メモリ管理の改善
[YACASL2.git] / src / word.c
1 #include "word.h"
2
3 /* wordのエラー定義 */
4 CERR cerr_word[] = {
5     { 114, "not integer" },
6     { 115, "not hex" },
7     { 116, "out of hex range" },
8 };
9
10 bool addcerrlist_word()
11 {
12     return addcerrlist(ARRAYSIZE(cerr_word), cerr_word);
13 }
14
15 /* 10進数の文字列をWORD値に変換 */
16 WORD n2word(const char *str)
17 {
18     assert(isdigit(*str) || *str == '-');
19
20     char *check;
21     int n;
22     /* WORD値に変換 */
23     n = strtol(str, &check, 10);
24     if(*check != '\0') {
25         setcerr(114, str);    /* not integer */
26         return 0x0;
27     }
28     /* nが-32768〜32767の範囲にないときは、その下位16ビットを格納 */
29     if(n < -32768 || n > 32767) {
30         n = n & 0xFFFF;
31     }
32     return (WORD)n;
33 }
34
35 /* 16進数の文字列をWORD値に変換 */
36 WORD h2word(const char *str)
37 {
38     assert(*str == '#');
39
40     WORD word = 0x0;
41     char *check;
42     str++;
43     if(*str == '-' || strlen(str) > 4) {
44         setcerr(116, str-1);    /* out of hex range */
45         return 0;
46     }
47     /* WORD値に変換 */
48     word = (WORD)strtol(str, &check, 16);
49     if(*check != '\0') {
50         setcerr(115, str-1);    /* not hex */
51         return 0x0;
52     }
53     return word;
54 }
55
56 /* 10進数または16進数の文字列をWORD値に変換 */
57 WORD nh2word(const char *str)
58 {
59     addcerrlist_word();
60     WORD word;
61     if(!isdigit(*str) && *str != '-' && *str != '#') {
62         setcerr(114, str);    /* not integer */
63         return 0x0;
64     }
65     if(*str == '#') {
66         word = h2word(str);
67     } else {
68         word = n2word(str);
69     }
70     return word;
71 }
72
73 /* WORD値を10進数の文字列に変換 */
74 char *word2n(WORD word)
75 {
76     enum {
77         MAXLEN = 6,        /* WORD値を10進数で表したときの最大桁数 */
78     };
79     char *p = malloc_chk(MAXLEN, "word2n.p"), *digit = malloc_chk(MAXLEN, "word2n.digit");
80     int i = 0, j;
81
82     do{
83         *(p + i++) = word % 10 + '0';
84     } while((word /= 10) > 0);
85     for(j = 0; j < i; j++) {
86         *(digit + j) = *(p + (i - 1) - j);
87     }
88     *(digit + j + 1) = '\0';
89     free_chk(p, "word2n.p");
90     return digit;
91 }
92
93 /* WORD値を2進数の文字列に変換 */
94 char *word2bit(const WORD word)
95 {
96     enum {
97         MAXLEN = 16,        /* WORD値を2進数で表したときの最大桁数 */
98     };
99     WORD mask = 0x8000;
100     char *bit, *p;
101
102     bit = p = malloc_chk(MAXLEN + 1, "word2bit.bit");
103     do {
104         *p++ = (word & mask) ? '1' : '0';
105     } while((mask >>= 1) > 0);
106     *p = '\0';
107     return bit;
108 }
109
110 /* WORD値を解析して表示 */
111 void print_dumpword(WORD word, bool logicalmode)
112 {
113     if(logicalmode == true) {
114         fprintf(stdout, "%6d", word);
115     } else {
116         fprintf(stdout, "%6d", (signed short)word);
117     }
118     fprintf(stdout, " = #%04X = %s", word, word2bit(word));
119     /* 「文字の組」の符号表に記載された文字と、改行(CR)/タブを表示 */
120     if(word >= 0x20 && word <= 0x7E) {
121         fprintf(stdout, " = \'%c\'", word);
122     } else if(word == 0xA) {
123         fprintf(stdout, " = \'\\n\'");
124     } else if(word == '\t') {
125         fprintf(stdout, " = \'\\t\'");
126     }
127     fprintf(stdout, "\n");
128 }