• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  *  Copyright 2018 Google, Inc
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 _LIBDM_DM_H_
18 #define _LIBDM_DM_H_
19 
20 #include <dirent.h>
21 #include <fcntl.h>
22 #include <linux/dm-ioctl.h>
23 #include <linux/kdev_t.h>
24 #include <linux/types.h>
25 #include <stdint.h>
26 #include <sys/sysmacros.h>
27 #include <sys/types.h>
28 #include <unistd.h>
29 
30 #include <chrono>
31 #include <map>
32 #include <memory>
33 #include <optional>
34 #include <string>
35 #include <utility>
36 #include <vector>
37 
38 #include "dm_table.h"
39 
40 // The minimum expected device mapper major.minor version
41 #define DM_VERSION0 (4)
42 #define DM_VERSION1 (0)
43 #define DM_VERSION2 (0)
44 
45 #define DM_ALIGN_MASK (7)
46 #define DM_ALIGN(x) (((x) + DM_ALIGN_MASK) & ~DM_ALIGN_MASK)
47 
48 namespace android {
49 namespace dm {
50 
51 enum class DmDeviceState { INVALID, SUSPENDED, ACTIVE };
52 
53 static constexpr uint64_t kSectorSize = 512;
54 
55 // Returns `path` without /dev/block prefix if and only if `path` starts with
56 // that prefix.
57 std::optional<std::string> ExtractBlockDeviceName(const std::string& path);
58 
59 // This interface is for testing purposes. See DeviceMapper proper for what these methods do.
60 class IDeviceMapper {
61   public:
~IDeviceMapper()62     virtual ~IDeviceMapper() {}
63 
64     struct TargetInfo {
65         struct dm_target_spec spec;
66         std::string data;
TargetInfoTargetInfo67         TargetInfo() {}
TargetInfoTargetInfo68         TargetInfo(const struct dm_target_spec& spec, const std::string& data)
69             : spec(spec), data(data) {}
70 
71         bool IsOverflowSnapshot() const;
72     };
73 
74     virtual bool CreateDevice(const std::string& name, const DmTable& table, std::string* path,
75                               const std::chrono::milliseconds& timeout_ms) = 0;
76     virtual DmDeviceState GetState(const std::string& name) const = 0;
77     virtual bool LoadTableAndActivate(const std::string& name, const DmTable& table) = 0;
78     virtual bool GetTableInfo(const std::string& name, std::vector<TargetInfo>* table) = 0;
79     virtual bool GetTableStatus(const std::string& name, std::vector<TargetInfo>* table) = 0;
80     virtual bool GetDmDevicePathByName(const std::string& name, std::string* path) = 0;
81     virtual bool GetDeviceString(const std::string& name, std::string* dev) = 0;
82     virtual bool DeleteDeviceIfExists(const std::string& name) = 0;
83 };
84 
85 class DeviceMapper final : public IDeviceMapper {
86   public:
87     class DmBlockDevice final {
88       public:
89         // only allow creating this with dm_name_list
90         DmBlockDevice() = delete;
91 
DmBlockDevice(struct dm_name_list * d)92         explicit DmBlockDevice(struct dm_name_list* d) : name_(d->name), dev_(d->dev){};
93 
94         // Returs device mapper name associated with the block device
name()95         const std::string& name() const { return name_; }
96 
97         // Return major number for the block device
Major()98         uint32_t Major() const { return major(dev_); }
99 
100         // Return minor number for the block device
Minor()101         uint32_t Minor() const { return minor(dev_); }
102         ~DmBlockDevice() = default;
103 
104       private:
105         std::string name_;
106         uint64_t dev_;
107     };
108 
109     class Info {
110         uint32_t flags_;
111 
112       public:
Info(uint32_t flags)113         explicit Info(uint32_t flags) : flags_(flags) {}
114 
IsActiveTablePresent()115         bool IsActiveTablePresent() const { return flags_ & DM_ACTIVE_PRESENT_FLAG; }
IsBufferFull()116         bool IsBufferFull() const { return flags_ & DM_BUFFER_FULL_FLAG; }
IsInactiveTablePresent()117         bool IsInactiveTablePresent() const { return flags_ & DM_INACTIVE_PRESENT_FLAG; }
IsReadOnly()118         bool IsReadOnly() const { return flags_ & DM_READONLY_FLAG; }
IsSuspended()119         bool IsSuspended() const { return flags_ & DM_SUSPEND_FLAG; }
120     };
121 
122     // Removes a device mapper device with the given name.
123     // Returns 'true' on success, false otherwise.
124     bool DeleteDevice(const std::string& name);
125     bool DeleteDeviceIfExists(const std::string& name) override;
126     // Removes a device mapper device with the given name and waits for |timeout_ms| milliseconds
127     // for the corresponding block device to be deleted.
128     bool DeleteDevice(const std::string& name, const std::chrono::milliseconds& timeout_ms);
129     bool DeleteDeviceIfExists(const std::string& name, const std::chrono::milliseconds& timeout_ms);
130 
131     // Enqueues a deletion of device mapper device with the given name once last reference is
132     // closed.
133     // Returns 'true' on success, false otherwise.
134     bool DeleteDeviceDeferred(const std::string& name);
135     bool DeleteDeviceIfExistsDeferred(const std::string& name);
136 
137     // Fetches and returns the complete state of the underlying device mapper
138     // device with given name.
139     std::optional<Info> GetDetailedInfo(const std::string& name) const;
140 
141     // Returns the current state of the underlying device mapper device
142     // with given name.
143     // One of INVALID, SUSPENDED or ACTIVE.
144     DmDeviceState GetState(const std::string& name) const override;
145 
146     // Puts the given device to the specified status, which must be either:
147     // - SUSPENDED: suspend the device, or
148     // - ACTIVE: resumes the device.
149     bool ChangeState(const std::string& name, DmDeviceState state);
150 
151     // Creates empty device.
152     // This supports a use case when a caller doesn't need a device straight away, but instead
153     // asks kernel to create it beforehand, thus avoiding blocking itself from waiting for ueventd
154     // to create user space paths.
155     // Callers are expected to then activate their device by calling LoadTableAndActivate function.
156     // To avoid race conditions, callers must still synchronize with ueventd by calling
157     // WaitForDevice function.
158     bool CreateEmptyDevice(const std::string& name);
159 
160     // Waits for device paths to be created in the user space.
161     bool WaitForDevice(const std::string& name, const std::chrono::milliseconds& timeout_ms,
162                        std::string* path);
163 
164     // Creates a device, loads the given table, and activates it. If the device
165     // is not able to be activated, it is destroyed, and false is returned.
166     // After creation, |path| contains the result of calling
167     // GetDmDevicePathByName, and the path is guaranteed to exist. If after
168     // |timeout_ms| the path is not available, the device will be deleted and
169     // this function will return false.
170     //
171     // This variant must be used when depending on the device path. The
172     // following manual sequence should not be used:
173     //
174     //   1. CreateDevice(name, table)
175     //   2. GetDmDevicePathByName(name, &path)
176     //   3. fs_mgr::WaitForFile(path, <timeout>)
177     //
178     // This sequence has a race condition where, if another process deletes a
179     // device, CreateDevice may acquire the same path. When this happens, the
180     // WaitForFile() may early-return since ueventd has not yet processed all
181     // of the outstanding udev events. The caller may unexpectedly get an
182     // ENOENT on a system call using the affected path.
183     //
184     // If |timeout_ms| is 0ms, then this function will return true whether or
185     // not |path| is available. It is the caller's responsibility to ensure
186     // there are no races.
187     bool CreateDevice(const std::string& name, const DmTable& table, std::string* path,
188                       const std::chrono::milliseconds& timeout_ms) override;
189 
190     // Create a device and activate the given table, without waiting to acquire
191     // a valid path. If the caller will use GetDmDevicePathByName(), it should
192     // use the timeout variant above.
193     bool CreateDevice(const std::string& name, const DmTable& table);
194 
195     // Loads the device mapper table from parameter into the underlying device
196     // mapper device with given name and activate / resumes the device in the
197     // process. A device with the given name must already exist.
198     //
199     // Returns 'true' on success, false otherwise.
200     bool LoadTableAndActivate(const std::string& name, const DmTable& table) override;
201 
202     // Returns true if a list of available device mapper targets registered in the kernel was
203     // successfully read and stored in 'targets'. Returns 'false' otherwise.
204     bool GetAvailableTargets(std::vector<DmTargetTypeInfo>* targets);
205 
206     // Finds a target by name and returns its information if found. |info| may
207     // be null to check for the existence of a target.
208     bool GetTargetByName(const std::string& name, DmTargetTypeInfo* info);
209 
210     // Return 'true' if it can successfully read the list of device mapper block devices
211     // currently created. 'devices' will be empty if the kernel interactions
212     // were successful and there are no block devices at the moment. Returns
213     // 'false' in case of any failure along the way.
214     bool GetAvailableDevices(std::vector<DmBlockDevice>* devices);
215 
216     // Returns the path to the device mapper device node in '/dev' corresponding to
217     // 'name'. If the device does not exist, false is returned, and the path
218     // parameter is not set.
219     //
220     // This returns a path in the format "/dev/block/dm-N" that can be easily
221     // re-used with sysfs.
222     //
223     // WaitForFile() should not be used in conjunction with this call, since it
224     // could race with ueventd.
225     bool GetDmDevicePathByName(const std::string& name, std::string* path);
226 
227     // Returns the device mapper UUID for a given name.  If the device does not
228     // exist, false is returned, and the path parameter is not set.
229     //
230     // WaitForFile() should not be used in conjunction with this call, since it
231     // could race with ueventd.
232     bool GetDmDeviceUuidByName(const std::string& name, std::string* path);
233 
234     // Returns a device's unique path as generated by ueventd. This will return
235     // true as long as the device has been created, even if ueventd has not
236     // processed it yet.
237     //
238     // The formatting of this path is /dev/block/mapper/by-uuid/<uuid>.
239     bool GetDeviceUniquePath(const std::string& name, std::string* path);
240 
241     // Returns the dev_t for the named device-mapper node.
242     bool GetDeviceNumber(const std::string& name, dev_t* dev);
243 
244     // Returns a major:minor string for the named device-mapper node, that can
245     // be used as inputs to DmTargets that take a block device.
246     bool GetDeviceString(const std::string& name, std::string* dev) override;
247 
248     // The only way to create a DeviceMapper object.
249     static DeviceMapper& Instance();
250 
~DeviceMapper()251     ~DeviceMapper() {
252         if (fd_ != -1) {
253             ::close(fd_);
254         }
255     }
256 
257     // Query the status of a table, given a device name. The output vector will
258     // contain one TargetInfo for each target in the table. If the device does
259     // not exist, or there were too many targets, the call will fail and return
260     // false.
261     bool GetTableStatus(const std::string& name, std::vector<TargetInfo>* table) override;
262 
263     // Identical to GetTableStatus, except also retrives the active table for the device
264     // mapper device from the kernel.
265     bool GetTableInfo(const std::string& name, std::vector<TargetInfo>* table) override;
266 
267     static std::string GetTargetType(const struct dm_target_spec& spec);
268 
269     // Returns true if given path is a path to a dm block device.
270     bool IsDmBlockDevice(const std::string& path);
271 
272     // Returns name of a dm-device with the given path, or std::nulloptr if given path is not a
273     // dm-device.
274     std::optional<std::string> GetDmDeviceNameByPath(const std::string& path);
275 
276     // Returns a parent block device of a dm device with the given path, or std::nullopt if:
277     //  * Given path doesn't correspond to a dm device.
278     //  * A dm device is based on top of more than one block devices.
279     //  * A failure occurred.
280     std::optional<std::string> GetParentBlockDeviceByPath(const std::string& path);
281 
282     // Iterate the content over "/sys/block/dm-x/dm/name" and find
283     // all the dm-wrapped block devices.
284     //
285     // Returns mapping <partition-name, /dev/block/dm-x>
286     std::map<std::string, std::string> FindDmPartitions();
287 
288   private:
289     // Maximum possible device mapper targets registered in the kernel.
290     // This is only used to read the list of targets from kernel so we allocate
291     // a finite amount of memory. This limit is in no way enforced by the kernel.
292     static constexpr uint32_t kMaxPossibleDmTargets = 256;
293 
294     // Maximum possible device mapper created block devices. Note that this is restricted by
295     // the minor numbers (that used to be 8 bits) that can be range from 0 to 2^20-1 in newer
296     // kernels. In Android systems however, we never expect these to grow beyond the artificial
297     // limit we are imposing here of 256.
298     static constexpr uint32_t kMaxPossibleDmDevices = 256;
299 
300     bool CreateDevice(const std::string& name, const std::string& uuid = {});
301     bool GetTable(const std::string& name, uint32_t flags, std::vector<TargetInfo>* table);
302     void InitIo(struct dm_ioctl* io, const std::string& name = std::string()) const;
303 
304     DeviceMapper();
305 
306     int fd_;
307     // Non-copyable & Non-movable
308     DeviceMapper(const DeviceMapper&) = delete;
309     DeviceMapper& operator=(const DeviceMapper&) = delete;
310     DeviceMapper& operator=(DeviceMapper&&) = delete;
311     DeviceMapper(DeviceMapper&&) = delete;
312 };
313 
314 }  // namespace dm
315 }  // namespace android
316 
317 #endif /* _LIBDM_DM_H_ */
318