1 | /* |
2 | * Copyright 2010 The Android Open Source Project |
3 | * |
4 | * Use of this source code is governed by a BSD-style license that can be |
5 | * found in the LICENSE file. |
6 | */ |
7 | |
8 | #include "src/pdf/SkDeflate.h" |
9 | |
10 | #include "include/core/SkData.h" |
11 | #include "include/private/SkMalloc.h" |
12 | #include "include/private/SkTo.h" |
13 | #include "src/core/SkTraceEvent.h" |
14 | |
15 | #include "zlib.h" |
16 | |
17 | #include <algorithm> |
18 | |
19 | namespace { |
20 | |
21 | // Different zlib implementations use different T. |
22 | // We've seen size_t and unsigned. |
23 | template <typename T> void* skia_alloc_func(void*, T items, T size) { |
24 | return sk_calloc_throw(SkToSizeT(items) * SkToSizeT(size)); |
25 | } |
26 | |
27 | void skia_free_func(void*, void* address) { sk_free(address); } |
28 | |
29 | } // namespace |
30 | |
31 | #define SKDEFLATEWSTREAM_INPUT_BUFFER_SIZE 4096 |
32 | #define SKDEFLATEWSTREAM_OUTPUT_BUFFER_SIZE 4224 // 4096 + 128, usually big |
33 | // enough to always do a |
34 | // single loop. |
35 | |
36 | // called by both write() and finalize() |
37 | static void do_deflate(int flush, |
38 | z_stream* zStream, |
39 | SkWStream* out, |
40 | unsigned char* inBuffer, |
41 | size_t inBufferSize) { |
42 | zStream->next_in = inBuffer; |
43 | zStream->avail_in = SkToInt(inBufferSize); |
44 | unsigned char outBuffer[SKDEFLATEWSTREAM_OUTPUT_BUFFER_SIZE]; |
45 | SkDEBUGCODE(int returnValue;) |
46 | do { |
47 | zStream->next_out = outBuffer; |
48 | zStream->avail_out = sizeof(outBuffer); |
49 | SkDEBUGCODE(returnValue =) deflate(zStream, flush); |
50 | SkASSERT(!zStream->msg); |
51 | |
52 | out->write(outBuffer, sizeof(outBuffer) - zStream->avail_out); |
53 | } while (zStream->avail_in || !zStream->avail_out); |
54 | SkASSERT(flush == Z_FINISH |
55 | ? returnValue == Z_STREAM_END |
56 | : returnValue == Z_OK); |
57 | } |
58 | |
59 | // Hide all zlib impl details. |
60 | struct SkDeflateWStream::Impl { |
61 | SkWStream* fOut; |
62 | unsigned char fInBuffer[SKDEFLATEWSTREAM_INPUT_BUFFER_SIZE]; |
63 | size_t fInBufferIndex; |
64 | z_stream fZStream; |
65 | }; |
66 | |
67 | SkDeflateWStream::SkDeflateWStream(SkWStream* out, |
68 | int compressionLevel, |
69 | bool gzip) |
70 | : fImpl(std::make_unique<SkDeflateWStream::Impl>()) { |
71 | fImpl->fOut = out; |
72 | fImpl->fInBufferIndex = 0; |
73 | if (!fImpl->fOut) { |
74 | return; |
75 | } |
76 | fImpl->fZStream.next_in = nullptr; |
77 | fImpl->fZStream.zalloc = &skia_alloc_func; |
78 | fImpl->fZStream.zfree = &skia_free_func; |
79 | fImpl->fZStream.opaque = nullptr; |
80 | SkASSERT(compressionLevel <= 9 && compressionLevel >= -1); |
81 | SkDEBUGCODE(int r =) deflateInit2(&fImpl->fZStream, compressionLevel, |
82 | Z_DEFLATED, gzip ? 0x1F : 0x0F, |
83 | 8, Z_DEFAULT_STRATEGY); |
84 | SkASSERT(Z_OK == r); |
85 | } |
86 | |
87 | SkDeflateWStream::~SkDeflateWStream() { this->finalize(); } |
88 | |
89 | void SkDeflateWStream::finalize() { |
90 | TRACE_EVENT0("skia" , TRACE_FUNC); |
91 | if (!fImpl->fOut) { |
92 | return; |
93 | } |
94 | do_deflate(Z_FINISH, &fImpl->fZStream, fImpl->fOut, fImpl->fInBuffer, |
95 | fImpl->fInBufferIndex); |
96 | (void)deflateEnd(&fImpl->fZStream); |
97 | fImpl->fOut = nullptr; |
98 | } |
99 | |
100 | bool SkDeflateWStream::write(const void* void_buffer, size_t len) { |
101 | TRACE_EVENT0("skia" , TRACE_FUNC); |
102 | if (!fImpl->fOut) { |
103 | return false; |
104 | } |
105 | const char* buffer = (const char*)void_buffer; |
106 | while (len > 0) { |
107 | size_t tocopy = |
108 | std::min(len, sizeof(fImpl->fInBuffer) - fImpl->fInBufferIndex); |
109 | memcpy(fImpl->fInBuffer + fImpl->fInBufferIndex, buffer, tocopy); |
110 | len -= tocopy; |
111 | buffer += tocopy; |
112 | fImpl->fInBufferIndex += tocopy; |
113 | SkASSERT(fImpl->fInBufferIndex <= sizeof(fImpl->fInBuffer)); |
114 | |
115 | // if the buffer isn't filled, don't call into zlib yet. |
116 | if (sizeof(fImpl->fInBuffer) == fImpl->fInBufferIndex) { |
117 | do_deflate(Z_NO_FLUSH, &fImpl->fZStream, fImpl->fOut, |
118 | fImpl->fInBuffer, fImpl->fInBufferIndex); |
119 | fImpl->fInBufferIndex = 0; |
120 | } |
121 | } |
122 | return true; |
123 | } |
124 | |
125 | size_t SkDeflateWStream::bytesWritten() const { |
126 | return fImpl->fZStream.total_in + fImpl->fInBufferIndex; |
127 | } |
128 | |