123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- use rustsbi::{HartMask, SbiRet};
- use spin::Mutex;
- use crate::board::SBI_IMPL;
- use crate::riscv_spec::current_hartid;
- use crate::sbi::fifo::{Fifo, FifoError};
- use crate::sbi::trap;
- use crate::sbi::trap_stack::ROOT_STACK;
- use core::sync::atomic::{AtomicU32, Ordering};
- /// Cell for managing remote fence operations between harts.
- pub(crate) struct RFenceCell {
- // Queue of fence operations with source hart ID
- queue: Mutex<Fifo<(RFenceContext, usize)>>,
- // Counter for tracking pending synchronization operations
- wait_sync_count: AtomicU32,
- }
- /// Context information for a remote fence operation.
- #[repr(C)]
- #[derive(Clone, Copy, Debug)]
- pub struct RFenceContext {
- /// Start address of memory region to fence.
- pub start_addr: usize,
- /// Size of memory region to fence.
- pub size: usize,
- /// Address space ID.
- pub asid: usize,
- /// Virtual machine ID.
- pub vmid: usize,
- /// Type of fence operation.
- pub op: RFenceType,
- }
- /// Types of remote fence operations supported.
- #[allow(unused)]
- #[derive(Clone, Copy, Debug)]
- pub enum RFenceType {
- /// Instruction fence.
- FenceI,
- /// Supervisor fence for virtual memory.
- SFenceVma,
- /// Supervisor fence for virtual memory with ASID.
- SFenceVmaAsid,
- /// Hypervisor fence for guest virtual memory with VMID.
- HFenceGvmaVmid,
- /// Hypervisor fence for guest virtual memory.
- HFenceGvma,
- /// Hypervisor fence for virtual machine virtual memory with ASID.
- HFenceVvmaAsid,
- /// Hypervisor fence for virtual machine virtual memory.
- HFenceVvma,
- }
- impl RFenceCell {
- /// Creates a new RFenceCell with empty queue and zero sync count.
- pub fn new() -> Self {
- Self {
- queue: Mutex::new(Fifo::new()),
- wait_sync_count: AtomicU32::new(0),
- }
- }
- /// Gets a local view of this fence cell for the current hart.
- #[inline]
- pub fn local(&self) -> LocalRFenceCell<'_> {
- LocalRFenceCell(self)
- }
- /// Gets a remote view of this fence cell for accessing from other harts.
- #[inline]
- pub fn remote(&self) -> RemoteRFenceCell<'_> {
- RemoteRFenceCell(self)
- }
- }
- // Mark RFenceCell as safe to share between threads
- unsafe impl Sync for RFenceCell {}
- unsafe impl Send for RFenceCell {}
- /// View of RFenceCell for operations on the current hart.
- pub struct LocalRFenceCell<'a>(&'a RFenceCell);
- /// View of RFenceCell for operations from other harts.
- pub struct RemoteRFenceCell<'a>(&'a RFenceCell);
- /// Gets the local fence context for the current hart.
- pub(crate) fn local_rfence() -> Option<LocalRFenceCell<'static>> {
- unsafe {
- ROOT_STACK
- .get_mut(current_hartid())
- .map(|x| x.hart_context().rfence.local())
- }
- }
- /// Gets the remote fence context for a specific hart.
- pub(crate) fn remote_rfence(hart_id: usize) -> Option<RemoteRFenceCell<'static>> {
- unsafe {
- ROOT_STACK
- .get_mut(hart_id)
- .map(|x| x.hart_context().rfence.remote())
- }
- }
- #[allow(unused)]
- impl LocalRFenceCell<'_> {
- /// Checks if all synchronization operations are complete.
- pub fn is_sync(&self) -> bool {
- self.0.wait_sync_count.load(Ordering::Relaxed) == 0
- }
- /// Increments the synchronization counter.
- pub fn add(&self) {
- self.0.wait_sync_count.fetch_add(1, Ordering::Relaxed);
- }
- /// Checks if the operation queue is empty.
- pub fn is_empty(&self) -> bool {
- self.0.queue.lock().is_empty()
- }
- /// Gets the next fence operation from the queue.
- pub fn get(&self) -> Option<(RFenceContext, usize)> {
- self.0.queue.lock().pop().ok()
- }
- /// Adds a fence operation to the queue, retrying if full.
- pub fn set(&self, ctx: RFenceContext) {
- let hart_id = current_hartid();
- loop {
- let mut queue = self.0.queue.lock();
- match queue.push((ctx, hart_id)) {
- Ok(_) => break,
- Err(FifoError::Full) => {
- drop(queue);
- trap::rfence_single_handler();
- }
- Err(_) => panic!("Unable to push fence ops to fifo"),
- }
- }
- }
- }
- #[allow(unused)]
- impl RemoteRFenceCell<'_> {
- /// Adds a fence operation to the queue from a remote hart.
- pub fn set(&self, ctx: RFenceContext) {
- let hart_id = current_hartid();
- loop {
- let mut queue = self.0.queue.lock();
- match queue.push((ctx, hart_id)) {
- Ok(_) => return,
- Err(FifoError::Full) => {
- drop(queue);
- trap::rfence_single_handler();
- }
- Err(_) => panic!("Unable to push fence ops to fifo"),
- }
- }
- }
- /// Decrements the synchronization counter.
- pub fn sub(&self) {
- self.0.wait_sync_count.fetch_sub(1, Ordering::Relaxed);
- }
- }
- /// Implementation of RISC-V remote fence operations.
- pub(crate) struct SbiRFence;
- /// Validates address range for fence operations
- #[inline(always)]
- fn validate_address_range(start_addr: usize, size: usize) -> Result<usize, SbiRet> {
- // Check page alignment using bitwise AND instead of modulo
- if start_addr & 0xFFF != 0 {
- return Err(SbiRet::invalid_address());
- }
- // Avoid checked_add by checking for overflow directly
- if size > usize::MAX - start_addr {
- return Err(SbiRet::invalid_address());
- }
- Ok(size)
- }
- /// Processes a remote fence operation by sending IPI to target harts.
- fn remote_fence_process(rfence_ctx: RFenceContext, hart_mask: HartMask) -> SbiRet {
- let sbi_ret = unsafe { SBI_IMPL.assume_init_mut() }
- .ipi
- .as_ref()
- .unwrap()
- .send_ipi_by_fence(hart_mask, rfence_ctx);
- sbi_ret
- }
- impl rustsbi::Fence for SbiRFence {
- /// Remote instruction fence for specified harts.
- fn remote_fence_i(&self, hart_mask: HartMask) -> SbiRet {
- remote_fence_process(
- RFenceContext {
- start_addr: 0,
- size: 0,
- asid: 0,
- vmid: 0,
- op: RFenceType::FenceI,
- },
- hart_mask,
- )
- }
- /// Remote supervisor fence for virtual memory on specified harts.
- fn remote_sfence_vma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
- let flush_size = match validate_address_range(start_addr, size) {
- Ok(size) => size,
- Err(e) => return e,
- };
- remote_fence_process(
- RFenceContext {
- start_addr,
- size: flush_size,
- asid: 0,
- vmid: 0,
- op: RFenceType::SFenceVma,
- },
- hart_mask,
- )
- }
- /// Remote supervisor fence for virtual memory with ASID on specified harts.
- fn remote_sfence_vma_asid(
- &self,
- hart_mask: HartMask,
- start_addr: usize,
- size: usize,
- asid: usize,
- ) -> SbiRet {
- let flush_size = match validate_address_range(start_addr, size) {
- Ok(size) => size,
- Err(e) => return e,
- };
- remote_fence_process(
- RFenceContext {
- start_addr,
- size: flush_size,
- asid,
- vmid: 0,
- op: RFenceType::SFenceVmaAsid,
- },
- hart_mask,
- )
- }
- }
|