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

Side by Side Diff: third_party/WebKit/Source/platform/image-decoders/SegmentReader.cpp

Issue 1812273003: Eliminate copies of encoded image data (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Respond to pkasting's comments in ps 18 Created 4 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
(Empty)
1 // Copyright 2016 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "platform/image-decoders/SegmentReader.h"
6
7 #include "platform/SharedBuffer.h"
8 #include "third_party/skia/include/core/SkData.h"
9 #include "wtf/Assertions.h"
10 #include "wtf/Noncopyable.h"
11 #include "wtf/PassRefPtr.h"
12 #include "wtf/RefPtr.h"
13
14 namespace blink {
15
16 // SharedBufferSegmentReader ---------------------------------------------------
17
18 // Interface for ImageDecoder to read a SharedBuffer.
19 class SharedBufferSegmentReader final : public SegmentReader {
20 WTF_MAKE_NONCOPYABLE(SharedBufferSegmentReader);
21 public:
22 SharedBufferSegmentReader(PassRefPtr<SharedBuffer>);
23 size_t size() const override;
24 size_t getSomeData(const char*& data, size_t position) const override;
25 PassRefPtr<SkData> getAsSkData() const override;
26 private:
27 RefPtr<SharedBuffer> m_sharedBuffer;
28 };
29
30 SharedBufferSegmentReader::SharedBufferSegmentReader(PassRefPtr<SharedBuffer> bu ffer)
31 : m_sharedBuffer(buffer) {}
32
33 size_t SharedBufferSegmentReader::size() const
34 {
35 return m_sharedBuffer->size();
36 }
37
38 size_t SharedBufferSegmentReader::getSomeData(const char*& data, size_t position ) const
39 {
40 return m_sharedBuffer->getSomeData(data, position);
41 }
42
43 PassRefPtr<SkData> SharedBufferSegmentReader::getAsSkData() const
44 {
45 return m_sharedBuffer->getAsSkData();
46 }
47
48 // DataSegmentReader -----------------------------------------------------------
49
50 // Interface for ImageDecoder to read an SkData.
51 class DataSegmentReader final : public SegmentReader {
52 WTF_MAKE_NONCOPYABLE(DataSegmentReader);
53 public:
54 DataSegmentReader(PassRefPtr<SkData>);
55 size_t size() const override;
56 size_t getSomeData(const char*& data, size_t position) const override;
57 PassRefPtr<SkData> getAsSkData() const override;
58 private:
59 RefPtr<SkData> m_data;
60 };
61
62 DataSegmentReader::DataSegmentReader(PassRefPtr<SkData> data)
63 : m_data(data) {}
64
65 size_t DataSegmentReader::size() const
66 {
67 return m_data->size();
68 }
69
70 size_t DataSegmentReader::getSomeData(const char*& data, size_t position) const
71 {
72 if (position >= m_data->size())
73 return 0;
74
75 data = reinterpret_cast<const char*>(m_data->bytes() + position);
76 return m_data->size() - position;
77 }
78
79 PassRefPtr<SkData> DataSegmentReader::getAsSkData() const
80 {
81 return m_data.get();
82 }
83
84 // ROBufferSegmentReader -------------------------------------------------------
85
86 class ROBufferSegmentReader final : public SegmentReader {
87 WTF_MAKE_NONCOPYABLE(ROBufferSegmentReader);
88 public:
89 ROBufferSegmentReader(PassRefPtr<SkROBuffer>);
90
91 size_t size() const override;
92 size_t getSomeData(const char*& data, size_t position) const override;
93 PassRefPtr<SkData> getAsSkData() const override;
94
95 private:
96 RefPtr<SkROBuffer> m_roBuffer;
97 // Position of the first char in the current block of m_iter.
98 mutable size_t m_positionOfBlock;
99 mutable SkROBuffer::Iter m_iter;
100 };
101
102 ROBufferSegmentReader::ROBufferSegmentReader(PassRefPtr<SkROBuffer> buffer)
103 : m_roBuffer(buffer)
104 , m_positionOfBlock(0)
105 , m_iter(m_roBuffer.get())
106 {}
107
108 size_t ROBufferSegmentReader::size() const
109 {
110 return m_roBuffer ? m_roBuffer->size() : 0;
111 }
112
113 size_t ROBufferSegmentReader::getSomeData(const char*& data, size_t position) co nst
114 {
115 if (!m_roBuffer)
116 return 0;
117
118 if (position < m_positionOfBlock) {
119 // SkROBuffer::Iter only iterates forwards. Start from the
120 // beginning.
Peter Kasting 2016/03/26 01:53:03 Nit: No need to linebreak comment
scroggo_chromium 2016/04/04 13:59:47 Done.
121 m_iter.reset(m_roBuffer.get());
122 m_positionOfBlock = 0;
123 }
124
125 for (size_t sizeOfBlock = m_iter.size(); sizeOfBlock != 0; m_positionOfBlock += sizeOfBlock) {
126 ASSERT(m_positionOfBlock <= position);
127
128 if (m_positionOfBlock + sizeOfBlock > position) {
129 // |position| is in this block.
130 const size_t positionInBlock = position - m_positionOfBlock;
131 data = static_cast<const char*>(m_iter.data()) + positionInBlock;
132 return sizeOfBlock - positionInBlock;
133 }
134
135 // Move to next block.
136 if (!m_iter.next()) {
137 // Reset to the beginning, so future calls can succeed.
138 m_iter.reset(m_roBuffer.get());
139 m_positionOfBlock = 0;
140 return 0;
141 }
142
Peter Kasting 2016/03/26 01:53:03 Nit: Extra blank line
scroggo_chromium 2016/04/04 13:59:47 Done.
143 }
144
145 return 0;
146 }
147
148 static void unrefROBuffer(const void* ptr, void* context)
149 {
150 static_cast<SkROBuffer*>(context)->unref();
151 }
152
153 PassRefPtr<SkData> ROBufferSegmentReader::getAsSkData() const
154 {
155 if (!m_roBuffer)
156 return nullptr;
157
158 // Check to see if the data is already contiguous.
159 SkROBuffer::Iter iter(m_roBuffer.get());
160 const bool multipleBlocks = iter.next();
161 iter.reset(m_roBuffer.get());
162
163 if (!multipleBlocks) {
164 // Contiguous data. No need to copy.
165 m_roBuffer->ref();
166 return adoptRef(SkData::NewWithProc(iter.data(), iter.size(), &unrefROBu ffer, m_roBuffer.get()));
167 }
168
169 RefPtr<SkData> data = adoptRef(SkData::NewUninitialized(m_roBuffer->size())) ;
170 char* dst = static_cast<char*>(data->writable_data());
171 do {
172 size_t size = iter.size();
173 memcpy(dst, iter.data(), size);
174 dst += size;
175 } while (iter.next());
176 return data.release();
177 }
178
179 // SegmentReader ---------------------------------------------------------------
180
181 PassRefPtr<SegmentReader> SegmentReader::createFromSharedBuffer(PassRefPtr<Share dBuffer> buffer)
182 {
183 return adoptRef(new SharedBufferSegmentReader(buffer));
184 }
185
186 PassRefPtr<SegmentReader> SegmentReader::createFromSkData(PassRefPtr<SkData> dat a)
187 {
188 return adoptRef(new DataSegmentReader(data));
189 }
190
191 PassRefPtr<SegmentReader> SegmentReader::createFromSkROBuffer(PassRefPtr<SkROBuf fer> buffer)
192 {
193 return adoptRef(new ROBufferSegmentReader(buffer));
194 }
195
196 } // namespace blink
197
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698