summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'dev-lang/ghc/files/ghc-6.12.3-ticket-2615-linker-script.patch')
-rw-r--r--dev-lang/ghc/files/ghc-6.12.3-ticket-2615-linker-script.patch278
1 files changed, 278 insertions, 0 deletions
diff --git a/dev-lang/ghc/files/ghc-6.12.3-ticket-2615-linker-script.patch b/dev-lang/ghc/files/ghc-6.12.3-ticket-2615-linker-script.patch
new file mode 100644
index 000000000000..96fdcb6a8091
--- /dev/null
+++ b/dev-lang/ghc/files/ghc-6.12.3-ticket-2615-linker-script.patch
@@ -0,0 +1,278 @@
+Wed Dec 16 19:51:55 CET 2009 howard_b_golden@yahoo.com
+ * FIX #2615 (linker scripts in .so files)
+ This patch does not apply to Windows. It only applies to systems with
+ ELF binaries.
+
+ This is a patch to rts/Linker.c to recognize linker scripts in .so
+ files and find the real target .so shared library for loading.
+diff -rN -u old-ghc/rts/Linker.c new-ghc/rts/Linker.c
+--- old-ghc/rts/Linker.c 2010-07-28 22:59:33.532759118 +0200
++++ new-ghc/rts/Linker.c 2010-07-28 22:59:34.853759052 +0200
+@@ -39,6 +39,8 @@
+
+ #include <stdlib.h>
+ #include <string.h>
++#include <stdio.h>
++#include <assert.h>
+
+ #ifdef HAVE_SYS_STAT_H
+ #include <sys/stat.h>
+@@ -79,6 +81,8 @@
+
+ #if defined(linux_HOST_OS) || defined(solaris2_HOST_OS) || defined(freebsd_HOST_OS) || defined(dragonfly_HOST_OS) || defined(netbsd_HOST_OS) || defined(openbsd_HOST_OS)
+ # define OBJFORMAT_ELF
++# include <regex.h> // regex is already used by dlopen() so this is OK
++ // to use here without requiring an additional lib
+ #elif defined(cygwin32_HOST_OS) || defined (mingw32_HOST_OS)
+ # define OBJFORMAT_PEi386
+ # include <windows.h>
+@@ -1095,12 +1099,19 @@
+
+ #if defined(OBJFORMAT_ELF) || defined(OBJFORMAT_MACHO)
+ static void *dl_prog_handle;
++static regex_t re_invalid;
++static regex_t re_realso;
++static void initLinkerCleanup( void );
++#ifdef THREADED_RTS
++static Mutex dl_mutex; // mutex to protect dlopen/dlerror critical section
++#endif
+ #endif
+
+ void
+ initLinker( void )
+ {
+ RtsSymbolVal *sym;
++ int compileResult;
+
+ /* Make initLinker idempotent, so we can call it
+ before evey relevant operation; that means we
+@@ -1109,6 +1120,9 @@
+ linker_init_done = 1;
+ }
+
++#ifdef THREADED_RTS
++ initMutex(&dl_mutex);
++#endif
+ stablehash = allocStrHashTable();
+ symhash = allocStrHashTable();
+
+@@ -1127,6 +1141,15 @@
+ # else
+ dl_prog_handle = dlopen(NULL, RTLD_LAZY);
+ # endif /* RTLD_DEFAULT */
++
++ compileResult = regcomp(&re_invalid,
++ "(([^ \t()])+\\.so([^ \t:()])*):([ \t])*invalid ELF header",
++ REG_EXTENDED);
++ ASSERT( compileResult == 0 );
++ compileResult = regcomp(&re_realso,
++ "GROUP *\\( *(([^ )])+)",
++ REG_EXTENDED);
++ ASSERT( compileResult == 0 );
+ # endif
+
+ #if defined(x86_64_HOST_ARCH)
+@@ -1147,6 +1170,19 @@
+ #endif
+ }
+
++void
++exitLinker( void ) {
++#if defined(OBJFORMAT_ELF) || defined(OBJFORMAT_MACHO)
++ if (linker_init_done == 1) {
++ regfree(&re_invalid);
++ regfree(&re_realso);
++#ifdef THREADED_RTS
++ closeMutex(&dl_mutex);
++#endif
++ }
++#endif
++}
++
+ /* -----------------------------------------------------------------------------
+ * Loading DLL or .so dynamic libraries
+ * -----------------------------------------------------------------------------
+@@ -1182,29 +1218,112 @@
+ static OpenedDLL* opened_dlls = NULL;
+ #endif
+
+-const char *
+-addDLL( char *dll_name )
+-{
+ # if defined(OBJFORMAT_ELF) || defined(OBJFORMAT_MACHO)
+- /* ------------------- ELF DLL loader ------------------- */
+- void *hdl;
+- const char *errmsg;
+
+- initLinker();
++static char *
++internal_dlopen(const char *dll_name)
++{
++ void *hdl;
++ char *errmsg, *errmsg_copy;
+
+ // omitted: RTLD_NOW
+ // see http://www.haskell.org/pipermail/cvs-ghc/2007-September/038570.html
+- hdl= dlopen(dll_name, RTLD_LAZY | RTLD_GLOBAL);
++ IF_DEBUG(linker,
++ debugBelch("internal_dlopen: dll_name = '%s'\n", dll_name));
+
++ //-------------- Begin critical section ------------------
++ // This critical section is necessary because dlerror() is not
++ // required to be reentrant (see POSIX -- IEEE Std 1003.1-2008)
++ // Also, the error message returned must be copied to preserve it
++ // (see POSIX also)
++
++ ACQUIRE_LOCK(&dl_mutex);
++ hdl = dlopen(dll_name, RTLD_LAZY | RTLD_GLOBAL);
++
++ errmsg = NULL;
+ if (hdl == NULL) {
+ /* dlopen failed; return a ptr to the error msg. */
+ errmsg = dlerror();
+ if (errmsg == NULL) errmsg = "addDLL: unknown error";
+- return errmsg;
+- } else {
++ errmsg_copy = stgMallocBytes(strlen(errmsg)+1, "addDLL");
++ strcpy(errmsg_copy, errmsg);
++ errmsg = errmsg_copy;
++ }
++ RELEASE_LOCK(&dl_mutex);
++ //--------------- End critical section -------------------
++
++ return errmsg;
++}
++# endif
++
++const char *
++addDLL( char *dll_name )
++{
++# if defined(OBJFORMAT_ELF) || defined(OBJFORMAT_MACHO)
++ /* ------------------- ELF DLL loader ------------------- */
++
++#define NMATCH 5
++ regmatch_t match[NMATCH];
++ char *errmsg;
++ FILE* fp;
++ size_t match_length;
++#define MAXLINE 1000
++ char line[MAXLINE];
++ int result;
++
++ initLinker();
++
++ IF_DEBUG(linker, debugBelch("addDLL: dll_name = '%s'\n", dll_name));
++ errmsg = internal_dlopen(dll_name);
++
++ if (errmsg == NULL) {
+ return NULL;
+ }
+- /*NOTREACHED*/
++
++ // GHC Trac ticket #2615
++ // On some systems (e.g., Gentoo Linux) dynamic files (e.g. libc.so)
++ // contain linker scripts rather than ELF-format object code. This
++ // code handles the situation by recognizing the real object code
++ // file name given in the linker script.
++ //
++ // If an "invalid ELF header" error occurs, it is assumed that the
++ // .so file contains a linker script instead of ELF object code.
++ // In this case, the code looks for the GROUP ( ... ) linker
++ // directive. If one is found, the first file name inside the
++ // parentheses is treated as the name of a dynamic library and the
++ // code attempts to dlopen that file. If this is also unsuccessful,
++ // an error message is returned.
++
++ // see if the error message is due to an invalid ELF header
++ IF_DEBUG(linker, debugBelch("errmsg = '%s'\n", errmsg));
++ result = regexec(&re_invalid, errmsg, (size_t) NMATCH, match, 0);
++ IF_DEBUG(linker, debugBelch("result = %i\n", result));
++ if (result == 0) {
++ // success -- try to read the named file as a linker script
++ match_length = (size_t) stg_min((match[1].rm_eo - match[1].rm_so),
++ MAXLINE-1);
++ strncpy(line, (errmsg+(match[1].rm_so)),match_length);
++ line[match_length] = '\0'; // make sure string is null-terminated
++ IF_DEBUG(linker, debugBelch ("file name = '%s'\n", line));
++ if ((fp = fopen(line, "r")) == NULL) {
++ return errmsg; // return original error if open fails
++ }
++ // try to find a GROUP ( ... ) command
++ while (fgets(line, MAXLINE, fp) != NULL) {
++ IF_DEBUG(linker, debugBelch("input line = %s", line));
++ if (regexec(&re_realso, line, (size_t) NMATCH, match, 0) == 0) {
++ // success -- try to dlopen the first named file
++ IF_DEBUG(linker, debugBelch("match%s\n",""));
++ line[match[1].rm_eo] = '\0';
++ errmsg = internal_dlopen(line+match[1].rm_so);
++ break;
++ }
++ // if control reaches here, no GROUP ( ... ) directive was found
++ // and the original error message is returned to the caller
++ }
++ fclose(fp);
++ }
++ return errmsg;
+
+ # elif defined(OBJFORMAT_PEi386)
+ /* ------------------- Win32 DLL loader ------------------- */
+@@ -4195,7 +4314,7 @@
+ i++;
+ }
+ #endif
+- else
++ else
+ {
+ barf ("Don't know how to handle this Mach-O "
+ "scattered relocation entry: "
+@@ -4204,7 +4323,7 @@
+ oc->fileName, scat->r_type, scat->r_address);
+ return 0;
+ }
+-
++
+ #ifdef powerpc_HOST_ARCH
+ if(scat->r_type == GENERIC_RELOC_VANILLA
+ || scat->r_type == PPC_RELOC_SECTDIFF)
+@@ -4250,8 +4369,8 @@
+ "object file %s; entry type %ld; address %#lx\n",
+ oc->fileName, scat->r_type, scat->r_address);
+ return 0;
+- }
+-
++ }
++
+ }
+ else /* !(relocs[i].r_address & R_SCATTERED) */
+ {
+diff -rN -u old-ghc/rts/LinkerInternals.h new-ghc/rts/LinkerInternals.h
+--- old-ghc/rts/LinkerInternals.h 2010-07-28 22:59:33.532759118 +0200
++++ new-ghc/rts/LinkerInternals.h 2010-07-28 22:59:34.854758341 +0200
+@@ -109,4 +109,6 @@
+
+ extern ObjectCode *objects;
+
++void exitLinker( void );
++
+ #endif /* LINKERINTERNALS_H */
+diff -rN -u old-ghc/rts/RtsStartup.c new-ghc/rts/RtsStartup.c
+--- old-ghc/rts/RtsStartup.c 2010-07-28 22:59:33.537758009 +0200
++++ new-ghc/rts/RtsStartup.c 2010-07-28 22:59:34.859758558 +0200
+@@ -34,6 +34,7 @@
+ #include "Profiling.h"
+ #include "Timer.h"
+ #include "Globals.h"
++void exitLinker( void ); // there is no Linker.h file to include
+
+ #if defined(RTS_GTK_FRONTPANEL)
+ #include "FrontPanel.h"
+@@ -388,6 +389,9 @@
+ /* free shared Typeable store */
+ exitGlobalStore();
+
++ /* free linker data */
++ exitLinker();
++
+ /* free file locking tables, if necessary */
+ #if !defined(mingw32_HOST_OS)
+ freeFileLocking();