mirror of https://github.com/bsnes-emu/bsnes.git
Update to v098r11 release.
byuu says: Changelog: - fixed nall/path.hpp compilation issue - fixed ruby/audio/xaudio header declaration compilation issue (again) - cleaned up xaudio2.hpp file to match my coding syntax (12.5% of the file was whitespace overkill) - added null terminator entry to nall/windows/utf8.hpp argc[] array - nall/windows/guid.hpp uses the Windows API for generating the GUID - this should stop all the bug reports where two nall users were generating GUIDs at the exact same second - fixed hiro/cocoa compilation issue with uint# types - fixed major higan/sfc Super Game Boy audio latency issue - fixed higan/sfc CPU core bug with pei, [dp], [dp]+y instructions - major cleanups to higan/processor/r65816 core - merged emulation/native-mode opcodes - use camel-case naming on memory.hpp functions - simplify address masking code for memory.hpp functions - simplify a few opcodes themselves (avoid redundant copies, etc) - rename regs.* to r.* to match modern convention of other CPU cores - removed device.order<> concept from Emulator::Interface - cores will now do the translation to make the job of the UI easier - fixed plurality naming of arrays in Emulator::Interface - example: emulator.ports[p].devices[d].inputs[i] - example: vector<Medium> media - probably more surprises Major show-stoppers to the next official release: - we need to work on GB core improvements: LY=153/0 case, multiple STAT IRQs case, GBC audio output regs, etc. - we need to re-add software cursors for light guns (Super Scope, Justifier) - after the above, we need to fix the turbo button for the Super Scope I really have no idea how I want to implement the light guns. Ideally, we'd want it in higan/video, so we can support the NES Zapper with the same code. But this isn't going to be easy, because only the SNES knows when its output is interlaced, and its resolutions can vary as {256,512}x{224,240,448,480} which requires pixel doubling that was hard-coded to the SNES-specific behavior, but isn't appropriate to be exposed in higan/video.
This commit is contained in:
parent
3ebc77c148
commit
ae5d380d06
|
@ -8,7 +8,7 @@ using namespace nall;
|
||||||
|
|
||||||
namespace Emulator {
|
namespace Emulator {
|
||||||
static const string Name = "higan";
|
static const string Name = "higan";
|
||||||
static const string Version = "098.10";
|
static const string Version = "098.11";
|
||||||
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/";
|
||||||
|
|
|
@ -17,13 +17,13 @@ struct Interface {
|
||||||
} capability;
|
} capability;
|
||||||
} information;
|
} information;
|
||||||
|
|
||||||
struct Media {
|
struct Medium {
|
||||||
uint id;
|
uint id;
|
||||||
string name;
|
string name;
|
||||||
string type;
|
string type;
|
||||||
bool bootable; //false for cartridge slots (eg Sufami Turbo cartridges)
|
bool bootable; //false for cartridge slots (eg Sufami Turbo cartridges)
|
||||||
};
|
};
|
||||||
vector<Media> media;
|
vector<Medium> media;
|
||||||
|
|
||||||
struct Device {
|
struct Device {
|
||||||
uint id;
|
uint id;
|
||||||
|
@ -35,16 +35,15 @@ struct Interface {
|
||||||
string name;
|
string name;
|
||||||
uintptr guid; //user data field
|
uintptr guid; //user data field
|
||||||
};
|
};
|
||||||
vector<Input> input;
|
vector<Input> inputs;
|
||||||
vector<uint> order;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Port {
|
struct Port {
|
||||||
uint id;
|
uint id;
|
||||||
string name;
|
string name;
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
vector<Port> port;
|
vector<Port> ports;
|
||||||
|
|
||||||
struct Bind {
|
struct Bind {
|
||||||
virtual auto loadRequest(uint, string, string, bool) -> void {}
|
virtual auto loadRequest(uint, string, string, bool) -> void {}
|
||||||
|
|
|
@ -15,12 +15,15 @@ auto Input::latch(bool data) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Input::data(bool port) -> bool {
|
auto Input::data(bool port) -> bool {
|
||||||
|
//table to convert native button ordering to Emulator::Interface ordering
|
||||||
|
static const uint lookup[] = {5, 4, 6, 7, 0, 1, 2, 3};
|
||||||
|
|
||||||
bool result = 0;
|
bool result = 0;
|
||||||
|
|
||||||
if(port == 0) {
|
if(port == 0) {
|
||||||
if(port1 == Device::Joypad) {
|
if(port1 == Device::Joypad) {
|
||||||
if(counter1 >= 8) return 1;
|
if(counter1 >= 8) return 1;
|
||||||
result = interface->inputPoll(0, 0u, counter1);
|
result = interface->inputPoll(0, 0u, lookup[counter1]);
|
||||||
if(latchdata == 0) counter1++;
|
if(latchdata == 0) counter1++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -28,7 +31,7 @@ auto Input::data(bool port) -> bool {
|
||||||
if(port == 1) {
|
if(port == 1) {
|
||||||
if(port2 == Device::Joypad) {
|
if(port2 == Device::Joypad) {
|
||||||
if(counter2 >= 8) return 1;
|
if(counter2 >= 8) return 1;
|
||||||
result = interface->inputPoll(1, 0u, counter2);
|
result = interface->inputPoll(1, 0u, lookup[counter2]);
|
||||||
if(latchdata == 0) counter2++;
|
if(latchdata == 0) counter2++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,25 +22,24 @@ Interface::Interface() {
|
||||||
media.append({ID::Famicom, "Famicom", "fc", true});
|
media.append({ID::Famicom, "Famicom", "fc", true});
|
||||||
|
|
||||||
{ Device device{0, ID::Port1 | ID::Port2, "Controller"};
|
{ Device device{0, ID::Port1 | ID::Port2, "Controller"};
|
||||||
device.input.append({0, 0, "A" });
|
device.inputs.append({0, 0, "Up" });
|
||||||
device.input.append({1, 0, "B" });
|
device.inputs.append({1, 0, "Down" });
|
||||||
device.input.append({2, 0, "Select"});
|
device.inputs.append({2, 0, "Left" });
|
||||||
device.input.append({3, 0, "Start" });
|
device.inputs.append({3, 0, "Right" });
|
||||||
device.input.append({4, 0, "Up" });
|
device.inputs.append({4, 0, "B" });
|
||||||
device.input.append({5, 0, "Down" });
|
device.inputs.append({5, 0, "A" });
|
||||||
device.input.append({6, 0, "Left" });
|
device.inputs.append({6, 0, "Select"});
|
||||||
device.input.append({7, 0, "Right" });
|
device.inputs.append({7, 0, "Start" });
|
||||||
device.order = {4, 5, 6, 7, 1, 0, 2, 3};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
port.append({0, "Port 1"});
|
ports.append({0, "Port 1"});
|
||||||
port.append({1, "Port 2"});
|
ports.append({1, "Port 2"});
|
||||||
|
|
||||||
for(auto& device : this->device) {
|
for(auto& device : devices) {
|
||||||
for(auto& port : this->port) {
|
for(auto& port : ports) {
|
||||||
if(device.portmask & (1 << port.id)) {
|
if(device.portmask & (1 << port.id)) {
|
||||||
port.device.append(device);
|
port.devices.append(device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,14 +97,14 @@ auto Interface::videoColor(uint32 n) -> uint64 {
|
||||||
v *= brightness / 12.0;
|
v *= brightness / 12.0;
|
||||||
|
|
||||||
y += v;
|
y += v;
|
||||||
i += v * std::cos((3.141592653 / 6.0) * (p + hue));
|
i += v * cos((3.141592653 / 6.0) * (p + hue));
|
||||||
q += v * std::sin((3.141592653 / 6.0) * (p + hue));
|
q += v * sin((3.141592653 / 6.0) * (p + hue));
|
||||||
}
|
}
|
||||||
|
|
||||||
i *= saturation;
|
i *= saturation;
|
||||||
q *= saturation;
|
q *= saturation;
|
||||||
|
|
||||||
auto gammaAdjust = [=](double f) { return f < 0.0 ? 0.0 : std::pow(f, 2.2 / gamma); };
|
auto gammaAdjust = [=](double f) { return f < 0.0 ? 0.0 : pow(f, 2.2 / gamma); };
|
||||||
uint64 r = uclamp<16>(65535.0 * gammaAdjust(y + 0.946882 * i + 0.623557 * q));
|
uint64 r = uclamp<16>(65535.0 * gammaAdjust(y + 0.946882 * i + 0.623557 * q));
|
||||||
uint64 g = uclamp<16>(65535.0 * gammaAdjust(y + -0.274788 * i + -0.635691 * q));
|
uint64 g = uclamp<16>(65535.0 * gammaAdjust(y + -0.274788 * i + -0.635691 * q));
|
||||||
uint64 b = uclamp<16>(65535.0 * gammaAdjust(y + -1.108545 * i + 1.709007 * q));
|
uint64 b = uclamp<16>(65535.0 * gammaAdjust(y + -1.108545 * i + 1.709007 * q));
|
||||||
|
|
|
@ -55,7 +55,7 @@ struct Interface : Emulator::Interface {
|
||||||
auto set(const string& name, const any& value) -> bool override;
|
auto set(const string& name, const any& value) -> bool override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Settings {
|
struct Settings {
|
||||||
|
|
|
@ -25,19 +25,18 @@ Interface::Interface() {
|
||||||
|
|
||||||
{
|
{
|
||||||
Device device{0, ID::Device, "Controller"};
|
Device device{0, ID::Device, "Controller"};
|
||||||
device.input.append({0, 0, "Up" });
|
device.inputs.append({0, 0, "Up" });
|
||||||
device.input.append({1, 0, "Down" });
|
device.inputs.append({1, 0, "Down" });
|
||||||
device.input.append({2, 0, "Left" });
|
device.inputs.append({2, 0, "Left" });
|
||||||
device.input.append({3, 0, "Right" });
|
device.inputs.append({3, 0, "Right" });
|
||||||
device.input.append({4, 0, "B" });
|
device.inputs.append({4, 0, "B" });
|
||||||
device.input.append({5, 0, "A" });
|
device.inputs.append({5, 0, "A" });
|
||||||
device.input.append({6, 0, "Select"});
|
device.inputs.append({6, 0, "Select"});
|
||||||
device.input.append({7, 0, "Start" });
|
device.inputs.append({7, 0, "Start" });
|
||||||
device.order = {0, 1, 2, 3, 4, 5, 6, 7};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
port.append({0, "Device", {device[0]}});
|
ports.append({0, "Device", {devices[0]}});
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Interface::manifest() -> string {
|
auto Interface::manifest() -> string {
|
||||||
|
|
|
@ -69,7 +69,7 @@ struct Interface : Emulator::Interface {
|
||||||
auto joypWrite(bool p15, bool p14) -> void;
|
auto joypWrite(bool p15, bool p14) -> void;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Settings {
|
struct Settings {
|
||||||
|
|
|
@ -91,12 +91,15 @@ auto CPU::sync_step(uint clocks) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::keypad_run() -> void {
|
auto CPU::keypad_run() -> void {
|
||||||
if(regs.keypad.control.enable == false) return;
|
//lookup table to convert button indexes to Emulator::Interface indexes
|
||||||
|
static const uint lookup[] = {5, 4, 8, 9, 3, 2, 0, 1, 7, 6};
|
||||||
|
|
||||||
|
if(!regs.keypad.control.enable) return;
|
||||||
|
|
||||||
bool test = regs.keypad.control.condition; //0 = OR, 1 = AND
|
bool test = regs.keypad.control.condition; //0 = OR, 1 = AND
|
||||||
for(auto n : range(10)) {
|
for(auto n : range(10)) {
|
||||||
if(regs.keypad.control.flag[n] == false) continue;
|
if(!regs.keypad.control.flag[n]) continue;
|
||||||
bool input = interface->inputPoll(0, 0, n);
|
bool input = interface->inputPoll(0, 0, lookup[n]);
|
||||||
if(regs.keypad.control.condition == 0) test |= input;
|
if(regs.keypad.control.condition == 0) test |= input;
|
||||||
if(regs.keypad.control.condition == 1) test &= input;
|
if(regs.keypad.control.condition == 1) test &= input;
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,9 +61,10 @@ auto CPU::read(uint32 addr) -> uint8 {
|
||||||
|
|
||||||
//KEYINPUT
|
//KEYINPUT
|
||||||
case 0x04000130: {
|
case 0x04000130: {
|
||||||
|
static const uint lookup[] = {5, 4, 8, 9, 3, 2, 0, 1};
|
||||||
if(auto result = player.keyinput()) return result() >> 0;
|
if(auto result = player.keyinput()) return result() >> 0;
|
||||||
uint8 result = 0;
|
uint8 result = 0;
|
||||||
for(uint n = 0; n < 8; n++) result |= interface->inputPoll(0, 0, n) << n;
|
for(uint n = 0; n < 8; n++) result |= interface->inputPoll(0, 0, lookup[n]) << n;
|
||||||
if((result & 0xc0) == 0xc0) result &= (uint8)~0xc0; //up+down cannot be pressed simultaneously
|
if((result & 0xc0) == 0xc0) result &= (uint8)~0xc0; //up+down cannot be pressed simultaneously
|
||||||
if((result & 0x30) == 0x30) result &= (uint8)~0x30; //left+right cannot be pressed simultaneously
|
if((result & 0x30) == 0x30) result &= (uint8)~0x30; //left+right cannot be pressed simultaneously
|
||||||
return result ^ 0xff;
|
return result ^ 0xff;
|
||||||
|
@ -71,8 +72,8 @@ auto CPU::read(uint32 addr) -> uint8 {
|
||||||
case 0x04000131: {
|
case 0x04000131: {
|
||||||
if(auto result = player.keyinput()) return result() >> 8;
|
if(auto result = player.keyinput()) return result() >> 8;
|
||||||
uint8 result = 0;
|
uint8 result = 0;
|
||||||
result |= interface->inputPoll(0, 0, 8) << 0;
|
result |= interface->inputPoll(0, 0, 7) << 0;
|
||||||
result |= interface->inputPoll(0, 0, 9) << 1;
|
result |= interface->inputPoll(0, 0, 6) << 1;
|
||||||
return result ^ 0x03;
|
return result ^ 0x03;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,22 +22,21 @@ Interface::Interface() {
|
||||||
media.append({ID::GameBoyAdvance, "Game Boy Advance", "gba", true});
|
media.append({ID::GameBoyAdvance, "Game Boy Advance", "gba", true});
|
||||||
|
|
||||||
{ Device device{0, ID::Device, "Controller"};
|
{ Device device{0, ID::Device, "Controller"};
|
||||||
device.input.append({ 0, 0, "A" });
|
device.inputs.append({ 0, 0, "Up" });
|
||||||
device.input.append({ 1, 0, "B" });
|
device.inputs.append({ 1, 0, "Down" });
|
||||||
device.input.append({ 2, 0, "Select"});
|
device.inputs.append({ 2, 0, "Left" });
|
||||||
device.input.append({ 3, 0, "Start" });
|
device.inputs.append({ 3, 0, "Right" });
|
||||||
device.input.append({ 4, 0, "Right" });
|
device.inputs.append({ 4, 0, "B" });
|
||||||
device.input.append({ 5, 0, "Left" });
|
device.inputs.append({ 5, 0, "A" });
|
||||||
device.input.append({ 6, 0, "Up" });
|
device.inputs.append({ 6, 0, "L" });
|
||||||
device.input.append({ 7, 0, "Down" });
|
device.inputs.append({ 7, 0, "R" });
|
||||||
device.input.append({ 8, 0, "R" });
|
device.inputs.append({ 8, 0, "Select"});
|
||||||
device.input.append({ 9, 0, "L" });
|
device.inputs.append({ 9, 0, "Start" });
|
||||||
device.input.append({10, 2, "Rumble"});
|
device.inputs.append({10, 2, "Rumble"});
|
||||||
device.order = {6, 7, 5, 4, 1, 0, 9, 8, 2, 3, 10};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
port.append({0, "Device", {device[0]}});
|
ports.append({0, "Device", {devices[0]}});
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Interface::manifest() -> string {
|
auto Interface::manifest() -> string {
|
||||||
|
|
|
@ -52,7 +52,7 @@ struct Interface : Emulator::Interface {
|
||||||
auto set(const string& name, const any& value) -> bool override;
|
auto set(const string& name, const any& value) -> bool override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Settings {
|
struct Settings {
|
||||||
|
|
|
@ -1,327 +1,327 @@
|
||||||
auto R65816::op_adc_b() {
|
auto R65816::op_adc_b() {
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
if(!regs.p.d) {
|
if(!r.p.d) {
|
||||||
result = regs.a.l + rd.l + regs.p.c;
|
result = r.a.l + rd.l + r.p.c;
|
||||||
} else {
|
} else {
|
||||||
result = (regs.a.l & 0x0f) + (rd.l & 0x0f) + (regs.p.c << 0);
|
result = (r.a.l & 0x0f) + (rd.l & 0x0f) + (r.p.c << 0);
|
||||||
if(result > 0x09) result += 0x06;
|
if(result > 0x09) result += 0x06;
|
||||||
regs.p.c = result > 0x0f;
|
r.p.c = result > 0x0f;
|
||||||
result = (regs.a.l & 0xf0) + (rd.l & 0xf0) + (regs.p.c << 4) + (result & 0x0f);
|
result = (r.a.l & 0xf0) + (rd.l & 0xf0) + (r.p.c << 4) + (result & 0x0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.p.v = ~(regs.a.l ^ rd.l) & (regs.a.l ^ result) & 0x80;
|
r.p.v = ~(r.a.l ^ rd.l) & (r.a.l ^ result) & 0x80;
|
||||||
if(regs.p.d && result > 0x9f) result += 0x60;
|
if(r.p.d && result > 0x9f) result += 0x60;
|
||||||
regs.p.c = result > 0xff;
|
r.p.c = result > 0xff;
|
||||||
regs.p.n = result & 0x80;
|
r.p.n = result & 0x80;
|
||||||
regs.p.z = (uint8_t)result == 0;
|
r.p.z = (uint8)result == 0;
|
||||||
|
|
||||||
regs.a.l = result;
|
r.a.l = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adc_w() {
|
auto R65816::op_adc_w() {
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
if(!regs.p.d) {
|
if(!r.p.d) {
|
||||||
result = regs.a.w + rd.w + regs.p.c;
|
result = r.a.w + rd.w + r.p.c;
|
||||||
} else {
|
} else {
|
||||||
result = (regs.a.w & 0x000f) + (rd.w & 0x000f) + (regs.p.c << 0);
|
result = (r.a.w & 0x000f) + (rd.w & 0x000f) + (r.p.c << 0);
|
||||||
if(result > 0x0009) result += 0x0006;
|
if(result > 0x0009) result += 0x0006;
|
||||||
regs.p.c = result > 0x000f;
|
r.p.c = result > 0x000f;
|
||||||
result = (regs.a.w & 0x00f0) + (rd.w & 0x00f0) + (regs.p.c << 4) + (result & 0x000f);
|
result = (r.a.w & 0x00f0) + (rd.w & 0x00f0) + (r.p.c << 4) + (result & 0x000f);
|
||||||
if(result > 0x009f) result += 0x0060;
|
if(result > 0x009f) result += 0x0060;
|
||||||
regs.p.c = result > 0x00ff;
|
r.p.c = result > 0x00ff;
|
||||||
result = (regs.a.w & 0x0f00) + (rd.w & 0x0f00) + (regs.p.c << 8) + (result & 0x00ff);
|
result = (r.a.w & 0x0f00) + (rd.w & 0x0f00) + (r.p.c << 8) + (result & 0x00ff);
|
||||||
if(result > 0x09ff) result += 0x0600;
|
if(result > 0x09ff) result += 0x0600;
|
||||||
regs.p.c = result > 0x0fff;
|
r.p.c = result > 0x0fff;
|
||||||
result = (regs.a.w & 0xf000) + (rd.w & 0xf000) + (regs.p.c << 12) + (result & 0x0fff);
|
result = (r.a.w & 0xf000) + (rd.w & 0xf000) + (r.p.c << 12) + (result & 0x0fff);
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.p.v = ~(regs.a.w ^ rd.w) & (regs.a.w ^ result) & 0x8000;
|
r.p.v = ~(r.a.w ^ rd.w) & (r.a.w ^ result) & 0x8000;
|
||||||
if(regs.p.d && result > 0x9fff) result += 0x6000;
|
if(r.p.d && result > 0x9fff) result += 0x6000;
|
||||||
regs.p.c = result > 0xffff;
|
r.p.c = result > 0xffff;
|
||||||
regs.p.n = result & 0x8000;
|
r.p.n = result & 0x8000;
|
||||||
regs.p.z = (uint16_t)result == 0;
|
r.p.z = (uint16)result == 0;
|
||||||
|
|
||||||
regs.a.w = result;
|
r.a.w = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_and_b() {
|
auto R65816::op_and_b() {
|
||||||
regs.a.l &= rd.l;
|
r.a.l &= rd.l;
|
||||||
regs.p.n = regs.a.l & 0x80;
|
r.p.n = r.a.l & 0x80;
|
||||||
regs.p.z = regs.a.l == 0;
|
r.p.z = r.a.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_and_w() {
|
auto R65816::op_and_w() {
|
||||||
regs.a.w &= rd.w;
|
r.a.w &= rd.w;
|
||||||
regs.p.n = regs.a.w & 0x8000;
|
r.p.n = r.a.w & 0x8000;
|
||||||
regs.p.z = regs.a.w == 0;
|
r.p.z = r.a.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_bit_b() {
|
auto R65816::op_bit_b() {
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.v = rd.l & 0x40;
|
r.p.v = rd.l & 0x40;
|
||||||
regs.p.z = (rd.l & regs.a.l) == 0;
|
r.p.z = (rd.l & r.a.l) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_bit_w() {
|
auto R65816::op_bit_w() {
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.v = rd.w & 0x4000;
|
r.p.v = rd.w & 0x4000;
|
||||||
regs.p.z = (rd.w & regs.a.w) == 0;
|
r.p.z = (rd.w & r.a.w) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cmp_b() {
|
auto R65816::op_cmp_b() {
|
||||||
int r = regs.a.l - rd.l;
|
int result = r.a.l - rd.l;
|
||||||
regs.p.n = r & 0x80;
|
r.p.n = result & 0x80;
|
||||||
regs.p.z = (uint8)r == 0;
|
r.p.z = (uint8)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cmp_w() {
|
auto R65816::op_cmp_w() {
|
||||||
int r = regs.a.w - rd.w;
|
int result = r.a.w - rd.w;
|
||||||
regs.p.n = r & 0x8000;
|
r.p.n = result & 0x8000;
|
||||||
regs.p.z = (uint16)r == 0;
|
r.p.z = (uint16)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cpx_b() {
|
auto R65816::op_cpx_b() {
|
||||||
int r = regs.x.l - rd.l;
|
int result = r.x.l - rd.l;
|
||||||
regs.p.n = r & 0x80;
|
r.p.n = result & 0x80;
|
||||||
regs.p.z = (uint8)r == 0;
|
r.p.z = (uint8)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cpx_w() {
|
auto R65816::op_cpx_w() {
|
||||||
int r = regs.x.w - rd.w;
|
int result = r.x.w - rd.w;
|
||||||
regs.p.n = r & 0x8000;
|
r.p.n = result & 0x8000;
|
||||||
regs.p.z = (uint16)r == 0;
|
r.p.z = (uint16)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cpy_b() {
|
auto R65816::op_cpy_b() {
|
||||||
int r = regs.y.l - rd.l;
|
int result = r.y.l - rd.l;
|
||||||
regs.p.n = r & 0x80;
|
r.p.n = result & 0x80;
|
||||||
regs.p.z = (uint8)r == 0;
|
r.p.z = (uint8)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_cpy_w() {
|
auto R65816::op_cpy_w() {
|
||||||
int r = regs.y.w - rd.w;
|
int result = r.y.w - rd.w;
|
||||||
regs.p.n = r & 0x8000;
|
r.p.n = result & 0x8000;
|
||||||
regs.p.z = (uint16)r == 0;
|
r.p.z = (uint16)result == 0;
|
||||||
regs.p.c = r >= 0;
|
r.p.c = result >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_eor_b() {
|
auto R65816::op_eor_b() {
|
||||||
regs.a.l ^= rd.l;
|
r.a.l ^= rd.l;
|
||||||
regs.p.n = regs.a.l & 0x80;
|
r.p.n = r.a.l & 0x80;
|
||||||
regs.p.z = regs.a.l == 0;
|
r.p.z = r.a.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_eor_w() {
|
auto R65816::op_eor_w() {
|
||||||
regs.a.w ^= rd.w;
|
r.a.w ^= rd.w;
|
||||||
regs.p.n = regs.a.w & 0x8000;
|
r.p.n = r.a.w & 0x8000;
|
||||||
regs.p.z = regs.a.w == 0;
|
r.p.z = r.a.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lda_b() {
|
auto R65816::op_lda_b() {
|
||||||
regs.a.l = rd.l;
|
r.a.l = rd.l;
|
||||||
regs.p.n = regs.a.l & 0x80;
|
r.p.n = r.a.l & 0x80;
|
||||||
regs.p.z = regs.a.l == 0;
|
r.p.z = r.a.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lda_w() {
|
auto R65816::op_lda_w() {
|
||||||
regs.a.w = rd.w;
|
r.a.w = rd.w;
|
||||||
regs.p.n = regs.a.w & 0x8000;
|
r.p.n = r.a.w & 0x8000;
|
||||||
regs.p.z = regs.a.w == 0;
|
r.p.z = r.a.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ldx_b() {
|
auto R65816::op_ldx_b() {
|
||||||
regs.x.l = rd.l;
|
r.x.l = rd.l;
|
||||||
regs.p.n = regs.x.l & 0x80;
|
r.p.n = r.x.l & 0x80;
|
||||||
regs.p.z = regs.x.l == 0;
|
r.p.z = r.x.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ldx_w() {
|
auto R65816::op_ldx_w() {
|
||||||
regs.x.w = rd.w;
|
r.x.w = rd.w;
|
||||||
regs.p.n = regs.x.w & 0x8000;
|
r.p.n = r.x.w & 0x8000;
|
||||||
regs.p.z = regs.x.w == 0;
|
r.p.z = r.x.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ldy_b() {
|
auto R65816::op_ldy_b() {
|
||||||
regs.y.l = rd.l;
|
r.y.l = rd.l;
|
||||||
regs.p.n = regs.y.l & 0x80;
|
r.p.n = r.y.l & 0x80;
|
||||||
regs.p.z = regs.y.l == 0;
|
r.p.z = r.y.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ldy_w() {
|
auto R65816::op_ldy_w() {
|
||||||
regs.y.w = rd.w;
|
r.y.w = rd.w;
|
||||||
regs.p.n = regs.y.w & 0x8000;
|
r.p.n = r.y.w & 0x8000;
|
||||||
regs.p.z = regs.y.w == 0;
|
r.p.z = r.y.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ora_b() {
|
auto R65816::op_ora_b() {
|
||||||
regs.a.l |= rd.l;
|
r.a.l |= rd.l;
|
||||||
regs.p.n = regs.a.l & 0x80;
|
r.p.n = r.a.l & 0x80;
|
||||||
regs.p.z = regs.a.l == 0;
|
r.p.z = r.a.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ora_w() {
|
auto R65816::op_ora_w() {
|
||||||
regs.a.w |= rd.w;
|
r.a.w |= rd.w;
|
||||||
regs.p.n = regs.a.w & 0x8000;
|
r.p.n = r.a.w & 0x8000;
|
||||||
regs.p.z = regs.a.w == 0;
|
r.p.z = r.a.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sbc_b() {
|
auto R65816::op_sbc_b() {
|
||||||
int result;
|
int result;
|
||||||
rd.l ^= 0xff;
|
rd.l ^= 0xff;
|
||||||
|
|
||||||
if(!regs.p.d) {
|
if(!r.p.d) {
|
||||||
result = regs.a.l + rd.l + regs.p.c;
|
result = r.a.l + rd.l + r.p.c;
|
||||||
} else {
|
} else {
|
||||||
result = (regs.a.l & 0x0f) + (rd.l & 0x0f) + (regs.p.c << 0);
|
result = (r.a.l & 0x0f) + (rd.l & 0x0f) + (r.p.c << 0);
|
||||||
if(result <= 0x0f) result -= 0x06;
|
if(result <= 0x0f) result -= 0x06;
|
||||||
regs.p.c = result > 0x0f;
|
r.p.c = result > 0x0f;
|
||||||
result = (regs.a.l & 0xf0) + (rd.l & 0xf0) + (regs.p.c << 4) + (result & 0x0f);
|
result = (r.a.l & 0xf0) + (rd.l & 0xf0) + (r.p.c << 4) + (result & 0x0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.p.v = ~(regs.a.l ^ rd.l) & (regs.a.l ^ result) & 0x80;
|
r.p.v = ~(r.a.l ^ rd.l) & (r.a.l ^ result) & 0x80;
|
||||||
if(regs.p.d && result <= 0xff) result -= 0x60;
|
if(r.p.d && result <= 0xff) result -= 0x60;
|
||||||
regs.p.c = result > 0xff;
|
r.p.c = result > 0xff;
|
||||||
regs.p.n = result & 0x80;
|
r.p.n = result & 0x80;
|
||||||
regs.p.z = (uint8_t)result == 0;
|
r.p.z = (uint8_t)result == 0;
|
||||||
|
|
||||||
regs.a.l = result;
|
r.a.l = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sbc_w() {
|
auto R65816::op_sbc_w() {
|
||||||
int result;
|
int result;
|
||||||
rd.w ^= 0xffff;
|
rd.w ^= 0xffff;
|
||||||
|
|
||||||
if(!regs.p.d) {
|
if(!r.p.d) {
|
||||||
result = regs.a.w + rd.w + regs.p.c;
|
result = r.a.w + rd.w + r.p.c;
|
||||||
} else {
|
} else {
|
||||||
result = (regs.a.w & 0x000f) + (rd.w & 0x000f) + (regs.p.c << 0);
|
result = (r.a.w & 0x000f) + (rd.w & 0x000f) + (r.p.c << 0);
|
||||||
if(result <= 0x000f) result -= 0x0006;
|
if(result <= 0x000f) result -= 0x0006;
|
||||||
regs.p.c = result > 0x000f;
|
r.p.c = result > 0x000f;
|
||||||
result = (regs.a.w & 0x00f0) + (rd.w & 0x00f0) + (regs.p.c << 4) + (result & 0x000f);
|
result = (r.a.w & 0x00f0) + (rd.w & 0x00f0) + (r.p.c << 4) + (result & 0x000f);
|
||||||
if(result <= 0x00ff) result -= 0x0060;
|
if(result <= 0x00ff) result -= 0x0060;
|
||||||
regs.p.c = result > 0x00ff;
|
r.p.c = result > 0x00ff;
|
||||||
result = (regs.a.w & 0x0f00) + (rd.w & 0x0f00) + (regs.p.c << 8) + (result & 0x00ff);
|
result = (r.a.w & 0x0f00) + (rd.w & 0x0f00) + (r.p.c << 8) + (result & 0x00ff);
|
||||||
if(result <= 0x0fff) result -= 0x0600;
|
if(result <= 0x0fff) result -= 0x0600;
|
||||||
regs.p.c = result > 0x0fff;
|
r.p.c = result > 0x0fff;
|
||||||
result = (regs.a.w & 0xf000) + (rd.w & 0xf000) + (regs.p.c << 12) + (result & 0x0fff);
|
result = (r.a.w & 0xf000) + (rd.w & 0xf000) + (r.p.c << 12) + (result & 0x0fff);
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.p.v = ~(regs.a.w ^ rd.w) & (regs.a.w ^ result) & 0x8000;
|
r.p.v = ~(r.a.w ^ rd.w) & (r.a.w ^ result) & 0x8000;
|
||||||
if(regs.p.d && result <= 0xffff) result -= 0x6000;
|
if(r.p.d && result <= 0xffff) result -= 0x6000;
|
||||||
regs.p.c = result > 0xffff;
|
r.p.c = result > 0xffff;
|
||||||
regs.p.n = result & 0x8000;
|
r.p.n = result & 0x8000;
|
||||||
regs.p.z = (uint16_t)result == 0;
|
r.p.z = (uint16_t)result == 0;
|
||||||
|
|
||||||
regs.a.w = result;
|
r.a.w = result;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_inc_b() {
|
auto R65816::op_inc_b() {
|
||||||
rd.l++;
|
rd.l++;
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_inc_w() {
|
auto R65816::op_inc_w() {
|
||||||
rd.w++;
|
rd.w++;
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_dec_b() {
|
auto R65816::op_dec_b() {
|
||||||
rd.l--;
|
rd.l--;
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_dec_w() {
|
auto R65816::op_dec_w() {
|
||||||
rd.w--;
|
rd.w--;
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_asl_b() {
|
auto R65816::op_asl_b() {
|
||||||
regs.p.c = rd.l & 0x80;
|
r.p.c = rd.l & 0x80;
|
||||||
rd.l <<= 1;
|
rd.l <<= 1;
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_asl_w() {
|
auto R65816::op_asl_w() {
|
||||||
regs.p.c = rd.w & 0x8000;
|
r.p.c = rd.w & 0x8000;
|
||||||
rd.w <<= 1;
|
rd.w <<= 1;
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lsr_b() {
|
auto R65816::op_lsr_b() {
|
||||||
regs.p.c = rd.l & 1;
|
r.p.c = rd.l & 1;
|
||||||
rd.l >>= 1;
|
rd.l >>= 1;
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lsr_w() {
|
auto R65816::op_lsr_w() {
|
||||||
regs.p.c = rd.w & 1;
|
r.p.c = rd.w & 1;
|
||||||
rd.w >>= 1;
|
rd.w >>= 1;
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rol_b() {
|
auto R65816::op_rol_b() {
|
||||||
unsigned carry = (unsigned)regs.p.c;
|
auto carry = (uint)r.p.c;
|
||||||
regs.p.c = rd.l & 0x80;
|
r.p.c = rd.l & 0x80;
|
||||||
rd.l = (rd.l << 1) | carry;
|
rd.l = (rd.l << 1) | carry;
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rol_w() {
|
auto R65816::op_rol_w() {
|
||||||
unsigned carry = (unsigned)regs.p.c;
|
auto carry = (uint)r.p.c;
|
||||||
regs.p.c = rd.w & 0x8000;
|
r.p.c = rd.w & 0x8000;
|
||||||
rd.w = (rd.w << 1) | carry;
|
rd.w = (rd.w << 1) | carry;
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ror_b() {
|
auto R65816::op_ror_b() {
|
||||||
unsigned carry = (unsigned)regs.p.c << 7;
|
auto carry = (uint)r.p.c << 7;
|
||||||
regs.p.c = rd.l & 1;
|
r.p.c = rd.l & 1;
|
||||||
rd.l = carry | (rd.l >> 1);
|
rd.l = carry | (rd.l >> 1);
|
||||||
regs.p.n = rd.l & 0x80;
|
r.p.n = rd.l & 0x80;
|
||||||
regs.p.z = rd.l == 0;
|
r.p.z = rd.l == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ror_w() {
|
auto R65816::op_ror_w() {
|
||||||
unsigned carry = (unsigned)regs.p.c << 15;
|
auto carry = (uint)r.p.c << 15;
|
||||||
regs.p.c = rd.w & 1;
|
r.p.c = rd.w & 1;
|
||||||
rd.w = carry | (rd.w >> 1);
|
rd.w = carry | (rd.w >> 1);
|
||||||
regs.p.n = rd.w & 0x8000;
|
r.p.n = rd.w & 0x8000;
|
||||||
regs.p.z = rd.w == 0;
|
r.p.z = rd.w == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_trb_b() {
|
auto R65816::op_trb_b() {
|
||||||
regs.p.z = (rd.l & regs.a.l) == 0;
|
r.p.z = (rd.l & r.a.l) == 0;
|
||||||
rd.l &= ~regs.a.l;
|
rd.l &= ~r.a.l;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_trb_w() {
|
auto R65816::op_trb_w() {
|
||||||
regs.p.z = (rd.w & regs.a.w) == 0;
|
r.p.z = (rd.w & r.a.w) == 0;
|
||||||
rd.w &= ~regs.a.w;
|
rd.w &= ~r.a.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_tsb_b() {
|
auto R65816::op_tsb_b() {
|
||||||
regs.p.z = (rd.l & regs.a.l) == 0;
|
r.p.z = (rd.l & r.a.l) == 0;
|
||||||
rd.l |= regs.a.l;
|
rd.l |= r.a.l;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_tsb_w() {
|
auto R65816::op_tsb_w() {
|
||||||
regs.p.z = (rd.w & regs.a.w) == 0;
|
r.p.z = (rd.w & r.a.w) == 0;
|
||||||
rd.w |= regs.a.w;
|
rd.w |= r.a.w;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,93 +23,93 @@ auto R65816::dreadl(uint24 addr) -> uint24 {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::decode(uint8 mode, uint24 addr) -> uint24 {
|
auto R65816::decode(uint8 mode, uint24 addr) -> uint24 {
|
||||||
uint24 r = 0;
|
uint24 a = 0;
|
||||||
|
|
||||||
switch(mode) {
|
switch(mode) {
|
||||||
case OPTYPE_DP:
|
case OPTYPE_DP:
|
||||||
r = (regs.d + (addr & 0xffff)) & 0xffff;
|
a = (r.d + (addr & 0xffff)) & 0xffff;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_DPX:
|
case OPTYPE_DPX:
|
||||||
r = (regs.d + regs.x + (addr & 0xffff)) & 0xffff;
|
a = (r.d + r.x + (addr & 0xffff)) & 0xffff;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_DPY:
|
case OPTYPE_DPY:
|
||||||
r = (regs.d + regs.y + (addr & 0xffff)) & 0xffff;
|
a = (r.d + r.y + (addr & 0xffff)) & 0xffff;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_IDP:
|
case OPTYPE_IDP:
|
||||||
addr = (regs.d + (addr & 0xffff)) & 0xffff;
|
addr = (r.d + (addr & 0xffff)) & 0xffff;
|
||||||
r = (regs.db << 16) + dreadw(addr);
|
a = (r.db << 16) + dreadw(addr);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_IDPX:
|
case OPTYPE_IDPX:
|
||||||
addr = (regs.d + regs.x + (addr & 0xffff)) & 0xffff;
|
addr = (r.d + r.x + (addr & 0xffff)) & 0xffff;
|
||||||
r = (regs.db << 16) + dreadw(addr);
|
a = (r.db << 16) + dreadw(addr);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_IDPY:
|
case OPTYPE_IDPY:
|
||||||
addr = (regs.d + (addr & 0xffff)) & 0xffff;
|
addr = (r.d + (addr & 0xffff)) & 0xffff;
|
||||||
r = (regs.db << 16) + dreadw(addr) + regs.y;
|
a = (r.db << 16) + dreadw(addr) + r.y;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ILDP:
|
case OPTYPE_ILDP:
|
||||||
addr = (regs.d + (addr & 0xffff)) & 0xffff;
|
addr = (r.d + (addr & 0xffff)) & 0xffff;
|
||||||
r = dreadl(addr);
|
a = dreadl(addr);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ILDPY:
|
case OPTYPE_ILDPY:
|
||||||
addr = (regs.d + (addr & 0xffff)) & 0xffff;
|
addr = (r.d + (addr & 0xffff)) & 0xffff;
|
||||||
r = dreadl(addr) + regs.y;
|
a = dreadl(addr) + r.y;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ADDR:
|
case OPTYPE_ADDR:
|
||||||
r = (regs.db << 16) + (addr & 0xffff);
|
a = (r.db << 16) + (addr & 0xffff);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ADDR_PC:
|
case OPTYPE_ADDR_PC:
|
||||||
r = (regs.pc.b << 16) + (addr & 0xffff);
|
a = (r.pc.b << 16) + (addr & 0xffff);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ADDRX:
|
case OPTYPE_ADDRX:
|
||||||
r = (regs.db << 16) + (addr & 0xffff) + regs.x;
|
a = (r.db << 16) + (addr & 0xffff) + r.x;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ADDRY:
|
case OPTYPE_ADDRY:
|
||||||
r = (regs.db << 16) + (addr & 0xffff) + regs.y;
|
a = (r.db << 16) + (addr & 0xffff) + r.y;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_IADDR_PC:
|
case OPTYPE_IADDR_PC:
|
||||||
r = (regs.pc.b << 16) + (addr & 0xffff);
|
a = (r.pc.b << 16) + (addr & 0xffff);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_IADDRX:
|
case OPTYPE_IADDRX:
|
||||||
r = (regs.pc.b << 16) + ((addr + regs.x) & 0xffff);
|
a = (r.pc.b << 16) + ((addr + r.x) & 0xffff);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ILADDR:
|
case OPTYPE_ILADDR:
|
||||||
r = addr;
|
a = addr;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_LONG:
|
case OPTYPE_LONG:
|
||||||
r = addr;
|
a = addr;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_LONGX:
|
case OPTYPE_LONGX:
|
||||||
r = (addr + regs.x);
|
a = (addr + r.x);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_SR:
|
case OPTYPE_SR:
|
||||||
r = (regs.s + (addr & 0xff)) & 0xffff;
|
a = (r.s + (addr & 0xff)) & 0xffff;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_ISRY:
|
case OPTYPE_ISRY:
|
||||||
addr = (regs.s + (addr & 0xff)) & 0xffff;
|
addr = (r.s + (addr & 0xff)) & 0xffff;
|
||||||
r = (regs.db << 16) + dreadw(addr) + regs.y;
|
a = (r.db << 16) + dreadw(addr) + r.y;
|
||||||
break;
|
break;
|
||||||
case OPTYPE_RELB:
|
case OPTYPE_RELB:
|
||||||
r = (regs.pc.b << 16) + ((regs.pc.w + 2) & 0xffff);
|
a = (r.pc.b << 16) + ((r.pc.w + 2) & 0xffff);
|
||||||
r += int8(addr);
|
a += int8(addr);
|
||||||
break;
|
break;
|
||||||
case OPTYPE_RELW:
|
case OPTYPE_RELW:
|
||||||
r = (regs.pc.b << 16) + ((regs.pc.w + 3) & 0xffff);
|
a = (r.pc.b << 16) + ((r.pc.w + 3) & 0xffff);
|
||||||
r += (int16)addr;
|
a += (int16)addr;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return r;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::disassemble() -> string {
|
auto R65816::disassemble() -> string {
|
||||||
return disassemble(regs.pc.d, regs.e, regs.p.m, regs.p.x);
|
return disassemble(r.pc.d, r.e, r.p.m, r.p.x);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::disassemble(uint24 addr, bool e, bool m, bool x) -> string {
|
auto R65816::disassemble(uint24 addr, bool e, bool m, bool x) -> string {
|
||||||
string s;
|
string s;
|
||||||
|
|
||||||
reg24_t pc;
|
reg24 pc;
|
||||||
pc.d = addr;
|
pc.d = addr;
|
||||||
s = {hex(pc, 6), " "};
|
s = {hex(pc, 6), " "};
|
||||||
|
|
||||||
|
@ -403,23 +403,23 @@ auto R65816::disassemble(uint24 addr, bool e, bool m, bool x) -> string {
|
||||||
#undef x8
|
#undef x8
|
||||||
|
|
||||||
s.append(t, " A:{0} X:{1} Y:{2} S:{3} D:{4} B:{5} ", format{
|
s.append(t, " A:{0} X:{1} Y:{2} S:{3} D:{4} B:{5} ", format{
|
||||||
hex(regs.a.w, 4), hex(regs.x.w, 4), hex(regs.y.w, 4),
|
hex(r.a.w, 4), hex(r.x.w, 4), hex(r.y.w, 4),
|
||||||
hex(regs.s.w, 4), hex(regs.d.w, 4), hex(regs.db, 2)
|
hex(r.s.w, 4), hex(r.d.w, 4), hex(r.db, 2)
|
||||||
});
|
});
|
||||||
|
|
||||||
if(regs.e) {
|
if(r.e) {
|
||||||
s.append(
|
s.append(
|
||||||
regs.p.n ? 'N' : 'n', regs.p.v ? 'V' : 'v',
|
r.p.n ? 'N' : 'n', r.p.v ? 'V' : 'v',
|
||||||
regs.p.m ? '1' : '0', regs.p.x ? 'B' : 'b',
|
r.p.m ? '1' : '0', r.p.x ? 'B' : 'b',
|
||||||
regs.p.d ? 'D' : 'd', regs.p.i ? 'I' : 'i',
|
r.p.d ? 'D' : 'd', r.p.i ? 'I' : 'i',
|
||||||
regs.p.z ? 'Z' : 'z', regs.p.c ? 'C' : 'c'
|
r.p.z ? 'Z' : 'z', r.p.c ? 'C' : 'c'
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
s.append(
|
s.append(
|
||||||
regs.p.n ? 'N' : 'n', regs.p.v ? 'V' : 'v',
|
r.p.n ? 'N' : 'n', r.p.v ? 'V' : 'v',
|
||||||
regs.p.m ? 'M' : 'm', regs.p.x ? 'X' : 'x',
|
r.p.m ? 'M' : 'm', r.p.x ? 'X' : 'x',
|
||||||
regs.p.d ? 'D' : 'd', regs.p.i ? 'I' : 'i',
|
r.p.d ? 'D' : 'd', r.p.i ? 'I' : 'i',
|
||||||
regs.p.z ? 'Z' : 'z', regs.p.c ? 'C' : 'c'
|
r.p.z ? 'Z' : 'z', r.p.c ? 'C' : 'c'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,77 +1,81 @@
|
||||||
alwaysinline auto readpc() -> uint8 {
|
alwaysinline auto readPC() -> uint8 {
|
||||||
return read((regs.pc.b << 16) + regs.pc.w++);
|
return read(r.pc.b << 16 | uint16(r.pc.w++));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readstack() -> uint8 {
|
alwaysinline auto readSP() -> uint8 {
|
||||||
regs.e ? regs.s.l++ : regs.s.w++;
|
r.e ? r.s.l++ : r.s.w++;
|
||||||
return read(regs.s.w);
|
return read(r.s.w);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readstackn() -> uint8 {
|
alwaysinline auto readSPn() -> uint8 {
|
||||||
return read(++regs.s.w);
|
return read(++r.s.w);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readaddr(uint32 addr) -> uint8 {
|
alwaysinline auto readAddr(uint addr) -> uint8 {
|
||||||
return read(addr & 0xffff);
|
return read(uint16(addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readlong(uint32 addr) -> uint8 {
|
alwaysinline auto readLong(uint addr) -> uint8 {
|
||||||
return read(addr & 0xffffff);
|
return read(uint24(addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readdbr(uint32 addr) -> uint8 {
|
alwaysinline auto readDB(uint addr) -> uint8 {
|
||||||
return read(((regs.db << 16) + addr) & 0xffffff);
|
return read(r.db << 16 | uint16(addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readpbr(uint32 addr) -> uint8 {
|
alwaysinline auto readPB(uint addr) -> uint8 {
|
||||||
return read((regs.pc.b << 16) + (addr & 0xffff));
|
return read(r.pc.b << 16 | uint16(addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readdp(uint32 addr) -> uint8 {
|
alwaysinline auto readDP(uint addr) -> uint8 {
|
||||||
if(regs.e && regs.d.l == 0x00) {
|
if(r.e && r.d.l == 0x00) {
|
||||||
return read((regs.d & 0xff00) + ((regs.d + (addr & 0xffff)) & 0xff));
|
return read(r.d | uint8(addr));
|
||||||
} else {
|
} else {
|
||||||
return read((regs.d + (addr & 0xffff)) & 0xffff);
|
return read(uint16(r.d + addr));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto readsp(uint32 addr) -> uint8 {
|
alwaysinline auto readDPn(uint addr) -> uint8 {
|
||||||
return read((regs.s + (addr & 0xffff)) & 0xffff);
|
return read(uint16(r.d + addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writestack(uint8 data) -> void {
|
alwaysinline auto readSP(uint addr) -> uint8 {
|
||||||
write(regs.s.w, data);
|
return read(uint16(r.s + addr));
|
||||||
regs.e ? regs.s.l-- : regs.s.w--;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writestackn(uint8 data) -> void {
|
alwaysinline auto writeSP(uint8 data) -> void {
|
||||||
write(regs.s.w--, data);
|
write(r.s.w, data);
|
||||||
|
r.e ? r.s.l-- : r.s.w--;
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writeaddr(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writeSPn(uint8 data) -> void {
|
||||||
write(addr & 0xffff, data);
|
write(r.s.w--, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writelong(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writeAddr(uint addr, uint8 data) -> void {
|
||||||
write(addr & 0xffffff, data);
|
write(uint16(addr), data);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writedbr(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writeLong(uint addr, uint8 data) -> void {
|
||||||
write(((regs.db << 16) + addr) & 0xffffff, data);
|
write(uint24(addr), data);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writepbr(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writeDB(uint addr, uint8 data) -> void {
|
||||||
write((regs.pc.b << 16) + (addr & 0xffff), data);
|
write(r.db << 16 | uint16(addr), data);
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writedp(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writePB(uint addr, uint8 data) -> void {
|
||||||
if(regs.e && regs.d.l == 0x00) {
|
write(r.pc.b << 16 | uint16(addr), data);
|
||||||
write((regs.d & 0xff00) + ((regs.d + (addr & 0xffff)) & 0xff), data);
|
}
|
||||||
|
|
||||||
|
alwaysinline auto writeDP(uint addr, uint8 data) -> void {
|
||||||
|
if(r.e && r.d.l == 0x00) {
|
||||||
|
write(r.d | uint8(addr), data);
|
||||||
} else {
|
} else {
|
||||||
write((regs.d + (addr & 0xffff)) & 0xffff, data);
|
write(uint16(r.d + addr), data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
alwaysinline auto writesp(uint32 addr, uint8 data) -> void {
|
alwaysinline auto writeSP(uint addr, uint8 data) -> void {
|
||||||
write((regs.s + (addr & 0xffff)) & 0xffff, data);
|
write(uint16(r.s + addr), data);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,81 +3,65 @@ L ioIRQ();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_wdm() {
|
auto R65816::op_wdm() {
|
||||||
L readpc();
|
L readPC();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_xba() {
|
auto R65816::op_xba() {
|
||||||
io();
|
io();
|
||||||
L io();
|
L io();
|
||||||
regs.a.l ^= regs.a.h;
|
swap(r.a.l, r.a.h);
|
||||||
regs.a.h ^= regs.a.l;
|
r.p.n = (r.a.l & 0x80);
|
||||||
regs.a.l ^= regs.a.h;
|
r.p.z = (r.a.l == 0);
|
||||||
regs.p.n = (regs.a.l & 0x80);
|
|
||||||
regs.p.z = (regs.a.l == 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_move_b(signed adjust) {
|
auto R65816::op_move_b(int adjust) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
regs.db = dp;
|
r.db = dp;
|
||||||
rd.l = readlong((sp << 16) | regs.x.w);
|
rd.l = readLong(sp << 16 | r.x.w);
|
||||||
writelong((dp << 16) | regs.y.w, rd.l);
|
writeLong(dp << 16 | r.y.w, rd.l);
|
||||||
io();
|
io();
|
||||||
regs.x.l += adjust;
|
r.x.l += adjust;
|
||||||
regs.y.l += adjust;
|
r.y.l += adjust;
|
||||||
L io();
|
L io();
|
||||||
if(regs.a.w--) regs.pc.w -= 3;
|
if(r.a.w--) r.pc.w -= 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_move_w(signed adjust) {
|
auto R65816::op_move_w(int adjust) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
regs.db = dp;
|
r.db = dp;
|
||||||
rd.l = readlong((sp << 16) | regs.x.w);
|
rd.l = readLong(sp << 16 | r.x.w);
|
||||||
writelong((dp << 16) | regs.y.w, rd.l);
|
writeLong(dp << 16 | r.y.w, rd.l);
|
||||||
io();
|
io();
|
||||||
regs.x.w += adjust;
|
r.x.w += adjust;
|
||||||
regs.y.w += adjust;
|
r.y.w += adjust;
|
||||||
L io();
|
L io();
|
||||||
if(regs.a.w--) regs.pc.w -= 3;
|
if(r.a.w--) r.pc.w -= 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_interrupt_e(uint16 vector) {
|
auto R65816::op_interrupt(uint16 vector) {
|
||||||
readpc();
|
readPC();
|
||||||
writestack(regs.pc.h);
|
N writeSP(r.pc.b);
|
||||||
writestack(regs.pc.l);
|
writeSP(r.pc.h);
|
||||||
writestack(regs.p);
|
writeSP(r.pc.l);
|
||||||
rd.l = readlong(vector + 0);
|
writeSP(r.p);
|
||||||
regs.pc.b = 0;
|
r.pc.l = readLong(vector + 0);
|
||||||
regs.p.i = 1;
|
r.p.i = 1;
|
||||||
regs.p.d = 0;
|
r.p.d = 0;
|
||||||
L rd.h = readlong(vector + 1);
|
L r.pc.h = readLong(vector + 1);
|
||||||
regs.pc.w = rd.w;
|
r.pc.b = 0x00;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_interrupt_n(uint16 vector) {
|
|
||||||
readpc();
|
|
||||||
writestack(regs.pc.b);
|
|
||||||
writestack(regs.pc.h);
|
|
||||||
writestack(regs.pc.l);
|
|
||||||
writestack(regs.p);
|
|
||||||
rd.l = readlong(vector + 0);
|
|
||||||
regs.pc.b = 0x00;
|
|
||||||
regs.p.i = 1;
|
|
||||||
regs.p.d = 0;
|
|
||||||
L rd.h = readlong(vector + 1);
|
|
||||||
regs.pc.w = rd.w;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_stp() {
|
auto R65816::op_stp() {
|
||||||
while(regs.wai = true) {
|
while(r.wai = true) {
|
||||||
L io();
|
L io();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_wai() {
|
auto R65816::op_wai() {
|
||||||
regs.wai = true;
|
r.wai = true;
|
||||||
while(regs.wai) {
|
while(r.wai) {
|
||||||
L io();
|
L io();
|
||||||
}
|
}
|
||||||
io();
|
io();
|
||||||
|
@ -85,15 +69,15 @@ L io();
|
||||||
|
|
||||||
auto R65816::op_xce() {
|
auto R65816::op_xce() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
bool carry = regs.p.c;
|
bool carry = r.p.c;
|
||||||
regs.p.c = regs.e;
|
r.p.c = r.e;
|
||||||
regs.e = carry;
|
r.e = carry;
|
||||||
if(regs.e) {
|
if(r.e) {
|
||||||
regs.p.m = 1;
|
r.p.m = 1;
|
||||||
regs.p.x = 1;
|
r.p.x = 1;
|
||||||
regs.x.h = 0x00;
|
r.x.h = 0x00;
|
||||||
regs.y.h = 0x00;
|
r.y.h = 0x00;
|
||||||
regs.s.h = 0x01;
|
r.s.h = 0x01;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,226 +86,160 @@ L ioIRQ();
|
||||||
flag = value;
|
flag = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pflag_e(bool mode) {
|
auto R65816::op_pflag(bool mode) {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
L io();
|
L io();
|
||||||
regs.p = (mode ? regs.p | rd.l : regs.p & ~rd.l);
|
r.p = (mode ? r.p | rd.l : r.p & ~rd.l);
|
||||||
regs.p.m = 1;
|
E r.p.m = 1, r.p.x = 1;
|
||||||
regs.p.x = 1;
|
if(r.p.x) {
|
||||||
regs.x.h = 0x00;
|
r.x.h = 0x00;
|
||||||
regs.y.h = 0x00;
|
r.y.h = 0x00;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_pflag_n(bool mode) {
|
|
||||||
rd.l = readpc();
|
|
||||||
L io();
|
|
||||||
regs.p = (mode ? regs.p | rd.l : regs.p & ~rd.l);
|
|
||||||
if(regs.p.x) {
|
|
||||||
regs.x.h = 0x00;
|
|
||||||
regs.y.h = 0x00;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_transfer_b(reg16_t& from, reg16_t& to) {
|
auto R65816::op_transfer_b(reg16& from, reg16& to) {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
to.l = from.l;
|
to.l = from.l;
|
||||||
regs.p.n = (to.l & 0x80);
|
r.p.n = (to.l & 0x80);
|
||||||
regs.p.z = (to.l == 0);
|
r.p.z = (to.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_transfer_w(reg16_t& from, reg16_t& to) {
|
auto R65816::op_transfer_w(reg16& from, reg16& to) {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
to.w = from.w;
|
to.w = from.w;
|
||||||
regs.p.n = (to.w & 0x8000);
|
r.p.n = (to.w & 0x8000);
|
||||||
regs.p.z = (to.w == 0);
|
r.p.z = (to.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_tcs_e() {
|
auto R65816::op_tcs() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.s.l = regs.a.l;
|
r.s.w = r.a.w;
|
||||||
}
|
E r.s.h = 0x01;
|
||||||
|
|
||||||
auto R65816::op_tcs_n() {
|
|
||||||
L ioIRQ();
|
|
||||||
regs.s.w = regs.a.w;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_tsx_b() {
|
auto R65816::op_tsx_b() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.x.l = regs.s.l;
|
r.x.l = r.s.l;
|
||||||
regs.p.n = (regs.x.l & 0x80);
|
r.p.n = (r.x.l & 0x80);
|
||||||
regs.p.z = (regs.x.l == 0);
|
r.p.z = (r.x.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_tsx_w() {
|
auto R65816::op_tsx_w() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.x.w = regs.s.w;
|
r.x.w = r.s.w;
|
||||||
regs.p.n = (regs.x.w & 0x8000);
|
r.p.n = (r.x.w & 0x8000);
|
||||||
regs.p.z = (regs.x.w == 0);
|
r.p.z = (r.x.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_txs_e() {
|
auto R65816::op_txs() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.s.l = regs.x.l;
|
E r.s.l = r.x.l;
|
||||||
|
N r.s.w = r.x.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_txs_n() {
|
auto R65816::op_push_b(reg16& reg) {
|
||||||
L ioIRQ();
|
|
||||||
regs.s.w = regs.x.w;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_push_b(reg16_t& reg) {
|
|
||||||
io();
|
io();
|
||||||
L writestack(reg.l);
|
L writeSP(reg.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_push_w(reg16_t& reg) {
|
auto R65816::op_push_w(reg16& reg) {
|
||||||
io();
|
io();
|
||||||
writestack(reg.h);
|
writeSP(reg.h);
|
||||||
L writestack(reg.l);
|
L writeSP(reg.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_phd_e() {
|
auto R65816::op_phd() {
|
||||||
io();
|
io();
|
||||||
writestackn(regs.d.h);
|
writeSPn(r.d.h);
|
||||||
L writestackn(regs.d.l);
|
L writeSPn(r.d.l);
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_phd_n() {
|
|
||||||
io();
|
|
||||||
writestackn(regs.d.h);
|
|
||||||
L writestackn(regs.d.l);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_phb() {
|
auto R65816::op_phb() {
|
||||||
io();
|
io();
|
||||||
L writestack(regs.db);
|
L writeSP(r.db);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_phk() {
|
auto R65816::op_phk() {
|
||||||
io();
|
io();
|
||||||
L writestack(regs.pc.b);
|
L writeSP(r.pc.b);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_php() {
|
auto R65816::op_php() {
|
||||||
io();
|
io();
|
||||||
L writestack(regs.p);
|
L writeSP(r.p);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pull_b(reg16_t& reg) {
|
auto R65816::op_pull_b(reg16& reg) {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
L reg.l = readstack();
|
L reg.l = readSP();
|
||||||
regs.p.n = (reg.l & 0x80);
|
r.p.n = (reg.l & 0x80);
|
||||||
regs.p.z = (reg.l == 0);
|
r.p.z = (reg.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pull_w(reg16_t& reg) {
|
auto R65816::op_pull_w(reg16& reg) {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
reg.l = readstack();
|
reg.l = readSP();
|
||||||
L reg.h = readstack();
|
L reg.h = readSP();
|
||||||
regs.p.n = (reg.w & 0x8000);
|
r.p.n = (reg.w & 0x8000);
|
||||||
regs.p.z = (reg.w == 0);
|
r.p.z = (reg.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pld_e() {
|
auto R65816::op_pld() {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
regs.d.l = readstackn();
|
r.d.l = readSPn();
|
||||||
L regs.d.h = readstackn();
|
L r.d.h = readSPn();
|
||||||
regs.p.n = (regs.d.w & 0x8000);
|
r.p.n = (r.d.w & 0x8000);
|
||||||
regs.p.z = (regs.d.w == 0);
|
r.p.z = (r.d.w == 0);
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_pld_n() {
|
|
||||||
io();
|
|
||||||
io();
|
|
||||||
regs.d.l = readstackn();
|
|
||||||
L regs.d.h = readstackn();
|
|
||||||
regs.p.n = (regs.d.w & 0x8000);
|
|
||||||
regs.p.z = (regs.d.w == 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_plb() {
|
auto R65816::op_plb() {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
L regs.db = readstack();
|
L r.db = readSP();
|
||||||
regs.p.n = (regs.db & 0x80);
|
r.p.n = (r.db & 0x80);
|
||||||
regs.p.z = (regs.db == 0);
|
r.p.z = (r.db == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_plp_e() {
|
auto R65816::op_plp() {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
L regs.p = readstack() | 0x30;
|
L r.p = readSP();
|
||||||
if(regs.p.x) {
|
E r.p.m = 1, r.p.x = 1;
|
||||||
regs.x.h = 0x00;
|
if(r.p.x) {
|
||||||
regs.y.h = 0x00;
|
r.x.h = 0x00;
|
||||||
|
r.y.h = 0x00;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_plp_n() {
|
auto R65816::op_pea() {
|
||||||
io();
|
aa.l = readPC();
|
||||||
io();
|
aa.h = readPC();
|
||||||
L regs.p = readstack();
|
writeSPn(aa.h);
|
||||||
if(regs.p.x) {
|
L writeSPn(aa.l);
|
||||||
regs.x.h = 0x00;
|
E r.s.h = 0x01;
|
||||||
regs.y.h = 0x00;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pea_e() {
|
auto R65816::op_pei() {
|
||||||
aa.l = readpc();
|
dp = readPC();
|
||||||
aa.h = readpc();
|
|
||||||
writestackn(aa.h);
|
|
||||||
L writestackn(aa.l);
|
|
||||||
regs.s.h = 0x01;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_pea_n() {
|
|
||||||
aa.l = readpc();
|
|
||||||
aa.h = readpc();
|
|
||||||
writestackn(aa.h);
|
|
||||||
L writestackn(aa.l);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_pei_e() {
|
|
||||||
dp = readpc();
|
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
writestackn(aa.h);
|
writeSPn(aa.h);
|
||||||
L writestackn(aa.l);
|
L writeSPn(aa.l);
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_pei_n() {
|
auto R65816::op_per() {
|
||||||
dp = readpc();
|
aa.l = readPC();
|
||||||
io2();
|
aa.h = readPC();
|
||||||
aa.l = readdp(dp + 0);
|
|
||||||
aa.h = readdp(dp + 1);
|
|
||||||
writestackn(aa.h);
|
|
||||||
L writestackn(aa.l);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_per_e() {
|
|
||||||
aa.l = readpc();
|
|
||||||
aa.h = readpc();
|
|
||||||
io();
|
io();
|
||||||
rd.w = regs.pc.d + (int16)aa.w;
|
rd.w = r.pc.d + (int16)aa.w;
|
||||||
writestackn(rd.h);
|
writeSPn(rd.h);
|
||||||
L writestackn(rd.l);
|
L writeSPn(rd.l);
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_per_n() {
|
|
||||||
aa.l = readpc();
|
|
||||||
aa.h = readpc();
|
|
||||||
io();
|
|
||||||
rd.w = regs.pc.d + (int16)aa.w;
|
|
||||||
writestackn(rd.h);
|
|
||||||
L writestackn(rd.l);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,176 +1,138 @@
|
||||||
auto R65816::op_branch(bool flag, bool value) {
|
auto R65816::op_branch(bool flag, bool value) {
|
||||||
if(flag != value) {
|
if(flag != value) {
|
||||||
L rd.l = readpc();
|
L rd.l = readPC();
|
||||||
} else {
|
} else {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
aa.w = regs.pc.d + (int8)rd.l;
|
aa.w = r.pc.d + (int8)rd.l;
|
||||||
io6(aa.w);
|
io6(aa.w);
|
||||||
L io();
|
L io();
|
||||||
regs.pc.w = aa.w;
|
r.pc.w = aa.w;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_bra() {
|
auto R65816::op_bra() {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
aa.w = regs.pc.d + (int8)rd.l;
|
aa.w = r.pc.d + (int8)rd.l;
|
||||||
io6(aa.w);
|
io6(aa.w);
|
||||||
L io();
|
L io();
|
||||||
regs.pc.w = aa.w;
|
r.pc.w = aa.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_brl() {
|
auto R65816::op_brl() {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
rd.h = readpc();
|
rd.h = readPC();
|
||||||
L io();
|
L io();
|
||||||
regs.pc.w = regs.pc.d + (int16)rd.w;
|
r.pc.w = r.pc.d + (int16)rd.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jmp_addr() {
|
auto R65816::op_jmp_addr() {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
L rd.h = readpc();
|
L rd.h = readPC();
|
||||||
regs.pc.w = rd.w;
|
r.pc.w = rd.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jmp_long() {
|
auto R65816::op_jmp_long() {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
rd.h = readpc();
|
rd.h = readPC();
|
||||||
L rd.b = readpc();
|
L rd.b = readPC();
|
||||||
regs.pc.d = rd.d & 0xffffff;
|
r.pc.d = rd.d;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jmp_iaddr() {
|
auto R65816::op_jmp_iaddr() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
rd.l = readaddr(aa.w + 0);
|
rd.l = readAddr(aa.w + 0);
|
||||||
L rd.h = readaddr(aa.w + 1);
|
L rd.h = readAddr(aa.w + 1);
|
||||||
regs.pc.w = rd.w;
|
r.pc.w = rd.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jmp_iaddrx() {
|
auto R65816::op_jmp_iaddrx() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
rd.l = readpbr(aa.w + regs.x.w + 0);
|
rd.l = readPB(aa.w + r.x.w + 0);
|
||||||
L rd.h = readpbr(aa.w + regs.x.w + 1);
|
L rd.h = readPB(aa.w + r.x.w + 1);
|
||||||
regs.pc.w = rd.w;
|
r.pc.w = rd.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jmp_iladdr() {
|
auto R65816::op_jmp_iladdr() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
rd.l = readaddr(aa.w + 0);
|
rd.l = readAddr(aa.w + 0);
|
||||||
rd.h = readaddr(aa.w + 1);
|
rd.h = readAddr(aa.w + 1);
|
||||||
L rd.b = readaddr(aa.w + 2);
|
L rd.b = readAddr(aa.w + 2);
|
||||||
regs.pc.d = rd.d & 0xffffff;
|
r.pc.d = rd.d;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jsr_addr() {
|
auto R65816::op_jsr_addr() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
regs.pc.w--;
|
r.pc.w--;
|
||||||
writestack(regs.pc.h);
|
writeSP(r.pc.h);
|
||||||
L writestack(regs.pc.l);
|
L writeSP(r.pc.l);
|
||||||
regs.pc.w = aa.w;
|
r.pc.w = aa.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jsr_long_e() {
|
auto R65816::op_jsr_long() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
writestackn(regs.pc.b);
|
writeSPn(r.pc.b);
|
||||||
io();
|
io();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
regs.pc.w--;
|
r.pc.w--;
|
||||||
writestackn(regs.pc.h);
|
writeSPn(r.pc.h);
|
||||||
L writestackn(regs.pc.l);
|
L writeSPn(r.pc.l);
|
||||||
regs.pc.d = aa.d & 0xffffff;
|
r.pc.d = aa.d;
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jsr_long_n() {
|
auto R65816::op_jsr_iaddrx() {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
writeSPn(r.pc.h);
|
||||||
writestackn(regs.pc.b);
|
writeSPn(r.pc.l);
|
||||||
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
aa.b = readpc();
|
rd.l = readPB(aa.w + r.x.w + 0);
|
||||||
regs.pc.w--;
|
L rd.h = readPB(aa.w + r.x.w + 1);
|
||||||
writestackn(regs.pc.h);
|
r.pc.w = rd.w;
|
||||||
L writestackn(regs.pc.l);
|
E r.s.h = 0x01;
|
||||||
regs.pc.d = aa.d & 0xffffff;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_jsr_iaddrx_e() {
|
auto R65816::op_rti() {
|
||||||
aa.l = readpc();
|
|
||||||
writestackn(regs.pc.h);
|
|
||||||
writestackn(regs.pc.l);
|
|
||||||
aa.h = readpc();
|
|
||||||
io();
|
|
||||||
rd.l = readpbr(aa.w + regs.x.w + 0);
|
|
||||||
L rd.h = readpbr(aa.w + regs.x.w + 1);
|
|
||||||
regs.pc.w = rd.w;
|
|
||||||
regs.s.h = 0x01;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_jsr_iaddrx_n() {
|
|
||||||
aa.l = readpc();
|
|
||||||
writestackn(regs.pc.h);
|
|
||||||
writestackn(regs.pc.l);
|
|
||||||
aa.h = readpc();
|
|
||||||
io();
|
|
||||||
rd.l = readpbr(aa.w + regs.x.w + 0);
|
|
||||||
L rd.h = readpbr(aa.w + regs.x.w + 1);
|
|
||||||
regs.pc.w = rd.w;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_rti_e() {
|
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
regs.p = readstack() | 0x30;
|
r.p = readSP();
|
||||||
rd.l = readstack();
|
E r.p.m = 1, r.p.x = 1;
|
||||||
L rd.h = readstack();
|
if(r.p.x) {
|
||||||
regs.pc.w = rd.w;
|
r.x.h = 0x00;
|
||||||
}
|
r.y.h = 0x00;
|
||||||
|
}
|
||||||
auto R65816::op_rti_n() {
|
r.pc.l = readSP();
|
||||||
io();
|
if(r.e) {
|
||||||
io();
|
L r.pc.h = readSP();
|
||||||
regs.p = readstack();
|
} else {
|
||||||
if(regs.p.x) {
|
r.pc.h = readSP();
|
||||||
regs.x.h = 0x00;
|
L r.pc.b = readSP();
|
||||||
regs.y.h = 0x00;
|
|
||||||
}
|
}
|
||||||
rd.l = readstack();
|
|
||||||
rd.h = readstack();
|
|
||||||
L rd.b = readstack();
|
|
||||||
regs.pc.d = rd.d & 0xffffff;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rts() {
|
auto R65816::op_rts() {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
rd.l = readstack();
|
rd.l = readSP();
|
||||||
rd.h = readstack();
|
rd.h = readSP();
|
||||||
L io();
|
L io();
|
||||||
regs.pc.w = ++rd.w;
|
r.pc.w = ++rd.w;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rtl_e() {
|
auto R65816::op_rtl() {
|
||||||
io();
|
io();
|
||||||
io();
|
io();
|
||||||
rd.l = readstackn();
|
rd.l = readSPn();
|
||||||
rd.h = readstackn();
|
rd.h = readSPn();
|
||||||
L rd.b = readstackn();
|
L rd.b = readSPn();
|
||||||
regs.pc.b = rd.b;
|
r.pc.b = rd.b;
|
||||||
regs.pc.w = ++rd.w;
|
r.pc.w = ++rd.w;
|
||||||
regs.s.h = 0x01;
|
E r.s.h = 0x01;
|
||||||
}
|
|
||||||
|
|
||||||
auto R65816::op_rtl_n() {
|
|
||||||
io();
|
|
||||||
io();
|
|
||||||
rd.l = readstackn();
|
|
||||||
rd.h = readstackn();
|
|
||||||
L rd.b = readstackn();
|
|
||||||
regs.pc.b = rd.b;
|
|
||||||
regs.pc.w = ++rd.w;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,275 +1,275 @@
|
||||||
auto R65816::op_read_const_b(fp op) {
|
auto R65816::op_read_const_b(fp op) {
|
||||||
L rd.l = readpc();
|
L rd.l = readPC();
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_const_w(fp op) {
|
auto R65816::op_read_const_w(fp op) {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
L rd.h = readpc();
|
L rd.h = readPC();
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_bit_const_b() {
|
auto R65816::op_read_bit_const_b() {
|
||||||
L rd.l = readpc();
|
L rd.l = readPC();
|
||||||
regs.p.z = ((rd.l & regs.a.l) == 0);
|
r.p.z = ((rd.l & r.a.l) == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_bit_const_w() {
|
auto R65816::op_read_bit_const_w() {
|
||||||
rd.l = readpc();
|
rd.l = readPC();
|
||||||
L rd.h = readpc();
|
L rd.h = readPC();
|
||||||
regs.p.z = ((rd.w & regs.a.w) == 0);
|
r.p.z = ((rd.w & r.a.w) == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addr_b(fp op) {
|
auto R65816::op_read_addr_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
L rd.l = readdbr(aa.w);
|
L rd.l = readDB(aa.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addr_w(fp op) {
|
auto R65816::op_read_addr_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
rd.l = readdbr(aa.w + 0);
|
rd.l = readDB(aa.w + 0);
|
||||||
L rd.h = readdbr(aa.w + 1);
|
L rd.h = readDB(aa.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addrx_b(fp op) {
|
auto R65816::op_read_addrx_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io4(aa.w, aa.w + regs.x.w);
|
io4(aa.w, aa.w + r.x.w);
|
||||||
L rd.l = readdbr(aa.w + regs.x.w);
|
L rd.l = readDB(aa.w + r.x.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addrx_w(fp op) {
|
auto R65816::op_read_addrx_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io4(aa.w, aa.w + regs.x.w);
|
io4(aa.w, aa.w + r.x.w);
|
||||||
rd.l = readdbr(aa.w + regs.x.w + 0);
|
rd.l = readDB(aa.w + r.x.w + 0);
|
||||||
L rd.h = readdbr(aa.w + regs.x.w + 1);
|
L rd.h = readDB(aa.w + r.x.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addry_b(fp op) {
|
auto R65816::op_read_addry_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io4(aa.w, aa.w + regs.y.w);
|
io4(aa.w, aa.w + r.y.w);
|
||||||
L rd.l = readdbr(aa.w + regs.y.w);
|
L rd.l = readDB(aa.w + r.y.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_addry_w(fp op) {
|
auto R65816::op_read_addry_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io4(aa.w, aa.w + regs.y.w);
|
io4(aa.w, aa.w + r.y.w);
|
||||||
rd.l = readdbr(aa.w + regs.y.w + 0);
|
rd.l = readDB(aa.w + r.y.w + 0);
|
||||||
L rd.h = readdbr(aa.w + regs.y.w + 1);
|
L rd.h = readDB(aa.w + r.y.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_long_b(fp op) {
|
auto R65816::op_read_long_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
L rd.l = readlong(aa.d);
|
L rd.l = readLong(aa.d);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_long_w(fp op) {
|
auto R65816::op_read_long_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
rd.l = readlong(aa.d + 0);
|
rd.l = readLong(aa.d + 0);
|
||||||
L rd.h = readlong(aa.d + 1);
|
L rd.h = readLong(aa.d + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_longx_b(fp op) {
|
auto R65816::op_read_longx_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
L rd.l = readlong(aa.d + regs.x.w);
|
L rd.l = readLong(aa.d + r.x.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_longx_w(fp op) {
|
auto R65816::op_read_longx_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
rd.l = readlong(aa.d + regs.x.w + 0);
|
rd.l = readLong(aa.d + r.x.w + 0);
|
||||||
L rd.h = readlong(aa.d + regs.x.w + 1);
|
L rd.h = readLong(aa.d + r.x.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_dp_b(fp op) {
|
auto R65816::op_read_dp_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
L rd.l = readdp(dp);
|
L rd.l = readDP(dp);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_dp_w(fp op) {
|
auto R65816::op_read_dp_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
rd.l = readdp(dp + 0);
|
rd.l = readDP(dp + 0);
|
||||||
L rd.h = readdp(dp + 1);
|
L rd.h = readDP(dp + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_dpr_b(fp op, reg16_t& reg) {
|
auto R65816::op_read_dpr_b(fp op, reg16& reg) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
L rd.l = readdp(dp + reg.w);
|
L rd.l = readDP(dp + reg.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_dpr_w(fp op, reg16_t& reg) {
|
auto R65816::op_read_dpr_w(fp op, reg16& reg) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
rd.l = readdp(dp + reg.w + 0);
|
rd.l = readDP(dp + reg.w + 0);
|
||||||
L rd.h = readdp(dp + reg.w + 1);
|
L rd.h = readDP(dp + reg.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idp_b(fp op) {
|
auto R65816::op_read_idp_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
L rd.l = readdbr(aa.w);
|
L rd.l = readDB(aa.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idp_w(fp op) {
|
auto R65816::op_read_idp_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
rd.l = readdbr(aa.w + 0);
|
rd.l = readDB(aa.w + 0);
|
||||||
L rd.h = readdbr(aa.w + 1);
|
L rd.h = readDB(aa.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idpx_b(fp op) {
|
auto R65816::op_read_idpx_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
aa.l = readdp(dp + regs.x.w + 0);
|
aa.l = readDP(dp + r.x.w + 0);
|
||||||
aa.h = readdp(dp + regs.x.w + 1);
|
aa.h = readDP(dp + r.x.w + 1);
|
||||||
L rd.l = readdbr(aa.w);
|
L rd.l = readDB(aa.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idpx_w(fp op) {
|
auto R65816::op_read_idpx_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
aa.l = readdp(dp + regs.x.w + 0);
|
aa.l = readDP(dp + r.x.w + 0);
|
||||||
aa.h = readdp(dp + regs.x.w + 1);
|
aa.h = readDP(dp + r.x.w + 1);
|
||||||
rd.l = readdbr(aa.w + 0);
|
rd.l = readDB(aa.w + 0);
|
||||||
L rd.h = readdbr(aa.w + 1);
|
L rd.h = readDB(aa.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idpy_b(fp op) {
|
auto R65816::op_read_idpy_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
io4(aa.w, aa.w + regs.y.w);
|
io4(aa.w, aa.w + r.y.w);
|
||||||
L rd.l = readdbr(aa.w + regs.y.w);
|
L rd.l = readDB(aa.w + r.y.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_idpy_w(fp op) {
|
auto R65816::op_read_idpy_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
io4(aa.w, aa.w + regs.y.w);
|
io4(aa.w, aa.w + r.y.w);
|
||||||
rd.l = readdbr(aa.w + regs.y.w + 0);
|
rd.l = readDB(aa.w + r.y.w + 0);
|
||||||
L rd.h = readdbr(aa.w + regs.y.w + 1);
|
L rd.h = readDB(aa.w + r.y.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_ildp_b(fp op) {
|
auto R65816::op_read_ildp_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
L rd.l = readlong(aa.d);
|
L rd.l = readLong(aa.d);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_ildp_w(fp op) {
|
auto R65816::op_read_ildp_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
rd.l = readlong(aa.d + 0);
|
rd.l = readLong(aa.d + 0);
|
||||||
L rd.h = readlong(aa.d + 1);
|
L rd.h = readLong(aa.d + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_ildpy_b(fp op) {
|
auto R65816::op_read_ildpy_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
L rd.l = readlong(aa.d + regs.y.w);
|
L rd.l = readLong(aa.d + r.y.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_ildpy_w(fp op) {
|
auto R65816::op_read_ildpy_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
rd.l = readlong(aa.d + regs.y.w + 0);
|
rd.l = readLong(aa.d + r.y.w + 0);
|
||||||
L rd.h = readlong(aa.d + regs.y.w + 1);
|
L rd.h = readLong(aa.d + r.y.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_sr_b(fp op) {
|
auto R65816::op_read_sr_b(fp op) {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
L rd.l = readsp(sp);
|
L rd.l = readSP(sp);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_sr_w(fp op) {
|
auto R65816::op_read_sr_w(fp op) {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
rd.l = readsp(sp + 0);
|
rd.l = readSP(sp + 0);
|
||||||
L rd.h = readsp(sp + 1);
|
L rd.h = readSP(sp + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_isry_b(fp op) {
|
auto R65816::op_read_isry_b(fp op) {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
aa.l = readsp(sp + 0);
|
aa.l = readSP(sp + 0);
|
||||||
aa.h = readsp(sp + 1);
|
aa.h = readSP(sp + 1);
|
||||||
io();
|
io();
|
||||||
L rd.l = readdbr(aa.w + regs.y.w);
|
L rd.l = readDB(aa.w + r.y.w);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_read_isry_w(fp op) {
|
auto R65816::op_read_isry_w(fp op) {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
aa.l = readsp(sp + 0);
|
aa.l = readSP(sp + 0);
|
||||||
aa.h = readsp(sp + 1);
|
aa.h = readSP(sp + 1);
|
||||||
io();
|
io();
|
||||||
rd.l = readdbr(aa.w + regs.y.w + 0);
|
rd.l = readDB(aa.w + r.y.w + 0);
|
||||||
L rd.h = readdbr(aa.w + regs.y.w + 1);
|
L rd.h = readDB(aa.w + r.y.w + 1);
|
||||||
call(op);
|
call(op);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,165 +1,165 @@
|
||||||
auto R65816::op_adjust_imm_b(reg16_t& reg, signed adjust) {
|
auto R65816::op_adjust_imm_b(reg16& reg, int adjust) {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
reg.l += adjust;
|
reg.l += adjust;
|
||||||
regs.p.n = (reg.l & 0x80);
|
r.p.n = (reg.l & 0x80);
|
||||||
regs.p.z = (reg.l == 0);
|
r.p.z = (reg.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_imm_w(reg16_t& reg, signed adjust) {
|
auto R65816::op_adjust_imm_w(reg16& reg, int adjust) {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
reg.w += adjust;
|
reg.w += adjust;
|
||||||
regs.p.n = (reg.w & 0x8000);
|
r.p.n = (reg.w & 0x8000);
|
||||||
regs.p.z = (reg.w == 0);
|
r.p.z = (reg.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_asl_imm_b() {
|
auto R65816::op_asl_imm_b() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.p.c = (regs.a.l & 0x80);
|
r.p.c = (r.a.l & 0x80);
|
||||||
regs.a.l <<= 1;
|
r.a.l <<= 1;
|
||||||
regs.p.n = (regs.a.l & 0x80);
|
r.p.n = (r.a.l & 0x80);
|
||||||
regs.p.z = (regs.a.l == 0);
|
r.p.z = (r.a.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_asl_imm_w() {
|
auto R65816::op_asl_imm_w() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.p.c = (regs.a.w & 0x8000);
|
r.p.c = (r.a.w & 0x8000);
|
||||||
regs.a.w <<= 1;
|
r.a.w <<= 1;
|
||||||
regs.p.n = (regs.a.w & 0x8000);
|
r.p.n = (r.a.w & 0x8000);
|
||||||
regs.p.z = (regs.a.w == 0);
|
r.p.z = (r.a.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lsr_imm_b() {
|
auto R65816::op_lsr_imm_b() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.p.c = (regs.a.l & 0x01);
|
r.p.c = (r.a.l & 0x01);
|
||||||
regs.a.l >>= 1;
|
r.a.l >>= 1;
|
||||||
regs.p.n = (regs.a.l & 0x80);
|
r.p.n = (r.a.l & 0x80);
|
||||||
regs.p.z = (regs.a.l == 0);
|
r.p.z = (r.a.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_lsr_imm_w() {
|
auto R65816::op_lsr_imm_w() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
regs.p.c = (regs.a.w & 0x0001);
|
r.p.c = (r.a.w & 0x0001);
|
||||||
regs.a.w >>= 1;
|
r.a.w >>= 1;
|
||||||
regs.p.n = (regs.a.w & 0x8000);
|
r.p.n = (r.a.w & 0x8000);
|
||||||
regs.p.z = (regs.a.w == 0);
|
r.p.z = (r.a.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rol_imm_b() {
|
auto R65816::op_rol_imm_b() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
bool carry = regs.p.c;
|
bool carry = r.p.c;
|
||||||
regs.p.c = (regs.a.l & 0x80);
|
r.p.c = (r.a.l & 0x80);
|
||||||
regs.a.l = (regs.a.l << 1) | carry;
|
r.a.l = (r.a.l << 1) | carry;
|
||||||
regs.p.n = (regs.a.l & 0x80);
|
r.p.n = (r.a.l & 0x80);
|
||||||
regs.p.z = (regs.a.l == 0);
|
r.p.z = (r.a.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_rol_imm_w() {
|
auto R65816::op_rol_imm_w() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
bool carry = regs.p.c;
|
bool carry = r.p.c;
|
||||||
regs.p.c = (regs.a.w & 0x8000);
|
r.p.c = (r.a.w & 0x8000);
|
||||||
regs.a.w = (regs.a.w << 1) | carry;
|
r.a.w = (r.a.w << 1) | carry;
|
||||||
regs.p.n = (regs.a.w & 0x8000);
|
r.p.n = (r.a.w & 0x8000);
|
||||||
regs.p.z = (regs.a.w == 0);
|
r.p.z = (r.a.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ror_imm_b() {
|
auto R65816::op_ror_imm_b() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
bool carry = regs.p.c;
|
bool carry = r.p.c;
|
||||||
regs.p.c = (regs.a.l & 0x01);
|
r.p.c = (r.a.l & 0x01);
|
||||||
regs.a.l = (carry << 7) | (regs.a.l >> 1);
|
r.a.l = (carry << 7) | (r.a.l >> 1);
|
||||||
regs.p.n = (regs.a.l & 0x80);
|
r.p.n = (r.a.l & 0x80);
|
||||||
regs.p.z = (regs.a.l == 0);
|
r.p.z = (r.a.l == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_ror_imm_w() {
|
auto R65816::op_ror_imm_w() {
|
||||||
L ioIRQ();
|
L ioIRQ();
|
||||||
bool carry = regs.p.c;
|
bool carry = r.p.c;
|
||||||
regs.p.c = (regs.a.w & 0x0001);
|
r.p.c = (r.a.w & 0x0001);
|
||||||
regs.a.w = (carry << 15) | (regs.a.w >> 1);
|
r.a.w = (carry << 15) | (r.a.w >> 1);
|
||||||
regs.p.n = (regs.a.w & 0x8000);
|
r.p.n = (r.a.w & 0x8000);
|
||||||
regs.p.z = (regs.a.w == 0);
|
r.p.z = (r.a.w == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_addr_b(fp op) {
|
auto R65816::op_adjust_addr_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
rd.l = readdbr(aa.w);
|
rd.l = readDB(aa.w);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
L writedbr(aa.w, rd.l);
|
L writeDB(aa.w, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_addr_w(fp op) {
|
auto R65816::op_adjust_addr_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
rd.l = readdbr(aa.w + 0);
|
rd.l = readDB(aa.w + 0);
|
||||||
rd.h = readdbr(aa.w + 1);
|
rd.h = readDB(aa.w + 1);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
writedbr(aa.w + 1, rd.h);
|
writeDB(aa.w + 1, rd.h);
|
||||||
L writedbr(aa.w + 0, rd.l);
|
L writeDB(aa.w + 0, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_addrx_b(fp op) {
|
auto R65816::op_adjust_addrx_b(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
rd.l = readdbr(aa.w + regs.x.w);
|
rd.l = readDB(aa.w + r.x.w);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
L writedbr(aa.w + regs.x.w, rd.l);
|
L writeDB(aa.w + r.x.w, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_addrx_w(fp op) {
|
auto R65816::op_adjust_addrx_w(fp op) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
rd.l = readdbr(aa.w + regs.x.w + 0);
|
rd.l = readDB(aa.w + r.x.w + 0);
|
||||||
rd.h = readdbr(aa.w + regs.x.w + 1);
|
rd.h = readDB(aa.w + r.x.w + 1);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
writedbr(aa.w + regs.x.w + 1, rd.h);
|
writeDB(aa.w + r.x.w + 1, rd.h);
|
||||||
L writedbr(aa.w + regs.x.w + 0, rd.l);
|
L writeDB(aa.w + r.x.w + 0, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_dp_b(fp op) {
|
auto R65816::op_adjust_dp_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
rd.l = readdp(dp);
|
rd.l = readDP(dp);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
L writedp(dp, rd.l);
|
L writeDP(dp, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_dp_w(fp op) {
|
auto R65816::op_adjust_dp_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
rd.l = readdp(dp + 0);
|
rd.l = readDP(dp + 0);
|
||||||
rd.h = readdp(dp + 1);
|
rd.h = readDP(dp + 1);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
writedp(dp + 1, rd.h);
|
writeDP(dp + 1, rd.h);
|
||||||
L writedp(dp + 0, rd.l);
|
L writeDP(dp + 0, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_dpx_b(fp op) {
|
auto R65816::op_adjust_dpx_b(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
rd.l = readdp(dp + regs.x.w);
|
rd.l = readDP(dp + r.x.w);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
L writedp(dp + regs.x.w, rd.l);
|
L writeDP(dp + r.x.w, rd.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_adjust_dpx_w(fp op) {
|
auto R65816::op_adjust_dpx_w(fp op) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
rd.l = readdp(dp + regs.x.w + 0);
|
rd.l = readDP(dp + r.x.w + 0);
|
||||||
rd.h = readdp(dp + regs.x.w + 1);
|
rd.h = readDP(dp + r.x.w + 1);
|
||||||
io();
|
io();
|
||||||
call(op);
|
call(op);
|
||||||
writedp(dp + regs.x.w + 1, rd.h);
|
writeDP(dp + r.x.w + 1, rd.h);
|
||||||
L writedp(dp + regs.x.w + 0, rd.l);
|
L writeDP(dp + r.x.w + 0, rd.l);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,195 +1,195 @@
|
||||||
auto R65816::op_write_addr_b(reg16_t& reg) {
|
auto R65816::op_write_addr_b(reg16& reg) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
L writedbr(aa.w, reg);
|
L writeDB(aa.w, reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_addr_w(reg16_t& reg) {
|
auto R65816::op_write_addr_w(reg16& reg) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
writedbr(aa.w + 0, reg >> 0);
|
writeDB(aa.w + 0, reg >> 0);
|
||||||
L writedbr(aa.w + 1, reg >> 8);
|
L writeDB(aa.w + 1, reg >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_addrr_b(reg16_t& reg, reg16_t& idx) {
|
auto R65816::op_write_addrr_b(reg16& reg, reg16& idx) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
L writedbr(aa.w + idx, reg);
|
L writeDB(aa.w + idx, reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_addrr_w(reg16_t& reg, reg16_t& idx) {
|
auto R65816::op_write_addrr_w(reg16& reg, reg16& idx) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
io();
|
io();
|
||||||
writedbr(aa.w + idx + 0, reg >> 0);
|
writeDB(aa.w + idx + 0, reg >> 0);
|
||||||
L writedbr(aa.w + idx + 1, reg >> 8);
|
L writeDB(aa.w + idx + 1, reg >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_longr_b(reg16_t& idx) {
|
auto R65816::op_write_longr_b(reg16& idx) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
L writelong(aa.d + idx, regs.a.l);
|
L writeLong(aa.d + idx, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_longr_w(reg16_t& idx) {
|
auto R65816::op_write_longr_w(reg16& idx) {
|
||||||
aa.l = readpc();
|
aa.l = readPC();
|
||||||
aa.h = readpc();
|
aa.h = readPC();
|
||||||
aa.b = readpc();
|
aa.b = readPC();
|
||||||
writelong(aa.d + idx + 0, regs.a.l);
|
writeLong(aa.d + idx + 0, r.a.l);
|
||||||
L writelong(aa.d + idx + 1, regs.a.h);
|
L writeLong(aa.d + idx + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_dp_b(reg16_t& reg) {
|
auto R65816::op_write_dp_b(reg16& reg) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
L writedp(dp, reg);
|
L writeDP(dp, reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_dp_w(reg16_t& reg) {
|
auto R65816::op_write_dp_w(reg16& reg) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
writedp(dp + 0, reg >> 0);
|
writeDP(dp + 0, reg >> 0);
|
||||||
L writedp(dp + 1, reg >> 8);
|
L writeDP(dp + 1, reg >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_dpr_b(reg16_t& reg, reg16_t& idx) {
|
auto R65816::op_write_dpr_b(reg16& reg, reg16& idx) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
L writedp(dp + idx, reg);
|
L writeDP(dp + idx, reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_write_dpr_w(reg16_t& reg, reg16_t& idx) {
|
auto R65816::op_write_dpr_w(reg16& reg, reg16& idx) {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
writedp(dp + idx + 0, reg >> 0);
|
writeDP(dp + idx + 0, reg >> 0);
|
||||||
L writedp(dp + idx + 1, reg >> 8);
|
L writeDP(dp + idx + 1, reg >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idp_b() {
|
auto R65816::op_sta_idp_b() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
L writedbr(aa.w, regs.a.l);
|
L writeDB(aa.w, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idp_w() {
|
auto R65816::op_sta_idp_w() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
writedbr(aa.w + 0, regs.a.l);
|
writeDB(aa.w + 0, r.a.l);
|
||||||
L writedbr(aa.w + 1, regs.a.h);
|
L writeDB(aa.w + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_ildp_b() {
|
auto R65816::op_sta_ildp_b() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
L writelong(aa.d, regs.a.l);
|
L writeLong(aa.d, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_ildp_w() {
|
auto R65816::op_sta_ildp_w() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
writelong(aa.d + 0, regs.a.l);
|
writeLong(aa.d + 0, r.a.l);
|
||||||
L writelong(aa.d + 1, regs.a.h);
|
L writeLong(aa.d + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idpx_b() {
|
auto R65816::op_sta_idpx_b() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
aa.l = readdp(dp + regs.x.w + 0);
|
aa.l = readDP(dp + r.x.w + 0);
|
||||||
aa.h = readdp(dp + regs.x.w + 1);
|
aa.h = readDP(dp + r.x.w + 1);
|
||||||
L writedbr(aa.w, regs.a.l);
|
L writeDB(aa.w, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idpx_w() {
|
auto R65816::op_sta_idpx_w() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
io();
|
io();
|
||||||
aa.l = readdp(dp + regs.x.w + 0);
|
aa.l = readDP(dp + r.x.w + 0);
|
||||||
aa.h = readdp(dp + regs.x.w + 1);
|
aa.h = readDP(dp + r.x.w + 1);
|
||||||
writedbr(aa.w + 0, regs.a.l);
|
writeDB(aa.w + 0, r.a.l);
|
||||||
L writedbr(aa.w + 1, regs.a.h);
|
L writeDB(aa.w + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idpy_b() {
|
auto R65816::op_sta_idpy_b() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
io();
|
io();
|
||||||
L writedbr(aa.w + regs.y.w, regs.a.l);
|
L writeDB(aa.w + r.y.w, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_idpy_w() {
|
auto R65816::op_sta_idpy_w() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDP(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDP(dp + 1);
|
||||||
io();
|
io();
|
||||||
writedbr(aa.w + regs.y.w + 0, regs.a.l);
|
writeDB(aa.w + r.y.w + 0, r.a.l);
|
||||||
L writedbr(aa.w + regs.y.w + 1, regs.a.h);
|
L writeDB(aa.w + r.y.w + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_ildpy_b() {
|
auto R65816::op_sta_ildpy_b() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
L writelong(aa.d + regs.y.w, regs.a.l);
|
L writeLong(aa.d + r.y.w, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_ildpy_w() {
|
auto R65816::op_sta_ildpy_w() {
|
||||||
dp = readpc();
|
dp = readPC();
|
||||||
io2();
|
io2();
|
||||||
aa.l = readdp(dp + 0);
|
aa.l = readDPn(dp + 0);
|
||||||
aa.h = readdp(dp + 1);
|
aa.h = readDPn(dp + 1);
|
||||||
aa.b = readdp(dp + 2);
|
aa.b = readDPn(dp + 2);
|
||||||
writelong(aa.d + regs.y.w + 0, regs.a.l);
|
writeLong(aa.d + r.y.w + 0, r.a.l);
|
||||||
L writelong(aa.d + regs.y.w + 1, regs.a.h);
|
L writeLong(aa.d + r.y.w + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_sr_b() {
|
auto R65816::op_sta_sr_b() {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
L writesp(sp, regs.a.l);
|
L writeSP(sp, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_sr_w() {
|
auto R65816::op_sta_sr_w() {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
writesp(sp + 0, regs.a.l);
|
writeSP(sp + 0, r.a.l);
|
||||||
L writesp(sp + 1, regs.a.h);
|
L writeSP(sp + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_isry_b() {
|
auto R65816::op_sta_isry_b() {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
aa.l = readsp(sp + 0);
|
aa.l = readSP(sp + 0);
|
||||||
aa.h = readsp(sp + 1);
|
aa.h = readSP(sp + 1);
|
||||||
io();
|
io();
|
||||||
L writedbr(aa.w + regs.y.w, regs.a.l);
|
L writeDB(aa.w + r.y.w, r.a.l);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::op_sta_isry_w() {
|
auto R65816::op_sta_isry_w() {
|
||||||
sp = readpc();
|
sp = readPC();
|
||||||
io();
|
io();
|
||||||
aa.l = readsp(sp + 0);
|
aa.l = readSP(sp + 0);
|
||||||
aa.h = readsp(sp + 1);
|
aa.h = readSP(sp + 1);
|
||||||
io();
|
io();
|
||||||
writedbr(aa.w + regs.y.w + 0, regs.a.l);
|
writeDB(aa.w + r.y.w + 0, r.a.l);
|
||||||
L writedbr(aa.w + regs.y.w + 1, regs.a.h);
|
L writeDB(aa.w + r.y.w + 1, r.a.h);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,8 @@ namespace Processor {
|
||||||
#include "disassembler.cpp"
|
#include "disassembler.cpp"
|
||||||
#include "serialization.cpp"
|
#include "serialization.cpp"
|
||||||
|
|
||||||
|
#define E if(r.e)
|
||||||
|
#define N if(!r.e)
|
||||||
#define L lastCycle();
|
#define L lastCycle();
|
||||||
#define call(op) (this->*op)()
|
#define call(op) (this->*op)()
|
||||||
|
|
||||||
|
@ -17,9 +19,6 @@ namespace Processor {
|
||||||
#include "opcode_misc.cpp"
|
#include "opcode_misc.cpp"
|
||||||
#include "switch.cpp"
|
#include "switch.cpp"
|
||||||
|
|
||||||
#undef L
|
|
||||||
#undef call
|
|
||||||
|
|
||||||
//immediate, 2-cycle opcodes with I/O cycle will become bus read
|
//immediate, 2-cycle opcodes with I/O cycle will become bus read
|
||||||
//when an IRQ is to be triggered immediately after opcode completion.
|
//when an IRQ is to be triggered immediately after opcode completion.
|
||||||
//this affects the following opcodes:
|
//this affects the following opcodes:
|
||||||
|
@ -31,43 +30,47 @@ namespace Processor {
|
||||||
auto R65816::ioIRQ() -> void {
|
auto R65816::ioIRQ() -> void {
|
||||||
if(interruptPending()) {
|
if(interruptPending()) {
|
||||||
//modify I/O cycle to bus read cycle, do not increment PC
|
//modify I/O cycle to bus read cycle, do not increment PC
|
||||||
read(regs.pc.d);
|
read(r.pc.d);
|
||||||
} else {
|
} else {
|
||||||
io();
|
io();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::io2() -> void {
|
auto R65816::io2() -> void {
|
||||||
if(regs.d.l != 0x00) {
|
if(r.d.l != 0x00) {
|
||||||
io();
|
io();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::io4(uint16 x, uint16 y) -> void {
|
auto R65816::io4(uint16 x, uint16 y) -> void {
|
||||||
if(!regs.p.x || (x & 0xff00) != (y & 0xff00)) {
|
if(!r.p.x || (x & 0xff00) != (y & 0xff00)) {
|
||||||
io();
|
io();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::io6(uint16 addr) -> void {
|
auto R65816::io6(uint16 addr) -> void {
|
||||||
if(regs.e && (regs.pc.w & 0xff00) != (addr & 0xff00)) {
|
if(r.e && (r.pc.w & 0xff00) != (addr & 0xff00)) {
|
||||||
io();
|
io();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto R65816::interrupt() -> void {
|
auto R65816::interrupt() -> void {
|
||||||
read(regs.pc.d);
|
read(r.pc.d);
|
||||||
io();
|
io();
|
||||||
if(!regs.e) writestack(regs.pc.b);
|
N writeSP(r.pc.b);
|
||||||
writestack(regs.pc.h);
|
writeSP(r.pc.h);
|
||||||
writestack(regs.pc.l);
|
writeSP(r.pc.l);
|
||||||
writestack(regs.e ? (regs.p & ~0x10) : regs.p);
|
writeSP(r.e ? (r.p & ~0x10) : r.p);
|
||||||
rd.l = read(regs.vector + 0);
|
r.pc.l = read(r.vector + 0);
|
||||||
regs.pc.b = 0x00;
|
r.p.i = 1;
|
||||||
regs.p.i = 1;
|
r.p.d = 0;
|
||||||
regs.p.d = 0;
|
r.pc.h = read(r.vector + 1);
|
||||||
rd.h = read(regs.vector + 1);
|
r.pc.b = 0x00;
|
||||||
regs.pc.w = rd.w;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#undef E
|
||||||
|
#undef N
|
||||||
|
#undef L
|
||||||
|
#undef call
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,7 +20,7 @@ struct R65816 {
|
||||||
virtual auto interruptPending() const -> bool = 0;
|
virtual auto interruptPending() const -> bool = 0;
|
||||||
virtual auto interrupt() -> void;
|
virtual auto interrupt() -> void;
|
||||||
|
|
||||||
virtual auto disassemblerRead(uint24 addr) -> uint8 { return 0u; }
|
virtual auto disassemblerRead(uint24 addr) -> uint8 { return 0; }
|
||||||
|
|
||||||
//r65816.cpp
|
//r65816.cpp
|
||||||
alwaysinline auto ioIRQ() -> void;
|
alwaysinline auto ioIRQ() -> void;
|
||||||
|
@ -88,8 +88,8 @@ struct R65816 {
|
||||||
auto op_read_longx_w(fp);
|
auto op_read_longx_w(fp);
|
||||||
auto op_read_dp_b(fp);
|
auto op_read_dp_b(fp);
|
||||||
auto op_read_dp_w(fp);
|
auto op_read_dp_w(fp);
|
||||||
auto op_read_dpr_b(fp, reg16_t&);
|
auto op_read_dpr_b(fp, reg16&);
|
||||||
auto op_read_dpr_w(fp, reg16_t&);
|
auto op_read_dpr_w(fp, reg16&);
|
||||||
auto op_read_idp_b(fp);
|
auto op_read_idp_b(fp);
|
||||||
auto op_read_idp_w(fp);
|
auto op_read_idp_w(fp);
|
||||||
auto op_read_idpx_b(fp);
|
auto op_read_idpx_b(fp);
|
||||||
|
@ -106,16 +106,16 @@ struct R65816 {
|
||||||
auto op_read_isry_w(fp);
|
auto op_read_isry_w(fp);
|
||||||
|
|
||||||
//opcode_write.cpp
|
//opcode_write.cpp
|
||||||
auto op_write_addr_b(reg16_t&);
|
auto op_write_addr_b(reg16&);
|
||||||
auto op_write_addr_w(reg16_t&);
|
auto op_write_addr_w(reg16&);
|
||||||
auto op_write_addrr_b(reg16_t&, reg16_t&);
|
auto op_write_addrr_b(reg16&, reg16&);
|
||||||
auto op_write_addrr_w(reg16_t&, reg16_t&);
|
auto op_write_addrr_w(reg16&, reg16&);
|
||||||
auto op_write_longr_b(reg16_t&);
|
auto op_write_longr_b(reg16&);
|
||||||
auto op_write_longr_w(reg16_t&);
|
auto op_write_longr_w(reg16&);
|
||||||
auto op_write_dp_b(reg16_t&);
|
auto op_write_dp_b(reg16&);
|
||||||
auto op_write_dp_w(reg16_t&);
|
auto op_write_dp_w(reg16&);
|
||||||
auto op_write_dpr_b(reg16_t&, reg16_t&);
|
auto op_write_dpr_b(reg16&, reg16&);
|
||||||
auto op_write_dpr_w(reg16_t&, reg16_t&);
|
auto op_write_dpr_w(reg16&, reg16&);
|
||||||
auto op_sta_idp_b();
|
auto op_sta_idp_b();
|
||||||
auto op_sta_idp_w();
|
auto op_sta_idp_w();
|
||||||
auto op_sta_ildp_b();
|
auto op_sta_ildp_b();
|
||||||
|
@ -132,8 +132,8 @@ struct R65816 {
|
||||||
auto op_sta_isry_w();
|
auto op_sta_isry_w();
|
||||||
|
|
||||||
//opcode_rmw.cpp
|
//opcode_rmw.cpp
|
||||||
auto op_adjust_imm_b(reg16_t&, signed);
|
auto op_adjust_imm_b(reg16&, int);
|
||||||
auto op_adjust_imm_w(reg16_t&, signed);
|
auto op_adjust_imm_w(reg16&, int);
|
||||||
auto op_asl_imm_b();
|
auto op_asl_imm_b();
|
||||||
auto op_asl_imm_w();
|
auto op_asl_imm_w();
|
||||||
auto op_lsr_imm_b();
|
auto op_lsr_imm_b();
|
||||||
|
@ -161,58 +161,44 @@ struct R65816 {
|
||||||
auto op_jmp_iaddrx();
|
auto op_jmp_iaddrx();
|
||||||
auto op_jmp_iladdr();
|
auto op_jmp_iladdr();
|
||||||
auto op_jsr_addr();
|
auto op_jsr_addr();
|
||||||
auto op_jsr_long_e();
|
auto op_jsr_long();
|
||||||
auto op_jsr_long_n();
|
auto op_jsr_iaddrx();
|
||||||
auto op_jsr_iaddrx_e();
|
auto op_rti();
|
||||||
auto op_jsr_iaddrx_n();
|
|
||||||
auto op_rti_e();
|
|
||||||
auto op_rti_n();
|
|
||||||
auto op_rts();
|
auto op_rts();
|
||||||
auto op_rtl_e();
|
auto op_rtl();
|
||||||
auto op_rtl_n();
|
|
||||||
|
|
||||||
//opcode_misc.cpp
|
//opcode_misc.cpp
|
||||||
auto op_nop();
|
auto op_nop();
|
||||||
auto op_wdm();
|
auto op_wdm();
|
||||||
auto op_xba();
|
auto op_xba();
|
||||||
auto op_move_b(signed adjust);
|
auto op_move_b(int adjust);
|
||||||
auto op_move_w(signed adjust);
|
auto op_move_w(int adjust);
|
||||||
auto op_interrupt_e(uint16);
|
auto op_interrupt(uint16);
|
||||||
auto op_interrupt_n(uint16);
|
|
||||||
auto op_stp();
|
auto op_stp();
|
||||||
auto op_wai();
|
auto op_wai();
|
||||||
auto op_xce();
|
auto op_xce();
|
||||||
auto op_flag(bool& flag, bool value);
|
auto op_flag(bool& flag, bool value);
|
||||||
auto op_pflag_e(bool);
|
auto op_pflag(bool);
|
||||||
auto op_pflag_n(bool);
|
auto op_transfer_b(reg16&, reg16&);
|
||||||
auto op_transfer_b(reg16_t&, reg16_t&);
|
auto op_transfer_w(reg16&, reg16&);
|
||||||
auto op_transfer_w(reg16_t&, reg16_t&);
|
auto op_tcs();
|
||||||
auto op_tcs_e();
|
|
||||||
auto op_tcs_n();
|
|
||||||
auto op_tsx_b();
|
auto op_tsx_b();
|
||||||
auto op_tsx_w();
|
auto op_tsx_w();
|
||||||
auto op_txs_e();
|
auto op_txs();
|
||||||
auto op_txs_n();
|
auto op_push_b(reg16&);
|
||||||
auto op_push_b(reg16_t&);
|
auto op_push_w(reg16&);
|
||||||
auto op_push_w(reg16_t&);
|
auto op_phd();
|
||||||
auto op_phd_e();
|
|
||||||
auto op_phd_n();
|
|
||||||
auto op_phb();
|
auto op_phb();
|
||||||
auto op_phk();
|
auto op_phk();
|
||||||
auto op_php();
|
auto op_php();
|
||||||
auto op_pull_b(reg16_t&);
|
auto op_pull_b(reg16&);
|
||||||
auto op_pull_w(reg16_t&);
|
auto op_pull_w(reg16&);
|
||||||
auto op_pld_e();
|
auto op_pld();
|
||||||
auto op_pld_n();
|
|
||||||
auto op_plb();
|
auto op_plb();
|
||||||
auto op_plp_e();
|
auto op_plp();
|
||||||
auto op_plp_n();
|
auto op_pea();
|
||||||
auto op_pea_e();
|
auto op_pei();
|
||||||
auto op_pea_n();
|
auto op_per();
|
||||||
auto op_pei_e();
|
|
||||||
auto op_pei_n();
|
|
||||||
auto op_per_e();
|
|
||||||
auto op_per_n();
|
|
||||||
|
|
||||||
//switch.cpp
|
//switch.cpp
|
||||||
auto instruction() -> void;
|
auto instruction() -> void;
|
||||||
|
@ -220,8 +206,8 @@ struct R65816 {
|
||||||
//serialization.cpp
|
//serialization.cpp
|
||||||
auto serialize(serializer&) -> void;
|
auto serialize(serializer&) -> void;
|
||||||
|
|
||||||
regs_t regs;
|
Registers r;
|
||||||
reg24_t aa, rd;
|
reg24 aa, rd;
|
||||||
uint8 sp, dp;
|
uint8 sp, dp;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
struct flag_t {
|
struct Flags {
|
||||||
bool n{0};
|
bool n{0};
|
||||||
bool v{0};
|
bool v{0};
|
||||||
bool m{0};
|
bool m{0};
|
||||||
|
@ -8,68 +8,68 @@ struct flag_t {
|
||||||
bool z{0};
|
bool z{0};
|
||||||
bool c{0};
|
bool c{0};
|
||||||
|
|
||||||
inline operator unsigned() const {
|
inline operator uint() const {
|
||||||
return (n << 7) + (v << 6) + (m << 5) + (x << 4)
|
return (n << 7) + (v << 6) + (m << 5) + (x << 4)
|
||||||
+ (d << 3) + (i << 2) + (z << 1) + (c << 0);
|
+ (d << 3) + (i << 2) + (z << 1) + (c << 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline auto operator=(uint8 data) -> unsigned {
|
inline auto operator=(uint8 data) -> uint {
|
||||||
n = data & 0x80; v = data & 0x40; m = data & 0x20; x = data & 0x10;
|
n = data & 0x80; v = data & 0x40; m = data & 0x20; x = data & 0x10;
|
||||||
d = data & 0x08; i = data & 0x04; z = data & 0x02; c = data & 0x01;
|
d = data & 0x08; i = data & 0x04; z = data & 0x02; c = data & 0x01;
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct reg16_t {
|
struct reg16 {
|
||||||
union {
|
union {
|
||||||
uint16_t w = 0;
|
uint16_t w = 0;
|
||||||
struct { uint8_t order_lsb2(l, h); };
|
struct { uint8_t order_lsb2(l, h); };
|
||||||
};
|
};
|
||||||
|
|
||||||
inline operator unsigned() const { return w; }
|
inline operator uint() const { return w; }
|
||||||
inline auto operator = (unsigned i) -> unsigned { return w = i; }
|
inline auto operator =(uint i) -> uint { return w = i; }
|
||||||
inline auto operator |= (unsigned i) -> unsigned { return w |= i; }
|
inline auto operator |=(uint i) -> uint { return w |= i; }
|
||||||
inline auto operator ^= (unsigned i) -> unsigned { return w ^= i; }
|
inline auto operator ^=(uint i) -> uint { return w ^= i; }
|
||||||
inline auto operator &= (unsigned i) -> unsigned { return w &= i; }
|
inline auto operator &=(uint i) -> uint { return w &= i; }
|
||||||
inline auto operator <<= (unsigned i) -> unsigned { return w <<= i; }
|
inline auto operator<<=(uint i) -> uint { return w <<= i; }
|
||||||
inline auto operator >>= (unsigned i) -> unsigned { return w >>= i; }
|
inline auto operator>>=(uint i) -> uint { return w >>= i; }
|
||||||
inline auto operator += (unsigned i) -> unsigned { return w += i; }
|
inline auto operator +=(uint i) -> uint { return w += i; }
|
||||||
inline auto operator -= (unsigned i) -> unsigned { return w -= i; }
|
inline auto operator -=(uint i) -> uint { return w -= i; }
|
||||||
inline auto operator *= (unsigned i) -> unsigned { return w *= i; }
|
inline auto operator *=(uint i) -> uint { return w *= i; }
|
||||||
inline auto operator /= (unsigned i) -> unsigned { return w /= i; }
|
inline auto operator /=(uint i) -> uint { return w /= i; }
|
||||||
inline auto operator %= (unsigned i) -> unsigned { return w %= i; }
|
inline auto operator %=(uint i) -> uint { return w %= i; }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct reg24_t {
|
struct reg24 {
|
||||||
union {
|
union {
|
||||||
uint32_t d = 0;
|
uint32_t d = 0;
|
||||||
struct { uint16_t order_lsb2(w, wh); };
|
struct { uint16_t order_lsb2(w, wh); };
|
||||||
struct { uint8_t order_lsb4(l, h, b, bh); };
|
struct { uint8_t order_lsb4(l, h, b, bh); };
|
||||||
};
|
};
|
||||||
|
|
||||||
inline operator unsigned() const { return d; }
|
inline operator uint() const { return d; }
|
||||||
inline auto operator = (unsigned i) -> unsigned { return d = uclip<24>(i); }
|
inline auto operator =(uint i) -> uint { return d = uclip<24>(i); }
|
||||||
inline auto operator |= (unsigned i) -> unsigned { return d = uclip<24>(d | i); }
|
inline auto operator |=(uint i) -> uint { return d = uclip<24>(d | i); }
|
||||||
inline auto operator ^= (unsigned i) -> unsigned { return d = uclip<24>(d ^ i); }
|
inline auto operator ^=(uint i) -> uint { return d = uclip<24>(d ^ i); }
|
||||||
inline auto operator &= (unsigned i) -> unsigned { return d = uclip<24>(d & i); }
|
inline auto operator &=(uint i) -> uint { return d = uclip<24>(d & i); }
|
||||||
inline auto operator <<= (unsigned i) -> unsigned { return d = uclip<24>(d << i); }
|
inline auto operator<<=(uint i) -> uint { return d = uclip<24>(d << i); }
|
||||||
inline auto operator >>= (unsigned i) -> unsigned { return d = uclip<24>(d >> i); }
|
inline auto operator>>=(uint i) -> uint { return d = uclip<24>(d >> i); }
|
||||||
inline auto operator += (unsigned i) -> unsigned { return d = uclip<24>(d + i); }
|
inline auto operator +=(uint i) -> uint { return d = uclip<24>(d + i); }
|
||||||
inline auto operator -= (unsigned i) -> unsigned { return d = uclip<24>(d - i); }
|
inline auto operator -=(uint i) -> uint { return d = uclip<24>(d - i); }
|
||||||
inline auto operator *= (unsigned i) -> unsigned { return d = uclip<24>(d * i); }
|
inline auto operator *=(uint i) -> uint { return d = uclip<24>(d * i); }
|
||||||
inline auto operator /= (unsigned i) -> unsigned { return d = uclip<24>(d / i); }
|
inline auto operator /=(uint i) -> uint { return d = uclip<24>(d / i); }
|
||||||
inline auto operator %= (unsigned i) -> unsigned { return d = uclip<24>(d % i); }
|
inline auto operator %=(uint i) -> uint { return d = uclip<24>(d % i); }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct regs_t {
|
struct Registers {
|
||||||
reg24_t pc;
|
reg24 pc;
|
||||||
reg16_t a;
|
reg16 a;
|
||||||
reg16_t x;
|
reg16 x;
|
||||||
reg16_t y;
|
reg16 y;
|
||||||
reg16_t z; //pseudo-register (zero register)
|
reg16 z; //pseudo-register (zero register)
|
||||||
reg16_t s;
|
reg16 s;
|
||||||
reg16_t d;
|
reg16 d;
|
||||||
flag_t p;
|
Flags p;
|
||||||
uint8 db{0};
|
uint8 db{0};
|
||||||
bool e{0};
|
bool e{0};
|
||||||
|
|
||||||
|
|
|
@ -1,28 +1,28 @@
|
||||||
auto R65816::serialize(serializer& s) -> void {
|
auto R65816::serialize(serializer& s) -> void {
|
||||||
s.integer(regs.pc.d);
|
s.integer(r.pc.d);
|
||||||
|
|
||||||
s.integer(regs.a.w);
|
s.integer(r.a.w);
|
||||||
s.integer(regs.x.w);
|
s.integer(r.x.w);
|
||||||
s.integer(regs.y.w);
|
s.integer(r.y.w);
|
||||||
s.integer(regs.z.w);
|
s.integer(r.z.w);
|
||||||
s.integer(regs.s.w);
|
s.integer(r.s.w);
|
||||||
s.integer(regs.d.w);
|
s.integer(r.d.w);
|
||||||
|
|
||||||
s.integer(regs.p.n);
|
s.integer(r.p.n);
|
||||||
s.integer(regs.p.v);
|
s.integer(r.p.v);
|
||||||
s.integer(regs.p.m);
|
s.integer(r.p.m);
|
||||||
s.integer(regs.p.x);
|
s.integer(r.p.x);
|
||||||
s.integer(regs.p.d);
|
s.integer(r.p.d);
|
||||||
s.integer(regs.p.i);
|
s.integer(r.p.i);
|
||||||
s.integer(regs.p.z);
|
s.integer(r.p.z);
|
||||||
s.integer(regs.p.c);
|
s.integer(r.p.c);
|
||||||
|
|
||||||
s.integer(regs.db);
|
s.integer(r.db);
|
||||||
s.integer(regs.e);
|
s.integer(r.e);
|
||||||
s.integer(regs.irq);
|
s.integer(r.irq);
|
||||||
s.integer(regs.wai);
|
s.integer(r.wai);
|
||||||
s.integer(regs.mdr);
|
s.integer(r.mdr);
|
||||||
s.integer(regs.vector);
|
s.integer(r.vector);
|
||||||
|
|
||||||
s.integer(aa.d);
|
s.integer(aa.d);
|
||||||
s.integer(rd.d);
|
s.integer(rd.d);
|
||||||
|
|
|
@ -1,24 +1,22 @@
|
||||||
auto R65816::instruction() -> void {
|
auto R65816::instruction() -> void {
|
||||||
#define opA( n, o ) case n: return op_##o();
|
#define opA( n, o ) case n: return op_##o();
|
||||||
|
#define opAI( n, o, i ) case n: return op_##o(i);
|
||||||
#define opAII(n, o, i, j) case n: return op_##o(i, j);
|
#define opAII(n, o, i, j) case n: return op_##o(i, j);
|
||||||
#define opE( n, o ) case n: return regs.e ? op_##o##_e() : op_##o##_n();
|
#define opM( n, o ) case n: return r.p.m ? op_##o##_b() : op_##o##_w();
|
||||||
#define opEI( n, o, i ) case n: return regs.e ? op_##o##_e(i) : op_##o##_n(i);
|
#define opMF( n, o, f ) case n: return r.p.m ? op_##o##_b(&R65816::op_##f##_b) : op_##o##_w(&R65816::op_##f##_w);
|
||||||
#define opEII(n, o, i, j) case n: return regs.e ? op_##o##_e(i) : op_##o##_n(j);
|
#define opMFI(n, o, f, i) case n: return r.p.m ? op_##o##_b(&R65816::op_##f##_b, i) : op_##o##_w(&R65816::op_##f##_w, i);
|
||||||
#define opM( n, o ) case n: return regs.p.m ? op_##o##_b() : op_##o##_w();
|
#define opMI( n, o, i ) case n: return r.p.m ? op_##o##_b(i) : op_##o##_w(i);
|
||||||
#define opMF( n, o, f ) case n: return regs.p.m ? op_##o##_b(&R65816::op_##f##_b) : op_##o##_w(&R65816::op_##f##_w);
|
#define opMII(n, o, i, j) case n: return r.p.m ? op_##o##_b(i, j) : op_##o##_w(i, j);
|
||||||
#define opMFI(n, o, f, i) case n: return regs.p.m ? op_##o##_b(&R65816::op_##f##_b, i) : op_##o##_w(&R65816::op_##f##_w, i);
|
#define opX( n, o) case n: return r.p.x ? op_##o##_b() : op_##o##_w();
|
||||||
#define opMI( n, o, i ) case n: return regs.p.m ? op_##o##_b(i) : op_##o##_w(i);
|
#define opXF( n, o, f ) case n: return r.p.x ? op_##o##_b(&R65816::op_##f##_b) : op_##o##_w(&R65816::op_##f##_w);
|
||||||
#define opMII(n, o, i, j) case n: return regs.p.m ? op_##o##_b(i, j) : op_##o##_w(i, j);
|
#define opXFI(n, o, f, i) case n: return r.p.x ? op_##o##_b(&R65816::op_##f##_b, i) : op_##o##_w(&R65816::op_##f##_w, i);
|
||||||
#define opX( n, o) case n: return regs.p.x ? op_##o##_b() : op_##o##_w();
|
#define opXI( n, o, i ) case n: return r.p.x ? op_##o##_b(i) : op_##o##_w(i);
|
||||||
#define opXF( n, o, f ) case n: return regs.p.x ? op_##o##_b(&R65816::op_##f##_b) : op_##o##_w(&R65816::op_##f##_w);
|
#define opXII(n, o, i, j) case n: return r.p.x ? op_##o##_b(i, j) : op_##o##_w(i, j);
|
||||||
#define opXFI(n, o, f, i) case n: return regs.p.x ? op_##o##_b(&R65816::op_##f##_b, i) : op_##o##_w(&R65816::op_##f##_w, i);
|
|
||||||
#define opXI( n, o, i ) case n: return regs.p.x ? op_##o##_b(i) : op_##o##_w(i);
|
|
||||||
#define opXII(n, o, i, j) case n: return regs.p.x ? op_##o##_b(i, j) : op_##o##_w(i, j);
|
|
||||||
|
|
||||||
switch(readpc()) {
|
switch(readPC()) {
|
||||||
opEII(0x00, interrupt, 0xfffe, 0xffe6)
|
opAI (0x00, interrupt, r.e ? 0xffe6 : 0xfffe)
|
||||||
opMF (0x01, read_idpx, ora)
|
opMF (0x01, read_idpx, ora)
|
||||||
opEII(0x02, interrupt, 0xfff4, 0xffe4)
|
opAI (0x02, interrupt, r.e ? 0xffe4 : 0xfff4)
|
||||||
opMF (0x03, read_sr, ora)
|
opMF (0x03, read_sr, ora)
|
||||||
opMF (0x04, adjust_dp, tsb)
|
opMF (0x04, adjust_dp, tsb)
|
||||||
opMF (0x05, read_dp, ora)
|
opMF (0x05, read_dp, ora)
|
||||||
|
@ -27,60 +25,60 @@ auto R65816::instruction() -> void {
|
||||||
opA (0x08, php)
|
opA (0x08, php)
|
||||||
opMF (0x09, read_const, ora)
|
opMF (0x09, read_const, ora)
|
||||||
opM (0x0a, asl_imm)
|
opM (0x0a, asl_imm)
|
||||||
opE (0x0b, phd)
|
opA (0x0b, phd)
|
||||||
opMF (0x0c, adjust_addr, tsb)
|
opMF (0x0c, adjust_addr, tsb)
|
||||||
opMF (0x0d, read_addr, ora)
|
opMF (0x0d, read_addr, ora)
|
||||||
opMF (0x0e, adjust_addr, asl)
|
opMF (0x0e, adjust_addr, asl)
|
||||||
opMF (0x0f, read_long, ora)
|
opMF (0x0f, read_long, ora)
|
||||||
opAII(0x10, branch, regs.p.n, 0)
|
opAII(0x10, branch, r.p.n, 0)
|
||||||
opMF (0x11, read_idpy, ora)
|
opMF (0x11, read_idpy, ora)
|
||||||
opMF (0x12, read_idp, ora)
|
opMF (0x12, read_idp, ora)
|
||||||
opMF (0x13, read_isry, ora)
|
opMF (0x13, read_isry, ora)
|
||||||
opMF (0x14, adjust_dp, trb)
|
opMF (0x14, adjust_dp, trb)
|
||||||
opMFI(0x15, read_dpr, ora, regs.x)
|
opMFI(0x15, read_dpr, ora, r.x)
|
||||||
opMF (0x16, adjust_dpx, asl)
|
opMF (0x16, adjust_dpx, asl)
|
||||||
opMF (0x17, read_ildpy, ora)
|
opMF (0x17, read_ildpy, ora)
|
||||||
opAII(0x18, flag, regs.p.c, 0)
|
opAII(0x18, flag, r.p.c, 0)
|
||||||
opMF (0x19, read_addry, ora)
|
opMF (0x19, read_addry, ora)
|
||||||
opMII(0x1a, adjust_imm, regs.a, +1)
|
opMII(0x1a, adjust_imm, r.a, +1)
|
||||||
opE (0x1b, tcs)
|
opA (0x1b, tcs)
|
||||||
opMF (0x1c, adjust_addr, trb)
|
opMF (0x1c, adjust_addr, trb)
|
||||||
opMF (0x1d, read_addrx, ora)
|
opMF (0x1d, read_addrx, ora)
|
||||||
opMF (0x1e, adjust_addrx, asl)
|
opMF (0x1e, adjust_addrx, asl)
|
||||||
opMF (0x1f, read_longx, ora)
|
opMF (0x1f, read_longx, ora)
|
||||||
opA (0x20, jsr_addr)
|
opA (0x20, jsr_addr)
|
||||||
opMF (0x21, read_idpx, and)
|
opMF (0x21, read_idpx, and)
|
||||||
opE (0x22, jsr_long)
|
opA (0x22, jsr_long)
|
||||||
opMF (0x23, read_sr, and)
|
opMF (0x23, read_sr, and)
|
||||||
opMF (0x24, read_dp, bit)
|
opMF (0x24, read_dp, bit)
|
||||||
opMF (0x25, read_dp, and)
|
opMF (0x25, read_dp, and)
|
||||||
opMF (0x26, adjust_dp, rol)
|
opMF (0x26, adjust_dp, rol)
|
||||||
opMF (0x27, read_ildp, and)
|
opMF (0x27, read_ildp, and)
|
||||||
opE (0x28, plp)
|
opA (0x28, plp)
|
||||||
opMF (0x29, read_const, and)
|
opMF (0x29, read_const, and)
|
||||||
opM (0x2a, rol_imm)
|
opM (0x2a, rol_imm)
|
||||||
opE (0x2b, pld)
|
opA (0x2b, pld)
|
||||||
opMF (0x2c, read_addr, bit)
|
opMF (0x2c, read_addr, bit)
|
||||||
opMF (0x2d, read_addr, and)
|
opMF (0x2d, read_addr, and)
|
||||||
opMF (0x2e, adjust_addr, rol)
|
opMF (0x2e, adjust_addr, rol)
|
||||||
opMF (0x2f, read_long, and)
|
opMF (0x2f, read_long, and)
|
||||||
opAII(0x30, branch, regs.p.n, 1)
|
opAII(0x30, branch, r.p.n, 1)
|
||||||
opMF (0x31, read_idpy, and)
|
opMF (0x31, read_idpy, and)
|
||||||
opMF (0x32, read_idp, and)
|
opMF (0x32, read_idp, and)
|
||||||
opMF (0x33, read_isry, and)
|
opMF (0x33, read_isry, and)
|
||||||
opMFI(0x34, read_dpr, bit, regs.x)
|
opMFI(0x34, read_dpr, bit, r.x)
|
||||||
opMFI(0x35, read_dpr, and, regs.x)
|
opMFI(0x35, read_dpr, and, r.x)
|
||||||
opMF (0x36, adjust_dpx, rol)
|
opMF (0x36, adjust_dpx, rol)
|
||||||
opMF (0x37, read_ildpy, and)
|
opMF (0x37, read_ildpy, and)
|
||||||
opAII(0x38, flag, regs.p.c, 1)
|
opAII(0x38, flag, r.p.c, 1)
|
||||||
opMF (0x39, read_addry, and)
|
opMF (0x39, read_addry, and)
|
||||||
opMII(0x3a, adjust_imm, regs.a, -1)
|
opMII(0x3a, adjust_imm, r.a, -1)
|
||||||
opAII(0x3b, transfer_w, regs.s, regs.a)
|
opAII(0x3b, transfer_w, r.s, r.a)
|
||||||
opMF (0x3c, read_addrx, bit)
|
opMF (0x3c, read_addrx, bit)
|
||||||
opMF (0x3d, read_addrx, and)
|
opMF (0x3d, read_addrx, and)
|
||||||
opMF (0x3e, adjust_addrx, rol)
|
opMF (0x3e, adjust_addrx, rol)
|
||||||
opMF (0x3f, read_longx, and)
|
opMF (0x3f, read_longx, and)
|
||||||
opE (0x40, rti)
|
opA (0x40, rti)
|
||||||
opMF (0x41, read_idpx, eor)
|
opMF (0x41, read_idpx, eor)
|
||||||
opA (0x42, wdm)
|
opA (0x42, wdm)
|
||||||
opMF (0x43, read_sr, eor)
|
opMF (0x43, read_sr, eor)
|
||||||
|
@ -88,7 +86,7 @@ auto R65816::instruction() -> void {
|
||||||
opMF (0x45, read_dp, eor)
|
opMF (0x45, read_dp, eor)
|
||||||
opMF (0x46, adjust_dp, lsr)
|
opMF (0x46, adjust_dp, lsr)
|
||||||
opMF (0x47, read_ildp, eor)
|
opMF (0x47, read_ildp, eor)
|
||||||
opMI (0x48, push, regs.a)
|
opMI (0x48, push, r.a)
|
||||||
opMF (0x49, read_const, eor)
|
opMF (0x49, read_const, eor)
|
||||||
opM (0x4a, lsr_imm)
|
opM (0x4a, lsr_imm)
|
||||||
opA (0x4b, phk)
|
opA (0x4b, phk)
|
||||||
|
@ -96,50 +94,50 @@ auto R65816::instruction() -> void {
|
||||||
opMF (0x4d, read_addr, eor)
|
opMF (0x4d, read_addr, eor)
|
||||||
opMF (0x4e, adjust_addr, lsr)
|
opMF (0x4e, adjust_addr, lsr)
|
||||||
opMF (0x4f, read_long, eor)
|
opMF (0x4f, read_long, eor)
|
||||||
opAII(0x50, branch, regs.p.v, 0)
|
opAII(0x50, branch, r.p.v, 0)
|
||||||
opMF (0x51, read_idpy, eor)
|
opMF (0x51, read_idpy, eor)
|
||||||
opMF (0x52, read_idp, eor)
|
opMF (0x52, read_idp, eor)
|
||||||
opMF (0x53, read_isry, eor)
|
opMF (0x53, read_isry, eor)
|
||||||
opXI (0x54, move, +1)
|
opXI (0x54, move, +1)
|
||||||
opMFI(0x55, read_dpr, eor, regs.x)
|
opMFI(0x55, read_dpr, eor, r.x)
|
||||||
opMF (0x56, adjust_dpx, lsr)
|
opMF (0x56, adjust_dpx, lsr)
|
||||||
opMF (0x57, read_ildpy, eor)
|
opMF (0x57, read_ildpy, eor)
|
||||||
opAII(0x58, flag, regs.p.i, 0)
|
opAII(0x58, flag, r.p.i, 0)
|
||||||
opMF (0x59, read_addry, eor)
|
opMF (0x59, read_addry, eor)
|
||||||
opXI (0x5a, push, regs.y)
|
opXI (0x5a, push, r.y)
|
||||||
opAII(0x5b, transfer_w, regs.a, regs.d)
|
opAII(0x5b, transfer_w, r.a, r.d)
|
||||||
opA (0x5c, jmp_long)
|
opA (0x5c, jmp_long)
|
||||||
opMF (0x5d, read_addrx, eor)
|
opMF (0x5d, read_addrx, eor)
|
||||||
opMF (0x5e, adjust_addrx, lsr)
|
opMF (0x5e, adjust_addrx, lsr)
|
||||||
opMF (0x5f, read_longx, eor)
|
opMF (0x5f, read_longx, eor)
|
||||||
opA (0x60, rts)
|
opA (0x60, rts)
|
||||||
opMF (0x61, read_idpx, adc)
|
opMF (0x61, read_idpx, adc)
|
||||||
opE (0x62, per)
|
opA (0x62, per)
|
||||||
opMF (0x63, read_sr, adc)
|
opMF (0x63, read_sr, adc)
|
||||||
opMI (0x64, write_dp, regs.z)
|
opMI (0x64, write_dp, r.z)
|
||||||
opMF (0x65, read_dp, adc)
|
opMF (0x65, read_dp, adc)
|
||||||
opMF (0x66, adjust_dp, ror)
|
opMF (0x66, adjust_dp, ror)
|
||||||
opMF (0x67, read_ildp, adc)
|
opMF (0x67, read_ildp, adc)
|
||||||
opMI (0x68, pull, regs.a)
|
opMI (0x68, pull, r.a)
|
||||||
opMF (0x69, read_const, adc)
|
opMF (0x69, read_const, adc)
|
||||||
opM (0x6a, ror_imm)
|
opM (0x6a, ror_imm)
|
||||||
opE (0x6b, rtl)
|
opA (0x6b, rtl)
|
||||||
opA (0x6c, jmp_iaddr)
|
opA (0x6c, jmp_iaddr)
|
||||||
opMF (0x6d, read_addr, adc)
|
opMF (0x6d, read_addr, adc)
|
||||||
opMF (0x6e, adjust_addr, ror)
|
opMF (0x6e, adjust_addr, ror)
|
||||||
opMF (0x6f, read_long, adc)
|
opMF (0x6f, read_long, adc)
|
||||||
opAII(0x70, branch, regs.p.v, 1)
|
opAII(0x70, branch, r.p.v, 1)
|
||||||
opMF (0x71, read_idpy, adc)
|
opMF (0x71, read_idpy, adc)
|
||||||
opMF (0x72, read_idp, adc)
|
opMF (0x72, read_idp, adc)
|
||||||
opMF (0x73, read_isry, adc)
|
opMF (0x73, read_isry, adc)
|
||||||
opMII(0x74, write_dpr, regs.z, regs.x)
|
opMII(0x74, write_dpr, r.z, r.x)
|
||||||
opMFI(0x75, read_dpr, adc, regs.x)
|
opMFI(0x75, read_dpr, adc, r.x)
|
||||||
opMF (0x76, adjust_dpx, ror)
|
opMF (0x76, adjust_dpx, ror)
|
||||||
opMF (0x77, read_ildpy, adc)
|
opMF (0x77, read_ildpy, adc)
|
||||||
opAII(0x78, flag, regs.p.i, 1)
|
opAII(0x78, flag, r.p.i, 1)
|
||||||
opMF (0x79, read_addry, adc)
|
opMF (0x79, read_addry, adc)
|
||||||
opXI (0x7a, pull, regs.y)
|
opXI (0x7a, pull, r.y)
|
||||||
opAII(0x7b, transfer_w, regs.d, regs.a)
|
opAII(0x7b, transfer_w, r.d, r.a)
|
||||||
opA (0x7c, jmp_iaddrx)
|
opA (0x7c, jmp_iaddrx)
|
||||||
opMF (0x7d, read_addrx, adc)
|
opMF (0x7d, read_addrx, adc)
|
||||||
opMF (0x7e, adjust_addrx, ror)
|
opMF (0x7e, adjust_addrx, ror)
|
||||||
|
@ -148,34 +146,34 @@ auto R65816::instruction() -> void {
|
||||||
opM (0x81, sta_idpx)
|
opM (0x81, sta_idpx)
|
||||||
opA (0x82, brl)
|
opA (0x82, brl)
|
||||||
opM (0x83, sta_sr)
|
opM (0x83, sta_sr)
|
||||||
opXI (0x84, write_dp, regs.y)
|
opXI (0x84, write_dp, r.y)
|
||||||
opMI (0x85, write_dp, regs.a)
|
opMI (0x85, write_dp, r.a)
|
||||||
opXI (0x86, write_dp, regs.x)
|
opXI (0x86, write_dp, r.x)
|
||||||
opM (0x87, sta_ildp)
|
opM (0x87, sta_ildp)
|
||||||
opXII(0x88, adjust_imm, regs.y, -1)
|
opXII(0x88, adjust_imm, r.y, -1)
|
||||||
opM (0x89, read_bit_const)
|
opM (0x89, read_bit_const)
|
||||||
opMII(0x8a, transfer, regs.x, regs.a)
|
opMII(0x8a, transfer, r.x, r.a)
|
||||||
opA (0x8b, phb)
|
opA (0x8b, phb)
|
||||||
opXI (0x8c, write_addr, regs.y)
|
opXI (0x8c, write_addr, r.y)
|
||||||
opMI (0x8d, write_addr, regs.a)
|
opMI (0x8d, write_addr, r.a)
|
||||||
opXI (0x8e, write_addr, regs.x)
|
opXI (0x8e, write_addr, r.x)
|
||||||
opMI (0x8f, write_longr, regs.z)
|
opMI (0x8f, write_longr, r.z)
|
||||||
opAII(0x90, branch, regs.p.c, 0)
|
opAII(0x90, branch, r.p.c, 0)
|
||||||
opM (0x91, sta_idpy)
|
opM (0x91, sta_idpy)
|
||||||
opM (0x92, sta_idp)
|
opM (0x92, sta_idp)
|
||||||
opM (0x93, sta_isry)
|
opM (0x93, sta_isry)
|
||||||
opXII(0x94, write_dpr, regs.y, regs.x)
|
opXII(0x94, write_dpr, r.y, r.x)
|
||||||
opMII(0x95, write_dpr, regs.a, regs.x)
|
opMII(0x95, write_dpr, r.a, r.x)
|
||||||
opXII(0x96, write_dpr, regs.x, regs.y)
|
opXII(0x96, write_dpr, r.x, r.y)
|
||||||
opM (0x97, sta_ildpy)
|
opM (0x97, sta_ildpy)
|
||||||
opMII(0x98, transfer, regs.y, regs.a)
|
opMII(0x98, transfer, r.y, r.a)
|
||||||
opMII(0x99, write_addrr, regs.a, regs.y)
|
opMII(0x99, write_addrr, r.a, r.y)
|
||||||
opE (0x9a, txs)
|
opA (0x9a, txs)
|
||||||
opXII(0x9b, transfer, regs.x, regs.y)
|
opXII(0x9b, transfer, r.x, r.y)
|
||||||
opMI (0x9c, write_addr, regs.z)
|
opMI (0x9c, write_addr, r.z)
|
||||||
opMII(0x9d, write_addrr, regs.a, regs.x)
|
opMII(0x9d, write_addrr, r.a, r.x)
|
||||||
opMII(0x9e, write_addrr, regs.z, regs.x)
|
opMII(0x9e, write_addrr, r.z, r.x)
|
||||||
opMI (0x9f, write_longr, regs.x)
|
opMI (0x9f, write_longr, r.x)
|
||||||
opXF (0xa0, read_const, ldy)
|
opXF (0xa0, read_const, ldy)
|
||||||
opMF (0xa1, read_idpx, lda)
|
opMF (0xa1, read_idpx, lda)
|
||||||
opXF (0xa2, read_const, ldx)
|
opXF (0xa2, read_const, ldx)
|
||||||
|
@ -184,57 +182,57 @@ auto R65816::instruction() -> void {
|
||||||
opMF (0xa5, read_dp, lda)
|
opMF (0xa5, read_dp, lda)
|
||||||
opXF (0xa6, read_dp, ldx)
|
opXF (0xa6, read_dp, ldx)
|
||||||
opMF (0xa7, read_ildp, lda)
|
opMF (0xa7, read_ildp, lda)
|
||||||
opXII(0xa8, transfer, regs.a, regs.y)
|
opXII(0xa8, transfer, r.a, r.y)
|
||||||
opMF (0xa9, read_const, lda)
|
opMF (0xa9, read_const, lda)
|
||||||
opXII(0xaa, transfer, regs.a, regs.x)
|
opXII(0xaa, transfer, r.a, r.x)
|
||||||
opA (0xab, plb)
|
opA (0xab, plb)
|
||||||
opXF (0xac, read_addr, ldy)
|
opXF (0xac, read_addr, ldy)
|
||||||
opMF (0xad, read_addr, lda)
|
opMF (0xad, read_addr, lda)
|
||||||
opXF (0xae, read_addr, ldx)
|
opXF (0xae, read_addr, ldx)
|
||||||
opMF (0xaf, read_long, lda)
|
opMF (0xaf, read_long, lda)
|
||||||
opAII(0xb0, branch, regs.p.c, 1)
|
opAII(0xb0, branch, r.p.c, 1)
|
||||||
opMF (0xb1, read_idpy, lda)
|
opMF (0xb1, read_idpy, lda)
|
||||||
opMF (0xb2, read_idp, lda)
|
opMF (0xb2, read_idp, lda)
|
||||||
opMF (0xb3, read_isry, lda)
|
opMF (0xb3, read_isry, lda)
|
||||||
opXFI(0xb4, read_dpr, ldy, regs.x)
|
opXFI(0xb4, read_dpr, ldy, r.x)
|
||||||
opMFI(0xb5, read_dpr, lda, regs.x)
|
opMFI(0xb5, read_dpr, lda, r.x)
|
||||||
opXFI(0xb6, read_dpr, ldx, regs.y)
|
opXFI(0xb6, read_dpr, ldx, r.y)
|
||||||
opMF (0xb7, read_ildpy, lda)
|
opMF (0xb7, read_ildpy, lda)
|
||||||
opAII(0xb8, flag, regs.p.v, 0)
|
opAII(0xb8, flag, r.p.v, 0)
|
||||||
opMF (0xb9, read_addry, lda)
|
opMF (0xb9, read_addry, lda)
|
||||||
opX (0xba, tsx)
|
opX (0xba, tsx)
|
||||||
opXII(0xbb, transfer, regs.y, regs.x)
|
opXII(0xbb, transfer, r.y, r.x)
|
||||||
opXF (0xbc, read_addrx, ldy)
|
opXF (0xbc, read_addrx, ldy)
|
||||||
opMF (0xbd, read_addrx, lda)
|
opMF (0xbd, read_addrx, lda)
|
||||||
opXF (0xbe, read_addry, ldx)
|
opXF (0xbe, read_addry, ldx)
|
||||||
opMF (0xbf, read_longx, lda)
|
opMF (0xbf, read_longx, lda)
|
||||||
opXF (0xc0, read_const, cpy)
|
opXF (0xc0, read_const, cpy)
|
||||||
opMF (0xc1, read_idpx, cmp)
|
opMF (0xc1, read_idpx, cmp)
|
||||||
opEI (0xc2, pflag, 0)
|
opAI (0xc2, pflag, 0)
|
||||||
opMF (0xc3, read_sr, cmp)
|
opMF (0xc3, read_sr, cmp)
|
||||||
opXF (0xc4, read_dp, cpy)
|
opXF (0xc4, read_dp, cpy)
|
||||||
opMF (0xc5, read_dp, cmp)
|
opMF (0xc5, read_dp, cmp)
|
||||||
opMF (0xc6, adjust_dp, dec)
|
opMF (0xc6, adjust_dp, dec)
|
||||||
opMF (0xc7, read_ildp, cmp)
|
opMF (0xc7, read_ildp, cmp)
|
||||||
opXII(0xc8, adjust_imm, regs.y, +1)
|
opXII(0xc8, adjust_imm, r.y, +1)
|
||||||
opMF (0xc9, read_const, cmp)
|
opMF (0xc9, read_const, cmp)
|
||||||
opXII(0xca, adjust_imm, regs.x, -1)
|
opXII(0xca, adjust_imm, r.x, -1)
|
||||||
opA (0xcb, wai)
|
opA (0xcb, wai)
|
||||||
opXF (0xcc, read_addr, cpy)
|
opXF (0xcc, read_addr, cpy)
|
||||||
opMF (0xcd, read_addr, cmp)
|
opMF (0xcd, read_addr, cmp)
|
||||||
opMF (0xce, adjust_addr, dec)
|
opMF (0xce, adjust_addr, dec)
|
||||||
opMF (0xcf, read_long, cmp)
|
opMF (0xcf, read_long, cmp)
|
||||||
opAII(0xd0, branch, regs.p.z, 0)
|
opAII(0xd0, branch, r.p.z, 0)
|
||||||
opMF (0xd1, read_idpy, cmp)
|
opMF (0xd1, read_idpy, cmp)
|
||||||
opMF (0xd2, read_idp, cmp)
|
opMF (0xd2, read_idp, cmp)
|
||||||
opMF (0xd3, read_isry, cmp)
|
opMF (0xd3, read_isry, cmp)
|
||||||
opE (0xd4, pei)
|
opA (0xd4, pei)
|
||||||
opMFI(0xd5, read_dpr, cmp, regs.x)
|
opMFI(0xd5, read_dpr, cmp, r.x)
|
||||||
opMF (0xd6, adjust_dpx, dec)
|
opMF (0xd6, adjust_dpx, dec)
|
||||||
opMF (0xd7, read_ildpy, cmp)
|
opMF (0xd7, read_ildpy, cmp)
|
||||||
opAII(0xd8, flag, regs.p.d, 0)
|
opAII(0xd8, flag, r.p.d, 0)
|
||||||
opMF (0xd9, read_addry, cmp)
|
opMF (0xd9, read_addry, cmp)
|
||||||
opXI (0xda, push, regs.x)
|
opXI (0xda, push, r.x)
|
||||||
opA (0xdb, stp)
|
opA (0xdb, stp)
|
||||||
opA (0xdc, jmp_iladdr)
|
opA (0xdc, jmp_iladdr)
|
||||||
opMF (0xdd, read_addrx, cmp)
|
opMF (0xdd, read_addrx, cmp)
|
||||||
|
@ -242,13 +240,13 @@ auto R65816::instruction() -> void {
|
||||||
opMF (0xdf, read_longx, cmp)
|
opMF (0xdf, read_longx, cmp)
|
||||||
opXF (0xe0, read_const, cpx)
|
opXF (0xe0, read_const, cpx)
|
||||||
opMF (0xe1, read_idpx, sbc)
|
opMF (0xe1, read_idpx, sbc)
|
||||||
opEI (0xe2, pflag, 1)
|
opAI (0xe2, pflag, 1)
|
||||||
opMF (0xe3, read_sr, sbc)
|
opMF (0xe3, read_sr, sbc)
|
||||||
opXF (0xe4, read_dp, cpx)
|
opXF (0xe4, read_dp, cpx)
|
||||||
opMF (0xe5, read_dp, sbc)
|
opMF (0xe5, read_dp, sbc)
|
||||||
opMF (0xe6, adjust_dp, inc)
|
opMF (0xe6, adjust_dp, inc)
|
||||||
opMF (0xe7, read_ildp, sbc)
|
opMF (0xe7, read_ildp, sbc)
|
||||||
opXII(0xe8, adjust_imm, regs.x, +1)
|
opXII(0xe8, adjust_imm, r.x, +1)
|
||||||
opMF (0xe9, read_const, sbc)
|
opMF (0xe9, read_const, sbc)
|
||||||
opA (0xea, nop)
|
opA (0xea, nop)
|
||||||
opA (0xeb, xba)
|
opA (0xeb, xba)
|
||||||
|
@ -256,29 +254,27 @@ auto R65816::instruction() -> void {
|
||||||
opMF (0xed, read_addr, sbc)
|
opMF (0xed, read_addr, sbc)
|
||||||
opMF (0xee, adjust_addr, inc)
|
opMF (0xee, adjust_addr, inc)
|
||||||
opMF (0xef, read_long, sbc)
|
opMF (0xef, read_long, sbc)
|
||||||
opAII(0xf0, branch, regs.p.z, 1)
|
opAII(0xf0, branch, r.p.z, 1)
|
||||||
opMF (0xf1, read_idpy, sbc)
|
opMF (0xf1, read_idpy, sbc)
|
||||||
opMF (0xf2, read_idp, sbc)
|
opMF (0xf2, read_idp, sbc)
|
||||||
opMF (0xf3, read_isry, sbc)
|
opMF (0xf3, read_isry, sbc)
|
||||||
opE (0xf4, pea)
|
opA (0xf4, pea)
|
||||||
opMFI(0xf5, read_dpr, sbc, regs.x)
|
opMFI(0xf5, read_dpr, sbc, r.x)
|
||||||
opMF (0xf6, adjust_dpx, inc)
|
opMF (0xf6, adjust_dpx, inc)
|
||||||
opMF (0xf7, read_ildpy, sbc)
|
opMF (0xf7, read_ildpy, sbc)
|
||||||
opAII(0xf8, flag, regs.p.d, 1)
|
opAII(0xf8, flag, r.p.d, 1)
|
||||||
opMF (0xf9, read_addry, sbc)
|
opMF (0xf9, read_addry, sbc)
|
||||||
opXI (0xfa, pull, regs.x)
|
opXI (0xfa, pull, r.x)
|
||||||
opA (0xfb, xce)
|
opA (0xfb, xce)
|
||||||
opE (0xfc, jsr_iaddrx)
|
opA (0xfc, jsr_iaddrx)
|
||||||
opMF (0xfd, read_addrx, sbc)
|
opMF (0xfd, read_addrx, sbc)
|
||||||
opMF (0xfe, adjust_addrx, inc)
|
opMF (0xfe, adjust_addrx, inc)
|
||||||
opMF (0xff, read_longx, sbc)
|
opMF (0xff, read_longx, sbc)
|
||||||
}
|
}
|
||||||
|
|
||||||
#undef opA
|
#undef opA
|
||||||
|
#undef opAI
|
||||||
#undef opAII
|
#undef opAII
|
||||||
#undef opE
|
|
||||||
#undef opEI
|
|
||||||
#undef opEII
|
|
||||||
#undef opM
|
#undef opM
|
||||||
#undef opMF
|
#undef opMF
|
||||||
#undef opMFI
|
#undef opMFI
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
struct Gamepad : Controller {
|
struct Gamepad : Controller {
|
||||||
enum : uint {
|
enum : uint {
|
||||||
B, Y, Select, Start, Up, Down, Left, Right, A, X, L, R,
|
Up, Down, Left, Right, B, A, Y, X, L, R, Select, Start,
|
||||||
};
|
};
|
||||||
|
|
||||||
Gamepad(bool port);
|
Gamepad(bool port);
|
||||||
|
|
|
@ -25,7 +25,7 @@ auto ICD2::main() -> void {
|
||||||
GameBoy::system._clocksExecuted = 0;
|
GameBoy::system._clocksExecuted = 0;
|
||||||
} else { //DMG halted
|
} else { //DMG halted
|
||||||
stream->sample(0, 0);
|
stream->sample(0, 0);
|
||||||
step(1);
|
step(2); //two clocks per audio sample
|
||||||
}
|
}
|
||||||
synchronizeCPU();
|
synchronizeCPU();
|
||||||
}
|
}
|
||||||
|
@ -54,7 +54,7 @@ auto ICD2::power() -> void {
|
||||||
|
|
||||||
auto ICD2::reset(bool soft) -> void {
|
auto ICD2::reset(bool soft) -> void {
|
||||||
create(ICD2::Enter, cpu.frequency / 5);
|
create(ICD2::Enter, cpu.frequency / 5);
|
||||||
if(!soft) stream = Emulator::audio.createStream(2, 4194304.0 / 2.0);
|
if(!soft) stream = Emulator::audio.createStream(2, cpu.frequency / 10);
|
||||||
|
|
||||||
r6003 = 0x00;
|
r6003 = 0x00;
|
||||||
r6004 = 0xff;
|
r6004 = 0xff;
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
|
|
||||||
auto SA1::dma_normal() -> void {
|
auto SA1::dma_normal() -> void {
|
||||||
while(mmio.dtc--) {
|
while(mmio.dtc--) {
|
||||||
uint8 data = regs.mdr;
|
uint8 data = r.mdr;
|
||||||
uint32 dsa = mmio.dsa++;
|
uint32 dsa = mmio.dsa++;
|
||||||
uint32 dda = mmio.dda++;
|
uint32 dda = mmio.dda++;
|
||||||
|
|
||||||
|
@ -61,7 +61,7 @@ auto SA1::dma_cc1() -> void {
|
||||||
mmio.chdma_irqfl = true;
|
mmio.chdma_irqfl = true;
|
||||||
if(mmio.chdma_irqen) {
|
if(mmio.chdma_irqen) {
|
||||||
mmio.chdma_irqcl = 0;
|
mmio.chdma_irqcl = 0;
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -115,13 +115,13 @@ auto SA1::io() -> void {
|
||||||
auto SA1::read(uint24 addr) -> uint8 {
|
auto SA1::read(uint24 addr) -> uint8 {
|
||||||
tick();
|
tick();
|
||||||
if(((addr & 0x40e000) == 0x006000) || ((addr & 0xd00000) == 0x400000)) tick();
|
if(((addr & 0x40e000) == 0x006000) || ((addr & 0xd00000) == 0x400000)) tick();
|
||||||
return bus_read(addr, regs.mdr);
|
return bus_read(addr, r.mdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SA1::write(uint24 addr, uint8 data) -> void {
|
auto SA1::write(uint24 addr, uint8 data) -> void {
|
||||||
tick();
|
tick();
|
||||||
if(((addr & 0x40e000) == 0x006000) || ((addr & 0xd00000) == 0x400000)) tick();
|
if(((addr & 0x40e000) == 0x006000) || ((addr & 0xd00000) == 0x400000)) tick();
|
||||||
bus_write(addr, regs.mdr = data);
|
bus_write(addr, r.mdr = data);
|
||||||
}
|
}
|
||||||
|
|
||||||
//note: addresses are translated prior to invoking this function:
|
//note: addresses are translated prior to invoking this function:
|
||||||
|
|
|
@ -2,8 +2,8 @@
|
||||||
auto SA1::mmio_w2200(uint8 data) -> void {
|
auto SA1::mmio_w2200(uint8 data) -> void {
|
||||||
if(mmio.sa1_resb && !(data & 0x80)) {
|
if(mmio.sa1_resb && !(data & 0x80)) {
|
||||||
//reset SA-1 CPU
|
//reset SA-1 CPU
|
||||||
regs.pc.w = mmio.crv;
|
r.pc.w = mmio.crv;
|
||||||
regs.pc.b = 0x00;
|
r.pc.b = 0x00;
|
||||||
}
|
}
|
||||||
|
|
||||||
mmio.sa1_irq = (data & 0x80);
|
mmio.sa1_irq = (data & 0x80);
|
||||||
|
@ -28,14 +28,14 @@ auto SA1::mmio_w2201(uint8 data) -> void {
|
||||||
if(!mmio.cpu_irqen && (data & 0x80)) {
|
if(!mmio.cpu_irqen && (data & 0x80)) {
|
||||||
if(mmio.cpu_irqfl) {
|
if(mmio.cpu_irqfl) {
|
||||||
mmio.cpu_irqcl = 0;
|
mmio.cpu_irqcl = 0;
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!mmio.chdma_irqen && (data & 0x20)) {
|
if(!mmio.chdma_irqen && (data & 0x20)) {
|
||||||
if(mmio.chdma_irqfl) {
|
if(mmio.chdma_irqfl) {
|
||||||
mmio.chdma_irqcl = 0;
|
mmio.chdma_irqcl = 0;
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -51,7 +51,7 @@ auto SA1::mmio_w2202(uint8 data) -> void {
|
||||||
if(mmio.cpu_irqcl ) mmio.cpu_irqfl = false;
|
if(mmio.cpu_irqcl ) mmio.cpu_irqfl = false;
|
||||||
if(mmio.chdma_irqcl) mmio.chdma_irqfl = false;
|
if(mmio.chdma_irqcl) mmio.chdma_irqfl = false;
|
||||||
|
|
||||||
if(!mmio.cpu_irqfl && !mmio.chdma_irqfl) cpu.regs.irq = 0;
|
if(!mmio.cpu_irqfl && !mmio.chdma_irqfl) cpu.r.irq = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
//(CRV) SA-1 reset vector
|
//(CRV) SA-1 reset vector
|
||||||
|
@ -77,7 +77,7 @@ auto SA1::mmio_w2209(uint8 data) -> void {
|
||||||
mmio.cpu_irqfl = true;
|
mmio.cpu_irqfl = true;
|
||||||
if(mmio.cpu_irqen) {
|
if(mmio.cpu_irqen) {
|
||||||
mmio.cpu_irqcl = 0;
|
mmio.cpu_irqcl = 0;
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,41 +32,41 @@ auto SA1::main() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SA1::interrupt() -> void {
|
auto SA1::interrupt() -> void {
|
||||||
read(regs.pc.d);
|
read(r.pc.d);
|
||||||
io();
|
io();
|
||||||
if(!regs.e) writestack(regs.pc.b);
|
if(!r.e) writeSP(r.pc.b);
|
||||||
writestack(regs.pc.h);
|
writeSP(r.pc.h);
|
||||||
writestack(regs.pc.l);
|
writeSP(r.pc.l);
|
||||||
writestack(regs.e ? (regs.p & ~0x10) : regs.p);
|
writeSP(r.e ? (r.p & ~0x10) : r.p);
|
||||||
regs.pc.w = regs.vector;
|
r.pc.w = r.vector;
|
||||||
regs.pc.b = 0x00;
|
r.pc.b = 0x00;
|
||||||
regs.p.i = 1;
|
r.p.i = 1;
|
||||||
regs.p.d = 0;
|
r.p.d = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SA1::lastCycle() -> void {
|
auto SA1::lastCycle() -> void {
|
||||||
if(mmio.sa1_nmi && !mmio.sa1_nmicl) {
|
if(mmio.sa1_nmi && !mmio.sa1_nmicl) {
|
||||||
status.interrupt_pending = true;
|
status.interrupt_pending = true;
|
||||||
regs.vector = mmio.cnv;
|
r.vector = mmio.cnv;
|
||||||
mmio.sa1_nmifl = true;
|
mmio.sa1_nmifl = true;
|
||||||
mmio.sa1_nmicl = 1;
|
mmio.sa1_nmicl = 1;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
} else if(!regs.p.i) {
|
} else if(!r.p.i) {
|
||||||
if(mmio.timer_irqen && !mmio.timer_irqcl) {
|
if(mmio.timer_irqen && !mmio.timer_irqcl) {
|
||||||
status.interrupt_pending = true;
|
status.interrupt_pending = true;
|
||||||
regs.vector = mmio.civ;
|
r.vector = mmio.civ;
|
||||||
mmio.timer_irqfl = true;
|
mmio.timer_irqfl = true;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
} else if(mmio.dma_irqen && !mmio.dma_irqcl) {
|
} else if(mmio.dma_irqen && !mmio.dma_irqcl) {
|
||||||
status.interrupt_pending = true;
|
status.interrupt_pending = true;
|
||||||
regs.vector = mmio.civ;
|
r.vector = mmio.civ;
|
||||||
mmio.dma_irqfl = true;
|
mmio.dma_irqfl = true;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
} else if(mmio.sa1_irq && !mmio.sa1_irqcl) {
|
} else if(mmio.sa1_irq && !mmio.sa1_irqcl) {
|
||||||
status.interrupt_pending = true;
|
status.interrupt_pending = true;
|
||||||
regs.vector = mmio.civ;
|
r.vector = mmio.civ;
|
||||||
mmio.sa1_irqfl = true;
|
mmio.sa1_irqfl = true;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -124,8 +124,10 @@ auto SA1::unload() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SA1::power() -> void {
|
auto SA1::power() -> void {
|
||||||
regs.a = regs.x = regs.y = 0x0000;
|
r.a = 0x0000;
|
||||||
regs.s = 0x01ff;
|
r.x = 0x0000;
|
||||||
|
r.y = 0x0000;
|
||||||
|
r.s = 0x01ff;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SA1::reset() -> void {
|
auto SA1::reset() -> void {
|
||||||
|
@ -136,17 +138,17 @@ auto SA1::reset() -> void {
|
||||||
iram.write(addr, 0x00);
|
iram.write(addr, 0x00);
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.pc.d = 0x000000;
|
r.pc.d = 0x000000;
|
||||||
regs.x.h = 0x00;
|
r.x.h = 0x00;
|
||||||
regs.y.h = 0x00;
|
r.y.h = 0x00;
|
||||||
regs.s.h = 0x01;
|
r.s.h = 0x01;
|
||||||
regs.d = 0x0000;
|
r.d = 0x0000;
|
||||||
regs.db = 0x00;
|
r.db = 0x00;
|
||||||
regs.p = 0x34;
|
r.p = 0x34;
|
||||||
regs.e = 1;
|
r.e = 1;
|
||||||
regs.mdr = 0x00;
|
r.mdr = 0x00;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
regs.vector = 0x0000;
|
r.vector = 0x0000;
|
||||||
|
|
||||||
status.tick_counter = 0;
|
status.tick_counter = 0;
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
auto SuperFX::stop() -> void {
|
auto SuperFX::stop() -> void {
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SuperFX::color(uint8 source) -> uint8 {
|
auto SuperFX::color(uint8 source) -> uint8 {
|
||||||
|
|
|
@ -18,7 +18,7 @@ auto SuperFX::mmio_read(uint24 addr, uint8) -> uint8 {
|
||||||
case 0x3031: {
|
case 0x3031: {
|
||||||
uint8 r = regs.sfr >> 8;
|
uint8 r = regs.sfr >> 8;
|
||||||
regs.sfr.irq = 0;
|
regs.sfr.irq = 0;
|
||||||
cpu.regs.irq = 0;
|
cpu.r.irq = 0;
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,28 +60,28 @@ auto CPU::main() -> void {
|
||||||
status.interrupt_pending = false;
|
status.interrupt_pending = false;
|
||||||
if(status.nmi_pending) {
|
if(status.nmi_pending) {
|
||||||
status.nmi_pending = false;
|
status.nmi_pending = false;
|
||||||
regs.vector = !regs.e ? 0xffea : 0xfffa;
|
r.vector = r.e ? 0xfffa : 0xffea;
|
||||||
interrupt();
|
interrupt();
|
||||||
debugger.op_nmi();
|
debugger.op_nmi();
|
||||||
} else if(status.irq_pending) {
|
} else if(status.irq_pending) {
|
||||||
status.irq_pending = false;
|
status.irq_pending = false;
|
||||||
regs.vector = !regs.e ? 0xffee : 0xfffe;
|
r.vector = r.e ? 0xfffe : 0xffee;
|
||||||
interrupt();
|
interrupt();
|
||||||
debugger.op_irq();
|
debugger.op_irq();
|
||||||
} else if(status.reset_pending) {
|
} else if(status.reset_pending) {
|
||||||
status.reset_pending = false;
|
status.reset_pending = false;
|
||||||
addClocks(132);
|
addClocks(132);
|
||||||
regs.vector = 0xfffc;
|
r.vector = 0xfffc;
|
||||||
interrupt();
|
interrupt();
|
||||||
} else if(status.power_pending) {
|
} else if(status.power_pending) {
|
||||||
status.power_pending = false;
|
status.power_pending = false;
|
||||||
addClocks(186);
|
addClocks(186);
|
||||||
regs.pc.l = bus.read(0xfffc, regs.mdr);
|
r.pc.l = bus.read(0xfffc, r.mdr);
|
||||||
regs.pc.h = bus.read(0xfffd, regs.mdr);
|
r.pc.h = bus.read(0xfffd, r.mdr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
debugger.op_exec(regs.pc.d);
|
debugger.op_exec(r.pc.d);
|
||||||
instruction();
|
instruction();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,8 +89,10 @@ auto CPU::power() -> void {
|
||||||
for(auto& byte : wram) byte = random(0x55);
|
for(auto& byte : wram) byte = random(0x55);
|
||||||
|
|
||||||
//CPU
|
//CPU
|
||||||
regs.a = regs.x = regs.y = 0x0000;
|
r.a = 0x0000;
|
||||||
regs.s = 0x01ff;
|
r.x = 0x0000;
|
||||||
|
r.y = 0x0000;
|
||||||
|
r.s = 0x01ff;
|
||||||
|
|
||||||
//DMA
|
//DMA
|
||||||
for(auto& channel : this->channel) {
|
for(auto& channel : this->channel) {
|
||||||
|
@ -144,17 +146,17 @@ auto CPU::reset() -> void {
|
||||||
bus.map(reader, writer, "7e-7f:0000-ffff", 0x20000);
|
bus.map(reader, writer, "7e-7f:0000-ffff", 0x20000);
|
||||||
|
|
||||||
//CPU
|
//CPU
|
||||||
regs.pc = 0x000000;
|
r.pc = 0x000000;
|
||||||
regs.x.h = 0x00;
|
r.x.h = 0x00;
|
||||||
regs.y.h = 0x00;
|
r.y.h = 0x00;
|
||||||
regs.s.h = 0x01;
|
r.s.h = 0x01;
|
||||||
regs.d = 0x0000;
|
r.d = 0x0000;
|
||||||
regs.db = 0x00;
|
r.db = 0x00;
|
||||||
regs.p = 0x34;
|
r.p = 0x34;
|
||||||
regs.e = 1;
|
r.e = 1;
|
||||||
regs.mdr = 0x00;
|
r.mdr = 0x00;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
regs.vector = 0xfffc; //reset vector address
|
r.vector = 0xfffc; //reset vector address
|
||||||
|
|
||||||
//$2140-217f
|
//$2140-217f
|
||||||
for(auto& port : status.port) port = 0x00;
|
for(auto& port : status.port) port = 0x00;
|
||||||
|
|
|
@ -24,7 +24,7 @@ auto CPU::dmaAddressValid(uint24 abus) -> bool {
|
||||||
|
|
||||||
auto CPU::dmaRead(uint24 abus) -> uint8 {
|
auto CPU::dmaRead(uint24 abus) -> uint8 {
|
||||||
if(!dmaAddressValid(abus)) return 0x00;
|
if(!dmaAddressValid(abus)) return 0x00;
|
||||||
return bus.read(abus, regs.mdr);
|
return bus.read(abus, r.mdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
//simulate two-stage pipeline for DMA transfers; example:
|
//simulate two-stage pipeline for DMA transfers; example:
|
||||||
|
@ -42,14 +42,14 @@ auto CPU::dmaWrite(bool valid, uint addr, uint8 data) -> void {
|
||||||
auto CPU::dmaTransfer(bool direction, uint8 bbus, uint24 abus) -> void {
|
auto CPU::dmaTransfer(bool direction, uint8 bbus, uint24 abus) -> void {
|
||||||
if(direction == 0) {
|
if(direction == 0) {
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
regs.mdr = dmaRead(abus);
|
r.mdr = dmaRead(abus);
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
dmaWrite(dmaTransferValid(bbus, abus), 0x2100 | bbus, regs.mdr);
|
dmaWrite(dmaTransferValid(bbus, abus), 0x2100 | bbus, r.mdr);
|
||||||
} else {
|
} else {
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
regs.mdr = dmaTransferValid(bbus, abus) ? bus.read(0x2100 | bbus, regs.mdr) : (uint8)0x00;
|
r.mdr = dmaTransferValid(bbus, abus) ? bus.read(0x2100 | bbus, r.mdr) : (uint8)0x00;
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
dmaWrite(dmaAddressValid(abus), abus, regs.mdr);
|
dmaWrite(dmaAddressValid(abus), abus, r.mdr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -155,12 +155,12 @@ auto CPU::dmaRun() -> void {
|
||||||
|
|
||||||
auto CPU::hdmaUpdate(uint n) -> void {
|
auto CPU::hdmaUpdate(uint n) -> void {
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
regs.mdr = dmaRead(channel[n].source_bank << 16 | channel[n].hdma_addr);
|
r.mdr = dmaRead(channel[n].source_bank << 16 | channel[n].hdma_addr);
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
dmaWrite(false);
|
dmaWrite(false);
|
||||||
|
|
||||||
if((channel[n].line_counter & 0x7f) == 0) {
|
if((channel[n].line_counter & 0x7f) == 0) {
|
||||||
channel[n].line_counter = regs.mdr;
|
channel[n].line_counter = r.mdr;
|
||||||
channel[n].hdma_addr++;
|
channel[n].hdma_addr++;
|
||||||
|
|
||||||
channel[n].hdma_completed = channel[n].line_counter == 0;
|
channel[n].hdma_completed = channel[n].line_counter == 0;
|
||||||
|
@ -168,16 +168,16 @@ auto CPU::hdmaUpdate(uint n) -> void {
|
||||||
|
|
||||||
if(channel[n].indirect) {
|
if(channel[n].indirect) {
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
regs.mdr = dmaRead(hdmaAddress(n));
|
r.mdr = dmaRead(hdmaAddress(n));
|
||||||
channel[n].indirect_addr = regs.mdr << 8;
|
channel[n].indirect_addr = r.mdr << 8;
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
dmaWrite(false);
|
dmaWrite(false);
|
||||||
|
|
||||||
if(!channel[n].hdma_completed || hdmaActiveAfter(n)) {
|
if(!channel[n].hdma_completed || hdmaActiveAfter(n)) {
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
regs.mdr = dmaRead(hdmaAddress(n));
|
r.mdr = dmaRead(hdmaAddress(n));
|
||||||
channel[n].indirect_addr >>= 8;
|
channel[n].indirect_addr >>= 8;
|
||||||
channel[n].indirect_addr |= regs.mdr << 8;
|
channel[n].indirect_addr |= r.mdr << 8;
|
||||||
dmaAddClocks(4);
|
dmaAddClocks(4);
|
||||||
dmaWrite(false);
|
dmaWrite(false);
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,13 +90,13 @@ auto CPU::timeup() -> bool {
|
||||||
auto CPU::nmiTest() -> bool {
|
auto CPU::nmiTest() -> bool {
|
||||||
if(!status.nmi_transition) return false;
|
if(!status.nmi_transition) return false;
|
||||||
status.nmi_transition = false;
|
status.nmi_transition = false;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::irqTest() -> bool {
|
auto CPU::irqTest() -> bool {
|
||||||
if(!status.irq_transition && !regs.irq) return false;
|
if(!status.irq_transition && !r.irq) return false;
|
||||||
status.irq_transition = false;
|
status.irq_transition = false;
|
||||||
regs.wai = false;
|
r.wai = false;
|
||||||
return !regs.p.i;
|
return !r.p.i;
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,11 +17,11 @@ auto CPU::read(uint24 addr) -> uint8 {
|
||||||
status.clock_count = speed(addr);
|
status.clock_count = speed(addr);
|
||||||
dmaEdge();
|
dmaEdge();
|
||||||
addClocks(status.clock_count - 4);
|
addClocks(status.clock_count - 4);
|
||||||
regs.mdr = bus.read(addr, regs.mdr);
|
r.mdr = bus.read(addr, r.mdr);
|
||||||
addClocks(4);
|
addClocks(4);
|
||||||
aluEdge();
|
aluEdge();
|
||||||
debugger.op_read(addr, regs.mdr);
|
debugger.op_read(addr, r.mdr);
|
||||||
return regs.mdr;
|
return r.mdr;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::write(uint24 addr, uint8 data) -> void {
|
auto CPU::write(uint24 addr, uint8 data) -> void {
|
||||||
|
@ -29,8 +29,8 @@ auto CPU::write(uint24 addr, uint8 data) -> void {
|
||||||
status.clock_count = speed(addr);
|
status.clock_count = speed(addr);
|
||||||
dmaEdge();
|
dmaEdge();
|
||||||
addClocks(status.clock_count);
|
addClocks(status.clock_count);
|
||||||
bus.write(addr, regs.mdr = data);
|
bus.write(addr, r.mdr = data);
|
||||||
debugger.op_write(addr, regs.mdr);
|
debugger.op_write(addr, r.mdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::speed(uint24 addr) const -> uint {
|
auto CPU::speed(uint24 addr) const -> uint {
|
||||||
|
@ -44,5 +44,5 @@ auto CPU::speed(uint24 addr) const -> uint {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::disassemblerRead(uint24 addr) -> uint8 {
|
auto CPU::disassemblerRead(uint24 addr) -> uint8 {
|
||||||
return bus.read(addr, regs.mdr);
|
return bus.read(addr, r.mdr);
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,16 +8,16 @@ auto CPU::cpuPortRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
|
|
||||||
//WMDATA
|
//WMDATA
|
||||||
if(addr == 0x2180) {
|
if(addr == 0x2180) {
|
||||||
return bus.read(0x7e0000 | status.wram_addr++, regs.mdr);
|
return bus.read(0x7e0000 | status.wram_addr++, r.mdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
//JOYSER0
|
//JOYSER0
|
||||||
//7-2 = MDR
|
//7-2 = MDR
|
||||||
//1-0 = Joypad serial data
|
//1-0 = Joypad serial data
|
||||||
if(addr == 0x4016) {
|
if(addr == 0x4016) {
|
||||||
uint8 r = regs.mdr & 0xfc;
|
uint8 v = r.mdr & 0xfc;
|
||||||
r |= SuperFamicom::peripherals.controllerPort1->data();
|
v |= SuperFamicom::peripherals.controllerPort1->data();
|
||||||
return r;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
//JOYSER1
|
//JOYSER1
|
||||||
|
@ -25,9 +25,9 @@ auto CPU::cpuPortRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
//7-5 = MDR
|
//7-5 = MDR
|
||||||
//4-2 = Always 1 (pins are connected to GND)
|
//4-2 = Always 1 (pins are connected to GND)
|
||||||
//1-0 = Joypad serial data
|
//1-0 = Joypad serial data
|
||||||
uint8 r = (regs.mdr & 0xe0) | 0x1c;
|
uint8 v = (r.mdr & 0xe0) | 0x1c;
|
||||||
r |= SuperFamicom::peripherals.controllerPort2->data();
|
v |= SuperFamicom::peripherals.controllerPort2->data();
|
||||||
return r;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
//RDNMI
|
//RDNMI
|
||||||
|
@ -35,19 +35,19 @@ auto CPU::cpuPortRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
//7 = NMI acknowledge
|
//7 = NMI acknowledge
|
||||||
//6-4 = MDR
|
//6-4 = MDR
|
||||||
//3-0 = CPU (5a22) version
|
//3-0 = CPU (5a22) version
|
||||||
uint8 r = (regs.mdr & 0x70);
|
uint8 v = (r.mdr & 0x70);
|
||||||
r |= (uint8)(rdnmi()) << 7;
|
v |= (uint8)(rdnmi()) << 7;
|
||||||
r |= (cpu_version & 0x0f);
|
v |= (cpu_version & 0x0f);
|
||||||
return r;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
//TIMEUP
|
//TIMEUP
|
||||||
if(addr == 0x4211) {
|
if(addr == 0x4211) {
|
||||||
//7 = IRQ acknowledge
|
//7 = IRQ acknowledge
|
||||||
//6-0 = MDR
|
//6-0 = MDR
|
||||||
uint8 r = (regs.mdr & 0x7f);
|
uint8 v = (r.mdr & 0x7f);
|
||||||
r |= (uint8)(timeup()) << 7;
|
v |= (uint8)(timeup()) << 7;
|
||||||
return r;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
//HVBJOY
|
//HVBJOY
|
||||||
|
@ -56,11 +56,11 @@ auto CPU::cpuPortRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
//6 = HBLANK acknowledge
|
//6 = HBLANK acknowledge
|
||||||
//5-1 = MDR
|
//5-1 = MDR
|
||||||
//0 = JOYPAD acknowledge
|
//0 = JOYPAD acknowledge
|
||||||
uint8 r = (regs.mdr & 0x3e);
|
uint8 v = (r.mdr & 0x3e);
|
||||||
if(status.auto_joypad_active) r |= 0x01;
|
if(status.auto_joypad_active) v |= 0x01;
|
||||||
if(hcounter() <= 2 || hcounter() >= 1096) r |= 0x40; //hblank
|
if(hcounter() <= 2 || hcounter() >= 1096) v |= 0x40; //hblank
|
||||||
if(vcounter() >= ppu.vdisp()) r |= 0x80; //vblank
|
if(vcounter() >= ppu.vdisp()) v |= 0x80; //vblank
|
||||||
return r;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
//RDIO
|
//RDIO
|
||||||
|
|
|
@ -28,15 +28,15 @@ auto SuperDisc::Enter() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SuperDisc::main() -> void {
|
auto SuperDisc::main() -> void {
|
||||||
cpu.regs.irq = 0;
|
cpu.r.irq = 0;
|
||||||
|
|
||||||
if(r.irqEnable.bit(3)) {
|
if(r.irqEnable.bit(3)) {
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
nec.data = necPollIRQ();
|
nec.data = necPollIRQ();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(r.irqEnable.bit(2)) {
|
if(r.irqEnable.bit(2)) {
|
||||||
cpu.regs.irq = 1;
|
cpu.r.irq = 1;
|
||||||
sony.data = sonyPollIRQ();
|
sony.data = sonyPollIRQ();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@ auto SuperDisc::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
}
|
}
|
||||||
|
|
||||||
if(addr == 0x21e1) {
|
if(addr == 0x21e1) {
|
||||||
cpu.regs.irq = 0;
|
cpu.r.irq = 0;
|
||||||
data = necReadData();
|
data = necReadData();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,7 +61,7 @@ auto SuperDisc::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
}
|
}
|
||||||
|
|
||||||
if(addr == 0x21e3) {
|
if(addr == 0x21e3) {
|
||||||
cpu.regs.irq = 0;
|
cpu.r.irq = 0;
|
||||||
data = sonyReadData();
|
data = sonyReadData();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,113 +26,105 @@ Interface::Interface() {
|
||||||
media.append({ID::SuperFamicom, "Sufami Turbo", "st", false});
|
media.append({ID::SuperFamicom, "Sufami Turbo", "st", false});
|
||||||
|
|
||||||
{ Device device{0, ID::ControllerPort1 | ID::ControllerPort2 | ID::ExpansionPort, "None"};
|
{ Device device{0, ID::ControllerPort1 | ID::ControllerPort2 | ID::ExpansionPort, "None"};
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{1, ID::ControllerPort1 | ID::ControllerPort2, "Gamepad"};
|
{ Device device{1, ID::ControllerPort1 | ID::ControllerPort2, "Gamepad"};
|
||||||
device.input.append({ 0, 0, "B" });
|
device.inputs.append({ 0, 0, "Up" });
|
||||||
device.input.append({ 1, 0, "Y" });
|
device.inputs.append({ 1, 0, "Down" });
|
||||||
device.input.append({ 2, 0, "Select"});
|
device.inputs.append({ 2, 0, "Left" });
|
||||||
device.input.append({ 3, 0, "Start" });
|
device.inputs.append({ 3, 0, "Right" });
|
||||||
device.input.append({ 4, 0, "Up" });
|
device.inputs.append({ 4, 0, "B" });
|
||||||
device.input.append({ 5, 0, "Down" });
|
device.inputs.append({ 5, 0, "A" });
|
||||||
device.input.append({ 6, 0, "Left" });
|
device.inputs.append({ 6, 0, "Y" });
|
||||||
device.input.append({ 7, 0, "Right" });
|
device.inputs.append({ 7, 0, "X" });
|
||||||
device.input.append({ 8, 0, "A" });
|
device.inputs.append({ 8, 0, "L" });
|
||||||
device.input.append({ 9, 0, "X" });
|
device.inputs.append({ 9, 0, "R" });
|
||||||
device.input.append({10, 0, "L" });
|
device.inputs.append({10, 0, "Select"});
|
||||||
device.input.append({11, 0, "R" });
|
device.inputs.append({11, 0, "Start" });
|
||||||
device.order = {4, 5, 6, 7, 0, 8, 1, 9, 10, 11, 2, 3};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{2, ID::ControllerPort1 | ID::ControllerPort2, "Multitap"};
|
{ Device device{2, ID::ControllerPort1 | ID::ControllerPort2, "Multitap"};
|
||||||
for(uint p = 1, n = 0; p <= 4; p++, n += 12) {
|
for(uint p = 1, n = 0; p <= 4; p++, n += 12) {
|
||||||
device.input.append({n + 0, 0, {"Port ", p, " - ", "B" }});
|
device.inputs.append({n + 0, 0, {"Port ", p, " - ", "Up" }});
|
||||||
device.input.append({n + 1, 0, {"Port ", p, " - ", "Y" }});
|
device.inputs.append({n + 1, 0, {"Port ", p, " - ", "Down" }});
|
||||||
device.input.append({n + 2, 0, {"Port ", p, " - ", "Select"}});
|
device.inputs.append({n + 2, 0, {"Port ", p, " - ", "Left" }});
|
||||||
device.input.append({n + 3, 0, {"Port ", p, " - ", "Start" }});
|
device.inputs.append({n + 3, 0, {"Port ", p, " - ", "Right" }});
|
||||||
device.input.append({n + 4, 0, {"Port ", p, " - ", "Up" }});
|
device.inputs.append({n + 4, 0, {"Port ", p, " - ", "B" }});
|
||||||
device.input.append({n + 5, 0, {"Port ", p, " - ", "Down" }});
|
device.inputs.append({n + 5, 0, {"Port ", p, " - ", "A" }});
|
||||||
device.input.append({n + 6, 0, {"Port ", p, " - ", "Left" }});
|
device.inputs.append({n + 6, 0, {"Port ", p, " - ", "Y" }});
|
||||||
device.input.append({n + 7, 0, {"Port ", p, " - ", "Right" }});
|
device.inputs.append({n + 7, 0, {"Port ", p, " - ", "X" }});
|
||||||
device.input.append({n + 8, 0, {"Port ", p, " - ", "A" }});
|
device.inputs.append({n + 8, 0, {"Port ", p, " - ", "L" }});
|
||||||
device.input.append({n + 9, 0, {"Port ", p, " - ", "X" }});
|
device.inputs.append({n + 9, 0, {"Port ", p, " - ", "R" }});
|
||||||
device.input.append({n + 10, 0, {"Port ", p, " - ", "L" }});
|
device.inputs.append({n + 10, 0, {"Port ", p, " - ", "Select"}});
|
||||||
device.input.append({n + 11, 0, {"Port ", p, " - ", "R" }});
|
device.inputs.append({n + 11, 0, {"Port ", p, " - ", "Start" }});
|
||||||
device.order.append({n + 4, n + 5, n + 6, n + 7, n + 0, n + 8});
|
|
||||||
device.order.append({n + 1, n + 9, n + 10, n + 11, n + 2, n + 3});
|
|
||||||
}
|
}
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{3, ID::ControllerPort1 | ID::ControllerPort2, "Mouse"};
|
{ Device device{3, ID::ControllerPort1 | ID::ControllerPort2, "Mouse"};
|
||||||
device.input.append({0, 1, "X-axis"});
|
device.inputs.append({0, 1, "X-axis"});
|
||||||
device.input.append({1, 1, "Y-axis"});
|
device.inputs.append({1, 1, "Y-axis"});
|
||||||
device.input.append({2, 0, "Left" });
|
device.inputs.append({2, 0, "Left" });
|
||||||
device.input.append({3, 0, "Right" });
|
device.inputs.append({3, 0, "Right" });
|
||||||
device.order = {0, 1, 2, 3};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{4, ID::ControllerPort2, "Super Scope"};
|
{ Device device{4, ID::ControllerPort2, "Super Scope"};
|
||||||
device.input.append({0, 1, "X-axis" });
|
device.inputs.append({0, 1, "X-axis" });
|
||||||
device.input.append({1, 1, "Y-axis" });
|
device.inputs.append({1, 1, "Y-axis" });
|
||||||
device.input.append({2, 0, "Trigger"});
|
device.inputs.append({2, 0, "Trigger"});
|
||||||
device.input.append({3, 0, "Cursor" });
|
device.inputs.append({3, 0, "Cursor" });
|
||||||
device.input.append({4, 0, "Turbo" });
|
device.inputs.append({4, 0, "Turbo" });
|
||||||
device.input.append({5, 0, "Pause" });
|
device.inputs.append({5, 0, "Pause" });
|
||||||
device.order = {0, 1, 2, 3, 4, 5};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{5, ID::ControllerPort2, "Justifier"};
|
{ Device device{5, ID::ControllerPort2, "Justifier"};
|
||||||
device.input.append({0, 1, "X-axis" });
|
device.inputs.append({0, 1, "X-axis" });
|
||||||
device.input.append({1, 1, "Y-axis" });
|
device.inputs.append({1, 1, "Y-axis" });
|
||||||
device.input.append({2, 0, "Trigger"});
|
device.inputs.append({2, 0, "Trigger"});
|
||||||
device.input.append({3, 0, "Start" });
|
device.inputs.append({3, 0, "Start" });
|
||||||
device.order = {0, 1, 2, 3};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{6, ID::ControllerPort2, "Justifiers"};
|
{ Device device{6, ID::ControllerPort2, "Justifiers"};
|
||||||
device.input.append({0, 1, "Port 1 - X-axis" });
|
device.inputs.append({0, 1, "Port 1 - X-axis" });
|
||||||
device.input.append({1, 1, "Port 1 - Y-axis" });
|
device.inputs.append({1, 1, "Port 1 - Y-axis" });
|
||||||
device.input.append({2, 0, "Port 1 - Trigger"});
|
device.inputs.append({2, 0, "Port 1 - Trigger"});
|
||||||
device.input.append({3, 0, "Port 1 - Start" });
|
device.inputs.append({3, 0, "Port 1 - Start" });
|
||||||
device.order.append({0, 1, 2, 3});
|
device.inputs.append({4, 1, "Port 2 - X-axis" });
|
||||||
device.input.append({4, 1, "Port 2 - X-axis" });
|
device.inputs.append({5, 1, "Port 2 - Y-axis" });
|
||||||
device.input.append({5, 1, "Port 2 - Y-axis" });
|
device.inputs.append({6, 0, "Port 2 - Trigger"});
|
||||||
device.input.append({6, 0, "Port 2 - Trigger"});
|
device.inputs.append({7, 0, "Port 2 - Start" });
|
||||||
device.input.append({7, 0, "Port 2 - Start" });
|
devices.append(device);
|
||||||
device.order.append({4, 5, 6, 7});
|
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{7, ID::ControllerPort1, "Serial USART"};
|
{ Device device{7, ID::ControllerPort1, "Serial USART"};
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{8, ID::ExpansionPort, "Satellaview"};
|
{ Device device{8, ID::ExpansionPort, "Satellaview"};
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{9, ID::ExpansionPort, "Super Disc"};
|
{ Device device{9, ID::ExpansionPort, "Super Disc"};
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{10, ID::ExpansionPort, "21fx"};
|
{ Device device{10, ID::ExpansionPort, "21fx"};
|
||||||
this->device.append(device);
|
devices.append(device);
|
||||||
}
|
}
|
||||||
|
|
||||||
port.append({0, "Controller Port 1"});
|
ports.append({0, "Controller Port 1"});
|
||||||
port.append({1, "Controller Port 2"});
|
ports.append({1, "Controller Port 2"});
|
||||||
port.append({2, "Expansion Port"});
|
ports.append({2, "Expansion Port"});
|
||||||
|
|
||||||
for(auto& device : this->device) {
|
for(auto& device : devices) {
|
||||||
for(auto& port : this->port) {
|
for(auto& port : ports) {
|
||||||
if(device.portmask & (1 << port.id)) {
|
if(device.portmask & (1 << port.id)) {
|
||||||
port.device.append(device);
|
port.devices.append(device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -124,7 +124,7 @@ struct Interface : Emulator::Interface {
|
||||||
auto get(const string& name) -> any override;
|
auto get(const string& name) -> any override;
|
||||||
auto set(const string& name, const any& value) -> bool override;
|
auto set(const string& name, const any& value) -> bool override;
|
||||||
|
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Settings {
|
struct Settings {
|
||||||
|
|
|
@ -13,21 +13,21 @@ auto PPU::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
|
|
||||||
//MPYL
|
//MPYL
|
||||||
case 0x2134: {
|
case 0x2134: {
|
||||||
unsigned result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
uint result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
||||||
regs.ppu1_mdr = (result >> 0);
|
regs.ppu1_mdr = (result >> 0);
|
||||||
return regs.ppu1_mdr;
|
return regs.ppu1_mdr;
|
||||||
}
|
}
|
||||||
|
|
||||||
//MPYM
|
//MPYM
|
||||||
case 0x2135: {
|
case 0x2135: {
|
||||||
unsigned result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
uint result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
||||||
regs.ppu1_mdr = (result >> 8);
|
regs.ppu1_mdr = (result >> 8);
|
||||||
return regs.ppu1_mdr;
|
return regs.ppu1_mdr;
|
||||||
}
|
}
|
||||||
|
|
||||||
//MPYH
|
//MPYH
|
||||||
case 0x2136: {
|
case 0x2136: {
|
||||||
unsigned result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
uint result = ((int16)regs.m7a * (int8)(regs.m7b >> 8));
|
||||||
regs.ppu1_mdr = (result >> 16);
|
regs.ppu1_mdr = (result >> 16);
|
||||||
return regs.ppu1_mdr;
|
return regs.ppu1_mdr;
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,7 @@ auto PPU::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
//SLHV
|
//SLHV
|
||||||
case 0x2137: {
|
case 0x2137: {
|
||||||
if(cpu.pio() & 0x80) latchCounters();
|
if(cpu.pio() & 0x80) latchCounters();
|
||||||
return cpu.regs.mdr;
|
return cpu.r.mdr;
|
||||||
}
|
}
|
||||||
|
|
||||||
//OAMDATAREAD
|
//OAMDATAREAD
|
||||||
|
|
|
@ -6,13 +6,13 @@ auto Program::loadRequest(uint id, string name, string type, bool required) -> v
|
||||||
.openFolder();
|
.openFolder();
|
||||||
if(!directory::exists(location)) return;
|
if(!directory::exists(location)) return;
|
||||||
|
|
||||||
mediaPaths(id) = location;
|
mediumPaths(id) = location;
|
||||||
folderPaths.append(location);
|
folderPaths.append(location);
|
||||||
emulator->load(id);
|
emulator->load(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
||||||
string pathname = mediaPaths(emulator->group(id));
|
string pathname = mediumPaths(emulator->group(id));
|
||||||
string location = {pathname, filename};
|
string location = {pathname, filename};
|
||||||
|
|
||||||
if(filename == "manifest.bml" && pathname && !pathname.endsWith("sys/")) {
|
if(filename == "manifest.bml" && pathname && !pathname.endsWith("sys/")) {
|
||||||
|
@ -36,7 +36,7 @@ auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::saveRequest(uint id, string filename) -> void {
|
auto Program::saveRequest(uint id, string filename) -> void {
|
||||||
string pathname = mediaPaths(emulator->group(id));
|
string pathname = mediumPaths(emulator->group(id));
|
||||||
string location = {pathname, filename};
|
string location = {pathname, filename};
|
||||||
|
|
||||||
//filestream stream{location, file::mode::write};
|
//filestream stream{location, file::mode::write};
|
||||||
|
@ -113,7 +113,7 @@ auto Program::dipSettings(const Markup::Node& node) -> uint {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::path(uint group) -> string {
|
auto Program::path(uint group) -> string {
|
||||||
return mediaPaths(group);
|
return mediumPaths(group);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::notify(string text) -> void {
|
auto Program::notify(string text) -> void {
|
||||||
|
|
|
@ -1,21 +1,21 @@
|
||||||
auto Program::loadMedia(string location) -> void {
|
auto Program::loadMedium(string location) -> void {
|
||||||
location.transform("\\", "/");
|
location.transform("\\", "/");
|
||||||
if(!location.endsWith("/")) location.append("/");
|
if(!location.endsWith("/")) location.append("/");
|
||||||
if(!directory::exists(location)) return;
|
if(!directory::exists(location)) return;
|
||||||
|
|
||||||
string type = suffixname(location).trimLeft(".", 1L);
|
string type = suffixname(location).trimLeft(".", 1L);
|
||||||
for(auto& media : emulator->media) {
|
for(auto& medium : emulator->media) {
|
||||||
if(!media.bootable) continue;
|
if(!medium.bootable) continue;
|
||||||
if(media.type != type) continue;
|
if(medium.type != type) continue;
|
||||||
return loadMedia(media, location);
|
return loadMedium(medium, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::loadMedia(Emulator::Interface::Media& media, string location) -> void {
|
auto Program::loadMedium(Emulator::Interface::Medium& medium, string location) -> void {
|
||||||
unloadMedia();
|
unloadMedium();
|
||||||
|
|
||||||
mediaPaths(0) = locate({media.name, ".sys/"});
|
mediumPaths(0) = locate({medium.name, ".sys/"});
|
||||||
mediaPaths(media.id) = location;
|
mediumPaths(medium.id) = location;
|
||||||
folderPaths.append(location);
|
folderPaths.append(location);
|
||||||
|
|
||||||
emulator->set("Blur Emulation", false);
|
emulator->set("Blur Emulation", false);
|
||||||
|
@ -28,17 +28,17 @@ auto Program::loadMedia(Emulator::Interface::Media& media, string location) -> v
|
||||||
emulator->connect((uint)SFC::Port::Controller2, (uint)SFC::Device::None);
|
emulator->connect((uint)SFC::Port::Controller2, (uint)SFC::Device::None);
|
||||||
emulator->connect((uint)SFC::Port::Expansion, (uint)SFC::Device::None);
|
emulator->connect((uint)SFC::Port::Expansion, (uint)SFC::Device::None);
|
||||||
|
|
||||||
emulator->load(media.id);
|
emulator->load(medium.id);
|
||||||
emulator->power();
|
emulator->power();
|
||||||
|
|
||||||
presentation->setTitle(emulator->title());
|
presentation->setTitle(emulator->title());
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::unloadMedia() -> void {
|
auto Program::unloadMedium() -> void {
|
||||||
if(!emulator->loaded()) return;
|
if(!emulator->loaded()) return;
|
||||||
|
|
||||||
emulator->unload();
|
emulator->unload();
|
||||||
mediaPaths.reset();
|
mediumPaths.reset();
|
||||||
folderPaths.reset();
|
folderPaths.reset();
|
||||||
|
|
||||||
presentation->setTitle("");
|
presentation->setTitle("");
|
|
@ -1,6 +1,6 @@
|
||||||
#include "../loki.hpp"
|
#include "../loki.hpp"
|
||||||
#include "interface.cpp"
|
#include "interface.cpp"
|
||||||
#include "media.cpp"
|
#include "medium.cpp"
|
||||||
unique_pointer<Program> program;
|
unique_pointer<Program> program;
|
||||||
|
|
||||||
Program::Program(lstring args) {
|
Program::Program(lstring args) {
|
||||||
|
@ -38,7 +38,7 @@ Program::Program(lstring args) {
|
||||||
"Legend of Zelda - A Link to the Past, The (USA) (1.0).sfc/"
|
"Legend of Zelda - A Link to the Past, The (USA) (1.0).sfc/"
|
||||||
};
|
};
|
||||||
|
|
||||||
if(directory::exists(location)) loadMedia(location);
|
if(directory::exists(location)) loadMedium(location);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::main() -> void {
|
auto Program::main() -> void {
|
||||||
|
@ -52,7 +52,7 @@ auto Program::main() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::quit() -> void {
|
auto Program::quit() -> void {
|
||||||
unloadMedia();
|
unloadMedium();
|
||||||
video.reset();
|
video.reset();
|
||||||
audio.reset();
|
audio.reset();
|
||||||
input.reset();
|
input.reset();
|
||||||
|
|
|
@ -5,9 +5,9 @@ struct Program : Emulator::Interface::Bind {
|
||||||
auto quit() -> void;
|
auto quit() -> void;
|
||||||
|
|
||||||
//media.cpp
|
//media.cpp
|
||||||
auto loadMedia(string location) -> void;
|
auto loadMedium(string location) -> void;
|
||||||
auto loadMedia(Emulator::Interface::Media& media, string location) -> void;
|
auto loadMedium(Emulator::Interface::Medium& medium, string location) -> void;
|
||||||
auto unloadMedia() -> void;
|
auto unloadMedium() -> void;
|
||||||
|
|
||||||
//interface.cpp
|
//interface.cpp
|
||||||
auto loadRequest(uint id, string name, string type, bool required) -> void override;
|
auto loadRequest(uint id, string name, string type, bool required) -> void override;
|
||||||
|
@ -22,7 +22,7 @@ struct Program : Emulator::Interface::Bind {
|
||||||
auto notify(string text) -> void override;
|
auto notify(string text) -> void override;
|
||||||
|
|
||||||
vector<shared_pointer<HID::Device>> devices;
|
vector<shared_pointer<HID::Device>> devices;
|
||||||
vector<string> mediaPaths;
|
vector<string> mediumPaths;
|
||||||
vector<string> folderPaths;
|
vector<string> folderPaths;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -150,16 +150,15 @@ InputManager::InputManager() {
|
||||||
auto& inputEmulator = emulators.right();
|
auto& inputEmulator = emulators.right();
|
||||||
inputEmulator.name = emulator->information.name;
|
inputEmulator.name = emulator->information.name;
|
||||||
|
|
||||||
for(auto& port : emulator->port) {
|
for(auto& port : emulator->ports) {
|
||||||
inputEmulator.ports.append(InputPort());
|
inputEmulator.ports.append(InputPort());
|
||||||
auto& inputPort = inputEmulator.ports.right();
|
auto& inputPort = inputEmulator.ports.right();
|
||||||
inputPort.name = port.name;
|
inputPort.name = port.name;
|
||||||
for(auto& device : port.device) {
|
for(auto& device : port.devices) {
|
||||||
inputPort.devices.append(InputDevice());
|
inputPort.devices.append(InputDevice());
|
||||||
auto& inputDevice = inputPort.devices.right();
|
auto& inputDevice = inputPort.devices.right();
|
||||||
inputDevice.name = device.name;
|
inputDevice.name = device.name;
|
||||||
for(auto number : device.order) {
|
for(auto& input : device.inputs) {
|
||||||
auto& input = device.input[number];
|
|
||||||
inputDevice.mappings.append(new InputMapping());
|
inputDevice.mappings.append(new InputMapping());
|
||||||
auto& inputMapping = inputDevice.mappings.right();
|
auto& inputMapping = inputDevice.mappings.right();
|
||||||
inputMapping->name = input.name;
|
inputMapping->name = input.name;
|
||||||
|
|
|
@ -17,7 +17,7 @@ Presentation::Presentation() {
|
||||||
.setFilters(string{media.name, "|*.", media.type})
|
.setFilters(string{media.name, "|*.", media.type})
|
||||||
.openFolder();
|
.openFolder();
|
||||||
if(directory::exists(location)) {
|
if(directory::exists(location)) {
|
||||||
program->loadMedia(location);
|
program->loadMedium(location);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
loadBootableMedia.append(item);
|
loadBootableMedia.append(item);
|
||||||
|
@ -29,7 +29,7 @@ Presentation::Presentation() {
|
||||||
libraryMenu.append(MenuItem().setText("Load ROM File ...").onActivate([&] {
|
libraryMenu.append(MenuItem().setText("Load ROM File ...").onActivate([&] {
|
||||||
audio->clear();
|
audio->clear();
|
||||||
if(auto location = execute("icarus", "--import")) {
|
if(auto location = execute("icarus", "--import")) {
|
||||||
program->loadMedia(location.output.strip());
|
program->loadMedium(location.output.strip());
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
libraryMenu.append(MenuItem().setText("Import ROM Files ...").onActivate([&] {
|
libraryMenu.append(MenuItem().setText("Import ROM Files ...").onActivate([&] {
|
||||||
|
@ -40,7 +40,7 @@ Presentation::Presentation() {
|
||||||
systemMenu.setText("System").setVisible(false);
|
systemMenu.setText("System").setVisible(false);
|
||||||
powerSystem.setText("Power").onActivate([&] { program->powerCycle(); });
|
powerSystem.setText("Power").onActivate([&] { program->powerCycle(); });
|
||||||
resetSystem.setText("Reset").onActivate([&] { program->softReset(); });
|
resetSystem.setText("Reset").onActivate([&] { program->softReset(); });
|
||||||
unloadSystem.setText("Unload").onActivate([&] { program->unloadMedia(); drawSplashScreen(); });
|
unloadSystem.setText("Unload").onActivate([&] { program->unloadMedium(); drawSplashScreen(); });
|
||||||
|
|
||||||
settingsMenu.setText("Settings");
|
settingsMenu.setText("Settings");
|
||||||
videoScaleMenu.setText("Video Scale");
|
videoScaleMenu.setText("Video Scale");
|
||||||
|
@ -173,14 +173,14 @@ auto Presentation::updateEmulator() -> void {
|
||||||
inputPort2.setVisible(false).reset();
|
inputPort2.setVisible(false).reset();
|
||||||
inputPort3.setVisible(false).reset();
|
inputPort3.setVisible(false).reset();
|
||||||
|
|
||||||
for(auto n : range(emulator->port)) {
|
for(auto n : range(emulator->ports)) {
|
||||||
if(n >= 3) break;
|
if(n >= 3) break;
|
||||||
auto& port = emulator->port[n];
|
auto& port = emulator->ports[n];
|
||||||
auto& menu = (n == 0 ? inputPort1 : n == 1 ? inputPort2 : inputPort3);
|
auto& menu = (n == 0 ? inputPort1 : n == 1 ? inputPort2 : inputPort3);
|
||||||
menu.setText(port.name);
|
menu.setText(port.name);
|
||||||
|
|
||||||
Group devices;
|
Group devices;
|
||||||
for(auto& device : port.device) {
|
for(auto& device : port.devices) {
|
||||||
MenuRadioItem item{&menu};
|
MenuRadioItem item{&menu};
|
||||||
item.setText(device.name).onActivate([=] {
|
item.setText(device.name).onActivate([=] {
|
||||||
auto path = string{emulator->information.name, "/", port.name}.replace(" ", "");
|
auto path = string{emulator->information.name, "/", port.name}.replace(" ", "");
|
||||||
|
|
|
@ -7,14 +7,14 @@ auto Program::loadRequest(uint id, string name, string type, bool required) -> v
|
||||||
.openFolder();
|
.openFolder();
|
||||||
if(!directory::exists(location)) return;
|
if(!directory::exists(location)) return;
|
||||||
|
|
||||||
mediaPaths(id) = location;
|
mediumPaths(id) = location;
|
||||||
folderPaths.append(location);
|
folderPaths.append(location);
|
||||||
emulator->load(id);
|
emulator->load(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
//request from emulation core to load non-volatile media file
|
//request from emulation core to load non-volatile media file
|
||||||
auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
||||||
string pathname = mediaPaths(emulator->group(id));
|
string pathname = mediumPaths(emulator->group(id));
|
||||||
string location = {pathname, filename};
|
string location = {pathname, filename};
|
||||||
|
|
||||||
if(filename == "manifest.bml" && pathname && !pathname.endsWith(".sys/")) {
|
if(filename == "manifest.bml" && pathname && !pathname.endsWith(".sys/")) {
|
||||||
|
@ -39,7 +39,7 @@ auto Program::loadRequest(uint id, string filename, bool required) -> void {
|
||||||
|
|
||||||
//request from emulation core to save non-volatile media file
|
//request from emulation core to save non-volatile media file
|
||||||
auto Program::saveRequest(uint id, string filename) -> void {
|
auto Program::saveRequest(uint id, string filename) -> void {
|
||||||
string pathname = mediaPaths(emulator->group(id));
|
string pathname = mediumPaths(emulator->group(id));
|
||||||
string location = {pathname, filename};
|
string location = {pathname, filename};
|
||||||
if(!pathname) return; //should never occur
|
if(!pathname) return; //should never occur
|
||||||
|
|
||||||
|
@ -89,19 +89,13 @@ auto Program::videoRefresh(const uint32* data, uint pitch, uint width, uint heig
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::audioSample(int16 lsample, int16 rsample) -> void {
|
auto Program::audioSample(int16 left, int16 right) -> void {
|
||||||
audio->sample(lsample, rsample);
|
audio->sample(left, right);
|
||||||
//int samples[] = {lsample, rsample};
|
|
||||||
//dsp.sample(samples);
|
|
||||||
//while(dsp.pending()) {
|
|
||||||
// dsp.read(samples);
|
|
||||||
// audio->sample(samples[0], samples[1]);
|
|
||||||
//}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::inputPoll(uint port, uint device, uint input) -> int16 {
|
auto Program::inputPoll(uint port, uint device, uint input) -> int16 {
|
||||||
if(presentation->focused() || settings["Input/FocusLoss/AllowInput"].boolean()) {
|
if(presentation->focused() || settings["Input/FocusLoss/AllowInput"].boolean()) {
|
||||||
auto guid = emulator->port[port].device[device].input[input].guid;
|
auto guid = emulator->ports[port].devices[device].inputs[input].guid;
|
||||||
auto mapping = (InputMapping*)guid;
|
auto mapping = (InputMapping*)guid;
|
||||||
if(mapping) return mapping->poll();
|
if(mapping) return mapping->poll();
|
||||||
}
|
}
|
||||||
|
@ -110,7 +104,7 @@ auto Program::inputPoll(uint port, uint device, uint input) -> int16 {
|
||||||
|
|
||||||
auto Program::inputRumble(uint port, uint device, uint input, bool enable) -> void {
|
auto Program::inputRumble(uint port, uint device, uint input, bool enable) -> void {
|
||||||
if(presentation->focused() || settings["Input/FocusLoss/AllowInput"].boolean() || !enable) {
|
if(presentation->focused() || settings["Input/FocusLoss/AllowInput"].boolean() || !enable) {
|
||||||
auto guid = emulator->port[port].device[device].input[input].guid;
|
auto guid = emulator->ports[port].devices[device].inputs[input].guid;
|
||||||
auto mapping = (InputMapping*)guid;
|
auto mapping = (InputMapping*)guid;
|
||||||
if(mapping) return mapping->rumble(enable);
|
if(mapping) return mapping->rumble(enable);
|
||||||
}
|
}
|
||||||
|
@ -121,7 +115,7 @@ auto Program::dipSettings(const Markup::Node& node) -> uint {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::path(uint group) -> string {
|
auto Program::path(uint group) -> string {
|
||||||
return mediaPaths(group);
|
return mediumPaths(group);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::notify(string text) -> void {
|
auto Program::notify(string text) -> void {
|
||||||
|
|
|
@ -1,23 +1,23 @@
|
||||||
auto Program::loadMedia(string location) -> void {
|
auto Program::loadMedium(string location) -> void {
|
||||||
location.transform("\\", "/");
|
location.transform("\\", "/");
|
||||||
if(!location.endsWith("/")) location.append("/");
|
if(!location.endsWith("/")) location.append("/");
|
||||||
if(!directory::exists(location)) return;
|
if(!directory::exists(location)) return;
|
||||||
|
|
||||||
string type = suffixname(location).trimLeft(".", 1L);
|
string type = suffixname(location).trimLeft(".", 1L);
|
||||||
for(auto& emulator : emulators) {
|
for(auto& emulator : emulators) {
|
||||||
for(auto& media : emulator->media) {
|
for(auto& medium : emulator->media) {
|
||||||
if(!media.bootable) continue;
|
if(!medium.bootable) continue;
|
||||||
if(media.type != type) continue;
|
if(medium.type != type) continue;
|
||||||
return loadMedia(*emulator, media, location);
|
return loadMedium(*emulator, medium, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::loadMedia(Emulator::Interface& interface, Emulator::Interface::Media& media, string location) -> void {
|
auto Program::loadMedium(Emulator::Interface& interface, Emulator::Interface::Medium& medium, string location) -> void {
|
||||||
unloadMedia();
|
unloadMedium();
|
||||||
|
|
||||||
mediaPaths(0) = locate({media.name, ".sys/"});
|
mediumPaths(0) = locate({medium.name, ".sys/"});
|
||||||
mediaPaths(media.id) = location;
|
mediumPaths(medium.id) = location;
|
||||||
folderPaths.append(location);
|
folderPaths.append(location);
|
||||||
|
|
||||||
//note: the order of operations in this block of code is critical
|
//note: the order of operations in this block of code is critical
|
||||||
|
@ -25,14 +25,14 @@ auto Program::loadMedia(Emulator::Interface& interface, Emulator::Interface::Med
|
||||||
Emulator::audio.setFrequency(audio->get(Audio::Frequency).get<uint>());
|
Emulator::audio.setFrequency(audio->get(Audio::Frequency).get<uint>());
|
||||||
emulator = &interface;
|
emulator = &interface;
|
||||||
connectDevices();
|
connectDevices();
|
||||||
emulator->load(media.id);
|
emulator->load(medium.id);
|
||||||
updateAudioDriver();
|
updateAudioDriver();
|
||||||
updateAudioEffects();
|
updateAudioEffects();
|
||||||
emulator->power();
|
emulator->power();
|
||||||
|
|
||||||
presentation->resizeViewport();
|
presentation->resizeViewport();
|
||||||
presentation->setTitle(emulator->title());
|
presentation->setTitle(emulator->title());
|
||||||
presentation->systemMenu.setText(media.name).setVisible(true);
|
presentation->systemMenu.setText(medium.name).setVisible(true);
|
||||||
presentation->toolsMenu.setVisible(true);
|
presentation->toolsMenu.setVisible(true);
|
||||||
presentation->updateEmulator();
|
presentation->updateEmulator();
|
||||||
toolsManager->cheatEditor.loadCheats();
|
toolsManager->cheatEditor.loadCheats();
|
||||||
|
@ -40,14 +40,14 @@ auto Program::loadMedia(Emulator::Interface& interface, Emulator::Interface::Med
|
||||||
toolsManager->manifestViewer.doRefresh();
|
toolsManager->manifestViewer.doRefresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::unloadMedia() -> void {
|
auto Program::unloadMedium() -> void {
|
||||||
if(!emulator) return;
|
if(!emulator) return;
|
||||||
|
|
||||||
toolsManager->cheatEditor.saveCheats();
|
toolsManager->cheatEditor.saveCheats();
|
||||||
emulator->unload();
|
emulator->unload();
|
||||||
emulator = nullptr;
|
emulator = nullptr;
|
||||||
|
|
||||||
mediaPaths.reset();
|
mediumPaths.reset();
|
||||||
folderPaths.reset();
|
folderPaths.reset();
|
||||||
|
|
||||||
presentation->setTitle({"higan v", Emulator::Version});
|
presentation->setTitle({"higan v", Emulator::Version});
|
|
@ -5,7 +5,7 @@
|
||||||
#include <gba/interface/interface.hpp>
|
#include <gba/interface/interface.hpp>
|
||||||
#include <ws/interface/interface.hpp>
|
#include <ws/interface/interface.hpp>
|
||||||
#include "interface.cpp"
|
#include "interface.cpp"
|
||||||
#include "media.cpp"
|
#include "medium.cpp"
|
||||||
#include "state.cpp"
|
#include "state.cpp"
|
||||||
#include "utility.cpp"
|
#include "utility.cpp"
|
||||||
unique_pointer<Program> program;
|
unique_pointer<Program> program;
|
||||||
|
@ -64,7 +64,7 @@ Program::Program(lstring args) {
|
||||||
|
|
||||||
auto Program::load(string location) -> void {
|
auto Program::load(string location) -> void {
|
||||||
if(directory::exists(location)) {
|
if(directory::exists(location)) {
|
||||||
loadMedia(location);
|
loadMedium(location);
|
||||||
} else if(file::exists(location)) {
|
} else if(file::exists(location)) {
|
||||||
//special handling to allow importing the Game Boy Advance BIOS
|
//special handling to allow importing the Game Boy Advance BIOS
|
||||||
if(file::size(location) == 16384 && file::sha256(location).beginsWith("fd2547724b505f48")) {
|
if(file::size(location) == 16384 && file::sha256(location).beginsWith("fd2547724b505f48")) {
|
||||||
|
@ -77,7 +77,7 @@ auto Program::load(string location) -> void {
|
||||||
|
|
||||||
//ask icarus to import the game; and play it upon success
|
//ask icarus to import the game; and play it upon success
|
||||||
if(auto result = execute("icarus", "--import", location)) {
|
if(auto result = execute("icarus", "--import", location)) {
|
||||||
loadMedia(result.output.strip());
|
loadMedium(result.output.strip());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -96,7 +96,7 @@ auto Program::main() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Program::quit() -> void {
|
auto Program::quit() -> void {
|
||||||
unloadMedia();
|
unloadMedium();
|
||||||
settings.quit();
|
settings.quit();
|
||||||
inputManager->quit();
|
inputManager->quit();
|
||||||
Application::quit();
|
Application::quit();
|
||||||
|
|
|
@ -17,10 +17,10 @@ struct Program : Emulator::Interface::Bind {
|
||||||
auto path(uint group) -> string override;
|
auto path(uint group) -> string override;
|
||||||
auto notify(string text) -> void override;
|
auto notify(string text) -> void override;
|
||||||
|
|
||||||
//media.cpp
|
//medium.cpp
|
||||||
auto loadMedia(string location) -> void;
|
auto loadMedium(string location) -> void;
|
||||||
auto loadMedia(Emulator::Interface& interface, Emulator::Interface::Media& media, string location) -> void;
|
auto loadMedium(Emulator::Interface& interface, Emulator::Interface::Medium& medium, string location) -> void;
|
||||||
auto unloadMedia() -> void;
|
auto unloadMedium() -> void;
|
||||||
|
|
||||||
//state.cpp
|
//state.cpp
|
||||||
auto stateName(uint slot, bool manager = false) -> string;
|
auto stateName(uint slot, bool manager = false) -> string;
|
||||||
|
@ -42,7 +42,7 @@ struct Program : Emulator::Interface::Bind {
|
||||||
|
|
||||||
vector<Emulator::Interface*> emulators;
|
vector<Emulator::Interface*> emulators;
|
||||||
|
|
||||||
vector<string> mediaPaths;
|
vector<string> mediumPaths;
|
||||||
vector<string> folderPaths;
|
vector<string> folderPaths;
|
||||||
|
|
||||||
string statusText;
|
string statusText;
|
||||||
|
|
|
@ -13,10 +13,10 @@ auto Program::softReset() -> void {
|
||||||
|
|
||||||
auto Program::connectDevices() -> void {
|
auto Program::connectDevices() -> void {
|
||||||
if(!emulator) return;
|
if(!emulator) return;
|
||||||
for(auto& port : emulator->port) {
|
for(auto& port : emulator->ports) {
|
||||||
auto path = string{emulator->information.name, "/", port.name}.replace(" ", "");
|
auto path = string{emulator->information.name, "/", port.name}.replace(" ", "");
|
||||||
auto name = settings(path).text();
|
auto name = settings(path).text();
|
||||||
for(auto& device : port.device) {
|
for(auto& device : port.devices) {
|
||||||
if(device.name == name) {
|
if(device.name == name) {
|
||||||
emulator->connect(port.id, device.id);
|
emulator->connect(port.id, device.id);
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -23,41 +23,39 @@ Interface::Interface() {
|
||||||
media.append({ID::WonderSwanColor, "WonderSwan Color", "wsc", true});
|
media.append({ID::WonderSwanColor, "WonderSwan Color", "wsc", true});
|
||||||
|
|
||||||
{ Device device{0, ID::DeviceHorizontal, "Controller"};
|
{ Device device{0, ID::DeviceHorizontal, "Controller"};
|
||||||
device.input.append({ 0, 0, "Y1"});
|
device.inputs.append({ 0, 0, "Y1"});
|
||||||
device.input.append({ 1, 0, "Y2"});
|
device.inputs.append({ 1, 0, "Y2"});
|
||||||
device.input.append({ 2, 0, "Y3"});
|
device.inputs.append({ 2, 0, "Y3"});
|
||||||
device.input.append({ 3, 0, "Y4"});
|
device.inputs.append({ 3, 0, "Y4"});
|
||||||
device.input.append({ 4, 0, "X1"});
|
device.inputs.append({ 4, 0, "X1"});
|
||||||
device.input.append({ 5, 0, "X2"});
|
device.inputs.append({ 5, 0, "X2"});
|
||||||
device.input.append({ 6, 0, "X3"});
|
device.inputs.append({ 6, 0, "X3"});
|
||||||
device.input.append({ 7, 0, "X4"});
|
device.inputs.append({ 7, 0, "X4"});
|
||||||
device.input.append({ 8, 0, "B"});
|
device.inputs.append({ 8, 0, "B"});
|
||||||
device.input.append({ 9, 0, "A"});
|
device.inputs.append({ 9, 0, "A"});
|
||||||
device.input.append({10, 0, "Start"});
|
device.inputs.append({10, 0, "Start"});
|
||||||
device.input.append({11, 0, "Rotate"});
|
device.inputs.append({11, 0, "Rotate"});
|
||||||
device.order = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{ Device device{1, ID::DeviceVertical, "Controller"};
|
{ Device device{1, ID::DeviceVertical, "Controller"};
|
||||||
device.input.append({ 0, 0, "Y1"});
|
device.inputs.append({ 0, 0, "Y1"});
|
||||||
device.input.append({ 1, 0, "Y2"});
|
device.inputs.append({ 1, 0, "Y2"});
|
||||||
device.input.append({ 2, 0, "Y3"});
|
device.inputs.append({ 2, 0, "Y3"});
|
||||||
device.input.append({ 3, 0, "Y4"});
|
device.inputs.append({ 3, 0, "Y4"});
|
||||||
device.input.append({ 4, 0, "X1"});
|
device.inputs.append({ 4, 0, "X1"});
|
||||||
device.input.append({ 5, 0, "X2"});
|
device.inputs.append({ 5, 0, "X2"});
|
||||||
device.input.append({ 6, 0, "X3"});
|
device.inputs.append({ 6, 0, "X3"});
|
||||||
device.input.append({ 7, 0, "X4"});
|
device.inputs.append({ 7, 0, "X4"});
|
||||||
device.input.append({ 8, 0, "B"});
|
device.inputs.append({ 8, 0, "B"});
|
||||||
device.input.append({ 9, 0, "A"});
|
device.inputs.append({ 9, 0, "A"});
|
||||||
device.input.append({10, 0, "Start"});
|
device.inputs.append({10, 0, "Start"});
|
||||||
device.input.append({11, 0, "Rotate"});
|
device.inputs.append({11, 0, "Rotate"});
|
||||||
device.order = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11};
|
devices.append(device);
|
||||||
this->device.append(device);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
port.append({0, "Horizontal Orientation", {device[0]}});
|
ports.append({0, "Horizontal Orientation", {devices[0]}});
|
||||||
port.append({1, "Vertical Orientation", {device[1]}});
|
ports.append({1, "Vertical Orientation", {devices[1]}});
|
||||||
}
|
}
|
||||||
|
|
||||||
auto Interface::manifest() -> string {
|
auto Interface::manifest() -> string {
|
||||||
|
|
|
@ -57,7 +57,7 @@ struct Interface : Emulator::Interface {
|
||||||
auto set(const string& name, const any& value) -> bool override;
|
auto set(const string& name, const any& value) -> bool override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
vector<Device> device;
|
vector<Device> devices;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Settings {
|
struct Settings {
|
||||||
|
|
|
@ -1,4 +1,8 @@
|
||||||
#define decimal decimal_cocoa
|
#define decimal decimal_cocoa
|
||||||
|
#define int8 int8_cocoa
|
||||||
|
#define int16 int16_cocoa
|
||||||
|
#define int32 int32_cocoa
|
||||||
|
#define int64 int64_cocoa
|
||||||
#define uint8 uint8_cocoa
|
#define uint8 uint8_cocoa
|
||||||
#define uint16 uint16_cocoa
|
#define uint16 uint16_cocoa
|
||||||
#define uint32 uint32_cocoa
|
#define uint32 uint32_cocoa
|
||||||
|
@ -6,6 +10,10 @@
|
||||||
#import <Cocoa/Cocoa.h>
|
#import <Cocoa/Cocoa.h>
|
||||||
#import <Carbon/Carbon.h>
|
#import <Carbon/Carbon.h>
|
||||||
#undef decimal
|
#undef decimal
|
||||||
|
#undef int8
|
||||||
|
#undef int16
|
||||||
|
#undef int32
|
||||||
|
#undef int64
|
||||||
#undef uint8
|
#undef uint8
|
||||||
#undef uint16
|
#undef uint16
|
||||||
#undef uint32
|
#undef uint32
|
||||||
|
|
36
nall/bit.hpp
36
nall/bit.hpp
|
@ -4,28 +4,28 @@
|
||||||
|
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
|
||||||
template<unsigned bits> inline auto uclamp(const uintmax_t x) -> uintmax_t {
|
template<uint bits> inline auto uclamp(const uintmax x) -> uintmax {
|
||||||
enum : uintmax_t { b = 1ull << (bits - 1), y = b * 2 - 1 };
|
enum : uintmax { b = 1ull << (bits - 1), y = b * 2 - 1 };
|
||||||
return y + ((x - y) & -(x < y)); //min(x, y);
|
return y + ((x - y) & -(x < y)); //min(x, y);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned bits> inline auto uclip(const uintmax_t x) -> uintmax_t {
|
template<uint bits> inline auto uclip(const uintmax x) -> uintmax {
|
||||||
enum : uintmax_t { b = 1ull << (bits - 1), m = b * 2 - 1 };
|
enum : uintmax { b = 1ull << (bits - 1), m = b * 2 - 1 };
|
||||||
return (x & m);
|
return (x & m);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned bits> inline auto sclamp(const intmax_t x) -> intmax_t {
|
template<uint bits> inline auto sclamp(const intmax x) -> intmax {
|
||||||
enum : intmax_t { b = 1ull << (bits - 1), m = b - 1 };
|
enum : intmax { b = 1ull << (bits - 1), m = b - 1 };
|
||||||
return (x > m) ? m : (x < -b) ? -b : x;
|
return (x > m) ? m : (x < -b) ? -b : x;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned bits> inline auto sclip(const intmax_t x) -> intmax_t {
|
template<uint bits> inline auto sclip(const intmax x) -> intmax {
|
||||||
enum : uintmax_t { b = 1ull << (bits - 1), m = b * 2 - 1 };
|
enum : uintmax { b = 1ull << (bits - 1), m = b * 2 - 1 };
|
||||||
return ((x & m) ^ b) - b;
|
return ((x & m) ^ b) - b;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace bit {
|
namespace bit {
|
||||||
constexpr inline auto mask(const char* s, uintmax_t sum = 0) -> uintmax_t {
|
constexpr inline auto mask(const char* s, uintmax sum = 0) -> uintmax {
|
||||||
return (
|
return (
|
||||||
*s == '0' || *s == '1' ? mask(s + 1, (sum << 1) | 1) :
|
*s == '0' || *s == '1' ? mask(s + 1, (sum << 1) | 1) :
|
||||||
*s == ' ' || *s == '_' ? mask(s + 1, sum) :
|
*s == ' ' || *s == '_' ? mask(s + 1, sum) :
|
||||||
|
@ -34,7 +34,7 @@ namespace bit {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr inline auto test(const char* s, uintmax_t sum = 0) -> uintmax_t {
|
constexpr inline auto test(const char* s, uintmax sum = 0) -> uintmax {
|
||||||
return (
|
return (
|
||||||
*s == '0' || *s == '1' ? test(s + 1, (sum << 1) | (*s - '0')) :
|
*s == '0' || *s == '1' ? test(s + 1, (sum << 1) | (*s - '0')) :
|
||||||
*s == ' ' || *s == '_' ? test(s + 1, sum) :
|
*s == ' ' || *s == '_' ? test(s + 1, sum) :
|
||||||
|
@ -44,38 +44,38 @@ namespace bit {
|
||||||
}
|
}
|
||||||
|
|
||||||
//lowest(0b1110) == 0b0010
|
//lowest(0b1110) == 0b0010
|
||||||
constexpr inline auto lowest(const uintmax_t x) -> uintmax_t {
|
constexpr inline auto lowest(const uintmax x) -> uintmax {
|
||||||
return x & -x;
|
return x & -x;
|
||||||
}
|
}
|
||||||
|
|
||||||
//clear_lowest(0b1110) == 0b1100
|
//clear_lowest(0b1110) == 0b1100
|
||||||
constexpr inline auto clear_lowest(const uintmax_t x) -> uintmax_t {
|
constexpr inline auto clear_lowest(const uintmax x) -> uintmax {
|
||||||
return x & (x - 1);
|
return x & (x - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
//set_lowest(0b0101) == 0b0111
|
//set_lowest(0b0101) == 0b0111
|
||||||
constexpr inline auto set_lowest(const uintmax_t x) -> uintmax_t {
|
constexpr inline auto set_lowest(const uintmax x) -> uintmax {
|
||||||
return x | (x + 1);
|
return x | (x + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
//count number of bits set in a byte
|
//count number of bits set in a byte
|
||||||
inline auto count(uintmax_t x) -> unsigned {
|
inline auto count(uintmax x) -> uint {
|
||||||
unsigned count = 0;
|
uint count = 0;
|
||||||
do count += x & 1; while(x >>= 1);
|
do count += x & 1; while(x >>= 1);
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
//return index of the first bit set (or zero of no bits are set)
|
//return index of the first bit set (or zero of no bits are set)
|
||||||
//first(0b1000) == 3
|
//first(0b1000) == 3
|
||||||
inline auto first(uintmax_t x) -> unsigned {
|
inline auto first(uintmax x) -> uint {
|
||||||
unsigned first = 0;
|
uint first = 0;
|
||||||
while(x) { if(x & 1) break; x >>= 1; first++; }
|
while(x) { if(x & 1) break; x >>= 1; first++; }
|
||||||
return first;
|
return first;
|
||||||
}
|
}
|
||||||
|
|
||||||
//round up to next highest single bit:
|
//round up to next highest single bit:
|
||||||
//round(15) == 16, round(16) == 16, round(17) == 32
|
//round(15) == 16, round(16) == 16, round(17) == 32
|
||||||
inline auto round(uintmax_t x) -> uintmax_t {
|
inline auto round(uintmax x) -> uintmax {
|
||||||
if((x & (x - 1)) == 0) return x;
|
if((x & (x - 1)) == 0) return x;
|
||||||
while(x & (x - 1)) x &= x - 1;
|
while(x & (x - 1)) x &= x - 1;
|
||||||
return x << 1;
|
return x << 1;
|
||||||
|
|
|
@ -10,7 +10,7 @@ namespace Configuration {
|
||||||
struct Node {
|
struct Node {
|
||||||
string name;
|
string name;
|
||||||
string desc;
|
string desc;
|
||||||
enum class Type : unsigned { Null, Boolean, Integer, Natural, Double, String } type = Type::Null;
|
enum class Type : uint { Null, Boolean, Integer, Natural, Double, String } type = Type::Null;
|
||||||
void* data = nullptr;
|
void* data = nullptr;
|
||||||
vector<Node> children;
|
vector<Node> children;
|
||||||
|
|
||||||
|
@ -74,7 +74,7 @@ struct Node {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto save(file& fp, unsigned depth = 0) -> void {
|
auto save(file& fp, uint depth = 0) -> void {
|
||||||
for(auto& child : children) {
|
for(auto& child : children) {
|
||||||
if(child.desc) {
|
if(child.desc) {
|
||||||
for(auto n : range(depth)) fp.print(" ");
|
for(auto n : range(depth)) fp.print(" ");
|
||||||
|
|
|
@ -65,7 +65,7 @@ inline auto library::close() -> void {
|
||||||
inline auto library::open(const string& name, const string& path) -> bool {
|
inline auto library::open(const string& name, const string& path) -> bool {
|
||||||
if(handle) close();
|
if(handle) close();
|
||||||
if(path) handle = (uintptr_t)dlopen(string(path, "lib", name, ".dylib"), RTLD_LAZY);
|
if(path) handle = (uintptr_t)dlopen(string(path, "lib", name, ".dylib"), RTLD_LAZY);
|
||||||
if(!handle) handle = (uintptr_t)dlopen(string(userpath(), ".local/lib/lib", name, ".dylib"), RTLD_LAZY);
|
if(!handle) handle = (uintptr_t)dlopen(string(Path::user(), ".local/lib/lib", name, ".dylib"), RTLD_LAZY);
|
||||||
if(!handle) handle = (uintptr_t)dlopen(string("/usr/local/lib/lib", name, ".dylib"), RTLD_LAZY);
|
if(!handle) handle = (uintptr_t)dlopen(string("/usr/local/lib/lib", name, ".dylib"), RTLD_LAZY);
|
||||||
if(!handle) handle = (uintptr_t)dlopen(string("lib", name, ".dylib"), RTLD_LAZY);
|
if(!handle) handle = (uintptr_t)dlopen(string("lib", name, ".dylib"), RTLD_LAZY);
|
||||||
return handle;
|
return handle;
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
//remove: O(1) average; O(n) worst
|
//remove: O(1) average; O(n) worst
|
||||||
//
|
//
|
||||||
//requirements:
|
//requirements:
|
||||||
// auto T::hash() const -> unsigned;
|
// auto T::hash() const -> uint;
|
||||||
// auto T::operator==(const T&) const -> bool;
|
// auto T::operator==(const T&) const -> bool;
|
||||||
|
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
@ -15,7 +15,7 @@ namespace nall {
|
||||||
template<typename T>
|
template<typename T>
|
||||||
struct hashset {
|
struct hashset {
|
||||||
hashset() = default;
|
hashset() = default;
|
||||||
hashset(unsigned length) : length(bit::round(length)) {}
|
hashset(uint length) : length(bit::round(length)) {}
|
||||||
hashset(const hashset& source) { operator=(source); }
|
hashset(const hashset& source) { operator=(source); }
|
||||||
hashset(hashset&& source) { operator=(move(source)); }
|
hashset(hashset&& source) { operator=(move(source)); }
|
||||||
~hashset() { reset(); }
|
~hashset() { reset(); }
|
||||||
|
@ -23,7 +23,7 @@ struct hashset {
|
||||||
auto operator=(const hashset& source) -> hashset& {
|
auto operator=(const hashset& source) -> hashset& {
|
||||||
reset();
|
reset();
|
||||||
if(source.pool) {
|
if(source.pool) {
|
||||||
for(unsigned n = 0; n < source.count; n++) {
|
for(uint n : range(source.count)) {
|
||||||
insert(*source.pool[n]);
|
insert(*source.pool[n]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -42,12 +42,12 @@ struct hashset {
|
||||||
}
|
}
|
||||||
|
|
||||||
explicit operator bool() const { return count; }
|
explicit operator bool() const { return count; }
|
||||||
auto capacity() const -> unsigned { return length; }
|
auto capacity() const -> uint { return length; }
|
||||||
auto size() const -> unsigned { return count; }
|
auto size() const -> uint { return count; }
|
||||||
|
|
||||||
auto reset() -> void {
|
auto reset() -> void {
|
||||||
if(pool) {
|
if(pool) {
|
||||||
for(unsigned n = 0; n < length; n++) {
|
for(uint n : range(length)) {
|
||||||
if(pool[n]) {
|
if(pool[n]) {
|
||||||
delete pool[n];
|
delete pool[n];
|
||||||
pool[n] = nullptr;
|
pool[n] = nullptr;
|
||||||
|
@ -60,15 +60,15 @@ struct hashset {
|
||||||
count = 0;
|
count = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto reserve(unsigned size) -> void {
|
auto reserve(uint size) -> void {
|
||||||
//ensure all items will fit into pool (with <= 50% load) and amortize growth
|
//ensure all items will fit into pool (with <= 50% load) and amortize growth
|
||||||
size = bit::round(max(size, count << 1));
|
size = bit::round(max(size, count << 1));
|
||||||
T** copy = new T*[size]();
|
T** copy = new T*[size]();
|
||||||
|
|
||||||
if(pool) {
|
if(pool) {
|
||||||
for(unsigned n = 0; n < length; n++) {
|
for(uint n : range(length)) {
|
||||||
if(pool[n]) {
|
if(pool[n]) {
|
||||||
unsigned hash = (*pool[n]).hash() & (size - 1);
|
uint hash = (*pool[n]).hash() & (size - 1);
|
||||||
while(copy[hash]) if(++hash >= size) hash = 0;
|
while(copy[hash]) if(++hash >= size) hash = 0;
|
||||||
copy[hash] = pool[n];
|
copy[hash] = pool[n];
|
||||||
pool[n] = nullptr;
|
pool[n] = nullptr;
|
||||||
|
@ -84,7 +84,7 @@ struct hashset {
|
||||||
auto find(const T& value) -> maybe<T&> {
|
auto find(const T& value) -> maybe<T&> {
|
||||||
if(!pool) return nothing;
|
if(!pool) return nothing;
|
||||||
|
|
||||||
unsigned hash = value.hash() & (length - 1);
|
uint hash = value.hash() & (length - 1);
|
||||||
while(pool[hash]) {
|
while(pool[hash]) {
|
||||||
if(value == *pool[hash]) return *pool[hash];
|
if(value == *pool[hash]) return *pool[hash];
|
||||||
if(++hash >= length) hash = 0;
|
if(++hash >= length) hash = 0;
|
||||||
|
@ -100,7 +100,7 @@ struct hashset {
|
||||||
if(count >= (length >> 1)) reserve(length << 1);
|
if(count >= (length >> 1)) reserve(length << 1);
|
||||||
count++;
|
count++;
|
||||||
|
|
||||||
unsigned hash = value.hash() & (length - 1);
|
uint hash = value.hash() & (length - 1);
|
||||||
while(pool[hash]) if(++hash >= length) hash = 0;
|
while(pool[hash]) if(++hash >= length) hash = 0;
|
||||||
pool[hash] = new T(value);
|
pool[hash] = new T(value);
|
||||||
|
|
||||||
|
@ -110,7 +110,7 @@ struct hashset {
|
||||||
auto remove(const T& value) -> bool {
|
auto remove(const T& value) -> bool {
|
||||||
if(!pool) return false;
|
if(!pool) return false;
|
||||||
|
|
||||||
unsigned hash = value.hash() & (length - 1);
|
uint hash = value.hash() & (length - 1);
|
||||||
while(pool[hash]) {
|
while(pool[hash]) {
|
||||||
if(value == *pool[hash]) {
|
if(value == *pool[hash]) {
|
||||||
delete pool[hash];
|
delete pool[hash];
|
||||||
|
@ -126,8 +126,8 @@ struct hashset {
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
T** pool = nullptr;
|
T** pool = nullptr;
|
||||||
unsigned length = 8; //length of pool
|
uint length = 8; //length of pool
|
||||||
unsigned count = 0; //number of objects inside of the pool
|
uint count = 0; //number of objects inside of the pool
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
14
nall/hid.hpp
14
nall/hid.hpp
|
@ -19,10 +19,10 @@ struct Group : vector<Input> {
|
||||||
Group(const string& name) : _name(name) {}
|
Group(const string& name) : _name(name) {}
|
||||||
|
|
||||||
auto name() const -> string { return _name; }
|
auto name() const -> string { return _name; }
|
||||||
auto input(unsigned id) -> Input& { return operator[](id); }
|
auto input(uint id) -> Input& { return operator[](id); }
|
||||||
auto append(const string& name) -> void { vector::append(Input{name}); }
|
auto append(const string& name) -> void { vector::append(Input{name}); }
|
||||||
|
|
||||||
auto find(const string& name) const -> maybe<unsigned> {
|
auto find(const string& name) const -> maybe<uint> {
|
||||||
for(auto id : range(size())) {
|
for(auto id : range(size())) {
|
||||||
if(operator[](id)._name == name) return id;
|
if(operator[](id)._name == name) return id;
|
||||||
}
|
}
|
||||||
|
@ -50,10 +50,10 @@ struct Device : vector<Group> {
|
||||||
auto name() const -> string { return _name; }
|
auto name() const -> string { return _name; }
|
||||||
auto id() const -> uint64_t { return _id; }
|
auto id() const -> uint64_t { return _id; }
|
||||||
auto setID(uint64_t id) -> void { _id = id; }
|
auto setID(uint64_t id) -> void { _id = id; }
|
||||||
auto group(unsigned id) -> Group& { return operator[](id); }
|
auto group(uint id) -> Group& { return operator[](id); }
|
||||||
auto append(const string& name) -> void { vector::append(Group{name}); }
|
auto append(const string& name) -> void { vector::append(Group{name}); }
|
||||||
|
|
||||||
auto find(const string& name) const -> maybe<unsigned> {
|
auto find(const string& name) const -> maybe<uint> {
|
||||||
for(auto id : range(size())) {
|
for(auto id : range(size())) {
|
||||||
if(operator[](id)._name == name) return id;
|
if(operator[](id)._name == name) return id;
|
||||||
}
|
}
|
||||||
|
@ -71,7 +71,7 @@ struct Null : Device {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Keyboard : Device {
|
struct Keyboard : Device {
|
||||||
enum GroupID : unsigned { Button };
|
enum GroupID : uint { Button };
|
||||||
|
|
||||||
Keyboard() : Device("Keyboard") { append("Button"); }
|
Keyboard() : Device("Keyboard") { append("Button"); }
|
||||||
auto isKeyboard() const -> bool { return true; }
|
auto isKeyboard() const -> bool { return true; }
|
||||||
|
@ -79,7 +79,7 @@ struct Keyboard : Device {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Mouse : Device {
|
struct Mouse : Device {
|
||||||
enum GroupID : unsigned { Axis, Button };
|
enum GroupID : uint { Axis, Button };
|
||||||
|
|
||||||
Mouse() : Device("Mouse") { append("Axis"), append("Button"); }
|
Mouse() : Device("Mouse") { append("Axis"), append("Button"); }
|
||||||
auto isMouse() const -> bool { return true; }
|
auto isMouse() const -> bool { return true; }
|
||||||
|
@ -88,7 +88,7 @@ struct Mouse : Device {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Joypad : Device {
|
struct Joypad : Device {
|
||||||
enum GroupID : unsigned { Axis, Hat, Trigger, Button };
|
enum GroupID : uint { Axis, Hat, Trigger, Button };
|
||||||
|
|
||||||
Joypad() : Device("Joypad") { append("Axis"), append("Hat"), append("Trigger"), append("Button"); }
|
Joypad() : Device("Joypad") { append("Axis"), append("Hat"), append("Trigger"), append("Button"); }
|
||||||
auto isJoypad() const -> bool { return true; }
|
auto isJoypad() const -> bool { return true; }
|
||||||
|
|
|
@ -9,7 +9,7 @@
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
|
||||||
struct inode {
|
struct inode {
|
||||||
enum class time : unsigned { access, modify };
|
enum class time : uint { access, modify };
|
||||||
|
|
||||||
static auto exists(const string& name) -> bool {
|
static auto exists(const string& name) -> bool {
|
||||||
return access(name, F_OK) == 0;
|
return access(name, F_OK) == 0;
|
||||||
|
@ -27,19 +27,19 @@ struct inode {
|
||||||
return access(name, X_OK) == 0;
|
return access(name, X_OK) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static auto uid(const string& name) -> unsigned {
|
static auto uid(const string& name) -> uint {
|
||||||
struct stat data{0};
|
struct stat data{0};
|
||||||
stat(name, &data);
|
stat(name, &data);
|
||||||
return data.st_uid;
|
return data.st_uid;
|
||||||
}
|
}
|
||||||
|
|
||||||
static auto gid(const string& name) -> unsigned {
|
static auto gid(const string& name) -> uint {
|
||||||
struct stat data{0};
|
struct stat data{0};
|
||||||
stat(name, &data);
|
stat(name, &data);
|
||||||
return data.st_gid;
|
return data.st_gid;
|
||||||
}
|
}
|
||||||
|
|
||||||
static auto mode(const string& name) -> unsigned {
|
static auto mode(const string& name) -> uint {
|
||||||
struct stat data{0};
|
struct stat data{0};
|
||||||
stat(name, &data);
|
stat(name, &data);
|
||||||
return data.st_mode;
|
return data.st_mode;
|
||||||
|
@ -55,7 +55,7 @@ struct inode {
|
||||||
}
|
}
|
||||||
|
|
||||||
//returns true if 'name' already exists
|
//returns true if 'name' already exists
|
||||||
static auto create(const string& name, unsigned permissions = 0755) -> bool {
|
static auto create(const string& name, uint permissions = 0755) -> bool {
|
||||||
if(exists(name)) return true;
|
if(exists(name)) return true;
|
||||||
if(name.endsWith("/")) return mkdir(name, permissions) == 0;
|
if(name.endsWith("/")) return mkdir(name, permissions) == 0;
|
||||||
int fd = open(name, O_CREAT | O_EXCL, permissions);
|
int fd = open(name, O_CREAT | O_EXCL, permissions);
|
||||||
|
|
|
@ -1,13 +1,15 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
using uint = unsigned;
|
||||||
|
|
||||||
struct Intrinsics {
|
struct Intrinsics {
|
||||||
enum class Compiler : unsigned { Clang, GCC, VisualCPP, Unknown };
|
enum class Compiler : uint { Clang, GCC, VisualCPP, Unknown };
|
||||||
enum class Platform : unsigned { Windows, MacOSX, Linux, BSD, Unknown };
|
enum class Platform : uint { Windows, MacOSX, Linux, BSD, Unknown };
|
||||||
enum class API : unsigned { Windows, Posix, Unknown };
|
enum class API : uint { Windows, Posix, Unknown };
|
||||||
enum class Display : unsigned { Windows, Quartz, Xorg, Unknown };
|
enum class Display : uint { Windows, Quartz, Xorg, Unknown };
|
||||||
enum class Processor : unsigned { x86, amd64, ARM, PPC32, PPC64, Unknown };
|
enum class Processor : uint { x86, amd64, ARM, PPC32, PPC64, Unknown };
|
||||||
enum class Endian : unsigned { LSB, MSB, Unknown };
|
enum class Endian : uint { LSB, MSB, Unknown };
|
||||||
|
|
||||||
static inline auto compiler() -> Compiler;
|
static inline auto compiler() -> Compiler;
|
||||||
static inline auto platform() -> Platform;
|
static inline auto platform() -> Platform;
|
||||||
|
|
|
@ -4,13 +4,13 @@ namespace nall {
|
||||||
|
|
||||||
namespace Matrix {
|
namespace Matrix {
|
||||||
|
|
||||||
template<typename T> inline auto Multiply(T* output, const T* xdata, unsigned xrows, unsigned xcols, const T* ydata, unsigned yrows, unsigned ycols) -> void {
|
template<typename T> inline auto Multiply(T* output, const T* xdata, uint xrows, uint xcols, const T* ydata, uint yrows, uint ycols) -> void {
|
||||||
if(xcols != yrows) return;
|
if(xcols != yrows) return;
|
||||||
|
|
||||||
for(unsigned y = 0; y < xrows; y++) {
|
for(uint y : range(xrows)) {
|
||||||
for(unsigned x = 0; x < ycols; x++) {
|
for(uint x : range(ycols)) {
|
||||||
T sum = 0;
|
T sum = 0;
|
||||||
for(unsigned z = 0; z < xcols; z++) {
|
for(uint z : range(xcols)) {
|
||||||
sum += xdata[y * xcols + z] * ydata[z * ycols + x];
|
sum += xdata[y * xcols + z] * ydata[z * ycols + x];
|
||||||
}
|
}
|
||||||
*output++ = sum;
|
*output++ = sum;
|
||||||
|
@ -18,7 +18,7 @@ template<typename T> inline auto Multiply(T* output, const T* xdata, unsigned xr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename T> inline auto Multiply(const T* xdata, unsigned xrows, unsigned xcols, const T* ydata, unsigned yrows, unsigned ycols) -> vector<T> {
|
template<typename T> inline auto Multiply(const T* xdata, uint xrows, uint xcols, const T* ydata, uint yrows, uint ycols) -> vector<T> {
|
||||||
vector<T> output;
|
vector<T> output;
|
||||||
output.resize(xrows * ycols);
|
output.resize(xrows * ycols);
|
||||||
Multiply(output.data(), xdata, xrows, xcols, ydata, yrows, ycols);
|
Multiply(output.data(), xdata, xrows, xcols, ydata, yrows, ycols);
|
||||||
|
|
|
@ -30,11 +30,11 @@ inline auto program() -> string {
|
||||||
GetModuleFileName(nullptr, path, PATH_MAX);
|
GetModuleFileName(nullptr, path, PATH_MAX);
|
||||||
string result = (const char*)utf8_t(path);
|
string result = (const char*)utf8_t(path);
|
||||||
result.transform("\\", "/");
|
result.transform("\\", "/");
|
||||||
return real(result);
|
return Path::real(result);
|
||||||
#else
|
#else
|
||||||
Dl_info info;
|
Dl_info info;
|
||||||
dladdr((void*)&program, &info);
|
dladdr((void*)&program, &info);
|
||||||
return real(info.dli_fname);
|
return Path::real(info.dli_fname);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,9 +78,9 @@ inline auto config() -> string {
|
||||||
string result = (const char*)utf8_t(path);
|
string result = (const char*)utf8_t(path);
|
||||||
result.transform("\\", "/");
|
result.transform("\\", "/");
|
||||||
#elif defined(PLATFORM_MACOSX)
|
#elif defined(PLATFORM_MACOSX)
|
||||||
string result = {user(), "Library/Application Support/"};
|
string result = {Path::user(), "Library/Application Support/"};
|
||||||
#else
|
#else
|
||||||
string result = {user(), ".config/"};
|
string result = {Path::user(), ".config/"};
|
||||||
#endif
|
#endif
|
||||||
if(!result) result = ".";
|
if(!result) result = ".";
|
||||||
if(!result.endsWith("/")) result.append("/");
|
if(!result.endsWith("/")) result.append("/");
|
||||||
|
@ -96,9 +96,9 @@ inline auto local() -> string {
|
||||||
string result = (const char*)utf8_t(path);
|
string result = (const char*)utf8_t(path);
|
||||||
result.transform("\\", "/");
|
result.transform("\\", "/");
|
||||||
#elif defined(PLATFORM_MACOSX)
|
#elif defined(PLATFORM_MACOSX)
|
||||||
string result = {user(), "Library/Application Support/"};
|
string result = {Path::user(), "Library/Application Support/"};
|
||||||
#else
|
#else
|
||||||
string result = {user(), ".local/share/"};
|
string result = {Path::user(), ".local/share/"};
|
||||||
#endif
|
#endif
|
||||||
if(!result) result = ".";
|
if(!result) result = ".";
|
||||||
if(!result.endsWith("/")) result.append("/");
|
if(!result.endsWith("/")) result.append("/");
|
||||||
|
|
|
@ -1,106 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <limits>
|
|
||||||
#include <nall/function.hpp>
|
|
||||||
#include <nall/serializer.hpp>
|
|
||||||
#include <nall/utility.hpp>
|
|
||||||
|
|
||||||
namespace nall {
|
|
||||||
|
|
||||||
template<typename type_t> auto priority_queue_nocallback(type_t) -> void {}
|
|
||||||
|
|
||||||
//priority queue implementation using binary min-heap array;
|
|
||||||
//does not require normalize() function.
|
|
||||||
//O(1) find (tick)
|
|
||||||
//O(log n) append (enqueue)
|
|
||||||
//O(log n) remove (dequeue)
|
|
||||||
template<typename type_t> struct priority_queue {
|
|
||||||
priority_queue(unsigned size, function<void (type_t)> callback = &priority_queue_nocallback<type_t>) : callback(callback) {
|
|
||||||
heap = new heap_t[size];
|
|
||||||
heapcapacity = size;
|
|
||||||
reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
~priority_queue() {
|
|
||||||
delete[] heap;
|
|
||||||
}
|
|
||||||
|
|
||||||
priority_queue(const priority_queue&) = delete;
|
|
||||||
auto operator=(const priority_queue&) -> priority_queue& = delete;
|
|
||||||
|
|
||||||
inline auto tick(unsigned ticks) -> void {
|
|
||||||
basecounter += ticks;
|
|
||||||
while(heapsize && gte(basecounter, heap[0].counter)) callback(dequeue());
|
|
||||||
}
|
|
||||||
|
|
||||||
//counter is relative to current time (eg enqueue(64, ...) fires in 64 ticks);
|
|
||||||
//counter cannot exceed std::numeric_limits<unsigned>::max() >> 1.
|
|
||||||
auto enqueue(unsigned counter, type_t event) -> void {
|
|
||||||
unsigned child = heapsize++;
|
|
||||||
counter += basecounter;
|
|
||||||
|
|
||||||
while(child) {
|
|
||||||
unsigned parent = (child - 1) >> 1;
|
|
||||||
if(gte(counter, heap[parent].counter)) break;
|
|
||||||
|
|
||||||
heap[child].counter = heap[parent].counter;
|
|
||||||
heap[child].event = heap[parent].event;
|
|
||||||
child = parent;
|
|
||||||
}
|
|
||||||
|
|
||||||
heap[child].counter = counter;
|
|
||||||
heap[child].event = event;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto dequeue() -> type_t {
|
|
||||||
type_t event(heap[0].event);
|
|
||||||
unsigned parent = 0;
|
|
||||||
unsigned counter = heap[--heapsize].counter;
|
|
||||||
|
|
||||||
while(true) {
|
|
||||||
unsigned child = (parent << 1) + 1;
|
|
||||||
if(child >= heapsize) break;
|
|
||||||
if(child + 1 < heapsize && gte(heap[child].counter, heap[child + 1].counter)) child++;
|
|
||||||
if(gte(heap[child].counter, counter)) break;
|
|
||||||
|
|
||||||
heap[parent].counter = heap[child].counter;
|
|
||||||
heap[parent].event = heap[child].event;
|
|
||||||
parent = child;
|
|
||||||
}
|
|
||||||
|
|
||||||
heap[parent].counter = counter;
|
|
||||||
heap[parent].event = heap[heapsize].event;
|
|
||||||
return event;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto reset() -> void {
|
|
||||||
basecounter = 0;
|
|
||||||
heapsize = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto serialize(serializer& s) -> void {
|
|
||||||
s.integer(basecounter);
|
|
||||||
s.integer(heapsize);
|
|
||||||
for(unsigned n = 0; n < heapcapacity; n++) {
|
|
||||||
s.integer(heap[n].counter);
|
|
||||||
s.integer(heap[n].event);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
function<void (type_t)> callback;
|
|
||||||
unsigned basecounter;
|
|
||||||
unsigned heapsize;
|
|
||||||
unsigned heapcapacity;
|
|
||||||
struct heap_t {
|
|
||||||
unsigned counter;
|
|
||||||
type_t event;
|
|
||||||
} *heap;
|
|
||||||
|
|
||||||
//return true if x is greater than or equal to y
|
|
||||||
inline auto gte(unsigned x, unsigned y) -> bool {
|
|
||||||
return x - y < (std::numeric_limits<unsigned>::max() >> 1);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
|
@ -15,11 +15,11 @@ struct RandomNumberGenerator {
|
||||||
struct LinearFeedbackShiftRegisterGenerator : RandomNumberGenerator {
|
struct LinearFeedbackShiftRegisterGenerator : RandomNumberGenerator {
|
||||||
auto seed(uint64_t seed) -> void {
|
auto seed(uint64_t seed) -> void {
|
||||||
lfsr = seed;
|
lfsr = seed;
|
||||||
for(unsigned n = 0; n < 8; n++) operator()();
|
for(uint n = 0; n < 8; n++) operator()();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto operator()() -> uint64_t {
|
auto operator()() -> uint64_t {
|
||||||
return lfsr = (lfsr >> 1) ^ (-(lfsr & 1) & crc64jones);
|
return lfsr = (lfsr >> 1) ^ (-(lfsr & 1) & crc64);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto serialize(serializer& s) -> void {
|
auto serialize(serializer& s) -> void {
|
||||||
|
@ -27,9 +27,8 @@ struct LinearFeedbackShiftRegisterGenerator : RandomNumberGenerator {
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static const uint64_t crc64ecma = 0x42f0e1eba9ea3693;
|
static const uint64_t crc64 = 0xc96c'5795'd787'0f42;
|
||||||
static const uint64_t crc64jones = 0xad93d23594c935a9;
|
uint64_t lfsr = crc64;
|
||||||
uint64_t lfsr = crc64ecma;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
inline auto random() -> uint64_t {
|
inline auto random() -> uint64_t {
|
||||||
|
|
16
nall/set.hpp
16
nall/set.hpp
|
@ -26,7 +26,7 @@ template<typename T> struct set {
|
||||||
};
|
};
|
||||||
|
|
||||||
node_t* root = nullptr;
|
node_t* root = nullptr;
|
||||||
unsigned nodes = 0;
|
uint nodes = 0;
|
||||||
|
|
||||||
set() = default;
|
set() = default;
|
||||||
set(const set& source) { operator=(source); }
|
set(const set& source) { operator=(source); }
|
||||||
|
@ -50,7 +50,7 @@ template<typename T> struct set {
|
||||||
}
|
}
|
||||||
|
|
||||||
explicit operator bool() const { return nodes; }
|
explicit operator bool() const { return nodes; }
|
||||||
auto size() const -> unsigned { return nodes; }
|
auto size() const -> uint { return nodes; }
|
||||||
|
|
||||||
auto reset() -> void {
|
auto reset() -> void {
|
||||||
reset(root);
|
reset(root);
|
||||||
|
@ -68,7 +68,7 @@ template<typename T> struct set {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto insert(const T& value) -> maybe<T&> {
|
auto insert(const T& value) -> maybe<T&> {
|
||||||
unsigned count = size();
|
uint count = size();
|
||||||
node_t* v = insert(root, value);
|
node_t* v = insert(root, value);
|
||||||
root->red = 0;
|
root->red = 0;
|
||||||
if(size() == count) return nothing;
|
if(size() == count) return nothing;
|
||||||
|
@ -82,7 +82,7 @@ template<typename T> struct set {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto remove(const T& value) -> bool {
|
auto remove(const T& value) -> bool {
|
||||||
unsigned count = size();
|
uint count = size();
|
||||||
bool done = 0;
|
bool done = 0;
|
||||||
remove(root, &value, done);
|
remove(root, &value, done);
|
||||||
if(root) root->red = 0;
|
if(root) root->red = 0;
|
||||||
|
@ -112,7 +112,7 @@ template<typename T> struct set {
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
base_iterator(const set& source, unsigned position) : source(source), position(position) {
|
base_iterator(const set& source, uint position) : source(source), position(position) {
|
||||||
node_t* node = source.root;
|
node_t* node = source.root;
|
||||||
while(node) {
|
while(node) {
|
||||||
stack.append(node);
|
stack.append(node);
|
||||||
|
@ -122,12 +122,12 @@ template<typename T> struct set {
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
const set& source;
|
const set& source;
|
||||||
unsigned position;
|
uint position;
|
||||||
vector<node_t*> stack;
|
vector<node_t*> stack;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct iterator : base_iterator {
|
struct iterator : base_iterator {
|
||||||
iterator(const set& source, unsigned position) : base_iterator(source, position) {}
|
iterator(const set& source, uint position) : base_iterator(source, position) {}
|
||||||
auto operator*() const -> T& { return base_iterator::stack.right()->value; }
|
auto operator*() const -> T& { return base_iterator::stack.right()->value; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -135,7 +135,7 @@ template<typename T> struct set {
|
||||||
auto end() -> iterator { return iterator(*this, size()); }
|
auto end() -> iterator { return iterator(*this, size()); }
|
||||||
|
|
||||||
struct const_iterator : base_iterator {
|
struct const_iterator : base_iterator {
|
||||||
const_iterator(const set& source, unsigned position) : base_iterator(source, position) {}
|
const_iterator(const set& source, uint position) : base_iterator(source, position) {}
|
||||||
auto operator*() const -> const T& { return base_iterator::stack.right()->value; }
|
auto operator*() const -> const T& { return base_iterator::stack.right()->value; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -18,14 +18,14 @@
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
|
||||||
struct SMTP {
|
struct SMTP {
|
||||||
enum class Format : unsigned { Plain, HTML };
|
enum class Format : uint { Plain, HTML };
|
||||||
|
|
||||||
inline auto server(string server, uint16_t port = 25) -> void;
|
inline auto server(string server, uint16_t port = 25) -> void;
|
||||||
inline auto from(string mail, string name = "") -> void;
|
inline auto from(string mail, string name = "") -> void;
|
||||||
inline auto to(string mail, string name = "") -> void;
|
inline auto to(string mail, string name = "") -> void;
|
||||||
inline auto cc(string mail, string name = "") -> void;
|
inline auto cc(string mail, string name = "") -> void;
|
||||||
inline auto bcc(string mail, string name = "") -> void;
|
inline auto bcc(string mail, string name = "") -> void;
|
||||||
inline auto attachment(const uint8_t* data, unsigned size, string name) -> void;
|
inline auto attachment(const uint8_t* data, uint size, string name) -> void;
|
||||||
inline auto attachment(string filename, string name = "") -> bool;
|
inline auto attachment(string filename, string name = "") -> bool;
|
||||||
inline auto subject(string subject) -> void;
|
inline auto subject(string subject) -> void;
|
||||||
inline auto body(string body, Format format = Format::Plain) -> void;
|
inline auto body(string body, Format format = Format::Plain) -> void;
|
||||||
|
@ -94,7 +94,7 @@ auto SMTP::bcc(string mail, string name) -> void {
|
||||||
info.bcc.append({mail, name});
|
info.bcc.append({mail, name});
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SMTP::attachment(const uint8_t* data, unsigned size, string name) -> void {
|
auto SMTP::attachment(const uint8_t* data, uint size, string name) -> void {
|
||||||
vector<uint8_t> buffer;
|
vector<uint8_t> buffer;
|
||||||
buffer.resize(size);
|
buffer.resize(size);
|
||||||
memcpy(buffer.data(), data, size);
|
memcpy(buffer.data(), data, size);
|
||||||
|
@ -223,7 +223,7 @@ auto SMTP::response() -> string {
|
||||||
|
|
||||||
auto SMTP::send(int sock, const string& text) -> bool {
|
auto SMTP::send(int sock, const string& text) -> bool {
|
||||||
const char* data = text.data();
|
const char* data = text.data();
|
||||||
unsigned size = text.size();
|
uint size = text.size();
|
||||||
while(size) {
|
while(size) {
|
||||||
int length = ::send(sock, (const char*)data, size, 0);
|
int length = ::send(sock, (const char*)data, size, 0);
|
||||||
if(length == -1) return false;
|
if(length == -1) return false;
|
||||||
|
@ -249,7 +249,7 @@ auto SMTP::boundary() -> string {
|
||||||
random_lfsr random;
|
random_lfsr random;
|
||||||
random.seed(time(0));
|
random.seed(time(0));
|
||||||
string boundary;
|
string boundary;
|
||||||
for(unsigned n = 0; n < 16; n++) boundary.append(hex<2>(random()));
|
for(uint n = 0; n < 16; n++) boundary.append(hex<2>(random()));
|
||||||
return boundary;
|
return boundary;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -279,9 +279,9 @@ auto SMTP::contacts(const vector<Information::Contact>& contacts) -> string {
|
||||||
auto SMTP::split(const string& text) -> string {
|
auto SMTP::split(const string& text) -> string {
|
||||||
string result;
|
string result;
|
||||||
|
|
||||||
unsigned offset = 0;
|
uint offset = 0;
|
||||||
while(offset < text.size()) {
|
while(offset < text.size()) {
|
||||||
unsigned length = min(76, text.size() - offset);
|
uint length = min(76, text.size() - offset);
|
||||||
if(length < 76) {
|
if(length < 76) {
|
||||||
result.append(text.slice(offset));
|
result.append(text.slice(offset));
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -19,13 +19,13 @@
|
||||||
|
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
|
||||||
template<typename T, typename Comparator> auto sort(T list[], unsigned size, const Comparator& lessthan) -> void {
|
template<typename T, typename Comparator> auto sort(T list[], uint size, const Comparator& lessthan) -> void {
|
||||||
if(size <= 1) return; //nothing to sort
|
if(size <= 1) return; //nothing to sort
|
||||||
|
|
||||||
//use insertion sort to quickly sort smaller blocks
|
//use insertion sort to quickly sort smaller blocks
|
||||||
if(size < 64) {
|
if(size < 64) {
|
||||||
#if defined(NALL_SORT_INSERTION)
|
#if defined(NALL_SORT_INSERTION)
|
||||||
for(signed i = 1, j; i < size; i++) {
|
for(int i = 1, j; i < size; i++) {
|
||||||
T copy = std::move(list[i]);
|
T copy = std::move(list[i]);
|
||||||
for(j = i - 1; j >= 0; j--) {
|
for(j = i - 1; j >= 0; j--) {
|
||||||
if(!lessthan(copy, list[j])) break;
|
if(!lessthan(copy, list[j])) break;
|
||||||
|
@ -34,9 +34,9 @@ template<typename T, typename Comparator> auto sort(T list[], unsigned size, con
|
||||||
list[j + 1] = std::move(copy);
|
list[j + 1] = std::move(copy);
|
||||||
}
|
}
|
||||||
#elif defined(NALL_SORT_SELECTION)
|
#elif defined(NALL_SORT_SELECTION)
|
||||||
for(unsigned i = 0; i < size; i++) {
|
for(uint i = 0; i < size; i++) {
|
||||||
unsigned min = i;
|
uint min = i;
|
||||||
for(unsigned j = i + 1; j < size; j++) {
|
for(uint j = i + 1; j < size; j++) {
|
||||||
if(lessthan(list[j], list[min])) min = j;
|
if(lessthan(list[j], list[min])) min = j;
|
||||||
}
|
}
|
||||||
if(min != i) std::swap(list[i], list[min]);
|
if(min != i) std::swap(list[i], list[min]);
|
||||||
|
@ -46,13 +46,13 @@ template<typename T, typename Comparator> auto sort(T list[], unsigned size, con
|
||||||
}
|
}
|
||||||
|
|
||||||
//split list in half and recursively sort both
|
//split list in half and recursively sort both
|
||||||
unsigned middle = size / 2;
|
uint middle = size / 2;
|
||||||
sort(list, middle, lessthan);
|
sort(list, middle, lessthan);
|
||||||
sort(list + middle, size - middle, lessthan);
|
sort(list + middle, size - middle, lessthan);
|
||||||
|
|
||||||
//left and right are sorted here; perform merge sort
|
//left and right are sorted here; perform merge sort
|
||||||
T* buffer = new T[size];
|
T* buffer = new T[size];
|
||||||
unsigned offset = 0, left = 0, right = middle;
|
uint offset = 0, left = 0, right = middle;
|
||||||
while(left < middle && right < size) {
|
while(left < middle && right < size) {
|
||||||
if(!lessthan(list[right], list[left])) {
|
if(!lessthan(list[right], list[left])) {
|
||||||
buffer[offset++] = std::move(list[left++]);
|
buffer[offset++] = std::move(list[left++]);
|
||||||
|
@ -63,11 +63,11 @@ template<typename T, typename Comparator> auto sort(T list[], unsigned size, con
|
||||||
while(left < middle) buffer[offset++] = std::move(list[left++]);
|
while(left < middle) buffer[offset++] = std::move(list[left++]);
|
||||||
while(right < size) buffer[offset++] = std::move(list[right++]);
|
while(right < size) buffer[offset++] = std::move(list[right++]);
|
||||||
|
|
||||||
for(unsigned i = 0; i < size; i++) list[i] = std::move(buffer[i]);
|
for(uint i = 0; i < size; i++) list[i] = std::move(buffer[i]);
|
||||||
delete[] buffer;
|
delete[] buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename T> auto sort(T list[], unsigned size) -> void {
|
template<typename T> auto sort(T list[], uint size) -> void {
|
||||||
return sort(list, size, [](const T& l, const T& r) { return l < r; });
|
return sort(list, size, [](const T& l, const T& r) { return l < r; });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,13 +19,13 @@ namespace nall {
|
||||||
struct thread {
|
struct thread {
|
||||||
inline auto join() -> void;
|
inline auto join() -> void;
|
||||||
|
|
||||||
static inline auto create(const function<void (uintptr_t)>& callback, uintptr_t parameter = 0, unsigned stacksize = 0) -> thread;
|
static inline auto create(const function<void (uintptr)>& callback, uintptr parameter = 0, uint stacksize = 0) -> thread;
|
||||||
static inline auto detach() -> void;
|
static inline auto detach() -> void;
|
||||||
static inline auto exit() -> void;
|
static inline auto exit() -> void;
|
||||||
|
|
||||||
struct context {
|
struct context {
|
||||||
function<void (uintptr_t)> callback;
|
function<auto (uintptr) -> void> callback;
|
||||||
uintptr_t parameter = 0;
|
uintptr parameter = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -43,7 +43,7 @@ auto thread::join() -> void {
|
||||||
pthread_join(handle, nullptr);
|
pthread_join(handle, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto thread::create(const function<void (uintptr_t)>& callback, uintptr_t parameter, unsigned stacksize) -> thread {
|
auto thread::create(const function<void (uintptr)>& callback, uintptr parameter, uint stacksize) -> thread {
|
||||||
thread instance;
|
thread instance;
|
||||||
|
|
||||||
auto context = new thread::context;
|
auto context = new thread::context;
|
||||||
|
@ -76,13 +76,13 @@ struct thread {
|
||||||
inline ~thread();
|
inline ~thread();
|
||||||
inline auto join() -> void;
|
inline auto join() -> void;
|
||||||
|
|
||||||
static inline auto create(const function<void (uintptr_t)>& callback, uintptr_t parameter = 0, unsigned stacksize = 0) -> thread;
|
static inline auto create(const function<void (uintptr)>& callback, uintptr parameter = 0, uint stacksize = 0) -> thread;
|
||||||
static inline auto detach() -> void;
|
static inline auto detach() -> void;
|
||||||
static inline auto exit() -> void;
|
static inline auto exit() -> void;
|
||||||
|
|
||||||
struct context {
|
struct context {
|
||||||
function<void (uintptr_t)> callback;
|
function<auto (uintptr) -> void> callback;
|
||||||
uintptr_t parameter = 0;
|
uintptr parameter = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -111,7 +111,7 @@ auto thread::join() -> void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto thread::create(const function<void (uintptr_t)>& callback, uintptr_t parameter, unsigned stacksize) -> thread {
|
auto thread::create(const function<void (uintptr)>& callback, uintptr parameter, uint stacksize) -> thread {
|
||||||
thread instance;
|
thread instance;
|
||||||
|
|
||||||
auto context = new thread::context;
|
auto context = new thread::context;
|
||||||
|
|
|
@ -9,9 +9,9 @@ template<typename T> struct base_from_member {
|
||||||
T value;
|
T value;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename T> inline auto allocate(unsigned size, const T& value) -> T* {
|
template<typename T> inline auto allocate(uint size, const T& value) -> T* {
|
||||||
T* array = new T[size];
|
T* array = new T[size];
|
||||||
for(unsigned i = 0; i < size; i++) array[i] = value;
|
for(uint i = 0; i < size; i++) array[i] = value;
|
||||||
return array;
|
return array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -10,8 +10,8 @@ struct varint {
|
||||||
virtual auto read() -> uint8_t = 0;
|
virtual auto read() -> uint8_t = 0;
|
||||||
virtual auto write(uint8_t) -> void = 0;
|
virtual auto write(uint8_t) -> void = 0;
|
||||||
|
|
||||||
auto readvu() -> uintmax_t {
|
auto readvu() -> uintmax {
|
||||||
uintmax_t data = 0, shift = 1;
|
uintmax data = 0, shift = 1;
|
||||||
while(true) {
|
while(true) {
|
||||||
uint8_t x = read();
|
uint8_t x = read();
|
||||||
data += (x & 0x7f) * shift;
|
data += (x & 0x7f) * shift;
|
||||||
|
@ -22,15 +22,15 @@ struct varint {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto readvs() -> intmax_t {
|
auto readvs() -> intmax {
|
||||||
uintmax_t data = readvu();
|
uintmax data = readvu();
|
||||||
bool negate = data & 1;
|
bool negate = data & 1;
|
||||||
data >>= 1;
|
data >>= 1;
|
||||||
if(negate) data = ~data;
|
if(negate) data = ~data;
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto writevu(uintmax_t data) -> void {
|
auto writevu(uintmax data) -> void {
|
||||||
while(true) {
|
while(true) {
|
||||||
uint8_t x = data & 0x7f;
|
uint8_t x = data & 0x7f;
|
||||||
data >>= 7;
|
data >>= 7;
|
||||||
|
@ -40,7 +40,7 @@ struct varint {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto writevs(intmax_t data) -> void {
|
auto writevs(intmax data) -> void {
|
||||||
bool negate = data < 0;
|
bool negate = data < 0;
|
||||||
if(negate) data = ~data;
|
if(negate) data = ~data;
|
||||||
data = (data << 1) | negate;
|
data = (data << 1) | negate;
|
||||||
|
|
|
@ -1,27 +1,17 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <nall/random.hpp>
|
|
||||||
#include <nall/string.hpp>
|
#include <nall/string.hpp>
|
||||||
|
|
||||||
namespace nall {
|
namespace nall {
|
||||||
|
|
||||||
//generate unique GUID
|
|
||||||
inline auto guid() -> string {
|
inline auto guid() -> string {
|
||||||
LinearFeedbackShiftRegisterGenerator lfsr;
|
GUID guidInstance;
|
||||||
lfsr.seed(time(nullptr));
|
CoCreateGuid(&guidInstance);
|
||||||
for(uint n = 0; n < 256; n++) lfsr();
|
|
||||||
|
|
||||||
string output;
|
wchar_t guidString[39];
|
||||||
for(uint n = 0; n < 4; n++) output.append(hex(lfsr(), 2L));
|
StringFromGUID2(guidInstance, guidString, 39);
|
||||||
output.append("-");
|
|
||||||
for(uint n = 0; n < 2; n++) output.append(hex(lfsr(), 2L));
|
return (char*)utf8_t(guidString);
|
||||||
output.append("-");
|
|
||||||
for(uint n = 0; n < 2; n++) output.append(hex(lfsr(), 2L));
|
|
||||||
output.append("-");
|
|
||||||
for(uint n = 0; n < 2; n++) output.append(hex(lfsr(), 2L));
|
|
||||||
output.append("-");
|
|
||||||
for(uint n = 0; n < 6; n++) output.append(hex(lfsr(), 2L));
|
|
||||||
return {"{", output, "}"};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,7 +77,7 @@ namespace nall {
|
||||||
|
|
||||||
inline auto utf8_args(int& argc, char**& argv) -> void {
|
inline auto utf8_args(int& argc, char**& argv) -> void {
|
||||||
wchar_t** wargv = CommandLineToArgvW(GetCommandLineW(), &argc);
|
wchar_t** wargv = CommandLineToArgvW(GetCommandLineW(), &argc);
|
||||||
argv = new char*[argc];
|
argv = new char*[argc + 1]();
|
||||||
for(uint i = 0; i < argc; i++) {
|
for(uint i = 0; i < argc; i++) {
|
||||||
argv[i] = new char[PATH_MAX];
|
argv[i] = new char[PATH_MAX];
|
||||||
strcpy(argv[i], nall::utf8_t(wargv[i]));
|
strcpy(argv[i], nall::utf8_t(wargv[i]));
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
#include "xaudio2.hpp"
|
#include "xaudio2.hpp"
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
#include <audioclient.h>
|
|
||||||
|
|
||||||
struct AudioXAudio2 : Audio, public IXAudio2VoiceCallback {
|
struct AudioXAudio2 : Audio, public IXAudio2VoiceCallback {
|
||||||
~AudioXAudio2() { term(); }
|
~AudioXAudio2() { term(); }
|
||||||
|
|
|
@ -6,14 +6,15 @@
|
||||||
#define GUID_EXT EXTERN_C
|
#define GUID_EXT EXTERN_C
|
||||||
#define GUID_SECT
|
#define GUID_SECT
|
||||||
|
|
||||||
#include <BaseTyps.h>
|
#include <audioclient.h>
|
||||||
|
#include <basetyps.h>
|
||||||
|
|
||||||
#define DEFINE_GUID_X(n,l,w1,w2,b1,b2,b3,b4,b5,b6,b7,b8) GUID_EXT const GUID n GUID_SECT = {l,w1,w2,{b1,b2,b3,b4,b5,b6,b7,b8}}
|
#define DEFINE_GUID_X(n,l,w1,w2,b1,b2,b3,b4,b5,b6,b7,b8) GUID_EXT const GUID n GUID_SECT = {l, w1, w2, {b1, b2, b3, b4, b5, b6, b7, b8}}
|
||||||
#define DEFINE_CLSID_X(className, l, w1, w2, b1, b2, b3, b4, b5, b6, b7, b8) \
|
#define DEFINE_CLSID_X(className, l, w1, w2, b1, b2, b3, b4, b5, b6, b7, b8) \
|
||||||
DEFINE_GUID_X(CLSID_##className, 0x##l, 0x##w1, 0x##w2, 0x##b1, 0x##b2, 0x##b3, 0x##b4, 0x##b5, 0x##b6, 0x##b7, 0x##b8)
|
DEFINE_GUID_X(CLSID_##className, 0x##l, 0x##w1, 0x##w2, 0x##b1, 0x##b2, 0x##b3, 0x##b4, 0x##b5, 0x##b6, 0x##b7, 0x##b8)
|
||||||
#define DEFINE_IID_X(interfaceName, l, w1, w2, b1, b2, b3, b4, b5, b6, b7, b8) \
|
#define DEFINE_IID_X(interfaceName, l, w1, w2, b1, b2, b3, b4, b5, b6, b7, b8) \
|
||||||
DEFINE_GUID_X(IID_##interfaceName, 0x##l, 0x##w1, 0x##w2, 0x##b1, 0x##b2, 0x##b3, 0x##b4, 0x##b5, 0x##b6, 0x##b7, 0x##b8)
|
DEFINE_GUID_X(IID_##interfaceName, 0x##l, 0x##w1, 0x##w2, 0x##b1, 0x##b2, 0x##b3, 0x##b4, 0x##b5, 0x##b6, 0x##b7, 0x##b8)
|
||||||
#define X2DEFAULT(x) =x
|
#define X2DEFAULT(x) = x
|
||||||
|
|
||||||
DEFINE_CLSID_X(XAudio2, e21a7345, eb21, 468e, be, 50, 80, 4d, b9, 7c, f7, 08);
|
DEFINE_CLSID_X(XAudio2, e21a7345, eb21, 468e, be, 50, 80, 4d, b9, 7c, f7, 08);
|
||||||
DEFINE_CLSID_X(XAudio2_Debug, f7a76c21, 53d4, 46bb, ac, 53, 8b, 45, 9c, ae, 46, bd);
|
DEFINE_CLSID_X(XAudio2_Debug, f7a76c21, 53d4, 46bb, ac, 53, 8b, 45, 9c, ae, 46, bd);
|
||||||
|
@ -28,34 +29,30 @@ DECLARE_INTERFACE(IXAudio2Voice);
|
||||||
#define XAUDIO2_DEBUG_ENGINE 0x0001
|
#define XAUDIO2_DEBUG_ENGINE 0x0001
|
||||||
#define XAUDIO2_VOICE_NOSRC 0x0004
|
#define XAUDIO2_VOICE_NOSRC 0x0004
|
||||||
|
|
||||||
typedef enum XAUDIO2_DEVICE_ROLE
|
typedef enum XAUDIO2_DEVICE_ROLE {
|
||||||
{
|
|
||||||
NotDefaultDevice = 0x0,
|
NotDefaultDevice = 0x0,
|
||||||
DefaultConsoleDevice = 0x1,
|
DefaultConsoleDevice = 0x1,
|
||||||
DefaultMultimediaDevice = 0x2,
|
DefaultMultimediaDevice = 0x2,
|
||||||
DefaultCommunicationsDevice = 0x4,
|
DefaultCommunicationsDevice = 0x4,
|
||||||
DefaultGameDevice = 0x8,
|
DefaultGameDevice = 0x8,
|
||||||
GlobalDefaultDevice = 0xf,
|
GlobalDefaultDevice = 0xf,
|
||||||
InvalidDeviceRole = ~GlobalDefaultDevice
|
InvalidDeviceRole = ~GlobalDefaultDevice,
|
||||||
} XAUDIO2_DEVICE_ROLE;
|
} XAUDIO2_DEVICE_ROLE;
|
||||||
|
|
||||||
typedef struct XAUDIO2_DEVICE_DETAILS
|
typedef struct XAUDIO2_DEVICE_DETAILS {
|
||||||
{
|
|
||||||
WCHAR DeviceID[256];
|
WCHAR DeviceID[256];
|
||||||
WCHAR DisplayName[256];
|
WCHAR DisplayName[256];
|
||||||
XAUDIO2_DEVICE_ROLE Role;
|
XAUDIO2_DEVICE_ROLE Role;
|
||||||
WAVEFORMATEXTENSIBLE OutputFormat;
|
WAVEFORMATEXTENSIBLE OutputFormat;
|
||||||
} XAUDIO2_DEVICE_DETAILS;
|
} XAUDIO2_DEVICE_DETAILS;
|
||||||
|
|
||||||
typedef struct XAUDIO2_VOICE_DETAILS
|
typedef struct XAUDIO2_VOICE_DETAILS {
|
||||||
{
|
|
||||||
UINT32 CreationFlags;
|
UINT32 CreationFlags;
|
||||||
UINT32 InputChannels;
|
UINT32 InputChannels;
|
||||||
UINT32 InputSampleRate;
|
UINT32 InputSampleRate;
|
||||||
} XAUDIO2_VOICE_DETAILS;
|
} XAUDIO2_VOICE_DETAILS;
|
||||||
|
|
||||||
typedef enum XAUDIO2_WINDOWS_PROCESSOR_SPECIFIER
|
typedef enum XAUDIO2_WINDOWS_PROCESSOR_SPECIFIER {
|
||||||
{
|
|
||||||
Processor1 = 0x00000001,
|
Processor1 = 0x00000001,
|
||||||
Processor2 = 0x00000002,
|
Processor2 = 0x00000002,
|
||||||
Processor3 = 0x00000004,
|
Processor3 = 0x00000004,
|
||||||
|
@ -89,45 +86,38 @@ typedef enum XAUDIO2_WINDOWS_PROCESSOR_SPECIFIER
|
||||||
Processor31 = 0x40000000,
|
Processor31 = 0x40000000,
|
||||||
Processor32 = 0x80000000,
|
Processor32 = 0x80000000,
|
||||||
XAUDIO2_ANY_PROCESSOR = 0xffffffff,
|
XAUDIO2_ANY_PROCESSOR = 0xffffffff,
|
||||||
XAUDIO2_DEFAULT_PROCESSOR = XAUDIO2_ANY_PROCESSOR
|
XAUDIO2_DEFAULT_PROCESSOR = XAUDIO2_ANY_PROCESSOR,
|
||||||
} XAUDIO2_WINDOWS_PROCESSOR_SPECIFIER, XAUDIO2_PROCESSOR;
|
} XAUDIO2_WINDOWS_PROCESSOR_SPECIFIER, XAUDIO2_PROCESSOR;
|
||||||
|
|
||||||
typedef struct XAUDIO2_VOICE_SENDS
|
typedef struct XAUDIO2_VOICE_SENDS {
|
||||||
{
|
|
||||||
UINT32 OutputCount;
|
UINT32 OutputCount;
|
||||||
IXAudio2Voice** pOutputVoices;
|
IXAudio2Voice** pOutputVoices;
|
||||||
} XAUDIO2_VOICE_SENDS;
|
} XAUDIO2_VOICE_SENDS;
|
||||||
|
|
||||||
typedef struct XAUDIO2_EFFECT_DESCRIPTOR
|
typedef struct XAUDIO2_EFFECT_DESCRIPTOR {
|
||||||
{
|
|
||||||
IUnknown* pEffect;
|
IUnknown* pEffect;
|
||||||
BOOL InitialState;
|
BOOL InitialState;
|
||||||
UINT32 OutputChannels;
|
UINT32 OutputChannels;
|
||||||
} XAUDIO2_EFFECT_DESCRIPTOR;
|
} XAUDIO2_EFFECT_DESCRIPTOR;
|
||||||
|
|
||||||
typedef struct XAUDIO2_EFFECT_CHAIN
|
typedef struct XAUDIO2_EFFECT_CHAIN {
|
||||||
{
|
|
||||||
UINT32 EffectCount;
|
UINT32 EffectCount;
|
||||||
const XAUDIO2_EFFECT_DESCRIPTOR* pEffectDescriptors;
|
const XAUDIO2_EFFECT_DESCRIPTOR* pEffectDescriptors;
|
||||||
} XAUDIO2_EFFECT_CHAIN;
|
} XAUDIO2_EFFECT_CHAIN;
|
||||||
|
|
||||||
typedef enum XAUDIO2_FILTER_TYPE
|
typedef enum XAUDIO2_FILTER_TYPE {
|
||||||
{
|
|
||||||
LowPassFilter,
|
LowPassFilter,
|
||||||
BandPassFilter,
|
BandPassFilter,
|
||||||
HighPassFilter
|
HighPassFilter,
|
||||||
} XAUDIO2_FILTER_TYPE;
|
} XAUDIO2_FILTER_TYPE;
|
||||||
|
|
||||||
typedef struct XAUDIO2_FILTER_PARAMETERS
|
typedef struct XAUDIO2_FILTER_PARAMETERS {
|
||||||
{
|
|
||||||
XAUDIO2_FILTER_TYPE Type;
|
XAUDIO2_FILTER_TYPE Type;
|
||||||
float Frequency;
|
float Frequency;
|
||||||
float OneOverQ;
|
float OneOverQ;
|
||||||
|
|
||||||
} XAUDIO2_FILTER_PARAMETERS;
|
} XAUDIO2_FILTER_PARAMETERS;
|
||||||
|
|
||||||
typedef struct XAUDIO2_BUFFER
|
typedef struct XAUDIO2_BUFFER {
|
||||||
{
|
|
||||||
UINT32 Flags;
|
UINT32 Flags;
|
||||||
UINT32 AudioBytes;
|
UINT32 AudioBytes;
|
||||||
const BYTE* pAudioData;
|
const BYTE* pAudioData;
|
||||||
|
@ -139,21 +129,18 @@ typedef struct XAUDIO2_BUFFER
|
||||||
void* pContext;
|
void* pContext;
|
||||||
} XAUDIO2_BUFFER;
|
} XAUDIO2_BUFFER;
|
||||||
|
|
||||||
typedef struct XAUDIO2_BUFFER_WMA
|
typedef struct XAUDIO2_BUFFER_WMA {
|
||||||
{
|
|
||||||
const UINT32* pDecodedPacketCumulativeBytes;
|
const UINT32* pDecodedPacketCumulativeBytes;
|
||||||
UINT32 PacketCount;
|
UINT32 PacketCount;
|
||||||
} XAUDIO2_BUFFER_WMA;
|
} XAUDIO2_BUFFER_WMA;
|
||||||
|
|
||||||
typedef struct XAUDIO2_VOICE_STATE
|
typedef struct XAUDIO2_VOICE_STATE {
|
||||||
{
|
|
||||||
void* pCurrentBufferContext;
|
void* pCurrentBufferContext;
|
||||||
UINT32 BuffersQueued;
|
UINT32 BuffersQueued;
|
||||||
UINT64 SamplesPlayed;
|
UINT64 SamplesPlayed;
|
||||||
} XAUDIO2_VOICE_STATE;
|
} XAUDIO2_VOICE_STATE;
|
||||||
|
|
||||||
typedef struct XAUDIO2_PERFORMANCE_DATA
|
typedef struct XAUDIO2_PERFORMANCE_DATA {
|
||||||
{
|
|
||||||
UINT64 AudioCyclesSinceLastQuery;
|
UINT64 AudioCyclesSinceLastQuery;
|
||||||
UINT64 TotalCyclesSinceLastQuery;
|
UINT64 TotalCyclesSinceLastQuery;
|
||||||
UINT32 MinimumCyclesPerQuantum;
|
UINT32 MinimumCyclesPerQuantum;
|
||||||
|
@ -169,8 +156,7 @@ typedef struct XAUDIO2_PERFORMANCE_DATA
|
||||||
UINT32 ActiveXmaStreams;
|
UINT32 ActiveXmaStreams;
|
||||||
} XAUDIO2_PERFORMANCE_DATA;
|
} XAUDIO2_PERFORMANCE_DATA;
|
||||||
|
|
||||||
typedef struct XAUDIO2_DEBUG_CONFIGURATION
|
typedef struct XAUDIO2_DEBUG_CONFIGURATION {
|
||||||
{
|
|
||||||
UINT32 TraceMask;
|
UINT32 TraceMask;
|
||||||
UINT32 BreakMask;
|
UINT32 BreakMask;
|
||||||
BOOL LogThreadID;
|
BOOL LogThreadID;
|
||||||
|
@ -179,15 +165,13 @@ typedef struct XAUDIO2_DEBUG_CONFIGURATION
|
||||||
BOOL LogTiming;
|
BOOL LogTiming;
|
||||||
} XAUDIO2_DEBUG_CONFIGURATION;
|
} XAUDIO2_DEBUG_CONFIGURATION;
|
||||||
|
|
||||||
DECLARE_INTERFACE(IXAudio2EngineCallback)
|
DECLARE_INTERFACE(IXAudio2EngineCallback) {
|
||||||
{
|
|
||||||
STDMETHOD_(void, OnProcessingPassStart) (THIS) PURE;
|
STDMETHOD_(void, OnProcessingPassStart) (THIS) PURE;
|
||||||
STDMETHOD_(void, OnProcessingPassEnd) (THIS) PURE;
|
STDMETHOD_(void, OnProcessingPassEnd) (THIS) PURE;
|
||||||
STDMETHOD_(void, OnCriticalError) (THIS_ HRESULT Error) PURE;
|
STDMETHOD_(void, OnCriticalError) (THIS_ HRESULT Error) PURE;
|
||||||
};
|
};
|
||||||
|
|
||||||
DECLARE_INTERFACE(IXAudio2VoiceCallback)
|
DECLARE_INTERFACE(IXAudio2VoiceCallback) {
|
||||||
{
|
|
||||||
STDMETHOD_(void, OnVoiceProcessingPassStart) (THIS_ UINT32 BytesRequired) PURE;
|
STDMETHOD_(void, OnVoiceProcessingPassStart) (THIS_ UINT32 BytesRequired) PURE;
|
||||||
STDMETHOD_(void, OnVoiceProcessingPassEnd) (THIS) PURE;
|
STDMETHOD_(void, OnVoiceProcessingPassEnd) (THIS) PURE;
|
||||||
STDMETHOD_(void, OnStreamEnd) (THIS) PURE;
|
STDMETHOD_(void, OnStreamEnd) (THIS) PURE;
|
||||||
|
@ -197,8 +181,7 @@ DECLARE_INTERFACE(IXAudio2VoiceCallback)
|
||||||
STDMETHOD_(void, OnVoiceError) (THIS_ void* pBufferContext, HRESULT Error) PURE;
|
STDMETHOD_(void, OnVoiceError) (THIS_ void* pBufferContext, HRESULT Error) PURE;
|
||||||
};
|
};
|
||||||
|
|
||||||
DECLARE_INTERFACE(IXAudio2Voice)
|
DECLARE_INTERFACE(IXAudio2Voice) {
|
||||||
{
|
|
||||||
#define Declare_IXAudio2Voice_Methods() \
|
#define Declare_IXAudio2Voice_Methods() \
|
||||||
STDMETHOD_(void, GetVoiceDetails) (THIS_ XAUDIO2_VOICE_DETAILS* pVoiceDetails) PURE; \
|
STDMETHOD_(void, GetVoiceDetails) (THIS_ XAUDIO2_VOICE_DETAILS* pVoiceDetails) PURE; \
|
||||||
STDMETHOD(SetOutputVoices) (THIS_ const XAUDIO2_VOICE_SENDS* pSendList) PURE; \
|
STDMETHOD(SetOutputVoices) (THIS_ const XAUDIO2_VOICE_SENDS* pSendList) PURE; \
|
||||||
|
@ -235,19 +218,15 @@ DECLARE_INTERFACE(IXAudio2Voice)
|
||||||
Declare_IXAudio2Voice_Methods();
|
Declare_IXAudio2Voice_Methods();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
DECLARE_INTERFACE_(IXAudio2MasteringVoice, IXAudio2Voice) {
|
||||||
DECLARE_INTERFACE_(IXAudio2MasteringVoice, IXAudio2Voice)
|
|
||||||
{
|
|
||||||
Declare_IXAudio2Voice_Methods();
|
Declare_IXAudio2Voice_Methods();
|
||||||
};
|
};
|
||||||
|
|
||||||
DECLARE_INTERFACE_(IXAudio2SubmixVoice, IXAudio2Voice)
|
DECLARE_INTERFACE_(IXAudio2SubmixVoice, IXAudio2Voice) {
|
||||||
{
|
|
||||||
Declare_IXAudio2Voice_Methods();
|
Declare_IXAudio2Voice_Methods();
|
||||||
};
|
};
|
||||||
|
|
||||||
DECLARE_INTERFACE_(IXAudio2SourceVoice, IXAudio2Voice)
|
DECLARE_INTERFACE_(IXAudio2SourceVoice, IXAudio2Voice) {
|
||||||
{
|
|
||||||
Declare_IXAudio2Voice_Methods();
|
Declare_IXAudio2Voice_Methods();
|
||||||
STDMETHOD(Start) (THIS_ UINT32 Flags, UINT32 OperationSet X2DEFAULT(XAUDIO2_COMMIT_NOW)) PURE;
|
STDMETHOD(Start) (THIS_ UINT32 Flags, UINT32 OperationSet X2DEFAULT(XAUDIO2_COMMIT_NOW)) PURE;
|
||||||
STDMETHOD(Stop) (THIS_ UINT32 Flags, UINT32 OperationSet X2DEFAULT(XAUDIO2_COMMIT_NOW)) PURE;
|
STDMETHOD(Stop) (THIS_ UINT32 Flags, UINT32 OperationSet X2DEFAULT(XAUDIO2_COMMIT_NOW)) PURE;
|
||||||
|
@ -261,8 +240,7 @@ DECLARE_INTERFACE_(IXAudio2SourceVoice, IXAudio2Voice)
|
||||||
STDMETHOD_(void, GetFrequencyRatio) (THIS_ float* pRatio) PURE;
|
STDMETHOD_(void, GetFrequencyRatio) (THIS_ float* pRatio) PURE;
|
||||||
};
|
};
|
||||||
|
|
||||||
DECLARE_INTERFACE_(IXAudio2, IUnknown)
|
DECLARE_INTERFACE_(IXAudio2, IUnknown) {
|
||||||
{
|
|
||||||
STDMETHOD(QueryInterface) (THIS_ REFIID riid, void** ppvInterface) PURE;
|
STDMETHOD(QueryInterface) (THIS_ REFIID riid, void** ppvInterface) PURE;
|
||||||
STDMETHOD_(ULONG, AddRef) (THIS) PURE;
|
STDMETHOD_(ULONG, AddRef) (THIS) PURE;
|
||||||
STDMETHOD_(ULONG, Release) (THIS) PURE;
|
STDMETHOD_(ULONG, Release) (THIS) PURE;
|
||||||
|
@ -298,20 +276,15 @@ DECLARE_INTERFACE_(IXAudio2, IUnknown)
|
||||||
};
|
};
|
||||||
|
|
||||||
__inline HRESULT XAudio2Create(IXAudio2** ppXAudio2, UINT32 Flags X2DEFAULT(0),
|
__inline HRESULT XAudio2Create(IXAudio2** ppXAudio2, UINT32 Flags X2DEFAULT(0),
|
||||||
XAUDIO2_PROCESSOR XAudio2Processor X2DEFAULT(XAUDIO2_DEFAULT_PROCESSOR))
|
XAUDIO2_PROCESSOR XAudio2Processor X2DEFAULT(XAUDIO2_DEFAULT_PROCESSOR)) {
|
||||||
{
|
|
||||||
IXAudio2* pXAudio2;
|
IXAudio2* pXAudio2;
|
||||||
HRESULT hr = CoCreateInstance((Flags & XAUDIO2_DEBUG_ENGINE) ? CLSID_XAudio2_Debug : CLSID_XAudio2,
|
HRESULT hr = CoCreateInstance((Flags & XAUDIO2_DEBUG_ENGINE) ? CLSID_XAudio2_Debug : CLSID_XAudio2,
|
||||||
NULL, CLSCTX_INPROC_SERVER, IID_IXAudio2, (void**)&pXAudio2);
|
NULL, CLSCTX_INPROC_SERVER, IID_IXAudio2, (void**)&pXAudio2);
|
||||||
if (SUCCEEDED(hr))
|
if(SUCCEEDED(hr)) {
|
||||||
{
|
|
||||||
hr = pXAudio2->Initialize(Flags, XAudio2Processor);
|
hr = pXAudio2->Initialize(Flags, XAudio2Processor);
|
||||||
if (SUCCEEDED(hr))
|
if(SUCCEEDED(hr)) {
|
||||||
{
|
|
||||||
*ppXAudio2 = pXAudio2;
|
*ppXAudio2 = pXAudio2;
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
pXAudio2->Release();
|
pXAudio2->Release();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue