1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192
#![no_std]
//! Defines traits for Files and Directories within the virtual filesystem. These files and directories mimic
//! that of a standard unix virtual filesystem, where directories follow a hierarchical system
//! and all directories have a parent directory (except for the special root directory).
//! All files must be contained within directories.
//!
//! Note that both File and Directory extend from FsNode, which is a trait that defines
//! common methods for both Files and Directories to enhance code reuse
//!
//! Some functions return an enum FileOrDir; this allows us to seamlessly call functions on the return types of
//! other filesystem functions, and then we simply match on the FSnode to extract the concrete type
//! to perform the desired function
#[macro_use] extern crate alloc;
extern crate spin;
extern crate memory;
extern crate io;
use core::fmt;
use alloc::string::String;
use alloc::vec::Vec;
use spin::Mutex;
use alloc::sync::{Arc, Weak};
use memory::MappedPages;
use io::{ByteReader, ByteWriter, KnownLength};
/// A reference to any type that implements the [`File`] trait,
/// which can only represent a File (not a Directory).
pub type FileRef = Arc<Mutex<dyn File + Send>>;
/// A weak reference to any type that implements the [`File`] trait,
/// which can only represent a File (not a Directory).
pub type WeakFileRef = Weak<Mutex<dyn File + Send>>;
/// A reference to any type that implements the [`Directory`] trait,
/// which can only represent a Directory (not a File).
pub type DirRef = Arc<Mutex<dyn Directory + Send>>;
/// A weak reference to any type that implements the [`Directory`] trait,
/// which can only represent a Directory (not a File).
pub type WeakDirRef = Weak<Mutex<dyn Directory + Send>>;
/// A trait that covers any filesystem node, both files and directories.
pub trait FsNode {
/// Recursively gets the absolute pathname as a String
fn get_absolute_path(&self) -> String {
let mut path = self.get_name();
if let Some(cur_dir) = self.get_parent_dir() {
let parent_path = &cur_dir.lock().get_absolute_path();
// Check if the parent path is root
if parent_path == "/" {
path.insert_str(0, parent_path);
return path;
}
path.insert_str(0, &format!("{parent_path}/"));
return path;
}
path
}
/// Returns the string name of the node
fn get_name(&self) -> String;
/// Returns the parent directory of the current node.
fn get_parent_dir(&self) -> Option<DirRef>;
/// Sets this node's parent directory.
/// This is useful for ensuring correctness when inserting or removing
/// files or directories from their parent directory.
fn set_parent_dir(&mut self, new_parent: WeakDirRef);
}
// Trait for files, implementors of File must also implement FsNode
pub trait File : FsNode + ByteReader + ByteWriter + KnownLength {
/// Returns a view of this file as an immutable memory-mapped region.
fn as_mapping(&self) -> Result<&MappedPages, &'static str>;
}
/// Trait for directories, implementors of Directory must also implement FsNode
pub trait Directory : FsNode {
/// Gets either the file or directory in this `Directory` on its name.
fn get(&self, name: &str) -> Option<FileOrDir>;
/// Like [`Directory::get()`], but only looks for **files** matching the given `name` in this `Directory`.
fn get_file(&self, name: &str) -> Option<FileRef> {
match self.get(name) {
Some(FileOrDir::File(f)) => Some(f),
_ => None,
}
}
/// Like [`Directory::get()`], but only looks for **directories** matching the given `name` in this `Directory`.
fn get_dir(&self, name: &str) -> Option<DirRef> {
match self.get(name) {
Some(FileOrDir::Dir(d)) => Some(d),
_ => None,
}
}
/// Inserts the given new file or directory into this directory.
/// If an existing node has the same name, that node is replaced and returned.
///
/// Note that this function **does not** set the given `node`'s parent directory;
/// that should be set when the `node` was originally created, before calling this function.
/// However, if a node is replaced, that old node's parent directory will be cleared
/// to reflect that it is no longer in this directory.
///
/// The lock on `node` must not be held because it will be acquired within this function.
fn insert(&mut self, node: FileOrDir) -> Result<Option<FileOrDir>, &'static str>;
/// Removes a file or directory from this directory and returns it if found.
/// Also, the returned node's parent directory reference is cleared.
///
/// The lock on `node` must not be held because it will be acquired within this function.
fn remove(&mut self, node: &FileOrDir) -> Option<FileOrDir>;
/// Lists the names of the nodes in this directory.
fn list(&self) -> Vec<String>;
}
/// Allows us to return a generic type that can be matched by the caller to extract the underlying type
#[derive(Clone)]
pub enum FileOrDir {
File(FileRef),
Dir(DirRef),
}
impl fmt::Debug for FileOrDir {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.get_absolute_path())
}
}
// Allows us to call methods directly on an enum so we don't have to match on the underlying type
impl FsNode for FileOrDir {
fn get_absolute_path(&self) -> String {
match self {
FileOrDir::File(file) => file.lock().get_absolute_path(),
FileOrDir::Dir(dir) => dir.lock().get_absolute_path(),
}
}
fn get_name(&self) -> String {
match self {
FileOrDir::File(file) => file.lock().get_name(),
FileOrDir::Dir(dir) => dir.lock().get_name(),
}
}
fn get_parent_dir(&self) -> Option<DirRef> {
match self {
FileOrDir::File(file) => file.lock().get_parent_dir(),
FileOrDir::Dir(dir) => dir.lock().get_parent_dir(),
}
}
fn set_parent_dir(&mut self, new_parent: WeakDirRef) {
match self {
FileOrDir::File(file) => file.lock().set_parent_dir(new_parent),
FileOrDir::Dir(dir) => dir.lock().set_parent_dir(new_parent),
}
}
}
impl KnownLength for FileOrDir {
/// Returns the length (size) in bytes of this `FileOrDir`.
///
/// Directories currently return `0`.
fn len(&self) -> usize {
match &self {
FileOrDir::File(f) => f.lock().len(),
FileOrDir::Dir(_) => 0,
}
}
}
impl FileOrDir {
/// Returns `true` if this is a `File`, `false` if it is a `Directory`.
pub fn is_file(&self) -> bool {
match &self {
FileOrDir::File(_) => true,
FileOrDir::Dir(_) => false,
}
}
/// Returns `true` if this is a `Directory`, `false` if it is a `File`.
pub fn is_dir(&self) -> bool {
match &self {
FileOrDir::File(_) => false,
FileOrDir::Dir(_) => true,
}
}
}