[SVG -> OTF Converter] Crashes when SVG font is invalid
[WebKit-https.git] / Source / WebCore / loader / cache / CachedSVGFont.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 "CachedSVGFont.h"
29
30 #if ENABLE(SVG_FONTS)
31
32 #include "FontDescription.h"
33 #include "FontPlatformData.h"
34 #include "SVGDocument.h"
35 #include "SVGFontData.h"
36 #include "SVGFontElement.h"
37 #include "SVGFontFaceElement.h"
38 #include "SharedBuffer.h"
39 #include "TextResourceDecoder.h"
40 #include "TypedElementDescendantIterator.h"
41
42 #if ENABLE(SVG_OTF_CONVERTER)
43 #include "SVGToOTFFontConversion.h"
44 #endif
45
46 namespace WebCore {
47
48 CachedSVGFont::CachedSVGFont(const ResourceRequest& resourceRequest, SessionID sessionID)
49     : CachedFont(resourceRequest, sessionID, SVGFontResource)
50     , m_externalSVGFontElement(nullptr)
51 {
52 }
53
54 PassRefPtr<SimpleFontData> CachedSVGFont::getFontData(const FontDescription& fontDescription, const AtomicString& remoteURI, bool syntheticBold, bool syntheticItalic, bool externalSVG)
55 {
56 #if ENABLE(SVG_OTF_CONVERTER)
57     if (!externalSVG || firstFontFace(remoteURI))
58         return CachedFont::getFontData(fontDescription, remoteURI, syntheticBold, syntheticItalic, externalSVG);
59 #else
60     if (!externalSVG)
61         return CachedFont::getFontData(fontDescription, remoteURI, syntheticBold, syntheticItalic, externalSVG);
62
63     if (SVGFontFaceElement* firstFontFace = this->firstFontFace(remoteURI))
64         return SimpleFontData::create(std::make_unique<SVGFontData>(firstFontFace), fontDescription.computedPixelSize(), syntheticBold, syntheticItalic);
65 #endif
66     return nullptr;
67 }
68
69 FontPlatformData CachedSVGFont::platformDataFromCustomData(float size, bool bold, bool italic, FontOrientation orientation, FontWidthVariant widthVariant, FontRenderingMode renderingMode)
70 {
71     if (m_externalSVGDocument)
72         return FontPlatformData(size, bold, italic);
73     return CachedFont::platformDataFromCustomData(size, bold, italic, orientation, widthVariant, renderingMode);
74 }
75
76 bool CachedSVGFont::ensureCustomFontData(bool externalSVG, const AtomicString& remoteURI)
77 {
78     if (!externalSVG)
79         return CachedFont::ensureCustomFontData(externalSVG, remoteURI);
80
81     if (!m_externalSVGDocument && !errorOccurred() && !isLoading() && m_data) {
82         m_externalSVGDocument = SVGDocument::create(nullptr, URL());
83         RefPtr<TextResourceDecoder> decoder = TextResourceDecoder::create("application/xml");
84         m_externalSVGDocument->setContent(decoder->decodeAndFlush(m_data->data(), m_data->size()));
85 #if !ENABLE(SVG_OTF_CONVERTER)
86         if (decoder->sawError())
87             m_externalSVGDocument = nullptr;
88 #else
89         if (decoder->sawError())
90             m_externalSVGDocument = nullptr;
91         else
92             maybeInitializeExternalSVGFontElement(remoteURI);
93         if (!m_externalSVGFontElement)
94             return false;
95         Vector<char> convertedFont = convertSVGToOTFFont(*m_externalSVGFontElement);
96         m_convertedFont = SharedBuffer::adoptVector(convertedFont);
97 #endif
98     }
99
100 #if !ENABLE(SVG_OTF_CONVERTER)
101     return m_externalSVGDocument;
102 #else
103     return m_externalSVGDocument && CachedFont::ensureCustomFontData(m_convertedFont.get());
104 #endif
105 }
106
107 SVGFontElement* CachedSVGFont::getSVGFontById(const String& fontName) const
108 {
109     ASSERT(m_externalSVGDocument);
110     auto elements = descendantsOfType<SVGFontElement>(*m_externalSVGDocument);
111
112     if (fontName.isEmpty())
113         return elements.first();
114
115     for (auto& element : elements) {
116         if (element.getIdAttribute() == fontName)
117             return &element;
118     }
119     return nullptr;
120 }
121
122 SVGFontElement* CachedSVGFont::maybeInitializeExternalSVGFontElement(const AtomicString& remoteURI)
123 {
124     if (m_externalSVGFontElement)
125         return m_externalSVGFontElement;
126     String fragmentIdentifier;
127     size_t start = remoteURI.find('#');
128     if (start != notFound)
129         fragmentIdentifier = remoteURI.string().substring(start + 1);
130     m_externalSVGFontElement = getSVGFontById(fragmentIdentifier);
131     return m_externalSVGFontElement;
132 }
133
134 SVGFontFaceElement* CachedSVGFont::firstFontFace(const AtomicString& remoteURI)
135 {
136     if (!maybeInitializeExternalSVGFontElement(remoteURI))
137         return nullptr;
138
139     if (auto* firstFontFace = childrenOfType<SVGFontFaceElement>(*m_externalSVGFontElement).first())
140         return firstFontFace;
141     return nullptr;
142 }
143
144 }
145
146 #endif