123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103 |
- // SPDX-License-Identifier: GPL-2.0
- #define _GNU_SOURCE
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <sched.h>
- #include <time.h>
- #include <stdio.h>
- #include <unistd.h>
- #include <sys/syscall.h>
- #include <dlfcn.h>
- #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)
- vdso = dlopen("linux-vdso32.so.1",
- RTLD_LAZY | RTLD_LOCAL | RTLD_NOLOAD);
- if (!vdso)
- vdso = dlopen("linux-vdso64.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)
- vdso_clock_gettime = (vgettime_t)dlsym(vdso, "__kernel_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;
- }
|