1 /*
2 * Copyright (C) 2019 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 #include "otautil/package.h"
18
19 #include <string.h>
20 #include <unistd.h>
21
22 #include <android-base/file.h>
23 #include <android-base/logging.h>
24 #include <android-base/stringprintf.h>
25 #include <android-base/unique_fd.h>
26
27 #include "otautil/error_code.h"
28 #include "otautil/sysutil.h"
29
30 // This class wraps the package in memory, i.e. a memory mapped package, or a package loaded
31 // to a string/vector.
32 class MemoryPackage : public Package {
33 public:
34 // Constructs the class from a file. We will memory maps the file later.
35 MemoryPackage(const std::string& path, std::unique_ptr<MemMapping> map,
36 const std::function<void(float)>& set_progress);
37
38 // Constructs the class from the package bytes in |content|.
39 MemoryPackage(std::vector<uint8_t> content, const std::function<void(float)>& set_progress);
40
41 ~MemoryPackage() override;
42
GetType() const43 PackageType GetType() const override {
44 return PackageType::kMemory;
45 }
46
47 // Memory maps the package file if necessary. Initializes the start address and size of the
48 // package.
GetPackageSize() const49 uint64_t GetPackageSize() const override {
50 return package_size_;
51 }
52
GetPath() const53 std::string GetPath() const override {
54 return path_;
55 }
56
57 bool ReadFullyAtOffset(uint8_t* buffer, uint64_t byte_count, uint64_t offset) override;
58
59 ZipArchiveHandle GetZipArchiveHandle() override;
60
61 bool UpdateHashAtOffset(const std::vector<HasherUpdateCallback>& hashers, uint64_t start,
62 uint64_t length) override;
63
64 private:
65 const uint8_t* addr_; // Start address of the package in memory.
66 uint64_t package_size_; // Package size in bytes.
67
68 // The memory mapped package.
69 std::unique_ptr<MemMapping> map_;
70 // A copy of the package content, valid only if we create the class with the exact bytes of
71 // the package.
72 std::vector<uint8_t> package_content_;
73 // The physical path to the package, empty if we create the class with the package content.
74 std::string path_;
75
76 // The ZipArchiveHandle of the package.
77 ZipArchiveHandle zip_handle_;
78 };
79
SetProgress(float progress)80 void Package::SetProgress(float progress) {
81 if (set_progress_) {
82 set_progress_(progress);
83 }
84 }
85
86 class FilePackage : public Package {
87 public:
88 FilePackage(android::base::unique_fd&& fd, uint64_t file_size, const std::string& path,
89 const std::function<void(float)>& set_progress);
90
91 ~FilePackage() override;
92
GetType() const93 PackageType GetType() const override {
94 return PackageType::kFile;
95 }
96
GetPackageSize() const97 uint64_t GetPackageSize() const override {
98 return package_size_;
99 }
100
GetPath() const101 std::string GetPath() const override {
102 return path_;
103 }
104
105 bool ReadFullyAtOffset(uint8_t* buffer, uint64_t byte_count, uint64_t offset) override;
106
107 ZipArchiveHandle GetZipArchiveHandle() override;
108
109 bool UpdateHashAtOffset(const std::vector<HasherUpdateCallback>& hashers, uint64_t start,
110 uint64_t length) override;
111
112 private:
113 android::base::unique_fd fd_; // The underlying fd to the open package.
114 uint64_t package_size_;
115 std::string path_; // The physical path to the package.
116
117 ZipArchiveHandle zip_handle_;
118 };
119
CreateMemoryPackage(const std::string & path,const std::function<void (float)> & set_progress)120 std::unique_ptr<Package> Package::CreateMemoryPackage(
121 const std::string& path, const std::function<void(float)>& set_progress) {
122 std::unique_ptr<MemMapping> mmap = std::make_unique<MemMapping>();
123 if (!mmap->MapFile(path)) {
124 LOG(ERROR) << "failed to map file";
125 return nullptr;
126 }
127
128 return std::make_unique<MemoryPackage>(path, std::move(mmap), set_progress);
129 }
130
CreateFilePackage(const std::string & path,const std::function<void (float)> & set_progress)131 std::unique_ptr<Package> Package::CreateFilePackage(
132 const std::string& path, const std::function<void(float)>& set_progress) {
133 android::base::unique_fd fd(open(path.c_str(), O_RDONLY));
134 if (fd == -1) {
135 PLOG(ERROR) << "Failed to open " << path;
136 return nullptr;
137 }
138
139 off64_t file_size = lseek64(fd.get(), 0, SEEK_END);
140 if (file_size == -1) {
141 PLOG(ERROR) << "Failed to get the package size";
142 return nullptr;
143 }
144
145 return std::make_unique<FilePackage>(std::move(fd), file_size, path, set_progress);
146 }
147
CreateMemoryPackage(std::vector<uint8_t> content,const std::function<void (float)> & set_progress)148 std::unique_ptr<Package> Package::CreateMemoryPackage(
149 std::vector<uint8_t> content, const std::function<void(float)>& set_progress) {
150 return std::make_unique<MemoryPackage>(std::move(content), set_progress);
151 }
152
MemoryPackage(const std::string & path,std::unique_ptr<MemMapping> map,const std::function<void (float)> & set_progress)153 MemoryPackage::MemoryPackage(const std::string& path, std::unique_ptr<MemMapping> map,
154 const std::function<void(float)>& set_progress)
155 : map_(std::move(map)), path_(path), zip_handle_(nullptr) {
156 addr_ = map_->addr;
157 package_size_ = map_->length;
158 set_progress_ = set_progress;
159 }
160
MemoryPackage(std::vector<uint8_t> content,const std::function<void (float)> & set_progress)161 MemoryPackage::MemoryPackage(std::vector<uint8_t> content,
162 const std::function<void(float)>& set_progress)
163 : package_content_(std::move(content)), zip_handle_(nullptr) {
164 CHECK(!package_content_.empty());
165 addr_ = package_content_.data();
166 package_size_ = package_content_.size();
167 set_progress_ = set_progress;
168 }
169
~MemoryPackage()170 MemoryPackage::~MemoryPackage() {
171 if (zip_handle_) {
172 CloseArchive(zip_handle_);
173 }
174 }
175
ReadFullyAtOffset(uint8_t * buffer,uint64_t byte_count,uint64_t offset)176 bool MemoryPackage::ReadFullyAtOffset(uint8_t* buffer, uint64_t byte_count, uint64_t offset) {
177 if (byte_count > package_size_ || offset > package_size_ - byte_count) {
178 LOG(ERROR) << "Out of bound read, offset: " << offset << ", size: " << byte_count
179 << ", total package_size: " << package_size_;
180 return false;
181 }
182 memcpy(buffer, addr_ + offset, byte_count);
183 return true;
184 }
185
UpdateHashAtOffset(const std::vector<HasherUpdateCallback> & hashers,uint64_t start,uint64_t length)186 bool MemoryPackage::UpdateHashAtOffset(const std::vector<HasherUpdateCallback>& hashers,
187 uint64_t start, uint64_t length) {
188 if (length > package_size_ || start > package_size_ - length) {
189 LOG(ERROR) << "Out of bound read, offset: " << start << ", size: " << length
190 << ", total package_size: " << package_size_;
191 return false;
192 }
193
194 for (const auto& hasher : hashers) {
195 hasher(addr_ + start, length);
196 }
197 return true;
198 }
199
GetZipArchiveHandle()200 ZipArchiveHandle MemoryPackage::GetZipArchiveHandle() {
201 if (zip_handle_) {
202 return zip_handle_;
203 }
204
205 if (auto err = OpenArchiveFromMemory(const_cast<uint8_t*>(addr_), package_size_, path_.c_str(),
206 &zip_handle_);
207 err != 0) {
208 LOG(ERROR) << "Can't open package" << path_ << " : " << ErrorCodeString(err);
209 return nullptr;
210 }
211
212 return zip_handle_;
213 }
214
FilePackage(android::base::unique_fd && fd,uint64_t file_size,const std::string & path,const std::function<void (float)> & set_progress)215 FilePackage::FilePackage(android::base::unique_fd&& fd, uint64_t file_size, const std::string& path,
216 const std::function<void(float)>& set_progress)
217 : fd_(std::move(fd)), package_size_(file_size), path_(path), zip_handle_(nullptr) {
218 set_progress_ = set_progress;
219 }
220
~FilePackage()221 FilePackage::~FilePackage() {
222 if (zip_handle_) {
223 CloseArchive(zip_handle_);
224 }
225 }
226
ReadFullyAtOffset(uint8_t * buffer,uint64_t byte_count,uint64_t offset)227 bool FilePackage::ReadFullyAtOffset(uint8_t* buffer, uint64_t byte_count, uint64_t offset) {
228 if (byte_count > package_size_ || offset > package_size_ - byte_count) {
229 LOG(ERROR) << "Out of bound read, offset: " << offset << ", size: " << byte_count
230 << ", total package_size: " << package_size_;
231 return false;
232 }
233
234 if (!android::base::ReadFullyAtOffset(fd_.get(), buffer, byte_count, offset)) {
235 PLOG(ERROR) << "Failed to read " << byte_count << " bytes data at offset " << offset;
236 return false;
237 }
238
239 return true;
240 }
241
UpdateHashAtOffset(const std::vector<HasherUpdateCallback> & hashers,uint64_t start,uint64_t length)242 bool FilePackage::UpdateHashAtOffset(const std::vector<HasherUpdateCallback>& hashers,
243 uint64_t start, uint64_t length) {
244 if (length > package_size_ || start > package_size_ - length) {
245 LOG(ERROR) << "Out of bound read, offset: " << start << ", size: " << length
246 << ", total package_size: " << package_size_;
247 return false;
248 }
249
250 uint64_t so_far = 0;
251 while (so_far < length) {
252 uint64_t read_size = std::min<uint64_t>(length - so_far, 16 * MiB);
253 std::vector<uint8_t> buffer(read_size);
254 if (!ReadFullyAtOffset(buffer.data(), read_size, start + so_far)) {
255 return false;
256 }
257
258 for (const auto& hasher : hashers) {
259 hasher(buffer.data(), read_size);
260 }
261 so_far += read_size;
262 }
263
264 return true;
265 }
266
GetZipArchiveHandle()267 ZipArchiveHandle FilePackage::GetZipArchiveHandle() {
268 if (zip_handle_) {
269 return zip_handle_;
270 }
271
272 if (auto err = OpenArchiveFd(fd_.get(), path_.c_str(), &zip_handle_, false); err != 0) {
273 LOG(ERROR) << "Can't open package" << path_ << " : " << ErrorCodeString(err);
274 return nullptr;
275 }
276
277 return zip_handle_;
278 }
279