builtins/
factory_items.rsuse anyhow::{anyhow, Error};
use fidl_fuchsia_boot as fboot;
use fuchsia_zbi::ZbiType::StorageBootfsFactory;
use fuchsia_zbi::{ZbiParser, ZbiParserError, ZbiResult};
use futures::prelude::*;
use lazy_static::lazy_static;
use std::collections::HashMap;
use std::sync::Arc;
use zx::{self as zx, HandleBased};
lazy_static! {
static ref IMMUTABLE_VMO_RIGHTS: zx::Rights = zx::Rights::DUPLICATE
| zx::Rights::TRANSFER
| zx::Rights::READ
| zx::Rights::MAP
| zx::Rights::GET_PROPERTY;
}
#[derive(Debug)]
struct FactoryItem {
vmo: zx::Vmo,
length: u32,
}
#[derive(Debug)]
pub struct FactoryItems {
items: HashMap<u32, FactoryItem>,
}
impl FactoryItems {
pub fn new(parser: &mut ZbiParser) -> Result<Arc<Self>, Error> {
match parser.try_get_item(StorageBootfsFactory.into_raw(), None) {
Ok(result) => {
parser.release_item(StorageBootfsFactory)?;
FactoryItems::from_parsed_zbi(result)
}
Err(err) if err == ZbiParserError::ItemNotFound { zbi_type: StorageBootfsFactory } => {
Ok(Arc::new(FactoryItems { items: HashMap::new() }))
}
Err(err) => {
Err(anyhow!(
"Failed to retrieve StorageBootfsFactory item with unexpected error: {}",
err
))
}
}
}
fn from_parsed_zbi(items: Vec<ZbiResult>) -> Result<Arc<Self>, Error> {
let mut parsed_items = HashMap::new();
for item in items {
if parsed_items.contains_key(&item.extra) {
return Err(anyhow!("Duplicate factory item found in ZBI: {}", item.extra));
}
let vmo = zx::Vmo::create(item.bytes.len().try_into()?)?;
vmo.write(&item.bytes, 0)?;
parsed_items
.insert(item.extra, FactoryItem { vmo, length: item.bytes.len().try_into()? });
}
Ok(Arc::new(FactoryItems { items: parsed_items }))
}
pub async fn serve(
self: Arc<Self>,
mut stream: fboot::FactoryItemsRequestStream,
) -> Result<(), Error> {
while let Some(fboot::FactoryItemsRequest::Get { extra, responder }) =
stream.try_next().await?
{
match self.items.get(&extra) {
Some(item) => responder
.send(Some(item.vmo.duplicate_handle(*IMMUTABLE_VMO_RIGHTS)?), item.length)?,
None => responder.send(None, 0)?,
};
}
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
use fidl::AsHandleRef;
use fuchsia_async as fasync;
fn serve_factory_items(items: Vec<ZbiResult>) -> Result<fboot::FactoryItemsProxy, Error> {
let (proxy, stream) =
fidl::endpoints::create_proxy_and_stream::<fboot::FactoryItemsMarker>();
fasync::Task::local(
FactoryItems::from_parsed_zbi(items)?
.serve(stream)
.unwrap_or_else(|e| panic!("Error while serving factory items service: {}", e)),
)
.detach();
Ok(proxy)
}
#[fuchsia::test]
async fn no_factory_items_in_zbi() {
let mut parser = ZbiParser::new(zx::Vmo::create(0).expect("Failed to create empty VMO"));
let items = FactoryItems::new(&mut parser);
assert!(items.is_ok());
assert_eq!(items.unwrap().items.len(), 0);
}
#[fuchsia::test]
async fn duplicate_factory_item() {
let mock_results = vec![
ZbiResult { bytes: b"abc".to_vec(), extra: 12345 },
ZbiResult { bytes: b"def".to_vec(), extra: 12345 },
ZbiResult { bytes: b"ghi".to_vec(), extra: 54321 },
];
let factory_items = serve_factory_items(mock_results);
assert!(factory_items.is_err());
}
#[fuchsia::test]
async fn no_matching_factory_item() {
let mock_results = vec![
ZbiResult { bytes: b"abc".to_vec(), extra: 123 },
ZbiResult { bytes: b"def".to_vec(), extra: 456 },
ZbiResult { bytes: b"ghi".to_vec(), extra: 789 },
];
let factory_items = serve_factory_items(mock_results).unwrap();
let (vmo, length) =
factory_items.get(314159).await.expect("Failed to query factory item service");
assert!(vmo.is_none());
assert_eq!(length, 0);
}
#[fuchsia::test]
async fn get_factory_items_success() {
let mock_results = vec![
ZbiResult { bytes: b"abc".to_vec(), extra: 123 },
ZbiResult { bytes: b"def".to_vec(), extra: 456 },
ZbiResult { bytes: b"ghi".to_vec(), extra: 789 },
];
let factory_items = serve_factory_items(mock_results).unwrap();
let (vmo, length) =
factory_items.get(456).await.expect("Failed to query factory item service");
let mut bytes = [0; b"def".len()];
assert_eq!(length, bytes.len() as u32);
assert!(vmo.is_some());
let vmo = vmo.unwrap();
vmo.read(&mut bytes, 0).unwrap();
assert_eq!(&bytes, b"def");
let rights = vmo.basic_info().unwrap().rights;
assert!(rights.contains(zx::Rights::DUPLICATE));
assert!(rights.contains(zx::Rights::TRANSFER));
assert!(rights.contains(zx::Rights::READ));
assert!(rights.contains(zx::Rights::MAP));
assert!(rights.contains(zx::Rights::GET_PROPERTY));
assert!(!rights.contains(zx::Rights::WRITE));
assert!(!rights.contains(zx::Rights::SET_PROPERTY));
}
}