blob: 7f92e43631421a9a33e368df276bf9719d7bd187 [file] [log] [blame]
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2010 The Android Open Source Project
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00003 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00004 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00006 */
7
edisonn@google.comd9dfa182013-04-24 13:01:01 +00008#include "SkBitmap.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +00009#include "SkCanvas.h"
reed@google.com8a85d0c2011-06-24 19:12:12 +000010#include "SkData.h"
halcanary3d32d502015-03-01 06:55:20 -080011#include "SkDocument.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000012#include "SkFlate.h"
edisonn@google.comd9dfa182013-04-24 13:01:01 +000013#include "SkImageEncoder.h"
14#include "SkMatrix.h"
halcanarya1f1ee92015-02-20 06:17:26 -080015#include "SkPDFCanon.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +000016#include "SkPDFDevice.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000017#include "SkPDFStream.h"
18#include "SkPDFTypes.h"
senorblancob0e89dc2014-10-20 14:03:12 -070019#include "SkReadBuffer.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000020#include "SkScalar.h"
21#include "SkStream.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000022#include "SkTypes.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000023#include "Test.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000024
commit-bot@chromium.org608ea652013-10-03 19:29:21 +000025#define DUMMY_TEXT "DCT compessed stream."
26
reed@google.com8a85d0c2011-06-24 19:12:12 +000027static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
28 const void* buffer, size_t len) {
29 SkAutoDataUnref data(stream.copyToData());
robertphillips@google.com59f46b82012-07-10 17:30:58 +000030 if (offset + len > data->size()) {
reed@google.com8a85d0c2011-06-24 19:12:12 +000031 return false;
32 }
robertphillips@google.com59f46b82012-07-10 17:30:58 +000033 return memcmp(data->bytes() + offset, buffer, len) == 0;
reed@google.com8a85d0c2011-06-24 19:12:12 +000034}
35
halcanary6a144342015-01-23 11:45:10 -080036static void emit_object(SkPDFObject* object,
37 SkWStream* stream,
halcanary37c46ca2015-03-31 12:30:20 -070038 const SkPDFObjNumMap& objNumMap,
39 const SkPDFSubstituteMap& substitutes,
halcanary6a144342015-01-23 11:45:10 -080040 bool indirect) {
halcanary37c46ca2015-03-31 12:30:20 -070041 SkPDFObject* realObject = substitutes.getSubstitute(object);
halcanary6a144342015-01-23 11:45:10 -080042 if (indirect) {
halcanary37c46ca2015-03-31 12:30:20 -070043 stream->writeDecAsText(objNumMap.getObjectNumber(realObject));
halcanarybf799cd2015-02-10 13:32:09 -080044 stream->writeText(" 0 obj\n"); // Generation number is always 0.
halcanary37c46ca2015-03-31 12:30:20 -070045 realObject->emitObject(stream, objNumMap, substitutes);
halcanary6a144342015-01-23 11:45:10 -080046 stream->writeText("\nendobj\n");
47 } else {
halcanary37c46ca2015-03-31 12:30:20 -070048 realObject->emitObject(stream, objNumMap, substitutes);
halcanary6a144342015-01-23 11:45:10 -080049 }
50}
51
52static size_t get_output_size(SkPDFObject* object,
halcanary37c46ca2015-03-31 12:30:20 -070053 const SkPDFObjNumMap& objNumMap,
54 const SkPDFSubstituteMap& substitutes,
halcanary6a144342015-01-23 11:45:10 -080055 bool indirect) {
56 SkDynamicMemoryWStream buffer;
halcanary37c46ca2015-03-31 12:30:20 -070057 emit_object(object, &buffer, objNumMap, substitutes, indirect);
halcanary6a144342015-01-23 11:45:10 -080058 return buffer.getOffset();
59}
60
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000061static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
vandebo@chromium.org421d6442011-07-20 17:39:01 +000062 const char* expectedData, size_t expectedSize,
mtkleincabc08c2015-02-19 08:29:24 -080063 bool indirect) {
halcanary37c46ca2015-03-31 12:30:20 -070064 SkPDFSubstituteMap substituteMap;
65 SkPDFObjNumMap catalog;
66 size_t directSize = get_output_size(obj, catalog, substituteMap, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000067 REPORTER_ASSERT(reporter, directSize == expectedSize);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000068
69 SkDynamicMemoryWStream buffer;
halcanary37c46ca2015-03-31 12:30:20 -070070 emit_object(obj, &buffer, catalog, substituteMap, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000071 REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
halcanary130444f2015-04-25 06:45:07 -070072 if (!stream_equals(buffer, 0, expectedData, directSize)) {
73 SkAutoTDelete<SkStreamAsset> asset(buffer.detachAsStream());
74 SkString s(asset->getLength());
75 asset->read(s.writable_str(), s.size());
76 ERRORF(reporter, "!stream_equals() '%s' '%s'", expectedData, s.c_str());
77 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000078
79 if (indirect) {
80 // Indirect output.
81 static char header[] = "1 0 obj\n";
82 static size_t headerLen = strlen(header);
83 static char footer[] = "\nendobj\n";
84 static size_t footerLen = strlen(footer);
85
halcanaryf41061c2015-03-25 11:29:18 -070086 catalog.addObject(obj);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000087
halcanary37c46ca2015-03-31 12:30:20 -070088 size_t indirectSize =
89 get_output_size(obj, catalog, substituteMap, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000090 REPORTER_ASSERT(reporter,
91 indirectSize == directSize + headerLen + footerLen);
92
93 buffer.reset();
halcanary37c46ca2015-03-31 12:30:20 -070094 emit_object(obj, &buffer, catalog, substituteMap, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000095 REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
reed@google.com8a85d0c2011-06-24 19:12:12 +000096 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
vandebo@chromium.org421d6442011-07-20 17:39:01 +000097 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +000098 directSize));
99 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
100 footer, footerLen));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000101 }
102}
103
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000104static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
105 SkPDFObject* obj,
robertphillips@google.com4debcac2012-05-14 16:33:36 +0000106 const char* expectedResult) {
107 CheckObjectOutput(reporter, obj, expectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800108 strlen(expectedResult), true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000109}
110
111static void TestPDFStream(skiatest::Reporter* reporter) {
112 char streamBytes[] = "Test\nFoo\tBar";
scroggoa1193e42015-01-21 12:09:53 -0800113 SkAutoTDelete<SkMemoryStream> streamData(new SkMemoryStream(
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000114 streamBytes, strlen(streamBytes), true));
115 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000116 SimpleCheckObjectOutput(
117 reporter, stream.get(),
halcanary37c46ca2015-03-31 12:30:20 -0700118 "<</Length 12>> stream\nTest\nFoo\tBar\nendstream");
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000119 stream->insert("Attribute", new SkPDFInt(42))->unref();
120 SimpleCheckObjectOutput(reporter, stream.get(),
halcanary37c46ca2015-03-31 12:30:20 -0700121 "<</Length 12\n/Attribute 42>> stream\n"
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000122 "Test\nFoo\tBar\nendstream");
123
halcanary91d1d622015-02-17 14:43:06 -0800124 {
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000125 char streamBytes2[] = "This is a longer string, so that compression "
126 "can do something with it. With shorter strings, "
127 "the short circuit logic cuts in and we end up "
128 "with an uncompressed string.";
129 SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
130 strlen(streamBytes2)));
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000131 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000132
133 SkDynamicMemoryWStream compressedByteStream;
134 SkFlate::Deflate(streamData2.get(), &compressedByteStream);
135 SkAutoDataUnref compressedData(compressedByteStream.copyToData());
136
mtkleincabc08c2015-02-19 08:29:24 -0800137 SkDynamicMemoryWStream expected;
halcanary37c46ca2015-03-31 12:30:20 -0700138 expected.writeText("<</Filter /FlateDecode\n/Length 116>> stream\n");
mtkleincabc08c2015-02-19 08:29:24 -0800139 expected.write(compressedData->data(), compressedData->size());
140 expected.writeText("\nendstream");
141 SkAutoDataUnref expectedResultData2(expected.copyToData());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000142 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000143 (const char*) expectedResultData2->data(),
mtkleincabc08c2015-02-19 08:29:24 -0800144 expectedResultData2->size(), true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000145 }
146}
147
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000148static void TestCatalog(skiatest::Reporter* reporter) {
halcanary37c46ca2015-03-31 12:30:20 -0700149 SkPDFSubstituteMap substituteMap;
150 SkPDFObjNumMap catalog;
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000151 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
152 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
153 SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
154 int1.get()->ref();
155 SkAutoTUnref<SkPDFInt> int1Again(int1.get());
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000156
halcanaryf41061c2015-03-25 11:29:18 -0700157 catalog.addObject(int1.get());
158 catalog.addObject(int2.get());
159 catalog.addObject(int3.get());
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000160
halcanarybf799cd2015-02-10 13:32:09 -0800161 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1.get()) == 1);
162 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int2.get()) == 2);
163 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int3.get()) == 3);
164 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1Again.get()) == 1);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000165}
166
167static void TestObjectRef(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000168 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
169 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
170 SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000171
halcanary37c46ca2015-03-31 12:30:20 -0700172 SkPDFSubstituteMap substituteMap;
173 SkPDFObjNumMap catalog;
halcanaryf41061c2015-03-25 11:29:18 -0700174 catalog.addObject(int1.get());
175 catalog.addObject(int2.get());
halcanarybf799cd2015-02-10 13:32:09 -0800176 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1.get()) == 1);
177 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int2.get()) == 2);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000178
179 char expectedResult[] = "2 0 R";
180 SkDynamicMemoryWStream buffer;
halcanary37c46ca2015-03-31 12:30:20 -0700181 int2ref->emitObject(&buffer, catalog, substituteMap);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000182 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
reed@google.com8a85d0c2011-06-24 19:12:12 +0000183 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
184 buffer.getOffset()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000185}
186
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000187static void TestSubstitute(skiatest::Reporter* reporter) {
halcanaryf3d53a02015-03-24 15:29:39 -0700188 SkAutoTUnref<SkPDFDict> proxy(new SkPDFDict());
189 SkAutoTUnref<SkPDFDict> stub(new SkPDFDict());
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000190
halcanarybf799cd2015-02-10 13:32:09 -0800191 proxy->insert("Value", new SkPDFInt(33))->unref();
192 stub->insert("Value", new SkPDFInt(44))->unref();
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000193
halcanary37c46ca2015-03-31 12:30:20 -0700194 SkPDFSubstituteMap substituteMap;
195 substituteMap.setSubstitute(proxy.get(), stub.get());
196 SkPDFObjNumMap catalog;
halcanaryf41061c2015-03-25 11:29:18 -0700197 catalog.addObject(proxy.get());
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000198
halcanary37c46ca2015-03-31 12:30:20 -0700199 REPORTER_ASSERT(reporter, stub.get() == substituteMap.getSubstitute(proxy));
200 REPORTER_ASSERT(reporter, proxy.get() != substituteMap.getSubstitute(stub));
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000201}
202
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000203// This test used to assert without the fix submitted for
204// http://code.google.com/p/skia/issues/detail?id=1083.
205// SKP files might have invalid glyph ids. This test ensures they are ignored,
206// and there is no assert on input data in Debug mode.
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000207static void test_issue1083() {
halcanary3d32d502015-03-01 06:55:20 -0800208 SkDynamicMemoryWStream outStream;
209 SkAutoTUnref<SkDocument> doc(SkDocument::CreatePDF(&outStream));
210 SkCanvas* canvas = doc->beginPage(100.0f, 100.0f);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000211 SkPaint paint;
212 paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
213
214 uint16_t glyphID = 65000;
halcanary3d32d502015-03-01 06:55:20 -0800215 canvas->drawText(&glyphID, 2, 0, 0, paint);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000216
halcanary3d32d502015-03-01 06:55:20 -0800217 doc->close();
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000218}
219
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000220DEF_TEST(PDFPrimitives, reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000221 SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000222 SimpleCheckObjectOutput(reporter, int42.get(), "42");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000223
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000224 SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000225 SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000226
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000227 SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000228#if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000229 SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000230#else
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000231 SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000232
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000233 SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000234 SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000235
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000236 SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000237 SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000238#endif
239
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000240 SkAutoTUnref<SkPDFString> stringSimple(
241 new SkPDFString("test ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000242 SimpleCheckObjectOutput(reporter, stringSimple.get(),
243 "(test \\) string \\( foo)");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000244 SkAutoTUnref<SkPDFString> stringComplex(
245 new SkPDFString("\ttest ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000246 SimpleCheckObjectOutput(reporter, stringComplex.get(),
247 "<0974657374202920737472696E67202820666F6F>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000248
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000249 SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000250 const char expectedResult[] = "/Test#20name#09with#23tab";
251 CheckObjectOutput(reporter, name.get(), expectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800252 strlen(expectedResult), false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000253
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000254 SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000255 const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
256 CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
mtkleincabc08c2015-02-19 08:29:24 -0800257 strlen(escapedNameExpected), false);
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000258
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000259 // Test that we correctly handle characters with the high-bit set.
bungeman@google.comf8aa18c2012-03-19 21:04:52 +0000260 const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000261 SkAutoTUnref<SkPDFName> highBitName(
262 new SkPDFName((const char*)highBitCString));
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000263 const char highBitExpectedResult[] = "/#DE#ADbe#EF";
264 CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800265 strlen(highBitExpectedResult), false);
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000266
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000267 SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000268 SimpleCheckObjectOutput(reporter, array.get(), "[]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000269 array->append(int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000270 SimpleCheckObjectOutput(reporter, array.get(), "[42]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000271 array->append(realHalf.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000272 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000273 SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000274 array->append(int0.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000275 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000276
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000277 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000278 SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
halcanary130444f2015-04-25 06:45:07 -0700279 dict->insert("n1", int42.get());
halcanary37c46ca2015-03-31 12:30:20 -0700280 SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42>>");
halcanary130444f2015-04-25 06:45:07 -0700281 SkString n3("n3");
282 dict->insert("n2", realHalf.get());
283 dict->insertObject(n3, array.detach());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000284 SimpleCheckObjectOutput(reporter, dict.get(),
halcanary2ad228e2015-04-08 08:40:36 -0700285 "<</n1 42\n/n2 0.5\n/n3 [42 0.5 0]>>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000286
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000287 TestPDFStream(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000288
289 TestCatalog(reporter);
290
291 TestObjectRef(reporter);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000292
293 TestSubstitute(reporter);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000294
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000295 test_issue1083();
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000296}
senorblancob0e89dc2014-10-20 14:03:12 -0700297
298namespace {
299
300class DummyImageFilter : public SkImageFilter {
301public:
302 DummyImageFilter(bool visited = false) : SkImageFilter(0, NULL), fVisited(visited) {}
mtklein36352bf2015-03-25 18:17:31 -0700303 ~DummyImageFilter() override {}
senorblancob0e89dc2014-10-20 14:03:12 -0700304 virtual bool onFilterImage(Proxy*, const SkBitmap& src, const Context&,
mtklein36352bf2015-03-25 18:17:31 -0700305 SkBitmap* result, SkIPoint* offset) const override {
senorblancob0e89dc2014-10-20 14:03:12 -0700306 fVisited = true;
307 offset->fX = offset->fY = 0;
308 *result = src;
309 return true;
310 }
robertphillipsf3f5bad2014-12-19 13:49:15 -0800311 SK_TO_STRING_OVERRIDE()
senorblancob0e89dc2014-10-20 14:03:12 -0700312 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(DummyImageFilter)
senorblancob0e89dc2014-10-20 14:03:12 -0700313 bool visited() const { return fVisited; }
314
315private:
316 mutable bool fVisited;
317};
318
319SkFlattenable* DummyImageFilter::CreateProc(SkReadBuffer& buffer) {
320 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 0);
321 bool visited = buffer.readBool();
322 return SkNEW_ARGS(DummyImageFilter, (visited));
323}
324
robertphillipsf3f5bad2014-12-19 13:49:15 -0800325#ifndef SK_IGNORE_TO_STRING
326void DummyImageFilter::toString(SkString* str) const {
327 str->appendf("DummyImageFilter: (");
328 str->append(")");
329}
330#endif
331
senorblancob0e89dc2014-10-20 14:03:12 -0700332};
333
334// Check that PDF rendering of image filters successfully falls back to
335// CPU rasterization.
336DEF_TEST(PDFImageFilter, reporter) {
halcanary3d32d502015-03-01 06:55:20 -0800337 SkDynamicMemoryWStream stream;
338 SkAutoTUnref<SkDocument> doc(SkDocument::CreatePDF(&stream));
339 SkCanvas* canvas = doc->beginPage(100.0f, 100.0f);
340
senorblancob0e89dc2014-10-20 14:03:12 -0700341 SkAutoTUnref<DummyImageFilter> filter(new DummyImageFilter());
342
343 // Filter just created; should be unvisited.
344 REPORTER_ASSERT(reporter, !filter->visited());
345 SkPaint paint;
346 paint.setImageFilter(filter.get());
halcanary3d32d502015-03-01 06:55:20 -0800347 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
348 doc->close();
senorblancob0e89dc2014-10-20 14:03:12 -0700349
350 // Filter was used in rendering; should be visited.
351 REPORTER_ASSERT(reporter, filter->visited());
352}