redox/kernel/syscall/validate.rs

45 lines
1.6 KiB
Rust
Raw Normal View History

2016-09-21 02:03:14 +02:00
use core::{mem, slice};
2016-09-17 02:50:47 +02:00
2016-09-21 02:03:14 +02:00
use arch::paging::{ActivePageTable, Page, VirtualAddress, entry};
use syscall::error::*;
2016-09-17 02:50:47 +02:00
2016-09-21 02:03:14 +02:00
fn validate(address: usize, size: usize, flags: entry::EntryFlags) -> Result<()> {
let active_table = unsafe { ActivePageTable::new() };
let start_page = Page::containing_address(VirtualAddress::new(address));
let end_page = Page::containing_address(VirtualAddress::new(address + size - 1));
for page in Page::range_inclusive(start_page, end_page) {
2016-12-27 19:18:41 +01:00
if let Some(page_flags) = active_table.translate_page_flags(page) {
if ! page_flags.contains(flags) {
//println!("{:X}: Not {:?}", page.start_address().get(), flags);
return Err(Error::new(EFAULT));
}
} else {
//println!("{:X}: Not found", page.start_address().get());
2016-09-21 02:03:14 +02:00
return Err(Error::new(EFAULT));
}
}
Ok(())
}
2016-09-17 02:50:47 +02:00
/// Convert a pointer and length to slice, if valid
pub fn validate_slice<T>(ptr: *const T, len: usize) -> Result<&'static [T]> {
2016-09-21 02:03:14 +02:00
if len == 0 {
Ok(&[])
} else {
validate(ptr as usize, len * mem::size_of::<T>(), entry::PRESENT /* TODO | entry::USER_ACCESSIBLE */)?;
Ok(unsafe { slice::from_raw_parts(ptr, len) })
}
2016-09-17 02:50:47 +02:00
}
/// Convert a pointer and length to slice, if valid
pub fn validate_slice_mut<T>(ptr: *mut T, len: usize) -> Result<&'static mut [T]> {
2016-09-21 02:03:14 +02:00
if len == 0 {
Ok(&mut [])
} else {
validate(ptr as usize, len * mem::size_of::<T>(), entry::PRESENT | entry::WRITABLE /* TODO | entry::USER_ACCESSIBLE */)?;
Ok(unsafe { slice::from_raw_parts_mut(ptr, len) })
}
2016-09-17 02:50:47 +02:00
}