use crate::format::SPARSE_HEADER_SIZE;
use crate::{Chunk, Reader, SparseHeader, Writer, BLK_SIZE};
use anyhow::{ensure, Result};
use std::io::{Cursor, SeekFrom};
pub enum DataSource {
Buffer(Box<[u8]>),
Reader(Box<dyn Reader>),
Skip(u64),
Fill(u32, u64),
}
pub struct SparseImageBuilder {
block_size: u32,
chunks: Vec<DataSource>,
}
impl SparseImageBuilder {
pub fn new() -> Self {
Self { block_size: BLK_SIZE as u32, chunks: vec![] }
}
pub fn set_block_size(mut self, block_size: u32) -> Self {
self.block_size = block_size;
self
}
pub fn add_chunk(mut self, source: DataSource) -> Self {
self.chunks.push(source);
self
}
pub fn build<W: Writer>(self, output: &mut W) -> Result<()> {
output.seek(SeekFrom::Start(SPARSE_HEADER_SIZE as u64))?;
let mut num_blocks = 0;
let num_chunks = self.chunks.len() as u32;
let mut current_offset = 0;
for input_chunk in self.chunks {
let (chunk, mut source) = match input_chunk {
DataSource::Buffer(buf) => {
ensure!(
buf.len() % self.block_size as usize == 0,
"Invalid buffer length {}",
buf.len()
);
(
Chunk::Raw { start: current_offset, size: buf.len() },
Some(Box::new(Cursor::new(buf)) as Box<dyn Reader>),
)
}
DataSource::Reader(mut reader) => {
let size = reader.seek(SeekFrom::End(0))?;
reader.seek(SeekFrom::Start(0))?;
ensure!(size % self.block_size as u64 == 0, "Invalid Reader length {}", size);
(Chunk::Raw { start: current_offset, size: size as usize }, Some(reader))
}
DataSource::Skip(size) => {
ensure!(size % self.block_size as u64 == 0, "Invalid Skip length {}", size);
(Chunk::DontCare { start: current_offset, size: size as usize }, None)
}
DataSource::Fill(value, count) => {
let size = count as usize * std::mem::size_of::<u32>();
ensure!(size % self.block_size as usize == 0, "Invalid Fill length {}", size);
(Chunk::Fill { start: current_offset, size, value }, None)
}
};
chunk.write(source.as_mut(), output)?;
num_blocks += chunk.output_blocks(self.block_size as usize) as u32;
current_offset += chunk.output_size() as u64;
}
output.seek(SeekFrom::Start(0))?;
let header = SparseHeader::new(self.block_size, num_blocks, num_chunks);
let header_bytes: Vec<u8> = bincode::serialize(&header)?;
std::io::copy(&mut Cursor::new(header_bytes), output)?;
output.flush()?;
Ok(())
}
}