#include "cerr.h"
-/* エラーメッセージ */
-int cerrno = 0;
-char *cerrmsg;
+/* mallocを実行し、メモリを確保できない場合は */
+/* エラーを出力して終了 */
+void *malloc_chk(size_t size, char *tag)
+{
+ void *p;
+
+ if((p = malloc(size)) == NULL) {
+ fprintf(stderr, "%s: cannot allocate memory\n", tag);
+ exit(-1);
+ }
+ return p;
+}
+
+/* callocを実行し、メモリを確保できない場合は */
+/* エラーを出力して終了 */
+void *calloc_chk(size_t nmemb, size_t size, char *tag)
+{
+ void *p;
+
+ if((p = calloc(nmemb, size)) == NULL) {
+ fprintf(stderr, "%s: cannot allocate memory\n", tag);
+ exit(-1);
+ }
+ return p;
+}
-/* エラー番号とエラーメッセージを設定する */
-void setcerr(int num, const char *val)
+/* エラーの初期化 */
+void cerr_init()
{
- assert(&cerr != NULL && num > 0);
+ cerr = malloc_chk(sizeof(CERR), "cerr");
+ cerr->num = 0;
+}
+
+/* malloc_chkを実行してメモリを確保してから、 */
+/* コピーした文字列を返す */
+char *strdup_chk(const char *s, char *tag)
+{
+ assert(s != NULL);
+ char *t;
+
+ t = malloc_chk(strlen(s) + 1, tag);
+ strcpy(t, s);
+ return t;
+}
- cerrno = num;
- cerrmsg = malloc(MSGSIZE + 1);
- if(val != NULL) {
- strcpy(cerrmsg, val);
- strcat(cerrmsg, ": ");
- strcat(cerrmsg, getcerrmsg(cerrno));
+/* 現在のエラー */
+CERR *cerr;
+
+/* エラーリスト */
+CERRLIST *cerrlist;
+
+/* エラーリストを作成・追加する */
+bool addcerrlist(int newerrc, CERR newerrv[])
+{
+ int i;
+ CERRLIST *p, *q;
+
+ assert(newerrc > 0 && newerrv != NULL);
+ if(cerrlist == NULL) {
+ p = cerrlist = malloc_chk(sizeof(CERRLIST), "cerrlist");
} else {
- strcpy(cerrmsg, getcerrmsg(cerrno));
+ for(p = cerrlist; p != NULL; p = p->next) {
+ q = p;
+ }
+ p = q->next = malloc_chk(sizeof(CERRLIST), "cerrlist.next");
+ }
+ for(i = 0; i < newerrc; i++) {
+ p->cerr = &(newerrv[i]);
+ p->next = malloc_chk(sizeof(CERRLIST), "cerrlist.next");
+ q = p;
+ p = p->next;
}
+ q->next = NULL;
+ return true;
}
-/* エラー番号からメッセージを返す */
+/* 現在のエラーを設定する */
+void setcerr(int num, const char *str)
+{
+ cerr->num = num;
+ cerr->msg = malloc_chk(CERRMSGSIZE + 1, "cerr.msg");
+ if(str != NULL && strlen(str) <= CERRSTRSIZE) {
+ sprintf(cerr->msg, "%s: %s", str, getcerrmsg(cerr->num));
+ } else {
+ strcpy(cerr->msg, getcerrmsg(cerr->num));
+ }
+}
+
+/* エラーリストから、エラー番号に対応するメッセージを返す */
char *getcerrmsg(int num)
{
- assert(&cerr != NULL && num > 0);
- int i = 0;
- CERRARRAY *ptr;
+ CERRLIST *p;
- do {
- if((ptr = &cerr[i++])->num == num) {
- return ptr->msg;
+ for(p = cerrlist; p != NULL; p = p->next) {
+ if(num == p->cerr->num) {
+ return p->cerr->msg;
}
- } while(ptr->num > 0);
- return "unkown error";
+ }
+ return "unknown error";
}
-/* エラーを解放する */
+/* ã\82¨ã\83©ã\83¼ã\83ªã\82¹ã\83\88ã\81¨ç\8f¾å\9c¨ã\81®ã\82¨ã\83©ã\83¼ã\82\92解æ\94¾ã\81\99ã\82\8b */
void freecerr()
{
- assert(cerrno > 0);
- cerrno = 0;
- if(strlen(cerrmsg) > 0) {
- free(cerrmsg);
+ CERRLIST *p = cerrlist, *q;
+
+ /* エラーリストを解放 */
+ while(p != NULL) {
+ q = p->next;
+ free(p);
+ p = q;
}
+ /* 現在のエラーを解放 */
+ free(cerr);
}