1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
/* vi: set sw=4 ts=4: */
/* testcase for futimens(2)
* Copyright (C) 2009 Bernhard Reutner-Fischer <uClibc@uClibc.org>
* Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
*/
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <stdlib.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
struct
{
char *name; /* name of file to open */
int flags; /* flags for file descriptor */
const struct timespec ts[2];
int err; /* expected errno */
} tests [] =
{
{"futimens.tst", (O_CREAT|O_TRUNC), {{0,0},{0,0}}, 0},
{"futimens.tst", (O_CREAT|O_TRUNC), {{99,0},{0,0}}, 0},
{"futimens.tst", (O_CREAT|O_TRUNC), {{0,99},{0,0}}, 0},
{"futimens.tst", (O_CREAT|O_TRUNC), {{0,0},{99,0}}, 0},
{"futimens.tst", (O_CREAT|O_TRUNC), {{0,0},{0,99}}, 0},
{"futimens.tst", (O_CREAT|O_TRUNC), {{11,2},{3,4}}, 0},
};
int do_test(int argc, char **argv) {
char *name;
int i, errors;
errors = argc - argc + 0;
unsigned has_stat_nsec = 0;
{
struct stat probe;
/* Let's attempt an educated guess if this filesystem supports
* nanosecond mtime. */
if ((!stat(".", &probe)) && probe.st_mtim.tv_nsec)
has_stat_nsec = 1;
else if ((!stat(argv[0], &probe)) && probe.st_mtim.tv_nsec)
has_stat_nsec = 1;
}
for (i=0; i < (int) (sizeof(tests)/sizeof(tests[0])); ++i) {
int err, fd;
struct stat sb;
name = tests[i].name;
if (*name != '.')
unlink(name);
fd = open(name, tests[i].flags, 0660);
if (fd < 0)
abort();
errno = 0;
err = futimens(fd, tests[i].ts);
if ((errno && !err) || (!errno && err)) {
err = errno;
printf("FAILED test %d (errno and return value disagree)\n", i);
++errors;
} else
err = errno;
if (err != tests[i].err) {
printf("FAILED test %d (expected errno %d, got %d)\n",
i, tests[i].err, err);
++errors;
continue;
}
if (stat(name, &sb) < 0) {
printf("FAILED test %d (verification)\n", i);
++errors;
continue;
} else {
unsigned wrong = tests[i].ts[0].tv_sec != sb.st_atim.tv_sec ||
tests[i].ts[0].tv_nsec != sb.st_atim.tv_nsec ||
tests[i].ts[1].tv_sec != sb.st_mtim.tv_sec ||
tests[i].ts[1].tv_nsec != sb.st_mtim.tv_nsec;
if (wrong) {
if (tests[i].ts[0].tv_sec != sb.st_atim.tv_sec) {
printf("FAILED test %d (access time, sec: expected %ld, got %ld)\n",
i, tests[i].ts[0].tv_sec, sb.st_atim.tv_sec);
++errors;
}
if (tests[i].ts[0].tv_nsec != sb.st_atim.tv_nsec) {
printf("FAILED test %d (access time, nsec: expected %ld, got %ld)\n",
i, tests[i].ts[0].tv_nsec, sb.st_atim.tv_nsec);
errors += has_stat_nsec;
}
if (tests[i].ts[1].tv_sec != sb.st_mtim.tv_sec) {
printf("FAILED test %d (modification time, sec: expected %ld, got %ld)\n",
i, tests[i].ts[1].tv_sec, sb.st_mtim.tv_sec);
++errors;
}
if (tests[i].ts[1].tv_nsec != sb.st_mtim.tv_nsec) {
printf("FAILED test %d (modification time, nsec: expected %ld, got %ld)\n",
i, tests[i].ts[1].tv_nsec, sb.st_mtim.tv_nsec);
errors += has_stat_nsec;
}
}
}
}
if (*name != '.')
unlink(name);
printf("%d errors.\n", errors);
return (!errors) ? EXIT_SUCCESS : EXIT_FAILURE;
}
#include <test-skeleton.c>
|