146e00343SAndrei Vagin // SPDX-License-Identifier: GPL-2.0
246e00343SAndrei Vagin #define _GNU_SOURCE
346e00343SAndrei Vagin #include <sched.h>
446e00343SAndrei Vagin 
546e00343SAndrei Vagin #include <sys/timerfd.h>
646e00343SAndrei Vagin #include <sys/syscall.h>
746e00343SAndrei Vagin #include <time.h>
846e00343SAndrei Vagin #include <unistd.h>
946e00343SAndrei Vagin #include <stdlib.h>
1046e00343SAndrei Vagin #include <stdio.h>
1146e00343SAndrei Vagin #include <stdint.h>
1246e00343SAndrei Vagin #include <pthread.h>
1346e00343SAndrei Vagin #include <signal.h>
1446e00343SAndrei Vagin #include <string.h>
1546e00343SAndrei Vagin 
1646e00343SAndrei Vagin #include "log.h"
1746e00343SAndrei Vagin #include "timens.h"
1846e00343SAndrei Vagin 
test_sig(int sig)1946e00343SAndrei Vagin void test_sig(int sig)
2046e00343SAndrei Vagin {
2146e00343SAndrei Vagin 	if (sig == SIGUSR2)
2246e00343SAndrei Vagin 		pthread_exit(NULL);
2346e00343SAndrei Vagin }
2446e00343SAndrei Vagin 
2546e00343SAndrei Vagin struct thread_args {
2646e00343SAndrei Vagin 	struct timespec *now, *rem;
2746e00343SAndrei Vagin 	pthread_mutex_t *lock;
2846e00343SAndrei Vagin 	int clockid;
2946e00343SAndrei Vagin 	int abs;
3046e00343SAndrei Vagin };
3146e00343SAndrei Vagin 
call_nanosleep(void * _args)3246e00343SAndrei Vagin void *call_nanosleep(void *_args)
3346e00343SAndrei Vagin {
3446e00343SAndrei Vagin 	struct thread_args *args = _args;
3546e00343SAndrei Vagin 
3646e00343SAndrei Vagin 	clock_nanosleep(args->clockid, args->abs ? TIMER_ABSTIME : 0, args->now, args->rem);
3746e00343SAndrei Vagin 	pthread_mutex_unlock(args->lock);
3846e00343SAndrei Vagin 	return NULL;
3946e00343SAndrei Vagin }
4046e00343SAndrei Vagin 
run_test(int clockid,int abs)4146e00343SAndrei Vagin int run_test(int clockid, int abs)
4246e00343SAndrei Vagin {
4346e00343SAndrei Vagin 	struct timespec now = {}, rem;
4446e00343SAndrei Vagin 	struct thread_args args = { .now = &now, .rem = &rem, .clockid = clockid};
4546e00343SAndrei Vagin 	struct timespec start;
4646e00343SAndrei Vagin 	pthread_mutex_t lock;
4746e00343SAndrei Vagin 	pthread_t thread;
4846e00343SAndrei Vagin 	int j, ok, ret;
4946e00343SAndrei Vagin 
5046e00343SAndrei Vagin 	signal(SIGUSR1, test_sig);
5146e00343SAndrei Vagin 	signal(SIGUSR2, test_sig);
5246e00343SAndrei Vagin 
5346e00343SAndrei Vagin 	pthread_mutex_init(&lock, NULL);
5446e00343SAndrei Vagin 	pthread_mutex_lock(&lock);
5546e00343SAndrei Vagin 
5646e00343SAndrei Vagin 	if (clock_gettime(clockid, &start) == -1) {
5746e00343SAndrei Vagin 		if (errno == EINVAL && check_skip(clockid))
5846e00343SAndrei Vagin 			return 0;
5946e00343SAndrei Vagin 		return pr_perror("clock_gettime");
6046e00343SAndrei Vagin 	}
6146e00343SAndrei Vagin 
6246e00343SAndrei Vagin 
6346e00343SAndrei Vagin 	if (abs) {
6446e00343SAndrei Vagin 		now.tv_sec = start.tv_sec;
6546e00343SAndrei Vagin 		now.tv_nsec = start.tv_nsec;
6646e00343SAndrei Vagin 	}
6746e00343SAndrei Vagin 
6846e00343SAndrei Vagin 	now.tv_sec += 3600;
6946e00343SAndrei Vagin 	args.abs = abs;
7046e00343SAndrei Vagin 	args.lock = &lock;
7146e00343SAndrei Vagin 	ret = pthread_create(&thread, NULL, call_nanosleep, &args);
7246e00343SAndrei Vagin 	if (ret != 0) {
7346e00343SAndrei Vagin 		pr_err("Unable to create a thread: %s", strerror(ret));
7446e00343SAndrei Vagin 		return 1;
7546e00343SAndrei Vagin 	}
7646e00343SAndrei Vagin 
7746e00343SAndrei Vagin 	/* Wait when the thread will call clock_nanosleep(). */
7846e00343SAndrei Vagin 	ok = 0;
7946e00343SAndrei Vagin 	for (j = 0; j < 8; j++) {
8046e00343SAndrei Vagin 		/* The maximum timeout is about 5 seconds. */
8146e00343SAndrei Vagin 		usleep(10000 << j);
8246e00343SAndrei Vagin 
8346e00343SAndrei Vagin 		/* Try to interrupt clock_nanosleep(). */
8446e00343SAndrei Vagin 		pthread_kill(thread, SIGUSR1);
8546e00343SAndrei Vagin 
8646e00343SAndrei Vagin 		usleep(10000 << j);
8746e00343SAndrei Vagin 		/* Check whether clock_nanosleep() has been interrupted or not. */
8846e00343SAndrei Vagin 		if (pthread_mutex_trylock(&lock) == 0) {
8946e00343SAndrei Vagin 			/**/
9046e00343SAndrei Vagin 			ok = 1;
9146e00343SAndrei Vagin 			break;
9246e00343SAndrei Vagin 		}
9346e00343SAndrei Vagin 	}
9446e00343SAndrei Vagin 	if (!ok)
9546e00343SAndrei Vagin 		pthread_kill(thread, SIGUSR2);
9646e00343SAndrei Vagin 	pthread_join(thread, NULL);
9746e00343SAndrei Vagin 	pthread_mutex_destroy(&lock);
9846e00343SAndrei Vagin 
9946e00343SAndrei Vagin 	if (!ok) {
10046e00343SAndrei Vagin 		ksft_test_result_pass("clockid: %d abs:%d timeout\n", clockid, abs);
10146e00343SAndrei Vagin 		return 1;
10246e00343SAndrei Vagin 	}
10346e00343SAndrei Vagin 
10446e00343SAndrei Vagin 	if (rem.tv_sec < 3300 || rem.tv_sec > 3900) {
10546e00343SAndrei Vagin 		pr_fail("clockid: %d abs: %d remain: %ld\n",
10646e00343SAndrei Vagin 			clockid, abs, rem.tv_sec);
10746e00343SAndrei Vagin 		return 1;
10846e00343SAndrei Vagin 	}
10946e00343SAndrei Vagin 	ksft_test_result_pass("clockid: %d abs:%d\n", clockid, abs);
11046e00343SAndrei Vagin 
11146e00343SAndrei Vagin 	return 0;
11246e00343SAndrei Vagin }
11346e00343SAndrei Vagin 
main(int argc,char * argv[])11446e00343SAndrei Vagin int main(int argc, char *argv[])
11546e00343SAndrei Vagin {
11646e00343SAndrei Vagin 	int ret, nsfd;
11746e00343SAndrei Vagin 
11846e00343SAndrei Vagin 	nscheck();
11946e00343SAndrei Vagin 
12046e00343SAndrei Vagin 	ksft_set_plan(4);
12146e00343SAndrei Vagin 
122558ae035SAndrei Vagin 	check_supported_timers();
12346e00343SAndrei Vagin 
12446e00343SAndrei Vagin 	if (unshare_timens())
12546e00343SAndrei Vagin 		return 1;
12646e00343SAndrei Vagin 
12746e00343SAndrei Vagin 	if (_settime(CLOCK_MONOTONIC, 7 * 24 * 3600))
12846e00343SAndrei Vagin 		return 1;
12946e00343SAndrei Vagin 	if (_settime(CLOCK_BOOTTIME, 9 * 24 * 3600))
13046e00343SAndrei Vagin 		return 1;
13146e00343SAndrei Vagin 
13246e00343SAndrei Vagin 	nsfd = open("/proc/self/ns/time_for_children", O_RDONLY);
13346e00343SAndrei Vagin 	if (nsfd < 0)
13446e00343SAndrei Vagin 		return pr_perror("Unable to open timens_for_children");
13546e00343SAndrei Vagin 
13646e00343SAndrei Vagin 	if (setns(nsfd, CLONE_NEWTIME))
13746e00343SAndrei Vagin 		return pr_perror("Unable to set timens");
13846e00343SAndrei Vagin 
13946e00343SAndrei Vagin 	ret = 0;
14046e00343SAndrei Vagin 	ret |= run_test(CLOCK_MONOTONIC, 0);
14146e00343SAndrei Vagin 	ret |= run_test(CLOCK_MONOTONIC, 1);
14246e00343SAndrei Vagin 	ret |= run_test(CLOCK_BOOTTIME_ALARM, 0);
14346e00343SAndrei Vagin 	ret |= run_test(CLOCK_BOOTTIME_ALARM, 1);
14446e00343SAndrei Vagin 
14546e00343SAndrei Vagin 	if (ret)
14646e00343SAndrei Vagin 		ksft_exit_fail();
14746e00343SAndrei Vagin 	ksft_exit_pass();
14846e00343SAndrei Vagin 	return ret;
14946e00343SAndrei Vagin }
150