Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * Utility functions for conversion procs.
4 : *
5 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
6 : * Portions Copyright (c) 1994, Regents of the University of California
7 : *
8 : * IDENTIFICATION
9 : * src/backend/utils/mb/conv.c
10 : *
11 : *-------------------------------------------------------------------------
12 : */
13 : #include "postgres.h"
14 : #include "mb/pg_wchar.h"
15 :
16 :
17 : /*
18 : * local2local: a generic single byte charset encoding
19 : * conversion between two ASCII-superset encodings.
20 : *
21 : * l points to the source string of length len
22 : * p is the output area (must be large enough!)
23 : * src_encoding is the PG identifier for the source encoding
24 : * dest_encoding is the PG identifier for the target encoding
25 : * tab holds conversion entries for the source charset
26 : * starting from 128 (0x80). each entry in the table holds the corresponding
27 : * code point for the target charset, or 0 if there is no equivalent code.
28 : *
29 : * Returns the number of input bytes consumed. If noError is true, this can
30 : * be less than 'len'.
31 : */
32 : int
33 228 : local2local(const unsigned char *l,
34 : unsigned char *p,
35 : int len,
36 : int src_encoding,
37 : int dest_encoding,
38 : const unsigned char *tab,
39 : bool noError)
40 : {
41 228 : const unsigned char *start = l;
42 : unsigned char c1,
43 : c2;
44 :
45 732 : while (len > 0)
46 : {
47 612 : c1 = *l;
48 612 : if (c1 == 0)
49 : {
50 108 : if (noError)
51 54 : break;
52 54 : report_invalid_encoding(src_encoding, (const char *) l, len);
53 : }
54 504 : if (!IS_HIGHBIT_SET(c1))
55 306 : *p++ = c1;
56 : else
57 : {
58 198 : c2 = tab[c1 - HIGHBIT];
59 198 : if (c2)
60 198 : *p++ = c2;
61 : else
62 : {
63 0 : if (noError)
64 0 : break;
65 0 : report_untranslatable_char(src_encoding, dest_encoding,
66 : (const char *) l, len);
67 : }
68 : }
69 504 : l++;
70 504 : len--;
71 : }
72 174 : *p = '\0';
73 :
74 174 : return l - start;
75 : }
76 :
77 : /*
78 : * LATINn ---> MIC when the charset's local codes map directly to MIC
79 : *
80 : * l points to the source string of length len
81 : * p is the output area (must be large enough!)
82 : * lc is the mule character set id for the local encoding
83 : * encoding is the PG identifier for the local encoding
84 : *
85 : * Returns the number of input bytes consumed. If noError is true, this can
86 : * be less than 'len'.
87 : */
88 : int
89 30 : latin2mic(const unsigned char *l, unsigned char *p, int len,
90 : int lc, int encoding, bool noError)
91 : {
92 30 : const unsigned char *start = l;
93 : int c1;
94 :
95 120 : while (len > 0)
96 : {
97 90 : c1 = *l;
98 90 : if (c1 == 0)
99 : {
100 0 : if (noError)
101 0 : break;
102 0 : report_invalid_encoding(encoding, (const char *) l, len);
103 : }
104 90 : if (IS_HIGHBIT_SET(c1))
105 0 : *p++ = lc;
106 90 : *p++ = c1;
107 90 : l++;
108 90 : len--;
109 : }
110 30 : *p = '\0';
111 :
112 30 : return l - start;
113 : }
114 :
115 : /*
116 : * MIC ---> LATINn when the charset's local codes map directly to MIC
117 : *
118 : * mic points to the source string of length len
119 : * p is the output area (must be large enough!)
120 : * lc is the mule character set id for the local encoding
121 : * encoding is the PG identifier for the local encoding
122 : *
123 : * Returns the number of input bytes consumed. If noError is true, this can
124 : * be less than 'len'.
125 : */
126 : int
127 354 : mic2latin(const unsigned char *mic, unsigned char *p, int len,
128 : int lc, int encoding, bool noError)
129 : {
130 354 : const unsigned char *start = mic;
131 : int c1;
132 :
133 840 : while (len > 0)
134 : {
135 774 : c1 = *mic;
136 774 : if (c1 == 0)
137 : {
138 0 : if (noError)
139 0 : break;
140 0 : report_invalid_encoding(PG_MULE_INTERNAL, (const char *) mic, len);
141 : }
142 774 : if (!IS_HIGHBIT_SET(c1))
143 : {
144 : /* easy for ASCII */
145 360 : *p++ = c1;
146 360 : mic++;
147 360 : len--;
148 : }
149 : else
150 : {
151 414 : int l = pg_mule_mblen(mic);
152 :
153 414 : if (len < l)
154 : {
155 108 : if (noError)
156 54 : break;
157 54 : report_invalid_encoding(PG_MULE_INTERNAL, (const char *) mic,
158 : len);
159 : }
160 306 : if (l != 2 || c1 != lc || !IS_HIGHBIT_SET(mic[1]))
161 : {
162 180 : if (noError)
163 90 : break;
164 90 : report_untranslatable_char(PG_MULE_INTERNAL, encoding,
165 : (const char *) mic, len);
166 : }
167 126 : *p++ = mic[1];
168 126 : mic += 2;
169 126 : len -= 2;
170 : }
171 : }
172 210 : *p = '\0';
173 :
174 210 : return mic - start;
175 : }
176 :
177 :
178 : /*
179 : * latin2mic_with_table: a generic single byte charset encoding
180 : * conversion from a local charset to the mule internal code.
181 : *
182 : * l points to the source string of length len
183 : * p is the output area (must be large enough!)
184 : * lc is the mule character set id for the local encoding
185 : * encoding is the PG identifier for the local encoding
186 : * tab holds conversion entries for the local charset
187 : * starting from 128 (0x80). each entry in the table holds the corresponding
188 : * code point for the mule encoding, or 0 if there is no equivalent code.
189 : *
190 : * Returns the number of input bytes consumed. If noError is true, this can
191 : * be less than 'len'.
192 : */
193 : int
194 168 : latin2mic_with_table(const unsigned char *l,
195 : unsigned char *p,
196 : int len,
197 : int lc,
198 : int encoding,
199 : const unsigned char *tab,
200 : bool noError)
201 : {
202 168 : const unsigned char *start = l;
203 : unsigned char c1,
204 : c2;
205 :
206 492 : while (len > 0)
207 : {
208 432 : c1 = *l;
209 432 : if (c1 == 0)
210 : {
211 108 : if (noError)
212 54 : break;
213 54 : report_invalid_encoding(encoding, (const char *) l, len);
214 : }
215 324 : if (!IS_HIGHBIT_SET(c1))
216 126 : *p++ = c1;
217 : else
218 : {
219 198 : c2 = tab[c1 - HIGHBIT];
220 198 : if (c2)
221 : {
222 198 : *p++ = lc;
223 198 : *p++ = c2;
224 : }
225 : else
226 : {
227 0 : if (noError)
228 0 : break;
229 0 : report_untranslatable_char(encoding, PG_MULE_INTERNAL,
230 : (const char *) l, len);
231 : }
232 : }
233 324 : l++;
234 324 : len--;
235 : }
236 114 : *p = '\0';
237 :
238 114 : return l - start;
239 : }
240 :
241 : /*
242 : * mic2latin_with_table: a generic single byte charset encoding
243 : * conversion from the mule internal code to a local charset.
244 : *
245 : * mic points to the source string of length len
246 : * p is the output area (must be large enough!)
247 : * lc is the mule character set id for the local encoding
248 : * encoding is the PG identifier for the local encoding
249 : * tab holds conversion entries for the mule internal code's second byte,
250 : * starting from 128 (0x80). each entry in the table holds the corresponding
251 : * code point for the local charset, or 0 if there is no equivalent code.
252 : *
253 : * Returns the number of input bytes consumed. If noError is true, this can
254 : * be less than 'len'.
255 : */
256 : int
257 348 : mic2latin_with_table(const unsigned char *mic,
258 : unsigned char *p,
259 : int len,
260 : int lc,
261 : int encoding,
262 : const unsigned char *tab,
263 : bool noError)
264 : {
265 348 : const unsigned char *start = mic;
266 : unsigned char c1,
267 : c2;
268 :
269 816 : while (len > 0)
270 : {
271 756 : c1 = *mic;
272 756 : if (c1 == 0)
273 : {
274 0 : if (noError)
275 0 : break;
276 0 : report_invalid_encoding(PG_MULE_INTERNAL, (const char *) mic, len);
277 : }
278 756 : if (!IS_HIGHBIT_SET(c1))
279 : {
280 : /* easy for ASCII */
281 342 : *p++ = c1;
282 342 : mic++;
283 342 : len--;
284 : }
285 : else
286 : {
287 414 : int l = pg_mule_mblen(mic);
288 :
289 414 : if (len < l)
290 : {
291 108 : if (noError)
292 54 : break;
293 54 : report_invalid_encoding(PG_MULE_INTERNAL, (const char *) mic,
294 : len);
295 : }
296 306 : if (l != 2 || c1 != lc || !IS_HIGHBIT_SET(mic[1]) ||
297 126 : (c2 = tab[mic[1] - HIGHBIT]) == 0)
298 : {
299 180 : if (noError)
300 90 : break;
301 90 : report_untranslatable_char(PG_MULE_INTERNAL, encoding,
302 : (const char *) mic, len);
303 : break; /* keep compiler quiet */
304 : }
305 126 : *p++ = c2;
306 126 : mic += 2;
307 126 : len -= 2;
308 : }
309 : }
310 204 : *p = '\0';
311 :
312 204 : return mic - start;
313 : }
314 :
315 : /*
316 : * comparison routine for bsearch()
317 : * this routine is intended for combined UTF8 -> local code
318 : */
319 : static int
320 468 : compare3(const void *p1, const void *p2)
321 : {
322 : uint32 s1,
323 : s2,
324 : d1,
325 : d2;
326 :
327 468 : s1 = *(const uint32 *) p1;
328 468 : s2 = *((const uint32 *) p1 + 1);
329 468 : d1 = ((const pg_utf_to_local_combined *) p2)->utf1;
330 468 : d2 = ((const pg_utf_to_local_combined *) p2)->utf2;
331 468 : return (s1 > d1 || (s1 == d1 && s2 > d2)) ? 1 : ((s1 == d1 && s2 == d2) ? 0 : -1);
332 : }
333 :
334 : /*
335 : * comparison routine for bsearch()
336 : * this routine is intended for local code -> combined UTF8
337 : */
338 : static int
339 162 : compare4(const void *p1, const void *p2)
340 : {
341 : uint32 v1,
342 : v2;
343 :
344 162 : v1 = *(const uint32 *) p1;
345 162 : v2 = ((const pg_local_to_utf_combined *) p2)->code;
346 162 : return (v1 > v2) ? 1 : ((v1 == v2) ? 0 : -1);
347 : }
348 :
349 : /*
350 : * store 32bit character representation into multibyte stream
351 : */
352 : static inline unsigned char *
353 1134 : store_coded_char(unsigned char *dest, uint32 code)
354 : {
355 1134 : if (code & 0xff000000)
356 126 : *dest++ = code >> 24;
357 1134 : if (code & 0x00ff0000)
358 522 : *dest++ = code >> 16;
359 1134 : if (code & 0x0000ff00)
360 1008 : *dest++ = code >> 8;
361 1134 : if (code & 0x000000ff)
362 1134 : *dest++ = code;
363 1134 : return dest;
364 : }
365 :
366 : /*
367 : * Convert a character using a conversion radix tree.
368 : *
369 : * 'l' is the length of the input character in bytes, and b1-b4 are
370 : * the input character's bytes.
371 : */
372 : static inline uint32
373 1980 : pg_mb_radix_conv(const pg_mb_radix_tree *rt,
374 : int l,
375 : unsigned char b1,
376 : unsigned char b2,
377 : unsigned char b3,
378 : unsigned char b4)
379 : {
380 1980 : if (l == 4)
381 : {
382 : /* 4-byte code */
383 :
384 : /* check code validity */
385 90 : if (b1 < rt->b4_1_lower || b1 > rt->b4_1_upper ||
386 90 : b2 < rt->b4_2_lower || b2 > rt->b4_2_upper ||
387 90 : b3 < rt->b4_3_lower || b3 > rt->b4_3_upper ||
388 90 : b4 < rt->b4_4_lower || b4 > rt->b4_4_upper)
389 0 : return 0;
390 :
391 : /* perform lookup */
392 90 : if (rt->chars32)
393 : {
394 90 : uint32 idx = rt->b4root;
395 :
396 90 : idx = rt->chars32[b1 + idx - rt->b4_1_lower];
397 90 : idx = rt->chars32[b2 + idx - rt->b4_2_lower];
398 90 : idx = rt->chars32[b3 + idx - rt->b4_3_lower];
399 90 : return rt->chars32[b4 + idx - rt->b4_4_lower];
400 : }
401 : else
402 : {
403 0 : uint16 idx = rt->b4root;
404 :
405 0 : idx = rt->chars16[b1 + idx - rt->b4_1_lower];
406 0 : idx = rt->chars16[b2 + idx - rt->b4_2_lower];
407 0 : idx = rt->chars16[b3 + idx - rt->b4_3_lower];
408 0 : return rt->chars16[b4 + idx - rt->b4_4_lower];
409 : }
410 : }
411 1890 : else if (l == 3)
412 : {
413 : /* 3-byte code */
414 :
415 : /* check code validity */
416 936 : if (b2 < rt->b3_1_lower || b2 > rt->b3_1_upper ||
417 288 : b3 < rt->b3_2_lower || b3 > rt->b3_2_upper ||
418 288 : b4 < rt->b3_3_lower || b4 > rt->b3_3_upper)
419 648 : return 0;
420 :
421 : /* perform lookup */
422 288 : if (rt->chars32)
423 : {
424 288 : uint32 idx = rt->b3root;
425 :
426 288 : idx = rt->chars32[b2 + idx - rt->b3_1_lower];
427 288 : idx = rt->chars32[b3 + idx - rt->b3_2_lower];
428 288 : return rt->chars32[b4 + idx - rt->b3_3_lower];
429 : }
430 : else
431 : {
432 0 : uint16 idx = rt->b3root;
433 :
434 0 : idx = rt->chars16[b2 + idx - rt->b3_1_lower];
435 0 : idx = rt->chars16[b3 + idx - rt->b3_2_lower];
436 0 : return rt->chars16[b4 + idx - rt->b3_3_lower];
437 : }
438 : }
439 954 : else if (l == 2)
440 : {
441 : /* 2-byte code */
442 :
443 : /* check code validity - first byte */
444 756 : if (b3 < rt->b2_1_lower || b3 > rt->b2_1_upper ||
445 684 : b4 < rt->b2_2_lower || b4 > rt->b2_2_upper)
446 72 : return 0;
447 :
448 : /* perform lookup */
449 684 : if (rt->chars32)
450 : {
451 522 : uint32 idx = rt->b2root;
452 :
453 522 : idx = rt->chars32[b3 + idx - rt->b2_1_lower];
454 522 : return rt->chars32[b4 + idx - rt->b2_2_lower];
455 : }
456 : else
457 : {
458 162 : uint16 idx = rt->b2root;
459 :
460 162 : idx = rt->chars16[b3 + idx - rt->b2_1_lower];
461 162 : return rt->chars16[b4 + idx - rt->b2_2_lower];
462 : }
463 : }
464 198 : else if (l == 1)
465 : {
466 : /* 1-byte code */
467 :
468 : /* check code validity - first byte */
469 198 : if (b4 < rt->b1_lower || b4 > rt->b1_upper)
470 0 : return 0;
471 :
472 : /* perform lookup */
473 198 : if (rt->chars32)
474 198 : return rt->chars32[b4 + rt->b1root - rt->b1_lower];
475 : else
476 0 : return rt->chars16[b4 + rt->b1root - rt->b1_lower];
477 : }
478 0 : return 0; /* shouldn't happen */
479 : }
480 :
481 : /*
482 : * UTF8 ---> local code
483 : *
484 : * utf: input string in UTF8 encoding (need not be null-terminated)
485 : * len: length of input string (in bytes)
486 : * iso: pointer to the output area (must be large enough!)
487 : (output string will be null-terminated)
488 : * map: conversion map for single characters
489 : * cmap: conversion map for combined characters
490 : * (optional, pass NULL if none)
491 : * cmapsize: number of entries in the conversion map for combined characters
492 : * (optional, pass 0 if none)
493 : * conv_func: algorithmic encoding conversion function
494 : * (optional, pass NULL if none)
495 : * encoding: PG identifier for the local encoding
496 : *
497 : * For each character, the cmap (if provided) is consulted first; if no match,
498 : * the map is consulted next; if still no match, the conv_func (if provided)
499 : * is applied. An error is raised if no match is found.
500 : *
501 : * See pg_wchar.h for more details about the data structures used here.
502 : *
503 : * Returns the number of input bytes consumed. If noError is true, this can
504 : * be less than 'len'.
505 : */
506 : int
507 2268 : UtfToLocal(const unsigned char *utf, int len,
508 : unsigned char *iso,
509 : const pg_mb_radix_tree *map,
510 : const pg_utf_to_local_combined *cmap, int cmapsize,
511 : utf_local_conversion_func conv_func,
512 : int encoding, bool noError)
513 : {
514 : uint32 iutf;
515 : int l;
516 : const pg_utf_to_local_combined *cp;
517 2268 : const unsigned char *start = utf;
518 :
519 2268 : if (!PG_VALID_ENCODING(encoding))
520 0 : ereport(ERROR,
521 : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
522 : errmsg("invalid encoding number: %d", encoding)));
523 :
524 7056 : for (; len > 0; len -= l)
525 : {
526 6408 : unsigned char b1 = 0;
527 6408 : unsigned char b2 = 0;
528 6408 : unsigned char b3 = 0;
529 6408 : unsigned char b4 = 0;
530 :
531 : /* "break" cases all represent errors */
532 6408 : if (*utf == '\0')
533 180 : break;
534 :
535 6228 : l = pg_utf_mblen(utf);
536 6228 : if (len < l)
537 216 : break;
538 :
539 6012 : if (!pg_utf8_islegal(utf, l))
540 360 : break;
541 :
542 5652 : if (l == 1)
543 : {
544 : /* ASCII case is easy, assume it's one-to-one conversion */
545 4248 : *iso++ = *utf++;
546 4248 : continue;
547 : }
548 :
549 : /* collect coded char of length l */
550 1404 : if (l == 2)
551 : {
552 414 : b3 = *utf++;
553 414 : b4 = *utf++;
554 : }
555 990 : else if (l == 3)
556 : {
557 990 : b2 = *utf++;
558 990 : b3 = *utf++;
559 990 : b4 = *utf++;
560 : }
561 0 : else if (l == 4)
562 : {
563 0 : b1 = *utf++;
564 0 : b2 = *utf++;
565 0 : b3 = *utf++;
566 0 : b4 = *utf++;
567 : }
568 : else
569 : {
570 0 : elog(ERROR, "unsupported character length %d", l);
571 : iutf = 0; /* keep compiler quiet */
572 : }
573 1404 : iutf = (b1 << 24 | b2 << 16 | b3 << 8 | b4);
574 :
575 : /* First, try with combined map if possible */
576 1404 : if (cmap && len > l)
577 : {
578 144 : const unsigned char *utf_save = utf;
579 144 : int len_save = len;
580 144 : int l_save = l;
581 :
582 : /* collect next character, same as above */
583 144 : len -= l;
584 :
585 144 : l = pg_utf_mblen(utf);
586 144 : if (len < l)
587 : {
588 : /* need more data to decide if this is a combined char */
589 36 : utf -= l_save;
590 36 : break;
591 : }
592 :
593 108 : if (!pg_utf8_islegal(utf, l))
594 : {
595 0 : if (!noError)
596 0 : report_invalid_encoding(PG_UTF8, (const char *) utf, len);
597 0 : utf -= l_save;
598 0 : break;
599 : }
600 :
601 : /* We assume ASCII character cannot be in combined map */
602 108 : if (l > 1)
603 : {
604 : uint32 iutf2;
605 : uint32 cutf[2];
606 :
607 108 : if (l == 2)
608 : {
609 54 : iutf2 = *utf++ << 8;
610 54 : iutf2 |= *utf++;
611 : }
612 54 : else if (l == 3)
613 : {
614 54 : iutf2 = *utf++ << 16;
615 54 : iutf2 |= *utf++ << 8;
616 54 : iutf2 |= *utf++;
617 : }
618 0 : else if (l == 4)
619 : {
620 0 : iutf2 = *utf++ << 24;
621 0 : iutf2 |= *utf++ << 16;
622 0 : iutf2 |= *utf++ << 8;
623 0 : iutf2 |= *utf++;
624 : }
625 : else
626 : {
627 0 : elog(ERROR, "unsupported character length %d", l);
628 : iutf2 = 0; /* keep compiler quiet */
629 : }
630 :
631 108 : cutf[0] = iutf;
632 108 : cutf[1] = iutf2;
633 :
634 108 : cp = bsearch(cutf, cmap, cmapsize,
635 : sizeof(pg_utf_to_local_combined), compare3);
636 :
637 108 : if (cp)
638 : {
639 18 : iso = store_coded_char(iso, cp->code);
640 18 : continue;
641 : }
642 : }
643 :
644 : /* fail, so back up to reprocess second character next time */
645 90 : utf = utf_save;
646 90 : len = len_save;
647 90 : l = l_save;
648 : }
649 :
650 : /* Now check ordinary map */
651 1350 : if (map)
652 : {
653 1350 : uint32 converted = pg_mb_radix_conv(map, l, b1, b2, b3, b4);
654 :
655 1350 : if (converted)
656 : {
657 450 : iso = store_coded_char(iso, converted);
658 450 : continue;
659 : }
660 : }
661 :
662 : /* if there's a conversion function, try that */
663 900 : if (conv_func)
664 : {
665 72 : uint32 converted = (*conv_func) (iutf);
666 :
667 72 : if (converted)
668 : {
669 72 : iso = store_coded_char(iso, converted);
670 72 : continue;
671 : }
672 : }
673 :
674 : /* failed to translate this character */
675 828 : utf -= l;
676 828 : if (noError)
677 414 : break;
678 414 : report_untranslatable_char(PG_UTF8, encoding,
679 : (const char *) utf, len);
680 : }
681 :
682 : /* if we broke out of loop early, must be invalid input */
683 1854 : if (len > 0 && !noError)
684 396 : report_invalid_encoding(PG_UTF8, (const char *) utf, len);
685 :
686 1458 : *iso = '\0';
687 :
688 1458 : return utf - start;
689 : }
690 :
691 : /*
692 : * local code ---> UTF8
693 : *
694 : * iso: input string in local encoding (need not be null-terminated)
695 : * len: length of input string (in bytes)
696 : * utf: pointer to the output area (must be large enough!)
697 : (output string will be null-terminated)
698 : * map: conversion map for single characters
699 : * cmap: conversion map for combined characters
700 : * (optional, pass NULL if none)
701 : * cmapsize: number of entries in the conversion map for combined characters
702 : * (optional, pass 0 if none)
703 : * conv_func: algorithmic encoding conversion function
704 : * (optional, pass NULL if none)
705 : * encoding: PG identifier for the local encoding
706 : *
707 : * For each character, the map is consulted first; if no match, the cmap
708 : * (if provided) is consulted next; if still no match, the conv_func
709 : * (if provided) is applied. An error is raised if no match is found.
710 : *
711 : * See pg_wchar.h for more details about the data structures used here.
712 : *
713 : * Returns the number of input bytes consumed. If noError is true, this can
714 : * be less than 'len'.
715 : */
716 : int
717 1284 : LocalToUtf(const unsigned char *iso, int len,
718 : unsigned char *utf,
719 : const pg_mb_radix_tree *map,
720 : const pg_local_to_utf_combined *cmap, int cmapsize,
721 : utf_local_conversion_func conv_func,
722 : int encoding,
723 : bool noError)
724 : {
725 : uint32 iiso;
726 : int l;
727 : const pg_local_to_utf_combined *cp;
728 1284 : const unsigned char *start = iso;
729 :
730 1284 : if (!PG_VALID_ENCODING(encoding))
731 0 : ereport(ERROR,
732 : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
733 : errmsg("invalid encoding number: %d", encoding)));
734 :
735 5812 : for (; len > 0; len -= l)
736 : {
737 5320 : unsigned char b1 = 0;
738 5320 : unsigned char b2 = 0;
739 5320 : unsigned char b3 = 0;
740 5320 : unsigned char b4 = 0;
741 :
742 : /* "break" cases all represent errors */
743 5320 : if (*iso == '\0')
744 324 : break;
745 :
746 4996 : if (!IS_HIGHBIT_SET(*iso))
747 : {
748 : /* ASCII case is easy, assume it's one-to-one conversion */
749 3970 : *utf++ = *iso++;
750 3970 : l = 1;
751 3970 : continue;
752 : }
753 :
754 1026 : l = pg_encoding_verifymbchar(encoding, (const char *) iso, len);
755 1026 : if (l < 0)
756 396 : break;
757 :
758 : /* collect coded char of length l */
759 630 : if (l == 1)
760 198 : b4 = *iso++;
761 432 : else if (l == 2)
762 : {
763 342 : b3 = *iso++;
764 342 : b4 = *iso++;
765 : }
766 90 : else if (l == 3)
767 : {
768 0 : b2 = *iso++;
769 0 : b3 = *iso++;
770 0 : b4 = *iso++;
771 : }
772 90 : else if (l == 4)
773 : {
774 90 : b1 = *iso++;
775 90 : b2 = *iso++;
776 90 : b3 = *iso++;
777 90 : b4 = *iso++;
778 : }
779 : else
780 : {
781 0 : elog(ERROR, "unsupported character length %d", l);
782 : iiso = 0; /* keep compiler quiet */
783 : }
784 630 : iiso = (b1 << 24 | b2 << 16 | b3 << 8 | b4);
785 :
786 630 : if (map)
787 : {
788 630 : uint32 converted = pg_mb_radix_conv(map, l, b1, b2, b3, b4);
789 :
790 630 : if (converted)
791 : {
792 468 : utf = store_coded_char(utf, converted);
793 468 : continue;
794 : }
795 :
796 : /* If there's a combined character map, try that */
797 162 : if (cmap)
798 : {
799 36 : cp = bsearch(&iiso, cmap, cmapsize,
800 : sizeof(pg_local_to_utf_combined), compare4);
801 :
802 36 : if (cp)
803 : {
804 36 : utf = store_coded_char(utf, cp->utf1);
805 36 : utf = store_coded_char(utf, cp->utf2);
806 36 : continue;
807 : }
808 : }
809 : }
810 :
811 : /* if there's a conversion function, try that */
812 126 : if (conv_func)
813 : {
814 90 : uint32 converted = (*conv_func) (iiso);
815 :
816 90 : if (converted)
817 : {
818 54 : utf = store_coded_char(utf, converted);
819 54 : continue;
820 : }
821 : }
822 :
823 : /* failed to translate this character */
824 72 : iso -= l;
825 72 : if (noError)
826 36 : break;
827 36 : report_untranslatable_char(encoding, PG_UTF8,
828 : (const char *) iso, len);
829 : }
830 :
831 : /* if we broke out of loop early, must be invalid input */
832 1248 : if (len > 0 && !noError)
833 354 : report_invalid_encoding(encoding, (const char *) iso, len);
834 :
835 894 : *utf = '\0';
836 :
837 894 : return iso - start;
838 : }
|