/* * Copyright (C) 1999 Lars Knoll (knoll@kde.org) * (C) 1999 Antti Koivisto (koivisto@kde.org) * (C) 2001 Dirk Mueller ( mueller@kde.org ) * Copyright (C) 2003-2020 Apple Inc. All rights reserved. * Copyright (C) 2006 Andrew Wellington (proton@wiretapped.net) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include #if STRING_STATS #include #include #endif namespace WTF { using namespace Unicode; static_assert(sizeof(StringImpl) == 2 * sizeof(int) + 2 * sizeof(void*), "StringImpl should stay small"); #if STRING_STATS StringStats StringImpl::m_stringStats; std::atomic StringStats::s_stringRemovesTillPrintStats(s_printStringStatsFrequency); void StringStats::removeString(StringImpl& string) { unsigned length = string.length(); bool isSubString = string.isSubString(); --m_totalNumberStrings; if (string.is8Bit()) { --m_number8BitStrings; if (!isSubString) m_total8BitData -= length; } else { --m_number16BitStrings; if (!isSubString) m_total16BitData -= length; } if (!--s_stringRemovesTillPrintStats) { s_stringRemovesTillPrintStats = s_printStringStatsFrequency; printStats(); } } void StringStats::printStats() { dataLogF("String stats for process id %d:\n", getCurrentProcessID()); unsigned long long totalNumberCharacters = m_total8BitData + m_total16BitData; double percent8Bit = m_totalNumberStrings ? ((double)m_number8BitStrings * 100) / (double)m_totalNumberStrings : 0.0; double average8bitLength = m_number8BitStrings ? (double)m_total8BitData / (double)m_number8BitStrings : 0.0; dataLogF("%8u (%5.2f%%) 8 bit %12llu chars %12llu bytes avg length %6.1f\n", m_number8BitStrings.load(), percent8Bit, m_total8BitData.load(), m_total8BitData.load(), average8bitLength); double percent16Bit = m_totalNumberStrings ? ((double)m_number16BitStrings * 100) / (double)m_totalNumberStrings : 0.0; double average16bitLength = m_number16BitStrings ? (double)m_total16BitData / (double)m_number16BitStrings : 0.0; dataLogF("%8u (%5.2f%%) 16 bit %12llu chars %12llu bytes avg length %6.1f\n", m_number16BitStrings.load(), percent16Bit, m_total16BitData.load(), m_total16BitData * 2, average16bitLength); double averageLength = m_totalNumberStrings ? (double)totalNumberCharacters / (double)m_totalNumberStrings : 0.0; unsigned long long totalDataBytes = m_total8BitData + m_total16BitData * 2; dataLogF("%8u Total %12llu chars %12llu bytes avg length %6.1f\n", m_totalNumberStrings.load(), totalNumberCharacters, totalDataBytes, averageLength); unsigned long long totalSavedBytes = m_total8BitData; double percentSavings = totalSavedBytes ? ((double)totalSavedBytes * 100) / (double)(totalDataBytes + totalSavedBytes) : 0.0; dataLogF(" Total savings %12llu bytes (%5.2f%%)\n", totalSavedBytes, percentSavings); dataLogF("%8u StringImpl::ref calls\n", m_refCalls.load()); dataLogF("%8u StringImpl::deref calls\n", m_derefCalls.load()); } #endif DEFINE_ALLOCATOR_WITH_HEAP_IDENTIFIER(StringImpl); StringImpl::StaticStringImpl StringImpl::s_emptyAtomString("", StringImpl::StringAtom); StringImpl::~StringImpl() { ASSERT(!isStatic()); StringView::invalidate(*this); STRING_STATS_REMOVE_STRING(*this); if (isAtom()) { ASSERT(!isSymbol()); if (length()) AtomStringImpl::remove(static_cast(this)); } else if (isSymbol()) { auto& symbol = static_cast(*this); auto* symbolRegistry = symbol.symbolRegistry(); if (symbolRegistry) symbolRegistry->remove(*symbol.asRegisteredSymbolImpl()); } BufferOwnership ownership = bufferOwnership(); if (ownership == BufferInternal) return; if (ownership == BufferOwned) { // We use m_data8, but since it is a union with m_data16 this works either way. ASSERT(m_data8); StringImplMalloc::free(const_cast(m_data8)); return; } if (ownership == BufferExternal) { auto* external = static_cast(this); external->freeExternalBuffer(const_cast(m_data8), sizeInBytes()); external->m_free.~ExternalStringImplFreeFunction(); return; } ASSERT(ownership == BufferSubstring); ASSERT(substringBuffer()); substringBuffer()->deref(); } void StringImpl::destroy(StringImpl* stringImpl) { stringImpl->~StringImpl(); StringImplMalloc::free(stringImpl); } Ref StringImpl::createFromLiteral(const char* characters, unsigned length) { ASSERT_WITH_MESSAGE(length, "Use StringImpl::empty() to create an empty string"); ASSERT(charactersAreAllASCII(reinterpret_cast(characters), length)); return adoptRef(*new StringImpl(reinterpret_cast(characters), length, ConstructWithoutCopying)); } Ref StringImpl::createFromLiteral(const char* characters) { return createFromLiteral(characters, strlen(characters)); } Ref StringImpl::createWithoutCopying(const UChar* characters, unsigned length) { if (!length) return *empty(); return adoptRef(*new StringImpl(characters, length, ConstructWithoutCopying)); } Ref StringImpl::createWithoutCopying(const LChar* characters, unsigned length) { if (!length) return *empty(); return adoptRef(*new StringImpl(characters, length, ConstructWithoutCopying)); } template inline Ref StringImpl::createUninitializedInternal(unsigned length, CharacterType*& data) { if (!length) { data = nullptr; return *empty(); } return createUninitializedInternalNonEmpty(length, data); } template inline Ref StringImpl::createUninitializedInternalNonEmpty(unsigned length, CharacterType*& data) { ASSERT(length); // Allocate a single buffer large enough to contain the StringImpl // struct as well as the data which it contains. This removes one // heap allocation from this call. if (length > maxInternalLength()) CRASH(); StringImpl* string = static_cast(StringImplMalloc::malloc(allocationSize(length))); data = string->tailPointer(); return constructInternal(*string, length); } Ref StringImpl::createUninitialized(unsigned length, LChar*& data) { return createUninitializedInternal(length, data); } Ref StringImpl::createUninitialized(unsigned length, UChar*& data) { return createUninitializedInternal(length, data); } template inline Expected, UTF8ConversionError> StringImpl::reallocateInternal(Ref&& originalString, unsigned length, CharacterType*& data) { ASSERT(originalString->hasOneRef()); ASSERT(originalString->bufferOwnership() == BufferInternal); if (!length) { data = nullptr; return Ref(*empty()); } // Same as createUninitialized() except here we use fastRealloc. if (length > maxInternalLength()) return makeUnexpected(UTF8ConversionError::OutOfMemory); originalString->~StringImpl(); auto* string = static_cast(StringImplMalloc::tryRealloc(&originalString.leakRef(), allocationSize(length))); if (!string) return makeUnexpected(UTF8ConversionError::OutOfMemory); data = string->tailPointer(); return constructInternal(*string, length); } Ref StringImpl::reallocate(Ref&& originalString, unsigned length, LChar*& data) { auto expectedStringImpl = tryReallocate(WTFMove(originalString), length, data); RELEASE_ASSERT(expectedStringImpl); return WTFMove(expectedStringImpl.value()); } Ref StringImpl::reallocate(Ref&& originalString, unsigned length, UChar*& data) { auto expectedStringImpl = tryReallocate(WTFMove(originalString), length, data); RELEASE_ASSERT(expectedStringImpl); return WTFMove(expectedStringImpl.value()); } Expected, UTF8ConversionError> StringImpl::tryReallocate(Ref&& originalString, unsigned length, LChar*& data) { ASSERT(originalString->is8Bit()); return reallocateInternal(WTFMove(originalString), length, data); } Expected, UTF8ConversionError> StringImpl::tryReallocate(Ref&& originalString, unsigned length, UChar*& data) { ASSERT(!originalString->is8Bit()); return reallocateInternal(WTFMove(originalString), length, data); } template inline Ref StringImpl::createInternal(const CharacterType* characters, unsigned length) { if (!characters || !length) return *empty(); CharacterType* data; auto string = createUninitializedInternalNonEmpty(length, data); copyCharacters(data, characters, length); return string; } Ref StringImpl::create(const UChar* characters, unsigned length) { return createInternal(characters, length); } Ref StringImpl::create(const LChar* characters, unsigned length) { return createInternal(characters, length); } Ref StringImpl::createStaticStringImpl(const char* characters, unsigned length) { const LChar* lcharCharacters = reinterpret_cast(characters); ASSERT(charactersAreAllASCII(lcharCharacters, length)); Ref result = createInternal(lcharCharacters, length); result->setHash(StringHasher::computeHashAndMaskTop8Bits(lcharCharacters, length)); result->m_refCount |= s_refCountFlagIsStaticString; return result; } Ref StringImpl::create8BitIfPossible(const UChar* characters, unsigned length) { if (!characters || !length) return *empty(); LChar* data; auto string = createUninitializedInternalNonEmpty(length, data); for (size_t i = 0; i < length; ++i) { if (!isLatin1(characters[i])) return create(characters, length); data[i] = static_cast(characters[i]); } return string; } Ref StringImpl::create8BitIfPossible(const UChar* string) { return StringImpl::create8BitIfPossible(string, lengthOfNullTerminatedString(string)); } Ref StringImpl::create(const LChar* string) { if (!string) return *empty(); size_t length = strlen(reinterpret_cast(string)); if (length > MaxLength) CRASH(); return create(string, length); } Ref StringImpl::substring(unsigned start, unsigned length) { if (start >= m_length) return *empty(); unsigned maxLength = m_length - start; if (length >= maxLength) { if (!start) return *this; length = maxLength; } if (is8Bit()) return create(m_data8 + start, length); return create(m_data16 + start, length); } UChar32 StringImpl::characterStartingAt(unsigned i) { if (is8Bit()) return m_data8[i]; if (U16_IS_SINGLE(m_data16[i])) return m_data16[i]; if (i + 1 < m_length && U16_IS_LEAD(m_data16[i]) && U16_IS_TRAIL(m_data16[i + 1])) return U16_GET_SUPPLEMENTARY(m_data16[i], m_data16[i + 1]); return 0; } Ref StringImpl::convertToLowercaseWithoutLocale() { // Note: At one time this was a hot function in the Dromaeo benchmark, specifically the // no-op code path that may return ourself if we find no upper case letters and no invalid // ASCII letters. // First scan the string for uppercase and non-ASCII characters: if (is8Bit()) { for (unsigned i = 0; i < m_length; ++i) { LChar character = m_data8[i]; if (UNLIKELY(!isASCII(character) || isASCIIUpper(character))) return convertToLowercaseWithoutLocaleStartingAtFailingIndex8Bit(i); } return *this; } bool noUpper = true; unsigned ored = 0; for (unsigned i = 0; i < m_length; ++i) { UChar character = m_data16[i]; if (UNLIKELY(isASCIIUpper(character))) noUpper = false; ored |= character; } // Nothing to do if the string is all ASCII with no uppercase. if (noUpper && !(ored & ~0x7F)) return *this; if (!(ored & ~0x7F)) { UChar* data16; auto newImpl = createUninitializedInternalNonEmpty(m_length, data16); for (unsigned i = 0; i < m_length; ++i) data16[i] = toASCIILower(m_data16[i]); return newImpl; } if (m_length > MaxLength) CRASH(); int32_t length = m_length; // Do a slower implementation for cases that include non-ASCII characters. UChar* data16; auto newImpl = createUninitializedInternalNonEmpty(m_length, data16); UErrorCode status = U_ZERO_ERROR; int32_t realLength = u_strToLower(data16, length, m_data16, m_length, "", &status); if (U_SUCCESS(status) && realLength == length) return newImpl; newImpl = createUninitialized(realLength, data16); status = U_ZERO_ERROR; u_strToLower(data16, realLength, m_data16, m_length, "", &status); if (U_FAILURE(status)) return *this; return newImpl; } Ref StringImpl::convertToLowercaseWithoutLocaleStartingAtFailingIndex8Bit(unsigned failingIndex) { ASSERT(is8Bit()); LChar* data8; auto newImpl = createUninitializedInternalNonEmpty(m_length, data8); for (unsigned i = 0; i < failingIndex; ++i) { ASSERT(isASCII(m_data8[i])); ASSERT(!isASCIIUpper(m_data8[i])); data8[i] = m_data8[i]; } for (unsigned i = failingIndex; i < m_length; ++i) { LChar character = m_data8[i]; if (isASCII(character)) data8[i] = toASCIILower(character); else { ASSERT(isLatin1(u_tolower(character))); data8[i] = static_cast(u_tolower(character)); } } return newImpl; } Ref StringImpl::convertToUppercaseWithoutLocale() { // This function could be optimized for no-op cases the way // convertToLowercaseWithoutLocale() is, but in empirical testing, // few actual calls to upper() are no-ops, so it wouldn't be worth // the extra time for pre-scanning. if (m_length > MaxLength) CRASH(); int32_t length = m_length; if (is8Bit()) { LChar* data8; auto newImpl = createUninitialized(m_length, data8); // Do a faster loop for the case where all the characters are ASCII. unsigned ored = 0; for (int i = 0; i < length; ++i) { LChar character = m_data8[i]; ored |= character; data8[i] = toASCIIUpper(character); } if (!(ored & ~0x7F)) return newImpl; // Do a slower implementation for cases that include non-ASCII Latin-1 characters. int numberSharpSCharacters = 0; // There are two special cases. // 1. Some Latin-1 characters when converted to upper case are 16 bit characters. // 2. Lower case sharp-S converts to "SS" (two characters) for (int32_t i = 0; i < length; ++i) { LChar character = m_data8[i]; if (UNLIKELY(character == smallLetterSharpS)) ++numberSharpSCharacters; ASSERT(u_toupper(character) <= 0xFFFF); UChar upper = u_toupper(character); if (UNLIKELY(!isLatin1(upper))) { // Since this upper-cased character does not fit in an 8-bit string, we need to take the 16-bit path. goto upconvert; } data8[i] = static_cast(upper); } if (!numberSharpSCharacters) return newImpl; // We have numberSSCharacters sharp-s characters, but none of the other special characters. newImpl = createUninitialized(m_length + numberSharpSCharacters, data8); LChar* dest = data8; for (int32_t i = 0; i < length; ++i) { LChar character = m_data8[i]; if (character == smallLetterSharpS) { *dest++ = 'S'; *dest++ = 'S'; } else { ASSERT(isLatin1(u_toupper(character))); *dest++ = static_cast(u_toupper(character)); } } return newImpl; } upconvert: auto upconvertedCharacters = StringView(*this).upconvertedCharacters(); const UChar* source16 = upconvertedCharacters; UChar* data16; auto newImpl = createUninitialized(m_length, data16); // Do a faster loop for the case where all the characters are ASCII. unsigned ored = 0; for (int i = 0; i < length; ++i) { UChar character = source16[i]; ored |= character; data16[i] = toASCIIUpper(character); } if (!(ored & ~0x7F)) return newImpl; // Do a slower implementation for cases that include non-ASCII characters. UErrorCode status = U_ZERO_ERROR; int32_t realLength = u_strToUpper(data16, length, source16, m_length, "", &status); if (U_SUCCESS(status) && realLength == length) return newImpl; newImpl = createUninitialized(realLength, data16); status = U_ZERO_ERROR; u_strToUpper(data16, realLength, source16, m_length, "", &status); if (U_FAILURE(status)) return *this; return newImpl; } static inline bool needsTurkishCasingRules(const AtomString& localeIdentifier) { // Either "tr" or "az" locale, with case sensitive comparison and allowing for an ignored subtag. UChar first = localeIdentifier[0]; UChar second = localeIdentifier[1]; return ((isASCIIAlphaCaselessEqual(first, 't') && isASCIIAlphaCaselessEqual(second, 'r')) || (isASCIIAlphaCaselessEqual(first, 'a') && isASCIIAlphaCaselessEqual(second, 'z'))) && (localeIdentifier.length() == 2 || localeIdentifier[2] == '-'); } Ref StringImpl::convertToLowercaseWithLocale(const AtomString& localeIdentifier) { // Use the more-optimized code path most of the time. // Assuming here that the only locale-specific lowercasing is the Turkish casing rules. // FIXME: Could possibly optimize further by looking for the specific sequences // that have locale-specific lowercasing. There are only three of them. if (!needsTurkishCasingRules(localeIdentifier)) return convertToLowercaseWithoutLocale(); // FIXME: Could share more code with the main StringImpl::lower by factoring out // this last part into a shared function that takes a locale string, since this is // just like the end of that function. if (m_length > MaxLength) CRASH(); int length = m_length; // Below, we pass in the hardcoded locale "tr". Passing that is more efficient than // allocating memory just to turn localeIdentifier into a C string, and we assume // there is no difference between the uppercasing for "tr" and "az" locales. auto upconvertedCharacters = StringView(*this).upconvertedCharacters(); const UChar* source16 = upconvertedCharacters; UChar* data16; auto newString = createUninitialized(length, data16); UErrorCode status = U_ZERO_ERROR; int realLength = u_strToLower(data16, length, source16, length, "tr", &status); if (U_SUCCESS(status) && realLength == length) return newString; newString = createUninitialized(realLength, data16); status = U_ZERO_ERROR; u_strToLower(data16, realLength, source16, length, "tr", &status); if (U_FAILURE(status)) return *this; return newString; } Ref StringImpl::convertToUppercaseWithLocale(const AtomString& localeIdentifier) { // Use the more-optimized code path most of the time. // Assuming here that the only locale-specific lowercasing is the Turkish casing rules, // and that the only affected character is lowercase "i". if (!needsTurkishCasingRules(localeIdentifier) || find('i') == notFound) return convertToUppercaseWithoutLocale(); if (m_length > MaxLength) CRASH(); int length = m_length; // Below, we pass in the hardcoded locale "tr". Passing that is more efficient than // allocating memory just to turn localeIdentifier into a C string, and we assume // there is no difference between the uppercasing for "tr" and "az" locales. auto upconvertedCharacters = StringView(*this).upconvertedCharacters(); const UChar* source16 = upconvertedCharacters; UChar* data16; auto newString = createUninitialized(length, data16); UErrorCode status = U_ZERO_ERROR; int realLength = u_strToUpper(data16, length, source16, length, "tr", &status); if (U_SUCCESS(status) && realLength == length) return newString; newString = createUninitialized(realLength, data16); status = U_ZERO_ERROR; u_strToUpper(data16, realLength, source16, length, "tr", &status); if (U_FAILURE(status)) return *this; return newString; } Ref StringImpl::foldCase() { if (is8Bit()) { unsigned failingIndex; for (unsigned i = 0; i < m_length; ++i) { auto character = m_data8[i]; if (UNLIKELY(!isASCII(character) || isASCIIUpper(character))) { failingIndex = i; goto SlowPath; } } // String was all ASCII and no uppercase, so just return as-is. return *this; SlowPath: bool need16BitCharacters = false; for (unsigned i = failingIndex; i < m_length; ++i) { auto character = m_data8[i]; if (character == 0xB5 || character == 0xDF) { need16BitCharacters = true; break; } } if (!need16BitCharacters) { LChar* data8; auto folded = createUninitializedInternalNonEmpty(m_length, data8); copyCharacters(data8, m_data8, failingIndex); for (unsigned i = failingIndex; i < m_length; ++i) { auto character = m_data8[i]; if (isASCII(character)) data8[i] = toASCIILower(character); else { ASSERT(isLatin1(u_foldCase(character, U_FOLD_CASE_DEFAULT))); data8[i] = static_cast(u_foldCase(character, U_FOLD_CASE_DEFAULT)); } } return folded; } } else { // FIXME: Unclear why we use goto in the 8-bit case, and a different approach in the 16-bit case. bool noUpper = true; unsigned ored = 0; for (unsigned i = 0; i < m_length; ++i) { UChar character = m_data16[i]; if (UNLIKELY(isASCIIUpper(character))) noUpper = false; ored |= character; } if (!(ored & ~0x7F)) { if (noUpper) { // String was all ASCII and no uppercase, so just return as-is. return *this; } UChar* data16; auto folded = createUninitializedInternalNonEmpty(m_length, data16); for (unsigned i = 0; i < m_length; ++i) data16[i] = toASCIILower(m_data16[i]); return folded; } } if (m_length > MaxLength) CRASH(); auto upconvertedCharacters = StringView(*this).upconvertedCharacters(); UChar* data; auto folded = createUninitializedInternalNonEmpty(m_length, data); int32_t length = m_length; UErrorCode status = U_ZERO_ERROR; int32_t realLength = u_strFoldCase(data, length, upconvertedCharacters, length, U_FOLD_CASE_DEFAULT, &status); if (U_SUCCESS(status) && realLength == length) return folded; ASSERT(realLength > length); folded = createUninitializedInternalNonEmpty(realLength, data); status = U_ZERO_ERROR; u_strFoldCase(data, realLength, upconvertedCharacters, length, U_FOLD_CASE_DEFAULT, &status); if (U_FAILURE(status)) return *this; return folded; } template ALWAYS_INLINE Ref StringImpl::convertASCIICase(StringImpl& impl, const CharacterType* data, unsigned length) { unsigned failingIndex; for (unsigned i = 0; i < length; ++i) { CharacterType character = data[i]; if (type == CaseConvertType::Lower ? UNLIKELY(isASCIIUpper(character)) : LIKELY(isASCIILower(character))) { failingIndex = i; goto SlowPath; } } return impl; SlowPath: CharacterType* newData; auto newImpl = createUninitializedInternalNonEmpty(length, newData); copyCharacters(newData, data, failingIndex); for (unsigned i = failingIndex; i < length; ++i) newData[i] = type == CaseConvertType::Lower ? toASCIILower(data[i]) : toASCIIUpper(data[i]); return newImpl; } Ref StringImpl::convertToASCIILowercase() { if (is8Bit()) return convertASCIICase(*this, m_data8, m_length); return convertASCIICase(*this, m_data16, m_length); } Ref StringImpl::convertToASCIIUppercase() { if (is8Bit()) return convertASCIICase(*this, m_data8, m_length); return convertASCIICase(*this, m_data16, m_length); } template inline Ref StringImpl::stripMatchedCharacters(CodeUnitPredicate predicate) { if (!m_length) return *this; unsigned start = 0; unsigned end = m_length - 1; // skip white space from start while (start <= end && predicate(is8Bit() ? m_data8[start] : m_data16[start])) ++start; // only white space if (start > end) return *empty(); // skip white space from end while (end && predicate(is8Bit() ? m_data8[end] : m_data16[end])) --end; if (!start && end == m_length - 1) return *this; if (is8Bit()) return create(m_data8 + start, end + 1 - start); return create(m_data16 + start, end + 1 - start); } Ref StringImpl::stripWhiteSpace() { return stripMatchedCharacters(isSpaceOrNewline); } Ref StringImpl::stripLeadingAndTrailingCharacters(CodeUnitMatchFunction predicate) { return stripMatchedCharacters(predicate); } template inline Ref StringImpl::simplifyMatchedCharactersToSpace(UCharPredicate predicate) { StringBuffer data(m_length); auto* from = characters(); auto* fromEnd = from + m_length; unsigned outc = 0; bool changedToSpace = false; auto* to = data.characters(); while (true) { while (from != fromEnd && predicate(*from)) { if (*from != ' ') changedToSpace = true; ++from; } while (from != fromEnd && !predicate(*from)) to[outc++] = *from++; if (from != fromEnd) to[outc++] = ' '; else break; } if (outc && to[outc - 1] == ' ') --outc; if (outc == m_length && !changedToSpace) return *this; data.shrink(outc); return adopt(WTFMove(data)); } Ref StringImpl::simplifyWhiteSpace() { if (is8Bit()) return StringImpl::simplifyMatchedCharactersToSpace(isSpaceOrNewline); return StringImpl::simplifyMatchedCharactersToSpace(isSpaceOrNewline); } Ref StringImpl::simplifyWhiteSpace(CodeUnitMatchFunction isWhiteSpace) { if (is8Bit()) return StringImpl::simplifyMatchedCharactersToSpace(isWhiteSpace); return StringImpl::simplifyMatchedCharactersToSpace(isWhiteSpace); } int StringImpl::toIntStrict(bool* ok, int base) { if (is8Bit()) return charactersToIntStrict(characters8(), m_length, ok, base); return charactersToIntStrict(characters16(), m_length, ok, base); } unsigned StringImpl::toUIntStrict(bool* ok, int base) { if (is8Bit()) return charactersToUIntStrict(characters8(), m_length, ok, base); return charactersToUIntStrict(characters16(), m_length, ok, base); } int64_t StringImpl::toInt64Strict(bool* ok, int base) { if (is8Bit()) return charactersToInt64Strict(characters8(), m_length, ok, base); return charactersToInt64Strict(characters16(), m_length, ok, base); } uint64_t StringImpl::toUInt64Strict(bool* ok, int base) { if (is8Bit()) return charactersToUInt64Strict(characters8(), m_length, ok, base); return charactersToUInt64Strict(characters16(), m_length, ok, base); } intptr_t StringImpl::toIntPtrStrict(bool* ok, int base) { if (is8Bit()) return charactersToIntPtrStrict(characters8(), m_length, ok, base); return charactersToIntPtrStrict(characters16(), m_length, ok, base); } int StringImpl::toInt(bool* ok) { if (is8Bit()) return charactersToInt(characters8(), m_length, ok); return charactersToInt(characters16(), m_length, ok); } unsigned StringImpl::toUInt(bool* ok) { if (is8Bit()) return charactersToUInt(characters8(), m_length, ok); return charactersToUInt(characters16(), m_length, ok); } int64_t StringImpl::toInt64(bool* ok) { if (is8Bit()) return charactersToInt64(characters8(), m_length, ok); return charactersToInt64(characters16(), m_length, ok); } uint64_t StringImpl::toUInt64(bool* ok) { if (is8Bit()) return charactersToUInt64(characters8(), m_length, ok); return charactersToUInt64(characters16(), m_length, ok); } intptr_t StringImpl::toIntPtr(bool* ok) { if (is8Bit()) return charactersToIntPtr(characters8(), m_length, ok); return charactersToIntPtr(characters16(), m_length, ok); } double StringImpl::toDouble(bool* ok) { if (is8Bit()) return charactersToDouble(characters8(), m_length, ok); return charactersToDouble(characters16(), m_length, ok); } float StringImpl::toFloat(bool* ok) { if (is8Bit()) return charactersToFloat(characters8(), m_length, ok); return charactersToFloat(characters16(), m_length, ok); } size_t StringImpl::find(CodeUnitMatchFunction matchFunction, unsigned start) { if (is8Bit()) return WTF::find(characters8(), m_length, matchFunction, start); return WTF::find(characters16(), m_length, matchFunction, start); } size_t StringImpl::find(const LChar* matchString, unsigned index) { // Check for null or empty string to match against if (!matchString) return notFound; size_t matchStringLength = strlen(reinterpret_cast(matchString)); if (matchStringLength > MaxLength) CRASH(); unsigned matchLength = matchStringLength; if (!matchLength) return std::min(index, length()); // Optimization 1: fast case for strings of length 1. if (matchLength == 1) { if (is8Bit()) return WTF::find(characters8(), length(), matchString[0], index); return WTF::find(characters16(), length(), *matchString, index); } // Check index & matchLength are in range. if (index > length()) return notFound; unsigned searchLength = length() - index; if (matchLength > searchLength) return notFound; // delta is the number of additional times to test; delta == 0 means test only once. unsigned delta = searchLength - matchLength; // Optimization 2: keep a running hash of the strings, // only call equal if the hashes match. if (is8Bit()) { const LChar* searchCharacters = characters8() + index; unsigned searchHash = 0; unsigned matchHash = 0; for (unsigned i = 0; i < matchLength; ++i) { searchHash += searchCharacters[i]; matchHash += matchString[i]; } unsigned i = 0; while (searchHash != matchHash || !equal(searchCharacters + i, matchString, matchLength)) { if (i == delta) return notFound; searchHash += searchCharacters[i + matchLength]; searchHash -= searchCharacters[i]; ++i; } return index + i; } const UChar* searchCharacters = characters16() + index; unsigned searchHash = 0; unsigned matchHash = 0; for (unsigned i = 0; i < matchLength; ++i) { searchHash += searchCharacters[i]; matchHash += matchString[i]; } unsigned i = 0; while (searchHash != matchHash || !equal(searchCharacters + i, matchString, matchLength)) { if (i == delta) return notFound; searchHash += searchCharacters[i + matchLength]; searchHash -= searchCharacters[i]; ++i; } return index + i; } size_t StringImpl::find(StringImpl* matchString) { // Check for null string to match against if (UNLIKELY(!matchString)) return notFound; unsigned matchLength = matchString->length(); // Optimization 1: fast case for strings of length 1. if (matchLength == 1) { if (is8Bit()) { if (matchString->is8Bit()) return WTF::find(characters8(), length(), matchString->characters8()[0]); return WTF::find(characters8(), length(), matchString->characters16()[0]); } if (matchString->is8Bit()) return WTF::find(characters16(), length(), matchString->characters8()[0]); return WTF::find(characters16(), length(), matchString->characters16()[0]); } // Check matchLength is in range. if (matchLength > length()) return notFound; // Check for empty string to match against if (UNLIKELY(!matchLength)) return 0; if (is8Bit()) { if (matchString->is8Bit()) return findInner(characters8(), matchString->characters8(), 0, length(), matchLength); return findInner(characters8(), matchString->characters16(), 0, length(), matchLength); } if (matchString->is8Bit()) return findInner(characters16(), matchString->characters8(), 0, length(), matchLength); return findInner(characters16(), matchString->characters16(), 0, length(), matchLength); } size_t StringImpl::find(StringImpl* matchString, unsigned index) { // Check for null or empty string to match against if (UNLIKELY(!matchString)) return notFound; return findCommon(*this, *matchString, index); } size_t StringImpl::findIgnoringASCIICase(const StringImpl& matchString) const { return ::WTF::findIgnoringASCIICase(*this, matchString, 0); } size_t StringImpl::findIgnoringASCIICase(const StringImpl& matchString, unsigned startOffset) const { return ::WTF::findIgnoringASCIICase(*this, matchString, startOffset); } size_t StringImpl::findIgnoringASCIICase(const StringImpl* matchString) const { if (!matchString) return notFound; return ::WTF::findIgnoringASCIICase(*this, *matchString, 0); } size_t StringImpl::findIgnoringASCIICase(const StringImpl* matchString, unsigned startOffset) const { if (!matchString) return notFound; return ::WTF::findIgnoringASCIICase(*this, *matchString, startOffset); } size_t StringImpl::reverseFind(UChar character, unsigned index) { if (is8Bit()) return WTF::reverseFind(characters8(), m_length, character, index); return WTF::reverseFind(characters16(), m_length, character, index); } template ALWAYS_INLINE static size_t reverseFindInner(const SearchCharacterType* searchCharacters, const MatchCharacterType* matchCharacters, unsigned index, unsigned length, unsigned matchLength) { // Optimization: keep a running hash of the strings, // only call equal if the hashes match. // delta is the number of additional times to test; delta == 0 means test only once. unsigned delta = std::min(index, length - matchLength); unsigned searchHash = 0; unsigned matchHash = 0; for (unsigned i = 0; i < matchLength; ++i) { searchHash += searchCharacters[delta + i]; matchHash += matchCharacters[i]; } // keep looping until we match while (searchHash != matchHash || !equal(searchCharacters + delta, matchCharacters, matchLength)) { if (!delta) return notFound; --delta; searchHash -= searchCharacters[delta + matchLength]; searchHash += searchCharacters[delta]; } return delta; } size_t StringImpl::reverseFind(StringImpl* matchString, unsigned index) { // Check for null or empty string to match against if (!matchString) return notFound; unsigned matchLength = matchString->length(); unsigned ourLength = length(); if (!matchLength) return std::min(index, ourLength); // Optimization 1: fast case for strings of length 1. if (matchLength == 1) { if (is8Bit()) return WTF::reverseFind(characters8(), ourLength, (*matchString)[0], index); return WTF::reverseFind(characters16(), ourLength, (*matchString)[0], index); } // Check index & matchLength are in range. if (matchLength > ourLength) return notFound; if (is8Bit()) { if (matchString->is8Bit()) return reverseFindInner(characters8(), matchString->characters8(), index, ourLength, matchLength); return reverseFindInner(characters8(), matchString->characters16(), index, ourLength, matchLength); } if (matchString->is8Bit()) return reverseFindInner(characters16(), matchString->characters8(), index, ourLength, matchLength); return reverseFindInner(characters16(), matchString->characters16(), index, ourLength, matchLength); } ALWAYS_INLINE static bool equalInner(const StringImpl& string, unsigned startOffset, const char* matchString, unsigned matchLength) { ASSERT(matchLength <= string.length()); ASSERT(startOffset + matchLength <= string.length()); if (string.is8Bit()) return equal(string.characters8() + startOffset, reinterpret_cast(matchString), matchLength); return equal(string.characters16() + startOffset, reinterpret_cast(matchString), matchLength); } ALWAYS_INLINE static bool equalInner(const StringImpl& string, unsigned startOffset, const StringImpl& matchString) { if (startOffset > string.length()) return false; if (matchString.length() > string.length()) return false; if (matchString.length() + startOffset > string.length()) return false; if (string.is8Bit()) { if (matchString.is8Bit()) return equal(string.characters8() + startOffset, matchString.characters8(), matchString.length()); return equal(string.characters8() + startOffset, matchString.characters16(), matchString.length()); } if (matchString.is8Bit()) return equal(string.characters16() + startOffset, matchString.characters8(), matchString.length()); return equal(string.characters16() + startOffset, matchString.characters16(), matchString.length()); } bool StringImpl::startsWith(const StringImpl* string) const { return !string || ::WTF::startsWith(*this, *string); } bool StringImpl::startsWith(const StringImpl& string) const { return ::WTF::startsWith(*this, string); } bool StringImpl::startsWithIgnoringASCIICase(const StringImpl* prefix) const { return prefix && ::WTF::startsWithIgnoringASCIICase(*this, *prefix); } bool StringImpl::startsWithIgnoringASCIICase(const StringImpl& prefix) const { return ::WTF::startsWithIgnoringASCIICase(*this, prefix); } bool StringImpl::startsWith(UChar character) const { return m_length && (*this)[0] == character; } bool StringImpl::startsWith(const char* matchString, unsigned matchLength) const { return matchLength <= length() && equalInner(*this, 0, matchString, matchLength); } bool StringImpl::hasInfixStartingAt(const StringImpl& matchString, unsigned startOffset) const { return equalInner(*this, startOffset, matchString); } bool StringImpl::endsWith(StringImpl* suffix) { return suffix && ::WTF::endsWith(*this, *suffix); } bool StringImpl::endsWith(StringImpl& suffix) { return ::WTF::endsWith(*this, suffix); } bool StringImpl::endsWithIgnoringASCIICase(const StringImpl* suffix) const { return suffix && ::WTF::endsWithIgnoringASCIICase(*this, *suffix); } bool StringImpl::endsWithIgnoringASCIICase(const StringImpl& suffix) const { return ::WTF::endsWithIgnoringASCIICase(*this, suffix); } bool StringImpl::endsWith(UChar character) const { return m_length && (*this)[m_length - 1] == character; } bool StringImpl::endsWith(const char* matchString, unsigned matchLength) const { return matchLength <= length() && equalInner(*this, length() - matchLength, matchString, matchLength); } bool StringImpl::hasInfixEndingAt(const StringImpl& matchString, unsigned endOffset) const { return endOffset >= matchString.length() && equalInner(*this, endOffset - matchString.length(), matchString); } Ref StringImpl::replace(UChar target, UChar replacement) { if (target == replacement) return *this; unsigned i; for (i = 0; i != m_length; ++i) { UChar character = is8Bit() ? m_data8[i] : m_data16[i]; if (character == target) break; } if (i == m_length) return *this; if (is8Bit()) { if (!isLatin1(target)) { // Looking for a 16-bit character in an 8-bit string, so we're done. return *this; } if (isLatin1(replacement)) { LChar* data; LChar oldChar = static_cast(target); LChar newChar = static_cast(replacement); auto newImpl = createUninitializedInternalNonEmpty(m_length, data); for (i = 0; i != m_length; ++i) { LChar character = m_data8[i]; if (character == oldChar) character = newChar; data[i] = character; } return newImpl; } UChar* data; auto newImpl = createUninitializedInternalNonEmpty(m_length, data); for (i = 0; i != m_length; ++i) { UChar character = m_data8[i]; if (character == target) character = replacement; data[i] = character; } return newImpl; } UChar* data; auto newImpl = createUninitializedInternalNonEmpty(m_length, data); copyCharacters(data, m_data16, i); for (unsigned j = i; j != m_length; ++j) { UChar character = m_data16[j]; if (character == target) character = replacement; data[j] = character; } return newImpl; } Ref StringImpl::replace(unsigned position, unsigned lengthToReplace, StringImpl* string) { position = std::min(position, length()); lengthToReplace = std::min(lengthToReplace, length() - position); unsigned lengthToInsert = string ? string->length() : 0; if (!lengthToReplace && !lengthToInsert) return *this; if ((length() - lengthToReplace) >= (MaxLength - lengthToInsert)) CRASH(); if (is8Bit() && (!string || string->is8Bit())) { LChar* data; auto newImpl = createUninitialized(length() - lengthToReplace + lengthToInsert, data); copyCharacters(data, m_data8, position); if (string) copyCharacters(data + position, string->m_data8, lengthToInsert); copyCharacters(data + position + lengthToInsert, m_data8 + position + lengthToReplace, length() - position - lengthToReplace); return newImpl; } UChar* data; auto newImpl = createUninitialized(length() - lengthToReplace + lengthToInsert, data); if (is8Bit()) copyCharacters(data, m_data8, position); else copyCharacters(data, m_data16, position); if (string) { if (string->is8Bit()) copyCharacters(data + position, string->m_data8, lengthToInsert); else copyCharacters(data + position, string->m_data16, lengthToInsert); } if (is8Bit()) copyCharacters(data + position + lengthToInsert, m_data8 + position + lengthToReplace, length() - position - lengthToReplace); else copyCharacters(data + position + lengthToInsert, m_data16 + position + lengthToReplace, length() - position - lengthToReplace); return newImpl; } Ref StringImpl::replace(UChar pattern, StringImpl* replacement) { if (!replacement) return *this; if (replacement->is8Bit()) return replace(pattern, replacement->m_data8, replacement->length()); return replace(pattern, replacement->m_data16, replacement->length()); } Ref StringImpl::replace(UChar pattern, const LChar* replacement, unsigned repStrLength) { ASSERT(replacement); size_t srcSegmentStart = 0; unsigned matchCount = 0; // Count the matches. while ((srcSegmentStart = find(pattern, srcSegmentStart)) != notFound) { ++matchCount; ++srcSegmentStart; } // If we have 0 matches then we don't have to do any more work. if (!matchCount) return *this; if (repStrLength && matchCount > MaxLength / repStrLength) CRASH(); unsigned replaceSize = matchCount * repStrLength; unsigned newSize = m_length - matchCount; if (newSize >= (MaxLength - replaceSize)) CRASH(); newSize += replaceSize; // Construct the new data. size_t srcSegmentEnd; unsigned srcSegmentLength; srcSegmentStart = 0; unsigned dstOffset = 0; if (is8Bit()) { LChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); dstOffset += srcSegmentLength; copyCharacters(data + dstOffset, replacement, repStrLength); dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + 1; } srcSegmentLength = m_length - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } UChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); dstOffset += srcSegmentLength; copyCharacters(data + dstOffset, replacement, repStrLength); dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + 1; } srcSegmentLength = m_length - srcSegmentStart; copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } Ref StringImpl::replace(UChar pattern, const UChar* replacement, unsigned repStrLength) { ASSERT(replacement); size_t srcSegmentStart = 0; unsigned matchCount = 0; // Count the matches. while ((srcSegmentStart = find(pattern, srcSegmentStart)) != notFound) { ++matchCount; ++srcSegmentStart; } // If we have 0 matches then we don't have to do any more work. if (!matchCount) return *this; if (repStrLength && matchCount > MaxLength / repStrLength) CRASH(); unsigned replaceSize = matchCount * repStrLength; unsigned newSize = m_length - matchCount; if (newSize >= (MaxLength - replaceSize)) CRASH(); newSize += replaceSize; // Construct the new data. size_t srcSegmentEnd; unsigned srcSegmentLength; srcSegmentStart = 0; unsigned dstOffset = 0; if (is8Bit()) { UChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); dstOffset += srcSegmentLength; copyCharacters(data + dstOffset, replacement, repStrLength); dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + 1; } srcSegmentLength = m_length - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } UChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); dstOffset += srcSegmentLength; copyCharacters(data + dstOffset, replacement, repStrLength); dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + 1; } srcSegmentLength = m_length - srcSegmentStart; copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } Ref StringImpl::replace(StringImpl* pattern, StringImpl* replacement) { if (!pattern || !replacement) return *this; unsigned patternLength = pattern->length(); if (!patternLength) return *this; unsigned repStrLength = replacement->length(); size_t srcSegmentStart = 0; unsigned matchCount = 0; // Count the matches. while ((srcSegmentStart = find(pattern, srcSegmentStart)) != notFound) { ++matchCount; srcSegmentStart += patternLength; } // If we have 0 matches, we don't have to do any more work if (!matchCount) return *this; unsigned newSize = m_length - matchCount * patternLength; if (repStrLength && matchCount > MaxLength / repStrLength) CRASH(); if (newSize > (MaxLength - matchCount * repStrLength)) CRASH(); newSize += matchCount * repStrLength; // Construct the new data size_t srcSegmentEnd; unsigned srcSegmentLength; srcSegmentStart = 0; unsigned dstOffset = 0; bool srcIs8Bit = is8Bit(); bool replacementIs8Bit = replacement->is8Bit(); // There are 4 cases: // 1. This and replacement are both 8 bit. // 2. This and replacement are both 16 bit. // 3. This is 8 bit and replacement is 16 bit. // 4. This is 16 bit and replacement is 8 bit. if (srcIs8Bit && replacementIs8Bit) { // Case 1 LChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); dstOffset += srcSegmentLength; copyCharacters(data + dstOffset, replacement->m_data8, repStrLength); dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + patternLength; } srcSegmentLength = m_length - srcSegmentStart; copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } UChar* data; auto newImpl = createUninitialized(newSize, data); while ((srcSegmentEnd = find(pattern, srcSegmentStart)) != notFound) { srcSegmentLength = srcSegmentEnd - srcSegmentStart; if (srcIs8Bit) { // Case 3. copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); } else { // Case 2 & 4. copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); } dstOffset += srcSegmentLength; if (replacementIs8Bit) { // Cases 2 & 3. copyCharacters(data + dstOffset, replacement->m_data8, repStrLength); } else { // Case 4 copyCharacters(data + dstOffset, replacement->m_data16, repStrLength); } dstOffset += repStrLength; srcSegmentStart = srcSegmentEnd + patternLength; } srcSegmentLength = m_length - srcSegmentStart; if (srcIs8Bit) { // Case 3. copyCharacters(data + dstOffset, m_data8 + srcSegmentStart, srcSegmentLength); } else { // Cases 2 & 4. copyCharacters(data + dstOffset, m_data16 + srcSegmentStart, srcSegmentLength); } ASSERT(dstOffset + srcSegmentLength == newImpl.get().length()); return newImpl; } bool equal(const StringImpl* a, const StringImpl* b) { return equalCommon(a, b); } template inline bool equalInternal(const StringImpl* a, const CharacterType* b, unsigned length) { if (!a) return !b; if (!b) return false; if (a->length() != length) return false; if (a->is8Bit()) return equal(a->characters8(), b, length); return equal(a->characters16(), b, length); } bool equal(const StringImpl* a, const LChar* b, unsigned length) { return equalInternal(a, b, length); } bool equal(const StringImpl* a, const UChar* b, unsigned length) { return equalInternal(a, b, length); } bool equal(const StringImpl* a, const LChar* b) { if (!a) return !b; if (!b) return !a; unsigned length = a->length(); if (a->is8Bit()) { const LChar* aPtr = a->characters8(); for (unsigned i = 0; i != length; ++i) { LChar bc = b[i]; LChar ac = aPtr[i]; if (!bc) return false; if (ac != bc) return false; } return !b[length]; } const UChar* aPtr = a->characters16(); for (unsigned i = 0; i != length; ++i) { LChar bc = b[i]; if (!bc) return false; if (aPtr[i] != bc) return false; } return !b[length]; } bool equal(const StringImpl& a, const StringImpl& b) { return equalCommon(a, b); } bool equalIgnoringNullity(StringImpl* a, StringImpl* b) { if (!a && b && !b->length()) return true; if (!b && a && !a->length()) return true; return equal(a, b); } bool equalIgnoringASCIICase(const StringImpl* a, const StringImpl* b) { return a == b || (a && b && equalIgnoringASCIICase(*a, *b)); } bool equalIgnoringASCIICaseNonNull(const StringImpl* a, const StringImpl* b) { ASSERT(a); ASSERT(b); return equalIgnoringASCIICase(*a, *b); } UCharDirection StringImpl::defaultWritingDirection(bool* hasStrongDirectionality) { for (unsigned i = 0; i < m_length; ++i) { auto charDirection = u_charDirection(is8Bit() ? m_data8[i] : m_data16[i]); if (charDirection == U_LEFT_TO_RIGHT) { if (hasStrongDirectionality) *hasStrongDirectionality = true; return U_LEFT_TO_RIGHT; } if (charDirection == U_RIGHT_TO_LEFT || charDirection == U_RIGHT_TO_LEFT_ARABIC) { if (hasStrongDirectionality) *hasStrongDirectionality = true; return U_RIGHT_TO_LEFT; } } if (hasStrongDirectionality) *hasStrongDirectionality = false; return U_LEFT_TO_RIGHT; } Ref StringImpl::adopt(StringBuffer&& buffer) { unsigned length = buffer.length(); if (!length) return *empty(); return adoptRef(*new StringImpl(buffer.release(), length)); } Ref StringImpl::adopt(StringBuffer&& buffer) { unsigned length = buffer.length(); if (!length) return *empty(); return adoptRef(*new StringImpl(buffer.release(), length)); } size_t StringImpl::sizeInBytes() const { // FIXME: support substrings size_t size = length(); if (!is8Bit()) size *= 2; return size + sizeof(*this); } // Helper to write a three-byte UTF-8 code point into the buffer; caller must ensure room is available. static inline void putUTF8Triple(char*& buffer, UChar character) { ASSERT(character >= 0x0800); *buffer++ = static_cast(((character >> 12) & 0x0F) | 0xE0); *buffer++ = static_cast(((character >> 6) & 0x3F) | 0x80); *buffer++ = static_cast((character & 0x3F) | 0x80); } UTF8ConversionError StringImpl::utf8Impl(const UChar* characters, unsigned length, char*& buffer, size_t bufferSize, ConversionMode mode) { if (mode == StrictConversionReplacingUnpairedSurrogatesWithFFFD) { const UChar* charactersEnd = characters + length; char* bufferEnd = buffer + bufferSize; while (characters < charactersEnd) { // Use strict conversion to detect unpaired surrogates. auto result = convertUTF16ToUTF8(&characters, charactersEnd, &buffer, bufferEnd); ASSERT(result != TargetExhausted); // Conversion fails when there is an unpaired surrogate. // Put replacement character (U+FFFD) instead of the unpaired surrogate. if (result != ConversionOK) { ASSERT((0xD800 <= *characters && *characters <= 0xDFFF)); // There should be room left, since one UChar hasn't been converted. ASSERT((buffer + 3) <= bufferEnd); putUTF8Triple(buffer, replacementCharacter); ++characters; } } } else { bool strict = mode == StrictConversion; const UChar* originalCharacters = characters; auto result = convertUTF16ToUTF8(&characters, characters + length, &buffer, buffer + bufferSize, strict); ASSERT(result != TargetExhausted); // (length * 3) should be sufficient for any conversion // Only produced from strict conversion. if (result == SourceIllegal) { ASSERT(strict); return UTF8ConversionError::IllegalSource; } // Check for an unconverted high surrogate. if (result == SourceExhausted) { if (strict) return UTF8ConversionError::SourceExhausted; // This should be one unpaired high surrogate. Treat it the same // was as an unpaired high surrogate would have been handled in // the middle of a string with non-strict conversion - which is // to say, simply encode it to UTF-8. ASSERT_UNUSED( originalCharacters, (characters + 1) == (originalCharacters + length)); ASSERT((*characters >= 0xD800) && (*characters <= 0xDBFF)); // There should be room left, since one UChar hasn't been converted. ASSERT((buffer + 3) <= (buffer + bufferSize)); putUTF8Triple(buffer, *characters); } } return UTF8ConversionError::None; } Expected StringImpl::utf8ForCharacters(const LChar* characters, unsigned length) { if (!length) return CString("", 0); if (length > MaxLength / 3) return makeUnexpected(UTF8ConversionError::OutOfMemory); Vector bufferVector(length * 3); char* buffer = bufferVector.data(); const LChar* source = characters; bool success = convertLatin1ToUTF8(&source, source + length, &buffer, buffer + bufferVector.size()); ASSERT_UNUSED(success, success); // (length * 3) should be sufficient for any conversion return CString(bufferVector.data(), buffer - bufferVector.data()); } Expected StringImpl::utf8ForCharacters(const UChar* characters, unsigned length, ConversionMode mode) { if (!length) return CString("", 0); if (length > MaxLength / 3) return makeUnexpected(UTF8ConversionError::OutOfMemory); Vector bufferVector(length * 3); char* buffer = bufferVector.data(); UTF8ConversionError error = utf8Impl(characters, length, buffer, bufferVector.size(), mode); if (error != UTF8ConversionError::None) return makeUnexpected(error); return CString(bufferVector.data(), buffer - bufferVector.data()); } Expected StringImpl::tryGetUtf8ForRange(unsigned offset, unsigned length, ConversionMode mode) const { ASSERT(offset <= this->length()); ASSERT(offset + length <= this->length()); if (!length) return CString("", 0); // Allocate a buffer big enough to hold all the characters // (an individual UTF-16 UChar can only expand to 3 UTF-8 bytes). // Optimization ideas, if we find this function is hot: // * We could speculatively create a CStringBuffer to contain 'length' // characters, and resize if necessary (i.e. if the buffer contains // non-ascii characters). (Alternatively, scan the buffer first for // ascii characters, so we know this will be sufficient). // * We could allocate a CStringBuffer with an appropriate size to // have a good chance of being able to write the string into the // buffer without reallocing (say, 1.5 x length). if (length > MaxLength / 3) return makeUnexpected(UTF8ConversionError::OutOfMemory); Vector bufferVector(length * 3); char* buffer = bufferVector.data(); if (is8Bit()) { const LChar* characters = this->characters8() + offset; auto success = convertLatin1ToUTF8(&characters, characters + length, &buffer, buffer + bufferVector.size()); ASSERT_UNUSED(success, success); // (length * 3) should be sufficient for any conversion } else { UTF8ConversionError error = utf8Impl(this->characters16() + offset, length, buffer, bufferVector.size(), mode); if (error != UTF8ConversionError::None) return makeUnexpected(error); } return CString(bufferVector.data(), buffer - bufferVector.data()); } Expected StringImpl::tryGetUtf8(ConversionMode mode) const { return tryGetUtf8ForRange(0, length(), mode); } CString StringImpl::utf8(ConversionMode mode) const { auto expectedString = tryGetUtf8ForRange(0, length(), mode); RELEASE_ASSERT(expectedString); return expectedString.value(); } NEVER_INLINE unsigned StringImpl::hashSlowCase() const { if (is8Bit()) setHash(StringHasher::computeHashAndMaskTop8Bits(m_data8, m_length)); else setHash(StringHasher::computeHashAndMaskTop8Bits(m_data16, m_length)); return existingHash(); } unsigned StringImpl::concurrentHash() const { unsigned hash; if (is8Bit()) hash = StringHasher::computeHashAndMaskTop8Bits(m_data8, m_length); else hash = StringHasher::computeHashAndMaskTop8Bits(m_data16, m_length); ASSERT(((hash << s_flagCount) >> s_flagCount) == hash); return hash; } bool equalIgnoringNullity(const UChar* a, size_t aLength, StringImpl* b) { if (!b) return !aLength; if (aLength != b->length()) return false; if (b->is8Bit()) { const LChar* bCharacters = b->characters8(); for (unsigned i = 0; i < aLength; ++i) { if (a[i] != bCharacters[i]) return false; } return true; } return !memcmp(a, b->characters16(), b->length() * sizeof(UChar)); } } // namespace WTF