pthread_atfork.c 931 B

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748
  1. #include <pthread.h>
  2. #include "libc.h"
  3. static struct atfork_funcs {
  4. void (*prepare)(void);
  5. void (*parent)(void);
  6. void (*child)(void);
  7. struct atfork_funcs *prev, *next;
  8. } *funcs;
  9. static int lock;
  10. static void fork_handler(int who)
  11. {
  12. struct atfork_funcs *p;
  13. if (who < 0) {
  14. LOCK(&lock);
  15. for (p=funcs; p; p = p->next) {
  16. if (p->prepare) p->prepare();
  17. funcs = p;
  18. }
  19. } else {
  20. for (p=funcs; p; p = p->prev) {
  21. if (!who && p->parent) p->parent();
  22. else if (who && p->child) p->child();
  23. funcs = p;
  24. }
  25. UNLOCK(&lock);
  26. }
  27. }
  28. int pthread_atfork(void (*prepare)(void), void (*parent)(void), void (*child)(void))
  29. {
  30. struct atfork_funcs *new = malloc(sizeof *new);
  31. if (!new) return -1;
  32. LOCK(&lock);
  33. libc.fork_handler = fork_handler;
  34. new->next = funcs;
  35. new->prev = 0;
  36. new->prepare = prepare;
  37. new->parent = parent;
  38. new->child = child;
  39. if (funcs) funcs->prev = new;
  40. funcs = new;
  41. UNLOCK(&lock);
  42. return 0;
  43. }