use crate::partition::PartitionBackend;
use crate::partitions_directory::PartitionsDirectory;
use anyhow::{anyhow, Context as _, Error};
use block_client::{
AsBlockProxy, BlockClient as _, BufferSlice, MutableBufferSlice, RemoteBlockClient, VmoId,
WriteOptions,
};
use block_server::async_interface::SessionManager;
use block_server::BlockServer;
use futures::lock::Mutex;
use futures::stream::TryStreamExt as _;
use std::collections::BTreeMap;
use std::ops::Range;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::{Arc, Weak};
use zx::AsHandleRef as _;
use {fidl_fuchsia_storage_partitions as fpartitions, fuchsia_async as fasync};
fn partition_directory_entry_name(index: u32) -> String {
format!("part-{:03}", index)
}
pub struct GptPartition {
gpt: Weak<GptManager>,
block_client: Arc<RemoteBlockClient>,
block_range: Range<u64>,
index: u32,
}
impl GptPartition {
pub fn new(
gpt: &Arc<GptManager>,
block_client: Arc<RemoteBlockClient>,
index: u32,
block_range: Range<u64>,
) -> Arc<Self> {
debug_assert!(block_range.end >= block_range.start);
Arc::new(Self { gpt: Arc::downgrade(gpt), block_client, block_range, index })
}
pub async fn terminate(&self) {
if let Err(error) = self.block_client.close().await {
log::warn!(error:?; "Failed to close block client");
}
}
pub fn index(&self) -> u32 {
self.index
}
pub fn block_size(&self) -> u32 {
self.block_client.block_size()
}
pub fn block_count(&self) -> u64 {
self.block_range.end - self.block_range.start
}
pub async fn attach_vmo(&self, vmo: &zx::Vmo) -> Result<VmoId, zx::Status> {
self.block_client.attach_vmo(vmo).await
}
pub async fn detach_vmo(&self, vmoid: VmoId) -> Result<(), zx::Status> {
self.block_client.detach_vmo(vmoid).await
}
pub async fn get_info(&self) -> Result<block_server::PartitionInfo, zx::Status> {
if let Some(gpt) = self.gpt.upgrade() {
let inner = gpt.inner.lock().await;
inner
.gpt
.partitions()
.get(&self.index)
.map(convert_partition_info)
.ok_or(zx::Status::BAD_STATE)
} else {
Err(zx::Status::BAD_STATE)
}
}
pub async fn read(
&self,
device_block_offset: u64,
block_count: u32,
vmo_id: &VmoId,
vmo_offset: u64, ) -> Result<(), zx::Status> {
let dev_offset = self
.absolute_offset(device_block_offset, block_count)
.map(|offset| offset * self.block_size() as u64)?;
let buffer = MutableBufferSlice::new_with_vmo_id(
vmo_id,
vmo_offset,
(block_count * self.block_size()) as u64,
);
self.block_client.read_at(buffer, dev_offset).await
}
pub async fn write(
&self,
device_block_offset: u64,
block_count: u32,
vmo_id: &VmoId,
vmo_offset: u64, opts: WriteOptions,
) -> Result<(), zx::Status> {
let dev_offset = self
.absolute_offset(device_block_offset, block_count)
.map(|offset| offset * self.block_size() as u64)?;
let buffer = BufferSlice::new_with_vmo_id(
vmo_id,
vmo_offset,
(block_count * self.block_size()) as u64,
);
self.block_client.write_at_with_opts(buffer, dev_offset, opts).await
}
pub async fn flush(&self) -> Result<(), zx::Status> {
self.block_client.flush().await
}
pub async fn trim(
&self,
mut device_block_offset: u64,
block_count: u32,
) -> Result<(), zx::Status> {
device_block_offset = self.absolute_offset(device_block_offset, block_count)?;
self.block_client.trim(device_block_offset..device_block_offset + block_count as u64).await
}
fn absolute_offset(&self, mut offset: u64, len: u32) -> Result<u64, zx::Status> {
offset = offset.checked_add(self.block_range.start).ok_or(zx::Status::OUT_OF_RANGE)?;
let end = offset.checked_add(len as u64).ok_or(zx::Status::OUT_OF_RANGE)?;
if end > self.block_range.end {
Err(zx::Status::OUT_OF_RANGE)
} else {
Ok(offset)
}
}
}
fn convert_partition_info(info: &gpt::PartitionInfo) -> block_server::PartitionInfo {
block_server::PartitionInfo {
block_range: Some(info.start_block..info.start_block + info.num_blocks),
type_guid: info.type_guid.to_bytes(),
instance_guid: info.instance_guid.to_bytes(),
name: Some(info.label.clone()),
flags: info.flags,
}
}
struct PendingTransaction {
transaction: gpt::Transaction,
client_koid: zx::Koid,
added_partitions: Vec<u32>,
_signal_task: fasync::Task<()>,
}
struct Inner {
gpt: gpt::Gpt,
partitions: BTreeMap<u32, Arc<BlockServer<SessionManager<PartitionBackend>>>>,
partitions_dir: PartitionsDirectory,
pending_transaction: Option<PendingTransaction>,
}
impl Inner {
fn ensure_transaction_matches(&self, transaction: &zx::EventPair) -> Result<(), zx::Status> {
if let Some(pending) = self.pending_transaction.as_ref() {
if transaction.get_koid()? == pending.client_koid {
Ok(())
} else {
Err(zx::Status::BAD_HANDLE)
}
} else {
Err(zx::Status::BAD_STATE)
}
}
async fn bind_partition(
&mut self,
parent: &Arc<GptManager>,
index: u32,
info: gpt::PartitionInfo,
) -> Result<(), Error> {
log::info!("GPT part {index}: {info:?}");
let partition = PartitionBackend::new(GptPartition::new(
parent,
self.gpt.client().clone(),
index,
info.start_block
..info
.start_block
.checked_add(info.num_blocks)
.ok_or_else(|| anyhow!("Overflow in partition range"))?,
));
let block_server = Arc::new(BlockServer::new(parent.block_size, partition));
self.partitions_dir.add_entry(
&partition_directory_entry_name(index),
Arc::downgrade(&block_server),
Arc::downgrade(parent),
index as usize,
);
self.partitions.insert(index, block_server);
Ok(())
}
async fn bind_all_partitions(&mut self, parent: &Arc<GptManager>) -> Result<(), Error> {
self.partitions.clear();
self.partitions_dir.clear();
for (index, info) in self.gpt.partitions().clone() {
self.bind_partition(parent, index, info).await?;
}
Ok(())
}
fn add_partition(&mut self, info: gpt::PartitionInfo) -> Result<usize, gpt::AddPartitionError> {
let pending = self.pending_transaction.as_mut().unwrap();
let idx = self.gpt.add_partition(&mut pending.transaction, info)?;
pending.added_partitions.push(idx as u32);
Ok(idx)
}
}
pub struct GptManager {
block_size: u32,
block_count: u64,
inner: Mutex<Inner>,
shutdown: AtomicBool,
}
impl std::fmt::Debug for GptManager {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
f.debug_struct("GptManager")
.field("block_size", &self.block_size)
.field("block_count", &self.block_count)
.finish()
}
}
impl GptManager {
pub async fn new(
block: impl AsBlockProxy,
partitions_dir: Arc<vfs::directory::immutable::Simple>,
) -> Result<Arc<Self>, Error> {
log::info!("Binding to GPT");
let client = Arc::new(RemoteBlockClient::new(block).await?);
let block_size = client.block_size();
let block_count = client.block_count();
let gpt = gpt::Gpt::open(client).await.context("Failed to load GPT")?;
let this = Arc::new(Self {
block_size,
block_count,
inner: Mutex::new(Inner {
gpt,
partitions: BTreeMap::new(),
partitions_dir: PartitionsDirectory::new(partitions_dir),
pending_transaction: None,
}),
shutdown: AtomicBool::new(false),
});
log::info!("Bind to GPT OK, binding partitions");
this.inner.lock().await.bind_all_partitions(&this).await?;
log::info!("Starting all partitions OK!");
Ok(this)
}
pub fn block_size(&self) -> u32 {
self.block_size
}
pub fn block_count(&self) -> u64 {
self.block_count
}
pub async fn create_transaction(self: &Arc<Self>) -> Result<zx::EventPair, zx::Status> {
let mut inner = self.inner.lock().await;
if inner.pending_transaction.is_some() {
return Err(zx::Status::ALREADY_EXISTS);
}
let transaction = inner.gpt.create_transaction().unwrap();
let (client_end, server_end) = zx::EventPair::create();
let client_koid = client_end.get_koid()?;
let signal_waiter = fasync::OnSignals::new(server_end, zx::Signals::EVENTPAIR_PEER_CLOSED);
let this = self.clone();
let task = fasync::Task::spawn(async move {
let _ = signal_waiter.await;
let mut inner = this.inner.lock().await;
if inner.pending_transaction.as_ref().map_or(false, |t| t.client_koid == client_koid) {
inner.pending_transaction = None;
}
});
inner.pending_transaction = Some(PendingTransaction {
transaction,
client_koid,
added_partitions: vec![],
_signal_task: task,
});
Ok(client_end)
}
pub async fn commit_transaction(
self: &Arc<Self>,
transaction: zx::EventPair,
) -> Result<(), zx::Status> {
let mut inner = self.inner.lock().await;
inner.ensure_transaction_matches(&transaction)?;
let pending = std::mem::take(&mut inner.pending_transaction).unwrap();
if let Err(err) = inner.gpt.commit_transaction(pending.transaction).await {
log::error!(err:?; "Failed to commit transaction");
return Err(zx::Status::IO);
}
for idx in pending.added_partitions {
let info = inner.gpt.partitions().get(&idx).ok_or(zx::Status::BAD_STATE)?.clone();
inner.bind_partition(self, idx, info).await.map_err(|err| {
log::error!(err:?; "Failed to bind partition");
zx::Status::BAD_STATE
})?;
}
Ok(())
}
pub async fn add_partition(
&self,
request: fpartitions::PartitionsManagerAddPartitionRequest,
) -> Result<(), zx::Status> {
let mut inner = self.inner.lock().await;
inner.ensure_transaction_matches(
request.transaction.as_ref().ok_or(zx::Status::BAD_HANDLE)?,
)?;
let info = gpt::PartitionInfo {
label: request.name.ok_or(zx::Status::INVALID_ARGS)?,
type_guid: request
.type_guid
.map(|value| gpt::Guid::from_bytes(value.value))
.ok_or(zx::Status::INVALID_ARGS)?,
instance_guid: request
.instance_guid
.map(|value| gpt::Guid::from_bytes(value.value))
.unwrap_or_else(|| gpt::Guid::generate()),
start_block: 0,
num_blocks: request.num_blocks.ok_or(zx::Status::INVALID_ARGS)?,
flags: request.flags.unwrap_or_default(),
};
let idx = inner.add_partition(info)?;
let partition =
inner.pending_transaction.as_ref().unwrap().transaction.partitions.get(idx).unwrap();
log::info!(
"Allocated partition {:?} at {:?}",
partition.label,
partition.start_block..partition.start_block + partition.num_blocks
);
Ok(())
}
pub async fn handle_partitions_requests(
&self,
gpt_index: usize,
mut requests: fpartitions::PartitionRequestStream,
) -> Result<(), zx::Status> {
while let Some(request) = requests.try_next().await.unwrap() {
match request {
fpartitions::PartitionRequest::UpdateMetadata { payload, responder } => {
responder
.send(
self.update_partition_metadata(gpt_index, payload)
.await
.map_err(|status| status.into_raw()),
)
.unwrap_or_else(
|err| log::error!(err:?; "Failed to send UpdateMetadata response"),
);
}
}
}
Ok(())
}
async fn update_partition_metadata(
&self,
gpt_index: usize,
request: fpartitions::PartitionUpdateMetadataRequest,
) -> Result<(), zx::Status> {
let mut inner = self.inner.lock().await;
inner.ensure_transaction_matches(
request.transaction.as_ref().ok_or(zx::Status::BAD_HANDLE)?,
)?;
let transaction = &mut inner.pending_transaction.as_mut().unwrap().transaction;
let entry = transaction.partitions.get_mut(gpt_index).ok_or(zx::Status::BAD_STATE)?;
if let Some(type_guid) = request.type_guid.as_ref().cloned() {
entry.type_guid = gpt::Guid::from_bytes(type_guid.value);
}
if let Some(flags) = request.flags.as_ref() {
entry.flags = *flags;
}
Ok(())
}
pub async fn reset_partition_table(
self: &Arc<Self>,
partitions: Vec<gpt::PartitionInfo>,
) -> Result<(), zx::Status> {
let mut inner = self.inner.lock().await;
if inner.pending_transaction.is_some() {
return Err(zx::Status::BAD_STATE);
}
log::info!("Resetting gpt. Expect data loss!!!");
let mut transaction = inner.gpt.create_transaction().unwrap();
transaction.partitions = partitions;
inner.gpt.commit_transaction(transaction).await?;
log::info!("Rebinding partitions...");
if let Err(err) = inner.bind_all_partitions(&self).await {
log::error!(err:?; "Failed to rebind partitions");
return Err(zx::Status::BAD_STATE);
}
log::info!("Rebinding partitions OK!");
Ok(())
}
pub async fn shutdown(self: Arc<Self>) {
log::info!("Shutting down gpt");
let mut inner = self.inner.lock().await;
inner.partitions_dir.clear();
inner.partitions.clear();
self.shutdown.store(true, Ordering::Relaxed);
log::info!("Shutting down gpt OK");
}
}
impl Drop for GptManager {
fn drop(&mut self) {
assert!(self.shutdown.load(Ordering::Relaxed), "Did you forget to shutdown?");
}
}
#[cfg(test)]
mod tests {
use super::GptManager;
use block_client::{BlockClient as _, BufferSlice, MutableBufferSlice, RemoteBlockClient};
use block_server::WriteOptions;
use fake_block_server::{FakeServer, FakeServerOptions};
use fidl::HandleBased as _;
use fuchsia_component::client::connect_to_named_protocol_at_dir_root;
use gpt::{Gpt, Guid, PartitionInfo};
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use vfs::directory::entry_container::Directory as _;
use vfs::execution_scope::ExecutionScope;
use vfs::ObjectRequest;
use {
fidl_fuchsia_hardware_block as fblock, fidl_fuchsia_hardware_block_volume as fvolume,
fidl_fuchsia_io as fio, fidl_fuchsia_storage_partitions as fpartitions,
fuchsia_async as fasync,
};
async fn setup(
block_size: u32,
block_count: u64,
partitions: Vec<PartitionInfo>,
) -> (Arc<FakeServer>, Arc<vfs::directory::immutable::Simple>) {
setup_with_options(
FakeServerOptions { block_count: Some(block_count), block_size, ..Default::default() },
partitions,
)
.await
}
async fn setup_with_options(
opts: FakeServerOptions<'_>,
partitions: Vec<PartitionInfo>,
) -> (Arc<FakeServer>, Arc<vfs::directory::immutable::Simple>) {
let server = Arc::new(FakeServer::from(opts));
{
let (block_client, block_server) =
fidl::endpoints::create_proxy::<fblock::BlockMarker>();
let volume_stream = fidl::endpoints::ServerEnd::<fvolume::VolumeMarker>::from(
block_server.into_channel(),
)
.into_stream();
let server_clone = server.clone();
let _task = fasync::Task::spawn(async move { server_clone.serve(volume_stream).await });
let client = Arc::new(RemoteBlockClient::new(block_client).await.unwrap());
Gpt::format(client, partitions).await.unwrap();
}
(server, vfs::directory::immutable::simple())
}
#[fuchsia::test]
async fn load_unformatted_gpt() {
let vmo = zx::Vmo::create(4096).unwrap();
let server = Arc::new(FakeServer::from_vmo(512, vmo));
GptManager::new(server.block_proxy(), vfs::directory::immutable::simple())
.await
.expect_err("load should fail");
}
#[fuchsia::test]
async fn load_formatted_empty_gpt() {
let (block_device, partitions_dir) = setup(512, 8, vec![]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir)
.await
.expect("load should succeed");
runner.shutdown().await;
}
#[fuchsia::test]
async fn load_formatted_gpt_with_one_partition() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_GUID: [u8; 16] = [2u8; 16];
const PART_NAME: &str = "part";
let (block_device, partitions_dir) = setup(
512,
8,
vec![PartitionInfo {
label: PART_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
}],
)
.await;
let partitions_dir_clone = partitions_dir.clone();
let runner = GptManager::new(block_device.block_proxy(), partitions_dir_clone)
.await
.expect("load should succeed");
partitions_dir.get_entry("part-000").expect("No entry found");
partitions_dir.get_entry("part-001").map(|_| ()).expect_err("Extra entry found");
runner.shutdown().await;
}
#[fuchsia::test]
async fn load_formatted_gpt_with_two_partitions() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_1_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_2_GUID: [u8; 16] = [3u8; 16];
const PART_1_NAME: &str = "part1";
const PART_2_NAME: &str = "part2";
let (block_device, partitions_dir) = setup(
512,
8,
vec![
PartitionInfo {
label: PART_1_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_1_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
},
PartitionInfo {
label: PART_2_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_2_GUID),
start_block: 5,
num_blocks: 1,
flags: 0,
},
],
)
.await;
let partitions_dir_clone = partitions_dir.clone();
let runner = GptManager::new(block_device.block_proxy(), partitions_dir_clone)
.await
.expect("load should succeed");
partitions_dir.get_entry("part-000").expect("No entry found");
partitions_dir.get_entry("part-001").expect("No entry found");
partitions_dir.get_entry("part-002").map(|_| ()).expect_err("Extra entry found");
runner.shutdown().await;
}
#[fuchsia::test]
async fn partition_io() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_GUID: [u8; 16] = [2u8; 16];
const PART_NAME: &str = "part";
let (block_device, partitions_dir) = setup(
512,
8,
vec![PartitionInfo {
label: PART_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
}],
)
.await;
let partitions_dir_clone = partitions_dir.clone();
let runner = GptManager::new(block_device.block_proxy(), partitions_dir_clone)
.await
.expect("load should succeed");
let (proxy, server_end) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let flags =
fio::Flags::PERM_CONNECT | fio::Flags::PERM_TRAVERSE | fio::Flags::PERM_ENUMERATE;
let options = fio::Options::default();
let scope = vfs::execution_scope::ExecutionScope::new();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-000").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end.into_channel().into()),
)
.unwrap();
let block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&proxy, "volume")
.expect("Failed to open block service");
let client = RemoteBlockClient::new(block).await.expect("Failed to create block client");
assert_eq!(client.block_count(), 1);
assert_eq!(client.block_size(), 512);
let buf = vec![0xabu8; 512];
client.write_at(BufferSlice::Memory(&buf[..]), 0).await.expect("write_at failed");
client
.write_at(BufferSlice::Memory(&buf[..]), 512)
.await
.expect_err("write_at should fail when writing past partition end");
let mut buf2 = vec![0u8; 512];
client.read_at(MutableBufferSlice::Memory(&mut buf2[..]), 0).await.expect("read_at failed");
assert_eq!(buf, buf2);
client
.read_at(MutableBufferSlice::Memory(&mut buf2[..]), 512)
.await
.expect_err("read_at should fail when reading past partition end");
runner.shutdown().await;
let mut buf = vec![0u8; 512];
let client = RemoteBlockClient::new(block_device.block_proxy()).await.unwrap();
client.read_at(MutableBufferSlice::Memory(&mut buf[..]), 2048).await.unwrap();
assert_eq!(&buf[..], &[0xabu8; 512]);
}
#[fuchsia::test]
async fn load_formatted_gpt_with_invalid_primary_header() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_1_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_2_GUID: [u8; 16] = [3u8; 16];
const PART_1_NAME: &str = "part1";
const PART_2_NAME: &str = "part2";
let (block_device, partitions_dir) = setup(
512,
8,
vec![
PartitionInfo {
label: PART_1_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_1_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
},
PartitionInfo {
label: PART_2_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_2_GUID),
start_block: 5,
num_blocks: 1,
flags: 0,
},
],
)
.await;
{
let (client, stream) =
fidl::endpoints::create_proxy_and_stream::<fvolume::VolumeMarker>();
let server = block_device.clone();
let _task = fasync::Task::spawn(async move { server.serve(stream).await });
let client = RemoteBlockClient::new(client).await.unwrap();
client.write_at(BufferSlice::Memory(&[0xffu8; 512]), 512).await.unwrap();
}
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
partitions_dir.get_entry("part-000").expect("No entry found");
partitions_dir.get_entry("part-001").expect("No entry found");
runner.shutdown().await;
}
#[fuchsia::test]
async fn load_formatted_gpt_with_invalid_primary_partition_table() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_1_GUID: [u8; 16] = [2u8; 16];
const PART_INSTANCE_2_GUID: [u8; 16] = [3u8; 16];
const PART_1_NAME: &str = "part1";
const PART_2_NAME: &str = "part2";
let (block_device, partitions_dir) = setup(
512,
8,
vec![
PartitionInfo {
label: PART_1_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_1_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
},
PartitionInfo {
label: PART_2_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_INSTANCE_2_GUID),
start_block: 5,
num_blocks: 1,
flags: 0,
},
],
)
.await;
{
let (client, stream) =
fidl::endpoints::create_proxy_and_stream::<fvolume::VolumeMarker>();
let server = block_device.clone();
let _task = fasync::Task::spawn(async move { server.serve(stream).await });
let client = RemoteBlockClient::new(client).await.unwrap();
client.write_at(BufferSlice::Memory(&[0xffu8; 512]), 1024).await.unwrap();
}
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
partitions_dir.get_entry("part-000").expect("No entry found");
partitions_dir.get_entry("part-001").expect("No entry found");
runner.shutdown().await;
}
#[fuchsia::test]
async fn force_access_passed_through() {
const BLOCK_SIZE: u32 = 512;
const BLOCK_COUNT: u64 = 1024;
struct Observer(Arc<AtomicBool>);
impl fake_block_server::Observer for Observer {
fn write(
&self,
_device_block_offset: u64,
_block_count: u32,
_vmo: &Arc<zx::Vmo>,
_vmo_offset: u64,
opts: WriteOptions,
) -> fake_block_server::WriteAction {
assert_eq!(
opts.contains(WriteOptions::FORCE_ACCESS),
self.0.load(Ordering::Relaxed)
);
fake_block_server::WriteAction::Write
}
}
let expect_force_access = Arc::new(AtomicBool::new(false));
let (server, partitions_dir) = setup_with_options(
FakeServerOptions {
block_count: Some(BLOCK_COUNT),
block_size: BLOCK_SIZE,
observer: Some(Box::new(Observer(expect_force_access.clone()))),
..Default::default()
},
vec![PartitionInfo {
label: "foo".to_string(),
type_guid: Guid::from_bytes([1; 16]),
instance_guid: Guid::from_bytes([2; 16]),
start_block: 4,
num_blocks: 1,
flags: 0,
}],
)
.await;
let manager = GptManager::new(server.block_proxy(), partitions_dir.clone()).await.unwrap();
let (proxy, server_end) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let flags =
fio::Flags::PERM_CONNECT | fio::Flags::PERM_TRAVERSE | fio::Flags::PERM_ENUMERATE;
let options = fio::Options::default();
let scope = ExecutionScope::new();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-000").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end.into_channel().into()),
)
.unwrap();
let block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&proxy, "volume")
.expect("Failed to open block service");
let client = RemoteBlockClient::new(block).await.expect("Failed to create block client");
let buffer = vec![0; BLOCK_SIZE as usize];
client.write_at(BufferSlice::Memory(&buffer), 0).await.unwrap();
expect_force_access.store(true, Ordering::Relaxed);
client
.write_at_with_opts(BufferSlice::Memory(&buffer), 0, WriteOptions::FORCE_ACCESS)
.await
.unwrap();
manager.shutdown().await;
}
#[fuchsia::test]
async fn commit_transaction() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_1_INSTANCE_GUID: [u8; 16] = [2u8; 16];
const PART_1_NAME: &str = "part";
const PART_2_INSTANCE_GUID: [u8; 16] = [3u8; 16];
const PART_2_NAME: &str = "part2";
let (block_device, partitions_dir) = setup(
512,
16,
vec![
PartitionInfo {
label: PART_1_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_1_INSTANCE_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
},
PartitionInfo {
label: PART_2_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_2_INSTANCE_GUID),
start_block: 5,
num_blocks: 1,
flags: 0,
},
],
)
.await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let (part_0_dir, server_end_0) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let (part_1_dir, server_end_1) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let flags =
fio::Flags::PERM_CONNECT | fio::Flags::PERM_TRAVERSE | fio::Flags::PERM_ENUMERATE;
let options = fio::Options::default();
let scope = vfs::execution_scope::ExecutionScope::new();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-000").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end_0.into_channel().into()),
)
.unwrap();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-001").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end_1.into_channel().into()),
)
.unwrap();
let part_0_proxy = connect_to_named_protocol_at_dir_root::<fpartitions::PartitionMarker>(
&part_0_dir,
"partition",
)
.expect("Failed to open Partition service");
let part_1_proxy = connect_to_named_protocol_at_dir_root::<fpartitions::PartitionMarker>(
&part_1_dir,
"partition",
)
.expect("Failed to open Partition service");
let transaction = runner.create_transaction().await.expect("Failed to create transaction");
part_0_proxy
.update_metadata(fpartitions::PartitionUpdateMetadataRequest {
transaction: Some(transaction.duplicate_handle(zx::Rights::SAME_RIGHTS).unwrap()),
type_guid: Some(fidl_fuchsia_hardware_block_partition::Guid {
value: [0xffu8; 16],
}),
..Default::default()
})
.await
.expect("FIDL error")
.expect("Failed to update_metadata");
part_1_proxy
.update_metadata(fpartitions::PartitionUpdateMetadataRequest {
transaction: Some(transaction.duplicate_handle(zx::Rights::SAME_RIGHTS).unwrap()),
flags: Some(u64::MAX),
..Default::default()
})
.await
.expect("FIDL error")
.expect("Failed to update_metadata");
runner.commit_transaction(transaction).await.expect("Failed to commit transaction");
let part_0_block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&part_0_dir, "volume")
.expect("Failed to open Volume service");
let (status, guid) = part_0_block.get_type_guid().await.expect("FIDL error");
assert_eq!(zx::Status::from_raw(status), zx::Status::OK);
assert_eq!(guid.unwrap().value, [0xffu8; 16]);
let part_1_block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&part_1_dir, "volume")
.expect("Failed to open Volume service");
let (status, guid) = part_1_block.get_type_guid().await.expect("FIDL error");
assert_eq!(zx::Status::from_raw(status), zx::Status::OK);
assert_eq!(guid.unwrap().value, PART_TYPE_GUID);
runner.shutdown().await;
}
#[fuchsia::test]
async fn reset_partition_tables() {
const PART_TYPE_GUID: [u8; 16] = [2u8; 16];
const PART_1_INSTANCE_GUID: [u8; 16] = [2u8; 16];
const PART_1_NAME: &str = "part";
const PART_2_INSTANCE_GUID: [u8; 16] = [3u8; 16];
const PART_2_NAME: &str = "part2";
const PART_3_NAME: &str = "part3";
const PART_4_NAME: &str = "part4";
let (block_device, partitions_dir) = setup(
512,
1048576 / 512,
vec![
PartitionInfo {
label: PART_1_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_1_INSTANCE_GUID),
start_block: 4,
num_blocks: 1,
flags: 0,
},
PartitionInfo {
label: PART_2_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes(PART_2_INSTANCE_GUID),
start_block: 5,
num_blocks: 1,
flags: 0,
},
],
)
.await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let nil_entry = PartitionInfo {
label: "".to_string(),
type_guid: Guid::from_bytes([0u8; 16]),
instance_guid: Guid::from_bytes([0u8; 16]),
start_block: 0,
num_blocks: 0,
flags: 0,
};
let mut new_partitions = vec![nil_entry; 128];
new_partitions[0] = PartitionInfo {
label: PART_3_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes([1u8; 16]),
start_block: 64,
num_blocks: 2,
flags: 0,
};
new_partitions[2] = PartitionInfo {
label: PART_4_NAME.to_string(),
type_guid: Guid::from_bytes(PART_TYPE_GUID),
instance_guid: Guid::from_bytes([2u8; 16]),
start_block: 66,
num_blocks: 4,
flags: 0,
};
runner.reset_partition_table(new_partitions).await.expect("reset_partition_table failed");
partitions_dir.get_entry("part-000").expect("No entry found");
partitions_dir.get_entry("part-001").map(|_| ()).expect_err("Extra entry found");
partitions_dir.get_entry("part-002").expect("No entry found");
let (proxy, server_end) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let flags =
fio::Flags::PERM_CONNECT | fio::Flags::PERM_TRAVERSE | fio::Flags::PERM_ENUMERATE;
let options = fio::Options::default();
let scope = vfs::execution_scope::ExecutionScope::new();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-000").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end.into_channel().into()),
)
.unwrap();
let block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&proxy, "volume")
.expect("Failed to open block service");
let (status, name) = block.get_name().await.expect("FIDL error");
assert_eq!(zx::Status::from_raw(status), zx::Status::OK);
assert_eq!(name.unwrap(), PART_3_NAME);
runner.shutdown().await;
}
#[fuchsia::test]
async fn reset_partition_tables_fails_if_too_many_partitions() {
let (block_device, partitions_dir) = setup(512, 8, vec![]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let nil_entry = PartitionInfo {
label: "".to_string(),
type_guid: Guid::from_bytes([0u8; 16]),
instance_guid: Guid::from_bytes([0u8; 16]),
start_block: 0,
num_blocks: 0,
flags: 0,
};
let new_partitions = vec![nil_entry; 128];
runner
.reset_partition_table(new_partitions)
.await
.expect_err("reset_partition_table should fail");
runner.shutdown().await;
}
#[fuchsia::test]
async fn reset_partition_tables_fails_if_too_large_partitions() {
let (block_device, partitions_dir) = setup(512, 64, vec![]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let new_partitions = vec![
PartitionInfo {
label: "a".to_string(),
type_guid: Guid::from_bytes([1u8; 16]),
instance_guid: Guid::from_bytes([1u8; 16]),
start_block: 4,
num_blocks: 2,
flags: 0,
},
PartitionInfo {
label: "b".to_string(),
type_guid: Guid::from_bytes([2u8; 16]),
instance_guid: Guid::from_bytes([2u8; 16]),
start_block: 6,
num_blocks: 200,
flags: 0,
},
];
runner
.reset_partition_table(new_partitions)
.await
.expect_err("reset_partition_table should fail");
runner.shutdown().await;
}
#[fuchsia::test]
async fn reset_partition_tables_fails_if_partition_overlaps_metadata() {
let (block_device, partitions_dir) = setup(512, 64, vec![]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let new_partitions = vec![PartitionInfo {
label: "a".to_string(),
type_guid: Guid::from_bytes([1u8; 16]),
instance_guid: Guid::from_bytes([1u8; 16]),
start_block: 1,
num_blocks: 2,
flags: 0,
}];
runner
.reset_partition_table(new_partitions)
.await
.expect_err("reset_partition_table should fail");
runner.shutdown().await;
}
#[fuchsia::test]
async fn reset_partition_tables_fails_if_partitions_overlap() {
let (block_device, partitions_dir) = setup(512, 64, vec![]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let new_partitions = vec![
PartitionInfo {
label: "a".to_string(),
type_guid: Guid::from_bytes([1u8; 16]),
instance_guid: Guid::from_bytes([1u8; 16]),
start_block: 32,
num_blocks: 2,
flags: 0,
},
PartitionInfo {
label: "b".to_string(),
type_guid: Guid::from_bytes([2u8; 16]),
instance_guid: Guid::from_bytes([2u8; 16]),
start_block: 33,
num_blocks: 1,
flags: 0,
},
];
runner
.reset_partition_table(new_partitions)
.await
.expect_err("reset_partition_table should fail");
runner.shutdown().await;
}
#[fuchsia::test]
async fn add_partition() {
let (block_device, partitions_dir) = setup(512, 64, vec![PartitionInfo::nil(); 64]).await;
let runner = GptManager::new(block_device.block_proxy(), partitions_dir.clone())
.await
.expect("load should succeed");
let transaction = runner.create_transaction().await.expect("Create transaction failed");
let request = fpartitions::PartitionsManagerAddPartitionRequest {
transaction: Some(transaction.duplicate_handle(zx::Rights::SAME_RIGHTS).unwrap()),
name: Some("a".to_string()),
type_guid: Some(fidl_fuchsia_hardware_block_partition::Guid { value: [1u8; 16] }),
num_blocks: Some(2),
..Default::default()
};
runner.add_partition(request).await.expect("add_partition failed");
runner.commit_transaction(transaction).await.expect("add_partition failed");
let (proxy, server_end) = fidl::endpoints::create_proxy::<fio::DirectoryMarker>();
let flags =
fio::Flags::PERM_CONNECT | fio::Flags::PERM_TRAVERSE | fio::Flags::PERM_ENUMERATE;
let options = fio::Options::default();
let scope = vfs::execution_scope::ExecutionScope::new();
partitions_dir
.clone()
.open3(
scope.clone(),
vfs::path::Path::validate_and_split("part-000").unwrap(),
flags.clone(),
&mut ObjectRequest::new(flags, &options, server_end.into_channel().into()),
)
.unwrap();
let block =
connect_to_named_protocol_at_dir_root::<fvolume::VolumeMarker>(&proxy, "volume")
.expect("Failed to open block service");
let client = RemoteBlockClient::new(block).await.expect("Failed to create block client");
assert_eq!(client.block_count(), 2);
assert_eq!(client.block_size(), 512);
runner.shutdown().await;
}
}