Revert "Remove FX_Alloc() null checks now that it can't return NULL."
[pdfium.git] / core / src / fxcrt / fx_basic_wstring.cpp
index 900f058..3c54ca9 100644 (file)
-// Copyright 2014 PDFium Authors. All rights reserved.\r
-// Use of this source code is governed by a BSD-style license that can be\r
-// found in the LICENSE file.\r
\r
-// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com\r
-\r
-#include "../../include/fxcrt/fx_basic.h"\r
-static CFX_StringDataW* FX_AllocStringW(int nLen)\r
-{\r
-    if (nLen == 0) {\r
-        return NULL;\r
-    }\r
-    CFX_StringDataW* pData = (CFX_StringDataW*)FX_Alloc(FX_BYTE, sizeof(long) * 3 + (nLen + 1) * sizeof(FX_WCHAR));\r
-    if (!pData) {\r
-        return NULL;\r
-    }\r
-    pData->m_nAllocLength = nLen;\r
-    pData->m_nDataLength = nLen;\r
-    pData->m_nRefs = 1;\r
-    pData->m_String[nLen] = 0;\r
-    return pData;\r
-}\r
-static void FX_ReleaseStringW(CFX_StringDataW* pData)\r
-{\r
-    if (pData == NULL) {\r
-        return;\r
-    }\r
-    pData->m_nRefs --;\r
-    if (pData->m_nRefs <= 0) {\r
-        FX_Free(pData);\r
-    }\r
-}\r
-CFX_WideString::~CFX_WideString()\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    m_pData->m_nRefs --;\r
-    if (m_pData->m_nRefs < 1) {\r
-        FX_Free(m_pData);\r
-    }\r
-}\r
-void CFX_WideString::InitStr(FX_LPCWSTR lpsz, FX_STRSIZE nLen)\r
-{\r
-    if (nLen < 0) {\r
-        nLen = lpsz ? (FX_STRSIZE)FXSYS_wcslen(lpsz) : 0;\r
-    }\r
-    if (nLen) {\r
-        m_pData = FX_AllocStringW(nLen);\r
-        if (!m_pData) {\r
-            return;\r
-        }\r
-        FXSYS_memcpy32(m_pData->m_String, lpsz, nLen * sizeof(FX_WCHAR));\r
-    } else {\r
-        m_pData = NULL;\r
-    }\r
-}\r
-CFX_WideString::CFX_WideString(const CFX_WideString& stringSrc)\r
-{\r
-    if (stringSrc.m_pData == NULL) {\r
-        m_pData = NULL;\r
-        return;\r
-    }\r
-    if (stringSrc.m_pData->m_nRefs >= 0) {\r
-        m_pData = stringSrc.m_pData;\r
-        m_pData->m_nRefs ++;\r
-    } else {\r
-        m_pData = NULL;\r
-        *this = stringSrc;\r
-    }\r
-}\r
-CFX_WideString::CFX_WideString(FX_WCHAR ch)\r
-{\r
-    m_pData = FX_AllocStringW(1);\r
-    if (m_pData) {\r
-        m_pData->m_String[0] = ch;\r
-    }\r
-}\r
-CFX_WideString::CFX_WideString(const CFX_WideStringC& str)\r
-{\r
-    if (str.IsEmpty()) {\r
-        m_pData = NULL;\r
-        return;\r
-    }\r
-    m_pData = FX_AllocStringW(str.GetLength());\r
-    if (m_pData) {\r
-        FXSYS_memcpy32(m_pData->m_String, str.GetPtr(), str.GetLength()*sizeof(FX_WCHAR));\r
-    }\r
-}\r
-CFX_WideString::CFX_WideString(const CFX_WideStringC& str1, const CFX_WideStringC& str2)\r
-{\r
-    m_pData = NULL;\r
-    int nNewLen = str1.GetLength() + str2.GetLength();\r
-    if (nNewLen == 0) {\r
-        return;\r
-    }\r
-    m_pData = FX_AllocStringW(nNewLen);\r
-    if (m_pData) {\r
-        FXSYS_memcpy32(m_pData->m_String, str1.GetPtr(), str1.GetLength()*sizeof(FX_WCHAR));\r
-        FXSYS_memcpy32(m_pData->m_String + str1.GetLength(), str2.GetPtr(), str2.GetLength()*sizeof(FX_WCHAR));\r
-    }\r
-}\r
-void CFX_WideString::ReleaseBuffer(FX_STRSIZE nNewLength)\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    CopyBeforeWrite();\r
-    if (nNewLength == -1) {\r
-        nNewLength = m_pData ? (FX_STRSIZE)FXSYS_wcslen(m_pData->m_String) : 0;\r
-    }\r
-    if (nNewLength == 0) {\r
-        Empty();\r
-        return;\r
-    }\r
-    FXSYS_assert(nNewLength <= m_pData->m_nAllocLength);\r
-    m_pData->m_nDataLength = nNewLength;\r
-    m_pData->m_String[nNewLength] = 0;\r
-}\r
-const CFX_WideString& CFX_WideString::operator=(FX_LPCWSTR lpsz)\r
-{\r
-    if (lpsz == NULL || lpsz[0] == 0) {\r
-        Empty();\r
-    } else {\r
-        AssignCopy((FX_STRSIZE)FXSYS_wcslen(lpsz), lpsz);\r
-    }\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator=(const CFX_WideStringC& stringSrc)\r
-{\r
-    if (stringSrc.IsEmpty()) {\r
-        Empty();\r
-    } else {\r
-        AssignCopy(stringSrc.GetLength(), stringSrc.GetPtr());\r
-    }\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator=(const CFX_WideString& stringSrc)\r
-{\r
-    if (m_pData == stringSrc.m_pData) {\r
-        return *this;\r
-    }\r
-    if (stringSrc.IsEmpty()) {\r
-        Empty();\r
-    } else if ((m_pData && m_pData->m_nRefs < 0) ||\r
-               (stringSrc.m_pData && stringSrc.m_pData->m_nRefs < 0)) {\r
-        AssignCopy(stringSrc.m_pData->m_nDataLength, stringSrc.m_pData->m_String);\r
-    } else {\r
-        Empty();\r
-        m_pData = stringSrc.m_pData;\r
-        if (m_pData) {\r
-            m_pData->m_nRefs ++;\r
-        }\r
-    }\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator+=(FX_WCHAR ch)\r
-{\r
-    ConcatInPlace(1, &ch);\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator+=(FX_LPCWSTR lpsz)\r
-{\r
-    if (lpsz) {\r
-        ConcatInPlace((FX_STRSIZE)FXSYS_wcslen(lpsz), lpsz);\r
-    }\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator+=(const CFX_WideString& string)\r
-{\r
-    if (string.m_pData == NULL) {\r
-        return *this;\r
-    }\r
-    ConcatInPlace(string.m_pData->m_nDataLength, string.m_pData->m_String);\r
-    return *this;\r
-}\r
-const CFX_WideString& CFX_WideString::operator+=(const CFX_WideStringC& string)\r
-{\r
-    if (string.IsEmpty()) {\r
-        return *this;\r
-    }\r
-    ConcatInPlace(string.GetLength(), string.GetPtr());\r
-    return *this;\r
-}\r
-bool operator==(const CFX_WideString& s1, FX_LPCWSTR s2)\r
-{\r
-    return s1.Equal(s2);\r
-}\r
-bool operator==(FX_LPCWSTR s1, const CFX_WideString& s2)\r
-{\r
-    return s2.Equal(s1);\r
-}\r
-bool operator==(const CFX_WideString& s1, const CFX_WideString& s2)\r
-{\r
-    return s1.Equal(s2);\r
-}\r
-bool operator==(const CFX_WideString& s1, const CFX_WideStringC& s2)\r
-{\r
-    return s1.Equal(s2);\r
-}\r
-bool operator==(const CFX_WideStringC& s1, const CFX_WideString& s2)\r
-{\r
-    return s2.Equal(s1);\r
-}\r
-bool operator != (const CFX_WideString& s1, FX_LPCWSTR s2)\r
-{\r
-    return !s1.Equal(s2);\r
-}\r
-bool operator!=(const CFX_WideString& s1, const CFX_WideString& s2)\r
-{\r
-    return !s1.Equal(s2);\r
-}\r
-bool operator!=(const CFX_WideString& s1, const CFX_WideStringC& s2)\r
-{\r
-    return !s1.Equal(s2);\r
-}\r
-bool operator!=(const CFX_WideStringC& s1, const CFX_WideString& s2)\r
-{\r
-    return !s2.Equal(s1);\r
-}\r
-bool CFX_WideString::Equal(const CFX_WideStringC& str) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return str.IsEmpty();\r
-    }\r
-    return str.GetLength() == m_pData->m_nDataLength &&\r
-           FXSYS_memcmp32(str.GetPtr(), m_pData->m_String, m_pData->m_nDataLength * sizeof(FX_WCHAR)) == 0;\r
-}\r
-void CFX_WideString::Empty()\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    if (m_pData->m_nRefs > 1) {\r
-        m_pData->m_nRefs --;\r
-    } else {\r
-        FX_Free(m_pData);\r
-    }\r
-    m_pData = NULL;\r
-}\r
-void CFX_WideString::ConcatInPlace(FX_STRSIZE nSrcLen, FX_LPCWSTR lpszSrcData)\r
-{\r
-    if (nSrcLen == 0 || lpszSrcData == NULL) {\r
-        return;\r
-    }\r
-    if (m_pData == NULL) {\r
-        m_pData = FX_AllocStringW(nSrcLen);\r
-        if (m_pData) {\r
-            FXSYS_memcpy32(m_pData->m_String, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));\r
-        }\r
-        return;\r
-    }\r
-    if (m_pData->m_nRefs > 1 || m_pData->m_nDataLength + nSrcLen > m_pData->m_nAllocLength) {\r
-        CFX_StringDataW* pOldData = m_pData;\r
-        ConcatCopy(m_pData->m_nDataLength, m_pData->m_String, nSrcLen, lpszSrcData);\r
-        FX_ReleaseStringW(pOldData);\r
-    } else {\r
-        FXSYS_memcpy32(m_pData->m_String + m_pData->m_nDataLength, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));\r
-        m_pData->m_nDataLength += nSrcLen;\r
-        m_pData->m_String[m_pData->m_nDataLength] = 0;\r
-    }\r
-}\r
-void CFX_WideString::ConcatCopy(FX_STRSIZE nSrc1Len, FX_LPCWSTR lpszSrc1Data,\r
-                                FX_STRSIZE nSrc2Len, FX_LPCWSTR lpszSrc2Data)\r
-{\r
-    FX_STRSIZE nNewLen = nSrc1Len + nSrc2Len;\r
-    if (nNewLen == 0) {\r
-        return;\r
-    }\r
-    m_pData = FX_AllocStringW(nNewLen);\r
-    if (m_pData) {\r
-        FXSYS_memcpy32(m_pData->m_String, lpszSrc1Data, nSrc1Len * sizeof(FX_WCHAR));\r
-        FXSYS_memcpy32(m_pData->m_String + nSrc1Len, lpszSrc2Data, nSrc2Len * sizeof(FX_WCHAR));\r
-    }\r
-}\r
-void CFX_WideString::CopyBeforeWrite()\r
-{\r
-    if (m_pData == NULL || m_pData->m_nRefs <= 1) {\r
-        return;\r
-    }\r
-    CFX_StringDataW* pData = m_pData;\r
-    m_pData->m_nRefs --;\r
-    FX_STRSIZE nDataLength = pData->m_nDataLength;\r
-    m_pData = FX_AllocStringW(nDataLength);\r
-    if (m_pData != NULL) {\r
-        FXSYS_memcpy32(m_pData->m_String, pData->m_String, (nDataLength + 1) * sizeof(FX_WCHAR));\r
-    }\r
-}\r
-void CFX_WideString::AllocBeforeWrite(FX_STRSIZE nLen)\r
-{\r
-    if (m_pData && m_pData->m_nRefs <= 1 && m_pData->m_nAllocLength >= nLen) {\r
-        return;\r
-    }\r
-    Empty();\r
-    m_pData = FX_AllocStringW(nLen);\r
-}\r
-void CFX_WideString::AssignCopy(FX_STRSIZE nSrcLen, FX_LPCWSTR lpszSrcData)\r
-{\r
-    AllocBeforeWrite(nSrcLen);\r
-    FXSYS_memcpy32(m_pData->m_String, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));\r
-    m_pData->m_nDataLength = nSrcLen;\r
-    m_pData->m_String[nSrcLen] = 0;\r
-}\r
-int CFX_WideString::Compare(FX_LPCWSTR lpsz) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return (lpsz == NULL || lpsz[0] == 0) ? 0 : -1;\r
-    }\r
-    return FXSYS_wcscmp(m_pData->m_String, lpsz);\r
-}\r
-CFX_ByteString CFX_WideString::UTF8Encode() const\r
-{\r
-    return FX_UTF8Encode(*this);\r
-}\r
-CFX_ByteString CFX_WideString::UTF16LE_Encode(FX_BOOL bTerminate) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return bTerminate ? CFX_ByteString(FX_BSTRC("\0\0")) : CFX_ByteString();\r
-    }\r
-    int len = m_pData->m_nDataLength;\r
-    CFX_ByteString result;\r
-    FX_LPSTR buffer = result.GetBuffer(len * 2 + (bTerminate ? 2 : 0));\r
-    for (int i = 0; i < len; i ++) {\r
-        buffer[i * 2] = m_pData->m_String[i] & 0xff;\r
-        buffer[i * 2 + 1] = m_pData->m_String[i] >> 8;\r
-    }\r
-    if (bTerminate) {\r
-        buffer[len * 2] = 0;\r
-        buffer[len * 2 + 1] = 0;\r
-        result.ReleaseBuffer(len * 2 + 2);\r
-    } else {\r
-        result.ReleaseBuffer(len * 2);\r
-    }\r
-    return result;\r
-}\r
-void CFX_WideString::ConvertFrom(const CFX_ByteString& str, CFX_CharMap* pCharMap)\r
-{\r
-    if (pCharMap == NULL) {\r
-        pCharMap = CFX_CharMap::GetDefaultMapper();\r
-    }\r
-    *this = pCharMap->m_GetWideString(pCharMap, str);\r
-}\r
-void CFX_WideString::Reserve(FX_STRSIZE len)\r
-{\r
-    GetBuffer(len);\r
-    ReleaseBuffer(GetLength());\r
-}\r
-FX_LPWSTR CFX_WideString::GetBuffer(FX_STRSIZE nMinBufLength)\r
-{\r
-    if (m_pData == NULL && nMinBufLength == 0) {\r
-        return NULL;\r
-    }\r
-    if (m_pData && m_pData->m_nRefs <= 1 && m_pData->m_nAllocLength >= nMinBufLength) {\r
-        return m_pData->m_String;\r
-    }\r
-    if (m_pData == NULL) {\r
-        m_pData = FX_AllocStringW(nMinBufLength);\r
-        if (!m_pData) {\r
-            return NULL;\r
-        }\r
-        m_pData->m_nDataLength = 0;\r
-        m_pData->m_String[0] = 0;\r
-        return m_pData->m_String;\r
-    }\r
-    CFX_StringDataW* pOldData = m_pData;\r
-    FX_STRSIZE nOldLen = pOldData->m_nDataLength;\r
-    if (nMinBufLength < nOldLen) {\r
-        nMinBufLength = nOldLen;\r
-    }\r
-    m_pData = FX_AllocStringW(nMinBufLength);\r
-    if (!m_pData) {\r
-        return NULL;\r
-    }\r
-    FXSYS_memcpy32(m_pData->m_String, pOldData->m_String, (nOldLen + 1)*sizeof(FX_WCHAR));\r
-    m_pData->m_nDataLength = nOldLen;\r
-    pOldData->m_nRefs --;\r
-    if (pOldData->m_nRefs <= 0) {\r
-        FX_Free(pOldData);\r
-    }\r
-    return m_pData->m_String;\r
-}\r
-CFX_WideString CFX_WideString::FromLocal(const char* str, FX_STRSIZE len)\r
-{\r
-    CFX_WideString result;\r
-    result.ConvertFrom(CFX_ByteString(str, len));\r
-    return result;\r
-}\r
-CFX_WideString CFX_WideString::FromUTF8(const char* str, FX_STRSIZE len)\r
-{\r
-    if (!str) {\r
-        return CFX_WideString();\r
-    }\r
-    if (len < 0) {\r
-        len = 0;\r
-        while (str[len]) {\r
-            len ++;\r
-        }\r
-    }\r
-    CFX_UTF8Decoder decoder;\r
-    for (FX_STRSIZE i = 0; i < len; i ++) {\r
-        decoder.Input(str[i]);\r
-    }\r
-    return decoder.GetResult();\r
-}\r
-CFX_WideString CFX_WideString::FromUTF16LE(const unsigned short* wstr, FX_STRSIZE wlen)\r
-{\r
-    if (!wstr || !wlen) {\r
-        return CFX_WideString();\r
-    }\r
-    if (wlen < 0) {\r
-        wlen = 0;\r
-        while (wstr[wlen]) {\r
-            wlen ++;\r
-        }\r
-    }\r
-    CFX_WideString result;\r
-    FX_WCHAR* buf = result.GetBuffer(wlen);\r
-    for (int i = 0; i < wlen; i ++) {\r
-        buf[i] = wstr[i];\r
-    }\r
-    result.ReleaseBuffer(wlen);\r
-    return result;\r
-}\r
-void CFX_WideString::AllocCopy(CFX_WideString& dest, FX_STRSIZE nCopyLen, FX_STRSIZE nCopyIndex,\r
-                               FX_STRSIZE nExtraLen) const\r
-{\r
-    FX_STRSIZE nNewLen = nCopyLen + nExtraLen;\r
-    if (nNewLen == 0) {\r
-        return;\r
-    }\r
-    ASSERT(dest.m_pData == NULL);\r
-    dest.m_pData = FX_AllocStringW(nNewLen);\r
-    if (dest.m_pData) {\r
-        FXSYS_memcpy32(dest.m_pData->m_String, m_pData->m_String + nCopyIndex, nCopyLen * sizeof(FX_WCHAR));\r
-    }\r
-}\r
-CFX_WideString CFX_WideString::Left(FX_STRSIZE nCount) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return CFX_WideString();\r
-    }\r
-    if (nCount < 0) {\r
-        nCount = 0;\r
-    }\r
-    if (nCount >= m_pData->m_nDataLength) {\r
-        return *this;\r
-    }\r
-    CFX_WideString dest;\r
-    AllocCopy(dest, nCount, 0, 0);\r
-    return dest;\r
-}\r
-CFX_WideString CFX_WideString::Mid(FX_STRSIZE nFirst) const\r
-{\r
-    return Mid(nFirst, m_pData->m_nDataLength - nFirst);\r
-}\r
-CFX_WideString CFX_WideString::Mid(FX_STRSIZE nFirst, FX_STRSIZE nCount) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return CFX_WideString();\r
-    }\r
-    if (nFirst < 0) {\r
-        nFirst = 0;\r
-    }\r
-    if (nCount < 0) {\r
-        nCount = 0;\r
-    }\r
-    if (nFirst + nCount > m_pData->m_nDataLength) {\r
-        nCount = m_pData->m_nDataLength - nFirst;\r
-    }\r
-    if (nFirst > m_pData->m_nDataLength) {\r
-        nCount = 0;\r
-    }\r
-    if (nFirst == 0 && nFirst + nCount == m_pData->m_nDataLength) {\r
-        return *this;\r
-    }\r
-    CFX_WideString dest;\r
-    AllocCopy(dest, nCount, nFirst, 0);\r
-    return dest;\r
-}\r
-CFX_WideString CFX_WideString::Right(FX_STRSIZE nCount) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return CFX_WideString();\r
-    }\r
-    if (nCount < 0) {\r
-        nCount = 0;\r
-    }\r
-    if (nCount >= m_pData->m_nDataLength) {\r
-        return *this;\r
-    }\r
-    CFX_WideString dest;\r
-    AllocCopy(dest, nCount, m_pData->m_nDataLength - nCount, 0);\r
-    return dest;\r
-}\r
-int CFX_WideString::CompareNoCase(FX_LPCWSTR lpsz) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return (lpsz == NULL || lpsz[0] == 0) ? 0 : -1;\r
-    }\r
-    return FXSYS_wcsicmp(m_pData->m_String, lpsz);\r
-}\r
-int CFX_WideString::Compare(const CFX_WideString& str) const\r
-{\r
-    if (m_pData == NULL) {\r
-        if (str.m_pData == NULL) {\r
-            return 0;\r
-        }\r
-        return -1;\r
-    } else if (str.m_pData == NULL) {\r
-        return 1;\r
-    }\r
-    int this_len = m_pData->m_nDataLength;\r
-    int that_len = str.m_pData->m_nDataLength;\r
-    int min_len = this_len < that_len ? this_len : that_len;\r
-    for (int i = 0; i < min_len; i ++) {\r
-        if (m_pData->m_String[i] < str.m_pData->m_String[i]) {\r
-            return -1;\r
-        } else if (m_pData->m_String[i] > str.m_pData->m_String[i]) {\r
-            return 1;\r
-        }\r
-    }\r
-    if (this_len < that_len) {\r
-        return -1;\r
-    } else if (this_len > that_len) {\r
-        return 1;\r
-    }\r
-    return 0;\r
-}\r
-FX_LPWSTR CFX_WideString::LockBuffer()\r
-{\r
-    if (m_pData == NULL) {\r
-        return NULL;\r
-    }\r
-    FX_LPWSTR lpsz = GetBuffer(0);\r
-    m_pData->m_nRefs = -1;\r
-    return lpsz;\r
-}\r
-void CFX_WideString::SetAt(FX_STRSIZE nIndex, FX_WCHAR ch)\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    ASSERT(nIndex >= 0);\r
-    ASSERT(nIndex < m_pData->m_nDataLength);\r
-    CopyBeforeWrite();\r
-    m_pData->m_String[nIndex] = ch;\r
-}\r
-void CFX_WideString::MakeLower()\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    CopyBeforeWrite();\r
-    if (GetLength() < 1) {\r
-        return;\r
-    }\r
-    FXSYS_wcslwr(m_pData->m_String);\r
-}\r
-void CFX_WideString::MakeUpper()\r
-{\r
-    if (m_pData == NULL) {\r
-        return;\r
-    }\r
-    CopyBeforeWrite();\r
-    if (GetLength() < 1) {\r
-        return;\r
-    }\r
-    FXSYS_wcsupr(m_pData->m_String);\r
-}\r
-FX_STRSIZE CFX_WideString::Find(FX_LPCWSTR lpszSub, FX_STRSIZE nStart) const\r
-{\r
-    FX_STRSIZE nLength = GetLength();\r
-    if (nLength < 1 || nStart > nLength) {\r
-        return -1;\r
-    }\r
-    FX_LPCWSTR lpsz = (FX_LPCWSTR)FXSYS_wcsstr(m_pData->m_String + nStart, lpszSub);\r
-    return (lpsz == NULL) ? -1 : (int)(lpsz - m_pData->m_String);\r
-}\r
-FX_STRSIZE CFX_WideString::Find(FX_WCHAR ch, FX_STRSIZE nStart) const\r
-{\r
-    if (m_pData == NULL) {\r
-        return -1;\r
-    }\r
-    FX_STRSIZE nLength = m_pData->m_nDataLength;\r
-    if (nStart >= nLength) {\r
-        return -1;\r
-    }\r
-    FX_LPCWSTR lpsz = (FX_LPCWSTR)FXSYS_wcschr(m_pData->m_String + nStart, ch);\r
-    return (lpsz == NULL) ? -1 : (int)(lpsz - m_pData->m_String);\r
-}\r
-void CFX_WideString::TrimRight(FX_LPCWSTR lpszTargetList)\r
-{\r
-    FXSYS_assert(lpszTargetList != NULL);\r
-    if (m_pData == NULL || *lpszTargetList == 0) {\r
-        return;\r
-    }\r
-    CopyBeforeWrite();\r
-    FX_STRSIZE len = GetLength();\r
-    if (len < 1) {\r
-        return;\r
-    }\r
-    FX_STRSIZE pos = len;\r
-    while (pos) {\r
-        if (FXSYS_wcschr(lpszTargetList, m_pData->m_String[pos - 1]) == NULL) {\r
-            break;\r
-        }\r
-        pos --;\r
-    }\r
-    if (pos < len) {\r
-        m_pData->m_String[pos] = 0;\r
-        m_pData->m_nDataLength = pos;\r
-    }\r
-}\r
-void CFX_WideString::TrimRight(FX_WCHAR chTarget)\r
-{\r
-    FX_WCHAR str[2] = {chTarget, 0};\r
-    TrimRight(str);\r
-}\r
-void CFX_WideString::TrimRight()\r
-{\r
-    TrimRight(L"\x09\x0a\x0b\x0c\x0d\x20");\r
-}\r
-void CFX_WideString::TrimLeft(FX_LPCWSTR lpszTargets)\r
-{\r
-    FXSYS_assert(lpszTargets != NULL);\r
-    if (m_pData == NULL || *lpszTargets == 0) {\r
-        return;\r
-    }\r
-    CopyBeforeWrite();\r
-    if (GetLength() < 1) {\r
-        return;\r
-    }\r
-    FX_LPCWSTR lpsz = m_pData->m_String;\r
-    while (*lpsz != 0) {\r
-        if (FXSYS_wcschr(lpszTargets, *lpsz) == NULL) {\r
-            break;\r
-        }\r
-        lpsz ++;\r
-    }\r
-    if (lpsz != m_pData->m_String) {\r
-        int nDataLength = m_pData->m_nDataLength - (FX_STRSIZE)(lpsz - m_pData->m_String);\r
-        FXSYS_memmove32(m_pData->m_String, lpsz, (nDataLength + 1)*sizeof(FX_WCHAR));\r
-        m_pData->m_nDataLength = nDataLength;\r
-    }\r
-}\r
-void CFX_WideString::TrimLeft(FX_WCHAR chTarget)\r
-{\r
-    FX_WCHAR str[2] = {chTarget, 0};\r
-    TrimLeft(str);\r
-}\r
-void CFX_WideString::TrimLeft()\r
-{\r
-    TrimLeft(L"\x09\x0a\x0b\x0c\x0d\x20");\r
-}\r
-FX_STRSIZE CFX_WideString::Replace(FX_LPCWSTR lpszOld, FX_LPCWSTR lpszNew)\r
-{\r
-    if (GetLength() < 1) {\r
-        return 0;\r
-    }\r
-    if (lpszOld == NULL) {\r
-        return 0;\r
-    }\r
-    FX_STRSIZE nSourceLen = (FX_STRSIZE)FXSYS_wcslen(lpszOld);\r
-    if (nSourceLen == 0) {\r
-        return 0;\r
-    }\r
-    FX_STRSIZE nReplacementLen = lpszNew ? (FX_STRSIZE)FXSYS_wcslen(lpszNew) : 0;\r
-    FX_STRSIZE nCount = 0;\r
-    FX_LPWSTR lpszStart = m_pData->m_String;\r
-    FX_LPWSTR lpszEnd = m_pData->m_String + m_pData->m_nDataLength;\r
-    FX_LPWSTR lpszTarget;\r
-    {\r
-        while ((lpszTarget = (FX_LPWSTR)FXSYS_wcsstr(lpszStart, lpszOld)) != NULL && lpszStart < lpszEnd) {\r
-            nCount++;\r
-            lpszStart = lpszTarget + nSourceLen;\r
-        }\r
-    }\r
-    if (nCount > 0) {\r
-        CopyBeforeWrite();\r
-        FX_STRSIZE nOldLength = m_pData->m_nDataLength;\r
-        FX_STRSIZE nNewLength =  nOldLength + (nReplacementLen - nSourceLen) * nCount;\r
-        if (m_pData->m_nAllocLength < nNewLength || m_pData->m_nRefs > 1) {\r
-            CFX_StringDataW* pOldData = m_pData;\r
-            FX_LPCWSTR pstr = m_pData->m_String;\r
-            m_pData = FX_AllocStringW(nNewLength);\r
-            if (!m_pData) {\r
-                return 0;\r
-            }\r
-            FXSYS_memcpy32(m_pData->m_String, pstr, pOldData->m_nDataLength * sizeof(FX_WCHAR));\r
-            FX_ReleaseStringW(pOldData);\r
-        }\r
-        lpszStart = m_pData->m_String;\r
-        lpszEnd = m_pData->m_String + FX_MAX(m_pData->m_nDataLength, nNewLength);\r
-        {\r
-            while ((lpszTarget = (FX_LPWSTR)FXSYS_wcsstr(lpszStart, lpszOld)) != NULL && lpszStart < lpszEnd) {\r
-                FX_STRSIZE nBalance = nOldLength - (FX_STRSIZE)(lpszTarget - m_pData->m_String + nSourceLen);\r
-                FXSYS_memmove32(lpszTarget + nReplacementLen, lpszTarget + nSourceLen, nBalance * sizeof(FX_WCHAR));\r
-                FXSYS_memcpy32(lpszTarget, lpszNew, nReplacementLen * sizeof(FX_WCHAR));\r
-                lpszStart = lpszTarget + nReplacementLen;\r
-                lpszStart[nBalance] = 0;\r
-                nOldLength += (nReplacementLen - nSourceLen);\r
-            }\r
-        }\r
-        ASSERT(m_pData->m_String[nNewLength] == 0);\r
-        m_pData->m_nDataLength = nNewLength;\r
-    }\r
-    return nCount;\r
-}\r
-FX_STRSIZE CFX_WideString::Insert(FX_STRSIZE nIndex, FX_WCHAR ch)\r
-{\r
-    CopyBeforeWrite();\r
-    if (nIndex < 0) {\r
-        nIndex = 0;\r
-    }\r
-    FX_STRSIZE nNewLength = GetLength();\r
-    if (nIndex > nNewLength) {\r
-        nIndex = nNewLength;\r
-    }\r
-    nNewLength++;\r
-    if (m_pData == NULL || m_pData->m_nAllocLength < nNewLength) {\r
-        CFX_StringDataW* pOldData = m_pData;\r
-        FX_LPCWSTR pstr = m_pData->m_String;\r
-        m_pData = FX_AllocStringW(nNewLength);\r
-        if (!m_pData) {\r
-            return 0;\r
-        }\r
-        if(pOldData != NULL) {\r
-            FXSYS_memmove32(m_pData->m_String, pstr, (pOldData->m_nDataLength + 1)*sizeof(FX_WCHAR));\r
-            FX_ReleaseStringW(pOldData);\r
-        } else {\r
-            m_pData->m_String[0] = 0;\r
-        }\r
-    }\r
-    FXSYS_memmove32(m_pData->m_String + nIndex + 1,\r
-                    m_pData->m_String + nIndex, (nNewLength - nIndex)*sizeof(FX_WCHAR));\r
-    m_pData->m_String[nIndex] = ch;\r
-    m_pData->m_nDataLength = nNewLength;\r
-    return nNewLength;\r
-}\r
-FX_STRSIZE CFX_WideString::Delete(FX_STRSIZE nIndex, FX_STRSIZE nCount)\r
-{\r
-    if (GetLength() < 1) {\r
-        return 0;\r
-    }\r
-    if (nIndex < 0) {\r
-        nIndex = 0;\r
-    }\r
-    FX_STRSIZE nOldLength = m_pData->m_nDataLength;\r
-    if (nCount > 0 && nIndex < nOldLength) {\r
-        CopyBeforeWrite();\r
-        int nBytesToCopy = nOldLength - (nIndex + nCount) + 1;\r
-        FXSYS_memmove32(m_pData->m_String + nIndex,\r
-                        m_pData->m_String + nIndex + nCount, nBytesToCopy * sizeof(FX_WCHAR));\r
-        m_pData->m_nDataLength = nOldLength - nCount;\r
-    }\r
-    return m_pData->m_nDataLength;\r
-}\r
-FX_STRSIZE CFX_WideString::Remove(FX_WCHAR chRemove)\r
-{\r
-    if (m_pData == NULL) {\r
-        return 0;\r
-    }\r
-    CopyBeforeWrite();\r
-    if (GetLength() < 1) {\r
-        return 0;\r
-    }\r
-    FX_LPWSTR pstrSource = m_pData->m_String;\r
-    FX_LPWSTR pstrDest = m_pData->m_String;\r
-    FX_LPWSTR pstrEnd = m_pData->m_String + m_pData->m_nDataLength;\r
-    while (pstrSource < pstrEnd) {\r
-        if (*pstrSource != chRemove) {\r
-            *pstrDest = *pstrSource;\r
-            pstrDest ++;\r
-        }\r
-        pstrSource ++;\r
-    }\r
-    *pstrDest = 0;\r
-    FX_STRSIZE nCount = (FX_STRSIZE)(pstrSource - pstrDest);\r
-    m_pData->m_nDataLength -= nCount;\r
-    return nCount;\r
-}\r
-#define FORCE_ANSI      0x10000\r
-#define FORCE_UNICODE   0x20000\r
-#define FORCE_INT64     0x40000\r
-void CFX_WideString::FormatV(FX_LPCWSTR lpszFormat, va_list argList)\r
-{\r
-    va_list argListSave;\r
-#if defined(__ARMCC_VERSION) || (!defined(_MSC_VER) && (_FX_CPU_ == _FX_X64_ || _FX_CPU_ == _FX_IA64_ || _FX_CPU_ == _FX_ARM64_)) || defined(__native_client__)\r
-    va_copy(argListSave, argList);\r
-#else\r
-    argListSave = argList;\r
-#endif\r
-    int nMaxLen = 0;\r
-    for (FX_LPCWSTR lpsz = lpszFormat; *lpsz != 0; lpsz ++) {\r
-        if (*lpsz != '%' || *(lpsz = lpsz + 1) == '%') {\r
-            nMaxLen += (FX_STRSIZE)FXSYS_wcslen(lpsz);\r
-            continue;\r
-        }\r
-        int nItemLen = 0;\r
-        int nWidth = 0;\r
-        for (; *lpsz != 0; lpsz ++) {\r
-            if (*lpsz == '#') {\r
-                nMaxLen += 2;\r
-            } else if (*lpsz == '*') {\r
-                nWidth = va_arg(argList, int);\r
-            } else if (*lpsz == '-' || *lpsz == '+' || *lpsz == '0' ||\r
-                       *lpsz == ' ')\r
-                ;\r
-            else {\r
-                break;\r
-            }\r
-        }\r
-        if (nWidth == 0) {\r
-            nWidth = FXSYS_wtoi(lpsz);\r
-            for (; *lpsz != 0 && (*lpsz) <= '9' && (*lpsz) >= '0'; lpsz ++)\r
-                ;\r
-        }\r
-        if (nWidth < 0 || nWidth > 128 * 1024) {\r
-            lpszFormat = (FX_LPCWSTR)L"Bad width";\r
-            nMaxLen = 10;\r
-            break;\r
-        }\r
-        int nPrecision = 0;\r
-        if (*lpsz == '.') {\r
-            lpsz ++;\r
-            if (*lpsz == '*') {\r
-                nPrecision = va_arg(argList, int);\r
-                lpsz ++;\r
-            } else {\r
-                nPrecision = FXSYS_wtoi(lpsz);\r
-                for (; *lpsz != 0 && (*lpsz) >= '0' && (*lpsz) <= '9'; lpsz ++)\r
-                    ;\r
-            }\r
-        }\r
-        if (nPrecision < 0 || nPrecision > 128 * 1024) {\r
-            lpszFormat = (FX_LPCWSTR)L"Bad precision";\r
-            nMaxLen = 14;\r
-            break;\r
-        }\r
-        int nModifier = 0;\r
-        if (*lpsz == L'I' && *(lpsz + 1) == L'6' && *(lpsz + 2) == L'4') {\r
-            lpsz += 3;\r
-            nModifier = FORCE_INT64;\r
-        } else {\r
-            switch (*lpsz) {\r
-                case 'h':\r
-                    nModifier = FORCE_ANSI;\r
-                    lpsz ++;\r
-                    break;\r
-                case 'l':\r
-                    nModifier = FORCE_UNICODE;\r
-                    lpsz ++;\r
-                    break;\r
-                case 'F':\r
-                case 'N':\r
-                case 'L':\r
-                    lpsz ++;\r
-                    break;\r
-            }\r
-        }\r
-        switch (*lpsz | nModifier) {\r
-            case 'c':\r
-            case 'C':\r
-                nItemLen = 2;\r
-                va_arg(argList, int);\r
-                break;\r
-            case 'c'|FORCE_ANSI:\r
-            case 'C'|FORCE_ANSI:\r
-                nItemLen = 2;\r
-                va_arg(argList, int);\r
-                break;\r
-            case 'c'|FORCE_UNICODE:\r
-            case 'C'|FORCE_UNICODE:\r
-                nItemLen = 2;\r
-                va_arg(argList, int);\r
-                break;\r
-            case 's': {\r
-                    FX_LPCWSTR pstrNextArg = va_arg(argList, FX_LPCWSTR);\r
-                    if (pstrNextArg == NULL) {\r
-                        nItemLen = 6;\r
-                    } else {\r
-                        nItemLen = (FX_STRSIZE)FXSYS_wcslen(pstrNextArg);\r
-                        if (nItemLen < 1) {\r
-                            nItemLen = 1;\r
-                        }\r
-                    }\r
-                }\r
-                break;\r
-            case 'S': {\r
-                    FX_LPCSTR pstrNextArg = va_arg(argList, FX_LPCSTR);\r
-                    if (pstrNextArg == NULL) {\r
-                        nItemLen = 6;\r
-                    } else {\r
-                        nItemLen = (FX_STRSIZE)FXSYS_strlen(pstrNextArg);\r
-                        if (nItemLen < 1) {\r
-                            nItemLen = 1;\r
-                        }\r
-                    }\r
-                }\r
-                break;\r
-            case 's'|FORCE_ANSI:\r
-            case 'S'|FORCE_ANSI: {\r
-                    FX_LPCSTR pstrNextArg = va_arg(argList, FX_LPCSTR);\r
-                    if (pstrNextArg == NULL) {\r
-                        nItemLen = 6;\r
-                    } else {\r
-                        nItemLen = (FX_STRSIZE)FXSYS_strlen(pstrNextArg);\r
-                        if (nItemLen < 1) {\r
-                            nItemLen = 1;\r
-                        }\r
-                    }\r
-                }\r
-                break;\r
-            case 's'|FORCE_UNICODE:\r
-            case 'S'|FORCE_UNICODE: {\r
-                    FX_LPWSTR pstrNextArg = va_arg(argList, FX_LPWSTR);\r
-                    if (pstrNextArg == NULL) {\r
-                        nItemLen = 6;\r
-                    } else {\r
-                        nItemLen = (FX_STRSIZE)FXSYS_wcslen(pstrNextArg);\r
-                        if (nItemLen < 1) {\r
-                            nItemLen = 1;\r
-                        }\r
-                    }\r
-                }\r
-                break;\r
-        }\r
-        if (nItemLen != 0) {\r
-            if (nPrecision != 0 && nItemLen > nPrecision) {\r
-                nItemLen = nPrecision;\r
-            }\r
-            if (nItemLen < nWidth) {\r
-                nItemLen = nWidth;\r
-            }\r
-        } else {\r
-            switch (*lpsz) {\r
-                case 'd':\r
-                case 'i':\r
-                case 'u':\r
-                case 'x':\r
-                case 'X':\r
-                case 'o':\r
-                    if (nModifier & FORCE_INT64) {\r
-                        va_arg(argList, FX_INT64);\r
-                    } else {\r
-                        va_arg(argList, int);\r
-                    }\r
-                    nItemLen = 32;\r
-                    if (nItemLen < nWidth + nPrecision) {\r
-                        nItemLen = nWidth + nPrecision;\r
-                    }\r
-                    break;\r
-                case 'a':\r
-                case 'A':\r
-                case 'e':\r
-                case 'E':\r
-                case 'g':\r
-                case 'G':\r
-                    va_arg(argList, double);\r
-                    nItemLen = 128;\r
-                    if (nItemLen < nWidth + nPrecision) {\r
-                        nItemLen = nWidth + nPrecision;\r
-                    }\r
-                    break;\r
-                case 'f':\r
-                    if (nWidth + nPrecision > 100) {\r
-                        nItemLen = nPrecision + nWidth + 128;\r
-                    } else {\r
-                        double f;\r
-                        char pszTemp[256];\r
-                        f = va_arg(argList, double);\r
-                        FXSYS_snprintf(pszTemp, sizeof(pszTemp), "%*.*f", nWidth, nPrecision + 6, f );\r
-                        nItemLen = (FX_STRSIZE)FXSYS_strlen(pszTemp);\r
-                    }\r
-                    break;\r
-                case 'p':\r
-                    va_arg(argList, void*);\r
-                    nItemLen = 32;\r
-                    if (nItemLen < nWidth + nPrecision) {\r
-                        nItemLen = nWidth + nPrecision;\r
-                    }\r
-                    break;\r
-                case 'n':\r
-                    va_arg(argList, int*);\r
-                    break;\r
-            }\r
-        }\r
-        nMaxLen += nItemLen;\r
-    }\r
-    GetBuffer(nMaxLen);\r
-    if (m_pData) {\r
-        FXSYS_vswprintf((wchar_t*)m_pData->m_String, nMaxLen + 1, (const wchar_t*)lpszFormat, argListSave);\r
-        ReleaseBuffer();\r
-    }\r
-    va_end(argListSave);\r
-}\r
-void CFX_WideString::Format(FX_LPCWSTR lpszFormat, ...)\r
-{\r
-    va_list argList;\r
-    va_start(argList, lpszFormat);\r
-    FormatV(lpszFormat, argList);\r
-    va_end(argList);\r
-}\r
-FX_FLOAT FX_wtof(FX_LPCWSTR str, int len)\r
-{\r
-    if (len == 0) {\r
-        return 0.0;\r
-    }\r
-    int cc = 0;\r
-    FX_BOOL bNegative = FALSE;\r
-    if (str[0] == '+') {\r
-        cc++;\r
-    } else if (str[0] == '-') {\r
-        bNegative = TRUE;\r
-        cc++;\r
-    }\r
-    int integer = 0;\r
-    while (cc < len) {\r
-        if (str[cc] == '.') {\r
-            break;\r
-        }\r
-        integer = integer * 10 + str[cc] - '0';\r
-        cc ++;\r
-    }\r
-    FX_FLOAT fraction = 0;\r
-    if (str[cc] == '.') {\r
-        cc ++;\r
-        FX_FLOAT scale = 0.1f;\r
-        while (cc < len) {\r
-            fraction += scale * (str[cc] - '0');\r
-            scale *= 0.1f;\r
-            cc ++;\r
-        }\r
-    }\r
-    fraction += (FX_FLOAT)integer;\r
-    return bNegative ? -fraction : fraction;\r
-}\r
-int CFX_WideString::GetInteger() const\r
-{\r
-    if (m_pData == NULL) {\r
-        return 0;\r
-    }\r
-    return FXSYS_wtoi(m_pData->m_String);\r
-}\r
-FX_FLOAT CFX_WideString::GetFloat() const\r
-{\r
-    if (m_pData == NULL) {\r
-        return 0.0;\r
-    }\r
-    return FX_wtof(m_pData->m_String, m_pData->m_nDataLength);\r
-}\r
-void CFX_WideStringL::Empty(IFX_Allocator* pAllocator)\r
-{\r
-    if (m_Ptr) {\r
-        FX_Allocator_Free(pAllocator, (FX_LPVOID)m_Ptr);\r
-    }\r
-    m_Ptr = NULL, m_Length = 0;\r
-}\r
-void CFX_WideStringL::Set(FX_WSTR src, IFX_Allocator* pAllocator)\r
-{\r
-    Empty(pAllocator);\r
-    if (src.GetPtr() != NULL && src.GetLength() > 0) {\r
-        FX_LPWSTR str = FX_Allocator_Alloc(pAllocator, FX_WCHAR, src.GetLength() + 1);\r
-        if (!str) {\r
-            return;\r
-        }\r
-        FXSYS_memcpy32(str, src.GetPtr(), src.GetLength()*sizeof(FX_WCHAR));\r
-        str[src.GetLength()] = '\0';\r
-        *(FX_LPWSTR*)(&m_Ptr) = str;\r
-        m_Length = src.GetLength();\r
-    }\r
-}\r
-int CFX_WideStringL::GetInteger() const\r
-{\r
-    if (!m_Ptr) {\r
-        return 0;\r
-    }\r
-    return FXSYS_wtoi(m_Ptr);\r
-}\r
-FX_FLOAT CFX_WideStringL::GetFloat() const\r
-{\r
-    if (!m_Ptr) {\r
-        return 0.0f;\r
-    }\r
-    return FX_wtof(m_Ptr, m_Length);\r
-}\r
-void CFX_WideStringL::TrimRight(FX_LPCWSTR lpszTargets)\r
-{\r
-    if (!lpszTargets || *lpszTargets == 0 || !m_Ptr || m_Length < 1) {\r
-        return;\r
-    }\r
-    FX_STRSIZE pos = m_Length;\r
-    while (pos) {\r
-        if (FXSYS_wcschr(lpszTargets, m_Ptr[pos - 1]) == NULL) {\r
-            break;\r
-        }\r
-        pos --;\r
-    }\r
-    if (pos < m_Length) {\r
-        (*(FX_LPWSTR*)(&m_Ptr))[pos] = 0;\r
-        m_Length = pos;\r
-    }\r
-}\r
-static CFX_ByteString _DefMap_GetByteString(CFX_CharMap* pCharMap, const CFX_WideString& widestr)\r
-{\r
-    int src_len = widestr.GetLength();\r
-    int codepage = pCharMap->m_GetCodePage ? pCharMap->m_GetCodePage() : 0;\r
-    int dest_len = FXSYS_WideCharToMultiByte(codepage, 0, widestr, src_len, NULL, 0, NULL, NULL);\r
-    if (dest_len == 0) {\r
-        return CFX_ByteString();\r
-    }\r
-    CFX_ByteString bytestr;\r
-    FX_LPSTR dest_buf = bytestr.GetBuffer(dest_len);\r
-    FXSYS_WideCharToMultiByte(codepage, 0, widestr, src_len, dest_buf, dest_len, NULL, NULL);\r
-    bytestr.ReleaseBuffer(dest_len);\r
-    return bytestr;\r
-}\r
-static CFX_WideString _DefMap_GetWideString(CFX_CharMap* pCharMap, const CFX_ByteString& bytestr)\r
-{\r
-    int src_len = bytestr.GetLength();\r
-    int codepage = pCharMap->m_GetCodePage ? pCharMap->m_GetCodePage() : 0;\r
-    int dest_len = FXSYS_MultiByteToWideChar(codepage, 0, bytestr, src_len, NULL, 0);\r
-    if (dest_len == 0) {\r
-        return CFX_WideString();\r
-    }\r
-    CFX_WideString widestr;\r
-    FX_LPWSTR dest_buf = widestr.GetBuffer(dest_len);\r
-    FXSYS_MultiByteToWideChar(codepage, 0, bytestr, src_len, dest_buf, dest_len);\r
-    widestr.ReleaseBuffer(dest_len);\r
-    return widestr;\r
-}\r
-static int _DefMap_GetGBKCodePage()\r
-{\r
-    return 936;\r
-}\r
-static int _DefMap_GetUHCCodePage()\r
-{\r
-    return 949;\r
-}\r
-static int _DefMap_GetJISCodePage()\r
-{\r
-    return 932;\r
-}\r
-static int _DefMap_GetBig5CodePage()\r
-{\r
-    return 950;\r
-}\r
-static const CFX_CharMap g_DefaultMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, NULL};\r
-static const CFX_CharMap g_DefaultGBKMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetGBKCodePage};\r
-static const CFX_CharMap g_DefaultJISMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetJISCodePage};\r
-static const CFX_CharMap g_DefaultUHCMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetUHCCodePage};\r
-static const CFX_CharMap g_DefaultBig5Mapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetBig5CodePage};\r
-CFX_CharMap* CFX_CharMap::GetDefaultMapper(FX_INT32 codepage)\r
-{\r
-    switch (codepage) {\r
-        case 0:\r
-            return (CFX_CharMap*)&g_DefaultMapper;\r
-        case 932:\r
-            return (CFX_CharMap*)&g_DefaultJISMapper;\r
-        case 936:\r
-            return (CFX_CharMap*)&g_DefaultGBKMapper;\r
-        case 949:\r
-            return (CFX_CharMap*)&g_DefaultUHCMapper;\r
-        case 950:\r
-            return (CFX_CharMap*)&g_DefaultBig5Mapper;\r
-    }\r
-    return NULL;\r
-}\r
+// Copyright 2014 PDFium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
+
+#include <stddef.h>  // For offsetof().
+
+#include "../../include/fxcrt/fx_basic.h"
+#include "../../../third_party/base/numerics/safe_math.h"
+
+// static
+CFX_WideString::StringData* CFX_WideString::StringData::Create(int nLen)
+{
+    // TODO(palmer): |nLen| should really be declared as |size_t|, or
+    // at least unsigned.
+    if (nLen == 0 || nLen < 0) {
+        return NULL;
+    }
+
+    // Fixed portion of header plus a NUL wide char not in m_nAllocLength.
+    int overhead = offsetof(StringData, m_String) + sizeof(FX_WCHAR);
+    pdfium::base::CheckedNumeric<int> iSize = nLen;
+    iSize *= sizeof(FX_WCHAR);
+    iSize += overhead;
+
+    // Now round to an 8-byte boundary. We'd expect that this is the minimum
+    // granularity of any of the underlying allocators, so there may be cases
+    // where we can save a re-alloc when adding a few characters to a string
+    // by using this otherwise wasted space.
+    iSize += 7;
+    int totalSize = iSize.ValueOrDie() & ~7;
+    int usableLen = (totalSize - overhead) / sizeof(FX_WCHAR);
+    FXSYS_assert(usableLen >= nLen);
+
+    void* pData = FX_Alloc(FX_BYTE, iSize.ValueOrDie());
+    if (!pData) {
+        return NULL;
+    }
+    return new (pData) StringData(nLen, usableLen);
+}
+CFX_WideString::~CFX_WideString()
+{
+    if (m_pData) {
+        m_pData->Release();
+    }
+}
+CFX_WideString::CFX_WideString(const CFX_WideString& stringSrc)
+{
+    if (stringSrc.m_pData == NULL) {
+        m_pData = NULL;
+        return;
+    }
+    if (stringSrc.m_pData->m_nRefs >= 0) {
+        m_pData = stringSrc.m_pData;
+        m_pData->Retain();
+    } else {
+        m_pData = NULL;
+        *this = stringSrc;
+    }
+}
+CFX_WideString::CFX_WideString(FX_LPCWSTR lpsz, FX_STRSIZE nLen) {
+    if (nLen < 0) {
+        nLen = lpsz ? FXSYS_wcslen(lpsz) : 0;
+    }
+    if (nLen) {
+        m_pData = StringData::Create(nLen);
+        if (m_pData) {
+            FXSYS_memcpy32(m_pData->m_String, lpsz, nLen * sizeof(FX_WCHAR));
+        }
+    } else {
+        m_pData = NULL;
+    }
+}
+CFX_WideString::CFX_WideString(FX_WCHAR ch)
+{
+    m_pData = StringData::Create(1);
+    if (m_pData) {
+        m_pData->m_String[0] = ch;
+    }
+}
+CFX_WideString::CFX_WideString(const CFX_WideStringC& str)
+{
+    if (str.IsEmpty()) {
+        m_pData = NULL;
+        return;
+    }
+    m_pData = StringData::Create(str.GetLength());
+    if (m_pData) {
+        FXSYS_memcpy32(m_pData->m_String, str.GetPtr(), str.GetLength()*sizeof(FX_WCHAR));
+    }
+}
+CFX_WideString::CFX_WideString(const CFX_WideStringC& str1, const CFX_WideStringC& str2)
+{
+    m_pData = NULL;
+    int nNewLen = str1.GetLength() + str2.GetLength();
+    if (nNewLen == 0) {
+        return;
+    }
+    m_pData = StringData::Create(nNewLen);
+    if (m_pData) {
+        FXSYS_memcpy32(m_pData->m_String, str1.GetPtr(), str1.GetLength()*sizeof(FX_WCHAR));
+        FXSYS_memcpy32(m_pData->m_String + str1.GetLength(), str2.GetPtr(), str2.GetLength()*sizeof(FX_WCHAR));
+    }
+}
+void CFX_WideString::ReleaseBuffer(FX_STRSIZE nNewLength)
+{
+    if (m_pData == NULL) {
+        return;
+    }
+    CopyBeforeWrite();
+    if (nNewLength == -1) {
+        nNewLength = m_pData ? FXSYS_wcslen(m_pData->m_String) : 0;
+    }
+    if (nNewLength == 0) {
+        Empty();
+        return;
+    }
+    FXSYS_assert(nNewLength <= m_pData->m_nAllocLength);
+    m_pData->m_nDataLength = nNewLength;
+    m_pData->m_String[nNewLength] = 0;
+}
+const CFX_WideString& CFX_WideString::operator=(FX_LPCWSTR lpsz)
+{
+    if (lpsz == NULL || lpsz[0] == 0) {
+        Empty();
+    } else {
+        AssignCopy(FXSYS_wcslen(lpsz), lpsz);
+    }
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator=(const CFX_WideStringC& stringSrc)
+{
+    if (stringSrc.IsEmpty()) {
+        Empty();
+    } else {
+        AssignCopy(stringSrc.GetLength(), stringSrc.GetPtr());
+    }
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator=(const CFX_WideString& stringSrc)
+{
+    if (m_pData == stringSrc.m_pData) {
+        return *this;
+    }
+    if (stringSrc.IsEmpty()) {
+        Empty();
+    } else if ((m_pData && m_pData->m_nRefs < 0) ||
+               (stringSrc.m_pData && stringSrc.m_pData->m_nRefs < 0)) {
+        AssignCopy(stringSrc.m_pData->m_nDataLength, stringSrc.m_pData->m_String);
+    } else {
+        Empty();
+        m_pData = stringSrc.m_pData;
+        if (m_pData) {
+            m_pData->Retain();
+        }
+    }
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator+=(FX_WCHAR ch)
+{
+    ConcatInPlace(1, &ch);
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator+=(FX_LPCWSTR lpsz)
+{
+    if (lpsz) {
+        ConcatInPlace(FXSYS_wcslen(lpsz), lpsz);
+    }
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator+=(const CFX_WideString& string)
+{
+    if (string.m_pData == NULL) {
+        return *this;
+    }
+    ConcatInPlace(string.m_pData->m_nDataLength, string.m_pData->m_String);
+    return *this;
+}
+const CFX_WideString& CFX_WideString::operator+=(const CFX_WideStringC& string)
+{
+    if (string.IsEmpty()) {
+        return *this;
+    }
+    ConcatInPlace(string.GetLength(), string.GetPtr());
+    return *this;
+}
+bool CFX_WideString::Equal(const wchar_t* ptr) const
+{
+    if (!m_pData) {
+        return !ptr || ptr[0] == L'\0';
+    }
+    if (!ptr) {
+        return m_pData->m_nDataLength == 0;
+    }
+    return wcslen(ptr) == m_pData->m_nDataLength &&
+            wmemcmp(ptr, m_pData->m_String, m_pData->m_nDataLength) == 0;
+}
+bool CFX_WideString::Equal(const CFX_WideStringC& str) const
+{
+    if (m_pData == NULL) {
+        return str.IsEmpty();
+    }
+    return str.GetLength() == m_pData->m_nDataLength &&
+        wmemcmp(str.GetPtr(), m_pData->m_String, m_pData->m_nDataLength) == 0;
+}
+bool CFX_WideString::Equal(const CFX_WideString& other) const
+{
+    if (IsEmpty()) {
+        return other.IsEmpty();
+    }
+    if (other.IsEmpty()) {
+        return false;
+    }
+    return other.m_pData->m_nDataLength == m_pData->m_nDataLength &&
+        wmemcmp(other.m_pData->m_String,
+                m_pData->m_String,
+                m_pData->m_nDataLength) == 0;
+}
+void CFX_WideString::Empty()
+{
+    if (m_pData) {
+        m_pData->Release();
+        m_pData = NULL;
+    }
+}
+void CFX_WideString::ConcatInPlace(FX_STRSIZE nSrcLen, FX_LPCWSTR lpszSrcData)
+{
+    if (nSrcLen == 0 || lpszSrcData == NULL) {
+        return;
+    }
+    if (m_pData == NULL) {
+        m_pData = StringData::Create(nSrcLen);
+        if (m_pData) {
+            FXSYS_memcpy32(m_pData->m_String, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));
+        }
+        return;
+    }
+    if (m_pData->m_nRefs > 1 || m_pData->m_nDataLength + nSrcLen > m_pData->m_nAllocLength) {
+        ConcatCopy(m_pData->m_nDataLength, m_pData->m_String, nSrcLen, lpszSrcData);
+    } else {
+        FXSYS_memcpy32(m_pData->m_String + m_pData->m_nDataLength, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));
+        m_pData->m_nDataLength += nSrcLen;
+        m_pData->m_String[m_pData->m_nDataLength] = 0;
+    }
+}
+void CFX_WideString::ConcatCopy(FX_STRSIZE nSrc1Len, FX_LPCWSTR lpszSrc1Data,
+                                FX_STRSIZE nSrc2Len, FX_LPCWSTR lpszSrc2Data)
+{
+    FX_STRSIZE nNewLen = nSrc1Len + nSrc2Len;
+    if (nNewLen <= 0) {
+        return;
+    }
+    // Don't release until done copying, might be one of the arguments.
+    StringData* pOldData = m_pData;
+    m_pData = StringData::Create(nNewLen);
+    if (m_pData) {
+        wmemcpy(m_pData->m_String, lpszSrc1Data, nSrc1Len);
+        wmemcpy(m_pData->m_String + nSrc1Len, lpszSrc2Data, nSrc2Len);
+    }
+    pOldData->Release();
+}
+void CFX_WideString::CopyBeforeWrite()
+{
+    if (m_pData == NULL || m_pData->m_nRefs <= 1) {
+        return;
+    }
+    StringData* pData = m_pData;
+    m_pData->Release();
+    FX_STRSIZE nDataLength = pData->m_nDataLength;
+    m_pData = StringData::Create(nDataLength);
+    if (m_pData != NULL) {
+        FXSYS_memcpy32(m_pData->m_String, pData->m_String, (nDataLength + 1) * sizeof(FX_WCHAR));
+    }
+}
+void CFX_WideString::AllocBeforeWrite(FX_STRSIZE nLen)
+{
+    if (m_pData && m_pData->m_nRefs <= 1 && m_pData->m_nAllocLength >= nLen) {
+        return;
+    }
+    Empty();
+    m_pData = StringData::Create(nLen);
+}
+void CFX_WideString::AssignCopy(FX_STRSIZE nSrcLen, FX_LPCWSTR lpszSrcData)
+{
+    AllocBeforeWrite(nSrcLen);
+    FXSYS_memcpy32(m_pData->m_String, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));
+    m_pData->m_nDataLength = nSrcLen;
+    m_pData->m_String[nSrcLen] = 0;
+}
+int CFX_WideString::Compare(FX_LPCWSTR lpsz) const
+{
+    if (m_pData == NULL) {
+        return (lpsz == NULL || lpsz[0] == 0) ? 0 : -1;
+    }
+    return FXSYS_wcscmp(m_pData->m_String, lpsz);
+}
+CFX_ByteString CFX_WideString::UTF8Encode() const
+{
+    return FX_UTF8Encode(*this);
+}
+CFX_ByteString CFX_WideString::UTF16LE_Encode() const
+{
+    if (m_pData == NULL) {
+        return CFX_ByteString(FX_BSTRC("\0\0"));
+    }
+    int len = m_pData->m_nDataLength;
+    CFX_ByteString result;
+    FX_LPSTR buffer = result.GetBuffer(len * 2 + 2);
+    for (int i = 0; i < len; i ++) {
+        buffer[i * 2] = m_pData->m_String[i] & 0xff;
+        buffer[i * 2 + 1] = m_pData->m_String[i] >> 8;
+    }
+    buffer[len * 2] = 0;
+    buffer[len * 2 + 1] = 0;
+    result.ReleaseBuffer(len * 2 + 2);
+    return result;
+}
+void CFX_WideString::ConvertFrom(const CFX_ByteString& str, CFX_CharMap* pCharMap)
+{
+    if (pCharMap == NULL) {
+        pCharMap = CFX_CharMap::GetDefaultMapper();
+    }
+    *this = pCharMap->m_GetWideString(pCharMap, str);
+}
+void CFX_WideString::Reserve(FX_STRSIZE len)
+{
+    GetBuffer(len);
+    ReleaseBuffer(GetLength());
+}
+FX_LPWSTR CFX_WideString::GetBuffer(FX_STRSIZE nMinBufLength)
+{
+    if (m_pData == NULL && nMinBufLength == 0) {
+        return NULL;
+    }
+    if (m_pData && m_pData->m_nRefs <= 1 && m_pData->m_nAllocLength >= nMinBufLength) {
+        return m_pData->m_String;
+    }
+    if (m_pData == NULL) {
+        m_pData = StringData::Create(nMinBufLength);
+        if (!m_pData) {
+            return NULL;
+        }
+        m_pData->m_nDataLength = 0;
+        m_pData->m_String[0] = 0;
+        return m_pData->m_String;
+    }
+    StringData* pOldData = m_pData;
+    FX_STRSIZE nOldLen = pOldData->m_nDataLength;
+    if (nMinBufLength < nOldLen) {
+        nMinBufLength = nOldLen;
+    }
+    m_pData = StringData::Create(nMinBufLength);
+    if (!m_pData) {
+        return NULL;
+    }
+    FXSYS_memcpy32(m_pData->m_String, pOldData->m_String, (nOldLen + 1)*sizeof(FX_WCHAR));
+    m_pData->m_nDataLength = nOldLen;
+    pOldData->Release();
+    return m_pData->m_String;
+}
+CFX_WideString CFX_WideString::FromLocal(const char* str, FX_STRSIZE len)
+{
+    CFX_WideString result;
+    result.ConvertFrom(CFX_ByteString(str, len));
+    return result;
+}
+CFX_WideString CFX_WideString::FromUTF8(const char* str, FX_STRSIZE len)
+{
+    if (!str || 0 == len) {
+        return CFX_WideString();
+    }
+
+    CFX_UTF8Decoder decoder;
+    for (FX_STRSIZE i = 0; i < len; i ++) {
+        decoder.Input(str[i]);
+    }
+    return decoder.GetResult();
+}
+CFX_WideString CFX_WideString::FromUTF16LE(const unsigned short* wstr, FX_STRSIZE wlen)
+{
+    if (!wstr || 0 == wlen) {
+        return CFX_WideString();
+    }
+
+    CFX_WideString result;
+    FX_WCHAR* buf = result.GetBuffer(wlen);
+    for (int i = 0; i < wlen; i ++) {
+        buf[i] = wstr[i];
+    }
+    result.ReleaseBuffer(wlen);
+    return result;
+}
+FX_STRSIZE CFX_WideString::WStringLength(const unsigned short* str)
+{
+    FX_STRSIZE len = 0;
+    if (str)
+        while (str[len]) len++;
+    return len;
+}
+
+
+
+void CFX_WideString::AllocCopy(CFX_WideString& dest, FX_STRSIZE nCopyLen, FX_STRSIZE nCopyIndex) const
+{
+    // |FX_STRSIZE| is currently typedef'd as in |int|. TODO(palmer): It
+    // should be a |size_t|, or at least unsigned.
+    if (nCopyLen == 0 || nCopyLen < 0) {
+        return;
+    }
+    pdfium::base::CheckedNumeric<FX_STRSIZE> iSize = static_cast<FX_STRSIZE>(sizeof(FX_WCHAR));
+    iSize *= nCopyLen;
+    ASSERT(dest.m_pData == NULL);
+    dest.m_pData = StringData::Create(nCopyLen);
+    if (dest.m_pData) {
+        FXSYS_memcpy32(dest.m_pData->m_String, m_pData->m_String + nCopyIndex, iSize.ValueOrDie());
+    }
+}
+CFX_WideString CFX_WideString::Left(FX_STRSIZE nCount) const
+{
+    if (m_pData == NULL) {
+        return CFX_WideString();
+    }
+    if (nCount < 0) {
+        nCount = 0;
+    }
+    if (nCount >= m_pData->m_nDataLength) {
+        return *this;
+    }
+    CFX_WideString dest;
+    AllocCopy(dest, nCount, 0);
+    return dest;
+}
+CFX_WideString CFX_WideString::Mid(FX_STRSIZE nFirst) const
+{
+    return Mid(nFirst, m_pData->m_nDataLength - nFirst);
+}
+CFX_WideString CFX_WideString::Mid(FX_STRSIZE nFirst, FX_STRSIZE nCount) const
+{
+    if (m_pData == NULL) {
+        return CFX_WideString();
+    }
+    if (nFirst < 0) {
+        nFirst = 0;
+    }
+    if (nCount < 0) {
+        nCount = 0;
+    }
+    if (nFirst + nCount > m_pData->m_nDataLength) {
+        nCount = m_pData->m_nDataLength - nFirst;
+    }
+    if (nFirst > m_pData->m_nDataLength) {
+        nCount = 0;
+    }
+    if (nFirst == 0 && nFirst + nCount == m_pData->m_nDataLength) {
+        return *this;
+    }
+    CFX_WideString dest;
+    AllocCopy(dest, nCount, nFirst);
+    return dest;
+}
+CFX_WideString CFX_WideString::Right(FX_STRSIZE nCount) const
+{
+    if (m_pData == NULL) {
+        return CFX_WideString();
+    }
+    if (nCount < 0) {
+        nCount = 0;
+    }
+    if (nCount >= m_pData->m_nDataLength) {
+        return *this;
+    }
+    CFX_WideString dest;
+    AllocCopy(dest, nCount, m_pData->m_nDataLength - nCount);
+    return dest;
+}
+int CFX_WideString::CompareNoCase(FX_LPCWSTR lpsz) const
+{
+    if (m_pData == NULL) {
+        return (lpsz == NULL || lpsz[0] == 0) ? 0 : -1;
+    }
+    return FXSYS_wcsicmp(m_pData->m_String, lpsz);
+}
+int CFX_WideString::Compare(const CFX_WideString& str) const
+{
+    if (m_pData == NULL) {
+        if (str.m_pData == NULL) {
+            return 0;
+        }
+        return -1;
+    } else if (str.m_pData == NULL) {
+        return 1;
+    }
+    int this_len = m_pData->m_nDataLength;
+    int that_len = str.m_pData->m_nDataLength;
+    int min_len = this_len < that_len ? this_len : that_len;
+    for (int i = 0; i < min_len; i ++) {
+        if (m_pData->m_String[i] < str.m_pData->m_String[i]) {
+            return -1;
+        } else if (m_pData->m_String[i] > str.m_pData->m_String[i]) {
+            return 1;
+        }
+    }
+    if (this_len < that_len) {
+        return -1;
+    } else if (this_len > that_len) {
+        return 1;
+    }
+    return 0;
+}
+void CFX_WideString::SetAt(FX_STRSIZE nIndex, FX_WCHAR ch)
+{
+    if (m_pData == NULL) {
+        return;
+    }
+    ASSERT(nIndex >= 0);
+    ASSERT(nIndex < m_pData->m_nDataLength);
+    CopyBeforeWrite();
+    m_pData->m_String[nIndex] = ch;
+}
+void CFX_WideString::MakeLower()
+{
+    if (m_pData == NULL) {
+        return;
+    }
+    CopyBeforeWrite();
+    if (GetLength() < 1) {
+        return;
+    }
+    FXSYS_wcslwr(m_pData->m_String);
+}
+void CFX_WideString::MakeUpper()
+{
+    if (m_pData == NULL) {
+        return;
+    }
+    CopyBeforeWrite();
+    if (GetLength() < 1) {
+        return;
+    }
+    FXSYS_wcsupr(m_pData->m_String);
+}
+FX_STRSIZE CFX_WideString::Find(FX_LPCWSTR lpszSub, FX_STRSIZE nStart) const
+{
+    FX_STRSIZE nLength = GetLength();
+    if (nLength < 1 || nStart > nLength) {
+        return -1;
+    }
+    FX_LPCWSTR lpsz = FXSYS_wcsstr(m_pData->m_String + nStart, lpszSub);
+    return (lpsz == NULL) ? -1 : (int)(lpsz - m_pData->m_String);
+}
+FX_STRSIZE CFX_WideString::Find(FX_WCHAR ch, FX_STRSIZE nStart) const
+{
+    if (m_pData == NULL) {
+        return -1;
+    }
+    FX_STRSIZE nLength = m_pData->m_nDataLength;
+    if (nStart >= nLength) {
+        return -1;
+    }
+    FX_LPCWSTR lpsz = FXSYS_wcschr(m_pData->m_String + nStart, ch);
+    return (lpsz == NULL) ? -1 : (int)(lpsz - m_pData->m_String);
+}
+void CFX_WideString::TrimRight(FX_LPCWSTR lpszTargetList)
+{
+    FXSYS_assert(lpszTargetList != NULL);
+    if (m_pData == NULL || *lpszTargetList == 0) {
+        return;
+    }
+    CopyBeforeWrite();
+    FX_STRSIZE len = GetLength();
+    if (len < 1) {
+        return;
+    }
+    FX_STRSIZE pos = len;
+    while (pos) {
+        if (FXSYS_wcschr(lpszTargetList, m_pData->m_String[pos - 1]) == NULL) {
+            break;
+        }
+        pos --;
+    }
+    if (pos < len) {
+        m_pData->m_String[pos] = 0;
+        m_pData->m_nDataLength = pos;
+    }
+}
+void CFX_WideString::TrimRight(FX_WCHAR chTarget)
+{
+    FX_WCHAR str[2] = {chTarget, 0};
+    TrimRight(str);
+}
+void CFX_WideString::TrimRight()
+{
+    TrimRight(L"\x09\x0a\x0b\x0c\x0d\x20");
+}
+void CFX_WideString::TrimLeft(FX_LPCWSTR lpszTargets)
+{
+    FXSYS_assert(lpszTargets != NULL);
+    if (m_pData == NULL || *lpszTargets == 0) {
+        return;
+    }
+    CopyBeforeWrite();
+    if (GetLength() < 1) {
+        return;
+    }
+    FX_LPCWSTR lpsz = m_pData->m_String;
+    while (*lpsz != 0) {
+        if (FXSYS_wcschr(lpszTargets, *lpsz) == NULL) {
+            break;
+        }
+        lpsz ++;
+    }
+    if (lpsz != m_pData->m_String) {
+        int nDataLength = m_pData->m_nDataLength - (FX_STRSIZE)(lpsz - m_pData->m_String);
+        FXSYS_memmove32(m_pData->m_String, lpsz, (nDataLength + 1)*sizeof(FX_WCHAR));
+        m_pData->m_nDataLength = nDataLength;
+    }
+}
+void CFX_WideString::TrimLeft(FX_WCHAR chTarget)
+{
+    FX_WCHAR str[2] = {chTarget, 0};
+    TrimLeft(str);
+}
+void CFX_WideString::TrimLeft()
+{
+    TrimLeft(L"\x09\x0a\x0b\x0c\x0d\x20");
+}
+FX_STRSIZE CFX_WideString::Replace(FX_LPCWSTR lpszOld, FX_LPCWSTR lpszNew)
+{
+    if (GetLength() < 1) {
+        return 0;
+    }
+    if (lpszOld == NULL) {
+        return 0;
+    }
+    FX_STRSIZE nSourceLen = FXSYS_wcslen(lpszOld);
+    if (nSourceLen == 0) {
+        return 0;
+    }
+    FX_STRSIZE nReplacementLen = lpszNew ? FXSYS_wcslen(lpszNew) : 0;
+    FX_STRSIZE nCount = 0;
+    FX_LPWSTR lpszStart = m_pData->m_String;
+    FX_LPWSTR lpszEnd = m_pData->m_String + m_pData->m_nDataLength;
+    FX_LPWSTR lpszTarget;
+    {
+        while ((lpszTarget = (FX_LPWSTR)FXSYS_wcsstr(lpszStart, lpszOld)) != NULL && lpszStart < lpszEnd) {
+            nCount++;
+            lpszStart = lpszTarget + nSourceLen;
+        }
+    }
+    if (nCount > 0) {
+        CopyBeforeWrite();
+        FX_STRSIZE nOldLength = m_pData->m_nDataLength;
+        FX_STRSIZE nNewLength =  nOldLength + (nReplacementLen - nSourceLen) * nCount;
+        if (m_pData->m_nAllocLength < nNewLength || m_pData->m_nRefs > 1) {
+            StringData* pOldData = m_pData;
+            FX_LPCWSTR pstr = m_pData->m_String;
+            m_pData = StringData::Create(nNewLength);
+            if (!m_pData) {
+                return 0;
+            }
+            FXSYS_memcpy32(m_pData->m_String, pstr, pOldData->m_nDataLength * sizeof(FX_WCHAR));
+            pOldData->Release();
+        }
+        lpszStart = m_pData->m_String;
+        lpszEnd = m_pData->m_String + FX_MAX(m_pData->m_nDataLength, nNewLength);
+        {
+            while ((lpszTarget = (FX_LPWSTR)FXSYS_wcsstr(lpszStart, lpszOld)) != NULL && lpszStart < lpszEnd) {
+                FX_STRSIZE nBalance = nOldLength - (FX_STRSIZE)(lpszTarget - m_pData->m_String + nSourceLen);
+                FXSYS_memmove32(lpszTarget + nReplacementLen, lpszTarget + nSourceLen, nBalance * sizeof(FX_WCHAR));
+                FXSYS_memcpy32(lpszTarget, lpszNew, nReplacementLen * sizeof(FX_WCHAR));
+                lpszStart = lpszTarget + nReplacementLen;
+                lpszStart[nBalance] = 0;
+                nOldLength += (nReplacementLen - nSourceLen);
+            }
+        }
+        ASSERT(m_pData->m_String[nNewLength] == 0);
+        m_pData->m_nDataLength = nNewLength;
+    }
+    return nCount;
+}
+FX_STRSIZE CFX_WideString::Insert(FX_STRSIZE nIndex, FX_WCHAR ch)
+{
+    CopyBeforeWrite();
+    if (nIndex < 0) {
+        nIndex = 0;
+    }
+    FX_STRSIZE nNewLength = GetLength();
+    if (nIndex > nNewLength) {
+        nIndex = nNewLength;
+    }
+    nNewLength++;
+    if (m_pData == NULL || m_pData->m_nAllocLength < nNewLength) {
+        StringData* pOldData = m_pData;
+        FX_LPCWSTR pstr = m_pData->m_String;
+        m_pData = StringData::Create(nNewLength);
+        if (!m_pData) {
+            return 0;
+        }
+        if(pOldData != NULL) {
+            FXSYS_memmove32(m_pData->m_String, pstr, (pOldData->m_nDataLength + 1)*sizeof(FX_WCHAR));
+            pOldData->Release();
+        } else {
+            m_pData->m_String[0] = 0;
+        }
+    }
+    FXSYS_memmove32(m_pData->m_String + nIndex + 1,
+                    m_pData->m_String + nIndex, (nNewLength - nIndex)*sizeof(FX_WCHAR));
+    m_pData->m_String[nIndex] = ch;
+    m_pData->m_nDataLength = nNewLength;
+    return nNewLength;
+}
+FX_STRSIZE CFX_WideString::Delete(FX_STRSIZE nIndex, FX_STRSIZE nCount)
+{
+    if (GetLength() < 1) {
+        return 0;
+    }
+    if (nIndex < 0) {
+        nIndex = 0;
+    }
+    FX_STRSIZE nOldLength = m_pData->m_nDataLength;
+    if (nCount > 0 && nIndex < nOldLength) {
+        CopyBeforeWrite();
+        int nBytesToCopy = nOldLength - (nIndex + nCount) + 1;
+        FXSYS_memmove32(m_pData->m_String + nIndex,
+                        m_pData->m_String + nIndex + nCount, nBytesToCopy * sizeof(FX_WCHAR));
+        m_pData->m_nDataLength = nOldLength - nCount;
+    }
+    return m_pData->m_nDataLength;
+}
+FX_STRSIZE CFX_WideString::Remove(FX_WCHAR chRemove)
+{
+    if (m_pData == NULL) {
+        return 0;
+    }
+    CopyBeforeWrite();
+    if (GetLength() < 1) {
+        return 0;
+    }
+    FX_LPWSTR pstrSource = m_pData->m_String;
+    FX_LPWSTR pstrDest = m_pData->m_String;
+    FX_LPWSTR pstrEnd = m_pData->m_String + m_pData->m_nDataLength;
+    while (pstrSource < pstrEnd) {
+        if (*pstrSource != chRemove) {
+            *pstrDest = *pstrSource;
+            pstrDest ++;
+        }
+        pstrSource ++;
+    }
+    *pstrDest = 0;
+    FX_STRSIZE nCount = (FX_STRSIZE)(pstrSource - pstrDest);
+    m_pData->m_nDataLength -= nCount;
+    return nCount;
+}
+#define FORCE_ANSI      0x10000
+#define FORCE_UNICODE   0x20000
+#define FORCE_INT64     0x40000
+void CFX_WideString::FormatV(FX_LPCWSTR lpszFormat, va_list argList)
+{
+    va_list argListSave;
+#if defined(__ARMCC_VERSION) || (!defined(_MSC_VER) && (_FX_CPU_ == _FX_X64_ || _FX_CPU_ == _FX_IA64_ || _FX_CPU_ == _FX_ARM64_)) || defined(__native_client__)
+    va_copy(argListSave, argList);
+#else
+    argListSave = argList;
+#endif
+    int nMaxLen = 0;
+    for (FX_LPCWSTR lpsz = lpszFormat; *lpsz != 0; lpsz ++) {
+        if (*lpsz != '%' || *(lpsz = lpsz + 1) == '%') {
+            nMaxLen += FXSYS_wcslen(lpsz);
+            continue;
+        }
+        int nItemLen = 0;
+        int nWidth = 0;
+        for (; *lpsz != 0; lpsz ++) {
+            if (*lpsz == '#') {
+                nMaxLen += 2;
+            } else if (*lpsz == '*') {
+                nWidth = va_arg(argList, int);
+            } else if (*lpsz == '-' || *lpsz == '+' || *lpsz == '0' ||
+                       *lpsz == ' ')
+                ;
+            else {
+                break;
+            }
+        }
+        if (nWidth == 0) {
+            nWidth = FXSYS_wtoi(lpsz);
+            for (; *lpsz != 0 && (*lpsz) <= '9' && (*lpsz) >= '0'; lpsz ++)
+                ;
+        }
+        if (nWidth < 0 || nWidth > 128 * 1024) {
+            lpszFormat = L"Bad width";
+            nMaxLen = 10;
+            break;
+        }
+        int nPrecision = 0;
+        if (*lpsz == '.') {
+            lpsz ++;
+            if (*lpsz == '*') {
+                nPrecision = va_arg(argList, int);
+                lpsz ++;
+            } else {
+                nPrecision = FXSYS_wtoi(lpsz);
+                for (; *lpsz != 0 && (*lpsz) >= '0' && (*lpsz) <= '9'; lpsz ++)
+                    ;
+            }
+        }
+        if (nPrecision < 0 || nPrecision > 128 * 1024) {
+            lpszFormat = L"Bad precision";
+            nMaxLen = 14;
+            break;
+        }
+        int nModifier = 0;
+        if (*lpsz == L'I' && *(lpsz + 1) == L'6' && *(lpsz + 2) == L'4') {
+            lpsz += 3;
+            nModifier = FORCE_INT64;
+        } else {
+            switch (*lpsz) {
+                case 'h':
+                    nModifier = FORCE_ANSI;
+                    lpsz ++;
+                    break;
+                case 'l':
+                    nModifier = FORCE_UNICODE;
+                    lpsz ++;
+                    break;
+                case 'F':
+                case 'N':
+                case 'L':
+                    lpsz ++;
+                    break;
+            }
+        }
+        switch (*lpsz | nModifier) {
+            case 'c':
+            case 'C':
+                nItemLen = 2;
+                va_arg(argList, int);
+                break;
+            case 'c'|FORCE_ANSI:
+            case 'C'|FORCE_ANSI:
+                nItemLen = 2;
+                va_arg(argList, int);
+                break;
+            case 'c'|FORCE_UNICODE:
+            case 'C'|FORCE_UNICODE:
+                nItemLen = 2;
+                va_arg(argList, int);
+                break;
+            case 's': {
+                    FX_LPCWSTR pstrNextArg = va_arg(argList, FX_LPCWSTR);
+                    if (pstrNextArg == NULL) {
+                        nItemLen = 6;
+                    } else {
+                        nItemLen = FXSYS_wcslen(pstrNextArg);
+                        if (nItemLen < 1) {
+                            nItemLen = 1;
+                        }
+                    }
+                }
+                break;
+            case 'S': {
+                    FX_LPCSTR pstrNextArg = va_arg(argList, FX_LPCSTR);
+                    if (pstrNextArg == NULL) {
+                        nItemLen = 6;
+                    } else {
+                        nItemLen = FXSYS_strlen(pstrNextArg);
+                        if (nItemLen < 1) {
+                            nItemLen = 1;
+                        }
+                    }
+                }
+                break;
+            case 's'|FORCE_ANSI:
+            case 'S'|FORCE_ANSI: {
+                    FX_LPCSTR pstrNextArg = va_arg(argList, FX_LPCSTR);
+                    if (pstrNextArg == NULL) {
+                        nItemLen = 6;
+                    } else {
+                        nItemLen = FXSYS_strlen(pstrNextArg);
+                        if (nItemLen < 1) {
+                            nItemLen = 1;
+                        }
+                    }
+                }
+                break;
+            case 's'|FORCE_UNICODE:
+            case 'S'|FORCE_UNICODE: {
+                    FX_LPWSTR pstrNextArg = va_arg(argList, FX_LPWSTR);
+                    if (pstrNextArg == NULL) {
+                        nItemLen = 6;
+                    } else {
+                        nItemLen = FXSYS_wcslen(pstrNextArg);
+                        if (nItemLen < 1) {
+                            nItemLen = 1;
+                        }
+                    }
+                }
+                break;
+        }
+        if (nItemLen != 0) {
+            if (nPrecision != 0 && nItemLen > nPrecision) {
+                nItemLen = nPrecision;
+            }
+            if (nItemLen < nWidth) {
+                nItemLen = nWidth;
+            }
+        } else {
+            switch (*lpsz) {
+                case 'd':
+                case 'i':
+                case 'u':
+                case 'x':
+                case 'X':
+                case 'o':
+                    if (nModifier & FORCE_INT64) {
+                        va_arg(argList, FX_INT64);
+                    } else {
+                        va_arg(argList, int);
+                    }
+                    nItemLen = 32;
+                    if (nItemLen < nWidth + nPrecision) {
+                        nItemLen = nWidth + nPrecision;
+                    }
+                    break;
+                case 'a':
+                case 'A':
+                case 'e':
+                case 'E':
+                case 'g':
+                case 'G':
+                    va_arg(argList, double);
+                    nItemLen = 128;
+                    if (nItemLen < nWidth + nPrecision) {
+                        nItemLen = nWidth + nPrecision;
+                    }
+                    break;
+                case 'f':
+                    if (nWidth + nPrecision > 100) {
+                        nItemLen = nPrecision + nWidth + 128;
+                    } else {
+                        double f;
+                        char pszTemp[256];
+                        f = va_arg(argList, double);
+                        FXSYS_snprintf(pszTemp, sizeof(pszTemp), "%*.*f", nWidth, nPrecision + 6, f );
+                        nItemLen = FXSYS_strlen(pszTemp);
+                    }
+                    break;
+                case 'p':
+                    va_arg(argList, void*);
+                    nItemLen = 32;
+                    if (nItemLen < nWidth + nPrecision) {
+                        nItemLen = nWidth + nPrecision;
+                    }
+                    break;
+                case 'n':
+                    va_arg(argList, int*);
+                    break;
+            }
+        }
+        nMaxLen += nItemLen;
+    }
+    GetBuffer(nMaxLen);
+    if (m_pData) {
+        FXSYS_vswprintf((wchar_t*)m_pData->m_String, nMaxLen + 1, (const wchar_t*)lpszFormat, argListSave);
+        ReleaseBuffer();
+    }
+    va_end(argListSave);
+}
+void CFX_WideString::Format(FX_LPCWSTR lpszFormat, ...)
+{
+    va_list argList;
+    va_start(argList, lpszFormat);
+    FormatV(lpszFormat, argList);
+    va_end(argList);
+}
+FX_FLOAT FX_wtof(FX_LPCWSTR str, int len)
+{
+    if (len == 0) {
+        return 0.0;
+    }
+    int cc = 0;
+    FX_BOOL bNegative = FALSE;
+    if (str[0] == '+') {
+        cc++;
+    } else if (str[0] == '-') {
+        bNegative = TRUE;
+        cc++;
+    }
+    int integer = 0;
+    while (cc < len) {
+        if (str[cc] == '.') {
+            break;
+        }
+        integer = integer * 10 + str[cc] - '0';
+        cc ++;
+    }
+    FX_FLOAT fraction = 0;
+    if (str[cc] == '.') {
+        cc ++;
+        FX_FLOAT scale = 0.1f;
+        while (cc < len) {
+            fraction += scale * (str[cc] - '0');
+            scale *= 0.1f;
+            cc ++;
+        }
+    }
+    fraction += (FX_FLOAT)integer;
+    return bNegative ? -fraction : fraction;
+}
+int CFX_WideString::GetInteger() const
+{
+    if (m_pData == NULL) {
+        return 0;
+    }
+    return FXSYS_wtoi(m_pData->m_String);
+}
+FX_FLOAT CFX_WideString::GetFloat() const
+{
+    if (m_pData == NULL) {
+        return 0.0;
+    }
+    return FX_wtof(m_pData->m_String, m_pData->m_nDataLength);
+}
+static CFX_ByteString _DefMap_GetByteString(CFX_CharMap* pCharMap, const CFX_WideString& widestr)
+{
+    int src_len = widestr.GetLength();
+    int codepage = pCharMap->m_GetCodePage ? pCharMap->m_GetCodePage() : 0;
+    int dest_len = FXSYS_WideCharToMultiByte(codepage, 0, widestr.c_str(), src_len, NULL, 0, NULL, NULL);
+    if (dest_len == 0) {
+        return CFX_ByteString();
+    }
+    CFX_ByteString bytestr;
+    FX_LPSTR dest_buf = bytestr.GetBuffer(dest_len);
+    FXSYS_WideCharToMultiByte(codepage, 0, widestr.c_str(), src_len, dest_buf, dest_len, NULL, NULL);
+    bytestr.ReleaseBuffer(dest_len);
+    return bytestr;
+}
+static CFX_WideString _DefMap_GetWideString(CFX_CharMap* pCharMap, const CFX_ByteString& bytestr)
+{
+    int src_len = bytestr.GetLength();
+    int codepage = pCharMap->m_GetCodePage ? pCharMap->m_GetCodePage() : 0;
+    int dest_len = FXSYS_MultiByteToWideChar(codepage, 0, bytestr, src_len, NULL, 0);
+    if (dest_len == 0) {
+        return CFX_WideString();
+    }
+    CFX_WideString widestr;
+    FX_LPWSTR dest_buf = widestr.GetBuffer(dest_len);
+    FXSYS_MultiByteToWideChar(codepage, 0, bytestr, src_len, dest_buf, dest_len);
+    widestr.ReleaseBuffer(dest_len);
+    return widestr;
+}
+static int _DefMap_GetGBKCodePage()
+{
+    return 936;
+}
+static int _DefMap_GetUHCCodePage()
+{
+    return 949;
+}
+static int _DefMap_GetJISCodePage()
+{
+    return 932;
+}
+static int _DefMap_GetBig5CodePage()
+{
+    return 950;
+}
+static const CFX_CharMap g_DefaultMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, NULL};
+static const CFX_CharMap g_DefaultGBKMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetGBKCodePage};
+static const CFX_CharMap g_DefaultJISMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetJISCodePage};
+static const CFX_CharMap g_DefaultUHCMapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetUHCCodePage};
+static const CFX_CharMap g_DefaultBig5Mapper = {&_DefMap_GetWideString, &_DefMap_GetByteString, &_DefMap_GetBig5CodePage};
+CFX_CharMap* CFX_CharMap::GetDefaultMapper(FX_INT32 codepage)
+{
+    switch (codepage) {
+        case 0:
+            return (CFX_CharMap*)&g_DefaultMapper;
+        case 932:
+            return (CFX_CharMap*)&g_DefaultJISMapper;
+        case 936:
+            return (CFX_CharMap*)&g_DefaultGBKMapper;
+        case 949:
+            return (CFX_CharMap*)&g_DefaultUHCMapper;
+        case 950:
+            return (CFX_CharMap*)&g_DefaultBig5Mapper;
+    }
+    return NULL;
+}