commit
00ef69d461
10
Makefile
10
Makefile
|
@ -13,8 +13,8 @@ ifeq ($(ARCH),arm)
|
||||||
else
|
else
|
||||||
LD=ld
|
LD=ld
|
||||||
QEMUFLAGS+=-enable-kvm -cpu host -machine q35 -smp 4
|
QEMUFLAGS+=-enable-kvm -cpu host -machine q35 -smp 4
|
||||||
|
QEMUFLAGS+=-nographic -vga none
|
||||||
#,int,pcall
|
#,int,pcall
|
||||||
#-nographic
|
|
||||||
#-device intel-iommu
|
#-device intel-iommu
|
||||||
|
|
||||||
UNAME := $(shell uname)
|
UNAME := $(shell uname)
|
||||||
|
@ -51,7 +51,13 @@ build/libcollections.rlib: rust/src/libcollections/lib.rs build/libcore.rlib bui
|
||||||
mkdir -p build
|
mkdir -p build
|
||||||
./rustc.sh $(RUSTCFLAGS) -o $@ $<
|
./rustc.sh $(RUSTCFLAGS) -o $@ $<
|
||||||
|
|
||||||
build/libkernel.a: build/libcore.rlib build/liballoc.rlib build/libcollections.rlib FORCE
|
build/libinit.a: init/Cargo.toml init/src/*.rs
|
||||||
|
RUSTC="./rustc.sh" cargo rustc --manifest-path $< $(CARGOFLAGS) -o $@
|
||||||
|
|
||||||
|
build/init: build/libinit.a
|
||||||
|
$(LD) -e _start --gc-sections -o $@ $<
|
||||||
|
|
||||||
|
build/libkernel.a: build/libcore.rlib build/liballoc.rlib build/libcollections.rlib build/init FORCE
|
||||||
mkdir -p build
|
mkdir -p build
|
||||||
RUSTC="./rustc.sh" cargo rustc $(CARGOFLAGS) -o $@
|
RUSTC="./rustc.sh" cargo rustc $(CARGOFLAGS) -o $@
|
||||||
|
|
||||||
|
|
|
@ -60,6 +60,7 @@ pub unsafe fn init() {
|
||||||
|
|
||||||
// Set syscall function
|
// Set syscall function
|
||||||
IDT[0x80].set_func(syscall::syscall);
|
IDT[0x80].set_func(syscall::syscall);
|
||||||
|
IDT[0x80].set_flags(IDT_PRESENT | IDT_RING_3 | IDT_INTERRUPT);
|
||||||
|
|
||||||
dtables::lidt(&IDTR);
|
dtables::lidt(&IDTR);
|
||||||
}
|
}
|
||||||
|
|
|
@ -71,7 +71,9 @@ interrupt_error!(protection, {
|
||||||
});
|
});
|
||||||
|
|
||||||
interrupt_error!(page, {
|
interrupt_error!(page, {
|
||||||
print!("Page fault\n");
|
let cr2: usize;
|
||||||
|
asm!("mov rax, cr2" : "={rax}"(cr2) : : : "intel", "volatile");
|
||||||
|
println!("Page fault: {:>016X}", cr2);
|
||||||
stack_trace();
|
stack_trace();
|
||||||
loop { halt(); }
|
loop { halt(); }
|
||||||
});
|
});
|
||||||
|
|
|
@ -52,13 +52,21 @@ pub unsafe fn stack_trace() {
|
||||||
//Maximum 64 frames
|
//Maximum 64 frames
|
||||||
let active_table = ActivePageTable::new();
|
let active_table = ActivePageTable::new();
|
||||||
for _frame in 0..64 {
|
for _frame in 0..64 {
|
||||||
if active_table.translate(VirtualAddress::new(rbp)).is_some() && active_table.translate(VirtualAddress::new(rbp + mem::size_of::<usize>())).is_some() {
|
if let Some(rip_rbp) = rbp.checked_add(mem::size_of::<usize>()) {
|
||||||
let rip = *(rbp as *const usize).offset(1);
|
if active_table.translate(VirtualAddress::new(rbp)).is_some() && active_table.translate(VirtualAddress::new(rip_rbp)).is_some() {
|
||||||
println!(" {:>016X}: {:>016X}", rbp, rip);
|
let rip = *(rip_rbp as *const usize);
|
||||||
rbp = *(rbp as *const usize);
|
if rip == 0 {
|
||||||
|
println!(" {:>016X}: EMPTY RETURN", rbp);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
println!(" {:>016X}: {:>016X}", rbp, rip);
|
||||||
|
rbp = *(rbp as *const usize);
|
||||||
|
} else {
|
||||||
|
println!(" {:>016X}: GUARD PAGE", rbp);
|
||||||
|
break;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
println!(" {:>016X}: GUARD PAGE", rbp);
|
println!(" {:>016X}: RBP OVERFLOW", rbp);
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,22 +1,38 @@
|
||||||
#[naked]
|
#[naked]
|
||||||
pub unsafe extern fn syscall() {
|
pub unsafe extern fn syscall() {
|
||||||
extern {
|
#[inline(never)]
|
||||||
fn syscall(a: usize, b: usize, c: usize, d: usize, e: usize, f: usize) -> usize;
|
unsafe fn inner() {
|
||||||
|
extern {
|
||||||
|
fn syscall(a: usize, b: usize, c: usize, d: usize, e: usize, f: usize) -> usize;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut a;
|
||||||
|
{
|
||||||
|
let b;
|
||||||
|
let c;
|
||||||
|
let d;
|
||||||
|
let e;
|
||||||
|
let f;
|
||||||
|
asm!("" : "={rax}"(a), "={rbx}"(b), "={rcx}"(c), "={rdx}"(d), "={rsi}"(e), "={rdi}"(f)
|
||||||
|
: : : "intel", "volatile");
|
||||||
|
|
||||||
|
a = syscall(a, b, c, d, e, f);
|
||||||
|
}
|
||||||
|
|
||||||
|
asm!("" : : "{rax}"(a) : : "intel", "volatile");
|
||||||
}
|
}
|
||||||
|
|
||||||
let a;
|
asm!("push fs
|
||||||
let b;
|
push rax
|
||||||
let c;
|
mov rax, 0x18
|
||||||
let d;
|
mov fs, ax
|
||||||
let e;
|
pop rax"
|
||||||
let f;
|
: : : : "intel", "volatile");
|
||||||
asm!("" : "={rax}"(a), "={rbx}"(b), "={rcx}"(c), "={rdx}"(d), "={rsi}"(e), "={rdi}"(f)
|
|
||||||
: : : "intel", "volatile");
|
|
||||||
|
|
||||||
let a = syscall(a, b, c, d, e, f);
|
inner();
|
||||||
|
|
||||||
asm!("" : : "{rax}"(a) : : "intel", "volatile");
|
// Interrupt return
|
||||||
|
asm!("pop fs
|
||||||
// Pop scratch registers, error code, and return
|
iretq"
|
||||||
asm!("iretq" : : : : "intel", "volatile");
|
: : : : "intel", "volatile");
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,7 @@ extern crate hole_list_allocator as allocator;
|
||||||
extern crate bitflags;
|
extern crate bitflags;
|
||||||
extern crate ransid;
|
extern crate ransid;
|
||||||
extern crate spin;
|
extern crate spin;
|
||||||
extern crate x86;
|
pub extern crate x86;
|
||||||
|
|
||||||
/// Print to console
|
/// Print to console
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
|
@ -56,14 +56,18 @@ macro_rules! interrupt {
|
||||||
push r8
|
push r8
|
||||||
push r9
|
push r9
|
||||||
push r10
|
push r10
|
||||||
push r11"
|
push r11
|
||||||
|
push fs
|
||||||
|
mov rax, 0x18
|
||||||
|
mov fs, ax"
|
||||||
: : : : "intel", "volatile");
|
: : : : "intel", "volatile");
|
||||||
|
|
||||||
// Call inner rust function
|
// Call inner rust function
|
||||||
inner();
|
inner();
|
||||||
|
|
||||||
// Pop scratch registers and return
|
// Pop scratch registers and return
|
||||||
asm!("pop r11
|
asm!("pop fs
|
||||||
|
pop r11
|
||||||
pop r10
|
pop r10
|
||||||
pop r9
|
pop r9
|
||||||
pop r8
|
pop r8
|
||||||
|
@ -97,14 +101,18 @@ macro_rules! interrupt_error {
|
||||||
push r8
|
push r8
|
||||||
push r9
|
push r9
|
||||||
push r10
|
push r10
|
||||||
push r11"
|
push r11
|
||||||
|
push fs
|
||||||
|
mov rax, 0x18
|
||||||
|
mov fs, ax"
|
||||||
: : : : "intel", "volatile");
|
: : : : "intel", "volatile");
|
||||||
|
|
||||||
// Call inner rust function
|
// Call inner rust function
|
||||||
inner();
|
inner();
|
||||||
|
|
||||||
// Pop scratch registers, error code, and return
|
// Pop scratch registers, error code, and return
|
||||||
asm!("pop r11
|
asm!("pop fs
|
||||||
|
pop r11
|
||||||
pop r10
|
pop r10
|
||||||
pop r9
|
pop r9
|
||||||
pop r8
|
pop r8
|
||||||
|
|
|
@ -46,6 +46,15 @@ impl Mapper {
|
||||||
self.map_to(page, frame, flags)
|
self.map_to(page, frame, flags)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Update flags for a page
|
||||||
|
pub fn remap(&mut self, page: Page, flags: EntryFlags) {
|
||||||
|
let mut p3 = self.p4_mut().next_table_mut(page.p4_index()).expect("failed to remap: no p3");
|
||||||
|
let mut p2 = p3.next_table_mut(page.p3_index()).expect("failed to remap: no p2");
|
||||||
|
let mut p1 = p2.next_table_mut(page.p2_index()).expect("failed to remap: no p1");
|
||||||
|
let frame = p1[page.p1_index()].pointed_frame().expect("failed to remap: not mapped");
|
||||||
|
p1[page.p1_index()].set(frame, flags | entry::PRESENT);
|
||||||
|
}
|
||||||
|
|
||||||
/// Identity map a frame
|
/// Identity map a frame
|
||||||
pub fn identity_map(&mut self, frame: Frame, flags: EntryFlags) {
|
pub fn identity_map(&mut self, frame: Frame, flags: EntryFlags) {
|
||||||
let page = Page::containing_address(VirtualAddress::new(frame.start_address().get()));
|
let page = Page::containing_address(VirtualAddress::new(frame.start_address().get()));
|
||||||
|
|
|
@ -303,8 +303,8 @@ impl Page {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn containing_address(address: VirtualAddress) -> Page {
|
pub fn containing_address(address: VirtualAddress) -> Page {
|
||||||
assert!(address.get() < 0x0000_8000_0000_0000 || address.get() >= 0xffff_8000_0000_0000,
|
//TODO assert!(address.get() < 0x0000_8000_0000_0000 || address.get() >= 0xffff_8000_0000_0000,
|
||||||
"invalid address: 0x{:x}", address.get());
|
// "invalid address: 0x{:x}", address.get());
|
||||||
Page { number: address.get() / PAGE_SIZE }
|
Page { number: address.get() / PAGE_SIZE }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -66,7 +66,7 @@ impl<L> Table<L> where L: HierarchicalLevel {
|
||||||
assert!(!self[index].flags().contains(HUGE_PAGE),
|
assert!(!self[index].flags().contains(HUGE_PAGE),
|
||||||
"mapping code does not support huge pages");
|
"mapping code does not support huge pages");
|
||||||
let frame = allocate_frame().expect("no frames available");
|
let frame = allocate_frame().expect("no frames available");
|
||||||
self[index].set(frame, PRESENT | WRITABLE);
|
self[index].set(frame, PRESENT | WRITABLE | USER_ACCESSIBLE /* Allow users to go down the page table, implement permissions at the page level */);
|
||||||
self.next_table_mut(index).unwrap().zero();
|
self.next_table_mut(index).unwrap().zero();
|
||||||
}
|
}
|
||||||
self.next_table_mut(index).unwrap()
|
self.next_table_mut(index).unwrap()
|
||||||
|
|
|
@ -137,28 +137,6 @@ pub unsafe extern fn kstart() -> ! {
|
||||||
kmain();
|
kmain();
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe fn usermode(ip: usize, sp: usize) {
|
|
||||||
// Test usermode
|
|
||||||
asm!("xchg bx, bx
|
|
||||||
mov rax, 0x2B
|
|
||||||
mov ds, ax
|
|
||||||
mov es, ax
|
|
||||||
mov fs, ax
|
|
||||||
mov gs, ax
|
|
||||||
|
|
||||||
push rax
|
|
||||||
push rbx
|
|
||||||
pushfq
|
|
||||||
mov rax, 0x23
|
|
||||||
push rax
|
|
||||||
push rcx
|
|
||||||
iretq"
|
|
||||||
:
|
|
||||||
: "{rbx}"(sp), "{rcx}"(ip)
|
|
||||||
: "rax", "rbx", "rcx", "sp"
|
|
||||||
: "intel", "volatile");
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Entry to rust for an AP
|
/// Entry to rust for an AP
|
||||||
pub unsafe extern fn kstart_ap(stack_start: usize, stack_end: usize) -> ! {
|
pub unsafe extern fn kstart_ap(stack_start: usize, stack_end: usize) -> ! {
|
||||||
{
|
{
|
||||||
|
@ -223,3 +201,24 @@ pub unsafe extern fn kstart_ap(stack_start: usize, stack_end: usize) -> ! {
|
||||||
|
|
||||||
kmain_ap(ap_number);
|
kmain_ap(ap_number);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub unsafe fn usermode(ip: usize, sp: usize) {
|
||||||
|
// Test usermode
|
||||||
|
asm!("mov rax, 0x2B
|
||||||
|
mov ds, ax
|
||||||
|
mov es, ax
|
||||||
|
mov fs, ax
|
||||||
|
mov gs, ax
|
||||||
|
|
||||||
|
push rax
|
||||||
|
push rbx
|
||||||
|
pushfq
|
||||||
|
mov rax, 0x23
|
||||||
|
push rax
|
||||||
|
push rcx
|
||||||
|
iretq"
|
||||||
|
:
|
||||||
|
: "{rbx}"(sp), "{rcx}"(ip)
|
||||||
|
: "rax", "rbx", "rcx", "sp"
|
||||||
|
: "intel", "volatile");
|
||||||
|
}
|
||||||
|
|
10
init/Cargo.toml
Normal file
10
init/Cargo.toml
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
[package]
|
||||||
|
name = "init"
|
||||||
|
version = "0.1.0"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "init"
|
||||||
|
crate-type = ["staticlib"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
syscall = { path = "../syscall/" }
|
97
init/src/lib.rs
Normal file
97
init/src/lib.rs
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
#![feature(asm)]
|
||||||
|
#![feature(lang_items)]
|
||||||
|
#![no_std]
|
||||||
|
|
||||||
|
#[macro_use]
|
||||||
|
extern crate syscall;
|
||||||
|
|
||||||
|
use syscall::{exit, write};
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern fn _start() {
|
||||||
|
let string = b"Hello, World!\n";
|
||||||
|
write(1, string);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(test))]
|
||||||
|
#[lang = "eh_personality"]
|
||||||
|
extern "C" fn eh_personality() {}
|
||||||
|
|
||||||
|
#[cfg(not(test))]
|
||||||
|
/// Required to handle panics
|
||||||
|
#[lang = "panic_fmt"]
|
||||||
|
extern "C" fn panic_fmt(_fmt: ::core::fmt::Arguments, _file: &str, _line: u32) -> ! {
|
||||||
|
loop {}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Memcpy
|
||||||
|
///
|
||||||
|
/// Copy N bytes of memory from one location to another.
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern fn memcpy(dest: *mut u8, src: *const u8,
|
||||||
|
n: usize) -> *mut u8 {
|
||||||
|
let mut i = 0;
|
||||||
|
while i < n {
|
||||||
|
*dest.offset(i as isize) = *src.offset(i as isize);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
dest
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Memmove
|
||||||
|
///
|
||||||
|
/// Copy N bytes of memory from src to dest. The memory areas may overlap.
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern fn memmove(dest: *mut u8, src: *const u8,
|
||||||
|
n: usize) -> *mut u8 {
|
||||||
|
if src < dest as *const u8 {
|
||||||
|
let mut i = n;
|
||||||
|
while i != 0 {
|
||||||
|
i -= 1;
|
||||||
|
*dest.offset(i as isize) = *src.offset(i as isize);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let mut i = 0;
|
||||||
|
while i < n {
|
||||||
|
*dest.offset(i as isize) = *src.offset(i as isize);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dest
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Memset
|
||||||
|
///
|
||||||
|
/// Fill a block of memory with a specified value.
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern fn memset(s: *mut u8, c: i32, n: usize) -> *mut u8 {
|
||||||
|
let mut i = 0;
|
||||||
|
while i < n {
|
||||||
|
*s.offset(i as isize) = c as u8;
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
s
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Memcmp
|
||||||
|
///
|
||||||
|
/// Compare two blocks of memory.
|
||||||
|
#[no_mangle]
|
||||||
|
pub unsafe extern fn memcmp(s1: *const u8, s2: *const u8, n: usize) -> i32 {
|
||||||
|
let mut i = 0;
|
||||||
|
|
||||||
|
while i < n {
|
||||||
|
let a = *s1.offset(i as isize);
|
||||||
|
let b = *s2.offset(i as isize);
|
||||||
|
if a != b {
|
||||||
|
return a as i32 - b as i32
|
||||||
|
}
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
0
|
||||||
|
}
|
140
kernel/elf.rs
140
kernel/elf.rs
|
@ -1,8 +1,8 @@
|
||||||
//! ELF executables
|
//! ELF executables
|
||||||
|
|
||||||
use collections::{String, Vec};
|
use collections::String;
|
||||||
|
|
||||||
use core::{ptr, str};
|
use core::str;
|
||||||
|
|
||||||
#[cfg(target_arch = "x86")]
|
#[cfg(target_arch = "x86")]
|
||||||
use goblin::elf32::{header, program_header};
|
use goblin::elf32::{header, program_header};
|
||||||
|
@ -10,9 +10,15 @@ use goblin::elf32::{header, program_header};
|
||||||
#[cfg(target_arch = "x86_64")]
|
#[cfg(target_arch = "x86_64")]
|
||||||
use goblin::elf64::{header, program_header};
|
use goblin::elf64::{header, program_header};
|
||||||
|
|
||||||
|
use arch::externs::{memcpy, memset};
|
||||||
|
use arch::paging::{entry, ActivePageTable, Page, VirtualAddress};
|
||||||
|
use arch::start::usermode;
|
||||||
|
use arch::x86::tlb;
|
||||||
|
|
||||||
/// An ELF executable
|
/// An ELF executable
|
||||||
pub struct Elf<'a> {
|
pub struct Elf<'a> {
|
||||||
pub data: &'a [u8],
|
pub data: &'a [u8],
|
||||||
|
header: &'a header::Header
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Elf<'a> {
|
impl<'a> Elf<'a> {
|
||||||
|
@ -25,29 +31,125 @@ impl<'a> Elf<'a> {
|
||||||
} else if data.get(header::EI_CLASS) != Some(&header::ELFCLASS) {
|
} else if data.get(header::EI_CLASS) != Some(&header::ELFCLASS) {
|
||||||
Err(format!("Elf: Invalid architecture: {:?} != {:?}", data.get(header::EI_CLASS), header::ELFCLASS))
|
Err(format!("Elf: Invalid architecture: {:?} != {:?}", data.get(header::EI_CLASS), header::ELFCLASS))
|
||||||
} else {
|
} else {
|
||||||
Ok(Elf { data: data })
|
Ok(Elf {
|
||||||
|
data: data,
|
||||||
|
header: unsafe { &*(data.as_ptr() as usize as *const header::Header) }
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub unsafe fn load_segments(&self) -> Vec<program_header::ProgramHeader> {
|
pub fn segments(&'a self) -> ElfSegments<'a> {
|
||||||
let mut segments = Vec::new();
|
ElfSegments {
|
||||||
|
data: self.data,
|
||||||
let header = &*(self.data.as_ptr() as usize as *const header::Header);
|
header: self.header,
|
||||||
|
i: 0
|
||||||
for i in 0..header.e_phnum {
|
|
||||||
let segment = ptr::read((self.data.as_ptr() as usize + header.e_phoff as usize + i as usize * header.e_phentsize as usize) as *const program_header::ProgramHeader);
|
|
||||||
|
|
||||||
if segment.p_type == program_header::PT_LOAD || segment.p_type == program_header::PT_TLS {
|
|
||||||
segments.push(segment);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
segments
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the entry field of the header
|
/// Get the entry field of the header
|
||||||
pub unsafe fn entry(&self) -> usize {
|
pub fn entry(&self) -> usize {
|
||||||
let header = &*(self.data.as_ptr() as usize as *const header::Header);
|
self.header.e_entry as usize
|
||||||
header.e_entry as usize
|
}
|
||||||
|
|
||||||
|
/// Test function to run. Remove and replace with proper syscall
|
||||||
|
pub fn run(self) {
|
||||||
|
let mut active_table = unsafe { ActivePageTable::new() };
|
||||||
|
|
||||||
|
for segment in self.segments() {
|
||||||
|
println!("Segment {:X} flags {:X} off {:X} virt {:X} phys {:X} file {} mem {} align {}",
|
||||||
|
segment.p_type, segment.p_flags, segment.p_offset,
|
||||||
|
segment.p_vaddr, segment.p_paddr, segment.p_filesz,
|
||||||
|
segment.p_memsz, segment.p_align);
|
||||||
|
|
||||||
|
if segment.p_type == program_header::PT_LOAD {
|
||||||
|
let start_page = Page::containing_address(VirtualAddress::new(segment.p_vaddr as usize));
|
||||||
|
let end_page = Page::containing_address(VirtualAddress::new((segment.p_vaddr + segment.p_memsz) as usize));
|
||||||
|
|
||||||
|
for page in Page::range_inclusive(start_page, end_page) {
|
||||||
|
active_table.map(page, entry::NO_EXECUTE | entry::WRITABLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
// Update the page table
|
||||||
|
tlb::flush_all();
|
||||||
|
|
||||||
|
// Copy file data
|
||||||
|
memcpy(segment.p_vaddr as *mut u8,
|
||||||
|
(self.data.as_ptr() as usize + segment.p_offset as usize) as *const u8,
|
||||||
|
segment.p_filesz as usize);
|
||||||
|
// Set BSS
|
||||||
|
memset((segment.p_vaddr + segment.p_filesz) as *mut u8,
|
||||||
|
0,
|
||||||
|
(segment.p_memsz - segment.p_filesz) as usize);
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut flags = entry::NO_EXECUTE | entry::USER_ACCESSIBLE;
|
||||||
|
|
||||||
|
if segment.p_flags & program_header::PF_R == program_header::PF_R {
|
||||||
|
flags.insert(entry::PRESENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
// W ^ X. If it is executable, do not allow it to be writable, even if requested
|
||||||
|
if segment.p_flags & program_header::PF_X == program_header::PF_X {
|
||||||
|
flags.remove(entry::NO_EXECUTE);
|
||||||
|
} else if segment.p_flags & program_header::PF_W == program_header::PF_W {
|
||||||
|
flags.insert(entry::WRITABLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
for page in Page::range_inclusive(start_page, end_page) {
|
||||||
|
println!("{:X}: {:?}", page.start_address().get(), flags);
|
||||||
|
active_table.remap(page, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
// Update the page table
|
||||||
|
tlb::flush_all();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
// Map stack
|
||||||
|
let start_page = Page::containing_address(VirtualAddress::new(0x80000000));
|
||||||
|
let end_page = Page::containing_address(VirtualAddress::new(0x80000000 + 64*1024 - 1));
|
||||||
|
|
||||||
|
for page in Page::range_inclusive(start_page, end_page) {
|
||||||
|
active_table.map(page, entry::NO_EXECUTE | entry::WRITABLE | entry::USER_ACCESSIBLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the page table
|
||||||
|
tlb::flush_all();
|
||||||
|
|
||||||
|
// Clear stack
|
||||||
|
memset(0x80000000 as *mut u8, 0, 64 * 1024);
|
||||||
|
|
||||||
|
// Go to usermode
|
||||||
|
usermode(self.entry(), 0x80000000 + 64*1024 - 256);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ElfSegments<'a> {
|
||||||
|
data: &'a [u8],
|
||||||
|
header: &'a header::Header,
|
||||||
|
i: usize
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Iterator for ElfSegments<'a> {
|
||||||
|
type Item = &'a program_header::ProgramHeader;
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
if self.i < self.header.e_phnum as usize {
|
||||||
|
let item = unsafe {
|
||||||
|
&* ((
|
||||||
|
self.data.as_ptr() as usize
|
||||||
|
+ self.header.e_phoff as usize
|
||||||
|
+ self.i * self.header.e_phentsize as usize
|
||||||
|
) as *const program_header::ProgramHeader)
|
||||||
|
};
|
||||||
|
self.i += 1;
|
||||||
|
Some(item)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -133,6 +133,18 @@ pub extern fn kmain() {
|
||||||
let pid = syscall::getpid();
|
let pid = syscall::getpid();
|
||||||
println!("BSP: {:?}", pid);
|
println!("BSP: {:?}", pid);
|
||||||
|
|
||||||
|
let stdin = syscall::open("debug:".as_bytes(), 0);
|
||||||
|
println!("STDIN: {:?}", stdin);
|
||||||
|
|
||||||
|
let stdout = syscall::open("debug:".as_bytes(), 0);
|
||||||
|
println!("STDOUT: {:?}", stdout);
|
||||||
|
|
||||||
|
let stderr = syscall::open("debug:".as_bytes(), 0);
|
||||||
|
println!("STDERR: {:?}", stderr);
|
||||||
|
|
||||||
|
let elf = elf::Elf::from(include_bytes!("../build/init")).expect("could not load elf");
|
||||||
|
elf.run();
|
||||||
|
|
||||||
/*
|
/*
|
||||||
if let Ok(_context_lock) = context::contexts_mut().spawn(context_test) {
|
if let Ok(_context_lock) = context::contexts_mut().spawn(context_test) {
|
||||||
print!("Spawned context\n");
|
print!("Spawned context\n");
|
||||||
|
@ -143,7 +155,7 @@ pub extern fn kmain() {
|
||||||
|
|
||||||
print!("Main halt\n");
|
print!("Main halt\n");
|
||||||
*/
|
*/
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
unsafe { interrupt::enable_and_halt(); }
|
unsafe { interrupt::enable_and_halt(); }
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,34 +7,24 @@ use super::{Error, Result};
|
||||||
|
|
||||||
/// Read syscall
|
/// Read syscall
|
||||||
pub fn read(fd: usize, buf: &mut [u8]) -> Result<usize> {
|
pub fn read(fd: usize, buf: &mut [u8]) -> Result<usize> {
|
||||||
println!("Read {}: {}", fd, buf.len());
|
println!("Read {}: {:X} {}", fd, buf.as_ptr() as usize, buf.len());
|
||||||
if let Some(context_lock) = context::contexts().current() {
|
let contexts = context::contexts();
|
||||||
let context = context_lock.read();
|
let context_lock = contexts.current().ok_or(Error::NoProcess)?;
|
||||||
if let Some(file) = context.files.get(fd) {
|
let context = context_lock.read();
|
||||||
println!("{:?}", file);
|
let file = context.files.get(fd).ok_or(Error::BadFile)?;
|
||||||
Ok(0)
|
println!("{:?}", file);
|
||||||
} else {
|
Ok(0)
|
||||||
Err(Error::BadFile)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Err(Error::NoProcess)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Write syscall
|
/// Write syscall
|
||||||
pub fn write(fd: usize, buf: &[u8]) -> Result<usize> {
|
pub fn write(fd: usize, buf: &[u8]) -> Result<usize> {
|
||||||
println!("Write {}: {}", fd, buf.len());
|
println!("Write {}: {:X} {}", fd, buf.as_ptr() as usize, buf.len());
|
||||||
if let Some(context_lock) = context::contexts().current() {
|
let contexts = context::contexts();
|
||||||
let context = context_lock.read();
|
let context_lock = contexts.current().ok_or(Error::NoProcess)?;
|
||||||
if let Some(file) = context.files.get(fd) {
|
let context = context_lock.read();
|
||||||
println!("{:?}: {:?}", file, ::core::str::from_utf8(buf));
|
let file = context.files.get(fd).ok_or(Error::BadFile);
|
||||||
Ok(buf.len())
|
println!("{:?}: {:?}", file, ::core::str::from_utf8(buf));
|
||||||
} else {
|
Ok(buf.len())
|
||||||
Err(Error::BadFile)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Err(Error::NoProcess)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Open syscall
|
/// Open syscall
|
||||||
|
@ -45,31 +35,20 @@ pub fn open(path: &[u8], flags: usize) -> Result<usize> {
|
||||||
println!("Open namespace {:?} reference {:?}: {:X}", namespace_opt.map(::core::str::from_utf8), reference_opt.map(::core::str::from_utf8), flags);
|
println!("Open namespace {:?} reference {:?}: {:X}", namespace_opt.map(::core::str::from_utf8), reference_opt.map(::core::str::from_utf8), flags);
|
||||||
|
|
||||||
let file = {
|
let file = {
|
||||||
if let Some(namespace) = namespace_opt {
|
let namespace = namespace_opt.ok_or(Error::NoEntry)?;
|
||||||
let schemes = scheme::schemes();
|
let schemes = scheme::schemes();
|
||||||
if let Some(scheme_mutex) = schemes.get(namespace) {
|
let scheme_mutex = schemes.get(namespace).ok_or(Error::NoEntry)?;
|
||||||
scheme_mutex.lock().open(reference_opt.unwrap_or(b""), flags)
|
let file = scheme_mutex.lock().open(reference_opt.unwrap_or(b""), flags)?;
|
||||||
} else {
|
file
|
||||||
Err(Error::NoEntry)
|
};
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Err(Error::NoEntry)
|
|
||||||
}
|
|
||||||
}?;
|
|
||||||
|
|
||||||
if let Some(context_lock) = context::contexts().current() {
|
let contexts = context::contexts();
|
||||||
let mut context = context_lock.write();
|
let context_lock = contexts.current().ok_or(Error::NoProcess)?;
|
||||||
if let Some(fd) = context.add_file(::context::file::File {
|
let mut context = context_lock.write();
|
||||||
scheme: 0,
|
context.add_file(::context::file::File {
|
||||||
number: file
|
scheme: 0,
|
||||||
}) {
|
number: file
|
||||||
Ok(fd)
|
}).ok_or(Error::TooManyFiles)
|
||||||
} else {
|
|
||||||
Err(Error::TooManyFiles)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Err(Error::NoProcess)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Close syscall
|
/// Close syscall
|
||||||
|
|
|
@ -23,10 +23,8 @@ pub fn exec(path: &[u8], args: &[[usize; 2]]) -> Result<usize> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn getpid() -> Result<usize> {
|
pub fn getpid() -> Result<usize> {
|
||||||
if let Some(context_lock) = context::contexts().current() {
|
let contexts = context::contexts();
|
||||||
let context = context_lock.read();
|
let context_lock = contexts.current().ok_or(Error::NoProcess)?;
|
||||||
Ok(context.id)
|
let context = context_lock.read();
|
||||||
} else {
|
Ok(context.id)
|
||||||
Err(Error::NoProcess)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
3
syscall/Cargo.toml
Normal file
3
syscall/Cargo.toml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
[package]
|
||||||
|
name = "syscall"
|
||||||
|
version = "0.1.0"
|
314
syscall/src/error.rs
Normal file
314
syscall/src/error.rs
Normal file
|
@ -0,0 +1,314 @@
|
||||||
|
use core::{fmt, result};
|
||||||
|
|
||||||
|
pub struct Error {
|
||||||
|
pub errno: isize,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub type Result<T> = result::Result<T, Error>;
|
||||||
|
|
||||||
|
impl Error {
|
||||||
|
pub fn new(errno: isize) -> Error {
|
||||||
|
Error { errno: errno }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mux(result: Result<usize>) -> usize {
|
||||||
|
match result {
|
||||||
|
Ok(value) => value,
|
||||||
|
Err(error) => -error.errno as usize,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn demux(value: usize) -> Result<usize> {
|
||||||
|
let errno = -(value as isize);
|
||||||
|
if errno >= 1 && errno < STR_ERROR.len() as isize {
|
||||||
|
Err(Error::new(errno))
|
||||||
|
} else {
|
||||||
|
Ok(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn text(&self) -> &str {
|
||||||
|
if let Some(description) = STR_ERROR.get(self.errno as usize) {
|
||||||
|
description
|
||||||
|
} else {
|
||||||
|
"Unknown Error"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Error {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> result::Result<(), fmt::Error> {
|
||||||
|
f.write_str(self.text())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for Error {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> result::Result<(), fmt::Error> {
|
||||||
|
f.write_str(self.text())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const EPERM: isize = 1; /* Operation not permitted */
|
||||||
|
pub const ENOENT: isize = 2; /* No such file or directory */
|
||||||
|
pub const ESRCH: isize = 3; /* No such process */
|
||||||
|
pub const EINTR: isize = 4; /* Interrupted system call */
|
||||||
|
pub const EIO: isize = 5; /* I/O error */
|
||||||
|
pub const ENXIO: isize = 6; /* No such device or address */
|
||||||
|
pub const E2BIG: isize = 7; /* Argument list too long */
|
||||||
|
pub const ENOEXEC: isize = 8; /* Exec format error */
|
||||||
|
pub const EBADF: isize = 9; /* Bad file number */
|
||||||
|
pub const ECHILD: isize = 10; /* No child processes */
|
||||||
|
pub const EAGAIN: isize = 11; /* Try again */
|
||||||
|
pub const ENOMEM: isize = 12; /* Out of memory */
|
||||||
|
pub const EACCES: isize = 13; /* Permission denied */
|
||||||
|
pub const EFAULT: isize = 14; /* Bad address */
|
||||||
|
pub const ENOTBLK: isize = 15; /* Block device required */
|
||||||
|
pub const EBUSY: isize = 16; /* Device or resource busy */
|
||||||
|
pub const EEXIST: isize = 17; /* File exists */
|
||||||
|
pub const EXDEV: isize = 18; /* Cross-device link */
|
||||||
|
pub const ENODEV: isize = 19; /* No such device */
|
||||||
|
pub const ENOTDIR: isize = 20; /* Not a directory */
|
||||||
|
pub const EISDIR: isize = 21; /* Is a directory */
|
||||||
|
pub const EINVAL: isize = 22; /* Invalid argument */
|
||||||
|
pub const ENFILE: isize = 23; /* File table overflow */
|
||||||
|
pub const EMFILE: isize = 24; /* Too many open files */
|
||||||
|
pub const ENOTTY: isize = 25; /* Not a typewriter */
|
||||||
|
pub const ETXTBSY: isize = 26; /* Text file busy */
|
||||||
|
pub const EFBIG: isize = 27; /* File too large */
|
||||||
|
pub const ENOSPC: isize = 28; /* No space left on device */
|
||||||
|
pub const ESPIPE: isize = 29; /* Illegal seek */
|
||||||
|
pub const EROFS: isize = 30; /* Read-only file system */
|
||||||
|
pub const EMLINK: isize = 31; /* Too many links */
|
||||||
|
pub const EPIPE: isize = 32; /* Broken pipe */
|
||||||
|
pub const EDOM: isize = 33; /* Math argument out of domain of func */
|
||||||
|
pub const ERANGE: isize = 34; /* Math result not representable */
|
||||||
|
pub const EDEADLK: isize = 35; /* Resource deadlock would occur */
|
||||||
|
pub const ENAMETOOLONG: isize = 36; /* File name too long */
|
||||||
|
pub const ENOLCK: isize = 37; /* No record locks available */
|
||||||
|
pub const ENOSYS: isize = 38; /* Function not implemented */
|
||||||
|
pub const ENOTEMPTY: isize = 39; /* Directory not empty */
|
||||||
|
pub const ELOOP: isize = 40; /* Too many symbolic links encountered */
|
||||||
|
pub const EWOULDBLOCK: isize = 41; /* Operation would block */
|
||||||
|
pub const ENOMSG: isize = 42; /* No message of desired type */
|
||||||
|
pub const EIDRM: isize = 43; /* Identifier removed */
|
||||||
|
pub const ECHRNG: isize = 44; /* Channel number out of range */
|
||||||
|
pub const EL2NSYNC: isize = 45; /* Level 2 not synchronized */
|
||||||
|
pub const EL3HLT: isize = 46; /* Level 3 halted */
|
||||||
|
pub const EL3RST: isize = 47; /* Level 3 reset */
|
||||||
|
pub const ELNRNG: isize = 48; /* Link number out of range */
|
||||||
|
pub const EUNATCH: isize = 49; /* Protocol driver not attached */
|
||||||
|
pub const ENOCSI: isize = 50; /* No CSI structure available */
|
||||||
|
pub const EL2HLT: isize = 51; /* Level 2 halted */
|
||||||
|
pub const EBADE: isize = 52; /* Invalid exchange */
|
||||||
|
pub const EBADR: isize = 53; /* Invalid request descriptor */
|
||||||
|
pub const EXFULL: isize = 54; /* Exchange full */
|
||||||
|
pub const ENOANO: isize = 55; /* No anode */
|
||||||
|
pub const EBADRQC: isize = 56; /* Invalid request code */
|
||||||
|
pub const EBADSLT: isize = 57; /* Invalid slot */
|
||||||
|
pub const EDEADLOCK: isize = 58; /* Resource deadlock would occur */
|
||||||
|
pub const EBFONT: isize = 59; /* Bad font file format */
|
||||||
|
pub const ENOSTR: isize = 60; /* Device not a stream */
|
||||||
|
pub const ENODATA: isize = 61; /* No data available */
|
||||||
|
pub const ETIME: isize = 62; /* Timer expired */
|
||||||
|
pub const ENOSR: isize = 63; /* Out of streams resources */
|
||||||
|
pub const ENONET: isize = 64; /* Machine is not on the network */
|
||||||
|
pub const ENOPKG: isize = 65; /* Package not installed */
|
||||||
|
pub const EREMOTE: isize = 66; /* Object is remote */
|
||||||
|
pub const ENOLINK: isize = 67; /* Link has been severed */
|
||||||
|
pub const EADV: isize = 68; /* Advertise error */
|
||||||
|
pub const ESRMNT: isize = 69; /* Srmount error */
|
||||||
|
pub const ECOMM: isize = 70; /* Communication error on send */
|
||||||
|
pub const EPROTO: isize = 71; /* Protocol error */
|
||||||
|
pub const EMULTIHOP: isize = 72; /* Multihop attempted */
|
||||||
|
pub const EDOTDOT: isize = 73; /* RFS specific error */
|
||||||
|
pub const EBADMSG: isize = 74; /* Not a data message */
|
||||||
|
pub const EOVERFLOW: isize = 75; /* Value too large for defined data type */
|
||||||
|
pub const ENOTUNIQ: isize = 76; /* Name not unique on network */
|
||||||
|
pub const EBADFD: isize = 77; /* File descriptor in bad state */
|
||||||
|
pub const EREMCHG: isize = 78; /* Remote address changed */
|
||||||
|
pub const ELIBACC: isize = 79; /* Can not access a needed shared library */
|
||||||
|
pub const ELIBBAD: isize = 80; /* Accessing a corrupted shared library */
|
||||||
|
pub const ELIBSCN: isize = 81; /* .lib section in a.out corrupted */
|
||||||
|
pub const ELIBMAX: isize = 82; /* Attempting to link in too many shared libraries */
|
||||||
|
pub const ELIBEXEC: isize = 83; /* Cannot exec a shared library directly */
|
||||||
|
pub const EILSEQ: isize = 84; /* Illegal byte sequence */
|
||||||
|
pub const ERESTART: isize = 85; /* Interrupted system call should be restarted */
|
||||||
|
pub const ESTRPIPE: isize = 86; /* Streams pipe error */
|
||||||
|
pub const EUSERS: isize = 87; /* Too many users */
|
||||||
|
pub const ENOTSOCK: isize = 88; /* Socket operation on non-socket */
|
||||||
|
pub const EDESTADDRREQ: isize = 89; /* Destination address required */
|
||||||
|
pub const EMSGSIZE: isize = 90; /* Message too long */
|
||||||
|
pub const EPROTOTYPE: isize = 91; /* Protocol wrong type for socket */
|
||||||
|
pub const ENOPROTOOPT: isize = 92; /* Protocol not available */
|
||||||
|
pub const EPROTONOSUPPORT: isize = 93; /* Protocol not supported */
|
||||||
|
pub const ESOCKTNOSUPPORT: isize = 94; /* Socket type not supported */
|
||||||
|
pub const EOPNOTSUPP: isize = 95; /* Operation not supported on transport endpoint */
|
||||||
|
pub const EPFNOSUPPORT: isize = 96; /* Protocol family not supported */
|
||||||
|
pub const EAFNOSUPPORT: isize = 97; /* Address family not supported by protocol */
|
||||||
|
pub const EADDRINUSE: isize = 98; /* Address already in use */
|
||||||
|
pub const EADDRNOTAVAIL: isize = 99; /* Cannot assign requested address */
|
||||||
|
pub const ENETDOWN: isize = 100; /* Network is down */
|
||||||
|
pub const ENETUNREACH: isize = 101; /* Network is unreachable */
|
||||||
|
pub const ENETRESET: isize = 102; /* Network dropped connection because of reset */
|
||||||
|
pub const ECONNABORTED: isize = 103; /* Software caused connection abort */
|
||||||
|
pub const ECONNRESET: isize = 104; /* Connection reset by peer */
|
||||||
|
pub const ENOBUFS: isize = 105; /* No buffer space available */
|
||||||
|
pub const EISCONN: isize = 106; /* Transport endpoint is already connected */
|
||||||
|
pub const ENOTCONN: isize = 107; /* Transport endpoint is not connected */
|
||||||
|
pub const ESHUTDOWN: isize = 108; /* Cannot send after transport endpoint shutdown */
|
||||||
|
pub const ETOOMANYREFS: isize = 109; /* Too many references: cannot splice */
|
||||||
|
pub const ETIMEDOUT: isize = 110; /* Connection timed out */
|
||||||
|
pub const ECONNREFUSED: isize = 111; /* Connection refused */
|
||||||
|
pub const EHOSTDOWN: isize = 112; /* Host is down */
|
||||||
|
pub const EHOSTUNREACH: isize = 113; /* No route to host */
|
||||||
|
pub const EALREADY: isize = 114; /* Operation already in progress */
|
||||||
|
pub const EINPROGRESS: isize = 115; /* Operation now in progress */
|
||||||
|
pub const ESTALE: isize = 116; /* Stale NFS file handle */
|
||||||
|
pub const EUCLEAN: isize = 117; /* Structure needs cleaning */
|
||||||
|
pub const ENOTNAM: isize = 118; /* Not a XENIX named type file */
|
||||||
|
pub const ENAVAIL: isize = 119; /* No XENIX semaphores available */
|
||||||
|
pub const EISNAM: isize = 120; /* Is a named type file */
|
||||||
|
pub const EREMOTEIO: isize = 121; /* Remote I/O error */
|
||||||
|
pub const EDQUOT: isize = 122; /* Quota exceeded */
|
||||||
|
pub const ENOMEDIUM: isize = 123; /* No medium found */
|
||||||
|
pub const EMEDIUMTYPE: isize = 124; /* Wrong medium type */
|
||||||
|
pub const ECANCELED: isize = 125; /* Operation Canceled */
|
||||||
|
pub const ENOKEY: isize = 126; /* Required key not available */
|
||||||
|
pub const EKEYEXPIRED: isize = 127; /* Key has expired */
|
||||||
|
pub const EKEYREVOKED: isize = 128; /* Key has been revoked */
|
||||||
|
pub const EKEYREJECTED: isize = 129; /* Key was rejected by service */
|
||||||
|
pub const EOWNERDEAD: isize = 130; /* Owner died */
|
||||||
|
pub const ENOTRECOVERABLE: isize = 131; /* State not recoverable */
|
||||||
|
|
||||||
|
pub static STR_ERROR: [&'static str; 132] = ["Success",
|
||||||
|
"Operation not permitted",
|
||||||
|
"No such file or directory",
|
||||||
|
"No such process",
|
||||||
|
"Interrupted system call",
|
||||||
|
"I/O error",
|
||||||
|
"No such device or address",
|
||||||
|
"Argument list too long",
|
||||||
|
"Exec format error",
|
||||||
|
"Bad file number",
|
||||||
|
"No child processes",
|
||||||
|
"Try again",
|
||||||
|
"Out of memory",
|
||||||
|
"Permission denied",
|
||||||
|
"Bad address",
|
||||||
|
"Block device required",
|
||||||
|
"Device or resource busy",
|
||||||
|
"File exists",
|
||||||
|
"Cross-device link",
|
||||||
|
"No such device",
|
||||||
|
"Not a directory",
|
||||||
|
"Is a directory",
|
||||||
|
"Invalid argument",
|
||||||
|
"File table overflow",
|
||||||
|
"Too many open files",
|
||||||
|
"Not a typewriter",
|
||||||
|
"Text file busy",
|
||||||
|
"File too large",
|
||||||
|
"No space left on device",
|
||||||
|
"Illegal seek",
|
||||||
|
"Read-only file system",
|
||||||
|
"Too many links",
|
||||||
|
"Broken pipe",
|
||||||
|
"Math argument out of domain of func",
|
||||||
|
"Math result not representable",
|
||||||
|
"Resource deadlock would occur",
|
||||||
|
"File name too long",
|
||||||
|
"No record locks available",
|
||||||
|
"Function not implemented",
|
||||||
|
"Directory not empty",
|
||||||
|
"Too many symbolic links encountered",
|
||||||
|
"Operation would block",
|
||||||
|
"No message of desired type",
|
||||||
|
"Identifier removed",
|
||||||
|
"Channel number out of range",
|
||||||
|
"Level 2 not synchronized",
|
||||||
|
"Level 3 halted",
|
||||||
|
"Level 3 reset",
|
||||||
|
"Link number out of range",
|
||||||
|
"Protocol driver not attached",
|
||||||
|
"No CSI structure available",
|
||||||
|
"Level 2 halted",
|
||||||
|
"Invalid exchange",
|
||||||
|
"Invalid request descriptor",
|
||||||
|
"Exchange full",
|
||||||
|
"No anode",
|
||||||
|
"Invalid request code",
|
||||||
|
"Invalid slot",
|
||||||
|
"Resource deadlock would occur",
|
||||||
|
"Bad font file format",
|
||||||
|
"Device not a stream",
|
||||||
|
"No data available",
|
||||||
|
"Timer expired",
|
||||||
|
"Out of streams resources",
|
||||||
|
"Machine is not on the network",
|
||||||
|
"Package not installed",
|
||||||
|
"Object is remote",
|
||||||
|
"Link has been severed",
|
||||||
|
"Advertise error",
|
||||||
|
"Srmount error",
|
||||||
|
"Communication error on send",
|
||||||
|
"Protocol error",
|
||||||
|
"Multihop attempted",
|
||||||
|
"RFS specific error",
|
||||||
|
"Not a data message",
|
||||||
|
"Value too large for defined data type",
|
||||||
|
"Name not unique on network",
|
||||||
|
"File descriptor in bad state",
|
||||||
|
"Remote address changed",
|
||||||
|
"Can not access a needed shared library",
|
||||||
|
"Accessing a corrupted shared library",
|
||||||
|
".lib section in a.out corrupted",
|
||||||
|
"Attempting to link in too many shared libraries",
|
||||||
|
"Cannot exec a shared library directly",
|
||||||
|
"Illegal byte sequence",
|
||||||
|
"Interrupted system call should be restarted",
|
||||||
|
"Streams pipe error",
|
||||||
|
"Too many users",
|
||||||
|
"Socket operation on non-socket",
|
||||||
|
"Destination address required",
|
||||||
|
"Message too long",
|
||||||
|
"Protocol wrong type for socket",
|
||||||
|
"Protocol not available",
|
||||||
|
"Protocol not supported",
|
||||||
|
"Socket type not supported",
|
||||||
|
"Operation not supported on transport endpoint",
|
||||||
|
"Protocol family not supported",
|
||||||
|
"Address family not supported by protocol",
|
||||||
|
"Address already in use",
|
||||||
|
"Cannot assign requested address",
|
||||||
|
"Network is down",
|
||||||
|
"Network is unreachable",
|
||||||
|
"Network dropped connection because of reset",
|
||||||
|
"Software caused connection abort",
|
||||||
|
"Connection reset by peer",
|
||||||
|
"No buffer space available",
|
||||||
|
"Transport endpoint is already connected",
|
||||||
|
"Transport endpoint is not connected",
|
||||||
|
"Cannot send after transport endpoint shutdown",
|
||||||
|
"Too many references: cannot splice",
|
||||||
|
"Connection timed out",
|
||||||
|
"Connection refused",
|
||||||
|
"Host is down",
|
||||||
|
"No route to host",
|
||||||
|
"Operation already in progress",
|
||||||
|
"Operation now in progress",
|
||||||
|
"Stale NFS file handle",
|
||||||
|
"Structure needs cleaning",
|
||||||
|
"Not a XENIX named type file",
|
||||||
|
"No XENIX semaphores available",
|
||||||
|
"Is a named type file",
|
||||||
|
"Remote I/O error",
|
||||||
|
"Quota exceeded",
|
||||||
|
"No medium found",
|
||||||
|
"Wrong medium type",
|
||||||
|
"Operation Canceled",
|
||||||
|
"Required key not available",
|
||||||
|
"Key has expired",
|
||||||
|
"Key has been revoked",
|
||||||
|
"Key was rejected by service",
|
||||||
|
"Owner died",
|
||||||
|
"State not recoverable"];
|
210
syscall/src/lib.rs
Normal file
210
syscall/src/lib.rs
Normal file
|
@ -0,0 +1,210 @@
|
||||||
|
#![feature(asm)]
|
||||||
|
#![no_std]
|
||||||
|
|
||||||
|
pub use self::arch::*;
|
||||||
|
pub use self::error::*;
|
||||||
|
|
||||||
|
#[cfg(target_arch = "x86")]
|
||||||
|
#[path="x86.rs"]
|
||||||
|
pub mod arch;
|
||||||
|
|
||||||
|
#[cfg(target_arch = "x86_64")]
|
||||||
|
#[path="x86_64.rs"]
|
||||||
|
pub mod arch;
|
||||||
|
|
||||||
|
pub mod error;
|
||||||
|
|
||||||
|
pub const SYS_BRK: usize = 45;
|
||||||
|
pub const SYS_CHDIR: usize = 12;
|
||||||
|
pub const SYS_CLONE: usize = 120;
|
||||||
|
pub const CLONE_VM: usize = 0x100;
|
||||||
|
pub const CLONE_FS: usize = 0x200;
|
||||||
|
pub const CLONE_FILES: usize = 0x400;
|
||||||
|
pub const CLONE_VFORK: usize = 0x4000;
|
||||||
|
/// Mark this clone as supervised.
|
||||||
|
///
|
||||||
|
/// This means that the process can run in supervised mode, even not being connected to
|
||||||
|
/// a supervisor yet. In other words, the parent can later on supervise the process and handle
|
||||||
|
/// the potential blocking syscall.
|
||||||
|
///
|
||||||
|
/// This is an important security measure, since otherwise the process would be able to fork it
|
||||||
|
/// self right after starting, making supervising it impossible.
|
||||||
|
pub const CLONE_SUPERVISE: usize = 0x400000;
|
||||||
|
pub const SYS_CLOSE: usize = 6;
|
||||||
|
pub const SYS_CLOCK_GETTIME: usize = 265;
|
||||||
|
pub const CLOCK_REALTIME: usize = 1;
|
||||||
|
pub const CLOCK_MONOTONIC: usize = 4;
|
||||||
|
pub const SYS_DUP: usize = 41;
|
||||||
|
pub const SYS_EXECVE: usize = 11;
|
||||||
|
pub const SYS_EXIT: usize = 1;
|
||||||
|
pub const SYS_FPATH: usize = 928;
|
||||||
|
pub const SYS_FSTAT: usize = 28;
|
||||||
|
pub const MODE_DIR: u16 = 0x4000;
|
||||||
|
pub const MODE_FILE: u16 = 0x8000;
|
||||||
|
pub const MODE_ALL: u16 = MODE_DIR | MODE_FILE;
|
||||||
|
pub const SYS_FSYNC: usize = 118;
|
||||||
|
pub const SYS_FTRUNCATE: usize = 93;
|
||||||
|
pub const SYS_FUTEX: usize = 240;
|
||||||
|
pub const FUTEX_WAIT: usize = 0;
|
||||||
|
pub const FUTEX_WAKE: usize = 1;
|
||||||
|
pub const FUTEX_REQUEUE: usize = 2;
|
||||||
|
pub const SYS_GETPID: usize = 20;
|
||||||
|
pub const SYS_IOPL: usize = 110;
|
||||||
|
pub const SYS_LINK: usize = 9;
|
||||||
|
pub const SYS_LSEEK: usize = 19;
|
||||||
|
pub const SEEK_SET: usize = 0;
|
||||||
|
pub const SEEK_CUR: usize = 1;
|
||||||
|
pub const SEEK_END: usize = 2;
|
||||||
|
pub const SYS_MKDIR: usize = 39;
|
||||||
|
pub const SYS_NANOSLEEP: usize = 162;
|
||||||
|
pub const SYS_OPEN: usize = 5;
|
||||||
|
pub const O_RDONLY: usize = 0;
|
||||||
|
pub const O_WRONLY: usize = 1;
|
||||||
|
pub const O_RDWR: usize = 2;
|
||||||
|
pub const O_NONBLOCK: usize = 4;
|
||||||
|
pub const O_APPEND: usize = 8;
|
||||||
|
pub const O_SHLOCK: usize = 0x10;
|
||||||
|
pub const O_EXLOCK: usize = 0x20;
|
||||||
|
pub const O_ASYNC: usize = 0x40;
|
||||||
|
pub const O_FSYNC: usize = 0x80;
|
||||||
|
pub const O_CREAT: usize = 0x200;
|
||||||
|
pub const O_TRUNC: usize = 0x400;
|
||||||
|
pub const O_EXCL: usize = 0x800;
|
||||||
|
pub const SYS_PIPE2: usize = 331;
|
||||||
|
pub const SYS_READ: usize = 3;
|
||||||
|
pub const SYS_RMDIR: usize = 84;
|
||||||
|
pub const SYS_UNLINK: usize = 10;
|
||||||
|
pub const SYS_WAITPID: usize = 7;
|
||||||
|
pub const SYS_WRITE: usize = 4;
|
||||||
|
pub const SYS_YIELD: usize = 158;
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, Default)]
|
||||||
|
#[repr(packed)]
|
||||||
|
pub struct Stat {
|
||||||
|
pub st_dev: u16,
|
||||||
|
pub st_ino: u16,
|
||||||
|
pub st_mode: u16,
|
||||||
|
pub st_nlink: u16,
|
||||||
|
pub st_uid: u16,
|
||||||
|
pub st_gid: u16,
|
||||||
|
pub st_rdev: u16,
|
||||||
|
pub st_size: u32,
|
||||||
|
pub st_atime: u32,
|
||||||
|
pub st_mtime: u32,
|
||||||
|
pub st_ctime: u32
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, Default)]
|
||||||
|
#[repr(packed)]
|
||||||
|
pub struct TimeSpec {
|
||||||
|
pub tv_sec: i64,
|
||||||
|
pub tv_nsec: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn brk(addr: usize) -> Result<usize> {
|
||||||
|
syscall1(SYS_BRK, addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn chdir(path: &str) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_CHDIR, path.as_ptr() as usize, path.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn clone(flags: usize) -> Result<usize> {
|
||||||
|
syscall1(SYS_CLONE, flags)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn close(fd: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall1(SYS_CLOSE, fd) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn clock_gettime(clock: usize, tp: &mut TimeSpec) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_CLOCK_GETTIME, clock, tp as *mut TimeSpec as usize) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn dup(fd: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall1(SYS_DUP, fd) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn execve(path: *const u8, args: *const *const u8) -> Result<usize> {
|
||||||
|
syscall2(SYS_EXECVE, path as usize, args as usize)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn exit(status: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall1(SYS_EXIT, status) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fpath(fd: usize, buf: &mut [u8]) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_FPATH, fd, buf.as_mut_ptr() as usize, buf.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fstat(fd: usize, stat: &mut Stat) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_FSTAT, fd, stat as *mut Stat as usize) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fsync(fd: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall1(SYS_FSYNC, fd) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn ftruncate(fd: usize, len: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_FTRUNCATE, fd, len) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn futex(addr: *mut i32, op: usize, val: i32, val2: usize, addr2: *mut i32) -> Result<usize> {
|
||||||
|
syscall5(SYS_FUTEX, addr as usize, op, (val as isize) as usize, val2, addr2 as usize)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn getpid() -> Result<usize> {
|
||||||
|
unsafe { syscall0(SYS_GETPID) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn iopl(level: usize) -> Result<usize> {
|
||||||
|
syscall1(SYS_IOPL, level)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn link(old: *const u8, new: *const u8) -> Result<usize> {
|
||||||
|
syscall2(SYS_LINK, old as usize, new as usize)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn lseek(fd: usize, offset: isize, whence: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_LSEEK, fd, offset as usize, whence) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mkdir(path: &str, mode: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_MKDIR, path.as_ptr() as usize, path.len(), mode) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn nanosleep(req: &TimeSpec, rem: &mut TimeSpec) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_NANOSLEEP, req as *const TimeSpec as usize, rem as *mut TimeSpec as usize) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn open(path: &str, flags: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_OPEN, path.as_ptr() as usize, path.len(), flags) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pipe2(fds: &mut [usize; 2], flags: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_PIPE2, fds.as_ptr() as usize, flags) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn read(fd: usize, buf: &mut [u8]) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_READ, fd, buf.as_mut_ptr() as usize, buf.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn rmdir(path: &str) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_RMDIR, path.as_ptr() as usize, path.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unlink(path: &str) -> Result<usize> {
|
||||||
|
unsafe { syscall2(SYS_UNLINK, path.as_ptr() as usize, path.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn waitpid(pid: usize, status: &mut usize, options: usize) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_WAITPID, pid, status as *mut usize as usize, options) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn write(fd: usize, buf: &[u8]) -> Result<usize> {
|
||||||
|
unsafe { syscall3(SYS_WRITE, fd, buf.as_ptr() as usize, buf.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn sched_yield() -> Result<usize> {
|
||||||
|
unsafe { syscall0(SYS_YIELD) }
|
||||||
|
}
|
61
syscall/src/x86.rs
Normal file
61
syscall/src/x86.rs
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
use error::{Error, Result};
|
||||||
|
|
||||||
|
pub unsafe fn syscall0(mut a: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall1(mut a: usize, b: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a), "{ebx}"(b)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall2(mut a: usize, b: usize, c: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a), "{ebx}"(b), "{ecx}"(c)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall3(mut a: usize, b: usize, c: usize, d: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a), "{ebx}"(b), "{ecx}"(c), "{edx}"(d)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall4(mut a: usize, b: usize, c: usize, d: usize, e: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a), "{ebx}"(b), "{ecx}"(c), "{edx}"(d), "{esi}"(e)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall5(mut a: usize, b: usize, c: usize, d: usize, e: usize, f: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={eax}"(a)
|
||||||
|
: "{eax}"(a), "{ebx}"(b), "{ecx}"(c), "{edx}"(d), "{esi}"(e), "{edi}"(f)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
61
syscall/src/x86_64.rs
Normal file
61
syscall/src/x86_64.rs
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
use error::{Error, Result};
|
||||||
|
|
||||||
|
pub unsafe fn syscall0(mut a: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall1(mut a: usize, b: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a), "{rbx}"(b)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall2(mut a: usize, b: usize, c: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a), "{rbx}"(b), "{rcx}"(c)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall3(mut a: usize, b: usize, c: usize, d: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a), "{rbx}"(b), "{rcx}"(c), "{rdx}"(d)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall4(mut a: usize, b: usize, c: usize, d: usize, e: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a), "{rbx}"(b), "{rcx}"(c), "{rdx}"(d), "{rsi}"(e)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub unsafe fn syscall5(mut a: usize, b: usize, c: usize, d: usize, e: usize, f: usize) -> Result<usize> {
|
||||||
|
asm!("int 0x80"
|
||||||
|
: "={rax}"(a)
|
||||||
|
: "{rax}"(a), "{rbx}"(b), "{rcx}"(c), "{rdx}"(d), "{rsi}"(e), "{rdi}"(f)
|
||||||
|
: "memory"
|
||||||
|
: "intel", "volatile");
|
||||||
|
|
||||||
|
Error::demux(a)
|
||||||
|
}
|
Loading…
Reference in a new issue