Update to v095r03 release and icarus 20151107.

byuu says:

Note: you will need the new icarus (and please use the "no manifest"
system) to run GBA games with this WIP.

Changelog:
- fixed caching of r(d) to pass armwrestler tests [Jonas Quinn]
- DMA to/from GBA BIOS should fail [Cydrak]
- fixed sign-extend and rotate on ldrs instructions [Cydrak]
- fixed 8-bit SRAM reading/writing [byuu]
- refactored GBA/cartridge
  - cartridge/rom,ram.type is now cartridge/mrom,sram,eeprom,flash
  - things won't crash horribly if you specify a RAM size larger than
    the largest legal size in the manifest
  - specialized MROM / SRAM classes replace all the shared read/write
    functions that didn't work right anyway
- there's a new ruby/video.glx2 driver, which is not enabled by default
  - use this if you are running Linux/BSD, but don't have OpenGL 3.2 yet
  - I'm not going to support OpenGL2 on Windows/OS X, because these OSes
    don't ship ancient video card drivers
- probably more. What am I, clairvoyant? :P

For endrift's tests, this gets us to 1348/1552 memory and 1016/1260
timing. Overall, this puts us back in second place. Only no$ is ahead
on memory, but bgba is even more ahead on timing.
This commit is contained in:
Tim Allen 2015-11-08 20:09:18 +11:00
parent b42ab2fcb3
commit 0fe55e3f5b
32 changed files with 607 additions and 280 deletions

View File

@ -3,12 +3,11 @@
#include <nall/nall.hpp> #include <nall/nall.hpp>
#include <nall/dsp.hpp> #include <nall/dsp.hpp>
#include <nall/priority-queue.hpp>
using namespace nall; using namespace nall;
namespace Emulator { namespace Emulator {
static const string Name = "higan"; static const string Name = "higan";
static const string Version = "095.02"; static const string Version = "095.03";
static const string Author = "byuu"; static const string Author = "byuu";
static const string License = "GPLv3"; static const string License = "GPLv3";
static const string Website = "http://byuu.org/"; static const string Website = "http://byuu.org/";

View File

@ -2,170 +2,128 @@
namespace GameBoyAdvance { namespace GameBoyAdvance {
#include "mrom.cpp"
#include "sram.cpp"
#include "eeprom.cpp" #include "eeprom.cpp"
#include "flashrom.cpp" #include "flash.cpp"
#include "serialization.cpp" #include "serialization.cpp"
Cartridge cartridge; Cartridge cartridge;
string Cartridge::title() { Cartridge::Cartridge() {
loaded = false;
mrom.data = new uint8[mrom.size = 32 * 1024 * 1024];
sram.data = new uint8[sram.size = 32 * 1024];
eeprom.data = new uint8[eeprom.size = 8 * 1024];
flash.data = new uint8[flash.size = 128 * 1024];
}
Cartridge::~Cartridge() {
delete[] mrom.data;
delete[] sram.data;
delete[] eeprom.data;
delete[] flash.data;
}
auto Cartridge::title() const -> string {
return information.title; return information.title;
} }
void Cartridge::load() { auto Cartridge::load() -> void {
interface->loadRequest(ID::Manifest, "manifest.bml", true); interface->loadRequest(ID::Manifest, "manifest.bml", true);
auto document = BML::unserialize(information.markup); auto document = BML::unserialize(information.markup);
information.title = document["information/title"].text(); information.title = document["information/title"].text();
unsigned rom_size = 0; unsigned mrom_size = 0;
if(document["cartridge/rom"]) { if(auto info = document["cartridge/mrom"]) {
auto info = document["cartridge/rom"]; interface->loadRequest(ID::MROM, info["name"].text(), true);
interface->loadRequest(ID::ROM, info["name"].text(), true); mrom_size = info["size"].decimal();
rom_size = info["size"].decimal(); for(unsigned addr = mrom_size; addr < mrom.size; addr++) {
for(unsigned addr = rom_size; addr < rom.size; addr++) { mrom.data[addr] = mrom.data[Bus::mirror(addr, mrom_size)];
rom.data[addr] = rom.data[Bus::mirror(addr, rom_size)];
} }
} }
has_sram = false; has_sram = false;
has_eeprom = false; has_eeprom = false;
has_flashrom = false; has_flash = false;
if(document["cartridge/ram"]) { if(auto info = document["cartridge/sram"]) {
auto info = document["cartridge/ram"];
if(info["type"].text() == "SRAM" || info["type"].text() == "FRAM") {
has_sram = true; has_sram = true;
ram.size = info["size"].decimal(); sram.size = min(32 * 1024, info["size"].decimal());
ram.mask = ram.size - 1; sram.mask = sram.size - 1;
for(unsigned n = 0; n < ram.size; n++) ram.data[n] = 0xff; for(auto n : range(sram.size)) sram.data[n] = 0xff;
interface->loadRequest(ID::RAM, info["name"].text(), false); interface->loadRequest(ID::SRAM, info["name"].text(), false);
memory.append({ID::RAM, info["name"].text()}); memory.append({ID::SRAM, info["name"].text()});
} }
if(info["type"].text() == "EEPROM") { if(auto info = document["cartridge/eeprom"]) {
has_eeprom = true; has_eeprom = true;
eeprom.size = info["size"].decimal(); eeprom.size = min(8 * 1024, info["size"].decimal());
eeprom.bits = eeprom.size <= 512 ? 6 : 14; eeprom.bits = eeprom.size <= 512 ? 6 : 14;
if(eeprom.size == 0) eeprom.size = 8192, eeprom.bits = 0; //auto-detect size if(eeprom.size == 0) eeprom.size = 8192, eeprom.bits = 0; //auto-detect size
eeprom.mask = rom_size > 16 * 1024 * 1024 ? 0x0fffff00 : 0x0f000000; eeprom.mask = mrom_size > 16 * 1024 * 1024 ? 0x0fffff00 : 0x0f000000;
eeprom.test = rom_size > 16 * 1024 * 1024 ? 0x0dffff00 : 0x0d000000; eeprom.test = mrom_size > 16 * 1024 * 1024 ? 0x0dffff00 : 0x0d000000;
for(unsigned n = 0; n < eeprom.size; n++) eeprom.data[n] = 0xff; for(auto n : range(eeprom.size)) eeprom.data[n] = 0xff;
interface->loadRequest(ID::EEPROM, info["name"].text(), false); interface->loadRequest(ID::EEPROM, info["name"].text(), false);
memory.append({ID::EEPROM, info["name"].text()}); memory.append({ID::EEPROM, info["name"].text()});
} }
if(info["type"].text() == "FlashROM") { if(auto info = document["cartridge/flash"]) {
has_flashrom = true; has_flash = true;
flashrom.id = info["id"].decimal(); flash.id = info["id"].decimal();
flashrom.size = info["size"].decimal(); flash.size = min(128 * 1024, info["size"].decimal());
for(unsigned n = 0; n < flashrom.size; n++) flashrom.data[n] = 0xff; for(auto n : range(flash.size)) flash.data[n] = 0xff;
//if FlashROM ID not provided; guess that it's a Macronix chip //if flash ID not provided; guess that it's a Macronix chip
//this will not work for all games; in which case, the ID must be specified manually //this will not work for all games; in which case, the ID must be specified manually
if(!flashrom.id && flashrom.size == 64 * 1024) flashrom.id = 0x1cc2; if(!flash.id && flash.size == 64 * 1024) flash.id = 0x1cc2;
if(!flashrom.id && flashrom.size == 128 * 1024) flashrom.id = 0x09c2; if(!flash.id && flash.size == 128 * 1024) flash.id = 0x09c2;
interface->loadRequest(ID::FlashROM, info["name"].text(), false); interface->loadRequest(ID::FLASH, info["name"].text(), false);
memory.append({ID::FlashROM, info["name"].text()}); memory.append({ID::FLASH, info["name"].text()});
}
} }
sha256 = Hash::SHA256(rom.data, rom_size).digest(); sha256 = Hash::SHA256(mrom.data, mrom_size).digest();
system.load(); system.load();
loaded = true; loaded = true;
} }
void Cartridge::unload() { auto Cartridge::unload() -> void {
if(loaded == false) return; if(loaded) {
loaded = false; loaded = false;
memory.reset(); memory.reset();
} }
}
void Cartridge::power() { auto Cartridge::power() -> void {
eeprom.power(); eeprom.power();
flashrom.power(); flash.power();
}
uint8* Cartridge::ram_data() {
if(has_sram) return ram.data;
if(has_eeprom) return eeprom.data;
if(has_flashrom) return flashrom.data;
return nullptr;
}
unsigned Cartridge::ram_size() {
if(has_sram) return ram.size;
if(has_eeprom) return eeprom.size;
if(has_flashrom) return flashrom.size;
return 0;
}
auto Cartridge::read(uint8 *data, unsigned mode, uint32 addr) -> uint32 {
if(mode & Word) addr &= ~3;
if(mode & Half) addr &= ~1;
data += addr;
if(mode & Word) return data[0] << 0 | data[1] << 8 | data[2] << 16 | data[3] << 24;
if(mode & Half) return data[0] << 0 | data[1] << 8;
if(mode & Byte) return data[0];
return 0; //should never occur
}
auto Cartridge::write(uint8 *data, unsigned mode, uint32 addr, uint32 word) -> void {
if(mode & Word) addr &= ~3;
if(mode & Half) addr &= ~1;
data += addr;
if(mode & Word) {
data[0] = word >> 0;
data[1] = word >> 8;
data[2] = word >> 16;
data[3] = word >> 24;
} else if(mode & Half) {
data[0] = word >> 0;
data[1] = word >> 8;
} else if(mode & Byte) {
data[0] = word >> 0;
}
} }
#define RAM_ANALYZE #define RAM_ANALYZE
auto Cartridge::rom_read(unsigned mode, uint32 addr) -> uint32 { auto Cartridge::read(unsigned mode, uint32 addr) -> uint32 {
if(addr < 0x0e00'0000) {
if(has_eeprom && (addr & eeprom.mask) == eeprom.test) return eeprom.read(); if(has_eeprom && (addr & eeprom.mask) == eeprom.test) return eeprom.read();
return read(rom.data, mode, addr & 0x01ff'ffff); return mrom.read(mode, addr);
} } else {
if(has_sram) return sram.read(mode, addr);
auto Cartridge::rom_write(unsigned mode, uint32 addr, uint32 word) -> void { if(has_flash) return flash.read(addr);
if(has_eeprom && (addr & eeprom.mask) == eeprom.test) return eeprom.write(word & 1);
}
auto Cartridge::ram_read(unsigned mode, uint32 addr) -> uint32 {
if(has_sram) return read(ram.data, mode, addr & ram.mask);
if(has_flashrom) return flashrom.read(addr);
return cpu.pipeline.fetch.instruction; return cpu.pipeline.fetch.instruction;
} }
auto Cartridge::ram_write(unsigned mode, uint32 addr, uint32 word) -> void {
if(has_sram) return write(ram.data, mode, addr & ram.mask, word);
if(has_flashrom) return flashrom.write(addr, word);
} }
Cartridge::Cartridge() { auto Cartridge::write(unsigned mode, uint32 addr, uint32 word) -> void {
loaded = false; if(addr < 0x0e00'0000) {
rom.data = new uint8[rom.size = 32 * 1024 * 1024]; if(has_eeprom && (addr & eeprom.mask) == eeprom.test) return eeprom.write(word & 1);
ram.data = new uint8[ram.size = 32 * 1024]; return mrom.write(mode, addr, word);
eeprom.data = new uint8[eeprom.size = 8 * 1024]; } else {
flashrom.data = new uint8[flashrom.size = 128 * 1024]; if(has_sram) return sram.write(mode, addr, word);
if(has_flash) return flash.write(addr, word);
} }
Cartridge::~Cartridge() {
delete[] rom.data;
delete[] ram.data;
delete[] eeprom.data;
delete[] flashrom.data;
} }
} }

View File

@ -6,41 +6,32 @@ struct Cartridge : property<Cartridge> {
readonly<bool> has_sram; readonly<bool> has_sram;
readonly<bool> has_eeprom; readonly<bool> has_eeprom;
readonly<bool> has_flashrom; readonly<bool> has_flash;
struct Information { struct Information {
string markup; string markup;
string title; string title;
} information; } information;
string title();
struct Media { struct Media {
unsigned id; unsigned id;
string name; string name;
}; };
vector<Media> memory; vector<Media> memory;
void load();
void unload();
void power();
uint8* ram_data();
unsigned ram_size();
auto read(uint8* data, unsigned mode, uint32 addr) -> uint32;
auto write(uint8* data, unsigned mode, uint32 addr, uint32 word) -> void;
auto rom_read(unsigned mode, uint32 addr) -> uint32;
auto rom_write(unsigned mode, uint32 addr, uint32 word) -> void;
auto ram_read(unsigned mode, uint32 addr) -> uint32;
auto ram_write(unsigned mode, uint32 addr, uint32 word) -> void;
void serialize(serializer&);
Cartridge(); Cartridge();
~Cartridge(); ~Cartridge();
auto title() const -> string;
auto load() -> void;
auto unload() -> void;
auto power() -> void;
auto read(unsigned mode, uint32 addr) -> uint32;
auto write(unsigned mode, uint32 addr, uint32 word) -> void;
auto serialize(serializer&) -> void;
}; };
extern Cartridge cartridge; extern Cartridge cartridge;

View File

@ -1,13 +1,13 @@
bool Cartridge::EEPROM::read(unsigned addr) { auto Cartridge::EEPROM::read(unsigned addr) -> bool {
return data[addr >> 3] & 0x80 >> (addr & 7); return data[addr >> 3] & 0x80 >> (addr & 7);
} }
void Cartridge::EEPROM::write(unsigned addr, bool bit) { auto Cartridge::EEPROM::write(unsigned addr, bool bit) -> void {
if(bit == 0) data[addr >> 3] &=~ (0x80 >> (addr & 7)); if(bit == 0) data[addr >> 3] &=~ (0x80 >> (addr & 7));
if(bit == 1) data[addr >> 3] |= (0x80 >> (addr & 7)); if(bit == 1) data[addr >> 3] |= (0x80 >> (addr & 7));
} }
bool Cartridge::EEPROM::read() { auto Cartridge::EEPROM::read() -> bool {
bool bit = 1; bool bit = 1;
//EEPROM size auto-detection //EEPROM size auto-detection
@ -28,7 +28,7 @@ bool Cartridge::EEPROM::read() {
return bit; return bit;
} }
void Cartridge::EEPROM::write(bool bit) { auto Cartridge::EEPROM::write(bool bit) -> void {
if(mode == Mode::Wait) { if(mode == Mode::Wait) {
if(bit == 1) mode = Mode::Command; if(bit == 1) mode = Mode::Command;
} }
@ -76,13 +76,13 @@ void Cartridge::EEPROM::write(bool bit) {
} }
} }
void Cartridge::EEPROM::power() { auto Cartridge::EEPROM::power() -> void {
mode = Mode::Wait; mode = Mode::Wait;
offset = 0; offset = 0;
address = 0; address = 0;
} }
void Cartridge::EEPROM::serialize(serializer& s) { auto Cartridge::EEPROM::serialize(serializer& s) -> void {
s.array(data, size); s.array(data, size);
s.integer(size); s.integer(size);
s.integer(mask); s.integer(mask);

View File

@ -7,7 +7,7 @@
//0x1362 128KB 32x4096 Sanyo //0x1362 128KB 32x4096 Sanyo
//0x09c2 128KB 32x4096 Macronix //0x09c2 128KB 32x4096 Macronix
uint8 Cartridge::FlashROM::read(uint16 addr) { auto Cartridge::FLASH::read(uint16 addr) -> uint8 {
if(idmode) { if(idmode) {
if(addr == 0x0000) return id >> 0; if(addr == 0x0000) return id >> 0;
if(addr == 0x0001) return id >> 8; if(addr == 0x0001) return id >> 8;
@ -17,7 +17,7 @@ uint8 Cartridge::FlashROM::read(uint16 addr) {
return data[bank << 16 | addr]; return data[bank << 16 | addr];
} }
void Cartridge::FlashROM::write(uint16 addr, uint8 byte) { auto Cartridge::FLASH::write(uint16 addr, uint8 byte) -> void {
if(bankselect) { if(bankselect) {
bankselect = false; bankselect = false;
//bank select is only applicable on 128KB chips //bank select is only applicable on 128KB chips
@ -77,7 +77,7 @@ void Cartridge::FlashROM::write(uint16 addr, uint8 byte) {
} }
} }
void Cartridge::FlashROM::power() { auto Cartridge::FLASH::power() -> void {
unlockhi = false; unlockhi = false;
unlocklo = false; unlocklo = false;
idmode = false; idmode = false;
@ -86,7 +86,7 @@ void Cartridge::FlashROM::power() {
bank = 0; bank = 0;
} }
void Cartridge::FlashROM::serialize(serializer& s) { auto Cartridge::FLASH::serialize(serializer& s) -> void {
s.array(data, size); s.array(data, size);
s.integer(size); s.integer(size);
s.integer(id); s.integer(id);

View File

@ -1,31 +1,53 @@
struct Memory { struct Memory {
auto ror(uint32 word, unsigned shift) -> uint32 {
return word << 32 - shift | word >> shift;
}
};
struct MROM : Memory {
uint8* data; uint8* data;
unsigned size; unsigned size;
unsigned mask; unsigned mask;
} rom, ram;
struct EEPROM { auto read(unsigned mode, uint32 addr) -> uint32;
auto write(unsigned mode, uint32 addr, uint32 word) -> void;
} mrom;
struct SRAM : Memory {
uint8* data;
unsigned size;
unsigned mask;
auto read(unsigned mode, uint32 addr) -> uint32;
auto write(unsigned mode, uint32 addr, uint32 word) -> void;
auto serialize(serializer&) -> void;
} sram;
struct EEPROM : Memory {
uint8* data; uint8* data;
unsigned size; unsigned size;
unsigned mask; unsigned mask;
unsigned test; unsigned test;
unsigned bits; unsigned bits;
enum class Mode : unsigned { Wait, Command, ReadAddress, ReadValidate, ReadData, WriteAddress, WriteData, WriteValidate } mode; enum class Mode : unsigned {
Wait, Command, ReadAddress, ReadValidate, ReadData, WriteAddress, WriteData, WriteValidate
} mode;
unsigned offset; unsigned offset;
unsigned address; unsigned address;
unsigned addressbits; unsigned addressbits;
bool read(unsigned addr); auto read(unsigned addr) -> bool;
void write(unsigned addr, bool bit); auto write(unsigned addr, bool bit) -> void;
bool read(); auto read() -> bool;
void write(bool bit); auto write(bool bit) -> void;
void power(); auto power() -> void;
void serialize(serializer&); auto serialize(serializer&) -> void;
} eeprom; } eeprom;
struct FlashROM { struct FLASH : Memory {
uint8* data; uint8* data;
unsigned size; unsigned size;
uint16 id; uint16 id;
@ -38,8 +60,9 @@ struct FlashROM {
bool writeselect; bool writeselect;
bool bank; bool bank;
uint8 read(uint16 addr); auto read(uint16 addr) -> uint8;
void write(uint16 addr, uint8 byte); auto write(uint16 addr, uint8 byte) -> void;
void power();
void serialize(serializer&); auto power() -> void;
} flashrom; auto serialize(serializer&) -> void;
} flash;

12
gba/cartridge/mrom.cpp Normal file
View File

@ -0,0 +1,12 @@
auto Cartridge::MROM::read(unsigned mode, uint32 addr) -> uint32 {
if(mode & Word) addr &= ~3;
if(mode & Half) addr &= ~1;
auto p = data + (addr & 0x01ff'ffff);
if(mode & Word) return p[0] << 0 | p[1] << 8 | p[2] << 16 | p[3] << 24;
if(mode & Half) return p[0] << 0 | p[1] << 8;
if(mode & Byte) return p[0] << 0;
return 0; //should never occur
}
auto Cartridge::MROM::write(unsigned mode, uint32 addr, uint32 word) -> void {
}

View File

@ -1,5 +1,5 @@
void Cartridge::serialize(serializer& s) { void Cartridge::serialize(serializer& s) {
if(has_sram) s.array(ram.data, ram.size); if(has_sram) sram.serialize(s);
if(has_eeprom) eeprom.serialize(s); if(has_eeprom) eeprom.serialize(s);
if(has_flashrom) flashrom.serialize(s); if(has_flash) flash.serialize(s);
} }

17
gba/cartridge/sram.cpp Normal file
View File

@ -0,0 +1,17 @@
auto Cartridge::SRAM::read(unsigned mode, uint32 addr) -> uint32 {
if(mode & Word) addr &= ~3;
if(mode & Half) addr &= ~1;
uint32 word = data[addr & mask];
word = ror(word, 8 * (addr & 3));
if(mode & Word) { word &= 0xff; word |= word << 8; word |= word << 16; }
if(mode & Half) { word &= 0xff; word |= word << 8; }
return word;
}
auto Cartridge::SRAM::write(unsigned mode, uint32 addr, uint32 word) -> void {
data[addr & mask] = ror(word, 8 * (addr & 3));
}
auto Cartridge::SRAM::serialize(serializer& s) -> void {
s.array(data, size);
}

View File

@ -16,7 +16,7 @@ auto CPU::bus_read(unsigned mode, uint32 addr) -> uint32 {
} else { } else {
if(!active.dma) prefetch_wait(); if(!active.dma) prefetch_wait();
step(wait - 1); step(wait - 1);
word = addr < 0x0e00'0000 ? cartridge.rom_read(mode, addr) : cartridge.ram_read(mode, addr); word = cartridge.read(mode, addr);
step(1); step(1);
} }
} else { } else {
@ -43,7 +43,7 @@ auto CPU::bus_write(unsigned mode, uint32 addr, uint32 word) -> void {
} else if(addr & 0x0800'0000) { } else if(addr & 0x0800'0000) {
if(!active.dma) prefetch_wait(); if(!active.dma) prefetch_wait();
step(wait); step(wait);
addr < 0x0e00'0000 ? cartridge.rom_write(mode, addr, word) : cartridge.ram_write(mode, addr, word); cartridge.write(mode, addr, word);
} else { } else {
prefetch_step(wait); prefetch_step(wait);
if(addr < 0x0200'0000); if(addr < 0x0200'0000);

View File

@ -96,6 +96,7 @@ auto CPU::power() -> void {
dma.source = 0; dma.source = 0;
dma.target = 0; dma.target = 0;
dma.length = 0; dma.length = 0;
dma.data = 0;
dma.control = 0; dma.control = 0;
dma.pending = 0; dma.pending = 0;
dma.run.target = 0; dma.run.target = 0;

View File

@ -33,8 +33,17 @@ auto CPU::dma_exec(Registers::DMA& dma) -> void {
} }
} }
uint32 word = bus_read(mode, dma.run.source); if(dma.run.source < 0x0200'0000) {
bus_write(mode, dma.run.target, word); idle(); //cannot access BIOS
} else {
dma.data = bus_read(mode, dma.run.source);
}
if(dma.run.target < 0x0200'0000) {
idle(); //cannot access BIOS
} else {
bus_write(mode, dma.run.target, dma.data);
}
switch(dma.control.sourcemode) { switch(dma.control.sourcemode) {
case 0: dma.run.source += seek; break; case 0: dma.run.source += seek; break;

View File

@ -12,7 +12,7 @@ auto CPU::prefetch_step(unsigned clocks) -> void {
while(!prefetch.full() && clocks--) { while(!prefetch.full() && clocks--) {
if(--prefetch.wait) continue; if(--prefetch.wait) continue;
prefetch.slot[prefetch.load >> 1 & 7] = cartridge.rom_read(Half, prefetch.load); prefetch.slot[prefetch.load >> 1 & 7] = cartridge.read(Half, prefetch.load);
prefetch.load += 2; prefetch.load += 2;
prefetch.wait = bus_wait(Half | Sequential, prefetch.load); prefetch.wait = bus_wait(Half | Sequential, prefetch.load);
} }

View File

@ -18,6 +18,7 @@ struct Registers {
varuint source; varuint source;
varuint target; varuint target;
varuint length; varuint length;
uint32 data;
DMAControl control; DMAControl control;
//internal //internal

View File

@ -9,6 +9,7 @@ auto CPU::serialize(serializer& s) -> void {
s.integer(dma.source); s.integer(dma.source);
s.integer(dma.target); s.integer(dma.target);
s.integer(dma.length); s.integer(dma.length);
s.integer(dma.data);
s.integer(dma.control.targetmode); s.integer(dma.control.targetmode);
s.integer(dma.control.sourcemode); s.integer(dma.control.sourcemode);
s.integer(dma.control.repeat); s.integer(dma.control.repeat);

View File

@ -26,10 +26,10 @@ unsigned Interface::group(unsigned id) {
case ID::BIOS: case ID::BIOS:
return ID::System; return ID::System;
case ID::Manifest: case ID::Manifest:
case ID::ROM: case ID::MROM:
case ID::RAM: case ID::SRAM:
case ID::EEPROM: case ID::EEPROM:
case ID::FlashROM: case ID::FLASH:
return ID::GameBoyAdvance; return ID::GameBoyAdvance;
} }
@ -59,34 +59,34 @@ void Interface::load(unsigned id, const stream& stream) {
cartridge.information.markup = stream.text(); cartridge.information.markup = stream.text();
} }
if(id == ID::ROM) { if(id == ID::MROM) {
stream.read(cartridge.rom.data, min(cartridge.rom.size, stream.size())); stream.read(cartridge.mrom.data, min(cartridge.mrom.size, stream.size()));
} }
if(id == ID::RAM) { if(id == ID::SRAM) {
stream.read(cartridge.ram.data, min(cartridge.ram.size, stream.size())); stream.read(cartridge.sram.data, min(cartridge.sram.size, stream.size()));
} }
if(id == ID::EEPROM) { if(id == ID::EEPROM) {
stream.read(cartridge.eeprom.data, min(cartridge.eeprom.size, stream.size())); stream.read(cartridge.eeprom.data, min(cartridge.eeprom.size, stream.size()));
} }
if(id == ID::FlashROM) { if(id == ID::FLASH) {
stream.read(cartridge.flashrom.data, min(cartridge.flashrom.size, stream.size())); stream.read(cartridge.flash.data, min(cartridge.flash.size, stream.size()));
} }
} }
void Interface::save(unsigned id, const stream& stream) { void Interface::save(unsigned id, const stream& stream) {
if(id == ID::RAM) { if(id == ID::SRAM) {
stream.write(cartridge.ram.data, cartridge.ram.size); stream.write(cartridge.sram.data, cartridge.sram.size);
} }
if(id == ID::EEPROM) { if(id == ID::EEPROM) {
stream.write(cartridge.eeprom.data, cartridge.eeprom.size); stream.write(cartridge.eeprom.data, cartridge.eeprom.size);
} }
if(id == ID::FlashROM) { if(id == ID::FLASH) {
stream.write(cartridge.flashrom.data, cartridge.flashrom.size); stream.write(cartridge.flash.data, cartridge.flash.size);
} }
} }

View File

@ -13,10 +13,10 @@ struct ID {
BIOS, BIOS,
Manifest, Manifest,
ROM, MROM,
RAM, SRAM,
EEPROM, EEPROM,
FlashROM, FLASH,
}; };
enum : unsigned { enum : unsigned {

View File

@ -11,12 +11,21 @@ auto pLayout::destruct() -> void {
} }
auto pLayout::setEnabled(bool enabled) -> void { auto pLayout::setEnabled(bool enabled) -> void {
for(auto& sizable : state().sizables) {
if(auto self = sizable->self()) self->setEnabled(sizable->enabled(true));
}
} }
auto pLayout::setFont(const Font& font) -> void { auto pLayout::setFont(const Font& font) -> void {
for(auto& sizable : state().sizables) {
if(auto self = sizable->self()) self->setFont(sizable->font(true));
}
} }
auto pLayout::setVisible(bool visible) -> void { auto pLayout::setVisible(bool visible) -> void {
for(auto& sizable : state().sizables) {
if(auto self = sizable->self()) self->setVisible(sizable->visible(true));
}
} }
} }

View File

@ -242,15 +242,15 @@ auto pWindow::setDroppable(bool droppable) -> void {
auto pWindow::setEnabled(bool enabled) -> void { auto pWindow::setEnabled(bool enabled) -> void {
if(auto& menuBar = state().menuBar) { if(auto& menuBar = state().menuBar) {
if(menuBar->self()) menuBar->self()->setEnabled(menuBar->enabled(true)); if(auto self = menuBar->self()) self->setEnabled(menuBar->enabled(true));
} }
if(auto& statusBar = state().statusBar) { if(auto& statusBar = state().statusBar) {
if(statusBar->self()) statusBar->self()->setEnabled(statusBar->enabled(true)); if(auto self = statusBar->self()) self->setEnabled(statusBar->enabled(true));
} }
if(auto& layout = state().layout) { if(auto& layout = state().layout) {
if(layout->self()) layout->self()->setEnabled(layout->enabled(true)); if(auto self = layout->self()) self->setEnabled(layout->enabled(true));
} }
} }

View File

@ -38,13 +38,13 @@ GameBoyAdvanceCartridge::GameBoyAdvanceCartridge(const uint8_t *data, unsigned s
identifiers = list.merge(","); identifiers = list.merge(",");
markup.append("cartridge\n"); markup.append("cartridge\n");
markup.append(" rom name=program.rom size=0x", hex(size), "\n"); markup.append(" mrom name=program.rom size=0x", hex(size), "\n");
if(0); if(0);
else if(identifiers.beginsWith("SRAM_V" )) markup.append(" ram name=save.ram type=SRAM size=0x8000\n"); else if(identifiers.beginsWith("SRAM_V" )) markup.append(" sram name=save.ram size=0x8000\n");
else if(identifiers.beginsWith("SRAM_F_V" )) markup.append(" ram name=save.ram type=FRAM size=0x8000\n"); else if(identifiers.beginsWith("SRAM_F_V" )) markup.append(" sram name=save.ram size=0x8000\n");
else if(identifiers.beginsWith("EEPROM_V" )) markup.append(" ram name=save.ram type=EEPROM size=0x0\n"); else if(identifiers.beginsWith("EEPROM_V" )) markup.append(" eeprom name=save.ram size=0x0\n");
else if(identifiers.beginsWith("FLASH_V" )) markup.append(" ram name=save.ram type=FlashROM size=0x10000\n"); else if(identifiers.beginsWith("FLASH_V" )) markup.append(" flash name=save.ram size=0x10000\n");
else if(identifiers.beginsWith("FLASH512_V")) markup.append(" ram name=save.ram type=FlashROM size=0x10000\n"); else if(identifiers.beginsWith("FLASH512_V")) markup.append(" flash name=save.ram size=0x10000\n");
else if(identifiers.beginsWith("FLASH1M_V" )) markup.append(" ram name=save.ram type=FlashROM size=0x20000\n"); else if(identifiers.beginsWith("FLASH1M_V" )) markup.append(" flash name=save.ram size=0x20000\n");
//if(identifiers.empty() == false) markup.append(" #detected: ", identifiers, "\n"); //if(identifiers.empty() == false) markup.append(" #detected: ", identifiers, "\n");
} }

View File

@ -128,15 +128,18 @@ public:
auto size() const -> unsigned { return _size; } auto size() const -> unsigned { return _size; }
auto capacity() const -> unsigned { return _capacity; } auto capacity() const -> unsigned { return _capacity; }
auto operator==(const string& s) const -> bool { return size() == s.size() && memory::compare(data(), s.data(), s.size()) == 0; } auto operator==(const string& source) const -> bool { return size() == source.size() && memory::compare(data(), source.data(), size()) == 0; }
auto operator!=(const string& s) const -> bool { return size() != s.size() || memory::compare(data(), s.data(), s.size()) != 0; } auto operator!=(const string& source) const -> bool { return size() != source.size() || memory::compare(data(), source.data(), size()) != 0; }
auto operator==(const char* s) const -> bool { return strcmp(data(), s) == 0; } auto operator==(const char* source) const -> bool { return strcmp(data(), source) == 0; }
auto operator!=(const char* s) const -> bool { return strcmp(data(), s) != 0; } auto operator!=(const char* source) const -> bool { return strcmp(data(), source) != 0; }
auto operator< (const char* s) const -> bool { return strcmp(data(), s) < 0; }
auto operator<=(const char* s) const -> bool { return strcmp(data(), s) <= 0; } auto operator==(rstring source) const -> bool { return compare(source) == 0; }
auto operator> (const char* s) const -> bool { return strcmp(data(), s) > 0; } auto operator!=(rstring source) const -> bool { return compare(source) != 0; }
auto operator>=(const char* s) const -> bool { return strcmp(data(), s) >= 0; } auto operator< (rstring source) const -> bool { return compare(source) < 0; }
auto operator<=(rstring source) const -> bool { return compare(source) <= 0; }
auto operator> (rstring source) const -> bool { return compare(source) > 0; }
auto operator>=(rstring source) const -> bool { return compare(source) >= 0; }
string(const string& source) : string() { operator=(source); } string(const string& source) : string() { operator=(source); }
string(string&& source) : string() { operator=(move(source)); } string(string&& source) : string() { operator=(move(source)); }
@ -183,6 +186,9 @@ public:
//compare.hpp //compare.hpp
template<bool> inline static auto _compare(const char*, unsigned, const char*, unsigned) -> signed; template<bool> inline static auto _compare(const char*, unsigned, const char*, unsigned) -> signed;
inline static auto compare(rstring, rstring) -> signed;
inline static auto icompare(rstring, rstring) -> signed;
inline auto compare(rstring source) const -> signed; inline auto compare(rstring source) const -> signed;
inline auto icompare(rstring source) const -> signed; inline auto icompare(rstring source) const -> signed;

View File

@ -8,12 +8,21 @@ auto string::_compare(const char* target, unsigned capacity, const char* source,
return memory::compare(target, capacity, source, size); return memory::compare(target, capacity, source, size);
} }
//size() + 1 includes null-terminator; required to properly compare strings of differing lengths
auto string::compare(rstring x, rstring y) -> signed {
return memory::compare(x.data(), x.size() + 1, y.data(), y.size() + 1);
}
auto string::icompare(rstring x, rstring y) -> signed {
return memory::icompare(x.data(), x.size() + 1, y.data(), y.size() + 1);
}
auto string::compare(rstring source) const -> signed { auto string::compare(rstring source) const -> signed {
return memory::compare(data(), size(), source.data(), source.size()); return memory::compare(data(), size() + 1, source.data(), source.size() + 1);
} }
auto string::icompare(rstring source) const -> signed { auto string::icompare(rstring source) const -> signed {
return memory::icompare(data(), size(), source.data(), source.size()); return memory::icompare(data(), size() + 1, source.data(), source.size() + 1);
} }
auto string::equals(rstring source) const -> bool { auto string::equals(rstring source) const -> bool {

View File

@ -13,11 +13,23 @@ struct ManagedNode {
ManagedNode(const string& name, const string& value) : _name(name), _value(value) {} ManagedNode(const string& name, const string& value) : _name(name), _value(value) {}
auto clone() const -> SharedNode { auto clone() const -> SharedNode {
SharedNode clone(new ManagedNode(_name, _value)); SharedNode clone{new ManagedNode(_name, _value)};
for(auto& child : _children) clone->_children.append(child->clone()); for(auto& child : _children) {
clone->_children.append(child->clone());
}
return clone; return clone;
} }
auto copy(SharedNode source) -> void {
_name = source->_name;
_value = source->_value;
_metadata = source->_metadata;
_children.reset();
for(auto child : source->_children) {
_children.append(child->clone());
}
}
protected: protected:
string _name; string _name;
string _value; string _value;
@ -40,6 +52,7 @@ struct Node {
auto unique() const -> bool { return shared.unique(); } auto unique() const -> bool { return shared.unique(); }
auto clone() const -> Node { return shared->clone(); } auto clone() const -> Node { return shared->clone(); }
auto copy(Node source) -> void { return shared->copy(source.shared); }
explicit operator bool() const { return shared->_name || shared->_children; } explicit operator bool() const { return shared->_name || shared->_children; }
auto name() const -> string { return shared->_name; } auto name() const -> string { return shared->_name; }
@ -50,8 +63,8 @@ struct Node {
auto integer() const -> intmax_t { return text().integer(); } auto integer() const -> intmax_t { return text().integer(); }
auto decimal() const -> uintmax_t { return text().decimal(); } auto decimal() const -> uintmax_t { return text().decimal(); }
auto setName(const string& name = "") -> void { shared->_name = name; } auto setName(const string& name = "") -> Node& { shared->_name = name; return *this; }
auto setValue(const string& value = "") -> void { shared->_value = value; } auto setValue(const string& value = "") -> Node& { shared->_value = value; return *this; }
auto reset() -> void { shared->_children.reset(); } auto reset() -> void { shared->_children.reset(); }
auto size() const -> unsigned { return shared->_children.size(); } auto size() const -> unsigned { return shared->_children.size(); }
@ -67,16 +80,6 @@ struct Node {
return false; return false;
} }
auto at(unsigned position) const -> Node {
if(position >= size()) return {};
return shared->_children[position];
}
auto swap(unsigned x, unsigned y) -> bool {
if(x >= size() || y >= size()) return false;
return std::swap(shared->_children[x], shared->_children[y]), true;
}
auto insert(unsigned position, const Node& node) -> bool { auto insert(unsigned position, const Node& node) -> bool {
if(position > size()) return false; //used > instead of >= to allow indexed-equivalent of append() if(position > size()) return false; //used > instead of >= to allow indexed-equivalent of append()
return shared->_children.insert(position, node.shared), true; return shared->_children.insert(position, node.shared), true;
@ -87,8 +90,26 @@ struct Node {
return shared->_children.remove(position), true; return shared->_children.remove(position), true;
} }
auto operator()(const string& path) -> Node { return shared->_create(path); } auto swap(unsigned x, unsigned y) -> bool {
if(x >= size() || y >= size()) return false;
return std::swap(shared->_children[x], shared->_children[y]), true;
}
auto sort(function<bool (Node, Node)> comparator = [](auto x, auto y) {
return string::compare(x.shared->_name, y.shared->_name) < 0;
}) -> void {
nall::sort(shared->_children.data(), shared->_children.size(), [&](auto x, auto y) {
return comparator(x, y); //this call converts SharedNode objects to Node objects
});
}
auto operator[](signed position) -> Node {
if(position >= size()) return {};
return shared->_children[position];
}
auto operator[](const string& path) const -> Node { return shared->_lookup(path); } auto operator[](const string& path) const -> Node { return shared->_lookup(path); }
auto operator()(const string& path) -> Node { return shared->_create(path); }
auto find(const string& query) const -> vector<Node> { return shared->_find(query); } auto find(const string& query) const -> vector<Node> { return shared->_find(query); }
struct iterator { struct iterator {

View File

@ -160,12 +160,10 @@ template<typename T> struct vector {
} }
} }
auto sort() -> void { auto sort(const function<bool (const T& lhs, const T& rhs)>& comparator = [](const T& lhs, const T& rhs) -> bool {
nall::sort(pool + poolbase, objectsize); return lhs < rhs;
} }) -> void {
nall::sort(pool + poolbase, objectsize, comparator);
template<typename Comparator> auto sort(const Comparator& lessthan) -> void {
nall::sort(pool + poolbase, objectsize, lessthan);
} }
auto find(const T& data) const -> maybe<unsigned> { auto find(const T& data) const -> maybe<unsigned> {

View File

@ -42,14 +42,23 @@ auto ARM::load(unsigned mode, uint32 addr) -> uint32 {
pipeline.nonsequential = true; pipeline.nonsequential = true;
uint32 word = bus_read(Load | mode, addr); uint32 word = bus_read(Load | mode, addr);
if(mode & Half) { word &= 0xffff; word |= word << 16; } if(mode & Half) {
if(mode & Byte) { word &= 0xff; word |= word << 8; word |= word << 16; } addr &= 1;
word = mode & Signed ? (int16)word : (uint16)word;
}
if(mode & Byte) {
addr &= 0;
word = mode & Signed ? (int8)word : (uint8)word;
}
if(mode & Signed) {
word = asr(word, 8 * (addr & 3));
} else {
word = ror(word, 8 * (addr & 3)); word = ror(word, 8 * (addr & 3));
idle(); }
if(mode & Half) word &= 0xffff; idle();
if(mode & Byte) word &= 0xff;
return word; return word;
} }

View File

@ -17,6 +17,7 @@ struct ARM {
Word = 32, //32-bit access Word = 32, //32-bit access
Load = 64, //load operation Load = 64, //load operation
Store = 128, //store operation Store = 128, //store operation
Signed = 256, //sign extend
}; };
#include "registers.hpp" #include "registers.hpp"

View File

@ -164,13 +164,15 @@ auto ARM::arm_op_move_half_register() {
uint32 rn = r(n); uint32 rn = r(n);
uint32 rm = r(m); uint32 rm = r(m);
uint32 rd = r(d);
if(pre == 1) rn = up ? rn + rm : rn - rm; if(pre == 1) rn = up ? rn + rm : rn - rm;
if(l == 1) r(d) = load(Half | Nonsequential, rn); if(l == 1) rd = load(Half | Nonsequential, rn);
if(l == 0) store(Half | Nonsequential, rn, r(d)); if(l == 0) store(Half | Nonsequential, rn, rd);
if(pre == 0) rn = up ? rn + rm : rn - rm; if(pre == 0) rn = up ? rn + rm : rn - rm;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
if(l == 1) r(d) = rd;
} }
//(ldr,str){condition}h rd,[rn{,+/-offset}]{!} //(ldr,str){condition}h rd,[rn{,+/-offset}]{!}
@ -196,14 +198,16 @@ auto ARM::arm_op_move_half_immediate() {
uint4 il = instruction(); uint4 il = instruction();
uint32 rn = r(n); uint32 rn = r(n);
uint32 rd = r(d);
uint8 immediate = (ih << 4) + (il << 0); uint8 immediate = (ih << 4) + (il << 0);
if(pre == 1) rn = up ? rn + immediate : rn - immediate; if(pre == 1) rn = up ? rn + immediate : rn - immediate;
if(l == 1) r(d) = load(Half | Nonsequential, rn); if(l == 1) rd = load(Half | Nonsequential, rn);
if(l == 0) store(Half | Nonsequential, rn, r(d)); if(l == 0) store(Half | Nonsequential, rn, rd);
if(pre == 0) rn = up ? rn + immediate : rn - immediate; if(pre == 0) rn = up ? rn + immediate : rn - immediate;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
if(l == 1) r(d) = rd;
} }
//ldr{condition}s(h,b) rd,[rn,rm]{!} //ldr{condition}s(h,b) rd,[rn,rm]{!}
@ -228,13 +232,14 @@ auto ARM::arm_op_load_register() {
uint32 rn = r(n); uint32 rn = r(n);
uint32 rm = r(m); uint32 rm = r(m);
uint32 rd = r(d);
if(pre == 1) rn = up ? rn + rm : rn - rm; if(pre == 1) rn = up ? rn + rm : rn - rm;
uint32 word = load((half ? Half : Byte) | Nonsequential, rn); rd = load((half ? Half : Byte) | Nonsequential | Signed, rn);
r(d) = half ? (int16)word : (int8)word;
if(pre == 0) rn = up ? rn + rm : rn - rm; if(pre == 0) rn = up ? rn + rm : rn - rm;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
r(d) = rd;
} }
//ldr{condition}s(h,b) rd,[rn{,+/-offset}]{!} //ldr{condition}s(h,b) rd,[rn{,+/-offset}]{!}
@ -260,14 +265,15 @@ auto ARM::arm_op_load_immediate() {
uint4 il = instruction(); uint4 il = instruction();
uint32 rn = r(n); uint32 rn = r(n);
uint32 rd = r(d);
uint8 immediate = (ih << 4) + (il << 0); uint8 immediate = (ih << 4) + (il << 0);
if(pre == 1) rn = up ? rn + immediate : rn - immediate; if(pre == 1) rn = up ? rn + immediate : rn - immediate;
uint32 word = load((half ? Half : Byte) | Nonsequential, rn); rd = load((half ? Half : Byte) | Nonsequential | Signed, rn);
r(d) = half ? (int16)word : (int8)word;
if(pre == 0) rn = up ? rn + immediate : rn - immediate; if(pre == 0) rn = up ? rn + immediate : rn - immediate;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
r(d) = rd;
} }
//mrs{condition} rd,(c,s)psr //mrs{condition} rd,(c,s)psr
@ -375,8 +381,8 @@ auto ARM::arm_op_data_register_shift() {
uint2 mode = instruction() >> 5; uint2 mode = instruction() >> 5;
uint4 m = instruction(); uint4 m = instruction();
uint8 rs = r(s); uint8 rs = r(s) + (s == 15 ? 4 : 0);
uint32 rm = r(m); uint32 rm = r(m) + (m == 15 ? 4 : 0);
carryout() = cpsr().c; carryout() = cpsr().c;
if(mode == 0 ) rm = lsl(rm, rs < 33 ? rs : 33); if(mode == 0 ) rm = lsl(rm, rs < 33 ? rs : 33);
@ -434,7 +440,7 @@ auto ARM::arm_op_move_immediate_offset() {
uint12 rm = instruction(); uint12 rm = instruction();
uint32 rn = r(n); uint32 rn = r(n);
auto& rd = r(d); uint32 rd = r(d);
if(pre == 1) rn = up ? rn + rm : rn - rm; if(pre == 1) rn = up ? rn + rm : rn - rm;
if(l == 1) rd = load((byte ? Byte : Word) | Nonsequential, rn); if(l == 1) rd = load((byte ? Byte : Word) | Nonsequential, rn);
@ -442,6 +448,7 @@ auto ARM::arm_op_move_immediate_offset() {
if(pre == 0) rn = up ? rn + rm : rn - rm; if(pre == 0) rn = up ? rn + rm : rn - rm;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
if(l == 1) r(d) = rd;
} }
//(ldr,str){condition}{b} rd,[rn,rm {mode} #immediate]{!} //(ldr,str){condition}{b} rd,[rn,rm {mode} #immediate]{!}
@ -471,7 +478,7 @@ auto ARM::arm_op_move_register_offset() {
uint4 m = instruction(); uint4 m = instruction();
uint32 rn = r(n); uint32 rn = r(n);
auto& rd = r(d); uint32 rd = r(d);
uint32 rs = immediate; uint32 rs = immediate;
uint32 rm = r(m); uint32 rm = r(m);
bool c = cpsr().c; bool c = cpsr().c;
@ -487,6 +494,7 @@ auto ARM::arm_op_move_register_offset() {
if(pre == 0) rn = up ? rn + rm : rn - rm; if(pre == 0) rn = up ? rn + rm : rn - rm;
if(pre == 0 || writeback == 1) r(n) = rn; if(pre == 0 || writeback == 1) r(n) = rn;
if(l == 1) r(d) = rd;
} }
//(ldm,stm){condition}{mode} rn{!},{r...} //(ldm,stm){condition}{mode} rn{!},{r...}

View File

@ -164,11 +164,11 @@ auto ARM::thumb_op_move_register_offset() {
case 0: store(Word | Nonsequential, r(n) + r(m), r(d)); break; //STR case 0: store(Word | Nonsequential, r(n) + r(m), r(d)); break; //STR
case 1: store(Half | Nonsequential, r(n) + r(m), r(d)); break; //STRH case 1: store(Half | Nonsequential, r(n) + r(m), r(d)); break; //STRH
case 2: store(Byte | Nonsequential, r(n) + r(m), r(d)); break; //STRB case 2: store(Byte | Nonsequential, r(n) + r(m), r(d)); break; //STRB
case 3: r(d) = (int8)load(Byte | Nonsequential, r(n) + r(m)); break; //LDSB case 3: r(d) = load(Byte | Nonsequential | Signed, r(n) + r(m)); break; //LDSB
case 4: r(d) = load(Word | Nonsequential, r(n) + r(m)); break; //LDR case 4: r(d) = load(Word | Nonsequential, r(n) + r(m)); break; //LDR
case 5: r(d) = load(Half | Nonsequential, r(n) + r(m)); break; //LDRH case 5: r(d) = load(Half | Nonsequential, r(n) + r(m)); break; //LDRH
case 6: r(d) = load(Byte | Nonsequential, r(n) + r(m)); break; //LDRB case 6: r(d) = load(Byte | Nonsequential, r(n) + r(m)); break; //LDRB
case 7: r(d) = (int16)load(Half | Nonsequential, r(n) + r(m)); break; //LDSH case 7: r(d) = load(Half | Nonsequential | Signed, r(n) + r(m)); break; //LDSH
} }
} }

View File

@ -43,6 +43,10 @@ using namespace ruby;
#include <ruby/video/glx.cpp> #include <ruby/video/glx.cpp>
#endif #endif
#if defined(VIDEO_GLX2)
#include <ruby/video/glx2.cpp>
#endif
#if defined(VIDEO_SDL) #if defined(VIDEO_SDL)
#include <ruby/video/sdl.cpp> #include <ruby/video/sdl.cpp>
#endif #endif
@ -93,6 +97,10 @@ auto Video::create(const string& driver) -> Video* {
if(driver == "OpenGL") return new VideoGLX; if(driver == "OpenGL") return new VideoGLX;
#endif #endif
#if defined(VIDEO_GLX2)
if(driver == "OpenGL2") return new VideoGLX2;
#endif
#if defined(VIDEO_SDL) #if defined(VIDEO_SDL)
if(driver == "SDL") return new VideoSDL; if(driver == "SDL") return new VideoSDL;
#endif #endif
@ -125,6 +133,8 @@ auto Video::optimalDriver() -> string {
return "OpenGL"; return "OpenGL";
#elif defined(VIDEO_GLX) #elif defined(VIDEO_GLX)
return "OpenGL"; return "OpenGL";
#elif defined(VIDEO_GLX2)
return "OpenGL2";
#elif defined(VIDEO_XV) #elif defined(VIDEO_XV)
return "X-Video"; return "X-Video";
#elif defined(VIDEO_XSHM) #elif defined(VIDEO_XSHM)
@ -153,6 +163,8 @@ auto Video::safestDriver() -> string {
return "SDL"; return "SDL";
#elif defined(VIDEO_XV) #elif defined(VIDEO_XV)
return "X-Video"; return "X-Video";
#elif defined(VIDEO_GLX2)
return "OpenGL2";
#elif defined(VIDEO_GLX) #elif defined(VIDEO_GLX)
return "OpenGL"; return "OpenGL";
#else #else
@ -187,6 +199,10 @@ auto Video::availableDrivers() -> lstring {
"OpenGL", "OpenGL",
#endif #endif
#if defined(VIDEO_GLX2)
"OpenGL2",
#endif
#if defined(VIDEO_XV) #if defined(VIDEO_XV)
"X-Video", "X-Video",
#endif #endif

245
ruby/video/glx2.cpp Normal file
View File

@ -0,0 +1,245 @@
//Xorg/GLX OpenGL 2.0 driver
//note: this is a fallback driver for use when OpenGL 3.2 is not available.
//see glx.cpp for comments on how this driver operates (they are very similar.)
struct VideoGLX2 : Video {
~VideoGLX2() { term(); }
auto (*glXSwapInterval)(signed) -> signed = nullptr;
Display* display = nullptr;
signed screen = 0;
Window xwindow = 0;
Colormap colormap = 0;
GLXContext glxcontext = nullptr;
GLXWindow glxwindow = 0;
struct {
Window handle = 0;
bool synchronize = false;
unsigned filter = 1; //linear
unsigned width = 256;
unsigned height = 256;
bool isDoubleBuffered = false;
bool isDirect = false;
} settings;
auto cap(const string& name) -> bool {
if(name == Video::Handle) return true;
if(name == Video::Synchronize) return true;
if(name == Video::Filter) return true;
return false;
}
auto get(const string& name) -> any {
if(name == Video::Handle) return (uintptr_t)settings.handle;
if(name == Video::Synchronize) return settings.synchronize;
if(name == Video::Filter) return settings.filter;
return {};
}
auto set(const string& name, const any& value) -> bool {
if(name == Video::Handle && value.is<uintptr_t>()) {
settings.handle = value.get<uintptr_t>();
return true;
}
if(name == Video::Synchronize && value.is<bool>()) {
if(settings.synchronize != value.get<bool>()) {
settings.synchronize = value.get<bool>();
if(glXSwapInterval) glXSwapInterval(settings.synchronize);
return true;
}
}
if(name == Video::Filter && value.is<unsigned>()) {
settings.filter = value.get<unsigned>();
return true;
}
return false;
}
auto lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool {
if(width != settings.width || height != settings.height) resize(width, height);
pitch = glwidth * sizeof(uint32_t);
return data = glbuffer;
}
auto unlock() -> void {
}
auto clear() -> void {
memory::fill(glbuffer, glwidth * glheight * sizeof(uint32_t));
glClearColor(0.0, 0.0, 0.0, 1.0);
glClear(GL_COLOR_BUFFER_BIT);
glFlush();
if(settings.isDoubleBuffered) glXSwapBuffers(display, glxwindow);
}
auto refresh() -> void {
XWindowAttributes parent, child;
XGetWindowAttributes(display, settings.handle, &parent);
XGetWindowAttributes(display, xwindow, &child);
if(child.width != parent.width || child.height != parent.height) {
XResizeWindow(display, xwindow, parent.width, parent.height);
}
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_BORDER);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_BORDER);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, settings.filter ? GL_LINEAR : GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, settings.filter ? GL_LINEAR : GL_NEAREST);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
glOrtho(0, parent.width, 0, parent.height, -1.0, 1.0);
glViewport(0, 0, parent.width, parent.height);
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
glPixelStorei(GL_UNPACK_ROW_LENGTH, glwidth);
glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, settings.width, settings.height,
GL_BGRA, GL_UNSIGNED_INT_8_8_8_8_REV, glbuffer);
double w = (double)settings.width / (double)glwidth;
double h = (double)settings.height / (double)glheight;
signed u = parent.width;
signed v = parent.height;
glBegin(GL_TRIANGLE_STRIP);
glTexCoord2f(0, 0); glVertex3i(0, v, 0);
glTexCoord2f(w, 0); glVertex3i(u, v, 0);
glTexCoord2f(0, h); glVertex3i(0, 0, 0);
glTexCoord2f(w, h); glVertex3i(u, 0, 0);
glEnd();
glFlush();
if(settings.isDoubleBuffered) glXSwapBuffers(display, glxwindow);
}
auto init() -> bool {
display = XOpenDisplay(0);
screen = DefaultScreen(display);
signed versionMajor = 0, versionMinor = 0;
glXQueryVersion(display, &versionMajor, &versionMinor);
if(versionMajor < 1 || (versionMajor == 1 && versionMinor < 2)) return false;
XWindowAttributes windowAttributes;
XGetWindowAttributes(display, settings.handle, &windowAttributes);
signed attributeList[] = {
GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT,
GLX_RENDER_TYPE, GLX_RGBA_BIT,
GLX_DOUBLEBUFFER, True,
GLX_RED_SIZE, 8,
GLX_GREEN_SIZE, 8,
GLX_BLUE_SIZE, 8,
None
};
signed fbCount = 0;
auto fbConfig = glXChooseFBConfig(display, screen, attributeList, &fbCount);
if(fbCount == 0) return false;
auto vi = glXGetVisualFromFBConfig(display, fbConfig[0]);
colormap = XCreateColormap(display, RootWindow(display, vi->screen), vi->visual, AllocNone);
XSetWindowAttributes attributes;
attributes.colormap = colormap;
attributes.border_pixel = 0;
xwindow = XCreateWindow(display, settings.handle, 0, 0, windowAttributes.width, windowAttributes.height,
0, vi->depth, InputOutput, vi->visual, CWColormap | CWBorderPixel, &attributes);
XSetWindowBackground(display, xwindow, 0);
XMapWindow(display, xwindow);
XFlush(display);
while(XPending(display)) {
XEvent event;
XNextEvent(display, &event);
}
glxcontext = glXCreateContext(display, vi, 0, GL_TRUE);
glXMakeCurrent(display, glxwindow = xwindow, glxcontext);
glXSwapInterval = (signed (*)(signed))glGetProcAddress("glXSwapIntervalEXT");
if(!glXSwapInterval) glXSwapInterval = (signed (*)(signed))glGetProcAddress("glXSwapIntervalMESA");
if(!glXSwapInterval) glXSwapInterval = (signed (*)(signed))glGetProcAddress("glXSwapIntervalSGI");
if(glXSwapInterval) glXSwapInterval(settings.synchronize);
signed value = 0;
glXGetConfig(display, vi, GLX_DOUBLEBUFFER, &value);
settings.isDoubleBuffered = value;
settings.isDirect = glXIsDirect(display, glxcontext);
glDisable(GL_ALPHA_TEST);
glDisable(GL_BLEND);
glDisable(GL_DEPTH_TEST);
glDisable(GL_POLYGON_SMOOTH);
glDisable(GL_STENCIL_TEST);
glEnable(GL_DITHER);
glEnable(GL_TEXTURE_2D);
resize(256, 256);
return true;
}
auto term() -> void {
if(gltexture) {
glDeleteTextures(1, &gltexture);
gltexture = 0;
}
if(glbuffer) {
delete[] glbuffer;
glbuffer = 0;
}
glwidth = 0;
glheight = 0;
if(glxcontext) {
glXDestroyContext(display, glxcontext);
glxcontext = nullptr;
}
if(xwindow) {
XUnmapWindow(display, xwindow);
xwindow = 0;
}
if(colormap) {
XFreeColormap(display, colormap);
colormap = 0;
}
if(display) {
XCloseDisplay(display);
display = nullptr;
}
}
private:
GLuint gltexture = 0;
uint32_t* glbuffer = nullptr;
unsigned glwidth = 0;
unsigned glheight = 0;
auto resize(unsigned width, unsigned height) -> void {
settings.width = width;
settings.height = height;
if(gltexture == 0) glGenTextures(1, &gltexture);
glwidth = max(glwidth, width);
glheight = max(glheight, height);
if(glbuffer) delete[] glbuffer;
glbuffer = new uint32_t[glwidth * glheight]();
glBindTexture(GL_TEXTURE_2D, gltexture);
glPixelStorei(GL_UNPACK_ROW_LENGTH, glwidth);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, glwidth, glheight, 0, GL_BGRA, GL_UNSIGNED_INT_8_8_8_8_REV, glbuffer);
}
};

View File

@ -1,13 +1,5 @@
template<typename T, unsigned size> template<typename T, unsigned size>
struct ModuloArray { struct ModuloArray {
ModuloArray() {
buffer = new T[size * 3]();
}
~ModuloArray() {
delete[] buffer;
}
inline auto operator[](signed index) const -> T { inline auto operator[](signed index) const -> T {
return buffer[size + index]; return buffer[size + index];
} }
@ -27,5 +19,5 @@ struct ModuloArray {
} }
private: private:
T* buffer; T buffer[size * 3] = {0};
}; };

View File

@ -2,6 +2,7 @@
#error "bsnes: debugger not supported with performance profile." #error "bsnes: debugger not supported with performance profile."
#endif #endif
#include <nall/priority-queue.hpp>
#include <sfc/alt/cpu/cpu.hpp> #include <sfc/alt/cpu/cpu.hpp>
#include <sfc/alt/smp/smp.hpp> #include <sfc/alt/smp/smp.hpp>
#include <sfc/alt/dsp/dsp.hpp> #include <sfc/alt/dsp/dsp.hpp>