summaryrefslogtreecommitdiffstats
path: root/sys/ufs
diff options
context:
space:
mode:
authorjdp <jdp@FreeBSD.org>1999-08-22 00:15:16 +0000
committerjdp <jdp@FreeBSD.org>1999-08-22 00:15:16 +0000
commit9f71d680aa0a36148f27d5ed32b29015c2497fd2 (patch)
tree2a7d73965f26e82ae8564395face352b791deb4e /sys/ufs
parent4039114e6fd13c7dc0d565c3af5f30bd26e3312b (diff)
downloadFreeBSD-src-9f71d680aa0a36148f27d5ed32b29015c2497fd2.zip
FreeBSD-src-9f71d680aa0a36148f27d5ed32b29015c2497fd2.tar.gz
Support full-precision file timestamps. Until now, only the seconds
have been maintained, and that is still the default. A new sysctl variable "vfs.timestamp_precision" can be used to enable higher levels of precision: 0 = seconds only; nanoseconds zeroed (default). 1 = seconds and nanoseconds, accurate within 1/HZ. 2 = seconds and nanoseconds, truncated to microseconds. >=3 = seconds and nanoseconds, maximum precision. Level 1 uses getnanotime(), which is fast but can be wrong by up to 1/HZ. Level 2 uses microtime(). It might be desirable for consistency with utimes() and friends, which take timeval structures rather than timespecs. Level 3 uses nanotime() for the higest precision. I benchmarked levels 0, 1, and 3 by copying a 550 MB tree with "cpio -pdu". There was almost negligible difference in the system times -- much less than 1%, and less than the variation among multiple runs at the same level. Bruce Evans dreamed up a torture test involving 1-byte reads with intervening fstat() calls, but the cpio test seems more realistic to me. This feature is currently implemented only for the UFS (FFS and MFS) filesystems. But I think it should be easy to support it in the others as well. An earlier version of this was reviewed by Bruce. He's not to blame for any breakage I've introduced since then. Reviewed by: bde (an earlier version of the code)
Diffstat (limited to 'sys/ufs')
-rw-r--r--sys/ufs/ufs/ufs_vnops.c29
1 files changed, 19 insertions, 10 deletions
diff --git a/sys/ufs/ufs/ufs_vnops.c b/sys/ufs/ufs/ufs_vnops.c
index e65f22b..f68e6da 100644
--- a/sys/ufs/ufs/ufs_vnops.c
+++ b/sys/ufs/ufs/ufs_vnops.c
@@ -36,7 +36,7 @@
* SUCH DAMAGE.
*
* @(#)ufs_vnops.c 8.27 (Berkeley) 5/27/95
- * $Id: ufs_vnops.c,v 1.118 1999/08/13 10:10:12 phk Exp $
+ * $Id: ufs_vnops.c,v 1.119 1999/08/13 10:56:02 phk Exp $
*/
#include "opt_quota.h"
@@ -138,26 +138,31 @@ ufs_itimes(vp)
struct vnode *vp;
{
struct inode *ip;
- time_t tv_sec;
+ struct timespec ts;
ip = VTOI(vp);
if ((ip->i_flag & (IN_ACCESS | IN_CHANGE | IN_UPDATE)) == 0)
return;
if ((vp->v_mount->mnt_flag & MNT_RDONLY) == 0) {
- tv_sec = time_second;
+ vfs_timestamp(&ts);
if ((vp->v_type == VBLK || vp->v_type == VCHR) &&
!DOINGSOFTDEP(vp))
ip->i_flag |= IN_LAZYMOD;
else
ip->i_flag |= IN_MODIFIED;
- if (ip->i_flag & IN_ACCESS)
- ip->i_atime = tv_sec;
+ if (ip->i_flag & IN_ACCESS) {
+ ip->i_atime = ts.tv_sec;
+ ip->i_atimensec = ts.tv_nsec;
+ }
if (ip->i_flag & IN_UPDATE) {
- ip->i_mtime = tv_sec;
+ ip->i_mtime = ts.tv_sec;
+ ip->i_mtimensec = ts.tv_nsec;
ip->i_modrev++;
}
- if (ip->i_flag & IN_CHANGE)
- ip->i_ctime = tv_sec;
+ if (ip->i_flag & IN_CHANGE) {
+ ip->i_ctime = ts.tv_sec;
+ ip->i_ctimensec = ts.tv_nsec;
+ }
}
ip->i_flag &= ~(IN_ACCESS | IN_CHANGE | IN_UPDATE);
}
@@ -500,10 +505,14 @@ ufs_setattr(ap)
if (vap->va_mtime.tv_sec != VNOVAL)
ip->i_flag |= IN_CHANGE | IN_UPDATE;
ufs_itimes(vp);
- if (vap->va_atime.tv_sec != VNOVAL)
+ if (vap->va_atime.tv_sec != VNOVAL) {
ip->i_atime = vap->va_atime.tv_sec;
- if (vap->va_mtime.tv_sec != VNOVAL)
+ ip->i_atimensec = vap->va_atime.tv_nsec;
+ }
+ if (vap->va_mtime.tv_sec != VNOVAL) {
ip->i_mtime = vap->va_mtime.tv_sec;
+ ip->i_mtimensec = vap->va_mtime.tv_nsec;
+ }
error = UFS_UPDATE(vp, 0);
if (error)
return (error);
OpenPOWER on IntegriCloud