X-Git-Url: https://git.saurik.com/apple/javascriptcore.git/blobdiff_plain/14957cd040308e3eeec43d26bae5d76da13fcd85..refs/heads/master:/runtime/JSStringBuilder.h diff --git a/runtime/JSStringBuilder.h b/runtime/JSStringBuilder.h index 49d4a63..34f43c4 100644 --- a/runtime/JSStringBuilder.h +++ b/runtime/JSStringBuilder.h @@ -28,105 +28,109 @@ #include "ExceptionHelpers.h" #include "JSString.h" -#include "UStringConcatenate.h" -#include "Vector.h" +#include namespace JSC { +// FIXME: Should move the last few callers over from this to WTF::StringBuilder. class JSStringBuilder { public: JSStringBuilder() : m_okay(true) + , m_is8Bit(true) { } - void append(const UChar u) + void append(LChar character) { - m_okay &= buffer.tryAppend(&u, 1); - } - - void append(const char* str) - { - append(str, strlen(str)); - } - - void append(const char* str, size_t len) - { - m_okay &= buffer.tryReserveCapacity(buffer.size() + len); - for (size_t i = 0; i < len; i++) { - UChar u = static_cast(str[i]); - m_okay &= buffer.tryAppend(&u, 1); + if (m_is8Bit) { + m_okay &= buffer8.tryAppend(&character, 1); + return; } + UChar upconvertedCharacter = character; + m_okay &= buffer16.tryAppend(&upconvertedCharacter, 1); } - void append(const UChar* str, size_t len) + void append(UChar character) { - m_okay &= buffer.tryAppend(str, len); + if (m_is8Bit) { + if (character < 0x100) { + LChar narrowedCharacter = character; + m_okay &= buffer8.tryAppend(&narrowedCharacter, 1); + return; + } + upConvert(); + } + m_okay &= buffer16.tryAppend(&character, 1); } - void append(const UString& str) + void append(const char* str) { - m_okay &= buffer.tryAppend(str.characters(), str.length()); + append(reinterpret_cast(str), strlen(str)); } JSValue build(ExecState* exec) { if (!m_okay) return throwOutOfMemoryError(exec); - buffer.shrinkToFit(); - if (!buffer.data()) + 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, UString::adopt(buffer)); + return jsString(exec, String::adopt(buffer16)); } -protected: - Vector buffer; - bool m_okay; -}; +private: + void append(const LChar* characters, size_t length) + { + if (m_is8Bit) { + m_okay &= buffer8.tryAppend(characters, length); + return; + } + // 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); + } + } -template -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2) -{ - PassRefPtr result = WTF::tryMakeString(string1, string2); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} + void upConvert() + { + ASSERT(m_is8Bit); + size_t len = buffer8.size(); -template -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3) -{ - PassRefPtr result = WTF::tryMakeString(string1, string2, string3); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} + for (size_t i = 0; i < len; i++) + buffer16.append(buffer8[i]); -template -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4) -{ - PassRefPtr result = WTF::tryMakeString(string1, string2, string3, string4); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); -} + buffer8.clear(); + m_is8Bit = false; + } + + Vector buffer8; + Vector buffer16; + bool m_okay; + bool m_is8Bit; +}; -template -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4, StringType5 string5) +template +inline JSValue jsMakeNontrivialString(ExecState* exec, StringType&& string) { - PassRefPtr result = WTF::tryMakeString(string1, string2, string3, string4, string5); - if (!result) - return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); + return jsNontrivialString(exec, std::forward(string)); } -template -inline JSValue jsMakeNontrivialString(ExecState* exec, StringType1 string1, StringType2 string2, StringType3 string3, StringType4 string4, StringType5 string5, StringType6 string6) +template +inline JSValue jsMakeNontrivialString(ExecState* exec, const StringType& string, const StringTypes&... strings) { - PassRefPtr result = WTF::tryMakeString(string1, string2, string3, string4, string5, string6); + RefPtr result = WTF::tryMakeString(string, strings...); if (!result) return throwOutOfMemoryError(exec); - return jsNontrivialString(exec, result); + return jsNontrivialString(exec, result.release()); } }