mirror of
https://github.com/italicsjenga/agb.git
synced 2025-01-22 15:16:40 +11:00
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:
commit
627c6dad1f
1 changed files with 45 additions and 10 deletions
|
@ -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;
|
||||||
|
|
Loading…
Add table
Reference in a new issue