cachepc-qemu

Fork of AMDESE/qemu with changes for cachepc side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-qemu
Log | Files | Refs | Submodules | LICENSE | sfeed.txt

host-cpu.c (5979B)


      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);