summaryrefslogtreecommitdiffstats
path: root/lib/libthr/thread/thr_fork.c
diff options
context:
space:
mode:
authordavidxu <davidxu@FreeBSD.org>2010-09-19 09:03:11 +0000
committerdavidxu <davidxu@FreeBSD.org>2010-09-19 09:03:11 +0000
commitfe5567c8f172d8218fb37af60118825748628d44 (patch)
tree21699abe5fd839789147e1721b9b3cd97787d6c1 /lib/libthr/thread/thr_fork.c
parent5418a235978733112ea2f114b3826d95346409e8 (diff)
downloadFreeBSD-src-fe5567c8f172d8218fb37af60118825748628d44.zip
FreeBSD-src-fe5567c8f172d8218fb37af60118825748628d44.tar.gz
Because atfork lock is held while forking, a thread cancellation triggered
by atfork handler is unsafe, use intenal flag no_cancel to disable it.
Diffstat (limited to 'lib/libthr/thread/thr_fork.c')
-rw-r--r--lib/libthr/thread/thr_fork.c9
1 files changed, 7 insertions, 2 deletions
diff --git a/lib/libthr/thread/thr_fork.c b/lib/libthr/thread/thr_fork.c
index dd6005a..fb5daa1 100644
--- a/lib/libthr/thread/thr_fork.c
+++ b/lib/libthr/thread/thr_fork.c
@@ -137,7 +137,7 @@ _fork(void)
struct pthread *curthread;
struct pthread_atfork *af;
pid_t ret;
- int errsave;
+ int errsave, cancelsave;
int was_threaded;
int rtld_locks[MAX_RTLD_LOCKS];
@@ -145,7 +145,8 @@ _fork(void)
return (__sys_fork());
curthread = _get_curthread();
-
+ cancelsave = curthread->no_cancel;
+ curthread->no_cancel = 1;
_thr_rwl_rdlock(&_thr_atfork_lock);
/* Run down atfork prepare handlers. */
@@ -223,6 +224,7 @@ _fork(void)
af->child();
}
_thr_rwlock_unlock(&_thr_atfork_lock);
+ curthread->no_cancel = cancelsave;
} else {
/* Parent process */
errsave = errno;
@@ -244,6 +246,9 @@ _fork(void)
}
_thr_rwlock_unlock(&_thr_atfork_lock);
+ curthread->no_cancel = cancelsave;
+ /* test async cancel */
+ _thr_testcancel(curthread);
}
errno = errsave;
OpenPOWER on IntegriCloud