]> git.baikalelectronics.ru Git - kernel.git/commitdiff
RISC-V: KVM: FP lazy save/restore
authorAtish Patra <atish.patra@wdc.com>
Mon, 27 Sep 2021 11:40:12 +0000 (17:10 +0530)
committerAnup Patel <anup@brainfault.org>
Mon, 4 Oct 2021 10:38:23 +0000 (16:08 +0530)
This patch adds floating point (F and D extension) context save/restore
for guest VCPUs. The FP context is saved and restored lazily only when
kernel enter/exits the in-kernel run loop and not during the KVM world
switch. This way FP save/restore has minimal impact on KVM performance.

Signed-off-by: Atish Patra <atish.patra@wdc.com>
Signed-off-by: Anup Patel <anup.patel@wdc.com>
Acked-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Reviewed-by: Alexander Graf <graf@amazon.com>
Acked-by: Palmer Dabbelt <palmerdabbelt@google.com>
arch/riscv/include/asm/kvm_host.h
arch/riscv/kernel/asm-offsets.c
arch/riscv/kvm/vcpu.c
arch/riscv/kvm/vcpu_switch.S

index c22d3f6519fa92524bf9b66f09571833019625d0..be159686da460be55abd5323ac2c6442fa82acb5 100644 (file)
@@ -125,6 +125,7 @@ struct kvm_cpu_context {
        unsigned long sepc;
        unsigned long sstatus;
        unsigned long hstatus;
+       union __riscv_fp_state fp;
 };
 
 struct kvm_vcpu_csr {
@@ -239,6 +240,10 @@ int kvm_riscv_vcpu_exit(struct kvm_vcpu *vcpu, struct kvm_run *run,
                        struct kvm_cpu_trap *trap);
 
 void __kvm_riscv_switch_to(struct kvm_vcpu_arch *vcpu_arch);
+void __kvm_riscv_fp_f_save(struct kvm_cpu_context *context);
+void __kvm_riscv_fp_f_restore(struct kvm_cpu_context *context);
+void __kvm_riscv_fp_d_save(struct kvm_cpu_context *context);
+void __kvm_riscv_fp_d_restore(struct kvm_cpu_context *context);
 
 int kvm_riscv_vcpu_set_interrupt(struct kvm_vcpu *vcpu, unsigned int irq);
 int kvm_riscv_vcpu_unset_interrupt(struct kvm_vcpu *vcpu, unsigned int irq);
index 91c77555d914541b79f8f07c9d94ea20808063e6..24d3827e48379995cb1e17fb63acdd2867e70814 100644 (file)
@@ -195,6 +195,78 @@ void asm_offsets(void)
        OFFSET(KVM_ARCH_TRAP_HTVAL, kvm_cpu_trap, htval);
        OFFSET(KVM_ARCH_TRAP_HTINST, kvm_cpu_trap, htinst);
 
+       /* F extension */
+
+       OFFSET(KVM_ARCH_FP_F_F0, kvm_cpu_context, fp.f.f[0]);
+       OFFSET(KVM_ARCH_FP_F_F1, kvm_cpu_context, fp.f.f[1]);
+       OFFSET(KVM_ARCH_FP_F_F2, kvm_cpu_context, fp.f.f[2]);
+       OFFSET(KVM_ARCH_FP_F_F3, kvm_cpu_context, fp.f.f[3]);
+       OFFSET(KVM_ARCH_FP_F_F4, kvm_cpu_context, fp.f.f[4]);
+       OFFSET(KVM_ARCH_FP_F_F5, kvm_cpu_context, fp.f.f[5]);
+       OFFSET(KVM_ARCH_FP_F_F6, kvm_cpu_context, fp.f.f[6]);
+       OFFSET(KVM_ARCH_FP_F_F7, kvm_cpu_context, fp.f.f[7]);
+       OFFSET(KVM_ARCH_FP_F_F8, kvm_cpu_context, fp.f.f[8]);
+       OFFSET(KVM_ARCH_FP_F_F9, kvm_cpu_context, fp.f.f[9]);
+       OFFSET(KVM_ARCH_FP_F_F10, kvm_cpu_context, fp.f.f[10]);
+       OFFSET(KVM_ARCH_FP_F_F11, kvm_cpu_context, fp.f.f[11]);
+       OFFSET(KVM_ARCH_FP_F_F12, kvm_cpu_context, fp.f.f[12]);
+       OFFSET(KVM_ARCH_FP_F_F13, kvm_cpu_context, fp.f.f[13]);
+       OFFSET(KVM_ARCH_FP_F_F14, kvm_cpu_context, fp.f.f[14]);
+       OFFSET(KVM_ARCH_FP_F_F15, kvm_cpu_context, fp.f.f[15]);
+       OFFSET(KVM_ARCH_FP_F_F16, kvm_cpu_context, fp.f.f[16]);
+       OFFSET(KVM_ARCH_FP_F_F17, kvm_cpu_context, fp.f.f[17]);
+       OFFSET(KVM_ARCH_FP_F_F18, kvm_cpu_context, fp.f.f[18]);
+       OFFSET(KVM_ARCH_FP_F_F19, kvm_cpu_context, fp.f.f[19]);
+       OFFSET(KVM_ARCH_FP_F_F20, kvm_cpu_context, fp.f.f[20]);
+       OFFSET(KVM_ARCH_FP_F_F21, kvm_cpu_context, fp.f.f[21]);
+       OFFSET(KVM_ARCH_FP_F_F22, kvm_cpu_context, fp.f.f[22]);
+       OFFSET(KVM_ARCH_FP_F_F23, kvm_cpu_context, fp.f.f[23]);
+       OFFSET(KVM_ARCH_FP_F_F24, kvm_cpu_context, fp.f.f[24]);
+       OFFSET(KVM_ARCH_FP_F_F25, kvm_cpu_context, fp.f.f[25]);
+       OFFSET(KVM_ARCH_FP_F_F26, kvm_cpu_context, fp.f.f[26]);
+       OFFSET(KVM_ARCH_FP_F_F27, kvm_cpu_context, fp.f.f[27]);
+       OFFSET(KVM_ARCH_FP_F_F28, kvm_cpu_context, fp.f.f[28]);
+       OFFSET(KVM_ARCH_FP_F_F29, kvm_cpu_context, fp.f.f[29]);
+       OFFSET(KVM_ARCH_FP_F_F30, kvm_cpu_context, fp.f.f[30]);
+       OFFSET(KVM_ARCH_FP_F_F31, kvm_cpu_context, fp.f.f[31]);
+       OFFSET(KVM_ARCH_FP_F_FCSR, kvm_cpu_context, fp.f.fcsr);
+
+       /* D extension */
+
+       OFFSET(KVM_ARCH_FP_D_F0, kvm_cpu_context, fp.d.f[0]);
+       OFFSET(KVM_ARCH_FP_D_F1, kvm_cpu_context, fp.d.f[1]);
+       OFFSET(KVM_ARCH_FP_D_F2, kvm_cpu_context, fp.d.f[2]);
+       OFFSET(KVM_ARCH_FP_D_F3, kvm_cpu_context, fp.d.f[3]);
+       OFFSET(KVM_ARCH_FP_D_F4, kvm_cpu_context, fp.d.f[4]);
+       OFFSET(KVM_ARCH_FP_D_F5, kvm_cpu_context, fp.d.f[5]);
+       OFFSET(KVM_ARCH_FP_D_F6, kvm_cpu_context, fp.d.f[6]);
+       OFFSET(KVM_ARCH_FP_D_F7, kvm_cpu_context, fp.d.f[7]);
+       OFFSET(KVM_ARCH_FP_D_F8, kvm_cpu_context, fp.d.f[8]);
+       OFFSET(KVM_ARCH_FP_D_F9, kvm_cpu_context, fp.d.f[9]);
+       OFFSET(KVM_ARCH_FP_D_F10, kvm_cpu_context, fp.d.f[10]);
+       OFFSET(KVM_ARCH_FP_D_F11, kvm_cpu_context, fp.d.f[11]);
+       OFFSET(KVM_ARCH_FP_D_F12, kvm_cpu_context, fp.d.f[12]);
+       OFFSET(KVM_ARCH_FP_D_F13, kvm_cpu_context, fp.d.f[13]);
+       OFFSET(KVM_ARCH_FP_D_F14, kvm_cpu_context, fp.d.f[14]);
+       OFFSET(KVM_ARCH_FP_D_F15, kvm_cpu_context, fp.d.f[15]);
+       OFFSET(KVM_ARCH_FP_D_F16, kvm_cpu_context, fp.d.f[16]);
+       OFFSET(KVM_ARCH_FP_D_F17, kvm_cpu_context, fp.d.f[17]);
+       OFFSET(KVM_ARCH_FP_D_F18, kvm_cpu_context, fp.d.f[18]);
+       OFFSET(KVM_ARCH_FP_D_F19, kvm_cpu_context, fp.d.f[19]);
+       OFFSET(KVM_ARCH_FP_D_F20, kvm_cpu_context, fp.d.f[20]);
+       OFFSET(KVM_ARCH_FP_D_F21, kvm_cpu_context, fp.d.f[21]);
+       OFFSET(KVM_ARCH_FP_D_F22, kvm_cpu_context, fp.d.f[22]);
+       OFFSET(KVM_ARCH_FP_D_F23, kvm_cpu_context, fp.d.f[23]);
+       OFFSET(KVM_ARCH_FP_D_F24, kvm_cpu_context, fp.d.f[24]);
+       OFFSET(KVM_ARCH_FP_D_F25, kvm_cpu_context, fp.d.f[25]);
+       OFFSET(KVM_ARCH_FP_D_F26, kvm_cpu_context, fp.d.f[26]);
+       OFFSET(KVM_ARCH_FP_D_F27, kvm_cpu_context, fp.d.f[27]);
+       OFFSET(KVM_ARCH_FP_D_F28, kvm_cpu_context, fp.d.f[28]);
+       OFFSET(KVM_ARCH_FP_D_F29, kvm_cpu_context, fp.d.f[29]);
+       OFFSET(KVM_ARCH_FP_D_F30, kvm_cpu_context, fp.d.f[30]);
+       OFFSET(KVM_ARCH_FP_D_F31, kvm_cpu_context, fp.d.f[31]);
+       OFFSET(KVM_ARCH_FP_D_FCSR, kvm_cpu_context, fp.d.fcsr);
+
        /*
         * THREAD_{F,X}* might be larger than a S-type offset can handle, but
         * these are used in performance-sensitive assembly so we can't resort
index 840f4586796f7d803857d93ffab6afce5e8e2ac9..cf6832365345f8528d8f31140acc6b20680ffbf0 100644 (file)
@@ -38,6 +38,86 @@ const struct kvm_stats_header kvm_vcpu_stats_header = {
                       sizeof(kvm_vcpu_stats_desc),
 };
 
+#ifdef CONFIG_FPU
+static void kvm_riscv_vcpu_fp_reset(struct kvm_vcpu *vcpu)
+{
+       unsigned long isa = vcpu->arch.isa;
+       struct kvm_cpu_context *cntx = &vcpu->arch.guest_context;
+
+       cntx->sstatus &= ~SR_FS;
+       if (riscv_isa_extension_available(&isa, f) ||
+           riscv_isa_extension_available(&isa, d))
+               cntx->sstatus |= SR_FS_INITIAL;
+       else
+               cntx->sstatus |= SR_FS_OFF;
+}
+
+static void kvm_riscv_vcpu_fp_clean(struct kvm_cpu_context *cntx)
+{
+       cntx->sstatus &= ~SR_FS;
+       cntx->sstatus |= SR_FS_CLEAN;
+}
+
+static void kvm_riscv_vcpu_guest_fp_save(struct kvm_cpu_context *cntx,
+                                        unsigned long isa)
+{
+       if ((cntx->sstatus & SR_FS) == SR_FS_DIRTY) {
+               if (riscv_isa_extension_available(&isa, d))
+                       __kvm_riscv_fp_d_save(cntx);
+               else if (riscv_isa_extension_available(&isa, f))
+                       __kvm_riscv_fp_f_save(cntx);
+               kvm_riscv_vcpu_fp_clean(cntx);
+       }
+}
+
+static void kvm_riscv_vcpu_guest_fp_restore(struct kvm_cpu_context *cntx,
+                                           unsigned long isa)
+{
+       if ((cntx->sstatus & SR_FS) != SR_FS_OFF) {
+               if (riscv_isa_extension_available(&isa, d))
+                       __kvm_riscv_fp_d_restore(cntx);
+               else if (riscv_isa_extension_available(&isa, f))
+                       __kvm_riscv_fp_f_restore(cntx);
+               kvm_riscv_vcpu_fp_clean(cntx);
+       }
+}
+
+static void kvm_riscv_vcpu_host_fp_save(struct kvm_cpu_context *cntx)
+{
+       /* No need to check host sstatus as it can be modified outside */
+       if (riscv_isa_extension_available(NULL, d))
+               __kvm_riscv_fp_d_save(cntx);
+       else if (riscv_isa_extension_available(NULL, f))
+               __kvm_riscv_fp_f_save(cntx);
+}
+
+static void kvm_riscv_vcpu_host_fp_restore(struct kvm_cpu_context *cntx)
+{
+       if (riscv_isa_extension_available(NULL, d))
+               __kvm_riscv_fp_d_restore(cntx);
+       else if (riscv_isa_extension_available(NULL, f))
+               __kvm_riscv_fp_f_restore(cntx);
+}
+#else
+static void kvm_riscv_vcpu_fp_reset(struct kvm_vcpu *vcpu)
+{
+}
+static void kvm_riscv_vcpu_guest_fp_save(struct kvm_cpu_context *cntx,
+                                        unsigned long isa)
+{
+}
+static void kvm_riscv_vcpu_guest_fp_restore(struct kvm_cpu_context *cntx,
+                                           unsigned long isa)
+{
+}
+static void kvm_riscv_vcpu_host_fp_save(struct kvm_cpu_context *cntx)
+{
+}
+static void kvm_riscv_vcpu_host_fp_restore(struct kvm_cpu_context *cntx)
+{
+}
+#endif
+
 #define KVM_RISCV_ISA_ALLOWED  (riscv_isa_extension_mask(a) | \
                                 riscv_isa_extension_mask(c) | \
                                 riscv_isa_extension_mask(d) | \
@@ -58,6 +138,8 @@ static void kvm_riscv_reset_vcpu(struct kvm_vcpu *vcpu)
 
        memcpy(cntx, reset_cntx, sizeof(*cntx));
 
+       kvm_riscv_vcpu_fp_reset(vcpu);
+
        kvm_riscv_vcpu_timer_reset(vcpu);
 
        WRITE_ONCE(vcpu->arch.irqs_pending, 0);
@@ -192,6 +274,7 @@ static int kvm_riscv_vcpu_set_reg_config(struct kvm_vcpu *vcpu,
                        vcpu->arch.isa = reg_val;
                        vcpu->arch.isa &= riscv_isa_extension_base(NULL);
                        vcpu->arch.isa &= KVM_RISCV_ISA_ALLOWED;
+                       kvm_riscv_vcpu_fp_reset(vcpu);
                } else {
                        return -EOPNOTSUPP;
                }
@@ -593,6 +676,10 @@ void kvm_arch_vcpu_load(struct kvm_vcpu *vcpu, int cpu)
 
        kvm_riscv_vcpu_timer_restore(vcpu);
 
+       kvm_riscv_vcpu_host_fp_save(&vcpu->arch.host_context);
+       kvm_riscv_vcpu_guest_fp_restore(&vcpu->arch.guest_context,
+                                       vcpu->arch.isa);
+
        vcpu->cpu = cpu;
 }
 
@@ -602,6 +689,10 @@ void kvm_arch_vcpu_put(struct kvm_vcpu *vcpu)
 
        vcpu->cpu = -1;
 
+       kvm_riscv_vcpu_guest_fp_save(&vcpu->arch.guest_context,
+                                    vcpu->arch.isa);
+       kvm_riscv_vcpu_host_fp_restore(&vcpu->arch.host_context);
+
        csr_write(CSR_HGATP, 0);
 
        csr->vsstatus = csr_read(CSR_VSSTATUS);
index e22721e1b89287c908d754c5e5ff3efe401c3717..029a28a195c69c08f9cc0c598a3b743905ee8ac3 100644 (file)
@@ -224,3 +224,177 @@ ENTRY(__kvm_riscv_unpriv_trap)
        REG_S   a1, (KVM_ARCH_TRAP_HTINST)(a0)
        sret
 ENDPROC(__kvm_riscv_unpriv_trap)
+
+#ifdef CONFIG_FPU
+       .align 3
+       .global __kvm_riscv_fp_f_save
+__kvm_riscv_fp_f_save:
+       csrr t2, CSR_SSTATUS
+       li t1, SR_FS
+       csrs CSR_SSTATUS, t1
+       frcsr t0
+       fsw f0,  KVM_ARCH_FP_F_F0(a0)
+       fsw f1,  KVM_ARCH_FP_F_F1(a0)
+       fsw f2,  KVM_ARCH_FP_F_F2(a0)
+       fsw f3,  KVM_ARCH_FP_F_F3(a0)
+       fsw f4,  KVM_ARCH_FP_F_F4(a0)
+       fsw f5,  KVM_ARCH_FP_F_F5(a0)
+       fsw f6,  KVM_ARCH_FP_F_F6(a0)
+       fsw f7,  KVM_ARCH_FP_F_F7(a0)
+       fsw f8,  KVM_ARCH_FP_F_F8(a0)
+       fsw f9,  KVM_ARCH_FP_F_F9(a0)
+       fsw f10, KVM_ARCH_FP_F_F10(a0)
+       fsw f11, KVM_ARCH_FP_F_F11(a0)
+       fsw f12, KVM_ARCH_FP_F_F12(a0)
+       fsw f13, KVM_ARCH_FP_F_F13(a0)
+       fsw f14, KVM_ARCH_FP_F_F14(a0)
+       fsw f15, KVM_ARCH_FP_F_F15(a0)
+       fsw f16, KVM_ARCH_FP_F_F16(a0)
+       fsw f17, KVM_ARCH_FP_F_F17(a0)
+       fsw f18, KVM_ARCH_FP_F_F18(a0)
+       fsw f19, KVM_ARCH_FP_F_F19(a0)
+       fsw f20, KVM_ARCH_FP_F_F20(a0)
+       fsw f21, KVM_ARCH_FP_F_F21(a0)
+       fsw f22, KVM_ARCH_FP_F_F22(a0)
+       fsw f23, KVM_ARCH_FP_F_F23(a0)
+       fsw f24, KVM_ARCH_FP_F_F24(a0)
+       fsw f25, KVM_ARCH_FP_F_F25(a0)
+       fsw f26, KVM_ARCH_FP_F_F26(a0)
+       fsw f27, KVM_ARCH_FP_F_F27(a0)
+       fsw f28, KVM_ARCH_FP_F_F28(a0)
+       fsw f29, KVM_ARCH_FP_F_F29(a0)
+       fsw f30, KVM_ARCH_FP_F_F30(a0)
+       fsw f31, KVM_ARCH_FP_F_F31(a0)
+       sw t0, KVM_ARCH_FP_F_FCSR(a0)
+       csrw CSR_SSTATUS, t2
+       ret
+
+       .align 3
+       .global __kvm_riscv_fp_d_save
+__kvm_riscv_fp_d_save:
+       csrr t2, CSR_SSTATUS
+       li t1, SR_FS
+       csrs CSR_SSTATUS, t1
+       frcsr t0
+       fsd f0,  KVM_ARCH_FP_D_F0(a0)
+       fsd f1,  KVM_ARCH_FP_D_F1(a0)
+       fsd f2,  KVM_ARCH_FP_D_F2(a0)
+       fsd f3,  KVM_ARCH_FP_D_F3(a0)
+       fsd f4,  KVM_ARCH_FP_D_F4(a0)
+       fsd f5,  KVM_ARCH_FP_D_F5(a0)
+       fsd f6,  KVM_ARCH_FP_D_F6(a0)
+       fsd f7,  KVM_ARCH_FP_D_F7(a0)
+       fsd f8,  KVM_ARCH_FP_D_F8(a0)
+       fsd f9,  KVM_ARCH_FP_D_F9(a0)
+       fsd f10, KVM_ARCH_FP_D_F10(a0)
+       fsd f11, KVM_ARCH_FP_D_F11(a0)
+       fsd f12, KVM_ARCH_FP_D_F12(a0)
+       fsd f13, KVM_ARCH_FP_D_F13(a0)
+       fsd f14, KVM_ARCH_FP_D_F14(a0)
+       fsd f15, KVM_ARCH_FP_D_F15(a0)
+       fsd f16, KVM_ARCH_FP_D_F16(a0)
+       fsd f17, KVM_ARCH_FP_D_F17(a0)
+       fsd f18, KVM_ARCH_FP_D_F18(a0)
+       fsd f19, KVM_ARCH_FP_D_F19(a0)
+       fsd f20, KVM_ARCH_FP_D_F20(a0)
+       fsd f21, KVM_ARCH_FP_D_F21(a0)
+       fsd f22, KVM_ARCH_FP_D_F22(a0)
+       fsd f23, KVM_ARCH_FP_D_F23(a0)
+       fsd f24, KVM_ARCH_FP_D_F24(a0)
+       fsd f25, KVM_ARCH_FP_D_F25(a0)
+       fsd f26, KVM_ARCH_FP_D_F26(a0)
+       fsd f27, KVM_ARCH_FP_D_F27(a0)
+       fsd f28, KVM_ARCH_FP_D_F28(a0)
+       fsd f29, KVM_ARCH_FP_D_F29(a0)
+       fsd f30, KVM_ARCH_FP_D_F30(a0)
+       fsd f31, KVM_ARCH_FP_D_F31(a0)
+       sw t0, KVM_ARCH_FP_D_FCSR(a0)
+       csrw CSR_SSTATUS, t2
+       ret
+
+       .align 3
+       .global __kvm_riscv_fp_f_restore
+__kvm_riscv_fp_f_restore:
+       csrr t2, CSR_SSTATUS
+       li t1, SR_FS
+       lw t0, KVM_ARCH_FP_F_FCSR(a0)
+       csrs CSR_SSTATUS, t1
+       flw f0,  KVM_ARCH_FP_F_F0(a0)
+       flw f1,  KVM_ARCH_FP_F_F1(a0)
+       flw f2,  KVM_ARCH_FP_F_F2(a0)
+       flw f3,  KVM_ARCH_FP_F_F3(a0)
+       flw f4,  KVM_ARCH_FP_F_F4(a0)
+       flw f5,  KVM_ARCH_FP_F_F5(a0)
+       flw f6,  KVM_ARCH_FP_F_F6(a0)
+       flw f7,  KVM_ARCH_FP_F_F7(a0)
+       flw f8,  KVM_ARCH_FP_F_F8(a0)
+       flw f9,  KVM_ARCH_FP_F_F9(a0)
+       flw f10, KVM_ARCH_FP_F_F10(a0)
+       flw f11, KVM_ARCH_FP_F_F11(a0)
+       flw f12, KVM_ARCH_FP_F_F12(a0)
+       flw f13, KVM_ARCH_FP_F_F13(a0)
+       flw f14, KVM_ARCH_FP_F_F14(a0)
+       flw f15, KVM_ARCH_FP_F_F15(a0)
+       flw f16, KVM_ARCH_FP_F_F16(a0)
+       flw f17, KVM_ARCH_FP_F_F17(a0)
+       flw f18, KVM_ARCH_FP_F_F18(a0)
+       flw f19, KVM_ARCH_FP_F_F19(a0)
+       flw f20, KVM_ARCH_FP_F_F20(a0)
+       flw f21, KVM_ARCH_FP_F_F21(a0)
+       flw f22, KVM_ARCH_FP_F_F22(a0)
+       flw f23, KVM_ARCH_FP_F_F23(a0)
+       flw f24, KVM_ARCH_FP_F_F24(a0)
+       flw f25, KVM_ARCH_FP_F_F25(a0)
+       flw f26, KVM_ARCH_FP_F_F26(a0)
+       flw f27, KVM_ARCH_FP_F_F27(a0)
+       flw f28, KVM_ARCH_FP_F_F28(a0)
+       flw f29, KVM_ARCH_FP_F_F29(a0)
+       flw f30, KVM_ARCH_FP_F_F30(a0)
+       flw f31, KVM_ARCH_FP_F_F31(a0)
+       fscsr t0
+       csrw CSR_SSTATUS, t2
+       ret
+
+       .align 3
+       .global __kvm_riscv_fp_d_restore
+__kvm_riscv_fp_d_restore:
+       csrr t2, CSR_SSTATUS
+       li t1, SR_FS
+       lw t0, KVM_ARCH_FP_D_FCSR(a0)
+       csrs CSR_SSTATUS, t1
+       fld f0,  KVM_ARCH_FP_D_F0(a0)
+       fld f1,  KVM_ARCH_FP_D_F1(a0)
+       fld f2,  KVM_ARCH_FP_D_F2(a0)
+       fld f3,  KVM_ARCH_FP_D_F3(a0)
+       fld f4,  KVM_ARCH_FP_D_F4(a0)
+       fld f5,  KVM_ARCH_FP_D_F5(a0)
+       fld f6,  KVM_ARCH_FP_D_F6(a0)
+       fld f7,  KVM_ARCH_FP_D_F7(a0)
+       fld f8,  KVM_ARCH_FP_D_F8(a0)
+       fld f9,  KVM_ARCH_FP_D_F9(a0)
+       fld f10, KVM_ARCH_FP_D_F10(a0)
+       fld f11, KVM_ARCH_FP_D_F11(a0)
+       fld f12, KVM_ARCH_FP_D_F12(a0)
+       fld f13, KVM_ARCH_FP_D_F13(a0)
+       fld f14, KVM_ARCH_FP_D_F14(a0)
+       fld f15, KVM_ARCH_FP_D_F15(a0)
+       fld f16, KVM_ARCH_FP_D_F16(a0)
+       fld f17, KVM_ARCH_FP_D_F17(a0)
+       fld f18, KVM_ARCH_FP_D_F18(a0)
+       fld f19, KVM_ARCH_FP_D_F19(a0)
+       fld f20, KVM_ARCH_FP_D_F20(a0)
+       fld f21, KVM_ARCH_FP_D_F21(a0)
+       fld f22, KVM_ARCH_FP_D_F22(a0)
+       fld f23, KVM_ARCH_FP_D_F23(a0)
+       fld f24, KVM_ARCH_FP_D_F24(a0)
+       fld f25, KVM_ARCH_FP_D_F25(a0)
+       fld f26, KVM_ARCH_FP_D_F26(a0)
+       fld f27, KVM_ARCH_FP_D_F27(a0)
+       fld f28, KVM_ARCH_FP_D_F28(a0)
+       fld f29, KVM_ARCH_FP_D_F29(a0)
+       fld f30, KVM_ARCH_FP_D_F30(a0)
+       fld f31, KVM_ARCH_FP_D_F31(a0)
+       fscsr t0
+       csrw CSR_SSTATUS, t2
+       ret
+#endif