diff options
| author | Matt Borgerson <mborgerson@gmail.com> | 2016-04-20 01:08:26 -0700 |
|---|---|---|
| committer | Matt Borgerson <mborgerson@gmail.com> | 2016-04-20 01:30:35 -0700 |
| commit | bceb2a5ae67e661b7ef00c07b14db425ca14453d (patch) | |
| tree | 6f79ff4903ec0213e676e4c2f7b97ce24d667d91 /gdbstub_sys.c | |
| parent | e677c14dc715da62ad5749a4ab740803e58b0a7f (diff) | |
| download | gdbstub-bceb2a5ae67e661b7ef00c07b14db425ca14453d.tar.gz | |
Add GDB stub code
Diffstat (limited to 'gdbstub_sys.c')
| -rw-r--r-- | gdbstub_sys.c | 341 |
1 files changed, 341 insertions, 0 deletions
diff --git a/gdbstub_sys.c b/gdbstub_sys.c new file mode 100644 index 0000000..2b17702 --- /dev/null +++ b/gdbstub_sys.c @@ -0,0 +1,341 @@ +/* + * Copyright (C) 2016 Matt Borgerson + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include "gdbstub.h" + +#ifdef __STRICT_ANSI__ +#define asm __asm__ +#endif + +#define SERIAL_COM1 0x3f8 +#define SERIAL_COM2 0x2f8 +#define SERIAL_PORT SERIAL_COM1 + +#define NUM_IDT_ENTRIES 32 + +/***************************************************************************** + * BSS Data + ****************************************************************************/ + +static struct dbg_idt_gate dbg_idt_gates[NUM_IDT_ENTRIES]; +static struct dbg_state dbg_state; + +/***************************************************************************** + * Misc. Functions + ****************************************************************************/ + +void *dbg_sys_memset(void *ptr, int data, size_t len) +{ + char *p = ptr; + + while (len--) { + *p++ = (char)data; + } + + return ptr; +} + +/* + * Get current code segment (CS register). + */ +uint32_t dbg_get_cs() +{ + uint32_t cs; + + asm( + "push %%cs;" + "pop %%eax;" + /* Outputs */ : "=a" (cs) + /* Inputs */ : /* None */ + /* Clobbers */ : /* None */ + ); + + return cs; +} + +/***************************************************************************** + * Interrupt Management Functions + ****************************************************************************/ + +/* + * Initialize idt_gates with the interrupt handlers. + */ +int dbg_init_gates() +{ + size_t i; + uint16_t cs; + + cs = dbg_get_cs(); + for (i = 0; i < NUM_IDT_ENTRIES; i++) { + dbg_idt_gates[i].flags = 0x8E00; + dbg_idt_gates[i].segment = cs; + dbg_idt_gates[i].offset_low = + ((uint32_t)dbg_int_handlers[i] ) & 0xffff; + dbg_idt_gates[i].offset_high = + ((uint32_t)dbg_int_handlers[i] >> 16) & 0xffff; + } + + return 0; +} + +/* + * Load a new IDT. + */ +int dbg_load_idt(struct dbg_idtr *idtr) +{ + asm( + "lidt %0" + /* Outputs */ : /* None */ + /* Inputs */ : "m" (*idtr) + /* Clobbers */ : /* None */ + ); + + return 0; +} + +/* + * Get current IDT. + */ +int dbg_store_idt(struct dbg_idtr *idtr) +{ + asm( + "sidt %0" + /* Outputs */ : "=m" (*idtr) + /* Inputs */ : /* None */ + /* Clobbers */ : /* None */ + ); + + return 0; +} + +/* + * Hook a vector of the current IDT. + */ +int dbg_hook_idt(uint8_t vector, const void *function) +{ + struct dbg_idtr idtr; + struct dbg_idt_gate *gates; + + dbg_store_idt(&idtr); + gates = (struct dbg_idt_gate *)idtr.offset; + gates[vector].flags = 0x8E00; + gates[vector].segment = dbg_get_cs(); + gates[vector].offset_low = (((uint32_t)function) ) & 0xffff; + gates[vector].offset_high = (((uint32_t)function) >> 16) & 0xffff; + + return 0; +} + +/* + * Initialize IDT gates and load the new IDT. + */ +int dbg_init_idt(void) +{ + struct dbg_idtr idtr; + + dbg_init_gates(); + idtr.len = sizeof(dbg_idt_gates)-1; + idtr.offset = (uint32_t)dbg_idt_gates; + dbg_load_idt(&idtr); + + return 0; +} + +/* + * Common interrupt handler routine. + */ +void dbg_int_handler(struct dbg_interrupt_state *istate) +{ + dbg_interrupt(istate); +} + +/* + * Debug interrupt handler. + */ +void dbg_interrupt(struct dbg_interrupt_state *istate) +{ + dbg_sys_memset(&dbg_state.registers, 0, sizeof(dbg_state.registers)); + + dbg_state.signum = istate->vector; + + /* Load Registers */ + dbg_state.registers[DBG_CPU_I386_REG_EAX] = istate->eax; + dbg_state.registers[DBG_CPU_I386_REG_ECX] = istate->ecx; + dbg_state.registers[DBG_CPU_I386_REG_EDX] = istate->edx; + dbg_state.registers[DBG_CPU_I386_REG_EBX] = istate->ebx; + dbg_state.registers[DBG_CPU_I386_REG_ESP] = istate->esp; + dbg_state.registers[DBG_CPU_I386_REG_EBP] = istate->ebp; + dbg_state.registers[DBG_CPU_I386_REG_ESI] = istate->esi; + dbg_state.registers[DBG_CPU_I386_REG_EDI] = istate->edi; + dbg_state.registers[DBG_CPU_I386_REG_PC] = istate->eip; + dbg_state.registers[DBG_CPU_I386_REG_CS] = istate->cs; + dbg_state.registers[DBG_CPU_I386_REG_PS] = istate->eflags; + dbg_state.registers[DBG_CPU_I386_REG_SS] = istate->ss; + dbg_state.registers[DBG_CPU_I386_REG_DS] = istate->ds; + dbg_state.registers[DBG_CPU_I386_REG_ES] = istate->es; + dbg_state.registers[DBG_CPU_I386_REG_FS] = istate->fs; + dbg_state.registers[DBG_CPU_I386_REG_GS] = istate->gs; + + dbg_main(&dbg_state); + + /* Restore Registers */ + istate->eax = dbg_state.registers[DBG_CPU_I386_REG_EAX]; + istate->ecx = dbg_state.registers[DBG_CPU_I386_REG_ECX]; + istate->edx = dbg_state.registers[DBG_CPU_I386_REG_EDX]; + istate->ebx = dbg_state.registers[DBG_CPU_I386_REG_EBX]; + istate->esp = dbg_state.registers[DBG_CPU_I386_REG_ESP]; + istate->ebp = dbg_state.registers[DBG_CPU_I386_REG_EBP]; + istate->esi = dbg_state.registers[DBG_CPU_I386_REG_ESI]; + istate->edi = dbg_state.registers[DBG_CPU_I386_REG_EDI]; + istate->eip = dbg_state.registers[DBG_CPU_I386_REG_PC]; + istate->cs = dbg_state.registers[DBG_CPU_I386_REG_CS]; + istate->eflags = dbg_state.registers[DBG_CPU_I386_REG_PS]; + istate->ss = dbg_state.registers[DBG_CPU_I386_REG_SS]; + istate->ds = dbg_state.registers[DBG_CPU_I386_REG_DS]; + istate->es = dbg_state.registers[DBG_CPU_I386_REG_ES]; + istate->fs = dbg_state.registers[DBG_CPU_I386_REG_FS]; + istate->gs = dbg_state.registers[DBG_CPU_I386_REG_GS]; +} + +/***************************************************************************** + * I/O Functions + ****************************************************************************/ + +/* + * Write to I/O port. + */ +void dbg_io_write_8(uint16_t port, uint8_t val) +{ + asm( + "outb %%al, %%dx;" + /* Outputs */ : /* None */ + /* Inputs */ : "a" (val), "d" (port) + /* Clobbers */ : /* None */ + ); +} + +/* + * Read from I/O port. + */ +uint8_t dbg_io_read_8(uint16_t port) +{ + uint8_t val; + + asm( + "inb %%dx, %%al;" + /* Outputs */ : "=a" (val) + /* Inputs */ : "d" (port) + /* Clobbers */ : /* None */ + ); + + return val; +} + +/***************************************************************************** + * NS16550 Serial Port (IO) + ****************************************************************************/ + +#define SERIAL_THR 0 +#define SERIAL_RBR 0 +#define SERIAL_LSR 5 + +int dbg_serial_getc(void) +{ + /* Wait for data */ + while ((dbg_io_read_8(SERIAL_PORT + SERIAL_LSR) & 1) == 0); + return dbg_io_read_8(SERIAL_PORT + SERIAL_RBR); +} + +int dbg_serial_putchar(int ch) +{ + /* Wait for THRE (bit 5) to be high */ + while ((dbg_io_read_8(SERIAL_PORT + SERIAL_LSR) & (1<<5)) == 0); + dbg_io_write_8(SERIAL_PORT + SERIAL_THR, ch); + return ch; +} + +/***************************************************************************** + * Debugging System Functions + ****************************************************************************/ + +/* + * Write one character to the debugging stream. + */ +int dbg_sys_putchar(int ch) +{ + return dbg_serial_putchar(ch); +} + +/* + * Read one character from the debugging stream. + */ +int dbg_sys_getc(void) +{ + return dbg_serial_getc() & 0xff; +} + +/* + * Read one byte from memory. + */ +int dbg_sys_mem_readb(address addr, char *val) +{ + *val = *(volatile char *)addr; + return 0; +} + +/* + * Write one byte to memory. + */ +int dbg_sys_mem_writeb(address addr, char val) +{ + *(volatile char *)addr = val; + return 0; +} + +/* + * Continue program execution. + */ +int dbg_sys_continue() +{ + dbg_state.registers[DBG_CPU_I386_REG_PS] &= ~(1<<8); + return 0; +} + +/* + * Single step the next instruction. + */ +int dbg_sys_step() +{ + dbg_state.registers[DBG_CPU_I386_REG_PS] |= 1<<8; + return 0; +} + +/* + * Debugger init function. + * + * Hooks the IDT to enable debugging. + */ +void dbg_start(void) +{ + /* Hook current IDT. */ + dbg_hook_idt(1, dbg_int_handlers[1]); + dbg_hook_idt(3, dbg_int_handlers[3]); + + /* Interrupt to start debugging. */ + asm("int3"); +} |
