xref: /aosp_15_r20/external/ltp/testcases/kernel/syscalls/sysinfo/sysinfo03.c (revision 49cdfc7efb34551c7342be41a7384b9c40d7cab7)
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