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_DMTARGET_H_
18 #define _LIBDM_DMTARGET_H_
19 
20 #include <linux/dm-ioctl.h>
21 #include <stdint.h>
22 
23 #include <string>
24 #include <vector>
25 
26 namespace android {
27 namespace dm {
28 
29 class DmTargetTypeInfo {
30   public:
DmTargetTypeInfo()31     DmTargetTypeInfo() : major_(0), minor_(0), patch_(0) {}
DmTargetTypeInfo(const struct dm_target_versions * info)32     DmTargetTypeInfo(const struct dm_target_versions* info)
33         : name_(info->name),
34           major_(info->version[0]),
35           minor_(info->version[1]),
36           patch_(info->version[2]) {}
37 
name()38     const std::string& name() const { return name_; }
version()39     std::string version() const {
40         return std::to_string(major_) + "." + std::to_string(minor_) + "." + std::to_string(patch_);
41     }
42 
major_version()43     uint32_t major_version() const { return major_; }
minor_version()44     uint32_t minor_version() const { return minor_; }
patch_level()45     uint32_t patch_level() const { return patch_; }
46 
IsAtLeast(uint32_t major,uint32_t minor,uint32_t patch)47     bool IsAtLeast(uint32_t major, uint32_t minor, uint32_t patch) const {
48         if (major_ > major) return true;
49         if (major_ < major) return false;
50         if (minor_ > minor) return true;
51         if (minor_ < minor) return false;
52         return patch_ >= patch;
53     }
54 
55   private:
56     std::string name_;
57     uint32_t major_;
58     uint32_t minor_;
59     uint32_t patch_;
60 };
61 
62 class DmTarget {
63   public:
DmTarget(uint64_t start,uint64_t length)64     DmTarget(uint64_t start, uint64_t length) : start_(start), length_(length) {}
65 
66     virtual ~DmTarget() = default;
67 
68     // Returns name of the target.
69     virtual std::string name() const = 0;
70 
71     // Return the first logical sector represented by this target.
start()72     uint64_t start() const { return start_; }
73 
74     // Returns size in number of sectors when this target is part of
75     // a DmTable, return 0 otherwise.
size()76     uint64_t size() const { return length_; }
77 
78     // Function that converts this object to a string of arguments that can
79     // be passed to the kernel for adding this target in a table. Each target (e.g. verity, linear)
80     // must implement this, for it to be used on a device.
81     std::string Serialize() const;
82 
Valid()83     virtual bool Valid() const { return true; }
84 
85   protected:
86     // Get the parameter string that is passed to the end of the dm_target_spec
87     // for this target type.
88     virtual std::string GetParameterString() const = 0;
89 
90   private:
91     // logical sector number start and total length (in terms of 512-byte sectors) represented
92     // by this target within a DmTable.
93     uint64_t start_, length_;
94 };
95 
96 class DmTargetZero final : public DmTarget {
97   public:
DmTargetZero(uint64_t start,uint64_t length)98     DmTargetZero(uint64_t start, uint64_t length) : DmTarget(start, length) {}
99 
name()100     std::string name() const override { return "zero"; }
101     std::string GetParameterString() const override;
102 };
103 
104 class DmTargetLinear final : public DmTarget {
105   public:
DmTargetLinear(uint64_t start,uint64_t length,const std::string & block_device,uint64_t physical_sector)106     DmTargetLinear(uint64_t start, uint64_t length, const std::string& block_device,
107                    uint64_t physical_sector)
108         : DmTarget(start, length), block_device_(block_device), physical_sector_(physical_sector) {}
109 
name()110     std::string name() const override { return "linear"; }
111     std::string GetParameterString() const override;
block_device()112     const std::string& block_device() const { return block_device_; }
113 
114   private:
115     std::string block_device_;
116     uint64_t physical_sector_;
117 };
118 
119 class DmTargetVerity final : public DmTarget {
120   public:
121     DmTargetVerity(uint64_t start, uint64_t length, uint32_t version,
122                    const std::string& block_device, const std::string& hash_device,
123                    uint32_t data_block_size, uint32_t hash_block_size, uint32_t num_data_blocks,
124                    uint32_t hash_start_block, const std::string& hash_algorithm,
125                    const std::string& root_digest, const std::string& salt);
126 
127     void UseFec(const std::string& device, uint32_t num_roots, uint32_t num_blocks, uint32_t start);
128     void SetVerityMode(const std::string& mode);
129     void IgnoreZeroBlocks();
130 
name()131     std::string name() const override { return "verity"; }
132     std::string GetParameterString() const override;
Valid()133     bool Valid() const override { return valid_; }
134 
135   private:
136     std::vector<std::string> base_args_;
137     std::vector<std::string> optional_args_;
138     bool valid_;
139 };
140 
141 class DmTargetAndroidVerity final : public DmTarget {
142   public:
DmTargetAndroidVerity(uint64_t start,uint64_t length,const std::string & block_device,const std::string & keyid)143     DmTargetAndroidVerity(uint64_t start, uint64_t length, const std::string& block_device,
144                           const std::string& keyid)
145         : DmTarget(start, length), keyid_(keyid), block_device_(block_device) {}
146 
name()147     std::string name() const override { return "android-verity"; }
148     std::string GetParameterString() const override;
149 
150   private:
151     std::string keyid_;
152     std::string block_device_;
153 };
154 
155 // This is the same as DmTargetVerity, but the table may be specified as a raw
156 // string. This code exists only for fs_mgr_verity and should be avoided. Use
157 // DmTargetVerity for new code instead.
158 class DmTargetVerityString final : public DmTarget {
159   public:
DmTargetVerityString(uint64_t start,uint64_t length,const std::string & target_string)160     DmTargetVerityString(uint64_t start, uint64_t length, const std::string& target_string)
161         : DmTarget(start, length), target_string_(target_string) {}
162 
name()163     std::string name() const override { return "verity"; }
GetParameterString()164     std::string GetParameterString() const override { return target_string_; }
Valid()165     bool Valid() const override { return true; }
166 
167   private:
168     std::string target_string_;
169 };
170 
171 // dm-bow is the backup on write target that can provide checkpoint capability
172 // for file systems that do not support checkpoints natively
173 class DmTargetBow final : public DmTarget {
174   public:
DmTargetBow(uint64_t start,uint64_t length,const std::string & target_string)175     DmTargetBow(uint64_t start, uint64_t length, const std::string& target_string)
176         : DmTarget(start, length), target_string_(target_string) {}
177 
SetBlockSize(uint32_t block_size)178     void SetBlockSize(uint32_t block_size) { block_size_ = block_size; }
179 
name()180     std::string name() const override { return "bow"; }
181     std::string GetParameterString() const override;
182 
183   private:
184     std::string target_string_;
185     uint32_t block_size_ = 0;
186 };
187 
188 enum class SnapshotStorageMode {
189     // The snapshot will be persisted to the COW device.
190     Persistent,
191     // The snapshot will be lost on reboot.
192     Transient,
193     // The snapshot will be merged from the COW device into the base device,
194     // in the background.
195     Merge
196 };
197 
198 // Writes to a snapshot device will be written to the given COW device. Reads
199 // will read from the COW device or base device. The chunk size is specified
200 // in sectors.
201 class DmTargetSnapshot final : public DmTarget {
202   public:
DmTargetSnapshot(uint64_t start,uint64_t length,const std::string & base_device,const std::string & cow_device,SnapshotStorageMode mode,uint64_t chunk_size)203     DmTargetSnapshot(uint64_t start, uint64_t length, const std::string& base_device,
204                      const std::string& cow_device, SnapshotStorageMode mode, uint64_t chunk_size)
205         : DmTarget(start, length),
206           base_device_(base_device),
207           cow_device_(cow_device),
208           mode_(mode),
209           chunk_size_(chunk_size) {}
210 
211     std::string name() const override;
212     std::string GetParameterString() const override;
Valid()213     bool Valid() const override { return true; }
214 
215     struct Status {
216         uint64_t sectors_allocated;
217         uint64_t total_sectors;
218         uint64_t metadata_sectors;
219         std::string error;
220     };
221 
222     static double MergePercent(const Status& status, uint64_t sectors_initial = 0);
223     static bool ParseStatusText(const std::string& text, Status* status);
224     static bool ReportsOverflow(const std::string& target_type);
225     static bool GetDevicesFromParams(const std::string& params, std::string* base_device,
226                                      std::string* cow_device);
227 
228   private:
229     std::string base_device_;
230     std::string cow_device_;
231     SnapshotStorageMode mode_;
232     uint64_t chunk_size_;
233 };
234 
235 // snapshot-origin will read/write directly to the backing device, updating any
236 // snapshot devices with a matching origin.
237 class DmTargetSnapshotOrigin final : public DmTarget {
238   public:
DmTargetSnapshotOrigin(uint64_t start,uint64_t length,const std::string & device)239     DmTargetSnapshotOrigin(uint64_t start, uint64_t length, const std::string& device)
240         : DmTarget(start, length), device_(device) {}
241 
name()242     std::string name() const override { return "snapshot-origin"; }
GetParameterString()243     std::string GetParameterString() const override { return device_; }
Valid()244     bool Valid() const override { return true; }
245 
246   private:
247     std::string device_;
248 };
249 
250 class DmTargetCrypt final : public DmTarget {
251   public:
DmTargetCrypt(uint64_t start,uint64_t length,const std::string & cipher,const std::string & key,uint64_t iv_sector_offset,const std::string & device,uint64_t device_sector)252     DmTargetCrypt(uint64_t start, uint64_t length, const std::string& cipher,
253                   const std::string& key, uint64_t iv_sector_offset, const std::string& device,
254                   uint64_t device_sector)
255         : DmTarget(start, length),
256           cipher_(cipher),
257           key_(key),
258           iv_sector_offset_(iv_sector_offset),
259           device_(device),
260           device_sector_(device_sector) {}
261 
AllowDiscards()262     void AllowDiscards() { allow_discards_ = true; }
AllowEncryptOverride()263     void AllowEncryptOverride() { allow_encrypt_override_ = true; }
SetIvLargeSectors()264     void SetIvLargeSectors() { iv_large_sectors_ = true; }
SetSectorSize(uint32_t sector_size)265     void SetSectorSize(uint32_t sector_size) { sector_size_ = sector_size; }
266 
name()267     std::string name() const override { return "crypt"; }
Valid()268     bool Valid() const override { return true; }
269     std::string GetParameterString() const override;
270 
271   private:
272     std::string cipher_;
273     std::string key_;
274     uint64_t iv_sector_offset_;
275     std::string device_;
276     uint64_t device_sector_;
277     bool allow_discards_ = false;
278     bool allow_encrypt_override_ = false;
279     bool iv_large_sectors_ = false;
280     uint32_t sector_size_ = 0;
281 };
282 
283 class DmTargetDefaultKey final : public DmTarget {
284   public:
DmTargetDefaultKey(uint64_t start,uint64_t length,const std::string & cipher,const std::string & key,const std::string & blockdev,uint64_t start_sector)285     DmTargetDefaultKey(uint64_t start, uint64_t length, const std::string& cipher,
286                        const std::string& key, const std::string& blockdev, uint64_t start_sector)
287         : DmTarget(start, length),
288           cipher_(cipher),
289           key_(key),
290           blockdev_(blockdev),
291           start_sector_(start_sector) {}
292 
name()293     std::string name() const override { return kName; }
294     bool Valid() const override;
295     std::string GetParameterString() const override;
SetUseLegacyOptionsFormat()296     void SetUseLegacyOptionsFormat() { use_legacy_options_format_ = true; }
SetSetDun()297     void SetSetDun() { set_dun_ = true; }
SetWrappedKeyV0()298     void SetWrappedKeyV0() { is_hw_wrapped_ = true; }
299 
300   private:
301     inline static const std::string kName = "default-key";
302 
303     std::string cipher_;
304     std::string key_;
305     std::string blockdev_;
306     uint64_t start_sector_;
307     bool use_legacy_options_format_ = false;
308     bool set_dun_ = false;
309     bool is_hw_wrapped_ = false;
310 };
311 
312 class DmTargetUser final : public DmTarget {
313   public:
DmTargetUser(uint64_t start,uint64_t length)314     DmTargetUser(uint64_t start, uint64_t length) : DmTarget(start, length) {}
315 
name()316     std::string name() const override { return "user"; }
317     std::string GetParameterString() const override;
318 };
319 
320 }  // namespace dm
321 }  // namespace android
322 
323 #endif /* _LIBDM_DMTARGET_H_ */
324