Merge branch 'master' into sgb
This commit is contained in:
commit
7b36ee10a4
16
Core/apu.c
16
Core/apu.c
@ -69,6 +69,7 @@ static void update_sample(GB_gameboy_t *gb, unsigned index, int8_t value, unsign
|
|||||||
static void render(GB_gameboy_t *gb, bool no_downsampling, GB_sample_t *dest)
|
static void render(GB_gameboy_t *gb, bool no_downsampling, GB_sample_t *dest)
|
||||||
{
|
{
|
||||||
GB_sample_t output = {0,0};
|
GB_sample_t output = {0,0};
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = GB_N_CHANNELS; i--;) {
|
for (unsigned i = GB_N_CHANNELS; i--;) {
|
||||||
double multiplier = CH_STEP;
|
double multiplier = CH_STEP;
|
||||||
if (!is_DAC_enabled(gb, i)) {
|
if (!is_DAC_enabled(gb, i)) {
|
||||||
@ -125,6 +126,7 @@ static void render(GB_gameboy_t *gb, bool no_downsampling, GB_sample_t *dest)
|
|||||||
unsigned mask = gb->io_registers[GB_IO_NR51];
|
unsigned mask = gb->io_registers[GB_IO_NR51];
|
||||||
unsigned left_volume = 0;
|
unsigned left_volume = 0;
|
||||||
unsigned right_volume = 0;
|
unsigned right_volume = 0;
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = GB_N_CHANNELS; i--;) {
|
for (unsigned i = GB_N_CHANNELS; i--;) {
|
||||||
if (gb->apu.is_active[i]) {
|
if (gb->apu.is_active[i]) {
|
||||||
if (mask & 1) {
|
if (mask & 1) {
|
||||||
@ -372,6 +374,7 @@ void GB_apu_run(GB_gameboy_t *gb)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = GB_SQUARE_2 + 1; i--;) {
|
for (unsigned i = GB_SQUARE_2 + 1; i--;) {
|
||||||
if (gb->apu.is_active[i]) {
|
if (gb->apu.is_active[i]) {
|
||||||
uint8_t cycles_left = cycles;
|
uint8_t cycles_left = cycles;
|
||||||
@ -454,10 +457,9 @@ void GB_apu_run(GB_gameboy_t *gb)
|
|||||||
|
|
||||||
if (gb->apu_output.sample_rate) {
|
if (gb->apu_output.sample_rate) {
|
||||||
gb->apu_output.cycles_since_render += cycles;
|
gb->apu_output.cycles_since_render += cycles;
|
||||||
double cycles_per_sample = 2 * GB_get_clock_rate(gb) / (double)gb->apu_output.sample_rate; /* 2 * because we use 8MHz units */
|
|
||||||
|
|
||||||
if (gb->apu_output.sample_cycles > cycles_per_sample) {
|
if (gb->apu_output.sample_cycles > gb->apu_output.cycles_per_sample) {
|
||||||
gb->apu_output.sample_cycles -= cycles_per_sample;
|
gb->apu_output.sample_cycles -= gb->apu_output.cycles_per_sample;
|
||||||
render(gb, false, NULL);
|
render(gb, false, NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -976,9 +978,17 @@ void GB_set_sample_rate(GB_gameboy_t *gb, unsigned int sample_rate)
|
|||||||
if (sample_rate) {
|
if (sample_rate) {
|
||||||
gb->apu_output.highpass_rate = pow(0.999958, GB_get_clock_rate(gb) / (double)sample_rate);
|
gb->apu_output.highpass_rate = pow(0.999958, GB_get_clock_rate(gb) / (double)sample_rate);
|
||||||
}
|
}
|
||||||
|
GB_apu_update_cycles_per_sample(gb);
|
||||||
}
|
}
|
||||||
|
|
||||||
void GB_set_highpass_filter_mode(GB_gameboy_t *gb, GB_highpass_mode_t mode)
|
void GB_set_highpass_filter_mode(GB_gameboy_t *gb, GB_highpass_mode_t mode)
|
||||||
{
|
{
|
||||||
gb->apu_output.highpass_mode = mode;
|
gb->apu_output.highpass_mode = mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GB_apu_update_cycles_per_sample(GB_gameboy_t *gb)
|
||||||
|
{
|
||||||
|
if (gb->apu_output.sample_rate) {
|
||||||
|
gb->apu_output.cycles_per_sample = 2 * GB_get_clock_rate(gb) / (double)gb->apu_output.sample_rate; /* 2 * because we use 8MHz units */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -140,6 +140,7 @@ typedef struct {
|
|||||||
volatile bool lock;
|
volatile bool lock;
|
||||||
|
|
||||||
double sample_cycles; // In 8 MHz units
|
double sample_cycles; // In 8 MHz units
|
||||||
|
double cycles_per_sample;
|
||||||
|
|
||||||
// Samples are NOT normalized to MAX_CH_AMP * 4 at this stage!
|
// Samples are NOT normalized to MAX_CH_AMP * 4 at this stage!
|
||||||
unsigned cycles_since_render;
|
unsigned cycles_since_render;
|
||||||
@ -164,6 +165,7 @@ uint8_t GB_apu_read(GB_gameboy_t *gb, uint8_t reg);
|
|||||||
void GB_apu_div_event(GB_gameboy_t *gb);
|
void GB_apu_div_event(GB_gameboy_t *gb);
|
||||||
void GB_apu_init(GB_gameboy_t *gb);
|
void GB_apu_init(GB_gameboy_t *gb);
|
||||||
void GB_apu_run(GB_gameboy_t *gb);
|
void GB_apu_run(GB_gameboy_t *gb);
|
||||||
|
void GB_apu_update_cycles_per_sample(GB_gameboy_t *gb);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif /* apu_h */
|
#endif /* apu_h */
|
||||||
|
@ -27,6 +27,7 @@ static GB_fifo_item_t *fifo_pop(GB_fifo_t *fifo)
|
|||||||
static void fifo_push_bg_row(GB_fifo_t *fifo, uint8_t lower, uint8_t upper, uint8_t palette, bool bg_priority, bool flip_x)
|
static void fifo_push_bg_row(GB_fifo_t *fifo, uint8_t lower, uint8_t upper, uint8_t palette, bool bg_priority, bool flip_x)
|
||||||
{
|
{
|
||||||
if (!flip_x) {
|
if (!flip_x) {
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = 8; i--;) {
|
for (unsigned i = 8; i--;) {
|
||||||
fifo->fifo[fifo->write_end] = (GB_fifo_item_t) {
|
fifo->fifo[fifo->write_end] = (GB_fifo_item_t) {
|
||||||
(lower >> 7) | ((upper >> 7) << 1),
|
(lower >> 7) | ((upper >> 7) << 1),
|
||||||
@ -42,6 +43,7 @@ static void fifo_push_bg_row(GB_fifo_t *fifo, uint8_t lower, uint8_t upper, uint
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = 8; i--;) {
|
for (unsigned i = 8; i--;) {
|
||||||
fifo->fifo[fifo->write_end] = (GB_fifo_item_t) {
|
fifo->fifo[fifo->write_end] = (GB_fifo_item_t) {
|
||||||
(lower & 1) | ((upper & 1) << 1),
|
(lower & 1) | ((upper & 1) << 1),
|
||||||
@ -68,6 +70,7 @@ static void fifo_overlay_object_row(GB_fifo_t *fifo, uint8_t lower, uint8_t uppe
|
|||||||
|
|
||||||
uint8_t flip_xor = flip_x? 0: 0x7;
|
uint8_t flip_xor = flip_x? 0: 0x7;
|
||||||
|
|
||||||
|
#pragma unroll
|
||||||
for (unsigned i = 8; i--;) {
|
for (unsigned i = 8; i--;) {
|
||||||
uint8_t pixel = (lower >> 7) | ((upper >> 7) << 1);
|
uint8_t pixel = (lower >> 7) | ((upper >> 7) << 1);
|
||||||
GB_fifo_item_t *target = &fifo->fifo[(fifo->read_end + (i ^ flip_xor)) & (GB_FIFO_LENGTH - 1)];
|
GB_fifo_item_t *target = &fifo->fifo[(fifo->read_end + (i ^ flip_xor)) & (GB_FIFO_LENGTH - 1)];
|
||||||
@ -1086,6 +1089,7 @@ uint8_t GB_get_oam_info(GB_gameboy_t *gb, GB_oam_info_t *dest, uint8_t *sprite_h
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (unsigned y = 0; y < *sprite_height; y++) {
|
for (unsigned y = 0; y < *sprite_height; y++) {
|
||||||
|
#pragma unroll
|
||||||
for (unsigned x = 0; x < 8; x++) {
|
for (unsigned x = 0; x < 8; x++) {
|
||||||
uint8_t color = (((gb->vram[vram_address ] >> ((~x)&7)) & 1 ) |
|
uint8_t color = (((gb->vram[vram_address ] >> ((~x)&7)) & 1 ) |
|
||||||
((gb->vram[vram_address + 1] >> ((~x)&7)) & 1) << 1 );
|
((gb->vram[vram_address + 1] >> ((~x)&7)) & 1) << 1 );
|
||||||
|
@ -680,6 +680,8 @@ void GB_reset(GB_gameboy_t *gb)
|
|||||||
/* Todo: Ugly, fixme, see comment in the timer state machine */
|
/* Todo: Ugly, fixme, see comment in the timer state machine */
|
||||||
gb->div_state = 3;
|
gb->div_state = 3;
|
||||||
|
|
||||||
|
GB_apu_update_cycles_per_sample(gb);
|
||||||
|
|
||||||
gb->magic = (uintptr_t)'SAME';
|
gb->magic = (uintptr_t)'SAME';
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -767,6 +769,7 @@ void *GB_get_direct_access(GB_gameboy_t *gb, GB_direct_access_t access, size_t *
|
|||||||
void GB_set_clock_multiplier(GB_gameboy_t *gb, double multiplier)
|
void GB_set_clock_multiplier(GB_gameboy_t *gb, double multiplier)
|
||||||
{
|
{
|
||||||
gb->clock_multiplier = multiplier;
|
gb->clock_multiplier = multiplier;
|
||||||
|
GB_apu_update_cycles_per_sample(gb);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t GB_get_clock_rate(GB_gameboy_t *gb)
|
uint32_t GB_get_clock_rate(GB_gameboy_t *gb)
|
||||||
|
@ -193,7 +193,7 @@ static uint8_t read_high_memory(GB_gameboy_t *gb, uint16_t addr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (addr < 0xFE00) {
|
if (addr < 0xFE00) {
|
||||||
return gb->ram[addr & 0x0FFF];
|
return read_banked_ram(gb, addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr < 0xFF00) {
|
if (addr < 0xFF00) {
|
||||||
@ -412,7 +412,7 @@ static GB_read_function_t * const read_map[] =
|
|||||||
read_vram, read_vram, /* 8XXX, 9XXX */
|
read_vram, read_vram, /* 8XXX, 9XXX */
|
||||||
read_mbc_ram, read_mbc_ram, /* AXXX, BXXX */
|
read_mbc_ram, read_mbc_ram, /* AXXX, BXXX */
|
||||||
read_ram, read_banked_ram, /* CXXX, DXXX */
|
read_ram, read_banked_ram, /* CXXX, DXXX */
|
||||||
read_high_memory, read_high_memory, /* EXXX FXXX */
|
read_ram, read_high_memory, /* EXXX FXXX */
|
||||||
};
|
};
|
||||||
|
|
||||||
uint8_t GB_read_memory(GB_gameboy_t *gb, uint16_t addr)
|
uint8_t GB_read_memory(GB_gameboy_t *gb, uint16_t addr)
|
||||||
@ -540,7 +540,7 @@ static void write_high_memory(GB_gameboy_t *gb, uint16_t addr, uint8_t value)
|
|||||||
{
|
{
|
||||||
if (addr < 0xFE00) {
|
if (addr < 0xFE00) {
|
||||||
GB_log(gb, "Wrote %02x to %04x (RAM Mirror)\n", value, addr);
|
GB_log(gb, "Wrote %02x to %04x (RAM Mirror)\n", value, addr);
|
||||||
gb->ram[addr & 0x0FFF] = value;
|
write_banked_ram(gb, addr, value);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -913,7 +913,7 @@ static GB_write_function_t * const write_map[] =
|
|||||||
write_vram, write_vram, /* 8XXX, 9XXX */
|
write_vram, write_vram, /* 8XXX, 9XXX */
|
||||||
write_mbc_ram, write_mbc_ram, /* AXXX, BXXX */
|
write_mbc_ram, write_mbc_ram, /* AXXX, BXXX */
|
||||||
write_ram, write_banked_ram, /* CXXX, DXXX */
|
write_ram, write_banked_ram, /* CXXX, DXXX */
|
||||||
write_high_memory, write_high_memory, /* EXXX FXXX */
|
write_ram, write_high_memory, /* EXXX FXXX */
|
||||||
};
|
};
|
||||||
|
|
||||||
void GB_write_memory(GB_gameboy_t *gb, uint16_t addr, uint8_t value)
|
void GB_write_memory(GB_gameboy_t *gb, uint16_t addr, uint8_t value)
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static const unsigned int GB_TAC_RATIOS[] = {1024, 16, 64, 256};
|
static const unsigned int GB_TAC_TRIGGER_BITS[] = {512, 8, 32, 128};
|
||||||
|
|
||||||
#ifndef DISABLE_TIMEKEEPING
|
#ifndef DISABLE_TIMEKEEPING
|
||||||
static int64_t get_nanoseconds(void)
|
static int64_t get_nanoseconds(void)
|
||||||
@ -108,11 +108,6 @@ static void advance_tima_state_machine(GB_gameboy_t *gb)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool counter_overflow_check(uint32_t old, uint32_t new, uint32_t max)
|
|
||||||
{
|
|
||||||
return (old & (max >> 1)) && !(new & (max >> 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void increase_tima(GB_gameboy_t *gb)
|
static void increase_tima(GB_gameboy_t *gb)
|
||||||
{
|
{
|
||||||
gb->io_registers[GB_IO_TIMA]++;
|
gb->io_registers[GB_IO_TIMA]++;
|
||||||
@ -126,13 +121,13 @@ static void GB_set_internal_div_counter(GB_gameboy_t *gb, uint32_t value)
|
|||||||
{
|
{
|
||||||
/* TIMA increases when a specific high-bit becomes a low-bit. */
|
/* TIMA increases when a specific high-bit becomes a low-bit. */
|
||||||
value &= INTERNAL_DIV_CYCLES - 1;
|
value &= INTERNAL_DIV_CYCLES - 1;
|
||||||
if ((gb->io_registers[GB_IO_TAC] & 4) &&
|
uint32_t triggers = gb->div_counter & ~value;
|
||||||
counter_overflow_check(gb->div_counter, value, GB_TAC_RATIOS[gb->io_registers[GB_IO_TAC] & 3])) {
|
if ((gb->io_registers[GB_IO_TAC] & 4) && (triggers & GB_TAC_TRIGGER_BITS[gb->io_registers[GB_IO_TAC] & 3])) {
|
||||||
increase_tima(gb);
|
increase_tima(gb);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO: Can switching to double speed mode trigger an event? */
|
/* TODO: Can switching to double speed mode trigger an event? */
|
||||||
if (counter_overflow_check(gb->div_counter, value, gb->cgb_double_speed? 0x4000 : 0x2000)) {
|
if (triggers & (gb->cgb_double_speed? 0x2000 : 0x1000)) {
|
||||||
GB_apu_run(gb);
|
GB_apu_run(gb);
|
||||||
GB_apu_div_event(gb);
|
GB_apu_div_event(gb);
|
||||||
}
|
}
|
||||||
@ -221,13 +216,13 @@ void GB_emulate_timer_glitch(GB_gameboy_t *gb, uint8_t old_tac, uint8_t new_tac)
|
|||||||
/* Glitch only happens when old_tac is enabled. */
|
/* Glitch only happens when old_tac is enabled. */
|
||||||
if (!(old_tac & 4)) return;
|
if (!(old_tac & 4)) return;
|
||||||
|
|
||||||
unsigned int old_clocks = GB_TAC_RATIOS[old_tac & 3];
|
unsigned int old_clocks = GB_TAC_TRIGGER_BITS[old_tac & 3];
|
||||||
unsigned int new_clocks = GB_TAC_RATIOS[new_tac & 3];
|
unsigned int new_clocks = GB_TAC_TRIGGER_BITS[new_tac & 3];
|
||||||
|
|
||||||
/* The bit used for overflow testing must have been 1 */
|
/* The bit used for overflow testing must have been 1 */
|
||||||
if (gb->div_counter & (old_clocks >> 1)) {
|
if (gb->div_counter & old_clocks) {
|
||||||
/* And now either the timer must be disabled, or the new bit used for overflow testing be 0. */
|
/* And now either the timer must be disabled, or the new bit used for overflow testing be 0. */
|
||||||
if (!(new_tac & 4) || gb->div_counter & (new_clocks >> 1)) {
|
if (!(new_tac & 4) || gb->div_counter & new_clocks) {
|
||||||
increase_tima(gb);
|
increase_tima(gb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user