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