mirror of
https://github.com/torvalds/linux.git
synced 2026-04-20 15:53:59 -04:00
x86 Xen-for-KVM: * Allow the Xen runstate information to cross a page boundary * Allow XEN_RUNSTATE_UPDATE flag behaviour to be configured * add support for 32-bit guests in SCHEDOP_poll x86 fixes: * One-off fixes for various emulation flows (SGX, VMXON, NRIPS=0). * Reinstate IBPB on emulated VM-Exit that was incorrectly dropped a few years back when eliminating unnecessary barriers when switching between vmcs01 and vmcs02. * Clean up the MSR filter docs. * Clean up vmread_error_trampoline() to make it more obvious that params must be passed on the stack, even for x86-64. * Let userspace set all supported bits in MSR_IA32_FEAT_CTL irrespective of the current guest CPUID. * Fudge around a race with TSC refinement that results in KVM incorrectly thinking a guest needs TSC scaling when running on a CPU with a constant TSC, but no hardware-enumerated TSC frequency. * Advertise (on AMD) that the SMM_CTL MSR is not supported * Remove unnecessary exports Selftests: * Fix an inverted check in the access tracking perf test, and restore support for asserting that there aren't too many idle pages when running on bare metal. * Fix an ordering issue in the AMX test introduced by recent conversions to use kvm_cpu_has(), and harden the code to guard against similar bugs in the future. Anything that tiggers caching of KVM's supported CPUID, kvm_cpu_has() in this case, effectively hides opt-in XSAVE features if the caching occurs before the test opts in via prctl(). * Fix build errors that occur in certain setups (unsure exactly what is unique about the problematic setup) due to glibc overriding static_assert() to a variant that requires a custom message. * Introduce actual atomics for clear/set_bit() in selftests Documentation: * Remove deleted ioctls from documentation * Various fixes
104 lines
2.0 KiB
C
104 lines
2.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
#include "kvm_util.h"
|
|
#include "linux/types.h"
|
|
#include "linux/bitmap.h"
|
|
#include "linux/atomic.h"
|
|
|
|
struct ucall_header {
|
|
DECLARE_BITMAP(in_use, KVM_MAX_VCPUS);
|
|
struct ucall ucalls[KVM_MAX_VCPUS];
|
|
};
|
|
|
|
/*
|
|
* ucall_pool holds per-VM values (global data is duplicated by each VM), it
|
|
* must not be accessed from host code.
|
|
*/
|
|
static struct ucall_header *ucall_pool;
|
|
|
|
void ucall_init(struct kvm_vm *vm, vm_paddr_t mmio_gpa)
|
|
{
|
|
struct ucall_header *hdr;
|
|
struct ucall *uc;
|
|
vm_vaddr_t vaddr;
|
|
int i;
|
|
|
|
vaddr = __vm_vaddr_alloc(vm, sizeof(*hdr), KVM_UTIL_MIN_VADDR, MEM_REGION_DATA);
|
|
hdr = (struct ucall_header *)addr_gva2hva(vm, vaddr);
|
|
memset(hdr, 0, sizeof(*hdr));
|
|
|
|
for (i = 0; i < KVM_MAX_VCPUS; ++i) {
|
|
uc = &hdr->ucalls[i];
|
|
uc->hva = uc;
|
|
}
|
|
|
|
write_guest_global(vm, ucall_pool, (struct ucall_header *)vaddr);
|
|
|
|
ucall_arch_init(vm, mmio_gpa);
|
|
}
|
|
|
|
static struct ucall *ucall_alloc(void)
|
|
{
|
|
struct ucall *uc;
|
|
int i;
|
|
|
|
GUEST_ASSERT(ucall_pool);
|
|
|
|
for (i = 0; i < KVM_MAX_VCPUS; ++i) {
|
|
if (!test_and_set_bit(i, ucall_pool->in_use)) {
|
|
uc = &ucall_pool->ucalls[i];
|
|
memset(uc->args, 0, sizeof(uc->args));
|
|
return uc;
|
|
}
|
|
}
|
|
|
|
GUEST_ASSERT(0);
|
|
return NULL;
|
|
}
|
|
|
|
static void ucall_free(struct ucall *uc)
|
|
{
|
|
/* Beware, here be pointer arithmetic. */
|
|
clear_bit(uc - ucall_pool->ucalls, ucall_pool->in_use);
|
|
}
|
|
|
|
void ucall(uint64_t cmd, int nargs, ...)
|
|
{
|
|
struct ucall *uc;
|
|
va_list va;
|
|
int i;
|
|
|
|
uc = ucall_alloc();
|
|
|
|
WRITE_ONCE(uc->cmd, cmd);
|
|
|
|
nargs = min(nargs, UCALL_MAX_ARGS);
|
|
|
|
va_start(va, nargs);
|
|
for (i = 0; i < nargs; ++i)
|
|
WRITE_ONCE(uc->args[i], va_arg(va, uint64_t));
|
|
va_end(va);
|
|
|
|
ucall_arch_do_ucall((vm_vaddr_t)uc->hva);
|
|
|
|
ucall_free(uc);
|
|
}
|
|
|
|
uint64_t get_ucall(struct kvm_vcpu *vcpu, struct ucall *uc)
|
|
{
|
|
struct ucall ucall;
|
|
void *addr;
|
|
|
|
if (!uc)
|
|
uc = &ucall;
|
|
|
|
addr = ucall_arch_get_ucall(vcpu);
|
|
if (addr) {
|
|
memcpy(uc, addr, sizeof(*uc));
|
|
vcpu_run_complete_io(vcpu);
|
|
} else {
|
|
memset(uc, 0, sizeof(*uc));
|
|
}
|
|
|
|
return uc->cmd;
|
|
}
|