Use pointers rather than pin to simplify the mixers buffer handling (#703)

Mentioned by @corwinkuiper that this might be a nicer way of handling
this

- [x] no changelog update needed
This commit is contained in:
Gwilym Inzani 2024-05-21 23:48:39 +01:00 committed by GitHub
commit 627c6dad1f
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -1,6 +1,5 @@
use core::cell::RefCell; use core::cell::RefCell;
use core::marker::PhantomData; use core::marker::PhantomData;
use core::pin::Pin;
use alloc::boxed::Box; use alloc::boxed::Box;
use alloc::vec::Vec; use alloc::vec::Vec;
@ -107,7 +106,7 @@ pub struct Mixer<'gba> {
// SAFETY: Has to go before buffer because it holds a reference to it // SAFETY: Has to go before buffer because it holds a reference to it
_interrupt_handler: InterruptHandler, _interrupt_handler: InterruptHandler,
buffer: Pin<Box<MixerBuffer, InternalAllocator>>, buffer: raw_box::RawBoxDrop<MixerBuffer, InternalAllocator>,
channels: [Option<SoundChannel>; 8], channels: [Option<SoundChannel>; 8],
indices: [i32; 8], indices: [i32; 8],
frequency: Frequency, frequency: Frequency,
@ -144,7 +143,8 @@ pub struct ChannelId(usize, i32);
impl Mixer<'_> { impl Mixer<'_> {
pub(super) fn new(frequency: Frequency) -> Self { pub(super) fn new(frequency: Frequency) -> Self {
let buffer = Box::pin_in(MixerBuffer::new(frequency), InternalAllocator); let buffer =
raw_box::RawBoxDrop::new(Box::new_in(MixerBuffer::new(frequency), InternalAllocator));
// SAFETY: you can only ever have 1 Mixer at a time // SAFETY: you can only ever have 1 Mixer at a time
let fifo_timer = unsafe { Timer::new(0) }; let fifo_timer = unsafe { Timer::new(0) };
@ -157,15 +157,21 @@ impl Mixer<'_> {
.set_interrupt(true) .set_interrupt(true)
.set_overflow_amount(frequency.buffer_size() as u16); .set_overflow_amount(frequency.buffer_size() as u16);
let buffer_pointer_for_interrupt_handler: &MixerBuffer = &buffer; struct SendPtr<T>(*const T);
unsafe impl<T> Send for SendPtr<T> {}
unsafe impl<T> Sync for SendPtr<T> {}
// SAFETY: dropping the lifetime, sound because interrupt handler dropped before the buffer is let ptr_for_interrupt_handler = SendPtr(&*buffer);
// In the case of the mixer being forgotten, both stay alive so okay
let buffer_pointer_for_interrupt_handler: &MixerBuffer = // SAFETY: the interrupt handler will be dropped before the buffer is so this never accesses
unsafe { core::mem::transmute(buffer_pointer_for_interrupt_handler) }; // freed memory. Also the dereference happens in a critical section to ensure that
// we don't end up accessing while dropping
let interrupt_handler = unsafe { let interrupt_handler = unsafe {
add_interrupt_handler(interrupt_timer.interrupt(), |cs| { add_interrupt_handler(interrupt_timer.interrupt(), move |cs| {
buffer_pointer_for_interrupt_handler.swap(cs); // needed to ensure that rust doesn't only capture the field
let _ = &ptr_for_interrupt_handler;
(*ptr_for_interrupt_handler.0).swap(cs);
}) })
}; };
@ -548,6 +554,35 @@ impl MixerBuffer {
} }
} }
mod raw_box {
use core::ops::Deref;
use alloc::boxed::Box;
pub struct RawBoxDrop<T, A: Clone + alloc::alloc::Allocator>(*mut T, A);
impl<T, A: Clone + alloc::alloc::Allocator> RawBoxDrop<T, A> {
pub fn new(inner: Box<T, A>) -> Self {
let (ptr, allocator) = Box::into_raw_with_allocator(inner);
Self(ptr, allocator)
}
}
impl<T, A: Clone + alloc::alloc::Allocator> Deref for RawBoxDrop<T, A> {
type Target = T;
fn deref(&self) -> &Self::Target {
unsafe { &*self.0 }
}
}
impl<T, A: Clone + alloc::alloc::Allocator> Drop for RawBoxDrop<T, A> {
fn drop(&mut self) {
unsafe { Box::from_raw_in(self.0, self.1.clone()) };
}
}
}
#[cfg(test)] #[cfg(test)]
mod test { mod test {
use crate::fixnum::num; use crate::fixnum::num;