MIPS: KVM: Expose MSA registers
Add KVM register numbers for the MIPS SIMD Architecture (MSA) registers, and implement access to them with the KVM_GET_ONE_REG / KVM_SET_ONE_REG ioctls when the MSA capability is enabled (exposed in a later patch) and present in the guest according to its Config3.MSAP bit. The MSA vector registers use the same register numbers as the FPU registers except with a different size (128bits). Since MSA depends on Status.FR=1, these registers are inaccessible when Status.FR=0. These registers are returned as a single native endian 128bit value, rather than least significant half first with each 64-bit half native endian as the kernel uses internally. Signed-off-by: James Hogan <james.hogan@imgtec.com> Cc: Paolo Bonzini <pbonzini@redhat.com> Cc: Paul Burton <paul.burton@imgtec.com> Cc: Ralf Baechle <ralf@linux-mips.org> Cc: Gleb Natapov <gleb@kernel.org> Cc: Jonathan Corbet <corbet@lwn.net> Cc: linux-mips@linux-mips.org Cc: kvm@vger.kernel.org Cc: linux-api@vger.kernel.org Cc: linux-doc@vger.kernel.org
This commit is contained in:
parent
c2537ed9fb
commit
ab86bd6004
|
@ -1981,8 +1981,11 @@ registers, find a list below:
|
||||||
MIPS | KVM_REG_MIPS_COUNT_HZ | 64
|
MIPS | KVM_REG_MIPS_COUNT_HZ | 64
|
||||||
MIPS | KVM_REG_MIPS_FPR_32(0..31) | 32
|
MIPS | KVM_REG_MIPS_FPR_32(0..31) | 32
|
||||||
MIPS | KVM_REG_MIPS_FPR_64(0..31) | 64
|
MIPS | KVM_REG_MIPS_FPR_64(0..31) | 64
|
||||||
|
MIPS | KVM_REG_MIPS_VEC_128(0..31) | 128
|
||||||
MIPS | KVM_REG_MIPS_FCR_IR | 32
|
MIPS | KVM_REG_MIPS_FCR_IR | 32
|
||||||
MIPS | KVM_REG_MIPS_FCR_CSR | 32
|
MIPS | KVM_REG_MIPS_FCR_CSR | 32
|
||||||
|
MIPS | KVM_REG_MIPS_MSA_IR | 32
|
||||||
|
MIPS | KVM_REG_MIPS_MSA_CSR | 32
|
||||||
|
|
||||||
ARM registers are mapped using the lower 32 bits. The upper 16 of that
|
ARM registers are mapped using the lower 32 bits. The upper 16 of that
|
||||||
is the register group type, or coprocessor number:
|
is the register group type, or coprocessor number:
|
||||||
|
@ -2040,14 +2043,21 @@ MIPS FPU registers (see KVM_REG_MIPS_FPR_{32,64}() above) have the following
|
||||||
id bit patterns depending on the size of the register being accessed. They are
|
id bit patterns depending on the size of the register being accessed. They are
|
||||||
always accessed according to the current guest FPU mode (Status.FR and
|
always accessed according to the current guest FPU mode (Status.FR and
|
||||||
Config5.FRE), i.e. as the guest would see them, and they become unpredictable
|
Config5.FRE), i.e. as the guest would see them, and they become unpredictable
|
||||||
if the guest FPU mode is changed:
|
if the guest FPU mode is changed. MIPS SIMD Architecture (MSA) vector
|
||||||
|
registers (see KVM_REG_MIPS_VEC_128() above) have similar patterns as they
|
||||||
|
overlap the FPU registers:
|
||||||
0x7020 0000 0003 00 <0:3> <reg:5> (32-bit FPU registers)
|
0x7020 0000 0003 00 <0:3> <reg:5> (32-bit FPU registers)
|
||||||
0x7030 0000 0003 00 <0:3> <reg:5> (64-bit FPU registers)
|
0x7030 0000 0003 00 <0:3> <reg:5> (64-bit FPU registers)
|
||||||
|
0x7040 0000 0003 00 <0:3> <reg:5> (128-bit MSA vector registers)
|
||||||
|
|
||||||
MIPS FPU control registers (see KVM_REG_MIPS_FCR_{IR,CSR} above) have the
|
MIPS FPU control registers (see KVM_REG_MIPS_FCR_{IR,CSR} above) have the
|
||||||
following id bit patterns:
|
following id bit patterns:
|
||||||
0x7020 0000 0003 01 <0:3> <reg:5>
|
0x7020 0000 0003 01 <0:3> <reg:5>
|
||||||
|
|
||||||
|
MIPS MSA control registers (see KVM_REG_MIPS_MSA_{IR,CSR} above) have the
|
||||||
|
following id bit patterns:
|
||||||
|
0x7020 0000 0003 02 <0:3> <reg:5>
|
||||||
|
|
||||||
|
|
||||||
4.69 KVM_GET_ONE_REG
|
4.69 KVM_GET_ONE_REG
|
||||||
|
|
||||||
|
|
|
@ -58,7 +58,7 @@ struct kvm_fpu {
|
||||||
*
|
*
|
||||||
* Register set = 2: KVM specific registers (see definitions below).
|
* Register set = 2: KVM specific registers (see definitions below).
|
||||||
*
|
*
|
||||||
* Register set = 3: FPU registers (see definitions below).
|
* Register set = 3: FPU / MSA registers (see definitions below).
|
||||||
*
|
*
|
||||||
* Other sets registers may be added in the future. Each set would
|
* Other sets registers may be added in the future. Each set would
|
||||||
* have its own identifier in bits[31..16].
|
* have its own identifier in bits[31..16].
|
||||||
|
@ -148,7 +148,7 @@ struct kvm_fpu {
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* KVM_REG_MIPS_FPU - Floating Point registers.
|
* KVM_REG_MIPS_FPU - Floating Point and MIPS SIMD Architecture (MSA) registers.
|
||||||
*
|
*
|
||||||
* bits[15..8] - Register subset (see definitions below).
|
* bits[15..8] - Register subset (see definitions below).
|
||||||
* bits[7..5] - Must be zero.
|
* bits[7..5] - Must be zero.
|
||||||
|
@ -157,12 +157,14 @@ struct kvm_fpu {
|
||||||
|
|
||||||
#define KVM_REG_MIPS_FPR (KVM_REG_MIPS_FPU | 0x0000000000000000ULL)
|
#define KVM_REG_MIPS_FPR (KVM_REG_MIPS_FPU | 0x0000000000000000ULL)
|
||||||
#define KVM_REG_MIPS_FCR (KVM_REG_MIPS_FPU | 0x0000000000000100ULL)
|
#define KVM_REG_MIPS_FCR (KVM_REG_MIPS_FPU | 0x0000000000000100ULL)
|
||||||
|
#define KVM_REG_MIPS_MSACR (KVM_REG_MIPS_FPU | 0x0000000000000200ULL)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* KVM_REG_MIPS_FPR - Floating point / Vector registers.
|
* KVM_REG_MIPS_FPR - Floating point / Vector registers.
|
||||||
*/
|
*/
|
||||||
#define KVM_REG_MIPS_FPR_32(n) (KVM_REG_MIPS_FPR | KVM_REG_SIZE_U32 | (n))
|
#define KVM_REG_MIPS_FPR_32(n) (KVM_REG_MIPS_FPR | KVM_REG_SIZE_U32 | (n))
|
||||||
#define KVM_REG_MIPS_FPR_64(n) (KVM_REG_MIPS_FPR | KVM_REG_SIZE_U64 | (n))
|
#define KVM_REG_MIPS_FPR_64(n) (KVM_REG_MIPS_FPR | KVM_REG_SIZE_U64 | (n))
|
||||||
|
#define KVM_REG_MIPS_VEC_128(n) (KVM_REG_MIPS_FPR | KVM_REG_SIZE_U128 | (n))
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* KVM_REG_MIPS_FCR - Floating point control registers.
|
* KVM_REG_MIPS_FCR - Floating point control registers.
|
||||||
|
@ -170,6 +172,12 @@ struct kvm_fpu {
|
||||||
#define KVM_REG_MIPS_FCR_IR (KVM_REG_MIPS_FCR | KVM_REG_SIZE_U32 | 0)
|
#define KVM_REG_MIPS_FCR_IR (KVM_REG_MIPS_FCR | KVM_REG_SIZE_U32 | 0)
|
||||||
#define KVM_REG_MIPS_FCR_CSR (KVM_REG_MIPS_FCR | KVM_REG_SIZE_U32 | 31)
|
#define KVM_REG_MIPS_FCR_CSR (KVM_REG_MIPS_FCR | KVM_REG_SIZE_U32 | 31)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* KVM_REG_MIPS_MSACR - MIPS SIMD Architecture (MSA) control registers.
|
||||||
|
*/
|
||||||
|
#define KVM_REG_MIPS_MSA_IR (KVM_REG_MIPS_MSACR | KVM_REG_SIZE_U32 | 0)
|
||||||
|
#define KVM_REG_MIPS_MSA_CSR (KVM_REG_MIPS_MSACR | KVM_REG_SIZE_U32 | 1)
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* KVM MIPS specific structures and definitions
|
* KVM MIPS specific structures and definitions
|
||||||
|
|
|
@ -531,6 +531,7 @@ static int kvm_mips_get_reg(struct kvm_vcpu *vcpu,
|
||||||
struct mips_fpu_struct *fpu = &vcpu->arch.fpu;
|
struct mips_fpu_struct *fpu = &vcpu->arch.fpu;
|
||||||
int ret;
|
int ret;
|
||||||
s64 v;
|
s64 v;
|
||||||
|
s64 vs[2];
|
||||||
unsigned int idx;
|
unsigned int idx;
|
||||||
|
|
||||||
switch (reg->id) {
|
switch (reg->id) {
|
||||||
|
@ -579,6 +580,35 @@ static int kvm_mips_get_reg(struct kvm_vcpu *vcpu,
|
||||||
v = fpu->fcr31;
|
v = fpu->fcr31;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
/* MIPS SIMD Architecture (MSA) registers */
|
||||||
|
case KVM_REG_MIPS_VEC_128(0) ... KVM_REG_MIPS_VEC_128(31):
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
/* Can't access MSA registers in FR=0 mode */
|
||||||
|
if (!(kvm_read_c0_guest_status(cop0) & ST0_FR))
|
||||||
|
return -EINVAL;
|
||||||
|
idx = reg->id - KVM_REG_MIPS_VEC_128(0);
|
||||||
|
#ifdef CONFIG_CPU_LITTLE_ENDIAN
|
||||||
|
/* least significant byte first */
|
||||||
|
vs[0] = get_fpr64(&fpu->fpr[idx], 0);
|
||||||
|
vs[1] = get_fpr64(&fpu->fpr[idx], 1);
|
||||||
|
#else
|
||||||
|
/* most significant byte first */
|
||||||
|
vs[0] = get_fpr64(&fpu->fpr[idx], 1);
|
||||||
|
vs[1] = get_fpr64(&fpu->fpr[idx], 0);
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
|
case KVM_REG_MIPS_MSA_IR:
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
v = boot_cpu_data.msa_id;
|
||||||
|
break;
|
||||||
|
case KVM_REG_MIPS_MSA_CSR:
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
v = fpu->msacsr;
|
||||||
|
break;
|
||||||
|
|
||||||
/* Co-processor 0 registers */
|
/* Co-processor 0 registers */
|
||||||
case KVM_REG_MIPS_CP0_INDEX:
|
case KVM_REG_MIPS_CP0_INDEX:
|
||||||
v = (long)kvm_read_c0_guest_index(cop0);
|
v = (long)kvm_read_c0_guest_index(cop0);
|
||||||
|
@ -664,6 +694,10 @@ static int kvm_mips_get_reg(struct kvm_vcpu *vcpu,
|
||||||
u32 v32 = (u32)v;
|
u32 v32 = (u32)v;
|
||||||
|
|
||||||
return put_user(v32, uaddr32);
|
return put_user(v32, uaddr32);
|
||||||
|
} else if ((reg->id & KVM_REG_SIZE_MASK) == KVM_REG_SIZE_U128) {
|
||||||
|
void __user *uaddr = (void __user *)(long)reg->addr;
|
||||||
|
|
||||||
|
return copy_to_user(uaddr, vs, 16);
|
||||||
} else {
|
} else {
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
@ -675,6 +709,7 @@ static int kvm_mips_set_reg(struct kvm_vcpu *vcpu,
|
||||||
struct mips_coproc *cop0 = vcpu->arch.cop0;
|
struct mips_coproc *cop0 = vcpu->arch.cop0;
|
||||||
struct mips_fpu_struct *fpu = &vcpu->arch.fpu;
|
struct mips_fpu_struct *fpu = &vcpu->arch.fpu;
|
||||||
s64 v;
|
s64 v;
|
||||||
|
s64 vs[2];
|
||||||
unsigned int idx;
|
unsigned int idx;
|
||||||
|
|
||||||
if ((reg->id & KVM_REG_SIZE_MASK) == KVM_REG_SIZE_U64) {
|
if ((reg->id & KVM_REG_SIZE_MASK) == KVM_REG_SIZE_U64) {
|
||||||
|
@ -689,6 +724,10 @@ static int kvm_mips_set_reg(struct kvm_vcpu *vcpu,
|
||||||
if (get_user(v32, uaddr32) != 0)
|
if (get_user(v32, uaddr32) != 0)
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
v = (s64)v32;
|
v = (s64)v32;
|
||||||
|
} else if ((reg->id & KVM_REG_SIZE_MASK) == KVM_REG_SIZE_U128) {
|
||||||
|
void __user *uaddr = (void __user *)(long)reg->addr;
|
||||||
|
|
||||||
|
return copy_from_user(vs, uaddr, 16);
|
||||||
} else {
|
} else {
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
@ -742,6 +781,32 @@ static int kvm_mips_set_reg(struct kvm_vcpu *vcpu,
|
||||||
fpu->fcr31 = v;
|
fpu->fcr31 = v;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
/* MIPS SIMD Architecture (MSA) registers */
|
||||||
|
case KVM_REG_MIPS_VEC_128(0) ... KVM_REG_MIPS_VEC_128(31):
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
idx = reg->id - KVM_REG_MIPS_VEC_128(0);
|
||||||
|
#ifdef CONFIG_CPU_LITTLE_ENDIAN
|
||||||
|
/* least significant byte first */
|
||||||
|
set_fpr64(&fpu->fpr[idx], 0, vs[0]);
|
||||||
|
set_fpr64(&fpu->fpr[idx], 1, vs[1]);
|
||||||
|
#else
|
||||||
|
/* most significant byte first */
|
||||||
|
set_fpr64(&fpu->fpr[idx], 1, vs[0]);
|
||||||
|
set_fpr64(&fpu->fpr[idx], 0, vs[1]);
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
|
case KVM_REG_MIPS_MSA_IR:
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
/* Read-only */
|
||||||
|
break;
|
||||||
|
case KVM_REG_MIPS_MSA_CSR:
|
||||||
|
if (!kvm_mips_guest_has_msa(&vcpu->arch))
|
||||||
|
return -EINVAL;
|
||||||
|
fpu->msacsr = v;
|
||||||
|
break;
|
||||||
|
|
||||||
/* Co-processor 0 registers */
|
/* Co-processor 0 registers */
|
||||||
case KVM_REG_MIPS_CP0_INDEX:
|
case KVM_REG_MIPS_CP0_INDEX:
|
||||||
kvm_write_c0_guest_index(cop0, v);
|
kvm_write_c0_guest_index(cop0, v);
|
||||||
|
|
Loading…
Reference in New Issue