OLD | NEW |
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 <ctype.h> | 8 #include <ctype.h> |
9 | 9 |
10 #include "SkData.h" | 10 #include "SkData.h" |
(...skipping 132 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
143 } else if (c <= '9') { | 143 } else if (c <= '9') { |
144 return c - '0'; | 144 return c - '0'; |
145 } else if (c <= 'F') { | 145 } else if (c <= 'F') { |
146 return c - 'A' + 10; | 146 return c - 'A' + 10; |
147 } else if (c <= 'f') { | 147 } else if (c <= 'f') { |
148 return c - 'a' + 10; | 148 return c - 'a' + 10; |
149 } | 149 } |
150 return -1; | 150 return -1; |
151 } | 151 } |
152 | 152 |
153 SkStream* handleType1Stream(SkStream* srcStream, size_t* headerLen, | 153 static SkData* handle_type1_stream(SkStream* srcStream, size_t* headerLen, |
154 size_t* dataLen, size_t* trailerLen) { | 154 size_t* dataLen, size_t* trailerLen) { |
155 // srcStream may be backed by a file or a unseekable fd, so we may not be | 155 // srcStream may be backed by a file or a unseekable fd, so we may not be |
156 // able to use skip(), rewind(), or getMemoryBase(). read()ing through | 156 // able to use skip(), rewind(), or getMemoryBase(). read()ing through |
157 // the input only once is doable, but very ugly. Furthermore, it'd be nice | 157 // the input only once is doable, but very ugly. Furthermore, it'd be nice |
158 // if the data was NUL terminated so that we can use strstr() to search it. | 158 // if the data was NUL terminated so that we can use strstr() to search it. |
159 // Make as few copies as possible given these constraints. | 159 // Make as few copies as possible given these constraints. |
160 SkDynamicMemoryWStream dynamicStream; | 160 SkDynamicMemoryWStream dynamicStream; |
161 SkAutoTUnref<SkMemoryStream> staticStream; | 161 SkAutoTUnref<SkMemoryStream> staticStream; |
162 SkData* data = NULL; | 162 SkData* data = NULL; |
163 const uint8_t* src; | 163 const uint8_t* src; |
164 size_t srcLen; | 164 size_t srcLen; |
(...skipping 27 matching lines...) Expand all Loading... |
192 data = dynamicStream.copyToData(); | 192 data = dynamicStream.copyToData(); |
193 src = data->bytes(); | 193 src = data->bytes(); |
194 srcLen = data->size() - 1; | 194 srcLen = data->size() - 1; |
195 } | 195 } |
196 | 196 |
197 // this handles releasing the data we may have gotten from dynamicStream. | 197 // this handles releasing the data we may have gotten from dynamicStream. |
198 // if data is null, it is a no-op | 198 // if data is null, it is a no-op |
199 SkAutoDataUnref aud(data); | 199 SkAutoDataUnref aud(data); |
200 | 200 |
201 if (parsePFB(src, srcLen, headerLen, dataLen, trailerLen)) { | 201 if (parsePFB(src, srcLen, headerLen, dataLen, trailerLen)) { |
202 SkMemoryStream* result = | 202 static const int kPFBSectionHeaderLength = 6; |
203 new SkMemoryStream(*headerLen + *dataLen + *trailerLen); | 203 const size_t length = *headerLen + *dataLen + *trailerLen; |
204 memcpy((char*)result->getAtPos(), src + 6, *headerLen); | 204 SkASSERT(length > 0); |
205 result->seek(*headerLen); | 205 SkASSERT(length + (2 * kPFBSectionHeaderLength) <= srcLen); |
206 memcpy((char*)result->getAtPos(), src + 6 + *headerLen + 6, *dataLen); | 206 |
207 result->seek(*headerLen + *dataLen); | 207 SkAutoTMalloc<uint8_t> buffer(length); |
208 memcpy((char*)result->getAtPos(), src + 6 + *headerLen + 6 + *dataLen, | 208 |
209 *trailerLen); | 209 const uint8_t* const srcHeader = src + kPFBSectionHeaderLength; |
210 result->rewind(); | 210 // There is a six-byte section header before header and data |
211 return result; | 211 // (but not trailer) that we're not going to copy. |
| 212 const uint8_t* const srcData |
| 213 = srcHeader + *headerLen + kPFBSectionHeaderLength; |
| 214 const uint8_t* const srcTrailer = srcData + *headerLen; |
| 215 |
| 216 uint8_t* const resultHeader = buffer.get(); |
| 217 uint8_t* const resultData = resultHeader + *headerLen; |
| 218 uint8_t* const resultTrailer = resultData + *dataLen; |
| 219 |
| 220 SkASSERT(resultTrailer + *trailerLen == resultHeader + length); |
| 221 |
| 222 memcpy(resultHeader, srcHeader, *headerLen); |
| 223 memcpy(resultData, srcData, *dataLen); |
| 224 memcpy(resultTrailer, srcTrailer, *trailerLen); |
| 225 |
| 226 return SkData::NewFromMalloc(buffer.detach(), length); |
212 } | 227 } |
213 | 228 |
214 // A PFA has to be converted for PDF. | 229 // A PFA has to be converted for PDF. |
215 size_t hexDataLen; | 230 size_t hexDataLen; |
216 if (parsePFA((const char*)src, srcLen, headerLen, &hexDataLen, dataLen, | 231 if (parsePFA((const char*)src, srcLen, headerLen, &hexDataLen, dataLen, |
217 trailerLen)) { | 232 trailerLen)) { |
218 SkMemoryStream* result = | 233 const size_t length = *headerLen + *dataLen + *trailerLen; |
219 new SkMemoryStream(*headerLen + *dataLen + *trailerLen); | 234 SkASSERT(length > 0); |
220 memcpy((char*)result->getAtPos(), src, *headerLen); | 235 SkAutoTMalloc<uint8_t> buffer(length); |
221 result->seek(*headerLen); | 236 |
| 237 memcpy(buffer.get(), src, *headerLen); |
| 238 uint8_t* const resultData = &(buffer[*headerLen]); |
222 | 239 |
223 const uint8_t* hexData = src + *headerLen; | 240 const uint8_t* hexData = src + *headerLen; |
224 const uint8_t* trailer = hexData + hexDataLen; | 241 const uint8_t* trailer = hexData + hexDataLen; |
225 size_t outputOffset = 0; | 242 size_t outputOffset = 0; |
226 uint8_t dataByte = 0; // To hush compiler. | 243 uint8_t dataByte = 0; // To hush compiler. |
227 bool highNibble = true; | 244 bool highNibble = true; |
228 for (; hexData < trailer; hexData++) { | 245 for (; hexData < trailer; hexData++) { |
229 int8_t curNibble = hexToBin(*hexData); | 246 int8_t curNibble = hexToBin(*hexData); |
230 if (curNibble < 0) { | 247 if (curNibble < 0) { |
231 continue; | 248 continue; |
232 } | 249 } |
233 if (highNibble) { | 250 if (highNibble) { |
234 dataByte = curNibble << 4; | 251 dataByte = curNibble << 4; |
235 highNibble = false; | 252 highNibble = false; |
236 } else { | 253 } else { |
237 dataByte |= curNibble; | 254 dataByte |= curNibble; |
238 highNibble = true; | 255 highNibble = true; |
239 ((char *)result->getAtPos())[outputOffset++] = dataByte; | 256 resultData[outputOffset++] = dataByte; |
240 } | 257 } |
241 } | 258 } |
242 if (!highNibble) { | 259 if (!highNibble) { |
243 ((char *)result->getAtPos())[outputOffset++] = dataByte; | 260 resultData[outputOffset++] = dataByte; |
244 } | 261 } |
245 SkASSERT(outputOffset == *dataLen); | 262 SkASSERT(outputOffset == *dataLen); |
246 result->seek(*headerLen + outputOffset); | |
247 | 263 |
248 memcpy((char *)result->getAtPos(), src + *headerLen + hexDataLen, | 264 uint8_t* const resultTrailer = &(buffer[*headerLen + outputOffset]); |
249 *trailerLen); | 265 memcpy(resultTrailer, src + *headerLen + hexDataLen, *trailerLen); |
250 result->rewind(); | 266 |
251 return result; | 267 return SkData::NewFromMalloc(buffer.detach(), length); |
252 } | 268 } |
253 | |
254 return NULL; | 269 return NULL; |
255 } | 270 } |
256 | 271 |
257 // scale from em-units to base-1000, returning as a SkScalar | 272 // scale from em-units to base-1000, returning as a SkScalar |
258 SkScalar scaleFromFontUnits(int16_t val, uint16_t emSize) { | 273 SkScalar scaleFromFontUnits(int16_t val, uint16_t emSize) { |
259 SkScalar scaled = SkIntToScalar(val); | 274 SkScalar scaled = SkIntToScalar(val); |
260 if (emSize == 1000) { | 275 if (emSize == 1000) { |
261 return scaled; | 276 return scaled; |
262 } else { | 277 } else { |
263 return SkScalarMulDiv(scaled, 1000, emSize); | 278 return SkScalarMulDiv(scaled, 1000, emSize); |
(...skipping 285 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
549 uint16_t lastGlyphID) { | 564 uint16_t lastGlyphID) { |
550 SkDynamicMemoryWStream cmap; | 565 SkDynamicMemoryWStream cmap; |
551 if (multiByteGlyphs) { | 566 if (multiByteGlyphs) { |
552 append_tounicode_header(&cmap, firstGlyphID, lastGlyphID); | 567 append_tounicode_header(&cmap, firstGlyphID, lastGlyphID); |
553 } else { | 568 } else { |
554 append_tounicode_header(&cmap, 1, lastGlyphID - firstGlyphID + 1); | 569 append_tounicode_header(&cmap, 1, lastGlyphID - firstGlyphID + 1); |
555 } | 570 } |
556 append_cmap_sections(glyphToUnicode, subset, &cmap, multiByteGlyphs, | 571 append_cmap_sections(glyphToUnicode, subset, &cmap, multiByteGlyphs, |
557 firstGlyphID, lastGlyphID); | 572 firstGlyphID, lastGlyphID); |
558 append_cmap_footer(&cmap); | 573 append_cmap_footer(&cmap); |
559 SkAutoTUnref<SkMemoryStream> cmapStream(new SkMemoryStream()); | 574 SkAutoTUnref<SkData> cmapData(cmap.copyToData()); |
560 cmapStream->setData(cmap.copyToData())->unref(); | 575 return new SkPDFStream(cmapData.get()); |
561 return new SkPDFStream(cmapStream.get()); | |
562 } | 576 } |
563 | 577 |
564 #if defined (SK_SFNTLY_SUBSETTER) | 578 #if defined (SK_SFNTLY_SUBSETTER) |
565 static void sk_delete_array(const void* ptr, size_t, void*) { | 579 static void sk_delete_array(const void* ptr, size_t, void*) { |
566 // Use C-style cast to cast away const and cast type simultaneously. | 580 // Use C-style cast to cast away const and cast type simultaneously. |
567 delete[] (unsigned char*)ptr; | 581 delete[] (unsigned char*)ptr; |
568 } | 582 } |
569 #endif | 583 #endif |
570 | 584 |
571 static size_t get_subset_font_stream(const char* fontName, | 585 static size_t get_subset_font_stream(const char* fontName, |
572 const SkTypeface* typeface, | 586 const SkTypeface* typeface, |
573 const SkTDArray<uint32_t>& subset, | 587 const SkTDArray<uint32_t>& subset, |
574 SkPDFStream** fontStream) { | 588 SkPDFStream** fontStream) { |
575 int ttcIndex; | 589 int ttcIndex; |
576 SkAutoTUnref<SkStream> fontData(typeface->openStream(&ttcIndex)); | 590 SkAutoTUnref<SkStream> fontData(typeface->openStream(&ttcIndex)); |
| 591 SkASSERT(fontData.get()); |
577 | 592 |
578 size_t fontSize = fontData->getLength(); | 593 size_t fontSize = fontData->getLength(); |
579 | 594 |
580 #if defined (SK_SFNTLY_SUBSETTER) | 595 #if defined (SK_SFNTLY_SUBSETTER) |
581 // Read font into buffer. | 596 // Read font into buffer. |
582 SkPDFStream* subsetFontStream = NULL; | 597 SkPDFStream* subsetFontStream = NULL; |
583 SkTDArray<unsigned char> originalFont; | 598 SkTDArray<unsigned char> originalFont; |
584 originalFont.setCount(SkToInt(fontSize)); | 599 originalFont.setCount(SkToInt(fontSize)); |
585 if (fontData->read(originalFont.begin(), fontSize) == fontSize) { | 600 if (fontData->read(originalFont.begin(), fontSize) == fontSize) { |
586 unsigned char* subsetFont = NULL; | 601 unsigned char* subsetFont = NULL; |
(...skipping 701 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1288 } | 1303 } |
1289 | 1304 |
1290 SkAutoTUnref<SkPDFDict> descriptor(new SkPDFDict("FontDescriptor")); | 1305 SkAutoTUnref<SkPDFDict> descriptor(new SkPDFDict("FontDescriptor")); |
1291 setFontDescriptor(descriptor.get()); | 1306 setFontDescriptor(descriptor.get()); |
1292 | 1307 |
1293 int ttcIndex; | 1308 int ttcIndex; |
1294 size_t header SK_INIT_TO_AVOID_WARNING; | 1309 size_t header SK_INIT_TO_AVOID_WARNING; |
1295 size_t data SK_INIT_TO_AVOID_WARNING; | 1310 size_t data SK_INIT_TO_AVOID_WARNING; |
1296 size_t trailer SK_INIT_TO_AVOID_WARNING; | 1311 size_t trailer SK_INIT_TO_AVOID_WARNING; |
1297 SkAutoTUnref<SkStream> rawFontData(typeface()->openStream(&ttcIndex)); | 1312 SkAutoTUnref<SkStream> rawFontData(typeface()->openStream(&ttcIndex)); |
1298 SkStream* fontData = handleType1Stream(rawFontData.get(), &header, &data, | 1313 SkData* fontData = handle_type1_stream(rawFontData.get(), &header, &data, |
1299 &trailer); | 1314 &trailer); |
1300 if (fontData == NULL) { | 1315 if (fontData == NULL) { |
1301 return false; | 1316 return false; |
1302 } | 1317 } |
1303 if (canEmbed()) { | 1318 if (canEmbed()) { |
1304 SkAutoTUnref<SkPDFStream> fontStream(new SkPDFStream(fontData)); | 1319 SkAutoTUnref<SkPDFStream> fontStream(new SkPDFStream(fontData)); |
1305 addResource(fontStream.get()); | 1320 addResource(fontStream.get()); |
1306 fontStream->insertInt("Length1", header); | 1321 fontStream->insertInt("Length1", header); |
1307 fontStream->insertInt("Length2", data); | 1322 fontStream->insertInt("Length2", data); |
1308 fontStream->insertInt("Length3", trailer); | 1323 fontStream->insertInt("Length3", trailer); |
(...skipping 164 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1473 | 1488 |
1474 insert("FontBBox", makeFontBBox(bbox, 1000))->unref(); | 1489 insert("FontBBox", makeFontBBox(bbox, 1000))->unref(); |
1475 insertInt("FirstChar", 1); | 1490 insertInt("FirstChar", 1); |
1476 insertInt("LastChar", lastGlyphID() - firstGlyphID() + 1); | 1491 insertInt("LastChar", lastGlyphID() - firstGlyphID() + 1); |
1477 insert("Widths", widthArray.get()); | 1492 insert("Widths", widthArray.get()); |
1478 insertName("CIDToGIDMap", "Identity"); | 1493 insertName("CIDToGIDMap", "Identity"); |
1479 | 1494 |
1480 populateToUnicodeTable(NULL); | 1495 populateToUnicodeTable(NULL); |
1481 return true; | 1496 return true; |
1482 } | 1497 } |
OLD | NEW |