| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622 | //-----------------------------------------------------------------------------// Copyright (c) 2013 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#include "unicode.h"#include "memory/frameAllocator.h"#include "math/mMath.h"#include "debug/profiler.h"#include <stdio.h>//-----------------------------------------------------------------------------/// replacement character. Standard correct value is 0xFFFD.#define kReplacementChar 0xFFFD/// Look up table. Shift a byte >> 1, then look up how many bytes to expect after it./// Contains -1's for illegal values.U8 firstByteLUT[128] = {   1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1, // 0x0F // single byte ascii   1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1, // 0x1F // single byte ascii   1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1, // 0x2F // single byte ascii   1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1,  1, 1, 1, 1, // 0x3F // single byte ascii   0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0, // 0x4F // trailing utf8   0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0, // 0x5F // trailing utf8   2, 2, 2, 2,  2, 2, 2, 2,  2, 2, 2, 2,  2, 2, 2, 2, // 0x6F // first of 2   3, 3, 3, 3,  3, 3, 3, 3,  4, 4, 4, 4,  5, 5, 6, 0, // 0x7F // first of 3,4,5,illegal in utf-8};/// Look up table. Shift a 16-bit word >> 10, then look up whether it is a surrogate,///  and which part. 0 means non-surrogate, 1 means 1st in pair, 2 means 2nd in pair.U8 surrogateLUT[64] = {   0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0, // 0x0F    0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0, // 0x1F    0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0,  0, 0, 0, 0, // 0x2F    0, 0, 0, 0,  0, 0, 1, 2,  0, 0, 0, 0,  0, 0, 0, 0, // 0x3F };/// Look up table. Feed value from firstByteLUT in, gives you/// the mask for the data bits of that UTF-8 code unit.U8  byteMask8LUT[]  = { 0x3f, 0x7f, 0x1f, 0x0f, 0x07, 0x03, 0x01 }; // last 0=6, 1=7, 2=5, 4, 3, 2, 1 bits/// Mask for the data bits of a UTF-16 surrogate.U16 byteMaskLow10 = 0x03ff;//-----------------------------------------------------------------------------inline bool isSurrogateRange(U32 codepoint){   return ( 0xd800 < codepoint && codepoint < 0xdfff );}inline bool isAboveBMP(U32 codepoint){   return ( codepoint > 0xFFFF );}//-----------------------------------------------------------------------------const U32 convertUTF8toUTF16(const UTF8 *unistring, UTF16 *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF8toUTF16);   U32 walked, nCodepoints;   UTF32 middleman;      nCodepoints=0;   while(*unistring != '\0' && nCodepoints < len)   {      walked = 1;      middleman = oneUTF8toUTF32(unistring,&walked);      outbuffer[nCodepoints] = oneUTF32toUTF16(middleman);      unistring+=walked;      nCodepoints++;   }   nCodepoints = getMin(nCodepoints,len - 1);   outbuffer[nCodepoints] = '\0';      PROFILE_END();   return nCodepoints; }//-----------------------------------------------------------------------------const U32 convertUTF8toUTF32(const UTF8 *unistring, UTF32 *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF8toUTF32);   U32 walked, nCodepoints;      nCodepoints=0;   while(*unistring != 0 && nCodepoints < len)   {      walked = 1;      outbuffer[nCodepoints] = oneUTF8toUTF32(unistring,&walked);      unistring+=walked;      nCodepoints++;   }   nCodepoints = getMin(nCodepoints,len - 1);   outbuffer[nCodepoints] = '\0';      PROFILE_END();   return nCodepoints; }//-----------------------------------------------------------------------------const U32 convertUTF16toUTF8( const UTF16 *unistring, UTF8  *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF16toUTF8);   U32 walked, nCodeunits, codeunitLen;   UTF32 middleman;      nCodeunits=0;   while( *unistring != '\0' && nCodeunits + 3 < len )   {      walked = 1;      middleman  = oneUTF16toUTF32(unistring,&walked);      codeunitLen = oneUTF32toUTF8(middleman, &outbuffer[nCodeunits]);      unistring += walked;      nCodeunits += codeunitLen;   }   nCodeunits = getMin(nCodeunits,len - 1);   outbuffer[nCodeunits] = '\0';      PROFILE_END();   return nCodeunits;}//-----------------------------------------------------------------------------const U32 convertUTF16toUTF32(const UTF16 *unistring, UTF32 *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF16toUTF32);   U32 walked, nCodepoints;      nCodepoints=0;   while( *unistring != '\0' && nCodepoints < len )   {      walked=1;      outbuffer[nCodepoints] = oneUTF16toUTF32(unistring,&walked);      unistring += walked;      nCodepoints++;   }      nCodepoints = getMin(nCodepoints,len);   outbuffer[nCodepoints] = '\0';      PROFILE_END();   return nCodepoints;}//-----------------------------------------------------------------------------const U32 convertUTF32toUTF8( const UTF32 *unistring, UTF8  *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF32toUTF8);   U32 nCodeunits, codeunitLen;      nCodeunits=0;   while( *unistring != '\0' && nCodeunits + 3 < len )   {      codeunitLen = oneUTF32toUTF8(*unistring, &outbuffer[nCodeunits]);      unistring++;      nCodeunits += codeunitLen;   }      nCodeunits = getMin(nCodeunits,len);   outbuffer[nCodeunits] = '\0';      PROFILE_END();   return nCodeunits;}//-----------------------------------------------------------------------------const U32 convertUTF32toUTF16(const UTF32 *unistring, UTF16 *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF32toUTF16);   U32 nCodepoints;      nCodepoints=0;   while(*unistring != '\0' && nCodepoints < len)   {      outbuffer[nCodepoints] = oneUTF32toUTF16(*unistring);      unistring++;      nCodepoints++;   }   nCodepoints = getMin(nCodepoints,len);   outbuffer[nCodepoints] = '\0';      PROFILE_END();   return nCodepoints; }//-----------------------------------------------------------------------------// Functions that convert buffers of unicode code points//-----------------------------------------------------------------------------UTF16* convertUTF8toUTF16( const UTF8* unistring){   PROFILE_START(convertUTF8toUTF16_create);   // allocate plenty of memory.   U32 nCodepoints, len = dStrlen(unistring) + 1;   FrameTemp<UTF16> buf(len);      // perform conversion   nCodepoints = convertUTF8toUTF16( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodepoints++;      // allocate the return buffer, copy over, and return it.   UTF16 *ret = new UTF16[nCodepoints];   dMemcpy(ret, buf, nCodepoints * sizeof(UTF16));      PROFILE_END();   return ret;}//-----------------------------------------------------------------------------UTF32* convertUTF8toUTF32( const UTF8* unistring){   PROFILE_START(convertUTF8toUTF32_create);   // allocate plenty of memory.   U32 nCodepoints, len = dStrlen(unistring) + 1;   FrameTemp<UTF32> buf(len);      // perform conversion   nCodepoints = convertUTF8toUTF32( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodepoints++;      // allocate the return buffer, copy over, and return it.   UTF32 *ret = new UTF32[nCodepoints];   dMemcpy(ret, buf, nCodepoints * sizeof(UTF32));      PROFILE_END();   return ret;}//-----------------------------------------------------------------------------UTF8*  convertUTF16toUTF8( const UTF16* unistring){   PROFILE_START(convertUTF16toUTF8_create);   // allocate plenty of memory.   U32 nCodeunits, len = dStrlen(unistring) * 3 + 1;   FrameTemp<UTF8> buf(len);      // perform conversion   nCodeunits = convertUTF16toUTF8( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodeunits++;      // allocate the return buffer, copy over, and return it.   UTF8 *ret = new UTF8[nCodeunits];   dMemcpy(ret, buf, nCodeunits * sizeof(UTF8));   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------UTF32* convertUTF16toUTF32(const UTF16* unistring){   PROFILE_START(convertUTF16toUTF32_create);   // allocate plenty of memory.   U32 nCodepoints, len = dStrlen(unistring) + 1;   FrameTemp<UTF32> buf(len);      // perform conversion   nCodepoints = convertUTF16toUTF32( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodepoints++;      // allocate the return buffer, copy over, and return it.   UTF32 *ret = new UTF32[nCodepoints];   dMemcpy(ret, buf, nCodepoints * sizeof(UTF32));   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------UTF8*  convertUTF32toUTF8( const UTF32* unistring){   PROFILE_START(convertUTF32toUTF8_create);   // allocate plenty of memory.   U32 nCodeunits, len = dStrlen(unistring) * 3 + 1;   FrameTemp<UTF8> buf(len);      // perform conversion   nCodeunits = convertUTF32toUTF8( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodeunits++;      // allocate the return buffer, copy over, and return it.   UTF8 *ret = new UTF8[nCodeunits];   dMemcpy(ret, buf, nCodeunits * sizeof(UTF8));   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------UTF16* convertUTF32toUTF16(const UTF32* unistring){   PROFILE_START(convertUTF32toUTF16_create);   // allocate plenty of memory.   U32 nCodepoints, len = dStrlen(unistring) + 1;   FrameTemp<UTF16> buf(len);      // perform conversion   nCodepoints = convertUTF32toUTF16( unistring, buf, len);      // add 1 for the NULL terminator the converter promises it included.   nCodepoints++;      // allocate the return buffer, copy over, and return it.   UTF16 *ret = new UTF16[nCodepoints];   dMemcpy(ret, buf, nCodepoints * sizeof(UTF16));   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------// Functions that converts one unicode codepoint at a time//-----------------------------------------------------------------------------const UTF32 oneUTF8toUTF32( const UTF8* codepoint, U32 *unitsWalked){   PROFILE_START(oneUTF8toUTF32);   // codepoints 6 codeunits long are read, but do not convert correctly,   // and are filtered out anyway.      // early out for ascii   if(!(*codepoint & 0x0080))   {      *unitsWalked = 1;      PROFILE_END();      return (UTF32)*codepoint;   }      U32 expectedByteCount;   UTF32  ret = 0;   U8 codeunit;      // check the first byte ( a.k.a. codeunit ) .   unsigned char c = codepoint[0];   c = c >> 1;   expectedByteCount = firstByteLUT[c];   if(expectedByteCount > 0) // 0 or negative is illegal to start with   {      // process 1st codeunit      ret |= byteMask8LUT[expectedByteCount] & codepoint[0]; // bug?            // process trailing codeunits      for(U32 i=1;i<expectedByteCount; i++)      {         codeunit = codepoint[i];         if( firstByteLUT[codeunit>>1] == 0 )         {            ret <<= 6;                 // shift up 6            ret |= (codeunit & 0x3f);  // mask in the low 6 bits of this codeunit byte.         }         else         {            // found a bad codepoint - did not get a medial where we wanted one.            // Dump the replacement, and claim to have parsed only 1 char,            // so that we'll dump a slew of replacements, instead of eating the next char.                        ret = kReplacementChar;            expectedByteCount = 1;            break;         }      }   }   else    {      // found a bad codepoint - got a medial or an illegal codeunit.       // Dump the replacement, and claim to have parsed only 1 char,      // so that we'll dump a slew of replacements, instead of eating the next char.      ret = kReplacementChar;      expectedByteCount = 1;   }      if(unitsWalked != NULL)      *unitsWalked = expectedByteCount;      // codepoints in the surrogate range are illegal, and should be replaced.   if(isSurrogateRange(ret))      ret = kReplacementChar;      // codepoints outside the Basic Multilingual Plane add complexity to our UTF16 string classes,   // we've read them correctly so they wont foul the byte stream,   // but we kill them here to make sure they wont foul anything else   if(isAboveBMP(ret))      ret = kReplacementChar;   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------const UTF32  oneUTF16toUTF32(const UTF16* codepoint, U32 *unitsWalked){   PROFILE_START(oneUTF16toUTF32);   U8    expectedType;   U32   unitCount;   UTF32 ret = 0;   UTF16 codeunit1,codeunit2;      codeunit1 = codepoint[0];   expectedType = surrogateLUT[codeunit1 >> 10];   switch(expectedType)   {      case 0: // simple         ret = codeunit1;         unitCount = 1;         break;      case 1: // 2 surrogates         codeunit2 = codepoint[1];         if( surrogateLUT[codeunit2 >> 10] == 2)         {            ret = ((codeunit1 & byteMaskLow10 ) << 10) | (codeunit2 & byteMaskLow10);            unitCount = 2;            break;         }         // else, did not find a trailing surrogate where we expected one,         // so fall through to the error      case 2: // error         // found a trailing surrogate where we expected a codepoint or leading surrogate.         // Dump the replacement.         ret = kReplacementChar;         unitCount = 1;         break;   }   if(unitsWalked != NULL)      *unitsWalked = unitCount;   // codepoints in the surrogate range are illegal, and should be replaced.   if(isSurrogateRange(ret))      ret = kReplacementChar;   // codepoints outside the Basic Multilingual Plane add complexity to our UTF16 string classes,   // we've read them correctly so they wont foul the byte stream,   // but we kill them here to make sure they wont foul anything else   // NOTE: these are perfectly legal codepoints, we just dont want to deal with them.   if(isAboveBMP(ret))      ret = kReplacementChar;   PROFILE_END();   return ret;}//-----------------------------------------------------------------------------const UTF16 oneUTF32toUTF16(const UTF32 codepoint){   // found a codepoint outside the codeable UTF-16 range!   // or, found an illegal codepoint!   if(codepoint >= 0x10FFFF || isSurrogateRange(codepoint))      return kReplacementChar;      // these are legal, we just dont want to deal with them.   if(isAboveBMP(codepoint))      return kReplacementChar;   return (UTF16)codepoint;}//-----------------------------------------------------------------------------const U32 oneUTF32toUTF8(const UTF32 codepoint, UTF8 *threeByteCodeunitBuf){   PROFILE_START(oneUTF32toUTF8);   U32 bytecount = 0;   UTF8 *buf;   U32 working = codepoint;   buf = threeByteCodeunitBuf;   //-----------------   if(isSurrogateRange(working))  // found an illegal codepoint!      working = kReplacementChar;      //return oneUTF32toUTF8(kReplacementChar, threeByteCodeunitBuf);      if(isAboveBMP(working))        // these are legal, we just dont want to deal with them.      working = kReplacementChar;      //return oneUTF32toUTF8(kReplacementChar, threeByteCodeunitBuf);   //-----------------   if( working < (1 << 7))        // codeable in 7 bits      bytecount = 1;   else if( working < (1 << 11))  // codeable in 11 bits      bytecount = 2;   else if( working < (1 << 16))  // codeable in 16 bits      bytecount = 3;   AssertISV( bytecount > 0, "Error converting to UTF-8 in oneUTF32toUTF8(). isAboveBMP() should have caught this!");   //-----------------   U8  mask = byteMask8LUT[0];            // 0011 1111   U8  marker = ( ~mask << 1);            // 1000 0000      // Process the low order bytes, shifting the codepoint down 6 each pass.   for( int i = bytecount-1; i > 0; i--)   {      threeByteCodeunitBuf[i] = marker | (working & mask);       working >>= 6;   }   // Process the 1st byte. filter based on the # of expected bytes.   mask = byteMask8LUT[bytecount];   marker = ( ~mask << 1 );   threeByteCodeunitBuf[0] = marker | (working & mask);      PROFILE_END();   return bytecount;}//-----------------------------------------------------------------------------const U32 dStrlen(const UTF16 *unistring){   U32 i = 0;   while(unistring[i] != '\0')      i++;         return i;}//-----------------------------------------------------------------------------const U32 dStrlen(const UTF32 *unistring){   U32 i = 0;   while(unistring[i] != '\0')      i++;         return i;}//-----------------------------------------------------------------------------const U32 dStrncmp(const UTF16* unistring1, const UTF16* unistring2, U32 len){   UTF16 c1, c2;   for(U32 i = 0; i<len; i++)   {      c1 = *unistring1++;      c2 = *unistring2++;      if(c1 < c2) return -1;      if(c1 > c2) return 1;      if(!c1) return 0;   }   return 0;}//-----------------------------------------------------------------------------const U32 dStrncmp(const UTF32* unistring1, const UTF32* unistring2, U32 len){   UTF32 c1, c2;   for(U32 i = 0; i<len; i++)   {      c1 = *unistring1++;      c2 = *unistring2++;      if(c1 < c2) return -1;      if(c1 > c2) return 1;      if(!c1) return 0;   }   return 0;}//-----------------------------------------------------------------------------const UTF8* getNthCodepoint(const UTF8 *unistring, const U32 n){   const UTF8* ret = unistring;   U32 charsseen = 0;   while( *ret && charsseen < n)   {      ret++;      if((*ret & 0xC0) != 0x80)         charsseen++;   }      return ret;}/* alternate utf-8 decode impl for speed, no error checking,    left here for your amusement:      U32 codeunit = codepoint + expectedByteCount - 1;   U32 i = 0;   switch(expectedByteCount)   {      case 6: ret |= ( *(codeunit--) & 0x3f ); i++;                  case 5: ret |= ( *(codeunit--) & 0x3f ) << (6 * i++);          case 4: ret |= ( *(codeunit--) & 0x3f ) << (6 * i++);          case 3: ret |= ( *(codeunit--) & 0x3f ) << (6 * i++);          case 2: ret |= ( *(codeunit--) & 0x3f ) << (6 * i++);          case 1: ret |= *(codeunit) & byteMask8LUT[expectedByteCount] << (6 * i);   }*/
 |