elf.c (elf_syminfo): Loop over the elf_syminfo_data chain.
authorAlexander Monakov <amonakov@ispras.ru>
Tue, 23 Jul 2013 09:17:10 +0000 (13:17 +0400)
committerAlexander Monakov <amonakov@gcc.gnu.org>
Tue, 23 Jul 2013 09:17:10 +0000 (13:17 +0400)
* elf.c (elf_syminfo): Loop over the elf_syminfo_data chain.

From-SVN: r201159

libbacktrace/ChangeLog
libbacktrace/elf.c

index 1f1d21e4829d7aee7bbd0e21051ddd13cb37472b..d6f7205f1d1bbd3051611d4252380933fc84b570 100644 (file)
@@ -1,3 +1,7 @@
+2013-07-23  Alexander Monakov  <amonakov@ispras.ru>
+
+       * elf.c (elf_syminfo): Loop over the elf_syminfo_data chain.
+
 2013-07-23  Alexander Monakov  <amonakov@ispras.ru>
 
        * elf.c (backtrace_initialize): Pass elf_fileline_fn to
index 36761899458b6859ea42a8fbd0a970442f72812c..c1dbc5492e0ef261def2b99c7c177890c1172593 100644 (file)
@@ -454,12 +454,46 @@ elf_syminfo (struct backtrace_state *state, uintptr_t pc,
             void *data)
 {
   struct elf_syminfo_data *edata;
-  struct elf_symbol *sym;
+  struct elf_symbol *sym = NULL;
+
+  if (!state->threaded)
+    {
+      for (edata = (struct elf_syminfo_data *) state->syminfo_data;
+          edata != NULL;
+          edata = edata->next)
+       {
+         sym = ((struct elf_symbol *)
+                bsearch (&pc, edata->symbols, edata->count,
+                         sizeof (struct elf_symbol), elf_symbol_search));
+         if (sym != NULL)
+           break;
+       }
+    }
+  else
+    {
+      struct elf_syminfo_data **pp;
+
+      pp = (struct elf_syminfo_data **) (void *) &state->syminfo_data;
+      while (1)
+       {
+         edata = *pp;
+         /* Atomic load.  */
+         while (!__sync_bool_compare_and_swap (pp, edata, edata))
+           edata = *pp;
+
+         if (edata == NULL)
+           break;
+
+         sym = ((struct elf_symbol *)
+                bsearch (&pc, edata->symbols, edata->count,
+                         sizeof (struct elf_symbol), elf_symbol_search));
+         if (sym != NULL)
+           break;
+
+         pp = &edata->next;
+       }
+    }
 
-  edata = (struct elf_syminfo_data *) state->syminfo_data;
-  sym = ((struct elf_symbol *)
-        bsearch (&pc, edata->symbols, edata->count,
-                 sizeof (struct elf_symbol), elf_symbol_search));
   if (sym == NULL)
     callback (data, pc, NULL, 0);
   else