cgb pcm registers

This commit is contained in:
Alex Janka 2023-04-24 11:35:38 +10:00
parent 5c61c642bf
commit c7b7a99cf9
3 changed files with 26 additions and 5 deletions

View file

@ -372,8 +372,8 @@ where
CgbIoAddress::Palette(address) => self.gpu.get_cgb_palette(address), CgbIoAddress::Palette(address) => self.gpu.get_cgb_palette(address),
CgbIoAddress::ObjPriority => self.gpu.get_obj_priority(), CgbIoAddress::ObjPriority => self.gpu.get_obj_priority(),
CgbIoAddress::WramBank => ((*selected) & 0b111) as u8, CgbIoAddress::WramBank => ((*selected) & 0b111) as u8,
CgbIoAddress::Pcm12 => todo!(), CgbIoAddress::Pcm12 => self.apu.get_pcm_1_2(),
CgbIoAddress::Pcm34 => todo!(), CgbIoAddress::Pcm34 => self.apu.get_pcm_3_4(),
CgbIoAddress::Unused(v) => { CgbIoAddress::Unused(v) => {
println!("attempt to get 0x{v:0>4X}"); println!("attempt to get 0x{v:0>4X}");
todo!() todo!()

View file

@ -153,6 +153,14 @@ impl Apu {
} }
} }
pub(crate) fn get_pcm_1_2(&self) -> u8 {
(self.channels.one.last & 0xF) | ((self.channels.two.last & 0xF0) << 4)
}
pub(crate) fn get_pcm_3_4(&self) -> u8 {
(self.channels.one.last & 0xF) | ((self.channels.two.last & 0xF0) << 4)
}
fn next_audio(&mut self) { fn next_audio(&mut self) {
self.out_buffer.append( self.out_buffer.append(
&mut self.converter.process( &mut self.converter.process(

View file

@ -127,6 +127,7 @@ impl DutyCycle {
#[derive(Serialize, Deserialize, Clone, Copy)] #[derive(Serialize, Deserialize, Clone, Copy)]
pub(super) struct PwmChannel { pub(super) struct PwmChannel {
pub(super) enabled: bool, pub(super) enabled: bool,
pub(super) last: u8,
sweep: Sweep, sweep: Sweep,
duty_cycle: DutyCycle, duty_cycle: DutyCycle,
length_enable: bool, length_enable: bool,
@ -143,6 +144,7 @@ impl PwmChannel {
let wavelength = 0x7FF; let wavelength = 0x7FF;
Self { Self {
enabled, enabled,
last: 0,
sweep: Sweep::default(), sweep: Sweep::default(),
duty_cycle: DutyCycle::Fifty, duty_cycle: DutyCycle::Fifty,
length_enable: false, length_enable: false,
@ -181,11 +183,13 @@ impl PwmChannel {
}) })
.collect() .collect()
} else { } else {
self.last = 0;
vec![0.; steps] vec![0.; steps]
} }
} }
fn dac(&self, digital: u8) -> f32 { fn dac(&mut self, digital: u8) -> f32 {
self.last = digital;
(((digital as f32) * (-2.)) + 1.) * ((self.envelope.current_volume as f32) / 15.) (((digital as f32) * (-2.)) + 1.) * ((self.envelope.current_volume as f32) / 15.)
} }
@ -339,6 +343,7 @@ impl WaveRam {
#[derive(Serialize, Deserialize, Clone, Copy)] #[derive(Serialize, Deserialize, Clone, Copy)]
pub(super) struct WaveChannel { pub(super) struct WaveChannel {
pub(super) enabled: bool, pub(super) enabled: bool,
pub(super) last: u8,
dac_enabled: bool, dac_enabled: bool,
length_enable: bool, length_enable: bool,
length_timer: u8, length_timer: u8,
@ -354,6 +359,7 @@ impl WaveChannel {
let wavelength = 0x7FF; let wavelength = 0x7FF;
Self { Self {
enabled, enabled,
last: 0,
dac_enabled: false, dac_enabled: false,
length_enable: false, length_enable: false,
length_timer: 0, length_timer: 0,
@ -390,14 +396,17 @@ impl WaveChannel {
}) })
.collect() .collect()
} else { } else {
self.last = 0;
vec![0.; steps] vec![0.; steps]
} }
} }
fn dac(&self, digital: u8) -> f32 { fn dac(&mut self, digital: u8) -> f32 {
if self.dac_enabled && self.volume != ShiftVolumePercent::Zero { if self.dac_enabled && self.volume != ShiftVolumePercent::Zero {
self.last = digital;
((((digital >> self.volume.as_shift_amount()) as f32) * (-2.)) + 1.) / 15. ((((digital >> self.volume.as_shift_amount()) as f32) * (-2.)) + 1.) / 15.
} else { } else {
self.last = 0;
0. 0.
} }
} }
@ -545,6 +554,7 @@ impl Default for Lfsr {
#[derive(Serialize, Deserialize, Clone, Copy)] #[derive(Serialize, Deserialize, Clone, Copy)]
pub(super) struct NoiseChannel { pub(super) struct NoiseChannel {
pub(super) enabled: bool, pub(super) enabled: bool,
pub(super) last: u8,
length_enable: bool, length_enable: bool,
length_timer: u8, length_timer: u8,
envelope: Envelope, envelope: Envelope,
@ -556,6 +566,7 @@ impl NoiseChannel {
pub(super) fn new(enabled: bool) -> Self { pub(super) fn new(enabled: bool) -> Self {
Self { Self {
enabled, enabled,
last: 0,
length_enable: false, length_enable: false,
length_timer: 0, length_timer: 0,
envelope: Envelope::default(), envelope: Envelope::default(),
@ -582,11 +593,13 @@ impl NoiseChannel {
}) })
.collect() .collect()
} else { } else {
self.last = 0;
vec![0.; steps] vec![0.; steps]
} }
} }
fn dac(&self, digital: u8) -> f32 { fn dac(&mut self, digital: u8) -> f32 {
self.last = digital;
(((digital as f32) * (-2.)) + 1.) * ((self.envelope.current_volume as f32) / 15.) (((digital as f32) * (-2.)) + 1.) * ((self.envelope.current_volume as f32) / 15.)
} }