| /* |
| * Copyright (C) 2007-2014 Apple Inc. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 59 Temple Place - Suite 330, |
| * Boston, MA 02111-1307, USA. |
| * |
| */ |
| |
| #pragma once |
| |
| #include <algorithm> |
| #include <wtf/MainThread.h> |
| #include <wtf/StdLibExtras.h> |
| #include <wtf/TZoneMalloc.h> |
| #include <wtf/text/AtomString.h> |
| |
| namespace WebCore { |
| |
| class SpaceSplitStringData { |
| WTF_MAKE_TZONE_ALLOCATED(SpaceSplitStringData); |
| WTF_MAKE_NONCOPYABLE(SpaceSplitStringData); |
| public: |
| static RefPtr<SpaceSplitStringData> create(const AtomString&); |
| |
| auto begin() const LIFETIME_BOUND { return std::to_address(tokenArray().begin()); } |
| auto end() const LIFETIME_BOUND { return std::to_address(tokenArray().end()); } |
| auto begin() LIFETIME_BOUND { return std::to_address(tokenArray().begin()); } |
| auto end() LIFETIME_BOUND { return std::to_address(tokenArray().end()); } |
| |
| bool contains(const AtomString& string) |
| { |
| auto tokens = tokenArray(); |
| return std::ranges::find(tokens, string) != tokens.end(); |
| } |
| |
| bool containsAll(SpaceSplitStringData&); |
| |
| unsigned size() const { return m_size; } |
| static constexpr ptrdiff_t sizeMemoryOffset() { return OBJECT_OFFSETOF(SpaceSplitStringData, m_size); } |
| |
| const AtomString& operator[](unsigned i) LIFETIME_BOUND { return tokenArray()[i]; } |
| |
| void ref() |
| { |
| ASSERT(isMainThread()); |
| ASSERT(m_refCount); |
| ++m_refCount; |
| } |
| |
| void deref() |
| { |
| ASSERT(isMainThread()); |
| ASSERT(m_refCount); |
| unsigned tempRefCount = m_refCount - 1; |
| if (!tempRefCount) { |
| destroy(this); |
| return; |
| } |
| m_refCount = tempRefCount; |
| } |
| |
| const AtomString& keyString() const LIFETIME_BOUND { return m_keyString; } |
| |
| static constexpr ptrdiff_t tokensMemoryOffset() { return sizeof(SpaceSplitStringData); } |
| |
| private: |
| static Ref<SpaceSplitStringData> create(const AtomString&, unsigned tokenCount); |
| SpaceSplitStringData(const AtomString& string, unsigned size) |
| : m_keyString(string) |
| , m_refCount(1) |
| , m_size(size) |
| { |
| ASSERT(!string.isEmpty()); |
| ASSERT_WITH_MESSAGE(m_size, "SpaceSplitStringData should never be empty by definition. There is no difference between empty and null."); |
| } |
| |
| ~SpaceSplitStringData() = default; |
| static void destroy(SpaceSplitStringData*); |
| |
| std::span<AtomString> tokenArray() LIFETIME_BOUND { return unsafeMakeSpan(m_tokens, m_size); } |
| std::span<const AtomString> tokenArray() const LIFETIME_BOUND { return unsafeMakeSpan(m_tokens, m_size); } |
| |
| AtomString m_keyString; |
| unsigned m_refCount; |
| unsigned m_size; |
| AtomString m_tokens[0]; |
| }; |
| |
| class SpaceSplitString { |
| public: |
| SpaceSplitString() = default; |
| |
| enum class ShouldFoldCase : bool { No, Yes }; |
| SpaceSplitString(const AtomString&, ShouldFoldCase); |
| |
| const AtomString& keyString() const LIFETIME_BOUND |
| { |
| if (m_data) |
| return m_data->keyString(); |
| return nullAtom(); |
| } |
| |
| friend bool operator==(const SpaceSplitString&, const SpaceSplitString&) = default; |
| void set(const AtomString&, ShouldFoldCase); |
| void clear() { m_data = nullptr; } |
| |
| bool contains(const AtomString& string) const { return m_data && m_data->contains(string); } |
| bool containsAll(const SpaceSplitString& names) const { return !names.m_data || (m_data && m_data->containsAll(*names.m_data)); } |
| |
| unsigned size() const { return m_data ? m_data->size() : 0; } |
| bool isEmpty() const { return !m_data; } |
| const AtomString& operator[](unsigned i) const LIFETIME_BOUND |
| { |
| ASSERT_WITH_SECURITY_IMPLICATION(m_data); |
| return (*m_data)[i]; |
| } |
| |
| auto begin() const LIFETIME_BOUND { return m_data ? m_data->begin() : nullptr; } |
| auto end() const LIFETIME_BOUND { return m_data ? m_data->end() : nullptr; } |
| auto begin() LIFETIME_BOUND { return m_data ? m_data->begin() : nullptr; } |
| auto end() LIFETIME_BOUND { return m_data ? m_data->end() : nullptr; } |
| |
| static bool spaceSplitStringContainsValue(StringView spaceSplitString, StringView value, ShouldFoldCase); |
| |
| private: |
| RefPtr<SpaceSplitStringData> m_data; |
| }; |
| |
| inline SpaceSplitString::SpaceSplitString(const AtomString& string, ShouldFoldCase shouldFoldCase) |
| : m_data(!string.isEmpty() ? SpaceSplitStringData::create(shouldFoldCase == ShouldFoldCase::Yes ? string.convertToASCIILowercase() : string) : nullptr) |
| { |
| } |
| |
| } // namespace WebCore |