1 /*
2 ******************************************************************************
3 *
4 *   Copyright (C) 1998-2013, International Business Machines
5 *   Corporation and others.  All Rights Reserved.
6 *
7 ******************************************************************************
8 *
9 *  ucnv.c:
10 *  Implements APIs for the ICU's codeset conversion library;
11 *  mostly calls through internal functions;
12 *  created by Bertrand A. Damiba
13 *
14 * Modification History:
15 *
16 *   Date        Name        Description
17 *   04/04/99    helena      Fixed internal header inclusion.
18 *   05/09/00    helena      Added implementation to handle fallback mappings.
19 *   06/20/2000  helena      OS/400 port changes; mostly typecast.
20 */
21 
22 #include "unicode/utypes.h"
23 
24 #if !UCONFIG_NO_CONVERSION
25 
26 #include "unicode/ustring.h"
27 #include "unicode/ucnv.h"
28 #include "unicode/ucnv_err.h"
29 #include "unicode/uset.h"
30 #include "unicode/utf.h"
31 #include "unicode/utf16.h"
32 #include "putilimp.h"
33 #include "cmemory.h"
34 #include "cstring.h"
35 #include "uassert.h"
36 #include "utracimp.h"
37 #include "ustr_imp.h"
38 #include "ucnv_imp.h"
39 #include "ucnv_cnv.h"
40 #include "ucnv_bld.h"
41 
42 /* size of intermediate and preflighting buffers in ucnv_convert() */
43 #define CHUNK_SIZE 1024
44 
45 typedef struct UAmbiguousConverter {
46     const char *name;
47     const UChar variant5c;
48 } UAmbiguousConverter;
49 
50 static const UAmbiguousConverter ambiguousConverters[]={
51     { "ibm-897_P100-1995", 0xa5 },
52     { "ibm-942_P120-1999", 0xa5 },
53     { "ibm-943_P130-1999", 0xa5 },
54     { "ibm-946_P100-1995", 0xa5 },
55     { "ibm-33722_P120-1999", 0xa5 },
56     { "ibm-1041_P100-1995", 0xa5 },
57     /*{ "ibm-54191_P100-2006", 0xa5 },*/
58     /*{ "ibm-62383_P100-2007", 0xa5 },*/
59     /*{ "ibm-891_P100-1995", 0x20a9 },*/
60     { "ibm-944_P100-1995", 0x20a9 },
61     { "ibm-949_P110-1999", 0x20a9 },
62     { "ibm-1363_P110-1997", 0x20a9 },
63     { "ISO_2022,locale=ko,version=0", 0x20a9 },
64     { "ibm-1088_P100-1995", 0x20a9 }
65 };
66 
67 /*Calls through createConverter */
68 U_CAPI UConverter* U_EXPORT2
ucnv_open(const char * name,UErrorCode * err)69 ucnv_open (const char *name,
70                        UErrorCode * err)
71 {
72     UConverter *r;
73 
74     if (err == NULL || U_FAILURE (*err)) {
75         return NULL;
76     }
77 
78     r =  ucnv_createConverter(NULL, name, err);
79     return r;
80 }
81 
82 U_CAPI UConverter* U_EXPORT2
ucnv_openPackage(const char * packageName,const char * converterName,UErrorCode * err)83 ucnv_openPackage   (const char *packageName, const char *converterName, UErrorCode * err)
84 {
85     return ucnv_createConverterFromPackage(packageName, converterName,  err);
86 }
87 
88 /*Extracts the UChar* to a char* and calls through createConverter */
89 U_CAPI UConverter*   U_EXPORT2
ucnv_openU(const UChar * name,UErrorCode * err)90 ucnv_openU (const UChar * name,
91                          UErrorCode * err)
92 {
93     char asciiName[UCNV_MAX_CONVERTER_NAME_LENGTH];
94 
95     if (err == NULL || U_FAILURE(*err))
96         return NULL;
97     if (name == NULL)
98         return ucnv_open (NULL, err);
99     if (u_strlen(name) >= UCNV_MAX_CONVERTER_NAME_LENGTH)
100     {
101         *err = U_ILLEGAL_ARGUMENT_ERROR;
102         return NULL;
103     }
104     return ucnv_open(u_austrcpy(asciiName, name), err);
105 }
106 
107 /* Copy the string that is represented by the UConverterPlatform enum
108  * @param platformString An output buffer
109  * @param platform An enum representing a platform
110  * @return the length of the copied string.
111  */
112 static int32_t
ucnv_copyPlatformString(char * platformString,UConverterPlatform pltfrm)113 ucnv_copyPlatformString(char *platformString, UConverterPlatform pltfrm)
114 {
115     switch (pltfrm)
116     {
117     case UCNV_IBM:
118         uprv_strcpy(platformString, "ibm-");
119         return 4;
120     case UCNV_UNKNOWN:
121         break;
122     }
123 
124     /* default to empty string */
125     *platformString = 0;
126     return 0;
127 }
128 
129 /*Assumes a $platform-#codepage.$CONVERTER_FILE_EXTENSION scheme and calls
130  *through createConverter*/
131 U_CAPI UConverter*   U_EXPORT2
ucnv_openCCSID(int32_t codepage,UConverterPlatform platform,UErrorCode * err)132 ucnv_openCCSID (int32_t codepage,
133                 UConverterPlatform platform,
134                 UErrorCode * err)
135 {
136     char myName[UCNV_MAX_CONVERTER_NAME_LENGTH];
137     int32_t myNameLen;
138 
139     if (err == NULL || U_FAILURE (*err))
140         return NULL;
141 
142     /* ucnv_copyPlatformString could return "ibm-" or "cp" */
143     myNameLen = ucnv_copyPlatformString(myName, platform);
144     T_CString_integerToString(myName + myNameLen, codepage, 10);
145 
146     return ucnv_createConverter(NULL, myName, err);
147 }
148 
149 /* Creating a temporary stack-based object that can be used in one thread,
150 and created from a converter that is shared across threads.
151 */
152 
153 U_CAPI UConverter* U_EXPORT2
ucnv_safeClone(const UConverter * cnv,void * stackBuffer,int32_t * pBufferSize,UErrorCode * status)154 ucnv_safeClone(const UConverter* cnv, void *stackBuffer, int32_t *pBufferSize, UErrorCode *status)
155 {
156     UConverter *localConverter, *allocatedConverter;
157     int32_t stackBufferSize;
158     int32_t bufferSizeNeeded;
159     char *stackBufferChars = (char *)stackBuffer;
160     UErrorCode cbErr;
161     UConverterToUnicodeArgs toUArgs = {
162         sizeof(UConverterToUnicodeArgs),
163             TRUE,
164             NULL,
165             NULL,
166             NULL,
167             NULL,
168             NULL,
169             NULL
170     };
171     UConverterFromUnicodeArgs fromUArgs = {
172         sizeof(UConverterFromUnicodeArgs),
173             TRUE,
174             NULL,
175             NULL,
176             NULL,
177             NULL,
178             NULL,
179             NULL
180     };
181 
182     UTRACE_ENTRY_OC(UTRACE_UCNV_CLONE);
183 
184     if (status == NULL || U_FAILURE(*status)){
185         UTRACE_EXIT_STATUS(status? *status: U_ILLEGAL_ARGUMENT_ERROR);
186         return NULL;
187     }
188 
189     if (cnv == NULL) {
190         *status = U_ILLEGAL_ARGUMENT_ERROR;
191         UTRACE_EXIT_STATUS(*status);
192         return NULL;
193     }
194 
195     UTRACE_DATA3(UTRACE_OPEN_CLOSE, "clone converter %s at %p into stackBuffer %p",
196                                     ucnv_getName(cnv, status), cnv, stackBuffer);
197 
198     if (cnv->sharedData->impl->safeClone != NULL) {
199         /* call the custom safeClone function for sizing */
200         bufferSizeNeeded = 0;
201         cnv->sharedData->impl->safeClone(cnv, NULL, &bufferSizeNeeded, status);
202         if (U_FAILURE(*status)) {
203             UTRACE_EXIT_STATUS(*status);
204             return NULL;
205         }
206     }
207     else
208     {
209         /* inherent sizing */
210         bufferSizeNeeded = sizeof(UConverter);
211     }
212 
213     if (pBufferSize == NULL) {
214         stackBufferSize = 1;
215         pBufferSize = &stackBufferSize;
216     } else {
217         stackBufferSize = *pBufferSize;
218         if (stackBufferSize <= 0){ /* 'preflighting' request - set needed size into *pBufferSize */
219             *pBufferSize = bufferSizeNeeded;
220             UTRACE_EXIT_VALUE(bufferSizeNeeded);
221             return NULL;
222         }
223     }
224 
225 
226     /* Pointers on 64-bit platforms need to be aligned
227      * on a 64-bit boundary in memory.
228      */
229     if (U_ALIGNMENT_OFFSET(stackBuffer) != 0) {
230         int32_t offsetUp = (int32_t)U_ALIGNMENT_OFFSET_UP(stackBufferChars);
231         if(stackBufferSize > offsetUp) {
232             stackBufferSize -= offsetUp;
233             stackBufferChars += offsetUp;
234         } else {
235             /* prevent using the stack buffer but keep the size > 0 so that we do not just preflight */
236             stackBufferSize = 1;
237         }
238     }
239 
240     stackBuffer = (void *)stackBufferChars;
241 
242     /* Now, see if we must allocate any memory */
243     if (stackBufferSize < bufferSizeNeeded || stackBuffer == NULL)
244     {
245         /* allocate one here...*/
246         localConverter = allocatedConverter = (UConverter *) uprv_malloc (bufferSizeNeeded);
247 
248         if(localConverter == NULL) {
249             *status = U_MEMORY_ALLOCATION_ERROR;
250             UTRACE_EXIT_STATUS(*status);
251             return NULL;
252         }
253         *status = U_SAFECLONE_ALLOCATED_WARNING;
254 
255         /* record the fact that memory was allocated */
256         *pBufferSize = bufferSizeNeeded;
257     } else {
258         /* just use the stack buffer */
259         localConverter = (UConverter*) stackBuffer;
260         allocatedConverter = NULL;
261     }
262 
263     uprv_memset(localConverter, 0, bufferSizeNeeded);
264 
265     /* Copy initial state */
266     uprv_memcpy(localConverter, cnv, sizeof(UConverter));
267     localConverter->isCopyLocal = localConverter->isExtraLocal = FALSE;
268 
269     /* copy the substitution string */
270     if (cnv->subChars == (uint8_t *)cnv->subUChars) {
271         localConverter->subChars = (uint8_t *)localConverter->subUChars;
272     } else {
273         localConverter->subChars = (uint8_t *)uprv_malloc(UCNV_ERROR_BUFFER_LENGTH * U_SIZEOF_UCHAR);
274         if (localConverter->subChars == NULL) {
275             uprv_free(allocatedConverter);
276             UTRACE_EXIT_STATUS(*status);
277             return NULL;
278         }
279         uprv_memcpy(localConverter->subChars, cnv->subChars, UCNV_ERROR_BUFFER_LENGTH * U_SIZEOF_UCHAR);
280     }
281 
282     /* now either call the safeclone fcn or not */
283     if (cnv->sharedData->impl->safeClone != NULL) {
284         /* call the custom safeClone function */
285         localConverter = cnv->sharedData->impl->safeClone(cnv, localConverter, pBufferSize, status);
286     }
287 
288     if(localConverter==NULL || U_FAILURE(*status)) {
289         if (allocatedConverter != NULL && allocatedConverter->subChars != (uint8_t *)allocatedConverter->subUChars) {
290             uprv_free(allocatedConverter->subChars);
291         }
292         uprv_free(allocatedConverter);
293         UTRACE_EXIT_STATUS(*status);
294         return NULL;
295     }
296 
297     /* increment refcount of shared data if needed */
298     /*
299     Checking whether it's an algorithic converter is okay
300     in multithreaded applications because the value never changes.
301     Don't check referenceCounter for any other value.
302     */
303     if (cnv->sharedData->referenceCounter != ~0) {
304         ucnv_incrementRefCount(cnv->sharedData);
305     }
306 
307     if(localConverter == (UConverter*)stackBuffer) {
308         /* we're using user provided data - set to not destroy */
309         localConverter->isCopyLocal = TRUE;
310     }
311 
312     /* allow callback functions to handle any memory allocation */
313     toUArgs.converter = fromUArgs.converter = localConverter;
314     cbErr = U_ZERO_ERROR;
315     cnv->fromCharErrorBehaviour(cnv->toUContext, &toUArgs, NULL, 0, UCNV_CLONE, &cbErr);
316     cbErr = U_ZERO_ERROR;
317     cnv->fromUCharErrorBehaviour(cnv->fromUContext, &fromUArgs, NULL, 0, 0, UCNV_CLONE, &cbErr);
318 
319     UTRACE_EXIT_PTR_STATUS(localConverter, *status);
320     return localConverter;
321 }
322 
323 
324 
325 /*Decreases the reference counter in the shared immutable section of the object
326  *and frees the mutable part*/
327 
328 U_CAPI void  U_EXPORT2
ucnv_close(UConverter * converter)329 ucnv_close (UConverter * converter)
330 {
331     UErrorCode errorCode = U_ZERO_ERROR;
332 
333     UTRACE_ENTRY_OC(UTRACE_UCNV_CLOSE);
334 
335     if (converter == NULL)
336     {
337         UTRACE_EXIT();
338         return;
339     }
340 
341     UTRACE_DATA3(UTRACE_OPEN_CLOSE, "close converter %s at %p, isCopyLocal=%b",
342         ucnv_getName(converter, &errorCode), converter, converter->isCopyLocal);
343 
344     /* In order to speed up the close, only call the callbacks when they have been changed.
345     This performance check will only work when the callbacks are set within a shared library
346     or from user code that statically links this code. */
347     /* first, notify the callback functions that the converter is closed */
348     if (converter->fromCharErrorBehaviour != UCNV_TO_U_DEFAULT_CALLBACK) {
349         UConverterToUnicodeArgs toUArgs = {
350             sizeof(UConverterToUnicodeArgs),
351                 TRUE,
352                 NULL,
353                 NULL,
354                 NULL,
355                 NULL,
356                 NULL,
357                 NULL
358         };
359 
360         toUArgs.converter = converter;
361         errorCode = U_ZERO_ERROR;
362         converter->fromCharErrorBehaviour(converter->toUContext, &toUArgs, NULL, 0, UCNV_CLOSE, &errorCode);
363     }
364     if (converter->fromUCharErrorBehaviour != UCNV_FROM_U_DEFAULT_CALLBACK) {
365         UConverterFromUnicodeArgs fromUArgs = {
366             sizeof(UConverterFromUnicodeArgs),
367                 TRUE,
368                 NULL,
369                 NULL,
370                 NULL,
371                 NULL,
372                 NULL,
373                 NULL
374         };
375         fromUArgs.converter = converter;
376         errorCode = U_ZERO_ERROR;
377         converter->fromUCharErrorBehaviour(converter->fromUContext, &fromUArgs, NULL, 0, 0, UCNV_CLOSE, &errorCode);
378     }
379 
380     if (converter->sharedData->impl->close != NULL) {
381         converter->sharedData->impl->close(converter);
382     }
383 
384     if (converter->subChars != (uint8_t *)converter->subUChars) {
385         uprv_free(converter->subChars);
386     }
387 
388     /*
389     Checking whether it's an algorithic converter is okay
390     in multithreaded applications because the value never changes.
391     Don't check referenceCounter for any other value.
392     */
393     if (converter->sharedData->referenceCounter != ~0) {
394         ucnv_unloadSharedDataIfReady(converter->sharedData);
395     }
396 
397     if(!converter->isCopyLocal){
398         uprv_free(converter);
399     }
400 
401     UTRACE_EXIT();
402 }
403 
404 /*returns a single Name from the list, will return NULL if out of bounds
405  */
406 U_CAPI const char*   U_EXPORT2
ucnv_getAvailableName(int32_t n)407 ucnv_getAvailableName (int32_t n)
408 {
409     if (0 <= n && n <= 0xffff) {
410         UErrorCode err = U_ZERO_ERROR;
411         const char *name = ucnv_bld_getAvailableConverter((uint16_t)n, &err);
412         if (U_SUCCESS(err)) {
413             return name;
414         }
415     }
416     return NULL;
417 }
418 
419 U_CAPI int32_t   U_EXPORT2
ucnv_countAvailable()420 ucnv_countAvailable ()
421 {
422     UErrorCode err = U_ZERO_ERROR;
423     return ucnv_bld_countAvailableConverters(&err);
424 }
425 
426 U_CAPI void    U_EXPORT2
ucnv_getSubstChars(const UConverter * converter,char * mySubChar,int8_t * len,UErrorCode * err)427 ucnv_getSubstChars (const UConverter * converter,
428                     char *mySubChar,
429                     int8_t * len,
430                     UErrorCode * err)
431 {
432     if (U_FAILURE (*err))
433         return;
434 
435     if (converter->subCharLen <= 0) {
436         /* Unicode string or empty string from ucnv_setSubstString(). */
437         *len = 0;
438         return;
439     }
440 
441     if (*len < converter->subCharLen) /*not enough space in subChars */
442     {
443         *err = U_INDEX_OUTOFBOUNDS_ERROR;
444         return;
445     }
446 
447     uprv_memcpy (mySubChar, converter->subChars, converter->subCharLen);   /*fills in the subchars */
448     *len = converter->subCharLen; /*store # of bytes copied to buffer */
449 }
450 
451 U_CAPI void    U_EXPORT2
ucnv_setSubstChars(UConverter * converter,const char * mySubChar,int8_t len,UErrorCode * err)452 ucnv_setSubstChars (UConverter * converter,
453                     const char *mySubChar,
454                     int8_t len,
455                     UErrorCode * err)
456 {
457     if (U_FAILURE (*err))
458         return;
459 
460     /*Makes sure that the subChar is within the codepages char length boundaries */
461     if ((len > converter->sharedData->staticData->maxBytesPerChar)
462      || (len < converter->sharedData->staticData->minBytesPerChar))
463     {
464         *err = U_ILLEGAL_ARGUMENT_ERROR;
465         return;
466     }
467 
468     uprv_memcpy (converter->subChars, mySubChar, len); /*copies the subchars */
469     converter->subCharLen = len;  /*sets the new len */
470 
471     /*
472     * There is currently (2001Feb) no separate API to set/get subChar1.
473     * In order to always have subChar written after it is explicitly set,
474     * we set subChar1 to 0.
475     */
476     converter->subChar1 = 0;
477 
478     return;
479 }
480 
481 U_CAPI void U_EXPORT2
ucnv_setSubstString(UConverter * cnv,const UChar * s,int32_t length,UErrorCode * err)482 ucnv_setSubstString(UConverter *cnv,
483                     const UChar *s,
484                     int32_t length,
485                     UErrorCode *err) {
486     UAlignedMemory cloneBuffer[U_CNV_SAFECLONE_BUFFERSIZE / sizeof(UAlignedMemory) + 1];
487     char chars[UCNV_ERROR_BUFFER_LENGTH];
488 
489     UConverter *clone;
490     uint8_t *subChars;
491     int32_t cloneSize, length8;
492 
493     /* Let the following functions check all arguments. */
494     cloneSize = sizeof(cloneBuffer);
495     clone = ucnv_safeClone(cnv, cloneBuffer, &cloneSize, err);
496     ucnv_setFromUCallBack(clone, UCNV_FROM_U_CALLBACK_STOP, NULL, NULL, NULL, err);
497     length8 = ucnv_fromUChars(clone, chars, (int32_t)sizeof(chars), s, length, err);
498     ucnv_close(clone);
499     if (U_FAILURE(*err)) {
500         return;
501     }
502 
503     if (cnv->sharedData->impl->writeSub == NULL
504 #if !UCONFIG_NO_LEGACY_CONVERSION
505         || (cnv->sharedData->staticData->conversionType == UCNV_MBCS &&
506          ucnv_MBCSGetType(cnv) != UCNV_EBCDIC_STATEFUL)
507 #endif
508     ) {
509         /* The converter is not stateful. Store the charset bytes as a fixed string. */
510         subChars = (uint8_t *)chars;
511     } else {
512         /*
513          * The converter has a non-default writeSub() function, indicating
514          * that it is stateful.
515          * Store the Unicode string for on-the-fly conversion for correct
516          * state handling.
517          */
518         if (length > UCNV_ERROR_BUFFER_LENGTH) {
519             /*
520              * Should not occur. The converter should output at least one byte
521              * per UChar, which means that ucnv_fromUChars() should catch all
522              * overflows.
523              */
524             *err = U_BUFFER_OVERFLOW_ERROR;
525             return;
526         }
527         subChars = (uint8_t *)s;
528         if (length < 0) {
529             length = u_strlen(s);
530         }
531         length8 = length * U_SIZEOF_UCHAR;
532     }
533 
534     /*
535      * For storing the substitution string, select either the small buffer inside
536      * UConverter or allocate a subChars buffer.
537      */
538     if (length8 > UCNV_MAX_SUBCHAR_LEN) {
539         /* Use a separate buffer for the string. Outside UConverter to not make it too large. */
540         if (cnv->subChars == (uint8_t *)cnv->subUChars) {
541             /* Allocate a new buffer for the string. */
542             cnv->subChars = (uint8_t *)uprv_malloc(UCNV_ERROR_BUFFER_LENGTH * U_SIZEOF_UCHAR);
543             if (cnv->subChars == NULL) {
544                 cnv->subChars = (uint8_t *)cnv->subUChars;
545                 *err = U_MEMORY_ALLOCATION_ERROR;
546                 return;
547             }
548             uprv_memset(cnv->subChars, 0, UCNV_ERROR_BUFFER_LENGTH * U_SIZEOF_UCHAR);
549         }
550     }
551 
552     /* Copy the substitution string into the UConverter or its subChars buffer. */
553     if (length8 == 0) {
554         cnv->subCharLen = 0;
555     } else {
556         uprv_memcpy(cnv->subChars, subChars, length8);
557         if (subChars == (uint8_t *)chars) {
558             cnv->subCharLen = (int8_t)length8;
559         } else /* subChars == s */ {
560             cnv->subCharLen = (int8_t)-length;
561         }
562     }
563 
564     /* See comment in ucnv_setSubstChars(). */
565     cnv->subChar1 = 0;
566 }
567 
568 /*resets the internal states of a converter
569  *goal : have the same behaviour than a freshly created converter
570  */
_reset(UConverter * converter,UConverterResetChoice choice,UBool callCallback)571 static void _reset(UConverter *converter, UConverterResetChoice choice,
572                    UBool callCallback) {
573     if(converter == NULL) {
574         return;
575     }
576 
577     if(callCallback) {
578         /* first, notify the callback functions that the converter is reset */
579         UErrorCode errorCode;
580 
581         if(choice<=UCNV_RESET_TO_UNICODE && converter->fromCharErrorBehaviour != UCNV_TO_U_DEFAULT_CALLBACK) {
582             UConverterToUnicodeArgs toUArgs = {
583                 sizeof(UConverterToUnicodeArgs),
584                 TRUE,
585                 NULL,
586                 NULL,
587                 NULL,
588                 NULL,
589                 NULL,
590                 NULL
591             };
592             toUArgs.converter = converter;
593             errorCode = U_ZERO_ERROR;
594             converter->fromCharErrorBehaviour(converter->toUContext, &toUArgs, NULL, 0, UCNV_RESET, &errorCode);
595         }
596         if(choice!=UCNV_RESET_TO_UNICODE && converter->fromUCharErrorBehaviour != UCNV_FROM_U_DEFAULT_CALLBACK) {
597             UConverterFromUnicodeArgs fromUArgs = {
598                 sizeof(UConverterFromUnicodeArgs),
599                 TRUE,
600                 NULL,
601                 NULL,
602                 NULL,
603                 NULL,
604                 NULL,
605                 NULL
606             };
607             fromUArgs.converter = converter;
608             errorCode = U_ZERO_ERROR;
609             converter->fromUCharErrorBehaviour(converter->fromUContext, &fromUArgs, NULL, 0, 0, UCNV_RESET, &errorCode);
610         }
611     }
612 
613     /* now reset the converter itself */
614     if(choice<=UCNV_RESET_TO_UNICODE) {
615         converter->toUnicodeStatus = converter->sharedData->toUnicodeStatus;
616         converter->mode = 0;
617         converter->toULength = 0;
618         converter->invalidCharLength = converter->UCharErrorBufferLength = 0;
619         converter->preToULength = 0;
620     }
621     if(choice!=UCNV_RESET_TO_UNICODE) {
622         converter->fromUnicodeStatus = 0;
623         converter->fromUChar32 = 0;
624         converter->invalidUCharLength = converter->charErrorBufferLength = 0;
625         converter->preFromUFirstCP = U_SENTINEL;
626         converter->preFromULength = 0;
627     }
628 
629     if (converter->sharedData->impl->reset != NULL) {
630         /* call the custom reset function */
631         converter->sharedData->impl->reset(converter, choice);
632     }
633 }
634 
635 U_CAPI void  U_EXPORT2
ucnv_reset(UConverter * converter)636 ucnv_reset(UConverter *converter)
637 {
638     _reset(converter, UCNV_RESET_BOTH, TRUE);
639 }
640 
641 U_CAPI void  U_EXPORT2
ucnv_resetToUnicode(UConverter * converter)642 ucnv_resetToUnicode(UConverter *converter)
643 {
644     _reset(converter, UCNV_RESET_TO_UNICODE, TRUE);
645 }
646 
647 U_CAPI void  U_EXPORT2
ucnv_resetFromUnicode(UConverter * converter)648 ucnv_resetFromUnicode(UConverter *converter)
649 {
650     _reset(converter, UCNV_RESET_FROM_UNICODE, TRUE);
651 }
652 
653 U_CAPI int8_t   U_EXPORT2
ucnv_getMaxCharSize(const UConverter * converter)654 ucnv_getMaxCharSize (const UConverter * converter)
655 {
656     return converter->maxBytesPerUChar;
657 }
658 
659 
660 U_CAPI int8_t   U_EXPORT2
ucnv_getMinCharSize(const UConverter * converter)661 ucnv_getMinCharSize (const UConverter * converter)
662 {
663     return converter->sharedData->staticData->minBytesPerChar;
664 }
665 
666 U_CAPI const char*   U_EXPORT2
ucnv_getName(const UConverter * converter,UErrorCode * err)667 ucnv_getName (const UConverter * converter, UErrorCode * err)
668 
669 {
670     if (U_FAILURE (*err))
671         return NULL;
672     if(converter->sharedData->impl->getName){
673         const char* temp= converter->sharedData->impl->getName(converter);
674         if(temp)
675             return temp;
676     }
677     return converter->sharedData->staticData->name;
678 }
679 
680 U_CAPI int32_t U_EXPORT2
ucnv_getCCSID(const UConverter * converter,UErrorCode * err)681 ucnv_getCCSID(const UConverter * converter,
682               UErrorCode * err)
683 {
684     int32_t ccsid;
685     if (U_FAILURE (*err))
686         return -1;
687 
688     ccsid = converter->sharedData->staticData->codepage;
689     if (ccsid == 0) {
690         /* Rare case. This is for cases like gb18030,
691         which doesn't have an IBM canonical name, but does have an IBM alias. */
692         const char *standardName = ucnv_getStandardName(ucnv_getName(converter, err), "IBM", err);
693         if (U_SUCCESS(*err) && standardName) {
694             const char *ccsidStr = uprv_strchr(standardName, '-');
695             if (ccsidStr) {
696                 ccsid = (int32_t)atol(ccsidStr+1);  /* +1 to skip '-' */
697             }
698         }
699     }
700     return ccsid;
701 }
702 
703 
704 U_CAPI UConverterPlatform   U_EXPORT2
ucnv_getPlatform(const UConverter * converter,UErrorCode * err)705 ucnv_getPlatform (const UConverter * converter,
706                                       UErrorCode * err)
707 {
708     if (U_FAILURE (*err))
709         return UCNV_UNKNOWN;
710 
711     return (UConverterPlatform)converter->sharedData->staticData->platform;
712 }
713 
714 U_CAPI void U_EXPORT2
ucnv_getToUCallBack(const UConverter * converter,UConverterToUCallback * action,const void ** context)715     ucnv_getToUCallBack (const UConverter * converter,
716                          UConverterToUCallback *action,
717                          const void **context)
718 {
719     *action = converter->fromCharErrorBehaviour;
720     *context = converter->toUContext;
721 }
722 
723 U_CAPI void U_EXPORT2
ucnv_getFromUCallBack(const UConverter * converter,UConverterFromUCallback * action,const void ** context)724     ucnv_getFromUCallBack (const UConverter * converter,
725                            UConverterFromUCallback *action,
726                            const void **context)
727 {
728     *action = converter->fromUCharErrorBehaviour;
729     *context = converter->fromUContext;
730 }
731 
732 U_CAPI void    U_EXPORT2
ucnv_setToUCallBack(UConverter * converter,UConverterToUCallback newAction,const void * newContext,UConverterToUCallback * oldAction,const void ** oldContext,UErrorCode * err)733 ucnv_setToUCallBack (UConverter * converter,
734                             UConverterToUCallback newAction,
735                             const void* newContext,
736                             UConverterToUCallback *oldAction,
737                             const void** oldContext,
738                             UErrorCode * err)
739 {
740     if (U_FAILURE (*err))
741         return;
742     if (oldAction) *oldAction = converter->fromCharErrorBehaviour;
743     converter->fromCharErrorBehaviour = newAction;
744     if (oldContext) *oldContext = converter->toUContext;
745     converter->toUContext = newContext;
746 }
747 
748 U_CAPI void  U_EXPORT2
ucnv_setFromUCallBack(UConverter * converter,UConverterFromUCallback newAction,const void * newContext,UConverterFromUCallback * oldAction,const void ** oldContext,UErrorCode * err)749 ucnv_setFromUCallBack (UConverter * converter,
750                             UConverterFromUCallback newAction,
751                             const void* newContext,
752                             UConverterFromUCallback *oldAction,
753                             const void** oldContext,
754                             UErrorCode * err)
755 {
756     if (U_FAILURE (*err))
757         return;
758     if (oldAction) *oldAction = converter->fromUCharErrorBehaviour;
759     converter->fromUCharErrorBehaviour = newAction;
760     if (oldContext) *oldContext = converter->fromUContext;
761     converter->fromUContext = newContext;
762 }
763 
764 static void
_updateOffsets(int32_t * offsets,int32_t length,int32_t sourceIndex,int32_t errorInputLength)765 _updateOffsets(int32_t *offsets, int32_t length,
766                int32_t sourceIndex, int32_t errorInputLength) {
767     int32_t *limit;
768     int32_t delta, offset;
769 
770     if(sourceIndex>=0) {
771         /*
772          * adjust each offset by adding the previous sourceIndex
773          * minus the length of the input sequence that caused an
774          * error, if any
775          */
776         delta=sourceIndex-errorInputLength;
777     } else {
778         /*
779          * set each offset to -1 because this conversion function
780          * does not handle offsets
781          */
782         delta=-1;
783     }
784 
785     limit=offsets+length;
786     if(delta==0) {
787         /* most common case, nothing to do */
788     } else if(delta>0) {
789         /* add the delta to each offset (but not if the offset is <0) */
790         while(offsets<limit) {
791             offset=*offsets;
792             if(offset>=0) {
793                 *offsets=offset+delta;
794             }
795             ++offsets;
796         }
797     } else /* delta<0 */ {
798         /*
799          * set each offset to -1 because this conversion function
800          * does not handle offsets
801          * or the error input sequence started in a previous buffer
802          */
803         while(offsets<limit) {
804             *offsets++=-1;
805         }
806     }
807 }
808 
809 /* ucnv_fromUnicode --------------------------------------------------------- */
810 
811 /*
812  * Implementation note for m:n conversions
813  *
814  * While collecting source units to find the longest match for m:n conversion,
815  * some source units may need to be stored for a partial match.
816  * When a second buffer does not yield a match on all of the previously stored
817  * source units, then they must be "replayed", i.e., fed back into the converter.
818  *
819  * The code relies on the fact that replaying will not nest -
820  * converting a replay buffer will not result in a replay.
821  * This is because a replay is necessary only after the _continuation_ of a
822  * partial match failed, but a replay buffer is converted as a whole.
823  * It may result in some of its units being stored again for a partial match,
824  * but there will not be a continuation _during_ the replay which could fail.
825  *
826  * It is conceivable that a callback function could call the converter
827  * recursively in a way that causes another replay to be stored, but that
828  * would be an error in the callback function.
829  * Such violations will cause assertion failures in a debug build,
830  * and wrong output, but they will not cause a crash.
831  */
832 
833 static void
_fromUnicodeWithCallback(UConverterFromUnicodeArgs * pArgs,UErrorCode * err)834 _fromUnicodeWithCallback(UConverterFromUnicodeArgs *pArgs, UErrorCode *err) {
835     UConverterFromUnicode fromUnicode;
836     UConverter *cnv;
837     const UChar *s;
838     char *t;
839     int32_t *offsets;
840     int32_t sourceIndex;
841     int32_t errorInputLength;
842     UBool converterSawEndOfInput, calledCallback;
843 
844     /* variables for m:n conversion */
845     UChar replay[UCNV_EXT_MAX_UCHARS];
846     const UChar *realSource, *realSourceLimit;
847     int32_t realSourceIndex;
848     UBool realFlush;
849 
850     cnv=pArgs->converter;
851     s=pArgs->source;
852     t=pArgs->target;
853     offsets=pArgs->offsets;
854 
855     /* get the converter implementation function */
856     sourceIndex=0;
857     if(offsets==NULL) {
858         fromUnicode=cnv->sharedData->impl->fromUnicode;
859     } else {
860         fromUnicode=cnv->sharedData->impl->fromUnicodeWithOffsets;
861         if(fromUnicode==NULL) {
862             /* there is no WithOffsets implementation */
863             fromUnicode=cnv->sharedData->impl->fromUnicode;
864             /* we will write -1 for each offset */
865             sourceIndex=-1;
866         }
867     }
868 
869     if(cnv->preFromULength>=0) {
870         /* normal mode */
871         realSource=NULL;
872 
873         /* avoid compiler warnings - not otherwise necessary, and the values do not matter */
874         realSourceLimit=NULL;
875         realFlush=FALSE;
876         realSourceIndex=0;
877     } else {
878         /*
879          * Previous m:n conversion stored source units from a partial match
880          * and failed to consume all of them.
881          * We need to "replay" them from a temporary buffer and convert them first.
882          */
883         realSource=pArgs->source;
884         realSourceLimit=pArgs->sourceLimit;
885         realFlush=pArgs->flush;
886         realSourceIndex=sourceIndex;
887 
888         uprv_memcpy(replay, cnv->preFromU, -cnv->preFromULength*U_SIZEOF_UCHAR);
889         pArgs->source=replay;
890         pArgs->sourceLimit=replay-cnv->preFromULength;
891         pArgs->flush=FALSE;
892         sourceIndex=-1;
893 
894         cnv->preFromULength=0;
895     }
896 
897     /*
898      * loop for conversion and error handling
899      *
900      * loop {
901      *   convert
902      *   loop {
903      *     update offsets
904      *     handle end of input
905      *     handle errors/call callback
906      *   }
907      * }
908      */
909     for(;;) {
910         if(U_SUCCESS(*err)) {
911             /* convert */
912             fromUnicode(pArgs, err);
913 
914             /*
915              * set a flag for whether the converter
916              * successfully processed the end of the input
917              *
918              * need not check cnv->preFromULength==0 because a replay (<0) will cause
919              * s<sourceLimit before converterSawEndOfInput is checked
920              */
921             converterSawEndOfInput=
922                 (UBool)(U_SUCCESS(*err) &&
923                         pArgs->flush && pArgs->source==pArgs->sourceLimit &&
924                         cnv->fromUChar32==0);
925         } else {
926             /* handle error from ucnv_convertEx() */
927             converterSawEndOfInput=FALSE;
928         }
929 
930         /* no callback called yet for this iteration */
931         calledCallback=FALSE;
932 
933         /* no sourceIndex adjustment for conversion, only for callback output */
934         errorInputLength=0;
935 
936         /*
937          * loop for offsets and error handling
938          *
939          * iterates at most 3 times:
940          * 1. to clean up after the conversion function
941          * 2. after the callback
942          * 3. after the callback again if there was truncated input
943          */
944         for(;;) {
945             /* update offsets if we write any */
946             if(offsets!=NULL) {
947                 int32_t length=(int32_t)(pArgs->target-t);
948                 if(length>0) {
949                     _updateOffsets(offsets, length, sourceIndex, errorInputLength);
950 
951                     /*
952                      * if a converter handles offsets and updates the offsets
953                      * pointer at the end, then pArgs->offset should not change
954                      * here;
955                      * however, some converters do not handle offsets at all
956                      * (sourceIndex<0) or may not update the offsets pointer
957                      */
958                     pArgs->offsets=offsets+=length;
959                 }
960 
961                 if(sourceIndex>=0) {
962                     sourceIndex+=(int32_t)(pArgs->source-s);
963                 }
964             }
965 
966             if(cnv->preFromULength<0) {
967                 /*
968                  * switch the source to new replay units (cannot occur while replaying)
969                  * after offset handling and before end-of-input and callback handling
970                  */
971                 if(realSource==NULL) {
972                     realSource=pArgs->source;
973                     realSourceLimit=pArgs->sourceLimit;
974                     realFlush=pArgs->flush;
975                     realSourceIndex=sourceIndex;
976 
977                     uprv_memcpy(replay, cnv->preFromU, -cnv->preFromULength*U_SIZEOF_UCHAR);
978                     pArgs->source=replay;
979                     pArgs->sourceLimit=replay-cnv->preFromULength;
980                     pArgs->flush=FALSE;
981                     if((sourceIndex+=cnv->preFromULength)<0) {
982                         sourceIndex=-1;
983                     }
984 
985                     cnv->preFromULength=0;
986                 } else {
987                     /* see implementation note before _fromUnicodeWithCallback() */
988                     U_ASSERT(realSource==NULL);
989                     *err=U_INTERNAL_PROGRAM_ERROR;
990                 }
991             }
992 
993             /* update pointers */
994             s=pArgs->source;
995             t=pArgs->target;
996 
997             if(U_SUCCESS(*err)) {
998                 if(s<pArgs->sourceLimit) {
999                     /*
1000                      * continue with the conversion loop while there is still input left
1001                      * (continue converting by breaking out of only the inner loop)
1002                      */
1003                     break;
1004                 } else if(realSource!=NULL) {
1005                     /* switch back from replaying to the real source and continue */
1006                     pArgs->source=realSource;
1007                     pArgs->sourceLimit=realSourceLimit;
1008                     pArgs->flush=realFlush;
1009                     sourceIndex=realSourceIndex;
1010 
1011                     realSource=NULL;
1012                     break;
1013                 } else if(pArgs->flush && cnv->fromUChar32!=0) {
1014                     /*
1015                      * the entire input stream is consumed
1016                      * and there is a partial, truncated input sequence left
1017                      */
1018 
1019                     /* inject an error and continue with callback handling */
1020                     *err=U_TRUNCATED_CHAR_FOUND;
1021                     calledCallback=FALSE; /* new error condition */
1022                 } else {
1023                     /* input consumed */
1024                     if(pArgs->flush) {
1025                         /*
1026                          * return to the conversion loop once more if the flush
1027                          * flag is set and the conversion function has not
1028                          * successfully processed the end of the input yet
1029                          *
1030                          * (continue converting by breaking out of only the inner loop)
1031                          */
1032                         if(!converterSawEndOfInput) {
1033                             break;
1034                         }
1035 
1036                         /* reset the converter without calling the callback function */
1037                         _reset(cnv, UCNV_RESET_FROM_UNICODE, FALSE);
1038                     }
1039 
1040                     /* done successfully */
1041                     return;
1042                 }
1043             }
1044 
1045             /* U_FAILURE(*err) */
1046             {
1047                 UErrorCode e;
1048 
1049                 if( calledCallback ||
1050                     (e=*err)==U_BUFFER_OVERFLOW_ERROR ||
1051                     (e!=U_INVALID_CHAR_FOUND &&
1052                      e!=U_ILLEGAL_CHAR_FOUND &&
1053                      e!=U_TRUNCATED_CHAR_FOUND)
1054                 ) {
1055                     /*
1056                      * the callback did not or cannot resolve the error:
1057                      * set output pointers and return
1058                      *
1059                      * the check for buffer overflow is redundant but it is
1060                      * a high-runner case and hopefully documents the intent
1061                      * well
1062                      *
1063                      * if we were replaying, then the replay buffer must be
1064                      * copied back into the UConverter
1065                      * and the real arguments must be restored
1066                      */
1067                     if(realSource!=NULL) {
1068                         int32_t length;
1069 
1070                         U_ASSERT(cnv->preFromULength==0);
1071 
1072                         length=(int32_t)(pArgs->sourceLimit-pArgs->source);
1073                         if(length>0) {
1074                             uprv_memcpy(cnv->preFromU, pArgs->source, length*U_SIZEOF_UCHAR);
1075                             cnv->preFromULength=(int8_t)-length;
1076                         }
1077 
1078                         pArgs->source=realSource;
1079                         pArgs->sourceLimit=realSourceLimit;
1080                         pArgs->flush=realFlush;
1081                     }
1082 
1083                     return;
1084                 }
1085             }
1086 
1087             /* callback handling */
1088             {
1089                 UChar32 codePoint;
1090 
1091                 /* get and write the code point */
1092                 codePoint=cnv->fromUChar32;
1093                 errorInputLength=0;
1094                 U16_APPEND_UNSAFE(cnv->invalidUCharBuffer, errorInputLength, codePoint);
1095                 cnv->invalidUCharLength=(int8_t)errorInputLength;
1096 
1097                 /* set the converter state to deal with the next character */
1098                 cnv->fromUChar32=0;
1099 
1100                 /* call the callback function */
1101                 cnv->fromUCharErrorBehaviour(cnv->fromUContext, pArgs,
1102                     cnv->invalidUCharBuffer, errorInputLength, codePoint,
1103                     *err==U_INVALID_CHAR_FOUND ? UCNV_UNASSIGNED : UCNV_ILLEGAL,
1104                     err);
1105             }
1106 
1107             /*
1108              * loop back to the offset handling
1109              *
1110              * this flag will indicate after offset handling
1111              * that a callback was called;
1112              * if the callback did not resolve the error, then we return
1113              */
1114             calledCallback=TRUE;
1115         }
1116     }
1117 }
1118 
1119 /*
1120  * Output the fromUnicode overflow buffer.
1121  * Call this function if(cnv->charErrorBufferLength>0).
1122  * @return TRUE if overflow
1123  */
1124 static UBool
ucnv_outputOverflowFromUnicode(UConverter * cnv,char ** target,const char * targetLimit,int32_t ** pOffsets,UErrorCode * err)1125 ucnv_outputOverflowFromUnicode(UConverter *cnv,
1126                                char **target, const char *targetLimit,
1127                                int32_t **pOffsets,
1128                                UErrorCode *err) {
1129     int32_t *offsets;
1130     char *overflow, *t;
1131     int32_t i, length;
1132 
1133     t=*target;
1134     if(pOffsets!=NULL) {
1135         offsets=*pOffsets;
1136     } else {
1137         offsets=NULL;
1138     }
1139 
1140     overflow=(char *)cnv->charErrorBuffer;
1141     length=cnv->charErrorBufferLength;
1142     i=0;
1143     while(i<length) {
1144         if(t==targetLimit) {
1145             /* the overflow buffer contains too much, keep the rest */
1146             int32_t j=0;
1147 
1148             do {
1149                 overflow[j++]=overflow[i++];
1150             } while(i<length);
1151 
1152             cnv->charErrorBufferLength=(int8_t)j;
1153             *target=t;
1154             if(offsets!=NULL) {
1155                 *pOffsets=offsets;
1156             }
1157             *err=U_BUFFER_OVERFLOW_ERROR;
1158             return TRUE;
1159         }
1160 
1161         /* copy the overflow contents to the target */
1162         *t++=overflow[i++];
1163         if(offsets!=NULL) {
1164             *offsets++=-1; /* no source index available for old output */
1165         }
1166     }
1167 
1168     /* the overflow buffer is completely copied to the target */
1169     cnv->charErrorBufferLength=0;
1170     *target=t;
1171     if(offsets!=NULL) {
1172         *pOffsets=offsets;
1173     }
1174     return FALSE;
1175 }
1176 
1177 U_CAPI void U_EXPORT2
ucnv_fromUnicode(UConverter * cnv,char ** target,const char * targetLimit,const UChar ** source,const UChar * sourceLimit,int32_t * offsets,UBool flush,UErrorCode * err)1178 ucnv_fromUnicode(UConverter *cnv,
1179                  char **target, const char *targetLimit,
1180                  const UChar **source, const UChar *sourceLimit,
1181                  int32_t *offsets,
1182                  UBool flush,
1183                  UErrorCode *err) {
1184     UConverterFromUnicodeArgs args;
1185     const UChar *s;
1186     char *t;
1187 
1188     /* check parameters */
1189     if(err==NULL || U_FAILURE(*err)) {
1190         return;
1191     }
1192 
1193     if(cnv==NULL || target==NULL || source==NULL) {
1194         *err=U_ILLEGAL_ARGUMENT_ERROR;
1195         return;
1196     }
1197 
1198     s=*source;
1199     t=*target;
1200 
1201     if ((const void *)U_MAX_PTR(sourceLimit) == (const void *)sourceLimit) {
1202         /*
1203         Prevent code from going into an infinite loop in case we do hit this
1204         limit. The limit pointer is expected to be on a UChar * boundary.
1205         This also prevents the next argument check from failing.
1206         */
1207         sourceLimit = (const UChar *)(((const char *)sourceLimit) - 1);
1208     }
1209 
1210     /*
1211      * All these conditions should never happen.
1212      *
1213      * 1) Make sure that the limits are >= to the address source or target
1214      *
1215      * 2) Make sure that the buffer sizes do not exceed the number range for
1216      * int32_t because some functions use the size (in units or bytes)
1217      * rather than comparing pointers, and because offsets are int32_t values.
1218      *
1219      * size_t is guaranteed to be unsigned and large enough for the job.
1220      *
1221      * Return with an error instead of adjusting the limits because we would
1222      * not be able to maintain the semantics that either the source must be
1223      * consumed or the target filled (unless an error occurs).
1224      * An adjustment would be targetLimit=t+0x7fffffff; for example.
1225      *
1226      * 3) Make sure that the user didn't incorrectly cast a UChar * pointer
1227      * to a char * pointer and provide an incomplete UChar code unit.
1228      */
1229     if (sourceLimit<s || targetLimit<t ||
1230         ((size_t)(sourceLimit-s)>(size_t)0x3fffffff && sourceLimit>s) ||
1231         ((size_t)(targetLimit-t)>(size_t)0x7fffffff && targetLimit>t) ||
1232         (((const char *)sourceLimit-(const char *)s) & 1) != 0)
1233     {
1234         *err=U_ILLEGAL_ARGUMENT_ERROR;
1235         return;
1236     }
1237 
1238     /* output the target overflow buffer */
1239     if( cnv->charErrorBufferLength>0 &&
1240         ucnv_outputOverflowFromUnicode(cnv, target, targetLimit, &offsets, err)
1241     ) {
1242         /* U_BUFFER_OVERFLOW_ERROR */
1243         return;
1244     }
1245     /* *target may have moved, therefore stop using t */
1246 
1247     if(!flush && s==sourceLimit && cnv->preFromULength>=0) {
1248         /* the overflow buffer is emptied and there is no new input: we are done */
1249         return;
1250     }
1251 
1252     /*
1253      * Do not simply return with a buffer overflow error if
1254      * !flush && t==targetLimit
1255      * because it is possible that the source will not generate any output.
1256      * For example, the skip callback may be called;
1257      * it does not output anything.
1258      */
1259 
1260     /* prepare the converter arguments */
1261     args.converter=cnv;
1262     args.flush=flush;
1263     args.offsets=offsets;
1264     args.source=s;
1265     args.sourceLimit=sourceLimit;
1266     args.target=*target;
1267     args.targetLimit=targetLimit;
1268     args.size=sizeof(args);
1269 
1270     _fromUnicodeWithCallback(&args, err);
1271 
1272     *source=args.source;
1273     *target=args.target;
1274 }
1275 
1276 /* ucnv_toUnicode() --------------------------------------------------------- */
1277 
1278 static void
_toUnicodeWithCallback(UConverterToUnicodeArgs * pArgs,UErrorCode * err)1279 _toUnicodeWithCallback(UConverterToUnicodeArgs *pArgs, UErrorCode *err) {
1280     UConverterToUnicode toUnicode;
1281     UConverter *cnv;
1282     const char *s;
1283     UChar *t;
1284     int32_t *offsets;
1285     int32_t sourceIndex;
1286     int32_t errorInputLength;
1287     UBool converterSawEndOfInput, calledCallback;
1288 
1289     /* variables for m:n conversion */
1290     char replay[UCNV_EXT_MAX_BYTES];
1291     const char *realSource, *realSourceLimit;
1292     int32_t realSourceIndex;
1293     UBool realFlush;
1294 
1295     cnv=pArgs->converter;
1296     s=pArgs->source;
1297     t=pArgs->target;
1298     offsets=pArgs->offsets;
1299 
1300     /* get the converter implementation function */
1301     sourceIndex=0;
1302     if(offsets==NULL) {
1303         toUnicode=cnv->sharedData->impl->toUnicode;
1304     } else {
1305         toUnicode=cnv->sharedData->impl->toUnicodeWithOffsets;
1306         if(toUnicode==NULL) {
1307             /* there is no WithOffsets implementation */
1308             toUnicode=cnv->sharedData->impl->toUnicode;
1309             /* we will write -1 for each offset */
1310             sourceIndex=-1;
1311         }
1312     }
1313 
1314     if(cnv->preToULength>=0) {
1315         /* normal mode */
1316         realSource=NULL;
1317 
1318         /* avoid compiler warnings - not otherwise necessary, and the values do not matter */
1319         realSourceLimit=NULL;
1320         realFlush=FALSE;
1321         realSourceIndex=0;
1322     } else {
1323         /*
1324          * Previous m:n conversion stored source units from a partial match
1325          * and failed to consume all of them.
1326          * We need to "replay" them from a temporary buffer and convert them first.
1327          */
1328         realSource=pArgs->source;
1329         realSourceLimit=pArgs->sourceLimit;
1330         realFlush=pArgs->flush;
1331         realSourceIndex=sourceIndex;
1332 
1333         uprv_memcpy(replay, cnv->preToU, -cnv->preToULength);
1334         pArgs->source=replay;
1335         pArgs->sourceLimit=replay-cnv->preToULength;
1336         pArgs->flush=FALSE;
1337         sourceIndex=-1;
1338 
1339         cnv->preToULength=0;
1340     }
1341 
1342     /*
1343      * loop for conversion and error handling
1344      *
1345      * loop {
1346      *   convert
1347      *   loop {
1348      *     update offsets
1349      *     handle end of input
1350      *     handle errors/call callback
1351      *   }
1352      * }
1353      */
1354     for(;;) {
1355         if(U_SUCCESS(*err)) {
1356             /* convert */
1357             toUnicode(pArgs, err);
1358 
1359             /*
1360              * set a flag for whether the converter
1361              * successfully processed the end of the input
1362              *
1363              * need not check cnv->preToULength==0 because a replay (<0) will cause
1364              * s<sourceLimit before converterSawEndOfInput is checked
1365              */
1366             converterSawEndOfInput=
1367                 (UBool)(U_SUCCESS(*err) &&
1368                         pArgs->flush && pArgs->source==pArgs->sourceLimit &&
1369                         cnv->toULength==0);
1370         } else {
1371             /* handle error from getNextUChar() or ucnv_convertEx() */
1372             converterSawEndOfInput=FALSE;
1373         }
1374 
1375         /* no callback called yet for this iteration */
1376         calledCallback=FALSE;
1377 
1378         /* no sourceIndex adjustment for conversion, only for callback output */
1379         errorInputLength=0;
1380 
1381         /*
1382          * loop for offsets and error handling
1383          *
1384          * iterates at most 3 times:
1385          * 1. to clean up after the conversion function
1386          * 2. after the callback
1387          * 3. after the callback again if there was truncated input
1388          */
1389         for(;;) {
1390             /* update offsets if we write any */
1391             if(offsets!=NULL) {
1392                 int32_t length=(int32_t)(pArgs->target-t);
1393                 if(length>0) {
1394                     _updateOffsets(offsets, length, sourceIndex, errorInputLength);
1395 
1396                     /*
1397                      * if a converter handles offsets and updates the offsets
1398                      * pointer at the end, then pArgs->offset should not change
1399                      * here;
1400                      * however, some converters do not handle offsets at all
1401                      * (sourceIndex<0) or may not update the offsets pointer
1402                      */
1403                     pArgs->offsets=offsets+=length;
1404                 }
1405 
1406                 if(sourceIndex>=0) {
1407                     sourceIndex+=(int32_t)(pArgs->source-s);
1408                 }
1409             }
1410 
1411             if(cnv->preToULength<0) {
1412                 /*
1413                  * switch the source to new replay units (cannot occur while replaying)
1414                  * after offset handling and before end-of-input and callback handling
1415                  */
1416                 if(realSource==NULL) {
1417                     realSource=pArgs->source;
1418                     realSourceLimit=pArgs->sourceLimit;
1419                     realFlush=pArgs->flush;
1420                     realSourceIndex=sourceIndex;
1421 
1422                     uprv_memcpy(replay, cnv->preToU, -cnv->preToULength);
1423                     pArgs->source=replay;
1424                     pArgs->sourceLimit=replay-cnv->preToULength;
1425                     pArgs->flush=FALSE;
1426                     if((sourceIndex+=cnv->preToULength)<0) {
1427                         sourceIndex=-1;
1428                     }
1429 
1430                     cnv->preToULength=0;
1431                 } else {
1432                     /* see implementation note before _fromUnicodeWithCallback() */
1433                     U_ASSERT(realSource==NULL);
1434                     *err=U_INTERNAL_PROGRAM_ERROR;
1435                 }
1436             }
1437 
1438             /* update pointers */
1439             s=pArgs->source;
1440             t=pArgs->target;
1441 
1442             if(U_SUCCESS(*err)) {
1443                 if(s<pArgs->sourceLimit) {
1444                     /*
1445                      * continue with the conversion loop while there is still input left
1446                      * (continue converting by breaking out of only the inner loop)
1447                      */
1448                     break;
1449                 } else if(realSource!=NULL) {
1450                     /* switch back from replaying to the real source and continue */
1451                     pArgs->source=realSource;
1452                     pArgs->sourceLimit=realSourceLimit;
1453                     pArgs->flush=realFlush;
1454                     sourceIndex=realSourceIndex;
1455 
1456                     realSource=NULL;
1457                     break;
1458                 } else if(pArgs->flush && cnv->toULength>0) {
1459                     /*
1460                      * the entire input stream is consumed
1461                      * and there is a partial, truncated input sequence left
1462                      */
1463 
1464                     /* inject an error and continue with callback handling */
1465                     *err=U_TRUNCATED_CHAR_FOUND;
1466                     calledCallback=FALSE; /* new error condition */
1467                 } else {
1468                     /* input consumed */
1469                     if(pArgs->flush) {
1470                         /*
1471                          * return to the conversion loop once more if the flush
1472                          * flag is set and the conversion function has not
1473                          * successfully processed the end of the input yet
1474                          *
1475                          * (continue converting by breaking out of only the inner loop)
1476                          */
1477                         if(!converterSawEndOfInput) {
1478                             break;
1479                         }
1480 
1481                         /* reset the converter without calling the callback function */
1482                         _reset(cnv, UCNV_RESET_TO_UNICODE, FALSE);
1483                     }
1484 
1485                     /* done successfully */
1486                     return;
1487                 }
1488             }
1489 
1490             /* U_FAILURE(*err) */
1491             {
1492                 UErrorCode e;
1493 
1494                 if( calledCallback ||
1495                     (e=*err)==U_BUFFER_OVERFLOW_ERROR ||
1496                     (e!=U_INVALID_CHAR_FOUND &&
1497                      e!=U_ILLEGAL_CHAR_FOUND &&
1498                      e!=U_TRUNCATED_CHAR_FOUND &&
1499                      e!=U_ILLEGAL_ESCAPE_SEQUENCE &&
1500                      e!=U_UNSUPPORTED_ESCAPE_SEQUENCE)
1501                 ) {
1502                     /*
1503                      * the callback did not or cannot resolve the error:
1504                      * set output pointers and return
1505                      *
1506                      * the check for buffer overflow is redundant but it is
1507                      * a high-runner case and hopefully documents the intent
1508                      * well
1509                      *
1510                      * if we were replaying, then the replay buffer must be
1511                      * copied back into the UConverter
1512                      * and the real arguments must be restored
1513                      */
1514                     if(realSource!=NULL) {
1515                         int32_t length;
1516 
1517                         U_ASSERT(cnv->preToULength==0);
1518 
1519                         length=(int32_t)(pArgs->sourceLimit-pArgs->source);
1520                         if(length>0) {
1521                             uprv_memcpy(cnv->preToU, pArgs->source, length);
1522                             cnv->preToULength=(int8_t)-length;
1523                         }
1524 
1525                         pArgs->source=realSource;
1526                         pArgs->sourceLimit=realSourceLimit;
1527                         pArgs->flush=realFlush;
1528                     }
1529 
1530                     return;
1531                 }
1532             }
1533 
1534             /* copy toUBytes[] to invalidCharBuffer[] */
1535             errorInputLength=cnv->invalidCharLength=cnv->toULength;
1536             if(errorInputLength>0) {
1537                 uprv_memcpy(cnv->invalidCharBuffer, cnv->toUBytes, errorInputLength);
1538             }
1539 
1540             /* set the converter state to deal with the next character */
1541             cnv->toULength=0;
1542 
1543             /* call the callback function */
1544             if(cnv->toUCallbackReason==UCNV_ILLEGAL && *err==U_INVALID_CHAR_FOUND) {
1545                 cnv->toUCallbackReason = UCNV_UNASSIGNED;
1546             }
1547             cnv->fromCharErrorBehaviour(cnv->toUContext, pArgs,
1548                 cnv->invalidCharBuffer, errorInputLength,
1549                 cnv->toUCallbackReason,
1550                 err);
1551             cnv->toUCallbackReason = UCNV_ILLEGAL; /* reset to default value */
1552 
1553             /*
1554              * loop back to the offset handling
1555              *
1556              * this flag will indicate after offset handling
1557              * that a callback was called;
1558              * if the callback did not resolve the error, then we return
1559              */
1560             calledCallback=TRUE;
1561         }
1562     }
1563 }
1564 
1565 /*
1566  * Output the toUnicode overflow buffer.
1567  * Call this function if(cnv->UCharErrorBufferLength>0).
1568  * @return TRUE if overflow
1569  */
1570 static UBool
ucnv_outputOverflowToUnicode(UConverter * cnv,UChar ** target,const UChar * targetLimit,int32_t ** pOffsets,UErrorCode * err)1571 ucnv_outputOverflowToUnicode(UConverter *cnv,
1572                              UChar **target, const UChar *targetLimit,
1573                              int32_t **pOffsets,
1574                              UErrorCode *err) {
1575     int32_t *offsets;
1576     UChar *overflow, *t;
1577     int32_t i, length;
1578 
1579     t=*target;
1580     if(pOffsets!=NULL) {
1581         offsets=*pOffsets;
1582     } else {
1583         offsets=NULL;
1584     }
1585 
1586     overflow=cnv->UCharErrorBuffer;
1587     length=cnv->UCharErrorBufferLength;
1588     i=0;
1589     while(i<length) {
1590         if(t==targetLimit) {
1591             /* the overflow buffer contains too much, keep the rest */
1592             int32_t j=0;
1593 
1594             do {
1595                 overflow[j++]=overflow[i++];
1596             } while(i<length);
1597 
1598             cnv->UCharErrorBufferLength=(int8_t)j;
1599             *target=t;
1600             if(offsets!=NULL) {
1601                 *pOffsets=offsets;
1602             }
1603             *err=U_BUFFER_OVERFLOW_ERROR;
1604             return TRUE;
1605         }
1606 
1607         /* copy the overflow contents to the target */
1608         *t++=overflow[i++];
1609         if(offsets!=NULL) {
1610             *offsets++=-1; /* no source index available for old output */
1611         }
1612     }
1613 
1614     /* the overflow buffer is completely copied to the target */
1615     cnv->UCharErrorBufferLength=0;
1616     *target=t;
1617     if(offsets!=NULL) {
1618         *pOffsets=offsets;
1619     }
1620     return FALSE;
1621 }
1622 
1623 U_CAPI void U_EXPORT2
ucnv_toUnicode(UConverter * cnv,UChar ** target,const UChar * targetLimit,const char ** source,const char * sourceLimit,int32_t * offsets,UBool flush,UErrorCode * err)1624 ucnv_toUnicode(UConverter *cnv,
1625                UChar **target, const UChar *targetLimit,
1626                const char **source, const char *sourceLimit,
1627                int32_t *offsets,
1628                UBool flush,
1629                UErrorCode *err) {
1630     UConverterToUnicodeArgs args;
1631     const char *s;
1632     UChar *t;
1633 
1634     /* check parameters */
1635     if(err==NULL || U_FAILURE(*err)) {
1636         return;
1637     }
1638 
1639     if(cnv==NULL || target==NULL || source==NULL) {
1640         *err=U_ILLEGAL_ARGUMENT_ERROR;
1641         return;
1642     }
1643 
1644     s=*source;
1645     t=*target;
1646 
1647     if ((const void *)U_MAX_PTR(targetLimit) == (const void *)targetLimit) {
1648         /*
1649         Prevent code from going into an infinite loop in case we do hit this
1650         limit. The limit pointer is expected to be on a UChar * boundary.
1651         This also prevents the next argument check from failing.
1652         */
1653         targetLimit = (const UChar *)(((const char *)targetLimit) - 1);
1654     }
1655 
1656     /*
1657      * All these conditions should never happen.
1658      *
1659      * 1) Make sure that the limits are >= to the address source or target
1660      *
1661      * 2) Make sure that the buffer sizes do not exceed the number range for
1662      * int32_t because some functions use the size (in units or bytes)
1663      * rather than comparing pointers, and because offsets are int32_t values.
1664      *
1665      * size_t is guaranteed to be unsigned and large enough for the job.
1666      *
1667      * Return with an error instead of adjusting the limits because we would
1668      * not be able to maintain the semantics that either the source must be
1669      * consumed or the target filled (unless an error occurs).
1670      * An adjustment would be sourceLimit=t+0x7fffffff; for example.
1671      *
1672      * 3) Make sure that the user didn't incorrectly cast a UChar * pointer
1673      * to a char * pointer and provide an incomplete UChar code unit.
1674      */
1675     if (sourceLimit<s || targetLimit<t ||
1676         ((size_t)(sourceLimit-s)>(size_t)0x7fffffff && sourceLimit>s) ||
1677         ((size_t)(targetLimit-t)>(size_t)0x3fffffff && targetLimit>t) ||
1678         (((const char *)targetLimit-(const char *)t) & 1) != 0
1679     ) {
1680         *err=U_ILLEGAL_ARGUMENT_ERROR;
1681         return;
1682     }
1683 
1684     /* output the target overflow buffer */
1685     if( cnv->UCharErrorBufferLength>0 &&
1686         ucnv_outputOverflowToUnicode(cnv, target, targetLimit, &offsets, err)
1687     ) {
1688         /* U_BUFFER_OVERFLOW_ERROR */
1689         return;
1690     }
1691     /* *target may have moved, therefore stop using t */
1692 
1693     if(!flush && s==sourceLimit && cnv->preToULength>=0) {
1694         /* the overflow buffer is emptied and there is no new input: we are done */
1695         return;
1696     }
1697 
1698     /*
1699      * Do not simply return with a buffer overflow error if
1700      * !flush && t==targetLimit
1701      * because it is possible that the source will not generate any output.
1702      * For example, the skip callback may be called;
1703      * it does not output anything.
1704      */
1705 
1706     /* prepare the converter arguments */
1707     args.converter=cnv;
1708     args.flush=flush;
1709     args.offsets=offsets;
1710     args.source=s;
1711     args.sourceLimit=sourceLimit;
1712     args.target=*target;
1713     args.targetLimit=targetLimit;
1714     args.size=sizeof(args);
1715 
1716     _toUnicodeWithCallback(&args, err);
1717 
1718     *source=args.source;
1719     *target=args.target;
1720 }
1721 
1722 /* ucnv_to/fromUChars() ----------------------------------------------------- */
1723 
1724 U_CAPI int32_t U_EXPORT2
ucnv_fromUChars(UConverter * cnv,char * dest,int32_t destCapacity,const UChar * src,int32_t srcLength,UErrorCode * pErrorCode)1725 ucnv_fromUChars(UConverter *cnv,
1726                 char *dest, int32_t destCapacity,
1727                 const UChar *src, int32_t srcLength,
1728                 UErrorCode *pErrorCode) {
1729     const UChar *srcLimit;
1730     char *originalDest, *destLimit;
1731     int32_t destLength;
1732 
1733     /* check arguments */
1734     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
1735         return 0;
1736     }
1737 
1738     if( cnv==NULL ||
1739         destCapacity<0 || (destCapacity>0 && dest==NULL) ||
1740         srcLength<-1 || (srcLength!=0 && src==NULL)
1741     ) {
1742         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
1743         return 0;
1744     }
1745 
1746     /* initialize */
1747     ucnv_resetFromUnicode(cnv);
1748     originalDest=dest;
1749     if(srcLength==-1) {
1750         srcLength=u_strlen(src);
1751     }
1752     if(srcLength>0) {
1753         srcLimit=src+srcLength;
1754         destLimit=dest+destCapacity;
1755 
1756         /* pin the destination limit to U_MAX_PTR; NULL check is for OS/400 */
1757         if(destLimit<dest || (destLimit==NULL && dest!=NULL)) {
1758             destLimit=(char *)U_MAX_PTR(dest);
1759         }
1760 
1761         /* perform the conversion */
1762         ucnv_fromUnicode(cnv, &dest, destLimit, &src, srcLimit, 0, TRUE, pErrorCode);
1763         destLength=(int32_t)(dest-originalDest);
1764 
1765         /* if an overflow occurs, then get the preflighting length */
1766         if(*pErrorCode==U_BUFFER_OVERFLOW_ERROR) {
1767             char buffer[1024];
1768 
1769             destLimit=buffer+sizeof(buffer);
1770             do {
1771                 dest=buffer;
1772                 *pErrorCode=U_ZERO_ERROR;
1773                 ucnv_fromUnicode(cnv, &dest, destLimit, &src, srcLimit, 0, TRUE, pErrorCode);
1774                 destLength+=(int32_t)(dest-buffer);
1775             } while(*pErrorCode==U_BUFFER_OVERFLOW_ERROR);
1776         }
1777     } else {
1778         destLength=0;
1779     }
1780 
1781     return u_terminateChars(originalDest, destCapacity, destLength, pErrorCode);
1782 }
1783 
1784 U_CAPI int32_t U_EXPORT2
ucnv_toUChars(UConverter * cnv,UChar * dest,int32_t destCapacity,const char * src,int32_t srcLength,UErrorCode * pErrorCode)1785 ucnv_toUChars(UConverter *cnv,
1786               UChar *dest, int32_t destCapacity,
1787               const char *src, int32_t srcLength,
1788               UErrorCode *pErrorCode) {
1789     const char *srcLimit;
1790     UChar *originalDest, *destLimit;
1791     int32_t destLength;
1792 
1793     /* check arguments */
1794     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
1795         return 0;
1796     }
1797 
1798     if( cnv==NULL ||
1799         destCapacity<0 || (destCapacity>0 && dest==NULL) ||
1800         srcLength<-1 || (srcLength!=0 && src==NULL))
1801     {
1802         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
1803         return 0;
1804     }
1805 
1806     /* initialize */
1807     ucnv_resetToUnicode(cnv);
1808     originalDest=dest;
1809     if(srcLength==-1) {
1810         srcLength=(int32_t)uprv_strlen(src);
1811     }
1812     if(srcLength>0) {
1813         srcLimit=src+srcLength;
1814         destLimit=dest+destCapacity;
1815 
1816         /* pin the destination limit to U_MAX_PTR; NULL check is for OS/400 */
1817         if(destLimit<dest || (destLimit==NULL && dest!=NULL)) {
1818             destLimit=(UChar *)U_MAX_PTR(dest);
1819         }
1820 
1821         /* perform the conversion */
1822         ucnv_toUnicode(cnv, &dest, destLimit, &src, srcLimit, 0, TRUE, pErrorCode);
1823         destLength=(int32_t)(dest-originalDest);
1824 
1825         /* if an overflow occurs, then get the preflighting length */
1826         if(*pErrorCode==U_BUFFER_OVERFLOW_ERROR)
1827         {
1828             UChar buffer[1024];
1829 
1830             destLimit=buffer+sizeof(buffer)/U_SIZEOF_UCHAR;
1831             do {
1832                 dest=buffer;
1833                 *pErrorCode=U_ZERO_ERROR;
1834                 ucnv_toUnicode(cnv, &dest, destLimit, &src, srcLimit, 0, TRUE, pErrorCode);
1835                 destLength+=(int32_t)(dest-buffer);
1836             }
1837             while(*pErrorCode==U_BUFFER_OVERFLOW_ERROR);
1838         }
1839     } else {
1840         destLength=0;
1841     }
1842 
1843     return u_terminateUChars(originalDest, destCapacity, destLength, pErrorCode);
1844 }
1845 
1846 /* ucnv_getNextUChar() ------------------------------------------------------ */
1847 
1848 U_CAPI UChar32 U_EXPORT2
ucnv_getNextUChar(UConverter * cnv,const char ** source,const char * sourceLimit,UErrorCode * err)1849 ucnv_getNextUChar(UConverter *cnv,
1850                   const char **source, const char *sourceLimit,
1851                   UErrorCode *err) {
1852     UConverterToUnicodeArgs args;
1853     UChar buffer[U16_MAX_LENGTH];
1854     const char *s;
1855     UChar32 c;
1856     int32_t i, length;
1857 
1858     /* check parameters */
1859     if(err==NULL || U_FAILURE(*err)) {
1860         return 0xffff;
1861     }
1862 
1863     if(cnv==NULL || source==NULL) {
1864         *err=U_ILLEGAL_ARGUMENT_ERROR;
1865         return 0xffff;
1866     }
1867 
1868     s=*source;
1869     if(sourceLimit<s) {
1870         *err=U_ILLEGAL_ARGUMENT_ERROR;
1871         return 0xffff;
1872     }
1873 
1874     /*
1875      * Make sure that the buffer sizes do not exceed the number range for
1876      * int32_t because some functions use the size (in units or bytes)
1877      * rather than comparing pointers, and because offsets are int32_t values.
1878      *
1879      * size_t is guaranteed to be unsigned and large enough for the job.
1880      *
1881      * Return with an error instead of adjusting the limits because we would
1882      * not be able to maintain the semantics that either the source must be
1883      * consumed or the target filled (unless an error occurs).
1884      * An adjustment would be sourceLimit=t+0x7fffffff; for example.
1885      */
1886     if(((size_t)(sourceLimit-s)>(size_t)0x7fffffff && sourceLimit>s)) {
1887         *err=U_ILLEGAL_ARGUMENT_ERROR;
1888         return 0xffff;
1889     }
1890 
1891     c=U_SENTINEL;
1892 
1893     /* flush the target overflow buffer */
1894     if(cnv->UCharErrorBufferLength>0) {
1895         UChar *overflow;
1896 
1897         overflow=cnv->UCharErrorBuffer;
1898         i=0;
1899         length=cnv->UCharErrorBufferLength;
1900         U16_NEXT(overflow, i, length, c);
1901 
1902         /* move the remaining overflow contents up to the beginning */
1903         if((cnv->UCharErrorBufferLength=(int8_t)(length-i))>0) {
1904             uprv_memmove(cnv->UCharErrorBuffer, cnv->UCharErrorBuffer+i,
1905                          cnv->UCharErrorBufferLength*U_SIZEOF_UCHAR);
1906         }
1907 
1908         if(!U16_IS_LEAD(c) || i<length) {
1909             return c;
1910         }
1911         /*
1912          * Continue if the overflow buffer contained only a lead surrogate,
1913          * in case the converter outputs single surrogates from complete
1914          * input sequences.
1915          */
1916     }
1917 
1918     /*
1919      * flush==TRUE is implied for ucnv_getNextUChar()
1920      *
1921      * do not simply return even if s==sourceLimit because the converter may
1922      * not have seen flush==TRUE before
1923      */
1924 
1925     /* prepare the converter arguments */
1926     args.converter=cnv;
1927     args.flush=TRUE;
1928     args.offsets=NULL;
1929     args.source=s;
1930     args.sourceLimit=sourceLimit;
1931     args.target=buffer;
1932     args.targetLimit=buffer+1;
1933     args.size=sizeof(args);
1934 
1935     if(c<0) {
1936         /*
1937          * call the native getNextUChar() implementation if we are
1938          * at a character boundary (toULength==0)
1939          *
1940          * unlike with _toUnicode(), getNextUChar() implementations must set
1941          * U_TRUNCATED_CHAR_FOUND for truncated input,
1942          * in addition to setting toULength/toUBytes[]
1943          */
1944         if(cnv->toULength==0 && cnv->sharedData->impl->getNextUChar!=NULL) {
1945             c=cnv->sharedData->impl->getNextUChar(&args, err);
1946             *source=s=args.source;
1947             if(*err==U_INDEX_OUTOFBOUNDS_ERROR) {
1948                 /* reset the converter without calling the callback function */
1949                 _reset(cnv, UCNV_RESET_TO_UNICODE, FALSE);
1950                 return 0xffff; /* no output */
1951             } else if(U_SUCCESS(*err) && c>=0) {
1952                 return c;
1953             /*
1954              * else fall through to use _toUnicode() because
1955              *   UCNV_GET_NEXT_UCHAR_USE_TO_U: the native function did not want to handle it after all
1956              *   U_FAILURE: call _toUnicode() for callback handling (do not output c)
1957              */
1958             }
1959         }
1960 
1961         /* convert to one UChar in buffer[0], or handle getNextUChar() errors */
1962         _toUnicodeWithCallback(&args, err);
1963 
1964         if(*err==U_BUFFER_OVERFLOW_ERROR) {
1965             *err=U_ZERO_ERROR;
1966         }
1967 
1968         i=0;
1969         length=(int32_t)(args.target-buffer);
1970     } else {
1971         /* write the lead surrogate from the overflow buffer */
1972         buffer[0]=(UChar)c;
1973         args.target=buffer+1;
1974         i=0;
1975         length=1;
1976     }
1977 
1978     /* buffer contents starts at i and ends before length */
1979 
1980     if(U_FAILURE(*err)) {
1981         c=0xffff; /* no output */
1982     } else if(length==0) {
1983         /* no input or only state changes */
1984         *err=U_INDEX_OUTOFBOUNDS_ERROR;
1985         /* no need to reset explicitly because _toUnicodeWithCallback() did it */
1986         c=0xffff; /* no output */
1987     } else {
1988         c=buffer[0];
1989         i=1;
1990         if(!U16_IS_LEAD(c)) {
1991             /* consume c=buffer[0], done */
1992         } else {
1993             /* got a lead surrogate, see if a trail surrogate follows */
1994             UChar c2;
1995 
1996             if(cnv->UCharErrorBufferLength>0) {
1997                 /* got overflow output from the conversion */
1998                 if(U16_IS_TRAIL(c2=cnv->UCharErrorBuffer[0])) {
1999                     /* got a trail surrogate, too */
2000                     c=U16_GET_SUPPLEMENTARY(c, c2);
2001 
2002                     /* move the remaining overflow contents up to the beginning */
2003                     if((--cnv->UCharErrorBufferLength)>0) {
2004                         uprv_memmove(cnv->UCharErrorBuffer, cnv->UCharErrorBuffer+1,
2005                                      cnv->UCharErrorBufferLength*U_SIZEOF_UCHAR);
2006                     }
2007                 } else {
2008                     /* c is an unpaired lead surrogate, just return it */
2009                 }
2010             } else if(args.source<sourceLimit) {
2011                 /* convert once more, to buffer[1] */
2012                 args.targetLimit=buffer+2;
2013                 _toUnicodeWithCallback(&args, err);
2014                 if(*err==U_BUFFER_OVERFLOW_ERROR) {
2015                     *err=U_ZERO_ERROR;
2016                 }
2017 
2018                 length=(int32_t)(args.target-buffer);
2019                 if(U_SUCCESS(*err) && length==2 && U16_IS_TRAIL(c2=buffer[1])) {
2020                     /* got a trail surrogate, too */
2021                     c=U16_GET_SUPPLEMENTARY(c, c2);
2022                     i=2;
2023                 }
2024             }
2025         }
2026     }
2027 
2028     /*
2029      * move leftover output from buffer[i..length[
2030      * into the beginning of the overflow buffer
2031      */
2032     if(i<length) {
2033         /* move further overflow back */
2034         int32_t delta=length-i;
2035         if((length=cnv->UCharErrorBufferLength)>0) {
2036             uprv_memmove(cnv->UCharErrorBuffer+delta, cnv->UCharErrorBuffer,
2037                          length*U_SIZEOF_UCHAR);
2038         }
2039         cnv->UCharErrorBufferLength=(int8_t)(length+delta);
2040 
2041         cnv->UCharErrorBuffer[0]=buffer[i++];
2042         if(delta>1) {
2043             cnv->UCharErrorBuffer[1]=buffer[i];
2044         }
2045     }
2046 
2047     *source=args.source;
2048     return c;
2049 }
2050 
2051 /* ucnv_convert() and siblings ---------------------------------------------- */
2052 
2053 U_CAPI void U_EXPORT2
ucnv_convertEx(UConverter * targetCnv,UConverter * sourceCnv,char ** target,const char * targetLimit,const char ** source,const char * sourceLimit,UChar * pivotStart,UChar ** pivotSource,UChar ** pivotTarget,const UChar * pivotLimit,UBool reset,UBool flush,UErrorCode * pErrorCode)2054 ucnv_convertEx(UConverter *targetCnv, UConverter *sourceCnv,
2055                char **target, const char *targetLimit,
2056                const char **source, const char *sourceLimit,
2057                UChar *pivotStart, UChar **pivotSource,
2058                UChar **pivotTarget, const UChar *pivotLimit,
2059                UBool reset, UBool flush,
2060                UErrorCode *pErrorCode) {
2061     UChar pivotBuffer[CHUNK_SIZE];
2062     const UChar *myPivotSource;
2063     UChar *myPivotTarget;
2064     const char *s;
2065     char *t;
2066 
2067     UConverterToUnicodeArgs toUArgs;
2068     UConverterFromUnicodeArgs fromUArgs;
2069     UConverterConvert convert;
2070 
2071     /* error checking */
2072     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
2073         return;
2074     }
2075 
2076     if( targetCnv==NULL || sourceCnv==NULL ||
2077         source==NULL || *source==NULL ||
2078         target==NULL || *target==NULL || targetLimit==NULL
2079     ) {
2080         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2081         return;
2082     }
2083 
2084     s=*source;
2085     t=*target;
2086     if((sourceLimit!=NULL && sourceLimit<s) || targetLimit<t) {
2087         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2088         return;
2089     }
2090 
2091     /*
2092      * Make sure that the buffer sizes do not exceed the number range for
2093      * int32_t. See ucnv_toUnicode() for a more detailed comment.
2094      */
2095     if(
2096         (sourceLimit!=NULL && ((size_t)(sourceLimit-s)>(size_t)0x7fffffff && sourceLimit>s)) ||
2097         ((size_t)(targetLimit-t)>(size_t)0x7fffffff && targetLimit>t)
2098     ) {
2099         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2100         return;
2101     }
2102 
2103     if(pivotStart==NULL) {
2104         if(!flush) {
2105             /* streaming conversion requires an explicit pivot buffer */
2106             *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2107             return;
2108         }
2109 
2110         /* use the stack pivot buffer */
2111         myPivotSource=myPivotTarget=pivotStart=pivotBuffer;
2112         pivotSource=(UChar **)&myPivotSource;
2113         pivotTarget=&myPivotTarget;
2114         pivotLimit=pivotBuffer+CHUNK_SIZE;
2115     } else if(  pivotStart>=pivotLimit ||
2116                 pivotSource==NULL || *pivotSource==NULL ||
2117                 pivotTarget==NULL || *pivotTarget==NULL ||
2118                 pivotLimit==NULL
2119     ) {
2120         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2121         return;
2122     }
2123 
2124     if(sourceLimit==NULL) {
2125         /* get limit of single-byte-NUL-terminated source string */
2126         sourceLimit=uprv_strchr(*source, 0);
2127     }
2128 
2129     if(reset) {
2130         ucnv_resetToUnicode(sourceCnv);
2131         ucnv_resetFromUnicode(targetCnv);
2132         *pivotSource=*pivotTarget=pivotStart;
2133     } else if(targetCnv->charErrorBufferLength>0) {
2134         /* output the targetCnv overflow buffer */
2135         if(ucnv_outputOverflowFromUnicode(targetCnv, target, targetLimit, NULL, pErrorCode)) {
2136             /* U_BUFFER_OVERFLOW_ERROR */
2137             return;
2138         }
2139         /* *target has moved, therefore stop using t */
2140 
2141         if( !flush &&
2142             targetCnv->preFromULength>=0 && *pivotSource==*pivotTarget &&
2143             sourceCnv->UCharErrorBufferLength==0 && sourceCnv->preToULength>=0 && s==sourceLimit
2144         ) {
2145             /* the fromUnicode overflow buffer is emptied and there is no new input: we are done */
2146             return;
2147         }
2148     }
2149 
2150     /* Is direct-UTF-8 conversion available? */
2151     if( sourceCnv->sharedData->staticData->conversionType==UCNV_UTF8 &&
2152         targetCnv->sharedData->impl->fromUTF8!=NULL
2153     ) {
2154         convert=targetCnv->sharedData->impl->fromUTF8;
2155     } else if( targetCnv->sharedData->staticData->conversionType==UCNV_UTF8 &&
2156                sourceCnv->sharedData->impl->toUTF8!=NULL
2157     ) {
2158         convert=sourceCnv->sharedData->impl->toUTF8;
2159     } else {
2160         convert=NULL;
2161     }
2162 
2163     /*
2164      * If direct-UTF-8 conversion is available, then we use a smaller
2165      * pivot buffer for error handling and partial matches
2166      * so that we quickly return to direct conversion.
2167      *
2168      * 32 is large enough for UCNV_EXT_MAX_UCHARS and UCNV_ERROR_BUFFER_LENGTH.
2169      *
2170      * We could reduce the pivot buffer size further, at the cost of
2171      * buffer overflows from callbacks.
2172      * The pivot buffer should not be smaller than the maximum number of
2173      * fromUnicode extension table input UChars
2174      * (for m:n conversion, see
2175      * targetCnv->sharedData->mbcs.extIndexes[UCNV_EXT_COUNT_UCHARS])
2176      * or 2 for surrogate pairs.
2177      *
2178      * Too small a buffer can cause thrashing between pivoting and direct
2179      * conversion, with function call overhead outweighing the benefits
2180      * of direct conversion.
2181      */
2182     if(convert!=NULL && (pivotLimit-pivotStart)>32) {
2183         pivotLimit=pivotStart+32;
2184     }
2185 
2186     /* prepare the converter arguments */
2187     fromUArgs.converter=targetCnv;
2188     fromUArgs.flush=FALSE;
2189     fromUArgs.offsets=NULL;
2190     fromUArgs.target=*target;
2191     fromUArgs.targetLimit=targetLimit;
2192     fromUArgs.size=sizeof(fromUArgs);
2193 
2194     toUArgs.converter=sourceCnv;
2195     toUArgs.flush=flush;
2196     toUArgs.offsets=NULL;
2197     toUArgs.source=s;
2198     toUArgs.sourceLimit=sourceLimit;
2199     toUArgs.targetLimit=pivotLimit;
2200     toUArgs.size=sizeof(toUArgs);
2201 
2202     /*
2203      * TODO: Consider separating this function into two functions,
2204      * extracting exactly the conversion loop,
2205      * for readability and to reduce the set of visible variables.
2206      *
2207      * Otherwise stop using s and t from here on.
2208      */
2209     s=t=NULL;
2210 
2211     /*
2212      * conversion loop
2213      *
2214      * The sequence of steps in the loop may appear backward,
2215      * but the principle is simple:
2216      * In the chain of
2217      *   source - sourceCnv overflow - pivot - targetCnv overflow - target
2218      * empty out later buffers before refilling them from earlier ones.
2219      *
2220      * The targetCnv overflow buffer is flushed out only once before the loop.
2221      */
2222     for(;;) {
2223         /*
2224          * if(pivot not empty or error or replay or flush fromUnicode) {
2225          *   fromUnicode(pivot -> target);
2226          * }
2227          *
2228          * For pivoting conversion; and for direct conversion for
2229          * error callback handling and flushing the replay buffer.
2230          */
2231         if( *pivotSource<*pivotTarget ||
2232             U_FAILURE(*pErrorCode) ||
2233             targetCnv->preFromULength<0 ||
2234             fromUArgs.flush
2235         ) {
2236             fromUArgs.source=*pivotSource;
2237             fromUArgs.sourceLimit=*pivotTarget;
2238             _fromUnicodeWithCallback(&fromUArgs, pErrorCode);
2239             if(U_FAILURE(*pErrorCode)) {
2240                 /* target overflow, or conversion error */
2241                 *pivotSource=(UChar *)fromUArgs.source;
2242                 break;
2243             }
2244 
2245             /*
2246              * _fromUnicodeWithCallback() must have consumed the pivot contents
2247              * (*pivotSource==*pivotTarget) since it returned with U_SUCCESS()
2248              */
2249         }
2250 
2251         /* The pivot buffer is empty; reset it so we start at pivotStart. */
2252         *pivotSource=*pivotTarget=pivotStart;
2253 
2254         /*
2255          * if(sourceCnv overflow buffer not empty) {
2256          *     move(sourceCnv overflow buffer -> pivot);
2257          *     continue;
2258          * }
2259          */
2260         /* output the sourceCnv overflow buffer */
2261         if(sourceCnv->UCharErrorBufferLength>0) {
2262             if(ucnv_outputOverflowToUnicode(sourceCnv, pivotTarget, pivotLimit, NULL, pErrorCode)) {
2263                 /* U_BUFFER_OVERFLOW_ERROR */
2264                 *pErrorCode=U_ZERO_ERROR;
2265             }
2266             continue;
2267         }
2268 
2269         /*
2270          * check for end of input and break if done
2271          *
2272          * Checking both flush and fromUArgs.flush ensures that the converters
2273          * have been called with the flush flag set if the ucnv_convertEx()
2274          * caller set it.
2275          */
2276         if( toUArgs.source==sourceLimit &&
2277             sourceCnv->preToULength>=0 && sourceCnv->toULength==0 &&
2278             (!flush || fromUArgs.flush)
2279         ) {
2280             /* done successfully */
2281             break;
2282         }
2283 
2284         /*
2285          * use direct conversion if available
2286          * but not if continuing a partial match
2287          * or flushing the toUnicode replay buffer
2288          */
2289         if(convert!=NULL && targetCnv->preFromUFirstCP<0 && sourceCnv->preToULength==0) {
2290             if(*pErrorCode==U_USING_DEFAULT_WARNING) {
2291                 /* remove a warning that may be set by this function */
2292                 *pErrorCode=U_ZERO_ERROR;
2293             }
2294             convert(&fromUArgs, &toUArgs, pErrorCode);
2295             if(*pErrorCode==U_BUFFER_OVERFLOW_ERROR) {
2296                 break;
2297             } else if(U_FAILURE(*pErrorCode)) {
2298                 if(sourceCnv->toULength>0) {
2299                     /*
2300                      * Fall through to calling _toUnicodeWithCallback()
2301                      * for callback handling.
2302                      *
2303                      * The pivot buffer will be reset with
2304                      *   *pivotSource=*pivotTarget=pivotStart;
2305                      * which indicates a toUnicode error to the caller
2306                      * (*pivotSource==pivotStart shows no pivot UChars consumed).
2307                      */
2308                 } else {
2309                     /*
2310                      * Indicate a fromUnicode error to the caller
2311                      * (*pivotSource>pivotStart shows some pivot UChars consumed).
2312                      */
2313                     *pivotSource=*pivotTarget=pivotStart+1;
2314                     /*
2315                      * Loop around to calling _fromUnicodeWithCallbacks()
2316                      * for callback handling.
2317                      */
2318                     continue;
2319                 }
2320             } else if(*pErrorCode==U_USING_DEFAULT_WARNING) {
2321                 /*
2322                  * No error, but the implementation requested to temporarily
2323                  * fall back to pivoting.
2324                  */
2325                 *pErrorCode=U_ZERO_ERROR;
2326             /*
2327              * The following else branches are almost identical to the end-of-input
2328              * handling in _toUnicodeWithCallback().
2329              * Avoid calling it just for the end of input.
2330              */
2331             } else if(flush && sourceCnv->toULength>0) { /* flush==toUArgs.flush */
2332                 /*
2333                  * the entire input stream is consumed
2334                  * and there is a partial, truncated input sequence left
2335                  */
2336 
2337                 /* inject an error and continue with callback handling */
2338                 *pErrorCode=U_TRUNCATED_CHAR_FOUND;
2339             } else {
2340                 /* input consumed */
2341                 if(flush) {
2342                     /* reset the converters without calling the callback functions */
2343                     _reset(sourceCnv, UCNV_RESET_TO_UNICODE, FALSE);
2344                     _reset(targetCnv, UCNV_RESET_FROM_UNICODE, FALSE);
2345                 }
2346 
2347                 /* done successfully */
2348                 break;
2349             }
2350         }
2351 
2352         /*
2353          * toUnicode(source -> pivot);
2354          *
2355          * For pivoting conversion; and for direct conversion for
2356          * error callback handling, continuing partial matches
2357          * and flushing the replay buffer.
2358          *
2359          * The pivot buffer is empty and reset.
2360          */
2361         toUArgs.target=pivotStart; /* ==*pivotTarget */
2362         /* toUArgs.targetLimit=pivotLimit; already set before the loop */
2363         _toUnicodeWithCallback(&toUArgs, pErrorCode);
2364         *pivotTarget=toUArgs.target;
2365         if(*pErrorCode==U_BUFFER_OVERFLOW_ERROR) {
2366             /* pivot overflow: continue with the conversion loop */
2367             *pErrorCode=U_ZERO_ERROR;
2368         } else if(U_FAILURE(*pErrorCode) || (!flush && *pivotTarget==pivotStart)) {
2369             /* conversion error, or there was nothing left to convert */
2370             break;
2371         }
2372         /*
2373          * else:
2374          * _toUnicodeWithCallback() wrote into the pivot buffer,
2375          * continue with fromUnicode conversion.
2376          *
2377          * Set the fromUnicode flush flag if we flush and if toUnicode has
2378          * processed the end of the input.
2379          */
2380         if( flush && toUArgs.source==sourceLimit &&
2381             sourceCnv->preToULength>=0 &&
2382             sourceCnv->UCharErrorBufferLength==0
2383         ) {
2384             fromUArgs.flush=TRUE;
2385         }
2386     }
2387 
2388     /*
2389      * The conversion loop is exited when one of the following is true:
2390      * - the entire source text has been converted successfully to the target buffer
2391      * - a target buffer overflow occurred
2392      * - a conversion error occurred
2393      */
2394 
2395     *source=toUArgs.source;
2396     *target=fromUArgs.target;
2397 
2398     /* terminate the target buffer if possible */
2399     if(flush && U_SUCCESS(*pErrorCode)) {
2400         if(*target!=targetLimit) {
2401             **target=0;
2402             if(*pErrorCode==U_STRING_NOT_TERMINATED_WARNING) {
2403                 *pErrorCode=U_ZERO_ERROR;
2404             }
2405         } else {
2406             *pErrorCode=U_STRING_NOT_TERMINATED_WARNING;
2407         }
2408     }
2409 }
2410 
2411 /* internal implementation of ucnv_convert() etc. with preflighting */
2412 static int32_t
ucnv_internalConvert(UConverter * outConverter,UConverter * inConverter,char * target,int32_t targetCapacity,const char * source,int32_t sourceLength,UErrorCode * pErrorCode)2413 ucnv_internalConvert(UConverter *outConverter, UConverter *inConverter,
2414                      char *target, int32_t targetCapacity,
2415                      const char *source, int32_t sourceLength,
2416                      UErrorCode *pErrorCode) {
2417     UChar pivotBuffer[CHUNK_SIZE];
2418     UChar *pivot, *pivot2;
2419 
2420     char *myTarget;
2421     const char *sourceLimit;
2422     const char *targetLimit;
2423     int32_t targetLength=0;
2424 
2425     /* set up */
2426     if(sourceLength<0) {
2427         sourceLimit=uprv_strchr(source, 0);
2428     } else {
2429         sourceLimit=source+sourceLength;
2430     }
2431 
2432     /* if there is no input data, we're done */
2433     if(source==sourceLimit) {
2434         return u_terminateChars(target, targetCapacity, 0, pErrorCode);
2435     }
2436 
2437     pivot=pivot2=pivotBuffer;
2438     myTarget=target;
2439     targetLength=0;
2440 
2441     if(targetCapacity>0) {
2442         /* perform real conversion */
2443         targetLimit=target+targetCapacity;
2444         ucnv_convertEx(outConverter, inConverter,
2445                        &myTarget, targetLimit,
2446                        &source, sourceLimit,
2447                        pivotBuffer, &pivot, &pivot2, pivotBuffer+CHUNK_SIZE,
2448                        FALSE,
2449                        TRUE,
2450                        pErrorCode);
2451         targetLength=(int32_t)(myTarget-target);
2452     }
2453 
2454     /*
2455      * If the output buffer is exhausted (or we are only "preflighting"), we need to stop writing
2456      * to it but continue the conversion in order to store in targetCapacity
2457      * the number of bytes that was required.
2458      */
2459     if(*pErrorCode==U_BUFFER_OVERFLOW_ERROR || targetCapacity==0)
2460     {
2461         char targetBuffer[CHUNK_SIZE];
2462 
2463         targetLimit=targetBuffer+CHUNK_SIZE;
2464         do {
2465             *pErrorCode=U_ZERO_ERROR;
2466             myTarget=targetBuffer;
2467             ucnv_convertEx(outConverter, inConverter,
2468                            &myTarget, targetLimit,
2469                            &source, sourceLimit,
2470                            pivotBuffer, &pivot, &pivot2, pivotBuffer+CHUNK_SIZE,
2471                            FALSE,
2472                            TRUE,
2473                            pErrorCode);
2474             targetLength+=(int32_t)(myTarget-targetBuffer);
2475         } while(*pErrorCode==U_BUFFER_OVERFLOW_ERROR);
2476 
2477         /* done with preflighting, set warnings and errors as appropriate */
2478         return u_terminateChars(target, targetCapacity, targetLength, pErrorCode);
2479     }
2480 
2481     /* no need to call u_terminateChars() because ucnv_convertEx() took care of that */
2482     return targetLength;
2483 }
2484 
2485 U_CAPI int32_t U_EXPORT2
ucnv_convert(const char * toConverterName,const char * fromConverterName,char * target,int32_t targetCapacity,const char * source,int32_t sourceLength,UErrorCode * pErrorCode)2486 ucnv_convert(const char *toConverterName, const char *fromConverterName,
2487              char *target, int32_t targetCapacity,
2488              const char *source, int32_t sourceLength,
2489              UErrorCode *pErrorCode) {
2490     UConverter in, out; /* stack-allocated */
2491     UConverter *inConverter, *outConverter;
2492     int32_t targetLength;
2493 
2494     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
2495         return 0;
2496     }
2497 
2498     if( source==NULL || sourceLength<-1 ||
2499         targetCapacity<0 || (targetCapacity>0 && target==NULL)
2500     ) {
2501         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2502         return 0;
2503     }
2504 
2505     /* if there is no input data, we're done */
2506     if(sourceLength==0 || (sourceLength<0 && *source==0)) {
2507         return u_terminateChars(target, targetCapacity, 0, pErrorCode);
2508     }
2509 
2510     /* create the converters */
2511     inConverter=ucnv_createConverter(&in, fromConverterName, pErrorCode);
2512     if(U_FAILURE(*pErrorCode)) {
2513         return 0;
2514     }
2515 
2516     outConverter=ucnv_createConverter(&out, toConverterName, pErrorCode);
2517     if(U_FAILURE(*pErrorCode)) {
2518         ucnv_close(inConverter);
2519         return 0;
2520     }
2521 
2522     targetLength=ucnv_internalConvert(outConverter, inConverter,
2523                                       target, targetCapacity,
2524                                       source, sourceLength,
2525                                       pErrorCode);
2526 
2527     ucnv_close(inConverter);
2528     ucnv_close(outConverter);
2529 
2530     return targetLength;
2531 }
2532 
2533 /* @internal */
2534 static int32_t
ucnv_convertAlgorithmic(UBool convertToAlgorithmic,UConverterType algorithmicType,UConverter * cnv,char * target,int32_t targetCapacity,const char * source,int32_t sourceLength,UErrorCode * pErrorCode)2535 ucnv_convertAlgorithmic(UBool convertToAlgorithmic,
2536                         UConverterType algorithmicType,
2537                         UConverter *cnv,
2538                         char *target, int32_t targetCapacity,
2539                         const char *source, int32_t sourceLength,
2540                         UErrorCode *pErrorCode) {
2541     UConverter algoConverterStatic; /* stack-allocated */
2542     UConverter *algoConverter, *to, *from;
2543     int32_t targetLength;
2544 
2545     if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
2546         return 0;
2547     }
2548 
2549     if( cnv==NULL || source==NULL || sourceLength<-1 ||
2550         targetCapacity<0 || (targetCapacity>0 && target==NULL)
2551     ) {
2552         *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
2553         return 0;
2554     }
2555 
2556     /* if there is no input data, we're done */
2557     if(sourceLength==0 || (sourceLength<0 && *source==0)) {
2558         return u_terminateChars(target, targetCapacity, 0, pErrorCode);
2559     }
2560 
2561     /* create the algorithmic converter */
2562     algoConverter=ucnv_createAlgorithmicConverter(&algoConverterStatic, algorithmicType,
2563                                                   "", 0, pErrorCode);
2564     if(U_FAILURE(*pErrorCode)) {
2565         return 0;
2566     }
2567 
2568     /* reset the other converter */
2569     if(convertToAlgorithmic) {
2570         /* cnv->Unicode->algo */
2571         ucnv_resetToUnicode(cnv);
2572         to=algoConverter;
2573         from=cnv;
2574     } else {
2575         /* algo->Unicode->cnv */
2576         ucnv_resetFromUnicode(cnv);
2577         from=algoConverter;
2578         to=cnv;
2579     }
2580 
2581     targetLength=ucnv_internalConvert(to, from,
2582                                       target, targetCapacity,
2583                                       source, sourceLength,
2584                                       pErrorCode);
2585 
2586     ucnv_close(algoConverter);
2587 
2588     return targetLength;
2589 }
2590 
2591 U_CAPI int32_t U_EXPORT2
ucnv_toAlgorithmic(UConverterType algorithmicType,UConverter * cnv,char * target,int32_t targetCapacity,const char * source,int32_t sourceLength,UErrorCode * pErrorCode)2592 ucnv_toAlgorithmic(UConverterType algorithmicType,
2593                    UConverter *cnv,
2594                    char *target, int32_t targetCapacity,
2595                    const char *source, int32_t sourceLength,
2596                    UErrorCode *pErrorCode) {
2597     return ucnv_convertAlgorithmic(TRUE, algorithmicType, cnv,
2598                                    target, targetCapacity,
2599                                    source, sourceLength,
2600                                    pErrorCode);
2601 }
2602 
2603 U_CAPI int32_t U_EXPORT2
ucnv_fromAlgorithmic(UConverter * cnv,UConverterType algorithmicType,char * target,int32_t targetCapacity,const char * source,int32_t sourceLength,UErrorCode * pErrorCode)2604 ucnv_fromAlgorithmic(UConverter *cnv,
2605                      UConverterType algorithmicType,
2606                      char *target, int32_t targetCapacity,
2607                      const char *source, int32_t sourceLength,
2608                      UErrorCode *pErrorCode) {
2609     return ucnv_convertAlgorithmic(FALSE, algorithmicType, cnv,
2610                                    target, targetCapacity,
2611                                    source, sourceLength,
2612                                    pErrorCode);
2613 }
2614 
2615 U_CAPI UConverterType  U_EXPORT2
ucnv_getType(const UConverter * converter)2616 ucnv_getType(const UConverter* converter)
2617 {
2618     int8_t type = converter->sharedData->staticData->conversionType;
2619 #if !UCONFIG_NO_LEGACY_CONVERSION
2620     if(type == UCNV_MBCS) {
2621         return ucnv_MBCSGetType(converter);
2622     }
2623 #endif
2624     return (UConverterType)type;
2625 }
2626 
2627 U_CAPI void  U_EXPORT2
ucnv_getStarters(const UConverter * converter,UBool starters[256],UErrorCode * err)2628 ucnv_getStarters(const UConverter* converter,
2629                  UBool starters[256],
2630                  UErrorCode* err)
2631 {
2632     if (err == NULL || U_FAILURE(*err)) {
2633         return;
2634     }
2635 
2636     if(converter->sharedData->impl->getStarters != NULL) {
2637         converter->sharedData->impl->getStarters(converter, starters, err);
2638     } else {
2639         *err = U_ILLEGAL_ARGUMENT_ERROR;
2640     }
2641 }
2642 
ucnv_getAmbiguous(const UConverter * cnv)2643 static const UAmbiguousConverter *ucnv_getAmbiguous(const UConverter *cnv)
2644 {
2645     UErrorCode errorCode;
2646     const char *name;
2647     int32_t i;
2648 
2649     if(cnv==NULL) {
2650         return NULL;
2651     }
2652 
2653     errorCode=U_ZERO_ERROR;
2654     name=ucnv_getName(cnv, &errorCode);
2655     if(U_FAILURE(errorCode)) {
2656         return NULL;
2657     }
2658 
2659     for(i=0; i<(int32_t)(sizeof(ambiguousConverters)/sizeof(UAmbiguousConverter)); ++i)
2660     {
2661         if(0==uprv_strcmp(name, ambiguousConverters[i].name))
2662         {
2663             return ambiguousConverters+i;
2664         }
2665     }
2666 
2667     return NULL;
2668 }
2669 
2670 U_CAPI void  U_EXPORT2
ucnv_fixFileSeparator(const UConverter * cnv,UChar * source,int32_t sourceLength)2671 ucnv_fixFileSeparator(const UConverter *cnv,
2672                       UChar* source,
2673                       int32_t sourceLength) {
2674     const UAmbiguousConverter *a;
2675     int32_t i;
2676     UChar variant5c;
2677 
2678     if(cnv==NULL || source==NULL || sourceLength<=0 || (a=ucnv_getAmbiguous(cnv))==NULL)
2679     {
2680         return;
2681     }
2682 
2683     variant5c=a->variant5c;
2684     for(i=0; i<sourceLength; ++i) {
2685         if(source[i]==variant5c) {
2686             source[i]=0x5c;
2687         }
2688     }
2689 }
2690 
2691 U_CAPI UBool  U_EXPORT2
ucnv_isAmbiguous(const UConverter * cnv)2692 ucnv_isAmbiguous(const UConverter *cnv) {
2693     return (UBool)(ucnv_getAmbiguous(cnv)!=NULL);
2694 }
2695 
2696 U_CAPI void  U_EXPORT2
ucnv_setFallback(UConverter * cnv,UBool usesFallback)2697 ucnv_setFallback(UConverter *cnv, UBool usesFallback)
2698 {
2699     cnv->useFallback = usesFallback;
2700 }
2701 
2702 U_CAPI UBool  U_EXPORT2
ucnv_usesFallback(const UConverter * cnv)2703 ucnv_usesFallback(const UConverter *cnv)
2704 {
2705     return cnv->useFallback;
2706 }
2707 
2708 U_CAPI void  U_EXPORT2
ucnv_getInvalidChars(const UConverter * converter,char * errBytes,int8_t * len,UErrorCode * err)2709 ucnv_getInvalidChars (const UConverter * converter,
2710                       char *errBytes,
2711                       int8_t * len,
2712                       UErrorCode * err)
2713 {
2714     if (err == NULL || U_FAILURE(*err))
2715     {
2716         return;
2717     }
2718     if (len == NULL || errBytes == NULL || converter == NULL)
2719     {
2720         *err = U_ILLEGAL_ARGUMENT_ERROR;
2721         return;
2722     }
2723     if (*len < converter->invalidCharLength)
2724     {
2725         *err = U_INDEX_OUTOFBOUNDS_ERROR;
2726         return;
2727     }
2728     if ((*len = converter->invalidCharLength) > 0)
2729     {
2730         uprv_memcpy (errBytes, converter->invalidCharBuffer, *len);
2731     }
2732 }
2733 
2734 U_CAPI void  U_EXPORT2
ucnv_getInvalidUChars(const UConverter * converter,UChar * errChars,int8_t * len,UErrorCode * err)2735 ucnv_getInvalidUChars (const UConverter * converter,
2736                        UChar *errChars,
2737                        int8_t * len,
2738                        UErrorCode * err)
2739 {
2740     if (err == NULL || U_FAILURE(*err))
2741     {
2742         return;
2743     }
2744     if (len == NULL || errChars == NULL || converter == NULL)
2745     {
2746         *err = U_ILLEGAL_ARGUMENT_ERROR;
2747         return;
2748     }
2749     if (*len < converter->invalidUCharLength)
2750     {
2751         *err = U_INDEX_OUTOFBOUNDS_ERROR;
2752         return;
2753     }
2754     if ((*len = converter->invalidUCharLength) > 0)
2755     {
2756         uprv_memcpy (errChars, converter->invalidUCharBuffer, sizeof(UChar) * (*len));
2757     }
2758 }
2759 
2760 #define SIG_MAX_LEN 5
2761 
2762 U_CAPI const char* U_EXPORT2
ucnv_detectUnicodeSignature(const char * source,int32_t sourceLength,int32_t * signatureLength,UErrorCode * pErrorCode)2763 ucnv_detectUnicodeSignature( const char* source,
2764                              int32_t sourceLength,
2765                              int32_t* signatureLength,
2766                              UErrorCode* pErrorCode) {
2767     int32_t dummy;
2768 
2769     /* initial 0xa5 bytes: make sure that if we read <SIG_MAX_LEN
2770      * bytes we don't misdetect something
2771      */
2772     char start[SIG_MAX_LEN]={ '\xa5', '\xa5', '\xa5', '\xa5', '\xa5' };
2773     int i = 0;
2774 
2775     if((pErrorCode==NULL) || U_FAILURE(*pErrorCode)){
2776         return NULL;
2777     }
2778 
2779     if(source == NULL || sourceLength < -1){
2780         *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
2781         return NULL;
2782     }
2783 
2784     if(signatureLength == NULL) {
2785         signatureLength = &dummy;
2786     }
2787 
2788     if(sourceLength==-1){
2789         sourceLength=(int32_t)uprv_strlen(source);
2790     }
2791 
2792 
2793     while(i<sourceLength&& i<SIG_MAX_LEN){
2794         start[i]=source[i];
2795         i++;
2796     }
2797 
2798     if(start[0] == '\xFE' && start[1] == '\xFF') {
2799         *signatureLength=2;
2800         return  "UTF-16BE";
2801     } else if(start[0] == '\xFF' && start[1] == '\xFE') {
2802         if(start[2] == '\x00' && start[3] =='\x00') {
2803             *signatureLength=4;
2804             return "UTF-32LE";
2805         } else {
2806             *signatureLength=2;
2807             return  "UTF-16LE";
2808         }
2809     } else if(start[0] == '\xEF' && start[1] == '\xBB' && start[2] == '\xBF') {
2810         *signatureLength=3;
2811         return  "UTF-8";
2812     } else if(start[0] == '\x00' && start[1] == '\x00' &&
2813               start[2] == '\xFE' && start[3]=='\xFF') {
2814         *signatureLength=4;
2815         return  "UTF-32BE";
2816     } else if(start[0] == '\x0E' && start[1] == '\xFE' && start[2] == '\xFF') {
2817         *signatureLength=3;
2818         return "SCSU";
2819     } else if(start[0] == '\xFB' && start[1] == '\xEE' && start[2] == '\x28') {
2820         *signatureLength=3;
2821         return "BOCU-1";
2822     } else if(start[0] == '\x2B' && start[1] == '\x2F' && start[2] == '\x76') {
2823         /*
2824          * UTF-7: Initial U+FEFF is encoded as +/v8  or  +/v9  or  +/v+  or  +/v/
2825          * depending on the second UTF-16 code unit.
2826          * Detect the entire, closed Unicode mode sequence +/v8- for only U+FEFF
2827          * if it occurs.
2828          *
2829          * So far we have +/v
2830          */
2831         if(start[3] == '\x38' && start[4] == '\x2D') {
2832             /* 5 bytes +/v8- */
2833             *signatureLength=5;
2834             return "UTF-7";
2835         } else if(start[3] == '\x38' || start[3] == '\x39' || start[3] == '\x2B' || start[3] == '\x2F') {
2836             /* 4 bytes +/v8  or  +/v9  or  +/v+  or  +/v/ */
2837             *signatureLength=4;
2838             return "UTF-7";
2839         }
2840     }else if(start[0]=='\xDD' && start[1]== '\x73'&& start[2]=='\x66' && start[3]=='\x73'){
2841         *signatureLength=4;
2842         return "UTF-EBCDIC";
2843     }
2844 
2845 
2846     /* no known Unicode signature byte sequence recognized */
2847     *signatureLength=0;
2848     return NULL;
2849 }
2850 
2851 U_CAPI int32_t U_EXPORT2
ucnv_fromUCountPending(const UConverter * cnv,UErrorCode * status)2852 ucnv_fromUCountPending(const UConverter* cnv, UErrorCode* status)
2853 {
2854     if(status == NULL || U_FAILURE(*status)){
2855         return -1;
2856     }
2857     if(cnv == NULL){
2858         *status = U_ILLEGAL_ARGUMENT_ERROR;
2859         return -1;
2860     }
2861 
2862     if(cnv->preFromUFirstCP >= 0){
2863         return U16_LENGTH(cnv->preFromUFirstCP)+cnv->preFromULength ;
2864     }else if(cnv->preFromULength < 0){
2865         return -cnv->preFromULength ;
2866     }else if(cnv->fromUChar32 > 0){
2867         return 1;
2868     }
2869     return 0;
2870 
2871 }
2872 
2873 U_CAPI int32_t U_EXPORT2
ucnv_toUCountPending(const UConverter * cnv,UErrorCode * status)2874 ucnv_toUCountPending(const UConverter* cnv, UErrorCode* status){
2875 
2876     if(status == NULL || U_FAILURE(*status)){
2877         return -1;
2878     }
2879     if(cnv == NULL){
2880         *status = U_ILLEGAL_ARGUMENT_ERROR;
2881         return -1;
2882     }
2883 
2884     if(cnv->preToULength > 0){
2885         return cnv->preToULength ;
2886     }else if(cnv->preToULength < 0){
2887         return -cnv->preToULength;
2888     }else if(cnv->toULength > 0){
2889         return cnv->toULength;
2890     }
2891     return 0;
2892 }
2893 
2894 U_CAPI UBool U_EXPORT2
ucnv_isFixedWidth(UConverter * cnv,UErrorCode * status)2895 ucnv_isFixedWidth(UConverter *cnv, UErrorCode *status){
2896     if (U_FAILURE(*status)) {
2897         return FALSE;
2898     }
2899 
2900     if (cnv == NULL) {
2901         *status = U_ILLEGAL_ARGUMENT_ERROR;
2902         return FALSE;
2903     }
2904 
2905     switch (ucnv_getType(cnv)) {
2906         case UCNV_SBCS:
2907         case UCNV_DBCS:
2908         case UCNV_UTF32_BigEndian:
2909         case UCNV_UTF32_LittleEndian:
2910         case UCNV_UTF32:
2911         case UCNV_US_ASCII:
2912             return TRUE;
2913         default:
2914             return FALSE;
2915     }
2916 }
2917 #endif
2918 
2919 /*
2920  * Hey, Emacs, please set the following:
2921  *
2922  * Local Variables:
2923  * indent-tabs-mode: nil
2924  * End:
2925  *
2926  */
2927