/src/elfutils/libdwfl_stacktrace/dwflst_process_tracker.c
Line | Count | Source |
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 | 0 | #define HTAB_DEFAULT_SIZE 1021 |
36 | | |
37 | | Dwflst_Process_Tracker *dwflst_tracker_begin (const Dwfl_Callbacks *callbacks) |
38 | 0 | { |
39 | 0 | Dwflst_Process_Tracker *tracker = calloc (1, sizeof *tracker); |
40 | 0 | if (tracker == NULL) |
41 | 0 | { |
42 | 0 | __libdwfl_seterrno (DWFL_E_NOMEM); |
43 | 0 | return tracker; |
44 | 0 | } |
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 | 0 | } |
54 | | |
55 | | Dwfl *dwflst_tracker_dwfl_begin (Dwflst_Process_Tracker *tracker) |
56 | 0 | { |
57 | 0 | Dwfl *dwfl = INTUSE(dwfl_begin) (tracker->callbacks); |
58 | 0 | if (dwfl == NULL) |
59 | 0 | 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 | 0 | } |
69 | | |
70 | | 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 | 0 | { |
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 | 0 | { |
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 | 0 | } |
93 | |
|
94 | 0 | return dwfl; |
95 | 0 | } |
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 | 0 | { |
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 | 0 | } |
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 | 0 | { |
125 | 0 | rwlock_unlock (tracker->dwfltab_lock); |
126 | 0 | __libdwfl_seterrno (DWFL_E_NOMEM); |
127 | 0 | return; |
128 | 0 | } |
129 | 0 | ent->dwfl = dwfl; |
130 | 0 | ent->invalid = false; |
131 | 0 | if (dwflst_tracker_dwfltab_insert(&tracker->dwfltab, hval, ent) != 0) |
132 | 0 | { |
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 | 0 | } |
137 | 0 | rwlock_unlock (tracker->dwfltab_lock); |
138 | 0 | } |
139 | | |
140 | | void |
141 | | internal_function |
142 | 0 | __libdwfl_stacktrace_remove_dwfl_from_tracker (Dwfl *dwfl) { |
143 | 0 | if (dwfl->tracker == NULL) |
144 | 0 | return; |
145 | 0 | Dwflst_Process_Tracker *tracker = dwfl->tracker; |
146 | 0 | dwflst_tracker_dwfl_info *ent = NULL; |
147 | 0 | if (dwfl->process == NULL) |
148 | 0 | 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 | 0 | { |
155 | 0 | ent->dwfl = NULL; |
156 | 0 | ent->invalid = true; |
157 | 0 | } |
158 | 0 | rwlock_unlock (tracker->dwfltab_lock); |
159 | 0 | } |
160 | | |
161 | | void dwflst_tracker_end (Dwflst_Process_Tracker *tracker) |
162 | 0 | { |
163 | 0 | if (tracker == NULL) |
164 | 0 | 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 | 0 | { |
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 | 0 | 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 | 0 | } |
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 | 0 | { |
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 | 0 | memory_order_relaxed); |
200 | 0 | if (t->dwfl != NULL) |
201 | 0 | INTUSE(dwfl_end) (t->dwfl); |
202 | 0 | free(t); |
203 | 0 | } |
204 | 0 | free (tracker->dwfltab.table); |
205 | |
|
206 | 0 | free (tracker); |
207 | 0 | } |