mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-07 01:33:56 -06:00
target-arm queue:
* Support gdbstub (guest debug) in HVF * xnlx-versal: Support CANFD controller * bpim2u: New board model: Banana Pi BPI-M2 Ultra * Emulate FEAT_LSE2 * allow DC CVA[D]P in user mode emulation * trap DCC access in user mode emulation -----BEGIN PGP SIGNATURE----- iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmR/AKUZHHBldGVyLm1h eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3jzIEACNepQGY44yPhrEG+wD4WAB fH670KI33HcsFd2rGsC369gcssQbRIW/29reOzNhRMuol+kHI6OFaONpuKSdO0Rz TLVIsnT2Uq8KwbYfLtDQt5knj027amPy75d4re8wIK1eZB4dOIHysqAvQrJYeync 9obKku8xXGLwZh/mYHoVgHcZU0cPJO9nri39n1tV3JUBsgmqEURjzbZrMcF+yMX7 bUzOYQvC1Iedmo+aWfx43u82AlNQFz1lsqmnQj7Z5rvv0HT+BRF5WzVMP0qRh5+Z njkqmBH9xb9kkgeHmeMvHpWox+J+obeSmVg/4gDNlJpThmpuU0Vr7EXUN3MBQlV9 lhyy6zrTwC/BToiQqdT2dnpao9FzXy5exfnqi/py5IuqfjAzSO+p61LlPPZ4cJri pCK4yq2gzQXYfrlZkUJipvRMH8Xa4IdQx+w7lXrQoJdduF4/+6aJW/GAWSu0e7eC zgBwaJjI7ENce8ixJnuEFUxUnaBo8dl72a0PGA1UU8PL+cJNOIpyhPk4goWQprdn iFF4ZnjhBRZ2gk/4HGD9u5Vo2lNqP93YS5QhkGkF+HJsBmcOZgidIUpfHhPQvvHO Np196T2cAETCWGV1xG4CaTpxN2ndRReq3C0/mzfhIbwhXEACtvAiSlO4KB8t6pJj MzinCABXHcovJbGbxZ9j6w== =8SdN -----END PGP SIGNATURE----- Merge tag 'pull-target-arm-20230606' of https://git.linaro.org/people/pmaydell/qemu-arm into staging target-arm queue: * Support gdbstub (guest debug) in HVF * xnlx-versal: Support CANFD controller * bpim2u: New board model: Banana Pi BPI-M2 Ultra * Emulate FEAT_LSE2 * allow DC CVA[D]P in user mode emulation * trap DCC access in user mode emulation # -----BEGIN PGP SIGNATURE----- # # iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmR/AKUZHHBldGVyLm1h # eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3jzIEACNepQGY44yPhrEG+wD4WAB # fH670KI33HcsFd2rGsC369gcssQbRIW/29reOzNhRMuol+kHI6OFaONpuKSdO0Rz # TLVIsnT2Uq8KwbYfLtDQt5knj027amPy75d4re8wIK1eZB4dOIHysqAvQrJYeync # 9obKku8xXGLwZh/mYHoVgHcZU0cPJO9nri39n1tV3JUBsgmqEURjzbZrMcF+yMX7 # bUzOYQvC1Iedmo+aWfx43u82AlNQFz1lsqmnQj7Z5rvv0HT+BRF5WzVMP0qRh5+Z # njkqmBH9xb9kkgeHmeMvHpWox+J+obeSmVg/4gDNlJpThmpuU0Vr7EXUN3MBQlV9 # lhyy6zrTwC/BToiQqdT2dnpao9FzXy5exfnqi/py5IuqfjAzSO+p61LlPPZ4cJri # pCK4yq2gzQXYfrlZkUJipvRMH8Xa4IdQx+w7lXrQoJdduF4/+6aJW/GAWSu0e7eC # zgBwaJjI7ENce8ixJnuEFUxUnaBo8dl72a0PGA1UU8PL+cJNOIpyhPk4goWQprdn # iFF4ZnjhBRZ2gk/4HGD9u5Vo2lNqP93YS5QhkGkF+HJsBmcOZgidIUpfHhPQvvHO # Np196T2cAETCWGV1xG4CaTpxN2ndRReq3C0/mzfhIbwhXEACtvAiSlO4KB8t6pJj # MzinCABXHcovJbGbxZ9j6w== # =8SdN # -----END PGP SIGNATURE----- # gpg: Signature made Tue 06 Jun 2023 02:47:17 AM PDT # gpg: using RSA key E1A5C593CD419DE28E8315CF3C2525ED14360CDE # gpg: issuer "peter.maydell@linaro.org" # gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>" [full] # gpg: aka "Peter Maydell <pmaydell@gmail.com>" [full] # gpg: aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>" [full] # gpg: aka "Peter Maydell <peter@archaic.org.uk>" [unknown] * tag 'pull-target-arm-20230606' of https://git.linaro.org/people/pmaydell/qemu-arm: (42 commits) target/arm: trap DCC access in user mode emulation tests/tcg/aarch64: add DC CVA[D]P tests target/arm: allow DC CVA[D]P in user mode emulation target/arm: Enable FEAT_LSE2 for -cpu max tests/tcg/multiarch: Adjust sigbus.c tests/tcg/aarch64: Use stz2g in mte-7.c target/arm: Move mte check for store-exclusive target/arm: Relax ordered/atomic alignment checks for LSE2 target/arm: Add SCTLR.nAA to TBFLAG_A64 target/arm: Check alignment in helper_mte_check target/arm: Pass single_memop to gen_mte_checkN target/arm: Pass memop to gen_mte_check1* target/arm: Hoist finalize_memop out of do_fp_{ld, st} target/arm: Hoist finalize_memop out of do_gpr_{ld, st} target/arm: Load/store integer pair with one tcg operation target/arm: Sink gen_mte_check1 into load/store_exclusive target/arm: Use tcg_gen_qemu_{ld, st}_i128 in gen_sve_{ld, st}r target/arm: Use tcg_gen_qemu_st_i128 for STZG, STZ2G target/arm: Use tcg_gen_qemu_{st, ld}_i128 for do_fp_{st, ld} target/arm: Use tcg_gen_qemu_ld_i128 for LDXP ... Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
This commit is contained in:
commit
f5e6786de4
63 changed files with 7386 additions and 733 deletions
|
@ -31,6 +31,118 @@
|
|||
#include "trace/trace-target_arm_hvf.h"
|
||||
#include "migration/vmstate.h"
|
||||
|
||||
#include "exec/gdbstub.h"
|
||||
|
||||
#define MDSCR_EL1_SS_SHIFT 0
|
||||
#define MDSCR_EL1_MDE_SHIFT 15
|
||||
|
||||
static uint16_t dbgbcr_regs[] = {
|
||||
HV_SYS_REG_DBGBCR0_EL1,
|
||||
HV_SYS_REG_DBGBCR1_EL1,
|
||||
HV_SYS_REG_DBGBCR2_EL1,
|
||||
HV_SYS_REG_DBGBCR3_EL1,
|
||||
HV_SYS_REG_DBGBCR4_EL1,
|
||||
HV_SYS_REG_DBGBCR5_EL1,
|
||||
HV_SYS_REG_DBGBCR6_EL1,
|
||||
HV_SYS_REG_DBGBCR7_EL1,
|
||||
HV_SYS_REG_DBGBCR8_EL1,
|
||||
HV_SYS_REG_DBGBCR9_EL1,
|
||||
HV_SYS_REG_DBGBCR10_EL1,
|
||||
HV_SYS_REG_DBGBCR11_EL1,
|
||||
HV_SYS_REG_DBGBCR12_EL1,
|
||||
HV_SYS_REG_DBGBCR13_EL1,
|
||||
HV_SYS_REG_DBGBCR14_EL1,
|
||||
HV_SYS_REG_DBGBCR15_EL1,
|
||||
};
|
||||
static uint16_t dbgbvr_regs[] = {
|
||||
HV_SYS_REG_DBGBVR0_EL1,
|
||||
HV_SYS_REG_DBGBVR1_EL1,
|
||||
HV_SYS_REG_DBGBVR2_EL1,
|
||||
HV_SYS_REG_DBGBVR3_EL1,
|
||||
HV_SYS_REG_DBGBVR4_EL1,
|
||||
HV_SYS_REG_DBGBVR5_EL1,
|
||||
HV_SYS_REG_DBGBVR6_EL1,
|
||||
HV_SYS_REG_DBGBVR7_EL1,
|
||||
HV_SYS_REG_DBGBVR8_EL1,
|
||||
HV_SYS_REG_DBGBVR9_EL1,
|
||||
HV_SYS_REG_DBGBVR10_EL1,
|
||||
HV_SYS_REG_DBGBVR11_EL1,
|
||||
HV_SYS_REG_DBGBVR12_EL1,
|
||||
HV_SYS_REG_DBGBVR13_EL1,
|
||||
HV_SYS_REG_DBGBVR14_EL1,
|
||||
HV_SYS_REG_DBGBVR15_EL1,
|
||||
};
|
||||
static uint16_t dbgwcr_regs[] = {
|
||||
HV_SYS_REG_DBGWCR0_EL1,
|
||||
HV_SYS_REG_DBGWCR1_EL1,
|
||||
HV_SYS_REG_DBGWCR2_EL1,
|
||||
HV_SYS_REG_DBGWCR3_EL1,
|
||||
HV_SYS_REG_DBGWCR4_EL1,
|
||||
HV_SYS_REG_DBGWCR5_EL1,
|
||||
HV_SYS_REG_DBGWCR6_EL1,
|
||||
HV_SYS_REG_DBGWCR7_EL1,
|
||||
HV_SYS_REG_DBGWCR8_EL1,
|
||||
HV_SYS_REG_DBGWCR9_EL1,
|
||||
HV_SYS_REG_DBGWCR10_EL1,
|
||||
HV_SYS_REG_DBGWCR11_EL1,
|
||||
HV_SYS_REG_DBGWCR12_EL1,
|
||||
HV_SYS_REG_DBGWCR13_EL1,
|
||||
HV_SYS_REG_DBGWCR14_EL1,
|
||||
HV_SYS_REG_DBGWCR15_EL1,
|
||||
};
|
||||
static uint16_t dbgwvr_regs[] = {
|
||||
HV_SYS_REG_DBGWVR0_EL1,
|
||||
HV_SYS_REG_DBGWVR1_EL1,
|
||||
HV_SYS_REG_DBGWVR2_EL1,
|
||||
HV_SYS_REG_DBGWVR3_EL1,
|
||||
HV_SYS_REG_DBGWVR4_EL1,
|
||||
HV_SYS_REG_DBGWVR5_EL1,
|
||||
HV_SYS_REG_DBGWVR6_EL1,
|
||||
HV_SYS_REG_DBGWVR7_EL1,
|
||||
HV_SYS_REG_DBGWVR8_EL1,
|
||||
HV_SYS_REG_DBGWVR9_EL1,
|
||||
HV_SYS_REG_DBGWVR10_EL1,
|
||||
HV_SYS_REG_DBGWVR11_EL1,
|
||||
HV_SYS_REG_DBGWVR12_EL1,
|
||||
HV_SYS_REG_DBGWVR13_EL1,
|
||||
HV_SYS_REG_DBGWVR14_EL1,
|
||||
HV_SYS_REG_DBGWVR15_EL1,
|
||||
};
|
||||
|
||||
static inline int hvf_arm_num_brps(hv_vcpu_config_t config)
|
||||
{
|
||||
uint64_t val;
|
||||
hv_return_t ret;
|
||||
ret = hv_vcpu_config_get_feature_reg(config, HV_FEATURE_REG_ID_AA64DFR0_EL1,
|
||||
&val);
|
||||
assert_hvf_ok(ret);
|
||||
return FIELD_EX64(val, ID_AA64DFR0, BRPS) + 1;
|
||||
}
|
||||
|
||||
static inline int hvf_arm_num_wrps(hv_vcpu_config_t config)
|
||||
{
|
||||
uint64_t val;
|
||||
hv_return_t ret;
|
||||
ret = hv_vcpu_config_get_feature_reg(config, HV_FEATURE_REG_ID_AA64DFR0_EL1,
|
||||
&val);
|
||||
assert_hvf_ok(ret);
|
||||
return FIELD_EX64(val, ID_AA64DFR0, WRPS) + 1;
|
||||
}
|
||||
|
||||
void hvf_arm_init_debug(void)
|
||||
{
|
||||
hv_vcpu_config_t config;
|
||||
config = hv_vcpu_config_create();
|
||||
|
||||
max_hw_bps = hvf_arm_num_brps(config);
|
||||
hw_breakpoints =
|
||||
g_array_sized_new(true, true, sizeof(HWBreakpoint), max_hw_bps);
|
||||
|
||||
max_hw_wps = hvf_arm_num_wrps(config);
|
||||
hw_watchpoints =
|
||||
g_array_sized_new(true, true, sizeof(HWWatchpoint), max_hw_wps);
|
||||
}
|
||||
|
||||
#define HVF_SYSREG(crn, crm, op0, op1, op2) \
|
||||
ENCODE_AA64_CP_REG(CP_REG_ARM64_SYSREG_CP, crn, crm, op0, op1, op2)
|
||||
#define PL1_WRITE_MASK 0x4
|
||||
|
@ -107,6 +219,72 @@
|
|||
#define SYSREG_ICC_SGI1R_EL1 SYSREG(3, 0, 12, 11, 5)
|
||||
#define SYSREG_ICC_SRE_EL1 SYSREG(3, 0, 12, 12, 5)
|
||||
|
||||
#define SYSREG_MDSCR_EL1 SYSREG(2, 0, 0, 2, 2)
|
||||
#define SYSREG_DBGBVR0_EL1 SYSREG(2, 0, 0, 0, 4)
|
||||
#define SYSREG_DBGBCR0_EL1 SYSREG(2, 0, 0, 0, 5)
|
||||
#define SYSREG_DBGWVR0_EL1 SYSREG(2, 0, 0, 0, 6)
|
||||
#define SYSREG_DBGWCR0_EL1 SYSREG(2, 0, 0, 0, 7)
|
||||
#define SYSREG_DBGBVR1_EL1 SYSREG(2, 0, 0, 1, 4)
|
||||
#define SYSREG_DBGBCR1_EL1 SYSREG(2, 0, 0, 1, 5)
|
||||
#define SYSREG_DBGWVR1_EL1 SYSREG(2, 0, 0, 1, 6)
|
||||
#define SYSREG_DBGWCR1_EL1 SYSREG(2, 0, 0, 1, 7)
|
||||
#define SYSREG_DBGBVR2_EL1 SYSREG(2, 0, 0, 2, 4)
|
||||
#define SYSREG_DBGBCR2_EL1 SYSREG(2, 0, 0, 2, 5)
|
||||
#define SYSREG_DBGWVR2_EL1 SYSREG(2, 0, 0, 2, 6)
|
||||
#define SYSREG_DBGWCR2_EL1 SYSREG(2, 0, 0, 2, 7)
|
||||
#define SYSREG_DBGBVR3_EL1 SYSREG(2, 0, 0, 3, 4)
|
||||
#define SYSREG_DBGBCR3_EL1 SYSREG(2, 0, 0, 3, 5)
|
||||
#define SYSREG_DBGWVR3_EL1 SYSREG(2, 0, 0, 3, 6)
|
||||
#define SYSREG_DBGWCR3_EL1 SYSREG(2, 0, 0, 3, 7)
|
||||
#define SYSREG_DBGBVR4_EL1 SYSREG(2, 0, 0, 4, 4)
|
||||
#define SYSREG_DBGBCR4_EL1 SYSREG(2, 0, 0, 4, 5)
|
||||
#define SYSREG_DBGWVR4_EL1 SYSREG(2, 0, 0, 4, 6)
|
||||
#define SYSREG_DBGWCR4_EL1 SYSREG(2, 0, 0, 4, 7)
|
||||
#define SYSREG_DBGBVR5_EL1 SYSREG(2, 0, 0, 5, 4)
|
||||
#define SYSREG_DBGBCR5_EL1 SYSREG(2, 0, 0, 5, 5)
|
||||
#define SYSREG_DBGWVR5_EL1 SYSREG(2, 0, 0, 5, 6)
|
||||
#define SYSREG_DBGWCR5_EL1 SYSREG(2, 0, 0, 5, 7)
|
||||
#define SYSREG_DBGBVR6_EL1 SYSREG(2, 0, 0, 6, 4)
|
||||
#define SYSREG_DBGBCR6_EL1 SYSREG(2, 0, 0, 6, 5)
|
||||
#define SYSREG_DBGWVR6_EL1 SYSREG(2, 0, 0, 6, 6)
|
||||
#define SYSREG_DBGWCR6_EL1 SYSREG(2, 0, 0, 6, 7)
|
||||
#define SYSREG_DBGBVR7_EL1 SYSREG(2, 0, 0, 7, 4)
|
||||
#define SYSREG_DBGBCR7_EL1 SYSREG(2, 0, 0, 7, 5)
|
||||
#define SYSREG_DBGWVR7_EL1 SYSREG(2, 0, 0, 7, 6)
|
||||
#define SYSREG_DBGWCR7_EL1 SYSREG(2, 0, 0, 7, 7)
|
||||
#define SYSREG_DBGBVR8_EL1 SYSREG(2, 0, 0, 8, 4)
|
||||
#define SYSREG_DBGBCR8_EL1 SYSREG(2, 0, 0, 8, 5)
|
||||
#define SYSREG_DBGWVR8_EL1 SYSREG(2, 0, 0, 8, 6)
|
||||
#define SYSREG_DBGWCR8_EL1 SYSREG(2, 0, 0, 8, 7)
|
||||
#define SYSREG_DBGBVR9_EL1 SYSREG(2, 0, 0, 9, 4)
|
||||
#define SYSREG_DBGBCR9_EL1 SYSREG(2, 0, 0, 9, 5)
|
||||
#define SYSREG_DBGWVR9_EL1 SYSREG(2, 0, 0, 9, 6)
|
||||
#define SYSREG_DBGWCR9_EL1 SYSREG(2, 0, 0, 9, 7)
|
||||
#define SYSREG_DBGBVR10_EL1 SYSREG(2, 0, 0, 10, 4)
|
||||
#define SYSREG_DBGBCR10_EL1 SYSREG(2, 0, 0, 10, 5)
|
||||
#define SYSREG_DBGWVR10_EL1 SYSREG(2, 0, 0, 10, 6)
|
||||
#define SYSREG_DBGWCR10_EL1 SYSREG(2, 0, 0, 10, 7)
|
||||
#define SYSREG_DBGBVR11_EL1 SYSREG(2, 0, 0, 11, 4)
|
||||
#define SYSREG_DBGBCR11_EL1 SYSREG(2, 0, 0, 11, 5)
|
||||
#define SYSREG_DBGWVR11_EL1 SYSREG(2, 0, 0, 11, 6)
|
||||
#define SYSREG_DBGWCR11_EL1 SYSREG(2, 0, 0, 11, 7)
|
||||
#define SYSREG_DBGBVR12_EL1 SYSREG(2, 0, 0, 12, 4)
|
||||
#define SYSREG_DBGBCR12_EL1 SYSREG(2, 0, 0, 12, 5)
|
||||
#define SYSREG_DBGWVR12_EL1 SYSREG(2, 0, 0, 12, 6)
|
||||
#define SYSREG_DBGWCR12_EL1 SYSREG(2, 0, 0, 12, 7)
|
||||
#define SYSREG_DBGBVR13_EL1 SYSREG(2, 0, 0, 13, 4)
|
||||
#define SYSREG_DBGBCR13_EL1 SYSREG(2, 0, 0, 13, 5)
|
||||
#define SYSREG_DBGWVR13_EL1 SYSREG(2, 0, 0, 13, 6)
|
||||
#define SYSREG_DBGWCR13_EL1 SYSREG(2, 0, 0, 13, 7)
|
||||
#define SYSREG_DBGBVR14_EL1 SYSREG(2, 0, 0, 14, 4)
|
||||
#define SYSREG_DBGBCR14_EL1 SYSREG(2, 0, 0, 14, 5)
|
||||
#define SYSREG_DBGWVR14_EL1 SYSREG(2, 0, 0, 14, 6)
|
||||
#define SYSREG_DBGWCR14_EL1 SYSREG(2, 0, 0, 14, 7)
|
||||
#define SYSREG_DBGBVR15_EL1 SYSREG(2, 0, 0, 15, 4)
|
||||
#define SYSREG_DBGBCR15_EL1 SYSREG(2, 0, 0, 15, 5)
|
||||
#define SYSREG_DBGWVR15_EL1 SYSREG(2, 0, 0, 15, 6)
|
||||
#define SYSREG_DBGWCR15_EL1 SYSREG(2, 0, 0, 15, 7)
|
||||
|
||||
#define WFX_IS_WFE (1 << 0)
|
||||
|
||||
#define TMR_CTL_ENABLE (1 << 0)
|
||||
|
@ -397,6 +575,92 @@ int hvf_get_registers(CPUState *cpu)
|
|||
continue;
|
||||
}
|
||||
|
||||
if (cpu->hvf->guest_debug_enabled) {
|
||||
/* Handle debug registers */
|
||||
switch (hvf_sreg_match[i].reg) {
|
||||
case HV_SYS_REG_DBGBVR0_EL1:
|
||||
case HV_SYS_REG_DBGBCR0_EL1:
|
||||
case HV_SYS_REG_DBGWVR0_EL1:
|
||||
case HV_SYS_REG_DBGWCR0_EL1:
|
||||
case HV_SYS_REG_DBGBVR1_EL1:
|
||||
case HV_SYS_REG_DBGBCR1_EL1:
|
||||
case HV_SYS_REG_DBGWVR1_EL1:
|
||||
case HV_SYS_REG_DBGWCR1_EL1:
|
||||
case HV_SYS_REG_DBGBVR2_EL1:
|
||||
case HV_SYS_REG_DBGBCR2_EL1:
|
||||
case HV_SYS_REG_DBGWVR2_EL1:
|
||||
case HV_SYS_REG_DBGWCR2_EL1:
|
||||
case HV_SYS_REG_DBGBVR3_EL1:
|
||||
case HV_SYS_REG_DBGBCR3_EL1:
|
||||
case HV_SYS_REG_DBGWVR3_EL1:
|
||||
case HV_SYS_REG_DBGWCR3_EL1:
|
||||
case HV_SYS_REG_DBGBVR4_EL1:
|
||||
case HV_SYS_REG_DBGBCR4_EL1:
|
||||
case HV_SYS_REG_DBGWVR4_EL1:
|
||||
case HV_SYS_REG_DBGWCR4_EL1:
|
||||
case HV_SYS_REG_DBGBVR5_EL1:
|
||||
case HV_SYS_REG_DBGBCR5_EL1:
|
||||
case HV_SYS_REG_DBGWVR5_EL1:
|
||||
case HV_SYS_REG_DBGWCR5_EL1:
|
||||
case HV_SYS_REG_DBGBVR6_EL1:
|
||||
case HV_SYS_REG_DBGBCR6_EL1:
|
||||
case HV_SYS_REG_DBGWVR6_EL1:
|
||||
case HV_SYS_REG_DBGWCR6_EL1:
|
||||
case HV_SYS_REG_DBGBVR7_EL1:
|
||||
case HV_SYS_REG_DBGBCR7_EL1:
|
||||
case HV_SYS_REG_DBGWVR7_EL1:
|
||||
case HV_SYS_REG_DBGWCR7_EL1:
|
||||
case HV_SYS_REG_DBGBVR8_EL1:
|
||||
case HV_SYS_REG_DBGBCR8_EL1:
|
||||
case HV_SYS_REG_DBGWVR8_EL1:
|
||||
case HV_SYS_REG_DBGWCR8_EL1:
|
||||
case HV_SYS_REG_DBGBVR9_EL1:
|
||||
case HV_SYS_REG_DBGBCR9_EL1:
|
||||
case HV_SYS_REG_DBGWVR9_EL1:
|
||||
case HV_SYS_REG_DBGWCR9_EL1:
|
||||
case HV_SYS_REG_DBGBVR10_EL1:
|
||||
case HV_SYS_REG_DBGBCR10_EL1:
|
||||
case HV_SYS_REG_DBGWVR10_EL1:
|
||||
case HV_SYS_REG_DBGWCR10_EL1:
|
||||
case HV_SYS_REG_DBGBVR11_EL1:
|
||||
case HV_SYS_REG_DBGBCR11_EL1:
|
||||
case HV_SYS_REG_DBGWVR11_EL1:
|
||||
case HV_SYS_REG_DBGWCR11_EL1:
|
||||
case HV_SYS_REG_DBGBVR12_EL1:
|
||||
case HV_SYS_REG_DBGBCR12_EL1:
|
||||
case HV_SYS_REG_DBGWVR12_EL1:
|
||||
case HV_SYS_REG_DBGWCR12_EL1:
|
||||
case HV_SYS_REG_DBGBVR13_EL1:
|
||||
case HV_SYS_REG_DBGBCR13_EL1:
|
||||
case HV_SYS_REG_DBGWVR13_EL1:
|
||||
case HV_SYS_REG_DBGWCR13_EL1:
|
||||
case HV_SYS_REG_DBGBVR14_EL1:
|
||||
case HV_SYS_REG_DBGBCR14_EL1:
|
||||
case HV_SYS_REG_DBGWVR14_EL1:
|
||||
case HV_SYS_REG_DBGWCR14_EL1:
|
||||
case HV_SYS_REG_DBGBVR15_EL1:
|
||||
case HV_SYS_REG_DBGBCR15_EL1:
|
||||
case HV_SYS_REG_DBGWVR15_EL1:
|
||||
case HV_SYS_REG_DBGWCR15_EL1: {
|
||||
/*
|
||||
* If the guest is being debugged, the vCPU's debug registers
|
||||
* are holding the gdbstub's view of the registers (set in
|
||||
* hvf_arch_update_guest_debug()).
|
||||
* Since the environment is used to store only the guest's view
|
||||
* of the registers, don't update it with the values from the
|
||||
* vCPU but simply keep the values from the previous
|
||||
* environment.
|
||||
*/
|
||||
const ARMCPRegInfo *ri;
|
||||
ri = get_arm_cp_reginfo(arm_cpu->cp_regs, hvf_sreg_match[i].key);
|
||||
val = read_raw_cp_reg(env, ri);
|
||||
|
||||
arm_cpu->cpreg_values[hvf_sreg_match[i].cp_idx] = val;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ret = hv_vcpu_get_sys_reg(cpu->hvf->fd, hvf_sreg_match[i].reg, &val);
|
||||
assert_hvf_ok(ret);
|
||||
|
||||
|
@ -448,6 +712,82 @@ int hvf_put_registers(CPUState *cpu)
|
|||
continue;
|
||||
}
|
||||
|
||||
if (cpu->hvf->guest_debug_enabled) {
|
||||
/* Handle debug registers */
|
||||
switch (hvf_sreg_match[i].reg) {
|
||||
case HV_SYS_REG_DBGBVR0_EL1:
|
||||
case HV_SYS_REG_DBGBCR0_EL1:
|
||||
case HV_SYS_REG_DBGWVR0_EL1:
|
||||
case HV_SYS_REG_DBGWCR0_EL1:
|
||||
case HV_SYS_REG_DBGBVR1_EL1:
|
||||
case HV_SYS_REG_DBGBCR1_EL1:
|
||||
case HV_SYS_REG_DBGWVR1_EL1:
|
||||
case HV_SYS_REG_DBGWCR1_EL1:
|
||||
case HV_SYS_REG_DBGBVR2_EL1:
|
||||
case HV_SYS_REG_DBGBCR2_EL1:
|
||||
case HV_SYS_REG_DBGWVR2_EL1:
|
||||
case HV_SYS_REG_DBGWCR2_EL1:
|
||||
case HV_SYS_REG_DBGBVR3_EL1:
|
||||
case HV_SYS_REG_DBGBCR3_EL1:
|
||||
case HV_SYS_REG_DBGWVR3_EL1:
|
||||
case HV_SYS_REG_DBGWCR3_EL1:
|
||||
case HV_SYS_REG_DBGBVR4_EL1:
|
||||
case HV_SYS_REG_DBGBCR4_EL1:
|
||||
case HV_SYS_REG_DBGWVR4_EL1:
|
||||
case HV_SYS_REG_DBGWCR4_EL1:
|
||||
case HV_SYS_REG_DBGBVR5_EL1:
|
||||
case HV_SYS_REG_DBGBCR5_EL1:
|
||||
case HV_SYS_REG_DBGWVR5_EL1:
|
||||
case HV_SYS_REG_DBGWCR5_EL1:
|
||||
case HV_SYS_REG_DBGBVR6_EL1:
|
||||
case HV_SYS_REG_DBGBCR6_EL1:
|
||||
case HV_SYS_REG_DBGWVR6_EL1:
|
||||
case HV_SYS_REG_DBGWCR6_EL1:
|
||||
case HV_SYS_REG_DBGBVR7_EL1:
|
||||
case HV_SYS_REG_DBGBCR7_EL1:
|
||||
case HV_SYS_REG_DBGWVR7_EL1:
|
||||
case HV_SYS_REG_DBGWCR7_EL1:
|
||||
case HV_SYS_REG_DBGBVR8_EL1:
|
||||
case HV_SYS_REG_DBGBCR8_EL1:
|
||||
case HV_SYS_REG_DBGWVR8_EL1:
|
||||
case HV_SYS_REG_DBGWCR8_EL1:
|
||||
case HV_SYS_REG_DBGBVR9_EL1:
|
||||
case HV_SYS_REG_DBGBCR9_EL1:
|
||||
case HV_SYS_REG_DBGWVR9_EL1:
|
||||
case HV_SYS_REG_DBGWCR9_EL1:
|
||||
case HV_SYS_REG_DBGBVR10_EL1:
|
||||
case HV_SYS_REG_DBGBCR10_EL1:
|
||||
case HV_SYS_REG_DBGWVR10_EL1:
|
||||
case HV_SYS_REG_DBGWCR10_EL1:
|
||||
case HV_SYS_REG_DBGBVR11_EL1:
|
||||
case HV_SYS_REG_DBGBCR11_EL1:
|
||||
case HV_SYS_REG_DBGWVR11_EL1:
|
||||
case HV_SYS_REG_DBGWCR11_EL1:
|
||||
case HV_SYS_REG_DBGBVR12_EL1:
|
||||
case HV_SYS_REG_DBGBCR12_EL1:
|
||||
case HV_SYS_REG_DBGWVR12_EL1:
|
||||
case HV_SYS_REG_DBGWCR12_EL1:
|
||||
case HV_SYS_REG_DBGBVR13_EL1:
|
||||
case HV_SYS_REG_DBGBCR13_EL1:
|
||||
case HV_SYS_REG_DBGWVR13_EL1:
|
||||
case HV_SYS_REG_DBGWCR13_EL1:
|
||||
case HV_SYS_REG_DBGBVR14_EL1:
|
||||
case HV_SYS_REG_DBGBCR14_EL1:
|
||||
case HV_SYS_REG_DBGWVR14_EL1:
|
||||
case HV_SYS_REG_DBGWCR14_EL1:
|
||||
case HV_SYS_REG_DBGBVR15_EL1:
|
||||
case HV_SYS_REG_DBGBCR15_EL1:
|
||||
case HV_SYS_REG_DBGWVR15_EL1:
|
||||
case HV_SYS_REG_DBGWCR15_EL1:
|
||||
/*
|
||||
* If the guest is being debugged, the vCPU's debug registers
|
||||
* are already holding the gdbstub's view of the registers (set
|
||||
* in hvf_arch_update_guest_debug()).
|
||||
*/
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
val = arm_cpu->cpreg_values[hvf_sreg_match[i].cp_idx];
|
||||
ret = hv_vcpu_set_sys_reg(cpu->hvf->fd, hvf_sreg_match[i].reg, val);
|
||||
assert_hvf_ok(ret);
|
||||
|
@ -933,6 +1273,78 @@ static int hvf_sysreg_read(CPUState *cpu, uint32_t reg, uint32_t rt)
|
|||
hvf_raise_exception(cpu, EXCP_UDEF, syn_uncategorized());
|
||||
}
|
||||
break;
|
||||
case SYSREG_DBGBVR0_EL1:
|
||||
case SYSREG_DBGBVR1_EL1:
|
||||
case SYSREG_DBGBVR2_EL1:
|
||||
case SYSREG_DBGBVR3_EL1:
|
||||
case SYSREG_DBGBVR4_EL1:
|
||||
case SYSREG_DBGBVR5_EL1:
|
||||
case SYSREG_DBGBVR6_EL1:
|
||||
case SYSREG_DBGBVR7_EL1:
|
||||
case SYSREG_DBGBVR8_EL1:
|
||||
case SYSREG_DBGBVR9_EL1:
|
||||
case SYSREG_DBGBVR10_EL1:
|
||||
case SYSREG_DBGBVR11_EL1:
|
||||
case SYSREG_DBGBVR12_EL1:
|
||||
case SYSREG_DBGBVR13_EL1:
|
||||
case SYSREG_DBGBVR14_EL1:
|
||||
case SYSREG_DBGBVR15_EL1:
|
||||
val = env->cp15.dbgbvr[SYSREG_CRM(reg)];
|
||||
break;
|
||||
case SYSREG_DBGBCR0_EL1:
|
||||
case SYSREG_DBGBCR1_EL1:
|
||||
case SYSREG_DBGBCR2_EL1:
|
||||
case SYSREG_DBGBCR3_EL1:
|
||||
case SYSREG_DBGBCR4_EL1:
|
||||
case SYSREG_DBGBCR5_EL1:
|
||||
case SYSREG_DBGBCR6_EL1:
|
||||
case SYSREG_DBGBCR7_EL1:
|
||||
case SYSREG_DBGBCR8_EL1:
|
||||
case SYSREG_DBGBCR9_EL1:
|
||||
case SYSREG_DBGBCR10_EL1:
|
||||
case SYSREG_DBGBCR11_EL1:
|
||||
case SYSREG_DBGBCR12_EL1:
|
||||
case SYSREG_DBGBCR13_EL1:
|
||||
case SYSREG_DBGBCR14_EL1:
|
||||
case SYSREG_DBGBCR15_EL1:
|
||||
val = env->cp15.dbgbcr[SYSREG_CRM(reg)];
|
||||
break;
|
||||
case SYSREG_DBGWVR0_EL1:
|
||||
case SYSREG_DBGWVR1_EL1:
|
||||
case SYSREG_DBGWVR2_EL1:
|
||||
case SYSREG_DBGWVR3_EL1:
|
||||
case SYSREG_DBGWVR4_EL1:
|
||||
case SYSREG_DBGWVR5_EL1:
|
||||
case SYSREG_DBGWVR6_EL1:
|
||||
case SYSREG_DBGWVR7_EL1:
|
||||
case SYSREG_DBGWVR8_EL1:
|
||||
case SYSREG_DBGWVR9_EL1:
|
||||
case SYSREG_DBGWVR10_EL1:
|
||||
case SYSREG_DBGWVR11_EL1:
|
||||
case SYSREG_DBGWVR12_EL1:
|
||||
case SYSREG_DBGWVR13_EL1:
|
||||
case SYSREG_DBGWVR14_EL1:
|
||||
case SYSREG_DBGWVR15_EL1:
|
||||
val = env->cp15.dbgwvr[SYSREG_CRM(reg)];
|
||||
break;
|
||||
case SYSREG_DBGWCR0_EL1:
|
||||
case SYSREG_DBGWCR1_EL1:
|
||||
case SYSREG_DBGWCR2_EL1:
|
||||
case SYSREG_DBGWCR3_EL1:
|
||||
case SYSREG_DBGWCR4_EL1:
|
||||
case SYSREG_DBGWCR5_EL1:
|
||||
case SYSREG_DBGWCR6_EL1:
|
||||
case SYSREG_DBGWCR7_EL1:
|
||||
case SYSREG_DBGWCR8_EL1:
|
||||
case SYSREG_DBGWCR9_EL1:
|
||||
case SYSREG_DBGWCR10_EL1:
|
||||
case SYSREG_DBGWCR11_EL1:
|
||||
case SYSREG_DBGWCR12_EL1:
|
||||
case SYSREG_DBGWCR13_EL1:
|
||||
case SYSREG_DBGWCR14_EL1:
|
||||
case SYSREG_DBGWCR15_EL1:
|
||||
val = env->cp15.dbgwcr[SYSREG_CRM(reg)];
|
||||
break;
|
||||
default:
|
||||
if (is_id_sysreg(reg)) {
|
||||
/* ID system registers read as RES0 */
|
||||
|
@ -1172,6 +1584,81 @@ static int hvf_sysreg_write(CPUState *cpu, uint32_t reg, uint64_t val)
|
|||
hvf_raise_exception(cpu, EXCP_UDEF, syn_uncategorized());
|
||||
}
|
||||
break;
|
||||
case SYSREG_MDSCR_EL1:
|
||||
env->cp15.mdscr_el1 = val;
|
||||
break;
|
||||
case SYSREG_DBGBVR0_EL1:
|
||||
case SYSREG_DBGBVR1_EL1:
|
||||
case SYSREG_DBGBVR2_EL1:
|
||||
case SYSREG_DBGBVR3_EL1:
|
||||
case SYSREG_DBGBVR4_EL1:
|
||||
case SYSREG_DBGBVR5_EL1:
|
||||
case SYSREG_DBGBVR6_EL1:
|
||||
case SYSREG_DBGBVR7_EL1:
|
||||
case SYSREG_DBGBVR8_EL1:
|
||||
case SYSREG_DBGBVR9_EL1:
|
||||
case SYSREG_DBGBVR10_EL1:
|
||||
case SYSREG_DBGBVR11_EL1:
|
||||
case SYSREG_DBGBVR12_EL1:
|
||||
case SYSREG_DBGBVR13_EL1:
|
||||
case SYSREG_DBGBVR14_EL1:
|
||||
case SYSREG_DBGBVR15_EL1:
|
||||
env->cp15.dbgbvr[SYSREG_CRM(reg)] = val;
|
||||
break;
|
||||
case SYSREG_DBGBCR0_EL1:
|
||||
case SYSREG_DBGBCR1_EL1:
|
||||
case SYSREG_DBGBCR2_EL1:
|
||||
case SYSREG_DBGBCR3_EL1:
|
||||
case SYSREG_DBGBCR4_EL1:
|
||||
case SYSREG_DBGBCR5_EL1:
|
||||
case SYSREG_DBGBCR6_EL1:
|
||||
case SYSREG_DBGBCR7_EL1:
|
||||
case SYSREG_DBGBCR8_EL1:
|
||||
case SYSREG_DBGBCR9_EL1:
|
||||
case SYSREG_DBGBCR10_EL1:
|
||||
case SYSREG_DBGBCR11_EL1:
|
||||
case SYSREG_DBGBCR12_EL1:
|
||||
case SYSREG_DBGBCR13_EL1:
|
||||
case SYSREG_DBGBCR14_EL1:
|
||||
case SYSREG_DBGBCR15_EL1:
|
||||
env->cp15.dbgbcr[SYSREG_CRM(reg)] = val;
|
||||
break;
|
||||
case SYSREG_DBGWVR0_EL1:
|
||||
case SYSREG_DBGWVR1_EL1:
|
||||
case SYSREG_DBGWVR2_EL1:
|
||||
case SYSREG_DBGWVR3_EL1:
|
||||
case SYSREG_DBGWVR4_EL1:
|
||||
case SYSREG_DBGWVR5_EL1:
|
||||
case SYSREG_DBGWVR6_EL1:
|
||||
case SYSREG_DBGWVR7_EL1:
|
||||
case SYSREG_DBGWVR8_EL1:
|
||||
case SYSREG_DBGWVR9_EL1:
|
||||
case SYSREG_DBGWVR10_EL1:
|
||||
case SYSREG_DBGWVR11_EL1:
|
||||
case SYSREG_DBGWVR12_EL1:
|
||||
case SYSREG_DBGWVR13_EL1:
|
||||
case SYSREG_DBGWVR14_EL1:
|
||||
case SYSREG_DBGWVR15_EL1:
|
||||
env->cp15.dbgwvr[SYSREG_CRM(reg)] = val;
|
||||
break;
|
||||
case SYSREG_DBGWCR0_EL1:
|
||||
case SYSREG_DBGWCR1_EL1:
|
||||
case SYSREG_DBGWCR2_EL1:
|
||||
case SYSREG_DBGWCR3_EL1:
|
||||
case SYSREG_DBGWCR4_EL1:
|
||||
case SYSREG_DBGWCR5_EL1:
|
||||
case SYSREG_DBGWCR6_EL1:
|
||||
case SYSREG_DBGWCR7_EL1:
|
||||
case SYSREG_DBGWCR8_EL1:
|
||||
case SYSREG_DBGWCR9_EL1:
|
||||
case SYSREG_DBGWCR10_EL1:
|
||||
case SYSREG_DBGWCR11_EL1:
|
||||
case SYSREG_DBGWCR12_EL1:
|
||||
case SYSREG_DBGWCR13_EL1:
|
||||
case SYSREG_DBGWCR14_EL1:
|
||||
case SYSREG_DBGWCR15_EL1:
|
||||
env->cp15.dbgwcr[SYSREG_CRM(reg)] = val;
|
||||
break;
|
||||
default:
|
||||
cpu_synchronize_state(cpu);
|
||||
trace_hvf_unhandled_sysreg_write(env->pc, reg,
|
||||
|
@ -1317,11 +1804,13 @@ int hvf_vcpu_exec(CPUState *cpu)
|
|||
{
|
||||
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
||||
CPUARMState *env = &arm_cpu->env;
|
||||
int ret;
|
||||
hv_vcpu_exit_t *hvf_exit = cpu->hvf->exit;
|
||||
hv_return_t r;
|
||||
bool advance_pc = false;
|
||||
|
||||
if (hvf_inject_interrupts(cpu)) {
|
||||
if (!(cpu->singlestep_enabled & SSTEP_NOIRQ) &&
|
||||
hvf_inject_interrupts(cpu)) {
|
||||
return EXCP_INTERRUPT;
|
||||
}
|
||||
|
||||
|
@ -1339,6 +1828,7 @@ int hvf_vcpu_exec(CPUState *cpu)
|
|||
uint64_t syndrome = hvf_exit->exception.syndrome;
|
||||
uint32_t ec = syn_get_ec(syndrome);
|
||||
|
||||
ret = 0;
|
||||
qemu_mutex_lock_iothread();
|
||||
switch (exit_reason) {
|
||||
case HV_EXIT_REASON_EXCEPTION:
|
||||
|
@ -1358,6 +1848,49 @@ int hvf_vcpu_exec(CPUState *cpu)
|
|||
hvf_sync_vtimer(cpu);
|
||||
|
||||
switch (ec) {
|
||||
case EC_SOFTWARESTEP: {
|
||||
ret = EXCP_DEBUG;
|
||||
|
||||
if (!cpu->singlestep_enabled) {
|
||||
error_report("EC_SOFTWARESTEP but single-stepping not enabled");
|
||||
}
|
||||
break;
|
||||
}
|
||||
case EC_AA64_BKPT: {
|
||||
ret = EXCP_DEBUG;
|
||||
|
||||
cpu_synchronize_state(cpu);
|
||||
|
||||
if (!hvf_find_sw_breakpoint(cpu, env->pc)) {
|
||||
/* Re-inject into the guest */
|
||||
ret = 0;
|
||||
hvf_raise_exception(cpu, EXCP_BKPT, syn_aa64_bkpt(0));
|
||||
}
|
||||
break;
|
||||
}
|
||||
case EC_BREAKPOINT: {
|
||||
ret = EXCP_DEBUG;
|
||||
|
||||
cpu_synchronize_state(cpu);
|
||||
|
||||
if (!find_hw_breakpoint(cpu, env->pc)) {
|
||||
error_report("EC_BREAKPOINT but unknown hw breakpoint");
|
||||
}
|
||||
break;
|
||||
}
|
||||
case EC_WATCHPOINT: {
|
||||
ret = EXCP_DEBUG;
|
||||
|
||||
cpu_synchronize_state(cpu);
|
||||
|
||||
CPUWatchpoint *wp =
|
||||
find_hw_watchpoint(cpu, hvf_exit->exception.virtual_address);
|
||||
if (!wp) {
|
||||
error_report("EXCP_DEBUG but unknown hw watchpoint");
|
||||
}
|
||||
cpu->watchpoint_hit = wp;
|
||||
break;
|
||||
}
|
||||
case EC_DATAABORT: {
|
||||
bool isv = syndrome & ARM_EL_ISV;
|
||||
bool iswrite = (syndrome >> 6) & 1;
|
||||
|
@ -1462,9 +1995,14 @@ int hvf_vcpu_exec(CPUState *cpu)
|
|||
pc += 4;
|
||||
r = hv_vcpu_set_reg(cpu->hvf->fd, HV_REG_PC, pc);
|
||||
assert_hvf_ok(r);
|
||||
|
||||
/* Handle single-stepping over instructions which trigger a VM exit */
|
||||
if (cpu->singlestep_enabled) {
|
||||
ret = EXCP_DEBUG;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_hvf_vtimer = {
|
||||
|
@ -1496,5 +2034,213 @@ int hvf_arch_init(void)
|
|||
hvf_state->vtimer_offset = mach_absolute_time();
|
||||
vmstate_register(NULL, 0, &vmstate_hvf_vtimer, &vtimer);
|
||||
qemu_add_vm_change_state_handler(hvf_vm_state_change, &vtimer);
|
||||
|
||||
hvf_arm_init_debug();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const uint32_t brk_insn = 0xd4200000;
|
||||
|
||||
int hvf_arch_insert_sw_breakpoint(CPUState *cpu, struct hvf_sw_breakpoint *bp)
|
||||
{
|
||||
if (cpu_memory_rw_debug(cpu, bp->pc, (uint8_t *)&bp->saved_insn, 4, 0) ||
|
||||
cpu_memory_rw_debug(cpu, bp->pc, (uint8_t *)&brk_insn, 4, 1)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int hvf_arch_remove_sw_breakpoint(CPUState *cpu, struct hvf_sw_breakpoint *bp)
|
||||
{
|
||||
static uint32_t brk;
|
||||
|
||||
if (cpu_memory_rw_debug(cpu, bp->pc, (uint8_t *)&brk, 4, 0) ||
|
||||
brk != brk_insn ||
|
||||
cpu_memory_rw_debug(cpu, bp->pc, (uint8_t *)&bp->saved_insn, 4, 1)) {
|
||||
return -EINVAL;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int hvf_arch_insert_hw_breakpoint(target_ulong addr, target_ulong len, int type)
|
||||
{
|
||||
switch (type) {
|
||||
case GDB_BREAKPOINT_HW:
|
||||
return insert_hw_breakpoint(addr);
|
||||
case GDB_WATCHPOINT_READ:
|
||||
case GDB_WATCHPOINT_WRITE:
|
||||
case GDB_WATCHPOINT_ACCESS:
|
||||
return insert_hw_watchpoint(addr, len, type);
|
||||
default:
|
||||
return -ENOSYS;
|
||||
}
|
||||
}
|
||||
|
||||
int hvf_arch_remove_hw_breakpoint(target_ulong addr, target_ulong len, int type)
|
||||
{
|
||||
switch (type) {
|
||||
case GDB_BREAKPOINT_HW:
|
||||
return delete_hw_breakpoint(addr);
|
||||
case GDB_WATCHPOINT_READ:
|
||||
case GDB_WATCHPOINT_WRITE:
|
||||
case GDB_WATCHPOINT_ACCESS:
|
||||
return delete_hw_watchpoint(addr, len, type);
|
||||
default:
|
||||
return -ENOSYS;
|
||||
}
|
||||
}
|
||||
|
||||
void hvf_arch_remove_all_hw_breakpoints(void)
|
||||
{
|
||||
if (cur_hw_wps > 0) {
|
||||
g_array_remove_range(hw_watchpoints, 0, cur_hw_wps);
|
||||
}
|
||||
if (cur_hw_bps > 0) {
|
||||
g_array_remove_range(hw_breakpoints, 0, cur_hw_bps);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Update the vCPU with the gdbstub's view of debug registers. This view
|
||||
* consists of all hardware breakpoints and watchpoints inserted so far while
|
||||
* debugging the guest.
|
||||
*/
|
||||
static void hvf_put_gdbstub_debug_registers(CPUState *cpu)
|
||||
{
|
||||
hv_return_t r = HV_SUCCESS;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < cur_hw_bps; i++) {
|
||||
HWBreakpoint *bp = get_hw_bp(i);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbcr_regs[i], bp->bcr);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbvr_regs[i], bp->bvr);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
for (i = cur_hw_bps; i < max_hw_bps; i++) {
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbcr_regs[i], 0);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbvr_regs[i], 0);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
|
||||
for (i = 0; i < cur_hw_wps; i++) {
|
||||
HWWatchpoint *wp = get_hw_wp(i);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwcr_regs[i], wp->wcr);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwvr_regs[i], wp->wvr);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
for (i = cur_hw_wps; i < max_hw_wps; i++) {
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwcr_regs[i], 0);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwvr_regs[i], 0);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Update the vCPU with the guest's view of debug registers. This view is kept
|
||||
* in the environment at all times.
|
||||
*/
|
||||
static void hvf_put_guest_debug_registers(CPUState *cpu)
|
||||
{
|
||||
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
||||
CPUARMState *env = &arm_cpu->env;
|
||||
hv_return_t r = HV_SUCCESS;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < max_hw_bps; i++) {
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbcr_regs[i],
|
||||
env->cp15.dbgbcr[i]);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgbvr_regs[i],
|
||||
env->cp15.dbgbvr[i]);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
|
||||
for (i = 0; i < max_hw_wps; i++) {
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwcr_regs[i],
|
||||
env->cp15.dbgwcr[i]);
|
||||
assert_hvf_ok(r);
|
||||
r = hv_vcpu_set_sys_reg(cpu->hvf->fd, dbgwvr_regs[i],
|
||||
env->cp15.dbgwvr[i]);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
}
|
||||
|
||||
static inline bool hvf_arm_hw_debug_active(CPUState *cpu)
|
||||
{
|
||||
return ((cur_hw_wps > 0) || (cur_hw_bps > 0));
|
||||
}
|
||||
|
||||
static void hvf_arch_set_traps(void)
|
||||
{
|
||||
CPUState *cpu;
|
||||
bool should_enable_traps = false;
|
||||
hv_return_t r = HV_SUCCESS;
|
||||
|
||||
/* Check whether guest debugging is enabled for at least one vCPU; if it
|
||||
* is, enable exiting the guest on all vCPUs */
|
||||
CPU_FOREACH(cpu) {
|
||||
should_enable_traps |= cpu->hvf->guest_debug_enabled;
|
||||
}
|
||||
CPU_FOREACH(cpu) {
|
||||
/* Set whether debug exceptions exit the guest */
|
||||
r = hv_vcpu_set_trap_debug_exceptions(cpu->hvf->fd,
|
||||
should_enable_traps);
|
||||
assert_hvf_ok(r);
|
||||
|
||||
/* Set whether accesses to debug registers exit the guest */
|
||||
r = hv_vcpu_set_trap_debug_reg_accesses(cpu->hvf->fd,
|
||||
should_enable_traps);
|
||||
assert_hvf_ok(r);
|
||||
}
|
||||
}
|
||||
|
||||
void hvf_arch_update_guest_debug(CPUState *cpu)
|
||||
{
|
||||
ARMCPU *arm_cpu = ARM_CPU(cpu);
|
||||
CPUARMState *env = &arm_cpu->env;
|
||||
|
||||
/* Check whether guest debugging is enabled */
|
||||
cpu->hvf->guest_debug_enabled = cpu->singlestep_enabled ||
|
||||
hvf_sw_breakpoints_active(cpu) ||
|
||||
hvf_arm_hw_debug_active(cpu);
|
||||
|
||||
/* Update debug registers */
|
||||
if (cpu->hvf->guest_debug_enabled) {
|
||||
hvf_put_gdbstub_debug_registers(cpu);
|
||||
} else {
|
||||
hvf_put_guest_debug_registers(cpu);
|
||||
}
|
||||
|
||||
cpu_synchronize_state(cpu);
|
||||
|
||||
/* Enable/disable single-stepping */
|
||||
if (cpu->singlestep_enabled) {
|
||||
env->cp15.mdscr_el1 =
|
||||
deposit64(env->cp15.mdscr_el1, MDSCR_EL1_SS_SHIFT, 1, 1);
|
||||
pstate_write(env, pstate_read(env) | PSTATE_SS);
|
||||
} else {
|
||||
env->cp15.mdscr_el1 =
|
||||
deposit64(env->cp15.mdscr_el1, MDSCR_EL1_SS_SHIFT, 1, 0);
|
||||
}
|
||||
|
||||
/* Enable/disable Breakpoint exceptions */
|
||||
if (hvf_arm_hw_debug_active(cpu)) {
|
||||
env->cp15.mdscr_el1 =
|
||||
deposit64(env->cp15.mdscr_el1, MDSCR_EL1_MDE_SHIFT, 1, 1);
|
||||
} else {
|
||||
env->cp15.mdscr_el1 =
|
||||
deposit64(env->cp15.mdscr_el1, MDSCR_EL1_MDE_SHIFT, 1, 0);
|
||||
}
|
||||
|
||||
hvf_arch_set_traps();
|
||||
}
|
||||
|
||||
inline bool hvf_arch_supports_guest_debug(void)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue