1999-01-19 Fernando Nasser <fnasser@totem.to.cygnus.com>
[deliverable/binutils-gdb.git] / gdb / d30v-tdep.c
index eb889900c58f8727b57955e89b0abad55cc892a4..ec46463314119b1ee8ec8ba866122f04e7c18c36 100644 (file)
@@ -35,7 +35,75 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
 
 void d30v_frame_find_saved_regs PARAMS ((struct frame_info *fi,
                                         struct frame_saved_regs *fsr));
+void d30v_frame_find_saved_regs_offsets PARAMS ((struct frame_info *fi,
+                                                struct frame_saved_regs *fsr));
 static void d30v_pop_dummy_frame PARAMS ((struct frame_info *fi));
+static void d30v_print_flags PARAMS ((void));
+static void print_flags_command PARAMS ((char *, int));
+
+/* the following defines assume:
+   fp is r61, lr is r62, sp is r63, and ?? is r22
+   if that changes, they will need to be updated */
+
+#define OP_MASK_ALL_BUT_RA     0x0ffc0fff /* throw away Ra, keep the rest */
+
+#define OP_STW_SPM             0x054c0fc0 /* stw Ra, @(sp-) */
+#define OP_STW_SP_R0           0x05400fc0 /* stw Ra, @(sp,r0) */
+#define OP_STW_SP_IMM0         0x05480fc0 /* st Ra, @(sp, 0x0) */
+#define OP_STW_R22P_R0         0x05440580 /* stw Ra, @(r22+,r0) */
+
+#define OP_ST2W_SPM            0x056c0fc0 /* st2w Ra, @(sp-) */
+#define OP_ST2W_SP_R0          0x05600fc0 /* st2w Ra, @(sp, r0) */
+#define OP_ST2W_SP_IMM0                0x05680fc0 /* st2w Ra, @(sp, 0x0) */
+#define OP_ST2W_R22P_R0                0x05640580 /* st2w Ra, @(r22+, r0) */
+
+#define OP_MASK_OPCODE         0x0ffc0000 /* just the opcode, ign operands */
+#define OP_NOP                 0x00f00000 /* nop */
+
+#define OP_MASK_ALL_BUT_IMM    0x0fffffc0 /* throw away imm, keep the rest */
+#define OP_SUB_SP_IMM          0x082bffc0 /* sub sp,sp,imm */
+#define OP_ADD_SP_IMM          0x080bffc0 /* add sp,sp,imm */
+#define OP_ADD_R22_SP_IMM      0x08096fc0 /* add r22,sp,imm */
+#define OP_STW_FP_SP_IMM       0x054bdfc0 /* stw fp,@(sp,imm) */
+#define OP_OR_SP_R0_IMM                0x03abf000 /* or sp,r0,imm */
+
+/* no mask */
+#define OP_OR_FP_R0_SP         0x03a3d03f /* or fp,r0,sp */
+#define OP_OR_FP_SP_R0         0x03a3dfc0 /* or fp,sp,r0 */
+#define OP_OR_FP_IMM0_SP       0x03abd03f /* or fp,0x0,sp */
+#define OP_STW_FP_R22P_R0      0x0547d580 /* stw fp,@(r22+,r0) */
+#define OP_STW_LR_R22P_R0      0x0547e580 /* stw lr,@(r22+,r0) */
+
+#define OP_MASK_OP_AND_RB      0x0ff80fc0 /* keep op and rb,throw away rest */
+#define OP_STW_SP_IMM          0x05480fc0 /* stw Ra,@(sp,imm) */
+#define OP_ST2W_SP_IMM         0x05680fc0 /* st2w Ra,@(sp,imm) */
+#define OP_STW_FP_IMM          0x05480f40 /* stw Ra,@(fp,imm) */
+#define OP_STW_FP_R0           0x05400f40 /* stw Ra,@(fp,r0) */
+
+#define OP_MASK_FM_BIT         0x80000000
+#define OP_MASK_CC_BITS                0x70000000
+#define OP_MASK_SUB_INST       0x0fffffff
+
+#define EXTRACT_RA(op)         (((op) >> 12) & 0x3f)
+#define EXTRACT_RB(op)         (((op) >> 6) & 0x3f)
+#define EXTRACT_RC(op)         (((op) & 0x3f)
+#define EXTRACT_UIMM6(op)      ((op) & 0x3f)
+#define EXTRACT_IMM6(op)       ((((int)EXTRACT_UIMM6(op)) << 26) >> 26)
+#define EXTRACT_IMM26(op)      ((((op)&0x0ff00000) >> 2) | ((op)&0x0003ffff))
+#define EXTRACT_IMM32(opl, opr)        ((EXTRACT_UIMM6(opl) << 26)|EXTRACT_IMM26(opr))
+
+
+int
+d30v_frame_chain_valid (chain, fi)
+     CORE_ADDR chain;
+     struct frame_info *fi;      /* not used here */
+{
+#if 0
+  return ((chain) != 0 && (fi) != 0 && (fi)->return_pc != 0);
+#else
+  return ((chain) != 0 && (fi) != 0 && (fi)->frame <= chain);
+#endif
+}
 
 /* Discard from the stack the innermost frame, restoring all saved
    registers.  */
@@ -73,15 +141,15 @@ d30v_pop_frame ()
     {
       if (fsr.regs[regnum])
        {
-         write_register (regnum, read_memory_unsigned_integer (fsr.regs[regnum], 2));
+         write_register (regnum, read_memory_unsigned_integer (fsr.regs[regnum], 4));
        }
     }
   if (fsr.regs[PSW_REGNUM])
     {
-      write_register (PSW_REGNUM, read_memory_unsigned_integer (fsr.regs[PSW_REGNUM], 2));
+      write_register (PSW_REGNUM, read_memory_unsigned_integer (fsr.regs[PSW_REGNUM], 4));
     }
 
-  write_register (PC_REGNUM, read_register(13));
+  write_register (PC_REGNUM, read_register(LR_REGNUM));
   write_register (SP_REGNUM, fp + frame->size);
   target_store_registers (-1);
   flush_cached_frames ();
@@ -89,36 +157,76 @@ d30v_pop_frame ()
 
 static int 
 check_prologue (op)
-     unsigned short op;
+     unsigned long op;
 {
-  /* st  rn, @-sp */
-  if ((op & 0x7E1F) == 0x6C1F)
+  /* add sp,sp,imm -- observed */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_ADD_SP_IMM)
     return 1;
 
-  /* st2w  rn, @-sp */
-  if ((op & 0x7E3F) == 0x6E1F)
+  /* add r22,sp,imm -- observed */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_ADD_R22_SP_IMM)
     return 1;
 
-  /* subi  sp, n */
-  if ((op & 0x7FE1) == 0x01E1)
+  /* or  fp,r0,sp -- observed */
+  if (op == OP_OR_FP_R0_SP)
     return 1;
 
-  /* mv  r11, sp */
-  if (op == 0x417E)
+  /* nop */
+  if ((op & OP_MASK_OPCODE) == OP_NOP)
     return 1;
 
-  /* nop */
-  if (op == 0x5E00)
+  /* stw  Ra,@(sp,r0) */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_SP_R0)
     return 1;
 
-  /* st  rn, @sp */
-  if ((op & 0x7E1F) == 0x681E)
+  /* stw  Ra,@(sp,0x0) */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_SP_IMM0)
     return 1;
 
-  /* st2w  rn, @sp */
- if ((op & 0x7E3F) == 0x3A1E)
+  /* st2w  Ra,@(sp,r0) */
+ if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SP_R0)
+   return 1;
+
+  /* st2w  Ra,@(sp,0x0) */
+ if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SP_IMM0)
+   return 1;
+
+ /* stw fp, @(r22+,r0) -- observed */
+ if (op == OP_STW_FP_R22P_R0)
+   return 1;
+
+ /* stw r62, @(r22+,r0) -- observed */
+ if (op == OP_STW_LR_R22P_R0)
    return 1;
 
+ /* stw Ra, @(fp,r0) -- observed */
+ if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_FP_R0)
+   return 1;                   /* first arg */
+
+ /* stw Ra, @(fp,imm) -- observed */
+ if ((op & OP_MASK_OP_AND_RB) == OP_STW_FP_IMM)
+   return 1;                   /* second and subsequent args */
+
+ /* stw fp,@(sp,imm) -- observed */
+ if ((op & OP_MASK_ALL_BUT_IMM) == OP_STW_FP_SP_IMM)
+   return 1;
+
+ /* st2w Ra,@(r22+,r0) */
+ if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_R22P_R0)
+   return 1;
+
+  /* stw  Ra, @(sp-) */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_SPM)
+    return 1;
+
+  /* st2w  Ra, @(sp-) */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SPM)
+    return 1;
+
+  /* sub.?  sp,sp,imm */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_SUB_SP_IMM)
+    return 1;
+
   return 0;
 }
 
@@ -126,8 +234,11 @@ CORE_ADDR
 d30v_skip_prologue (pc)
      CORE_ADDR pc;
 {
-  unsigned long op;
-  unsigned short op1, op2;
+  unsigned long op[2];
+  unsigned long opl, opr;      /* left / right sub operations */
+  unsigned long fm0, fm1;      /* left / right mode bits */
+  unsigned long cc0, cc1;
+  unsigned long op1, op2;
   CORE_ADDR func_addr, func_end;
   struct symtab_and_line sal;
 
@@ -140,32 +251,44 @@ d30v_skip_prologue (pc)
        return sal.end;
     }
   
-  if (target_read_memory (pc, (char *)&op, 4))
+  if (target_read_memory (pc, (char *)&op[0], 8))
     return pc;                 /* Can't access it -- assume no prologue. */
 
   while (1)
     {
-      op = (unsigned long)read_memory_integer (pc, 4);
-      if ((op & 0xC0000000) == 0xC0000000)
+      opl = (unsigned long)read_memory_integer (pc, 4);
+      opr = (unsigned long)read_memory_integer (pc+4, 4);
+
+      fm0 = (opl & OP_MASK_FM_BIT);
+      fm1 = (opr & OP_MASK_FM_BIT);
+
+      cc0 = (opl & OP_MASK_CC_BITS);
+      cc1 = (opr & OP_MASK_CC_BITS);
+
+      opl = (opl & OP_MASK_SUB_INST);
+      opr = (opr & OP_MASK_SUB_INST);
+
+      if (fm0 && fm1)
        {
-         /* long instruction */
-         if ( ((op & 0x3FFF0000) != 0x01FF0000) &&   /* add3 sp,sp,n */
-              ((op & 0x3F0F0000) != 0x340F0000) &&   /* st  rn, @(offset,sp) */
-              ((op & 0x3F1F0000) != 0x350F0000))     /* st2w  rn, @(offset,sp) */
+         /* long instruction (opl contains the opcode) */
+         if (((opl & OP_MASK_ALL_BUT_IMM) != OP_ADD_SP_IMM) && /* add sp,sp,imm */
+             ((opl & OP_MASK_ALL_BUT_IMM) != OP_ADD_R22_SP_IMM) && /* add r22,sp,imm */
+             ((opl & OP_MASK_OP_AND_RB) != OP_STW_SP_IMM) && /* stw Ra, @(sp,imm) */
+             ((opl & OP_MASK_OP_AND_RB) != OP_ST2W_SP_IMM)) /* st2w Ra, @(sp,imm) */
            break;
        }
       else
        {
          /* short instructions */
-         if ((op & 0xC0000000) == 0x80000000)
+         if (fm0 && !fm1)
            {
-             op2 = (op & 0x3FFF8000) >> 15;
-             op1 = op & 0x7FFF;
+             op1 = opr;
+             op2 = opl;
            } 
          else 
            {
-             op1 = (op & 0x3FFF8000) >> 15;
-             op2 = op & 0x7FFF;
+             op1 = opl;
+             op2 = opr;
            }
          if (check_prologue(op1))
            {
@@ -173,19 +296,21 @@ d30v_skip_prologue (pc)
                {
                  /* if the previous opcode was really part of the prologue */
                  /* and not just a NOP, then we want to break after both instructions */
-                 if (op1 != 0x5E00)
-                   pc += 4;
+                 if ((op1 & OP_MASK_OPCODE) != OP_NOP)
+                   pc += 8;
                  break;
                }
            }
          else
            break;
        }
-      pc += 4;
+      pc += 8;
     }
   return pc;
 }
 
+static int end_of_stack;
+
 /* Given a GDB frame, determine the address of the calling function's frame.
    This will be used to create a new GDB frame struct, and then
    INIT_EXTRA_FRAME_INFO and INIT_FRAME_PC will be called for the new frame.
@@ -199,6 +324,9 @@ d30v_frame_chain (frame)
 
   d30v_frame_find_saved_regs (frame, &fsr);
 
+  if (end_of_stack)
+    return (CORE_ADDR)0;
+
   if (frame->return_pc == IMEM_START)
     return (CORE_ADDR)0;
 
@@ -210,79 +338,159 @@ d30v_frame_chain (frame)
       return fsr.regs[SP_REGNUM];
     }
 
-  if (!read_memory_unsigned_integer(fsr.regs[FP_REGNUM],2))
+  if (!read_memory_unsigned_integer(fsr.regs[FP_REGNUM],4))
     return (CORE_ADDR)0;
 
-  return read_memory_unsigned_integer(fsr.regs[FP_REGNUM],2)| DMEM_START;
+  return read_memory_unsigned_integer(fsr.regs[FP_REGNUM],4);
 }  
 
 static int next_addr, uses_frame;
+static int frame_size;
 
 static int 
 prologue_find_regs (op, fsr, addr)
-     unsigned short op;
+     unsigned long op;
      struct frame_saved_regs *fsr;
      CORE_ADDR addr;
 {
   int n;
+  int offset;
 
-  /* st  rn, @-sp */
-  if ((op & 0x7E1F) == 0x6C1F)
+  /* add sp,sp,imm -- observed */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_ADD_SP_IMM)
     {
-      n = (op & 0x1E0) >> 5;
-      next_addr -= 2;
-      fsr->regs[n] = next_addr;
+      offset = EXTRACT_IMM6(op);
+      /*next_addr += offset;*/
+      frame_size += -offset;
       return 1;
     }
 
-  /* st2w  rn, @-sp */
-  else if ((op & 0x7E3F) == 0x6E1F)
+  /* add r22,sp,imm -- observed */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_ADD_R22_SP_IMM)
     {
-      n = (op & 0x1E0) >> 5;
-      next_addr -= 4;
-      fsr->regs[n] = next_addr;
-      fsr->regs[n+1] = next_addr+2;
+      offset = EXTRACT_IMM6(op);
+      next_addr = (offset - frame_size);
       return 1;
     }
 
-  /* subi  sp, n */
-  if ((op & 0x7FE1) == 0x01E1)
+  /* stw Ra, @(fp, offset) -- observed */
+  if ((op & OP_MASK_OP_AND_RB) == OP_STW_FP_IMM)
     {
-      n = (op & 0x1E) >> 1;
-      if (n == 0)
-       n = 16;
-      next_addr -= n;
+      n = EXTRACT_RA(op);
+      offset = EXTRACT_IMM6(op);
+      fsr->regs[n] = (offset - frame_size);
       return 1;
     }
 
-  /* mv  r11, sp */
-  if (op == 0x417E)
+  /* stw Ra, @(fp, r0) -- observed */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_FP_R0)
+    {
+      n = EXTRACT_RA(op);
+      fsr->regs[n] = (- frame_size);
+      return 1;
+    }
+
+  /* or  fp,0,sp -- observed */
+  if ((op == OP_OR_FP_R0_SP) ||
+      (op == OP_OR_FP_SP_R0) ||
+      (op == OP_OR_FP_IMM0_SP))
     {
       uses_frame = 1;
       return 1;
     }
 
   /* nop */
-  if (op == 0x5E00)
+  if ((op & OP_MASK_OPCODE) == OP_NOP)
     return 1;
 
-  /* st  rn, @sp */
-  if ((op & 0x7E1F) == 0x681E)
+  /* stw Ra,@(r22+,r0) -- observed */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_R22P_R0)
+    {
+      n = EXTRACT_RA(op);
+      fsr->regs[n] = next_addr;
+      next_addr += 4;
+      return 1;
+    }
+#if 0                          /* subsumed in pattern above */
+  /* stw fp,@(r22+,r0) -- observed */
+  if (op == OP_STW_FP_R22P_R0)
+    {
+      fsr->regs[FP_REGNUM] = next_addr;        /* XXX */
+      next_addr += 4;
+      return 1;
+    }
+
+  /* stw r62,@(r22+,r0) -- observed */
+  if (op == OP_STW_LR_R22P_R0)
+    {
+      fsr->regs[LR_REGNUM] = next_addr;
+      next_addr += 4;
+      return 1;
+    }
+#endif
+  /* st2w Ra,@(r22+,r0) -- observed */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_R22P_R0)
     {
-      n = (op & 0x1E0) >> 5;
+      n = EXTRACT_RA(op);
       fsr->regs[n] = next_addr;
+      fsr->regs[n+1] = next_addr + 4;
+      next_addr += 8;
       return 1;
     }
 
-  /* st2w  rn, @sp */
-  if ((op & 0x7E3F) == 0x3A1E)
+  /* stw  rn, @(sp-) */
+  if ((op & OP_MASK_ALL_BUT_RA) == OP_STW_SPM)
     {
-      n = (op & 0x1E0) >> 5;
+      n = EXTRACT_RA(op);
       fsr->regs[n] = next_addr;
-      fsr->regs[n+1] = next_addr+2;
+      next_addr -= 4;
+      return 1;
+    }
+
+  /* st2w  Ra, @(sp-) */
+  else if ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SPM)
+    {
+      n = EXTRACT_RA(op);
+      fsr->regs[n] = next_addr;
+      fsr->regs[n+1] = next_addr+4;
+      next_addr -= 8;
+      return 1;
+    }
+
+  /* sub  sp,sp,imm */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_SUB_SP_IMM)
+    {
+      offset = EXTRACT_IMM6(op);
+      frame_size += -offset;
+      return 1;
+    }
+
+  /* st  rn, @(sp,0) -- observed */
+  if (((op & OP_MASK_ALL_BUT_RA) == OP_STW_SP_R0) ||
+      ((op & OP_MASK_ALL_BUT_RA) == OP_STW_SP_IMM0))
+    {
+      n = EXTRACT_RA(op);
+      fsr->regs[n] = (- frame_size);
       return 1;
     }
 
+  /* st2w  rn, @(sp,0) */
+  if (((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SP_R0) ||
+      ((op & OP_MASK_ALL_BUT_RA) == OP_ST2W_SP_IMM0))
+    {
+      n = EXTRACT_RA(op);
+      fsr->regs[n] = (- frame_size);
+      fsr->regs[n+1] = (- frame_size) + 4;
+      return 1;
+    }
+
+  /* stw fp,@(sp,imm) -- observed */
+  if ((op & OP_MASK_ALL_BUT_IMM) == OP_STW_FP_SP_IMM)
+    {
+      offset = EXTRACT_IMM6(op);
+      fsr->regs[FP_REGNUM] = (offset - frame_size);
+      return 1;
+    }
   return 0;
 }
 
@@ -297,43 +505,117 @@ d30v_frame_find_saved_regs (fi, fsr)
      struct frame_saved_regs *fsr;
 {
   CORE_ADDR fp, pc;
-  unsigned long op;
-  unsigned short op1, op2;
+  unsigned long opl, opr;
+  unsigned long op1, op2;
+  unsigned long fm0, fm1;
+  int i;
+
+  fp = fi->frame;
+  memset (fsr, 0, sizeof (*fsr));
+  next_addr = 0;
+  frame_size = 0;
+  end_of_stack = 0;
+
+  uses_frame = 0;
+
+  d30v_frame_find_saved_regs_offsets (fi, fsr);
+  
+  fi->size = frame_size;
+
+  if (!fp)
+    fp = read_register(SP_REGNUM);
+
+  for (i=0; i<NUM_REGS-1; i++)
+    if (fsr->regs[i])
+      {
+       fsr->regs[i] = fsr->regs[i] + fp + frame_size;
+      }
+
+  if (fsr->regs[LR_REGNUM])
+    fi->return_pc = read_memory_unsigned_integer(fsr->regs[LR_REGNUM],4);
+  else
+    fi->return_pc = read_register(LR_REGNUM);
+  
+  /* the SP is not normally (ever?) saved, but check anyway */
+  if (!fsr->regs[SP_REGNUM])
+    {
+      /* if the FP was saved, that means the current FP is valid, */
+      /* otherwise, it isn't being used, so we use the SP instead */
+      if (uses_frame)
+       fsr->regs[SP_REGNUM] = read_register(FP_REGNUM) + fi->size;
+      else
+       {
+         fsr->regs[SP_REGNUM] = fp + fi->size;
+         fi->frameless = 1;
+         fsr->regs[FP_REGNUM] = 0;
+       }
+    }
+}
+
+void
+d30v_frame_find_saved_regs_offsets (fi, fsr)
+     struct frame_info *fi;
+     struct frame_saved_regs *fsr;
+{
+  CORE_ADDR fp, pc;
+  unsigned long opl, opr;
+  unsigned long op1, op2;
+  unsigned long fm0, fm1;
   int i;
 
   fp = fi->frame;
   memset (fsr, 0, sizeof (*fsr));
   next_addr = 0;
+  frame_size = 0;
+  end_of_stack = 0;
 
   pc = get_pc_function_start (fi->pc);
 
   uses_frame = 0;
-  while (1)
+  while (pc < fi->pc)
     {
-      op = (unsigned long)read_memory_integer (pc, 4);
-      if ((op & 0xC0000000) == 0xC0000000)
+      opl = (unsigned long)read_memory_integer (pc, 4);
+      opr = (unsigned long)read_memory_integer (pc+4, 4);
+
+      fm0 = (opl & OP_MASK_FM_BIT);
+      fm1 = (opr & OP_MASK_FM_BIT);
+
+      opl = (opl & OP_MASK_SUB_INST);
+      opr = (opr & OP_MASK_SUB_INST);
+
+      if (fm0 && fm1)
        {
          /* long instruction */
-         if ((op & 0x3FFF0000) == 0x01FF0000)
+         if ((opl & OP_MASK_ALL_BUT_IMM) == OP_ADD_SP_IMM)
+           {
+             /* add sp,sp,n */
+             long offset = EXTRACT_IMM32(opl, opr);
+             frame_size += -offset;
+           }
+         else if ((opl & OP_MASK_ALL_BUT_IMM) == OP_ADD_R22_SP_IMM)
            {
-             /* add3 sp,sp,n */
-             short n = op & 0xFFFF;
-             next_addr += n;
+             /* add r22,sp,offset */
+             long offset = EXTRACT_IMM32(opl,opr);
+             next_addr = (offset - frame_size);
            }
-         else if ((op & 0x3F0F0000) == 0x340F0000)
+         else if ((opl & OP_MASK_OP_AND_RB) == OP_STW_SP_IMM)
            {
-             /* st  rn, @(offset,sp) */
-             short offset = op & 0xFFFF;
-             short n = (op >> 20) & 0xF;
-             fsr->regs[n] = next_addr + offset;
+             /* st Ra, @(sp,imm) */
+             long offset = EXTRACT_IMM32(opl, opr);
+             short n = EXTRACT_RA(opl);
+             fsr->regs[n] = (offset - frame_size);
            }
-         else if ((op & 0x3F1F0000) == 0x350F0000)
+         else if ((opl & OP_MASK_OP_AND_RB) == OP_ST2W_SP_IMM)
            {
-             /* st2w  rn, @(offset,sp) */
-             short offset = op & 0xFFFF;
-             short n = (op >> 20) & 0xF;
-             fsr->regs[n] = next_addr + offset;
-             fsr->regs[n+1] = next_addr + offset + 2;
+             /* st2w Ra, @(sp,offset) */
+             long offset = EXTRACT_IMM32(opl, opr);
+             short n = EXTRACT_RA(opl);
+             fsr->regs[n] = (offset - frame_size);
+             fsr->regs[n+1] = (offset - frame_size) + 4;
+           }
+         else if ((opl & OP_MASK_ALL_BUT_IMM) == OP_OR_SP_R0_IMM)
+           {
+             end_of_stack = 1;
            }
          else
            break;
@@ -341,39 +623,40 @@ d30v_frame_find_saved_regs (fi, fsr)
       else
        {
          /* short instructions */
-         if ((op & 0xC0000000) == 0x80000000)
+         if (fm0 && !fm1)
            {
-             op2 = (op & 0x3FFF8000) >> 15;
-             op1 = op & 0x7FFF;
+             op2 = opl;
+             op1 = opr;
            } 
          else 
            {
-             op1 = (op & 0x3FFF8000) >> 15;
-             op2 = op & 0x7FFF;
+             op1 = opl;
+             op2 = opr;
            }
          if (!prologue_find_regs(op1,fsr,pc) || !prologue_find_regs(op2,fsr,pc))
            break;
        }
-      pc += 4;
+      pc += 8;
     }
   
-  fi->size = -next_addr;
+#if 0
+  fi->size = frame_size;
 
-  if (!(fp & 0xffff))
-    fp = read_register(SP_REGNUM) | DMEM_START;
+  if (!fp)
+    fp = read_register(SP_REGNUM);
 
   for (i=0; i<NUM_REGS-1; i++)
     if (fsr->regs[i])
       {
-       fsr->regs[i] = fp - (next_addr - fsr->regs[i]); 
+       fsr->regs[i] = fsr->regs[i] + fp + frame_size;
       }
 
   if (fsr->regs[LR_REGNUM])
-    fi->return_pc = ((read_memory_unsigned_integer(fsr->regs[LR_REGNUM],2) - 1) << 2) | IMEM_START;
+    fi->return_pc = read_memory_unsigned_integer(fsr->regs[LR_REGNUM],4);
   else
-    fi->return_pc = ((read_register(LR_REGNUM) - 1) << 2) | IMEM_START;
+    fi->return_pc = read_register(LR_REGNUM);
   
-  /* th SP is not normally (ever?) saved, but check anyway */
+  /* the SP is not normally (ever?) saved, but check anyway */
   if (!fsr->regs[SP_REGNUM])
     {
       /* if the FP was saved, that means the current FP is valid, */
@@ -387,6 +670,7 @@ d30v_frame_find_saved_regs (fi, fsr)
          fsr->regs[FP_REGNUM] = 0;
        }
     }
+#endif
 }
 
 void
@@ -396,10 +680,40 @@ d30v_init_extra_frame_info (fromleaf, fi)
 {
   struct frame_saved_regs dummy;
 
-  if (fi->next && ((fi->pc & 0xffff) == 0)) 
+  if (fi->next && (fi->pc == 0))
     fi->pc = fi->next->return_pc; 
 
-  d30v_frame_find_saved_regs (fi, &dummy);
+  d30v_frame_find_saved_regs_offsets (fi, &dummy);
+
+  if (uses_frame == 0)
+    fi->frameless = 1;
+  else
+    fi->frameless = 0;
+
+  if ((fi->next == 0) && (uses_frame == 0))
+    /* innermost frame and it's "frameless",
+       so the fi->frame field is wrong, fix it! */
+    fi->frame = read_sp ();
+
+  if (dummy.regs[LR_REGNUM])
+    {
+      /* it was saved, grab it! */
+      dummy.regs[LR_REGNUM] += (fi->frame + frame_size);
+      fi->return_pc = read_memory_unsigned_integer(dummy.regs[LR_REGNUM],4);
+    }
+  else
+    fi->return_pc = read_register(LR_REGNUM);
+}
+
+void
+d30v_init_frame_pc (fromleaf, prev)
+     int fromleaf;
+     struct frame_info *prev;
+{
+  /* default value, put here so we can breakpoint on it and
+     see if the default value is really the right thing to use */
+  prev->pc = (fromleaf ? SAVED_PC_AFTER_CALL (prev->next) : \
+             prev->next ? FRAME_SAVED_PC (prev->next) : read_pc ());
 }
 
 static void d30v_print_register PARAMS ((int regnum, int tabular));
@@ -423,31 +737,64 @@ d30v_print_register (regnum, tabular)
 
       read_relative_register_raw_bytes (regnum, regbuf);
 
-      val_print (REGISTER_VIRTUAL_TYPE (regnum), regbuf, 0,
+      val_print (REGISTER_VIRTUAL_TYPE (regnum), regbuf, 0, 0,
                 gdb_stdout, 'x', 1, 0, Val_pretty_default);
 
       if (!tabular)
        {
          printf_filtered ("    ");
-         val_print (REGISTER_VIRTUAL_TYPE (regnum), regbuf, 0,
+         val_print (REGISTER_VIRTUAL_TYPE (regnum), regbuf, 0, 0,
                 gdb_stdout, 'd', 1, 0, Val_pretty_default);
        }
     }
 }
 
+static void
+d30v_print_flags ()
+{
+  long psw = read_register (PSW_REGNUM);
+  printf_filtered ("flags #1");
+  printf_filtered ("   (sm) %d", (psw & PSW_SM) != 0);
+  printf_filtered ("   (ea) %d", (psw & PSW_EA) != 0);
+  printf_filtered ("   (db) %d", (psw & PSW_DB) != 0);
+  printf_filtered ("   (ds) %d", (psw & PSW_DS) != 0);
+  printf_filtered ("   (ie) %d", (psw & PSW_IE) != 0);
+  printf_filtered ("   (rp) %d", (psw & PSW_RP) != 0);
+  printf_filtered ("   (md) %d\n", (psw & PSW_MD) != 0);
+
+  printf_filtered ("flags #2");
+  printf_filtered ("   (f0) %d", (psw & PSW_F0) != 0);
+  printf_filtered ("   (f1) %d", (psw & PSW_F1) != 0);
+  printf_filtered ("   (f2) %d", (psw & PSW_F2) != 0);
+  printf_filtered ("   (f3) %d", (psw & PSW_F3) != 0);
+  printf_filtered ("    (s) %d", (psw & PSW_S) != 0);
+  printf_filtered ("    (v) %d", (psw & PSW_V) != 0);
+  printf_filtered ("   (va) %d", (psw & PSW_VA) != 0);
+  printf_filtered ("    (c) %d\n", (psw & PSW_C) != 0);
+}
+
+static void
+print_flags_command (args, from_tty)
+     char *args;
+     int from_tty;
+{
+  d30v_print_flags ();
+}
+
 void
 d30v_do_registers_info (regnum, fpregs)
      int regnum;
      int fpregs;
 {
   long long num1, num2;
+  long psw;
 
   if (regnum != -1)
     {
-      if (reg_names[0] == NULL || reg_names[0][0] == '\000')
+      if (REGISTER_NAME (0) == NULL || REGISTER_NAME (0)[0] == '\000')
        return;
 
-      printf_filtered ("%s ", reg_names[regnum]);
+      printf_filtered ("%s ", REGISTER_NAME (regnum));
       d30v_print_register (regnum, 0);
 
       printf_filtered ("\n");
@@ -506,6 +853,7 @@ d30v_do_registers_info (regnum, fpregs)
   d30v_print_register (INT_M_REGNUM, 1);
   printf_filtered ("\n");
 
+  d30v_print_flags ();
   for (regnum = 0; regnum <= 63;)
     {
       int i;
@@ -548,7 +896,7 @@ d30v_fix_call_dummy (dummyname, start_sp, fun, nargs, args, type, gcc_p)
   char buffer[MAX_REGISTER_RAW_SIZE];
   struct frame_info *frame = get_current_frame ();
   frame->dummy = start_sp;
-  start_sp |= DMEM_START;
+  /*start_sp |= DMEM_START;*/
 
   sp = start_sp;
   for (regnum = 0; regnum < NUM_REGS; regnum++)
@@ -557,9 +905,9 @@ d30v_fix_call_dummy (dummyname, start_sp, fun, nargs, args, type, gcc_p)
       store_address (buffer, REGISTER_RAW_SIZE(regnum), read_register(regnum));
       write_memory (sp, buffer, REGISTER_RAW_SIZE(regnum));
     }
-  write_register (SP_REGNUM, (LONGEST)(sp & 0xffff)); 
+  write_register (SP_REGNUM, (LONGEST)sp);
   /* now we need to load LR with the return address */
-  write_register (LR_REGNUM, (LONGEST)(d30v_call_dummy_address() & 0xffff) >> 2);  
+  write_register (LR_REGNUM, (LONGEST)d30v_call_dummy_address());  
   return sp;
 }
 
@@ -592,6 +940,7 @@ d30v_push_arguments (nargs, args, sp, struct_return, struct_addr)
   LONGEST val;
   CORE_ADDR ptrs[10];
 
+#if 0
   /* Pass 1. Put all large args on stack */
   for (i = 0; i < nargs; i++)
     {
@@ -608,7 +957,7 @@ d30v_push_arguments (nargs, args, sp, struct_return, struct_addr)
          ptrs[index++] = sp;
        }
     }
-
+#endif
   index = 0;
 
   for (i = 0; i < nargs; i++)
@@ -617,36 +966,60 @@ d30v_push_arguments (nargs, args, sp, struct_return, struct_addr)
       struct type *arg_type = check_typedef (VALUE_TYPE (arg));
       len = TYPE_LENGTH (arg_type);
       contents = VALUE_CONTENTS(arg);
-      val = extract_signed_integer (contents, len);
       if (len > 4)
        {
-         /* use a pointer to previously saved data */
-         if (regnum < 6)
-           write_register (regnum++, ptrs[index++]);
-         else
+         /* we need multiple registers */
+         int ndx;
+
+         for (ndx = 0; len > 0; ndx += 8, len -= 8)
            {
-             /* no more registers available.  put it on the stack */
-             sp -= 2;
-             store_address (buffer, 2, ptrs[index++]);
-             write_memory (sp, buffer, 2);
+             if (regnum & 1)
+               regnum++;       /* all args > 4 bytes start in even register */
+
+             if (regnum < 18)
+               {
+                 val = extract_signed_integer (&contents[ndx], 4);
+                 write_register (regnum++, val);
+
+                 if (len >= 8)
+                   val = extract_signed_integer (&contents[ndx+4], 4);
+                 else
+                   val = extract_signed_integer (&contents[ndx+4], len-4);
+                 write_register (regnum++, val);
+               }
+             else
+               {
+                 /* no more registers available.  put it on the stack */
+
+                 /* all args > 4 bytes are padded to a multiple of 8 bytes
+                  and start on an 8 byte boundary */
+                 if (sp & 7)
+                   sp -= (sp & 7); /* align it */
+
+                 sp -= ((len + 7) & ~7); /* allocate space */
+                 write_memory (sp, &contents[ndx], len);
+                 break;
+               }
            }
        }
       else
        {
-         if (regnum < 6 )
+         if (regnum < 18 )
            {
-             if (len == 4)
-               write_register (regnum++, val>>16);
-             write_register (regnum++, val & 0xffff);
+             val = extract_signed_integer (contents, len);
+             write_register (regnum++, val);
            }
          else
            {
-             sp -= len;
-             store_address (buffer, len, val);
-             write_memory (sp, buffer, len);
+             /* all args are padded to a multiple of 4 bytes (at least) */
+             sp -= ((len + 3) & ~3);
+             write_memory (sp, contents, len);
            }
        }
     }
+  if (sp & 7)
+    /* stack pointer is not on an 8 byte boundary -- align it */
+    sp -= (sp & 7);
   return sp;
 }
 
@@ -995,11 +1368,13 @@ _initialize_d30v_tdep ()
   target_resume_hook = d30v_eva_prepare_to_trace;
   target_wait_loop_hook = d30v_eva_get_trace_data;
 
+  add_info ("flags", print_flags_command, "Print d30v flags.");
+
   add_com ("trace", class_support, trace_command,
           "Enable tracing of instruction execution.");
 
   add_com ("untrace", class_support, untrace_command,
-          "Disable tracing of instruction execution.");
+          "Disable tracing of instruction execution.");
 
   add_com ("tdisassemble", class_vars, tdisassemble_command,
           "Disassemble the trace buffer.\n\
This page took 0.034719 seconds and 4 git commands to generate.