SameBoy/Core/apu.c

1753 lines
72 KiB
C
Raw Permalink Normal View History

2016-03-30 20:07:55 +00:00
#include <stdint.h>
#include <math.h>
#include <string.h>
#include <assert.h>
2022-05-20 15:36:54 +00:00
#include <errno.h>
2016-03-30 20:07:55 +00:00
#include "gb.h"
static const uint8_t duties[] = {
0, 0, 0, 0, 0, 0, 0, 1,
1, 0, 0, 0, 0, 0, 0, 1,
1, 0, 0, 0, 0, 1, 1, 1,
0, 1, 1, 1, 1, 1, 1, 0,
};
static void refresh_channel(GB_gameboy_t *gb, unsigned index, unsigned cycles_offset)
{
unsigned multiplier = gb->apu_output.cycles_since_render + cycles_offset - gb->apu_output.last_update[index];
gb->apu_output.summed_samples[index].left += gb->apu_output.current_sample[index].left * multiplier;
gb->apu_output.summed_samples[index].right += gb->apu_output.current_sample[index].right * multiplier;
gb->apu_output.last_update[index] = gb->apu_output.cycles_since_render + cycles_offset;
}
bool GB_apu_is_DAC_enabled(GB_gameboy_t *gb, unsigned index)
{
if (gb->model > GB_MODEL_CGB_E) {
2019-05-16 21:08:34 +00:00
/* On the AGB, mixing is done digitally, so there are no per-channel
DACs. Instead, all channels are summed digital regardless of
whatever the DAC state would be on a CGB or earlier model. */
return true;
}
switch (index) {
case GB_SQUARE_1:
return gb->io_registers[GB_IO_NR12] & 0xF8;
case GB_SQUARE_2:
return gb->io_registers[GB_IO_NR22] & 0xF8;
case GB_WAVE:
return gb->apu.wave_channel.enable;
case GB_NOISE:
return gb->io_registers[GB_IO_NR42] & 0xF8;
2022-01-03 15:17:35 +00:00
nodefault;
}
2019-05-16 21:08:34 +00:00
return false;
}
static uint8_t agb_bias_for_channel(GB_gameboy_t *gb, unsigned index)
{
if (!gb->apu.is_active[index]) return 0;
switch (index) {
case GB_SQUARE_1:
return gb->apu.square_channels[GB_SQUARE_1].current_volume;
case GB_SQUARE_2:
return gb->apu.square_channels[GB_SQUARE_2].current_volume;
case GB_WAVE:
return 0;
case GB_NOISE:
return gb->apu.noise_channel.current_volume;
2022-01-03 15:17:35 +00:00
nodefault;
}
return 0;
}
2018-02-15 23:26:37 +00:00
static void update_sample(GB_gameboy_t *gb, unsigned index, int8_t value, unsigned cycles_offset)
2016-03-30 20:07:55 +00:00
{
2022-01-21 23:11:23 +00:00
if (gb->model > GB_MODEL_CGB_E) {
2019-05-16 21:08:34 +00:00
/* On the AGB, because no analog mixing is done, the behavior of NR51 is a bit different.
A channel that is not connected to a terminal is idenitcal to a connected channel
playing PCM sample 0. */
gb->apu.samples[index] = value;
if (gb->apu_output.sample_rate) {
unsigned right_volume = (gb->io_registers[GB_IO_NR50] & 7) + 1;
unsigned left_volume = ((gb->io_registers[GB_IO_NR50] >> 4) & 7) + 1;
2022-05-19 17:59:48 +00:00
if (index == GB_WAVE) {
/* For some reason, channel 3 is inverted on the AGB */
value ^= 0xF;
}
2019-05-16 21:08:34 +00:00
GB_sample_t output;
uint8_t bias = agb_bias_for_channel(gb, index);
2019-05-16 21:08:34 +00:00
if (gb->io_registers[GB_IO_NR51] & (1 << index)) {
2022-02-13 14:58:44 +00:00
output.right = (0xF - value * 2 + bias) * right_volume;
2019-05-16 21:08:34 +00:00
}
else {
2022-02-13 14:58:44 +00:00
output.right = 0xF * right_volume;
2019-05-16 21:08:34 +00:00
}
if (gb->io_registers[GB_IO_NR51] & (0x10 << index)) {
2022-02-13 14:58:44 +00:00
output.left = (0xF - value * 2 + bias) * left_volume;
2019-05-16 21:08:34 +00:00
}
else {
2022-02-13 14:58:44 +00:00
output.left = 0xF * left_volume;
2019-05-16 21:08:34 +00:00
}
if (*(uint32_t *)&(gb->apu_output.current_sample[index]) != *(uint32_t *)&output) {
refresh_channel(gb, index, cycles_offset);
gb->apu_output.current_sample[index] = output;
}
}
return;
}
2022-01-21 23:11:23 +00:00
if (value == 0 && gb->apu.samples[index] == 0) return;
if (!GB_apu_is_DAC_enabled(gb, index)) {
value = gb->apu.samples[index];
}
else {
gb->apu.samples[index] = value;
}
if (gb->apu_output.sample_rate) {
unsigned right_volume = 0;
if (gb->io_registers[GB_IO_NR51] & (1 << index)) {
right_volume = (gb->io_registers[GB_IO_NR50] & 7) + 1;
}
unsigned left_volume = 0;
if (gb->io_registers[GB_IO_NR51] & (0x10 << index)) {
left_volume = ((gb->io_registers[GB_IO_NR50] >> 4) & 7) + 1;
}
2022-02-13 14:58:44 +00:00
GB_sample_t output = {(0xF - value * 2) * left_volume, (0xF - value * 2) * right_volume};
if (*(uint32_t *)&(gb->apu_output.current_sample[index]) != *(uint32_t *)&output) {
refresh_channel(gb, index, cycles_offset);
gb->apu_output.current_sample[index] = output;
}
}
}
static double smooth(double x)
{
return 3*x*x - 2*x*x*x;
}
2020-12-30 22:06:36 +00:00
static signed interference(GB_gameboy_t *gb)
{
/* These aren't scientifically measured, but based on ear based on several recordings */
signed ret = 0;
if (gb->halted) {
if (gb->model <= GB_MODEL_CGB_E) {
2020-12-30 22:06:36 +00:00
ret -= MAX_CH_AMP / 5;
}
else {
ret -= MAX_CH_AMP / 12;
}
}
if (gb->io_registers[GB_IO_LCDC] & 0x80) {
ret += MAX_CH_AMP / 7;
if ((gb->io_registers[GB_IO_STAT] & 3) == 3 && gb->model <= GB_MODEL_CGB_E) {
2020-12-30 22:06:36 +00:00
ret += MAX_CH_AMP / 14;
}
else if ((gb->io_registers[GB_IO_STAT] & 3) == 1) {
ret -= MAX_CH_AMP / 7;
}
}
if (gb->apu.global_enable) {
ret += MAX_CH_AMP / 10;
}
if (GB_is_cgb(gb) && gb->model <= GB_MODEL_CGB_E && (gb->io_registers[GB_IO_RP] & 1)) {
2020-12-30 22:06:36 +00:00
ret += MAX_CH_AMP / 10;
}
if (!GB_is_cgb(gb)) {
ret /= 4;
}
ret += rand() % (MAX_CH_AMP / 12);
return ret;
}
static void render(GB_gameboy_t *gb)
{
2020-04-24 17:37:57 +00:00
GB_sample_t output = {0, 0};
unrolled for (unsigned i = 0; i < GB_N_CHANNELS; i++) {
double multiplier = CH_STEP;
2019-05-16 21:08:34 +00:00
if (gb->model <= GB_MODEL_CGB_E) {
2019-05-17 17:31:52 +00:00
if (!GB_apu_is_DAC_enabled(gb, i)) {
2019-05-16 21:08:34 +00:00
gb->apu_output.dac_discharge[i] -= ((double) DAC_DECAY_SPEED) / gb->apu_output.sample_rate;
if (gb->apu_output.dac_discharge[i] < 0) {
multiplier = 0;
gb->apu_output.dac_discharge[i] = 0;
}
else {
multiplier *= smooth(gb->apu_output.dac_discharge[i]);
2019-05-16 21:08:34 +00:00
}
}
else {
2019-05-16 21:08:34 +00:00
gb->apu_output.dac_discharge[i] += ((double) DAC_ATTACK_SPEED) / gb->apu_output.sample_rate;
if (gb->apu_output.dac_discharge[i] > 1) {
gb->apu_output.dac_discharge[i] = 1;
}
else {
multiplier *= smooth(gb->apu_output.dac_discharge[i]);
2019-05-16 21:08:34 +00:00
}
}
}
if (likely(gb->apu_output.last_update[i] == 0)) {
output.left += gb->apu_output.current_sample[i].left * multiplier;
output.right += gb->apu_output.current_sample[i].right * multiplier;
}
else {
refresh_channel(gb, i, 0);
output.left += (signed long) gb->apu_output.summed_samples[i].left * multiplier
/ gb->apu_output.cycles_since_render;
output.right += (signed long) gb->apu_output.summed_samples[i].right * multiplier
/ gb->apu_output.cycles_since_render;
gb->apu_output.summed_samples[i] = (GB_sample_t){0, 0};
}
gb->apu_output.last_update[i] = 0;
}
gb->apu_output.cycles_since_render = 0;
2022-06-24 22:44:50 +00:00
if (gb->sgb && gb->sgb->intro_animation < GB_SGB_INTRO_ANIMATION_LENGTH) return;
2017-08-15 18:14:55 +00:00
GB_sample_t filtered_output = gb->apu_output.highpass_mode?
(GB_sample_t) {output.left - gb->apu_output.highpass_diff.left,
output.right - gb->apu_output.highpass_diff.right} :
output;
2017-08-15 18:14:55 +00:00
switch (gb->apu_output.highpass_mode) {
case GB_HIGHPASS_OFF:
gb->apu_output.highpass_diff = (GB_double_sample_t) {0, 0};
break;
case GB_HIGHPASS_ACCURATE:
gb->apu_output.highpass_diff = (GB_double_sample_t)
{output.left - filtered_output.left * gb->apu_output.highpass_rate,
output.right - filtered_output.right * gb->apu_output.highpass_rate};
break;
case GB_HIGHPASS_REMOVE_DC_OFFSET: {
unsigned mask = gb->io_registers[GB_IO_NR51];
unsigned left_volume = 0;
unsigned right_volume = 0;
unrolled for (unsigned i = GB_N_CHANNELS; i--;) {
2022-02-23 18:34:50 +00:00
if (GB_apu_is_DAC_enabled(gb, i)) {
if (mask & 1) {
2022-02-23 18:34:50 +00:00
left_volume += ((gb->io_registers[GB_IO_NR50] & 7) + 1) * CH_STEP * 0xF;
}
if (mask & 0x10) {
2022-02-23 18:34:50 +00:00
right_volume += (((gb->io_registers[GB_IO_NR50] >> 4) & 7) + 1) * CH_STEP * 0xF;
}
2017-08-15 18:14:55 +00:00
}
mask >>= 1;
}
gb->apu_output.highpass_diff = (GB_double_sample_t)
{left_volume * (1 - gb->apu_output.highpass_rate) + gb->apu_output.highpass_diff.left * gb->apu_output.highpass_rate,
right_volume * (1 - gb->apu_output.highpass_rate) + gb->apu_output.highpass_diff.right * gb->apu_output.highpass_rate};
case GB_HIGHPASS_MAX:;
2017-08-15 18:14:55 +00:00
}
2017-08-15 18:14:55 +00:00
}
2020-12-30 22:06:36 +00:00
if (gb->apu_output.interference_volume) {
signed interference_bias = interference(gb);
int16_t interference_sample = (interference_bias - gb->apu_output.interference_highpass);
gb->apu_output.interference_highpass = gb->apu_output.interference_highpass * gb->apu_output.highpass_rate +
(1 - gb->apu_output.highpass_rate) * interference_sample;
interference_bias *= gb->apu_output.interference_volume;
filtered_output.left = MAX(MIN(filtered_output.left + interference_bias, 0x7FFF), -0x8000);
filtered_output.right = MAX(MIN(filtered_output.right + interference_bias, 0x7FFF), -0x8000);
}
assert(gb->apu_output.sample_callback);
gb->apu_output.sample_callback(gb, &filtered_output);
2022-05-20 15:36:54 +00:00
if (unlikely(gb->apu_output.output_file)) {
#ifdef GB_BIG_ENDIAN
if (gb->apu_output.output_format == GB_AUDIO_FORMAT_WAV) {
filtered_output.left = LE16(filtered_output.left);
filtered_output.right = LE16(filtered_output.right);
}
#endif
if (fwrite(&filtered_output, sizeof(filtered_output), 1, gb->apu_output.output_file) != 1) {
fclose(gb->apu_output.output_file);
gb->apu_output.output_file = NULL;
gb->apu_output.output_error = errno;
}
}
2016-03-30 20:07:55 +00:00
}
static void update_square_sample(GB_gameboy_t *gb, unsigned index)
{
2022-01-21 23:11:23 +00:00
if (gb->apu.square_channels[index].sample_surpressed) {
if (gb->model > GB_MODEL_CGB_E) {
2022-01-21 23:11:23 +00:00
update_sample(gb, index, gb->apu.samples[index], 0);
}
return;
}
uint8_t duty = gb->io_registers[index == GB_SQUARE_1? GB_IO_NR11 :GB_IO_NR21] >> 6;
update_sample(gb, index,
duties[gb->apu.square_channels[index].current_sample_index + duty * 8]?
gb->apu.square_channels[index].current_volume : 0,
0);
}
2021-10-16 23:06:33 +00:00
static inline void update_wave_sample(GB_gameboy_t *gb, unsigned cycles)
{
if (gb->apu.wave_channel.current_sample_index & 1) {
update_sample(gb, GB_WAVE,
(gb->apu.wave_channel.current_sample_byte & 0xF) >> gb->apu.wave_channel.shift,
cycles);
}
else {
update_sample(gb, GB_WAVE,
(gb->apu.wave_channel.current_sample_byte >> 4) >> gb->apu.wave_channel.shift,
cycles);
}
}
static void _nrx2_glitch(uint8_t *volume, uint8_t value, uint8_t old_value, uint8_t *countdown, GB_envelope_clock_t *lock)
{
if (lock->clock) {
*countdown = value & 7;
}
bool should_tick = (value & 7) && !(old_value & 7) && !lock->locked;
bool should_invert = (value & 8) ^ (old_value & 8);
if ((value & 0xF) == 8 && (old_value & 0xF) == 8 && !lock->locked) {
should_tick = true;
}
if (should_invert) {
// The weird way and over-the-top way clocks for this counter are connected cause
// some weird ways for it to invert
if (value & 8) {
if (!(old_value & 7) && !lock->locked) {
*volume ^= 0xF;
}
else {
*volume = 0xE - *volume;
*volume &= 0xF;
}
should_tick = false; // Somehow prevents ticking?
}
else {
*volume = 0x10 - *volume;
*volume &= 0xF;
}
}
if (should_tick) {
if (value & 8) {
(*volume)++;
}
else {
(*volume)--;
}
*volume &= 0xF;
}
else if (!(value & 7) && lock->clock) {
// *lock->locked = false; // Excepted from the schematics, but doesn't actually happen on any model?
if (!should_invert) {
if (*volume == 0xF && (value & 8)) {
lock->locked = true;
}
else if (*volume == 0 && !(value & 8)) {
lock->locked = true;
}
}
else if (*volume == 1 && !(value & 8)) {
lock->locked = true;
}
else if (*volume == 0xE && (value & 8)) {
lock->locked = true;
}
lock->clock = false;
}
}
2017-10-02 19:59:03 +00:00
static void nrx2_glitch(GB_gameboy_t *gb, uint8_t *volume, uint8_t value, uint8_t old_value, uint8_t *countdown, GB_envelope_clock_t *lock)
{
if (gb->model <= GB_MODEL_CGB_C) {
_nrx2_glitch(volume, 0xFF, old_value, countdown, lock);
_nrx2_glitch(volume, value, 0xFF, countdown, lock);
}
else {
_nrx2_glitch(volume, value, old_value, countdown, lock);
}
}
static void tick_square_envelope(GB_gameboy_t *gb, enum GB_CHANNELS index)
{
uint8_t nrx2 = gb->io_registers[index == GB_SQUARE_1? GB_IO_NR12 : GB_IO_NR22];
if (gb->apu.square_channels[index].envelope_clock.locked) return;
if (!(nrx2 & 7)) return;
if (gb->cgb_double_speed) {
if (index == GB_SQUARE_1) {
gb->apu.pcm_mask[0] &= gb->apu.square_channels[GB_SQUARE_1].current_volume | 0xF1;
}
else {
gb->apu.pcm_mask[0] &= (gb->apu.square_channels[GB_SQUARE_2].current_volume << 2) | 0x1F;
}
}
2021-02-22 11:48:56 +00:00
if (nrx2 & 8) {
if (gb->apu.square_channels[index].current_volume < 0xF) {
gb->apu.square_channels[index].current_volume++;
}
else {
gb->apu.square_channels[index].envelope_clock.locked = true;
2021-02-22 11:48:56 +00:00
}
}
2021-02-22 11:48:56 +00:00
else {
if (gb->apu.square_channels[index].current_volume > 0) {
gb->apu.square_channels[index].current_volume--;
}
else {
gb->apu.square_channels[index].envelope_clock.locked = true;
2021-02-22 11:48:56 +00:00
}
}
if (gb->apu.is_active[index]) {
update_square_sample(gb, index);
}
}
static void tick_noise_envelope(GB_gameboy_t *gb)
{
uint8_t nr42 = gb->io_registers[GB_IO_NR42];
if (gb->apu.noise_channel.envelope_clock.locked) return;
if (!(nr42 & 7)) return;
if (gb->cgb_double_speed) {
gb->apu.pcm_mask[0] &= (gb->apu.noise_channel.current_volume << 2) | 0x1F;
}
2021-02-22 11:48:56 +00:00
if (nr42 & 8) {
if (gb->apu.noise_channel.current_volume < 0xF) {
gb->apu.noise_channel.current_volume++;
}
else {
gb->apu.noise_channel.envelope_clock.locked = true;
2021-02-22 11:48:56 +00:00
}
}
2021-02-22 11:48:56 +00:00
else {
if (gb->apu.noise_channel.current_volume > 0) {
gb->apu.noise_channel.current_volume--;
}
else {
gb->apu.noise_channel.envelope_clock.locked = true;
2021-02-22 11:48:56 +00:00
}
}
if (gb->apu.is_active[GB_NOISE]) {
update_sample(gb, GB_NOISE,
(gb->apu.noise_channel.lfsr & 1) ?
gb->apu.noise_channel.current_volume : 0,
0);
}
}
static void trigger_sweep_calculation(GB_gameboy_t *gb)
{
if ((gb->io_registers[GB_IO_NR10] & 0x70) && gb->apu.square_sweep_countdown == 7) {
if (gb->io_registers[GB_IO_NR10] & 0x07) {
gb->apu.square_channels[GB_SQUARE_1].sample_length =
gb->apu.sweep_length_addend + gb->apu.shadow_sweep_sample_length + !!(gb->io_registers[GB_IO_NR10] & 0x8);
gb->apu.square_channels[GB_SQUARE_1].sample_length &= 0x7FF;
}
2020-12-01 20:37:13 +00:00
if (gb->apu.channel_1_restart_hold == 0) {
gb->apu.sweep_length_addend = gb->apu.square_channels[GB_SQUARE_1].sample_length;
gb->apu.sweep_length_addend >>= (gb->io_registers[GB_IO_NR10] & 7);
}
/* Recalculation and overflow check only occurs after a delay */
2020-12-01 20:37:13 +00:00
gb->apu.square_sweep_calculate_countdown = (gb->io_registers[GB_IO_NR10] & 0x7) * 2 + 5 - gb->apu.lf_div;
if (gb->model <= GB_MODEL_CGB_C && gb->apu.lf_div) {
// gb->apu.square_sweep_calculate_countdown += 2;
}
2021-01-08 14:43:00 +00:00
gb->apu.enable_zombie_calculate_stepping = false;
gb->apu.unshifted_sweep = !(gb->io_registers[GB_IO_NR10] & 0x7);
gb->apu.square_sweep_countdown = ((gb->io_registers[GB_IO_NR10] >> 4) & 7) ^ 7;
}
}
void GB_apu_div_event(GB_gameboy_t *gb)
{
2021-12-28 22:43:10 +00:00
GB_apu_run(gb, true);
if (!gb->apu.global_enable) return;
2020-02-01 21:36:16 +00:00
if (gb->apu.skip_div_event == GB_SKIP_DIV_EVENT_SKIP) {
gb->apu.skip_div_event = GB_SKIP_DIV_EVENT_SKIPPED;
return;
}
2020-02-01 21:36:16 +00:00
if (gb->apu.skip_div_event == GB_SKIP_DIV_EVENT_SKIPPED) {
gb->apu.skip_div_event = GB_SKIP_DIV_EVENT_INACTIVE;
}
else {
gb->apu.div_divider++;
}
if ((gb->apu.div_divider & 7) == 7) {
unrolled for (unsigned i = GB_SQUARE_2 + 1; i--;) {
if (!gb->apu.square_channels[i].envelope_clock.clock) {
gb->apu.square_channels[i].volume_countdown--;
gb->apu.square_channels[i].volume_countdown &= 7;
}
2016-03-30 20:07:55 +00:00
}
if (!gb->apu.noise_channel.envelope_clock.clock) {
gb->apu.noise_channel.volume_countdown--;
gb->apu.noise_channel.volume_countdown &= 7;
2017-08-11 14:57:08 +00:00
}
}
unrolled for (unsigned i = GB_SQUARE_2 + 1; i--;) {
if (gb->apu.square_channels[i].envelope_clock.clock) {
tick_square_envelope(gb, i);
gb->apu.square_channels[i].envelope_clock.clock = false;
}
}
if (gb->apu.noise_channel.envelope_clock.clock) {
tick_noise_envelope(gb);
gb->apu.noise_channel.envelope_clock.clock = false;
}
2017-08-15 19:27:15 +00:00
if ((gb->apu.div_divider & 1) == 1) {
unrolled for (unsigned i = GB_SQUARE_2 + 1; i--;) {
if (gb->apu.square_channels[i].length_enabled) {
if (gb->apu.square_channels[i].pulse_length) {
if (!--gb->apu.square_channels[i].pulse_length) {
gb->apu.is_active[i] = false;
update_sample(gb, i, 0, 0);
}
}
}
}
if (gb->apu.wave_channel.length_enabled) {
if (gb->apu.wave_channel.pulse_length) {
if (!--gb->apu.wave_channel.pulse_length) {
if (gb->apu.is_active[GB_WAVE] && gb->model > GB_MODEL_CGB_E) {
if (gb->apu.wave_channel.sample_countdown == 0) {
gb->apu.wave_channel.current_sample_byte =
gb->io_registers[GB_IO_WAV_START + (((gb->apu.wave_channel.current_sample_index + 1) & 0xF) >> 1)];
}
else if (gb->apu.wave_channel.sample_countdown == 9) {
// TODO: wtf?
gb->apu.wave_channel.current_sample_byte = gb->io_registers[GB_IO_WAV_START];
}
}
gb->apu.is_active[GB_WAVE] = false;
update_sample(gb, GB_WAVE, 0, 0);
}
}
}
if (gb->apu.noise_channel.length_enabled) {
if (gb->apu.noise_channel.pulse_length) {
if (!--gb->apu.noise_channel.pulse_length) {
gb->apu.is_active[GB_NOISE] = false;
update_sample(gb, GB_NOISE, 0, 0);
}
}
}
}
if ((gb->apu.div_divider & 3) == 3) {
gb->apu.square_sweep_countdown++;
gb->apu.square_sweep_countdown &= 7;
trigger_sweep_calculation(gb);
}
2016-03-30 20:07:55 +00:00
}
void GB_apu_div_secondary_event(GB_gameboy_t *gb)
{
2021-12-28 22:43:10 +00:00
GB_apu_run(gb, true);
if (!gb->apu.global_enable) return;
unrolled for (unsigned i = GB_SQUARE_2 + 1; i--;) {
uint8_t nrx2 = gb->io_registers[i == GB_SQUARE_1? GB_IO_NR12 : GB_IO_NR22];
if (gb->apu.is_active[i] && gb->apu.square_channels[i].volume_countdown == 0) {
gb->apu.square_channels[i].envelope_clock.clock = (gb->apu.square_channels[i].volume_countdown = nrx2 & 7);
}
}
if (gb->apu.is_active[GB_NOISE] && gb->apu.noise_channel.volume_countdown == 0) {
gb->apu.noise_channel.envelope_clock.clock = (gb->apu.noise_channel.volume_countdown = gb->io_registers[GB_IO_NR42] & 7);
}
}
static void step_lfsr(GB_gameboy_t *gb, unsigned cycles_offset)
{
unsigned high_bit_mask = gb->apu.noise_channel.narrow ? 0x4040 : 0x4000;
bool new_high_bit = (gb->apu.noise_channel.lfsr ^ (gb->apu.noise_channel.lfsr >> 1) ^ 1) & 1;
gb->apu.noise_channel.lfsr >>= 1;
if (new_high_bit) {
gb->apu.noise_channel.lfsr |= high_bit_mask;
}
else {
/* This code is not redundent, it's relevant when switching LFSR widths */
gb->apu.noise_channel.lfsr &= ~high_bit_mask;
}
gb->apu.noise_channel.current_lfsr_sample = gb->apu.noise_channel.lfsr & 1;
if (gb->apu.is_active[GB_NOISE]) {
update_sample(gb, GB_NOISE,
gb->apu.noise_channel.current_lfsr_sample ?
gb->apu.noise_channel.current_volume : 0,
cycles_offset);
}
}
2016-09-13 14:33:48 +00:00
2021-12-28 22:43:10 +00:00
void GB_apu_run(GB_gameboy_t *gb, bool force)
2016-03-30 20:07:55 +00:00
{
2021-12-28 22:43:10 +00:00
uint32_t clock_rate = GB_get_clock_rate(gb) * 2;
if (!force ||
(gb->apu.apu_cycles > 0x1000) ||
(gb->apu_output.sample_cycles >= clock_rate) ||
(gb->apu.square_sweep_calculate_countdown || gb->apu.channel_1_restart_hold) ||
(gb->model <= GB_MODEL_CGB_E && (gb->apu.wave_channel.bugged_read_countdown || (gb->apu.wave_channel.enable && gb->apu.wave_channel.pulsed)))) {
2021-12-28 22:43:10 +00:00
force = true;
}
if (!force) {
return;
}
/* Convert 4MHZ to 2MHz. apu_cycles is always divisable by 4. */
2021-12-28 22:43:10 +00:00
uint16_t cycles = gb->apu.apu_cycles >> 2;
gb->apu.apu_cycles = 0;
if (!cycles) return;
2021-01-08 14:43:00 +00:00
if (unlikely(gb->apu.wave_channel.bugged_read_countdown)) {
2021-12-28 22:43:10 +00:00
uint16_t cycles_left = cycles;
while (cycles_left) {
cycles_left--;
if (--gb->apu.wave_channel.bugged_read_countdown == 0) {
gb->apu.wave_channel.current_sample_byte =
gb->io_registers[GB_IO_WAV_START + (gb->address_bus & 0xF)];
if (gb->apu.is_active[GB_WAVE]) {
update_wave_sample(gb, 0);
}
break;
}
}
}
bool start_ch4 = false;
if (likely(!gb->stopped || GB_is_cgb(gb))) {
if (gb->apu.noise_channel.dmg_delayed_start) {
if (gb->apu.noise_channel.dmg_delayed_start == cycles) {
gb->apu.noise_channel.dmg_delayed_start = 0;
2021-01-08 14:43:00 +00:00
start_ch4 = true;
}
else if (gb->apu.noise_channel.dmg_delayed_start > cycles) {
gb->apu.noise_channel.dmg_delayed_start -= cycles;
2021-01-08 14:43:00 +00:00
}
else {
/* Split it into two */
cycles -= gb->apu.noise_channel.dmg_delayed_start;
gb->apu.apu_cycles = gb->apu.noise_channel.dmg_delayed_start * 4;
2021-12-28 22:43:10 +00:00
GB_apu_run(gb, true);
2021-01-08 14:43:00 +00:00
}
}
/* To align the square signal to 1MHz */
gb->apu.lf_div ^= cycles & 1;
gb->apu.noise_channel.alignment += cycles;
if (gb->apu.square_sweep_calculate_countdown &&
(((gb->io_registers[GB_IO_NR10] & 7) || gb->apu.unshifted_sweep) ||
2021-01-08 14:43:00 +00:00
gb->apu.square_sweep_calculate_countdown <= 3)) { // Calculation is paused if the lower bits are 0
if (gb->apu.square_sweep_calculate_countdown > cycles) {
gb->apu.square_sweep_calculate_countdown -= cycles;
}
else {
/* APU bug: sweep frequency is checked after adding the sweep delta twice */
2020-12-01 20:37:13 +00:00
if (gb->apu.channel_1_restart_hold == 0) {
gb->apu.shadow_sweep_sample_length = gb->apu.square_channels[GB_SQUARE_1].sample_length;
}
if (gb->io_registers[GB_IO_NR10] & 8) {
gb->apu.sweep_length_addend ^= 0x7FF;
}
if (gb->apu.shadow_sweep_sample_length + gb->apu.sweep_length_addend > 0x7FF && !(gb->io_registers[GB_IO_NR10] & 8)) {
gb->apu.is_active[GB_SQUARE_1] = false;
update_sample(gb, GB_SQUARE_1, 0, gb->apu.square_sweep_calculate_countdown - cycles);
}
2021-01-08 14:43:00 +00:00
gb->apu.channel1_completed_addend = gb->apu.sweep_length_addend;
gb->apu.square_sweep_calculate_countdown = 0;
}
}
2020-12-01 20:37:13 +00:00
if (gb->apu.channel_1_restart_hold) {
if (gb->apu.channel_1_restart_hold > cycles) {
gb->apu.channel_1_restart_hold -= cycles;
}
else {
gb->apu.channel_1_restart_hold = 0;
}
}
unrolled for (unsigned i = GB_SQUARE_1; i <= GB_SQUARE_2; i++) {
if (gb->apu.is_active[i]) {
2021-12-28 22:43:10 +00:00
uint16_t cycles_left = cycles;
if (unlikely(gb->apu.square_channels[i].delay)) {
if (gb->apu.square_channels[i].delay < cycles_left) {
gb->apu.square_channels[i].delay = 0;
}
else {
gb->apu.square_channels[i].delay -= cycles_left;
}
}
while (unlikely(cycles_left > gb->apu.square_channels[i].sample_countdown)) {
cycles_left -= gb->apu.square_channels[i].sample_countdown + 1;
gb->apu.square_channels[i].sample_countdown = (gb->apu.square_channels[i].sample_length ^ 0x7FF) * 2 + 1;
gb->apu.square_channels[i].current_sample_index++;
gb->apu.square_channels[i].current_sample_index &= 0x7;
gb->apu.square_channels[i].sample_surpressed = false;
if (cycles_left == 0 && gb->apu.samples[i] == 0) {
gb->apu.pcm_mask[0] &= i == GB_SQUARE_1? 0xF0 : 0x0F;
}
gb->apu.square_channels[i].did_tick = true;
update_square_sample(gb, i);
}
if (cycles_left) {
gb->apu.square_channels[i].sample_countdown -= cycles_left;
}
}
}
gb->apu.wave_channel.wave_form_just_read = false;
if (gb->apu.is_active[GB_WAVE]) {
2021-12-28 22:43:10 +00:00
uint16_t cycles_left = cycles;
while (unlikely(cycles_left > gb->apu.wave_channel.sample_countdown)) {
cycles_left -= gb->apu.wave_channel.sample_countdown + 1;
gb->apu.wave_channel.sample_countdown = gb->apu.wave_channel.sample_length ^ 0x7FF;
gb->apu.wave_channel.current_sample_index++;
gb->apu.wave_channel.current_sample_index &= 0x1F;
2021-10-16 23:06:33 +00:00
gb->apu.wave_channel.current_sample_byte =
gb->io_registers[GB_IO_WAV_START + (gb->apu.wave_channel.current_sample_index >> 1)];
update_wave_sample(gb, cycles - cycles_left);
gb->apu.wave_channel.wave_form_just_read = true;
}
if (cycles_left) {
gb->apu.wave_channel.sample_countdown -= cycles_left;
gb->apu.wave_channel.wave_form_just_read = false;
}
}
else if (gb->apu.wave_channel.enable && gb->apu.wave_channel.pulsed && gb->model <= GB_MODEL_CGB_E) {
2021-12-28 22:43:10 +00:00
uint16_t cycles_left = cycles;
while (unlikely(cycles_left > gb->apu.wave_channel.sample_countdown)) {
cycles_left -= gb->apu.wave_channel.sample_countdown + 1;
gb->apu.wave_channel.sample_countdown = gb->apu.wave_channel.sample_length ^ 0x7FF;
if (cycles_left) {
gb->apu.wave_channel.current_sample_byte =
gb->io_registers[GB_IO_WAV_START + (gb->address_bus & 0xF)];
}
else {
gb->apu.wave_channel.bugged_read_countdown = 1;
}
}
if (cycles_left) {
gb->apu.wave_channel.sample_countdown -= cycles_left;
}
if (gb->apu.wave_channel.sample_countdown == 0) {
gb->apu.wave_channel.bugged_read_countdown = 2;
}
}
// The noise channel can step even if inactive on the DMG
if (gb->apu.is_active[GB_NOISE] || !GB_is_cgb(gb)) {
2021-12-28 22:43:10 +00:00
uint16_t cycles_left = cycles;
unsigned divisor = (gb->io_registers[GB_IO_NR43] & 0x07) << 2;
if (!divisor) divisor = 2;
if (gb->apu.noise_channel.counter_countdown == 0) {
gb->apu.noise_channel.counter_countdown = divisor;
}
2021-12-28 22:43:10 +00:00
// This while doesn't get an unlikely because the noise channel steps frequently enough
while (cycles_left >= gb->apu.noise_channel.counter_countdown) {
cycles_left -= gb->apu.noise_channel.counter_countdown;
gb->apu.noise_channel.counter_countdown = divisor + gb->apu.noise_channel.delta;
gb->apu.noise_channel.delta = 0;
bool old_bit = (gb->apu.noise_channel.counter >> (gb->io_registers[GB_IO_NR43] >> 4)) & 1;
gb->apu.noise_channel.counter++;
gb->apu.noise_channel.counter &= 0x3FFF;
bool new_bit = (gb->apu.noise_channel.counter >> (gb->io_registers[GB_IO_NR43] >> 4)) & 1;
/* Step LFSR */
if (new_bit && !old_bit) {
if (cycles_left == 0 && gb->apu.samples[GB_NOISE] == 0) {
gb->apu.pcm_mask[1] &= 0x0F;
}
step_lfsr(gb, cycles - cycles_left);
}
2017-08-11 14:57:08 +00:00
}
if (cycles_left) {
gb->apu.noise_channel.counter_countdown -= cycles_left;
gb->apu.noise_channel.countdown_reloaded = false;
}
else {
gb->apu.noise_channel.countdown_reloaded = true;
2018-07-04 18:55:12 +00:00
}
2017-08-11 14:57:08 +00:00
}
}
if (gb->apu_output.sample_rate) {
gb->apu_output.cycles_since_render += cycles;
2021-12-26 13:20:46 +00:00
if (gb->apu_output.sample_cycles >= clock_rate) {
gb->apu_output.sample_cycles -= clock_rate;
render(gb);
}
}
if (start_ch4) {
GB_apu_write(gb, GB_IO_NR44, gb->io_registers[GB_IO_NR44] | 0x80);
}
2016-03-30 20:07:55 +00:00
}
2021-01-08 14:43:00 +00:00
void GB_apu_init(GB_gameboy_t *gb)
2016-03-30 20:07:55 +00:00
{
memset(&gb->apu, 0, sizeof(gb->apu));
2017-08-11 14:57:08 +00:00
gb->apu.lf_div = 1;
gb->apu.wave_channel.shift = 4;
2018-06-09 12:12:42 +00:00
/* APU glitch: When turning the APU on while DIV's bit 4 (or 5 in double speed mode) is on,
the first DIV/APU event is skipped. */
if (gb->div_counter & (gb->cgb_double_speed? 0x2000 : 0x1000)) {
2020-02-01 21:36:16 +00:00
gb->apu.skip_div_event = GB_SKIP_DIV_EVENT_SKIP;
gb->apu.div_divider = 1;
}
2022-03-12 12:32:16 +00:00
gb->apu.square_channels[GB_SQUARE_1].sample_countdown = -1;
gb->apu.square_channels[GB_SQUARE_2].sample_countdown = -1;
2016-03-30 20:07:55 +00:00
}
uint8_t GB_apu_read(GB_gameboy_t *gb, uint8_t reg)
2016-03-30 20:07:55 +00:00
{
2021-12-28 22:43:10 +00:00
GB_apu_run(gb, true);
2016-03-30 20:07:55 +00:00
if (reg == GB_IO_NR52) {
uint8_t value = 0;
2020-04-09 11:32:52 +00:00
for (unsigned i = 0; i < GB_N_CHANNELS; i++) {
2016-03-30 20:07:55 +00:00
value >>= 1;
if (gb->apu.is_active[i]) {
2016-03-30 20:07:55 +00:00
value |= 0x8;
}
}
if (gb->apu.global_enable) {
value |= 0x80;
}
value |= 0x70;
return value;
}
static const char read_mask[GB_IO_WAV_END - GB_IO_NR10 + 1] = {
/* NRX0 NRX1 NRX2 NRX3 NRX4 */
0x80, 0x3F, 0x00, 0xFF, 0xBF, // NR1X
0xFF, 0x3F, 0x00, 0xFF, 0xBF, // NR2X
0x7F, 0xFF, 0x9F, 0xFF, 0xBF, // NR3X
0xFF, 0xFF, 0x00, 0x00, 0xBF, // NR4X
0x00, 0x00, 0x70, 0xFF, 0xFF, // NR5X
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, // Unused
// Wave RAM
0, /* ... */
};
if (reg >= GB_IO_WAV_START && reg <= GB_IO_WAV_END && gb->apu.is_active[GB_WAVE]) {
if (!GB_is_cgb(gb) && !gb->apu.wave_channel.wave_form_just_read) {
return 0xFF;
2016-09-13 14:40:10 +00:00
}
if (gb->model > GB_MODEL_CGB_E) {
return 0xFF;
}
reg = GB_IO_WAV_START + gb->apu.wave_channel.current_sample_index / 2;
2016-03-30 20:07:55 +00:00
}
return gb->io_registers[reg] | read_mask[reg - GB_IO_NR10];
}
static inline uint16_t effective_channel4_counter(GB_gameboy_t *gb)
{
2021-01-16 13:31:09 +00:00
/*
TODO: On revisions older than the CGB-D, this behaves differently when
the counter advanced this exact T-cycle. Also, in these revisions,
it seems that "passive" changes (due to the temporary FF value NR43
has during writes) behave slightly different from non-passive ones.
*/
uint16_t effective_counter = gb->apu.noise_channel.counter;
/* Ladies and gentlemen, I present you the holy grail glitch of revision detection! */
switch (gb->model) {
/* Pre CGB revisions are assumed to be like CGB-C, A and 0 for the lack of a better guess.
TODO: It could be verified with audio based test ROMs. */
case GB_MODEL_CGB_B:
if (effective_counter & 8) {
effective_counter |= 0xE; // Seems to me F under some circumstances?
}
if (effective_counter & 0x80) {
effective_counter |= 0xFF;
}
if (effective_counter & 0x100) {
effective_counter |= 0x1;
}
if (effective_counter & 0x200) {
effective_counter |= 0x2;
}
if (effective_counter & 0x400) {
effective_counter |= 0x4;
}
if (effective_counter & 0x800) {
2021-01-09 21:28:30 +00:00
effective_counter |= 0x408; // TODO: Only my CGB-B does that! Others behave like C!
}
if (effective_counter & 0x1000) {
effective_counter |= 0x10;
}
if (effective_counter & 0x2000) {
effective_counter |= 0x20;
}
break;
case GB_MODEL_DMG_B:
2021-10-23 20:28:54 +00:00
case GB_MODEL_MGB:
case GB_MODEL_SGB_NTSC:
case GB_MODEL_SGB_PAL:
case GB_MODEL_SGB_NTSC_NO_SFC:
case GB_MODEL_SGB_PAL_NO_SFC:
case GB_MODEL_SGB2:
case GB_MODEL_SGB2_NO_SFC:
2021-11-03 22:32:15 +00:00
case GB_MODEL_CGB_0:
2022-01-05 19:55:46 +00:00
case GB_MODEL_CGB_A:
case GB_MODEL_CGB_C:
if (effective_counter & 8) {
2021-01-24 18:57:46 +00:00
effective_counter |= 0xE; // Sometimes F on some instances
}
if (effective_counter & 0x80) {
effective_counter |= 0xFF;
}
if (effective_counter & 0x100) {
effective_counter |= 0x1;
}
if (effective_counter & 0x200) {
effective_counter |= 0x2;
}
if (effective_counter & 0x400) {
effective_counter |= 0x4;
}
if (effective_counter & 0x800) {
2021-01-24 18:57:46 +00:00
if ((gb->io_registers[GB_IO_NR43] & 8)) {
effective_counter |= 0x400;
}
effective_counter |= 0x8;
}
if (effective_counter & 0x1000) {
effective_counter |= 0x10;
}
if (effective_counter & 0x2000) {
effective_counter |= 0x20;
}
break;
case GB_MODEL_CGB_D:
if (effective_counter & ((gb->io_registers[GB_IO_NR43] & 8)? 0x40 : 0x80)) { // This is so weird
effective_counter |= 0xFF;
}
if (effective_counter & 0x100) {
effective_counter |= 0x1;
}
if (effective_counter & 0x200) {
effective_counter |= 0x2;
}
if (effective_counter & 0x400) {
effective_counter |= 0x4;
}
if (effective_counter & 0x800) {
effective_counter |= 0x8;
}
if (effective_counter & 0x1000) {
effective_counter |= 0x10;
}
break;
case GB_MODEL_CGB_E:
if (effective_counter & ((gb->io_registers[GB_IO_NR43] & 8)? 0x40 : 0x80)) { // This is so weird
effective_counter |= 0xFF;
}
if (effective_counter & 0x1000) {
effective_counter |= 0x10;
}
break;
case GB_MODEL_AGB_A:
/* TODO: AGBs are not affected, but AGSes are. They don't seem to follow a simple
pattern like the other revisions. */
/* For the most part, AGS seems to do:
0x20 -> 0xA0
0x200 -> 0xA00
2021-01-24 18:57:46 +00:00
0x1000 -> 0x1010, but only if wide
*/
break;
}
return effective_counter;
}
void GB_apu_write(GB_gameboy_t *gb, uint8_t reg, uint8_t value)
2016-03-30 20:07:55 +00:00
{
2021-12-28 22:43:10 +00:00
GB_apu_run(gb, true);
if (!gb->apu.global_enable && reg != GB_IO_NR52 && reg < GB_IO_WAV_START && (GB_is_cgb(gb) ||
(
reg != GB_IO_NR11 &&
reg != GB_IO_NR21 &&
reg != GB_IO_NR31 &&
reg != GB_IO_NR41
)
)) {
2016-03-30 20:07:55 +00:00
return;
}
if (reg >= GB_IO_WAV_START && reg <= GB_IO_WAV_END && gb->apu.is_active[GB_WAVE]) {
if ((!GB_is_cgb(gb) && !gb->apu.wave_channel.wave_form_just_read) || gb->model > GB_MODEL_CGB_E) {
return;
}
reg = GB_IO_WAV_START + gb->apu.wave_channel.current_sample_index / 2;
}
/* Todo: this can and should be rewritten with a function table. */
2016-03-30 20:07:55 +00:00
switch (reg) {
/* Globals */
case GB_IO_NR50:
case GB_IO_NR51:
gb->io_registers[reg] = value;
/* These registers affect the output of all 4 channels (but not the output of the PCM registers).*/
/* We call update_samples with the current value so the APU output is updated with the new outputs */
for (unsigned i = GB_N_CHANNELS; i--;) {
int8_t sample = gb->apu.samples[i];
gb->apu.samples[i] = 0x10; // Invalidate to force update
update_sample(gb, i, sample, 0);
}
break;
case GB_IO_NR52: {
uint8_t old_pulse_lengths[] = {
gb->apu.square_channels[0].pulse_length,
gb->apu.square_channels[1].pulse_length,
gb->apu.wave_channel.pulse_length,
gb->apu.noise_channel.pulse_length
};
if ((value & 0x80) && !gb->apu.global_enable) {
GB_apu_init(gb);
gb->apu.global_enable = true;
2016-03-30 20:07:55 +00:00
}
else if (!(value & 0x80) && gb->apu.global_enable) {
for (unsigned i = GB_N_CHANNELS; i--;) {
update_sample(gb, i, 0, 0);
}
memset(&gb->apu, 0, sizeof(gb->apu));
memset(gb->io_registers + GB_IO_NR10, 0, GB_IO_WAV_START - GB_IO_NR10);
gb->apu.global_enable = false;
2016-03-30 20:07:55 +00:00
}
if (!GB_is_cgb(gb) && (value & 0x80)) {
gb->apu.square_channels[0].pulse_length = old_pulse_lengths[0];
gb->apu.square_channels[1].pulse_length = old_pulse_lengths[1];
gb->apu.wave_channel.pulse_length = old_pulse_lengths[2];
gb->apu.noise_channel.pulse_length = old_pulse_lengths[3];
}
}
break;
2017-08-11 14:57:08 +00:00
/* Square channels */
case GB_IO_NR10:{
bool old_negate = gb->io_registers[GB_IO_NR10] & 8;
gb->io_registers[GB_IO_NR10] = value;
2021-01-08 14:43:00 +00:00
if (gb->apu.shadow_sweep_sample_length + gb->apu.channel1_completed_addend + old_negate > 0x7FF &&
2020-12-01 20:37:13 +00:00
!(value & 8)) {
2017-09-21 15:18:10 +00:00
gb->apu.is_active[GB_SQUARE_1] = false;
update_sample(gb, GB_SQUARE_1, 0, 0);
}
trigger_sweep_calculation(gb);
break;
}
case GB_IO_NR11:
case GB_IO_NR21: {
unsigned index = reg == GB_IO_NR21? GB_SQUARE_2: GB_SQUARE_1;
2022-02-13 14:58:44 +00:00
gb->apu.square_channels[index].pulse_length = (0x40 - (value & 0x3F));
2017-09-21 15:32:21 +00:00
if (!gb->apu.global_enable) {
2022-02-13 14:58:44 +00:00
value &= 0x3F;
2017-09-21 15:32:21 +00:00
}
break;
}
case GB_IO_NR12:
case GB_IO_NR22: {
unsigned index = reg == GB_IO_NR22? GB_SQUARE_2: GB_SQUARE_1;
if ((value & 0xF8) == 0) {
/* This disables the DAC */
gb->io_registers[reg] = value;
gb->apu.is_active[index] = false;
update_sample(gb, index, 0, 0);
}
else if (gb->apu.is_active[index]) {
nrx2_glitch(gb, &gb->apu.square_channels[index].current_volume,
value, gb->io_registers[reg], &gb->apu.square_channels[index].volume_countdown,
&gb->apu.square_channels[index].envelope_clock);
update_square_sample(gb, index);
}
break;
}
case GB_IO_NR13:
case GB_IO_NR23: {
unsigned index = reg == GB_IO_NR23? GB_SQUARE_2: GB_SQUARE_1;
if (gb->apu.is_active[index]) {
/* On an AGB, as well as on CGB C and earlier (TODO: Tested: 0, B and C), it behaves slightly different on
double speed. */
if (gb->model == GB_MODEL_CGB_E || gb->model == GB_MODEL_CGB_D || gb->apu.square_channels[index].sample_countdown & 1) {
if (gb->apu.square_channels[index].did_tick &&
gb->apu.square_channels[index].sample_countdown >> 1 == (gb->apu.square_channels[index].sample_length ^ 0x7FF)) {
gb->apu.square_channels[index].current_sample_index--;
gb->apu.square_channels[index].current_sample_index &= 7;
gb->apu.square_channels[index].sample_surpressed = false;
}
}
}
gb->apu.square_channels[index].sample_length &= ~0xFF;
gb->apu.square_channels[index].sample_length |= value & 0xFF;
break;
}
case GB_IO_NR14:
case GB_IO_NR24: {
unsigned index = reg == GB_IO_NR24? GB_SQUARE_2: GB_SQUARE_1;
2021-01-02 14:23:34 +00:00
bool was_active = gb->apu.is_active[index];
2019-09-29 21:09:25 +00:00
/* TODO: When the sample length changes right before being updated, the countdown should change to the
old length, but the current sample should not change. Because our write timing isn't accurate to
the T-cycle, we hack around it by stepping the sample index backwards. */
if ((value & 0x80) == 0 && gb->apu.is_active[index]) {
/* On an AGB, as well as on CGB C and earlier (TODO: Tested: 0, B and C), it behaves slightly different on
double speed. */
2021-10-26 22:40:28 +00:00
if (gb->model == GB_MODEL_CGB_E || gb->model == GB_MODEL_CGB_D || gb->apu.square_channels[index].sample_countdown & 1) {
if (gb->apu.square_channels[index].did_tick &&
gb->apu.square_channels[index].sample_countdown >> 1 == (gb->apu.square_channels[index].sample_length ^ 0x7FF)) {
2019-09-29 21:09:25 +00:00
gb->apu.square_channels[index].current_sample_index--;
gb->apu.square_channels[index].current_sample_index &= 7;
gb->apu.square_channels[index].sample_surpressed = false;
2019-09-29 21:09:25 +00:00
}
}
}
uint16_t old_sample_length = gb->apu.square_channels[index].sample_length;
gb->apu.square_channels[index].sample_length &= 0xFF;
gb->apu.square_channels[index].sample_length |= (value & 7) << 8;
if (value & 0x80) {
/* Current sample index remains unchanged when restarting channels 1 or 2. It is only reset by
turning the APU off. */
gb->apu.square_channels[index].envelope_clock.locked = false;
gb->apu.square_channels[index].envelope_clock.clock = false;
gb->apu.square_channels[index].did_tick = false;
2022-03-12 12:32:16 +00:00
bool force_unsurpressed = false;
if (!gb->apu.is_active[index]) {
2022-03-12 12:32:16 +00:00
if (gb->model == GB_MODEL_CGB_E || gb->model == GB_MODEL_CGB_D) {
if (!(value & 4) && !(((gb->apu.square_channels[index].sample_countdown - gb->apu.square_channels[index].delay) / 2) & 0x400)) {
gb->apu.square_channels[index].current_sample_index++;
gb->apu.square_channels[index].current_sample_index &= 0x7;
force_unsurpressed = true;
}
}
gb->apu.square_channels[index].delay = 6 - gb->apu.lf_div;
gb->apu.square_channels[index].sample_countdown = (gb->apu.square_channels[index].sample_length ^ 0x7FF) * 2 + gb->apu.square_channels[index].delay;
if (gb->model <= GB_MODEL_CGB_C && gb->apu.lf_div) {
gb->apu.square_channels[index].sample_countdown += 2;
2022-03-12 12:32:16 +00:00
gb->apu.square_channels[index].delay += 2;
}
}
else {
unsigned extra_delay = 0;
2021-10-26 22:40:28 +00:00
if (gb->model == GB_MODEL_CGB_E || gb->model == GB_MODEL_CGB_D) {
if (!(value & 4) && !(((gb->apu.square_channels[index].sample_countdown - 1 - gb->apu.square_channels[index].delay) / 2) & 0x400)) {
gb->apu.square_channels[index].current_sample_index++;
gb->apu.square_channels[index].current_sample_index &= 0x7;
gb->apu.square_channels[index].sample_surpressed = false;
}
/* Todo: verify with the schematics what's going on in here */
else if (gb->apu.square_channels[index].sample_length == 0x7FF &&
old_sample_length != 0x7FF &&
(gb->apu.square_channels[index].sample_surpressed)) {
extra_delay += 2;
}
2021-01-02 16:27:21 +00:00
}
/* Timing quirk: if already active, sound starts 2 (2MHz) ticks earlier.*/
gb->apu.square_channels[index].delay = 4 - gb->apu.lf_div + extra_delay;
gb->apu.square_channels[index].sample_countdown = (gb->apu.square_channels[index].sample_length ^ 0x7FF) * 2 + gb->apu.square_channels[index].delay;
if (gb->model <= GB_MODEL_CGB_C && gb->apu.lf_div) {
gb->apu.square_channels[index].sample_countdown += 2;
2022-03-12 12:32:16 +00:00
gb->apu.square_channels[index].delay += 2;
}
}
gb->apu.square_channels[index].current_volume = gb->io_registers[index == GB_SQUARE_1 ? GB_IO_NR12 : GB_IO_NR22] >> 4;
/* The volume changes caused by NRX4 sound start take effect instantly (i.e. the effect the previously
started sound). The playback itself is not instant which is why we don't update the sample for other
cases. */
if (gb->apu.is_active[index]) {
update_square_sample(gb, index);
}
2017-08-15 19:27:15 +00:00
gb->apu.square_channels[index].volume_countdown = gb->io_registers[index == GB_SQUARE_1 ? GB_IO_NR12 : GB_IO_NR22] & 7;
if ((gb->io_registers[index == GB_SQUARE_1 ? GB_IO_NR12 : GB_IO_NR22] & 0xF8) != 0 && !gb->apu.is_active[index]) {
gb->apu.is_active[index] = true;
update_sample(gb, index, 0, 0);
2022-03-12 12:32:16 +00:00
gb->apu.square_channels[index].sample_surpressed = true && !force_unsurpressed;
}
if (gb->apu.square_channels[index].pulse_length == 0) {
gb->apu.square_channels[index].pulse_length = 0x40;
gb->apu.square_channels[index].length_enabled = false;
}
2017-08-15 19:05:20 +00:00
if (index == GB_SQUARE_1) {
2020-12-01 20:37:13 +00:00
gb->apu.shadow_sweep_sample_length = 0;
2021-01-08 14:43:00 +00:00
gb->apu.channel1_completed_addend = 0;
2017-09-22 11:39:39 +00:00
if (gb->io_registers[GB_IO_NR10] & 7) {
/* APU bug: if shift is nonzero, overflow check also occurs on trigger */
gb->apu.square_sweep_calculate_countdown = (gb->io_registers[GB_IO_NR10] & 0x7) * 2 + 5 - gb->apu.lf_div;
if (gb->model <= GB_MODEL_CGB_C && gb->apu.lf_div) {
/* TODO: I used to think this is correct, but it caused several regressions.
More research is needed to figure how calculation time is different
in models prior to CGB-D */
// gb->apu.square_sweep_calculate_countdown += 2;
}
2021-01-08 14:43:00 +00:00
gb->apu.enable_zombie_calculate_stepping = false;
gb->apu.unshifted_sweep = false;
2021-01-08 14:43:00 +00:00
if (!was_active) {
gb->apu.square_sweep_calculate_countdown += 2;
}
gb->apu.sweep_length_addend = gb->apu.square_channels[GB_SQUARE_1].sample_length;
gb->apu.sweep_length_addend >>= (gb->io_registers[GB_IO_NR10] & 7);
2017-09-22 11:39:39 +00:00
}
2017-09-22 21:23:02 +00:00
else {
gb->apu.sweep_length_addend = 0;
2017-09-22 21:23:02 +00:00
}
2021-10-26 22:40:28 +00:00
gb->apu.channel_1_restart_hold = 2 - gb->apu.lf_div + (GB_is_cgb(gb) && gb->model != GB_MODEL_CGB_D) * 2;
/*
if (GB_is_cgb(gb) && gb->model <= GB_MODEL_CGB_C && gb->apu.lf_div) {
// TODO: This if makes channel_1_sweep_restart_2 fail on CGB-C mode
gb->apu.channel_1_restart_hold += 2;
2021-10-26 22:40:28 +00:00
}*/
gb->apu.square_sweep_countdown = ((gb->io_registers[GB_IO_NR10] >> 4) & 7) ^ 7;
}
}
/* APU glitch - if length is enabled while the DIV-divider's LSB is 1, tick the length once. */
2021-10-30 17:58:57 +00:00
if (((value & 0x40) || (GB_is_cgb(gb) && gb->model <= GB_MODEL_CGB_B)) && // Current value is irrelevant on CGB-B and older
!gb->apu.square_channels[index].length_enabled &&
(gb->apu.div_divider & 1) &&
gb->apu.square_channels[index].pulse_length) {
gb->apu.square_channels[index].pulse_length--;
if (gb->apu.square_channels[index].pulse_length == 0) {
if (value & 0x80) {
gb->apu.square_channels[index].pulse_length = 0x3F;
}
else {
gb->apu.is_active[index] = false;
update_sample(gb, index, 0, 0);
}
}
}
gb->apu.square_channels[index].length_enabled = value & 0x40;
break;
}
/* Wave channel */
2016-03-30 20:07:55 +00:00
case GB_IO_NR30:
gb->apu.wave_channel.enable = value & 0x80;
if (!gb->apu.wave_channel.enable) {
gb->apu.wave_channel.pulsed = false;
2021-10-16 23:06:33 +00:00
if (gb->apu.is_active[GB_WAVE]) {
// Todo: I assume this happens on pre-CGB models; test this with an audible test
if (gb->apu.wave_channel.sample_countdown == 0 && gb->model <= GB_MODEL_CGB_E) {
2021-10-17 09:52:08 +00:00
gb->apu.wave_channel.current_sample_byte = gb->io_registers[GB_IO_WAV_START + (gb->pc & 0xF)];
2021-10-16 23:06:33 +00:00
}
else if (gb->apu.wave_channel.wave_form_just_read && gb->model <= GB_MODEL_CGB_C) {
gb->apu.wave_channel.current_sample_byte = gb->io_registers[GB_IO_WAV_START + (GB_IO_NR30 & 0xF)];
2021-10-16 23:06:33 +00:00
}
}
gb->apu.is_active[GB_WAVE] = false;
update_sample(gb, GB_WAVE, 0, 0);
}
2016-03-30 20:07:55 +00:00
break;
case GB_IO_NR31:
gb->apu.wave_channel.pulse_length = (0x100 - value);
2016-03-30 20:07:55 +00:00
break;
case GB_IO_NR32:
gb->apu.wave_channel.shift = (uint8_t[]){4, 0, 1, 2}[(value >> 5) & 3];
if (gb->apu.is_active[GB_WAVE]) {
2021-10-16 23:06:33 +00:00
update_wave_sample(gb, 0);
}
2016-03-30 20:07:55 +00:00
break;
case GB_IO_NR33:
gb->apu.wave_channel.sample_length &= ~0xFF;
gb->apu.wave_channel.sample_length |= value & 0xFF;
2016-03-30 20:07:55 +00:00
break;
case GB_IO_NR34:
gb->apu.wave_channel.sample_length &= 0xFF;
gb->apu.wave_channel.sample_length |= (value & 7) << 8;
2021-10-07 15:25:54 +00:00
if (value & 0x80) {
gb->apu.wave_channel.pulsed = true;
/* DMG bug: wave RAM gets corrupted if the channel is retriggerred 1 cycle before the APU
reads from it. */
if (!GB_is_cgb(gb) &&
gb->apu.is_active[GB_WAVE] &&
2021-09-04 15:15:22 +00:00
gb->apu.wave_channel.sample_countdown == 0) {
unsigned offset = ((gb->apu.wave_channel.current_sample_index + 1) >> 1) & 0xF;
2018-05-15 20:02:07 +00:00
/* This glitch varies between models and even specific instances:
DMG-B: Most of them behave as emulated. A few behave differently.
SGB: As far as I know, all tested instances behave as emulated.
MGB, SGB2: Most instances behave non-deterministically, a few behave as emulated.
2021-10-23 20:28:54 +00:00
For DMG-B emulation I emulate the most common behavior, which blargg's tests expect (not my own DMG-B, which fails it)
For MGB emulation, I emulate my Game Boy Light, which happens to be deterministic.
2018-05-15 20:02:07 +00:00
Additionally, I believe DMGs, including those we behave differently than emulated,
are all deterministic. */
2021-10-23 20:28:54 +00:00
if (offset < 4 && gb->model != GB_MODEL_MGB) {
gb->io_registers[GB_IO_WAV_START] = gb->io_registers[GB_IO_WAV_START + offset];
}
else {
memcpy(gb->io_registers + GB_IO_WAV_START,
gb->io_registers + GB_IO_WAV_START + (offset & ~3),
4);
}
}
2021-10-16 23:06:33 +00:00
gb->apu.wave_channel.current_sample_index = 0;
if (gb->apu.is_active[GB_WAVE] && gb->apu.wave_channel.sample_countdown == 0) {
gb->apu.wave_channel.current_sample_byte = gb->io_registers[GB_IO_WAV_START];
}
if (gb->apu.wave_channel.enable) {
gb->apu.is_active[GB_WAVE] = true;
update_sample(gb, GB_WAVE,
2021-10-16 23:06:33 +00:00
(gb->apu.wave_channel.current_sample_byte >> 4) >> gb->apu.wave_channel.shift,
0);
}
gb->apu.wave_channel.sample_countdown = (gb->apu.wave_channel.sample_length ^ 0x7FF) + 3;
if (gb->apu.wave_channel.pulse_length == 0) {
gb->apu.wave_channel.pulse_length = 0x100;
gb->apu.wave_channel.length_enabled = false;
}
/* Note that we don't change the sample just yet! This was verified on hardware. */
2016-03-30 20:07:55 +00:00
}
/* APU glitch - if length is enabled while the DIV-divider's LSB is 1, tick the length once. */
2021-10-30 17:58:57 +00:00
if (((value & 0x40) || (GB_is_cgb(gb) && gb->model <= GB_MODEL_CGB_B)) && // Current value is irrelevant on CGB-B and older
!gb->apu.wave_channel.length_enabled &&
(gb->apu.div_divider & 1) &&
gb->apu.wave_channel.pulse_length) {
gb->apu.wave_channel.pulse_length--;
if (gb->apu.wave_channel.pulse_length == 0) {
if (value & 0x80) {
gb->apu.wave_channel.pulse_length = 0xFF;
}
else {
gb->apu.is_active[GB_WAVE] = false;
update_sample(gb, GB_WAVE, 0, 0);
}
}
}
gb->apu.wave_channel.length_enabled = value & 0x40;
2016-03-30 20:07:55 +00:00
break;
2017-08-11 14:57:08 +00:00
/* Noise Channel */
2017-08-11 14:57:08 +00:00
case GB_IO_NR41: {
2022-02-13 14:58:44 +00:00
gb->apu.noise_channel.pulse_length = (0x40 - (value & 0x3F));
2017-08-11 14:57:08 +00:00
break;
}
2017-08-11 14:57:08 +00:00
case GB_IO_NR42: {
if ((value & 0xF8) == 0) {
/* This disables the DAC */
gb->io_registers[reg] = value;
2017-08-11 14:57:08 +00:00
gb->apu.is_active[GB_NOISE] = false;
update_sample(gb, GB_NOISE, 0, 0);
2017-08-11 14:57:08 +00:00
}
2020-04-24 17:37:57 +00:00
else if (gb->apu.is_active[GB_NOISE]) {
nrx2_glitch(gb, &gb->apu.noise_channel.current_volume,
value, gb->io_registers[reg], &gb->apu.noise_channel.volume_countdown,
&gb->apu.noise_channel.envelope_clock);
update_sample(gb, GB_NOISE,
gb->apu.noise_channel.current_lfsr_sample ?
gb->apu.noise_channel.current_volume : 0,
0);
}
2017-08-11 14:57:08 +00:00
break;
}
2017-08-11 14:57:08 +00:00
case GB_IO_NR43: {
gb->apu.noise_channel.narrow = value & 8;
uint16_t effective_counter = effective_channel4_counter(gb);
bool old_bit = (effective_counter >> (gb->io_registers[GB_IO_NR43] >> 4)) & 1;
gb->io_registers[GB_IO_NR43] = value;
bool new_bit = (effective_counter >> (gb->io_registers[GB_IO_NR43] >> 4)) & 1;
if (gb->apu.noise_channel.countdown_reloaded) {
unsigned divisor = (gb->io_registers[GB_IO_NR43] & 0x07) << 2;
if (!divisor) divisor = 2;
2021-01-16 12:43:32 +00:00
if (gb->model > GB_MODEL_CGB_C) {
gb->apu.noise_channel.counter_countdown =
divisor + (divisor == 2? 0 : (uint8_t[]){2, 1, 0, 3}[(gb->apu.noise_channel.alignment) & 3]);
}
else {
gb->apu.noise_channel.counter_countdown =
divisor + (divisor == 2? 0 : (uint8_t[]){2, 1, 4, 3}[(gb->apu.noise_channel.alignment) & 3]);
}
gb->apu.noise_channel.delta = 0;
}
/* Step LFSR */
if (new_bit && (!old_bit || gb->model <= GB_MODEL_CGB_C)) {
if (gb->model <= GB_MODEL_CGB_C) {
bool previous_narrow = gb->apu.noise_channel.narrow;
gb->apu.noise_channel.narrow = true;
step_lfsr(gb, 0);
gb->apu.noise_channel.narrow = previous_narrow;
}
else {
step_lfsr(gb, 0);
}
}
break;
2017-08-11 14:57:08 +00:00
}
2017-08-11 14:57:08 +00:00
case GB_IO_NR44: {
if (value & 0x80) {
gb->apu.noise_channel.envelope_clock.locked = false;
gb->apu.noise_channel.envelope_clock.clock = false;
if (!GB_is_cgb(gb) && (gb->apu.noise_channel.alignment & 3) != 0) {
gb->apu.noise_channel.dmg_delayed_start = 6;
}
else {
unsigned divisor = (gb->io_registers[GB_IO_NR43] & 0x07) << 2;
if (!divisor) divisor = 2;
gb->apu.noise_channel.delta = 0;
gb->apu.noise_channel.counter_countdown = divisor + 4;
if (divisor == 2) {
if (gb->model <= GB_MODEL_CGB_C) {
gb->apu.noise_channel.counter_countdown += gb->apu.lf_div;
if (!gb->cgb_double_speed) {
gb->apu.noise_channel.counter_countdown -= 1;
}
}
else {
gb->apu.noise_channel.counter_countdown += 1 - gb->apu.lf_div;
}
}
else {
if (gb->model <= GB_MODEL_CGB_C) {
gb->apu.noise_channel.counter_countdown += (uint8_t[]){2, 1, 4, 3}[gb->apu.noise_channel.alignment & 3];
}
else {
gb->apu.noise_channel.counter_countdown += (uint8_t[]){2, 1, 0, 3}[gb->apu.noise_channel.alignment & 3];
}
if (((gb->apu.noise_channel.alignment + 1) & 3) < 2) {
if ((gb->io_registers[GB_IO_NR43] & 0x07) == 1) {
gb->apu.noise_channel.counter_countdown -= 2;
gb->apu.noise_channel.delta = 2;
}
else {
gb->apu.noise_channel.counter_countdown -= 4;
}
}
}
2021-01-24 18:57:46 +00:00
/* TODO: These are quite weird. Verify further */
if (gb->model <= GB_MODEL_CGB_C) {
if (gb->cgb_double_speed) {
if (!(gb->io_registers[GB_IO_NR43] & 0xF0) && (gb->io_registers[GB_IO_NR43] & 0x07)) {
gb->apu.noise_channel.counter_countdown -= 1;
}
else if ((gb->io_registers[GB_IO_NR43] & 0xF0) && !(gb->io_registers[GB_IO_NR43] & 0x07)) {
gb->apu.noise_channel.counter_countdown += 1;
}
}
2021-01-24 18:57:46 +00:00
else {
gb->apu.noise_channel.counter_countdown -= 2;
}
}
gb->apu.noise_channel.current_volume = gb->io_registers[GB_IO_NR42] >> 4;
/* The volume changes caused by NRX4 sound start take effect instantly (i.e. the effect the previously
started sound). The playback itself is not instant which is why we don't update the sample for other
cases. */
if (gb->apu.is_active[GB_NOISE]) {
update_sample(gb, GB_NOISE,
gb->apu.noise_channel.current_lfsr_sample ?
gb->apu.noise_channel.current_volume : 0,
0);
}
gb->apu.noise_channel.lfsr = 0;
gb->apu.noise_channel.current_lfsr_sample = false;
gb->apu.noise_channel.volume_countdown = gb->io_registers[GB_IO_NR42] & 7;
if (!gb->apu.is_active[GB_NOISE] && (gb->io_registers[GB_IO_NR42] & 0xF8) != 0) {
gb->apu.is_active[GB_NOISE] = true;
update_sample(gb, GB_NOISE, 0, 0);
}
if (gb->apu.noise_channel.pulse_length == 0) {
gb->apu.noise_channel.pulse_length = 0x40;
gb->apu.noise_channel.length_enabled = false;
}
2017-08-11 14:57:08 +00:00
}
}
2017-08-11 14:57:08 +00:00
/* APU glitch - if length is enabled while the DIV-divider's LSB is 1, tick the length once. */
if ((value & 0x40) &&
!gb->apu.noise_channel.length_enabled &&
(gb->apu.div_divider & 1) &&
gb->apu.noise_channel.pulse_length) {
gb->apu.noise_channel.pulse_length--;
if (gb->apu.noise_channel.pulse_length == 0) {
if (value & 0x80) {
gb->apu.noise_channel.pulse_length = 0x3F;
}
else {
gb->apu.is_active[GB_NOISE] = false;
update_sample(gb, GB_NOISE, 0, 0);
2017-08-11 14:57:08 +00:00
}
}
}
gb->apu.noise_channel.length_enabled = value & 0x40;
break;
}
2016-03-30 20:07:55 +00:00
}
gb->io_registers[reg] = value;
}
void GB_set_sample_rate(GB_gameboy_t *gb, unsigned sample_rate)
{
2017-08-15 18:14:55 +00:00
gb->apu_output.sample_rate = sample_rate;
if (sample_rate) {
gb->apu_output.highpass_rate = pow(0.999958, GB_get_clock_rate(gb) / (double)sample_rate);
2017-08-15 18:14:55 +00:00
}
}
void GB_set_sample_rate_by_clocks(GB_gameboy_t *gb, double cycles_per_sample)
{
if (cycles_per_sample == 0) {
GB_set_sample_rate(gb, 0);
return;
}
gb->apu_output.sample_rate = GB_get_clock_rate(gb) / cycles_per_sample * 2;
gb->apu_output.highpass_rate = pow(0.999958, cycles_per_sample);
}
2022-05-20 15:36:54 +00:00
unsigned GB_get_sample_rate(GB_gameboy_t *gb)
{
return gb->apu_output.sample_rate;
}
void GB_apu_set_sample_callback(GB_gameboy_t *gb, GB_sample_callback_t callback)
{
gb->apu_output.sample_callback = callback;
}
2017-08-15 18:14:55 +00:00
void GB_set_highpass_filter_mode(GB_gameboy_t *gb, GB_highpass_mode_t mode)
{
gb->apu_output.highpass_mode = mode;
}
2020-12-30 22:06:36 +00:00
void GB_set_interference_volume(GB_gameboy_t *gb, double volume)
{
gb->apu_output.interference_volume = volume;
}
2022-05-20 15:36:54 +00:00
typedef struct __attribute__((packed)) {
uint32_t format_chunk; // = BE32('FORM')
uint32_t size; // = BE32(file size - 8)
uint32_t format; // = BE32('AIFC')
uint32_t fver_chunk; // = BE32('FVER')
uint32_t fver_size; // = BE32(4)
uint32_t fver;
uint32_t comm_chunk; // = BE32('COMM')
uint32_t comm_size; // = BE32(0x18)
uint16_t channels; // = BE16(2)
uint32_t samples_per_channel; // = BE32(total number of samples / 2)
uint16_t bit_depth; // = BE16(16)
uint16_t frequency_exponent;
uint64_t frequency_significand;
uint32_t compression_type; // = 'NONE' (BE) or 'twos' (LE)
uint16_t compression_name; // = 0
uint32_t ssnd_chunk; // = BE32('SSND')
uint32_t ssnd_size; // = BE32(length of samples - 8)
uint32_t ssnd_offset; // = 0
uint32_t ssnd_block; // = 0
} aiff_header_t;
typedef struct __attribute__((packed)) {
uint32_t marker; // = BE32('RIFF')
uint32_t size; // = LE32(file size - 8)
uint32_t type; // = BE32('WAVE')
uint32_t fmt_chunk; // = BE32('fmt ')
uint32_t fmt_size; // = LE16(16)
uint16_t format; // = LE16(1)
uint16_t channels; // = LE16(2)
uint32_t sample_rate; // = LE32(sample_rate)
uint32_t byte_rate; // = LE32(sample_rate * 4)
uint16_t frame_size; // = LE32(4)
uint16_t bit_depth; // = LE16(16)
uint32_t data_chunk; // = BE32('data')
uint32_t data_size; // = LE32(length of samples)
} wav_header_t;
int GB_start_audio_recording(GB_gameboy_t *gb, const char *path, GB_audio_format_t format)
{
if (gb->apu_output.sample_rate == 0) {
return EINVAL;
}
if (gb->apu_output.output_file) {
GB_stop_audio_recording(gb);
}
gb->apu_output.output_file = fopen(path, "wb");
if (!gb->apu_output.output_file) return errno;
gb->apu_output.output_format = format;
switch (format) {
case GB_AUDIO_FORMAT_RAW:
return 0;
case GB_AUDIO_FORMAT_AIFF: {
aiff_header_t header = {0,};
if (fwrite(&header, sizeof(header), 1, gb->apu_output.output_file) != 1) {
fclose(gb->apu_output.output_file);
gb->apu_output.output_file = NULL;
return errno;
}
return 0;
}
case GB_AUDIO_FORMAT_WAV: {
wav_header_t header = {0,};
if (fwrite(&header, sizeof(header), 1, gb->apu_output.output_file) != 1) {
fclose(gb->apu_output.output_file);
gb->apu_output.output_file = NULL;
return errno;
}
return 0;
}
default:
fclose(gb->apu_output.output_file);
gb->apu_output.output_file = NULL;
return EINVAL;
}
}
int GB_stop_audio_recording(GB_gameboy_t *gb)
{
if (!gb->apu_output.output_file) {
int ret = gb->apu_output.output_error ?: -1;
gb->apu_output.output_error = 0;
return ret;
}
gb->apu_output.output_error = 0;
switch (gb->apu_output.output_format) {
case GB_AUDIO_FORMAT_RAW:
break;
case GB_AUDIO_FORMAT_AIFF: {
size_t file_size = ftell(gb->apu_output.output_file);
size_t frames = (file_size - sizeof(aiff_header_t)) / sizeof(GB_sample_t);
aiff_header_t header = {
.format_chunk = BE32('FORM'),
.size = BE32(file_size - 8),
.format = BE32('AIFC'),
.fver_chunk = BE32('FVER'),
.fver_size = BE32(4),
.fver = BE32(0xA2805140),
.comm_chunk = BE32('COMM'),
.comm_size = BE32(0x18),
.channels = BE16(2),
.samples_per_channel = BE32(frames),
.bit_depth = BE16(16),
#ifdef GB_BIG_ENDIAN
.compression_type = 'NONE',
#else
.compression_type = 'twos',
#endif
.compression_name = 0,
.ssnd_chunk = BE32('SSND'),
.ssnd_size = BE32(frames * sizeof(GB_sample_t) - 8),
.ssnd_offset = 0,
.ssnd_block = 0,
};
uint64_t significand = gb->apu_output.sample_rate;
uint16_t exponent = 0x403E;
while ((int64_t)significand > 0) {
significand <<= 1;
exponent--;
}
header.frequency_exponent = BE16(exponent);
header.frequency_significand = BE64(significand);
fseek(gb->apu_output.output_file, 0, SEEK_SET);
if (fwrite(&header, sizeof(header), 1, gb->apu_output.output_file) != 1) {
gb->apu_output.output_error = errno;
}
break;
}
case GB_AUDIO_FORMAT_WAV: {
size_t file_size = ftell(gb->apu_output.output_file);
size_t frames = (file_size - sizeof(wav_header_t)) / sizeof(GB_sample_t);
wav_header_t header = {
.marker = BE32('RIFF'),
.size = LE32(file_size - 8),
.type = BE32('WAVE'),
.fmt_chunk = BE32('fmt '),
.fmt_size = LE16(16),
.format = LE16(1),
.channels = LE16(2),
.sample_rate = LE32(gb->apu_output.sample_rate),
.byte_rate = LE32(gb->apu_output.sample_rate * 4),
.frame_size = LE32(4),
.bit_depth = LE16(16),
.data_chunk = BE32('data'),
.data_size = LE32(frames * sizeof(GB_sample_t)),
};
fseek(gb->apu_output.output_file, 0, SEEK_SET);
if (fwrite(&header, sizeof(header), 1, gb->apu_output.output_file) != 1) {
gb->apu_output.output_error = errno;
}
break;
}
}
fclose(gb->apu_output.output_file);
gb->apu_output.output_file = NULL;
int ret = gb->apu_output.output_error;
gb->apu_output.output_error = 0;
return ret;
}