* rs6000-tdep.c (frame_saved_pc): Check for signal handler caller
authorPeter Schauer <Peter.Schauer@mytum.de>
Sun, 30 Jul 1995 08:45:15 +0000 (08:45 +0000)
committerPeter Schauer <Peter.Schauer@mytum.de>
Sun, 30 Jul 1995 08:45:15 +0000 (08:45 +0000)
before trying to determine the start of the function.
(skip_prologue):  Skip subroutine call which might save the
floating point registers only if it is within the first three
instructions.
Reinstate setting of alloca_reg if setup of a gcc frame pointer
is found.
(frame_get_cache_fsr):  Use new fields in rs6000_framedata.

gdb/rs6000-tdep.c

index 4c41808c7df2404c32fc3f11d76b970190a2ded5..c53e5423936c25330648441f82524f5e66dbf3b2 100644 (file)
@@ -292,6 +292,12 @@ skip_prologue (pc, fdata)
        continue;
 
       } else if ((op & 0xfc000000) == 0x48000000) {    /* bl foo, to save fprs??? */
+
+       /* Don't skip over the subroutine call if it is not within the first
+          three instructions of the prologue.  */
+       if ((pc - orig_pc) > 8)
+         break;
+
        op = read_memory_integer (pc+4, 4);
 
        /* At this point, make sure this is not a trampoline function
@@ -300,7 +306,7 @@ skip_prologue (pc, fdata)
           prologue. */
 
        if (op == 0x4def7b82 || op == 0)                /* crorc 15, 15, 15 */
-         return pc;                                    /* don't skip over this branch */
+         break;                                        /* don't skip over this branch */
 
        continue;
 
@@ -335,6 +341,7 @@ skip_prologue (pc, fdata)
       } else if (op == 0x603f0000                      /* oril r31, r1, 0x0 */
                 || op == 0x7c3f0b78) {                 /* mr r31, r1 */
        framep = 1;
+       fdata->alloca_reg = 31;
        continue;
 
       } else {
@@ -927,6 +934,9 @@ frame_saved_pc (fi)
   struct rs6000_framedata fdata;
   int frameless;
 
+  if (fi->signal_handler_caller)
+    return read_memory_integer (fi->frame + SIG_FRAME_PC_OFFSET, 4);
+
   func_start = get_pc_function_start (fi->pc) + FUNCTION_START_OFFSET;
 
   /* If we failed to find the start of the function, it is a mistake
@@ -936,9 +946,6 @@ frame_saved_pc (fi)
 
   (void) skip_prologue (func_start, &fdata);
 
-  if (fi->signal_handler_caller)
-    return read_memory_integer (fi->frame + SIG_FRAME_PC_OFFSET, 4);
-
   if (fdata.lr_offset == 0 && fi->next != NULL)
     return read_memory_integer (rs6000_frame_chain (fi) + DEFAULT_LR_SAVE, 4);
 
@@ -979,22 +986,36 @@ frame_get_cache_fsr (fi, fdatap)
     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. */
+     All fpr's from saved_fpr to fp31 are saved.  */
 
   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;
+    int fpr_offset = frame_addr + fdatap->fpr_offset;
+    for (ii = fdatap->saved_fpr; ii < 32; ii++) {
+      fi->cache_fsr->regs [FP0_REGNUM + ii] = fpr_offset;
+      fpr_offset += 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. */
+     All gpr's from saved_gpr to gpr31 are saved.  */
   
-  if (fdatap->saved_gpr >= 0)
-    for (ii=31; ii >= fdatap->saved_gpr; --ii)
-      fi->cache_fsr->regs [ii] = frame_addr - ((32 - ii) * 4);
+  if (fdatap->saved_gpr >= 0) {
+    int gpr_offset = frame_addr + fdatap->gpr_offset;
+    for (ii = fdatap->saved_gpr; ii < 32; ii++) {
+      fi->cache_fsr->regs [ii] = gpr_offset;
+      gpr_offset += 4;
+    }
+  }
+
+  /* If != 0, fdatap->cr_offset is the offset from the frame that holds
+     the CR.  */
+  if (fdatap->cr_offset != 0)
+    fi->cache_fsr->regs [CR_REGNUM] = frame_addr + fdatap->cr_offset;
+
+  /* If != 0, fdatap->lr_offset is the offset from the frame that holds
+     the LR.  */
+  if (fdatap->lr_offset != 0)
+    fi->cache_fsr->regs [LR_REGNUM] = frame_addr + fdatap->lr_offset;
 }
 
 /* Return the address of a frame. This is the inital %sp value when the frame