Path: blob/master/tools/testing/selftests/kvm/x86/amx_test.c
52540 views
// SPDX-License-Identifier: GPL-2.0-only1/*2* amx tests3*4* Copyright (C) 2021, Intel, Inc.5*6* Tests for amx #NM exception and save/restore.7*/8#include <fcntl.h>9#include <stdio.h>10#include <stdlib.h>11#include <string.h>12#include <sys/ioctl.h>13#include <sys/syscall.h>1415#include "test_util.h"1617#include "kvm_util.h"18#include "processor.h"19#include "vmx.h"2021#ifndef __x86_64__22# error This test is 64-bit only23#endif2425#define NUM_TILES 826#define TILE_SIZE 102427#define XSAVE_SIZE ((NUM_TILES * TILE_SIZE) + PAGE_SIZE)2829/* Tile configuration associated: */30#define PALETTE_TABLE_INDEX 131#define MAX_TILES 1632#define RESERVED_BYTES 143334#define XSAVE_HDR_OFFSET 5123536struct tile_config {37u8 palette_id;38u8 start_row;39u8 reserved[RESERVED_BYTES];40u16 colsb[MAX_TILES];41u8 rows[MAX_TILES];42};4344struct tile_data {45u8 data[NUM_TILES * TILE_SIZE];46};4748struct xtile_info {49u16 bytes_per_tile;50u16 bytes_per_row;51u16 max_names;52u16 max_rows;53u32 xsave_offset;54u32 xsave_size;55};5657static struct xtile_info xtile;5859static inline void __ldtilecfg(void *cfg)60{61asm volatile(".byte 0xc4,0xe2,0x78,0x49,0x00"62: : "a"(cfg));63}6465static inline void __tileloadd(void *tile)66{67asm volatile(".byte 0xc4,0xe2,0x7b,0x4b,0x04,0x10"68: : "a"(tile), "d"(0));69}7071static inline int tileloadd_safe(void *tile)72{73return kvm_asm_safe(".byte 0xc4,0xe2,0x7b,0x4b,0x04,0x10",74"a"(tile), "d"(0));75}7677static inline void __tilerelease(void)78{79asm volatile(".byte 0xc4, 0xe2, 0x78, 0x49, 0xc0" ::);80}8182static inline void __xsavec(struct xstate *xstate, uint64_t rfbm)83{84uint32_t rfbm_lo = rfbm;85uint32_t rfbm_hi = rfbm >> 32;8687asm volatile("xsavec (%%rdi)"88: : "D" (xstate), "a" (rfbm_lo), "d" (rfbm_hi)89: "memory");90}9192static void check_xtile_info(void)93{94GUEST_ASSERT((xgetbv(0) & XFEATURE_MASK_XTILE) == XFEATURE_MASK_XTILE);9596GUEST_ASSERT(this_cpu_has_p(X86_PROPERTY_XSTATE_MAX_SIZE_XCR0));97GUEST_ASSERT(this_cpu_property(X86_PROPERTY_XSTATE_MAX_SIZE_XCR0) <= XSAVE_SIZE);9899xtile.xsave_offset = this_cpu_property(X86_PROPERTY_XSTATE_TILE_OFFSET);100GUEST_ASSERT(xtile.xsave_offset == 2816);101xtile.xsave_size = this_cpu_property(X86_PROPERTY_XSTATE_TILE_SIZE);102GUEST_ASSERT(xtile.xsave_size == 8192);103GUEST_ASSERT(sizeof(struct tile_data) >= xtile.xsave_size);104105GUEST_ASSERT(this_cpu_has_p(X86_PROPERTY_AMX_MAX_PALETTE_TABLES));106GUEST_ASSERT(this_cpu_property(X86_PROPERTY_AMX_MAX_PALETTE_TABLES) >=107PALETTE_TABLE_INDEX);108109GUEST_ASSERT(this_cpu_has_p(X86_PROPERTY_AMX_NR_TILE_REGS));110xtile.max_names = this_cpu_property(X86_PROPERTY_AMX_NR_TILE_REGS);111GUEST_ASSERT(xtile.max_names == 8);112xtile.bytes_per_tile = this_cpu_property(X86_PROPERTY_AMX_BYTES_PER_TILE);113GUEST_ASSERT(xtile.bytes_per_tile == 1024);114xtile.bytes_per_row = this_cpu_property(X86_PROPERTY_AMX_BYTES_PER_ROW);115GUEST_ASSERT(xtile.bytes_per_row == 64);116xtile.max_rows = this_cpu_property(X86_PROPERTY_AMX_MAX_ROWS);117GUEST_ASSERT(xtile.max_rows == 16);118}119120static void set_tilecfg(struct tile_config *cfg)121{122int i;123124/* Only palette id 1 */125cfg->palette_id = 1;126for (i = 0; i < xtile.max_names; i++) {127cfg->colsb[i] = xtile.bytes_per_row;128cfg->rows[i] = xtile.max_rows;129}130}131132enum {133/* Retrieve TMM0 from guest, stash it for TEST_RESTORE_TILEDATA */134TEST_SAVE_TILEDATA = 1,135136/* Check TMM0 against tiledata */137TEST_COMPARE_TILEDATA = 2,138139/* Restore TMM0 from earlier save */140TEST_RESTORE_TILEDATA = 4,141142/* Full VM save/restore */143TEST_SAVE_RESTORE = 8,144};145146static void __attribute__((__flatten__)) guest_code(struct tile_config *amx_cfg,147struct tile_data *tiledata,148struct xstate *xstate)149{150int vector;151152GUEST_ASSERT(this_cpu_has(X86_FEATURE_XSAVE) &&153this_cpu_has(X86_FEATURE_OSXSAVE));154check_xtile_info();155GUEST_SYNC(TEST_SAVE_RESTORE);156157/* xfd=0, enable amx */158wrmsr(MSR_IA32_XFD, 0);159GUEST_SYNC(TEST_SAVE_RESTORE);160GUEST_ASSERT(rdmsr(MSR_IA32_XFD) == 0);161set_tilecfg(amx_cfg);162__ldtilecfg(amx_cfg);163GUEST_SYNC(TEST_SAVE_RESTORE);164/* Check save/restore when trap to userspace */165__tileloadd(tiledata);166GUEST_SYNC(TEST_SAVE_TILEDATA | TEST_COMPARE_TILEDATA | TEST_SAVE_RESTORE);167168/* xfd=0x40000, disable amx tiledata */169wrmsr(MSR_IA32_XFD, XFEATURE_MASK_XTILE_DATA);170171/* host tries setting tiledata while guest XFD is set */172GUEST_SYNC(TEST_RESTORE_TILEDATA);173GUEST_SYNC(TEST_SAVE_RESTORE);174175wrmsr(MSR_IA32_XFD, 0);176__tilerelease();177GUEST_SYNC(TEST_SAVE_RESTORE);178/*179* After XSAVEC, XTILEDATA is cleared in the xstate_bv but is set in180* the xcomp_bv.181*/182xstate->header.xstate_bv = XFEATURE_MASK_XTILE_DATA;183__xsavec(xstate, XFEATURE_MASK_XTILE_DATA);184GUEST_ASSERT(!(xstate->header.xstate_bv & XFEATURE_MASK_XTILE_DATA));185GUEST_ASSERT(xstate->header.xcomp_bv & XFEATURE_MASK_XTILE_DATA);186187/* #NM test */188189/* xfd=0x40000, disable amx tiledata */190wrmsr(MSR_IA32_XFD, XFEATURE_MASK_XTILE_DATA);191192/*193* XTILEDATA is cleared in xstate_bv but set in xcomp_bv, this property194* remains the same even when amx tiledata is disabled by IA32_XFD.195*/196xstate->header.xstate_bv = XFEATURE_MASK_XTILE_DATA;197__xsavec(xstate, XFEATURE_MASK_XTILE_DATA);198GUEST_ASSERT(!(xstate->header.xstate_bv & XFEATURE_MASK_XTILE_DATA));199GUEST_ASSERT((xstate->header.xcomp_bv & XFEATURE_MASK_XTILE_DATA));200201GUEST_SYNC(TEST_SAVE_RESTORE);202GUEST_ASSERT(rdmsr(MSR_IA32_XFD) == XFEATURE_MASK_XTILE_DATA);203set_tilecfg(amx_cfg);204__ldtilecfg(amx_cfg);205206/* Trigger #NM exception */207vector = tileloadd_safe(tiledata);208__GUEST_ASSERT(vector == NM_VECTOR,209"Wanted #NM on tileloadd with XFD[18]=1, got %s",210ex_str(vector));211212GUEST_ASSERT(!(get_cr0() & X86_CR0_TS));213GUEST_ASSERT(rdmsr(MSR_IA32_XFD_ERR) == XFEATURE_MASK_XTILE_DATA);214GUEST_ASSERT(rdmsr(MSR_IA32_XFD) == XFEATURE_MASK_XTILE_DATA);215GUEST_SYNC(TEST_SAVE_RESTORE);216GUEST_ASSERT(rdmsr(MSR_IA32_XFD_ERR) == XFEATURE_MASK_XTILE_DATA);217GUEST_ASSERT(rdmsr(MSR_IA32_XFD) == XFEATURE_MASK_XTILE_DATA);218/* Clear xfd_err */219wrmsr(MSR_IA32_XFD_ERR, 0);220/* xfd=0, enable amx */221wrmsr(MSR_IA32_XFD, 0);222GUEST_SYNC(TEST_SAVE_RESTORE);223224__tileloadd(tiledata);225GUEST_SYNC(TEST_COMPARE_TILEDATA | TEST_SAVE_RESTORE);226227GUEST_DONE();228}229230int main(int argc, char *argv[])231{232struct kvm_regs regs1, regs2;233struct kvm_vcpu *vcpu;234struct kvm_vm *vm;235struct kvm_x86_state *state;236struct kvm_x86_state *tile_state = NULL;237int xsave_restore_size;238vm_vaddr_t amx_cfg, tiledata, xstate;239struct ucall uc;240int ret;241242/*243* Note, all off-by-default features must be enabled before anything244* caches KVM_GET_SUPPORTED_CPUID, e.g. before using kvm_cpu_has().245*/246vm_xsave_require_permission(XFEATURE_MASK_XTILE_DATA);247248TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XFD));249TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XSAVE));250TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_AMX_TILE));251TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XTILECFG));252TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XTILEDATA));253TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_XTILEDATA_XFD));254255/* Create VM */256vm = vm_create_with_one_vcpu(&vcpu, guest_code);257258TEST_ASSERT(kvm_cpu_has_p(X86_PROPERTY_XSTATE_MAX_SIZE),259"KVM should enumerate max XSAVE size when XSAVE is supported");260xsave_restore_size = kvm_cpu_property(X86_PROPERTY_XSTATE_MAX_SIZE);261262vcpu_regs_get(vcpu, ®s1);263264/* amx cfg for guest_code */265amx_cfg = vm_vaddr_alloc_page(vm);266memset(addr_gva2hva(vm, amx_cfg), 0x0, getpagesize());267268/* amx tiledata for guest_code */269tiledata = vm_vaddr_alloc_pages(vm, 2);270memset(addr_gva2hva(vm, tiledata), rand() | 1, 2 * getpagesize());271272/* XSAVE state for guest_code */273xstate = vm_vaddr_alloc_pages(vm, DIV_ROUND_UP(XSAVE_SIZE, PAGE_SIZE));274memset(addr_gva2hva(vm, xstate), 0, PAGE_SIZE * DIV_ROUND_UP(XSAVE_SIZE, PAGE_SIZE));275vcpu_args_set(vcpu, 3, amx_cfg, tiledata, xstate);276277int iter = 0;278for (;;) {279vcpu_run(vcpu);280TEST_ASSERT_KVM_EXIT_REASON(vcpu, KVM_EXIT_IO);281282switch (get_ucall(vcpu, &uc)) {283case UCALL_ABORT:284REPORT_GUEST_ASSERT(uc);285/* NOT REACHED */286case UCALL_SYNC:287++iter;288if (uc.args[1] & TEST_SAVE_TILEDATA) {289fprintf(stderr, "GUEST_SYNC #%d, save tiledata\n", iter);290tile_state = vcpu_save_state(vcpu);291}292if (uc.args[1] & TEST_COMPARE_TILEDATA) {293fprintf(stderr, "GUEST_SYNC #%d, check TMM0 contents\n", iter);294295/* Compacted mode, get amx offset by xsave area296* size subtract 8K amx size.297*/298u32 amx_offset = xsave_restore_size - NUM_TILES*TILE_SIZE;299void *amx_start = (void *)tile_state->xsave + amx_offset;300void *tiles_data = (void *)addr_gva2hva(vm, tiledata);301/* Only check TMM0 register, 1 tile */302ret = memcmp(amx_start, tiles_data, TILE_SIZE);303TEST_ASSERT(ret == 0, "memcmp failed, ret=%d", ret);304}305if (uc.args[1] & TEST_RESTORE_TILEDATA) {306fprintf(stderr, "GUEST_SYNC #%d, before KVM_SET_XSAVE\n", iter);307vcpu_xsave_set(vcpu, tile_state->xsave);308fprintf(stderr, "GUEST_SYNC #%d, after KVM_SET_XSAVE\n", iter);309}310if (uc.args[1] & TEST_SAVE_RESTORE) {311fprintf(stderr, "GUEST_SYNC #%d, save/restore VM state\n", iter);312state = vcpu_save_state(vcpu);313memset(®s1, 0, sizeof(regs1));314vcpu_regs_get(vcpu, ®s1);315316kvm_vm_release(vm);317318/* Restore state in a new VM. */319vcpu = vm_recreate_with_one_vcpu(vm);320vcpu_load_state(vcpu, state);321kvm_x86_state_cleanup(state);322323memset(®s2, 0, sizeof(regs2));324vcpu_regs_get(vcpu, ®s2);325TEST_ASSERT(!memcmp(®s1, ®s2, sizeof(regs2)),326"Unexpected register values after vcpu_load_state; rdi: %lx rsi: %lx",327(ulong) regs2.rdi, (ulong) regs2.rsi);328}329break;330case UCALL_DONE:331fprintf(stderr, "UCALL_DONE\n");332goto done;333default:334TEST_FAIL("Unknown ucall %lu", uc.cmd);335}336337}338done:339kvm_vm_free(vm);340}341342343