mirror of https://gitee.com/openkylin/linux.git
KVM: arm64: vgic-its: Enable ITS emulation as a virtual MSI controller
Now that all ITS emulation functionality is in place, we advertise MSI functionality to userland and also the ITS device to the guest - if userland has configured that. Signed-off-by: Andre Przywara <andre.przywara@arm.com> Reviewed-by: Marc Zyngier <marc.zyngier@arm.com> Tested-by: Eric Auger <eric.auger@redhat.com> Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
This commit is contained in:
parent
2891a7dfb6
commit
0e4e82f154
|
@ -2162,7 +2162,7 @@ after pausing the vcpu, but before it is resumed.
|
||||||
4.71 KVM_SIGNAL_MSI
|
4.71 KVM_SIGNAL_MSI
|
||||||
|
|
||||||
Capability: KVM_CAP_SIGNAL_MSI
|
Capability: KVM_CAP_SIGNAL_MSI
|
||||||
Architectures: x86
|
Architectures: x86 arm64
|
||||||
Type: vm ioctl
|
Type: vm ioctl
|
||||||
Parameters: struct kvm_msi (in)
|
Parameters: struct kvm_msi (in)
|
||||||
Returns: >0 on delivery, 0 if guest blocked the MSI, and -1 on error
|
Returns: >0 on delivery, 0 if guest blocked the MSI, and -1 on error
|
||||||
|
|
|
@ -36,6 +36,7 @@ config KVM
|
||||||
select HAVE_KVM_IRQFD
|
select HAVE_KVM_IRQFD
|
||||||
select KVM_ARM_VGIC_V3
|
select KVM_ARM_VGIC_V3
|
||||||
select KVM_ARM_PMU if HW_PERF_EVENTS
|
select KVM_ARM_PMU if HW_PERF_EVENTS
|
||||||
|
select HAVE_KVM_MSI
|
||||||
---help---
|
---help---
|
||||||
Support hosting virtualized guest machines.
|
Support hosting virtualized guest machines.
|
||||||
We don't support KVM with 16K page tables yet, due to the multiple
|
We don't support KVM with 16K page tables yet, due to the multiple
|
||||||
|
|
|
@ -29,5 +29,6 @@ kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-mmio.o
|
||||||
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-mmio-v2.o
|
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-mmio-v2.o
|
||||||
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-mmio-v3.o
|
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-mmio-v3.o
|
||||||
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-kvm-device.o
|
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-kvm-device.o
|
||||||
|
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/vgic/vgic-its.o
|
||||||
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/arch_timer.o
|
kvm-$(CONFIG_KVM_ARM_HOST) += $(KVM)/arm/arch_timer.o
|
||||||
kvm-$(CONFIG_KVM_ARM_PMU) += $(KVM)/arm/pmu.o
|
kvm-$(CONFIG_KVM_ARM_PMU) += $(KVM)/arm/pmu.o
|
||||||
|
|
|
@ -86,6 +86,12 @@ int kvm_arch_dev_ioctl_check_extension(struct kvm *kvm, long ext)
|
||||||
case KVM_CAP_VCPU_ATTRIBUTES:
|
case KVM_CAP_VCPU_ATTRIBUTES:
|
||||||
r = 1;
|
r = 1;
|
||||||
break;
|
break;
|
||||||
|
case KVM_CAP_MSI_DEVID:
|
||||||
|
if (!kvm)
|
||||||
|
r = -EINVAL;
|
||||||
|
else
|
||||||
|
r = kvm->arch.vgic.msis_require_devid;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
r = 0;
|
r = 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -163,6 +163,9 @@ struct vgic_dist {
|
||||||
/* vGIC model the kernel emulates for the guest (GICv2 or GICv3) */
|
/* vGIC model the kernel emulates for the guest (GICv2 or GICv3) */
|
||||||
u32 vgic_model;
|
u32 vgic_model;
|
||||||
|
|
||||||
|
/* Do injected MSIs require an additional device ID? */
|
||||||
|
bool msis_require_devid;
|
||||||
|
|
||||||
int nr_spis;
|
int nr_spis;
|
||||||
|
|
||||||
/* TODO: Consider moving to global state */
|
/* TODO: Consider moving to global state */
|
||||||
|
@ -308,4 +311,6 @@ static inline int kvm_vgic_get_max_vcpus(void)
|
||||||
return kvm_vgic_global_state.max_gic_vcpus;
|
return kvm_vgic_global_state.max_gic_vcpus;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int kvm_send_userspace_msi(struct kvm *kvm, struct kvm_msi *msi);
|
||||||
|
|
||||||
#endif /* __KVM_ARM_VGIC_H */
|
#endif /* __KVM_ARM_VGIC_H */
|
||||||
|
|
|
@ -258,6 +258,9 @@ int vgic_init(struct kvm *kvm)
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
if (vgic_has_its(kvm))
|
||||||
|
dist->msis_require_devid = true;
|
||||||
|
|
||||||
kvm_for_each_vcpu(i, vcpu, kvm)
|
kvm_for_each_vcpu(i, vcpu, kvm)
|
||||||
kvm_vgic_vcpu_init(vcpu);
|
kvm_vgic_vcpu_init(vcpu);
|
||||||
|
|
||||||
|
|
|
@ -223,6 +223,9 @@ int kvm_register_vgic_device(unsigned long type)
|
||||||
case KVM_DEV_TYPE_ARM_VGIC_V3:
|
case KVM_DEV_TYPE_ARM_VGIC_V3:
|
||||||
ret = kvm_register_device_ops(&kvm_arm_vgic_v3_ops,
|
ret = kvm_register_device_ops(&kvm_arm_vgic_v3_ops,
|
||||||
KVM_DEV_TYPE_ARM_VGIC_V3);
|
KVM_DEV_TYPE_ARM_VGIC_V3);
|
||||||
|
if (ret)
|
||||||
|
break;
|
||||||
|
ret = kvm_vgic_register_its_device();
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,7 +66,12 @@ static unsigned long vgic_mmio_read_v3_misc(struct kvm_vcpu *vcpu,
|
||||||
case GICD_TYPER:
|
case GICD_TYPER:
|
||||||
value = vcpu->kvm->arch.vgic.nr_spis + VGIC_NR_PRIVATE_IRQS;
|
value = vcpu->kvm->arch.vgic.nr_spis + VGIC_NR_PRIVATE_IRQS;
|
||||||
value = (value >> 5) - 1;
|
value = (value >> 5) - 1;
|
||||||
value |= (INTERRUPT_ID_BITS_SPIS - 1) << 19;
|
if (vgic_has_its(vcpu->kvm)) {
|
||||||
|
value |= (INTERRUPT_ID_BITS_ITS - 1) << 19;
|
||||||
|
value |= GICD_TYPER_LPIS;
|
||||||
|
} else {
|
||||||
|
value |= (INTERRUPT_ID_BITS_SPIS - 1) << 19;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case GICD_IIDR:
|
case GICD_IIDR:
|
||||||
value = (PRODUCT_ID_KVM << 24) | (IMPLEMENTER_ARM << 0);
|
value = (PRODUCT_ID_KVM << 24) | (IMPLEMENTER_ARM << 0);
|
||||||
|
@ -163,9 +168,8 @@ static void vgic_mmio_write_v3r_ctlr(struct kvm_vcpu *vcpu,
|
||||||
|
|
||||||
vgic_cpu->lpis_enabled = val & GICR_CTLR_ENABLE_LPIS;
|
vgic_cpu->lpis_enabled = val & GICR_CTLR_ENABLE_LPIS;
|
||||||
|
|
||||||
if (!was_enabled && vgic_cpu->lpis_enabled) {
|
if (!was_enabled && vgic_cpu->lpis_enabled)
|
||||||
/* Eventually do something */
|
vgic_enable_lpis(vcpu);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned long vgic_mmio_read_v3r_typer(struct kvm_vcpu *vcpu,
|
static unsigned long vgic_mmio_read_v3r_typer(struct kvm_vcpu *vcpu,
|
||||||
|
@ -179,6 +183,8 @@ static unsigned long vgic_mmio_read_v3r_typer(struct kvm_vcpu *vcpu,
|
||||||
value |= ((target_vcpu_id & 0xffff) << 8);
|
value |= ((target_vcpu_id & 0xffff) << 8);
|
||||||
if (target_vcpu_id == atomic_read(&vcpu->kvm->online_vcpus) - 1)
|
if (target_vcpu_id == atomic_read(&vcpu->kvm->online_vcpus) - 1)
|
||||||
value |= GICR_TYPER_LAST;
|
value |= GICR_TYPER_LAST;
|
||||||
|
if (vgic_has_its(vcpu->kvm))
|
||||||
|
value |= GICR_TYPER_PLPIS;
|
||||||
|
|
||||||
return extract_bytes(value, addr & 7, len);
|
return extract_bytes(value, addr & 7, len);
|
||||||
}
|
}
|
||||||
|
|
|
@ -718,3 +718,11 @@ bool kvm_vgic_map_is_active(struct kvm_vcpu *vcpu, unsigned int virt_irq)
|
||||||
|
|
||||||
return map_is_active;
|
return map_is_active;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int kvm_send_userspace_msi(struct kvm *kvm, struct kvm_msi *msi)
|
||||||
|
{
|
||||||
|
if (vgic_has_its(kvm))
|
||||||
|
return vgic_its_inject_msi(kvm, msi);
|
||||||
|
else
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
|
@ -77,6 +77,7 @@ int vgic_v3_probe(const struct gic_kvm_info *info);
|
||||||
int vgic_v3_map_resources(struct kvm *kvm);
|
int vgic_v3_map_resources(struct kvm *kvm);
|
||||||
int vgic_register_redist_iodevs(struct kvm *kvm, gpa_t dist_base_address);
|
int vgic_register_redist_iodevs(struct kvm *kvm, gpa_t dist_base_address);
|
||||||
bool vgic_has_its(struct kvm *kvm);
|
bool vgic_has_its(struct kvm *kvm);
|
||||||
|
int kvm_vgic_register_its_device(void);
|
||||||
void vgic_enable_lpis(struct kvm_vcpu *vcpu);
|
void vgic_enable_lpis(struct kvm_vcpu *vcpu);
|
||||||
int vgic_its_inject_msi(struct kvm *kvm, struct kvm_msi *msi);
|
int vgic_its_inject_msi(struct kvm *kvm, struct kvm_msi *msi);
|
||||||
#else
|
#else
|
||||||
|
@ -136,6 +137,11 @@ static inline bool vgic_has_its(struct kvm *kvm)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int kvm_vgic_register_its_device(void)
|
||||||
|
{
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
static inline void vgic_enable_lpis(struct kvm_vcpu *vcpu)
|
static inline void vgic_enable_lpis(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue