2008-12-05 15:48:36 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 00:54:04 +03:00
|
|
|
// $Id$
|
2008-12-05 15:48:36 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2019-12-20 10:42:07 +03:00
|
|
|
// Copyright (c) 2008-2019 Stanislav Shwartsman
|
2008-12-05 15:48:36 +03:00
|
|
|
// Written by Stanislav Shwartsman [sshwarts at sourceforge net]
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
2009-01-16 21:18:59 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA B 02110-1301 USA
|
2008-12-05 15:48:36 +03:00
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#define NEED_CPU_REG_SHORTCUTS 1
|
|
|
|
#include "bochs.h"
|
|
|
|
#include "cpu.h"
|
2023-12-29 22:27:00 +03:00
|
|
|
#include "cpuid.h"
|
2019-10-26 23:09:30 +03:00
|
|
|
#include "msr.h"
|
2008-12-05 15:48:36 +03:00
|
|
|
#define LOG_THIS BX_CPU_THIS_PTR
|
|
|
|
|
2024-01-12 01:56:09 +03:00
|
|
|
#if BX_SUPPORT_SVM
|
|
|
|
#include "svm.h"
|
|
|
|
#endif
|
|
|
|
|
2023-11-28 12:08:42 +03:00
|
|
|
#if BX_SUPPORT_APIC
|
|
|
|
#include "apic.h"
|
|
|
|
#endif
|
|
|
|
|
2019-12-20 10:42:07 +03:00
|
|
|
#if BX_SUPPORT_CET
|
2021-01-30 11:35:35 +03:00
|
|
|
extern bool is_invalid_cet_control(bx_address val);
|
2019-12-20 10:42:07 +03:00
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_CPU_LEVEL >= 5
|
2021-01-30 11:35:35 +03:00
|
|
|
bool BX_CPP_AttrRegparmN(2) BX_CPU_C::rdmsr(Bit32u index, Bit64u *msr)
|
2009-01-19 21:08:38 +03:00
|
|
|
{
|
2009-01-19 20:43:54 +03:00
|
|
|
Bit64u val64 = 0;
|
2008-12-28 23:30:48 +03:00
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
2024-01-27 16:50:51 +03:00
|
|
|
VMCS_CACHE *vm = &BX_CPU_THIS_PTR vmcs;
|
2023-12-16 22:59:34 +03:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest) {
|
2024-01-27 16:50:51 +03:00
|
|
|
if (vm->vmexec_ctrls2.VIRTUALIZE_X2APIC_MODE()) {
|
2023-12-16 22:59:34 +03:00
|
|
|
if (index >= 0x800 && index <= 0x8FF) {
|
2024-01-27 16:50:51 +03:00
|
|
|
if (index == 0x808 || vm->vmexec_ctrls2.VIRTUALIZE_APIC_REGISTERS()) {
|
2023-12-16 22:59:34 +03:00
|
|
|
unsigned vapic_offset = (index & 0xff) << 4;
|
|
|
|
Bit32u msr_lo = VMX_Read_Virtual_APIC(vapic_offset);
|
|
|
|
Bit32u msr_hi = VMX_Read_Virtual_APIC(vapic_offset + 4);
|
|
|
|
*msr = GET64_FROM_HI32_LO32(msr_hi, msr_lo);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-04-21 17:27:42 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2014-08-31 23:22:41 +04:00
|
|
|
if (is_cpu_extension_supported(BX_ISA_X2APIC)) {
|
2019-12-11 00:07:19 +03:00
|
|
|
if (is_x2apic_msr_range(index)) {
|
2023-11-21 16:43:38 +03:00
|
|
|
if (x2apic_mode())
|
2023-11-28 12:08:42 +03:00
|
|
|
return BX_CPU_THIS_PTR lapic->read_x2apic(index, msr);
|
2011-04-21 17:27:42 +04:00
|
|
|
else
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2011-04-21 17:27:42 +04:00
|
|
|
}
|
2010-04-08 19:50:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-12-28 23:30:48 +03:00
|
|
|
switch(index) {
|
2010-02-26 14:44:50 +03:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_SYSENTER_CS:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_SYSENTER_CS: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.sysenter_cs_msr;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_SYSENTER_ESP:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_SYSENTER_ESP: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.sysenter_esp_msr;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_SYSENTER_EIP:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_SYSENTER_EIP: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.sysenter_eip_msr;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2009-06-15 19:10:05 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_MTRRCAP: // read only MSR
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_MTRRCAP: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-21 00:50:59 +03:00
|
|
|
val64 = BX_CONST64(0x0000000000000500) | BX_NUM_VARIABLE_RANGE_MTRRS;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_MTRRPHYSBASE0:
|
|
|
|
case BX_MSR_MTRRPHYSMASK0:
|
|
|
|
case BX_MSR_MTRRPHYSBASE1:
|
|
|
|
case BX_MSR_MTRRPHYSMASK1:
|
|
|
|
case BX_MSR_MTRRPHYSBASE2:
|
|
|
|
case BX_MSR_MTRRPHYSMASK2:
|
|
|
|
case BX_MSR_MTRRPHYSBASE3:
|
|
|
|
case BX_MSR_MTRRPHYSMASK3:
|
|
|
|
case BX_MSR_MTRRPHYSBASE4:
|
|
|
|
case BX_MSR_MTRRPHYSMASK4:
|
|
|
|
case BX_MSR_MTRRPHYSBASE5:
|
|
|
|
case BX_MSR_MTRRPHYSMASK5:
|
|
|
|
case BX_MSR_MTRRPHYSBASE6:
|
|
|
|
case BX_MSR_MTRRPHYSMASK6:
|
|
|
|
case BX_MSR_MTRRPHYSBASE7:
|
|
|
|
case BX_MSR_MTRRPHYSMASK7:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.mtrrphys[index - BX_MSR_MTRRPHYSBASE0];
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_MTRRFIX64K_00000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-24 00:17:33 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.mtrrfix64k.u64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
case BX_MSR_MTRRFIX16K_80000:
|
|
|
|
case BX_MSR_MTRRFIX16K_A0000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-24 00:17:33 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.mtrrfix16k[index - BX_MSR_MTRRFIX16K_80000].u64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_MTRRFIX4K_C0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_C8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_D0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_D8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_E0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_E8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_F0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_F8000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-24 00:17:33 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.mtrrfix4k[index - BX_MSR_MTRRFIX4K_C0000].u64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2013-04-17 23:59:56 +04:00
|
|
|
case BX_MSR_MTRR_DEFTYPE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_MTRR_DEFTYPE: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.mtrr_deftype;
|
|
|
|
break;
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_PAT:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_PAT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR BX_MSR_PAT: PAT is not enabled in the cpu model"));
|
2013-04-11 23:41:54 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-24 00:17:33 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.pat.u64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case BX_MSR_TSC:
|
2023-11-25 16:00:39 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR get_Virtual_TSC();
|
2020-05-21 22:58:16 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_TSC_ADJUST:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_TSC_ADJUST)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_TSC_ADJUST: TSC_ADJUST is not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR tsc_adjust;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2023-12-01 19:00:03 +03:00
|
|
|
#if BX_SUPPORT_MONITOR_MWAIT
|
|
|
|
case BX_MSR_IA32_UMWAIT_CONTROL:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_WAITPKG)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UMWAIT_CONTROL: WAITPKG is not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_umwait_ctrl;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_SUPPORT_APIC
|
|
|
|
case BX_MSR_APICBASE:
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.apicbase;
|
|
|
|
BX_INFO(("RDMSR: Read %08x:%08x from MSR_APICBASE", GET32H(val64), GET32L(val64)));
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2017-03-16 00:44:15 +03:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
|
|
|
case BX_MSR_XSS:
|
2019-12-06 12:23:28 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_xss;
|
2017-03-16 00:44:15 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2019-12-20 10:42:07 +03:00
|
|
|
#if BX_SUPPORT_CET
|
|
|
|
case BX_MSR_IA32_U_CET:
|
|
|
|
case BX_MSR_IA32_S_CET:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_U_CET/BX_MSR_IA32_S_CET: CET not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_cet_control[index == BX_MSR_IA32_U_CET];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_PL0_SSP:
|
|
|
|
case BX_MSR_IA32_PL1_SSP:
|
|
|
|
case BX_MSR_IA32_PL2_SSP:
|
|
|
|
case BX_MSR_IA32_PL3_SSP:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_PLi_SSP: CET not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_pl_ssp[index - BX_MSR_IA32_PL0_SSP];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_INTERRUPT_SSP_TABLE_ADDR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_INTERRUPT_SSP_TABLE_ADDR: CET not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_interrupt_ssp_table;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2023-11-25 17:43:47 +03:00
|
|
|
#if BX_SUPPORT_UINTR
|
|
|
|
case BX_MSR_IA32_UINTR_RR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_RR: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR uintr.uirr;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_HANDLER:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_HANDLER: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR uintr.ui_handler;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_STACKADJUST:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_STACKADJUST: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR uintr.stack_adjust;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_MISC:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_MISC: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = GET64_FROM_HI32_LO32(BX_CPU_THIS_PTR uintr.uinv, BX_CPU_THIS_PTR uintr.uitt_size);
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_PD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_PD: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR uintr.upid_addr;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_TT:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IA32_UINTR_TT: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR uintr.uitt_addr;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2020-05-29 15:35:30 +03:00
|
|
|
#if BX_SUPPORT_PKEYS
|
|
|
|
case BX_MSR_IA32_PKRS:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_PKS)) {
|
|
|
|
BX_ERROR(("RDMSR BX_MSR_IS32_PKS: Supervisor-Mode Protection Keys not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR pkrs;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2011-11-21 16:51:50 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
|
|
|
case BX_MSR_TSC_DEADLINE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_TSC_DEADLINE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR BX_MSR_TSC_DEADLINE: TSC-Deadline not enabled in the cpu model"));
|
2011-11-21 16:51:50 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2023-11-28 12:08:42 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR lapic->get_tsc_deadline();
|
2011-11-21 16:51:50 +04:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
// artificial MSR for MRSLIST serialization
|
|
|
|
case BX_MSR_IA32_BARRIER:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MSRLIST)) {
|
|
|
|
BX_ERROR(("RDMSR IA32_BARRIER: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// SCA prevention MSRs
|
2019-10-26 23:09:30 +03:00
|
|
|
case BX_MSR_IA32_ARCH_CAPABILITIES:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
|
|
|
BX_ERROR(("RDMSR IA32_ARCH_CAPABILITIES: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
// [0]: RDCL_NO: The processor is not susceptible to Rogue Data Cache Load (RDCL)
|
|
|
|
// [1]: IBRS_ALL: The processor supports enhanced IBRS
|
|
|
|
// [2]: RSBA: The processor supports RSB Alternate
|
|
|
|
// [3]: SKIP_L1DFL_VMENTRY: indicates the hypervisor need not flush the L1D on VM entry
|
|
|
|
// [4]: SSB_NO: Processor is not susceptible to Speculative Store Bypass
|
|
|
|
// [63:5]: reserved
|
|
|
|
val64 = 0x1F; // set bits [4:0]
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_SPEC_CTRL:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
2024-01-13 22:58:05 +03:00
|
|
|
BX_ERROR(("RDMSR IA32_SPEC_CTRL: not enabled in the cpu model"));
|
2019-10-26 23:09:30 +03:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
// [0] - Enable IBRS: Indirect Branch Restricted Speculation
|
|
|
|
// [1] - Enable STIBP: Single Thread Indirect Branch Predictors
|
|
|
|
// [2] - Enable SSCB: Speculative Store Bypass Disable
|
|
|
|
// [63:3] - reserved
|
2024-01-27 19:30:22 +03:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
2024-01-27 16:50:51 +03:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest && vm->vmexec_ctrls3.VIRTUALIZE_IA32_SPEC_CTRL())
|
|
|
|
val64 = vm->ia32_spec_ctrl_shadow;
|
2024-01-13 22:58:05 +03:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_spec_ctrl;
|
2019-10-26 23:09:30 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_PRED_CMD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
2024-01-13 22:58:05 +03:00
|
|
|
BX_ERROR(("RDMSR IA32_PRED_CMD: not enabled in the cpu model"));
|
2019-10-26 23:09:30 +03:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
// write only MSR, no need to remember written value
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
|
|
|
|
case BX_MSR_IA32_FLUSH_CMD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
2024-01-13 22:58:05 +03:00
|
|
|
BX_ERROR(("RDMSR IA32_FLUSH_CMD: not enabled in the cpu model"));
|
2019-10-26 23:09:30 +03:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
// write only MSR, no need to remember written value
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
|
2009-01-31 13:43:24 +03:00
|
|
|
#if BX_SUPPORT_VMX
|
2009-10-24 21:33:39 +04:00
|
|
|
/*
|
2009-10-22 17:33:13 +04:00
|
|
|
case BX_MSR_IA32_SMM_MONITOR_CTL:
|
|
|
|
BX_PANIC(("Dual-monitor treatment of SMI and SMM is not implemented"));
|
|
|
|
break;
|
2009-10-24 21:33:39 +04:00
|
|
|
*/
|
2010-03-06 19:59:05 +03:00
|
|
|
case BX_MSR_IA32_FEATURE_CONTROL:
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.ia32_feature_ctrl;
|
|
|
|
break;
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_BASIC:
|
|
|
|
val64 = VMX_MSR_VMX_BASIC;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_PINBASED_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_PINBASED_CTRLS;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_PROCBASED_CTRLS;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_VMEXIT_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_VMEXIT_CTRLS;
|
|
|
|
break;
|
2024-01-27 14:34:51 +03:00
|
|
|
case BX_MSR_VMX_VMEXIT_CTRLS2:
|
|
|
|
if (BX_CPU_THIS_PTR vmx_cap.vmx_vmexit_ctrl2_supported_bits) {
|
|
|
|
val64 = VMX_MSR_VMX_VMEXIT_CTRLS2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_VMENTRY_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_VMENTRY_CTRLS;
|
|
|
|
break;
|
2010-03-15 18:48:01 +03:00
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS2:
|
2011-09-27 00:36:26 +04:00
|
|
|
if (BX_CPU_THIS_PTR vmx_cap.vmx_vmexec_ctrl2_supported_bits) {
|
|
|
|
val64 = VMX_MSR_VMX_PROCBASED_CTRLS2;
|
|
|
|
break;
|
|
|
|
}
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-12-16 22:59:34 +03:00
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS3:
|
|
|
|
if (BX_CPU_THIS_PTR vmx_cap.vmx_vmexec_ctrl3_supported_bits) {
|
|
|
|
val64 = VMX_MSR_VMX_PROCBASED_CTRLS3;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
2011-09-27 00:36:26 +04:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
2010-04-03 11:30:23 +04:00
|
|
|
case BX_MSR_VMX_TRUE_PINBASED_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_TRUE_PINBASED_CTRLS;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_TRUE_PROCBASED_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_TRUE_PROCBASED_CTRLS;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_TRUE_VMEXIT_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_TRUE_VMEXIT_CTRLS;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_TRUE_VMENTRY_CTRLS:
|
|
|
|
val64 = VMX_MSR_VMX_TRUE_VMENTRY_CTRLS;
|
|
|
|
break;
|
2011-09-27 00:36:26 +04:00
|
|
|
case BX_MSR_VMX_EPT_VPID_CAP:
|
2012-05-02 22:11:39 +04:00
|
|
|
if (VMX_MSR_VMX_EPT_VPID_CAP != 0) {
|
2011-09-27 00:36:26 +04:00
|
|
|
val64 = VMX_MSR_VMX_EPT_VPID_CAP;
|
|
|
|
break;
|
|
|
|
}
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2011-11-05 11:31:51 +04:00
|
|
|
case BX_MSR_VMX_VMFUNC:
|
|
|
|
if (BX_CPU_THIS_PTR vmx_cap.vmx_vmfunc_supported_bits) {
|
|
|
|
val64 = BX_CPU_THIS_PTR vmx_cap.vmx_vmfunc_supported_bits;
|
|
|
|
break;
|
|
|
|
}
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2010-04-03 11:30:23 +04:00
|
|
|
#endif
|
2010-03-13 00:20:42 +03:00
|
|
|
case BX_MSR_VMX_MISC:
|
|
|
|
val64 = VMX_MSR_MISC;
|
|
|
|
break;
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_CR0_FIXED0:
|
|
|
|
val64 = VMX_MSR_CR0_FIXED0;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_CR0_FIXED1:
|
|
|
|
val64 = VMX_MSR_CR0_FIXED1;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_CR4_FIXED0:
|
|
|
|
val64 = VMX_MSR_CR4_FIXED0;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_CR4_FIXED1:
|
|
|
|
val64 = VMX_MSR_CR4_FIXED1;
|
|
|
|
break;
|
|
|
|
case BX_MSR_VMX_VMCS_ENUM:
|
|
|
|
val64 = VMX_MSR_VMCS_ENUM;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_EFER:
|
2011-12-28 20:12:28 +04:00
|
|
|
if (! BX_CPU_THIS_PTR efer_suppmask) {
|
|
|
|
BX_ERROR(("RDMSR MSR_EFER: EFER MSR is not supported !"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR efer.get32();
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2011-12-31 17:58:55 +04:00
|
|
|
#if BX_SUPPORT_SVM
|
2024-01-16 08:42:28 +03:00
|
|
|
case BX_SVM_VM_CR_MSR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SVM)) {
|
|
|
|
BX_ERROR(("RDMSR SVM_VM_CR_MSR: SVM support not enabled in the cpu model"));
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.svm_vm_cr;
|
|
|
|
break;
|
|
|
|
|
2011-12-31 17:58:55 +04:00
|
|
|
case BX_SVM_HSAVE_PA_MSR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SVM)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR SVM_HSAVE_PA_MSR: SVM support not enabled in the cpu model"));
|
2011-12-31 17:58:55 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
val64 = BX_CPU_THIS_PTR msr.svm_hsave_pa;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_STAR:
|
2011-09-25 21:36:20 +04:00
|
|
|
if ((BX_CPU_THIS_PTR efer_suppmask & BX_EFER_SCE_MASK) == 0) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_STAR: SYSCALL/SYSRET support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-07-13 23:24:14 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.star;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2011-08-31 02:00:27 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_LSTAR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_LSTAR: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.lstar;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_CSTAR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_CSTAR: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.cstar;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_FMASK:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_FMASK: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.fmask;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_FSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_FSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = MSR_FSBASE;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_GSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_GSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
val64 = MSR_GSBASE;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_KERNELGSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_KERNELGSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
2015-07-13 23:24:14 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.kernelgsbase;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_TSC_AUX:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_RDTSCP)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("RDMSR MSR_TSC_AUX: RTDSCP feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_rdmsr(index, msr);
|
2011-08-21 23:09:35 +04:00
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
val64 = BX_CPU_THIS_PTR msr.tsc_aux; // 32 bit MSR
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
2011-08-11 02:04:33 +04:00
|
|
|
#endif
|
2008-12-05 15:48:36 +03:00
|
|
|
|
2011-09-25 21:36:20 +04:00
|
|
|
default:
|
|
|
|
return handle_unknown_rdmsr(index, msr);
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_DEBUG(("RDMSR: read %08x:%08x from MSR %x", GET32H(val64), GET32L(val64), index));
|
2011-08-10 02:11:56 +04:00
|
|
|
|
2011-09-25 21:36:20 +04:00
|
|
|
*msr = val64;
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2011-09-25 21:36:20 +04:00
|
|
|
}
|
|
|
|
|
2021-01-30 11:35:35 +03:00
|
|
|
bool BX_CPP_AttrRegparmN(2) BX_CPU_C::handle_unknown_rdmsr(Bit32u index, Bit64u *msr)
|
2011-09-25 21:36:20 +04:00
|
|
|
{
|
|
|
|
Bit64u val_64 = 0;
|
|
|
|
|
|
|
|
// Try to check cpuid_t first (can implement some MSRs)
|
|
|
|
int result = BX_CPU_THIS_PTR cpuid->rdmsr(index, &val_64);
|
|
|
|
if (result == 0)
|
2023-12-16 20:07:03 +03:00
|
|
|
return false; // #GP fault due to not supported MSR
|
2011-09-25 21:36:20 +04:00
|
|
|
|
|
|
|
if (result < 0) {
|
|
|
|
// cpuid_t have no idea about this MSR
|
2008-12-28 23:30:48 +03:00
|
|
|
#if BX_CONFIGURE_MSRS
|
2011-09-25 21:36:20 +04:00
|
|
|
if (index < BX_MSR_MAX_INDEX && BX_CPU_THIS_PTR msrs[index]) {
|
|
|
|
val_64 = BX_CPU_THIS_PTR msrs[index]->get64();
|
|
|
|
}
|
|
|
|
else
|
2008-12-28 23:30:48 +03:00
|
|
|
#endif
|
2011-09-25 21:36:20 +04:00
|
|
|
{
|
|
|
|
// failed to find the MSR, could #GP or ignore it silently
|
|
|
|
BX_ERROR(("RDMSR: Unknown register %#x", index));
|
2009-01-19 21:08:38 +03:00
|
|
|
|
2022-07-31 18:47:13 +03:00
|
|
|
if (! BX_CPU_THIS_PTR ignore_bad_msrs)
|
2023-12-16 20:07:03 +03:00
|
|
|
return false; // will result in #GP fault due to unknown MSR
|
2011-08-10 02:11:56 +04:00
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-01-19 20:43:54 +03:00
|
|
|
|
2011-09-25 21:36:20 +04:00
|
|
|
*msr = val_64;
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2009-01-19 21:08:38 +03:00
|
|
|
}
|
2011-09-25 21:36:20 +04:00
|
|
|
|
2009-01-19 21:08:38 +03:00
|
|
|
#endif // BX_CPU_LEVEL >= 5
|
|
|
|
|
2018-02-16 10:57:32 +03:00
|
|
|
void BX_CPP_AttrRegparmN(1) BX_CPU_C::RDMSR(bxInstruction_c *i)
|
2009-01-19 21:08:38 +03:00
|
|
|
{
|
|
|
|
#if BX_CPU_LEVEL >= 5
|
2011-08-24 01:25:34 +04:00
|
|
|
// CPL is always 0 in real mode
|
|
|
|
if (/* !real_mode() && */ CPL!=0) {
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_ERROR(("RDMSR: CPL != 0 not in real mode"));
|
2010-03-14 18:51:27 +03:00
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
2009-01-19 21:08:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Bit32u index = ECX;
|
|
|
|
Bit64u val64 = 0;
|
|
|
|
|
2011-12-26 23:57:39 +04:00
|
|
|
#if BX_SUPPORT_SVM
|
|
|
|
if (BX_CPU_THIS_PTR in_svm_guest) {
|
2011-12-27 23:42:11 +04:00
|
|
|
if (SVM_INTERCEPT(SVM_INTERCEPT0_MSR)) SvmInterceptMSR(BX_READ, index);
|
2011-12-26 23:57:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-01-31 13:43:24 +03:00
|
|
|
#if BX_SUPPORT_VMX
|
2011-08-10 00:50:51 +04:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest)
|
2012-07-26 20:03:26 +04:00
|
|
|
VMexit_MSR(VMX_VMEXIT_RDMSR, index);
|
2009-01-31 13:43:24 +03:00
|
|
|
#endif
|
|
|
|
|
2009-01-19 21:08:38 +03:00
|
|
|
if (!rdmsr(index, &val64))
|
2010-03-14 18:51:27 +03:00
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
2009-01-19 21:08:38 +03:00
|
|
|
|
2009-01-19 20:43:54 +03:00
|
|
|
RAX = GET32L(val64);
|
|
|
|
RDX = GET32H(val64);
|
2008-12-05 15:48:36 +03:00
|
|
|
#endif
|
2011-07-07 00:01:18 +04:00
|
|
|
|
|
|
|
BX_NEXT_INSTR(i);
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
|
|
|
|
2009-06-15 19:10:05 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2021-01-30 11:35:35 +03:00
|
|
|
bool isMemTypeValidMTRR(unsigned memtype)
|
2008-12-05 15:48:36 +03:00
|
|
|
{
|
|
|
|
switch(memtype) {
|
2009-01-03 23:04:03 +03:00
|
|
|
case BX_MEMTYPE_UC:
|
|
|
|
case BX_MEMTYPE_WC:
|
|
|
|
case BX_MEMTYPE_WT:
|
|
|
|
case BX_MEMTYPE_WP:
|
|
|
|
case BX_MEMTYPE_WB:
|
2019-12-09 19:44:36 +03:00
|
|
|
return true;
|
2008-12-05 15:48:36 +03:00
|
|
|
default:
|
2019-12-09 19:44:36 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-30 11:35:35 +03:00
|
|
|
BX_CPP_INLINE bool isMemTypeValidPAT(unsigned memtype)
|
2008-12-05 15:48:36 +03:00
|
|
|
{
|
2009-01-19 19:45:54 +03:00
|
|
|
return (memtype == 0x07) /* UC- */ || isMemTypeValidMTRR(memtype);
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2010-03-27 00:26:08 +03:00
|
|
|
|
2021-01-30 11:35:35 +03:00
|
|
|
bool isValidMSR_PAT(Bit64u pat_val)
|
2010-03-27 00:26:08 +03:00
|
|
|
{
|
2015-02-24 00:17:33 +03:00
|
|
|
// use packed register as 64-bit value with convinient accessors
|
|
|
|
BxPackedRegister pat_msr = pat_val;
|
|
|
|
for (unsigned i=0; i<8; i++)
|
2019-12-09 19:44:36 +03:00
|
|
|
if (! isMemTypeValidPAT(pat_msr.ubyte(i))) return false;
|
2010-03-27 00:26:08 +03:00
|
|
|
|
2019-12-09 19:44:36 +03:00
|
|
|
return true;
|
2010-03-27 00:26:08 +03:00
|
|
|
}
|
|
|
|
|
2021-01-30 11:35:35 +03:00
|
|
|
bool isValidMSR_FixedMTRR(Bit64u fixed_mtrr_val)
|
2010-03-27 00:26:08 +03:00
|
|
|
{
|
2015-02-24 00:17:33 +03:00
|
|
|
// use packed register as 64-bit value with convinient accessors
|
|
|
|
BxPackedRegister fixed_mtrr_msr = fixed_mtrr_val;
|
|
|
|
for (unsigned i=0; i<8; i++)
|
2019-12-09 19:44:36 +03:00
|
|
|
if (! isMemTypeValidMTRR(fixed_mtrr_msr.ubyte(i))) return false;
|
2010-03-27 00:26:08 +03:00
|
|
|
|
2019-12-09 19:44:36 +03:00
|
|
|
return true;
|
2010-03-27 00:26:08 +03:00
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
#endif
|
|
|
|
|
2024-04-27 09:44:32 +03:00
|
|
|
bool isValidMSR_IA32_SPEC_CTRL(Bit64u val_64)
|
|
|
|
{
|
|
|
|
// [0] - Enable IBRS: Indirect Branch Restricted Speculation
|
|
|
|
// [1] - Enable STIBP: Single Thread Indirect Branch Predictors
|
|
|
|
// [2] - Enable SSCB: Speculative Store Bypass Disable
|
|
|
|
// [63:3] - reserved
|
|
|
|
if (val_64 & ~(BX_CONST64(0x7)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_CPU_LEVEL >= 5
|
2021-01-30 11:35:35 +03:00
|
|
|
bool BX_CPP_AttrRegparmN(2) BX_CPU_C::wrmsr(Bit32u index, Bit64u val_64)
|
2009-01-19 21:08:38 +03:00
|
|
|
{
|
|
|
|
Bit32u val32_lo = GET32L(val_64);
|
|
|
|
Bit32u val32_hi = GET32H(val_64);
|
2008-12-05 15:48:36 +03:00
|
|
|
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_INSTR_WRMSR(BX_CPU_ID, index, val_64);
|
2008-12-05 15:48:36 +03:00
|
|
|
|
2010-05-16 17:56:22 +04:00
|
|
|
BX_DEBUG(("WRMSR: write %08x:%08x to MSR %x", val32_hi, val32_lo, index));
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
2024-01-27 16:50:51 +03:00
|
|
|
VMCS_CACHE *vm = &BX_CPU_THIS_PTR vmcs;
|
2023-12-16 22:59:34 +03:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest) {
|
2024-01-27 16:50:51 +03:00
|
|
|
if (vm->vmexec_ctrls2.VIRTUALIZE_X2APIC_MODE()) {
|
2023-12-16 22:59:34 +03:00
|
|
|
if (Virtualize_X2APIC_Write(index, val_64))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-04-21 17:27:42 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2014-08-31 23:22:41 +04:00
|
|
|
if (is_cpu_extension_supported(BX_ISA_X2APIC)) {
|
2019-12-11 00:07:19 +03:00
|
|
|
if (is_x2apic_msr_range(index)) {
|
2023-11-21 16:43:38 +03:00
|
|
|
if (x2apic_mode())
|
2023-11-28 12:08:42 +03:00
|
|
|
return BX_CPU_THIS_PTR lapic->write_x2apic(index, val32_hi, val32_lo);
|
2011-04-21 17:27:42 +04:00
|
|
|
else
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2011-04-21 17:27:42 +04:00
|
|
|
}
|
2010-04-08 19:50:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-12-28 23:30:48 +03:00
|
|
|
switch(index) {
|
2008-12-05 15:48:36 +03:00
|
|
|
|
2014-11-07 16:15:54 +03:00
|
|
|
#if BX_SUPPORT_PERFMON
|
2014-10-15 23:04:28 +04:00
|
|
|
case BX_MSR_PERFEVTSEL0:
|
|
|
|
case BX_MSR_PERFEVTSEL1:
|
|
|
|
case BX_MSR_PERFEVTSEL2:
|
|
|
|
case BX_MSR_PERFEVTSEL3:
|
|
|
|
case BX_MSR_PERFEVTSEL4:
|
|
|
|
case BX_MSR_PERFEVTSEL5:
|
|
|
|
case BX_MSR_PERFEVTSEL6:
|
|
|
|
case BX_MSR_PERFEVTSEL7:
|
|
|
|
BX_INFO(("WRMSR: write into IA32_PERFEVTSEL%d MSR: %08x:%08x", index - BX_MSR_PERFEVTSEL0, val32_hi, val32_lo));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
2014-11-07 16:15:54 +03:00
|
|
|
#endif
|
2014-10-15 23:04:28 +04:00
|
|
|
|
2010-02-26 14:44:50 +03:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_SYSENTER_CS:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_SYSENTER_CS: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.sysenter_cs_msr = val32_lo;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_SYSENTER_ESP:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_SYSENTER_ESP: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_SUPPORT_X86_64
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_SYSENTER_ESP !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
|
|
|
#endif
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.sysenter_esp_msr = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_SYSENTER_EIP:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SYSENTER_SYSEXIT)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_SYSENTER_EIP: SYSENTER/SYSEXIT feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
2010-02-26 14:44:50 +03:00
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_SUPPORT_X86_64
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_SYSENTER_EIP !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
|
|
|
#endif
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.sysenter_eip_msr = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2009-06-15 19:10:05 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_MTRRCAP:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_MTRRCAP: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRRCAP is read only MSR"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
|
|
|
|
case BX_MSR_MTRRPHYSBASE0:
|
|
|
|
case BX_MSR_MTRRPHYSBASE1:
|
|
|
|
case BX_MSR_MTRRPHYSBASE2:
|
|
|
|
case BX_MSR_MTRRPHYSBASE3:
|
|
|
|
case BX_MSR_MTRRPHYSBASE4:
|
|
|
|
case BX_MSR_MTRRPHYSBASE5:
|
|
|
|
case BX_MSR_MTRRPHYSBASE6:
|
|
|
|
case BX_MSR_MTRRPHYSBASE7:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-11-04 20:04:28 +03:00
|
|
|
if (! IsValidPhyAddr(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR[0x%08x]: attempt to write invalid phy addr to variable range MTRR %08x:%08x", index, val32_hi, val32_lo));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-11-04 20:04:28 +03:00
|
|
|
}
|
2009-09-28 19:39:52 +04:00
|
|
|
// handle 8-11 reserved bits
|
|
|
|
if (! isMemTypeValidMTRR(val32_lo & 0xFFF)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to BX_MSR_MTRRPHYSBASE"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-11-04 09:46:04 +03:00
|
|
|
BX_CPU_THIS_PTR msr.mtrrphys[index - BX_MSR_MTRRPHYSBASE0] = val_64;
|
|
|
|
break;
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_MTRRPHYSMASK0:
|
|
|
|
case BX_MSR_MTRRPHYSMASK1:
|
|
|
|
case BX_MSR_MTRRPHYSMASK2:
|
|
|
|
case BX_MSR_MTRRPHYSMASK3:
|
|
|
|
case BX_MSR_MTRRPHYSMASK4:
|
|
|
|
case BX_MSR_MTRRPHYSMASK5:
|
|
|
|
case BX_MSR_MTRRPHYSMASK6:
|
|
|
|
case BX_MSR_MTRRPHYSMASK7:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-09-28 19:39:52 +04:00
|
|
|
if (! IsValidPhyAddr(val_64)) {
|
2009-10-03 18:29:31 +04:00
|
|
|
BX_ERROR(("WRMSR[0x%08x]: attempt to write invalid phy addr to variable range MTRR %08x:%08x", index, val32_hi, val32_lo));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-09-28 19:39:52 +04:00
|
|
|
}
|
2009-11-04 20:04:28 +03:00
|
|
|
// handle 10-0 reserved bits
|
2009-11-03 15:34:36 +03:00
|
|
|
if (val32_lo & 0x7ff) {
|
|
|
|
BX_ERROR(("WRMSR[0x%08x]: variable range MTRR reserved bits violation %08x:%08x", index, val32_hi, val32_lo));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-11-03 15:34:36 +03:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.mtrrphys[index - BX_MSR_MTRRPHYSBASE0] = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_MTRRFIX64K_00000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2010-03-27 00:26:08 +03:00
|
|
|
if (! isValidMSR_FixedMTRR(val_64)) {
|
2009-05-21 17:25:30 +04:00
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to MSR_MTRRFIX64K_00000 !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
BX_CPU_THIS_PTR msr.mtrrfix64k = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
2013-04-17 23:59:56 +04:00
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_MTRRFIX16K_80000:
|
|
|
|
case BX_MSR_MTRRFIX16K_A0000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2010-03-27 00:26:08 +03:00
|
|
|
if (! isValidMSR_FixedMTRR(val_64)) {
|
2019-12-09 19:29:23 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to MSR_MTRRFIX16K register !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-05-21 17:25:30 +04:00
|
|
|
BX_CPU_THIS_PTR msr.mtrrfix16k[index - BX_MSR_MTRRFIX16K_80000] = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_MTRRFIX4K_C0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_C8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_D0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_D8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_E0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_E8000:
|
|
|
|
case BX_MSR_MTRRFIX4K_F0000:
|
|
|
|
case BX_MSR_MTRRFIX4K_F8000:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2010-03-27 00:26:08 +03:00
|
|
|
if (! isValidMSR_FixedMTRR(val_64)) {
|
2009-05-21 17:25:30 +04:00
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to fixed memory range MTRR !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-05-21 17:25:30 +04:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.mtrrfix4k[index - BX_MSR_MTRRFIX4K_C0000] = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2013-04-17 23:59:56 +04:00
|
|
|
case BX_MSR_MTRR_DEFTYPE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MTRR)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_MTRR_DEFTYPE: MTRR is not enabled in the cpu model"));
|
2013-04-17 23:59:56 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! isMemTypeValidMTRR(val32_lo & 0xFF)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to MSR_MTRR_DEFTYPE"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2013-04-17 23:59:56 +04:00
|
|
|
}
|
|
|
|
if (val32_hi || (val32_lo & 0xfffff300)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to reserved bits in MSR_MTRR_DEFTYPE"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2013-04-17 23:59:56 +04:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.mtrr_deftype = val32_lo;
|
|
|
|
break;
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_PAT:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_PAT)) {
|
2013-04-11 23:41:54 +04:00
|
|
|
BX_ERROR(("WRMSR BX_MSR_PAT: PAT is not enabled !"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
|
2010-03-27 00:26:08 +03:00
|
|
|
if (! isValidMSR_PAT(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write invalid Memory Type to MSR_PAT"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2022-08-23 21:38:00 +03:00
|
|
|
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_CPU_THIS_PTR msr.pat = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case BX_MSR_TSC:
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_INFO(("WRMSR: write 0x%08x%08x to MSR_TSC", val32_hi, val32_lo));
|
|
|
|
BX_CPU_THIS_PTR set_TSC(val_64);
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2020-05-21 22:58:16 +03:00
|
|
|
case BX_MSR_TSC_ADJUST:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_TSC_ADJUST)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_TSC_ADJUST: TSC_ADJUST is not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR tsc_adjust = (Bit64s) val_64;
|
|
|
|
break;
|
|
|
|
|
2023-12-01 19:00:03 +03:00
|
|
|
#if BX_SUPPORT_MONITOR_MWAIT
|
|
|
|
case BX_MSR_IA32_UMWAIT_CONTROL:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_WAITPKG)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UMWAIT_CONTROL: WAITPKG is not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.ia32_umwait_ctrl = val32_lo;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
#if BX_SUPPORT_APIC
|
|
|
|
case BX_MSR_APICBASE:
|
2009-02-19 01:25:04 +03:00
|
|
|
return relocate_apic(val_64);
|
2008-12-05 15:48:36 +03:00
|
|
|
#endif
|
|
|
|
|
2017-03-16 00:44:15 +03:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
|
|
|
case BX_MSR_XSS:
|
2019-12-20 10:42:07 +03:00
|
|
|
{
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_XSAVES)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_XSS: XSAVES not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2020-01-03 22:33:16 +03:00
|
|
|
Bit32u xss_suport_mask = get_ia32_xss_allow_mask();
|
|
|
|
if (val_64 & ~Bit64u(xss_suport_mask)) {
|
2019-12-20 10:42:07 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to set reserved/not supported bit in BX_MSR_XSS"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-12-20 10:42:07 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.ia32_xss = val_64;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if BX_SUPPORT_CET
|
|
|
|
case BX_MSR_IA32_U_CET:
|
|
|
|
case BX_MSR_IA32_S_CET:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_U_CET/BX_MSR_IA32_S_CET: CET not enabled in the cpu model"));
|
2017-03-16 00:44:15 +03:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2019-12-20 10:42:07 +03:00
|
|
|
if (! IsCanonical(val_64) || is_invalid_cet_control(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical or invalid value to BX_MSR_IA32_U_CET/BX_MSR_IA32_S_CET !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-12-20 10:42:07 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.ia32_cet_control[index == BX_MSR_IA32_U_CET] = val_64;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_PL0_SSP:
|
|
|
|
case BX_MSR_IA32_PL1_SSP:
|
|
|
|
case BX_MSR_IA32_PL2_SSP:
|
|
|
|
case BX_MSR_IA32_PL3_SSP:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_PLi_SSP: CET not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_PLi_SSP !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-12-20 10:42:07 +03:00
|
|
|
}
|
|
|
|
if (val_64 & 0x03) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non 4byte-aligned address to BX_MSR_IA32_PLi_SSP !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-12-20 10:42:07 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.ia32_pl_ssp[index - BX_MSR_IA32_PL0_SSP] = val_64;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_INTERRUPT_SSP_TABLE_ADDR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_CET)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_INTERRUPT_SSP_TABLE_ADDR: CET not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_INTERRUPT_SSP_TABLE_ADDR !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-12-20 10:42:07 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.ia32_interrupt_ssp_table = val_64;
|
|
|
|
break;
|
2017-03-16 00:44:15 +03:00
|
|
|
#endif
|
|
|
|
|
2023-11-25 17:43:47 +03:00
|
|
|
#if BX_SUPPORT_UINTR
|
|
|
|
case BX_MSR_IA32_UINTR_RR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_RR: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.uirr = val_64;
|
|
|
|
uintr_uirr_update(); // potentially signal or clear user-level-interrupt
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_HANDLER:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_HANDLER: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_UINTR_HANDLER !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.ui_handler = val_64;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_STACKADJUST:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_STACKADJUST: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_UINTR_STACKADJUST !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.stack_adjust = val_64;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_MISC:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_MISC: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (val_64 & BX_CONST64(0xffffff0000000000)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write to reserved bits of BX_MSR_IA32_UINTR_MISC !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.uitt_size = GET32L(val_64);
|
|
|
|
BX_CPU_THIS_PTR uintr.uinv = GET32H(val_64);
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_PD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_PD: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_UINTR_PD !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
if (val_64 & 0x3f) { // bits [5:0] are reserved and MBZ
|
2024-01-11 09:10:42 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write to reserved bits of BX_MSR_IA32_UINTR_PD !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.upid_addr = val_64;
|
|
|
|
break;
|
|
|
|
case BX_MSR_IA32_UINTR_TT:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_UINTR)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IA32_UINTR_TT: UINTR not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
if (! IsCanonical(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to BX_MSR_IA32_UINTR_TT !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
if (val_64 & 0x0E) { // bits [3:1] are reserved and MBZ
|
|
|
|
BX_ERROR(("WRMSR: attempt to write to reserved bits of BX_MSR_IA32_UINTR_TT !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2023-11-25 17:43:47 +03:00
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR uintr.uitt_addr = val_64;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2020-05-29 15:35:30 +03:00
|
|
|
#if BX_SUPPORT_PKEYS
|
|
|
|
case BX_MSR_IA32_PKRS:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_PKS)) {
|
|
|
|
BX_ERROR(("WRMSR BX_MSR_IS32_PKS: Supervisor-Mode Protection Keys not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR pkrs = val_64;
|
|
|
|
set_PKeys(BX_CPU_THIS_PTR pkru, BX_CPU_THIS_PTR pkrs);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2011-11-21 16:51:50 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
|
|
|
case BX_MSR_TSC_DEADLINE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_TSC_DEADLINE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR BX_MSR_TSC_DEADLINE: TSC-Deadline not enabled in the cpu model"));
|
2011-11-21 16:51:50 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2023-11-28 12:08:42 +03:00
|
|
|
BX_CPU_THIS_PTR lapic->set_tsc_deadline(val_64);
|
2011-11-21 16:51:50 +04:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
// artificial MSR for MRSLIST serialization
|
|
|
|
case BX_MSR_IA32_BARRIER:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_MSRLIST)) {
|
|
|
|
BX_ERROR(("WRMSR IA32_BARRIER: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// SCA prevention MSRs
|
2019-10-26 23:09:30 +03:00
|
|
|
case BX_MSR_IA32_ARCH_CAPABILITIES:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
|
|
|
BX_ERROR(("WRMSR IA32_ARCH_CAPABILITIES: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
BX_ERROR(("WRMSR: IA32_ARCH_CAPABILITIES is read only MSR"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
|
|
|
|
case BX_MSR_IA32_SPEC_CTRL:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
|
|
|
BX_ERROR(("WRMSR IA32_SPEC_CTRL: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2024-01-27 19:30:22 +03:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
2024-01-27 16:50:51 +03:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest && vm->vmexec_ctrls3.VIRTUALIZE_IA32_SPEC_CTRL())
|
|
|
|
val_64 = (BX_CPU_THIS_PTR msr.ia32_spec_ctrl & vm->ia32_spec_ctrl_mask) | (val_64 & ~vm->ia32_spec_ctrl_mask);
|
2024-01-13 22:58:05 +03:00
|
|
|
#endif
|
2024-04-27 09:44:32 +03:00
|
|
|
if (! isValidMSR_IA32_SPEC_CTRL(val_64)) {
|
2019-10-26 23:09:30 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to set reserved bits of IA32_SPEC_CTRL !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
}
|
2024-01-13 22:58:05 +03:00
|
|
|
BX_CPU_THIS_PTR msr.ia32_spec_ctrl = GET32L(val_64);
|
2019-10-26 23:09:30 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_PRED_CMD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
|
|
|
BX_ERROR(("WRMSR IA32_PRED_CMD: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
// [0] - Indirect Branch Prediction Barrier (IBPB)
|
|
|
|
// [63:1] - reserved
|
|
|
|
if (val_64 & ~(BX_CONST64(0x1))) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to set reserved bits of IA32_PRED_CMD !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
}
|
|
|
|
// write only MSR, no need to remember written value
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_IA32_FLUSH_CMD:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SCA_MITIGATIONS)) {
|
|
|
|
BX_ERROR(("WRMSR IA32_FLUSH_CMD: not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
// [0] - WBINVD DL1 Cache
|
|
|
|
// [63:1] - reserved
|
|
|
|
if (val_64 & ~(BX_CONST64(0x1))) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to set reserved bits of IA32_FLUSH_CMD !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2019-10-26 23:09:30 +03:00
|
|
|
}
|
|
|
|
// write only MSR, no need to remember written value
|
|
|
|
break;
|
|
|
|
|
2009-01-31 13:43:24 +03:00
|
|
|
#if BX_SUPPORT_VMX
|
2010-03-06 19:59:05 +03:00
|
|
|
// Support only two bits: lock bit (bit 0) and VMX enable (bit 2)
|
|
|
|
case BX_MSR_IA32_FEATURE_CONTROL:
|
|
|
|
if (BX_CPU_THIS_PTR msr.ia32_feature_ctrl & 0x1) {
|
|
|
|
BX_ERROR(("WRMSR: IA32_FEATURE_CONTROL_MSR VMX lock bit is set !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2010-03-06 19:59:05 +03:00
|
|
|
}
|
2010-03-06 20:07:25 +03:00
|
|
|
/*
|
2010-03-06 19:59:05 +03:00
|
|
|
if (val_64 & ~((Bit64u)(BX_IA32_FEATURE_CONTROL_BITS))) {
|
|
|
|
BX_ERROR(("WRMSR: attempt to set reserved bits of IA32_FEATURE_CONTROL_MSR !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2010-03-06 19:59:05 +03:00
|
|
|
}
|
2010-03-06 20:07:25 +03:00
|
|
|
*/
|
2010-03-06 19:59:05 +03:00
|
|
|
BX_CPU_THIS_PTR msr.ia32_feature_ctrl = val32_lo;
|
2010-03-06 20:09:25 +03:00
|
|
|
break;
|
2010-03-06 19:59:05 +03:00
|
|
|
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_BASIC:
|
|
|
|
case BX_MSR_VMX_PINBASED_CTRLS:
|
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS:
|
2010-04-03 11:30:23 +04:00
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS2:
|
2023-12-16 22:59:34 +03:00
|
|
|
case BX_MSR_VMX_PROCBASED_CTRLS3:
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_VMEXIT_CTRLS:
|
2024-01-27 14:34:51 +03:00
|
|
|
case BX_MSR_VMX_VMEXIT_CTRLS2:
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_VMENTRY_CTRLS:
|
|
|
|
case BX_MSR_VMX_MISC:
|
|
|
|
case BX_MSR_VMX_CR0_FIXED0:
|
|
|
|
case BX_MSR_VMX_CR0_FIXED1:
|
|
|
|
case BX_MSR_VMX_CR4_FIXED0:
|
|
|
|
case BX_MSR_VMX_CR4_FIXED1:
|
|
|
|
case BX_MSR_VMX_VMCS_ENUM:
|
2011-09-27 00:36:26 +04:00
|
|
|
case BX_MSR_VMX_EPT_VPID_CAP:
|
2009-01-31 13:43:24 +03:00
|
|
|
case BX_MSR_VMX_TRUE_PINBASED_CTRLS:
|
|
|
|
case BX_MSR_VMX_TRUE_PROCBASED_CTRLS:
|
|
|
|
case BX_MSR_VMX_TRUE_VMEXIT_CTRLS:
|
|
|
|
case BX_MSR_VMX_TRUE_VMENTRY_CTRLS:
|
2011-11-05 11:31:51 +04:00
|
|
|
case BX_MSR_VMX_VMFUNC:
|
2009-01-31 13:43:24 +03:00
|
|
|
BX_ERROR(("WRMSR: VMX read only MSR"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-01-31 13:43:24 +03:00
|
|
|
#endif
|
|
|
|
|
2011-12-31 17:58:55 +04:00
|
|
|
#if BX_SUPPORT_SVM
|
2024-01-16 08:42:28 +03:00
|
|
|
case BX_SVM_VM_CR_MSR:
|
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SVM)) {
|
|
|
|
BX_ERROR(("WRMSR SVM_VM_CR_MSR: SVM support not enabled in the cpu model"));
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
|
|
|
Svm_Update_VM_CR_MSR(val_64);
|
|
|
|
break;
|
|
|
|
|
2011-12-31 17:58:55 +04:00
|
|
|
case BX_SVM_HSAVE_PA_MSR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_SVM)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR SVM_HSAVE_PA_MSR: SVM support not enabled in the cpu model"));
|
2011-12-31 17:58:55 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2012-12-27 01:59:16 +04:00
|
|
|
if (! IsValidPageAlignedPhyAddr(val_64)) {
|
2011-12-31 17:58:55 +04:00
|
|
|
BX_ERROR(("WRMSR SVM_HSAVE_PA_MSR: invalid or not page aligned physical address !"));
|
|
|
|
}
|
|
|
|
BX_CPU_THIS_PTR msr.svm_hsave_pa = val_64;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_EFER:
|
2023-12-16 20:07:03 +03:00
|
|
|
if (! SetEFER(val_64)) return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_STAR:
|
2011-09-25 21:36:20 +04:00
|
|
|
if ((BX_CPU_THIS_PTR efer_suppmask & BX_EFER_SCE_MASK) == 0) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_STAR: SYSCALL/SYSRET support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2015-07-13 23:24:14 +03:00
|
|
|
BX_CPU_THIS_PTR msr.star = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
2011-08-31 02:00:27 +04:00
|
|
|
#if BX_SUPPORT_X86_64
|
2008-12-05 15:48:36 +03:00
|
|
|
case BX_MSR_LSTAR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_LSTAR: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_LSTAR !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
BX_CPU_THIS_PTR msr.lstar = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_CSTAR:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_CSTAR: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_CSTAR !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
BX_CPU_THIS_PTR msr.cstar = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_FMASK:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_FMASK: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
BX_CPU_THIS_PTR msr.fmask = (Bit32u) val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_FSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_FSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_FSBASE !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
MSR_FSBASE = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_GSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_GSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_GSBASE !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
MSR_GSBASE = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_KERNELGSBASE:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_LONG_MODE)) {
|
2015-02-21 00:50:59 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_KERNELGSBASE: long mode support not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
if (! IsCanonical(val_64)) {
|
2008-12-05 15:48:36 +03:00
|
|
|
BX_ERROR(("WRMSR: attempt to write non-canonical value to MSR_KERNELGSBASE !"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2015-07-13 23:24:14 +03:00
|
|
|
BX_CPU_THIS_PTR msr.kernelgsbase = val_64;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BX_MSR_TSC_AUX:
|
2014-08-31 23:22:41 +04:00
|
|
|
if (! is_cpu_extension_supported(BX_ISA_RDTSCP)) {
|
2020-01-03 22:33:16 +03:00
|
|
|
BX_ERROR(("WRMSR MSR_TSC_AUX: RDTSCP feature not enabled in the cpu model"));
|
2011-09-25 21:36:20 +04:00
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
2011-08-21 23:09:35 +04:00
|
|
|
}
|
2015-02-19 23:23:08 +03:00
|
|
|
BX_CPU_THIS_PTR msr.tsc_aux = val32_lo;
|
2008-12-05 15:48:36 +03:00
|
|
|
break;
|
|
|
|
#endif // #if BX_SUPPORT_X86_64
|
|
|
|
|
2011-09-25 21:36:20 +04:00
|
|
|
default:
|
|
|
|
return handle_unknown_wrmsr(index, val_64);
|
|
|
|
}
|
2011-08-10 02:11:56 +04:00
|
|
|
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2011-09-25 21:36:20 +04:00
|
|
|
}
|
|
|
|
|
2021-01-30 11:35:35 +03:00
|
|
|
bool BX_CPP_AttrRegparmN(2) BX_CPU_C::handle_unknown_wrmsr(Bit32u index, Bit64u val_64)
|
2011-09-25 21:36:20 +04:00
|
|
|
{
|
|
|
|
// Try to check cpuid_t first (can implement some MSRs)
|
|
|
|
int result = BX_CPU_THIS_PTR cpuid->wrmsr(index, val_64);
|
|
|
|
if (result == 0)
|
2023-12-16 20:07:03 +03:00
|
|
|
return false; // #GP fault due to not supported MSR
|
2011-09-25 21:36:20 +04:00
|
|
|
|
|
|
|
if (result < 0) {
|
|
|
|
// cpuid_t have no idea about this MSR
|
2008-12-28 23:30:48 +03:00
|
|
|
#if BX_CONFIGURE_MSRS
|
2011-09-25 21:36:20 +04:00
|
|
|
if (index < BX_MSR_MAX_INDEX && BX_CPU_THIS_PTR msrs[index]) {
|
|
|
|
if (! BX_CPU_THIS_PTR msrs[index]->set64(val_64)) {
|
|
|
|
BX_ERROR(("WRMSR: Write failed to MSR %#x - #GP fault", index));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2011-08-10 02:11:56 +04:00
|
|
|
}
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2011-08-10 02:11:56 +04:00
|
|
|
}
|
2011-09-25 21:36:20 +04:00
|
|
|
#endif
|
|
|
|
// failed to find the MSR, could #GP or ignore it silently
|
|
|
|
BX_ERROR(("WRMSR: Unknown register %#x", index));
|
2022-07-31 18:47:13 +03:00
|
|
|
if (! BX_CPU_THIS_PTR ignore_bad_msrs)
|
2023-12-16 20:07:03 +03:00
|
|
|
return false; // will result in #GP fault due to unknown MSR
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2009-01-19 21:08:38 +03:00
|
|
|
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2009-01-19 21:08:38 +03:00
|
|
|
}
|
2011-09-25 21:36:20 +04:00
|
|
|
|
2009-01-19 21:08:38 +03:00
|
|
|
#endif // BX_CPU_LEVEL >= 5
|
|
|
|
|
2009-02-17 22:44:02 +03:00
|
|
|
#if BX_SUPPORT_APIC
|
2021-01-30 11:35:35 +03:00
|
|
|
bool BX_CPU_C::relocate_apic(Bit64u val_64)
|
2009-02-17 22:44:02 +03:00
|
|
|
{
|
|
|
|
/* MSR_APICBASE
|
2010-03-03 17:33:35 +03:00
|
|
|
* [0:7] Reserved
|
|
|
|
* [8] This is set if CPU is BSP
|
|
|
|
* [9] Reserved
|
|
|
|
* [10] X2APIC mode bit (1=enabled 0=disabled)
|
|
|
|
* [11] APIC Global Enable bit (1=enabled 0=disabled)
|
|
|
|
* [12:M] APIC Base Address (physical)
|
|
|
|
* [M:63] Reserved
|
2009-02-17 22:44:02 +03:00
|
|
|
*/
|
2009-02-19 01:25:04 +03:00
|
|
|
|
2022-07-27 23:20:47 +03:00
|
|
|
const Bit32u BX_MSR_APICBASE_RESERVED_BITS = (0x2ff | (is_cpu_extension_supported(BX_ISA_X2APIC) ? 0 : 0x400));
|
2009-02-19 01:25:04 +03:00
|
|
|
|
2023-11-21 16:43:38 +03:00
|
|
|
if (apic_global_enable_on()) {
|
2009-02-17 22:44:02 +03:00
|
|
|
Bit32u val32_hi = GET32H(val_64), val32_lo = GET32L(val_64);
|
|
|
|
BX_INFO(("WRMSR: wrote %08x:%08x to MSR_APICBASE", val32_hi, val32_lo));
|
2009-02-19 01:25:04 +03:00
|
|
|
if (! IsValidPhyAddr(val_64)) {
|
|
|
|
BX_ERROR(("relocate_apic: invalid physical address"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-02-17 22:44:02 +03:00
|
|
|
}
|
2009-02-19 01:25:04 +03:00
|
|
|
if (val32_lo & BX_MSR_APICBASE_RESERVED_BITS) {
|
|
|
|
BX_ERROR(("relocate_apic: attempt to set reserved bits"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2009-02-19 01:25:04 +03:00
|
|
|
}
|
2010-04-08 19:50:39 +04:00
|
|
|
|
2011-04-21 17:27:42 +04:00
|
|
|
#if BX_CPU_LEVEL >= 6
|
2014-08-31 23:22:41 +04:00
|
|
|
if (is_cpu_extension_supported(BX_ISA_X2APIC)) {
|
2011-04-21 17:27:42 +04:00
|
|
|
unsigned apic_state = (BX_CPU_THIS_PTR msr.apicbase >> 10) & 3;
|
|
|
|
unsigned new_state = (val32_lo >> 10) & 3;
|
2012-10-25 20:09:34 +04:00
|
|
|
|
|
|
|
if (new_state != apic_state) {
|
|
|
|
if (new_state == BX_APIC_STATE_INVALID) {
|
|
|
|
BX_ERROR(("relocate_apic: attempt to set invalid apic state"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2012-10-25 20:09:34 +04:00
|
|
|
}
|
|
|
|
if (apic_state == BX_APIC_X2APIC_MODE && new_state != BX_APIC_GLOBALLY_DISABLED) {
|
|
|
|
BX_ERROR(("relocate_apic: attempt to switch from x2apic -> xapic"));
|
2023-12-16 20:07:03 +03:00
|
|
|
return false;
|
2012-10-25 20:09:34 +04:00
|
|
|
}
|
2011-04-21 17:27:42 +04:00
|
|
|
}
|
2010-04-08 19:50:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-04-07 20:12:19 +04:00
|
|
|
BX_CPU_THIS_PTR msr.apicbase = (bx_phy_address) val_64;
|
2023-11-28 12:08:42 +03:00
|
|
|
BX_CPU_THIS_PTR lapic->set_base(BX_CPU_THIS_PTR msr.apicbase);
|
2009-02-17 22:44:02 +03:00
|
|
|
// TLB flush is required for emulation correctness
|
|
|
|
TLB_flush(); // don't care about performance of apic relocation
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
BX_INFO(("WRMSR: MSR_APICBASE APIC global enable bit cleared !"));
|
|
|
|
}
|
2009-02-19 01:25:04 +03:00
|
|
|
|
2023-12-16 20:07:03 +03:00
|
|
|
return true;
|
2009-02-17 22:44:02 +03:00
|
|
|
}
|
2023-11-21 16:43:38 +03:00
|
|
|
|
|
|
|
bool BX_CPU_C::apic_global_enable_on()
|
|
|
|
{
|
2023-12-16 20:07:03 +03:00
|
|
|
return (BX_CPU_THIS_PTR msr.apicbase & 0x800) != 0;
|
2023-11-21 16:43:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool BX_CPU_C::x2apic_mode()
|
|
|
|
{
|
2023-12-16 20:07:03 +03:00
|
|
|
return (BX_CPU_THIS_PTR msr.apicbase & 0x400) != 0;
|
2023-11-21 16:43:38 +03:00
|
|
|
}
|
2009-02-17 22:44:02 +03:00
|
|
|
#endif
|
|
|
|
|
2018-02-16 10:57:32 +03:00
|
|
|
void BX_CPP_AttrRegparmN(1) BX_CPU_C::WRMSR(bxInstruction_c *i)
|
2009-01-19 21:08:38 +03:00
|
|
|
{
|
|
|
|
#if BX_CPU_LEVEL >= 5
|
2011-08-24 01:25:34 +04:00
|
|
|
// CPL is always 0 in real mode
|
|
|
|
if (/* !real_mode() && */ CPL!=0) {
|
2009-01-19 21:08:38 +03:00
|
|
|
BX_ERROR(("WRMSR: CPL != 0 not in real mode"));
|
2010-03-14 18:51:27 +03:00
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
2009-01-19 21:08:38 +03:00
|
|
|
}
|
|
|
|
|
2012-10-26 22:43:53 +04:00
|
|
|
invalidate_prefetch_q();
|
|
|
|
|
2023-11-18 00:31:38 +03:00
|
|
|
Bit64u val_64 = GET64_FROM_HI32_LO32(EDX, EAX);
|
2009-01-19 21:08:38 +03:00
|
|
|
Bit32u index = ECX;
|
|
|
|
|
2011-12-26 23:57:39 +04:00
|
|
|
#if BX_SUPPORT_SVM
|
|
|
|
if (BX_CPU_THIS_PTR in_svm_guest) {
|
2011-12-27 23:42:11 +04:00
|
|
|
if (SVM_INTERCEPT(SVM_INTERCEPT0_MSR)) SvmInterceptMSR(BX_WRITE, index);
|
2011-12-26 23:57:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-01-31 13:43:24 +03:00
|
|
|
#if BX_SUPPORT_VMX
|
2011-08-10 00:50:51 +04:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest)
|
2012-07-26 20:03:26 +04:00
|
|
|
VMexit_MSR(VMX_VMEXIT_WRMSR, index);
|
2009-01-31 13:43:24 +03:00
|
|
|
#endif
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
if (! wrmsr(index, val_64))
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
BX_NEXT_TRACE(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if BX_SUPPORT_X86_64
|
|
|
|
|
|
|
|
#include "scalar_arith.h"
|
|
|
|
|
|
|
|
void BX_CPP_AttrRegparmN(1) BX_CPU_C::RDMSRLIST(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
#if BX_SUPPORT_VMX
|
2012-10-26 22:43:53 +04:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest) {
|
2024-01-27 16:50:51 +03:00
|
|
|
if (! BX_CPU_THIS_PTR vmcs.vmexec_ctrls3.ENABLE_MSRLIST())
|
2023-12-16 22:59:34 +03:00
|
|
|
exception(BX_UD_EXCEPTION, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!long64_mode() || CPL!=0) {
|
|
|
|
BX_ERROR(("RDMSRLIST: CPL != 0 cause #GP(0)"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((ESI | EDI) & 0x7) != 0) {
|
|
|
|
BX_ERROR(("RDMSRLIST: RSI and RDI must be 8-byte aligned"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Bit64u val64;
|
|
|
|
|
|
|
|
while (RCX != 0) {
|
|
|
|
unsigned MSR_index = tzcntq(RCX); // position of least significant bit set in RCX
|
|
|
|
Bit64u MSR_mask = (BX_CONST64(1) << MSR_index);
|
|
|
|
Bit64u MSR_address = read_linear_qword(BX_SEG_REG_DS, RSI + MSR_index*8);
|
|
|
|
if (GET32H(MSR_address)) {
|
|
|
|
BX_ERROR(("RDMSRLIST index=%d #GP(0): reserved bits are set in MSR address table entry", MSR_index));
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
2010-04-08 19:50:39 +04:00
|
|
|
}
|
2023-12-16 22:59:34 +03:00
|
|
|
|
|
|
|
#if BX_SUPPORT_VMX >= 2
|
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest)
|
|
|
|
VMexit_MSR(VMX_VMEXIT_RDMSRLIST, (Bit32u) MSR_address);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!rdmsr((Bit32u) MSR_address, &val64))
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
|
|
|
|
write_linear_qword(BX_SEG_REG_DS, RDI + MSR_index*8, val64);
|
|
|
|
|
|
|
|
RCX &= ~MSR_mask;
|
|
|
|
|
|
|
|
// allow delivery of any pending interrupts or traps
|
|
|
|
if (BX_CPU_THIS_PTR async_event) {
|
|
|
|
RIP = BX_CPU_THIS_PTR prev_rip; // loop not done, restore RIP
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_NEXT_TRACE(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BX_CPP_AttrRegparmN(1) BX_CPU_C::WRMSRLIST(bxInstruction_c *i)
|
|
|
|
{
|
|
|
|
#if BX_SUPPORT_VMX
|
2024-01-27 16:50:51 +03:00
|
|
|
VMCS_CACHE *vm = &BX_CPU_THIS_PTR vmcs;
|
2023-12-16 22:59:34 +03:00
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest) {
|
2024-01-27 16:50:51 +03:00
|
|
|
if (! vm->vmexec_ctrls3.ENABLE_MSRLIST())
|
2023-12-16 22:59:34 +03:00
|
|
|
exception(BX_UD_EXCEPTION, 0);
|
2010-04-08 19:50:39 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
if (!long64_mode() || CPL!=0) {
|
|
|
|
BX_ERROR(("WRMSRLIST: CPL != 0 cause #GP(0)"));
|
2010-03-14 18:51:27 +03:00
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
2023-12-16 22:59:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (((ESI | EDI) & 0x7) != 0) {
|
|
|
|
BX_ERROR(("WRMSRLIST: RSI and RDI must be 8-byte aligned"));
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidate_prefetch_q();
|
|
|
|
|
|
|
|
while (RCX != 0) {
|
|
|
|
unsigned MSR_index = tzcntq(RCX); // position of least significant bit set in RCX
|
|
|
|
Bit64u MSR_mask = (BX_CONST64(1) << MSR_index);
|
|
|
|
Bit64u MSR_address = read_linear_qword(BX_SEG_REG_DS, RSI + MSR_index*8);
|
|
|
|
if (GET32H(MSR_address)) {
|
|
|
|
BX_ERROR(("WRMSRLIST index=%d #GP(0): reserved bits are set in MSR address table entry", MSR_index));
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
}
|
|
|
|
|
2023-12-23 11:34:03 +03:00
|
|
|
Bit64u MSR_data = read_linear_qword(BX_SEG_REG_DS, RDI + MSR_index*8);
|
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
#if BX_SUPPORT_VMX >= 2
|
|
|
|
if (BX_CPU_THIS_PTR in_vmx_guest) {
|
|
|
|
vm->msr_data = MSR_data;
|
|
|
|
VMexit_MSR(VMX_VMEXIT_WRMSRLIST, (Bit32u) MSR_address);
|
|
|
|
}
|
2008-12-05 15:48:36 +03:00
|
|
|
#endif
|
2011-07-07 00:01:18 +04:00
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
if (! wrmsr((Bit32u) MSR_address, MSR_data))
|
|
|
|
exception(BX_GP_EXCEPTION, 0);
|
|
|
|
|
|
|
|
RCX &= ~MSR_mask;
|
|
|
|
|
|
|
|
// allow delivery of any pending interrupts or traps
|
|
|
|
if (BX_CPU_THIS_PTR async_event) {
|
|
|
|
RIP = BX_CPU_THIS_PTR prev_rip; // loop not done, restore RIP
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-25 20:49:22 +04:00
|
|
|
BX_NEXT_TRACE(i);
|
2008-12-05 15:48:36 +03:00
|
|
|
}
|
2008-12-28 23:30:48 +03:00
|
|
|
|
2023-12-16 22:59:34 +03:00
|
|
|
#endif
|
|
|
|
|
2008-12-28 23:30:48 +03:00
|
|
|
#if BX_CONFIGURE_MSRS
|
|
|
|
|
|
|
|
int BX_CPU_C::load_MSRs(const char *file)
|
|
|
|
{
|
|
|
|
char line[512];
|
|
|
|
unsigned linenum = 0;
|
|
|
|
Bit32u index, type;
|
|
|
|
Bit32u reset_hi, reset_lo;
|
|
|
|
Bit32u rsrv_hi, rsrv_lo;
|
|
|
|
Bit32u ignr_hi, ignr_lo;
|
|
|
|
|
|
|
|
FILE *fd = fopen (file, "r");
|
|
|
|
if (fd == NULL) return -1;
|
|
|
|
int retval = 0;
|
|
|
|
do {
|
|
|
|
linenum++;
|
|
|
|
char* ret = fgets(line, sizeof(line)-1, fd);
|
|
|
|
line[sizeof(line) - 1] = '\0';
|
|
|
|
size_t len = strlen(line);
|
|
|
|
if (len>0 && line[len-1] < ' ')
|
|
|
|
line[len-1] = '\0';
|
|
|
|
|
|
|
|
if (ret != NULL && strlen(line)) {
|
|
|
|
if (line[0] == '#') continue;
|
|
|
|
retval = sscanf(line, "%x %d %08x %08x %08x %08x %08x %08x",
|
|
|
|
&index, &type, &reset_hi, &reset_lo, &rsrv_hi, &rsrv_lo, &ignr_hi, &ignr_lo);
|
|
|
|
|
|
|
|
if (retval < 8) {
|
|
|
|
retval = -1;
|
|
|
|
BX_PANIC(("%s:%d > error parsing MSRs config file!", file, linenum));
|
|
|
|
break; // quit parsing after first error
|
|
|
|
}
|
2008-12-30 21:16:30 +03:00
|
|
|
if (index >= BX_MSR_MAX_INDEX) {
|
2008-12-28 23:30:48 +03:00
|
|
|
BX_PANIC(("%s:%d > MSR index is too big !", file, linenum));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (BX_CPU_THIS_PTR msrs[index]) {
|
|
|
|
BX_PANIC(("%s:%d > MSR[0x%03x] is already defined!", file, linenum, index));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (type > 2) {
|
|
|
|
BX_PANIC(("%s:%d > MSR[0x%03x] unknown type !", file, linenum, index));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_INFO(("loaded MSR[0x%03x] type=%d %08x:%08x %08x:%08x %08x:%08x", index, type,
|
|
|
|
reset_hi, reset_lo, rsrv_hi, rsrv_lo, ignr_hi, ignr_lo));
|
|
|
|
|
|
|
|
BX_CPU_THIS_PTR msrs[index] = new MSR(index, type,
|
2023-11-18 00:31:38 +03:00
|
|
|
GET64_FROM_HI32_LO32(reset_hi, reset_lo),
|
|
|
|
GET64_FROM_HI32_LO32(rsrv_hi, rsrv_lo),
|
|
|
|
GET64_FROM_HI32_LO32(ignr_hi, ignr_lo));
|
2008-12-28 23:30:48 +03:00
|
|
|
}
|
|
|
|
} while (!feof(fd));
|
|
|
|
|
|
|
|
fclose(fd);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|