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