x86 queue, 2016-10-17
-----BEGIN PGP SIGNATURE----- iQIcBAABCAAGBQJYBQ+LAAoJECgHk2+YTcWmaF0QAISVeb39vJyyXNhxXsy1Y5iM WSYA8Dym0TWCXTd7Fq7Ck4VS9ZC7hAREKNSBs2hgVPutecL56iB/IjWrB0AyFAMK u5y4H1pI6l9TiH+6GcDWwQjthM/0v2pEHzQ2udLWBSpKJGjDPTSQIafZUgrW2uu0 J/Drxg17FJ6KixqCg3FemPBXucbuU1PSW2qEWIgVElwj843j3d/Wc5l1wNb24irN jnOcvJd9WQsuT2fUDXezrCRVQle92tHR1cHtu5bZvC1aMFbGuHfSA4pm7pXw3l5N 8H0fhrCoj6JGKRY/pzHGmLgwMTWJL4qASxr6sEKkMAyu59DdjQ0+U8EhOwoAHYhp gSrNgpwPKRr2OKrSUJXil7w1cQ+hsokgEo44SDEgsV4k9Rgbz8VVVct+LwOxwfwW l9sC9L5ONheFODfB3rgVFiyAbspYxzwOvGZ29VoeMyb4CS1CUBrsvka8DledFi+m By26W6IMtXBa4NZoYqp49zHqUZ5Wu62I32LCaWDKscUQfaEJKrf1DtFQ9FlWhy5F 4NeSzTo4eAp3WPRDscbvXIyEJfYqzf7gs8KQA9QD+aTceDOIPiZeMz6oMokFukE8 Lt1fWzzppFJ6ZyPLO1YI/T91fOskl45r8b3T242fovOKGAlujkunpRcTAYaybN5C qUv5Qrq6ZujcRecupLfE =KM/h -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/ehabkost/tags/x86-pull-request' into staging x86 queue, 2016-10-17 # gpg: Signature made Mon 17 Oct 2016 18:51:07 BST # gpg: using RSA key 0x2807936F984DC5A6 # gpg: Good signature from "Eduardo Habkost <ehabkost@redhat.com>" # Primary key fingerprint: 5A32 2FD5 ABC4 D3DB ACCF D1AA 2807 936F 984D C5A6 * remotes/ehabkost/tags/x86-pull-request: (21 commits) target-i386: Don't use cpu->migratable when filtering features target-i386: Return runnability information on query-cpu-definitions target-i386: x86_cpu_load_features() function target-i386: Unset cannot_destroy_with_object_finalize_yet target-i386/kvm: cache the return value of kvm_enable_x2apic() intel_iommu: reject broken EIM intel_iommu: add OnOffAuto intr_eim as "eim" property intel_iommu: redo configuraton check in realize intel_iommu: pass whole remapped addresses to apic apic: add send_msi() to APICCommonClass apic: add global apic_get_class() target-i386: Move warning code outside x86_cpu_filter_features() qmp: Add runnability information to query-cpu-definitions target-i386: xsave: Add FP and SSE bits to x86_ext_save_areas target-i386: Register properties for feature aliases manually target-i386: Remove underscores from feat_names arrays target-i386: Make plus_features/minus_features QOM-based target-i386: Register aliases for feature names with underscores target-i386: Disable VME by default with TCG target-i386: List CPU models using subclass list ... Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
e8ddc2eae5
@ -21,6 +21,7 @@
|
||||
|
||||
#include "qemu/osdep.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include "qapi/error.h"
|
||||
#include "hw/sysbus.h"
|
||||
#include "exec/address-spaces.h"
|
||||
#include "intel_iommu_internal.h"
|
||||
@ -32,6 +33,8 @@
|
||||
#include "hw/i386/x86-iommu.h"
|
||||
#include "hw/pci-host/q35.h"
|
||||
#include "sysemu/kvm.h"
|
||||
#include "hw/i386/apic_internal.h"
|
||||
#include "kvm_i386.h"
|
||||
|
||||
/*#define DEBUG_INTEL_IOMMU*/
|
||||
#ifdef DEBUG_INTEL_IOMMU
|
||||
@ -280,18 +283,17 @@ static void vtd_update_iotlb(IntelIOMMUState *s, uint16_t source_id,
|
||||
static void vtd_generate_interrupt(IntelIOMMUState *s, hwaddr mesg_addr_reg,
|
||||
hwaddr mesg_data_reg)
|
||||
{
|
||||
hwaddr addr;
|
||||
uint32_t data;
|
||||
MSIMessage msi;
|
||||
|
||||
assert(mesg_data_reg < DMAR_REG_SIZE);
|
||||
assert(mesg_addr_reg < DMAR_REG_SIZE);
|
||||
|
||||
addr = vtd_get_long_raw(s, mesg_addr_reg);
|
||||
data = vtd_get_long_raw(s, mesg_data_reg);
|
||||
msi.address = vtd_get_long_raw(s, mesg_addr_reg);
|
||||
msi.data = vtd_get_long_raw(s, mesg_data_reg);
|
||||
|
||||
VTD_DPRINTF(FLOG, "msi: addr 0x%"PRIx64 " data 0x%"PRIx32, addr, data);
|
||||
address_space_stl_le(&address_space_memory, addr, data,
|
||||
MEMTXATTRS_UNSPECIFIED, NULL);
|
||||
VTD_DPRINTF(FLOG, "msi: addr 0x%"PRIx64 " data 0x%"PRIx32,
|
||||
msi.address, msi.data);
|
||||
apic_get_class()->send_msi(&msi);
|
||||
}
|
||||
|
||||
/* Generate a fault event to software via MSI if conditions are met.
|
||||
@ -2012,6 +2014,9 @@ static const MemoryRegionOps vtd_mem_ops = {
|
||||
|
||||
static Property vtd_properties[] = {
|
||||
DEFINE_PROP_UINT32("version", IntelIOMMUState, version, 0),
|
||||
DEFINE_PROP_ON_OFF_AUTO("eim", IntelIOMMUState, intr_eim,
|
||||
ON_OFF_AUTO_AUTO),
|
||||
DEFINE_PROP_BOOL("x-buggy-eim", IntelIOMMUState, buggy_eim, false),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@ -2134,6 +2139,7 @@ static void vtd_generate_msi_message(VTDIrq *irq, MSIMessage *msg_out)
|
||||
msg.dest_mode = irq->dest_mode;
|
||||
msg.redir_hint = irq->redir_hint;
|
||||
msg.dest = irq->dest;
|
||||
msg.__addr_hi = irq->dest & 0xffffff00;
|
||||
msg.__addr_head = cpu_to_le32(0xfee);
|
||||
/* Keep this from original MSI address bits */
|
||||
msg.__not_used = irq->msi_addr_last_bits;
|
||||
@ -2293,11 +2299,7 @@ static MemTxResult vtd_mem_ir_write(void *opaque, hwaddr addr,
|
||||
" for device sid 0x%04x",
|
||||
to.address, to.data, sid);
|
||||
|
||||
if (dma_memory_write(&address_space_memory, to.address,
|
||||
&to.data, size)) {
|
||||
VTD_DPRINTF(GENERAL, "error: fail to write 0x%"PRIx64
|
||||
" value 0x%"PRIx32, to.address, to.data);
|
||||
}
|
||||
apic_get_class()->send_msi(&to);
|
||||
|
||||
return MEMTX_OK;
|
||||
}
|
||||
@ -2382,7 +2384,11 @@ static void vtd_init(IntelIOMMUState *s)
|
||||
s->ecap = VTD_ECAP_QI | VTD_ECAP_IRO;
|
||||
|
||||
if (x86_iommu->intr_supported) {
|
||||
s->ecap |= VTD_ECAP_IR | VTD_ECAP_EIM | VTD_ECAP_MHMV;
|
||||
s->ecap |= VTD_ECAP_IR | VTD_ECAP_MHMV;
|
||||
if (s->intr_eim == ON_OFF_AUTO_ON) {
|
||||
s->ecap |= VTD_ECAP_EIM;
|
||||
}
|
||||
assert(s->intr_eim != ON_OFF_AUTO_AUTO);
|
||||
}
|
||||
|
||||
vtd_reset_context_cache(s);
|
||||
@ -2463,6 +2469,42 @@ static AddressSpace *vtd_host_dma_iommu(PCIBus *bus, void *opaque, int devfn)
|
||||
return &vtd_as->as;
|
||||
}
|
||||
|
||||
static bool vtd_decide_config(IntelIOMMUState *s, Error **errp)
|
||||
{
|
||||
X86IOMMUState *x86_iommu = X86_IOMMU_DEVICE(s);
|
||||
|
||||
/* Currently Intel IOMMU IR only support "kernel-irqchip={off|split}" */
|
||||
if (x86_iommu->intr_supported && kvm_irqchip_in_kernel() &&
|
||||
!kvm_irqchip_is_split()) {
|
||||
error_setg(errp, "Intel Interrupt Remapping cannot work with "
|
||||
"kernel-irqchip=on, please use 'split|off'.");
|
||||
return false;
|
||||
}
|
||||
if (s->intr_eim == ON_OFF_AUTO_ON && !x86_iommu->intr_supported) {
|
||||
error_setg(errp, "eim=on cannot be selected without intremap=on");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (s->intr_eim == ON_OFF_AUTO_AUTO) {
|
||||
s->intr_eim = (kvm_irqchip_in_kernel() || s->buggy_eim)
|
||||
&& x86_iommu->intr_supported ?
|
||||
ON_OFF_AUTO_ON : ON_OFF_AUTO_OFF;
|
||||
}
|
||||
if (s->intr_eim == ON_OFF_AUTO_ON && !s->buggy_eim) {
|
||||
if (!kvm_irqchip_in_kernel()) {
|
||||
error_setg(errp, "eim=on requires accel=kvm,kernel-irqchip=split");
|
||||
return false;
|
||||
}
|
||||
if (!kvm_enable_x2apic()) {
|
||||
error_setg(errp, "eim=on requires support on the KVM side"
|
||||
"(X2APIC_API, first shipped in v4.7)");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static void vtd_realize(DeviceState *dev, Error **errp)
|
||||
{
|
||||
PCMachineState *pcms = PC_MACHINE(qdev_get_machine());
|
||||
@ -2472,6 +2514,11 @@ static void vtd_realize(DeviceState *dev, Error **errp)
|
||||
|
||||
VTD_DPRINTF(GENERAL, "");
|
||||
x86_iommu->type = TYPE_INTEL;
|
||||
|
||||
if (!vtd_decide_config(s, errp)) {
|
||||
return;
|
||||
}
|
||||
|
||||
memset(s->vtd_as_by_bus_num, 0, sizeof(s->vtd_as_by_bus_num));
|
||||
memory_region_init_io(&s->csrmem, OBJECT(s), &vtd_mem_ops, s,
|
||||
"intel_iommu", DMAR_REG_SIZE);
|
||||
@ -2486,14 +2533,6 @@ static void vtd_realize(DeviceState *dev, Error **errp)
|
||||
pci_setup_iommu(bus, vtd_host_dma_iommu, dev);
|
||||
/* Pseudo address space under root PCI bus. */
|
||||
pcms->ioapic_as = vtd_host_dma_iommu(bus, s, Q35_PSEUDO_DEVFN_IOAPIC);
|
||||
|
||||
/* Currently Intel IOMMU IR only support "kernel-irqchip={off|split}" */
|
||||
if (x86_iommu->intr_supported && kvm_irqchip_in_kernel() &&
|
||||
!kvm_irqchip_is_split()) {
|
||||
error_report("Intel Interrupt Remapping cannot work with "
|
||||
"kernel-irqchip=on, please use 'split|off'.");
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
static void vtd_class_init(ObjectClass *klass, void *data)
|
||||
|
@ -169,6 +169,17 @@ static void kvm_apic_external_nmi(APICCommonState *s)
|
||||
run_on_cpu(CPU(s->cpu), do_inject_external_nmi, s);
|
||||
}
|
||||
|
||||
static void kvm_send_msi(MSIMessage *msg)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = kvm_irqchip_send_msi(kvm_state, *msg);
|
||||
if (ret < 0) {
|
||||
fprintf(stderr, "KVM: injection failed, MSI lost (%s)\n",
|
||||
strerror(-ret));
|
||||
}
|
||||
}
|
||||
|
||||
static uint64_t kvm_apic_mem_read(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
@ -179,13 +190,8 @@ static void kvm_apic_mem_write(void *opaque, hwaddr addr,
|
||||
uint64_t data, unsigned size)
|
||||
{
|
||||
MSIMessage msg = { .address = addr, .data = data };
|
||||
int ret;
|
||||
|
||||
ret = kvm_irqchip_send_msi(kvm_state, msg);
|
||||
if (ret < 0) {
|
||||
fprintf(stderr, "KVM: injection failed, MSI lost (%s)\n",
|
||||
strerror(-ret));
|
||||
}
|
||||
kvm_send_msi(&msg);
|
||||
}
|
||||
|
||||
static const MemoryRegionOps kvm_apic_io_ops = {
|
||||
@ -232,6 +238,7 @@ static void kvm_apic_class_init(ObjectClass *klass, void *data)
|
||||
k->enable_tpr_reporting = kvm_apic_enable_tpr_reporting;
|
||||
k->vapic_base_update = kvm_apic_vapic_base_update;
|
||||
k->external_nmi = kvm_apic_external_nmi;
|
||||
k->send_msi = kvm_send_msi;
|
||||
}
|
||||
|
||||
static const TypeInfo kvm_apic_info = {
|
||||
|
@ -68,6 +68,11 @@ static void xen_apic_external_nmi(APICCommonState *s)
|
||||
{
|
||||
}
|
||||
|
||||
static void xen_send_msi(MSIMessage *msi)
|
||||
{
|
||||
xen_hvm_inject_msi(msi->address, msi->data);
|
||||
}
|
||||
|
||||
static void xen_apic_class_init(ObjectClass *klass, void *data)
|
||||
{
|
||||
APICCommonClass *k = APIC_COMMON_CLASS(klass);
|
||||
@ -78,6 +83,7 @@ static void xen_apic_class_init(ObjectClass *klass, void *data)
|
||||
k->get_tpr = xen_apic_get_tpr;
|
||||
k->vapic_base_update = xen_apic_vapic_base_update;
|
||||
k->external_nmi = xen_apic_external_nmi;
|
||||
k->send_msi = xen_send_msi;
|
||||
}
|
||||
|
||||
static const TypeInfo xen_apic_info = {
|
||||
|
@ -740,8 +740,10 @@ static uint32_t apic_mem_readl(void *opaque, hwaddr addr)
|
||||
return val;
|
||||
}
|
||||
|
||||
static void apic_send_msi(hwaddr addr, uint32_t data)
|
||||
static void apic_send_msi(MSIMessage *msi)
|
||||
{
|
||||
uint64_t addr = msi->address;
|
||||
uint32_t data = msi->data;
|
||||
uint8_t dest = (addr & MSI_ADDR_DEST_ID_MASK) >> MSI_ADDR_DEST_ID_SHIFT;
|
||||
uint8_t vector = (data & MSI_DATA_VECTOR_MASK) >> MSI_DATA_VECTOR_SHIFT;
|
||||
uint8_t dest_mode = (addr >> MSI_ADDR_DEST_MODE_SHIFT) & 0x1;
|
||||
@ -762,7 +764,8 @@ static void apic_mem_writel(void *opaque, hwaddr addr, uint32_t val)
|
||||
* APIC is connected directly to the CPU.
|
||||
* Mapping them on the global bus happens to work because
|
||||
* MSI registers are reserved in APIC MMIO and vice versa. */
|
||||
apic_send_msi(addr, val);
|
||||
MSIMessage msi = { .address = addr, .data = val };
|
||||
apic_send_msi(&msi);
|
||||
return;
|
||||
}
|
||||
|
||||
@ -913,6 +916,7 @@ static void apic_class_init(ObjectClass *klass, void *data)
|
||||
k->external_nmi = apic_external_nmi;
|
||||
k->pre_save = apic_pre_save;
|
||||
k->post_load = apic_post_load;
|
||||
k->send_msi = apic_send_msi;
|
||||
}
|
||||
|
||||
static const TypeInfo apic_info = {
|
||||
|
@ -18,6 +18,7 @@
|
||||
* License along with this library; if not, see <http://www.gnu.org/licenses/>
|
||||
*/
|
||||
#include "qemu/osdep.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include "qapi/error.h"
|
||||
#include "qemu-common.h"
|
||||
#include "cpu.h"
|
||||
|
@ -14,6 +14,10 @@
|
||||
.driver = "ioapic",\
|
||||
.property = "version",\
|
||||
.value = "0x11",\
|
||||
},{\
|
||||
.driver = "intel-iommu",\
|
||||
.property = "x-buggy-eim",\
|
||||
.value = "true",\
|
||||
},
|
||||
|
||||
#define HW_COMPAT_2_6 \
|
||||
|
@ -146,6 +146,10 @@ typedef struct APICCommonClass
|
||||
void (*pre_save)(APICCommonState *s);
|
||||
void (*post_load)(APICCommonState *s);
|
||||
void (*reset)(APICCommonState *s);
|
||||
/* send_msi emulates an APIC bus and its proper place would be in a new
|
||||
* device, but it's convenient to have it here for now.
|
||||
*/
|
||||
void (*send_msi)(MSIMessage *msi);
|
||||
} APICCommonClass;
|
||||
|
||||
struct APICCommonState {
|
||||
@ -222,4 +226,6 @@ static inline int apic_get_bit(uint32_t *tab, int index)
|
||||
return !!(tab[i] & mask);
|
||||
}
|
||||
|
||||
APICCommonClass *apic_get_class(void);
|
||||
|
||||
#endif /* QEMU_APIC_INTERNAL_H */
|
||||
|
@ -289,6 +289,8 @@ struct IntelIOMMUState {
|
||||
dma_addr_t intr_root; /* Interrupt remapping table pointer */
|
||||
uint32_t intr_size; /* Number of IR table entries */
|
||||
bool intr_eime; /* Extended interrupt mode enabled */
|
||||
OnOffAuto intr_eim; /* Toggle for EIM cabability */
|
||||
bool buggy_eim; /* Force buggy EIM unless eim=off */
|
||||
};
|
||||
|
||||
/* Find the VTD Address space associated with the given bus pointer,
|
||||
|
@ -3101,10 +3101,31 @@
|
||||
# QEMU version, machine type, machine options and accelerator options.
|
||||
# A static model is always migration-safe. (since 2.8)
|
||||
#
|
||||
# @unavailable-features: #optional List of properties that prevent
|
||||
# the CPU model from running in the current
|
||||
# host. (since 2.8)
|
||||
#
|
||||
# @unavailable-features is a list of QOM property names that
|
||||
# represent CPU model attributes that prevent the CPU from running.
|
||||
# If the QOM property is read-only, that means there's no known
|
||||
# way to make the CPU model run in the current host. Implementations
|
||||
# that choose not to provide specific information return the
|
||||
# property name "type".
|
||||
# If the property is read-write, it means that it MAY be possible
|
||||
# to run the CPU model in the current host if that property is
|
||||
# changed. Management software can use it as hints to suggest or
|
||||
# choose an alternative for the user, or just to generate meaningful
|
||||
# error messages explaining why the CPU model can't be used.
|
||||
# If @unavailable-features is an empty list, the CPU model is
|
||||
# runnable using the current host and machine-type.
|
||||
# If @unavailable-features is not present, runnability
|
||||
# information for the CPU is not available.
|
||||
#
|
||||
# Since: 1.2.0
|
||||
##
|
||||
{ 'struct': 'CpuDefinitionInfo',
|
||||
'data': { 'name': 'str', '*migration-safe': 'bool', 'static': 'bool' } }
|
||||
'data': { 'name': 'str', '*migration-safe': 'bool', 'static': 'bool',
|
||||
'*unavailable-features': [ 'str' ] } }
|
||||
|
||||
##
|
||||
# @query-cpu-definitions:
|
||||
|
@ -63,6 +63,10 @@ typedef struct X86CPUClass {
|
||||
|
||||
bool kvm_required;
|
||||
|
||||
/* Optional description of CPU model.
|
||||
* If unavailable, cpu_def->model_id is used */
|
||||
const char *model_description;
|
||||
|
||||
DeviceRealize parent_realize;
|
||||
void (*parent_reset)(CPUState *cpu);
|
||||
} X86CPUClass;
|
||||
|
@ -278,12 +278,12 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
|
||||
},
|
||||
[FEAT_1_ECX] = {
|
||||
.feat_names = {
|
||||
"pni|sse3" /* Intel,AMD sse3 */, "pclmulqdq|pclmuldq", "dtes64", "monitor",
|
||||
"ds_cpl", "vmx", "smx", "est",
|
||||
"pni" /* Intel,AMD sse3 */, "pclmulqdq", "dtes64", "monitor",
|
||||
"ds-cpl", "vmx", "smx", "est",
|
||||
"tm2", "ssse3", "cid", NULL,
|
||||
"fma", "cx16", "xtpr", "pdcm",
|
||||
NULL, "pcid", "dca", "sse4.1|sse4_1",
|
||||
"sse4.2|sse4_2", "x2apic", "movbe", "popcnt",
|
||||
NULL, "pcid", "dca", "sse4.1",
|
||||
"sse4.2", "x2apic", "movbe", "popcnt",
|
||||
"tsc-deadline", "aes", "xsave", "osxsave",
|
||||
"avx", "f16c", "rdrand", "hypervisor",
|
||||
},
|
||||
@ -302,22 +302,22 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
|
||||
NULL /* cx8 */, NULL /* apic */, NULL, "syscall",
|
||||
NULL /* mtrr */, NULL /* pge */, NULL /* mca */, NULL /* cmov */,
|
||||
NULL /* pat */, NULL /* pse36 */, NULL, NULL /* Linux mp */,
|
||||
"nx|xd", NULL, "mmxext", NULL /* mmx */,
|
||||
NULL /* fxsr */, "fxsr_opt|ffxsr", "pdpe1gb", "rdtscp",
|
||||
NULL, "lm|i64", "3dnowext", "3dnow",
|
||||
"nx", NULL, "mmxext", NULL /* mmx */,
|
||||
NULL /* fxsr */, "fxsr-opt", "pdpe1gb", "rdtscp",
|
||||
NULL, "lm", "3dnowext", "3dnow",
|
||||
},
|
||||
.cpuid_eax = 0x80000001, .cpuid_reg = R_EDX,
|
||||
.tcg_features = TCG_EXT2_FEATURES,
|
||||
},
|
||||
[FEAT_8000_0001_ECX] = {
|
||||
.feat_names = {
|
||||
"lahf_lm", "cmp_legacy", "svm", "extapic",
|
||||
"lahf-lm", "cmp-legacy", "svm", "extapic",
|
||||
"cr8legacy", "abm", "sse4a", "misalignsse",
|
||||
"3dnowprefetch", "osvw", "ibs", "xop",
|
||||
"skinit", "wdt", NULL, "lwp",
|
||||
"fma4", "tce", NULL, "nodeid_msr",
|
||||
NULL, "tbm", "topoext", "perfctr_core",
|
||||
"perfctr_nb", NULL, NULL, NULL,
|
||||
"fma4", "tce", NULL, "nodeid-msr",
|
||||
NULL, "tbm", "topoext", "perfctr-core",
|
||||
"perfctr-nb", NULL, NULL, NULL,
|
||||
NULL, NULL, NULL, NULL,
|
||||
},
|
||||
.cpuid_eax = 0x80000001, .cpuid_reg = R_ECX,
|
||||
@ -339,8 +339,8 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
|
||||
},
|
||||
[FEAT_KVM] = {
|
||||
.feat_names = {
|
||||
"kvmclock", "kvm_nopiodelay", "kvm_mmu", "kvmclock",
|
||||
"kvm_asyncpf", "kvm_steal_time", "kvm_pv_eoi", "kvm_pv_unhalt",
|
||||
"kvmclock", "kvm-nopiodelay", "kvm-mmu", "kvmclock",
|
||||
"kvm-asyncpf", "kvm-steal-time", "kvm-pv-eoi", "kvm-pv-unhalt",
|
||||
NULL, NULL, NULL, NULL,
|
||||
NULL, NULL, NULL, NULL,
|
||||
NULL, NULL, NULL, NULL,
|
||||
@ -400,9 +400,9 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
|
||||
},
|
||||
[FEAT_SVM] = {
|
||||
.feat_names = {
|
||||
"npt", "lbrv", "svm_lock", "nrip_save",
|
||||
"tsc_scale", "vmcb_clean", "flushbyasid", "decodeassists",
|
||||
NULL, NULL, "pause_filter", NULL,
|
||||
"npt", "lbrv", "svm-lock", "nrip-save",
|
||||
"tsc-scale", "vmcb-clean", "flushbyasid", "decodeassists",
|
||||
NULL, NULL, "pause-filter", NULL,
|
||||
"pfthreshold", NULL, NULL, NULL,
|
||||
NULL, NULL, NULL, NULL,
|
||||
NULL, NULL, NULL, NULL,
|
||||
@ -414,7 +414,7 @@ static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
|
||||
},
|
||||
[FEAT_7_0_EBX] = {
|
||||
.feat_names = {
|
||||
"fsgsbase", "tsc_adjust", NULL, "bmi1",
|
||||
"fsgsbase", "tsc-adjust", NULL, "bmi1",
|
||||
"hle", "avx2", NULL, "smep",
|
||||
"bmi2", "erms", "invpcid", "rtm",
|
||||
NULL, NULL, "mpx", NULL,
|
||||
@ -535,6 +535,20 @@ typedef struct ExtSaveArea {
|
||||
} ExtSaveArea;
|
||||
|
||||
static const ExtSaveArea x86_ext_save_areas[] = {
|
||||
[XSTATE_FP_BIT] = {
|
||||
/* x87 FP state component is always enabled if XSAVE is supported */
|
||||
.feature = FEAT_1_ECX, .bits = CPUID_EXT_XSAVE,
|
||||
/* x87 state is in the legacy region of the XSAVE area */
|
||||
.offset = 0,
|
||||
.size = sizeof(X86LegacyXSaveArea) + sizeof(X86XSaveHeader),
|
||||
},
|
||||
[XSTATE_SSE_BIT] = {
|
||||
/* SSE state component is always enabled if XSAVE is supported */
|
||||
.feature = FEAT_1_ECX, .bits = CPUID_EXT_XSAVE,
|
||||
/* SSE state is in the legacy region of the XSAVE area */
|
||||
.offset = 0,
|
||||
.size = sizeof(X86LegacyXSaveArea) + sizeof(X86XSaveHeader),
|
||||
},
|
||||
[XSTATE_YMM_BIT] =
|
||||
{ .feature = FEAT_1_ECX, .bits = CPUID_EXT_AVX,
|
||||
.offset = offsetof(X86XSaveArea, avx_state),
|
||||
@ -568,9 +582,9 @@ static const ExtSaveArea x86_ext_save_areas[] = {
|
||||
static uint32_t xsave_area_size(uint64_t mask)
|
||||
{
|
||||
int i;
|
||||
uint64_t ret = sizeof(X86LegacyXSaveArea) + sizeof(X86XSaveHeader);
|
||||
uint64_t ret = 0;
|
||||
|
||||
for (i = 2; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
|
||||
for (i = 0; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
|
||||
const ExtSaveArea *esa = &x86_ext_save_areas[i];
|
||||
if ((mask >> i) & 1) {
|
||||
ret = MAX(ret, esa->offset + esa->size);
|
||||
@ -650,85 +664,6 @@ void host_cpuid(uint32_t function, uint32_t count,
|
||||
*edx = vec[3];
|
||||
}
|
||||
|
||||
#define iswhite(c) ((c) && ((c) <= ' ' || '~' < (c)))
|
||||
|
||||
/* general substring compare of *[s1..e1) and *[s2..e2). sx is start of
|
||||
* a substring. ex if !NULL points to the first char after a substring,
|
||||
* otherwise the string is assumed to sized by a terminating nul.
|
||||
* Return lexical ordering of *s1:*s2.
|
||||
*/
|
||||
static int sstrcmp(const char *s1, const char *e1,
|
||||
const char *s2, const char *e2)
|
||||
{
|
||||
for (;;) {
|
||||
if (!*s1 || !*s2 || *s1 != *s2)
|
||||
return (*s1 - *s2);
|
||||
++s1, ++s2;
|
||||
if (s1 == e1 && s2 == e2)
|
||||
return (0);
|
||||
else if (s1 == e1)
|
||||
return (*s2);
|
||||
else if (s2 == e2)
|
||||
return (*s1);
|
||||
}
|
||||
}
|
||||
|
||||
/* compare *[s..e) to *altstr. *altstr may be a simple string or multiple
|
||||
* '|' delimited (possibly empty) strings in which case search for a match
|
||||
* within the alternatives proceeds left to right. Return 0 for success,
|
||||
* non-zero otherwise.
|
||||
*/
|
||||
static int altcmp(const char *s, const char *e, const char *altstr)
|
||||
{
|
||||
const char *p, *q;
|
||||
|
||||
for (q = p = altstr; ; ) {
|
||||
while (*p && *p != '|')
|
||||
++p;
|
||||
if ((q == p && !*s) || (q != p && !sstrcmp(s, e, q, p)))
|
||||
return (0);
|
||||
if (!*p)
|
||||
return (1);
|
||||
else
|
||||
q = ++p;
|
||||
}
|
||||
}
|
||||
|
||||
/* search featureset for flag *[s..e), if found set corresponding bit in
|
||||
* *pval and return true, otherwise return false
|
||||
*/
|
||||
static bool lookup_feature(uint32_t *pval, const char *s, const char *e,
|
||||
const char **featureset)
|
||||
{
|
||||
uint32_t mask;
|
||||
const char **ppc;
|
||||
bool found = false;
|
||||
|
||||
for (mask = 1, ppc = featureset; mask; mask <<= 1, ++ppc) {
|
||||
if (*ppc && !altcmp(s, e, *ppc)) {
|
||||
*pval |= mask;
|
||||
found = true;
|
||||
}
|
||||
}
|
||||
return found;
|
||||
}
|
||||
|
||||
static void add_flagname_to_bitmaps(const char *flagname,
|
||||
FeatureWordArray words,
|
||||
Error **errp)
|
||||
{
|
||||
FeatureWord w;
|
||||
for (w = 0; w < FEATURE_WORDS; w++) {
|
||||
FeatureWordInfo *wi = &feature_word_info[w];
|
||||
if (lookup_feature(&words[w], flagname, NULL, wi->feat_names)) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (w == FEATURE_WORDS) {
|
||||
error_setg(errp, "CPU feature %s not found", flagname);
|
||||
}
|
||||
}
|
||||
|
||||
/* CPU class name definitions: */
|
||||
|
||||
#define X86_CPU_TYPE_SUFFIX "-" TYPE_X86_CPU
|
||||
@ -1550,6 +1485,14 @@ static PropValue kvm_default_props[] = {
|
||||
{ NULL, NULL },
|
||||
};
|
||||
|
||||
/* TCG-specific defaults that override all CPU models when using TCG
|
||||
*/
|
||||
static PropValue tcg_default_props[] = {
|
||||
{ "vme", "off" },
|
||||
{ NULL, NULL },
|
||||
};
|
||||
|
||||
|
||||
void x86_cpu_change_kvm_default(const char *prop, const char *value)
|
||||
{
|
||||
PropValue *pv;
|
||||
@ -1628,6 +1571,9 @@ static void host_x86_cpu_class_init(ObjectClass *oc, void *data)
|
||||
cpu_x86_fill_model_id(host_cpudef.model_id);
|
||||
|
||||
xcc->cpu_def = &host_cpudef;
|
||||
xcc->model_description =
|
||||
"KVM processor with all supported host features "
|
||||
"(only available in KVM mode)";
|
||||
|
||||
/* level, xlevel, xlevel2, and the feature words are initialized on
|
||||
* instance_init, because they require KVM to be initialized.
|
||||
@ -1999,13 +1945,33 @@ static inline void feat2prop(char *s)
|
||||
}
|
||||
}
|
||||
|
||||
/* Return the feature property name for a feature flag bit */
|
||||
static const char *x86_cpu_feature_name(FeatureWord w, int bitnr)
|
||||
{
|
||||
/* XSAVE components are automatically enabled by other features,
|
||||
* so return the original feature name instead
|
||||
*/
|
||||
if (w == FEAT_XSAVE_COMP_LO || w == FEAT_XSAVE_COMP_HI) {
|
||||
int comp = (w == FEAT_XSAVE_COMP_HI) ? bitnr + 32 : bitnr;
|
||||
|
||||
if (comp < ARRAY_SIZE(x86_ext_save_areas) &&
|
||||
x86_ext_save_areas[comp].bits) {
|
||||
w = x86_ext_save_areas[comp].feature;
|
||||
bitnr = ctz32(x86_ext_save_areas[comp].bits);
|
||||
}
|
||||
}
|
||||
|
||||
assert(bitnr < 32);
|
||||
assert(w < FEATURE_WORDS);
|
||||
return feature_word_info[w].feat_names[bitnr];
|
||||
}
|
||||
|
||||
/* Compatibily hack to maintain legacy +-feat semantic,
|
||||
* where +-feat overwrites any feature set by
|
||||
* feat=on|feat even if the later is parsed after +-feat
|
||||
* (i.e. "-x2apic,x2apic=on" will result in x2apic disabled)
|
||||
*/
|
||||
static FeatureWordArray plus_features = { 0 };
|
||||
static FeatureWordArray minus_features = { 0 };
|
||||
static GList *plus_features, *minus_features;
|
||||
|
||||
/* Parse "+feature,-feature,feature=foo" CPU feature string
|
||||
*/
|
||||
@ -2036,10 +2002,12 @@ static void x86_cpu_parse_featurestr(const char *typename, char *features,
|
||||
|
||||
/* Compatibility syntax: */
|
||||
if (featurestr[0] == '+') {
|
||||
add_flagname_to_bitmaps(featurestr + 1, plus_features, &local_err);
|
||||
plus_features = g_list_append(plus_features,
|
||||
g_strdup(featurestr + 1));
|
||||
continue;
|
||||
} else if (featurestr[0] == '-') {
|
||||
add_flagname_to_bitmaps(featurestr + 1, minus_features, &local_err);
|
||||
minus_features = g_list_append(minus_features,
|
||||
g_strdup(featurestr + 1));
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -2083,6 +2051,59 @@ static void x86_cpu_parse_featurestr(const char *typename, char *features,
|
||||
}
|
||||
}
|
||||
|
||||
static void x86_cpu_load_features(X86CPU *cpu, Error **errp);
|
||||
static int x86_cpu_filter_features(X86CPU *cpu);
|
||||
|
||||
/* Check for missing features that may prevent the CPU class from
|
||||
* running using the current machine and accelerator.
|
||||
*/
|
||||
static void x86_cpu_class_check_missing_features(X86CPUClass *xcc,
|
||||
strList **missing_feats)
|
||||
{
|
||||
X86CPU *xc;
|
||||
FeatureWord w;
|
||||
Error *err = NULL;
|
||||
strList **next = missing_feats;
|
||||
|
||||
if (xcc->kvm_required && !kvm_enabled()) {
|
||||
strList *new = g_new0(strList, 1);
|
||||
new->value = g_strdup("kvm");;
|
||||
*missing_feats = new;
|
||||
return;
|
||||
}
|
||||
|
||||
xc = X86_CPU(object_new(object_class_get_name(OBJECT_CLASS(xcc))));
|
||||
|
||||
x86_cpu_load_features(xc, &err);
|
||||
if (err) {
|
||||
/* Errors at x86_cpu_load_features should never happen,
|
||||
* but in case it does, just report the model as not
|
||||
* runnable at all using the "type" property.
|
||||
*/
|
||||
strList *new = g_new0(strList, 1);
|
||||
new->value = g_strdup("type");
|
||||
*next = new;
|
||||
next = &new->next;
|
||||
}
|
||||
|
||||
x86_cpu_filter_features(xc);
|
||||
|
||||
for (w = 0; w < FEATURE_WORDS; w++) {
|
||||
uint32_t filtered = xc->filtered_features[w];
|
||||
int i;
|
||||
for (i = 0; i < 32; i++) {
|
||||
if (filtered & (1UL << i)) {
|
||||
strList *new = g_new0(strList, 1);
|
||||
new->value = g_strdup(x86_cpu_feature_name(w, i));
|
||||
*next = new;
|
||||
next = &new->next;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
object_unref(OBJECT(xc));
|
||||
}
|
||||
|
||||
/* Print all cpuid feature names in featureset
|
||||
*/
|
||||
static void listflags(FILE *f, fprintf_function print, const char **featureset)
|
||||
@ -2098,23 +2119,62 @@ static void listflags(FILE *f, fprintf_function print, const char **featureset)
|
||||
}
|
||||
}
|
||||
|
||||
/* generate CPU information. */
|
||||
/* Sort alphabetically by type name, listing kvm_required models last. */
|
||||
static gint x86_cpu_list_compare(gconstpointer a, gconstpointer b)
|
||||
{
|
||||
ObjectClass *class_a = (ObjectClass *)a;
|
||||
ObjectClass *class_b = (ObjectClass *)b;
|
||||
X86CPUClass *cc_a = X86_CPU_CLASS(class_a);
|
||||
X86CPUClass *cc_b = X86_CPU_CLASS(class_b);
|
||||
const char *name_a, *name_b;
|
||||
|
||||
if (cc_a->kvm_required != cc_b->kvm_required) {
|
||||
/* kvm_required items go last */
|
||||
return cc_a->kvm_required ? 1 : -1;
|
||||
} else {
|
||||
name_a = object_class_get_name(class_a);
|
||||
name_b = object_class_get_name(class_b);
|
||||
return strcmp(name_a, name_b);
|
||||
}
|
||||
}
|
||||
|
||||
static GSList *get_sorted_cpu_model_list(void)
|
||||
{
|
||||
GSList *list = object_class_get_list(TYPE_X86_CPU, false);
|
||||
list = g_slist_sort(list, x86_cpu_list_compare);
|
||||
return list;
|
||||
}
|
||||
|
||||
static void x86_cpu_list_entry(gpointer data, gpointer user_data)
|
||||
{
|
||||
ObjectClass *oc = data;
|
||||
X86CPUClass *cc = X86_CPU_CLASS(oc);
|
||||
CPUListState *s = user_data;
|
||||
char *name = x86_cpu_class_get_model_name(cc);
|
||||
const char *desc = cc->model_description;
|
||||
if (!desc) {
|
||||
desc = cc->cpu_def->model_id;
|
||||
}
|
||||
|
||||
(*s->cpu_fprintf)(s->file, "x86 %16s %-48s\n",
|
||||
name, desc);
|
||||
g_free(name);
|
||||
}
|
||||
|
||||
/* list available CPU models and flags */
|
||||
void x86_cpu_list(FILE *f, fprintf_function cpu_fprintf)
|
||||
{
|
||||
X86CPUDefinition *def;
|
||||
char buf[256];
|
||||
int i;
|
||||
CPUListState s = {
|
||||
.file = f,
|
||||
.cpu_fprintf = cpu_fprintf,
|
||||
};
|
||||
GSList *list;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(builtin_x86_defs); i++) {
|
||||
def = &builtin_x86_defs[i];
|
||||
snprintf(buf, sizeof(buf), "%s", def->name);
|
||||
(*cpu_fprintf)(f, "x86 %16s %-48s\n", buf, def->model_id);
|
||||
}
|
||||
#ifdef CONFIG_KVM
|
||||
(*cpu_fprintf)(f, "x86 %16s %-48s\n", "host",
|
||||
"KVM processor with all supported host features "
|
||||
"(only available in KVM mode)");
|
||||
#endif
|
||||
(*cpu_fprintf)(f, "Available CPUs:\n");
|
||||
list = get_sorted_cpu_model_list();
|
||||
g_slist_foreach(list, x86_cpu_list_entry, &s);
|
||||
g_slist_free(list);
|
||||
|
||||
(*cpu_fprintf)(f, "\nRecognized CPUID flags:\n");
|
||||
for (i = 0; i < ARRAY_SIZE(feature_word_info); i++) {
|
||||
@ -2126,26 +2186,31 @@ void x86_cpu_list(FILE *f, fprintf_function cpu_fprintf)
|
||||
}
|
||||
}
|
||||
|
||||
static void x86_cpu_definition_entry(gpointer data, gpointer user_data)
|
||||
{
|
||||
ObjectClass *oc = data;
|
||||
X86CPUClass *cc = X86_CPU_CLASS(oc);
|
||||
CpuDefinitionInfoList **cpu_list = user_data;
|
||||
CpuDefinitionInfoList *entry;
|
||||
CpuDefinitionInfo *info;
|
||||
|
||||
info = g_malloc0(sizeof(*info));
|
||||
info->name = x86_cpu_class_get_model_name(cc);
|
||||
x86_cpu_class_check_missing_features(cc, &info->unavailable_features);
|
||||
info->has_unavailable_features = true;
|
||||
|
||||
entry = g_malloc0(sizeof(*entry));
|
||||
entry->value = info;
|
||||
entry->next = *cpu_list;
|
||||
*cpu_list = entry;
|
||||
}
|
||||
|
||||
CpuDefinitionInfoList *arch_query_cpu_definitions(Error **errp)
|
||||
{
|
||||
CpuDefinitionInfoList *cpu_list = NULL;
|
||||
X86CPUDefinition *def;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(builtin_x86_defs); i++) {
|
||||
CpuDefinitionInfoList *entry;
|
||||
CpuDefinitionInfo *info;
|
||||
|
||||
def = &builtin_x86_defs[i];
|
||||
info = g_malloc0(sizeof(*info));
|
||||
info->name = g_strdup(def->name);
|
||||
|
||||
entry = g_malloc0(sizeof(*entry));
|
||||
entry->value = info;
|
||||
entry->next = cpu_list;
|
||||
cpu_list = entry;
|
||||
}
|
||||
|
||||
GSList *list = get_sorted_cpu_model_list();
|
||||
g_slist_foreach(list, x86_cpu_definition_entry, &cpu_list);
|
||||
g_slist_free(list);
|
||||
return cpu_list;
|
||||
}
|
||||
|
||||
@ -2183,14 +2248,11 @@ static int x86_cpu_filter_features(X86CPU *cpu)
|
||||
|
||||
for (w = 0; w < FEATURE_WORDS; w++) {
|
||||
uint32_t host_feat =
|
||||
x86_cpu_get_supported_feature_word(w, cpu->migratable);
|
||||
x86_cpu_get_supported_feature_word(w, false);
|
||||
uint32_t requested_features = env->features[w];
|
||||
env->features[w] &= host_feat;
|
||||
cpu->filtered_features[w] = requested_features & ~env->features[w];
|
||||
if (cpu->filtered_features[w]) {
|
||||
if (cpu->check_cpuid || cpu->enforce_cpuid) {
|
||||
report_unavailable_features(w, cpu->filtered_features[w]);
|
||||
}
|
||||
rv = 1;
|
||||
}
|
||||
}
|
||||
@ -2198,6 +2260,15 @@ static int x86_cpu_filter_features(X86CPU *cpu)
|
||||
return rv;
|
||||
}
|
||||
|
||||
static void x86_cpu_report_filtered_features(X86CPU *cpu)
|
||||
{
|
||||
FeatureWord w;
|
||||
|
||||
for (w = 0; w < FEATURE_WORDS; w++) {
|
||||
report_unavailable_features(w, cpu->filtered_features[w]);
|
||||
}
|
||||
}
|
||||
|
||||
static void x86_cpu_apply_props(X86CPU *cpu, PropValue *props)
|
||||
{
|
||||
PropValue *pv;
|
||||
@ -2238,6 +2309,8 @@ static void x86_cpu_load_def(X86CPU *cpu, X86CPUDefinition *def, Error **errp)
|
||||
}
|
||||
|
||||
x86_cpu_apply_props(cpu, kvm_default_props);
|
||||
} else if (tcg_enabled()) {
|
||||
x86_cpu_apply_props(cpu, tcg_default_props);
|
||||
}
|
||||
|
||||
env->features[FEAT_1_ECX] |= CPUID_EXT_HYPERVISOR;
|
||||
@ -2848,9 +2921,8 @@ static void mce_init(X86CPU *cpu)
|
||||
}
|
||||
|
||||
#ifndef CONFIG_USER_ONLY
|
||||
static void x86_cpu_apic_create(X86CPU *cpu, Error **errp)
|
||||
APICCommonClass *apic_get_class(void)
|
||||
{
|
||||
APICCommonState *apic;
|
||||
const char *apic_type = "apic";
|
||||
|
||||
if (kvm_apic_in_kernel()) {
|
||||
@ -2859,7 +2931,15 @@ static void x86_cpu_apic_create(X86CPU *cpu, Error **errp)
|
||||
apic_type = "xen-apic";
|
||||
}
|
||||
|
||||
cpu->apic_state = DEVICE(object_new(apic_type));
|
||||
return APIC_COMMON_CLASS(object_class_by_name(apic_type));
|
||||
}
|
||||
|
||||
static void x86_cpu_apic_create(X86CPU *cpu, Error **errp)
|
||||
{
|
||||
APICCommonState *apic;
|
||||
ObjectClass *apic_class = OBJECT_CLASS(apic_get_class());
|
||||
|
||||
cpu->apic_state = DEVICE(object_new(object_class_get_name(apic_class)));
|
||||
|
||||
object_property_add_child(OBJECT(cpu), "lapic",
|
||||
OBJECT(cpu->apic_state), &error_abort);
|
||||
@ -2984,8 +3064,8 @@ static void x86_cpu_enable_xsave_components(X86CPU *cpu)
|
||||
return;
|
||||
}
|
||||
|
||||
mask = (XSTATE_FP_MASK | XSTATE_SSE_MASK);
|
||||
for (i = 2; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
|
||||
mask = 0;
|
||||
for (i = 0; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
|
||||
const ExtSaveArea *esa = &x86_ext_save_areas[i];
|
||||
if (env->features[esa->feature] & esa->bits) {
|
||||
mask |= (1ULL << i);
|
||||
@ -2996,33 +3076,13 @@ static void x86_cpu_enable_xsave_components(X86CPU *cpu)
|
||||
env->features[FEAT_XSAVE_COMP_HI] = mask >> 32;
|
||||
}
|
||||
|
||||
#define IS_INTEL_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_INTEL_1 && \
|
||||
(env)->cpuid_vendor2 == CPUID_VENDOR_INTEL_2 && \
|
||||
(env)->cpuid_vendor3 == CPUID_VENDOR_INTEL_3)
|
||||
#define IS_AMD_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_AMD_1 && \
|
||||
(env)->cpuid_vendor2 == CPUID_VENDOR_AMD_2 && \
|
||||
(env)->cpuid_vendor3 == CPUID_VENDOR_AMD_3)
|
||||
static void x86_cpu_realizefn(DeviceState *dev, Error **errp)
|
||||
/* Load CPUID data based on configured features */
|
||||
static void x86_cpu_load_features(X86CPU *cpu, Error **errp)
|
||||
{
|
||||
CPUState *cs = CPU(dev);
|
||||
X86CPU *cpu = X86_CPU(dev);
|
||||
X86CPUClass *xcc = X86_CPU_GET_CLASS(dev);
|
||||
CPUX86State *env = &cpu->env;
|
||||
Error *local_err = NULL;
|
||||
static bool ht_warned;
|
||||
FeatureWord w;
|
||||
|
||||
if (xcc->kvm_required && !kvm_enabled()) {
|
||||
char *name = x86_cpu_class_get_model_name(xcc);
|
||||
error_setg(&local_err, "CPU model '%s' requires KVM", name);
|
||||
g_free(name);
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (cpu->apic_id == UNASSIGNED_APIC_ID) {
|
||||
error_setg(errp, "apic-id property was not initialized properly");
|
||||
return;
|
||||
}
|
||||
GList *l;
|
||||
Error *local_err = NULL;
|
||||
|
||||
/*TODO: cpu->host_features incorrectly overwrites features
|
||||
* set using "feat=on|off". Once we fix this, we can convert
|
||||
@ -3036,9 +3096,20 @@ static void x86_cpu_realizefn(DeviceState *dev, Error **errp)
|
||||
}
|
||||
}
|
||||
|
||||
for (w = 0; w < FEATURE_WORDS; w++) {
|
||||
cpu->env.features[w] |= plus_features[w];
|
||||
cpu->env.features[w] &= ~minus_features[w];
|
||||
for (l = plus_features; l; l = l->next) {
|
||||
const char *prop = l->data;
|
||||
object_property_set_bool(OBJECT(cpu), true, prop, &local_err);
|
||||
if (local_err) {
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
|
||||
for (l = minus_features; l; l = l->next) {
|
||||
const char *prop = l->data;
|
||||
object_property_set_bool(OBJECT(cpu), false, prop, &local_err);
|
||||
if (local_err) {
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
|
||||
if (!kvm_enabled() || !cpu->expose_kvm) {
|
||||
@ -3077,14 +3148,56 @@ static void x86_cpu_realizefn(DeviceState *dev, Error **errp)
|
||||
env->cpuid_xlevel2 = env->cpuid_min_xlevel2;
|
||||
}
|
||||
|
||||
if (x86_cpu_filter_features(cpu) && cpu->enforce_cpuid) {
|
||||
error_setg(&local_err,
|
||||
kvm_enabled() ?
|
||||
"Host doesn't support requested features" :
|
||||
"TCG doesn't support requested features");
|
||||
out:
|
||||
if (local_err != NULL) {
|
||||
error_propagate(errp, local_err);
|
||||
}
|
||||
}
|
||||
|
||||
#define IS_INTEL_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_INTEL_1 && \
|
||||
(env)->cpuid_vendor2 == CPUID_VENDOR_INTEL_2 && \
|
||||
(env)->cpuid_vendor3 == CPUID_VENDOR_INTEL_3)
|
||||
#define IS_AMD_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_AMD_1 && \
|
||||
(env)->cpuid_vendor2 == CPUID_VENDOR_AMD_2 && \
|
||||
(env)->cpuid_vendor3 == CPUID_VENDOR_AMD_3)
|
||||
static void x86_cpu_realizefn(DeviceState *dev, Error **errp)
|
||||
{
|
||||
CPUState *cs = CPU(dev);
|
||||
X86CPU *cpu = X86_CPU(dev);
|
||||
X86CPUClass *xcc = X86_CPU_GET_CLASS(dev);
|
||||
CPUX86State *env = &cpu->env;
|
||||
Error *local_err = NULL;
|
||||
static bool ht_warned;
|
||||
|
||||
if (xcc->kvm_required && !kvm_enabled()) {
|
||||
char *name = x86_cpu_class_get_model_name(xcc);
|
||||
error_setg(&local_err, "CPU model '%s' requires KVM", name);
|
||||
g_free(name);
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (cpu->apic_id == UNASSIGNED_APIC_ID) {
|
||||
error_setg(errp, "apic-id property was not initialized properly");
|
||||
return;
|
||||
}
|
||||
|
||||
x86_cpu_load_features(cpu, &local_err);
|
||||
if (local_err) {
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (x86_cpu_filter_features(cpu) &&
|
||||
(cpu->check_cpuid || cpu->enforce_cpuid)) {
|
||||
x86_cpu_report_filtered_features(cpu);
|
||||
if (cpu->enforce_cpuid) {
|
||||
error_setg(&local_err,
|
||||
kvm_enabled() ?
|
||||
"Host doesn't support requested features" :
|
||||
"TCG doesn't support requested features");
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
|
||||
/* On AMD CPUs, some CPUID[8000_0001].EDX bits must match the bits on
|
||||
* CPUID[1].EDX.
|
||||
*/
|
||||
@ -3332,27 +3445,22 @@ static void x86_cpu_register_feature_bit_props(X86CPU *cpu,
|
||||
FeatureWord w,
|
||||
int bitnr)
|
||||
{
|
||||
Object *obj = OBJECT(cpu);
|
||||
int i;
|
||||
char **names;
|
||||
FeatureWordInfo *fi = &feature_word_info[w];
|
||||
const char *name = fi->feat_names[bitnr];
|
||||
|
||||
if (!fi->feat_names[bitnr]) {
|
||||
if (!name) {
|
||||
return;
|
||||
}
|
||||
|
||||
names = g_strsplit(fi->feat_names[bitnr], "|", 0);
|
||||
|
||||
feat2prop(names[0]);
|
||||
x86_cpu_register_bit_prop(cpu, names[0], &cpu->env.features[w], bitnr);
|
||||
|
||||
for (i = 1; names[i]; i++) {
|
||||
feat2prop(names[i]);
|
||||
object_property_add_alias(obj, names[i], obj, names[0],
|
||||
&error_abort);
|
||||
}
|
||||
|
||||
g_strfreev(names);
|
||||
/* Property names should use "-" instead of "_".
|
||||
* Old names containing underscores are registered as aliases
|
||||
* using object_property_add_alias()
|
||||
*/
|
||||
assert(!strchr(name, '_'));
|
||||
/* aliases don't use "|" delimiters anymore, they are registered
|
||||
* manually using object_property_add_alias() */
|
||||
assert(!strchr(name, '|'));
|
||||
x86_cpu_register_bit_prop(cpu, name, &cpu->env.features[w], bitnr);
|
||||
}
|
||||
|
||||
static void x86_cpu_initfn(Object *obj)
|
||||
@ -3400,6 +3508,36 @@ static void x86_cpu_initfn(Object *obj)
|
||||
}
|
||||
}
|
||||
|
||||
object_property_add_alias(obj, "sse3", obj, "pni", &error_abort);
|
||||
object_property_add_alias(obj, "pclmuldq", obj, "pclmulqdq", &error_abort);
|
||||
object_property_add_alias(obj, "sse4-1", obj, "sse4.1", &error_abort);
|
||||
object_property_add_alias(obj, "sse4-2", obj, "sse4.2", &error_abort);
|
||||
object_property_add_alias(obj, "xd", obj, "nx", &error_abort);
|
||||
object_property_add_alias(obj, "ffxsr", obj, "fxsr-opt", &error_abort);
|
||||
object_property_add_alias(obj, "i64", obj, "lm", &error_abort);
|
||||
|
||||
object_property_add_alias(obj, "ds_cpl", obj, "ds-cpl", &error_abort);
|
||||
object_property_add_alias(obj, "tsc_adjust", obj, "tsc-adjust", &error_abort);
|
||||
object_property_add_alias(obj, "fxsr_opt", obj, "fxsr-opt", &error_abort);
|
||||
object_property_add_alias(obj, "lahf_lm", obj, "lahf-lm", &error_abort);
|
||||
object_property_add_alias(obj, "cmp_legacy", obj, "cmp-legacy", &error_abort);
|
||||
object_property_add_alias(obj, "nodeid_msr", obj, "nodeid-msr", &error_abort);
|
||||
object_property_add_alias(obj, "perfctr_core", obj, "perfctr-core", &error_abort);
|
||||
object_property_add_alias(obj, "perfctr_nb", obj, "perfctr-nb", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_nopiodelay", obj, "kvm-nopiodelay", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_mmu", obj, "kvm-mmu", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_asyncpf", obj, "kvm-asyncpf", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_steal_time", obj, "kvm-steal-time", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_pv_eoi", obj, "kvm-pv-eoi", &error_abort);
|
||||
object_property_add_alias(obj, "kvm_pv_unhalt", obj, "kvm-pv-unhalt", &error_abort);
|
||||
object_property_add_alias(obj, "svm_lock", obj, "svm-lock", &error_abort);
|
||||
object_property_add_alias(obj, "nrip_save", obj, "nrip-save", &error_abort);
|
||||
object_property_add_alias(obj, "tsc_scale", obj, "tsc-scale", &error_abort);
|
||||
object_property_add_alias(obj, "vmcb_clean", obj, "vmcb-clean", &error_abort);
|
||||
object_property_add_alias(obj, "pause_filter", obj, "pause-filter", &error_abort);
|
||||
object_property_add_alias(obj, "sse4_1", obj, "sse4.1", &error_abort);
|
||||
object_property_add_alias(obj, "sse4_2", obj, "sse4.2", &error_abort);
|
||||
|
||||
x86_cpu_load_def(cpu, xcc->cpu_def, &error_abort);
|
||||
}
|
||||
|
||||
@ -3537,11 +3675,6 @@ static void x86_cpu_common_class_init(ObjectClass *oc, void *data)
|
||||
cc->cpu_exec_exit = x86_cpu_exec_exit;
|
||||
|
||||
dc->cannot_instantiate_with_device_add_yet = false;
|
||||
/*
|
||||
* Reason: x86_cpu_initfn() calls cpu_exec_init(), which saves the
|
||||
* object in cpus -> dangling pointer after final object_unref().
|
||||
*/
|
||||
dc->cannot_destroy_with_object_finalize_yet = true;
|
||||
}
|
||||
|
||||
static const TypeInfo x86_cpu_type_info = {
|
||||
|
@ -25,6 +25,11 @@ bool kvm_has_smm(void)
|
||||
return 1;
|
||||
}
|
||||
|
||||
bool kvm_enable_x2apic(void)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
/* This function is only called inside conditionals which we
|
||||
* rely on the compiler to optimize out when CONFIG_KVM is not
|
||||
* defined.
|
||||
|
@ -122,6 +122,32 @@ bool kvm_allows_irq0_override(void)
|
||||
return !kvm_irqchip_in_kernel() || kvm_has_gsi_routing();
|
||||
}
|
||||
|
||||
static bool kvm_x2apic_api_set_flags(uint64_t flags)
|
||||
{
|
||||
KVMState *s = KVM_STATE(current_machine->accelerator);
|
||||
|
||||
return !kvm_vm_enable_cap(s, KVM_CAP_X2APIC_API, 0, flags);
|
||||
}
|
||||
|
||||
#define MEMORIZE(fn) \
|
||||
({ \
|
||||
static typeof(fn) _result; \
|
||||
static bool _memorized; \
|
||||
\
|
||||
if (_memorized) { \
|
||||
return _result; \
|
||||
} \
|
||||
_memorized = true; \
|
||||
_result = fn; \
|
||||
})
|
||||
|
||||
bool kvm_enable_x2apic(void)
|
||||
{
|
||||
return MEMORIZE(
|
||||
kvm_x2apic_api_set_flags(KVM_X2APIC_API_USE_32BIT_IDS |
|
||||
KVM_X2APIC_API_DISABLE_BROADCAST_QUIRK));
|
||||
}
|
||||
|
||||
static int kvm_get_tsc(CPUState *cs)
|
||||
{
|
||||
X86CPU *cpu = X86_CPU(cs);
|
||||
|
@ -43,4 +43,5 @@ int kvm_device_msix_deassign(KVMState *s, uint32_t dev_id);
|
||||
|
||||
void kvm_put_apicbase(X86CPU *cpu, uint64_t value);
|
||||
|
||||
bool kvm_enable_x2apic(void);
|
||||
#endif
|
||||
|
@ -3,6 +3,7 @@
|
||||
#include "qapi/qmp/qlist.h"
|
||||
#include "qapi/qmp/qdict.h"
|
||||
#include "qapi/qmp/qint.h"
|
||||
#include "qapi/qmp/qbool.h"
|
||||
#include "libqtest.h"
|
||||
|
||||
static char *get_cpu0_qom_path(void)
|
||||
@ -34,6 +35,15 @@ static QObject *qom_get(const char *path, const char *prop)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool qom_get_bool(const char *path, const char *prop)
|
||||
{
|
||||
QBool *value = qobject_to_qbool(qom_get(path, prop));
|
||||
bool b = qbool_get_bool(value);
|
||||
|
||||
QDECREF(value);
|
||||
return b;
|
||||
}
|
||||
|
||||
typedef struct CpuidTestArgs {
|
||||
const char *cmdline;
|
||||
const char *property;
|
||||
@ -66,10 +76,44 @@ static void add_cpuid_test(const char *name, const char *cmdline,
|
||||
qtest_add_data_func(name, args, test_cpuid_prop);
|
||||
}
|
||||
|
||||
static void test_plus_minus(void)
|
||||
{
|
||||
char *path;
|
||||
|
||||
/* Rules:
|
||||
* 1)"-foo" overrides "+foo"
|
||||
* 2) "[+-]foo" overrides "foo=..."
|
||||
* 3) Old feature names with underscores (e.g. "sse4_2")
|
||||
* should keep working
|
||||
*
|
||||
* Note: rules 1 and 2 are planned to be removed soon, but we
|
||||
* need to keep compatibility for a while until we start
|
||||
* warning users about it.
|
||||
*/
|
||||
qtest_start("-cpu pentium,-fpu,+fpu,-mce,mce=on,+cx8,cx8=off,+sse4_1,sse4_2=on");
|
||||
path = get_cpu0_qom_path();
|
||||
|
||||
g_assert_false(qom_get_bool(path, "fpu"));
|
||||
g_assert_false(qom_get_bool(path, "mce"));
|
||||
g_assert_true(qom_get_bool(path, "cx8"));
|
||||
|
||||
/* Test both the original and the alias feature names: */
|
||||
g_assert_true(qom_get_bool(path, "sse4-1"));
|
||||
g_assert_true(qom_get_bool(path, "sse4.1"));
|
||||
|
||||
g_assert_true(qom_get_bool(path, "sse4-2"));
|
||||
g_assert_true(qom_get_bool(path, "sse4.2"));
|
||||
|
||||
qtest_end();
|
||||
g_free(path);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
g_test_init(&argc, &argv, NULL);
|
||||
|
||||
qtest_add_func("x86/cpuid/parsing-plus-minus", test_plus_minus);
|
||||
|
||||
/* Original level values for CPU models: */
|
||||
add_cpuid_test("x86/cpuid/phenom/level",
|
||||
"-cpu phenom", "level", 5);
|
||||
|
Loading…
x
Reference in New Issue
Block a user