X-Git-Url: http://j8takagi.net/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fcerr.c;h=990673900bb3d2ad61b73abe082d92581c949c10;hb=8ec48a2a7f650cd1eb4d782c0968d47ebaeebfec;hp=1aeb8c42c29ba3ab3b0813ebec91ae4f5f6c1f8f;hpb=285d44a446b45fa1a5ac66617b6920bc7ba81fa6;p=YACASL2.git diff --git a/src/cerr.c b/src/cerr.c index 1aeb8c4..9906739 100644 --- a/src/cerr.c +++ b/src/cerr.c @@ -1,44 +1,117 @@ #include "cerr.h" -/* エラーメッセージ */ -int cerrno = 0; -char *cerrmsg; +/* mallocを実行し、メモリを確保できない場合は */ +/* エラーを出力して終了 */ +void *malloc_chk(size_t size, char *tag) +{ + void *p; -/* エラー番号とエラーメッセージを設定する */ -void setcerr(int num, const char *str) + 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) { - assert(cerr != NULL && num > 0); + void *p; + + if((p = calloc(nmemb, size)) == NULL) { + fprintf(stderr, "%s: cannot allocate memory\n", tag); + exit(-1); + } + return p; +} + +/* エラーの初期化 */ +void cerr_init() +{ + 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(CERRMSGSIZE + 1); - if(str != NULL && strlen(str) < 10) { - sprintf(cerrmsg, "%s: %s", str, 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 { + 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(cerrmsg, getcerrmsg(cerrno)); + 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"; } -/* エラーを解放する */ +/* エラーリストと現在のエラーを解放する */ 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); }