1use crate::buffer::{BufferFuture, BufferRef, MutableBufferRef};
6use anyhow::{bail, Error};
7use async_trait::async_trait;
8use block_protocol::WriteOptions;
9use futures::channel::oneshot::{channel, Sender};
10use std::future::Future;
11use std::mem::ManuallyDrop;
12use std::ops::{Deref, Range};
13use std::sync::{Arc, OnceLock};
14
15pub mod buffer;
16pub mod buffer_allocator;
17
18#[cfg(target_os = "fuchsia")]
19pub mod block_device;
20
21#[cfg(target_family = "unix")]
22pub mod file_backed_device;
23
24pub mod fake_device;
25
26#[async_trait]
27pub trait Device: Send + Sync {
29 fn allocate_buffer(&self, size: usize) -> BufferFuture<'_>;
32
33 fn block_size(&self) -> u32;
35
36 fn block_count(&self) -> u64;
39
40 fn size(&self) -> u64 {
42 self.block_size() as u64 * self.block_count()
43 }
44
45 async fn read(&self, offset: u64, buffer: MutableBufferRef<'_>) -> Result<(), Error>;
47
48 async fn write(&self, offset: u64, buffer: BufferRef<'_>) -> Result<(), Error> {
50 self.write_with_opts(offset, buffer, WriteOptions::empty()).await
51 }
52
53 async fn write_with_opts(
55 &self,
56 offset: u64,
57 buffer: BufferRef<'_>,
58 opts: WriteOptions,
59 ) -> Result<(), Error>;
60
61 async fn trim(&self, range: Range<u64>) -> Result<(), Error>;
63
64 async fn close(&self) -> Result<(), Error>;
67
68 async fn flush(&self) -> Result<(), Error>;
70
71 fn reopen(&self, _read_only: bool) {
73 unreachable!();
74 }
75 fn is_read_only(&self) -> bool;
77
78 fn supports_trim(&self) -> bool;
80
81 fn snapshot(&self) -> Result<DeviceHolder, Error> {
83 bail!("Not supported");
84 }
85
86 fn discard_random_since_last_flush(&self) -> Result<(), Error> {
88 bail!("Not supported");
89 }
90
91 fn poison(&self) -> Result<(), Error> {
93 bail!("Not supported");
94 }
95}
96
97pub struct DeviceHolder {
104 device: ManuallyDrop<Arc<dyn Device>>,
105 on_drop: OnceLock<Sender<DeviceHolder>>,
106}
107
108impl DeviceHolder {
109 pub fn new(device: impl Device + 'static) -> Self {
110 DeviceHolder { device: ManuallyDrop::new(Arc::new(device)), on_drop: OnceLock::new() }
111 }
112
113 pub fn ensure_unique(&self) {
116 assert_eq!(Arc::strong_count(&self.device), 1);
117 }
118
119 pub fn take_when_dropped(&self) -> impl Future<Output = DeviceHolder> {
120 let (sender, receiver) = channel::<DeviceHolder>();
121 self.on_drop
122 .set(sender)
123 .unwrap_or_else(|_| panic!("take_when_dropped should only be called once"));
124 async { receiver.await.unwrap() }
125 }
126}
127
128impl Drop for DeviceHolder {
129 fn drop(&mut self) {
130 if let Some(sender) = self.on_drop.take() {
131 let device = ManuallyDrop::new(unsafe { ManuallyDrop::take(&mut self.device) });
133 let _ = sender.send(DeviceHolder { device, on_drop: OnceLock::new() });
135 } else {
136 unsafe { ManuallyDrop::drop(&mut self.device) }
138 }
139 }
140}
141
142impl Deref for DeviceHolder {
143 type Target = Arc<dyn Device>;
144
145 fn deref(&self) -> &Self::Target {
146 &self.device
147 }
148}
149
150#[cfg(test)]
151mod tests {
152 use super::DeviceHolder;
153 use crate::fake_device::FakeDevice;
154
155 #[fuchsia::test]
156 async fn test_take_when_dropped() {
157 let holder = DeviceHolder::new(FakeDevice::new(1, 512));
158 let fut = holder.take_when_dropped();
159 std::mem::drop(holder);
160 fut.await.ensure_unique();
161 }
162}