Load init from initfs
This commit is contained in:
parent
00db6ddd62
commit
f2ca411cd6
|
@ -133,11 +133,23 @@ pub extern fn kmain() {
|
||||||
let pid = syscall::getpid();
|
let pid = syscall::getpid();
|
||||||
println!("BSP: {:?}", pid);
|
println!("BSP: {:?}", pid);
|
||||||
|
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(0));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(0));
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(1));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(1));
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(2));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(2));
|
||||||
|
|
||||||
let elf = elf::Elf::from(include_bytes!("../build/userspace/init")).expect("could not load elf");
|
let init_file = syscall::open(b"initfs:init", 0).expect("failed to open initfs:init");
|
||||||
|
let mut init_data = collections::Vec::new();
|
||||||
|
loop {
|
||||||
|
let mut buf = [0; 65536];
|
||||||
|
let count = syscall::read(init_file, &mut buf).expect("failed to read initfs:init");
|
||||||
|
if count > 0 {
|
||||||
|
init_data.extend_from_slice(&buf[..count]);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let elf = elf::Elf::from(&init_data).expect("could not load elf");
|
||||||
elf.run();
|
elf.run();
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -163,12 +175,9 @@ pub extern fn kmain_ap(id: usize) {
|
||||||
let pid = syscall::getpid();
|
let pid = syscall::getpid();
|
||||||
println!("AP {}: {:?}", id, pid);
|
println!("AP {}: {:?}", id, pid);
|
||||||
|
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(0));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(0));
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(1));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(1));
|
||||||
assert_eq!(syscall::open("debug:".as_bytes(), 0), Ok(2));
|
assert_eq!(syscall::open(b"debug:", 0), Ok(2));
|
||||||
|
|
||||||
let elf = elf::Elf::from(include_bytes!("../build/userspace/init")).expect("could not load elf");
|
|
||||||
elf.run();
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
unsafe { interrupt::enable_and_halt() }
|
unsafe { interrupt::enable_and_halt() }
|
||||||
|
|
70
kernel/scheme/initfs.rs
Normal file
70
kernel/scheme/initfs.rs
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
use collections::BTreeMap;
|
||||||
|
|
||||||
|
use syscall::{Error, Result};
|
||||||
|
use super::Scheme;
|
||||||
|
|
||||||
|
struct Handle {
|
||||||
|
data: &'static [u8],
|
||||||
|
seek: usize
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct InitFsScheme {
|
||||||
|
next_id: usize,
|
||||||
|
files: BTreeMap<&'static [u8], &'static [u8]>,
|
||||||
|
handles: BTreeMap<usize, Handle>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InitFsScheme {
|
||||||
|
pub fn new() -> InitFsScheme {
|
||||||
|
let mut files: BTreeMap<&'static [u8], &'static [u8]> = BTreeMap::new();
|
||||||
|
|
||||||
|
files.insert(b"init", include_bytes!("../../build/userspace/init"));
|
||||||
|
|
||||||
|
InitFsScheme {
|
||||||
|
next_id: 0,
|
||||||
|
files: files,
|
||||||
|
handles: BTreeMap::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Scheme for InitFsScheme {
|
||||||
|
fn open(&mut self, path: &[u8], _flags: usize) -> Result<usize> {
|
||||||
|
let data = self.files.get(path).ok_or(Error::NoEntry)?;
|
||||||
|
let id = self.next_id;
|
||||||
|
self.next_id += 1;
|
||||||
|
self.handles.insert(id, Handle {
|
||||||
|
data: data,
|
||||||
|
seek: 0
|
||||||
|
});
|
||||||
|
Ok(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Read the file `number` into the `buffer`
|
||||||
|
///
|
||||||
|
/// Returns the number of bytes read
|
||||||
|
fn read(&mut self, file: usize, buffer: &mut [u8]) -> Result<usize> {
|
||||||
|
let mut handle = self.handles.get_mut(&file).ok_or(Error::BadFile)?;
|
||||||
|
|
||||||
|
let mut i = 0;
|
||||||
|
while i < buffer.len() && handle.seek < handle.data.len() {
|
||||||
|
buffer[i] = handle.data[handle.seek];
|
||||||
|
i += 1;
|
||||||
|
handle.seek += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(i)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Write the `buffer` to the `file`
|
||||||
|
///
|
||||||
|
/// Returns the number of bytes written
|
||||||
|
fn write(&mut self, _file: usize, _buffer: &[u8]) -> Result<usize> {
|
||||||
|
Err(Error::NotPermitted)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Close the file `number`
|
||||||
|
fn close(&mut self, file: usize) -> Result<()> {
|
||||||
|
self.handles.remove(&file).ok_or(Error::BadFile).and(Ok(()))
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,10 +16,14 @@ use spin::{Once, Mutex, RwLock, RwLockReadGuard, RwLockWriteGuard};
|
||||||
use syscall::{Error, Result};
|
use syscall::{Error, Result};
|
||||||
|
|
||||||
use self::debug::DebugScheme;
|
use self::debug::DebugScheme;
|
||||||
|
use self::initfs::InitFsScheme;
|
||||||
|
|
||||||
/// Debug scheme
|
/// Debug scheme
|
||||||
pub mod debug;
|
pub mod debug;
|
||||||
|
|
||||||
|
/// InitFS scheme
|
||||||
|
pub mod initfs;
|
||||||
|
|
||||||
/// Limit on number of schemes
|
/// Limit on number of schemes
|
||||||
pub const SCHEME_MAX_SCHEMES: usize = 65536;
|
pub const SCHEME_MAX_SCHEMES: usize = 65536;
|
||||||
|
|
||||||
|
@ -88,6 +92,7 @@ static SCHEMES: Once<RwLock<SchemeList>> = Once::new();
|
||||||
fn init_schemes() -> RwLock<SchemeList> {
|
fn init_schemes() -> RwLock<SchemeList> {
|
||||||
let mut list: SchemeList = SchemeList::new();
|
let mut list: SchemeList = SchemeList::new();
|
||||||
list.insert(Box::new(*b"debug"), Arc::new(Mutex::new(Box::new(DebugScheme)))).expect("failed to insert debug: scheme");
|
list.insert(Box::new(*b"debug"), Arc::new(Mutex::new(Box::new(DebugScheme)))).expect("failed to insert debug: scheme");
|
||||||
|
list.insert(Box::new(*b"initfs"), Arc::new(Mutex::new(Box::new(InitFsScheme::new())))).expect("failed to insert initfs: scheme");
|
||||||
RwLock::new(list)
|
RwLock::new(list)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue