Path: blob/master/tools/testing/selftests/kvm/lib/test_util.c
38237 views
// SPDX-License-Identifier: GPL-2.0-only1/*2* tools/testing/selftests/kvm/lib/test_util.c3*4* Copyright (C) 2020, Google LLC.5*/6#include <stdio.h>7#include <stdarg.h>8#include <assert.h>9#include <ctype.h>10#include <limits.h>11#include <stdlib.h>12#include <time.h>13#include <sys/stat.h>14#include <sys/syscall.h>15#include <linux/mman.h>16#include "linux/kernel.h"1718#include "test_util.h"1920sigjmp_buf expect_sigbus_jmpbuf;2122void __attribute__((used)) expect_sigbus_handler(int signum)23{24siglongjmp(expect_sigbus_jmpbuf, 1);25}2627/*28* Random number generator that is usable from guest code. This is the29* Park-Miller LCG using standard constants.30*/3132struct guest_random_state new_guest_random_state(uint32_t seed)33{34struct guest_random_state s = {.seed = seed};35return s;36}3738uint32_t guest_random_u32(struct guest_random_state *state)39{40state->seed = (uint64_t)state->seed * 48271 % ((uint32_t)(1 << 31) - 1);41return state->seed;42}4344/*45* Parses "[0-9]+[kmgt]?".46*/47size_t parse_size(const char *size)48{49size_t base;50char *scale;51int shift = 0;5253TEST_ASSERT(size && isdigit(size[0]), "Need at least one digit in '%s'", size);5455base = strtoull(size, &scale, 0);5657TEST_ASSERT(base != ULLONG_MAX, "Overflow parsing size!");5859switch (tolower(*scale)) {60case 't':61shift = 40;62break;63case 'g':64shift = 30;65break;66case 'm':67shift = 20;68break;69case 'k':70shift = 10;71break;72case 'b':73case '\0':74shift = 0;75break;76default:77TEST_ASSERT(false, "Unknown size letter %c", *scale);78}7980TEST_ASSERT((base << shift) >> shift == base, "Overflow scaling size!");8182return base << shift;83}8485int64_t timespec_to_ns(struct timespec ts)86{87return (int64_t)ts.tv_nsec + 1000000000LL * (int64_t)ts.tv_sec;88}8990struct timespec timespec_add_ns(struct timespec ts, int64_t ns)91{92struct timespec res;9394res.tv_nsec = ts.tv_nsec + ns;95res.tv_sec = ts.tv_sec + res.tv_nsec / 1000000000LL;96res.tv_nsec %= 1000000000LL;9798return res;99}100101struct timespec timespec_add(struct timespec ts1, struct timespec ts2)102{103int64_t ns1 = timespec_to_ns(ts1);104int64_t ns2 = timespec_to_ns(ts2);105return timespec_add_ns((struct timespec){0}, ns1 + ns2);106}107108struct timespec timespec_sub(struct timespec ts1, struct timespec ts2)109{110int64_t ns1 = timespec_to_ns(ts1);111int64_t ns2 = timespec_to_ns(ts2);112return timespec_add_ns((struct timespec){0}, ns1 - ns2);113}114115struct timespec timespec_elapsed(struct timespec start)116{117struct timespec end;118119clock_gettime(CLOCK_MONOTONIC, &end);120return timespec_sub(end, start);121}122123struct timespec timespec_div(struct timespec ts, int divisor)124{125int64_t ns = timespec_to_ns(ts) / divisor;126127return timespec_add_ns((struct timespec){0}, ns);128}129130void print_skip(const char *fmt, ...)131{132va_list ap;133134assert(fmt);135va_start(ap, fmt);136vprintf(fmt, ap);137va_end(ap);138puts(", skipping test");139}140141static bool test_sysfs_path(const char *path)142{143struct stat statbuf;144int ret;145146ret = stat(path, &statbuf);147TEST_ASSERT(ret == 0 || (ret == -1 && errno == ENOENT),148"Error in stat()ing '%s'", path);149150return ret == 0;151}152153bool thp_configured(void)154{155return test_sysfs_path("/sys/kernel/mm/transparent_hugepage");156}157158static size_t get_sysfs_val(const char *path)159{160size_t size;161FILE *f;162int ret;163164f = fopen(path, "r");165TEST_ASSERT(f, "Error opening '%s'", path);166167ret = fscanf(f, "%ld", &size);168TEST_ASSERT(ret > 0, "Error reading '%s'", path);169170/* Re-scan the input stream to verify the entire file was read. */171ret = fscanf(f, "%ld", &size);172TEST_ASSERT(ret < 1, "Error reading '%s'", path);173174fclose(f);175return size;176}177178size_t get_trans_hugepagesz(void)179{180TEST_ASSERT(thp_configured(), "THP is not configured in host kernel");181182return get_sysfs_val("/sys/kernel/mm/transparent_hugepage/hpage_pmd_size");183}184185bool is_numa_balancing_enabled(void)186{187if (!test_sysfs_path("/proc/sys/kernel/numa_balancing"))188return false;189return get_sysfs_val("/proc/sys/kernel/numa_balancing") == 1;190}191192size_t get_def_hugetlb_pagesz(void)193{194char buf[64];195const char *hugepagesize = "Hugepagesize:";196const char *hugepages_total = "HugePages_Total:";197FILE *f;198199f = fopen("/proc/meminfo", "r");200TEST_ASSERT(f != NULL, "Error in opening /proc/meminfo");201202while (fgets(buf, sizeof(buf), f) != NULL) {203if (strstr(buf, hugepages_total) == buf) {204unsigned long long total = strtoull(buf + strlen(hugepages_total), NULL, 10);205if (!total) {206fprintf(stderr, "HUGETLB is not enabled in /proc/sys/vm/nr_hugepages\n");207exit(KSFT_SKIP);208}209}210if (strstr(buf, hugepagesize) == buf) {211fclose(f);212return strtoull(buf + strlen(hugepagesize), NULL, 10) << 10;213}214}215216if (feof(f)) {217fprintf(stderr, "HUGETLB is not configured in host kernel");218exit(KSFT_SKIP);219}220221TEST_FAIL("Error in reading /proc/meminfo");222}223224#define ANON_FLAGS (MAP_PRIVATE | MAP_ANONYMOUS)225#define ANON_HUGE_FLAGS (ANON_FLAGS | MAP_HUGETLB)226227const struct vm_mem_backing_src_alias *vm_mem_backing_src_alias(uint32_t i)228{229static const struct vm_mem_backing_src_alias aliases[] = {230[VM_MEM_SRC_ANONYMOUS] = {231.name = "anonymous",232.flag = ANON_FLAGS,233},234[VM_MEM_SRC_ANONYMOUS_THP] = {235.name = "anonymous_thp",236.flag = ANON_FLAGS,237},238[VM_MEM_SRC_ANONYMOUS_HUGETLB] = {239.name = "anonymous_hugetlb",240.flag = ANON_HUGE_FLAGS,241},242[VM_MEM_SRC_ANONYMOUS_HUGETLB_16KB] = {243.name = "anonymous_hugetlb_16kb",244.flag = ANON_HUGE_FLAGS | MAP_HUGE_16KB,245},246[VM_MEM_SRC_ANONYMOUS_HUGETLB_64KB] = {247.name = "anonymous_hugetlb_64kb",248.flag = ANON_HUGE_FLAGS | MAP_HUGE_64KB,249},250[VM_MEM_SRC_ANONYMOUS_HUGETLB_512KB] = {251.name = "anonymous_hugetlb_512kb",252.flag = ANON_HUGE_FLAGS | MAP_HUGE_512KB,253},254[VM_MEM_SRC_ANONYMOUS_HUGETLB_1MB] = {255.name = "anonymous_hugetlb_1mb",256.flag = ANON_HUGE_FLAGS | MAP_HUGE_1MB,257},258[VM_MEM_SRC_ANONYMOUS_HUGETLB_2MB] = {259.name = "anonymous_hugetlb_2mb",260.flag = ANON_HUGE_FLAGS | MAP_HUGE_2MB,261},262[VM_MEM_SRC_ANONYMOUS_HUGETLB_8MB] = {263.name = "anonymous_hugetlb_8mb",264.flag = ANON_HUGE_FLAGS | MAP_HUGE_8MB,265},266[VM_MEM_SRC_ANONYMOUS_HUGETLB_16MB] = {267.name = "anonymous_hugetlb_16mb",268.flag = ANON_HUGE_FLAGS | MAP_HUGE_16MB,269},270[VM_MEM_SRC_ANONYMOUS_HUGETLB_32MB] = {271.name = "anonymous_hugetlb_32mb",272.flag = ANON_HUGE_FLAGS | MAP_HUGE_32MB,273},274[VM_MEM_SRC_ANONYMOUS_HUGETLB_256MB] = {275.name = "anonymous_hugetlb_256mb",276.flag = ANON_HUGE_FLAGS | MAP_HUGE_256MB,277},278[VM_MEM_SRC_ANONYMOUS_HUGETLB_512MB] = {279.name = "anonymous_hugetlb_512mb",280.flag = ANON_HUGE_FLAGS | MAP_HUGE_512MB,281},282[VM_MEM_SRC_ANONYMOUS_HUGETLB_1GB] = {283.name = "anonymous_hugetlb_1gb",284.flag = ANON_HUGE_FLAGS | MAP_HUGE_1GB,285},286[VM_MEM_SRC_ANONYMOUS_HUGETLB_2GB] = {287.name = "anonymous_hugetlb_2gb",288.flag = ANON_HUGE_FLAGS | MAP_HUGE_2GB,289},290[VM_MEM_SRC_ANONYMOUS_HUGETLB_16GB] = {291.name = "anonymous_hugetlb_16gb",292.flag = ANON_HUGE_FLAGS | MAP_HUGE_16GB,293},294[VM_MEM_SRC_SHMEM] = {295.name = "shmem",296.flag = MAP_SHARED,297},298[VM_MEM_SRC_SHARED_HUGETLB] = {299.name = "shared_hugetlb",300/*301* No MAP_HUGETLB, we use MFD_HUGETLB instead. Since302* we're using "file backed" memory, we need to specify303* this when the FD is created, not when the area is304* mapped.305*/306.flag = MAP_SHARED,307},308};309_Static_assert(ARRAY_SIZE(aliases) == NUM_SRC_TYPES,310"Missing new backing src types?");311312TEST_ASSERT(i < NUM_SRC_TYPES, "Backing src type ID %d too big", i);313314return &aliases[i];315}316317#define MAP_HUGE_PAGE_SIZE(x) (1ULL << ((x >> MAP_HUGE_SHIFT) & MAP_HUGE_MASK))318319size_t get_backing_src_pagesz(uint32_t i)320{321uint32_t flag = vm_mem_backing_src_alias(i)->flag;322323switch (i) {324case VM_MEM_SRC_ANONYMOUS:325case VM_MEM_SRC_SHMEM:326return getpagesize();327case VM_MEM_SRC_ANONYMOUS_THP:328return get_trans_hugepagesz();329case VM_MEM_SRC_ANONYMOUS_HUGETLB:330case VM_MEM_SRC_SHARED_HUGETLB:331return get_def_hugetlb_pagesz();332default:333return MAP_HUGE_PAGE_SIZE(flag);334}335}336337bool is_backing_src_hugetlb(uint32_t i)338{339return !!(vm_mem_backing_src_alias(i)->flag & MAP_HUGETLB);340}341342static void print_available_backing_src_types(const char *prefix)343{344int i;345346printf("%sAvailable backing src types:\n", prefix);347348for (i = 0; i < NUM_SRC_TYPES; i++)349printf("%s %s\n", prefix, vm_mem_backing_src_alias(i)->name);350}351352void backing_src_help(const char *flag)353{354printf(" %s: specify the type of memory that should be used to\n"355" back the guest data region. (default: %s)\n",356flag, vm_mem_backing_src_alias(DEFAULT_VM_MEM_SRC)->name);357print_available_backing_src_types(" ");358}359360enum vm_mem_backing_src_type parse_backing_src_type(const char *type_name)361{362int i;363364for (i = 0; i < NUM_SRC_TYPES; i++)365if (!strcmp(type_name, vm_mem_backing_src_alias(i)->name))366return i;367368print_available_backing_src_types("");369TEST_FAIL("Unknown backing src type: %s", type_name);370return -1;371}372373long get_run_delay(void)374{375char path[64];376long val[2];377FILE *fp;378379sprintf(path, "/proc/%ld/schedstat", syscall(SYS_gettid));380fp = fopen(path, "r");381/* Return MIN_RUN_DELAY_NS upon failure just to be safe */382if (fscanf(fp, "%ld %ld ", &val[0], &val[1]) < 2)383val[1] = MIN_RUN_DELAY_NS;384fclose(fp);385386return val[1];387}388389int atoi_paranoid(const char *num_str)390{391char *end_ptr;392long num;393394errno = 0;395num = strtol(num_str, &end_ptr, 0);396TEST_ASSERT(!errno, "strtol(\"%s\") failed", num_str);397TEST_ASSERT(num_str != end_ptr,398"strtol(\"%s\") didn't find a valid integer.", num_str);399TEST_ASSERT(*end_ptr == '\0',400"strtol(\"%s\") failed to parse trailing characters \"%s\".",401num_str, end_ptr);402TEST_ASSERT(num >= INT_MIN && num <= INT_MAX,403"%ld not in range of [%d, %d]", num, INT_MIN, INT_MAX);404405return num;406}407408char *strdup_printf(const char *fmt, ...)409{410va_list ap;411char *str;412413va_start(ap, fmt);414TEST_ASSERT(vasprintf(&str, fmt, ap) >= 0, "vasprintf() failed");415va_end(ap);416417return str;418}419420#define CLOCKSOURCE_PATH "/sys/devices/system/clocksource/clocksource0/current_clocksource"421422char *sys_get_cur_clocksource(void)423{424char *clk_name;425struct stat st;426FILE *fp;427428fp = fopen(CLOCKSOURCE_PATH, "r");429TEST_ASSERT(fp, "failed to open clocksource file, errno: %d", errno);430431TEST_ASSERT(!fstat(fileno(fp), &st), "failed to stat clocksource file, errno: %d",432errno);433434clk_name = malloc(st.st_size);435TEST_ASSERT(clk_name, "failed to allocate buffer to read file");436437TEST_ASSERT(fgets(clk_name, st.st_size, fp), "failed to read clocksource file: %d",438ferror(fp));439440fclose(fp);441442return clk_name;443}444445446