Skip to content

Commit 773b818

Browse files
committed
Documentation: add a new file documenting multigrain timestamps
JIRA: https://issues.redhat.com/browse/RHEL-121527 Add a high-level document that describes how multigrain timestamps work, rationale for them, and some info about implementation and tradeoffs. Reviewed-by: Josef Bacik <josef@toxicpanda.com> Reviewed-by: Darrick J. Wong <djwong@kernel.org> Reviewed-by: Randy Dunlap <rdunlap@infradead.org> Reviewed-by: Jan Kara <jack@suse.cz> Tested-by: Randy Dunlap <rdunlap@infradead.org> # documentation bits Signed-off-by: Jeff Layton <jlayton@kernel.org> Link: https://lore.kernel.org/r/20241002-mgtime-v10-8-d1c4717f5284@kernel.org Signed-off-by: Christian Brauner <brauner@kernel.org> (cherry picked from commit e3fad03) Signed-off-by: Carlos Maiolino <cmaiolino@redhat.com>
1 parent de7f7d8 commit 773b818

File tree

2 files changed

+126
-0
lines changed

2 files changed

+126
-0
lines changed

Documentation/filesystems/index.rst

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ algorithms work.
2929
fiemap
3030
files
3131
locks
32+
multigrain-ts
3233
mount_api
3334
quota
3435
seq_file
Lines changed: 125 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
.. SPDX-License-Identifier: GPL-2.0
2+
3+
=====================
4+
Multigrain Timestamps
5+
=====================
6+
7+
Introduction
8+
============
9+
Historically, the kernel has always used coarse time values to stamp inodes.
10+
This value is updated every jiffy, so any change that happens within that jiffy
11+
will end up with the same timestamp.
12+
13+
When the kernel goes to stamp an inode (due to a read or write), it first gets
14+
the current time and then compares it to the existing timestamp(s) to see
15+
whether anything will change. If nothing changed, then it can avoid updating
16+
the inode's metadata.
17+
18+
Coarse timestamps are therefore good from a performance standpoint, since they
19+
reduce the need for metadata updates, but bad from the standpoint of
20+
determining whether anything has changed, since a lot of things can happen in a
21+
jiffy.
22+
23+
They are particularly troublesome with NFSv3, where unchanging timestamps can
24+
make it difficult to tell whether to invalidate caches. NFSv4 provides a
25+
dedicated change attribute that should always show a visible change, but not
26+
all filesystems implement this properly, causing the NFS server to substitute
27+
the ctime in many cases.
28+
29+
Multigrain timestamps aim to remedy this by selectively using fine-grained
30+
timestamps when a file has had its timestamps queried recently, and the current
31+
coarse-grained time does not cause a change.
32+
33+
Inode Timestamps
34+
================
35+
There are currently 3 timestamps in the inode that are updated to the current
36+
wallclock time on different activity:
37+
38+
ctime:
39+
The inode change time. This is stamped with the current time whenever
40+
the inode's metadata is changed. Note that this value is not settable
41+
from userland.
42+
43+
mtime:
44+
The inode modification time. This is stamped with the current time
45+
any time a file's contents change.
46+
47+
atime:
48+
The inode access time. This is stamped whenever an inode's contents are
49+
read. Widely considered to be a terrible mistake. Usually avoided with
50+
options like noatime or relatime.
51+
52+
Updating the mtime always implies a change to the ctime, but updating the
53+
atime due to a read request does not.
54+
55+
Multigrain timestamps are only tracked for the ctime and the mtime. atimes are
56+
not affected and always use the coarse-grained value (subject to the floor).
57+
58+
Inode Timestamp Ordering
59+
========================
60+
61+
In addition to just providing info about changes to individual files, file
62+
timestamps also serve an important purpose in applications like "make". These
63+
programs measure timestamps in order to determine whether source files might be
64+
newer than cached objects.
65+
66+
Userland applications like make can only determine ordering based on
67+
operational boundaries. For a syscall those are the syscall entry and exit
68+
points. For io_uring or nfsd operations, that's the request submission and
69+
response. In the case of concurrent operations, userland can make no
70+
determination about the order in which things will occur.
71+
72+
For instance, if a single thread modifies one file, and then another file in
73+
sequence, the second file must show an equal or later mtime than the first. The
74+
same is true if two threads are issuing similar operations that do not overlap
75+
in time.
76+
77+
If however, two threads have racing syscalls that overlap in time, then there
78+
is no such guarantee, and the second file may appear to have been modified
79+
before, after or at the same time as the first, regardless of which one was
80+
submitted first.
81+
82+
Note that the above assumes that the system doesn't experience a backward jump
83+
of the realtime clock. If that occurs at an inopportune time, then timestamps
84+
can appear to go backward, even on a properly functioning system.
85+
86+
Multigrain Timestamp Implementation
87+
===================================
88+
Multigrain timestamps are aimed at ensuring that changes to a single file are
89+
always recognizable, without violating the ordering guarantees when multiple
90+
different files are modified. This affects the mtime and the ctime, but the
91+
atime will always use coarse-grained timestamps.
92+
93+
It uses an unused bit in the i_ctime_nsec field to indicate whether the mtime
94+
or ctime has been queried. If either or both have, then the kernel takes
95+
special care to ensure the next timestamp update will display a visible change.
96+
This ensures tight cache coherency for use-cases like NFS, without sacrificing
97+
the benefits of reduced metadata updates when files aren't being watched.
98+
99+
The Ctime Floor Value
100+
=====================
101+
It's not sufficient to simply use fine or coarse-grained timestamps based on
102+
whether the mtime or ctime has been queried. A file could get a fine grained
103+
timestamp, and then a second file modified later could get a coarse-grained one
104+
that appears earlier than the first, which would break the kernel's timestamp
105+
ordering guarantees.
106+
107+
To mitigate this problem, maintain a global floor value that ensures that
108+
this can't happen. The two files in the above example may appear to have been
109+
modified at the same time in such a case, but they will never show the reverse
110+
order. To avoid problems with realtime clock jumps, the floor is managed as a
111+
monotonic ktime_t, and the values are converted to realtime clock values as
112+
needed.
113+
114+
Implementation Notes
115+
====================
116+
Multigrain timestamps are intended for use by local filesystems that get
117+
ctime values from the local clock. This is in contrast to network filesystems
118+
and the like that just mirror timestamp values from a server.
119+
120+
For most filesystems, it's sufficient to just set the FS_MGTIME flag in the
121+
fstype->fs_flags in order to opt-in, providing the ctime is only ever set via
122+
inode_set_ctime_current(). If the filesystem has a ->getattr routine that
123+
doesn't call generic_fillattr, then it should call fill_mg_cmtime() to
124+
fill those values. For setattr, it should use setattr_copy() to update the
125+
timestamps, or otherwise mimic its behavior.

0 commit comments

Comments
 (0)