| Index: third_party/WebKit/Source/platform/image-decoders/SegmentReader.cpp
|
| diff --git a/third_party/WebKit/Source/platform/image-decoders/SegmentReader.cpp b/third_party/WebKit/Source/platform/image-decoders/SegmentReader.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..4a2e6b243ab985400a650a1aaa0ccc9c42833855
|
| --- /dev/null
|
| +++ b/third_party/WebKit/Source/platform/image-decoders/SegmentReader.cpp
|
| @@ -0,0 +1,200 @@
|
| +// Copyright 2016 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "platform/image-decoders/SegmentReader.h"
|
| +
|
| +#include "platform/SharedBuffer.h"
|
| +#include "third_party/skia/include/core/SkData.h"
|
| +#include "wtf/Assertions.h"
|
| +#include "wtf/Noncopyable.h"
|
| +#include "wtf/PassRefPtr.h"
|
| +#include "wtf/RefPtr.h"
|
| +#include "wtf/ThreadingPrimitives.h"
|
| +
|
| +namespace blink {
|
| +
|
| +// SharedBufferSegmentReader ---------------------------------------------------
|
| +
|
| +// Interface for ImageDecoder to read a SharedBuffer.
|
| +class SharedBufferSegmentReader final : public SegmentReader {
|
| + WTF_MAKE_NONCOPYABLE(SharedBufferSegmentReader);
|
| +public:
|
| + SharedBufferSegmentReader(PassRefPtr<SharedBuffer>);
|
| + size_t size() const override;
|
| + size_t getSomeData(const char*& data, size_t position) const override;
|
| + PassRefPtr<SkData> getAsSkData() const override;
|
| +private:
|
| + RefPtr<SharedBuffer> m_sharedBuffer;
|
| +};
|
| +
|
| +SharedBufferSegmentReader::SharedBufferSegmentReader(PassRefPtr<SharedBuffer> buffer)
|
| + : m_sharedBuffer(buffer) {}
|
| +
|
| +size_t SharedBufferSegmentReader::size() const
|
| +{
|
| + return m_sharedBuffer->size();
|
| +}
|
| +
|
| +size_t SharedBufferSegmentReader::getSomeData(const char*& data, size_t position) const
|
| +{
|
| + return m_sharedBuffer->getSomeData(data, position);
|
| +}
|
| +
|
| +PassRefPtr<SkData> SharedBufferSegmentReader::getAsSkData() const
|
| +{
|
| + return m_sharedBuffer->getAsSkData();
|
| +}
|
| +
|
| +// DataSegmentReader -----------------------------------------------------------
|
| +
|
| +// Interface for ImageDecoder to read an SkData.
|
| +class DataSegmentReader final : public SegmentReader {
|
| + WTF_MAKE_NONCOPYABLE(DataSegmentReader);
|
| +public:
|
| + DataSegmentReader(PassRefPtr<SkData>);
|
| + size_t size() const override;
|
| + size_t getSomeData(const char*& data, size_t position) const override;
|
| + PassRefPtr<SkData> getAsSkData() const override;
|
| +private:
|
| + RefPtr<SkData> m_data;
|
| +};
|
| +
|
| +DataSegmentReader::DataSegmentReader(PassRefPtr<SkData> data)
|
| + : m_data(data) {}
|
| +
|
| +size_t DataSegmentReader::size() const
|
| +{
|
| + return m_data->size();
|
| +}
|
| +
|
| +size_t DataSegmentReader::getSomeData(const char*& data, size_t position) const
|
| +{
|
| + if (position >= m_data->size())
|
| + return 0;
|
| +
|
| + data = reinterpret_cast<const char*>(m_data->bytes() + position);
|
| + return m_data->size() - position;
|
| +}
|
| +
|
| +PassRefPtr<SkData> DataSegmentReader::getAsSkData() const
|
| +{
|
| + return m_data.get();
|
| +}
|
| +
|
| +// ROBufferSegmentReader -------------------------------------------------------
|
| +
|
| +class ROBufferSegmentReader final : public SegmentReader {
|
| + WTF_MAKE_NONCOPYABLE(ROBufferSegmentReader);
|
| +public:
|
| + ROBufferSegmentReader(PassRefPtr<SkROBuffer>);
|
| +
|
| + size_t size() const override;
|
| + size_t getSomeData(const char*& data, size_t position) const override;
|
| + PassRefPtr<SkData> getAsSkData() const override;
|
| +
|
| +private:
|
| + RefPtr<SkROBuffer> m_roBuffer;
|
| + // Protects access to mutable fields.
|
| + mutable Mutex m_readMutex;
|
| + // Position of the first char in the current block of m_iter.
|
| + mutable size_t m_positionOfBlock;
|
| + mutable SkROBuffer::Iter m_iter;
|
| +};
|
| +
|
| +ROBufferSegmentReader::ROBufferSegmentReader(PassRefPtr<SkROBuffer> buffer)
|
| + : m_roBuffer(buffer)
|
| + , m_positionOfBlock(0)
|
| + , m_iter(m_roBuffer.get())
|
| + {}
|
| +
|
| +size_t ROBufferSegmentReader::size() const
|
| +{
|
| + return m_roBuffer ? m_roBuffer->size() : 0;
|
| +}
|
| +
|
| +size_t ROBufferSegmentReader::getSomeData(const char*& data, size_t position) const
|
| +{
|
| + if (!m_roBuffer)
|
| + return 0;
|
| +
|
| + MutexLocker lock(m_readMutex);
|
| +
|
| + if (position < m_positionOfBlock) {
|
| + // SkROBuffer::Iter only iterates forwards. Start from the beginning.
|
| + m_iter.reset(m_roBuffer.get());
|
| + m_positionOfBlock = 0;
|
| + }
|
| +
|
| + for (size_t sizeOfBlock = m_iter.size(); sizeOfBlock != 0; m_positionOfBlock += sizeOfBlock, sizeOfBlock = m_iter.size()) {
|
| + ASSERT(m_positionOfBlock <= position);
|
| +
|
| + if (m_positionOfBlock + sizeOfBlock > position) {
|
| + // |position| is in this block.
|
| + const size_t positionInBlock = position - m_positionOfBlock;
|
| + data = static_cast<const char*>(m_iter.data()) + positionInBlock;
|
| + return sizeOfBlock - positionInBlock;
|
| + }
|
| +
|
| + // Move to next block.
|
| + if (!m_iter.next()) {
|
| + // Reset to the beginning, so future calls can succeed.
|
| + m_iter.reset(m_roBuffer.get());
|
| + m_positionOfBlock = 0;
|
| + return 0;
|
| + }
|
| + }
|
| +
|
| + return 0;
|
| +}
|
| +
|
| +static void unrefROBuffer(const void* ptr, void* context)
|
| +{
|
| + static_cast<SkROBuffer*>(context)->unref();
|
| +}
|
| +
|
| +PassRefPtr<SkData> ROBufferSegmentReader::getAsSkData() const
|
| +{
|
| + if (!m_roBuffer)
|
| + return nullptr;
|
| +
|
| + // Check to see if the data is already contiguous.
|
| + SkROBuffer::Iter iter(m_roBuffer.get());
|
| + const bool multipleBlocks = iter.next();
|
| + iter.reset(m_roBuffer.get());
|
| +
|
| + if (!multipleBlocks) {
|
| + // Contiguous data. No need to copy.
|
| + m_roBuffer->ref();
|
| + return adoptRef(SkData::NewWithProc(iter.data(), iter.size(), &unrefROBuffer, m_roBuffer.get()));
|
| + }
|
| +
|
| + RefPtr<SkData> data = adoptRef(SkData::NewUninitialized(m_roBuffer->size()));
|
| + char* dst = static_cast<char*>(data->writable_data());
|
| + do {
|
| + size_t size = iter.size();
|
| + memcpy(dst, iter.data(), size);
|
| + dst += size;
|
| + } while (iter.next());
|
| + return data.release();
|
| +}
|
| +
|
| +// SegmentReader ---------------------------------------------------------------
|
| +
|
| +PassRefPtr<SegmentReader> SegmentReader::createFromSharedBuffer(PassRefPtr<SharedBuffer> buffer)
|
| +{
|
| + return adoptRef(new SharedBufferSegmentReader(buffer));
|
| +}
|
| +
|
| +PassRefPtr<SegmentReader> SegmentReader::createFromSkData(PassRefPtr<SkData> data)
|
| +{
|
| + return adoptRef(new DataSegmentReader(data));
|
| +}
|
| +
|
| +PassRefPtr<SegmentReader> SegmentReader::createFromSkROBuffer(PassRefPtr<SkROBuffer> buffer)
|
| +{
|
| + return adoptRef(new ROBufferSegmentReader(buffer));
|
| +}
|
| +
|
| +} // namespace blink
|
| +
|
|
|