#include "defs.h"
#include "gdbtypes.h"
#include "gdbcore.h"
#include "regcache.h"
#include "arch-utils.h"
#include "i386-tdep.h"
#include "i387-tdep.h"
#include "nbsd-tdep.h"
#include "solib-svr4.h"
static int regmap[] =
{
( 0 * 4),
( 1 * 4),
( 2 * 4),
( 3 * 4),
( 4 * 4),
( 5 * 4),
( 6 * 4),
( 7 * 4),
( 8 * 4),
( 9 * 4),
(10 * 4),
(11 * 4),
(12 * 4),
(13 * 4),
(14 * 4),
(15 * 4),
};
#define SIZEOF_STRUCT_REG (16 * 4)
static void
i386nbsd_supply_reg (char *regs, int regno)
{
int i;
for (i = 0; i <= 15; i++)
if (regno == i || regno == -1)
supply_register (i, regs + regmap[i]);
}
static void
fetch_core_registers (char *core_reg_sect, unsigned core_reg_size, int which,
CORE_ADDR ignore)
{
char *regs, *fsave;
if (which != 0)
return;
if (core_reg_size < (SIZEOF_STRUCT_REG + 108))
{
warning ("Wrong size register set in core file.");
return;
}
regs = core_reg_sect;
fsave = core_reg_sect + SIZEOF_STRUCT_REG;
i386nbsd_supply_reg (regs, -1);
i387_supply_fsave (fsave);
}
static void
fetch_elfcore_registers (char *core_reg_sect, unsigned core_reg_size,
int which, CORE_ADDR ignore)
{
switch (which)
{
case 0:
if (core_reg_size != SIZEOF_STRUCT_REG)
warning ("Wrong size register set in core file.");
else
i386nbsd_supply_reg (core_reg_sect, -1);
break;
case 2:
if (core_reg_size != 108)
warning ("Wrong size FP register set in core file.");
else
i387_supply_fsave (core_reg_sect);
break;
case 3:
if (core_reg_size != 512)
warning ("Wrong size XMM register set in core file.");
else
i387_supply_fxsave (core_reg_sect);
break;
default:
break;
}
}
static struct core_fns i386nbsd_core_fns =
{
bfd_target_unknown_flavour,
default_check_format,
default_core_sniffer,
fetch_core_registers,
NULL
};
static struct core_fns i386nbsd_elfcore_fns =
{
bfd_target_elf_flavour,
default_check_format,
default_core_sniffer,
fetch_elfcore_registers,
NULL
};
#define RETCODE_INSN1 0x8d
#define RETCODE_INSN2 0x50
#define RETCODE_INSN3 0x50
#define RETCODE_INSN4 0xb8
#define RETCODE_INSN5 0xcd
#define RETCODE_INSN2_OFF 4
#define RETCODE_INSN3_OFF 5
#define RETCODE_INSN4_OFF 6
#define RETCODE_INSN5_OFF 11
static const unsigned char sigtramp_retcode[] =
{
RETCODE_INSN1, 0x44, 0x24, 0x10,
RETCODE_INSN2,
RETCODE_INSN3,
RETCODE_INSN4, 0x27, 0x01, 0x00, 0x00,
RETCODE_INSN5, 0x80,
};
static LONGEST
i386nbsd_sigtramp_offset (CORE_ADDR pc)
{
unsigned char ret[sizeof(sigtramp_retcode)], insn;
LONGEST off;
int i;
if (read_memory_nobpt (pc, &insn, 1) != 0)
return -1;
switch (insn)
{
case RETCODE_INSN1:
off = 0;
break;
case RETCODE_INSN2:
if (read_memory_nobpt (pc + 1, &insn, 1) != 0)
return -1;
if (insn == RETCODE_INSN3)
off = RETCODE_INSN2_OFF;
else
off = RETCODE_INSN3_OFF;
break;
case RETCODE_INSN4:
off = RETCODE_INSN4_OFF;
break;
case RETCODE_INSN5:
off = RETCODE_INSN5_OFF;
break;
default:
return -1;
}
pc -= off;
if (read_memory_nobpt (pc, (char *) ret, sizeof (ret)) != 0)
return -1;
if (memcmp (ret, sigtramp_retcode, sizeof (ret)) == 0)
return off;
return -1;
}
static int
i386nbsd_pc_in_sigtramp (CORE_ADDR pc, char *name)
{
return (nbsd_pc_in_sigtramp (pc, name)
|| i386nbsd_sigtramp_offset (pc) >= 0);
}
int i386nbsd_sc_pc_offset = 44;
int i386nbsd_sc_sp_offset = 56;
static void
i386nbsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
{
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
i386bsd_init_abi (info, gdbarch);
set_gdbarch_pc_in_sigtramp (gdbarch, i386nbsd_pc_in_sigtramp);
set_gdbarch_sigtramp_start (gdbarch, NULL);
set_gdbarch_sigtramp_end (gdbarch, NULL);
tdep->struct_return = reg_struct_return;
tdep->sc_pc_offset = i386nbsd_sc_pc_offset;
tdep->sc_sp_offset = i386nbsd_sc_sp_offset;
}
static void
i386nbsdelf_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
{
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
i386nbsd_init_abi (info, gdbarch);
i386_elf_init_abi (info, gdbarch);
set_gdbarch_in_solib_call_trampoline (gdbarch,
generic_in_solib_call_trampoline);
set_solib_svr4_fetch_link_map_offsets (gdbarch,
nbsd_ilp32_solib_svr4_fetch_link_map_offsets);
tdep->struct_return = pcc_struct_return;
tdep->num_xmm_regs = I386_NUM_XREGS - 1;
set_gdbarch_num_regs (gdbarch, I386_NUM_GREGS + I386_NUM_FREGS
+ I386_NUM_XREGS);
}
void
_initialize_i386nbsd_tdep (void)
{
add_core_fns (&i386nbsd_core_fns);
add_core_fns (&i386nbsd_elfcore_fns);
gdbarch_register_osabi (bfd_arch_i386, GDB_OSABI_NETBSD_AOUT,
i386nbsd_init_abi);
gdbarch_register_osabi (bfd_arch_i386, GDB_OSABI_NETBSD_ELF,
i386nbsdelf_init_abi);
}