]> granicus.if.org Git - zfs/commitdiff
Fix truncate(2) mtime and ctime handling
authorLOLi <loli10K@users.noreply.github.com>
Mon, 13 Nov 2017 17:24:26 +0000 (18:24 +0100)
committerTony Hutter <hutter2@llnl.gov>
Tue, 21 Nov 2017 19:11:29 +0000 (13:11 -0600)
On Linux, ftruncate(2) always changes the file timestamps, even if the
file size is not changed. However, in case of a successfull
truncate(2), the timestamps are updated only if the file size changes.
This translates to the VFS calling the ZFS Posix Layer "setattr"
function (zpl_setattr) with ATTR_MTIME and ATTR_CTIME unconditionally
set on the iattr mask only when doing a ftruncate(2), while the
truncate(2) is left to the filesystem implementation to be dealt with.

This behaviour is consistent with POSIX:2004/SUSv3 specifications
where there's no explicit requirement for file size changes to update
the timestamps only for ftruncate(2):

http://pubs.opengroup.org/onlinepubs/009695399/functions/truncate.html
http://pubs.opengroup.org/onlinepubs/009695399/functions/ftruncate.html

This has been later updated in POSIX:2008/SUSv4 where, for both
truncate(2)/ftruncate(2), there's no mention of this size change
requirement:

http://austingroupbugs.net/view.php?id=489
http://pubs.opengroup.org/onlinepubs/9699919799/functions/truncate.html
http://pubs.opengroup.org/onlinepubs/9699919799/functions/ftruncate.html

Unfortunately the Linux VFS is still calling into the ZPL without
ATTR_MTIME/ATTR_CTIME set in the truncate(2) case: we fix this by
explicitly updating the timestamps when detecting the ATTR_SIZE bit,
which is always set in do_truncate(), on the iattr mask.

Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Signed-off-by: loli10K <ezomori.nozomu@gmail.com>
Closes #6811
Closes #6819

module/zfs/zfs_vnops.c
tests/runfiles/linux.run
tests/zfs-tests/include/math.shlib
tests/zfs-tests/tests/functional/truncate/.gitignore [new file with mode: 0644]
tests/zfs-tests/tests/functional/truncate/Makefile.am
tests/zfs-tests/tests/functional/truncate/truncate_test.c [new file with mode: 0644]
tests/zfs-tests/tests/functional/truncate/truncate_timestamps.ksh [new file with mode: 0755]

index a88f2482e722043e81cffe746b1786b1ecea8bd4..6a1dab5c984e9aef79c10a5745caf7ff1dd445fc 100644 (file)
@@ -3155,7 +3155,7 @@ top:
                    &atime, sizeof (atime));
        }
 
-       if (mask & ATTR_MTIME) {
+       if (mask & (ATTR_MTIME | ATTR_SIZE)) {
                ZFS_TIME_ENCODE(&vap->va_mtime, mtime);
                ZTOI(zp)->i_mtime = timespec_trunc(vap->va_mtime,
                    ZTOI(zp)->i_sb->s_time_gran);
@@ -3164,7 +3164,7 @@ top:
                    mtime, sizeof (mtime));
        }
 
-       if (mask & ATTR_CTIME) {
+       if (mask & (ATTR_CTIME | ATTR_SIZE)) {
                ZFS_TIME_ENCODE(&vap->va_ctime, ctime);
                ZTOI(zp)->i_ctime = timespec_trunc(vap->va_ctime,
                    ZTOI(zp)->i_sb->s_time_gran);
index 5583a2554168e3d55f4d1de70cb00f8668c29d3a..7d6d13b4b29c4b38e3212b3cdabec3d3f4a3daee 100644 (file)
@@ -537,7 +537,8 @@ tests = ['threadsappend_001_pos']
 tests = ['tmpfile_001_pos', 'tmpfile_002_pos', 'tmpfile_003_pos']
 
 [tests/functional/truncate]
-tests = ['truncate_001_pos', 'truncate_002_pos']
+tests = ['truncate_001_pos', 'truncate_002_pos', 'truncate_timestamps']
+tags = ['functional', 'truncate']
 
 [tests/functional/upgrade]
 tests = [ 'upgrade_userobj_001_pos' ]
index 4ed11fb4b72bf35729ae23f45639ec17d70a4ed3..66658cdda9d35f3c8d0bcc0401a5beb1df21cfdb 100644 (file)
@@ -66,3 +66,33 @@ function to_bytes
 
        return 0
 }
+
+#
+# Verify $a is equal to $b, otherwise raise an error specifying
+# the $type of values being compared
+#
+function verify_eq # <a> <b> <type>
+{
+       typeset a=$1
+       typeset b=$2
+       typeset type=$3
+
+       if [[ $a -ne $b ]]; then
+               log_fail "Compared $type should be equal: $a != $b"
+       fi
+}
+
+#
+# Verify $a is not equal to $b, otherwise raise an error specifying
+# the $type of values being compared
+#
+function verify_ne # <a> <b> <type>
+{
+       typeset a=$1
+       typeset b=$2
+       typeset type=$3
+
+       if [[ $a -eq $b ]]; then
+               log_fail "Compared $type should be not equal: $a == $b"
+       fi
+}
diff --git a/tests/zfs-tests/tests/functional/truncate/.gitignore b/tests/zfs-tests/tests/functional/truncate/.gitignore
new file mode 100644 (file)
index 0000000..f28d935
--- /dev/null
@@ -0,0 +1 @@
+/truncate_test
index 16cadf29d75dbafe076d1f94d4216a6b5817803c..0071e8f87070d00be7557299881606003bb2f6be 100644 (file)
@@ -1,7 +1,16 @@
+include $(top_srcdir)/config/Rules.am
+
 pkgdatadir = $(datadir)/@PACKAGE@/zfs-tests/tests/functional/truncate
+
 dist_pkgdata_SCRIPTS = \
        setup.ksh \
        cleanup.ksh \
        truncate.cfg \
        truncate_001_pos.ksh \
-       truncate_002_pos.ksh
+       truncate_002_pos.ksh \
+       truncate_timestamps.ksh
+
+pkgexecdir = $(datadir)/@PACKAGE@/zfs-tests/tests/functional/truncate
+
+pkgexec_PROGRAMS = truncate_test
+truncate_test_SOURCES = truncate_test.c
diff --git a/tests/zfs-tests/tests/functional/truncate/truncate_test.c b/tests/zfs-tests/tests/functional/truncate/truncate_test.c
new file mode 100644 (file)
index 0000000..3e277e8
--- /dev/null
@@ -0,0 +1,103 @@
+/*
+ * This file and its contents are supplied under the terms of the
+ * Common Development and Distribution License ("CDDL"), version 1.0.
+ * You may only use this file in accordance with the terms of version
+ * 1.0 of the CDDL.
+ *
+ * A full copy of the text of the CDDL should have accompanied this
+ * source.  A copy of the CDDL is also available via the Internet at
+ * http://www.illumos.org/license/CDDL.
+ */
+
+/*
+ * Copyright (c) 2012, 2014 by Delphix. All rights reserved.
+ * Copyright 2017, loli10K <ezomori.nozomu@gmail.com>. All rights reserved.
+ */
+
+#include <fcntl.h>
+#include <sys/stat.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <errno.h>
+#include <stdio.h>
+#include <stdlib.h>
+
+#define        FSIZE   256*1024*1024
+
+static long    fsize = FSIZE;
+static int     errflag = 0;
+static char    *filename = NULL;
+static int     ftruncflag = 0;
+
+static void parse_options(int argc, char *argv[]);
+
+static void
+usage(char *execname)
+{
+       (void) fprintf(stderr,
+           "usage: %s [-s filesize] [-f] /path/to/file\n", execname);
+       (void) exit(1);
+}
+
+int
+main(int argc, char *argv[])
+{
+       int fd;
+
+       parse_options(argc, argv);
+
+       if (ftruncflag) {
+               fd = open(filename, O_RDWR|O_CREAT, 0666);
+               if (fd < 0) {
+                       perror("open");
+                       return (1);
+               }
+               if (ftruncate(fd, fsize) < 0) {
+                       perror("ftruncate");
+                       return (1);
+               }
+               if (close(fd)) {
+                       perror("close");
+                       return (1);
+               }
+       } else {
+               if (truncate(filename, fsize) < 0) {
+                       perror("truncate");
+                       return (1);
+               }
+       }
+       return (0);
+}
+
+static void
+parse_options(int argc, char *argv[])
+{
+       int c;
+       extern char *optarg;
+       extern int optind, optopt;
+
+       while ((c = getopt(argc, argv, "s:f")) != -1) {
+               switch (c) {
+                       case 's':
+                               fsize = atoi(optarg);
+                               break;
+                       case 'f':
+                               ftruncflag++;
+                               break;
+                       case ':':
+                               (void) fprintf(stderr,
+                                   "Option -%c requires an operand\n", optopt);
+                               errflag++;
+                               break;
+               }
+               if (errflag) {
+                       (void) usage(argv[0]);
+               }
+       }
+
+       if (argc <= optind) {
+               (void) fprintf(stderr, "No filename specified\n");
+               usage(argv[0]);
+       }
+       filename = argv[optind];
+}
diff --git a/tests/zfs-tests/tests/functional/truncate/truncate_timestamps.ksh b/tests/zfs-tests/tests/functional/truncate/truncate_timestamps.ksh
new file mode 100755 (executable)
index 0000000..c365c74
--- /dev/null
@@ -0,0 +1,74 @@
+#!/bin/ksh -p
+#
+# This file and its contents are supplied under the terms of the
+# Common Development and Distribution License ("CDDL"), version 1.0.
+# You may only use this file in accordance with the terms of version
+# 1.0 of the CDDL.
+#
+# A full copy of the text of the CDDL should have accompanied this
+# source.  A copy of the CDDL is also available via the Internet at
+# http://www.illumos.org/license/CDDL.
+#
+
+#
+# Copyright 2017, loli10K <ezomori.nozomu@gmail.com>. All rights reserved.
+#
+
+. $STF_SUITE/tests/functional/truncate/truncate.cfg
+. $STF_SUITE/include/libtest.shlib
+. $STF_SUITE/include/math.shlib
+
+#
+# DESCRIPTION:
+# Ensure both truncate(2)/ftruncate(2) update target file mtime/ctime attributes
+#
+# STRATEGY:
+# 1. Create a file
+# 2. Truncate the file
+# 3. Verify both mtime/ctime are updated
+# 4. Rinse and repeat for both truncate(2) and ftruncate(2) with various sizes
+#
+
+verify_runnable "both"
+
+function verify_truncate # <filename> <filesize> <option>
+{
+       typeset filename="$1"
+       typeset -i size="$2"
+       typeset option="$3"
+
+       log_must mkfile $sizeavg $filename # always start with $sizeavg
+       typeset -i timestm="$(stat -c %Y $filename)"
+       typeset -i timestc="$(stat -c %Z $filename)"
+       log_must sleep 1
+       log_must $STF_SUITE/tests/functional/truncate/truncate_test -s $size $filename $option
+       verify_eq $size "$(stat -c %s $filename)" "size"
+       verify_ne $timestm "$(stat -c %Y $filename)" "mtime"
+       verify_ne $timestc "$(stat -c %Z $filename)" "ctime"
+       log_must rm -f $filename
+}
+
+function cleanup
+{
+       [[ -f $truncfile ]] && rm -f $truncfile
+}
+
+log_assert "Ensure both truncate(2)/ftruncate(2) update target file timestamps"
+log_onexit cleanup
+
+truncfile="$TESTDIR/truncate.$$"
+sizemin="123"
+sizeavg="$((256*1024))"
+sizemax="$((1024*1024))"
+
+# truncate(2)
+verify_truncate $truncfile $sizemin ""
+verify_truncate $truncfile $sizeavg ""
+verify_truncate $truncfile $sizemax ""
+
+# ftruncate(2)
+verify_truncate $truncfile $sizemin "-f"
+verify_truncate $truncfile $sizeavg "-f"
+verify_truncate $truncfile $sizemax "-f"
+
+log_pass "Successful truncation correctly update timestamps"