Path: blob/master/tools/testing/selftests/kvm/steal_time.c
38189 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#endif304305static void *do_steal_time(void *arg)306{307struct timespec ts, stop;308309clock_gettime(CLOCK_MONOTONIC, &ts);310stop = timespec_add_ns(ts, MIN_RUN_DELAY_NS);311312while (1) {313clock_gettime(CLOCK_MONOTONIC, &ts);314if (timespec_to_ns(timespec_sub(ts, stop)) >= 0)315break;316}317318return NULL;319}320321static void run_vcpu(struct kvm_vcpu *vcpu)322{323struct ucall uc;324325vcpu_run(vcpu);326327switch (get_ucall(vcpu, &uc)) {328case UCALL_SYNC:329case UCALL_DONE:330break;331case UCALL_ABORT:332REPORT_GUEST_ASSERT(uc);333default:334TEST_ASSERT(false, "Unexpected exit: %s",335exit_reason_str(vcpu->run->exit_reason));336}337}338339int main(int ac, char **av)340{341struct kvm_vcpu *vcpus[NR_VCPUS];342struct kvm_vm *vm;343pthread_attr_t attr;344pthread_t thread;345cpu_set_t cpuset;346unsigned int gpages;347long stolen_time;348long run_delay;349bool verbose;350int i;351352verbose = ac > 1 && (!strncmp(av[1], "-v", 3) || !strncmp(av[1], "--verbose", 10));353354/* Set CPU affinity so we can force preemption of the VCPU */355CPU_ZERO(&cpuset);356CPU_SET(0, &cpuset);357pthread_attr_init(&attr);358pthread_attr_setaffinity_np(&attr, sizeof(cpu_set_t), &cpuset);359pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);360361/* Create a VM and an identity mapped memslot for the steal time structure */362vm = vm_create_with_vcpus(NR_VCPUS, guest_code, vcpus);363gpages = vm_calc_num_guest_pages(VM_MODE_DEFAULT, STEAL_TIME_SIZE * NR_VCPUS);364vm_userspace_mem_region_add(vm, VM_MEM_SRC_ANONYMOUS, ST_GPA_BASE, 1, gpages, 0);365virt_map(vm, ST_GPA_BASE, ST_GPA_BASE, gpages);366367ksft_print_header();368TEST_REQUIRE(is_steal_time_supported(vcpus[0]));369ksft_set_plan(NR_VCPUS);370371/* Run test on each VCPU */372for (i = 0; i < NR_VCPUS; ++i) {373steal_time_init(vcpus[i], i);374375vcpu_args_set(vcpus[i], 1, i);376377/* First VCPU run initializes steal-time */378run_vcpu(vcpus[i]);379380/* Second VCPU run, expect guest stolen time to be <= run_delay */381run_vcpu(vcpus[i]);382sync_global_from_guest(vm, guest_stolen_time[i]);383stolen_time = guest_stolen_time[i];384run_delay = get_run_delay();385TEST_ASSERT(stolen_time <= run_delay,386"Expected stolen time <= %ld, got %ld",387run_delay, stolen_time);388389/* Steal time from the VCPU. The steal time thread has the same CPU affinity as the VCPUs. */390run_delay = get_run_delay();391pthread_create(&thread, &attr, do_steal_time, NULL);392do393sched_yield();394while (get_run_delay() - run_delay < MIN_RUN_DELAY_NS);395pthread_join(thread, NULL);396run_delay = get_run_delay() - run_delay;397TEST_ASSERT(run_delay >= MIN_RUN_DELAY_NS,398"Expected run_delay >= %ld, got %ld",399MIN_RUN_DELAY_NS, run_delay);400401/* Run VCPU again to confirm stolen time is consistent with run_delay */402run_vcpu(vcpus[i]);403sync_global_from_guest(vm, guest_stolen_time[i]);404stolen_time = guest_stolen_time[i] - stolen_time;405TEST_ASSERT(stolen_time >= run_delay,406"Expected stolen time >= %ld, got %ld",407run_delay, stolen_time);408409if (verbose) {410ksft_print_msg("VCPU%d: total-stolen-time=%ld test-stolen-time=%ld%s\n",411i, guest_stolen_time[i], stolen_time,412stolen_time == run_delay ?413" (BONUS: guest test-stolen-time even exactly matches test-run_delay)" : "");414steal_time_dump(vm, i);415}416ksft_test_result_pass("vcpu%d\n", i);417}418419/* Print results and exit() accordingly */420ksft_finished();421}422423424