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