2 * Copyright 2010 The Android Open Source Project
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
12 #include "SkImageEncoder.h"
14 #include "SkPDFCatalog.h"
15 #include "SkPDFDevice.h"
16 #include "SkPDFStream.h"
17 #include "SkPDFTypes.h"
18 #include "SkReadBuffer.h"
24 class SkPDFTestDict : public SkPDFDict {
26 virtual void getResources(const SkTSet<SkPDFObject*>& knownResourceObjects,
27 SkTSet<SkPDFObject*>* newResourceObjects) {
28 for (int i = 0; i < fResources.count(); i++) {
29 newResourceObjects->add(fResources[i]);
34 void addResource(SkPDFObject* object) {
35 fResources.append(1, &object);
39 SkTDArray<SkPDFObject*> fResources;
42 #define DUMMY_TEXT "DCT compessed stream."
44 static SkData* encode_to_dct_data(size_t* pixelRefOffset, const SkBitmap& bitmap) {
46 return SkData::NewWithProc(DUMMY_TEXT, sizeof(DUMMY_TEXT) - 1, NULL, NULL);
49 static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
50 const void* buffer, size_t len) {
51 SkAutoDataUnref data(stream.copyToData());
52 if (offset + len > data->size()) {
55 return memcmp(data->bytes() + offset, buffer, len) == 0;
58 static bool stream_contains(const SkDynamicMemoryWStream& stream,
60 SkAutoDataUnref data(stream.copyToData());
61 int len = strlen(buffer); // our buffer does not have EOSs.
63 for (int offset = 0 ; offset < (int)data->size() - len; offset++) {
64 if (memcmp(data->bytes() + offset, buffer, len) == 0) {
72 static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
73 const char* expectedData, size_t expectedSize,
74 bool indirect, bool compression) {
75 SkPDFDocument::Flags docFlags = (SkPDFDocument::Flags) 0;
77 docFlags = SkTBitOr(docFlags, SkPDFDocument::kFavorSpeedOverSize_Flags);
79 SkPDFCatalog catalog(docFlags);
80 size_t directSize = obj->getOutputSize(&catalog, false);
81 REPORTER_ASSERT(reporter, directSize == expectedSize);
83 SkDynamicMemoryWStream buffer;
84 obj->emit(&buffer, &catalog, false);
85 REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
86 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedData,
91 static char header[] = "1 0 obj\n";
92 static size_t headerLen = strlen(header);
93 static char footer[] = "\nendobj\n";
94 static size_t footerLen = strlen(footer);
96 catalog.addObject(obj, false);
98 size_t indirectSize = obj->getOutputSize(&catalog, true);
99 REPORTER_ASSERT(reporter,
100 indirectSize == directSize + headerLen + footerLen);
103 obj->emit(&buffer, &catalog, true);
104 REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
105 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
106 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
108 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
113 static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
115 const char* expectedResult) {
116 CheckObjectOutput(reporter, obj, expectedResult,
117 strlen(expectedResult), true, false);
120 static void TestPDFStream(skiatest::Reporter* reporter) {
121 char streamBytes[] = "Test\nFoo\tBar";
122 SkAutoTUnref<SkMemoryStream> streamData(new SkMemoryStream(
123 streamBytes, strlen(streamBytes), true));
124 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
125 SimpleCheckObjectOutput(
126 reporter, stream.get(),
127 "<</Length 12\n>> stream\nTest\nFoo\tBar\nendstream");
128 stream->insert("Attribute", new SkPDFInt(42))->unref();
129 SimpleCheckObjectOutput(reporter, stream.get(),
130 "<</Length 12\n/Attribute 42\n>> stream\n"
131 "Test\nFoo\tBar\nendstream");
133 if (SkFlate::HaveFlate()) {
134 char streamBytes2[] = "This is a longer string, so that compression "
135 "can do something with it. With shorter strings, "
136 "the short circuit logic cuts in and we end up "
137 "with an uncompressed string.";
138 SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
139 strlen(streamBytes2)));
140 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
142 SkDynamicMemoryWStream compressedByteStream;
143 SkFlate::Deflate(streamData2.get(), &compressedByteStream);
144 SkAutoDataUnref compressedData(compressedByteStream.copyToData());
146 // Check first without compression.
147 SkDynamicMemoryWStream expectedResult1;
148 expectedResult1.writeText("<</Length 167\n>> stream\n");
149 expectedResult1.writeText(streamBytes2);
150 expectedResult1.writeText("\nendstream");
151 SkAutoDataUnref expectedResultData1(expectedResult1.copyToData());
152 CheckObjectOutput(reporter, stream.get(),
153 (const char*) expectedResultData1->data(),
154 expectedResultData1->size(), true, false);
156 // Then again with compression.
157 SkDynamicMemoryWStream expectedResult2;
158 expectedResult2.writeText("<</Filter /FlateDecode\n/Length 116\n"
160 expectedResult2.write(compressedData->data(), compressedData->size());
161 expectedResult2.writeText("\nendstream");
162 SkAutoDataUnref expectedResultData2(expectedResult2.copyToData());
163 CheckObjectOutput(reporter, stream.get(),
164 (const char*) expectedResultData2->data(),
165 expectedResultData2->size(), true, true);
169 static void TestCatalog(skiatest::Reporter* reporter) {
170 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
171 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
172 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
173 SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
175 SkAutoTUnref<SkPDFInt> int1Again(int1.get());
177 catalog.addObject(int1.get(), false);
178 catalog.addObject(int2.get(), false);
179 catalog.addObject(int3.get(), false);
181 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
182 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
183 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int3.get()) == 3);
185 SkDynamicMemoryWStream buffer;
186 catalog.emitObjectNumber(&buffer, int1.get());
187 catalog.emitObjectNumber(&buffer, int2.get());
188 catalog.emitObjectNumber(&buffer, int3.get());
189 catalog.emitObjectNumber(&buffer, int1Again.get());
190 char expectedResult[] = "1 02 03 01 0";
191 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
192 strlen(expectedResult)));
195 static void TestObjectRef(skiatest::Reporter* reporter) {
196 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
197 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
198 SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
200 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
201 catalog.addObject(int1.get(), false);
202 catalog.addObject(int2.get(), false);
203 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
204 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
206 char expectedResult[] = "2 0 R";
207 SkDynamicMemoryWStream buffer;
208 int2ref->emitObject(&buffer, &catalog, false);
209 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
210 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
211 buffer.getOffset()));
214 static void TestSubstitute(skiatest::Reporter* reporter) {
215 SkAutoTUnref<SkPDFTestDict> proxy(new SkPDFTestDict());
216 SkAutoTUnref<SkPDFTestDict> stub(new SkPDFTestDict());
217 SkAutoTUnref<SkPDFInt> int33(new SkPDFInt(33));
218 SkAutoTUnref<SkPDFDict> stubResource(new SkPDFDict());
219 SkAutoTUnref<SkPDFInt> int44(new SkPDFInt(44));
221 stub->insert("Value", int33.get());
222 stubResource->insert("InnerValue", int44.get());
223 stub->addResource(stubResource.get());
225 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
226 catalog.addObject(proxy.get(), false);
227 catalog.setSubstitute(proxy.get(), stub.get());
229 SkDynamicMemoryWStream buffer;
230 proxy->emit(&buffer, &catalog, false);
231 catalog.emitSubstituteResources(&buffer, false);
233 char objectResult[] = "2 0 obj\n<</Value 33\n>>\nendobj\n";
236 catalog.setFileOffset(proxy.get(), 0) == strlen(objectResult));
238 char expectedResult[] =
239 "<</Value 33\n>>1 0 obj\n<</InnerValue 44\n>>\nendobj\n";
240 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
241 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
242 buffer.getOffset()));
245 // Create a bitmap that would be very eficiently compressed in a ZIP.
246 static void setup_bitmap(SkBitmap* bitmap, int width, int height) {
247 bitmap->allocN32Pixels(width, height);
248 bitmap->eraseColor(SK_ColorWHITE);
251 static void TestImage(skiatest::Reporter* reporter, const SkBitmap& bitmap,
252 const char* expected, bool useDCTEncoder) {
253 SkISize pageSize = SkISize::Make(bitmap.width(), bitmap.height());
254 SkAutoTUnref<SkPDFDevice> dev(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
257 dev->setDCTEncoder(encode_to_dct_data);
261 c.drawBitmap(bitmap, 0, 0, NULL);
266 SkDynamicMemoryWStream stream;
267 doc.emitPDF(&stream);
269 REPORTER_ASSERT(reporter, stream_contains(stream, expected));
272 static void TestUncompressed(skiatest::Reporter* reporter) {
274 setup_bitmap(&bitmap, 1, 1);
275 TestImage(reporter, bitmap,
279 "/ColorSpace /DeviceRGB\n"
280 "/BitsPerComponent 8\n"
286 static void TestFlateDecode(skiatest::Reporter* reporter) {
287 if (!SkFlate::HaveFlate()) {
291 setup_bitmap(&bitmap, 10, 10);
292 TestImage(reporter, bitmap,
296 "/ColorSpace /DeviceRGB\n"
297 "/BitsPerComponent 8\n"
298 "/Filter /FlateDecode\n"
304 static void TestDCTDecode(skiatest::Reporter* reporter) {
306 setup_bitmap(&bitmap, 32, 32);
307 TestImage(reporter, bitmap,
311 "/ColorSpace /DeviceRGB\n"
312 "/BitsPerComponent 8\n"
313 "/Filter /DCTDecode\n"
314 "/ColorTransform 0\n"
320 static void TestImages(skiatest::Reporter* reporter) {
321 TestUncompressed(reporter);
322 TestFlateDecode(reporter);
323 TestDCTDecode(reporter);
326 // This test used to assert without the fix submitted for
327 // http://code.google.com/p/skia/issues/detail?id=1083.
328 // SKP files might have invalid glyph ids. This test ensures they are ignored,
329 // and there is no assert on input data in Debug mode.
330 static void test_issue1083() {
331 SkISize pageSize = SkISize::Make(100, 100);
332 SkAutoTUnref<SkPDFDevice> dev(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
336 paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
338 uint16_t glyphID = 65000;
339 c.drawText(&glyphID, 2, 0, 0, paint);
344 SkDynamicMemoryWStream stream;
345 doc.emitPDF(&stream);
348 DEF_TEST(PDFPrimitives, reporter) {
349 SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
350 SimpleCheckObjectOutput(reporter, int42.get(), "42");
352 SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
353 SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
355 SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
356 #if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
357 SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
359 SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
361 SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
362 SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
364 SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
365 SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
368 SkAutoTUnref<SkPDFString> stringSimple(
369 new SkPDFString("test ) string ( foo"));
370 SimpleCheckObjectOutput(reporter, stringSimple.get(),
371 "(test \\) string \\( foo)");
372 SkAutoTUnref<SkPDFString> stringComplex(
373 new SkPDFString("\ttest ) string ( foo"));
374 SimpleCheckObjectOutput(reporter, stringComplex.get(),
375 "<0974657374202920737472696E67202820666F6F>");
377 SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
378 const char expectedResult[] = "/Test#20name#09with#23tab";
379 CheckObjectOutput(reporter, name.get(), expectedResult,
380 strlen(expectedResult), false, false);
382 SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
383 const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
384 CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
385 strlen(escapedNameExpected), false, false);
387 // Test that we correctly handle characters with the high-bit set.
388 const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
389 SkAutoTUnref<SkPDFName> highBitName(
390 new SkPDFName((const char*)highBitCString));
391 const char highBitExpectedResult[] = "/#DE#ADbe#EF";
392 CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
393 strlen(highBitExpectedResult), false, false);
395 SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
396 SimpleCheckObjectOutput(reporter, array.get(), "[]");
397 array->append(int42.get());
398 SimpleCheckObjectOutput(reporter, array.get(), "[42]");
399 array->append(realHalf.get());
400 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
401 SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
402 array->append(int0.get());
403 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
404 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
405 array->setAt(0, int1.get());
406 SimpleCheckObjectOutput(reporter, array.get(), "[1 0.5 0]");
408 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
409 SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
410 SkAutoTUnref<SkPDFName> n1(new SkPDFName("n1"));
411 dict->insert(n1.get(), int42.get());
412 SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42\n>>");
413 SkAutoTUnref<SkPDFName> n2(new SkPDFName("n2"));
414 SkAutoTUnref<SkPDFName> n3(new SkPDFName("n3"));
415 dict->insert(n2.get(), realHalf.get());
416 dict->insert(n3.get(), array.get());
417 SimpleCheckObjectOutput(reporter, dict.get(),
418 "<</n1 42\n/n2 0.5\n/n3 [1 0.5 0]\n>>");
420 TestPDFStream(reporter);
422 TestCatalog(reporter);
424 TestObjectRef(reporter);
426 TestSubstitute(reporter);
430 TestImages(reporter);
435 class DummyImageFilter : public SkImageFilter {
437 DummyImageFilter(bool visited = false) : SkImageFilter(0, NULL), fVisited(visited) {}
438 virtual ~DummyImageFilter() SK_OVERRIDE {}
439 virtual bool onFilterImage(Proxy*, const SkBitmap& src, const Context&,
440 SkBitmap* result, SkIPoint* offset) const {
442 offset->fX = offset->fY = 0;
446 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(DummyImageFilter)
447 #ifdef SK_SUPPORT_LEGACY_DEEPFLATTENING
448 explicit DummyImageFilter(SkReadBuffer& buffer) : SkImageFilter(0, NULL) {
449 fVisited = buffer.readBool();
452 bool visited() const { return fVisited; }
455 mutable bool fVisited;
458 SkFlattenable* DummyImageFilter::CreateProc(SkReadBuffer& buffer) {
459 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 0);
460 bool visited = buffer.readBool();
461 return SkNEW_ARGS(DummyImageFilter, (visited));
466 // Check that PDF rendering of image filters successfully falls back to
467 // CPU rasterization.
468 DEF_TEST(PDFImageFilter, reporter) {
469 SkISize pageSize = SkISize::Make(100, 100);
470 SkAutoTUnref<SkPDFDevice> device(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
471 SkCanvas canvas(device.get());
472 SkAutoTUnref<DummyImageFilter> filter(new DummyImageFilter());
474 // Filter just created; should be unvisited.
475 REPORTER_ASSERT(reporter, !filter->visited());
477 paint.setImageFilter(filter.get());
478 canvas.drawRect(SkRect::MakeWH(100, 100), paint);
480 // Filter was used in rendering; should be visited.
481 REPORTER_ASSERT(reporter, filter->visited());