|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | #define _GNU_SOURCE | 
|  | #include <errno.h> | 
|  | #include <fcntl.h> | 
|  | #include <sched.h> | 
|  | #include <stdio.h> | 
|  | #include <stdbool.h> | 
|  | #include <sys/stat.h> | 
|  | #include <sys/syscall.h> | 
|  | #include <sys/types.h> | 
|  | #include <sys/wait.h> | 
|  | #include <time.h> | 
|  | #include <unistd.h> | 
|  | #include <string.h> | 
|  | #include <pthread.h> | 
|  |  | 
|  | #include "log.h" | 
|  | #include "timens.h" | 
|  |  | 
|  | #define OFFSET (36000) | 
|  |  | 
|  | struct thread_args { | 
|  | char *tst_name; | 
|  | struct timespec *now; | 
|  | }; | 
|  |  | 
|  | static void *tcheck(void *_args) | 
|  | { | 
|  | struct thread_args *args = _args; | 
|  | struct timespec *now = args->now, tst; | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < 2; i++) { | 
|  | _gettime(CLOCK_MONOTONIC, &tst, i); | 
|  | if (abs(tst.tv_sec - now->tv_sec) > 5) { | 
|  | pr_fail("%s: in-thread: unexpected value: %ld (%ld)\n", | 
|  | args->tst_name, tst.tv_sec, now->tv_sec); | 
|  | return (void *)1UL; | 
|  | } | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static int check_in_thread(char *tst_name, struct timespec *now) | 
|  | { | 
|  | struct thread_args args = { | 
|  | .tst_name = tst_name, | 
|  | .now = now, | 
|  | }; | 
|  | pthread_t th; | 
|  | void *retval; | 
|  |  | 
|  | if (pthread_create(&th, NULL, tcheck, &args)) | 
|  | return pr_perror("thread"); | 
|  | if (pthread_join(th, &retval)) | 
|  | return pr_perror("pthread_join"); | 
|  | return !(retval == NULL); | 
|  | } | 
|  |  | 
|  | static int check(char *tst_name, struct timespec *now) | 
|  | { | 
|  | struct timespec tst; | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < 2; i++) { | 
|  | _gettime(CLOCK_MONOTONIC, &tst, i); | 
|  | if (abs(tst.tv_sec - now->tv_sec) > 5) | 
|  | return pr_fail("%s: unexpected value: %ld (%ld)\n", | 
|  | tst_name, tst.tv_sec, now->tv_sec); | 
|  | } | 
|  | if (check_in_thread(tst_name, now)) | 
|  | return 1; | 
|  | ksft_test_result_pass("%s\n", tst_name); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int main(int argc, char *argv[]) | 
|  | { | 
|  | struct timespec now; | 
|  | int status; | 
|  | pid_t pid; | 
|  |  | 
|  | if (argc > 1) { | 
|  | char *endptr; | 
|  |  | 
|  | ksft_cnt.ksft_pass = 1; | 
|  | now.tv_sec = strtoul(argv[1], &endptr, 0); | 
|  | if (*endptr != 0) | 
|  | return pr_perror("strtoul"); | 
|  |  | 
|  | return check("child after exec", &now); | 
|  | } | 
|  |  | 
|  | nscheck(); | 
|  |  | 
|  | ksft_set_plan(4); | 
|  |  | 
|  | clock_gettime(CLOCK_MONOTONIC, &now); | 
|  |  | 
|  | if (unshare_timens()) | 
|  | return 1; | 
|  |  | 
|  | if (_settime(CLOCK_MONOTONIC, OFFSET)) | 
|  | return 1; | 
|  |  | 
|  | if (check("parent before vfork", &now)) | 
|  | return 1; | 
|  |  | 
|  | pid = vfork(); | 
|  | if (pid < 0) | 
|  | return pr_perror("fork"); | 
|  |  | 
|  | if (pid == 0) { | 
|  | char now_str[64]; | 
|  | char *cargv[] = {"exec", now_str, NULL}; | 
|  | char *cenv[] = {NULL}; | 
|  |  | 
|  | /* Check for proper vvar offsets after execve. */ | 
|  | snprintf(now_str, sizeof(now_str), "%ld", now.tv_sec + OFFSET); | 
|  | execve("/proc/self/exe", cargv, cenv); | 
|  | pr_perror("execve"); | 
|  | _exit(1); | 
|  | } | 
|  |  | 
|  | if (waitpid(pid, &status, 0) != pid) | 
|  | return pr_perror("waitpid"); | 
|  |  | 
|  | if (status) | 
|  | ksft_exit_fail(); | 
|  | ksft_inc_pass_cnt(); | 
|  | ksft_test_result_pass("wait for child\n"); | 
|  |  | 
|  | /* Check that we are still in the source timens. */ | 
|  | if (check("parent after vfork", &now)) | 
|  | return 1; | 
|  |  | 
|  | ksft_exit_pass(); | 
|  | return 0; | 
|  | } |