New gdbarch method "dwarf_cfa_op" and migrate SPARC to it

Message ID d6197185-945d-d90b-6eb9-729fa6d3705b@foss.arm.com
State New, archived
Headers

Commit Message

Jiong Wang April 19, 2017, 10:17 a.m. UTC
  Hi,

Recently a feature called "return address signing" has been added to GCC to
prevent stack smash stack on AArch64.  For details please refer:

   https://gcc.gnu.org/ml/gcc-patches/2017-01/msg00376.html

GDB needs to be aware of this feature so it can restore the original return
address which is critical for unwinding.

On compiler side, whenever return address, i.e. LR register, is mangled or
restored by hardware instruction, compiler is expected to generate a
DW_CFA_GNU_window_save CFA instruction to toggle the LR signing status.
DW_CFA_GNU_window_save was introduced by SPARC for their register window
feature, here AArch64 want to multiplex this DWARF CFA instruction as it's in
vendor space.  So, a new gdbarch method, dwarf_cfa_op, is introduced.

Currently, I have restricted it's parameter to those only needed by SPARC and
AArch64 for DW_CFA_GNU_window_save which is a instruction takes none operand.
Should any further DWARF CFA instructions want to be multiplexed in the future,
the parameter list can be extended.  Below is the current function prototype.

    typedef int (gdbarch_dwarf_multiplex_cfa_op_ftype)
      (struct gdbarch *gdbarch, gdb_byte op, struct dwarf2_frame_state *fs);

   DW_CFA_GNU_window_save support for SPARC is migrated to this new gdbarch
method.

   SPARC-linux cross gdb build OK.

   OK for master?

gdb/
2017-04-19  Jiong Wang  <jiong.wang@arm.com>

         * dwarf2-frame.c (dwarf2_frame_state_alloc_regs): Made the
         visibility external.
         (execute_cfa_program): Multiplex DW_CFA_GNU_window_save.
         (enum cfa_how_kind): Move to ...
         (struct dwarf2_frame_state_reg_info): Likewise.
         (struct dwarf2_frame_state): Likewise.
         * dwarf2-frame.h: ... here.
         (dwarf2_frame_state_alloc_regs): New declaration.
         * gdbarch.sh: New target hook dwarf_cfa_op.
         * gdbarch.c: Regenerated.
         * gdbarch.h: Regenerated.
         * sparc-tdep.c (sparc_dwarf_cfa_op): New function.
         (sparc32_gdbarch_init): Register dwarf_cfa_op hook.
  

Comments

Ivo Raisr April 21, 2017, 2:43 p.m. UTC | #1
On 19.4.2017 12:17, Jiong Wang wrote:
> Hi,
>
> Recently a feature called "return address signing" has been added to GCC to
> prevent stack smash stack on AArch64.  For details please refer:
>
>   https://gcc.gnu.org/ml/gcc-patches/2017-01/msg00376.html
>
> GDB needs to be aware of this feature so it can restore the original return
> address which is critical for unwinding.
>
> On compiler side, whenever return address, i.e. LR register, is mangled or
> restored by hardware instruction, compiler is expected to generate a
> DW_CFA_GNU_window_save CFA instruction to toggle the LR signing status.
> DW_CFA_GNU_window_save was introduced by SPARC for their register window
> feature, here AArch64 want to multiplex this DWARF CFA instruction as
> it's in vendor space.  So, a new gdbarch method, dwarf_cfa_op, is introduced.

Are you sure to reuse DW_CFA_GNU_window_save for AArch64 purposes? 
Because semantics of DW_CFA_GNU_window_save (implied from "window save") 
mean much more than just fiddling with return address. For example as 
you learned from dwarf2-frame.c, whole register window needs to be copied.


>   SPARC-linux cross gdb build OK.

I have built gdb on sparc64-linux with your changes and no new 
regressions are introduced in the test suite. So the patch is ok w.r.t. 
this.

I.
  
Jiong Wang April 21, 2017, 2:56 p.m. UTC | #2
On 21/04/17 15:43, Ivo Raisr wrote:
>
>
> On 19.4.2017 12:17, Jiong Wang wrote:
>> Hi,
>>
>> Recently a feature called "return address signing" has been added to 
>> GCC to
>> prevent stack smash stack on AArch64.  For details please refer:
>>
>>   https://gcc.gnu.org/ml/gcc-patches/2017-01/msg00376.html
>>
>> GDB needs to be aware of this feature so it can restore the original 
>> return
>> address which is critical for unwinding.
>>
>> On compiler side, whenever return address, i.e. LR register, is 
>> mangled or
>> restored by hardware instruction, compiler is expected to generate a
>> DW_CFA_GNU_window_save CFA instruction to toggle the LR signing status.
>> DW_CFA_GNU_window_save was introduced by SPARC for their register window
>> feature, here AArch64 want to multiplex this DWARF CFA instruction as
>> it's in vendor space.  So, a new gdbarch method, dwarf_cfa_op, is 
>> introduced.
>
> Are you sure to reuse DW_CFA_GNU_window_save for AArch64 purposes? 
> Because semantics of DW_CFA_GNU_window_save (implied from "window 
> save") mean much more than just fiddling with return address. For 
> example as you learned from dwarf2-frame.c, whole register window 
> needs to be copied.

Hi Ivo,

   Thanks very much for testing this on SPARC platform.

   What's really reused is the DWARF CFA number 0x2d behind 
DW_CFA_GNU_window_save.  It is in vendor extension space ( 
DW_CFA_lo_user.. DW_CFA_hi_user) so the semantics depends on vendor 
interpreation.

Regards,
Jiong.
  
Pedro Alves April 21, 2017, 7:37 p.m. UTC | #3
On 04/21/2017 03:56 PM, Jiong Wang wrote:

> Hi Ivo,
> 
>   Thanks very much for testing this on SPARC platform.
> 
>   What's really reused is the DWARF CFA number 0x2d behind
> DW_CFA_GNU_window_save.  It is in vendor extension space (
> DW_CFA_lo_user.. DW_CFA_hi_user) so the semantics depends on vendor
> interpreation.

Maybe the commit log could/should be simplified, because
I was confused too.

Doesn't the Aarch64 version of the opcode have its own
name, like DW_CFA_GNU_Aarch64_whatever, even if it reuses the
opcode number?  I think that would help a lot going forward
if it had one.  E.g., it'd avoid confusion, allow for easier
searching, etc.

On the patch, I think it would be better if in execute_cfa_program:

>  	    case DW_CFA_GNU_window_save:
> -	      /* This is SPARC-specific code, and contains hard-coded
> -		 constants for the register numbering scheme used by

you removed "case CFA_GNU_window_save:" too, and moved the 
gdbarch_dwarf_cfa_op call to the default case.  Then make the hook
return a boolean indication about whether it /recognized the
opcode, and make the caller throw an error if the hook returns
false.  And make that error call be a regular "error()" call instead
of the current internal_error call.  The internal error is bogus
here because we reach that with an unrecognized opcode that comes
from a binary, i.e., input, not a gdb bug.

Replace the gdb_assert in sparc_dwarf_cfa_op by returning false for
unrecognized opcodes, and likewise change default_dwarf_cfa_op
to return false instead of calling error itself.

Add a comment at the hook call site about letting the backend
handle vendor-specific opcodes, and generalize a bit the comment
describing the hook in gdbarch.sh in that direction as well.

Maybe rename to hook from dwarf_cfa_op to something like:

  handle_dwarf_cfa_op
  handle_dwarf_cfa_vendor_op
  execute_dwarf_cfa_op
  execute_dwarf_cfa_vendor_op

too, while at it?

Thanks,
Pedro Alves
  
Ivo Raisr April 21, 2017, 7:53 p.m. UTC | #4
On 21.4.2017 21:37, Pedro Alves wrote:
> On 04/21/2017 03:56 PM, Jiong Wang wrote:
>
>> Hi Ivo,
>>
>>   Thanks very much for testing this on SPARC platform.
>>
>>   What's really reused is the DWARF CFA number 0x2d behind
>> DW_CFA_GNU_window_save.  It is in vendor extension space (
>> DW_CFA_lo_user.. DW_CFA_hi_user) so the semantics depends on vendor
>> interpreation.
>
> Maybe the commit log could/should be simplified, because
> I was confused too.
>
> Doesn't the Aarch64 version of the opcode have its own
> name, like DW_CFA_GNU_Aarch64_whatever, even if it reuses the
> opcode number?  I think that would help a lot going forward
> if it had one.  E.g., it'd avoid confusion, allow for easier
> searching, etc.

Introducing Aarch64 specific name would be very much appreciated.
+1 here!
I.
  

Patch

diff --git a/gdb/arch-utils.c b/gdb/arch-utils.c
index 3b22e51..36a98b5 100644
--- a/gdb/arch-utils.c
+++ b/gdb/arch-utils.c
@@ -204,6 +204,15 @@  default_adjust_dwarf2_line (CORE_ADDR addr, int rel)
   return addr;
 }
 
+/* See arch-utils.h.  */
+
+void
+default_dwarf_cfa_op (struct gdbarch *gdbarch, gdb_byte op,
+		      struct dwarf2_frame_state *fs)
+{
+  error (_("DW_CFA_GNU_window_save is not handled."));
+}
+
 int
 cannot_register_not (struct gdbarch *gdbarch, int regnum)
 {
diff --git a/gdb/arch-utils.h b/gdb/arch-utils.h
index 4d7b499..447f1df 100644
--- a/gdb/arch-utils.h
+++ b/gdb/arch-utils.h
@@ -25,6 +25,7 @@  struct frame_info;
 struct minimal_symbol;
 struct type;
 struct gdbarch_info;
+struct dwarf2_frame_state;
 
 template <size_t bp_size, const gdb_byte *break_insn>
 struct bp_manipulation
@@ -130,6 +131,11 @@  CORE_ADDR default_adjust_dwarf2_addr (CORE_ADDR pc);
 
 CORE_ADDR default_adjust_dwarf2_line (CORE_ADDR addr, int rel);
 
+/* Default DWARF CFA handler does nothing but simply error out.  */
+
+void default_dwarf_cfa_op (struct gdbarch *gdbarch, gdb_byte op,
+			   struct dwarf2_frame_state *fs);
+
 /* Version of cannot_fetch_register() / cannot_store_register() that
    always fails.  */
 
diff --git a/gdb/dwarf2-frame.c b/gdb/dwarf2-frame.c
index f8dd1df..ce5b2a5 100644
--- a/gdb/dwarf2-frame.c
+++ b/gdb/dwarf2-frame.c
@@ -165,58 +165,6 @@  static CORE_ADDR read_encoded_value (struct comp_unit *unit, gdb_byte encoding,
 				     CORE_ADDR func_base);
 
 
-enum cfa_how_kind
-{
-  CFA_UNSET,
-  CFA_REG_OFFSET,
-  CFA_EXP
-};
-
-struct dwarf2_frame_state_reg_info
-{
-  struct dwarf2_frame_state_reg *reg;
-  int num_regs;
-
-  LONGEST cfa_offset;
-  ULONGEST cfa_reg;
-  enum cfa_how_kind cfa_how;
-  const gdb_byte *cfa_exp;
-
-  /* Used to implement DW_CFA_remember_state.  */
-  struct dwarf2_frame_state_reg_info *prev;
-};
-
-/* Structure describing a frame state.  */
-
-struct dwarf2_frame_state
-{
-  /* Each register save state can be described in terms of a CFA slot,
-     another register, or a location expression.  */
-  struct dwarf2_frame_state_reg_info regs;
-
-  /* The PC described by the current frame state.  */
-  CORE_ADDR pc;
-
-  /* Initial register set from the CIE.
-     Used to implement DW_CFA_restore.  */
-  struct dwarf2_frame_state_reg_info initial;
-
-  /* The information we care about from the CIE.  */
-  LONGEST data_align;
-  ULONGEST code_align;
-  ULONGEST retaddr_column;
-
-  /* Flags for known producer quirks.  */
-
-  /* The ARM compilers, in DWARF2 mode, assume that DW_CFA_def_cfa
-     and DW_CFA_def_cfa_offset takes a factored offset.  */
-  int armcc_cfa_offsets_sf;
-
-  /* The ARM compilers, in DWARF2 or DWARF3 mode, may assume that
-     the CFA is defined as REG - OFFSET rather than REG + OFFSET.  */
-  int armcc_cfa_offsets_reversed;
-};
-
 /* Store the length the expression for the CFA in the `cfa_reg' field,
    which is unused in that case.  */
 #define cfa_exp_len cfa_reg
@@ -224,7 +172,7 @@  struct dwarf2_frame_state
 /* Assert that the register set RS is large enough to store gdbarch_num_regs
    columns.  If necessary, enlarge the register set.  */
 
-static void
+void
 dwarf2_frame_state_alloc_regs (struct dwarf2_frame_state_reg_info *rs,
 			       int num_regs)
 {
@@ -673,28 +621,7 @@  bad CFI data; mismatched DW_CFA_restore_state at %s"),
 	      break;
 
 	    case DW_CFA_GNU_window_save:
-	      /* This is SPARC-specific code, and contains hard-coded
-		 constants for the register numbering scheme used by
-		 GCC.  Rather than having a architecture-specific
-		 operation that's only ever used by a single
-		 architecture, we provide the implementation here.
-		 Incidentally that's what GCC does too in its
-		 unwinder.  */
-	      {
-		int size = register_size (gdbarch, 0);
-
-		dwarf2_frame_state_alloc_regs (&fs->regs, 32);
-		for (reg = 8; reg < 16; reg++)
-		  {
-		    fs->regs.reg[reg].how = DWARF2_FRAME_REG_SAVED_REG;
-		    fs->regs.reg[reg].loc.reg = reg + 16;
-		  }
-		for (reg = 16; reg < 32; reg++)
-		  {
-		    fs->regs.reg[reg].how = DWARF2_FRAME_REG_SAVED_OFFSET;
-		    fs->regs.reg[reg].loc.offset = (reg - 16) * size;
-		  }
-	      }
+	      gdbarch_dwarf_cfa_op (gdbarch, insn, fs);
 	      break;
 
 	    case DW_CFA_GNU_args_size:
diff --git a/gdb/dwarf2-frame.h b/gdb/dwarf2-frame.h
index 2ada436..9e8668e 100644
--- a/gdb/dwarf2-frame.h
+++ b/gdb/dwarf2-frame.h
@@ -82,6 +82,58 @@  struct dwarf2_frame_state_reg
   enum dwarf2_frame_reg_rule how;
 };
 
+enum cfa_how_kind
+{
+  CFA_UNSET,
+  CFA_REG_OFFSET,
+  CFA_EXP
+};
+
+struct dwarf2_frame_state_reg_info
+{
+  struct dwarf2_frame_state_reg *reg;
+  int num_regs;
+
+  LONGEST cfa_offset;
+  ULONGEST cfa_reg;
+  enum cfa_how_kind cfa_how;
+  const gdb_byte *cfa_exp;
+
+  /* Used to implement DW_CFA_remember_state.  */
+  struct dwarf2_frame_state_reg_info *prev;
+};
+
+/* Structure describing a frame state.  */
+
+struct dwarf2_frame_state
+{
+  /* Each register save state can be described in terms of a CFA slot,
+     another register, or a location expression.  */
+  struct dwarf2_frame_state_reg_info regs;
+
+  /* The PC described by the current frame state.  */
+  CORE_ADDR pc;
+
+  /* Initial register set from the CIE.
+     Used to implement DW_CFA_restore.  */
+  struct dwarf2_frame_state_reg_info initial;
+
+  /* The information we care about from the CIE.  */
+  LONGEST data_align;
+  ULONGEST code_align;
+  ULONGEST retaddr_column;
+
+  /* Flags for known producer quirks.  */
+
+  /* The ARM compilers, in DWARF2 mode, assume that DW_CFA_def_cfa
+     and DW_CFA_def_cfa_offset takes a factored offset.  */
+  int armcc_cfa_offsets_sf;
+
+  /* The ARM compilers, in DWARF2 or DWARF3 mode, may assume that
+     the CFA is defined as REG - OFFSET rather than REG + OFFSET.  */
+  int armcc_cfa_offsets_reversed;
+};
+
 /* Set the architecture-specific register state initialization
    function for GDBARCH to INIT_REG.  */
 
@@ -120,6 +172,12 @@  extern const struct frame_base *
 
 CORE_ADDR dwarf2_frame_cfa (struct frame_info *this_frame);
 
+/* Assert that the register set RS is large enough to store gdbarch_num_regs
+   columns.  If necessary, enlarge the register set.  */
+
+void dwarf2_frame_state_alloc_regs (struct dwarf2_frame_state_reg_info *rs,
+				    int num_regs);
+
 /* Find the CFA information for PC.
 
    Return 1 if a register is used for the CFA, or 0 if another
diff --git a/gdb/gdbarch.c b/gdb/gdbarch.c
index 5664325..fb5607b 100644
--- a/gdb/gdbarch.c
+++ b/gdb/gdbarch.c
@@ -275,6 +275,7 @@  struct gdbarch
   int have_nonsteppable_watchpoint;
   gdbarch_address_class_type_flags_ftype *address_class_type_flags;
   gdbarch_address_class_type_flags_to_name_ftype *address_class_type_flags_to_name;
+  gdbarch_dwarf_cfa_op_ftype *dwarf_cfa_op;
   gdbarch_address_class_name_to_type_flags_ftype *address_class_name_to_type_flags;
   gdbarch_register_reggroup_p_ftype *register_reggroup_p;
   gdbarch_fetch_pointer_argument_ftype *fetch_pointer_argument;
@@ -436,6 +437,7 @@  gdbarch_alloc (const struct gdbarch_info *info,
   gdbarch->make_symbol_special = default_make_symbol_special;
   gdbarch->adjust_dwarf2_addr = default_adjust_dwarf2_addr;
   gdbarch->adjust_dwarf2_line = default_adjust_dwarf2_line;
+  gdbarch->dwarf_cfa_op = default_dwarf_cfa_op;
   gdbarch->register_reggroup_p = default_register_reggroup_p;
   gdbarch->skip_permanent_breakpoint = default_skip_permanent_breakpoint;
   gdbarch->displaced_step_hw_singlestep = default_displaced_step_hw_singlestep;
@@ -634,6 +636,7 @@  verify_gdbarch (struct gdbarch *gdbarch)
   /* Skip verify of have_nonsteppable_watchpoint, invalid_p == 0 */
   /* Skip verify of address_class_type_flags, has predicate.  */
   /* Skip verify of address_class_type_flags_to_name, has predicate.  */
+  /* Skip verify of dwarf_cfa_op, invalid_p == 0 */
   /* Skip verify of address_class_name_to_type_flags, has predicate.  */
   /* Skip verify of register_reggroup_p, invalid_p == 0 */
   /* Skip verify of fetch_pointer_argument, has predicate.  */
@@ -960,6 +963,9 @@  gdbarch_dump (struct gdbarch *gdbarch, struct ui_file *file)
                       "gdbarch_dump: dwarf2_reg_to_regnum = <%s>\n",
                       host_address_to_string (gdbarch->dwarf2_reg_to_regnum));
   fprintf_unfiltered (file,
+                      "gdbarch_dump: dwarf_cfa_op = <%s>\n",
+                      host_address_to_string (gdbarch->dwarf_cfa_op));
+  fprintf_unfiltered (file,
                       "gdbarch_dump: ecoff_reg_to_regnum = <%s>\n",
                       host_address_to_string (gdbarch->ecoff_reg_to_regnum));
   fprintf_unfiltered (file,
@@ -3523,6 +3529,23 @@  set_gdbarch_address_class_type_flags_to_name (struct gdbarch *gdbarch,
   gdbarch->address_class_type_flags_to_name = address_class_type_flags_to_name;
 }
 
+void
+gdbarch_dwarf_cfa_op (struct gdbarch *gdbarch, gdb_byte op, struct dwarf2_frame_state *fs)
+{
+  gdb_assert (gdbarch != NULL);
+  gdb_assert (gdbarch->dwarf_cfa_op != NULL);
+  if (gdbarch_debug >= 2)
+    fprintf_unfiltered (gdb_stdlog, "gdbarch_dwarf_cfa_op called\n");
+  gdbarch->dwarf_cfa_op (gdbarch, op, fs);
+}
+
+void
+set_gdbarch_dwarf_cfa_op (struct gdbarch *gdbarch,
+                          gdbarch_dwarf_cfa_op_ftype dwarf_cfa_op)
+{
+  gdbarch->dwarf_cfa_op = dwarf_cfa_op;
+}
+
 int
 gdbarch_address_class_name_to_type_flags_p (struct gdbarch *gdbarch)
 {
diff --git a/gdb/gdbarch.h b/gdb/gdbarch.h
index 4845f23..88025cb 100644
--- a/gdb/gdbarch.h
+++ b/gdb/gdbarch.h
@@ -827,6 +827,15 @@  typedef const char * (gdbarch_address_class_type_flags_to_name_ftype) (struct gd
 extern const char * gdbarch_address_class_type_flags_to_name (struct gdbarch *gdbarch, int type_flags);
 extern void set_gdbarch_address_class_type_flags_to_name (struct gdbarch *gdbarch, gdbarch_address_class_type_flags_to_name_ftype *address_class_type_flags_to_name);
 
+/* Multiplex DWARF expression operation.  OP is the DWARF CFA operation we want
+   to multiplex.  FS are passed from the generic execute_cfa_program function.
+   This hook is currently used by SPARC and AArch64 backends to multiplex
+   DW_CFA_GNU_window_save. */
+
+typedef void (gdbarch_dwarf_cfa_op_ftype) (struct gdbarch *gdbarch, gdb_byte op, struct dwarf2_frame_state *fs);
+extern void gdbarch_dwarf_cfa_op (struct gdbarch *gdbarch, gdb_byte op, struct dwarf2_frame_state *fs);
+extern void set_gdbarch_dwarf_cfa_op (struct gdbarch *gdbarch, gdbarch_dwarf_cfa_op_ftype *dwarf_cfa_op);
+
 /* Return the appropriate type_flags for the supplied address class.
    This function should return 1 if the address class was recognized and
    type_flags was set, zero otherwise. */
diff --git a/gdb/gdbarch.sh b/gdb/gdbarch.sh
index a42dc43..cbae457 100755
--- a/gdb/gdbarch.sh
+++ b/gdb/gdbarch.sh
@@ -708,6 +708,11 @@  v:int:cannot_step_breakpoint:::0:0::0
 v:int:have_nonsteppable_watchpoint:::0:0::0
 F:int:address_class_type_flags:int byte_size, int dwarf2_addr_class:byte_size, dwarf2_addr_class
 M:const char *:address_class_type_flags_to_name:int type_flags:type_flags
+# Multiplex DWARF expression operation.  OP is the DWARF CFA operation we want
+# to multiplex.  FS are passed from the generic execute_cfa_program function.
+# This hook is currently used by SPARC and AArch64 backends to multiplex
+# DW_CFA_GNU_window_save.
+m:void:dwarf_cfa_op:gdb_byte op, struct dwarf2_frame_state *fs:op, fs::default_dwarf_cfa_op::0
 
 # Return the appropriate type_flags for the supplied address class.
 # This function should return 1 if the address class was recognized and
diff --git a/gdb/sparc-tdep.c b/gdb/sparc-tdep.c
index 078907a..53cb0a4 100644
--- a/gdb/sparc-tdep.c
+++ b/gdb/sparc-tdep.c
@@ -20,6 +20,7 @@ 
 #include "defs.h"
 #include "arch-utils.h"
 #include "dis-asm.h"
+#include "dwarf2.h"
 #include "dwarf2-frame.h"
 #include "floatformat.h"
 #include "frame.h"
@@ -1536,6 +1537,31 @@  sparc32_dwarf2_frame_init_reg (struct gdbarch *gdbarch, int regnum,
     }
 }
 
+/* Implement the dwarf_cfa_op method.  */
+
+static void
+sparc_dwarf_cfa_op (struct gdbarch *gdbarch, gdb_byte op,
+		    struct dwarf2_frame_state *fs)
+{
+  /* Only DW_CFA_GNU_window_save is expected on SPARC.  */
+  gdb_assert (op == DW_CFA_GNU_window_save);
+
+  uint64_t reg;
+  int size = register_size (gdbarch, 0);
+
+  dwarf2_frame_state_alloc_regs (&fs->regs, 32);
+  for (reg = 8; reg < 16; reg++)
+    {
+      fs->regs.reg[reg].how = DWARF2_FRAME_REG_SAVED_REG;
+      fs->regs.reg[reg].loc.reg = reg + 16;
+    }
+  for (reg = 16; reg < 32; reg++)
+    {
+      fs->regs.reg[reg].how = DWARF2_FRAME_REG_SAVED_OFFSET;
+      fs->regs.reg[reg].loc.offset = (reg - 16) * size;
+    }
+}
+
 
 /* The SPARC Architecture doesn't have hardware single-step support,
    and most operating systems don't implement it either, so we provide
@@ -1801,6 +1827,8 @@  sparc32_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches)
 
   /* Hook in the DWARF CFI frame unwinder.  */
   dwarf2_frame_set_init_reg (gdbarch, sparc32_dwarf2_frame_init_reg);
+  /* Register DWARF CFA handler.  */
+  set_gdbarch_dwarf_cfa_op (gdbarch, sparc_dwarf_cfa_op);
   /* FIXME: kettenis/20050423: Don't enable the unwinder until the
      StackGhost issues have been resolved.  */