diff options
-rw-r--r-- | fs/fuse/dir.c | 28 | ||||
-rw-r--r-- | include/linux/fuse.h | 2 |
2 files changed, 27 insertions, 3 deletions
diff --git a/fs/fuse/dir.c b/fs/fuse/dir.c index 1e941b3f2453..537d38bee13c 100644 --- a/fs/fuse/dir.c +++ b/fs/fuse/dir.c | |||
@@ -1014,6 +1014,20 @@ static int fuse_dir_fsync(struct file *file, struct dentry *de, int datasync) | |||
1014 | return file ? fuse_fsync_common(file, de, datasync, 1) : 0; | 1014 | return file ? fuse_fsync_common(file, de, datasync, 1) : 0; |
1015 | } | 1015 | } |
1016 | 1016 | ||
1017 | static bool update_mtime(unsigned ivalid) | ||
1018 | { | ||
1019 | /* Always update if mtime is explicitly set */ | ||
1020 | if (ivalid & ATTR_MTIME_SET) | ||
1021 | return true; | ||
1022 | |||
1023 | /* If it's an open(O_TRUNC) or an ftruncate(), don't update */ | ||
1024 | if ((ivalid & ATTR_SIZE) && (ivalid & (ATTR_OPEN | ATTR_FILE))) | ||
1025 | return false; | ||
1026 | |||
1027 | /* In all other cases update */ | ||
1028 | return true; | ||
1029 | } | ||
1030 | |||
1017 | static void iattr_to_fattr(struct iattr *iattr, struct fuse_setattr_in *arg) | 1031 | static void iattr_to_fattr(struct iattr *iattr, struct fuse_setattr_in *arg) |
1018 | { | 1032 | { |
1019 | unsigned ivalid = iattr->ia_valid; | 1033 | unsigned ivalid = iattr->ia_valid; |
@@ -1026,11 +1040,19 @@ static void iattr_to_fattr(struct iattr *iattr, struct fuse_setattr_in *arg) | |||
1026 | arg->valid |= FATTR_GID, arg->gid = iattr->ia_gid; | 1040 | arg->valid |= FATTR_GID, arg->gid = iattr->ia_gid; |
1027 | if (ivalid & ATTR_SIZE) | 1041 | if (ivalid & ATTR_SIZE) |
1028 | arg->valid |= FATTR_SIZE, arg->size = iattr->ia_size; | 1042 | arg->valid |= FATTR_SIZE, arg->size = iattr->ia_size; |
1029 | /* You can only _set_ these together (they may change by themselves) */ | 1043 | if (ivalid & ATTR_ATIME) { |
1030 | if ((ivalid & (ATTR_ATIME | ATTR_MTIME)) == (ATTR_ATIME | ATTR_MTIME)) { | 1044 | arg->valid |= FATTR_ATIME; |
1031 | arg->valid |= FATTR_ATIME | FATTR_MTIME; | ||
1032 | arg->atime = iattr->ia_atime.tv_sec; | 1045 | arg->atime = iattr->ia_atime.tv_sec; |
1046 | arg->atimensec = iattr->ia_atime.tv_nsec; | ||
1047 | if (!(ivalid & ATTR_ATIME_SET)) | ||
1048 | arg->valid |= FATTR_ATIME_NOW; | ||
1049 | } | ||
1050 | if ((ivalid & ATTR_MTIME) && update_mtime(ivalid)) { | ||
1051 | arg->valid |= FATTR_MTIME; | ||
1033 | arg->mtime = iattr->ia_mtime.tv_sec; | 1052 | arg->mtime = iattr->ia_mtime.tv_sec; |
1053 | arg->mtimensec = iattr->ia_mtime.tv_nsec; | ||
1054 | if (!(ivalid & ATTR_MTIME_SET)) | ||
1055 | arg->valid |= FATTR_MTIME_NOW; | ||
1034 | } | 1056 | } |
1035 | } | 1057 | } |
1036 | 1058 | ||
diff --git a/include/linux/fuse.h b/include/linux/fuse.h index a50d0d9ac7ae..43a77d73349c 100644 --- a/include/linux/fuse.h +++ b/include/linux/fuse.h | |||
@@ -83,6 +83,8 @@ struct fuse_file_lock { | |||
83 | #define FATTR_ATIME (1 << 4) | 83 | #define FATTR_ATIME (1 << 4) |
84 | #define FATTR_MTIME (1 << 5) | 84 | #define FATTR_MTIME (1 << 5) |
85 | #define FATTR_FH (1 << 6) | 85 | #define FATTR_FH (1 << 6) |
86 | #define FATTR_ATIME_NOW (1 << 7) | ||
87 | #define FATTR_MTIME_NOW (1 << 8) | ||
86 | 88 | ||
87 | /** | 89 | /** |
88 | * Flags returned by the OPEN request | 90 | * Flags returned by the OPEN request |