I have a library which registers an atfork
handler (via pthread_atfork()
) which does not support multiple threads when fork()
is called. In my case, I don't need the forked environment to be usable because all I want is to call exec()
right after the fork()
. So, I want the fork()
but without any atfork
handlers. Is that possible? Do I miss any important edge cases?
For background info, the library is OpenBlas, the issue is described here and here.
You could use vfork()
(NPTL implementation doesn't call fork handlers). Although POSIX has removed vfork
from the standard, it's likely available on your implementation.
Fork handlers established using pthread_atfork(3) are not called when
a multithreaded program employing the NPTL threading library calls
vfork(). Fork handlers are called in this case in a program using
the LinuxThreads threading library. (See pthreads(7) for a
description of Linux threading libraries.)
Or, posix_spawn()
. This is similar to vfork
. Man page says:
According to POSIX, it unspecified whether fork handlers established with pthread_atfork(3) are called when posix_spawn() is invoked. On glibc, fork handlers are called only if the child is created using fork(2).
Or, syscall
and directly use SYS_clone
. SYS_clone
is the system call number used to create threads and processes on Linux. So syscall(SYS_clone, SIGCHLD, 0);
should work, provided you would exec immediately.
syscall(SYS_fork);
(as answered by Shachar) would likely work too. But note that SYS_fork
not available on some platforms (e.g., aarch64, ia64). SYS_fork
is considered as obsolete in Linux and it's only there for backward compatibility and Linux kernel uses SYS_clone for creating all "types" of processes.
(Note: These options are mostly limited to glibc/Linux).
Yes. The following should work on Linux (and, I think, all glibc
based platforms):
#define _GNU_SOURCE
#include <unistd.h>
#include <sys/syscall.h>
...
syscall(SYS_fork);
This bypasses the library and directly calls the system call for fork
. You might run into trouble if your platform does not implement fork
as a single system call. For Linux, that simply means that you should use clone
instead.
With that in mind, I'm not sure I'd recomment doing that. Since you're a library, you have no idea why someone registered an atfork
. Assuming it's irrelevant is bad programming practice.
So you lose portability in order to do something that may or may not break stuff, all in the name of, what? Saving a few function calls? Personally, I'd just use fork
.