[Chromium] Uninitialized access in SimpleFontDataSkia::platformInit
[WebKit-https.git] / Source / WebCore / platform / graphics / skia / SimpleFontDataSkia.cpp
1 /*
2  * Copyright (c) 2008, 2009, Google Inc. All rights reserved.
3  * 
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  * 
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  * 
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "SimpleFontData.h"
33
34 #include "FloatRect.h"
35 #include "Font.h"
36 #include "FontCache.h"
37 #include "FontDescription.h"
38 #include "Logging.h"
39 #include "SkFontHost.h"
40 #include "SkPaint.h"
41 #include "SkTime.h"
42 #include "SkTypeface.h"
43 #include "SkTypes.h"
44 #include "VDMXParser.h"
45
46 namespace WebCore {
47
48 // Smallcaps versions of fonts are 70% the size of the normal font.
49 static const float smallCapsFraction = 0.7f;
50 static const float emphasisMarkFraction = .5;
51 // This is the largest VDMX table which we'll try to load and parse.
52 static const size_t maxVDMXTableSize = 1024 * 1024; // 1 MB
53
54 void SimpleFontData::platformInit()
55 {
56     if (!m_platformData.size()) {
57         m_fontMetrics.reset();
58         m_avgCharWidth = 0;
59         m_maxCharWidth = 0;
60         return;
61     }
62
63     SkPaint paint;
64     SkPaint::FontMetrics metrics;
65
66     m_platformData.setupPaint(&paint);
67     paint.getFontMetrics(&metrics);
68     const SkFontID fontID = m_platformData.uniqueID();
69
70     static const uint32_t vdmxTag = SkSetFourByteTag('V', 'D', 'M', 'X');
71     int pixelSize = m_platformData.size() + 0.5;
72     int vdmxAscent, vdmxDescent;
73     bool isVDMXValid = false;
74
75     size_t vdmxSize = SkFontHost::GetTableSize(fontID, vdmxTag);
76     if (vdmxSize && vdmxSize < maxVDMXTableSize) {
77         uint8_t* vdmxTable = (uint8_t*) fastMalloc(vdmxSize);
78         if (vdmxTable
79             && SkFontHost::GetTableData(fontID, vdmxTag, 0, vdmxSize, vdmxTable) == vdmxSize
80             && parseVDMX(&vdmxAscent, &vdmxDescent, vdmxTable, vdmxSize, pixelSize))
81             isVDMXValid = true;
82         fastFree(vdmxTable);
83     }
84
85     float ascent;
86     float descent;
87
88     // Beware those who step here: This code is designed to match Win32 font
89     // metrics *exactly*.
90     if (isVDMXValid) {
91         ascent = vdmxAscent;
92         descent = -vdmxDescent;
93     } else {
94         SkScalar height = -metrics.fAscent + metrics.fDescent + metrics.fLeading;
95         ascent = SkScalarRound(-metrics.fAscent);
96         descent = SkScalarRound(height) - ascent;
97     }
98
99     m_fontMetrics.setAscent(ascent);
100     m_fontMetrics.setDescent(descent);
101
102     float xHeight;
103     if (metrics.fXHeight)
104         xHeight = metrics.fXHeight;
105     else {
106         // hack taken from the Windows port
107         xHeight = ascent * 0.56f;
108     }
109
110     float lineGap = SkScalarToFloat(metrics.fLeading);
111     m_fontMetrics.setLineGap(lineGap);
112     m_fontMetrics.setXHeight(xHeight);
113     m_fontMetrics.setLineSpacing(lroundf(ascent) + lroundf(descent) + lroundf(lineGap));
114
115     if (platformData().orientation() == Vertical && !isTextOrientationFallback()) {
116         static const uint32_t vheaTag = SkSetFourByteTag('v', 'h', 'e', 'a');
117         static const uint32_t vorgTag = SkSetFourByteTag('V', 'O', 'R', 'G');
118         size_t vheaSize = SkFontHost::GetTableSize(fontID, vheaTag);
119         size_t vorgSize = SkFontHost::GetTableSize(fontID, vorgTag);
120         if ((vheaSize > 0) || (vorgSize > 0))
121             m_hasVerticalGlyphs = true;
122     }
123
124     // In WebKit/WebCore/platform/graphics/SimpleFontData.cpp, m_spaceWidth is
125     // calculated for us, but we need to calculate m_maxCharWidth and
126     // m_avgCharWidth in order for text entry widgets to be sized correctly.
127
128     SkScalar xRange = metrics.fXMax - metrics.fXMin;
129     m_maxCharWidth = SkScalarRound(xRange * SkScalarRound(m_platformData.size()));
130
131     if (metrics.fAvgCharWidth)
132         m_avgCharWidth = SkScalarRound(metrics.fAvgCharWidth);
133     else {
134         m_avgCharWidth = xHeight;
135
136         GlyphPage* glyphPageZero = GlyphPageTreeNode::getRootChild(this, 0)->page();
137
138         if (glyphPageZero) {
139             static const UChar32 xChar = 'x';
140             const Glyph xGlyph = glyphPageZero->glyphDataForCharacter(xChar).glyph;
141
142             if (xGlyph) {
143                 // In widthForGlyph(), xGlyph will be compared with
144                 // m_zeroWidthSpaceGlyph, which isn't initialized yet here.
145                 // Initialize it with zero to make sure widthForGlyph() returns
146                 // the right width.
147                 m_zeroWidthSpaceGlyph = 0;
148                 m_avgCharWidth = widthForGlyph(xGlyph);
149             }
150         }
151     }
152 }
153
154 void SimpleFontData::platformCharWidthInit()
155 {
156     // charwidths are set in platformInit.
157 }
158
159 void SimpleFontData::platformDestroy()
160 {
161 }
162
163 PassOwnPtr<SimpleFontData> SimpleFontData::createScaledFontData(const FontDescription& fontDescription, float scaleFactor) const
164 {
165     const float scaledSize = lroundf(fontDescription.computedSize() * scaleFactor);
166     return adoptPtr(new SimpleFontData(FontPlatformData(m_platformData, scaledSize), isCustomFont(), false));
167 }
168
169 SimpleFontData* SimpleFontData::smallCapsFontData(const FontDescription& fontDescription) const
170 {
171     if (!m_derivedFontData)
172         m_derivedFontData = DerivedFontData::create(isCustomFont());
173     if (!m_derivedFontData->smallCaps)
174         m_derivedFontData->smallCaps = createScaledFontData(fontDescription, smallCapsFraction);
175
176     return m_derivedFontData->smallCaps.get();
177 }
178
179 SimpleFontData* SimpleFontData::emphasisMarkFontData(const FontDescription& fontDescription) const
180 {
181     if (!m_derivedFontData)
182         m_derivedFontData = DerivedFontData::create(isCustomFont());
183     if (!m_derivedFontData->emphasisMark)
184         m_derivedFontData->emphasisMark = createScaledFontData(fontDescription, emphasisMarkFraction);
185
186     return m_derivedFontData->emphasisMark.get();
187 }
188
189 bool SimpleFontData::containsCharacters(const UChar* characters, int length) const
190 {
191     SkPaint paint;
192     static const unsigned maxBufferCount = 64;
193     uint16_t glyphs[maxBufferCount];
194
195     m_platformData.setupPaint(&paint);
196     paint.setTextEncoding(SkPaint::kUTF16_TextEncoding);
197
198     while (length > 0) {
199         int n = SkMin32(length, SK_ARRAY_COUNT(glyphs));
200
201         // textToGlyphs takes a byte count so we double the character count.
202         int count = paint.textToGlyphs(characters, n * 2, glyphs);
203         for (int i = 0; i < count; i++) {
204             if (!glyphs[i])
205                 return false; // missing glyph
206         }
207
208         characters += n;
209         length -= n;
210     }
211
212     return true;
213 }
214
215 void SimpleFontData::determinePitch()
216 {
217     m_treatAsFixedPitch = platformData().isFixedPitch();
218 }
219
220 FloatRect SimpleFontData::platformBoundsForGlyph(Glyph) const
221 {
222     return FloatRect();
223 }
224     
225 float SimpleFontData::platformWidthForGlyph(Glyph glyph) const
226 {
227     if (!m_platformData.size())
228         return 0;
229
230     SkASSERT(sizeof(glyph) == 2); // compile-time assert
231
232     SkPaint paint;
233
234     m_platformData.setupPaint(&paint);
235
236     paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
237     SkScalar width = paint.measureText(&glyph, 2);
238
239     // Though WebKit supports non-integral advances, Skia only supports them
240     // for "subpixel" (distinct from LCD subpixel antialiasing) text, which
241     // we don't use.
242     return round(SkScalarToFloat(width));
243 }
244
245 } // namespace WebCore