* HVF fixes
* Extra qos-test debugging output (Christian) * SEV secret address autodetection (James) * SEV-ES support (Thomas) * Relocatable paths bugfix (Stefan) * RR fix (Pavel) * EventNotifier fix (Greg) -----BEGIN PGP SIGNATURE----- iQFIBAABCAAyFiEE8TM4V0tmI4mGbHaCv/vSX3jHroMFAmAr778UHHBib256aW5p QHJlZGhhdC5jb20ACgkQv/vSX3jHroNVLwf/V3lb/HbyqFkhacB9eqEsEXGC3Hdp hU4J11P3lGS84muByxCdfw1axCGZ5x2cJmJSE71LfCcHXxEQSx4FmfxX5xeKbp1n vHPJ1XKhsFkOYA2O6mCW4yynTfizmp+JK36wwjmG3BEXTMMC5o2V8gAnzkP1sT9l 0h454CtPq2lD0upgVIvI7AStpWXZwysh0hQEDk8TsIfFfzLNs+MJyvlPGn4pj+kN k+G3475FinPdncIBGsnRNMfiBmA4/L0L4lriQzZPV57lDfZ8sJkrmh1+/JfK6vsb FWIe6Suior6JGorzATbXrFhmNJ+FxNNEmlzSdqRxRz7CDv0SDZb7Ckv37Q== =FDIr -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/bonzini-gitlab/tags/for-upstream' into staging * HVF fixes * Extra qos-test debugging output (Christian) * SEV secret address autodetection (James) * SEV-ES support (Thomas) * Relocatable paths bugfix (Stefan) * RR fix (Pavel) * EventNotifier fix (Greg) # gpg: Signature made Tue 16 Feb 2021 16:15:59 GMT # gpg: using RSA key F13338574B662389866C7682BFFBD25F78C7AE83 # gpg: issuer "pbonzini@redhat.com" # gpg: Good signature from "Paolo Bonzini <bonzini@gnu.org>" [full] # gpg: aka "Paolo Bonzini <pbonzini@redhat.com>" [full] # Primary key fingerprint: 46F5 9FBD 57D6 12E7 BFD4 E2F7 7E15 100C CD36 69B1 # Subkey fingerprint: F133 3857 4B66 2389 866C 7682 BFFB D25F 78C7 AE83 * remotes/bonzini-gitlab/tags/for-upstream: (21 commits) replay: fix icount request when replaying clock access event_notifier: Set ->initialized earlier in event_notifier_init() hvf: Fetch cr4 before evaluating CPUID(1) target/i386/hvf: add rdmsr 35H MSR_CORE_THREAD_COUNT hvf: x86: Remove unused definitions target/i386/hvf: add vmware-cpuid-freq cpu feature hvf: Guard xgetbv call util/cutils: Skip "." when looking for next directory component tests/qtest/qos-test: dump QEMU command if verbose tests/qtest/qos-test: dump environment variables if verbose tests/qtest/qos-test: dump qos graph if verbose libqos/qgraph_internal: add qos_printf() and qos_printf_literal() libqos/qgraph: add qos_node_create_driver_named() sev/i386: Enable an SEV-ES guest based on SEV policy kvm/i386: Use a per-VM check for SMM capability sev/i386: Don't allow a system reset under an SEV-ES guest sev/i386: Allow AP booting under SEV-ES sev/i386: Require in-kernel irqchip support for SEV-ES guests sev/i386: Add initial support for SEV-ES sev: update sev-inject-launch-secret to make gpa optional ... Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
f0f75dc174
@ -39,7 +39,6 @@
|
||||
#include "qemu/main-loop.h"
|
||||
#include "trace.h"
|
||||
#include "hw/irq.h"
|
||||
#include "sysemu/sev.h"
|
||||
#include "qapi/visitor.h"
|
||||
#include "qapi/qapi-types-common.h"
|
||||
#include "qapi/qapi-visit-common.h"
|
||||
@ -2313,6 +2312,11 @@ void kvm_flush_coalesced_mmio_buffer(void)
|
||||
s->coalesced_flush_in_progress = false;
|
||||
}
|
||||
|
||||
bool kvm_cpu_check_are_resettable(void)
|
||||
{
|
||||
return kvm_arch_cpu_check_are_resettable();
|
||||
}
|
||||
|
||||
static void do_kvm_cpu_synchronize_state(CPUState *cpu, run_on_cpu_data arg)
|
||||
{
|
||||
if (!cpu->vcpu_dirty) {
|
||||
|
@ -125,6 +125,113 @@ void pc_system_flash_cleanup_unused(PCMachineState *pcms)
|
||||
}
|
||||
}
|
||||
|
||||
#define OVMF_TABLE_FOOTER_GUID "96b582de-1fb2-45f7-baea-a366c55a082d"
|
||||
|
||||
static uint8_t *ovmf_table;
|
||||
static int ovmf_table_len;
|
||||
|
||||
static void pc_system_parse_ovmf_flash(uint8_t *flash_ptr, size_t flash_size)
|
||||
{
|
||||
uint8_t *ptr;
|
||||
QemuUUID guid;
|
||||
int tot_len;
|
||||
|
||||
/* should only be called once */
|
||||
if (ovmf_table) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (flash_size < TARGET_PAGE_SIZE) {
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* if this is OVMF there will be a table footer
|
||||
* guid 48 bytes before the end of the flash file. If it's
|
||||
* not found, silently abort the flash parsing.
|
||||
*/
|
||||
qemu_uuid_parse(OVMF_TABLE_FOOTER_GUID, &guid);
|
||||
guid = qemu_uuid_bswap(guid); /* guids are LE */
|
||||
ptr = flash_ptr + flash_size - 48;
|
||||
if (!qemu_uuid_is_equal((QemuUUID *)ptr, &guid)) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* if found, just before is two byte table length */
|
||||
ptr -= sizeof(uint16_t);
|
||||
tot_len = le16_to_cpu(*(uint16_t *)ptr) - sizeof(guid) - sizeof(uint16_t);
|
||||
|
||||
if (tot_len <= 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
ovmf_table = g_malloc(tot_len);
|
||||
ovmf_table_len = tot_len;
|
||||
|
||||
/*
|
||||
* ptr is the foot of the table, so copy it all to the newly
|
||||
* allocated ovmf_table and then set the ovmf_table pointer
|
||||
* to the table foot
|
||||
*/
|
||||
memcpy(ovmf_table, ptr - tot_len, tot_len);
|
||||
ovmf_table += tot_len;
|
||||
}
|
||||
|
||||
bool pc_system_ovmf_table_find(const char *entry, uint8_t **data,
|
||||
int *data_len)
|
||||
{
|
||||
uint8_t *ptr = ovmf_table;
|
||||
int tot_len = ovmf_table_len;
|
||||
QemuUUID entry_guid;
|
||||
|
||||
if (qemu_uuid_parse(entry, &entry_guid) < 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!ptr) {
|
||||
return false;
|
||||
}
|
||||
|
||||
entry_guid = qemu_uuid_bswap(entry_guid); /* guids are LE */
|
||||
while (tot_len >= sizeof(QemuUUID) + sizeof(uint16_t)) {
|
||||
int len;
|
||||
QemuUUID *guid;
|
||||
|
||||
/*
|
||||
* The data structure is
|
||||
* arbitrary length data
|
||||
* 2 byte length of entire entry
|
||||
* 16 byte guid
|
||||
*/
|
||||
guid = (QemuUUID *)(ptr - sizeof(QemuUUID));
|
||||
len = le16_to_cpu(*(uint16_t *)(ptr - sizeof(QemuUUID) -
|
||||
sizeof(uint16_t)));
|
||||
|
||||
/*
|
||||
* just in case the table is corrupt, wouldn't want to spin in
|
||||
* the zero case
|
||||
*/
|
||||
if (len < sizeof(QemuUUID) + sizeof(uint16_t)) {
|
||||
return false;
|
||||
} else if (len > tot_len) {
|
||||
return false;
|
||||
}
|
||||
|
||||
ptr -= len;
|
||||
tot_len -= len;
|
||||
if (qemu_uuid_is_equal(guid, &entry_guid)) {
|
||||
if (data) {
|
||||
*data = ptr;
|
||||
}
|
||||
if (data_len) {
|
||||
*data_len = len - sizeof(QemuUUID) - sizeof(uint16_t);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/*
|
||||
* Map the pcms->flash[] from 4GiB downward, and realize.
|
||||
* Map them in descending order, i.e. pcms->flash[0] at the top,
|
||||
@ -149,6 +256,7 @@ static void pc_system_flash_map(PCMachineState *pcms,
|
||||
MemoryRegion *flash_mem;
|
||||
void *flash_ptr;
|
||||
int flash_size;
|
||||
int ret;
|
||||
|
||||
assert(PC_MACHINE_GET_CLASS(pcms)->pci_enabled);
|
||||
|
||||
@ -192,10 +300,24 @@ static void pc_system_flash_map(PCMachineState *pcms,
|
||||
flash_mem = pflash_cfi01_get_memory(system_flash);
|
||||
pc_isa_bios_init(rom_memory, flash_mem, size);
|
||||
|
||||
/* Encrypt the pflash boot ROM, if necessary */
|
||||
flash_ptr = memory_region_get_ram_ptr(flash_mem);
|
||||
flash_size = memory_region_size(flash_mem);
|
||||
sev_encrypt_flash(flash_ptr, flash_size, &error_fatal);
|
||||
/* Encrypt the pflash boot ROM */
|
||||
if (sev_enabled()) {
|
||||
flash_ptr = memory_region_get_ram_ptr(flash_mem);
|
||||
flash_size = memory_region_size(flash_mem);
|
||||
/*
|
||||
* OVMF places a GUIDed structures in the flash, so
|
||||
* search for them
|
||||
*/
|
||||
pc_system_parse_ovmf_flash(flash_ptr, flash_size);
|
||||
|
||||
ret = sev_es_save_reset_vector(flash_ptr, flash_size);
|
||||
if (ret) {
|
||||
error_report("failed to locate and/or save reset vector");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
sev_encrypt_flash(flash_ptr, flash_size, &error_fatal);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -3,6 +3,7 @@
|
||||
|
||||
#include "qemu/notify.h"
|
||||
#include "qapi/qapi-types-common.h"
|
||||
#include "qemu/uuid.h"
|
||||
#include "hw/boards.h"
|
||||
#include "hw/block/fdc.h"
|
||||
#include "hw/block/flash.h"
|
||||
@ -191,6 +192,9 @@ ISADevice *pc_find_fdc0(void);
|
||||
void pc_system_flash_create(PCMachineState *pcms);
|
||||
void pc_system_flash_cleanup_unused(PCMachineState *pcms);
|
||||
void pc_system_firmware_init(PCMachineState *pcms, MemoryRegion *rom_memory);
|
||||
bool pc_system_ovmf_table_find(const char *entry, uint8_t **data,
|
||||
int *data_len);
|
||||
|
||||
|
||||
/* acpi-build.c */
|
||||
void pc_madt_cpu_entry(AcpiDeviceIf *adev, int uid,
|
||||
|
@ -41,6 +41,8 @@ extern int icount_align_option;
|
||||
/* Unblock cpu */
|
||||
void qemu_cpu_kick_self(void);
|
||||
|
||||
bool cpus_are_resettable(void);
|
||||
|
||||
void cpu_synchronize_all_states(void);
|
||||
void cpu_synchronize_all_post_reset(void);
|
||||
void cpu_synchronize_all_post_init(void);
|
||||
|
@ -22,4 +22,9 @@ void cpu_synchronize_post_reset(CPUState *cpu);
|
||||
void cpu_synchronize_post_init(CPUState *cpu);
|
||||
void cpu_synchronize_pre_loadvm(CPUState *cpu);
|
||||
|
||||
static inline bool cpu_check_are_resettable(void)
|
||||
{
|
||||
return kvm_enabled() ? kvm_cpu_check_are_resettable() : true;
|
||||
}
|
||||
|
||||
#endif /* QEMU_HW_ACCEL_H */
|
||||
|
@ -541,4 +541,14 @@ int kvm_get_max_memslots(void);
|
||||
/* Notify resamplefd for EOI of specific interrupts. */
|
||||
void kvm_resample_fd_notify(int gsi);
|
||||
|
||||
/**
|
||||
* kvm_cpu_check_are_resettable - return whether CPUs can be reset
|
||||
*
|
||||
* Returns: true: CPUs are resettable
|
||||
* false: CPUs are not resettable
|
||||
*/
|
||||
bool kvm_cpu_check_are_resettable(void);
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void);
|
||||
|
||||
#endif
|
||||
|
@ -128,18 +128,20 @@ bool replay_has_interrupt(void);
|
||||
int64_t replay_save_clock(ReplayClockKind kind, int64_t clock,
|
||||
int64_t raw_icount);
|
||||
/*! Read the specified clock from the log or return cached data */
|
||||
int64_t replay_read_clock(ReplayClockKind kind);
|
||||
int64_t replay_read_clock(ReplayClockKind kind, int64_t raw_icount);
|
||||
/*! Saves or reads the clock depending on the current replay mode. */
|
||||
#define REPLAY_CLOCK(clock, value) \
|
||||
(replay_mode == REPLAY_MODE_PLAY ? replay_read_clock((clock)) \
|
||||
(replay_mode == REPLAY_MODE_PLAY \
|
||||
? replay_read_clock((clock), icount_get_raw()) \
|
||||
: replay_mode == REPLAY_MODE_RECORD \
|
||||
? replay_save_clock((clock), (value), icount_get_raw()) \
|
||||
: (value))
|
||||
? replay_save_clock((clock), (value), icount_get_raw()) \
|
||||
: (value))
|
||||
#define REPLAY_CLOCK_LOCKED(clock, value) \
|
||||
(replay_mode == REPLAY_MODE_PLAY ? replay_read_clock((clock)) \
|
||||
(replay_mode == REPLAY_MODE_PLAY \
|
||||
? replay_read_clock((clock), icount_get_raw_locked()) \
|
||||
: replay_mode == REPLAY_MODE_RECORD \
|
||||
? replay_save_clock((clock), (value), icount_get_raw_locked()) \
|
||||
: (value))
|
||||
: (value))
|
||||
|
||||
/* Processing data from random generators */
|
||||
|
||||
|
@ -16,8 +16,13 @@
|
||||
|
||||
#include "sysemu/kvm.h"
|
||||
|
||||
bool sev_enabled(void);
|
||||
int sev_kvm_init(ConfidentialGuestSupport *cgs, Error **errp);
|
||||
int sev_encrypt_flash(uint8_t *ptr, uint64_t len, Error **errp);
|
||||
int sev_inject_launch_secret(const char *hdr, const char *secret,
|
||||
uint64_t gpa, Error **errp);
|
||||
|
||||
int sev_es_save_reset_vector(void *flash_ptr, uint64_t flash_size);
|
||||
void sev_es_set_reset_vector(CPUState *cpu);
|
||||
|
||||
#endif
|
||||
|
@ -216,7 +216,7 @@
|
||||
#
|
||||
##
|
||||
{ 'command': 'sev-inject-launch-secret',
|
||||
'data': { 'packet-header': 'str', 'secret': 'str', 'gpa': 'uint64' },
|
||||
'data': { 'packet-header': 'str', 'secret': 'str', '*gpa': 'uint64' },
|
||||
'if': 'defined(TARGET_I386)' }
|
||||
|
||||
##
|
||||
|
@ -247,10 +247,31 @@ void replay_advance_current_icount(uint64_t current_icount)
|
||||
/* Time can only go forward */
|
||||
assert(diff >= 0);
|
||||
|
||||
if (diff > 0) {
|
||||
replay_put_event(EVENT_INSTRUCTION);
|
||||
replay_put_dword(diff);
|
||||
replay_state.current_icount += diff;
|
||||
if (replay_mode == REPLAY_MODE_RECORD) {
|
||||
if (diff > 0) {
|
||||
replay_put_event(EVENT_INSTRUCTION);
|
||||
replay_put_dword(diff);
|
||||
replay_state.current_icount += diff;
|
||||
}
|
||||
} else if (replay_mode == REPLAY_MODE_PLAY) {
|
||||
if (diff > 0) {
|
||||
replay_state.instruction_count -= diff;
|
||||
replay_state.current_icount += diff;
|
||||
if (replay_state.instruction_count == 0) {
|
||||
assert(replay_state.data_kind == EVENT_INSTRUCTION);
|
||||
replay_finish_event();
|
||||
/* Wake up iothread. This is required because
|
||||
timers will not expire until clock counters
|
||||
will be read from the log. */
|
||||
qemu_notify_event();
|
||||
}
|
||||
}
|
||||
/* Execution reached the break step */
|
||||
if (replay_break_icount == replay_state.current_icount) {
|
||||
/* Cannot make callback directly from the vCPU thread */
|
||||
timer_mod_ns(replay_break_timer,
|
||||
qemu_clock_get_ns(QEMU_CLOCK_REALTIME));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -46,12 +46,12 @@ void replay_read_next_clock(ReplayClockKind kind)
|
||||
}
|
||||
|
||||
/*! Reads next clock event from the input. */
|
||||
int64_t replay_read_clock(ReplayClockKind kind)
|
||||
int64_t replay_read_clock(ReplayClockKind kind, int64_t raw_icount)
|
||||
{
|
||||
int64_t ret;
|
||||
g_assert(replay_file && replay_mutex_locked());
|
||||
|
||||
replay_account_executed_instructions();
|
||||
replay_advance_current_icount(raw_icount);
|
||||
|
||||
if (replay_next_event_is(EVENT_CLOCK + kind)) {
|
||||
replay_read_next_clock(kind);
|
||||
|
@ -94,28 +94,7 @@ void replay_account_executed_instructions(void)
|
||||
if (replay_mode == REPLAY_MODE_PLAY) {
|
||||
g_assert(replay_mutex_locked());
|
||||
if (replay_state.instruction_count > 0) {
|
||||
int count = (int)(replay_get_current_icount()
|
||||
- replay_state.current_icount);
|
||||
|
||||
/* Time can only go forward */
|
||||
assert(count >= 0);
|
||||
|
||||
replay_state.instruction_count -= count;
|
||||
replay_state.current_icount += count;
|
||||
if (replay_state.instruction_count == 0) {
|
||||
assert(replay_state.data_kind == EVENT_INSTRUCTION);
|
||||
replay_finish_event();
|
||||
/* Wake up iothread. This is required because
|
||||
timers will not expire until clock counters
|
||||
will be read from the log. */
|
||||
qemu_notify_event();
|
||||
}
|
||||
/* Execution reached the break step */
|
||||
if (replay_break_icount == replay_state.current_icount) {
|
||||
/* Cannot make callback directly from the vCPU thread */
|
||||
timer_mod_ns(replay_break_timer,
|
||||
qemu_clock_get_ns(QEMU_CLOCK_REALTIME));
|
||||
}
|
||||
replay_advance_current_icount(replay_get_current_icount());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -194,6 +194,11 @@ void cpu_synchronize_pre_loadvm(CPUState *cpu)
|
||||
}
|
||||
}
|
||||
|
||||
bool cpus_are_resettable(void)
|
||||
{
|
||||
return cpu_check_are_resettable();
|
||||
}
|
||||
|
||||
int64_t cpus_get_virtual_clock(void)
|
||||
{
|
||||
/*
|
||||
|
@ -528,6 +528,9 @@ void qemu_system_reset_request(ShutdownCause reason)
|
||||
if (reboot_action == REBOOT_ACTION_SHUTDOWN &&
|
||||
reason != SHUTDOWN_CAUSE_SUBSYSTEM_RESET) {
|
||||
shutdown_requested = reason;
|
||||
} else if (!cpus_are_resettable()) {
|
||||
error_report("cpus are not resettable, terminating");
|
||||
shutdown_requested = reason;
|
||||
} else {
|
||||
reset_requested = reason;
|
||||
}
|
||||
|
@ -13,7 +13,7 @@ int64_t replay_save_clock(unsigned int kind, int64_t clock, int64_t raw_icount)
|
||||
return 0;
|
||||
}
|
||||
|
||||
int64_t replay_read_clock(unsigned int kind)
|
||||
int64_t replay_read_clock(unsigned int kind, int64_t raw_icount)
|
||||
{
|
||||
abort();
|
||||
return 0;
|
||||
|
@ -1045,3 +1045,8 @@ int kvm_arch_msi_data_to_gsi(uint32_t data)
|
||||
{
|
||||
return (data - 32) & 0xffff;
|
||||
}
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
@ -5984,6 +5984,7 @@ void cpu_x86_cpuid(CPUX86State *env, uint32_t index, uint32_t count,
|
||||
break;
|
||||
case 0x8000001F:
|
||||
*eax = sev_enabled() ? 0x2 : 0;
|
||||
*eax |= sev_es_enabled() ? 0x8 : 0;
|
||||
*ebx = sev_get_cbit_position();
|
||||
*ebx |= sev_get_reduced_phys_bits() << 6;
|
||||
*ecx = 0;
|
||||
|
@ -368,6 +368,7 @@ typedef enum X86Seg {
|
||||
|
||||
#define MSR_IA32_SMBASE 0x9e
|
||||
#define MSR_SMI_COUNT 0x34
|
||||
#define MSR_CORE_THREAD_COUNT 0x35
|
||||
#define MSR_MTRRcap 0xfe
|
||||
#define MSR_MTRRcap_VCNT 8
|
||||
#define MSR_MTRRcap_FIXRANGE_SUPPORT (1 << 8)
|
||||
|
@ -21,21 +21,6 @@
|
||||
#include "cpu.h"
|
||||
#include "x86.h"
|
||||
|
||||
#define HVF_MAX_VCPU 0x10
|
||||
|
||||
extern struct hvf_state hvf_global;
|
||||
|
||||
struct hvf_vm {
|
||||
int id;
|
||||
struct hvf_vcpu_state *vcpus[HVF_MAX_VCPU];
|
||||
};
|
||||
|
||||
struct hvf_state {
|
||||
uint32_t version;
|
||||
struct hvf_vm *vm;
|
||||
uint64_t mem_quota;
|
||||
};
|
||||
|
||||
/* hvf_slot flags */
|
||||
#define HVF_SLOT_LOG (1 << 0)
|
||||
|
||||
@ -75,7 +60,6 @@ hvf_slot *hvf_find_overlap_slot(uint64_t, uint64_t);
|
||||
|
||||
/* Host specific functions */
|
||||
int hvf_inject_interrupt(CPUArchState *env, int vector);
|
||||
int hvf_vcpu_run(struct hvf_vcpu_state *vcpu);
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
@ -65,6 +65,7 @@
|
||||
|
||||
#include <Hypervisor/hv.h>
|
||||
#include <Hypervisor/hv_vmx.h>
|
||||
#include <sys/sysctl.h>
|
||||
|
||||
#include "exec/address-spaces.h"
|
||||
#include "hw/i386/apic_internal.h"
|
||||
@ -456,6 +457,48 @@ static void dummy_signal(int sig)
|
||||
{
|
||||
}
|
||||
|
||||
static void init_tsc_freq(CPUX86State *env)
|
||||
{
|
||||
size_t length;
|
||||
uint64_t tsc_freq;
|
||||
|
||||
if (env->tsc_khz != 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
length = sizeof(uint64_t);
|
||||
if (sysctlbyname("machdep.tsc.frequency", &tsc_freq, &length, NULL, 0)) {
|
||||
return;
|
||||
}
|
||||
env->tsc_khz = tsc_freq / 1000; /* Hz to KHz */
|
||||
}
|
||||
|
||||
static void init_apic_bus_freq(CPUX86State *env)
|
||||
{
|
||||
size_t length;
|
||||
uint64_t bus_freq;
|
||||
|
||||
if (env->apic_bus_freq != 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
length = sizeof(uint64_t);
|
||||
if (sysctlbyname("hw.busfrequency", &bus_freq, &length, NULL, 0)) {
|
||||
return;
|
||||
}
|
||||
env->apic_bus_freq = bus_freq;
|
||||
}
|
||||
|
||||
static inline bool tsc_is_known(CPUX86State *env)
|
||||
{
|
||||
return env->tsc_khz != 0;
|
||||
}
|
||||
|
||||
static inline bool apic_bus_freq_is_known(CPUX86State *env)
|
||||
{
|
||||
return env->apic_bus_freq != 0;
|
||||
}
|
||||
|
||||
int hvf_init_vcpu(CPUState *cpu)
|
||||
{
|
||||
|
||||
@ -480,6 +523,15 @@ int hvf_init_vcpu(CPUState *cpu)
|
||||
hvf_state->hvf_caps = g_new0(struct hvf_vcpu_caps, 1);
|
||||
env->hvf_mmio_buf = g_new(char, 4096);
|
||||
|
||||
if (x86cpu->vmware_cpuid_freq) {
|
||||
init_tsc_freq(env);
|
||||
init_apic_bus_freq(env);
|
||||
|
||||
if (!tsc_is_known(env) || !apic_bus_freq_is_known(env)) {
|
||||
error_report("vmware-cpuid-freq: feature couldn't be enabled");
|
||||
}
|
||||
}
|
||||
|
||||
r = hv_vcpu_create((hv_vcpuid_t *)&cpu->hvf_fd, HV_VCPU_DEFAULT);
|
||||
cpu->vcpu_dirty = 1;
|
||||
assert_hvf_ok(r);
|
||||
@ -597,6 +649,48 @@ static void hvf_store_events(CPUState *cpu, uint32_t ins_len, uint64_t idtvec_in
|
||||
}
|
||||
}
|
||||
|
||||
static void hvf_cpu_x86_cpuid(CPUX86State *env, uint32_t index, uint32_t count,
|
||||
uint32_t *eax, uint32_t *ebx,
|
||||
uint32_t *ecx, uint32_t *edx)
|
||||
{
|
||||
/*
|
||||
* A wrapper extends cpu_x86_cpuid with 0x40000000 and 0x40000010 leafs,
|
||||
* leafs 0x40000001-0x4000000F are filled with zeros
|
||||
* Provides vmware-cpuid-freq support to hvf
|
||||
*
|
||||
* Note: leaf 0x40000000 not exposes HVF,
|
||||
* leaving hypervisor signature empty
|
||||
*/
|
||||
|
||||
if (index < 0x40000000 || index > 0x40000010 ||
|
||||
!tsc_is_known(env) || !apic_bus_freq_is_known(env)) {
|
||||
|
||||
cpu_x86_cpuid(env, index, count, eax, ebx, ecx, edx);
|
||||
return;
|
||||
}
|
||||
|
||||
switch (index) {
|
||||
case 0x40000000:
|
||||
*eax = 0x40000010; /* Max available cpuid leaf */
|
||||
*ebx = 0; /* Leave signature empty */
|
||||
*ecx = 0;
|
||||
*edx = 0;
|
||||
break;
|
||||
case 0x40000010:
|
||||
*eax = env->tsc_khz;
|
||||
*ebx = env->apic_bus_freq / 1000; /* Hz to KHz */
|
||||
*ecx = 0;
|
||||
*edx = 0;
|
||||
break;
|
||||
default:
|
||||
*eax = 0;
|
||||
*ebx = 0;
|
||||
*ecx = 0;
|
||||
*edx = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int hvf_vcpu_exec(CPUState *cpu)
|
||||
{
|
||||
X86CPU *x86_cpu = X86_CPU(cpu);
|
||||
@ -734,7 +828,11 @@ int hvf_vcpu_exec(CPUState *cpu)
|
||||
uint32_t rcx = (uint32_t)rreg(cpu->hvf_fd, HV_X86_RCX);
|
||||
uint32_t rdx = (uint32_t)rreg(cpu->hvf_fd, HV_X86_RDX);
|
||||
|
||||
cpu_x86_cpuid(env, rax, rcx, &rax, &rbx, &rcx, &rdx);
|
||||
if (rax == 1) {
|
||||
/* CPUID1.ecx.OSXSAVE needs to know CR4 */
|
||||
env->cr[4] = rvmcs(cpu->hvf_fd, VMCS_GUEST_CR4);
|
||||
}
|
||||
hvf_cpu_x86_cpuid(env, rax, rcx, &rax, &rbx, &rcx, &rdx);
|
||||
|
||||
wreg(cpu->hvf_fd, HV_X86_RAX, rax);
|
||||
wreg(cpu->hvf_fd, HV_X86_RBX, rbx);
|
||||
|
@ -27,15 +27,22 @@
|
||||
#include "vmx.h"
|
||||
#include "sysemu/hvf.h"
|
||||
|
||||
static uint64_t xgetbv(uint32_t xcr)
|
||||
static bool xgetbv(uint32_t cpuid_ecx, uint32_t idx, uint64_t *xcr)
|
||||
{
|
||||
uint32_t eax, edx;
|
||||
uint32_t xcrl, xcrh;
|
||||
|
||||
__asm__ volatile ("xgetbv"
|
||||
: "=a" (eax), "=d" (edx)
|
||||
: "c" (xcr));
|
||||
if (cpuid_ecx & CPUID_EXT_OSXSAVE) {
|
||||
/*
|
||||
* The xgetbv instruction is not available to older versions of
|
||||
* the assembler, so we encode the instruction manually.
|
||||
*/
|
||||
asm(".byte 0x0f, 0x01, 0xd0" : "=a" (xcrl), "=d" (xcrh) : "c" (idx));
|
||||
|
||||
return (((uint64_t)edx) << 32) | eax;
|
||||
*xcr = (((uint64_t)xcrh) << 32) | xcrl;
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
uint32_t hvf_get_supported_cpuid(uint32_t func, uint32_t idx,
|
||||
@ -100,12 +107,15 @@ uint32_t hvf_get_supported_cpuid(uint32_t func, uint32_t idx,
|
||||
break;
|
||||
case 0xD:
|
||||
if (idx == 0) {
|
||||
uint64_t host_xcr0 = xgetbv(0);
|
||||
uint64_t supp_xcr0 = host_xcr0 & (XSTATE_FP_MASK | XSTATE_SSE_MASK |
|
||||
XSTATE_YMM_MASK | XSTATE_BNDREGS_MASK |
|
||||
XSTATE_BNDCSR_MASK | XSTATE_OPMASK_MASK |
|
||||
XSTATE_ZMM_Hi256_MASK | XSTATE_Hi16_ZMM_MASK);
|
||||
eax &= supp_xcr0;
|
||||
uint64_t host_xcr0;
|
||||
if (xgetbv(ecx, 0, &host_xcr0)) {
|
||||
uint64_t supp_xcr0 = host_xcr0 & (XSTATE_FP_MASK |
|
||||
XSTATE_SSE_MASK | XSTATE_YMM_MASK |
|
||||
XSTATE_BNDREGS_MASK | XSTATE_BNDCSR_MASK |
|
||||
XSTATE_OPMASK_MASK | XSTATE_ZMM_Hi256_MASK |
|
||||
XSTATE_Hi16_ZMM_MASK);
|
||||
eax &= supp_xcr0;
|
||||
}
|
||||
} else if (idx == 1) {
|
||||
hv_vmx_read_capability(HV_VMX_CAP_PROCBASED2, &cap);
|
||||
eax &= CPUID_XSAVE_XSAVEOPT | CPUID_XSAVE_XGETBV1;
|
||||
|
@ -668,6 +668,7 @@ void simulate_rdmsr(struct CPUState *cpu)
|
||||
{
|
||||
X86CPU *x86_cpu = X86_CPU(cpu);
|
||||
CPUX86State *env = &x86_cpu->env;
|
||||
CPUState *cs = env_cpu(env);
|
||||
uint32_t msr = ECX(env);
|
||||
uint64_t val = 0;
|
||||
|
||||
@ -745,6 +746,10 @@ void simulate_rdmsr(struct CPUState *cpu)
|
||||
case MSR_MTRRdefType:
|
||||
val = env->mtrr_deftype;
|
||||
break;
|
||||
case MSR_CORE_THREAD_COUNT:
|
||||
val = cs->nr_threads * cs->nr_cores; /* thread count, bits 15..0 */
|
||||
val |= ((uint32_t)cs->nr_cores << 16); /* core count, bits 31..16 */
|
||||
break;
|
||||
default:
|
||||
/* fprintf(stderr, "%s: unknown msr 0x%x\n", __func__, msr); */
|
||||
val = 0;
|
||||
|
@ -27,6 +27,7 @@
|
||||
#include "sysemu/kvm_int.h"
|
||||
#include "sysemu/runstate.h"
|
||||
#include "kvm_i386.h"
|
||||
#include "sev_i386.h"
|
||||
#include "hyperv.h"
|
||||
#include "hyperv-proto.h"
|
||||
|
||||
@ -136,7 +137,7 @@ int kvm_has_pit_state2(void)
|
||||
|
||||
bool kvm_has_smm(void)
|
||||
{
|
||||
return kvm_check_extension(kvm_state, KVM_CAP_X86_SMM);
|
||||
return kvm_vm_check_extension(kvm_state, KVM_CAP_X86_SMM);
|
||||
}
|
||||
|
||||
bool kvm_has_adjust_clock_stable(void)
|
||||
@ -1922,6 +1923,8 @@ void kvm_arch_reset_vcpu(X86CPU *cpu)
|
||||
}
|
||||
/* enabled by default */
|
||||
env->poll_control_msr = 1;
|
||||
|
||||
sev_es_set_reset_vector(CPU(cpu));
|
||||
}
|
||||
|
||||
void kvm_arch_do_init_vcpu(X86CPU *cpu)
|
||||
@ -4819,3 +4822,8 @@ bool kvm_has_waitpkg(void)
|
||||
{
|
||||
return has_msr_umwait;
|
||||
}
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void)
|
||||
{
|
||||
return !sev_es_enabled();
|
||||
}
|
||||
|
@ -34,6 +34,7 @@
|
||||
#include "sev_i386.h"
|
||||
#include "qapi/qapi-commands-misc-target.h"
|
||||
#include "qapi/qapi-commands-misc.h"
|
||||
#include "hw/i386/pc.h"
|
||||
|
||||
/* Perform linear address sign extension */
|
||||
static hwaddr addr_canonical(CPUArchState *env, hwaddr addr)
|
||||
@ -730,9 +731,29 @@ SevCapability *qmp_query_sev_capabilities(Error **errp)
|
||||
return sev_get_capabilities(errp);
|
||||
}
|
||||
|
||||
#define SEV_SECRET_GUID "4c2eb361-7d9b-4cc3-8081-127c90d3d294"
|
||||
struct sev_secret_area {
|
||||
uint32_t base;
|
||||
uint32_t size;
|
||||
};
|
||||
|
||||
void qmp_sev_inject_launch_secret(const char *packet_hdr,
|
||||
const char *secret, uint64_t gpa,
|
||||
const char *secret,
|
||||
bool has_gpa, uint64_t gpa,
|
||||
Error **errp)
|
||||
{
|
||||
if (!has_gpa) {
|
||||
uint8_t *data;
|
||||
struct sev_secret_area *area;
|
||||
|
||||
if (!pc_system_ovmf_table_find(SEV_SECRET_GUID, &data, NULL)) {
|
||||
error_setg(errp, "SEV: no secret area found in OVMF,"
|
||||
" gpa must be specified.");
|
||||
return;
|
||||
}
|
||||
area = (struct sev_secret_area *)data;
|
||||
gpa = area->base;
|
||||
}
|
||||
|
||||
sev_inject_launch_secret(packet_hdr, secret, gpa, errp);
|
||||
}
|
||||
|
@ -49,6 +49,7 @@ SevCapability *sev_get_capabilities(Error **errp)
|
||||
error_setg(errp, "SEV is not available in this QEMU");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int sev_inject_launch_secret(const char *hdr, const char *secret,
|
||||
uint64_t gpa, Error **errp)
|
||||
{
|
||||
@ -59,3 +60,17 @@ int sev_encrypt_flash(uint8_t *ptr, uint64_t len, Error **errp)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool sev_es_enabled(void)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
void sev_es_set_reset_vector(CPUState *cpu)
|
||||
{
|
||||
}
|
||||
|
||||
int sev_es_save_reset_vector(void *flash_ptr, uint64_t flash_size)
|
||||
{
|
||||
abort();
|
||||
}
|
||||
|
@ -22,6 +22,7 @@
|
||||
#include "qom/object_interfaces.h"
|
||||
#include "qemu/base64.h"
|
||||
#include "qemu/module.h"
|
||||
#include "qemu/uuid.h"
|
||||
#include "sysemu/kvm.h"
|
||||
#include "sev_i386.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
@ -32,6 +33,7 @@
|
||||
#include "exec/address-spaces.h"
|
||||
#include "monitor/monitor.h"
|
||||
#include "exec/confidential-guest-support.h"
|
||||
#include "hw/i386/pc.h"
|
||||
|
||||
#define TYPE_SEV_GUEST "sev-guest"
|
||||
OBJECT_DECLARE_SIMPLE_TYPE(SevGuestState, SEV_GUEST)
|
||||
@ -67,11 +69,21 @@ struct SevGuestState {
|
||||
int sev_fd;
|
||||
SevState state;
|
||||
gchar *measurement;
|
||||
|
||||
uint32_t reset_cs;
|
||||
uint32_t reset_ip;
|
||||
bool reset_data_valid;
|
||||
};
|
||||
|
||||
#define DEFAULT_GUEST_POLICY 0x1 /* disable debug */
|
||||
#define DEFAULT_SEV_DEVICE "/dev/sev"
|
||||
|
||||
#define SEV_INFO_BLOCK_GUID "00f771de-1a7e-4fcb-890e-68c77e2fb44e"
|
||||
typedef struct __attribute__((__packed__)) SevInfoBlock {
|
||||
/* SEV-ES Reset Vector Address */
|
||||
uint32_t reset_addr;
|
||||
} SevInfoBlock;
|
||||
|
||||
static SevGuestState *sev_guest;
|
||||
static Error *sev_mig_blocker;
|
||||
|
||||
@ -341,6 +353,12 @@ sev_enabled(void)
|
||||
return !!sev_guest;
|
||||
}
|
||||
|
||||
bool
|
||||
sev_es_enabled(void)
|
||||
{
|
||||
return sev_enabled() && (sev_guest->policy & SEV_POLICY_ES);
|
||||
}
|
||||
|
||||
uint64_t
|
||||
sev_get_me_mask(void)
|
||||
{
|
||||
@ -561,6 +579,20 @@ sev_launch_update_data(SevGuestState *sev, uint8_t *addr, uint64_t len)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
sev_launch_update_vmsa(SevGuestState *sev)
|
||||
{
|
||||
int ret, fw_error;
|
||||
|
||||
ret = sev_ioctl(sev->sev_fd, KVM_SEV_LAUNCH_UPDATE_VMSA, NULL, &fw_error);
|
||||
if (ret) {
|
||||
error_report("%s: LAUNCH_UPDATE_VMSA ret=%d fw_error=%d '%s'",
|
||||
__func__, ret, fw_error, fw_error_to_str(fw_error));
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
sev_launch_get_measure(Notifier *notifier, void *unused)
|
||||
{
|
||||
@ -573,6 +605,14 @@ sev_launch_get_measure(Notifier *notifier, void *unused)
|
||||
return;
|
||||
}
|
||||
|
||||
if (sev_es_enabled()) {
|
||||
/* measure all the VM save areas before getting launch_measure */
|
||||
ret = sev_launch_update_vmsa(sev);
|
||||
if (ret) {
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
measurement = g_new0(struct kvm_sev_launch_measure, 1);
|
||||
|
||||
/* query the measurement blob length */
|
||||
@ -667,7 +707,7 @@ int sev_kvm_init(ConfidentialGuestSupport *cgs, Error **errp)
|
||||
SevGuestState *sev
|
||||
= (SevGuestState *)object_dynamic_cast(OBJECT(cgs), TYPE_SEV_GUEST);
|
||||
char *devname;
|
||||
int ret, fw_error;
|
||||
int ret, fw_error, cmd;
|
||||
uint32_t ebx;
|
||||
uint32_t host_cbitpos;
|
||||
struct sev_user_data_status status = {};
|
||||
@ -724,8 +764,26 @@ int sev_kvm_init(ConfidentialGuestSupport *cgs, Error **errp)
|
||||
sev->api_major = status.api_major;
|
||||
sev->api_minor = status.api_minor;
|
||||
|
||||
if (sev_es_enabled()) {
|
||||
if (!kvm_kernel_irqchip_allowed()) {
|
||||
error_report("%s: SEV-ES guests require in-kernel irqchip support",
|
||||
__func__);
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (!(status.flags & SEV_STATUS_FLAGS_CONFIG_ES)) {
|
||||
error_report("%s: guest policy requires SEV-ES, but "
|
||||
"host SEV-ES support unavailable",
|
||||
__func__);
|
||||
goto err;
|
||||
}
|
||||
cmd = KVM_SEV_ES_INIT;
|
||||
} else {
|
||||
cmd = KVM_SEV_INIT;
|
||||
}
|
||||
|
||||
trace_kvm_sev_init();
|
||||
ret = sev_ioctl(sev->sev_fd, KVM_SEV_INIT, NULL, &fw_error);
|
||||
ret = sev_ioctl(sev->sev_fd, cmd, NULL, &fw_error);
|
||||
if (ret) {
|
||||
error_setg(errp, "%s: failed to initialize ret=%d fw_error=%d '%s'",
|
||||
__func__, ret, fw_error, fw_error_to_str(fw_error));
|
||||
@ -833,6 +891,122 @@ int sev_inject_launch_secret(const char *packet_hdr, const char *secret,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
sev_es_parse_reset_block(SevInfoBlock *info, uint32_t *addr)
|
||||
{
|
||||
if (!info->reset_addr) {
|
||||
error_report("SEV-ES reset address is zero");
|
||||
return 1;
|
||||
}
|
||||
|
||||
*addr = info->reset_addr;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
sev_es_find_reset_vector(void *flash_ptr, uint64_t flash_size,
|
||||
uint32_t *addr)
|
||||
{
|
||||
QemuUUID info_guid, *guid;
|
||||
SevInfoBlock *info;
|
||||
uint8_t *data;
|
||||
uint16_t *len;
|
||||
|
||||
/*
|
||||
* Initialize the address to zero. An address of zero with a successful
|
||||
* return code indicates that SEV-ES is not active.
|
||||
*/
|
||||
*addr = 0;
|
||||
|
||||
/*
|
||||
* Extract the AP reset vector for SEV-ES guests by locating the SEV GUID.
|
||||
* The SEV GUID is located on its own (original implementation) or within
|
||||
* the Firmware GUID Table (new implementation), either of which are
|
||||
* located 32 bytes from the end of the flash.
|
||||
*
|
||||
* Check the Firmware GUID Table first.
|
||||
*/
|
||||
if (pc_system_ovmf_table_find(SEV_INFO_BLOCK_GUID, &data, NULL)) {
|
||||
return sev_es_parse_reset_block((SevInfoBlock *)data, addr);
|
||||
}
|
||||
|
||||
/*
|
||||
* SEV info block not found in the Firmware GUID Table (or there isn't
|
||||
* a Firmware GUID Table), fall back to the original implementation.
|
||||
*/
|
||||
data = flash_ptr + flash_size - 0x20;
|
||||
|
||||
qemu_uuid_parse(SEV_INFO_BLOCK_GUID, &info_guid);
|
||||
info_guid = qemu_uuid_bswap(info_guid); /* GUIDs are LE */
|
||||
|
||||
guid = (QemuUUID *)(data - sizeof(info_guid));
|
||||
if (!qemu_uuid_is_equal(guid, &info_guid)) {
|
||||
error_report("SEV information block/Firmware GUID Table block not found in pflash rom");
|
||||
return 1;
|
||||
}
|
||||
|
||||
len = (uint16_t *)((uint8_t *)guid - sizeof(*len));
|
||||
info = (SevInfoBlock *)(data - le16_to_cpu(*len));
|
||||
|
||||
return sev_es_parse_reset_block(info, addr);
|
||||
}
|
||||
|
||||
void sev_es_set_reset_vector(CPUState *cpu)
|
||||
{
|
||||
X86CPU *x86;
|
||||
CPUX86State *env;
|
||||
|
||||
/* Only update if we have valid reset information */
|
||||
if (!sev_guest || !sev_guest->reset_data_valid) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* Do not update the BSP reset state */
|
||||
if (cpu->cpu_index == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
x86 = X86_CPU(cpu);
|
||||
env = &x86->env;
|
||||
|
||||
cpu_x86_load_seg_cache(env, R_CS, 0xf000, sev_guest->reset_cs, 0xffff,
|
||||
DESC_P_MASK | DESC_S_MASK | DESC_CS_MASK |
|
||||
DESC_R_MASK | DESC_A_MASK);
|
||||
|
||||
env->eip = sev_guest->reset_ip;
|
||||
}
|
||||
|
||||
int sev_es_save_reset_vector(void *flash_ptr, uint64_t flash_size)
|
||||
{
|
||||
CPUState *cpu;
|
||||
uint32_t addr;
|
||||
int ret;
|
||||
|
||||
if (!sev_es_enabled()) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
addr = 0;
|
||||
ret = sev_es_find_reset_vector(flash_ptr, flash_size,
|
||||
&addr);
|
||||
if (ret) {
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (addr) {
|
||||
sev_guest->reset_cs = addr & 0xffff0000;
|
||||
sev_guest->reset_ip = addr & 0x0000ffff;
|
||||
sev_guest->reset_data_valid = true;
|
||||
|
||||
CPU_FOREACH(cpu) {
|
||||
sev_es_set_reset_vector(cpu);
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
sev_register_types(void)
|
||||
{
|
||||
|
@ -28,7 +28,7 @@
|
||||
#define SEV_POLICY_DOMAIN 0x10
|
||||
#define SEV_POLICY_SEV 0x20
|
||||
|
||||
extern bool sev_enabled(void);
|
||||
extern bool sev_es_enabled(void);
|
||||
extern uint64_t sev_get_me_mask(void);
|
||||
extern SevInfo *sev_get_info(void);
|
||||
extern uint32_t sev_get_cbit_position(void);
|
||||
|
@ -1290,3 +1290,8 @@ int mips_kvm_type(MachineState *machine, const char *vm_type)
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
@ -2929,3 +2929,8 @@ void kvmppc_set_reg_tb_offset(PowerPCCPU *cpu, int64_t tb_offset)
|
||||
kvm_set_one_reg(cs, KVM_REG_PPC_TB_OFFSET, &tb_offset);
|
||||
}
|
||||
}
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
@ -2599,3 +2599,8 @@ void kvm_s390_stop_interrupt(S390CPU *cpu)
|
||||
|
||||
kvm_s390_vcpu_interrupt(cpu, &irq);
|
||||
}
|
||||
|
||||
bool kvm_arch_cpu_check_are_resettable(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
@ -153,6 +153,7 @@ static QOSGraphNode *create_node(const char *name, QOSNodeType type)
|
||||
static void destroy_node(void *val)
|
||||
{
|
||||
QOSGraphNode *node = val;
|
||||
g_free(node->qemu_name);
|
||||
g_free(node->command_line);
|
||||
g_free(node);
|
||||
}
|
||||
@ -286,7 +287,8 @@ static void build_machine_cmd_line(QOSGraphNode *node, const char *args)
|
||||
*/
|
||||
static void build_driver_cmd_line(QOSGraphNode *node)
|
||||
{
|
||||
node->command_line = g_strconcat(" -device ", node->name, NULL);
|
||||
const char *name = node->qemu_name ?: node->name;
|
||||
node->command_line = g_strconcat(" -device ", name, NULL);
|
||||
}
|
||||
|
||||
/* qos_print_cb(): callback prints all path found by the DFS algorithm. */
|
||||
@ -631,6 +633,15 @@ void qos_node_create_driver(const char *name, QOSCreateDriverFunc function)
|
||||
node->u.driver.constructor = function;
|
||||
}
|
||||
|
||||
void qos_node_create_driver_named(const char *name, const char *qemu_name,
|
||||
QOSCreateDriverFunc function)
|
||||
{
|
||||
QOSGraphNode *node = create_node(name, QNODE_DRIVER);
|
||||
node->qemu_name = g_strdup(qemu_name);
|
||||
build_driver_cmd_line(node);
|
||||
node->u.driver.constructor = function;
|
||||
}
|
||||
|
||||
void qos_node_contains(const char *container, const char *contained,
|
||||
QOSGraphEdgeOptions *opts, ...)
|
||||
{
|
||||
@ -663,7 +674,7 @@ void qos_node_consumes(const char *consumer, const char *interface,
|
||||
add_edge(interface, consumer, QEDGE_CONSUMED_BY, opts);
|
||||
}
|
||||
|
||||
void qos_graph_node_set_availability(const char *node, bool av)
|
||||
static void qos_graph_node_set_availability_explicit(const char *node, bool av)
|
||||
{
|
||||
QOSGraphEdgeList *elist;
|
||||
QOSGraphNode *n = search_node(node);
|
||||
@ -678,11 +689,48 @@ void qos_graph_node_set_availability(const char *node, bool av)
|
||||
}
|
||||
QSLIST_FOREACH_SAFE(e, elist, edge_list, next) {
|
||||
if (e->type == QEDGE_CONTAINS || e->type == QEDGE_PRODUCES) {
|
||||
qos_graph_node_set_availability(e->dest, av);
|
||||
qos_graph_node_set_availability_explicit(e->dest, av);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Behaves as qos_graph_node_set_availability_explicit(), except that the
|
||||
* former always matches by node name only, whereas this function matches both
|
||||
* by node name and node's optional 'qemu_name' field.
|
||||
*/
|
||||
void qos_graph_node_set_availability(const char *node, bool av)
|
||||
{
|
||||
GList *l;
|
||||
QOSGraphEdgeList *elist;
|
||||
QOSGraphEdge *e, *next;
|
||||
QOSGraphNode *n;
|
||||
GList *keys = g_hash_table_get_keys(node_table);
|
||||
|
||||
for (l = keys; l != NULL; l = l->next) {
|
||||
const gchar *key = l->data;
|
||||
n = g_hash_table_lookup(node_table, key);
|
||||
/*
|
||||
* node's 'qemu_name' is set if there is more than one device with
|
||||
* the same QEMU (QMP) device name
|
||||
*/
|
||||
const char *node_name = n->qemu_name ?: n->name;
|
||||
if (g_strcmp0(node_name, node) == 0) {
|
||||
n->available = av;
|
||||
elist = get_edgelist(n->name);
|
||||
if (elist) {
|
||||
QSLIST_FOREACH_SAFE(e, elist, edge_list, next) {
|
||||
if (e->type == QEDGE_CONTAINS || e->type == QEDGE_PRODUCES)
|
||||
{
|
||||
qos_graph_node_set_availability_explicit(e->dest, av);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
g_list_free(keys);
|
||||
}
|
||||
|
||||
void qos_graph_foreach_test_path(QOSTestCallback fn)
|
||||
{
|
||||
QOSGraphNode *root = qos_graph_get_node(QOS_ROOT);
|
||||
@ -757,3 +805,48 @@ void qos_delete_cmd_line(const char *name)
|
||||
node->command_line = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
void qos_dump_graph(void)
|
||||
{
|
||||
GList *keys;
|
||||
GList *l;
|
||||
QOSGraphEdgeList *list;
|
||||
QOSGraphEdge *e, *next;
|
||||
QOSGraphNode *dest_node, *node;
|
||||
|
||||
qos_printf("ALL QGRAPH EDGES: {\n");
|
||||
keys = g_hash_table_get_keys(edge_table);
|
||||
for (l = keys; l != NULL; l = l->next) {
|
||||
const gchar *key = l->data;
|
||||
qos_printf("\t src='%s'\n", key);
|
||||
list = get_edgelist(key);
|
||||
QSLIST_FOREACH_SAFE(e, list, edge_list, next) {
|
||||
dest_node = g_hash_table_lookup(node_table, e->dest);
|
||||
qos_printf("\t\t|-> dest='%s' type=%d (node=%p)",
|
||||
e->dest, e->type, dest_node);
|
||||
if (!dest_node) {
|
||||
qos_printf_literal(" <------- ERROR !");
|
||||
}
|
||||
qos_printf_literal("\n");
|
||||
}
|
||||
}
|
||||
g_list_free(keys);
|
||||
qos_printf("}\n");
|
||||
|
||||
qos_printf("ALL QGRAPH NODES: {\n");
|
||||
keys = g_hash_table_get_keys(node_table);
|
||||
for (l = keys; l != NULL; l = l->next) {
|
||||
const gchar *key = l->data;
|
||||
node = g_hash_table_lookup(node_table, key);
|
||||
qos_printf("\t name='%s' ", key);
|
||||
if (node->qemu_name) {
|
||||
qos_printf_literal("qemu_name='%s' ", node->qemu_name);
|
||||
}
|
||||
qos_printf_literal("type=%d cmd_line='%s' [%s]\n",
|
||||
node->type, node->command_line,
|
||||
node->available ? "available" : "UNAVAILBLE"
|
||||
);
|
||||
}
|
||||
g_list_free(keys);
|
||||
qos_printf("}\n");
|
||||
}
|
||||
|
@ -452,6 +452,22 @@ void qos_node_create_machine_args(const char *name,
|
||||
*/
|
||||
void qos_node_create_driver(const char *name, QOSCreateDriverFunc function);
|
||||
|
||||
/**
|
||||
* Behaves as qos_node_create_driver() with the extension of allowing to
|
||||
* specify a different node name vs. associated QEMU device name.
|
||||
*
|
||||
* Use this function instead of qos_node_create_driver() if you need to create
|
||||
* several instances of the same QEMU device. You are free to choose a custom
|
||||
* node name, however the chosen node name must always be unique.
|
||||
*
|
||||
* @param name: custom, unique name of the node to be created
|
||||
* @param qemu_name: actual (official) QEMU driver name the node shall be
|
||||
* associated with
|
||||
* @param function: driver constructor
|
||||
*/
|
||||
void qos_node_create_driver_named(const char *name, const char *qemu_name,
|
||||
QOSCreateDriverFunc function);
|
||||
|
||||
/**
|
||||
* qos_node_contains(): creates one or more edges of type QEDGE_CONTAINS
|
||||
* and adds them to the edge list mapped to @container in the
|
||||
@ -570,5 +586,25 @@ QOSGraphObject *qos_machine_new(QOSGraphNode *node, QTestState *qts);
|
||||
QOSGraphObject *qos_driver_new(QOSGraphNode *node, QOSGraphObject *parent,
|
||||
QGuestAllocator *alloc, void *arg);
|
||||
|
||||
/**
|
||||
* Just for debugging purpose: prints all currently existing nodes and
|
||||
* edges to stdout.
|
||||
*
|
||||
* All qtests add themselves to the overall qos graph by calling qgraph
|
||||
* functions that add device nodes and edges between the individual graph
|
||||
* nodes for tests. As the actual graph is assmbled at runtime by the qos
|
||||
* subsystem, it is sometimes not obvious how the overall graph looks like.
|
||||
* E.g. when writing new tests it may happen that those new tests are simply
|
||||
* ignored by the qtest framework.
|
||||
*
|
||||
* This function allows to identify problems in the created qgraph. Keep in
|
||||
* mind: only tests with a path down from the actual test case node (leaf) up
|
||||
* to the graph's root node are actually executed by the qtest framework. And
|
||||
* the qtest framework uses QMP to automatically check which QEMU drivers are
|
||||
* actually currently available, and accordingly qos marks certain pathes as
|
||||
* 'unavailable' in such cases (e.g. when QEMU was compiled without support for
|
||||
* a certain feature).
|
||||
*/
|
||||
void qos_dump_graph(void);
|
||||
|
||||
#endif
|
||||
|
@ -56,6 +56,7 @@ struct QOSGraphNode {
|
||||
bool available; /* set by QEMU via QMP, used during graph walk */
|
||||
bool visited; /* used during graph walk */
|
||||
char *name; /* used to identify the node */
|
||||
char *qemu_name; /* optional: see qos_node_create_driver_named() */
|
||||
char *command_line; /* used to start QEMU at test execution */
|
||||
union {
|
||||
struct {
|
||||
@ -254,4 +255,15 @@ void qos_delete_cmd_line(const char *name);
|
||||
*/
|
||||
void qos_graph_node_set_availability(const char *node, bool av);
|
||||
|
||||
/*
|
||||
* Prepends a '#' character in front for not breaking TAP output format.
|
||||
*/
|
||||
#define qos_printf(...) printf("# " __VA_ARGS__)
|
||||
|
||||
/*
|
||||
* Intended for printing something literally, i.e. for appending text as is
|
||||
* to a line already been started by qos_printf() before.
|
||||
*/
|
||||
#define qos_printf_literal printf
|
||||
|
||||
#endif
|
||||
|
@ -89,6 +89,9 @@ static void qos_set_machines_devices_available(void)
|
||||
|
||||
static void restart_qemu_or_continue(char *path)
|
||||
{
|
||||
if (g_test_verbose()) {
|
||||
qos_printf("Run QEMU with: '%s'\n", path);
|
||||
}
|
||||
/* compares the current command line with the
|
||||
* one previously executed: if they are the same,
|
||||
* don't restart QEMU, if they differ, stop previous
|
||||
@ -313,15 +316,25 @@ static void walk_path(QOSGraphNode *orig_path, int len)
|
||||
* machine/drivers/test objects
|
||||
* - Cleans up everything
|
||||
*/
|
||||
int main(int argc, char **argv)
|
||||
int main(int argc, char **argv, char** envp)
|
||||
{
|
||||
g_test_init(&argc, &argv, NULL);
|
||||
if (g_test_verbose()) {
|
||||
qos_printf("ENVIRONMENT VARIABLES: {\n");
|
||||
for (char **env = envp; *env != 0; env++) {
|
||||
qos_printf("\t%s\n", *env);
|
||||
}
|
||||
qos_printf("}\n");
|
||||
}
|
||||
qos_graph_init();
|
||||
module_call_init(MODULE_INIT_QOM);
|
||||
module_call_init(MODULE_INIT_LIBQOS);
|
||||
qos_set_machines_devices_available();
|
||||
|
||||
qos_graph_foreach_test_path(walk_path);
|
||||
if (g_test_verbose()) {
|
||||
qos_dump_graph();
|
||||
}
|
||||
g_test_run();
|
||||
qtest_end();
|
||||
qos_graph_destroy();
|
||||
|
@ -916,7 +916,8 @@ static inline bool starts_with_prefix(const char *dir)
|
||||
static inline const char *next_component(const char *dir, int *p_len)
|
||||
{
|
||||
int len;
|
||||
while (*dir && G_IS_DIR_SEPARATOR(*dir)) {
|
||||
while ((*dir && G_IS_DIR_SEPARATOR(*dir)) ||
|
||||
(*dir == '.' && (G_IS_DIR_SEPARATOR(dir[1]) || dir[1] == '\0'))) {
|
||||
dir++;
|
||||
}
|
||||
len = 0;
|
||||
|
@ -66,10 +66,10 @@ int event_notifier_init(EventNotifier *e, int active)
|
||||
e->rfd = fds[0];
|
||||
e->wfd = fds[1];
|
||||
}
|
||||
e->initialized = true;
|
||||
if (active) {
|
||||
event_notifier_set(e);
|
||||
}
|
||||
e->initialized = true;
|
||||
return 0;
|
||||
|
||||
fail:
|
||||
|
Loading…
Reference in New Issue
Block a user