メモリ管理の改善
[YACASL2.git] / src / cerr.c
index 121b680..02b6a3b 100644 (file)
@@ -1,46 +1,83 @@
 #include "cerr.h"
 
-/* エラーメッセージ */
-int cerrno = 0;
-char *cerrmsg;
+/* エラーの初期化 */
+void cerr_init()
+{
+    cerr = malloc_chk(sizeof(CERR), "cerr");
+    cerr->num = 0;
+}
+
+/* 現在のエラー */
+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 *val)
+/* 現在のエラーを設定する */
+void setcerr(int num, const char *str)
 {
-    assert(&cerr != NULL && num > 0);
-
-    cerrno = num;
-    cerrmsg = malloc(MSGSIZE + 1);
-    if(val != NULL) {
-        strcpy(cerrmsg, val);
-        strcat(cerrmsg, ": ");
-        strcat(cerrmsg, getcerrmsg(cerrno));
+    /* 現在のエラー番号を設定  */
+    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";
 }
 
-/* エラーを解放する */
+/* ã\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_chk(p, "freecerr.p");
+        p = q;
     }
+    /* 現在のエラーメッセージを解放 */
+    free_chk(cerr->msg, "cerr.msg");
+    /* 現在のエラーを解放 */
+    free_chk(cerr, "cerr");
 }