#include <pthread.h> | |
#include "libc.h" | |
#include "lock.h" | |
static struct atfork_funcs { | |
void (*prepare)(void); | |
void (*parent)(void); | |
void (*child)(void); | |
struct atfork_funcs *prev, *next; | |
} *funcs; | |
static volatile int lock[1]; | |
void __fork_handler(int who) | |
{ | |
struct atfork_funcs *p; | |
if (!funcs) return; | |
if (who < 0) { | |
LOCK(lock); | |
for (p=funcs; p; p = p->next) { | |
if (p->prepare) p->prepare(); | |
funcs = p; | |
} | |
} else { | |
for (p=funcs; p; p = p->prev) { | |
if (!who && p->parent) p->parent(); | |
else if (who && p->child) p->child(); | |
funcs = p; | |
} | |
UNLOCK(lock); | |
} | |
} | |
int pthread_atfork(void (*prepare)(void), void (*parent)(void), void (*child)(void)) | |
{ | |
struct atfork_funcs *new = malloc(sizeof *new); | |
if (!new) return -1; | |
LOCK(lock); | |
new->next = funcs; | |
new->prev = 0; | |
new->prepare = prepare; | |
new->parent = parent; | |
new->child = child; | |
if (funcs) funcs->prev = new; | |
funcs = new; | |
UNLOCK(lock); | |
return 0; | |
} |