Cleanup some numeric code.
[pdfium.git] / core / src / fxcrt / fx_basic_wstring.cpp
index 900f058..3310df7 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(uint8_t, totalSize);
+  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(const FX_WCHAR* lpsz, FX_STRSIZE nLen) {
+  if (nLen < 0) {
+    nLen = lpsz ? FXSYS_wcslen(lpsz) : 0;
+  }
+  if (nLen) {
+    m_pData = StringData::Create(nLen);
+    if (m_pData) {
+      FXSYS_memcpy(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_memcpy(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_memcpy(m_pData->m_String, str1.GetPtr(),
+                 str1.GetLength() * sizeof(FX_WCHAR));
+    FXSYS_memcpy(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=(const FX_WCHAR* 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+=(const FX_WCHAR* 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,
+                                   const FX_WCHAR* lpszSrcData) {
+  if (nSrcLen == 0 || lpszSrcData == NULL) {
+    return;
+  }
+  if (m_pData == NULL) {
+    m_pData = StringData::Create(nSrcLen);
+    if (m_pData) {
+      FXSYS_memcpy(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_memcpy(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,
+                                const FX_WCHAR* lpszSrc1Data,
+                                FX_STRSIZE nSrc2Len,
+                                const FX_WCHAR* 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_memcpy(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,
+                                const FX_WCHAR* lpszSrcData) {
+  AllocBeforeWrite(nSrcLen);
+  FXSYS_memcpy(m_pData->m_String, lpszSrcData, nSrcLen * sizeof(FX_WCHAR));
+  m_pData->m_nDataLength = nSrcLen;
+  m_pData->m_String[nSrcLen] = 0;
+}
+int CFX_WideString::Compare(const FX_WCHAR* 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_CHAR* 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_WCHAR* 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_memcpy(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_memcpy(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(const FX_WCHAR* 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;
+  }
+  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;
+    }
+    if (m_pData->m_String[i] > str.m_pData->m_String[i]) {
+      return 1;
+    }
+  }
+  if (this_len < that_len) {
+    return -1;
+  }
+  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(const FX_WCHAR* lpszSub,
+                                FX_STRSIZE nStart) const {
+  FX_STRSIZE nLength = GetLength();
+  if (nLength < 1 || nStart > nLength) {
+    return -1;
+  }
+  const FX_WCHAR* 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;
+  }
+  const FX_WCHAR* lpsz = FXSYS_wcschr(m_pData->m_String + nStart, ch);
+  return (lpsz == NULL) ? -1 : (int)(lpsz - m_pData->m_String);
+}
+void CFX_WideString::TrimRight(const FX_WCHAR* 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(const FX_WCHAR* lpszTargets) {
+  FXSYS_assert(lpszTargets != NULL);
+  if (m_pData == NULL || *lpszTargets == 0) {
+    return;
+  }
+  CopyBeforeWrite();
+  if (GetLength() < 1) {
+    return;
+  }
+  const FX_WCHAR* 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_memmove(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(const FX_WCHAR* lpszOld,
+                                   const FX_WCHAR* 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_WCHAR* lpszStart = m_pData->m_String;
+  FX_WCHAR* lpszEnd = m_pData->m_String + m_pData->m_nDataLength;
+  FX_WCHAR* lpszTarget;
+  {
+    while ((lpszTarget = (FX_WCHAR*)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;
+      const FX_WCHAR* pstr = m_pData->m_String;
+      m_pData = StringData::Create(nNewLength);
+      if (!m_pData) {
+        return 0;
+      }
+      FXSYS_memcpy(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_WCHAR*)FXSYS_wcsstr(lpszStart, lpszOld)) !=
+                 NULL &&
+             lpszStart < lpszEnd) {
+        FX_STRSIZE nBalance =
+            nOldLength -
+            (FX_STRSIZE)(lpszTarget - m_pData->m_String + nSourceLen);
+        FXSYS_memmove(lpszTarget + nReplacementLen, lpszTarget + nSourceLen,
+                      nBalance * sizeof(FX_WCHAR));
+        FXSYS_memcpy(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;
+    const FX_WCHAR* pstr = m_pData->m_String;
+    m_pData = StringData::Create(nNewLength);
+    if (!m_pData) {
+      return 0;
+    }
+    if (pOldData != NULL) {
+      FXSYS_memmove(m_pData->m_String, pstr,
+                    (pOldData->m_nDataLength + 1) * sizeof(FX_WCHAR));
+      pOldData->Release();
+    } else {
+      m_pData->m_String[0] = 0;
+    }
+  }
+  FXSYS_memmove(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_memmove(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_WCHAR* pstrSource = m_pData->m_String;
+  FX_WCHAR* pstrDest = m_pData->m_String;
+  FX_WCHAR* 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(const FX_WCHAR* 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 (const FX_WCHAR* 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 && std::isdigit(*lpsz); 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 && std::isdigit(*lpsz); 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': {
+        const FX_WCHAR* pstrNextArg = va_arg(argList, const FX_WCHAR*);
+        if (pstrNextArg == NULL) {
+          nItemLen = 6;
+        } else {
+          nItemLen = FXSYS_wcslen(pstrNextArg);
+          if (nItemLen < 1) {
+            nItemLen = 1;
+          }
+        }
+      } break;
+      case 'S': {
+        const FX_CHAR* pstrNextArg = va_arg(argList, const FX_CHAR*);
+        if (pstrNextArg == NULL) {
+          nItemLen = 6;
+        } else {
+          nItemLen = FXSYS_strlen(pstrNextArg);
+          if (nItemLen < 1) {
+            nItemLen = 1;
+          }
+        }
+      } break;
+      case 's' | FORCE_ANSI:
+      case 'S' | FORCE_ANSI: {
+        const FX_CHAR* pstrNextArg = va_arg(argList, const FX_CHAR*);
+        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_WCHAR* pstrNextArg = va_arg(argList, FX_WCHAR*);
+        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, int64_t);
+          } 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(const FX_WCHAR* lpszFormat, ...) {
+  va_list argList;
+  va_start(argList, lpszFormat);
+  FormatV(lpszFormat, argList);
+  va_end(argList);
+}
+FX_FLOAT FX_wtof(const FX_WCHAR* 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_CHAR* 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_WCHAR* 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(int32_t 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;
+}