xref: /qemu/target/i386/host-cpu.c (revision f5cc5a5c168674f84bf061cdb307c2d25fba5448)
1*f5cc5a5cSClaudio Fontana /*
2*f5cc5a5cSClaudio Fontana  * x86 host CPU functions, and "host" cpu type initialization
3*f5cc5a5cSClaudio Fontana  *
4*f5cc5a5cSClaudio Fontana  * Copyright 2021 SUSE LLC
5*f5cc5a5cSClaudio Fontana  *
6*f5cc5a5cSClaudio Fontana  * This work is licensed under the terms of the GNU GPL, version 2 or later.
7*f5cc5a5cSClaudio Fontana  * See the COPYING file in the top-level directory.
8*f5cc5a5cSClaudio Fontana  */
9*f5cc5a5cSClaudio Fontana 
10*f5cc5a5cSClaudio Fontana #include "qemu/osdep.h"
11*f5cc5a5cSClaudio Fontana #include "cpu.h"
12*f5cc5a5cSClaudio Fontana #include "host-cpu.h"
13*f5cc5a5cSClaudio Fontana #include "qapi/error.h"
14*f5cc5a5cSClaudio Fontana #include "sysemu/sysemu.h"
15*f5cc5a5cSClaudio Fontana 
16*f5cc5a5cSClaudio Fontana /* Note: Only safe for use on x86(-64) hosts */
17*f5cc5a5cSClaudio Fontana static uint32_t host_cpu_phys_bits(void)
18*f5cc5a5cSClaudio Fontana {
19*f5cc5a5cSClaudio Fontana     uint32_t eax;
20*f5cc5a5cSClaudio Fontana     uint32_t host_phys_bits;
21*f5cc5a5cSClaudio Fontana 
22*f5cc5a5cSClaudio Fontana     host_cpuid(0x80000000, 0, &eax, NULL, NULL, NULL);
23*f5cc5a5cSClaudio Fontana     if (eax >= 0x80000008) {
24*f5cc5a5cSClaudio Fontana         host_cpuid(0x80000008, 0, &eax, NULL, NULL, NULL);
25*f5cc5a5cSClaudio Fontana         /*
26*f5cc5a5cSClaudio Fontana          * Note: According to AMD doc 25481 rev 2.34 they have a field
27*f5cc5a5cSClaudio Fontana          * at 23:16 that can specify a maximum physical address bits for
28*f5cc5a5cSClaudio Fontana          * the guest that can override this value; but I've not seen
29*f5cc5a5cSClaudio Fontana          * anything with that set.
30*f5cc5a5cSClaudio Fontana          */
31*f5cc5a5cSClaudio Fontana         host_phys_bits = eax & 0xff;
32*f5cc5a5cSClaudio Fontana     } else {
33*f5cc5a5cSClaudio Fontana         /*
34*f5cc5a5cSClaudio Fontana          * It's an odd 64 bit machine that doesn't have the leaf for
35*f5cc5a5cSClaudio Fontana          * physical address bits; fall back to 36 that's most older
36*f5cc5a5cSClaudio Fontana          * Intel.
37*f5cc5a5cSClaudio Fontana          */
38*f5cc5a5cSClaudio Fontana         host_phys_bits = 36;
39*f5cc5a5cSClaudio Fontana     }
40*f5cc5a5cSClaudio Fontana 
41*f5cc5a5cSClaudio Fontana     return host_phys_bits;
42*f5cc5a5cSClaudio Fontana }
43*f5cc5a5cSClaudio Fontana 
44*f5cc5a5cSClaudio Fontana static void host_cpu_enable_cpu_pm(X86CPU *cpu)
45*f5cc5a5cSClaudio Fontana {
46*f5cc5a5cSClaudio Fontana     CPUX86State *env = &cpu->env;
47*f5cc5a5cSClaudio Fontana 
48*f5cc5a5cSClaudio Fontana     host_cpuid(5, 0, &cpu->mwait.eax, &cpu->mwait.ebx,
49*f5cc5a5cSClaudio Fontana                &cpu->mwait.ecx, &cpu->mwait.edx);
50*f5cc5a5cSClaudio Fontana     env->features[FEAT_1_ECX] |= CPUID_EXT_MONITOR;
51*f5cc5a5cSClaudio Fontana }
52*f5cc5a5cSClaudio Fontana 
53*f5cc5a5cSClaudio Fontana static uint32_t host_cpu_adjust_phys_bits(X86CPU *cpu, Error **errp)
54*f5cc5a5cSClaudio Fontana {
55*f5cc5a5cSClaudio Fontana     uint32_t host_phys_bits = host_cpu_phys_bits();
56*f5cc5a5cSClaudio Fontana     uint32_t phys_bits = cpu->phys_bits;
57*f5cc5a5cSClaudio Fontana     static bool warned;
58*f5cc5a5cSClaudio Fontana 
59*f5cc5a5cSClaudio Fontana     /*
60*f5cc5a5cSClaudio Fontana      * Print a warning if the user set it to a value that's not the
61*f5cc5a5cSClaudio Fontana      * host value.
62*f5cc5a5cSClaudio Fontana      */
63*f5cc5a5cSClaudio Fontana     if (phys_bits != host_phys_bits && phys_bits != 0 &&
64*f5cc5a5cSClaudio Fontana         !warned) {
65*f5cc5a5cSClaudio Fontana         warn_report("Host physical bits (%u)"
66*f5cc5a5cSClaudio Fontana                     " does not match phys-bits property (%u)",
67*f5cc5a5cSClaudio Fontana                     host_phys_bits, phys_bits);
68*f5cc5a5cSClaudio Fontana         warned = true;
69*f5cc5a5cSClaudio Fontana     }
70*f5cc5a5cSClaudio Fontana 
71*f5cc5a5cSClaudio Fontana     if (cpu->host_phys_bits) {
72*f5cc5a5cSClaudio Fontana         /* The user asked for us to use the host physical bits */
73*f5cc5a5cSClaudio Fontana         phys_bits = host_phys_bits;
74*f5cc5a5cSClaudio Fontana         if (cpu->host_phys_bits_limit &&
75*f5cc5a5cSClaudio Fontana             phys_bits > cpu->host_phys_bits_limit) {
76*f5cc5a5cSClaudio Fontana             phys_bits = cpu->host_phys_bits_limit;
77*f5cc5a5cSClaudio Fontana         }
78*f5cc5a5cSClaudio Fontana     }
79*f5cc5a5cSClaudio Fontana 
80*f5cc5a5cSClaudio Fontana     if (phys_bits &&
81*f5cc5a5cSClaudio Fontana         (phys_bits > TARGET_PHYS_ADDR_SPACE_BITS ||
82*f5cc5a5cSClaudio Fontana          phys_bits < 32)) {
83*f5cc5a5cSClaudio Fontana         error_setg(errp, "phys-bits should be between 32 and %u "
84*f5cc5a5cSClaudio Fontana                    " (but is %u)",
85*f5cc5a5cSClaudio Fontana                    TARGET_PHYS_ADDR_SPACE_BITS, phys_bits);
86*f5cc5a5cSClaudio Fontana     }
87*f5cc5a5cSClaudio Fontana 
88*f5cc5a5cSClaudio Fontana     return phys_bits;
89*f5cc5a5cSClaudio Fontana }
90*f5cc5a5cSClaudio Fontana 
91*f5cc5a5cSClaudio Fontana void host_cpu_realizefn(CPUState *cs, Error **errp)
92*f5cc5a5cSClaudio Fontana {
93*f5cc5a5cSClaudio Fontana     X86CPU *cpu = X86_CPU(cs);
94*f5cc5a5cSClaudio Fontana     CPUX86State *env = &cpu->env;
95*f5cc5a5cSClaudio Fontana 
96*f5cc5a5cSClaudio Fontana     if (cpu->max_features && enable_cpu_pm) {
97*f5cc5a5cSClaudio Fontana         host_cpu_enable_cpu_pm(cpu);
98*f5cc5a5cSClaudio Fontana     }
99*f5cc5a5cSClaudio Fontana     if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) {
100*f5cc5a5cSClaudio Fontana         cpu->phys_bits = host_cpu_adjust_phys_bits(cpu, errp);
101*f5cc5a5cSClaudio Fontana     }
102*f5cc5a5cSClaudio Fontana }
103*f5cc5a5cSClaudio Fontana 
104*f5cc5a5cSClaudio Fontana #define CPUID_MODEL_ID_SZ 48
105*f5cc5a5cSClaudio Fontana /**
106*f5cc5a5cSClaudio Fontana  * cpu_x86_fill_model_id:
107*f5cc5a5cSClaudio Fontana  * Get CPUID model ID string from host CPU.
108*f5cc5a5cSClaudio Fontana  *
109*f5cc5a5cSClaudio Fontana  * @str should have at least CPUID_MODEL_ID_SZ bytes
110*f5cc5a5cSClaudio Fontana  *
111*f5cc5a5cSClaudio Fontana  * The function does NOT add a null terminator to the string
112*f5cc5a5cSClaudio Fontana  * automatically.
113*f5cc5a5cSClaudio Fontana  */
114*f5cc5a5cSClaudio Fontana static int host_cpu_fill_model_id(char *str)
115*f5cc5a5cSClaudio Fontana {
116*f5cc5a5cSClaudio Fontana     uint32_t eax = 0, ebx = 0, ecx = 0, edx = 0;
117*f5cc5a5cSClaudio Fontana     int i;
118*f5cc5a5cSClaudio Fontana 
119*f5cc5a5cSClaudio Fontana     for (i = 0; i < 3; i++) {
120*f5cc5a5cSClaudio Fontana         host_cpuid(0x80000002 + i, 0, &eax, &ebx, &ecx, &edx);
121*f5cc5a5cSClaudio Fontana         memcpy(str + i * 16 +  0, &eax, 4);
122*f5cc5a5cSClaudio Fontana         memcpy(str + i * 16 +  4, &ebx, 4);
123*f5cc5a5cSClaudio Fontana         memcpy(str + i * 16 +  8, &ecx, 4);
124*f5cc5a5cSClaudio Fontana         memcpy(str + i * 16 + 12, &edx, 4);
125*f5cc5a5cSClaudio Fontana     }
126*f5cc5a5cSClaudio Fontana     return 0;
127*f5cc5a5cSClaudio Fontana }
128*f5cc5a5cSClaudio Fontana 
129*f5cc5a5cSClaudio Fontana void host_cpu_vendor_fms(char *vendor, int *family, int *model, int *stepping)
130*f5cc5a5cSClaudio Fontana {
131*f5cc5a5cSClaudio Fontana     uint32_t eax, ebx, ecx, edx;
132*f5cc5a5cSClaudio Fontana 
133*f5cc5a5cSClaudio Fontana     host_cpuid(0x0, 0, &eax, &ebx, &ecx, &edx);
134*f5cc5a5cSClaudio Fontana     x86_cpu_vendor_words2str(vendor, ebx, edx, ecx);
135*f5cc5a5cSClaudio Fontana 
136*f5cc5a5cSClaudio Fontana     host_cpuid(0x1, 0, &eax, &ebx, &ecx, &edx);
137*f5cc5a5cSClaudio Fontana     if (family) {
138*f5cc5a5cSClaudio Fontana         *family = ((eax >> 8) & 0x0F) + ((eax >> 20) & 0xFF);
139*f5cc5a5cSClaudio Fontana     }
140*f5cc5a5cSClaudio Fontana     if (model) {
141*f5cc5a5cSClaudio Fontana         *model = ((eax >> 4) & 0x0F) | ((eax & 0xF0000) >> 12);
142*f5cc5a5cSClaudio Fontana     }
143*f5cc5a5cSClaudio Fontana     if (stepping) {
144*f5cc5a5cSClaudio Fontana         *stepping = eax & 0x0F;
145*f5cc5a5cSClaudio Fontana     }
146*f5cc5a5cSClaudio Fontana }
147*f5cc5a5cSClaudio Fontana 
148*f5cc5a5cSClaudio Fontana void host_cpu_instance_init(X86CPU *cpu)
149*f5cc5a5cSClaudio Fontana {
150*f5cc5a5cSClaudio Fontana     uint32_t ebx = 0, ecx = 0, edx = 0;
151*f5cc5a5cSClaudio Fontana     char vendor[CPUID_VENDOR_SZ + 1];
152*f5cc5a5cSClaudio Fontana 
153*f5cc5a5cSClaudio Fontana     host_cpuid(0, 0, NULL, &ebx, &ecx, &edx);
154*f5cc5a5cSClaudio Fontana     x86_cpu_vendor_words2str(vendor, ebx, edx, ecx);
155*f5cc5a5cSClaudio Fontana 
156*f5cc5a5cSClaudio Fontana     object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort);
157*f5cc5a5cSClaudio Fontana }
158*f5cc5a5cSClaudio Fontana 
159*f5cc5a5cSClaudio Fontana void host_cpu_max_instance_init(X86CPU *cpu)
160*f5cc5a5cSClaudio Fontana {
161*f5cc5a5cSClaudio Fontana     char vendor[CPUID_VENDOR_SZ + 1] = { 0 };
162*f5cc5a5cSClaudio Fontana     char model_id[CPUID_MODEL_ID_SZ + 1] = { 0 };
163*f5cc5a5cSClaudio Fontana     int family, model, stepping;
164*f5cc5a5cSClaudio Fontana 
165*f5cc5a5cSClaudio Fontana     /* Use max host physical address bits if -cpu max option is applied */
166*f5cc5a5cSClaudio Fontana     object_property_set_bool(OBJECT(cpu), "host-phys-bits", true, &error_abort);
167*f5cc5a5cSClaudio Fontana 
168*f5cc5a5cSClaudio Fontana     host_cpu_vendor_fms(vendor, &family, &model, &stepping);
169*f5cc5a5cSClaudio Fontana     host_cpu_fill_model_id(model_id);
170*f5cc5a5cSClaudio Fontana 
171*f5cc5a5cSClaudio Fontana     object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort);
172*f5cc5a5cSClaudio Fontana     object_property_set_int(OBJECT(cpu), "family", family, &error_abort);
173*f5cc5a5cSClaudio Fontana     object_property_set_int(OBJECT(cpu), "model", model, &error_abort);
174*f5cc5a5cSClaudio Fontana     object_property_set_int(OBJECT(cpu), "stepping", stepping,
175*f5cc5a5cSClaudio Fontana                             &error_abort);
176*f5cc5a5cSClaudio Fontana     object_property_set_str(OBJECT(cpu), "model-id", model_id,
177*f5cc5a5cSClaudio Fontana                             &error_abort);
178*f5cc5a5cSClaudio Fontana }
179*f5cc5a5cSClaudio Fontana 
180*f5cc5a5cSClaudio Fontana static void host_cpu_class_init(ObjectClass *oc, void *data)
181*f5cc5a5cSClaudio Fontana {
182*f5cc5a5cSClaudio Fontana     X86CPUClass *xcc = X86_CPU_CLASS(oc);
183*f5cc5a5cSClaudio Fontana 
184*f5cc5a5cSClaudio Fontana     xcc->host_cpuid_required = true;
185*f5cc5a5cSClaudio Fontana     xcc->ordering = 8;
186*f5cc5a5cSClaudio Fontana     xcc->model_description =
187*f5cc5a5cSClaudio Fontana         g_strdup_printf("processor with all supported host features ");
188*f5cc5a5cSClaudio Fontana }
189*f5cc5a5cSClaudio Fontana 
190*f5cc5a5cSClaudio Fontana static const TypeInfo host_cpu_type_info = {
191*f5cc5a5cSClaudio Fontana     .name = X86_CPU_TYPE_NAME("host"),
192*f5cc5a5cSClaudio Fontana     .parent = X86_CPU_TYPE_NAME("max"),
193*f5cc5a5cSClaudio Fontana     .class_init = host_cpu_class_init,
194*f5cc5a5cSClaudio Fontana };
195*f5cc5a5cSClaudio Fontana 
196*f5cc5a5cSClaudio Fontana static void host_cpu_type_init(void)
197*f5cc5a5cSClaudio Fontana {
198*f5cc5a5cSClaudio Fontana     type_register_static(&host_cpu_type_info);
199*f5cc5a5cSClaudio Fontana }
200*f5cc5a5cSClaudio Fontana 
201*f5cc5a5cSClaudio Fontana type_init(host_cpu_type_init);
202