PDFPrimitivesTest.cpp revision d1c53aae59ee44377be8bc0cc15e54d46aa530ce
1
2/*
3 * Copyright 2010 The Android Open Source Project
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
9
10#include "Test.h"
11#include "SkCanvas.h"
12#include "SkData.h"
13#include "SkFlate.h"
14#include "SkPDFCatalog.h"
15#include "SkPDFDevice.h"
16#include "SkPDFStream.h"
17#include "SkPDFTypes.h"
18#include "SkScalar.h"
19#include "SkStream.h"
20#include "SkTypes.h"
21
22class SkPDFTestDict : public SkPDFDict {
23public:
24    void getResources(SkTDArray<SkPDFObject*>* resourceList) {
25        resourceList->setReserve(resourceList->count() + fResources.count());
26        for (int i = 0; i < fResources.count(); i++) {
27            resourceList->push(fResources[i]);
28            fResources[i]->ref();
29        }
30    }
31
32    void addResource(SkPDFObject* object) {
33        fResources.append(1, &object);
34    }
35
36private:
37    SkTDArray<SkPDFObject*> fResources;
38};
39
40static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
41                          const void* buffer, size_t len) {
42    SkAutoDataUnref data(stream.copyToData());
43    if (offset + len > data->size()) {
44        return false;
45    }
46    return memcmp(data->bytes() + offset, buffer, len) == 0;
47}
48
49static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
50                              const char* expectedData, size_t expectedSize,
51                              bool indirect, bool compression) {
52    SkPDFDocument::Flags docFlags = (SkPDFDocument::Flags) 0;
53    if (!compression) {
54        docFlags = SkTBitOr(docFlags, SkPDFDocument::kNoCompression_Flags);
55    }
56    SkPDFCatalog catalog(docFlags);
57    size_t directSize = obj->getOutputSize(&catalog, false);
58    REPORTER_ASSERT(reporter, directSize == expectedSize);
59
60    SkDynamicMemoryWStream buffer;
61    obj->emit(&buffer, &catalog, false);
62    REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
63    REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedData,
64                                            directSize));
65
66    if (indirect) {
67        // Indirect output.
68        static char header[] = "1 0 obj\n";
69        static size_t headerLen = strlen(header);
70        static char footer[] = "\nendobj\n";
71        static size_t footerLen = strlen(footer);
72
73        catalog.addObject(obj, false);
74
75        size_t indirectSize = obj->getOutputSize(&catalog, true);
76        REPORTER_ASSERT(reporter,
77                        indirectSize == directSize + headerLen + footerLen);
78
79        buffer.reset();
80        obj->emit(&buffer, &catalog, true);
81        REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
82        REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
83        REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
84                                                directSize));
85        REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
86                                                footer, footerLen));
87    }
88}
89
90static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
91                                    SkPDFObject* obj,
92                                    const char* expectedResult) {
93    CheckObjectOutput(reporter, obj, expectedResult,
94                      strlen(expectedResult), true, false);
95}
96
97static void TestPDFStream(skiatest::Reporter* reporter) {
98    char streamBytes[] = "Test\nFoo\tBar";
99    SkAutoTUnref<SkMemoryStream> streamData(new SkMemoryStream(
100        streamBytes, strlen(streamBytes), true));
101    SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
102    SimpleCheckObjectOutput(
103        reporter, stream.get(),
104        "<</Length 12\n>> stream\nTest\nFoo\tBar\nendstream");
105    stream->insert("Attribute", new SkPDFInt(42))->unref();
106    SimpleCheckObjectOutput(reporter, stream.get(),
107                            "<</Length 12\n/Attribute 42\n>> stream\n"
108                                "Test\nFoo\tBar\nendstream");
109
110    if (SkFlate::HaveFlate()) {
111        char streamBytes2[] = "This is a longer string, so that compression "
112                              "can do something with it. With shorter strings, "
113                              "the short circuit logic cuts in and we end up "
114                              "with an uncompressed string.";
115        SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
116                                                        strlen(streamBytes2)));
117        SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
118
119        SkDynamicMemoryWStream compressedByteStream;
120        SkFlate::Deflate(streamData2.get(), &compressedByteStream);
121        SkAutoDataUnref compressedData(compressedByteStream.copyToData());
122
123        // Check first without compression.
124        SkDynamicMemoryWStream expectedResult1;
125        expectedResult1.writeText("<</Length 167\n>> stream\n");
126        expectedResult1.writeText(streamBytes2);
127        expectedResult1.writeText("\nendstream");
128        SkAutoDataUnref expectedResultData1(expectedResult1.copyToData());
129        CheckObjectOutput(reporter, stream.get(),
130                          (const char*) expectedResultData1->data(),
131                          expectedResultData1->size(), true, false);
132
133        // Then again with compression.
134        SkDynamicMemoryWStream expectedResult2;
135        expectedResult2.writeText("<</Filter /FlateDecode\n/Length 116\n"
136                                 ">> stream\n");
137        expectedResult2.write(compressedData->data(), compressedData->size());
138        expectedResult2.writeText("\nendstream");
139        SkAutoDataUnref expectedResultData2(expectedResult2.copyToData());
140        CheckObjectOutput(reporter, stream.get(),
141                          (const char*) expectedResultData2->data(),
142                          expectedResultData2->size(), true, true);
143    }
144}
145
146static void TestCatalog(skiatest::Reporter* reporter) {
147    SkPDFCatalog catalog((SkPDFDocument::Flags)0);
148    SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
149    SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
150    SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
151    int1.get()->ref();
152    SkAutoTUnref<SkPDFInt> int1Again(int1.get());
153
154    catalog.addObject(int1.get(), false);
155    catalog.addObject(int2.get(), false);
156    catalog.addObject(int3.get(), false);
157
158    REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
159    REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
160    REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int3.get()) == 3);
161
162    SkDynamicMemoryWStream buffer;
163    catalog.emitObjectNumber(&buffer, int1.get());
164    catalog.emitObjectNumber(&buffer, int2.get());
165    catalog.emitObjectNumber(&buffer, int3.get());
166    catalog.emitObjectNumber(&buffer, int1Again.get());
167    char expectedResult[] = "1 02 03 01 0";
168    REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
169                                            strlen(expectedResult)));
170}
171
172static void TestObjectRef(skiatest::Reporter* reporter) {
173    SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
174    SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
175    SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
176
177    SkPDFCatalog catalog((SkPDFDocument::Flags)0);
178    catalog.addObject(int1.get(), false);
179    catalog.addObject(int2.get(), false);
180    REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
181    REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
182
183    char expectedResult[] = "2 0 R";
184    SkDynamicMemoryWStream buffer;
185    int2ref->emitObject(&buffer, &catalog, false);
186    REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
187    REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
188                                            buffer.getOffset()));
189}
190
191static void TestSubstitute(skiatest::Reporter* reporter) {
192    SkAutoTUnref<SkPDFTestDict> proxy(new SkPDFTestDict());
193    SkAutoTUnref<SkPDFTestDict> stub(new SkPDFTestDict());
194    SkAutoTUnref<SkPDFInt> int33(new SkPDFInt(33));
195    SkAutoTUnref<SkPDFDict> stubResource(new SkPDFDict());
196    SkAutoTUnref<SkPDFInt> int44(new SkPDFInt(44));
197
198    stub->insert("Value", int33.get());
199    stubResource->insert("InnerValue", int44.get());
200    stub->addResource(stubResource.get());
201
202    SkPDFCatalog catalog((SkPDFDocument::Flags)0);
203    catalog.addObject(proxy.get(), false);
204    catalog.setSubstitute(proxy.get(), stub.get());
205
206    SkDynamicMemoryWStream buffer;
207    proxy->emit(&buffer, &catalog, false);
208    catalog.emitSubstituteResources(&buffer, false);
209
210    char objectResult[] = "2 0 obj\n<</Value 33\n>>\nendobj\n";
211    REPORTER_ASSERT(
212        reporter,
213        catalog.setFileOffset(proxy.get(), 0) == strlen(objectResult));
214
215    char expectedResult[] =
216        "<</Value 33\n>>1 0 obj\n<</InnerValue 44\n>>\nendobj\n";
217    REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
218    REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
219                                            buffer.getOffset()));
220}
221
222// This test used to assert without the fix submitted for
223// http://code.google.com/p/skia/issues/detail?id=1083.
224// SKP files might have invalid glyph ids. This test ensures they are ignored,
225// and there is no assert on input data in Debug mode.
226static void test_issue1083(skiatest::Reporter* reporter) {
227    SkISize pageSize = SkISize::Make(100, 100);
228    SkPDFDevice* dev = new SkPDFDevice(pageSize, pageSize, SkMatrix::I());
229
230    SkCanvas c(dev);
231    SkPaint paint;
232    paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
233
234    uint16_t glyphID = 65000;
235    c.drawText(&glyphID, 2, 0, 0, paint);
236
237    SkPDFDocument doc;
238    doc.appendPage(dev);
239
240    SkDynamicMemoryWStream stream;
241    doc.emitPDF(&stream);
242}
243
244static void TestPDFPrimitives(skiatest::Reporter* reporter) {
245    SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
246    SimpleCheckObjectOutput(reporter, int42.get(), "42");
247
248    SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
249    SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
250
251#if defined(SK_SCALAR_IS_FLOAT)
252    SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
253#if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
254    SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
255#else
256    SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
257
258    SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
259    SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
260
261    SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
262    SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
263#endif
264#endif
265
266    SkAutoTUnref<SkPDFString> stringSimple(
267        new SkPDFString("test ) string ( foo"));
268    SimpleCheckObjectOutput(reporter, stringSimple.get(),
269                            "(test \\) string \\( foo)");
270    SkAutoTUnref<SkPDFString> stringComplex(
271        new SkPDFString("\ttest ) string ( foo"));
272    SimpleCheckObjectOutput(reporter, stringComplex.get(),
273                            "<0974657374202920737472696E67202820666F6F>");
274
275    SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
276    const char expectedResult[] = "/Test#20name#09with#23tab";
277    CheckObjectOutput(reporter, name.get(), expectedResult,
278                      strlen(expectedResult), false, false);
279
280    SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
281    const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
282    CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
283                      strlen(escapedNameExpected), false, false);
284
285    // Test that we correctly handle characters with the high-bit set.
286    const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
287    SkAutoTUnref<SkPDFName> highBitName(
288        new SkPDFName((const char*)highBitCString));
289    const char highBitExpectedResult[] = "/#DE#ADbe#EF";
290    CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
291                      strlen(highBitExpectedResult), false, false);
292
293    SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
294    SimpleCheckObjectOutput(reporter, array.get(), "[]");
295    array->append(int42.get());
296    SimpleCheckObjectOutput(reporter, array.get(), "[42]");
297    array->append(realHalf.get());
298    SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
299    SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
300    array->append(int0.get());
301    SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
302    SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
303    array->setAt(0, int1.get());
304    SimpleCheckObjectOutput(reporter, array.get(), "[1 0.5 0]");
305
306    SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
307    SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
308    SkAutoTUnref<SkPDFName> n1(new SkPDFName("n1"));
309    dict->insert(n1.get(), int42.get());
310    SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42\n>>");
311    SkAutoTUnref<SkPDFName> n2(new SkPDFName("n2"));
312    SkAutoTUnref<SkPDFName> n3(new SkPDFName("n3"));
313    dict->insert(n2.get(), realHalf.get());
314    dict->insert(n3.get(), array.get());
315    SimpleCheckObjectOutput(reporter, dict.get(),
316                            "<</n1 42\n/n2 0.5\n/n3 [1 0.5 0]\n>>");
317
318    TestPDFStream(reporter);
319
320    TestCatalog(reporter);
321
322    TestObjectRef(reporter);
323
324    TestSubstitute(reporter);
325
326    test_issue1083(reporter);
327}
328
329#include "TestClassDef.h"
330DEFINE_TESTCLASS("PDFPrimitives", PDFPrimitivesTestClass, TestPDFPrimitives)
331