Home | History | Annotate | Download | only in i18n
      1 /*
      2 ******************************************************************************
      3 *
      4 *   Copyright (C) 2008-2010, International Business Machines
      5 *   Corporation and others.  All Rights Reserved.
      6 *
      7 ******************************************************************************
      8 *   file name:  uspoof_conf.cpp
      9 *   encoding:   US-ASCII
     10 *   tab size:   8 (not used)
     11 *   indentation:4
     12 *
     13 *   created on: 2009Jan05  (refactoring earlier files)
     14 *   created by: Andy Heninger
     15 *
     16 *   Internal classes for compililing confusable data into its binary (runtime) form.
     17 */
     18 
     19 #include "unicode/utypes.h"
     20 #include "unicode/uspoof.h"
     21 #if !UCONFIG_NO_REGULAR_EXPRESSIONS
     22 #if !UCONFIG_NO_NORMALIZATION
     23 
     24 #include "unicode/unorm.h"
     25 #include "unicode/uregex.h"
     26 #include "unicode/ustring.h"
     27 #include "cmemory.h"
     28 #include "uspoof_impl.h"
     29 #include "uhash.h"
     30 #include "uvector.h"
     31 #include "uassert.h"
     32 #include "uarrsort.h"
     33 #include "uspoof_conf.h"
     34 
     35 U_NAMESPACE_USE
     36 
     37 
     38 //---------------------------------------------------------------------
     39 //
     40 //  buildConfusableData   Compile the source confusable data, as defined by
     41 //                        the Unicode data file confusables.txt, into the binary
     42 //                        structures used by the confusable detector.
     43 //
     44 //                        The binary structures are described in uspoof_impl.h
     45 //
     46 //     1.  parse the data, building 4 hash tables, one each for the SL, SA, ML and MA
     47 //         tables.  Each maps from a UChar32 to a String.
     48 //
     49 //     2.  Sort all of the strings encountered by length, since they will need to
     50 //         be stored in that order in the final string table.
     51 //
     52 //     3.  Build a list of keys (UChar32s) from the four mapping tables.  Sort the
     53 //         list because that will be the ordering of our runtime table.
     54 //
     55 //     4.  Generate the run time string table.  This is generated before the key & value
     56 //         tables because we need the string indexes when building those tables.
     57 //
     58 //     5.  Build the run-time key and value tables.  These are parallel tables, and are built
     59 //         at the same time
     60 //
     61 
     62 SPUString::SPUString(UnicodeString *s) {
     63     fStr = s;
     64     fStrTableIndex = 0;
     65 }
     66 
     67 
     68 SPUString::~SPUString() {
     69     delete fStr;
     70 }
     71 
     72 
     73 SPUStringPool::SPUStringPool(UErrorCode &status) : fVec(NULL), fHash(NULL) {
     74     fVec = new UVector(status);
     75     fHash = uhash_open(uhash_hashUnicodeString,           // key hash function
     76                        uhash_compareUnicodeString,        // Key Comparator
     77                        NULL,                              // Value Comparator
     78                        &status);
     79 }
     80 
     81 
     82 SPUStringPool::~SPUStringPool() {
     83     int i;
     84     for (i=fVec->size()-1; i>=0; i--) {
     85         SPUString *s = static_cast<SPUString *>(fVec->elementAt(i));
     86         delete s;
     87     }
     88     delete fVec;
     89     uhash_close(fHash);
     90 }
     91 
     92 
     93 int32_t SPUStringPool::size() {
     94     return fVec->size();
     95 }
     96 
     97 SPUString *SPUStringPool::getByIndex(int32_t index) {
     98     SPUString *retString = (SPUString *)fVec->elementAt(index);
     99     return retString;
    100 }
    101 
    102 
    103 // Comparison function for ordering strings in the string pool.
    104 // Compare by length first, then, within a group of the same length,
    105 // by code point order.
    106 // Conforms to the type signature for a USortComparator in uvector.h
    107 
    108 static int8_t U_CALLCONV SPUStringCompare(UHashTok left, UHashTok right) {
    109 	const SPUString *sL = const_cast<const SPUString *>(
    110         static_cast<SPUString *>(left.pointer));
    111  	const SPUString *sR = const_cast<const SPUString *>(
    112  	    static_cast<SPUString *>(right.pointer));
    113     int32_t lenL = sL->fStr->length();
    114     int32_t lenR = sR->fStr->length();
    115     if (lenL < lenR) {
    116         return -1;
    117     } else if (lenL > lenR) {
    118         return 1;
    119     } else {
    120         return sL->fStr->compare(*(sR->fStr));
    121     }
    122 }
    123 
    124 void SPUStringPool::sort(UErrorCode &status) {
    125     fVec->sort(SPUStringCompare, status);
    126 }
    127 
    128 
    129 SPUString *SPUStringPool::addString(UnicodeString *src, UErrorCode &status) {
    130     SPUString *hashedString = static_cast<SPUString *>(uhash_get(fHash, src));
    131     if (hashedString != NULL) {
    132         delete src;
    133     } else {
    134         hashedString = new SPUString(src);
    135         uhash_put(fHash, src, hashedString, &status);
    136         fVec->addElement(hashedString, status);
    137     }
    138     return hashedString;
    139 }
    140 
    141 
    142 
    143 ConfusabledataBuilder::ConfusabledataBuilder(SpoofImpl *spImpl, UErrorCode &status) :
    144     fSpoofImpl(spImpl),
    145     fInput(NULL),
    146     fSLTable(NULL),
    147     fSATable(NULL),
    148     fMLTable(NULL),
    149     fMATable(NULL),
    150     fKeySet(NULL),
    151     fKeyVec(NULL),
    152     fValueVec(NULL),
    153     fStringTable(NULL),
    154     fStringLengthsTable(NULL),
    155     stringPool(NULL),
    156     fParseLine(NULL),
    157     fParseHexNum(NULL),
    158     fLineNum(0)
    159 {
    160     if (U_FAILURE(status)) {
    161         return;
    162     }
    163     fSLTable    = uhash_open(uhash_hashLong, uhash_compareLong, NULL, &status);
    164     fSATable    = uhash_open(uhash_hashLong, uhash_compareLong, NULL, &status);
    165     fMLTable    = uhash_open(uhash_hashLong, uhash_compareLong, NULL, &status);
    166     fMATable    = uhash_open(uhash_hashLong, uhash_compareLong, NULL, &status);
    167     fKeySet     = new UnicodeSet();
    168     fKeyVec     = new UVector(status);
    169     fValueVec   = new UVector(status);
    170     stringPool = new SPUStringPool(status);
    171 }
    172 
    173 
    174 ConfusabledataBuilder::~ConfusabledataBuilder() {
    175     uprv_free(fInput);
    176     uregex_close(fParseLine);
    177     uregex_close(fParseHexNum);
    178     uhash_close(fSLTable);
    179     uhash_close(fSATable);
    180     uhash_close(fMLTable);
    181     uhash_close(fMATable);
    182     delete fKeySet;
    183     delete fKeyVec;
    184     delete fStringTable;
    185     delete fStringLengthsTable;
    186     delete fValueVec;
    187     delete stringPool;
    188 }
    189 
    190 
    191 void ConfusabledataBuilder::buildConfusableData(SpoofImpl * spImpl, const char * confusables,
    192     int32_t confusablesLen, int32_t *errorType, UParseError *pe, UErrorCode &status) {
    193 
    194     if (U_FAILURE(status)) {
    195         return;
    196     }
    197     ConfusabledataBuilder builder(spImpl, status);
    198     builder.build(confusables, confusablesLen, status);
    199     if (U_FAILURE(status) && errorType != NULL) {
    200         *errorType = USPOOF_SINGLE_SCRIPT_CONFUSABLE;
    201         pe->line = builder.fLineNum;
    202     }
    203 }
    204 
    205 
    206 void ConfusabledataBuilder::build(const char * confusables, int32_t confusablesLen,
    207                UErrorCode &status) {
    208 
    209     // Convert the user input data from UTF-8 to UChar (UTF-16)
    210     int32_t inputLen = 0;
    211     if (U_FAILURE(status)) {
    212         return;
    213     }
    214     u_strFromUTF8(NULL, 0, &inputLen, confusables, confusablesLen, &status);
    215     if (status != U_BUFFER_OVERFLOW_ERROR) {
    216         return;
    217     }
    218     status = U_ZERO_ERROR;
    219     fInput = static_cast<UChar *>(uprv_malloc((inputLen+1) * sizeof(UChar)));
    220     if (fInput == NULL) {
    221         status = U_MEMORY_ALLOCATION_ERROR;
    222     }
    223     u_strFromUTF8(fInput, inputLen+1, NULL, confusables, confusablesLen, &status);
    224 
    225 
    226     // Regular Expression to parse a line from Confusables.txt.  The expression will match
    227     // any line.  What was matched is determined by examining which capture groups have a match.
    228     //   Capture Group 1:  the source char
    229     //   Capture Group 2:  the replacement chars
    230     //   Capture Group 3-6  the table type, SL, SA, ML, or MA
    231     //   Capture Group 7:  A blank or comment only line.
    232     //   Capture Group 8:  A syntactically invalid line.  Anything that didn't match before.
    233     // Example Line from the confusables.txt source file:
    234     //   "1D702 ;	006E 0329 ;	SL	# MATHEMATICAL ITALIC SMALL ETA ... "
    235     fParseLine = uregex_openC(
    236         "(?m)^[ \\t]*([0-9A-Fa-f]+)[ \\t]+;"      // Match the source char
    237         "[ \\t]*([0-9A-Fa-f]+"                    // Match the replacement char(s)
    238            "(?:[ \\t]+[0-9A-Fa-f]+)*)[ \\t]*;"    //     (continued)
    239         "\\s*(?:(SL)|(SA)|(ML)|(MA))"             // Match the table type
    240         "[ \\t]*(?:#.*?)?$"                       // Match any trailing #comment
    241         "|^([ \\t]*(?:#.*?)?)$"       // OR match empty lines or lines with only a #comment
    242         "|^(.*?)$",                   // OR match any line, which catches illegal lines.
    243         0, NULL, &status);
    244 
    245     // Regular expression for parsing a hex number out of a space-separated list of them.
    246     //   Capture group 1 gets the number, with spaces removed.
    247     fParseHexNum = uregex_openC("\\s*([0-9A-F]+)", 0, NULL, &status);
    248 
    249     // Zap any Byte Order Mark at the start of input.  Changing it to a space is benign
    250     //   given the syntax of the input.
    251     if (*fInput == 0xfeff) {
    252         *fInput = 0x20;
    253     }
    254 
    255     // Parse the input, one line per iteration of this loop.
    256     uregex_setText(fParseLine, fInput, inputLen, &status);
    257     while (uregex_findNext(fParseLine, &status)) {
    258         fLineNum++;
    259         if (uregex_start(fParseLine, 7, &status) >= 0) {
    260             // this was a blank or comment line.
    261             continue;
    262         }
    263         if (uregex_start(fParseLine, 8, &status) >= 0) {
    264             // input file syntax error.
    265             status = U_PARSE_ERROR;
    266             return;
    267         }
    268 
    269         // We have a good input line.  Extract the key character and mapping string, and
    270         //    put them into the appropriate mapping table.
    271         UChar32 keyChar = SpoofImpl::ScanHex(fInput, uregex_start(fParseLine, 1, &status),
    272                           uregex_end(fParseLine, 1, &status), status);
    273 
    274         int32_t mapStringStart = uregex_start(fParseLine, 2, &status);
    275         int32_t mapStringLength = uregex_end(fParseLine, 2, &status) - mapStringStart;
    276         uregex_setText(fParseHexNum, &fInput[mapStringStart], mapStringLength, &status);
    277 
    278         UnicodeString  *mapString = new UnicodeString();
    279         if (mapString == NULL) {
    280             status = U_MEMORY_ALLOCATION_ERROR;
    281             return;
    282         }
    283         while (uregex_findNext(fParseHexNum, &status)) {
    284             UChar32 c = SpoofImpl::ScanHex(&fInput[mapStringStart], uregex_start(fParseHexNum, 1, &status),
    285                                  uregex_end(fParseHexNum, 1, &status), status);
    286             mapString->append(c);
    287         }
    288         U_ASSERT(mapString->length() >= 1);
    289 
    290         // Put the map (value) string into the string pool
    291         // This a little like a Java intern() - any duplicates will be eliminated.
    292         SPUString *smapString = stringPool->addString(mapString, status);
    293 
    294         // Add the UChar32 -> string mapping to the appropriate table.
    295         UHashtable *table = uregex_start(fParseLine, 3, &status) >= 0 ? fSLTable :
    296                             uregex_start(fParseLine, 4, &status) >= 0 ? fSATable :
    297                             uregex_start(fParseLine, 5, &status) >= 0 ? fMLTable :
    298                             uregex_start(fParseLine, 6, &status) >= 0 ? fMATable :
    299                             NULL;
    300         U_ASSERT(table != NULL);
    301         uhash_iput(table, keyChar, smapString, &status);
    302         fKeySet->add(keyChar);
    303         if (U_FAILURE(status)) {
    304             return;
    305         }
    306     }
    307 
    308     // Input data is now all parsed and collected.
    309     // Now create the run-time binary form of the data.
    310     //
    311     // This is done in two steps.  First the data is assembled into vectors and strings,
    312     //   for ease of construction, then the contents of these collections are dumped
    313     //   into the actual raw-bytes data storage.
    314 
    315     // Build up the string array, and record the index of each string therein
    316     //  in the (build time only) string pool.
    317     // Strings of length one are not entered into the strings array.
    318     // At the same time, build up the string lengths table, which records the
    319     // position in the string table of the first string of each length >= 4.
    320     // (Strings in the table are sorted by length)
    321     stringPool->sort(status);
    322     fStringTable = new UnicodeString();
    323     fStringLengthsTable = new UVector(status);
    324     int32_t previousStringLength = 0;
    325     int32_t previousStringIndex  = 0;
    326     int32_t poolSize = stringPool->size();
    327     int32_t i;
    328     for (i=0; i<poolSize; i++) {
    329         SPUString *s = stringPool->getByIndex(i);
    330         int32_t strLen = s->fStr->length();
    331         int32_t strIndex = fStringTable->length();
    332         U_ASSERT(strLen >= previousStringLength);
    333         if (strLen == 1) {
    334             // strings of length one do not get an entry in the string table.
    335             // Keep the single string character itself here, which is the same
    336             //  convention that is used in the final run-time string table index.
    337             s->fStrTableIndex = s->fStr->charAt(0);
    338         } else {
    339             if ((strLen > previousStringLength) && (previousStringLength >= 4)) {
    340                 fStringLengthsTable->addElement(previousStringIndex, status);
    341                 fStringLengthsTable->addElement(previousStringLength, status);
    342             }
    343             s->fStrTableIndex = strIndex;
    344             fStringTable->append(*(s->fStr));
    345         }
    346         previousStringLength = strLen;
    347         previousStringIndex  = strIndex;
    348     }
    349     // Make the final entry to the string lengths table.
    350     //   (it holds an entry for the _last_ string of each length, so adding the
    351     //    final one doesn't happen in the main loop because no longer string was encountered.)
    352     if (previousStringLength >= 4) {
    353         fStringLengthsTable->addElement(previousStringIndex, status);
    354         fStringLengthsTable->addElement(previousStringLength, status);
    355     }
    356 
    357     // Construct the compile-time Key and Value tables
    358     //
    359     // For each key code point, check which mapping tables it applies to,
    360     //   and create the final data for the key & value structures.
    361     //
    362     //   The four logical mapping tables are conflated into one combined table.
    363     //   If multiple logical tables have the same mapping for some key, they
    364     //     share a single entry in the combined table.
    365     //   If more than one mapping exists for the same key code point, multiple
    366     //     entries will be created in the table
    367 
    368     for (int32_t range=0; range<fKeySet->getRangeCount(); range++) {
    369         // It is an oddity of the UnicodeSet API that simply enumerating the contained
    370         //   code points requires a nested loop.
    371         for (UChar32 keyChar=fKeySet->getRangeStart(range);
    372                 keyChar <= fKeySet->getRangeEnd(range); keyChar++) {
    373             addKeyEntry(keyChar, fSLTable, USPOOF_SL_TABLE_FLAG, status);
    374             addKeyEntry(keyChar, fSATable, USPOOF_SA_TABLE_FLAG, status);
    375             addKeyEntry(keyChar, fMLTable, USPOOF_ML_TABLE_FLAG, status);
    376             addKeyEntry(keyChar, fMATable, USPOOF_MA_TABLE_FLAG, status);
    377         }
    378     }
    379 
    380     // Put the assembled data into the flat runtime array
    381     outputData(status);
    382 
    383     // All of the intermediate allocated data belongs to the ConfusabledataBuilder
    384     //  object  (this), and is deleted in the destructor.
    385     return;
    386 }
    387 
    388 //
    389 // outputData     The confusable data has been compiled and stored in intermediate
    390 //                collections and strings.  Copy it from there to the final flat
    391 //                binary array.
    392 //
    393 //                Note that as each section is added to the output data, the
    394 //                expand (reserveSpace() function will likely relocate it in memory.
    395 //                Be careful with pointers.
    396 //
    397 void ConfusabledataBuilder::outputData(UErrorCode &status) {
    398 
    399     U_ASSERT(fSpoofImpl->fSpoofData->fDataOwned == TRUE);
    400 
    401     //  The Key Table
    402     //     While copying the keys to the runtime array,
    403     //       also sanity check that they are sorted.
    404 
    405     int32_t numKeys = fKeyVec->size();
    406     int32_t *keys =
    407         static_cast<int32_t *>(fSpoofImpl->fSpoofData->reserveSpace(numKeys*sizeof(int32_t), status));
    408     if (U_FAILURE(status)) {
    409         return;
    410     }
    411     int i;
    412     int32_t previousKey = 0;
    413     for (i=0; i<numKeys; i++) {
    414         int32_t key =  fKeyVec->elementAti(i);
    415         U_ASSERT((key & 0x00ffffff) >= (previousKey & 0x00ffffff));
    416         U_ASSERT((key & 0xff000000) != 0);
    417         keys[i] = key;
    418         previousKey = key;
    419     }
    420     SpoofDataHeader *rawData = fSpoofImpl->fSpoofData->fRawData;
    421     rawData->fCFUKeys = (int32_t)((char *)keys - (char *)rawData);
    422     rawData->fCFUKeysSize = numKeys;
    423     fSpoofImpl->fSpoofData->fCFUKeys = keys;
    424 
    425 
    426     // The Value Table, parallels the key table
    427     int32_t numValues = fValueVec->size();
    428     U_ASSERT(numKeys == numValues);
    429     uint16_t *values =
    430         static_cast<uint16_t *>(fSpoofImpl->fSpoofData->reserveSpace(numKeys*sizeof(uint16_t), status));
    431     if (U_FAILURE(status)) {
    432         return;
    433     }
    434     for (i=0; i<numValues; i++) {
    435         uint32_t value = static_cast<uint32_t>(fValueVec->elementAti(i));
    436         U_ASSERT(value < 0xffff);
    437         values[i] = static_cast<uint16_t>(value);
    438     }
    439     rawData = fSpoofImpl->fSpoofData->fRawData;
    440     rawData->fCFUStringIndex = (int32_t)((char *)values - (char *)rawData);
    441     rawData->fCFUStringIndexSize = numValues;
    442     fSpoofImpl->fSpoofData->fCFUValues = values;
    443 
    444     // The Strings Table.
    445 
    446     uint32_t stringsLength = fStringTable->length();
    447     // Reserve an extra space so the string will be nul-terminated.  This is
    448     // only a convenience, for when debugging; it is not needed otherwise.
    449     UChar *strings =
    450         static_cast<UChar *>(fSpoofImpl->fSpoofData->reserveSpace(stringsLength*sizeof(UChar)+2, status));
    451     if (U_FAILURE(status)) {
    452         return;
    453     }
    454     fStringTable->extract(strings, stringsLength+1, status);
    455     rawData = fSpoofImpl->fSpoofData->fRawData;
    456     U_ASSERT(rawData->fCFUStringTable == 0);
    457     rawData->fCFUStringTable = (int32_t)((char *)strings - (char *)rawData);
    458     rawData->fCFUStringTableLen = stringsLength;
    459     fSpoofImpl->fSpoofData->fCFUStrings = strings;
    460 
    461     // The String Lengths Table
    462     //    While copying into the runtime array do some sanity checks on the values
    463     //    Each complete entry contains two fields, an index and an offset.
    464     //    Lengths should increase with each entry.
    465     //    Offsets should be less than the size of the string table.
    466     int32_t lengthTableLength = fStringLengthsTable->size();
    467     uint16_t *stringLengths =
    468         static_cast<uint16_t *>(fSpoofImpl->fSpoofData->reserveSpace(lengthTableLength*sizeof(uint16_t), status));
    469     if (U_FAILURE(status)) {
    470         return;
    471     }
    472     int32_t destIndex = 0;
    473     uint32_t previousLength = 0;
    474     for (i=0; i<lengthTableLength; i+=2) {
    475         uint32_t offset = static_cast<uint32_t>(fStringLengthsTable->elementAti(i));
    476         uint32_t length = static_cast<uint32_t>(fStringLengthsTable->elementAti(i+1));
    477         U_ASSERT(offset < stringsLength);
    478         U_ASSERT(length < 40);
    479         U_ASSERT(length > previousLength);
    480         stringLengths[destIndex++] = static_cast<uint16_t>(offset);
    481         stringLengths[destIndex++] = static_cast<uint16_t>(length);
    482         previousLength = length;
    483     }
    484     rawData = fSpoofImpl->fSpoofData->fRawData;
    485     rawData->fCFUStringLengths = (int32_t)((char *)stringLengths - (char *)rawData);
    486     // Note: StringLengthsSize in the raw data is the number of complete entries,
    487     //       each consisting of a pair of 16 bit values, hence the divide by 2.
    488     rawData->fCFUStringLengthsSize = lengthTableLength / 2;
    489     fSpoofImpl->fSpoofData->fCFUStringLengths =
    490         reinterpret_cast<SpoofStringLengthsElement *>(stringLengths);
    491 }
    492 
    493 
    494 
    495 //  addKeyEntry   Construction of the confusable Key and Mapping Values tables.
    496 //                This is an intermediate point in the building process.
    497 //                We already have the mappings in the hash tables fSLTable, etc.
    498 //                This function builds corresponding run-time style table entries into
    499 //                  fKeyVec and fValueVec
    500 
    501 void ConfusabledataBuilder::addKeyEntry(
    502     UChar32     keyChar,     // The key character
    503     UHashtable *table,       // The table, one of SATable, MATable, etc.
    504     int32_t     tableFlag,   // One of USPOOF_SA_TABLE_FLAG, etc.
    505     UErrorCode &status) {
    506 
    507     SPUString *targetMapping = static_cast<SPUString *>(uhash_iget(table, keyChar));
    508     if (targetMapping == NULL) {
    509         // No mapping for this key character.
    510         //   (This function is called for all four tables for each key char that
    511         //    is seen anywhere, so this no entry cases are very much expected.)
    512         return;
    513     }
    514 
    515     // Check whether there is already an entry with the correct mapping.
    516     // If so, simply set the flag in the keyTable saying that the existing entry
    517     // applies to the table that we're doing now.
    518 
    519     UBool keyHasMultipleValues = FALSE;
    520     int32_t i;
    521     for (i=fKeyVec->size()-1; i>=0 ; i--) {
    522         int32_t key = fKeyVec->elementAti(i);
    523         if ((key & 0x0ffffff) != keyChar) {
    524             // We have now checked all existing key entries for this key char (if any)
    525             //  without finding one with the same mapping.
    526             break;
    527         }
    528         UnicodeString mapping = getMapping(i);
    529         if (mapping == *(targetMapping->fStr)) {
    530             // The run time entry we are currently testing has the correct mapping.
    531             // Set the flag in it indicating that it applies to the new table also.
    532             key |= tableFlag;
    533             fKeyVec->setElementAt(key, i);
    534             return;
    535         }
    536         keyHasMultipleValues = TRUE;
    537     }
    538 
    539     // Need to add a new entry to the binary data being built for this mapping.
    540     // Includes adding entries to both the key table and the parallel values table.
    541 
    542     int32_t newKey = keyChar | tableFlag;
    543     if (keyHasMultipleValues) {
    544         newKey |= USPOOF_KEY_MULTIPLE_VALUES;
    545     }
    546     int32_t adjustedMappingLength = targetMapping->fStr->length() - 1;
    547     if (adjustedMappingLength>3) {
    548         adjustedMappingLength = 3;
    549     }
    550     newKey |= adjustedMappingLength << USPOOF_KEY_LENGTH_SHIFT;
    551 
    552     int32_t newData = targetMapping->fStrTableIndex;
    553 
    554     fKeyVec->addElement(newKey, status);
    555     fValueVec->addElement(newData, status);
    556 
    557     // If the preceding key entry is for the same key character (but with a different mapping)
    558     //   set the multiple-values flag on it.
    559     if (keyHasMultipleValues) {
    560         int32_t previousKeyIndex = fKeyVec->size() - 2;
    561         int32_t previousKey = fKeyVec->elementAti(previousKeyIndex);
    562         previousKey |= USPOOF_KEY_MULTIPLE_VALUES;
    563         fKeyVec->setElementAt(previousKey, previousKeyIndex);
    564     }
    565 }
    566 
    567 
    568 
    569 UnicodeString ConfusabledataBuilder::getMapping(int32_t index) {
    570     int32_t key = fKeyVec->elementAti(index);
    571     int32_t value = fValueVec->elementAti(index);
    572     int32_t length = USPOOF_KEY_LENGTH_FIELD(key);
    573     int32_t lastIndexWithLen;
    574     switch (length) {
    575       case 0:
    576         return UnicodeString(static_cast<UChar>(value));
    577       case 1:
    578       case 2:
    579         return UnicodeString(*fStringTable, value, length+1);
    580       case 3:
    581         length = 0;
    582         int32_t i;
    583         for (i=0; i<fStringLengthsTable->size(); i+=2) {
    584             lastIndexWithLen = fStringLengthsTable->elementAti(i);
    585             if (value <= lastIndexWithLen) {
    586                 length = fStringLengthsTable->elementAti(i+1);
    587                 break;
    588             }
    589         }
    590         U_ASSERT(length>=3);
    591         return UnicodeString(*fStringTable, value, length);
    592       default:
    593         U_ASSERT(FALSE);
    594     }
    595     return UnicodeString();
    596 }
    597 
    598 #endif
    599 #endif // !UCONFIG_NO_REGULAR_EXPRESSIONS
    600 
    601