2015-06-07 11:19:33 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <avr/pgmspace.h>
|
|
|
|
|
|
|
|
#include "hd44780.h"
|
|
|
|
#include "status.h"
|
|
|
|
|
|
|
|
#define CTRL_PORT PORTB
|
|
|
|
#define CTRL_DDR DDRB
|
2015-06-07 17:58:14 +01:00
|
|
|
#define CTRL_DIN PINB
|
|
|
|
|
|
|
|
// Outputs
|
2015-06-07 11:19:33 +01:00
|
|
|
#define STEP_MASK 0x01
|
2015-06-07 11:45:34 +01:00
|
|
|
#define SINGLE_MASK 0x02
|
|
|
|
#define RESET_MASK 0x04
|
2015-06-07 17:58:14 +01:00
|
|
|
#define BRKPT_ENABLE_MASK 0x08
|
|
|
|
#define BRKPT_CLOCK_MASK 0x10
|
|
|
|
#define BRKPT_DATA_MASK 0x20
|
|
|
|
// Inputs
|
|
|
|
#define BRKPT_INTERRUPTED_MASK 0x40
|
|
|
|
#define BRKPT_ACTIVE_MASK 0x80
|
|
|
|
|
|
|
|
#define CTRL_MASK (SINGLE_MASK | STEP_MASK | RESET_MASK | BRKPT_ENABLE_MASK | BRKPT_CLOCK_MASK | BRKPT_DATA_MASK)
|
2015-06-07 11:19:33 +01:00
|
|
|
|
|
|
|
#define AL_PORT PORTD
|
|
|
|
#define AL_DIN PIND
|
|
|
|
#define AL_MASK 0x00
|
|
|
|
#define AL_DDR DDRD
|
|
|
|
|
|
|
|
#define AH_PORT PORTE
|
|
|
|
#define AH_DIN PINE
|
|
|
|
#define AH_MASK 0x00
|
|
|
|
#define AH_DDR DDRE
|
|
|
|
|
2015-06-07 17:58:14 +01:00
|
|
|
#define VERSION "0.11"
|
2015-06-07 11:19:33 +01:00
|
|
|
|
|
|
|
#define NUMCMDS 10
|
|
|
|
#define MAXBKPTS 4
|
|
|
|
|
|
|
|
int numbkpts = 0;
|
|
|
|
|
|
|
|
int single;
|
|
|
|
long trace;
|
|
|
|
long instructions = 1;
|
|
|
|
|
|
|
|
unsigned int breakpoints[MAXBKPTS] = {
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
char *cmdStrings[NUMCMDS] = {
|
|
|
|
"help",
|
|
|
|
"reset",
|
2015-06-07 17:58:14 +01:00
|
|
|
"interrupt",
|
2015-06-07 11:19:33 +01:00
|
|
|
"address",
|
|
|
|
"step",
|
|
|
|
"trace",
|
|
|
|
"blist",
|
2015-06-07 17:58:14 +01:00
|
|
|
"break",
|
|
|
|
"bclear",
|
|
|
|
"continue",
|
2015-06-07 11:19:33 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#define Delay_us(__us) \
|
|
|
|
if((unsigned long) (F_CPU/1000000.0 * __us) != F_CPU/1000000.0 * __us)\
|
|
|
|
__builtin_avr_delay_cycles((unsigned long) ( F_CPU/1000000.0 * __us)+1);\
|
|
|
|
else __builtin_avr_delay_cycles((unsigned long) ( F_CPU/1000000.0 * __us))
|
|
|
|
|
|
|
|
#define Delay_ms(__ms) \
|
|
|
|
if((unsigned long) (F_CPU/1000.0 * __ms) != F_CPU/1000.0 * __ms)\
|
|
|
|
__builtin_avr_delay_cycles((unsigned long) ( F_CPU/1000.0 * __ms)+1);\
|
|
|
|
else __builtin_avr_delay_cycles((unsigned long) ( F_CPU/1000.0 * __ms))
|
|
|
|
|
|
|
|
char message[32];
|
|
|
|
char command[32];
|
|
|
|
|
|
|
|
void readCmd(char *cmd) {
|
|
|
|
char c;
|
|
|
|
int i = 0;
|
|
|
|
log0(">> ");
|
|
|
|
while (1) {
|
|
|
|
c = Serial_RxByte0();
|
|
|
|
if (c == 8) {
|
|
|
|
// Handle backspace/delete
|
|
|
|
if (i > 0) {
|
|
|
|
i--;
|
|
|
|
Serial_TxByte0(c);
|
|
|
|
Serial_TxByte0(32);
|
|
|
|
Serial_TxByte0(c);
|
|
|
|
}
|
|
|
|
} else if (c == 13) {
|
|
|
|
// Handle return
|
|
|
|
if (i == 0) {
|
|
|
|
while (cmd[i]) {
|
|
|
|
Serial_TxByte0(cmd[i++]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cmd[i] = 0;
|
|
|
|
}
|
|
|
|
Serial_TxByte0(10);
|
|
|
|
Serial_TxByte0(13);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
// Handle any other character
|
|
|
|
Serial_TxByte0(c);
|
|
|
|
cmd[i] = c;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSingle(int i) {
|
|
|
|
single = i;
|
|
|
|
if (single) {
|
|
|
|
CTRL_PORT |= SINGLE_MASK;
|
|
|
|
} else {
|
|
|
|
CTRL_PORT &= ~SINGLE_MASK;
|
|
|
|
}
|
2015-06-07 17:58:14 +01:00
|
|
|
Delay_us(10);
|
2015-06-07 11:19:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void setTrace(long i) {
|
|
|
|
trace = i;
|
|
|
|
if (trace) {
|
2015-06-07 17:58:14 +01:00
|
|
|
log0("Tracing every %ld instructions while single stepping\n", trace);
|
2015-06-07 11:19:33 +01:00
|
|
|
} else {
|
|
|
|
log0("Tracing disabled\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-07 17:58:14 +01:00
|
|
|
void version() {
|
|
|
|
log0("Atom Bus Monitor version %s\n", VERSION);
|
|
|
|
log0("Compiled at %s on %s\n",__TIME__,__DATE__);
|
|
|
|
}
|
2015-06-07 11:19:33 +01:00
|
|
|
|
|
|
|
/*******************************************
|
|
|
|
* Commands
|
|
|
|
*******************************************/
|
|
|
|
|
|
|
|
void doCmdHelp(char *params) {
|
|
|
|
int i;
|
2015-06-07 17:58:14 +01:00
|
|
|
version();
|
2015-06-07 11:19:33 +01:00
|
|
|
log0("Commands:\n");
|
|
|
|
for (i = 0; i < NUMCMDS; i++) {
|
|
|
|
log0(" %s\n", cmdStrings[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdAddr() {
|
|
|
|
unsigned int addr = AH_DIN << 8 | AL_DIN;
|
|
|
|
sprintf(message, "%04X", addr);
|
|
|
|
lcd_goto(6);
|
|
|
|
lcd_puts(message);
|
|
|
|
log0("%s\n", message);
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdStep(char *params) {
|
|
|
|
long i;
|
|
|
|
long j;
|
|
|
|
|
|
|
|
if (!single) {
|
|
|
|
log0("Use the break command to stop the 6502\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sscanf(params, "%ld", &instructions);
|
|
|
|
if (instructions <= 0) {
|
|
|
|
log0("Number of instuctions must be positive\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
log0("Stepping %ld instructions\n", instructions);
|
|
|
|
|
|
|
|
j = trace;
|
|
|
|
for (i = 1; i <= instructions; i++) {
|
|
|
|
// Step the 6502
|
|
|
|
CTRL_PORT &= ~STEP_MASK;
|
2015-06-07 17:58:14 +01:00
|
|
|
Delay_us(2);
|
2015-06-07 11:19:33 +01:00
|
|
|
CTRL_PORT |= STEP_MASK;
|
2015-06-07 17:58:14 +01:00
|
|
|
Delay_us(2);
|
2015-06-07 11:19:33 +01:00
|
|
|
if (i == instructions || (trace && (--j == 0))) {
|
2015-06-07 17:58:14 +01:00
|
|
|
Delay_us(10);
|
2015-06-07 11:19:33 +01:00
|
|
|
doCmdAddr();
|
|
|
|
j = trace;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdReset(char *params) {
|
2015-06-07 11:45:34 +01:00
|
|
|
log0("Resetting 6502\n");
|
|
|
|
CTRL_PORT |= RESET_MASK;
|
|
|
|
Delay_us(100);
|
|
|
|
CTRL_PORT &= ~RESET_MASK;
|
2015-06-07 11:19:33 +01:00
|
|
|
}
|
|
|
|
|
2015-06-07 17:58:14 +01:00
|
|
|
void doCmdInterrupt(char *params) {
|
|
|
|
setSingle(1);
|
2015-06-07 11:19:33 +01:00
|
|
|
doCmdAddr();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void doCmdTrace(char *params) {
|
|
|
|
long i;
|
|
|
|
sscanf(params, "%ld", &i);
|
|
|
|
setTrace(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdBList(char *params) {
|
|
|
|
int i;
|
|
|
|
if (numbkpts) {
|
|
|
|
for (i = 0; i < numbkpts; i++) {
|
|
|
|
log0("%d: %04X\n", i, breakpoints[i]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log0("No breakpoints set\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-07 17:58:14 +01:00
|
|
|
void doCmdBreak(char *params) {
|
2015-06-07 11:19:33 +01:00
|
|
|
int i;
|
|
|
|
unsigned int addr;
|
|
|
|
sscanf(params, "%x", &addr);
|
|
|
|
if (numbkpts == MAXBKPTS) {
|
|
|
|
log0("All breakpoints are already set\n");
|
|
|
|
doCmdBList(NULL);
|
|
|
|
return;
|
|
|
|
}
|
2015-06-07 17:58:14 +01:00
|
|
|
for (i = 0; i < numbkpts; i++) {
|
|
|
|
if (breakpoints[i] == addr) {
|
|
|
|
log0("Breakpoint already set at %04X\n", addr);
|
|
|
|
doCmdBList(NULL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-07 11:19:33 +01:00
|
|
|
numbkpts++;
|
|
|
|
for (i = numbkpts - 2; i >= -1; i--) {
|
|
|
|
if (i == -1 || breakpoints[i] < addr) {
|
|
|
|
log0("Setting breakpoint at %04X\n", addr);
|
|
|
|
breakpoints[i + 1] = addr;
|
|
|
|
doCmdBList(NULL);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
breakpoints[i + 1] = breakpoints[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdBClear(char *params) {
|
|
|
|
int i;
|
2015-06-07 17:58:14 +01:00
|
|
|
int n = 0;
|
2015-06-07 11:19:33 +01:00
|
|
|
sscanf(params, "%d", &n);
|
|
|
|
if (n >= numbkpts) {
|
|
|
|
log0("Breakpoint %d not set\n", n);
|
|
|
|
} else {
|
|
|
|
log0("Removing breakpoint at %04X\n", breakpoints[n]);
|
|
|
|
for (i = n; i < numbkpts; i++) {
|
|
|
|
breakpoints[i] = breakpoints[i + 1];
|
|
|
|
}
|
|
|
|
numbkpts--;
|
|
|
|
}
|
|
|
|
doCmdBList(NULL);
|
|
|
|
}
|
|
|
|
|
2015-06-07 17:58:14 +01:00
|
|
|
void shiftBreakpointRegister(unsigned int addr) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
CTRL_PORT &= ~BRKPT_CLOCK_MASK;
|
|
|
|
if (addr & 1) {
|
|
|
|
CTRL_PORT |= BRKPT_DATA_MASK;
|
|
|
|
} else {
|
|
|
|
CTRL_PORT &= ~BRKPT_DATA_MASK;
|
|
|
|
}
|
|
|
|
Delay_us(10);
|
|
|
|
CTRL_PORT |= BRKPT_CLOCK_MASK;
|
|
|
|
Delay_us(10);
|
|
|
|
addr >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void doCmdContinue(char *params) {
|
|
|
|
int i;
|
|
|
|
int status;
|
|
|
|
doCmdBList(NULL);
|
|
|
|
|
|
|
|
// Disable breakpoints to allow loading
|
|
|
|
CTRL_PORT &= ~BRKPT_ENABLE_MASK;
|
|
|
|
|
|
|
|
// Load breakpoints into comparators
|
|
|
|
for (i = 0; i < MAXBKPTS; i++) {
|
|
|
|
shiftBreakpointRegister(i < numbkpts ? breakpoints[i] : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable breakpoints
|
|
|
|
CTRL_PORT |= BRKPT_ENABLE_MASK;
|
|
|
|
|
|
|
|
// Disable single stepping
|
|
|
|
setSingle(0);
|
|
|
|
|
|
|
|
// Wait for breakpoint to become active
|
|
|
|
log0("6502 free running...\n");
|
|
|
|
do {
|
|
|
|
status = CTRL_DIN;
|
|
|
|
} while (!(status & BRKPT_ACTIVE_MASK) && !(status && BRKPT_INTERRUPTED_MASK));
|
|
|
|
|
|
|
|
// Output cause
|
|
|
|
if (status & BRKPT_ACTIVE_MASK) {
|
|
|
|
log0("Breakpoint hit at ");
|
|
|
|
} else {
|
|
|
|
log0("Interrupted at ");
|
|
|
|
}
|
|
|
|
doCmdAddr();
|
|
|
|
|
|
|
|
// Enable single stepping
|
|
|
|
setSingle(1);
|
|
|
|
|
|
|
|
// Disable breakpoints
|
|
|
|
CTRL_PORT &= ~BRKPT_ENABLE_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-07 11:19:33 +01:00
|
|
|
void initialize() {
|
|
|
|
CTRL_DDR = CTRL_MASK;
|
|
|
|
AL_DDR = AL_MASK;
|
|
|
|
AH_DDR = AH_MASK;
|
2015-06-07 17:58:14 +01:00
|
|
|
CTRL_PORT &= 0;
|
2015-06-07 11:19:33 +01:00
|
|
|
Serial_Init(57600,57600);
|
|
|
|
lcd_init();
|
|
|
|
lcd_puts("Addr: xxxx");
|
2015-06-07 17:58:14 +01:00
|
|
|
version();
|
|
|
|
setSingle(1);
|
|
|
|
setTrace(1);
|
|
|
|
log0("6502 paused...\n");
|
|
|
|
doCmdAddr();
|
2015-06-07 11:19:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void (*cmdFuncs[NUMCMDS])(char *params) = {
|
|
|
|
doCmdHelp,
|
|
|
|
doCmdReset,
|
2015-06-07 17:58:14 +01:00
|
|
|
doCmdInterrupt,
|
2015-06-07 11:19:33 +01:00
|
|
|
doCmdAddr,
|
|
|
|
doCmdStep,
|
|
|
|
doCmdTrace,
|
|
|
|
doCmdBList,
|
2015-06-07 17:58:14 +01:00
|
|
|
doCmdBreak,
|
|
|
|
doCmdBClear,
|
|
|
|
doCmdContinue
|
2015-06-07 11:19:33 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void dispatchCmd(char *cmd) {
|
|
|
|
int i;
|
|
|
|
char *cmdString;
|
|
|
|
|
|
|
|
|
|
|
|
int minLen;
|
|
|
|
int cmdStringLen;
|
|
|
|
|
|
|
|
int cmdLen = 0;
|
|
|
|
while (cmd[cmdLen] >= 'a' && cmd[cmdLen] <= 'z') {
|
|
|
|
cmdLen++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < NUMCMDS; i++) {
|
|
|
|
cmdString = cmdStrings[i];
|
|
|
|
cmdStringLen = strlen(cmdString);
|
|
|
|
minLen = cmdLen < cmdStringLen ? cmdLen : cmdStringLen;
|
|
|
|
if (strncmp(cmdString, cmd, minLen) == 0) {
|
|
|
|
(*cmdFuncs[i])(command + cmdLen);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log0("Unknown command %s\n", cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void) {
|
|
|
|
initialize();
|
|
|
|
while (1) {
|
|
|
|
readCmd(command);
|
|
|
|
dispatchCmd(command);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|