1*49cdfc7eSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*49cdfc7eSAndroid Build Coastguard Worker /*
3*49cdfc7eSAndroid Build Coastguard Worker Copyright (c) 2020 Cyril Hrubis <[email protected]>
4*49cdfc7eSAndroid Build Coastguard Worker */
5*49cdfc7eSAndroid Build Coastguard Worker /*
6*49cdfc7eSAndroid Build Coastguard Worker
7*49cdfc7eSAndroid Build Coastguard Worker Test if CLOCK_BOOTTIME namespace offset is applied to sysinfo uptime and that
8*49cdfc7eSAndroid Build Coastguard Worker it's consistent with /proc/uptime as well.
9*49cdfc7eSAndroid Build Coastguard Worker
10*49cdfc7eSAndroid Build Coastguard Worker After a call to unshare(CLONE_NEWTIME) a new timer namespace is created, the
11*49cdfc7eSAndroid Build Coastguard Worker process that has called the unshare() can adjust offsets for CLOCK_MONOTONIC
12*49cdfc7eSAndroid Build Coastguard Worker and CLOCK_BOOTTIME for its children by writing to the '/proc/self/timens_offsets'.
13*49cdfc7eSAndroid Build Coastguard Worker
14*49cdfc7eSAndroid Build Coastguard Worker */
15*49cdfc7eSAndroid Build Coastguard Worker
16*49cdfc7eSAndroid Build Coastguard Worker #include <sys/sysinfo.h>
17*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/posix_clocks.h"
18*49cdfc7eSAndroid Build Coastguard Worker #include "tst_test.h"
19*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/sched.h"
20*49cdfc7eSAndroid Build Coastguard Worker
21*49cdfc7eSAndroid Build Coastguard Worker static int offsets[] = {
22*49cdfc7eSAndroid Build Coastguard Worker 10,
23*49cdfc7eSAndroid Build Coastguard Worker -10,
24*49cdfc7eSAndroid Build Coastguard Worker 3600,
25*49cdfc7eSAndroid Build Coastguard Worker };
26*49cdfc7eSAndroid Build Coastguard Worker
read_proc_uptime(void)27*49cdfc7eSAndroid Build Coastguard Worker static long read_proc_uptime(void)
28*49cdfc7eSAndroid Build Coastguard Worker {
29*49cdfc7eSAndroid Build Coastguard Worker long sec, sec_rem;
30*49cdfc7eSAndroid Build Coastguard Worker
31*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF("/proc/uptime", "%li.%li", &sec, &sec_rem);
32*49cdfc7eSAndroid Build Coastguard Worker
33*49cdfc7eSAndroid Build Coastguard Worker return sec + (sec_rem ? 1 : 0);
34*49cdfc7eSAndroid Build Coastguard Worker }
35*49cdfc7eSAndroid Build Coastguard Worker
verify_sysinfo(unsigned int n)36*49cdfc7eSAndroid Build Coastguard Worker static void verify_sysinfo(unsigned int n)
37*49cdfc7eSAndroid Build Coastguard Worker {
38*49cdfc7eSAndroid Build Coastguard Worker struct sysinfo si;
39*49cdfc7eSAndroid Build Coastguard Worker long uptime;
40*49cdfc7eSAndroid Build Coastguard Worker int off = offsets[n];
41*49cdfc7eSAndroid Build Coastguard Worker
42*49cdfc7eSAndroid Build Coastguard Worker SAFE_UNSHARE(CLONE_NEWTIME);
43*49cdfc7eSAndroid Build Coastguard Worker
44*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_PRINTF("/proc/self/timens_offsets", "%d %d 0",
45*49cdfc7eSAndroid Build Coastguard Worker CLOCK_BOOTTIME, off);
46*49cdfc7eSAndroid Build Coastguard Worker
47*49cdfc7eSAndroid Build Coastguard Worker sysinfo(&si);
48*49cdfc7eSAndroid Build Coastguard Worker
49*49cdfc7eSAndroid Build Coastguard Worker uptime = si.uptime;
50*49cdfc7eSAndroid Build Coastguard Worker
51*49cdfc7eSAndroid Build Coastguard Worker if (!SAFE_FORK()) {
52*49cdfc7eSAndroid Build Coastguard Worker sysinfo(&si);
53*49cdfc7eSAndroid Build Coastguard Worker long proc_uptime = read_proc_uptime();
54*49cdfc7eSAndroid Build Coastguard Worker
55*49cdfc7eSAndroid Build Coastguard Worker long diff = si.uptime - uptime;
56*49cdfc7eSAndroid Build Coastguard Worker
57*49cdfc7eSAndroid Build Coastguard Worker if (diff < off || diff > off + 1)
58*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL, "Wrong sysinfo uptime offset %li", diff);
59*49cdfc7eSAndroid Build Coastguard Worker else
60*49cdfc7eSAndroid Build Coastguard Worker tst_res(TPASS, "Correct sysinfo uptime offset %i", off);
61*49cdfc7eSAndroid Build Coastguard Worker
62*49cdfc7eSAndroid Build Coastguard Worker if (si.uptime < proc_uptime || si.uptime > proc_uptime + 1) {
63*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL, "/proc/uptime %li differs from sysinfo %li",
64*49cdfc7eSAndroid Build Coastguard Worker proc_uptime, si.uptime);
65*49cdfc7eSAndroid Build Coastguard Worker } else {
66*49cdfc7eSAndroid Build Coastguard Worker tst_res(TPASS, "/proc/uptime is consistent with sysinfo");
67*49cdfc7eSAndroid Build Coastguard Worker }
68*49cdfc7eSAndroid Build Coastguard Worker }
69*49cdfc7eSAndroid Build Coastguard Worker }
70*49cdfc7eSAndroid Build Coastguard Worker
71*49cdfc7eSAndroid Build Coastguard Worker static struct tst_test test = {
72*49cdfc7eSAndroid Build Coastguard Worker .tcnt = ARRAY_SIZE(offsets),
73*49cdfc7eSAndroid Build Coastguard Worker .test = verify_sysinfo,
74*49cdfc7eSAndroid Build Coastguard Worker .needs_root = 1,
75*49cdfc7eSAndroid Build Coastguard Worker .forks_child = 1,
76*49cdfc7eSAndroid Build Coastguard Worker .needs_kconfigs = (const char *[]) {
77*49cdfc7eSAndroid Build Coastguard Worker "CONFIG_TIME_NS=y",
78*49cdfc7eSAndroid Build Coastguard Worker NULL
79*49cdfc7eSAndroid Build Coastguard Worker },
80*49cdfc7eSAndroid Build Coastguard Worker .tags = (const struct tst_tag[]) {
81*49cdfc7eSAndroid Build Coastguard Worker {"linux-git", "ecc421e05bab"},
82*49cdfc7eSAndroid Build Coastguard Worker {}
83*49cdfc7eSAndroid Build Coastguard Worker }
84*49cdfc7eSAndroid Build Coastguard Worker };
85