#![cfg_attr(not(test), no_std)]
#![allow(dead_code)]
#![deny(missing_docs)]
use byteorder::{ByteOrder, LittleEndian};
use core::convert::TryFrom;
use log::debug;
#[macro_use]
mod structure;
mod blockdevice;
mod fat;
mod filesystem;
mod sdmmc;
mod sdmmc_proto;
pub use crate::blockdevice::{Block, BlockCount, BlockDevice, BlockIdx};
pub use crate::fat::FatVolume;
use crate::fat::RESERVED_ENTRIES;
pub use crate::filesystem::{
Attributes, Cluster, DirEntry, Directory, File, FilenameError, Mode, ShortFileName, TimeSource,
Timestamp, MAX_FILE_SIZE,
};
pub use crate::sdmmc::Error as SdMmcError;
pub use crate::sdmmc::SdMmcSpi;
#[derive(Debug, Clone)]
pub enum Error<E>
where
E: core::fmt::Debug,
{
DeviceError(E),
FormatError(&'static str),
NoSuchVolume,
FilenameError(FilenameError),
TooManyOpenDirs,
TooManyOpenFiles,
FileNotFound,
FileAlreadyOpen,
DirAlreadyOpen,
OpenedDirAsFile,
Unsupported,
EndOfFile,
BadCluster,
ConversionError,
NotEnoughSpace,
AllocationError,
JumpedFree,
ReadOnly,
FileAlreadyExists,
}
pub const MAX_OPEN_DIRS: usize = 4;
pub const MAX_OPEN_FILES: usize = 4;
pub struct Controller<D, T>
where
D: BlockDevice,
T: TimeSource,
<D as BlockDevice>::Error: core::fmt::Debug,
{
block_device: D,
timesource: T,
open_dirs: [(VolumeIdx, Cluster); MAX_OPEN_DIRS],
open_files: [(VolumeIdx, Cluster); MAX_OPEN_DIRS],
}
#[derive(Debug, PartialEq, Eq)]
pub struct Volume {
idx: VolumeIdx,
volume_type: VolumeType,
}
#[derive(Debug, PartialEq, Eq)]
pub enum VolumeType {
Fat(FatVolume),
}
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
pub struct VolumeIdx(pub usize);
const PARTITION_ID_FAT32_LBA: u8 = 0x0C;
const PARTITION_ID_FAT16_LBA: u8 = 0x0E;
const PARTITION_ID_FAT16: u8 = 0x06;
const PARTITION_ID_FAT32_CHS_LBA: u8 = 0x0B;
impl<D, T> Controller<D, T>
where
D: BlockDevice,
T: TimeSource,
<D as BlockDevice>::Error: core::fmt::Debug,
{
pub fn new(block_device: D, timesource: T) -> Controller<D, T> {
debug!("Creating new embedded-sdmmc::Controller");
Controller {
block_device,
timesource,
open_dirs: [(VolumeIdx(0), Cluster::INVALID); 4],
open_files: [(VolumeIdx(0), Cluster::INVALID); 4],
}
}
pub fn device(&mut self) -> &mut D {
&mut self.block_device
}
pub fn get_volume(&mut self, volume_idx: VolumeIdx) -> Result<Volume, Error<D::Error>> {
const PARTITION1_START: usize = 446;
const PARTITION2_START: usize = PARTITION1_START + PARTITION_INFO_LENGTH;
const PARTITION3_START: usize = PARTITION2_START + PARTITION_INFO_LENGTH;
const PARTITION4_START: usize = PARTITION3_START + PARTITION_INFO_LENGTH;
const FOOTER_START: usize = 510;
const FOOTER_VALUE: u16 = 0xAA55;
const PARTITION_INFO_LENGTH: usize = 16;
const PARTITION_INFO_STATUS_INDEX: usize = 0;
const PARTITION_INFO_TYPE_INDEX: usize = 4;
const PARTITION_INFO_LBA_START_INDEX: usize = 8;
const PARTITION_INFO_NUM_BLOCKS_INDEX: usize = 12;
let (part_type, lba_start, num_blocks) = {
let mut blocks = [Block::new()];
self.block_device
.read(&mut blocks, BlockIdx(0), "read_mbr")
.map_err(Error::DeviceError)?;
let block = &blocks[0];
if LittleEndian::read_u16(&block[FOOTER_START..FOOTER_START + 2]) != FOOTER_VALUE {
return Err(Error::FormatError("Invalid MBR signature"));
}
let partition = match volume_idx {
VolumeIdx(0) => {
&block[PARTITION1_START..(PARTITION1_START + PARTITION_INFO_LENGTH)]
}
VolumeIdx(1) => {
&block[PARTITION2_START..(PARTITION2_START + PARTITION_INFO_LENGTH)]
}
VolumeIdx(2) => {
&block[PARTITION3_START..(PARTITION3_START + PARTITION_INFO_LENGTH)]
}
VolumeIdx(3) => {
&block[PARTITION4_START..(PARTITION4_START + PARTITION_INFO_LENGTH)]
}
_ => {
return Err(Error::NoSuchVolume);
}
};
if (partition[PARTITION_INFO_STATUS_INDEX] & 0x7F) != 0x00 {
return Err(Error::FormatError("Invalid partition status"));
}
let lba_start = LittleEndian::read_u32(
&partition[PARTITION_INFO_LBA_START_INDEX..(PARTITION_INFO_LBA_START_INDEX + 4)],
);
let num_blocks = LittleEndian::read_u32(
&partition[PARTITION_INFO_NUM_BLOCKS_INDEX..(PARTITION_INFO_NUM_BLOCKS_INDEX + 4)],
);
(
partition[PARTITION_INFO_TYPE_INDEX],
BlockIdx(lba_start),
BlockCount(num_blocks),
)
};
match part_type {
PARTITION_ID_FAT32_CHS_LBA
| PARTITION_ID_FAT32_LBA
| PARTITION_ID_FAT16_LBA
| PARTITION_ID_FAT16 => {
let volume = fat::parse_volume(self, lba_start, num_blocks)?;
Ok(Volume {
idx: volume_idx,
volume_type: volume,
})
}
_ => Err(Error::FormatError("Partition type not supported")),
}
}
pub fn open_root_dir(&mut self, volume: &Volume) -> Result<Directory, Error<D::Error>> {
let mut open_dirs_row = None;
for (i, d) in self.open_dirs.iter().enumerate() {
if *d == (volume.idx, Cluster::ROOT_DIR) {
return Err(Error::DirAlreadyOpen);
}
if d.1 == Cluster::INVALID {
open_dirs_row = Some(i);
break;
}
}
let open_dirs_row = open_dirs_row.ok_or(Error::TooManyOpenDirs)?;
self.open_dirs[open_dirs_row] = (volume.idx, Cluster::ROOT_DIR);
Ok(Directory {
cluster: Cluster::ROOT_DIR,
entry: None,
})
}
pub fn open_dir(
&mut self,
volume: &Volume,
parent_dir: &Directory,
name: &str,
) -> Result<Directory, Error<D::Error>> {
let mut open_dirs_row = None;
for (i, d) in self.open_dirs.iter().enumerate() {
if d.1 == Cluster::INVALID {
open_dirs_row = Some(i);
}
}
let open_dirs_row = open_dirs_row.ok_or(Error::TooManyOpenDirs)?;
let dir_entry = match &volume.volume_type {
VolumeType::Fat(fat) => fat.find_directory_entry(self, parent_dir, name)?,
};
if !dir_entry.attributes.is_directory() {
return Err(Error::OpenedDirAsFile);
}
for (_i, dir_table_row) in self.open_dirs.iter().enumerate() {
if *dir_table_row == (volume.idx, dir_entry.cluster) {
return Err(Error::DirAlreadyOpen);
}
}
self.open_dirs[open_dirs_row] = (volume.idx, dir_entry.cluster);
Ok(Directory {
cluster: dir_entry.cluster,
entry: Some(dir_entry),
})
}
pub fn close_dir(&mut self, volume: &Volume, dir: Directory) {
let target = (volume.idx, dir.cluster);
for d in self.open_dirs.iter_mut() {
if *d == target {
d.1 = Cluster::INVALID;
break;
}
}
drop(dir);
}
pub fn find_directory_entry(
&mut self,
volume: &Volume,
dir: &Directory,
name: &str,
) -> Result<DirEntry, Error<D::Error>> {
match &volume.volume_type {
VolumeType::Fat(fat) => fat.find_directory_entry(self, dir, name),
}
}
pub fn iterate_dir<F>(
&mut self,
volume: &Volume,
dir: &Directory,
func: F,
) -> Result<(), Error<D::Error>>
where
F: FnMut(&DirEntry),
{
match &volume.volume_type {
VolumeType::Fat(fat) => fat.iterate_dir(self, dir, func),
}
}
pub fn open_file_in_dir(
&mut self,
volume: &mut Volume,
dir: &Directory,
name: &str,
mode: Mode,
) -> Result<File, Error<D::Error>> {
let mut open_files_row = None;
for (i, d) in self.open_files.iter().enumerate() {
if d.1 == Cluster::INVALID {
open_files_row = Some(i);
}
}
let open_files_row = open_files_row.ok_or(Error::TooManyOpenDirs)?;
let dir_entry = match &volume.volume_type {
VolumeType::Fat(fat) => fat.find_directory_entry(self, dir, name),
};
let dir_entry = match dir_entry {
Ok(entry) => Some(entry),
Err(_)
if (mode == Mode::ReadWriteCreate)
| (mode == Mode::ReadWriteCreateOrTruncate)
| (mode == Mode::ReadWriteCreateOrAppend) =>
{
None
}
_ => return Err(Error::FileNotFound),
};
if let Some(entry) = &dir_entry {
for dir_table_row in self.open_files.iter() {
if *dir_table_row == (volume.idx, entry.cluster) {
return Err(Error::DirAlreadyOpen);
}
}
if entry.attributes.is_directory() {
return Err(Error::OpenedDirAsFile);
}
if entry.attributes.is_read_only() && mode != Mode::ReadOnly {
return Err(Error::ReadOnly);
}
};
let mut mode = mode;
if mode == Mode::ReadWriteCreateOrAppend {
if dir_entry.is_some() {
mode = Mode::ReadWriteAppend;
} else {
mode = Mode::ReadWriteCreate;
}
} else if mode == Mode::ReadWriteCreateOrTruncate {
if dir_entry.is_some() {
mode = Mode::ReadWriteTruncate;
} else {
mode = Mode::ReadWriteCreate;
}
}
let file = match mode {
Mode::ReadOnly => {
let dir_entry = dir_entry.unwrap();
File {
starting_cluster: dir_entry.cluster,
current_cluster: (0, dir_entry.cluster),
current_offset: 0,
length: dir_entry.size,
mode,
entry: dir_entry,
}
}
Mode::ReadWriteAppend => {
let dir_entry = dir_entry.unwrap();
let mut file = File {
starting_cluster: dir_entry.cluster,
current_cluster: (0, dir_entry.cluster),
current_offset: 0,
length: dir_entry.size,
mode,
entry: dir_entry,
};
file.seek_from_end(0).ok();
file
}
Mode::ReadWriteTruncate => {
let dir_entry = dir_entry.unwrap();
let mut file = File {
starting_cluster: dir_entry.cluster,
current_cluster: (0, dir_entry.cluster),
current_offset: 0,
length: dir_entry.size,
mode,
entry: dir_entry,
};
match &mut volume.volume_type {
VolumeType::Fat(fat) => {
fat.truncate_cluster_chain(self, file.starting_cluster)?
}
};
file.update_length(0);
match &volume.volume_type {
VolumeType::Fat(fat) => {
let fat_type = fat.get_fat_type();
self.write_entry_to_disk(fat_type, &file.entry)?;
}
};
file
}
Mode::ReadWriteCreate => {
if dir_entry.is_some() {
return Err(Error::FileAlreadyExists);
}
let file_name =
ShortFileName::create_from_str(name).map_err(Error::FilenameError)?;
let att = Attributes::create_from_fat(0);
let entry = match &mut volume.volume_type {
VolumeType::Fat(fat) => {
fat.write_new_directory_entry(self, dir, file_name, att)?
}
};
File {
starting_cluster: entry.cluster,
current_cluster: (0, entry.cluster),
current_offset: 0,
length: entry.size,
mode,
entry,
}
}
_ => return Err(Error::Unsupported),
};
self.open_files[open_files_row] = (volume.idx, file.starting_cluster);
Ok(file)
}
pub fn read(
&mut self,
volume: &Volume,
file: &mut File,
buffer: &mut [u8],
) -> Result<usize, Error<D::Error>> {
let mut space = buffer.len();
let mut read = 0;
while space > 0 && !file.eof() {
let (block_idx, block_offset, block_avail) =
self.find_data_on_disk(volume, &mut file.current_cluster, file.current_offset)?;
let mut blocks = [Block::new()];
self.block_device
.read(&mut blocks, block_idx, "read")
.map_err(Error::DeviceError)?;
let block = &blocks[0];
let to_copy = block_avail.min(space).min(file.left() as usize);
assert!(to_copy != 0);
buffer[read..read + to_copy]
.copy_from_slice(&block[block_offset..block_offset + to_copy]);
read += to_copy;
space -= to_copy;
file.seek_from_current(to_copy as i32).unwrap();
}
Ok(read)
}
pub fn write(
&mut self,
volume: &mut Volume,
file: &mut File,
buffer: &[u8],
) -> Result<usize, Error<D::Error>> {
debug!(
"write(volume={:?}, file={:?}, buffer={:x?}",
volume, file, buffer
);
if file.mode == Mode::ReadOnly {
return Err(Error::ReadOnly);
}
if file.starting_cluster.0 < RESERVED_ENTRIES {
file.starting_cluster = match &mut volume.volume_type {
VolumeType::Fat(fat) => fat.alloc_cluster(self, None, false)?,
};
file.entry.cluster = file.starting_cluster;
debug!("Alloc first cluster {:?}", file.starting_cluster);
}
if (file.current_cluster.1).0 < file.starting_cluster.0 {
debug!("Rewinding to start");
file.current_cluster = (0, file.starting_cluster);
}
let bytes_until_max = usize::try_from(MAX_FILE_SIZE - file.current_offset)
.map_err(|_| Error::ConversionError)?;
let bytes_to_write = core::cmp::min(buffer.len(), bytes_until_max);
let mut written = 0;
while written < bytes_to_write {
let mut current_cluster = file.current_cluster;
debug!(
"Have written bytes {}/{}, finding cluster {:?}",
written, bytes_to_write, current_cluster
);
let (block_idx, block_offset, block_avail) =
match self.find_data_on_disk(volume, &mut current_cluster, file.current_offset) {
Ok(vars) => {
debug!(
"Found block_idx={:?}, block_offset={:?}, block_avail={}",
vars.0, vars.1, vars.2
);
vars
}
Err(Error::EndOfFile) => {
debug!("Extending file");
match &mut volume.volume_type {
VolumeType::Fat(ref mut fat) => {
if fat
.alloc_cluster(self, Some(current_cluster.1), false)
.is_err()
{
return Ok(written);
}
debug!("Allocated new FAT cluster, finding offsets...");
let new_offset = self
.find_data_on_disk(
volume,
&mut current_cluster,
file.current_offset,
)
.map_err(|_| Error::AllocationError)?;
debug!("New offset {:?}", new_offset);
new_offset
}
}
}
Err(e) => return Err(e),
};
let mut blocks = [Block::new()];
let to_copy = core::cmp::min(block_avail, bytes_to_write - written);
if block_offset != 0 {
debug!("Partial block write");
self.block_device
.read(&mut blocks, block_idx, "read")
.map_err(Error::DeviceError)?;
}
let block = &mut blocks[0];
block[block_offset..block_offset + to_copy]
.copy_from_slice(&buffer[written..written + to_copy]);
debug!("Writing block {:?}", block_idx);
self.block_device
.write(&blocks, block_idx)
.map_err(Error::DeviceError)?;
written += to_copy;
file.current_cluster = current_cluster;
let to_copy = i32::try_from(to_copy).map_err(|_| Error::ConversionError)?;
file.update_length(file.length + (to_copy as u32));
file.seek_from_current(to_copy).unwrap();
file.entry.attributes.set_archive(true);
file.entry.mtime = self.timesource.get_timestamp();
debug!("Updating FAT info sector");
match &mut volume.volume_type {
VolumeType::Fat(fat) => {
fat.update_info_sector(self)?;
debug!("Updating dir entry");
self.write_entry_to_disk(fat.get_fat_type(), &file.entry)?;
}
}
}
Ok(written)
}
pub fn close_file(&mut self, volume: &Volume, file: File) -> Result<(), Error<D::Error>> {
let target = (volume.idx, file.starting_cluster);
for d in self.open_files.iter_mut() {
if *d == target {
d.1 = Cluster::INVALID;
break;
}
}
drop(file);
Ok(())
}
fn find_data_on_disk(
&mut self,
volume: &Volume,
start: &mut (u32, Cluster),
desired_offset: u32,
) -> Result<(BlockIdx, usize, usize), Error<D::Error>> {
let bytes_per_cluster = match &volume.volume_type {
VolumeType::Fat(fat) => fat.bytes_per_cluster(),
};
let offset_from_cluster = desired_offset - start.0;
let num_clusters = offset_from_cluster / bytes_per_cluster;
for _ in 0..num_clusters {
start.1 = match &volume.volume_type {
VolumeType::Fat(fat) => fat.next_cluster(self, start.1)?,
};
start.0 += bytes_per_cluster;
}
let offset_from_cluster = desired_offset - start.0;
assert!(offset_from_cluster < bytes_per_cluster);
let num_blocks = BlockCount(offset_from_cluster / Block::LEN_U32);
let block_idx = match &volume.volume_type {
VolumeType::Fat(fat) => fat.cluster_to_block(start.1),
} + num_blocks;
let block_offset = (desired_offset % Block::LEN_U32) as usize;
let available = Block::LEN - block_offset;
Ok((block_idx, block_offset, available))
}
fn write_entry_to_disk(
&mut self,
fat_type: fat::FatType,
entry: &DirEntry,
) -> Result<(), Error<D::Error>> {
let mut blocks = [Block::new()];
self.block_device
.read(&mut blocks, entry.entry_block, "read")
.map_err(Error::DeviceError)?;
let block = &mut blocks[0];
let start = usize::try_from(entry.entry_offset).map_err(|_| Error::ConversionError)?;
block[start..start + 32].copy_from_slice(&entry.serialize(fat_type)[..]);
self.block_device
.write(&blocks, entry.entry_block)
.map_err(Error::DeviceError)?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
struct DummyBlockDevice;
struct Clock;
#[derive(Debug)]
enum Error {
Unknown,
}
impl TimeSource for Clock {
fn get_timestamp(&self) -> Timestamp {
Timestamp {
year_since_1970: 0,
zero_indexed_month: 0,
zero_indexed_day: 0,
hours: 0,
minutes: 0,
seconds: 0,
}
}
}
impl BlockDevice for DummyBlockDevice {
type Error = Error;
fn read(
&self,
blocks: &mut [Block],
start_block_idx: BlockIdx,
_reason: &str,
) -> Result<(), Self::Error> {
static BLOCKS: [Block; 3] = [
Block {
contents: [
0xfa, 0xb8, 0x00, 0x10, 0x8e, 0xd0, 0xbc, 0x00, 0xb0, 0xb8, 0x00, 0x00,
0x8e, 0xd8, 0x8e, 0xc0, 0xfb, 0xbe, 0x00, 0x7c, 0xbf, 0x00, 0x06, 0xb9, 0x00, 0x02, 0xf3, 0xa4,
0xea, 0x21, 0x06, 0x00, 0x00, 0xbe, 0xbe, 0x07, 0x38, 0x04, 0x75, 0x0b, 0x83, 0xc6, 0x10, 0x81,
0xfe, 0xfe, 0x07, 0x75, 0xf3, 0xeb, 0x16, 0xb4, 0x02, 0xb0, 0x01, 0xbb, 0x00, 0x7c, 0xb2, 0x80,
0x8a, 0x74, 0x01, 0x8b, 0x4c, 0x02, 0xcd, 0x13, 0xea, 0x00, 0x7c, 0x00, 0x00, 0xeb, 0xfe, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x4c, 0xca, 0xde, 0x06,
0x00, 0x00, 0x00, 0x04, 0x01, 0x04, 0x0c, 0xfe, 0xc2, 0xff, 0x01, 0x00, 0x00, 0x00, 0x33, 0x22,
0x11, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x55, 0xaa, ],
},
Block {
contents: [
0xeb, 0x58, 0x90, 0x6d, 0x6b, 0x66, 0x73, 0x2e, 0x66, 0x61, 0x74, 0x00,
0x02, 0x08, 0x20, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0xf8, 0x00, 0x00, 0x10, 0x00, 0x04, 0x00,
0x00, 0x08, 0x00, 0x00, 0x00, 0x20, 0x76, 0x00, 0x80, 0x1d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x02, 0x00, 0x00, 0x00, 0x01, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x80, 0x01, 0x29, 0x0b, 0xa8, 0x89, 0x27, 0x50, 0x69, 0x63, 0x74, 0x75,
0x72, 0x65, 0x73, 0x20, 0x20, 0x20, 0x46, 0x41, 0x54, 0x33, 0x32, 0x20, 0x20, 0x20, 0x0e, 0x1f,
0xbe, 0x77, 0x7c, 0xac, 0x22, 0xc0, 0x74, 0x0b, 0x56, 0xb4, 0x0e, 0xbb, 0x07, 0x00, 0xcd, 0x10,
0x5e, 0xeb, 0xf0, 0x32, 0xe4, 0xcd, 0x16, 0xcd, 0x19, 0xeb, 0xfe, 0x54, 0x68, 0x69, 0x73, 0x20,
0x69, 0x73, 0x20, 0x6e, 0x6f, 0x74, 0x20, 0x61, 0x20, 0x62, 0x6f, 0x6f, 0x74, 0x61, 0x62, 0x6c,
0x65, 0x20, 0x64, 0x69, 0x73, 0x6b, 0x2e, 0x20, 0x20, 0x50, 0x6c, 0x65, 0x61, 0x73, 0x65, 0x20,
0x69, 0x6e, 0x73, 0x65, 0x72, 0x74, 0x20, 0x61, 0x20, 0x62, 0x6f, 0x6f, 0x74, 0x61, 0x62, 0x6c,
0x65, 0x20, 0x66, 0x6c, 0x6f, 0x70, 0x70, 0x79, 0x20, 0x61, 0x6e, 0x64, 0x0d, 0x0a, 0x70, 0x72,
0x65, 0x73, 0x73, 0x20, 0x61, 0x6e, 0x79, 0x20, 0x6b, 0x65, 0x79, 0x20, 0x74, 0x6f, 0x20, 0x74,
0x72, 0x79, 0x20, 0x61, 0x67, 0x61, 0x69, 0x6e, 0x20, 0x2e, 0x2e, 0x2e, 0x20, 0x0d, 0x0a, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x55, 0xaa, ],
},
Block {
contents: [
0x52, 0x52, 0x61, 0x41, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x72, 0x72, 0x41, 0x61, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x55, 0xAA,
],
},
];
println!(
"Reading block {} to {}",
start_block_idx.0,
start_block_idx.0 as usize + blocks.len()
);
for (idx, block) in blocks.iter_mut().enumerate() {
let block_idx = start_block_idx.0 as usize + idx;
if block_idx < BLOCKS.len() {
*block = BLOCKS[block_idx].clone();
} else {
return Err(Error::Unknown);
}
}
Ok(())
}
fn write(&self, _blocks: &[Block], _start_block_idx: BlockIdx) -> Result<(), Self::Error> {
unimplemented!();
}
fn num_blocks(&self) -> Result<BlockCount, Self::Error> {
Ok(BlockCount(2))
}
}
#[test]
fn partition0() {
let mut c = Controller::new(DummyBlockDevice, Clock);
let v = c.get_volume(VolumeIdx(0)).unwrap();
assert_eq!(
v,
Volume {
idx: VolumeIdx(0),
volume_type: VolumeType::Fat(FatVolume {
lba_start: BlockIdx(1),
num_blocks: BlockCount(0x0011_2233),
blocks_per_cluster: 8,
first_data_block: BlockCount(15136),
fat_start: BlockCount(32),
name: fat::VolumeName {
data: *b"Pictures "
},
free_clusters_count: None,
next_free_cluster: None,
cluster_count: 965_788,
fat_specific_info: fat::FatSpecificInfo::Fat32(fat::Fat32Info {
first_root_dir_cluster: Cluster(2),
info_location: BlockIdx(1) + BlockCount(1),
})
})
}
);
}
}