mirror of https://github.com/bsnes-emu/bsnes.git
410 lines
15 KiB
C
410 lines
15 KiB
C
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
#include "debugger.h"
|
||
|
#include "memory.h"
|
||
|
#include "z80_cpu.h"
|
||
|
#include "gb.h"
|
||
|
|
||
|
|
||
|
typedef struct {
|
||
|
enum {
|
||
|
LVALUE_MEMORY,
|
||
|
LVALUE_REG16,
|
||
|
LVALUE_REG_H,
|
||
|
LVALUE_REG_L,
|
||
|
} kind;
|
||
|
union {
|
||
|
unsigned short *register_address;
|
||
|
unsigned short memory_address;
|
||
|
};
|
||
|
} lvalue_t;
|
||
|
|
||
|
static unsigned short read_lvalue(GB_gameboy_t *gb, lvalue_t lvalue)
|
||
|
{
|
||
|
/* Not used until we add support for operators like += */
|
||
|
switch (lvalue.kind) {
|
||
|
case LVALUE_MEMORY:
|
||
|
return read_memory(gb, lvalue.memory_address);
|
||
|
|
||
|
case LVALUE_REG16:
|
||
|
return *lvalue.register_address;
|
||
|
|
||
|
case LVALUE_REG_L:
|
||
|
return *lvalue.register_address & 0x00FF;
|
||
|
|
||
|
case LVALUE_REG_H:
|
||
|
return *lvalue.register_address >> 8;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void write_lvalue(GB_gameboy_t *gb, lvalue_t lvalue, unsigned short value)
|
||
|
{
|
||
|
switch (lvalue.kind) {
|
||
|
case LVALUE_MEMORY:
|
||
|
write_memory(gb, lvalue.memory_address, value);
|
||
|
return;
|
||
|
|
||
|
case LVALUE_REG16:
|
||
|
*lvalue.register_address = value;
|
||
|
return;
|
||
|
|
||
|
case LVALUE_REG_L:
|
||
|
*lvalue.register_address &= 0xFF00;
|
||
|
*lvalue.register_address |= value & 0xFF;
|
||
|
return;
|
||
|
|
||
|
case LVALUE_REG_H:
|
||
|
*lvalue.register_address &= 0x00FF;
|
||
|
*lvalue.register_address |= value << 8;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static unsigned short add(unsigned short a, unsigned short b) {return a + b;};
|
||
|
static unsigned short sub(unsigned short a, unsigned short b) {return a - b;};
|
||
|
static unsigned short mul(unsigned short a, unsigned short b) {return a * b;};
|
||
|
static unsigned short _div(unsigned short a, unsigned short b) {
|
||
|
if (b == 0) {
|
||
|
return 0;
|
||
|
}
|
||
|
return a / b;
|
||
|
};
|
||
|
static unsigned short mod(unsigned short a, unsigned short b) {
|
||
|
if (b == 0) {
|
||
|
return 0;
|
||
|
}
|
||
|
return a % b;
|
||
|
};
|
||
|
static unsigned short and(unsigned short a, unsigned short b) {return a & b;};
|
||
|
static unsigned short or(unsigned short a, unsigned short b) {return a | b;};
|
||
|
static unsigned short xor(unsigned short a, unsigned short b) {return a ^ b;};
|
||
|
static unsigned short shleft(unsigned short a, unsigned short b) {return a << b;};
|
||
|
static unsigned short shright(unsigned short a, unsigned short b) {return a >> b;};
|
||
|
static unsigned short assign(GB_gameboy_t *gb, lvalue_t a, unsigned short b)
|
||
|
{
|
||
|
write_lvalue(gb, a, b);
|
||
|
return read_lvalue(gb, a);
|
||
|
}
|
||
|
|
||
|
static struct {
|
||
|
const char *string;
|
||
|
char priority;
|
||
|
unsigned short (*operator)(unsigned short, unsigned short);
|
||
|
unsigned short (*lvalue_operator)(GB_gameboy_t *, lvalue_t, unsigned short);
|
||
|
} operators[] =
|
||
|
{
|
||
|
// Yes. This is not C-like. But it makes much more sense.
|
||
|
// Deal with it.
|
||
|
{"+", 0, add},
|
||
|
{"-", 0, sub},
|
||
|
{"|", 0, or},
|
||
|
{"*", 1, mul},
|
||
|
{"/", 1, _div},
|
||
|
{"%", 1, mod},
|
||
|
{"&", 1, and},
|
||
|
{"^", 1, xor},
|
||
|
{"<<", 2, shleft},
|
||
|
{">>", 2, shright},
|
||
|
{"=", 2, NULL, assign},
|
||
|
};
|
||
|
|
||
|
unsigned short debugger_evaluate(GB_gameboy_t *gb, const char *string, unsigned int length, bool *error);
|
||
|
|
||
|
static lvalue_t debugger_evaluate_lvalue(GB_gameboy_t *gb, const char *string, unsigned int length, bool *error)
|
||
|
{
|
||
|
*error = false;
|
||
|
// Strip whitespace
|
||
|
while (length && (string[0] == ' ' || string[0] == '\n' || string[0] == '\r' || string[0] == '\t')) {
|
||
|
string++;
|
||
|
length--;
|
||
|
}
|
||
|
while (length && (string[length-1] == ' ' || string[length-1] == '\n' || string[length-1] == '\r' || string[length-1] == '\t')) {
|
||
|
length--;
|
||
|
}
|
||
|
if (length == 0)
|
||
|
{
|
||
|
gb_log(gb, "Expected expression.\n");
|
||
|
*error = true;
|
||
|
return (lvalue_t){0,};
|
||
|
}
|
||
|
if (string[0] == '(' && string[length - 1] == ')') {
|
||
|
// Attempt to strip parentheses
|
||
|
signed int depth = 0;
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
if (string[i] == '(') depth++;
|
||
|
if (depth == 0) {
|
||
|
// First and last are not matching
|
||
|
depth = 1;
|
||
|
break;
|
||
|
}
|
||
|
if (string[i] == ')') depth--;
|
||
|
}
|
||
|
if (depth == 0) return debugger_evaluate_lvalue(gb, string + 1, length - 2, error);
|
||
|
}
|
||
|
else if (string[0] == '[' && string[length - 1] == ']') {
|
||
|
// Attempt to strip square parentheses (memory dereference)
|
||
|
signed int depth = 0;
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
if (string[i] == '[') depth++;
|
||
|
if (depth == 0) {
|
||
|
// First and last are not matching
|
||
|
depth = 1;
|
||
|
break;
|
||
|
}
|
||
|
if (string[i] == ']') depth--;
|
||
|
}
|
||
|
if (depth == 0) {
|
||
|
return (lvalue_t){LVALUE_MEMORY, .memory_address = debugger_evaluate(gb, string + 1, length - 2, error)};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Registers
|
||
|
if (string[0] == '$') {
|
||
|
if (length == 2) {
|
||
|
switch (string[1]) {
|
||
|
case 'a': return (lvalue_t){LVALUE_REG_H, .register_address = &gb->registers[GB_REGISTER_AF]};
|
||
|
case 'f': return (lvalue_t){LVALUE_REG_L, .register_address = &gb->registers[GB_REGISTER_AF]};
|
||
|
case 'b': return (lvalue_t){LVALUE_REG_H, .register_address = &gb->registers[GB_REGISTER_BC]};
|
||
|
case 'c': return (lvalue_t){LVALUE_REG_L, .register_address = &gb->registers[GB_REGISTER_BC]};
|
||
|
case 'd': return (lvalue_t){LVALUE_REG_H, .register_address = &gb->registers[GB_REGISTER_DE]};
|
||
|
case 'e': return (lvalue_t){LVALUE_REG_L, .register_address = &gb->registers[GB_REGISTER_DE]};
|
||
|
case 'h': return (lvalue_t){LVALUE_REG_H, .register_address = &gb->registers[GB_REGISTER_HL]};
|
||
|
case 'l': return (lvalue_t){LVALUE_REG_L, .register_address = &gb->registers[GB_REGISTER_HL]};
|
||
|
}
|
||
|
}
|
||
|
else if (length == 3) {
|
||
|
switch (string[1]) {
|
||
|
case 'a': if (string[2] == 'f') return (lvalue_t){LVALUE_REG16, .register_address = &gb->registers[GB_REGISTER_AF]};
|
||
|
case 'b': if (string[2] == 'c') return (lvalue_t){LVALUE_REG16, .register_address = &gb->registers[GB_REGISTER_BC]};
|
||
|
case 'd': if (string[2] == 'e') return (lvalue_t){LVALUE_REG16, .register_address = &gb->registers[GB_REGISTER_DE]};
|
||
|
case 'h': if (string[2] == 'l') return (lvalue_t){LVALUE_REG16, .register_address = &gb->registers[GB_REGISTER_HL]};
|
||
|
case 's': if (string[2] == 'p') return (lvalue_t){LVALUE_REG16, .register_address = &gb->registers[GB_REGISTER_SP]};
|
||
|
case 'p': if (string[2] == 'c') return (lvalue_t){LVALUE_REG16, .register_address = &gb->pc};
|
||
|
}
|
||
|
}
|
||
|
gb_log(gb, "Unknown register: %.*s\n", length, string);
|
||
|
*error = true;
|
||
|
return (lvalue_t){0,};
|
||
|
}
|
||
|
|
||
|
gb_log(gb, "Expression is not an lvalue: %.*s\n", length, string);
|
||
|
*error = true;
|
||
|
return (lvalue_t){0,};
|
||
|
}
|
||
|
|
||
|
unsigned short debugger_evaluate(GB_gameboy_t *gb, const char *string, unsigned int length, bool *error)
|
||
|
{
|
||
|
*error = false;
|
||
|
// Strip whitespace
|
||
|
while (length && (string[0] == ' ' || string[0] == '\n' || string[0] == '\r' || string[0] == '\t')) {
|
||
|
string++;
|
||
|
length--;
|
||
|
}
|
||
|
while (length && (string[length-1] == ' ' || string[length-1] == '\n' || string[length-1] == '\r' || string[length-1] == '\t')) {
|
||
|
length--;
|
||
|
}
|
||
|
if (length == 0)
|
||
|
{
|
||
|
gb_log(gb, "Expected expression.\n");
|
||
|
*error = true;
|
||
|
return -1;
|
||
|
}
|
||
|
if (string[0] == '(' && string[length - 1] == ')') {
|
||
|
// Attempt to strip parentheses
|
||
|
signed int depth = 0;
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
if (string[i] == '(') depth++;
|
||
|
if (depth == 0) {
|
||
|
// First and last are not matching
|
||
|
depth = 1;
|
||
|
break;
|
||
|
}
|
||
|
if (string[i] == ')') depth--;
|
||
|
}
|
||
|
if (depth == 0) return debugger_evaluate(gb, string + 1, length - 2, error);
|
||
|
}
|
||
|
else if (string[0] == '[' && string[length - 1] == ']') {
|
||
|
// Attempt to strip square parentheses (memory dereference)
|
||
|
signed int depth = 0;
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
if (string[i] == '[') depth++;
|
||
|
if (depth == 0) {
|
||
|
// First and last are not matching
|
||
|
depth = 1;
|
||
|
break;
|
||
|
}
|
||
|
if (string[i] == ']') depth--;
|
||
|
}
|
||
|
if (depth == 0) return read_memory(gb, debugger_evaluate(gb, string + 1, length - 2, error));
|
||
|
}
|
||
|
// Search for lowest priority operator
|
||
|
signed int depth = 0;
|
||
|
unsigned int operator_index = -1;
|
||
|
unsigned int operator_pos = 0;
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
if (string[i] == '(') depth++;
|
||
|
else if (string[i] == ')') depth--;
|
||
|
else if (string[i] == '[') depth++;
|
||
|
else if (string[i] == ']') depth--;
|
||
|
else if (depth == 0) {
|
||
|
for (int j = 0; j < sizeof(operators) / sizeof(operators[0]); j++) {
|
||
|
if (strlen(operators[j].string) > length - i) continue; // Operator too big.
|
||
|
// Priority higher than what we already have.
|
||
|
if (operator_index != -1 && operators[operator_index].priority > operators[j].priority) continue;
|
||
|
if (memcmp(string + i, operators[j].string, strlen(operators[j].string)) == 0) {
|
||
|
// Found an operator!
|
||
|
operator_pos = i;
|
||
|
operator_index = j;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (operator_index != -1) {
|
||
|
unsigned int right_start = (unsigned int)(operator_pos + strlen(operators[operator_index].string));
|
||
|
unsigned short right = debugger_evaluate(gb, string + right_start, length - right_start, error);
|
||
|
if (*error) return -1;
|
||
|
if (operators[operator_index].lvalue_operator) {
|
||
|
lvalue_t left = debugger_evaluate_lvalue(gb, string, operator_pos, error);
|
||
|
if (*error) return -1;
|
||
|
return operators[operator_index].lvalue_operator(gb, left, right);
|
||
|
}
|
||
|
unsigned short left = debugger_evaluate(gb, string, operator_pos, error);
|
||
|
if (*error) return -1;
|
||
|
return operators[operator_index].operator(left, right);
|
||
|
}
|
||
|
|
||
|
// Not an expression - must be a register or a literal
|
||
|
|
||
|
// Registers
|
||
|
if (string[0] == '$') {
|
||
|
if (length == 2) {
|
||
|
switch (string[1]) {
|
||
|
case 'a': return gb->registers[GB_REGISTER_AF] >> 8;
|
||
|
case 'f': return gb->registers[GB_REGISTER_AF] & 0xFF;
|
||
|
case 'b': return gb->registers[GB_REGISTER_BC] >> 8;
|
||
|
case 'c': return gb->registers[GB_REGISTER_BC] & 0xFF;
|
||
|
case 'd': return gb->registers[GB_REGISTER_DE] >> 8;
|
||
|
case 'e': return gb->registers[GB_REGISTER_DE] & 0xFF;
|
||
|
case 'h': return gb->registers[GB_REGISTER_HL] >> 8;
|
||
|
case 'l': return gb->registers[GB_REGISTER_HL] & 0xFF;
|
||
|
}
|
||
|
}
|
||
|
else if (length == 3) {
|
||
|
switch (string[1]) {
|
||
|
case 'a': if (string[2] == 'f') return gb->registers[GB_REGISTER_AF];
|
||
|
case 'b': if (string[2] == 'c') return gb->registers[GB_REGISTER_BC];
|
||
|
case 'd': if (string[2] == 'e') return gb->registers[GB_REGISTER_DE];
|
||
|
case 'h': if (string[2] == 'l') return gb->registers[GB_REGISTER_HL];
|
||
|
case 's': if (string[2] == 'p') return gb->registers[GB_REGISTER_SP];
|
||
|
case 'p': if (string[2] == 'c') return gb->pc;
|
||
|
}
|
||
|
}
|
||
|
gb_log(gb, "Unknown register: %.*s\n", length, string);
|
||
|
*error = true;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
char *end;
|
||
|
unsigned short literal = (unsigned short) (strtol(string, &end, 16));
|
||
|
if (end != string + length) {
|
||
|
gb_log(gb, "Failed to parse: %.*s\n", length, string);
|
||
|
*error = true;
|
||
|
return -1;
|
||
|
}
|
||
|
return literal;
|
||
|
}
|
||
|
|
||
|
|
||
|
/* The debugger interface is quite primitive. One letter commands with a single parameter maximum.
|
||
|
Only one breakpoint is allowed at a time. More features will be added later. */
|
||
|
void debugger_run(GB_gameboy_t *gb)
|
||
|
{
|
||
|
char *input = NULL;
|
||
|
if (gb->debug_next_command && gb->debug_call_depth == 0) {
|
||
|
gb->debug_stopped = true;
|
||
|
}
|
||
|
if (gb->debug_fin_command && gb->debug_call_depth == -1) {
|
||
|
gb->debug_stopped = true;
|
||
|
}
|
||
|
if (gb->debug_stopped) {
|
||
|
cpu_disassemble(gb, gb->pc, 5);
|
||
|
}
|
||
|
next_command:
|
||
|
if (input) {
|
||
|
free(input);
|
||
|
}
|
||
|
if (gb->pc == gb->breakpoint && !gb->debug_stopped) {
|
||
|
gb->debug_stopped = true;
|
||
|
gb_log(gb, "Breakpoint: PC = %04x\n", gb->pc);
|
||
|
cpu_disassemble(gb, gb->pc, 5);
|
||
|
}
|
||
|
if (gb->debug_stopped) {
|
||
|
gb->debug_next_command = false;
|
||
|
gb->debug_fin_command = false;
|
||
|
input = gb->input_callback(gb);
|
||
|
switch (*input) {
|
||
|
case 'c':
|
||
|
gb->debug_stopped = false;
|
||
|
break;
|
||
|
case 'n':
|
||
|
gb->debug_stopped = false;
|
||
|
gb->debug_next_command = true;
|
||
|
gb->debug_call_depth = 0;
|
||
|
break;
|
||
|
case 'f':
|
||
|
gb->debug_stopped = false;
|
||
|
gb->debug_fin_command = true;
|
||
|
gb->debug_call_depth = 0;
|
||
|
break;
|
||
|
case 's':
|
||
|
break;
|
||
|
case 'r':
|
||
|
gb_log(gb, "AF = %04x\n", gb->registers[GB_REGISTER_AF]);
|
||
|
gb_log(gb, "BC = %04x\n", gb->registers[GB_REGISTER_BC]);
|
||
|
gb_log(gb, "DE = %04x\n", gb->registers[GB_REGISTER_DE]);
|
||
|
gb_log(gb, "HL = %04x\n", gb->registers[GB_REGISTER_HL]);
|
||
|
gb_log(gb, "SP = %04x\n", gb->registers[GB_REGISTER_SP]);
|
||
|
gb_log(gb, "PC = %04x\n", gb->pc);
|
||
|
gb_log(gb, "TIMA = %d/%lu\n", gb->io_registers[GB_IO_TIMA], gb->tima_cycles);
|
||
|
gb_log(gb, "Display Controller: LY = %d/%lu\n", gb->io_registers[GB_IO_LY], gb->display_cycles % 456);
|
||
|
goto next_command;
|
||
|
case 'x':
|
||
|
{
|
||
|
bool error;
|
||
|
unsigned short addr = debugger_evaluate(gb, input + 1, (unsigned int)strlen(input + 1), &error);
|
||
|
if (!error) {
|
||
|
gb_log(gb, "%4x: ", addr);
|
||
|
for (int i = 0; i < 16; i++) {
|
||
|
gb_log(gb, "%02x ", read_memory(gb, addr + i));
|
||
|
}
|
||
|
gb_log(gb, "\n");
|
||
|
}
|
||
|
goto next_command;
|
||
|
}
|
||
|
case 'b':
|
||
|
{
|
||
|
bool error;
|
||
|
unsigned short result = debugger_evaluate(gb, input + 1, (unsigned int)strlen(input + 1), &error);
|
||
|
if (!error) {
|
||
|
gb_log(gb, "Breakpoint moved to %04x\n", gb->breakpoint = result);
|
||
|
}
|
||
|
goto next_command;
|
||
|
}
|
||
|
|
||
|
case 'p':
|
||
|
{
|
||
|
bool error;
|
||
|
unsigned short result = debugger_evaluate(gb, input + 1, (unsigned int)strlen(input + 1), &error);
|
||
|
if (!error) {
|
||
|
gb_log(gb, "=%04x\n", result);
|
||
|
}
|
||
|
goto next_command;
|
||
|
}
|
||
|
|
||
|
default:
|
||
|
goto next_command;
|
||
|
}
|
||
|
free(input);
|
||
|
}
|
||
|
}
|