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