blob: f93eecd50dfd017eb001cc541ec966743dba0a18 [file] [log] [blame]
/*
* Copyright (C) 2010-2018 Apple Inc. All rights reserved.
* Copyright (C) 2012 Google Inc. All rights reserved.
* Copyright (C) 2017 Yusuke Suzuki <utatane.tea@gmail.com>. All rights reserved.
* Copyright (C) 2017 Mozilla Foundation. All rights reserved.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include "config.h"
#include <wtf/text/StringBuilder.h>
#include <wtf/text/StringBuilderInternals.h>
#include <wtf/text/WTFString.h>
namespace WTF {
// This table driven escaping is ported from SpiderMonkey.
static const constexpr LChar escapedFormsForJSON[0x100] = {
'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u',
'b', 't', 'n', 'u', 'f', 'r', 'u', 'u',
'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u',
'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u',
0, 0, '\"', 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, '\\', 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
};
template<typename OutputCharacterType, typename InputCharacterType>
ALWAYS_INLINE static void appendQuotedJSONStringInternal(OutputCharacterType*& output, const InputCharacterType* input, unsigned length)
{
for (auto* end = input + length; input != end; ++input) {
auto character = *input;
if (LIKELY(character <= 0xFF)) {
auto escaped = escapedFormsForJSON[character];
if (LIKELY(!escaped)) {
*output++ = character;
continue;
}
*output++ = '\\';
*output++ = escaped;
if (UNLIKELY(escaped == 'u')) {
*output++ = '0';
*output++ = '0';
*output++ = upperNibbleToLowercaseASCIIHexDigit(character);
*output++ = lowerNibbleToLowercaseASCIIHexDigit(character);
}
continue;
}
if (LIKELY(!U16_IS_SURROGATE(character))) {
*output++ = character;
continue;
}
auto next = input + 1;
bool isValidSurrogatePair = U16_IS_SURROGATE_LEAD(character) && next != end && U16_IS_TRAIL(*next);
if (isValidSurrogatePair) {
*output++ = character;
*output++ = *next;
++input;
continue;
}
uint8_t upper = static_cast<uint32_t>(character) >> 8;
uint8_t lower = static_cast<uint8_t>(character);
*output++ = '\\';
*output++ = 'u';
*output++ = upperNibbleToLowercaseASCIIHexDigit(upper);
*output++ = lowerNibbleToLowercaseASCIIHexDigit(upper);
*output++ = upperNibbleToLowercaseASCIIHexDigit(lower);
*output++ = lowerNibbleToLowercaseASCIIHexDigit(lower);
continue;
}
}
void StringBuilder::appendQuotedJSONString(const String& string)
{
if (hasOverflowed())
return;
// Make sure we have enough buffer space to append this string for worst case without reallocating.
// The 2 is for the '"' quotes on each end.
// The 6 is the worst case for a single code unit that could be encoded as \uNNNN.
CheckedUint32 stringLength = string.length();
stringLength *= 6;
stringLength += 2;
if (stringLength.hasOverflowed()) {
didOverflow();
return;
}
auto stringLengthValue = stringLength.value();
if (is8Bit() && string.is8Bit()) {
if (auto* output = extendBufferForAppending<LChar>(saturatedSum<uint32_t>(m_length, stringLengthValue))) {
auto* end = output + stringLengthValue;
*output++ = '"';
appendQuotedJSONStringInternal(output, string.characters8(), string.length());
*output++ = '"';
if (output < end)
shrink(m_length - (end - output));
}
} else {
if (auto* output = extendBufferForAppendingWithUpconvert(saturatedSum<uint32_t>(m_length, stringLengthValue))) {
auto* end = output + stringLengthValue;
*output++ = '"';
if (string.is8Bit())
appendQuotedJSONStringInternal(output, string.characters8(), string.length());
else
appendQuotedJSONStringInternal(output, string.characters16(), string.length());
*output++ = '"';
if (output < end)
shrink(m_length - (end - output));
}
}
}
} // namespace WTF