我放入exit()
了一个线程,但我的程序有时不会退出。
根据这个链接,exit()
是不是async-signal-safe
。我想知道exit()
在线程中使用是否会导致未定义的行为。
我放入exit()
了一个线程,但我的程序有时不会退出。
根据这个链接,exit()
是不是async-signal-safe
。我想知道exit()
在线程中使用是否会导致未定义的行为。
普通exit
(而不是_exit
例如)需要进行所有通常的atexit
清理、输出刷新等工作。可以构建在某些情况下挂起的代码,但我不得不提出一个“明显的问题”来展示它。如果库例程(例如,内部 stdio fflush
)试图在其他线程持有的退出线程中获取锁(例如,在 stdio 流上),那么即使没有您自己的atexit
. 由于您没有显示您的代码,我只是在猜测。
这是一个测试程序(有故意的,明显的问题),在被告知时会挂起,至少在 FreeBSD 上是这样。(格式化很棘手,因为剪切和粘贴保留了标签,但后来我不得不将一些编辑到空格中......)
#include <pthread.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
pthread_mutex_t global_mtx;
void die(int error, const char *fmt, ...) {
va_list ap;
va_start(ap, fmt);
vfprintf(stderr, fmt, ap);
va_end(ap);
if (error)
fprintf(stderr, ": %s\n", strerror(error));
else
putc('\n', stderr);
fflush(stderr);
_exit(0);
}
enum behavior { NORMAL, EXIT_WO_HANG, EXIT_W_HANG };
struct behave {
enum behavior how;
pthread_mutex_t lock;
pthread_cond_t cond;
int th1_entered;
int th2_entered;
};
void hanger(void);
void *th1_main(void *);
void *th2_main(void *);
#define WR(x) (void)write(1, x, sizeof(x) - 1)
int main(int argc, char **argv) {
int error;
struct behave how;
pthread_t th1, th2;
error = pthread_mutex_init(&global_mtx, NULL);
if (error)
die(error, "pthread_mutex_init global_mtx");
error = pthread_mutex_init(&how.lock, NULL);
if (error)
die(error, "pthread_mutex_init how.lock");
error = pthread_cond_init(&how.cond, NULL);
if (error)
die(error, "pthread_cond_init how.cond");
how.how = NORMAL;
how.th1_entered = 0;
how.th2_entered = 0;
if (argc > 1) {
if (strcmp(argv[1], "exit") == 0)
how.how = EXIT_WO_HANG;
else if (strcmp(argv[1], "hang") == 0)
how.how = EXIT_W_HANG;
else if (strcmp(argv[1], "normal") != 0)
die(0, "usage: example [normal|exit|hang]");
}
atexit(hanger);
error = pthread_create(&th1, NULL, th1_main, &how);
if (error)
die(error, "pthread_create th1");
error = pthread_create(&th2, NULL, th2_main, &how);
if (error)
die(error, "pthread_create th2");
/* now wait for threads */
error = pthread_join(th1, NULL);
error = pthread_join(th2, NULL);
printf("joined, normal exit\n");
return 0;
}
void *th1_main(void *arg) {
struct behave *how = arg;
WR("thread 1 start\n");
(void) pthread_mutex_lock(&global_mtx);
(void) pthread_mutex_lock(&how->lock);
how->th1_entered = 1;
pthread_cond_signal(&how->cond);
while (how->th2_entered == 0)
(void) pthread_cond_wait(&how->cond, &how->lock);
WR("thread 1 sees thread 2 started\n");
(void) pthread_mutex_unlock(&how->lock);
if (how->how == EXIT_W_HANG)
WR("thread 1 not unlocking\n");
else
(void) pthread_mutex_unlock(&global_mtx);
return NULL;
}
void *th2_main(void *arg) {
struct behave *how = arg;
WR("thread 2 start\n");
(void) pthread_mutex_lock(&how->lock);
how->th2_entered = 1;
pthread_cond_signal(&how->cond);
while (how->th1_entered == 0)
(void) pthread_cond_wait(&how->cond, &how->lock);
WR("thread 2 sees thread 1 started\n");
(void) pthread_mutex_unlock(&how->lock);
if (how->how != NORMAL) {
WR("thread 2 exit()\n");
exit(1);
}
return NULL;
}
void hanger(void) {
/* this is what will cause us to hang, in the one case */
WR("hanger start\n");
pthread_mutex_lock(&global_mtx);
WR("hanger got global mutex\n");
pthread_mutex_unlock(&global_mtx);
WR("hanger finish\n");
}