1/* Conversion to and from ARMSCII-8
2 Copyright (C) 1997-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
18
19#include <dlfcn.h>
20#include <stdint.h>
21
22/* Definitions used in the body of the `gconv' function. */
23#define CHARSET_NAME "ARMSCII-8//"
24#define FROM_LOOP from_armscii_8
25#define TO_LOOP to_armscii_8
26#define DEFINE_INIT 1
27#define DEFINE_FINI 1
28#define MIN_NEEDED_FROM 1
29#define MIN_NEEDED_TO 4
30#define ONE_DIRECTION 0
31
32
33static const uint16_t map_from_armscii_8[0xfe - 0xa2 + 1] =
34 {
35 0x0587, 0x0589, 0x0029, 0x0028, 0x00bb, 0x00ab, 0x2014, 0x002e,
36 0x055d, 0x002c, 0x002d, 0x058a, 0x2026, 0x055c, 0x055b, 0x055e,
37 0x0531, 0x0561, 0x0532, 0x0562, 0x0533, 0x0563, 0x0534, 0x0564,
38 0x0535, 0x0565, 0x0536, 0x0566, 0x0537, 0x0567, 0x0538, 0x0568,
39 0x0539, 0x0569, 0x053a, 0x056a, 0x053b, 0x056b, 0x053c, 0x056c,
40 0x053d, 0x056d, 0x053e, 0x056e, 0x053f, 0x056f, 0x0540, 0x0570,
41 0x0541, 0x0571, 0x0542, 0x0572, 0x0543, 0x0573, 0x0544, 0x0574,
42 0x0545, 0x0575, 0x0546, 0x0576, 0x0547, 0x0577, 0x0548, 0x0578,
43 0x0549, 0x0579, 0x054a, 0x057a, 0x054b, 0x057b, 0x054c, 0x057c,
44 0x054d, 0x057d, 0x054e, 0x057e, 0x054f, 0x057f, 0x0550, 0x0580,
45 0x0551, 0x0581, 0x0552, 0x0582, 0x0553, 0x0583, 0x0554, 0x0584,
46 0x0555, 0x0585, 0x0556, 0x0586, 0x055a
47 };
48
49
50/* First define the conversion function from ARMSCII-8 to UCS4. */
51#define MIN_NEEDED_INPUT MIN_NEEDED_FROM
52#define MIN_NEEDED_OUTPUT MIN_NEEDED_TO
53#define LOOPFCT FROM_LOOP
54#define BODY \
55 { \
56 uint_fast8_t ch = *inptr; \
57 \
58 if (ch <= 0xa0) \
59 { \
60 /* Upto and including 0xa0 the ARMSCII-8 corresponds to Unicode. */ \
61 *((uint32_t *) outptr) = ch; \
62 outptr += sizeof (uint32_t); \
63 } \
64 else if (ch >= 0xa2 && ch <= 0xfe) \
65 { \
66 /* Use the table. */ \
67 *((uint32_t *) outptr) = map_from_armscii_8[ch - 0xa2]; \
68 outptr += sizeof (uint32_t); \
69 } \
70 else \
71 { \
72 /* This is an illegal character. */ \
73 STANDARD_FROM_LOOP_ERR_HANDLER (1); \
74 } \
75 \
76 ++inptr; \
77 }
78#define LOOP_NEED_FLAGS
79#define ONEBYTE_BODY \
80 { \
81 if (c <= 0xa0) \
82 /* Upto and including 0xa0 the ARMSCII-8 corresponds to Unicode. */ \
83 return c; \
84 else if (c >= 0xa2 && c <= 0xfe) \
85 /* Use the table. */ \
86 return map_from_armscii_8[c - 0xa2]; \
87 else \
88 return WEOF; \
89 }
90#include <iconv/loop.c>
91
92
93static const unsigned char map_to_armscii_8[0x58a - 0x531 + 1] =
94 {
95 0xb2, 0xb4, 0xb6, 0xb8, 0xba, 0xbc, 0xbe, 0xc0,
96 0xc2, 0xc4, 0xc6, 0xc8, 0xca, 0xcc, 0xce, 0xd0,
97 0xd2, 0xd4, 0xd6, 0xd8, 0xda, 0xdc, 0xde, 0xe0,
98 0xe2, 0xe4, 0xe6, 0xe8, 0xea, 0xec, 0xee, 0xf0,
99 0xf2, 0xf4, 0xf6, 0xf8, 0xfa, 0xfc, 0x00, 0x00,
100 0x00, 0xfe, 0xb0, 0xaf, 0xaa, 0xb1, 0x00, 0x00,
101 0xb3, 0xb5, 0xb7, 0xb9, 0xbb, 0xbd, 0xbf, 0xc1,
102 0xc3, 0xc5, 0xc7, 0xc9, 0xcb, 0xcd, 0xcf, 0xd1,
103 0xd3, 0xd5, 0xd7, 0xd9, 0xdb, 0xdd, 0xdf, 0xe1,
104 0xe3, 0xe5, 0xe7, 0xe9, 0xeb, 0xed, 0xef, 0xf1,
105 0xf3, 0xf5, 0xf7, 0xf9, 0xfb, 0xfd, 0xa2, 0x00,
106 0xa3, 0xad
107 };
108
109
110/* Next, define the other direction. */
111#define MIN_NEEDED_INPUT MIN_NEEDED_TO
112#define MIN_NEEDED_OUTPUT MIN_NEEDED_FROM
113#define LOOPFCT TO_LOOP
114#define BODY \
115 { \
116 uint32_t ch = *((const uint32_t *) inptr); \
117 \
118 if (ch <= 0xa0) \
119 /* Upto and including 0xa0 the ARMSCII-8 corresponds to Unicode. */ \
120 *outptr = (unsigned char) ch; \
121 else if (ch == 0xab) \
122 *outptr = 0xa7; \
123 else if (ch == 0xbb) \
124 *outptr = 0xa6; \
125 else if (ch >= 0x531 && ch <= 0x58a) \
126 { \
127 unsigned char oc = map_to_armscii_8[ch - 0x531]; \
128 \
129 if (oc == 0) \
130 /* No valid mapping. */ \
131 goto err; \
132 \
133 *outptr = oc; \
134 } \
135 else if (ch == 0x2014) \
136 *outptr = 0xa8; \
137 else if (ch == 0x2026) \
138 *outptr = 0xae; \
139 else \
140 { \
141 UNICODE_TAG_HANDLER (ch, 4); \
142 \
143 /* We have an illegal character. */ \
144 err: \
145 STANDARD_TO_LOOP_ERR_HANDLER (4); \
146 } \
147 ++outptr; \
148 inptr += 4; \
149 }
150#define LOOP_NEED_FLAGS
151#include <iconv/loop.c>
152
153
154/* Now define the toplevel functions. */
155#include <iconv/skeleton.c>
156

source code of glibc/iconvdata/armscii-8.c