1caf54c4fSMartin Matuska /*-
2bd66c1b4SMartin Matuska * SPDX-License-Identifier: BSD-2-Clause
3bd66c1b4SMartin Matuska *
4caf54c4fSMartin Matuska * Copyright (c) 2003-2008 Tim Kientzle
5caf54c4fSMartin Matuska * All rights reserved.
6caf54c4fSMartin Matuska */
7caf54c4fSMartin Matuska #include "test.h"
8caf54c4fSMartin Matuska #if defined(HAVE_UTIME_H)
9caf54c4fSMartin Matuska #include <utime.h>
10caf54c4fSMartin Matuska #elif defined(HAVE_SYS_UTIME_H)
11caf54c4fSMartin Matuska #include <sys/utime.h>
12caf54c4fSMartin Matuska #endif
13caf54c4fSMartin Matuska
14caf54c4fSMartin Matuska static struct {
15caf54c4fSMartin Matuska const char *name;
16caf54c4fSMartin Matuska time_t atime_sec;
17caf54c4fSMartin Matuska } files[] = {
18caf54c4fSMartin Matuska { "f0", 0 },
19caf54c4fSMartin Matuska { "f1", 0 },
20caf54c4fSMartin Matuska { "f2", 0 },
21caf54c4fSMartin Matuska { "f3", 0 },
22caf54c4fSMartin Matuska { "f4", 0 },
23caf54c4fSMartin Matuska { "f5", 0 }
24caf54c4fSMartin Matuska };
25caf54c4fSMartin Matuska
26caf54c4fSMartin Matuska /*
27caf54c4fSMartin Matuska * Create a bunch of test files and record their atimes.
28caf54c4fSMartin Matuska * For the atime preserve/change tests, the files must have
29caf54c4fSMartin Matuska * atimes in the past. We can accomplish this by explicitly invoking
30caf54c4fSMartin Matuska * utime() on platforms that support it or by simply sleeping
31caf54c4fSMartin Matuska * for a second after creating the files. (Creating all of the files
32caf54c4fSMartin Matuska * at once means we only need to sleep once.)
33caf54c4fSMartin Matuska */
34caf54c4fSMartin Matuska static void
test_create(void)35caf54c4fSMartin Matuska test_create(void)
36caf54c4fSMartin Matuska {
37caf54c4fSMartin Matuska struct stat st;
38caf54c4fSMartin Matuska struct utimbuf times;
39caf54c4fSMartin Matuska static const int numfiles = sizeof(files) / sizeof(files[0]);
40caf54c4fSMartin Matuska int i;
41caf54c4fSMartin Matuska
42caf54c4fSMartin Matuska for (i = 0; i < numfiles; ++i) {
43caf54c4fSMartin Matuska /*
44caf54c4fSMartin Matuska * Note: Have to write at least one byte to the file.
45caf54c4fSMartin Matuska * cpio doesn't bother reading the file if it's zero length,
46caf54c4fSMartin Matuska * so the atime never gets changed in that case, which
47caf54c4fSMartin Matuska * makes the tests below rather pointless.
48caf54c4fSMartin Matuska */
49caf54c4fSMartin Matuska assertMakeFile(files[i].name, 0644, "a");
50caf54c4fSMartin Matuska
51caf54c4fSMartin Matuska /* If utime() isn't supported on your platform, just
52caf54c4fSMartin Matuska * #ifdef this section out. Most of the test below is
53caf54c4fSMartin Matuska * still valid. */
54caf54c4fSMartin Matuska memset(×, 0, sizeof(times));
55*2e113ef8SMartin Matuska #if defined(_WIN32) && !defined(__CYGWIN__)
5652c2bb75SMartin Matuska times.actime = 86400;
5752c2bb75SMartin Matuska times.modtime = 86400;
5852c2bb75SMartin Matuska #else
59caf54c4fSMartin Matuska times.actime = 1;
60caf54c4fSMartin Matuska times.modtime = 3;
6152c2bb75SMartin Matuska #endif
62caf54c4fSMartin Matuska assertEqualInt(0, utime(files[i].name, ×));
63caf54c4fSMartin Matuska
64caf54c4fSMartin Matuska /* Record whatever atime the file ended up with. */
65caf54c4fSMartin Matuska /* If utime() is available, this should be 1, but there's
66caf54c4fSMartin Matuska * no harm in being careful. */
67caf54c4fSMartin Matuska assertEqualInt(0, stat(files[i].name, &st));
68caf54c4fSMartin Matuska files[i].atime_sec = st.st_atime;
69caf54c4fSMartin Matuska }
70caf54c4fSMartin Matuska
71caf54c4fSMartin Matuska /* Wait until the atime on the last file is actually in the past. */
72caf54c4fSMartin Matuska sleepUntilAfter(files[numfiles - 1].atime_sec);
73caf54c4fSMartin Matuska }
74caf54c4fSMartin Matuska
DEFINE_TEST(test_option_a)75caf54c4fSMartin Matuska DEFINE_TEST(test_option_a)
76caf54c4fSMartin Matuska {
77caf54c4fSMartin Matuska struct stat st;
78caf54c4fSMartin Matuska int r;
79caf54c4fSMartin Matuska char *p;
80caf54c4fSMartin Matuska
81caf54c4fSMartin Matuska /* Create all of the test files. */
82caf54c4fSMartin Matuska test_create();
83caf54c4fSMartin Matuska
84caf54c4fSMartin Matuska /* Sanity check; verify that atimes really do get modified. */
85a8fc61d5SMartin Matuska p = slurpfile(NULL, "f0");
86a8fc61d5SMartin Matuska assert(p != NULL);
87caf54c4fSMartin Matuska free(p);
88caf54c4fSMartin Matuska assertEqualInt(0, stat("f0", &st));
89caf54c4fSMartin Matuska if (st.st_atime == files[0].atime_sec) {
90caf54c4fSMartin Matuska skipping("Cannot verify -a option\n"
91caf54c4fSMartin Matuska " Your system appears to not support atime.");
92caf54c4fSMartin Matuska }
93caf54c4fSMartin Matuska else
94caf54c4fSMartin Matuska {
95caf54c4fSMartin Matuska /*
96caf54c4fSMartin Matuska * If this disk is mounted noatime, then we can't
97caf54c4fSMartin Matuska * verify correct operation without -a.
98caf54c4fSMartin Matuska */
99caf54c4fSMartin Matuska
100caf54c4fSMartin Matuska /* Copy the file without -a; should change the atime. */
101caf54c4fSMartin Matuska r = systemf("echo %s | %s -pd copy-no-a > copy-no-a.out 2>copy-no-a.err", files[1].name, testprog);
102caf54c4fSMartin Matuska assertEqualInt(r, 0);
103caf54c4fSMartin Matuska assertTextFileContents("1 block\n", "copy-no-a.err");
104caf54c4fSMartin Matuska assertEmptyFile("copy-no-a.out");
105caf54c4fSMartin Matuska assertEqualInt(0, stat(files[1].name, &st));
106caf54c4fSMartin Matuska failure("Copying file without -a should have changed atime.");
107caf54c4fSMartin Matuska assert(st.st_atime != files[1].atime_sec);
108caf54c4fSMartin Matuska
109caf54c4fSMartin Matuska /* Archive the file without -a; should change the atime. */
110caf54c4fSMartin Matuska r = systemf("echo %s | %s -o > archive-no-a.out 2>archive-no-a.err", files[2].name, testprog);
111caf54c4fSMartin Matuska assertEqualInt(r, 0);
112caf54c4fSMartin Matuska assertTextFileContents("1 block\n", "copy-no-a.err");
113caf54c4fSMartin Matuska assertEqualInt(0, stat(files[2].name, &st));
114caf54c4fSMartin Matuska failure("Archiving file without -a should have changed atime.");
115caf54c4fSMartin Matuska assert(st.st_atime != files[2].atime_sec);
116caf54c4fSMartin Matuska }
117caf54c4fSMartin Matuska
118caf54c4fSMartin Matuska /*
119caf54c4fSMartin Matuska * We can, of course, still verify that the atime is unchanged
120caf54c4fSMartin Matuska * when using the -a option.
121caf54c4fSMartin Matuska */
122caf54c4fSMartin Matuska
123caf54c4fSMartin Matuska /* Copy the file with -a; should not change the atime. */
124caf54c4fSMartin Matuska r = systemf("echo %s | %s -pad copy-a > copy-a.out 2>copy-a.err",
125caf54c4fSMartin Matuska files[3].name, testprog);
126caf54c4fSMartin Matuska assertEqualInt(r, 0);
127caf54c4fSMartin Matuska assertTextFileContents("1 block\n", "copy-a.err");
128caf54c4fSMartin Matuska assertEmptyFile("copy-a.out");
129caf54c4fSMartin Matuska assertEqualInt(0, stat(files[3].name, &st));
130caf54c4fSMartin Matuska failure("Copying file with -a should not have changed atime.");
131caf54c4fSMartin Matuska assertEqualInt(st.st_atime, files[3].atime_sec);
132caf54c4fSMartin Matuska
133caf54c4fSMartin Matuska /* Archive the file with -a; should not change the atime. */
134caf54c4fSMartin Matuska r = systemf("echo %s | %s -oa > archive-a.out 2>archive-a.err",
135caf54c4fSMartin Matuska files[4].name, testprog);
136caf54c4fSMartin Matuska assertEqualInt(r, 0);
137caf54c4fSMartin Matuska assertTextFileContents("1 block\n", "copy-a.err");
138caf54c4fSMartin Matuska assertEqualInt(0, stat(files[4].name, &st));
139caf54c4fSMartin Matuska failure("Archiving file with -a should not have changed atime.");
140caf54c4fSMartin Matuska assertEqualInt(st.st_atime, files[4].atime_sec);
141caf54c4fSMartin Matuska }
142