2016-03-30 23:07:55 +03:00
|
|
|
#ifndef apu_h
|
|
|
|
#define apu_h
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
2018-05-27 19:30:23 +03:00
|
|
|
#include <stddef.h>
|
2017-04-17 20:16:17 +03:00
|
|
|
#include "gb_struct_def.h"
|
2017-07-21 18:24:28 +03:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef GB_INTERNAL
|
2018-02-16 18:01:50 +02:00
|
|
|
/* Speed = 1 / Length (in seconds) */
|
2019-05-25 19:12:09 +03:00
|
|
|
#define DAC_DECAY_SPEED 20000
|
|
|
|
#define DAC_ATTACK_SPEED 20000
|
2018-10-19 23:53:01 +03:00
|
|
|
|
2018-02-16 18:01:50 +02:00
|
|
|
|
2018-01-06 11:58:07 +02:00
|
|
|
/* Divides nicely and never overflows with 4 channels and 8 (1-8) volume levels */
|
2018-02-07 15:27:28 -05:00
|
|
|
#ifdef WIIU
|
2018-02-10 15:02:22 +02:00
|
|
|
/* Todo: Remove this hack once https://github.com/libretro/RetroArch/issues/6252 is fixed*/
|
2018-02-16 01:26:37 +02:00
|
|
|
#define MAX_CH_AMP (0xFF0 / 2)
|
2018-02-07 15:27:28 -05:00
|
|
|
#else
|
2018-02-16 01:26:37 +02:00
|
|
|
#define MAX_CH_AMP 0xFF0
|
2018-02-07 15:27:28 -05:00
|
|
|
#endif
|
2018-01-06 11:58:07 +02:00
|
|
|
#define CH_STEP (MAX_CH_AMP/0xF/8)
|
2017-07-21 18:24:28 +03:00
|
|
|
#endif
|
2016-03-30 23:07:55 +03:00
|
|
|
|
2018-02-16 18:01:50 +02:00
|
|
|
|
|
|
|
|
2017-08-02 21:14:23 +03:00
|
|
|
/* APU ticks are 2MHz, triggered by an internal APU clock. */
|
2016-03-30 23:07:55 +03:00
|
|
|
|
2016-06-10 15:28:50 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int16_t left;
|
|
|
|
int16_t right;
|
|
|
|
} GB_sample_t;
|
|
|
|
|
2017-08-15 21:14:55 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
double left;
|
|
|
|
double right;
|
|
|
|
} GB_double_sample_t;
|
|
|
|
|
2017-07-21 18:24:28 +03:00
|
|
|
enum GB_CHANNELS {
|
|
|
|
GB_SQUARE_1,
|
|
|
|
GB_SQUARE_2,
|
|
|
|
GB_WAVE,
|
|
|
|
GB_NOISE,
|
|
|
|
GB_N_CHANNELS
|
|
|
|
};
|
2016-03-30 23:07:55 +03:00
|
|
|
|
2019-06-15 23:22:27 +03:00
|
|
|
typedef void (*GB_sample_callback_t)(GB_gameboy_t *gb, GB_sample_t *sample);
|
|
|
|
|
2016-03-30 23:07:55 +03:00
|
|
|
typedef struct
|
|
|
|
{
|
2016-09-13 01:21:47 +03:00
|
|
|
bool global_enable;
|
2017-07-27 23:11:33 +03:00
|
|
|
uint8_t apu_cycles;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-21 18:24:28 +03:00
|
|
|
uint8_t samples[GB_N_CHANNELS];
|
|
|
|
bool is_active[GB_N_CHANNELS];
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-02 21:14:23 +03:00
|
|
|
uint8_t div_divider; // The DIV register ticks the APU at 512Hz, but is then divided
|
|
|
|
// once more to generate 128Hz and 64Hz clocks
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-11 17:57:08 +03:00
|
|
|
uint8_t lf_div; // The APU runs in 2MHz, but channels 1, 2 and 4 run in 1MHZ so we divide
|
|
|
|
// need to divide the signal.
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-27 23:11:33 +03:00
|
|
|
uint8_t square_sweep_countdown; // In 128Hz
|
2017-08-15 22:05:20 +03:00
|
|
|
uint8_t square_sweep_calculate_countdown; // In 2 MHz
|
2020-11-28 19:31:25 +02:00
|
|
|
uint16_t sweep_length_addend;
|
2020-02-27 18:11:10 +01:00
|
|
|
uint16_t shadow_sweep_sample_length;
|
2020-12-01 14:17:35 +02:00
|
|
|
GB_PADDING(bool, sweep_enabled);
|
2020-11-28 19:31:25 +02:00
|
|
|
GB_PADDING(bool, sweep_decreasing);
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-27 23:11:33 +03:00
|
|
|
struct {
|
2017-08-02 21:14:23 +03:00
|
|
|
uint16_t pulse_length; // Reloaded from NRX1 (xorred), in 256Hz DIV ticks
|
2017-07-27 23:11:33 +03:00
|
|
|
uint8_t current_volume; // Reloaded from NRX2
|
|
|
|
uint8_t volume_countdown; // Reloaded from NRX2
|
2018-10-29 00:44:43 +02:00
|
|
|
uint8_t current_sample_index; /* For save state compatibility,
|
|
|
|
highest bit is reused (See NR14/NR24's
|
|
|
|
write code)*/
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-10 19:42:23 +03:00
|
|
|
uint16_t sample_countdown; // in APU ticks (Reloaded from sample_length, xorred $7FF)
|
2017-08-02 21:14:23 +03:00
|
|
|
uint16_t sample_length; // From NRX3, NRX4, in APU ticks
|
2017-07-27 23:11:33 +03:00
|
|
|
bool length_enabled; // NRX4
|
|
|
|
|
|
|
|
} square_channels[2];
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-21 18:24:28 +03:00
|
|
|
struct {
|
|
|
|
bool enable; // NR30
|
2017-08-02 21:14:23 +03:00
|
|
|
uint16_t pulse_length; // Reloaded from NR31 (xorred), in 256Hz DIV ticks
|
2017-07-21 18:24:28 +03:00
|
|
|
uint8_t shift; // NR32
|
|
|
|
uint16_t sample_length; // NR33, NR34, in APU ticks
|
|
|
|
bool length_enabled; // NR34
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-10 19:42:23 +03:00
|
|
|
uint16_t sample_countdown; // in APU ticks (Reloaded from sample_length, xorred $7FF)
|
2017-07-21 18:24:28 +03:00
|
|
|
uint8_t current_sample_index;
|
|
|
|
uint8_t current_sample; // Current sample before shifting.
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-21 18:24:28 +03:00
|
|
|
int8_t wave_form[32];
|
|
|
|
bool wave_form_just_read;
|
|
|
|
} wave_channel;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-11 17:57:08 +03:00
|
|
|
struct {
|
|
|
|
uint16_t pulse_length; // Reloaded from NR41 (xorred), in 256Hz DIV ticks
|
|
|
|
uint8_t current_volume; // Reloaded from NR42
|
|
|
|
uint8_t volume_countdown; // Reloaded from NR42
|
|
|
|
uint16_t lfsr;
|
|
|
|
bool narrow;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2020-12-12 16:02:25 +02:00
|
|
|
uint8_t counter_countdown; // Counts from 0-7 to 0 to tick counter (Scaled from 512KHz to 2MHz)
|
|
|
|
uint8_t __padding;
|
|
|
|
uint16_t counter; // A bit from this 14-bit register ticks LFSR
|
2017-08-11 17:57:08 +03:00
|
|
|
bool length_enabled; // NR44
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-12 19:50:39 +03:00
|
|
|
uint8_t alignment; // If (NR43 & 7) != 0, samples are aligned to 512KHz clock instead of
|
|
|
|
// 1MHz. This variable keeps track of the alignment.
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-11 17:57:08 +03:00
|
|
|
} noise_channel;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2020-02-01 23:36:16 +02:00
|
|
|
#define GB_SKIP_DIV_EVENT_INACTIVE 0
|
|
|
|
#define GB_SKIP_DIV_EVENT_SKIPPED 1
|
|
|
|
#define GB_SKIP_DIV_EVENT_SKIP 2
|
|
|
|
uint8_t skip_div_event;
|
2018-07-04 21:55:12 +03:00
|
|
|
bool current_lfsr_sample;
|
2020-05-10 00:37:52 +03:00
|
|
|
uint8_t pcm_mask[2]; // For CGB-0 to CGB-C PCM read glitch
|
2020-12-01 22:37:13 +02:00
|
|
|
uint8_t channel_1_restart_hold;
|
2020-12-12 16:02:25 +02:00
|
|
|
int8_t channel_4_delta;
|
|
|
|
bool channel_4_countdown_reloaded;
|
2020-12-12 18:13:55 +02:00
|
|
|
uint8_t channel_4_dmg_delayed_start;
|
2016-03-30 23:07:55 +03:00
|
|
|
} GB_apu_t;
|
|
|
|
|
2017-08-15 21:14:55 +03:00
|
|
|
typedef enum {
|
|
|
|
GB_HIGHPASS_OFF, // Do not apply any filter, keep DC offset
|
|
|
|
GB_HIGHPASS_ACCURATE, // Apply a highpass filter similar to the one used on hardware
|
|
|
|
GB_HIGHPASS_REMOVE_DC_OFFSET, // Remove DC Offset without affecting the waveform
|
2017-12-23 21:11:44 +02:00
|
|
|
GB_HIGHPASS_MAX
|
2017-08-15 21:14:55 +03:00
|
|
|
} GB_highpass_mode_t;
|
|
|
|
|
2017-07-21 23:06:02 +03:00
|
|
|
typedef struct {
|
|
|
|
unsigned sample_rate;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2018-02-20 21:17:12 +02:00
|
|
|
double sample_cycles; // In 8 MHz units
|
2019-01-01 00:42:40 +02:00
|
|
|
double cycles_per_sample;
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-07-21 23:06:02 +03:00
|
|
|
// Samples are NOT normalized to MAX_CH_AMP * 4 at this stage!
|
|
|
|
unsigned cycles_since_render;
|
|
|
|
unsigned last_update[GB_N_CHANNELS];
|
|
|
|
GB_sample_t current_sample[GB_N_CHANNELS];
|
|
|
|
GB_sample_t summed_samples[GB_N_CHANNELS];
|
2018-02-16 18:01:50 +02:00
|
|
|
double dac_discharge[GB_N_CHANNELS];
|
2019-05-15 12:39:08 +02:00
|
|
|
|
2017-08-15 21:14:55 +03:00
|
|
|
GB_highpass_mode_t highpass_mode;
|
|
|
|
double highpass_rate;
|
|
|
|
GB_double_sample_t highpass_diff;
|
2019-06-15 23:22:27 +03:00
|
|
|
|
|
|
|
GB_sample_callback_t sample_callback;
|
2019-10-08 15:10:24 +03:00
|
|
|
|
|
|
|
bool rate_set_in_clocks;
|
2020-12-31 00:06:36 +02:00
|
|
|
double interference_volume;
|
|
|
|
double interference_highpass;
|
2017-07-21 23:06:02 +03:00
|
|
|
} GB_apu_output_t;
|
|
|
|
|
2019-06-15 23:22:27 +03:00
|
|
|
void GB_set_sample_rate(GB_gameboy_t *gb, unsigned sample_rate);
|
2019-11-03 22:02:33 +02:00
|
|
|
void GB_set_sample_rate_by_clocks(GB_gameboy_t *gb, double cycles_per_sample); /* Cycles are in 8MHz units */
|
2017-08-15 21:14:55 +03:00
|
|
|
void GB_set_highpass_filter_mode(GB_gameboy_t *gb, GB_highpass_mode_t mode);
|
2020-12-31 00:06:36 +02:00
|
|
|
void GB_set_interference_volume(GB_gameboy_t *gb, double volume);
|
2019-06-15 23:22:27 +03:00
|
|
|
void GB_apu_set_sample_callback(GB_gameboy_t *gb, GB_sample_callback_t callback);
|
2020-12-31 00:06:36 +02:00
|
|
|
|
2017-04-17 20:16:17 +03:00
|
|
|
#ifdef GB_INTERNAL
|
2019-05-15 12:39:08 +02:00
|
|
|
bool GB_apu_is_DAC_enabled(GB_gameboy_t *gb, unsigned index);
|
2016-06-18 20:29:11 +03:00
|
|
|
void GB_apu_write(GB_gameboy_t *gb, uint8_t reg, uint8_t value);
|
|
|
|
uint8_t GB_apu_read(GB_gameboy_t *gb, uint8_t reg);
|
2017-07-21 18:24:28 +03:00
|
|
|
void GB_apu_div_event(GB_gameboy_t *gb);
|
2016-06-18 20:29:11 +03:00
|
|
|
void GB_apu_init(GB_gameboy_t *gb);
|
2017-07-27 23:11:33 +03:00
|
|
|
void GB_apu_run(GB_gameboy_t *gb);
|
2019-01-01 00:42:40 +02:00
|
|
|
void GB_apu_update_cycles_per_sample(GB_gameboy_t *gb);
|
2020-02-08 13:28:46 +02:00
|
|
|
void GB_borrow_sgb_border(GB_gameboy_t *gb);
|
2017-04-17 20:16:17 +03:00
|
|
|
#endif
|
2016-03-30 23:07:55 +03:00
|
|
|
|
|
|
|
#endif /* apu_h */
|