diff options
Diffstat (limited to 'intl/uconv/ucvcn/nsGBKConvUtil.cpp')
-rw-r--r-- | intl/uconv/ucvcn/nsGBKConvUtil.cpp | 95 |
1 files changed, 95 insertions, 0 deletions
diff --git a/intl/uconv/ucvcn/nsGBKConvUtil.cpp b/intl/uconv/ucvcn/nsGBKConvUtil.cpp new file mode 100644 index 000000000..c611cee11 --- /dev/null +++ b/intl/uconv/ucvcn/nsGBKConvUtil.cpp @@ -0,0 +1,95 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "nsGBKConvUtil.h" +#include "gbku.h" +#include "nsDebug.h" +#define MAX_GBK_LENGTH 24066 /* (0xfe-0x80)*(0xfe-0x3f) */ +//-------------------------------------------------------------------- +// nsGBKConvUtil +//-------------------------------------------------------------------- + +static const char16_t gGBKToUnicodeTable[MAX_GBK_LENGTH] = { +#include "cp936map.h" +}; +static const uint16_t gUnicodeToGBKTable[0xA000-0x4e00] = { +#include "cp936invmap.h" +}; + +bool nsGBKConvUtil::UnicodeToGBKChar( + char16_t aChar, bool aToGL, char* + aOutByte1, char* aOutByte2) +{ + bool found=false; + *aOutByte1 = *aOutByte2 = 0; + if(UNICHAR_IN_RANGE(0xd800, aChar, 0xdfff)) + { + // surrogate is not in here + return false; + } + if(UNICHAR_IN_RANGE(0x4e00, aChar, 0x9FFF)) + { + uint16_t item = gUnicodeToGBKTable[aChar - 0x4e00]; + if(item != 0) + { + *aOutByte1 = item >> 8; + *aOutByte2 = item & 0x00FF; + found = true; + } else { + return false; + } + } else if (aChar == UCS2_NO_MAPPING) { + return false; + } else { + // ugly linear search + for( int32_t i = 0; i < MAX_GBK_LENGTH; i++ ) + { + if( aChar == gGBKToUnicodeTable[i]) + { + *aOutByte1 = (i / 0x00BF + 0x0081) ; + *aOutByte2 = (i % 0x00BF + 0x0040) ; + found = true; + break; + } + } + } + if(! found) + return false; + + if(aToGL) { + // to GL, we only return if it is in the range + if(UINT8_IN_RANGE(0xA1, *aOutByte1, 0xFE) && + UINT8_IN_RANGE(0xA1, *aOutByte2, 0xFE)) + { + // mask them to GL + *aOutByte1 &= 0x7F; + *aOutByte2 &= 0x7F; + } else { + // if it does not fit into 0xa1-0xfe 0xa1-0xfe range that mean + // it is not a GB2312 character, we cannot map to GL + *aOutByte1 = 0x00; + *aOutByte2 = 0x00; + return false; + } + } + return true; +} +char16_t nsGBKConvUtil::GBKCharToUnicode(char aByte1, char aByte2) +{ + NS_ASSERTION(UINT8_IN_RANGE(0x81,aByte1, 0xFE), "first byte out of range"); + NS_ASSERTION(UINT8_IN_RANGE(0x40,aByte2, 0xFE), "second byte out of range"); + + uint8_t i1 = (uint8_t)aByte1; + uint8_t i2 = (uint8_t)aByte2; + uint16_t idx = (i1 - 0x0081) * 0x00bf + i2 - 0x0040 ; + + NS_ASSERTION(idx < MAX_GBK_LENGTH, "ARB"); + // play it safe- add if statement here ot protect ARB + // probably not necessary + if(idx < MAX_GBK_LENGTH) + return gGBKToUnicodeTable[ idx ]; + else + return UCS2_NO_MAPPING; +} |