int writeback;
int flock;
int xattr;
- const char *source;
+ char *source;
double timeout;
int cache;
int timeout_set;
}
} else {
- lo.source = "/";
+ lo.source = strdup("/");
+ if(!lo.source) {
+ fuse_log(FUSE_LOG_ERR, "fuse: memory allocation failed\n");
+ exit(1);
+ }
}
if (!lo.timeout_set) {
switch (lo.cache) {
if (lo.root.fd >= 0)
close(lo.root.fd);
+ free(lo.source);
return ret ? 1 : 0;
}
waitpid(pid, NULL, 0);
}
+static int setup_auto_unmount(const char *mountpoint, int quiet)
+{
+ int fds[2], pid;
+ int res;
+
+ if (!mountpoint) {
+ fuse_log(FUSE_LOG_ERR, "fuse: missing mountpoint parameter\n");
+ return -1;
+ }
+
+ res = socketpair(PF_UNIX, SOCK_STREAM, 0, fds);
+ if(res == -1) {
+ perror("fuse: socketpair() failed");
+ return -1;
+ }
+
+ pid = fork();
+ if(pid == -1) {
+ perror("fuse: fork() failed");
+ close(fds[0]);
+ close(fds[1]);
+ return -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;
+
+ 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);
+ }
+
+ close(fds[0]);
+
+ // Now fusermount3 will only exit when fds[1] closes automatically when our
+ // process exits.
+ return 0;
+}
+
static int fuse_mount_fusermount(const char *mountpoint, struct mount_opts *mo,
const char *opts, int quiet)
{
return -1;
}
- if (mo->auto_unmount) {
- /* Tell the caller to fallback to fusermount3 because
- auto-unmount does not work otherwise. */
- return -2;
- }
-
fd = open(devname, O_RDWR | O_CLOEXEC);
if (fd == -1) {
if (errno == ENODEV || errno == ENOENT)
goto out;
res = fuse_mount_sys(mountpoint, mo, mnt_opts);
- if (res == -2) {
+ if (res >= 0 && mo->auto_unmount) {
+ if(0 > setup_auto_unmount(mountpoint, 0)) {
+ // Something went wrong, let's umount like in fuse_mount_sys.
+ umount2(mountpoint, MNT_DETACH); /* lazy umount */
+ res = -1;
+ }
+ } else if (res == -2) {
if (mo->fusermount_opts &&
fuse_opt_add_opt(&mnt_opts, mo->fusermount_opts) == -1)
goto out;
else:
umount(mount_process, mnt_dir)
+@pytest.mark.parametrize("intended_user", ('root', 'non_root'))
+def test_dev_auto_unmount(short_tmpdir, output_checker, intended_user):
+ """Check that root can mount with dev and auto_unmount
+ (but non-root cannot).
+ Split into root vs non-root, so that the output of pytest
+ makes clear what functionality is being tested."""
+ if os.getuid() == 0 and intended_user == 'non_root':
+ pytest.skip('needs to run as non-root')
+ if os.getuid() != 0 and intended_user == 'root':
+ pytest.skip('needs to run as root')
+ mnt_dir = str(short_tmpdir.mkdir('mnt'))
+ src_dir = str('/dev')
+ cmdline = base_cmdline + \
+ [ pjoin(basename, 'example', 'passthrough_ll'),
+ '-o', f'source={src_dir},dev,auto_unmount',
+ '-f', mnt_dir ]
+ mount_process = subprocess.Popen(cmdline, stdout=output_checker.fd,
+ stderr=output_checker.fd)
+ try:
+ wait_for_mount(mount_process, mnt_dir)
+ if os.getuid() == 0:
+ open(pjoin(mnt_dir, 'null')).close()
+ else:
+ with pytest.raises(PermissionError):
+ open(pjoin(mnt_dir, 'null')).close()
+ except:
+ cleanup(mount_process, mnt_dir)
+ raise
+ else:
+ umount(mount_process, mnt_dir)
+
@pytest.mark.skipif(os.getuid() != 0,
reason='needs to run as root')
def test_cuse(output_checker):
int cfd;
const char *opts = "";
const char *type = NULL;
+ int setup_auto_unmount_only = 0;
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'},
case 'u':
unmount = 1;
break;
-
+ case 'U':
+ unmount = 1;
+ auto_unmount = 1;
+ setup_auto_unmount_only = 1;
+ break;
case 'z':
lazy = 1;
break;
exit(1);
umask(033);
- if (unmount)
+ if (!setup_auto_unmount_only && unmount)
goto do_unmount;
commfd = getenv(FUSE_COMMFD_ENV);
goto err_out;
}
+ cfd = atoi(commfd);
+
+ if (setup_auto_unmount_only)
+ goto wait_for_auto_unmount;
+
fd = mount_fuse(mnt, opts, &type);
if (fd == -1)
goto err_out;
- cfd = atoi(commfd);
res = send_fd(cfd, fd);
if (res == -1)
goto err_out;
return 0;
}
+wait_for_auto_unmount:
/* Become a daemon and wait for the parent to exit or die.
ie For the control socket to get closed.
btw We don't want to use daemon() function here because