olga / rpms / glibc

Forked from rpms/glibc 5 years ago
Clone

Blame SOURCES/glibc-rh1165212.patch

00db10
#
00db10
# This is a special patch for rhel-6 to fix recursive dlopen.
00db10
# It is likely the upstream patch will always be too risky for
00db10
# rhel-6 and will involve reorganizing the way in which recursive
00db10
# dlopen is allowed to operate and how the _r_debug and stap
00db10
# points are used by gdb for the recursive case.
00db10
#
00db10
# This fix changes the internal API to duplicate the ldconfig
00db10
# cache data. This means that at any point the cache can be
00db10
# unmapped without any consequences. The caller is responsible
00db10
# fore freeing the returned string.
00db10
#
00db10
# A regression test is added to verify the assertion for _r_debug
00db10
# is no longer triggered due to the recursive dlopen. The test to
00db10
# verify the fix in _dl_load_cache_lookup is not automated and
00db10
# has to be run by hand.
00db10
#
00db10
# The original version of this patch was based on the first version
00db10
# of the upstream patch posted here:
00db10
#   https://sourceware.org/ml/libc-alpha/2014-12/msg00446.html
00db10
# The current version has been modified to reflect the changes
00db10
# made in the revision of the patch committed to trunk after
00db10
# being posted for review here:
00db10
#   https://sourceware.org/ml/libc-alpha/2014-12/msg00483.html
00db10
00db10
This was committed upstream as:
00db10
00db10
commit ccdb048df457d581f6ac7ede8b0c7a593a891dfa
00db10
Author: Carlos O'Donell <carlos@systemhalted.org>
00db10
Date:   Wed Jan 21 01:51:10 2015 -0500
00db10
00db10
    Fix recursive dlopen.
00db10
00db10
--- glibc-2.17-c758a686/dlfcn/Makefile	2012-12-24 22:02:13.000000000 -0500
00db10
+++ glibc-2.17-c758a686/dlfcn/Makefile	2015-06-22 12:44:41.000000000 -0400
00db10
@@ -35,12 +35,12 @@ endif
00db10
 ifeq (yes,$(build-shared))
00db10
 tests = glrefmain failtest tst-dladdr default errmsg1 tstcxaatexit \
00db10
 	bug-dlopen1 bug-dlsym1 tst-dlinfo bug-atexit1 bug-atexit2 \
00db10
-	bug-atexit3 tstatexit
00db10
+	bug-atexit3 tstatexit tst-rec-dlopen
00db10
 endif
00db10
 modules-names = glreflib1 glreflib2 glreflib3 failtestmod defaultmod1 \
00db10
 		defaultmod2 errmsg1mod modatexit modcxaatexit \
00db10
 		bug-dlsym1-lib1 bug-dlsym1-lib2 bug-atexit1-lib \
00db10
-		bug-atexit2-lib bug-atexit3-lib
00db10
+		bug-atexit2-lib bug-atexit3-lib moddummy1 moddummy2
00db10
 
00db10
 failtestmod.so-no-z-defs = yes
00db10
 glreflib2.so-no-z-defs = yes
00db10
@@ -122,6 +122,8 @@ LDLIBS-bug-atexit3-lib.so = -lstdc++ -lg
00db10
 $(objpfx)bug-atexit3: $(libdl)
00db10
 $(objpfx)bug-atexit3.out: $(objpfx)bug-atexit3-lib.so
00db10
 
00db10
+$(objpfx)tst-rec-dlopen: $(libdl)
00db10
+$(objpfx)tst-rec-dlopen.out: $(objpfx)moddummy1.so $(objpfx)moddummy2.so
00db10
 
00db10
 # Depend on libc.so so a DT_NEEDED is generated in the shared objects.
00db10
 # This ensures they will load libc.so for needed symbols if loaded by
00db10
--- glibc-2.17-c758a686/elf/dl-cache.c	2012-12-24 22:02:13.000000000 -0500
00db10
+++ glibc-2.17-c758a686/elf/dl-cache.c	2015-06-22 12:44:39.000000000 -0400
00db10
@@ -174,9 +174,12 @@ _dl_cache_libcmp (const char *p1, const
00db10
 
00db10
 
00db10
 /* Look up NAME in ld.so.cache and return the file name stored there,
00db10
-   or null if none is found.  */
00db10
-
00db10
-const char *
00db10
+   or null if none is found. 
00db10
+   The caller is responsible for freeing the returned string.  The ld.so.cache
00db10
+   may be unmapped at any time by a completing recursive dlopen and
00db10
+   this function must take care that it does not return references to
00db10
+   any data in the mapping.  */
00db10
+char *
00db10
 internal_function
00db10
 _dl_load_cache_lookup (const char *name)
00db10
 {
00db10
@@ -289,7 +292,17 @@ _dl_load_cache_lookup (const char *name)
00db10
       && best != NULL)
00db10
     _dl_debug_printf ("  trying file=%s\n", best);
00db10
 
00db10
-  return best;
00db10
+  if (best == NULL)
00db10
+    return NULL;
00db10
+
00db10
+  /* The double copy is *required* since malloc may be interposed
00db10
+     and call dlopen itself whose completion would unmap the data
00db10
+     we are accessing. Therefore we must make the copy of the
00db10
+     mapping data without using malloc.  */
00db10
+  char *temp;
00db10
+  temp = alloca (strlen (best) + 1);
00db10
+  strcpy (temp, best);
00db10
+  return strdup (temp);
00db10
 }
00db10
 
00db10
 #ifndef MAP_COPY
00db10
--- glibc-2.17-c758a686/elf/dl-load.c	2015-06-22 12:41:10.748836414 -0400
00db10
+++ glibc-2.17-c758a686/elf/dl-load.c	2015-06-22 12:44:39.000000000 -0400
00db10
@@ -2232,7 +2232,7 @@ _dl_map_object (struct link_map *loader,
00db10
 	{
00db10
 	  /* Check the list of libraries in the file /etc/ld.so.cache,
00db10
 	     for compatibility with Linux's ldconfig program.  */
00db10
-	  const char *cached = _dl_load_cache_lookup (name);
00db10
+	  char *cached = _dl_load_cache_lookup (name);
00db10
 
00db10
 	  if (cached != NULL)
00db10
 	    {
00db10
@@ -2262,6 +2262,7 @@ _dl_map_object (struct link_map *loader,
00db10
 		      if (memcmp (cached, dirp, system_dirs_len[cnt]) == 0)
00db10
 			{
00db10
 			  /* The prefix matches.  Don't use the entry.  */
00db10
+			  free (cached);
00db10
 			  cached = NULL;
00db10
 			  break;
00db10
 			}
00db10
@@ -2278,14 +2279,9 @@ _dl_map_object (struct link_map *loader,
00db10
 				    &fb, loader ?: GL(dl_ns)[nsid]._ns_loaded,
00db10
 				    LA_SER_CONFIG, &found_other_class, false);
00db10
 		  if (__builtin_expect (fd != -1, 1))
00db10
-		    {
00db10
-		      realname = local_strdup (cached);
00db10
-		      if (realname == NULL)
00db10
-			{
00db10
-			  __close (fd);
00db10
-			  fd = -1;
00db10
-			}
00db10
-		    }
00db10
+		    realname = cached;
00db10
+		  else
00db10
+		    free (cached);
00db10
 		}
00db10
 	    }
00db10
 	}
00db10
--- glibc-2.17-c758a686/elf/dl-open.c	2015-06-22 12:41:16.348913620 -0400
00db10
+++ glibc-2.17-c758a686/elf/dl-open.c	2015-06-22 12:44:40.000000000 -0400
00db10
@@ -221,7 +221,11 @@ dl_open_worker (void *a)
00db10
 	}
00db10
     }
00db10
 
00db10
-  assert (_dl_debug_initialize (0, args->nsid)->r_state == RT_CONSISTENT);
00db10
+  /* One might be tempted to assert that we are RT_CONSISTENT at this point, but that
00db10
+     may not be true if this is a recursive call to dlopen.
00db10
+     TODO: Fix all of the debug state so we end up at RT_CONSISTENT only when the last
00db10
+     recursive dlopen completes.  */
00db10
+  _dl_debug_initialize (0, args->nsid);
00db10
 
00db10
   /* Load the named object.  */
00db10
   struct link_map *new;
00db10
--- glibc-2.17-c758a686/sysdeps/generic/ldsodefs.h	2015-06-22 12:41:16.328913344 -0400
00db10
+++ glibc-2.17-c758a686/sysdeps/generic/ldsodefs.h	2015-06-22 12:44:41.000000000 -0400
00db10
@@ -895,8 +895,8 @@
00db10
      internal_function;
00db10
 
00db10
 /* Look up NAME in ld.so.cache and return the file name stored there,
00db10
-   or null if none is found.  */
00db10
-extern const char *_dl_load_cache_lookup (const char *name)
00db10
+   or null if none is found.  Caller must free returned string.  */
00db10
+extern char *_dl_load_cache_lookup (const char *name)
00db10
      internal_function;
00db10
 
00db10
 /* If the system does not support MAP_COPY we cannot leave the file open
00db10
--- glibc-2.17-c758a686/dlfcn/tst-rec-dlopen.c	1969-12-31 19:00:00.000000000 -0500
00db10
+++ glibc-2.17-c758a686/dlfcn/tst-rec-dlopen.c	2015-06-22 12:44:41.000000000 -0400
00db10
@@ -0,0 +1,143 @@
00db10
+/* Test recursive dlopen using malloc hooks.
00db10
+   Copyright (C) 2015 Free Software Foundation, Inc.
00db10
+   This file is part of the GNU C Library.
00db10
+
00db10
+   The GNU C Library is free software; you can redistribute it and/or
00db10
+   modify it under the terms of the GNU Lesser General Public
00db10
+   License as published by the Free Software Foundation; either
00db10
+   version 2.1 of the License, or (at your option) any later version.
00db10
+
00db10
+   The GNU C Library is distributed in the hope that it will be useful,
00db10
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
00db10
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
00db10
+   Lesser General Public License for more details.
00db10
+
00db10
+   You should have received a copy of the GNU Lesser General Public
00db10
+   License along with the GNU C Library; if not, see
00db10
+   <http://www.gnu.org/licenses/>.  */
00db10
+
00db10
+#include <stdio.h>
00db10
+#include <stdlib.h>
00db10
+#include <malloc.h>
00db10
+#include <dlfcn.h>
00db10
+
00db10
+#define DSO "moddummy1.so"
00db10
+#define FUNC "dummy1"
00db10
+
00db10
+#define DSO1 "moddummy2.so"
00db10
+#define FUNC1 "dummy2"
00db10
+
00db10
+/* Result of the called function.  */
00db10
+int func_result;
00db10
+
00db10
+/* Prototype for my hook.  */
00db10
+void *custom_malloc_hook (size_t, const void *);
00db10
+
00db10
+/* Pointer to old malloc hooks.  */
00db10
+void *(*old_malloc_hook) (size_t, const void *);
00db10
+
00db10
+/* Call function func_name in DSO dso_name via dlopen.  */
00db10
+void
00db10
+call_func (const char *dso_name, const char *func_name)
00db10
+{
00db10
+  int ret;
00db10
+  void *dso;
00db10
+  int (*func) (void);
00db10
+  char *err;
00db10
+
00db10
+  /* Open the DSO.  */
00db10
+  dso = dlopen (dso_name, RTLD_NOW|RTLD_GLOBAL);
00db10
+  if (dso == NULL)
00db10
+    {
00db10
+      err = dlerror ();
00db10
+      fprintf (stderr, "%s\n", err);
00db10
+      exit (1);
00db10
+    }
00db10
+  /* Clear any errors.  */
00db10
+  dlerror ();
00db10
+
00db10
+  /* Lookup func.  */
00db10
+  func = (int (*) (void)) dlsym (dso, func_name);
00db10
+  if (func == NULL)
00db10
+    {
00db10
+      err = dlerror ();
00db10
+      if (err != NULL)
00db10
+        {
00db10
+	  fprintf (stderr, "%s\n", err);
00db10
+	  exit (1);
00db10
+        }
00db10
+    }
00db10
+  /* Call func.  */
00db10
+  func_result = (*func) ();
00db10
+
00db10
+  /* Close the library and look for errors too.  */
00db10
+  ret = dlclose (dso);
00db10
+  if (ret != 0)
00db10
+    {
00db10
+      err = dlerror ();
00db10
+      fprintf (stderr, "%s\n", err);
00db10
+      exit (1);
00db10
+    }
00db10
+
00db10
+}
00db10
+
00db10
+/* Empty hook that does nothing.  */
00db10
+void *
00db10
+custom_malloc_hook (size_t size, const void *caller)
00db10
+{
00db10
+  void *result;
00db10
+  /* Restore old hooks.  */
00db10
+  __malloc_hook = old_malloc_hook;
00db10
+  /* First call a function in another library via dlopen.  */
00db10
+  call_func (DSO1, FUNC1);
00db10
+  /* Called recursively.  */
00db10
+  result = malloc (size);
00db10
+  /* Restore new hooks.  */
00db10
+  __malloc_hook = custom_malloc_hook;
00db10
+  return result;
00db10
+}
00db10
+
00db10
+static int
00db10
+do_test (void)
00db10
+{
00db10
+  /* Save old hook.  */
00db10
+  old_malloc_hook = __malloc_hook;
00db10
+  /* Install new hook.  */
00db10
+  __malloc_hook = custom_malloc_hook;
00db10
+
00db10
+  /* Bug 17702 fixes two things:
00db10
+       * A recursive dlopen unmapping the ld.so.cache.
00db10
+       * An assertion that _r_debug is RT_CONSISTENT at entry to dlopen.
00db10
+     We can only test the latter. Testing the former requires modifying
00db10
+     ld.so.conf to cache the dummy libraries, then running ldconfig,
00db10
+     then run the test. If you do all of that (and glibc's test
00db10
+     infrastructure doesn't support that yet) then the test will
00db10
+     SEGFAULT without the fix. If you don't do that, then the test
00db10
+     will abort because of the assert described in detail below.  */
00db10
+  call_func (DSO, FUNC);
00db10
+
00db10
+  /* Restore old hook.  */
00db10
+  __malloc_hook = old_malloc_hook;
00db10
+
00db10
+  /* The function dummy2() is called by the malloc hook. Check to
00db10
+     see that it was called. This ensures the second recursive
00db10
+     dlopen happened and we called the function in that library.
00db10
+     Before the fix you either get a SIGSEGV when accessing mmap'd
00db10
+     ld.so.cache data or an assertion failure about _r_debug not
00db10
+     beint RT_CONSISTENT.  We don't test for the SIGSEGV since it
00db10
+     would require finding moddummy1 or moddummy2 in the cache and
00db10
+     we don't have any infrastructure to test that, but the _r_debug
00db10
+     assertion triggers.  */
00db10
+  printf ("Returned result is %d\n", func_result);
00db10
+  if (func_result <= 0)
00db10
+    {
00db10
+      printf ("FAIL: Function call_func() not called.\n");
00db10
+      exit (1);
00db10
+    }
00db10
+
00db10
+  printf ("PASS: Function call_func() called more than once.\n");
00db10
+  return 0;
00db10
+}
00db10
+
00db10
+#define TEST_FUNCTION do_test ()
00db10
+#include "../test-skeleton.c"
00db10
--- glibc-2.17-c758a686/dlfcn/moddummy1.c	1969-12-31 19:00:00.000000000 -0500
00db10
+++ glibc-2.17-c758a686/dlfcn/moddummy1.c	2015-06-22 12:44:41.000000000 -0400
00db10
@@ -0,0 +1,10 @@
00db10
+/* Provide a dummy DSO for tst-rec-dlopen to use.  */
00db10
+#include <stdio.h>
00db10
+#include <stdlib.h>
00db10
+
00db10
+int
00db10
+dummy1 (void)
00db10
+{
00db10
+  printf ("Called dummy1()\n");
00db10
+  return 1;
00db10
+}
00db10
--- glibc-2.17-c758a686/dlfcn/moddummy2.c	1969-12-31 19:00:00.000000000 -0500
00db10
+++ glibc-2.17-c758a686/dlfcn/moddummy2.c	2015-06-22 12:44:41.000000000 -0400
00db10
@@ -0,0 +1,13 @@
00db10
+/* Provide a dummy DSO for tst-rec-dlopen to use.  */
00db10
+#include <stdio.h>
00db10
+#include <stdlib.h>
00db10
+
00db10
+int
00db10
+dummy2 (void)
00db10
+{
00db10
+  printf ("Called dummy2()\n");
00db10
+  /* If the outer dlopen is not dummy1 (becuase of some error)
00db10
+     then tst-rec-dlopen will see a value of -1 as the returned
00db10
+     result and fail.  */
00db10
+  return -1;
00db10
+}