LCOV - code coverage report
Current view: top level - libdwfl_stacktrace - dwflst_process_tracker.c (source / functions) Coverage Total Hit
Test: elfutils-0.193 Lines: 0.0 % 96 0
Test Date: 2025-08-30 14:31:09 Functions: 0.0 % 6 0
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
Branches: 0.0 % 46 0

             Branch data     Line data    Source code
       1                 :             : /* Track multiple Dwfl structs for multiple processes.
       2                 :             :    Copyright (C) 2025, Red Hat, Inc.
       3                 :             :    This file is part of elfutils.
       4                 :             : 
       5                 :             :    This file is free software; you can redistribute it and/or modify
       6                 :             :    it under the terms of either
       7                 :             : 
       8                 :             :      * the GNU Lesser General Public License as published by the Free
       9                 :             :        Software Foundation; either version 3 of the License, or (at
      10                 :             :        your option) any later version
      11                 :             : 
      12                 :             :    or
      13                 :             : 
      14                 :             :      * the GNU General Public License as published by the Free
      15                 :             :        Software Foundation; either version 2 of the License, or (at
      16                 :             :        your option) any later version
      17                 :             : 
      18                 :             :    or both in parallel, as here.
      19                 :             : 
      20                 :             :    elfutils is distributed in the hope that it will be useful, but
      21                 :             :    WITHOUT ANY WARRANTY; without even the implied warranty of
      22                 :             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      23                 :             :    General Public License for more details.
      24                 :             : 
      25                 :             :    You should have received copies of the GNU General Public License and
      26                 :             :    the GNU Lesser General Public License along with this program.  If
      27                 :             :    not, see <http://www.gnu.org/licenses/>.  */
      28                 :             : 
      29                 :             : #ifdef HAVE_CONFIG_H
      30                 :             : # include <config.h>
      31                 :             : #endif
      32                 :             : 
      33                 :             : #include "libdwfl_stacktraceP.h"
      34                 :             : 
      35                 :             : #define HTAB_DEFAULT_SIZE 1021
      36                 :             : 
      37                 :           0 : Dwflst_Process_Tracker *dwflst_tracker_begin (const Dwfl_Callbacks *callbacks)
      38                 :             : {
      39                 :           0 :   Dwflst_Process_Tracker *tracker = calloc (1, sizeof *tracker);
      40         [ #  # ]:           0 :   if (tracker == NULL)
      41                 :             :     {
      42                 :           0 :       __libdwfl_seterrno (DWFL_E_NOMEM);
      43                 :           0 :       return tracker;
      44                 :             :     }
      45                 :             : 
      46                 :           0 :   dwflst_tracker_elftab_init (&tracker->elftab, HTAB_DEFAULT_SIZE);
      47                 :           0 :   rwlock_init (tracker->elftab_lock);
      48                 :           0 :   dwflst_tracker_dwfltab_init (&tracker->dwfltab, HTAB_DEFAULT_SIZE);
      49                 :           0 :   rwlock_init (tracker->dwfltab_lock);
      50                 :             : 
      51                 :           0 :   tracker->callbacks = callbacks;
      52                 :           0 :   return tracker;
      53                 :             : }
      54                 :             : 
      55                 :           0 : Dwfl *dwflst_tracker_dwfl_begin (Dwflst_Process_Tracker *tracker)
      56                 :             : {
      57                 :           0 :   Dwfl *dwfl = INTUSE(dwfl_begin) (tracker->callbacks);
      58         [ #  # ]:           0 :   if (dwfl == NULL)
      59                 :             :     return dwfl;
      60                 :             : 
      61                 :             :   /* TODO: Could also share dwfl->debuginfod, but thread-safely? */
      62                 :           0 :   dwfl->tracker = tracker;
      63                 :             : 
      64                 :             :   /* XXX: dwfl added to dwfltab when dwfl->process set in dwfl_attach_state. */
      65                 :             :   /* XXX: dwfl removed from dwfltab in dwfl_end() */
      66                 :             : 
      67                 :           0 :   return dwfl;
      68                 :             : }
      69                 :             : 
      70                 :           0 : Dwfl *dwflst_tracker_find_pid (Dwflst_Process_Tracker *tracker,
      71                 :             :                                pid_t pid,
      72                 :             :                                Dwfl *(*callback) (Dwflst_Process_Tracker *,
      73                 :             :                                                   pid_t, void *),
      74                 :             :                                void *arg)
      75                 :             : {
      76                 :           0 :   Dwfl *dwfl = NULL;
      77                 :             : 
      78                 :           0 :   rwlock_rdlock (tracker->dwfltab_lock);
      79                 :           0 :   dwflst_tracker_dwfl_info *ent
      80                 :           0 :     = dwflst_tracker_dwfltab_find(&tracker->dwfltab, pid);
      81                 :           0 :   rwlock_unlock (tracker->dwfltab_lock);
      82                 :             : 
      83   [ #  #  #  # ]:           0 :   if (ent != NULL && !ent->invalid)
      84                 :           0 :     dwfl = ent->dwfl;
      85         [ #  # ]:           0 :   if (dwfl == NULL && callback != NULL)
      86                 :           0 :     dwfl = callback(tracker, pid, arg);
      87         [ #  # ]:           0 :   if (dwfl != NULL)
      88                 :             :     {
      89         [ #  # ]:           0 :       assert (dwfl->tracker == tracker);
      90                 :             :       /* XXX: dwfl added to dwfltab when dwfl->process set in dwfl_attach_state.
      91                 :             :          Prior to that, the pid is not confirmed. */
      92                 :             :     }
      93                 :             : 
      94                 :           0 :   return dwfl;
      95                 :             : }
      96                 :             : 
      97                 :             : void
      98                 :             : internal_function
      99                 :           0 : __libdwfl_stacktrace_add_dwfl_to_tracker (Dwfl *dwfl) {
     100                 :           0 :   Dwflst_Process_Tracker *tracker = dwfl->tracker;
     101         [ #  # ]:           0 :   assert (tracker != NULL);
     102                 :             : 
     103                 :             :   /* First try to find an existing entry to replace: */
     104                 :           0 :   dwflst_tracker_dwfl_info *ent = NULL;
     105                 :           0 :   unsigned long int hval = dwfl->process->pid;
     106                 :             : 
     107                 :           0 :   rwlock_wrlock (tracker->dwfltab_lock);
     108                 :           0 :   ent = dwflst_tracker_dwfltab_find(&tracker->dwfltab, hval);
     109         [ #  # ]:           0 :   if (ent != NULL)
     110                 :             :     {
     111                 :             :       /* TODO: This is a bare-minimum solution. Ideally
     112                 :             :          we would clean up the existing ent->dwfl, but
     113                 :             :          this needs to be coordinated with any users of
     114                 :             :          the dwfl library that might still be holding it. */
     115                 :           0 :       ent->dwfl = dwfl;
     116                 :           0 :       ent->invalid = false;
     117                 :           0 :       rwlock_unlock (tracker->dwfltab_lock);
     118                 :           0 :       return;
     119                 :             :     }
     120                 :             : 
     121                 :             :   /* Only otherwise try to insert an entry: */
     122                 :           0 :   ent = calloc (1, sizeof(dwflst_tracker_dwfl_info));
     123         [ #  # ]:           0 :   if (ent == NULL)
     124                 :             :     {
     125                 :           0 :       rwlock_unlock (tracker->dwfltab_lock);
     126                 :           0 :       __libdwfl_seterrno (DWFL_E_NOMEM);
     127                 :           0 :       return;
     128                 :             :     }
     129                 :           0 :   ent->dwfl = dwfl;
     130                 :           0 :   ent->invalid = false;
     131         [ #  # ]:           0 :   if (dwflst_tracker_dwfltab_insert(&tracker->dwfltab, hval, ent) != 0)
     132                 :             :     {
     133                 :           0 :       free(ent);
     134                 :           0 :       rwlock_unlock (tracker->dwfltab_lock);
     135                 :           0 :       assert(false); /* Should not occur due to the wrlock on dwfltab. */
     136                 :             :     }
     137                 :           0 :   rwlock_unlock (tracker->dwfltab_lock);
     138                 :             : }
     139                 :             : 
     140                 :             : void
     141                 :             : internal_function
     142                 :           0 : __libdwfl_stacktrace_remove_dwfl_from_tracker (Dwfl *dwfl) {
     143         [ #  # ]:           0 :   if (dwfl->tracker == NULL)
     144                 :             :     return;
     145                 :           0 :   Dwflst_Process_Tracker *tracker = dwfl->tracker;
     146                 :           0 :   dwflst_tracker_dwfl_info *ent = NULL;
     147         [ #  # ]:           0 :   if (dwfl->process == NULL)
     148                 :             :     return;
     149                 :           0 :   unsigned long int hval = dwfl->process->pid;
     150                 :             : 
     151                 :           0 :   rwlock_wrlock (tracker->dwfltab_lock);
     152                 :           0 :   ent = dwflst_tracker_dwfltab_find(&tracker->dwfltab, hval);
     153   [ #  #  #  # ]:           0 :   if (ent != NULL && ent->dwfl == dwfl)
     154                 :             :     {
     155                 :           0 :       ent->dwfl = NULL;
     156                 :           0 :       ent->invalid = true;
     157                 :             :     }
     158                 :           0 :   rwlock_unlock (tracker->dwfltab_lock);
     159                 :             : }
     160                 :             : 
     161                 :           0 : void dwflst_tracker_end (Dwflst_Process_Tracker *tracker)
     162                 :             : {
     163         [ #  # ]:           0 :   if (tracker == NULL)
     164                 :             :     return;
     165                 :             : 
     166                 :           0 :   size_t idx;
     167                 :             : 
     168                 :             :   /* HACK to allow iteration of dynamicsizehash_concurrent.  */
     169                 :             :   /* XXX Based on lib/dynamicsizehash_concurrent.c free().  */
     170                 :           0 :   rwlock_fini (tracker->elftab_lock);
     171                 :           0 :   pthread_rwlock_destroy(&tracker->elftab.resize_rwl);
     172         [ #  # ]:           0 :   for (idx = 1; idx <= tracker->elftab.size; idx++)
     173                 :             :     {
     174                 :           0 :       dwflst_tracker_elftab_ent *ent = &tracker->elftab.table[idx];
     175         [ #  # ]:           0 :       if (ent->hashval == 0)
     176                 :           0 :         continue;
     177                 :           0 :       dwflst_tracker_elf_info *t =
     178                 :           0 :         (dwflst_tracker_elf_info *) atomic_load_explicit (&ent->val_ptr,
     179                 :             :                                                           memory_order_relaxed);
     180                 :           0 :       free(t->module_name);
     181         [ #  # ]:           0 :       if (t->fd >= 0)
     182                 :           0 :         close(t->fd);
     183         [ #  # ]:           0 :       if (t->elf != NULL)
     184                 :           0 :         elf_end(t->elf);
     185                 :           0 :       free(t); /* TODO: Check necessity. */
     186                 :             :     }
     187                 :           0 :   free (tracker->elftab.table);
     188                 :             : 
     189                 :             :   /* XXX Based on lib/dynamicsizehash_concurrent.c free().  */
     190                 :           0 :   rwlock_fini (tracker->dwfltab_lock);
     191                 :           0 :   pthread_rwlock_destroy(&tracker->dwfltab.resize_rwl);
     192         [ #  # ]:           0 :   for (idx = 1; idx <= tracker->dwfltab.size; idx++)
     193                 :             :     {
     194                 :           0 :       dwflst_tracker_dwfltab_ent *ent = &tracker->dwfltab.table[idx];
     195         [ #  # ]:           0 :       if (ent->hashval == 0)
     196                 :           0 :         continue;
     197                 :           0 :       dwflst_tracker_dwfl_info *t =
     198                 :           0 :         (dwflst_tracker_dwfl_info *) atomic_load_explicit (&ent->val_ptr,
     199                 :             :                                                            memory_order_relaxed);
     200         [ #  # ]:           0 :       if (t->dwfl != NULL)
     201                 :           0 :         INTUSE(dwfl_end) (t->dwfl);
     202                 :           0 :       free(t);
     203                 :             :     }
     204                 :           0 :   free (tracker->dwfltab.table);
     205                 :             : 
     206                 :           0 :   free (tracker);
     207                 :             : }
        

Generated by: LCOV version 2.0-1