aboutsummaryrefslogtreecommitdiffstats
path: root/virt
diff options
context:
space:
mode:
authorMarc Zyngier <marc.zyngier@arm.com>2015-06-08 10:37:26 -0400
committerMarc Zyngier <marc.zyngier@arm.com>2015-08-12 06:28:24 -0400
commitfb182cf84568cc33ab41121bc8cc999f7aacbd47 (patch)
tree4f4db678e56aff1dd7953e2936054b455db8355e /virt
parent32d2d8010cd7080a0f1712f1084b92657858428e (diff)
KVM: arm/arm64: vgic: Allow HW irq to be encoded in LR
Now that struct vgic_lr supports the LR_HW bit and carries a hwirq field, we can encode that information into the list registers. This patch provides implementations for both GICv2 and GICv3. Reviewed-by: Christoffer Dall <christoffer.dall@linaro.org> Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
Diffstat (limited to 'virt')
-rw-r--r--virt/kvm/arm/vgic-v2.c16
-rw-r--r--virt/kvm/arm/vgic-v3.c21
2 files changed, 33 insertions, 4 deletions
diff --git a/virt/kvm/arm/vgic-v2.c b/virt/kvm/arm/vgic-v2.c
index f9b9c7c51372..8d7b04db8471 100644
--- a/virt/kvm/arm/vgic-v2.c
+++ b/virt/kvm/arm/vgic-v2.c
@@ -48,6 +48,10 @@ static struct vgic_lr vgic_v2_get_lr(const struct kvm_vcpu *vcpu, int lr)
48 lr_desc.state |= LR_STATE_ACTIVE; 48 lr_desc.state |= LR_STATE_ACTIVE;
49 if (val & GICH_LR_EOI) 49 if (val & GICH_LR_EOI)
50 lr_desc.state |= LR_EOI_INT; 50 lr_desc.state |= LR_EOI_INT;
51 if (val & GICH_LR_HW) {
52 lr_desc.state |= LR_HW;
53 lr_desc.hwirq = (val & GICH_LR_PHYSID_CPUID) >> GICH_LR_PHYSID_CPUID_SHIFT;
54 }
51 55
52 return lr_desc; 56 return lr_desc;
53} 57}
@@ -55,7 +59,9 @@ static struct vgic_lr vgic_v2_get_lr(const struct kvm_vcpu *vcpu, int lr)
55static void vgic_v2_set_lr(struct kvm_vcpu *vcpu, int lr, 59static void vgic_v2_set_lr(struct kvm_vcpu *vcpu, int lr,
56 struct vgic_lr lr_desc) 60 struct vgic_lr lr_desc)
57{ 61{
58 u32 lr_val = (lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT) | lr_desc.irq; 62 u32 lr_val;
63
64 lr_val = lr_desc.irq;
59 65
60 if (lr_desc.state & LR_STATE_PENDING) 66 if (lr_desc.state & LR_STATE_PENDING)
61 lr_val |= GICH_LR_PENDING_BIT; 67 lr_val |= GICH_LR_PENDING_BIT;
@@ -64,6 +70,14 @@ static void vgic_v2_set_lr(struct kvm_vcpu *vcpu, int lr,
64 if (lr_desc.state & LR_EOI_INT) 70 if (lr_desc.state & LR_EOI_INT)
65 lr_val |= GICH_LR_EOI; 71 lr_val |= GICH_LR_EOI;
66 72
73 if (lr_desc.state & LR_HW) {
74 lr_val |= GICH_LR_HW;
75 lr_val |= (u32)lr_desc.hwirq << GICH_LR_PHYSID_CPUID_SHIFT;
76 }
77
78 if (lr_desc.irq < VGIC_NR_SGIS)
79 lr_val |= (lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT);
80
67 vcpu->arch.vgic_cpu.vgic_v2.vgic_lr[lr] = lr_val; 81 vcpu->arch.vgic_cpu.vgic_v2.vgic_lr[lr] = lr_val;
68} 82}
69 83
diff --git a/virt/kvm/arm/vgic-v3.c b/virt/kvm/arm/vgic-v3.c
index dff06021e748..afbf925b00f4 100644
--- a/virt/kvm/arm/vgic-v3.c
+++ b/virt/kvm/arm/vgic-v3.c
@@ -67,6 +67,10 @@ static struct vgic_lr vgic_v3_get_lr(const struct kvm_vcpu *vcpu, int lr)
67 lr_desc.state |= LR_STATE_ACTIVE; 67 lr_desc.state |= LR_STATE_ACTIVE;
68 if (val & ICH_LR_EOI) 68 if (val & ICH_LR_EOI)
69 lr_desc.state |= LR_EOI_INT; 69 lr_desc.state |= LR_EOI_INT;
70 if (val & ICH_LR_HW) {
71 lr_desc.state |= LR_HW;
72 lr_desc.hwirq = (val >> ICH_LR_PHYS_ID_SHIFT) & GENMASK(9, 0);
73 }
70 74
71 return lr_desc; 75 return lr_desc;
72} 76}
@@ -84,10 +88,17 @@ static void vgic_v3_set_lr(struct kvm_vcpu *vcpu, int lr,
84 * Eventually we want to make this configurable, so we may revisit 88 * Eventually we want to make this configurable, so we may revisit
85 * this in the future. 89 * this in the future.
86 */ 90 */
87 if (vcpu->kvm->arch.vgic.vgic_model == KVM_DEV_TYPE_ARM_VGIC_V3) 91 switch (vcpu->kvm->arch.vgic.vgic_model) {
92 case KVM_DEV_TYPE_ARM_VGIC_V3:
88 lr_val |= ICH_LR_GROUP; 93 lr_val |= ICH_LR_GROUP;
89 else 94 break;
90 lr_val |= (u32)lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT; 95 case KVM_DEV_TYPE_ARM_VGIC_V2:
96 if (lr_desc.irq < VGIC_NR_SGIS)
97 lr_val |= (u32)lr_desc.source << GICH_LR_PHYSID_CPUID_SHIFT;
98 break;
99 default:
100 BUG();
101 }
91 102
92 if (lr_desc.state & LR_STATE_PENDING) 103 if (lr_desc.state & LR_STATE_PENDING)
93 lr_val |= ICH_LR_PENDING_BIT; 104 lr_val |= ICH_LR_PENDING_BIT;
@@ -95,6 +106,10 @@ static void vgic_v3_set_lr(struct kvm_vcpu *vcpu, int lr,
95 lr_val |= ICH_LR_ACTIVE_BIT; 106 lr_val |= ICH_LR_ACTIVE_BIT;
96 if (lr_desc.state & LR_EOI_INT) 107 if (lr_desc.state & LR_EOI_INT)
97 lr_val |= ICH_LR_EOI; 108 lr_val |= ICH_LR_EOI;
109 if (lr_desc.state & LR_HW) {
110 lr_val |= ICH_LR_HW;
111 lr_val |= ((u64)lr_desc.hwirq) << ICH_LR_PHYS_ID_SHIFT;
112 }
98 113
99 vcpu->arch.vgic_cpu.vgic_v3.vgic_lr[LR_INDEX(lr)] = lr_val; 114 vcpu->arch.vgic_cpu.vgic_v3.vgic_lr[LR_INDEX(lr)] = lr_val;
100} 115}