| Index: pdf/chunk_stream.h
|
| diff --git a/pdf/chunk_stream.h b/pdf/chunk_stream.h
|
| index d2d8d2a13d4a4343bc5acadb263e1a5c0b83bd09..f25baf9dbebcd7d2cb94fa850eedd90ece501c66 100644
|
| --- a/pdf/chunk_stream.h
|
| +++ b/pdf/chunk_stream.h
|
| @@ -6,46 +6,103 @@
|
| #define PDF_CHUNK_STREAM_H_
|
|
|
| #include <stddef.h>
|
| +#include <string.h>
|
|
|
| -#include <map>
|
| -#include <utility>
|
| +#include <algorithm>
|
| +#include <array>
|
| +#include <memory>
|
| #include <vector>
|
|
|
| +#include "pdf/range_set.h"
|
| +
|
| namespace chrome_pdf {
|
|
|
| // This class collects a chunks of data into one data stream. Client can check
|
| // if data in certain range is available, and get missing chunks of data.
|
| +template <size_t N>
|
| class ChunkStream {
|
| public:
|
| - ChunkStream();
|
| - ~ChunkStream();
|
| + static const int kChunkSize = N;
|
| + typedef typename std::array<unsigned char, N> ChunkData;
|
|
|
| - void Clear();
|
| + ChunkStream() {}
|
| + ~ChunkStream() {}
|
|
|
| - void Preallocate(size_t stream_size);
|
| - size_t GetSize() const;
|
| + void SetChunkData(int chunk_index, std::unique_ptr<ChunkData> data) {
|
| + if (!data)
|
| + return;
|
| + if (chunk_index >= static_cast<int>(data_.size())) {
|
| + data_.resize(chunk_index + 1);
|
| + }
|
| + if (!data_[chunk_index]) {
|
| + ++filled_chunks_count_;
|
| + }
|
| + data_[chunk_index] = std::move(data);
|
| + filled_chunks_.Union(gfx::Range(chunk_index, chunk_index + 1));
|
| + }
|
|
|
| - bool WriteData(size_t offset, void* buffer, size_t size);
|
| - bool ReadData(size_t offset, size_t size, void* buffer) const;
|
| + bool ReadData(const gfx::Range& range, void* buffer) const {
|
| + if (!IsRangeAvailable(range)) {
|
| + return false;
|
| + }
|
| + unsigned char* data_buffer = static_cast<unsigned char*>(buffer);
|
| + uint32_t start = range.start();
|
| + while (start != range.end()) {
|
| + const uint32_t chunk_index = GetChunkIndex(start);
|
| + const uint32_t chunk_start = start % kChunkSize;
|
| + const uint32_t len =
|
| + std::min(kChunkSize - chunk_start, range.end() - start);
|
| + memcpy(data_buffer, data_[chunk_index]->data() + chunk_start, len);
|
| + data_buffer += len;
|
| + start += len;
|
| + }
|
| + return true;
|
| + }
|
|
|
| - // Returns vector of pairs where first is an offset, second is a size.
|
| - bool GetMissedRanges(size_t offset, size_t size,
|
| - std::vector<std::pair<size_t, size_t> >* ranges) const;
|
| - bool IsRangeAvailable(size_t offset, size_t size) const;
|
| - size_t GetFirstMissingByte() const;
|
| + uint32_t GetChunkIndex(uint32_t offset) const { return offset / kChunkSize; }
|
|
|
| - // Finds the first byte of the missing byte interval that offset belongs to.
|
| - size_t GetFirstMissingByteInInterval(size_t offset) const;
|
| - // Returns the last byte of the missing byte interval that offset belongs to.
|
| - size_t GetLastMissingByteInInterval(size_t offset) const;
|
| + gfx::Range GetChunksRange(uint32_t offset, uint32_t size) const {
|
| + return gfx::Range(GetChunkIndex(offset),
|
| + GetChunkIndex(offset + size + kChunkSize - 1));
|
| + }
|
|
|
| - private:
|
| - std::vector<unsigned char> data_;
|
| + bool IsRangeAvailable(const gfx::Range& range) const {
|
| + if (!range.IsValid() || range.is_reversed() ||
|
| + (eof_pos_ && eof_pos_ < range.end()))
|
| + return false;
|
| + if (range.is_empty())
|
| + return true;
|
| + const gfx::Range chunks_range(GetChunkIndex(range.start()),
|
| + GetChunkIndex(range.end() + kChunkSize - 1));
|
| + return filled_chunks_.Contains(chunks_range);
|
| + }
|
| +
|
| + void set_eof_pos(uint32_t eof_pos) { eof_pos_ = eof_pos; }
|
| + uint32_t eof_pos() const { return eof_pos_; }
|
| +
|
| + const RangeSet& filled_chunks() const { return filled_chunks_; }
|
|
|
| - // Pair, first - begining of the chunk, second - size of the chunk.
|
| - std::map<size_t, size_t> chunks_;
|
| + bool IsComplete() const {
|
| + return eof_pos_ > 0 && IsRangeAvailable(gfx::Range(0, eof_pos_));
|
| + }
|
|
|
| - size_t stream_size_;
|
| + void Clear() {
|
| + data_.clear();
|
| + eof_pos_ = 0;
|
| + filled_chunks_.Clear();
|
| + filled_chunks_count_ = 0;
|
| + }
|
| +
|
| + uint32_t filled_chunks_count() const { return filled_chunks_count_; }
|
| + uint32_t total_chunks_count() const {
|
| + return GetChunkIndex(eof_pos_ + kChunkSize - 1);
|
| + }
|
| +
|
| + private:
|
| + std::vector<std::unique_ptr<ChunkData>> data_;
|
| + uint32_t eof_pos_ = 0;
|
| + RangeSet filled_chunks_;
|
| + uint32_t filled_chunks_count_ = 0;
|
| };
|
|
|
| }; // namespace chrome_pdf
|
|
|