From ddf6d09136cd93df420a6bf570dd60034e983cf1 Mon Sep 17 00:00:00 2001 From: Berkus Decker Date: Thu, 5 May 2022 22:01:49 +0300 Subject: [PATCH] =?UTF-8?q?feat:=20=E2=9C=A8=20Switch=20mailboxes=20to=20c?= =?UTF-8?q?orrect=20DMA-backed=20storage=20by=20default?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Allocate DmaBackedMailboxStorage out of DMA_ALLOCATOR. Replace DMA bump_allocator with buddy_alloc. --- Cargo.lock | 7 +++ machine/Cargo.toml | 2 + machine/src/lib.rs | 38 ++++++++------ machine/src/platform/rpi3/mailbox.rs | 76 +++++++++++++++++++++++++--- 4 files changed, 100 insertions(+), 23 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 87769ee..9eabea1 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -67,6 +67,11 @@ version = "1.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" +[[package]] +name = "buddy-alloc" +version = "0.5.0" +source = "git+https://github.com/metta-systems/buddy-alloc?branch=feature/allocator-api#70c17c858100282252754981f13e9442b86e07e0" + [[package]] name = "bytes" version = "1.1.0" @@ -385,8 +390,10 @@ version = "0.0.1" dependencies = [ "bit_field", "bitflags", + "buddy-alloc", "cfg-if", "cortex-a", + "once_cell", "qemu-exit", "r0", "snafu", diff --git a/machine/Cargo.toml b/machine/Cargo.toml index ad72d91..c1f7ea5 100644 --- a/machine/Cargo.toml +++ b/machine/Cargo.toml @@ -38,3 +38,5 @@ bit_field = "0.10" bitflags = "1.3" cfg-if = "1.0" snafu = { version = "0.7", default-features = false } +buddy-alloc = { git = "https://github.com/metta-systems/buddy-alloc", version = "0.5.0", branch = "feature/allocator-api" } +once_cell = { version = "1.10.0", default-features = false, features = ["unstable"] } diff --git a/machine/src/lib.rs b/machine/src/lib.rs index b644d91..bc39adb 100644 --- a/machine/src/lib.rs +++ b/machine/src/lib.rs @@ -5,6 +5,7 @@ #![feature(format_args_nl)] #![feature(core_intrinsics)] #![feature(stmt_expr_attributes)] +#![feature(slice_ptr_get)] #![feature(nonnull_slice_from_raw_parts)] #![feature(custom_test_frameworks)] #![test_runner(crate::tests::test_runner)] @@ -18,6 +19,12 @@ #[cfg(not(target_arch = "aarch64"))] use architecture_not_supported_sorry; +use { + buddy_alloc::{BuddyAlloc, BuddyAllocParam}, + once_cell::unsync::Lazy, + platform::memory::map::virt::{DMA_HEAP_END, DMA_HEAP_START}, +}; + /// Architecture-specific code. #[macro_use] pub mod arch; @@ -41,19 +48,20 @@ pub static CONSOLE: sync::NullLock = sync::NullLock::new(devic /// The global allocator for DMA-able memory. That is, memory which is tagged /// non-cacheable in the page tables. #[allow(dead_code)] -static DMA_ALLOCATOR: sync::NullLock = - sync::NullLock::new(mm::BumpAllocator::new( - // @todo Init this after we loaded boot memory map - platform::memory::map::virt::DMA_HEAP_START as usize, - platform::memory::map::virt::DMA_HEAP_END as usize, - "Global DMA Allocator", - // Try the following arguments instead to see all mailbox operations - // fail. It will cause the allocator to use memory that are marked - // cacheable and therefore not DMA-safe. The answer from the VideoCore - // won't be received by the CPU because it reads an old cached value - // that resembles an error case instead. +static DMA_ALLOCATOR: sync::NullLock> = + sync::NullLock::new(Lazy::new(|| unsafe { + BuddyAlloc::new(BuddyAllocParam::new( + // @todo Init this after we loaded boot memory map + DMA_HEAP_START as *const u8, + DMA_HEAP_END - DMA_HEAP_START, + 64, + )) + })); +// Try the following arguments instead to see all mailbox operations +// fail. It will cause the allocator to use memory that are marked +// cacheable and therefore not DMA-safe. The answer from the VideoCore +// won't be received by the CPU because it reads an old cached value +// that resembles an error case instead. - // 0x00600000 as usize, - // 0x007FFFFF as usize, - // "Global Non-DMA Allocator", - )); +// 0x00600000 as usize, +// 0x007FFFFF as usize, diff --git a/machine/src/platform/rpi3/mailbox.rs b/machine/src/platform/rpi3/mailbox.rs index bac0533..e5c299e 100644 --- a/machine/src/platform/rpi3/mailbox.rs +++ b/machine/src/platform/rpi3/mailbox.rs @@ -13,8 +13,11 @@ use { super::BcmHost, - crate::{platform::MMIODerefWrapper, println}, + crate::{platform::MMIODerefWrapper, println, DMA_ALLOCATOR}, core::{ + alloc::{AllocError, Allocator, Layout}, + mem, + ptr::NonNull, result::Result as CoreResult, sync::atomic::{compiler_fence, Ordering}, }, @@ -31,14 +34,14 @@ use { /// The address for the buffer needs to be 16-byte aligned /// so that the VideoCore can handle it properly. /// The reason is that lowest 4 bits of the address will contain the channel number. -pub struct Mailbox> { +pub struct Mailbox> { registers: Registers, pub buffer: Storage, } /// Mailbox that is ready to be called. /// This prevents invalid use of the mailbox until it is fully prepared. -pub struct PreparedMailbox>( +pub struct PreparedMailbox>( Mailbox, ); @@ -96,6 +99,8 @@ pub enum MailboxError { Unknown, #[snafu(display("Timeout"))] Timeout, + #[snafu(display("AllocError"))] + Alloc, } pub type Result = CoreResult; @@ -111,7 +116,9 @@ pub trait MailboxOps { } pub trait MailboxStorage { - fn new() -> Self; + fn new() -> Result + where + Self: Sized; } pub trait MailboxStorageRef { @@ -127,11 +134,45 @@ pub struct LocalMailboxStorage { pub storage: [u32; N_SLOTS], } +pub struct DmaBackedMailboxStorage { + pub storage: *mut u32, +} + impl MailboxStorage for LocalMailboxStorage { - fn new() -> Self { - Self { + fn new() -> Result { + Ok(Self { storage: [0u32; N_SLOTS], - } + }) + } +} + +impl MailboxStorage for DmaBackedMailboxStorage { + fn new() -> Result { + Ok(Self { + storage: DMA_ALLOCATOR + .lock(|a| { + a.allocate( + Layout::from_size_align(N_SLOTS * mem::size_of::(), 16) + .map_err(|_| AllocError)?, + ) + }) + .map_err(|_| MailboxError::Alloc)? + .as_mut_ptr() as *mut u32, + }) + } +} + +impl Drop for DmaBackedMailboxStorage { + fn drop(&mut self) { + DMA_ALLOCATOR + .lock::<_, Result<()>>(|a| unsafe { + Ok(a.deallocate( + NonNull::new_unchecked(self.storage as *mut u8), + Layout::from_size_align(N_SLOTS * mem::size_of::(), 16) + .map_err(|_| MailboxError::Alloc)?, + )) + }) + .unwrap_or(()) } } @@ -154,6 +195,25 @@ impl MailboxStorageRef for LocalMailboxStorage { } } +impl MailboxStorageRef for DmaBackedMailboxStorage { + fn as_ref(&self) -> &[u32] { + unsafe { core::slice::from_raw_parts(self.storage.cast(), N_SLOTS) } + } + + fn as_mut(&mut self) -> &mut [u32] { + unsafe { core::slice::from_raw_parts_mut(self.storage.cast(), N_SLOTS) } + } + + fn as_ptr(&self) -> *const u32 { + self.storage.cast() + } + + // @todo Probably need a ResultMailbox for accessing data after call()? + fn value_at(&self, index: usize) -> u32 { + self.as_ref()[index] + } +} + /* * Source https://elinux.org/RPi_Framebuffer * Source for channels 8 and 9: https://github.com/raspberrypi/firmware/wiki/Mailboxes @@ -315,7 +375,7 @@ impl Mailbox< pub unsafe fn new(base_addr: usize) -> Result> { Ok(Mailbox { registers: Registers::new(base_addr), - buffer: Storage::new(), + buffer: Storage::new()?, }) }