/* * Copyright (C) 2010 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. */ #ifndef StringImplBase_h #define StringImplBase_h #include <wtf/unicode/Unicode.h> namespace WTF { class StringImplBase { WTF_MAKE_NONCOPYABLE(StringImplBase); WTF_MAKE_FAST_ALLOCATED; public: bool isStringImpl() { return (m_refCountAndFlags & s_refCountInvalidForStringImpl) != s_refCountInvalidForStringImpl; } unsigned length() const { return m_length; } void ref() { m_refCountAndFlags += s_refCountIncrement; } protected: enum BufferOwnership { BufferInternal, BufferOwned, BufferSubstring, BufferShared, }; // For SmallStringStorage, which allocates an array and uses an in-place new. StringImplBase() { } StringImplBase(unsigned length, BufferOwnership ownership) : m_refCountAndFlags(s_refCountIncrement | s_refCountFlagShouldReportedCost | ownership) , m_length(length) { ASSERT(isStringImpl()); } enum StaticStringConstructType { ConstructStaticString }; StringImplBase(unsigned length, StaticStringConstructType) : m_refCountAndFlags(s_refCountFlagStatic | s_refCountFlagIsIdentifier | BufferOwned) , m_length(length) { ASSERT(isStringImpl()); } // This constructor is not used when creating StringImpl objects, // and sets the flags into a state marking the object as such. enum NonStringImplConstructType { ConstructNonStringImpl }; StringImplBase(NonStringImplConstructType) : m_refCountAndFlags(s_refCountIncrement | s_refCountInvalidForStringImpl) , m_length(0) { ASSERT(!isStringImpl()); } // The bottom 7 bits hold flags, the top 25 bits hold the ref count. // When dereferencing StringImpls we check for the ref count AND the // static bit both being zero - static strings are never deleted. static const unsigned s_refCountMask = 0xFFFFFF80; static const unsigned s_refCountIncrement = 0x80; static const unsigned s_refCountFlagStatic = 0x40; static const unsigned s_refCountFlagHasTerminatingNullCharacter = 0x20; static const unsigned s_refCountFlagIsAtomic = 0x10; static const unsigned s_refCountFlagShouldReportedCost = 0x8; static const unsigned s_refCountFlagIsIdentifier = 0x4; static const unsigned s_refCountMaskBufferOwnership = 0x3; // An invalid permutation of flags (static & shouldReportedCost - static strings do not // set shouldReportedCost in the constructor, and this bit is only ever cleared, not set). // Used by "ConstructNonStringImpl" constructor, above. static const unsigned s_refCountInvalidForStringImpl = s_refCountFlagStatic | s_refCountFlagShouldReportedCost; unsigned m_refCountAndFlags; unsigned m_length; }; } // namespace WTF using WTF::StringImplBase; #endif