// SPDX-License-Identifier: GPL-2.0 #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include #include "log.h" #include "timens.h" typedef int (*vgettime_t)(clockid_t, struct timespec *); vgettime_t vdso_clock_gettime; static void fill_function_pointers(void) { void *vdso = dlopen("linux-vdso.so.1", RTLD_LAZY | RTLD_LOCAL | RTLD_NOLOAD); if (!vdso) vdso = dlopen("linux-gate.so.1", RTLD_LAZY | RTLD_LOCAL | RTLD_NOLOAD); if (!vdso) { pr_err("[WARN]\tfailed to find vDSO\n"); return; } vdso_clock_gettime = (vgettime_t)dlsym(vdso, "__vdso_clock_gettime"); if (!vdso_clock_gettime) pr_err("Warning: failed to find clock_gettime in vDSO\n"); } static void test(clock_t clockid, char *clockstr, bool in_ns) { struct timespec tp, start; long i = 0; const int timeout = 3; vdso_clock_gettime(clockid, &start); tp = start; for (tp = start; start.tv_sec + timeout > tp.tv_sec || (start.tv_sec + timeout == tp.tv_sec && start.tv_nsec > tp.tv_nsec); i++) { vdso_clock_gettime(clockid, &tp); } ksft_test_result_pass("%s:\tclock: %10s\tcycles:\t%10ld\n", in_ns ? "ns" : "host", clockstr, i); } int main(int argc, char *argv[]) { time_t offset = 10; int nsfd; ksft_set_plan(8); fill_function_pointers(); test(CLOCK_MONOTONIC, "monotonic", false); test(CLOCK_MONOTONIC_COARSE, "monotonic-coarse", false); test(CLOCK_MONOTONIC_RAW, "monotonic-raw", false); test(CLOCK_BOOTTIME, "boottime", false); nscheck(); if (unshare_timens()) return 1; nsfd = open("/proc/self/ns/time_for_children", O_RDONLY); if (nsfd < 0) return pr_perror("Can't open a time namespace"); if (_settime(CLOCK_MONOTONIC, offset)) return 1; if (_settime(CLOCK_BOOTTIME, offset)) return 1; if (setns(nsfd, CLONE_NEWTIME)) return pr_perror("setns"); test(CLOCK_MONOTONIC, "monotonic", true); test(CLOCK_MONOTONIC_COARSE, "monotonic-coarse", true); test(CLOCK_MONOTONIC_RAW, "monotonic-raw", true); test(CLOCK_BOOTTIME, "boottime", true); ksft_exit_pass(); return 0; }