#include <time.h>
#include <string.h>
#include <utime.h>
+#include <errno.h>
void usage(char *prog)
{
int main(int argc, char *argv[])
{
long before_mknod, after_mknod;
- long before_utime, after_utime;
+ const char *prog = argv[0];
+ const char *filename = argv[1];
+ struct utimbuf utb;
struct stat st;
int rc;
+ utb.actime = 0x47114711;
+ utb.modtime = 0x11471147;
+
+
if (argc != 2)
usage(argv[0]);
- before_mknod = time(0);
- rc = mknod(argv[1], 0700, S_IFREG);
+ /* Adjust the before time back one second, because the kernel's
+ * CURRENT_TIME (lockless clock reading, used to set inode times)
+ * may drift against the do_gettimeofday() time (TSC-corrected and
+ * locked clock reading, used to return timestamps to user space).
+ * This means that the mknod time could be a second older than the
+ * before time, even for a local filesystem such as ext3.
+ */
+ before_mknod = time(0) - 1;
+ rc = mknod(filename, 0700, S_IFREG);
after_mknod = time(0);
- if (rc) {
+ if (rc && errno != EEXIST) {
fprintf(stderr, "%s: mknod(%s) failed: rc %d: %s\n",
- argv[0], argv[1], rc, strerror(rc));
+ prog, filename, errno, strerror(errno));
return 2;
- }
+ } else if (!rc) {
+ rc = stat(filename, &st);
+ if (rc) {
+ fprintf(stderr, "%s: stat(%s) failed: rc %d: %s\n",
+ prog, filename, errno, strerror(errno));
+ return 3;
+ }
- rc = stat(argv[1], &st);
- if (rc) {
- fprintf(stderr, "%s: stat(%s) failed: rc %d: %s\n",
- argv[0], argv[1], rc, strerror(rc));
- return 3;
- }
+ if (st.st_mtime < before_mknod || st.st_mtime > after_mknod) {
+ fprintf(stderr,
+ "%s: bad mknod times %lu <= %lu <= %lu false\n",
+ prog, before_mknod, st.st_mtime, after_mknod);
+ return 4;
+ }
- if (st.st_mtime < before_mknod || st.st_mtime > after_mknod) {
- fprintf(stderr, "%s: bad mknod times %lu <= %lu <= %lu false\n",
- argv[0], before_mknod, st.st_mtime, after_mknod);
- return 4;
- }
+ printf("%s: good mknod times %lu%s <= %lu <= %lu\n",
+ prog, before_mknod, before_mknod == st.st_mtime ? "*":"",
+ st.st_mtime, after_mknod);
- printf("%s: good mknod times %lu <= %lu <= %lu\n",
- argv[0], before_mknod, st.st_mtime, after_mknod);
-
- sleep(5);
+ }
- before_utime = time(0);
- rc = utime(argv[0], NULL);
- after_utime = time(0);
+ /* See above */
+ rc = utime(filename, &utb);
if (rc) {
- fprintf(stderr, "%s: stat(%s) failed: rc %d: %s\n",
- argv[0], argv[1], rc, strerror(rc));
+ fprintf(stderr, "%s: utime(%s) failed: rc %d: %s\n",
+ prog, filename, errno, strerror(errno));
return 5;
}
- rc = stat(argv[1], &st);
+ rc = stat(filename, &st);
if (rc) {
fprintf(stderr, "%s: second stat(%s) failed: rc %d: %s\n",
- argv[0], argv[1], rc, strerror(rc));
+ prog, filename, errno, strerror(errno));
return 6;
}
- if (st.st_mtime < before_utime || st.st_mtime > after_utime) {
- fprintf(stderr, "%s: bad utime times %lu <= %lu <= %lu false\n",
- argv[0], before_utime, st.st_mtime, after_utime);
+ if (st.st_mtime != utb.modtime ) {
+ fprintf(stderr, "%s: bad utime mtime %lu should be %lu\n",
+ prog, st.st_mtime, utb.modtime);
+ return 7;
+ }
+
+ if (st.st_atime != utb.actime ) {
+ fprintf(stderr, "%s: bad utime mtime %lu should be %lu\n",
+ prog, st.st_atime, utb.actime);
return 7;
}
- printf("%s: good utime times %lu <= %lu <= %lu\n",
- argv[0], before_mknod, st.st_mtime, after_mknod);
+ printf("%s: good utime mtimes %lu, atime %lu\n",
+ prog, utb.modtime, utb.actime);
return 0;
}