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