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" 14cc37d98bSRichard 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 59f5cc5a5cSClaudio Fontana /* 60f5cc5a5cSClaudio Fontana * Print a warning if the user set it to a value that's not the 61f5cc5a5cSClaudio Fontana * host value. 62f5cc5a5cSClaudio Fontana */ 63*7502ffb2SZhao Liu if (phys_bits != host_phys_bits && phys_bits != 0) { 64*7502ffb2SZhao Liu warn_report_once("Host physical bits (%u)" 65f5cc5a5cSClaudio Fontana " does not match phys-bits property (%u)", 66f5cc5a5cSClaudio Fontana host_phys_bits, phys_bits); 67f5cc5a5cSClaudio Fontana } 68f5cc5a5cSClaudio Fontana 69f5cc5a5cSClaudio Fontana if (cpu->host_phys_bits) { 70f5cc5a5cSClaudio Fontana /* The user asked for us to use the host physical bits */ 71f5cc5a5cSClaudio Fontana phys_bits = host_phys_bits; 72f5cc5a5cSClaudio Fontana if (cpu->host_phys_bits_limit && 73f5cc5a5cSClaudio Fontana phys_bits > cpu->host_phys_bits_limit) { 74f5cc5a5cSClaudio Fontana phys_bits = cpu->host_phys_bits_limit; 75f5cc5a5cSClaudio Fontana } 76f5cc5a5cSClaudio Fontana } 77f5cc5a5cSClaudio Fontana 78f5cc5a5cSClaudio Fontana return phys_bits; 79f5cc5a5cSClaudio Fontana } 80f5cc5a5cSClaudio Fontana 819ea057dcSClaudio Fontana bool host_cpu_realizefn(CPUState *cs, Error **errp) 82f5cc5a5cSClaudio Fontana { 83f5cc5a5cSClaudio Fontana X86CPU *cpu = X86_CPU(cs); 84f5cc5a5cSClaudio Fontana CPUX86State *env = &cpu->env; 85f5cc5a5cSClaudio Fontana 86f5cc5a5cSClaudio Fontana if (cpu->max_features && enable_cpu_pm) { 87f5cc5a5cSClaudio Fontana host_cpu_enable_cpu_pm(cpu); 88f5cc5a5cSClaudio Fontana } 89f5cc5a5cSClaudio Fontana if (env->features[FEAT_8000_0001_EDX] & CPUID_EXT2_LM) { 90ce217265SClaudio Fontana uint32_t phys_bits = host_cpu_adjust_phys_bits(cpu); 91ce217265SClaudio Fontana 92ce217265SClaudio Fontana if (phys_bits && 93ce217265SClaudio Fontana (phys_bits > TARGET_PHYS_ADDR_SPACE_BITS || 94ce217265SClaudio Fontana phys_bits < 32)) { 95ce217265SClaudio Fontana error_setg(errp, "phys-bits should be between 32 and %u " 96ce217265SClaudio Fontana " (but is %u)", 97ce217265SClaudio Fontana TARGET_PHYS_ADDR_SPACE_BITS, phys_bits); 989ea057dcSClaudio Fontana return false; 99ce217265SClaudio Fontana } 100ce217265SClaudio Fontana cpu->phys_bits = phys_bits; 101f5cc5a5cSClaudio Fontana } 1029ea057dcSClaudio Fontana return true; 103f5cc5a5cSClaudio Fontana } 104f5cc5a5cSClaudio Fontana 105f5cc5a5cSClaudio Fontana #define CPUID_MODEL_ID_SZ 48 106f5cc5a5cSClaudio Fontana /** 107f5cc5a5cSClaudio Fontana * cpu_x86_fill_model_id: 108f5cc5a5cSClaudio Fontana * Get CPUID model ID string from host CPU. 109f5cc5a5cSClaudio Fontana * 110f5cc5a5cSClaudio Fontana * @str should have at least CPUID_MODEL_ID_SZ bytes 111f5cc5a5cSClaudio Fontana * 112f5cc5a5cSClaudio Fontana * The function does NOT add a null terminator to the string 113f5cc5a5cSClaudio Fontana * automatically. 114f5cc5a5cSClaudio Fontana */ 115f5cc5a5cSClaudio Fontana static int host_cpu_fill_model_id(char *str) 116f5cc5a5cSClaudio Fontana { 117f5cc5a5cSClaudio Fontana uint32_t eax = 0, ebx = 0, ecx = 0, edx = 0; 118f5cc5a5cSClaudio Fontana int i; 119f5cc5a5cSClaudio Fontana 120f5cc5a5cSClaudio Fontana for (i = 0; i < 3; i++) { 121f5cc5a5cSClaudio Fontana host_cpuid(0x80000002 + i, 0, &eax, &ebx, &ecx, &edx); 122f5cc5a5cSClaudio Fontana memcpy(str + i * 16 + 0, &eax, 4); 123f5cc5a5cSClaudio Fontana memcpy(str + i * 16 + 4, &ebx, 4); 124f5cc5a5cSClaudio Fontana memcpy(str + i * 16 + 8, &ecx, 4); 125f5cc5a5cSClaudio Fontana memcpy(str + i * 16 + 12, &edx, 4); 126f5cc5a5cSClaudio Fontana } 127f5cc5a5cSClaudio Fontana return 0; 128f5cc5a5cSClaudio Fontana } 129f5cc5a5cSClaudio Fontana 130f5cc5a5cSClaudio Fontana void host_cpu_vendor_fms(char *vendor, int *family, int *model, int *stepping) 131f5cc5a5cSClaudio Fontana { 132f5cc5a5cSClaudio Fontana uint32_t eax, ebx, ecx, edx; 133f5cc5a5cSClaudio Fontana 134f5cc5a5cSClaudio Fontana host_cpuid(0x0, 0, &eax, &ebx, &ecx, &edx); 135f5cc5a5cSClaudio Fontana x86_cpu_vendor_words2str(vendor, ebx, edx, ecx); 136f5cc5a5cSClaudio Fontana 137f5cc5a5cSClaudio Fontana host_cpuid(0x1, 0, &eax, &ebx, &ecx, &edx); 138f5cc5a5cSClaudio Fontana if (family) { 139f5cc5a5cSClaudio Fontana *family = ((eax >> 8) & 0x0F) + ((eax >> 20) & 0xFF); 140f5cc5a5cSClaudio Fontana } 141f5cc5a5cSClaudio Fontana if (model) { 142f5cc5a5cSClaudio Fontana *model = ((eax >> 4) & 0x0F) | ((eax & 0xF0000) >> 12); 143f5cc5a5cSClaudio Fontana } 144f5cc5a5cSClaudio Fontana if (stepping) { 145f5cc5a5cSClaudio Fontana *stepping = eax & 0x0F; 146f5cc5a5cSClaudio Fontana } 147f5cc5a5cSClaudio Fontana } 148f5cc5a5cSClaudio Fontana 149f5cc5a5cSClaudio Fontana void host_cpu_instance_init(X86CPU *cpu) 150f5cc5a5cSClaudio Fontana { 1515b8978d8SClaudio Fontana X86CPUClass *xcc = X86_CPU_GET_CLASS(cpu); 1525b8978d8SClaudio Fontana 1535b8978d8SClaudio Fontana if (xcc->model) { 154f5cc5a5cSClaudio Fontana uint32_t ebx = 0, ecx = 0, edx = 0; 155f5cc5a5cSClaudio Fontana char vendor[CPUID_VENDOR_SZ + 1]; 156f5cc5a5cSClaudio Fontana 157f5cc5a5cSClaudio Fontana host_cpuid(0, 0, NULL, &ebx, &ecx, &edx); 158f5cc5a5cSClaudio Fontana x86_cpu_vendor_words2str(vendor, ebx, edx, ecx); 159f5cc5a5cSClaudio Fontana object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort); 160f5cc5a5cSClaudio Fontana } 1615b8978d8SClaudio Fontana } 162f5cc5a5cSClaudio Fontana 163f5cc5a5cSClaudio Fontana void host_cpu_max_instance_init(X86CPU *cpu) 164f5cc5a5cSClaudio Fontana { 165f5cc5a5cSClaudio Fontana char vendor[CPUID_VENDOR_SZ + 1] = { 0 }; 166f5cc5a5cSClaudio Fontana char model_id[CPUID_MODEL_ID_SZ + 1] = { 0 }; 167f5cc5a5cSClaudio Fontana int family, model, stepping; 168f5cc5a5cSClaudio Fontana 169f5cc5a5cSClaudio Fontana /* Use max host physical address bits if -cpu max option is applied */ 170f5cc5a5cSClaudio Fontana object_property_set_bool(OBJECT(cpu), "host-phys-bits", true, &error_abort); 171f5cc5a5cSClaudio Fontana 172f5cc5a5cSClaudio Fontana host_cpu_vendor_fms(vendor, &family, &model, &stepping); 173f5cc5a5cSClaudio Fontana host_cpu_fill_model_id(model_id); 174f5cc5a5cSClaudio Fontana 175f5cc5a5cSClaudio Fontana object_property_set_str(OBJECT(cpu), "vendor", vendor, &error_abort); 176f5cc5a5cSClaudio Fontana object_property_set_int(OBJECT(cpu), "family", family, &error_abort); 177f5cc5a5cSClaudio Fontana object_property_set_int(OBJECT(cpu), "model", model, &error_abort); 178f5cc5a5cSClaudio Fontana object_property_set_int(OBJECT(cpu), "stepping", stepping, 179f5cc5a5cSClaudio Fontana &error_abort); 180f5cc5a5cSClaudio Fontana object_property_set_str(OBJECT(cpu), "model-id", model_id, 181f5cc5a5cSClaudio Fontana &error_abort); 182f5cc5a5cSClaudio Fontana } 183f5cc5a5cSClaudio Fontana 184f5cc5a5cSClaudio Fontana static void host_cpu_class_init(ObjectClass *oc, void *data) 185f5cc5a5cSClaudio Fontana { 186f5cc5a5cSClaudio Fontana X86CPUClass *xcc = X86_CPU_CLASS(oc); 187f5cc5a5cSClaudio Fontana 188f5cc5a5cSClaudio Fontana xcc->host_cpuid_required = true; 189f5cc5a5cSClaudio Fontana xcc->ordering = 8; 190f5cc5a5cSClaudio Fontana xcc->model_description = 191f5cc5a5cSClaudio Fontana g_strdup_printf("processor with all supported host features "); 192f5cc5a5cSClaudio Fontana } 193f5cc5a5cSClaudio Fontana 194f5cc5a5cSClaudio Fontana static const TypeInfo host_cpu_type_info = { 195f5cc5a5cSClaudio Fontana .name = X86_CPU_TYPE_NAME("host"), 196f5cc5a5cSClaudio Fontana .parent = X86_CPU_TYPE_NAME("max"), 197f5cc5a5cSClaudio Fontana .class_init = host_cpu_class_init, 198f5cc5a5cSClaudio Fontana }; 199f5cc5a5cSClaudio Fontana 200f5cc5a5cSClaudio Fontana static void host_cpu_type_init(void) 201f5cc5a5cSClaudio Fontana { 202f5cc5a5cSClaudio Fontana type_register_static(&host_cpu_type_info); 203f5cc5a5cSClaudio Fontana } 204f5cc5a5cSClaudio Fontana 205f5cc5a5cSClaudio Fontana type_init(host_cpu_type_init); 206