aboutsummaryrefslogtreecommitdiff
path: root/gdb/rs6000-xdep.c
diff options
context:
space:
mode:
Diffstat (limited to 'gdb/rs6000-xdep.c')
-rw-r--r--gdb/rs6000-xdep.c98
1 files changed, 60 insertions, 38 deletions
diff --git a/gdb/rs6000-xdep.c b/gdb/rs6000-xdep.c
index afb24c6..2dd170a 100644
--- a/gdb/rs6000-xdep.c
+++ b/gdb/rs6000-xdep.c
@@ -61,6 +61,7 @@ static int special_regs[] = {
extern int one_stepped;
extern char register_valid[];
+extern struct obstack frame_cache_obstack;
void
@@ -242,20 +243,66 @@ struct frame_info *fi;
}
+/* If saved registers of frame FI are not known yet, read and cache them.
+ &FDATAP contains aix_framedata; TDATAP can be NULL,
+ in which case the framedata are read.
+ */
+
+static void
+frame_get_cache_fsr (fi, fdatap)
+ struct frame_info *fi;
+ struct aix_framedata *fdatap;
+{
+ int ii;
+ CORE_ADDR frame_addr;
+ struct aix_framedata work_fdata;
+ if (fi->cache_fsr)
+ return;
+
+ if (fdatap = NULL) {
+ fdatap = &work_fdata;
+ function_frame_info (get_pc_function_start (fi->pc), fdatap);
+ }
+
+ fi->cache_fsr = (struct frame_saved_regs *)
+ obstack_alloc (&frame_cache_obstack, sizeof (struct frame_saved_regs));
+ bzero (fi->cache_fsr, sizeof (struct frame_saved_regs));
+
+ if (fi->prev && fi->prev->frame)
+ frame_addr = fi->prev->frame;
+ else
+ frame_addr = read_memory_integer (fi->frame, 4);
+
+ /* if != -1, fdatap->saved_fpr is the smallest number of saved_fpr.
+ All fpr's from saved_fpr to fp31 are saved right underneath caller
+ stack pointer, starting from fp31 first. */
+
+ if (fdatap->saved_fpr >= 0) {
+ for (ii=31; ii >= fdatap->saved_fpr; --ii)
+ fi->cache_fsr->regs [FP0_REGNUM + ii] = frame_addr - ((32 - ii) * 8);
+ frame_addr -= (32 - fdatap->saved_fpr) * 8;
+ }
+
+ /* if != -1, fdatap->saved_gpr is the smallest number of saved_gpr.
+ All gpr's from saved_gpr to gpr31 are saved right under saved fprs,
+ starting from r31 first. */
+
+ if (fdatap->saved_gpr >= 0)
+ for (ii=31; ii >= fdatap->saved_gpr; --ii)
+ fi->cache_fsr->regs [ii] = frame_addr - ((32 - ii) * 4);
+}
+
/* Return the address of a frame. This is the inital %sp value when the frame
was first allocated. For functions calling alloca(), it might be saved in
an alloca register. */
CORE_ADDR
frame_initial_stack_address (fi)
-struct frame_info *fi;
+ struct frame_info *fi;
{
- CORE_ADDR frame_addr, tmpaddr;
+ CORE_ADDR tmpaddr;
struct aix_framedata fdata;
struct frame_info *callee_fi;
- int ii;
-
- extern struct obstack frame_cache_obstack;
/* if the initial stack pointer (frame address) of this frame is known,
just return it. */
@@ -265,39 +312,12 @@ struct frame_info *fi;
/* find out if this function is using an alloca register.. */
- tmpaddr = get_pc_function_start (fi->pc);
- function_frame_info (tmpaddr, &fdata);
+ function_frame_info (get_pc_function_start (fi->pc), &fdata);
/* if saved registers of this frame are not known yet, read and cache them. */
- if (!fi->cache_fsr) {
- fi->cache_fsr = (struct frame_saved_regs *)
- obstack_alloc (&frame_cache_obstack, sizeof (struct frame_saved_regs));
- bzero (fi->cache_fsr, sizeof (struct frame_saved_regs));
-
- if (fi->prev && fi->prev->frame)
- frame_addr = fi->prev->frame;
- else
- frame_addr = read_memory_integer (fi->frame, 4);
-
- /* if != -1, fdata.saved_fpr is the smallest number of saved_fpr. All fpr's
- from saved_fpr to fp31 are saved right underneath caller stack pointer,
- starting from fp31 first. */
-
- if (fdata.saved_fpr >= 0) {
- for (ii=31; ii >= fdata.saved_fpr; --ii)
- fi->cache_fsr->regs [FP0_REGNUM + ii] = frame_addr - ((32 - ii) * 8);
- frame_addr -= (32 - fdata.saved_fpr) * 8;
- }
-
- /* if != -1, fdata.saved_gpr is the smallest number of saved_gpr. All gpr's
- from saved_gpr to gpr31 are saved right under saved fprs, starting
- from r31 first. */
-
- if (fdata.saved_gpr >= 0)
- for (ii=31; ii >= fdata.saved_gpr; --ii)
- fi->cache_fsr->regs [ii] = frame_addr - ((32 - ii) * 4);
- }
+ if (!fi->cache_fsr)
+ frame_get_cache_fsr (fi, &fdata);
/* If no alloca register used, then fi->frame is the value of the %sp for
this frame, and it is good enough. */
@@ -313,13 +333,15 @@ struct frame_info *fi;
if (!fi->next)
return fi->initial_sp = read_register (fdata.alloca_reg);
- /* Otherwise, this is a caller frame. Callee has already saved (???) its
- registers. Find the address in which caller's alloca register is saved. */
+ /* Otherwise, this is a caller frame. Callee has usually already saved
+ registers, but there are are exceptions (such as when the callee
+ has no parameters). Find the address in which caller's alloca
+ register is saved. */
for (callee_fi = fi->next; callee_fi; callee_fi = callee_fi->next) {
if (!callee_fi->cache_fsr)
- fatal ("Callee has not saved caller's registers.");
+ frame_get_cache_fsr (fi, NULL);
/* this is the address in which alloca register is saved. */