Avoid problems with plugins being loaded multiple times.
authorZenith423 <zenith432@users.sourceforge.net>
Thu, 23 Aug 2018 15:22:56 +0000 (16:22 +0100)
committerNick Clifton <nickc@redhat.com>
Thu, 23 Aug 2018 15:22:56 +0000 (16:22 +0100)
PR 23460
* plugin.c (struct plugin_list_entry): New structure.
(plugin_list): New variable.
(try_load_plugin): Place opened plugins on a list.  Ensure that
the refcount in the dynamic loader is kept at 1.

bfd/ChangeLog
bfd/plugin.c

index 97c8e0851c97d1d8bf49547c127cfa27d4830f9b..1cadfc4b7f554aecf2b66e9298fbd335699a198a 100644 (file)
@@ -1,3 +1,11 @@
+2018-08-23  Zenith423  <zenith432@users.sourceforge.net>
+
+       PR 23460
+       * plugin.c (struct plugin_list_entry): New structure.
+       (plugin_list): New variable.
+       (try_load_plugin): Place opened plugins on a list.  Ensure that
+       the refcount in the dynamic loader is kept at 1.
+
 2018-08-23  Alan Modra  <amodra@gmail.com>
 
        * elf64-ppc.c (ppc64_elf_relocate_section): Revise "call lacks
index d9b9e2f174e96c8348e1004e8fc40d91a5c3ffaa..fe3411531d7b8d2732ce680d3dd80db9c98c091d 100644 (file)
@@ -219,6 +219,15 @@ try_claim (bfd *abfd)
   return claimed;
 }
 
+struct plugin_list_entry
+{
+  void *                        handle;
+  ld_plugin_claim_file_handler  claim_file;
+  struct plugin_list_entry *    next;
+};
+
+static struct plugin_list_entry * plugin_list = NULL;
+
 static int
 try_load_plugin (const char *pname, bfd *abfd, int *has_plugin_p)
 {
@@ -227,9 +236,7 @@ try_load_plugin (const char *pname, bfd *abfd, int *has_plugin_p)
   int i;
   ld_plugin_onload onload;
   enum ld_plugin_status status;
-
-  if (claim_file)
-    goto have_claim_file;
+  struct plugin_list_entry *plugin_list_iter;
 
   *has_plugin_p = 0;
 
@@ -240,9 +247,30 @@ try_load_plugin (const char *pname, bfd *abfd, int *has_plugin_p)
       return 0;
     }
 
+  for (plugin_list_iter = plugin_list;
+       plugin_list_iter;
+       plugin_list_iter = plugin_list_iter->next)
+    {
+      if (plugin_handle == plugin_list_iter->handle)
+        {
+          dlclose (plugin_handle);
+          if (!plugin_list_iter->claim_file)
+            return 0;
+
+          register_claim_file (plugin_list_iter->claim_file);
+          goto have_claim_file;
+        }
+    }
+
+  plugin_list_iter = (struct plugin_list_entry *) xmalloc (sizeof *plugin_list_iter);
+  plugin_list_iter->handle = plugin_handle;
+  plugin_list_iter->claim_file = NULL;
+  plugin_list_iter->next = plugin_list;
+  plugin_list = plugin_list_iter;
+
   onload = dlsym (plugin_handle, "onload");
   if (!onload)
-    goto err;
+    return 0;
 
   i = 0;
   tv[i].tv_tag = LDPT_MESSAGE;
@@ -263,7 +291,9 @@ try_load_plugin (const char *pname, bfd *abfd, int *has_plugin_p)
   status = (*onload)(tv);
 
   if (status != LDPS_OK)
-    goto err;
+    return 0;
+
+  plugin_list_iter->claim_file = claim_file;
 
 have_claim_file:
   *has_plugin_p = 1;
@@ -271,20 +301,13 @@ have_claim_file:
   abfd->plugin_format = bfd_plugin_no;
 
   if (!claim_file)
-    goto err;
+    return 0;
 
   if (!try_claim (abfd))
-    goto err;
+    return 0;
 
   abfd->plugin_format = bfd_plugin_yes;
-
   return 1;
-
- err:
-  if (plugin_handle)
-    dlclose (plugin_handle);
-  register_claim_file (NULL);
-  return 0;
 }
 
 /* There may be plugin libraries in lib/bfd-plugins.  */