static struct dentry_operations fuse_dentry_operations;
-/* FIXME: This should be user configurable */
-#define FUSE_REVALIDATE_TIME (1 * HZ)
-
#ifndef KERNEL_2_6
#define new_decode_dev(x) (x)
#define new_encode_dev(x) (x)
}
static int fuse_do_lookup(struct inode *dir, struct dentry *entry,
- struct fuse_lookup_out *outarg, int *version)
+ struct fuse_entry_out *outarg, int *version)
{
struct fuse_conn *fc = INO_FC(dir);
struct fuse_in in = FUSE_IN_INIT;
in.args[0].size = entry->d_name.len + 1;
in.args[0].value = entry->d_name.name;
out.numargs = 1;
- out.args[0].size = sizeof(struct fuse_lookup_out);
+ out.args[0].size = sizeof(struct fuse_entry_out);
out.args[0].value = outarg;
request_send(fc, &in, &out);
return out.h.error;
}
+static inline unsigned long time_to_jiffies(unsigned long sec,
+ unsigned long nsec)
+{
+ /* prevent wrapping of jiffies */
+ if(sec + 1 >= LONG_MAX / HZ)
+ return 0;
+
+ return jiffies + sec * HZ + nsec / (1000000000 / HZ);
+}
+
static int fuse_lookup_iget(struct inode *dir, struct dentry *entry,
struct inode **inodep)
{
int err;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
int version;
struct inode *inode = NULL;
} else if(err != -ENOENT)
return err;
- entry->d_time = jiffies;
+ entry->d_time = time_to_jiffies(outarg.entry_valid,
+ outarg.entry_valid_nsec);
entry->d_op = &fuse_dentry_operations;
*inodep = inode;
return 0;
}
static int lookup_new_entry(struct inode *dir, struct dentry *entry,
- struct fuse_lookup_out *outarg, int version,
+ struct fuse_entry_out *outarg, int version,
int mode)
{
struct inode *inode;
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
struct fuse_mknod_in inarg;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
memset(&inarg, 0, sizeof(inarg));
inarg.mode = mode;
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
struct fuse_mkdir_in inarg;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
memset(&inarg, 0, sizeof(inarg));
inarg.mode = mode;
struct fuse_conn *fc = INO_FC(dir);
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
in.h.opcode = FUSE_SYMLINK;
in.h.ino = dir->i_ino;
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
struct fuse_link_in inarg;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
memset(&inarg, 0, sizeof(inarg));
inarg.newdir = newdir->i_ino;
struct fuse_conn *fc = INO_FC(inode);
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
- struct fuse_getattr_out arg;
+ struct fuse_attr_out arg;
in.h.opcode = FUSE_GETATTR;
in.h.ino = inode->i_ino;
if(!(fc->flags & FUSE_ALLOW_OTHER) &&
current->fsuid != fc->uid)
return -EACCES;
- } else if(time_before_eq(jiffies, entry->d_time + FUSE_REVALIDATE_TIME))
+ } else if(!entry->d_time || time_before_eq(jiffies, entry->d_time))
return 0;
return fuse_do_getattr(inode);
struct fuse_conn *fc = INO_FC(inode);
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
- struct fuse_getdir_out outarg;
+ struct fuse_getdir_out_i outarg;
in.h.opcode = FUSE_GETDIR;
in.h.ino = inode->i_ino;
out.numargs = 1;
- out.args[0].size = sizeof(outarg);
+ out.args[0].size = sizeof(struct fuse_getdir_out);
out.args[0].value = &outarg;
request_send(fc, &in, &out);
if(!out.h.error) {
struct fuse_in in = FUSE_IN_INIT;
struct fuse_out out = FUSE_OUT_INIT;
struct fuse_setattr_in inarg;
- struct fuse_setattr_out outarg;
+ struct fuse_attr_out outarg;
memset(&inarg, 0, sizeof(inarg));
inarg.valid = iattr_to_fattr(attr, &inarg.attr);
{
if(!entry->d_inode)
return 0;
- else if(time_after(jiffies, entry->d_time + FUSE_REVALIDATE_TIME)) {
+ else if(entry->d_time && time_after(jiffies, entry->d_time)) {
struct inode *inode = entry->d_inode;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
int version;
int ret;
change_attributes(inode, &outarg.attr);
inode->i_version = version;
- entry->d_time = jiffies;
+ entry->d_time = time_to_jiffies(outarg.entry_valid,
+ outarg.entry_valid_nsec);
}
return 1;
}
#define FUSE_MAX_PATH 4096
#define PARAM(inarg) (((char *)(inarg)) + sizeof(*inarg))
+#define ENTRY_REVALIDATE_TIME 1 /* sec */
+#define ATTR_REVALIDATE_TIME 1 /* sec */
+
static const char *opname(enum fuse_opcode opcode)
{
switch(opcode) {
}
static int lookup_path(struct fuse *f, fino_t ino, int version, char *name,
- const char *path, struct fuse_lookup_out *arg)
+ const char *path, struct fuse_entry_out *arg)
{
int res;
struct stat buf;
if(res == 0) {
struct node *node;
- memset(arg, 0, sizeof(struct fuse_lookup_out));
+ memset(arg, 0, sizeof(struct fuse_entry_out));
convert_stat(&buf, &arg->attr);
node = find_node(f, ino, name, &arg->attr, version);
arg->ino = node->ino;
arg->generation = node->generation;
+ arg->entry_valid = ENTRY_REVALIDATE_TIME;
+ arg->entry_valid_nsec = 0;
+ arg->attr_valid = ATTR_REVALIDATE_TIME;
+ arg->attr_valid_nsec = 0;
if(f->flags & FUSE_DEBUG) {
printf(" INO: %li\n", arg->ino);
fflush(stdout);
{
int res;
char *path;
- struct fuse_lookup_out arg;
+ struct fuse_entry_out arg;
res = -ENOENT;
path = get_path_name(f, in->ino, name);
int res;
char *path;
struct stat buf;
- struct fuse_getattr_out arg;
+ struct fuse_attr_out arg;
res = -ENOENT;
path = get_path(f, in->ino);
}
if(res == 0) {
- memset(&arg, 0, sizeof(struct fuse_getattr_out));
+ memset(&arg, 0, sizeof(struct fuse_attr_out));
+ arg.attr_valid = ATTR_REVALIDATE_TIME;
+ arg.attr_valid_nsec = 0;
convert_stat(&buf, &arg.attr);
}
char *path;
int valid = arg->valid;
struct fuse_attr *attr = &arg->attr;
- struct fuse_setattr_out outarg;
+ struct fuse_attr_out outarg;
res = -ENOENT;
path = get_path(f, in->ino);
struct stat buf;
res = f->op.getattr(path, &buf);
if(!res) {
- memset(&outarg, 0, sizeof(struct fuse_setattr_out));
+ memset(&outarg, 0, sizeof(struct fuse_attr_out));
+ outarg.attr_valid = ATTR_REVALIDATE_TIME;
+ outarg.attr_valid_nsec = 0;
convert_stat(&buf, &outarg.attr);
}
}
int res;
char *path;
char *name = PARAM(inarg);
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
res = -ENOENT;
path = get_path_name(f, in->ino, name);
int res;
char *path;
char *name = PARAM(inarg);
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
res = -ENOENT;
path = get_path_name(f, in->ino, name);
{
int res;
char *path;
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
res = -ENOENT;
path = get_path_name(f, in->ino, name);
char *oldpath;
char *newpath;
char *name = PARAM(arg);
- struct fuse_lookup_out outarg;
+ struct fuse_entry_out outarg;
res = -ENOENT;
oldpath = get_path(f, in->ino);