qemu/target/i386/host-cpu.c
<<
>>
Prefs
   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 "sysemu/sysemu.h"
  15
  16/* Note: Only safe for use on x86(-64) hosts */
  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         * Note: According to AMD doc 25481 rev 2.34 they have a field
  27         * at 23:16 that can specify a maximum physical address bits for
  28         * the guest that can override this value; but I've not seen
  29         * anything with that set.
  30         */
  31        host_phys_bits = eax & 0xff;
  32    } else {
  33        /*
  34         * It's an odd 64 bit machine that doesn't have the leaf for
  35         * physical address bits; fall back to 36 that's most older
  36         * Intel.
  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     * Print a warning if the user set it to a value that's not the
  61     * host value.
  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        /* The user asked for us to use the host physical bits */
  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 * cpu_x86_fill_model_id:
 110 * Get CPUID model ID string from host CPU.
 111 *
 112 * @str should have at least CPUID_MODEL_ID_SZ bytes
 113 *
 114 * The function does NOT add a null terminator to the string
 115 * automatically.
 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    /* Use max host physical address bits if -cpu max option is applied */
 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