]> git.xonotic.org Git - xonotic/darkplaces.git/blob - utf8lib.c
physics: fix and refactor unsticking
[xonotic/darkplaces.git] / utf8lib.c
1 #include "darkplaces.h"
2 #include "utf8lib.h"
3 #include "draw.h"
4
5 /*
6 ================================================================================
7 Initialization of UTF-8 support and new cvars.
8 ================================================================================
9 */
10 // for compatibility this defaults to 0
11 cvar_t utf8_enable = {CF_CLIENT | CF_SERVER | CF_ARCHIVE, "utf8_enable", "0", "Enable UTF-8 support. For compatibility, this is disabled by default in most games."};
12
13 void   u8_Init(void)
14 {
15         Cvar_RegisterVariable(&utf8_enable);
16 }
17
18 /*
19 ================================================================================
20 UTF-8 encoding and decoding functions follow.
21 ================================================================================
22 */
23
24 unsigned char utf8_lengths[256] = { // 0 = invalid
25         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // ascii characters
26         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
27         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
28         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
29         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
30         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
31         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
32         1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
33         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 0x80 - 0xBF are within multibyte sequences
34         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // they could be interpreted as 2-byte starts but
35         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // the codepoint would be < 127
36         0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
37         0, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, // C0 and C1 would also result in overlong encodings
38         2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
39         3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
40         4, 4, 4, 4, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
41         // with F5 the codepoint is above 0x10FFFF,
42         // F8-FB would start 5-byte sequences
43         // FC-FD would start 6-byte sequences
44         // ...
45 };
46 Uchar utf8_range[5] = {
47         1,       // invalid - let's not allow the creation of 0-bytes :P
48         1,       // ascii minimum
49         0x80,    // 2-byte minimum
50         0x800,   // 3-byte minimum
51         0x10000, // 4-byte minimum
52 };
53
54 /** Analyze the next character and return various information if requested.
55  * @param _s      An utf-8 string.
56  * @param _start  Filled with the start byte-offset of the next valid character
57  * @param _len    Fileed with the length of the next valid character
58  * @param _ch     Filled with the unicode value of the next character
59  * @param _maxlen Maximum number of bytes to read from _s
60  * @return        Whether or not another valid character is in the string
61  */
62 #define U8_ANALYZE_INFINITY 7
63 static qbool u8_analyze(const char *_s, size_t *_start, size_t *_len, Uchar *_ch, size_t _maxlen)
64 {
65         const unsigned char *s = (const unsigned char*)_s;
66         size_t i, j;
67         size_t bits = 0;
68         Uchar ch;
69
70         i = 0;
71 findchar:
72         while (i < _maxlen && s[i] && (bits = utf8_lengths[s[i]]) == 0)
73                 ++i;
74
75         if (i >= _maxlen || !s[i]) {
76                 if (_start) *_start = i;
77                 if (_len) *_len = 0;
78                 return false;
79         }
80         
81         if (bits == 1) { // ascii
82                 if (_start) *_start = i;
83                 if (_len) *_len = 1;
84                 if (_ch) *_ch = (Uchar)s[i];
85                 return true;
86         }
87
88         ch = (s[i] & (0xFF >> bits));
89         for (j = 1; j < bits; ++j)
90         {
91                 if ( (s[i+j] & 0xC0) != 0x80 )
92                 {
93                         i += j;
94                         goto findchar;
95                 }
96                 ch = (ch << 6) | (s[i+j] & 0x3F);
97         }
98         if (ch < utf8_range[bits] || ch >= 0x10FFFF)
99         {
100                 i += bits;
101                 goto findchar;
102         }
103 #if 0
104         // <0xC2 is always an overlong encoding, they're invalid, thus skipped
105         while (i < _maxlen && s[i] && s[i] >= 0x80 && s[i] < 0xC2) {
106                 //fprintf(stderr, "skipping\n");
107                 ++i;
108         }
109
110         // If we hit the end, well, we're out and invalid
111         if(i >= _maxlen || !s[i]) {
112                 if (_start) *_start = i;
113                 if (_len) *_len = 0;
114                 return false;
115         }
116
117         // I'll leave that in - if you remove it, also change the part below
118         // to support 1-byte chars correctly
119         if (s[i] < 0x80)
120         {
121                 if (_start) *_start = i;
122                 if (_len) *_len = 1;
123                 if (_ch) *_ch = (Uchar)s[i];
124                 //fprintf(stderr, "valid ascii\n");
125                 return true;
126         }
127
128         // Figure out the next char's length
129         bc = s[i];
130         bits = 1;
131         // count the 1 bits, they're the # of bytes
132         for (bt = 0x40; bt && (bc & bt); bt >>= 1, ++bits);
133         if (!bt)
134         {
135                 //fprintf(stderr, "superlong\n");
136                 ++i;
137                 goto findchar;
138         }
139         if(i + bits > _maxlen) {
140                 /*
141                 if (_start) *_start = i;
142                 if (_len) *_len = 0;
143                 return false;
144                 */
145                 ++i;
146                 goto findchar;
147         }
148         // turn bt into a mask and give ch a starting value
149         --bt;
150         ch = (s[i] & bt);
151         // check the byte sequence for invalid bytes
152         for (j = 1; j < bits; ++j)
153         {
154                 // valid bit value: 10xx xxxx
155                 //if (s[i+j] < 0x80 || s[i+j] >= 0xC0)
156                 if ( (s[i+j] & 0xC0) != 0x80 )
157                 {
158                         //fprintf(stderr, "sequence of %i f'd at %i by %x\n", bits, j, (unsigned int)s[i+j]);
159                         // this byte sequence is invalid, skip it
160                         i += j;
161                         // find a character after it
162                         goto findchar;
163                 }
164                 // at the same time, decode the character
165                 ch = (ch << 6) | (s[i+j] & 0x3F);
166         }
167
168         // Now check the decoded byte for an overlong encoding
169         if ( (bits >= 2 && ch < 0x80) ||
170              (bits >= 3 && ch < 0x800) ||
171              (bits >= 4 && ch < 0x10000) ||
172              ch >= 0x10FFFF // RFC 3629
173         )
174         {
175                 i += bits;
176                 //fprintf(stderr, "overlong: %i bytes for %x\n", bits, ch);
177                 goto findchar;
178         }
179 #endif
180
181         if (_start)
182                 *_start = i;
183         if (_len)
184                 *_len = bits;
185         if (_ch)
186                 *_ch = ch;
187         //fprintf(stderr, "valid utf8\n");
188         return true;
189 }
190
191 /** Get the number of characters in an UTF-8 string.
192  * @param _s    An utf-8 encoded null-terminated string.
193  * @return      The number of unicode characters in the string.
194  */
195 size_t u8_strlen(const char *_s)
196 {
197         size_t st, ln;
198         size_t len = 0;
199         const unsigned char *s = (const unsigned char*)_s;
200
201         if (!utf8_enable.integer)
202                 return strlen(_s);
203
204         while (*s)
205         {
206                 // ascii char, skip u8_analyze
207                 if (*s < 0x80)
208                 {
209                         ++len;
210                         ++s;
211                         continue;
212                 }
213
214                 // invalid, skip u8_analyze
215                 if (*s < 0xC2)
216                 {
217                         ++s;
218                         continue;
219                 }
220
221                 if (!u8_analyze((const char*)s, &st, &ln, NULL, U8_ANALYZE_INFINITY))
222                         break;
223                 // valid character, skip after it
224                 s += st + ln;
225                 ++len;
226         }
227         return len;
228 }
229
230 static int colorcode_skipwidth(const unsigned char *s)
231 {
232         if(*s == STRING_COLOR_TAG)
233         {
234                 if(s[1] <= '9' && s[1] >= '0') // ^[0-9] found
235                         return 2;
236                 else if(s[1] == STRING_COLOR_RGB_TAG_CHAR && isxdigit(s[2]) && isxdigit(s[3]) && isxdigit(s[4]))
237                         return 5;
238                 else if(s[1] == STRING_COLOR_TAG)
239                         return 1; // special case, do NOT call colorcode_skipwidth for next char
240         }
241         return 0;
242 }
243
244 /** Get the number of characters in a part of an UTF-8 string.
245  * @param _s    An utf-8 encoded null-terminated string.
246  * @param n     The maximum number of bytes.
247  * @return      The number of unicode characters in the string.
248  */
249 size_t u8_strnlen(const char *_s, size_t n)
250 {
251         size_t st, ln;
252         size_t len = 0;
253         const unsigned char *s = (const unsigned char*)_s;
254
255         if (!utf8_enable.integer)
256         {
257                 len = strlen(_s);
258                 return (len < n) ? len : n;
259         }
260
261         while (*s && n)
262         {
263                 // ascii char, skip u8_analyze
264                 if (*s < 0x80)
265                 {
266                         ++len;
267                         ++s;
268                         --n;
269                         continue;
270                 }
271
272                 // invalid, skip u8_analyze
273                 if (*s < 0xC2)
274                 {
275                         ++s;
276                         --n;
277                         continue;
278                 }
279
280                 if (!u8_analyze((const char*)s, &st, &ln, NULL, n))
281                         break;
282                 // valid character, see if it's still inside the range specified by n:
283                 if (n < st + ln)
284                         return len;
285                 ++len;
286                 n -= st + ln;
287                 s += st + ln;
288         }
289         return len;
290 }
291
292 static size_t u8_strnlen_colorcodes(const char *_s, size_t n)
293 {
294         size_t st, ln;
295         size_t len = 0;
296         const unsigned char *s = (const unsigned char*)_s;
297
298         while (*s && n)
299         {
300                 int w = colorcode_skipwidth(s);
301                 n -= w;
302                 s += w;
303                 if(w > 1) // == 1 means single caret
304                         continue;
305
306                 // ascii char, skip u8_analyze
307                 if (*s < 0x80 || !utf8_enable.integer)
308                 {
309                         ++len;
310                         ++s;
311                         --n;
312                         continue;
313                 }
314
315                 // invalid, skip u8_analyze
316                 if (*s < 0xC2)
317                 {
318                         ++s;
319                         --n;
320                         continue;
321                 }
322
323                 if (!u8_analyze((const char*)s, &st, &ln, NULL, n))
324                         break;
325                 // valid character, see if it's still inside the range specified by n:
326                 if (n < st + ln)
327                         return len;
328                 ++len;
329                 n -= st + ln;
330                 s += st + ln;
331         }
332         return len;
333 }
334
335 /** Get the number of bytes used in a string to represent an amount of characters.
336  * @param _s    An utf-8 encoded null-terminated string.
337  * @param n     The number of characters we want to know the byte-size for.
338  * @return      The number of bytes used to represent n characters.
339  */
340 size_t u8_bytelen(const char *_s, size_t n)
341 {
342         size_t st, ln;
343         size_t len = 0;
344         const unsigned char *s = (const unsigned char*)_s;
345
346         if (!utf8_enable.integer) {
347                 len = strlen(_s);
348                 return (len < n) ? len : n;
349         }
350
351         while (*s && n)
352         {
353                 // ascii char, skip u8_analyze
354                 if (*s < 0x80)
355                 {
356                         ++len;
357                         ++s;
358                         --n;
359                         continue;
360                 }
361
362                 // invalid, skip u8_analyze
363                 if (*s < 0xC2)
364                 {
365                         ++s;
366                         ++len;
367                         continue;
368                 }
369
370                 if (!u8_analyze((const char*)s, &st, &ln, NULL, U8_ANALYZE_INFINITY))
371                         break;
372                 --n;
373                 s += st + ln;
374                 len += st + ln;
375         }
376         return len;
377 }
378
379 static size_t u8_bytelen_colorcodes(const char *_s, size_t n)
380 {
381         size_t st, ln;
382         size_t len = 0;
383         const unsigned char *s = (const unsigned char*)_s;
384
385         while (*s && n)
386         {
387                 int w = colorcode_skipwidth(s);
388                 len += w;
389                 s += w;
390                 if(w > 1) // == 1 means single caret
391                         continue;
392
393                 // ascii char, skip u8_analyze
394                 if (*s < 0x80 || !utf8_enable.integer)
395                 {
396                         ++len;
397                         ++s;
398                         --n;
399                         continue;
400                 }
401
402                 // invalid, skip u8_analyze
403                 if (*s < 0xC2)
404                 {
405                         ++s;
406                         ++len;
407                         continue;
408                 }
409
410                 if (!u8_analyze((const char*)s, &st, &ln, NULL, U8_ANALYZE_INFINITY))
411                         break;
412                 --n;
413                 s += st + ln;
414                 len += st + ln;
415         }
416         return len;
417 }
418
419 /** Get the byte-index for a character-index.
420  * @param _s      An utf-8 encoded string.
421  * @param i       The character-index for which you want the byte offset.
422  * @param len     If not null, character's length will be stored in there.
423  * @return        The byte-index at which the character begins, or -1 if the string is too short.
424  */
425 int u8_byteofs(const char *_s, size_t i, size_t *len)
426 {
427         size_t st, ln;
428         size_t ofs = 0;
429         const unsigned char *s = (const unsigned char*)_s;
430
431         if (!utf8_enable.integer)
432         {
433                 if (strlen(_s) < i)
434                 {
435                         if (len) *len = 0;
436                         return -1;
437                 }
438
439                 if (len) *len = 1;
440                 return (int)i;
441         }
442
443         st = ln = 0;
444         do
445         {
446                 ofs += ln;
447                 if (!u8_analyze((const char*)s + ofs, &st, &ln, NULL, U8_ANALYZE_INFINITY))
448                         return -1;
449                 ofs += st;
450         } while(i-- > 0);
451         if (len)
452                 *len = ln;
453         return (int)ofs;
454 }
455
456 /** Get the char-index for a byte-index.
457  * @param _s      An utf-8 encoded string.
458  * @param i       The byte offset for which you want the character index.
459  * @param len     If not null, the offset within the character is stored here.
460  * @return        The character-index, or -1 if the string is too short.
461  */
462 int u8_charidx(const char *_s, size_t i, size_t *len)
463 {
464         size_t st, ln;
465         size_t ofs = 0;
466         size_t pofs = 0;
467         int idx = 0;
468         const unsigned char *s = (const unsigned char*)_s;
469
470         if (!utf8_enable.integer)
471         {
472                 if (len) *len = 0;
473                 return (int)i;
474         }
475
476         while (ofs < i && s[ofs])
477         {
478                 // ascii character, skip u8_analyze
479                 if (s[ofs] < 0x80)
480                 {
481                         pofs = ofs;
482                         ++idx;
483                         ++ofs;
484                         continue;
485                 }
486
487                 // invalid, skip u8_analyze
488                 if (s[ofs] < 0xC2)
489                 {
490                         ++ofs;
491                         continue;
492                 }
493
494                 if (!u8_analyze((const char*)s+ofs, &st, &ln, NULL, U8_ANALYZE_INFINITY))
495                         return -1;
496                 // see if next char is after the bytemark
497                 if (ofs + st > i)
498                 {
499                         if (len)
500                                 *len = i - pofs;
501                         return idx;
502                 }
503                 ++idx;
504                 pofs = ofs + st;
505                 ofs += st + ln;
506                 // see if bytemark is within the char
507                 if (ofs > i)
508                 {
509                         if (len)
510                                 *len = i - pofs;
511                         return idx;
512                 }
513         }
514         if (len) *len = 0;
515         return idx;
516 }
517
518 /** Get the byte offset of the previous byte.
519  * The result equals:
520  * prevchar_pos = u8_byteofs(text, u8_charidx(text, thischar_pos, NULL) - 1, NULL)
521  * @param _s      An utf-8 encoded string.
522  * @param i       The current byte offset.
523  * @return        The byte offset of the previous character
524  */
525 size_t u8_prevbyte(const char *_s, size_t i)
526 {
527         size_t st, ln;
528         const unsigned char *s = (const unsigned char*)_s;
529         size_t lastofs = 0;
530         size_t ofs = 0;
531
532         if (!utf8_enable.integer)
533         {
534                 if (i > 0)
535                         return i-1;
536                 return 0;
537         }
538
539         while (ofs < i && s[ofs])
540         {
541                 // ascii character, skip u8_analyze
542                 if (s[ofs] < 0x80)
543                 {
544                         lastofs = ofs++;
545                         continue;
546                 }
547
548                 // invalid, skip u8_analyze
549                 if (s[ofs] < 0xC2)
550                 {
551                         ++ofs;
552                         continue;
553                 }
554
555                 if (!u8_analyze((const char*)s+ofs, &st, &ln, NULL, U8_ANALYZE_INFINITY))
556                         return lastofs;
557                 if (ofs + st > i)
558                         return lastofs;
559                 if (ofs + st + ln >= i)
560                         return ofs + st;
561
562                 lastofs = ofs;
563                 ofs += st + ln;
564         }
565         return lastofs;
566 }
567
568 Uchar u8_quake2utf8map[256] = {
569         0xE000, 0xE001, 0xE002, 0xE003, 0xE004, 0xE005, 0xE006, 0xE007, 0xE008, 0xE009, 0xE00A, 0xE00B, 0xE00C, 0xE00D, 0xE00E, 0xE00F, // specials
570         0xE010, 0xE011, 0xE012, 0xE013, 0xE014, 0xE015, 0xE016, 0xE017, 0xE018, 0xE019, 0xE01A, 0xE01B, 0xE01C, 0xE01D, 0xE01E, 0xE01F, // specials
571         0x0020, 0x0021, 0x0022, 0x0023, 0x0024, 0x0025, 0x0026, 0x0027, 0x0028, 0x0029, 0x002A, 0x002B, 0x002C, 0x002D, 0x002E, 0x002F, // shift+digit line
572         0x0030, 0x0031, 0x0032, 0x0033, 0x0034, 0x0035, 0x0036, 0x0037, 0x0038, 0x0039, 0x003A, 0x003B, 0x003C, 0x003D, 0x003E, 0x003F, // digits
573         0x0040, 0x0041, 0x0042, 0x0043, 0x0044, 0x0045, 0x0046, 0x0047, 0x0048, 0x0049, 0x004A, 0x004B, 0x004C, 0x004D, 0x004E, 0x004F, // caps
574         0x0050, 0x0051, 0x0052, 0x0053, 0x0054, 0x0055, 0x0056, 0x0057, 0x0058, 0x0059, 0x005A, 0x005B, 0x005C, 0x005D, 0x005E, 0x005F, // caps
575         0x0060, 0x0061, 0x0062, 0x0063, 0x0064, 0x0065, 0x0066, 0x0067, 0x0068, 0x0069, 0x006A, 0x006B, 0x006C, 0x006D, 0x006E, 0x006F, // small
576         0x0070, 0x0071, 0x0072, 0x0073, 0x0074, 0x0075, 0x0076, 0x0077, 0x0078, 0x0079, 0x007A, 0x007B, 0x007C, 0x007D, 0x007E, 0x007F, // small
577         0xE080, 0xE081, 0xE082, 0xE083, 0xE084, 0xE085, 0xE086, 0xE087, 0xE088, 0xE089, 0xE08A, 0xE08B, 0xE08C, 0xE08D, 0xE08E, 0xE08F, // specials
578         0xE090, 0xE091, 0xE092, 0xE093, 0xE094, 0xE095, 0xE096, 0xE097, 0xE098, 0xE099, 0xE09A, 0xE09B, 0xE09C, 0xE09D, 0xE09E, 0xE09F, // faces
579         0xE0A0, 0xE0A1, 0xE0A2, 0xE0A3, 0xE0A4, 0xE0A5, 0xE0A6, 0xE0A7, 0xE0A8, 0xE0A9, 0xE0AA, 0xE0AB, 0xE0AC, 0xE0AD, 0xE0AE, 0xE0AF,
580         0xE0B0, 0xE0B1, 0xE0B2, 0xE0B3, 0xE0B4, 0xE0B5, 0xE0B6, 0xE0B7, 0xE0B8, 0xE0B9, 0xE0BA, 0xE0BB, 0xE0BC, 0xE0BD, 0xE0BE, 0xE0BF,
581         0xE0C0, 0xE0C1, 0xE0C2, 0xE0C3, 0xE0C4, 0xE0C5, 0xE0C6, 0xE0C7, 0xE0C8, 0xE0C9, 0xE0CA, 0xE0CB, 0xE0CC, 0xE0CD, 0xE0CE, 0xE0CF,
582         0xE0D0, 0xE0D1, 0xE0D2, 0xE0D3, 0xE0D4, 0xE0D5, 0xE0D6, 0xE0D7, 0xE0D8, 0xE0D9, 0xE0DA, 0xE0DB, 0xE0DC, 0xE0DD, 0xE0DE, 0xE0DF,
583         0xE0E0, 0xE0E1, 0xE0E2, 0xE0E3, 0xE0E4, 0xE0E5, 0xE0E6, 0xE0E7, 0xE0E8, 0xE0E9, 0xE0EA, 0xE0EB, 0xE0EC, 0xE0ED, 0xE0EE, 0xE0EF,
584         0xE0F0, 0xE0F1, 0xE0F2, 0xE0F3, 0xE0F4, 0xE0F5, 0xE0F6, 0xE0F7, 0xE0F8, 0xE0F9, 0xE0FA, 0xE0FB, 0xE0FC, 0xE0FD, 0xE0FE, 0xE0FF,
585 };
586
587 /** Fetch a character from an utf-8 encoded string.
588  * @param _s      The start of an utf-8 encoded multi-byte character.
589  * @param _end    Will point to after the first multi-byte character.
590  * @return        The 32-bit integer representation of the first multi-byte character or 0 for invalid characters.
591  */
592 Uchar u8_getchar_utf8_enabled(const char *_s, const char **_end)
593 {
594         size_t st, ln;
595         Uchar ch;
596
597         if (!u8_analyze(_s, &st, &ln, &ch, U8_ANALYZE_INFINITY))
598                 ch = 0;
599         if (_end)
600                 *_end = _s + st + ln;
601         return ch;
602 }
603
604 /** Fetch a character from an utf-8 encoded string.
605  * @param _s      The start of an utf-8 encoded multi-byte character.
606  * @param _end    Will point to after the first multi-byte character.
607  * @return        The 32-bit integer representation of the first multi-byte character or 0 for invalid characters.
608  */
609 Uchar u8_getnchar_utf8_enabled(const char *_s, const char **_end, size_t _maxlen)
610 {
611         size_t st, ln;
612         Uchar ch;
613
614         if (!u8_analyze(_s, &st, &ln, &ch, _maxlen))
615                 ch = 0;
616         if (_end)
617                 *_end = _s + st + ln;
618         return ch;
619 }
620
621 /** Encode a wide-character into utf-8.
622  * @param w        The wide character to encode.
623  * @param to       The target buffer the utf-8 encoded string is stored to.
624  * @param maxlen   The maximum number of bytes that fit into the target buffer.
625  * @return         Number of bytes written to the buffer not including the terminating null.
626  *                 Less or equal to 0 if the buffer is too small.
627  */
628 int u8_fromchar(Uchar w, char *to, size_t maxlen)
629 {
630         if (maxlen < 1)
631                 return 0;
632
633         if (!w)
634                 return 0;
635
636         if (w >= 0xE000 && !utf8_enable.integer)
637                 w -= 0xE000;
638
639         if (w < 0x80 || !utf8_enable.integer)
640         {
641                 to[0] = (char)w;
642                 if (maxlen < 2)
643                         return -1;
644                 to[1] = 0;
645                 return 1;
646         }
647         // for a little speedup
648         if (w < 0x800)
649         {
650                 if (maxlen < 3)
651                 {
652                         to[0] = 0;
653                         return -1;
654                 }
655                 to[2] = 0;
656                 to[1] = 0x80 | (w & 0x3F); w >>= 6;
657                 to[0] = 0xC0 | w;
658                 return 2;
659         }
660         if (w < 0x10000)
661         {
662                 if (maxlen < 4)
663                 {
664                         to[0] = 0;
665                         return -1;
666                 }
667                 to[3] = 0;
668                 to[2] = 0x80 | (w & 0x3F); w >>= 6;
669                 to[1] = 0x80 | (w & 0x3F); w >>= 6;
670                 to[0] = 0xE0 | w;
671                 return 3;
672         }
673
674         // RFC 3629
675         if (w <= 0x10FFFF)
676         {
677                 if (maxlen < 5)
678                 {
679                         to[0] = 0;
680                         return -1;
681                 }
682                 to[4] = 0;
683                 to[3] = 0x80 | (w & 0x3F); w >>= 6;
684                 to[2] = 0x80 | (w & 0x3F); w >>= 6;
685                 to[1] = 0x80 | (w & 0x3F); w >>= 6;
686                 to[0] = 0xF0 | w;
687                 return 4;
688         }
689         return 0;
690 }
691
692 /** uses u8_fromchar on a static buffer
693  * @param ch        The unicode character to convert to encode
694  * @param l         The number of bytes without the terminating null.
695  * @return          A statically allocated buffer containing the character's utf8 representation, or NULL if it fails.
696  */
697 char *u8_encodech(Uchar ch, size_t *l, char *buf16)
698 {
699         size_t len;
700         len = u8_fromchar(ch, buf16, 16);
701         if (len > 0)
702         {
703                 if (l) *l = len;
704                 return buf16;
705         }
706         return NULL;
707 }
708
709 /** Convert a utf-8 multibyte string to a wide character string.
710  * @param wcs       The target wide-character buffer.
711  * @param mb        The utf-8 encoded multibyte string to convert.
712  * @param maxlen    The maximum number of wide-characters that fit into the target buffer.
713  * @return          The number of characters written to the target buffer.
714  */
715 size_t u8_mbstowcs(Uchar *wcs, const char *mb, size_t maxlen)
716 {
717         size_t i;
718         Uchar ch;
719         if (maxlen < 1)
720                 return 0;
721         for (i = 0; *mb && i < maxlen-1; ++i)
722         {
723                 ch = u8_getchar(mb, &mb);
724                 if (!ch)
725                         break;
726                 wcs[i] = ch;
727         }
728         wcs[i] = 0;
729         return i;
730 }
731
732 /** Convert a wide-character string to a utf-8 multibyte string.
733  * @param mb      The target buffer the utf-8 string is written to.
734  * @param wcs     The wide-character string to convert.
735  * @param maxlen  The number bytes that fit into the multibyte target buffer.
736  * @return        The number of bytes written, not including the terminating \0
737  */
738 size_t u8_wcstombs(char *mb, const Uchar *wcs, size_t maxlen)
739 {
740         size_t i;
741         const char *start = mb;
742         if (maxlen < 2)
743                 return 0;
744         for (i = 0; wcs[i] && i < maxlen-1; ++i)
745         {
746                 /*
747                 int len;
748                 if ( (len = u8_fromchar(wcs[i], mb, maxlen - i)) < 0)
749                         return (mb - start);
750                 mb += len;
751                 */
752                 mb += u8_fromchar(wcs[i], mb, maxlen - i);
753         }
754         *mb = 0;
755         return (mb - start);
756 }
757
758 /*
759 ============
760 UTF-8 aware COM_StringLengthNoColors
761
762 calculates the visible width of a color coded string.
763
764 *valid is filled with TRUE if the string is a valid colored string (that is, if
765 it does not end with an unfinished color code). If it gets filled with FALSE, a
766 fix would be adding a STRING_COLOR_TAG at the end of the string.
767
768 valid can be set to NULL if the caller doesn't care.
769
770 For size_s, specify the maximum number of characters from s to use, or 0 to use
771 all characters until the zero terminator.
772 ============
773 */
774 size_t
775 COM_StringLengthNoColors(const char *s, size_t size_s, qbool *valid);
776 size_t
777 u8_COM_StringLengthNoColors(const char *_s, size_t size_s, qbool *valid)
778 {
779         const unsigned char *s = (const unsigned char*)_s;
780         const unsigned char *end;
781         size_t len = 0;
782         size_t st, ln;
783
784         if (!utf8_enable.integer)
785                 return COM_StringLengthNoColors(_s, size_s, valid);
786
787         end = size_s ? (s + size_s) : NULL;
788
789         for(;;)
790         {
791                 switch((s == end) ? 0 : *s)
792                 {
793                         case 0:
794                                 if(valid)
795                                         *valid = true;
796                                 return len;
797                         case STRING_COLOR_TAG:
798                                 ++s;
799                                 switch((s == end) ? 0 : *s)
800                                 {
801                                         case STRING_COLOR_RGB_TAG_CHAR:
802                                                 if (s+1 != end && isxdigit(s[1]) &&
803                                                         s+2 != end && isxdigit(s[2]) &&
804                                                         s+3 != end && isxdigit(s[3]) )
805                                                 {
806                                                         s+=3;
807                                                         break;
808                                                 }
809                                                 ++len; // STRING_COLOR_TAG
810                                                 ++len; // STRING_COLOR_RGB_TAG_CHAR
811                                                 break;
812                                         case 0: // ends with unfinished color code!
813                                                 ++len;
814                                                 if(valid)
815                                                         *valid = false;
816                                                 return len;
817                                         case STRING_COLOR_TAG: // escaped ^
818                                                 ++len;
819                                                 break;
820                                         case '0': case '1': case '2': case '3': case '4':
821                                         case '5': case '6': case '7': case '8': case '9': // color code
822                                                 break;
823                                         default: // not a color code
824                                                 ++len; // STRING_COLOR_TAG
825                                                 ++len; // the character
826                                                 break;
827                                 }
828                                 ++s;
829                                 continue;
830                         default:
831                                 break;
832                 }
833
834                 // ascii char, skip u8_analyze
835                 if (*s < 0x80)
836                 {
837                         ++len;
838                         ++s;
839                         continue;
840                 }
841
842                 // invalid, skip u8_analyze
843                 if (*s < 0xC2)
844                 {
845                         ++s;
846                         continue;
847                 }
848
849                 if (!u8_analyze((const char*)s, &st, &ln, NULL, U8_ANALYZE_INFINITY))
850                 {
851                         // we CAN end up here, if an invalid char is between this one and the end of the string
852                         if(valid)
853                                 *valid = true;
854                         return len;
855                 }
856
857                 if(end && s + st + ln > end)
858                 {
859                         // string length exceeded by new character
860                         if(valid)
861                                 *valid = true;
862                         return len;
863                 }
864
865                 // valid character, skip after it
866                 s += st + ln;
867                 ++len;
868         }
869         // never get here
870 }
871
872 /** Pads a utf-8 string
873  * @param out     The target buffer the utf-8 string is written to.
874  * @param outsize The size of the target buffer, including the final NUL
875  * @param in      The input utf-8 buffer
876  * @param leftalign Left align the output string (by default right alignment is done)
877  * @param minwidth The minimum output width
878  * @param maxwidth The maximum output width
879  * @return        The number of bytes written, not including the terminating \0
880  */
881 size_t u8_strpad(char *out, size_t outsize, const char *in, qbool leftalign, size_t minwidth, size_t maxwidth)
882 {
883         if(!utf8_enable.integer)
884         {
885                 return dpsnprintf(out, outsize, "%*.*s", leftalign ? -(int) minwidth : (int) minwidth, (int) maxwidth, in);
886         }
887         else
888         {
889                 size_t l = u8_bytelen(in, maxwidth);
890                 size_t actual_width = u8_strnlen(in, l);
891                 int pad = (int)((actual_width >= minwidth) ? 0 : (minwidth - actual_width));
892                 int prec = (int)l;
893                 int lpad = leftalign ? 0 : pad;
894                 int rpad = leftalign ? pad : 0;
895                 return dpsnprintf(out, outsize, "%*s%.*s%*s", lpad, "", prec, in, rpad, "");
896         }
897 }
898
899 size_t u8_strpad_colorcodes(char *out, size_t outsize, const char *in, qbool leftalign, size_t minwidth, size_t maxwidth)
900 {
901         size_t l = u8_bytelen_colorcodes(in, maxwidth);
902         size_t actual_width = u8_strnlen_colorcodes(in, l);
903         int pad = (int)((actual_width >= minwidth) ? 0 : (minwidth - actual_width));
904         int prec = (int)l;
905         int lpad = leftalign ? 0 : pad;
906         int rpad = leftalign ? pad : 0;
907         return dpsnprintf(out, outsize, "%*s%.*s%*s", lpad, "", prec, in, rpad, "");
908 }
909
910 #ifdef WIN32
911
912 /**
913  * Convert Windows "wide" characters to WTF-8 for internal manipulation
914  */
915 int towtf8(const wchar *wstr, int wlen, char *cstr, int maxclen)
916 {
917         int p = 0;
918         int i;
919         for (i = 0; i < wlen; ++i)
920         {
921                 wchar point = wstr[i];
922                 if (point < 0x80)
923                 {
924                         if (p + 1 >= maxclen) break;
925                         cstr[p++] = point;
926                 }
927                 else if (point < 0x800)
928                 {
929                         if (p + 2 >= maxclen) break;
930                         cstr[p++] = (0xc0 | ((point >>  6) & 0x1f));
931                         cstr[p++] = (0x80 | ((point >>  0) & 0x3f));
932                 }
933                 else
934                 #if WTF8U32
935                 if (point < 0x10000)
936                 #endif
937                 {
938                         if (p + 3 >= maxclen) break;
939                         cstr[p++] = (0xe0 | ((point >> 12) & 0x0f));
940                         cstr[p++] = (0x80 | ((point >>  6) & 0x3f));
941                         cstr[p++] = (0x80 | ((point >>  0) & 0x3f));
942                 }
943                 #if WTF8U32
944                 else
945                 #if WTF8CHECKS
946                 if (point < 0x110000)
947                 #endif
948                 {
949                         if (p + 4 >= maxclen) break;
950                         cstr[p++] = (0xf0 | ((point >> 18) & 0x07));
951                         cstr[p++] = (0x80 | ((point >> 12) & 0x3f));
952                         cstr[p++] = (0x80 | ((point >>  6) & 0x3f));
953                         cstr[p++] = (0x80 | ((point >>  0) & 0x3f));
954                 }
955                 #endif
956         }
957         cstr[p] = 0x00;
958         return p;
959 }
960
961 /**
962  * Convert WTF-8 string to "wide" characters used by Windows
963  */
964 int fromwtf8(const char *cstr, int clen, wchar *wstr, int maxwlen)
965 {
966         int p = 0;
967         int i;
968         for (i = 0; i < clen;)
969         {
970                 char byte = cstr[i++];
971                 wchar point = byte;
972                 int length = 1;
973                 if (p + 1 >= maxwlen) break;
974                 #if WTF8CHECKS
975                 if ((byte & 0xf8) == 0xf8)
976                         return -1;
977                 #endif
978                 #if WTF8U32
979                 if ((byte & 0xf8) == 0xf0)
980                 {
981                         length = 4;
982                         point = byte & 0x07;
983                 }
984                 else
985                 #endif
986                 if ((byte & 0xf0) == 0xe0)
987                 {
988                         length = 3;
989                         point = byte & 0x0f;
990                 }
991                 else if ((byte & 0xe0) == 0xc0)
992                 {
993                         length = 2;
994                         point = byte & 0x1f;
995                 }
996                 #if WTF8CHECKS
997                 else if ((byte & 0xc0) == 0x80)
998                 {
999                         return -1;
1000                 }
1001                 #endif
1002                 while (--length)
1003                 {
1004                         byte = cstr[i++];
1005                         #if WTF8CHECKS
1006                         if (byte == -1) return -1;
1007                         else if ((byte & 0xc0) != 0x80) return -1;
1008                         #endif
1009                         point = (point << 6) | (byte & 0x3f);
1010                 }
1011                 wstr[p++] = point;
1012         }
1013         wstr[p] = 0x00;
1014         return p;
1015 }
1016
1017 #endif // WIN32
1018
1019
1020 /*
1021 The two following functions (u8_toupper, u8_tolower) are derived from
1022 ftp://ftp.unicode.org/Public/UNIDATA/UnicodeData.txt and the following license
1023 holds for these:
1024
1025 Copyright Â© 1991-2011 Unicode, Inc. All rights reserved. Distributed under the
1026 Terms of Use in http://www.unicode.org/copyright.html.
1027
1028 Permission is hereby granted, free of charge, to any person obtaining a copy of
1029 the Unicode data files and any associated documentation (the "Data Files") or
1030 Unicode software and any associated documentation (the "Software") to deal in
1031 the Data Files or Software without restriction, including without limitation
1032 the rights to use, copy, modify, merge, publish, distribute, and/or sell copies
1033 of the Data Files or Software, and to permit persons to whom the Data Files or
1034 Software are furnished to do so, provided that (a) the above copyright
1035 notice(s) and this permission notice appear with all copies of the Data Files
1036 or Software, (b) both the above copyright notice(s) and this permission notice
1037 appear in associated documentation, and (c) there is clear notice in each
1038 modified Data File or in the Software as well as in the documentation
1039 associated with the Data File(s) or Software that the data or software has been
1040 modified.
1041
1042 THE DATA FILES AND SOFTWARE ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
1043 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
1044 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD
1045 PARTY RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS INCLUDED IN
1046 THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL
1047 DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
1048 WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
1049 OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THE DATA FILES OR
1050 SOFTWARE.
1051
1052 Except as contained in this notice, the name of a copyright holder shall not be
1053 used in advertising or otherwise to promote the sale, use or other dealings in
1054 these Data Files or Software without prior written authorization of the
1055 copyright holder.
1056 */
1057
1058 Uchar u8_toupper(Uchar ch)
1059 {
1060         switch(ch)
1061         {
1062                 case 0x0061: return 0x0041;
1063                 case 0x0062: return 0x0042;
1064                 case 0x0063: return 0x0043;
1065                 case 0x0064: return 0x0044;
1066                 case 0x0065: return 0x0045;
1067                 case 0x0066: return 0x0046;
1068                 case 0x0067: return 0x0047;
1069                 case 0x0068: return 0x0048;
1070                 case 0x0069: return 0x0049;
1071                 case 0x006A: return 0x004A;
1072                 case 0x006B: return 0x004B;
1073                 case 0x006C: return 0x004C;
1074                 case 0x006D: return 0x004D;
1075                 case 0x006E: return 0x004E;
1076                 case 0x006F: return 0x004F;
1077                 case 0x0070: return 0x0050;
1078                 case 0x0071: return 0x0051;
1079                 case 0x0072: return 0x0052;
1080                 case 0x0073: return 0x0053;
1081                 case 0x0074: return 0x0054;
1082                 case 0x0075: return 0x0055;
1083                 case 0x0076: return 0x0056;
1084                 case 0x0077: return 0x0057;
1085                 case 0x0078: return 0x0058;
1086                 case 0x0079: return 0x0059;
1087                 case 0x007A: return 0x005A;
1088                 case 0x00B5: return 0x039C;
1089                 case 0x00E0: return 0x00C0;
1090                 case 0x00E1: return 0x00C1;
1091                 case 0x00E2: return 0x00C2;
1092                 case 0x00E3: return 0x00C3;
1093                 case 0x00E4: return 0x00C4;
1094                 case 0x00E5: return 0x00C5;
1095                 case 0x00E6: return 0x00C6;
1096                 case 0x00E7: return 0x00C7;
1097                 case 0x00E8: return 0x00C8;
1098                 case 0x00E9: return 0x00C9;
1099                 case 0x00EA: return 0x00CA;
1100                 case 0x00EB: return 0x00CB;
1101                 case 0x00EC: return 0x00CC;
1102                 case 0x00ED: return 0x00CD;
1103                 case 0x00EE: return 0x00CE;
1104                 case 0x00EF: return 0x00CF;
1105                 case 0x00F0: return 0x00D0;
1106                 case 0x00F1: return 0x00D1;
1107                 case 0x00F2: return 0x00D2;
1108                 case 0x00F3: return 0x00D3;
1109                 case 0x00F4: return 0x00D4;
1110                 case 0x00F5: return 0x00D5;
1111                 case 0x00F6: return 0x00D6;
1112                 case 0x00F8: return 0x00D8;
1113                 case 0x00F9: return 0x00D9;
1114                 case 0x00FA: return 0x00DA;
1115                 case 0x00FB: return 0x00DB;
1116                 case 0x00FC: return 0x00DC;
1117                 case 0x00FD: return 0x00DD;
1118                 case 0x00FE: return 0x00DE;
1119                 case 0x00FF: return 0x0178;
1120                 case 0x0101: return 0x0100;
1121                 case 0x0103: return 0x0102;
1122                 case 0x0105: return 0x0104;
1123                 case 0x0107: return 0x0106;
1124                 case 0x0109: return 0x0108;
1125                 case 0x010B: return 0x010A;
1126                 case 0x010D: return 0x010C;
1127                 case 0x010F: return 0x010E;
1128                 case 0x0111: return 0x0110;
1129                 case 0x0113: return 0x0112;
1130                 case 0x0115: return 0x0114;
1131                 case 0x0117: return 0x0116;
1132                 case 0x0119: return 0x0118;
1133                 case 0x011B: return 0x011A;
1134                 case 0x011D: return 0x011C;
1135                 case 0x011F: return 0x011E;
1136                 case 0x0121: return 0x0120;
1137                 case 0x0123: return 0x0122;
1138                 case 0x0125: return 0x0124;
1139                 case 0x0127: return 0x0126;
1140                 case 0x0129: return 0x0128;
1141                 case 0x012B: return 0x012A;
1142                 case 0x012D: return 0x012C;
1143                 case 0x012F: return 0x012E;
1144                 case 0x0131: return 0x0049;
1145                 case 0x0133: return 0x0132;
1146                 case 0x0135: return 0x0134;
1147                 case 0x0137: return 0x0136;
1148                 case 0x013A: return 0x0139;
1149                 case 0x013C: return 0x013B;
1150                 case 0x013E: return 0x013D;
1151                 case 0x0140: return 0x013F;
1152                 case 0x0142: return 0x0141;
1153                 case 0x0144: return 0x0143;
1154                 case 0x0146: return 0x0145;
1155                 case 0x0148: return 0x0147;
1156                 case 0x014B: return 0x014A;
1157                 case 0x014D: return 0x014C;
1158                 case 0x014F: return 0x014E;
1159                 case 0x0151: return 0x0150;
1160                 case 0x0153: return 0x0152;
1161                 case 0x0155: return 0x0154;
1162                 case 0x0157: return 0x0156;
1163                 case 0x0159: return 0x0158;
1164                 case 0x015B: return 0x015A;
1165                 case 0x015D: return 0x015C;
1166                 case 0x015F: return 0x015E;
1167                 case 0x0161: return 0x0160;
1168                 case 0x0163: return 0x0162;
1169                 case 0x0165: return 0x0164;
1170                 case 0x0167: return 0x0166;
1171                 case 0x0169: return 0x0168;
1172                 case 0x016B: return 0x016A;
1173                 case 0x016D: return 0x016C;
1174                 case 0x016F: return 0x016E;
1175                 case 0x0171: return 0x0170;
1176                 case 0x0173: return 0x0172;
1177                 case 0x0175: return 0x0174;
1178                 case 0x0177: return 0x0176;
1179                 case 0x017A: return 0x0179;
1180                 case 0x017C: return 0x017B;
1181                 case 0x017E: return 0x017D;
1182                 case 0x017F: return 0x0053;
1183                 case 0x0180: return 0x0243;
1184                 case 0x0183: return 0x0182;
1185                 case 0x0185: return 0x0184;
1186                 case 0x0188: return 0x0187;
1187                 case 0x018C: return 0x018B;
1188                 case 0x0192: return 0x0191;
1189                 case 0x0195: return 0x01F6;
1190                 case 0x0199: return 0x0198;
1191                 case 0x019A: return 0x023D;
1192                 case 0x019E: return 0x0220;
1193                 case 0x01A1: return 0x01A0;
1194                 case 0x01A3: return 0x01A2;
1195                 case 0x01A5: return 0x01A4;
1196                 case 0x01A8: return 0x01A7;
1197                 case 0x01AD: return 0x01AC;
1198                 case 0x01B0: return 0x01AF;
1199                 case 0x01B4: return 0x01B3;
1200                 case 0x01B6: return 0x01B5;
1201                 case 0x01B9: return 0x01B8;
1202                 case 0x01BD: return 0x01BC;
1203                 case 0x01BF: return 0x01F7;
1204                 case 0x01C5: return 0x01C4;
1205                 case 0x01C6: return 0x01C4;
1206                 case 0x01C8: return 0x01C7;
1207                 case 0x01C9: return 0x01C7;
1208                 case 0x01CB: return 0x01CA;
1209                 case 0x01CC: return 0x01CA;
1210                 case 0x01CE: return 0x01CD;
1211                 case 0x01D0: return 0x01CF;
1212                 case 0x01D2: return 0x01D1;
1213                 case 0x01D4: return 0x01D3;
1214                 case 0x01D6: return 0x01D5;
1215                 case 0x01D8: return 0x01D7;
1216                 case 0x01DA: return 0x01D9;
1217                 case 0x01DC: return 0x01DB;
1218                 case 0x01DD: return 0x018E;
1219                 case 0x01DF: return 0x01DE;
1220                 case 0x01E1: return 0x01E0;
1221                 case 0x01E3: return 0x01E2;
1222                 case 0x01E5: return 0x01E4;
1223                 case 0x01E7: return 0x01E6;
1224                 case 0x01E9: return 0x01E8;
1225                 case 0x01EB: return 0x01EA;
1226                 case 0x01ED: return 0x01EC;
1227                 case 0x01EF: return 0x01EE;
1228                 case 0x01F2: return 0x01F1;
1229                 case 0x01F3: return 0x01F1;
1230                 case 0x01F5: return 0x01F4;
1231                 case 0x01F9: return 0x01F8;
1232                 case 0x01FB: return 0x01FA;
1233                 case 0x01FD: return 0x01FC;
1234                 case 0x01FF: return 0x01FE;
1235                 case 0x0201: return 0x0200;
1236                 case 0x0203: return 0x0202;
1237                 case 0x0205: return 0x0204;
1238                 case 0x0207: return 0x0206;
1239                 case 0x0209: return 0x0208;
1240                 case 0x020B: return 0x020A;
1241                 case 0x020D: return 0x020C;
1242                 case 0x020F: return 0x020E;
1243                 case 0x0211: return 0x0210;
1244                 case 0x0213: return 0x0212;
1245                 case 0x0215: return 0x0214;
1246                 case 0x0217: return 0x0216;
1247                 case 0x0219: return 0x0218;
1248                 case 0x021B: return 0x021A;
1249                 case 0x021D: return 0x021C;
1250                 case 0x021F: return 0x021E;
1251                 case 0x0223: return 0x0222;
1252                 case 0x0225: return 0x0224;
1253                 case 0x0227: return 0x0226;
1254                 case 0x0229: return 0x0228;
1255                 case 0x022B: return 0x022A;
1256                 case 0x022D: return 0x022C;
1257                 case 0x022F: return 0x022E;
1258                 case 0x0231: return 0x0230;
1259                 case 0x0233: return 0x0232;
1260                 case 0x023C: return 0x023B;
1261                 case 0x023F: return 0x2C7E;
1262                 case 0x0240: return 0x2C7F;
1263                 case 0x0242: return 0x0241;
1264                 case 0x0247: return 0x0246;
1265                 case 0x0249: return 0x0248;
1266                 case 0x024B: return 0x024A;
1267                 case 0x024D: return 0x024C;
1268                 case 0x024F: return 0x024E;
1269                 case 0x0250: return 0x2C6F;
1270                 case 0x0251: return 0x2C6D;
1271                 case 0x0252: return 0x2C70;
1272                 case 0x0253: return 0x0181;
1273                 case 0x0254: return 0x0186;
1274                 case 0x0256: return 0x0189;
1275                 case 0x0257: return 0x018A;
1276                 case 0x0259: return 0x018F;
1277                 case 0x025B: return 0x0190;
1278                 case 0x0260: return 0x0193;
1279                 case 0x0263: return 0x0194;
1280                 case 0x0265: return 0xA78D;
1281                 case 0x0268: return 0x0197;
1282                 case 0x0269: return 0x0196;
1283                 case 0x026B: return 0x2C62;
1284                 case 0x026F: return 0x019C;
1285                 case 0x0271: return 0x2C6E;
1286                 case 0x0272: return 0x019D;
1287                 case 0x0275: return 0x019F;
1288                 case 0x027D: return 0x2C64;
1289                 case 0x0280: return 0x01A6;
1290                 case 0x0283: return 0x01A9;
1291                 case 0x0288: return 0x01AE;
1292                 case 0x0289: return 0x0244;
1293                 case 0x028A: return 0x01B1;
1294                 case 0x028B: return 0x01B2;
1295                 case 0x028C: return 0x0245;
1296                 case 0x0292: return 0x01B7;
1297                 case 0x0345: return 0x0399;
1298                 case 0x0371: return 0x0370;
1299                 case 0x0373: return 0x0372;
1300                 case 0x0377: return 0x0376;
1301                 case 0x037B: return 0x03FD;
1302                 case 0x037C: return 0x03FE;
1303                 case 0x037D: return 0x03FF;
1304                 case 0x03AC: return 0x0386;
1305                 case 0x03AD: return 0x0388;
1306                 case 0x03AE: return 0x0389;
1307                 case 0x03AF: return 0x038A;
1308                 case 0x03B1: return 0x0391;
1309                 case 0x03B2: return 0x0392;
1310                 case 0x03B3: return 0x0393;
1311                 case 0x03B4: return 0x0394;
1312                 case 0x03B5: return 0x0395;
1313                 case 0x03B6: return 0x0396;
1314                 case 0x03B7: return 0x0397;
1315                 case 0x03B8: return 0x0398;
1316                 case 0x03B9: return 0x0399;
1317                 case 0x03BA: return 0x039A;
1318                 case 0x03BB: return 0x039B;
1319                 case 0x03BC: return 0x039C;
1320                 case 0x03BD: return 0x039D;
1321                 case 0x03BE: return 0x039E;
1322                 case 0x03BF: return 0x039F;
1323                 case 0x03C0: return 0x03A0;
1324                 case 0x03C1: return 0x03A1;
1325                 case 0x03C2: return 0x03A3;
1326                 case 0x03C3: return 0x03A3;
1327                 case 0x03C4: return 0x03A4;
1328                 case 0x03C5: return 0x03A5;
1329                 case 0x03C6: return 0x03A6;
1330                 case 0x03C7: return 0x03A7;
1331                 case 0x03C8: return 0x03A8;
1332                 case 0x03C9: return 0x03A9;
1333                 case 0x03CA: return 0x03AA;
1334                 case 0x03CB: return 0x03AB;
1335                 case 0x03CC: return 0x038C;
1336                 case 0x03CD: return 0x038E;
1337                 case 0x03CE: return 0x038F;
1338                 case 0x03D0: return 0x0392;
1339                 case 0x03D1: return 0x0398;
1340                 case 0x03D5: return 0x03A6;
1341                 case 0x03D6: return 0x03A0;
1342                 case 0x03D7: return 0x03CF;
1343                 case 0x03D9: return 0x03D8;
1344                 case 0x03DB: return 0x03DA;
1345                 case 0x03DD: return 0x03DC;
1346                 case 0x03DF: return 0x03DE;
1347                 case 0x03E1: return 0x03E0;
1348                 case 0x03E3: return 0x03E2;
1349                 case 0x03E5: return 0x03E4;
1350                 case 0x03E7: return 0x03E6;
1351                 case 0x03E9: return 0x03E8;
1352                 case 0x03EB: return 0x03EA;
1353                 case 0x03ED: return 0x03EC;
1354                 case 0x03EF: return 0x03EE;
1355                 case 0x03F0: return 0x039A;
1356                 case 0x03F1: return 0x03A1;
1357                 case 0x03F2: return 0x03F9;
1358                 case 0x03F5: return 0x0395;
1359                 case 0x03F8: return 0x03F7;
1360                 case 0x03FB: return 0x03FA;
1361                 case 0x0430: return 0x0410;
1362                 case 0x0431: return 0x0411;
1363                 case 0x0432: return 0x0412;
1364                 case 0x0433: return 0x0413;
1365                 case 0x0434: return 0x0414;
1366                 case 0x0435: return 0x0415;
1367                 case 0x0436: return 0x0416;
1368                 case 0x0437: return 0x0417;
1369                 case 0x0438: return 0x0418;
1370                 case 0x0439: return 0x0419;
1371                 case 0x043A: return 0x041A;
1372                 case 0x043B: return 0x041B;
1373                 case 0x043C: return 0x041C;
1374                 case 0x043D: return 0x041D;
1375                 case 0x043E: return 0x041E;
1376                 case 0x043F: return 0x041F;
1377                 case 0x0440: return 0x0420;
1378                 case 0x0441: return 0x0421;
1379                 case 0x0442: return 0x0422;
1380                 case 0x0443: return 0x0423;
1381                 case 0x0444: return 0x0424;
1382                 case 0x0445: return 0x0425;
1383                 case 0x0446: return 0x0426;
1384                 case 0x0447: return 0x0427;
1385                 case 0x0448: return 0x0428;
1386                 case 0x0449: return 0x0429;
1387                 case 0x044A: return 0x042A;
1388                 case 0x044B: return 0x042B;
1389                 case 0x044C: return 0x042C;
1390                 case 0x044D: return 0x042D;
1391                 case 0x044E: return 0x042E;
1392                 case 0x044F: return 0x042F;
1393                 case 0x0450: return 0x0400;
1394                 case 0x0451: return 0x0401;
1395                 case 0x0452: return 0x0402;
1396                 case 0x0453: return 0x0403;
1397                 case 0x0454: return 0x0404;
1398                 case 0x0455: return 0x0405;
1399                 case 0x0456: return 0x0406;
1400                 case 0x0457: return 0x0407;
1401                 case 0x0458: return 0x0408;
1402                 case 0x0459: return 0x0409;
1403                 case 0x045A: return 0x040A;
1404                 case 0x045B: return 0x040B;
1405                 case 0x045C: return 0x040C;
1406                 case 0x045D: return 0x040D;
1407                 case 0x045E: return 0x040E;
1408                 case 0x045F: return 0x040F;
1409                 case 0x0461: return 0x0460;
1410                 case 0x0463: return 0x0462;
1411                 case 0x0465: return 0x0464;
1412                 case 0x0467: return 0x0466;
1413                 case 0x0469: return 0x0468;
1414                 case 0x046B: return 0x046A;
1415                 case 0x046D: return 0x046C;
1416                 case 0x046F: return 0x046E;
1417                 case 0x0471: return 0x0470;
1418                 case 0x0473: return 0x0472;
1419                 case 0x0475: return 0x0474;
1420                 case 0x0477: return 0x0476;
1421                 case 0x0479: return 0x0478;
1422                 case 0x047B: return 0x047A;
1423                 case 0x047D: return 0x047C;
1424                 case 0x047F: return 0x047E;
1425                 case 0x0481: return 0x0480;
1426                 case 0x048B: return 0x048A;
1427                 case 0x048D: return 0x048C;
1428                 case 0x048F: return 0x048E;
1429                 case 0x0491: return 0x0490;
1430                 case 0x0493: return 0x0492;
1431                 case 0x0495: return 0x0494;
1432                 case 0x0497: return 0x0496;
1433                 case 0x0499: return 0x0498;
1434                 case 0x049B: return 0x049A;
1435                 case 0x049D: return 0x049C;
1436                 case 0x049F: return 0x049E;
1437                 case 0x04A1: return 0x04A0;
1438                 case 0x04A3: return 0x04A2;
1439                 case 0x04A5: return 0x04A4;
1440                 case 0x04A7: return 0x04A6;
1441                 case 0x04A9: return 0x04A8;
1442                 case 0x04AB: return 0x04AA;
1443                 case 0x04AD: return 0x04AC;
1444                 case 0x04AF: return 0x04AE;
1445                 case 0x04B1: return 0x04B0;
1446                 case 0x04B3: return 0x04B2;
1447                 case 0x04B5: return 0x04B4;
1448                 case 0x04B7: return 0x04B6;
1449                 case 0x04B9: return 0x04B8;
1450                 case 0x04BB: return 0x04BA;
1451                 case 0x04BD: return 0x04BC;
1452                 case 0x04BF: return 0x04BE;
1453                 case 0x04C2: return 0x04C1;
1454                 case 0x04C4: return 0x04C3;
1455                 case 0x04C6: return 0x04C5;
1456                 case 0x04C8: return 0x04C7;
1457                 case 0x04CA: return 0x04C9;
1458                 case 0x04CC: return 0x04CB;
1459                 case 0x04CE: return 0x04CD;
1460                 case 0x04CF: return 0x04C0;
1461                 case 0x04D1: return 0x04D0;
1462                 case 0x04D3: return 0x04D2;
1463                 case 0x04D5: return 0x04D4;
1464                 case 0x04D7: return 0x04D6;
1465                 case 0x04D9: return 0x04D8;
1466                 case 0x04DB: return 0x04DA;
1467                 case 0x04DD: return 0x04DC;
1468                 case 0x04DF: return 0x04DE;
1469                 case 0x04E1: return 0x04E0;
1470                 case 0x04E3: return 0x04E2;
1471                 case 0x04E5: return 0x04E4;
1472                 case 0x04E7: return 0x04E6;
1473                 case 0x04E9: return 0x04E8;
1474                 case 0x04EB: return 0x04EA;
1475                 case 0x04ED: return 0x04EC;
1476                 case 0x04EF: return 0x04EE;
1477                 case 0x04F1: return 0x04F0;
1478                 case 0x04F3: return 0x04F2;
1479                 case 0x04F5: return 0x04F4;
1480                 case 0x04F7: return 0x04F6;
1481                 case 0x04F9: return 0x04F8;
1482                 case 0x04FB: return 0x04FA;
1483                 case 0x04FD: return 0x04FC;
1484                 case 0x04FF: return 0x04FE;
1485                 case 0x0501: return 0x0500;
1486                 case 0x0503: return 0x0502;
1487                 case 0x0505: return 0x0504;
1488                 case 0x0507: return 0x0506;
1489                 case 0x0509: return 0x0508;
1490                 case 0x050B: return 0x050A;
1491                 case 0x050D: return 0x050C;
1492                 case 0x050F: return 0x050E;
1493                 case 0x0511: return 0x0510;
1494                 case 0x0513: return 0x0512;
1495                 case 0x0515: return 0x0514;
1496                 case 0x0517: return 0x0516;
1497                 case 0x0519: return 0x0518;
1498                 case 0x051B: return 0x051A;
1499                 case 0x051D: return 0x051C;
1500                 case 0x051F: return 0x051E;
1501                 case 0x0521: return 0x0520;
1502                 case 0x0523: return 0x0522;
1503                 case 0x0525: return 0x0524;
1504                 case 0x0527: return 0x0526;
1505                 case 0x0561: return 0x0531;
1506                 case 0x0562: return 0x0532;
1507                 case 0x0563: return 0x0533;
1508                 case 0x0564: return 0x0534;
1509                 case 0x0565: return 0x0535;
1510                 case 0x0566: return 0x0536;
1511                 case 0x0567: return 0x0537;
1512                 case 0x0568: return 0x0538;
1513                 case 0x0569: return 0x0539;
1514                 case 0x056A: return 0x053A;
1515                 case 0x056B: return 0x053B;
1516                 case 0x056C: return 0x053C;
1517                 case 0x056D: return 0x053D;
1518                 case 0x056E: return 0x053E;
1519                 case 0x056F: return 0x053F;
1520                 case 0x0570: return 0x0540;
1521                 case 0x0571: return 0x0541;
1522                 case 0x0572: return 0x0542;
1523                 case 0x0573: return 0x0543;
1524                 case 0x0574: return 0x0544;
1525                 case 0x0575: return 0x0545;
1526                 case 0x0576: return 0x0546;
1527                 case 0x0577: return 0x0547;
1528                 case 0x0578: return 0x0548;
1529                 case 0x0579: return 0x0549;
1530                 case 0x057A: return 0x054A;
1531                 case 0x057B: return 0x054B;
1532                 case 0x057C: return 0x054C;
1533                 case 0x057D: return 0x054D;
1534                 case 0x057E: return 0x054E;
1535                 case 0x057F: return 0x054F;
1536                 case 0x0580: return 0x0550;
1537                 case 0x0581: return 0x0551;
1538                 case 0x0582: return 0x0552;
1539                 case 0x0583: return 0x0553;
1540                 case 0x0584: return 0x0554;
1541                 case 0x0585: return 0x0555;
1542                 case 0x0586: return 0x0556;
1543                 case 0x1D79: return 0xA77D;
1544                 case 0x1D7D: return 0x2C63;
1545                 case 0x1E01: return 0x1E00;
1546                 case 0x1E03: return 0x1E02;
1547                 case 0x1E05: return 0x1E04;
1548                 case 0x1E07: return 0x1E06;
1549                 case 0x1E09: return 0x1E08;
1550                 case 0x1E0B: return 0x1E0A;
1551                 case 0x1E0D: return 0x1E0C;
1552                 case 0x1E0F: return 0x1E0E;
1553                 case 0x1E11: return 0x1E10;
1554                 case 0x1E13: return 0x1E12;
1555                 case 0x1E15: return 0x1E14;
1556                 case 0x1E17: return 0x1E16;
1557                 case 0x1E19: return 0x1E18;
1558                 case 0x1E1B: return 0x1E1A;
1559                 case 0x1E1D: return 0x1E1C;
1560                 case 0x1E1F: return 0x1E1E;
1561                 case 0x1E21: return 0x1E20;
1562                 case 0x1E23: return 0x1E22;
1563                 case 0x1E25: return 0x1E24;
1564                 case 0x1E27: return 0x1E26;
1565                 case 0x1E29: return 0x1E28;
1566                 case 0x1E2B: return 0x1E2A;
1567                 case 0x1E2D: return 0x1E2C;
1568                 case 0x1E2F: return 0x1E2E;
1569                 case 0x1E31: return 0x1E30;
1570                 case 0x1E33: return 0x1E32;
1571                 case 0x1E35: return 0x1E34;
1572                 case 0x1E37: return 0x1E36;
1573                 case 0x1E39: return 0x1E38;
1574                 case 0x1E3B: return 0x1E3A;
1575                 case 0x1E3D: return 0x1E3C;
1576                 case 0x1E3F: return 0x1E3E;
1577                 case 0x1E41: return 0x1E40;
1578                 case 0x1E43: return 0x1E42;
1579                 case 0x1E45: return 0x1E44;
1580                 case 0x1E47: return 0x1E46;
1581                 case 0x1E49: return 0x1E48;
1582                 case 0x1E4B: return 0x1E4A;
1583                 case 0x1E4D: return 0x1E4C;
1584                 case 0x1E4F: return 0x1E4E;
1585                 case 0x1E51: return 0x1E50;
1586                 case 0x1E53: return 0x1E52;
1587                 case 0x1E55: return 0x1E54;
1588                 case 0x1E57: return 0x1E56;
1589                 case 0x1E59: return 0x1E58;
1590                 case 0x1E5B: return 0x1E5A;
1591                 case 0x1E5D: return 0x1E5C;
1592                 case 0x1E5F: return 0x1E5E;
1593                 case 0x1E61: return 0x1E60;
1594                 case 0x1E63: return 0x1E62;
1595                 case 0x1E65: return 0x1E64;
1596                 case 0x1E67: return 0x1E66;
1597                 case 0x1E69: return 0x1E68;
1598                 case 0x1E6B: return 0x1E6A;
1599                 case 0x1E6D: return 0x1E6C;
1600                 case 0x1E6F: return 0x1E6E;
1601                 case 0x1E71: return 0x1E70;
1602                 case 0x1E73: return 0x1E72;
1603                 case 0x1E75: return 0x1E74;
1604                 case 0x1E77: return 0x1E76;
1605                 case 0x1E79: return 0x1E78;
1606                 case 0x1E7B: return 0x1E7A;
1607                 case 0x1E7D: return 0x1E7C;
1608                 case 0x1E7F: return 0x1E7E;
1609                 case 0x1E81: return 0x1E80;
1610                 case 0x1E83: return 0x1E82;
1611                 case 0x1E85: return 0x1E84;
1612                 case 0x1E87: return 0x1E86;
1613                 case 0x1E89: return 0x1E88;
1614                 case 0x1E8B: return 0x1E8A;
1615                 case 0x1E8D: return 0x1E8C;
1616                 case 0x1E8F: return 0x1E8E;
1617                 case 0x1E91: return 0x1E90;
1618                 case 0x1E93: return 0x1E92;
1619                 case 0x1E95: return 0x1E94;
1620                 case 0x1E9B: return 0x1E60;
1621                 case 0x1EA1: return 0x1EA0;
1622                 case 0x1EA3: return 0x1EA2;
1623                 case 0x1EA5: return 0x1EA4;
1624                 case 0x1EA7: return 0x1EA6;
1625                 case 0x1EA9: return 0x1EA8;
1626                 case 0x1EAB: return 0x1EAA;
1627                 case 0x1EAD: return 0x1EAC;
1628                 case 0x1EAF: return 0x1EAE;
1629                 case 0x1EB1: return 0x1EB0;
1630                 case 0x1EB3: return 0x1EB2;
1631                 case 0x1EB5: return 0x1EB4;
1632                 case 0x1EB7: return 0x1EB6;
1633                 case 0x1EB9: return 0x1EB8;
1634                 case 0x1EBB: return 0x1EBA;
1635                 case 0x1EBD: return 0x1EBC;
1636                 case 0x1EBF: return 0x1EBE;
1637                 case 0x1EC1: return 0x1EC0;
1638                 case 0x1EC3: return 0x1EC2;
1639                 case 0x1EC5: return 0x1EC4;
1640                 case 0x1EC7: return 0x1EC6;
1641                 case 0x1EC9: return 0x1EC8;
1642                 case 0x1ECB: return 0x1ECA;
1643                 case 0x1ECD: return 0x1ECC;
1644                 case 0x1ECF: return 0x1ECE;
1645                 case 0x1ED1: return 0x1ED0;
1646                 case 0x1ED3: return 0x1ED2;
1647                 case 0x1ED5: return 0x1ED4;
1648                 case 0x1ED7: return 0x1ED6;
1649                 case 0x1ED9: return 0x1ED8;
1650                 case 0x1EDB: return 0x1EDA;
1651                 case 0x1EDD: return 0x1EDC;
1652                 case 0x1EDF: return 0x1EDE;
1653                 case 0x1EE1: return 0x1EE0;
1654                 case 0x1EE3: return 0x1EE2;
1655                 case 0x1EE5: return 0x1EE4;
1656                 case 0x1EE7: return 0x1EE6;
1657                 case 0x1EE9: return 0x1EE8;
1658                 case 0x1EEB: return 0x1EEA;
1659                 case 0x1EED: return 0x1EEC;
1660                 case 0x1EEF: return 0x1EEE;
1661                 case 0x1EF1: return 0x1EF0;
1662                 case 0x1EF3: return 0x1EF2;
1663                 case 0x1EF5: return 0x1EF4;
1664                 case 0x1EF7: return 0x1EF6;
1665                 case 0x1EF9: return 0x1EF8;
1666                 case 0x1EFB: return 0x1EFA;
1667                 case 0x1EFD: return 0x1EFC;
1668                 case 0x1EFF: return 0x1EFE;
1669                 case 0x1F00: return 0x1F08;
1670                 case 0x1F01: return 0x1F09;
1671                 case 0x1F02: return 0x1F0A;
1672                 case 0x1F03: return 0x1F0B;
1673                 case 0x1F04: return 0x1F0C;
1674                 case 0x1F05: return 0x1F0D;
1675                 case 0x1F06: return 0x1F0E;
1676                 case 0x1F07: return 0x1F0F;
1677                 case 0x1F10: return 0x1F18;
1678                 case 0x1F11: return 0x1F19;
1679                 case 0x1F12: return 0x1F1A;
1680                 case 0x1F13: return 0x1F1B;
1681                 case 0x1F14: return 0x1F1C;
1682                 case 0x1F15: return 0x1F1D;
1683                 case 0x1F20: return 0x1F28;
1684                 case 0x1F21: return 0x1F29;
1685                 case 0x1F22: return 0x1F2A;
1686                 case 0x1F23: return 0x1F2B;
1687                 case 0x1F24: return 0x1F2C;
1688                 case 0x1F25: return 0x1F2D;
1689                 case 0x1F26: return 0x1F2E;
1690                 case 0x1F27: return 0x1F2F;
1691                 case 0x1F30: return 0x1F38;
1692                 case 0x1F31: return 0x1F39;
1693                 case 0x1F32: return 0x1F3A;
1694                 case 0x1F33: return 0x1F3B;
1695                 case 0x1F34: return 0x1F3C;
1696                 case 0x1F35: return 0x1F3D;
1697                 case 0x1F36: return 0x1F3E;
1698                 case 0x1F37: return 0x1F3F;
1699                 case 0x1F40: return 0x1F48;
1700                 case 0x1F41: return 0x1F49;
1701                 case 0x1F42: return 0x1F4A;
1702                 case 0x1F43: return 0x1F4B;
1703                 case 0x1F44: return 0x1F4C;
1704                 case 0x1F45: return 0x1F4D;
1705                 case 0x1F51: return 0x1F59;
1706                 case 0x1F53: return 0x1F5B;
1707                 case 0x1F55: return 0x1F5D;
1708                 case 0x1F57: return 0x1F5F;
1709                 case 0x1F60: return 0x1F68;
1710                 case 0x1F61: return 0x1F69;
1711                 case 0x1F62: return 0x1F6A;
1712                 case 0x1F63: return 0x1F6B;
1713                 case 0x1F64: return 0x1F6C;
1714                 case 0x1F65: return 0x1F6D;
1715                 case 0x1F66: return 0x1F6E;
1716                 case 0x1F67: return 0x1F6F;
1717                 case 0x1F70: return 0x1FBA;
1718                 case 0x1F71: return 0x1FBB;
1719                 case 0x1F72: return 0x1FC8;
1720                 case 0x1F73: return 0x1FC9;
1721                 case 0x1F74: return 0x1FCA;
1722                 case 0x1F75: return 0x1FCB;
1723                 case 0x1F76: return 0x1FDA;
1724                 case 0x1F77: return 0x1FDB;
1725                 case 0x1F78: return 0x1FF8;
1726                 case 0x1F79: return 0x1FF9;
1727                 case 0x1F7A: return 0x1FEA;
1728                 case 0x1F7B: return 0x1FEB;
1729                 case 0x1F7C: return 0x1FFA;
1730                 case 0x1F7D: return 0x1FFB;
1731                 case 0x1F80: return 0x1F88;
1732                 case 0x1F81: return 0x1F89;
1733                 case 0x1F82: return 0x1F8A;
1734                 case 0x1F83: return 0x1F8B;
1735                 case 0x1F84: return 0x1F8C;
1736                 case 0x1F85: return 0x1F8D;
1737                 case 0x1F86: return 0x1F8E;
1738                 case 0x1F87: return 0x1F8F;
1739                 case 0x1F90: return 0x1F98;
1740                 case 0x1F91: return 0x1F99;
1741                 case 0x1F92: return 0x1F9A;
1742                 case 0x1F93: return 0x1F9B;
1743                 case 0x1F94: return 0x1F9C;
1744                 case 0x1F95: return 0x1F9D;
1745                 case 0x1F96: return 0x1F9E;
1746                 case 0x1F97: return 0x1F9F;
1747                 case 0x1FA0: return 0x1FA8;
1748                 case 0x1FA1: return 0x1FA9;
1749                 case 0x1FA2: return 0x1FAA;
1750                 case 0x1FA3: return 0x1FAB;
1751                 case 0x1FA4: return 0x1FAC;
1752                 case 0x1FA5: return 0x1FAD;
1753                 case 0x1FA6: return 0x1FAE;
1754                 case 0x1FA7: return 0x1FAF;
1755                 case 0x1FB0: return 0x1FB8;
1756                 case 0x1FB1: return 0x1FB9;
1757                 case 0x1FB3: return 0x1FBC;
1758                 case 0x1FBE: return 0x0399;
1759                 case 0x1FC3: return 0x1FCC;
1760                 case 0x1FD0: return 0x1FD8;
1761                 case 0x1FD1: return 0x1FD9;
1762                 case 0x1FE0: return 0x1FE8;
1763                 case 0x1FE1: return 0x1FE9;
1764                 case 0x1FE5: return 0x1FEC;
1765                 case 0x1FF3: return 0x1FFC;
1766                 case 0x214E: return 0x2132;
1767                 case 0x2170: return 0x2160;
1768                 case 0x2171: return 0x2161;
1769                 case 0x2172: return 0x2162;
1770                 case 0x2173: return 0x2163;
1771                 case 0x2174: return 0x2164;
1772                 case 0x2175: return 0x2165;
1773                 case 0x2176: return 0x2166;
1774                 case 0x2177: return 0x2167;
1775                 case 0x2178: return 0x2168;
1776                 case 0x2179: return 0x2169;
1777                 case 0x217A: return 0x216A;
1778                 case 0x217B: return 0x216B;
1779                 case 0x217C: return 0x216C;
1780                 case 0x217D: return 0x216D;
1781                 case 0x217E: return 0x216E;
1782                 case 0x217F: return 0x216F;
1783                 case 0x2184: return 0x2183;
1784                 case 0x24D0: return 0x24B6;
1785                 case 0x24D1: return 0x24B7;
1786                 case 0x24D2: return 0x24B8;
1787                 case 0x24D3: return 0x24B9;
1788                 case 0x24D4: return 0x24BA;
1789                 case 0x24D5: return 0x24BB;
1790                 case 0x24D6: return 0x24BC;
1791                 case 0x24D7: return 0x24BD;
1792                 case 0x24D8: return 0x24BE;
1793                 case 0x24D9: return 0x24BF;
1794                 case 0x24DA: return 0x24C0;
1795                 case 0x24DB: return 0x24C1;
1796                 case 0x24DC: return 0x24C2;
1797                 case 0x24DD: return 0x24C3;
1798                 case 0x24DE: return 0x24C4;
1799                 case 0x24DF: return 0x24C5;
1800                 case 0x24E0: return 0x24C6;
1801                 case 0x24E1: return 0x24C7;
1802                 case 0x24E2: return 0x24C8;
1803                 case 0x24E3: return 0x24C9;
1804                 case 0x24E4: return 0x24CA;
1805                 case 0x24E5: return 0x24CB;
1806                 case 0x24E6: return 0x24CC;
1807                 case 0x24E7: return 0x24CD;
1808                 case 0x24E8: return 0x24CE;
1809                 case 0x24E9: return 0x24CF;
1810                 case 0x2C30: return 0x2C00;
1811                 case 0x2C31: return 0x2C01;
1812                 case 0x2C32: return 0x2C02;
1813                 case 0x2C33: return 0x2C03;
1814                 case 0x2C34: return 0x2C04;
1815                 case 0x2C35: return 0x2C05;
1816                 case 0x2C36: return 0x2C06;
1817                 case 0x2C37: return 0x2C07;
1818                 case 0x2C38: return 0x2C08;
1819                 case 0x2C39: return 0x2C09;
1820                 case 0x2C3A: return 0x2C0A;
1821                 case 0x2C3B: return 0x2C0B;
1822                 case 0x2C3C: return 0x2C0C;
1823                 case 0x2C3D: return 0x2C0D;
1824                 case 0x2C3E: return 0x2C0E;
1825                 case 0x2C3F: return 0x2C0F;
1826                 case 0x2C40: return 0x2C10;
1827                 case 0x2C41: return 0x2C11;
1828                 case 0x2C42: return 0x2C12;
1829                 case 0x2C43: return 0x2C13;
1830                 case 0x2C44: return 0x2C14;
1831                 case 0x2C45: return 0x2C15;
1832                 case 0x2C46: return 0x2C16;
1833                 case 0x2C47: return 0x2C17;
1834                 case 0x2C48: return 0x2C18;
1835                 case 0x2C49: return 0x2C19;
1836                 case 0x2C4A: return 0x2C1A;
1837                 case 0x2C4B: return 0x2C1B;
1838                 case 0x2C4C: return 0x2C1C;
1839                 case 0x2C4D: return 0x2C1D;
1840                 case 0x2C4E: return 0x2C1E;
1841                 case 0x2C4F: return 0x2C1F;
1842                 case 0x2C50: return 0x2C20;
1843                 case 0x2C51: return 0x2C21;
1844                 case 0x2C52: return 0x2C22;
1845                 case 0x2C53: return 0x2C23;
1846                 case 0x2C54: return 0x2C24;
1847                 case 0x2C55: return 0x2C25;
1848                 case 0x2C56: return 0x2C26;
1849                 case 0x2C57: return 0x2C27;
1850                 case 0x2C58: return 0x2C28;
1851                 case 0x2C59: return 0x2C29;
1852                 case 0x2C5A: return 0x2C2A;
1853                 case 0x2C5B: return 0x2C2B;
1854                 case 0x2C5C: return 0x2C2C;
1855                 case 0x2C5D: return 0x2C2D;
1856                 case 0x2C5E: return 0x2C2E;
1857                 case 0x2C61: return 0x2C60;
1858                 case 0x2C65: return 0x023A;
1859                 case 0x2C66: return 0x023E;
1860                 case 0x2C68: return 0x2C67;
1861                 case 0x2C6A: return 0x2C69;
1862                 case 0x2C6C: return 0x2C6B;
1863                 case 0x2C73: return 0x2C72;
1864                 case 0x2C76: return 0x2C75;
1865                 case 0x2C81: return 0x2C80;
1866                 case 0x2C83: return 0x2C82;
1867                 case 0x2C85: return 0x2C84;
1868                 case 0x2C87: return 0x2C86;
1869                 case 0x2C89: return 0x2C88;
1870                 case 0x2C8B: return 0x2C8A;
1871                 case 0x2C8D: return 0x2C8C;
1872                 case 0x2C8F: return 0x2C8E;
1873                 case 0x2C91: return 0x2C90;
1874                 case 0x2C93: return 0x2C92;
1875                 case 0x2C95: return 0x2C94;
1876                 case 0x2C97: return 0x2C96;
1877                 case 0x2C99: return 0x2C98;
1878                 case 0x2C9B: return 0x2C9A;
1879                 case 0x2C9D: return 0x2C9C;
1880                 case 0x2C9F: return 0x2C9E;
1881                 case 0x2CA1: return 0x2CA0;
1882                 case 0x2CA3: return 0x2CA2;
1883                 case 0x2CA5: return 0x2CA4;
1884                 case 0x2CA7: return 0x2CA6;
1885                 case 0x2CA9: return 0x2CA8;
1886                 case 0x2CAB: return 0x2CAA;
1887                 case 0x2CAD: return 0x2CAC;
1888                 case 0x2CAF: return 0x2CAE;
1889                 case 0x2CB1: return 0x2CB0;
1890                 case 0x2CB3: return 0x2CB2;
1891                 case 0x2CB5: return 0x2CB4;
1892                 case 0x2CB7: return 0x2CB6;
1893                 case 0x2CB9: return 0x2CB8;
1894                 case 0x2CBB: return 0x2CBA;
1895                 case 0x2CBD: return 0x2CBC;
1896                 case 0x2CBF: return 0x2CBE;
1897                 case 0x2CC1: return 0x2CC0;
1898                 case 0x2CC3: return 0x2CC2;
1899                 case 0x2CC5: return 0x2CC4;
1900                 case 0x2CC7: return 0x2CC6;
1901                 case 0x2CC9: return 0x2CC8;
1902                 case 0x2CCB: return 0x2CCA;
1903                 case 0x2CCD: return 0x2CCC;
1904                 case 0x2CCF: return 0x2CCE;
1905                 case 0x2CD1: return 0x2CD0;
1906                 case 0x2CD3: return 0x2CD2;
1907                 case 0x2CD5: return 0x2CD4;
1908                 case 0x2CD7: return 0x2CD6;
1909                 case 0x2CD9: return 0x2CD8;
1910                 case 0x2CDB: return 0x2CDA;
1911                 case 0x2CDD: return 0x2CDC;
1912                 case 0x2CDF: return 0x2CDE;
1913                 case 0x2CE1: return 0x2CE0;
1914                 case 0x2CE3: return 0x2CE2;
1915                 case 0x2CEC: return 0x2CEB;
1916                 case 0x2CEE: return 0x2CED;
1917                 case 0x2D00: return 0x10A0;
1918                 case 0x2D01: return 0x10A1;
1919                 case 0x2D02: return 0x10A2;
1920                 case 0x2D03: return 0x10A3;
1921                 case 0x2D04: return 0x10A4;
1922                 case 0x2D05: return 0x10A5;
1923                 case 0x2D06: return 0x10A6;
1924                 case 0x2D07: return 0x10A7;
1925                 case 0x2D08: return 0x10A8;
1926                 case 0x2D09: return 0x10A9;
1927                 case 0x2D0A: return 0x10AA;
1928                 case 0x2D0B: return 0x10AB;
1929                 case 0x2D0C: return 0x10AC;
1930                 case 0x2D0D: return 0x10AD;
1931                 case 0x2D0E: return 0x10AE;
1932                 case 0x2D0F: return 0x10AF;
1933                 case 0x2D10: return 0x10B0;
1934                 case 0x2D11: return 0x10B1;
1935                 case 0x2D12: return 0x10B2;
1936                 case 0x2D13: return 0x10B3;
1937                 case 0x2D14: return 0x10B4;
1938                 case 0x2D15: return 0x10B5;
1939                 case 0x2D16: return 0x10B6;
1940                 case 0x2D17: return 0x10B7;
1941                 case 0x2D18: return 0x10B8;
1942                 case 0x2D19: return 0x10B9;
1943                 case 0x2D1A: return 0x10BA;
1944                 case 0x2D1B: return 0x10BB;
1945                 case 0x2D1C: return 0x10BC;
1946                 case 0x2D1D: return 0x10BD;
1947                 case 0x2D1E: return 0x10BE;
1948                 case 0x2D1F: return 0x10BF;
1949                 case 0x2D20: return 0x10C0;
1950                 case 0x2D21: return 0x10C1;
1951                 case 0x2D22: return 0x10C2;
1952                 case 0x2D23: return 0x10C3;
1953                 case 0x2D24: return 0x10C4;
1954                 case 0x2D25: return 0x10C5;
1955                 case 0xA641: return 0xA640;
1956                 case 0xA643: return 0xA642;
1957                 case 0xA645: return 0xA644;
1958                 case 0xA647: return 0xA646;
1959                 case 0xA649: return 0xA648;
1960                 case 0xA64B: return 0xA64A;
1961                 case 0xA64D: return 0xA64C;
1962                 case 0xA64F: return 0xA64E;
1963                 case 0xA651: return 0xA650;
1964                 case 0xA653: return 0xA652;
1965                 case 0xA655: return 0xA654;
1966                 case 0xA657: return 0xA656;
1967                 case 0xA659: return 0xA658;
1968                 case 0xA65B: return 0xA65A;
1969                 case 0xA65D: return 0xA65C;
1970                 case 0xA65F: return 0xA65E;
1971                 case 0xA661: return 0xA660;
1972                 case 0xA663: return 0xA662;
1973                 case 0xA665: return 0xA664;
1974                 case 0xA667: return 0xA666;
1975                 case 0xA669: return 0xA668;
1976                 case 0xA66B: return 0xA66A;
1977                 case 0xA66D: return 0xA66C;
1978                 case 0xA681: return 0xA680;
1979                 case 0xA683: return 0xA682;
1980                 case 0xA685: return 0xA684;
1981                 case 0xA687: return 0xA686;
1982                 case 0xA689: return 0xA688;
1983                 case 0xA68B: return 0xA68A;
1984                 case 0xA68D: return 0xA68C;
1985                 case 0xA68F: return 0xA68E;
1986                 case 0xA691: return 0xA690;
1987                 case 0xA693: return 0xA692;
1988                 case 0xA695: return 0xA694;
1989                 case 0xA697: return 0xA696;
1990                 case 0xA723: return 0xA722;
1991                 case 0xA725: return 0xA724;
1992                 case 0xA727: return 0xA726;
1993                 case 0xA729: return 0xA728;
1994                 case 0xA72B: return 0xA72A;
1995                 case 0xA72D: return 0xA72C;
1996                 case 0xA72F: return 0xA72E;
1997                 case 0xA733: return 0xA732;
1998                 case 0xA735: return 0xA734;
1999                 case 0xA737: return 0xA736;
2000                 case 0xA739: return 0xA738;
2001                 case 0xA73B: return 0xA73A;
2002                 case 0xA73D: return 0xA73C;
2003                 case 0xA73F: return 0xA73E;
2004                 case 0xA741: return 0xA740;
2005                 case 0xA743: return 0xA742;
2006                 case 0xA745: return 0xA744;
2007                 case 0xA747: return 0xA746;
2008                 case 0xA749: return 0xA748;
2009                 case 0xA74B: return 0xA74A;
2010                 case 0xA74D: return 0xA74C;
2011                 case 0xA74F: return 0xA74E;
2012                 case 0xA751: return 0xA750;
2013                 case 0xA753: return 0xA752;
2014                 case 0xA755: return 0xA754;
2015                 case 0xA757: return 0xA756;
2016                 case 0xA759: return 0xA758;
2017                 case 0xA75B: return 0xA75A;
2018                 case 0xA75D: return 0xA75C;
2019                 case 0xA75F: return 0xA75E;
2020                 case 0xA761: return 0xA760;
2021                 case 0xA763: return 0xA762;
2022                 case 0xA765: return 0xA764;
2023                 case 0xA767: return 0xA766;
2024                 case 0xA769: return 0xA768;
2025                 case 0xA76B: return 0xA76A;
2026                 case 0xA76D: return 0xA76C;
2027                 case 0xA76F: return 0xA76E;
2028                 case 0xA77A: return 0xA779;
2029                 case 0xA77C: return 0xA77B;
2030                 case 0xA77F: return 0xA77E;
2031                 case 0xA781: return 0xA780;
2032                 case 0xA783: return 0xA782;
2033                 case 0xA785: return 0xA784;
2034                 case 0xA787: return 0xA786;
2035                 case 0xA78C: return 0xA78B;
2036                 case 0xA791: return 0xA790;
2037                 case 0xA7A1: return 0xA7A0;
2038                 case 0xA7A3: return 0xA7A2;
2039                 case 0xA7A5: return 0xA7A4;
2040                 case 0xA7A7: return 0xA7A6;
2041                 case 0xA7A9: return 0xA7A8;
2042                 case 0xFF41: return 0xFF21;
2043                 case 0xFF42: return 0xFF22;
2044                 case 0xFF43: return 0xFF23;
2045                 case 0xFF44: return 0xFF24;
2046                 case 0xFF45: return 0xFF25;
2047                 case 0xFF46: return 0xFF26;
2048                 case 0xFF47: return 0xFF27;
2049                 case 0xFF48: return 0xFF28;
2050                 case 0xFF49: return 0xFF29;
2051                 case 0xFF4A: return 0xFF2A;
2052                 case 0xFF4B: return 0xFF2B;
2053                 case 0xFF4C: return 0xFF2C;
2054                 case 0xFF4D: return 0xFF2D;
2055                 case 0xFF4E: return 0xFF2E;
2056                 case 0xFF4F: return 0xFF2F;
2057                 case 0xFF50: return 0xFF30;
2058                 case 0xFF51: return 0xFF31;
2059                 case 0xFF52: return 0xFF32;
2060                 case 0xFF53: return 0xFF33;
2061                 case 0xFF54: return 0xFF34;
2062                 case 0xFF55: return 0xFF35;
2063                 case 0xFF56: return 0xFF36;
2064                 case 0xFF57: return 0xFF37;
2065                 case 0xFF58: return 0xFF38;
2066                 case 0xFF59: return 0xFF39;
2067                 case 0xFF5A: return 0xFF3A;
2068                 case 0x10428: return 0x10400;
2069                 case 0x10429: return 0x10401;
2070                 case 0x1042A: return 0x10402;
2071                 case 0x1042B: return 0x10403;
2072                 case 0x1042C: return 0x10404;
2073                 case 0x1042D: return 0x10405;
2074                 case 0x1042E: return 0x10406;
2075                 case 0x1042F: return 0x10407;
2076                 case 0x10430: return 0x10408;
2077                 case 0x10431: return 0x10409;
2078                 case 0x10432: return 0x1040A;
2079                 case 0x10433: return 0x1040B;
2080                 case 0x10434: return 0x1040C;
2081                 case 0x10435: return 0x1040D;
2082                 case 0x10436: return 0x1040E;
2083                 case 0x10437: return 0x1040F;
2084                 case 0x10438: return 0x10410;
2085                 case 0x10439: return 0x10411;
2086                 case 0x1043A: return 0x10412;
2087                 case 0x1043B: return 0x10413;
2088                 case 0x1043C: return 0x10414;
2089                 case 0x1043D: return 0x10415;
2090                 case 0x1043E: return 0x10416;
2091                 case 0x1043F: return 0x10417;
2092                 case 0x10440: return 0x10418;
2093                 case 0x10441: return 0x10419;
2094                 case 0x10442: return 0x1041A;
2095                 case 0x10443: return 0x1041B;
2096                 case 0x10444: return 0x1041C;
2097                 case 0x10445: return 0x1041D;
2098                 case 0x10446: return 0x1041E;
2099                 case 0x10447: return 0x1041F;
2100                 case 0x10448: return 0x10420;
2101                 case 0x10449: return 0x10421;
2102                 case 0x1044A: return 0x10422;
2103                 case 0x1044B: return 0x10423;
2104                 case 0x1044C: return 0x10424;
2105                 case 0x1044D: return 0x10425;
2106                 case 0x1044E: return 0x10426;
2107                 case 0x1044F: return 0x10427;
2108                 default: return ch;
2109         }
2110 }
2111
2112 Uchar u8_tolower(Uchar ch)
2113 {
2114         switch(ch)
2115         {
2116                 case 0x0041: return 0x0061;
2117                 case 0x0042: return 0x0062;
2118                 case 0x0043: return 0x0063;
2119                 case 0x0044: return 0x0064;
2120                 case 0x0045: return 0x0065;
2121                 case 0x0046: return 0x0066;
2122                 case 0x0047: return 0x0067;
2123                 case 0x0048: return 0x0068;
2124                 case 0x0049: return 0x0069;
2125                 case 0x004A: return 0x006A;
2126                 case 0x004B: return 0x006B;
2127                 case 0x004C: return 0x006C;
2128                 case 0x004D: return 0x006D;
2129                 case 0x004E: return 0x006E;
2130                 case 0x004F: return 0x006F;
2131                 case 0x0050: return 0x0070;
2132                 case 0x0051: return 0x0071;
2133                 case 0x0052: return 0x0072;
2134                 case 0x0053: return 0x0073;
2135                 case 0x0054: return 0x0074;
2136                 case 0x0055: return 0x0075;
2137                 case 0x0056: return 0x0076;
2138                 case 0x0057: return 0x0077;
2139                 case 0x0058: return 0x0078;
2140                 case 0x0059: return 0x0079;
2141                 case 0x005A: return 0x007A;
2142                 case 0x00C0: return 0x00E0;
2143                 case 0x00C1: return 0x00E1;
2144                 case 0x00C2: return 0x00E2;
2145                 case 0x00C3: return 0x00E3;
2146                 case 0x00C4: return 0x00E4;
2147                 case 0x00C5: return 0x00E5;
2148                 case 0x00C6: return 0x00E6;
2149                 case 0x00C7: return 0x00E7;
2150                 case 0x00C8: return 0x00E8;
2151                 case 0x00C9: return 0x00E9;
2152                 case 0x00CA: return 0x00EA;
2153                 case 0x00CB: return 0x00EB;
2154                 case 0x00CC: return 0x00EC;
2155                 case 0x00CD: return 0x00ED;
2156                 case 0x00CE: return 0x00EE;
2157                 case 0x00CF: return 0x00EF;
2158                 case 0x00D0: return 0x00F0;
2159                 case 0x00D1: return 0x00F1;
2160                 case 0x00D2: return 0x00F2;
2161                 case 0x00D3: return 0x00F3;
2162                 case 0x00D4: return 0x00F4;
2163                 case 0x00D5: return 0x00F5;
2164                 case 0x00D6: return 0x00F6;
2165                 case 0x00D8: return 0x00F8;
2166                 case 0x00D9: return 0x00F9;
2167                 case 0x00DA: return 0x00FA;
2168                 case 0x00DB: return 0x00FB;
2169                 case 0x00DC: return 0x00FC;
2170                 case 0x00DD: return 0x00FD;
2171                 case 0x00DE: return 0x00FE;
2172                 case 0x0100: return 0x0101;
2173                 case 0x0102: return 0x0103;
2174                 case 0x0104: return 0x0105;
2175                 case 0x0106: return 0x0107;
2176                 case 0x0108: return 0x0109;
2177                 case 0x010A: return 0x010B;
2178                 case 0x010C: return 0x010D;
2179                 case 0x010E: return 0x010F;
2180                 case 0x0110: return 0x0111;
2181                 case 0x0112: return 0x0113;
2182                 case 0x0114: return 0x0115;
2183                 case 0x0116: return 0x0117;
2184                 case 0x0118: return 0x0119;
2185                 case 0x011A: return 0x011B;
2186                 case 0x011C: return 0x011D;
2187                 case 0x011E: return 0x011F;
2188                 case 0x0120: return 0x0121;
2189                 case 0x0122: return 0x0123;
2190                 case 0x0124: return 0x0125;
2191                 case 0x0126: return 0x0127;
2192                 case 0x0128: return 0x0129;
2193                 case 0x012A: return 0x012B;
2194                 case 0x012C: return 0x012D;
2195                 case 0x012E: return 0x012F;
2196                 case 0x0130: return 0x0069;
2197                 case 0x0132: return 0x0133;
2198                 case 0x0134: return 0x0135;
2199                 case 0x0136: return 0x0137;
2200                 case 0x0139: return 0x013A;
2201                 case 0x013B: return 0x013C;
2202                 case 0x013D: return 0x013E;
2203                 case 0x013F: return 0x0140;
2204                 case 0x0141: return 0x0142;
2205                 case 0x0143: return 0x0144;
2206                 case 0x0145: return 0x0146;
2207                 case 0x0147: return 0x0148;
2208                 case 0x014A: return 0x014B;
2209                 case 0x014C: return 0x014D;
2210                 case 0x014E: return 0x014F;
2211                 case 0x0150: return 0x0151;
2212                 case 0x0152: return 0x0153;
2213                 case 0x0154: return 0x0155;
2214                 case 0x0156: return 0x0157;
2215                 case 0x0158: return 0x0159;
2216                 case 0x015A: return 0x015B;
2217                 case 0x015C: return 0x015D;
2218                 case 0x015E: return 0x015F;
2219                 case 0x0160: return 0x0161;
2220                 case 0x0162: return 0x0163;
2221                 case 0x0164: return 0x0165;
2222                 case 0x0166: return 0x0167;
2223                 case 0x0168: return 0x0169;
2224                 case 0x016A: return 0x016B;
2225                 case 0x016C: return 0x016D;
2226                 case 0x016E: return 0x016F;
2227                 case 0x0170: return 0x0171;
2228                 case 0x0172: return 0x0173;
2229                 case 0x0174: return 0x0175;
2230                 case 0x0176: return 0x0177;
2231                 case 0x0178: return 0x00FF;
2232                 case 0x0179: return 0x017A;
2233                 case 0x017B: return 0x017C;
2234                 case 0x017D: return 0x017E;
2235                 case 0x0181: return 0x0253;
2236                 case 0x0182: return 0x0183;
2237                 case 0x0184: return 0x0185;
2238                 case 0x0186: return 0x0254;
2239                 case 0x0187: return 0x0188;
2240                 case 0x0189: return 0x0256;
2241                 case 0x018A: return 0x0257;
2242                 case 0x018B: return 0x018C;
2243                 case 0x018E: return 0x01DD;
2244                 case 0x018F: return 0x0259;
2245                 case 0x0190: return 0x025B;
2246                 case 0x0191: return 0x0192;
2247                 case 0x0193: return 0x0260;
2248                 case 0x0194: return 0x0263;
2249                 case 0x0196: return 0x0269;
2250                 case 0x0197: return 0x0268;
2251                 case 0x0198: return 0x0199;
2252                 case 0x019C: return 0x026F;
2253                 case 0x019D: return 0x0272;
2254                 case 0x019F: return 0x0275;
2255                 case 0x01A0: return 0x01A1;
2256                 case 0x01A2: return 0x01A3;
2257                 case 0x01A4: return 0x01A5;
2258                 case 0x01A6: return 0x0280;
2259                 case 0x01A7: return 0x01A8;
2260                 case 0x01A9: return 0x0283;
2261                 case 0x01AC: return 0x01AD;
2262                 case 0x01AE: return 0x0288;
2263                 case 0x01AF: return 0x01B0;
2264                 case 0x01B1: return 0x028A;
2265                 case 0x01B2: return 0x028B;
2266                 case 0x01B3: return 0x01B4;
2267                 case 0x01B5: return 0x01B6;
2268                 case 0x01B7: return 0x0292;
2269                 case 0x01B8: return 0x01B9;
2270                 case 0x01BC: return 0x01BD;
2271                 case 0x01C4: return 0x01C6;
2272                 case 0x01C5: return 0x01C6;
2273                 case 0x01C7: return 0x01C9;
2274                 case 0x01C8: return 0x01C9;
2275                 case 0x01CA: return 0x01CC;
2276                 case 0x01CB: return 0x01CC;
2277                 case 0x01CD: return 0x01CE;
2278                 case 0x01CF: return 0x01D0;
2279                 case 0x01D1: return 0x01D2;
2280                 case 0x01D3: return 0x01D4;
2281                 case 0x01D5: return 0x01D6;
2282                 case 0x01D7: return 0x01D8;
2283                 case 0x01D9: return 0x01DA;
2284                 case 0x01DB: return 0x01DC;
2285                 case 0x01DE: return 0x01DF;
2286                 case 0x01E0: return 0x01E1;
2287                 case 0x01E2: return 0x01E3;
2288                 case 0x01E4: return 0x01E5;
2289                 case 0x01E6: return 0x01E7;
2290                 case 0x01E8: return 0x01E9;
2291                 case 0x01EA: return 0x01EB;
2292                 case 0x01EC: return 0x01ED;
2293                 case 0x01EE: return 0x01EF;
2294                 case 0x01F1: return 0x01F3;
2295                 case 0x01F2: return 0x01F3;
2296                 case 0x01F4: return 0x01F5;
2297                 case 0x01F6: return 0x0195;
2298                 case 0x01F7: return 0x01BF;
2299                 case 0x01F8: return 0x01F9;
2300                 case 0x01FA: return 0x01FB;
2301                 case 0x01FC: return 0x01FD;
2302                 case 0x01FE: return 0x01FF;
2303                 case 0x0200: return 0x0201;
2304                 case 0x0202: return 0x0203;
2305                 case 0x0204: return 0x0205;
2306                 case 0x0206: return 0x0207;
2307                 case 0x0208: return 0x0209;
2308                 case 0x020A: return 0x020B;
2309                 case 0x020C: return 0x020D;
2310                 case 0x020E: return 0x020F;
2311                 case 0x0210: return 0x0211;
2312                 case 0x0212: return 0x0213;
2313                 case 0x0214: return 0x0215;
2314                 case 0x0216: return 0x0217;
2315                 case 0x0218: return 0x0219;
2316                 case 0x021A: return 0x021B;
2317                 case 0x021C: return 0x021D;
2318                 case 0x021E: return 0x021F;
2319                 case 0x0220: return 0x019E;
2320                 case 0x0222: return 0x0223;
2321                 case 0x0224: return 0x0225;
2322                 case 0x0226: return 0x0227;
2323                 case 0x0228: return 0x0229;
2324                 case 0x022A: return 0x022B;
2325                 case 0x022C: return 0x022D;
2326                 case 0x022E: return 0x022F;
2327                 case 0x0230: return 0x0231;
2328                 case 0x0232: return 0x0233;
2329                 case 0x023A: return 0x2C65;
2330                 case 0x023B: return 0x023C;
2331                 case 0x023D: return 0x019A;
2332                 case 0x023E: return 0x2C66;
2333                 case 0x0241: return 0x0242;
2334                 case 0x0243: return 0x0180;
2335                 case 0x0244: return 0x0289;
2336                 case 0x0245: return 0x028C;
2337                 case 0x0246: return 0x0247;
2338                 case 0x0248: return 0x0249;
2339                 case 0x024A: return 0x024B;
2340                 case 0x024C: return 0x024D;
2341                 case 0x024E: return 0x024F;
2342                 case 0x0370: return 0x0371;
2343                 case 0x0372: return 0x0373;
2344                 case 0x0376: return 0x0377;
2345                 case 0x0386: return 0x03AC;
2346                 case 0x0388: return 0x03AD;
2347                 case 0x0389: return 0x03AE;
2348                 case 0x038A: return 0x03AF;
2349                 case 0x038C: return 0x03CC;
2350                 case 0x038E: return 0x03CD;
2351                 case 0x038F: return 0x03CE;
2352                 case 0x0391: return 0x03B1;
2353                 case 0x0392: return 0x03B2;
2354                 case 0x0393: return 0x03B3;
2355                 case 0x0394: return 0x03B4;
2356                 case 0x0395: return 0x03B5;
2357                 case 0x0396: return 0x03B6;
2358                 case 0x0397: return 0x03B7;
2359                 case 0x0398: return 0x03B8;
2360                 case 0x0399: return 0x03B9;
2361                 case 0x039A: return 0x03BA;
2362                 case 0x039B: return 0x03BB;
2363                 case 0x039C: return 0x03BC;
2364                 case 0x039D: return 0x03BD;
2365                 case 0x039E: return 0x03BE;
2366                 case 0x039F: return 0x03BF;
2367                 case 0x03A0: return 0x03C0;
2368                 case 0x03A1: return 0x03C1;
2369                 case 0x03A3: return 0x03C3;
2370                 case 0x03A4: return 0x03C4;
2371                 case 0x03A5: return 0x03C5;
2372                 case 0x03A6: return 0x03C6;
2373                 case 0x03A7: return 0x03C7;
2374                 case 0x03A8: return 0x03C8;
2375                 case 0x03A9: return 0x03C9;
2376                 case 0x03AA: return 0x03CA;
2377                 case 0x03AB: return 0x03CB;
2378                 case 0x03CF: return 0x03D7;
2379                 case 0x03D8: return 0x03D9;
2380                 case 0x03DA: return 0x03DB;
2381                 case 0x03DC: return 0x03DD;
2382                 case 0x03DE: return 0x03DF;
2383                 case 0x03E0: return 0x03E1;
2384                 case 0x03E2: return 0x03E3;
2385                 case 0x03E4: return 0x03E5;
2386                 case 0x03E6: return 0x03E7;
2387                 case 0x03E8: return 0x03E9;
2388                 case 0x03EA: return 0x03EB;
2389                 case 0x03EC: return 0x03ED;
2390                 case 0x03EE: return 0x03EF;
2391                 case 0x03F4: return 0x03B8;
2392                 case 0x03F7: return 0x03F8;
2393                 case 0x03F9: return 0x03F2;
2394                 case 0x03FA: return 0x03FB;
2395                 case 0x03FD: return 0x037B;
2396                 case 0x03FE: return 0x037C;
2397                 case 0x03FF: return 0x037D;
2398                 case 0x0400: return 0x0450;
2399                 case 0x0401: return 0x0451;
2400                 case 0x0402: return 0x0452;
2401                 case 0x0403: return 0x0453;
2402                 case 0x0404: return 0x0454;
2403                 case 0x0405: return 0x0455;
2404                 case 0x0406: return 0x0456;
2405                 case 0x0407: return 0x0457;
2406                 case 0x0408: return 0x0458;
2407                 case 0x0409: return 0x0459;
2408                 case 0x040A: return 0x045A;
2409                 case 0x040B: return 0x045B;
2410                 case 0x040C: return 0x045C;
2411                 case 0x040D: return 0x045D;
2412                 case 0x040E: return 0x045E;
2413                 case 0x040F: return 0x045F;
2414                 case 0x0410: return 0x0430;
2415                 case 0x0411: return 0x0431;
2416                 case 0x0412: return 0x0432;
2417                 case 0x0413: return 0x0433;
2418                 case 0x0414: return 0x0434;
2419                 case 0x0415: return 0x0435;
2420                 case 0x0416: return 0x0436;
2421                 case 0x0417: return 0x0437;
2422                 case 0x0418: return 0x0438;
2423                 case 0x0419: return 0x0439;
2424                 case 0x041A: return 0x043A;
2425                 case 0x041B: return 0x043B;
2426                 case 0x041C: return 0x043C;
2427                 case 0x041D: return 0x043D;
2428                 case 0x041E: return 0x043E;
2429                 case 0x041F: return 0x043F;
2430                 case 0x0420: return 0x0440;
2431                 case 0x0421: return 0x0441;
2432                 case 0x0422: return 0x0442;
2433                 case 0x0423: return 0x0443;
2434                 case 0x0424: return 0x0444;
2435                 case 0x0425: return 0x0445;
2436                 case 0x0426: return 0x0446;
2437                 case 0x0427: return 0x0447;
2438                 case 0x0428: return 0x0448;
2439                 case 0x0429: return 0x0449;
2440                 case 0x042A: return 0x044A;
2441                 case 0x042B: return 0x044B;
2442                 case 0x042C: return 0x044C;
2443                 case 0x042D: return 0x044D;
2444                 case 0x042E: return 0x044E;
2445                 case 0x042F: return 0x044F;
2446                 case 0x0460: return 0x0461;
2447                 case 0x0462: return 0x0463;
2448                 case 0x0464: return 0x0465;
2449                 case 0x0466: return 0x0467;
2450                 case 0x0468: return 0x0469;
2451                 case 0x046A: return 0x046B;
2452                 case 0x046C: return 0x046D;
2453                 case 0x046E: return 0x046F;
2454                 case 0x0470: return 0x0471;
2455                 case 0x0472: return 0x0473;
2456                 case 0x0474: return 0x0475;
2457                 case 0x0476: return 0x0477;
2458                 case 0x0478: return 0x0479;
2459                 case 0x047A: return 0x047B;
2460                 case 0x047C: return 0x047D;
2461                 case 0x047E: return 0x047F;
2462                 case 0x0480: return 0x0481;
2463                 case 0x048A: return 0x048B;
2464                 case 0x048C: return 0x048D;
2465                 case 0x048E: return 0x048F;
2466                 case 0x0490: return 0x0491;
2467                 case 0x0492: return 0x0493;
2468                 case 0x0494: return 0x0495;
2469                 case 0x0496: return 0x0497;
2470                 case 0x0498: return 0x0499;
2471                 case 0x049A: return 0x049B;
2472                 case 0x049C: return 0x049D;
2473                 case 0x049E: return 0x049F;
2474                 case 0x04A0: return 0x04A1;
2475                 case 0x04A2: return 0x04A3;
2476                 case 0x04A4: return 0x04A5;
2477                 case 0x04A6: return 0x04A7;
2478                 case 0x04A8: return 0x04A9;
2479                 case 0x04AA: return 0x04AB;
2480                 case 0x04AC: return 0x04AD;
2481                 case 0x04AE: return 0x04AF;
2482                 case 0x04B0: return 0x04B1;
2483                 case 0x04B2: return 0x04B3;
2484                 case 0x04B4: return 0x04B5;
2485                 case 0x04B6: return 0x04B7;
2486                 case 0x04B8: return 0x04B9;
2487                 case 0x04BA: return 0x04BB;
2488                 case 0x04BC: return 0x04BD;
2489                 case 0x04BE: return 0x04BF;
2490                 case 0x04C0: return 0x04CF;
2491                 case 0x04C1: return 0x04C2;
2492                 case 0x04C3: return 0x04C4;
2493                 case 0x04C5: return 0x04C6;
2494                 case 0x04C7: return 0x04C8;
2495                 case 0x04C9: return 0x04CA;
2496                 case 0x04CB: return 0x04CC;
2497                 case 0x04CD: return 0x04CE;
2498                 case 0x04D0: return 0x04D1;
2499                 case 0x04D2: return 0x04D3;
2500                 case 0x04D4: return 0x04D5;
2501                 case 0x04D6: return 0x04D7;
2502                 case 0x04D8: return 0x04D9;
2503                 case 0x04DA: return 0x04DB;
2504                 case 0x04DC: return 0x04DD;
2505                 case 0x04DE: return 0x04DF;
2506                 case 0x04E0: return 0x04E1;
2507                 case 0x04E2: return 0x04E3;
2508                 case 0x04E4: return 0x04E5;
2509                 case 0x04E6: return 0x04E7;
2510                 case 0x04E8: return 0x04E9;
2511                 case 0x04EA: return 0x04EB;
2512                 case 0x04EC: return 0x04ED;
2513                 case 0x04EE: return 0x04EF;
2514                 case 0x04F0: return 0x04F1;
2515                 case 0x04F2: return 0x04F3;
2516                 case 0x04F4: return 0x04F5;
2517                 case 0x04F6: return 0x04F7;
2518                 case 0x04F8: return 0x04F9;
2519                 case 0x04FA: return 0x04FB;
2520                 case 0x04FC: return 0x04FD;
2521                 case 0x04FE: return 0x04FF;
2522                 case 0x0500: return 0x0501;
2523                 case 0x0502: return 0x0503;
2524                 case 0x0504: return 0x0505;
2525                 case 0x0506: return 0x0507;
2526                 case 0x0508: return 0x0509;
2527                 case 0x050A: return 0x050B;
2528                 case 0x050C: return 0x050D;
2529                 case 0x050E: return 0x050F;
2530                 case 0x0510: return 0x0511;
2531                 case 0x0512: return 0x0513;
2532                 case 0x0514: return 0x0515;
2533                 case 0x0516: return 0x0517;
2534                 case 0x0518: return 0x0519;
2535                 case 0x051A: return 0x051B;
2536                 case 0x051C: return 0x051D;
2537                 case 0x051E: return 0x051F;
2538                 case 0x0520: return 0x0521;
2539                 case 0x0522: return 0x0523;
2540                 case 0x0524: return 0x0525;
2541                 case 0x0526: return 0x0527;
2542                 case 0x0531: return 0x0561;
2543                 case 0x0532: return 0x0562;
2544                 case 0x0533: return 0x0563;
2545                 case 0x0534: return 0x0564;
2546                 case 0x0535: return 0x0565;
2547                 case 0x0536: return 0x0566;
2548                 case 0x0537: return 0x0567;
2549                 case 0x0538: return 0x0568;
2550                 case 0x0539: return 0x0569;
2551                 case 0x053A: return 0x056A;
2552                 case 0x053B: return 0x056B;
2553                 case 0x053C: return 0x056C;
2554                 case 0x053D: return 0x056D;
2555                 case 0x053E: return 0x056E;
2556                 case 0x053F: return 0x056F;
2557                 case 0x0540: return 0x0570;
2558                 case 0x0541: return 0x0571;
2559                 case 0x0542: return 0x0572;
2560                 case 0x0543: return 0x0573;
2561                 case 0x0544: return 0x0574;
2562                 case 0x0545: return 0x0575;
2563                 case 0x0546: return 0x0576;
2564                 case 0x0547: return 0x0577;
2565                 case 0x0548: return 0x0578;
2566                 case 0x0549: return 0x0579;
2567                 case 0x054A: return 0x057A;
2568                 case 0x054B: return 0x057B;
2569                 case 0x054C: return 0x057C;
2570                 case 0x054D: return 0x057D;
2571                 case 0x054E: return 0x057E;
2572                 case 0x054F: return 0x057F;
2573                 case 0x0550: return 0x0580;
2574                 case 0x0551: return 0x0581;
2575                 case 0x0552: return 0x0582;
2576                 case 0x0553: return 0x0583;
2577                 case 0x0554: return 0x0584;
2578                 case 0x0555: return 0x0585;
2579                 case 0x0556: return 0x0586;
2580                 case 0x10A0: return 0x2D00;
2581                 case 0x10A1: return 0x2D01;
2582                 case 0x10A2: return 0x2D02;
2583                 case 0x10A3: return 0x2D03;
2584                 case 0x10A4: return 0x2D04;
2585                 case 0x10A5: return 0x2D05;
2586                 case 0x10A6: return 0x2D06;
2587                 case 0x10A7: return 0x2D07;
2588                 case 0x10A8: return 0x2D08;
2589                 case 0x10A9: return 0x2D09;
2590                 case 0x10AA: return 0x2D0A;
2591                 case 0x10AB: return 0x2D0B;
2592                 case 0x10AC: return 0x2D0C;
2593                 case 0x10AD: return 0x2D0D;
2594                 case 0x10AE: return 0x2D0E;
2595                 case 0x10AF: return 0x2D0F;
2596                 case 0x10B0: return 0x2D10;
2597                 case 0x10B1: return 0x2D11;
2598                 case 0x10B2: return 0x2D12;
2599                 case 0x10B3: return 0x2D13;
2600                 case 0x10B4: return 0x2D14;
2601                 case 0x10B5: return 0x2D15;
2602                 case 0x10B6: return 0x2D16;
2603                 case 0x10B7: return 0x2D17;
2604                 case 0x10B8: return 0x2D18;
2605                 case 0x10B9: return 0x2D19;
2606                 case 0x10BA: return 0x2D1A;
2607                 case 0x10BB: return 0x2D1B;
2608                 case 0x10BC: return 0x2D1C;
2609                 case 0x10BD: return 0x2D1D;
2610                 case 0x10BE: return 0x2D1E;
2611                 case 0x10BF: return 0x2D1F;
2612                 case 0x10C0: return 0x2D20;
2613                 case 0x10C1: return 0x2D21;
2614                 case 0x10C2: return 0x2D22;
2615                 case 0x10C3: return 0x2D23;
2616                 case 0x10C4: return 0x2D24;
2617                 case 0x10C5: return 0x2D25;
2618                 case 0x1E00: return 0x1E01;
2619                 case 0x1E02: return 0x1E03;
2620                 case 0x1E04: return 0x1E05;
2621                 case 0x1E06: return 0x1E07;
2622                 case 0x1E08: return 0x1E09;
2623                 case 0x1E0A: return 0x1E0B;
2624                 case 0x1E0C: return 0x1E0D;
2625                 case 0x1E0E: return 0x1E0F;
2626                 case 0x1E10: return 0x1E11;
2627                 case 0x1E12: return 0x1E13;
2628                 case 0x1E14: return 0x1E15;
2629                 case 0x1E16: return 0x1E17;
2630                 case 0x1E18: return 0x1E19;
2631                 case 0x1E1A: return 0x1E1B;
2632                 case 0x1E1C: return 0x1E1D;
2633                 case 0x1E1E: return 0x1E1F;
2634                 case 0x1E20: return 0x1E21;
2635                 case 0x1E22: return 0x1E23;
2636                 case 0x1E24: return 0x1E25;
2637                 case 0x1E26: return 0x1E27;
2638                 case 0x1E28: return 0x1E29;
2639                 case 0x1E2A: return 0x1E2B;
2640                 case 0x1E2C: return 0x1E2D;
2641                 case 0x1E2E: return 0x1E2F;
2642                 case 0x1E30: return 0x1E31;
2643                 case 0x1E32: return 0x1E33;
2644                 case 0x1E34: return 0x1E35;
2645                 case 0x1E36: return 0x1E37;
2646                 case 0x1E38: return 0x1E39;
2647                 case 0x1E3A: return 0x1E3B;
2648                 case 0x1E3C: return 0x1E3D;
2649                 case 0x1E3E: return 0x1E3F;
2650                 case 0x1E40: return 0x1E41;
2651                 case 0x1E42: return 0x1E43;
2652                 case 0x1E44: return 0x1E45;
2653                 case 0x1E46: return 0x1E47;
2654                 case 0x1E48: return 0x1E49;
2655                 case 0x1E4A: return 0x1E4B;
2656                 case 0x1E4C: return 0x1E4D;
2657                 case 0x1E4E: return 0x1E4F;
2658                 case 0x1E50: return 0x1E51;
2659                 case 0x1E52: return 0x1E53;
2660                 case 0x1E54: return 0x1E55;
2661                 case 0x1E56: return 0x1E57;
2662                 case 0x1E58: return 0x1E59;
2663                 case 0x1E5A: return 0x1E5B;
2664                 case 0x1E5C: return 0x1E5D;
2665                 case 0x1E5E: return 0x1E5F;
2666                 case 0x1E60: return 0x1E61;
2667                 case 0x1E62: return 0x1E63;
2668                 case 0x1E64: return 0x1E65;
2669                 case 0x1E66: return 0x1E67;
2670                 case 0x1E68: return 0x1E69;
2671                 case 0x1E6A: return 0x1E6B;
2672                 case 0x1E6C: return 0x1E6D;
2673                 case 0x1E6E: return 0x1E6F;
2674                 case 0x1E70: return 0x1E71;
2675                 case 0x1E72: return 0x1E73;
2676                 case 0x1E74: return 0x1E75;
2677                 case 0x1E76: return 0x1E77;
2678                 case 0x1E78: return 0x1E79;
2679                 case 0x1E7A: return 0x1E7B;
2680                 case 0x1E7C: return 0x1E7D;
2681                 case 0x1E7E: return 0x1E7F;
2682                 case 0x1E80: return 0x1E81;
2683                 case 0x1E82: return 0x1E83;
2684                 case 0x1E84: return 0x1E85;
2685                 case 0x1E86: return 0x1E87;
2686                 case 0x1E88: return 0x1E89;
2687                 case 0x1E8A: return 0x1E8B;
2688                 case 0x1E8C: return 0x1E8D;
2689                 case 0x1E8E: return 0x1E8F;
2690                 case 0x1E90: return 0x1E91;
2691                 case 0x1E92: return 0x1E93;
2692                 case 0x1E94: return 0x1E95;
2693                 case 0x1E9E: return 0x00DF;
2694                 case 0x1EA0: return 0x1EA1;
2695                 case 0x1EA2: return 0x1EA3;
2696                 case 0x1EA4: return 0x1EA5;
2697                 case 0x1EA6: return 0x1EA7;
2698                 case 0x1EA8: return 0x1EA9;
2699                 case 0x1EAA: return 0x1EAB;
2700                 case 0x1EAC: return 0x1EAD;
2701                 case 0x1EAE: return 0x1EAF;
2702                 case 0x1EB0: return 0x1EB1;
2703                 case 0x1EB2: return 0x1EB3;
2704                 case 0x1EB4: return 0x1EB5;
2705                 case 0x1EB6: return 0x1EB7;
2706                 case 0x1EB8: return 0x1EB9;
2707                 case 0x1EBA: return 0x1EBB;
2708                 case 0x1EBC: return 0x1EBD;
2709                 case 0x1EBE: return 0x1EBF;
2710                 case 0x1EC0: return 0x1EC1;
2711                 case 0x1EC2: return 0x1EC3;
2712                 case 0x1EC4: return 0x1EC5;
2713                 case 0x1EC6: return 0x1EC7;
2714                 case 0x1EC8: return 0x1EC9;
2715                 case 0x1ECA: return 0x1ECB;
2716                 case 0x1ECC: return 0x1ECD;
2717                 case 0x1ECE: return 0x1ECF;
2718                 case 0x1ED0: return 0x1ED1;
2719                 case 0x1ED2: return 0x1ED3;
2720                 case 0x1ED4: return 0x1ED5;
2721                 case 0x1ED6: return 0x1ED7;
2722                 case 0x1ED8: return 0x1ED9;
2723                 case 0x1EDA: return 0x1EDB;
2724                 case 0x1EDC: return 0x1EDD;
2725                 case 0x1EDE: return 0x1EDF;
2726                 case 0x1EE0: return 0x1EE1;
2727                 case 0x1EE2: return 0x1EE3;
2728                 case 0x1EE4: return 0x1EE5;
2729                 case 0x1EE6: return 0x1EE7;
2730                 case 0x1EE8: return 0x1EE9;
2731                 case 0x1EEA: return 0x1EEB;
2732                 case 0x1EEC: return 0x1EED;
2733                 case 0x1EEE: return 0x1EEF;
2734                 case 0x1EF0: return 0x1EF1;
2735                 case 0x1EF2: return 0x1EF3;
2736                 case 0x1EF4: return 0x1EF5;
2737                 case 0x1EF6: return 0x1EF7;
2738                 case 0x1EF8: return 0x1EF9;
2739                 case 0x1EFA: return 0x1EFB;
2740                 case 0x1EFC: return 0x1EFD;
2741                 case 0x1EFE: return 0x1EFF;
2742                 case 0x1F08: return 0x1F00;
2743                 case 0x1F09: return 0x1F01;
2744                 case 0x1F0A: return 0x1F02;
2745                 case 0x1F0B: return 0x1F03;
2746                 case 0x1F0C: return 0x1F04;
2747                 case 0x1F0D: return 0x1F05;
2748                 case 0x1F0E: return 0x1F06;
2749                 case 0x1F0F: return 0x1F07;
2750                 case 0x1F18: return 0x1F10;
2751                 case 0x1F19: return 0x1F11;
2752                 case 0x1F1A: return 0x1F12;
2753                 case 0x1F1B: return 0x1F13;
2754                 case 0x1F1C: return 0x1F14;
2755                 case 0x1F1D: return 0x1F15;
2756                 case 0x1F28: return 0x1F20;
2757                 case 0x1F29: return 0x1F21;
2758                 case 0x1F2A: return 0x1F22;
2759                 case 0x1F2B: return 0x1F23;
2760                 case 0x1F2C: return 0x1F24;
2761                 case 0x1F2D: return 0x1F25;
2762                 case 0x1F2E: return 0x1F26;
2763                 case 0x1F2F: return 0x1F27;
2764                 case 0x1F38: return 0x1F30;
2765                 case 0x1F39: return 0x1F31;
2766                 case 0x1F3A: return 0x1F32;
2767                 case 0x1F3B: return 0x1F33;
2768                 case 0x1F3C: return 0x1F34;
2769                 case 0x1F3D: return 0x1F35;
2770                 case 0x1F3E: return 0x1F36;
2771                 case 0x1F3F: return 0x1F37;
2772                 case 0x1F48: return 0x1F40;
2773                 case 0x1F49: return 0x1F41;
2774                 case 0x1F4A: return 0x1F42;
2775                 case 0x1F4B: return 0x1F43;
2776                 case 0x1F4C: return 0x1F44;
2777                 case 0x1F4D: return 0x1F45;
2778                 case 0x1F59: return 0x1F51;
2779                 case 0x1F5B: return 0x1F53;
2780                 case 0x1F5D: return 0x1F55;
2781                 case 0x1F5F: return 0x1F57;
2782                 case 0x1F68: return 0x1F60;
2783                 case 0x1F69: return 0x1F61;
2784                 case 0x1F6A: return 0x1F62;
2785                 case 0x1F6B: return 0x1F63;
2786                 case 0x1F6C: return 0x1F64;
2787                 case 0x1F6D: return 0x1F65;
2788                 case 0x1F6E: return 0x1F66;
2789                 case 0x1F6F: return 0x1F67;
2790                 case 0x1F88: return 0x1F80;
2791                 case 0x1F89: return 0x1F81;
2792                 case 0x1F8A: return 0x1F82;
2793                 case 0x1F8B: return 0x1F83;
2794                 case 0x1F8C: return 0x1F84;
2795                 case 0x1F8D: return 0x1F85;
2796                 case 0x1F8E: return 0x1F86;
2797                 case 0x1F8F: return 0x1F87;
2798                 case 0x1F98: return 0x1F90;
2799                 case 0x1F99: return 0x1F91;
2800                 case 0x1F9A: return 0x1F92;
2801                 case 0x1F9B: return 0x1F93;
2802                 case 0x1F9C: return 0x1F94;
2803                 case 0x1F9D: return 0x1F95;
2804                 case 0x1F9E: return 0x1F96;
2805                 case 0x1F9F: return 0x1F97;
2806                 case 0x1FA8: return 0x1FA0;
2807                 case 0x1FA9: return 0x1FA1;
2808                 case 0x1FAA: return 0x1FA2;
2809                 case 0x1FAB: return 0x1FA3;
2810                 case 0x1FAC: return 0x1FA4;
2811                 case 0x1FAD: return 0x1FA5;
2812                 case 0x1FAE: return 0x1FA6;
2813                 case 0x1FAF: return 0x1FA7;
2814                 case 0x1FB8: return 0x1FB0;
2815                 case 0x1FB9: return 0x1FB1;
2816                 case 0x1FBA: return 0x1F70;
2817                 case 0x1FBB: return 0x1F71;
2818                 case 0x1FBC: return 0x1FB3;
2819                 case 0x1FC8: return 0x1F72;
2820                 case 0x1FC9: return 0x1F73;
2821                 case 0x1FCA: return 0x1F74;
2822                 case 0x1FCB: return 0x1F75;
2823                 case 0x1FCC: return 0x1FC3;
2824                 case 0x1FD8: return 0x1FD0;
2825                 case 0x1FD9: return 0x1FD1;
2826                 case 0x1FDA: return 0x1F76;
2827                 case 0x1FDB: return 0x1F77;
2828                 case 0x1FE8: return 0x1FE0;
2829                 case 0x1FE9: return 0x1FE1;
2830                 case 0x1FEA: return 0x1F7A;
2831                 case 0x1FEB: return 0x1F7B;
2832                 case 0x1FEC: return 0x1FE5;
2833                 case 0x1FF8: return 0x1F78;
2834                 case 0x1FF9: return 0x1F79;
2835                 case 0x1FFA: return 0x1F7C;
2836                 case 0x1FFB: return 0x1F7D;
2837                 case 0x1FFC: return 0x1FF3;
2838                 case 0x2126: return 0x03C9;
2839                 case 0x212A: return 0x006B;
2840                 case 0x212B: return 0x00E5;
2841                 case 0x2132: return 0x214E;
2842                 case 0x2160: return 0x2170;
2843                 case 0x2161: return 0x2171;
2844                 case 0x2162: return 0x2172;
2845                 case 0x2163: return 0x2173;
2846                 case 0x2164: return 0x2174;
2847                 case 0x2165: return 0x2175;
2848                 case 0x2166: return 0x2176;
2849                 case 0x2167: return 0x2177;
2850                 case 0x2168: return 0x2178;
2851                 case 0x2169: return 0x2179;
2852                 case 0x216A: return 0x217A;
2853                 case 0x216B: return 0x217B;
2854                 case 0x216C: return 0x217C;
2855                 case 0x216D: return 0x217D;
2856                 case 0x216E: return 0x217E;
2857                 case 0x216F: return 0x217F;
2858                 case 0x2183: return 0x2184;
2859                 case 0x24B6: return 0x24D0;
2860                 case 0x24B7: return 0x24D1;
2861                 case 0x24B8: return 0x24D2;
2862                 case 0x24B9: return 0x24D3;
2863                 case 0x24BA: return 0x24D4;
2864                 case 0x24BB: return 0x24D5;
2865                 case 0x24BC: return 0x24D6;
2866                 case 0x24BD: return 0x24D7;
2867                 case 0x24BE: return 0x24D8;
2868                 case 0x24BF: return 0x24D9;
2869                 case 0x24C0: return 0x24DA;
2870                 case 0x24C1: return 0x24DB;
2871                 case 0x24C2: return 0x24DC;
2872                 case 0x24C3: return 0x24DD;
2873                 case 0x24C4: return 0x24DE;
2874                 case 0x24C5: return 0x24DF;
2875                 case 0x24C6: return 0x24E0;
2876                 case 0x24C7: return 0x24E1;
2877                 case 0x24C8: return 0x24E2;
2878                 case 0x24C9: return 0x24E3;
2879                 case 0x24CA: return 0x24E4;
2880                 case 0x24CB: return 0x24E5;
2881                 case 0x24CC: return 0x24E6;
2882                 case 0x24CD: return 0x24E7;
2883                 case 0x24CE: return 0x24E8;
2884                 case 0x24CF: return 0x24E9;
2885                 case 0x2C00: return 0x2C30;
2886                 case 0x2C01: return 0x2C31;
2887                 case 0x2C02: return 0x2C32;
2888                 case 0x2C03: return 0x2C33;
2889                 case 0x2C04: return 0x2C34;
2890                 case 0x2C05: return 0x2C35;
2891                 case 0x2C06: return 0x2C36;
2892                 case 0x2C07: return 0x2C37;
2893                 case 0x2C08: return 0x2C38;
2894                 case 0x2C09: return 0x2C39;
2895                 case 0x2C0A: return 0x2C3A;
2896                 case 0x2C0B: return 0x2C3B;
2897                 case 0x2C0C: return 0x2C3C;
2898                 case 0x2C0D: return 0x2C3D;
2899                 case 0x2C0E: return 0x2C3E;
2900                 case 0x2C0F: return 0x2C3F;
2901                 case 0x2C10: return 0x2C40;
2902                 case 0x2C11: return 0x2C41;
2903                 case 0x2C12: return 0x2C42;
2904                 case 0x2C13: return 0x2C43;
2905                 case 0x2C14: return 0x2C44;
2906                 case 0x2C15: return 0x2C45;
2907                 case 0x2C16: return 0x2C46;
2908                 case 0x2C17: return 0x2C47;
2909                 case 0x2C18: return 0x2C48;
2910                 case 0x2C19: return 0x2C49;
2911                 case 0x2C1A: return 0x2C4A;
2912                 case 0x2C1B: return 0x2C4B;
2913                 case 0x2C1C: return 0x2C4C;
2914                 case 0x2C1D: return 0x2C4D;
2915                 case 0x2C1E: return 0x2C4E;
2916                 case 0x2C1F: return 0x2C4F;
2917                 case 0x2C20: return 0x2C50;
2918                 case 0x2C21: return 0x2C51;
2919                 case 0x2C22: return 0x2C52;
2920                 case 0x2C23: return 0x2C53;
2921                 case 0x2C24: return 0x2C54;
2922                 case 0x2C25: return 0x2C55;
2923                 case 0x2C26: return 0x2C56;
2924                 case 0x2C27: return 0x2C57;
2925                 case 0x2C28: return 0x2C58;
2926                 case 0x2C29: return 0x2C59;
2927                 case 0x2C2A: return 0x2C5A;
2928                 case 0x2C2B: return 0x2C5B;
2929                 case 0x2C2C: return 0x2C5C;
2930                 case 0x2C2D: return 0x2C5D;
2931                 case 0x2C2E: return 0x2C5E;
2932                 case 0x2C60: return 0x2C61;
2933                 case 0x2C62: return 0x026B;
2934                 case 0x2C63: return 0x1D7D;
2935                 case 0x2C64: return 0x027D;
2936                 case 0x2C67: return 0x2C68;
2937                 case 0x2C69: return 0x2C6A;
2938                 case 0x2C6B: return 0x2C6C;
2939                 case 0x2C6D: return 0x0251;
2940                 case 0x2C6E: return 0x0271;
2941                 case 0x2C6F: return 0x0250;
2942                 case 0x2C70: return 0x0252;
2943                 case 0x2C72: return 0x2C73;
2944                 case 0x2C75: return 0x2C76;
2945                 case 0x2C7E: return 0x023F;
2946                 case 0x2C7F: return 0x0240;
2947                 case 0x2C80: return 0x2C81;
2948                 case 0x2C82: return 0x2C83;
2949                 case 0x2C84: return 0x2C85;
2950                 case 0x2C86: return 0x2C87;
2951                 case 0x2C88: return 0x2C89;
2952                 case 0x2C8A: return 0x2C8B;
2953                 case 0x2C8C: return 0x2C8D;
2954                 case 0x2C8E: return 0x2C8F;
2955                 case 0x2C90: return 0x2C91;
2956                 case 0x2C92: return 0x2C93;
2957                 case 0x2C94: return 0x2C95;
2958                 case 0x2C96: return 0x2C97;
2959                 case 0x2C98: return 0x2C99;
2960                 case 0x2C9A: return 0x2C9B;
2961                 case 0x2C9C: return 0x2C9D;
2962                 case 0x2C9E: return 0x2C9F;
2963                 case 0x2CA0: return 0x2CA1;
2964                 case 0x2CA2: return 0x2CA3;
2965                 case 0x2CA4: return 0x2CA5;
2966                 case 0x2CA6: return 0x2CA7;
2967                 case 0x2CA8: return 0x2CA9;
2968                 case 0x2CAA: return 0x2CAB;
2969                 case 0x2CAC: return 0x2CAD;
2970                 case 0x2CAE: return 0x2CAF;
2971                 case 0x2CB0: return 0x2CB1;
2972                 case 0x2CB2: return 0x2CB3;
2973                 case 0x2CB4: return 0x2CB5;
2974                 case 0x2CB6: return 0x2CB7;
2975                 case 0x2CB8: return 0x2CB9;
2976                 case 0x2CBA: return 0x2CBB;
2977                 case 0x2CBC: return 0x2CBD;
2978                 case 0x2CBE: return 0x2CBF;
2979                 case 0x2CC0: return 0x2CC1;
2980                 case 0x2CC2: return 0x2CC3;
2981                 case 0x2CC4: return 0x2CC5;
2982                 case 0x2CC6: return 0x2CC7;
2983                 case 0x2CC8: return 0x2CC9;
2984                 case 0x2CCA: return 0x2CCB;
2985                 case 0x2CCC: return 0x2CCD;
2986                 case 0x2CCE: return 0x2CCF;
2987                 case 0x2CD0: return 0x2CD1;
2988                 case 0x2CD2: return 0x2CD3;
2989                 case 0x2CD4: return 0x2CD5;
2990                 case 0x2CD6: return 0x2CD7;
2991                 case 0x2CD8: return 0x2CD9;
2992                 case 0x2CDA: return 0x2CDB;
2993                 case 0x2CDC: return 0x2CDD;
2994                 case 0x2CDE: return 0x2CDF;
2995                 case 0x2CE0: return 0x2CE1;
2996                 case 0x2CE2: return 0x2CE3;
2997                 case 0x2CEB: return 0x2CEC;
2998                 case 0x2CED: return 0x2CEE;
2999                 case 0xA640: return 0xA641;
3000                 case 0xA642: return 0xA643;
3001                 case 0xA644: return 0xA645;
3002                 case 0xA646: return 0xA647;
3003                 case 0xA648: return 0xA649;
3004                 case 0xA64A: return 0xA64B;
3005                 case 0xA64C: return 0xA64D;
3006                 case 0xA64E: return 0xA64F;
3007                 case 0xA650: return 0xA651;
3008                 case 0xA652: return 0xA653;
3009                 case 0xA654: return 0xA655;
3010                 case 0xA656: return 0xA657;
3011                 case 0xA658: return 0xA659;
3012                 case 0xA65A: return 0xA65B;
3013                 case 0xA65C: return 0xA65D;
3014                 case 0xA65E: return 0xA65F;
3015                 case 0xA660: return 0xA661;
3016                 case 0xA662: return 0xA663;
3017                 case 0xA664: return 0xA665;
3018                 case 0xA666: return 0xA667;
3019                 case 0xA668: return 0xA669;
3020                 case 0xA66A: return 0xA66B;
3021                 case 0xA66C: return 0xA66D;
3022                 case 0xA680: return 0xA681;
3023                 case 0xA682: return 0xA683;
3024                 case 0xA684: return 0xA685;
3025                 case 0xA686: return 0xA687;
3026                 case 0xA688: return 0xA689;
3027                 case 0xA68A: return 0xA68B;
3028                 case 0xA68C: return 0xA68D;
3029                 case 0xA68E: return 0xA68F;
3030                 case 0xA690: return 0xA691;
3031                 case 0xA692: return 0xA693;
3032                 case 0xA694: return 0xA695;
3033                 case 0xA696: return 0xA697;
3034                 case 0xA722: return 0xA723;
3035                 case 0xA724: return 0xA725;
3036                 case 0xA726: return 0xA727;
3037                 case 0xA728: return 0xA729;
3038                 case 0xA72A: return 0xA72B;
3039                 case 0xA72C: return 0xA72D;
3040                 case 0xA72E: return 0xA72F;
3041                 case 0xA732: return 0xA733;
3042                 case 0xA734: return 0xA735;
3043                 case 0xA736: return 0xA737;
3044                 case 0xA738: return 0xA739;
3045                 case 0xA73A: return 0xA73B;
3046                 case 0xA73C: return 0xA73D;
3047                 case 0xA73E: return 0xA73F;
3048                 case 0xA740: return 0xA741;
3049                 case 0xA742: return 0xA743;
3050                 case 0xA744: return 0xA745;
3051                 case 0xA746: return 0xA747;
3052                 case 0xA748: return 0xA749;
3053                 case 0xA74A: return 0xA74B;
3054                 case 0xA74C: return 0xA74D;
3055                 case 0xA74E: return 0xA74F;
3056                 case 0xA750: return 0xA751;
3057                 case 0xA752: return 0xA753;
3058                 case 0xA754: return 0xA755;
3059                 case 0xA756: return 0xA757;
3060                 case 0xA758: return 0xA759;
3061                 case 0xA75A: return 0xA75B;
3062                 case 0xA75C: return 0xA75D;
3063                 case 0xA75E: return 0xA75F;
3064                 case 0xA760: return 0xA761;
3065                 case 0xA762: return 0xA763;
3066                 case 0xA764: return 0xA765;
3067                 case 0xA766: return 0xA767;
3068                 case 0xA768: return 0xA769;
3069                 case 0xA76A: return 0xA76B;
3070                 case 0xA76C: return 0xA76D;
3071                 case 0xA76E: return 0xA76F;
3072                 case 0xA779: return 0xA77A;
3073                 case 0xA77B: return 0xA77C;
3074                 case 0xA77D: return 0x1D79;
3075                 case 0xA77E: return 0xA77F;
3076                 case 0xA780: return 0xA781;
3077                 case 0xA782: return 0xA783;
3078                 case 0xA784: return 0xA785;
3079                 case 0xA786: return 0xA787;
3080                 case 0xA78B: return 0xA78C;
3081                 case 0xA78D: return 0x0265;
3082                 case 0xA790: return 0xA791;
3083                 case 0xA7A0: return 0xA7A1;
3084                 case 0xA7A2: return 0xA7A3;
3085                 case 0xA7A4: return 0xA7A5;
3086                 case 0xA7A6: return 0xA7A7;
3087                 case 0xA7A8: return 0xA7A9;
3088                 case 0xFF21: return 0xFF41;
3089                 case 0xFF22: return 0xFF42;
3090                 case 0xFF23: return 0xFF43;
3091                 case 0xFF24: return 0xFF44;
3092                 case 0xFF25: return 0xFF45;
3093                 case 0xFF26: return 0xFF46;
3094                 case 0xFF27: return 0xFF47;
3095                 case 0xFF28: return 0xFF48;
3096                 case 0xFF29: return 0xFF49;
3097                 case 0xFF2A: return 0xFF4A;
3098                 case 0xFF2B: return 0xFF4B;
3099                 case 0xFF2C: return 0xFF4C;
3100                 case 0xFF2D: return 0xFF4D;
3101                 case 0xFF2E: return 0xFF4E;
3102                 case 0xFF2F: return 0xFF4F;
3103                 case 0xFF30: return 0xFF50;
3104                 case 0xFF31: return 0xFF51;
3105                 case 0xFF32: return 0xFF52;
3106                 case 0xFF33: return 0xFF53;
3107                 case 0xFF34: return 0xFF54;
3108                 case 0xFF35: return 0xFF55;
3109                 case 0xFF36: return 0xFF56;
3110                 case 0xFF37: return 0xFF57;
3111                 case 0xFF38: return 0xFF58;
3112                 case 0xFF39: return 0xFF59;
3113                 case 0xFF3A: return 0xFF5A;
3114                 case 0x10400: return 0x10428;
3115                 case 0x10401: return 0x10429;
3116                 case 0x10402: return 0x1042A;
3117                 case 0x10403: return 0x1042B;
3118                 case 0x10404: return 0x1042C;
3119                 case 0x10405: return 0x1042D;
3120                 case 0x10406: return 0x1042E;
3121                 case 0x10407: return 0x1042F;
3122                 case 0x10408: return 0x10430;
3123                 case 0x10409: return 0x10431;
3124                 case 0x1040A: return 0x10432;
3125                 case 0x1040B: return 0x10433;
3126                 case 0x1040C: return 0x10434;
3127                 case 0x1040D: return 0x10435;
3128                 case 0x1040E: return 0x10436;
3129                 case 0x1040F: return 0x10437;
3130                 case 0x10410: return 0x10438;
3131                 case 0x10411: return 0x10439;
3132                 case 0x10412: return 0x1043A;
3133                 case 0x10413: return 0x1043B;
3134                 case 0x10414: return 0x1043C;
3135                 case 0x10415: return 0x1043D;
3136                 case 0x10416: return 0x1043E;
3137                 case 0x10417: return 0x1043F;
3138                 case 0x10418: return 0x10440;
3139                 case 0x10419: return 0x10441;
3140                 case 0x1041A: return 0x10442;
3141                 case 0x1041B: return 0x10443;
3142                 case 0x1041C: return 0x10444;
3143                 case 0x1041D: return 0x10445;
3144                 case 0x1041E: return 0x10446;
3145                 case 0x1041F: return 0x10447;
3146                 case 0x10420: return 0x10448;
3147                 case 0x10421: return 0x10449;
3148                 case 0x10422: return 0x1044A;
3149                 case 0x10423: return 0x1044B;
3150                 case 0x10424: return 0x1044C;
3151                 case 0x10425: return 0x1044D;
3152                 case 0x10426: return 0x1044E;
3153                 case 0x10427: return 0x1044F;
3154                 default: return ch;
3155         }
3156 }