| /* | 
 |  * 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 <memory> | 
 |  | 
 | #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<IData> 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<const uint8_t*>(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<const uint8_t*>(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<IData> 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<android::FileMap>(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<const uint8_t*>(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<const uint8_t[]> 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<const uint8_t[]> 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 */ |