xref: /aosp_15_r20/bootable/recovery/fuse_sideload/fuse_provider.cpp (revision e7c364b630b241adcb6c7726a21055250b91fdac)
1*e7c364b6SAndroid Build Coastguard Worker /*
2*e7c364b6SAndroid Build Coastguard Worker  * Copyright (C) 2019 The Android Open Source Project
3*e7c364b6SAndroid Build Coastguard Worker  *
4*e7c364b6SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*e7c364b6SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*e7c364b6SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*e7c364b6SAndroid Build Coastguard Worker  *
8*e7c364b6SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*e7c364b6SAndroid Build Coastguard Worker  *
10*e7c364b6SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*e7c364b6SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*e7c364b6SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*e7c364b6SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*e7c364b6SAndroid Build Coastguard Worker  * limitations under the License.
15*e7c364b6SAndroid Build Coastguard Worker  */
16*e7c364b6SAndroid Build Coastguard Worker 
17*e7c364b6SAndroid Build Coastguard Worker #include "fuse_provider.h"
18*e7c364b6SAndroid Build Coastguard Worker 
19*e7c364b6SAndroid Build Coastguard Worker #include <errno.h>
20*e7c364b6SAndroid Build Coastguard Worker #include <fcntl.h>
21*e7c364b6SAndroid Build Coastguard Worker #include <inttypes.h>
22*e7c364b6SAndroid Build Coastguard Worker #include <stdio.h>
23*e7c364b6SAndroid Build Coastguard Worker #include <string.h>
24*e7c364b6SAndroid Build Coastguard Worker #include <sys/stat.h>
25*e7c364b6SAndroid Build Coastguard Worker #include <unistd.h>
26*e7c364b6SAndroid Build Coastguard Worker 
27*e7c364b6SAndroid Build Coastguard Worker #include <functional>
28*e7c364b6SAndroid Build Coastguard Worker 
29*e7c364b6SAndroid Build Coastguard Worker #include <android-base/file.h>
30*e7c364b6SAndroid Build Coastguard Worker #include <android-base/logging.h>
31*e7c364b6SAndroid Build Coastguard Worker #include <android-base/strings.h>
32*e7c364b6SAndroid Build Coastguard Worker 
33*e7c364b6SAndroid Build Coastguard Worker #include "fuse_sideload.h"
34*e7c364b6SAndroid Build Coastguard Worker #include "otautil/sysutil.h"
35*e7c364b6SAndroid Build Coastguard Worker 
FuseFileDataProvider(const std::string & path,uint32_t block_size)36*e7c364b6SAndroid Build Coastguard Worker FuseFileDataProvider::FuseFileDataProvider(const std::string& path, uint32_t block_size) {
37*e7c364b6SAndroid Build Coastguard Worker   struct stat sb;
38*e7c364b6SAndroid Build Coastguard Worker   if (stat(path.c_str(), &sb) == -1) {
39*e7c364b6SAndroid Build Coastguard Worker     fprintf(stderr, "failed to stat %s: %s\n", path.c_str(), strerror(errno));
40*e7c364b6SAndroid Build Coastguard Worker     return;
41*e7c364b6SAndroid Build Coastguard Worker   }
42*e7c364b6SAndroid Build Coastguard Worker 
43*e7c364b6SAndroid Build Coastguard Worker   fd_.reset(open(path.c_str(), O_RDONLY));
44*e7c364b6SAndroid Build Coastguard Worker   if (fd_ == -1) {
45*e7c364b6SAndroid Build Coastguard Worker     fprintf(stderr, "failed to open %s: %s\n", path.c_str(), strerror(errno));
46*e7c364b6SAndroid Build Coastguard Worker     return;
47*e7c364b6SAndroid Build Coastguard Worker   }
48*e7c364b6SAndroid Build Coastguard Worker   file_size_ = sb.st_size;
49*e7c364b6SAndroid Build Coastguard Worker   fuse_block_size_ = block_size;
50*e7c364b6SAndroid Build Coastguard Worker }
51*e7c364b6SAndroid Build Coastguard Worker 
CreateFromFile(const std::string & path,uint32_t block_size)52*e7c364b6SAndroid Build Coastguard Worker std::unique_ptr<FuseDataProvider> FuseFileDataProvider::CreateFromFile(const std::string& path,
53*e7c364b6SAndroid Build Coastguard Worker                                                                        uint32_t block_size) {
54*e7c364b6SAndroid Build Coastguard Worker   return std::make_unique<FuseFileDataProvider>(path, block_size);
55*e7c364b6SAndroid Build Coastguard Worker }
56*e7c364b6SAndroid Build Coastguard Worker 
ReadBlockAlignedData(uint8_t * buffer,uint32_t fetch_size,uint32_t start_block) const57*e7c364b6SAndroid Build Coastguard Worker bool FuseFileDataProvider::ReadBlockAlignedData(uint8_t* buffer, uint32_t fetch_size,
58*e7c364b6SAndroid Build Coastguard Worker                                                 uint32_t start_block) const {
59*e7c364b6SAndroid Build Coastguard Worker   uint64_t offset = static_cast<uint64_t>(start_block) * fuse_block_size_;
60*e7c364b6SAndroid Build Coastguard Worker   if (fetch_size > file_size_ || offset > file_size_ - fetch_size) {
61*e7c364b6SAndroid Build Coastguard Worker     fprintf(stderr,
62*e7c364b6SAndroid Build Coastguard Worker             "Out of bound read, start block: %" PRIu32 ", fetch size: %" PRIu32
63*e7c364b6SAndroid Build Coastguard Worker             ", file size %" PRIu64 "\n",
64*e7c364b6SAndroid Build Coastguard Worker             start_block, fetch_size, file_size_);
65*e7c364b6SAndroid Build Coastguard Worker     return false;
66*e7c364b6SAndroid Build Coastguard Worker   }
67*e7c364b6SAndroid Build Coastguard Worker 
68*e7c364b6SAndroid Build Coastguard Worker   if (!android::base::ReadFullyAtOffset(fd_, buffer, fetch_size, offset)) {
69*e7c364b6SAndroid Build Coastguard Worker     fprintf(stderr, "Failed to read fetch size: %" PRIu32 " bytes data at offset %" PRIu64 ": %s\n",
70*e7c364b6SAndroid Build Coastguard Worker             fetch_size, offset, strerror(errno));
71*e7c364b6SAndroid Build Coastguard Worker     return false;
72*e7c364b6SAndroid Build Coastguard Worker   }
73*e7c364b6SAndroid Build Coastguard Worker 
74*e7c364b6SAndroid Build Coastguard Worker   return true;
75*e7c364b6SAndroid Build Coastguard Worker }
76*e7c364b6SAndroid Build Coastguard Worker 
Close()77*e7c364b6SAndroid Build Coastguard Worker void FuseFileDataProvider::Close() {
78*e7c364b6SAndroid Build Coastguard Worker   fd_.reset();
79*e7c364b6SAndroid Build Coastguard Worker }
80*e7c364b6SAndroid Build Coastguard Worker 
FuseBlockDataProvider(uint64_t file_size,uint32_t fuse_block_size,android::base::unique_fd && fd,uint32_t source_block_size,RangeSet ranges)81*e7c364b6SAndroid Build Coastguard Worker FuseBlockDataProvider::FuseBlockDataProvider(uint64_t file_size, uint32_t fuse_block_size,
82*e7c364b6SAndroid Build Coastguard Worker                                              android::base::unique_fd&& fd,
83*e7c364b6SAndroid Build Coastguard Worker                                              uint32_t source_block_size, RangeSet ranges)
84*e7c364b6SAndroid Build Coastguard Worker     : FuseDataProvider(file_size, fuse_block_size),
85*e7c364b6SAndroid Build Coastguard Worker       fd_(std::move(fd)),
86*e7c364b6SAndroid Build Coastguard Worker       source_block_size_(source_block_size),
87*e7c364b6SAndroid Build Coastguard Worker       ranges_(std::move(ranges)) {
88*e7c364b6SAndroid Build Coastguard Worker   // Make sure the offset is also aligned with the blocks on the block device when we call
89*e7c364b6SAndroid Build Coastguard Worker   // ReadBlockAlignedData().
90*e7c364b6SAndroid Build Coastguard Worker   CHECK_EQ(0, fuse_block_size_ % source_block_size_);
91*e7c364b6SAndroid Build Coastguard Worker }
92*e7c364b6SAndroid Build Coastguard Worker 
ReadBlockAlignedData(uint8_t * buffer,uint32_t fetch_size,uint32_t start_block) const93*e7c364b6SAndroid Build Coastguard Worker bool FuseBlockDataProvider::ReadBlockAlignedData(uint8_t* buffer, uint32_t fetch_size,
94*e7c364b6SAndroid Build Coastguard Worker                                                  uint32_t start_block) const {
95*e7c364b6SAndroid Build Coastguard Worker   uint64_t offset = static_cast<uint64_t>(start_block) * fuse_block_size_;
96*e7c364b6SAndroid Build Coastguard Worker   if (fetch_size > file_size_ || offset > file_size_ - fetch_size) {
97*e7c364b6SAndroid Build Coastguard Worker     LOG(ERROR) << "Out of bound read, offset: " << offset << ", fetch size: " << fetch_size
98*e7c364b6SAndroid Build Coastguard Worker                << ", file size " << file_size_;
99*e7c364b6SAndroid Build Coastguard Worker     return false;
100*e7c364b6SAndroid Build Coastguard Worker   }
101*e7c364b6SAndroid Build Coastguard Worker 
102*e7c364b6SAndroid Build Coastguard Worker   auto read_ranges =
103*e7c364b6SAndroid Build Coastguard Worker       ranges_.GetSubRanges(offset / source_block_size_, fetch_size / source_block_size_);
104*e7c364b6SAndroid Build Coastguard Worker   if (!read_ranges) {
105*e7c364b6SAndroid Build Coastguard Worker     return false;
106*e7c364b6SAndroid Build Coastguard Worker   }
107*e7c364b6SAndroid Build Coastguard Worker 
108*e7c364b6SAndroid Build Coastguard Worker   uint8_t* next_out = buffer;
109*e7c364b6SAndroid Build Coastguard Worker   for (const auto& [range_start, range_end] : read_ranges.value()) {
110*e7c364b6SAndroid Build Coastguard Worker     uint64_t bytes_start = static_cast<uint64_t>(range_start) * source_block_size_;
111*e7c364b6SAndroid Build Coastguard Worker     uint64_t bytes_to_read = static_cast<uint64_t>(range_end - range_start) * source_block_size_;
112*e7c364b6SAndroid Build Coastguard Worker     if (!android::base::ReadFullyAtOffset(fd_, next_out, bytes_to_read, bytes_start)) {
113*e7c364b6SAndroid Build Coastguard Worker       PLOG(ERROR) << "Failed to read " << bytes_to_read << " bytes at offset " << bytes_start;
114*e7c364b6SAndroid Build Coastguard Worker       return false;
115*e7c364b6SAndroid Build Coastguard Worker     }
116*e7c364b6SAndroid Build Coastguard Worker 
117*e7c364b6SAndroid Build Coastguard Worker     next_out += bytes_to_read;
118*e7c364b6SAndroid Build Coastguard Worker   }
119*e7c364b6SAndroid Build Coastguard Worker 
120*e7c364b6SAndroid Build Coastguard Worker   if (uint64_t tailing_bytes = fetch_size % source_block_size_; tailing_bytes != 0) {
121*e7c364b6SAndroid Build Coastguard Worker     // Calculate the offset to last partial block. Two possibilities as below:
122*e7c364b6SAndroid Build Coastguard Worker     // 1: fetch_size < source_block_size_, the read_ranges is a blank range_set.
123*e7c364b6SAndroid Build Coastguard Worker     //    Get the last block num through GetBlockNumber() of the offset block.
124*e7c364b6SAndroid Build Coastguard Worker     // 2: fetch_size >= source_block_size_, the last block num is already stored
125*e7c364b6SAndroid Build Coastguard Worker     //    in read-ranges by GetSubRanges() above.
126*e7c364b6SAndroid Build Coastguard Worker     uint64_t tailing_offset =
127*e7c364b6SAndroid Build Coastguard Worker         read_ranges.value()
128*e7c364b6SAndroid Build Coastguard Worker             ? static_cast<uint64_t>((read_ranges->cend() - 1)->second) * source_block_size_
129*e7c364b6SAndroid Build Coastguard Worker             : static_cast<uint64_t>(ranges_.GetBlockNumber(offset / source_block_size_)) *
130*e7c364b6SAndroid Build Coastguard Worker                   source_block_size_;
131*e7c364b6SAndroid Build Coastguard Worker     if (!android::base::ReadFullyAtOffset(fd_, next_out, tailing_bytes, tailing_offset)) {
132*e7c364b6SAndroid Build Coastguard Worker       PLOG(ERROR) << "Failed to read tailing " << tailing_bytes << " bytes at offset "
133*e7c364b6SAndroid Build Coastguard Worker                   << tailing_offset;
134*e7c364b6SAndroid Build Coastguard Worker       return false;
135*e7c364b6SAndroid Build Coastguard Worker     }
136*e7c364b6SAndroid Build Coastguard Worker   }
137*e7c364b6SAndroid Build Coastguard Worker   return true;
138*e7c364b6SAndroid Build Coastguard Worker }
139*e7c364b6SAndroid Build Coastguard Worker 
CreateFromBlockMap(const std::string & block_map_path,uint32_t fuse_block_size)140*e7c364b6SAndroid Build Coastguard Worker std::unique_ptr<FuseDataProvider> FuseBlockDataProvider::CreateFromBlockMap(
141*e7c364b6SAndroid Build Coastguard Worker     const std::string& block_map_path, uint32_t fuse_block_size) {
142*e7c364b6SAndroid Build Coastguard Worker   auto block_map = BlockMapData::ParseBlockMapFile(block_map_path);
143*e7c364b6SAndroid Build Coastguard Worker   if (!block_map) {
144*e7c364b6SAndroid Build Coastguard Worker     return nullptr;
145*e7c364b6SAndroid Build Coastguard Worker   }
146*e7c364b6SAndroid Build Coastguard Worker 
147*e7c364b6SAndroid Build Coastguard Worker   android::base::unique_fd fd(TEMP_FAILURE_RETRY(open(block_map.path().c_str(), O_RDONLY)));
148*e7c364b6SAndroid Build Coastguard Worker   if (fd == -1) {
149*e7c364b6SAndroid Build Coastguard Worker     PLOG(ERROR) << "Failed to open " << block_map.path();
150*e7c364b6SAndroid Build Coastguard Worker     return nullptr;
151*e7c364b6SAndroid Build Coastguard Worker   }
152*e7c364b6SAndroid Build Coastguard Worker 
153*e7c364b6SAndroid Build Coastguard Worker   return std::unique_ptr<FuseBlockDataProvider>(
154*e7c364b6SAndroid Build Coastguard Worker       new FuseBlockDataProvider(block_map.file_size(), fuse_block_size, std::move(fd),
155*e7c364b6SAndroid Build Coastguard Worker                                 block_map.block_size(), block_map.block_ranges()));
156*e7c364b6SAndroid Build Coastguard Worker }
157*e7c364b6SAndroid Build Coastguard Worker 
Close()158*e7c364b6SAndroid Build Coastguard Worker void FuseBlockDataProvider::Close() {
159*e7c364b6SAndroid Build Coastguard Worker   fd_.reset();
160*e7c364b6SAndroid Build Coastguard Worker }
161