fs_management/
partition.rs

1// Copyright 2022 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use crate::format::{detect_disk_format, DiskFormat};
6use anyhow::{anyhow, Context, Error};
7use fidl_fuchsia_device::{ControllerMarker, ControllerProxy};
8use fidl_fuchsia_hardware_block_partition::{Guid, PartitionMarker};
9use fidl_fuchsia_hardware_block_volume::VolumeManagerProxy;
10use fidl_fuchsia_io as fio;
11use fuchsia_async::TimeoutExt;
12use fuchsia_component::client::connect_to_named_protocol_at_dir_root;
13use fuchsia_fs::directory::{WatchEvent, Watcher};
14use futures::StreamExt;
15use zx::{self as zx, MonotonicDuration};
16
17/// Set of parameters to use for identifying the correct partition to open via
18/// [`open_partition`]
19///
20/// If multiple matchers are specified, the first partition that satisfies any set
21/// of matchers will be used. At least one of [`type_guids`], [`instance_guids`], [`labels`],
22/// [`detected_formats`], or [`parent_device`] must be specified.
23#[derive(Default, Clone)]
24pub struct PartitionMatcher {
25    /// Set of type GUIDs the partition must match. Ignored if empty.
26    pub type_guids: Option<Vec<[u8; 16]>>,
27    /// Set of instance GUIDs the partition must match. Ignored if empty.
28    pub instance_guids: Option<Vec<[u8; 16]>>,
29    pub labels: Option<Vec<String>>,
30    pub detected_disk_formats: Option<Vec<DiskFormat>>,
31    /// partition must be a child of this device.
32    pub parent_device: Option<String>,
33    /// The topological path must not start with this prefix.
34    pub ignore_prefix: Option<String>,
35    /// The topological path must not contain this substring.
36    pub ignore_if_path_contains: Option<String>,
37}
38
39const BLOCK_DEV_PATH: &str = "/dev/class/block/";
40
41/// Waits for a partition to appear on BLOCK_DEV_PATH that matches the fields in the
42/// PartitionMatcher. Returns the path of the partition if found. Errors after timeout duration.
43// TODO(https://fxbug.dev/42072982): Most users end up wanting the things we open for checking the partition,
44// like the partition proxy and the topological path. We should consider returning all those
45// resources instead of forcing them to retrieve them again.
46pub async fn find_partition(
47    matcher: PartitionMatcher,
48    timeout: MonotonicDuration,
49) -> Result<ControllerProxy, Error> {
50    let dir = fuchsia_fs::directory::open_in_namespace(BLOCK_DEV_PATH, fio::Flags::empty())?;
51    find_partition_in(&dir, matcher, timeout).await
52}
53
54/// Waits for a partition to appear in [`dir`] that matches the fields in [`matcher`]. Returns the
55/// topological path of the partition if found. Returns an error after the timeout duration
56/// expires.
57pub async fn find_partition_in(
58    dir: &fio::DirectoryProxy,
59    matcher: PartitionMatcher,
60    timeout: MonotonicDuration,
61) -> Result<ControllerProxy, Error> {
62    let timeout_seconds = timeout.into_seconds();
63    async {
64        let mut watcher = Watcher::new(dir).await.context("making watcher")?;
65        while let Some(message) = watcher.next().await {
66            let message = message.context("watcher channel returned error")?;
67            match message.event {
68                WatchEvent::ADD_FILE | WatchEvent::EXISTING => {
69                    let filename = message.filename.to_str().unwrap();
70                    if filename == "." {
71                        continue;
72                    }
73                    let proxy = connect_to_named_protocol_at_dir_root::<ControllerMarker>(
74                        &dir,
75                        &format!("{filename}/device_controller"),
76                    )
77                    .context("opening partition path")?;
78                    match partition_matches_with_proxy(&proxy, &matcher).await {
79                        Ok(true) => {
80                            return Ok(proxy);
81                        }
82                        Ok(false) => {}
83                        Err(error) => {
84                            log::info!(error:?; "Failure in partition match. Transient device?");
85                        }
86                    }
87                }
88                _ => (),
89            }
90        }
91        Err(anyhow!("Watch stream unexpectedly ended"))
92    }
93    .on_timeout(timeout, || {
94        Err(anyhow!("Timed out after {}s without finding expected partition", timeout_seconds))
95    })
96    .await
97}
98
99/// Checks if the partition associated with proxy matches the matcher.
100/// An error isn't necessarily an issue - we might be using a matcher that wants a type guid,
101/// but the device we are currently checking doesn't implement get_type_guid. The error message may
102/// help debugging why no partition was matched but should generally be considered recoverable.
103pub async fn partition_matches_with_proxy(
104    controller_proxy: &ControllerProxy,
105    matcher: &PartitionMatcher,
106) -> Result<bool, Error> {
107    assert!(
108        matcher.type_guids.is_some()
109            || matcher.instance_guids.is_some()
110            || matcher.detected_disk_formats.is_some()
111            || matcher.parent_device.is_some()
112            || matcher.labels.is_some()
113    );
114
115    let (partition_proxy, partition_server_end) =
116        fidl::endpoints::create_proxy::<PartitionMarker>();
117    controller_proxy
118        .connect_to_device_fidl(partition_server_end.into_channel())
119        .context("connecting to partition protocol")?;
120
121    if let Some(matcher_type_guids) = &matcher.type_guids {
122        let (status, guid_option) =
123            partition_proxy.get_type_guid().await.context("transport error on get_type_guid")?;
124        zx::Status::ok(status).context("get_type_guid failed")?;
125        let guid = guid_option.ok_or_else(|| anyhow!("Expected type guid"))?;
126        if !matcher_type_guids.into_iter().any(|x| x == &guid.value) {
127            return Ok(false);
128        }
129    }
130
131    if let Some(matcher_instance_guids) = &matcher.instance_guids {
132        let (status, guid_option) = partition_proxy
133            .get_instance_guid()
134            .await
135            .context("transport error on get_instance_guid")?;
136        zx::Status::ok(status).context("get_instance_guid failed")?;
137        let guid = guid_option.ok_or_else(|| anyhow!("Expected instance guid"))?;
138        if !matcher_instance_guids.into_iter().any(|x| x == &guid.value) {
139            return Ok(false);
140        }
141    }
142
143    if let Some(matcher_labels) = &matcher.labels {
144        let (status, name) =
145            partition_proxy.get_name().await.context("transport error on get_name")?;
146        zx::Status::ok(status).context("get_name failed")?;
147        let name = name.ok_or_else(|| anyhow!("Expected name"))?;
148        if name.is_empty() {
149            return Ok(false);
150        }
151        let mut matches_label = false;
152        for label in matcher_labels {
153            if name == *label {
154                matches_label = true;
155                break;
156            }
157        }
158        if !matches_label {
159            return Ok(false);
160        }
161    }
162
163    let topological_path = controller_proxy
164        .get_topological_path()
165        .await
166        .context("get_topological_path failed")?
167        .map_err(zx::Status::from_raw)?;
168
169    if let Some(matcher_parent_device) = &matcher.parent_device {
170        if !topological_path.starts_with(matcher_parent_device) {
171            return Ok(false);
172        }
173    }
174
175    if let Some(matcher_ignore_prefix) = &matcher.ignore_prefix {
176        if topological_path.starts_with(matcher_ignore_prefix) {
177            return Ok(false);
178        }
179    }
180
181    if let Some(matcher_ignore_if_path_contains) = &matcher.ignore_if_path_contains {
182        if topological_path.find(matcher_ignore_if_path_contains) != None {
183            return Ok(false);
184        }
185    }
186
187    if let Some(matcher_detected_disk_formats) = &matcher.detected_disk_formats {
188        let detected_format = detect_disk_format(&partition_proxy).await;
189        if !matcher_detected_disk_formats.into_iter().any(|x| x == &detected_format) {
190            return Ok(false);
191        }
192    }
193    Ok(true)
194}
195
196pub async fn fvm_allocate_partition(
197    fvm_proxy: &VolumeManagerProxy,
198    type_guid: [u8; 16],
199    instance_guid: [u8; 16],
200    name: &str,
201    flags: u32,
202    slice_count: u64,
203) -> Result<ControllerProxy, Error> {
204    let status = fvm_proxy
205        .allocate_partition(
206            slice_count,
207            &Guid { value: type_guid },
208            &Guid { value: instance_guid },
209            name,
210            flags,
211        )
212        .await?;
213    zx::Status::ok(status)?;
214
215    let matcher = PartitionMatcher {
216        type_guids: Some(vec![type_guid]),
217        instance_guids: Some(vec![instance_guid]),
218        ..Default::default()
219    };
220
221    find_partition(matcher, MonotonicDuration::from_seconds(40)).await
222}
223
224#[cfg(test)]
225mod tests {
226    use super::{partition_matches_with_proxy, PartitionMatcher};
227    use crate::format::{constants, DiskFormat};
228    use fake_block_server::FakeServer;
229    use fidl::endpoints::{create_proxy_and_stream, RequestStream as _};
230    use fidl_fuchsia_device::{ControllerMarker, ControllerRequest};
231    use fidl_fuchsia_hardware_block_volume::VolumeRequestStream;
232    use fuchsia_async as fasync;
233    use futures::{pin_mut, select, FutureExt, StreamExt};
234    use std::sync::Arc;
235
236    const VALID_TYPE_GUID: [u8; 16] = fake_block_server::TYPE_GUID;
237
238    const VALID_INSTANCE_GUID: [u8; 16] = fake_block_server::INSTANCE_GUID;
239
240    const INVALID_GUID_1: [u8; 16] = [
241        0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e,
242        0x2f,
243    ];
244
245    const INVALID_GUID_2: [u8; 16] = [
246        0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e,
247        0x3f,
248    ];
249
250    const VALID_LABEL: &str = fake_block_server::PARTITION_NAME;
251    const INVALID_LABEL_1: &str = "TheWrongLabel";
252    const INVALID_LABEL_2: &str = "StillTheWrongLabel";
253    const PARENT_DEVICE_PATH: &str = "/fake/block/device/1";
254    const NOT_PARENT_DEVICE_PATH: &str = "/fake/block/device/2";
255    const DEFAULT_PATH: &str = "/fake/block/device/1/partition/001";
256
257    async fn check_partition_matches(matcher: &PartitionMatcher) -> bool {
258        let (proxy, mut stream) = create_proxy_and_stream::<ControllerMarker>();
259
260        let fake_block_server = Arc::new(FakeServer::new(1000, 512, &constants::FVM_MAGIC));
261
262        let mock_controller = async {
263            while let Some(request) = stream.next().await {
264                match request {
265                    Ok(ControllerRequest::GetTopologicalPath { responder }) => {
266                        responder.send(Ok(DEFAULT_PATH)).unwrap();
267                    }
268                    Ok(ControllerRequest::ConnectToDeviceFidl { server, .. }) => {
269                        let fake_block_server = fake_block_server.clone();
270                        fasync::Task::spawn(async move {
271                            if let Err(e) = fake_block_server
272                                .serve(VolumeRequestStream::from_channel(
273                                    fasync::Channel::from_channel(server),
274                                ))
275                                .await
276                            {
277                                println!("FakeServer::serve failed: {e:?}");
278                            }
279                        })
280                        .detach();
281                    }
282                    _ => {
283                        println!("Unexpected request: {:?}", request);
284                        unreachable!()
285                    }
286                }
287            }
288        }
289        .fuse();
290
291        pin_mut!(mock_controller);
292
293        select! {
294            _ = mock_controller => unreachable!(),
295            matches = partition_matches_with_proxy(&proxy, &matcher).fuse() => matches,
296        }
297        .unwrap_or(false)
298    }
299
300    #[fuchsia::test]
301    async fn test_type_guid_match() {
302        let matcher = PartitionMatcher {
303            type_guids: Some(vec![VALID_TYPE_GUID, INVALID_GUID_1]),
304            ..Default::default()
305        };
306        assert_eq!(check_partition_matches(&matcher).await, true);
307    }
308
309    #[fuchsia::test]
310    async fn test_instance_guid_match() {
311        let matcher = PartitionMatcher {
312            instance_guids: Some(vec![VALID_INSTANCE_GUID, INVALID_GUID_1]),
313            ..Default::default()
314        };
315        assert_eq!(check_partition_matches(&matcher).await, true);
316    }
317
318    #[fuchsia::test]
319    async fn test_type_and_instance_guid_match() {
320        let matcher = PartitionMatcher {
321            type_guids: Some(vec![VALID_TYPE_GUID, INVALID_GUID_1]),
322            instance_guids: Some(vec![VALID_INSTANCE_GUID, INVALID_GUID_2]),
323            ..Default::default()
324        };
325        assert_eq!(check_partition_matches(&matcher).await, true);
326    }
327
328    #[fuchsia::test]
329    async fn test_parent_match() {
330        let matcher = PartitionMatcher {
331            parent_device: Some(PARENT_DEVICE_PATH.to_string()),
332            ..Default::default()
333        };
334        assert_eq!(check_partition_matches(&matcher).await, true);
335
336        let matcher2 = PartitionMatcher {
337            parent_device: Some(NOT_PARENT_DEVICE_PATH.to_string()),
338            ..Default::default()
339        };
340        assert_eq!(check_partition_matches(&matcher2).await, false);
341    }
342
343    #[fuchsia::test]
344    async fn test_single_label_match() {
345        let the_labels = vec![VALID_LABEL.to_string()];
346        let matcher = PartitionMatcher { labels: Some(the_labels), ..Default::default() };
347        assert_eq!(check_partition_matches(&matcher).await, true);
348    }
349
350    #[fuchsia::test]
351    async fn test_multi_label_match() {
352        let mut the_labels = vec![VALID_LABEL.to_string()];
353        the_labels.push(INVALID_LABEL_1.to_string());
354        the_labels.push(INVALID_LABEL_2.to_string());
355        let matcher = PartitionMatcher { labels: Some(the_labels), ..Default::default() };
356        assert_eq!(check_partition_matches(&matcher).await, true);
357    }
358
359    #[fuchsia::test]
360    async fn test_ignore_prefix_mismatch() {
361        let matcher = PartitionMatcher {
362            type_guids: Some(vec![VALID_TYPE_GUID]),
363            ignore_prefix: Some("/fake/block/device".to_string()),
364            ..Default::default()
365        };
366        assert_eq!(check_partition_matches(&matcher).await, false);
367    }
368
369    #[fuchsia::test]
370    async fn test_ignore_prefix_match() {
371        let matcher = PartitionMatcher {
372            type_guids: Some(vec![VALID_TYPE_GUID]),
373            ignore_prefix: Some("/real/block/device".to_string()),
374            ..Default::default()
375        };
376        assert_eq!(check_partition_matches(&matcher).await, true);
377    }
378
379    #[fuchsia::test]
380    async fn test_ignore_if_path_contains_mismatch() {
381        let matcher = PartitionMatcher {
382            type_guids: Some(vec![VALID_TYPE_GUID]),
383            ignore_if_path_contains: Some("/device/1".to_string()),
384            ..Default::default()
385        };
386        assert_eq!(check_partition_matches(&matcher).await, false);
387    }
388
389    #[fuchsia::test]
390    async fn test_ignore_if_path_contains_match() {
391        let matcher = PartitionMatcher {
392            type_guids: Some(vec![VALID_TYPE_GUID]),
393            ignore_if_path_contains: Some("/device/0".to_string()),
394            ..Default::default()
395        };
396        assert_eq!(check_partition_matches(&matcher).await, true);
397    }
398
399    #[fuchsia::test]
400    async fn test_type_and_label_match() {
401        let the_labels = vec![VALID_LABEL.to_string()];
402        let matcher = PartitionMatcher {
403            type_guids: Some(vec![VALID_TYPE_GUID]),
404            labels: Some(the_labels),
405            ..Default::default()
406        };
407        assert_eq!(check_partition_matches(&matcher).await, true);
408    }
409
410    #[fuchsia::test]
411    async fn test_type_guid_mismatch() {
412        let matcher = PartitionMatcher {
413            type_guids: Some(vec![INVALID_GUID_1, INVALID_GUID_2]),
414            ..Default::default()
415        };
416        assert_eq!(check_partition_matches(&matcher).await, false);
417    }
418
419    #[fuchsia::test]
420    async fn test_instance_guid_mismatch() {
421        let matcher = PartitionMatcher {
422            instance_guids: Some(vec![INVALID_GUID_1, INVALID_GUID_2]),
423            ..Default::default()
424        };
425        assert_eq!(check_partition_matches(&matcher).await, false);
426    }
427
428    #[fuchsia::test]
429    async fn test_label_mismatch() {
430        let mut the_labels = vec![INVALID_LABEL_1.to_string()];
431        the_labels.push(INVALID_LABEL_2.to_string());
432        let matcher = PartitionMatcher { labels: Some(the_labels), ..Default::default() };
433        assert_eq!(check_partition_matches(&matcher).await, false);
434    }
435
436    #[fuchsia::test]
437    async fn test_detected_disk_format_match() {
438        let matcher = PartitionMatcher {
439            detected_disk_formats: Some(vec![DiskFormat::Fvm, DiskFormat::Minfs]),
440            ..Default::default()
441        };
442        assert_eq!(check_partition_matches(&matcher).await, true);
443    }
444
445    #[fuchsia::test]
446    async fn test_detected_disk_format_mismatch() {
447        let matcher = PartitionMatcher {
448            detected_disk_formats: Some(vec![DiskFormat::Fxfs, DiskFormat::Minfs]),
449            ..Default::default()
450        };
451        assert_eq!(check_partition_matches(&matcher).await, false);
452    }
453}