use an accessor function for __libc data pointer when compiled as PIC
authorRich Felker <dalias@aerifal.cx>
Mon, 21 Feb 2011 03:30:06 +0000 (22:30 -0500)
committerRich Felker <dalias@aerifal.cx>
Mon, 21 Feb 2011 03:30:06 +0000 (22:30 -0500)
prior to this change, a large portion of libc was unusable prior to
relocation by the dynamic linker, due to dependence on the global data
in the __libc structure and the need to obtain its address through the
GOT. with this patch, the accessor function __libc_loc is now able to
obtain the address of __libc via PC-relative addressing without using
the GOT. this means the majority of libc functionality is now
accessible right away.

naturally, the above statements all depend on having an architecture
where PC-relative addressing and jumps/calls are feasible, and a
compiler that generates the appropriate code.

src/internal/libc.c
src/internal/libc.h

index 5e8e9d95311f346013ce0c12c99f474fd98ea0ab..954efedf8b04eee03e09d26b36563097ef2c65db 100644 (file)
@@ -1,3 +1,11 @@
 #include "libc.h"
 
-struct libc libc;
+#ifdef __PIC__
+struct __libc *__libc_loc()
+{
+       static struct __libc __libc;
+       return &__libc;
+}
+#else
+struct __libc __libc;
+#endif
index ea221b6f9a4deeaba93e0879aeb333298009401f..8a84be0b90f144ccc661921c1e1553f7119d72fe 100644 (file)
@@ -4,8 +4,7 @@
 #include <stdlib.h>
 #include <stdio.h>
 
-#define libc __libc
-extern struct libc {
+struct __libc {
        void (*lock)(volatile int *);
        void (*cancelpt)(int);
        int (*atexit)(void (*)(void));
@@ -16,7 +15,15 @@ extern struct libc {
        int (*rsyscall)(int, long, long, long, long, long, long);
        void (**tsd_keys)(void *);
        void (*fork_handler)(int);
-} libc;
+};
+
+#ifdef __PIC__
+extern struct __libc *__libc_loc(void) __attribute__((const));
+#define libc (*__libc_loc())
+#else
+extern struct __libc __libc;
+#define libc __libc
+#endif
 
 
 /* Designed to avoid any overhead in non-threaded processes */