Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 /* | |
| 2 * Copyright 2015 Google Inc. | |
| 3 * | |
| 4 * Use of this source code is governed by a BSD-style license that can be | |
| 5 * found in the LICENSE file. | |
| 6 */ | |
| 7 | |
| 8 #include "SkChecksum.h" | |
| 9 #include "SkFontDescriptor.h" | |
| 10 #include "SkStream.h" | |
| 11 #include "SkString.h" | |
| 12 #include "SkTypeface.h" | |
| 13 #include "SkUtils.h" | |
| 14 | |
| 15 #include "SkWhitelistChecksums.cpp" | |
| 16 | |
| 17 #define WHITELIST_DEBUG 0 | |
| 18 | |
| 19 extern void WhitelistSerializeTypeface(const SkTypeface*, SkWStream* ); | |
| 20 extern SkTypeface* WhitelistDeserializeTypeface(SkStream* ); | |
| 21 extern bool CheckChecksums(); | |
| 22 extern bool GenerateChecksums(); | |
| 23 | |
| 24 #if WHITELIST_DEBUG | |
| 25 static bool timesNewRomanSerializedNameOnly = false; | |
| 26 #endif | |
| 27 | |
| 28 struct NameRecord { | |
| 29 unsigned short fPlatformID; | |
| 30 unsigned short fEncodingID; | |
| 31 unsigned short fLanguageID; | |
| 32 unsigned short fNameID; | |
| 33 unsigned short fLength; | |
| 34 unsigned short fOffset; | |
| 35 }; | |
| 36 | |
| 37 struct NameTable { | |
|
bungeman-skia
2015/08/31 15:59:32
Why not SkOTTableName?
| |
| 38 unsigned short fFormat; | |
| 39 unsigned short fCount; | |
| 40 unsigned short fStringOffset; | |
| 41 NameRecord fRecord[1]; | |
| 42 }; | |
| 43 | |
| 44 #define SUBNAME_PREFIX "sk_" | |
| 45 | |
| 46 static unsigned short swizzle(unsigned short x) { | |
| 47 return x << 8 | (x >> 8 & 0xff); | |
| 48 } | |
| 49 | |
| 50 static bool font_name_is_local(const char* fontName, SkTypeface::Style style) { | |
| 51 if (!strcmp(fontName, "DejaVu Sans")) { | |
| 52 return true; | |
| 53 } | |
| 54 SkTypeface* defaultFace = SkTypeface::CreateFromName(nullptr, style); | |
| 55 SkTypeface* foundFace = SkTypeface::CreateFromName(fontName, style); | |
| 56 return defaultFace != foundFace; | |
| 57 } | |
| 58 | |
| 59 static int name_table(const NameTable* nameTable, int tableIndex, const char** s tringLocPtr) { | |
| 60 int nameTableCount = swizzle(nameTable->fCount); | |
| 61 for (int i = 0; i < nameTableCount; ++i) { | |
| 62 const NameRecord* nameRecord = &nameTable->fRecord[i]; | |
| 63 int recordNameID = swizzle(nameRecord->fNameID); | |
| 64 if (recordNameID != tableIndex) { | |
| 65 continue; | |
| 66 } | |
| 67 int stringLen = swizzle(nameRecord->fLength); | |
| 68 if (!stringLen) { | |
| 69 break; | |
| 70 } | |
| 71 int recordOffset = swizzle(nameRecord->fOffset); | |
| 72 const char* stringLoc = (const char* ) nameTable + swizzle(nameTable->fS tringOffset); | |
| 73 stringLoc += recordOffset; | |
| 74 *stringLocPtr = stringLoc; | |
| 75 return stringLen; | |
| 76 } | |
| 77 return -1; | |
| 78 } | |
| 79 | |
| 80 static int whitelist_name_index(const SkTypeface* tf) { | |
| 81 static const SkFontTableTag nameTag = SkSetFourByteTag('n', 'a', 'm', 'e'); | |
| 82 size_t nameSize = tf->getTableSize(nameTag); | |
| 83 if (!nameSize) { | |
| 84 return -1; | |
| 85 } | |
| 86 SkTDArray<char> name; | |
| 87 name.setCount((int) nameSize); | |
| 88 tf->getTableData(nameTag, 0, nameSize, name.begin()); | |
| 89 const NameTable* nameTable = (const NameTable* ) name.begin(); | |
| 90 const char* stringLoc; | |
| 91 int stringLen = name_table(nameTable, 1, &stringLoc); | |
|
bungeman-skia
2015/08/31 15:59:32
Any reason not to use SkOTTableName::Iterator or L
caryclark
2015/09/01 12:17:37
I'll switch over to LocalizedStrings_NameTable in
| |
| 92 if (stringLen < 0) { | |
| 93 stringLen = name_table(nameTable, 16, &stringLoc); | |
| 94 } | |
| 95 if (stringLen < 0) { | |
| 96 stringLen = name_table(nameTable, 21, &stringLoc); | |
| 97 } | |
| 98 if (stringLen < 0) { | |
| 99 return -1; | |
| 100 } | |
| 101 SkString fontNameStr; | |
| 102 if (!*stringLoc) { | |
| 103 stringLen /= 2; | |
| 104 for (int i = 0; i < stringLen; ++i) { | |
| 105 fontNameStr.appendUnichar(swizzle(((const uint16_t*) stringLoc)[i])) ; | |
| 106 } | |
| 107 } else { | |
| 108 fontNameStr.resize(stringLen); | |
| 109 strncpy(fontNameStr.writable_str(), stringLoc, stringLen); | |
| 110 } | |
| 111 // check against permissible list of names | |
| 112 for (int i = 0; i < whitelistCount; ++i) { | |
| 113 if (fontNameStr.equals(whitelist[i].fFontName)) { | |
| 114 return i; | |
| 115 } | |
| 116 } | |
| 117 for (int i = 0; i < whitelistCount; ++i) { | |
| 118 if (fontNameStr.startsWith(whitelist[i].fFontName)) { | |
| 119 #if WHITELIST_DEBUG | |
| 120 SkDebugf("partial match whitelist=\"%s\" fontName=\"%s\"\n", whiteli st[i].fFontName, | |
| 121 fontNameStr.c_str()); | |
| 122 #endif | |
| 123 return -1; | |
| 124 } | |
| 125 } | |
| 126 #if WHITELIST_DEBUG | |
| 127 SkDebugf("no match fontName=\"%s\"\n", fontNameStr.c_str()); | |
| 128 #endif | |
| 129 return -1; | |
| 130 } | |
| 131 | |
| 132 static uint32_t compute_checksum(const SkTypeface* tf) { | |
| 133 SkFontData* fontData = tf->createFontData(); | |
| 134 if (!fontData) { | |
| 135 return 0; | |
| 136 } | |
| 137 SkStreamAsset* fontStream = fontData->getStream(); | |
| 138 if (!fontStream) { | |
| 139 return 0; | |
| 140 } | |
| 141 SkTDArray<char> data; | |
| 142 size_t length = fontStream->getLength(); | |
| 143 if (!length) { | |
| 144 return 0; | |
| 145 } | |
| 146 data.setCount((int) length); | |
| 147 if (!fontStream->peek(data.begin(), length)) { | |
| 148 return 0; | |
| 149 } | |
| 150 return SkChecksum::Murmur3(data.begin(), length); | |
| 151 } | |
| 152 | |
| 153 static void serialize_sub(const char* fontName, SkTypeface::Style style, SkWStre am* wstream) { | |
| 154 SkFontDescriptor desc(style); | |
| 155 SkString subName(SUBNAME_PREFIX); | |
| 156 subName.append(fontName); | |
| 157 const char* familyName = subName.c_str(); | |
| 158 desc.setFamilyName(familyName); | |
| 159 desc.serialize(wstream); | |
| 160 #if WHITELIST_DEBUG | |
| 161 for (int i = 0; i < whitelistCount; ++i) { | |
| 162 if (!strcmp(fontName, whitelist[i].fFontName)) { | |
| 163 if (!whitelist[i].fSerializedSub) { | |
| 164 whitelist[i].fSerializedSub = true; | |
| 165 SkDebugf("%s %s\n", __FUNCTION__, familyName); | |
| 166 } | |
| 167 break; | |
| 168 } | |
| 169 } | |
| 170 #endif | |
| 171 } | |
| 172 | |
| 173 static bool is_local(const SkTypeface* tf) { | |
| 174 bool isLocal = false; | |
| 175 SkFontDescriptor desc(tf->style()); | |
| 176 tf->getFontDescriptor(&desc, &isLocal); | |
| 177 return isLocal; | |
| 178 } | |
| 179 | |
| 180 static void serialize_full(const SkTypeface* tf, SkWStream* wstream) { | |
| 181 bool isLocal = false; | |
| 182 SkFontDescriptor desc(tf->style()); | |
| 183 tf->getFontDescriptor(&desc, &isLocal); | |
| 184 | |
| 185 // Embed font data if it's a local font. | |
| 186 if (isLocal && !desc.hasFontData()) { | |
| 187 desc.setFontData(tf->createFontData()); | |
| 188 } | |
| 189 desc.serialize(wstream); | |
| 190 } | |
| 191 | |
| 192 static void serialize_name_only(const SkTypeface* tf, SkWStream* wstream) { | |
| 193 bool isLocal = false; | |
| 194 SkFontDescriptor desc(tf->style()); | |
| 195 tf->getFontDescriptor(&desc, &isLocal); | |
| 196 SkASSERT(!isLocal); | |
| 197 #if WHITELIST_DEBUG | |
| 198 const char* familyName = desc.getFamilyName(); | |
| 199 if (familyName) { | |
| 200 if (!strcmp(familyName, "Times New Roman")) { | |
| 201 if (!timesNewRomanSerializedNameOnly) { | |
| 202 timesNewRomanSerializedNameOnly = true; | |
| 203 SkDebugf("%s %s\n", __FUNCTION__, familyName); | |
| 204 } | |
| 205 } else { | |
| 206 for (int i = 0; i < whitelistCount; ++i) { | |
| 207 if (!strcmp(familyName, whitelist[i].fFontName)) { | |
| 208 if (!whitelist[i].fSerializedNameOnly) { | |
| 209 whitelist[i].fSerializedNameOnly = true; | |
| 210 SkDebugf("%s %s\n", __FUNCTION__, familyName); | |
| 211 } | |
| 212 break; | |
| 213 } | |
| 214 } | |
| 215 } | |
| 216 } | |
| 217 #endif | |
| 218 desc.serialize(wstream); | |
| 219 } | |
| 220 | |
| 221 void WhitelistSerializeTypeface(const SkTypeface* tf, SkWStream* wstream) { | |
| 222 if (!is_local(tf)) { | |
| 223 serialize_name_only(tf, wstream); | |
| 224 return; | |
| 225 } | |
| 226 int whitelistIndex = whitelist_name_index(tf); | |
| 227 if (whitelistIndex < 0) { | |
| 228 serialize_full(tf, wstream); | |
| 229 return; | |
| 230 } | |
| 231 const char* fontName = whitelist[whitelistIndex].fFontName; | |
| 232 if (!font_name_is_local(fontName, tf->style())) { | |
| 233 #if WHITELIST_DEBUG | |
| 234 SkDebugf("name not found locally \"%s\" style=%d\n", fontName, tf->style ()); | |
| 235 #endif | |
| 236 serialize_full(tf, wstream); | |
| 237 return; | |
| 238 } | |
| 239 uint32_t checksum = compute_checksum(tf); | |
| 240 if (whitelist[whitelistIndex].fChecksum != checksum) { | |
| 241 #if WHITELIST_DEBUG | |
| 242 if (whitelist[whitelistIndex].fChecksum) { | |
| 243 SkDebugf("!!! checksum changed !!!\n"); | |
| 244 } | |
| 245 SkDebugf("checksum updated\n"); | |
| 246 SkDebugf(" { \"%s\", 0x%08x },\n", fontName, checksum); | |
| 247 #endif | |
| 248 whitelist[whitelistIndex].fChecksum = checksum; | |
| 249 } | |
| 250 serialize_sub(fontName, tf->style(), wstream); | |
| 251 } | |
| 252 | |
| 253 SkTypeface* WhitelistDeserializeTypeface(SkStream* stream) { | |
| 254 SkFontDescriptor desc(stream); | |
| 255 SkFontData* data = desc.detachFontData(); | |
| 256 if (data) { | |
| 257 SkTypeface* typeface = SkTypeface::CreateFromFontData(data); | |
| 258 if (typeface) { | |
| 259 return typeface; | |
| 260 } | |
| 261 } | |
| 262 const char* familyName = desc.getFamilyName(); | |
| 263 if (!strncmp(SUBNAME_PREFIX, familyName, sizeof(SUBNAME_PREFIX) - 1)) { | |
| 264 familyName += sizeof(SUBNAME_PREFIX) - 1; | |
| 265 } | |
| 266 return SkTypeface::CreateFromName(desc.getFamilyName(), desc.getStyle()); | |
| 267 } | |
| 268 | |
| 269 bool CheckChecksums() { | |
| 270 for (int i = 0; i < whitelistCount; ++i) { | |
| 271 const char* fontName = whitelist[i].fFontName; | |
| 272 SkTypeface* tf = SkTypeface::CreateFromName(fontName, SkTypeface::kNorma l); | |
| 273 uint32_t checksum = compute_checksum(tf); | |
| 274 if (whitelist[i].fChecksum != checksum) { | |
| 275 return false; | |
| 276 } | |
| 277 } | |
| 278 return true; | |
| 279 } | |
| 280 | |
| 281 const char checksumFileName[] = "SkWhitelistChecksums.cpp"; | |
| 282 | |
| 283 const char checksumHeader[] = | |
| 284 "/*" "\n" | |
| 285 " * Copyright 2015 Google Inc." "\n" | |
| 286 " *" "\n" | |
| 287 " * Use of this source code is governed by a BSD-style license that can be" "\n" | |
| 288 " * found in the LICENSE file." "\n" | |
| 289 " *" "\n" | |
| 290 " * %s() in %s generated %s." "\n" | |
| 291 " * Run 'whitelist_typefaces --generate' to create anew." "\n" | |
| 292 " */" "\n" | |
| 293 "" "\n" | |
| 294 "#include \"SkTDArray.h\"" "\n" | |
| 295 "" "\n" | |
| 296 "struct Whitelist {" "\n" | |
| 297 " const char* fFontName;" "\n" | |
| 298 " uint32_t fChecksum;" "\n" | |
| 299 " bool fSerializedNameOnly;" "\n" | |
| 300 " bool fSerializedSub;" "\n" | |
| 301 "};" "\n" | |
| 302 "" "\n" | |
| 303 "static Whitelist whitelist[] = {" "\n" ; | |
| 304 | |
| 305 const char checksumEntry[] = | |
| 306 " { \"%s\", 0x%08x, false, false }," "\n" ; | |
| 307 | |
| 308 const char checksumTrailer[] = | |
| 309 "};" "\n" | |
| 310 "" "\n" | |
| 311 "static const int whitelistCount = (int) SK_ARRAY_COUNT(whitelist);" "\n" ; | |
| 312 | |
| 313 | |
| 314 #include "SkOSFile.h" | |
| 315 | |
| 316 bool GenerateChecksums() { | |
| 317 SkFILE* file = sk_fopen(checksumFileName, kWrite_SkFILE_Flag); | |
| 318 if (!file) { | |
| 319 SkDebugf("Can't open %s for writing.\n", checksumFileName); | |
| 320 return false; | |
| 321 } | |
| 322 SkString line; | |
| 323 line.printf(checksumHeader, __FUNCTION__, __FILE__, checksumFileName); | |
| 324 sk_fwrite(line.c_str(), line.size(), file); | |
| 325 for (int i = 0; i < whitelistCount; ++i) { | |
| 326 const char* fontName = whitelist[i].fFontName; | |
| 327 SkTypeface* tf = SkTypeface::CreateFromName(fontName, SkTypeface::kNorma l); | |
| 328 uint32_t checksum = compute_checksum(tf); | |
| 329 line.printf(checksumEntry, fontName, checksum); | |
| 330 sk_fwrite(line.c_str(), line.size(), file); | |
| 331 } | |
| 332 sk_fwrite(checksumTrailer, sizeof(checksumTrailer) - 1, file); | |
| 333 sk_fclose(file); | |
| 334 return true; | |
| 335 } | |
| OLD | NEW |