diff options
Diffstat (limited to 'Source/JavaScriptCore/runtime/JSStringBuilder.h')
-rw-r--r-- | Source/JavaScriptCore/runtime/JSStringBuilder.h | 158 |
1 files changed, 52 insertions, 106 deletions
diff --git a/Source/JavaScriptCore/runtime/JSStringBuilder.h b/Source/JavaScriptCore/runtime/JSStringBuilder.h index 7b1c1a505..88ad6ddef 100644 --- a/Source/JavaScriptCore/runtime/JSStringBuilder.h +++ b/Source/JavaScriptCore/runtime/JSStringBuilder.h @@ -1,5 +1,5 @@ /* - * Copyright (C) 2009 Apple Inc. All rights reserved. + * Copyright (C) 2009, 2016 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions @@ -23,8 +23,7 @@ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ -#ifndef JSStringBuilder_h -#define JSStringBuilder_h +#pragma once #include "ExceptionHelpers.h" #include "JSString.h" @@ -32,6 +31,7 @@ namespace JSC { +// FIXME: Should move the last few callers over from this to WTF::StringBuilder. class JSStringBuilder { public: JSStringBuilder() @@ -40,73 +40,66 @@ public: { } - void append(const UChar u) + void append(LChar character) { if (m_is8Bit) { - if (u < 0xff) { - LChar c = u; - m_okay &= buffer8.tryAppend(&c, 1); + m_okay &= buffer8.tryAppend(&character, 1); + return; + } + UChar upconvertedCharacter = character; + m_okay &= buffer16.tryAppend(&upconvertedCharacter, 1); + } + + void append(UChar character) + { + if (m_is8Bit) { + if (character < 0x100) { + LChar narrowedCharacter = character; + m_okay &= buffer8.tryAppend(&narrowedCharacter, 1); return; } upConvert(); } - m_okay &= buffer16.tryAppend(&u, 1); + m_okay &= buffer16.tryAppend(&character, 1); } void append(const char* str) { - append(str, strlen(str)); + append(reinterpret_cast<const LChar*>(str), strlen(str)); } - void append(const char* str, size_t len) + JSValue build(ExecState* exec) { + VM& vm = exec->vm(); + auto scope = DECLARE_THROW_SCOPE(vm); + if (!m_okay) + return throwOutOfMemoryError(exec, scope); if (m_is8Bit) { - m_okay &= buffer8.tryAppend(reinterpret_cast<const LChar*>(str), len); - return; - } - m_okay &= buffer8.tryReserveCapacity(buffer16.size() + len); - for (size_t i = 0; i < len; i++) { - UChar u = static_cast<unsigned char>(str[i]); - m_okay &= buffer16.tryAppend(&u, 1); + buffer8.shrinkToFit(); + if (!buffer8.data()) + return throwOutOfMemoryError(exec, scope); + return jsString(exec, String::adopt(WTFMove(buffer8))); } + buffer16.shrinkToFit(); + if (!buffer16.data()) + return throwOutOfMemoryError(exec, scope); + return jsString(exec, String::adopt(WTFMove(buffer16))); } - void append(const LChar* str, size_t len) +private: + void append(const LChar* characters, size_t length) { if (m_is8Bit) { - m_okay &= buffer8.tryAppend(str, len); + m_okay &= buffer8.tryAppend(characters, length); return; } - m_okay &= buffer8.tryReserveCapacity(buffer16.size() + len); - for (size_t i = 0; i < len; i++) { - UChar u = str[i]; - m_okay &= buffer16.tryAppend(&u, 1); + // FIXME: There must be a more efficient way of doing this. + m_okay &= buffer16.tryReserveCapacity(buffer16.size() + length); + for (size_t i = 0; i < length; i++) { + UChar upconvertedCharacter = characters[i]; + m_okay &= buffer16.tryAppend(&upconvertedCharacter, 1); } } - - void append(const UChar* str, size_t len) - { - if (m_is8Bit) - upConvert(); // FIXME: We could check character by character its size. - m_okay &= buffer16.tryAppend(str, len); - } - - void append(const String& str) - { - unsigned length = str.length(); - - if (!length) - return; - - if (m_is8Bit) { - if (str.is8Bit()) { - m_okay &= buffer8.tryAppend(str.characters8(), length); - return; - } - upConvert(); - } - m_okay &= buffer16.tryAppend(str.deprecatedCharacters(), length); - } void upConvert() { @@ -120,74 +113,27 @@ public: m_is8Bit = false; } - JSValue build(ExecState* exec) - { - if (!m_okay) - return throwOutOfMemoryError(exec); - if (m_is8Bit) { - buffer8.shrinkToFit(); - if (!buffer8.data()) - return throwOutOfMemoryError(exec); - return jsString(exec, String::adopt(buffer8)); - } - buffer16.shrinkToFit(); - if (!buffer16.data()) - return throwOutOfMemoryError(exec); - return jsString(exec, String::adopt(buffer16)); - } - -protected: Vector<LChar, 64, UnsafeVectorOverflow> buffer8; Vector<UChar, 64, UnsafeVectorOverflow> buffer16; bool m_okay; bool m_is8Bit; }; -template<typename StringType1, typename StringType2> -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2) -{ - PassRefPtr<StringImpl> result = WTF::tryMakeString(string1, string2); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} - -template<typename StringType1, typename StringType2, typename StringType3> -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3) -{ - PassRefPtr<StringImpl> result = WTF::tryMakeString(string1, string2, string3); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} - -template<typename StringType1, typename StringType2, typename StringType3, typename StringType4> -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4) -{ - PassRefPtr<StringImpl> result = WTF::tryMakeString(string1, string2, string3, string4); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} - -template<typename StringType1, typename StringType2, typename StringType3, typename StringType4, typename StringType5> -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4, StringType5 string5) +template<typename StringType> +inline JSValue jsMakeNontrivialString(ExecState* exec, StringType&& string) { - PassRefPtr<StringImpl> result = WTF::tryMakeString(string1, string2, string3, string4, string5); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); + return jsNontrivialString(exec, std::forward<StringType>(string)); } -template<typename StringType1, typename StringType2, typename StringType3, typename StringType4, typename StringType5, typename StringType6> -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4, StringType5 string5, StringType6 string6) +template<typename StringType, typename... StringTypes> +inline JSValue jsMakeNontrivialString(ExecState* exec, const StringType& string, const StringTypes&... strings) { - PassRefPtr<StringImpl> result = WTF::tryMakeString(string1, string2, string3, string4, string5, string6); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} - + VM& vm = exec->vm(); + auto scope = DECLARE_THROW_SCOPE(vm); + String result = tryMakeString(string, strings...); + if (UNLIKELY(!result || !JSString::isValidLength(result.length()))) + return throwOutOfMemoryError(exec, scope); + return jsNontrivialString(exec, WTFMove(result)); } -#endif +} // namespace JSC |