See the file COPYING.LIB
*/
+#define _GNU_SOURCE
+
#include "fuse_i.h"
#include "fuse_kernel.h"
#include "fuse_opt.h"
#include <limits.h>
#include <errno.h>
+#ifndef F_LINUX_SPECIFIC_BASE
+#define F_LINUX_SPECIFIC_BASE 1024
+#endif
+#ifndef F_SETPIPE_SZ
+#define F_SETPIPE_SZ (F_LINUX_SPECIFIC_BASE + 7)
+#endif
+
+
#define PARAM(inarg) (((char *)(inarg)) + sizeof(*(inarg)))
#define OFFSET_MAX 0x7fffffffffffffffLL
padded_iov = malloc((count + 1) * sizeof(struct iovec));
if (padded_iov == NULL)
- return fuse_reply_err(req, -ENOMEM);
+ return fuse_reply_err(req, ENOMEM);
memcpy(padded_iov + 1, iov, count * sizeof(struct iovec));
count++;
return send_reply_ok(req, buf, size);
}
+struct fuse_ll_pipe {
+ size_t size;
+ int can_grow;
+ int pipe[2];
+};
+
+static void fuse_ll_pipe_free(struct fuse_ll_pipe *llp)
+{
+ close(llp->pipe[0]);
+ close(llp->pipe[1]);
+ free(llp);
+}
+
+static struct fuse_ll_pipe *fuse_ll_get_pipe(struct fuse_ll *f)
+{
+ struct fuse_ll_pipe *llp = pthread_getspecific(f->pipe_key);
+ if (llp == NULL) {
+ int res;
+
+ llp = malloc(sizeof(struct fuse_ll_pipe));
+ if (llp == NULL)
+ return NULL;
+
+ res = pipe(llp->pipe);
+ if (res == -1) {
+ free(llp);
+ return NULL;
+ }
+ /* the default size is 16 pages on linux
+ */
+ llp->size = getpagesize() * 16;
+ llp->can_grow = 1;
+
+ pthread_setspecific(f->pipe_key, llp);
+ }
+
+ return llp;
+}
+
+int fuse_reply_fd(fuse_req_t req, int fd, loff_t *off, size_t len,
+ unsigned int flags)
+{
+ int res;
+ void *buf;
+ struct fuse_out_header out;
+ struct iovec iov;
+ struct fuse_ll_pipe *llp;
+ int splice_flags;
+ size_t pipesize;
+
+ static size_t pagesize = 0;
+ if (!pagesize)
+ pagesize = getpagesize();
+
+ if (req->f->conn.proto_minor < 14 ||
+ !(req->f->conn.want & FUSE_CAP_SPLICE_WRITE))
+ goto fallback;
+
+ llp = fuse_ll_get_pipe(req->f);
+ if (llp == NULL)
+ goto fallback;
+
+
+ /*
+ * Heuristic for the required pipe size, does not work if the
+ * source contains less than page size fragments
+ */
+ pipesize = pagesize * 2 + len;
+
+ if (llp->size < pipesize) {
+ if (llp->can_grow) {
+ res = fcntl(llp->pipe[0], F_SETPIPE_SZ, pipesize);
+ if (res == -1) {
+ llp->can_grow = 0;
+ goto fallback;
+ }
+ llp->size = res;
+ }
+ if (llp->size < pipesize)
+ goto fallback;
+ }
+
+ out.unique = req->unique;
+ out.error = 0;
+ out.len = len + sizeof(struct fuse_out_header);
+
+ iov.iov_base = &out;
+ iov.iov_len = sizeof(struct fuse_out_header);
+
+ res = vmsplice(llp->pipe[1], &iov, 1, 0);
+ if (res == -1) {
+ res = -errno;
+ perror("fuse: vmsplice to pipe");
+ return res;
+ }
+ if (res != sizeof(struct fuse_out_header)) {
+ res = -EIO;
+ fprintf(stderr, "fuse: short vmsplice to pipe: %u/%zu\n", res,
+ sizeof(struct fuse_out_header));
+ goto clear_pipe;
+ }
+
+ res = splice(fd, off, llp->pipe[1], NULL, len, 0);
+ if (res == -1) {
+ res = fuse_reply_err(req, errno);
+ goto clear_pipe;
+ }
+ len = res;
+ out.len = len + sizeof(struct fuse_out_header);
+
+ if (req->f->debug) {
+ fprintf(stderr,
+ " unique: %llu, success, outsize: %i (splice)\n",
+ (unsigned long long) out.unique, out.len);
+ }
+
+ splice_flags = 0;
+ if ((flags & FUSE_REPLY_FD_MOVE) &&
+ (req->f->conn.want & FUSE_CAP_SPLICE_MOVE))
+ splice_flags |= SPLICE_F_MOVE;
+
+ res = splice(llp->pipe[0], NULL,
+ fuse_chan_fd(req->ch), NULL, out.len, flags);
+ if (res == -1) {
+ res = -errno;
+ perror("fuse: splice from pipe");
+ goto clear_pipe;
+ }
+ if (res != out.len) {
+ res = -EIO;
+ fprintf(stderr, "fuse: short splice from pipe: %u/%u\n",
+ res, out.len);
+ goto clear_pipe;
+ }
+ return 0;
+
+clear_pipe:
+ pthread_setspecific(req->f->pipe_key, NULL);
+ fuse_ll_pipe_free(llp);
+ return res;
+
+fallback:
+ res = posix_memalign(&buf, pagesize, len);
+ if (res != 0)
+ return fuse_reply_err(req, res);
+
+ if (off != NULL) {
+ res = pread(fd, buf, len, *off);
+ if (res > 0)
+ *off += res;
+ } else {
+ res = read(fd, buf, len);
+ }
+ if (res == -1)
+ res = fuse_reply_err(req, errno);
+ else
+ res = fuse_reply_buf(req, buf, res);
+ free(buf);
+
+ return res;
+}
+
int fuse_reply_statfs(fuse_req_t req, const struct statvfs *stbuf)
{
struct fuse_statfs_out arg;
padded_iov = malloc((count + 2) * sizeof(struct iovec));
if (padded_iov == NULL)
- return fuse_reply_err(req, -ENOMEM);
+ return fuse_reply_err(req, ENOMEM);
memset(&arg, 0, sizeof(arg));
arg.result = result;
f->conn.max_readahead = 0;
}
+ if (req->f->conn.proto_minor >= 14) {
+ f->conn.capable |= FUSE_CAP_SPLICE_WRITE | FUSE_CAP_SPLICE_MOVE;
+ if (!f->no_splice_write)
+ f->conn.want |= FUSE_CAP_SPLICE_WRITE;
+ if (!f->no_splice_move)
+ f->conn.want |= FUSE_CAP_SPLICE_MOVE;
+ }
+
if (f->atomic_o_trunc)
f->conn.want |= FUSE_CAP_ATOMIC_O_TRUNC;
if (f->op.getlk && f->op.setlk && !f->no_remote_lock)
{ "atomic_o_trunc", offsetof(struct fuse_ll, atomic_o_trunc), 1},
{ "no_remote_lock", offsetof(struct fuse_ll, no_remote_lock), 1},
{ "big_writes", offsetof(struct fuse_ll, big_writes), 1},
+ { "no_splice_write", offsetof(struct fuse_ll, no_splice_write), 1},
+ { "no_splice_move", offsetof(struct fuse_ll, no_splice_move), 1},
FUSE_OPT_KEY("max_read=", FUSE_OPT_KEY_DISCARD),
FUSE_OPT_KEY("-h", KEY_HELP),
FUSE_OPT_KEY("--help", KEY_HELP),
" -o sync_read perform reads synchronously\n"
" -o atomic_o_trunc enable atomic open+truncate support\n"
" -o big_writes enable larger than 4kB writes\n"
-" -o no_remote_lock disable remote file locking\n");
+" -o no_remote_lock disable remote file locking\n"
+" -o no_splice_write don't use splice to write to the fuse device\n"
+" -o no_splice_move don't move data while splicing to the fuse device\n"
+);
}
static int fuse_ll_opt_proc(void *data, const char *arg, int key,
static void fuse_ll_destroy(void *data)
{
struct fuse_ll *f = (struct fuse_ll *) data;
+ struct fuse_ll_pipe *llp;
if (f->got_init && !f->got_destroy) {
if (f->op.destroy)
f->op.destroy(f->userdata);
}
-
+ llp = pthread_getspecific(f->pipe_key);
+ if (llp != NULL)
+ fuse_ll_pipe_free(llp);
+ pthread_key_delete(f->pipe_key);
pthread_mutex_destroy(&f->lock);
free(f->cuse_data);
free(f);
}
+static void fuse_ll_pipe_destructor(void *data)
+{
+ struct fuse_ll_pipe *llp = data;
+ fuse_ll_pipe_free(llp);
+}
+
/*
* always call fuse_lowlevel_new_common() internally, to work around a
* misfeature in the FreeBSD runtime linker, which links the old
const struct fuse_lowlevel_ops *op,
size_t op_size, void *userdata)
{
+ int err;
struct fuse_ll *f;
struct fuse_session *se;
struct fuse_session_ops sop = {
list_init_req(&f->interrupts);
fuse_mutex_init(&f->lock);
- if (fuse_opt_parse(args, f, fuse_ll_opts, fuse_ll_opt_proc) == -1)
+ err = pthread_key_create(&f->pipe_key, fuse_ll_pipe_destructor);
+ if (err) {
+ fprintf(stderr, "fuse: failed to create thread specific key: %s\n",
+ strerror(err));
goto out_free;
+ }
+
+ if (fuse_opt_parse(args, f, fuse_ll_opts, fuse_ll_opt_proc) == -1)
+ goto out_key_destroy;
if (f->debug)
fprintf(stderr, "FUSE library version: %s\n", PACKAGE_VERSION);
se = fuse_session_new(&sop, f);
if (!se)
- goto out_free;
+ goto out_key_destroy;
return se;
+out_key_destroy:
+ pthread_key_delete(f->pipe_key);
out_free:
+ pthread_mutex_destroy(&f->lock);
free(f);
out:
return NULL;