Update to v082r26 release.

byuu says:

.cht files now use BML-formatted data. I'm still going to request the
cheats.xml file as-is, and will write my own converter for embedding
during releases.
This is where parsing 2MB markup files in 10ms is really going to be
nice. Had to use an evil hack before for actually searching for games.

This has the start of the board/chip separation from mappers for NES,
and it has a barebones iNES->board markup converter.
You can specify your own board markup and bypass the need for an iNES
header, so in other words it will load No-Intro style games with
a proper board file.
Long-term, we'll have an internal database for commercial boards, and
probably folder.fc/prg.rom{,chr.rom} loading support.

Since they can't co-exist, the mappers are currently disabled, and I've
only ported the easy ones. So no MMC1/MMC3/VRC6 in this release. I need
to make them into chips first.
This commit is contained in:
Tim Allen 2011-09-29 22:44:49 +10:00
parent e8b1af0917
commit 7115047d85
23 changed files with 597 additions and 285 deletions

View File

@ -34,7 +34,7 @@ protected:
unsigned sourceSize;
public:
inline lzss() : sourceData(nullptr), sourceSize(nullptr) {}
inline lzss() : sourceData(nullptr), sourceSize(0) {}
};
void lzss::source(const uint8_t *data, unsigned size) {

View File

@ -20,7 +20,9 @@
#include <nall/windows/utf8.hpp>
#define NALL_STRING_INTERNAL_HPP
#include <nall/string/base.hpp>
#include <nall/string/bml.hpp>
#include <nall/string/bsv.hpp>
#include <nall/string/core.hpp>
#include <nall/string/cast.hpp>
@ -38,5 +40,6 @@
#include <nall/string/variadic.hpp>
#include <nall/string/wrapper.hpp>
#include <nall/string/xml.hpp>
#undef NALL_STRING_INTERNAL_HPP
#endif

127
bsnes/nall/string/bml.hpp Executable file
View File

@ -0,0 +1,127 @@
#ifdef NALL_STRING_INTERNAL_HPP
//BML parser
//version 0.02
namespace nall {
namespace BML {
struct Node : linear_vector<Node> {
Node *parent;
const char *cname;
const char *cvalue;
inline string name() { return cname; }
inline string value() { return cvalue; }
private:
inline bool valid(char p) const {
if(p >= 'A' && p <= 'Z') return true;
if(p >= 'a' && p <= 'z') return true;
if(p >= '0' && p <= '9') return true;
if(p == '+' || p == '-') return true;
if(p == '.' || p == '_') return true;
if(p == ':') return true;
return false;
}
inline bool space(char p) const {
return p == ' ' || p == '\t';
}
inline unsigned parseDepth(char *p) {
unsigned depth = 0;
while(space(*p)) depth++, p++;
return depth;
}
inline void parseNode(char *&p) {
if(valid(*p)) {
cname = p;
while(valid(*p)) p++;
if(*p != '=') return;
*p++ = 0;
}
if(*p == '\n') throw "Missing node value";
if(valid(*p)) {
cvalue = p;
while(valid(*p)) p++;
} else {
char terminal = *p++;
cvalue = p;
while(*p && *p != terminal) p++;
if(*p == 0) throw "Unclosed terminal";
*p++ = 0;
}
}
inline void parseLine(char *&p) {
unsigned depth = parseDepth(p);
while(space(*p)) p++;
parseNode(p);
while(space(*p)) {
*p++ = 0;
Node node(this);
node.parseNode(p);
append(node);
}
if(*p == '\n') *p++ = 0;
while(parseDepth(p) > depth) {
Node node(this);
node.parseLine(p);
append(node);
}
}
inline void parse(char *&p, unsigned parentDepth = 0) {
while(*p) {
while(*p == '\n') *p++ = 0;
Node node(this);
node.parseLine(p);
append(node);
}
}
public:
inline Node& operator[](const string &name) {
for(auto &node : *this) {
if(!strcmp(node.cname, name)) return node;
}
static Node node;
return node;
}
inline bool exists() {
return parent != nullptr;
}
inline string content(const string &separator = "\n") const {
string result;
for(auto &node : *this) result.append(node.cvalue, separator);
result.rtrim<1>(separator);
return result;
}
inline Node(const string &document) : parent(nullptr), cname(nullptr), cvalue(nullptr) {
char *p = strdup(document);
cvalue = p;
try {
parse(p);
} catch(const char *error) {
reset();
}
}
inline Node(Node *parent) : parent(parent), cname(""), cvalue("") {}
inline Node() : parent(nullptr), cname(""), cvalue("") {}
inline ~Node() { if(cname == nullptr && cvalue) free((void*)cvalue); }
};
}
}
#endif

View File

@ -1,74 +1,75 @@
#ifndef NALL_STRING_BSV_HPP
#define NALL_STRING_BSV_HPP
#ifdef NALL_STRING_INTERNAL_HPP
//BSV parser
//version 0.01
//version 0.02
namespace nall {
inline string bsv_decode(const char *input) {
string output;
unsigned offset = 0;
while(*input) {
//illegal characters
if(*input == '}' ) return "";
if(*input == '\r') return "";
if(*input == '\n') return "";
struct BSV {
static inline string decode(const char *input) {
string output;
unsigned offset = 0;
while(*input) {
//illegal characters
if(*input == '}' ) return "";
if(*input == '\r') return "";
if(*input == '\n') return "";
//normal characters
if(*input != '{') { output[offset++] = *input++; continue; }
//normal characters
if(*input != '{') { output[offset++] = *input++; continue; }
//entities
if(strbegin(input, "{lf}")) { output[offset++] = '\n'; input += 4; continue; }
if(strbegin(input, "{lb}")) { output[offset++] = '{'; input += 4; continue; }
if(strbegin(input, "{rb}")) { output[offset++] = '}'; input += 4; continue; }
//entities
if(strbegin(input, "{lf}")) { output[offset++] = '\n'; input += 4; continue; }
if(strbegin(input, "{lb}")) { output[offset++] = '{'; input += 4; continue; }
if(strbegin(input, "{rb}")) { output[offset++] = '}'; input += 4; continue; }
//illegal entities
return "";
//illegal entities
return "";
}
output[offset] = 0;
return output;
}
output[offset] = 0;
return output;
}
inline string bsv_encode(const char *input) {
string output;
unsigned offset = 0;
while(*input) {
//illegal characters
if(*input == '\r') return "";
static inline string encode(const char *input) {
string output;
unsigned offset = 0;
while(*input) {
//illegal characters
if(*input == '\r') return "";
if(*input == '\n') {
output[offset++] = '{';
output[offset++] = 'l';
output[offset++] = 'f';
output[offset++] = '}';
input++;
continue;
if(*input == '\n') {
output[offset++] = '{';
output[offset++] = 'l';
output[offset++] = 'f';
output[offset++] = '}';
input++;
continue;
}
if(*input == '{') {
output[offset++] = '{';
output[offset++] = 'l';
output[offset++] = 'b';
output[offset++] = '}';
input++;
continue;
}
if(*input == '}') {
output[offset++] = '{';
output[offset++] = 'r';
output[offset++] = 'b';
output[offset++] = '}';
input++;
continue;
}
output[offset++] = *input++;
}
if(*input == '{') {
output[offset++] = '{';
output[offset++] = 'l';
output[offset++] = 'b';
output[offset++] = '}';
input++;
continue;
}
if(*input == '}') {
output[offset++] = '{';
output[offset++] = 'r';
output[offset++] = 'b';
output[offset++] = '}';
input++;
continue;
}
output[offset++] = *input++;
output[offset] = 0;
return output;
}
output[offset] = 0;
return output;
}
};
}

View File

@ -1,6 +1,5 @@
nes_objects := nes-interface nes-system nes-scheduler nes-input
nes_objects += nes-mapper nes-cartridge nes-memory
nes_objects += nes-cpu nes-apu nes-ppu
nes_objects += nes-memory nes-cartridge nes-cpu nes-apu nes-ppu
nes_objects += nes-cheat
objects += $(nes_objects)
@ -8,9 +7,8 @@ obj/nes-interface.o: $(nes)/interface/interface.cpp $(call rwildcard,$(nes)/inte
obj/nes-system.o: $(nes)/system/system.cpp $(call rwildcard,$(nes)/system/)
obj/nes-scheduler.o: $(nes)/scheduler/scheduler.cpp $(call rwildcard,$(nes)/scheduler/)
obj/nes-input.o: $(nes)/input/input.cpp $(call rwildcard,$(nes)/input/)
obj/nes-mapper.o: $(nes)/mapper/mapper.cpp $(call rwildcard,$(nes)/mapper/)
obj/nes-cartridge.o: $(nes)/cartridge/cartridge.cpp $(call rwildcard,$(nes)/cartridge/)
obj/nes-memory.o: $(nes)/memory/memory.cpp $(call rwildcard,$(nes)/memory/)
obj/nes-cartridge.o: $(nes)/cartridge/cartridge.cpp $(call rwildcard,$(nes)/cartridge/)
obj/nes-cpu.o: $(nes)/cpu/cpu.cpp $(call rwildcard,$(nes)/cpu/)
obj/nes-apu.o: $(nes)/apu/apu.cpp $(call rwildcard,$(nes)/apu/)
obj/nes-ppu.o: $(nes)/ppu/ppu.cpp $(call rwildcard,$(nes)/ppu/)

View File

@ -0,0 +1,50 @@
//NES-AMROM
//NES-ANROM
//NES-AN1ROM
//NES-AOROM
struct AxROM : Board {
uint4 prg_bank;
bool mirror_select;
uint8 prg_read(unsigned addr) {
if(addr & 0x8000) return Board::prg_read((prg_bank << 15) | (addr & 0x7fff));
return cpu.mdr();
}
void prg_write(unsigned addr, uint8 data) {
if(addr & 0x8000) {
prg_bank = data & 0x0f;
mirror_select = data & 0x10;
}
}
uint8 chr_read(unsigned addr) {
if(addr & 0x2000) return ppu.ciram_read((mirror_select << 10) | (addr & 0x03ff));
return Board::chr_read(addr);
}
void chr_write(unsigned addr, uint8 data) {
if(addr & 0x2000) return ppu.ciram_write((mirror_select << 10) | (addr & 0x03ff), data);
return Board::chr_write(addr, data);
}
void power() {
reset();
}
void reset() {
prg_bank = 0x0f;
mirror_select = 0;
}
void serialize(serializer &s) {
s.integer(prg_bank);
s.integer(mirror_select);
}
AxROM(BML::Node &board, const uint8_t *data, unsigned size) : Board(board, data, size) {
}
};

View File

@ -1,4 +1,8 @@
#include "axrom.cpp"
#include "cnrom.cpp"
#include "nrom.cpp"
#include "sxrom.cpp"
#include "uxrom.cpp"
unsigned Board::mirror(unsigned addr, unsigned size) const {
unsigned base = 0;
@ -18,73 +22,94 @@ unsigned Board::mirror(unsigned addr, unsigned size) const {
return base;
}
void Board::main() {
while(true) {
if(scheduler.sync == Scheduler::SynchronizeMode::All) {
scheduler.exit(Scheduler::ExitReason::SynchronizeEvent);
}
cartridge.clock += 12 * 4095;
tick();
}
}
void Board::tick() {
cartridge.clock += 12;
if(cartridge.clock >= 0 && scheduler.sync != Scheduler::SynchronizeMode::All) co_switch(cpu.thread);
}
uint8 Board::prg_read(unsigned addr) {
return prg.data[mirror(addr, prg.size)];
return prgrom.data[mirror(addr, prgrom.size)];
}
void Board::prg_write(unsigned addr, uint8 data) {
prg.data[mirror(addr, prg.size)] = data;
prgrom.data[mirror(addr, prgrom.size)] = data;
}
uint8 Board::chr_read(unsigned addr) {
return chr.data[mirror(addr, chr.size)];
if(chrrom.size) return chrrom.data[mirror(addr, chrrom.size)];
if(chrram.size) return chrram.data[mirror(addr, chrram.size)];
return 0u;
}
void Board::chr_write(unsigned addr, uint8 data) {
chr.data[mirror(addr, chr.size)] = data;
if(chrram.size) chrram.data[mirror(addr, chrram.size)] = data;
}
Board* Board::create(const string &xml, const uint8_t *data, unsigned size) {
string type;
string configuration;
xml_element document = xml_parse(xml);
for(auto &head : document.element) {
if(head.name == "cartridge") {
for(auto &node : head.element) {
if(node.name == "board") {
configuration = node.content;
for(auto &attr : node.attribute) {
if(attr.name == "type") type = attr.parse();
}
}
}
}
}
Board *board = nullptr;
if(type == "NES-NROM-256") board = new NROM;
assert(board != nullptr);
for(auto &head : document.element) {
if(head.name == "cartridge") {
for(auto &node : head.element) {
if(node.name == "board") {
for(auto &leaf : node.element) {
if(leaf.name == "prg") {
for(auto &attr : leaf.attribute) {
if(attr.name == "size") board->prg.size = decimal(attr.content);
}
}
if(leaf.name == "chr") {
for(auto &attr : leaf.attribute) {
if(attr.name == "size") board->chr.size = decimal(attr.content);
}
}
}
}
}
}
}
board->prg.data = new uint8[board->prg.size];
memcpy(board->prg.data, data, board->prg.size);
board->chr.data = new uint8[board->chr.size];
memcpy(board->chr.data, data + board->prg.size, board->chr.size);
board->configure({ "<board>\n", configuration, "</board>\n" });
return board;
Board::Memory Board::memory() {
return prgram;
}
void Board::power() {
}
void Board::reset() {
}
void Board::serialize(serializer &s) {
if(prgram.size) s.array(prgram.data, prgram.size);
if(chrram.size) s.array(chrram.data, chrram.size);
}
Board::Board(BML::Node &board, const uint8_t *data, unsigned size) {
information.type = board["type"].value();
information.battery = board["prg"]["battery"].value();
prgrom.size = decimal(board["prg"]["rom"].value());
prgram.size = decimal(board["prg"]["ram"].value());
chrrom.size = decimal(board["chr"]["rom"].value());
chrram.size = decimal(board["chr"]["ram"].value());
if(prgrom.size) prgrom.data = new uint8[prgrom.size]();
if(prgram.size) prgram.data = new uint8[prgram.size]();
if(chrrom.size) chrrom.data = new uint8[chrrom.size]();
if(chrram.size) chrram.data = new uint8[chrram.size]();
if(prgrom.size) memcpy(prgrom.data, data, prgrom.size);
if(chrrom.size) memcpy(chrrom.data, data + prgrom.size, chrrom.size);
}
Board::~Board() {
if(prgrom.size) delete[] prgrom.data;
if(prgram.size) delete[] prgram.data;
if(chrrom.size) delete[] chrrom.data;
if(chrram.size) delete[] chrram.data;
}
Board* Board::load(const string &markup, const uint8_t *data, unsigned size) {
BML::Node document(markup);
auto &board = document["cartridge"]["board"];
string type = board["type"].value();
if(type == "NES-AMROM" ) return new AxROM(board, data, size);
if(type == "NES-ANROM" ) return new AxROM(board, data, size);
if(type == "NES-AN1ROM" ) return new AxROM(board, data, size);
if(type == "NES-AOROM" ) return new AxROM(board, data, size);
if(type == "NES-CNROM" ) return new CNROM(board, data, size);
if(type == "NES-NROM-256") return new NROM(board, data, size);
if(type == "NES-UNROM" ) return new UxROM(board, data, size);
if(type == "NES-SXROM" ) return new SxROM(board, data, size);
if(type == "NES-UOROM" ) return new UxROM(board, data, size);
return nullptr;
}

View File

@ -1,25 +1,40 @@
struct Board {
struct Memory {
uint8_t *data;
unsigned size;
inline Memory() : data(nullptr), size(0u) {}
};
unsigned mirror(unsigned addr, unsigned size) const;
virtual void main();
virtual void tick();
virtual uint8 prg_read(unsigned addr);
virtual void prg_write(unsigned addr, uint8 data);
virtual uint8 chr_read(unsigned addr);
virtual void chr_write(unsigned addr, uint8 data);
virtual void configure(const string &xml) = 0;
virtual Memory memory();
static Board* create(const string &xml, const uint8_t *data, unsigned size);
virtual void power();
virtual void reset();
virtual void serialize(serializer&);
Board(BML::Node &board, const uint8_t *data, unsigned size);
~Board();
static Board* load(const string &markup, const uint8_t *data, unsigned size);
struct Information {
string type;
bool battery;
} information;
struct Memory {
uint8_t *data;
unsigned size;
};
Memory prg;
Memory chr;
protected:
Memory prgrom;
Memory prgram;
Memory chrrom;
Memory chrram;
};

View File

@ -0,0 +1,54 @@
//NES-CNROM
struct CNROM : Board {
struct Settings {
bool mirror; //0 = horizontal, 1 = vertical
} settings;
uint2 chr_bank;
uint8 prg_read(unsigned addr) {
if(addr & 0x8000) return Board::prg_read(addr & 0x7fff);
return cpu.mdr();
}
void prg_write(unsigned addr, uint8 data) {
if(addr & 0x8000) chr_bank = data & 0x03;
}
uint8 chr_read(unsigned addr) {
if(addr & 0x2000) {
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_read(addr & 0x07ff);
}
addr = (chr_bank * 0x2000) + (addr & 0x1fff);
return Board::chr_read(addr);
}
void chr_write(unsigned addr, uint8 data) {
if(addr & 0x2000) {
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_write(addr & 0x07ff, data);
}
addr = (chr_bank * 0x2000) + (addr & 0x1fff);
Board::chr_write(addr, data);
}
void power() {
reset();
}
void reset() {
chr_bank = 0;
}
void serialize(serializer &s) {
s.integer(chr_bank);
}
CNROM(BML::Node &board, const uint8_t *data, unsigned size) : Board(board, data, size) {
settings.mirror = board["mirror"].value() == "vertical" ? 1 : 0;
}
};

View File

@ -1,7 +1,10 @@
//NES-NROM-128
//NES-NROM-256
struct NROM : Board {
struct Settings {
enum class Mirror : unsigned { Horizontal, Vertical } mirror;
bool mirror; //0 = horizontal, 1 = vertical
} settings;
uint8 prg_read(unsigned addr) {
@ -14,7 +17,7 @@ void prg_write(unsigned addr, uint8 data) {
uint8 chr_read(unsigned addr) {
if(addr & 0x2000) {
if(settings.mirror == Settings::Mirror::Horizontal) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_read(addr & 0x07ff);
}
return Board::chr_read(addr);
@ -22,26 +25,14 @@ uint8 chr_read(unsigned addr) {
void chr_write(unsigned addr, uint8 data) {
if(addr & 0x2000) {
if(settings.mirror == Settings::Mirror::Horizontal) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_write(addr, data);
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_write(addr & 0x07ff, data);
}
return Board::chr_write(addr, data);
}
void configure(const string &xml) {
xml_element document = xml_parse(xml);
for(auto &node : document.element) {
if(node.name == "mirror") {
for(auto &attr : node.attribute) {
if(attr.name == "type") {
if(attr.content == "horizontal") settings.mirror = Settings::Mirror::Horizontal;
if(attr.content == "vertical" ) settings.mirror = Settings::Mirror::Vertical;
}
}
}
}
NROM(BML::Node &board, const uint8_t *data, unsigned size) : Board(board, data, size) {
settings.mirror = board["mirror"].value() == "vertical" ? 1 : 0;
}
};
NROM nrom;

View File

@ -0,0 +1,29 @@
//NES-SAROM
//NES-SBROM
//NES-SCROM
//NES-SC1ROM
//NES-SEROM
//NES-SFROM
//NES-SGROM
//NES-SHROM
//NES-SH1ROM
//NES-SIROM
//NES-SJROM
//NES-SKROM
//NES-SLROM
//NES-SL1ROM
//NES-SL2ROM
//NES-SL3ROM
//NES-SLRROM
//NES-SMROM
//NES-SNROM
//NES-SOROM
//NES-SUROM
//NES-SXROM
struct SxROM : Board {
SxROM(BML::Node &board, const uint8_t *data, unsigned size) : Board(board, data, size) {
}
};

View File

@ -0,0 +1,54 @@
//NES-UNROM
//NES-UOROM
struct UxROM : Board {
struct Settings {
bool mirror; //0 = horizontal, 1 = vertical
} settings;
uint4 prg_bank;
uint8 prg_read(unsigned addr) {
if((addr & 0xc000) == 0x8000) return Board::prg_read((prg_bank << 14) | (addr & 0x3fff));
if((addr & 0xc000) == 0xc000) return Board::prg_read(( 0x0f << 14) | (addr & 0x3fff));
return cpu.mdr();
}
void prg_write(unsigned addr, uint8 data) {
if(addr & 0x8000) prg_bank = data & 0x0f;
}
uint8 chr_read(unsigned addr) {
if(addr & 0x2000) {
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_read(addr);
}
return Board::chr_read(addr);
}
void chr_write(unsigned addr, uint8 data) {
if(addr & 0x2000) {
if(settings.mirror == 0) addr = ((addr & 0x0800) >> 1) | (addr & 0x03ff);
return ppu.ciram_write(addr, data);
}
return Board::chr_write(addr, data);
}
void power() {
reset();
}
void reset() {
prg_bank = 0;
}
void serialize(serializer &s) {
s.integer(prg_bank);
}
UxROM(BML::Node &board, const uint8_t *data, unsigned size) : Board(board, data, size) {
settings.mirror = board["mirror"].value() == "vertical" ? 1 : 0;
}
};

View File

@ -2,10 +2,9 @@
namespace NES {
#include "ines.cpp"
#include "chip/chip.cpp"
#include "board/board.cpp"
//#define BOARD
Cartridge cartridge;
void Cartridge::Main() {
@ -13,92 +12,42 @@ void Cartridge::Main() {
}
void Cartridge::main() {
mapper->main();
board->main();
}
void Cartridge::load(const string &xml, const uint8_t *data, unsigned size) {
#ifdef BOARD
rom_size = size;
rom_data = new uint8[rom_size];
memcpy(rom_data, data, size);
#else
rom_size = size - 16;
rom_data = new uint8[rom_size];
memcpy(rom_data, data + 16, size - 16);
#endif
#ifdef BOARD
prg_size = 32768;
chr_size = 8192;
#else
prg_size = data[4] * 0x4000;
chr_size = data[5] * 0x2000;
#endif
prg_data = new uint8[prg_size];
memcpy(prg_data, rom_data, prg_size);
if(chr_size) {
chr_ram = false;
chr_data = new uint8[chr_size];
memcpy(chr_data, rom_data + prg_size, chr_size);
void Cartridge::load(const string &markup, const uint8_t *data, unsigned size) {
if((size & 0xff) == 0) {
sha256 = nall::sha256(data, size);
board = Board::load(markup, data, size);
} else {
chr_ram = true;
chr_size = 0x2000;
chr_data = new uint8[chr_size]();
board = Board::load(markup != "" ? markup : iNES(data, size), data + 16, size - 16);
}
mirroring = ((data[6] & 0x08) >> 2) | (data[6] & 0x01);
uint8 mapperNumber = ((data[7] >> 4) << 4) | (data[6] >> 4);
switch(mapperNumber) {
default : mapper = &Mapper::none; break;
case 1: mapper = &Mapper::mmc1; break;
case 2: mapper = &Mapper::uorom; break;
case 3: mapper = &Mapper::cnrom; break;
case 4: mapper = &Mapper::mmc3; break;
case 7: mapper = &Mapper::aorom; break;
case 16: mapper = &Mapper::bandaiFCG; break;
case 24: mapper = &Mapper::vrc6; Mapper::vrc6.abus_swap = 0; break;
case 26: mapper = &Mapper::vrc6; Mapper::vrc6.abus_swap = 1; break;
}
sha256 = nall::sha256(rom_data, rom_size);
#ifdef BOARD
board = Board::create(xml, rom_data, rom_size);
#endif
system.load();
loaded = true;
}
void Cartridge::unload() {
if(loaded == false) return;
delete[] rom_data;
delete[] prg_data;
delete[] chr_data;
loaded = false;
}
unsigned Cartridge::ram_size() {
return mapper->ram_size();
return board->memory().size;
}
uint8* Cartridge::ram_data() {
return mapper->ram_data();
return board->memory().data;
}
void Cartridge::power() {
create(Cartridge::Main, 21477272);
mapper->power();
board->power();
}
void Cartridge::reset() {
create(Cartridge::Main, 21477272);
mapper->reset();
board->reset();
}
Cartridge::Cartridge() {
@ -106,37 +55,23 @@ Cartridge::Cartridge() {
}
uint8 Cartridge::prg_read(unsigned addr) {
#ifdef BOARD
return board->prg_read(addr);
#endif
return mapper->prg_read(addr);
}
void Cartridge::prg_write(unsigned addr, uint8 data) {
#ifdef BOARD
return board->prg_write(addr, data);
#endif
return mapper->prg_write(addr, data);
}
uint8 Cartridge::chr_read(unsigned addr) {
#ifdef BOARD
return board->chr_read(addr);
#endif
return mapper->chr_read(addr);
}
void Cartridge::chr_write(unsigned addr, uint8 data) {
#ifdef BOARD
return board->chr_write(addr, data);
#endif
return mapper->chr_write(addr, data);
}
void Cartridge::serialize(serializer &s) {
if(chr_ram) s.array(chr_data, chr_size);
return mapper->serialize(s);
return board->serialize(s);
}
}

View File

@ -1,10 +1,11 @@
#include "chip/chip.hpp"
#include "board/board.hpp"
struct Cartridge : Processor, property<Cartridge> {
static void Main();
void main();
void load(const string &xml, const uint8_t *data, unsigned size);
void load(const string &markup, const uint8_t *data, unsigned size);
void unload();
unsigned ram_size();
@ -21,25 +22,12 @@ struct Cartridge : Processor, property<Cartridge> {
//privileged:
Board *board;
Mapper::Mapper *mapper;
uint8 prg_read(unsigned addr);
void prg_write(unsigned addr, uint8 data);
uint8 chr_read(unsigned addr);
void chr_write(unsigned addr, uint8 data);
uint8 *rom_data;
unsigned rom_size;
uint8 *prg_data;
unsigned prg_size;
uint8 *chr_data;
unsigned chr_size;
bool chr_ram;
unsigned mirroring;
};
extern Cartridge cartridge;

View File

View File

@ -0,0 +1,2 @@
struct Chip {
};

66
bsnes/nes/cartridge/ines.cpp Executable file
View File

@ -0,0 +1,66 @@
static string iNES(const uint8_t *data, unsigned size) {
if(size < 16) return "";
if(data[0] != 'N') return "";
if(data[1] != 'E') return "";
if(data[2] != 'S') return "";
if(data[3] != 0x1a) return "";
string output;
unsigned mapper = ((data[7] >> 4) << 4) | (data[6] >> 4);
unsigned mirror = ((data[6] & 0x08) >> 2) | (data[6] & 0x01);
unsigned prgrom = data[4] * 0x4000;
unsigned chrrom = data[5] * 0x2000;
unsigned prgram = 0;
unsigned chrram = chrrom == 0 ? 8192 : 0;
print("iNES mapper: ", mapper, "\n");
output.append("cartridge\n");
switch(mapper) {
default:
output.append(" board type=NES-NROM-256\n");
output.append(" mirror=", mirror == 0 ? "horizontal" : "vertical", "\n");
break;
case 1:
output.append(" board type=NES-SXROM\n");
break;
case 2:
output.append(" board type=NES-UOROM\n");
output.append(" mirror=", mirror == 0 ? "horizontal" : "vertical", "\n");
break;
case 3:
output.append(" board type=NES-CNROM\n");
output.append(" mirror=", mirror == 0 ? "horizontal" : "vertical", "\n");
break;
case 7:
output.append(" board type=NES-AOROM\n");
break;
}
output.append(" prg rom=", prgrom, " ram=", prgram, "\n");
output.append(" chr rom=", chrrom, " ram=", chrram, "\n");
print(output, "\n");
return output;
}
/*
switch(mapperNumber) {
//default : mapper = &Mapper::none; break;
//case 1: mapper = &Mapper::mmc1; break;
//case 2: mapper = &Mapper::uorom; break;
//case 3: mapper = &Mapper::cnrom; break;
case 4: mapper = &Mapper::mmc3; break;
//case 7: mapper = &Mapper::aorom; break;
case 16: mapper = &Mapper::bandaiFCG; break;
case 24: mapper = &Mapper::vrc6; Mapper::vrc6.abus_swap = 0; break;
case 26: mapper = &Mapper::vrc6; Mapper::vrc6.abus_swap = 1; break;
}
*/

View File

@ -27,8 +27,8 @@ bool Interface::cartridgeLoaded() {
return cartridge.loaded();
}
void Interface::loadCartridge(const string &xml, const uint8_t *data, unsigned size) {
cartridge.load(xml, data, size);
void Interface::loadCartridge(const string &markup, const uint8_t *data, unsigned size) {
cartridge.load(markup, data, size);
system.power();
}

View File

@ -8,7 +8,7 @@ struct Interface {
virtual void initialize(Interface*);
virtual bool cartridgeLoaded();
virtual void loadCartridge(const string &xml, const uint8_t *data, unsigned size);
virtual void loadCartridge(const string &markup, const uint8_t *data, unsigned size);
virtual void unloadCartridge();
enum class Memory : unsigned {

View File

@ -99,9 +99,8 @@ namespace NES {
#include <nes/system/system.hpp>
#include <nes/scheduler/scheduler.hpp>
#include <nes/input/input.hpp>
#include <nes/mapper/mapper.hpp>
#include <nes/cartridge/cartridge.hpp>
#include <nes/memory/memory.hpp>
#include <nes/cartridge/cartridge.hpp>
#include <nes/cpu/cpu.hpp>
#include <nes/apu/apu.hpp>
#include <nes/ppu/ppu.hpp>

View File

@ -20,10 +20,10 @@ bool InterfaceNES::loadCartridge(const string &filename) {
interface->unloadCartridge();
interface->baseName = nall::basename(filename);
string xml;
xml.readfile({ interface->baseName, ".xml" });
string markup;
markup.readfile({ interface->baseName, ".bml" });
NES::Interface::loadCartridge(xml, fp.data(), fp.size());
NES::Interface::loadCartridge(markup, fp.data(), fp.size());
fp.close();
if(NES::Interface::memorySize(NES::Interface::Memory::RAM) > 0) {

View File

@ -49,7 +49,7 @@ Application::Application(int argc, char **argv) {
inputManager = new InputManager;
utility = new Utility;
title = "bsnes v082.25";
title = "bsnes v082.26";
string fontFamily = Intrinsics::platform() == Intrinsics::Platform::Windows ? "Tahoma, " : "Sans, ";
normalFont = { fontFamily, "8" };

View File

@ -129,31 +129,13 @@ bool CheatEditor::load(const string &filename) {
if(data.readfile(filename) == false) return false;
unsigned n = 0;
xml_element document = xml_parse(data);
for(auto &head : document.element) {
if(head.name == "cartridge") {
for(auto &node : head.element) {
if(node.name == "cheat") {
bool enable = false;
string description;
string code;
for(auto &attribute : node.attribute) {
if(attribute.name == "enabled") enable = (attribute.parse() == "true");
}
for(auto &element : node.element) {
if(element.name == "description") description = element.parse();
else if(element.name == "code") code.append(element.parse(), "+");
}
code.rtrim<1>("+");
cheatList.setChecked(n, enable);
cheatText[n][Code] = code;
cheatText[n][Desc] = description;
if(++n >= 128) break;
}
}
}
BML::Node document(data);
for(auto &cheat : document["cartridge"]) {
if(cheat.name() != "cheat") continue;
cheatList.setChecked(n, cheat["enable"].exists());
cheatText[n][Code] = cheat["code"].value();
cheatText[n][Desc] = cheat["description"].value();
if(++n >= 128) break;
}
updateUI(), updateInterface();
@ -179,21 +161,14 @@ bool CheatEditor::save(const string &filename) {
file fp;
if(fp.open(filename, file::mode::write) == false) return false;
fp.print("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
fp.print("<cartridge>\n");
fp.print("cartridge sha256=", interface->sha256(), "\n");
for(unsigned n = 0; n <= lastSave; n++) {
fp.print(" <cheat enabled=\"", cheatList.checked(n) ? "true" : "false", "\">\n");
fp.print(" <description>", cheatText[n][Desc], "</description>\n");
lstring list;
list.split("+", cheatText[n][Code]);
for(auto &code : list) {
fp.print(" <code>", code, "</code>\n");
}
fp.print(" </cheat>\n");
fp.print(" cheat", cheatList.checked(n) ? " enable" : "", "\n");
fp.print(" description=|", cheatText[n][Desc], "|\n");
fp.print(" code=|", cheatText[n][Code], "|\n");
}
fp.print("</cartridge>\n");
fp.close();
fp.close();
return true;
}