Path: blob/master/tools/testing/selftests/kvm/steal_time.c
53293 views
// SPDX-License-Identifier: GPL-2.01/*2* steal/stolen time test3*4* Copyright (C) 2020, Red Hat, Inc.5*/6#include <stdio.h>7#include <time.h>8#include <sched.h>9#include <pthread.h>10#include <linux/kernel.h>11#include <asm/kvm.h>12#ifdef __riscv13#include "sbi.h"14#else15#include <asm/kvm_para.h>16#endif1718#include "test_util.h"19#include "kvm_util.h"20#include "processor.h"21#include "ucall_common.h"2223#define NR_VCPUS 424#define ST_GPA_BASE (1 << 30)2526static void *st_gva[NR_VCPUS];27static uint64_t guest_stolen_time[NR_VCPUS];2829#if defined(__x86_64__)3031/* steal_time must have 64-byte alignment */32#define STEAL_TIME_SIZE ((sizeof(struct kvm_steal_time) + 63) & ~63)3334static void check_status(struct kvm_steal_time *st)35{36GUEST_ASSERT(!(READ_ONCE(st->version) & 1));37GUEST_ASSERT_EQ(READ_ONCE(st->flags), 0);38GUEST_ASSERT_EQ(READ_ONCE(st->preempted), 0);39}4041static void guest_code(int cpu)42{43struct kvm_steal_time *st = st_gva[cpu];44uint32_t version;4546GUEST_ASSERT_EQ(rdmsr(MSR_KVM_STEAL_TIME), ((uint64_t)st_gva[cpu] | KVM_MSR_ENABLED));4748memset(st, 0, sizeof(*st));49GUEST_SYNC(0);5051check_status(st);52WRITE_ONCE(guest_stolen_time[cpu], st->steal);53version = READ_ONCE(st->version);54check_status(st);55GUEST_SYNC(1);5657check_status(st);58GUEST_ASSERT(version < READ_ONCE(st->version));59WRITE_ONCE(guest_stolen_time[cpu], st->steal);60check_status(st);61GUEST_DONE();62}6364static bool is_steal_time_supported(struct kvm_vcpu *vcpu)65{66return kvm_cpu_has(X86_FEATURE_KVM_STEAL_TIME);67}6869static void steal_time_init(struct kvm_vcpu *vcpu, uint32_t i)70{71int ret;7273/* ST_GPA_BASE is identity mapped */74st_gva[i] = (void *)(ST_GPA_BASE + i * STEAL_TIME_SIZE);75sync_global_to_guest(vcpu->vm, st_gva[i]);7677ret = _vcpu_set_msr(vcpu, MSR_KVM_STEAL_TIME,78(ulong)st_gva[i] | KVM_STEAL_RESERVED_MASK);79TEST_ASSERT(ret == 0, "Bad GPA didn't fail");8081vcpu_set_msr(vcpu, MSR_KVM_STEAL_TIME, (ulong)st_gva[i] | KVM_MSR_ENABLED);82}8384static void steal_time_dump(struct kvm_vm *vm, uint32_t vcpu_idx)85{86struct kvm_steal_time *st = addr_gva2hva(vm, (ulong)st_gva[vcpu_idx]);8788ksft_print_msg("VCPU%d:\n", vcpu_idx);89ksft_print_msg(" steal: %lld\n", st->steal);90ksft_print_msg(" version: %d\n", st->version);91ksft_print_msg(" flags: %d\n", st->flags);92ksft_print_msg(" preempted: %d\n", st->preempted);93ksft_print_msg(" u8_pad: %d %d %d\n",94st->u8_pad[0], st->u8_pad[1], st->u8_pad[2]);95ksft_print_msg(" pad: %d %d %d %d %d %d %d %d %d %d %d\n",96st->pad[0], st->pad[1], st->pad[2], st->pad[3],97st->pad[4], st->pad[5], st->pad[6], st->pad[7],98st->pad[8], st->pad[9], st->pad[10]);99}100101#elif defined(__aarch64__)102103/* PV_TIME_ST must have 64-byte alignment */104#define STEAL_TIME_SIZE ((sizeof(struct st_time) + 63) & ~63)105106#define SMCCC_ARCH_FEATURES 0x80000001107#define PV_TIME_FEATURES 0xc5000020108#define PV_TIME_ST 0xc5000021109110struct st_time {111uint32_t rev;112uint32_t attr;113uint64_t st_time;114};115116static int64_t smccc(uint32_t func, uint64_t arg)117{118struct arm_smccc_res res;119120do_smccc(func, arg, 0, 0, 0, 0, 0, 0, &res);121return res.a0;122}123124static void check_status(struct st_time *st)125{126GUEST_ASSERT_EQ(READ_ONCE(st->rev), 0);127GUEST_ASSERT_EQ(READ_ONCE(st->attr), 0);128}129130static void guest_code(int cpu)131{132struct st_time *st;133int64_t status;134135status = smccc(SMCCC_ARCH_FEATURES, PV_TIME_FEATURES);136GUEST_ASSERT_EQ(status, 0);137status = smccc(PV_TIME_FEATURES, PV_TIME_FEATURES);138GUEST_ASSERT_EQ(status, 0);139status = smccc(PV_TIME_FEATURES, PV_TIME_ST);140GUEST_ASSERT_EQ(status, 0);141142status = smccc(PV_TIME_ST, 0);143GUEST_ASSERT_NE(status, -1);144GUEST_ASSERT_EQ(status, (ulong)st_gva[cpu]);145146st = (struct st_time *)status;147GUEST_SYNC(0);148149check_status(st);150WRITE_ONCE(guest_stolen_time[cpu], st->st_time);151GUEST_SYNC(1);152153check_status(st);154WRITE_ONCE(guest_stolen_time[cpu], st->st_time);155GUEST_DONE();156}157158static bool is_steal_time_supported(struct kvm_vcpu *vcpu)159{160struct kvm_device_attr dev = {161.group = KVM_ARM_VCPU_PVTIME_CTRL,162.attr = KVM_ARM_VCPU_PVTIME_IPA,163};164165return !__vcpu_ioctl(vcpu, KVM_HAS_DEVICE_ATTR, &dev);166}167168static void steal_time_init(struct kvm_vcpu *vcpu, uint32_t i)169{170struct kvm_vm *vm = vcpu->vm;171uint64_t st_ipa;172int ret;173174struct kvm_device_attr dev = {175.group = KVM_ARM_VCPU_PVTIME_CTRL,176.attr = KVM_ARM_VCPU_PVTIME_IPA,177.addr = (uint64_t)&st_ipa,178};179180vcpu_ioctl(vcpu, KVM_HAS_DEVICE_ATTR, &dev);181182/* ST_GPA_BASE is identity mapped */183st_gva[i] = (void *)(ST_GPA_BASE + i * STEAL_TIME_SIZE);184sync_global_to_guest(vm, st_gva[i]);185186st_ipa = (ulong)st_gva[i] | 1;187ret = __vcpu_ioctl(vcpu, KVM_SET_DEVICE_ATTR, &dev);188TEST_ASSERT(ret == -1 && errno == EINVAL, "Bad IPA didn't report EINVAL");189190st_ipa = (ulong)st_gva[i];191vcpu_ioctl(vcpu, KVM_SET_DEVICE_ATTR, &dev);192193ret = __vcpu_ioctl(vcpu, KVM_SET_DEVICE_ATTR, &dev);194TEST_ASSERT(ret == -1 && errno == EEXIST, "Set IPA twice without EEXIST");195}196197static void steal_time_dump(struct kvm_vm *vm, uint32_t vcpu_idx)198{199struct st_time *st = addr_gva2hva(vm, (ulong)st_gva[vcpu_idx]);200201ksft_print_msg("VCPU%d:\n", vcpu_idx);202ksft_print_msg(" rev: %d\n", st->rev);203ksft_print_msg(" attr: %d\n", st->attr);204ksft_print_msg(" st_time: %ld\n", st->st_time);205}206207#elif defined(__riscv)208209/* SBI STA shmem must have 64-byte alignment */210#define STEAL_TIME_SIZE ((sizeof(struct sta_struct) + 63) & ~63)211212static vm_paddr_t st_gpa[NR_VCPUS];213214struct sta_struct {215uint32_t sequence;216uint32_t flags;217uint64_t steal;218uint8_t preempted;219uint8_t pad[47];220} __packed;221222static void sta_set_shmem(vm_paddr_t gpa, unsigned long flags)223{224unsigned long lo = (unsigned long)gpa;225#if __riscv_xlen == 32226unsigned long hi = (unsigned long)(gpa >> 32);227#else228unsigned long hi = gpa == -1 ? -1 : 0;229#endif230struct sbiret ret = sbi_ecall(SBI_EXT_STA, 0, lo, hi, flags, 0, 0, 0);231232GUEST_ASSERT(ret.value == 0 && ret.error == 0);233}234235static void check_status(struct sta_struct *st)236{237GUEST_ASSERT(!(READ_ONCE(st->sequence) & 1));238GUEST_ASSERT(READ_ONCE(st->flags) == 0);239GUEST_ASSERT(READ_ONCE(st->preempted) == 0);240}241242static void guest_code(int cpu)243{244struct sta_struct *st = st_gva[cpu];245uint32_t sequence;246long out_val = 0;247bool probe;248249probe = guest_sbi_probe_extension(SBI_EXT_STA, &out_val);250GUEST_ASSERT(probe && out_val == 1);251252sta_set_shmem(st_gpa[cpu], 0);253GUEST_SYNC(0);254255check_status(st);256WRITE_ONCE(guest_stolen_time[cpu], st->steal);257sequence = READ_ONCE(st->sequence);258check_status(st);259GUEST_SYNC(1);260261check_status(st);262GUEST_ASSERT(sequence < READ_ONCE(st->sequence));263WRITE_ONCE(guest_stolen_time[cpu], st->steal);264check_status(st);265GUEST_DONE();266}267268static bool is_steal_time_supported(struct kvm_vcpu *vcpu)269{270uint64_t id = RISCV_SBI_EXT_REG(KVM_RISCV_SBI_EXT_STA);271unsigned long enabled = vcpu_get_reg(vcpu, id);272273TEST_ASSERT(enabled == 0 || enabled == 1, "Expected boolean result");274275return enabled;276}277278static void steal_time_init(struct kvm_vcpu *vcpu, uint32_t i)279{280/* ST_GPA_BASE is identity mapped */281st_gva[i] = (void *)(ST_GPA_BASE + i * STEAL_TIME_SIZE);282st_gpa[i] = addr_gva2gpa(vcpu->vm, (vm_vaddr_t)st_gva[i]);283sync_global_to_guest(vcpu->vm, st_gva[i]);284sync_global_to_guest(vcpu->vm, st_gpa[i]);285}286287static void steal_time_dump(struct kvm_vm *vm, uint32_t vcpu_idx)288{289struct sta_struct *st = addr_gva2hva(vm, (ulong)st_gva[vcpu_idx]);290int i;291292pr_info("VCPU%d:\n", vcpu_idx);293pr_info(" sequence: %d\n", st->sequence);294pr_info(" flags: %d\n", st->flags);295pr_info(" steal: %"PRIu64"\n", st->steal);296pr_info(" preempted: %d\n", st->preempted);297pr_info(" pad: ");298for (i = 0; i < 47; ++i)299pr_info("%d", st->pad[i]);300pr_info("\n");301}302303#elif defined(__loongarch__)304305/* steal_time must have 64-byte alignment */306#define STEAL_TIME_SIZE ((sizeof(struct kvm_steal_time) + 63) & ~63)307#define KVM_STEAL_PHYS_VALID BIT_ULL(0)308309struct kvm_steal_time {310__u64 steal;311__u32 version;312__u32 flags;313__u8 preempted;314__u8 pad[47];315};316317static void check_status(struct kvm_steal_time *st)318{319GUEST_ASSERT(!(READ_ONCE(st->version) & 1));320GUEST_ASSERT_EQ(READ_ONCE(st->flags), 0);321GUEST_ASSERT_EQ(READ_ONCE(st->preempted), 0);322}323324static void guest_code(int cpu)325{326uint32_t version;327struct kvm_steal_time *st = st_gva[cpu];328329memset(st, 0, sizeof(*st));330GUEST_SYNC(0);331332check_status(st);333WRITE_ONCE(guest_stolen_time[cpu], st->steal);334version = READ_ONCE(st->version);335check_status(st);336GUEST_SYNC(1);337338check_status(st);339GUEST_ASSERT(version < READ_ONCE(st->version));340WRITE_ONCE(guest_stolen_time[cpu], st->steal);341check_status(st);342GUEST_DONE();343}344345static bool is_steal_time_supported(struct kvm_vcpu *vcpu)346{347int err;348uint64_t val;349struct kvm_device_attr attr = {350.group = KVM_LOONGARCH_VCPU_CPUCFG,351.attr = CPUCFG_KVM_FEATURE,352.addr = (uint64_t)&val,353};354355err = __vcpu_ioctl(vcpu, KVM_HAS_DEVICE_ATTR, &attr);356if (err)357return false;358359err = __vcpu_ioctl(vcpu, KVM_GET_DEVICE_ATTR, &attr);360if (err)361return false;362363return val & BIT(KVM_FEATURE_STEAL_TIME);364}365366static void steal_time_init(struct kvm_vcpu *vcpu, uint32_t i)367{368int err;369uint64_t st_gpa;370struct kvm_vm *vm = vcpu->vm;371struct kvm_device_attr attr = {372.group = KVM_LOONGARCH_VCPU_PVTIME_CTRL,373.attr = KVM_LOONGARCH_VCPU_PVTIME_GPA,374.addr = (uint64_t)&st_gpa,375};376377/* ST_GPA_BASE is identity mapped */378st_gva[i] = (void *)(ST_GPA_BASE + i * STEAL_TIME_SIZE);379sync_global_to_guest(vm, st_gva[i]);380381err = __vcpu_ioctl(vcpu, KVM_HAS_DEVICE_ATTR, &attr);382TEST_ASSERT(err == 0, "No PV stealtime Feature");383384st_gpa = (unsigned long)st_gva[i] | KVM_STEAL_PHYS_VALID;385err = __vcpu_ioctl(vcpu, KVM_SET_DEVICE_ATTR, &attr);386TEST_ASSERT(err == 0, "Fail to set PV stealtime GPA");387}388389static void steal_time_dump(struct kvm_vm *vm, uint32_t vcpu_idx)390{391struct kvm_steal_time *st = addr_gva2hva(vm, (ulong)st_gva[vcpu_idx]);392393ksft_print_msg("VCPU%d:\n", vcpu_idx);394ksft_print_msg(" steal: %lld\n", st->steal);395ksft_print_msg(" flags: %d\n", st->flags);396ksft_print_msg(" version: %d\n", st->version);397ksft_print_msg(" preempted: %d\n", st->preempted);398}399#endif400401static void *do_steal_time(void *arg)402{403struct timespec ts, stop;404405clock_gettime(CLOCK_MONOTONIC, &ts);406stop = timespec_add_ns(ts, MIN_RUN_DELAY_NS);407408while (1) {409clock_gettime(CLOCK_MONOTONIC, &ts);410if (timespec_to_ns(timespec_sub(ts, stop)) >= 0)411break;412}413414return NULL;415}416417static void run_vcpu(struct kvm_vcpu *vcpu)418{419struct ucall uc;420421vcpu_run(vcpu);422423switch (get_ucall(vcpu, &uc)) {424case UCALL_SYNC:425case UCALL_DONE:426break;427case UCALL_ABORT:428REPORT_GUEST_ASSERT(uc);429default:430TEST_ASSERT(false, "Unexpected exit: %s",431exit_reason_str(vcpu->run->exit_reason));432}433}434435int main(int ac, char **av)436{437struct kvm_vcpu *vcpus[NR_VCPUS];438struct kvm_vm *vm;439pthread_attr_t attr;440pthread_t thread;441cpu_set_t cpuset;442unsigned int gpages;443long stolen_time;444long run_delay;445bool verbose;446int i;447448verbose = ac > 1 && (!strncmp(av[1], "-v", 3) || !strncmp(av[1], "--verbose", 10));449450/* Set CPU affinity so we can force preemption of the VCPU */451CPU_ZERO(&cpuset);452CPU_SET(0, &cpuset);453pthread_attr_init(&attr);454pthread_attr_setaffinity_np(&attr, sizeof(cpu_set_t), &cpuset);455pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);456457/* Create a VM and an identity mapped memslot for the steal time structure */458vm = vm_create_with_vcpus(NR_VCPUS, guest_code, vcpus);459gpages = vm_calc_num_guest_pages(VM_MODE_DEFAULT, STEAL_TIME_SIZE * NR_VCPUS);460vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, ST_GPA_BASE, 1, gpages, 0);461virt_map(vm, ST_GPA_BASE, ST_GPA_BASE, gpages);462463ksft_print_header();464TEST_REQUIRE(is_steal_time_supported(vcpus[0]));465ksft_set_plan(NR_VCPUS);466467/* Run test on each VCPU */468for (i = 0; i < NR_VCPUS; ++i) {469steal_time_init(vcpus[i], i);470471vcpu_args_set(vcpus[i], 1, i);472473/* First VCPU run initializes steal-time */474run_vcpu(vcpus[i]);475476/* Second VCPU run, expect guest stolen time to be <= run_delay */477run_vcpu(vcpus[i]);478sync_global_from_guest(vm, guest_stolen_time[i]);479stolen_time = guest_stolen_time[i];480run_delay = get_run_delay();481TEST_ASSERT(stolen_time <= run_delay,482"Expected stolen time <= %ld, got %ld",483run_delay, stolen_time);484485/* Steal time from the VCPU. The steal time thread has the same CPU affinity as the VCPUs. */486run_delay = get_run_delay();487pthread_create(&thread, &attr, do_steal_time, NULL);488do489sched_yield();490while (get_run_delay() - run_delay < MIN_RUN_DELAY_NS);491pthread_join(thread, NULL);492run_delay = get_run_delay() - run_delay;493TEST_ASSERT(run_delay >= MIN_RUN_DELAY_NS,494"Expected run_delay >= %ld, got %ld",495MIN_RUN_DELAY_NS, run_delay);496497/* Run VCPU again to confirm stolen time is consistent with run_delay */498run_vcpu(vcpus[i]);499sync_global_from_guest(vm, guest_stolen_time[i]);500stolen_time = guest_stolen_time[i] - stolen_time;501TEST_ASSERT(stolen_time >= run_delay,502"Expected stolen time >= %ld, got %ld",503run_delay, stolen_time);504505if (verbose) {506ksft_print_msg("VCPU%d: total-stolen-time=%ld test-stolen-time=%ld%s\n",507i, guest_stolen_time[i], stolen_time,508stolen_time == run_delay ?509" (BONUS: guest test-stolen-time even exactly matches test-run_delay)" : "");510steal_time_dump(vm, i);511}512ksft_test_result_pass("vcpu%d\n", i);513}514515/* Print results and exit() accordingly */516ksft_finished();517}518519520