CGT_HCR_TPU_TOCU,
        CGT_HCR_NV1_nNV2_ENSCXT,
        CGT_MDCR_TPM_TPMCR,
+       CGT_MDCR_TPM_HPMN,
        CGT_MDCR_TDE_TDA,
        CGT_MDCR_TDE_TDOSA,
        CGT_MDCR_TDE_TDRA,
        CGT_CNTHCTL_EL1PTEN,
 
        CGT_CPTR_TTA,
+       CGT_MDCR_HPMN,
 
        /* Must be last */
        __NR_CGT_GROUP_IDS__
        MCB(CGT_HCR_TPU_TOCU,           CGT_HCR_TPU, CGT_HCR_TOCU),
        MCB(CGT_HCR_NV1_nNV2_ENSCXT,    CGT_HCR_NV1_nNV2, CGT_HCR_ENSCXT),
        MCB(CGT_MDCR_TPM_TPMCR,         CGT_MDCR_TPM, CGT_MDCR_TPMCR),
+       MCB(CGT_MDCR_TPM_HPMN,          CGT_MDCR_TPM, CGT_MDCR_HPMN),
        MCB(CGT_MDCR_TDE_TDA,           CGT_MDCR_TDE, CGT_MDCR_TDA),
        MCB(CGT_MDCR_TDE_TDOSA,         CGT_MDCR_TDE, CGT_MDCR_TDOSA),
        MCB(CGT_MDCR_TDE_TDRA,          CGT_MDCR_TDE, CGT_MDCR_TDRA),
        return BEHAVE_HANDLE_LOCALLY;
 }
 
+static enum trap_behaviour check_mdcr_hpmn(struct kvm_vcpu *vcpu)
+{
+       u32 sysreg = esr_sys64_to_sysreg(kvm_vcpu_get_esr(vcpu));
+       unsigned int idx;
+
+
+       switch (sysreg) {
+       case SYS_PMEVTYPERn_EL0(0) ... SYS_PMEVTYPERn_EL0(30):
+       case SYS_PMEVCNTRn_EL0(0) ... SYS_PMEVCNTRn_EL0(30):
+               idx = (sys_reg_CRm(sysreg) & 0x3) << 3 | sys_reg_Op2(sysreg);
+               break;
+       case SYS_PMXEVTYPER_EL0:
+       case SYS_PMXEVCNTR_EL0:
+               idx = SYS_FIELD_GET(PMSELR_EL0, SEL,
+                                   __vcpu_sys_reg(vcpu, PMSELR_EL0));
+               break;
+       default:
+               /* Someone used this trap helper for something else... */
+               KVM_BUG_ON(1, vcpu->kvm);
+               return BEHAVE_HANDLE_LOCALLY;
+       }
+
+       if (kvm_pmu_counter_is_hyp(vcpu, idx))
+               return BEHAVE_FORWARD_RW | BEHAVE_FORWARD_IN_HOST_EL0;
+
+       return BEHAVE_HANDLE_LOCALLY;
+}
+
 #define CCC(id, fn)                            \
        [id - __COMPLEX_CONDITIONS__] = fn
 
        CCC(CGT_CNTHCTL_EL1PCTEN, check_cnthctl_el1pcten),
        CCC(CGT_CNTHCTL_EL1PTEN, check_cnthctl_el1pten),
        CCC(CGT_CPTR_TTA, check_cptr_tta),
+       CCC(CGT_MDCR_HPMN, check_mdcr_hpmn),
 };
 
 /*
        SR_TRAP(SYS_PMOVSCLR_EL0,       CGT_MDCR_TPM),
        SR_TRAP(SYS_PMCEID0_EL0,        CGT_MDCR_TPM),
        SR_TRAP(SYS_PMCEID1_EL0,        CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMXEVTYPER_EL0,     CGT_MDCR_TPM),
+       SR_TRAP(SYS_PMXEVTYPER_EL0,     CGT_MDCR_TPM_HPMN),
        SR_TRAP(SYS_PMSWINC_EL0,        CGT_MDCR_TPM),
        SR_TRAP(SYS_PMSELR_EL0,         CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMXEVCNTR_EL0,      CGT_MDCR_TPM),
+       SR_TRAP(SYS_PMXEVCNTR_EL0,      CGT_MDCR_TPM_HPMN),
        SR_TRAP(SYS_PMCCNTR_EL0,        CGT_MDCR_TPM),
        SR_TRAP(SYS_PMUSERENR_EL0,      CGT_MDCR_TPM),
        SR_TRAP(SYS_PMINTENSET_EL1,     CGT_MDCR_TPM),
        SR_TRAP(SYS_PMINTENCLR_EL1,     CGT_MDCR_TPM),
        SR_TRAP(SYS_PMMIR_EL1,          CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(0),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(1),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(2),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(3),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(4),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(5),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(6),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(7),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(8),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(9),   CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(10),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(11),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(12),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(13),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(14),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(15),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(16),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(17),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(18),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(19),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(20),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(21),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(22),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(23),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(24),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(25),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(26),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(27),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(28),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(29),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVCNTRn_EL0(30),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(0),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(1),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(2),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(3),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(4),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(5),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(6),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(7),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(8),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(9),  CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(10), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(11), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(12), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(13), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(14), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(15), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(16), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(17), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(18), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(19), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(20), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(21), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(22), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(23), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(24), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(25), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(26), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(27), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(28), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(29), CGT_MDCR_TPM),
-       SR_TRAP(SYS_PMEVTYPERn_EL0(30), CGT_MDCR_TPM),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(0),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(1),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(2),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(3),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(4),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(5),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(6),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(7),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(8),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(9),   CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(10),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(11),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(12),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(13),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(14),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(15),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(16),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(17),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(18),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(19),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(20),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(21),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(22),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(23),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(24),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(25),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(26),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(27),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(28),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(29),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVCNTRn_EL0(30),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(0),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(1),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(2),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(3),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(4),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(5),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(6),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(7),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(8),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(9),  CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(10), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(11), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(12), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(13), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(14), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(15), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(16), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(17), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(18), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(19), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(20), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(21), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(22), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(23), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(24), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(25), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(26), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(27), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(28), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(29), CGT_MDCR_TPM_HPMN),
+       SR_TRAP(SYS_PMEVTYPERn_EL0(30), CGT_MDCR_TPM_HPMN),
        SR_TRAP(SYS_PMCCFILTR_EL0,      CGT_MDCR_TPM),
        SR_TRAP(SYS_MDCCSR_EL0,         CGT_MDCR_TDCC_TDE_TDA),
        SR_TRAP(SYS_MDCCINT_EL1,        CGT_MDCR_TDCC_TDE_TDA),