1
2
3
4
5
6
7
8
9
10#include "qemu/osdep.h"
11#include "cpu.h"
12#include "host-cpu.h"
13#include "qapi/error.h"
14#include "sysemu/sysemu.h"
15
16
17static uint32_t host_cpu_phys_bits(void)
18{
19 uint32_t eax;
20 uint32_t host_phys_bits;
21
22 host_cpuid(0x80000000, 0, &eax, NULL, NULL, NULL);
23 if (eax >= 0x80000008) {
24 host_cpuid(0x80000008, 0, &eax, NULL, NULL, NULL);
25
26
27
28
29
30
31 host_phys_bits = eax & 0xff;
32 } else {
33
34
35
36
37
38 host_phys_bits = 36;
39 }
40
41 return host_phys_bits;
42}
43
44static void host_cpu_enable_cpu_pm(X86CPU *cpu)
45{
46 CPUX86State *env = &cpu->env;
47
48 host_cpuid(5, 0, &cpu->mwait.eax, &cpu->mwait.ebx,
49 &cpu->mwait.ecx, &cpu->mwait.edx);
50 env->features[FEAT_1_ECX] |= CPUID_EXT_MONITOR;
51}
52
53static uint32_t host_cpu_adjust_phys_bits(X86CPU *cpu)
54{
55 uint32_t host_phys_bits = host_cpu_phys_bits();
56 uint32_t phys_bits = cpu->phys_bits;
57 static bool warned;
58
59
60
61
62
63 if (phys_bits != host_phys_bits && phys_bits != 0 &&
64 !warned) {
65 warn_report("Host physical bits (%u)"
66 " does not match phys-bits property (%u)",
67 host_phys_bits, phys_bits);
68 warned = true;
69 }
70
71 if (cpu->host_phys_bits) {
72
73 phys_bits = host_phys_bits;
74 if (cpu->host_phys_bits_limit &&
75 phys_bits > cpu->host_phys_bits_limit) {
76 phys_bits = cpu->host_phys_bits_limit;
77 }
78 }
79
80 return phys_bits;
81}
82
83bool host_cpu_realizefn(CPUState *cs, Error **errp)
84{
85 X86CPU *cpu = X86_CPU(cs);
86 CPUX86State *env = &cpu->env;
87
88 if (cpu->max_features && enable_cpu_pm) {
89 host_cpu_enable_cpu_pm(cpu);
90 }
91 if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) {
92 uint32_t phys_bits = host_cpu_adjust_phys_bits(cpu);
93
94 if (phys_bits &&
95 (phys_bits > TARGET_PHYS_ADDR_SPACE_BITS ||
96 phys_bits < 32)) {
97 error_setg(errp, "phys-bits should be between 32 and %u "
98 " (but is %u)",
99 TARGET_PHYS_ADDR_SPACE_BITS, phys_bits);
100 return false;
101 }
102 cpu->phys_bits = phys_bits;
103 }
104 return true;
105}
106
107#define CPUID_MODEL_ID_SZ 48
108
109
110
111
112
113
114
115
116
117static int host_cpu_fill_model_id(char *str)
118{
119 uint32_t eax = 0, ebx = 0, ecx = 0, edx = 0;
120 int i;
121
122 for (i = 0; i < 3; i++) {
123 host_cpuid(0x80000002 + i, 0, &eax, &ebx, &ecx, &edx);
124 memcpy(str + i * 16 + 0, &eax, 4);
125 memcpy(str + i * 16 + 4, &ebx, 4);
126 memcpy(str + i * 16 + 8, &ecx, 4);
127 memcpy(str + i * 16 + 12, &edx, 4);
128 }
129 return 0;
130}
131
132void host_cpu_vendor_fms(char *vendor, int *family, int *model, int *stepping)
133{
134 uint32_t eax, ebx, ecx, edx;
135
136 host_cpuid(0x0, 0, &eax, &ebx, &ecx, &edx);
137 x86_cpu_vendor_words2str(vendor, ebx, edx, ecx);
138
139 host_cpuid(0x1, 0, &eax, &ebx, &ecx, &edx);
140 if (family) {
141 *family = ((eax >> 8) & 0x0F) + ((eax >> 20) & 0xFF);
142 }
143 if (model) {
144 *model = ((eax >> 4) & 0x0F) | ((eax & 0xF0000) >> 12);
145 }
146 if (stepping) {
147 *stepping = eax & 0x0F;
148 }
149}
150
151void host_cpu_instance_init(X86CPU *cpu)
152{
153 X86CPUClass *xcc = X86_CPU_GET_CLASS(cpu);
154
155 if (xcc->model) {
156 uint32_t ebx = 0, ecx = 0, edx = 0;
157 char vendor[CPUID_VENDOR_SZ + 1];
158
159 host_cpuid(0, 0, NULL, &ebx, &ecx, &edx);
160 x86_cpu_vendor_words2str(vendor, ebx, edx, ecx);
161 object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort);
162 }
163}
164
165void host_cpu_max_instance_init(X86CPU *cpu)
166{
167 char vendor[CPUID_VENDOR_SZ + 1] = { 0 };
168 char model_id[CPUID_MODEL_ID_SZ + 1] = { 0 };
169 int family, model, stepping;
170
171
172 object_property_set_bool(OBJECT(cpu), "host-phys-bits", true, &error_abort);
173
174 host_cpu_vendor_fms(vendor, &family, &model, &stepping);
175 host_cpu_fill_model_id(model_id);
176
177 object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort);
178 object_property_set_int(OBJECT(cpu), "family", family, &error_abort);
179 object_property_set_int(OBJECT(cpu), "model", model, &error_abort);
180 object_property_set_int(OBJECT(cpu), "stepping", stepping,
181 &error_abort);
182 object_property_set_str(OBJECT(cpu), "model-id", model_id,
183 &error_abort);
184}
185
186static void host_cpu_class_init(ObjectClass *oc, void *data)
187{
188 X86CPUClass *xcc = X86_CPU_CLASS(oc);
189
190 xcc->host_cpuid_required = true;
191 xcc->ordering = 8;
192 xcc->model_description =
193 g_strdup_printf("processor with all supported host features ");
194}
195
196static const TypeInfo host_cpu_type_info = {
197 .name = X86_CPU_TYPE_NAME("host"),
198 .parent = X86_CPU_TYPE_NAME("max"),
199 .class_init = host_cpu_class_init,
200};
201
202static void host_cpu_type_init(void)
203{
204 type_register_static(&host_cpu_type_info);
205}
206
207type_init(host_cpu_type_init);
208