blob: 25a3d80221e623086399e2c8b93016077e1474b6 [file] [log] [blame]
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001/*
2 * i386 CPUID helper functions
3 *
4 * Copyright (c) 2003 Fabrice Bellard
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18 */
Peter Maydell1ef26b12016-02-23 11:58:02 +000019#include "qemu/osdep.h"
Veronia Bahaaf348b6d2016-03-20 19:16:19 +020020#include "qemu/cutils.h"
Andre Przywarac6dc6f62010-03-11 14:38:55 +010021
22#include "cpu.h"
Paolo Bonzini63c91552016-03-15 13:18:37 +010023#include "exec/exec-all.h"
Paolo Bonzini9c17d612012-12-17 18:20:04 +010024#include "sysemu/kvm.h"
Eduardo Habkost8932cfd2013-01-22 18:25:09 -020025#include "sysemu/cpus.h"
Paolo Bonzini50a2c6e2013-03-20 13:11:56 +010026#include "kvm_i386.h"
Andre Przywarac6dc6f62010-03-11 14:38:55 +010027
Markus Armbrusterd49b6832015-03-17 18:29:20 +010028#include "qemu/error-report.h"
Paolo Bonzini1de7afc2012-12-17 18:20:00 +010029#include "qemu/option.h"
30#include "qemu/config-file.h"
Paolo Bonzini7b1b5d12012-12-17 18:19:43 +010031#include "qapi/qmp/qerror.h"
Andre Przywarac6dc6f62010-03-11 14:38:55 +010032
Eduardo Habkost8e8aba52013-05-06 13:20:07 -030033#include "qapi-types.h"
34#include "qapi-visit.h"
Paolo Bonzini7b1b5d12012-12-17 18:19:43 +010035#include "qapi/visitor.h"
Paolo Bonzini9c17d612012-12-17 18:20:04 +010036#include "sysemu/arch_init.h"
Andreas Färber71ad61d2012-04-17 12:10:29 +020037
Stefan Weilb834b502012-08-30 22:28:31 +020038#if defined(CONFIG_KVM)
Anthony Liguorief8621b2012-08-29 09:32:41 -050039#include <linux/kvm_para.h>
Stefan Weilb834b502012-08-30 22:28:31 +020040#endif
Igor Mammedov65dee382012-07-23 15:22:28 +020041
Paolo Bonzini9c17d612012-12-17 18:20:04 +010042#include "sysemu/sysemu.h"
Igor Mammedov53a89e22013-04-29 19:03:01 +020043#include "hw/qdev-properties.h"
Radim Krčmář5232d002016-05-12 19:24:26 +020044#include "hw/i386/topology.h"
Igor Mammedovbdeec802012-10-13 22:35:39 +020045#ifndef CONFIG_USER_ONLY
Paolo Bonzini2001d0c2015-03-31 14:11:09 +020046#include "exec/address-spaces.h"
Paolo Bonzini741da0d2014-06-27 08:40:04 +020047#include "hw/hw.h"
Paolo Bonzini0d09e412013-02-05 17:06:20 +010048#include "hw/xen/xen.h"
Paolo Bonzini0d09e412013-02-05 17:06:20 +010049#include "hw/i386/apic_internal.h"
Igor Mammedovbdeec802012-10-13 22:35:39 +020050#endif
51
Eduardo Habkost5e891bf2013-08-27 12:24:37 -030052
53/* Cache topology CPUID constants: */
54
55/* CPUID Leaf 2 Descriptors */
56
57#define CPUID_2_L1D_32KB_8WAY_64B 0x2c
58#define CPUID_2_L1I_32KB_8WAY_64B 0x30
59#define CPUID_2_L2_2MB_8WAY_64B 0x7d
Longpeng(Mike)14c985c2016-09-07 13:21:13 +080060#define CPUID_2_L3_16MB_16WAY_64B 0x4d
Eduardo Habkost5e891bf2013-08-27 12:24:37 -030061
62
63/* CPUID Leaf 4 constants: */
64
65/* EAX: */
66#define CPUID_4_TYPE_DCACHE 1
67#define CPUID_4_TYPE_ICACHE 2
68#define CPUID_4_TYPE_UNIFIED 3
69
70#define CPUID_4_LEVEL(l) ((l) << 5)
71
72#define CPUID_4_SELF_INIT_LEVEL (1 << 8)
73#define CPUID_4_FULLY_ASSOC (1 << 9)
74
75/* EDX: */
76#define CPUID_4_NO_INVD_SHARING (1 << 0)
77#define CPUID_4_INCLUSIVE (1 << 1)
78#define CPUID_4_COMPLEX_IDX (1 << 2)
79
80#define ASSOC_FULL 0xFF
81
82/* AMD associativity encoding used on CPUID Leaf 0x80000006: */
83#define AMD_ENC_ASSOC(a) (a <= 1 ? a : \
84 a == 2 ? 0x2 : \
85 a == 4 ? 0x4 : \
86 a == 8 ? 0x6 : \
87 a == 16 ? 0x8 : \
88 a == 32 ? 0xA : \
89 a == 48 ? 0xB : \
90 a == 64 ? 0xC : \
91 a == 96 ? 0xD : \
92 a == 128 ? 0xE : \
93 a == ASSOC_FULL ? 0xF : \
94 0 /* invalid value */)
95
96
97/* Definitions of the hardcoded cache entries we expose: */
98
99/* L1 data cache: */
100#define L1D_LINE_SIZE 64
101#define L1D_ASSOCIATIVITY 8
102#define L1D_SETS 64
103#define L1D_PARTITIONS 1
104/* Size = LINE_SIZE*ASSOCIATIVITY*SETS*PARTITIONS = 32KiB */
105#define L1D_DESCRIPTOR CPUID_2_L1D_32KB_8WAY_64B
106/*FIXME: CPUID leaf 0x80000005 is inconsistent with leaves 2 & 4 */
107#define L1D_LINES_PER_TAG 1
108#define L1D_SIZE_KB_AMD 64
109#define L1D_ASSOCIATIVITY_AMD 2
110
111/* L1 instruction cache: */
112#define L1I_LINE_SIZE 64
113#define L1I_ASSOCIATIVITY 8
114#define L1I_SETS 64
115#define L1I_PARTITIONS 1
116/* Size = LINE_SIZE*ASSOCIATIVITY*SETS*PARTITIONS = 32KiB */
117#define L1I_DESCRIPTOR CPUID_2_L1I_32KB_8WAY_64B
118/*FIXME: CPUID leaf 0x80000005 is inconsistent with leaves 2 & 4 */
119#define L1I_LINES_PER_TAG 1
120#define L1I_SIZE_KB_AMD 64
121#define L1I_ASSOCIATIVITY_AMD 2
122
123/* Level 2 unified cache: */
124#define L2_LINE_SIZE 64
125#define L2_ASSOCIATIVITY 16
126#define L2_SETS 4096
127#define L2_PARTITIONS 1
128/* Size = LINE_SIZE*ASSOCIATIVITY*SETS*PARTITIONS = 4MiB */
129/*FIXME: CPUID leaf 2 descriptor is inconsistent with CPUID leaf 4 */
130#define L2_DESCRIPTOR CPUID_2_L2_2MB_8WAY_64B
131/*FIXME: CPUID leaf 0x80000006 is inconsistent with leaves 2 & 4 */
132#define L2_LINES_PER_TAG 1
133#define L2_SIZE_KB_AMD 512
134
Longpeng(Mike)14c985c2016-09-07 13:21:13 +0800135/* Level 3 unified cache: */
Eduardo Habkost5e891bf2013-08-27 12:24:37 -0300136#define L3_SIZE_KB 0 /* disabled */
137#define L3_ASSOCIATIVITY 0 /* disabled */
138#define L3_LINES_PER_TAG 0 /* disabled */
139#define L3_LINE_SIZE 0 /* disabled */
Longpeng(Mike)14c985c2016-09-07 13:21:13 +0800140#define L3_N_LINE_SIZE 64
141#define L3_N_ASSOCIATIVITY 16
142#define L3_N_SETS 16384
143#define L3_N_PARTITIONS 1
144#define L3_N_DESCRIPTOR CPUID_2_L3_16MB_16WAY_64B
145#define L3_N_LINES_PER_TAG 1
146#define L3_N_SIZE_KB_AMD 16384
Eduardo Habkost5e891bf2013-08-27 12:24:37 -0300147
148/* TLB definitions: */
149
150#define L1_DTLB_2M_ASSOC 1
151#define L1_DTLB_2M_ENTRIES 255
152#define L1_DTLB_4K_ASSOC 1
153#define L1_DTLB_4K_ENTRIES 255
154
155#define L1_ITLB_2M_ASSOC 1
156#define L1_ITLB_2M_ENTRIES 255
157#define L1_ITLB_4K_ASSOC 1
158#define L1_ITLB_4K_ENTRIES 255
159
160#define L2_DTLB_2M_ASSOC 0 /* disabled */
161#define L2_DTLB_2M_ENTRIES 0 /* disabled */
162#define L2_DTLB_4K_ASSOC 4
163#define L2_DTLB_4K_ENTRIES 512
164
165#define L2_ITLB_2M_ASSOC 0 /* disabled */
166#define L2_ITLB_2M_ENTRIES 0 /* disabled */
167#define L2_ITLB_4K_ASSOC 4
168#define L2_ITLB_4K_ENTRIES 512
169
170
171
Igor Mammedov99b88a12013-01-21 15:06:36 +0100172static void x86_cpu_vendor_words2str(char *dst, uint32_t vendor1,
173 uint32_t vendor2, uint32_t vendor3)
174{
175 int i;
176 for (i = 0; i < 4; i++) {
177 dst[i] = vendor1 >> (8 * i);
178 dst[i + 4] = vendor2 >> (8 * i);
179 dst[i + 8] = vendor3 >> (8 * i);
180 }
181 dst[CPUID_VENDOR_SZ] = '\0';
182}
183
Eduardo Habkost621626c2014-04-30 13:48:36 -0300184#define I486_FEATURES (CPUID_FP87 | CPUID_VME | CPUID_PSE)
185#define PENTIUM_FEATURES (I486_FEATURES | CPUID_DE | CPUID_TSC | \
186 CPUID_MSR | CPUID_MCE | CPUID_CX8 | CPUID_MMX | CPUID_APIC)
187#define PENTIUM2_FEATURES (PENTIUM_FEATURES | CPUID_PAE | CPUID_SEP | \
188 CPUID_MTRR | CPUID_PGE | CPUID_MCA | CPUID_CMOV | CPUID_PAT | \
189 CPUID_PSE36 | CPUID_FXSR)
190#define PENTIUM3_FEATURES (PENTIUM2_FEATURES | CPUID_SSE)
191#define PPRO_FEATURES (CPUID_FP87 | CPUID_DE | CPUID_PSE | CPUID_TSC | \
192 CPUID_MSR | CPUID_MCE | CPUID_CX8 | CPUID_PGE | CPUID_CMOV | \
193 CPUID_PAT | CPUID_FXSR | CPUID_MMX | CPUID_SSE | CPUID_SSE2 | \
194 CPUID_PAE | CPUID_SEP | CPUID_APIC)
195
196#define TCG_FEATURES (CPUID_FP87 | CPUID_PSE | CPUID_TSC | CPUID_MSR | \
197 CPUID_PAE | CPUID_MCE | CPUID_CX8 | CPUID_APIC | CPUID_SEP | \
198 CPUID_MTRR | CPUID_PGE | CPUID_MCA | CPUID_CMOV | CPUID_PAT | \
199 CPUID_PSE36 | CPUID_CLFLUSH | CPUID_ACPI | CPUID_MMX | \
Eduardo Habkostb6c5a6f2015-10-07 16:39:43 -0300200 CPUID_FXSR | CPUID_SSE | CPUID_SSE2 | CPUID_SS | CPUID_DE)
Eduardo Habkost621626c2014-04-30 13:48:36 -0300201 /* partly implemented:
202 CPUID_MTRR, CPUID_MCA, CPUID_CLFLUSH (needed for Win64) */
203 /* missing:
204 CPUID_VME, CPUID_DTS, CPUID_SS, CPUID_HT, CPUID_TM, CPUID_PBE */
205#define TCG_EXT_FEATURES (CPUID_EXT_SSE3 | CPUID_EXT_PCLMULQDQ | \
206 CPUID_EXT_MONITOR | CPUID_EXT_SSSE3 | CPUID_EXT_CX16 | \
207 CPUID_EXT_SSE41 | CPUID_EXT_SSE42 | CPUID_EXT_POPCNT | \
Richard Henderson19dc85d2015-07-02 14:53:40 +0100208 CPUID_EXT_XSAVE | /* CPUID_EXT_OSXSAVE is dynamic */ \
Eduardo Habkost621626c2014-04-30 13:48:36 -0300209 CPUID_EXT_MOVBE | CPUID_EXT_AES | CPUID_EXT_HYPERVISOR)
210 /* missing:
211 CPUID_EXT_DTES64, CPUID_EXT_DSCPL, CPUID_EXT_VMX, CPUID_EXT_SMX,
212 CPUID_EXT_EST, CPUID_EXT_TM2, CPUID_EXT_CID, CPUID_EXT_FMA,
213 CPUID_EXT_XTPR, CPUID_EXT_PDCM, CPUID_EXT_PCID, CPUID_EXT_DCA,
Richard Henderson19dc85d2015-07-02 14:53:40 +0100214 CPUID_EXT_X2APIC, CPUID_EXT_TSC_DEADLINE_TIMER, CPUID_EXT_AVX,
215 CPUID_EXT_F16C, CPUID_EXT_RDRAND */
Eduardo Habkost621626c2014-04-30 13:48:36 -0300216
217#ifdef TARGET_X86_64
218#define TCG_EXT2_X86_64_FEATURES (CPUID_EXT2_SYSCALL | CPUID_EXT2_LM)
219#else
220#define TCG_EXT2_X86_64_FEATURES 0
221#endif
222
223#define TCG_EXT2_FEATURES ((TCG_FEATURES & CPUID_EXT2_AMD_ALIASES) | \
224 CPUID_EXT2_NX | CPUID_EXT2_MMXEXT | CPUID_EXT2_RDTSCP | \
225 CPUID_EXT2_3DNOW | CPUID_EXT2_3DNOWEXT | CPUID_EXT2_PDPE1GB | \
226 TCG_EXT2_X86_64_FEATURES)
227#define TCG_EXT3_FEATURES (CPUID_EXT3_LAHF_LM | CPUID_EXT3_SVM | \
228 CPUID_EXT3_CR8LEG | CPUID_EXT3_ABM | CPUID_EXT3_SSE4A)
229#define TCG_EXT4_FEATURES 0
230#define TCG_SVM_FEATURES 0
231#define TCG_KVM_FEATURES 0
232#define TCG_7_0_EBX_FEATURES (CPUID_7_0_EBX_SMEP | CPUID_7_0_EBX_SMAP | \
Xiao Guangrong0c472422015-10-29 15:31:39 +0800233 CPUID_7_0_EBX_BMI1 | CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ADX | \
234 CPUID_7_0_EBX_PCOMMIT | CPUID_7_0_EBX_CLFLUSHOPT | \
Paolo Bonzini7eb24382016-06-17 17:47:13 +0200235 CPUID_7_0_EBX_CLWB | CPUID_7_0_EBX_MPX | CPUID_7_0_EBX_FSGSBASE | \
236 CPUID_7_0_EBX_ERMS)
Eduardo Habkost621626c2014-04-30 13:48:36 -0300237 /* missing:
Richard Henderson07929f22015-11-18 12:55:47 +0100238 CPUID_7_0_EBX_HLE, CPUID_7_0_EBX_AVX2,
Paolo Bonzini7eb24382016-06-17 17:47:13 +0200239 CPUID_7_0_EBX_INVPCID, CPUID_7_0_EBX_RTM,
Eduardo Habkost621626c2014-04-30 13:48:36 -0300240 CPUID_7_0_EBX_RDSEED */
Paolo Bonzini0f70ed42016-02-09 14:14:28 +0100241#define TCG_7_0_ECX_FEATURES (CPUID_7_0_ECX_PKU | CPUID_7_0_ECX_OSPKE)
Marcelo Tosatti303752a2014-04-30 13:48:45 -0300242#define TCG_APM_FEATURES 0
Jan Kiszka28b8e4d2015-06-07 11:15:08 +0200243#define TCG_6_EAX_FEATURES CPUID_6_EAX_ARAT
Richard Hendersonc9cfe8f2015-07-02 15:21:23 +0100244#define TCG_XSAVE_FEATURES (CPUID_XSAVE_XSAVEOPT | CPUID_XSAVE_XGETBV1)
245 /* missing:
246 CPUID_XSAVE_XSAVEC, CPUID_XSAVE_XSAVES */
Eduardo Habkost621626c2014-04-30 13:48:36 -0300247
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200248typedef struct FeatureWordInfo {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300249 /* feature flags names are taken from "Intel Processor Identification and
250 * the CPUID Instruction" and AMD's "CPUID Specification".
251 * In cases of disagreement between feature naming conventions,
252 * aliases may be added.
253 */
254 const char *feat_names[32];
Eduardo Habkost04d104b2013-04-22 16:00:16 -0300255 uint32_t cpuid_eax; /* Input EAX for CPUID */
256 bool cpuid_needs_ecx; /* CPUID instruction uses ECX as input */
257 uint32_t cpuid_ecx; /* Input ECX value for CPUID */
258 int cpuid_reg; /* output register (R_* constant) */
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300259 uint32_t tcg_features; /* Feature flags supported by TCG */
Eduardo Habkost84f1b922014-04-30 13:48:41 -0300260 uint32_t unmigratable_flags; /* Feature flags known to be unmigratable */
Eduardo Habkost6fb2fff2016-09-28 12:00:29 -0300261 uint32_t migratable_flags; /* Feature flags known to be migratable */
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200262} FeatureWordInfo;
263
264static FeatureWordInfo feature_word_info[FEATURE_WORDS] = {
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200265 [FEAT_1_EDX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300266 .feat_names = {
267 "fpu", "vme", "de", "pse",
268 "tsc", "msr", "pae", "mce",
269 "cx8", "apic", NULL, "sep",
270 "mtrr", "pge", "mca", "cmov",
271 "pat", "pse36", "pn" /* Intel psn */, "clflush" /* Intel clfsh */,
272 NULL, "ds" /* Intel dts */, "acpi", "mmx",
273 "fxsr", "sse", "sse2", "ss",
274 "ht" /* Intel htt */, "tm", "ia64", "pbe",
275 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200276 .cpuid_eax = 1, .cpuid_reg = R_EDX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300277 .tcg_features = TCG_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200278 },
279 [FEAT_1_ECX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300280 .feat_names = {
281 "pni|sse3" /* Intel,AMD sse3 */, "pclmulqdq|pclmuldq", "dtes64", "monitor",
282 "ds_cpl", "vmx", "smx", "est",
283 "tm2", "ssse3", "cid", NULL,
284 "fma", "cx16", "xtpr", "pdcm",
285 NULL, "pcid", "dca", "sse4.1|sse4_1",
286 "sse4.2|sse4_2", "x2apic", "movbe", "popcnt",
287 "tsc-deadline", "aes", "xsave", "osxsave",
288 "avx", "f16c", "rdrand", "hypervisor",
289 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200290 .cpuid_eax = 1, .cpuid_reg = R_ECX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300291 .tcg_features = TCG_EXT_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200292 },
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300293 /* Feature names that are already defined on feature_name[] but
294 * are set on CPUID[8000_0001].EDX on AMD CPUs don't have their
295 * names on feat_names below. They are copied automatically
296 * to features[FEAT_8000_0001_EDX] if and only if CPU vendor is AMD.
297 */
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200298 [FEAT_8000_0001_EDX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300299 .feat_names = {
300 NULL /* fpu */, NULL /* vme */, NULL /* de */, NULL /* pse */,
301 NULL /* tsc */, NULL /* msr */, NULL /* pae */, NULL /* mce */,
302 NULL /* cx8 */, NULL /* apic */, NULL, "syscall",
303 NULL /* mtrr */, NULL /* pge */, NULL /* mca */, NULL /* cmov */,
304 NULL /* pat */, NULL /* pse36 */, NULL, NULL /* Linux mp */,
305 "nx|xd", NULL, "mmxext", NULL /* mmx */,
306 NULL /* fxsr */, "fxsr_opt|ffxsr", "pdpe1gb", "rdtscp",
307 NULL, "lm|i64", "3dnowext", "3dnow",
308 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200309 .cpuid_eax = 0x80000001, .cpuid_reg = R_EDX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300310 .tcg_features = TCG_EXT2_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200311 },
312 [FEAT_8000_0001_ECX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300313 .feat_names = {
314 "lahf_lm", "cmp_legacy", "svm", "extapic",
315 "cr8legacy", "abm", "sse4a", "misalignsse",
316 "3dnowprefetch", "osvw", "ibs", "xop",
317 "skinit", "wdt", NULL, "lwp",
318 "fma4", "tce", NULL, "nodeid_msr",
319 NULL, "tbm", "topoext", "perfctr_core",
320 "perfctr_nb", NULL, NULL, NULL,
321 NULL, NULL, NULL, NULL,
322 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200323 .cpuid_eax = 0x80000001, .cpuid_reg = R_ECX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300324 .tcg_features = TCG_EXT3_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200325 },
Eduardo Habkost89e49c82013-01-07 16:20:47 -0200326 [FEAT_C000_0001_EDX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300327 .feat_names = {
328 NULL, NULL, "xstore", "xstore-en",
329 NULL, NULL, "xcrypt", "xcrypt-en",
330 "ace2", "ace2-en", "phe", "phe-en",
331 "pmm", "pmm-en", NULL, NULL,
332 NULL, NULL, NULL, NULL,
333 NULL, NULL, NULL, NULL,
334 NULL, NULL, NULL, NULL,
335 NULL, NULL, NULL, NULL,
336 },
Eduardo Habkost89e49c82013-01-07 16:20:47 -0200337 .cpuid_eax = 0xC0000001, .cpuid_reg = R_EDX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300338 .tcg_features = TCG_EXT4_FEATURES,
Eduardo Habkost89e49c82013-01-07 16:20:47 -0200339 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200340 [FEAT_KVM] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300341 .feat_names = {
342 "kvmclock", "kvm_nopiodelay", "kvm_mmu", "kvmclock",
343 "kvm_asyncpf", "kvm_steal_time", "kvm_pv_eoi", "kvm_pv_unhalt",
344 NULL, NULL, NULL, NULL,
345 NULL, NULL, NULL, NULL,
346 NULL, NULL, NULL, NULL,
347 NULL, NULL, NULL, NULL,
348 "kvmclock-stable-bit", NULL, NULL, NULL,
349 NULL, NULL, NULL, NULL,
350 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200351 .cpuid_eax = KVM_CPUID_FEATURES, .cpuid_reg = R_EAX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300352 .tcg_features = TCG_KVM_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200353 },
Evgeny Yakovlevc35bd192016-06-24 13:49:36 +0300354 [FEAT_HYPERV_EAX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300355 .feat_names = {
356 NULL /* hv_msr_vp_runtime_access */, NULL /* hv_msr_time_refcount_access */,
357 NULL /* hv_msr_synic_access */, NULL /* hv_msr_stimer_access */,
358 NULL /* hv_msr_apic_access */, NULL /* hv_msr_hypercall_access */,
359 NULL /* hv_vpindex_access */, NULL /* hv_msr_reset_access */,
360 NULL /* hv_msr_stats_access */, NULL /* hv_reftsc_access */,
361 NULL /* hv_msr_idle_access */, NULL /* hv_msr_frequency_access */,
362 NULL, NULL, NULL, NULL,
363 NULL, NULL, NULL, NULL,
364 NULL, NULL, NULL, NULL,
365 NULL, NULL, NULL, NULL,
366 NULL, NULL, NULL, NULL,
367 },
Evgeny Yakovlevc35bd192016-06-24 13:49:36 +0300368 .cpuid_eax = 0x40000003, .cpuid_reg = R_EAX,
369 },
370 [FEAT_HYPERV_EBX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300371 .feat_names = {
372 NULL /* hv_create_partitions */, NULL /* hv_access_partition_id */,
373 NULL /* hv_access_memory_pool */, NULL /* hv_adjust_message_buffers */,
374 NULL /* hv_post_messages */, NULL /* hv_signal_events */,
375 NULL /* hv_create_port */, NULL /* hv_connect_port */,
376 NULL /* hv_access_stats */, NULL, NULL, NULL /* hv_debugging */,
377 NULL /* hv_cpu_power_management */, NULL /* hv_configure_profiler */,
378 NULL, NULL,
379 NULL, NULL, NULL, NULL,
380 NULL, NULL, NULL, NULL,
381 NULL, NULL, NULL, NULL,
382 NULL, NULL, NULL, NULL,
383 },
Evgeny Yakovlevc35bd192016-06-24 13:49:36 +0300384 .cpuid_eax = 0x40000003, .cpuid_reg = R_EBX,
385 },
386 [FEAT_HYPERV_EDX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300387 .feat_names = {
388 NULL /* hv_mwait */, NULL /* hv_guest_debugging */,
389 NULL /* hv_perf_monitor */, NULL /* hv_cpu_dynamic_part */,
390 NULL /* hv_hypercall_params_xmm */, NULL /* hv_guest_idle_state */,
391 NULL, NULL,
392 NULL, NULL, NULL /* hv_guest_crash_msr */, NULL,
393 NULL, NULL, NULL, NULL,
394 NULL, NULL, NULL, NULL,
395 NULL, NULL, NULL, NULL,
396 NULL, NULL, NULL, NULL,
397 NULL, NULL, NULL, NULL,
398 },
Evgeny Yakovlevc35bd192016-06-24 13:49:36 +0300399 .cpuid_eax = 0x40000003, .cpuid_reg = R_EDX,
400 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200401 [FEAT_SVM] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300402 .feat_names = {
403 "npt", "lbrv", "svm_lock", "nrip_save",
404 "tsc_scale", "vmcb_clean", "flushbyasid", "decodeassists",
405 NULL, NULL, "pause_filter", NULL,
406 "pfthreshold", NULL, NULL, NULL,
407 NULL, NULL, NULL, NULL,
408 NULL, NULL, NULL, NULL,
409 NULL, NULL, NULL, NULL,
410 NULL, NULL, NULL, NULL,
411 },
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200412 .cpuid_eax = 0x8000000A, .cpuid_reg = R_EDX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300413 .tcg_features = TCG_SVM_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200414 },
415 [FEAT_7_0_EBX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300416 .feat_names = {
417 "fsgsbase", "tsc_adjust", NULL, "bmi1",
418 "hle", "avx2", NULL, "smep",
419 "bmi2", "erms", "invpcid", "rtm",
420 NULL, NULL, "mpx", NULL,
421 "avx512f", "avx512dq", "rdseed", "adx",
422 "smap", "avx512ifma", "pcommit", "clflushopt",
423 "clwb", NULL, "avx512pf", "avx512er",
424 "avx512cd", NULL, "avx512bw", "avx512vl",
425 },
Eduardo Habkost04d104b2013-04-22 16:00:16 -0300426 .cpuid_eax = 7,
427 .cpuid_needs_ecx = true, .cpuid_ecx = 0,
428 .cpuid_reg = R_EBX,
Eduardo Habkost37ce3522014-04-30 13:48:38 -0300429 .tcg_features = TCG_7_0_EBX_FEATURES,
Eduardo Habkostbffd67b2013-01-07 16:20:46 -0200430 },
Huaitong Hanf74eefe2015-11-18 10:20:15 +0800431 [FEAT_7_0_ECX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300432 .feat_names = {
433 NULL, "avx512vbmi", "umip", "pku",
434 "ospke", NULL, NULL, NULL,
435 NULL, NULL, NULL, NULL,
436 NULL, NULL, NULL, NULL,
437 NULL, NULL, NULL, NULL,
438 NULL, NULL, "rdpid", NULL,
439 NULL, NULL, NULL, NULL,
440 NULL, NULL, NULL, NULL,
441 },
Huaitong Hanf74eefe2015-11-18 10:20:15 +0800442 .cpuid_eax = 7,
443 .cpuid_needs_ecx = true, .cpuid_ecx = 0,
444 .cpuid_reg = R_ECX,
445 .tcg_features = TCG_7_0_ECX_FEATURES,
446 },
Marcelo Tosatti303752a2014-04-30 13:48:45 -0300447 [FEAT_8000_0007_EDX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300448 .feat_names = {
449 NULL, NULL, NULL, NULL,
450 NULL, NULL, NULL, NULL,
451 "invtsc", NULL, NULL, NULL,
452 NULL, NULL, NULL, NULL,
453 NULL, NULL, NULL, NULL,
454 NULL, NULL, NULL, NULL,
455 NULL, NULL, NULL, NULL,
456 NULL, NULL, NULL, NULL,
457 },
Marcelo Tosatti303752a2014-04-30 13:48:45 -0300458 .cpuid_eax = 0x80000007,
459 .cpuid_reg = R_EDX,
460 .tcg_features = TCG_APM_FEATURES,
461 .unmigratable_flags = CPUID_APM_INVTSC,
462 },
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +0100463 [FEAT_XSAVE] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300464 .feat_names = {
465 "xsaveopt", "xsavec", "xgetbv1", "xsaves",
466 NULL, NULL, NULL, NULL,
467 NULL, NULL, NULL, NULL,
468 NULL, NULL, NULL, NULL,
469 NULL, NULL, NULL, NULL,
470 NULL, NULL, NULL, NULL,
471 NULL, NULL, NULL, NULL,
472 NULL, NULL, NULL, NULL,
473 },
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +0100474 .cpuid_eax = 0xd,
475 .cpuid_needs_ecx = true, .cpuid_ecx = 1,
476 .cpuid_reg = R_EAX,
Richard Hendersonc9cfe8f2015-07-02 15:21:23 +0100477 .tcg_features = TCG_XSAVE_FEATURES,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +0100478 },
Jan Kiszka28b8e4d2015-06-07 11:15:08 +0200479 [FEAT_6_EAX] = {
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300480 .feat_names = {
481 NULL, NULL, "arat", NULL,
482 NULL, NULL, NULL, NULL,
483 NULL, NULL, NULL, NULL,
484 NULL, NULL, NULL, NULL,
485 NULL, NULL, NULL, NULL,
486 NULL, NULL, NULL, NULL,
487 NULL, NULL, NULL, NULL,
488 NULL, NULL, NULL, NULL,
489 },
Jan Kiszka28b8e4d2015-06-07 11:15:08 +0200490 .cpuid_eax = 6, .cpuid_reg = R_EAX,
491 .tcg_features = TCG_6_EAX_FEATURES,
492 },
Eduardo Habkost96193c22016-09-22 17:41:35 -0300493 [FEAT_XSAVE_COMP_LO] = {
494 .cpuid_eax = 0xD,
495 .cpuid_needs_ecx = true, .cpuid_ecx = 0,
496 .cpuid_reg = R_EAX,
497 .tcg_features = ~0U,
Eduardo Habkost6fb2fff2016-09-28 12:00:29 -0300498 .migratable_flags = XSTATE_FP_MASK | XSTATE_SSE_MASK |
499 XSTATE_YMM_MASK | XSTATE_BNDREGS_MASK | XSTATE_BNDCSR_MASK |
500 XSTATE_OPMASK_MASK | XSTATE_ZMM_Hi256_MASK | XSTATE_Hi16_ZMM_MASK |
501 XSTATE_PKRU_MASK,
Eduardo Habkost96193c22016-09-22 17:41:35 -0300502 },
503 [FEAT_XSAVE_COMP_HI] = {
504 .cpuid_eax = 0xD,
505 .cpuid_needs_ecx = true, .cpuid_ecx = 0,
506 .cpuid_reg = R_EDX,
507 .tcg_features = ~0U,
508 },
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200509};
510
Eduardo Habkost8e8aba52013-05-06 13:20:07 -0300511typedef struct X86RegisterInfo32 {
512 /* Name of register */
513 const char *name;
514 /* QAPI enum value register */
515 X86CPURegister32 qapi_enum;
516} X86RegisterInfo32;
517
518#define REGISTER(reg) \
Wenchao Xia5d371f42014-03-04 18:44:40 -0800519 [R_##reg] = { .name = #reg, .qapi_enum = X86_CPU_REGISTER32_##reg }
Stefan Weila443bc32014-03-16 15:03:41 +0100520static const X86RegisterInfo32 x86_reg_info_32[CPU_NB_REGS32] = {
Eduardo Habkost8e8aba52013-05-06 13:20:07 -0300521 REGISTER(EAX),
522 REGISTER(ECX),
523 REGISTER(EDX),
524 REGISTER(EBX),
525 REGISTER(ESP),
526 REGISTER(EBP),
527 REGISTER(ESI),
528 REGISTER(EDI),
529};
530#undef REGISTER
531
Richard Henderson3f32bd22016-07-06 13:35:00 -0700532typedef struct ExtSaveArea {
533 uint32_t feature, bits;
534 uint32_t offset, size;
535} ExtSaveArea;
536
537static const ExtSaveArea x86_ext_save_areas[] = {
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100538 [XSTATE_YMM_BIT] =
539 { .feature = FEAT_1_ECX, .bits = CPUID_EXT_AVX,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200540 .offset = offsetof(X86XSaveArea, avx_state),
541 .size = sizeof(XSaveAVX) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100542 [XSTATE_BNDREGS_BIT] =
543 { .feature = FEAT_7_0_EBX, .bits = CPUID_7_0_EBX_MPX,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200544 .offset = offsetof(X86XSaveArea, bndreg_state),
545 .size = sizeof(XSaveBNDREG) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100546 [XSTATE_BNDCSR_BIT] =
547 { .feature = FEAT_7_0_EBX, .bits = CPUID_7_0_EBX_MPX,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200548 .offset = offsetof(X86XSaveArea, bndcsr_state),
549 .size = sizeof(XSaveBNDCSR) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100550 [XSTATE_OPMASK_BIT] =
551 { .feature = FEAT_7_0_EBX, .bits = CPUID_7_0_EBX_AVX512F,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200552 .offset = offsetof(X86XSaveArea, opmask_state),
553 .size = sizeof(XSaveOpmask) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100554 [XSTATE_ZMM_Hi256_BIT] =
555 { .feature = FEAT_7_0_EBX, .bits = CPUID_7_0_EBX_AVX512F,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200556 .offset = offsetof(X86XSaveArea, zmm_hi256_state),
557 .size = sizeof(XSaveZMM_Hi256) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100558 [XSTATE_Hi16_ZMM_BIT] =
559 { .feature = FEAT_7_0_EBX, .bits = CPUID_7_0_EBX_AVX512F,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200560 .offset = offsetof(X86XSaveArea, hi16_zmm_state),
561 .size = sizeof(XSaveHi16_ZMM) },
Paolo Bonzinicfc3b072016-02-17 10:54:53 +0100562 [XSTATE_PKRU_BIT] =
563 { .feature = FEAT_7_0_ECX, .bits = CPUID_7_0_ECX_PKU,
Eduardo Habkostee1b09f2015-11-28 14:32:26 -0200564 .offset = offsetof(X86XSaveArea, pkru_state),
565 .size = sizeof(XSavePKRU) },
Paolo Bonzini2560f192013-10-02 17:54:57 +0200566};
Eduardo Habkost8e8aba52013-05-06 13:20:07 -0300567
Eduardo Habkost1fda6192016-09-22 17:14:07 -0300568static uint32_t xsave_area_size(uint64_t mask)
569{
570 int i;
571 uint64_t ret = sizeof(X86LegacyXSaveArea) + sizeof(X86XSaveHeader);
572
573 for (i = 2; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
574 const ExtSaveArea *esa = &x86_ext_save_areas[i];
575 if ((mask >> i) & 1) {
576 ret = MAX(ret, esa->offset + esa->size);
577 }
578 }
579 return ret;
580}
581
Eduardo Habkost96193c22016-09-22 17:41:35 -0300582static inline uint64_t x86_cpu_xsave_components(X86CPU *cpu)
583{
584 return ((uint64_t)cpu->env.features[FEAT_XSAVE_COMP_HI]) << 32 |
585 cpu->env.features[FEAT_XSAVE_COMP_LO];
586}
587
Eduardo Habkost8b4bedd2013-01-04 20:01:06 -0200588const char *get_register_name_32(unsigned int reg)
589{
Igor Mammedov31ccdde2013-06-03 18:23:27 +0200590 if (reg >= CPU_NB_REGS32) {
Eduardo Habkost8b4bedd2013-01-04 20:01:06 -0200591 return NULL;
592 }
Eduardo Habkost8e8aba52013-05-06 13:20:07 -0300593 return x86_reg_info_32[reg].name;
Eduardo Habkost8b4bedd2013-01-04 20:01:06 -0200594}
595
Eduardo Habkost84f1b922014-04-30 13:48:41 -0300596/*
597 * Returns the set of feature flags that are supported and migratable by
598 * QEMU, for a given FeatureWord.
599 */
600static uint32_t x86_cpu_get_migratable_flags(FeatureWord w)
601{
602 FeatureWordInfo *wi = &feature_word_info[w];
603 uint32_t r = 0;
604 int i;
605
606 for (i = 0; i < 32; i++) {
607 uint32_t f = 1U << i;
Eduardo Habkost6fb2fff2016-09-28 12:00:29 -0300608
609 /* If the feature name is known, it is implicitly considered migratable,
610 * unless it is explicitly set in unmigratable_flags */
611 if ((wi->migratable_flags & f) ||
612 (wi->feat_names[i] && !(wi->unmigratable_flags & f))) {
613 r |= f;
Eduardo Habkost84f1b922014-04-30 13:48:41 -0300614 }
Eduardo Habkost84f1b922014-04-30 13:48:41 -0300615 }
616 return r;
617}
618
Jan Kiszkabb44e0d2011-01-21 21:48:07 +0100619void host_cpuid(uint32_t function, uint32_t count,
620 uint32_t *eax, uint32_t *ebx, uint32_t *ecx, uint32_t *edx)
Andre Przywarabdde4762010-03-11 14:38:58 +0100621{
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600622 uint32_t vec[4];
623
624#ifdef __x86_64__
625 asm volatile("cpuid"
626 : "=a"(vec[0]), "=b"(vec[1]),
627 "=c"(vec[2]), "=d"(vec[3])
628 : "0"(function), "c"(count) : "cc");
Eduardo Habkostc1f41222014-01-30 17:48:53 -0200629#elif defined(__i386__)
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600630 asm volatile("pusha \n\t"
631 "cpuid \n\t"
632 "mov %%eax, 0(%2) \n\t"
633 "mov %%ebx, 4(%2) \n\t"
634 "mov %%ecx, 8(%2) \n\t"
635 "mov %%edx, 12(%2) \n\t"
636 "popa"
637 : : "a"(function), "c"(count), "S"(vec)
638 : "memory", "cc");
Eduardo Habkostc1f41222014-01-30 17:48:53 -0200639#else
640 abort();
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600641#endif
642
Andre Przywarabdde4762010-03-11 14:38:58 +0100643 if (eax)
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600644 *eax = vec[0];
Andre Przywarabdde4762010-03-11 14:38:58 +0100645 if (ebx)
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600646 *ebx = vec[1];
Andre Przywarabdde4762010-03-11 14:38:58 +0100647 if (ecx)
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600648 *ecx = vec[2];
Andre Przywarabdde4762010-03-11 14:38:58 +0100649 if (edx)
Anthony Liguoria1fd24a2011-11-27 11:13:01 -0600650 *edx = vec[3];
Andre Przywarabdde4762010-03-11 14:38:58 +0100651}
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100652
653#define iswhite(c) ((c) && ((c) <= ' ' || '~' < (c)))
654
655/* general substring compare of *[s1..e1) and *[s2..e2). sx is start of
656 * a substring. ex if !NULL points to the first char after a substring,
657 * otherwise the string is assumed to sized by a terminating nul.
658 * Return lexical ordering of *s1:*s2.
659 */
Chen Fan8f9d9892014-11-05 16:40:33 +0800660static int sstrcmp(const char *s1, const char *e1,
661 const char *s2, const char *e2)
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100662{
663 for (;;) {
664 if (!*s1 || !*s2 || *s1 != *s2)
665 return (*s1 - *s2);
666 ++s1, ++s2;
667 if (s1 == e1 && s2 == e2)
668 return (0);
669 else if (s1 == e1)
670 return (*s2);
671 else if (s2 == e2)
672 return (*s1);
673 }
674}
675
676/* compare *[s..e) to *altstr. *altstr may be a simple string or multiple
677 * '|' delimited (possibly empty) strings in which case search for a match
678 * within the alternatives proceeds left to right. Return 0 for success,
679 * non-zero otherwise.
680 */
681static int altcmp(const char *s, const char *e, const char *altstr)
682{
683 const char *p, *q;
684
685 for (q = p = altstr; ; ) {
686 while (*p && *p != '|')
687 ++p;
688 if ((q == p && !*s) || (q != p && !sstrcmp(s, e, q, p)))
689 return (0);
690 if (!*p)
691 return (1);
692 else
693 q = ++p;
694 }
695}
696
697/* search featureset for flag *[s..e), if found set corresponding bit in
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200698 * *pval and return true, otherwise return false
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100699 */
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200700static bool lookup_feature(uint32_t *pval, const char *s, const char *e,
701 const char **featureset)
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100702{
703 uint32_t mask;
704 const char **ppc;
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200705 bool found = false;
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100706
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200707 for (mask = 1, ppc = featureset; mask; mask <<= 1, ++ppc) {
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100708 if (*ppc && !altcmp(s, e, *ppc)) {
709 *pval |= mask;
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200710 found = true;
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100711 }
Jan Kiszkae41e0fc2011-04-19 13:06:06 +0200712 }
713 return found;
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100714}
715
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200716static void add_flagname_to_bitmaps(const char *flagname,
Eduardo Habkostc00c94a2014-08-21 17:22:56 -0300717 FeatureWordArray words,
718 Error **errp)
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100719{
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200720 FeatureWord w;
721 for (w = 0; w < FEATURE_WORDS; w++) {
722 FeatureWordInfo *wi = &feature_word_info[w];
Eduardo Habkost2d5312d2016-05-12 11:24:04 -0300723 if (lookup_feature(&words[w], flagname, NULL, wi->feat_names)) {
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200724 break;
725 }
726 }
727 if (w == FEATURE_WORDS) {
Eduardo Habkostc00c94a2014-08-21 17:22:56 -0300728 error_setg(errp, "CPU feature %s not found", flagname);
Eduardo Habkost5ef57872013-01-07 16:20:45 -0200729 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100730}
731
Eduardo Habkostd940ee92014-02-10 08:21:30 -0200732/* CPU class name definitions: */
733
734#define X86_CPU_TYPE_SUFFIX "-" TYPE_X86_CPU
735#define X86_CPU_TYPE_NAME(name) (name X86_CPU_TYPE_SUFFIX)
736
737/* Return type name for a given CPU model name
738 * Caller is responsible for freeing the returned string.
739 */
740static char *x86_cpu_type_name(const char *model_name)
741{
742 return g_strdup_printf(X86_CPU_TYPE_NAME("%s"), model_name);
743}
744
Andreas Färber500050d2014-02-10 22:02:44 +0100745static ObjectClass *x86_cpu_class_by_name(const char *cpu_model)
746{
Eduardo Habkostd940ee92014-02-10 08:21:30 -0200747 ObjectClass *oc;
748 char *typename;
749
Andreas Färber500050d2014-02-10 22:02:44 +0100750 if (cpu_model == NULL) {
751 return NULL;
752 }
753
Eduardo Habkostd940ee92014-02-10 08:21:30 -0200754 typename = x86_cpu_type_name(cpu_model);
755 oc = object_class_by_name(typename);
756 g_free(typename);
757 return oc;
Andreas Färber500050d2014-02-10 22:02:44 +0100758}
759
Igor Mammedov104494e2016-06-06 17:16:45 +0200760static char *x86_cpu_class_get_model_name(X86CPUClass *cc)
761{
762 const char *class_name = object_class_get_name(OBJECT_CLASS(cc));
763 assert(g_str_has_suffix(class_name, X86_CPU_TYPE_SUFFIX));
764 return g_strndup(class_name,
765 strlen(class_name) - strlen(X86_CPU_TYPE_SUFFIX));
766}
767
Eduardo Habkostd940ee92014-02-10 08:21:30 -0200768struct X86CPUDefinition {
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100769 const char *name;
770 uint32_t level;
Eduardo Habkost90e4b0c2013-04-22 16:00:12 -0300771 uint32_t xlevel;
Igor Mammedov99b88a12013-01-21 15:06:36 +0100772 /* vendor is zero-terminated, 12 character ASCII string */
773 char vendor[CPUID_VENDOR_SZ + 1];
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100774 int family;
775 int model;
776 int stepping;
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300777 FeatureWordArray features;
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100778 char model_id[48];
Eduardo Habkostd940ee92014-02-10 08:21:30 -0200779};
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100780
Eduardo Habkost9576de72014-01-30 17:48:58 -0200781static X86CPUDefinition builtin_x86_defs[] = {
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100782 {
783 .name = "qemu64",
Radim Krčmář3046bb52015-07-09 21:07:39 +0200784 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100785 .vendor = CPUID_VENDOR_AMD,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100786 .family = 6,
Eduardo Habkostf8e6a112013-09-10 17:48:59 -0300787 .model = 6,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100788 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300789 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300790 PPRO_FEATURES |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100791 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100792 CPUID_PSE36,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300793 .features[FEAT_1_ECX] =
Eduardo Habkost6aa91e42015-11-03 17:24:18 -0200794 CPUID_EXT_SSE3 | CPUID_EXT_CX16,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300795 .features[FEAT_8000_0001_EDX] =
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100796 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300797 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost71195672015-11-03 17:24:18 -0200798 CPUID_EXT3_LAHF_LM | CPUID_EXT3_SVM,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100799 .xlevel = 0x8000000A,
Eduardo Habkost9cf2cc32016-04-09 16:44:20 -0300800 .model_id = "QEMU Virtual CPU version " QEMU_HW_VERSION,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100801 },
802 {
803 .name = "phenom",
804 .level = 5,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100805 .vendor = CPUID_VENDOR_AMD,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100806 .family = 16,
807 .model = 2,
808 .stepping = 3,
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300809 /* Missing: CPUID_HT */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300810 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300811 PPRO_FEATURES |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100812 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA |
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300813 CPUID_PSE36 | CPUID_VME,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300814 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300815 CPUID_EXT_SSE3 | CPUID_EXT_MONITOR | CPUID_EXT_CX16 |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100816 CPUID_EXT_POPCNT,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300817 .features[FEAT_8000_0001_EDX] =
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100818 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX |
819 CPUID_EXT2_3DNOW | CPUID_EXT2_3DNOWEXT | CPUID_EXT2_MMXEXT |
Aurelien Jarno8560efe2010-03-13 16:43:15 +0100820 CPUID_EXT2_FFXSR | CPUID_EXT2_PDPE1GB | CPUID_EXT2_RDTSCP,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100821 /* Missing: CPUID_EXT3_CMP_LEG, CPUID_EXT3_EXTAPIC,
822 CPUID_EXT3_CR8LEG,
823 CPUID_EXT3_MISALIGNSSE, CPUID_EXT3_3DNOWPREFETCH,
824 CPUID_EXT3_OSVW, CPUID_EXT3_IBS */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300825 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300826 CPUID_EXT3_LAHF_LM | CPUID_EXT3_SVM |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100827 CPUID_EXT3_ABM | CPUID_EXT3_SSE4A,
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300828 /* Missing: CPUID_SVM_LBRV */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300829 .features[FEAT_SVM] =
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300830 CPUID_SVM_NPT,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100831 .xlevel = 0x8000001A,
832 .model_id = "AMD Phenom(tm) 9550 Quad-Core Processor"
833 },
834 {
835 .name = "core2duo",
836 .level = 10,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100837 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100838 .family = 6,
839 .model = 15,
840 .stepping = 11,
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300841 /* Missing: CPUID_DTS, CPUID_HT, CPUID_TM, CPUID_PBE */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300842 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300843 PPRO_FEATURES |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100844 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA |
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300845 CPUID_PSE36 | CPUID_VME | CPUID_ACPI | CPUID_SS,
846 /* Missing: CPUID_EXT_DTES64, CPUID_EXT_DSCPL, CPUID_EXT_EST,
Eduardo Habkoste93abc12014-10-03 16:39:50 -0300847 * CPUID_EXT_TM2, CPUID_EXT_XTPR, CPUID_EXT_PDCM, CPUID_EXT_VMX */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300848 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300849 CPUID_EXT_SSE3 | CPUID_EXT_MONITOR | CPUID_EXT_SSSE3 |
Eduardo Habkoste93abc12014-10-03 16:39:50 -0300850 CPUID_EXT_CX16,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300851 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300852 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300853 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300854 CPUID_EXT3_LAHF_LM,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100855 .xlevel = 0x80000008,
856 .model_id = "Intel(R) Core(TM)2 Duo CPU T7700 @ 2.40GHz",
857 },
858 {
859 .name = "kvm64",
Radim Krčmář3046bb52015-07-09 21:07:39 +0200860 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100861 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100862 .family = 15,
863 .model = 6,
864 .stepping = 1,
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -0200865 /* Missing: CPUID_HT */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300866 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -0200867 PPRO_FEATURES | CPUID_VME |
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100868 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA |
869 CPUID_PSE36,
870 /* Missing: CPUID_EXT_POPCNT, CPUID_EXT_MONITOR */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300871 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300872 CPUID_EXT_SSE3 | CPUID_EXT_CX16,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100873 /* Missing: CPUID_EXT2_PDPE1GB, CPUID_EXT2_RDTSCP */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300874 .features[FEAT_8000_0001_EDX] =
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100875 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX,
876 /* Missing: CPUID_EXT3_LAHF_LM, CPUID_EXT3_CMP_LEG, CPUID_EXT3_EXTAPIC,
877 CPUID_EXT3_CR8LEG, CPUID_EXT3_ABM, CPUID_EXT3_SSE4A,
878 CPUID_EXT3_MISALIGNSSE, CPUID_EXT3_3DNOWPREFETCH,
879 CPUID_EXT3_OSVW, CPUID_EXT3_IBS, CPUID_EXT3_SVM */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300880 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300881 0,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100882 .xlevel = 0x80000008,
883 .model_id = "Common KVM processor"
884 },
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100885 {
886 .name = "qemu32",
887 .level = 4,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100888 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100889 .family = 6,
Eduardo Habkostf8e6a112013-09-10 17:48:59 -0300890 .model = 6,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100891 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300892 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300893 PPRO_FEATURES,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300894 .features[FEAT_1_ECX] =
Eduardo Habkost6aa91e42015-11-03 17:24:18 -0200895 CPUID_EXT_SSE3,
Andre Przywara58012d62010-03-11 14:39:06 +0100896 .xlevel = 0x80000004,
Eduardo Habkost9cf2cc32016-04-09 16:44:20 -0300897 .model_id = "QEMU Virtual CPU version " QEMU_HW_VERSION,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100898 },
899 {
Andre Przywaraeafaf1e2010-05-21 09:50:51 +0200900 .name = "kvm32",
901 .level = 5,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100902 .vendor = CPUID_VENDOR_INTEL,
Andre Przywaraeafaf1e2010-05-21 09:50:51 +0200903 .family = 15,
904 .model = 6,
905 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300906 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -0200907 PPRO_FEATURES | CPUID_VME |
Andre Przywaraeafaf1e2010-05-21 09:50:51 +0200908 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA | CPUID_PSE36,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300909 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300910 CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300911 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300912 0,
Andre Przywaraeafaf1e2010-05-21 09:50:51 +0200913 .xlevel = 0x80000008,
914 .model_id = "Common 32-bit KVM processor"
915 },
916 {
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100917 .name = "coreduo",
918 .level = 10,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100919 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100920 .family = 6,
921 .model = 14,
922 .stepping = 8,
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300923 /* Missing: CPUID_DTS, CPUID_HT, CPUID_TM, CPUID_PBE */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300924 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300925 PPRO_FEATURES | CPUID_VME |
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -0300926 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA | CPUID_ACPI |
927 CPUID_SS,
928 /* Missing: CPUID_EXT_EST, CPUID_EXT_TM2 , CPUID_EXT_XTPR,
Eduardo Habkoste93abc12014-10-03 16:39:50 -0300929 * CPUID_EXT_PDCM, CPUID_EXT_VMX */
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300930 .features[FEAT_1_ECX] =
Eduardo Habkoste93abc12014-10-03 16:39:50 -0300931 CPUID_EXT_SSE3 | CPUID_EXT_MONITOR,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300932 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300933 CPUID_EXT2_NX,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100934 .xlevel = 0x80000008,
935 .model_id = "Genuine Intel(R) CPU T2600 @ 2.16GHz",
936 },
937 {
938 .name = "486",
Andre Przywara58012d62010-03-11 14:39:06 +0100939 .level = 1,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100940 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100941 .family = 4,
Andreas Färberb2a856d2013-05-01 17:30:51 +0200942 .model = 8,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100943 .stepping = 0,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300944 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300945 I486_FEATURES,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100946 .xlevel = 0,
947 },
948 {
949 .name = "pentium",
950 .level = 1,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100951 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100952 .family = 5,
953 .model = 4,
954 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300955 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300956 PENTIUM_FEATURES,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100957 .xlevel = 0,
958 },
959 {
960 .name = "pentium2",
961 .level = 2,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100962 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100963 .family = 6,
964 .model = 5,
965 .stepping = 2,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300966 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300967 PENTIUM2_FEATURES,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100968 .xlevel = 0,
969 },
970 {
971 .name = "pentium3",
Radim Krčmář3046bb52015-07-09 21:07:39 +0200972 .level = 3,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100973 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100974 .family = 6,
975 .model = 7,
976 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300977 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300978 PENTIUM3_FEATURES,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100979 .xlevel = 0,
980 },
981 {
982 .name = "athlon",
983 .level = 2,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100984 .vendor = CPUID_VENDOR_AMD,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100985 .family = 6,
986 .model = 2,
987 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300988 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -0300989 PPRO_FEATURES | CPUID_PSE36 | CPUID_VME | CPUID_MTRR |
Eduardo Habkost60032ac2012-09-06 10:05:37 +0000990 CPUID_MCA,
Eduardo Habkost0514ef22013-04-22 16:00:15 -0300991 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost60032ac2012-09-06 10:05:37 +0000992 CPUID_EXT2_MMXEXT | CPUID_EXT2_3DNOW | CPUID_EXT2_3DNOWEXT,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100993 .xlevel = 0x80000008,
Eduardo Habkost9cf2cc32016-04-09 16:44:20 -0300994 .model_id = "QEMU Virtual CPU version " QEMU_HW_VERSION,
Andre Przywarac6dc6f62010-03-11 14:38:55 +0100995 },
996 {
997 .name = "n270",
Radim Krčmář3046bb52015-07-09 21:07:39 +0200998 .level = 10,
Igor Mammedov99b88a12013-01-21 15:06:36 +0100999 .vendor = CPUID_VENDOR_INTEL,
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001000 .family = 6,
1001 .model = 28,
1002 .stepping = 2,
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -03001003 /* Missing: CPUID_DTS, CPUID_HT, CPUID_TM, CPUID_PBE */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001004 .features[FEAT_1_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001005 PPRO_FEATURES |
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -03001006 CPUID_MTRR | CPUID_CLFLUSH | CPUID_MCA | CPUID_VME |
1007 CPUID_ACPI | CPUID_SS,
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001008 /* Some CPUs got no CPUID_SEP */
Eduardo Habkostb9fc20b2014-10-03 16:39:49 -03001009 /* Missing: CPUID_EXT_DSCPL, CPUID_EXT_EST, CPUID_EXT_TM2,
1010 * CPUID_EXT_XTPR */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001011 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001012 CPUID_EXT_SSE3 | CPUID_EXT_MONITOR | CPUID_EXT_SSSE3 |
Borislav Petkov4458c232013-04-25 15:43:04 -03001013 CPUID_EXT_MOVBE,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001014 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost60032ac2012-09-06 10:05:37 +00001015 CPUID_EXT2_NX,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001016 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001017 CPUID_EXT3_LAHF_LM,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001018 .xlevel = 0x80000008,
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001019 .model_id = "Intel(R) Atom(TM) CPU N270 @ 1.60GHz",
1020 },
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001021 {
1022 .name = "Conroe",
Radim Krčmář3046bb52015-07-09 21:07:39 +02001023 .level = 10,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001024 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001025 .family = 6,
Eduardo Habkostffce9eb2013-05-27 17:23:54 -03001026 .model = 15,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001027 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001028 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001029 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001030 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1031 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1032 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1033 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001034 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001035 CPUID_EXT_SSSE3 | CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001036 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001037 CPUID_EXT2_LM | CPUID_EXT2_NX | CPUID_EXT2_SYSCALL,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001038 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001039 CPUID_EXT3_LAHF_LM,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001040 .xlevel = 0x80000008,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001041 .model_id = "Intel Celeron_4x0 (Conroe/Merom Class Core 2)",
1042 },
1043 {
1044 .name = "Penryn",
Radim Krčmář3046bb52015-07-09 21:07:39 +02001045 .level = 10,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001046 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001047 .family = 6,
Eduardo Habkostffce9eb2013-05-27 17:23:54 -03001048 .model = 23,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001049 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001050 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001051 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001052 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1053 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1054 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1055 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001056 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001057 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001058 CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001059 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001060 CPUID_EXT2_LM | CPUID_EXT2_NX | CPUID_EXT2_SYSCALL,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001061 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001062 CPUID_EXT3_LAHF_LM,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001063 .xlevel = 0x80000008,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001064 .model_id = "Intel Core 2 Duo P9xxx (Penryn Class Core 2)",
1065 },
1066 {
1067 .name = "Nehalem",
Radim Krčmář3046bb52015-07-09 21:07:39 +02001068 .level = 11,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001069 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001070 .family = 6,
Eduardo Habkostffce9eb2013-05-27 17:23:54 -03001071 .model = 26,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001072 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001073 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001074 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001075 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1076 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1077 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1078 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001079 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001080 CPUID_EXT_POPCNT | CPUID_EXT_SSE42 | CPUID_EXT_SSE41 |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001081 CPUID_EXT_CX16 | CPUID_EXT_SSSE3 | CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001082 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001083 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001084 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001085 CPUID_EXT3_LAHF_LM,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001086 .xlevel = 0x80000008,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001087 .model_id = "Intel Core i7 9xx (Nehalem Class Core i7)",
1088 },
1089 {
1090 .name = "Westmere",
1091 .level = 11,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001092 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001093 .family = 6,
1094 .model = 44,
1095 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001096 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001097 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001098 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1099 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1100 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1101 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001102 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001103 CPUID_EXT_AES | CPUID_EXT_POPCNT | CPUID_EXT_SSE42 |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001104 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1105 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001106 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001107 CPUID_EXT2_LM | CPUID_EXT2_SYSCALL | CPUID_EXT2_NX,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001108 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001109 CPUID_EXT3_LAHF_LM,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001110 .features[FEAT_6_EAX] =
1111 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001112 .xlevel = 0x80000008,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001113 .model_id = "Westmere E56xx/L56xx/X56xx (Nehalem-C)",
1114 },
1115 {
1116 .name = "SandyBridge",
1117 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001118 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001119 .family = 6,
1120 .model = 42,
1121 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001122 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001123 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001124 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1125 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1126 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1127 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001128 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001129 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001130 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_POPCNT |
1131 CPUID_EXT_X2APIC | CPUID_EXT_SSE42 | CPUID_EXT_SSE41 |
1132 CPUID_EXT_CX16 | CPUID_EXT_SSSE3 | CPUID_EXT_PCLMULQDQ |
1133 CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001134 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001135 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001136 CPUID_EXT2_SYSCALL,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001137 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001138 CPUID_EXT3_LAHF_LM,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01001139 .features[FEAT_XSAVE] =
1140 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001141 .features[FEAT_6_EAX] =
1142 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001143 .xlevel = 0x80000008,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001144 .model_id = "Intel Xeon E312xx (Sandy Bridge)",
1145 },
1146 {
Paolo Bonzini2f9ac422014-12-05 10:55:23 +01001147 .name = "IvyBridge",
1148 .level = 0xd,
1149 .vendor = CPUID_VENDOR_INTEL,
1150 .family = 6,
1151 .model = 58,
1152 .stepping = 9,
1153 .features[FEAT_1_EDX] =
1154 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
1155 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1156 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1157 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1158 CPUID_DE | CPUID_FP87,
1159 .features[FEAT_1_ECX] =
1160 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
1161 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_POPCNT |
1162 CPUID_EXT_X2APIC | CPUID_EXT_SSE42 | CPUID_EXT_SSE41 |
1163 CPUID_EXT_CX16 | CPUID_EXT_SSSE3 | CPUID_EXT_PCLMULQDQ |
1164 CPUID_EXT_SSE3 | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
1165 .features[FEAT_7_0_EBX] =
1166 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_SMEP |
1167 CPUID_7_0_EBX_ERMS,
1168 .features[FEAT_8000_0001_EDX] =
1169 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
1170 CPUID_EXT2_SYSCALL,
1171 .features[FEAT_8000_0001_ECX] =
1172 CPUID_EXT3_LAHF_LM,
1173 .features[FEAT_XSAVE] =
1174 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001175 .features[FEAT_6_EAX] =
1176 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001177 .xlevel = 0x80000008,
Paolo Bonzini2f9ac422014-12-05 10:55:23 +01001178 .model_id = "Intel Xeon E3-12xx v2 (Ivy Bridge)",
1179 },
1180 {
Eduardo Habkosta3568502015-03-13 15:58:09 -03001181 .name = "Haswell-noTSX",
1182 .level = 0xd,
1183 .vendor = CPUID_VENDOR_INTEL,
1184 .family = 6,
1185 .model = 60,
1186 .stepping = 1,
1187 .features[FEAT_1_EDX] =
1188 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
1189 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1190 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1191 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1192 CPUID_DE | CPUID_FP87,
1193 .features[FEAT_1_ECX] =
1194 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
1195 CPUID_EXT_POPCNT | CPUID_EXT_X2APIC | CPUID_EXT_SSE42 |
1196 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1197 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3 |
1198 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_FMA | CPUID_EXT_MOVBE |
1199 CPUID_EXT_PCID | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
1200 .features[FEAT_8000_0001_EDX] =
1201 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
1202 CPUID_EXT2_SYSCALL,
1203 .features[FEAT_8000_0001_ECX] =
Paolo Bonzinibecb6662015-09-28 14:00:18 +02001204 CPUID_EXT3_ABM | CPUID_EXT3_LAHF_LM,
Eduardo Habkosta3568502015-03-13 15:58:09 -03001205 .features[FEAT_7_0_EBX] =
1206 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_BMI1 |
1207 CPUID_7_0_EBX_AVX2 | CPUID_7_0_EBX_SMEP |
1208 CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ERMS | CPUID_7_0_EBX_INVPCID,
1209 .features[FEAT_XSAVE] =
1210 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001211 .features[FEAT_6_EAX] =
1212 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001213 .xlevel = 0x80000008,
Eduardo Habkosta3568502015-03-13 15:58:09 -03001214 .model_id = "Intel Core Processor (Haswell, no TSX)",
1215 }, {
Eduardo Habkost37507092012-11-14 16:28:54 -02001216 .name = "Haswell",
1217 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001218 .vendor = CPUID_VENDOR_INTEL,
Eduardo Habkost37507092012-11-14 16:28:54 -02001219 .family = 6,
1220 .model = 60,
1221 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001222 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001223 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001224 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1225 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1226 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1227 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001228 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001229 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001230 CPUID_EXT_POPCNT | CPUID_EXT_X2APIC | CPUID_EXT_SSE42 |
1231 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1232 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3 |
1233 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_FMA | CPUID_EXT_MOVBE |
Paolo Bonzini78a611f2014-12-05 10:52:46 +01001234 CPUID_EXT_PCID | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001235 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001236 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001237 CPUID_EXT2_SYSCALL,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001238 .features[FEAT_8000_0001_ECX] =
Paolo Bonzinibecb6662015-09-28 14:00:18 +02001239 CPUID_EXT3_ABM | CPUID_EXT3_LAHF_LM,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001240 .features[FEAT_7_0_EBX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001241 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_BMI1 |
Eduardo Habkost1ee91592015-03-13 15:39:43 -03001242 CPUID_7_0_EBX_HLE | CPUID_7_0_EBX_AVX2 | CPUID_7_0_EBX_SMEP |
1243 CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ERMS | CPUID_7_0_EBX_INVPCID |
1244 CPUID_7_0_EBX_RTM,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01001245 .features[FEAT_XSAVE] =
1246 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001247 .features[FEAT_6_EAX] =
1248 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001249 .xlevel = 0x80000008,
Eduardo Habkost37507092012-11-14 16:28:54 -02001250 .model_id = "Intel Core Processor (Haswell)",
1251 },
1252 {
Eduardo Habkosta3568502015-03-13 15:58:09 -03001253 .name = "Broadwell-noTSX",
1254 .level = 0xd,
1255 .vendor = CPUID_VENDOR_INTEL,
1256 .family = 6,
1257 .model = 61,
1258 .stepping = 2,
1259 .features[FEAT_1_EDX] =
1260 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
1261 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1262 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1263 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1264 CPUID_DE | CPUID_FP87,
1265 .features[FEAT_1_ECX] =
1266 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
1267 CPUID_EXT_POPCNT | CPUID_EXT_X2APIC | CPUID_EXT_SSE42 |
1268 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1269 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3 |
1270 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_FMA | CPUID_EXT_MOVBE |
1271 CPUID_EXT_PCID | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
1272 .features[FEAT_8000_0001_EDX] =
1273 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
1274 CPUID_EXT2_SYSCALL,
1275 .features[FEAT_8000_0001_ECX] =
Paolo Bonzinibecb6662015-09-28 14:00:18 +02001276 CPUID_EXT3_ABM | CPUID_EXT3_LAHF_LM | CPUID_EXT3_3DNOWPREFETCH,
Eduardo Habkosta3568502015-03-13 15:58:09 -03001277 .features[FEAT_7_0_EBX] =
1278 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_BMI1 |
1279 CPUID_7_0_EBX_AVX2 | CPUID_7_0_EBX_SMEP |
1280 CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ERMS | CPUID_7_0_EBX_INVPCID |
1281 CPUID_7_0_EBX_RDSEED | CPUID_7_0_EBX_ADX |
1282 CPUID_7_0_EBX_SMAP,
1283 .features[FEAT_XSAVE] =
1284 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001285 .features[FEAT_6_EAX] =
1286 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001287 .xlevel = 0x80000008,
Eduardo Habkosta3568502015-03-13 15:58:09 -03001288 .model_id = "Intel Core Processor (Broadwell, no TSX)",
1289 },
1290 {
Eduardo Habkostece01352014-06-17 17:11:40 -03001291 .name = "Broadwell",
1292 .level = 0xd,
1293 .vendor = CPUID_VENDOR_INTEL,
1294 .family = 6,
1295 .model = 61,
1296 .stepping = 2,
1297 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001298 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostece01352014-06-17 17:11:40 -03001299 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1300 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1301 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1302 CPUID_DE | CPUID_FP87,
1303 .features[FEAT_1_ECX] =
1304 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
1305 CPUID_EXT_POPCNT | CPUID_EXT_X2APIC | CPUID_EXT_SSE42 |
1306 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1307 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3 |
1308 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_FMA | CPUID_EXT_MOVBE |
Paolo Bonzini78a611f2014-12-05 10:52:46 +01001309 CPUID_EXT_PCID | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
Eduardo Habkostece01352014-06-17 17:11:40 -03001310 .features[FEAT_8000_0001_EDX] =
1311 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
1312 CPUID_EXT2_SYSCALL,
1313 .features[FEAT_8000_0001_ECX] =
Paolo Bonzinibecb6662015-09-28 14:00:18 +02001314 CPUID_EXT3_ABM | CPUID_EXT3_LAHF_LM | CPUID_EXT3_3DNOWPREFETCH,
Eduardo Habkostece01352014-06-17 17:11:40 -03001315 .features[FEAT_7_0_EBX] =
1316 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_BMI1 |
Eduardo Habkost1ee91592015-03-13 15:39:43 -03001317 CPUID_7_0_EBX_HLE | CPUID_7_0_EBX_AVX2 | CPUID_7_0_EBX_SMEP |
Eduardo Habkostece01352014-06-17 17:11:40 -03001318 CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ERMS | CPUID_7_0_EBX_INVPCID |
Eduardo Habkost1ee91592015-03-13 15:39:43 -03001319 CPUID_7_0_EBX_RTM | CPUID_7_0_EBX_RDSEED | CPUID_7_0_EBX_ADX |
Eduardo Habkostece01352014-06-17 17:11:40 -03001320 CPUID_7_0_EBX_SMAP,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01001321 .features[FEAT_XSAVE] =
1322 CPUID_XSAVE_XSAVEOPT,
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02001323 .features[FEAT_6_EAX] =
1324 CPUID_6_EAX_ARAT,
Radim Krčmář3046bb52015-07-09 21:07:39 +02001325 .xlevel = 0x80000008,
Eduardo Habkostece01352014-06-17 17:11:40 -03001326 .model_id = "Intel Core Processor (Broadwell)",
1327 },
1328 {
Eduardo Habkostf6f949e2016-04-27 16:13:06 +08001329 .name = "Skylake-Client",
1330 .level = 0xd,
1331 .vendor = CPUID_VENDOR_INTEL,
1332 .family = 6,
1333 .model = 94,
1334 .stepping = 3,
1335 .features[FEAT_1_EDX] =
1336 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
1337 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1338 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1339 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1340 CPUID_DE | CPUID_FP87,
1341 .features[FEAT_1_ECX] =
1342 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
1343 CPUID_EXT_POPCNT | CPUID_EXT_X2APIC | CPUID_EXT_SSE42 |
1344 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_SSSE3 |
1345 CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3 |
1346 CPUID_EXT_TSC_DEADLINE_TIMER | CPUID_EXT_FMA | CPUID_EXT_MOVBE |
1347 CPUID_EXT_PCID | CPUID_EXT_F16C | CPUID_EXT_RDRAND,
1348 .features[FEAT_8000_0001_EDX] =
1349 CPUID_EXT2_LM | CPUID_EXT2_RDTSCP | CPUID_EXT2_NX |
1350 CPUID_EXT2_SYSCALL,
1351 .features[FEAT_8000_0001_ECX] =
1352 CPUID_EXT3_ABM | CPUID_EXT3_LAHF_LM | CPUID_EXT3_3DNOWPREFETCH,
1353 .features[FEAT_7_0_EBX] =
1354 CPUID_7_0_EBX_FSGSBASE | CPUID_7_0_EBX_BMI1 |
1355 CPUID_7_0_EBX_HLE | CPUID_7_0_EBX_AVX2 | CPUID_7_0_EBX_SMEP |
1356 CPUID_7_0_EBX_BMI2 | CPUID_7_0_EBX_ERMS | CPUID_7_0_EBX_INVPCID |
1357 CPUID_7_0_EBX_RTM | CPUID_7_0_EBX_RDSEED | CPUID_7_0_EBX_ADX |
1358 CPUID_7_0_EBX_SMAP | CPUID_7_0_EBX_MPX,
1359 /* Missing: XSAVES (not supported by some Linux versions,
1360 * including v4.1 to v4.6).
1361 * KVM doesn't yet expose any XSAVES state save component,
1362 * and the only one defined in Skylake (processor tracing)
1363 * probably will block migration anyway.
1364 */
1365 .features[FEAT_XSAVE] =
1366 CPUID_XSAVE_XSAVEOPT | CPUID_XSAVE_XSAVEC |
1367 CPUID_XSAVE_XGETBV1,
1368 .features[FEAT_6_EAX] =
1369 CPUID_6_EAX_ARAT,
1370 .xlevel = 0x80000008,
1371 .model_id = "Intel Core Processor (Skylake)",
1372 },
1373 {
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001374 .name = "Opteron_G1",
1375 .level = 5,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001376 .vendor = CPUID_VENDOR_AMD,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001377 .family = 15,
1378 .model = 6,
1379 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001380 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001381 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001382 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1383 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1384 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1385 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001386 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001387 CPUID_EXT_SSE3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001388 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001389 CPUID_EXT2_LM | CPUID_EXT2_FXSR | CPUID_EXT2_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001390 CPUID_EXT2_NX | CPUID_EXT2_PSE36 | CPUID_EXT2_PAT |
1391 CPUID_EXT2_CMOV | CPUID_EXT2_MCA | CPUID_EXT2_PGE |
1392 CPUID_EXT2_MTRR | CPUID_EXT2_SYSCALL | CPUID_EXT2_APIC |
1393 CPUID_EXT2_CX8 | CPUID_EXT2_MCE | CPUID_EXT2_PAE | CPUID_EXT2_MSR |
1394 CPUID_EXT2_TSC | CPUID_EXT2_PSE | CPUID_EXT2_DE | CPUID_EXT2_FPU,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001395 .xlevel = 0x80000008,
1396 .model_id = "AMD Opteron 240 (Gen 1 Class Opteron)",
1397 },
1398 {
1399 .name = "Opteron_G2",
1400 .level = 5,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001401 .vendor = CPUID_VENDOR_AMD,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001402 .family = 15,
1403 .model = 6,
1404 .stepping = 1,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001405 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001406 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001407 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1408 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1409 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1410 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001411 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001412 CPUID_EXT_CX16 | CPUID_EXT_SSE3,
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001413 /* Missing: CPUID_EXT2_RDTSCP */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001414 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001415 CPUID_EXT2_LM | CPUID_EXT2_FXSR |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001416 CPUID_EXT2_MMX | CPUID_EXT2_NX | CPUID_EXT2_PSE36 |
1417 CPUID_EXT2_PAT | CPUID_EXT2_CMOV | CPUID_EXT2_MCA |
1418 CPUID_EXT2_PGE | CPUID_EXT2_MTRR | CPUID_EXT2_SYSCALL |
1419 CPUID_EXT2_APIC | CPUID_EXT2_CX8 | CPUID_EXT2_MCE |
1420 CPUID_EXT2_PAE | CPUID_EXT2_MSR | CPUID_EXT2_TSC | CPUID_EXT2_PSE |
1421 CPUID_EXT2_DE | CPUID_EXT2_FPU,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001422 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001423 CPUID_EXT3_SVM | CPUID_EXT3_LAHF_LM,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001424 .xlevel = 0x80000008,
1425 .model_id = "AMD Opteron 22xx (Gen 2 Class Opteron)",
1426 },
1427 {
1428 .name = "Opteron_G3",
1429 .level = 5,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001430 .vendor = CPUID_VENDOR_AMD,
Evgeny Yakovlev339892d2016-10-03 14:50:02 +03001431 .family = 16,
1432 .model = 2,
1433 .stepping = 3,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001434 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001435 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001436 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1437 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1438 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1439 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001440 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001441 CPUID_EXT_POPCNT | CPUID_EXT_CX16 | CPUID_EXT_MONITOR |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001442 CPUID_EXT_SSE3,
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001443 /* Missing: CPUID_EXT2_RDTSCP */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001444 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001445 CPUID_EXT2_LM | CPUID_EXT2_FXSR |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001446 CPUID_EXT2_MMX | CPUID_EXT2_NX | CPUID_EXT2_PSE36 |
1447 CPUID_EXT2_PAT | CPUID_EXT2_CMOV | CPUID_EXT2_MCA |
1448 CPUID_EXT2_PGE | CPUID_EXT2_MTRR | CPUID_EXT2_SYSCALL |
1449 CPUID_EXT2_APIC | CPUID_EXT2_CX8 | CPUID_EXT2_MCE |
1450 CPUID_EXT2_PAE | CPUID_EXT2_MSR | CPUID_EXT2_TSC | CPUID_EXT2_PSE |
1451 CPUID_EXT2_DE | CPUID_EXT2_FPU,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001452 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001453 CPUID_EXT3_MISALIGNSSE | CPUID_EXT3_SSE4A |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001454 CPUID_EXT3_ABM | CPUID_EXT3_SVM | CPUID_EXT3_LAHF_LM,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001455 .xlevel = 0x80000008,
1456 .model_id = "AMD Opteron 23xx (Gen 3 Class Opteron)",
1457 },
1458 {
1459 .name = "Opteron_G4",
1460 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001461 .vendor = CPUID_VENDOR_AMD,
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001462 .family = 21,
1463 .model = 1,
1464 .stepping = 2,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001465 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001466 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001467 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1468 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1469 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1470 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001471 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001472 CPUID_EXT_AVX | CPUID_EXT_XSAVE | CPUID_EXT_AES |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001473 CPUID_EXT_POPCNT | CPUID_EXT_SSE42 | CPUID_EXT_SSE41 |
1474 CPUID_EXT_CX16 | CPUID_EXT_SSSE3 | CPUID_EXT_PCLMULQDQ |
1475 CPUID_EXT_SSE3,
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001476 /* Missing: CPUID_EXT2_RDTSCP */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001477 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001478 CPUID_EXT2_LM |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001479 CPUID_EXT2_PDPE1GB | CPUID_EXT2_FXSR | CPUID_EXT2_MMX |
1480 CPUID_EXT2_NX | CPUID_EXT2_PSE36 | CPUID_EXT2_PAT |
1481 CPUID_EXT2_CMOV | CPUID_EXT2_MCA | CPUID_EXT2_PGE |
1482 CPUID_EXT2_MTRR | CPUID_EXT2_SYSCALL | CPUID_EXT2_APIC |
1483 CPUID_EXT2_CX8 | CPUID_EXT2_MCE | CPUID_EXT2_PAE | CPUID_EXT2_MSR |
1484 CPUID_EXT2_TSC | CPUID_EXT2_PSE | CPUID_EXT2_DE | CPUID_EXT2_FPU,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001485 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001486 CPUID_EXT3_FMA4 | CPUID_EXT3_XOP |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001487 CPUID_EXT3_3DNOWPREFETCH | CPUID_EXT3_MISALIGNSSE |
1488 CPUID_EXT3_SSE4A | CPUID_EXT3_ABM | CPUID_EXT3_SVM |
1489 CPUID_EXT3_LAHF_LM,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01001490 /* no xsaveopt! */
Eduardo Habkost3eca4642012-09-05 17:41:10 -03001491 .xlevel = 0x8000001A,
1492 .model_id = "AMD Opteron 62xx class CPU",
1493 },
Andre Przywara021941b2012-11-14 16:28:53 -02001494 {
1495 .name = "Opteron_G5",
1496 .level = 0xd,
Igor Mammedov99b88a12013-01-21 15:06:36 +01001497 .vendor = CPUID_VENDOR_AMD,
Andre Przywara021941b2012-11-14 16:28:53 -02001498 .family = 21,
1499 .model = 2,
1500 .stepping = 0,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001501 .features[FEAT_1_EDX] =
Paolo Bonzinib3a4f0b2014-12-10 14:12:41 -02001502 CPUID_VME | CPUID_SSE2 | CPUID_SSE | CPUID_FXSR | CPUID_MMX |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001503 CPUID_CLFLUSH | CPUID_PSE36 | CPUID_PAT | CPUID_CMOV | CPUID_MCA |
1504 CPUID_PGE | CPUID_MTRR | CPUID_SEP | CPUID_APIC | CPUID_CX8 |
1505 CPUID_MCE | CPUID_PAE | CPUID_MSR | CPUID_TSC | CPUID_PSE |
1506 CPUID_DE | CPUID_FP87,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001507 .features[FEAT_1_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001508 CPUID_EXT_F16C | CPUID_EXT_AVX | CPUID_EXT_XSAVE |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001509 CPUID_EXT_AES | CPUID_EXT_POPCNT | CPUID_EXT_SSE42 |
1510 CPUID_EXT_SSE41 | CPUID_EXT_CX16 | CPUID_EXT_FMA |
1511 CPUID_EXT_SSSE3 | CPUID_EXT_PCLMULQDQ | CPUID_EXT_SSE3,
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001512 /* Missing: CPUID_EXT2_RDTSCP */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001513 .features[FEAT_8000_0001_EDX] =
Eduardo Habkost33b5e8c2015-11-13 17:07:13 -02001514 CPUID_EXT2_LM |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001515 CPUID_EXT2_PDPE1GB | CPUID_EXT2_FXSR | CPUID_EXT2_MMX |
1516 CPUID_EXT2_NX | CPUID_EXT2_PSE36 | CPUID_EXT2_PAT |
1517 CPUID_EXT2_CMOV | CPUID_EXT2_MCA | CPUID_EXT2_PGE |
1518 CPUID_EXT2_MTRR | CPUID_EXT2_SYSCALL | CPUID_EXT2_APIC |
1519 CPUID_EXT2_CX8 | CPUID_EXT2_MCE | CPUID_EXT2_PAE | CPUID_EXT2_MSR |
1520 CPUID_EXT2_TSC | CPUID_EXT2_PSE | CPUID_EXT2_DE | CPUID_EXT2_FPU,
Eduardo Habkost0514ef22013-04-22 16:00:15 -03001521 .features[FEAT_8000_0001_ECX] =
Eduardo Habkost27861ec2013-04-22 16:00:14 -03001522 CPUID_EXT3_TBM | CPUID_EXT3_FMA4 | CPUID_EXT3_XOP |
Eduardo Habkostb3fb3a22014-06-17 20:05:29 -03001523 CPUID_EXT3_3DNOWPREFETCH | CPUID_EXT3_MISALIGNSSE |
1524 CPUID_EXT3_SSE4A | CPUID_EXT3_ABM | CPUID_EXT3_SVM |
1525 CPUID_EXT3_LAHF_LM,
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01001526 /* no xsaveopt! */
Andre Przywara021941b2012-11-14 16:28:53 -02001527 .xlevel = 0x8000001A,
1528 .model_id = "AMD Opteron 63xx class CPU",
1529 },
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001530};
1531
Eduardo Habkost5114e842015-09-11 12:40:27 -03001532typedef struct PropValue {
1533 const char *prop, *value;
1534} PropValue;
1535
1536/* KVM-specific features that are automatically added/removed
1537 * from all CPU models when KVM is enabled.
1538 */
1539static PropValue kvm_default_props[] = {
1540 { "kvmclock", "on" },
1541 { "kvm-nopiodelay", "on" },
1542 { "kvm-asyncpf", "on" },
1543 { "kvm-steal-time", "on" },
1544 { "kvm-pv-eoi", "on" },
1545 { "kvmclock-stable-bit", "on" },
1546 { "x2apic", "on" },
1547 { "acpi", "off" },
1548 { "monitor", "off" },
1549 { "svm", "off" },
1550 { NULL, NULL },
1551};
1552
Eduardo Habkost04d99c32016-09-30 15:49:37 -03001553/* TCG-specific defaults that override all CPU models when using TCG
1554 */
1555static PropValue tcg_default_props[] = {
1556 { "vme", "off" },
1557 { NULL, NULL },
1558};
1559
1560
Eduardo Habkost5114e842015-09-11 12:40:27 -03001561void x86_cpu_change_kvm_default(const char *prop, const char *value)
1562{
1563 PropValue *pv;
1564 for (pv = kvm_default_props; pv->prop; pv++) {
1565 if (!strcmp(pv->prop, prop)) {
1566 pv->value = value;
1567 break;
1568 }
1569 }
1570
1571 /* It is valid to call this function only for properties that
1572 * are already present in the kvm_default_props table.
1573 */
1574 assert(pv->prop);
1575}
1576
Eduardo Habkost4d1b2792014-08-20 17:30:12 -03001577static uint32_t x86_cpu_get_supported_feature_word(FeatureWord w,
1578 bool migratable_only);
1579
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001580#ifdef CONFIG_KVM
1581
Haozhong Zhang40bfe482016-06-22 14:56:23 +08001582static bool lmce_supported(void)
1583{
1584 uint64_t mce_cap;
1585
1586 if (kvm_ioctl(kvm_state, KVM_X86_GET_MCE_CAP_SUPPORTED, &mce_cap) < 0) {
1587 return false;
1588 }
1589
1590 return !!(mce_cap & MCG_LMCE_P);
1591}
1592
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001593static int cpu_x86_fill_model_id(char *str)
1594{
1595 uint32_t eax = 0, ebx = 0, ecx = 0, edx = 0;
1596 int i;
1597
1598 for (i = 0; i < 3; i++) {
1599 host_cpuid(0x80000002 + i, 0, &eax, &ebx, &ecx, &edx);
1600 memcpy(str + i * 16 + 0, &eax, 4);
1601 memcpy(str + i * 16 + 4, &ebx, 4);
1602 memcpy(str + i * 16 + 8, &ecx, 4);
1603 memcpy(str + i * 16 + 12, &edx, 4);
1604 }
1605 return 0;
1606}
1607
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001608static X86CPUDefinition host_cpudef;
1609
Eduardo Habkost84f1b922014-04-30 13:48:41 -03001610static Property host_x86_cpu_properties[] = {
Eduardo Habkost120eee72014-06-17 17:31:53 -03001611 DEFINE_PROP_BOOL("migratable", X86CPU, migratable, true),
Eduardo Habkoste265e3e2015-09-02 11:19:11 -03001612 DEFINE_PROP_BOOL("host-cache-info", X86CPU, cache_info_passthrough, false),
Eduardo Habkost84f1b922014-04-30 13:48:41 -03001613 DEFINE_PROP_END_OF_LIST()
1614};
1615
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001616/* class_init for the "host" CPU model
Eduardo Habkost6e746f32012-10-24 19:44:06 -02001617 *
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001618 * This function may be called before KVM is initialized.
Eduardo Habkost6e746f32012-10-24 19:44:06 -02001619 */
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001620static void host_x86_cpu_class_init(ObjectClass *oc, void *data)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001621{
Eduardo Habkost84f1b922014-04-30 13:48:41 -03001622 DeviceClass *dc = DEVICE_CLASS(oc);
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001623 X86CPUClass *xcc = X86_CPU_CLASS(oc);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001624 uint32_t eax = 0, ebx = 0, ecx = 0, edx = 0;
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001625
1626 xcc->kvm_required = true;
1627
1628 host_cpuid(0x0, 0, &eax, &ebx, &ecx, &edx);
1629 x86_cpu_vendor_words2str(host_cpudef.vendor, ebx, edx, ecx);
1630
1631 host_cpuid(0x1, 0, &eax, &ebx, &ecx, &edx);
1632 host_cpudef.family = ((eax >> 8) & 0x0F) + ((eax >> 20) & 0xFF);
1633 host_cpudef.model = ((eax >> 4) & 0x0F) | ((eax & 0xF0000) >> 12);
1634 host_cpudef.stepping = eax & 0x0F;
1635
1636 cpu_x86_fill_model_id(host_cpudef.model_id);
1637
1638 xcc->cpu_def = &host_cpudef;
Eduardo Habkostee465a32016-09-30 15:49:36 -03001639 xcc->model_description =
1640 "KVM processor with all supported host features "
1641 "(only available in KVM mode)";
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001642
1643 /* level, xlevel, xlevel2, and the feature words are initialized on
1644 * instance_init, because they require KVM to be initialized.
1645 */
Eduardo Habkost84f1b922014-04-30 13:48:41 -03001646
1647 dc->props = host_x86_cpu_properties;
Markus Armbruster4c315c22015-10-01 10:59:58 +02001648 /* Reason: host_x86_cpu_initfn() dies when !kvm_enabled() */
1649 dc->cannot_destroy_with_object_finalize_yet = true;
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001650}
1651
1652static void host_x86_cpu_initfn(Object *obj)
1653{
1654 X86CPU *cpu = X86_CPU(obj);
1655 CPUX86State *env = &cpu->env;
1656 KVMState *s = kvm_state;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001657
Eduardo Habkost4d1b2792014-08-20 17:30:12 -03001658 /* We can't fill the features array here because we don't know yet if
1659 * "migratable" is true or false.
1660 */
1661 cpu->host_features = true;
1662
Igor Mammedov104494e2016-06-06 17:16:45 +02001663 /* If KVM is disabled, x86_cpu_realizefn() will report an error later */
Eduardo Habkoste4356012016-06-10 08:42:08 -03001664 if (kvm_enabled()) {
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03001665 env->cpuid_min_level =
1666 kvm_arch_get_supported_cpuid(s, 0x0, 0, R_EAX);
1667 env->cpuid_min_xlevel =
1668 kvm_arch_get_supported_cpuid(s, 0x80000000, 0, R_EAX);
1669 env->cpuid_min_xlevel2 =
1670 kvm_arch_get_supported_cpuid(s, 0xC0000000, 0, R_EAX);
Haozhong Zhang40bfe482016-06-22 14:56:23 +08001671
1672 if (lmce_supported()) {
1673 object_property_set_bool(OBJECT(cpu), true, "lmce", &error_abort);
1674 }
Eduardo Habkoste4356012016-06-10 08:42:08 -03001675 }
Eduardo Habkost2a573252014-01-20 14:41:12 -02001676
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001677 object_property_set_bool(OBJECT(cpu), true, "pmu", &error_abort);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001678}
1679
Eduardo Habkostd940ee92014-02-10 08:21:30 -02001680static const TypeInfo host_x86_cpu_type_info = {
1681 .name = X86_CPU_TYPE_NAME("host"),
1682 .parent = TYPE_X86_CPU,
1683 .instance_init = host_x86_cpu_initfn,
1684 .class_init = host_x86_cpu_class_init,
1685};
1686
1687#endif
1688
Eduardo Habkost8459e392014-04-30 13:48:31 -03001689static void report_unavailable_features(FeatureWord w, uint32_t mask)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001690{
Eduardo Habkost8459e392014-04-30 13:48:31 -03001691 FeatureWordInfo *f = &feature_word_info[w];
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001692 int i;
1693
Eduardo Habkost857aee32014-04-30 13:48:29 -03001694 for (i = 0; i < 32; ++i) {
Eduardo Habkost72370dc2015-09-29 17:34:22 -03001695 if ((1UL << i) & mask) {
Eduardo Habkostbffd67b2013-01-07 16:20:46 -02001696 const char *reg = get_register_name_32(f->cpuid_reg);
Eduardo Habkost8b4bedd2013-01-04 20:01:06 -02001697 assert(reg);
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03001698 fprintf(stderr, "warning: %s doesn't support requested feature: "
Eduardo Habkost8b4bedd2013-01-04 20:01:06 -02001699 "CPUID.%02XH:%s%s%s [bit %d]\n",
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03001700 kvm_enabled() ? "host" : "TCG",
Eduardo Habkostbffd67b2013-01-07 16:20:46 -02001701 f->cpuid_eax, reg,
1702 f->feat_names[i] ? "." : "",
1703 f->feat_names[i] ? f->feat_names[i] : "", i);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001704 }
Eduardo Habkost857aee32014-04-30 13:48:29 -03001705 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01001706}
1707
Eric Blaked7bce992016-01-29 06:48:55 -07001708static void x86_cpuid_version_get_family(Object *obj, Visitor *v,
1709 const char *name, void *opaque,
1710 Error **errp)
Andreas Färber95b85192012-04-17 14:42:22 +02001711{
1712 X86CPU *cpu = X86_CPU(obj);
1713 CPUX86State *env = &cpu->env;
1714 int64_t value;
1715
1716 value = (env->cpuid_version >> 8) & 0xf;
1717 if (value == 0xf) {
1718 value += (env->cpuid_version >> 20) & 0xff;
1719 }
Eric Blake51e72bc2016-01-29 06:48:54 -07001720 visit_type_int(v, name, &value, errp);
Andreas Färber95b85192012-04-17 14:42:22 +02001721}
1722
Eric Blaked7bce992016-01-29 06:48:55 -07001723static void x86_cpuid_version_set_family(Object *obj, Visitor *v,
1724 const char *name, void *opaque,
1725 Error **errp)
Andreas Färbered5e1ec2012-02-17 17:46:01 +01001726{
Andreas Färber71ad61d2012-04-17 12:10:29 +02001727 X86CPU *cpu = X86_CPU(obj);
1728 CPUX86State *env = &cpu->env;
1729 const int64_t min = 0;
1730 const int64_t max = 0xff + 0xf;
Markus Armbruster65cd9062014-04-25 12:44:22 +02001731 Error *local_err = NULL;
Andreas Färber71ad61d2012-04-17 12:10:29 +02001732 int64_t value;
1733
Eric Blake51e72bc2016-01-29 06:48:54 -07001734 visit_type_int(v, name, &value, &local_err);
Markus Armbruster65cd9062014-04-25 12:44:22 +02001735 if (local_err) {
1736 error_propagate(errp, local_err);
Andreas Färber71ad61d2012-04-17 12:10:29 +02001737 return;
1738 }
1739 if (value < min || value > max) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +01001740 error_setg(errp, QERR_PROPERTY_VALUE_OUT_OF_RANGE, "",
1741 name ? name : "null", value, min, max);
Andreas Färber71ad61d2012-04-17 12:10:29 +02001742 return;
1743 }
1744
Andreas Färbered5e1ec2012-02-17 17:46:01 +01001745 env->cpuid_version &= ~0xff00f00;
Andreas Färber71ad61d2012-04-17 12:10:29 +02001746 if (value > 0x0f) {
1747 env->cpuid_version |= 0xf00 | ((value - 0x0f) << 20);
Andreas Färbered5e1ec2012-02-17 17:46:01 +01001748 } else {
Andreas Färber71ad61d2012-04-17 12:10:29 +02001749 env->cpuid_version |= value << 8;
Andreas Färbered5e1ec2012-02-17 17:46:01 +01001750 }
1751}
1752
Eric Blaked7bce992016-01-29 06:48:55 -07001753static void x86_cpuid_version_get_model(Object *obj, Visitor *v,
1754 const char *name, void *opaque,
1755 Error **errp)
Andreas Färber67e30c82012-04-17 14:48:14 +02001756{
1757 X86CPU *cpu = X86_CPU(obj);
1758 CPUX86State *env = &cpu->env;
1759 int64_t value;
1760
1761 value = (env->cpuid_version >> 4) & 0xf;
1762 value |= ((env->cpuid_version >> 16) & 0xf) << 4;
Eric Blake51e72bc2016-01-29 06:48:54 -07001763 visit_type_int(v, name, &value, errp);
Andreas Färber67e30c82012-04-17 14:48:14 +02001764}
1765
Eric Blaked7bce992016-01-29 06:48:55 -07001766static void x86_cpuid_version_set_model(Object *obj, Visitor *v,
1767 const char *name, void *opaque,
1768 Error **errp)
Andreas Färberb0704cb2012-02-17 17:46:02 +01001769{
Andreas Färberc5291a42012-04-17 12:16:39 +02001770 X86CPU *cpu = X86_CPU(obj);
1771 CPUX86State *env = &cpu->env;
1772 const int64_t min = 0;
1773 const int64_t max = 0xff;
Markus Armbruster65cd9062014-04-25 12:44:22 +02001774 Error *local_err = NULL;
Andreas Färberc5291a42012-04-17 12:16:39 +02001775 int64_t value;
1776
Eric Blake51e72bc2016-01-29 06:48:54 -07001777 visit_type_int(v, name, &value, &local_err);
Markus Armbruster65cd9062014-04-25 12:44:22 +02001778 if (local_err) {
1779 error_propagate(errp, local_err);
Andreas Färberc5291a42012-04-17 12:16:39 +02001780 return;
1781 }
1782 if (value < min || value > max) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +01001783 error_setg(errp, QERR_PROPERTY_VALUE_OUT_OF_RANGE, "",
1784 name ? name : "null", value, min, max);
Andreas Färberc5291a42012-04-17 12:16:39 +02001785 return;
1786 }
1787
Andreas Färberb0704cb2012-02-17 17:46:02 +01001788 env->cpuid_version &= ~0xf00f0;
Andreas Färberc5291a42012-04-17 12:16:39 +02001789 env->cpuid_version |= ((value & 0xf) << 4) | ((value >> 4) << 16);
Andreas Färberb0704cb2012-02-17 17:46:02 +01001790}
1791
Andreas Färber35112e42012-04-17 14:50:53 +02001792static void x86_cpuid_version_get_stepping(Object *obj, Visitor *v,
Eric Blaked7bce992016-01-29 06:48:55 -07001793 const char *name, void *opaque,
Andreas Färber35112e42012-04-17 14:50:53 +02001794 Error **errp)
1795{
1796 X86CPU *cpu = X86_CPU(obj);
1797 CPUX86State *env = &cpu->env;
1798 int64_t value;
1799
1800 value = env->cpuid_version & 0xf;
Eric Blake51e72bc2016-01-29 06:48:54 -07001801 visit_type_int(v, name, &value, errp);
Andreas Färber35112e42012-04-17 14:50:53 +02001802}
1803
Andreas Färber036e2222012-04-17 14:14:18 +02001804static void x86_cpuid_version_set_stepping(Object *obj, Visitor *v,
Eric Blaked7bce992016-01-29 06:48:55 -07001805 const char *name, void *opaque,
Andreas Färber036e2222012-04-17 14:14:18 +02001806 Error **errp)
Andreas Färber38c3dc42012-02-17 17:46:03 +01001807{
Andreas Färber036e2222012-04-17 14:14:18 +02001808 X86CPU *cpu = X86_CPU(obj);
1809 CPUX86State *env = &cpu->env;
1810 const int64_t min = 0;
1811 const int64_t max = 0xf;
Markus Armbruster65cd9062014-04-25 12:44:22 +02001812 Error *local_err = NULL;
Andreas Färber036e2222012-04-17 14:14:18 +02001813 int64_t value;
1814
Eric Blake51e72bc2016-01-29 06:48:54 -07001815 visit_type_int(v, name, &value, &local_err);
Markus Armbruster65cd9062014-04-25 12:44:22 +02001816 if (local_err) {
1817 error_propagate(errp, local_err);
Andreas Färber036e2222012-04-17 14:14:18 +02001818 return;
1819 }
1820 if (value < min || value > max) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +01001821 error_setg(errp, QERR_PROPERTY_VALUE_OUT_OF_RANGE, "",
1822 name ? name : "null", value, min, max);
Andreas Färber036e2222012-04-17 14:14:18 +02001823 return;
1824 }
1825
Andreas Färber38c3dc42012-02-17 17:46:03 +01001826 env->cpuid_version &= ~0xf;
Andreas Färber036e2222012-04-17 14:14:18 +02001827 env->cpuid_version |= value & 0xf;
Andreas Färber38c3dc42012-02-17 17:46:03 +01001828}
1829
Andreas Färberd480e1a2012-04-17 19:22:58 +02001830static char *x86_cpuid_get_vendor(Object *obj, Error **errp)
1831{
1832 X86CPU *cpu = X86_CPU(obj);
1833 CPUX86State *env = &cpu->env;
1834 char *value;
Andreas Färberd480e1a2012-04-17 19:22:58 +02001835
Markus Armbrustere42a92a2014-12-04 14:46:46 +01001836 value = g_malloc(CPUID_VENDOR_SZ + 1);
Igor Mammedov99b88a12013-01-21 15:06:36 +01001837 x86_cpu_vendor_words2str(value, env->cpuid_vendor1, env->cpuid_vendor2,
1838 env->cpuid_vendor3);
Andreas Färberd480e1a2012-04-17 19:22:58 +02001839 return value;
1840}
1841
1842static void x86_cpuid_set_vendor(Object *obj, const char *value,
1843 Error **errp)
1844{
1845 X86CPU *cpu = X86_CPU(obj);
1846 CPUX86State *env = &cpu->env;
1847 int i;
1848
Igor Mammedov9df694e2012-10-22 17:03:10 +02001849 if (strlen(value) != CPUID_VENDOR_SZ) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +01001850 error_setg(errp, QERR_PROPERTY_VALUE_BAD, "", "vendor", value);
Andreas Färberd480e1a2012-04-17 19:22:58 +02001851 return;
1852 }
1853
1854 env->cpuid_vendor1 = 0;
1855 env->cpuid_vendor2 = 0;
1856 env->cpuid_vendor3 = 0;
1857 for (i = 0; i < 4; i++) {
1858 env->cpuid_vendor1 |= ((uint8_t)value[i ]) << (8 * i);
1859 env->cpuid_vendor2 |= ((uint8_t)value[i + 4]) << (8 * i);
1860 env->cpuid_vendor3 |= ((uint8_t)value[i + 8]) << (8 * i);
1861 }
Andreas Färberd480e1a2012-04-17 19:22:58 +02001862}
1863
Andreas Färber63e886e2012-04-17 23:02:26 +02001864static char *x86_cpuid_get_model_id(Object *obj, Error **errp)
1865{
1866 X86CPU *cpu = X86_CPU(obj);
1867 CPUX86State *env = &cpu->env;
1868 char *value;
1869 int i;
1870
1871 value = g_malloc(48 + 1);
1872 for (i = 0; i < 48; i++) {
1873 value[i] = env->cpuid_model[i >> 2] >> (8 * (i & 3));
1874 }
1875 value[48] = '\0';
1876 return value;
1877}
1878
Andreas Färber938d4c22012-04-17 15:17:27 +02001879static void x86_cpuid_set_model_id(Object *obj, const char *model_id,
1880 Error **errp)
Andreas Färberdcce6672012-02-17 17:46:04 +01001881{
Andreas Färber938d4c22012-04-17 15:17:27 +02001882 X86CPU *cpu = X86_CPU(obj);
1883 CPUX86State *env = &cpu->env;
Andreas Färberdcce6672012-02-17 17:46:04 +01001884 int c, len, i;
1885
1886 if (model_id == NULL) {
1887 model_id = "";
1888 }
1889 len = strlen(model_id);
Andreas Färberd0a6acf2012-04-17 18:21:52 +02001890 memset(env->cpuid_model, 0, 48);
Andreas Färberdcce6672012-02-17 17:46:04 +01001891 for (i = 0; i < 48; i++) {
1892 if (i >= len) {
1893 c = '\0';
1894 } else {
1895 c = (uint8_t)model_id[i];
1896 }
1897 env->cpuid_model[i >> 2] |= c << (8 * (i & 3));
1898 }
1899}
1900
Eric Blaked7bce992016-01-29 06:48:55 -07001901static void x86_cpuid_get_tsc_freq(Object *obj, Visitor *v, const char *name,
1902 void *opaque, Error **errp)
Andreas Färber89e48962012-04-18 00:12:23 +02001903{
1904 X86CPU *cpu = X86_CPU(obj);
1905 int64_t value;
1906
1907 value = cpu->env.tsc_khz * 1000;
Eric Blake51e72bc2016-01-29 06:48:54 -07001908 visit_type_int(v, name, &value, errp);
Andreas Färber89e48962012-04-18 00:12:23 +02001909}
1910
Eric Blaked7bce992016-01-29 06:48:55 -07001911static void x86_cpuid_set_tsc_freq(Object *obj, Visitor *v, const char *name,
1912 void *opaque, Error **errp)
Andreas Färber89e48962012-04-18 00:12:23 +02001913{
1914 X86CPU *cpu = X86_CPU(obj);
1915 const int64_t min = 0;
Don Slutz2e848492012-09-21 20:13:13 -04001916 const int64_t max = INT64_MAX;
Markus Armbruster65cd9062014-04-25 12:44:22 +02001917 Error *local_err = NULL;
Andreas Färber89e48962012-04-18 00:12:23 +02001918 int64_t value;
1919
Eric Blake51e72bc2016-01-29 06:48:54 -07001920 visit_type_int(v, name, &value, &local_err);
Markus Armbruster65cd9062014-04-25 12:44:22 +02001921 if (local_err) {
1922 error_propagate(errp, local_err);
Andreas Färber89e48962012-04-18 00:12:23 +02001923 return;
1924 }
1925 if (value < min || value > max) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +01001926 error_setg(errp, QERR_PROPERTY_VALUE_OUT_OF_RANGE, "",
1927 name ? name : "null", value, min, max);
Andreas Färber89e48962012-04-18 00:12:23 +02001928 return;
1929 }
1930
Haozhong Zhang36f96c42015-11-24 11:33:57 +08001931 cpu->env.tsc_khz = cpu->env.user_tsc_khz = value / 1000;
Andreas Färber89e48962012-04-18 00:12:23 +02001932}
1933
Eduardo Habkost7e5292b2013-05-06 13:20:09 -03001934/* Generic getter for "feature-words" and "filtered-features" properties */
Eric Blaked7bce992016-01-29 06:48:55 -07001935static void x86_cpu_get_feature_words(Object *obj, Visitor *v,
1936 const char *name, void *opaque,
1937 Error **errp)
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03001938{
Eduardo Habkost7e5292b2013-05-06 13:20:09 -03001939 uint32_t *array = (uint32_t *)opaque;
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03001940 FeatureWord w;
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03001941 X86CPUFeatureWordInfo word_infos[FEATURE_WORDS] = { };
1942 X86CPUFeatureWordInfoList list_entries[FEATURE_WORDS] = { };
1943 X86CPUFeatureWordInfoList *list = NULL;
1944
1945 for (w = 0; w < FEATURE_WORDS; w++) {
1946 FeatureWordInfo *wi = &feature_word_info[w];
1947 X86CPUFeatureWordInfo *qwi = &word_infos[w];
1948 qwi->cpuid_input_eax = wi->cpuid_eax;
1949 qwi->has_cpuid_input_ecx = wi->cpuid_needs_ecx;
1950 qwi->cpuid_input_ecx = wi->cpuid_ecx;
1951 qwi->cpuid_register = x86_reg_info_32[wi->cpuid_reg].qapi_enum;
Eduardo Habkost7e5292b2013-05-06 13:20:09 -03001952 qwi->features = array[w];
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03001953
1954 /* List will be in reverse order, but order shouldn't matter */
1955 list_entries[w].next = list;
1956 list_entries[w].value = &word_infos[w];
1957 list = &list_entries[w];
1958 }
1959
Eduardo Habkost6b62d962016-06-13 18:57:57 -03001960 visit_type_X86CPUFeatureWordInfoList(v, "feature-words", &list, errp);
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03001961}
1962
Eric Blaked7bce992016-01-29 06:48:55 -07001963static void x86_get_hv_spinlocks(Object *obj, Visitor *v, const char *name,
1964 void *opaque, Error **errp)
Igor Mammedovc8f0f882013-06-04 15:05:25 +02001965{
1966 X86CPU *cpu = X86_CPU(obj);
1967 int64_t value = cpu->hyperv_spinlock_attempts;
1968
Eric Blake51e72bc2016-01-29 06:48:54 -07001969 visit_type_int(v, name, &value, errp);
Igor Mammedovc8f0f882013-06-04 15:05:25 +02001970}
1971
Eric Blaked7bce992016-01-29 06:48:55 -07001972static void x86_set_hv_spinlocks(Object *obj, Visitor *v, const char *name,
1973 void *opaque, Error **errp)
Igor Mammedovc8f0f882013-06-04 15:05:25 +02001974{
1975 const int64_t min = 0xFFF;
1976 const int64_t max = UINT_MAX;
1977 X86CPU *cpu = X86_CPU(obj);
1978 Error *err = NULL;
1979 int64_t value;
1980
Eric Blake51e72bc2016-01-29 06:48:54 -07001981 visit_type_int(v, name, &value, &err);
Igor Mammedovc8f0f882013-06-04 15:05:25 +02001982 if (err) {
1983 error_propagate(errp, err);
1984 return;
1985 }
1986
1987 if (value < min || value > max) {
1988 error_setg(errp, "Property %s.%s doesn't take value %" PRId64
chenfan5bb4c352014-07-28 15:13:06 +08001989 " (minimum: %" PRId64 ", maximum: %" PRId64 ")",
1990 object_get_typename(obj), name ? name : "null",
1991 value, min, max);
Igor Mammedovc8f0f882013-06-04 15:05:25 +02001992 return;
1993 }
1994 cpu->hyperv_spinlock_attempts = value;
1995}
1996
1997static PropertyInfo qdev_prop_spinlocks = {
1998 .name = "int",
1999 .get = x86_get_hv_spinlocks,
2000 .set = x86_set_hv_spinlocks,
2001};
2002
Igor Mammedov72ac2e82013-04-26 18:04:32 +02002003/* Convert all '_' in a feature string option name to '-', to make feature
2004 * name conform to QOM property naming rule, which uses '-' instead of '_'.
2005 */
2006static inline void feat2prop(char *s)
2007{
2008 while ((s = strchr(s, '_'))) {
2009 *s = '-';
2010 }
2011}
2012
Igor Mammedovdc15c052016-06-06 17:16:44 +02002013/* Compatibily hack to maintain legacy +-feat semantic,
2014 * where +-feat overwrites any feature set by
2015 * feat=on|feat even if the later is parsed after +-feat
2016 * (i.e. "-x2apic,x2apic=on" will result in x2apic disabled)
2017 */
2018static FeatureWordArray plus_features = { 0 };
2019static FeatureWordArray minus_features = { 0 };
2020
Eduardo Habkost8f961352012-12-04 17:34:39 -02002021/* Parse "+feature,-feature,feature=foo" CPU feature string
2022 */
Igor Mammedov62a48a22016-06-09 19:11:01 +02002023static void x86_cpu_parse_featurestr(const char *typename, char *features,
Andreas Färber94a444b2014-03-03 23:19:19 +01002024 Error **errp)
Eduardo Habkost8f961352012-12-04 17:34:39 -02002025{
Eduardo Habkost8f961352012-12-04 17:34:39 -02002026 char *featurestr; /* Single 'key=value" string being parsed */
Andreas Färber94a444b2014-03-03 23:19:19 +01002027 Error *local_err = NULL;
Igor Mammedov62a48a22016-06-09 19:11:01 +02002028 static bool cpu_globals_initialized;
2029
2030 if (cpu_globals_initialized) {
2031 return;
2032 }
2033 cpu_globals_initialized = true;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002034
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002035 if (!features) {
2036 return;
2037 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002038
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002039 for (featurestr = strtok(features, ",");
2040 featurestr && !local_err;
2041 featurestr = strtok(NULL, ",")) {
2042 const char *name;
2043 const char *val = NULL;
2044 char *eq = NULL;
Paolo Bonzinicf2887c2016-06-21 14:04:40 +02002045 char num[32];
Igor Mammedov62a48a22016-06-09 19:11:01 +02002046 GlobalProperty *prop;
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002047
2048 /* Compatibility syntax: */
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002049 if (featurestr[0] == '+') {
Eduardo Habkostc00c94a2014-08-21 17:22:56 -03002050 add_flagname_to_bitmaps(featurestr + 1, plus_features, &local_err);
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002051 continue;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002052 } else if (featurestr[0] == '-') {
Eduardo Habkostc00c94a2014-08-21 17:22:56 -03002053 add_flagname_to_bitmaps(featurestr + 1, minus_features, &local_err);
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002054 continue;
2055 }
Joerg Roedelb862d1f2011-07-07 16:13:12 +02002056
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002057 eq = strchr(featurestr, '=');
2058 if (eq) {
2059 *eq++ = 0;
2060 val = eq;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002061 } else {
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002062 val = "on";
Igor Mammedova91987c2013-01-21 15:06:38 +01002063 }
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002064
2065 feat2prop(featurestr);
2066 name = featurestr;
2067
2068 /* Special case: */
2069 if (!strcmp(name, "tsc-freq")) {
2070 int64_t tsc_freq;
2071 char *err;
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002072
2073 tsc_freq = qemu_strtosz_suffix_unit(val, &err,
2074 QEMU_STRTOSZ_DEFSUFFIX_B, 1000);
2075 if (tsc_freq < 0 || *err) {
2076 error_setg(errp, "bad numerical value %s", val);
2077 return;
2078 }
2079 snprintf(num, sizeof(num), "%" PRId64, tsc_freq);
2080 val = num;
2081 name = "tsc-frequency";
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002082 }
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002083
Igor Mammedov62a48a22016-06-09 19:11:01 +02002084 prop = g_new0(typeof(*prop), 1);
2085 prop->driver = typename;
2086 prop->property = g_strdup(name);
2087 prop->value = g_strdup(val);
2088 prop->errp = &error_fatal;
2089 qdev_prop_register_global(prop);
Eduardo Habkostf6750e92016-06-09 19:10:58 +02002090 }
2091
2092 if (local_err) {
2093 error_propagate(errp, local_err);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002094 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002095}
2096
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002097/* Print all cpuid feature names in featureset
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002098 */
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002099static void listflags(FILE *f, fprintf_function print, const char **featureset)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002100{
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002101 int bit;
2102 bool first = true;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002103
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002104 for (bit = 0; bit < 32; bit++) {
2105 if (featureset[bit]) {
2106 print(f, "%s%s", first ? "" : " ", featureset[bit]);
2107 first = false;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002108 }
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002109 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002110}
2111
Eduardo Habkostee465a32016-09-30 15:49:36 -03002112/* Sort alphabetically by type name, listing kvm_required models last. */
2113static gint x86_cpu_list_compare(gconstpointer a, gconstpointer b)
2114{
2115 ObjectClass *class_a = (ObjectClass *)a;
2116 ObjectClass *class_b = (ObjectClass *)b;
2117 X86CPUClass *cc_a = X86_CPU_CLASS(class_a);
2118 X86CPUClass *cc_b = X86_CPU_CLASS(class_b);
2119 const char *name_a, *name_b;
2120
2121 if (cc_a->kvm_required != cc_b->kvm_required) {
2122 /* kvm_required items go last */
2123 return cc_a->kvm_required ? 1 : -1;
2124 } else {
2125 name_a = object_class_get_name(class_a);
2126 name_b = object_class_get_name(class_b);
2127 return strcmp(name_a, name_b);
2128 }
2129}
2130
2131static GSList *get_sorted_cpu_model_list(void)
2132{
2133 GSList *list = object_class_get_list(TYPE_X86_CPU, false);
2134 list = g_slist_sort(list, x86_cpu_list_compare);
2135 return list;
2136}
2137
2138static void x86_cpu_list_entry(gpointer data, gpointer user_data)
2139{
2140 ObjectClass *oc = data;
2141 X86CPUClass *cc = X86_CPU_CLASS(oc);
2142 CPUListState *s = user_data;
2143 char *name = x86_cpu_class_get_model_name(cc);
2144 const char *desc = cc->model_description;
2145 if (!desc) {
2146 desc = cc->cpu_def->model_id;
2147 }
2148
2149 (*s->cpu_fprintf)(s->file, "x86 %16s %-48s\n",
2150 name, desc);
2151 g_free(name);
2152}
2153
2154/* list available CPU models and flags */
Peter Maydelle916cbf2012-09-05 17:41:08 -03002155void x86_cpu_list(FILE *f, fprintf_function cpu_fprintf)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002156{
Andreas Färber7fc9b712013-01-21 01:02:28 +01002157 int i;
Eduardo Habkostee465a32016-09-30 15:49:36 -03002158 CPUListState s = {
2159 .file = f,
2160 .cpu_fprintf = cpu_fprintf,
2161 };
2162 GSList *list;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002163
Eduardo Habkostee465a32016-09-30 15:49:36 -03002164 (*cpu_fprintf)(f, "Available CPUs:\n");
2165 list = get_sorted_cpu_model_list();
2166 g_slist_foreach(list, x86_cpu_list_entry, &s);
2167 g_slist_free(list);
Jan Kiszka21ad7782013-03-24 17:01:02 +01002168
Peter Maydell6cdf8852012-09-05 17:41:07 -03002169 (*cpu_fprintf)(f, "\nRecognized CPUID flags:\n");
Jan Kiszka3af60be2013-02-27 10:15:51 +01002170 for (i = 0; i < ARRAY_SIZE(feature_word_info); i++) {
2171 FeatureWordInfo *fw = &feature_word_info[i];
2172
Eduardo Habkost8c3329e2015-02-03 15:48:55 -02002173 (*cpu_fprintf)(f, " ");
2174 listflags(f, cpu_fprintf, fw->feat_names);
2175 (*cpu_fprintf)(f, "\n");
Jan Kiszka3af60be2013-02-27 10:15:51 +01002176 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002177}
2178
Eduardo Habkostee465a32016-09-30 15:49:36 -03002179static void x86_cpu_definition_entry(gpointer data, gpointer user_data)
2180{
2181 ObjectClass *oc = data;
2182 X86CPUClass *cc = X86_CPU_CLASS(oc);
2183 CpuDefinitionInfoList **cpu_list = user_data;
2184 CpuDefinitionInfoList *entry;
2185 CpuDefinitionInfo *info;
2186
2187 info = g_malloc0(sizeof(*info));
2188 info->name = x86_cpu_class_get_model_name(cc);
2189
2190 entry = g_malloc0(sizeof(*entry));
2191 entry->value = info;
2192 entry->next = *cpu_list;
2193 *cpu_list = entry;
2194}
2195
Anthony Liguori76b64a72012-08-14 22:17:36 -05002196CpuDefinitionInfoList *arch_query_cpu_definitions(Error **errp)
Anthony Liguorie3966122012-08-10 11:04:14 -05002197{
2198 CpuDefinitionInfoList *cpu_list = NULL;
Eduardo Habkostee465a32016-09-30 15:49:36 -03002199 GSList *list = get_sorted_cpu_model_list();
2200 g_slist_foreach(list, x86_cpu_definition_entry, &cpu_list);
2201 g_slist_free(list);
Anthony Liguorie3966122012-08-10 11:04:14 -05002202 return cpu_list;
2203}
2204
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002205static uint32_t x86_cpu_get_supported_feature_word(FeatureWord w,
2206 bool migratable_only)
Eduardo Habkost27418ad2014-04-30 13:48:32 -03002207{
2208 FeatureWordInfo *wi = &feature_word_info[w];
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002209 uint32_t r;
Eduardo Habkost27418ad2014-04-30 13:48:32 -03002210
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03002211 if (kvm_enabled()) {
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002212 r = kvm_arch_get_supported_cpuid(kvm_state, wi->cpuid_eax,
2213 wi->cpuid_ecx,
2214 wi->cpuid_reg);
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03002215 } else if (tcg_enabled()) {
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002216 r = wi->tcg_features;
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03002217 } else {
2218 return ~0;
2219 }
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002220 if (migratable_only) {
2221 r &= x86_cpu_get_migratable_flags(w);
2222 }
2223 return r;
Eduardo Habkost27418ad2014-04-30 13:48:32 -03002224}
2225
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002226/*
2227 * Filters CPU feature words based on host availability of each feature.
2228 *
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002229 * Returns: 0 if all flags are supported by the host, non-zero otherwise.
2230 */
Eduardo Habkost27418ad2014-04-30 13:48:32 -03002231static int x86_cpu_filter_features(X86CPU *cpu)
Eduardo Habkostbc74b7d2012-10-04 17:49:05 -03002232{
2233 CPUX86State *env = &cpu->env;
Eduardo Habkostbd87d2a2013-04-22 16:00:18 -03002234 FeatureWord w;
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002235 int rv = 0;
2236
Eduardo Habkostbd87d2a2013-04-22 16:00:18 -03002237 for (w = 0; w < FEATURE_WORDS; w++) {
Eduardo Habkost84f1b922014-04-30 13:48:41 -03002238 uint32_t host_feat =
2239 x86_cpu_get_supported_feature_word(w, cpu->migratable);
Eduardo Habkost034acf42013-05-06 13:20:08 -03002240 uint32_t requested_features = env->features[w];
2241 env->features[w] &= host_feat;
2242 cpu->filtered_features[w] = requested_features & ~env->features[w];
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002243 if (cpu->filtered_features[w]) {
2244 if (cpu->check_cpuid || cpu->enforce_cpuid) {
Eduardo Habkost8459e392014-04-30 13:48:31 -03002245 report_unavailable_features(w, cpu->filtered_features[w]);
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002246 }
2247 rv = 1;
2248 }
Eduardo Habkostbd87d2a2013-04-22 16:00:18 -03002249 }
Eduardo Habkost51f63ae2014-04-30 13:48:30 -03002250
2251 return rv;
Eduardo Habkostbc74b7d2012-10-04 17:49:05 -03002252}
Eduardo Habkostbc74b7d2012-10-04 17:49:05 -03002253
Eduardo Habkost5114e842015-09-11 12:40:27 -03002254static void x86_cpu_apply_props(X86CPU *cpu, PropValue *props)
2255{
2256 PropValue *pv;
2257 for (pv = props; pv->prop; pv++) {
2258 if (!pv->value) {
2259 continue;
2260 }
2261 object_property_parse(OBJECT(cpu), pv->value, pv->prop,
2262 &error_abort);
2263 }
2264}
2265
Eduardo Habkostd940ee92014-02-10 08:21:30 -02002266/* Load data from X86CPUDefinition
Eduardo Habkostc080e302014-01-30 17:48:56 -02002267 */
Eduardo Habkostd940ee92014-02-10 08:21:30 -02002268static void x86_cpu_load_def(X86CPU *cpu, X86CPUDefinition *def, Error **errp)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002269{
Andreas Färber61dcd772012-04-17 12:00:51 +02002270 CPUX86State *env = &cpu->env;
Eduardo Habkost74f54bc2014-02-19 16:39:21 -03002271 const char *vendor;
2272 char host_vendor[CPUID_VENDOR_SZ + 1];
Eduardo Habkoste1c224b2014-04-30 13:48:37 -03002273 FeatureWord w;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002274
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03002275 /* CPU models only set _minimum_ values for level/xlevel: */
2276 object_property_set_int(OBJECT(cpu), def->level, "min-level", errp);
2277 object_property_set_int(OBJECT(cpu), def->xlevel, "min-xlevel", errp);
2278
Andreas Färber2d642552013-02-15 14:06:56 +01002279 object_property_set_int(OBJECT(cpu), def->family, "family", errp);
2280 object_property_set_int(OBJECT(cpu), def->model, "model", errp);
2281 object_property_set_int(OBJECT(cpu), def->stepping, "stepping", errp);
Andreas Färber2d642552013-02-15 14:06:56 +01002282 object_property_set_str(OBJECT(cpu), def->model_id, "model-id", errp);
Eduardo Habkoste1c224b2014-04-30 13:48:37 -03002283 for (w = 0; w < FEATURE_WORDS; w++) {
2284 env->features[w] = def->features[w];
2285 }
Eduardo Habkost82beb532014-01-30 17:48:54 -02002286
Eduardo Habkost9576de72014-01-30 17:48:58 -02002287 /* Special cases not set in the X86CPUDefinition structs: */
Eduardo Habkost82beb532014-01-30 17:48:54 -02002288 if (kvm_enabled()) {
Lan Tianyu492a4c92016-02-25 23:15:12 +08002289 if (!kvm_irqchip_in_kernel()) {
2290 x86_cpu_change_kvm_default("x2apic", "off");
2291 }
2292
Eduardo Habkost5114e842015-09-11 12:40:27 -03002293 x86_cpu_apply_props(cpu, kvm_default_props);
Eduardo Habkost04d99c32016-09-30 15:49:37 -03002294 } else if (tcg_enabled()) {
2295 x86_cpu_apply_props(cpu, tcg_default_props);
Eduardo Habkost82beb532014-01-30 17:48:54 -02002296 }
Eduardo Habkost5fcca9f2014-02-19 11:58:10 -03002297
Eduardo Habkost82beb532014-01-30 17:48:54 -02002298 env->features[FEAT_1_ECX] |= CPUID_EXT_HYPERVISOR;
Eduardo Habkost7c08db32014-01-30 17:48:55 -02002299
2300 /* sysenter isn't supported in compatibility mode on AMD,
2301 * syscall isn't supported in compatibility mode on Intel.
2302 * Normally we advertise the actual CPU vendor, but you can
2303 * override this using the 'vendor' property if you want to use
2304 * KVM's sysenter/syscall emulation in compatibility mode and
2305 * when doing cross vendor migration
2306 */
Eduardo Habkost74f54bc2014-02-19 16:39:21 -03002307 vendor = def->vendor;
Eduardo Habkost7c08db32014-01-30 17:48:55 -02002308 if (kvm_enabled()) {
2309 uint32_t ebx = 0, ecx = 0, edx = 0;
2310 host_cpuid(0, 0, NULL, &ebx, &ecx, &edx);
2311 x86_cpu_vendor_words2str(host_vendor, ebx, edx, ecx);
2312 vendor = host_vendor;
2313 }
2314
2315 object_property_set_str(OBJECT(cpu), vendor, "vendor", errp);
2316
Andreas Färber2d642552013-02-15 14:06:56 +01002317}
2318
Peter Maydell08565792015-03-03 00:29:17 +00002319X86CPU *cpu_x86_init(const char *cpu_model)
Igor Mammedov7f833242013-04-11 16:51:40 +02002320{
Igor Mammedova57d0162016-06-06 17:16:46 +02002321 return X86_CPU(cpu_generic_init(TYPE_X86_CPU, cpu_model));
Andreas Färber5c3c6a62013-02-01 15:12:13 +01002322}
2323
Eduardo Habkostd940ee92014-02-10 08:21:30 -02002324static void x86_cpu_cpudef_class_init(ObjectClass *oc, void *data)
2325{
2326 X86CPUDefinition *cpudef = data;
2327 X86CPUClass *xcc = X86_CPU_CLASS(oc);
2328
2329 xcc->cpu_def = cpudef;
2330}
2331
2332static void x86_register_cpudef_type(X86CPUDefinition *def)
2333{
2334 char *typename = x86_cpu_type_name(def->name);
2335 TypeInfo ti = {
2336 .name = typename,
2337 .parent = TYPE_X86_CPU,
2338 .class_init = x86_cpu_cpudef_class_init,
2339 .class_data = def,
2340 };
2341
2342 type_register(&ti);
2343 g_free(typename);
2344}
2345
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002346#if !defined(CONFIG_USER_ONLY)
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002347
Blue Swirl0e26b7b2010-06-19 10:42:34 +03002348void cpu_clear_apic_feature(CPUX86State *env)
2349{
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002350 env->features[FEAT_1_EDX] &= ~CPUID_APIC;
Blue Swirl0e26b7b2010-06-19 10:42:34 +03002351}
2352
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002353#endif /* !CONFIG_USER_ONLY */
2354
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002355void cpu_x86_cpuid(CPUX86State *env, uint32_t index, uint32_t count,
2356 uint32_t *eax, uint32_t *ebx,
2357 uint32_t *ecx, uint32_t *edx)
2358{
Andreas Färbera60f24b2012-12-01 05:35:08 +01002359 X86CPU *cpu = x86_env_get_cpu(env);
2360 CPUState *cs = CPU(cpu);
Longpeng(Mike)14c985c2016-09-07 13:21:13 +08002361 uint32_t pkg_offset;
Andreas Färbera60f24b2012-12-01 05:35:08 +01002362
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002363 /* test if maximum index reached */
2364 if (index & 0x80000000) {
brillywu@viatech.com.cnb3baa152011-06-01 09:59:52 +08002365 if (index > env->cpuid_xlevel) {
2366 if (env->cpuid_xlevel2 > 0) {
2367 /* Handle the Centaur's CPUID instruction. */
2368 if (index > env->cpuid_xlevel2) {
2369 index = env->cpuid_xlevel2;
2370 } else if (index < 0xC0000000) {
2371 index = env->cpuid_xlevel;
2372 }
2373 } else {
Eduardo Habkost57f26ae2012-12-20 16:43:48 -02002374 /* Intel documentation states that invalid EAX input will
2375 * return the same information as EAX=cpuid_level
2376 * (Intel SDM Vol. 2A - Instruction Set Reference - CPUID)
2377 */
2378 index = env->cpuid_level;
brillywu@viatech.com.cnb3baa152011-06-01 09:59:52 +08002379 }
2380 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002381 } else {
2382 if (index > env->cpuid_level)
2383 index = env->cpuid_level;
2384 }
2385
2386 switch(index) {
2387 case 0:
2388 *eax = env->cpuid_level;
Eduardo Habkost5eb2f7a2015-02-03 15:57:50 -02002389 *ebx = env->cpuid_vendor1;
2390 *edx = env->cpuid_vendor2;
2391 *ecx = env->cpuid_vendor3;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002392 break;
2393 case 1:
2394 *eax = env->cpuid_version;
Eduardo Habkost7e72a452014-12-18 23:20:10 -02002395 *ebx = (cpu->apic_id << 24) |
2396 8 << 8; /* CLFLUSH size in quad words, Linux wants it. */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002397 *ecx = env->features[FEAT_1_ECX];
Richard Henderson19dc85d2015-07-02 14:53:40 +01002398 if ((*ecx & CPUID_EXT_XSAVE) && (env->cr[4] & CR4_OSXSAVE_MASK)) {
2399 *ecx |= CPUID_EXT_OSXSAVE;
2400 }
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002401 *edx = env->features[FEAT_1_EDX];
Andreas Färberce3960e2012-12-17 03:27:07 +01002402 if (cs->nr_cores * cs->nr_threads > 1) {
2403 *ebx |= (cs->nr_cores * cs->nr_threads) << 16;
Richard Henderson19dc85d2015-07-02 14:53:40 +01002404 *edx |= CPUID_HT;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002405 }
2406 break;
2407 case 2:
2408 /* cache info: needed for Pentium Pro compatibility */
Benoît Canet787aaf52013-09-02 17:06:37 +02002409 if (cpu->cache_info_passthrough) {
2410 host_cpuid(index, 0, eax, ebx, ecx, edx);
2411 break;
2412 }
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002413 *eax = 1; /* Number of CPUID[EAX=2] calls required */
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002414 *ebx = 0;
Longpeng(Mike)14c985c2016-09-07 13:21:13 +08002415 if (!cpu->enable_l3_cache) {
2416 *ecx = 0;
2417 } else {
2418 *ecx = L3_N_DESCRIPTOR;
2419 }
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002420 *edx = (L1D_DESCRIPTOR << 16) | \
2421 (L1I_DESCRIPTOR << 8) | \
2422 (L2_DESCRIPTOR);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002423 break;
2424 case 4:
2425 /* cache info: needed for Core compatibility */
Benoît Canet787aaf52013-09-02 17:06:37 +02002426 if (cpu->cache_info_passthrough) {
2427 host_cpuid(index, count, eax, ebx, ecx, edx);
Paolo Bonzini76c29752013-11-19 17:49:46 +01002428 *eax &= ~0xFC000000;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002429 } else {
Aurelien Jarno2f7a21c2010-03-13 16:46:33 +01002430 *eax = 0;
Paolo Bonzini76c29752013-11-19 17:49:46 +01002431 switch (count) {
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002432 case 0: /* L1 dcache info */
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002433 *eax |= CPUID_4_TYPE_DCACHE | \
2434 CPUID_4_LEVEL(1) | \
2435 CPUID_4_SELF_INIT_LEVEL;
2436 *ebx = (L1D_LINE_SIZE - 1) | \
2437 ((L1D_PARTITIONS - 1) << 12) | \
2438 ((L1D_ASSOCIATIVITY - 1) << 22);
2439 *ecx = L1D_SETS - 1;
2440 *edx = CPUID_4_NO_INVD_SHARING;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002441 break;
2442 case 1: /* L1 icache info */
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002443 *eax |= CPUID_4_TYPE_ICACHE | \
2444 CPUID_4_LEVEL(1) | \
2445 CPUID_4_SELF_INIT_LEVEL;
2446 *ebx = (L1I_LINE_SIZE - 1) | \
2447 ((L1I_PARTITIONS - 1) << 12) | \
2448 ((L1I_ASSOCIATIVITY - 1) << 22);
2449 *ecx = L1I_SETS - 1;
2450 *edx = CPUID_4_NO_INVD_SHARING;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002451 break;
2452 case 2: /* L2 cache info */
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002453 *eax |= CPUID_4_TYPE_UNIFIED | \
2454 CPUID_4_LEVEL(2) | \
2455 CPUID_4_SELF_INIT_LEVEL;
Andreas Färberce3960e2012-12-17 03:27:07 +01002456 if (cs->nr_threads > 1) {
2457 *eax |= (cs->nr_threads - 1) << 14;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002458 }
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002459 *ebx = (L2_LINE_SIZE - 1) | \
2460 ((L2_PARTITIONS - 1) << 12) | \
2461 ((L2_ASSOCIATIVITY - 1) << 22);
2462 *ecx = L2_SETS - 1;
2463 *edx = CPUID_4_NO_INVD_SHARING;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002464 break;
Longpeng(Mike)14c985c2016-09-07 13:21:13 +08002465 case 3: /* L3 cache info */
2466 if (!cpu->enable_l3_cache) {
2467 *eax = 0;
2468 *ebx = 0;
2469 *ecx = 0;
2470 *edx = 0;
2471 break;
2472 }
2473 *eax |= CPUID_4_TYPE_UNIFIED | \
2474 CPUID_4_LEVEL(3) | \
2475 CPUID_4_SELF_INIT_LEVEL;
2476 pkg_offset = apicid_pkg_offset(cs->nr_cores, cs->nr_threads);
2477 *eax |= ((1 << pkg_offset) - 1) << 14;
2478 *ebx = (L3_N_LINE_SIZE - 1) | \
2479 ((L3_N_PARTITIONS - 1) << 12) | \
2480 ((L3_N_ASSOCIATIVITY - 1) << 22);
2481 *ecx = L3_N_SETS - 1;
2482 *edx = CPUID_4_INCLUSIVE | CPUID_4_COMPLEX_IDX;
2483 break;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002484 default: /* end of info */
2485 *eax = 0;
2486 *ebx = 0;
2487 *ecx = 0;
2488 *edx = 0;
2489 break;
Paolo Bonzini76c29752013-11-19 17:49:46 +01002490 }
2491 }
2492
2493 /* QEMU gives out its own APIC IDs, never pass down bits 31..26. */
2494 if ((*eax & 31) && cs->nr_cores > 1) {
2495 *eax |= (cs->nr_cores - 1) << 26;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002496 }
2497 break;
2498 case 5:
2499 /* mwait info: needed for Core compatibility */
2500 *eax = 0; /* Smallest monitor-line size in bytes */
2501 *ebx = 0; /* Largest monitor-line size in bytes */
2502 *ecx = CPUID_MWAIT_EMX | CPUID_MWAIT_IBE;
2503 *edx = 0;
2504 break;
2505 case 6:
2506 /* Thermal and Power Leaf */
Jan Kiszka28b8e4d2015-06-07 11:15:08 +02002507 *eax = env->features[FEAT_6_EAX];
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002508 *ebx = 0;
2509 *ecx = 0;
2510 *edx = 0;
2511 break;
Yang, Wei Yf7911682011-05-30 23:17:42 +08002512 case 7:
Eduardo Habkost13526722012-05-21 11:27:02 -03002513 /* Structured Extended Feature Flags Enumeration Leaf */
2514 if (count == 0) {
2515 *eax = 0; /* Maximum ECX value for sub-leaves */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002516 *ebx = env->features[FEAT_7_0_EBX]; /* Feature flags */
Huaitong Hanf74eefe2015-11-18 10:20:15 +08002517 *ecx = env->features[FEAT_7_0_ECX]; /* Feature flags */
Paolo Bonzini0f70ed42016-02-09 14:14:28 +01002518 if ((*ecx & CPUID_7_0_ECX_PKU) && env->cr[4] & CR4_PKE_MASK) {
2519 *ecx |= CPUID_7_0_ECX_OSPKE;
2520 }
Eduardo Habkost13526722012-05-21 11:27:02 -03002521 *edx = 0; /* Reserved */
Yang, Wei Yf7911682011-05-30 23:17:42 +08002522 } else {
2523 *eax = 0;
2524 *ebx = 0;
2525 *ecx = 0;
2526 *edx = 0;
2527 }
2528 break;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002529 case 9:
2530 /* Direct Cache Access Information Leaf */
2531 *eax = 0; /* Bits 0-31 in DCA_CAP MSR */
2532 *ebx = 0;
2533 *ecx = 0;
2534 *edx = 0;
2535 break;
2536 case 0xA:
2537 /* Architectural Performance Monitoring Leaf */
Eduardo Habkost9337e3b2013-07-26 17:09:36 -03002538 if (kvm_enabled() && cpu->enable_pmu) {
Andreas Färbera60f24b2012-12-01 05:35:08 +01002539 KVMState *s = cs->kvm_state;
Gleb Natapova0fa8202011-12-15 12:44:05 +02002540
2541 *eax = kvm_arch_get_supported_cpuid(s, 0xA, count, R_EAX);
2542 *ebx = kvm_arch_get_supported_cpuid(s, 0xA, count, R_EBX);
2543 *ecx = kvm_arch_get_supported_cpuid(s, 0xA, count, R_ECX);
2544 *edx = kvm_arch_get_supported_cpuid(s, 0xA, count, R_EDX);
2545 } else {
2546 *eax = 0;
2547 *ebx = 0;
2548 *ecx = 0;
2549 *edx = 0;
2550 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002551 break;
Radim Krčmář5232d002016-05-12 19:24:26 +02002552 case 0xB:
2553 /* Extended Topology Enumeration Leaf */
2554 if (!cpu->enable_cpuid_0xb) {
2555 *eax = *ebx = *ecx = *edx = 0;
2556 break;
2557 }
2558
2559 *ecx = count & 0xff;
2560 *edx = cpu->apic_id;
2561
2562 switch (count) {
2563 case 0:
Marc-André Lureaueab60fb2016-09-16 19:50:24 +04002564 *eax = apicid_core_offset(cs->nr_cores, cs->nr_threads);
2565 *ebx = cs->nr_threads;
Radim Krčmář5232d002016-05-12 19:24:26 +02002566 *ecx |= CPUID_TOPOLOGY_LEVEL_SMT;
2567 break;
2568 case 1:
Marc-André Lureaueab60fb2016-09-16 19:50:24 +04002569 *eax = apicid_pkg_offset(cs->nr_cores, cs->nr_threads);
2570 *ebx = cs->nr_cores * cs->nr_threads;
Radim Krčmář5232d002016-05-12 19:24:26 +02002571 *ecx |= CPUID_TOPOLOGY_LEVEL_CORE;
2572 break;
2573 default:
2574 *eax = 0;
2575 *ebx = 0;
2576 *ecx |= CPUID_TOPOLOGY_LEVEL_INVALID;
2577 }
2578
2579 assert(!(*eax & ~0x1f));
2580 *ebx &= 0xffff; /* The count doesn't need to be reliable. */
2581 break;
Paolo Bonzini2560f192013-10-02 17:54:57 +02002582 case 0xD: {
Sheng Yang51e49432010-06-17 15:18:14 +08002583 /* Processor Extended State */
Paolo Bonzini2560f192013-10-02 17:54:57 +02002584 *eax = 0;
2585 *ebx = 0;
2586 *ecx = 0;
2587 *edx = 0;
Richard Henderson19dc85d2015-07-02 14:53:40 +01002588 if (!(env->features[FEAT_1_ECX] & CPUID_EXT_XSAVE)) {
Sheng Yang51e49432010-06-17 15:18:14 +08002589 break;
2590 }
Eduardo Habkost4928cd62016-09-22 14:33:01 -03002591
Paolo Bonzini2560f192013-10-02 17:54:57 +02002592 if (count == 0) {
Eduardo Habkost96193c22016-09-22 17:41:35 -03002593 *ecx = xsave_area_size(x86_cpu_xsave_components(cpu));
2594 *eax = env->features[FEAT_XSAVE_COMP_LO];
2595 *edx = env->features[FEAT_XSAVE_COMP_HI];
Paolo Bonzini2560f192013-10-02 17:54:57 +02002596 *ebx = *ecx;
2597 } else if (count == 1) {
Paolo Bonzini0bb0b2d2014-11-24 15:54:43 +01002598 *eax = env->features[FEAT_XSAVE];
Richard Hendersonf4f11102015-07-02 15:57:14 +01002599 } else if (count < ARRAY_SIZE(x86_ext_save_areas)) {
Eduardo Habkost96193c22016-09-22 17:41:35 -03002600 if ((x86_cpu_xsave_components(cpu) >> count) & 1) {
2601 const ExtSaveArea *esa = &x86_ext_save_areas[count];
Liu Jinsong33f373d2013-12-03 04:17:50 +08002602 *eax = esa->size;
2603 *ebx = esa->offset;
Paolo Bonzini2560f192013-10-02 17:54:57 +02002604 }
Sheng Yang51e49432010-06-17 15:18:14 +08002605 }
2606 break;
Paolo Bonzini2560f192013-10-02 17:54:57 +02002607 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002608 case 0x80000000:
2609 *eax = env->cpuid_xlevel;
2610 *ebx = env->cpuid_vendor1;
2611 *edx = env->cpuid_vendor2;
2612 *ecx = env->cpuid_vendor3;
2613 break;
2614 case 0x80000001:
2615 *eax = env->cpuid_version;
2616 *ebx = 0;
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002617 *ecx = env->features[FEAT_8000_0001_ECX];
2618 *edx = env->features[FEAT_8000_0001_EDX];
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002619
2620 /* The Linux kernel checks for the CMPLegacy bit and
2621 * discards multiple thread information if it is set.
Stefan Weilcb8d4c82016-03-23 15:59:57 +01002622 * So don't set it here for Intel to make Linux guests happy.
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002623 */
Andreas Färberce3960e2012-12-17 03:27:07 +01002624 if (cs->nr_cores * cs->nr_threads > 1) {
Eduardo Habkost5eb2f7a2015-02-03 15:57:50 -02002625 if (env->cpuid_vendor1 != CPUID_VENDOR_INTEL_1 ||
2626 env->cpuid_vendor2 != CPUID_VENDOR_INTEL_2 ||
2627 env->cpuid_vendor3 != CPUID_VENDOR_INTEL_3) {
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002628 *ecx |= 1 << 1; /* CmpLegacy bit */
2629 }
2630 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002631 break;
2632 case 0x80000002:
2633 case 0x80000003:
2634 case 0x80000004:
2635 *eax = env->cpuid_model[(index - 0x80000002) * 4 + 0];
2636 *ebx = env->cpuid_model[(index - 0x80000002) * 4 + 1];
2637 *ecx = env->cpuid_model[(index - 0x80000002) * 4 + 2];
2638 *edx = env->cpuid_model[(index - 0x80000002) * 4 + 3];
2639 break;
2640 case 0x80000005:
2641 /* cache info (L1 cache) */
Benoît Canet787aaf52013-09-02 17:06:37 +02002642 if (cpu->cache_info_passthrough) {
2643 host_cpuid(index, 0, eax, ebx, ecx, edx);
2644 break;
2645 }
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002646 *eax = (L1_DTLB_2M_ASSOC << 24) | (L1_DTLB_2M_ENTRIES << 16) | \
2647 (L1_ITLB_2M_ASSOC << 8) | (L1_ITLB_2M_ENTRIES);
2648 *ebx = (L1_DTLB_4K_ASSOC << 24) | (L1_DTLB_4K_ENTRIES << 16) | \
2649 (L1_ITLB_4K_ASSOC << 8) | (L1_ITLB_4K_ENTRIES);
2650 *ecx = (L1D_SIZE_KB_AMD << 24) | (L1D_ASSOCIATIVITY_AMD << 16) | \
2651 (L1D_LINES_PER_TAG << 8) | (L1D_LINE_SIZE);
2652 *edx = (L1I_SIZE_KB_AMD << 24) | (L1I_ASSOCIATIVITY_AMD << 16) | \
2653 (L1I_LINES_PER_TAG << 8) | (L1I_LINE_SIZE);
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002654 break;
2655 case 0x80000006:
2656 /* cache info (L2 cache) */
Benoît Canet787aaf52013-09-02 17:06:37 +02002657 if (cpu->cache_info_passthrough) {
2658 host_cpuid(index, 0, eax, ebx, ecx, edx);
2659 break;
2660 }
Eduardo Habkost5e891bf2013-08-27 12:24:37 -03002661 *eax = (AMD_ENC_ASSOC(L2_DTLB_2M_ASSOC) << 28) | \
2662 (L2_DTLB_2M_ENTRIES << 16) | \
2663 (AMD_ENC_ASSOC(L2_ITLB_2M_ASSOC) << 12) | \
2664 (L2_ITLB_2M_ENTRIES);
2665 *ebx = (AMD_ENC_ASSOC(L2_DTLB_4K_ASSOC) << 28) | \
2666 (L2_DTLB_4K_ENTRIES << 16) | \
2667 (AMD_ENC_ASSOC(L2_ITLB_4K_ASSOC) << 12) | \
2668 (L2_ITLB_4K_ENTRIES);
2669 *ecx = (L2_SIZE_KB_AMD << 16) | \
2670 (AMD_ENC_ASSOC(L2_ASSOCIATIVITY) << 12) | \
2671 (L2_LINES_PER_TAG << 8) | (L2_LINE_SIZE);
Longpeng(Mike)14c985c2016-09-07 13:21:13 +08002672 if (!cpu->enable_l3_cache) {
2673 *edx = ((L3_SIZE_KB / 512) << 18) | \
2674 (AMD_ENC_ASSOC(L3_ASSOCIATIVITY) << 12) | \
2675 (L3_LINES_PER_TAG << 8) | (L3_LINE_SIZE);
2676 } else {
2677 *edx = ((L3_N_SIZE_KB_AMD / 512) << 18) | \
2678 (AMD_ENC_ASSOC(L3_N_ASSOCIATIVITY) << 12) | \
2679 (L3_N_LINES_PER_TAG << 8) | (L3_N_LINE_SIZE);
2680 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002681 break;
Marcelo Tosatti303752a2014-04-30 13:48:45 -03002682 case 0x80000007:
2683 *eax = 0;
2684 *ebx = 0;
2685 *ecx = 0;
2686 *edx = env->features[FEAT_8000_0007_EDX];
2687 break;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002688 case 0x80000008:
2689 /* virtual & phys address size in low 2 bytes. */
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002690 if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) {
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01002691 /* 64 bit processor, 48 bits virtual, configurable
2692 * physical bits.
2693 */
2694 *eax = 0x00003000 + cpu->phys_bits;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002695 } else {
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01002696 *eax = cpu->phys_bits;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002697 }
2698 *ebx = 0;
2699 *ecx = 0;
2700 *edx = 0;
Andreas Färberce3960e2012-12-17 03:27:07 +01002701 if (cs->nr_cores * cs->nr_threads > 1) {
2702 *ecx |= (cs->nr_cores * cs->nr_threads) - 1;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002703 }
2704 break;
2705 case 0x8000000A:
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002706 if (env->features[FEAT_8000_0001_ECX] & CPUID_EXT3_SVM) {
Eduardo Habkost9f3fb562012-12-04 17:34:38 -02002707 *eax = 0x00000001; /* SVM Revision */
2708 *ebx = 0x00000010; /* nr of ASIDs */
2709 *ecx = 0;
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002710 *edx = env->features[FEAT_SVM]; /* optional features */
Eduardo Habkost9f3fb562012-12-04 17:34:38 -02002711 } else {
2712 *eax = 0;
2713 *ebx = 0;
2714 *ecx = 0;
2715 *edx = 0;
2716 }
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002717 break;
brillywu@viatech.com.cnb3baa152011-06-01 09:59:52 +08002718 case 0xC0000000:
2719 *eax = env->cpuid_xlevel2;
2720 *ebx = 0;
2721 *ecx = 0;
2722 *edx = 0;
2723 break;
2724 case 0xC0000001:
2725 /* Support for VIA CPU's CPUID instruction */
2726 *eax = env->cpuid_version;
2727 *ebx = 0;
2728 *ecx = 0;
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002729 *edx = env->features[FEAT_C000_0001_EDX];
brillywu@viatech.com.cnb3baa152011-06-01 09:59:52 +08002730 break;
2731 case 0xC0000002:
2732 case 0xC0000003:
2733 case 0xC0000004:
2734 /* Reserved for the future, and now filled with zero */
2735 *eax = 0;
2736 *ebx = 0;
2737 *ecx = 0;
2738 *edx = 0;
2739 break;
Andre Przywarac6dc6f62010-03-11 14:38:55 +01002740 default:
2741 /* reserved values: zero */
2742 *eax = 0;
2743 *ebx = 0;
2744 *ecx = 0;
2745 *edx = 0;
2746 break;
2747 }
2748}
Andreas Färber5fd20872012-04-02 23:20:08 +02002749
2750/* CPUClass::reset() */
2751static void x86_cpu_reset(CPUState *s)
2752{
2753 X86CPU *cpu = X86_CPU(s);
2754 X86CPUClass *xcc = X86_CPU_GET_CLASS(cpu);
2755 CPUX86State *env = &cpu->env;
Richard Hendersona114d252015-07-07 16:13:10 +01002756 target_ulong cr4;
2757 uint64_t xcr0;
Andreas Färberc1958ae2012-04-03 00:16:24 +02002758 int i;
2759
Andreas Färber5fd20872012-04-02 23:20:08 +02002760 xcc->parent_reset(s);
2761
Eduardo Habkost5e992a82015-04-24 15:49:15 -03002762 memset(env, 0, offsetof(CPUX86State, end_reset_fields));
Andreas Färberc1958ae2012-04-03 00:16:24 +02002763
Andreas Färber00c8cb02013-09-04 02:19:44 +02002764 tlb_flush(s, 1);
Andreas Färberc1958ae2012-04-03 00:16:24 +02002765
2766 env->old_exception = -1;
2767
2768 /* init to reset state */
2769
Andreas Färberc1958ae2012-04-03 00:16:24 +02002770 env->hflags2 |= HF2_GIF_MASK;
2771
2772 cpu_x86_update_cr0(env, 0x60000010);
2773 env->a20_mask = ~0x0;
2774 env->smbase = 0x30000;
2775
2776 env->idt.limit = 0xffff;
2777 env->gdt.limit = 0xffff;
2778 env->ldt.limit = 0xffff;
2779 env->ldt.flags = DESC_P_MASK | (2 << DESC_TYPE_SHIFT);
2780 env->tr.limit = 0xffff;
2781 env->tr.flags = DESC_P_MASK | (11 << DESC_TYPE_SHIFT);
2782
2783 cpu_x86_load_seg_cache(env, R_CS, 0xf000, 0xffff0000, 0xffff,
2784 DESC_P_MASK | DESC_S_MASK | DESC_CS_MASK |
2785 DESC_R_MASK | DESC_A_MASK);
2786 cpu_x86_load_seg_cache(env, R_DS, 0, 0, 0xffff,
2787 DESC_P_MASK | DESC_S_MASK | DESC_W_MASK |
2788 DESC_A_MASK);
2789 cpu_x86_load_seg_cache(env, R_ES, 0, 0, 0xffff,
2790 DESC_P_MASK | DESC_S_MASK | DESC_W_MASK |
2791 DESC_A_MASK);
2792 cpu_x86_load_seg_cache(env, R_SS, 0, 0, 0xffff,
2793 DESC_P_MASK | DESC_S_MASK | DESC_W_MASK |
2794 DESC_A_MASK);
2795 cpu_x86_load_seg_cache(env, R_FS, 0, 0, 0xffff,
2796 DESC_P_MASK | DESC_S_MASK | DESC_W_MASK |
2797 DESC_A_MASK);
2798 cpu_x86_load_seg_cache(env, R_GS, 0, 0, 0xffff,
2799 DESC_P_MASK | DESC_S_MASK | DESC_W_MASK |
2800 DESC_A_MASK);
2801
2802 env->eip = 0xfff0;
2803 env->regs[R_EDX] = env->cpuid_version;
2804
2805 env->eflags = 0x2;
2806
2807 /* FPU init */
2808 for (i = 0; i < 8; i++) {
2809 env->fptags[i] = 1;
2810 }
Pavel Dovgalyuk5bde1402014-09-17 12:05:19 +04002811 cpu_set_fpuc(env, 0x37f);
Andreas Färberc1958ae2012-04-03 00:16:24 +02002812
2813 env->mxcsr = 0x1f80;
Richard Hendersona114d252015-07-07 16:13:10 +01002814 /* All units are in INIT state. */
2815 env->xstate_bv = 0;
Andreas Färberc1958ae2012-04-03 00:16:24 +02002816
2817 env->pat = 0x0007040600070406ULL;
2818 env->msr_ia32_misc_enable = MSR_IA32_MISC_ENABLE_DEFAULT;
2819
2820 memset(env->dr, 0, sizeof(env->dr));
2821 env->dr[6] = DR6_FIXED_1;
2822 env->dr[7] = DR7_FIXED_1;
Andreas Färberb3310ab2013-09-02 17:26:20 +02002823 cpu_breakpoint_remove_all(s, BP_CPU);
Andreas Färber75a34032013-09-02 16:57:02 +02002824 cpu_watchpoint_remove_all(s, BP_CPU);
Igor Mammedovdd673282012-07-23 15:22:27 +02002825
Richard Hendersona114d252015-07-07 16:13:10 +01002826 cr4 = 0;
Paolo Bonzinicfc3b072016-02-17 10:54:53 +01002827 xcr0 = XSTATE_FP_MASK;
Richard Hendersona114d252015-07-07 16:13:10 +01002828
2829#ifdef CONFIG_USER_ONLY
2830 /* Enable all the features for user-mode. */
2831 if (env->features[FEAT_1_EDX] & CPUID_SSE) {
Paolo Bonzinicfc3b072016-02-17 10:54:53 +01002832 xcr0 |= XSTATE_SSE_MASK;
Richard Hendersona114d252015-07-07 16:13:10 +01002833 }
Paolo Bonzini0f70ed42016-02-09 14:14:28 +01002834 for (i = 2; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
2835 const ExtSaveArea *esa = &x86_ext_save_areas[i];
Eduardo Habkost9646f492016-09-22 18:58:39 -03002836 if (env->features[esa->feature] & esa->bits) {
Paolo Bonzini0f70ed42016-02-09 14:14:28 +01002837 xcr0 |= 1ull << i;
2838 }
Richard Hendersona114d252015-07-07 16:13:10 +01002839 }
Paolo Bonzini0f70ed42016-02-09 14:14:28 +01002840
Richard Hendersona114d252015-07-07 16:13:10 +01002841 if (env->features[FEAT_1_ECX] & CPUID_EXT_XSAVE) {
2842 cr4 |= CR4_OSFXSR_MASK | CR4_OSXSAVE_MASK;
2843 }
Richard Henderson07929f22015-11-18 12:55:47 +01002844 if (env->features[FEAT_7_0_EBX] & CPUID_7_0_EBX_FSGSBASE) {
2845 cr4 |= CR4_FSGSBASE_MASK;
2846 }
Richard Hendersona114d252015-07-07 16:13:10 +01002847#endif
2848
2849 env->xcr0 = xcr0;
2850 cpu_x86_update_cr4(env, cr4);
Fernando Luis Vázquez Cao05226042013-12-06 17:33:01 +09002851
Alex Williamson9db2efd2014-08-14 15:39:39 -06002852 /*
2853 * SDM 11.11.5 requires:
2854 * - IA32_MTRR_DEF_TYPE MSR.E = 0
2855 * - IA32_MTRR_PHYSMASKn.V = 0
2856 * All other bits are undefined. For simplification, zero it all.
2857 */
2858 env->mtrr_deftype = 0;
2859 memset(env->mtrr_var, 0, sizeof(env->mtrr_var));
2860 memset(env->mtrr_fixed, 0, sizeof(env->mtrr_fixed));
2861
Igor Mammedovdd673282012-07-23 15:22:27 +02002862#if !defined(CONFIG_USER_ONLY)
2863 /* We hard-wire the BSP to the first CPU. */
Nadav Amit9cb11fd2015-04-02 02:58:36 +03002864 apic_designate_bsp(cpu->apic_state, s->cpu_index == 0);
Igor Mammedovdd673282012-07-23 15:22:27 +02002865
Andreas Färber259186a2013-01-17 18:51:17 +01002866 s->halted = !cpu_is_bsp(cpu);
Paolo Bonzini50a2c6e2013-03-20 13:11:56 +01002867
2868 if (kvm_enabled()) {
2869 kvm_arch_reset_vcpu(cpu);
2870 }
Igor Mammedovdd673282012-07-23 15:22:27 +02002871#endif
Andreas Färber5fd20872012-04-02 23:20:08 +02002872}
2873
Igor Mammedovdd673282012-07-23 15:22:27 +02002874#ifndef CONFIG_USER_ONLY
2875bool cpu_is_bsp(X86CPU *cpu)
2876{
Chen Fan02e51482013-12-23 17:04:02 +08002877 return cpu_get_apic_base(cpu->apic_state) & MSR_IA32_APICBASE_BSP;
Igor Mammedovdd673282012-07-23 15:22:27 +02002878}
Igor Mammedov65dee382012-07-23 15:22:28 +02002879
2880/* TODO: remove me, when reset over QOM tree is implemented */
2881static void x86_cpu_machine_reset_cb(void *opaque)
2882{
2883 X86CPU *cpu = opaque;
2884 cpu_reset(CPU(cpu));
2885}
Igor Mammedovdd673282012-07-23 15:22:27 +02002886#endif
2887
Andreas Färberde024812012-04-03 00:00:17 +02002888static void mce_init(X86CPU *cpu)
2889{
2890 CPUX86State *cenv = &cpu->env;
2891 unsigned int bank;
2892
2893 if (((cenv->cpuid_version >> 8) & 0xf) >= 6
Eduardo Habkost0514ef22013-04-22 16:00:15 -03002894 && (cenv->features[FEAT_1_EDX] & (CPUID_MCE | CPUID_MCA)) ==
Andreas Färberde024812012-04-03 00:00:17 +02002895 (CPUID_MCE | CPUID_MCA)) {
Ashok Raj87f8b622016-06-22 14:56:21 +08002896 cenv->mcg_cap = MCE_CAP_DEF | MCE_BANKS_DEF |
2897 (cpu->enable_lmce ? MCG_LMCE_P : 0);
Andreas Färberde024812012-04-03 00:00:17 +02002898 cenv->mcg_ctl = ~(uint64_t)0;
2899 for (bank = 0; bank < MCE_BANKS_DEF; bank++) {
2900 cenv->mce_banks[bank * 4] = ~(uint64_t)0;
2901 }
2902 }
2903}
2904
Igor Mammedovbdeec802012-10-13 22:35:39 +02002905#ifndef CONFIG_USER_ONLY
Igor Mammedovd3c64d62013-04-05 16:36:54 +02002906static void x86_cpu_apic_create(X86CPU *cpu, Error **errp)
Igor Mammedovbdeec802012-10-13 22:35:39 +02002907{
Andreas Färber449994e2012-10-10 12:18:02 +02002908 APICCommonState *apic;
Igor Mammedovbdeec802012-10-13 22:35:39 +02002909 const char *apic_type = "apic";
2910
Paolo Bonzini15eafc22015-12-17 17:16:08 +01002911 if (kvm_apic_in_kernel()) {
Igor Mammedovbdeec802012-10-13 22:35:39 +02002912 apic_type = "kvm-apic";
2913 } else if (xen_enabled()) {
2914 apic_type = "xen-apic";
2915 }
2916
Chen Fan46232aa2015-09-16 17:19:14 +08002917 cpu->apic_state = DEVICE(object_new(apic_type));
Igor Mammedovbdeec802012-10-13 22:35:39 +02002918
Igor Mammedov6816b1b2016-07-06 08:20:52 +02002919 object_property_add_child(OBJECT(cpu), "lapic",
2920 OBJECT(cpu->apic_state), &error_abort);
Igor Mammedov67e55ca2016-07-06 08:20:53 +02002921 object_unref(OBJECT(cpu->apic_state));
Igor Mammedov6816b1b2016-07-06 08:20:52 +02002922
Eduardo Habkost7e72a452014-12-18 23:20:10 -02002923 qdev_prop_set_uint8(cpu->apic_state, "id", cpu->apic_id);
Igor Mammedovbdeec802012-10-13 22:35:39 +02002924 /* TODO: convert to link<> */
Chen Fan02e51482013-12-23 17:04:02 +08002925 apic = APIC_COMMON(cpu->apic_state);
Andreas Färber60671e52012-10-10 14:10:07 +02002926 apic->cpu = cpu;
Chen Fan8d42d2d2015-09-16 17:19:11 +08002927 apic->apicbase = APIC_DEFAULT_ADDRESS | MSR_IA32_APICBASE_ENABLE;
Igor Mammedovd3c64d62013-04-05 16:36:54 +02002928}
2929
2930static void x86_cpu_apic_realize(X86CPU *cpu, Error **errp)
2931{
Chen Fan8d42d2d2015-09-16 17:19:11 +08002932 APICCommonState *apic;
2933 static bool apic_mmio_map_once;
2934
Chen Fan02e51482013-12-23 17:04:02 +08002935 if (cpu->apic_state == NULL) {
Igor Mammedovd3c64d62013-04-05 16:36:54 +02002936 return;
2937 }
Markus Armbruster6e8e2652015-02-05 10:29:15 +01002938 object_property_set_bool(OBJECT(cpu->apic_state), true, "realized",
2939 errp);
Chen Fan8d42d2d2015-09-16 17:19:11 +08002940
2941 /* Map APIC MMIO area */
2942 apic = APIC_COMMON(cpu->apic_state);
2943 if (!apic_mmio_map_once) {
2944 memory_region_add_subregion_overlap(get_system_memory(),
2945 apic->apicbase &
2946 MSR_IA32_APICBASE_BASE,
2947 &apic->io_memory,
2948 0x1000);
2949 apic_mmio_map_once = true;
2950 }
Igor Mammedovbdeec802012-10-13 22:35:39 +02002951}
Paolo Bonzinif809c602015-03-31 14:12:25 +02002952
2953static void x86_cpu_machine_done(Notifier *n, void *unused)
2954{
2955 X86CPU *cpu = container_of(n, X86CPU, machine_done);
2956 MemoryRegion *smram =
2957 (MemoryRegion *) object_resolve_path("/machine/smram", NULL);
2958
2959 if (smram) {
2960 cpu->smram = g_new(MemoryRegion, 1);
2961 memory_region_init_alias(cpu->smram, OBJECT(cpu), "smram",
2962 smram, 0, 1ull << 32);
2963 memory_region_set_enabled(cpu->smram, false);
2964 memory_region_add_subregion_overlap(cpu->cpu_as_root, 0, cpu->smram, 1);
2965 }
2966}
Igor Mammedovd3c64d62013-04-05 16:36:54 +02002967#else
2968static void x86_cpu_apic_realize(X86CPU *cpu, Error **errp)
2969{
2970}
Igor Mammedovbdeec802012-10-13 22:35:39 +02002971#endif
2972
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01002973/* Note: Only safe for use on x86(-64) hosts */
2974static uint32_t x86_host_phys_bits(void)
2975{
2976 uint32_t eax;
2977 uint32_t host_phys_bits;
2978
2979 host_cpuid(0x80000000, 0, &eax, NULL, NULL, NULL);
2980 if (eax >= 0x80000008) {
2981 host_cpuid(0x80000008, 0, &eax, NULL, NULL, NULL);
2982 /* Note: According to AMD doc 25481 rev 2.34 they have a field
2983 * at 23:16 that can specify a maximum physical address bits for
2984 * the guest that can override this value; but I've not seen
2985 * anything with that set.
2986 */
2987 host_phys_bits = eax & 0xff;
2988 } else {
2989 /* It's an odd 64 bit machine that doesn't have the leaf for
2990 * physical address bits; fall back to 36 that's most older
2991 * Intel.
2992 */
2993 host_phys_bits = 36;
2994 }
2995
2996 return host_phys_bits;
2997}
Wei Huange48638f2014-10-21 11:00:45 -04002998
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03002999static void x86_cpu_adjust_level(X86CPU *cpu, uint32_t *min, uint32_t value)
3000{
3001 if (*min < value) {
3002 *min = value;
3003 }
3004}
3005
3006/* Increase cpuid_min_{level,xlevel,xlevel2} automatically, if appropriate */
3007static void x86_cpu_adjust_feat_level(X86CPU *cpu, FeatureWord w)
3008{
3009 CPUX86State *env = &cpu->env;
3010 FeatureWordInfo *fi = &feature_word_info[w];
3011 uint32_t eax = fi->cpuid_eax;
3012 uint32_t region = eax & 0xF0000000;
3013
3014 if (!env->features[w]) {
3015 return;
3016 }
3017
3018 switch (region) {
3019 case 0x00000000:
3020 x86_cpu_adjust_level(cpu, &env->cpuid_min_level, eax);
3021 break;
3022 case 0x80000000:
3023 x86_cpu_adjust_level(cpu, &env->cpuid_min_xlevel, eax);
3024 break;
3025 case 0xC0000000:
3026 x86_cpu_adjust_level(cpu, &env->cpuid_min_xlevel2, eax);
3027 break;
3028 }
3029}
3030
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003031/* Calculate XSAVE components based on the configured CPU feature flags */
3032static void x86_cpu_enable_xsave_components(X86CPU *cpu)
3033{
3034 CPUX86State *env = &cpu->env;
3035 int i;
Eduardo Habkost96193c22016-09-22 17:41:35 -03003036 uint64_t mask;
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003037
3038 if (!(env->features[FEAT_1_ECX] & CPUID_EXT_XSAVE)) {
3039 return;
3040 }
3041
Eduardo Habkost96193c22016-09-22 17:41:35 -03003042 mask = (XSTATE_FP_MASK | XSTATE_SSE_MASK);
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003043 for (i = 2; i < ARRAY_SIZE(x86_ext_save_areas); i++) {
3044 const ExtSaveArea *esa = &x86_ext_save_areas[i];
3045 if (env->features[esa->feature] & esa->bits) {
Eduardo Habkost96193c22016-09-22 17:41:35 -03003046 mask |= (1ULL << i);
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003047 }
3048 }
3049
Eduardo Habkost96193c22016-09-22 17:41:35 -03003050 env->features[FEAT_XSAVE_COMP_LO] = mask;
3051 env->features[FEAT_XSAVE_COMP_HI] = mask >> 32;
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003052}
3053
Wei Huange48638f2014-10-21 11:00:45 -04003054#define IS_INTEL_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_INTEL_1 && \
3055 (env)->cpuid_vendor2 == CPUID_VENDOR_INTEL_2 && \
3056 (env)->cpuid_vendor3 == CPUID_VENDOR_INTEL_3)
3057#define IS_AMD_CPU(env) ((env)->cpuid_vendor1 == CPUID_VENDOR_AMD_1 && \
3058 (env)->cpuid_vendor2 == CPUID_VENDOR_AMD_2 && \
3059 (env)->cpuid_vendor3 == CPUID_VENDOR_AMD_3)
Andreas Färber2b6f2942013-01-16 03:41:47 +01003060static void x86_cpu_realizefn(DeviceState *dev, Error **errp)
Andreas Färber7a059952012-05-09 23:15:32 +02003061{
Andreas Färber14a10fc2013-07-27 02:53:25 +02003062 CPUState *cs = CPU(dev);
Andreas Färber2b6f2942013-01-16 03:41:47 +01003063 X86CPU *cpu = X86_CPU(dev);
3064 X86CPUClass *xcc = X86_CPU_GET_CLASS(dev);
Igor Mammedovb34d12d2012-10-22 17:03:00 +02003065 CPUX86State *env = &cpu->env;
Andreas Färber2b6f2942013-01-16 03:41:47 +01003066 Error *local_err = NULL;
Wei Huange48638f2014-10-21 11:00:45 -04003067 static bool ht_warned;
Igor Mammedovdc15c052016-06-06 17:16:44 +02003068 FeatureWord w;
Igor Mammedovb34d12d2012-10-22 17:03:00 +02003069
Igor Mammedov104494e2016-06-06 17:16:45 +02003070 if (xcc->kvm_required && !kvm_enabled()) {
3071 char *name = x86_cpu_class_get_model_name(xcc);
3072 error_setg(&local_err, "CPU model '%s' requires KVM", name);
3073 g_free(name);
3074 goto out;
3075 }
3076
Igor Mammedovd9c84f12016-07-06 08:20:37 +02003077 if (cpu->apic_id == UNASSIGNED_APIC_ID) {
Eduardo Habkost9886e832014-12-18 23:31:11 -02003078 error_setg(errp, "apic-id property was not initialized properly");
3079 return;
3080 }
3081
Igor Mammedovdc15c052016-06-06 17:16:44 +02003082 /*TODO: cpu->host_features incorrectly overwrites features
3083 * set using "feat=on|off". Once we fix this, we can convert
3084 * plus_features & minus_features to global properties
3085 * inside x86_cpu_parse_featurestr() too.
3086 */
3087 if (cpu->host_features) {
3088 for (w = 0; w < FEATURE_WORDS; w++) {
3089 env->features[w] =
3090 x86_cpu_get_supported_feature_word(w, cpu->migratable);
3091 }
3092 }
3093
3094 for (w = 0; w < FEATURE_WORDS; w++) {
3095 cpu->env.features[w] |= plus_features[w];
3096 cpu->env.features[w] &= ~minus_features[w];
3097 }
3098
Eduardo Habkostaec661d2016-09-26 19:11:14 -03003099 if (!kvm_enabled() || !cpu->expose_kvm) {
3100 env->features[FEAT_KVM] = 0;
3101 }
3102
Eduardo Habkost2ca8a8b2016-09-22 17:27:56 -03003103 x86_cpu_enable_xsave_components(cpu);
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03003104
3105 /* CPUID[EAX=7,ECX=0].EBX always increased level automatically: */
3106 x86_cpu_adjust_feat_level(cpu, FEAT_7_0_EBX);
3107 if (cpu->full_cpuid_auto_level) {
3108 x86_cpu_adjust_feat_level(cpu, FEAT_1_EDX);
3109 x86_cpu_adjust_feat_level(cpu, FEAT_1_ECX);
3110 x86_cpu_adjust_feat_level(cpu, FEAT_6_EAX);
3111 x86_cpu_adjust_feat_level(cpu, FEAT_7_0_ECX);
3112 x86_cpu_adjust_feat_level(cpu, FEAT_8000_0001_EDX);
3113 x86_cpu_adjust_feat_level(cpu, FEAT_8000_0001_ECX);
3114 x86_cpu_adjust_feat_level(cpu, FEAT_8000_0007_EDX);
3115 x86_cpu_adjust_feat_level(cpu, FEAT_C000_0001_EDX);
3116 x86_cpu_adjust_feat_level(cpu, FEAT_SVM);
3117 x86_cpu_adjust_feat_level(cpu, FEAT_XSAVE);
Eduardo Habkost0c3d7c02016-09-21 15:01:35 -03003118 /* SVM requires CPUID[0x8000000A] */
3119 if (env->features[FEAT_8000_0001_ECX] & CPUID_EXT3_SVM) {
3120 x86_cpu_adjust_level(cpu, &env->cpuid_min_xlevel, 0x8000000A);
3121 }
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03003122 }
3123
3124 /* Set cpuid_*level* based on cpuid_min_*level, if not explicitly set */
3125 if (env->cpuid_level == UINT32_MAX) {
3126 env->cpuid_level = env->cpuid_min_level;
3127 }
3128 if (env->cpuid_xlevel == UINT32_MAX) {
3129 env->cpuid_xlevel = env->cpuid_min_xlevel;
3130 }
3131 if (env->cpuid_xlevel2 == UINT32_MAX) {
3132 env->cpuid_xlevel2 = env->cpuid_min_xlevel2;
Igor Mammedovb34d12d2012-10-22 17:03:00 +02003133 }
Andreas Färber7a059952012-05-09 23:15:32 +02003134
Eduardo Habkost9997cf72016-04-15 14:54:26 -03003135 if (x86_cpu_filter_features(cpu) && cpu->enforce_cpuid) {
3136 error_setg(&local_err,
3137 kvm_enabled() ?
3138 "Host doesn't support requested features" :
3139 "TCG doesn't support requested features");
3140 goto out;
3141 }
3142
Igor Mammedov9b15cd92012-12-28 21:01:17 +01003143 /* On AMD CPUs, some CPUID[8000_0001].EDX bits must match the bits on
3144 * CPUID[1].EDX.
3145 */
Wei Huange48638f2014-10-21 11:00:45 -04003146 if (IS_AMD_CPU(env)) {
Eduardo Habkost0514ef22013-04-22 16:00:15 -03003147 env->features[FEAT_8000_0001_EDX] &= ~CPUID_EXT2_AMD_ALIASES;
3148 env->features[FEAT_8000_0001_EDX] |= (env->features[FEAT_1_EDX]
Igor Mammedov9b15cd92012-12-28 21:01:17 +01003149 & CPUID_EXT2_AMD_ALIASES);
3150 }
3151
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01003152 /* For 64bit systems think about the number of physical bits to present.
3153 * ideally this should be the same as the host; anything other than matching
3154 * the host can cause incorrect guest behaviour.
3155 * QEMU used to pick the magic value of 40 bits that corresponds to
3156 * consumer AMD devices but nothing else.
3157 */
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003158 if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) {
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003159 if (kvm_enabled()) {
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01003160 uint32_t host_phys_bits = x86_host_phys_bits();
3161 static bool warned;
3162
3163 if (cpu->host_phys_bits) {
3164 /* The user asked for us to use the host physical bits */
3165 cpu->phys_bits = host_phys_bits;
3166 }
3167
3168 /* Print a warning if the user set it to a value that's not the
3169 * host value.
3170 */
3171 if (cpu->phys_bits != host_phys_bits && cpu->phys_bits != 0 &&
3172 !warned) {
3173 error_report("Warning: Host physical bits (%u)"
3174 " does not match phys-bits property (%u)",
3175 host_phys_bits, cpu->phys_bits);
3176 warned = true;
3177 }
3178
3179 if (cpu->phys_bits &&
3180 (cpu->phys_bits > TARGET_PHYS_ADDR_SPACE_BITS ||
3181 cpu->phys_bits < 32)) {
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003182 error_setg(errp, "phys-bits should be between 32 and %u "
3183 " (but is %u)",
3184 TARGET_PHYS_ADDR_SPACE_BITS, cpu->phys_bits);
3185 return;
3186 }
3187 } else {
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01003188 if (cpu->phys_bits && cpu->phys_bits != TCG_PHYS_ADDR_BITS) {
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003189 error_setg(errp, "TCG only supports phys-bits=%u",
3190 TCG_PHYS_ADDR_BITS);
3191 return;
3192 }
3193 }
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01003194 /* 0 means it was not explicitly set by the user (or by machine
3195 * compat_props or by the host code above). In this case, the default
3196 * is the value used by TCG (40).
3197 */
3198 if (cpu->phys_bits == 0) {
3199 cpu->phys_bits = TCG_PHYS_ADDR_BITS;
3200 }
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003201 } else {
3202 /* For 32 bit systems don't use the user set value, but keep
3203 * phys_bits consistent with what we tell the guest.
3204 */
3205 if (cpu->phys_bits != 0) {
3206 error_setg(errp, "phys-bits is not user-configurable in 32 bit");
3207 return;
3208 }
Eduardo Habkostfefb41b2014-04-30 13:48:39 -03003209
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003210 if (env->features[FEAT_1_EDX] & CPUID_PSE36) {
3211 cpu->phys_bits = 36;
3212 } else {
3213 cpu->phys_bits = 32;
3214 }
3215 }
Eduardo Habkost42ecaba2015-02-12 23:04:50 -02003216 cpu_exec_init(cs, &error_abort);
3217
Eduardo Habkost57f24532015-02-12 22:57:44 -02003218 if (tcg_enabled()) {
3219 tcg_x86_init();
3220 }
3221
Igor Mammedov65dee382012-07-23 15:22:28 +02003222#ifndef CONFIG_USER_ONLY
3223 qemu_register_reset(x86_cpu_machine_reset_cb, cpu);
Igor Mammedovbdeec802012-10-13 22:35:39 +02003224
Eduardo Habkost0514ef22013-04-22 16:00:15 -03003225 if (cpu->env.features[FEAT_1_EDX] & CPUID_APIC || smp_cpus > 1) {
Igor Mammedovd3c64d62013-04-05 16:36:54 +02003226 x86_cpu_apic_create(cpu, &local_err);
Andreas Färber2b6f2942013-01-16 03:41:47 +01003227 if (local_err != NULL) {
Igor Mammedov4dc1f442013-04-05 16:36:53 +02003228 goto out;
Igor Mammedovbdeec802012-10-13 22:35:39 +02003229 }
3230 }
Igor Mammedov65dee382012-07-23 15:22:28 +02003231#endif
3232
Andreas Färber7a059952012-05-09 23:15:32 +02003233 mce_init(cpu);
Paolo Bonzini2001d0c2015-03-31 14:11:09 +02003234
3235#ifndef CONFIG_USER_ONLY
3236 if (tcg_enabled()) {
Peter Maydell56943e82016-01-21 14:15:04 +00003237 AddressSpace *newas = g_new(AddressSpace, 1);
3238
Paolo Bonzinif809c602015-03-31 14:12:25 +02003239 cpu->cpu_as_mem = g_new(MemoryRegion, 1);
Paolo Bonzini2001d0c2015-03-31 14:11:09 +02003240 cpu->cpu_as_root = g_new(MemoryRegion, 1);
Paolo Bonzinif809c602015-03-31 14:12:25 +02003241
3242 /* Outer container... */
3243 memory_region_init(cpu->cpu_as_root, OBJECT(cpu), "memory", ~0ull);
Paolo Bonzini2001d0c2015-03-31 14:11:09 +02003244 memory_region_set_enabled(cpu->cpu_as_root, true);
Paolo Bonzinif809c602015-03-31 14:12:25 +02003245
3246 /* ... with two regions inside: normal system memory with low
3247 * priority, and...
3248 */
3249 memory_region_init_alias(cpu->cpu_as_mem, OBJECT(cpu), "memory",
3250 get_system_memory(), 0, ~0ull);
3251 memory_region_add_subregion_overlap(cpu->cpu_as_root, 0, cpu->cpu_as_mem, 0);
3252 memory_region_set_enabled(cpu->cpu_as_mem, true);
Peter Maydell56943e82016-01-21 14:15:04 +00003253 address_space_init(newas, cpu->cpu_as_root, "CPU");
Peter Maydell12ebc9a2016-01-21 14:15:04 +00003254 cs->num_ases = 1;
Peter Maydell56943e82016-01-21 14:15:04 +00003255 cpu_address_space_init(cs, newas, 0);
Paolo Bonzinif809c602015-03-31 14:12:25 +02003256
3257 /* ... SMRAM with higher priority, linked from /machine/smram. */
3258 cpu->machine_done.notify = x86_cpu_machine_done;
3259 qemu_add_machine_init_done_notifier(&cpu->machine_done);
Paolo Bonzini2001d0c2015-03-31 14:11:09 +02003260 }
3261#endif
3262
Andreas Färber14a10fc2013-07-27 02:53:25 +02003263 qemu_init_vcpu(cs);
Igor Mammedovd3c64d62013-04-05 16:36:54 +02003264
Wei Huange48638f2014-10-21 11:00:45 -04003265 /* Only Intel CPUs support hyperthreading. Even though QEMU fixes this
3266 * issue by adjusting CPUID_0000_0001_EBX and CPUID_8000_0008_ECX
3267 * based on inputs (sockets,cores,threads), it is still better to gives
3268 * users a warning.
3269 *
3270 * NOTE: the following code has to follow qemu_init_vcpu(). Otherwise
3271 * cs->nr_threads hasn't be populated yet and the checking is incorrect.
3272 */
3273 if (!IS_INTEL_CPU(env) && cs->nr_threads > 1 && !ht_warned) {
3274 error_report("AMD CPU doesn't support hyperthreading. Please configure"
3275 " -smp options properly.");
3276 ht_warned = true;
3277 }
3278
Igor Mammedovd3c64d62013-04-05 16:36:54 +02003279 x86_cpu_apic_realize(cpu, &local_err);
3280 if (local_err != NULL) {
3281 goto out;
3282 }
Andreas Färber14a10fc2013-07-27 02:53:25 +02003283 cpu_reset(cs);
Andreas Färber2b6f2942013-01-16 03:41:47 +01003284
Igor Mammedov4dc1f442013-04-05 16:36:53 +02003285 xcc->parent_realize(dev, &local_err);
Paolo Bonzini2001d0c2015-03-31 14:11:09 +02003286
Igor Mammedov4dc1f442013-04-05 16:36:53 +02003287out:
3288 if (local_err != NULL) {
3289 error_propagate(errp, local_err);
3290 return;
3291 }
Andreas Färber7a059952012-05-09 23:15:32 +02003292}
3293
Igor Mammedovc8847762016-06-24 16:01:02 +02003294static void x86_cpu_unrealizefn(DeviceState *dev, Error **errp)
3295{
3296 X86CPU *cpu = X86_CPU(dev);
3297
3298#ifndef CONFIG_USER_ONLY
3299 cpu_remove_sync(CPU(dev));
3300 qemu_unregister_reset(x86_cpu_machine_reset_cb, dev);
3301#endif
3302
3303 if (cpu->apic_state) {
3304 object_unparent(OBJECT(cpu->apic_state));
3305 cpu->apic_state = NULL;
3306 }
3307}
3308
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003309typedef struct BitProperty {
3310 uint32_t *ptr;
3311 uint32_t mask;
3312} BitProperty;
3313
Eric Blaked7bce992016-01-29 06:48:55 -07003314static void x86_cpu_get_bit_prop(Object *obj, Visitor *v, const char *name,
3315 void *opaque, Error **errp)
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003316{
3317 BitProperty *fp = opaque;
3318 bool value = (*fp->ptr & fp->mask) == fp->mask;
Eric Blake51e72bc2016-01-29 06:48:54 -07003319 visit_type_bool(v, name, &value, errp);
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003320}
3321
Eric Blaked7bce992016-01-29 06:48:55 -07003322static void x86_cpu_set_bit_prop(Object *obj, Visitor *v, const char *name,
3323 void *opaque, Error **errp)
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003324{
3325 DeviceState *dev = DEVICE(obj);
3326 BitProperty *fp = opaque;
3327 Error *local_err = NULL;
3328 bool value;
3329
3330 if (dev->realized) {
3331 qdev_prop_set_after_realize(dev, name, errp);
3332 return;
3333 }
3334
Eric Blake51e72bc2016-01-29 06:48:54 -07003335 visit_type_bool(v, name, &value, &local_err);
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003336 if (local_err) {
3337 error_propagate(errp, local_err);
3338 return;
3339 }
3340
3341 if (value) {
3342 *fp->ptr |= fp->mask;
3343 } else {
3344 *fp->ptr &= ~fp->mask;
3345 }
3346}
3347
3348static void x86_cpu_release_bit_prop(Object *obj, const char *name,
3349 void *opaque)
3350{
3351 BitProperty *prop = opaque;
3352 g_free(prop);
3353}
3354
3355/* Register a boolean property to get/set a single bit in a uint32_t field.
3356 *
3357 * The same property name can be registered multiple times to make it affect
3358 * multiple bits in the same FeatureWord. In that case, the getter will return
3359 * true only if all bits are set.
3360 */
3361static void x86_cpu_register_bit_prop(X86CPU *cpu,
3362 const char *prop_name,
3363 uint32_t *field,
3364 int bitnr)
3365{
3366 BitProperty *fp;
3367 ObjectProperty *op;
3368 uint32_t mask = (1UL << bitnr);
3369
3370 op = object_property_find(OBJECT(cpu), prop_name, NULL);
3371 if (op) {
3372 fp = op->opaque;
3373 assert(fp->ptr == field);
3374 fp->mask |= mask;
3375 } else {
3376 fp = g_new0(BitProperty, 1);
3377 fp->ptr = field;
3378 fp->mask = mask;
3379 object_property_add(OBJECT(cpu), prop_name, "bool",
3380 x86_cpu_get_bit_prop,
3381 x86_cpu_set_bit_prop,
3382 x86_cpu_release_bit_prop, fp, &error_abort);
3383 }
3384}
3385
3386static void x86_cpu_register_feature_bit_props(X86CPU *cpu,
3387 FeatureWord w,
3388 int bitnr)
3389{
3390 Object *obj = OBJECT(cpu);
3391 int i;
3392 char **names;
3393 FeatureWordInfo *fi = &feature_word_info[w];
3394
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003395 if (!fi->feat_names[bitnr]) {
3396 return;
3397 }
3398
3399 names = g_strsplit(fi->feat_names[bitnr], "|", 0);
3400
3401 feat2prop(names[0]);
3402 x86_cpu_register_bit_prop(cpu, names[0], &cpu->env.features[w], bitnr);
3403
3404 for (i = 1; names[i]; i++) {
3405 feat2prop(names[i]);
Eduardo Habkostd461a442015-07-09 12:24:43 -03003406 object_property_add_alias(obj, names[i], obj, names[0],
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003407 &error_abort);
3408 }
3409
3410 g_strfreev(names);
3411}
3412
Andreas Färberde024812012-04-03 00:00:17 +02003413static void x86_cpu_initfn(Object *obj)
3414{
Andreas Färber55e5c282012-12-17 06:18:02 +01003415 CPUState *cs = CPU(obj);
Andreas Färberde024812012-04-03 00:00:17 +02003416 X86CPU *cpu = X86_CPU(obj);
Eduardo Habkostd940ee92014-02-10 08:21:30 -02003417 X86CPUClass *xcc = X86_CPU_GET_CLASS(obj);
Andreas Färberde024812012-04-03 00:00:17 +02003418 CPUX86State *env = &cpu->env;
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003419 FeatureWord w;
Andreas Färberde024812012-04-03 00:00:17 +02003420
Andreas Färberc05efcb2013-01-17 12:13:41 +01003421 cs->env_ptr = env;
Andreas Färber71ad61d2012-04-17 12:10:29 +02003422
3423 object_property_add(obj, "family", "int",
Andreas Färber95b85192012-04-17 14:42:22 +02003424 x86_cpuid_version_get_family,
Andreas Färber71ad61d2012-04-17 12:10:29 +02003425 x86_cpuid_version_set_family, NULL, NULL, NULL);
Andreas Färberc5291a42012-04-17 12:16:39 +02003426 object_property_add(obj, "model", "int",
Andreas Färber67e30c82012-04-17 14:48:14 +02003427 x86_cpuid_version_get_model,
Andreas Färberc5291a42012-04-17 12:16:39 +02003428 x86_cpuid_version_set_model, NULL, NULL, NULL);
Andreas Färber036e2222012-04-17 14:14:18 +02003429 object_property_add(obj, "stepping", "int",
Andreas Färber35112e42012-04-17 14:50:53 +02003430 x86_cpuid_version_get_stepping,
Andreas Färber036e2222012-04-17 14:14:18 +02003431 x86_cpuid_version_set_stepping, NULL, NULL, NULL);
Andreas Färberd480e1a2012-04-17 19:22:58 +02003432 object_property_add_str(obj, "vendor",
3433 x86_cpuid_get_vendor,
3434 x86_cpuid_set_vendor, NULL);
Andreas Färber938d4c22012-04-17 15:17:27 +02003435 object_property_add_str(obj, "model-id",
Andreas Färber63e886e2012-04-17 23:02:26 +02003436 x86_cpuid_get_model_id,
Andreas Färber938d4c22012-04-17 15:17:27 +02003437 x86_cpuid_set_model_id, NULL);
Andreas Färber89e48962012-04-18 00:12:23 +02003438 object_property_add(obj, "tsc-frequency", "int",
3439 x86_cpuid_get_tsc_freq,
3440 x86_cpuid_set_tsc_freq, NULL, NULL, NULL);
Eduardo Habkost8e8aba52013-05-06 13:20:07 -03003441 object_property_add(obj, "feature-words", "X86CPUFeatureWordInfo",
3442 x86_cpu_get_feature_words,
Eduardo Habkost7e5292b2013-05-06 13:20:09 -03003443 NULL, NULL, (void *)env->features, NULL);
3444 object_property_add(obj, "filtered-features", "X86CPUFeatureWordInfo",
3445 x86_cpu_get_feature_words,
3446 NULL, NULL, (void *)cpu->filtered_features, NULL);
Andreas Färber71ad61d2012-04-17 12:10:29 +02003447
Igor Mammedov92067bf2013-06-05 15:18:40 +02003448 cpu->hyperv_spinlock_attempts = HYPERV_SPINLOCK_NEVER_RETRY;
Igor Mammedovd65e9812012-06-19 15:39:46 +02003449
Eduardo Habkost38e5c112015-03-23 17:29:32 -03003450 for (w = 0; w < FEATURE_WORDS; w++) {
3451 int bitnr;
3452
3453 for (bitnr = 0; bitnr < 32; bitnr++) {
3454 x86_cpu_register_feature_bit_props(cpu, w, bitnr);
3455 }
3456 }
3457
Eduardo Habkostd940ee92014-02-10 08:21:30 -02003458 x86_cpu_load_def(cpu, xcc->cpu_def, &error_abort);
Andreas Färberde024812012-04-03 00:00:17 +02003459}
3460
Igor Mammedov997395d2013-04-23 10:29:41 +02003461static int64_t x86_cpu_get_arch_id(CPUState *cs)
3462{
3463 X86CPU *cpu = X86_CPU(cs);
Igor Mammedov997395d2013-04-23 10:29:41 +02003464
Eduardo Habkost7e72a452014-12-18 23:20:10 -02003465 return cpu->apic_id;
Igor Mammedov997395d2013-04-23 10:29:41 +02003466}
3467
Andreas Färber444d5592013-05-28 13:28:38 +02003468static bool x86_cpu_get_paging_enabled(const CPUState *cs)
3469{
3470 X86CPU *cpu = X86_CPU(cs);
3471
3472 return cpu->env.cr[0] & CR0_PG_MASK;
3473}
3474
Andreas Färberf45748f2013-06-21 19:09:18 +02003475static void x86_cpu_set_pc(CPUState *cs, vaddr value)
3476{
3477 X86CPU *cpu = X86_CPU(cs);
3478
3479 cpu->env.eip = value;
3480}
3481
Andreas Färberbdf7ae52013-06-28 19:31:32 +02003482static void x86_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock *tb)
3483{
3484 X86CPU *cpu = X86_CPU(cs);
3485
3486 cpu->env.eip = tb->pc - tb->cs_base;
3487}
3488
Andreas Färber8c2e1b02013-08-25 18:53:55 +02003489static bool x86_cpu_has_work(CPUState *cs)
3490{
3491 X86CPU *cpu = X86_CPU(cs);
3492 CPUX86State *env = &cpu->env;
3493
Pavel Dovgalyuk6220e902015-09-17 19:23:31 +03003494 return ((cs->interrupt_request & (CPU_INTERRUPT_HARD |
3495 CPU_INTERRUPT_POLL)) &&
Andreas Färber8c2e1b02013-08-25 18:53:55 +02003496 (env->eflags & IF_MASK)) ||
3497 (cs->interrupt_request & (CPU_INTERRUPT_NMI |
3498 CPU_INTERRUPT_INIT |
3499 CPU_INTERRUPT_SIPI |
Paolo Bonzinia9bad652015-05-19 13:46:47 +02003500 CPU_INTERRUPT_MCE)) ||
3501 ((cs->interrupt_request & CPU_INTERRUPT_SMI) &&
3502 !(env->hflags & HF_SMM_MASK));
Andreas Färber8c2e1b02013-08-25 18:53:55 +02003503}
3504
Eduardo Habkost9337e3b2013-07-26 17:09:36 -03003505static Property x86_cpu_properties[] = {
Igor Mammedov2da00e32016-07-06 08:20:41 +02003506#ifdef CONFIG_USER_ONLY
3507 /* apic_id = 0 by default for *-user, see commit 9886e834 */
3508 DEFINE_PROP_UINT32("apic-id", X86CPU, apic_id, 0),
Igor Mammedovd89c2b82016-07-06 08:20:42 +02003509 DEFINE_PROP_INT32("thread-id", X86CPU, thread_id, 0),
3510 DEFINE_PROP_INT32("core-id", X86CPU, core_id, 0),
3511 DEFINE_PROP_INT32("socket-id", X86CPU, socket_id, 0),
Igor Mammedov2da00e32016-07-06 08:20:41 +02003512#else
3513 DEFINE_PROP_UINT32("apic-id", X86CPU, apic_id, UNASSIGNED_APIC_ID),
Igor Mammedovd89c2b82016-07-06 08:20:42 +02003514 DEFINE_PROP_INT32("thread-id", X86CPU, thread_id, -1),
3515 DEFINE_PROP_INT32("core-id", X86CPU, core_id, -1),
3516 DEFINE_PROP_INT32("socket-id", X86CPU, socket_id, -1),
Igor Mammedov2da00e32016-07-06 08:20:41 +02003517#endif
Eduardo Habkost9337e3b2013-07-26 17:09:36 -03003518 DEFINE_PROP_BOOL("pmu", X86CPU, enable_pmu, false),
Igor Mammedovc8f0f882013-06-04 15:05:25 +02003519 { .name = "hv-spinlocks", .info = &qdev_prop_spinlocks },
Igor Mammedov89314502012-08-08 13:52:51 +02003520 DEFINE_PROP_BOOL("hv-relaxed", X86CPU, hyperv_relaxed_timing, false),
Igor Mammedov0f466852012-08-08 13:54:27 +02003521 DEFINE_PROP_BOOL("hv-vapic", X86CPU, hyperv_vapic, false),
Vadim Rozenfeld48a5f3b2014-01-24 00:40:49 +11003522 DEFINE_PROP_BOOL("hv-time", X86CPU, hyperv_time, false),
Andrey Smetaninf2a53c92015-09-09 14:41:30 +02003523 DEFINE_PROP_BOOL("hv-crash", X86CPU, hyperv_crash, false),
Andrey Smetanin744b8a92015-09-16 12:59:42 +03003524 DEFINE_PROP_BOOL("hv-reset", X86CPU, hyperv_reset, false),
Andrey Smetanin8c145d72015-09-16 12:59:43 +03003525 DEFINE_PROP_BOOL("hv-vpindex", X86CPU, hyperv_vpindex, false),
Andrey Smetanin46eb8f92015-09-16 12:59:44 +03003526 DEFINE_PROP_BOOL("hv-runtime", X86CPU, hyperv_runtime, false),
Andrey Smetanin866eea92015-11-11 13:18:38 +03003527 DEFINE_PROP_BOOL("hv-synic", X86CPU, hyperv_synic, false),
Andrey Smetaninff99aa62015-11-25 18:21:25 +03003528 DEFINE_PROP_BOOL("hv-stimer", X86CPU, hyperv_stimer, false),
Eduardo Habkost15e41342015-08-26 13:25:44 -03003529 DEFINE_PROP_BOOL("check", X86CPU, check_cpuid, true),
Igor Mammedov912ffc42013-06-04 15:13:14 +02003530 DEFINE_PROP_BOOL("enforce", X86CPU, enforce_cpuid, false),
Alex Williamsonf522d2a2014-06-02 11:28:50 -06003531 DEFINE_PROP_BOOL("kvm", X86CPU, expose_kvm, true),
Dr. David Alan Gilbertaf459072016-07-08 16:01:36 +01003532 DEFINE_PROP_UINT32("phys-bits", X86CPU, phys_bits, 0),
Dr. David Alan Gilbert11f6fee2016-07-11 20:28:46 +01003533 DEFINE_PROP_BOOL("host-phys-bits", X86CPU, host_phys_bits, false),
Dr. David Alan Gilbertfcc35e72016-07-08 16:01:38 +01003534 DEFINE_PROP_BOOL("fill-mtrr-mask", X86CPU, fill_mtrr_mask, true),
Eduardo Habkostc39c0ed2016-09-21 13:30:12 -03003535 DEFINE_PROP_UINT32("level", X86CPU, env.cpuid_level, UINT32_MAX),
3536 DEFINE_PROP_UINT32("xlevel", X86CPU, env.cpuid_xlevel, UINT32_MAX),
3537 DEFINE_PROP_UINT32("xlevel2", X86CPU, env.cpuid_xlevel2, UINT32_MAX),
3538 DEFINE_PROP_UINT32("min-level", X86CPU, env.cpuid_min_level, 0),
3539 DEFINE_PROP_UINT32("min-xlevel", X86CPU, env.cpuid_min_xlevel, 0),
3540 DEFINE_PROP_UINT32("min-xlevel2", X86CPU, env.cpuid_min_xlevel2, 0),
3541 DEFINE_PROP_BOOL("full-cpuid-auto-level", X86CPU, full_cpuid_auto_level, true),
Alex Williamson1c4a55d2015-10-16 09:38:22 -06003542 DEFINE_PROP_STRING("hv-vendor-id", X86CPU, hyperv_vendor_id),
Radim Krčmář5232d002016-05-12 19:24:26 +02003543 DEFINE_PROP_BOOL("cpuid-0xb", X86CPU, enable_cpuid_0xb, true),
Ashok Raj87f8b622016-06-22 14:56:21 +08003544 DEFINE_PROP_BOOL("lmce", X86CPU, enable_lmce, false),
Longpeng(Mike)14c985c2016-09-07 13:21:13 +08003545 DEFINE_PROP_BOOL("l3-cache", X86CPU, enable_l3_cache, true),
Eduardo Habkost9337e3b2013-07-26 17:09:36 -03003546 DEFINE_PROP_END_OF_LIST()
3547};
3548
Andreas Färber5fd20872012-04-02 23:20:08 +02003549static void x86_cpu_common_class_init(ObjectClass *oc, void *data)
3550{
3551 X86CPUClass *xcc = X86_CPU_CLASS(oc);
3552 CPUClass *cc = CPU_CLASS(oc);
Andreas Färber2b6f2942013-01-16 03:41:47 +01003553 DeviceClass *dc = DEVICE_CLASS(oc);
3554
3555 xcc->parent_realize = dc->realize;
3556 dc->realize = x86_cpu_realizefn;
Igor Mammedovc8847762016-06-24 16:01:02 +02003557 dc->unrealize = x86_cpu_unrealizefn;
Eduardo Habkost9337e3b2013-07-26 17:09:36 -03003558 dc->props = x86_cpu_properties;
Andreas Färber5fd20872012-04-02 23:20:08 +02003559
3560 xcc->parent_reset = cc->reset;
3561 cc->reset = x86_cpu_reset;
Andreas Färber91b1df82013-06-16 07:49:48 +02003562 cc->reset_dump_flags = CPU_DUMP_FPU | CPU_DUMP_CCOP;
Andreas Färberf56e3a12013-02-02 13:38:08 +01003563
Andreas Färber500050d2014-02-10 22:02:44 +01003564 cc->class_by_name = x86_cpu_class_by_name;
Andreas Färber94a444b2014-03-03 23:19:19 +01003565 cc->parse_features = x86_cpu_parse_featurestr;
Andreas Färber8c2e1b02013-08-25 18:53:55 +02003566 cc->has_work = x86_cpu_has_work;
Andreas Färber97a8ea52013-02-02 10:57:51 +01003567 cc->do_interrupt = x86_cpu_do_interrupt;
Richard Henderson42f53fe2014-09-13 09:45:33 -07003568 cc->cpu_exec_interrupt = x86_cpu_exec_interrupt;
Andreas Färber878096e2013-05-27 01:33:50 +02003569 cc->dump_state = x86_cpu_dump_state;
Andreas Färberf45748f2013-06-21 19:09:18 +02003570 cc->set_pc = x86_cpu_set_pc;
Andreas Färberbdf7ae52013-06-28 19:31:32 +02003571 cc->synchronize_from_tb = x86_cpu_synchronize_from_tb;
Andreas Färber5b50e792013-06-29 04:18:45 +02003572 cc->gdb_read_register = x86_cpu_gdb_read_register;
3573 cc->gdb_write_register = x86_cpu_gdb_write_register;
Andreas Färber444d5592013-05-28 13:28:38 +02003574 cc->get_arch_id = x86_cpu_get_arch_id;
3575 cc->get_paging_enabled = x86_cpu_get_paging_enabled;
Andreas Färber75104542013-08-26 03:01:33 +02003576#ifdef CONFIG_USER_ONLY
3577 cc->handle_mmu_fault = x86_cpu_handle_mmu_fault;
3578#else
Andreas Färbera23bbfd2013-05-28 13:52:01 +02003579 cc->get_memory_mapping = x86_cpu_get_memory_mapping;
Andreas Färber00b941e2013-06-29 18:55:54 +02003580 cc->get_phys_page_debug = x86_cpu_get_phys_page_debug;
Jens Freimannc72bf462013-04-19 16:45:06 +02003581 cc->write_elf64_note = x86_cpu_write_elf64_note;
3582 cc->write_elf64_qemunote = x86_cpu_write_elf64_qemunote;
3583 cc->write_elf32_note = x86_cpu_write_elf32_note;
3584 cc->write_elf32_qemunote = x86_cpu_write_elf32_qemunote;
Andreas Färber00b941e2013-06-29 18:55:54 +02003585 cc->vmsd = &vmstate_x86_cpu;
Jens Freimannc72bf462013-04-19 16:45:06 +02003586#endif
Andreas Färbera0e372f2013-06-28 23:18:47 +02003587 cc->gdb_num_core_regs = CPU_NB_REGS * 2 + 25;
Peter Maydell86025ee2014-09-12 14:06:48 +01003588#ifndef CONFIG_USER_ONLY
3589 cc->debug_excp_handler = breakpoint_handler;
3590#endif
Richard Henderson374e0cd2014-09-13 09:45:14 -07003591 cc->cpu_exec_enter = x86_cpu_exec_enter;
3592 cc->cpu_exec_exit = x86_cpu_exec_exit;
Markus Armbruster4c315c22015-10-01 10:59:58 +02003593
Igor Mammedovedd12112016-05-27 13:50:48 +02003594 dc->cannot_instantiate_with_device_add_yet = false;
Markus Armbruster4c315c22015-10-01 10:59:58 +02003595 /*
3596 * Reason: x86_cpu_initfn() calls cpu_exec_init(), which saves the
3597 * object in cpus -> dangling pointer after final object_unref().
3598 */
3599 dc->cannot_destroy_with_object_finalize_yet = true;
Andreas Färber5fd20872012-04-02 23:20:08 +02003600}
3601
3602static const TypeInfo x86_cpu_type_info = {
3603 .name = TYPE_X86_CPU,
3604 .parent = TYPE_CPU,
3605 .instance_size = sizeof(X86CPU),
Andreas Färberde024812012-04-03 00:00:17 +02003606 .instance_init = x86_cpu_initfn,
Eduardo Habkostd940ee92014-02-10 08:21:30 -02003607 .abstract = true,
Andreas Färber5fd20872012-04-02 23:20:08 +02003608 .class_size = sizeof(X86CPUClass),
3609 .class_init = x86_cpu_common_class_init,
3610};
3611
3612static void x86_cpu_register_types(void)
3613{
Eduardo Habkostd940ee92014-02-10 08:21:30 -02003614 int i;
3615
Andreas Färber5fd20872012-04-02 23:20:08 +02003616 type_register_static(&x86_cpu_type_info);
Eduardo Habkostd940ee92014-02-10 08:21:30 -02003617 for (i = 0; i < ARRAY_SIZE(builtin_x86_defs); i++) {
3618 x86_register_cpudef_type(&builtin_x86_defs[i]);
3619 }
3620#ifdef CONFIG_KVM
3621 type_register_static(&host_x86_cpu_type_info);
3622#endif
Andreas Färber5fd20872012-04-02 23:20:08 +02003623}
3624
3625type_init(x86_cpu_register_types)