Index: gdb/gdbserver/linux-ppc-low.c |
diff --git a/gdb/gdbserver/linux-ppc-low.c b/gdb/gdbserver/linux-ppc-low.c |
index ed254fac067c4dcaf60f89dd8c08edaf2f204945..aed17491b55eff87e3c505174649c848f038a2c9 100644 |
--- a/gdb/gdbserver/linux-ppc-low.c |
+++ b/gdb/gdbserver/linux-ppc-low.c |
@@ -1,7 +1,6 @@ |
/* GNU/Linux/PowerPC specific low level interface, for the remote server for |
GDB. |
- Copyright (C) 1995-1996, 1998-2002, 2005, 2007-2012 Free Software |
- Foundation, Inc. |
+ Copyright (C) 1995-2013 Free Software Foundation, Inc. |
This file is part of GDB. |
@@ -36,34 +35,63 @@ static unsigned long ppc_hwcap; |
/* Defined in auto-generated file powerpc-32l.c. */ |
void init_registers_powerpc_32l (void); |
+extern const struct target_desc *tdesc_powerpc_32l; |
+ |
/* Defined in auto-generated file powerpc-altivec32l.c. */ |
void init_registers_powerpc_altivec32l (void); |
+extern const struct target_desc *tdesc_powerpc_altivec32l; |
+ |
/* Defined in auto-generated file powerpc-cell32l.c. */ |
void init_registers_powerpc_cell32l (void); |
+extern const struct target_desc *tdesc_powerpc_cell32l; |
+ |
/* Defined in auto-generated file powerpc-vsx32l.c. */ |
void init_registers_powerpc_vsx32l (void); |
+extern const struct target_desc *tdesc_powerpc_vsx32l; |
+ |
/* Defined in auto-generated file powerpc-isa205-32l.c. */ |
void init_registers_powerpc_isa205_32l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_32l; |
+ |
/* Defined in auto-generated file powerpc-isa205-altivec32l.c. */ |
void init_registers_powerpc_isa205_altivec32l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_altivec32l; |
+ |
/* Defined in auto-generated file powerpc-isa205-vsx32l.c. */ |
void init_registers_powerpc_isa205_vsx32l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_vsx32l; |
+ |
/* Defined in auto-generated file powerpc-e500l.c. */ |
void init_registers_powerpc_e500l (void); |
+extern const struct target_desc *tdesc_powerpc_e500l; |
+ |
/* Defined in auto-generated file powerpc-64l.c. */ |
void init_registers_powerpc_64l (void); |
+extern const struct target_desc *tdesc_powerpc_64l; |
+ |
/* Defined in auto-generated file powerpc-altivec64l.c. */ |
void init_registers_powerpc_altivec64l (void); |
+extern const struct target_desc *tdesc_powerpc_altivec64l; |
+ |
/* Defined in auto-generated file powerpc-cell64l.c. */ |
void init_registers_powerpc_cell64l (void); |
+extern const struct target_desc *tdesc_powerpc_cell64l; |
+ |
/* Defined in auto-generated file powerpc-vsx64l.c. */ |
void init_registers_powerpc_vsx64l (void); |
+extern const struct target_desc *tdesc_powerpc_vsx64l; |
+ |
/* Defined in auto-generated file powerpc-isa205-64l.c. */ |
void init_registers_powerpc_isa205_64l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_64l; |
+ |
/* Defined in auto-generated file powerpc-isa205-altivec64l.c. */ |
void init_registers_powerpc_isa205_altivec64l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_altivec64l; |
+ |
/* Defined in auto-generated file powerpc-isa205-vsx64l.c. */ |
void init_registers_powerpc_isa205_vsx64l (void); |
+extern const struct target_desc *tdesc_powerpc_isa205_vsx64l; |
#define ppc_num_regs 73 |
@@ -148,15 +176,18 @@ static int ppc_regmap_e500[] = |
static int |
ppc_cannot_store_register (int regno) |
{ |
+ const struct target_desc *tdesc = current_process ()->tdesc; |
+ |
#ifndef __powerpc64__ |
/* Some kernels do not allow us to store fpscr. */ |
- if (!(ppc_hwcap & PPC_FEATURE_HAS_SPE) && regno == find_regno ("fpscr")) |
+ if (!(ppc_hwcap & PPC_FEATURE_HAS_SPE) |
+ && regno == find_regno (tdesc, "fpscr")) |
return 2; |
#endif |
/* Some kernels do not allow us to store orig_r3 or trap. */ |
- if (regno == find_regno ("orig_r3") |
- || regno == find_regno ("trap")) |
+ if (regno == find_regno (tdesc, "orig_r3") |
+ || regno == find_regno (tdesc, "trap")) |
return 2; |
return 0; |
@@ -171,7 +202,7 @@ ppc_cannot_fetch_register (int regno) |
static void |
ppc_collect_ptrace_register (struct regcache *regcache, int regno, char *buf) |
{ |
- int size = register_size (regno); |
+ int size = register_size (regcache->tdesc, regno); |
memset (buf, 0, sizeof (long)); |
@@ -185,7 +216,7 @@ static void |
ppc_supply_ptrace_register (struct regcache *regcache, |
int regno, const char *buf) |
{ |
- int size = register_size (regno); |
+ int size = register_size (regcache->tdesc, regno); |
if (size < sizeof (long)) |
supply_register (regcache, regno, buf + sizeof (long) - size); |
else |
@@ -206,7 +237,7 @@ parse_spufs_run (struct regcache *regcache, int *fd, CORE_ADDR *addr) |
int curr_insn; |
int curr_r0; |
- if (register_size (0) == 4) |
+ if (register_size (regcache->tdesc, 0) == 4) |
{ |
unsigned int pc, r0, r3, r4; |
collect_register_by_name (regcache, "pc", &pc); |
@@ -258,7 +289,7 @@ ppc_get_pc (struct regcache *regcache) |
return ((CORE_ADDR)1 << 63) |
| ((CORE_ADDR)fd << 32) | (CORE_ADDR) (pc - 4); |
} |
- else if (register_size (0) == 4) |
+ else if (register_size (regcache->tdesc, 0) == 4) |
{ |
unsigned int pc; |
collect_register_by_name (regcache, "pc", &pc); |
@@ -283,7 +314,7 @@ ppc_set_pc (struct regcache *regcache, CORE_ADDR pc) |
unsigned int newpc = pc; |
(*the_target->write_memory) (addr, (unsigned char *) &newpc, 4); |
} |
- else if (register_size (0) == 4) |
+ else if (register_size (regcache->tdesc, 0) == 4) |
{ |
unsigned int newpc = pc; |
supply_register_by_name (regcache, "pc", &newpc); |
@@ -299,7 +330,8 @@ ppc_set_pc (struct regcache *regcache, CORE_ADDR pc) |
static int |
ppc_get_hwcap (unsigned long *valp) |
{ |
- int wordsize = register_size (0); |
+ const struct target_desc *tdesc = current_process ()->tdesc; |
+ int wordsize = register_size (tdesc, 0); |
unsigned char *data = alloca (2 * wordsize); |
int offset = 0; |
@@ -331,9 +363,16 @@ ppc_get_hwcap (unsigned long *valp) |
return 0; |
} |
+/* Forward declaration. */ |
+static struct usrregs_info ppc_usrregs_info; |
+#ifndef __powerpc64__ |
+static int ppc_regmap_adjusted; |
+#endif |
+ |
static void |
ppc_arch_setup (void) |
{ |
+ const struct target_desc *tdesc; |
#ifdef __powerpc64__ |
long msr; |
struct regcache *regcache; |
@@ -341,20 +380,21 @@ ppc_arch_setup (void) |
/* On a 64-bit host, assume 64-bit inferior process with no |
AltiVec registers. Reset ppc_hwcap to ensure that the |
collect_register call below does not fail. */ |
- init_registers_powerpc_64l (); |
+ tdesc = tdesc_powerpc_64l; |
+ current_process ()->tdesc = tdesc; |
ppc_hwcap = 0; |
/* Only if the high bit of the MSR is set, we actually have |
a 64-bit inferior. */ |
- regcache = new_register_cache (); |
- fetch_inferior_registers (regcache, find_regno ("msr")); |
+ regcache = new_register_cache (tdesc); |
+ fetch_inferior_registers (regcache, find_regno (tdesc, "msr")); |
collect_register_by_name (regcache, "msr", &msr); |
free_register_cache (regcache); |
if (msr < 0) |
{ |
ppc_get_hwcap (&ppc_hwcap); |
if (ppc_hwcap & PPC_FEATURE_CELL) |
- init_registers_powerpc_cell64l (); |
+ tdesc = tdesc_powerpc_cell64l; |
else if (ppc_hwcap & PPC_FEATURE_HAS_VSX) |
{ |
/* Power ISA 2.05 (implemented by Power 6 and newer processors) |
@@ -365,59 +405,67 @@ ppc_arch_setup (void) |
Point, we check if that feature is available to decide the size |
of the FPSCR. */ |
if (ppc_hwcap & PPC_FEATURE_HAS_DFP) |
- init_registers_powerpc_isa205_vsx64l (); |
+ tdesc = tdesc_powerpc_isa205_vsx64l; |
else |
- init_registers_powerpc_vsx64l (); |
+ tdesc = tdesc_powerpc_vsx64l; |
} |
else if (ppc_hwcap & PPC_FEATURE_HAS_ALTIVEC) |
{ |
if (ppc_hwcap & PPC_FEATURE_HAS_DFP) |
- init_registers_powerpc_isa205_altivec64l (); |
+ tdesc = tdesc_powerpc_isa205_altivec64l; |
else |
- init_registers_powerpc_altivec64l (); |
+ tdesc = tdesc_powerpc_altivec64l; |
} |
+ current_process ()->tdesc = tdesc; |
return; |
} |
#endif |
/* OK, we have a 32-bit inferior. */ |
- init_registers_powerpc_32l (); |
+ tdesc = tdesc_powerpc_32l; |
+ current_process ()->tdesc = tdesc; |
ppc_get_hwcap (&ppc_hwcap); |
if (ppc_hwcap & PPC_FEATURE_CELL) |
- init_registers_powerpc_cell32l (); |
+ tdesc = tdesc_powerpc_cell32l; |
else if (ppc_hwcap & PPC_FEATURE_HAS_VSX) |
{ |
if (ppc_hwcap & PPC_FEATURE_HAS_DFP) |
- init_registers_powerpc_isa205_vsx32l (); |
+ tdesc = tdesc_powerpc_isa205_vsx32l; |
else |
- init_registers_powerpc_vsx32l (); |
+ tdesc = tdesc_powerpc_vsx32l; |
} |
else if (ppc_hwcap & PPC_FEATURE_HAS_ALTIVEC) |
{ |
if (ppc_hwcap & PPC_FEATURE_HAS_DFP) |
- init_registers_powerpc_isa205_altivec32l (); |
+ tdesc = tdesc_powerpc_isa205_altivec32l; |
else |
- init_registers_powerpc_altivec32l (); |
+ tdesc = tdesc_powerpc_altivec32l; |
} |
/* On 32-bit machines, check for SPE registers. |
Set the low target's regmap field as appropriately. */ |
#ifndef __powerpc64__ |
- the_low_target.regmap = ppc_regmap; |
if (ppc_hwcap & PPC_FEATURE_HAS_SPE) |
+ tdesc = tdesc_powerpc_e500l; |
+ |
+ if (!ppc_regmap_adjusted) |
{ |
- init_registers_powerpc_e500l (); |
- the_low_target.regmap = ppc_regmap_e500; |
- } |
+ if (ppc_hwcap & PPC_FEATURE_HAS_SPE) |
+ ppc_usrregs_info.regmap = ppc_regmap_e500; |
- /* If the FPSCR is 64-bit wide, we need to fetch the whole 64-bit |
- slot and not just its second word. The PT_FPSCR supplied in a |
- 32-bit GDB compilation doesn't reflect this. */ |
- if (register_size (70) == 8) |
- ppc_regmap[70] = (48 + 2*32) * sizeof (long); |
+ /* If the FPSCR is 64-bit wide, we need to fetch the whole |
+ 64-bit slot and not just its second word. The PT_FPSCR |
+ supplied in a 32-bit GDB compilation doesn't reflect |
+ this. */ |
+ if (register_size (tdesc, 70) == 8) |
+ ppc_regmap[70] = (48 + 2*32) * sizeof (long); |
+ |
+ ppc_regmap_adjusted = 1; |
+ } |
#endif |
+ current_process ()->tdesc = tdesc; |
} |
/* Correct in either endianness. |
@@ -485,7 +533,7 @@ ppc_fill_vsxregset (struct regcache *regcache, void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_VSX)) |
return; |
- base = find_regno ("vs0h"); |
+ base = find_regno (regcache->tdesc, "vs0h"); |
for (i = 0; i < 32; i++) |
collect_register (regcache, base + i, ®set[i * 8]); |
} |
@@ -499,7 +547,7 @@ ppc_store_vsxregset (struct regcache *regcache, const void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_VSX)) |
return; |
- base = find_regno ("vs0h"); |
+ base = find_regno (regcache->tdesc, "vs0h"); |
for (i = 0; i < 32; i++) |
supply_register (regcache, base + i, ®set[i * 8]); |
} |
@@ -520,7 +568,7 @@ ppc_fill_vrregset (struct regcache *regcache, void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_ALTIVEC)) |
return; |
- base = find_regno ("vr0"); |
+ base = find_regno (regcache->tdesc, "vr0"); |
for (i = 0; i < 32; i++) |
collect_register (regcache, base + i, ®set[i * 16]); |
@@ -537,7 +585,7 @@ ppc_store_vrregset (struct regcache *regcache, const void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_ALTIVEC)) |
return; |
- base = find_regno ("vr0"); |
+ base = find_regno (regcache->tdesc, "vr0"); |
for (i = 0; i < 32; i++) |
supply_register (regcache, base + i, ®set[i * 16]); |
@@ -566,7 +614,7 @@ ppc_fill_evrregset (struct regcache *regcache, void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_SPE)) |
return; |
- ev0 = find_regno ("ev0h"); |
+ ev0 = find_regno (regcache->tdesc, "ev0h"); |
for (i = 0; i < 32; i++) |
collect_register (regcache, ev0 + i, ®set->evr[i]); |
@@ -583,7 +631,7 @@ ppc_store_evrregset (struct regcache *regcache, const void *buf) |
if (!(ppc_hwcap & PPC_FEATURE_HAS_SPE)) |
return; |
- ev0 = find_regno ("ev0h"); |
+ ev0 = find_regno (regcache->tdesc, "ev0h"); |
for (i = 0; i < 32; i++) |
supply_register (regcache, ev0 + i, ®set->evr[i]); |
@@ -591,7 +639,7 @@ ppc_store_evrregset (struct regcache *regcache, const void *buf) |
supply_register_by_name (regcache, "spefscr", ®set->spefscr); |
} |
-struct regset_info target_regsets[] = { |
+static struct regset_info ppc_regsets[] = { |
/* List the extra register sets before GENERAL_REGS. That way we will |
fetch them every time, but still fall back to PTRACE_PEEKUSER for the |
general registers. Some kernels support these, but not the newer |
@@ -606,11 +654,35 @@ struct regset_info target_regsets[] = { |
{ 0, 0, 0, -1, -1, NULL, NULL } |
}; |
+static struct usrregs_info ppc_usrregs_info = |
+ { |
+ ppc_num_regs, |
+ ppc_regmap, |
+ }; |
+ |
+static struct regsets_info ppc_regsets_info = |
+ { |
+ ppc_regsets, /* regsets */ |
+ 0, /* num_regsets */ |
+ NULL, /* disabled_regsets */ |
+ }; |
+ |
+static struct regs_info regs_info = |
+ { |
+ NULL, /* regset_bitmap */ |
+ &ppc_usrregs_info, |
+ &ppc_regsets_info |
+ }; |
+ |
+static const struct regs_info * |
+ppc_regs_info (void) |
+{ |
+ return ®s_info; |
+} |
+ |
struct linux_target_ops the_low_target = { |
ppc_arch_setup, |
- ppc_num_regs, |
- ppc_regmap, |
- NULL, |
+ ppc_regs_info, |
ppc_cannot_fetch_register, |
ppc_cannot_store_register, |
NULL, /* fetch_register */ |
@@ -628,3 +700,27 @@ struct linux_target_ops the_low_target = { |
ppc_collect_ptrace_register, |
ppc_supply_ptrace_register, |
}; |
+ |
+void |
+initialize_low_arch (void) |
+{ |
+ /* Initialize the Linux target descriptions. */ |
+ |
+ init_registers_powerpc_32l (); |
+ init_registers_powerpc_altivec32l (); |
+ init_registers_powerpc_cell32l (); |
+ init_registers_powerpc_vsx32l (); |
+ init_registers_powerpc_isa205_32l (); |
+ init_registers_powerpc_isa205_altivec32l (); |
+ init_registers_powerpc_isa205_vsx32l (); |
+ init_registers_powerpc_e500l (); |
+ init_registers_powerpc_64l (); |
+ init_registers_powerpc_altivec64l (); |
+ init_registers_powerpc_cell64l (); |
+ init_registers_powerpc_vsx64l (); |
+ init_registers_powerpc_isa205_64l (); |
+ init_registers_powerpc_isa205_altivec64l (); |
+ init_registers_powerpc_isa205_vsx64l (); |
+ |
+ initialize_regsets_info (&ppc_regsets_info); |
+} |