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