[iOS] [OSX] Don't transcode WOFF on platforms that support it natively
[WebKit-https.git] / Source / WebCore / loader / cache / CachedFont.cpp
1 /*
2  * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
3  * Copyright (C) 2009 Torch Mobile, Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
15  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
18  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
25  */
26
27 #include "config.h"
28 #include "CachedFont.h"
29
30 #include "CachedFontClient.h"
31 #include "CachedResourceClientWalker.h"
32 #include "CachedResourceLoader.h"
33 #include "FontCustomPlatformData.h"
34 #include "FontPlatformData.h"
35 #include "MemoryCache.h"
36 #include "ResourceBuffer.h"
37 #include "SharedBuffer.h"
38 #include "TextResourceDecoder.h"
39 #include "TypedElementDescendantIterator.h"
40 #include "WOFFFileFormat.h"
41 #include <wtf/Vector.h>
42
43 #if ENABLE(SVG_FONTS)
44 #include "NodeList.h"
45 #include "SVGDocument.h"
46 #include "SVGElement.h"
47 #include "SVGFontElement.h"
48 #include "SVGGElement.h"
49 #include "SVGNames.h"
50 #endif
51
52 namespace WebCore {
53
54 CachedFont::CachedFont(const ResourceRequest& resourceRequest, SessionID sessionID)
55     : CachedResource(resourceRequest, FontResource, sessionID)
56     , m_loadInitiated(false)
57     , m_hasCreatedFontDataWrappingResource(false)
58 {
59 }
60
61 CachedFont::~CachedFont()
62 {
63 }
64
65 void CachedFont::load(CachedResourceLoader*, const ResourceLoaderOptions& options)
66 {
67     // Don't load the file yet.  Wait for an access before triggering the load.
68     setLoading(true);
69     m_options = options;
70 }
71
72 void CachedFont::didAddClient(CachedResourceClient* c)
73 {
74     ASSERT(c->resourceClientType() == CachedFontClient::expectedType());
75     if (!isLoading())
76         static_cast<CachedFontClient*>(c)->fontLoaded(this);
77 }
78
79 void CachedFont::finishLoading(ResourceBuffer* data)
80 {
81     m_data = data;
82     setEncodedSize(m_data.get() ? m_data->size() : 0);
83     setLoading(false);
84     checkNotify();
85 }
86
87 void CachedFont::beginLoadIfNeeded(CachedResourceLoader* dl)
88 {
89     if (!m_loadInitiated) {
90         m_loadInitiated = true;
91         CachedResource::load(dl, m_options);
92     }
93 }
94
95 bool CachedFont::ensureCustomFontData()
96 {
97     if (!m_fontData && !errorOccurred() && !isLoading() && m_data) {
98         SharedBuffer* buffer = m_data.get()->sharedBuffer();
99         ASSERT(buffer);
100
101         bool fontIsWOFF = false;
102 #if (!PLATFORM(MAC) || __MAC_OS_X_VERSION_MIN_REQUIRED <= 1090) && (!PLATFORM(IOS) || __IPHONE_OS_VERSION_MIN_REQUIRED < 80000)
103         RefPtr<SharedBuffer> sfntBuffer;
104
105         fontIsWOFF = isWOFF(buffer);
106         if (fontIsWOFF) {
107             Vector<char> sfnt;
108             if (convertWOFFToSfnt(buffer, sfnt)) {
109                 sfntBuffer = SharedBuffer::adoptVector(sfnt);
110                 buffer = sfntBuffer.get();
111             } else
112                 buffer = nullptr;
113         }
114 #endif
115
116         m_fontData = buffer ? createFontCustomPlatformData(*buffer) : nullptr;
117         if (m_fontData)
118             m_hasCreatedFontDataWrappingResource = !fontIsWOFF;
119         else
120             setStatus(DecodeError);
121     }
122     return m_fontData.get();
123 }
124
125 FontPlatformData CachedFont::platformDataFromCustomData(float size, bool bold, bool italic, FontOrientation orientation, FontWidthVariant widthVariant, FontRenderingMode renderingMode)
126 {
127 #if ENABLE(SVG_FONTS)
128     if (m_externalSVGDocument)
129         return FontPlatformData(size, bold, italic);
130 #endif
131     ASSERT(m_fontData);
132     return m_fontData->fontPlatformData(static_cast<int>(size), bold, italic, orientation, widthVariant, renderingMode);
133 }
134
135 #if ENABLE(SVG_FONTS)
136
137 bool CachedFont::ensureSVGFontData()
138 {
139     if (!m_externalSVGDocument && !errorOccurred() && !isLoading() && m_data) {
140         m_externalSVGDocument = SVGDocument::create(nullptr, URL());
141         RefPtr<TextResourceDecoder> decoder = TextResourceDecoder::create("application/xml");
142         m_externalSVGDocument->setContent(decoder->decodeAndFlush(m_data->data(), m_data->size()));
143         if (decoder->sawError())
144             m_externalSVGDocument = nullptr;
145     }
146     return m_externalSVGDocument;
147 }
148
149 SVGFontElement* CachedFont::getSVGFontById(const String& fontName) const
150 {
151     auto elements = descendantsOfType<SVGFontElement>(*m_externalSVGDocument);
152
153     if (fontName.isEmpty())
154         return elements.first();
155
156     for (auto& element : elements) {
157         if (element.getIdAttribute() == fontName)
158             return &element;
159     }
160     return nullptr;
161 }
162
163 #endif
164
165 void CachedFont::allClientsRemoved()
166 {
167     m_fontData = nullptr;
168 }
169
170 void CachedFont::checkNotify()
171 {
172     if (isLoading())
173         return;
174     
175     CachedResourceClientWalker<CachedFontClient> w(m_clients);
176     while (CachedFontClient* c = w.next())
177          c->fontLoaded(this);
178 }
179
180 bool CachedFont::mayTryReplaceEncodedData() const
181 {
182     // If a FontCustomPlatformData has ever been constructed to wrap the internal resource buffer then it still might be in use somewhere.
183     // That platform font object might directly reference the encoded data buffer behind this CachedFont,
184     // so replacing it is unsafe.
185
186     return !m_hasCreatedFontDataWrappingResource;
187 }
188
189 }