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 |
