Skip site navigation (1)Skip section navigation (2)
Date:      Tue, 16 Oct 2018 15:29:45 -0700
From:      Mark Millard <marklmi@yahoo.com>
To:        FreeBSD PowerPC ML <freebsd-ppc@freebsd.org>, FreeBSD Toolchain <freebsd-toolchain@freebsd.org>, FreeBSD <freebsd-hackers@freebsd.org>
Cc:        Justin Hibbits <chmeeedalf@gmail.com>
Subject:   How to get devel/powerpc-gcc based WITH_LIBCPLUSPLUS= buildworld to have some throwing of C++ exceptions work (patch)
Message-ID:  <86E4687B-280A-4625-A56E-8D6FC4C4675B@yahoo.com>

next in thread | raw e-mail | index | archive | help
I now have a patch that gets some basic C++ exception throwing
going for WITH_LIBCPLUSPLUS=3D use when building via
devel/powerpc64-gcc . But the overall mechanism seems to mess
up the handling of powerpc64's "red zone" style of stack
processing in various cases.

I've had recent list submittals reporting that buildworld
using WITH_LIBCPLUSPLUS=3D based on devel/powerpc64-gcc would
get stuck looping in _Unwind_RaiseException. This prevented
use of devel/gdb --which makes extensive use of throwing C++
exceptions in normal operation.

Well, I now have a patch that avoids the problem in
libcxxrt's throw_exception itself that made all throws
get stuck --and so allows some C++ exceptions to be thrown.
See below. (I'm not sure leading spaces will all be
preserved.) Most of text is commentary, not code.

# svnlite diff /usr/src/contrib/libcxxrt/
Index: /usr/src/contrib/libcxxrt/exception.cc
=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=
=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=
=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D
--- /usr/src/contrib/libcxxrt/exception.cc	(revision 339076)
+++ /usr/src/contrib/libcxxrt/exception.cc	(working copy)
@@ -772,10 +772,71 @@
 	info->globals.uncaughtExceptions++;
=20
 	_Unwind_Reason_Code err =3D =
_Unwind_RaiseException(&ex->unwindHeader);
+#if !defined(__powerpc64__) && !defined(__ppc64__)
 	// The _Unwind_RaiseException() function should not return, it =
should
 	// unwind the stack past this function.  If it does return, then =
something
 	// has gone wrong.
 	report_failure(err, ex);
+#else
+// NOTE: Only tested for devel/powerpc64-gcc based buildworld
+//       because clang still silently ignores
+//       __builtin_eh_return(offset,handler) for powerpc64
+//       (and powerpc), thus not generating correct output.
+//
+// NOTE: I've no clue if other archtiectures might have
+//       analogous issues to powerpc64. I'm not sure
+//       about powerpc because of it still being stuck
+//       at gcc 4.2.1 . (clang problems and no devel/powerpc-gcc .)
+//
+// The above/normal code produced the following sort of structure
+// for throw_exception. r1 is the stack pointer, note its adjustments
+// via stdu r1,-128(r1) and via addi r1,r1,128 .
+//
+// <throw_exception+0>:	mflr    r0
+// <throw_exception+4>:	std     r31,-8(r1)
+// <throw_exception+8>:	mr      r31,r3
+// <throw_exception+12>:	std     r0,16(r1)
+// <throw_exception+16>:	stdu    r1,-128(r1)
+// . . .
+// <throw_exception+140>:	bl      =
<00000018.plt_call._Unwind_RaiseException@@GCC_3.0>
+// <throw_exception+144>:	ld      r2,40(r1)
+// <throw_exception+148>:	addi    r1,r1,128
+// <throw_exception+152>:	mr      r4,r31
+// <throw_exception+156>:	ld      r0,16(r1)
+// <throw_exception+160>:	ld      r31,-8(r1)
+// <throw_exception+164>:	mtlr    r0
+// <throw_exception+168>:	b       <report_failure>
+//
+// The loop in __Unwind_RaiseException had its "fs"
+// used with uw_frame_state_for and uw_update_context get
+// stuck with the pc field having the address for
+// throw_exception+152 (just after the stack adjustment
+// addi r1,r1,128). Effectively, throw_exception unwinds
+// its stack use before calling report_failure in a
+// way that throw_exception is no longer on the stack.
+// The exception unwinding logic did not handle this
+// correctly and got stuck looping.
+//
+// The below avoids having any such stack adjustment here
+// by avoiding the report_failure call and directly doing
+// what case _URC_END_OF_STACK in report_failure does for
+// its first couple of lines. (It is also the kind of
+// thing that src/contrib/libstdc++/libsupc++/eh_throw.cc
+// has in its __cxxabiv1::__cxa_throw after the
+// _Unwind_RaiseException call.)
+//
+// Another option could be to turn report_failure into
+// a macro so that no subroutine call could be involved.
+// That should avoid the early stack pointer kadjsutment.
+//
+// Also: For the other archtiectures that I looked at, no
+//       such stack adjsutments were involved in the code
+//       generated (or the matching dwarfdump output).
+//       But I did not look at many.
+
+	__cxa_begin_catch (&(ex->unwindHeader));
+	std::terminate();
+#endif
 }
=20
=20
However, code such as the following from devel/kyua leads to
other examples of _Unwind_RaiseException looping without
making progress. Note the stdu r1,-368(r1) and the
addi r1,r1,368 stack pointer adjustments and their timing
relative to stack usage (the "red zone" style used for
FreeBSD's powerpc64 ABI):

(gdb) x/64i 0x100a8528-88
   0x100a84d0 <utils::fs::mkdir(utils::fs::path const&, int)>:	mflr    =
r0
   0x100a84d4 <utils::fs::mkdir(utils::fs::path const&, int)+4>:	=
std     r30,-16(r1)
   0x100a84d8 <utils::fs::mkdir(utils::fs::path const&, int)+8>:	=
std     r31,-8(r1)
   0x100a84dc <utils::fs::mkdir(utils::fs::path const&, int)+12>:	=
std     r29,-24(r1)
   0x100a84e0 <utils::fs::mkdir(utils::fs::path const&, int)+16>:	=
mr      r31,r4
   0x100a84e4 <utils::fs::mkdir(utils::fs::path const&, int)+20>:	=
std     r0,16(r1)
   0x100a84e8 <utils::fs::mkdir(utils::fs::path const&, int)+24>:	=
stdu    r1,-368(r1)
   0x100a84ec <utils::fs::mkdir(utils::fs::path const&, int)+28>:	=
mr      r30,r3
   0x100a84f0 <utils::fs::mkdir(utils::fs::path const&, int)+32>:	=
bl      0x100abab0 <utils::fs::path::c_str() const>
   0x100a84f4 <utils::fs::mkdir(utils::fs::path const&, int)+36>:	=
nop
   0x100a84f8 <utils::fs::mkdir(utils::fs::path const&, int)+40>:	=
clrlwi  r4,r31,16
   0x100a84fc <utils::fs::mkdir(utils::fs::path const&, int)+44>:	=
bl      0x10009fc0 <000000af.plt_call.mkdir@@FBSD_1.0>
   0x100a8500 <utils::fs::mkdir(utils::fs::path const&, int)+48>:	=
ld      r2,40(r1)
   0x100a8504 <utils::fs::mkdir(utils::fs::path const&, int)+52>:	=
cmpwi   cr7,r3,-1
   0x100a8508 <utils::fs::mkdir(utils::fs::path const&, int)+56>:	=
beq     cr7,0x100a8528 <utils::fs::mkdir(utils::fs::path const&, =
int)+88>
   0x100a850c <utils::fs::mkdir(utils::fs::path const&, int)+60>:	=
addi    r1,r1,368
   0x100a8510 <utils::fs::mkdir(utils::fs::path const&, int)+64>:	=
ld      r0,16(r1)
   0x100a8514 <utils::fs::mkdir(utils::fs::path const&, int)+68>:	=
ld      r29,-24(r1)
   0x100a8518 <utils::fs::mkdir(utils::fs::path const&, int)+72>:	=
ld      r30,-16(r1)
   0x100a851c <utils::fs::mkdir(utils::fs::path const&, int)+76>:	=
ld      r31,-8(r1)
   0x100a8520 <utils::fs::mkdir(utils::fs::path const&, int)+80>:	=
mtlr    r0
   0x100a8524 <utils::fs::mkdir(utils::fs::path const&, int)+84>:	=
blr
   0x100a8528 <utils::fs::mkdir(utils::fs::path const&, int)+88>:	=
bl      0x10009d40 <000000af.plt_call.__error@@FBSD_1.0>
. . . (more not shown here) . . .

This goes along with (darfdump -v -v -F output):

< 1323><0x100a84d0:0x100a8670><mkdir><cie offset 0x0000f100::cie index   =
  1><fde offset 0x0000f3d8 length: 0x00000034>
       <eh aug data len 0x8 bytes 0x00 00 00 00 00 00 f3 c7 >
        0x100a84d0: <off cfa=3D00(r1) >=20
        0x100a84e0: <off cfa=3D00(r1) > <off r29=3D-24(cfa) > <off =
r30=3D-16(cfa) > <off r31=3D-8(cfa) > <off r65=3Dr0 >=20
        0x100a84ec: <off cfa=3D368(r1) > <off r29=3D-24(cfa) > <off =
r30=3D-16(cfa) > <off r31=3D-8(cfa) > <off r65=3D16(cfa) >=20
        0x100a8510: <off cfa=3D00(r1) > <off r29=3D-24(cfa) > <off =
r30=3D-16(cfa) > <off r31=3D-8(cfa) > <off r65=3D16(cfa) >=20
        0x100a8524: <off cfa=3D00(r1) >=20
        0x100a8528: <off cfa=3D368(r1) > <off r29=3D-24(cfa) > <off =
r30=3D-16(cfa) > <off r31=3D-8(cfa) > <off r65=3D16(cfa) >=20
 fde section offset 62424 0x0000f3d8 cie offset for fde: 61696 =
0x0000f100
         0 DW_CFA_advance_loc 16  (4 * 4)
         1 DW_CFA_register r65 =3D r0
         4 DW_CFA_offset r30 -16  (2 * -8)
         6 DW_CFA_offset r31 -8  (1 * -8)
         8 DW_CFA_offset r29 -24  (3 * -8)
        10 DW_CFA_advance_loc 12  (3 * 4)
        11 DW_CFA_def_cfa_offset 368
        14 DW_CFA_offset_extended_sf r65 16  (-2 * -8)
        17 DW_CFA_advance_loc 36  (9 * 4)
        18 DW_CFA_remember_state
        19 DW_CFA_def_cfa_offset 0
        21 DW_CFA_advance_loc 20  (5 * 4)
        22 DW_CFA_restore_extended r65
        24 DW_CFA_restore r31
        25 DW_CFA_restore r30
        26 DW_CFA_restore r29
        27 DW_CFA_advance_loc 4  (1 * 4)
        28 DW_CFA_restore_state
        29 DW_CFA_nop
        30 DW_CFA_nop


The _Unwind_RaiseException's fs ends up reaching and the
holding the following value in my attempted devel/kyua
run for FreeBSD's test suite:

(gdb) print fs
$9 =3D {regs =3D {reg =3D {{loc =3D {reg =3D 0, offset =3D 0, exp =3D =
0x0}, how =3D REG_UNSAVED} <repeats 29 times>, {loc =3D {reg =3D =
18446744073709551592, offset =3D -24,=20
          exp =3D 0xffffffffffffffe8 <error: Cannot access memory at =
address 0xffffffffffffffe8>}, how =3D REG_SAVED_OFFSET}, {loc =3D {reg =3D=
 18446744073709551600, offset =3D -16,=20
          exp =3D 0xfffffffffffffff0 <error: Cannot access memory at =
address 0xfffffffffffffff0>}, how =3D REG_SAVED_OFFSET}, {loc =3D {reg =3D=
 18446744073709551608, offset =3D -8,=20
          exp =3D 0xfffffffffffffff8 <error: Cannot access memory at =
address 0xfffffffffffffff8>}, how =3D REG_SAVED_OFFSET}, {loc =3D {reg =3D=
 0, offset =3D 0, exp =3D 0x0},=20
        how =3D REG_UNSAVED} <repeats 33 times>, {loc =3D {reg =3D 16, =
offset =3D 16, exp =3D 0x10 <error: Cannot access memory at address =
0x10>}, how =3D REG_SAVED_OFFSET}, {loc =3D {reg =3D 0, offset =3D 0,=20
          exp =3D 0x0}, how =3D REG_UNSAVED} <repeats 80 times>}, prev =3D=
 0x0}, cfa_offset =3D 0, cfa_reg =3D 1, cfa_exp =3D 0x0, cfa_how =3D =
CFA_REG_OFFSET,=20
  pc =3D 0x100a8528 <utils::fs::mkdir(utils::fs::path const&, int)+88>,=20=

  personality =3D @0x810549c40: 0x81053c900 <__gxx_personality_v0(int, =
_Unwind_Action, uint64_t, _Unwind_Exception*, _Unwind_Context*)>, =
data_align =3D -8, code_align =3D 4, retaddr_column =3D 65,=20
  fde_encoding =3D 27 '\033', lsda_encoding =3D 20 '\024', saw_z =3D 1 =
'\001', signal_frame =3D 0 '\000', eh_ptr =3D 0x0}

Note the invariant value it is looping with (fs.pc value):

pc =3D 0x100a8528 <utils::fs::mkdir(utils::fs::path const&, int)+88>

But the routine is at 0x00000000100a85ec :

(gdb) bt
#0  _Unwind_RaiseException (exc=3D0x8109582e0) at =
/usr/src/gnu/lib/libgcc/../../../contrib/gcc/unwind.inc:103
#1  0x000000081053d704 in throw_exception (ex=3D0x810958288) at =
/usr/src/contrib/libcxxrt/exception.cc:774
#2  0x00000000100a85ec in utils::fs::mkdir (dir=3D..., =
mode=3Dmode@entry=3D493) at utils/fs/operations.cpp:484
#3  0x00000000100a8694 in utils::fs::mkdir_p (dir=3D..., mode=3D<optimized=
 out>) at utils/fs/operations.cpp:502
#4  0x000000001000cbf4 in (anonymous namespace)::safe_main =
(mock_command=3D..., argv=3D0x3fffffffffffdb88, argc=3D4, =
ui=3D0x3fffffffffffd960, this=3D<optimized out>, this=3D<optimized out>) =
at cli/main.cpp:207
#5  cli::main (ui=3Dui@entry=3D0x3fffffffffffd960, argc=3Dargc@entry=3D4, =
argv=3Dargv@entry=3D0x3fffffffffffdb88, mock_command=3D...) at =
cli/main.cpp:280
#6  0x000000001000e9dc in cli::main (argc=3D<optimized out>, =
argv=3D0x3fffffffffffdb88) at cli/main.cpp:353
#7  0x000000001000c570 in main (argc=3D<optimized out>, argv=3D<optimized =
out>) at main.cpp:49

where utils::fs::mkdir:

   0x00000000100a85d0 <+256>:	bne     0x100a85f0 =
<utils::fs::mkdir(utils::fs::path const&, int)+288>
   0x00000000100a85d4 <+260>:	addis   r9,r2,-1
   0x00000000100a85d8 <+264>:	addis   r10,r2,-1
   0x00000000100a85dc <+268>:	mr      r3,r29
   0x00000000100a85e0 <+272>:	addi    r5,r9,20864
   0x00000000100a85e4 <+276>:	addi    r4,r10,-12216
   0x00000000100a85e8 <+280>:	bl      0x1000a140 =
<000000af.plt_call.__cxa_throw@@CXXABI_1.3>
=3D> 0x00000000100a85ec <+284>:	ld      r2,40(r1)
   0x00000000100a85f0 <+288>:	ld      r3,320(r1)
   0x00000000100a85f4 <+292>:	bl      0x1000a8c0 =
<000000af.plt_call._ZdlPv>
   0x00000000100a85f8 <+296>:	ld      r2,40(r1)
   0x00000000100a85fc <+300>:	b       0x100a85d4 =
<utils::fs::mkdir(utils::fs::path const&, int)+260>

This is for the devel/kyua code:

void
fs::mkdir(const fs::path& dir, const int mode)
{
    if (::mkdir(dir.c_str(), static_cast< mode_t >(mode)) =3D=3D -1) {
        const int original_errno =3D errno;
        throw fs::system_error(F("Failed to create directory %s") % dir,
                               original_errno);
    }
}

I've not figured out how to make general throwing of C++
exceptions avoid this _Unwind_RaiseException unbounded
looping with a fixed fs.pc value.

But at least now I can use devel/gdb for some of the
investigation's activity.



Other notes:

WITH_LLVM_LIBUNWIND=3D does not even compile for targeting
powerpc64 --so that is not currently a way around the
problem.

WITHOUT_LIB32=3D yse is because, for every post-gcc 4.2.1
that I've tried, the lib32 produced misuses R30 in
crtbeginS code (vs. the ABI for FreeBSD) and 32-bit
code just produces core files from a bad so-called
address that is dereferenced via R30 content.


=3D=3D=3D
Mark Millard
marklmi at yahoo.com
( dsl-only.net went
away in early 2018-Mar)




Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?86E4687B-280A-4625-A56E-8D6FC4C4675B>