mirror of
https://github.com/sehugg/8bitworkshop.git
synced 2025-01-11 08:30:02 +00:00
probe shows values; click gutter to run to cursor
This commit is contained in:
parent
0e91abe6bb
commit
313dc2f863
@ -4,6 +4,9 @@
|
|||||||
.gutter-offset {
|
.gutter-offset {
|
||||||
width: 3em;
|
width: 3em;
|
||||||
}
|
}
|
||||||
|
.CodeMirror-gutter-elt:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
.gutter-bytes {
|
.gutter-bytes {
|
||||||
width: 6em;
|
width: 6em;
|
||||||
}
|
}
|
||||||
|
@ -120,20 +120,20 @@ export class StateRecorderImpl implements EmuRecorder {
|
|||||||
import { Probeable, ProbeAll } from "./devices";
|
import { Probeable, ProbeAll } from "./devices";
|
||||||
|
|
||||||
export enum ProbeFlags {
|
export enum ProbeFlags {
|
||||||
CLOCKS = 0x00000000,
|
CLOCKS = 0x00000000,
|
||||||
EXECUTE = 0x01000000,
|
EXECUTE = 0x01000000,
|
||||||
MEM_READ = 0x02000000,
|
MEM_READ = 0x12000000,
|
||||||
MEM_WRITE = 0x03000000,
|
MEM_WRITE = 0x13000000,
|
||||||
IO_READ = 0x04000000,
|
IO_READ = 0x14000000,
|
||||||
IO_WRITE = 0x05000000,
|
IO_WRITE = 0x15000000,
|
||||||
VRAM_READ = 0x06000000,
|
VRAM_READ = 0x16000000,
|
||||||
VRAM_WRITE = 0x07000000,
|
VRAM_WRITE= 0x17000000,
|
||||||
INTERRUPT = 0x08000000,
|
INTERRUPT = 0x08000000,
|
||||||
ILLEGAL = 0x09000000,
|
ILLEGAL = 0x09000000,
|
||||||
SP_PUSH = 0x0a000000,
|
SP_PUSH = 0x0a000000,
|
||||||
SP_POP = 0x0b000000,
|
SP_POP = 0x0b000000,
|
||||||
SCANLINE = 0x7e000000,
|
SCANLINE = 0x7e000000,
|
||||||
FRAME = 0x7f000000,
|
FRAME = 0x7f000000,
|
||||||
}
|
}
|
||||||
|
|
||||||
class ProbeFrame {
|
class ProbeFrame {
|
||||||
@ -216,23 +216,26 @@ export class ProbeRecorder implements ProbeAll {
|
|||||||
logInterrupt(type:number) {
|
logInterrupt(type:number) {
|
||||||
this.log(type | ProbeFlags.INTERRUPT);
|
this.log(type | ProbeFlags.INTERRUPT);
|
||||||
}
|
}
|
||||||
|
logValue(address:number, value:number, op:number) {
|
||||||
|
this.log((address & 0xffff) | ((value & 0xff)<<16) | op);
|
||||||
|
}
|
||||||
logRead(address:number, value:number) {
|
logRead(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.MEM_READ);
|
this.logValue(address, value, ProbeFlags.MEM_READ);
|
||||||
}
|
}
|
||||||
logWrite(address:number, value:number) {
|
logWrite(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.MEM_WRITE);
|
this.logValue(address, value, ProbeFlags.MEM_WRITE);
|
||||||
}
|
}
|
||||||
logIORead(address:number, value:number) {
|
logIORead(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.IO_READ);
|
this.logValue(address, value, ProbeFlags.IO_READ);
|
||||||
}
|
}
|
||||||
logIOWrite(address:number, value:number) {
|
logIOWrite(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.IO_WRITE);
|
this.logValue(address, value, ProbeFlags.IO_WRITE);
|
||||||
}
|
}
|
||||||
logVRAMRead(address:number, value:number) {
|
logVRAMRead(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.VRAM_READ);
|
this.logValue(address, value, ProbeFlags.VRAM_READ);
|
||||||
}
|
}
|
||||||
logVRAMWrite(address:number, value:number) {
|
logVRAMWrite(address:number, value:number) {
|
||||||
this.log(address | ProbeFlags.VRAM_WRITE);
|
this.logValue(address, value, ProbeFlags.VRAM_WRITE);
|
||||||
}
|
}
|
||||||
logIllegal(address:number) {
|
logIllegal(address:number) {
|
||||||
this.log(address | ProbeFlags.ILLEGAL);
|
this.log(address | ProbeFlags.ILLEGAL);
|
||||||
|
@ -1211,22 +1211,23 @@ function getEditorPC() : number {
|
|||||||
return wnd && wnd.getCursorPC && wnd.getCursorPC();
|
return wnd && wnd.getCursorPC && wnd.getCursorPC();
|
||||||
}
|
}
|
||||||
|
|
||||||
function runToCursor() {
|
export function runToPC(pc: number) {
|
||||||
if (!checkRunReady()) return;
|
if (!checkRunReady() || !(pc >= 0)) return;
|
||||||
setupBreakpoint("toline");
|
setupBreakpoint("toline");
|
||||||
var pc = getEditorPC();
|
console.log("Run to", pc.toString(16));
|
||||||
if (pc >= 0) {
|
if (platform.runToPC) {
|
||||||
console.log("Run to", pc.toString(16));
|
platform.runToPC(pc);
|
||||||
if (platform.runToPC) {
|
} else {
|
||||||
platform.runToPC(pc);
|
platform.runEval((c) => {
|
||||||
} else {
|
return c.PC == pc;
|
||||||
platform.runEval((c) => {
|
});
|
||||||
return c.PC == pc;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function runToCursor() {
|
||||||
|
runToPC(getEditorPC());
|
||||||
|
}
|
||||||
|
|
||||||
function runUntilReturn() {
|
function runUntilReturn() {
|
||||||
if (!checkRunReady()) return;
|
if (!checkRunReady()) return;
|
||||||
setupBreakpoint("stepout");
|
setupBreakpoint("stepout");
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
|
|
||||||
//import CodeMirror = require("codemirror");
|
//import CodeMirror = require("codemirror");
|
||||||
import { SourceFile, WorkerError, Segment, FileData } from "../common/workertypes";
|
import { SourceFile, WorkerError, Segment, FileData } from "../common/workertypes";
|
||||||
import { Platform, EmuState, lookupSymbol, BaseDebugPlatform, BaseZ80MachinePlatform, BaseZ80Platform } from "../common/baseplatform";
|
import { Platform, EmuState, lookupSymbol, BaseDebugPlatform, BaseZ80MachinePlatform, BaseZ80Platform, CpuState } from "../common/baseplatform";
|
||||||
import { hex, lpad, rpad, safeident, rgb2bgr } from "../common/util";
|
import { hex, lpad, rpad, safeident, rgb2bgr } from "../common/util";
|
||||||
import { CodeAnalyzer } from "../common/analysis";
|
import { CodeAnalyzer } from "../common/analysis";
|
||||||
import { platform, platform_id, compparams, current_project, lastDebugState, projectWindows } from "./ui";
|
import { platform, platform_id, compparams, current_project, lastDebugState, projectWindows, runToPC } from "./ui";
|
||||||
import { ProbeRecorder, ProbeFlags } from "../common/recorder";
|
import { ProbeRecorder, ProbeFlags } from "../common/recorder";
|
||||||
import { getMousePos } from "../common/emu";
|
import { getMousePos } from "../common/emu";
|
||||||
import * as pixed from "./pixeleditor";
|
import * as pixed from "./pixeleditor";
|
||||||
@ -108,12 +108,14 @@ export class SourceEditor implements ProjectView {
|
|||||||
|
|
||||||
setupEditor() {
|
setupEditor() {
|
||||||
var timer;
|
var timer;
|
||||||
|
// update file in project (and recompile) when edits made
|
||||||
this.editor.on('changes', (ed, changeobj) => {
|
this.editor.on('changes', (ed, changeobj) => {
|
||||||
clearTimeout(timer);
|
clearTimeout(timer);
|
||||||
timer = setTimeout( () => {
|
timer = setTimeout( () => {
|
||||||
current_project.updateFile(this.path, this.editor.getValue());
|
current_project.updateFile(this.path, this.editor.getValue());
|
||||||
}, 300);
|
}, 300);
|
||||||
});
|
});
|
||||||
|
// inspect symbol when it's highlighted (double-click)
|
||||||
this.editor.on('cursorActivity', (ed) => {
|
this.editor.on('cursorActivity', (ed) => {
|
||||||
var start = this.editor.getCursor(true);
|
var start = this.editor.getCursor(true);
|
||||||
var end = this.editor.getCursor(false);
|
var end = this.editor.getCursor(false);
|
||||||
@ -124,10 +126,15 @@ export class SourceEditor implements ProjectView {
|
|||||||
this.inspect(null);
|
this.inspect(null);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
//scrollProfileView(editor);
|
// gutter clicked
|
||||||
|
this.editor.on("gutterClick", (cm, n) => {
|
||||||
|
this.toggleBreakpoint(n);
|
||||||
|
});
|
||||||
|
// set editor mode for highlighting, etc
|
||||||
this.editor.setOption("mode", this.mode);
|
this.editor.setOption("mode", this.mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
inspect(ident : string) : void {
|
inspect(ident : string) : void {
|
||||||
var result;
|
var result;
|
||||||
if (platform.inspect) {
|
if (platform.inspect) {
|
||||||
@ -333,7 +340,6 @@ export class SourceEditor implements ProjectView {
|
|||||||
var line = this.getActiveLine();
|
var line = this.getActiveLine();
|
||||||
if (line >= 0) {
|
if (line >= 0) {
|
||||||
this.setCurrentLine(line, moveCursor);
|
this.setCurrentLine(line, moveCursor);
|
||||||
// TODO: switch to disasm?
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -353,6 +359,10 @@ export class SourceEditor implements ProjectView {
|
|||||||
this.refreshListing();
|
this.refreshListing();
|
||||||
this.refreshDebugState(moveCursor);
|
this.refreshDebugState(moveCursor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tick() {
|
||||||
|
this.refreshDebugState(false);
|
||||||
|
}
|
||||||
|
|
||||||
getLine(line : number) {
|
getLine(line : number) {
|
||||||
return this.editor.getLine(line-1);
|
return this.editor.getLine(line-1);
|
||||||
@ -375,6 +385,13 @@ export class SourceEditor implements ProjectView {
|
|||||||
undoStep() {
|
undoStep() {
|
||||||
this.editor.execCommand('undo');
|
this.editor.execCommand('undo');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
toggleBreakpoint(lineno: number) {
|
||||||
|
if (this.sourcefile != null) {
|
||||||
|
var targetPC = this.sourcefile.line2offset[lineno+1];
|
||||||
|
runToPC(targetPC);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
@ -859,7 +876,7 @@ abstract class ProbeViewBaseBase {
|
|||||||
|
|
||||||
abstract tick() : void;
|
abstract tick() : void;
|
||||||
|
|
||||||
redraw( eventfn:(op,addr,col,row,clk) => void ) {
|
redraw( eventfn:(op,addr,col,row,clk,value) => void ) {
|
||||||
var p = this.probe;
|
var p = this.probe;
|
||||||
if (!p || !p.idx) return; // if no probe, or if empty
|
if (!p || !p.idx) return; // if no probe, or if empty
|
||||||
var row=0;
|
var row=0;
|
||||||
@ -867,20 +884,21 @@ abstract class ProbeViewBaseBase {
|
|||||||
var clk=0;
|
var clk=0;
|
||||||
for (var i=0; i<p.idx; i++) {
|
for (var i=0; i<p.idx; i++) {
|
||||||
var word = p.buf[i];
|
var word = p.buf[i];
|
||||||
var addr = word & 0xffffff;
|
var addr = word & 0xffff;
|
||||||
|
var value = (word >> 16) & 0xff;
|
||||||
var op = word & 0xff000000;
|
var op = word & 0xff000000;
|
||||||
switch (op) {
|
switch (op) {
|
||||||
case ProbeFlags.SCANLINE: row++; col=0; break;
|
case ProbeFlags.SCANLINE: row++; col=0; break;
|
||||||
case ProbeFlags.FRAME: row=0; col=0; break;
|
case ProbeFlags.FRAME: row=0; col=0; break;
|
||||||
case ProbeFlags.CLOCKS: col += addr; clk += addr; break;
|
case ProbeFlags.CLOCKS: col += addr; clk += addr; break;
|
||||||
default:
|
default:
|
||||||
eventfn(op, addr, col, row, clk);
|
eventfn(op, addr, col, row, clk, value);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
opToString(op:number, addr?:number) {
|
opToString(op:number, addr?:number, value?:number) {
|
||||||
var s = "";
|
var s = "";
|
||||||
switch (op) {
|
switch (op) {
|
||||||
case ProbeFlags.EXECUTE: s = "Exec"; break;
|
case ProbeFlags.EXECUTE: s = "Exec"; break;
|
||||||
@ -894,7 +912,9 @@ abstract class ProbeViewBaseBase {
|
|||||||
case ProbeFlags.ILLEGAL: s = "Error"; break;
|
case ProbeFlags.ILLEGAL: s = "Error"; break;
|
||||||
default: return "";
|
default: return "";
|
||||||
}
|
}
|
||||||
return typeof addr == 'number' ? s + " " + this.addr2str(addr) : s;
|
if (typeof addr == 'number') s += " " + this.addr2str(addr);
|
||||||
|
if (typeof value == 'number') s += " = $" + hex(value,2);
|
||||||
|
return s;
|
||||||
}
|
}
|
||||||
getOpRGB(op:number) : number {
|
getOpRGB(op:number) : number {
|
||||||
switch (op) {
|
switch (op) {
|
||||||
@ -987,9 +1007,9 @@ abstract class ProbeBitmapViewBase extends ProbeViewBase {
|
|||||||
x = x|0;
|
x = x|0;
|
||||||
y = y|0;
|
y = y|0;
|
||||||
var s = "";
|
var s = "";
|
||||||
this.redraw( (op,addr,col,row) => {
|
this.redraw( (op,addr,col,row,clk,value) => {
|
||||||
if (y == row && x == col) {
|
if (y == row && x == col) {
|
||||||
s += "\n" + this.opToString(op, addr);
|
s += "\n" + this.opToString(op, addr, value);
|
||||||
}
|
}
|
||||||
} );
|
} );
|
||||||
return 'X: ' + x + ' Y: ' + y + ' ' + s;
|
return 'X: ' + x + ' Y: ' + y + ' ' + s;
|
||||||
@ -1038,13 +1058,13 @@ export class AddressHeatMapView extends ProbeBitmapViewBase implements ProjectVi
|
|||||||
var s = this.addr2str(a);
|
var s = this.addr2str(a);
|
||||||
var pc = -1;
|
var pc = -1;
|
||||||
var already = {};
|
var already = {};
|
||||||
this.redraw( (op,addr,col,row) => {
|
this.redraw( (op,addr,col,row,clk,value) => {
|
||||||
if (op == ProbeFlags.EXECUTE) {
|
if (op == ProbeFlags.EXECUTE) {
|
||||||
pc = addr;
|
pc = addr;
|
||||||
}
|
}
|
||||||
var key = op|pc;
|
var key = op|pc;
|
||||||
if (addr == a && !already[key]) {
|
if (addr == a && !already[key]) {
|
||||||
s += "\nPC " + this.addr2str(pc) + " " + this.opToString(op);
|
s += "\nPC " + this.addr2str(pc) + " " + this.opToString(op, null, value);
|
||||||
already[key] = 1;
|
already[key] = 1;
|
||||||
}
|
}
|
||||||
} );
|
} );
|
||||||
@ -1134,7 +1154,7 @@ export class ProbeLogView extends ProbeViewBaseBase {
|
|||||||
getMemoryLineAt(row : number) : string {
|
getMemoryLineAt(row : number) : string {
|
||||||
var line = this.dumplines && this.dumplines[row];
|
var line = this.dumplines && this.dumplines[row];
|
||||||
if (line != null) {
|
if (line != null) {
|
||||||
var xtra = line.info.join(" ");
|
var xtra = line.info.join(", ");
|
||||||
return "(" + lpad(line.row,3) + ", " + lpad(line.col,3) + ") " + rpad(line.asm||"",20) + xtra;
|
return "(" + lpad(line.row,3) + ", " + lpad(line.col,3) + ") " + rpad(line.asm||"",20) + xtra;
|
||||||
} else return "";
|
} else return "";
|
||||||
}
|
}
|
||||||
@ -1145,7 +1165,7 @@ export class ProbeLogView extends ProbeViewBaseBase {
|
|||||||
const isz80 = platform instanceof BaseZ80MachinePlatform || platform instanceof BaseZ80Platform; // TODO?
|
const isz80 = platform instanceof BaseZ80MachinePlatform || platform instanceof BaseZ80Platform; // TODO?
|
||||||
// cache each line in frame
|
// cache each line in frame
|
||||||
this.dumplines = {};
|
this.dumplines = {};
|
||||||
this.redraw((op,addr,col,row,clk) => {
|
this.redraw((op,addr,col,row,clk,value) => {
|
||||||
if (isz80) clk >>= 2;
|
if (isz80) clk >>= 2;
|
||||||
var line = this.dumplines[clk];
|
var line = this.dumplines[clk];
|
||||||
if (line == null) {
|
if (line == null) {
|
||||||
@ -1160,7 +1180,7 @@ export class ProbeLogView extends ProbeViewBaseBase {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
var xtra = this.opToString(op, addr);
|
var xtra = this.opToString(op, addr, value);
|
||||||
if (xtra != "") line.info.push(xtra);
|
if (xtra != "") line.info.push(xtra);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user