diff options
author | Deepa Dinamani <deepa.kernel@gmail.com> | 2018-01-21 21:04:25 -0500 |
---|---|---|
committer | Deepa Dinamani <deepa.kernel@gmail.com> | 2019-08-30 10:27:17 -0400 |
commit | 50e17c000c467fbc927fc001df99beb4027a5323 (patch) | |
tree | fd820b8e8992477f3fa474f152358ed8c09247fd | |
parent | 188d20bcd1ebd8277d9b8a79525bd66b66d40a2a (diff) |
vfs: Add timestamp_truncate() api
timespec_trunc() function is used to truncate a
filesystem timestamp to the right granularity.
But, the function does not clamp tv_sec part of the
timestamps according to the filesystem timestamp limits.
The replacement api: timestamp_truncate() also alters the
signature of the function to accommodate filesystem
timestamp clamping according to flesystem limits.
Note that the tv_nsec part is set to 0 if tv_sec is not within
the range supported for the filesystem.
Signed-off-by: Deepa Dinamani <deepa.kernel@gmail.com>
Acked-by: Jeff Layton <jlayton@kernel.org>
-rw-r--r-- | fs/inode.c | 33 | ||||
-rw-r--r-- | include/linux/fs.h | 2 |
2 files changed, 34 insertions, 1 deletions
diff --git a/fs/inode.c b/fs/inode.c index 0f1e3b563c47..64bf28cf05cd 100644 --- a/fs/inode.c +++ b/fs/inode.c | |||
@@ -2167,6 +2167,37 @@ struct timespec64 timespec64_trunc(struct timespec64 t, unsigned gran) | |||
2167 | EXPORT_SYMBOL(timespec64_trunc); | 2167 | EXPORT_SYMBOL(timespec64_trunc); |
2168 | 2168 | ||
2169 | /** | 2169 | /** |
2170 | * timestamp_truncate - Truncate timespec to a granularity | ||
2171 | * @t: Timespec | ||
2172 | * @inode: inode being updated | ||
2173 | * | ||
2174 | * Truncate a timespec to the granularity supported by the fs | ||
2175 | * containing the inode. Always rounds down. gran must | ||
2176 | * not be 0 nor greater than a second (NSEC_PER_SEC, or 10^9 ns). | ||
2177 | */ | ||
2178 | struct timespec64 timestamp_truncate(struct timespec64 t, struct inode *inode) | ||
2179 | { | ||
2180 | struct super_block *sb = inode->i_sb; | ||
2181 | unsigned int gran = sb->s_time_gran; | ||
2182 | |||
2183 | t.tv_sec = clamp(t.tv_sec, sb->s_time_min, sb->s_time_max); | ||
2184 | if (unlikely(t.tv_sec == sb->s_time_max || t.tv_sec == sb->s_time_min)) | ||
2185 | t.tv_nsec = 0; | ||
2186 | |||
2187 | /* Avoid division in the common cases 1 ns and 1 s. */ | ||
2188 | if (gran == 1) | ||
2189 | ; /* nothing */ | ||
2190 | else if (gran == NSEC_PER_SEC) | ||
2191 | t.tv_nsec = 0; | ||
2192 | else if (gran > 1 && gran < NSEC_PER_SEC) | ||
2193 | t.tv_nsec -= t.tv_nsec % gran; | ||
2194 | else | ||
2195 | WARN(1, "invalid file time granularity: %u", gran); | ||
2196 | return t; | ||
2197 | } | ||
2198 | EXPORT_SYMBOL(timestamp_truncate); | ||
2199 | |||
2200 | /** | ||
2170 | * current_time - Return FS time | 2201 | * current_time - Return FS time |
2171 | * @inode: inode. | 2202 | * @inode: inode. |
2172 | * | 2203 | * |
@@ -2187,7 +2218,7 @@ struct timespec64 current_time(struct inode *inode) | |||
2187 | return now; | 2218 | return now; |
2188 | } | 2219 | } |
2189 | 2220 | ||
2190 | return timespec64_trunc(now, inode->i_sb->s_time_gran); | 2221 | return timestamp_truncate(now, inode); |
2191 | } | 2222 | } |
2192 | EXPORT_SYMBOL(current_time); | 2223 | EXPORT_SYMBOL(current_time); |
2193 | 2224 | ||
diff --git a/include/linux/fs.h b/include/linux/fs.h index 4b349851b00c..7e6be3bf0ce0 100644 --- a/include/linux/fs.h +++ b/include/linux/fs.h | |||
@@ -726,6 +726,8 @@ struct inode { | |||
726 | void *i_private; /* fs or device private pointer */ | 726 | void *i_private; /* fs or device private pointer */ |
727 | } __randomize_layout; | 727 | } __randomize_layout; |
728 | 728 | ||
729 | struct timespec64 timestamp_truncate(struct timespec64 t, struct inode *inode); | ||
730 | |||
729 | static inline unsigned int i_blocksize(const struct inode *node) | 731 | static inline unsigned int i_blocksize(const struct inode *node) |
730 | { | 732 | { |
731 | return (1 << node->i_blkbits); | 733 | return (1 << node->i_blkbits); |