fat32.rs 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. mod dir;
  2. mod file;
  3. use crate::{
  4. io::{Buffer, ByteBuffer, UninitBuffer},
  5. kernel::{
  6. block::{make_device, BlockDevice, BlockDeviceRequest},
  7. constants::{S_IFDIR, S_IFREG},
  8. mem::paging::Page,
  9. vfs::{
  10. dentry::Dentry,
  11. inode::{define_struct_inode, Ino, Inode, InodeData},
  12. mount::{register_filesystem, Mount, MountCreator},
  13. vfs::Vfs,
  14. DevId,
  15. },
  16. },
  17. prelude::*,
  18. KResult,
  19. };
  20. use alloc::{
  21. collections::btree_map::BTreeMap,
  22. sync::{Arc, Weak},
  23. vec::Vec,
  24. };
  25. use bindings::EIO;
  26. use core::{ops::ControlFlow, sync::atomic::Ordering};
  27. use dir::Dirs as _;
  28. use eonix_runtime::task::Task;
  29. use eonix_sync::RwLock;
  30. use file::ClusterRead;
  31. type ClusterNo = u32;
  32. #[derive(Clone, Copy)]
  33. #[repr(C, packed)]
  34. struct Bootsector {
  35. jmp: [u8; 3],
  36. oem: [u8; 8],
  37. bytes_per_sector: u16,
  38. sectors_per_cluster: u8,
  39. reserved_sectors: u16,
  40. fat_copies: u8,
  41. root_entries: u16, // should be 0 for FAT32
  42. _total_sectors: u16, // outdated
  43. media: u8,
  44. _sectors_per_fat: u16, // outdated
  45. sectors_per_track: u16,
  46. heads: u16,
  47. hidden_sectors: u32,
  48. total_sectors: u32,
  49. sectors_per_fat: u32,
  50. flags: u16,
  51. fat_version: u16,
  52. root_cluster: ClusterNo,
  53. fsinfo_sector: u16,
  54. backup_bootsector: u16,
  55. _reserved: [u8; 12],
  56. drive_number: u8,
  57. _reserved2: u8,
  58. ext_sig: u8,
  59. serial: u32,
  60. volume_label: [u8; 11],
  61. fs_type: [u8; 8],
  62. bootcode: [u8; 420],
  63. mbr_signature: u16,
  64. }
  65. impl_any!(FatFs);
  66. /// # Lock order
  67. /// 2. FatTable
  68. /// 3. Inodes
  69. ///
  70. struct FatFs {
  71. sectors_per_cluster: u8,
  72. rootdir_cluster: ClusterNo,
  73. data_start: u64,
  74. volume_label: [u8; 11],
  75. device: Arc<BlockDevice>,
  76. fat: RwLock<Vec<ClusterNo>>,
  77. weak: Weak<FatFs>,
  78. icache: BTreeMap<Ino, FatInode>,
  79. }
  80. impl Vfs for FatFs {
  81. fn io_blksize(&self) -> usize {
  82. 4096
  83. }
  84. fn fs_devid(&self) -> DevId {
  85. self.device.devid()
  86. }
  87. fn is_read_only(&self) -> bool {
  88. true
  89. }
  90. }
  91. impl FatFs {
  92. fn read_cluster(&self, cluster: ClusterNo, buf: &Page) -> KResult<()> {
  93. let cluster = cluster - 2;
  94. let rq = BlockDeviceRequest {
  95. sector: self.data_start as u64 + cluster as u64 * self.sectors_per_cluster as u64,
  96. count: self.sectors_per_cluster as u64,
  97. buffer: core::slice::from_ref(buf),
  98. };
  99. self.device.read_raw(rq)?;
  100. Ok(())
  101. }
  102. fn get_or_alloc_inode(&self, ino: Ino, is_directory: bool, size: u32) -> Arc<dyn Inode> {
  103. self.icache
  104. .get(&ino)
  105. .cloned()
  106. .map(FatInode::unwrap)
  107. .unwrap_or_else(|| {
  108. if is_directory {
  109. DirInode::new(ino, self.weak.clone(), size)
  110. } else {
  111. FileInode::new(ino, self.weak.clone(), size)
  112. }
  113. })
  114. }
  115. }
  116. impl FatFs {
  117. pub fn create(device: DevId) -> KResult<(Arc<Self>, Arc<dyn Inode>)> {
  118. let device = BlockDevice::get(device)?;
  119. let mut fatfs_arc = Arc::new_cyclic(|weak: &Weak<FatFs>| Self {
  120. device,
  121. sectors_per_cluster: 0,
  122. rootdir_cluster: 0,
  123. data_start: 0,
  124. fat: RwLock::new(Vec::new()),
  125. weak: weak.clone(),
  126. icache: BTreeMap::new(),
  127. volume_label: [0; 11],
  128. });
  129. let fatfs = unsafe { Arc::get_mut_unchecked(&mut fatfs_arc) };
  130. let mut info: UninitBuffer<Bootsector> = UninitBuffer::new();
  131. fatfs.device.read_some(0, &mut info)?.ok_or(EIO)?;
  132. let info = info.assume_filled_ref()?;
  133. fatfs.sectors_per_cluster = info.sectors_per_cluster;
  134. fatfs.rootdir_cluster = info.root_cluster;
  135. fatfs.data_start =
  136. info.reserved_sectors as u64 + info.fat_copies as u64 * info.sectors_per_fat as u64;
  137. let fat = fatfs.fat.get_mut();
  138. fat.resize(
  139. 512 * info.sectors_per_fat as usize / core::mem::size_of::<ClusterNo>(),
  140. 0,
  141. );
  142. let mut buffer = ByteBuffer::from(fat.as_mut_slice());
  143. fatfs
  144. .device
  145. .read_some(info.reserved_sectors as usize * 512, &mut buffer)?
  146. .ok_or(EIO)?;
  147. info.volume_label
  148. .iter()
  149. .take_while(|&&c| c != ' ' as u8)
  150. .take(11)
  151. .enumerate()
  152. .for_each(|(idx, c)| fatfs.volume_label[idx] = *c);
  153. let root_dir_cluster_count = ClusterIterator::new(fat, fatfs.rootdir_cluster).count();
  154. let root_dir_size = root_dir_cluster_count as u32 * info.sectors_per_cluster as u32 * 512;
  155. let root_inode = DirInode::new(info.root_cluster as Ino, fatfs.weak.clone(), root_dir_size);
  156. Ok((fatfs_arc, root_inode))
  157. }
  158. }
  159. struct ClusterIterator<'fat> {
  160. fat: &'fat [ClusterNo],
  161. cur: ClusterNo,
  162. }
  163. impl<'fat> ClusterIterator<'fat> {
  164. fn new(fat: &'fat [ClusterNo], start: ClusterNo) -> Self {
  165. Self { fat, cur: start }
  166. }
  167. }
  168. impl<'fat> Iterator for ClusterIterator<'fat> {
  169. type Item = ClusterNo;
  170. fn next(&mut self) -> Option<Self::Item> {
  171. const EOC: ClusterNo = 0x0FFFFFF8;
  172. let next = self.cur;
  173. if next >= EOC {
  174. None
  175. } else {
  176. self.cur = self.fat[next as usize];
  177. Some(next)
  178. }
  179. }
  180. }
  181. #[allow(dead_code)]
  182. #[derive(Clone)]
  183. enum FatInode {
  184. File(Arc<FileInode>),
  185. Dir(Arc<DirInode>),
  186. }
  187. impl FatInode {
  188. fn unwrap(self) -> Arc<dyn Inode> {
  189. match self {
  190. FatInode::File(inode) => inode,
  191. FatInode::Dir(inode) => inode,
  192. }
  193. }
  194. }
  195. define_struct_inode! {
  196. struct FileInode;
  197. }
  198. impl FileInode {
  199. fn new(ino: Ino, weak: Weak<FatFs>, size: u32) -> Arc<Self> {
  200. let inode = Arc::new(Self {
  201. idata: InodeData::new(ino, weak),
  202. });
  203. // Safety: We are initializing the inode
  204. inode.nlink.store(1, Ordering::Relaxed);
  205. inode.mode.store(S_IFREG | 0o777, Ordering::Relaxed);
  206. inode.size.store(size as u64, Ordering::Relaxed);
  207. inode
  208. }
  209. }
  210. impl Inode for FileInode {
  211. fn read(&self, buffer: &mut dyn Buffer, offset: usize) -> KResult<usize> {
  212. let vfs = self.vfs.upgrade().ok_or(EIO)?;
  213. let vfs = vfs.as_any().downcast_ref::<FatFs>().unwrap();
  214. let fat = Task::block_on(vfs.fat.read());
  215. if self.size.load(Ordering::Relaxed) as usize == 0 {
  216. return Ok(0);
  217. }
  218. let iter = ClusterIterator::new(fat.as_ref(), self.ino as ClusterNo).read(vfs, offset);
  219. for data in iter {
  220. if buffer.fill(data?)?.should_stop() {
  221. break;
  222. }
  223. }
  224. Ok(buffer.wrote())
  225. }
  226. }
  227. define_struct_inode! {
  228. struct DirInode;
  229. }
  230. impl DirInode {
  231. fn new(ino: Ino, weak: Weak<FatFs>, size: u32) -> Arc<Self> {
  232. let inode = Arc::new(Self {
  233. idata: InodeData::new(ino, weak),
  234. });
  235. // Safety: We are initializing the inode
  236. inode.nlink.store(2, Ordering::Relaxed);
  237. inode.mode.store(S_IFDIR | 0o777, Ordering::Relaxed);
  238. inode.size.store(size as u64, Ordering::Relaxed);
  239. inode
  240. }
  241. }
  242. impl Inode for DirInode {
  243. fn lookup(&self, dentry: &Arc<Dentry>) -> KResult<Option<Arc<dyn Inode>>> {
  244. let vfs = self.vfs.upgrade().ok_or(EIO)?;
  245. let vfs = vfs.as_any().downcast_ref::<FatFs>().unwrap();
  246. let fat = Task::block_on(vfs.fat.read());
  247. let mut entries = ClusterIterator::new(fat.as_ref(), self.ino as ClusterNo)
  248. .read(vfs, 0)
  249. .dirs();
  250. let entry = entries.find(|entry| {
  251. entry
  252. .as_ref()
  253. .map(|entry| &entry.filename == dentry.name())
  254. .unwrap_or(true)
  255. });
  256. match entry {
  257. None => Ok(None),
  258. Some(Err(err)) => Err(err),
  259. Some(Ok(entry)) => Ok(Some(vfs.get_or_alloc_inode(
  260. entry.cluster as Ino,
  261. entry.is_directory,
  262. entry.size,
  263. ))),
  264. }
  265. }
  266. fn do_readdir(
  267. &self,
  268. offset: usize,
  269. callback: &mut dyn FnMut(&[u8], Ino) -> KResult<ControlFlow<(), ()>>,
  270. ) -> KResult<usize> {
  271. let vfs = self.vfs.upgrade().ok_or(EIO)?;
  272. let vfs = vfs.as_any().downcast_ref::<FatFs>().unwrap();
  273. let fat = Task::block_on(vfs.fat.read());
  274. let cluster_iter = ClusterIterator::new(fat.as_ref(), self.ino as ClusterNo)
  275. .read(vfs, offset)
  276. .dirs();
  277. let mut nread = 0usize;
  278. for entry in cluster_iter {
  279. let entry = entry?;
  280. vfs.get_or_alloc_inode(entry.cluster as Ino, entry.is_directory, entry.size);
  281. if callback(&entry.filename, entry.cluster as Ino)?.is_break() {
  282. break;
  283. }
  284. nread += entry.entry_offset as usize;
  285. }
  286. Ok(nread)
  287. }
  288. }
  289. struct FatMountCreator;
  290. impl MountCreator for FatMountCreator {
  291. fn create_mount(&self, _source: &str, _flags: u64, mp: &Arc<Dentry>) -> KResult<Mount> {
  292. let (fatfs, root_inode) = FatFs::create(make_device(8, 1))?;
  293. Mount::new(mp, fatfs, root_inode)
  294. }
  295. }
  296. pub fn init() {
  297. register_filesystem("fat32", Arc::new(FatMountCreator)).unwrap();
  298. }