| |
| /* |
| * Copyright 2010 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| |
| #include "SkData.h" |
| #include "SkFlate.h" |
| #include "SkPDFCatalog.h" |
| #include "SkPDFStream.h" |
| #include "SkStream.h" |
| #include "SkStreamPriv.h" |
| |
| static bool skip_compression(SkPDFCatalog* catalog) { |
| return SkToBool(catalog->getDocumentFlags() & |
| SkPDFDocument::kFavorSpeedOverSize_Flags); |
| } |
| |
| SkPDFStream::SkPDFStream(SkStream* stream) : fState(kUnused_State) { |
| this->setData(stream); |
| } |
| |
| SkPDFStream::SkPDFStream(SkData* data) : fState(kUnused_State) { |
| this->setData(data); |
| } |
| |
| SkPDFStream::SkPDFStream(const SkPDFStream& pdfStream) |
| : SkPDFDict(), |
| fState(kUnused_State) { |
| this->setData(pdfStream.fDataStream.get()); |
| bool removeLength = true; |
| // Don't uncompress an already compressed stream, but we could. |
| if (pdfStream.fState == kCompressed_State) { |
| fState = kCompressed_State; |
| removeLength = false; |
| } |
| this->mergeFrom(pdfStream); |
| if (removeLength) { |
| this->remove("Length"); |
| } |
| } |
| |
| SkPDFStream::~SkPDFStream() {} |
| |
| void SkPDFStream::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) { |
| return emitIndirectObject(stream, catalog); |
| } |
| SkAutoMutexAcquire lock(fMutex); // multiple threads could be calling emit |
| if (!this->populate(catalog)) { |
| return fSubstitute->emitObject(stream, catalog, indirect); |
| } |
| |
| this->INHERITED::emitObject(stream, catalog, false); |
| stream->writeText(" stream\n"); |
| stream->writeStream(fDataStream.get(), fDataStream->getLength()); |
| SkAssertResult(fDataStream->rewind()); |
| stream->writeText("\nendstream"); |
| } |
| |
| size_t SkPDFStream::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| if (indirect) { |
| return getIndirectOutputSize(catalog); |
| } |
| SkAutoMutexAcquire lock(fMutex); // multiple threads could be calling emit |
| if (!this->populate(catalog)) { |
| return fSubstitute->getOutputSize(catalog, indirect); |
| } |
| |
| return this->INHERITED::getOutputSize(catalog, false) + |
| strlen(" stream\n\nendstream") + this->dataSize(); |
| } |
| |
| SkPDFStream::SkPDFStream() : fState(kUnused_State) {} |
| |
| void SkPDFStream::setData(SkData* data) { |
| fMemoryStream.setData(data); |
| if (&fMemoryStream != fDataStream.get()) { |
| fDataStream.reset(SkRef(&fMemoryStream)); |
| } |
| } |
| |
| void SkPDFStream::setData(SkStream* stream) { |
| // Code assumes that the stream starts at the beginning and is rewindable. |
| if (&fMemoryStream == fDataStream.get()) { |
| SkASSERT(&fMemoryStream != stream); |
| fMemoryStream.setData(NULL); |
| } |
| SkASSERT(0 == fMemoryStream.getLength()); |
| if (stream) { |
| // SkStreamRewindableFromSkStream will try stream->duplicate(). |
| fDataStream.reset(SkStreamRewindableFromSkStream(stream)); |
| SkASSERT(fDataStream.get()); |
| } else { |
| fDataStream.reset(SkRef(&fMemoryStream)); |
| } |
| } |
| |
| size_t SkPDFStream::dataSize() const { |
| SkASSERT(fDataStream->hasLength()); |
| return fDataStream->getLength(); |
| } |
| |
| bool SkPDFStream::populate(SkPDFCatalog* catalog) { |
| if (fState == kUnused_State) { |
| if (!skip_compression(catalog) && SkFlate::HaveFlate()) { |
| SkDynamicMemoryWStream compressedData; |
| |
| SkAssertResult( |
| SkFlate::Deflate(fDataStream.get(), &compressedData)); |
| SkAssertResult(fDataStream->rewind()); |
| if (compressedData.getOffset() < this->dataSize()) { |
| SkAutoTUnref<SkStream> compressed( |
| compressedData.detachAsStream()); |
| this->setData(compressed.get()); |
| insertName("Filter", "FlateDecode"); |
| } |
| fState = kCompressed_State; |
| } else { |
| fState = kNoCompression_State; |
| } |
| insertInt("Length", this->dataSize()); |
| } else if (fState == kNoCompression_State && !skip_compression(catalog) && |
| SkFlate::HaveFlate()) { |
| if (!fSubstitute.get()) { |
| fSubstitute.reset(new SkPDFStream(*this)); |
| catalog->setSubstitute(this, fSubstitute.get()); |
| } |
| return false; |
| } |
| return true; |
| } |