[test] Add test case for UTIME
This commit is contained in:
parent
4fab368127
commit
df36a6bbbb
@ -22,7 +22,7 @@ TESTS ?= env empty hello_world malloc mmap file fs_perms getpid spawn sched pipe
|
||||
truncate readdir mkdir open stat link symlink chmod chown tls pthread system_info rlimit \
|
||||
server server_epoll unix_socket cout hostfs cpuid rdtsc device sleep exit_group posix_flock \
|
||||
ioctl fcntl eventfd emulate_syscall access signal sysinfo prctl rename procfs wait \
|
||||
spawn_attribute exec statfs random umask pgrp vfork mount flock
|
||||
spawn_attribute exec statfs random umask pgrp vfork mount flock utimes
|
||||
# Benchmarks: need to be compiled and run by bench-% target
|
||||
BENCHES := spawn_and_exit_latency pipe_throughput unix_socket_throughput
|
||||
|
||||
|
5
test/utimes/Makefile
Normal file
5
test/utimes/Makefile
Normal file
@ -0,0 +1,5 @@
|
||||
include ../test_common.mk
|
||||
|
||||
EXTRA_C_FLAGS :=
|
||||
EXTRA_LINK_FLAGS :=
|
||||
BIN_ARGS :=
|
331
test/utimes/main.c
Normal file
331
test/utimes/main.c
Normal file
@ -0,0 +1,331 @@
|
||||
#define _GNU_SOURCE
|
||||
#include <sys/types.h>
|
||||
#include <sys/time.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/syscall.h>
|
||||
#include <fcntl.h>
|
||||
#include <time.h>
|
||||
#include <unistd.h>
|
||||
#include <utime.h>
|
||||
#include "test_fs.h"
|
||||
|
||||
// ============================================================================
|
||||
// Global variables
|
||||
// ============================================================================
|
||||
|
||||
static int SUCCESS = 1;
|
||||
static int FAIL = -1;
|
||||
const static struct timespec period_of_100ms = {
|
||||
.tv_sec = 0,
|
||||
.tv_nsec = 100 * (1000 * 1000)
|
||||
};
|
||||
|
||||
// ============================================================================
|
||||
// Helper function
|
||||
// ============================================================================
|
||||
|
||||
static int create_file(const char *file_path) {
|
||||
int fd;
|
||||
int flags = O_RDONLY | O_CREAT | O_TRUNC;
|
||||
int mode = 00444;
|
||||
|
||||
fd = open(file_path, flags, mode);
|
||||
if (fd < 0) {
|
||||
THROW_ERROR("failed to create a file");
|
||||
}
|
||||
close(fd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int remove_file(const char *file_path) {
|
||||
int ret;
|
||||
|
||||
ret = unlink(file_path);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to unlink the created file");
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// Test cases for utime
|
||||
// ============================================================================
|
||||
|
||||
|
||||
static int __test_utime(const char *file_path) {
|
||||
struct stat stat_buf;
|
||||
struct utimbuf times;
|
||||
struct timeval timeval;
|
||||
time_t actime, modtime;
|
||||
int ret;
|
||||
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
actime = stat_buf.st_atim.tv_sec + 1;
|
||||
modtime = stat_buf.st_mtim.tv_sec + 2;
|
||||
times.actime = actime;
|
||||
times.modtime = modtime;
|
||||
ret = syscall(SYS_utime, file_path, ×);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to utime file");
|
||||
}
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
if (stat_buf.st_atim.tv_sec != actime ||
|
||||
stat_buf.st_atim.tv_nsec != 0 ||
|
||||
stat_buf.st_mtim.tv_sec != modtime ||
|
||||
stat_buf.st_mtim.tv_nsec != 0) {
|
||||
THROW_ERROR("check utime result failed");
|
||||
}
|
||||
|
||||
// If times is NULL, then the access and modification times
|
||||
// of the file are set to the current time.
|
||||
ret = gettimeofday(&timeval, NULL);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to gettimeofday");
|
||||
}
|
||||
ret = syscall(SYS_utime, file_path, NULL);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to utime file");
|
||||
}
|
||||
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
|
||||
if (stat_buf.st_atim.tv_sec != timeval.tv_sec ||
|
||||
stat_buf.st_mtim.tv_sec != timeval.tv_sec) {
|
||||
THROW_ERROR("check utime result failed");
|
||||
}
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int __test_utimes(const char *file_path) {
|
||||
struct stat stat_buf;
|
||||
struct timeval actime, modtime, times[2];
|
||||
int ret;
|
||||
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&actime, NULL);
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&modtime, NULL);
|
||||
times[0] = actime;
|
||||
times[1] = modtime;
|
||||
ret = syscall(SYS_utimes, file_path, times);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to utime file");
|
||||
}
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
if (stat_buf.st_atim.tv_sec != actime.tv_sec ||
|
||||
stat_buf.st_atim.tv_nsec / 1000 != actime.tv_usec ||
|
||||
stat_buf.st_mtim.tv_sec != modtime.tv_sec ||
|
||||
stat_buf.st_mtim.tv_nsec / 1000 != modtime.tv_usec) {
|
||||
THROW_ERROR("check utimes result failed");
|
||||
}
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int __test_futimesat(const char *file_path) {
|
||||
struct stat stat_buf;
|
||||
struct timeval actime, modtime, times[2];
|
||||
char dir_buf[PATH_MAX] = { 0 };
|
||||
char base_buf[PATH_MAX] = { 0 };
|
||||
char *dir_name, *file_name;
|
||||
int dirfd, ret;
|
||||
|
||||
if (fs_split_path(file_path, dir_buf, &dir_name, base_buf, &file_name) < 0) {
|
||||
THROW_ERROR("failed to split path");
|
||||
}
|
||||
dirfd = open(dir_name, O_RDONLY);
|
||||
if (dirfd < 0) {
|
||||
THROW_ERROR("failed to split path");
|
||||
}
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&actime, NULL);
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&modtime, NULL);
|
||||
times[0] = actime;
|
||||
times[1] = modtime;
|
||||
ret = syscall(SYS_futimesat, dirfd, file_name, times);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to futimesat file with dirfd");
|
||||
}
|
||||
close(dirfd);
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
if (stat_buf.st_atim.tv_sec != actime.tv_sec ||
|
||||
stat_buf.st_atim.tv_nsec / 1000 != actime.tv_usec ||
|
||||
stat_buf.st_mtim.tv_sec != modtime.tv_sec ||
|
||||
stat_buf.st_mtim.tv_nsec / 1000 != modtime.tv_usec) {
|
||||
THROW_ERROR("check utimes result failed");
|
||||
}
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int __test_futimesat_nullpath(const char *file_path) {
|
||||
struct stat stat_buf;
|
||||
struct timeval actime, modtime, times[2];
|
||||
int dirfd, ret;
|
||||
|
||||
dirfd = open(file_path, O_RDONLY);
|
||||
if (dirfd < 0) {
|
||||
THROW_ERROR("failed to open dir");
|
||||
}
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&actime, NULL);
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
gettimeofday(&modtime, NULL);
|
||||
times[0] = actime;
|
||||
times[1] = modtime;
|
||||
ret = syscall(SYS_futimesat, dirfd, NULL, times);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to futimesat file with dirfd");
|
||||
}
|
||||
close(dirfd);
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
if (stat_buf.st_atim.tv_sec != actime.tv_sec ||
|
||||
stat_buf.st_atim.tv_nsec / 1000 != actime.tv_usec ||
|
||||
stat_buf.st_mtim.tv_sec != modtime.tv_sec ||
|
||||
stat_buf.st_mtim.tv_nsec / 1000 != modtime.tv_usec) {
|
||||
THROW_ERROR("check utimes result failed");
|
||||
}
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int __test_utimensat(const char *file_path) {
|
||||
struct stat stat_buf;
|
||||
struct timespec actime, modtime, times[2];
|
||||
char dir_buf[PATH_MAX] = { 0 };
|
||||
char base_buf[PATH_MAX] = { 0 };
|
||||
char *dir_name, *file_name;
|
||||
int dirfd, ret;
|
||||
|
||||
if (fs_split_path(file_path, dir_buf, &dir_name, base_buf, &file_name) < 0) {
|
||||
THROW_ERROR("failed to split path");
|
||||
}
|
||||
dirfd = open(dir_name, O_RDONLY);
|
||||
if (dirfd < 0) {
|
||||
THROW_ERROR("failed to open dir");
|
||||
}
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
clock_gettime(CLOCK_REALTIME, &actime);
|
||||
nanosleep(&period_of_100ms, NULL);
|
||||
clock_gettime(CLOCK_REALTIME, &modtime);
|
||||
times[0] = actime;
|
||||
times[1] = modtime;
|
||||
ret = syscall(SYS_utimensat, dirfd, file_name, times, 0);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to futimesat file with dirfd");
|
||||
}
|
||||
close(dirfd);
|
||||
ret = stat(file_path, &stat_buf);
|
||||
if (ret < 0) {
|
||||
THROW_ERROR("failed to stat file");
|
||||
}
|
||||
if (stat_buf.st_atim.tv_sec != actime.tv_sec ||
|
||||
stat_buf.st_atim.tv_nsec != actime.tv_nsec ||
|
||||
stat_buf.st_mtim.tv_sec != modtime.tv_sec ||
|
||||
stat_buf.st_mtim.tv_nsec != modtime.tv_nsec) {
|
||||
THROW_ERROR("check utimes result failed");
|
||||
}
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int __test_utimensat_invalid_flag(const char *file_path) {
|
||||
struct timespec times[2] = {{10, 0}, {20, 0}};
|
||||
char dir_buf[PATH_MAX] = { 0 };
|
||||
char base_buf[PATH_MAX] = { 0 };
|
||||
char *dir_name, *file_name;
|
||||
int dirfd, ret;
|
||||
|
||||
if (fs_split_path(file_path, dir_buf, &dir_name, base_buf, &file_name) < 0) {
|
||||
THROW_ERROR("failed to split path");
|
||||
}
|
||||
dirfd = open(dir_name, O_RDONLY);
|
||||
if (dirfd < 0) {
|
||||
THROW_ERROR("failed to open dir");
|
||||
}
|
||||
// AT_SYMLINK_NOFOLLOW is invalid if we modify timestamps of the file
|
||||
// referred to by the file descriptor dirfd
|
||||
ret = syscall(SYS_utimensat, dirfd, NULL, times, AT_SYMLINK_NOFOLLOW);
|
||||
if (ret != -1 && errno != EINVAL) {
|
||||
THROW_ERROR("utimnsat() should return EINVAL");
|
||||
}
|
||||
close(dirfd);
|
||||
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
typedef int(*test_utimes_func_t)(const char *);
|
||||
|
||||
static int test_utimes_framework(test_utimes_func_t fn) {
|
||||
const char *file_path = "/root/test_filesystem_utimes.txt";
|
||||
|
||||
if (create_file(file_path) < 0) {
|
||||
return FAIL;
|
||||
}
|
||||
if (fn(file_path) < 0) {
|
||||
return FAIL;
|
||||
}
|
||||
if (remove_file(file_path) < 0) {
|
||||
return FAIL;
|
||||
}
|
||||
return SUCCESS;
|
||||
}
|
||||
|
||||
static int test_utime() {
|
||||
return test_utimes_framework(__test_utime);
|
||||
}
|
||||
|
||||
static int test_utimes() {
|
||||
return test_utimes_framework(__test_utimes);
|
||||
}
|
||||
|
||||
static int test_futimesat() {
|
||||
return test_utimes_framework(__test_futimesat);
|
||||
}
|
||||
|
||||
static int test_futimesat_nullpath() {
|
||||
return test_utimes_framework(__test_futimesat_nullpath);
|
||||
}
|
||||
|
||||
static int test_utimensat() {
|
||||
return test_utimes_framework(__test_utimensat);
|
||||
}
|
||||
|
||||
static int test_utimensat_invalid_flag() {
|
||||
return test_utimes_framework(__test_utimensat_invalid_flag);
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// Test suite main
|
||||
// ============================================================================
|
||||
|
||||
static test_case_t test_cases[] = {
|
||||
TEST_CASE(test_utime),
|
||||
TEST_CASE(test_utimes),
|
||||
TEST_CASE(test_futimesat),
|
||||
TEST_CASE(test_futimesat_nullpath),
|
||||
TEST_CASE(test_utimensat),
|
||||
TEST_CASE(test_utimensat_invalid_flag),
|
||||
};
|
||||
|
||||
int main(int argc, const char *argv[]) {
|
||||
return test_suite_run(test_cases, ARRAY_SIZE(test_cases));
|
||||
}
|
Loading…
Reference in New Issue
Block a user