From 3e08c80fb0d6053c422a9dd3455e07681cc5b6ee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Rafa=C5=82=20Harabie=C5=84?= Date: Sat, 7 Oct 2017 02:19:19 +0200 Subject: [PATCH] Dont assume cluster iterator always succeeds. --- src/file.rs | 12 ++++++++++-- src/fs.rs | 2 +- src/table.rs | 34 +++++++++++++++++++++++++++------- 3 files changed, 38 insertions(+), 10 deletions(-) diff --git a/src/file.rs b/src/file.rs index afda65f..fc2dcce 100644 --- a/src/file.rs +++ b/src/file.rs @@ -50,7 +50,11 @@ impl <'a, 'b> Read for FatFile<'a, 'b> { self.offset += read_bytes as u32; buf_offset += read_bytes; if self.offset % cluster_size == 0 { - self.current_cluster = self.fs.cluster_iter(current_cluster).skip(1).next(); + match self.fs.cluster_iter(current_cluster).skip(1).next() { + Some(Err(err)) => return Err(err), + Some(Ok(n)) => self.current_cluster = Some(n), + None => self.current_cluster = None, + } } } Ok(buf_offset) @@ -71,7 +75,11 @@ impl <'a, 'b> Seek for FatFile<'a, 'b> { let cluster_count = (new_offset / cluster_size as i64) as usize; let mut new_cluster = Some(self.first_cluster); if cluster_count > 0 { - new_cluster = self.fs.cluster_iter(new_cluster.unwrap()).skip(cluster_count).next(); + match self.fs.cluster_iter(new_cluster.unwrap()).skip(cluster_count).next() { + Some(Err(err)) => return Err(err), + Some(Ok(n)) => new_cluster = Some(n), + None => new_cluster = None, + } } self.offset = new_offset as u32; self.current_cluster = new_cluster; diff --git a/src/fs.rs b/src/fs.rs index 1ae6922..6a11c03 100644 --- a/src/fs.rs +++ b/src/fs.rs @@ -214,7 +214,7 @@ impl <'a> FatFileSystem<'a> { self.offset_from_sector(self.sector_from_cluster(cluser)) } - pub(crate) fn cluster_iter<'b>(&'b self, cluster: u32) -> iter::Chain, FatClusterIterator<'b, 'a>> { + pub(crate) fn cluster_iter<'b>(&'b self, cluster: u32) -> iter::Chain>, FatClusterIterator<'b, 'a>> { let bytes_per_sector = self.boot.bpb.bytes_per_sector as u64; let fat_offset = self.boot.bpb.reserved_sectors as u64 * bytes_per_sector; let sectors_per_fat = diff --git a/src/table.rs b/src/table.rs index 3037734..2df22ea 100644 --- a/src/table.rs +++ b/src/table.rs @@ -51,27 +51,47 @@ fn get_next_cluster_32(rdr: &mut ReadSeek, cluster: u32) -> io::Result { - part: FatSlice<'a, 'b>, + rdr: FatSlice<'a, 'b>, fat_type: FatType, cluster: Option, + err: bool, } impl <'a, 'b> FatClusterIterator<'a, 'b> { - pub(crate) fn new(part: FatSlice<'a, 'b>, fat_type: FatType, cluster: u32) -> iter::Chain, FatClusterIterator<'a, 'b>> { + pub(crate) fn new(rdr: FatSlice<'a, 'b>, fat_type: FatType, cluster: u32) + -> iter::Chain>, FatClusterIterator<'a, 'b>> { let iter = FatClusterIterator { - part: part, + rdr: rdr, fat_type: fat_type, cluster: Some(cluster), + err: false, }; - iter::once(cluster).chain(iter) + iter::once(Ok(cluster)).chain(iter) } } impl <'a, 'b> Iterator for FatClusterIterator<'a, 'b> { - type Item = u32; + type Item = io::Result; fn next(&mut self) -> Option { - self.cluster = get_next_cluster(&mut self.part, self.fat_type, self.cluster.unwrap()).unwrap(); // FIXME: unwrap! - self.cluster + if self.err { + return None; + } + match self.cluster { + Some(current_cluster) => { + self.cluster = match get_next_cluster(&mut self.rdr, self.fat_type, current_cluster) { + Ok(next_cluster) => next_cluster, + Err(err) => { + self.err = true; + return Some(Err(err)); + }, + } + }, + None => {}, + }; + match self.cluster { + Some(n) => Some(Ok(n)), + None => None, + } } }