diff options
author | guenther <> | 2015-04-07 01:27:07 +0000 |
---|---|---|
committer | guenther <> | 2015-04-07 01:27:07 +0000 |
commit | 8a98fefea13a82a7e7355353723c8ee1722456e7 (patch) | |
tree | f516e404d864d5abc848130431ad93c0aa43674c /src/lib/libc/stdlib/atexit.c | |
parent | 5bc886238be15ae40e7b95f920d80bf191b2ae70 (diff) | |
download | openbsd-8a98fefea13a82a7e7355353723c8ee1722456e7.tar.gz openbsd-8a98fefea13a82a7e7355353723c8ee1722456e7.tar.bz2 openbsd-8a98fefea13a82a7e7355353723c8ee1722456e7.zip |
Make pthread_atfork() track the DSO that called it like atexit() does,
unregistering callbacks if the DSO is unloaded. Move the callback
handling from libpthread to libc, though libpthread still overrides the
inner call to handle locking and thread-library reinitialization.
Major version bump for both libc and libpthread.
verification that this fixes various ports ajacoutot@
asm assistance miod@; ok millert@ deraadt@
Diffstat (limited to 'src/lib/libc/stdlib/atexit.c')
-rw-r--r-- | src/lib/libc/stdlib/atexit.c | 20 |
1 files changed, 19 insertions, 1 deletions
diff --git a/src/lib/libc/stdlib/atexit.c b/src/lib/libc/stdlib/atexit.c index 6532b382ea..a33080571f 100644 --- a/src/lib/libc/stdlib/atexit.c +++ b/src/lib/libc/stdlib/atexit.c | |||
@@ -1,4 +1,4 @@ | |||
1 | /* $OpenBSD: atexit.c,v 1.20 2014/07/11 09:51:37 kettenis Exp $ */ | 1 | /* $OpenBSD: atexit.c,v 1.21 2015/04/07 01:27:07 guenther Exp $ */ |
2 | /* | 2 | /* |
3 | * Copyright (c) 2002 Daniel Hartmeier | 3 | * Copyright (c) 2002 Daniel Hartmeier |
4 | * All rights reserved. | 4 | * All rights reserved. |
@@ -35,6 +35,7 @@ | |||
35 | #include <string.h> | 35 | #include <string.h> |
36 | #include <unistd.h> | 36 | #include <unistd.h> |
37 | #include "atexit.h" | 37 | #include "atexit.h" |
38 | #include "atfork.h" | ||
38 | #include "thread_private.h" | 39 | #include "thread_private.h" |
39 | 40 | ||
40 | struct atexit *__atexit; | 41 | struct atexit *__atexit; |
@@ -161,6 +162,23 @@ restart: | |||
161 | __atexit = NULL; | 162 | __atexit = NULL; |
162 | } | 163 | } |
163 | _ATEXIT_UNLOCK(); | 164 | _ATEXIT_UNLOCK(); |
165 | |||
166 | /* | ||
167 | * If unloading a DSO, unregister any atfork handlers registered | ||
168 | * by it. Skip the locking if the list is currently empty. | ||
169 | */ | ||
170 | if (dso != NULL && TAILQ_FIRST(&_atfork_list) != NULL) { | ||
171 | struct atfork_fn *af, *afnext; | ||
172 | |||
173 | _ATFORK_LOCK(); | ||
174 | TAILQ_FOREACH_SAFE(af, &_atfork_list, fn_next, afnext) | ||
175 | if (af->fn_dso == dso) { | ||
176 | TAILQ_REMOVE(&_atfork_list, af, fn_next); | ||
177 | free(af); | ||
178 | } | ||
179 | _ATFORK_UNLOCK(); | ||
180 | |||
181 | } | ||
164 | } | 182 | } |
165 | 183 | ||
166 | /* | 184 | /* |