blob: a62125e4f44ead8948bd884e404d6fa5828723b3 [file] [log] [blame]
Hridya Valsaraju31d2c262018-07-20 13:35:50 -07001/*
2 * Copyright (C) 2018 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#pragma once
17
David Anderson12211d12018-07-24 15:21:20 -070018#include <optional>
Hridya Valsaraju31d2c262018-07-20 13:35:50 -070019#include <string>
20
Konstantin Vyshetsky1cee2ed2022-03-17 17:57:28 -070021#include <android-base/file.h>
22#include <android-base/logging.h>
David Anderson12211d12018-07-24 15:21:20 -070023#include <android-base/unique_fd.h>
David Anderson23243492019-12-17 00:58:31 -080024#include <fstab/fstab.h>
David Andersond25f1c32018-11-09 20:41:33 -080025#include <liblp/liblp.h>
Hridya Valsaraju31d2c262018-07-20 13:35:50 -070026
David Anderson12211d12018-07-24 15:21:20 -070027// Logical partitions are only mapped to a block device as needed, and
28// immediately unmapped when no longer needed. In order to enforce this we
29// require accessing partitions through a Handle abstraction, which may perform
30// additional operations after closing its file descriptor.
31class PartitionHandle {
32 public:
33 PartitionHandle() {}
34 explicit PartitionHandle(const std::string& path) : path_(path) {}
David Anderson88ef0b12018-08-09 10:40:00 -070035 PartitionHandle(const std::string& path, std::function<void()>&& closer)
36 : path_(path), closer_(std::move(closer)) {}
37 PartitionHandle(PartitionHandle&& other) = default;
38 PartitionHandle& operator=(PartitionHandle&& other) = default;
39 ~PartitionHandle() {
40 if (closer_) {
41 // Make sure the device is closed first.
42 fd_ = {};
43 closer_();
44 }
45 }
David Anderson12211d12018-07-24 15:21:20 -070046 const std::string& path() const { return path_; }
47 int fd() const { return fd_.get(); }
Konstantin Vyshetsky1cee2ed2022-03-17 17:57:28 -070048 bool Open(int flags) {
49 flags |= (O_EXCL | O_CLOEXEC | O_BINARY);
David Anderson12211d12018-07-24 15:21:20 -070050
Konstantin Vyshetsky1cee2ed2022-03-17 17:57:28 -070051 // Attempts to open a second device can fail with EBUSY if the device is already open.
52 // Explicitly close any previously opened devices as unique_fd won't close them until
53 // after the attempt to open.
54 fd_.reset();
55
56 fd_ = android::base::unique_fd(TEMP_FAILURE_RETRY(open(path_.c_str(), flags)));
57 if (fd_ < 0) {
58 PLOG(ERROR) << "Failed to open block device: " << path_;
59 return false;
60 }
61 flags_ = flags;
62
63 return true;
64 }
65 bool Reset(int flags) {
66 if (fd_.ok() && (flags | O_EXCL | O_CLOEXEC | O_BINARY) == flags_) {
67 return true;
68 }
69
70 off_t offset = fd_.ok() ? lseek(fd_.get(), 0, SEEK_CUR) : 0;
71 if (offset < 0) {
72 PLOG(ERROR) << "Failed lseek on block device: " << path_;
73 return false;
74 }
75
76 sync();
77
78 if (Open(flags) == false) {
79 return false;
80 }
81
82 if (lseek(fd_.get(), offset, SEEK_SET) != offset) {
83 PLOG(ERROR) << "Failed lseek on block device: " << path_;
84 return false;
85 }
86
87 return true;
88 }
David Anderson12211d12018-07-24 15:21:20 -070089 private:
90 std::string path_;
91 android::base::unique_fd fd_;
Konstantin Vyshetsky1cee2ed2022-03-17 17:57:28 -070092 int flags_;
David Anderson88ef0b12018-08-09 10:40:00 -070093 std::function<void()> closer_;
David Anderson12211d12018-07-24 15:21:20 -070094};
95
David Anderson23243492019-12-17 00:58:31 -080096class AutoMountMetadata {
97 public:
98 AutoMountMetadata();
99 ~AutoMountMetadata();
100 explicit operator bool() const { return mounted_; }
101
102 private:
103 android::fs_mgr::Fstab fstab_;
104 bool mounted_ = false;
105 bool should_unmount_ = false;
106};
107
David Anderson12211d12018-07-24 15:21:20 -0700108class FastbootDevice;
109
David Andersond25f1c32018-11-09 20:41:33 -0800110// On normal devices, the super partition is always named "super". On retrofit
111// devices, the name must be derived from the partition name or current slot.
112// This helper assists in choosing the correct super for a given partition
113// name.
114std::string GetSuperSlotSuffix(FastbootDevice* device, const std::string& partition_name);
115
David Anderson12211d12018-07-24 15:21:20 -0700116std::optional<std::string> FindPhysicalPartition(const std::string& name);
David Andersond25f1c32018-11-09 20:41:33 -0800117bool LogicalPartitionExists(FastbootDevice* device, const std::string& name,
David Anderson88ef0b12018-08-09 10:40:00 -0700118 bool* is_zero_length = nullptr);
Yifan Hongc4073d32021-02-18 12:35:42 -0800119
Konstantin Vyshetsky81cc1192021-11-04 10:27:06 -0700120// Partition is O_WRONLY by default, caller should pass O_RDONLY for reading.
121// Caller may pass additional flags if needed. (O_EXCL | O_CLOEXEC | O_BINARY)
122// will be logically ORed internally.
Yifan Hongc4073d32021-02-18 12:35:42 -0800123bool OpenPartition(FastbootDevice* device, const std::string& name, PartitionHandle* handle,
Konstantin Vyshetsky81cc1192021-11-04 10:27:06 -0700124 int flags = O_WRONLY);
Yifan Hongc4073d32021-02-18 12:35:42 -0800125
Kelvin Zhang6befe782022-06-21 14:20:54 -0700126bool GetSlotNumber(const std::string& slot, int32_t* number);
David Anderson0f626632018-08-31 16:44:25 -0700127std::vector<std::string> ListPartitions(FastbootDevice* device);
Hridya Valsarajudca328d2018-09-24 16:01:35 -0700128bool GetDeviceLockStatus();
David Andersond25f1c32018-11-09 20:41:33 -0800129
130// Update all copies of metadata.
David Anderson4d307b02018-12-17 17:07:34 -0800131bool UpdateAllPartitionMetadata(FastbootDevice* device, const std::string& super_name,
David Andersond25f1c32018-11-09 20:41:33 -0800132 const android::fs_mgr::LpMetadata& metadata);