Re: [PATCH 4/7] KVM: arm/arm64: enable irqchip routing

2015-06-30 Thread Andre Przywara
Hi Eric,

On 29/06/15 16:37, Eric Auger wrote:
 This patch adds compilation and link against irqchip.
 
 On ARM, irqchip routing is not really useful since there is
 a single irqchip. However main motivation behind using irqchip
 code is to enable MSI routing code. With the support of in-kernel
 GICv3 ITS emulation, it now seems to be a MUST HAVE requirement.
 
 Functions previously implemented in vgic.c and substitute
 to more complex irqchip implementation are removed:
 
 - kvm_send_userspace_msi
 - kvm_irq_map_chip_pin
 - kvm_set_irq
 - kvm_irq_map_gsi.
 
 They implemented a kernel default identity GSI routing. This is now
 replaced by user-side provided routing.
 
 Routing standard hooks are now implemented in vgic:
 - kvm_set_routing_entry
 - kvm_set_irq
 - kvm_set_msi
 
 Both HAVE_KVM_IRQCHIP and HAVE_KVM_IRQ_ROUTING are defined.
 KVM_CAP_IRQ_ROUTING is advertised and KVM_SET_GSI_ROUTING is allowed.
 
 MSI routing is not yet allowed.
 
 Signed-off-by: Eric Auger eric.au...@linaro.org
 
...

 diff --git a/virt/kvm/arm/vgic.c b/virt/kvm/arm/vgic.c
 index 09b1f46..212a5ff 100644
 --- a/virt/kvm/arm/vgic.c
 +++ b/virt/kvm/arm/vgic.c
 @@ -2220,47 +2220,67 @@ out_free_irq:
   return ret;
  }
  
 -int kvm_irq_map_gsi(struct kvm *kvm,
 - struct kvm_kernel_irq_routing_entry *entries,
 - int gsi)
 +int vgic_irqfd_set_irq(struct kvm_kernel_irq_routing_entry *e,
 + struct kvm *kvm, int irq_source_id,
 + int level, bool line_status)
  {
 - return 0;
 -}
 -
 -int kvm_irq_map_chip_pin(struct kvm *kvm, unsigned irqchip, unsigned pin)
 -{
 - return pin;
 -}
 -
 -int kvm_set_irq(struct kvm *kvm, int irq_source_id,
 - u32 irq, int level, bool line_status)
 -{
 - unsigned int spi = irq + VGIC_NR_PRIVATE_IRQS;
 + unsigned int spi_id = e-irqchip.pin + VGIC_NR_PRIVATE_IRQS;
  
 - trace_kvm_set_irq(irq, level, irq_source_id);
 + trace_kvm_set_irq(spi_id, level, irq_source_id);
  
   BUG_ON(!vgic_initialized(kvm));
  
 - if (spi  kvm-arch.vgic.nr_irqs)
 + if (spi_id  kvm-arch.vgic.nr_irqs)
   return -EINVAL;
 - return kvm_vgic_inject_irq(kvm, 0, spi, level);
 + return kvm_vgic_inject_irq(kvm, 0, spi_id, level);
  
  }
  
 -/* MSI not implemented yet */
 +/**
 + * Populates a kvm routing entry from a user routing entry
 + * @e: kvm internal formatted entry
 + * @ue: user api formatted entry
 + * return 0 on success, -EINVAL on errors.
 + */
 +int kvm_set_routing_entry(struct kvm_kernel_irq_routing_entry *e,
 +   const struct kvm_irq_routing_entry *ue)
 +{
 + int r = -EINVAL;
 +
 + switch (ue-type) {
 + case KVM_IRQ_ROUTING_IRQCHIP:
 + e-set = vgic_irqfd_set_irq;
 + e-irqchip.irqchip = ue-u.irqchip.irqchip;
 + e-irqchip.pin = ue-u.irqchip.pin;
 + if ((e-irqchip.pin = KVM_IRQCHIP_NUM_PINS) ||
 + (e-irqchip.irqchip = KVM_NR_IRQCHIPS))
 + goto out;
 + break;
 + default:
 + goto out;
 + }
 + r = 0;
 +out:
 + return r;
 +}
 +
  int kvm_set_msi(struct kvm_kernel_irq_routing_entry *e,
   struct kvm *kvm, int irq_source_id,
   int level, bool line_status)
  {
 - return 0;
 -}
 + struct kvm_msi msi;
  
 -#ifdef CONFIG_HAVE_KVM_MSI
 -int kvm_send_userspace_msi(struct kvm *kvm, struct kvm_msi *msi)
 -{
 - if (kvm-arch.vgic.vm_ops.inject_msi)
 - return kvm-arch.vgic.vm_ops.inject_msi(kvm, msi);
 - else
 - return -ENODEV;
 + switch (e-type) {
 + case KVM_IRQ_ROUTING_EXTENDED_MSI:
 + msi.address_lo = e-ext_msi.address_lo;
 + msi.address_hi = e-ext_msi.address_hi;
 + msi.data = e-ext_msi.data;
 + msi.flags = e-ext_msi.devid;

You probably meant to write:
+   msi.flags = KVM_MSI_VALID_DEVID;
+   msi.devid = e-ext_msi.devid;

With this change I could get it (your patches on top of my v1.5) to work
with my hacked kvmtool - at least with vhost=0. On to fixing irqfd now ...

Cheers,
Andre.

 + if (kvm-arch.vgic.vm_ops.inject_msi)
 + return kvm-arch.vgic.vm_ops.inject_msi(kvm, msi);
 + else
 + return -ENODEV;
 + default:
 + return -EINVAL;
 + }
  }
 -#endif
--
To unsubscribe from this list: send the line unsubscribe kvm in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


Re: [PATCH 4/7] KVM: arm/arm64: enable irqchip routing

2015-06-30 Thread Eric Auger
On 06/30/2015 03:39 PM, Andre Przywara wrote:
 Hi Eric,
 
 On 29/06/15 16:37, Eric Auger wrote:
 This patch adds compilation and link against irqchip.

 On ARM, irqchip routing is not really useful since there is
 a single irqchip. However main motivation behind using irqchip
 code is to enable MSI routing code. With the support of in-kernel
 GICv3 ITS emulation, it now seems to be a MUST HAVE requirement.

 Functions previously implemented in vgic.c and substitute
 to more complex irqchip implementation are removed:

 - kvm_send_userspace_msi
 - kvm_irq_map_chip_pin
 - kvm_set_irq
 - kvm_irq_map_gsi.

 They implemented a kernel default identity GSI routing. This is now
 replaced by user-side provided routing.

 Routing standard hooks are now implemented in vgic:
 - kvm_set_routing_entry
 - kvm_set_irq
 - kvm_set_msi

 Both HAVE_KVM_IRQCHIP and HAVE_KVM_IRQ_ROUTING are defined.
 KVM_CAP_IRQ_ROUTING is advertised and KVM_SET_GSI_ROUTING is allowed.

 MSI routing is not yet allowed.

 Signed-off-by: Eric Auger eric.au...@linaro.org

 ...
 
 diff --git a/virt/kvm/arm/vgic.c b/virt/kvm/arm/vgic.c
 index 09b1f46..212a5ff 100644
 --- a/virt/kvm/arm/vgic.c
 +++ b/virt/kvm/arm/vgic.c
 @@ -2220,47 +2220,67 @@ out_free_irq:
  return ret;
  }
  
 -int kvm_irq_map_gsi(struct kvm *kvm,
 -struct kvm_kernel_irq_routing_entry *entries,
 -int gsi)
 +int vgic_irqfd_set_irq(struct kvm_kernel_irq_routing_entry *e,
 +struct kvm *kvm, int irq_source_id,
 +int level, bool line_status)
  {
 -return 0;
 -}
 -
 -int kvm_irq_map_chip_pin(struct kvm *kvm, unsigned irqchip, unsigned pin)
 -{
 -return pin;
 -}
 -
 -int kvm_set_irq(struct kvm *kvm, int irq_source_id,
 -u32 irq, int level, bool line_status)
 -{
 -unsigned int spi = irq + VGIC_NR_PRIVATE_IRQS;
 +unsigned int spi_id = e-irqchip.pin + VGIC_NR_PRIVATE_IRQS;
  
 -trace_kvm_set_irq(irq, level, irq_source_id);
 +trace_kvm_set_irq(spi_id, level, irq_source_id);
  
  BUG_ON(!vgic_initialized(kvm));
  
 -if (spi  kvm-arch.vgic.nr_irqs)
 +if (spi_id  kvm-arch.vgic.nr_irqs)
  return -EINVAL;
 -return kvm_vgic_inject_irq(kvm, 0, spi, level);
 +return kvm_vgic_inject_irq(kvm, 0, spi_id, level);
  
  }
  
 -/* MSI not implemented yet */
 +/**
 + * Populates a kvm routing entry from a user routing entry
 + * @e: kvm internal formatted entry
 + * @ue: user api formatted entry
 + * return 0 on success, -EINVAL on errors.
 + */
 +int kvm_set_routing_entry(struct kvm_kernel_irq_routing_entry *e,
 +  const struct kvm_irq_routing_entry *ue)
 +{
 +int r = -EINVAL;
 +
 +switch (ue-type) {
 +case KVM_IRQ_ROUTING_IRQCHIP:
 +e-set = vgic_irqfd_set_irq;
 +e-irqchip.irqchip = ue-u.irqchip.irqchip;
 +e-irqchip.pin = ue-u.irqchip.pin;
 +if ((e-irqchip.pin = KVM_IRQCHIP_NUM_PINS) ||
 +(e-irqchip.irqchip = KVM_NR_IRQCHIPS))
 +goto out;
 +break;
 +default:
 +goto out;
 +}
 +r = 0;
 +out:
 +return r;
 +}
 +
  int kvm_set_msi(struct kvm_kernel_irq_routing_entry *e,
  struct kvm *kvm, int irq_source_id,
  int level, bool line_status)
  {
 -return 0;
 -}
 +struct kvm_msi msi;
  
 -#ifdef CONFIG_HAVE_KVM_MSI
 -int kvm_send_userspace_msi(struct kvm *kvm, struct kvm_msi *msi)
 -{
 -if (kvm-arch.vgic.vm_ops.inject_msi)
 -return kvm-arch.vgic.vm_ops.inject_msi(kvm, msi);
 -else
 -return -ENODEV;
 +switch (e-type) {
 +case KVM_IRQ_ROUTING_EXTENDED_MSI:
 +msi.address_lo = e-ext_msi.address_lo;
 +msi.address_hi = e-ext_msi.address_hi;
 +msi.data = e-ext_msi.data;
 +msi.flags = e-ext_msi.devid;
 
 You probably meant to write:
 + msi.flags = KVM_MSI_VALID_DEVID;
 + msi.devid = e-ext_msi.devid;
 
 With this change I could get it (your patches on top of my v1.5) to work
 with my hacked kvmtool - at least with vhost=0. On to fixing irqfd now ...

Yes sure. Thanks for the catch!

I will correct in next respin

Thanks

Eric
 
 Cheers,
 Andre.
 
 +if (kvm-arch.vgic.vm_ops.inject_msi)
 +return kvm-arch.vgic.vm_ops.inject_msi(kvm, msi);
 +else
 +return -ENODEV;
 +default:
 +return -EINVAL;
 +}
  }
 -#endif

--
To unsubscribe from this list: send the line unsubscribe kvm in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 4/7] KVM: arm/arm64: enable irqchip routing

2015-06-29 Thread Eric Auger
This patch adds compilation and link against irqchip.

On ARM, irqchip routing is not really useful since there is
a single irqchip. However main motivation behind using irqchip
code is to enable MSI routing code. With the support of in-kernel
GICv3 ITS emulation, it now seems to be a MUST HAVE requirement.

Functions previously implemented in vgic.c and substitute
to more complex irqchip implementation are removed:

- kvm_send_userspace_msi
- kvm_irq_map_chip_pin
- kvm_set_irq
- kvm_irq_map_gsi.

They implemented a kernel default identity GSI routing. This is now
replaced by user-side provided routing.

Routing standard hooks are now implemented in vgic:
- kvm_set_routing_entry
- kvm_set_irq
- kvm_set_msi

Both HAVE_KVM_IRQCHIP and HAVE_KVM_IRQ_ROUTING are defined.
KVM_CAP_IRQ_ROUTING is advertised and KVM_SET_GSI_ROUTING is allowed.

MSI routing is not yet allowed.

Signed-off-by: Eric Auger eric.au...@linaro.org

---

RFC - PATCH
- reword api.txt:
  x move MSI routing comments in a subsequent patch,
  x clearly state GSI routing does not apply to KVM_IRQ_LINE
---
 Documentation/virtual/kvm/api.txt | 12 --
 arch/arm/include/asm/kvm_host.h   |  2 +
 arch/arm/kvm/Kconfig  |  2 +
 arch/arm/kvm/Makefile |  2 +-
 arch/arm64/include/asm/kvm_host.h |  1 +
 arch/arm64/kvm/Kconfig|  2 +
 arch/arm64/kvm/Makefile   |  2 +-
 include/kvm/arm_vgic.h|  9 -
 virt/kvm/arm/vgic.c   | 78 ---
 virt/kvm/irqchip.c|  2 +
 10 files changed, 68 insertions(+), 44 deletions(-)

diff --git a/Documentation/virtual/kvm/api.txt 
b/Documentation/virtual/kvm/api.txt
index 6426ae9..1e0d5f5 100644
--- a/Documentation/virtual/kvm/api.txt
+++ b/Documentation/virtual/kvm/api.txt
@@ -1395,13 +1395,16 @@ KVM_ASSIGN_DEV_IRQ. Partial deassignment of host or 
guest IRQ is allowed.
 4.52 KVM_SET_GSI_ROUTING
 
 Capability: KVM_CAP_IRQ_ROUTING
-Architectures: x86 s390
+Architectures: x86 s390 arm arm64
 Type: vm ioctl
 Parameters: struct kvm_irq_routing (in)
 Returns: 0 on success, -1 on error
 
 Sets the GSI routing table entries, overwriting any previously set entries.
 
+On arm/arm64, GSI routing has the following limitation:
+- GSI routing does not apply to KVM_IRQ_LINE but only to KVM_IRQFD.
+
 struct kvm_irq_routing {
__u32 nr;
__u32 flags;
@@ -2308,9 +2311,10 @@ Note that closing the resamplefd is not sufficient to 
disable the
 irqfd.  The KVM_IRQFD_FLAG_RESAMPLE is only necessary on assignment
 and need not be specified with KVM_IRQFD_FLAG_DEASSIGN.
 
-On ARM/ARM64, the gsi field in the kvm_irqfd struct specifies the Shared
-Peripheral Interrupt (SPI) index, such that the GIC interrupt ID is
-given by gsi + 32.
+On arm/arm64, gsi routing being supported, the following can happen:
+- in case no routing entry is associated to this gsi, injection fails
+- in case the gsi is associated to an irqchip routing entry,
+  irqchip.pin + 32 corresponds to the injected SPI ID.
 
 4.76 KVM_PPC_ALLOCATE_HTAB
 
diff --git a/arch/arm/include/asm/kvm_host.h b/arch/arm/include/asm/kvm_host.h
index d71607c..452697e 100644
--- a/arch/arm/include/asm/kvm_host.h
+++ b/arch/arm/include/asm/kvm_host.h
@@ -42,6 +42,8 @@
 
 #define KVM_VCPU_MAX_FEATURES 2
 
+#define KVM_IRQCHIP_NUM_PINS 988 /* 1020 -32 is the number of SPI */
+
 #include kvm/arm_vgic.h
 
 u32 *kvm_vcpu_reg(struct kvm_vcpu *vcpu, u8 reg_num, u32 mode);
diff --git a/arch/arm/kvm/Kconfig b/arch/arm/kvm/Kconfig
index bfb915d..151e710 100644
--- a/arch/arm/kvm/Kconfig
+++ b/arch/arm/kvm/Kconfig
@@ -31,6 +31,8 @@ config KVM
select KVM_VFIO
select HAVE_KVM_EVENTFD
select HAVE_KVM_IRQFD
+   select HAVE_KVM_IRQCHIP
+   select HAVE_KVM_IRQ_ROUTING
depends on ARM_VIRT_EXT  ARM_LPAE  ARM_ARCH_TIMER
---help---
  Support hosting virtualized guest machines.
diff --git a/arch/arm/kvm/Makefile b/arch/arm/kvm/Makefile
index c5eef02c..1a8f48a 100644
--- a/arch/arm/kvm/Makefile
+++ b/arch/arm/kvm/Makefile
@@ -15,7 +15,7 @@ AFLAGS_init.o := -Wa,-march=armv7-a$(plus_virt)
 AFLAGS_interrupts.o := -Wa,-march=armv7-a$(plus_virt)
 
 KVM := ../../../virt/kvm
-kvm-arm-y = $(KVM)/kvm_main.o $(KVM)/coalesced_mmio.o $(KVM)/eventfd.o 
$(KVM)/vfio.o
+kvm-arm-y = $(KVM)/kvm_main.o $(KVM)/coalesced_mmio.o $(KVM)/eventfd.o 
$(KVM)/vfio.o $(KVM)/irqchip.o
 
 obj-y += kvm-arm.o init.o interrupts.o
 obj-y += arm.o handle_exit.o guest.o mmu.o emulate.o reset.o
diff --git a/arch/arm64/include/asm/kvm_host.h 
b/arch/arm64/include/asm/kvm_host.h
index f0f58c9..751210a 100644
--- a/arch/arm64/include/asm/kvm_host.h
+++ b/arch/arm64/include/asm/kvm_host.h
@@ -44,6 +44,7 @@
 #include kvm/arm_arch_timer.h
 
 #define KVM_VCPU_MAX_FEATURES 3
+#define KVM_IRQCHIP_NUM_PINS 988 /* 1020 -32 is the number of SPI */
 
 int __attribute_const__ kvm_target_cpu(void);
 int kvm_reset_vcpu(struct kvm_vcpu *vcpu);
diff --git a/arch/arm64/kvm/Kconfig