blob: f15f06090d7b10787e32f9f6498290224abcf127 [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"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000011#include "SkFlate.h"
edisonn@google.comd9dfa182013-04-24 13:01:01 +000012#include "SkImageEncoder.h"
13#include "SkMatrix.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000014#include "SkPDFCatalog.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +000015#include "SkPDFDevice.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000016#include "SkPDFStream.h"
17#include "SkPDFTypes.h"
senorblancob0e89dc2014-10-20 14:03:12 -070018#include "SkReadBuffer.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000019#include "SkScalar.h"
20#include "SkStream.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000021#include "SkTypes.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000022#include "Test.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000023
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000024class SkPDFTestDict : public SkPDFDict {
25public:
edisonn@google.com6addb192013-04-02 15:33:08 +000026 virtual void getResources(const SkTSet<SkPDFObject*>& knownResourceObjects,
27 SkTSet<SkPDFObject*>* newResourceObjects) {
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000028 for (int i = 0; i < fResources.count(); i++) {
edisonn@google.com6addb192013-04-02 15:33:08 +000029 newResourceObjects->add(fResources[i]);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000030 fResources[i]->ref();
31 }
32 }
33
34 void addResource(SkPDFObject* object) {
35 fResources.append(1, &object);
36 }
37
38private:
39 SkTDArray<SkPDFObject*> fResources;
40};
41
commit-bot@chromium.org608ea652013-10-03 19:29:21 +000042#define DUMMY_TEXT "DCT compessed stream."
43
44static SkData* encode_to_dct_data(size_t* pixelRefOffset, const SkBitmap& bitmap) {
45 *pixelRefOffset = 0;
46 return SkData::NewWithProc(DUMMY_TEXT, sizeof(DUMMY_TEXT) - 1, NULL, NULL);
edisonn@google.comd9dfa182013-04-24 13:01:01 +000047}
48
reed@google.com8a85d0c2011-06-24 19:12:12 +000049static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
50 const void* buffer, size_t len) {
51 SkAutoDataUnref data(stream.copyToData());
robertphillips@google.com59f46b82012-07-10 17:30:58 +000052 if (offset + len > data->size()) {
reed@google.com8a85d0c2011-06-24 19:12:12 +000053 return false;
54 }
robertphillips@google.com59f46b82012-07-10 17:30:58 +000055 return memcmp(data->bytes() + offset, buffer, len) == 0;
reed@google.com8a85d0c2011-06-24 19:12:12 +000056}
57
edisonn@google.comd9dfa182013-04-24 13:01:01 +000058static bool stream_contains(const SkDynamicMemoryWStream& stream,
59 const char* buffer) {
60 SkAutoDataUnref data(stream.copyToData());
bsalomonfbaace02014-12-12 16:41:46 -080061 size_t len = strlen(buffer); // our buffer does not have EOSs.
edisonn@google.comd9dfa182013-04-24 13:01:01 +000062
bsalomonfbaace02014-12-12 16:41:46 -080063 for (size_t offset = 0 ; offset < data->size() - len; offset++) {
edisonn@google.comd9dfa182013-04-24 13:01:01 +000064 if (memcmp(data->bytes() + offset, buffer, len) == 0) {
65 return true;
66 }
67 }
68
69 return false;
70}
71
halcanary6a144342015-01-23 11:45:10 -080072static void emit_object(SkPDFObject* object,
73 SkWStream* stream,
74 SkPDFCatalog* catalog,
75 bool indirect) {
76 SkPDFObject* realObject = catalog->getSubstituteObject(object);
77 if (indirect) {
78 catalog->emitObjectNumber(stream, realObject);
79 stream->writeText(" obj\n");
80 realObject->emitObject(stream, catalog);
81 stream->writeText("\nendobj\n");
82 } else {
83 realObject->emitObject(stream, catalog);
84 }
85}
86
87static size_t get_output_size(SkPDFObject* object,
88 SkPDFCatalog* catalog,
89 bool indirect) {
90 SkDynamicMemoryWStream buffer;
91 emit_object(object, &buffer, catalog, indirect);
92 return buffer.getOffset();
93}
94
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000095static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
vandebo@chromium.org421d6442011-07-20 17:39:01 +000096 const char* expectedData, size_t expectedSize,
97 bool indirect, bool compression) {
98 SkPDFDocument::Flags docFlags = (SkPDFDocument::Flags) 0;
99 if (!compression) {
edisonn@google.com8c020612013-03-12 19:53:16 +0000100 docFlags = SkTBitOr(docFlags, SkPDFDocument::kFavorSpeedOverSize_Flags);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000101 }
102 SkPDFCatalog catalog(docFlags);
halcanary6a144342015-01-23 11:45:10 -0800103 size_t directSize = get_output_size(obj, &catalog, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000104 REPORTER_ASSERT(reporter, directSize == expectedSize);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000105
106 SkDynamicMemoryWStream buffer;
halcanary6a144342015-01-23 11:45:10 -0800107 emit_object(obj, &buffer, &catalog, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000108 REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000109 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +0000110 directSize));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000111
112 if (indirect) {
113 // Indirect output.
114 static char header[] = "1 0 obj\n";
115 static size_t headerLen = strlen(header);
116 static char footer[] = "\nendobj\n";
117 static size_t footerLen = strlen(footer);
118
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000119 catalog.addObject(obj, false);
120
halcanary6a144342015-01-23 11:45:10 -0800121 size_t indirectSize = get_output_size(obj, &catalog, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000122 REPORTER_ASSERT(reporter,
123 indirectSize == directSize + headerLen + footerLen);
124
125 buffer.reset();
halcanary6a144342015-01-23 11:45:10 -0800126 emit_object(obj, &buffer, &catalog, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000127 REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
reed@google.com8a85d0c2011-06-24 19:12:12 +0000128 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000129 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +0000130 directSize));
131 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
132 footer, footerLen));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000133 }
134}
135
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000136static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
137 SkPDFObject* obj,
robertphillips@google.com4debcac2012-05-14 16:33:36 +0000138 const char* expectedResult) {
139 CheckObjectOutput(reporter, obj, expectedResult,
140 strlen(expectedResult), true, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000141}
142
143static void TestPDFStream(skiatest::Reporter* reporter) {
144 char streamBytes[] = "Test\nFoo\tBar";
scroggoa1193e42015-01-21 12:09:53 -0800145 SkAutoTDelete<SkMemoryStream> streamData(new SkMemoryStream(
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000146 streamBytes, strlen(streamBytes), true));
147 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000148 SimpleCheckObjectOutput(
149 reporter, stream.get(),
150 "<</Length 12\n>> stream\nTest\nFoo\tBar\nendstream");
151 stream->insert("Attribute", new SkPDFInt(42))->unref();
152 SimpleCheckObjectOutput(reporter, stream.get(),
153 "<</Length 12\n/Attribute 42\n>> stream\n"
154 "Test\nFoo\tBar\nendstream");
155
156 if (SkFlate::HaveFlate()) {
157 char streamBytes2[] = "This is a longer string, so that compression "
158 "can do something with it. With shorter strings, "
159 "the short circuit logic cuts in and we end up "
160 "with an uncompressed string.";
161 SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
162 strlen(streamBytes2)));
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000163 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000164
165 SkDynamicMemoryWStream compressedByteStream;
166 SkFlate::Deflate(streamData2.get(), &compressedByteStream);
167 SkAutoDataUnref compressedData(compressedByteStream.copyToData());
168
169 // Check first without compression.
170 SkDynamicMemoryWStream expectedResult1;
171 expectedResult1.writeText("<</Length 167\n>> stream\n");
172 expectedResult1.writeText(streamBytes2);
173 expectedResult1.writeText("\nendstream");
174 SkAutoDataUnref expectedResultData1(expectedResult1.copyToData());
175 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000176 (const char*) expectedResultData1->data(),
177 expectedResultData1->size(), true, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000178
179 // Then again with compression.
180 SkDynamicMemoryWStream expectedResult2;
181 expectedResult2.writeText("<</Filter /FlateDecode\n/Length 116\n"
182 ">> stream\n");
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000183 expectedResult2.write(compressedData->data(), compressedData->size());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000184 expectedResult2.writeText("\nendstream");
185 SkAutoDataUnref expectedResultData2(expectedResult2.copyToData());
186 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000187 (const char*) expectedResultData2->data(),
188 expectedResultData2->size(), true, true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000189 }
190}
191
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000192static void TestCatalog(skiatest::Reporter* reporter) {
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000193 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000194 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
195 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
196 SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
197 int1.get()->ref();
198 SkAutoTUnref<SkPDFInt> int1Again(int1.get());
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000199
200 catalog.addObject(int1.get(), false);
201 catalog.addObject(int2.get(), false);
202 catalog.addObject(int3.get(), false);
203
204 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
205 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
206 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int3.get()) == 3);
207
208 SkDynamicMemoryWStream buffer;
209 catalog.emitObjectNumber(&buffer, int1.get());
210 catalog.emitObjectNumber(&buffer, int2.get());
211 catalog.emitObjectNumber(&buffer, int3.get());
212 catalog.emitObjectNumber(&buffer, int1Again.get());
213 char expectedResult[] = "1 02 03 01 0";
reed@google.com8a85d0c2011-06-24 19:12:12 +0000214 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
215 strlen(expectedResult)));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000216}
217
218static void TestObjectRef(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000219 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
220 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
221 SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000222
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000223 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000224 catalog.addObject(int1.get(), false);
225 catalog.addObject(int2.get(), false);
226 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
227 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
228
229 char expectedResult[] = "2 0 R";
230 SkDynamicMemoryWStream buffer;
halcanary4fc48af2015-01-12 10:07:50 -0800231 int2ref->emitObject(&buffer, &catalog);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000232 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
reed@google.com8a85d0c2011-06-24 19:12:12 +0000233 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
234 buffer.getOffset()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000235}
236
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000237static void TestSubstitute(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000238 SkAutoTUnref<SkPDFTestDict> proxy(new SkPDFTestDict());
239 SkAutoTUnref<SkPDFTestDict> stub(new SkPDFTestDict());
240 SkAutoTUnref<SkPDFInt> int33(new SkPDFInt(33));
241 SkAutoTUnref<SkPDFDict> stubResource(new SkPDFDict());
242 SkAutoTUnref<SkPDFInt> int44(new SkPDFInt(44));
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000243
244 stub->insert("Value", int33.get());
245 stubResource->insert("InnerValue", int44.get());
246 stub->addResource(stubResource.get());
247
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000248 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000249 catalog.addObject(proxy.get(), false);
250 catalog.setSubstitute(proxy.get(), stub.get());
251
252 SkDynamicMemoryWStream buffer;
halcanary6a144342015-01-23 11:45:10 -0800253 emit_object(proxy, &buffer, &catalog, false);
halcanaryf361b712015-01-13 07:12:57 -0800254 SkTSet<SkPDFObject*>* substituteResources =
255 catalog.getSubstituteList(false);
256 for (int i = 0; i < substituteResources->count(); ++i) {
halcanary6a144342015-01-23 11:45:10 -0800257 emit_object((*substituteResources)[i], &buffer, &catalog, true);
halcanaryf361b712015-01-13 07:12:57 -0800258 }
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000259
vandebo@chromium.orgd3a094c2011-07-25 22:22:25 +0000260 char objectResult[] = "2 0 obj\n<</Value 33\n>>\nendobj\n";
halcanaryf361b712015-01-13 07:12:57 -0800261 catalog.setFileOffset(proxy.get(), 0);
262
halcanary6a144342015-01-23 11:45:10 -0800263 size_t outputSize = get_output_size(
264 catalog.getSubstituteObject(proxy.get()), &catalog, true);
halcanaryf361b712015-01-13 07:12:57 -0800265 REPORTER_ASSERT(reporter, outputSize == strlen(objectResult));
vandebo@chromium.orgd3a094c2011-07-25 22:22:25 +0000266
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000267 char expectedResult[] =
268 "<</Value 33\n>>1 0 obj\n<</InnerValue 44\n>>\nendobj\n";
269 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
270 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
271 buffer.getOffset()));
272}
273
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000274// Create a bitmap that would be very eficiently compressed in a ZIP.
edisonn@google.comf1a358f2013-04-24 13:21:49 +0000275static void setup_bitmap(SkBitmap* bitmap, int width, int height) {
mike@reedtribe.orgdeee4962014-02-13 14:41:43 +0000276 bitmap->allocN32Pixels(width, height);
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000277 bitmap->eraseColor(SK_ColorWHITE);
278}
279
280static void TestImage(skiatest::Reporter* reporter, const SkBitmap& bitmap,
281 const char* expected, bool useDCTEncoder) {
282 SkISize pageSize = SkISize::Make(bitmap.width(), bitmap.height());
robertphillips@google.com57c56722013-08-26 11:10:13 +0000283 SkAutoTUnref<SkPDFDevice> dev(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000284
285 if (useDCTEncoder) {
commit-bot@chromium.org608ea652013-10-03 19:29:21 +0000286 dev->setDCTEncoder(encode_to_dct_data);
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000287 }
288
289 SkCanvas c(dev);
290 c.drawBitmap(bitmap, 0, 0, NULL);
291
292 SkPDFDocument doc;
293 doc.appendPage(dev);
294
295 SkDynamicMemoryWStream stream;
296 doc.emitPDF(&stream);
297
298 REPORTER_ASSERT(reporter, stream_contains(stream, expected));
299}
300
301static void TestUncompressed(skiatest::Reporter* reporter) {
302 SkBitmap bitmap;
edisonn@google.comf1a358f2013-04-24 13:21:49 +0000303 setup_bitmap(&bitmap, 1, 1);
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000304 TestImage(reporter, bitmap,
305 "/Subtype /Image\n"
306 "/Width 1\n"
307 "/Height 1\n"
308 "/ColorSpace /DeviceRGB\n"
309 "/BitsPerComponent 8\n"
310 "/Length 3\n"
311 ">> stream",
312 true);
313}
314
315static void TestFlateDecode(skiatest::Reporter* reporter) {
316 if (!SkFlate::HaveFlate()) {
317 return;
318 }
skia.committer@gmail.com83f0d302013-04-25 07:01:04 +0000319 SkBitmap bitmap;
edisonn@google.comf1a358f2013-04-24 13:21:49 +0000320 setup_bitmap(&bitmap, 10, 10);
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000321 TestImage(reporter, bitmap,
322 "/Subtype /Image\n"
323 "/Width 10\n"
324 "/Height 10\n"
325 "/ColorSpace /DeviceRGB\n"
326 "/BitsPerComponent 8\n"
327 "/Filter /FlateDecode\n"
328 "/Length 13\n"
329 ">> stream",
330 false);
331}
332
333static void TestDCTDecode(skiatest::Reporter* reporter) {
334 SkBitmap bitmap;
edisonn@google.comf1a358f2013-04-24 13:21:49 +0000335 setup_bitmap(&bitmap, 32, 32);
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000336 TestImage(reporter, bitmap,
337 "/Subtype /Image\n"
338 "/Width 32\n"
339 "/Height 32\n"
340 "/ColorSpace /DeviceRGB\n"
341 "/BitsPerComponent 8\n"
342 "/Filter /DCTDecode\n"
343 "/ColorTransform 0\n"
344 "/Length 21\n"
345 ">> stream",
346 true);
347}
348
349static void TestImages(skiatest::Reporter* reporter) {
350 TestUncompressed(reporter);
351 TestFlateDecode(reporter);
352 TestDCTDecode(reporter);
353}
354
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000355// This test used to assert without the fix submitted for
356// http://code.google.com/p/skia/issues/detail?id=1083.
357// SKP files might have invalid glyph ids. This test ensures they are ignored,
358// and there is no assert on input data in Debug mode.
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000359static void test_issue1083() {
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000360 SkISize pageSize = SkISize::Make(100, 100);
robertphillips@google.com35300c42013-03-21 17:38:49 +0000361 SkAutoTUnref<SkPDFDevice> dev(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000362
363 SkCanvas c(dev);
364 SkPaint paint;
365 paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
366
367 uint16_t glyphID = 65000;
368 c.drawText(&glyphID, 2, 0, 0, paint);
369
370 SkPDFDocument doc;
371 doc.appendPage(dev);
372
373 SkDynamicMemoryWStream stream;
374 doc.emitPDF(&stream);
375}
376
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000377DEF_TEST(PDFPrimitives, reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000378 SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000379 SimpleCheckObjectOutput(reporter, int42.get(), "42");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000380
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000381 SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000382 SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000383
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000384 SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000385#if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000386 SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000387#else
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000388 SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000389
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000390 SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000391 SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000392
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000393 SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000394 SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000395#endif
396
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000397 SkAutoTUnref<SkPDFString> stringSimple(
398 new SkPDFString("test ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000399 SimpleCheckObjectOutput(reporter, stringSimple.get(),
400 "(test \\) string \\( foo)");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000401 SkAutoTUnref<SkPDFString> stringComplex(
402 new SkPDFString("\ttest ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000403 SimpleCheckObjectOutput(reporter, stringComplex.get(),
404 "<0974657374202920737472696E67202820666F6F>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000405
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000406 SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000407 const char expectedResult[] = "/Test#20name#09with#23tab";
408 CheckObjectOutput(reporter, name.get(), expectedResult,
409 strlen(expectedResult), false, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000410
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000411 SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000412 const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
413 CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
414 strlen(escapedNameExpected), false, false);
415
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000416 // Test that we correctly handle characters with the high-bit set.
bungeman@google.comf8aa18c2012-03-19 21:04:52 +0000417 const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000418 SkAutoTUnref<SkPDFName> highBitName(
419 new SkPDFName((const char*)highBitCString));
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000420 const char highBitExpectedResult[] = "/#DE#ADbe#EF";
421 CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
422 strlen(highBitExpectedResult), false, false);
423
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000424 SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000425 SimpleCheckObjectOutput(reporter, array.get(), "[]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000426 array->append(int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000427 SimpleCheckObjectOutput(reporter, array.get(), "[42]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000428 array->append(realHalf.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000429 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000430 SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000431 array->append(int0.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000432 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000433 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000434 array->setAt(0, int1.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000435 SimpleCheckObjectOutput(reporter, array.get(), "[1 0.5 0]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000436
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000437 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000438 SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000439 SkAutoTUnref<SkPDFName> n1(new SkPDFName("n1"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000440 dict->insert(n1.get(), int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000441 SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42\n>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000442 SkAutoTUnref<SkPDFName> n2(new SkPDFName("n2"));
443 SkAutoTUnref<SkPDFName> n3(new SkPDFName("n3"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000444 dict->insert(n2.get(), realHalf.get());
445 dict->insert(n3.get(), array.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000446 SimpleCheckObjectOutput(reporter, dict.get(),
447 "<</n1 42\n/n2 0.5\n/n3 [1 0.5 0]\n>>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000448
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000449 TestPDFStream(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000450
451 TestCatalog(reporter);
452
453 TestObjectRef(reporter);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000454
455 TestSubstitute(reporter);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000456
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000457 test_issue1083();
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000458
skia.committer@gmail.com83f0d302013-04-25 07:01:04 +0000459 TestImages(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000460}
senorblancob0e89dc2014-10-20 14:03:12 -0700461
462namespace {
463
464class DummyImageFilter : public SkImageFilter {
465public:
466 DummyImageFilter(bool visited = false) : SkImageFilter(0, NULL), fVisited(visited) {}
mtklein72c9faa2015-01-09 10:06:39 -0800467 ~DummyImageFilter() SK_OVERRIDE {}
senorblancob0e89dc2014-10-20 14:03:12 -0700468 virtual bool onFilterImage(Proxy*, const SkBitmap& src, const Context&,
469 SkBitmap* result, SkIPoint* offset) const {
470 fVisited = true;
471 offset->fX = offset->fY = 0;
472 *result = src;
473 return true;
474 }
robertphillipsf3f5bad2014-12-19 13:49:15 -0800475 SK_TO_STRING_OVERRIDE()
senorblancob0e89dc2014-10-20 14:03:12 -0700476 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(DummyImageFilter)
senorblancob0e89dc2014-10-20 14:03:12 -0700477 bool visited() const { return fVisited; }
478
479private:
480 mutable bool fVisited;
481};
482
483SkFlattenable* DummyImageFilter::CreateProc(SkReadBuffer& buffer) {
484 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 0);
485 bool visited = buffer.readBool();
486 return SkNEW_ARGS(DummyImageFilter, (visited));
487}
488
robertphillipsf3f5bad2014-12-19 13:49:15 -0800489#ifndef SK_IGNORE_TO_STRING
490void DummyImageFilter::toString(SkString* str) const {
491 str->appendf("DummyImageFilter: (");
492 str->append(")");
493}
494#endif
495
senorblancob0e89dc2014-10-20 14:03:12 -0700496};
497
498// Check that PDF rendering of image filters successfully falls back to
499// CPU rasterization.
500DEF_TEST(PDFImageFilter, reporter) {
501 SkISize pageSize = SkISize::Make(100, 100);
502 SkAutoTUnref<SkPDFDevice> device(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
503 SkCanvas canvas(device.get());
504 SkAutoTUnref<DummyImageFilter> filter(new DummyImageFilter());
505
506 // Filter just created; should be unvisited.
507 REPORTER_ASSERT(reporter, !filter->visited());
508 SkPaint paint;
509 paint.setImageFilter(filter.get());
510 canvas.drawRect(SkRect::MakeWH(100, 100), paint);
511
512 // Filter was used in rendering; should be visited.
513 REPORTER_ASSERT(reporter, filter->visited());
514}