|
@@ -67,6 +67,10 @@ static struct vgic_lr vgic_v3_get_lr(const struct kvm_vcpu *vcpu, int lr)
|
|
|
lr_desc.state |= LR_STATE_ACTIVE;
|
|
|
if (val & ICH_LR_EOI)
|
|
|
lr_desc.state |= LR_EOI_INT;
|
|
|
+ if (val & ICH_LR_HW) {
|
|
|
+ lr_desc.state |= LR_HW;
|
|
|
+ lr_desc.hwirq = (val >> ICH_LR_PHYS_ID_SHIFT) & GENMASK(9, 0);
|
|
|
+ }
|
|
|
|
|
|
return lr_desc;
|
|
|
}
|
|
@@ -84,10 +88,17 @@ static void vgic_v3_set_lr(struct kvm_vcpu *vcpu, int lr,
|
|
|
* Eventually we want to make this configurable, so we may revisit
|
|
|
* this in the future.
|
|
|
*/
|
|
|
- if (vcpu->kvm->arch.vgic.vgic_model == KVM_DEV_TYPE_ARM_VGIC_V3)
|
|
|
+ switch (vcpu->kvm->arch.vgic.vgic_model) {
|
|
|
+ case KVM_DEV_TYPE_ARM_VGIC_V3:
|
|
|
lr_val |= ICH_LR_GROUP;
|
|
|
- else
|
|
|
- lr_val |= (u32)lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT;
|
|
|
+ break;
|
|
|
+ case KVM_DEV_TYPE_ARM_VGIC_V2:
|
|
|
+ if (lr_desc.irq < VGIC_NR_SGIS)
|
|
|
+ lr_val |= (u32)lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ BUG();
|
|
|
+ }
|
|
|
|
|
|
if (lr_desc.state & LR_STATE_PENDING)
|
|
|
lr_val |= ICH_LR_PENDING_BIT;
|
|
@@ -95,6 +106,10 @@ static void vgic_v3_set_lr(struct kvm_vcpu *vcpu, int lr,
|
|
|
lr_val |= ICH_LR_ACTIVE_BIT;
|
|
|
if (lr_desc.state & LR_EOI_INT)
|
|
|
lr_val |= ICH_LR_EOI;
|
|
|
+ if (lr_desc.state & LR_HW) {
|
|
|
+ lr_val |= ICH_LR_HW;
|
|
|
+ lr_val |= ((u64)lr_desc.hwirq) << ICH_LR_PHYS_ID_SHIFT;
|
|
|
+ }
|
|
|
|
|
|
vcpu->arch.vgic_cpu.vgic_v3.vgic_lr[LR_INDEX(lr)] = lr_val;
|
|
|
}
|