Commit 22eeef4b authored by Jason Wessel's avatar Jason Wessel

kgdb,arm: Individual register get/set for arm

Implement the ability to individually get and set registers for kdb
and kgdb for arm.
Signed-off-by: default avatarJason Wessel <jason.wessel@windriver.com>
CC: Russell King <linux@arm.linux.org.uk>
CC: linux-arm-kernel@lists.infradead.org
parent 0896a9be
...@@ -70,11 +70,11 @@ extern int kgdb_fault_expected; ...@@ -70,11 +70,11 @@ extern int kgdb_fault_expected;
#define _GP_REGS 16 #define _GP_REGS 16
#define _FP_REGS 8 #define _FP_REGS 8
#define _EXTRA_REGS 2 #define _EXTRA_REGS 2
#define GDB_MAX_REGS (_GP_REGS + (_FP_REGS * 3) + _EXTRA_REGS) #define DBG_MAX_REG_NUM (_GP_REGS + (_FP_REGS * 3) + _EXTRA_REGS)
#define KGDB_MAX_NO_CPUS 1 #define KGDB_MAX_NO_CPUS 1
#define BUFMAX 400 #define BUFMAX 400
#define NUMREGBYTES (GDB_MAX_REGS << 2) #define NUMREGBYTES (DBG_MAX_REG_NUM << 2)
#define NUMCRITREGBYTES (32 << 2) #define NUMCRITREGBYTES (32 << 2)
#define _R0 0 #define _R0 0
...@@ -93,7 +93,7 @@ extern int kgdb_fault_expected; ...@@ -93,7 +93,7 @@ extern int kgdb_fault_expected;
#define _SPT 13 #define _SPT 13
#define _LR 14 #define _LR 14
#define _PC 15 #define _PC 15
#define _CPSR (GDB_MAX_REGS - 1) #define _CPSR (DBG_MAX_REG_NUM - 1)
/* /*
* So that we can denote the end of a frame for tracing, * So that we can denote the end of a frame for tracing,
......
...@@ -13,54 +13,58 @@ ...@@ -13,54 +13,58 @@
#include <linux/kgdb.h> #include <linux/kgdb.h>
#include <asm/traps.h> #include <asm/traps.h>
/* Make a local copy of the registers passed into the handler (bletch) */ struct dbg_reg_def_t dbg_reg_def[DBG_MAX_REG_NUM] =
void pt_regs_to_gdb_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs)
{ {
int regno; { "r0", 4, offsetof(struct pt_regs, ARM_r0)},
{ "r1", 4, offsetof(struct pt_regs, ARM_r1)},
/* Initialize all to zero. */ { "r2", 4, offsetof(struct pt_regs, ARM_r2)},
for (regno = 0; regno < GDB_MAX_REGS; regno++) { "r3", 4, offsetof(struct pt_regs, ARM_r3)},
gdb_regs[regno] = 0; { "r4", 4, offsetof(struct pt_regs, ARM_r4)},
{ "r5", 4, offsetof(struct pt_regs, ARM_r5)},
{ "r6", 4, offsetof(struct pt_regs, ARM_r6)},
{ "r7", 4, offsetof(struct pt_regs, ARM_r7)},
{ "r8", 4, offsetof(struct pt_regs, ARM_r8)},
{ "r9", 4, offsetof(struct pt_regs, ARM_r9)},
{ "r10", 4, offsetof(struct pt_regs, ARM_r10)},
{ "fp", 4, offsetof(struct pt_regs, ARM_fp)},
{ "ip", 4, offsetof(struct pt_regs, ARM_ip)},
{ "sp", 4, offsetof(struct pt_regs, ARM_sp)},
{ "lr", 4, offsetof(struct pt_regs, ARM_lr)},
{ "pc", 4, offsetof(struct pt_regs, ARM_pc)},
{ "f0", 12, -1 },
{ "f1", 12, -1 },
{ "f2", 12, -1 },
{ "f3", 12, -1 },
{ "f4", 12, -1 },
{ "f5", 12, -1 },
{ "f6", 12, -1 },
{ "f7", 12, -1 },
{ "fps", 4, -1 },
{ "cpsr", 4, offsetof(struct pt_regs, ARM_cpsr)},
};
gdb_regs[_R0] = kernel_regs->ARM_r0; char *dbg_get_reg(int regno, void *mem, struct pt_regs *regs)
gdb_regs[_R1] = kernel_regs->ARM_r1; {
gdb_regs[_R2] = kernel_regs->ARM_r2; if (regno >= DBG_MAX_REG_NUM || regno < 0)
gdb_regs[_R3] = kernel_regs->ARM_r3; return NULL;
gdb_regs[_R4] = kernel_regs->ARM_r4;
gdb_regs[_R5] = kernel_regs->ARM_r5; if (dbg_reg_def[regno].offset != -1)
gdb_regs[_R6] = kernel_regs->ARM_r6; memcpy(mem, (void *)regs + dbg_reg_def[regno].offset,
gdb_regs[_R7] = kernel_regs->ARM_r7; dbg_reg_def[regno].size);
gdb_regs[_R8] = kernel_regs->ARM_r8; else
gdb_regs[_R9] = kernel_regs->ARM_r9; memset(mem, 0, dbg_reg_def[regno].size);
gdb_regs[_R10] = kernel_regs->ARM_r10; return dbg_reg_def[regno].name;
gdb_regs[_FP] = kernel_regs->ARM_fp;
gdb_regs[_IP] = kernel_regs->ARM_ip;
gdb_regs[_SPT] = kernel_regs->ARM_sp;
gdb_regs[_LR] = kernel_regs->ARM_lr;
gdb_regs[_PC] = kernel_regs->ARM_pc;
gdb_regs[_CPSR] = kernel_regs->ARM_cpsr;
} }
/* Copy local gdb registers back to kgdb regs, for later copy to kernel */ int dbg_set_reg(int regno, void *mem, struct pt_regs *regs)
void gdb_regs_to_pt_regs(unsigned long *gdb_regs, struct pt_regs *kernel_regs)
{ {
kernel_regs->ARM_r0 = gdb_regs[_R0]; if (regno >= DBG_MAX_REG_NUM || regno < 0)
kernel_regs->ARM_r1 = gdb_regs[_R1]; return -EINVAL;
kernel_regs->ARM_r2 = gdb_regs[_R2];
kernel_regs->ARM_r3 = gdb_regs[_R3]; if (dbg_reg_def[regno].offset != -1)
kernel_regs->ARM_r4 = gdb_regs[_R4]; memcpy((void *)regs + dbg_reg_def[regno].offset, mem,
kernel_regs->ARM_r5 = gdb_regs[_R5]; dbg_reg_def[regno].size);
kernel_regs->ARM_r6 = gdb_regs[_R6]; return 0;
kernel_regs->ARM_r7 = gdb_regs[_R7];
kernel_regs->ARM_r8 = gdb_regs[_R8];
kernel_regs->ARM_r9 = gdb_regs[_R9];
kernel_regs->ARM_r10 = gdb_regs[_R10];
kernel_regs->ARM_fp = gdb_regs[_FP];
kernel_regs->ARM_ip = gdb_regs[_IP];
kernel_regs->ARM_sp = gdb_regs[_SPT];
kernel_regs->ARM_lr = gdb_regs[_LR];
kernel_regs->ARM_pc = gdb_regs[_PC];
kernel_regs->ARM_cpsr = gdb_regs[_CPSR];
} }
void void
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment