Re: [PATCH v6 1/3] LoongArch: KVM: Enable paravirt feature control from VMM

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi, Bibo,

On Mon, Aug 12, 2024 at 11:02 AM Bibo Mao <maobibo@xxxxxxxxxxx> wrote:
>
> Export kernel paravirt features to user space, so that VMM can control
> the single paravirt feature. By default paravirt features will be the same
> with kvm supported features if VMM does not set it.
>
> Also a new feature KVM_FEATURE_VIRT_EXTIOI is added which can be set from
> user space. This feature indicates that the virt EXTIOI can route
> interrupts to 256 vCPUs, rather than 4 vCPUs like with real HW.
>
> Signed-off-by: Bibo Mao <maobibo@xxxxxxxxxxx>
> ---
>  arch/loongarch/include/asm/kvm_host.h      |  7 ++++
>  arch/loongarch/include/asm/kvm_para.h      |  1 +
>  arch/loongarch/include/asm/kvm_vcpu.h      |  4 ++
>  arch/loongarch/include/asm/loongarch.h     | 13 ------
>  arch/loongarch/include/uapi/asm/Kbuild     |  2 -
>  arch/loongarch/include/uapi/asm/kvm.h      |  5 +++
>  arch/loongarch/include/uapi/asm/kvm_para.h | 24 +++++++++++
>  arch/loongarch/kernel/paravirt.c           |  8 ++--
>  arch/loongarch/kvm/exit.c                  | 19 ++++-----
>  arch/loongarch/kvm/vcpu.c                  | 47 ++++++++++++++++++----
>  arch/loongarch/kvm/vm.c                    | 43 +++++++++++++++++++-
>  11 files changed, 137 insertions(+), 36 deletions(-)
>  create mode 100644 arch/loongarch/include/uapi/asm/kvm_para.h
>
> diff --git a/arch/loongarch/include/asm/kvm_host.h b/arch/loongarch/include/asm/kvm_host.h
> index 5f0677e03817..b73f6678e38a 100644
> --- a/arch/loongarch/include/asm/kvm_host.h
> +++ b/arch/loongarch/include/asm/kvm_host.h
> @@ -107,6 +107,8 @@ struct kvm_arch {
>         unsigned int  root_level;
>         spinlock_t    phyid_map_lock;
>         struct kvm_phyid_map  *phyid_map;
> +       /* Enabled PV features */
> +       unsigned long pv_features;
>
>         s64 time_offset;
>         struct kvm_context __percpu *vmcs;
> @@ -136,6 +138,11 @@ enum emulation_result {
>  #define KVM_LARCH_SWCSR_LATEST (0x1 << 3)
>  #define KVM_LARCH_HWCSR_USABLE (0x1 << 4)
>
> +#define LOONGARCH_PV_FEAT_UPDATED              BIT_ULL(63)
> +#define LOONGARCH_PV_FEAT_MASK                                         \
> +               (BIT(KVM_FEATURE_IPI) | BIT(KVM_FEATURE_STEAL_TIME) |   \
> +                BIT(KVM_FEATURE_VIRT_EXTIOI))
> +
>  struct kvm_vcpu_arch {
>         /*
>          * Switch pointer-to-function type to unsigned long
> diff --git a/arch/loongarch/include/asm/kvm_para.h b/arch/loongarch/include/asm/kvm_para.h
> index 43ec61589e6c..39d7483ab8fd 100644
> --- a/arch/loongarch/include/asm/kvm_para.h
> +++ b/arch/loongarch/include/asm/kvm_para.h
> @@ -2,6 +2,7 @@
>  #ifndef _ASM_LOONGARCH_KVM_PARA_H
>  #define _ASM_LOONGARCH_KVM_PARA_H
>
> +#include <uapi/asm/kvm_para.h>
>  /*
>   * Hypercall code field
>   */
> diff --git a/arch/loongarch/include/asm/kvm_vcpu.h b/arch/loongarch/include/asm/kvm_vcpu.h
> index c416cb7125c0..a1fc24a48fd1 100644
> --- a/arch/loongarch/include/asm/kvm_vcpu.h
> +++ b/arch/loongarch/include/asm/kvm_vcpu.h
> @@ -125,4 +125,8 @@ static inline bool kvm_pvtime_supported(void)
>         return !!sched_info_on();
>  }
>
> +static __always_inline bool guest_pv_has(struct kvm_vcpu *vcpu, unsigned int feature)
> +{
> +       return vcpu->kvm->arch.pv_features & BIT(feature);
> +}
We have similar functions
kvm_guest_has_fpu/kvm_guest_has_lsx/kvm_guest_has_lasx, so maybe it is
better to rename it as kvm_guest_has_pv_feature().

>  #endif /* __ASM_LOONGARCH_KVM_VCPU_H__ */
> diff --git a/arch/loongarch/include/asm/loongarch.h b/arch/loongarch/include/asm/loongarch.h
> index 04a78010fc72..eb82230f52c3 100644
> --- a/arch/loongarch/include/asm/loongarch.h
> +++ b/arch/loongarch/include/asm/loongarch.h
> @@ -158,19 +158,6 @@
>  #define  CPUCFG48_VFPU_CG              BIT(2)
>  #define  CPUCFG48_RAM_CG               BIT(3)
>
> -/*
> - * CPUCFG index area: 0x40000000 -- 0x400000ff
> - * SW emulation for KVM hypervirsor
> - */
> -#define CPUCFG_KVM_BASE                        0x40000000
> -#define CPUCFG_KVM_SIZE                        0x100
> -
> -#define CPUCFG_KVM_SIG                 (CPUCFG_KVM_BASE + 0)
> -#define  KVM_SIGNATURE                 "KVM\0"
> -#define CPUCFG_KVM_FEATURE             (CPUCFG_KVM_BASE + 4)
> -#define  KVM_FEATURE_IPI               BIT(1)
> -#define  KVM_FEATURE_STEAL_TIME                BIT(2)
It is a little better to keep these definitions here (at least
convenient for grep).



Huacai

> -
>  #ifndef __ASSEMBLY__
>
>  /* CSR */
> diff --git a/arch/loongarch/include/uapi/asm/Kbuild b/arch/loongarch/include/uapi/asm/Kbuild
> index c6d141d7b7d7..517761419999 100644
> --- a/arch/loongarch/include/uapi/asm/Kbuild
> +++ b/arch/loongarch/include/uapi/asm/Kbuild
> @@ -1,4 +1,2 @@
>  # SPDX-License-Identifier: GPL-2.0
>  syscall-y += unistd_64.h
> -
> -generic-y += kvm_para.h
> diff --git a/arch/loongarch/include/uapi/asm/kvm.h b/arch/loongarch/include/uapi/asm/kvm.h
> index ddc5cab0ffd0..719490e64e1c 100644
> --- a/arch/loongarch/include/uapi/asm/kvm.h
> +++ b/arch/loongarch/include/uapi/asm/kvm.h
> @@ -82,6 +82,11 @@ struct kvm_fpu {
>  #define KVM_IOC_CSRID(REG)             LOONGARCH_REG_64(KVM_REG_LOONGARCH_CSR, REG)
>  #define KVM_IOC_CPUCFG(REG)            LOONGARCH_REG_64(KVM_REG_LOONGARCH_CPUCFG, REG)
>
> +/* Device Control API on vm fd */
> +#define KVM_LOONGARCH_VM_FEAT_CTRL             0
> +#define  KVM_LOONGARCH_VM_FEAT_PV_IPI          5
> +#define  KVM_LOONGARCH_VM_FEAT_PV_STEALTIME    6
> +
>  /* Device Control API on vcpu fd */
>  #define KVM_LOONGARCH_VCPU_CPUCFG      0
>  #define KVM_LOONGARCH_VCPU_PVTIME_CTRL 1
> diff --git a/arch/loongarch/include/uapi/asm/kvm_para.h b/arch/loongarch/include/uapi/asm/kvm_para.h
> new file mode 100644
> index 000000000000..5dfe675709ab
> --- /dev/null
> +++ b/arch/loongarch/include/uapi/asm/kvm_para.h
> @@ -0,0 +1,24 @@
> +/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
> +#ifndef _UAPI_ASM_KVM_PARA_H
> +#define _UAPI_ASM_KVM_PARA_H
> +
> +#include <linux/types.h>
> +
> +/*
> + * CPUCFG index area: 0x40000000 -- 0x400000ff
> + * SW emulation for KVM hypervirsor
> + */
> +#define CPUCFG_KVM_BASE                        0x40000000
> +#define CPUCFG_KVM_SIZE                        0x100
> +#define CPUCFG_KVM_SIG                 (CPUCFG_KVM_BASE + 0)
> +#define  KVM_SIGNATURE                 "KVM\0"
> +#define CPUCFG_KVM_FEATURE             (CPUCFG_KVM_BASE + 4)
> +#define  KVM_FEATURE_IPI               1
> +#define  KVM_FEATURE_STEAL_TIME                2
> +/*
> + * BIT 24 - 31 is features configurable by user space vmm
> + * With VIRT_EXTIOI feature, interrupt can route to 256 VCPUs
> + */
> +#define  KVM_FEATURE_VIRT_EXTIOI       24
> +
> +#endif /* _UAPI_ASM_KVM_PARA_H */
> diff --git a/arch/loongarch/kernel/paravirt.c b/arch/loongarch/kernel/paravirt.c
> index 9c9b75b76f62..cc6bf096cb88 100644
> --- a/arch/loongarch/kernel/paravirt.c
> +++ b/arch/loongarch/kernel/paravirt.c
> @@ -175,7 +175,7 @@ int __init pv_ipi_init(void)
>                 return 0;
>
>         feature = read_cpucfg(CPUCFG_KVM_FEATURE);
> -       if (!(feature & KVM_FEATURE_IPI))
> +       if (!(feature & BIT(KVM_FEATURE_IPI)))
>                 return 0;
>
>  #ifdef CONFIG_SMP
> @@ -206,7 +206,7 @@ static int pv_enable_steal_time(void)
>         }
>
>         addr |= KVM_STEAL_PHYS_VALID;
> -       kvm_hypercall2(KVM_HCALL_FUNC_NOTIFY, KVM_FEATURE_STEAL_TIME, addr);
> +       kvm_hypercall2(KVM_HCALL_FUNC_NOTIFY, BIT(KVM_FEATURE_STEAL_TIME), addr);
>
>         return 0;
>  }
> @@ -214,7 +214,7 @@ static int pv_enable_steal_time(void)
>  static void pv_disable_steal_time(void)
>  {
>         if (has_steal_clock)
> -               kvm_hypercall2(KVM_HCALL_FUNC_NOTIFY, KVM_FEATURE_STEAL_TIME, 0);
> +               kvm_hypercall2(KVM_HCALL_FUNC_NOTIFY, BIT(KVM_FEATURE_STEAL_TIME), 0);
>  }
>
>  #ifdef CONFIG_SMP
> @@ -266,7 +266,7 @@ int __init pv_time_init(void)
>                 return 0;
>
>         feature = read_cpucfg(CPUCFG_KVM_FEATURE);
> -       if (!(feature & KVM_FEATURE_STEAL_TIME))
> +       if (!(feature & BIT(KVM_FEATURE_STEAL_TIME)))
>                 return 0;
>
>         has_steal_clock = 1;
> diff --git a/arch/loongarch/kvm/exit.c b/arch/loongarch/kvm/exit.c
> index ea73f9dc2cc6..54f78864a617 100644
> --- a/arch/loongarch/kvm/exit.c
> +++ b/arch/loongarch/kvm/exit.c
> @@ -50,9 +50,7 @@ static int kvm_emu_cpucfg(struct kvm_vcpu *vcpu, larch_inst inst)
>                 vcpu->arch.gprs[rd] = *(unsigned int *)KVM_SIGNATURE;
>                 break;
>         case CPUCFG_KVM_FEATURE:
> -               ret = KVM_FEATURE_IPI;
> -               if (kvm_pvtime_supported())
> -                       ret |= KVM_FEATURE_STEAL_TIME;
> +               ret = vcpu->kvm->arch.pv_features & LOONGARCH_PV_FEAT_MASK;
>                 vcpu->arch.gprs[rd] = ret;
>                 break;
>         default:
> @@ -697,8 +695,8 @@ static long kvm_save_notify(struct kvm_vcpu *vcpu)
>         id   = kvm_read_reg(vcpu, LOONGARCH_GPR_A1);
>         data = kvm_read_reg(vcpu, LOONGARCH_GPR_A2);
>         switch (id) {
> -       case KVM_FEATURE_STEAL_TIME:
> -               if (!kvm_pvtime_supported())
> +       case BIT(KVM_FEATURE_STEAL_TIME):
> +               if (!guest_pv_has(vcpu, KVM_FEATURE_STEAL_TIME))
>                         return KVM_HCALL_INVALID_CODE;
>
>                 if (data & ~(KVM_STEAL_PHYS_MASK | KVM_STEAL_PHYS_VALID))
> @@ -712,10 +710,10 @@ static long kvm_save_notify(struct kvm_vcpu *vcpu)
>                 kvm_make_request(KVM_REQ_STEAL_UPDATE, vcpu);
>                 break;
>         default:
> -               break;
> +               return KVM_HCALL_INVALID_CODE;
>         };
>
> -       return 0;
> +       return KVM_HCALL_INVALID_CODE;
>  };
>
>  /*
> @@ -786,8 +784,11 @@ static void kvm_handle_service(struct kvm_vcpu *vcpu)
>
>         switch (func) {
>         case KVM_HCALL_FUNC_IPI:
> -               kvm_send_pv_ipi(vcpu);
> -               ret = KVM_HCALL_SUCCESS;
> +               if (guest_pv_has(vcpu, KVM_FEATURE_IPI)) {
> +                       kvm_send_pv_ipi(vcpu);
> +                       ret = KVM_HCALL_SUCCESS;
> +               } else
> +                       ret = KVM_HCALL_INVALID_CODE;
>                 break;
>         case KVM_HCALL_FUNC_NOTIFY:
>                 ret = kvm_save_notify(vcpu);
> diff --git a/arch/loongarch/kvm/vcpu.c b/arch/loongarch/kvm/vcpu.c
> index 16756ffb55e8..2a7d7f91facd 100644
> --- a/arch/loongarch/kvm/vcpu.c
> +++ b/arch/loongarch/kvm/vcpu.c
> @@ -730,6 +730,8 @@ static int kvm_loongarch_cpucfg_has_attr(struct kvm_vcpu *vcpu,
>         switch (attr->attr) {
>         case 2:
>                 return 0;
> +       case CPUCFG_KVM_FEATURE:
> +               return 0;
>         default:
>                 return -ENXIO;
>         }
> @@ -740,7 +742,7 @@ static int kvm_loongarch_cpucfg_has_attr(struct kvm_vcpu *vcpu,
>  static int kvm_loongarch_pvtime_has_attr(struct kvm_vcpu *vcpu,
>                                          struct kvm_device_attr *attr)
>  {
> -       if (!kvm_pvtime_supported() ||
> +       if (!guest_pv_has(vcpu, KVM_FEATURE_STEAL_TIME) ||
>                         attr->attr != KVM_LOONGARCH_VCPU_PVTIME_GPA)
>                 return -ENXIO;
>
> @@ -773,9 +775,18 @@ static int kvm_loongarch_cpucfg_get_attr(struct kvm_vcpu *vcpu,
>         uint64_t val;
>         uint64_t __user *uaddr = (uint64_t __user *)attr->addr;
>
> -       ret = _kvm_get_cpucfg_mask(attr->attr, &val);
> -       if (ret)
> -               return ret;
> +       switch (attr->attr) {
> +       case 0 ... (KVM_MAX_CPUCFG_REGS - 1):
> +               ret = _kvm_get_cpucfg_mask(attr->attr, &val);
> +               if (ret)
> +                       return ret;
> +               break;
> +       case CPUCFG_KVM_FEATURE:
> +               val = vcpu->kvm->arch.pv_features & LOONGARCH_PV_FEAT_MASK;
> +               break;
> +       default:
> +               return -ENXIO;
> +       }
>
>         put_user(val, uaddr);
>
> @@ -788,7 +799,7 @@ static int kvm_loongarch_pvtime_get_attr(struct kvm_vcpu *vcpu,
>         u64 gpa;
>         u64 __user *user = (u64 __user *)attr->addr;
>
> -       if (!kvm_pvtime_supported() ||
> +       if (!guest_pv_has(vcpu, KVM_FEATURE_STEAL_TIME) ||
>                         attr->attr != KVM_LOONGARCH_VCPU_PVTIME_GPA)
>                 return -ENXIO;
>
> @@ -821,7 +832,29 @@ static int kvm_loongarch_vcpu_get_attr(struct kvm_vcpu *vcpu,
>  static int kvm_loongarch_cpucfg_set_attr(struct kvm_vcpu *vcpu,
>                                          struct kvm_device_attr *attr)
>  {
> -       return -ENXIO;
> +       u64 __user *user = (u64 __user *)attr->addr;
> +       u64 val, valid;
> +       struct kvm *kvm = vcpu->kvm;
> +
> +       switch (attr->attr) {
> +       case CPUCFG_KVM_FEATURE:
> +               if (get_user(val, user))
> +                       return -EFAULT;
> +
> +               valid = LOONGARCH_PV_FEAT_MASK;
> +               if (val & ~valid)
> +                       return -EINVAL;
> +
> +               /* All vCPUs need set the same pv features */
> +               if ((kvm->arch.pv_features & LOONGARCH_PV_FEAT_UPDATED) &&
> +                               ((kvm->arch.pv_features & valid) != val))
> +                       return -EINVAL;
> +               kvm->arch.pv_features = val | LOONGARCH_PV_FEAT_UPDATED;
> +               return 0;
> +
> +       default:
> +               return -ENXIO;
> +       }
>  }
>
>  static int kvm_loongarch_pvtime_set_attr(struct kvm_vcpu *vcpu,
> @@ -831,7 +864,7 @@ static int kvm_loongarch_pvtime_set_attr(struct kvm_vcpu *vcpu,
>         u64 gpa, __user *user = (u64 __user *)attr->addr;
>         struct kvm *kvm = vcpu->kvm;
>
> -       if (!kvm_pvtime_supported() ||
> +       if (!guest_pv_has(vcpu, KVM_FEATURE_STEAL_TIME) ||
>                         attr->attr != KVM_LOONGARCH_VCPU_PVTIME_GPA)
>                 return -ENXIO;
>
> diff --git a/arch/loongarch/kvm/vm.c b/arch/loongarch/kvm/vm.c
> index 6b2e4f66ad26..3234f3e85dc0 100644
> --- a/arch/loongarch/kvm/vm.c
> +++ b/arch/loongarch/kvm/vm.c
> @@ -5,6 +5,7 @@
>
>  #include <linux/kvm_host.h>
>  #include <asm/kvm_mmu.h>
> +#include <asm/kvm_vcpu.h>
>
>  const struct _kvm_stats_desc kvm_vm_stats_desc[] = {
>         KVM_GENERIC_VM_STATS(),
> @@ -39,6 +40,10 @@ int kvm_arch_init_vm(struct kvm *kvm, unsigned long type)
>         spin_lock_init(&kvm->arch.phyid_map_lock);
>
>         kvm_init_vmcs(kvm);
> +       /* Enable all pv features by default */
> +       kvm->arch.pv_features = BIT(KVM_FEATURE_IPI);
> +       if (kvm_pvtime_supported())
> +               kvm->arch.pv_features |= BIT(KVM_FEATURE_STEAL_TIME);
>         kvm->arch.gpa_size = BIT(cpu_vabits - 1);
>         kvm->arch.root_level = CONFIG_PGTABLE_LEVELS - 1;
>         kvm->arch.invalid_ptes[0] = 0;
> @@ -99,7 +104,43 @@ int kvm_vm_ioctl_check_extension(struct kvm *kvm, long ext)
>         return r;
>  }
>
> +static int kvm_vm_feature_has_attr(struct kvm *kvm, struct kvm_device_attr *attr)
> +{
> +       switch (attr->attr) {
> +       case KVM_LOONGARCH_VM_FEAT_PV_IPI:
> +               return 0;
> +       case KVM_LOONGARCH_VM_FEAT_PV_STEALTIME:
> +               if (kvm_pvtime_supported())
> +                       return 0;
> +               return -ENXIO;
> +       default:
> +               return -ENXIO;
> +       }
> +}
> +
> +static int kvm_vm_has_attr(struct kvm *kvm, struct kvm_device_attr *attr)
> +{
> +       switch (attr->group) {
> +       case KVM_LOONGARCH_VM_FEAT_CTRL:
> +               return kvm_vm_feature_has_attr(kvm, attr);
> +       default:
> +               return -ENXIO;
> +       }
> +}
> +
>  int kvm_arch_vm_ioctl(struct file *filp, unsigned int ioctl, unsigned long arg)
>  {
> -       return -ENOIOCTLCMD;
> +       struct kvm *kvm = filp->private_data;
> +       void __user *argp = (void __user *)arg;
> +       struct kvm_device_attr attr;
> +
> +       switch (ioctl) {
> +       case KVM_HAS_DEVICE_ATTR:
> +               if (copy_from_user(&attr, argp, sizeof(attr)))
> +                       return -EFAULT;
> +
> +               return kvm_vm_has_attr(kvm, &attr);
> +       default:
> +               return -EINVAL;
> +       }
>  }
> --
> 2.39.3
>
>





[Index of Archives]     [KVM Development]     [Libvirt Development]     [Libvirt Users]     [CentOS Virtualization]     [Netdev]     [Ethernet Bridging]     [Linux Wireless]     [Kernel Newbies]     [Security]     [Linux for Hams]     [Netfilter]     [Bugtraq]     [Yosemite Forum]     [MIPS Linux]     [ARM Linux]     [Linux RAID]     [Linux Admin]     [Samba]

  Powered by Linux