Home | History | Annotate | Download | only in i18n
      1 /*
      2  **********************************************************************
      3  *   Copyright (C) 2005-2015, International Business Machines
      4  *   Corporation and others.  All Rights Reserved.
      5  **********************************************************************
      6  */
      7 
      8 #include "unicode/utypes.h"
      9 
     10 #if !UCONFIG_NO_CONVERSION
     11 
     12 #include "unicode/ucsdet.h"
     13 
     14 #include "csdetect.h"
     15 #include "csmatch.h"
     16 #include "uenumimp.h"
     17 
     18 #include "cmemory.h"
     19 #include "cstring.h"
     20 #include "umutex.h"
     21 #include "ucln_in.h"
     22 #include "uarrsort.h"
     23 #include "inputext.h"
     24 #include "csrsbcs.h"
     25 #include "csrmbcs.h"
     26 #include "csrutf8.h"
     27 #include "csrucode.h"
     28 #include "csr2022.h"
     29 
     30 #define ARRAY_SIZE(array) (sizeof array / sizeof array[0])
     31 
     32 #define NEW_ARRAY(type,count) (type *) uprv_malloc((count) * sizeof(type))
     33 #define DELETE_ARRAY(array) uprv_free((void *) (array))
     34 
     35 U_NAMESPACE_BEGIN
     36 
     37 struct CSRecognizerInfo : public UMemory {
     38     CSRecognizerInfo(CharsetRecognizer *recognizer, UBool isDefaultEnabled)
     39         : recognizer(recognizer), isDefaultEnabled(isDefaultEnabled) {};
     40 
     41     ~CSRecognizerInfo() {delete recognizer;};
     42 
     43     CharsetRecognizer *recognizer;
     44     UBool isDefaultEnabled;
     45 };
     46 
     47 U_NAMESPACE_END
     48 
     49 static icu::CSRecognizerInfo **fCSRecognizers = NULL;
     50 static icu::UInitOnce gCSRecognizersInitOnce;
     51 static int32_t fCSRecognizers_size = 0;
     52 
     53 U_CDECL_BEGIN
     54 static UBool U_CALLCONV csdet_cleanup(void)
     55 {
     56     U_NAMESPACE_USE
     57     if (fCSRecognizers != NULL) {
     58         for(int32_t r = 0; r < fCSRecognizers_size; r += 1) {
     59             delete fCSRecognizers[r];
     60             fCSRecognizers[r] = NULL;
     61         }
     62 
     63         DELETE_ARRAY(fCSRecognizers);
     64         fCSRecognizers = NULL;
     65         fCSRecognizers_size = 0;
     66     }
     67     gCSRecognizersInitOnce.reset();
     68 
     69     return TRUE;
     70 }
     71 
     72 static int32_t U_CALLCONV
     73 charsetMatchComparator(const void * /*context*/, const void *left, const void *right)
     74 {
     75     U_NAMESPACE_USE
     76 
     77     const CharsetMatch **csm_l = (const CharsetMatch **) left;
     78     const CharsetMatch **csm_r = (const CharsetMatch **) right;
     79 
     80     // NOTE: compare is backwards to sort from highest to lowest.
     81     return (*csm_r)->getConfidence() - (*csm_l)->getConfidence();
     82 }
     83 
     84 static void U_CALLCONV initRecognizers(UErrorCode &status) {
     85     U_NAMESPACE_USE
     86     ucln_i18n_registerCleanup(UCLN_I18N_CSDET, csdet_cleanup);
     87     CSRecognizerInfo *tempArray[] = {
     88         new CSRecognizerInfo(new CharsetRecog_UTF8(), TRUE),
     89 
     90         new CSRecognizerInfo(new CharsetRecog_UTF_16_BE(), TRUE),
     91         new CSRecognizerInfo(new CharsetRecog_UTF_16_LE(), TRUE),
     92         new CSRecognizerInfo(new CharsetRecog_UTF_32_BE(), TRUE),
     93         new CSRecognizerInfo(new CharsetRecog_UTF_32_LE(), TRUE),
     94 
     95         new CSRecognizerInfo(new CharsetRecog_8859_1(), TRUE),
     96         new CSRecognizerInfo(new CharsetRecog_8859_2(), TRUE),
     97         new CSRecognizerInfo(new CharsetRecog_8859_5_ru(), TRUE),
     98         new CSRecognizerInfo(new CharsetRecog_8859_6_ar(), TRUE),
     99         new CSRecognizerInfo(new CharsetRecog_8859_7_el(), TRUE),
    100         new CSRecognizerInfo(new CharsetRecog_8859_8_I_he(), TRUE),
    101         new CSRecognizerInfo(new CharsetRecog_8859_8_he(), TRUE),
    102         new CSRecognizerInfo(new CharsetRecog_windows_1251(), TRUE),
    103         new CSRecognizerInfo(new CharsetRecog_windows_1256(), TRUE),
    104         new CSRecognizerInfo(new CharsetRecog_KOI8_R(), TRUE),
    105         new CSRecognizerInfo(new CharsetRecog_8859_9_tr(), TRUE),
    106         new CSRecognizerInfo(new CharsetRecog_sjis(), TRUE),
    107         new CSRecognizerInfo(new CharsetRecog_gb_18030(), TRUE),
    108         new CSRecognizerInfo(new CharsetRecog_euc_jp(), TRUE),
    109         new CSRecognizerInfo(new CharsetRecog_euc_kr(), TRUE),
    110         new CSRecognizerInfo(new CharsetRecog_big5(), TRUE),
    111 
    112         new CSRecognizerInfo(new CharsetRecog_2022JP(), TRUE),
    113 #if !UCONFIG_ONLY_HTML_CONVERSION
    114         new CSRecognizerInfo(new CharsetRecog_2022KR(), TRUE),
    115         new CSRecognizerInfo(new CharsetRecog_2022CN(), TRUE),
    116 
    117         new CSRecognizerInfo(new CharsetRecog_IBM424_he_rtl(), FALSE),
    118         new CSRecognizerInfo(new CharsetRecog_IBM424_he_ltr(), FALSE),
    119         new CSRecognizerInfo(new CharsetRecog_IBM420_ar_rtl(), FALSE),
    120         new CSRecognizerInfo(new CharsetRecog_IBM420_ar_ltr(), FALSE)
    121 #endif
    122     };
    123     int32_t rCount = ARRAY_SIZE(tempArray);
    124 
    125     fCSRecognizers = NEW_ARRAY(CSRecognizerInfo *, rCount);
    126 
    127     if (fCSRecognizers == NULL) {
    128         status = U_MEMORY_ALLOCATION_ERROR;
    129     }
    130     else {
    131         fCSRecognizers_size = rCount;
    132         for (int32_t r = 0; r < rCount; r += 1) {
    133             fCSRecognizers[r] = tempArray[r];
    134             if (fCSRecognizers[r] == NULL) {
    135                 status = U_MEMORY_ALLOCATION_ERROR;
    136             }
    137         }
    138     }
    139 }
    140 
    141 U_CDECL_END
    142 
    143 U_NAMESPACE_BEGIN
    144 
    145 void CharsetDetector::setRecognizers(UErrorCode &status)
    146 {
    147     umtx_initOnce(gCSRecognizersInitOnce, &initRecognizers, status);
    148 }
    149 
    150 CharsetDetector::CharsetDetector(UErrorCode &status)
    151   : textIn(new InputText(status)), resultArray(NULL),
    152     resultCount(0), fStripTags(FALSE), fFreshTextSet(FALSE),
    153     fEnabledRecognizers(NULL)
    154 {
    155     if (U_FAILURE(status)) {
    156         return;
    157     }
    158 
    159     setRecognizers(status);
    160 
    161     if (U_FAILURE(status)) {
    162         return;
    163     }
    164 
    165     resultArray = (CharsetMatch **)uprv_malloc(sizeof(CharsetMatch *)*fCSRecognizers_size);
    166 
    167     if (resultArray == NULL) {
    168         status = U_MEMORY_ALLOCATION_ERROR;
    169         return;
    170     }
    171 
    172     for(int32_t i = 0; i < fCSRecognizers_size; i += 1) {
    173         resultArray[i] = new CharsetMatch();
    174 
    175         if (resultArray[i] == NULL) {
    176             status = U_MEMORY_ALLOCATION_ERROR;
    177             break;
    178         }
    179     }
    180 }
    181 
    182 CharsetDetector::~CharsetDetector()
    183 {
    184     delete textIn;
    185 
    186     for(int32_t i = 0; i < fCSRecognizers_size; i += 1) {
    187         delete resultArray[i];
    188     }
    189 
    190     uprv_free(resultArray);
    191 
    192     if (fEnabledRecognizers) {
    193         uprv_free(fEnabledRecognizers);
    194     }
    195 }
    196 
    197 void CharsetDetector::setText(const char *in, int32_t len)
    198 {
    199     textIn->setText(in, len);
    200     fFreshTextSet = TRUE;
    201 }
    202 
    203 UBool CharsetDetector::setStripTagsFlag(UBool flag)
    204 {
    205     UBool temp = fStripTags;
    206     fStripTags = flag;
    207     fFreshTextSet = TRUE;
    208     return temp;
    209 }
    210 
    211 UBool CharsetDetector::getStripTagsFlag() const
    212 {
    213     return fStripTags;
    214 }
    215 
    216 void CharsetDetector::setDeclaredEncoding(const char *encoding, int32_t len) const
    217 {
    218     textIn->setDeclaredEncoding(encoding,len);
    219 }
    220 
    221 int32_t CharsetDetector::getDetectableCount()
    222 {
    223     UErrorCode status = U_ZERO_ERROR;
    224 
    225     setRecognizers(status);
    226 
    227     return fCSRecognizers_size;
    228 }
    229 
    230 const CharsetMatch *CharsetDetector::detect(UErrorCode &status)
    231 {
    232     int32_t maxMatchesFound = 0;
    233 
    234     detectAll(maxMatchesFound, status);
    235 
    236     if(maxMatchesFound > 0) {
    237         return resultArray[0];
    238     } else {
    239         return NULL;
    240     }
    241 }
    242 
    243 const CharsetMatch * const *CharsetDetector::detectAll(int32_t &maxMatchesFound, UErrorCode &status)
    244 {
    245     if(!textIn->isSet()) {
    246         status = U_MISSING_RESOURCE_ERROR;// TODO:  Need to set proper status code for input text not set
    247 
    248         return NULL;
    249     } else if (fFreshTextSet) {
    250         CharsetRecognizer *csr;
    251         int32_t            i;
    252 
    253         textIn->MungeInput(fStripTags);
    254 
    255         // Iterate over all possible charsets, remember all that
    256         // give a match quality > 0.
    257         resultCount = 0;
    258         for (i = 0; i < fCSRecognizers_size; i += 1) {
    259             csr = fCSRecognizers[i]->recognizer;
    260             if (csr->match(textIn, resultArray[resultCount])) {
    261                 resultCount++;
    262             }
    263         }
    264 
    265         if (resultCount > 1) {
    266             uprv_sortArray(resultArray, resultCount, sizeof resultArray[0], charsetMatchComparator, NULL, TRUE, &status);
    267         }
    268         fFreshTextSet = FALSE;
    269     }
    270 
    271     maxMatchesFound = resultCount;
    272 
    273     return resultArray;
    274 }
    275 
    276 void CharsetDetector::setDetectableCharset(const char *encoding, UBool enabled, UErrorCode &status)
    277 {
    278     if (U_FAILURE(status)) {
    279         return;
    280     }
    281 
    282     int32_t modIdx = -1;
    283     UBool isDefaultVal = FALSE;
    284     for (int32_t i = 0; i < fCSRecognizers_size; i++) {
    285         CSRecognizerInfo *csrinfo = fCSRecognizers[i];
    286         if (uprv_strcmp(csrinfo->recognizer->getName(), encoding) == 0) {
    287             modIdx = i;
    288             isDefaultVal = (csrinfo->isDefaultEnabled == enabled);
    289             break;
    290         }
    291     }
    292     if (modIdx < 0) {
    293         // No matching encoding found
    294         status = U_ILLEGAL_ARGUMENT_ERROR;
    295         return;
    296     }
    297 
    298     if (fEnabledRecognizers == NULL && !isDefaultVal) {
    299         // Create an array storing the non default setting
    300         fEnabledRecognizers = NEW_ARRAY(UBool, fCSRecognizers_size);
    301         if (fEnabledRecognizers == NULL) {
    302             status = U_MEMORY_ALLOCATION_ERROR;
    303             return;
    304         }
    305         // Initialize the array with default info
    306         for (int32_t i = 0; i < fCSRecognizers_size; i++) {
    307             fEnabledRecognizers[i] = fCSRecognizers[i]->isDefaultEnabled;
    308         }
    309     }
    310 
    311     if (fEnabledRecognizers != NULL) {
    312         fEnabledRecognizers[modIdx] = enabled;
    313     }
    314 }
    315 
    316 /*const char *CharsetDetector::getCharsetName(int32_t index, UErrorCode &status) const
    317 {
    318     if( index > fCSRecognizers_size-1 || index < 0) {
    319         status = U_INDEX_OUTOFBOUNDS_ERROR;
    320 
    321         return 0;
    322     } else {
    323         return fCSRecognizers[index]->getName();
    324     }
    325 }*/
    326 
    327 U_NAMESPACE_END
    328 
    329 U_CDECL_BEGIN
    330 typedef struct {
    331     int32_t currIndex;
    332     UBool all;
    333     UBool *enabledRecognizers;
    334 } Context;
    335 
    336 
    337 
    338 static void U_CALLCONV
    339 enumClose(UEnumeration *en) {
    340     if(en->context != NULL) {
    341         DELETE_ARRAY(en->context);
    342     }
    343 
    344     DELETE_ARRAY(en);
    345 }
    346 
    347 static int32_t U_CALLCONV
    348 enumCount(UEnumeration *en, UErrorCode *) {
    349     if (((Context *)en->context)->all) {
    350         // ucsdet_getAllDetectableCharsets, all charset detector names
    351         return fCSRecognizers_size;
    352     }
    353 
    354     // Otherwise, ucsdet_getDetectableCharsets - only enabled ones
    355     int32_t count = 0;
    356     UBool *enabledArray = ((Context *)en->context)->enabledRecognizers;
    357     if (enabledArray != NULL) {
    358         // custom set
    359         for (int32_t i = 0; i < fCSRecognizers_size; i++) {
    360             if (enabledArray[i]) {
    361                 count++;
    362             }
    363         }
    364     } else {
    365         // default set
    366         for (int32_t i = 0; i < fCSRecognizers_size; i++) {
    367             if (fCSRecognizers[i]->isDefaultEnabled) {
    368                 count++;
    369             }
    370         }
    371     }
    372     return count;
    373 }
    374 
    375 static const char* U_CALLCONV
    376 enumNext(UEnumeration *en, int32_t *resultLength, UErrorCode * /*status*/) {
    377     const char *currName = NULL;
    378 
    379     if (((Context *)en->context)->currIndex < fCSRecognizers_size) {
    380         if (((Context *)en->context)->all) {
    381             // ucsdet_getAllDetectableCharsets, all charset detector names
    382             currName = fCSRecognizers[((Context *)en->context)->currIndex]->recognizer->getName();
    383             ((Context *)en->context)->currIndex++;
    384         } else {
    385             // ucsdet_getDetectableCharsets
    386             UBool *enabledArray = ((Context *)en->context)->enabledRecognizers;
    387             if (enabledArray != NULL) {
    388                 // custome set
    389                 while (currName == NULL && ((Context *)en->context)->currIndex < fCSRecognizers_size) {
    390                     if (enabledArray[((Context *)en->context)->currIndex]) {
    391                         currName = fCSRecognizers[((Context *)en->context)->currIndex]->recognizer->getName();
    392                     }
    393                     ((Context *)en->context)->currIndex++;
    394                 }
    395             } else {
    396                 // default set
    397                 while (currName == NULL && ((Context *)en->context)->currIndex < fCSRecognizers_size) {
    398                     if (fCSRecognizers[((Context *)en->context)->currIndex]->isDefaultEnabled) {
    399                         currName = fCSRecognizers[((Context *)en->context)->currIndex]->recognizer->getName();
    400                     }
    401                     ((Context *)en->context)->currIndex++;
    402                 }
    403             }
    404         }
    405     }
    406 
    407     if(resultLength != NULL) {
    408         *resultLength = currName == NULL ? 0 : (int32_t)uprv_strlen(currName);
    409     }
    410 
    411     return currName;
    412 }
    413 
    414 
    415 static void U_CALLCONV
    416 enumReset(UEnumeration *en, UErrorCode *) {
    417     ((Context *)en->context)->currIndex = 0;
    418 }
    419 
    420 static const UEnumeration gCSDetEnumeration = {
    421     NULL,
    422     NULL,
    423     enumClose,
    424     enumCount,
    425     uenum_unextDefault,
    426     enumNext,
    427     enumReset
    428 };
    429 
    430 U_CDECL_END
    431 
    432 U_NAMESPACE_BEGIN
    433 
    434 UEnumeration * CharsetDetector::getAllDetectableCharsets(UErrorCode &status)
    435 {
    436 
    437     /* Initialize recognized charsets. */
    438     setRecognizers(status);
    439 
    440     if(U_FAILURE(status)) {
    441         return 0;
    442     }
    443 
    444     UEnumeration *en = NEW_ARRAY(UEnumeration, 1);
    445     if (en == NULL) {
    446         status = U_MEMORY_ALLOCATION_ERROR;
    447         return 0;
    448     }
    449     memcpy(en, &gCSDetEnumeration, sizeof(UEnumeration));
    450     en->context = (void*)NEW_ARRAY(Context, 1);
    451     if (en->context == NULL) {
    452         status = U_MEMORY_ALLOCATION_ERROR;
    453         DELETE_ARRAY(en);
    454         return 0;
    455     }
    456     uprv_memset(en->context, 0, sizeof(Context));
    457     ((Context*)en->context)->all = TRUE;
    458     return en;
    459 }
    460 
    461 UEnumeration * CharsetDetector::getDetectableCharsets(UErrorCode &status) const
    462 {
    463     if(U_FAILURE(status)) {
    464         return 0;
    465     }
    466 
    467     UEnumeration *en = NEW_ARRAY(UEnumeration, 1);
    468     if (en == NULL) {
    469         status = U_MEMORY_ALLOCATION_ERROR;
    470         return 0;
    471     }
    472     memcpy(en, &gCSDetEnumeration, sizeof(UEnumeration));
    473     en->context = (void*)NEW_ARRAY(Context, 1);
    474     if (en->context == NULL) {
    475         status = U_MEMORY_ALLOCATION_ERROR;
    476         DELETE_ARRAY(en);
    477         return 0;
    478     }
    479     uprv_memset(en->context, 0, sizeof(Context));
    480     ((Context*)en->context)->all = FALSE;
    481     ((Context*)en->context)->enabledRecognizers = fEnabledRecognizers;
    482     return en;
    483 }
    484 
    485 U_NAMESPACE_END
    486 
    487 #endif
    488