use alloc::sync::Arc; use hashbrown::HashMap; use spin::Lazy; use crate::{ filesystem::{StorageDevice, FsNode}, handle::{Handle, HandleResource}, }; pub static KERNEL_STATE: Lazy> = Lazy::new(|| spin::Mutex::new(KernelInternalState::new())); pub struct KernelInternalState { pub hostname: String, storage_devices: HashMap>, // FIXME: should this be per-process? file_table: HashMap, should_shutdown: bool, } impl KernelInternalState { pub fn new() -> Self { Self { should_shutdown: false, storage_devices: HashMap::new(), file_table: HashMap::new(), hostname: "".to_string(), } } pub fn set_hostname(&mut self, hostname: String) { self.hostname = hostname; } pub fn add_storage_device(&mut self, device: impl StorageDevice + Send + 'static) -> Handle { let handle = Handle::new(HandleResource::StorageDevice); self.storage_devices.insert(handle, Box::new(device)); handle } pub fn get_storage_device(&self, handle: Handle) -> Option<&dyn StorageDevice> { self.storage_devices.get(&handle).map(|d| &**d) } // TODO: implement flags here pub fn open_file_descriptor(&mut self, fs_node: Arc) -> Handle { let handle = Handle::new(HandleResource::FileDescriptor); self.file_table.insert(handle, FileTableEntry::new(fs_node)); handle } pub fn get_file_descriptor(&self, handle: Handle) -> Option<&FileTableEntry> { self.file_table.get(&handle) } pub fn close_file_descriptor(&mut self, handle: Handle) { self.file_table.remove(&handle); } pub fn shutdown(&mut self) { self.should_shutdown = true; } pub fn update_state(&mut self) { if self.should_shutdown { crate::arch::shutdown(); } } } impl Default for KernelInternalState { fn default() -> Self { Self::new() } } pub struct FileTableEntry { fs_node: Arc, // TODO: permissions, flags, owner, maybe cache stuff here? } impl FileTableEntry { fn new(fs_node: Arc) -> Self { Self { fs_node } } }