mirror of
https://github.com/sehugg/8bitworkshop.git
synced 2024-11-25 03:34:05 +00:00
more 7800 fixes; testing additional logging for profiler
This commit is contained in:
parent
4d5c6b9063
commit
fef73d9b54
@ -26,10 +26,10 @@ BACKGRND equ $20 ;Background Color
|
|||||||
P0C1 equ $21 ;Palette 0 - Color 1
|
P0C1 equ $21 ;Palette 0 - Color 1
|
||||||
P0C2 equ $22 ;Palette 0 - Color 2
|
P0C2 equ $22 ;Palette 0 - Color 2
|
||||||
P0C3 equ $23 ;Palette 0 - Color 3
|
P0C3 equ $23 ;Palette 0 - Color 3
|
||||||
WSYNC equ $20 ;Wait For Sync
|
WSYNC equ $24 ;Wait For Sync
|
||||||
P1C1 equ $21 ;Palette 1 - Color 1
|
P1C1 equ $25 ;Palette 1 - Color 1
|
||||||
P1C2 equ $22 ;Palette 1 - Color 2
|
P1C2 equ $26 ;Palette 1 - Color 2
|
||||||
P1C3 equ $23 ;Palette 1 - Color 3
|
P1C3 equ $27 ;Palette 1 - Color 3
|
||||||
MSTAT equ $28 ;Maria Status
|
MSTAT equ $28 ;Maria Status
|
||||||
P2C1 equ $29 ;Palette 2 - Color 1
|
P2C1 equ $29 ;Palette 2 - Color 1
|
||||||
P2C2 equ $2A ;Palette 2 - Color 2
|
P2C2 equ $2A ;Palette 2 - Color 2
|
||||||
@ -372,6 +372,19 @@ vbloop
|
|||||||
lda MSTAT ;Wait for VBLANK to end
|
lda MSTAT ;Wait for VBLANK to end
|
||||||
and #$80
|
and #$80
|
||||||
bne vbloop
|
bne vbloop
|
||||||
|
; test WSYNC
|
||||||
|
ldx #$04
|
||||||
|
sta WSYNC
|
||||||
|
stx BACKGRND
|
||||||
|
dex
|
||||||
|
sta WSYNC
|
||||||
|
stx BACKGRND
|
||||||
|
dex
|
||||||
|
sta WSYNC
|
||||||
|
stx BACKGRND
|
||||||
|
dex
|
||||||
|
sta WSYNC
|
||||||
|
stx BACKGRND
|
||||||
|
|
||||||
jmp mainloop ;Loop
|
jmp mainloop ;Loop
|
||||||
|
|
||||||
|
@ -48,7 +48,7 @@ export class DebugSymbols {
|
|||||||
constructor(symbolmap : SymbolMap) {
|
constructor(symbolmap : SymbolMap) {
|
||||||
this.symbolmap = symbolmap;
|
this.symbolmap = symbolmap;
|
||||||
this.addr2symbol = invertMap(symbolmap);
|
this.addr2symbol = invertMap(symbolmap);
|
||||||
// TODO: shouldn't be necc.
|
//// TODO: shouldn't be necc.
|
||||||
if (!this.addr2symbol[0x0]) this.addr2symbol[0x0] = '__START__'; // needed for ...
|
if (!this.addr2symbol[0x0]) this.addr2symbol[0x0] = '__START__'; // needed for ...
|
||||||
this.addr2symbol[0x10000] = '__END__'; // ... dump memory to work
|
this.addr2symbol[0x10000] = '__END__'; // ... dump memory to work
|
||||||
}
|
}
|
||||||
@ -101,8 +101,6 @@ export interface Platform {
|
|||||||
showHelp?(tool:string, ident?:string) : void;
|
showHelp?(tool:string, ident?:string) : void;
|
||||||
resize?() : void;
|
resize?() : void;
|
||||||
|
|
||||||
startProfiling?() : ProfilerOutput;
|
|
||||||
stopProfiling?() : void;
|
|
||||||
getRasterScanline?() : number;
|
getRasterScanline?() : number;
|
||||||
setBreakpoint?(id : string, cond : DebugCondition);
|
setBreakpoint?(id : string, cond : DebugCondition);
|
||||||
clearBreakpoint?(id : string);
|
clearBreakpoint?(id : string);
|
||||||
@ -166,12 +164,17 @@ export interface ProfilerOutput {
|
|||||||
export interface EmuProfiler {
|
export interface EmuProfiler {
|
||||||
start() : ProfilerOutput;
|
start() : ProfilerOutput;
|
||||||
stop();
|
stop();
|
||||||
|
// TODO?
|
||||||
|
logRead(a : number);
|
||||||
|
logWrite(a : number);
|
||||||
|
logInterrupt(a : number);
|
||||||
}
|
}
|
||||||
|
|
||||||
/////
|
/////
|
||||||
|
|
||||||
export abstract class BasePlatform {
|
export abstract class BasePlatform {
|
||||||
recorder : EmuRecorder = null;
|
recorder : EmuRecorder = null;
|
||||||
|
profiler : EmuProfiler = null;
|
||||||
debugSymbols : DebugSymbols;
|
debugSymbols : DebugSymbols;
|
||||||
|
|
||||||
abstract loadState(state : EmuState) : void;
|
abstract loadState(state : EmuState) : void;
|
||||||
|
@ -16,22 +16,21 @@ var Atari7800_PRESETS = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
const SWCHA = 0;
|
const SWCHA = 0;
|
||||||
const SWCHB = 1;
|
const SWCHB = 2;
|
||||||
const INPT4 = 2;
|
const INPT0 = 8;
|
||||||
const INPT5 = 3;
|
|
||||||
|
|
||||||
const Atari7800_KEYCODE_MAP = makeKeycodeMap([
|
const Atari7800_KEYCODE_MAP = makeKeycodeMap([
|
||||||
[Keys.A, INPT4, -0x80],
|
[Keys.A, INPT0+0, 0x80],
|
||||||
//[Keys.B, 0, 1],
|
[Keys.B, INPT0+1, 0x80],
|
||||||
//[Keys.SELECT, 0, 2],
|
[Keys.SELECT, SWCHB, -0x02],
|
||||||
//[Keys.START, 0, 3],
|
[Keys.START, SWCHB, -0x01],
|
||||||
[Keys.UP, SWCHA, -0x10],
|
[Keys.UP, SWCHA, -0x10],
|
||||||
[Keys.DOWN, SWCHA, -0x20],
|
[Keys.DOWN, SWCHA, -0x20],
|
||||||
[Keys.LEFT, SWCHA, -0x40],
|
[Keys.LEFT, SWCHA, -0x40],
|
||||||
[Keys.RIGHT, SWCHA, -0x80],
|
[Keys.RIGHT, SWCHA, -0x80],
|
||||||
|
|
||||||
[Keys.P2_A, INPT5, -0x80],
|
[Keys.P2_A, INPT0+2, 0x80],
|
||||||
//[Keys.P2_B, 1, 1],
|
[Keys.P2_B, INPT0+3, 0x80],
|
||||||
//[Keys.P2_SELECT, 1, 2],
|
//[Keys.P2_SELECT, 1, 2],
|
||||||
//[Keys.P2_START, 1, 3],
|
//[Keys.P2_START, 1, 3],
|
||||||
[Keys.P2_UP, SWCHA, -0x01],
|
[Keys.P2_UP, SWCHA, -0x01],
|
||||||
@ -46,7 +45,8 @@ const CLK = 3579545;
|
|||||||
const cpuFrequency = 1789772;
|
const cpuFrequency = 1789772;
|
||||||
const linesPerFrame = 262;
|
const linesPerFrame = 262;
|
||||||
const numVisibleLines = 258-16;
|
const numVisibleLines = 258-16;
|
||||||
const colorClocksPerLine = 454;
|
const colorClocksPerLine = 454; // 456?
|
||||||
|
const colorClocksPreDMA = 28;
|
||||||
const romLength = 0xc000;
|
const romLength = 0xc000;
|
||||||
|
|
||||||
// TIA chip
|
// TIA chip
|
||||||
@ -55,7 +55,7 @@ class TIA {
|
|||||||
regs = new Uint8Array(0x20);
|
regs = new Uint8Array(0x20);
|
||||||
|
|
||||||
reset() {
|
reset() {
|
||||||
// TODO
|
this.regs.fill(0);
|
||||||
}
|
}
|
||||||
read(a : number) : number {
|
read(a : number) : number {
|
||||||
return this.regs[a] | 0;
|
return this.regs[a] | 0;
|
||||||
@ -83,6 +83,7 @@ class TIA {
|
|||||||
|
|
||||||
class MARIA {
|
class MARIA {
|
||||||
bus;
|
bus;
|
||||||
|
profiler;
|
||||||
cycles : number = 0;
|
cycles : number = 0;
|
||||||
regs = new Uint8Array(0x20);
|
regs = new Uint8Array(0x20);
|
||||||
offset : number = -1;
|
offset : number = -1;
|
||||||
@ -92,15 +93,17 @@ class MARIA {
|
|||||||
h16 : boolean = false;
|
h16 : boolean = false;
|
||||||
h8 : boolean = false;
|
h8 : boolean = false;
|
||||||
pixels = new Uint8Array(320);
|
pixels = new Uint8Array(320);
|
||||||
|
WSYNC : number = 0;
|
||||||
|
|
||||||
reset() {
|
reset() {
|
||||||
// TODO
|
this.regs.fill(0);
|
||||||
}
|
}
|
||||||
read(a : number) : number {
|
read(a : number) : number {
|
||||||
return this.regs[a] | 0;
|
return this.regs[a] | 0;
|
||||||
}
|
}
|
||||||
write(a : number, v : number) {
|
write(a : number, v : number) {
|
||||||
this.regs[a] = v;
|
this.regs[a] = v;
|
||||||
|
if (a == 0x04) this.WSYNC++;
|
||||||
//console.log(hex(a), '=', hex(v));
|
//console.log(hex(a), '=', hex(v));
|
||||||
}
|
}
|
||||||
saveState() {
|
saveState() {
|
||||||
@ -143,15 +146,15 @@ class MARIA {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
readDLLEntry(bus) {
|
readDLLEntry(bus) {
|
||||||
|
this.profiler && this.profiler.logRead(this.dll);
|
||||||
var x = bus.read(this.dll);
|
var x = bus.read(this.dll);
|
||||||
this.offset = (x & 0xf);
|
this.offset = (x & 0xf);
|
||||||
this.dli = (x & 0x80) != 0;
|
|
||||||
this.h16 = (x & 0x40) != 0;
|
this.h16 = (x & 0x40) != 0;
|
||||||
this.h8 = (x & 0x20) != 0;
|
this.h8 = (x & 0x20) != 0;
|
||||||
this.dlstart = (bus.read(this.dll+1)<<8) + bus.read(this.dll+2);
|
this.dlstart = (bus.read(this.dll+1)<<8) + bus.read(this.dll+2);
|
||||||
//console.log(hex(this.dll,4), this.offset, hex(this.dlstart,4));
|
//console.log(hex(this.dll,4), this.offset, hex(this.dlstart,4));
|
||||||
this.dll = (this.dll + 3) & 0xffff; // TODO: can only cross 1 page
|
this.dll = (this.dll + 3) & 0xffff; // TODO: can also only cross 1 page?
|
||||||
this.cycles += 8;
|
this.dli = (bus.read(this.dll) & 0x80) != 0; // DLI flag is from next DLL entry
|
||||||
}
|
}
|
||||||
isHoley(a : number) : boolean {
|
isHoley(a : number) : boolean {
|
||||||
if (a & 0x8000) {
|
if (a & 0x8000) {
|
||||||
@ -165,11 +168,13 @@ class MARIA {
|
|||||||
return 0;
|
return 0;
|
||||||
else {
|
else {
|
||||||
this.cycles += 3;
|
this.cycles += 3;
|
||||||
|
//this.profiler && this.profiler.logRead(a);
|
||||||
return this.bus.read(a);
|
return this.bus.read(a);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
doDMA(bus) {
|
doDMA(platform : Atari7800Platform) {
|
||||||
this.bus = bus;
|
var bus = this.bus = platform.bus;
|
||||||
|
var profiler = this.profiler = platform.profiler;
|
||||||
this.cycles = 0;
|
this.cycles = 0;
|
||||||
this.pixels.fill(this.regs[0x0]);
|
this.pixels.fill(this.regs[0x0]);
|
||||||
if (this.isDMAEnabled()) {
|
if (this.isDMAEnabled()) {
|
||||||
@ -183,31 +188,33 @@ class MARIA {
|
|||||||
var dlofs = this.dlstart & 0xff;
|
var dlofs = this.dlstart & 0xff;
|
||||||
do {
|
do {
|
||||||
// read DL entry
|
// read DL entry
|
||||||
|
profiler && profiler.logRead(dlhi + ((dlofs+0) & 0x1ff));
|
||||||
var b0 = bus.read(dlhi + ((dlofs+0) & 0x1ff));
|
var b0 = bus.read(dlhi + ((dlofs+0) & 0x1ff));
|
||||||
var b1 = bus.read(dlhi + ((dlofs+1) & 0x1ff));
|
var b1 = bus.read(dlhi + ((dlofs+1) & 0x1ff));
|
||||||
if (b1 == 0) break; // end of DL
|
if (b1 == 0) break; // end of DL
|
||||||
var b2 = bus.read(dlhi + ((dlofs+2) & 0x1ff));
|
var b2 = bus.read(dlhi + ((dlofs+2) & 0x1ff));
|
||||||
var b3 = bus.read(dlhi + ((dlofs+3) & 0x1ff));
|
var b3 = bus.read(dlhi + ((dlofs+3) & 0x1ff));
|
||||||
var writemode = 0;
|
|
||||||
var indirect = false;
|
var indirect = false;
|
||||||
// extended header?
|
// extended header?
|
||||||
if ((b1 & 31) == 0) {
|
if ((b1 & 31) == 0) {
|
||||||
var pal = b3 >> 5;
|
var pal = b3 >> 5;
|
||||||
var width = 32 - (b3 & 31);
|
var width = 32 - (b3 & 31);
|
||||||
var xpos = bus.read(dlhi + ((dlofs+4) & 0x1ff));
|
var xpos = bus.read(dlhi + ((dlofs+4) & 0x1ff));
|
||||||
writemode = b1 & 0x80;
|
var writemode = b1 & 0x80;
|
||||||
indirect = (b1 & 0x20) != 0;
|
indirect = (b1 & 0x20) != 0;
|
||||||
dlofs += 5;
|
dlofs += 5;
|
||||||
|
this.cycles += 10;
|
||||||
} else {
|
} else {
|
||||||
// direct mode
|
// direct mode
|
||||||
var xpos = b3;
|
var xpos = b3;
|
||||||
var pal = b1 >> 5;
|
var pal = b1 >> 5;
|
||||||
var width = 32 - (b1 & 31);
|
var width = 32 - (b1 & 31);
|
||||||
|
var writemode = 0;
|
||||||
dlofs += 4;
|
dlofs += 4;
|
||||||
|
this.cycles += 8;
|
||||||
}
|
}
|
||||||
var gfxadr = b0 + (((b2 + (indirect?0:this.offset)) & 0xff) << 8);
|
var gfxadr = b0 + (((b2 + (indirect?0:this.offset)) & 0xff) << 8);
|
||||||
xpos *= 2;
|
xpos *= 2;
|
||||||
this.cycles += 4;
|
|
||||||
// copy graphics data (direct)
|
// copy graphics data (direct)
|
||||||
// TODO
|
// TODO
|
||||||
var readmode = (this.regs[0x1c] & 0x3) + (writemode?4:0);
|
var readmode = (this.regs[0x1c] & 0x3) + (writemode?4:0);
|
||||||
@ -229,27 +236,32 @@ class MARIA {
|
|||||||
xpos = (xpos + 2) & 0x1ff;
|
xpos = (xpos + 2) & 0x1ff;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 2: // 320 B/D
|
case 2: // 320 B/D (TODO?)
|
||||||
case 3: // 320 A/C
|
case 3: // 320 A/C
|
||||||
for (var j=0; j<8; j++) {
|
for (var j=0; j<8; j++) {
|
||||||
var col = data & 1;
|
var col = (data & 128) ? 1 : 0;
|
||||||
if (col > 0) {
|
if (col > 0) {
|
||||||
this.pixels[xpos] = this.regs[(pal<<2) + col];
|
this.pixels[xpos] = this.regs[(pal<<2) + col];
|
||||||
}
|
}
|
||||||
data >>= 1;
|
data <<= 1;
|
||||||
xpos = (xpos + 1) & 0x1ff;
|
xpos = (xpos + 1) & 0x1ff;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} while (1);
|
} while (this.cycles < colorClocksPerLine); // TODO?
|
||||||
// decrement offset
|
// decrement offset
|
||||||
this.offset -= 1;
|
this.offset -= 1;
|
||||||
}
|
}
|
||||||
return this.cycles;
|
return this.cycles;
|
||||||
}
|
}
|
||||||
doInterrupt() : boolean {
|
doInterrupt() : boolean {
|
||||||
return this.dli && this.offset == 0;
|
if (this.dli && this.offset < 0) {
|
||||||
|
this.dli = false;
|
||||||
|
return true;
|
||||||
|
} else
|
||||||
|
return false;
|
||||||
|
//return this.dli;// && this.offset == 1;
|
||||||
}
|
}
|
||||||
static stateToLongString(state) : string {
|
static stateToLongString(state) : string {
|
||||||
let s = "";
|
let s = "";
|
||||||
@ -275,7 +287,7 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
video;
|
video;
|
||||||
audio;
|
audio;
|
||||||
timer : AnimationTimer;
|
timer : AnimationTimer;
|
||||||
inputs = new Uint8Array(8);
|
inputs = new Uint8Array(16);
|
||||||
regs6532 = new Uint8Array(4);
|
regs6532 = new Uint8Array(4);
|
||||||
scanline : number = 0;
|
scanline : number = 0;
|
||||||
tia : TIA = new TIA();
|
tia : TIA = new TIA();
|
||||||
@ -284,24 +296,31 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
constructor(mainElement : HTMLElement) {
|
constructor(mainElement : HTMLElement) {
|
||||||
super();
|
super();
|
||||||
this.mainElement = mainElement;
|
this.mainElement = mainElement;
|
||||||
this.inputs.fill(0xff);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getPresets() {
|
getPresets() {
|
||||||
return Atari7800_PRESETS;
|
return Atari7800_PRESETS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
readInput(a:number) : number {
|
||||||
|
this.profiler && this.profiler.logRead(a+0x20);
|
||||||
|
switch (a) {
|
||||||
|
case 0xc: return ~this.inputs[0x8] & 0x80; //INPT4
|
||||||
|
case 0xd: return ~this.inputs[0x9] & 0x80; //INPT5
|
||||||
|
default: return this.inputs[a]|0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
start() {
|
start() {
|
||||||
this.cpu = new jt.M6502();
|
this.cpu = new jt.M6502();
|
||||||
this.ram = new Uint8Array(0x2800 - 0x1600);
|
this.ram = new Uint8Array(0x1000);
|
||||||
this.bios = new Uint8Array(0x1000); // TODO
|
//this.bios = new Uint8Array(0x1000); // TODO
|
||||||
// TODO: TIA access wastes a cycle
|
// TODO: TIA access wastes a cycle
|
||||||
this.bus = {
|
this.bus = {
|
||||||
read: newAddressDecoder([
|
read: newAddressDecoder([
|
||||||
[0x0008, 0x000b, 0x04, (a) => { return this.inputs[a]; }],
|
[0x0008, 0x000d, 0x0f, (a) => { return this.readInput(a); }],
|
||||||
[0x000c, 0x000d, 0x01, (a) => { return (this.inputs[INPT4 + a] & 0x80) ? 0x80 : 0x00; }],
|
|
||||||
[0x0000, 0x001f, 0x1f, (a) => { return this.tia.read(a); }],
|
[0x0000, 0x001f, 0x1f, (a) => { return this.tia.read(a); }],
|
||||||
[0x0020, 0x003f, 0x1f, (a) => { return this.maria.read(a); }],
|
[0x0020, 0x003f, 0x1f, (a) => { return this.maria.read(a); this.profiler && this.profiler.logRead(a+0x20); }],
|
||||||
[0x0040, 0x00ff, 0xff, (a) => { return this.ram[a + 0x800]; }],
|
[0x0040, 0x00ff, 0xff, (a) => { return this.ram[a + 0x800]; }],
|
||||||
[0x0100, 0x013f, 0xff, (a) => { return this.bus.read(a); }], // shadow
|
[0x0100, 0x013f, 0xff, (a) => { return this.bus.read(a); }], // shadow
|
||||||
[0x0140, 0x01ff, 0x1ff, (a) => { return this.ram[a + 0x800]; }],
|
[0x0140, 0x01ff, 0x1ff, (a) => { return this.ram[a + 0x800]; }],
|
||||||
@ -309,18 +328,19 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
[0x1800, 0x27ff, 0xffff, (a) => { return this.ram[a - 0x1800]; }],
|
[0x1800, 0x27ff, 0xffff, (a) => { return this.ram[a - 0x1800]; }],
|
||||||
[0x2800, 0x3fff, 0x7ff, (a) => { return this.bus.read(a | 0x2000); }], // shadow
|
[0x2800, 0x3fff, 0x7ff, (a) => { return this.bus.read(a | 0x2000); }], // shadow
|
||||||
[0x4000, 0xffff, 0xffff, (a) => { return this.rom ? this.rom[a - 0x4000] : 0; }],
|
[0x4000, 0xffff, 0xffff, (a) => { return this.rom ? this.rom[a - 0x4000] : 0; }],
|
||||||
[0x0000, 0xffff, 0xffff, (a) => { throw new EmuHalt("Read @ " + hex(a,4)); }]
|
[0x0000, 0xffff, 0xffff, (a) => { throw new EmuHalt("Read @ " + hex(a,4)); }],
|
||||||
]),
|
]),
|
||||||
write: newAddressDecoder([
|
write: newAddressDecoder([
|
||||||
[0x0015, 0x001A, 0x1f, (a,v) => { this.audio.pokey1.setTIARegister(a, v); }],
|
[0x0015, 0x001A, 0x1f, (a,v) => { this.audio.pokey1.setTIARegister(a, v); }],
|
||||||
[0x0000, 0x001f, 0x1f, (a,v) => { this.tia.write(a,v); }],
|
[0x0000, 0x001f, 0x1f, (a,v) => { this.tia.write(a,v); this.profiler && this.profiler.logWrite(a); }],
|
||||||
[0x0020, 0x003f, 0x1f, (a,v) => { this.maria.write(a,v); }],
|
[0x0020, 0x003f, 0x1f, (a,v) => { this.maria.write(a,v); this.profiler && this.profiler.logWrite(a+0x20); }],
|
||||||
[0x0040, 0x00ff, 0xff, (a,v) => { this.ram[a + 0x800] = v; }],
|
[0x0040, 0x00ff, 0xff, (a,v) => { this.ram[a + 0x800] = v; }],
|
||||||
[0x0140, 0x01ff, 0x1ff, (a,v) => { this.ram[a + 0x800] = v; }],
|
[0x0140, 0x01ff, 0x1ff, (a,v) => { this.ram[a + 0x800] = v; }],
|
||||||
[0x0280, 0x02ff, 0x3, (a,v) => { this.regs6532[a] = v; /*TODO*/ }],
|
[0x0280, 0x02ff, 0x3, (a,v) => { this.regs6532[a] = v; /*TODO*/ }],
|
||||||
[0x1800, 0x27ff, 0xffff, (a,v) => { this.ram[a - 0x1800] = v; }],
|
[0x1800, 0x27ff, 0xffff, (a,v) => { this.ram[a - 0x1800] = v; }],
|
||||||
[0x2800, 0x3fff, 0x7ff, (a,v) => { this.bus.write(a | 0x2000, v); }],
|
[0x2800, 0x3fff, 0x7ff, (a,v) => { this.bus.write(a | 0x2000, v); }],
|
||||||
[0x0000, 0xffff, 0xffff, (a,v) => { throw new EmuHalt("Write @ " + hex(a,4) + " " + hex(v,2)); }]
|
[0xbfff, 0xbfff, 0xffff, (a,v) => { }], // TODO: bank switching?
|
||||||
|
[0x0000, 0xffff, 0xffff, (a,v) => { throw new EmuHalt("Write @ " + hex(a,4) + " " + hex(v,2)); }],
|
||||||
]),
|
]),
|
||||||
};
|
};
|
||||||
this.cpu.connectBus(this.bus);
|
this.cpu.connectBus(this.bus);
|
||||||
@ -350,27 +370,24 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
var iofs = 0;
|
var iofs = 0;
|
||||||
var debugCond = this.getDebugCallback();
|
var debugCond = this.getDebugCallback();
|
||||||
var rgb;
|
var rgb;
|
||||||
var mariaClocks = 0;
|
var mariaClocks = colorClocksPreDMA; // 7 CPU cycles until DMA
|
||||||
// visible lines
|
// visible lines
|
||||||
for (var sl=0; sl<linesPerFrame; sl++) {
|
for (var sl=0; sl<linesPerFrame; sl++) {
|
||||||
this.scanline = sl;
|
this.scanline = sl;
|
||||||
this.maria.setVBLANK(sl >= numVisibleLines);
|
var visible = sl < numVisibleLines;
|
||||||
mariaClocks += 454;
|
this.maria.setVBLANK(!visible);
|
||||||
// TODO: 7 cycles at start of line
|
|
||||||
if (this.maria.isDMAEnabled()) {
|
|
||||||
// do DMA for scanline
|
|
||||||
mariaClocks -= this.maria.doDMA(this.bus);
|
|
||||||
// copy line to frame buffer
|
|
||||||
for (var i=0; i<320; i++) {
|
|
||||||
idata[iofs++] = COLORS_RGBA[this.maria.pixels[i]];
|
|
||||||
}
|
|
||||||
// do interrupt?
|
|
||||||
if (this.maria.doInterrupt()) {
|
|
||||||
mariaClocks -= this.cpu.setNMIAndWait() * 4;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// iterate CPU with free clocks
|
// iterate CPU with free clocks
|
||||||
while (mariaClocks > 0) {
|
while (mariaClocks > 0) {
|
||||||
|
// wait for WSYNC? (end of line)
|
||||||
|
if (this.maria.WSYNC) {
|
||||||
|
if (mariaClocks >= colorClocksPreDMA) {
|
||||||
|
this.maria.WSYNC--;
|
||||||
|
mariaClocks = colorClocksPreDMA; // 7 CPU cycles until DMA
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// next CPU clock
|
||||||
mariaClocks -= 4;
|
mariaClocks -= 4;
|
||||||
if (debugCond && debugCond()) {
|
if (debugCond && debugCond()) {
|
||||||
debugCond = null;
|
debugCond = null;
|
||||||
@ -379,6 +396,20 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
}
|
}
|
||||||
this.cpu.clockPulse();
|
this.cpu.clockPulse();
|
||||||
}
|
}
|
||||||
|
mariaClocks += colorClocksPerLine;
|
||||||
|
// do DMA for scanline?
|
||||||
|
if (this.maria.isDMAEnabled() && visible) {
|
||||||
|
mariaClocks -= this.maria.doDMA(this);
|
||||||
|
// copy line to frame buffer
|
||||||
|
for (var i=0; i<320; i++) {
|
||||||
|
idata[iofs++] = COLORS_RGBA[this.maria.pixels[i]];
|
||||||
|
}
|
||||||
|
// do interrupt?
|
||||||
|
if (this.maria.doInterrupt()) {
|
||||||
|
this.profiler && this.profiler.logInterrupt(0);
|
||||||
|
mariaClocks -= this.cpu.setNMIAndWait() * 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// update video frame
|
// update video frame
|
||||||
if (!novideo) {
|
if (!novideo) {
|
||||||
@ -390,15 +421,16 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
}
|
}
|
||||||
|
|
||||||
loadROM(title, data) {
|
loadROM(title, data) {
|
||||||
|
if (data.length == 0xc080) data = data.slice(0x80); // strip header
|
||||||
this.rom = padBytes(data, romLength, true);
|
this.rom = padBytes(data, romLength, true);
|
||||||
this.reset();
|
this.reset();
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
loadBIOS(title, data) {
|
loadBIOS(title, data) {
|
||||||
this.bios = padBytes(data, 0x1000);
|
this.bios = padBytes(data, 0x1000);
|
||||||
this.reset();
|
this.reset();
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
isRunning() {
|
isRunning() {
|
||||||
return this.timer.isRunning();
|
return this.timer.isRunning();
|
||||||
}
|
}
|
||||||
@ -417,14 +449,10 @@ class Atari7800Platform extends Base6502Platform implements Platform {
|
|||||||
this.cpu.reset();
|
this.cpu.reset();
|
||||||
this.tia.reset();
|
this.tia.reset();
|
||||||
this.maria.reset();
|
this.maria.reset();
|
||||||
/*
|
this.inputs.fill(0x0);
|
||||||
// execute until out of BIOS
|
this.inputs[SWCHA] = 0xff;
|
||||||
for (var i=0; i<20000; i++) {
|
this.inputs[SWCHB] = 1+2+8;
|
||||||
this.cpu.clockPulse();
|
this.cpu.clockPulse(); // TODO: needed for test to pass?
|
||||||
if (this.getCPUState().PC < 0xf000)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
readAddress(addr : number) {
|
readAddress(addr : number) {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
|
||||||
import { Platform, EmuState, EmuControlsState, EmuRecorder } from "./baseplatform";
|
import { Platform, BasePlatform, EmuState, EmuControlsState, EmuRecorder } from "./baseplatform";
|
||||||
import { BaseDebugPlatform, EmuProfiler, ProfilerOutput } from "./baseplatform";
|
import { BaseDebugPlatform, EmuProfiler, ProfilerOutput } from "./baseplatform";
|
||||||
import { getNoiseSeed, setNoiseSeed } from "./emu";
|
import { getNoiseSeed, setNoiseSeed } from "./emu";
|
||||||
|
|
||||||
@ -117,41 +117,63 @@ export class StateRecorderImpl implements EmuRecorder {
|
|||||||
|
|
||||||
// PROFILER
|
// PROFILER
|
||||||
|
|
||||||
|
const PROFOP_READ = 0x100000;
|
||||||
|
const PROFOP_WRITE = 0x200000;
|
||||||
|
const PROFOP_INTERRUPT = 0x400000;
|
||||||
|
|
||||||
export class EmuProfilerImpl implements EmuProfiler {
|
export class EmuProfilerImpl implements EmuProfiler {
|
||||||
|
|
||||||
platform : Platform;
|
platform : Platform;
|
||||||
|
frame = null;
|
||||||
|
output = {frame:null};
|
||||||
|
i = 0;
|
||||||
|
lastsl = 9999;
|
||||||
|
starti = 0;
|
||||||
|
|
||||||
constructor(platform : Platform) {
|
constructor(platform : Platform) {
|
||||||
this.platform = platform;
|
this.platform = platform;
|
||||||
}
|
}
|
||||||
|
|
||||||
start() : ProfilerOutput {
|
start() : ProfilerOutput {
|
||||||
var frame = null;
|
if (this.platform instanceof BasePlatform) this.platform.profiler = this;
|
||||||
var output = {frame:null};
|
|
||||||
var i = 0;
|
|
||||||
var lastsl = 9999;
|
|
||||||
var start = 0;
|
|
||||||
this.platform.setBreakpoint('profile', () => {
|
this.platform.setBreakpoint('profile', () => {
|
||||||
var sl = this.platform.getRasterScanline();
|
|
||||||
if (sl != lastsl) {
|
|
||||||
if (frame) {
|
|
||||||
frame.lines.push({start:start,end:i-1});
|
|
||||||
}
|
|
||||||
if (sl < lastsl) {
|
|
||||||
output.frame = frame;
|
|
||||||
frame = {iptab:new Uint32Array(0x8000), lines:[]}; // TODO: const
|
|
||||||
i = 0;
|
|
||||||
}
|
|
||||||
start = i;
|
|
||||||
lastsl = sl;
|
|
||||||
}
|
|
||||||
var c = this.platform.getCPUState();
|
var c = this.platform.getCPUState();
|
||||||
frame.iptab[i++] = c.EPC || c.PC;
|
this.log(c.EPC || c.PC);
|
||||||
return false; // profile forever
|
return false; // profile forever
|
||||||
});
|
});
|
||||||
return output;
|
this.output = {frame:null};
|
||||||
|
return this.output;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log(op : number) {
|
||||||
|
var sl = this.platform.getRasterScanline();
|
||||||
|
if (sl != this.lastsl) {
|
||||||
|
if (this.frame) {
|
||||||
|
this.frame.lines.push({start:this.starti, end:this.i-1});
|
||||||
|
}
|
||||||
|
if (sl < this.lastsl) {
|
||||||
|
this.output.frame = this.frame;
|
||||||
|
this.frame = {iptab:new Uint32Array(0x8000), lines:[]}; // TODO: const
|
||||||
|
this.i = 0;
|
||||||
|
}
|
||||||
|
this.starti = this.i;
|
||||||
|
this.lastsl = sl;
|
||||||
|
}
|
||||||
|
this.frame.iptab[this.i++] = op;
|
||||||
|
}
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
this.platform.clearBreakpoint('profile');
|
this.platform.clearBreakpoint('profile');
|
||||||
|
if (this.platform instanceof BasePlatform) this.platform.profiler = null;
|
||||||
|
}
|
||||||
|
// TODO?
|
||||||
|
logRead(a : number) {
|
||||||
|
this.log(a | PROFOP_READ);
|
||||||
|
}
|
||||||
|
logWrite(a : number) {
|
||||||
|
this.log(a | PROFOP_WRITE);
|
||||||
|
}
|
||||||
|
logInterrupt(a : number) {
|
||||||
|
this.log(a | PROFOP_INTERRUPT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -866,6 +866,12 @@ export class ProfileView implements ProjectView {
|
|||||||
for (let i=l.start; i<=l.end; i++) {
|
for (let i=l.start; i<=l.end; i++) {
|
||||||
let pc = f.iptab[i];
|
let pc = f.iptab[i];
|
||||||
let sym = this.symcache[pc];
|
let sym = this.symcache[pc];
|
||||||
|
let op = pc >> 20;
|
||||||
|
switch (op) { // TODO: const
|
||||||
|
case 1: sym = "r$" + hex(pc & 0xffff); break;
|
||||||
|
case 2: sym = "W$" + hex(pc & 0xffff); break;
|
||||||
|
case 4: sym = "I$" + hex(pc & 0xffff); break;
|
||||||
|
}
|
||||||
if (!sym) {
|
if (!sym) {
|
||||||
sym = lookupSymbol(platform, pc, false);
|
sym = lookupSymbol(platform, pc, false);
|
||||||
this.symcache[pc] = sym;
|
this.symcache[pc] = sym;
|
||||||
|
@ -42,6 +42,7 @@ var _williams = require('gen/platform/williams.js');
|
|||||||
var _sound_williams = require('gen/platform/sound_williams.js');
|
var _sound_williams = require('gen/platform/sound_williams.js');
|
||||||
var _astrocade = require('gen/platform/astrocade.js');
|
var _astrocade = require('gen/platform/astrocade.js');
|
||||||
var _atari8 = require('gen/platform/atari8.js');
|
var _atari8 = require('gen/platform/atari8.js');
|
||||||
|
var _atari7800 = require('gen/platform/atari7800.js');
|
||||||
var _coleco = require('gen/platform/coleco.js');
|
var _coleco = require('gen/platform/coleco.js');
|
||||||
var _sms = require('gen/platform/sms.js');
|
var _sms = require('gen/platform/sms.js');
|
||||||
|
|
||||||
@ -278,13 +279,11 @@ describe('Platform Replay', () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
/* TODO
|
it('Should run atari7800', () => {
|
||||||
it('Should run atari8-5200', () => {
|
var platform = testPlatform('atari7800', 'sprites.dasm.rom', 92, (platform, frameno) => {
|
||||||
var platform = testPlatform('atari8-5200', 'hello.a.rom', 92, (platform, frameno) => {
|
|
||||||
if (frameno == 62) {
|
if (frameno == 62) {
|
||||||
keycallback(Keys.VK_SPACE.c, Keys.VK_SPACE.c, 1);
|
keycallback(Keys.VK_DOWN.c, Keys.VK_DOWN.c, 1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
*/
|
|
||||||
});
|
});
|
||||||
|
BIN
test/roms/atari7800/sprites.dasm.rom
Normal file
BIN
test/roms/atari7800/sprites.dasm.rom
Normal file
Binary file not shown.
Loading…
Reference in New Issue
Block a user