Files
libguestfs/lib/fuse.c
Pino Toscano ab0cf0012f fuse: use the configured program name
When initializing FUSE, use the program name as set (either
automatically or manually) in the guestfs handle.
2017-02-06 10:09:58 +01:00

1527 lines
37 KiB
C

/* libguestfs
* Copyright (C) 2009-2017 Red Hat Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <config.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <string.h>
#include <libintl.h>
#include <sys/sysmacros.h>
#if HAVE_FUSE
/* See <attr/xattr.h> */
#ifndef ENOATTR
#define ENOATTR ENODATA
#endif
#define FUSE_USE_VERSION 26
#include <fuse.h>
#include <fuse_lowlevel.h>
#endif
#include "cloexec.h"
#include "glthread/lock.h"
#include "hash.h"
#include "hash-pjw.h"
#include "guestfs.h"
#include "guestfs-internal.h"
#include "guestfs-internal-actions.h"
#if HAVE_FUSE
/* Functions handling the directory cache. */
static int init_dir_caches (guestfs_h *);
static void free_dir_caches (guestfs_h *);
static void dir_cache_remove_all_expired (guestfs_h *, time_t now);
static void dir_cache_invalidate (guestfs_h *, const char *path);
static int lsc_insert (guestfs_h *, const char *path, const char *name, time_t now, struct stat const *statbuf);
static int xac_insert (guestfs_h *, const char *path, const char *name, time_t now, struct guestfs_xattr_list *xattrs);
static int rlc_insert (guestfs_h *, const char *path, const char *name, time_t now, char *link);
static const struct stat *lsc_lookup (guestfs_h *, const char *pathname);
static const struct guestfs_xattr_list *xac_lookup (guestfs_h *, const char *pathname);
static const char *rlc_lookup (guestfs_h *, const char *pathname);
/* This lock protects access to g->localmountpoint. */
gl_lock_define_initialized (static, mount_local_lock);
#define DECL_G() guestfs_h *g = fuse_get_context()->private_data
#define DEBUG_CALL(fs,...) \
if (g->ml_debug_calls) { \
debug (g, \
"%s: %s (" fs ")", \
g->localmountpoint, __func__, ## __VA_ARGS__); \
}
#define RETURN_ERRNO \
do { \
int ret_errno = guestfs_last_errno (g); \
\
/* 0 doesn't mean "no error". It means the errno was not \
* captured. Therefore we have to substitute an errno here. \
*/ \
if (ret_errno == 0) \
ret_errno = EINVAL; \
\
return -ret_errno; \
} while (0)
static struct guestfs_xattr_list *
copy_xattr_list (guestfs_h *g, const struct guestfs_xattr *first, size_t num)
{
struct guestfs_xattr_list *xattrs;
size_t i;
xattrs = malloc (sizeof *xattrs);
if (xattrs == NULL) {
perrorf (g, "malloc");
return NULL;
}
xattrs->len = num;
xattrs->val = malloc (num * sizeof (struct guestfs_xattr));
if (xattrs->val == NULL) {
perrorf (g, "malloc");
free (xattrs);
return NULL;
}
for (i = 0; i < num; ++i) {
xattrs->val[i].attrname = strdup (first[i].attrname);
xattrs->val[i].attrval_len = first[i].attrval_len;
xattrs->val[i].attrval = malloc (first[i].attrval_len);
memcpy (xattrs->val[i].attrval, first[i].attrval, first[i].attrval_len);
}
return xattrs;
}
static int
mount_local_readdir (const char *path, void *buf, fuse_fill_dir_t filler,
off_t offset, struct fuse_file_info *fi)
{
time_t now;
size_t i;
char **names;
CLEANUP_FREE_DIRENT_LIST struct guestfs_dirent_list *ents = NULL;
DECL_G ();
DEBUG_CALL ("%s, %p, %ld", path, buf, (long) offset);
time (&now);
dir_cache_remove_all_expired (g, now);
ents = guestfs_readdir (g, path);
if (ents == NULL)
RETURN_ERRNO;
for (i = 0; i < ents->len; ++i) {
struct stat stat;
memset (&stat, 0, sizeof stat);
stat.st_ino = ents->val[i].ino;
switch (ents->val[i].ftyp) {
case 'b': stat.st_mode = S_IFBLK; break;
case 'c': stat.st_mode = S_IFCHR; break;
case 'd': stat.st_mode = S_IFDIR; break;
case 'f': stat.st_mode = S_IFIFO; break;
case 'l': stat.st_mode = S_IFLNK; break;
case 'r': stat.st_mode = S_IFREG; break;
case 's': stat.st_mode = S_IFSOCK; break;
case 'u':
case '?':
default: stat.st_mode = 0;
}
/* Copied from the example, which also ignores 'offset'. I'm
* not quite sure how this is ever supposed to work on large
* directories. XXX
*/
if (filler (buf, ents->val[i].name, &stat, 0))
break;
}
/* Now prepopulate the directory caches. This step is just an
* optimization, don't worry if it fails.
*/
names = malloc ((ents->len + 1) * sizeof (char *));
if (names) {
CLEANUP_FREE_STATNS_LIST struct guestfs_statns_list *ss = NULL;
CLEANUP_FREE_XATTR_LIST struct guestfs_xattr_list *xattrs = NULL;
char **links;
for (i = 0; i < ents->len; ++i)
names[i] = ents->val[i].name;
names[i] = NULL;
ss = guestfs_lstatnslist (g, path, names);
if (ss) {
for (i = 0; i < ss->len; ++i) {
if (ss->val[i].st_ino >= 0) {
struct stat statbuf;
memset (&statbuf, 0, sizeof statbuf);
statbuf.st_dev = ss->val[i].st_dev;
statbuf.st_ino = ss->val[i].st_ino;
statbuf.st_mode = ss->val[i].st_mode;
statbuf.st_nlink = ss->val[i].st_nlink;
statbuf.st_uid = ss->val[i].st_uid;
statbuf.st_gid = ss->val[i].st_gid;
statbuf.st_rdev = ss->val[i].st_rdev;
statbuf.st_size = ss->val[i].st_size;
statbuf.st_blksize = ss->val[i].st_blksize;
statbuf.st_blocks = ss->val[i].st_blocks;
statbuf.st_atime = ss->val[i].st_atime_sec;
#ifdef HAVE_STRUCT_STAT_ST_ATIM_TV_NSEC
statbuf.st_atim.tv_nsec = ss->val[i].st_atime_nsec;
#endif
statbuf.st_mtime = ss->val[i].st_mtime_sec;
#ifdef HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
statbuf.st_mtim.tv_nsec = ss->val[i].st_mtime_nsec;
#endif
statbuf.st_ctime = ss->val[i].st_ctime_sec;
#ifdef HAVE_STRUCT_STAT_ST_CTIM_TV_NSEC
statbuf.st_ctim.tv_nsec = ss->val[i].st_ctime_nsec;
#endif
lsc_insert (g, path, names[i], now, &statbuf);
}
}
}
xattrs = guestfs_lxattrlist (g, path, names);
if (xattrs) {
size_t ni, num;
struct guestfs_xattr *first;
struct guestfs_xattr_list *copy;
for (i = 0, ni = 0; i < xattrs->len; ++i, ++ni) {
/* assert (strlen (xattrs->val[i].attrname) == 0); */
if (xattrs->val[i].attrval_len > 0) {
++i;
first = &xattrs->val[i];
num = 0;
for (; i < xattrs->len && strlen (xattrs->val[i].attrname) > 0; ++i)
num++;
copy = copy_xattr_list (g, first, num);
if (copy)
xac_insert (g, path, names[ni], now, copy);
i--;
}
}
}
links = guestfs_readlinklist (g, path, names);
if (links) {
for (i = 0; names[i] != NULL; ++i) {
if (links[i][0])
/* Note that rlc_insert owns the string links[i] after this, */
rlc_insert (g, path, names[i], now, links[i]);
else
/* which is why we have to free links[i] here. */
free (links[i]);
}
free (links); /* free the array, not the strings */
}
free (names);
}
return 0;
}
static int
mount_local_getattr (const char *path, struct stat *statbuf)
{
const struct stat *buf;
CLEANUP_FREE_STAT struct guestfs_statns *r = NULL;
DECL_G ();
DEBUG_CALL ("%s, %p", path, statbuf);
buf = lsc_lookup (g, path);
if (buf) {
memcpy (statbuf, buf, sizeof *statbuf);
return 0;
}
r = guestfs_lstatns (g, path);
if (r == NULL)
RETURN_ERRNO;
memset (statbuf, 0, sizeof *statbuf);
statbuf->st_dev = r->st_dev;
statbuf->st_ino = r->st_ino;
statbuf->st_mode = r->st_mode;
statbuf->st_nlink = r->st_nlink;
statbuf->st_uid = r->st_uid;
statbuf->st_gid = r->st_gid;
statbuf->st_rdev = r->st_rdev;
statbuf->st_size = r->st_size;
statbuf->st_blksize = r->st_blksize;
statbuf->st_blocks = r->st_blocks;
statbuf->st_atime = r->st_atime_sec;
#ifdef HAVE_STRUCT_STAT_ST_ATIM_TV_NSEC
statbuf->st_atim.tv_nsec = r->st_atime_nsec;
#endif
statbuf->st_mtime = r->st_mtime_sec;
#ifdef HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
statbuf->st_mtim.tv_nsec = r->st_mtime_nsec;
#endif
statbuf->st_ctime = r->st_ctime_sec;
#ifdef HAVE_STRUCT_STAT_ST_CTIM_TV_NSEC
statbuf->st_ctim.tv_nsec = r->st_ctime_nsec;
#endif
return 0;
}
/* Nautilus loves to use access(2) to test everything about a file,
* such as whether it's executable. Therefore treat this a lot like
* mount_local_getattr.
*/
static int
mount_local_access (const char *path, int mask)
{
struct stat statbuf;
int r;
struct fuse_context *fuse;
int ok = 1;
DECL_G ();
DEBUG_CALL ("%s, %d", path, mask);
if (g->ml_read_only && (mask & W_OK))
return -EROFS;
r = mount_local_getattr (path, &statbuf);
if (r < 0 || mask == F_OK) {
debug (g, "%s: mount_local_getattr returned r = %d", path, r);
return r;
}
fuse = fuse_get_context ();
/* Root user should be able to access everything, so only bother
* with these fine-grained tests for non-root. (RHBZ#1106548).
*/
if (fuse->uid != 0) {
if (mask & R_OK)
ok = ok &&
( fuse->uid == statbuf.st_uid ? statbuf.st_mode & S_IRUSR
: fuse->gid == statbuf.st_gid ? statbuf.st_mode & S_IRGRP
: statbuf.st_mode & S_IROTH);
if (mask & W_OK)
ok = ok &&
( fuse->uid == statbuf.st_uid ? statbuf.st_mode & S_IWUSR
: fuse->gid == statbuf.st_gid ? statbuf.st_mode & S_IWGRP
: statbuf.st_mode & S_IWOTH);
if (mask & X_OK)
ok = ok &&
( fuse->uid == statbuf.st_uid ? statbuf.st_mode & S_IXUSR
: fuse->gid == statbuf.st_gid ? statbuf.st_mode & S_IXGRP
: statbuf.st_mode & S_IXOTH);
}
debug (g, "%s: "
"testing access mask%s%s%s%s: "
"caller UID:GID = %ju:%ju, "
"file UID:GID = %ju:%ju, "
"file mode = %o, "
"result = %s",
path,
mask & R_OK ? " R_OK" : "",
mask & W_OK ? " W_OK" : "",
mask & X_OK ? " X_OK" : "",
mask == 0 ? " 0" : "",
(uintmax_t) fuse->uid, (uintmax_t) fuse->gid,
(uintmax_t) statbuf.st_uid, (uintmax_t) statbuf.st_gid,
statbuf.st_mode,
ok ? "OK" : "EACCESS");
return ok ? 0 : -EACCES;
}
static int
mount_local_readlink (const char *path, char *buf, size_t size)
{
const char *r;
int free_it = 0;
size_t len;
DECL_G ();
DEBUG_CALL ("%s, %p, %zu", path, buf, size);
r = rlc_lookup (g, path);
if (!r) {
r = guestfs_readlink (g, path);
if (r == NULL)
RETURN_ERRNO;
free_it = 1;
}
/* Note this is different from the real readlink(2) syscall. FUSE wants
* the string to be always nul-terminated, even if truncated.
*/
len = strlen (r);
if (len > size - 1)
len = size - 1;
memcpy (buf, r, len);
buf[len] = '\0';
if (free_it) {
char *tmp = (char *) r;
free (tmp);
}
return 0;
}
static int
mount_local_mknod (const char *path, mode_t mode, dev_t rdev)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, 0%o, 0x%jx", path, mode, (uintmax_t) rdev);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_mknod (g, mode, major (rdev), minor (rdev), path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_mkdir (const char *path, mode_t mode)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, 0%o", path, mode);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_mkdir_mode (g, path, mode);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_unlink (const char *path)
{
int r;
DECL_G ();
DEBUG_CALL ("%s", path);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_rm (g, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_rmdir (const char *path)
{
int r;
DECL_G ();
DEBUG_CALL ("%s", path);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_rmdir (g, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_symlink (const char *from, const char *to)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %s", from, to);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, to);
r = guestfs_ln_s (g, from, to);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_rename (const char *from, const char *to)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %s", from, to);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, from);
dir_cache_invalidate (g, to);
r = guestfs_rename (g, from, to);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_link (const char *from, const char *to)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %s", from, to);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, from);
dir_cache_invalidate (g, to);
r = guestfs_ln (g, from, to);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_chmod (const char *path, mode_t mode)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, 0%o", path, mode);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_chmod (g, mode, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_chown (const char *path, uid_t uid, gid_t gid)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %ju, %ju", path, (uintmax_t) uid, (uintmax_t) gid);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_lchown (g, uid, gid, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_truncate (const char *path, off_t size)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %ld", path, (long) size);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_truncate_size (g, path, size);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_utimens (const char *path, const struct timespec ts[2])
{
int r;
time_t atsecs, mtsecs;
long atnsecs, mtnsecs;
DECL_G ();
DEBUG_CALL ("%s, [{ %ld, %ld }, { %ld, %ld }]",
path, ts[0].tv_sec, ts[0].tv_nsec, ts[1].tv_sec, ts[1].tv_nsec);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
atsecs = ts[0].tv_sec;
atnsecs = ts[0].tv_nsec;
mtsecs = ts[1].tv_sec;
mtnsecs = ts[1].tv_nsec;
#ifdef UTIME_NOW
if (atnsecs == UTIME_NOW)
atnsecs = -1;
#endif
#ifdef UTIME_OMIT
if (atnsecs == UTIME_OMIT)
atnsecs = -2;
#endif
#ifdef UTIME_NOW
if (mtnsecs == UTIME_NOW)
mtnsecs = -1;
#endif
#ifdef UTIME_OMIT
if (mtnsecs == UTIME_OMIT)
mtnsecs = -2;
#endif
r = guestfs_utimens (g, path, atsecs, atnsecs, mtsecs, mtnsecs);
if (r == -1)
RETURN_ERRNO;
return 0;
}
/* All this function needs to do is to check that the requested open
* flags are valid. See the notes in <fuse/fuse.h>.
*/
static int
mount_local_open (const char *path, struct fuse_file_info *fi)
{
const int flags = fi->flags & O_ACCMODE;
DECL_G ();
DEBUG_CALL ("%s, 0%o", path, (unsigned) fi->flags);
if (g->ml_read_only && flags != O_RDONLY)
return -EROFS;
return 0;
}
static int
mount_local_read (const char *path, char *buf, size_t size, off_t offset,
struct fuse_file_info *fi)
{
char *r;
size_t rsize;
const size_t limit = 2 * 1024 * 1024;
DECL_G ();
DEBUG_CALL ("%s, %p, %zu, %ld", path, buf, size, (long) offset);
/* The guestfs protocol limits size to somewhere over 2MB. We just
* reduce the requested size here accordingly and push the problem
* up to every user. http://www.jwz.org/doc/worse-is-better.html
*/
if (size > limit)
size = limit;
r = guestfs_pread (g, path, size, offset, &rsize);
if (r == NULL)
RETURN_ERRNO;
/* This should never happen, but at least it stops us overflowing
* the output buffer if it does happen.
*/
if (rsize > size)
rsize = size;
memcpy (buf, r, rsize);
free (r);
return rsize;
}
static int
mount_local_write (const char *path, const char *buf, size_t size,
off_t offset, struct fuse_file_info *fi)
{
const size_t limit = 2 * 1024 * 1024;
int r;
DECL_G ();
DEBUG_CALL ("%s, %p, %zu, %ld", path, buf, size, (long) offset);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
/* See mount_local_read. */
if (size > limit)
size = limit;
r = guestfs_pwrite (g, path, buf, size, offset);
if (r == -1)
RETURN_ERRNO;
return r;
}
static int
mount_local_statfs (const char *path, struct statvfs *stbuf)
{
CLEANUP_FREE_STATVFS struct guestfs_statvfs *r;
DECL_G ();
DEBUG_CALL ("%s, %p", path, stbuf);
r = guestfs_statvfs (g, path);
if (r == NULL)
RETURN_ERRNO;
stbuf->f_bsize = r->bsize;
stbuf->f_frsize = r->frsize;
stbuf->f_blocks = r->blocks;
stbuf->f_bfree = r->bfree;
stbuf->f_bavail = r->bavail;
stbuf->f_files = r->files;
stbuf->f_ffree = r->ffree;
stbuf->f_favail = r->favail;
stbuf->f_fsid = r->fsid;
stbuf->f_flag = r->flag;
stbuf->f_namemax = r->namemax;
return 0;
}
static int
mount_local_release (const char *path, struct fuse_file_info *fi)
{
DECL_G ();
DEBUG_CALL ("%s", path);
/* Just a stub. This method is optional and can safely be left
* unimplemented.
*/
return 0;
}
/* Emulate this by calling sync. */
static int
mount_local_fsync (const char *path, int isdatasync,
struct fuse_file_info *fi)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %d", path, isdatasync);
r = guestfs_sync (g);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_setxattr (const char *path, const char *name, const char *value,
size_t size, int flags)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %s, %p, %zu", path, name, value, size);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
/* XXX Underlying guestfs(3) API doesn't understand the flags. */
r = guestfs_lsetxattr (g, name, value, size, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
/* The guestfs(3) API for getting xattrs is much easier to use
* than the real syscall. Unfortunately we now have to emulate
* the real syscall using that API :-(
*/
static int
mount_local_getxattr (const char *path, const char *name, char *value,
size_t size)
{
const struct guestfs_xattr_list *xattrs;
int free_attrs = 0;
ssize_t r;
size_t i, sz;
DECL_G ();
DEBUG_CALL ("%s, %s, %p, %zu", path, name, value, size);
xattrs = xac_lookup (g, path);
if (xattrs == NULL) {
xattrs = guestfs_lgetxattrs (g, path);
if (xattrs == NULL)
RETURN_ERRNO;
free_attrs = 1;
}
/* Find the matching attribute (index in 'i'). */
for (i = 0; i < xattrs->len; ++i) {
if (STREQ (xattrs->val[i].attrname, name))
break;
}
if (i == xattrs->len) { /* not found */
r = -ENOATTR;
goto out;
}
/* The getxattr man page is unclear, but if value == NULL then we
* return the space required (the caller then makes a second syscall
* after allocating the required amount of space). If value != NULL
* then it's not clear what we should do, but it appears we should
* copy as much as possible and return -ERANGE if there's not enough
* space in the buffer.
*/
sz = xattrs->val[i].attrval_len;
if (value == NULL) {
r = sz;
goto out;
}
if (sz <= size)
r = sz;
else {
r = -ERANGE;
sz = size;
}
memcpy (value, xattrs->val[i].attrval, sz);
out:
if (free_attrs)
guestfs_free_xattr_list ((struct guestfs_xattr_list *) xattrs);
return r;
}
/* Ditto as above. */
static int
mount_local_listxattr (const char *path, char *list, size_t size)
{
const struct guestfs_xattr_list *xattrs;
int free_attrs = 0;
size_t space = 0;
size_t len;
size_t i;
ssize_t r;
DECL_G ();
DEBUG_CALL ("%s, %p, %zu", path, list, size);
xattrs = xac_lookup (g, path);
if (xattrs == NULL) {
xattrs = guestfs_lgetxattrs (g, path);
if (xattrs == NULL)
RETURN_ERRNO;
free_attrs = 1;
}
/* Calculate how much space is required to hold the result. */
for (i = 0; i < xattrs->len; ++i) {
len = strlen (xattrs->val[i].attrname) + 1;
space += len;
}
/* The listxattr man page is unclear, but if list == NULL then we
* return the space required (the caller then makes a second syscall
* after allocating the required amount of space). If list != NULL
* then it's not clear what we should do, but it appears we should
* copy as much as possible and return -ERANGE if there's not enough
* space in the buffer.
*/
if (list == NULL) {
r = space;
goto out;
}
r = 0;
for (i = 0; i < xattrs->len; ++i) {
len = strlen (xattrs->val[i].attrname) + 1;
if (size >= len) {
memcpy (list, xattrs->val[i].attrname, len);
size -= len;
list += len;
r += len;
} else {
r = -ERANGE;
break;
}
}
out:
if (free_attrs)
guestfs_free_xattr_list ((struct guestfs_xattr_list *) xattrs);
return r;
}
static int
mount_local_removexattr(const char *path, const char *name)
{
int r;
DECL_G ();
DEBUG_CALL ("%s, %s", path, name);
if (g->ml_read_only) return -EROFS;
dir_cache_invalidate (g, path);
r = guestfs_lremovexattr (g, name, path);
if (r == -1)
RETURN_ERRNO;
return 0;
}
static int
mount_local_flush(const char *path, struct fuse_file_info *fi)
{
DECL_G ();
DEBUG_CALL ("%s", path);
/* Just a stub. This method is called whenever FUSE wants to flush the
* pending changes (f.ex. to attributes) to a file. Since we don't have
* anything to do and don't want FUSE to think something went badly,
* just return 0.
*/
return 0;
}
static struct fuse_operations mount_local_operations = {
.getattr = mount_local_getattr,
.access = mount_local_access,
.readlink = mount_local_readlink,
.readdir = mount_local_readdir,
.mknod = mount_local_mknod,
.mkdir = mount_local_mkdir,
.symlink = mount_local_symlink,
.unlink = mount_local_unlink,
.rmdir = mount_local_rmdir,
.rename = mount_local_rename,
.link = mount_local_link,
.chmod = mount_local_chmod,
.chown = mount_local_chown,
.truncate = mount_local_truncate,
.utimens = mount_local_utimens,
.open = mount_local_open,
.read = mount_local_read,
.write = mount_local_write,
.statfs = mount_local_statfs,
.release = mount_local_release,
.fsync = mount_local_fsync,
.setxattr = mount_local_setxattr,
.getxattr = mount_local_getxattr,
.listxattr = mount_local_listxattr,
.removexattr = mount_local_removexattr,
.flush = mount_local_flush,
};
int
guestfs_impl_mount_local (guestfs_h *g, const char *localmountpoint,
const struct guestfs_mount_local_argv *optargs)
{
const char *t;
struct fuse_args args = FUSE_ARGS_INIT (0, NULL);
struct fuse_chan *ch;
int fd;
/* You can only mount each handle in one place in one thread. */
gl_lock_lock (mount_local_lock);
t = g->localmountpoint;
gl_lock_unlock (mount_local_lock);
if (t) {
error (g, _("filesystem is already mounted in another thread"));
return -1;
}
if (optargs->bitmask & GUESTFS_MOUNT_LOCAL_READONLY_BITMASK)
g->ml_read_only = optargs->readonly;
else
g->ml_read_only = 0;
if (optargs->bitmask & GUESTFS_MOUNT_LOCAL_CACHETIMEOUT_BITMASK)
g->ml_dir_cache_timeout = optargs->cachetimeout;
else
g->ml_dir_cache_timeout = 60;
if (optargs->bitmask & GUESTFS_MOUNT_LOCAL_DEBUGCALLS_BITMASK)
g->ml_debug_calls = optargs->debugcalls;
else
g->ml_debug_calls = 0;
/* Initialize the directory caches in the handle. */
if (init_dir_caches (g) == -1)
return -1;
/* Create the FUSE 'args'. */
if (fuse_opt_add_arg (&args, g->program) == -1) {
arg_error:
perrorf (g, _("fuse_opt_add_arg: %s"), localmountpoint);
fuse_opt_free_args (&args);
guestfs_int_free_fuse (g);
return -1;
}
if (optargs->bitmask & GUESTFS_MOUNT_LOCAL_OPTIONS_BITMASK) {
if (fuse_opt_add_arg (&args, "-o") == -1 ||
fuse_opt_add_arg (&args, optargs->options) == -1)
goto arg_error;
}
debug (g, "%s: fuse_mount %s", __func__, localmountpoint);
/* Create the FUSE mountpoint. */
ch = fuse_mount (localmountpoint, &args);
if (ch == NULL) {
perrorf (g, _("fuse_mount: %s"), localmountpoint);
fuse_opt_free_args (&args);
guestfs_int_free_fuse (g);
return -1;
}
/* Set F_CLOEXEC on the channel. XXX libfuse should do this. */
fd = fuse_chan_fd (ch);
if (fd >= 0)
set_cloexec_flag (fd, 1);
debug (g, "%s: fuse_new", __func__);
/* Create the FUSE handle. */
g->fuse = fuse_new (ch, &args,
&mount_local_operations, sizeof mount_local_operations,
g);
if (!g->fuse) {
perrorf (g, _("fuse_new: %s"), localmountpoint);
fuse_unmount (localmountpoint, ch);
fuse_opt_free_args (&args);
guestfs_int_free_fuse (g);
return -1;
}
fuse_opt_free_args (&args);
debug (g, "%s: leaving fuse_mount_local", __func__);
/* Set g->localmountpoint in the handle. */
gl_lock_lock (mount_local_lock);
g->localmountpoint = localmountpoint;
gl_lock_unlock (mount_local_lock);
return 0;
}
int
guestfs_impl_mount_local_run (guestfs_h *g)
{
int r, mounted;
gl_lock_lock (mount_local_lock);
mounted = g->localmountpoint != NULL;
gl_lock_unlock (mount_local_lock);
if (!mounted) {
error (g, _("you must call guestfs_mount_local first"));
return -1;
}
/* Test if root is mounted. We do this by using a side-effect of
* guestfs_exists (which is that it calls NEED_ROOT).
*/
guestfs_push_error_handler (g, NULL, NULL);
r = guestfs_exists (g, "/");
guestfs_pop_error_handler (g);
if (r == -1) {
error (g, _("you must call 'guestfs_mount' first to mount a filesystem on '/'.\nNote: '%s' is still mounted. Use 'guestunmount %s' to clean up."),
g->localmountpoint, g->localmountpoint);
return -1;
}
debug (g, "%s: entering fuse_loop", __func__);
/* Enter the main loop. */
r = fuse_loop (g->fuse);
if (r != 0)
perrorf (g, _("fuse_loop: %s"), g->localmountpoint);
debug (g, "%s: leaving fuse_loop", __func__);
guestfs_int_free_fuse (g);
gl_lock_lock (mount_local_lock);
g->localmountpoint = NULL;
gl_lock_unlock (mount_local_lock);
/* By inspection, I found that fuse_loop only returns 0 or -1, but
* don't rely on this in future.
*/
return r == 0 ? 0 : -1;
}
void
guestfs_int_free_fuse (guestfs_h *g)
{
if (g->fuse)
fuse_destroy (g->fuse); /* also closes the channel */
g->fuse = NULL;
free_dir_caches (g);
}
int
guestfs_impl_umount_local (guestfs_h *g,
const struct guestfs_umount_local_argv *optargs)
{
const char *retry;
int r;
CLEANUP_FREE char *localmountpoint = NULL;
CLEANUP_CMD_CLOSE struct command *cmd = NULL;
/* How many times should we try the fusermount command? */
if (optargs->bitmask & GUESTFS_UMOUNT_LOCAL_RETRY_BITMASK)
retry = optargs->retry ? "--retry=5" : "--no-retry";
else
retry = "--no-retry";
/* Make a local copy of g->localmountpoint. It could be freed from
* under us by another thread, except when we are holding the lock.
*/
gl_lock_lock (mount_local_lock);
if (g->localmountpoint)
localmountpoint = safe_strdup (g, g->localmountpoint);
else
localmountpoint = NULL;
gl_lock_unlock (mount_local_lock);
if (!localmountpoint) {
error (g, _("no filesystem is mounted"));
return -1;
}
/* Run guestunmount --retry=... localmountpoint. */
cmd = guestfs_int_new_command (g);
guestfs_int_cmd_add_arg (cmd, "guestunmount");
guestfs_int_cmd_add_arg (cmd, retry);
guestfs_int_cmd_add_arg (cmd, localmountpoint);
r = guestfs_int_cmd_run (cmd);
if (r == -1)
return -1;
if (WIFEXITED (r) && WEXITSTATUS (r) == EXIT_SUCCESS)
/* External fusermount succeeded. Note that the original thread
* is responsible for setting g->localmountpoint to NULL.
*/
return 0;
return -1;
}
/* Functions handling the directory cache.
*
* Note on attribute caching: FUSE can cache filesystem attributes for
* short periods of time (configurable via -o attr_timeout). It
* doesn't cache xattrs, and in any case FUSE caching doesn't solve
* the problem that we have to make a series of guestfs_lstatns and
* guestfs_lgetxattr calls when we first list a directory (thus, many
* round trips).
*
* For this reason, we also implement a readdir cache here which is
* invoked when a readdir call is made. readdir is modified so that
* as well as reading the directory, it also requests all the stat
* structures, xattrs and readlinks of all entries in the directory,
* and these are added to the cache here (for a short, configurable
* period of time) in anticipation that they will be needed
* immediately afterwards, which is usually the case when the user is
* doing an "ls"-like operation.
*
* You can still use FUSE attribute caching on top of this mechanism
* if you like.
*/
struct entry_common {
char *pathname; /* full path to the file */
time_t timeout; /* when this entry expires */
};
struct lsc_entry { /* lstat cache entry */
struct entry_common c;
struct stat statbuf; /* statbuf */
};
struct xac_entry { /* xattr cache entry */
struct entry_common c;
struct guestfs_xattr_list *xattrs;
};
struct rlc_entry { /* readlink cache entry */
struct entry_common c;
char *link;
};
static size_t
gen_hash (void const *x, size_t table_size)
{
struct entry_common const *p = x;
return hash_pjw (p->pathname, table_size);
}
static bool
gen_compare (void const *x, void const *y)
{
struct entry_common const *a = x;
struct entry_common const *b = y;
return STREQ (a->pathname, b->pathname);
}
static void
lsc_free (void *x)
{
if (x) {
struct entry_common *p = x;
free (p->pathname);
free (p);
}
}
static void
xac_free (void *x)
{
if (x) {
struct xac_entry *p = x;
guestfs_free_xattr_list (p->xattrs);
lsc_free (x);
}
}
static void
rlc_free (void *x)
{
if (x) {
struct rlc_entry *p = x;
free (p->link);
lsc_free (x);
}
}
static int
init_dir_caches (guestfs_h *g)
{
g->lsc_ht = hash_initialize (1024, NULL, gen_hash, gen_compare, lsc_free);
g->xac_ht = hash_initialize (1024, NULL, gen_hash, gen_compare, xac_free);
g->rlc_ht = hash_initialize (1024, NULL, gen_hash, gen_compare, rlc_free);
if (!g->lsc_ht || !g->xac_ht || !g->rlc_ht) {
error (g, _("could not initialize dir cache hashtables"));
return -1;
}
return 0;
}
static void
free_dir_caches (guestfs_h *g)
{
if (g->lsc_ht)
hash_free (g->lsc_ht);
if (g->xac_ht)
hash_free (g->xac_ht);
if (g->rlc_ht)
hash_free (g->rlc_ht);
g->lsc_ht = NULL;
g->xac_ht = NULL;
g->rlc_ht = NULL;
}
struct gen_remove_data {
time_t now;
Hash_table *ht;
Hash_data_freer freer;
};
static bool
gen_remove_if_expired (void *x, void *data)
{
/* XXX hash_do_for_each was observed calling this function
* with x == NULL.
*/
if (x) {
struct entry_common *p = x;
struct gen_remove_data *d = data;
if (p->timeout < d->now)
d->freer (hash_delete (d->ht, x));
}
return 1;
}
static void
gen_remove_all_expired (Hash_table *ht, Hash_data_freer freer, time_t now)
{
struct gen_remove_data data;
data.now = now;
data.ht = ht;
data.freer = freer;
/* Careful reading of the documentation to hash _seems_ to indicate
* that this is safe, _provided_ we use the default thresholds (in
* particular, no shrink threshold).
*/
hash_do_for_each (ht, gen_remove_if_expired, &data);
}
static void
dir_cache_remove_all_expired (guestfs_h *g, time_t now)
{
gen_remove_all_expired (g->lsc_ht, lsc_free, now);
gen_remove_all_expired (g->xac_ht, xac_free, now);
gen_remove_all_expired (g->rlc_ht, rlc_free, now);
}
static int
gen_replace (guestfs_h *g, Hash_table *ht,
struct entry_common *new_entry, Hash_data_freer freer)
{
struct entry_common *old_entry;
old_entry = hash_delete (ht, new_entry);
freer (old_entry);
old_entry = hash_insert (ht, new_entry);
if (old_entry == NULL) {
perrorf (g, "hash_insert");
freer (new_entry);
return -1;
}
/* assert (old_entry == new_entry); */
return 0;
}
static int
lsc_insert (guestfs_h *g,
const char *path, const char *name, time_t now,
struct stat const *statbuf)
{
struct lsc_entry *entry;
size_t len;
entry = malloc (sizeof *entry);
if (entry == NULL) {
perrorf (g, "malloc");
return -1;
}
len = strlen (path) + strlen (name) + 2;
entry->c.pathname = malloc (len);
if (entry->c.pathname == NULL) {
perrorf (g, "malloc");
free (entry);
return -1;
}
if (STREQ (path, "/"))
snprintf (entry->c.pathname, len, "/%s", name);
else
snprintf (entry->c.pathname, len, "%s/%s", path, name);
memcpy (&entry->statbuf, statbuf, sizeof entry->statbuf);
entry->c.timeout = now + g->ml_dir_cache_timeout;
return gen_replace (g, g->lsc_ht, (struct entry_common *) entry, lsc_free);
}
static int
xac_insert (guestfs_h *g,
const char *path, const char *name, time_t now,
struct guestfs_xattr_list *xattrs)
{
struct xac_entry *entry;
size_t len;
entry = malloc (sizeof *entry);
if (entry == NULL) {
perrorf (g, "malloc");
return -1;
}
len = strlen (path) + strlen (name) + 2;
entry->c.pathname = malloc (len);
if (entry->c.pathname == NULL) {
perrorf (g, "malloc");
free (entry);
return -1;
}
if (STREQ (path, "/"))
snprintf (entry->c.pathname, len, "/%s", name);
else
snprintf (entry->c.pathname, len, "%s/%s", path, name);
entry->xattrs = xattrs;
entry->c.timeout = now + g->ml_dir_cache_timeout;
return gen_replace (g, g->xac_ht, (struct entry_common *) entry, xac_free);
}
static int
rlc_insert (guestfs_h *g,
const char *path, const char *name, time_t now,
char *link)
{
struct rlc_entry *entry;
size_t len;
entry = malloc (sizeof *entry);
if (entry == NULL) {
perrorf (g, "malloc");
return -1;
}
len = strlen (path) + strlen (name) + 2;
entry->c.pathname = malloc (len);
if (entry->c.pathname == NULL) {
perrorf (g, "malloc");
free (entry);
return -1;
}
if (STREQ (path, "/"))
snprintf (entry->c.pathname, len, "/%s", name);
else
snprintf (entry->c.pathname, len, "%s/%s", path, name);
entry->link = link;
entry->c.timeout = now + g->ml_dir_cache_timeout;
return gen_replace (g, g->rlc_ht, (struct entry_common *) entry, rlc_free);
}
static const struct stat *
lsc_lookup (guestfs_h *g, const char *pathname)
{
const struct entry_common key = { .pathname = (char *) pathname };
struct lsc_entry *entry;
time_t now;
time (&now);
entry = hash_lookup (g->lsc_ht, &key);
if (entry && entry->c.timeout >= now)
return &entry->statbuf;
else
return NULL;
}
static const struct guestfs_xattr_list *
xac_lookup (guestfs_h *g, const char *pathname)
{
const struct entry_common key = { .pathname = (char *) pathname };
struct xac_entry *entry;
time_t now;
time (&now);
entry = hash_lookup (g->xac_ht, &key);
if (entry && entry->c.timeout >= now)
return entry->xattrs;
else
return NULL;
}
static const char *
rlc_lookup (guestfs_h *g, const char *pathname)
{
const struct entry_common key = { .pathname = (char *) pathname };
struct rlc_entry *entry;
time_t now;
time (&now);
entry = hash_lookup (g->rlc_ht, &key);
if (entry && entry->c.timeout >= now)
return entry->link;
else
return NULL;
}
static void
gen_remove (Hash_table *ht, const char *pathname, Hash_data_freer freer)
{
const struct entry_common key = { .pathname = (char *) pathname };
struct entry_common *entry;
entry = hash_delete (ht, &key);
freer (entry);
}
static void
dir_cache_invalidate (guestfs_h *g, const char *path)
{
gen_remove (g->lsc_ht, path, lsc_free);
gen_remove (g->xac_ht, path, xac_free);
gen_remove (g->rlc_ht, path, rlc_free);
}
#else /* !HAVE_FUSE */
#define FUSE_NOT_SUPPORTED() \
NOT_SUPPORTED (g, -1, _("FUSE is not supported in this build of " \
"libguestfs because libfuse was not found " \
"when libguestfs was compiled"))
int
guestfs_impl_mount_local (guestfs_h *g, const char *localmountpoint,
const struct guestfs_mount_local_argv *optargs)
{
FUSE_NOT_SUPPORTED ();
}
int
guestfs_impl_mount_local_run (guestfs_h *g)
{
FUSE_NOT_SUPPORTED ();
}
int
guestfs_impl_umount_local (guestfs_h *g,
const struct guestfs_umount_local_argv *optargs)
{
FUSE_NOT_SUPPORTED ();
}
#endif /* !HAVE_FUSE */