hi, there!

On Fri, 15 Sep 2000, John Polstra wrote:

> > there were some problems with pthreads initialization (but David will not
> > like the way I did it -- some files in contrib/gcc should be changed).
> > this happened because we initialize pthreads from static object
> > constructor (which is not the right thing anyway).
> 
> It seemed like a good idea at the time. :-) What is the problem --
> is it that libc_r is initialized at the wrong time?  What would you
> suggest as a better method for initializing the library?  I'd hate to
> introduce a bunch of "if (!initted) { ... }" checks, since those add
> runtime overhead on every call.

`__register_frame_info' should be called from `do_ctors' in
src/lib/csu/common/crtbegin.c to load frame information from .eh_frame
sections before any constructors are executed because try/catch can be
used in constructors of static objects (`__register_frame_info'
is defined in src/contrib/gcc/frame.c, this file is linked into
libgcc[_r].a). however `__register_frame_info' uses locks and other
threading stuff (using pthreads on FreeBSD) so we must have pthreads
initialized before locks are used.

another issue with our pthreads initialization is that pthreads can be
used in static object constructors so the first `pthread_xxx' call can
happen before `_thread_init' is called from constructor of our
_thread_init_invoker (it depends on order in which constructors are placed
in .ctors section).

I see two solutions (both seem to be hacks for me):
1) we can insert if (!initted) ... in `pthread_once' (this seem to be the
first pthreads function called from __register_frame_info)
2) we can insert 

#if __GTHREADS
  _thread_init();
#endif
 
   at the very beginning of `__register_frame_info'

The second approach seems to be cleaner but require changes in
src/contrib/gcc/

/fjoe

PS I have attached patches and Makefile.inc which should be placed
into src/lib/csu. Makefile.inc can be slightly out of date (I made it
somewhere around May/Jun 2000)
--- contrib/gcc/config/freebsd.h.orig   Wed Jul 19 07:30:53 2000
+++ contrib/gcc/config/freebsd.h        Sun Jul 23 17:59:33 2000
@@ -137,12 +137,6 @@
 #undef  HANDLE_SYSV_PRAGMA
 #define HANDLE_SYSV_PRAGMA
 
-/* FreeBSD ELF using our home-grown crtbegin.o/crtend.o does not support the
-   DWARF2 unwinding mechanisms.  Once `make world' bootstraping problems with
-   the EGCS crtstuff.c is overcome, we will switch to the non-sjlj-exceptions 
-   type exception machanism.  */
-#define DWARF2_UNWIND_INFO 0
-
 /* Do not use ``thunks'' to implement C++ vtables.  This method still has
    fatal bugs.  Also, GCC 3.0 will have a new C++ ABI that may not even
    support `thunks'.  */
--- contrib/gcc/frame.c.orig    Mon Jul 24 00:29:25 2000
+++ contrib/gcc/frame.c Mon Jul 24 00:33:28 2000
@@ -725,6 +725,10 @@
 void
 __register_frame_info (void *begin, struct object *ob)
 {
+#if __GTHREADS
+  _thread_init();
+#endif
+
   ob->fde_begin = begin;
 
   ob->pc_begin = ob->pc_end = 0;
--- lib/csu/common/crtbegin.c.orig      Mon Jul 10 16:15:25 2000
+++ lib/csu/common/crtbegin.c   Mon Jul 24 00:36:25 2000
@@ -26,6 +26,9 @@
  */
 
 #include <sys/param.h>
+#include "tm.h"
+#include "defaults.h"
+#include <frame.h>
 
 #define ABI_VENDOR     "FreeBSD"
 #define ABI_SECTION    ".note.ABI-tag"
@@ -33,13 +36,23 @@
 
 typedef void (*fptr)(void);
 
+extern void __register_frame_info(void *, struct object *)
+                       __attribute__((weak));
+extern void *__deregister_frame_info(void *)
+                       __attribute__((weak));
+
 static fptr ctor_list[1] __attribute__((section(".ctors"))) = { (fptr) -1 };
 static fptr dtor_list[1] __attribute__((section(".dtors"))) = { (fptr) -1 };
+static char eh_frame_begin[] __attribute__((section(".eh_frame"))) = { };
 
 static void
 do_ctors(void)
 {
     fptr *fpp;
+    static struct object object;
+
+    if (__register_frame_info)
+       __register_frame_info(eh_frame_begin, &object);
 
     for(fpp = ctor_list + 1;  *fpp != 0;  ++fpp)
        ;
@@ -54,6 +67,9 @@
 
     for(fpp = dtor_list + 1;  *fpp != 0;  ++fpp)
        (**fpp)();
+
+    if (__deregister_frame_info)
+       __deregister_frame_info(eh_frame_begin);
 }
 
 /*
--- lib/csu/common/crtend.c.orig        Mon Jul 10 16:15:28 2000
+++ lib/csu/common/crtend.c     Sun Jul 23 17:59:33 2000
@@ -26,8 +26,11 @@
  */
 
 #include <sys/cdefs.h>
+#include <sys/types.h>
 
 typedef void (*fptr)(void);
 
 static fptr ctor_end[1] __attribute__((section(".ctors"))) __unused = { 0 };
 static fptr dtor_end[1] __attribute__((section(".dtors"))) __unused = { 0 };
+static u_int32_t eh_frame_end[1] __attribute__((section(".eh_frame")))
+                                __unused = { 0 };
TARGET_ARCH?=   ${MACHINE_ARCH}

.if ${TARGET_ARCH} == "mipsel" || ${TARGET_ARCH} == "mipseb"
GCC_ARCH=       mips
.else 
GCC_ARCH=       ${TARGET_ARCH}
.endif
GCCDIR=         ${.CURDIR}/../../../contrib/gcc

GENSRCS=        tm.h
SRCS+=          ${GENSRCS}
CLEANFILES+=    ${GENSRCS}
CFLAGS+=        -I. -I${GCCDIR} -I${GCCDIR}/config\
                -I${.CURDIR}/../../../gnu/usr.bin/cc/cc_tools

#       KEEP THIS IN SYNC with src/gcc/lib/libgcc/Makefile !!
tm.h:
        echo '#include "${GCC_ARCH}/${GCC_ARCH}.h"'     > ${.TARGET}
.if ${GCC_ARCH} == "i386"
        echo '#include "${GCC_ARCH}/att.h"'             >> ${.TARGET}
.endif
        echo '#include <freebsd.h>'                     >> ${.TARGET}
        echo '#include "${GCC_ARCH}/freebsd.h"'         >> ${.TARGET}
.if ${GCC_ARCH} == "i386"
        echo '#include "${GCC_ARCH}/perform.h"'         >> ${.TARGET}
.endif
        echo '#include <freebsd-native.h>'              >> ${.TARGET}

Reply via email to