diff options
author | Marc Zyngier <marc.zyngier@arm.com> | 2015-06-08 10:37:26 -0400 |
---|---|---|
committer | Marc Zyngier <marc.zyngier@arm.com> | 2015-08-12 06:28:24 -0400 |
commit | fb182cf84568cc33ab41121bc8cc999f7aacbd47 (patch) | |
tree | 4f4db678e56aff1dd7953e2936054b455db8355e /virt | |
parent | 32d2d8010cd7080a0f1712f1084b92657858428e (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.c | 16 | ||||
-rw-r--r-- | virt/kvm/arm/vgic-v3.c | 21 |
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) | |||
55 | static void vgic_v2_set_lr(struct kvm_vcpu *vcpu, int lr, | 59 | static 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 | } |