| /* | 
 |  * Copyright (C) 2008-2021 Apple Inc. All Rights Reserved. | 
 |  * | 
 |  * Redistribution and use in source and binary forms, with or without | 
 |  * modification, are permitted provided that the following conditions | 
 |  * are met: | 
 |  * 1. Redistributions of source code must retain the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer. | 
 |  * 2. Redistributions in binary form must reproduce the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer in the | 
 |  *    documentation and/or other materials provided with the distribution. | 
 |  * | 
 |  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | 
 |  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
 |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 
 |  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR | 
 |  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
 |  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
 |  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | 
 |  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | 
 |  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
 |  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
 |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
 |  */ | 
 |  | 
 | #include "config.h" | 
 | #include "SmallStrings.h" | 
 |  | 
 | #include "JSCJSValueInlines.h" | 
 | #include <wtf/text/StringImpl.h> | 
 |  | 
 | namespace JSC { | 
 |  | 
 | SmallStrings::SmallStrings() | 
 | { | 
 |     static_assert(singleCharacterStringCount == sizeof(m_singleCharacterStrings) / sizeof(m_singleCharacterStrings[0]), "characters count is in sync with class usage"); | 
 |  | 
 |     for (unsigned i = 0; i < singleCharacterStringCount; ++i) | 
 |         m_singleCharacterStrings[i] = nullptr; | 
 | } | 
 |  | 
 | void SmallStrings::initializeCommonStrings(VM& vm) | 
 | { | 
 |     ASSERT(!m_emptyString); | 
 |     m_emptyString = JSString::createEmptyString(vm); | 
 |     ASSERT(m_needsToBeVisited); | 
 |  | 
 |     for (unsigned i = 0; i < singleCharacterStringCount; ++i) { | 
 |         ASSERT(!m_singleCharacterStrings[i]); | 
 |         const LChar string[] = { static_cast<LChar>(i) }; | 
 |         m_singleCharacterStrings[i] = JSString::createHasOtherOwner(vm, AtomStringImpl::add(string, 1).releaseNonNull()); | 
 |         ASSERT(m_needsToBeVisited); | 
 |     } | 
 |  | 
 | #define JSC_COMMON_STRINGS_ATTRIBUTE_INITIALIZE(name) initialize(&vm, m_##name, #name ## _s); | 
 |     JSC_COMMON_STRINGS_EACH_NAME(JSC_COMMON_STRINGS_ATTRIBUTE_INITIALIZE) | 
 | #undef JSC_COMMON_STRINGS_ATTRIBUTE_INITIALIZE | 
 |     initialize(&vm, m_objectStringStart, "[object "_s); | 
 |     initialize(&vm, m_objectNullString, "[object Null]"_s); | 
 |     initialize(&vm, m_objectUndefinedString, "[object Undefined]"_s); | 
 |     initialize(&vm, m_objectObjectString, "[object Object]"_s); | 
 |     initialize(&vm, m_objectArrayString, "[object Array]"_s); | 
 |     initialize(&vm, m_objectFunctionString, "[object Function]"_s); | 
 |     initialize(&vm, m_objectArgumentsString, "[object Arguments]"_s); | 
 |     initialize(&vm, m_objectDateString, "[object Date]"_s); | 
 |     initialize(&vm, m_objectRegExpString, "[object RegExp]"_s); | 
 |     initialize(&vm, m_objectErrorString, "[object Error]"_s); | 
 |     initialize(&vm, m_objectBooleanString, "[object Boolean]"_s); | 
 |     initialize(&vm, m_objectNumberString, "[object Number]"_s); | 
 |     initialize(&vm, m_objectStringString, "[object String]"_s); | 
 |     initialize(&vm, m_boundPrefixString, "bound "_s); | 
 |     initialize(&vm, m_notEqualString, "not-equal"_s); | 
 |     initialize(&vm, m_timedOutString, "timed-out"_s); | 
 |     initialize(&vm, m_okString, "ok"_s); | 
 |     initialize(&vm, m_sentinelString, "$"_s); | 
 |  | 
 |     setIsInitialized(true); | 
 | } | 
 |  | 
 | template<typename Visitor> | 
 | void SmallStrings::visitStrongReferences(Visitor& visitor) | 
 | { | 
 |     m_needsToBeVisited = false; | 
 |     visitor.appendUnbarriered(m_emptyString); | 
 |     for (unsigned i = 0; i <= maxSingleCharacterString; ++i) | 
 |         visitor.appendUnbarriered(m_singleCharacterStrings[i]); | 
 | #define JSC_COMMON_STRINGS_ATTRIBUTE_VISIT(name) visitor.appendUnbarriered(m_##name); | 
 |     JSC_COMMON_STRINGS_EACH_NAME(JSC_COMMON_STRINGS_ATTRIBUTE_VISIT) | 
 | #undef JSC_COMMON_STRINGS_ATTRIBUTE_VISIT | 
 |     visitor.appendUnbarriered(m_objectStringStart); | 
 |     visitor.appendUnbarriered(m_objectNullString); | 
 |     visitor.appendUnbarriered(m_objectUndefinedString); | 
 |     visitor.appendUnbarriered(m_objectObjectString); | 
 |     visitor.appendUnbarriered(m_objectArrayString); | 
 |     visitor.appendUnbarriered(m_objectFunctionString); | 
 |     visitor.appendUnbarriered(m_objectArgumentsString); | 
 |     visitor.appendUnbarriered(m_objectDateString); | 
 |     visitor.appendUnbarriered(m_objectRegExpString); | 
 |     visitor.appendUnbarriered(m_objectErrorString); | 
 |     visitor.appendUnbarriered(m_objectBooleanString); | 
 |     visitor.appendUnbarriered(m_objectNumberString); | 
 |     visitor.appendUnbarriered(m_objectStringString); | 
 |     visitor.appendUnbarriered(m_boundPrefixString); | 
 |     visitor.appendUnbarriered(m_notEqualString); | 
 |     visitor.appendUnbarriered(m_timedOutString); | 
 |     visitor.appendUnbarriered(m_okString); | 
 |     visitor.appendUnbarriered(m_sentinelString); | 
 | } | 
 |  | 
 | template void SmallStrings::visitStrongReferences(AbstractSlotVisitor&); | 
 | template void SmallStrings::visitStrongReferences(SlotVisitor&); | 
 |  | 
 | SmallStrings::~SmallStrings() | 
 | { | 
 | } | 
 |  | 
 | Ref<AtomStringImpl> SmallStrings::singleCharacterStringRep(unsigned char character) | 
 | { | 
 |     if (LIKELY(m_isInitialized)) | 
 |         return *static_cast<AtomStringImpl*>(const_cast<StringImpl*>(m_singleCharacterStrings[character]->tryGetValueImpl())); | 
 |     const LChar string[] = { static_cast<LChar>(character) }; | 
 |     return AtomStringImpl::add(string, 1).releaseNonNull(); | 
 | } | 
 |  | 
 | void SmallStrings::initialize(VM* vm, JSString*& string, ASCIILiteral value) | 
 | { | 
 |     string = JSString::create(*vm, AtomStringImpl::add(value)); | 
 |     ASSERT(m_needsToBeVisited); | 
 | } | 
 |  | 
 | } // namespace JSC |