162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * Copyright © 2018 Alexey Dobriyan <adobriyan@gmail.com>
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * Permission to use, copy, modify, and distribute this software for any
562306a36Sopenharmony_ci * purpose with or without fee is hereby granted, provided that the above
662306a36Sopenharmony_ci * copyright notice and this permission notice appear in all copies.
762306a36Sopenharmony_ci *
862306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
962306a36Sopenharmony_ci * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1062306a36Sopenharmony_ci * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1162306a36Sopenharmony_ci * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1262306a36Sopenharmony_ci * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1362306a36Sopenharmony_ci * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1462306a36Sopenharmony_ci * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1562306a36Sopenharmony_ci */
1662306a36Sopenharmony_ci// Test that boottime value in /proc/uptime and CLOCK_BOOTTIME increment
1762306a36Sopenharmony_ci// monotonically while shifting across CPUs. We don't test idle time
1862306a36Sopenharmony_ci// monotonicity due to broken iowait task counting, cf: comment above
1962306a36Sopenharmony_ci// get_cpu_idle_time_us()
2062306a36Sopenharmony_ci#undef NDEBUG
2162306a36Sopenharmony_ci#include <assert.h>
2262306a36Sopenharmony_ci#include <errno.h>
2362306a36Sopenharmony_ci#include <unistd.h>
2462306a36Sopenharmony_ci#include <sys/syscall.h>
2562306a36Sopenharmony_ci#include <stdlib.h>
2662306a36Sopenharmony_ci#include <string.h>
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci#include <stdint.h>
2962306a36Sopenharmony_ci#include <sys/types.h>
3062306a36Sopenharmony_ci#include <sys/stat.h>
3162306a36Sopenharmony_ci#include <fcntl.h>
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci#include "proc-uptime.h"
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_cistatic inline int sys_sched_getaffinity(pid_t pid, unsigned int len, unsigned long *m)
3662306a36Sopenharmony_ci{
3762306a36Sopenharmony_ci	return syscall(SYS_sched_getaffinity, pid, len, m);
3862306a36Sopenharmony_ci}
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_cistatic inline int sys_sched_setaffinity(pid_t pid, unsigned int len, unsigned long *m)
4162306a36Sopenharmony_ci{
4262306a36Sopenharmony_ci	return syscall(SYS_sched_setaffinity, pid, len, m);
4362306a36Sopenharmony_ci}
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ciint main(void)
4662306a36Sopenharmony_ci{
4762306a36Sopenharmony_ci	uint64_t u0, u1, c0, c1;
4862306a36Sopenharmony_ci	unsigned int len;
4962306a36Sopenharmony_ci	unsigned long *m;
5062306a36Sopenharmony_ci	unsigned int cpu;
5162306a36Sopenharmony_ci	int fd;
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	/* find out "nr_cpu_ids" */
5462306a36Sopenharmony_ci	m = NULL;
5562306a36Sopenharmony_ci	len = 0;
5662306a36Sopenharmony_ci	do {
5762306a36Sopenharmony_ci		len += sizeof(unsigned long);
5862306a36Sopenharmony_ci		free(m);
5962306a36Sopenharmony_ci		m = malloc(len);
6062306a36Sopenharmony_ci	} while (sys_sched_getaffinity(0, len, m) == -1 && errno == EINVAL);
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci	fd = open("/proc/uptime", O_RDONLY);
6362306a36Sopenharmony_ci	assert(fd >= 0);
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_ci	u0 = proc_uptime(fd);
6662306a36Sopenharmony_ci	c0 = clock_boottime();
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	for (cpu = 0; cpu < len * 8; cpu++) {
6962306a36Sopenharmony_ci		memset(m, 0, len);
7062306a36Sopenharmony_ci		m[cpu / (8 * sizeof(unsigned long))] |= 1UL << (cpu % (8 * sizeof(unsigned long)));
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci		/* CPU might not exist, ignore error */
7362306a36Sopenharmony_ci		sys_sched_setaffinity(0, len, m);
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci		u1 = proc_uptime(fd);
7662306a36Sopenharmony_ci		c1 = clock_boottime();
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci		/* Is /proc/uptime monotonic ? */
7962306a36Sopenharmony_ci		assert(u1 >= u0);
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci		/* Is CLOCK_BOOTTIME monotonic ? */
8262306a36Sopenharmony_ci		assert(c1 >= c0);
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ci		/* Is CLOCK_BOOTTIME VS /proc/uptime monotonic ? */
8562306a36Sopenharmony_ci		assert(c0 >= u0);
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci		u0 = u1;
8862306a36Sopenharmony_ci		c0 = c1;
8962306a36Sopenharmony_ci	}
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_ci	return 0;
9262306a36Sopenharmony_ci}
93