See the file COPYING.LIB.
 */
 
+/* For environ */
+#define _GNU_SOURCE
+
 #include "fuse_config.h"
 #include "fuse_i.h"
 #include "fuse_misc.h"
 #include <fcntl.h>
 #include <errno.h>
 #include <poll.h>
+#include <spawn.h>
 #include <sys/socket.h>
 #include <sys/un.h>
 #include <sys/wait.h>
 #define FUSERMOUNT_PROG                "fusermount3"
 #define FUSE_COMMFD_ENV                "_FUSE_COMMFD"
 
-#ifndef HAVE_FORK
-#define fork() vfork()
-#endif
-
 #ifndef MS_DIRSYNC
 #define MS_DIRSYNC 128
 #endif
        FUSE_OPT_END
 };
 
-static void exec_fusermount(const char *argv[])
-{
-       execv(FUSERMOUNT_DIR "/" FUSERMOUNT_PROG, (char **) argv);
-       execvp(FUSERMOUNT_PROG, (char **) argv);
-}
-
 void fuse_mount_version(void)
 {
-       int pid = fork();
-       if (!pid) {
-               const char *argv[] = { FUSERMOUNT_PROG, "--version", NULL };
-               exec_fusermount(argv);
-               _exit(1);
-       } else if (pid != -1)
+       char const *const argv[] = {FUSERMOUNT_PROG, "--version", NULL};
+       pid_t pid;
+       int status = posix_spawn(&pid, FUSERMOUNT_DIR "/" FUSERMOUNT_PROG, NULL,
+                       NULL, (char * const *) argv, environ)
+               && posix_spawnp(&pid, FUSERMOUNT_PROG, NULL,
+                       NULL, (char * const *) argv, environ);
+       if(status != 0)
+               perror("posix_spawn");
+       else
                waitpid(pid, NULL, 0);
 }
 
 void fuse_kern_unmount(const char *mountpoint, int fd)
 {
        int res;
-       int pid;
+       pid_t pid;
 
        if (fd != -1) {
                struct pollfd pfd;
        if (res == 0)
                return;
 
-       pid = fork();
-       if(pid == -1)
+       char const * const argv[] =
+               { FUSERMOUNT_PROG, "--unmount", "--quiet", "--lazy",
+                               "--", mountpoint, NULL };
+       int status = posix_spawn(
+               &pid, FUSERMOUNT_DIR "/" FUSERMOUNT_PROG, NULL, NULL, (char * const *) argv, environ)
+               && posix_spawnp(
+                       &pid, FUSERMOUNT_PROG, NULL, NULL, (char * const *) argv, environ);
+       if(status != 0) {
+               perror("posix_spawn");
                return;
-
-       if(pid == 0) {
-               const char *argv[] = { FUSERMOUNT_PROG, "-u", "-q", "-z",
-                                      "--", mountpoint, NULL };
-
-               exec_fusermount(argv);
-               _exit(1);
        }
        waitpid(pid, NULL, 0);
 }
 
 static int setup_auto_unmount(const char *mountpoint, int quiet)
 {
-       int fds[2], pid;
+       int fds[2];
+       pid_t pid;
        int res;
 
        if (!mountpoint) {
                return -1;
        }
 
-       pid = fork();
-       if(pid == -1) {
-               perror("fuse: fork() failed");
-               close(fds[0]);
-               close(fds[1]);
-               return -1;
+       char arg_fd_entry[30];
+       snprintf(arg_fd_entry, sizeof(arg_fd_entry), "%i", fds[0]);
+       setenv(FUSE_COMMFD_ENV, arg_fd_entry, 1);
+
+       char const *const argv[] = {
+               FUSERMOUNT_PROG,
+               "--auto-unmount",
+               "--",
+               mountpoint,
+               NULL,
+       };
+
+       // TODO: add error handling for all manipulations of action.
+       posix_spawn_file_actions_t action;
+       posix_spawn_file_actions_init(&action);
+
+       if (quiet) {
+               posix_spawn_file_actions_addclose(&action, 1);
+               posix_spawn_file_actions_addclose(&action, 2);
        }
+       posix_spawn_file_actions_addclose(&action, fds[1]);
 
-       if(pid == 0) {
-               char env[10];
-               const char *argv[32];
-               int a = 0;
-
-               if (quiet) {
-                       int fd = open("/dev/null", O_RDONLY);
-                       if (fd != -1) {
-                               dup2(fd, 1);
-                               dup2(fd, 2);
-                       }
-               }
-
-               argv[a++] = FUSERMOUNT_PROG;
-               argv[a++] = "--auto-unmount";
-               argv[a++] = "--";
-               argv[a++] = mountpoint;
-               argv[a++] = NULL;
+       int status = posix_spawn(&pid, FUSERMOUNT_DIR "/" FUSERMOUNT_PROG, &action,
+                       NULL, (char *const *) argv, environ)
+               && posix_spawnp(&pid, FUSERMOUNT_PROG, &action,
+                       NULL, (char *const *) argv, environ);
+       posix_spawn_file_actions_destroy(&action);
 
+       if(status != 0) {
+               close(fds[0]);
                close(fds[1]);
-               fcntl(fds[0], F_SETFD, 0);
-               snprintf(env, sizeof(env), "%i", fds[0]);
-               setenv(FUSE_COMMFD_ENV, env, 1);
-               exec_fusermount(argv);
-               perror("fuse: failed to exec fusermount3");
-               _exit(1);
+               perror("fuse: posix_spawnp() of fusermount3 failed");
+               return -1;
        }
-
+       // passed to child now, so can close here.
        close(fds[0]);
 
        // Now fusermount3 will only exit when fds[1] closes automatically when our
        // process exits.
        return 0;
+       // Note: fds[1] is leakend and doesn't get FD_CLOEXEC
 }
 
 static int fuse_mount_fusermount(const char *mountpoint, struct mount_opts *mo,
                const char *opts, int quiet)
 {
-       int fds[2], pid;
+       int fds[2];
+       pid_t pid;
        int res;
-       int rv;
 
        if (!mountpoint) {
                fuse_log(FUSE_LOG_ERR, "fuse: missing mountpoint parameter\n");
                return -1;
        }
 
-       pid = fork();
-       if(pid == -1) {
-               perror("fuse: fork() failed");
-               close(fds[0]);
-               close(fds[1]);
-               return -1;
-       }
+       char arg_fd_entry[30];
+       snprintf(arg_fd_entry, sizeof(arg_fd_entry), "%i", fds[0]);
+       setenv(FUSE_COMMFD_ENV, arg_fd_entry, 1);
 
-       if(pid == 0) {
-               char env[10];
-               const char *argv[32];
-               int a = 0;
+       char const *const argv[] = {
+               FUSERMOUNT_PROG,
+               "-o", opts ? opts : "",
+               "--",
+               mountpoint,
+               NULL,
+       };
 
-               if (quiet) {
-                       int fd = open("/dev/null", O_RDONLY);
-                       if (fd != -1) {
-                               dup2(fd, 1);
-                               dup2(fd, 2);
-                       }
-               }
 
-               argv[a++] = FUSERMOUNT_PROG;
-               if (opts) {
-                       argv[a++] = "-o";
-                       argv[a++] = opts;
-               }
-               argv[a++] = "--";
-               argv[a++] = mountpoint;
-               argv[a++] = NULL;
+       posix_spawn_file_actions_t action;
+       posix_spawn_file_actions_init(&action);
+
+       if (quiet) {
+               posix_spawn_file_actions_addclose(&action, 1);
+               posix_spawn_file_actions_addclose(&action, 2);
+       }
+       posix_spawn_file_actions_addclose(&action, fds[1]);
 
+       int status = posix_spawn(&pid, FUSERMOUNT_DIR "/" FUSERMOUNT_PROG, &action,
+                       NULL, (char *const *) argv, environ)
+               && posix_spawnp(&pid, FUSERMOUNT_PROG, &action,
+                       NULL, (char *const *) argv, environ);
+       posix_spawn_file_actions_destroy(&action);
+
+       if(status != 0) {
+               close(fds[0]);
                close(fds[1]);
-               fcntl(fds[0], F_SETFD, 0);
-               snprintf(env, sizeof(env), "%i", fds[0]);
-               setenv(FUSE_COMMFD_ENV, env, 1);
-               exec_fusermount(argv);
-               perror("fuse: failed to exec fusermount3");
-               _exit(1);
+               perror("fuse: posix_spawnp() of fusermount3 failed");
+               return -1;
        }
 
+       // passed to child now, so can close here.
        close(fds[0]);
-       rv = receive_fd(fds[1]);
+
+       int rv = receive_fd(fds[1]);
 
        if (!mo->auto_unmount) {
                /* with auto_unmount option fusermount3 will not exit until
 
        static int unmount = 0;
        static int lazy = 0;
        static int quiet = 0;
-       char *commfd;
+       char *commfd = NULL;
        int cfd;
        const char *opts = "";
        const char *type = NULL;
 
        static const struct option long_opts[] = {
                {"unmount", no_argument, NULL, 'u'},
-               // Note: auto-unmount deliberately does not have a short version.
-               // It's meant for internal use by mount.c's setup_auto_unmount.
-               {"auto-unmount", no_argument, NULL, 'U'},
                {"lazy",    no_argument, NULL, 'z'},
                {"quiet",   no_argument, NULL, 'q'},
                {"help",    no_argument, NULL, 'h'},
                {"version", no_argument, NULL, 'V'},
                {"options", required_argument, NULL, 'o'},
+               // Note: auto-unmount and comm-fd don't have short versions.
+               // They'ne meant for internal use by mount.c
+               {"auto-unmount", no_argument, NULL, 'U'},
+               {"comm-fd", required_argument, NULL, 'c'},
                {0, 0, 0, 0}};
 
        progname = strdup(argc > 0 ? argv[0] : "fusermount");
                        auto_unmount = 1;
                        setup_auto_unmount_only = 1;
                        break;
+               case 'c':
+                       commfd = optarg;
+                       break;
                case 'z':
                        lazy = 1;
                        break;
        if (!setup_auto_unmount_only && unmount)
                goto do_unmount;
 
-       commfd = getenv(FUSE_COMMFD_ENV);
+       if(commfd == NULL)
+               commfd = getenv(FUSE_COMMFD_ENV);
        if (commfd == NULL) {
                fprintf(stderr, "%s: old style mounting not supported\n",
                        progname);