/* * Copyright (C) 2006-2021 Apple Inc. All rights reserved. * Copyright (C) 2009 Torch Mobile, Inc. * * 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 "CachedFont.h" #include "CachedFontClient.h" #include "CachedResourceClientWalker.h" #include "CachedResourceLoader.h" #include "FontCreationContext.h" #include "FontCustomPlatformData.h" #include "FontDescription.h" #include "FontPlatformData.h" #include "SharedBuffer.h" #include "TextResourceDecoder.h" #include "TypedElementDescendantIterator.h" #include "WOFFFileFormat.h" #include namespace WebCore { CachedFont::CachedFont(CachedResourceRequest&& request, PAL::SessionID sessionID, const CookieJar* cookieJar, Type type) : CachedResource(WTFMove(request), type, sessionID, cookieJar) , m_loadInitiated(false) , m_hasCreatedFontDataWrappingResource(false) { } CachedFont::~CachedFont() = default; void CachedFont::load(CachedResourceLoader&) { // Don't load the file yet. Wait for an access before triggering the load. setLoading(true); } void CachedFont::didAddClient(CachedResourceClient& client) { ASSERT(client.resourceClientType() == CachedFontClient::expectedType()); if (!isLoading()) static_cast(client).fontLoaded(*this); } void CachedFont::finishLoading(const FragmentedSharedBuffer* data, const NetworkLoadMetrics& metrics) { if (data) { m_data = data->makeContiguous(); setEncodedSize(m_data->size()); } else { m_data = nullptr; setEncodedSize(0); } setLoading(false); checkNotify(metrics); } void CachedFont::beginLoadIfNeeded(CachedResourceLoader& loader) { if (!m_loadInitiated) { m_loadInitiated = true; CachedResource::load(loader); } } bool CachedFont::ensureCustomFontData(const AtomString&) { return ensureCustomFontData(m_data.get()); } String CachedFont::calculateItemInCollection() const { return url().fragmentIdentifier().toString(); } bool CachedFont::ensureCustomFontData(SharedBuffer* data) { if (!m_fontCustomPlatformData && !errorOccurred() && !isLoading() && data) { bool wrapping; m_fontCustomPlatformData = createCustomFontData(*data, calculateItemInCollection(), wrapping); m_hasCreatedFontDataWrappingResource = m_fontCustomPlatformData && wrapping; if (!m_fontCustomPlatformData) setStatus(DecodeError); } return m_fontCustomPlatformData.get(); } std::unique_ptr CachedFont::createCustomFontData(SharedBuffer& bytes, const String& itemInCollection, bool& wrapping) { RefPtr buffer = { &bytes }; wrapping = !convertWOFFToSfntIfNecessary(buffer); return buffer ? createFontCustomPlatformData(*buffer, itemInCollection) : nullptr; } RefPtr CachedFont::createFont(const FontDescription& fontDescription, const AtomString&, bool syntheticBold, bool syntheticItalic, const FontCreationContext& fontCreationContext) { return Font::create(platformDataFromCustomData(fontDescription, syntheticBold, syntheticItalic, fontCreationContext), Font::Origin::Remote); } FontPlatformData CachedFont::platformDataFromCustomData(const FontDescription& fontDescription, bool bold, bool italic, const FontCreationContext& fontCreationContext) { ASSERT(m_fontCustomPlatformData); return platformDataFromCustomData(*m_fontCustomPlatformData, fontDescription, bold, italic, fontCreationContext); } FontPlatformData CachedFont::platformDataFromCustomData(FontCustomPlatformData& fontCustomPlatformData, const FontDescription& fontDescription, bool bold, bool italic, const FontCreationContext& fontCreationContext) { return fontCustomPlatformData.fontPlatformData(fontDescription, bold, italic, fontCreationContext); } void CachedFont::allClientsRemoved() { m_fontCustomPlatformData = nullptr; } void CachedFont::checkNotify(const NetworkLoadMetrics&) { if (isLoading()) return; CachedResourceClientWalker walker(m_clients); while (CachedFontClient* client = walker.next()) client->fontLoaded(*this); } bool CachedFont::mayTryReplaceEncodedData() const { // If a FontCustomPlatformData has ever been constructed to wrap the internal resource buffer then it still might be in use somewhere. // That platform font object might directly reference the encoded data buffer behind this CachedFont, // so replacing it is unsafe. return !m_hasCreatedFontDataWrappingResource; } }