Blob Blame History Raw
// commit: 370f78f2c80c64b7b0780a01e672494a26b5678e 2011-03-09
// commit: 0bed7e0acfd34e3fb63ca0e4d99b7592571355a9 2011-03-09
// raise should be robust against async fork in a signal handler
#include <pthread.h>
#include <signal.h>
#include <errno.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>
#include "test.h"

static volatile int c0;
static volatile int c1;
static volatile int child;

static void handler0(int sig)
{
	c0++;
}

static void handler1(int sig)
{
	c1++;
	switch (fork()) {
	case 0: child=1; break;
	case -1: t_error("fork failed: %s\n", strerror(errno));
	}
}

static void *start(void *arg)
{
	int i,r,s;

	for (i = 0; i < 1000; i++) {
		r = raise(SIGRTMIN);
		if (r)
			t_error("raise failed: %s\n", strerror(errno));
	}
	if (c0 != 1000)
		t_error("lost signals: got %d, wanted 1000 (ischild %d forks %d)\n", c0, child, c1);
	if (child)
		_exit(t_status);

	/* make sure we got all pthread_kills, then wait the forked children */
	while (c1 < 100);
	for (i = 0; i < 100; i++) {
		r = wait(&s);
		if (r == -1)
			t_error("wait failed: %s\n", strerror(errno));
		else if (!WIFEXITED(s) || WTERMSIG(s))
			t_error("child failed: pid:%d status:%d\n", r, s);
	}
	return 0;
}

int main(void)
{
	pthread_t t;
	void *p;
	int r, i, s;

	if (signal(SIGRTMIN, handler0) == SIG_ERR)
		t_error("registering signal handler failed: %s\n", strerror(errno));
	if (signal(SIGRTMIN+1, handler1) == SIG_ERR)
		t_error("registering signal handler failed: %s\n", strerror(errno));

	r = pthread_create(&t, 0, start, 0);
	if (r)
		t_error("pthread_create failed: %s\n", strerror(r));
	for (i = 0; i < 100; i++) {
		r = pthread_kill(t, SIGRTMIN+1);
		if (r)
			t_error("phread_kill failed: %s\n", strerror(r));
	}
	r = pthread_join(t,&p);
	if (r)
		t_error("pthread_join failed: %s\n", strerror(r));
	return t_status;
}