blob: 5eaa6c28b69b3968247b0f72cc11f0dff2d3d0a5 [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"
halcanary67ec1f82014-06-27 11:36:20 -070015#include "SkStreamPriv.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000016
vandebo@chromium.org421d6442011-07-20 17:39:01 +000017static bool skip_compression(SkPDFCatalog* catalog) {
vandebo@chromium.org238be8c2012-07-13 20:06:02 +000018 return SkToBool(catalog->getDocumentFlags() &
edisonn@google.com8c020612013-03-12 19:53:16 +000019 SkPDFDocument::kFavorSpeedOverSize_Flags);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000020}
vandebo@chromium.org1cfa2c42011-01-31 19:35:43 +000021
vandebo@chromium.org26e14492013-08-26 22:52:09 +000022SkPDFStream::SkPDFStream(SkStream* stream) : fState(kUnused_State) {
halcanary67ec1f82014-06-27 11:36:20 -070023 this->setData(stream);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000024}
25
26SkPDFStream::SkPDFStream(SkData* data) : fState(kUnused_State) {
halcanary67ec1f82014-06-27 11:36:20 -070027 this->setData(data);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000028}
29
30SkPDFStream::SkPDFStream(const SkPDFStream& pdfStream)
31 : SkPDFDict(),
vandebo@chromium.org26e14492013-08-26 22:52:09 +000032 fState(kUnused_State) {
halcanarye3224822014-07-14 09:12:12 -070033 this->setData(pdfStream.fDataStream.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +000034 bool removeLength = true;
35 // Don't uncompress an already compressed stream, but we could.
36 if (pdfStream.fState == kCompressed_State) {
37 fState = kCompressed_State;
38 removeLength = false;
vandebo@chromium.orga09ef972010-12-01 22:17:20 +000039 }
halcanary1f8ed022014-06-27 10:37:27 -070040 this->mergeFrom(pdfStream);
41 if (removeLength) {
42 this->remove("Length");
vandebo@chromium.org421d6442011-07-20 17:39:01 +000043 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000044}
45
vandebo@chromium.org421d6442011-07-20 17:39:01 +000046SkPDFStream::~SkPDFStream() {}
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000047
halcanary4fc48af2015-01-12 10:07:50 -080048void SkPDFStream::emitObject(SkWStream* stream, SkPDFCatalog* catalog) {
halcanary67ec1f82014-06-27 11:36:20 -070049 SkAutoMutexAcquire lock(fMutex); // multiple threads could be calling emit
vandebo@chromium.org421d6442011-07-20 17:39:01 +000050 if (!this->populate(catalog)) {
halcanary4fc48af2015-01-12 10:07:50 -080051 return fSubstitute->emitObject(stream, catalog);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000052 }
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000053
halcanary4fc48af2015-01-12 10:07:50 -080054 this->INHERITED::emitObject(stream, catalog);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000055 stream->writeText(" stream\n");
halcanarye3224822014-07-14 09:12:12 -070056 stream->writeStream(fDataStream.get(), fDataStream->getLength());
57 SkAssertResult(fDataStream->rewind());
vandebo@chromium.org9b49dc02010-10-20 22:23:29 +000058 stream->writeText("\nendstream");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000059}
60
vandebo@chromium.org421d6442011-07-20 17:39:01 +000061SkPDFStream::SkPDFStream() : fState(kUnused_State) {}
62
commit-bot@chromium.orgd8d976e2013-07-08 23:17:57 +000063void SkPDFStream::setData(SkData* data) {
scroggoa1193e42015-01-21 12:09:53 -080064 // FIXME: Don't swap if the data is the same.
65 fDataStream.reset(SkNEW_ARGS(SkMemoryStream, (data)));
commit-bot@chromium.orgd8d976e2013-07-08 23:17:57 +000066}
67
vandebo@chromium.org421d6442011-07-20 17:39:01 +000068void SkPDFStream::setData(SkStream* stream) {
vandebo@chromium.org26e14492013-08-26 22:52:09 +000069 // Code assumes that the stream starts at the beginning and is rewindable.
70 if (stream) {
halcanarye3224822014-07-14 09:12:12 -070071 // SkStreamRewindableFromSkStream will try stream->duplicate().
72 fDataStream.reset(SkStreamRewindableFromSkStream(stream));
73 SkASSERT(fDataStream.get());
halcanary67ec1f82014-06-27 11:36:20 -070074 } else {
scroggoa1193e42015-01-21 12:09:53 -080075 // Use an empty memory stream.
76 fDataStream.reset(SkNEW(SkMemoryStream));
vandebo@chromium.org26e14492013-08-26 22:52:09 +000077 }
halcanary67ec1f82014-06-27 11:36:20 -070078}
79
80size_t SkPDFStream::dataSize() const {
halcanarye3224822014-07-14 09:12:12 -070081 SkASSERT(fDataStream->hasLength());
82 return fDataStream->getLength();
vandebo@chromium.org421d6442011-07-20 17:39:01 +000083}
84
85bool SkPDFStream::populate(SkPDFCatalog* catalog) {
86 if (fState == kUnused_State) {
87 if (!skip_compression(catalog) && SkFlate::HaveFlate()) {
88 SkDynamicMemoryWStream compressedData;
89
halcanarye3224822014-07-14 09:12:12 -070090 SkAssertResult(
91 SkFlate::Deflate(fDataStream.get(), &compressedData));
92 SkAssertResult(fDataStream->rewind());
halcanary67ec1f82014-06-27 11:36:20 -070093 if (compressedData.getOffset() < this->dataSize()) {
scroggoa1193e42015-01-21 12:09:53 -080094 SkAutoTDelete<SkStream> compressed(
halcanaryf459c852014-07-14 12:59:59 -070095 compressedData.detachAsStream());
96 this->setData(compressed.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +000097 insertName("Filter", "FlateDecode");
98 }
99 fState = kCompressed_State;
100 } else {
101 fState = kNoCompression_State;
102 }
halcanary67ec1f82014-06-27 11:36:20 -0700103 insertInt("Length", this->dataSize());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000104 } else if (fState == kNoCompression_State && !skip_compression(catalog) &&
105 SkFlate::HaveFlate()) {
106 if (!fSubstitute.get()) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000107 fSubstitute.reset(new SkPDFStream(*this));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000108 catalog->setSubstitute(this, fSubstitute.get());
109 }
110 return false;
111 }
112 return true;
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000113}