mirror of https://github.com/bsnes-emu/bsnes.git
Update to v098r06 release.
byuu says: Changelog: - emulation cores now refresh video from host thread instead of cothreads (fix AMD crash) - SFC: fixed another bug with leap year months in SharpRTC emulation - SFC: cleaned up camelCase on function names for armdsp,epsonrtc,hitachidsp,mcc,nss,sharprtc classes - GB: added MBC1M emulation (requires manually setting mapper=MBC1M in manifest.bml for now, sorry) - audio: implemented Emulator::Audio mixer and effects processor - audio: implemented Emulator::Stream interface - it is now possible to have more than two audio streams: eg SNES + SGB + MSU1 + Voicer-Kun (eventually) - audio: added reverb delay + reverb level settings; exposed balance configuration in UI - video: reworked palette generation to re-enable saturation, gamma, luminance adjustments - higan/emulator.cpp is gone since there was nothing left in it I know you guys are going to say the color adjust/balance/reverb stuff is pointless. And indeed it mostly is. But I like the idea of allowing some fun special effects and configurability that isn't system-wide. Note: there seems to be some kind of added audio lag in the SGB emulation now, and I don't really understand why. The code should be effectively identical to what I had before. The only main thing is that I'm sampling things to 48000hz instead of 32040hz before mixing. There's no point where I'm intentionally introducing added latency though. I'm kind of stumped, so if anyone wouldn't mind taking a look at it, it'd be much appreciated :/ I don't have an MSU1 test ROM, but the latency issue may affect MSU1 as well, and that would be very bad.
This commit is contained in:
parent
55e507d5df
commit
e2ee6689a0
|
@ -5,7 +5,7 @@ target := tomoko
|
||||||
# console := true
|
# console := true
|
||||||
|
|
||||||
flags += -I. -I.. -O3
|
flags += -I. -I.. -O3
|
||||||
objects := libco emulator audio video
|
objects := libco audio video
|
||||||
|
|
||||||
# profile-guided optimization mode
|
# profile-guided optimization mode
|
||||||
# pgo := instrument
|
# pgo := instrument
|
||||||
|
@ -54,7 +54,6 @@ compile = \
|
||||||
all: build;
|
all: build;
|
||||||
|
|
||||||
obj/libco.o: ../libco/libco.c $(call rwildcard,../libco/)
|
obj/libco.o: ../libco/libco.c $(call rwildcard,../libco/)
|
||||||
obj/emulator.o: emulator/emulator.cpp $(call rwildcard,emulator/)
|
|
||||||
obj/audio.o: audio/audio.cpp $(call rwildcard,audio/)
|
obj/audio.o: audio/audio.cpp $(call rwildcard,audio/)
|
||||||
obj/video.o: video/video.cpp $(call rwildcard,video/)
|
obj/video.o: video/video.cpp $(call rwildcard,video/)
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,93 @@
|
||||||
#include <emulator/emulator.hpp>
|
#include <emulator/emulator.hpp>
|
||||||
|
|
||||||
namespace {
|
namespace Emulator {
|
||||||
|
|
||||||
|
Audio audio;
|
||||||
|
|
||||||
|
Stream::Stream(double inputFrequency, double outputFrequency, double volume, double balance) {
|
||||||
|
dsp.setChannels(2);
|
||||||
|
dsp.setPrecision(16);
|
||||||
|
dsp.setFrequency(inputFrequency);
|
||||||
|
dsp.setResampler(DSP::ResampleEngine::Sinc);
|
||||||
|
dsp.setResamplerFrequency(outputFrequency);
|
||||||
|
dsp.setVolume(volume);
|
||||||
|
dsp.setBalance(balance);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Stream::sample(int16 left, int16 right) -> void {
|
||||||
|
int samples[] = {left, right};
|
||||||
|
dsp.sample(samples);
|
||||||
|
audio.poll();
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
auto Audio::reset() -> void {
|
||||||
|
streams.reset();
|
||||||
|
setReverbDelay(reverbDelay);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setInterface(Interface* interface) -> void {
|
||||||
|
this->interface = interface;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setFrequency(double frequency) -> void {
|
||||||
|
this->frequency = frequency;
|
||||||
|
for(auto& stream : streams) stream->dsp.setResamplerFrequency(frequency);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setVolume(double volume) -> void {
|
||||||
|
this->volume = volume;
|
||||||
|
for(auto& stream : streams) stream->dsp.setVolume(volume);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setBalance(double balance) -> void {
|
||||||
|
this->balance = balance;
|
||||||
|
for(auto& stream : streams) stream->dsp.setBalance(balance);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setReverbDelay(uint reverbDelay) -> void {
|
||||||
|
this->reverbDelay = reverbDelay;
|
||||||
|
reverbLeft.resize(frequency * reverbDelay / 1000.0);
|
||||||
|
reverbRight.resize(frequency * reverbDelay / 1000.0);
|
||||||
|
memory::fill(reverbLeft.data(), reverbLeft.size() * sizeof(int16));
|
||||||
|
memory::fill(reverbRight.data(), reverbRight.size() * sizeof(int16));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::setReverbLevel(double reverbLevel) -> void {
|
||||||
|
this->reverbLevel = reverbLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Audio::createStream(double frequency) -> shared_pointer<Stream> {
|
||||||
|
shared_pointer<Stream> stream = new Stream{frequency, this->frequency, volume, balance};
|
||||||
|
streams.append(stream);
|
||||||
|
return stream;
|
||||||
|
}
|
||||||
|
|
||||||
|
//audio mixer
|
||||||
|
auto Audio::poll() -> void {
|
||||||
|
while(true) {
|
||||||
|
for(auto& stream : streams) {
|
||||||
|
if(!stream->dsp.pending()) return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int left = 0, right = 0;
|
||||||
|
for(auto& stream : streams) {
|
||||||
|
int samples[2];
|
||||||
|
stream->dsp.read(samples);
|
||||||
|
left += samples[0];
|
||||||
|
right += samples[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
if(reverbDelay) {
|
||||||
|
reverbLeft.append(left);
|
||||||
|
reverbRight.append(right);
|
||||||
|
left += reverbLeft.takeFirst() * reverbLevel;
|
||||||
|
right += reverbRight.takeFirst() * reverbLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface->audioSample(sclamp<16>(left), sclamp<16>(right));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,48 @@
|
||||||
|
|
||||||
#include "core.hpp"
|
#include "core.hpp"
|
||||||
|
|
||||||
namespace {
|
namespace Emulator {
|
||||||
|
|
||||||
|
struct Interface;
|
||||||
|
|
||||||
|
struct Stream {
|
||||||
|
Stream(double inputFrequency, double outputFrequency, double volume, double balance);
|
||||||
|
auto sample(int16 left, int16 right) -> void;
|
||||||
|
|
||||||
|
private:
|
||||||
|
nall::DSP dsp;
|
||||||
|
|
||||||
|
friend class Audio;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Audio {
|
||||||
|
auto reset() -> void;
|
||||||
|
auto setInterface(Interface*) -> void;
|
||||||
|
|
||||||
|
auto setFrequency(double frequency) -> void;
|
||||||
|
auto setVolume(double volume) -> void;
|
||||||
|
auto setBalance(double balance) -> void;
|
||||||
|
auto setReverbDelay(uint milliseconds) -> void;
|
||||||
|
auto setReverbLevel(double level) -> void;
|
||||||
|
|
||||||
|
auto createStream(double frequency) -> shared_pointer<Stream>;
|
||||||
|
|
||||||
|
auto poll() -> void;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Interface* interface = nullptr;
|
||||||
|
vector<shared_pointer<Stream>> streams;
|
||||||
|
double frequency = 0.0;
|
||||||
|
double volume = 1.0;
|
||||||
|
double balance = 0.0;
|
||||||
|
uint reverbDelay = 0; //0 = disabled
|
||||||
|
double reverbLevel = 0.0;
|
||||||
|
vector<int16> reverbLeft;
|
||||||
|
vector<int16> reverbRight;
|
||||||
|
|
||||||
|
friend class Stream;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern Audio audio;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,34 +0,0 @@
|
||||||
#include <emulator/emulator.hpp>
|
|
||||||
|
|
||||||
namespace Emulator {
|
|
||||||
|
|
||||||
auto Interface::videoColor(uint16 r, uint16 g, uint16 b) -> uint32 {
|
|
||||||
double saturation = 1.0;
|
|
||||||
double gamma = 1.0;
|
|
||||||
double luminance = 1.0;
|
|
||||||
|
|
||||||
if(saturation != 1.0) {
|
|
||||||
uint16 grayscale = uclamp<16>((r + g + b) / 3);
|
|
||||||
double inverse = max(0.0, 1.0 - saturation);
|
|
||||||
r = uclamp<16>(r * saturation + grayscale * inverse);
|
|
||||||
g = uclamp<16>(g * saturation + grayscale * inverse);
|
|
||||||
b = uclamp<16>(b * saturation + grayscale * inverse);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(gamma != 1.0) {
|
|
||||||
double reciprocal = 1.0 / 32767.0;
|
|
||||||
r = r > 32767 ? r : uint16(32767 * pow(r * reciprocal, gamma));
|
|
||||||
g = g > 32767 ? g : uint16(32767 * pow(g * reciprocal, gamma));
|
|
||||||
b = b > 32767 ? b : uint16(32767 * pow(b * reciprocal, gamma));
|
|
||||||
}
|
|
||||||
|
|
||||||
if(luminance != 1.0) {
|
|
||||||
r = uclamp<16>(r * luminance);
|
|
||||||
g = uclamp<16>(g * luminance);
|
|
||||||
b = uclamp<16>(b * luminance);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 255 << 24 | (r >> 8) << 16 | (g >> 8) << 8 | (b >> 8) << 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -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.05";
|
static const string Version = "098.06";
|
||||||
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/";
|
||||||
|
|
|
@ -75,7 +75,13 @@ struct Interface {
|
||||||
//information
|
//information
|
||||||
virtual auto manifest() -> string = 0;
|
virtual auto manifest() -> string = 0;
|
||||||
virtual auto title() -> string = 0;
|
virtual auto title() -> string = 0;
|
||||||
|
|
||||||
|
//video information
|
||||||
virtual auto videoFrequency() -> double = 0;
|
virtual auto videoFrequency() -> double = 0;
|
||||||
|
virtual auto videoColors() -> uint32 { return 1 << 19; }
|
||||||
|
virtual auto videoColor(uint32 color) -> uint64 { return 0; }
|
||||||
|
|
||||||
|
//audio information
|
||||||
virtual auto audioFrequency() -> double = 0;
|
virtual auto audioFrequency() -> double = 0;
|
||||||
|
|
||||||
//media interface
|
//media interface
|
||||||
|
|
|
@ -57,7 +57,7 @@ auto APU::main() -> void {
|
||||||
//output = filter.run_lopass(output);
|
//output = filter.run_lopass(output);
|
||||||
output = sclamp<16>(output);
|
output = sclamp<16>(output);
|
||||||
|
|
||||||
interface->audioSample(output, output);
|
stream->sample(output, output);
|
||||||
|
|
||||||
tick();
|
tick();
|
||||||
}
|
}
|
||||||
|
@ -89,6 +89,7 @@ auto APU::power() -> void {
|
||||||
|
|
||||||
auto APU::reset() -> void {
|
auto APU::reset() -> void {
|
||||||
create(APU::Enter, 21'477'272);
|
create(APU::Enter, 21'477'272);
|
||||||
|
stream = Emulator::audio.createStream(21'477'272.0 / 12.0);
|
||||||
|
|
||||||
pulse[0].reset();
|
pulse[0].reset();
|
||||||
pulse[1].reset();
|
pulse[1].reset();
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
struct APU : Thread {
|
struct APU : Thread {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
APU();
|
APU();
|
||||||
|
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
|
|
|
@ -22,7 +22,7 @@ namespace Famicom {
|
||||||
|
|
||||||
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
||||||
if(thread) co_delete(thread);
|
if(thread) co_delete(thread);
|
||||||
thread = co_create(262'144 * sizeof(void*), entrypoint);
|
thread = co_create(65'536 * sizeof(void*), entrypoint);
|
||||||
this->frequency = frequency;
|
this->frequency = frequency;
|
||||||
clock = 0;
|
clock = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,6 +58,61 @@ auto Interface::videoFrequency() -> double {
|
||||||
return 21477272.0 / (262.0 * 1364.0 - 4.0);
|
return 21477272.0 / (262.0 * 1364.0 - 4.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColors() -> uint32 {
|
||||||
|
return 1 << 9;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColor(uint32 n) -> uint64 {
|
||||||
|
double saturation = 2.0;
|
||||||
|
double hue = 0.0;
|
||||||
|
double contrast = 1.0;
|
||||||
|
double brightness = 1.0;
|
||||||
|
double gamma = settings.colorEmulation ? 1.8 : 2.2;
|
||||||
|
|
||||||
|
int color = (n & 0x0f), level = color < 0xe ? (n >> 4) & 3 : 1;
|
||||||
|
|
||||||
|
static const double black = 0.518, white = 1.962, attenuation = 0.746;
|
||||||
|
static const double levels[8] = {
|
||||||
|
0.350, 0.518, 0.962, 1.550,
|
||||||
|
1.094, 1.506, 1.962, 1.962,
|
||||||
|
};
|
||||||
|
|
||||||
|
double lo_and_hi[2] = {
|
||||||
|
levels[level + 4 * (color == 0x0)],
|
||||||
|
levels[level + 4 * (color < 0xd)],
|
||||||
|
};
|
||||||
|
|
||||||
|
double y = 0.0, i = 0.0, q = 0.0;
|
||||||
|
auto wave = [](int p, int color) { return (color + p + 8) % 12 < 6; };
|
||||||
|
for(int p : range(12)) {
|
||||||
|
double spot = lo_and_hi[wave(p, color)];
|
||||||
|
|
||||||
|
if(((n & 0x040) && wave(p, 12))
|
||||||
|
|| ((n & 0x080) && wave(p, 4))
|
||||||
|
|| ((n & 0x100) && wave(p, 8))
|
||||||
|
) spot *= attenuation;
|
||||||
|
|
||||||
|
double v = (spot - black) / (white - black);
|
||||||
|
|
||||||
|
v = (v - 0.5) * contrast + 0.5;
|
||||||
|
v *= brightness / 12.0;
|
||||||
|
|
||||||
|
y += v;
|
||||||
|
i += v * std::cos((3.141592653 / 6.0) * (p + hue));
|
||||||
|
q += v * std::sin((3.141592653 / 6.0) * (p + hue));
|
||||||
|
}
|
||||||
|
|
||||||
|
i *= saturation;
|
||||||
|
q *= saturation;
|
||||||
|
|
||||||
|
auto gammaAdjust = [=](double f) { return f < 0.0 ? 0.0 : std::pow(f, 2.2 / gamma); };
|
||||||
|
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 b = uclamp<16>(65535.0 * gammaAdjust(y + -1.108545 * i + 1.709007 * q));
|
||||||
|
|
||||||
|
return r << 32 | g << 16 | b << 0;
|
||||||
|
}
|
||||||
|
|
||||||
auto Interface::audioFrequency() -> double {
|
auto Interface::audioFrequency() -> double {
|
||||||
return 21477272.0 / 12.0;
|
return 21477272.0 / 12.0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,8 @@ struct Interface : Emulator::Interface {
|
||||||
auto manifest() -> string;
|
auto manifest() -> string;
|
||||||
auto title() -> string;
|
auto title() -> string;
|
||||||
auto videoFrequency() -> double;
|
auto videoFrequency() -> double;
|
||||||
|
auto videoColors() -> uint32;
|
||||||
|
auto videoColor(uint32 color) -> uint64;
|
||||||
auto audioFrequency() -> double;
|
auto audioFrequency() -> double;
|
||||||
|
|
||||||
auto loaded() -> bool;
|
auto loaded() -> bool;
|
||||||
|
|
|
@ -42,10 +42,13 @@ auto PPU::scanline() -> void {
|
||||||
|
|
||||||
auto PPU::frame() -> void {
|
auto PPU::frame() -> void {
|
||||||
status.field ^= 1;
|
status.field ^= 1;
|
||||||
Emulator::video.refresh(buffer, 256 * sizeof(uint32), 256, 240);
|
|
||||||
scheduler.exit(Scheduler::Event::Frame);
|
scheduler.exit(Scheduler::Event::Frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto PPU::refresh() -> void {
|
||||||
|
Emulator::video.refresh(buffer, 256 * sizeof(uint32), 256, 240);
|
||||||
|
}
|
||||||
|
|
||||||
auto PPU::power() -> void {
|
auto PPU::power() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@ struct PPU : Thread {
|
||||||
|
|
||||||
auto scanline() -> void;
|
auto scanline() -> void;
|
||||||
auto frame() -> void;
|
auto frame() -> void;
|
||||||
|
auto refresh() -> void;
|
||||||
|
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto reset() -> void;
|
auto reset() -> void;
|
||||||
|
|
|
@ -13,6 +13,7 @@ auto Scheduler::enter(Mode mode_) -> Event {
|
||||||
mode = mode_;
|
mode = mode_;
|
||||||
host = co_active();
|
host = co_active();
|
||||||
co_switch(resume);
|
co_switch(resume);
|
||||||
|
if(event == Event::Frame) ppu.refresh();
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,6 @@ auto System::load() -> void {
|
||||||
auto document = BML::unserialize(information.manifest);
|
auto document = BML::unserialize(information.manifest);
|
||||||
cartridge.load();
|
cartridge.load();
|
||||||
serializeInit();
|
serializeInit();
|
||||||
configureVideo();
|
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,6 +43,14 @@ auto System::power() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::reset() -> void {
|
auto System::reset() -> void {
|
||||||
|
Emulator::video.reset();
|
||||||
|
Emulator::video.setInterface(interface);
|
||||||
|
configureVideoPalette();
|
||||||
|
configureVideoEffects();
|
||||||
|
|
||||||
|
Emulator::audio.reset();
|
||||||
|
Emulator::audio.setInterface(interface);
|
||||||
|
|
||||||
cartridge.reset();
|
cartridge.reset();
|
||||||
cpu.reset();
|
cpu.reset();
|
||||||
apu.reset();
|
apu.reset();
|
||||||
|
|
|
@ -13,8 +13,8 @@ struct System {
|
||||||
auto term() -> void;
|
auto term() -> void;
|
||||||
|
|
||||||
//video.cpp
|
//video.cpp
|
||||||
auto configureVideo() -> void;
|
|
||||||
auto configureVideoPalette() -> void;
|
auto configureVideoPalette() -> void;
|
||||||
|
auto configureVideoEffects() -> void;
|
||||||
|
|
||||||
//serialization.cpp
|
//serialization.cpp
|
||||||
auto serialize() -> serializer;
|
auto serialize() -> serializer;
|
||||||
|
|
|
@ -1,57 +1,6 @@
|
||||||
auto System::configureVideo() -> void {
|
|
||||||
Emulator::video.reset();
|
|
||||||
Emulator::video.setInterface(interface);
|
|
||||||
configureVideoPalette();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto System::configureVideoPalette() -> void {
|
auto System::configureVideoPalette() -> void {
|
||||||
auto generateColor = [](uint n, double saturation, double hue, double contrast, double brightness, double gamma) -> uint64 {
|
Emulator::video.setPalette();
|
||||||
int color = (n & 0x0f), level = color < 0xe ? (n >> 4) & 3 : 1;
|
}
|
||||||
|
|
||||||
static const double black = 0.518, white = 1.962, attenuation = 0.746;
|
auto System::configureVideoEffects() -> void {
|
||||||
static const double levels[8] = {
|
|
||||||
0.350, 0.518, 0.962, 1.550,
|
|
||||||
1.094, 1.506, 1.962, 1.962,
|
|
||||||
};
|
|
||||||
|
|
||||||
double lo_and_hi[2] = {
|
|
||||||
levels[level + 4 * (color == 0x0)],
|
|
||||||
levels[level + 4 * (color < 0xd)],
|
|
||||||
};
|
|
||||||
|
|
||||||
double y = 0.0, i = 0.0, q = 0.0;
|
|
||||||
auto wave = [](int p, int color) { return (color + p + 8) % 12 < 6; };
|
|
||||||
for(int p : range(12)) {
|
|
||||||
double spot = lo_and_hi[wave(p, color)];
|
|
||||||
|
|
||||||
if(((n & 0x040) && wave(p, 12))
|
|
||||||
|| ((n & 0x080) && wave(p, 4))
|
|
||||||
|| ((n & 0x100) && wave(p, 8))
|
|
||||||
) spot *= attenuation;
|
|
||||||
|
|
||||||
double v = (spot - black) / (white - black);
|
|
||||||
|
|
||||||
v = (v - 0.5) * contrast + 0.5;
|
|
||||||
v *= brightness / 12.0;
|
|
||||||
|
|
||||||
y += v;
|
|
||||||
i += v * std::cos((3.141592653 / 6.0) * (p + hue));
|
|
||||||
q += v * std::sin((3.141592653 / 6.0) * (p + hue));
|
|
||||||
}
|
|
||||||
|
|
||||||
i *= saturation;
|
|
||||||
q *= saturation;
|
|
||||||
|
|
||||||
auto gammaAdjust = [=](double f) { return f < 0.0 ? 0.0 : std::pow(f, 2.2 / gamma); };
|
|
||||||
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 b = uclamp<16>(65535.0 * gammaAdjust(y + -1.108545 * i + 1.709007 * q));
|
|
||||||
|
|
||||||
return r << 32 | g << 16 | b << 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
Emulator::video.setPalette(1 << 9, [&](uint32 color) -> uint64 {
|
|
||||||
auto gamma = settings.colorEmulation ? 1.8 : 2.2;
|
|
||||||
return generateColor(color, 2.0, 0.0, 1.0, 1.0, gamma);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,7 +25,11 @@ auto APU::main() -> void {
|
||||||
hipass(sequencer.left, sequencer.leftBias);
|
hipass(sequencer.left, sequencer.leftBias);
|
||||||
hipass(sequencer.right, sequencer.rightBias);
|
hipass(sequencer.right, sequencer.rightBias);
|
||||||
|
|
||||||
|
if(!system.sgb()) {
|
||||||
|
stream->sample(sequencer.left, sequencer.right);
|
||||||
|
} else {
|
||||||
interface->audioSample(sequencer.left, sequencer.right);
|
interface->audioSample(sequencer.left, sequencer.right);
|
||||||
|
}
|
||||||
|
|
||||||
if(cycle == 0) { //512hz
|
if(cycle == 0) { //512hz
|
||||||
if(phase == 0 || phase == 2 || phase == 4 || phase == 6) { //256hz
|
if(phase == 0 || phase == 2 || phase == 4 || phase == 6) { //256hz
|
||||||
|
@ -58,6 +62,7 @@ auto APU::hipass(int16& sample, int64& bias) -> void {
|
||||||
|
|
||||||
auto APU::power() -> void {
|
auto APU::power() -> void {
|
||||||
create(Enter, 2 * 1024 * 1024);
|
create(Enter, 2 * 1024 * 1024);
|
||||||
|
if(!system.sgb()) stream = Emulator::audio.createStream(2 * 1024 * 1024);
|
||||||
for(uint n = 0xff10; n <= 0xff3f; n++) bus.mmio[n] = this;
|
for(uint n = 0xff10; n <= 0xff3f; n++) bus.mmio[n] = this;
|
||||||
|
|
||||||
square1.power();
|
square1.power();
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
struct APU : Thread, MMIO {
|
struct APU : Thread, MMIO {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
auto main() -> void;
|
auto main() -> void;
|
||||||
auto hipass(int16& sample, int64& bias) -> void;
|
auto hipass(int16& sample, int64& bias) -> void;
|
||||||
|
|
|
@ -4,6 +4,7 @@ namespace GameBoy {
|
||||||
|
|
||||||
#include "mbc0/mbc0.cpp"
|
#include "mbc0/mbc0.cpp"
|
||||||
#include "mbc1/mbc1.cpp"
|
#include "mbc1/mbc1.cpp"
|
||||||
|
#include "mbc1m/mbc1m.cpp"
|
||||||
#include "mbc2/mbc2.cpp"
|
#include "mbc2/mbc2.cpp"
|
||||||
#include "mbc3/mbc3.cpp"
|
#include "mbc3/mbc3.cpp"
|
||||||
#include "mbc5/mbc5.cpp"
|
#include "mbc5/mbc5.cpp"
|
||||||
|
@ -40,6 +41,7 @@ auto Cartridge::load(System::Revision revision) -> void {
|
||||||
auto mapperid = document["board/mapper"].text();
|
auto mapperid = document["board/mapper"].text();
|
||||||
if(mapperid == "none" ) information.mapper = Mapper::MBC0;
|
if(mapperid == "none" ) information.mapper = Mapper::MBC0;
|
||||||
if(mapperid == "MBC1" ) information.mapper = Mapper::MBC1;
|
if(mapperid == "MBC1" ) information.mapper = Mapper::MBC1;
|
||||||
|
if(mapperid == "MBC1M") information.mapper = Mapper::MBC1M;
|
||||||
if(mapperid == "MBC2" ) information.mapper = Mapper::MBC2;
|
if(mapperid == "MBC2" ) information.mapper = Mapper::MBC2;
|
||||||
if(mapperid == "MBC3" ) information.mapper = Mapper::MBC3;
|
if(mapperid == "MBC3" ) information.mapper = Mapper::MBC3;
|
||||||
if(mapperid == "MBC5" ) information.mapper = Mapper::MBC5;
|
if(mapperid == "MBC5" ) information.mapper = Mapper::MBC5;
|
||||||
|
@ -70,6 +72,7 @@ auto Cartridge::load(System::Revision revision) -> void {
|
||||||
switch(information.mapper) { default:
|
switch(information.mapper) { default:
|
||||||
case Mapper::MBC0: mapper = &mbc0; break;
|
case Mapper::MBC0: mapper = &mbc0; break;
|
||||||
case Mapper::MBC1: mapper = &mbc1; break;
|
case Mapper::MBC1: mapper = &mbc1; break;
|
||||||
|
case Mapper::MBC1M: mapper = &mbc1m; break;
|
||||||
case Mapper::MBC2: mapper = &mbc2; break;
|
case Mapper::MBC2: mapper = &mbc2; break;
|
||||||
case Mapper::MBC3: mapper = &mbc3; break;
|
case Mapper::MBC3: mapper = &mbc3; break;
|
||||||
case Mapper::MBC5: mapper = &mbc5; break;
|
case Mapper::MBC5: mapper = &mbc5; break;
|
||||||
|
@ -139,6 +142,7 @@ auto Cartridge::power() -> void {
|
||||||
|
|
||||||
mbc0.power();
|
mbc0.power();
|
||||||
mbc1.power();
|
mbc1.power();
|
||||||
|
mbc1m.power();
|
||||||
mbc2.power();
|
mbc2.power();
|
||||||
mbc3.power();
|
mbc3.power();
|
||||||
mbc5.power();
|
mbc5.power();
|
||||||
|
|
|
@ -16,6 +16,7 @@ struct Cartridge : MMIO, property<Cartridge> {
|
||||||
|
|
||||||
#include "mbc0/mbc0.hpp"
|
#include "mbc0/mbc0.hpp"
|
||||||
#include "mbc1/mbc1.hpp"
|
#include "mbc1/mbc1.hpp"
|
||||||
|
#include "mbc1m/mbc1m.hpp"
|
||||||
#include "mbc2/mbc2.hpp"
|
#include "mbc2/mbc2.hpp"
|
||||||
#include "mbc3/mbc3.hpp"
|
#include "mbc3/mbc3.hpp"
|
||||||
#include "mbc5/mbc5.hpp"
|
#include "mbc5/mbc5.hpp"
|
||||||
|
@ -26,6 +27,7 @@ struct Cartridge : MMIO, property<Cartridge> {
|
||||||
enum Mapper : uint {
|
enum Mapper : uint {
|
||||||
MBC0,
|
MBC0,
|
||||||
MBC1,
|
MBC1,
|
||||||
|
MBC1M,
|
||||||
MBC2,
|
MBC2,
|
||||||
MBC3,
|
MBC3,
|
||||||
MBC5,
|
MBC5,
|
||||||
|
|
|
@ -0,0 +1,26 @@
|
||||||
|
auto Cartridge::MBC1M::mmio_read(uint16 addr) -> uint8 {
|
||||||
|
if((addr & 0xc000) == 0x0000) { //$0000-3fff
|
||||||
|
return cartridge.rom_read((romHi << 4) * 0x4000 + addr.bits(0,13));
|
||||||
|
}
|
||||||
|
|
||||||
|
if((addr & 0xc000) == 0x4000) { //$4000-7fff
|
||||||
|
return cartridge.rom_read((romHi << 4 | romLo) * 0x4000 + addr.bits(0,13));
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Cartridge::MBC1M::mmio_write(uint16 addr, uint8 data) -> void {
|
||||||
|
if((addr & 0xe000) == 0x2000) { //$2000-3fff
|
||||||
|
romLo = data.bits(0,3);
|
||||||
|
}
|
||||||
|
|
||||||
|
if((addr & 0xe000) == 0x4000) { //$4000-5fff
|
||||||
|
romHi = data.bits(0,1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Cartridge::MBC1M::power() -> void {
|
||||||
|
romHi = 0;
|
||||||
|
romLo = 1;
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
struct MBC1M : MMIO {
|
||||||
|
auto mmio_read(uint16 addr) -> uint8;
|
||||||
|
auto mmio_write(uint16 addr, uint8 data) -> void;
|
||||||
|
auto power() -> void;
|
||||||
|
|
||||||
|
uint4 romLo;
|
||||||
|
uint2 romHi;
|
||||||
|
} mbc1m;
|
|
@ -7,6 +7,9 @@ auto Cartridge::serialize(serializer& s) -> void {
|
||||||
s.integer(mbc1.ram_select);
|
s.integer(mbc1.ram_select);
|
||||||
s.integer(mbc1.mode_select);
|
s.integer(mbc1.mode_select);
|
||||||
|
|
||||||
|
s.integer(mbc1m.romLo);
|
||||||
|
s.integer(mbc1m.romHi);
|
||||||
|
|
||||||
s.integer(mbc2.ram_enable);
|
s.integer(mbc2.ram_enable);
|
||||||
s.integer(mbc2.rom_select);
|
s.integer(mbc2.rom_select);
|
||||||
|
|
||||||
|
|
|
@ -22,7 +22,7 @@ namespace GameBoy {
|
||||||
|
|
||||||
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
||||||
if(thread) co_delete(thread);
|
if(thread) co_delete(thread);
|
||||||
thread = co_create(262'144 * sizeof(void*), entrypoint);
|
thread = co_create(65'536 * sizeof(void*), entrypoint);
|
||||||
this->frequency = frequency;
|
this->frequency = frequency;
|
||||||
clock = 0;
|
clock = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,6 +52,67 @@ auto Interface::videoFrequency() -> double {
|
||||||
return 4194304.0 / (154.0 * 456.0);
|
return 4194304.0 / (154.0 * 456.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColors() -> uint32 {
|
||||||
|
return !system.cgb() ? 1 << 2 : 1 << 15;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColor(uint32 color) -> uint64 {
|
||||||
|
if(!system.cgb()) {
|
||||||
|
if(!settings.colorEmulation) {
|
||||||
|
uint64 L = image::normalize(3 - color, 2, 16);
|
||||||
|
return L << 32 | L << 16 | L << 0;
|
||||||
|
} else {
|
||||||
|
#define DMG_PALETTE_GREEN
|
||||||
|
//#define DMG_PALETTE_YELLOW
|
||||||
|
//#define DMG_PALETTE_WHITE
|
||||||
|
|
||||||
|
const uint16 monochrome[4][3] = {
|
||||||
|
#if defined(DMG_PALETTE_GREEN)
|
||||||
|
{0xaeae, 0xd9d9, 0x2727},
|
||||||
|
{0x5858, 0xa0a0, 0x2828},
|
||||||
|
{0x2020, 0x6262, 0x2929},
|
||||||
|
{0x1a1a, 0x4545, 0x2a2a},
|
||||||
|
#elif defined(DMG_PALETTE_YELLOW)
|
||||||
|
{0xffff, 0xf7f7, 0x7b7b},
|
||||||
|
{0xb5b5, 0xaeae, 0x4a4a},
|
||||||
|
{0x6b6b, 0x6969, 0x3131},
|
||||||
|
{0x2121, 0x2020, 0x1010},
|
||||||
|
#elif defined(DMG_PALETTE_WHITE)
|
||||||
|
{0xffff, 0xffff, 0xffff},
|
||||||
|
{0xaaaa, 0xaaaa, 0xaaaa},
|
||||||
|
{0x5555, 0x5555, 0x5555},
|
||||||
|
{0x0000, 0x0000, 0x0000},
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
uint64 R = monochrome[color][0];
|
||||||
|
uint64 G = monochrome[color][1];
|
||||||
|
uint64 B = monochrome[color][2];
|
||||||
|
|
||||||
|
return R << 32 | G << 16 | B << 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
uint r = color.bits( 0, 4);
|
||||||
|
uint g = color.bits( 5, 9);
|
||||||
|
uint b = color.bits(10,14);
|
||||||
|
|
||||||
|
uint64_t R = image::normalize(r, 5, 16);
|
||||||
|
uint64_t G = image::normalize(g, 5, 16);
|
||||||
|
uint64_t B = image::normalize(b, 5, 16);
|
||||||
|
|
||||||
|
if(settings.colorEmulation) {
|
||||||
|
R = (r * 26 + g * 4 + b * 2);
|
||||||
|
G = ( g * 24 + b * 8);
|
||||||
|
B = (r * 6 + g * 4 + b * 22);
|
||||||
|
R = image::normalize(min(960, R), 10, 16);
|
||||||
|
G = image::normalize(min(960, G), 10, 16);
|
||||||
|
B = image::normalize(min(960, B), 10, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
return R << 32 | G << 16 | B << 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
auto Interface::audioFrequency() -> double {
|
auto Interface::audioFrequency() -> double {
|
||||||
return 4194304.0 / 2.0;
|
return 4194304.0 / 2.0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,8 @@ struct Interface : Emulator::Interface {
|
||||||
auto manifest() -> string;
|
auto manifest() -> string;
|
||||||
auto title() -> string;
|
auto title() -> string;
|
||||||
auto videoFrequency() -> double;
|
auto videoFrequency() -> double;
|
||||||
|
auto videoColors() -> uint32;
|
||||||
|
auto videoColor(uint32 color) -> uint64;
|
||||||
auto audioFrequency() -> double;
|
auto audioFrequency() -> double;
|
||||||
|
|
||||||
auto loaded() -> bool;
|
auto loaded() -> bool;
|
||||||
|
|
|
@ -60,11 +60,14 @@ auto PPU::main() -> void {
|
||||||
|
|
||||||
if(++status.ly == 154) {
|
if(++status.ly == 154) {
|
||||||
status.ly = 0;
|
status.ly = 0;
|
||||||
if(!system.sgb()) Emulator::video.refresh(screen, 160 * sizeof(uint32), 160, 144);
|
|
||||||
scheduler.exit(Scheduler::Event::Frame);
|
scheduler.exit(Scheduler::Event::Frame);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto PPU::refresh() -> void {
|
||||||
|
if(!system.sgb()) Emulator::video.refresh(screen, 160 * sizeof(uint32), 160, 144);
|
||||||
|
}
|
||||||
|
|
||||||
auto PPU::add_clocks(uint clocks) -> void {
|
auto PPU::add_clocks(uint clocks) -> void {
|
||||||
while(clocks--) {
|
while(clocks--) {
|
||||||
if(status.dma_active) {
|
if(status.dma_active) {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
struct PPU : Thread, MMIO {
|
struct PPU : Thread, MMIO {
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
auto main() -> void;
|
auto main() -> void;
|
||||||
|
auto refresh() -> void;
|
||||||
auto add_clocks(uint clocks) -> void;
|
auto add_clocks(uint clocks) -> void;
|
||||||
|
|
||||||
auto hflip(uint data) const -> uint;
|
auto hflip(uint data) const -> uint;
|
||||||
|
|
|
@ -13,6 +13,7 @@ auto Scheduler::enter(Mode mode_) -> Event {
|
||||||
mode = mode_;
|
mode = mode_;
|
||||||
host = co_active();
|
host = co_active();
|
||||||
co_switch(resume);
|
co_switch(resume);
|
||||||
|
if(event == Event::Frame) ppu.refresh();
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -50,7 +50,6 @@ auto System::load(Revision revision) -> void {
|
||||||
|
|
||||||
cartridge.load(revision);
|
cartridge.load(revision);
|
||||||
serializeInit();
|
serializeInit();
|
||||||
configureVideo();
|
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,6 +60,16 @@ auto System::unload() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::power() -> void {
|
auto System::power() -> void {
|
||||||
|
if(!system.sgb()) {
|
||||||
|
Emulator::video.reset();
|
||||||
|
Emulator::video.setInterface(interface);
|
||||||
|
configureVideoPalette();
|
||||||
|
configureVideoEffects();
|
||||||
|
|
||||||
|
Emulator::audio.reset();
|
||||||
|
Emulator::audio.setInterface(interface);
|
||||||
|
}
|
||||||
|
|
||||||
bus.power();
|
bus.power();
|
||||||
cartridge.power();
|
cartridge.power();
|
||||||
cpu.power();
|
cpu.power();
|
||||||
|
|
|
@ -30,7 +30,6 @@ struct System {
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
|
|
||||||
//video.cpp
|
//video.cpp
|
||||||
auto configureVideo() -> void;
|
|
||||||
auto configureVideoPalette() -> void;
|
auto configureVideoPalette() -> void;
|
||||||
auto configureVideoEffects() -> void;
|
auto configureVideoEffects() -> void;
|
||||||
|
|
||||||
|
|
|
@ -1,75 +1,9 @@
|
||||||
auto System::configureVideo() -> void {
|
|
||||||
if(sgb()) return;
|
|
||||||
|
|
||||||
Emulator::video.reset();
|
|
||||||
Emulator::video.setInterface(interface);
|
|
||||||
configureVideoPalette();
|
|
||||||
configureVideoEffects();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto System::configureVideoPalette() -> void {
|
auto System::configureVideoPalette() -> void {
|
||||||
if(sgb()) return;
|
if(sgb()) return;
|
||||||
|
Emulator::video.setPalette();
|
||||||
if(dmg()) Emulator::video.setPalette(1 << 2, [&](uint32 color) -> uint64 {
|
|
||||||
if(!settings.colorEmulation) {
|
|
||||||
uint64 L = image::normalize(3 - color, 2, 16);
|
|
||||||
return L << 32 | L << 16 | L << 0;
|
|
||||||
} else {
|
|
||||||
#define DMG_PALETTE_GREEN
|
|
||||||
//#define DMG_PALETTE_YELLOW
|
|
||||||
//#define DMG_PALETTE_WHITE
|
|
||||||
|
|
||||||
const uint16 monochrome[4][3] = {
|
|
||||||
#if defined(DMG_PALETTE_GREEN)
|
|
||||||
{0xaeae, 0xd9d9, 0x2727},
|
|
||||||
{0x5858, 0xa0a0, 0x2828},
|
|
||||||
{0x2020, 0x6262, 0x2929},
|
|
||||||
{0x1a1a, 0x4545, 0x2a2a},
|
|
||||||
#elif defined(DMG_PALETTE_YELLOW)
|
|
||||||
{0xffff, 0xf7f7, 0x7b7b},
|
|
||||||
{0xb5b5, 0xaeae, 0x4a4a},
|
|
||||||
{0x6b6b, 0x6969, 0x3131},
|
|
||||||
{0x2121, 0x2020, 0x1010},
|
|
||||||
#elif defined(DMG_PALETTE_WHITE)
|
|
||||||
{0xffff, 0xffff, 0xffff},
|
|
||||||
{0xaaaa, 0xaaaa, 0xaaaa},
|
|
||||||
{0x5555, 0x5555, 0x5555},
|
|
||||||
{0x0000, 0x0000, 0x0000},
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
uint64 R = monochrome[color][0];
|
|
||||||
uint64 G = monochrome[color][1];
|
|
||||||
uint64 B = monochrome[color][2];
|
|
||||||
|
|
||||||
return R << 32 | G << 16 | B << 0;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if(cgb()) Emulator::video.setPalette(1 << 15, [&](uint32 color) -> uint64 {
|
|
||||||
uint r = color.bits( 0, 4);
|
|
||||||
uint g = color.bits( 5, 9);
|
|
||||||
uint b = color.bits(10,14);
|
|
||||||
|
|
||||||
uint64_t R = image::normalize(r, 5, 16);
|
|
||||||
uint64_t G = image::normalize(g, 5, 16);
|
|
||||||
uint64_t B = image::normalize(b, 5, 16);
|
|
||||||
|
|
||||||
if(settings.colorEmulation) {
|
|
||||||
R = (r * 26 + g * 4 + b * 2);
|
|
||||||
G = ( g * 24 + b * 8);
|
|
||||||
B = (r * 6 + g * 4 + b * 22);
|
|
||||||
R = image::normalize(min(960, R), 10, 16);
|
|
||||||
G = image::normalize(min(960, G), 10, 16);
|
|
||||||
B = image::normalize(min(960, B), 10, 16);
|
|
||||||
}
|
|
||||||
|
|
||||||
return R << 32 | G << 16 | B << 0;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::configureVideoEffects() -> void {
|
auto System::configureVideoEffects() -> void {
|
||||||
if(sgb()) return;
|
if(sgb()) return;
|
||||||
|
|
||||||
Emulator::video.setEffect(Emulator::Video::Effect::InterframeBlending, settings.blurEmulation);
|
Emulator::video.setEffect(Emulator::Video::Effect::InterframeBlending, settings.blurEmulation);
|
||||||
}
|
}
|
||||||
|
|
|
@ -63,7 +63,7 @@ auto APU::main() -> void {
|
||||||
if(regs.bias.amplitude == 3) lsample &= ~15, rsample &= ~15;
|
if(regs.bias.amplitude == 3) lsample &= ~15, rsample &= ~15;
|
||||||
|
|
||||||
if(cpu.regs.mode == CPU::Registers::Mode::Stop) lsample = 0, rsample = 0;
|
if(cpu.regs.mode == CPU::Registers::Mode::Stop) lsample = 0, rsample = 0;
|
||||||
interface->audioSample(sclamp<16>(lsample << 6), sclamp<16>(rsample << 6)); //should be <<5, use <<6 for added volume
|
stream->sample(sclamp<16>(lsample << 6), sclamp<16>(rsample << 6)); //should be <<5; use <<6 for added volume
|
||||||
step(512);
|
step(512);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -74,6 +74,7 @@ auto APU::step(uint clocks) -> void {
|
||||||
|
|
||||||
auto APU::power() -> void {
|
auto APU::power() -> void {
|
||||||
create(APU::Enter, 16'777'216);
|
create(APU::Enter, 16'777'216);
|
||||||
|
stream = Emulator::audio.createStream(16'777'216.0 / 512.0);
|
||||||
|
|
||||||
square1.power();
|
square1.power();
|
||||||
square2.power();
|
square2.power();
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
struct APU : Thread, MMIO {
|
struct APU : Thread, MMIO {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
#include "registers.hpp"
|
#include "registers.hpp"
|
||||||
|
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
auto CPU::bus_idle() -> void {
|
auto CPU::busIdle() -> void {
|
||||||
prefetch_step(1);
|
prefetch_step(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::bus_read(unsigned mode, uint32 addr) -> uint32 {
|
auto CPU::busRead(uint mode, uint32 addr) -> uint32 {
|
||||||
unsigned wait = bus_wait(mode, addr);
|
uint wait = busWait(mode, addr);
|
||||||
unsigned word = pipeline.fetch.instruction;
|
uint word = pipeline.fetch.instruction;
|
||||||
|
|
||||||
if(addr >= 0x1000'0000) {
|
if(addr >= 0x1000'0000) {
|
||||||
prefetch_step(wait);
|
prefetch_step(wait);
|
||||||
|
@ -35,8 +35,8 @@ auto CPU::bus_read(unsigned mode, uint32 addr) -> uint32 {
|
||||||
return word;
|
return word;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::bus_write(unsigned mode, uint32 addr, uint32 word) -> void {
|
auto CPU::busWrite(uint mode, uint32 addr, uint32 word) -> void {
|
||||||
unsigned wait = bus_wait(mode, addr);
|
uint wait = busWait(mode, addr);
|
||||||
|
|
||||||
if(addr >= 0x1000'0000) {
|
if(addr >= 0x1000'0000) {
|
||||||
prefetch_step(wait);
|
prefetch_step(wait);
|
||||||
|
@ -57,7 +57,7 @@ auto CPU::bus_write(unsigned mode, uint32 addr, uint32 word) -> void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::bus_wait(unsigned mode, uint32 addr) -> unsigned {
|
auto CPU::busWait(uint mode, uint32 addr) -> uint {
|
||||||
if(addr >= 0x1000'0000) return 1; //unmapped
|
if(addr >= 0x1000'0000) return 1; //unmapped
|
||||||
if(addr < 0x0200'0000) return 1;
|
if(addr < 0x0200'0000) return 1;
|
||||||
if(addr < 0x0300'0000) return (16 - regs.memory.control.ewramwait) * (mode & Word ? 2 : 1);
|
if(addr < 0x0300'0000) return (16 - regs.memory.control.ewramwait) * (mode & Word ? 2 : 1);
|
||||||
|
@ -65,9 +65,9 @@ auto CPU::bus_wait(unsigned mode, uint32 addr) -> unsigned {
|
||||||
if(addr < 0x0700'0000) return mode & Word ? 2 : 1;
|
if(addr < 0x0700'0000) return mode & Word ? 2 : 1;
|
||||||
if(addr < 0x0800'0000) return 1;
|
if(addr < 0x0800'0000) return 1;
|
||||||
|
|
||||||
static unsigned timings[] = {5, 4, 3, 9};
|
static uint timings[] = {5, 4, 3, 9};
|
||||||
unsigned n = timings[regs.wait.control.nwait[addr >> 25 & 3]];
|
uint n = timings[regs.wait.control.nwait[addr >> 25 & 3]];
|
||||||
unsigned s = regs.wait.control.swait[addr >> 25 & 3];
|
uint s = regs.wait.control.swait[addr >> 25 & 3];
|
||||||
|
|
||||||
switch(addr & 0x0e00'0000) {
|
switch(addr & 0x0e00'0000) {
|
||||||
case 0x0800'0000: s = s ? 2 : 3; break;
|
case 0x0800'0000: s = s ? 2 : 3; break;
|
||||||
|
@ -79,7 +79,7 @@ auto CPU::bus_wait(unsigned mode, uint32 addr) -> unsigned {
|
||||||
bool sequential = (mode & Sequential);
|
bool sequential = (mode & Sequential);
|
||||||
if((addr & 0x1fffe) == 0) sequential = false; //N cycle on 16-bit ROM crossing 128KB page boundary (RAM S==N)
|
if((addr & 0x1fffe) == 0) sequential = false; //N cycle on 16-bit ROM crossing 128KB page boundary (RAM S==N)
|
||||||
|
|
||||||
unsigned clocks = sequential ? s : n;
|
uint clocks = sequential ? s : n;
|
||||||
if(mode & Word) clocks += s; //16-bit bus requires two transfers for words
|
if(mode & Word) clocks += s; //16-bit bus requires two transfers for words
|
||||||
return clocks;
|
return clocks;
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,10 +39,10 @@ struct CPU : Processor::ARM, Thread, MMIO {
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
|
|
||||||
//bus.cpp
|
//bus.cpp
|
||||||
auto bus_idle() -> void override;
|
auto busIdle() -> void override;
|
||||||
auto bus_read(uint mode, uint32 addr) -> uint32 override;
|
auto busRead(uint mode, uint32 addr) -> uint32 override;
|
||||||
auto bus_write(uint mode, uint32 addr, uint32 word) -> void override;
|
auto busWrite(uint mode, uint32 addr, uint32 word) -> void override;
|
||||||
auto bus_wait(uint mode, uint32 addr) -> uint;
|
auto busWait(uint mode, uint32 addr) -> uint;
|
||||||
|
|
||||||
//mmio.cpp
|
//mmio.cpp
|
||||||
auto read(uint32 addr) -> uint8;
|
auto read(uint32 addr) -> uint8;
|
||||||
|
|
|
@ -39,7 +39,7 @@ auto CPU::dma_exec(Registers::DMA& dma) -> void {
|
||||||
uint32 addr = dma.run.source;
|
uint32 addr = dma.run.source;
|
||||||
if(mode & Word) addr &= ~3;
|
if(mode & Word) addr &= ~3;
|
||||||
if(mode & Half) addr &= ~1;
|
if(mode & Half) addr &= ~1;
|
||||||
dma.data = bus_read(mode, addr);
|
dma.data = busRead(mode, addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(dma.run.target < 0x0200'0000) {
|
if(dma.run.target < 0x0200'0000) {
|
||||||
|
@ -48,7 +48,7 @@ auto CPU::dma_exec(Registers::DMA& dma) -> void {
|
||||||
uint32 addr = dma.run.target;
|
uint32 addr = dma.run.target;
|
||||||
if(mode & Word) addr &= ~3;
|
if(mode & Word) addr &= ~3;
|
||||||
if(mode & Half) addr &= ~1;
|
if(mode & Half) addr &= ~1;
|
||||||
bus_write(mode, addr, dma.data);
|
busWrite(mode, addr, dma.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
switch(dma.control.sourcemode) {
|
switch(dma.control.sourcemode) {
|
||||||
|
|
|
@ -3,7 +3,7 @@ auto CPU::prefetch_sync(uint32 addr) -> void {
|
||||||
|
|
||||||
prefetch.addr = addr;
|
prefetch.addr = addr;
|
||||||
prefetch.load = addr;
|
prefetch.load = addr;
|
||||||
prefetch.wait = bus_wait(Half | Nonsequential, prefetch.load);
|
prefetch.wait = busWait(Half | Nonsequential, prefetch.load);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::prefetch_step(uint clocks) -> void {
|
auto CPU::prefetch_step(uint clocks) -> void {
|
||||||
|
@ -14,7 +14,7 @@ auto CPU::prefetch_step(uint clocks) -> void {
|
||||||
if(--prefetch.wait) continue;
|
if(--prefetch.wait) continue;
|
||||||
prefetch.slot[prefetch.load >> 1 & 7] = cartridge.read(Half, prefetch.load);
|
prefetch.slot[prefetch.load >> 1 & 7] = cartridge.read(Half, prefetch.load);
|
||||||
prefetch.load += 2;
|
prefetch.load += 2;
|
||||||
prefetch.wait = bus_wait(Half | Sequential, prefetch.load);
|
prefetch.wait = busWait(Half | Sequential, prefetch.load);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,14 +22,14 @@ auto CPU::prefetch_wait() -> void {
|
||||||
if(!regs.wait.control.prefetch || active.dma || prefetch.full()) return;
|
if(!regs.wait.control.prefetch || active.dma || prefetch.full()) return;
|
||||||
|
|
||||||
prefetch_step(prefetch.wait);
|
prefetch_step(prefetch.wait);
|
||||||
prefetch.wait = bus_wait(Half | Nonsequential, prefetch.load);
|
prefetch.wait = busWait(Half | Nonsequential, prefetch.load);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto CPU::prefetch_read() -> uint16 {
|
auto CPU::prefetch_read() -> uint16 {
|
||||||
if(prefetch.empty()) prefetch_step(prefetch.wait);
|
if(prefetch.empty()) prefetch_step(prefetch.wait);
|
||||||
else prefetch_step(1);
|
else prefetch_step(1);
|
||||||
|
|
||||||
if(prefetch.full()) prefetch.wait = bus_wait(Half | Sequential, prefetch.load);
|
if(prefetch.full()) prefetch.wait = busWait(Half | Sequential, prefetch.load);
|
||||||
|
|
||||||
uint16 half = prefetch.slot[prefetch.addr >> 1 & 7];
|
uint16 half = prefetch.slot[prefetch.addr >> 1 & 7];
|
||||||
prefetch.addr += 2;
|
prefetch.addr += 2;
|
||||||
|
|
|
@ -34,7 +34,7 @@ namespace GameBoyAdvance {
|
||||||
|
|
||||||
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
||||||
if(thread) co_delete(thread);
|
if(thread) co_delete(thread);
|
||||||
thread = co_create(262'144 * sizeof(void*), entrypoint);
|
thread = co_create(65'536 * sizeof(void*), entrypoint);
|
||||||
this->frequency = frequency;
|
this->frequency = frequency;
|
||||||
clock = 0;
|
clock = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,6 +52,32 @@ auto Interface::videoFrequency() -> double {
|
||||||
return 16777216.0 / (228.0 * 1232.0);
|
return 16777216.0 / (228.0 * 1232.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColors() -> uint32 {
|
||||||
|
return 1 << 15;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColor(uint32 color) -> uint64 {
|
||||||
|
uint R = color.bits( 0, 4);
|
||||||
|
uint G = color.bits( 5, 9);
|
||||||
|
uint B = color.bits(10,14);
|
||||||
|
|
||||||
|
uint64 r = image::normalize(R, 5, 16);
|
||||||
|
uint64 g = image::normalize(G, 5, 16);
|
||||||
|
uint64 b = image::normalize(B, 5, 16);
|
||||||
|
|
||||||
|
if(settings.colorEmulation) {
|
||||||
|
double lcdGamma = 4.0, outGamma = 2.2;
|
||||||
|
double lb = pow(B / 31.0, lcdGamma);
|
||||||
|
double lg = pow(G / 31.0, lcdGamma);
|
||||||
|
double lr = pow(R / 31.0, lcdGamma);
|
||||||
|
r = pow(( 0 * lb + 50 * lg + 255 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
||||||
|
g = pow(( 30 * lb + 230 * lg + 10 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
||||||
|
b = pow((220 * lb + 10 * lg + 50 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
||||||
|
}
|
||||||
|
|
||||||
|
return r << 32 | g << 16 | b << 0;
|
||||||
|
}
|
||||||
|
|
||||||
auto Interface::audioFrequency() -> double {
|
auto Interface::audioFrequency() -> double {
|
||||||
return 16777216.0 / 512.0;
|
return 16777216.0 / 512.0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,8 @@ struct Interface : Emulator::Interface {
|
||||||
auto manifest() -> string;
|
auto manifest() -> string;
|
||||||
auto title() -> string;
|
auto title() -> string;
|
||||||
auto videoFrequency() -> double;
|
auto videoFrequency() -> double;
|
||||||
|
auto videoColors() -> uint32;
|
||||||
|
auto videoColor(uint32 color) -> uint64;
|
||||||
auto audioFrequency() -> double;
|
auto audioFrequency() -> double;
|
||||||
|
|
||||||
auto loaded() -> bool;
|
auto loaded() -> bool;
|
||||||
|
|
|
@ -178,8 +178,11 @@ auto PPU::scanline() -> void {
|
||||||
|
|
||||||
auto PPU::frame() -> void {
|
auto PPU::frame() -> void {
|
||||||
player.frame();
|
player.frame();
|
||||||
Emulator::video.refresh(output, 240 * sizeof(uint32), 240, 160);
|
|
||||||
scheduler.exit(Scheduler::Event::Frame);
|
scheduler.exit(Scheduler::Event::Frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto PPU::refresh() -> void {
|
||||||
|
Emulator::video.refresh(output, 240 * sizeof(uint32), 240, 160);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,7 @@ struct PPU : Thread, MMIO {
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto scanline() -> void;
|
auto scanline() -> void;
|
||||||
auto frame() -> void;
|
auto frame() -> void;
|
||||||
|
auto refresh() -> void;
|
||||||
|
|
||||||
auto read(uint32 addr) -> uint8;
|
auto read(uint32 addr) -> uint8;
|
||||||
auto write(uint32 addr, uint8 byte) -> void;
|
auto write(uint32 addr, uint8 byte) -> void;
|
||||||
|
|
|
@ -13,6 +13,7 @@ auto Scheduler::enter(Mode mode_) -> Event {
|
||||||
mode = mode_;
|
mode = mode_;
|
||||||
host = co_active();
|
host = co_active();
|
||||||
co_switch(resume);
|
co_switch(resume);
|
||||||
|
if(event == Event::Frame) ppu.refresh();
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,14 @@ auto System::term() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::power() -> void {
|
auto System::power() -> void {
|
||||||
|
Emulator::video.reset();
|
||||||
|
Emulator::video.setInterface(interface);
|
||||||
|
configureVideoPalette();
|
||||||
|
configureVideoEffects();
|
||||||
|
|
||||||
|
Emulator::audio.reset();
|
||||||
|
Emulator::audio.setInterface(interface);
|
||||||
|
|
||||||
bus.power();
|
bus.power();
|
||||||
player.power();
|
player.power();
|
||||||
cpu.power();
|
cpu.power();
|
||||||
|
@ -36,7 +44,6 @@ auto System::load() -> void {
|
||||||
|
|
||||||
cartridge.load();
|
cartridge.load();
|
||||||
serializeInit();
|
serializeInit();
|
||||||
configureVideo();
|
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,6 @@ struct System {
|
||||||
auto runToSave() -> void;
|
auto runToSave() -> void;
|
||||||
|
|
||||||
//video.cpp
|
//video.cpp
|
||||||
auto configureVideo() -> void;
|
|
||||||
auto configureVideoPalette() -> void;
|
auto configureVideoPalette() -> void;
|
||||||
auto configureVideoEffects() -> void;
|
auto configureVideoEffects() -> void;
|
||||||
|
|
||||||
|
|
|
@ -1,32 +1,5 @@
|
||||||
auto System::configureVideo() -> void {
|
|
||||||
Emulator::video.reset();
|
|
||||||
Emulator::video.setInterface(interface);
|
|
||||||
configureVideoPalette();
|
|
||||||
configureVideoEffects();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto System::configureVideoPalette() -> void {
|
auto System::configureVideoPalette() -> void {
|
||||||
Emulator::video.setPalette(1 << 15, [&](uint32 color) -> uint64 {
|
Emulator::video.setPalette();
|
||||||
uint R = color.bits( 0, 4);
|
|
||||||
uint G = color.bits( 5, 9);
|
|
||||||
uint B = color.bits(10,14);
|
|
||||||
|
|
||||||
uint64 r = image::normalize(R, 5, 16);
|
|
||||||
uint64 g = image::normalize(G, 5, 16);
|
|
||||||
uint64 b = image::normalize(B, 5, 16);
|
|
||||||
|
|
||||||
if(settings.colorEmulation) {
|
|
||||||
double lcdGamma = 4.0, outGamma = 2.2;
|
|
||||||
double lb = pow(B / 31.0, lcdGamma);
|
|
||||||
double lg = pow(G / 31.0, lcdGamma);
|
|
||||||
double lr = pow(R / 31.0, lcdGamma);
|
|
||||||
r = pow(( 0 * lb + 50 * lg + 255 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
|
||||||
g = pow(( 30 * lb + 230 * lg + 10 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
|
||||||
b = pow((220 * lb + 10 * lg + 50 * lr) / 255, 1 / outGamma) * (0xffff * 255 / 280);
|
|
||||||
}
|
|
||||||
|
|
||||||
return r << 32 | g << 16 | b << 0;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::configureVideoEffects() -> void {
|
auto System::configureVideoEffects() -> void {
|
||||||
|
|
|
@ -31,16 +31,16 @@ auto ARM::exec() -> void {
|
||||||
|
|
||||||
auto ARM::idle() -> void {
|
auto ARM::idle() -> void {
|
||||||
pipeline.nonsequential = true;
|
pipeline.nonsequential = true;
|
||||||
return bus_idle();
|
return busIdle();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ARM::read(unsigned mode, uint32 addr) -> uint32 {
|
auto ARM::read(unsigned mode, uint32 addr) -> uint32 {
|
||||||
return bus_read(mode, addr);
|
return busRead(mode, addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ARM::load(unsigned mode, uint32 addr) -> uint32 {
|
auto ARM::load(unsigned mode, uint32 addr) -> uint32 {
|
||||||
pipeline.nonsequential = true;
|
pipeline.nonsequential = true;
|
||||||
uint32 word = bus_read(Load | mode, addr);
|
uint32 word = busRead(Load | mode, addr);
|
||||||
|
|
||||||
if(mode & Half) {
|
if(mode & Half) {
|
||||||
addr &= 1;
|
addr &= 1;
|
||||||
|
@ -64,7 +64,7 @@ auto ARM::load(unsigned mode, uint32 addr) -> uint32 {
|
||||||
|
|
||||||
auto ARM::write(unsigned mode, uint32 addr, uint32 word) -> void {
|
auto ARM::write(unsigned mode, uint32 addr, uint32 word) -> void {
|
||||||
pipeline.nonsequential = true;
|
pipeline.nonsequential = true;
|
||||||
return bus_write(mode, addr, word);
|
return busWrite(mode, addr, word);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ARM::store(unsigned mode, uint32 addr, uint32 word) -> void {
|
auto ARM::store(unsigned mode, uint32 addr, uint32 word) -> void {
|
||||||
|
@ -73,7 +73,7 @@ auto ARM::store(unsigned mode, uint32 addr, uint32 word) -> void {
|
||||||
if(mode & Half) { word &= 0xffff; word |= word << 16; }
|
if(mode & Half) { word &= 0xffff; word |= word << 16; }
|
||||||
if(mode & Byte) { word &= 0xff; word |= word << 8; word |= word << 16; }
|
if(mode & Byte) { word &= 0xff; word |= word << 8; word |= word << 16; }
|
||||||
|
|
||||||
return bus_write(Store | mode, addr, word);
|
return busWrite(Store | mode, addr, word);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ARM::vector(uint32 addr, Processor::Mode mode) -> void {
|
auto ARM::vector(uint32 addr, Processor::Mode mode) -> void {
|
||||||
|
|
|
@ -25,9 +25,9 @@ struct ARM {
|
||||||
#include "disassembler.hpp"
|
#include "disassembler.hpp"
|
||||||
|
|
||||||
virtual auto step(unsigned clocks) -> void = 0;
|
virtual auto step(unsigned clocks) -> void = 0;
|
||||||
virtual auto bus_idle() -> void = 0;
|
virtual auto busIdle() -> void = 0;
|
||||||
virtual auto bus_read(unsigned mode, uint32 addr) -> uint32 = 0;
|
virtual auto busRead(unsigned mode, uint32 addr) -> uint32 = 0;
|
||||||
virtual auto bus_write(unsigned mode, uint32 addr, uint32 word) -> void = 0;
|
virtual auto busWrite(unsigned mode, uint32 addr, uint32 word) -> void = 0;
|
||||||
|
|
||||||
//arm.cpp
|
//arm.cpp
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
|
|
|
@ -10,8 +10,8 @@ namespace Processor {
|
||||||
auto HG51B::exec(uint24 addr) -> void {
|
auto HG51B::exec(uint24 addr) -> void {
|
||||||
if(regs.halt) return;
|
if(regs.halt) return;
|
||||||
addr = addr + regs.pc * 2;
|
addr = addr + regs.pc * 2;
|
||||||
opcode = bus_read(addr++) << 0;
|
opcode = read(addr++) << 0;
|
||||||
opcode |= bus_read(addr++) << 8;
|
opcode |= read(addr++) << 8;
|
||||||
regs.pc = (regs.pc & 0xffff00) | ((regs.pc + 1) & 0x0000ff);
|
regs.pc = (regs.pc & 0xffff00) | ((regs.pc + 1) & 0x0000ff);
|
||||||
instruction();
|
instruction();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
//Hitachi HG51B169 (HG51BS family/derivative?)
|
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
//Hitachi HG51B169 (HG51BS family/derivative?)
|
||||||
|
|
||||||
namespace Processor {
|
namespace Processor {
|
||||||
|
|
||||||
struct HG51B {
|
struct HG51B {
|
||||||
auto exec(uint24 addr) -> void;
|
auto exec(uint24 addr) -> void;
|
||||||
virtual auto bus_read(uint24 addr) -> uint8 = 0;
|
virtual auto read(uint24 addr) -> uint8 = 0;
|
||||||
virtual auto bus_write(uint24 addr, uint8 data) -> void = 0;
|
virtual auto write(uint24 addr, uint8 data) -> void = 0;
|
||||||
|
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto serialize(serializer&) -> void;
|
auto serialize(serializer&) -> void;
|
||||||
|
@ -25,8 +25,8 @@ protected:
|
||||||
auto instruction() -> void;
|
auto instruction() -> void;
|
||||||
|
|
||||||
//registers.cpp
|
//registers.cpp
|
||||||
auto reg_read(uint8 addr) const -> uint24;
|
auto regRead(uint8 addr) const -> uint24;
|
||||||
auto reg_write(uint8 addr, uint24 data) -> void;
|
auto regWrite(uint8 addr, uint24 data) -> void;
|
||||||
|
|
||||||
struct Registers {
|
struct Registers {
|
||||||
bool halt;
|
bool halt;
|
||||||
|
|
|
@ -34,7 +34,7 @@ auto HG51B::sa() -> uint {
|
||||||
//Register-or-Immediate: most opcodes can load from a register or immediate
|
//Register-or-Immediate: most opcodes can load from a register or immediate
|
||||||
auto HG51B::ri() -> uint {
|
auto HG51B::ri() -> uint {
|
||||||
if(opcode & 0x0400) return opcode & 0xff;
|
if(opcode & 0x0400) return opcode & 0xff;
|
||||||
return reg_read(opcode & 0xff);
|
return regRead(opcode & 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
//New-PC: determine jump target address; opcode.d9 = long jump flag (1 = yes)
|
//New-PC: determine jump target address; opcode.d9 = long jump flag (1 = yes)
|
||||||
|
@ -115,7 +115,7 @@ auto HG51B::instruction() -> void {
|
||||||
else if((opcode & 0xffff) == 0x4000) {
|
else if((opcode & 0xffff) == 0x4000) {
|
||||||
//0100 0000 0000 0000
|
//0100 0000 0000 0000
|
||||||
//rdbus
|
//rdbus
|
||||||
regs.busdata = bus_read(regs.busaddr++);
|
regs.busdata = read(regs.busaddr++);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if((opcode & 0xf800) == 0x4800) {
|
else if((opcode & 0xf800) == 0x4800) {
|
||||||
|
@ -306,7 +306,7 @@ auto HG51B::instruction() -> void {
|
||||||
else if((opcode & 0xff00) == 0xe000) {
|
else if((opcode & 0xff00) == 0xe000) {
|
||||||
//1110 0000 .... ....
|
//1110 0000 .... ....
|
||||||
//st r,a
|
//st r,a
|
||||||
reg_write(opcode & 0xff, regs.a);
|
regWrite(opcode & 0xff, regs.a);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if((opcode & 0xfb00) == 0xe800) {
|
else if((opcode & 0xfb00) == 0xe800) {
|
||||||
|
@ -333,10 +333,10 @@ auto HG51B::instruction() -> void {
|
||||||
else if((opcode & 0xff00) == 0xf000) {
|
else if((opcode & 0xff00) == 0xf000) {
|
||||||
//1111 0000 .... ....
|
//1111 0000 .... ....
|
||||||
//swap a,r
|
//swap a,r
|
||||||
uint24 source = reg_read(opcode & 0xff);
|
uint24 source = regRead(opcode & 0xff);
|
||||||
uint24 target = regs.a;
|
uint24 target = regs.a;
|
||||||
regs.a = source;
|
regs.a = source;
|
||||||
reg_write(opcode & 0xff, target);
|
regWrite(opcode & 0xff, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if((opcode & 0xffff) == 0xfc00) {
|
else if((opcode & 0xffff) == 0xfc00) {
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
auto HG51B::reg_read(uint8 addr) const -> uint24 {
|
auto HG51B::regRead(uint8 addr) const -> uint24 {
|
||||||
switch(addr) {
|
switch(addr) {
|
||||||
case 0x00: return regs.a;
|
case 0x00: return regs.a;
|
||||||
case 0x01: return regs.acch;
|
case 0x01: return regs.acch;
|
||||||
|
@ -44,7 +44,7 @@ auto HG51B::reg_read(uint8 addr) const -> uint24 {
|
||||||
return 0x000000;
|
return 0x000000;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HG51B::reg_write(uint8 addr, uint24 data) -> void {
|
auto HG51B::regWrite(uint8 addr, uint24 data) -> void {
|
||||||
switch(addr) {
|
switch(addr) {
|
||||||
case 0x00: regs.a = data; return;
|
case 0x00: regs.a = data; return;
|
||||||
case 0x01: regs.acch = data; return;
|
case 0x01: regs.acch = data; return;
|
||||||
|
|
|
@ -98,7 +98,7 @@ auto Cartridge::parseMarkupMCC(Markup::Node root) -> void {
|
||||||
|
|
||||||
for(auto node : root.find("map")) {
|
for(auto node : root.find("map")) {
|
||||||
if(node.text() == "mcu") {
|
if(node.text() == "mcu") {
|
||||||
parseMarkupMap(node, {&MCC::mcu_read, &mcc}, {&MCC::mcu_write, &mcc});
|
parseMarkupMap(node, {&MCC::mcuRead, &mcc}, {&MCC::mcuWrite, &mcc});
|
||||||
} else {
|
} else {
|
||||||
parseMarkupMap(node, {&MCC::read, &mcc}, {&MCC::write, &mcc});
|
parseMarkupMap(node, {&MCC::read, &mcc}, {&MCC::write, &mcc});
|
||||||
}
|
}
|
||||||
|
@ -231,7 +231,7 @@ auto Cartridge::parseMarkupARMDSP(Markup::Node root) -> void {
|
||||||
memory.append({ID::ArmDSPRAM, root["dram"]["name"].text()});
|
memory.append({ID::ArmDSPRAM, root["dram"]["name"].text()});
|
||||||
|
|
||||||
for(auto node : root.find("map")) {
|
for(auto node : root.find("map")) {
|
||||||
parseMarkupMap(node, {&ArmDSP::mmio_read, &armdsp}, {&ArmDSP::mmio_write, &armdsp});
|
parseMarkupMap(node, {&ArmDSP::read, &armdsp}, {&ArmDSP::write, &armdsp});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -252,19 +252,19 @@ auto Cartridge::parseMarkupHitachiDSP(Markup::Node root, uint roms) -> void {
|
||||||
parseMarkupMemory(hitachidsp.ram, root["ram"], ID::HitachiDSPRAM, true);
|
parseMarkupMemory(hitachidsp.ram, root["ram"], ID::HitachiDSPRAM, true);
|
||||||
|
|
||||||
for(auto node : root.find("map")) {
|
for(auto node : root.find("map")) {
|
||||||
parseMarkupMap(node, {&HitachiDSP::dsp_read, &hitachidsp}, {&HitachiDSP::dsp_write, &hitachidsp});
|
parseMarkupMap(node, {&HitachiDSP::dspRead, &hitachidsp}, {&HitachiDSP::dspWrite, &hitachidsp});
|
||||||
}
|
}
|
||||||
|
|
||||||
for(auto node : root["rom"].find("map")) {
|
for(auto node : root["rom"].find("map")) {
|
||||||
parseMarkupMap(node, {&HitachiDSP::rom_read, &hitachidsp}, {&HitachiDSP::rom_write, &hitachidsp});
|
parseMarkupMap(node, {&HitachiDSP::romRead, &hitachidsp}, {&HitachiDSP::romWrite, &hitachidsp});
|
||||||
}
|
}
|
||||||
|
|
||||||
for(auto node : root["ram"].find("map")) {
|
for(auto node : root["ram"].find("map")) {
|
||||||
parseMarkupMap(node, {&HitachiDSP::ram_read, &hitachidsp}, {&HitachiDSP::ram_write, &hitachidsp});
|
parseMarkupMap(node, {&HitachiDSP::ramRead, &hitachidsp}, {&HitachiDSP::ramWrite, &hitachidsp});
|
||||||
}
|
}
|
||||||
|
|
||||||
for(auto node : root["dram"].find("map")) {
|
for(auto node : root["dram"].find("map")) {
|
||||||
parseMarkupMap(node, {&HitachiDSP::dram_read, &hitachidsp}, {&HitachiDSP::dram_write, &hitachidsp});
|
parseMarkupMap(node, {&HitachiDSP::dramRead, &hitachidsp}, {&HitachiDSP::dramWrite, &hitachidsp});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -380,6 +380,6 @@ auto Cartridge::parseMarkupMSU1(Markup::Node root) -> void {
|
||||||
hasMSU1 = true;
|
hasMSU1 = true;
|
||||||
|
|
||||||
for(auto node : root.find("map")) {
|
for(auto node : root.find("map")) {
|
||||||
parseMarkupMap(node, {&MSU1::mmioRead, &msu1}, {&MSU1::mmioWrite, &msu1});
|
parseMarkupMap(node, {&MSU1::read, &msu1}, {&MSU1::write, &msu1});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,7 @@ auto ArmDSP::boot() -> void {
|
||||||
|
|
||||||
//reset sequence delay
|
//reset sequence delay
|
||||||
if(bridge.ready == false) {
|
if(bridge.ready == false) {
|
||||||
step(65536);
|
step(65'536);
|
||||||
bridge.ready = true;
|
bridge.ready = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -58,7 +58,7 @@ auto ArmDSP::step(uint clocks) -> void {
|
||||||
//3800-3807 mirrored throughout
|
//3800-3807 mirrored throughout
|
||||||
//a0 ignored
|
//a0 ignored
|
||||||
|
|
||||||
auto ArmDSP::mmio_read(uint24 addr, uint8) -> uint8 {
|
auto ArmDSP::read(uint24 addr, uint8) -> uint8 {
|
||||||
cpu.synchronizeCoprocessors();
|
cpu.synchronizeCoprocessors();
|
||||||
|
|
||||||
uint8 data = 0x00;
|
uint8 data = 0x00;
|
||||||
|
@ -82,7 +82,7 @@ auto ArmDSP::mmio_read(uint24 addr, uint8) -> uint8 {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ArmDSP::mmio_write(uint24 addr, uint8 data) -> void {
|
auto ArmDSP::write(uint24 addr, uint8 data) -> void {
|
||||||
cpu.synchronizeCoprocessors();
|
cpu.synchronizeCoprocessors();
|
||||||
|
|
||||||
addr &= 0xff06;
|
addr &= 0xff06;
|
||||||
|
|
|
@ -11,12 +11,12 @@ struct ArmDSP : Processor::ARM, Cothread {
|
||||||
auto main() -> void;
|
auto main() -> void;
|
||||||
|
|
||||||
auto step(uint clocks) -> void override;
|
auto step(uint clocks) -> void override;
|
||||||
auto bus_idle() -> void override;
|
auto busIdle() -> void override;
|
||||||
auto bus_read(uint mode, uint32 addr) -> uint32 override;
|
auto busRead(uint mode, uint32 addr) -> uint32 override;
|
||||||
auto bus_write(uint mode, uint32 addr, uint32 word) -> void override;
|
auto busWrite(uint mode, uint32 addr, uint32 word) -> void override;
|
||||||
|
|
||||||
auto mmio_read(uint24 addr, uint8 data) -> uint8;
|
auto read(uint24 addr, uint8 data) -> uint8;
|
||||||
auto mmio_write(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
auto init() -> void;
|
auto init() -> void;
|
||||||
auto load() -> void;
|
auto load() -> void;
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
//note: timings are completely unverified
|
//note: timings are completely unverified
|
||||||
//due to the ST018 chip design (on-die ROM), testing is nearly impossible
|
//due to the ST018 chip design (on-die ROM), testing is nearly impossible
|
||||||
|
|
||||||
auto ArmDSP::bus_idle() -> void {
|
auto ArmDSP::busIdle() -> void {
|
||||||
step(1);
|
step(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ArmDSP::bus_read(unsigned mode, uint32 addr) -> uint32 {
|
auto ArmDSP::busRead(uint mode, uint32 addr) -> uint32 {
|
||||||
step(1);
|
step(1);
|
||||||
|
|
||||||
static auto memory = [&](const uint8* memory, uint mode, uint32 addr) -> uint32 {
|
static auto memory = [&](const uint8* memory, uint mode, uint32 addr) -> uint32 {
|
||||||
|
@ -19,34 +19,34 @@ auto ArmDSP::bus_read(unsigned mode, uint32 addr) -> uint32 {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
switch(addr & 0xe0000000) {
|
switch(addr & 0xe000'0000) {
|
||||||
case 0x00000000: return memory(programROM, mode, addr & 0x1ffff);
|
case 0x0000'0000: return memory(programROM, mode, addr & 0x1ffff);
|
||||||
case 0x20000000: return pipeline.fetch.instruction;
|
case 0x2000'0000: return pipeline.fetch.instruction;
|
||||||
case 0x40000000: break;
|
case 0x4000'0000: break;
|
||||||
case 0x60000000: return 0x40404001;
|
case 0x6000'0000: return 0x40404001;
|
||||||
case 0x80000000: return pipeline.fetch.instruction;
|
case 0x8000'0000: return pipeline.fetch.instruction;
|
||||||
case 0xa0000000: return memory(dataROM, mode, addr & 0x7fff);
|
case 0xa000'0000: return memory(dataROM, mode, addr & 0x7fff);
|
||||||
case 0xc0000000: return pipeline.fetch.instruction;
|
case 0xc000'0000: return pipeline.fetch.instruction;
|
||||||
case 0xe0000000: return memory(programRAM, mode, addr & 0x3fff);
|
case 0xe000'0000: return memory(programRAM, mode, addr & 0x3fff);
|
||||||
}
|
}
|
||||||
|
|
||||||
addr &= 0xe000003f;
|
addr &= 0xe000'003f;
|
||||||
|
|
||||||
if(addr == 0x40000010) {
|
if(addr == 0x4000'0010) {
|
||||||
if(bridge.cputoarm.ready) {
|
if(bridge.cputoarm.ready) {
|
||||||
bridge.cputoarm.ready = false;
|
bridge.cputoarm.ready = false;
|
||||||
return bridge.cputoarm.data;
|
return bridge.cputoarm.data;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(addr == 0x40000020) {
|
if(addr == 0x4000'0020) {
|
||||||
return bridge.status();
|
return bridge.status();
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ArmDSP::bus_write(uint mode, uint32 addr, uint32 word) -> void {
|
auto ArmDSP::busWrite(uint mode, uint32 addr, uint32 word) -> void {
|
||||||
step(1);
|
step(1);
|
||||||
|
|
||||||
static auto memory = [](uint8* memory, uint mode, uint32 addr, uint32 word) {
|
static auto memory = [](uint8* memory, uint mode, uint32 addr, uint32 word) {
|
||||||
|
@ -62,37 +62,30 @@ auto ArmDSP::bus_write(uint mode, uint32 addr, uint32 word) -> void {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
switch(addr & 0xe0000000) {
|
switch(addr & 0xe000'0000) {
|
||||||
case 0x00000000: return;
|
case 0x0000'0000: return;
|
||||||
case 0x20000000: return;
|
case 0x2000'0000: return;
|
||||||
case 0x40000000: break;
|
case 0x4000'0000: break;
|
||||||
case 0x60000000: return;
|
case 0x6000'0000: return;
|
||||||
case 0x80000000: return;
|
case 0x8000'0000: return;
|
||||||
case 0xa0000000: return;
|
case 0xa000'0000: return;
|
||||||
case 0xc0000000: return;
|
case 0xc000'0000: return;
|
||||||
case 0xe0000000: return memory(programRAM, mode, addr & 0x3fff, word);
|
case 0xe000'0000: return memory(programRAM, mode, addr & 0x3fff, word);
|
||||||
}
|
}
|
||||||
|
|
||||||
addr &= 0xe000003f;
|
addr &= 0xe000'003f;
|
||||||
word &= 0x000000ff;
|
word &= 0x0000'00ff;
|
||||||
|
|
||||||
if(addr == 0x40000000) {
|
if(addr == 0x4000'0000) {
|
||||||
bridge.armtocpu.ready = true;
|
bridge.armtocpu.ready = true;
|
||||||
bridge.armtocpu.data = word;
|
bridge.armtocpu.data = word;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(addr == 0x40000010) {
|
if(addr == 0x4000'0010) bridge.signal = true;
|
||||||
bridge.signal = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(addr == 0x40000020) { bridge.timerlatch = (bridge.timerlatch & 0xffff00) | (word << 0); return; }
|
if(addr == 0x4000'0020) bridge.timerlatch.byte(0) = word;
|
||||||
if(addr == 0x40000024) { bridge.timerlatch = (bridge.timerlatch & 0xff00ff) | (word << 8); return; }
|
if(addr == 0x4000'0024) bridge.timerlatch.byte(1) = word;
|
||||||
if(addr == 0x40000028) { bridge.timerlatch = (bridge.timerlatch & 0x00ffff) | (word << 16); return; }
|
if(addr == 0x4000'0028) bridge.timerlatch.byte(2) = word;
|
||||||
|
|
||||||
if(addr == 0x4000002c) {
|
if(addr == 0x4000'002c) bridge.timer = bridge.timerlatch;
|
||||||
bridge.timer = bridge.timerlatch;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,11 @@ struct Bridge {
|
||||||
bool signal;
|
bool signal;
|
||||||
|
|
||||||
auto status() const -> uint8 {
|
auto status() const -> uint8 {
|
||||||
return (ready << 7) | (cputoarm.ready << 3) | (signal << 2) | (armtocpu.ready << 0);
|
return (
|
||||||
|
armtocpu.ready << 0
|
||||||
|
| signal << 2
|
||||||
|
| cputoarm.ready << 3
|
||||||
|
| ready << 7
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} bridge;
|
} bridge;
|
||||||
|
|
|
@ -15,7 +15,7 @@ auto EpsonRTC::main() -> void {
|
||||||
if(wait) { if(--wait == 0) ready = 1; }
|
if(wait) { if(--wait == 0) ready = 1; }
|
||||||
|
|
||||||
clocks++;
|
clocks++;
|
||||||
if((clocks & ~0x00ff) == 0) round_seconds(); //125 microseconds
|
if((clocks & ~0x00ff) == 0) roundSeconds(); //125 microseconds
|
||||||
if((clocks & ~0x3fff) == 0) duty(); //1/128th second
|
if((clocks & ~0x3fff) == 0) duty(); //1/128th second
|
||||||
if((clocks & ~0x7fff) == 0) irq(0); //1/64th second
|
if((clocks & ~0x7fff) == 0) irq(0); //1/64th second
|
||||||
if(clocks == 0) { //1 second
|
if(clocks == 0) { //1 second
|
||||||
|
@ -81,7 +81,7 @@ auto EpsonRTC::power() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::reset() -> void {
|
auto EpsonRTC::reset() -> void {
|
||||||
create(EpsonRTC::Enter, 32768 * 64);
|
create(EpsonRTC::Enter, 32'768 * 64);
|
||||||
|
|
||||||
clocks = 0;
|
clocks = 0;
|
||||||
seconds = 0;
|
seconds = 0;
|
||||||
|
@ -150,7 +150,7 @@ auto EpsonRTC::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
if(state != State::Read) return 0;
|
if(state != State::Read) return 0;
|
||||||
ready = 0;
|
ready = 0;
|
||||||
wait = 8;
|
wait = 8;
|
||||||
return rtc_read(offset++);
|
return rtcRead(offset++);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(addr == 2) {
|
if(addr == 2) {
|
||||||
|
@ -166,7 +166,7 @@ auto EpsonRTC::write(uint24 addr, uint8 data) -> void {
|
||||||
|
|
||||||
if(addr == 0) {
|
if(addr == 0) {
|
||||||
chipselect = data;
|
chipselect = data;
|
||||||
if(chipselect != 1) rtc_reset();
|
if(chipselect != 1) rtcReset();
|
||||||
ready = 1;
|
ready = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -192,7 +192,7 @@ auto EpsonRTC::write(uint24 addr, uint8 data) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
else if(state == State::Write) {
|
else if(state == State::Write) {
|
||||||
rtc_write(offset++, data);
|
rtcWrite(offset++, data);
|
||||||
ready = 0;
|
ready = 0;
|
||||||
wait = 8;
|
wait = 8;
|
||||||
mdr = data;
|
mdr = data;
|
||||||
|
|
|
@ -67,9 +67,9 @@ struct EpsonRTC : Cothread {
|
||||||
uint1 test;
|
uint1 test;
|
||||||
|
|
||||||
//memory.cpp
|
//memory.cpp
|
||||||
auto rtc_reset() -> void;
|
auto rtcReset() -> void;
|
||||||
auto rtc_read(uint4 addr) -> uint4;
|
auto rtcRead(uint4 addr) -> uint4;
|
||||||
auto rtc_write(uint4 addr, uint4 data) -> void;
|
auto rtcWrite(uint4 addr, uint4 data) -> void;
|
||||||
|
|
||||||
auto load(const uint8* data) -> void;
|
auto load(const uint8* data) -> void;
|
||||||
auto save(uint8* data) -> void;
|
auto save(uint8* data) -> void;
|
||||||
|
@ -77,15 +77,15 @@ struct EpsonRTC : Cothread {
|
||||||
//time.cpp
|
//time.cpp
|
||||||
auto irq(uint2 period) -> void;
|
auto irq(uint2 period) -> void;
|
||||||
auto duty() -> void;
|
auto duty() -> void;
|
||||||
auto round_seconds() -> void;
|
auto roundSeconds() -> void;
|
||||||
auto tick() -> void;
|
auto tick() -> void;
|
||||||
|
|
||||||
auto tick_second() -> void;
|
auto tickSecond() -> void;
|
||||||
auto tick_minute() -> void;
|
auto tickMinute() -> void;
|
||||||
auto tick_hour() -> void;
|
auto tickHour() -> void;
|
||||||
auto tick_day() -> void;
|
auto tickDay() -> void;
|
||||||
auto tick_month() -> void;
|
auto tickMonth() -> void;
|
||||||
auto tick_year() -> void;
|
auto tickYear() -> void;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern EpsonRTC epsonrtc;
|
extern EpsonRTC epsonrtc;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
auto EpsonRTC::rtc_reset() -> void {
|
auto EpsonRTC::rtcReset() -> void {
|
||||||
state = State::Mode;
|
state = State::Mode;
|
||||||
offset = 0;
|
offset = 0;
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@ auto EpsonRTC::rtc_reset() -> void {
|
||||||
test = 0;
|
test = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::rtc_read(uint4 addr) -> uint4 {
|
auto EpsonRTC::rtcRead(uint4 addr) -> uint4 {
|
||||||
switch(addr) { default:
|
switch(addr) { default:
|
||||||
case 0: return secondlo;
|
case 0: return secondlo;
|
||||||
case 1: return secondhi | batteryfailure << 3;
|
case 1: return secondhi | batteryfailure << 3;
|
||||||
|
@ -32,7 +32,7 @@ auto EpsonRTC::rtc_read(uint4 addr) -> uint4 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::rtc_write(uint4 addr, uint4 data) -> void {
|
auto EpsonRTC::rtcWrite(uint4 addr, uint4 data) -> void {
|
||||||
switch(addr) {
|
switch(addr) {
|
||||||
case 0:
|
case 0:
|
||||||
secondlo = data;
|
secondlo = data;
|
||||||
|
@ -87,7 +87,7 @@ auto EpsonRTC::rtc_write(uint4 addr, uint4 data) -> void {
|
||||||
if(held == 1 && hold == 0 && holdtick == 1) {
|
if(held == 1 && hold == 0 && holdtick == 1) {
|
||||||
//if a second has passed during hold, increment one second upon resuming
|
//if a second has passed during hold, increment one second upon resuming
|
||||||
holdtick = 0;
|
holdtick = 0;
|
||||||
tick_second();
|
tickSecond();
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
case 14:
|
case 14:
|
||||||
|
@ -156,10 +156,10 @@ auto EpsonRTC::load(const uint8* data) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64 diff = (uint64)time(0) - timestamp;
|
uint64 diff = (uint64)time(0) - timestamp;
|
||||||
while(diff >= 60 * 60 * 24) { tick_day(); diff -= 60 * 60 * 24; }
|
while(diff >= 60 * 60 * 24) { tickDay(); diff -= 60 * 60 * 24; }
|
||||||
while(diff >= 60 * 60) { tick_hour(); diff -= 60 * 60; }
|
while(diff >= 60 * 60) { tickHour(); diff -= 60 * 60; }
|
||||||
while(diff >= 60) { tick_minute(); diff -= 60; }
|
while(diff >= 60) { tickMinute(); diff -= 60; }
|
||||||
while(diff--) tick_second();
|
while(diff--) tickSecond();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::save(uint8* data) -> void {
|
auto EpsonRTC::save(uint8* data) -> void {
|
||||||
|
|
|
@ -8,11 +8,11 @@ auto EpsonRTC::duty() -> void {
|
||||||
if(irqduty) irqflag = 0;
|
if(irqduty) irqflag = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::round_seconds() -> void {
|
auto EpsonRTC::roundSeconds() -> void {
|
||||||
if(roundseconds == 0) return;
|
if(roundseconds == 0) return;
|
||||||
roundseconds = 0;
|
roundseconds = 0;
|
||||||
|
|
||||||
if(secondhi >= 3) tick_minute();
|
if(secondhi >= 3) tickMinute();
|
||||||
secondlo = 0;
|
secondlo = 0;
|
||||||
secondhi = 0;
|
secondhi = 0;
|
||||||
}
|
}
|
||||||
|
@ -26,13 +26,13 @@ auto EpsonRTC::tick() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
resync = 1;
|
resync = 1;
|
||||||
tick_second();
|
tickSecond();
|
||||||
}
|
}
|
||||||
|
|
||||||
//below code provides bit-perfect emulation of invalid BCD values on the RTC-4513
|
//below code provides bit-perfect emulation of invalid BCD values on the RTC-4513
|
||||||
//code makes extensive use of variable-length integers (see epsonrtc.hpp for sizes)
|
//code makes extensive use of variable-length integers (see epsonrtc.hpp for sizes)
|
||||||
|
|
||||||
auto EpsonRTC::tick_second() -> void {
|
auto EpsonRTC::tickSecond() -> void {
|
||||||
if(secondlo <= 8 || secondlo == 12) {
|
if(secondlo <= 8 || secondlo == 12) {
|
||||||
secondlo++;
|
secondlo++;
|
||||||
} else {
|
} else {
|
||||||
|
@ -41,12 +41,12 @@ auto EpsonRTC::tick_second() -> void {
|
||||||
secondhi++;
|
secondhi++;
|
||||||
} else {
|
} else {
|
||||||
secondhi = 0;
|
secondhi = 0;
|
||||||
tick_minute();
|
tickMinute();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::tick_minute() -> void {
|
auto EpsonRTC::tickMinute() -> void {
|
||||||
if(minutelo <= 8 || minutelo == 12) {
|
if(minutelo <= 8 || minutelo == 12) {
|
||||||
minutelo++;
|
minutelo++;
|
||||||
} else {
|
} else {
|
||||||
|
@ -55,12 +55,12 @@ auto EpsonRTC::tick_minute() -> void {
|
||||||
minutehi++;
|
minutehi++;
|
||||||
} else {
|
} else {
|
||||||
minutehi = 0;
|
minutehi = 0;
|
||||||
tick_hour();
|
tickHour();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::tick_hour() -> void {
|
auto EpsonRTC::tickHour() -> void {
|
||||||
if(atime) {
|
if(atime) {
|
||||||
if(hourhi < 2) {
|
if(hourhi < 2) {
|
||||||
if(hourlo <= 8 || hourlo == 12) {
|
if(hourlo <= 8 || hourlo == 12) {
|
||||||
|
@ -80,7 +80,7 @@ auto EpsonRTC::tick_hour() -> void {
|
||||||
} else {
|
} else {
|
||||||
hourlo = !(hourlo & 1);
|
hourlo = !(hourlo & 1);
|
||||||
hourhi = 0;
|
hourhi = 0;
|
||||||
tick_day();
|
tickDay();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -99,12 +99,12 @@ auto EpsonRTC::tick_hour() -> void {
|
||||||
hourlo = !(hourlo & 1);
|
hourlo = !(hourlo & 1);
|
||||||
hourhi ^= 1;
|
hourhi ^= 1;
|
||||||
}
|
}
|
||||||
if(meridian == 0 && !(hourlo & 1)) tick_day();
|
if(meridian == 0 && !(hourlo & 1)) tickDay();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::tick_day() -> void {
|
auto EpsonRTC::tickDay() -> void {
|
||||||
if(calendar == 0) return;
|
if(calendar == 0) return;
|
||||||
weekday = (weekday + 1) + (weekday == 6);
|
weekday = (weekday + 1) + (weekday == 6);
|
||||||
|
|
||||||
|
@ -124,25 +124,25 @@ auto EpsonRTC::tick_day() -> void {
|
||||||
if(days == 28 && (dayhi == 3 || (dayhi == 2 && daylo >= 8))) {
|
if(days == 28 && (dayhi == 3 || (dayhi == 2 && daylo >= 8))) {
|
||||||
daylo = 1;
|
daylo = 1;
|
||||||
dayhi = 0;
|
dayhi = 0;
|
||||||
return tick_month();
|
return tickMonth();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(days == 29 && (dayhi == 3 || (dayhi == 2 && (daylo > 8 && daylo != 12)))) {
|
if(days == 29 && (dayhi == 3 || (dayhi == 2 && (daylo > 8 && daylo != 12)))) {
|
||||||
daylo = 1;
|
daylo = 1;
|
||||||
dayhi = 0;
|
dayhi = 0;
|
||||||
return tick_month();
|
return tickMonth();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(days == 30 && (dayhi == 3 || (dayhi == 2 && (daylo == 10 || daylo == 14)))) {
|
if(days == 30 && (dayhi == 3 || (dayhi == 2 && (daylo == 10 || daylo == 14)))) {
|
||||||
daylo = 1;
|
daylo = 1;
|
||||||
dayhi = 0;
|
dayhi = 0;
|
||||||
return tick_month();
|
return tickMonth();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(days == 31 && (dayhi == 3 && (daylo & 3))) {
|
if(days == 31 && (dayhi == 3 && (daylo & 3))) {
|
||||||
daylo = 1;
|
daylo = 1;
|
||||||
dayhi = 0;
|
dayhi = 0;
|
||||||
return tick_month();
|
return tickMonth();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(daylo <= 8 || daylo == 12) {
|
if(daylo <= 8 || daylo == 12) {
|
||||||
|
@ -153,7 +153,7 @@ auto EpsonRTC::tick_day() -> void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::tick_month() -> void {
|
auto EpsonRTC::tickMonth() -> void {
|
||||||
if(monthhi == 0 || !(monthlo & 2)) {
|
if(monthhi == 0 || !(monthlo & 2)) {
|
||||||
if(monthlo <= 8 || monthlo == 12) {
|
if(monthlo <= 8 || monthlo == 12) {
|
||||||
monthlo++;
|
monthlo++;
|
||||||
|
@ -164,11 +164,11 @@ auto EpsonRTC::tick_month() -> void {
|
||||||
} else {
|
} else {
|
||||||
monthlo = !(monthlo & 1);
|
monthlo = !(monthlo & 1);
|
||||||
monthhi = 0;
|
monthhi = 0;
|
||||||
tick_year();
|
tickYear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EpsonRTC::tick_year() -> void {
|
auto EpsonRTC::tickYear() -> void {
|
||||||
if(yearlo <= 8 || yearlo == 12) {
|
if(yearlo <= 8 || yearlo == 12) {
|
||||||
yearlo++;
|
yearlo++;
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -13,7 +13,7 @@ auto HitachiDSP::Enter() -> void {
|
||||||
auto HitachiDSP::main() -> void {
|
auto HitachiDSP::main() -> void {
|
||||||
if(mmio.dma) {
|
if(mmio.dma) {
|
||||||
for(auto n : range(mmio.dma_length)) {
|
for(auto n : range(mmio.dma_length)) {
|
||||||
bus_write(mmio.dma_target + n, bus_read(mmio.dma_source + n));
|
write(mmio.dma_target + n, read(mmio.dma_source + n));
|
||||||
step(2);
|
step(2);
|
||||||
}
|
}
|
||||||
mmio.dma = false;
|
mmio.dma = false;
|
||||||
|
|
|
@ -14,24 +14,24 @@ struct HitachiDSP : Processor::HG51B, Cothread {
|
||||||
auto reset() -> void;
|
auto reset() -> void;
|
||||||
|
|
||||||
//HG51B read/write
|
//HG51B read/write
|
||||||
auto bus_read(uint24 addr) -> uint8;
|
auto read(uint24 addr) -> uint8 override;
|
||||||
auto bus_write(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void override;
|
||||||
|
|
||||||
//CPU ROM read/write
|
//CPU ROM read/write
|
||||||
auto rom_read(uint24 addr, uint8 data) -> uint8;
|
auto romRead(uint24 addr, uint8 data) -> uint8;
|
||||||
auto rom_write(uint24 addr, uint8 data) -> void;
|
auto romWrite(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
//CPU RAM read/write
|
//CPU RAM read/write
|
||||||
auto ram_read(uint24 addr, uint8 data) -> uint8;
|
auto ramRead(uint24 addr, uint8 data) -> uint8;
|
||||||
auto ram_write(uint24 addr, uint8 data) -> void;
|
auto ramWrite(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
//HG51B data RAM read/write
|
//HG51B data RAM read/write
|
||||||
auto dram_read(uint24 addr, uint8 data) -> uint8;
|
auto dramRead(uint24 addr, uint8 data) -> uint8;
|
||||||
auto dram_write(uint24 addr, uint8 data) -> void;
|
auto dramWrite(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
//CPU MMIO read/write
|
//CPU MMIO read/write
|
||||||
auto dsp_read(uint24 addr, uint8 data) -> uint8;
|
auto dspRead(uint24 addr, uint8 data) -> uint8;
|
||||||
auto dsp_write(uint24 addr, uint8 data) -> void;
|
auto dspWrite(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
auto firmware() const -> vector<uint8>;
|
auto firmware() const -> vector<uint8>;
|
||||||
auto serialize(serializer&) -> void;
|
auto serialize(serializer&) -> void;
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
auto HitachiDSP::bus_read(uint24 addr) -> uint8 {
|
auto HitachiDSP::read(uint24 addr) -> uint8 {
|
||||||
if((addr & 0x40ec00) == 0x006c00) { //$00-3f,80-bf:6c00-6cff,7c00-7cff
|
if((addr & 0x40ec00) == 0x006c00) { //$00-3f,80-bf:6c00-6cff,7c00-7cff
|
||||||
return dsp_read(addr, 0x00);
|
return dspRead(addr, 0x00);
|
||||||
}
|
}
|
||||||
if((addr & 0x40e000) == 0x006000) { //$00-3f,80-bf:6000-6bff,7000-7bff
|
if((addr & 0x40e000) == 0x006000) { //$00-3f,80-bf:6000-6bff,7000-7bff
|
||||||
return dram_read(addr, 0x00);
|
return dramRead(addr, 0x00);
|
||||||
}
|
}
|
||||||
if((addr & 0x408000) == 0x008000) { //$00-3f,80-bf:8000-ffff
|
if((addr & 0x408000) == 0x008000) { //$00-3f,80-bf:8000-ffff
|
||||||
if(rom.size() == 0) return 0x00;
|
if(rom.size() == 0) return 0x00;
|
||||||
|
@ -20,12 +20,12 @@ auto HitachiDSP::bus_read(uint24 addr) -> uint8 {
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::bus_write(uint24 addr, uint8 data) -> void {
|
auto HitachiDSP::write(uint24 addr, uint8 data) -> void {
|
||||||
if((addr & 0x40ec00) == 0x006c00) { //$00-3f,80-bf:6c00-6fff,7c00-7fff
|
if((addr & 0x40ec00) == 0x006c00) { //$00-3f,80-bf:6c00-6fff,7c00-7fff
|
||||||
return dsp_write(addr, data);
|
return dspWrite(addr, data);
|
||||||
}
|
}
|
||||||
if((addr & 0x40e000) == 0x006000) { //$00-3f,80-bf:6000-6bff,7000-7bff
|
if((addr & 0x40e000) == 0x006000) { //$00-3f,80-bf:6000-6bff,7000-7bff
|
||||||
return dram_write(addr, data);
|
return dramWrite(addr, data);
|
||||||
}
|
}
|
||||||
if((addr & 0xf88000) == 0x700000) { //$70-77:0000-7fff
|
if((addr & 0xf88000) == 0x700000) { //$70-77:0000-7fff
|
||||||
if(ram.size() == 0) return;
|
if(ram.size() == 0) return;
|
||||||
|
@ -35,7 +35,7 @@ auto HitachiDSP::bus_write(uint24 addr, uint8 data) -> void {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::rom_read(uint24 addr, uint8 data) -> uint8 {
|
auto HitachiDSP::romRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
if(co_active() == hitachidsp.thread || regs.halt) {
|
if(co_active() == hitachidsp.thread || regs.halt) {
|
||||||
addr = Bus::mirror(addr, rom.size());
|
addr = Bus::mirror(addr, rom.size());
|
||||||
//if(Roms == 2 && mmio.r1f52 == 1 && addr >= (bit::round(rom.size()) >> 1)) return 0x00;
|
//if(Roms == 2 && mmio.r1f52 == 1 && addr >= (bit::round(rom.size()) >> 1)) return 0x00;
|
||||||
|
@ -45,32 +45,32 @@ auto HitachiDSP::rom_read(uint24 addr, uint8 data) -> uint8 {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::rom_write(uint24 addr, uint8 data) -> void {
|
auto HitachiDSP::romWrite(uint24 addr, uint8 data) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::ram_read(uint24 addr, uint8 data) -> uint8 {
|
auto HitachiDSP::ramRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
if(ram.size() == 0) return 0x00; //not open bus
|
if(ram.size() == 0) return 0x00; //not open bus
|
||||||
return ram.read(Bus::mirror(addr, ram.size()), data);
|
return ram.read(Bus::mirror(addr, ram.size()), data);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::ram_write(uint24 addr, uint8 data) -> void {
|
auto HitachiDSP::ramWrite(uint24 addr, uint8 data) -> void {
|
||||||
if(ram.size() == 0) return;
|
if(ram.size() == 0) return;
|
||||||
return ram.write(Bus::mirror(addr, ram.size()), data);
|
return ram.write(Bus::mirror(addr, ram.size()), data);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::dram_read(uint24 addr, uint8 data) -> uint8 {
|
auto HitachiDSP::dramRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
addr &= 0xfff;
|
addr &= 0xfff;
|
||||||
if(addr >= 0xc00) return data;
|
if(addr >= 0xc00) return data;
|
||||||
return dataRAM[addr];
|
return dataRAM[addr];
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::dram_write(uint24 addr, uint8 data) -> void {
|
auto HitachiDSP::dramWrite(uint24 addr, uint8 data) -> void {
|
||||||
addr &= 0xfff;
|
addr &= 0xfff;
|
||||||
if(addr >= 0xc00) return;
|
if(addr >= 0xc00) return;
|
||||||
dataRAM[addr] = data;
|
dataRAM[addr] = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::dsp_read(uint24 addr, uint8) -> uint8 {
|
auto HitachiDSP::dspRead(uint24 addr, uint8) -> uint8 {
|
||||||
addr = 0x7c00 | (addr & 0x03ff);
|
addr = 0x7c00 | (addr & 0x03ff);
|
||||||
|
|
||||||
//MMIO
|
//MMIO
|
||||||
|
@ -115,7 +115,7 @@ auto HitachiDSP::dsp_read(uint24 addr, uint8) -> uint8 {
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto HitachiDSP::dsp_write(uint24 addr, uint8 data) -> void {
|
auto HitachiDSP::dspWrite(uint24 addr, uint8 data) -> void {
|
||||||
addr = 0x7c00 | (addr & 0x03ff);
|
addr = 0x7c00 | (addr & 0x03ff);
|
||||||
|
|
||||||
//MMIO
|
//MMIO
|
||||||
|
|
|
@ -24,7 +24,7 @@ auto ICD2::main() -> void {
|
||||||
step(GameBoy::system._clocksExecuted);
|
step(GameBoy::system._clocksExecuted);
|
||||||
GameBoy::system._clocksExecuted = 0;
|
GameBoy::system._clocksExecuted = 0;
|
||||||
} else { //DMG halted
|
} else { //DMG halted
|
||||||
audio.coprocessorSample(0, 0);
|
stream->sample(0, 0);
|
||||||
step(1);
|
step(1);
|
||||||
}
|
}
|
||||||
synchronizeCPU();
|
synchronizeCPU();
|
||||||
|
@ -50,12 +50,11 @@ auto ICD2::unload() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ICD2::power() -> void {
|
auto ICD2::power() -> void {
|
||||||
audio.coprocessorEnable(true);
|
|
||||||
audio.coprocessorFrequency(2 * 1024 * 1024);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ICD2::reset() -> 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(4194304.0 / 2.0);
|
||||||
|
|
||||||
r6003 = 0x00;
|
r6003 = 0x00;
|
||||||
r6004 = 0xff;
|
r6004 = 0xff;
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#if defined(SFC_SUPERGAMEBOY)
|
#if defined(SFC_SUPERGAMEBOY)
|
||||||
|
|
||||||
struct ICD2 : Emulator::Interface::Bind, GameBoy::Interface::Hook, Cothread {
|
struct ICD2 : Emulator::Interface::Bind, GameBoy::Interface::Hook, Cothread {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
auto main() -> void;
|
auto main() -> void;
|
||||||
|
|
||||||
|
@ -8,7 +10,7 @@ struct ICD2 : Emulator::Interface::Bind, GameBoy::Interface::Hook, Cothread {
|
||||||
auto load() -> void;
|
auto load() -> void;
|
||||||
auto unload() -> void;
|
auto unload() -> void;
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto reset() -> void;
|
auto reset(bool soft = false) -> void;
|
||||||
|
|
||||||
auto read(uint24 addr, uint8 data) -> uint8;
|
auto read(uint24 addr, uint8 data) -> uint8;
|
||||||
auto write(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void;
|
||||||
|
|
|
@ -120,7 +120,7 @@ auto ICD2::videoRefresh(const uint32* data, uint pitch, uint width, uint height)
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ICD2::audioSample(int16 left, int16 right) -> void {
|
auto ICD2::audioSample(int16 left, int16 right) -> void {
|
||||||
audio.coprocessorSample(left, right);
|
stream->sample(left, right);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ICD2::inputPoll(uint port, uint device, uint id) -> int16 {
|
auto ICD2::inputPoll(uint port, uint device, uint id) -> int16 {
|
||||||
|
|
|
@ -54,7 +54,7 @@ auto ICD2::write(uint24 addr, uint8 data) -> void {
|
||||||
//d1,d0: 0 = frequency divider (clock rate adjust)
|
//d1,d0: 0 = frequency divider (clock rate adjust)
|
||||||
if(addr == 0x6003) {
|
if(addr == 0x6003) {
|
||||||
if((r6003 & 0x80) == 0x00 && (data & 0x80) == 0x80) {
|
if((r6003 & 0x80) == 0x00 && (data & 0x80) == 0x80) {
|
||||||
reset();
|
reset(true);
|
||||||
}
|
}
|
||||||
switch(data & 3) {
|
switch(data & 3) {
|
||||||
case 0: frequency = cpu.frequency / 4; break; //fast (glitchy, even on real hardware)
|
case 0: frequency = cpu.frequency / 4; break; //fast (glitchy, even on real hardware)
|
||||||
|
|
|
@ -26,7 +26,7 @@ auto MCC::reset() -> void {
|
||||||
commit();
|
commit();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MCC::memory_access(bool write, Memory& memory, uint24 addr, uint8 data) -> uint8 {
|
auto MCC::memoryAccess(bool write, Memory& memory, uint24 addr, uint8 data) -> uint8 {
|
||||||
addr = bus.mirror(addr, memory.size());
|
addr = bus.mirror(addr, memory.size());
|
||||||
if(!write) {
|
if(!write) {
|
||||||
return memory.read(addr, data);
|
return memory.read(addr, data);
|
||||||
|
@ -37,7 +37,7 @@ auto MCC::memory_access(bool write, Memory& memory, uint24 addr, uint8 data) ->
|
||||||
|
|
||||||
//map address=00-3f,80-bf:8000-ffff mask=0x408000
|
//map address=00-3f,80-bf:8000-ffff mask=0x408000
|
||||||
//map address=40-7d,c0-ff:0000-ffff
|
//map address=40-7d,c0-ff:0000-ffff
|
||||||
auto MCC::mcu_access(bool write, uint24 addr, uint8 data) -> uint8 {
|
auto MCC::mcuAccess(bool write, uint24 addr, uint8 data) -> uint8 {
|
||||||
if(addr < 0x400000) {
|
if(addr < 0x400000) {
|
||||||
//note: manifest maps 00-3f,80-bf:8000-ffff mask=0x408000 => 00-3f:0000-ffff
|
//note: manifest maps 00-3f,80-bf:8000-ffff mask=0x408000 => 00-3f:0000-ffff
|
||||||
//the intention is consistency in pre-decoding as much as possible
|
//the intention is consistency in pre-decoding as much as possible
|
||||||
|
@ -50,31 +50,31 @@ auto MCC::mcu_access(bool write, uint24 addr, uint8 data) -> uint8 {
|
||||||
if((addr & 0xe08000) == 0x008000) { //$00-1f:8000-ffff
|
if((addr & 0xe08000) == 0x008000) { //$00-1f:8000-ffff
|
||||||
if(r07 == 1) {
|
if(r07 == 1) {
|
||||||
addr = ((addr & 0x1f0000) >> 1) | (addr & 0x7fff);
|
addr = ((addr & 0x1f0000) >> 1) | (addr & 0x7fff);
|
||||||
return memory_access(write, rom, addr, data);
|
return memoryAccess(write, rom, addr, data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if((addr & 0xe08000) == 0x808000) { //$80-9f:8000-ffff
|
if((addr & 0xe08000) == 0x808000) { //$80-9f:8000-ffff
|
||||||
if(r08 == 1) {
|
if(r08 == 1) {
|
||||||
addr = ((addr & 0x1f0000) >> 1) | (addr & 0x7fff);
|
addr = ((addr & 0x1f0000) >> 1) | (addr & 0x7fff);
|
||||||
return memory_access(write, rom, addr, data);
|
return memoryAccess(write, rom, addr, data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if((addr & 0xf00000) == 0x400000) { //$40-4f:0000-ffff
|
if((addr & 0xf00000) == 0x400000) { //$40-4f:0000-ffff
|
||||||
if(r05 == 0) return memory_access(write, ram, addr & 0x0fffff, data);
|
if(r05 == 0) return memoryAccess(write, ram, addr & 0x0fffff, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if((addr & 0xf00000) == 0x500000) { //$50-5f:0000-ffff
|
if((addr & 0xf00000) == 0x500000) { //$50-5f:0000-ffff
|
||||||
if(r06 == 0) return memory_access(write, ram, addr & 0x0fffff, data);
|
if(r06 == 0) return memoryAccess(write, ram, addr & 0x0fffff, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if((addr & 0xf00000) == 0x600000) { //$60-6f:0000-ffff
|
if((addr & 0xf00000) == 0x600000) { //$60-6f:0000-ffff
|
||||||
if(r03 == 1) return memory_access(write, ram, addr & 0x0fffff, data);
|
if(r03 == 1) return memoryAccess(write, ram, addr & 0x0fffff, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if((addr & 0xf80000) == 0x700000) { //$70-77:0000-ffff
|
if((addr & 0xf80000) == 0x700000) { //$70-77:0000-ffff
|
||||||
return memory_access(write, ram, addr & 0x07ffff, data);
|
return memoryAccess(write, ram, addr & 0x07ffff, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(((addr & 0x408000) == 0x008000) //$00-3f,80-bf:8000-ffff
|
if(((addr & 0x408000) == 0x008000) //$00-3f,80-bf:8000-ffff
|
||||||
|
@ -82,18 +82,18 @@ auto MCC::mcu_access(bool write, uint24 addr, uint8 data) -> uint8 {
|
||||||
) {
|
) {
|
||||||
if(r02 == 0) addr = ((addr & 0x7f0000) >> 1) | (addr & 0x7fff);
|
if(r02 == 0) addr = ((addr & 0x7f0000) >> 1) | (addr & 0x7fff);
|
||||||
Memory& memory = (r01 == 0 ? (Memory&)bsmemory : (Memory&)ram);
|
Memory& memory = (r01 == 0 ? (Memory&)bsmemory : (Memory&)ram);
|
||||||
return memory_access(write, memory, addr & 0x7fffff, data);
|
return memoryAccess(write, memory, addr & 0x7fffff, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MCC::mcu_read(uint24 addr, uint8 data) -> uint8 {
|
auto MCC::mcuRead(uint24 addr, uint8 data) -> uint8 {
|
||||||
return mcu_access(false, addr, data);
|
return mcuAccess(false, addr, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MCC::mcu_write(uint24 addr, uint8 data) -> void {
|
auto MCC::mcuWrite(uint24 addr, uint8 data) -> void {
|
||||||
mcu_access(true, addr, data);
|
mcuAccess(true, addr, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MCC::read(uint24 addr, uint8 data) -> uint8 {
|
auto MCC::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
|
|
|
@ -10,11 +10,11 @@ struct MCC {
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto reset() -> void;
|
auto reset() -> void;
|
||||||
|
|
||||||
auto memory_access(bool write, Memory& memory, uint24 addr, uint8 data) -> uint8;
|
auto memoryAccess(bool write, Memory& memory, uint24 addr, uint8 data) -> uint8;
|
||||||
auto mcu_access(bool write, uint24 addr, uint8 data) -> uint8;
|
auto mcuAccess(bool write, uint24 addr, uint8 data) -> uint8;
|
||||||
|
|
||||||
auto mcu_read(uint24 addr, uint8 data) -> uint8;
|
auto mcuRead(uint24 addr, uint8 data) -> uint8;
|
||||||
auto mcu_write(uint24 addr, uint8 data) -> void;
|
auto mcuWrite(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
auto read(uint24 addr, uint8 data) -> uint8;
|
auto read(uint24 addr, uint8 data) -> uint8;
|
||||||
auto write(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void;
|
||||||
|
|
|
@ -38,7 +38,7 @@ auto MSU1::main() -> void {
|
||||||
right = sclamp<16>(rchannel);
|
right = sclamp<16>(rchannel);
|
||||||
if(dsp.mute()) left = 0, right = 0;
|
if(dsp.mute()) left = 0, right = 0;
|
||||||
|
|
||||||
audio.coprocessorSample(left, right);
|
stream->sample(left, right);
|
||||||
step(1);
|
step(1);
|
||||||
synchronizeCPU();
|
synchronizeCPU();
|
||||||
}
|
}
|
||||||
|
@ -55,12 +55,11 @@ auto MSU1::unload() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MSU1::power() -> void {
|
auto MSU1::power() -> void {
|
||||||
audio.coprocessorEnable(true);
|
|
||||||
audio.coprocessorFrequency(44100.0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MSU1::reset() -> void {
|
auto MSU1::reset() -> void {
|
||||||
create(MSU1::Enter, 44100);
|
create(MSU1::Enter, 44100);
|
||||||
|
stream = Emulator::audio.createStream(44100.0);
|
||||||
|
|
||||||
mmio.dataSeekOffset = 0;
|
mmio.dataSeekOffset = 0;
|
||||||
mmio.dataReadOffset = 0;
|
mmio.dataReadOffset = 0;
|
||||||
|
@ -119,7 +118,7 @@ auto MSU1::audioOpen() -> void {
|
||||||
mmio.audioError = true;
|
mmio.audioError = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MSU1::mmioRead(uint24 addr, uint8) -> uint8 {
|
auto MSU1::read(uint24 addr, uint8) -> uint8 {
|
||||||
cpu.synchronizeCoprocessors();
|
cpu.synchronizeCoprocessors();
|
||||||
addr = 0x2000 | (addr & 7);
|
addr = 0x2000 | (addr & 7);
|
||||||
|
|
||||||
|
@ -147,7 +146,7 @@ auto MSU1::mmioRead(uint24 addr, uint8) -> uint8 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto MSU1::mmioWrite(uint24 addr, uint8 data) -> void {
|
auto MSU1::write(uint24 addr, uint8 data) -> void {
|
||||||
cpu.synchronizeCoprocessors();
|
cpu.synchronizeCoprocessors();
|
||||||
addr = 0x2000 | (addr & 7);
|
addr = 0x2000 | (addr & 7);
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
struct MSU1 : Cothread {
|
struct MSU1 : Cothread {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
static auto Enter() -> void;
|
static auto Enter() -> void;
|
||||||
auto main() -> void;
|
auto main() -> void;
|
||||||
auto init() -> void;
|
auto init() -> void;
|
||||||
|
@ -10,8 +12,8 @@ struct MSU1 : Cothread {
|
||||||
auto dataOpen() -> void;
|
auto dataOpen() -> void;
|
||||||
auto audioOpen() -> void;
|
auto audioOpen() -> void;
|
||||||
|
|
||||||
auto mmioRead(uint24 addr, uint8 data) -> uint8;
|
auto read(uint24 addr, uint8 data) -> uint8;
|
||||||
auto mmioWrite(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
auto serialize(serializer&) -> void;
|
auto serialize(serializer&) -> void;
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,7 @@ auto NSS::power() -> void {
|
||||||
auto NSS::reset() -> void {
|
auto NSS::reset() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto NSS::set_dip(uint16 dip) -> void {
|
auto NSS::setDip(uint16 dip) -> void {
|
||||||
this->dip = dip;
|
this->dip = dip;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ struct NSS {
|
||||||
auto power() -> void;
|
auto power() -> void;
|
||||||
auto reset() -> void;
|
auto reset() -> void;
|
||||||
|
|
||||||
auto set_dip(uint16 dip) -> void;
|
auto setDip(uint16 dip) -> void;
|
||||||
auto read(uint24 addr, uint8 data) -> uint8;
|
auto read(uint24 addr, uint8 data) -> uint8;
|
||||||
auto write(uint24 addr, uint8 data) -> void;
|
auto write(uint24 addr, uint8 data) -> void;
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
auto SharpRTC::rtc_read(uint4 addr) -> uint4 {
|
auto SharpRTC::rtcRead(uint4 addr) -> uint4 {
|
||||||
switch(addr) {
|
switch(addr) {
|
||||||
case 0: return second % 10;
|
case 0: return second % 10;
|
||||||
case 1: return second / 10;
|
case 1: return second / 10;
|
||||||
|
@ -17,7 +17,7 @@ auto SharpRTC::rtc_read(uint4 addr) -> uint4 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::rtc_write(uint4 addr, uint4 data) -> void {
|
auto SharpRTC::rtcWrite(uint4 addr, uint4 data) -> void {
|
||||||
switch(addr) {
|
switch(addr) {
|
||||||
case 0: second = second / 10 * 10 + data; break;
|
case 0: second = second / 10 * 10 + data; break;
|
||||||
case 1: second = data * 10 + second % 10; break;
|
case 1: second = data * 10 + second % 10; break;
|
||||||
|
@ -37,8 +37,8 @@ auto SharpRTC::rtc_write(uint4 addr, uint4 data) -> void {
|
||||||
|
|
||||||
auto SharpRTC::load(const uint8* data) -> void {
|
auto SharpRTC::load(const uint8* data) -> void {
|
||||||
for(auto byte : range(8)) {
|
for(auto byte : range(8)) {
|
||||||
rtc_write(byte * 2 + 0, data[byte] >> 0);
|
rtcWrite(byte * 2 + 0, data[byte] >> 0);
|
||||||
rtc_write(byte * 2 + 1, data[byte] >> 4);
|
rtcWrite(byte * 2 + 1, data[byte] >> 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64 timestamp = 0;
|
uint64 timestamp = 0;
|
||||||
|
@ -47,16 +47,16 @@ auto SharpRTC::load(const uint8* data) -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64 diff = (uint64)time(0) - timestamp;
|
uint64 diff = (uint64)time(0) - timestamp;
|
||||||
while(diff >= 60 * 60 * 24) { tick_day(); diff -= 60 * 60 * 24; }
|
while(diff >= 60 * 60 * 24) { tickDay(); diff -= 60 * 60 * 24; }
|
||||||
while(diff >= 60 * 60) { tick_hour(); diff -= 60 * 60; }
|
while(diff >= 60 * 60) { tickHour(); diff -= 60 * 60; }
|
||||||
while(diff >= 60) { tick_minute(); diff -= 60; }
|
while(diff >= 60) { tickMinute(); diff -= 60; }
|
||||||
while(diff--) tick_second();
|
while(diff--) tickSecond();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::save(uint8* data) -> void {
|
auto SharpRTC::save(uint8* data) -> void {
|
||||||
for(auto byte : range(8)) {
|
for(auto byte : range(8)) {
|
||||||
data[byte] = rtc_read(byte * 2 + 0) << 0;
|
data[byte] = rtcRead(byte * 2 + 0) << 0;
|
||||||
data[byte] |= rtc_read(byte * 2 + 1) << 4;
|
data[byte] |= rtcRead(byte * 2 + 1) << 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64 timestamp = (uint64)time(nullptr);
|
uint64 timestamp = (uint64)time(nullptr);
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
auto SharpRTC::serialize(serializer& s) -> void {
|
auto SharpRTC::serialize(serializer& s) -> void {
|
||||||
Thread::serialize(s);
|
Thread::serialize(s);
|
||||||
|
|
||||||
s.integer((uint&)rtc_state);
|
s.integer((uint&)state);
|
||||||
s.integer(rtc_index);
|
s.integer(index);
|
||||||
|
|
||||||
s.integer(second);
|
s.integer(second);
|
||||||
s.integer(minute);
|
s.integer(minute);
|
||||||
|
|
|
@ -12,7 +12,7 @@ auto SharpRTC::Enter() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::main() -> void {
|
auto SharpRTC::main() -> void {
|
||||||
tick_second();
|
tickSecond();
|
||||||
|
|
||||||
step(1);
|
step(1);
|
||||||
synchronizeCPU();
|
synchronizeCPU();
|
||||||
|
@ -42,8 +42,8 @@ auto SharpRTC::power() -> void {
|
||||||
auto SharpRTC::reset() -> void {
|
auto SharpRTC::reset() -> void {
|
||||||
create(SharpRTC::Enter, 1);
|
create(SharpRTC::Enter, 1);
|
||||||
|
|
||||||
rtc_state = State::Read;
|
state = State::Read;
|
||||||
rtc_index = -1;
|
index = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::sync() -> void {
|
auto SharpRTC::sync() -> void {
|
||||||
|
@ -63,16 +63,16 @@ auto SharpRTC::read(uint24 addr, uint8 data) -> uint8 {
|
||||||
addr &= 1;
|
addr &= 1;
|
||||||
|
|
||||||
if(addr == 0) {
|
if(addr == 0) {
|
||||||
if(rtc_state != State::Read) return 0;
|
if(state != State::Read) return 0;
|
||||||
|
|
||||||
if(rtc_index < 0) {
|
if(index < 0) {
|
||||||
rtc_index++;
|
index++;
|
||||||
return 15;
|
return 15;
|
||||||
} else if(rtc_index > 12) {
|
} else if(index > 12) {
|
||||||
rtc_index = -1;
|
index = -1;
|
||||||
return 15;
|
return 15;
|
||||||
} else {
|
} else {
|
||||||
return rtc_read(rtc_index++);
|
return rtcRead(index++);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,25 +84,25 @@ auto SharpRTC::write(uint24 addr, uint8 data) -> void {
|
||||||
|
|
||||||
if(addr == 1) {
|
if(addr == 1) {
|
||||||
if(data == 0x0d) {
|
if(data == 0x0d) {
|
||||||
rtc_state = State::Read;
|
state = State::Read;
|
||||||
rtc_index = -1;
|
index = -1;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(data == 0x0e) {
|
if(data == 0x0e) {
|
||||||
rtc_state = State::Command;
|
state = State::Command;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(data == 0x0f) return; //unknown behavior
|
if(data == 0x0f) return; //unknown behavior
|
||||||
|
|
||||||
if(rtc_state == State::Command) {
|
if(state == State::Command) {
|
||||||
if(data == 0) {
|
if(data == 0) {
|
||||||
rtc_state = State::Write;
|
state = State::Write;
|
||||||
rtc_index = 0;
|
index = 0;
|
||||||
} else if(data == 4) {
|
} else if(data == 4) {
|
||||||
rtc_state = State::Ready;
|
state = State::Ready;
|
||||||
rtc_index = -1;
|
index = -1;
|
||||||
//reset time
|
//reset time
|
||||||
second = 0;
|
second = 0;
|
||||||
minute = 0;
|
minute = 0;
|
||||||
|
@ -113,17 +113,17 @@ auto SharpRTC::write(uint24 addr, uint8 data) -> void {
|
||||||
weekday = 0;
|
weekday = 0;
|
||||||
} else {
|
} else {
|
||||||
//unknown behavior
|
//unknown behavior
|
||||||
rtc_state = State::Ready;
|
state = State::Ready;
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(rtc_state == State::Write) {
|
if(state == State::Write) {
|
||||||
if(rtc_index >= 0 && rtc_index < 12) {
|
if(index >= 0 && index < 12) {
|
||||||
rtc_write(rtc_index++, data);
|
rtcWrite(index++, data);
|
||||||
if(rtc_index == 12) {
|
if(index == 12) {
|
||||||
//day of week is automatically calculated and written
|
//day of week is automatically calculated and written
|
||||||
weekday = calculate_weekday(1000 + year, month, day);
|
weekday = calculateWeekday(1000 + year, month, day);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -14,8 +14,8 @@ struct SharpRTC : Cothread {
|
||||||
|
|
||||||
auto serialize(serializer&) -> void;
|
auto serialize(serializer&) -> void;
|
||||||
|
|
||||||
enum class State : uint { Ready, Command, Read, Write } rtc_state;
|
enum class State : uint { Ready, Command, Read, Write } state;
|
||||||
int rtc_index;
|
int index;
|
||||||
|
|
||||||
uint second;
|
uint second;
|
||||||
uint minute;
|
uint minute;
|
||||||
|
@ -26,22 +26,22 @@ struct SharpRTC : Cothread {
|
||||||
uint weekday;
|
uint weekday;
|
||||||
|
|
||||||
//memory.cpp
|
//memory.cpp
|
||||||
auto rtc_read(uint4 addr) -> uint4;
|
auto rtcRead(uint4 addr) -> uint4;
|
||||||
auto rtc_write(uint4 addr, uint4 data) -> void;
|
auto rtcWrite(uint4 addr, uint4 data) -> void;
|
||||||
|
|
||||||
auto load(const uint8* data) -> void;
|
auto load(const uint8* data) -> void;
|
||||||
auto save(uint8* data) -> void;
|
auto save(uint8* data) -> void;
|
||||||
|
|
||||||
//time.cpp
|
//time.cpp
|
||||||
static const uint daysinmonth[12];
|
static const uint daysInMonth[12];
|
||||||
auto tick_second() -> void;
|
auto tickSecond() -> void;
|
||||||
auto tick_minute() -> void;
|
auto tickMinute() -> void;
|
||||||
auto tick_hour() -> void;
|
auto tickHour() -> void;
|
||||||
auto tick_day() -> void;
|
auto tickDay() -> void;
|
||||||
auto tick_month() -> void;
|
auto tickMonth() -> void;
|
||||||
auto tick_year() -> void;
|
auto tickYear() -> void;
|
||||||
|
|
||||||
auto calculate_weekday(uint year, uint month, uint day) -> uint;
|
auto calculateWeekday(uint year, uint month, uint day) -> uint;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern SharpRTC sharprtc;
|
extern SharpRTC sharprtc;
|
||||||
|
|
|
@ -1,28 +1,28 @@
|
||||||
const uint SharpRTC::daysinmonth[12] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
|
const uint SharpRTC::daysInMonth[12] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
|
||||||
|
|
||||||
auto SharpRTC::tick_second() -> void {
|
auto SharpRTC::tickSecond() -> void {
|
||||||
if(++second < 60) return;
|
if(++second < 60) return;
|
||||||
second = 0;
|
second = 0;
|
||||||
tick_minute();
|
tickMinute();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::tick_minute() -> void {
|
auto SharpRTC::tickMinute() -> void {
|
||||||
if(++minute < 60) return;
|
if(++minute < 60) return;
|
||||||
minute = 0;
|
minute = 0;
|
||||||
tick_hour();
|
tickHour();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::tick_hour() -> void {
|
auto SharpRTC::tickHour() -> void {
|
||||||
if(++hour < 24) return;
|
if(++hour < 24) return;
|
||||||
hour = 0;
|
hour = 0;
|
||||||
tick_day();
|
tickDay();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::tick_day() -> void {
|
auto SharpRTC::tickDay() -> void {
|
||||||
uint days = daysinmonth[month % 12];
|
uint days = daysInMonth[(month - 1) % 12];
|
||||||
|
|
||||||
//add one day in February for leap years
|
//add one day in February for leap years
|
||||||
if(month == 1) {
|
if(month == 2) {
|
||||||
if(year % 400 == 0) days++;
|
if(year % 400 == 0) days++;
|
||||||
else if(year % 100 == 0);
|
else if(year % 100 == 0);
|
||||||
else if(year % 4 == 0) days++;
|
else if(year % 4 == 0) days++;
|
||||||
|
@ -30,16 +30,16 @@ auto SharpRTC::tick_day() -> void {
|
||||||
|
|
||||||
if(day++ < days) return;
|
if(day++ < days) return;
|
||||||
day = 1;
|
day = 1;
|
||||||
tick_month();
|
tickMonth();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::tick_month() -> void {
|
auto SharpRTC::tickMonth() -> void {
|
||||||
if(month++ < 12) return;
|
if(month++ < 12) return;
|
||||||
month = 1;
|
month = 1;
|
||||||
tick_year();
|
tickYear();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto SharpRTC::tick_year() -> void {
|
auto SharpRTC::tickYear() -> void {
|
||||||
year++;
|
year++;
|
||||||
year = (uint12)year;
|
year = (uint12)year;
|
||||||
}
|
}
|
||||||
|
@ -47,7 +47,7 @@ auto SharpRTC::tick_year() -> void {
|
||||||
//returns day of week for specified date
|
//returns day of week for specified date
|
||||||
//eg 0 = Sunday, 1 = Monday, ... 6 = Saturday
|
//eg 0 = Sunday, 1 = Monday, ... 6 = Saturday
|
||||||
//usage: calculate_weekday(2008, 1, 1) returns weekday of January 1st, 2008
|
//usage: calculate_weekday(2008, 1, 1) returns weekday of January 1st, 2008
|
||||||
auto SharpRTC::calculate_weekday(uint year, uint month, uint day) -> uint {
|
auto SharpRTC::calculateWeekday(uint year, uint month, uint day) -> uint {
|
||||||
uint y = 1000, m = 1; //SharpRTC epoch is 1000-01-01
|
uint y = 1000, m = 1; //SharpRTC epoch is 1000-01-01
|
||||||
uint sum = 0; //number of days passed since epoch
|
uint sum = 0; //number of days passed since epoch
|
||||||
|
|
||||||
|
@ -66,7 +66,7 @@ auto SharpRTC::calculate_weekday(uint year, uint month, uint day) -> uint {
|
||||||
}
|
}
|
||||||
|
|
||||||
while(m < month) {
|
while(m < month) {
|
||||||
uint days = daysinmonth[m - 1];
|
uint days = daysInMonth[(m - 1) % 12];
|
||||||
bool leapyearmonth = false;
|
bool leapyearmonth = false;
|
||||||
if(days == 28) {
|
if(days == 28) {
|
||||||
if(y % 4 == 0) {
|
if(y % 4 == 0) {
|
||||||
|
|
|
@ -1,46 +0,0 @@
|
||||||
Audio audio;
|
|
||||||
|
|
||||||
auto Audio::coprocessorEnable(bool enable) -> void {
|
|
||||||
mixer.clear();
|
|
||||||
mixerEnable = enable;
|
|
||||||
dsp.read = dsp.write = 0;
|
|
||||||
mix.read = mix.write = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto Audio::coprocessorFrequency(double frequency) -> void {
|
|
||||||
mixer.setFrequency(frequency);
|
|
||||||
mixer.setResampler(nall::DSP::ResampleEngine::Sinc);
|
|
||||||
mixer.setResamplerFrequency(system.apuFrequency() / 768.0);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto Audio::sample(int16 left, int16 right) -> void {
|
|
||||||
if(!mixerEnable) return interface->audioSample(left, right);
|
|
||||||
|
|
||||||
dsp.left[dsp.write] = left;
|
|
||||||
dsp.right[dsp.write] = right;
|
|
||||||
dsp.write++;
|
|
||||||
flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto Audio::coprocessorSample(int16 left, int16 right) -> void {
|
|
||||||
int samples[] = {left, right};
|
|
||||||
mixer.sample(samples);
|
|
||||||
while(mixer.pending()) {
|
|
||||||
mixer.read(samples);
|
|
||||||
mix.left[mix.write] = samples[0];
|
|
||||||
mix.right[mix.write] = samples[1];
|
|
||||||
mix.write++;
|
|
||||||
flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
auto Audio::flush() -> void {
|
|
||||||
while(dsp.read != dsp.write && mix.read != mix.write) {
|
|
||||||
interface->audioSample(
|
|
||||||
sclamp<16>(dsp.left[dsp.read] + mix.left[mix.read]),
|
|
||||||
sclamp<16>(dsp.right[dsp.read] + mix.right[mix.read])
|
|
||||||
);
|
|
||||||
dsp.read++;
|
|
||||||
mix.read++;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,20 +0,0 @@
|
||||||
struct Audio {
|
|
||||||
auto sample(int16 left, int16 right) -> void;
|
|
||||||
|
|
||||||
auto coprocessorEnable(bool enable) -> void;
|
|
||||||
auto coprocessorFrequency(double frequency) -> void;
|
|
||||||
auto coprocessorSample(int16 left, int16 right) -> void;
|
|
||||||
auto flush() -> void;
|
|
||||||
|
|
||||||
private:
|
|
||||||
nall::DSP mixer;
|
|
||||||
bool mixerEnable = false;
|
|
||||||
struct Buffer {
|
|
||||||
int16 left[256];
|
|
||||||
int16 right[256];
|
|
||||||
uint8 read;
|
|
||||||
uint8 write;
|
|
||||||
} dsp, mix;
|
|
||||||
};
|
|
||||||
|
|
||||||
extern Audio audio;
|
|
|
@ -3,7 +3,6 @@
|
||||||
namespace SuperFamicom {
|
namespace SuperFamicom {
|
||||||
|
|
||||||
DSP dsp;
|
DSP dsp;
|
||||||
#include "audio.cpp"
|
|
||||||
|
|
||||||
#define REG(n) state.regs[n]
|
#define REG(n) state.regs[n]
|
||||||
#define VREG(n) state.regs[v.vidx + n]
|
#define VREG(n) state.regs[v.vidx + n]
|
||||||
|
@ -238,12 +237,11 @@ auto DSP::power() -> void {
|
||||||
voice[n].vbit = 1 << n;
|
voice[n].vbit = 1 << n;
|
||||||
voice[n].vidx = n * 0x10;
|
voice[n].vidx = n * 0x10;
|
||||||
}
|
}
|
||||||
|
|
||||||
audio.coprocessorEnable(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto DSP::reset() -> void {
|
auto DSP::reset() -> void {
|
||||||
create(Enter, system.apuFrequency());
|
create(Enter, system.apuFrequency());
|
||||||
|
stream = Emulator::audio.createStream(system.apuFrequency() / 768.0);
|
||||||
|
|
||||||
REG(FLG) = 0xe0;
|
REG(FLG) = 0xe0;
|
||||||
state.noise = 0x4000;
|
state.noise = 0x4000;
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
//Sony CXD1222Q-1
|
//Sony CXD1222Q-1
|
||||||
|
|
||||||
#include "audio.hpp"
|
|
||||||
|
|
||||||
struct DSP : Thread {
|
struct DSP : Thread {
|
||||||
|
shared_pointer<Emulator::Stream> stream;
|
||||||
|
|
||||||
DSP();
|
DSP();
|
||||||
|
|
||||||
alwaysinline auto step(uint clocks) -> void;
|
alwaysinline auto step(uint clocks) -> void;
|
||||||
|
|
|
@ -103,7 +103,7 @@ auto DSP::echo27() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
//output sample to DAC
|
//output sample to DAC
|
||||||
audio.sample(outl, outr);
|
stream->sample(outl, outr);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto DSP::echo28() -> void {
|
auto DSP::echo28() -> void {
|
||||||
|
|
|
@ -153,6 +153,36 @@ auto Interface::videoFrequency() -> double {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColors() -> uint32 {
|
||||||
|
return 1 << 19;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto Interface::videoColor(uint32 color) -> uint64 {
|
||||||
|
uint r = color.bits( 0, 4);
|
||||||
|
uint g = color.bits( 5, 9);
|
||||||
|
uint b = color.bits(10,14);
|
||||||
|
uint l = color.bits(15,18);
|
||||||
|
|
||||||
|
double L = (1.0 + l) / 16.0 * (l ? 1.0 : 0.5);
|
||||||
|
uint64 R = L * image::normalize(r, 5, 16);
|
||||||
|
uint64 G = L * image::normalize(g, 5, 16);
|
||||||
|
uint64 B = L * image::normalize(b, 5, 16);
|
||||||
|
|
||||||
|
if(settings.colorEmulation) {
|
||||||
|
static const uint8 gammaRamp[32] = {
|
||||||
|
0x00, 0x01, 0x03, 0x06, 0x0a, 0x0f, 0x15, 0x1c,
|
||||||
|
0x24, 0x2d, 0x37, 0x42, 0x4e, 0x5b, 0x69, 0x78,
|
||||||
|
0x88, 0x90, 0x98, 0xa0, 0xa8, 0xb0, 0xb8, 0xc0,
|
||||||
|
0xc8, 0xd0, 0xd8, 0xe0, 0xe8, 0xf0, 0xf8, 0xff,
|
||||||
|
};
|
||||||
|
R = L * gammaRamp[r] * 0x0101;
|
||||||
|
G = L * gammaRamp[g] * 0x0101;
|
||||||
|
B = L * gammaRamp[b] * 0x0101;
|
||||||
|
}
|
||||||
|
|
||||||
|
return R << 32 | G << 16 | B << 0;
|
||||||
|
}
|
||||||
|
|
||||||
auto Interface::audioFrequency() -> double {
|
auto Interface::audioFrequency() -> double {
|
||||||
return system.apuFrequency() / 768.0;
|
return system.apuFrequency() / 768.0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,6 +94,8 @@ struct Interface : Emulator::Interface {
|
||||||
auto manifest() -> string;
|
auto manifest() -> string;
|
||||||
auto title() -> string;
|
auto title() -> string;
|
||||||
auto videoFrequency() -> double;
|
auto videoFrequency() -> double;
|
||||||
|
auto videoColors() -> uint32;
|
||||||
|
auto videoColor(uint32 color) -> uint64;
|
||||||
auto audioFrequency() -> double;
|
auto audioFrequency() -> double;
|
||||||
|
|
||||||
auto loaded() -> bool;
|
auto loaded() -> bool;
|
||||||
|
|
|
@ -213,12 +213,6 @@ auto PPU::scanline() -> void {
|
||||||
screen.scanline();
|
screen.scanline();
|
||||||
|
|
||||||
if(vcounter() == 241) {
|
if(vcounter() == 241) {
|
||||||
auto output = this->output;
|
|
||||||
if(!overscan()) output -= 14 * 512;
|
|
||||||
auto pitch = 1024 >> interlace();
|
|
||||||
auto width = 512;
|
|
||||||
auto height = !interlace() ? 240 : 480;
|
|
||||||
Emulator::video.refresh(output, pitch * sizeof(uint32), width, height);
|
|
||||||
scheduler.exit(Scheduler::Event::Frame);
|
scheduler.exit(Scheduler::Event::Frame);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -230,4 +224,13 @@ auto PPU::frame() -> void {
|
||||||
display.overscan = regs.overscan;
|
display.overscan = regs.overscan;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto PPU::refresh() -> void {
|
||||||
|
auto output = this->output;
|
||||||
|
if(!overscan()) output -= 14 * 512;
|
||||||
|
auto pitch = 1024 >> interlace();
|
||||||
|
auto width = 512;
|
||||||
|
auto height = !interlace() ? 240 : 480;
|
||||||
|
Emulator::video.refresh(output, pitch * sizeof(uint32), width, height);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,7 @@ privileged:
|
||||||
|
|
||||||
auto scanline() -> void;
|
auto scanline() -> void;
|
||||||
auto frame() -> void;
|
auto frame() -> void;
|
||||||
|
auto refresh() -> void;
|
||||||
|
|
||||||
struct Registers {
|
struct Registers {
|
||||||
uint8 ppu1_mdr;
|
uint8 ppu1_mdr;
|
||||||
|
@ -152,7 +153,7 @@ privileged:
|
||||||
friend class PPU::Sprite;
|
friend class PPU::Sprite;
|
||||||
friend class PPU::Window;
|
friend class PPU::Window;
|
||||||
friend class PPU::Screen;
|
friend class PPU::Screen;
|
||||||
friend class Video;
|
friend class Scheduler;
|
||||||
|
|
||||||
struct Debugger {
|
struct Debugger {
|
||||||
hook<auto (uint16, uint8) -> void> vram_read;
|
hook<auto (uint16, uint8) -> void> vram_read;
|
||||||
|
|
|
@ -13,6 +13,7 @@ auto Scheduler::enter(Mode mode_) -> Event {
|
||||||
mode = mode_;
|
mode = mode_;
|
||||||
host = co_active();
|
host = co_active();
|
||||||
co_switch(resume);
|
co_switch(resume);
|
||||||
|
if(event == Event::Frame) ppu.refresh();
|
||||||
return event;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,7 @@ namespace SuperFamicom {
|
||||||
|
|
||||||
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
auto create(auto (*entrypoint)() -> void, uint frequency) -> void {
|
||||||
if(thread) co_delete(thread);
|
if(thread) co_delete(thread);
|
||||||
thread = co_create(262'144 * sizeof(void*), entrypoint);
|
thread = co_create(65'536 * sizeof(void*), entrypoint);
|
||||||
this->frequency = frequency;
|
this->frequency = frequency;
|
||||||
clock = 0;
|
clock = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -91,7 +91,6 @@ auto System::load() -> void {
|
||||||
if(cartridge.hasSufamiTurboSlots()) sufamiturboA.load(), sufamiturboB.load();
|
if(cartridge.hasSufamiTurboSlots()) sufamiturboA.load(), sufamiturboB.load();
|
||||||
|
|
||||||
serializeInit();
|
serializeInit();
|
||||||
configureVideo();
|
|
||||||
_loaded = true;
|
_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -152,6 +151,14 @@ auto System::power() -> void {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::reset() -> void {
|
auto System::reset() -> void {
|
||||||
|
Emulator::video.reset();
|
||||||
|
Emulator::video.setInterface(interface);
|
||||||
|
configureVideoPalette();
|
||||||
|
configureVideoEffects();
|
||||||
|
|
||||||
|
Emulator::audio.reset();
|
||||||
|
Emulator::audio.setInterface(interface);
|
||||||
|
|
||||||
cpu.reset();
|
cpu.reset();
|
||||||
smp.reset();
|
smp.reset();
|
||||||
dsp.reset();
|
dsp.reset();
|
||||||
|
|
|
@ -21,7 +21,6 @@ struct System {
|
||||||
auto reset() -> void;
|
auto reset() -> void;
|
||||||
|
|
||||||
//video.cpp
|
//video.cpp
|
||||||
auto configureVideo() -> void;
|
|
||||||
auto configureVideoPalette() -> void;
|
auto configureVideoPalette() -> void;
|
||||||
auto configureVideoEffects() -> void;
|
auto configureVideoEffects() -> void;
|
||||||
|
|
||||||
|
|
|
@ -1,36 +1,5 @@
|
||||||
auto System::configureVideo() -> void {
|
|
||||||
Emulator::video.reset();
|
|
||||||
Emulator::video.setInterface(interface);
|
|
||||||
configureVideoPalette();
|
|
||||||
configureVideoEffects();
|
|
||||||
}
|
|
||||||
|
|
||||||
auto System::configureVideoPalette() -> void {
|
auto System::configureVideoPalette() -> void {
|
||||||
Emulator::video.setPalette(1 << 19, [&](uint32 color) -> uint64 {
|
Emulator::video.setPalette();
|
||||||
uint r = color.bits( 0, 4);
|
|
||||||
uint g = color.bits( 5, 9);
|
|
||||||
uint b = color.bits(10,14);
|
|
||||||
uint l = color.bits(15,18);
|
|
||||||
|
|
||||||
double L = (1.0 + l) / 16.0 * (l ? 1.0 : 0.5);
|
|
||||||
uint64 R = L * image::normalize(r, 5, 16);
|
|
||||||
uint64 G = L * image::normalize(g, 5, 16);
|
|
||||||
uint64 B = L * image::normalize(b, 5, 16);
|
|
||||||
|
|
||||||
if(settings.colorEmulation) {
|
|
||||||
static const uint8 gammaRamp[32] = {
|
|
||||||
0x00, 0x01, 0x03, 0x06, 0x0a, 0x0f, 0x15, 0x1c,
|
|
||||||
0x24, 0x2d, 0x37, 0x42, 0x4e, 0x5b, 0x69, 0x78,
|
|
||||||
0x88, 0x90, 0x98, 0xa0, 0xa8, 0xb0, 0xb8, 0xc0,
|
|
||||||
0xc8, 0xd0, 0xd8, 0xe0, 0xe8, 0xf0, 0xf8, 0xff,
|
|
||||||
};
|
|
||||||
R = L * gammaRamp[r] * 0x0101;
|
|
||||||
G = L * gammaRamp[g] * 0x0101;
|
|
||||||
B = L * gammaRamp[b] * 0x0101;
|
|
||||||
}
|
|
||||||
|
|
||||||
return R << 32 | G << 16 | B << 0;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto System::configureVideoEffects() -> void {
|
auto System::configureVideoEffects() -> void {
|
||||||
|
|
|
@ -23,6 +23,10 @@ Settings::Settings() {
|
||||||
set("Video/ColorEmulation", true);
|
set("Video/ColorEmulation", true);
|
||||||
set("Video/ScanlineEmulation", false);
|
set("Video/ScanlineEmulation", false);
|
||||||
|
|
||||||
|
set("Video/Saturation", 100);
|
||||||
|
set("Video/Gamma", 100);
|
||||||
|
set("Video/Luminance", 100);
|
||||||
|
|
||||||
set("Video/Overscan/Mask", false);
|
set("Video/Overscan/Mask", false);
|
||||||
set("Video/Overscan/Horizontal", 8);
|
set("Video/Overscan/Horizontal", 8);
|
||||||
set("Video/Overscan/Vertical", 8);
|
set("Video/Overscan/Vertical", 8);
|
||||||
|
@ -33,6 +37,9 @@ Settings::Settings() {
|
||||||
set("Audio/Synchronize", true);
|
set("Audio/Synchronize", true);
|
||||||
set("Audio/Mute", false);
|
set("Audio/Mute", false);
|
||||||
set("Audio/Volume", 100);
|
set("Audio/Volume", 100);
|
||||||
|
set("Audio/Balance", 50);
|
||||||
|
set("Audio/Reverb/Delay", 0);
|
||||||
|
set("Audio/Reverb/Level", 0);
|
||||||
set("Audio/Latency", 60);
|
set("Audio/Latency", 60);
|
||||||
set("Audio/Resampler", "Sinc");
|
set("Audio/Resampler", "Sinc");
|
||||||
|
|
||||||
|
|
|
@ -99,7 +99,7 @@ Presentation::Presentation() {
|
||||||
});
|
});
|
||||||
muteAudio.setText("Mute Audio").setChecked(settings["Audio/Mute"].boolean()).onToggle([&] {
|
muteAudio.setText("Mute Audio").setChecked(settings["Audio/Mute"].boolean()).onToggle([&] {
|
||||||
settings["Audio/Mute"].setValue(muteAudio.checked());
|
settings["Audio/Mute"].setValue(muteAudio.checked());
|
||||||
program->updateAudioVolume();
|
program->updateAudioEffects();
|
||||||
});
|
});
|
||||||
showStatusBar.setText("Show Status Bar").setChecked(settings["UserInterface/ShowStatusBar"].boolean()).onToggle([&] {
|
showStatusBar.setText("Show Status Bar").setChecked(settings["UserInterface/ShowStatusBar"].boolean()).onToggle([&] {
|
||||||
settings["UserInterface/ShowStatusBar"].setValue(showStatusBar.checked());
|
settings["UserInterface/ShowStatusBar"].setValue(showStatusBar.checked());
|
||||||
|
|
|
@ -90,12 +90,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 lsample, int16 rsample) -> void {
|
||||||
int samples[] = {lsample, rsample};
|
audio->sample(lsample, rsample);
|
||||||
dsp.sample(samples);
|
//int samples[] = {lsample, rsample};
|
||||||
while(dsp.pending()) {
|
//dsp.sample(samples);
|
||||||
dsp.read(samples);
|
//while(dsp.pending()) {
|
||||||
audio->sample(samples[0], samples[1]);
|
// 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 {
|
||||||
|
|
|
@ -21,10 +21,13 @@ auto Program::loadMedia(Emulator::Interface& interface, Emulator::Interface::Med
|
||||||
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
|
||||||
|
Emulator::audio.reset();
|
||||||
|
Emulator::audio.setFrequency(audio->get(Audio::Frequency).get<uint>());
|
||||||
emulator = &interface;
|
emulator = &interface;
|
||||||
connectDevices();
|
connectDevices();
|
||||||
emulator->load(media.id);
|
emulator->load(media.id);
|
||||||
updateAudio();
|
updateAudioDriver();
|
||||||
|
updateAudioEffects();
|
||||||
emulator->power();
|
emulator->power();
|
||||||
|
|
||||||
presentation->resizeViewport();
|
presentation->resizeViewport();
|
||||||
|
|
|
@ -41,12 +41,6 @@ Program::Program(lstring args) {
|
||||||
input->onChange({&InputManager::onChange, &inputManager()});
|
input->onChange({&InputManager::onChange, &inputManager()});
|
||||||
if(!input->init()) input = Input::create("None");
|
if(!input->init()) input = Input::create("None");
|
||||||
|
|
||||||
dsp.setPrecision(16);
|
|
||||||
dsp.setBalance(0.0);
|
|
||||||
dsp.setFrequency(44100);
|
|
||||||
dsp.setResampler(DSP::ResampleEngine::Sinc);
|
|
||||||
dsp.setResamplerFrequency(audio->get(Audio::Frequency).get<uint>());
|
|
||||||
|
|
||||||
presentation->drawSplashScreen();
|
presentation->drawSplashScreen();
|
||||||
|
|
||||||
new InputManager;
|
new InputManager;
|
||||||
|
@ -55,7 +49,8 @@ Program::Program(lstring args) {
|
||||||
new ToolsManager;
|
new ToolsManager;
|
||||||
|
|
||||||
updateVideoShader();
|
updateVideoShader();
|
||||||
updateAudio();
|
updateAudioDriver();
|
||||||
|
updateAudioEffects();
|
||||||
|
|
||||||
args.takeFirst(); //ignore program location in argument parsing
|
args.takeFirst(); //ignore program location in argument parsing
|
||||||
for(auto& argument : args) {
|
for(auto& argument : args) {
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue