Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(146)

Side by Side Diff: src/pdf/SkPDFFont.cpp

Issue 2278703002: SkPDF: Glyph validation change (Closed) Base URL: https://skia.googlesource.com/skia.git@master
Patch Set: fix compile Created 4 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« src/pdf/SkPDFDevice.cpp ('K') | « src/pdf/SkPDFFont.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright 2011 Google Inc. 2 * Copyright 2011 Google Inc.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license that can be 4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file. 5 * found in the LICENSE file.
6 */ 6 */
7 7
8 #include "SkData.h" 8 #include "SkData.h"
9 #include "SkGlyphCache.h" 9 #include "SkGlyphCache.h"
10 #include "SkPaint.h" 10 #include "SkPaint.h"
11 #include "SkPDFCanon.h" 11 #include "SkPDFCanon.h"
12 #include "SkPDFConvertType1FontStream.h" 12 #include "SkPDFConvertType1FontStream.h"
13 #include "SkPDFDevice.h" 13 #include "SkPDFDevice.h"
14 #include "SkPDFMakeCIDGlyphWidthsArray.h" 14 #include "SkPDFMakeCIDGlyphWidthsArray.h"
15 #include "SkPDFMakeToUnicodeCmap.h" 15 #include "SkPDFMakeToUnicodeCmap.h"
16 #include "SkPDFFont.h" 16 #include "SkPDFFont.h"
17 #include "SkPDFUtils.h" 17 #include "SkPDFUtils.h"
18 #include "SkRefCnt.h" 18 #include "SkRefCnt.h"
19 #include "SkScalar.h" 19 #include "SkScalar.h"
20 #include "SkStream.h" 20 #include "SkStream.h"
21 #include "SkTypefacePriv.h"
22 #include "SkTypes.h" 21 #include "SkTypes.h"
23 #include "SkUtils.h" 22 #include "SkUtils.h"
24 23
25 // TODO(halcanary): Switch Chrome from SK_SFNTLY_SUBSETTER to SK_PDF_USE_SFNTLY. 24 // TODO(halcanary): Switch Chrome from SK_SFNTLY_SUBSETTER to SK_PDF_USE_SFNTLY.
26 #if defined(SK_SFNTLY_SUBSETTER) && !defined(SK_PDF_USE_SFNTLY) 25 #if defined(SK_SFNTLY_SUBSETTER) && !defined(SK_PDF_USE_SFNTLY)
27 #define SK_PDF_USE_SFNTLY 26 #define SK_PDF_USE_SFNTLY
28 #endif 27 #endif
29 28
30 #ifdef SK_PDF_USE_SFNTLY 29 #ifdef SK_PDF_USE_SFNTLY
31 #include "sample/chromium/font_subsetter.h" 30 #include "sample/chromium/font_subsetter.h"
(...skipping 109 matching lines...) Expand 10 before | Expand all | Expand 10 after
141 static bool can_embed(const SkAdvancedTypefaceMetrics& metrics) { 140 static bool can_embed(const SkAdvancedTypefaceMetrics& metrics) {
142 return !SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kNotEmbeddable_ FontFlag); 141 return !SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kNotEmbeddable_ FontFlag);
143 } 142 }
144 143
145 #ifdef SK_PDF_USE_SFNTLY 144 #ifdef SK_PDF_USE_SFNTLY
146 static bool can_subset(const SkAdvancedTypefaceMetrics& metrics) { 145 static bool can_subset(const SkAdvancedTypefaceMetrics& metrics) {
147 return !SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kNotSubsettable _FontFlag); 146 return !SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kNotSubsettable _FontFlag);
148 } 147 }
149 #endif 148 #endif
150 149
151 int SkPDFFont::glyphsToPDFFontEncoding(SkGlyphID* glyphIDs, int numGlyphs) const {
152 // A font with multibyte glyphs will support all glyph IDs in a single font.
153 if (this->multiByteGlyphs()) {
154 return numGlyphs;
155 }
156
157 for (int i = 0; i < numGlyphs; i++) {
158 if (glyphIDs[i] == 0) {
159 continue;
160 }
161 if (glyphIDs[i] < fFirstGlyphID || glyphIDs[i] > fLastGlyphID) {
162 return i;
163 }
164 glyphIDs[i] -= (fFirstGlyphID - 1);
165 }
166
167 return numGlyphs;
168 }
169
170 int SkPDFFont::glyphsToPDFFontEncodingCount(const SkGlyphID* glyphIDs,
171 int numGlyphs) const {
172 if (this->multiByteGlyphs()) { // A font with multibyte glyphs will
173 return numGlyphs; // support all glyph IDs in a single font.
174 }
175 for (int i = 0; i < numGlyphs; i++) {
176 if (glyphIDs[i] != 0 &&
177 (glyphIDs[i] < fFirstGlyphID || glyphIDs[i] > fLastGlyphID)) {
178 return i;
179 }
180 }
181 return numGlyphs;
182 }
183
184
185 const SkAdvancedTypefaceMetrics* SkPDFFont::GetMetrics(SkTypeface* typeface, 150 const SkAdvancedTypefaceMetrics* SkPDFFont::GetMetrics(SkTypeface* typeface,
186 SkPDFCanon* canon) { 151 SkPDFCanon* canon) {
187 SkFontID id = SkTypeface::UniqueID(typeface); 152 SkASSERT(typeface);
153 SkFontID id = typeface->uniqueID();
188 if (SkAdvancedTypefaceMetrics** ptr = canon->fTypefaceMetrics.find(id)) { 154 if (SkAdvancedTypefaceMetrics** ptr = canon->fTypefaceMetrics.find(id)) {
189 return *ptr; 155 return *ptr;
190 } 156 }
191 sk_sp<SkTypeface> defaultFace; 157 sk_sp<SkAdvancedTypefaceMetrics> metrics;
192 if (!typeface) { 158 SkPDFFont::TypefaceInfo faceInfo = SkPDFFont::GetTypefaceInfo(typeface);
193 defaultFace = SkTypeface::MakeDefault(); 159 if (faceInfo.fGood) {
194 typeface = defaultFace.get(); 160 metrics.reset(
195 } 161 typeface->getAdvancedTypefaceMetrics(
196 sk_sp<SkAdvancedTypefaceMetrics> metrics( 162 SkTypeface::kGlyphNames_PerGlyphInfo | SkTypeface::kToUn icode_PerGlyphInfo,
197 typeface->getAdvancedTypefaceMetrics( 163 nullptr, 0));
198 SkTypeface::kGlyphNames_PerGlyphInfo | SkTypeface::kToUnicod e_PerGlyphInfo, 164 if (!metrics) {
199 nullptr, 0));
200 if (!metrics) {
201 if (typeface->countGlyphs() > 0) {
202 metrics = sk_make_sp<SkAdvancedTypefaceMetrics>(); 165 metrics = sk_make_sp<SkAdvancedTypefaceMetrics>();
166 metrics->fLastGlyphID = faceInfo.fMaxGlyphID;
203 } else { 167 } else {
204 SkDEBUGF(("SkPDF: SkTypeface:getAdvancedTypefaceMetrics() returned n ull.\n")); 168 SkASSERT(metrics->fLastGlyphID == faceInfo.fMaxGlyphID);
205 } 169 }
170 } else {
171 SkDEBUGF(("SkPDF: SkTypeface:countGlyphs() bad value.\n"));
206 } 172 }
207 // May cache null to skip this check. use SkSafeUnref. 173 // May cache null to skip this check. use SkSafeUnref.
208 return *canon->fTypefaceMetrics.set(id, metrics.release()); 174 return *canon->fTypefaceMetrics.set(id, metrics.release());
209 } 175 }
210 176
211 SkAdvancedTypefaceMetrics::FontType font_type(const SkAdvancedTypefaceMetrics& m etrics) { 177 SkAdvancedTypefaceMetrics::FontType font_type(const SkAdvancedTypefaceMetrics& m etrics) {
212 if (SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kMultiMaster_FontFl ag)) { 178 if (SkToBool(metrics.fFlags & SkAdvancedTypefaceMetrics::kMultiMaster_FontFl ag)) {
213 // force Type3 fallback. 179 // force Type3 fallback.
214 return SkAdvancedTypefaceMetrics::kOther_Font; 180 return SkAdvancedTypefaceMetrics::kOther_Font;
215 } 181 }
216 return metrics.fType; 182 return metrics.fType;
217 } 183 }
218 184
219 static SkGlyphID first_glyph_for_single_byte_encoding(SkGlyphID gid) { 185 static SkGlyphID first_nonzero_glyph_for_single_byte_encoding(SkGlyphID gid) {
220 return gid != 0 ? gid - (gid - 1) % 255 : 1; 186 return gid != 0 ? gid - (gid - 1) % 255 : 1;
221 } 187 }
222 188
223 SkPDFFont* SkPDFFont::GetFontResource(SkPDFCanon* canon, 189 SkPDFFont* SkPDFFont::GetFontResource(SkPDFCanon* canon,
224 SkTypeface* face, 190 SkTypeface* face,
225 SkGlyphID glyphID) { 191 SkGlyphID glyphID) {
226 SkASSERT(canon); 192 SkASSERT(canon);
193 SkASSERT(face); // All SkPDFDevice::internalDrawText ensures this.
227 const SkAdvancedTypefaceMetrics* fontMetrics = SkPDFFont::GetMetrics(face, c anon); 194 const SkAdvancedTypefaceMetrics* fontMetrics = SkPDFFont::GetMetrics(face, c anon);
228 if (!fontMetrics) { 195 SkASSERT(fontMetrics); // SkPDFDevice::internalDrawText ensures the typefac e is good.
229 return nullptr; // bad font, return early. 196 // GetMetrics only returns null to signify a bad typ eface.
230 }
231 const SkAdvancedTypefaceMetrics& metrics = *fontMetrics; 197 const SkAdvancedTypefaceMetrics& metrics = *fontMetrics;
232 SkAdvancedTypefaceMetrics::FontType type = font_type(metrics); 198 SkAdvancedTypefaceMetrics::FontType type = font_type(metrics);
233 bool multibyte = SkPDFFont::IsMultiByte(type); 199 bool multibyte = SkPDFFont::IsMultiByte(type);
234 SkGlyphID firstGlyph = multibyte ? 0 : first_glyph_for_single_byte_encoding( glyphID); 200 SkGlyphID subsetCode = multibyte ? 0 : first_nonzero_glyph_for_single_byte_e ncoding(glyphID);
235 uint64_t fontID = (SkTypeface::UniqueID(face) << 16) | firstGlyph; 201 uint64_t fontID = (SkTypeface::UniqueID(face) << 16) | subsetCode;
236 202
237 if (SkPDFFont** found = canon->fFontMap.find(fontID)) { 203 if (SkPDFFont** found = canon->fFontMap.find(fontID)) {
238 SkASSERT(multibyte == (*found)->multiByteGlyphs()); 204 SkPDFFont* foundFont = *found;
239 return SkRef(*found); 205 SkASSERT(foundFont && multibyte == foundFont->multiByteGlyphs());
206 return SkRef(foundFont);
240 } 207 }
241 208
242 sk_sp<SkTypeface> typeface(face ? sk_ref_sp(face) : SkTypeface::MakeDefault( )); 209 sk_sp<SkTypeface> typeface(sk_ref_sp(face));
243 SkASSERT(typeface); 210 SkASSERT(typeface);
244 int glyphCount = typeface->countGlyphs(); 211
245 // Validate typeface + glyph; 212 SkGlyphID lastGlyph = metrics.fLastGlyphID;
246 if (glyphCount < 1 || // typeface lacks even a NOTDEF glyph. 213 SkASSERT(typeface->countGlyphs() == SkToInt(1 + metrics.fLastGlyphID));
247 glyphCount > 1 + SK_MaxU16 || // invalid glyphCount 214
248 glyphID >= glyphCount) { // invalid glyph 215 // should be caught by SkPDFDevice::internalDrawText
249 return nullptr; 216 SkASSERT(glyphID <= lastGlyph);
250 }
251 217
252 SkGlyphID firstNonZeroGlyph; 218 SkGlyphID firstNonZeroGlyph;
253 SkGlyphID lastGlyph;
254 if (multibyte) { 219 if (multibyte) {
255 firstNonZeroGlyph = 1; 220 firstNonZeroGlyph = 1;
256 lastGlyph = SkToU16(glyphCount - 1);
257 } else { 221 } else {
258 firstNonZeroGlyph = firstGlyph; 222 firstNonZeroGlyph = subsetCode;
259 lastGlyph = SkToU16(SkTMin<int>(glyphCount - 1, 223 lastGlyph = SkToU16(SkTMin<int>((int)lastGlyph, 254 + (int)subsetCode));
260 254 + (int)firstGlyph));
261 } 224 }
262 SkPDFFont::Info info = {std::move(typeface), firstNonZeroGlyph, lastGlyph, t ype}; 225 SkPDFFont::Info info = {std::move(typeface), firstNonZeroGlyph, lastGlyph, t ype};
263 sk_sp<SkPDFFont> font; 226 sk_sp<SkPDFFont> font;
264 switch (type) { 227 switch (type) {
265 case SkAdvancedTypefaceMetrics::kType1CID_Font: 228 case SkAdvancedTypefaceMetrics::kType1CID_Font:
266 case SkAdvancedTypefaceMetrics::kTrueType_Font: 229 case SkAdvancedTypefaceMetrics::kTrueType_Font:
267 SkASSERT(multibyte); 230 SkASSERT(multibyte);
268 font = sk_make_sp<SkPDFType0Font>(std::move(info), metrics); 231 font = sk_make_sp<SkPDFType0Font>(std::move(info), metrics);
269 break; 232 break;
270 case SkAdvancedTypefaceMetrics::kType1_Font: 233 case SkAdvancedTypefaceMetrics::kType1_Font:
(...skipping 484 matching lines...) Expand 10 before | Expand all | Expand 10 after
755 718
756 bool SkPDFFont::CanEmbedTypeface(SkTypeface* typeface, SkPDFCanon* canon) { 719 bool SkPDFFont::CanEmbedTypeface(SkTypeface* typeface, SkPDFCanon* canon) {
757 const SkAdvancedTypefaceMetrics* metrics = SkPDFFont::GetMetrics(typeface, c anon); 720 const SkAdvancedTypefaceMetrics* metrics = SkPDFFont::GetMetrics(typeface, c anon);
758 return metrics && can_embed(*metrics); 721 return metrics && can_embed(*metrics);
759 } 722 }
760 723
761 void SkPDFFont::drop() { 724 void SkPDFFont::drop() {
762 fTypeface = nullptr; 725 fTypeface = nullptr;
763 this->SkPDFDict::drop(); 726 this->SkPDFDict::drop();
764 } 727 }
OLDNEW
« src/pdf/SkPDFDevice.cpp ('K') | « src/pdf/SkPDFFont.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698