1use 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#[derive(Default, Clone)]
24pub struct PartitionMatcher {
25 pub type_guids: Option<Vec<[u8; 16]>>,
27 pub instance_guids: Option<Vec<[u8; 16]>>,
29 pub labels: Option<Vec<String>>,
30 pub detected_disk_formats: Option<Vec<DiskFormat>>,
31 pub parent_device: Option<String>,
33 pub ignore_prefix: Option<String>,
35 pub ignore_if_path_contains: Option<String>,
37}
38
39const BLOCK_DEV_PATH: &str = "/dev/class/block/";
40
41pub 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
54pub 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
99pub 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}