X-Git-Url: https://git.saurik.com/apple/javascriptcore.git/blobdiff_plain/14957cd040308e3eeec43d26bae5d76da13fcd85..6fe7ccc865dc7d7541b93c5bcaf6368d2c98a174:/runtime/JSStringBuilder.h?ds=sidebyside diff --git a/runtime/JSStringBuilder.h b/runtime/JSStringBuilder.h index 49d4a63..1a2b812 100644 --- a/runtime/JSStringBuilder.h +++ b/runtime/JSStringBuilder.h @@ -29,7 +29,7 @@ #include "ExceptionHelpers.h" #include "JSString.h" #include "UStringConcatenate.h" -#include "Vector.h" +#include namespace JSC { @@ -37,12 +37,21 @@ class JSStringBuilder { public: JSStringBuilder() : m_okay(true) + , m_is8Bit(true) { } void append(const UChar u) { - m_okay &= buffer.tryAppend(&u, 1); + if (m_is8Bit) { + if (u < 0xff) { + LChar c = u; + m_okay &= buffer8.tryAppend(&c, 1); + return; + } + upConvert(); + } + m_okay &= buffer16.tryAppend(&u, 1); } void append(const char* str) @@ -52,36 +61,87 @@ public: void append(const char* str, size_t len) { - m_okay &= buffer.tryReserveCapacity(buffer.size() + len); + if (m_is8Bit) { + m_okay &= buffer8.tryAppend(reinterpret_cast(str), len); + return; + } + m_okay &= buffer8.tryReserveCapacity(buffer16.size() + len); for (size_t i = 0; i < len; i++) { UChar u = static_cast(str[i]); - m_okay &= buffer.tryAppend(&u, 1); + m_okay &= buffer16.tryAppend(&u, 1); } } + void append(const LChar* str, size_t len) + { + if (m_is8Bit) { + m_okay &= buffer8.tryAppend(str, len); + 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); + } + } + void append(const UChar* str, size_t len) { - m_okay &= buffer.tryAppend(str, len); + if (m_is8Bit) + upConvert(); // FIXME: We could check character by character its size. + m_okay &= buffer16.tryAppend(str, len); } void append(const UString& str) { - m_okay &= buffer.tryAppend(str.characters(), str.length()); + 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.characters(), length); + } + + void upConvert() + { + ASSERT(m_is8Bit); + size_t len = buffer8.size(); + + for (size_t i = 0; i < len; i++) + buffer16.append(buffer8[i]); + + buffer8.clear(); + m_is8Bit = false; } 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, UString::adopt(buffer8)); + } + buffer16.shrinkToFit(); + if (!buffer16.data()) return throwOutOfMemoryError(exec); - return jsString(exec, UString::adopt(buffer)); + return jsString(exec, UString::adopt(buffer16)); } protected: - Vector buffer; + Vector buffer8; + Vector buffer16; bool m_okay; + bool m_is8Bit; }; template