mirror of https://github.com/bsnes-emu/bsnes.git
Update to v068r04 release.
(there was no r03 release posted to the WIP thread) byuu says: This should provide hardware-accurate mosaic support in the accurate renderer, with the exception that I'm still not sure what mid-frame vertical mosaic or mid-scanline horizontal mosaic writes do. Either the code I have is correct, or it bypasses the mosaic adjust and gives the exact H/V positions. I've also renamed the fast folder to alternative (thinking about naming it simply alt instead), and started on a brand new PPU renderer. So far it's just a barebones setup with some MMIO support and VRAM/OAM/CGRAM writing. I'm not even confident that I can get this to be faster than the current scanline renderer, but it's the only avenue that we have left for any kind of significant bsnes speedup, so I have to try. I'm going to finish up the MMIO stuff first, that way we have a clean slate with no actual rendering. And then from here we can try various different approaches.
This commit is contained in:
parent
39b1acb177
commit
c434e8a0d5
|
@ -87,6 +87,6 @@ clean: ui_clean
|
||||||
-@$(call delete,*.manifest)
|
-@$(call delete,*.manifest)
|
||||||
|
|
||||||
archive-all:
|
archive-all:
|
||||||
tar -cjf bsnes-`date +%Y%m%d`.tar.bz2 launcher libco nall obj out qt ruby snes Makefile sync.sh cc.bat clean.bat
|
tar -cjf bsnes.tar.bz2 launcher libco nall obj out qt ruby snes Makefile sync.sh cc.bat clean.bat
|
||||||
|
|
||||||
help:;
|
help:;
|
||||||
|
|
|
@ -20,14 +20,14 @@ else ifeq ($(profile),compatibility)
|
||||||
flags += -DPROFILE_COMPATIBILITY
|
flags += -DPROFILE_COMPATIBILITY
|
||||||
snescpu := $(snes)/cpu
|
snescpu := $(snes)/cpu
|
||||||
snessmp := $(snes)/smp
|
snessmp := $(snes)/smp
|
||||||
snesdsp := $(snes)/fast/dsp
|
snesdsp := $(snes)/alternative/dsp
|
||||||
snesppu := $(snes)/fast/ppu
|
snesppu := $(snes)/alternative/ppu
|
||||||
else ifeq ($(profile),performance)
|
else ifeq ($(profile),performance)
|
||||||
flags += -DPROFILE_PERFORMANCE
|
flags += -DPROFILE_PERFORMANCE
|
||||||
snescpu := $(snes)/fast/cpu
|
snescpu := $(snes)/alternative/cpu
|
||||||
snessmp := $(snes)/smp
|
snessmp := $(snes)/smp
|
||||||
snesdsp := $(snes)/fast/dsp
|
snesdsp := $(snes)/alternative/dsp
|
||||||
snesppu := $(snes)/fast/ppu
|
snesppu := $(snes)/alternative/ppu-fast
|
||||||
endif
|
endif
|
||||||
|
|
||||||
obj/libco.o : libco/libco.c libco/*
|
obj/libco.o : libco/libco.c libco/*
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
#ifdef PPU_CPP
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,7 @@
|
||||||
|
#ifdef PPU_CPP
|
||||||
|
|
||||||
|
bool PPUDebugger::property(unsigned id, string &name, string &value) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,10 @@
|
||||||
|
class PPUDebugger : public PPU, public ChipDebugger {
|
||||||
|
public:
|
||||||
|
bool property(unsigned id, string &name, string &value);
|
||||||
|
|
||||||
|
bool bg1_enabled[2];
|
||||||
|
bool bg2_enabled[2];
|
||||||
|
bool bg3_enabled[2];
|
||||||
|
bool bg4_enabled[2];
|
||||||
|
bool oam_enabled[4];
|
||||||
|
};
|
|
@ -0,0 +1,48 @@
|
||||||
|
#ifdef PPU_CPP
|
||||||
|
|
||||||
|
void PPU::latch_counters() {
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16 PPU::get_vram_addr() {
|
||||||
|
uint16 addr = regs.vram_addr;
|
||||||
|
switch(regs.vram_mapping) {
|
||||||
|
case 0: break;
|
||||||
|
case 1: addr = (addr & 0xff00) | ((addr & 0x001f) << 3) | ((addr >> 5) & 7); break;
|
||||||
|
case 2: addr = (addr & 0xfe00) | ((addr & 0x003f) << 3) | ((addr >> 6) & 7); break;
|
||||||
|
case 3: addr = (addr & 0xfc00) | ((addr & 0x007f) << 3) | ((addr >> 7) & 7); break;
|
||||||
|
}
|
||||||
|
return (addr << 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8 PPU::vram_read(unsigned addr) {
|
||||||
|
if(regs.display_disabled == true) return memory::vram[addr];
|
||||||
|
if(cpu.vcounter() >= regs.height) return memory::vram[addr];
|
||||||
|
return 0x00;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::vram_write(unsigned addr, uint8 data) {
|
||||||
|
if(regs.display_disabled == true) { memory::vram[addr] = data; return; }
|
||||||
|
if(cpu.vcounter() >= regs.height) { memory::vram[addr] = data; return; }
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8 PPU::oam_read(unsigned addr) {
|
||||||
|
if(regs.display_disabled == true) return memory::oam[addr];
|
||||||
|
if(cpu.vcounter() >= regs.height) return memory::oam[addr];
|
||||||
|
return memory::oam[0x0218];
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::oam_write(unsigned addr, uint8 data) {
|
||||||
|
if(regs.display_disabled == true) { memory::oam[addr] = data; return; }
|
||||||
|
if(cpu.vcounter() >= regs.height) { memory::oam[addr] = data; return; }
|
||||||
|
memory::oam[0x0218] = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8 PPU::cgram_read(unsigned addr) {
|
||||||
|
return memory::cgram[addr];
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::cgram_write(unsigned addr, uint8 data) {
|
||||||
|
memory::cgram[addr] = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,227 @@
|
||||||
|
#ifdef PPU_CPP
|
||||||
|
|
||||||
|
uint8 PPU::mmio_read(unsigned addr) {
|
||||||
|
switch(addr & 0xffff) {
|
||||||
|
}
|
||||||
|
return 0x00;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::mmio_write(unsigned addr, uint8 data) {
|
||||||
|
switch(addr & 0xffff) {
|
||||||
|
case 0x2100: {
|
||||||
|
regs.display_disabled = data & 0x80;
|
||||||
|
regs.display_brightness = data & 0x0f;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2101: {
|
||||||
|
regs.oam_basesize = (data >> 5) & 7;
|
||||||
|
regs.oam_nameselect = (data >> 3) & 3;
|
||||||
|
regs.oam_tdaddr = (data & 3) << 14;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2102: {
|
||||||
|
regs.oam_baseaddr = (regs.oam_baseaddr & 0xff00) | (data << 0);
|
||||||
|
regs.oam_baseaddr &= 0x01ff;
|
||||||
|
regs.oam_addr = regs.oam_baseaddr << 1;
|
||||||
|
regs.oam_firstsprite = (regs.oam_priority == false ? 0 : (regs.oam_addr >> 2) & 127);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2103: {
|
||||||
|
regs.oam_priority = data & 0x80;
|
||||||
|
regs.oam_baseaddr = (data << 8) | (regs.oam_baseaddr & 0x00ff);
|
||||||
|
regs.oam_baseaddr &= 0x01ff;
|
||||||
|
regs.oam_addr = regs.oam_baseaddr << 1;
|
||||||
|
regs.oam_firstsprite = (regs.oam_priority == false ? 0 : (regs.oam_addr >> 2) & 127);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2104: {
|
||||||
|
if(regs.oam_addr & 0x0200) {
|
||||||
|
oam_write(regs.oam_addr, data);
|
||||||
|
} else if((regs.oam_addr & 1) == 0) {
|
||||||
|
regs.oam_latchdata = data;
|
||||||
|
} else {
|
||||||
|
oam_write((regs.oam_addr & ~1) + 0, regs.oam_latchdata);
|
||||||
|
oam_write((regs.oam_addr & ~1) + 1, data);
|
||||||
|
}
|
||||||
|
regs.oam_addr = (regs.oam_addr + 1) & 0x03ff;
|
||||||
|
regs.oam_firstsprite = (regs.oam_priority == false ? 0 : (regs.oam_addr >> 2) & 127);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2105: {
|
||||||
|
regs.bg_tilesize[BG4] = data & 0x80;
|
||||||
|
regs.bg_tilesize[BG3] = data & 0x40;
|
||||||
|
regs.bg_tilesize[BG2] = data & 0x20;
|
||||||
|
regs.bg_tilesize[BG1] = data & 0x10;
|
||||||
|
regs.bg3_priority = data & 0x08;
|
||||||
|
regs.bg_mode = data & 0x07;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2106: {
|
||||||
|
regs.mosaic_size = (data >> 4) & 15;
|
||||||
|
regs.mosaic_enabled[BG4] = data & 0x08;
|
||||||
|
regs.mosaic_enabled[BG3] = data & 0x04;
|
||||||
|
regs.mosaic_enabled[BG2] = data & 0x02;
|
||||||
|
regs.mosaic_enabled[BG1] = data & 0x01;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2107: {
|
||||||
|
regs.bg_scaddr[BG1] = (data & 0x7c) << 9;
|
||||||
|
regs.bg_scsize[BG1] = (data & 0x03);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2108: {
|
||||||
|
regs.bg_scaddr[BG2] = (data & 0x7c) << 9;
|
||||||
|
regs.bg_scsize[BG2] = (data & 0x03);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2109: {
|
||||||
|
regs.bg_scaddr[BG3] = (data & 0x7c) << 9;
|
||||||
|
regs.bg_scsize[BG3] = (data & 0x03);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x210a: {
|
||||||
|
regs.bg_scaddr[BG4] = (data & 0x7c) << 9;
|
||||||
|
regs.bg_scsize[BG4] = (data & 0x03);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x210b: {
|
||||||
|
regs.bg_tdaddr[BG1] = (data & 0x07) << 13;
|
||||||
|
regs.bg_tdaddr[BG2] = (data & 0x70) << 9;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x210c: {
|
||||||
|
regs.bg_tdaddr[BG3] = (data & 0x07) << 13;
|
||||||
|
regs.bg_tdaddr[BG4] = (data & 0x70) << 9;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2115: {
|
||||||
|
regs.vram_incmode = data & 0x80;
|
||||||
|
regs.vram_mapping = (data >> 2) & 3;
|
||||||
|
switch(data & 3) {
|
||||||
|
case 0: regs.vram_incsize = 1; break;
|
||||||
|
case 1: regs.vram_incsize = 32; break;
|
||||||
|
case 2: regs.vram_incsize = 128; break;
|
||||||
|
case 3: regs.vram_incsize = 128; break;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2116: {
|
||||||
|
regs.vram_addr = (regs.vram_addr & 0xff00) | (data << 0);
|
||||||
|
uint16 addr = get_vram_addr();
|
||||||
|
regs.vram_readbuffer = vram_read(addr + 0) << 0;
|
||||||
|
regs.vram_readbuffer |= vram_read(addr + 1) << 8;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2117: {
|
||||||
|
regs.vram_addr = (data << 8) | (regs.vram_addr & 0x00ff);
|
||||||
|
uint16 addr = get_vram_addr();
|
||||||
|
regs.vram_readbuffer = vram_read(addr + 0) << 0;
|
||||||
|
regs.vram_readbuffer |= vram_read(addr + 1) << 8;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2118: {
|
||||||
|
uint16 addr = get_vram_addr() + 0;
|
||||||
|
vram_write(addr, data);
|
||||||
|
if(regs.vram_incmode == 0) regs.vram_addr += regs.vram_incsize;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2119: {
|
||||||
|
uint16 addr = get_vram_addr() + 1;
|
||||||
|
vram_write(addr, data);
|
||||||
|
if(regs.vram_incmode == 1) regs.vram_addr += regs.vram_incsize;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2121: {
|
||||||
|
regs.cgram_addr = data << 1;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 0x2122: {
|
||||||
|
if((regs.cgram_addr & 1) == 0) {
|
||||||
|
regs.cgram_latchdata = data;
|
||||||
|
} else {
|
||||||
|
cgram_write((regs.cgram_addr & ~1) + 0, regs.cgram_latchdata);
|
||||||
|
cgram_write((regs.cgram_addr & ~1) + 1, data & 0x7f);
|
||||||
|
}
|
||||||
|
regs.cgram_addr = (regs.cgram_addr + 1) & 0x01ff;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::mmio_reset() {
|
||||||
|
//internal
|
||||||
|
regs.width = 256;
|
||||||
|
regs.height = 225;
|
||||||
|
|
||||||
|
//$2100
|
||||||
|
regs.display_disabled = true;
|
||||||
|
regs.display_brightness = 0;
|
||||||
|
|
||||||
|
//$2101
|
||||||
|
regs.oam_basesize = 0;
|
||||||
|
regs.oam_nameselect = 0;
|
||||||
|
regs.oam_tdaddr = 0;
|
||||||
|
|
||||||
|
//$2102-$2103
|
||||||
|
regs.oam_baseaddr = 0;
|
||||||
|
regs.oam_addr = 0;
|
||||||
|
regs.oam_priority = 0;
|
||||||
|
regs.oam_firstsprite = 0;
|
||||||
|
|
||||||
|
//$2104
|
||||||
|
regs.oam_latchdata = 0;
|
||||||
|
|
||||||
|
//$2105
|
||||||
|
for(unsigned i = 0; i < 4; i++) regs.bg_tilesize[i] = 0;
|
||||||
|
regs.bg3_priority = 0;
|
||||||
|
regs.bg_mode = 0;
|
||||||
|
|
||||||
|
//$2106
|
||||||
|
regs.mosaic_size = 0;
|
||||||
|
for(unsigned i = 0; i < 4; i++) regs.mosaic_enabled[i] = 0;
|
||||||
|
|
||||||
|
//$2107-$210a
|
||||||
|
for(unsigned i = 0; i < 4; i++) regs.bg_scaddr[i] = 0;
|
||||||
|
for(unsigned i = 0; i < 4; i++) regs.bg_scsize[i] = 0;
|
||||||
|
|
||||||
|
//$210b-$210c
|
||||||
|
for(unsigned i = 0; i < 4; i++) regs.bg_tdaddr[i] = 0;
|
||||||
|
|
||||||
|
//$2115
|
||||||
|
regs.vram_incmode = 0;
|
||||||
|
regs.vram_mapping = 0;
|
||||||
|
regs.vram_incsize = 0;
|
||||||
|
|
||||||
|
//$2116-$2117
|
||||||
|
regs.vram_addr = 0;
|
||||||
|
|
||||||
|
//$2121
|
||||||
|
regs.cgram_addr = 0;
|
||||||
|
|
||||||
|
//$2122
|
||||||
|
regs.cgram_latchdata = 0;
|
||||||
|
|
||||||
|
//$2139-$213a
|
||||||
|
regs.vram_readbuffer = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,124 @@
|
||||||
|
#include <snes.hpp>
|
||||||
|
|
||||||
|
#define PPU_CPP
|
||||||
|
namespace SNES {
|
||||||
|
|
||||||
|
#if defined(DEBUGGER)
|
||||||
|
#include "debugger/debugger.cpp"
|
||||||
|
PPUDebugger ppu;
|
||||||
|
#else
|
||||||
|
PPU ppu;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "background.cpp"
|
||||||
|
#include "memory.cpp"
|
||||||
|
#include "mmio.cpp"
|
||||||
|
#include "screen.cpp"
|
||||||
|
|
||||||
|
void PPU::step(unsigned clocks) {
|
||||||
|
clock += clocks;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::synchronize_cpu() {
|
||||||
|
if(CPU::Threaded == true) {
|
||||||
|
if(clock >= 0 && scheduler.sync != Scheduler::SynchronizeMode::All) co_switch(cpu.thread);
|
||||||
|
} else {
|
||||||
|
while(clock >= 0) cpu.enter();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::Enter() { ppu.enter(); }
|
||||||
|
|
||||||
|
void PPU::enter() {
|
||||||
|
while(true) {
|
||||||
|
if(scheduler.sync == Scheduler::SynchronizeMode::All) {
|
||||||
|
scheduler.exit(Scheduler::ExitReason::SynchronizeEvent);
|
||||||
|
}
|
||||||
|
|
||||||
|
scanline();
|
||||||
|
add_clocks(512);
|
||||||
|
if(vcounter() < regs.height) render_scanline();
|
||||||
|
add_clocks(lineclocks() - 512);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::add_clocks(unsigned clocks) {
|
||||||
|
tick(clocks);
|
||||||
|
step(clocks);
|
||||||
|
synchronize_cpu();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::scanline() {
|
||||||
|
if(vcounter() == 0) frame();
|
||||||
|
regs.width = !hires() ? 256 : 512;
|
||||||
|
regs.height = !overscan() ? 225 : 240;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::frame() {
|
||||||
|
system.frame();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::render_scanline() {
|
||||||
|
screen_render();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::power() {
|
||||||
|
for(unsigned i = 0; i < memory::vram.size(); i++) memory::vram[i] = 0x00;
|
||||||
|
for(unsigned i = 0; i < memory::oam.size(); i++) memory::oam[i] = 0x00;
|
||||||
|
for(unsigned i = 0; i < memory::cgram.size(); i++) memory::cgram[i] = 0x00;
|
||||||
|
reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::reset() {
|
||||||
|
create(Enter, system.cpu_frequency());
|
||||||
|
PPUcounter::reset();
|
||||||
|
memset(surface, 0, 512 * 512 * sizeof(uint16));
|
||||||
|
|
||||||
|
mmio_reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
PPU::PPU() {
|
||||||
|
surface = new uint16[512 * 512];
|
||||||
|
output = surface + 16 * 512;
|
||||||
|
|
||||||
|
light_table = new uint16*[16];
|
||||||
|
for(unsigned l = 0; l < 16; l++) {
|
||||||
|
light_table[l] = new uint16[32768];
|
||||||
|
for(unsigned r = 0; r < 32; r++) {
|
||||||
|
for(unsigned g = 0; g < 32; g++) {
|
||||||
|
for(unsigned b = 0; b < 32; b++) {
|
||||||
|
double luma = (double)l / 15.0;
|
||||||
|
unsigned ar = (luma * r + 0.5);
|
||||||
|
unsigned ag = (luma * g + 0.5);
|
||||||
|
unsigned ab = (luma * b + 0.5);
|
||||||
|
light_table[l][(r << 10) + (g << 5) + (b << 0)] = (ab << 10) + (ag << 5) + (ar << 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
PPU::~PPU() {
|
||||||
|
delete[] surface;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PPU::interlace() const { return false; }
|
||||||
|
bool PPU::overscan() const { return false; }
|
||||||
|
bool PPU::hires() const { return false; }
|
||||||
|
|
||||||
|
void PPUcounter::serialize(serializer &s) {
|
||||||
|
s.integer(status.interlace);
|
||||||
|
s.integer(status.field);
|
||||||
|
s.integer(status.vcounter);
|
||||||
|
s.integer(status.hcounter);
|
||||||
|
|
||||||
|
s.array(history.field);
|
||||||
|
s.array(history.vcounter);
|
||||||
|
s.array(history.hcounter);
|
||||||
|
s.integer(history.index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::serialize(serializer &s) {
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,118 @@
|
||||||
|
class PPU : public Processor, public PPUcounter, public MMIO {
|
||||||
|
public:
|
||||||
|
enum : bool { Threaded = true };
|
||||||
|
alwaysinline void step(unsigned clocks);
|
||||||
|
alwaysinline void synchronize_cpu();
|
||||||
|
|
||||||
|
void latch_counters();
|
||||||
|
bool interlace() const;
|
||||||
|
bool overscan() const;
|
||||||
|
bool hires() const;
|
||||||
|
|
||||||
|
uint8 mmio_read(unsigned addr);
|
||||||
|
void mmio_write(unsigned addr, uint8 data);
|
||||||
|
|
||||||
|
void enter();
|
||||||
|
void power();
|
||||||
|
void reset();
|
||||||
|
|
||||||
|
void serialize(serializer&);
|
||||||
|
PPU();
|
||||||
|
~PPU();
|
||||||
|
|
||||||
|
private:
|
||||||
|
enum : unsigned { BG1 = 0, BG2 = 1, BG3 = 2, BG4 = 3, OAM = 4, COL = 5, BACK = 5 };
|
||||||
|
uint16 *surface;
|
||||||
|
uint16 *output;
|
||||||
|
|
||||||
|
//background.cpp
|
||||||
|
|
||||||
|
//memory.cpp
|
||||||
|
uint16 get_vram_addr();
|
||||||
|
uint8 vram_read(unsigned addr);
|
||||||
|
void vram_write(unsigned addr, uint8 data);
|
||||||
|
uint8 oam_read(unsigned addr);
|
||||||
|
void oam_write(unsigned addr, uint8 data);
|
||||||
|
uint8 cgram_read(unsigned addr);
|
||||||
|
void cgram_write(unsigned addr, uint8 data);
|
||||||
|
|
||||||
|
//mmio.cpp
|
||||||
|
struct Regs {
|
||||||
|
//internal
|
||||||
|
unsigned width;
|
||||||
|
unsigned height;
|
||||||
|
|
||||||
|
//$2100
|
||||||
|
bool display_disabled;
|
||||||
|
unsigned display_brightness;
|
||||||
|
|
||||||
|
//$2101
|
||||||
|
unsigned oam_basesize;
|
||||||
|
unsigned oam_nameselect;
|
||||||
|
uint16 oam_tdaddr;
|
||||||
|
|
||||||
|
//$2102-$2103
|
||||||
|
uint16 oam_baseaddr;
|
||||||
|
uint16 oam_addr;
|
||||||
|
bool oam_priority;
|
||||||
|
unsigned oam_firstsprite;
|
||||||
|
|
||||||
|
//$2104
|
||||||
|
uint8 oam_latchdata;
|
||||||
|
|
||||||
|
//$2105
|
||||||
|
bool bg_tilesize[4];
|
||||||
|
bool bg3_priority;
|
||||||
|
unsigned bg_mode;
|
||||||
|
|
||||||
|
//$2106
|
||||||
|
unsigned mosaic_size;
|
||||||
|
bool mosaic_enabled[4];
|
||||||
|
|
||||||
|
//$2107-$210a
|
||||||
|
uint16 bg_scaddr[4];
|
||||||
|
unsigned bg_scsize[4];
|
||||||
|
|
||||||
|
//$210b-$210c
|
||||||
|
uint16 bg_tdaddr[4];
|
||||||
|
|
||||||
|
//$2115
|
||||||
|
bool vram_incmode;
|
||||||
|
unsigned vram_mapping;
|
||||||
|
unsigned vram_incsize;
|
||||||
|
|
||||||
|
//$2116-$2117
|
||||||
|
uint16 vram_addr;
|
||||||
|
|
||||||
|
//$2121
|
||||||
|
uint16 cgram_addr;
|
||||||
|
|
||||||
|
//$2122
|
||||||
|
uint8 cgram_latchdata;
|
||||||
|
|
||||||
|
//$2139-$213a
|
||||||
|
uint16 vram_readbuffer;
|
||||||
|
} regs;
|
||||||
|
|
||||||
|
void mmio_reset();
|
||||||
|
|
||||||
|
//ppu.cpp
|
||||||
|
static void Enter();
|
||||||
|
void add_clocks(unsigned clocks);
|
||||||
|
void scanline();
|
||||||
|
void frame();
|
||||||
|
void render_scanline();
|
||||||
|
|
||||||
|
//screen.cpp
|
||||||
|
uint16 **light_table;
|
||||||
|
void screen_render();
|
||||||
|
|
||||||
|
friend class Video;
|
||||||
|
};
|
||||||
|
|
||||||
|
#if defined(DEBUGGER)
|
||||||
|
#include "debugger/debugger.hpp"
|
||||||
|
extern PPUDebugger ppu;
|
||||||
|
#else
|
||||||
|
extern PPU ppu;
|
||||||
|
#endif
|
|
@ -0,0 +1,17 @@
|
||||||
|
#ifdef PPU_CPP
|
||||||
|
|
||||||
|
void PPU::screen_render() {
|
||||||
|
uint16 *data = output + vcounter() * 1024;
|
||||||
|
if(regs.display_disabled) {
|
||||||
|
memset(data, 0x00, regs.width << 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16 *table = light_table[regs.display_brightness];
|
||||||
|
uint16 color = memory::cgram[0];
|
||||||
|
color |= memory::cgram[1] << 8;
|
||||||
|
color = table[color];
|
||||||
|
for(unsigned i = 0; i < 256; i++) data[i] = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -7,17 +7,15 @@ void PPU::Background::frame() {
|
||||||
|
|
||||||
void PPU::Background::scanline() {
|
void PPU::Background::scanline() {
|
||||||
if(self.vcounter() == 1) {
|
if(self.vcounter() == 1) {
|
||||||
t.mosaic_y = regs.mosaic ? 1 : 0; //TODO: this is most likely incorrect
|
t.mosaic_vcounter = regs.mosaic + 1;
|
||||||
t.mosaic_vcounter = 0;
|
t.mosaic_y = 1;
|
||||||
}
|
} else if(--t.mosaic_vcounter == 0) {
|
||||||
|
t.mosaic_vcounter = regs.mosaic + 1;
|
||||||
if(t.mosaic_vcounter++ == regs.mosaic) {
|
|
||||||
t.mosaic_vcounter = 0;
|
|
||||||
t.mosaic_y += regs.mosaic + 1;
|
t.mosaic_y += regs.mosaic + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
t.mosaic_x = 0;
|
t.mosaic_x = 0;
|
||||||
t.mosaic_hcounter = 0;
|
t.mosaic_hcounter = regs.mosaic + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PPU::Background::run() {
|
void PPU::Background::run() {
|
||||||
|
@ -36,8 +34,8 @@ void PPU::Background::run() {
|
||||||
|
|
||||||
unsigned x = t.mosaic_x;
|
unsigned x = t.mosaic_x;
|
||||||
unsigned y = t.mosaic_y;
|
unsigned y = t.mosaic_y;
|
||||||
if(t.mosaic_hcounter++ == regs.mosaic) {
|
if(--t.mosaic_hcounter == 0) {
|
||||||
t.mosaic_hcounter = 0;
|
t.mosaic_hcounter = regs.mosaic + 1;
|
||||||
t.mosaic_x += regs.mosaic + 1;
|
t.mosaic_x += regs.mosaic + 1;
|
||||||
}
|
}
|
||||||
if(regs.mode == Mode::Mode7) return run_mode7();
|
if(regs.mode == Mode::Mode7) return run_mode7();
|
||||||
|
|
|
@ -4,5 +4,5 @@ namespace Info {
|
||||||
|
|
||||||
#include <cpu/cpu.hpp>
|
#include <cpu/cpu.hpp>
|
||||||
#include <smp/smp.hpp>
|
#include <smp/smp.hpp>
|
||||||
#include <fast/dsp/dsp.hpp>
|
#include <alternative/dsp/dsp.hpp>
|
||||||
#include <fast/ppu/ppu.hpp>
|
#include <alternative/ppu/ppu.hpp>
|
||||||
|
|
|
@ -2,7 +2,7 @@ namespace Info {
|
||||||
static const char Profile[] = "Performance";
|
static const char Profile[] = "Performance";
|
||||||
}
|
}
|
||||||
|
|
||||||
#include <fast/cpu/cpu.hpp>
|
#include <alternative/cpu/cpu.hpp>
|
||||||
#include <smp/smp.hpp>
|
#include <smp/smp.hpp>
|
||||||
#include <fast/dsp/dsp.hpp>
|
#include <alternative/dsp/dsp.hpp>
|
||||||
#include <fast/ppu/ppu.hpp>
|
#include <alternative/ppu-fast/ppu.hpp>
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
namespace SNES {
|
namespace SNES {
|
||||||
namespace Info {
|
namespace Info {
|
||||||
static const char Name[] = "bsnes";
|
static const char Name[] = "bsnes";
|
||||||
static const char Version[] = "068.02";
|
static const char Version[] = "068.04";
|
||||||
static const unsigned SerializerVersion = 13;
|
static const unsigned SerializerVersion = 13;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue