pub mod constants;
use anyhow::{anyhow, ensure, Context as _, Error};
use block_client::{AsBlockProxy, BlockClient, MutableBufferSlice, RemoteBlockClient};
use fuchsia_zircon as zx;
#[derive(Debug, PartialEq, Clone, Copy)]
pub enum DiskFormat {
Unknown,
Gpt,
Mbr,
Minfs,
Fat,
Blobfs,
Fvm,
Zxcrypt,
BlockVerity,
VbMeta,
BootPart,
Fxfs,
F2fs,
NandBroker,
}
impl DiskFormat {
pub fn as_str(&self) -> &'static str {
match self {
Self::Unknown => "unknown!",
Self::Gpt => "gpt",
Self::Mbr => "mbr",
Self::Minfs => "minfs",
Self::Fat => "fat",
Self::Blobfs => "blobfs",
Self::Fvm => "fvm",
Self::Zxcrypt => "zxcrypt",
Self::BlockVerity => "block verity",
Self::VbMeta => "vbmeta",
Self::BootPart => "bootpart",
Self::Fxfs => "fxfs",
Self::F2fs => "f2fs",
Self::NandBroker => "nand broker",
}
}
}
impl From<&str> for DiskFormat {
fn from(s: &str) -> Self {
match s {
"gpt" => Self::Gpt,
"mbr" => Self::Mbr,
"minfs" => Self::Minfs,
"fat" => Self::Fat,
"blobfs" => Self::Blobfs,
"fvm" => Self::Fvm,
"zxcrypt" => Self::Zxcrypt,
"block verity" => Self::BlockVerity,
"vbmeta" => Self::VbMeta,
"bootpart" => Self::BootPart,
"fxfs" => Self::Fxfs,
"f2fs" => Self::F2fs,
"nand broker" => Self::NandBroker,
_ => Self::Unknown,
}
}
}
pub fn round_up(val: u64, divisor: u64) -> Option<u64> {
((val.checked_add(divisor.checked_sub(1)?)?).checked_div(divisor)?).checked_mul(divisor)
}
pub async fn detect_disk_format(block_proxy: impl AsBlockProxy) -> DiskFormat {
match detect_disk_format_res(block_proxy).await {
Ok(format) => format,
Err(e) => {
tracing::error!("detect_disk_format failed: {:?}", e);
return DiskFormat::Unknown;
}
}
}
async fn detect_disk_format_res(block_proxy: impl AsBlockProxy) -> Result<DiskFormat, Error> {
let block_info = block_proxy
.get_info()
.await
.context("transport error on get_info call")?
.map_err(zx::Status::from_raw)
.context("get_info call failed")?;
ensure!(block_info.block_size > 0, "block size expected to be non-zero");
let double_block_size = (block_info.block_size)
.checked_mul(2)
.ok_or_else(|| anyhow!("overflow calculating double block size"))?;
let header_size = if constants::HEADER_SIZE > double_block_size {
constants::HEADER_SIZE
} else {
double_block_size
};
let device_size = (block_info.block_size as u64)
.checked_mul(block_info.block_count)
.ok_or_else(|| anyhow!("overflow calculating device size"))?;
if header_size as u64 > device_size {
return Ok(DiskFormat::Unknown);
}
let buffer_size = round_up(header_size as u64, block_info.block_size as u64)
.ok_or_else(|| anyhow!("overflow rounding header size up"))?;
let client =
RemoteBlockClient::new(block_proxy).await.context("RemoteBlockClient::new failed")?;
let mut data = vec![0; buffer_size as usize];
client.read_at(MutableBufferSlice::Memory(&mut data), 0).await.context("read_at failed")?;
if data.starts_with(&constants::FVM_MAGIC) {
return Ok(DiskFormat::Fvm);
}
if data.starts_with(&constants::ZXCRYPT_MAGIC) {
return Ok(DiskFormat::Zxcrypt);
}
if data.starts_with(&constants::BLOCK_VERITY_MAGIC) {
return Ok(DiskFormat::BlockVerity);
}
if &data[block_info.block_size as usize..block_info.block_size as usize + 16]
== &constants::GPT_MAGIC
{
return Ok(DiskFormat::Gpt);
}
if data.starts_with(&constants::MINFS_MAGIC) {
return Ok(DiskFormat::Minfs);
}
if data.starts_with(&constants::BLOBFS_MAGIC) {
return Ok(DiskFormat::Blobfs);
}
if data.starts_with(&constants::VB_META_MAGIC) {
return Ok(DiskFormat::VbMeta);
}
if &data[1024..1024 + constants::F2FS_MAGIC.len()] == &constants::F2FS_MAGIC {
return Ok(DiskFormat::F2fs);
}
if data.starts_with(&constants::FXFS_MAGIC) {
return Ok(DiskFormat::Fxfs);
}
if data[510] == 0x55 && data[511] == 0xAA {
if data[38] == 0x29 || data[66] == 0x29 {
return Ok(DiskFormat::Fat);
}
return Ok(DiskFormat::Mbr);
}
return Ok(DiskFormat::Unknown);
}
#[cfg(test)]
mod tests {
use super::{constants, detect_disk_format_res, DiskFormat};
use anyhow::Error;
use fake_block_server::FakeServer;
use fidl::endpoints::create_proxy_and_stream;
use fidl_fuchsia_hardware_block_volume::VolumeMarker;
use futures::{select, FutureExt};
use std::pin::pin;
async fn get_detected_disk_format(
content: &[u8],
block_count: u64,
block_size: u32,
) -> Result<DiskFormat, Error> {
let (proxy, stream) = create_proxy_and_stream::<VolumeMarker>().unwrap();
let fake_server = FakeServer::new(block_count, block_size, content);
let mut request_handler = pin!(fake_server.serve(stream).fuse());
select! {
_ = request_handler => unreachable!(),
format = detect_disk_format_res(&proxy).fuse() => format,
}
}
#[test]
fn to_from_str_test() {
for fmt in vec![
DiskFormat::Unknown,
DiskFormat::Gpt,
DiskFormat::Mbr,
DiskFormat::Minfs,
DiskFormat::Fat,
DiskFormat::Blobfs,
DiskFormat::Fvm,
DiskFormat::Zxcrypt,
DiskFormat::BlockVerity,
DiskFormat::VbMeta,
DiskFormat::BootPart,
DiskFormat::Fxfs,
DiskFormat::F2fs,
DiskFormat::NandBroker,
] {
assert_eq!(fmt, fmt.as_str().into());
}
}
#[fuchsia::test]
async fn detect_disk_format_too_small() {
assert_eq!(
get_detected_disk_format(&Vec::new(), 1, 512).await.unwrap(),
DiskFormat::Unknown
);
}
#[fuchsia::test]
async fn detect_format_fvm() {
let mut data = vec![0; 4096];
data[..constants::FVM_MAGIC.len()].copy_from_slice(&constants::FVM_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Fvm);
}
#[fuchsia::test]
async fn detect_format_zxcrypt() {
let mut data = vec![0; 4096];
data[0..constants::ZXCRYPT_MAGIC.len()].copy_from_slice(&constants::ZXCRYPT_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Zxcrypt);
}
#[fuchsia::test]
async fn detect_format_block_verity() {
let mut data = vec![0; 4096];
data[0..constants::BLOCK_VERITY_MAGIC.len()]
.copy_from_slice(&constants::BLOCK_VERITY_MAGIC);
assert_eq!(
get_detected_disk_format(&data, 1000, 512).await.unwrap(),
DiskFormat::BlockVerity
);
}
#[fuchsia::test]
async fn detect_format_gpt() {
let mut data = vec![0; 4096];
data[512..512 + 16].copy_from_slice(&constants::GPT_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Gpt);
}
#[fuchsia::test]
async fn detect_format_minfs() {
let mut data = vec![0; 4096];
data[0..constants::MINFS_MAGIC.len()].copy_from_slice(&constants::MINFS_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Minfs);
}
#[fuchsia::test]
async fn detect_format_minfs_large_block_device() {
let mut data = vec![0; 32768];
data[0..constants::MINFS_MAGIC.len()].copy_from_slice(&constants::MINFS_MAGIC);
assert_eq!(
get_detected_disk_format(&data, 1250000, 16384).await.unwrap(),
DiskFormat::Minfs
);
}
#[fuchsia::test]
async fn detect_format_blobfs() {
let mut data = vec![0; 4096];
data[0..constants::BLOBFS_MAGIC.len()].copy_from_slice(&constants::BLOBFS_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Blobfs);
}
#[fuchsia::test]
async fn detect_format_vb_meta() {
let mut data = vec![0; 4096];
data[0..constants::VB_META_MAGIC.len()].copy_from_slice(&constants::VB_META_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::VbMeta);
}
#[fuchsia::test]
async fn detect_format_mbr() {
let mut data = vec![0; 4096];
data[510] = 0x55 as u8;
data[511] = 0xAA as u8;
data[38] = 0x30 as u8;
data[66] = 0x30 as u8;
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Mbr);
}
#[fuchsia::test]
async fn detect_format_fat_1() {
let mut data = vec![0; 4096];
data[510] = 0x55 as u8;
data[511] = 0xAA as u8;
data[38] = 0x29 as u8;
data[66] = 0x30 as u8;
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Fat);
}
#[fuchsia::test]
async fn detect_format_fat_2() {
let mut data = vec![0; 4096];
data[510] = 0x55 as u8;
data[511] = 0xAA as u8;
data[38] = 0x30 as u8;
data[66] = 0x29 as u8;
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Fat);
}
#[fuchsia::test]
async fn detect_format_f2fs() {
let mut data = vec![0; 4096];
data[1024..1024 + constants::F2FS_MAGIC.len()].copy_from_slice(&constants::F2FS_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::F2fs);
}
#[fuchsia::test]
async fn detect_format_fxfs() {
let mut data = vec![0; 4096];
data[0..constants::FXFS_MAGIC.len()].copy_from_slice(&constants::FXFS_MAGIC);
assert_eq!(get_detected_disk_format(&data, 1000, 512).await.unwrap(), DiskFormat::Fxfs);
}
#[fuchsia::test]
async fn detect_format_unknown() {
assert_eq!(
get_detected_disk_format(&vec![0; 4096], 1000, 512).await.unwrap(),
DiskFormat::Unknown
);
}
}