Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(129)

Side by Side Diff: third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoder.h

Issue 2565323003: Move gif image decoder to SkCodec (Closed)
Patch Set: Zero fill pixel data when allocating Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2006 Apple Computer, Inc. All rights reserved. 2 * Copyright (C) 2006 Apple Computer, Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions 5 * modification, are permitted provided that the following conditions
6 * are met: 6 * are met:
7 * 1. Redistributions of source code must retain the above copyright 7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer. 8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright 9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the 10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution. 11 * documentation and/or other materials provided with the distribution.
12 * 12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY 13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */ 24 */
25 25
26 #ifndef GIFImageDecoder_h 26 #ifndef GIFImageDecoder_h
27 #define GIFImageDecoder_h 27 #define GIFImageDecoder_h
28 28
29 #include <algorithm>
30 #include <memory>
29 #include "platform/image-decoders/ImageDecoder.h" 31 #include "platform/image-decoders/ImageDecoder.h"
32 #include "third_party/skia/include/codec/SkCodec.h"
33 #include "third_party/skia/include/core/SkStream.h"
30 #include "wtf/Noncopyable.h" 34 #include "wtf/Noncopyable.h"
31 #include <memory> 35 #include "wtf/RefPtr.h"
32
33 class GIFImageReader;
34
35 typedef Vector<unsigned char> GIFRow;
36 36
37 namespace blink { 37 namespace blink {
38 38
39 // This class decodes the GIF image format. 39 // This class decodes the GIF image format.
40 class PLATFORM_EXPORT GIFImageDecoder final : public ImageDecoder { 40 class PLATFORM_EXPORT GIFImageDecoder final : public ImageDecoder {
41 WTF_MAKE_NONCOPYABLE(GIFImageDecoder); 41 WTF_MAKE_NONCOPYABLE(GIFImageDecoder);
42 42
43 public: 43 public:
44 GIFImageDecoder(AlphaOption, const ColorBehavior&, size_t maxDecodedBytes); 44 GIFImageDecoder(AlphaOption, const ColorBehavior&, size_t maxDecodedBytes);
45 ~GIFImageDecoder() override; 45 ~GIFImageDecoder() override;
46 46
47 enum GIFParseQuery { GIFSizeQuery, GIFFrameCountQuery };
48
49 // ImageDecoder: 47 // ImageDecoder:
50 String filenameExtension() const override { return "gif"; } 48 String filenameExtension() const override { return "gif"; }
51 void onSetData(SegmentReader* data) override; 49 void onSetData(SegmentReader* data) override;
52 int repetitionCount() const override; 50 int repetitionCount() const override;
53 bool frameIsCompleteAtIndex(size_t) const override; 51 bool frameIsCompleteAtIndex(size_t) const override;
54 float frameDurationAtIndex(size_t) const override; 52 float frameDurationAtIndex(size_t) const override;
55 // CAUTION: setFailed() deletes |m_reader|. Be careful to avoid
56 // accessing deleted memory, especially when calling this from inside
57 // GIFImageReader!
58 bool setFailed() override;
59
60 // Callbacks from the GIF reader.
61 bool haveDecodedRow(size_t frameIndex,
62 GIFRow::const_iterator rowBegin,
63 size_t width,
64 size_t rowNumber,
65 unsigned repeatCount,
66 bool writeTransparentPixels);
67 bool frameComplete(size_t frameIndex);
68
69 // For testing.
70 bool parseCompleted() const;
71 53
72 private: 54 private:
73 // ImageDecoder: 55 // ImageDecoder:
74 void clearFrameBuffer(size_t frameIndex) override; 56 void decodeSize() override {}
75 virtual void decodeSize() { parse(GIFSizeQuery); }
76 size_t decodeFrameCount() override; 57 size_t decodeFrameCount() override;
77 void initializeNewFrame(size_t) override; 58 void initializeNewFrame(size_t index) override;
78 void decode(size_t) override; 59 void decode(size_t) override;
79 60 bool frameStatusSufficientForSuccessors(size_t index) {
80 // Parses as much as is needed to answer the query, ignoring bitmap 61 DCHECK(index < m_frameBufferCache.size());
81 // data. If parsing fails, sets the "decode failure" flag. 62 return m_frameBufferCache[index].getStatus() == ImageFrame::FrameComplete;
82 void parse(GIFParseQuery); 63 }
83
84 // Reset the alpha tracker for this frame. Before calling this method, the
85 // caller must verify that the frame exists.
86 void onInitFrameBuffer(size_t) override;
87
88 // When the disposal method of the frame is DisposeOverWritePrevious, the 64 // When the disposal method of the frame is DisposeOverWritePrevious, the
89 // next frame will use the previous frame's buffer as its starting state, so 65 // next frame will use the previous frame's buffer as its starting state, so
90 // we can't take over the data in that case. Before calling this method, the 66 // we can't take over the data in that case. Before calling this method, the
91 // caller must verify that the frame exists. 67 // caller must verify that the frame exists.
92 bool canReusePreviousFrameBuffer(size_t) const override; 68 bool canReusePreviousFrameBuffer(size_t) const override;
93 69
94 bool m_currentBufferSawAlpha; 70 class SegmentStream : public SkStream {
95 mutable int m_repetitionCount; 71 public:
96 std::unique_ptr<GIFImageReader> m_reader; 72 SegmentStream()
73 : m_reader(),
74 m_position(0),
75 m_hasReadAllContents(true),
76 m_isCleared(true) {}
77
78 void setReader(SegmentReader* reader, bool allContentsReceived) {
79 m_reader = reader;
80 if (reader) {
81 m_hasReadAllContents = reader->size() == m_position;
82 m_isCleared = m_position > m_reader->size();
83 } else {
84 m_hasReadAllContents = true;
85 m_isCleared = true;
86 }
87 }
88
89 size_t read(void* dst, size_t len) {
90 if (m_isCleared)
91 return 0;
92
93 len = std::min(len, m_reader->size() - m_position);
94
95 size_t bytesAdvanced = 0;
96 if (!dst) { // skipping, not reading
97 bytesAdvanced = len;
98 } else {
99 bytesAdvanced = peek(dst, len);
100 }
101
102 m_position += bytesAdvanced;
103 m_hasReadAllContents = m_position == m_reader->size();
104
105 return bytesAdvanced;
106 }
107
108 size_t peek(void* buffer, size_t size) const override {
109 size = std::min(size, m_reader->size() - m_position);
110
111 size_t peekPosition = m_position;
112 size_t totalBytesPeeked = 0;
113 char* bufferAsCharPtr = reinterpret_cast<char*>(buffer);
114 while (size) {
115 const char* segment = nullptr;
116 size_t bytesPeeked = m_reader->getSomeData(segment, peekPosition);
117 if (!bytesPeeked) {
118 break;
119 }
120 if (bytesPeeked > size) {
121 bytesPeeked = size;
122 }
123
124 memcpy(bufferAsCharPtr, segment, bytesPeeked);
125 bufferAsCharPtr += bytesPeeked;
126 size -= bytesPeeked;
127 totalBytesPeeked += bytesPeeked;
128 peekPosition += bytesPeeked;
129 }
130
131 return totalBytesPeeked;
132 }
133
134 bool isAtEnd() const override { return m_hasReadAllContents; }
135
136 bool rewind() override {
137 m_position = 0;
138 return true;
139 }
140
141 bool hasPosition() const override { return true; }
142 size_t getPosition() const override { return m_position; }
143
144 bool seek(size_t position) override {
145 position = std::min(position, m_reader->size());
146 m_position = position;
147
148 return true;
149 }
150
151 bool move(long offset) override {
152 long absolutePosition = m_position + offset;
153
154 // clamp inside the bounds of the buffer size
155 absolutePosition = std::max(absolutePosition, 0l);
156 absolutePosition =
157 std::min(static_cast<size_t>(absolutePosition), m_reader->size());
158
159 m_position = absolutePosition;
160 m_hasReadAllContents = m_position == m_reader->size();
161
162 return true;
163 }
164
165 bool hasLength() const override { return true; }
166 size_t getLength() const override { return m_reader->size(); }
167
168 bool isCleared() const { return m_isCleared; }
169
170 private:
171 WTF::RefPtr<SegmentReader> m_reader;
172 size_t m_position;
173 bool m_hasReadAllContents;
174 bool m_isCleared;
175 };
176
177 std::unique_ptr<SkCodec> m_codec;
178 // m_segmentStream is a raw pointer because it passes ownership to
179 // m_codec when m_codec is created. However, we still need the
180 // pointer so we can append more data as it arrives.
181 SegmentStream* m_segmentStream;
182 std::vector<SkCodec::FrameInfo> m_frameInfos;
97 }; 183 };
98 184
99 } // namespace blink 185 } // namespace blink
100 186
101 #endif 187 #endif
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698