blob: d113a0b355d16ce26a13b2313dbdffef28fc5cb6 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2010 Google Inc.
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +00004 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00005 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +00007 */
8
epoger@google.comec3ed6a2011-07-28 14:26:00 +00009
reed@google.com8a85d0c2011-06-24 19:12:12 +000010#include "SkData.h"
vandebo@chromium.orga09ef972010-12-01 22:17:20 +000011#include "SkFlate.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000012#include "SkPDFCatalog.h"
13#include "SkPDFStream.h"
14#include "SkStream.h"
15
vandebo@chromium.org421d6442011-07-20 17:39:01 +000016static bool skip_compression(SkPDFCatalog* catalog) {
vandebo@chromium.org238be8c2012-07-13 20:06:02 +000017 return SkToBool(catalog->getDocumentFlags() &
18 SkPDFDocument::kNoCompression_Flags);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000019}
vandebo@chromium.org1cfa2c42011-01-31 19:35:43 +000020
vandebo@chromium.org421d6442011-07-20 17:39:01 +000021SkPDFStream::SkPDFStream(SkStream* stream)
22 : fState(kUnused_State),
23 fData(stream) {
24}
25
26SkPDFStream::SkPDFStream(SkData* data) : fState(kUnused_State) {
27 SkMemoryStream* stream = new SkMemoryStream;
28 stream->setData(data);
29 fData = stream;
30 fData->unref(); // SkRefPtr and new both took a reference.
31}
32
33SkPDFStream::SkPDFStream(const SkPDFStream& pdfStream)
34 : SkPDFDict(),
35 fState(kUnused_State),
36 fData(pdfStream.fData) {
37 bool removeLength = true;
38 // Don't uncompress an already compressed stream, but we could.
39 if (pdfStream.fState == kCompressed_State) {
40 fState = kCompressed_State;
41 removeLength = false;
vandebo@chromium.orga09ef972010-12-01 22:17:20 +000042 }
vandebo@chromium.org421d6442011-07-20 17:39:01 +000043 SkPDFDict::Iter dict(pdfStream);
44 SkPDFName* key;
45 SkPDFObject* value;
46 SkPDFName lengthName("Length");
47 for (key = dict.next(&value); key != NULL; key = dict.next(&value)) {
48 if (removeLength && *key == lengthName) {
49 continue;
50 }
51 this->insert(key, value);
52 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000053}
54
vandebo@chromium.org421d6442011-07-20 17:39:01 +000055SkPDFStream::~SkPDFStream() {}
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000056
57void SkPDFStream::emitObject(SkWStream* stream, SkPDFCatalog* catalog,
58 bool indirect) {
vandebo@chromium.org421d6442011-07-20 17:39:01 +000059 if (indirect) {
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000060 return emitIndirectObject(stream, catalog);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000061 }
62 if (!this->populate(catalog)) {
63 return fSubstitute->emitObject(stream, catalog, indirect);
64 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000065
vandebo@chromium.orgd90c1412011-02-24 21:50:04 +000066 this->INHERITED::emitObject(stream, catalog, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000067 stream->writeText(" stream\n");
vandebo@chromium.org421d6442011-07-20 17:39:01 +000068 stream->write(fData->getMemoryBase(), fData->getLength());
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +000069 stream->writeText("\nendstream");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000070}
71
72size_t SkPDFStream::getOutputSize(SkPDFCatalog* catalog, bool indirect) {
vandebo@chromium.org421d6442011-07-20 17:39:01 +000073 if (indirect) {
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000074 return getIndirectOutputSize(catalog);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000075 }
76 if (!this->populate(catalog)) {
77 return fSubstitute->getOutputSize(catalog, indirect);
78 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000079
vandebo@chromium.orgd90c1412011-02-24 21:50:04 +000080 return this->INHERITED::getOutputSize(catalog, false) +
vandebo@chromium.org421d6442011-07-20 17:39:01 +000081 strlen(" stream\n\nendstream") + fData->getLength();
82}
83
84SkPDFStream::SkPDFStream() : fState(kUnused_State) {}
85
86void SkPDFStream::setData(SkStream* stream) {
87 fData = stream;
88}
89
90bool SkPDFStream::populate(SkPDFCatalog* catalog) {
91 if (fState == kUnused_State) {
92 if (!skip_compression(catalog) && SkFlate::HaveFlate()) {
93 SkDynamicMemoryWStream compressedData;
94
95 SkAssertResult(SkFlate::Deflate(fData.get(), &compressedData));
96 if (compressedData.getOffset() < fData->getLength()) {
97 SkMemoryStream* stream = new SkMemoryStream;
vandebo@chromium.org610f7162012-03-14 18:34:15 +000098 stream->setData(compressedData.copyToData())->unref();
vandebo@chromium.org421d6442011-07-20 17:39:01 +000099 fData = stream;
100 fData->unref(); // SkRefPtr and new both took a reference.
101 insertName("Filter", "FlateDecode");
102 }
103 fState = kCompressed_State;
104 } else {
105 fState = kNoCompression_State;
106 }
107 insertInt("Length", fData->getLength());
108 } else if (fState == kNoCompression_State && !skip_compression(catalog) &&
109 SkFlate::HaveFlate()) {
110 if (!fSubstitute.get()) {
111 fSubstitute = new SkPDFStream(*this);
112 fSubstitute->unref(); // SkRefPtr and new both took a reference.
113 catalog->setSubstitute(this, fSubstitute.get());
114 }
115 return false;
116 }
117 return true;
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000118}