From 12abe152324ba327d8c3462a20a6674d6b7e6ad6 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Sat, 17 Aug 2024 00:14:21 +0000 Subject: [PATCH 01/45] MetaPtr + usize/u32 changes This changes some u32s into usizes as appropriate. It also introduces a new MetaPtr type that is a pointer that has explicit target dependant metadata. On many platforms, this will just be a wrapper around usize. On CHERI platforms MetaPtr will be a capability. See later commit. It also adds a new syscall encoding `encode_syscall_return_metaptr` That is intended to work for both 32/64 platforms with or without this extra metadata. Change-Id: I40faa11c1fd53debc6e9b21d00772660cacf8cab --- arch/cortex-m/src/syscall.rs | 13 +- arch/rv32i/src/syscall.rs | 10 +- kernel/src/grant.rs | 4 +- kernel/src/kernel.rs | 13 +- kernel/src/lib.rs | 1 + kernel/src/memop.rs | 22 +-- kernel/src/metaptr.rs | 113 ++++++++++++ kernel/src/process.rs | 17 +- kernel/src/process_standard.rs | 49 ++++-- kernel/src/syscall.rs | 294 ++++++++++++++++++++++---------- kernel/src/upcall.rs | 37 ++-- libraries/tock-tbf/src/types.rs | 6 +- 12 files changed, 425 insertions(+), 154 deletions(-) create mode 100644 kernel/src/metaptr.rs diff --git a/arch/cortex-m/src/syscall.rs b/arch/cortex-m/src/syscall.rs index 18b2130365..f6b70689f9 100644 --- a/arch/cortex-m/src/syscall.rs +++ b/arch/cortex-m/src/syscall.rs @@ -232,9 +232,9 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall // - Stack offset 4 is R12, which the syscall interface ignores let stack_bottom = state.psp as *mut usize; ptr::write(stack_bottom.offset(7), state.psr); //......... -> APSR - ptr::write(stack_bottom.offset(6), callback.pc | 1); //... -> PC + ptr::write(stack_bottom.offset(6), usize::from(callback.pc) | 1); //... -> PC ptr::write(stack_bottom.offset(5), state.yield_pc | 1); // -> LR - ptr::write(stack_bottom.offset(3), callback.argument3); // -> R3 + ptr::write(stack_bottom.offset(3), callback.argument3.into()); // -> R3 ptr::write(stack_bottom.offset(2), callback.argument2); // -> R2 ptr::write(stack_bottom.offset(1), callback.argument1); // -> R1 ptr::write(stack_bottom.offset(0), callback.argument0); // -> R0 @@ -300,8 +300,13 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall // Use the helper function to convert these raw values into a Tock // `Syscall` type. - let syscall = - kernel::syscall::Syscall::from_register_arguments(svc_num, r0, r1, r2, r3); + let syscall = kernel::syscall::Syscall::from_register_arguments( + svc_num, + r0, + r1.into(), + r2.into(), + r3.into(), + ); match syscall { Some(s) => kernel::syscall::ContextSwitchReason::SyscallFired { syscall: s }, diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index 1dec998943..a2bc8e3c84 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -195,7 +195,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { state.regs[R_A0] = callback.argument0 as u32; state.regs[R_A1] = callback.argument1 as u32; state.regs[R_A2] = callback.argument2 as u32; - state.regs[R_A3] = callback.argument3 as u32; + state.regs[R_A3] = callback.argument3.as_ptr() as usize as u32; // We also need to set the return address (ra) register so that the new // function that the process is running returns to the correct location. @@ -206,7 +206,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { state.regs[R_RA] = state.pc; // Save the PC we expect to execute. - state.pc = callback.pc as u32; + state.pc = callback.pc.as_ptr() as usize as u32; Ok(()) } @@ -631,9 +631,9 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { let syscall = kernel::syscall::Syscall::from_register_arguments( state.regs[R_A4] as u8, state.regs[R_A0] as usize, - state.regs[R_A1] as usize, - state.regs[R_A2] as usize, - state.regs[R_A3] as usize, + (state.regs[R_A1] as usize).into(), + (state.regs[R_A2] as usize).into(), + (state.regs[R_A3] as usize).into(), ); match syscall { diff --git a/kernel/src/grant.rs b/kernel/src/grant.rs index 2788bbd7c9..dbe931ccac 100644 --- a/kernel/src/grant.rs +++ b/kernel/src/grant.rs @@ -707,8 +707,8 @@ impl<'a> GrantKernelData<'a> { #[repr(C)] #[derive(Default)] struct SavedUpcall { - appdata: usize, - fn_ptr: Option>, + appdata: crate::upcall::AppdataType, + fn_ptr: crate::upcall::FnPtrType, } /// A minimal representation of a read-only allow from app, used for storing a diff --git a/kernel/src/kernel.rs b/kernel/src/kernel.rs index b9a22625ef..b24ed03a6e 100644 --- a/kernel/src/kernel.rs +++ b/kernel/src/kernel.rs @@ -9,7 +9,6 @@ //! etc.) is defined in the `scheduler` subcrate and selected by a board. use core::cell::Cell; -use core::ptr::NonNull; use crate::capabilities; use crate::config; @@ -877,15 +876,17 @@ impl Kernel { subscribe_num: subdriver_number, }; + // TODO: when the compiler supports capability types bring this back + // as a NonNull type. // First check if `upcall_ptr` is null. A null // `upcall_ptr` will result in `None` here and // represents the special "unsubscribe" operation. - let ptr = NonNull::new(upcall_ptr); + // let ptr = NonNull::new(upcall_ptr); // For convenience create an `Upcall` type now. This is // just a data structure and doesn't do any checking or // conversion. - let upcall = Upcall::new(process.processid(), upcall_id, appdata, ptr); + let upcall = Upcall::new(process.processid(), upcall_id, appdata, upcall_ptr); // If `ptr` is not null, we must first verify that the // upcall function pointer is within process accessible @@ -895,8 +896,8 @@ impl Kernel { // > process executable memory...), the kernel...MUST // > immediately return a failure with a error code of // > `INVALID`. - let rval1 = ptr.and_then(|upcall_ptr_nonnull| { - if !process.is_valid_upcall_function_pointer(upcall_ptr_nonnull) { + let rval1 = upcall_ptr.map_or(None, |upcall_ptr_nonnull| { + if !process.is_valid_upcall_function_pointer(upcall_ptr_nonnull.as_ptr()) { Some(ErrorCode::INVAL) } else { None @@ -1010,7 +1011,7 @@ impl Kernel { process.processid(), driver_number, subdriver_number, - upcall_ptr as usize, + upcall_ptr, appdata, rval ); diff --git a/kernel/src/lib.rs b/kernel/src/lib.rs index 65b8443cd8..a1d797c762 100644 --- a/kernel/src/lib.rs +++ b/kernel/src/lib.rs @@ -114,6 +114,7 @@ pub mod grant; pub mod hil; pub mod introspection; pub mod ipc; +pub mod metaptr; pub mod platform; pub mod process; pub mod process_checker; diff --git a/kernel/src/memop.rs b/kernel/src/memop.rs index 35d4cf9177..358c77c8a9 100644 --- a/kernel/src/memop.rs +++ b/kernel/src/memop.rs @@ -52,35 +52,35 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall // Op Type 1: SBRK 1 => process .sbrk(r1 as isize) - .map(|addr| SyscallReturn::SuccessU32(addr as u32)) + .map(|addr| SyscallReturn::SuccessPtr(addr)) .unwrap_or(SyscallReturn::Failure(ErrorCode::NOMEM)), // Op Type 2: Process memory start - 2 => SyscallReturn::SuccessU32(process.get_addresses().sram_start as u32), + 2 => SyscallReturn::SuccessUSize(process.get_addresses().sram_start), // Op Type 3: Process memory end - 3 => SyscallReturn::SuccessU32(process.get_addresses().sram_end as u32), + 3 => SyscallReturn::SuccessUSize(process.get_addresses().sram_end), // Op Type 4: Process flash start - 4 => SyscallReturn::SuccessU32(process.get_addresses().flash_start as u32), + 4 => SyscallReturn::SuccessUSize(process.get_addresses().flash_start), // Op Type 5: Process flash end - 5 => SyscallReturn::SuccessU32(process.get_addresses().flash_end as u32), + 5 => SyscallReturn::SuccessUSize(process.get_addresses().flash_end), // Op Type 6: Grant region begin - 6 => SyscallReturn::SuccessU32(process.get_addresses().sram_grant_start as u32), + 6 => SyscallReturn::SuccessUSize(process.get_addresses().sram_grant_start), // Op Type 7: Number of defined writeable regions in the TBF header. - 7 => SyscallReturn::SuccessU32(process.number_writeable_flash_regions() as u32), + 7 => SyscallReturn::SuccessUSize(process.number_writeable_flash_regions()), // Op Type 8: The start address of the writeable region indexed by r1. 8 => { - let flash_start = process.get_addresses().flash_start as u32; + let flash_start = process.get_addresses().flash_start; let (offset, size) = process.get_writeable_flash_region(r1); if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessU32(flash_start + offset) + SyscallReturn::SuccessUSize(flash_start + offset) } } @@ -88,12 +88,12 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall // Returns (void*) -1 on failure, meaning the selected writeable region // does not exist. 9 => { - let flash_start = process.get_addresses().flash_start as u32; + let flash_start = process.get_addresses().flash_start; let (offset, size) = process.get_writeable_flash_region(r1); if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessU32(flash_start + offset + size) + SyscallReturn::SuccessUSize(flash_start + offset + size) } } diff --git a/kernel/src/metaptr.rs b/kernel/src/metaptr.rs new file mode 100644 index 0000000000..f65ce1e0ef --- /dev/null +++ b/kernel/src/metaptr.rs @@ -0,0 +1,113 @@ +// Licensed under the Apache License, Version 2.0 or the MIT License. +// SPDX-License-Identifier: Apache-2.0 OR MIT +// Copyright Google LLC 2024. + +//! Defines the MetaPtr type + +use core::fmt::{Formatter, LowerHex, UpperHex}; +use core::ops::AddAssign; + +/// A pointer with target specific metadata. +/// This should be used any time the kernel wishes to grant authority to the user, or any time +/// the user should be required to prove validity of a pointer. +#[derive(Default, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] +#[repr(transparent)] +pub struct MetaPtr { + ptr: usize, +} + +#[derive(Copy, Clone, PartialEq)] +pub enum MetaPermissions { + Any, + Read, + Write, + ReadWrite, + Execute, +} + +impl From for usize { + #[inline] + fn from(from: MetaPtr) -> Self { + from.ptr + } +} + +impl From for MetaPtr { + #[inline] + fn from(from: usize) -> Self { + Self { ptr: from } + } +} + +impl UpperHex for MetaPtr { + #[inline] + fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { + UpperHex::fmt(&self.ptr, f) + } +} + +impl LowerHex for MetaPtr { + #[inline] + fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { + LowerHex::fmt(&self.ptr, f) + } +} + +impl AddAssign for MetaPtr { + #[inline] + fn add_assign(&mut self, rhs: usize) { + self.ptr.add_assign(rhs) + } +} + +impl MetaPtr { + pub fn as_ptr(&self) -> *const () { + self.ptr as *const () + } + + /// Convert to a raw pointer, checking that metadata allows a particular set of permissions over + /// a given number of bytes. + /// If the metadata does not allow for this, returns null. + /// If no such metadata exists, this succeeds. + #[inline] + pub fn as_ptr_checked(&self, _length: usize, _perms: MetaPermissions) -> *const () { + self.ptr as *const () + } + + #[inline] + pub fn new_with_metadata( + ptr: *const (), + _base: usize, + _length: usize, + _perms: MetaPermissions, + ) -> Self { + Self { ptr: ptr as usize } + } + + #[inline] + pub fn map_or(&self, default: U, f: F) -> U + where + F: FnOnce(&Self) -> U, + { + if self.ptr == 0usize { + default + } else { + f(self) + } + } + + #[inline] + pub fn map_or_else(&self, default: D, f: F) -> U + where + D: FnOnce() -> U, + F: FnOnce(&Self) -> U, + { + let addr: usize = (*self).into(); + + if addr == 0 { + default() + } else { + f(self) + } + } +} diff --git a/kernel/src/process.rs b/kernel/src/process.rs index 323112c5fd..c19f7a2438 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -14,6 +14,7 @@ use crate::capabilities; use crate::errorcode::ErrorCode; use crate::ipc; use crate::kernel::Kernel; +use crate::metaptr::MetaPtr; use crate::platform::mpu::{self}; use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions; @@ -539,7 +540,7 @@ pub trait Process { /// process's memory region. /// - [`Error::KernelError`] if there was an internal kernel error. This is /// a bug. - fn brk(&self, new_break: *const u8) -> Result<*const u8, Error>; + fn brk(&self, new_break: *const u8) -> Result; /// Change the location of the program break by `increment` bytes, /// reallocate the MPU region covering program memory, and return the @@ -559,7 +560,7 @@ pub trait Process { /// process's memory region. /// - [`Error::KernelError`] if there was an internal kernel error. This is /// a bug. - fn sbrk(&self, increment: isize) -> Result<*const u8, Error>; + fn sbrk(&self, increment: isize) -> Result; /// How many writeable flash regions defined in the TBF header for this /// process. @@ -574,10 +575,10 @@ pub trait Process { /// /// ## Returns /// - /// A tuple containing the a `u32` of the offset from the beginning of the - /// process's flash region where the writeable region starts and a `u32` of + /// A tuple containing the a `usize` of the offset from the beginning of the + /// process's flash region where the writeable region starts and a `usize` of /// the size of the region in bytes. - fn get_writeable_flash_region(&self, region_index: usize) -> (u32, u32); + fn get_writeable_flash_region(&self, region_index: usize) -> (usize, usize); /// Debug function to update the kernel on where the stack starts for this /// process. Processes are not required to call this through the memop @@ -790,7 +791,7 @@ pub trait Process { /// /// Returns `true` if the upcall function pointer is valid for this process, /// and `false` otherwise. - fn is_valid_upcall_function_pointer(&self, upcall_fn: NonNull<()>) -> bool; + fn is_valid_upcall_function_pointer(&self, upcall_fn: *const ()) -> bool; // functions for processes that are architecture specific @@ -1078,9 +1079,9 @@ pub struct FunctionCall { /// The third argument to the function. pub argument2: usize, /// The fourth argument to the function. - pub argument3: usize, + pub argument3: MetaPtr, /// The PC of the function to execute. - pub pc: usize, + pub pc: MetaPtr, } /// This is similar to `FunctionCall` but for the special case of the Null diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 146334da0a..ce52f54bb7 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -20,6 +20,8 @@ use crate::config; use crate::debug; use crate::errorcode::ErrorCode; use crate::kernel::Kernel; +use crate::metaptr::MetaPermissions::Execute; +use crate::metaptr::{MetaPermissions, MetaPtr}; use crate::platform::chip::Chip; use crate::platform::mpu::{self, MPU}; use crate::process::BinaryVersion; @@ -503,7 +505,7 @@ impl Process for ProcessStandard<'_, C> { self.header.number_writeable_flash_regions() } - fn get_writeable_flash_region(&self, region_index: usize) -> (u32, u32) { + fn get_writeable_flash_region(&self, region_index: usize) -> (usize, usize) { self.header.get_writeable_flash_region(region_index) } @@ -582,7 +584,7 @@ impl Process for ProcessStandard<'_, C> { }) } - fn sbrk(&self, increment: isize) -> Result<*const u8, Error> { + fn sbrk(&self, increment: isize) -> Result { // Do not modify an inactive process. if !self.is_running() { return Err(Error::InactiveApp); @@ -592,7 +594,7 @@ impl Process for ProcessStandard<'_, C> { self.brk(new_break) } - fn brk(&self, new_break: *const u8) -> Result<*const u8, Error> { + fn brk(&self, new_break: *const u8) -> Result { // Do not modify an inactive process. if !self.is_running() { return Err(Error::InactiveApp); @@ -614,7 +616,16 @@ impl Process for ProcessStandard<'_, C> { let old_break = self.app_break.get(); self.app_break.set(new_break); self.chip.mpu().configure_mpu(config); - Ok(old_break) + + let base = self.mem_start() as usize; + let break_result = MetaPtr::new_with_metadata( + old_break as *const (), + base, + (new_break as usize) - base, + MetaPermissions::ReadWrite, + ); + + Ok(break_result) } }) } @@ -984,8 +995,8 @@ impl Process for ProcessStandard<'_, C> { }) } - fn is_valid_upcall_function_pointer(&self, upcall_fn: NonNull<()>) -> bool { - let ptr = upcall_fn.as_ptr() as *const u8; + fn is_valid_upcall_function_pointer(&self, upcall_fn: *const ()) -> bool { + let ptr = upcall_fn as *const u8; let size = mem::size_of::<*const u8>(); // It is okay if this function is in memory or flash. @@ -1739,8 +1750,11 @@ impl ProcessStandard<'_, C> { let flash_start = process.flash.as_ptr(); let app_start = flash_start.wrapping_add(process.header.get_app_start_offset() as usize) as usize; - let init_fn = + let init_addr = flash_start.wrapping_add(process.header.get_init_function_offset() as usize) as usize; + let fn_base = flash_start as usize; + let fn_len = process.flash.len(); + let init_fn = MetaPtr::new_with_metadata(init_addr as *const (), fn_base, fn_len, Execute); process.tasks.map(|tasks| { tasks.enqueue(Task::FunctionCall(FunctionCall { @@ -1749,7 +1763,7 @@ impl ProcessStandard<'_, C> { argument0: app_start, argument1: process.memory_start as usize, argument2: process.memory_len, - argument3: process.app_break.get() as usize, + argument3: (process.app_break.get() as usize).into(), })); }); @@ -1902,16 +1916,23 @@ impl ProcessStandard<'_, C> { let flash_start = self.flash_start(); let app_start = flash_start.wrapping_add(self.header.get_app_start_offset() as usize) as usize; - let init_fn = + let init_addr = flash_start.wrapping_add(self.header.get_init_function_offset() as usize) as usize; + let init_fn = MetaPtr::new_with_metadata( + init_addr as *const (), + flash_start as usize, + (self.flash_end() as usize) - (flash_start as usize), + Execute, + ); + self.enqueue_task(Task::FunctionCall(FunctionCall { source: FunctionCallSource::Kernel, pc: init_fn, argument0: app_start, argument1: self.memory_start as usize, argument2: self.memory_len, - argument3: self.app_break.get() as usize, + argument3: (self.app_break.get() as usize).into(), })) } @@ -1921,6 +1942,10 @@ impl ProcessStandard<'_, C> { /// to be accessible to the process and to not overlap with the grant /// region. fn in_app_owned_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { + // TODO: On CHERI platforms, it impossible to form syscalls with pointers + // that are not in app memory. However, buf_start_addr is not the right + // type to make this function always return true. If MetaPtr makes it + // slightly further, we can skip this check. let buf_end_addr = buf_start_addr.wrapping_add(size); buf_end_addr >= buf_start_addr @@ -1933,6 +1958,10 @@ impl ProcessStandard<'_, C> { /// this method returns true, the buffer is guaranteed to be readable to the /// process. fn in_app_flash_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { + // TODO: On CHERI platforms, it impossible to form syscalls with pointers + // that are not in app memory. However, buf_start_addr is not the right + // type to make this function always return true. If MetaPtr makes it + // slightly further, we can skip this check. let buf_end_addr = buf_start_addr.wrapping_add(size); buf_end_addr >= buf_start_addr diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index e8975f4a7c..b08a11d446 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -69,6 +69,7 @@ use core::fmt::Write; use crate::errorcode::ErrorCode; +use crate::metaptr::{MetaPermissions, MetaPtr}; use crate::process; pub use crate::syscall_driver::{CommandReturn, SyscallDriver}; @@ -166,10 +167,12 @@ pub enum Syscall { driver_number: usize, /// The subscribe identifier. subdriver_number: usize, + // On CHERI platforms we need to maintain metadata for + // these two as they are used to access code/data. /// Upcall pointer to the upcall function. - upcall_ptr: *mut (), + upcall_ptr: MetaPtr, /// Userspace application data. - appdata: usize, + appdata: MetaPtr, }, /// Structure representing an invocation of the Command system call class. @@ -252,53 +255,53 @@ impl Syscall { pub fn from_register_arguments( syscall_number: u8, r0: usize, - r1: usize, - r2: usize, - r3: usize, + r1: MetaPtr, + r2: MetaPtr, + r3: MetaPtr, ) -> Option { match SyscallClass::try_from(syscall_number) { Ok(SyscallClass::Yield) => Some(Syscall::Yield { which: r0, - param_a: r1, - param_b: r2, + param_a: r1.into(), + param_b: r2.into(), }), Ok(SyscallClass::Subscribe) => Some(Syscall::Subscribe { driver_number: r0, - subdriver_number: r1, - upcall_ptr: r2 as *mut (), + subdriver_number: r1.into(), + upcall_ptr: r2, appdata: r3, }), Ok(SyscallClass::Command) => Some(Syscall::Command { driver_number: r0, - subdriver_number: r1, - arg0: r2, - arg1: r3, + subdriver_number: r1.into(), + arg0: r2.into(), + arg1: r3.into(), }), Ok(SyscallClass::ReadWriteAllow) => Some(Syscall::ReadWriteAllow { driver_number: r0, - subdriver_number: r1, - allow_address: r2 as *mut u8, - allow_size: r3, + subdriver_number: r1.into(), + allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::ReadWrite) as *mut u8, + allow_size: r3.into(), }), Ok(SyscallClass::UserspaceReadableAllow) => Some(Syscall::UserspaceReadableAllow { driver_number: r0, - subdriver_number: r1, - allow_address: r2 as *mut u8, - allow_size: r3, + subdriver_number: r1.into(), + allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::Read) as *mut u8, + allow_size: r3.into(), }), Ok(SyscallClass::ReadOnlyAllow) => Some(Syscall::ReadOnlyAllow { driver_number: r0, - subdriver_number: r1, - allow_address: r2 as *const u8, - allow_size: r3, + subdriver_number: r1.into(), + allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::Read) as *mut u8, + allow_size: r3.into(), }), Ok(SyscallClass::Memop) => Some(Syscall::Memop { operand: r0, - arg0: r1, + arg0: r1.into(), }), Ok(SyscallClass::Exit) => Some(Syscall::Exit { which: r0, - completion_code: r1, + completion_code: r1.into(), }), Err(_) => None, } @@ -437,6 +440,11 @@ pub enum SyscallReturn { /// Generic success case, with an additional 32-bit and 64-bit data field SuccessU32U64(u32, u64), + /// Generic success case, with an additional pointer with metadata + /// On CHERI, this grants authority. + /// Access to this return is therefore privileged. + SuccessPtr(MetaPtr), + // These following types are used by the scheduler so that it can return // values to userspace in an architecture (pointer-width) independent way. // The kernel passes these types (rather than ProcessBuffer or Upcall) for @@ -446,6 +454,15 @@ pub enum SyscallReturn { // (pointers out of valid memory), the kernel cannot construct an // ProcessBuffer or Upcall type but needs to be able to return a failure. // -pal 11/24/20 + + // FIXME: We need to think about what these look like on CHERI + // Really, things that were capabilities should come back as capabilities. + // However, we discarded all capability information at the syscall boundary. + // We could always use our own DDC, with just the permissions and length implied by the + // specific syscall. This would certainly got give userspace _extra_ authority, + // but might rob them of some bounds / permissions. This is what is implemented currently. + // Preferable behavior is not to discard the capability so early (it should make it as far + // as grant is stored in grant allow slots) /// Read/Write allow success case, returns the previous allowed buffer and /// size to the process. AllowReadWriteSuccess(*mut u8, usize), @@ -494,6 +511,18 @@ impl SyscallReturn { res.into_inner() } + /// Construct either SuccessU32 or SuccessU64 depending on platform. + #[allow(non_snake_case)] + pub(crate) fn SuccessUSize(val: usize) -> Self { + if core::mem::size_of::() == 8 { + SyscallReturn::SuccessU64(val as u64) + } else if core::mem::size_of::() == 4 { + SyscallReturn::SuccessU32(val as u32) + } else { + panic!(); + } + } + /// Returns true if the [`SyscallReturn`] is any success type. pub(crate) fn is_success(&self) -> bool { match self { @@ -503,6 +532,7 @@ impl SyscallReturn { SyscallReturn::SuccessU32U32U32(_, _, _) => true, SyscallReturn::SuccessU64(_) => true, SyscallReturn::SuccessU32U64(_, _) => true, + SyscallReturn::SuccessPtr(_) => true, SyscallReturn::AllowReadWriteSuccess(_, _) => true, SyscallReturn::UserspaceReadableAllowSuccess(_, _) => true, SyscallReturn::AllowReadOnlySuccess(_, _) => true, @@ -522,111 +552,197 @@ impl SyscallReturn { /// Encode the system call return value into 4 registers, following the /// encoding specified in TRD104. Architectures which do not follow TRD104 /// are free to define their own encoding. + /// TODO: deprecate in favour of the more general one pub fn encode_syscall_return(&self, a0: &mut u32, a1: &mut u32, a2: &mut u32, a3: &mut u32) { + if core::mem::size_of::() == core::mem::size_of::() { + // SAFETY: if the two unsized integers are the same size references to them + // can be safely transmuted. + // Ugly coercion could be avoided by first copying to the stack, then assigning with + // "as" in order to satisfy the compiler. But I expect this function will disappear + // in favour of just using the usize one. + unsafe { + let a0 = &mut *(core::ptr::from_mut(a0) as *mut MetaPtr); + let a1 = &mut *(core::ptr::from_mut(a1) as *mut MetaPtr); + let a2 = &mut *(core::ptr::from_mut(a2) as *mut MetaPtr); + let a3 = &mut *(core::ptr::from_mut(a3) as *mut MetaPtr); + self.encode_syscall_return_mptr(a0, a1, a2, a3); + } + } else { + panic!("encode_syscall_return used on a 64-bit platform or CHERI platform") + } + } + + /// The obvious extension of TRD104 that works for 32-bit and 64-bit platforms. + /// This makes no changes to TRD104 on 32-bit platforms. + /// On 64-bit platforms, both 64-bit and usize values are passed as a single register, + /// shifting down register number if that means fewer registers are needed needed. + /// For usize, this is the obvious choice. + /// For explicitly 64-bit arguments, this would require rewriting prototypes for userspace + /// functions between 32 and 64 bit platforms. + /// However, no driver currently USES any 64-bit values. + /// Any new ones on 64-bit platforms would likely prefer just passing the value. + /// If they would not, I suspect many really want usize anyway (and that is what is used for + /// the syscalls handled directly by the kernel). Maybe they should be written in terms of that, + /// and some helpful aliases created for FailureUSIZE etc. + /// I think packing two 32-bit values into 64-bits would gain nothing and pollute a whole lot + /// of user code. + /// I have not considered usize other than 4 and 8 bytes. + /// Also handles the CHERI extension as follows: + /// the high part of any MetaPtr register is zero'd if any non capability-sized arguments are + /// passed. + /// SuccessPtr is as passed the full MetaPtr register. + /// Pointers from allow'd buffers have minimal bounds attached that cover their length, + /// and the same permissions that were checked at the syscall boundary. + pub fn encode_syscall_return_mptr( + &self, + a0: &mut MetaPtr, + a1: &mut MetaPtr, + a2: &mut MetaPtr, + a3: &mut MetaPtr, + ) { + // On 32-bit CHERI, given that capabilities cannot be used as 64-bit integers, 64-bit + // integers will still be returned as two 32-bit values in different registers. + fn write_64(a: &mut MetaPtr, b: &mut MetaPtr, val: u64) { + let is_64_bit = core::mem::size_of::() == 8; + if !is_64_bit { + let (msb, lsb) = u64_to_be_u32s(val); + *a = (lsb as usize).into(); + *b = (msb as usize).into(); + } else { + *a = (val as usize).into(); + } + } + match *self { SyscallReturn::Failure(e) => { - *a0 = SyscallReturnVariant::Failure as u32; - *a1 = usize::from(e) as u32; + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(e)).into(); } SyscallReturn::FailureU32(e, data0) => { - *a0 = SyscallReturnVariant::FailureU32 as u32; - *a1 = usize::from(e) as u32; - *a2 = data0; + *a0 = (SyscallReturnVariant::FailureU32 as usize).into(); + *a1 = usize::from(e).into(); + *a2 = (data0 as usize).into(); } SyscallReturn::FailureU32U32(e, data0, data1) => { - *a0 = SyscallReturnVariant::FailureU32U32 as u32; - *a1 = usize::from(e) as u32; - *a2 = data0; - *a3 = data1; + *a0 = (SyscallReturnVariant::FailureU32U32 as usize).into(); + *a1 = (usize::from(e)).into(); + *a2 = (data0 as usize).into(); + *a3 = (data1 as usize).into(); } SyscallReturn::FailureU64(e, data0) => { - let (data0_msb, data0_lsb) = u64_to_be_u32s(data0); - *a0 = SyscallReturnVariant::FailureU64 as u32; - *a1 = usize::from(e) as u32; - *a2 = data0_lsb; - *a3 = data0_msb; + *a0 = (SyscallReturnVariant::FailureU64 as usize).into(); + *a1 = (usize::from(e)).into(); + write_64(a2, a3, data0) } SyscallReturn::Success => { - *a0 = SyscallReturnVariant::Success as u32; + *a0 = (SyscallReturnVariant::Success as usize).into(); } SyscallReturn::SuccessU32(data0) => { - *a0 = SyscallReturnVariant::SuccessU32 as u32; - *a1 = data0; + *a0 = (SyscallReturnVariant::SuccessU32 as usize).into(); + *a1 = (data0 as usize).into(); } SyscallReturn::SuccessU32U32(data0, data1) => { - *a0 = SyscallReturnVariant::SuccessU32U32 as u32; - *a1 = data0; - *a2 = data1; + *a0 = (SyscallReturnVariant::SuccessU32U32 as usize).into(); + *a1 = (data0 as usize).into(); + *a2 = (data1 as usize).into(); } SyscallReturn::SuccessU32U32U32(data0, data1, data2) => { - *a0 = SyscallReturnVariant::SuccessU32U32U32 as u32; - *a1 = data0; - *a2 = data1; - *a3 = data2; + *a0 = (SyscallReturnVariant::SuccessU32U32U32 as usize).into(); + *a1 = (data0 as usize).into(); + *a2 = (data1 as usize).into(); + *a3 = (data2 as usize).into(); } SyscallReturn::SuccessU64(data0) => { - let (data0_msb, data0_lsb) = u64_to_be_u32s(data0); - - *a0 = SyscallReturnVariant::SuccessU64 as u32; - *a1 = data0_lsb; - *a2 = data0_msb; + *a0 = (SyscallReturnVariant::SuccessU64 as usize).into(); + write_64(a1, a2, data0); } SyscallReturn::SuccessU32U64(data0, data1) => { - let (data1_msb, data1_lsb) = u64_to_be_u32s(data1); - - *a0 = SyscallReturnVariant::SuccessU32U64 as u32; - *a1 = data0; - *a2 = data1_lsb; - *a3 = data1_msb; + *a0 = (SyscallReturnVariant::SuccessU32U64 as usize).into(); + *a1 = (data0 as usize).into(); + write_64(a2, a3, data1); } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = SyscallReturnVariant::SuccessU32U32 as u32; - *a1 = ptr as u32; - *a2 = len as u32; + *a0 = (SyscallReturnVariant::Success as usize).into(); + *a1 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::ReadWrite, + ); + *a2 = len.into(); } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = SyscallReturnVariant::SuccessU32U32 as u32; - *a1 = ptr as u32; - *a2 = len as u32; + *a0 = (SyscallReturnVariant::Success as usize).into(); + *a1 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::Read, + ); + *a2 = len.into(); } SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { - *a0 = SyscallReturnVariant::FailureU32U32 as u32; - *a1 = usize::from(err) as u32; - *a2 = ptr as u32; - *a3 = len as u32; + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::ReadWrite, + ); + *a3 = len.into(); } SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { - *a0 = SyscallReturnVariant::FailureU32U32 as u32; - *a1 = usize::from(err) as u32; - *a2 = ptr as u32; - *a3 = len as u32; + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::Read, + ); + *a3 = len.into(); } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = SyscallReturnVariant::SuccessU32U32 as u32; - *a1 = ptr as u32; - *a2 = len as u32; + *a0 = (SyscallReturnVariant::Success as usize).into(); + *a1 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::Read, + ); + *a2 = len.into(); } SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { - *a0 = SyscallReturnVariant::FailureU32U32 as u32; - *a1 = usize::from(err) as u32; - *a2 = ptr as u32; - *a3 = len as u32; + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = MetaPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + MetaPermissions::Read, + ); + *a3 = len.into(); } SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = SyscallReturnVariant::SuccessU32U32 as u32; - *a1 = ptr as u32; - *a2 = data as u32; + *a0 = (SyscallReturnVariant::Success as usize).into(); + *a1 = (ptr as usize).into(); + *a2 = data.into(); } SyscallReturn::SubscribeFailure(err, ptr, data) => { - *a0 = SyscallReturnVariant::FailureU32U32 as u32; - *a1 = usize::from(err) as u32; - *a2 = ptr as u32; - *a3 = data as u32; + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = (ptr as usize).into(); + *a3 = data.into(); + } + SyscallReturn::SuccessPtr(metaptr) => { + *a0 = (SyscallReturnVariant::Success as usize).into(); + *a1 = metaptr; } SyscallReturn::YieldWaitFor(data0, data1, data2) => { - *a0 = data0 as u32; - *a1 = data1 as u32; - *a2 = data2 as u32; + *a0 = data0.into(); + *a1 = data1.into(); + *a2 = data2.into(); } } } diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index a719f3d37d..d0187647c5 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -4,10 +4,9 @@ //! Data structure for storing an upcall from the kernel to a process. -use core::ptr::NonNull; - use crate::config; use crate::debug; +use crate::metaptr::MetaPtr; use crate::process; use crate::process::ProcessId; use crate::syscall::SyscallReturn; @@ -70,6 +69,11 @@ pub enum UpcallError { KernelError, } +// FIXME: When we get CHERI compiler support, these can go back to the proper types +// b/274586199 +pub(crate) type AppdataType = MetaPtr; +pub(crate) type FnPtrType = MetaPtr; + /// Type for calling an upcall in a process. /// /// This is essentially a wrapper around a function pointer with associated @@ -83,7 +87,7 @@ pub(crate) struct Upcall { pub(crate) upcall_id: UpcallId, /// The application data passed by the app when `subscribe()` was called. - pub(crate) appdata: usize, + pub(crate) appdata: AppdataType, /// A pointer to the first instruction of the function in the app that /// corresponds to this upcall. @@ -91,15 +95,15 @@ pub(crate) struct Upcall { /// If this value is `None`, this is a null upcall, which cannot actually be /// scheduled. An `Upcall` can be null when it is first created, or after an /// app unsubscribes from an upcall. - pub(crate) fn_ptr: Option>, + pub(crate) fn_ptr: FnPtrType, } impl Upcall { pub(crate) fn new( process_id: ProcessId, upcall_id: UpcallId, - appdata: usize, - fn_ptr: Option>, + appdata: AppdataType, + fn_ptr: FnPtrType, ) -> Upcall { Upcall { process_id, @@ -147,7 +151,7 @@ impl Upcall { argument1: r1, argument2: r2, argument3: self.appdata, - pc: fp.as_ptr() as usize, + pc: *fp, })) }, ); @@ -178,7 +182,8 @@ impl Upcall { self.upcall_id.driver_num, self.upcall_id.subscribe_num, self.fn_ptr - .map_or(core::ptr::null_mut::<()>(), |fp| fp.as_ptr()) as usize, + .map_or(core::ptr::null_mut::<()>(), |fp| fp.as_ptr() as *mut ()) + as usize, r0, r1, r2, @@ -198,10 +203,10 @@ impl Upcall { /// We provide this `.into` function because the return type needs to /// include the function pointer of the upcall. pub(crate) fn into_subscribe_success(self) -> SyscallReturn { - match self.fn_ptr { - Some(fp) => SyscallReturn::SubscribeSuccess(fp.as_ptr(), self.appdata), - None => SyscallReturn::SubscribeSuccess(core::ptr::null::<()>(), self.appdata), - } + self.fn_ptr.map_or( + SyscallReturn::SubscribeSuccess(core::ptr::null::<()>(), self.appdata.into()), + |fp| SyscallReturn::SubscribeSuccess(fp.as_ptr(), self.appdata.into()), + ) } /// Create a failure case syscall return type suitable for returning to @@ -214,9 +219,9 @@ impl Upcall { /// We provide this `.into` function because the return type needs to /// include the function pointer of the upcall. pub(crate) fn into_subscribe_failure(self, err: ErrorCode) -> SyscallReturn { - match self.fn_ptr { - Some(fp) => SyscallReturn::SubscribeFailure(err, fp.as_ptr(), self.appdata), - None => SyscallReturn::SubscribeFailure(err, core::ptr::null::<()>(), self.appdata), - } + self.fn_ptr.map_or( + SyscallReturn::SubscribeFailure(err, core::ptr::null::<()>(), self.appdata.into()), + |fp| SyscallReturn::SubscribeFailure(err, fp.as_ptr(), self.appdata.into()), + ) } } diff --git a/libraries/tock-tbf/src/types.rs b/libraries/tock-tbf/src/types.rs index c12ac77937..46b0c8409f 100644 --- a/libraries/tock-tbf/src/types.rs +++ b/libraries/tock-tbf/src/types.rs @@ -788,7 +788,7 @@ impl TbfHeader { } /// Get the offset and size of a given flash region. - pub fn get_writeable_flash_region(&self, index: usize) -> (u32, u32) { + pub fn get_writeable_flash_region(&self, index: usize) -> (usize, usize) { match *self { TbfHeader::TbfHeaderV2(hd) => hd.writeable_regions.map_or((0, 0), |wr_slice| { fn get_region( @@ -807,8 +807,8 @@ impl TbfHeader { match get_region(wr_slice, index) { Ok(wr) => ( - wr.writeable_flash_region_offset, - wr.writeable_flash_region_size, + wr.writeable_flash_region_offset as usize, + wr.writeable_flash_region_size as usize, ), Err(()) => (0, 0), } From c32507d822eb91edfcfc4b2431f55ce85e910bf1 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 14 Oct 2024 18:40:59 +0000 Subject: [PATCH 02/45] Have MetaPtr wrap a pointer not usize If we ever refactored to support strict provenance or a cheri purecap rustc, *const() is a far better choice than usize. Change-Id: I56947d53dc2f2ef9d1d5ac80641bc4410f383813 --- arch/rv32i/src/syscall.rs | 4 ++-- kernel/src/metaptr.rs | 40 +++++++++++++++++++++++---------------- kernel/src/syscall.rs | 7 ++++--- kernel/src/upcall.rs | 4 ++-- 4 files changed, 32 insertions(+), 23 deletions(-) diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index a2bc8e3c84..7581a07d15 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -195,7 +195,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { state.regs[R_A0] = callback.argument0 as u32; state.regs[R_A1] = callback.argument1 as u32; state.regs[R_A2] = callback.argument2 as u32; - state.regs[R_A3] = callback.argument3.as_ptr() as usize as u32; + state.regs[R_A3] = callback.argument3.as_ptr::<()>() as usize as u32; // We also need to set the return address (ra) register so that the new // function that the process is running returns to the correct location. @@ -206,7 +206,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { state.regs[R_RA] = state.pc; // Save the PC we expect to execute. - state.pc = callback.pc.as_ptr() as usize as u32; + state.pc = callback.pc.as_ptr::<()>() as usize as u32; Ok(()) } diff --git a/kernel/src/metaptr.rs b/kernel/src/metaptr.rs index f65ce1e0ef..28eb55a850 100644 --- a/kernel/src/metaptr.rs +++ b/kernel/src/metaptr.rs @@ -10,10 +10,18 @@ use core::ops::AddAssign; /// A pointer with target specific metadata. /// This should be used any time the kernel wishes to grant authority to the user, or any time /// the user should be required to prove validity of a pointer. -#[derive(Default, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] +#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] #[repr(transparent)] pub struct MetaPtr { - ptr: usize, + ptr: *const (), +} + +impl Default for MetaPtr { + fn default() -> Self { + Self { + ptr: core::ptr::null(), + } + } } #[derive(Copy, Clone, PartialEq)] @@ -28,41 +36,43 @@ pub enum MetaPermissions { impl From for usize { #[inline] fn from(from: MetaPtr) -> Self { - from.ptr + from.ptr as usize } } impl From for MetaPtr { #[inline] fn from(from: usize) -> Self { - Self { ptr: from } + Self { + ptr: from as *const (), + } } } impl UpperHex for MetaPtr { #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { - UpperHex::fmt(&self.ptr, f) + UpperHex::fmt(&(self.ptr as usize), f) } } impl LowerHex for MetaPtr { #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { - LowerHex::fmt(&self.ptr, f) + LowerHex::fmt(&(self.ptr as usize), f) } } impl AddAssign for MetaPtr { #[inline] fn add_assign(&mut self, rhs: usize) { - self.ptr.add_assign(rhs) + self.ptr = (self.ptr as *const u8).wrapping_add(rhs) as *const (); } } impl MetaPtr { - pub fn as_ptr(&self) -> *const () { - self.ptr as *const () + pub fn as_ptr(&self) -> *const T { + self.ptr as *const T } /// Convert to a raw pointer, checking that metadata allows a particular set of permissions over @@ -70,8 +80,8 @@ impl MetaPtr { /// If the metadata does not allow for this, returns null. /// If no such metadata exists, this succeeds. #[inline] - pub fn as_ptr_checked(&self, _length: usize, _perms: MetaPermissions) -> *const () { - self.ptr as *const () + pub fn as_ptr_checked(&self, _length: usize, _perms: MetaPermissions) -> *const T { + self.ptr as *const T } #[inline] @@ -81,7 +91,7 @@ impl MetaPtr { _length: usize, _perms: MetaPermissions, ) -> Self { - Self { ptr: ptr as usize } + Self { ptr } } #[inline] @@ -89,7 +99,7 @@ impl MetaPtr { where F: FnOnce(&Self) -> U, { - if self.ptr == 0usize { + if self.ptr.is_null() { default } else { f(self) @@ -102,9 +112,7 @@ impl MetaPtr { D: FnOnce() -> U, F: FnOnce(&Self) -> U, { - let addr: usize = (*self).into(); - - if addr == 0 { + if self.ptr.is_null() { default() } else { f(self) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index b08a11d446..0434a14856 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -280,19 +280,20 @@ impl Syscall { Ok(SyscallClass::ReadWriteAllow) => Some(Syscall::ReadWriteAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::ReadWrite) as *mut u8, + allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::ReadWrite) + as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::UserspaceReadableAllow) => Some(Syscall::UserspaceReadableAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::Read) as *mut u8, + allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::Read) as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::ReadOnlyAllow) => Some(Syscall::ReadOnlyAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked(r3.into(), MetaPermissions::Read) as *mut u8, + allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::Read) as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::Memop) => Some(Syscall::Memop { diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index d0187647c5..98a44aacde 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -182,8 +182,8 @@ impl Upcall { self.upcall_id.driver_num, self.upcall_id.subscribe_num, self.fn_ptr - .map_or(core::ptr::null_mut::<()>(), |fp| fp.as_ptr() as *mut ()) - as usize, + .map_or(core::ptr::null_mut::<()>(), |fp| fp.as_ptr::<()>() + as *mut ()) as usize, r0, r1, r2, From 677183fbb519c612fb82ab10c252325b48fbc398 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 14 Oct 2024 18:48:16 +0000 Subject: [PATCH 03/45] Link to CHERI tracking issue Change-Id: I8049bb48e48a162feb261cb4e1e15f7e5b4d8f6e --- kernel/src/kernel.rs | 2 +- kernel/src/process.rs | 2 ++ kernel/src/upcall.rs | 1 + 3 files changed, 4 insertions(+), 1 deletion(-) diff --git a/kernel/src/kernel.rs b/kernel/src/kernel.rs index b24ed03a6e..45ee440c35 100644 --- a/kernel/src/kernel.rs +++ b/kernel/src/kernel.rs @@ -877,7 +877,7 @@ impl Kernel { }; // TODO: when the compiler supports capability types bring this back - // as a NonNull type. + // as a NonNull type. https://github.com/tock/tock/issues/4134. // First check if `upcall_ptr` is null. A null // `upcall_ptr` will result in `None` here and // represents the special "unsubscribe" operation. diff --git a/kernel/src/process.rs b/kernel/src/process.rs index c19f7a2438..ccd16383d3 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -791,6 +791,8 @@ pub trait Process { /// /// Returns `true` if the upcall function pointer is valid for this process, /// and `false` otherwise. + /// upcall_fn can eventually be a better type: + /// https://github.com/tock/tock/issues/4134 fn is_valid_upcall_function_pointer(&self, upcall_fn: *const ()) -> bool; // functions for processes that are architecture specific diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index 98a44aacde..51b6661c85 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -71,6 +71,7 @@ pub enum UpcallError { // FIXME: When we get CHERI compiler support, these can go back to the proper types // b/274586199 +// https://github.com/tock/tock/issues/4134 pub(crate) type AppdataType = MetaPtr; pub(crate) type FnPtrType = MetaPtr; From 6254b9ed0ccc5a7cbc0defbc1f47428c6f4ffb7f Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 14 Oct 2024 21:10:35 +0000 Subject: [PATCH 04/45] Add explicit return codes for usize/ptr Make return variants that contain pointers and usizes explict about this type information. A backwards compatability mapping is also provided for legacy u32 platforms. Change-Id: Ied5513c58bfa87be3599ba366692f66af3da1691 --- kernel/src/memop.rs | 16 +++++------ kernel/src/syscall.rs | 64 +++++++++++++++++++++++++++++-------------- 2 files changed, 51 insertions(+), 29 deletions(-) diff --git a/kernel/src/memop.rs b/kernel/src/memop.rs index 358c77c8a9..d04ef315f0 100644 --- a/kernel/src/memop.rs +++ b/kernel/src/memop.rs @@ -56,22 +56,22 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall .unwrap_or(SyscallReturn::Failure(ErrorCode::NOMEM)), // Op Type 2: Process memory start - 2 => SyscallReturn::SuccessUSize(process.get_addresses().sram_start), + 2 => SyscallReturn::SuccessUsize(process.get_addresses().sram_start), // Op Type 3: Process memory end - 3 => SyscallReturn::SuccessUSize(process.get_addresses().sram_end), + 3 => SyscallReturn::SuccessUsize(process.get_addresses().sram_end), // Op Type 4: Process flash start - 4 => SyscallReturn::SuccessUSize(process.get_addresses().flash_start), + 4 => SyscallReturn::SuccessUsize(process.get_addresses().flash_start), // Op Type 5: Process flash end - 5 => SyscallReturn::SuccessUSize(process.get_addresses().flash_end), + 5 => SyscallReturn::SuccessUsize(process.get_addresses().flash_end), // Op Type 6: Grant region begin - 6 => SyscallReturn::SuccessUSize(process.get_addresses().sram_grant_start), + 6 => SyscallReturn::SuccessUsize(process.get_addresses().sram_grant_start), // Op Type 7: Number of defined writeable regions in the TBF header. - 7 => SyscallReturn::SuccessUSize(process.number_writeable_flash_regions()), + 7 => SyscallReturn::SuccessUsize(process.number_writeable_flash_regions()), // Op Type 8: The start address of the writeable region indexed by r1. 8 => { @@ -80,7 +80,7 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessUSize(flash_start + offset) + SyscallReturn::SuccessUsize(flash_start + offset) } } @@ -93,7 +93,7 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessUSize(flash_start + offset + size) + SyscallReturn::SuccessUsize(flash_start + offset + size) } } diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 0434a14856..82d5f0b1d5 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -398,12 +398,38 @@ pub enum SyscallReturnVariant { FailureU32 = 1, FailureU32U32 = 2, FailureU64 = 3, + FailurePtrUsize = 4, + FailurePtrPtr = 5, Success = 128, SuccessU32 = 129, SuccessU32U32 = 130, SuccessU64 = 131, SuccessU32U32U32 = 132, SuccessU32U64 = 133, + SuccessUsize = 134, + SucessPtr = 135, + SucessPtrUsize = 136, + SucessPtrPtr = 137, +} + +impl SyscallReturnVariant { + /// Maps newly introduced return variants (usize and ptr) + /// to old ones (u32) for backwards compatibility. + pub const fn into_compat(self) -> Self { + // We only need to be backwards compatible on 32-bit systems + let compat = core::mem::size_of::() == core::mem::size_of::(); + if compat { + match self { + // Map all usizes and ptrs to u32 + Self::SuccessUsize | Self::SucessPtr => Self::SuccessU32, + Self::SucessPtrUsize | Self::SucessPtrPtr => Self::SuccessU32U32, + Self::FailurePtrUsize | Self::FailurePtrPtr => Self::FailureU32U32, + x => x, + } + } else { + self + } + } } /// Enumeration of the possible system call return variants specified in TRD104. @@ -441,6 +467,9 @@ pub enum SyscallReturn { /// Generic success case, with an additional 32-bit and 64-bit data field SuccessU32U64(u32, u64), + /// Generic success case, with an additional usize data field + SuccessUsize(usize), + /// Generic success case, with an additional pointer with metadata /// On CHERI, this grants authority. /// Access to this return is therefore privileged. @@ -512,18 +541,6 @@ impl SyscallReturn { res.into_inner() } - /// Construct either SuccessU32 or SuccessU64 depending on platform. - #[allow(non_snake_case)] - pub(crate) fn SuccessUSize(val: usize) -> Self { - if core::mem::size_of::() == 8 { - SyscallReturn::SuccessU64(val as u64) - } else if core::mem::size_of::() == 4 { - SyscallReturn::SuccessU32(val as u32) - } else { - panic!(); - } - } - /// Returns true if the [`SyscallReturn`] is any success type. pub(crate) fn is_success(&self) -> bool { match self { @@ -547,6 +564,7 @@ impl SyscallReturn { SyscallReturn::AllowReadOnlyFailure(_, _, _) => false, SyscallReturn::SubscribeFailure(_, _, _) => false, SyscallReturn::YieldWaitFor(_, _, _) => true, + SyscallReturn::SuccessUsize(_) => true, } } @@ -663,7 +681,7 @@ impl SyscallReturn { write_64(a2, a3, data1); } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::Success as usize).into(); + *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -673,7 +691,7 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::Success as usize).into(); + *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -683,7 +701,7 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); *a2 = MetaPtr::new_with_metadata( ptr as *const (), @@ -694,7 +712,7 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); *a2 = MetaPtr::new_with_metadata( ptr as *const (), @@ -705,7 +723,7 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::Success as usize).into(); + *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -715,7 +733,7 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); *a2 = MetaPtr::new_with_metadata( ptr as *const (), @@ -726,18 +744,18 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = (SyscallReturnVariant::Success as usize).into(); + *a0 = (SyscallReturnVariant::SucessPtrPtr.into_compat() as usize).into(); *a1 = (ptr as usize).into(); *a2 = data.into(); } SyscallReturn::SubscribeFailure(err, ptr, data) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrPtr.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); *a2 = (ptr as usize).into(); *a3 = data.into(); } SyscallReturn::SuccessPtr(metaptr) => { - *a0 = (SyscallReturnVariant::Success as usize).into(); + *a0 = (SyscallReturnVariant::SucessPtr.into_compat() as usize).into(); *a1 = metaptr; } SyscallReturn::YieldWaitFor(data0, data1, data2) => { @@ -745,6 +763,10 @@ impl SyscallReturn { *a1 = data1.into(); *a2 = data2.into(); } + SyscallReturn::SuccessUsize(data) => { + *a0 = (SyscallReturnVariant::SuccessUsize.into_compat() as usize).into(); + *a1 = data.into(); + } } } } From a5ee39681cd26f4c527bab5b2be6912a3ec4fff6 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 14 Oct 2024 21:32:22 +0000 Subject: [PATCH 05/45] Better comment for MetaPtr Change-Id: Iccb152457179f1a48eb110c4e4eb7c2efc19150d --- kernel/src/metaptr.rs | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/kernel/src/metaptr.rs b/kernel/src/metaptr.rs index 28eb55a850..bedcb5fb50 100644 --- a/kernel/src/metaptr.rs +++ b/kernel/src/metaptr.rs @@ -7,9 +7,26 @@ use core::fmt::{Formatter, LowerHex, UpperHex}; use core::ops::AddAssign; -/// A pointer with target specific metadata. +/// A pointer with target specific metadata concerning validity or access rights. +/// /// This should be used any time the kernel wishes to grant authority to the user, or any time /// the user should be required to prove validity of a pointer. +/// +/// Values that are just raw addresses but imply nothing about a rust object at that location +/// should be `usize`. +/// Values that are references, but do not cross the boundary between the user and the +/// kernel (or do cross the boundary but are merely informative and do not imply any rights) +/// can be `*const T` (or `&T` if the kernel knows they are valid). +/// Values that are references, and do need to cross the boundary, should be this type. +/// +/// For example, allow is meant to grant authority to the kernel to access a buffer, so is `MetaPtr`. +/// When the user tells the kernel the location of its stack (for debug diagnostics) it need not +/// be `MetaPtr` as the kernel is not making any access. +/// +/// `MetaPtr` is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or +/// A `usize`, possibly podding with extra bits. It is therefore an appropriate choice for the type +/// of a register that may contain any one of these in the syscall ABI at a point where it is not +/// yet clear which of these it is yet. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] #[repr(transparent)] pub struct MetaPtr { From fd510980aed34ee030933295ac3018e56b9f7a43 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 14 Oct 2024 23:13:53 +0000 Subject: [PATCH 06/45] Fix typo Change-Id: I4fdbed879d85013c7e6f0e0157dcba7f92fb943b --- doc/wg/core/notes/core-notes-2022-03-04.md | 2 +- kernel/src/syscall.rs | 20 ++++++++++---------- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/doc/wg/core/notes/core-notes-2022-03-04.md b/doc/wg/core/notes/core-notes-2022-03-04.md index 65c28df938..ab6d6bd3bb 100644 --- a/doc/wg/core/notes/core-notes-2022-03-04.md +++ b/doc/wg/core/notes/core-notes-2022-03-04.md @@ -53,7 +53,7 @@ Attendees: * Hudson: Does anyone else have any context about the want for using `usize`-sized return values? Jett posted issue #2981 -- he wants to have - `FailureUsize` and `SuccessUsize` return variants. + `FailureUsize` and `SUCCESS_USIZE` return variants. * Alyssa: I can give a bit of background on this. Essentially, it is because of host emulation, where `usize` is 64 bits. You can't really return pointers from Command anymore. [Editor's note: Alyssa's last sentence was hard to hear] diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 82d5f0b1d5..bcffc844b6 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -407,9 +407,9 @@ pub enum SyscallReturnVariant { SuccessU32U32U32 = 132, SuccessU32U64 = 133, SuccessUsize = 134, - SucessPtr = 135, - SucessPtrUsize = 136, - SucessPtrPtr = 137, + SuccessPtr = 135, + SuccessPtrUsize = 136, + SuccessPtrPtr = 137, } impl SyscallReturnVariant { @@ -421,8 +421,8 @@ impl SyscallReturnVariant { if compat { match self { // Map all usizes and ptrs to u32 - Self::SuccessUsize | Self::SucessPtr => Self::SuccessU32, - Self::SucessPtrUsize | Self::SucessPtrPtr => Self::SuccessU32U32, + Self::SuccessUsize | Self::SuccessPtr => Self::SuccessU32, + Self::SuccessPtrUsize | Self::SuccessPtrPtr => Self::SuccessU32U32, Self::FailurePtrUsize | Self::FailurePtrPtr => Self::FailureU32U32, x => x, } @@ -681,7 +681,7 @@ impl SyscallReturn { write_64(a2, a3, data1); } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -691,7 +691,7 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -723,7 +723,7 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SucessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); *a1 = MetaPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -744,7 +744,7 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = (SyscallReturnVariant::SucessPtrPtr.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrPtr.into_compat() as usize).into(); *a1 = (ptr as usize).into(); *a2 = data.into(); } @@ -755,7 +755,7 @@ impl SyscallReturn { *a3 = data.into(); } SyscallReturn::SuccessPtr(metaptr) => { - *a0 = (SyscallReturnVariant::SucessPtr.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtr.into_compat() as usize).into(); *a1 = metaptr; } SyscallReturn::YieldWaitFor(data0, data1, data2) => { From c97deeac20a01cb33cf71152f5c512fc09d53df8 Mon Sep 17 00:00:00 2001 From: LawrenceEsswood Date: Mon, 21 Oct 2024 18:35:48 +0100 Subject: [PATCH 07/45] Apply comment suggestions Co-authored-by: Amit Levy --- kernel/src/metaptr.rs | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/kernel/src/metaptr.rs b/kernel/src/metaptr.rs index bedcb5fb50..b5d6bdeadb 100644 --- a/kernel/src/metaptr.rs +++ b/kernel/src/metaptr.rs @@ -12,18 +12,18 @@ use core::ops::AddAssign; /// This should be used any time the kernel wishes to grant authority to the user, or any time /// the user should be required to prove validity of a pointer. /// -/// Values that are just raw addresses but imply nothing about a rust object at that location +/// Values that are just raw addresses but imply nothing about a Rust object at that location /// should be `usize`. /// Values that are references, but do not cross the boundary between the user and the /// kernel (or do cross the boundary but are merely informative and do not imply any rights) /// can be `*const T` (or `&T` if the kernel knows they are valid). /// Values that are references, and do need to cross the boundary, should be this type. /// -/// For example, allow is meant to grant authority to the kernel to access a buffer, so is `MetaPtr`. -/// When the user tells the kernel the location of its stack (for debug diagnostics) it need not -/// be `MetaPtr` as the kernel is not making any access. +/// For example, `allow` grants authority to the kernel to access a buffer, so passes [MetaPtr]s. +/// Conversely, when a process communicates its stack location to the kernel it need not be +/// passed as a [MetaPtr], as the kernel does not access it. /// -/// `MetaPtr` is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or +/// [MetaPtr] is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or /// A `usize`, possibly podding with extra bits. It is therefore an appropriate choice for the type /// of a register that may contain any one of these in the syscall ABI at a point where it is not /// yet clear which of these it is yet. From 291c8f403b6c1705a3a4cd848c2d7de8826b6e3a Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Tue, 29 Oct 2024 23:10:38 +0000 Subject: [PATCH 08/45] Rename MetaPtr -> CapabilityPtr Change-Id: I0b59feb2d1d6f60be4aaa46adc430785651837ca --- kernel/src/{metaptr.rs => capability_ptr.rs} | 26 +++++----- kernel/src/lib.rs | 2 +- kernel/src/process.rs | 10 ++-- kernel/src/process_standard.rs | 19 +++---- kernel/src/syscall.rs | 54 ++++++++++---------- kernel/src/upcall.rs | 6 +-- 6 files changed, 59 insertions(+), 58 deletions(-) rename kernel/src/{metaptr.rs => capability_ptr.rs} (85%) diff --git a/kernel/src/metaptr.rs b/kernel/src/capability_ptr.rs similarity index 85% rename from kernel/src/metaptr.rs rename to kernel/src/capability_ptr.rs index b5d6bdeadb..f4f5086ac1 100644 --- a/kernel/src/metaptr.rs +++ b/kernel/src/capability_ptr.rs @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 OR MIT // Copyright Google LLC 2024. -//! Defines the MetaPtr type +//! Defines the CapabilityPtr type use core::fmt::{Formatter, LowerHex, UpperHex}; use core::ops::AddAssign; @@ -19,21 +19,21 @@ use core::ops::AddAssign; /// can be `*const T` (or `&T` if the kernel knows they are valid). /// Values that are references, and do need to cross the boundary, should be this type. /// -/// For example, `allow` grants authority to the kernel to access a buffer, so passes [MetaPtr]s. +/// For example, `allow` grants authority to the kernel to access a buffer, so passes [CapabilityPtr]s. /// Conversely, when a process communicates its stack location to the kernel it need not be -/// passed as a [MetaPtr], as the kernel does not access it. +/// passed as a [CapabilityPtr], as the kernel does not access it. /// -/// [MetaPtr] is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or +/// [CapabilityPtr] is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or /// A `usize`, possibly podding with extra bits. It is therefore an appropriate choice for the type /// of a register that may contain any one of these in the syscall ABI at a point where it is not /// yet clear which of these it is yet. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] #[repr(transparent)] -pub struct MetaPtr { +pub struct CapabilityPtr { ptr: *const (), } -impl Default for MetaPtr { +impl Default for CapabilityPtr { fn default() -> Self { Self { ptr: core::ptr::null(), @@ -50,14 +50,14 @@ pub enum MetaPermissions { Execute, } -impl From for usize { +impl From for usize { #[inline] - fn from(from: MetaPtr) -> Self { + fn from(from: CapabilityPtr) -> Self { from.ptr as usize } } -impl From for MetaPtr { +impl From for CapabilityPtr { #[inline] fn from(from: usize) -> Self { Self { @@ -66,28 +66,28 @@ impl From for MetaPtr { } } -impl UpperHex for MetaPtr { +impl UpperHex for CapabilityPtr { #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { UpperHex::fmt(&(self.ptr as usize), f) } } -impl LowerHex for MetaPtr { +impl LowerHex for CapabilityPtr { #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { LowerHex::fmt(&(self.ptr as usize), f) } } -impl AddAssign for MetaPtr { +impl AddAssign for CapabilityPtr { #[inline] fn add_assign(&mut self, rhs: usize) { self.ptr = (self.ptr as *const u8).wrapping_add(rhs) as *const (); } } -impl MetaPtr { +impl CapabilityPtr { pub fn as_ptr(&self) -> *const T { self.ptr as *const T } diff --git a/kernel/src/lib.rs b/kernel/src/lib.rs index a1d797c762..412d00432d 100644 --- a/kernel/src/lib.rs +++ b/kernel/src/lib.rs @@ -105,6 +105,7 @@ pub const KERNEL_MAJOR_VERSION: u16 = 2; pub const KERNEL_MINOR_VERSION: u16 = 1; pub mod capabilities; +pub mod capability_ptr; pub mod collections; pub mod component; pub mod debug; @@ -114,7 +115,6 @@ pub mod grant; pub mod hil; pub mod introspection; pub mod ipc; -pub mod metaptr; pub mod platform; pub mod process; pub mod process_checker; diff --git a/kernel/src/process.rs b/kernel/src/process.rs index ccd16383d3..29709660f1 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -11,10 +11,10 @@ use core::ptr::NonNull; use core::str; use crate::capabilities; +use crate::capability_ptr::CapabilityPtr; use crate::errorcode::ErrorCode; use crate::ipc; use crate::kernel::Kernel; -use crate::metaptr::MetaPtr; use crate::platform::mpu::{self}; use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions; @@ -540,7 +540,7 @@ pub trait Process { /// process's memory region. /// - [`Error::KernelError`] if there was an internal kernel error. This is /// a bug. - fn brk(&self, new_break: *const u8) -> Result; + fn brk(&self, new_break: *const u8) -> Result; /// Change the location of the program break by `increment` bytes, /// reallocate the MPU region covering program memory, and return the @@ -560,7 +560,7 @@ pub trait Process { /// process's memory region. /// - [`Error::KernelError`] if there was an internal kernel error. This is /// a bug. - fn sbrk(&self, increment: isize) -> Result; + fn sbrk(&self, increment: isize) -> Result; /// How many writeable flash regions defined in the TBF header for this /// process. @@ -1081,9 +1081,9 @@ pub struct FunctionCall { /// The third argument to the function. pub argument2: usize, /// The fourth argument to the function. - pub argument3: MetaPtr, + pub argument3: CapabilityPtr, /// The PC of the function to execute. - pub pc: MetaPtr, + pub pc: CapabilityPtr, } /// This is similar to `FunctionCall` but for the special case of the Null diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index ce52f54bb7..92da6c3ac0 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -14,14 +14,14 @@ use core::num::NonZeroU32; use core::ptr::NonNull; use core::{mem, ptr, slice, str}; +use crate::capability_ptr::MetaPermissions::Execute; +use crate::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::collections::queue::Queue; use crate::collections::ring_buffer::RingBuffer; use crate::config; use crate::debug; use crate::errorcode::ErrorCode; use crate::kernel::Kernel; -use crate::metaptr::MetaPermissions::Execute; -use crate::metaptr::{MetaPermissions, MetaPtr}; use crate::platform::chip::Chip; use crate::platform::mpu::{self, MPU}; use crate::process::BinaryVersion; @@ -584,7 +584,7 @@ impl Process for ProcessStandard<'_, C> { }) } - fn sbrk(&self, increment: isize) -> Result { + fn sbrk(&self, increment: isize) -> Result { // Do not modify an inactive process. if !self.is_running() { return Err(Error::InactiveApp); @@ -594,7 +594,7 @@ impl Process for ProcessStandard<'_, C> { self.brk(new_break) } - fn brk(&self, new_break: *const u8) -> Result { + fn brk(&self, new_break: *const u8) -> Result { // Do not modify an inactive process. if !self.is_running() { return Err(Error::InactiveApp); @@ -618,7 +618,7 @@ impl Process for ProcessStandard<'_, C> { self.chip.mpu().configure_mpu(config); let base = self.mem_start() as usize; - let break_result = MetaPtr::new_with_metadata( + let break_result = CapabilityPtr::new_with_metadata( old_break as *const (), base, (new_break as usize) - base, @@ -1754,7 +1754,8 @@ impl ProcessStandard<'_, C> { flash_start.wrapping_add(process.header.get_init_function_offset() as usize) as usize; let fn_base = flash_start as usize; let fn_len = process.flash.len(); - let init_fn = MetaPtr::new_with_metadata(init_addr as *const (), fn_base, fn_len, Execute); + let init_fn = + CapabilityPtr::new_with_metadata(init_addr as *const (), fn_base, fn_len, Execute); process.tasks.map(|tasks| { tasks.enqueue(Task::FunctionCall(FunctionCall { @@ -1919,7 +1920,7 @@ impl ProcessStandard<'_, C> { let init_addr = flash_start.wrapping_add(self.header.get_init_function_offset() as usize) as usize; - let init_fn = MetaPtr::new_with_metadata( + let init_fn = CapabilityPtr::new_with_metadata( init_addr as *const (), flash_start as usize, (self.flash_end() as usize) - (flash_start as usize), @@ -1944,7 +1945,7 @@ impl ProcessStandard<'_, C> { fn in_app_owned_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { // TODO: On CHERI platforms, it impossible to form syscalls with pointers // that are not in app memory. However, buf_start_addr is not the right - // type to make this function always return true. If MetaPtr makes it + // type to make this function always return true. If CapabilityPtr makes it // slightly further, we can skip this check. let buf_end_addr = buf_start_addr.wrapping_add(size); @@ -1960,7 +1961,7 @@ impl ProcessStandard<'_, C> { fn in_app_flash_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { // TODO: On CHERI platforms, it impossible to form syscalls with pointers // that are not in app memory. However, buf_start_addr is not the right - // type to make this function always return true. If MetaPtr makes it + // type to make this function always return true. If CapabilityPtr makes it // slightly further, we can skip this check. let buf_end_addr = buf_start_addr.wrapping_add(size); diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index bcffc844b6..14ab78c1e1 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -68,8 +68,8 @@ use core::fmt::Write; +use crate::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::errorcode::ErrorCode; -use crate::metaptr::{MetaPermissions, MetaPtr}; use crate::process; pub use crate::syscall_driver::{CommandReturn, SyscallDriver}; @@ -170,9 +170,9 @@ pub enum Syscall { // On CHERI platforms we need to maintain metadata for // these two as they are used to access code/data. /// Upcall pointer to the upcall function. - upcall_ptr: MetaPtr, + upcall_ptr: CapabilityPtr, /// Userspace application data. - appdata: MetaPtr, + appdata: CapabilityPtr, }, /// Structure representing an invocation of the Command system call class. @@ -255,9 +255,9 @@ impl Syscall { pub fn from_register_arguments( syscall_number: u8, r0: usize, - r1: MetaPtr, - r2: MetaPtr, - r3: MetaPtr, + r1: CapabilityPtr, + r2: CapabilityPtr, + r3: CapabilityPtr, ) -> Option { match SyscallClass::try_from(syscall_number) { Ok(SyscallClass::Yield) => Some(Syscall::Yield { @@ -473,7 +473,7 @@ pub enum SyscallReturn { /// Generic success case, with an additional pointer with metadata /// On CHERI, this grants authority. /// Access to this return is therefore privileged. - SuccessPtr(MetaPtr), + SuccessPtr(CapabilityPtr), // These following types are used by the scheduler so that it can return // values to userspace in an architecture (pointer-width) independent way. @@ -573,17 +573,17 @@ impl SyscallReturn { /// are free to define their own encoding. /// TODO: deprecate in favour of the more general one pub fn encode_syscall_return(&self, a0: &mut u32, a1: &mut u32, a2: &mut u32, a3: &mut u32) { - if core::mem::size_of::() == core::mem::size_of::() { + if core::mem::size_of::() == core::mem::size_of::() { // SAFETY: if the two unsized integers are the same size references to them // can be safely transmuted. // Ugly coercion could be avoided by first copying to the stack, then assigning with // "as" in order to satisfy the compiler. But I expect this function will disappear // in favour of just using the usize one. unsafe { - let a0 = &mut *(core::ptr::from_mut(a0) as *mut MetaPtr); - let a1 = &mut *(core::ptr::from_mut(a1) as *mut MetaPtr); - let a2 = &mut *(core::ptr::from_mut(a2) as *mut MetaPtr); - let a3 = &mut *(core::ptr::from_mut(a3) as *mut MetaPtr); + let a0 = &mut *(core::ptr::from_mut(a0) as *mut CapabilityPtr); + let a1 = &mut *(core::ptr::from_mut(a1) as *mut CapabilityPtr); + let a2 = &mut *(core::ptr::from_mut(a2) as *mut CapabilityPtr); + let a3 = &mut *(core::ptr::from_mut(a3) as *mut CapabilityPtr); self.encode_syscall_return_mptr(a0, a1, a2, a3); } } else { @@ -607,21 +607,21 @@ impl SyscallReturn { /// of user code. /// I have not considered usize other than 4 and 8 bytes. /// Also handles the CHERI extension as follows: - /// the high part of any MetaPtr register is zero'd if any non capability-sized arguments are + /// the high part of any CapabilityPtr register is zero'd if any non capability-sized arguments are /// passed. - /// SuccessPtr is as passed the full MetaPtr register. + /// SuccessPtr is as passed the full CapabilityPtr register. /// Pointers from allow'd buffers have minimal bounds attached that cover their length, /// and the same permissions that were checked at the syscall boundary. pub fn encode_syscall_return_mptr( &self, - a0: &mut MetaPtr, - a1: &mut MetaPtr, - a2: &mut MetaPtr, - a3: &mut MetaPtr, + a0: &mut CapabilityPtr, + a1: &mut CapabilityPtr, + a2: &mut CapabilityPtr, + a3: &mut CapabilityPtr, ) { // On 32-bit CHERI, given that capabilities cannot be used as 64-bit integers, 64-bit // integers will still be returned as two 32-bit values in different registers. - fn write_64(a: &mut MetaPtr, b: &mut MetaPtr, val: u64) { + fn write_64(a: &mut CapabilityPtr, b: &mut CapabilityPtr, val: u64) { let is_64_bit = core::mem::size_of::() == 8; if !is_64_bit { let (msb, lsb) = u64_to_be_u32s(val); @@ -682,7 +682,7 @@ impl SyscallReturn { } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); - *a1 = MetaPtr::new_with_metadata( + *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -692,7 +692,7 @@ impl SyscallReturn { } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); - *a1 = MetaPtr::new_with_metadata( + *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -703,7 +703,7 @@ impl SyscallReturn { SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = MetaPtr::new_with_metadata( + *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -714,7 +714,7 @@ impl SyscallReturn { SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = MetaPtr::new_with_metadata( + *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -724,7 +724,7 @@ impl SyscallReturn { } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); - *a1 = MetaPtr::new_with_metadata( + *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -735,7 +735,7 @@ impl SyscallReturn { SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = MetaPtr::new_with_metadata( + *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, len, @@ -754,9 +754,9 @@ impl SyscallReturn { *a2 = (ptr as usize).into(); *a3 = data.into(); } - SyscallReturn::SuccessPtr(metaptr) => { + SyscallReturn::SuccessPtr(ptr) => { *a0 = (SyscallReturnVariant::SuccessPtr.into_compat() as usize).into(); - *a1 = metaptr; + *a1 = ptr; } SyscallReturn::YieldWaitFor(data0, data1, data2) => { *a0 = data0.into(); diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index 51b6661c85..a41fc467de 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -4,9 +4,9 @@ //! Data structure for storing an upcall from the kernel to a process. +use crate::capability_ptr::CapabilityPtr; use crate::config; use crate::debug; -use crate::metaptr::MetaPtr; use crate::process; use crate::process::ProcessId; use crate::syscall::SyscallReturn; @@ -72,8 +72,8 @@ pub enum UpcallError { // FIXME: When we get CHERI compiler support, these can go back to the proper types // b/274586199 // https://github.com/tock/tock/issues/4134 -pub(crate) type AppdataType = MetaPtr; -pub(crate) type FnPtrType = MetaPtr; +pub(crate) type AppdataType = CapabilityPtr; +pub(crate) type FnPtrType = CapabilityPtr; /// Type for calling an upcall in a process. /// From 695af1b02fbacccfba5bbc8ab938b44f01b2fa0f Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Tue, 29 Oct 2024 23:13:21 +0000 Subject: [PATCH 09/45] Add comment saying into_compat is for legacy only Change-Id: Iccd7ff9f675786cf04abfa16ce89e8c3667a756d --- kernel/src/syscall.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 14ab78c1e1..ee46e81ebe 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -415,6 +415,8 @@ pub enum SyscallReturnVariant { impl SyscallReturnVariant { /// Maps newly introduced return variants (usize and ptr) /// to old ones (u32) for backwards compatibility. + /// This should not be used for any newly designed interfaces, + /// and will eventually be deprecated once all interfaces are updated. pub const fn into_compat(self) -> Self { // We only need to be backwards compatible on 32-bit systems let compat = core::mem::size_of::() == core::mem::size_of::(); From 7ec6bdd51152cc054c2f68dcab3425928ad11876 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Mon, 4 Nov 2024 14:35:04 -0800 Subject: [PATCH 10/45] capability_ptr: update doc to reflect meaning --- kernel/src/capability_ptr.rs | 31 ++++++++++++++----------------- kernel/src/process.rs | 2 +- 2 files changed, 15 insertions(+), 18 deletions(-) diff --git a/kernel/src/capability_ptr.rs b/kernel/src/capability_ptr.rs index f4f5086ac1..99df89de8d 100644 --- a/kernel/src/capability_ptr.rs +++ b/kernel/src/capability_ptr.rs @@ -7,26 +7,23 @@ use core::fmt::{Formatter, LowerHex, UpperHex}; use core::ops::AddAssign; -/// A pointer with target specific metadata concerning validity or access rights. +/// A pointer to userspace memory. /// -/// This should be used any time the kernel wishes to grant authority to the user, or any time -/// the user should be required to prove validity of a pointer. +/// A [CapabilityPtr] points to memory a userspace processes may be +/// permitted to read, write, or execute. It is sized exacly to a +/// register that can pass values between userspace and kernel and at +/// least the size of a word ([usize]) [^note1]. Operations on the +/// pointer may affect permissions, e.g. offsetting the pointer beyond +/// the bounds of the memory object invalidates it. Like a `*const +/// ()`, a [CapabilityPtr] may also "hide" information by storing a +/// word of data with no memory access permissions. /// -/// Values that are just raw addresses but imply nothing about a Rust object at that location -/// should be `usize`. -/// Values that are references, but do not cross the boundary between the user and the -/// kernel (or do cross the boundary but are merely informative and do not imply any rights) -/// can be `*const T` (or `&T` if the kernel knows they are valid). -/// Values that are references, and do need to cross the boundary, should be this type. +/// [CapabilityPtr] should be used to store or pass between kernel and +/// userspace a value that may represent a valid userspace reference. /// -/// For example, `allow` grants authority to the kernel to access a buffer, so passes [CapabilityPtr]s. -/// Conversely, when a process communicates its stack location to the kernel it need not be -/// passed as a [CapabilityPtr], as the kernel does not access it. -/// -/// [CapabilityPtr] is also assumed to be wide enough that it could contain a raw pointer (`*const ()`) or -/// A `usize`, possibly podding with extra bits. It is therefore an appropriate choice for the type -/// of a register that may contain any one of these in the syscall ABI at a point where it is not -/// yet clear which of these it is yet. +/// [^note1]: Depending on the architecutre, this the size of a +/// [CapabilityPtr] may be a word size or larger, e.g., if registers +/// can store metadata such as access permissions. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] #[repr(transparent)] pub struct CapabilityPtr { diff --git a/kernel/src/process.rs b/kernel/src/process.rs index 29709660f1..fa87a6d9f6 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -792,7 +792,7 @@ pub trait Process { /// Returns `true` if the upcall function pointer is valid for this process, /// and `false` otherwise. /// upcall_fn can eventually be a better type: - /// https://github.com/tock/tock/issues/4134 + /// fn is_valid_upcall_function_pointer(&self, upcall_fn: *const ()) -> bool; // functions for processes that are architecture specific From d0fa5fc2a9a9b5cd80dbf409cb44704fac8b548d Mon Sep 17 00:00:00 2001 From: LawrenceEsswood Date: Fri, 8 Nov 2024 20:15:18 +0000 Subject: [PATCH 11/45] Update kernel/src/syscall.rs Co-authored-by: Amit Levy --- kernel/src/syscall.rs | 2 -- 1 file changed, 2 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index ee46e81ebe..f5445169ba 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -167,8 +167,6 @@ pub enum Syscall { driver_number: usize, /// The subscribe identifier. subdriver_number: usize, - // On CHERI platforms we need to maintain metadata for - // these two as they are used to access code/data. /// Upcall pointer to the upcall function. upcall_ptr: CapabilityPtr, /// Userspace application data. From be62a66112feeaa793462016675c305636fefd45 Mon Sep 17 00:00:00 2001 From: LawrenceEsswood Date: Fri, 8 Nov 2024 20:15:30 +0000 Subject: [PATCH 12/45] Update kernel/src/process.rs Co-authored-by: Amit Levy --- kernel/src/process.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kernel/src/process.rs b/kernel/src/process.rs index fa87a6d9f6..e5da5f0ccc 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -1080,7 +1080,7 @@ pub struct FunctionCall { pub argument1: usize, /// The third argument to the function. pub argument2: usize, - /// The fourth argument to the function. + /// The userdata provided by the process via `subscribe` pub argument3: CapabilityPtr, /// The PC of the function to execute. pub pc: CapabilityPtr, From 950deb207ecaabb565c71bca533940e662fdee3c Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:24:42 +0000 Subject: [PATCH 13/45] Revert notes change Change-Id: Iab143ff2364b3bafa801db8a0b3b653e57188388 --- doc/wg/core/notes/core-notes-2022-03-04.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/doc/wg/core/notes/core-notes-2022-03-04.md b/doc/wg/core/notes/core-notes-2022-03-04.md index ab6d6bd3bb..65c28df938 100644 --- a/doc/wg/core/notes/core-notes-2022-03-04.md +++ b/doc/wg/core/notes/core-notes-2022-03-04.md @@ -53,7 +53,7 @@ Attendees: * Hudson: Does anyone else have any context about the want for using `usize`-sized return values? Jett posted issue #2981 -- he wants to have - `FailureUsize` and `SUCCESS_USIZE` return variants. + `FailureUsize` and `SuccessUsize` return variants. * Alyssa: I can give a bit of background on this. Essentially, it is because of host emulation, where `usize` is 64 bits. You can't really return pointers from Command anymore. [Editor's note: Alyssa's last sentence was hard to hear] From 236a8ac00158c9e367cf2d7d4ee88fd49adc201c Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:41:42 +0000 Subject: [PATCH 14/45] Remove extra cast step Change-Id: I7a8a40899a3d26e93fab3ea3275d27cc3595e666 --- arch/rv32i/src/syscall.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index 7581a07d15..88f692780a 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -206,7 +206,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { state.regs[R_RA] = state.pc; // Save the PC we expect to execute. - state.pc = callback.pc.as_ptr::<()>() as usize as u32; + state.pc = usize::from(callback.pc) as u32; Ok(()) } From a4e593675d437d3686da1d45bd37d544fabcef69 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:42:16 +0000 Subject: [PATCH 15/45] Preface internal Google bug tracker Change-Id: I46cfdb257dd2b8bd4a15610c57fd4d8225d013b6 --- kernel/src/upcall.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index a41fc467de..7f6b9b0a1f 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -70,7 +70,7 @@ pub enum UpcallError { } // FIXME: When we get CHERI compiler support, these can go back to the proper types -// b/274586199 +// Google-internal issue: b/274586199 // https://github.com/tock/tock/issues/4134 pub(crate) type AppdataType = CapabilityPtr; pub(crate) type FnPtrType = CapabilityPtr; From 9ac666bb625d7b3c6a8c3dc7eb9a2f66e88afa3f Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:43:33 +0000 Subject: [PATCH 16/45] Remove as_checked_ptr until CHERI lands Change-Id: Icafa29180ff1f6ad89673899ef27df2994c47c57 --- kernel/src/capability_ptr.rs | 9 --------- kernel/src/syscall.rs | 7 +++---- 2 files changed, 3 insertions(+), 13 deletions(-) diff --git a/kernel/src/capability_ptr.rs b/kernel/src/capability_ptr.rs index 99df89de8d..115914ac2c 100644 --- a/kernel/src/capability_ptr.rs +++ b/kernel/src/capability_ptr.rs @@ -89,15 +89,6 @@ impl CapabilityPtr { self.ptr as *const T } - /// Convert to a raw pointer, checking that metadata allows a particular set of permissions over - /// a given number of bytes. - /// If the metadata does not allow for this, returns null. - /// If no such metadata exists, this succeeds. - #[inline] - pub fn as_ptr_checked(&self, _length: usize, _perms: MetaPermissions) -> *const T { - self.ptr as *const T - } - #[inline] pub fn new_with_metadata( ptr: *const (), diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index f5445169ba..78fd12e23c 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -278,20 +278,19 @@ impl Syscall { Ok(SyscallClass::ReadWriteAllow) => Some(Syscall::ReadWriteAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::ReadWrite) - as *mut u8, + allow_address: r2.as_ptr::() as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::UserspaceReadableAllow) => Some(Syscall::UserspaceReadableAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::Read) as *mut u8, + allow_address: r2.as_ptr::() as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::ReadOnlyAllow) => Some(Syscall::ReadOnlyAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr_checked::(r3.into(), MetaPermissions::Read) as *mut u8, + allow_address: r2.as_ptr::() as *mut u8, allow_size: r3.into(), }), Ok(SyscallClass::Memop) => Some(Syscall::Memop { From 010bf2244f1561112578295c9399b4823e950905 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:44:00 +0000 Subject: [PATCH 17/45] Change panic for assert and checked alignment too Change-Id: I8b9b74afd900fdbf1d2c192f1ef3af3bc004ed5d --- kernel/src/syscall.rs | 32 +++++++++++++++++--------------- 1 file changed, 17 insertions(+), 15 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 78fd12e23c..143d54f5b1 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -572,21 +572,23 @@ impl SyscallReturn { /// are free to define their own encoding. /// TODO: deprecate in favour of the more general one pub fn encode_syscall_return(&self, a0: &mut u32, a1: &mut u32, a2: &mut u32, a3: &mut u32) { - if core::mem::size_of::() == core::mem::size_of::() { - // SAFETY: if the two unsized integers are the same size references to them - // can be safely transmuted. - // Ugly coercion could be avoided by first copying to the stack, then assigning with - // "as" in order to satisfy the compiler. But I expect this function will disappear - // in favour of just using the usize one. - unsafe { - let a0 = &mut *(core::ptr::from_mut(a0) as *mut CapabilityPtr); - let a1 = &mut *(core::ptr::from_mut(a1) as *mut CapabilityPtr); - let a2 = &mut *(core::ptr::from_mut(a2) as *mut CapabilityPtr); - let a3 = &mut *(core::ptr::from_mut(a3) as *mut CapabilityPtr); - self.encode_syscall_return_mptr(a0, a1, a2, a3); - } - } else { - panic!("encode_syscall_return used on a 64-bit platform or CHERI platform") + assert!( + core::mem::size_of::() == core::mem::size_of::() + && core::mem::align_of::() >= align_of::(), + "encode_syscall_return used on a 64-bit platform or CHERI platform" + ); + + // SAFETY: if the two integers are the same size (and alignment permits) references + // to them can be safely transmuted. + // Ugly coercion could be avoided by first copying to the stack, then assigning with + // "as" in order to satisfy the compiler. But I expect this function will disappear + // in favour of just using the usize one. + unsafe { + let a0 = &mut *(core::ptr::from_mut(a0) as *mut CapabilityPtr); + let a1 = &mut *(core::ptr::from_mut(a1) as *mut CapabilityPtr); + let a2 = &mut *(core::ptr::from_mut(a2) as *mut CapabilityPtr); + let a3 = &mut *(core::ptr::from_mut(a3) as *mut CapabilityPtr); + self.encode_syscall_return_mptr(a0, a1, a2, a3); } } From 604460c247a5f0b83d10e11589d9ace912044b9e Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:44:53 +0000 Subject: [PATCH 18/45] Make the oddly chosen ANY permission a NONE permission Change-Id: Ifa4f132c75d0dbb78963f4990929adb3f25d48c1 --- kernel/src/capability_ptr.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kernel/src/capability_ptr.rs b/kernel/src/capability_ptr.rs index 115914ac2c..a3775dc2f7 100644 --- a/kernel/src/capability_ptr.rs +++ b/kernel/src/capability_ptr.rs @@ -40,7 +40,7 @@ impl Default for CapabilityPtr { #[derive(Copy, Clone, PartialEq)] pub enum MetaPermissions { - Any, + None, Read, Write, ReadWrite, From 79b2b8db8c102d3a3140d5db44afb110675f35f8 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 20:45:24 +0000 Subject: [PATCH 19/45] Add provenance notes Change-Id: Icd62c30f25da58ff83f3a4f59655920aefc805cb --- kernel/src/capability_ptr.rs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/kernel/src/capability_ptr.rs b/kernel/src/capability_ptr.rs index a3775dc2f7..e77f12dac5 100644 --- a/kernel/src/capability_ptr.rs +++ b/kernel/src/capability_ptr.rs @@ -48,6 +48,7 @@ pub enum MetaPermissions { } impl From for usize { + /// Provenance note: may not expose provenance #[inline] fn from(from: CapabilityPtr) -> Self { from.ptr as usize @@ -55,6 +56,7 @@ impl From for usize { } impl From for CapabilityPtr { + /// Provenance note: may have null provenance #[inline] fn from(from: usize) -> Self { Self { @@ -89,6 +91,8 @@ impl CapabilityPtr { self.ptr as *const T } + /// Provenance note: may derive from a pointer other than the input to provide something with + /// valid provenance to justify the other arguments. #[inline] pub fn new_with_metadata( ptr: *const (), From 6feb692df9fe08daa32cb12245e7ea5fa49bf28d Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 21:33:53 +0000 Subject: [PATCH 20/45] Refine CapabilityPtr description slightly Change-Id: Ifd89e7b62bb42404588f4559f51f038efc7cb577 --- kernel/src/capability_ptr.rs | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/kernel/src/capability_ptr.rs b/kernel/src/capability_ptr.rs index e77f12dac5..d0ed2a6853 100644 --- a/kernel/src/capability_ptr.rs +++ b/kernel/src/capability_ptr.rs @@ -7,10 +7,10 @@ use core::fmt::{Formatter, LowerHex, UpperHex}; use core::ops::AddAssign; -/// A pointer to userspace memory. +/// A pointer to userspace memory with implied authority. /// /// A [CapabilityPtr] points to memory a userspace processes may be -/// permitted to read, write, or execute. It is sized exacly to a +/// permitted to read, write, or execute. It is sized exactly to a /// register that can pass values between userspace and kernel and at /// least the size of a word ([usize]) [^note1]. Operations on the /// pointer may affect permissions, e.g. offsetting the pointer beyond @@ -18,10 +18,11 @@ use core::ops::AddAssign; /// ()`, a [CapabilityPtr] may also "hide" information by storing a /// word of data with no memory access permissions. /// -/// [CapabilityPtr] should be used to store or pass between kernel and -/// userspace a value that may represent a valid userspace reference. +/// [CapabilityPtr] should be used to store or pass between the kernel +/// and userspace a value that may represent a valid userspace reference, +/// when one party intends the other to access it. /// -/// [^note1]: Depending on the architecutre, this the size of a +/// [^note1]: Depending on the architecture, the size of a /// [CapabilityPtr] may be a word size or larger, e.g., if registers /// can store metadata such as access permissions. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] From f800ddd4a9f3571b1651c0ee19759d5b1ef81146 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 21:34:02 +0000 Subject: [PATCH 21/45] Minor changes to encode Change names to clearly indicate what they do and make the usize variant not available publicy until it is used. Also removes the CHERI comments that are not yet relevent. Change-Id: Ifa7ac0804ea984a1d0ede6274c075e7c5e84a70d --- arch/cortex-m/src/syscall.rs | 2 +- arch/rv32i/src/syscall.rs | 2 +- kernel/src/syscall.rs | 42 ++++++++++++++---------------------- 3 files changed, 18 insertions(+), 28 deletions(-) diff --git a/arch/cortex-m/src/syscall.rs b/arch/cortex-m/src/syscall.rs index f6b70689f9..7895a95c6d 100644 --- a/arch/cortex-m/src/syscall.rs +++ b/arch/cortex-m/src/syscall.rs @@ -195,7 +195,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall // // Refer to // https://doc.rust-lang.org/std/primitive.pointer.html#safety-13 - return_value.encode_syscall_return(&mut *r0, &mut *r1, &mut *r2, &mut *r3); + return_value.encode_syscall_return_32bit_trd104(&mut *r0, &mut *r1, &mut *r2, &mut *r3); Ok(()) } diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index 88f692780a..074fbdb5c9 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -172,7 +172,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { let (a1slice, r) = r.split_at_mut(R_A2 - R_A1); let (a2slice, a3slice) = r.split_at_mut(R_A3 - R_A2); - return_value.encode_syscall_return( + return_value.encode_syscall_return_32bit_trd104( &mut a0slice[0], &mut a1slice[0], &mut a2slice[0], diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 143d54f5b1..6fd150ea4b 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -570,8 +570,13 @@ impl SyscallReturn { /// Encode the system call return value into 4 registers, following the /// encoding specified in TRD104. Architectures which do not follow TRD104 /// are free to define their own encoding. - /// TODO: deprecate in favour of the more general one - pub fn encode_syscall_return(&self, a0: &mut u32, a1: &mut u32, a2: &mut u32, a3: &mut u32) { + pub fn encode_syscall_return_32bit_trd104( + &self, + a0: &mut u32, + a1: &mut u32, + a2: &mut u32, + a3: &mut u32, + ) { assert!( core::mem::size_of::() == core::mem::size_of::() && core::mem::align_of::() >= align_of::(), @@ -581,39 +586,24 @@ impl SyscallReturn { // SAFETY: if the two integers are the same size (and alignment permits) references // to them can be safely transmuted. // Ugly coercion could be avoided by first copying to the stack, then assigning with - // "as" in order to satisfy the compiler. But I expect this function will disappear - // in favour of just using the usize one. + // "as" in order to satisfy the compiler. unsafe { let a0 = &mut *(core::ptr::from_mut(a0) as *mut CapabilityPtr); let a1 = &mut *(core::ptr::from_mut(a1) as *mut CapabilityPtr); let a2 = &mut *(core::ptr::from_mut(a2) as *mut CapabilityPtr); let a3 = &mut *(core::ptr::from_mut(a3) as *mut CapabilityPtr); - self.encode_syscall_return_mptr(a0, a1, a2, a3); + self.encode_syscall_return_usize_trd104_compat(a0, a1, a2, a3); } } - /// The obvious extension of TRD104 that works for 32-bit and 64-bit platforms. - /// This makes no changes to TRD104 on 32-bit platforms. + /// An extension of TRD104 that works for 32-bit and 64-bit platforms. + /// This implements TRD104 exactly on 32-bit platforms by mapping new codes intended for + /// 64-bit platforms to existing ones. /// On 64-bit platforms, both 64-bit and usize values are passed as a single register, - /// shifting down register number if that means fewer registers are needed needed. - /// For usize, this is the obvious choice. - /// For explicitly 64-bit arguments, this would require rewriting prototypes for userspace - /// functions between 32 and 64 bit platforms. - /// However, no driver currently USES any 64-bit values. - /// Any new ones on 64-bit platforms would likely prefer just passing the value. - /// If they would not, I suspect many really want usize anyway (and that is what is used for - /// the syscalls handled directly by the kernel). Maybe they should be written in terms of that, - /// and some helpful aliases created for FailureUSIZE etc. - /// I think packing two 32-bit values into 64-bits would gain nothing and pollute a whole lot - /// of user code. - /// I have not considered usize other than 4 and 8 bytes. - /// Also handles the CHERI extension as follows: - /// the high part of any CapabilityPtr register is zero'd if any non capability-sized arguments are - /// passed. - /// SuccessPtr is as passed the full CapabilityPtr register. - /// Pointers from allow'd buffers have minimal bounds attached that cover their length, - /// and the same permissions that were checked at the syscall boundary. - pub fn encode_syscall_return_mptr( + /// Does not handle usize other than 4 and 8 bytes. + /// Pointers from allow'd buffers have permissions and length reattached matching + /// those that were checked at the syscall boundary. + fn encode_syscall_return_usize_trd104_compat( &self, a0: &mut CapabilityPtr, a1: &mut CapabilityPtr, From 2cd33d243d647e93168fd09294218393c9fee3d6 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Fri, 8 Nov 2024 21:39:22 +0000 Subject: [PATCH 22/45] Move capability_ptr to utilities Change-Id: If76e4b713fca94fa4f5ba827b994cd8b24624ed5 --- kernel/src/lib.rs | 1 - kernel/src/process.rs | 2 +- kernel/src/process_standard.rs | 4 ++-- kernel/src/syscall.rs | 2 +- kernel/src/upcall.rs | 2 +- kernel/src/{ => utilities}/capability_ptr.rs | 0 kernel/src/utilities/mod.rs | 1 + 7 files changed, 6 insertions(+), 6 deletions(-) rename kernel/src/{ => utilities}/capability_ptr.rs (100%) diff --git a/kernel/src/lib.rs b/kernel/src/lib.rs index 412d00432d..65b8443cd8 100644 --- a/kernel/src/lib.rs +++ b/kernel/src/lib.rs @@ -105,7 +105,6 @@ pub const KERNEL_MAJOR_VERSION: u16 = 2; pub const KERNEL_MINOR_VERSION: u16 = 1; pub mod capabilities; -pub mod capability_ptr; pub mod collections; pub mod component; pub mod debug; diff --git a/kernel/src/process.rs b/kernel/src/process.rs index e5da5f0ccc..1e1a5ce614 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -11,7 +11,6 @@ use core::ptr::NonNull; use core::str; use crate::capabilities; -use crate::capability_ptr::CapabilityPtr; use crate::errorcode::ErrorCode; use crate::ipc; use crate::kernel::Kernel; @@ -20,6 +19,7 @@ use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions; use crate::syscall::{self, Syscall, SyscallReturn}; use crate::upcall::UpcallId; +use crate::utilities::capability_ptr::CapabilityPtr; use tock_tbf::types::CommandPermissions; // Export all process related types via `kernel::process::`. diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 92da6c3ac0..9a09518652 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -14,8 +14,6 @@ use core::num::NonZeroU32; use core::ptr::NonNull; use core::{mem, ptr, slice, str}; -use crate::capability_ptr::MetaPermissions::Execute; -use crate::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::collections::queue::Queue; use crate::collections::ring_buffer::RingBuffer; use crate::config; @@ -38,6 +36,8 @@ use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions::StoragePermissions; use crate::syscall::{self, Syscall, SyscallReturn, UserspaceKernelBoundary}; use crate::upcall::UpcallId; +use crate::utilities::capability_ptr::MetaPermissions::Execute; +use crate::utilities::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::utilities::cells::{MapCell, NumericCellExt, OptionalCell}; use tock_tbf::types::CommandPermissions; diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 6fd150ea4b..9f8a828818 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -68,9 +68,9 @@ use core::fmt::Write; -use crate::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::errorcode::ErrorCode; use crate::process; +use crate::utilities::capability_ptr::{CapabilityPtr, MetaPermissions}; pub use crate::syscall_driver::{CommandReturn, SyscallDriver}; diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index 7f6b9b0a1f..86d60a033d 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -4,12 +4,12 @@ //! Data structure for storing an upcall from the kernel to a process. -use crate::capability_ptr::CapabilityPtr; use crate::config; use crate::debug; use crate::process; use crate::process::ProcessId; use crate::syscall::SyscallReturn; +use crate::utilities::capability_ptr::CapabilityPtr; use crate::ErrorCode; /// Type to uniquely identify an upcall subscription across all drivers. diff --git a/kernel/src/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs similarity index 100% rename from kernel/src/capability_ptr.rs rename to kernel/src/utilities/capability_ptr.rs diff --git a/kernel/src/utilities/mod.rs b/kernel/src/utilities/mod.rs index 5fc700c0fe..6719454748 100644 --- a/kernel/src/utilities/mod.rs +++ b/kernel/src/utilities/mod.rs @@ -14,6 +14,7 @@ pub mod peripheral_management; pub mod static_init; pub mod storage_volume; +pub mod capability_ptr; mod static_ref; pub use self::static_ref::StaticRef; From bf24e7231e408b8382c0c520e7f5c23e0c73e8ab Mon Sep 17 00:00:00 2001 From: LawrenceEsswood Date: Mon, 11 Nov 2024 18:48:58 +0000 Subject: [PATCH 23/45] Update kernel/src/process.rs Co-authored-by: Brad Campbell --- kernel/src/process.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/kernel/src/process.rs b/kernel/src/process.rs index 1e1a5ce614..88c978c95b 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -791,8 +791,8 @@ pub trait Process { /// /// Returns `true` if the upcall function pointer is valid for this process, /// and `false` otherwise. - /// upcall_fn can eventually be a better type: - /// + // `upcall_fn` can eventually be a better type: + // fn is_valid_upcall_function_pointer(&self, upcall_fn: *const ()) -> bool; // functions for processes that are architecture specific From d5697767bb6fc291684621ac29527d2aee62d4f3 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 18:40:29 +0000 Subject: [PATCH 24/45] Rename into_compat Change-Id: I57b8cfee00e00bf2dabc7ab6632b02471f637904 --- kernel/src/syscall.rs | 31 ++++++++++++++++++++----------- 1 file changed, 20 insertions(+), 11 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 9f8a828818..b09a53234f 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -414,7 +414,7 @@ impl SyscallReturnVariant { /// to old ones (u32) for backwards compatibility. /// This should not be used for any newly designed interfaces, /// and will eventually be deprecated once all interfaces are updated. - pub const fn into_compat(self) -> Self { + pub const fn into_compat_32bit_trd104(self) -> Self { // We only need to be backwards compatible on 32-bit systems let compat = core::mem::size_of::() == core::mem::size_of::(); if compat { @@ -672,7 +672,8 @@ impl SyscallReturn { write_64(a2, a3, data1); } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -682,7 +683,8 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -692,7 +694,8 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = (usize::from(err)).into(); *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), @@ -703,7 +706,8 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = (usize::from(err)).into(); *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), @@ -714,7 +718,8 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = CapabilityPtr::new_with_metadata( ptr as *const (), ptr as usize, @@ -724,7 +729,8 @@ impl SyscallReturn { *a2 = len.into(); } SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) + .into(); *a1 = (usize::from(err)).into(); *a2 = CapabilityPtr::new_with_metadata( ptr as *const (), @@ -735,18 +741,20 @@ impl SyscallReturn { *a3 = len.into(); } SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = (SyscallReturnVariant::SuccessPtrPtr.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtrPtr.into_compat_32bit_trd104() as usize) + .into(); *a1 = (ptr as usize).into(); *a2 = data.into(); } SyscallReturn::SubscribeFailure(err, ptr, data) => { - *a0 = (SyscallReturnVariant::FailurePtrPtr.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::FailurePtrPtr.into_compat_32bit_trd104() as usize) + .into(); *a1 = (usize::from(err)).into(); *a2 = (ptr as usize).into(); *a3 = data.into(); } SyscallReturn::SuccessPtr(ptr) => { - *a0 = (SyscallReturnVariant::SuccessPtr.into_compat() as usize).into(); + *a0 = (SyscallReturnVariant::SuccessPtr.into_compat_32bit_trd104() as usize).into(); *a1 = ptr; } SyscallReturn::YieldWaitFor(data0, data1, data2) => { @@ -755,7 +763,8 @@ impl SyscallReturn { *a2 = data2.into(); } SyscallReturn::SuccessUsize(data) => { - *a0 = (SyscallReturnVariant::SuccessUsize.into_compat() as usize).into(); + *a0 = + (SyscallReturnVariant::SuccessUsize.into_compat_32bit_trd104() as usize).into(); *a1 = data.into(); } } From 743c8cd341521f5bb9119346f8194b4531f2a5e8 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 18:47:05 +0000 Subject: [PATCH 25/45] Document what authority the CapabilityPtr from brk/sbrk should have Change-Id: Id0ee752b6e6c8577c557bb2b0f799dedb9297f2d --- kernel/src/process.rs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/kernel/src/process.rs b/kernel/src/process.rs index 88c978c95b..53534e4a4a 100644 --- a/kernel/src/process.rs +++ b/kernel/src/process.rs @@ -528,7 +528,8 @@ pub trait Process { /// /// ## Returns /// - /// On success, return the previous break address. + /// On success, return the previous break address with authority that + /// has RW permissions from the start of process RAM to the new break. /// /// On error, return: /// - [`Error::InactiveApp`] if the process is not running and adjusting the @@ -548,7 +549,8 @@ pub trait Process { /// /// ## Returns /// - /// On success, return the previous break address. + /// On success, return the previous break address with authority that + /// has RW permissions from the start of process RAM to the new break. /// /// On error, return: /// - [`Error::InactiveApp`] if the process is not running and adjusting the From 144269eeb3bff6b9cc359b543f7b351f9c572186 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 18:56:24 +0000 Subject: [PATCH 26/45] Do not pollute scope with Execute Change-Id: Ic0d56133529ceb1a6a0a33f60299ee57db8f2ff3 --- kernel/src/process_standard.rs | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 9a09518652..10be3e5678 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -36,7 +36,6 @@ use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions::StoragePermissions; use crate::syscall::{self, Syscall, SyscallReturn, UserspaceKernelBoundary}; use crate::upcall::UpcallId; -use crate::utilities::capability_ptr::MetaPermissions::Execute; use crate::utilities::capability_ptr::{CapabilityPtr, MetaPermissions}; use crate::utilities::cells::{MapCell, NumericCellExt, OptionalCell}; @@ -1754,8 +1753,12 @@ impl ProcessStandard<'_, C> { flash_start.wrapping_add(process.header.get_init_function_offset() as usize) as usize; let fn_base = flash_start as usize; let fn_len = process.flash.len(); - let init_fn = - CapabilityPtr::new_with_metadata(init_addr as *const (), fn_base, fn_len, Execute); + let init_fn = CapabilityPtr::new_with_metadata( + init_addr as *const (), + fn_base, + fn_len, + MetaPermissions::Execute, + ); process.tasks.map(|tasks| { tasks.enqueue(Task::FunctionCall(FunctionCall { @@ -1924,7 +1927,7 @@ impl ProcessStandard<'_, C> { init_addr as *const (), flash_start as usize, (self.flash_end() as usize) - (flash_start as usize), - Execute, + MetaPermissions::Execute, ); self.enqueue_task(Task::FunctionCall(FunctionCall { From cc0b53ec00d0b60e56fe80a6cbddd4008fc32646 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 19:13:02 +0000 Subject: [PATCH 27/45] Add comment to construction of intial fn Change-Id: I70b793b2417880bb46831e56a20bdcdc2a925d0c --- kernel/src/process_standard.rs | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 10be3e5678..aa5ba9a513 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -1753,6 +1753,10 @@ impl ProcessStandard<'_, C> { flash_start.wrapping_add(process.header.get_init_function_offset() as usize) as usize; let fn_base = flash_start as usize; let fn_len = process.flash.len(); + + // We need to construct a capability with sufficient authority to cover all of a user's + // code, with permissions to execute it. The entirety of flash is sufficient. + let init_fn = CapabilityPtr::new_with_metadata( init_addr as *const (), fn_base, @@ -1923,6 +1927,9 @@ impl ProcessStandard<'_, C> { let init_addr = flash_start.wrapping_add(self.header.get_init_function_offset() as usize) as usize; + // We need to construct a capability with sufficient authority to cover all of a user's + // code, with permissions to execute it. The entirety of flash is sufficient. + let init_fn = CapabilityPtr::new_with_metadata( init_addr as *const (), flash_start as usize, From 45f8a442587f59ab646a65f63d2d192667f91980 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 19:27:57 +0000 Subject: [PATCH 28/45] Remove mention of CHERI Change-Id: I68b5d9e4e40a0659f64be021f2b8c4eff68203b1 --- kernel/src/process_standard.rs | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index aa5ba9a513..082519a892 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -1953,10 +1953,11 @@ impl ProcessStandard<'_, C> { /// to be accessible to the process and to not overlap with the grant /// region. fn in_app_owned_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { - // TODO: On CHERI platforms, it impossible to form syscalls with pointers - // that are not in app memory. However, buf_start_addr is not the right - // type to make this function always return true. If CapabilityPtr makes it - // slightly further, we can skip this check. + // TODO: On some platforms, CapabilityPtr has sufficient authority that we + // could skip this check. + // CapabilityPtr needs to make it slightly further, and we need to add + // interfaces that tell us how much assurance it gives on the current + // platform. let buf_end_addr = buf_start_addr.wrapping_add(size); buf_end_addr >= buf_start_addr @@ -1969,10 +1970,11 @@ impl ProcessStandard<'_, C> { /// this method returns true, the buffer is guaranteed to be readable to the /// process. fn in_app_flash_memory(&self, buf_start_addr: *const u8, size: usize) -> bool { - // TODO: On CHERI platforms, it impossible to form syscalls with pointers - // that are not in app memory. However, buf_start_addr is not the right - // type to make this function always return true. If CapabilityPtr makes it - // slightly further, we can skip this check. + // TODO: On some platforms, CapabilityPtr has sufficient authority that we + // could skip this check. + // CapabilityPtr needs to make it slightly further, and we need to add + // interfaces that tell us how much assurance it gives on the current + // platform. let buf_end_addr = buf_start_addr.wrapping_add(size); buf_end_addr >= buf_start_addr From 391dfc817a52caec91e6892eab00e2224fee148f Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 19:28:09 +0000 Subject: [PATCH 29/45] Use .cast_mut(), not as Change-Id: I45e8e7d8e368c8ff74e76f783355d588916281df --- kernel/src/upcall.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index 86d60a033d..0ade28698a 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -182,9 +182,9 @@ impl Upcall { self.process_id, self.upcall_id.driver_num, self.upcall_id.subscribe_num, - self.fn_ptr - .map_or(core::ptr::null_mut::<()>(), |fp| fp.as_ptr::<()>() - as *mut ()) as usize, + self.fn_ptr.map_or(core::ptr::null_mut::<()>(), |fp| fp + .as_ptr::<()>() + .cast_mut()) as usize, r0, r1, r2, From 8f8ce44f5a31e560fc68c28aa6df3f7280b08e69 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 19:35:49 +0000 Subject: [PATCH 30/45] Document all methods on CapabilityPtr Change-Id: I061271386a151786843df3cf854c1114c492ec67 --- kernel/src/utilities/capability_ptr.rs | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index d0ed2a6853..744fcfc274 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -39,6 +39,8 @@ impl Default for CapabilityPtr { } } +/// Permission sets a [CapabilityPtr] may grant. +/// These may not be enforced or exist on a given platform. #[derive(Copy, Clone, PartialEq)] pub enum MetaPermissions { None, @@ -49,7 +51,8 @@ pub enum MetaPermissions { } impl From for usize { - /// Provenance note: may not expose provenance + /// Returns the address of the [CapabilityPtr]. + /// Provenance note: may not expose provenance. #[inline] fn from(from: CapabilityPtr) -> Self { from.ptr as usize @@ -57,7 +60,8 @@ impl From for usize { } impl From for CapabilityPtr { - /// Provenance note: may have null provenance + /// Constructs a [CapabilityPtr] with a given address. + /// Provenance note: may have null provenance. #[inline] fn from(from: usize) -> Self { Self { @@ -81,6 +85,7 @@ impl LowerHex for CapabilityPtr { } impl AddAssign for CapabilityPtr { + /// Increments the address of a [CapabilityPtr] #[inline] fn add_assign(&mut self, rhs: usize) { self.ptr = (self.ptr as *const u8).wrapping_add(rhs) as *const (); @@ -88,10 +93,13 @@ impl AddAssign for CapabilityPtr { } impl CapabilityPtr { + /// Returns the pointer component of a [CapabilityPtr] but without any of the authority. pub fn as_ptr(&self) -> *const T { self.ptr as *const T } + /// Construct a [CapabilityPtr] from a raw pointer, with the authority requested by other + /// arguments. /// Provenance note: may derive from a pointer other than the input to provide something with /// valid provenance to justify the other arguments. #[inline] @@ -104,6 +112,7 @@ impl CapabilityPtr { Self { ptr } } + /// If the [CapabilityPtr] is null returns `default`, otherwise applies `f` to `self`. #[inline] pub fn map_or(&self, default: U, f: F) -> U where @@ -116,6 +125,8 @@ impl CapabilityPtr { } } + /// If the [CapabilityPtr] is null returns `default`, otherwise applies `f` to `self`. + /// default is only evaluated if `self` is not null. #[inline] pub fn map_or_else(&self, default: D, f: F) -> U where From a5e5a6d2a919ab857d68b847f0a7f4fe7e47df78 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 19:53:53 +0000 Subject: [PATCH 31/45] More formatting/comments/name change for permissions Change-Id: I15820045ccc5011221203f387371b613a070cb46 --- kernel/src/process_standard.rs | 8 ++--- kernel/src/syscall.rs | 14 ++++----- kernel/src/utilities/capability_ptr.rs | 41 +++++++++++++++----------- kernel/src/utilities/mod.rs | 5 ++-- 4 files changed, 36 insertions(+), 32 deletions(-) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 082519a892..a0cd1707a2 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -36,7 +36,7 @@ use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::storage_permissions::StoragePermissions; use crate::syscall::{self, Syscall, SyscallReturn, UserspaceKernelBoundary}; use crate::upcall::UpcallId; -use crate::utilities::capability_ptr::{CapabilityPtr, MetaPermissions}; +use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; use crate::utilities::cells::{MapCell, NumericCellExt, OptionalCell}; use tock_tbf::types::CommandPermissions; @@ -621,7 +621,7 @@ impl Process for ProcessStandard<'_, C> { old_break as *const (), base, (new_break as usize) - base, - MetaPermissions::ReadWrite, + CapabilityPtrPermissions::ReadWrite, ); Ok(break_result) @@ -1761,7 +1761,7 @@ impl ProcessStandard<'_, C> { init_addr as *const (), fn_base, fn_len, - MetaPermissions::Execute, + CapabilityPtrPermissions::Execute, ); process.tasks.map(|tasks| { @@ -1934,7 +1934,7 @@ impl ProcessStandard<'_, C> { init_addr as *const (), flash_start as usize, (self.flash_end() as usize) - (flash_start as usize), - MetaPermissions::Execute, + CapabilityPtrPermissions::Execute, ); self.enqueue_task(Task::FunctionCall(FunctionCall { diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index b09a53234f..9ea892f796 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -70,7 +70,7 @@ use core::fmt::Write; use crate::errorcode::ErrorCode; use crate::process; -use crate::utilities::capability_ptr::{CapabilityPtr, MetaPermissions}; +use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; pub use crate::syscall_driver::{CommandReturn, SyscallDriver}; @@ -678,7 +678,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::ReadWrite, + CapabilityPtrPermissions::ReadWrite, ); *a2 = len.into(); } @@ -689,7 +689,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::Read, + CapabilityPtrPermissions::Read, ); *a2 = len.into(); } @@ -701,7 +701,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::ReadWrite, + CapabilityPtrPermissions::ReadWrite, ); *a3 = len.into(); } @@ -713,7 +713,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::Read, + CapabilityPtrPermissions::Read, ); *a3 = len.into(); } @@ -724,7 +724,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::Read, + CapabilityPtrPermissions::Read, ); *a2 = len.into(); } @@ -736,7 +736,7 @@ impl SyscallReturn { ptr as *const (), ptr as usize, len, - MetaPermissions::Read, + CapabilityPtrPermissions::Read, ); *a3 = len.into(); } diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index 744fcfc274..3eab391f1c 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -9,21 +9,22 @@ use core::ops::AddAssign; /// A pointer to userspace memory with implied authority. /// -/// A [CapabilityPtr] points to memory a userspace processes may be +/// A [`CapabilityPtr`] points to memory a userspace process may be /// permitted to read, write, or execute. It is sized exactly to a -/// register that can pass values between userspace and kernel and at -/// least the size of a word ([usize]) [^note1]. Operations on the +/// CPU register that can pass values between userspace and the kernel. +/// Because it is register sized, [`CapabilityPtr`] is guaranteed to be +/// at least the size of a word ([usize]) [^note1]. Operations on the /// pointer may affect permissions, e.g. offsetting the pointer beyond /// the bounds of the memory object invalidates it. Like a `*const -/// ()`, a [CapabilityPtr] may also "hide" information by storing a +/// ()`, a [`CapabilityPtr`] may also "hide" information by storing a /// word of data with no memory access permissions. /// -/// [CapabilityPtr] should be used to store or pass between the kernel -/// and userspace a value that may represent a valid userspace reference, +/// [`CapabilityPtr`] should be used to store or pass a value between the +/// kernel and userspace that may represent a valid userspace reference, /// when one party intends the other to access it. /// /// [^note1]: Depending on the architecture, the size of a -/// [CapabilityPtr] may be a word size or larger, e.g., if registers +/// [`CapabilityPtr`] may be a word size or larger, e.g., if registers /// can store metadata such as access permissions. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)] #[repr(transparent)] @@ -39,10 +40,10 @@ impl Default for CapabilityPtr { } } -/// Permission sets a [CapabilityPtr] may grant. +/// Permission sets a [`CapabilityPtr`] may grant. /// These may not be enforced or exist on a given platform. #[derive(Copy, Clone, PartialEq)] -pub enum MetaPermissions { +pub enum CapabilityPtrPermissions { None, Read, Write, @@ -51,7 +52,7 @@ pub enum MetaPermissions { } impl From for usize { - /// Returns the address of the [CapabilityPtr]. + /// Returns the address of the [`CapabilityPtr`]. /// Provenance note: may not expose provenance. #[inline] fn from(from: CapabilityPtr) -> Self { @@ -60,7 +61,7 @@ impl From for usize { } impl From for CapabilityPtr { - /// Constructs a [CapabilityPtr] with a given address. + /// Constructs a [`CapabilityPtr`] with a given address. /// Provenance note: may have null provenance. #[inline] fn from(from: usize) -> Self { @@ -71,6 +72,8 @@ impl From for CapabilityPtr { } impl UpperHex for CapabilityPtr { + /// Format the capability as an uppercase hex string. + /// Will print at least the address, and any platform specific metadata if it exists. #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { UpperHex::fmt(&(self.ptr as usize), f) @@ -78,6 +81,8 @@ impl UpperHex for CapabilityPtr { } impl LowerHex for CapabilityPtr { + /// Format the capability as a lowercase hex string. + /// Will print at least the address, and any platform specific metadata if it exists. #[inline] fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { LowerHex::fmt(&(self.ptr as usize), f) @@ -85,7 +90,7 @@ impl LowerHex for CapabilityPtr { } impl AddAssign for CapabilityPtr { - /// Increments the address of a [CapabilityPtr] + /// Increments the address of a [`CapabilityPtr`] #[inline] fn add_assign(&mut self, rhs: usize) { self.ptr = (self.ptr as *const u8).wrapping_add(rhs) as *const (); @@ -93,13 +98,13 @@ impl AddAssign for CapabilityPtr { } impl CapabilityPtr { - /// Returns the pointer component of a [CapabilityPtr] but without any of the authority. + /// Returns the pointer component of a [`CapabilityPtr`] but without any of the authority. pub fn as_ptr(&self) -> *const T { self.ptr as *const T } - /// Construct a [CapabilityPtr] from a raw pointer, with the authority requested by other - /// arguments. + /// Construct a [`CapabilityPtr`] from a raw pointer, with authority ranging over + /// [`base`, `base + length`) and permissions `perms`. /// Provenance note: may derive from a pointer other than the input to provide something with /// valid provenance to justify the other arguments. #[inline] @@ -107,12 +112,12 @@ impl CapabilityPtr { ptr: *const (), _base: usize, _length: usize, - _perms: MetaPermissions, + _perms: CapabilityPtrPermissions, ) -> Self { Self { ptr } } - /// If the [CapabilityPtr] is null returns `default`, otherwise applies `f` to `self`. + /// If the [`CapabilityPtr`] is null returns `default`, otherwise applies `f` to `self`. #[inline] pub fn map_or(&self, default: U, f: F) -> U where @@ -125,7 +130,7 @@ impl CapabilityPtr { } } - /// If the [CapabilityPtr] is null returns `default`, otherwise applies `f` to `self`. + /// If the [`CapabilityPtr`] is null returns `default`, otherwise applies `f` to `self`. /// default is only evaluated if `self` is not null. #[inline] pub fn map_or_else(&self, default: D, f: F) -> U diff --git a/kernel/src/utilities/mod.rs b/kernel/src/utilities/mod.rs index 6719454748..b9b53bacfe 100644 --- a/kernel/src/utilities/mod.rs +++ b/kernel/src/utilities/mod.rs @@ -5,6 +5,7 @@ //! Utility functions and macros provided by the kernel crate. pub mod binary_write; +pub mod capability_ptr; pub mod copy_slice; pub mod helpers; pub mod leasable_buffer; @@ -12,10 +13,8 @@ pub mod math; pub mod mut_imut_buffer; pub mod peripheral_management; pub mod static_init; -pub mod storage_volume; - -pub mod capability_ptr; mod static_ref; +pub mod storage_volume; pub use self::static_ref::StaticRef; From f3feec688d8add0946e73f3c23c31246c5372d27 Mon Sep 17 00:00:00 2001 From: Lawrence Esswood Date: Mon, 11 Nov 2024 20:10:55 +0000 Subject: [PATCH 32/45] Style changes Change-Id: I1972748578e6e685ed8dc6126e1c4da6753f5f4f --- kernel/src/syscall.rs | 35 ++++++++++++++++++----------------- 1 file changed, 18 insertions(+), 17 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 9ea892f796..d62b0a67cd 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -278,19 +278,19 @@ impl Syscall { Ok(SyscallClass::ReadWriteAllow) => Some(Syscall::ReadWriteAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr::() as *mut u8, + allow_address: r2.as_ptr::().cast_mut(), allow_size: r3.into(), }), Ok(SyscallClass::UserspaceReadableAllow) => Some(Syscall::UserspaceReadableAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr::() as *mut u8, + allow_address: r2.as_ptr::().cast_mut(), allow_size: r3.into(), }), Ok(SyscallClass::ReadOnlyAllow) => Some(Syscall::ReadOnlyAllow { driver_number: r0, subdriver_number: r1.into(), - allow_address: r2.as_ptr::() as *mut u8, + allow_address: r2.as_ptr::().cast_mut(), allow_size: r3.into(), }), Ok(SyscallClass::Memop) => Some(Syscall::Memop { @@ -417,16 +417,17 @@ impl SyscallReturnVariant { pub const fn into_compat_32bit_trd104(self) -> Self { // We only need to be backwards compatible on 32-bit systems let compat = core::mem::size_of::() == core::mem::size_of::(); - if compat { - match self { - // Map all usizes and ptrs to u32 - Self::SuccessUsize | Self::SuccessPtr => Self::SuccessU32, - Self::SuccessPtrUsize | Self::SuccessPtrPtr => Self::SuccessU32U32, - Self::FailurePtrUsize | Self::FailurePtrPtr => Self::FailureU32U32, - x => x, - } - } else { - self + + if !compat { + return self; + } + + match self { + // Map all usizes and ptrs to u32 + Self::SuccessUsize | Self::SuccessPtr => Self::SuccessU32, + Self::SuccessPtrUsize | Self::SuccessPtrPtr => Self::SuccessU32U32, + Self::FailurePtrUsize | Self::FailurePtrPtr => Self::FailureU32U32, + x => x, } } } @@ -588,10 +589,10 @@ impl SyscallReturn { // Ugly coercion could be avoided by first copying to the stack, then assigning with // "as" in order to satisfy the compiler. unsafe { - let a0 = &mut *(core::ptr::from_mut(a0) as *mut CapabilityPtr); - let a1 = &mut *(core::ptr::from_mut(a1) as *mut CapabilityPtr); - let a2 = &mut *(core::ptr::from_mut(a2) as *mut CapabilityPtr); - let a3 = &mut *(core::ptr::from_mut(a3) as *mut CapabilityPtr); + let a0 = &mut *(core::ptr::from_mut(a0).cast::()); + let a1 = &mut *(core::ptr::from_mut(a1).cast::()); + let a2 = &mut *(core::ptr::from_mut(a2).cast::()); + let a3 = &mut *(core::ptr::from_mut(a3).cast::()); self.encode_syscall_return_usize_trd104_compat(a0, a1, a2, a3); } } From a001b334cbdbce9ddb9e2b7e749c9e5ca3e30f31 Mon Sep 17 00:00:00 2001 From: Leon Schuermann Date: Fri, 8 Nov 2024 16:37:20 -0500 Subject: [PATCH 33/45] kernel/syscall: split out encode_syscall_return, create TRD104 subset Because all architecture crates depend on the kernel crate, we use it not just for architecture-agnostic core kernel infrastructure, but also architecture-specific code that happens to be shared between two or more `arch` crates. Prior to this change, we conflated these helpers with the core kernel infrastructure in `syscall.rs`. With this change, we split out the 32-bit and TRD 104-specific `encode_syscall_return` helper, shared between the 32-bit RISC-V and Cortex-M architecture implementations, into its own dedicated module under utilities. We create a separate `SyscallReturn` subset for those return values as specified in TRD 104, and create a function that translates between these types. This allows architectures which use these functions to guarantee that they are conformant to TRD 104 by first explicitly converting into this type, and then encoding it into their stored registers using the TRD 104 encoding helper. --- arch/cortex-m/src/syscall.rs | 10 +- arch/rv32i/src/syscall.rs | 5 +- kernel/src/syscall.rs | 270 +---------------- kernel/src/utilities/arch_helpers.rs | 437 +++++++++++++++++++++++++++ kernel/src/utilities/mod.rs | 1 + 5 files changed, 454 insertions(+), 269 deletions(-) create mode 100644 kernel/src/utilities/arch_helpers.rs diff --git a/arch/cortex-m/src/syscall.rs b/arch/cortex-m/src/syscall.rs index 7895a95c6d..840240cd67 100644 --- a/arch/cortex-m/src/syscall.rs +++ b/arch/cortex-m/src/syscall.rs @@ -195,7 +195,15 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall // // Refer to // https://doc.rust-lang.org/std/primitive.pointer.html#safety-13 - return_value.encode_syscall_return_32bit_trd104(&mut *r0, &mut *r1, &mut *r2, &mut *r3); + kernel::utilities::arch_helpers::encode_syscall_return_32bit_trd104( + &kernel::utilities::arch_helpers::TRD104SyscallReturn::from_syscall_return( + return_value, + ), + &mut *r0, + &mut *r1, + &mut *r2, + &mut *r3, + ); Ok(()) } diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index 074fbdb5c9..d787020741 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -172,7 +172,10 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { let (a1slice, r) = r.split_at_mut(R_A2 - R_A1); let (a2slice, a3slice) = r.split_at_mut(R_A3 - R_A2); - return_value.encode_syscall_return_32bit_trd104( + kernel::utilities::arch_helpers::encode_syscall_return_32bit_trd104( + &kernel::utilities::arch_helpers::TRD104SyscallReturn::from_syscall_return( + return_value, + ), &mut a0slice[0], &mut a1slice[0], &mut a2slice[0], diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index d62b0a67cd..09654593ef 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -70,22 +70,10 @@ use core::fmt::Write; use crate::errorcode::ErrorCode; use crate::process; -use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; +use crate::utilities::capability_ptr::CapabilityPtr; pub use crate::syscall_driver::{CommandReturn, SyscallDriver}; -/// Helper function to split a [`u64`] into a higher and lower [`u32`]. -/// -/// Used in encoding 64-bit wide system call return values on 32-bit platforms. -#[inline] -fn u64_to_be_u32s(src: u64) -> (u32, u32) { - let src_bytes = src.to_be_bytes(); - let src_msb = u32::from_be_bytes([src_bytes[0], src_bytes[1], src_bytes[2], src_bytes[3]]); - let src_lsb = u32::from_be_bytes([src_bytes[4], src_bytes[5], src_bytes[6], src_bytes[7]]); - - (src_msb, src_lsb) -} - // ---------- SYSTEMCALL ARGUMENT DECODING ---------- /// Enumeration of the system call classes based on the identifiers specified in @@ -381,58 +369,9 @@ impl Syscall { } } -// ---------- SYSCALL RETURN VALUE ENCODING ---------- +// ---------- SYSCALL RETURN VALUES ---------- -/// Enumeration of the system call return type variant identifiers described -/// in TRD104. -/// -/// Each variant is associated with the respective variant identifier that would -/// be passed along with the return value to userspace. -#[repr(u32)] -#[derive(Copy, Clone, Debug)] -pub enum SyscallReturnVariant { - Failure = 0, - FailureU32 = 1, - FailureU32U32 = 2, - FailureU64 = 3, - FailurePtrUsize = 4, - FailurePtrPtr = 5, - Success = 128, - SuccessU32 = 129, - SuccessU32U32 = 130, - SuccessU64 = 131, - SuccessU32U32U32 = 132, - SuccessU32U64 = 133, - SuccessUsize = 134, - SuccessPtr = 135, - SuccessPtrUsize = 136, - SuccessPtrPtr = 137, -} - -impl SyscallReturnVariant { - /// Maps newly introduced return variants (usize and ptr) - /// to old ones (u32) for backwards compatibility. - /// This should not be used for any newly designed interfaces, - /// and will eventually be deprecated once all interfaces are updated. - pub const fn into_compat_32bit_trd104(self) -> Self { - // We only need to be backwards compatible on 32-bit systems - let compat = core::mem::size_of::() == core::mem::size_of::(); - - if !compat { - return self; - } - - match self { - // Map all usizes and ptrs to u32 - Self::SuccessUsize | Self::SuccessPtr => Self::SuccessU32, - Self::SuccessPtrUsize | Self::SuccessPtrPtr => Self::SuccessU32U32, - Self::FailurePtrUsize | Self::FailurePtrPtr => Self::FailureU32U32, - x => x, - } - } -} - -/// Enumeration of the possible system call return variants specified in TRD104. +/// Enumeration of the possible system call return variants. /// /// This struct operates over primitive types such as integers of fixed length /// and pointers. It is constructed by the scheduler and passed down to the @@ -567,209 +506,6 @@ impl SyscallReturn { SyscallReturn::SuccessUsize(_) => true, } } - - /// Encode the system call return value into 4 registers, following the - /// encoding specified in TRD104. Architectures which do not follow TRD104 - /// are free to define their own encoding. - pub fn encode_syscall_return_32bit_trd104( - &self, - a0: &mut u32, - a1: &mut u32, - a2: &mut u32, - a3: &mut u32, - ) { - assert!( - core::mem::size_of::() == core::mem::size_of::() - && core::mem::align_of::() >= align_of::(), - "encode_syscall_return used on a 64-bit platform or CHERI platform" - ); - - // SAFETY: if the two integers are the same size (and alignment permits) references - // to them can be safely transmuted. - // Ugly coercion could be avoided by first copying to the stack, then assigning with - // "as" in order to satisfy the compiler. - unsafe { - let a0 = &mut *(core::ptr::from_mut(a0).cast::()); - let a1 = &mut *(core::ptr::from_mut(a1).cast::()); - let a2 = &mut *(core::ptr::from_mut(a2).cast::()); - let a3 = &mut *(core::ptr::from_mut(a3).cast::()); - self.encode_syscall_return_usize_trd104_compat(a0, a1, a2, a3); - } - } - - /// An extension of TRD104 that works for 32-bit and 64-bit platforms. - /// This implements TRD104 exactly on 32-bit platforms by mapping new codes intended for - /// 64-bit platforms to existing ones. - /// On 64-bit platforms, both 64-bit and usize values are passed as a single register, - /// Does not handle usize other than 4 and 8 bytes. - /// Pointers from allow'd buffers have permissions and length reattached matching - /// those that were checked at the syscall boundary. - fn encode_syscall_return_usize_trd104_compat( - &self, - a0: &mut CapabilityPtr, - a1: &mut CapabilityPtr, - a2: &mut CapabilityPtr, - a3: &mut CapabilityPtr, - ) { - // On 32-bit CHERI, given that capabilities cannot be used as 64-bit integers, 64-bit - // integers will still be returned as two 32-bit values in different registers. - fn write_64(a: &mut CapabilityPtr, b: &mut CapabilityPtr, val: u64) { - let is_64_bit = core::mem::size_of::() == 8; - if !is_64_bit { - let (msb, lsb) = u64_to_be_u32s(val); - *a = (lsb as usize).into(); - *b = (msb as usize).into(); - } else { - *a = (val as usize).into(); - } - } - - match *self { - SyscallReturn::Failure(e) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); - *a1 = (usize::from(e)).into(); - } - SyscallReturn::FailureU32(e, data0) => { - *a0 = (SyscallReturnVariant::FailureU32 as usize).into(); - *a1 = usize::from(e).into(); - *a2 = (data0 as usize).into(); - } - SyscallReturn::FailureU32U32(e, data0, data1) => { - *a0 = (SyscallReturnVariant::FailureU32U32 as usize).into(); - *a1 = (usize::from(e)).into(); - *a2 = (data0 as usize).into(); - *a3 = (data1 as usize).into(); - } - SyscallReturn::FailureU64(e, data0) => { - *a0 = (SyscallReturnVariant::FailureU64 as usize).into(); - *a1 = (usize::from(e)).into(); - write_64(a2, a3, data0) - } - SyscallReturn::Success => { - *a0 = (SyscallReturnVariant::Success as usize).into(); - } - SyscallReturn::SuccessU32(data0) => { - *a0 = (SyscallReturnVariant::SuccessU32 as usize).into(); - *a1 = (data0 as usize).into(); - } - SyscallReturn::SuccessU32U32(data0, data1) => { - *a0 = (SyscallReturnVariant::SuccessU32U32 as usize).into(); - *a1 = (data0 as usize).into(); - *a2 = (data1 as usize).into(); - } - SyscallReturn::SuccessU32U32U32(data0, data1, data2) => { - *a0 = (SyscallReturnVariant::SuccessU32U32U32 as usize).into(); - *a1 = (data0 as usize).into(); - *a2 = (data1 as usize).into(); - *a3 = (data2 as usize).into(); - } - SyscallReturn::SuccessU64(data0) => { - *a0 = (SyscallReturnVariant::SuccessU64 as usize).into(); - write_64(a1, a2, data0); - } - SyscallReturn::SuccessU32U64(data0, data1) => { - *a0 = (SyscallReturnVariant::SuccessU32U64 as usize).into(); - *a1 = (data0 as usize).into(); - write_64(a2, a3, data1); - } - SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ); - *a2 = len.into(); - } - SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); - *a2 = len.into(); - } - SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ); - *a3 = len.into(); - } - SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); - *a3 = len.into(); - } - SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); - *a2 = len.into(); - } - SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize.into_compat_32bit_trd104() as usize) - .into(); - *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); - *a3 = len.into(); - } - SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = (SyscallReturnVariant::SuccessPtrPtr.into_compat_32bit_trd104() as usize) - .into(); - *a1 = (ptr as usize).into(); - *a2 = data.into(); - } - SyscallReturn::SubscribeFailure(err, ptr, data) => { - *a0 = (SyscallReturnVariant::FailurePtrPtr.into_compat_32bit_trd104() as usize) - .into(); - *a1 = (usize::from(err)).into(); - *a2 = (ptr as usize).into(); - *a3 = data.into(); - } - SyscallReturn::SuccessPtr(ptr) => { - *a0 = (SyscallReturnVariant::SuccessPtr.into_compat_32bit_trd104() as usize).into(); - *a1 = ptr; - } - SyscallReturn::YieldWaitFor(data0, data1, data2) => { - *a0 = data0.into(); - *a1 = data1.into(); - *a2 = data2.into(); - } - SyscallReturn::SuccessUsize(data) => { - *a0 = - (SyscallReturnVariant::SuccessUsize.into_compat_32bit_trd104() as usize).into(); - *a1 = data.into(); - } - } - } } // ---------- USERSPACE KERNEL BOUNDARY ---------- diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs new file mode 100644 index 0000000000..90d3359f46 --- /dev/null +++ b/kernel/src/utilities/arch_helpers.rs @@ -0,0 +1,437 @@ +// Licensed under the Apache License, Version 2.0 or the MIT License. +// SPDX-License-Identifier: Apache-2.0 OR MIT +// Copyright Tock Contributors 2024. + +//! Helper functions and types shared between multiple `arch` crates. +//! +//! This function contains functions and types that do not have to be in the +//! core kernel and are architecture-specific, but are shared by two or more +//! `arch` crates. While these could also live in a dedicated crate, we use the +//! `kernel` crate as all `arch` crates already depend on it. + +use crate::syscall::SyscallReturn; +use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; +use crate::ErrorCode; + +/// Helper function to split a [`u64`] into a higher and lower [`u32`]. +/// +/// Used in encoding 64-bit wide system call return values on 32-bit +/// platforms. +#[inline] +fn u64_to_be_u32s(src: u64) -> (u32, u32) { + let src_bytes = src.to_be_bytes(); + let src_msb = u32::from_be_bytes([src_bytes[0], src_bytes[1], src_bytes[2], src_bytes[3]]); + let src_lsb = u32::from_be_bytes([src_bytes[4], src_bytes[5], src_bytes[6], src_bytes[7]]); + + (src_msb, src_lsb) +} + +/// Enumeration of the system call return type variant identifiers described in +/// TRD104. +/// +/// Each variant is associated with the respective variant identifier that would +/// be passed along with the return value to userspace. +#[repr(u32)] +#[derive(Copy, Clone, Debug)] +pub enum TRD104SyscallReturnVariant { + Failure = 0, + FailureU32 = 1, + FailureU32U32 = 2, + FailureU64 = 3, + Success = 128, + SuccessU32 = 129, + SuccessU32U32 = 130, + SuccessU64 = 131, + SuccessU32U32U32 = 132, + SuccessU32U64 = 133, +} + +/// System call return variants defined as defined in TRD104. These are a strict +/// subset of the variants defined in the core kernel's [`SyscallReturn`] +/// enum. For documentation on the individual variants, refer to this type +/// instead. +#[derive(Copy, Clone, Debug)] +pub enum TRD104SyscallReturn { + Failure(ErrorCode), + FailureU32(ErrorCode, u32), + FailureU32U32(ErrorCode, u32, u32), + FailureU64(ErrorCode, u64), + Success, + SuccessU32(u32), + SuccessU32U32(u32, u32), + SuccessU32U32U32(u32, u32, u32), + SuccessU64(u64), + SuccessU32U64(u32, u64), + AllowReadWriteSuccess(*mut u8, usize), + AllowReadWriteFailure(ErrorCode, *mut u8, usize), + UserspaceReadableAllowSuccess(*mut u8, usize), + UserspaceReadableAllowFailure(ErrorCode, *mut u8, usize), + AllowReadOnlySuccess(*const u8, usize), + AllowReadOnlyFailure(ErrorCode, *const u8, usize), + SubscribeSuccess(*const (), usize), + SubscribeFailure(ErrorCode, *const (), usize), + YieldWaitFor(usize, usize, usize), +} + +impl TRD104SyscallReturn { + /// Map from the kernel's [`SyscallReturn`] enum to the subset of return + /// values specified in TRD104. This ensures backwards compatibility with + /// architectures implementing the ABI as specified in TRD104. + pub fn from_syscall_return(syscall_return: SyscallReturn) -> Self { + match syscall_return { + // Identical variants: + SyscallReturn::Failure(a) => TRD104SyscallReturn::Failure(a), + SyscallReturn::FailureU32(a, b) => TRD104SyscallReturn::FailureU32(a, b), + SyscallReturn::FailureU32U32(a, b, c) => TRD104SyscallReturn::FailureU32U32(a, b, c), + SyscallReturn::FailureU64(a, b) => TRD104SyscallReturn::FailureU64(a, b), + SyscallReturn::Success => TRD104SyscallReturn::Success, + SyscallReturn::SuccessU32(a) => TRD104SyscallReturn::SuccessU32(a), + SyscallReturn::SuccessU32U32(a, b) => TRD104SyscallReturn::SuccessU32U32(a, b), + SyscallReturn::SuccessU32U32U32(a, b, c) => { + TRD104SyscallReturn::SuccessU32U32U32(a, b, c) + } + SyscallReturn::SuccessU64(a) => TRD104SyscallReturn::SuccessU64(a), + SyscallReturn::SuccessU32U64(a, b) => TRD104SyscallReturn::SuccessU32U64(a, b), + SyscallReturn::AllowReadWriteSuccess(a, b) => { + TRD104SyscallReturn::AllowReadWriteSuccess(a, b) + } + SyscallReturn::AllowReadWriteFailure(a, b, c) => { + TRD104SyscallReturn::AllowReadWriteFailure(a, b, c) + } + SyscallReturn::UserspaceReadableAllowSuccess(a, b) => { + TRD104SyscallReturn::UserspaceReadableAllowSuccess(a, b) + } + SyscallReturn::UserspaceReadableAllowFailure(a, b, c) => { + TRD104SyscallReturn::UserspaceReadableAllowFailure(a, b, c) + } + SyscallReturn::AllowReadOnlySuccess(a, b) => { + TRD104SyscallReturn::AllowReadOnlySuccess(a, b) + } + SyscallReturn::AllowReadOnlyFailure(a, b, c) => { + TRD104SyscallReturn::AllowReadOnlyFailure(a, b, c) + } + SyscallReturn::SubscribeSuccess(a, b) => TRD104SyscallReturn::SubscribeSuccess(a, b), + SyscallReturn::SubscribeFailure(a, b, c) => { + TRD104SyscallReturn::SubscribeFailure(a, b, c) + } + SyscallReturn::YieldWaitFor(a, b, c) => TRD104SyscallReturn::YieldWaitFor(a, b, c), + + // Compatibility mapping: + SyscallReturn::SuccessUsize(a) => TRD104SyscallReturn::SuccessU32(a as u32), + SyscallReturn::SuccessPtr(a) => { + TRD104SyscallReturn::SuccessU32(a.as_ptr::<()>() as u32) + } + } + } +} + +/// Encode the system call return value into 4 registers, following the encoding +/// specified in TRD104. Architectures which do not follow TRD104 are free to +/// define their own encoding. +pub fn encode_syscall_return_32bit_trd104( + syscall_return: &TRD104SyscallReturn, + a0: &mut u32, + a1: &mut u32, + a2: &mut u32, + a3: &mut u32, +) { + match *syscall_return { + TRD104SyscallReturn::Failure(e) => { + *a0 = TRD104SyscallReturnVariant::Failure as u32; + *a1 = usize::from(e) as u32; + } + TRD104SyscallReturn::FailureU32(e, data0) => { + *a0 = TRD104SyscallReturnVariant::FailureU32 as u32; + *a1 = usize::from(e) as u32; + *a2 = data0; + } + TRD104SyscallReturn::FailureU32U32(e, data0, data1) => { + *a0 = TRD104SyscallReturnVariant::FailureU32U32 as u32; + *a1 = usize::from(e) as u32; + *a2 = data0; + *a3 = data1; + } + TRD104SyscallReturn::FailureU64(e, data0) => { + let (data0_msb, data0_lsb) = u64_to_be_u32s(data0); + *a0 = TRD104SyscallReturnVariant::FailureU64 as u32; + *a1 = usize::from(e) as u32; + *a2 = data0_lsb; + *a3 = data0_msb; + } + TRD104SyscallReturn::Success => { + *a0 = TRD104SyscallReturnVariant::Success as u32; + } + TRD104SyscallReturn::SuccessU32(data0) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32 as u32; + *a1 = data0; + } + TRD104SyscallReturn::SuccessU32U32(data0, data1) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32 as u32; + *a1 = data0; + *a2 = data1; + } + TRD104SyscallReturn::SuccessU32U32U32(data0, data1, data2) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32U32 as u32; + *a1 = data0; + *a2 = data1; + *a3 = data2; + } + TRD104SyscallReturn::SuccessU64(data0) => { + let (data0_msb, data0_lsb) = u64_to_be_u32s(data0); + + *a0 = TRD104SyscallReturnVariant::SuccessU64 as u32; + *a1 = data0_lsb; + *a2 = data0_msb; + } + TRD104SyscallReturn::SuccessU32U64(data0, data1) => { + let (data1_msb, data1_lsb) = u64_to_be_u32s(data1); + + *a0 = TRD104SyscallReturnVariant::SuccessU32U64 as u32; + *a1 = data0; + *a2 = data1_lsb; + *a3 = data1_msb; + } + TRD104SyscallReturn::AllowReadWriteSuccess(ptr, len) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32 as u32; + *a1 = ptr as u32; + *a2 = len as u32; + } + TRD104SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32 as u32; + *a1 = ptr as u32; + *a2 = len as u32; + } + TRD104SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { + *a0 = TRD104SyscallReturnVariant::FailureU32U32 as u32; + *a1 = usize::from(err) as u32; + *a2 = ptr as u32; + *a3 = len as u32; + } + TRD104SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { + *a0 = TRD104SyscallReturnVariant::FailureU32U32 as u32; + *a1 = usize::from(err) as u32; + *a2 = ptr as u32; + *a3 = len as u32; + } + TRD104SyscallReturn::AllowReadOnlySuccess(ptr, len) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32 as u32; + *a1 = ptr as u32; + *a2 = len as u32; + } + TRD104SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { + *a0 = TRD104SyscallReturnVariant::FailureU32U32 as u32; + *a1 = usize::from(err) as u32; + *a2 = ptr as u32; + *a3 = len as u32; + } + TRD104SyscallReturn::SubscribeSuccess(ptr, data) => { + *a0 = TRD104SyscallReturnVariant::SuccessU32U32 as u32; + *a1 = ptr as u32; + *a2 = data as u32; + } + TRD104SyscallReturn::SubscribeFailure(err, ptr, data) => { + *a0 = TRD104SyscallReturnVariant::FailureU32U32 as u32; + *a1 = usize::from(err) as u32; + *a2 = ptr as u32; + *a3 = data as u32; + } + TRD104SyscallReturn::YieldWaitFor(data0, data1, data2) => { + *a0 = data0 as u32; + *a1 = data1 as u32; + *a2 = data2 as u32; + } + } +} + +/// Enumeration of all system call return type variant identifiers, as a +/// superset of those defined in [`TRD104`]. This is not yet standardized. +/// +/// Each variant is associated with the respective variant identifier that would +/// be passed along with the return value to userspace. +#[repr(u32)] +#[derive(Copy, Clone, Debug)] +pub enum SyscallReturnVariant { + Failure = 0, + FailureU32 = 1, + FailureU32U32 = 2, + FailureU64 = 3, + FailurePtrUsize = 4, + FailurePtrPtr = 5, + Success = 128, + SuccessU32 = 129, + SuccessU32U32 = 130, + SuccessU64 = 131, + SuccessU32U32U32 = 132, + SuccessU32U64 = 133, + SuccessUsize = 134, + SuccessPtr = 135, + SuccessPtrUsize = 136, + SuccessPtrPtr = 137, +} + +/// Encode the system call return value into 4 registers, with an encoding that +/// is a superset of the return variants specified in TRD104. This extension +/// behaves identical to TRD104 for the [`SyscallReturn`] variants that are +/// defined in TRD104. This does not implement any standardized encoding. +/// +/// On 64-bit platforms, both 64-bit and usize values are passed as a single register, +/// Does not handle usize other than 4 and 8 bytes. +/// Pointers from allow'd buffers have permissions and length reattached matching +/// those that were checked at the syscall boundary. +pub fn encode_syscall_return_usize( + syscall_return: &SyscallReturn, + a0: &mut CapabilityPtr, + a1: &mut CapabilityPtr, + a2: &mut CapabilityPtr, + a3: &mut CapabilityPtr, +) { + // On 32-bit CHERI, given that capabilities cannot be used as 64-bit integers, 64-bit + // integers will still be returned as two 32-bit values in different registers. + fn write_64(a: &mut CapabilityPtr, b: &mut CapabilityPtr, val: u64) { + let is_64_bit = core::mem::size_of::() == 8; + if !is_64_bit { + let (msb, lsb) = u64_to_be_u32s(val); + *a = (lsb as usize).into(); + *b = (msb as usize).into(); + } else { + *a = (val as usize).into(); + } + } + + match *syscall_return { + SyscallReturn::Failure(e) => { + *a0 = (SyscallReturnVariant::Failure as usize).into(); + *a1 = (usize::from(e)).into(); + } + SyscallReturn::FailureU32(e, data0) => { + *a0 = (SyscallReturnVariant::FailureU32 as usize).into(); + *a1 = usize::from(e).into(); + *a2 = (data0 as usize).into(); + } + SyscallReturn::FailureU32U32(e, data0, data1) => { + *a0 = (SyscallReturnVariant::FailureU32U32 as usize).into(); + *a1 = (usize::from(e)).into(); + *a2 = (data0 as usize).into(); + *a3 = (data1 as usize).into(); + } + SyscallReturn::FailureU64(e, data0) => { + *a0 = (SyscallReturnVariant::FailureU64 as usize).into(); + *a1 = (usize::from(e)).into(); + write_64(a2, a3, data0) + } + SyscallReturn::Success => { + *a0 = (SyscallReturnVariant::Success as usize).into(); + } + SyscallReturn::SuccessU32(data0) => { + *a0 = (SyscallReturnVariant::SuccessU32 as usize).into(); + *a1 = (data0 as usize).into(); + } + SyscallReturn::SuccessU32U32(data0, data1) => { + *a0 = (SyscallReturnVariant::SuccessU32U32 as usize).into(); + *a1 = (data0 as usize).into(); + *a2 = (data1 as usize).into(); + } + SyscallReturn::SuccessU32U32U32(data0, data1, data2) => { + *a0 = (SyscallReturnVariant::SuccessU32U32U32 as usize).into(); + *a1 = (data0 as usize).into(); + *a2 = (data1 as usize).into(); + *a3 = (data2 as usize).into(); + } + SyscallReturn::SuccessU64(data0) => { + *a0 = (SyscallReturnVariant::SuccessU64 as usize).into(); + write_64(a1, a2, data0); + } + SyscallReturn::SuccessU32U64(data0, data1) => { + *a0 = (SyscallReturnVariant::SuccessU32U64 as usize).into(); + *a1 = (data0 as usize).into(); + write_64(a2, a3, data1); + } + SyscallReturn::AllowReadWriteSuccess(ptr, len) => { + *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); + *a1 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::ReadWrite, + ); + *a2 = len.into(); + } + SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { + *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); + *a1 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ); + *a2 = len.into(); + } + SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { + *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::ReadWrite, + ); + *a3 = len.into(); + } + SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { + *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ); + *a3 = len.into(); + } + SyscallReturn::AllowReadOnlySuccess(ptr, len) => { + *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); + *a1 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ); + *a2 = len.into(); + } + SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { + *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ); + *a3 = len.into(); + } + SyscallReturn::SubscribeSuccess(ptr, data) => { + *a0 = (SyscallReturnVariant::SuccessPtrPtr as usize).into(); + *a1 = (ptr as usize).into(); + *a2 = data.into(); + } + SyscallReturn::SubscribeFailure(err, ptr, data) => { + *a0 = (SyscallReturnVariant::FailurePtrPtr as usize).into(); + *a1 = (usize::from(err)).into(); + *a2 = (ptr as usize).into(); + *a3 = data.into(); + } + SyscallReturn::SuccessPtr(ptr) => { + *a0 = (SyscallReturnVariant::SuccessPtr as usize).into(); + *a1 = ptr; + } + SyscallReturn::YieldWaitFor(data0, data1, data2) => { + *a0 = data0.into(); + *a1 = data1.into(); + *a2 = data2.into(); + } + SyscallReturn::SuccessUsize(data) => { + *a0 = (SyscallReturnVariant::SuccessUsize as usize).into(); + *a1 = data.into(); + } + } +} diff --git a/kernel/src/utilities/mod.rs b/kernel/src/utilities/mod.rs index b9b53bacfe..caa8d22f24 100644 --- a/kernel/src/utilities/mod.rs +++ b/kernel/src/utilities/mod.rs @@ -4,6 +4,7 @@ //! Utility functions and macros provided by the kernel crate. +pub mod arch_helpers; pub mod binary_write; pub mod capability_ptr; pub mod copy_slice; From 7ea663ffa2d188d2da560426890abaa0eb3214c5 Mon Sep 17 00:00:00 2001 From: Leon Schuermann Date: Mon, 11 Nov 2024 21:52:11 -0500 Subject: [PATCH 34/45] kernel: fix rustdoc links --- kernel/src/syscall.rs | 6 ++++-- kernel/src/utilities/arch_helpers.rs | 2 +- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index 09654593ef..d7ccdbdcce 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -375,8 +375,10 @@ impl Syscall { /// /// This struct operates over primitive types such as integers of fixed length /// and pointers. It is constructed by the scheduler and passed down to the -/// architecture to be encoded into registers, using the provided -/// [`encode_syscall_return`](SyscallReturn::encode_syscall_return) method. +/// architecture to be encoded into registers. Architectures may use the various +/// helper functions defined in +/// [`utilities::arch_helpers`](crate::utilities::arch_helpers), but are free to +/// define their own ABI and encoding. /// /// Capsules do not use this struct. Capsules use higher level Rust types (e.g. /// [`ReadWriteProcessBuffer`](crate::processbuffer::ReadWriteProcessBuffer) and diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index 90d3359f46..7ce631c9a0 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -244,7 +244,7 @@ pub fn encode_syscall_return_32bit_trd104( } /// Enumeration of all system call return type variant identifiers, as a -/// superset of those defined in [`TRD104`]. This is not yet standardized. +/// superset of those defined in TRD 104. This is not yet standardized. /// /// Each variant is associated with the respective variant identifier that would /// be passed along with the return value to userspace. From 670a5d00d4d9e4d4ed2664ee1c96ab16236cb59a Mon Sep 17 00:00:00 2001 From: Leon Schuermann Date: Wed, 13 Nov 2024 17:27:43 -0500 Subject: [PATCH 35/45] kernel: handle_syscall: elaborate on NonNull change for CapabilityPtr --- kernel/src/kernel.rs | 26 ++++++++++++++++++++------ 1 file changed, 20 insertions(+), 6 deletions(-) diff --git a/kernel/src/kernel.rs b/kernel/src/kernel.rs index 45ee440c35..9fe455d250 100644 --- a/kernel/src/kernel.rs +++ b/kernel/src/kernel.rs @@ -876,12 +876,26 @@ impl Kernel { subscribe_num: subdriver_number, }; - // TODO: when the compiler supports capability types bring this back - // as a NonNull type. https://github.com/tock/tock/issues/4134. - // First check if `upcall_ptr` is null. A null - // `upcall_ptr` will result in `None` here and - // represents the special "unsubscribe" operation. - // let ptr = NonNull::new(upcall_ptr); + // TODO: when the compiler supports capability types + // bring this back as a NonNull + // type. https://github.com/tock/tock/issues/4134. + // + // Previously, we had a NonNull type (that had a niche) + // here, and could wrap that in Option to fill the niche + // and handle the Null case. CapabilityPtr is filling + // the gap left by * const(), which does not have the + // niche and allows NULL internally. Having a CHERI + // capability type with a niche is (maybe?) predicated + // on having better compiler support. + // Option> is preferable here, and it should + // go back to it just as soon as we can express "non + // null capability". For now, checking for the null case + // is handled internally in each `map_or` call. + // + //First check if `upcall_ptr` is null. A null + //`upcall_ptr` will result in `None` here and + //represents the special "unsubscribe" operation. + //let ptr = NonNull::new(upcall_ptr); // For convenience create an `Upcall` type now. This is // just a data structure and doesn't do any checking or From 3810ca8eebed76d0c9d67cc3f1589e46c8fa5f23 Mon Sep 17 00:00:00 2001 From: Leon Schuermann Date: Wed, 13 Nov 2024 17:39:09 -0500 Subject: [PATCH 36/45] kernel/capability_ptr: make new_with_metadata an unsafe method --- kernel/src/process_standard.rs | 28 +++++---- kernel/src/utilities/arch_helpers.rs | 84 +++++++++++++++----------- kernel/src/utilities/capability_ptr.rs | 15 ++++- 3 files changed, 78 insertions(+), 49 deletions(-) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 0f9ca284f4..31d2836992 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -862,12 +862,14 @@ impl Process for ProcessStandard<'_, self.chip.mpu().configure_mpu(config); let base = self.mem_start() as usize; - let break_result = CapabilityPtr::new_with_metadata( - old_break as *const (), - base, - (new_break as usize) - base, - CapabilityPtrPermissions::ReadWrite, - ); + let break_result = unsafe { + CapabilityPtr::new_with_metadata( + old_break as *const (), + base, + (new_break as usize) - base, + CapabilityPtrPermissions::ReadWrite, + ) + }; Ok(break_result) } @@ -2147,12 +2149,14 @@ impl ProcessStandard<'_, C // We need to construct a capability with sufficient authority to cover all of a user's // code, with permissions to execute it. The entirety of flash is sufficient. - let init_fn = CapabilityPtr::new_with_metadata( - init_addr as *const (), - flash_start as usize, - (self.flash_end() as usize) - (flash_start as usize), - CapabilityPtrPermissions::Execute, - ); + let init_fn = unsafe { + CapabilityPtr::new_with_metadata( + init_addr as *const (), + flash_start as usize, + (self.flash_end() as usize) - (flash_start as usize), + CapabilityPtrPermissions::Execute, + ) + }; self.enqueue_task(Task::FunctionCall(FunctionCall { source: FunctionCallSource::Kernel, diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index 7ce631c9a0..61466f55a3 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -348,65 +348,77 @@ pub fn encode_syscall_return_usize( } SyscallReturn::AllowReadWriteSuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ); + *a1 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::ReadWrite, + ) + }; *a2 = len.into(); } SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); + *a1 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ) + }; *a2 = len.into(); } SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ); + *a2 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::ReadWrite, + ) + }; *a3 = len.into(); } SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); + *a2 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ) + }; *a3 = len.into(); } SyscallReturn::AllowReadOnlySuccess(ptr, len) => { *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); + *a1 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ) + }; *a2 = len.into(); } SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); *a1 = (usize::from(err)).into(); - *a2 = CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ); + *a2 = unsafe { + CapabilityPtr::new_with_metadata( + ptr as *const (), + ptr as usize, + len, + CapabilityPtrPermissions::Read, + ) + }; *a3 = len.into(); } SyscallReturn::SubscribeSuccess(ptr, data) => { diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index 3eab391f1c..d60a7d4640 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -105,10 +105,23 @@ impl CapabilityPtr { /// Construct a [`CapabilityPtr`] from a raw pointer, with authority ranging over /// [`base`, `base + length`) and permissions `perms`. + /// /// Provenance note: may derive from a pointer other than the input to provide something with /// valid provenance to justify the other arguments. + /// + /// ## Safety + /// + /// Constructing a [`CapabilityPtr`] with metadata may convey authority to + /// dereference this pointer, such as on userspace. When these pointers + /// serve as the only memory isolation primitive in the system, this method + /// can thus break Tock's isolation model. As semi-trusted kernel code can + /// name this type and method, it is thus marked as `unsafe`. + /// + /// TODO: Once Tock supports hardware that uses the [`CapabilityPtr`]'s + /// metdata to convey authority, this comment should incorporate the exact + /// safety conditions of this function. #[inline] - pub fn new_with_metadata( + pub unsafe fn new_with_metadata( ptr: *const (), _base: usize, _length: usize, From 0709e6a41a943e453794b3592dcff1091b717a90 Mon Sep 17 00:00:00 2001 From: Leon Schuermann Date: Wed, 13 Nov 2024 17:45:03 -0500 Subject: [PATCH 37/45] kernel/arch_helpers: remove 32bit infix from encode_syscall_return_trd104 --- arch/cortex-m/src/syscall.rs | 2 +- arch/rv32i/src/syscall.rs | 2 +- kernel/src/utilities/arch_helpers.rs | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/arch/cortex-m/src/syscall.rs b/arch/cortex-m/src/syscall.rs index 840240cd67..773158ba70 100644 --- a/arch/cortex-m/src/syscall.rs +++ b/arch/cortex-m/src/syscall.rs @@ -195,7 +195,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall // // Refer to // https://doc.rust-lang.org/std/primitive.pointer.html#safety-13 - kernel::utilities::arch_helpers::encode_syscall_return_32bit_trd104( + kernel::utilities::arch_helpers::encode_syscall_return_trd104( &kernel::utilities::arch_helpers::TRD104SyscallReturn::from_syscall_return( return_value, ), diff --git a/arch/rv32i/src/syscall.rs b/arch/rv32i/src/syscall.rs index d787020741..0ea6aa9507 100644 --- a/arch/rv32i/src/syscall.rs +++ b/arch/rv32i/src/syscall.rs @@ -172,7 +172,7 @@ impl kernel::syscall::UserspaceKernelBoundary for SysCall { let (a1slice, r) = r.split_at_mut(R_A2 - R_A1); let (a2slice, a3slice) = r.split_at_mut(R_A3 - R_A2); - kernel::utilities::arch_helpers::encode_syscall_return_32bit_trd104( + kernel::utilities::arch_helpers::encode_syscall_return_trd104( &kernel::utilities::arch_helpers::TRD104SyscallReturn::from_syscall_return( return_value, ), diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index 61466f55a3..4198eafa15 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -128,7 +128,7 @@ impl TRD104SyscallReturn { /// Encode the system call return value into 4 registers, following the encoding /// specified in TRD104. Architectures which do not follow TRD104 are free to /// define their own encoding. -pub fn encode_syscall_return_32bit_trd104( +pub fn encode_syscall_return_trd104( syscall_return: &TRD104SyscallReturn, a0: &mut u32, a1: &mut u32, From 1ebf5090aac086db3f19a3b301e7013131473764 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:11:43 -0800 Subject: [PATCH 38/45] kernel: remove CapabilityPtr aliases for now Defer choosing ergonomic/expressive aliases or types for later --- kernel/src/grant.rs | 5 +++-- kernel/src/upcall.rs | 14 ++++---------- 2 files changed, 7 insertions(+), 12 deletions(-) diff --git a/kernel/src/grant.rs b/kernel/src/grant.rs index dbe931ccac..d8522d9b5c 100644 --- a/kernel/src/grant.rs +++ b/kernel/src/grant.rs @@ -138,6 +138,7 @@ use crate::process::{Error, Process, ProcessCustomGrantIdentifier, ProcessId}; use crate::processbuffer::{ReadOnlyProcessBuffer, ReadWriteProcessBuffer}; use crate::processbuffer::{ReadOnlyProcessBufferRef, ReadWriteProcessBufferRef}; use crate::upcall::{Upcall, UpcallError, UpcallId}; +use crate::utilities::capability_ptr::CapabilityPtr; use crate::ErrorCode; /// Tracks how many upcalls a grant instance supports automatically. @@ -707,8 +708,8 @@ impl<'a> GrantKernelData<'a> { #[repr(C)] #[derive(Default)] struct SavedUpcall { - appdata: crate::upcall::AppdataType, - fn_ptr: crate::upcall::FnPtrType, + appdata: CapabilityPtr, + fn_ptr: CapabilityPtr, } /// A minimal representation of a read-only allow from app, used for storing a diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index 0ade28698a..c727fbd95a 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -69,12 +69,6 @@ pub enum UpcallError { KernelError, } -// FIXME: When we get CHERI compiler support, these can go back to the proper types -// Google-internal issue: b/274586199 -// https://github.com/tock/tock/issues/4134 -pub(crate) type AppdataType = CapabilityPtr; -pub(crate) type FnPtrType = CapabilityPtr; - /// Type for calling an upcall in a process. /// /// This is essentially a wrapper around a function pointer with associated @@ -88,7 +82,7 @@ pub(crate) struct Upcall { pub(crate) upcall_id: UpcallId, /// The application data passed by the app when `subscribe()` was called. - pub(crate) appdata: AppdataType, + pub(crate) appdata: CapabilityPtr, /// A pointer to the first instruction of the function in the app that /// corresponds to this upcall. @@ -96,15 +90,15 @@ pub(crate) struct Upcall { /// If this value is `None`, this is a null upcall, which cannot actually be /// scheduled. An `Upcall` can be null when it is first created, or after an /// app unsubscribes from an upcall. - pub(crate) fn_ptr: FnPtrType, + pub(crate) fn_ptr: CapabilityPtr, } impl Upcall { pub(crate) fn new( process_id: ProcessId, upcall_id: UpcallId, - appdata: AppdataType, - fn_ptr: FnPtrType, + appdata: CapabilityPtr, + fn_ptr: CapabilityPtr, ) -> Upcall { Upcall { process_id, From d23676d4179dd955727ef26a4677bea5a9a32855 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:14:34 -0800 Subject: [PATCH 39/45] kernel: remove stale reference in upcall doc --- kernel/src/upcall.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/kernel/src/upcall.rs b/kernel/src/upcall.rs index c727fbd95a..213f06aa25 100644 --- a/kernel/src/upcall.rs +++ b/kernel/src/upcall.rs @@ -87,9 +87,9 @@ pub(crate) struct Upcall { /// A pointer to the first instruction of the function in the app that /// corresponds to this upcall. /// - /// If this value is `None`, this is a null upcall, which cannot actually be - /// scheduled. An `Upcall` can be null when it is first created, or after an - /// app unsubscribes from an upcall. + /// If this value is `null`, it should not actually be + /// scheduled. An `Upcall` can be null when it is first created, + /// or after an app unsubscribes from an upcall. pub(crate) fn_ptr: CapabilityPtr, } From 38a518845b7d5cb91c59fb81a3c19fc49908bc48 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:16:33 -0800 Subject: [PATCH 40/45] kernel: remove unused syscall return variant Wait until we have something actually using it to implement --- kernel/src/utilities/arch_helpers.rs | 206 --------------------------- 1 file changed, 206 deletions(-) diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index 4198eafa15..af48807f9b 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -10,7 +10,6 @@ //! `kernel` crate as all `arch` crates already depend on it. use crate::syscall::SyscallReturn; -use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; use crate::ErrorCode; /// Helper function to split a [`u64`] into a higher and lower [`u32`]. @@ -242,208 +241,3 @@ pub fn encode_syscall_return_trd104( } } } - -/// Enumeration of all system call return type variant identifiers, as a -/// superset of those defined in TRD 104. This is not yet standardized. -/// -/// Each variant is associated with the respective variant identifier that would -/// be passed along with the return value to userspace. -#[repr(u32)] -#[derive(Copy, Clone, Debug)] -pub enum SyscallReturnVariant { - Failure = 0, - FailureU32 = 1, - FailureU32U32 = 2, - FailureU64 = 3, - FailurePtrUsize = 4, - FailurePtrPtr = 5, - Success = 128, - SuccessU32 = 129, - SuccessU32U32 = 130, - SuccessU64 = 131, - SuccessU32U32U32 = 132, - SuccessU32U64 = 133, - SuccessUsize = 134, - SuccessPtr = 135, - SuccessPtrUsize = 136, - SuccessPtrPtr = 137, -} - -/// Encode the system call return value into 4 registers, with an encoding that -/// is a superset of the return variants specified in TRD104. This extension -/// behaves identical to TRD104 for the [`SyscallReturn`] variants that are -/// defined in TRD104. This does not implement any standardized encoding. -/// -/// On 64-bit platforms, both 64-bit and usize values are passed as a single register, -/// Does not handle usize other than 4 and 8 bytes. -/// Pointers from allow'd buffers have permissions and length reattached matching -/// those that were checked at the syscall boundary. -pub fn encode_syscall_return_usize( - syscall_return: &SyscallReturn, - a0: &mut CapabilityPtr, - a1: &mut CapabilityPtr, - a2: &mut CapabilityPtr, - a3: &mut CapabilityPtr, -) { - // On 32-bit CHERI, given that capabilities cannot be used as 64-bit integers, 64-bit - // integers will still be returned as two 32-bit values in different registers. - fn write_64(a: &mut CapabilityPtr, b: &mut CapabilityPtr, val: u64) { - let is_64_bit = core::mem::size_of::() == 8; - if !is_64_bit { - let (msb, lsb) = u64_to_be_u32s(val); - *a = (lsb as usize).into(); - *b = (msb as usize).into(); - } else { - *a = (val as usize).into(); - } - } - - match *syscall_return { - SyscallReturn::Failure(e) => { - *a0 = (SyscallReturnVariant::Failure as usize).into(); - *a1 = (usize::from(e)).into(); - } - SyscallReturn::FailureU32(e, data0) => { - *a0 = (SyscallReturnVariant::FailureU32 as usize).into(); - *a1 = usize::from(e).into(); - *a2 = (data0 as usize).into(); - } - SyscallReturn::FailureU32U32(e, data0, data1) => { - *a0 = (SyscallReturnVariant::FailureU32U32 as usize).into(); - *a1 = (usize::from(e)).into(); - *a2 = (data0 as usize).into(); - *a3 = (data1 as usize).into(); - } - SyscallReturn::FailureU64(e, data0) => { - *a0 = (SyscallReturnVariant::FailureU64 as usize).into(); - *a1 = (usize::from(e)).into(); - write_64(a2, a3, data0) - } - SyscallReturn::Success => { - *a0 = (SyscallReturnVariant::Success as usize).into(); - } - SyscallReturn::SuccessU32(data0) => { - *a0 = (SyscallReturnVariant::SuccessU32 as usize).into(); - *a1 = (data0 as usize).into(); - } - SyscallReturn::SuccessU32U32(data0, data1) => { - *a0 = (SyscallReturnVariant::SuccessU32U32 as usize).into(); - *a1 = (data0 as usize).into(); - *a2 = (data1 as usize).into(); - } - SyscallReturn::SuccessU32U32U32(data0, data1, data2) => { - *a0 = (SyscallReturnVariant::SuccessU32U32U32 as usize).into(); - *a1 = (data0 as usize).into(); - *a2 = (data1 as usize).into(); - *a3 = (data2 as usize).into(); - } - SyscallReturn::SuccessU64(data0) => { - *a0 = (SyscallReturnVariant::SuccessU64 as usize).into(); - write_64(a1, a2, data0); - } - SyscallReturn::SuccessU32U64(data0, data1) => { - *a0 = (SyscallReturnVariant::SuccessU32U64 as usize).into(); - *a1 = (data0 as usize).into(); - write_64(a2, a3, data1); - } - SyscallReturn::AllowReadWriteSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ) - }; - *a2 = len.into(); - } - SyscallReturn::UserspaceReadableAllowSuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ) - }; - *a2 = len.into(); - } - SyscallReturn::AllowReadWriteFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); - *a1 = (usize::from(err)).into(); - *a2 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::ReadWrite, - ) - }; - *a3 = len.into(); - } - SyscallReturn::UserspaceReadableAllowFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); - *a1 = (usize::from(err)).into(); - *a2 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ) - }; - *a3 = len.into(); - } - SyscallReturn::AllowReadOnlySuccess(ptr, len) => { - *a0 = (SyscallReturnVariant::SuccessPtrUsize as usize).into(); - *a1 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ) - }; - *a2 = len.into(); - } - SyscallReturn::AllowReadOnlyFailure(err, ptr, len) => { - *a0 = (SyscallReturnVariant::FailurePtrUsize as usize).into(); - *a1 = (usize::from(err)).into(); - *a2 = unsafe { - CapabilityPtr::new_with_metadata( - ptr as *const (), - ptr as usize, - len, - CapabilityPtrPermissions::Read, - ) - }; - *a3 = len.into(); - } - SyscallReturn::SubscribeSuccess(ptr, data) => { - *a0 = (SyscallReturnVariant::SuccessPtrPtr as usize).into(); - *a1 = (ptr as usize).into(); - *a2 = data.into(); - } - SyscallReturn::SubscribeFailure(err, ptr, data) => { - *a0 = (SyscallReturnVariant::FailurePtrPtr as usize).into(); - *a1 = (usize::from(err)).into(); - *a2 = (ptr as usize).into(); - *a3 = data.into(); - } - SyscallReturn::SuccessPtr(ptr) => { - *a0 = (SyscallReturnVariant::SuccessPtr as usize).into(); - *a1 = ptr; - } - SyscallReturn::YieldWaitFor(data0, data1, data2) => { - *a0 = data0.into(); - *a1 = data1.into(); - *a2 = data2.into(); - } - SyscallReturn::SuccessUsize(data) => { - *a0 = (SyscallReturnVariant::SuccessUsize as usize).into(); - *a1 = data.into(); - } - } -} From 08caaa518e8cdc76f5f4a40bc87acb315a35a02b Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:20:24 -0800 Subject: [PATCH 41/45] kernel: rename capability_ptr constructor --- kernel/src/process_standard.rs | 6 +++--- kernel/src/utilities/capability_ptr.rs | 8 ++++---- 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/kernel/src/process_standard.rs b/kernel/src/process_standard.rs index 31d2836992..f5c89fd751 100644 --- a/kernel/src/process_standard.rs +++ b/kernel/src/process_standard.rs @@ -863,7 +863,7 @@ impl Process for ProcessStandard<'_, let base = self.mem_start() as usize; let break_result = unsafe { - CapabilityPtr::new_with_metadata( + CapabilityPtr::new_with_authority( old_break as *const (), base, (new_break as usize) - base, @@ -1978,7 +1978,7 @@ impl ProcessStandard<'_, C // We need to construct a capability with sufficient authority to cover all of a user's // code, with permissions to execute it. The entirety of flash is sufficient. - let init_fn = CapabilityPtr::new_with_metadata( + let init_fn = CapabilityPtr::new_with_authority( init_addr as *const (), fn_base, fn_len, @@ -2150,7 +2150,7 @@ impl ProcessStandard<'_, C // code, with permissions to execute it. The entirety of flash is sufficient. let init_fn = unsafe { - CapabilityPtr::new_with_metadata( + CapabilityPtr::new_with_authority( init_addr as *const (), flash_start as usize, (self.flash_end() as usize) - (flash_start as usize), diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index d60a7d4640..7a41762cf5 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -117,11 +117,11 @@ impl CapabilityPtr { /// can thus break Tock's isolation model. As semi-trusted kernel code can /// name this type and method, it is thus marked as `unsafe`. /// - /// TODO: Once Tock supports hardware that uses the [`CapabilityPtr`]'s - /// metdata to convey authority, this comment should incorporate the exact - /// safety conditions of this function. + // TODO: Once Tock supports hardware that uses the [`CapabilityPtr`]'s + // metdata to convey authority, this comment should incorporate the exact + // safety conditions of this function. #[inline] - pub unsafe fn new_with_metadata( + pub unsafe fn new_with_authority( ptr: *const (), _base: usize, _length: usize, From c6156c8be64c7b835d0ff5698edb26abc44b5cc2 Mon Sep 17 00:00:00 2001 From: Amit Levy Date: Wed, 13 Nov 2024 16:23:04 -0800 Subject: [PATCH 42/45] kernel: documentation nits Co-authored-by: Brad Campbell --- kernel/src/syscall.rs | 3 +-- kernel/src/utilities/capability_ptr.rs | 2 +- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index d7ccdbdcce..d2692d1bf1 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -377,8 +377,7 @@ impl Syscall { /// and pointers. It is constructed by the scheduler and passed down to the /// architecture to be encoded into registers. Architectures may use the various /// helper functions defined in -/// [`utilities::arch_helpers`](crate::utilities::arch_helpers), but are free to -/// define their own ABI and encoding. +/// [`utilities::arch_helpers`](crate::utilities::arch_helpers). /// /// Capsules do not use this struct. Capsules use higher level Rust types (e.g. /// [`ReadWriteProcessBuffer`](crate::processbuffer::ReadWriteProcessBuffer) and diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index 7a41762cf5..241912761f 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -112,7 +112,7 @@ impl CapabilityPtr { /// ## Safety /// /// Constructing a [`CapabilityPtr`] with metadata may convey authority to - /// dereference this pointer, such as on userspace. When these pointers + /// dereference this pointer, such as in userspace. When these pointers /// serve as the only memory isolation primitive in the system, this method /// can thus break Tock's isolation model. As semi-trusted kernel code can /// name this type and method, it is thus marked as `unsafe`. From a803e1e27910444d94e247129642c06a962c7189 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:35:46 -0800 Subject: [PATCH 43/45] kernel: remove SuccessUsize in favor of SuccessPtr Noting that SuccessPtr can return a `CapabilityPtr` with no authority granted. --- kernel/src/memop.rs | 16 ++++++++-------- kernel/src/syscall.rs | 10 +++------- kernel/src/utilities/arch_helpers.rs | 1 - kernel/src/utilities/capability_ptr.rs | 3 ++- 4 files changed, 13 insertions(+), 17 deletions(-) diff --git a/kernel/src/memop.rs b/kernel/src/memop.rs index d04ef315f0..7c4c8c81d9 100644 --- a/kernel/src/memop.rs +++ b/kernel/src/memop.rs @@ -56,22 +56,22 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall .unwrap_or(SyscallReturn::Failure(ErrorCode::NOMEM)), // Op Type 2: Process memory start - 2 => SyscallReturn::SuccessUsize(process.get_addresses().sram_start), + 2 => SyscallReturn::SuccessPtr(process.get_addresses().sram_start.into()), // Op Type 3: Process memory end - 3 => SyscallReturn::SuccessUsize(process.get_addresses().sram_end), + 3 => SyscallReturn::SuccessPtr(process.get_addresses().sram_end.into()), // Op Type 4: Process flash start - 4 => SyscallReturn::SuccessUsize(process.get_addresses().flash_start), + 4 => SyscallReturn::SuccessPtr(process.get_addresses().flash_start.into()), // Op Type 5: Process flash end - 5 => SyscallReturn::SuccessUsize(process.get_addresses().flash_end), + 5 => SyscallReturn::SuccessPtr(process.get_addresses().flash_end.into()), // Op Type 6: Grant region begin - 6 => SyscallReturn::SuccessUsize(process.get_addresses().sram_grant_start), + 6 => SyscallReturn::SuccessPtr(process.get_addresses().sram_grant_start.into()), // Op Type 7: Number of defined writeable regions in the TBF header. - 7 => SyscallReturn::SuccessUsize(process.number_writeable_flash_regions()), + 7 => SyscallReturn::SuccessU32(process.number_writeable_flash_regions() as u32), // Op Type 8: The start address of the writeable region indexed by r1. 8 => { @@ -80,7 +80,7 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessUsize(flash_start + offset) + SyscallReturn::SuccessPtr((flash_start + offset).into()) } } @@ -93,7 +93,7 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessUsize(flash_start + offset + size) + SyscallReturn::SuccessPtr((flash_start + offset + size).into()) } } diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index d2692d1bf1..b06f5d5dad 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -407,12 +407,9 @@ pub enum SyscallReturn { /// Generic success case, with an additional 32-bit and 64-bit data field SuccessU32U64(u32, u64), - /// Generic success case, with an additional usize data field - SuccessUsize(usize), - - /// Generic success case, with an additional pointer with metadata - /// On CHERI, this grants authority. - /// Access to this return is therefore privileged. + /// Generic success case, with an additional pointer. + /// This pointer may or may imply access permission to the + /// process. SuccessPtr(CapabilityPtr), // These following types are used by the scheduler so that it can return @@ -504,7 +501,6 @@ impl SyscallReturn { SyscallReturn::AllowReadOnlyFailure(_, _, _) => false, SyscallReturn::SubscribeFailure(_, _, _) => false, SyscallReturn::YieldWaitFor(_, _, _) => true, - SyscallReturn::SuccessUsize(_) => true, } } } diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index af48807f9b..186dde4604 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -116,7 +116,6 @@ impl TRD104SyscallReturn { SyscallReturn::YieldWaitFor(a, b, c) => TRD104SyscallReturn::YieldWaitFor(a, b, c), // Compatibility mapping: - SyscallReturn::SuccessUsize(a) => TRD104SyscallReturn::SuccessU32(a as u32), SyscallReturn::SuccessPtr(a) => { TRD104SyscallReturn::SuccessU32(a.as_ptr::<()>() as u32) } diff --git a/kernel/src/utilities/capability_ptr.rs b/kernel/src/utilities/capability_ptr.rs index 241912761f..bf77ad4695 100644 --- a/kernel/src/utilities/capability_ptr.rs +++ b/kernel/src/utilities/capability_ptr.rs @@ -61,7 +61,8 @@ impl From for usize { } impl From for CapabilityPtr { - /// Constructs a [`CapabilityPtr`] with a given address. + /// Constructs a [`CapabilityPtr`] with a given address and no authority + /// /// Provenance note: may have null provenance. #[inline] fn from(from: usize) -> Self { From 096536c10efc33da0c45fcc667d9b8ab702816fd Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 16:40:09 -0800 Subject: [PATCH 44/45] kernel: fix import order nit --- kernel/src/utilities/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kernel/src/utilities/mod.rs b/kernel/src/utilities/mod.rs index 38e78ffd65..8e146ac544 100644 --- a/kernel/src/utilities/mod.rs +++ b/kernel/src/utilities/mod.rs @@ -14,10 +14,10 @@ pub mod math; pub mod mut_imut_buffer; pub mod peripheral_management; pub mod static_init; -mod static_ref; pub mod storage_volume; pub mod streaming_process_slice; +mod static_ref; pub use self::static_ref::StaticRef; /// The Tock Register Interface. From 3c1876b9ad8248a1b6d12f5ec637e711871902f6 Mon Sep 17 00:00:00 2001 From: Amit Aryeh Levy Date: Wed, 13 Nov 2024 21:07:23 -0800 Subject: [PATCH 45/45] kernel: memop: add provenance to memop returns --- kernel/src/memop.rs | 61 ++++++++++++++++++++++++---- kernel/src/syscall.rs | 9 +++- kernel/src/utilities/arch_helpers.rs | 1 + 3 files changed, 62 insertions(+), 9 deletions(-) diff --git a/kernel/src/memop.rs b/kernel/src/memop.rs index 7c4c8c81d9..b50333a165 100644 --- a/kernel/src/memop.rs +++ b/kernel/src/memop.rs @@ -6,6 +6,7 @@ use crate::process::Process; use crate::syscall::SyscallReturn; +use crate::utilities::capability_ptr::{CapabilityPtr, CapabilityPtrPermissions}; use crate::ErrorCode; /// Handle the `memop` syscall. @@ -56,19 +57,51 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall .unwrap_or(SyscallReturn::Failure(ErrorCode::NOMEM)), // Op Type 2: Process memory start - 2 => SyscallReturn::SuccessPtr(process.get_addresses().sram_start.into()), + 2 => SyscallReturn::SuccessPtr(unsafe { + let addresses = process.get_addresses(); + CapabilityPtr::new_with_authority( + addresses.sram_start as *const _, + addresses.sram_start, + addresses.sram_app_brk - addresses.sram_start, + CapabilityPtrPermissions::ReadWrite, + ) + }), // Op Type 3: Process memory end - 3 => SyscallReturn::SuccessPtr(process.get_addresses().sram_end.into()), + 3 => SyscallReturn::SuccessPtr(unsafe { + let addresses = process.get_addresses(); + CapabilityPtr::new_with_authority( + addresses.sram_end as *const _, + addresses.sram_start, + addresses.sram_end - addresses.sram_start, + CapabilityPtrPermissions::ReadWrite, + ) + }), // Op Type 4: Process flash start - 4 => SyscallReturn::SuccessPtr(process.get_addresses().flash_start.into()), + 4 => SyscallReturn::SuccessPtr(unsafe { + let addresses = process.get_addresses(); + CapabilityPtr::new_with_authority( + addresses.flash_start as *const _, + addresses.flash_start, + addresses.flash_end - addresses.flash_start, + CapabilityPtrPermissions::Execute, + ) + }), // Op Type 5: Process flash end - 5 => SyscallReturn::SuccessPtr(process.get_addresses().flash_end.into()), + 5 => SyscallReturn::SuccessPtr(unsafe { + let addresses = process.get_addresses(); + CapabilityPtr::new_with_authority( + addresses.flash_end as *const _, + addresses.flash_start, + addresses.flash_end - addresses.flash_start, + CapabilityPtrPermissions::Execute, + ) + }), // Op Type 6: Grant region begin - 6 => SyscallReturn::SuccessPtr(process.get_addresses().sram_grant_start.into()), + 6 => SyscallReturn::SuccessAddr(process.get_addresses().sram_grant_start), // Op Type 7: Number of defined writeable regions in the TBF header. 7 => SyscallReturn::SuccessU32(process.number_writeable_flash_regions() as u32), @@ -80,7 +113,14 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessPtr((flash_start + offset).into()) + SyscallReturn::SuccessPtr(unsafe { + CapabilityPtr::new_with_authority( + (flash_start + offset) as *const _, + flash_start + offset, + size, + CapabilityPtrPermissions::ReadWrite, + ) + }) } } @@ -93,7 +133,14 @@ pub(crate) fn memop(process: &dyn Process, op_type: usize, r1: usize) -> Syscall if size == 0 { SyscallReturn::Failure(ErrorCode::FAIL) } else { - SyscallReturn::SuccessPtr((flash_start + offset + size).into()) + SyscallReturn::SuccessPtr(unsafe { + CapabilityPtr::new_with_authority( + (flash_start + offset + size) as *const _, + flash_start + offset, + size, + CapabilityPtrPermissions::ReadWrite, + ) + }) } } diff --git a/kernel/src/syscall.rs b/kernel/src/syscall.rs index b06f5d5dad..8f8564fe3d 100644 --- a/kernel/src/syscall.rs +++ b/kernel/src/syscall.rs @@ -407,9 +407,13 @@ pub enum SyscallReturn { /// Generic success case, with an additional 32-bit and 64-bit data field SuccessU32U64(u32, u64), + /// Generic success case with an additional address-sized value + /// that does not impute access permissions to the process. + SuccessAddr(usize), + /// Generic success case, with an additional pointer. - /// This pointer may or may imply access permission to the - /// process. + /// This pointer is provenance bearing and implies access + /// permission to the process. SuccessPtr(CapabilityPtr), // These following types are used by the scheduler so that it can return @@ -487,6 +491,7 @@ impl SyscallReturn { SyscallReturn::SuccessU32U32U32(_, _, _) => true, SyscallReturn::SuccessU64(_) => true, SyscallReturn::SuccessU32U64(_, _) => true, + SyscallReturn::SuccessAddr(_) => true, SyscallReturn::SuccessPtr(_) => true, SyscallReturn::AllowReadWriteSuccess(_, _) => true, SyscallReturn::UserspaceReadableAllowSuccess(_, _) => true, diff --git a/kernel/src/utilities/arch_helpers.rs b/kernel/src/utilities/arch_helpers.rs index 186dde4604..cdee406022 100644 --- a/kernel/src/utilities/arch_helpers.rs +++ b/kernel/src/utilities/arch_helpers.rs @@ -116,6 +116,7 @@ impl TRD104SyscallReturn { SyscallReturn::YieldWaitFor(a, b, c) => TRD104SyscallReturn::YieldWaitFor(a, b, c), // Compatibility mapping: + SyscallReturn::SuccessAddr(a) => TRD104SyscallReturn::SuccessU32(a as u32), SyscallReturn::SuccessPtr(a) => { TRD104SyscallReturn::SuccessU32(a.as_ptr::<()>() as u32) }