1999-05-03 14:27:23 +04:00
|
|
|
|
/* NetBSD/powerpc native-dependent code for GDB, the GNU debugger.
|
|
|
|
|
Copyright 1986, 1987, 1989, 1991, 1992, 1994, 1995, 1996, 1997
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
|
|
This file is part of GDB.
|
|
|
|
|
|
|
|
|
|
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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
|
|
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
|
#include "inferior.h"
|
|
|
|
|
#include "target.h"
|
|
|
|
|
#include "gdbcore.h"
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
|
#include <sys/ptrace.h>
|
|
|
|
|
#include <machine/frame.h>
|
|
|
|
|
#include <machine/reg.h>
|
|
|
|
|
#include <machine/pcb.h>
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fetch_core_registers PARAMS ((char *, unsigned int, int, CORE_ADDR));
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
fetch_inferior_registers (regno)
|
|
|
|
|
int regno;
|
|
|
|
|
{
|
|
|
|
|
struct reg infreg;
|
2000-06-04 13:55:53 +04:00
|
|
|
|
struct fpreg inffpreg;
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
/* Integer registers */
|
|
|
|
|
ptrace(PT_GETREGS, inferior_pid, (PTRACE_ARG3_TYPE) &infreg, 0);
|
|
|
|
|
|
|
|
|
|
memcpy(®isters[REGISTER_BYTE (GP0_REGNUM)], infreg.fixreg,
|
|
|
|
|
sizeof(infreg.fixreg));
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (PC_REGNUM)] = infreg.pc;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (PS_REGNUM)] = 0; /* XXX */
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (CR_REGNUM)] = infreg.cr;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (LR_REGNUM)] = infreg.lr;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (CTR_REGNUM)] = infreg.ctr;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (XER_REGNUM)] = infreg.xer;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (MQ_REGNUM)] = 0;
|
|
|
|
|
|
|
|
|
|
/* Floating point registers */
|
2000-06-04 13:55:53 +04:00
|
|
|
|
ptrace(PT_GETFPREGS, inferior_pid, (PTRACE_ARG3_TYPE) &inffpreg, 0);
|
|
|
|
|
|
|
|
|
|
memcpy(®isters[REGISTER_BYTE (FP0_REGNUM)], inffpreg.fpreg,
|
|
|
|
|
sizeof(inffpreg.fpreg));
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
registers_fetched ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
store_inferior_registers (regno)
|
|
|
|
|
int regno;
|
|
|
|
|
{
|
|
|
|
|
struct reg infreg;
|
2000-06-04 13:55:53 +04:00
|
|
|
|
struct fpreg inffpreg;
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
/* Integer registers */
|
|
|
|
|
memcpy(infreg.fixreg, ®isters[REGISTER_BYTE (GP0_REGNUM)],
|
|
|
|
|
sizeof(infreg.fixreg));
|
|
|
|
|
infreg.pc = *(long *) ®isters[REGISTER_BYTE (PC_REGNUM)];
|
|
|
|
|
infreg.cr = *(long *) ®isters[REGISTER_BYTE (CR_REGNUM)];
|
|
|
|
|
infreg.lr = *(long *) ®isters[REGISTER_BYTE (LR_REGNUM)];
|
|
|
|
|
infreg.ctr = *(long *) ®isters[REGISTER_BYTE (CTR_REGNUM)];
|
|
|
|
|
infreg.xer = *(long *) ®isters[REGISTER_BYTE (XER_REGNUM)];
|
|
|
|
|
|
|
|
|
|
ptrace(PT_SETREGS, inferior_pid, (PTRACE_ARG3_TYPE) &infreg, 0);
|
|
|
|
|
|
|
|
|
|
/* Floating point registers */
|
2000-06-04 13:55:53 +04:00
|
|
|
|
memset(&inffpreg, 0, sizeof(inffpreg));
|
|
|
|
|
memcpy(inffpreg.fpreg, ®isters[REGISTER_BYTE (FP0_REGNUM)],
|
|
|
|
|
sizeof(inffpreg.fpreg));
|
|
|
|
|
|
|
|
|
|
ptrace(PT_SETFPREGS, inferior_pid, (PTRACE_ARG3_TYPE) &inffpreg, 0);
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
registers_fetched ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fetch_core_registers (core_reg_sect, core_reg_size, which, reg_addr)
|
|
|
|
|
char *core_reg_sect;
|
|
|
|
|
unsigned core_reg_size;
|
|
|
|
|
int which;
|
|
|
|
|
CORE_ADDR reg_addr; /* Unused in this version */
|
|
|
|
|
{
|
|
|
|
|
struct md_coredump *core_reg;
|
|
|
|
|
struct trapframe *tf;
|
2000-06-04 16:06:43 +04:00
|
|
|
|
struct fpu *fs;
|
1999-05-03 14:27:23 +04:00
|
|
|
|
register int regnum;
|
|
|
|
|
|
|
|
|
|
/* We get everything from the .reg section. */
|
|
|
|
|
if (which != 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
core_reg = (struct md_coredump *)core_reg_sect;
|
|
|
|
|
tf = &core_reg->frame;
|
2000-06-04 16:06:43 +04:00
|
|
|
|
fs = &core_reg->fpstate;
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
if (core_reg_size < sizeof(*core_reg)) {
|
|
|
|
|
fprintf_unfiltered (gdb_stderr, "Couldn't read regs from core file\n");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Integer registers */
|
|
|
|
|
for (regnum = 0; regnum < 32; regnum++)
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (regnum)] = tf->fixreg[regnum];
|
|
|
|
|
|
|
|
|
|
/* Floating point registers */
|
2000-06-04 16:06:43 +04:00
|
|
|
|
memcpy (®isters[REGISTER_BYTE (FP0_REGNUM)], fs->fpr, sizeof(fs->fpr));
|
1999-05-03 14:27:23 +04:00
|
|
|
|
|
|
|
|
|
/* Special registers (PC, LR) */
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (PC_REGNUM)] = tf->srr0;
|
|
|
|
|
*(long *) ®isters[REGISTER_BYTE (LR_REGNUM)] = tf->lr;
|
|
|
|
|
|
|
|
|
|
registers_fetched ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Register that we are able to handle rs6000 core file formats. */
|
|
|
|
|
|
|
|
|
|
static struct core_fns ppcnbsd_core_fns =
|
|
|
|
|
{
|
|
|
|
|
bfd_target_elf_flavour,
|
|
|
|
|
fetch_core_registers,
|
|
|
|
|
NULL
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
_initialize_core_ppcnbsd ()
|
|
|
|
|
{
|
|
|
|
|
add_core_fns (&ppcnbsd_core_fns);
|
|
|
|
|
}
|