1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef AAPT_IO_DATA_H
18 #define AAPT_IO_DATA_H
19 
20 #include <memory>
21 
22 #include "android-base/macros.h"
23 #include "utils/FileMap.h"
24 
25 #include "io/Io.h"
26 
27 namespace aapt {
28 namespace io {
29 
30 // Interface for a block of contiguous memory. An instance of this interface owns the data.
31 class IData : public InputStream {
32  public:
33   virtual ~IData() = default;
34 
35   virtual const void* data() const = 0;
36   virtual size_t size() const = 0;
37 };
38 
39 class DataSegment : public IData {
40  public:
DataSegment(std::unique_ptr<IData> data,size_t offset,size_t len)41   explicit DataSegment(std::unique_ptr<IData> data, size_t offset, size_t len)
42       : data_(std::move(data)), offset_(offset), len_(len), next_read_(offset) {}
43   virtual ~DataSegment() = default;
44 
data()45   const void* data() const override {
46     return static_cast<const uint8_t*>(data_->data()) + offset_;
47   }
48 
size()49   size_t size() const override { return len_; }
50 
Next(const void ** data,size_t * size)51   bool Next(const void** data, size_t* size) override {
52     if (next_read_ == offset_ + len_) {
53       return false;
54     }
55     *data = static_cast<const uint8_t*>(data_->data()) + next_read_;
56     *size = len_ - (next_read_ - offset_);
57     next_read_ = offset_ + len_;
58     return true;
59   }
60 
BackUp(size_t count)61   void BackUp(size_t count) override {
62     if (count > next_read_ - offset_) {
63       next_read_ = offset_;
64     } else {
65       next_read_ -= count;
66     }
67   }
68 
CanRewind()69   bool CanRewind() const override { return true; }
70 
Rewind()71   bool Rewind() override {
72     next_read_ = offset_;
73     return true;
74   }
75 
ByteCount()76   size_t ByteCount() const override { return next_read_ - offset_; }
77 
HadError()78   bool HadError() const override { return false; }
79 
80  private:
81   DISALLOW_COPY_AND_ASSIGN(DataSegment);
82 
83   std::unique_ptr<IData> data_;
84   size_t offset_;
85   size_t len_;
86   size_t next_read_;
87 };
88 
89 // Implementation of IData that exposes a memory mapped file.
90 // The mmapped file is owned by this object.
91 class MmappedData : public IData {
92  public:
MmappedData(android::FileMap && map)93   explicit MmappedData(android::FileMap&& map) : map_(std::forward<android::FileMap>(map)) {}
94   virtual ~MmappedData() = default;
95 
data()96   const void* data() const override { return map_.getDataPtr(); }
97 
size()98   size_t size() const override { return map_.getDataLength(); }
99 
Next(const void ** data,size_t * size)100   bool Next(const void** data, size_t* size) override {
101     if (next_read_ == map_.getDataLength()) {
102       return false;
103     }
104     *data = reinterpret_cast<const uint8_t*>(map_.getDataPtr()) + next_read_;
105     *size = map_.getDataLength() - next_read_;
106     next_read_ = map_.getDataLength();
107     return true;
108   }
109 
BackUp(size_t count)110   void BackUp(size_t count) override {
111     if (count > next_read_) {
112       next_read_ = 0;
113     } else {
114       next_read_ -= count;
115     }
116   }
117 
CanRewind()118   bool CanRewind() const override { return true; }
119 
Rewind()120   bool Rewind() override {
121     next_read_ = 0;
122     return true;
123   }
124 
ByteCount()125   size_t ByteCount() const override { return next_read_; }
126 
HadError()127   bool HadError() const override { return false; }
128 
129  private:
130   DISALLOW_COPY_AND_ASSIGN(MmappedData);
131 
132   android::FileMap map_;
133   size_t next_read_ = 0;
134 };
135 
136 // Implementation of IData that exposes a block of memory that was malloc'ed (new'ed).
137 // The memory is owned by this object.
138 class MallocData : public IData {
139  public:
MallocData(std::unique_ptr<const uint8_t[]> data,size_t size)140   MallocData(std::unique_ptr<const uint8_t[]> data, size_t size)
141       : data_(std::move(data)), size_(size) {}
142   virtual ~MallocData() = default;
143 
data()144   const void* data() const override { return data_.get(); }
145 
size()146   size_t size() const override { return size_; }
147 
Next(const void ** data,size_t * size)148   bool Next(const void** data, size_t* size) override {
149     if (next_read_ == size_) {
150       return false;
151     }
152     *data = data_.get() + next_read_;
153     *size = size_ - next_read_;
154     next_read_ = size_;
155     return true;
156   }
157 
BackUp(size_t count)158   void BackUp(size_t count) override {
159     if (count > next_read_) {
160       next_read_ = 0;
161     } else {
162       next_read_ -= count;
163     }
164   }
165 
CanRewind()166   bool CanRewind() const override { return true; }
167 
Rewind()168   bool Rewind() override {
169     next_read_ = 0;
170     return true;
171   }
172 
ByteCount()173   size_t ByteCount() const override { return next_read_; }
174 
HadError()175   bool HadError() const override { return false; }
176 
177  private:
178   DISALLOW_COPY_AND_ASSIGN(MallocData);
179 
180   std::unique_ptr<const uint8_t[]> data_;
181   size_t size_;
182   size_t next_read_ = 0;
183 };
184 
185 // When mmap fails because the file has length 0, we use the EmptyData to simulate data of length 0.
186 class EmptyData : public IData {
187  public:
188   virtual ~EmptyData() = default;
189 
data()190   const void* data() const override {
191     static const uint8_t d = 0;
192     return &d;
193   }
194 
size()195   size_t size() const override { return 0u; }
196 
Next(const void **,size_t *)197   bool Next(const void** /*data*/, size_t* /*size*/) override { return false; }
198 
BackUp(size_t)199   void BackUp(size_t /*count*/) override {}
200 
CanRewind()201   bool CanRewind() const override { return true; }
202 
Rewind()203   bool Rewind() override { return true; }
204 
ByteCount()205   size_t ByteCount() const override { return 0u; }
206 
HadError()207   bool HadError() const override { return false; }
208 };
209 
210 }  // namespace io
211 }  // namespace aapt
212 
213 #endif /* AAPT_IO_DATA_H */
214