/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef AAPT_IO_DATA_H #define AAPT_IO_DATA_H #include #include "android-base/macros.h" #include "utils/FileMap.h" #include "io/Io.h" namespace aapt { namespace io { // Interface for a block of contiguous memory. An instance of this interface owns the data. class IData : public KnownSizeInputStream { public: virtual ~IData() = default; virtual const void* data() const = 0; virtual size_t size() const = 0; virtual size_t TotalSize() const override { return size(); } }; class DataSegment : public IData { public: explicit DataSegment(std::unique_ptr data, size_t offset, size_t len) : data_(std::move(data)), offset_(offset), len_(len), next_read_(offset) {} virtual ~DataSegment() = default; const void* data() const override { return static_cast(data_->data()) + offset_; } size_t size() const override { return len_; } bool Next(const void** data, size_t* size) override { if (next_read_ == offset_ + len_) { return false; } *data = static_cast(data_->data()) + next_read_; *size = len_ - (next_read_ - offset_); next_read_ = offset_ + len_; return true; } void BackUp(size_t count) override { if (count > next_read_ - offset_) { next_read_ = offset_; } else { next_read_ -= count; } } bool CanRewind() const override { return true; } bool Rewind() override { next_read_ = offset_; return true; } size_t ByteCount() const override { return next_read_ - offset_; } bool HadError() const override { return false; } private: DISALLOW_COPY_AND_ASSIGN(DataSegment); std::unique_ptr data_; size_t offset_; size_t len_; size_t next_read_; }; // Implementation of IData that exposes a memory mapped file. // The mmapped file is owned by this object. class MmappedData : public IData { public: explicit MmappedData(android::FileMap&& map) : map_(std::forward(map)) {} virtual ~MmappedData() = default; const void* data() const override { return map_.getDataPtr(); } size_t size() const override { return map_.getDataLength(); } bool Next(const void** data, size_t* size) override { if (next_read_ == map_.getDataLength()) { return false; } *data = reinterpret_cast(map_.getDataPtr()) + next_read_; *size = map_.getDataLength() - next_read_; next_read_ = map_.getDataLength(); return true; } void BackUp(size_t count) override { if (count > next_read_) { next_read_ = 0; } else { next_read_ -= count; } } bool CanRewind() const override { return true; } bool Rewind() override { next_read_ = 0; return true; } size_t ByteCount() const override { return next_read_; } bool HadError() const override { return false; } private: DISALLOW_COPY_AND_ASSIGN(MmappedData); android::FileMap map_; size_t next_read_ = 0; }; // Implementation of IData that exposes a block of memory that was malloc'ed (new'ed). // The memory is owned by this object. class MallocData : public IData { public: MallocData(std::unique_ptr data, size_t size) : data_(std::move(data)), size_(size) {} virtual ~MallocData() = default; const void* data() const override { return data_.get(); } size_t size() const override { return size_; } bool Next(const void** data, size_t* size) override { if (next_read_ == size_) { return false; } *data = data_.get() + next_read_; *size = size_ - next_read_; next_read_ = size_; return true; } void BackUp(size_t count) override { if (count > next_read_) { next_read_ = 0; } else { next_read_ -= count; } } bool CanRewind() const override { return true; } bool Rewind() override { next_read_ = 0; return true; } size_t ByteCount() const override { return next_read_; } bool HadError() const override { return false; } private: DISALLOW_COPY_AND_ASSIGN(MallocData); std::unique_ptr data_; size_t size_; size_t next_read_ = 0; }; // When mmap fails because the file has length 0, we use the EmptyData to simulate data of length 0. class EmptyData : public IData { public: virtual ~EmptyData() = default; const void* data() const override { static const uint8_t d = 0; return &d; } size_t size() const override { return 0u; } bool Next(const void** /*data*/, size_t* /*size*/) override { return false; } void BackUp(size_t /*count*/) override {} bool CanRewind() const override { return true; } bool Rewind() override { return true; } size_t ByteCount() const override { return 0u; } bool HadError() const override { return false; } }; } // namespace io } // namespace aapt #endif /* AAPT_IO_DATA_H */