VirtualBox

Changeset 87310 in vbox


Ignore:
Timestamp:
Jan 19, 2021 11:39:19 PM (4 years ago)
Author:
vboxsync
Message:

VMM: Split out the generic VMX/SVM instruction wrapper utility method into a separate file as we'd probably like to keep this compiling on 32-bit too for testing purposes.

Location:
trunk/src/VBox/VMM
Files:
2 edited
1 copied

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/VMM/Makefile.kmk

    r87134 r87310  
    496496        VMMR0/HMR0.cpp \
    497497        VMMR0/HMR0A.asm \
     498        VMMR0/HMR0UtilA.asm \
    498499        VMMR0/HMVMXR0.cpp \
    499500        VMMR0/HMSVMR0.cpp \
  • trunk/src/VBox/VMM/VMMR0/HMR0A.asm

    r83067 r87310  
    2727%include "HMInternal.mac"
    2828
    29 %ifdef RT_OS_OS2 ;; @todo fix OMF support in yasm and kick nasm out completely.
    30  %macro vmwrite 2,
    31     int3
    32  %endmacro
    33  %define vmlaunch int3
    34  %define vmresume int3
    35  %define vmsave int3
    36  %define vmload int3
    37  %define vmrun int3
    38  %define clgi int3
    39  %define stgi int3
    40  %macro invlpga 2,
    41     int3
    42  %endmacro
     29%ifndef RT_ARCH_AMD64
     30 %error AMD64 only.
    4331%endif
     32
    4433
    4534;*********************************************************************************************************************************
     
    447436
    448437
    449 ;;
    450 ; Executes VMWRITE, 64-bit value.
    451 ;
    452 ; @returns VBox status code.
    453 ; @param   idxField   x86: [ebp + 08h]  msc: rcx  gcc: rdi   VMCS index.
    454 ; @param   u64Data    x86: [ebp + 0ch]  msc: rdx  gcc: rsi   VM field value.
    455 ;
    456 ALIGNCODE(16)
    457 BEGINPROC VMXWriteVmcs64
    458 %ifdef RT_ARCH_AMD64
    459  %ifdef ASM_CALL64_GCC
    460     and         edi, 0ffffffffh
    461     xor         rax, rax
    462     vmwrite     rdi, rsi
    463  %else
    464     and         ecx, 0ffffffffh
    465     xor         rax, rax
    466     vmwrite     rcx, rdx
    467  %endif
    468 %else  ; RT_ARCH_X86
    469     mov         ecx, [esp + 4]          ; idxField
    470     lea         edx, [esp + 8]          ; &u64Data
    471     vmwrite     ecx, [edx]              ; low dword
    472     jz          .done
    473     jc          .done
    474     inc         ecx
    475     xor         eax, eax
    476     vmwrite     ecx, [edx + 4]          ; high dword
    477 .done:
    478 %endif ; RT_ARCH_X86
    479     jnc         .valid_vmcs
    480     mov         eax, VERR_VMX_INVALID_VMCS_PTR
    481     ret
    482 .valid_vmcs:
    483     jnz         .the_end
    484     mov         eax, VERR_VMX_INVALID_VMCS_FIELD
    485 .the_end:
    486     ret
    487 ENDPROC VMXWriteVmcs64
    488 
    489 
    490 ;;
    491 ; Executes VMREAD, 64-bit value.
    492 ;
    493 ; @returns VBox status code.
    494 ; @param   idxField        VMCS index.
    495 ; @param   pData           Where to store VM field value.
    496 ;
    497 ;DECLASM(int) VMXReadVmcs64(uint32_t idxField, uint64_t *pData);
    498 ALIGNCODE(16)
    499 BEGINPROC VMXReadVmcs64
    500 %ifdef RT_ARCH_AMD64
    501  %ifdef ASM_CALL64_GCC
    502     and         edi, 0ffffffffh
    503     xor         rax, rax
    504     vmread      [rsi], rdi
    505  %else
    506     and         ecx, 0ffffffffh
    507     xor         rax, rax
    508     vmread      [rdx], rcx
    509  %endif
    510 %else  ; RT_ARCH_X86
    511     mov         ecx, [esp + 4]          ; idxField
    512     mov         edx, [esp + 8]          ; pData
    513     vmread      [edx], ecx              ; low dword
    514     jz          .done
    515     jc          .done
    516     inc         ecx
    517     xor         eax, eax
    518     vmread      [edx + 4], ecx          ; high dword
    519 .done:
    520 %endif ; RT_ARCH_X86
    521     jnc         .valid_vmcs
    522     mov         eax, VERR_VMX_INVALID_VMCS_PTR
    523     ret
    524 .valid_vmcs:
    525     jnz         .the_end
    526     mov         eax, VERR_VMX_INVALID_VMCS_FIELD
    527 .the_end:
    528     ret
    529 ENDPROC VMXReadVmcs64
    530 
    531 
    532 ;;
    533 ; Executes VMREAD, 32-bit value.
    534 ;
    535 ; @returns VBox status code.
    536 ; @param   idxField        VMCS index.
    537 ; @param   pu32Data        Where to store VM field value.
    538 ;
    539 ;DECLASM(int) VMXReadVmcs32(uint32_t idxField, uint32_t *pu32Data);
    540 ALIGNCODE(16)
    541 BEGINPROC VMXReadVmcs32
    542 %ifdef RT_ARCH_AMD64
    543  %ifdef ASM_CALL64_GCC
    544     and     edi, 0ffffffffh
    545     xor     rax, rax
    546     vmread  r10, rdi
    547     mov     [rsi], r10d
    548  %else
    549     and     ecx, 0ffffffffh
    550     xor     rax, rax
    551     vmread  r10, rcx
    552     mov     [rdx], r10d
    553  %endif
    554 %else  ; RT_ARCH_X86
    555     mov     ecx, [esp + 4]              ; idxField
    556     mov     edx, [esp + 8]              ; pu32Data
    557     xor     eax, eax
    558     vmread  [edx], ecx
    559 %endif ; RT_ARCH_X86
    560     jnc     .valid_vmcs
    561     mov     eax, VERR_VMX_INVALID_VMCS_PTR
    562     ret
    563 .valid_vmcs:
    564     jnz     .the_end
    565     mov     eax, VERR_VMX_INVALID_VMCS_FIELD
    566 .the_end:
    567     ret
    568 ENDPROC VMXReadVmcs32
    569 
    570 
    571 ;;
    572 ; Executes VMWRITE, 32-bit value.
    573 ;
    574 ; @returns VBox status code.
    575 ; @param   idxField        VMCS index.
    576 ; @param   u32Data         Where to store VM field value.
    577 ;
    578 ;DECLASM(int) VMXWriteVmcs32(uint32_t idxField, uint32_t u32Data);
    579 ALIGNCODE(16)
    580 BEGINPROC VMXWriteVmcs32
    581 %ifdef RT_ARCH_AMD64
    582  %ifdef ASM_CALL64_GCC
    583     and     edi, 0ffffffffh
    584     and     esi, 0ffffffffh
    585     xor     rax, rax
    586     vmwrite rdi, rsi
    587  %else
    588     and     ecx, 0ffffffffh
    589     and     edx, 0ffffffffh
    590     xor     rax, rax
    591     vmwrite rcx, rdx
    592  %endif
    593 %else  ; RT_ARCH_X86
    594     mov     ecx, [esp + 4]              ; idxField
    595     mov     edx, [esp + 8]              ; u32Data
    596     xor     eax, eax
    597     vmwrite ecx, edx
    598 %endif ; RT_ARCH_X86
    599     jnc     .valid_vmcs
    600     mov     eax, VERR_VMX_INVALID_VMCS_PTR
    601     ret
    602 .valid_vmcs:
    603     jnz     .the_end
    604     mov     eax, VERR_VMX_INVALID_VMCS_FIELD
    605 .the_end:
    606     ret
    607 ENDPROC VMXWriteVmcs32
    608 
    609 
    610 ;;
    611 ; Executes VMXON.
    612 ;
    613 ; @returns VBox status code.
    614 ; @param   HCPhysVMXOn      Physical address of VMXON structure.
    615 ;
    616 ;DECLASM(int) VMXEnable(RTHCPHYS HCPhysVMXOn);
    617 BEGINPROC VMXEnable
    618 %ifdef RT_ARCH_AMD64
    619     xor     rax, rax
    620  %ifdef ASM_CALL64_GCC
    621     push    rdi
    622  %else
    623     push    rcx
    624  %endif
    625     vmxon   [rsp]
    626 %else  ; RT_ARCH_X86
    627     xor     eax, eax
    628     vmxon   [esp + 4]
    629 %endif ; RT_ARCH_X86
    630     jnc     .good
    631     mov     eax, VERR_VMX_INVALID_VMXON_PTR
    632     jmp     .the_end
    633 
    634 .good:
    635     jnz     .the_end
    636     mov     eax, VERR_VMX_VMXON_FAILED
    637 
    638 .the_end:
    639 %ifdef RT_ARCH_AMD64
    640     add     rsp, 8
    641 %endif
    642     ret
    643 ENDPROC VMXEnable
    644 
    645 
    646 ;;
    647 ; Executes VMXOFF.
    648 ;
    649 ;DECLASM(void) VMXDisable(void);
    650 BEGINPROC VMXDisable
    651     vmxoff
    652 .the_end:
    653     ret
    654 ENDPROC VMXDisable
    655 
    656 
    657 ;;
    658 ; Executes VMCLEAR.
    659 ;
    660 ; @returns VBox status code.
    661 ; @param   HCPhysVmcs     Physical address of VM control structure.
    662 ;
    663 ;DECLASM(int) VMXClearVmcs(RTHCPHYS HCPhysVmcs);
    664 ALIGNCODE(16)
    665 BEGINPROC VMXClearVmcs
    666 %ifdef RT_ARCH_AMD64
    667     xor     rax, rax
    668  %ifdef ASM_CALL64_GCC
    669     push    rdi
    670  %else
    671     push    rcx
    672  %endif
    673     vmclear [rsp]
    674 %else  ; RT_ARCH_X86
    675     xor     eax, eax
    676     vmclear [esp + 4]
    677 %endif ; RT_ARCH_X86
    678     jnc     .the_end
    679     mov     eax, VERR_VMX_INVALID_VMCS_PTR
    680 .the_end:
    681 %ifdef RT_ARCH_AMD64
    682     add     rsp, 8
    683 %endif
    684     ret
    685 ENDPROC VMXClearVmcs
    686 
    687 
    688 ;;
    689 ; Executes VMPTRLD.
    690 ;
    691 ; @returns VBox status code.
    692 ; @param   HCPhysVmcs     Physical address of VMCS structure.
    693 ;
    694 ;DECLASM(int) VMXLoadVmcs(RTHCPHYS HCPhysVmcs);
    695 ALIGNCODE(16)
    696 BEGINPROC VMXLoadVmcs
    697 %ifdef RT_ARCH_AMD64
    698     xor     rax, rax
    699  %ifdef ASM_CALL64_GCC
    700     push    rdi
    701  %else
    702     push    rcx
    703  %endif
    704     vmptrld [rsp]
    705 %else
    706     xor     eax, eax
    707     vmptrld [esp + 4]
    708 %endif
    709     jnc     .the_end
    710     mov     eax, VERR_VMX_INVALID_VMCS_PTR
    711 .the_end:
    712 %ifdef RT_ARCH_AMD64
    713     add     rsp, 8
    714 %endif
    715     ret
    716 ENDPROC VMXLoadVmcs
    717 
    718 
    719 ;;
    720 ; Executes VMPTRST.
    721 ;
    722 ; @returns VBox status code.
    723 ; @param    [esp + 04h]  gcc:rdi  msc:rcx   Param 1 - First parameter - Address that will receive the current pointer.
    724 ;
    725 ;DECLASM(int) VMXGetCurrentVmcs(RTHCPHYS *pVMCS);
    726 BEGINPROC VMXGetCurrentVmcs
    727 %ifdef RT_OS_OS2
    728     mov     eax, VERR_NOT_SUPPORTED
    729     ret
    730 %else
    731  %ifdef RT_ARCH_AMD64
    732   %ifdef ASM_CALL64_GCC
    733     vmptrst qword [rdi]
    734   %else
    735     vmptrst qword [rcx]
    736   %endif
    737  %else
    738     vmptrst qword [esp+04h]
    739  %endif
    740     xor     eax, eax
    741 .the_end:
    742     ret
    743 %endif
    744 ENDPROC VMXGetCurrentVmcs
    745 
    746 ;;
    747 ; Invalidate a page using INVEPT.
    748 ;
    749 ; @param   enmTlbFlush  msc:ecx  gcc:edi  x86:[esp+04]  Type of flush.
    750 ; @param   pDescriptor  msc:edx  gcc:esi  x86:[esp+08]  Descriptor pointer.
    751 ;
    752 ;DECLASM(int) VMXR0InvEPT(VMXTLBFLUSHEPT enmTlbFlush, uint64_t *pDescriptor);
    753 BEGINPROC VMXR0InvEPT
    754 %ifdef RT_ARCH_AMD64
    755  %ifdef ASM_CALL64_GCC
    756     and         edi, 0ffffffffh
    757     xor         rax, rax
    758 ;    invept      rdi, qword [rsi]
    759     DB          0x66, 0x0F, 0x38, 0x80, 0x3E
    760  %else
    761     and         ecx, 0ffffffffh
    762     xor         rax, rax
    763 ;    invept      rcx, qword [rdx]
    764     DB          0x66, 0x0F, 0x38, 0x80, 0xA
    765  %endif
    766 %else
    767     mov         ecx, [esp + 4]
    768     mov         edx, [esp + 8]
    769     xor         eax, eax
    770 ;    invept      ecx, qword [edx]
    771     DB          0x66, 0x0F, 0x38, 0x80, 0xA
    772 %endif
    773     jnc         .valid_vmcs
    774     mov         eax, VERR_VMX_INVALID_VMCS_PTR
    775     ret
    776 .valid_vmcs:
    777     jnz         .the_end
    778     mov         eax, VERR_INVALID_PARAMETER
    779 .the_end:
    780     ret
    781 ENDPROC VMXR0InvEPT
    782 
    783 
    784 ;;
    785 ; Invalidate a page using INVVPID.
    786 ;
    787 ; @param   enmTlbFlush  msc:ecx  gcc:edi  x86:[esp+04]  Type of flush
    788 ; @param   pDescriptor  msc:edx  gcc:esi  x86:[esp+08]  Descriptor pointer
    789 ;
    790 ;DECLASM(int) VMXR0InvVPID(VMXTLBFLUSHVPID enmTlbFlush, uint64_t *pDescriptor);
    791 BEGINPROC VMXR0InvVPID
    792 %ifdef RT_ARCH_AMD64
    793  %ifdef ASM_CALL64_GCC
    794     and         edi, 0ffffffffh
    795     xor         rax, rax
    796 ;    invvpid     rdi, qword [rsi]
    797     DB          0x66, 0x0F, 0x38, 0x81, 0x3E
    798  %else
    799     and         ecx, 0ffffffffh
    800     xor         rax, rax
    801 ;    invvpid     rcx, qword [rdx]
    802     DB          0x66, 0x0F, 0x38, 0x81, 0xA
    803  %endif
    804 %else
    805     mov         ecx, [esp + 4]
    806     mov         edx, [esp + 8]
    807     xor         eax, eax
    808 ;    invvpid     ecx, qword [edx]
    809     DB          0x66, 0x0F, 0x38, 0x81, 0xA
    810 %endif
    811     jnc         .valid_vmcs
    812     mov         eax, VERR_VMX_INVALID_VMCS_PTR
    813     ret
    814 .valid_vmcs:
    815     jnz         .the_end
    816     mov         eax, VERR_INVALID_PARAMETER
    817 .the_end:
    818     ret
    819 ENDPROC VMXR0InvVPID
    820 
    821 
    822 %if GC_ARCH_BITS == 64
    823 ;;
    824 ; Executes INVLPGA.
    825 ;
    826 ; @param   pPageGC  msc:rcx  gcc:rdi  x86:[esp+04]  Virtual page to invalidate
    827 ; @param   uASID    msc:rdx  gcc:rsi  x86:[esp+0C]  Tagged TLB id
    828 ;
    829 ;DECLASM(void) SVMR0InvlpgA(RTGCPTR pPageGC, uint32_t uASID);
    830 BEGINPROC SVMR0InvlpgA
    831 %ifdef RT_ARCH_AMD64
    832  %ifdef ASM_CALL64_GCC
    833     mov     rax, rdi
    834     mov     rcx, rsi
    835  %else
    836     mov     rax, rcx
    837     mov     rcx, rdx
    838  %endif
    839 %else
    840     mov     eax, [esp + 4]
    841     mov     ecx, [esp + 0Ch]
    842 %endif
    843     invlpga [xAX], ecx
    844     ret
    845 ENDPROC SVMR0InvlpgA
    846 
    847 %else ; GC_ARCH_BITS != 64
    848 ;;
    849 ; Executes INVLPGA
    850 ;
    851 ; @param   pPageGC  msc:ecx  gcc:edi  x86:[esp+04]  Virtual page to invalidate
    852 ; @param   uASID    msc:edx  gcc:esi  x86:[esp+08]  Tagged TLB id
    853 ;
    854 ;DECLASM(void) SVMR0InvlpgA(RTGCPTR pPageGC, uint32_t uASID);
    855 BEGINPROC SVMR0InvlpgA
    856 %ifdef RT_ARCH_AMD64
    857  %ifdef ASM_CALL64_GCC
    858     movzx   rax, edi
    859     mov     ecx, esi
    860  %else
    861     ; from http://www.cs.cmu.edu/~fp/courses/15213-s06/misc/asm64-handout.pdf:
    862     ; "Perhaps unexpectedly, instructions that move or generate 32-bit register
    863     ;  values also set the upper 32 bits of the register to zero. Consequently
    864     ;  there is no need for an instruction movzlq."
    865     mov     eax, ecx
    866     mov     ecx, edx
    867  %endif
    868 %else
    869     mov     eax, [esp + 4]
    870     mov     ecx, [esp + 8]
    871 %endif
    872     invlpga [xAX], ecx
    873     ret
    874 ENDPROC SVMR0InvlpgA
    875 
    876 %endif ; GC_ARCH_BITS != 64
    877 
    878 
    879438%ifdef VBOX_WITH_KERNEL_USING_XMM
    880439
  • trunk/src/VBox/VMM/VMMR0/HMR0UtilA.asm

    r87305 r87310  
    11; $Id$
    22;; @file
    3 ; HM - Ring-0 VMX, SVM world-switch and helper routines.
     3; HM - Ring-0 VMX & SVM Helpers.
    44;
    55
     
    2222%include "VBox/err.mac"
    2323%include "VBox/vmm/hm_vmx.mac"
    24 %include "VBox/vmm/cpum.mac"
    25 %include "VBox/vmm/vm.mac"
    2624%include "iprt/x86.mac"
    27 %include "HMInternal.mac"
    28 
    29 %ifdef RT_OS_OS2 ;; @todo fix OMF support in yasm and kick nasm out completely.
    30  %macro vmwrite 2,
    31     int3
    32  %endmacro
    33  %define vmlaunch int3
    34  %define vmresume int3
    35  %define vmsave int3
    36  %define vmload int3
    37  %define vmrun int3
    38  %define clgi int3
    39  %define stgi int3
    40  %macro invlpga 2,
    41     int3
    42  %endmacro
    43 %endif
    44 
    45 ;*********************************************************************************************************************************
    46 ;*  Defined Constants And Macros                                                                                                 *
    47 ;*********************************************************************************************************************************
    48 ;; The offset of the XMM registers in X86FXSTATE.
    49 ; Use define because I'm too lazy to convert the struct.
    50 %define XMM_OFF_IN_X86FXSTATE   160
    51 
    52 ;; Spectre filler for 32-bit mode.
    53 ; Some user space address that points to a 4MB page boundrary in hope that it
    54 ; will somehow make it less useful.
    55 %define SPECTRE_FILLER32        0x227fffff
    56 ;; Spectre filler for 64-bit mode.
    57 ; Choosen to be an invalid address (also with 5 level paging).
    58 %define SPECTRE_FILLER64        0x02204204207fffff
    59 ;; Spectre filler for the current CPU mode.
    60 %ifdef RT_ARCH_AMD64
    61  %define SPECTRE_FILLER         SPECTRE_FILLER64
    62 %else
    63  %define SPECTRE_FILLER         SPECTRE_FILLER32
    64 %endif
    65 
    66 ;;
    67 ; Determine skipping restoring of GDTR, IDTR, TR across VMX non-root operation.
    68 ;
    69 %ifdef RT_ARCH_AMD64
    70  %define VMX_SKIP_GDTR
    71  %define VMX_SKIP_TR
    72  %define VBOX_SKIP_RESTORE_SEG
    73  %ifdef RT_OS_DARWIN
    74   ; Load the NULL selector into DS, ES, FS and GS on 64-bit darwin so we don't
    75   ; risk loading a stale LDT value or something invalid.
    76   %define HM_64_BIT_USE_NULL_SEL
    77   ; Darwin (Mavericks) uses IDTR limit to store the CPU Id so we need to restore it always.
    78   ; See @bugref{6875}.
    79  %else
    80   %define VMX_SKIP_IDTR
    81  %endif
    82 %endif
    83 
    84 ;; @def MYPUSHAD
    85 ; Macro generating an equivalent to PUSHAD instruction.
    86 
    87 ;; @def MYPOPAD
    88 ; Macro generating an equivalent to POPAD instruction.
    89 
    90 ;; @def MYPUSHSEGS
    91 ; Macro saving all segment registers on the stack.
    92 ; @param 1  Full width register name.
    93 ; @param 2  16-bit register name for \a 1.
    94 
    95 ;; @def MYPOPSEGS
    96 ; Macro restoring all segment registers on the stack.
    97 ; @param 1  Full width register name.
    98 ; @param 2  16-bit register name for \a 1.
    99 
    100 %ifdef ASM_CALL64_GCC
    101  %macro MYPUSHAD64 0
    102    push    r15
    103    push    r14
    104    push    r13
    105    push    r12
    106    push    rbx
    107  %endmacro
    108  %macro MYPOPAD64 0
    109    pop     rbx
    110    pop     r12
    111    pop     r13
    112    pop     r14
    113    pop     r15
    114  %endmacro
    115 
    116 %else ; ASM_CALL64_MSC
    117  %macro MYPUSHAD64 0
    118    push    r15
    119    push    r14
    120    push    r13
    121    push    r12
    122    push    rbx
    123    push    rsi
    124    push    rdi
    125  %endmacro
    126  %macro MYPOPAD64 0
    127    pop     rdi
    128    pop     rsi
    129    pop     rbx
    130    pop     r12
    131    pop     r13
    132    pop     r14
    133    pop     r15
    134  %endmacro
    135 %endif
    136 
    137 %ifdef VBOX_SKIP_RESTORE_SEG
    138  %macro MYPUSHSEGS64 2
    139  %endmacro
    140 
    141  %macro MYPOPSEGS64 2
    142  %endmacro
    143 %else       ; !VBOX_SKIP_RESTORE_SEG
    144  ; Trashes, rax, rdx & rcx.
    145  %macro MYPUSHSEGS64 2
    146   %ifndef HM_64_BIT_USE_NULL_SEL
    147    mov     %2, es
    148    push    %1
    149    mov     %2, ds
    150    push    %1
    151   %endif
    152 
    153    ; Special case for FS; Windows and Linux either don't use it or restore it when leaving kernel mode,
    154    ; Solaris OTOH doesn't and we must save it.
    155    mov     ecx, MSR_K8_FS_BASE
    156    rdmsr
    157    push    rdx
    158    push    rax
    159   %ifndef HM_64_BIT_USE_NULL_SEL
    160    push    fs
    161   %endif
    162 
    163    ; Special case for GS; OSes typically use swapgs to reset the hidden base register for GS on entry into the kernel.
    164    ; The same happens on exit.
    165    mov     ecx, MSR_K8_GS_BASE
    166    rdmsr
    167    push    rdx
    168    push    rax
    169   %ifndef HM_64_BIT_USE_NULL_SEL
    170    push    gs
    171   %endif
    172  %endmacro
    173 
    174  ; trashes, rax, rdx & rcx
    175  %macro MYPOPSEGS64 2
    176    ; Note: do not step through this code with a debugger!
    177   %ifndef HM_64_BIT_USE_NULL_SEL
    178    xor     eax, eax
    179    mov     ds, ax
    180    mov     es, ax
    181    mov     fs, ax
    182    mov     gs, ax
    183   %endif
    184 
    185   %ifndef HM_64_BIT_USE_NULL_SEL
    186    pop     gs
    187   %endif
    188    pop     rax
    189    pop     rdx
    190    mov     ecx, MSR_K8_GS_BASE
    191    wrmsr
    192 
    193   %ifndef HM_64_BIT_USE_NULL_SEL
    194    pop     fs
    195   %endif
    196    pop     rax
    197    pop     rdx
    198    mov     ecx, MSR_K8_FS_BASE
    199    wrmsr
    200    ; Now it's safe to step again
    201 
    202   %ifndef HM_64_BIT_USE_NULL_SEL
    203    pop     %1
    204    mov     ds, %2
    205    pop     %1
    206    mov     es, %2
    207   %endif
    208  %endmacro
    209 %endif ; VBOX_SKIP_RESTORE_SEG
    210 
    211 %macro MYPUSHAD32 0
    212   pushad
    213 %endmacro
    214 %macro MYPOPAD32 0
    215   popad
    216 %endmacro
    217 
    218 %macro MYPUSHSEGS32 2
    219   push    ds
    220   push    es
    221   push    fs
    222   push    gs
    223 %endmacro
    224 %macro MYPOPSEGS32 2
    225   pop     gs
    226   pop     fs
    227   pop     es
    228   pop     ds
    229 %endmacro
    230 
    231 %ifdef RT_ARCH_AMD64
    232  %define MYPUSHAD       MYPUSHAD64
    233  %define MYPOPAD        MYPOPAD64
    234  %define MYPUSHSEGS     MYPUSHSEGS64
    235  %define MYPOPSEGS      MYPOPSEGS64
    236 %else
    237  %define MYPUSHAD       MYPUSHAD32
    238  %define MYPOPAD        MYPOPAD32
    239  %define MYPUSHSEGS     MYPUSHSEGS32
    240  %define MYPOPSEGS      MYPOPSEGS32
    241 %endif
    242 
    243 ;;
    244 ; Creates an indirect branch prediction barrier on CPUs that need and supports that.
    245 ; @clobbers eax, edx, ecx
    246 ; @param    1   How to address CPUMCTX.
    247 ; @param    2   Which flag to test for (CPUMCTX_WSF_IBPB_ENTRY or CPUMCTX_WSF_IBPB_EXIT)
    248 %macro INDIRECT_BRANCH_PREDICTION_BARRIER 2
    249     test    byte [%1 + CPUMCTX.fWorldSwitcher], %2
    250     jz      %%no_indirect_branch_barrier
    251     mov     ecx, MSR_IA32_PRED_CMD
    252     mov     eax, MSR_IA32_PRED_CMD_F_IBPB
    253     xor     edx, edx
    254     wrmsr
    255 %%no_indirect_branch_barrier:
    256 %endmacro
    257 
    258 ;;
    259 ; Creates an indirect branch prediction and L1D barrier on CPUs that need and supports that.
    260 ; @clobbers eax, edx, ecx
    261 ; @param    1   How to address CPUMCTX.
    262 ; @param    2   Which IBPB flag to test for (CPUMCTX_WSF_IBPB_ENTRY or CPUMCTX_WSF_IBPB_EXIT)
    263 ; @param    3   Which FLUSH flag to test for (CPUMCTX_WSF_L1D_ENTRY)
    264 ; @param    4   Which MDS flag to test for (CPUMCTX_WSF_MDS_ENTRY)
    265 %macro INDIRECT_BRANCH_PREDICTION_AND_L1_CACHE_BARRIER 4
    266     ; Only one test+jmp when disabled CPUs.
    267     test    byte [%1 + CPUMCTX.fWorldSwitcher], (%2 | %3 | %4)
    268     jz      %%no_barrier_needed
    269 
    270     ; The eax:edx value is the same for both.
    271     AssertCompile(MSR_IA32_PRED_CMD_F_IBPB == MSR_IA32_FLUSH_CMD_F_L1D)
    272     mov     eax, MSR_IA32_PRED_CMD_F_IBPB
    273     xor     edx, edx
    274 
    275     ; Indirect branch barrier.
    276     test    byte [%1 + CPUMCTX.fWorldSwitcher], %2
    277     jz      %%no_indirect_branch_barrier
    278     mov     ecx, MSR_IA32_PRED_CMD
    279     wrmsr
    280 %%no_indirect_branch_barrier:
    281 
    282     ; Level 1 data cache flush.
    283     test    byte [%1 + CPUMCTX.fWorldSwitcher], %3
    284     jz      %%no_cache_flush_barrier
    285     mov     ecx, MSR_IA32_FLUSH_CMD
    286     wrmsr
    287     jmp     %%no_mds_buffer_flushing    ; MDS flushing is included in L1D_FLUSH
    288 %%no_cache_flush_barrier:
    289 
    290     ; MDS buffer flushing.
    291     test    byte [%1 + CPUMCTX.fWorldSwitcher], %4
    292     jz      %%no_mds_buffer_flushing
    293     sub     xSP, xSP
    294     mov     [xSP], ds
    295     verw    [xSP]
    296     add     xSP, xSP
    297 %%no_mds_buffer_flushing:
    298 
    299 %%no_barrier_needed:
    300 %endmacro
    301 
    302 
    303 ;*********************************************************************************************************************************
    304 ;*  External Symbols                                                                                                             *
    305 ;*********************************************************************************************************************************
    306 %ifdef VBOX_WITH_KERNEL_USING_XMM
    307 extern NAME(CPUMIsGuestFPUStateActive)
    308 %endif
     25
    30926
    31027
    31128BEGINCODE
    312 
    313 
    314 ;;
    315 ; Restores host-state fields.
    316 ;
    317 ; @returns VBox status code
    318 ; @param   f32RestoreHost x86: [ebp + 08h]  msc: ecx  gcc: edi   RestoreHost flags.
    319 ; @param   pRestoreHost   x86: [ebp + 0ch]  msc: rdx  gcc: rsi   Pointer to the RestoreHost struct.
    320 ;
    321 ALIGNCODE(16)
    322 BEGINPROC VMXRestoreHostState
    323 %ifdef RT_ARCH_AMD64
    324  %ifndef ASM_CALL64_GCC
    325     ; Use GCC's input registers since we'll be needing both rcx and rdx further
    326     ; down with the wrmsr instruction.  Use the R10 and R11 register for saving
    327     ; RDI and RSI since MSC preserve the two latter registers.
    328     mov         r10, rdi
    329     mov         r11, rsi
    330     mov         rdi, rcx
    331     mov         rsi, rdx
    332  %endif
    333 
    334     test        edi, VMX_RESTORE_HOST_GDTR
    335     jz          .test_idtr
    336     lgdt        [rsi + VMXRESTOREHOST.HostGdtr]
    337 
    338 .test_idtr:
    339     test        edi, VMX_RESTORE_HOST_IDTR
    340     jz          .test_ds
    341     lidt        [rsi + VMXRESTOREHOST.HostIdtr]
    342 
    343 .test_ds:
    344     test        edi, VMX_RESTORE_HOST_SEL_DS
    345     jz          .test_es
    346     mov         ax, [rsi + VMXRESTOREHOST.uHostSelDS]
    347     mov         ds, eax
    348 
    349 .test_es:
    350     test        edi, VMX_RESTORE_HOST_SEL_ES
    351     jz          .test_tr
    352     mov         ax, [rsi + VMXRESTOREHOST.uHostSelES]
    353     mov         es, eax
    354 
    355 .test_tr:
    356     test        edi, VMX_RESTORE_HOST_SEL_TR
    357     jz          .test_fs
    358     ; When restoring the TR, we must first clear the busy flag or we'll end up faulting.
    359     mov         dx, [rsi + VMXRESTOREHOST.uHostSelTR]
    360     mov         ax, dx
    361     and         eax, X86_SEL_MASK_OFF_RPL                       ; mask away TI and RPL bits leaving only the descriptor offset
    362     test        edi, VMX_RESTORE_HOST_GDT_READ_ONLY | VMX_RESTORE_HOST_GDT_NEED_WRITABLE
    363     jnz         .gdt_readonly
    364     add         rax, qword [rsi + VMXRESTOREHOST.HostGdtr + 2]  ; xAX <- descriptor offset + GDTR.pGdt.
    365     and         dword [rax + 4], ~RT_BIT(9)                     ; clear the busy flag in TSS desc (bits 0-7=base, bit 9=busy bit)
    366     ltr         dx
    367     jmp short   .test_fs
    368 .gdt_readonly:
    369     test        edi, VMX_RESTORE_HOST_GDT_NEED_WRITABLE
    370     jnz         .gdt_readonly_need_writable
    371     mov         rcx, cr0
    372     mov         r9, rcx
    373     add         rax, qword [rsi + VMXRESTOREHOST.HostGdtr + 2]  ; xAX <- descriptor offset + GDTR.pGdt.
    374     and         rcx, ~X86_CR0_WP
    375     mov         cr0, rcx
    376     and         dword [rax + 4], ~RT_BIT(9)                     ; clear the busy flag in TSS desc (bits 0-7=base, bit 9=busy bit)
    377     ltr         dx
    378     mov         cr0, r9
    379     jmp short   .test_fs
    380 .gdt_readonly_need_writable:
    381     add         rax, qword [rsi + VMXRESTOREHOST.HostGdtrRw + 2]  ; xAX <- descriptor offset + GDTR.pGdtRw
    382     and         dword [rax + 4], ~RT_BIT(9)                     ; clear the busy flag in TSS desc (bits 0-7=base, bit 9=busy bit)
    383     lgdt        [rsi + VMXRESTOREHOST.HostGdtrRw]
    384     ltr         dx
    385     lgdt        [rsi + VMXRESTOREHOST.HostGdtr]                 ; load the original GDT
    386 
    387 .test_fs:
    388     ;
    389     ; When restoring the selector values for FS and GS, we'll temporarily trash
    390     ; the base address (at least the high 32-bit bits, but quite possibly the
    391     ; whole base address), the wrmsr will restore it correctly. (VT-x actually
    392     ; restores the base correctly when leaving guest mode, but not the selector
    393     ; value, so there is little problem with interrupts being enabled prior to
    394     ; this restore job.)
    395     ; We'll disable ints once for both FS and GS as that's probably faster.
    396     ;
    397     test        edi, VMX_RESTORE_HOST_SEL_FS | VMX_RESTORE_HOST_SEL_GS
    398     jz          .restore_success
    399     pushfq
    400     cli                                   ; (see above)
    401 
    402     test        edi, VMX_RESTORE_HOST_SEL_FS
    403     jz          .test_gs
    404     mov         ax, word [rsi + VMXRESTOREHOST.uHostSelFS]
    405     mov         fs, eax
    406     mov         eax, dword [rsi + VMXRESTOREHOST.uHostFSBase]         ; uHostFSBase - Lo
    407     mov         edx, dword [rsi + VMXRESTOREHOST.uHostFSBase + 4h]    ; uHostFSBase - Hi
    408     mov         ecx, MSR_K8_FS_BASE
    409     wrmsr
    410 
    411 .test_gs:
    412     test        edi, VMX_RESTORE_HOST_SEL_GS
    413     jz          .restore_flags
    414     mov         ax, word [rsi + VMXRESTOREHOST.uHostSelGS]
    415     mov         gs, eax
    416     mov         eax, dword [rsi + VMXRESTOREHOST.uHostGSBase]         ; uHostGSBase - Lo
    417     mov         edx, dword [rsi + VMXRESTOREHOST.uHostGSBase + 4h]    ; uHostGSBase - Hi
    418     mov         ecx, MSR_K8_GS_BASE
    419     wrmsr
    420 
    421 .restore_flags:
    422     popfq
    423 
    424 .restore_success:
    425     mov         eax, VINF_SUCCESS
    426  %ifndef ASM_CALL64_GCC
    427     ; Restore RDI and RSI on MSC.
    428     mov         rdi, r10
    429     mov         rsi, r11
    430  %endif
    431 %else  ; RT_ARCH_X86
    432     mov         eax, VERR_NOT_IMPLEMENTED
    433 %endif
    434     ret
    435 ENDPROC VMXRestoreHostState
    436 
    437 
    438 ;;
    439 ; Dispatches an NMI to the host.
    440 ;
    441 ALIGNCODE(16)
    442 BEGINPROC VMXDispatchHostNmi
    443     ; NMI is always vector 2. The IDT[2] IRQ handler cannot be anything else. See Intel spec. 6.3.1 "External Interrupts".
    444     int 2
    445     ret
    446 ENDPROC VMXDispatchHostNmi
    447 
    44829
    44930;;
     
    744325ENDPROC VMXGetCurrentVmcs
    745326
     327
    746328;;
    747329; Invalidate a page using INVEPT.
     
    876458%endif ; GC_ARCH_BITS != 64
    877459
    878 
    879 %ifdef VBOX_WITH_KERNEL_USING_XMM
    880 
    881 ;;
    882 ; Wrapper around vmx.pfnStartVM that preserves host XMM registers and
    883 ; load the guest ones when necessary.
    884 ;
    885 ; @cproto       DECLASM(int) hmR0VMXStartVMWrapXMM(RTHCUINT fResume, PCPUMCTX pCtx, void *pvUnused, PVM pVM,
    886 ;                                                  PVMCPU pVCpu, PFNHMVMXSTARTVM pfnStartVM);
    887 ;
    888 ; @returns      eax
    889 ;
    890 ; @param        fResumeVM       msc:rcx
    891 ; @param        pCtx            msc:rdx
    892 ; @param        pvUnused        msc:r8
    893 ; @param        pVM             msc:r9
    894 ; @param        pVCpu           msc:[rbp+30h]   The cross context virtual CPU structure of the calling EMT.
    895 ; @param        pfnStartVM      msc:[rbp+38h]
    896 ;
    897 ; @remarks      This is essentially the same code as hmR0SVMRunWrapXMM, only the parameters differ a little bit.
    898 ;
    899 ; @remarks      Drivers shouldn't use AVX registers without saving+loading:
    900 ;                   https://msdn.microsoft.com/en-us/library/windows/hardware/ff545910%28v=vs.85%29.aspx?f=255&MSPPError=-2147217396
    901 ;               However the compiler docs have different idea:
    902 ;                   https://msdn.microsoft.com/en-us/library/9z1stfyw.aspx
    903 ;               We'll go with the former for now.
    904 ;
    905 ; ASSUMING 64-bit and windows for now.
    906 ;
    907 ALIGNCODE(16)
    908 BEGINPROC hmR0VMXStartVMWrapXMM
    909         push    xBP
    910         mov     xBP, xSP
    911         sub     xSP, 0b0h + 040h ; Don't bother optimizing the frame size.
    912 
    913         ; Spill input parameters.
    914         mov     [xBP + 010h], rcx       ; fResumeVM
    915         mov     [xBP + 018h], rdx       ; pCtx
    916         mov     [xBP + 020h], r8        ; pvUnused
    917         mov     [xBP + 028h], r9        ; pVM
    918 
    919         ; Ask CPUM whether we've started using the FPU yet.
    920         mov     rcx, [xBP + 30h]        ; pVCpu
    921         call    NAME(CPUMIsGuestFPUStateActive)
    922         test    al, al
    923         jnz     .guest_fpu_state_active
    924 
    925         ; No need to mess with XMM registers just call the start routine and return.
    926         mov     r11, [xBP + 38h]        ; pfnStartVM
    927         mov     r10, [xBP + 30h]        ; pVCpu
    928         mov     [xSP + 020h], r10
    929         mov     rcx, [xBP + 010h]       ; fResumeVM
    930         mov     rdx, [xBP + 018h]       ; pCtx
    931         mov     r8,  [xBP + 020h]       ; pvUnused
    932         mov     r9,  [xBP + 028h]       ; pVM
    933         call    r11
    934 
    935         leave
    936         ret
    937 
    938 ALIGNCODE(8)
    939 .guest_fpu_state_active:
    940         ; Save the non-volatile host XMM registers.
    941         movdqa  [rsp + 040h + 000h], xmm6
    942         movdqa  [rsp + 040h + 010h], xmm7
    943         movdqa  [rsp + 040h + 020h], xmm8
    944         movdqa  [rsp + 040h + 030h], xmm9
    945         movdqa  [rsp + 040h + 040h], xmm10
    946         movdqa  [rsp + 040h + 050h], xmm11
    947         movdqa  [rsp + 040h + 060h], xmm12
    948         movdqa  [rsp + 040h + 070h], xmm13
    949         movdqa  [rsp + 040h + 080h], xmm14
    950         movdqa  [rsp + 040h + 090h], xmm15
    951         stmxcsr [rsp + 040h + 0a0h]
    952 
    953         mov     r10, [xBP + 018h]       ; pCtx
    954         mov     eax, [r10 + CPUMCTX.fXStateMask]
    955         test    eax, eax
    956         jz      .guest_fpu_state_manually
    957 
    958         ;
    959         ; Using XSAVE to load the guest XMM, YMM and ZMM registers.
    960         ;
    961         and     eax, CPUM_VOLATILE_XSAVE_GUEST_COMPONENTS
    962         xor     edx, edx
    963         mov     r10, [r10 + CPUMCTX.pXStateR0]
    964         xrstor  [r10]
    965 
    966         ; Make the call (same as in the other case).
    967         mov     r11, [xBP + 38h]        ; pfnStartVM
    968         mov     r10, [xBP + 30h]        ; pVCpu
    969         mov     [xSP + 020h], r10
    970         mov     rcx, [xBP + 010h]       ; fResumeVM
    971         mov     rdx, [xBP + 018h]       ; pCtx
    972         mov     r8,  [xBP + 020h]       ; pvUnused
    973         mov     r9,  [xBP + 028h]       ; pVM
    974         call    r11
    975 
    976         mov     r11d, eax               ; save return value (xsave below uses eax)
    977 
    978         ; Save the guest XMM registers.
    979         mov     r10, [xBP + 018h]       ; pCtx
    980         mov     eax, [r10 + CPUMCTX.fXStateMask]
    981         and     eax, CPUM_VOLATILE_XSAVE_GUEST_COMPONENTS
    982         xor     edx, edx
    983         mov     r10, [r10 + CPUMCTX.pXStateR0]
    984         xsave  [r10]
    985 
    986         mov     eax, r11d               ; restore return value
    987 
    988 .restore_non_volatile_host_xmm_regs:
    989         ; Load the non-volatile host XMM registers.
    990         movdqa  xmm6,  [rsp + 040h + 000h]
    991         movdqa  xmm7,  [rsp + 040h + 010h]
    992         movdqa  xmm8,  [rsp + 040h + 020h]
    993         movdqa  xmm9,  [rsp + 040h + 030h]
    994         movdqa  xmm10, [rsp + 040h + 040h]
    995         movdqa  xmm11, [rsp + 040h + 050h]
    996         movdqa  xmm12, [rsp + 040h + 060h]
    997         movdqa  xmm13, [rsp + 040h + 070h]
    998         movdqa  xmm14, [rsp + 040h + 080h]
    999         movdqa  xmm15, [rsp + 040h + 090h]
    1000         ldmxcsr        [rsp + 040h + 0a0h]
    1001         leave
    1002         ret
    1003 
    1004         ;
    1005         ; No XSAVE, load and save the guest XMM registers manually.
    1006         ;
    1007 .guest_fpu_state_manually:
    1008         ; Load the full guest XMM register state.
    1009         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1010         movdqa  xmm0,  [r10 + XMM_OFF_IN_X86FXSTATE + 000h]
    1011         movdqa  xmm1,  [r10 + XMM_OFF_IN_X86FXSTATE + 010h]
    1012         movdqa  xmm2,  [r10 + XMM_OFF_IN_X86FXSTATE + 020h]
    1013         movdqa  xmm3,  [r10 + XMM_OFF_IN_X86FXSTATE + 030h]
    1014         movdqa  xmm4,  [r10 + XMM_OFF_IN_X86FXSTATE + 040h]
    1015         movdqa  xmm5,  [r10 + XMM_OFF_IN_X86FXSTATE + 050h]
    1016         movdqa  xmm6,  [r10 + XMM_OFF_IN_X86FXSTATE + 060h]
    1017         movdqa  xmm7,  [r10 + XMM_OFF_IN_X86FXSTATE + 070h]
    1018         movdqa  xmm8,  [r10 + XMM_OFF_IN_X86FXSTATE + 080h]
    1019         movdqa  xmm9,  [r10 + XMM_OFF_IN_X86FXSTATE + 090h]
    1020         movdqa  xmm10, [r10 + XMM_OFF_IN_X86FXSTATE + 0a0h]
    1021         movdqa  xmm11, [r10 + XMM_OFF_IN_X86FXSTATE + 0b0h]
    1022         movdqa  xmm12, [r10 + XMM_OFF_IN_X86FXSTATE + 0c0h]
    1023         movdqa  xmm13, [r10 + XMM_OFF_IN_X86FXSTATE + 0d0h]
    1024         movdqa  xmm14, [r10 + XMM_OFF_IN_X86FXSTATE + 0e0h]
    1025         movdqa  xmm15, [r10 + XMM_OFF_IN_X86FXSTATE + 0f0h]
    1026         ldmxcsr        [r10 + X86FXSTATE.MXCSR]
    1027 
    1028         ; Make the call (same as in the other case).
    1029         mov     r11, [xBP + 38h]        ; pfnStartVM
    1030         mov     r10, [xBP + 30h]        ; pVCpu
    1031         mov     [xSP + 020h], r10
    1032         mov     rcx, [xBP + 010h]       ; fResumeVM
    1033         mov     rdx, [xBP + 018h]       ; pCtx
    1034         mov     r8,  [xBP + 020h]       ; pvUnused
    1035         mov     r9,  [xBP + 028h]       ; pVM
    1036         call    r11
    1037 
    1038         ; Save the guest XMM registers.
    1039         mov     r10, [xBP + 018h]       ; pCtx
    1040         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1041         stmxcsr [r10 + X86FXSTATE.MXCSR]
    1042         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 000h], xmm0
    1043         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 010h], xmm1
    1044         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 020h], xmm2
    1045         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 030h], xmm3
    1046         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 040h], xmm4
    1047         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 050h], xmm5
    1048         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 060h], xmm6
    1049         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 070h], xmm7
    1050         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 080h], xmm8
    1051         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 090h], xmm9
    1052         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0a0h], xmm10
    1053         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0b0h], xmm11
    1054         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0c0h], xmm12
    1055         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0d0h], xmm13
    1056         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0e0h], xmm14
    1057         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0f0h], xmm15
    1058         jmp     .restore_non_volatile_host_xmm_regs
    1059 ENDPROC   hmR0VMXStartVMWrapXMM
    1060 
    1061 ;;
    1062 ; Wrapper around svm.pfnVMRun that preserves host XMM registers and
    1063 ; load the guest ones when necessary.
    1064 ;
    1065 ; @cproto       DECLASM(int) hmR0SVMRunWrapXMM(RTHCPHYS HCPhysVmcbHost, RTHCPHYS HCPhysVmcb, PCPUMCTX pCtx, PVM pVM, PVMCPU pVCpu,
    1066 ;                                              PFNHMSVMVMRUN pfnVMRun);
    1067 ;
    1068 ; @returns      eax
    1069 ;
    1070 ; @param        HCPhysVmcbHost  msc:rcx
    1071 ; @param        HCPhysVmcb      msc:rdx
    1072 ; @param        pCtx            msc:r8
    1073 ; @param        pVM             msc:r9
    1074 ; @param        pVCpu           msc:[rbp+30h]   The cross context virtual CPU structure of the calling EMT.
    1075 ; @param        pfnVMRun        msc:[rbp+38h]
    1076 ;
    1077 ; @remarks      This is essentially the same code as hmR0VMXStartVMWrapXMM, only the parameters differ a little bit.
    1078 ;
    1079 ; @remarks      Drivers shouldn't use AVX registers without saving+loading:
    1080 ;                   https://msdn.microsoft.com/en-us/library/windows/hardware/ff545910%28v=vs.85%29.aspx?f=255&MSPPError=-2147217396
    1081 ;               However the compiler docs have different idea:
    1082 ;                   https://msdn.microsoft.com/en-us/library/9z1stfyw.aspx
    1083 ;               We'll go with the former for now.
    1084 ;
    1085 ; ASSUMING 64-bit and windows for now.
    1086 ALIGNCODE(16)
    1087 BEGINPROC hmR0SVMRunWrapXMM
    1088         push    xBP
    1089         mov     xBP, xSP
    1090         sub     xSP, 0b0h + 040h        ; don't bother optimizing the frame size
    1091 
    1092         ; Spill input parameters.
    1093         mov     [xBP + 010h], rcx       ; HCPhysVmcbHost
    1094         mov     [xBP + 018h], rdx       ; HCPhysVmcb
    1095         mov     [xBP + 020h], r8        ; pCtx
    1096         mov     [xBP + 028h], r9        ; pVM
    1097 
    1098         ; Ask CPUM whether we've started using the FPU yet.
    1099         mov     rcx, [xBP + 30h]        ; pVCpu
    1100         call    NAME(CPUMIsGuestFPUStateActive)
    1101         test    al, al
    1102         jnz     .guest_fpu_state_active
    1103 
    1104         ; No need to mess with XMM registers just call the start routine and return.
    1105         mov     r11, [xBP + 38h]        ; pfnVMRun
    1106         mov     r10, [xBP + 30h]        ; pVCpu
    1107         mov     [xSP + 020h], r10
    1108         mov     rcx, [xBP + 010h]       ; HCPhysVmcbHost
    1109         mov     rdx, [xBP + 018h]       ; HCPhysVmcb
    1110         mov     r8,  [xBP + 020h]       ; pCtx
    1111         mov     r9,  [xBP + 028h]       ; pVM
    1112         call    r11
    1113 
    1114         leave
    1115         ret
    1116 
    1117 ALIGNCODE(8)
    1118 .guest_fpu_state_active:
    1119         ; Save the non-volatile host XMM registers.
    1120         movdqa  [rsp + 040h + 000h], xmm6
    1121         movdqa  [rsp + 040h + 010h], xmm7
    1122         movdqa  [rsp + 040h + 020h], xmm8
    1123         movdqa  [rsp + 040h + 030h], xmm9
    1124         movdqa  [rsp + 040h + 040h], xmm10
    1125         movdqa  [rsp + 040h + 050h], xmm11
    1126         movdqa  [rsp + 040h + 060h], xmm12
    1127         movdqa  [rsp + 040h + 070h], xmm13
    1128         movdqa  [rsp + 040h + 080h], xmm14
    1129         movdqa  [rsp + 040h + 090h], xmm15
    1130         stmxcsr [rsp + 040h + 0a0h]
    1131 
    1132         mov     r10, [xBP + 020h]       ; pCtx
    1133         mov     eax, [r10 + CPUMCTX.fXStateMask]
    1134         test    eax, eax
    1135         jz      .guest_fpu_state_manually
    1136 
    1137         ;
    1138         ; Using XSAVE.
    1139         ;
    1140         and     eax, CPUM_VOLATILE_XSAVE_GUEST_COMPONENTS
    1141         xor     edx, edx
    1142         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1143         xrstor  [r10]
    1144 
    1145         ; Make the call (same as in the other case).
    1146         mov     r11, [xBP + 38h]        ; pfnVMRun
    1147         mov     r10, [xBP + 30h]        ; pVCpu
    1148         mov     [xSP + 020h], r10
    1149         mov     rcx, [xBP + 010h]       ; HCPhysVmcbHost
    1150         mov     rdx, [xBP + 018h]       ; HCPhysVmcb
    1151         mov     r8,  [xBP + 020h]       ; pCtx
    1152         mov     r9,  [xBP + 028h]       ; pVM
    1153         call    r11
    1154 
    1155         mov     r11d, eax               ; save return value (xsave below uses eax)
    1156 
    1157         ; Save the guest XMM registers.
    1158         mov     r10, [xBP + 020h]       ; pCtx
    1159         mov     eax, [r10 + CPUMCTX.fXStateMask]
    1160         and     eax, CPUM_VOLATILE_XSAVE_GUEST_COMPONENTS
    1161         xor     edx, edx
    1162         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1163         xsave  [r10]
    1164 
    1165         mov     eax, r11d               ; restore return value
    1166 
    1167 .restore_non_volatile_host_xmm_regs:
    1168         ; Load the non-volatile host XMM registers.
    1169         movdqa  xmm6,  [rsp + 040h + 000h]
    1170         movdqa  xmm7,  [rsp + 040h + 010h]
    1171         movdqa  xmm8,  [rsp + 040h + 020h]
    1172         movdqa  xmm9,  [rsp + 040h + 030h]
    1173         movdqa  xmm10, [rsp + 040h + 040h]
    1174         movdqa  xmm11, [rsp + 040h + 050h]
    1175         movdqa  xmm12, [rsp + 040h + 060h]
    1176         movdqa  xmm13, [rsp + 040h + 070h]
    1177         movdqa  xmm14, [rsp + 040h + 080h]
    1178         movdqa  xmm15, [rsp + 040h + 090h]
    1179         ldmxcsr [rsp + 040h + 0a0h]
    1180         leave
    1181         ret
    1182 
    1183         ;
    1184         ; No XSAVE, load and save the guest XMM registers manually.
    1185         ;
    1186 .guest_fpu_state_manually:
    1187         ; Load the full guest XMM register state.
    1188         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1189         movdqa  xmm0,  [r10 + XMM_OFF_IN_X86FXSTATE + 000h]
    1190         movdqa  xmm1,  [r10 + XMM_OFF_IN_X86FXSTATE + 010h]
    1191         movdqa  xmm2,  [r10 + XMM_OFF_IN_X86FXSTATE + 020h]
    1192         movdqa  xmm3,  [r10 + XMM_OFF_IN_X86FXSTATE + 030h]
    1193         movdqa  xmm4,  [r10 + XMM_OFF_IN_X86FXSTATE + 040h]
    1194         movdqa  xmm5,  [r10 + XMM_OFF_IN_X86FXSTATE + 050h]
    1195         movdqa  xmm6,  [r10 + XMM_OFF_IN_X86FXSTATE + 060h]
    1196         movdqa  xmm7,  [r10 + XMM_OFF_IN_X86FXSTATE + 070h]
    1197         movdqa  xmm8,  [r10 + XMM_OFF_IN_X86FXSTATE + 080h]
    1198         movdqa  xmm9,  [r10 + XMM_OFF_IN_X86FXSTATE + 090h]
    1199         movdqa  xmm10, [r10 + XMM_OFF_IN_X86FXSTATE + 0a0h]
    1200         movdqa  xmm11, [r10 + XMM_OFF_IN_X86FXSTATE + 0b0h]
    1201         movdqa  xmm12, [r10 + XMM_OFF_IN_X86FXSTATE + 0c0h]
    1202         movdqa  xmm13, [r10 + XMM_OFF_IN_X86FXSTATE + 0d0h]
    1203         movdqa  xmm14, [r10 + XMM_OFF_IN_X86FXSTATE + 0e0h]
    1204         movdqa  xmm15, [r10 + XMM_OFF_IN_X86FXSTATE + 0f0h]
    1205         ldmxcsr        [r10 + X86FXSTATE.MXCSR]
    1206 
    1207         ; Make the call (same as in the other case).
    1208         mov     r11, [xBP + 38h]        ; pfnVMRun
    1209         mov     r10, [xBP + 30h]        ; pVCpu
    1210         mov     [xSP + 020h], r10
    1211         mov     rcx, [xBP + 010h]       ; HCPhysVmcbHost
    1212         mov     rdx, [xBP + 018h]       ; HCPhysVmcb
    1213         mov     r8,  [xBP + 020h]       ; pCtx
    1214         mov     r9,  [xBP + 028h]       ; pVM
    1215         call    r11
    1216 
    1217         ; Save the guest XMM registers.
    1218         mov     r10, [xBP + 020h]       ; pCtx
    1219         mov     r10, [r10 + CPUMCTX.pXStateR0]
    1220         stmxcsr [r10 + X86FXSTATE.MXCSR]
    1221         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 000h], xmm0
    1222         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 010h], xmm1
    1223         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 020h], xmm2
    1224         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 030h], xmm3
    1225         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 040h], xmm4
    1226         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 050h], xmm5
    1227         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 060h], xmm6
    1228         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 070h], xmm7
    1229         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 080h], xmm8
    1230         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 090h], xmm9
    1231         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0a0h], xmm10
    1232         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0b0h], xmm11
    1233         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0c0h], xmm12
    1234         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0d0h], xmm13
    1235         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0e0h], xmm14
    1236         movdqa  [r10 + XMM_OFF_IN_X86FXSTATE + 0f0h], xmm15
    1237         jmp     .restore_non_volatile_host_xmm_regs
    1238 ENDPROC   hmR0SVMRunWrapXMM
    1239 
    1240 %endif ; VBOX_WITH_KERNEL_USING_XMM
    1241 
    1242 
    1243 %ifdef RT_ARCH_AMD64
    1244 ;; @def RESTORE_STATE_VM64
    1245 ; Macro restoring essential host state and updating guest state
    1246 ; for 64-bit host, 64-bit guest for VT-x.
    1247 ;
    1248 %macro RESTORE_STATE_VM64 0
    1249     ; Restore base and limit of the IDTR & GDTR.
    1250  %ifndef VMX_SKIP_IDTR
    1251     lidt    [xSP]
    1252     add     xSP, xCB * 2
    1253  %endif
    1254  %ifndef VMX_SKIP_GDTR
    1255     lgdt    [xSP]
    1256     add     xSP, xCB * 2
    1257  %endif
    1258 
    1259     push    xDI
    1260  %ifndef VMX_SKIP_TR
    1261     mov     xDI, [xSP + xCB * 3]        ; pCtx (*3 to skip the saved xDI, TR, LDTR)
    1262  %else
    1263     mov     xDI, [xSP + xCB * 2]        ; pCtx (*2 to skip the saved xDI, LDTR)
    1264  %endif
    1265 
    1266     mov     qword [xDI + CPUMCTX.eax], rax
    1267     mov     rax, SPECTRE_FILLER64
    1268     mov     qword [xDI + CPUMCTX.ebx], rbx
    1269     mov     rbx, rax
    1270     mov     qword [xDI + CPUMCTX.ecx], rcx
    1271     mov     rcx, rax
    1272     mov     qword [xDI + CPUMCTX.edx], rdx
    1273     mov     rdx, rax
    1274     mov     qword [xDI + CPUMCTX.esi], rsi
    1275     mov     rsi, rax
    1276     mov     qword [xDI + CPUMCTX.ebp], rbp
    1277     mov     rbp, rax
    1278     mov     qword [xDI + CPUMCTX.r8],  r8
    1279     mov     r8, rax
    1280     mov     qword [xDI + CPUMCTX.r9],  r9
    1281     mov     r9, rax
    1282     mov     qword [xDI + CPUMCTX.r10], r10
    1283     mov     r10, rax
    1284     mov     qword [xDI + CPUMCTX.r11], r11
    1285     mov     r11, rax
    1286     mov     qword [xDI + CPUMCTX.r12], r12
    1287     mov     r12, rax
    1288     mov     qword [xDI + CPUMCTX.r13], r13
    1289     mov     r13, rax
    1290     mov     qword [xDI + CPUMCTX.r14], r14
    1291     mov     r14, rax
    1292     mov     qword [xDI + CPUMCTX.r15], r15
    1293     mov     r15, rax
    1294     mov     rax, cr2
    1295     mov     qword [xDI + CPUMCTX.cr2], rax
    1296 
    1297     pop     xAX                                 ; The guest rdi we pushed above
    1298     mov     qword [xDI + CPUMCTX.edi], rax
    1299 
    1300     ; Fight spectre.
    1301     INDIRECT_BRANCH_PREDICTION_BARRIER xDI, CPUMCTX_WSF_IBPB_EXIT
    1302 
    1303  %ifndef VMX_SKIP_TR
    1304     ; Restore TSS selector; must mark it as not busy before using ltr!
    1305     ; ASSUME that this is supposed to be 'BUSY' (saves 20-30 ticks on the T42p).
    1306     ; @todo get rid of sgdt
    1307     pop     xBX         ; Saved TR
    1308     sub     xSP, xCB * 2
    1309     sgdt    [xSP]
    1310     mov     xAX, xBX
    1311     and     eax, X86_SEL_MASK_OFF_RPL           ; mask away TI and RPL bits leaving only the descriptor offset
    1312     add     xAX, [xSP + 2]                      ; eax <- GDTR.address + descriptor offset
    1313     and     dword [xAX + 4], ~RT_BIT(9)         ; clear the busy flag in TSS desc (bits 0-7=base, bit 9=busy bit)
    1314     ltr     bx
    1315     add     xSP, xCB * 2
    1316  %endif
    1317 
    1318     pop     xAX         ; Saved LDTR
    1319     cmp     eax, 0
    1320     je      %%skip_ldt_write64
    1321     lldt    ax
    1322 
    1323 %%skip_ldt_write64:
    1324     pop     xSI         ; pCtx (needed in rsi by the macros below)
    1325 
    1326     ; Restore segment registers.
    1327     MYPOPSEGS xAX, ax
    1328 
    1329     ; Restore the host XCR0 if necessary.
    1330     pop     xCX
    1331     test    ecx, ecx
    1332     jnz     %%xcr0_after_skip
    1333     pop     xAX
    1334     pop     xDX
    1335     xsetbv                              ; ecx is already zero.
    1336 %%xcr0_after_skip:
    1337 
    1338     ; Restore general purpose registers.
    1339     MYPOPAD
    1340 %endmacro
    1341 
    1342 
    1343 ;;
    1344 ; Prepares for and executes VMLAUNCH/VMRESUME (64 bits guest mode)
    1345 ;
    1346 ; @returns VBox status code
    1347 ; @param    fResume    msc:rcx, gcc:rdi       Whether to use vmlauch/vmresume.
    1348 ; @param    pCtx       msc:rdx, gcc:rsi       Pointer to the guest-CPU context.
    1349 ; @param    pvUnused   msc:r8,  gcc:rdx       Unused argument.
    1350 ; @param    pVM        msc:r9,  gcc:rcx       The cross context VM structure.
    1351 ; @param    pVCpu      msc:[ebp+30], gcc:r8   The cross context virtual CPU structure of the calling EMT.
    1352 ;
    1353 ALIGNCODE(16)
    1354 BEGINPROC VMXR0StartVM64
    1355     push    xBP
    1356     mov     xBP, xSP
    1357 
    1358     pushf
    1359     cli
    1360 
    1361     ; Save all general purpose host registers.
    1362     MYPUSHAD
    1363 
    1364     ; First we have to save some final CPU context registers.
    1365     lea     r10, [.vmlaunch64_done wrt rip]
    1366     mov     rax, VMX_VMCS_HOST_RIP      ; return address (too difficult to continue after VMLAUNCH?)
    1367     vmwrite rax, r10
    1368     ; Note: ASSUMES success!
    1369 
    1370     ;
    1371     ; Unify the input parameter registers.
    1372     ;
    1373 %ifdef ASM_CALL64_GCC
    1374     ; fResume already in rdi
    1375     ; pCtx    already in rsi
    1376     mov     rbx, rdx        ; pvUnused
    1377 %else
    1378     mov     rdi, rcx        ; fResume
    1379     mov     rsi, rdx        ; pCtx
    1380     mov     rbx, r8         ; pvUnused
    1381 %endif
    1382 
    1383     ;
    1384     ; Save the host XCR0 and load the guest one if necessary.
    1385     ; Note! Trashes rdx and rcx.
    1386     ;
    1387 %ifdef ASM_CALL64_MSC
    1388     mov     rax, [xBP + 30h]            ; pVCpu
    1389 %else
    1390     mov     rax, r8                     ; pVCpu
    1391 %endif
    1392     test    byte [xAX + VMCPU.hm + HMCPU.fLoadSaveGuestXcr0], 1
    1393     jz      .xcr0_before_skip
    1394 
    1395     xor     ecx, ecx
    1396     xgetbv                              ; save the host one on the stack
    1397     push    xDX
    1398     push    xAX
    1399 
    1400     mov     eax, [xSI + CPUMCTX.aXcr]   ; load the guest one
    1401     mov     edx, [xSI + CPUMCTX.aXcr + 4]
    1402     xor     ecx, ecx                    ; paranoia
    1403     xsetbv
    1404 
    1405     push    0                           ; indicate that we must restore XCR0 (popped into ecx, thus 0)
    1406     jmp     .xcr0_before_done
    1407 
    1408 .xcr0_before_skip:
    1409     push    3fh                         ; indicate that we need not
    1410 .xcr0_before_done:
    1411 
    1412     ;
    1413     ; Save segment registers.
    1414     ; Note! Trashes rdx & rcx, so we moved it here (amd64 case).
    1415     ;
    1416     MYPUSHSEGS xAX, ax
    1417 
    1418     ; Save the pCtx pointer.
    1419     push    xSI
    1420 
    1421     ; Save host LDTR.
    1422     xor     eax, eax
    1423     sldt    ax
    1424     push    xAX
    1425 
    1426 %ifndef VMX_SKIP_TR
    1427     ; The host TR limit is reset to 0x67; save & restore it manually.
    1428     str     eax
    1429     push    xAX
    1430 %endif
    1431 
    1432 %ifndef VMX_SKIP_GDTR
    1433     ; VT-x only saves the base of the GDTR & IDTR and resets the limit to 0xffff; we must restore the limit correctly!
    1434     sub     xSP, xCB * 2
    1435     sgdt    [xSP]
    1436 %endif
    1437 %ifndef VMX_SKIP_IDTR
    1438     sub     xSP, xCB * 2
    1439     sidt    [xSP]
    1440 %endif
    1441 
    1442     ; Load CR2 if necessary (may be expensive as writing CR2 is a synchronizing instruction).
    1443     mov     rbx, qword [xSI + CPUMCTX.cr2]
    1444     mov     rdx, cr2
    1445     cmp     rbx, rdx
    1446     je      .skip_cr2_write
    1447     mov     cr2, rbx
    1448 
    1449 .skip_cr2_write:
    1450     mov     eax, VMX_VMCS_HOST_RSP
    1451     vmwrite xAX, xSP
    1452     ; Note: ASSUMES success!
    1453     ; Don't mess with ESP anymore!!!
    1454 
    1455     ; Fight spectre and similar.
    1456     INDIRECT_BRANCH_PREDICTION_AND_L1_CACHE_BARRIER xSI, CPUMCTX_WSF_IBPB_ENTRY, CPUMCTX_WSF_L1D_ENTRY, CPUMCTX_WSF_MDS_ENTRY
    1457 
    1458     ; Load guest general purpose registers.
    1459     mov     rax, qword [xSI + CPUMCTX.eax]
    1460     mov     rbx, qword [xSI + CPUMCTX.ebx]
    1461     mov     rcx, qword [xSI + CPUMCTX.ecx]
    1462     mov     rdx, qword [xSI + CPUMCTX.edx]
    1463     mov     rbp, qword [xSI + CPUMCTX.ebp]
    1464     mov     r8,  qword [xSI + CPUMCTX.r8]
    1465     mov     r9,  qword [xSI + CPUMCTX.r9]
    1466     mov     r10, qword [xSI + CPUMCTX.r10]
    1467     mov     r11, qword [xSI + CPUMCTX.r11]
    1468     mov     r12, qword [xSI + CPUMCTX.r12]
    1469     mov     r13, qword [xSI + CPUMCTX.r13]
    1470     mov     r14, qword [xSI + CPUMCTX.r14]
    1471     mov     r15, qword [xSI + CPUMCTX.r15]
    1472 
    1473     ; Resume or start VM?
    1474     cmp     xDI, 0                  ; fResume
    1475 
    1476     ; Load guest rdi & rsi.
    1477     mov     rdi, qword [xSI + CPUMCTX.edi]
    1478     mov     rsi, qword [xSI + CPUMCTX.esi]
    1479 
    1480     je      .vmlaunch64_launch
    1481 
    1482     vmresume
    1483     jc      near .vmxstart64_invalid_vmcs_ptr
    1484     jz      near .vmxstart64_start_failed
    1485     jmp     .vmlaunch64_done;      ; here if vmresume detected a failure
    1486 
    1487 .vmlaunch64_launch:
    1488     vmlaunch
    1489     jc      near .vmxstart64_invalid_vmcs_ptr
    1490     jz      near .vmxstart64_start_failed
    1491     jmp     .vmlaunch64_done;      ; here if vmlaunch detected a failure
    1492 
    1493 ALIGNCODE(16)
    1494 .vmlaunch64_done:
    1495     RESTORE_STATE_VM64
    1496     mov     eax, VINF_SUCCESS
    1497 
    1498 .vmstart64_end:
    1499     popf
    1500     pop     xBP
    1501     ret
    1502 
    1503 .vmxstart64_invalid_vmcs_ptr:
    1504     RESTORE_STATE_VM64
    1505     mov     eax, VERR_VMX_INVALID_VMCS_PTR_TO_START_VM
    1506     jmp     .vmstart64_end
    1507 
    1508 .vmxstart64_start_failed:
    1509     RESTORE_STATE_VM64
    1510     mov     eax, VERR_VMX_UNABLE_TO_START_VM
    1511     jmp     .vmstart64_end
    1512 ENDPROC VMXR0StartVM64
    1513 %endif ; RT_ARCH_AMD64
    1514 
    1515 
    1516 ;;
    1517 ; Clears the MDS buffers using VERW.
    1518 ALIGNCODE(16)
    1519 BEGINPROC hmR0MdsClear
    1520         sub     xSP, xCB
    1521         mov     [xSP], ds
    1522         verw    [xSP]
    1523         add     xSP, xCB
    1524         ret
    1525 ENDPROC   hmR0MdsClear
    1526 
    1527 
    1528 %ifdef RT_ARCH_AMD64
    1529 ;;
    1530 ; Prepares for and executes VMRUN (32-bit and 64-bit guests).
    1531 ;
    1532 ; @returns  VBox status code
    1533 ; @param    HCPhysVmcbHost  msc:rcx,gcc:rdi     Physical address of host VMCB.
    1534 ; @param    HCPhysVmcb      msc:rdx,gcc:rsi     Physical address of guest VMCB.
    1535 ; @param    pCtx            msc:r8,gcc:rdx      Pointer to the guest-CPU context.
    1536 ; @param    pVM             msc:r9,gcc:rcx      The cross context VM structure.
    1537 ; @param    pVCpu           msc:[rsp+28],gcc:r8 The cross context virtual CPU structure of the calling EMT.
    1538 ;
    1539 ALIGNCODE(16)
    1540 BEGINPROC SVMR0VMRun
    1541     ; Fake a cdecl stack frame
    1542  %ifdef ASM_CALL64_GCC
    1543     push    r8                ; pVCpu
    1544     push    rcx               ; pVM
    1545     push    rdx               ; pCtx
    1546     push    rsi               ; HCPhysVmcb
    1547     push    rdi               ; HCPhysVmcbHost
    1548  %else
    1549     mov     rax, [rsp + 28h]
    1550     push    rax               ; rbp + 30h pVCpu
    1551     push    r9                ; rbp + 28h pVM
    1552     push    r8                ; rbp + 20h pCtx
    1553     push    rdx               ; rbp + 18h HCPhysVmcb
    1554     push    rcx               ; rbp + 10h HCPhysVmcbHost
    1555  %endif
    1556     push    0                 ; rbp + 08h "fake ret addr"
    1557     push    rbp               ; rbp + 00h
    1558     mov     rbp, rsp
    1559     pushf
    1560 
    1561     ; Manual save and restore:
    1562     ;  - General purpose registers except RIP, RSP, RAX
    1563     ;
    1564     ; Trashed:
    1565     ;  - CR2 (we don't care)
    1566     ;  - LDTR (reset to 0)
    1567     ;  - DRx (presumably not changed at all)
    1568     ;  - DR7 (reset to 0x400)
    1569 
    1570     ; Save all general purpose host registers.
    1571     MYPUSHAD
    1572 
    1573     ; Load pCtx into xSI.
    1574     mov     xSI, [rbp + xCB * 2 + RTHCPHYS_CB * 2]
    1575 
    1576     ; Save the host XCR0 and load the guest one if necessary.
    1577     mov     rax, [xBP + 30h]                ; pVCpu
    1578     test    byte [xAX + VMCPU.hm + HMCPU.fLoadSaveGuestXcr0], 1
    1579     jz      .xcr0_before_skip
    1580 
    1581     xor     ecx, ecx
    1582     xgetbv                                  ; save the host XCR0 on the stack
    1583     push    xDX
    1584     push    xAX
    1585 
    1586     mov     xSI, [xBP + xCB * 2 + RTHCPHYS_CB * 2]  ; pCtx
    1587     mov     eax, [xSI + CPUMCTX.aXcr]       ; load the guest XCR0
    1588     mov     edx, [xSI + CPUMCTX.aXcr + 4]
    1589     xor     ecx, ecx                        ; paranoia
    1590     xsetbv
    1591 
    1592     push    0                               ; indicate that we must restore XCR0 (popped into ecx, thus 0)
    1593     jmp     .xcr0_before_done
    1594 
    1595 .xcr0_before_skip:
    1596     push    3fh                             ; indicate that we need not restore XCR0
    1597 .xcr0_before_done:
    1598 
    1599     ; Save guest CPU-context pointer for simplifying saving of the GPRs afterwards.
    1600     push    rsi
    1601 
    1602     ; Save host fs, gs, sysenter msr etc.
    1603     mov     rax, [rbp + xCB * 2]            ; HCPhysVmcbHost (64 bits physical address; x86: take low dword only)
    1604     push    rax                             ; save for the vmload after vmrun
    1605     vmsave
    1606 
    1607     ; Fight spectre.
    1608     INDIRECT_BRANCH_PREDICTION_BARRIER xSI, CPUMCTX_WSF_IBPB_ENTRY
    1609 
    1610     ; Setup rax for VMLOAD.
    1611     mov     rax, [rbp + xCB * 2 + RTHCPHYS_CB] ; HCPhysVmcb (64 bits physical address; take low dword only)
    1612 
    1613     ; Load guest general purpose registers (rax is loaded from the VMCB by VMRUN).
    1614     mov     rbx, qword [xSI + CPUMCTX.ebx]
    1615     mov     rcx, qword [xSI + CPUMCTX.ecx]
    1616     mov     rdx, qword [xSI + CPUMCTX.edx]
    1617     mov     rdi, qword [xSI + CPUMCTX.edi]
    1618     mov     rbp, qword [xSI + CPUMCTX.ebp]
    1619     mov     r8,  qword [xSI + CPUMCTX.r8]
    1620     mov     r9,  qword [xSI + CPUMCTX.r9]
    1621     mov     r10, qword [xSI + CPUMCTX.r10]
    1622     mov     r11, qword [xSI + CPUMCTX.r11]
    1623     mov     r12, qword [xSI + CPUMCTX.r12]
    1624     mov     r13, qword [xSI + CPUMCTX.r13]
    1625     mov     r14, qword [xSI + CPUMCTX.r14]
    1626     mov     r15, qword [xSI + CPUMCTX.r15]
    1627     mov     rsi, qword [xSI + CPUMCTX.esi]
    1628 
    1629     ; Clear the global interrupt flag & execute sti to make sure external interrupts cause a world switch.
    1630     clgi
    1631     sti
    1632 
    1633     ; Load guest FS, GS, Sysenter MSRs etc.
    1634     vmload
    1635 
    1636     ; Run the VM.
    1637     vmrun
    1638 
    1639     ; Save guest fs, gs, sysenter msr etc.
    1640     vmsave
    1641 
    1642     ; Load host fs, gs, sysenter msr etc.
    1643     pop     rax                         ; load HCPhysVmcbHost (pushed above)
    1644     vmload
    1645 
    1646     ; Set the global interrupt flag again, but execute cli to make sure IF=0.
    1647     cli
    1648     stgi
    1649 
    1650     ; Pop the context pointer (pushed above) and save the guest GPRs (sans RSP and RAX).
    1651     pop     rax
    1652 
    1653     mov     qword [rax + CPUMCTX.ebx], rbx
    1654     mov     rbx, SPECTRE_FILLER64
    1655     mov     qword [rax + CPUMCTX.ecx], rcx
    1656     mov     rcx, rbx
    1657     mov     qword [rax + CPUMCTX.edx], rdx
    1658     mov     rdx, rbx
    1659     mov     qword [rax + CPUMCTX.esi], rsi
    1660     mov     rsi, rbx
    1661     mov     qword [rax + CPUMCTX.edi], rdi
    1662     mov     rdi, rbx
    1663     mov     qword [rax + CPUMCTX.ebp], rbp
    1664     mov     rbp, rbx
    1665     mov     qword [rax + CPUMCTX.r8],  r8
    1666     mov     r8, rbx
    1667     mov     qword [rax + CPUMCTX.r9],  r9
    1668     mov     r9, rbx
    1669     mov     qword [rax + CPUMCTX.r10], r10
    1670     mov     r10, rbx
    1671     mov     qword [rax + CPUMCTX.r11], r11
    1672     mov     r11, rbx
    1673     mov     qword [rax + CPUMCTX.r12], r12
    1674     mov     r12, rbx
    1675     mov     qword [rax + CPUMCTX.r13], r13
    1676     mov     r13, rbx
    1677     mov     qword [rax + CPUMCTX.r14], r14
    1678     mov     r14, rbx
    1679     mov     qword [rax + CPUMCTX.r15], r15
    1680     mov     r15, rbx
    1681 
    1682     ; Fight spectre.  Note! Trashes rax!
    1683     INDIRECT_BRANCH_PREDICTION_BARRIER rax, CPUMCTX_WSF_IBPB_EXIT
    1684 
    1685     ; Restore the host xcr0 if necessary.
    1686     pop     xCX
    1687     test    ecx, ecx
    1688     jnz     .xcr0_after_skip
    1689     pop     xAX
    1690     pop     xDX
    1691     xsetbv                              ; ecx is already zero
    1692 .xcr0_after_skip:
    1693 
    1694     ; Restore host general purpose registers.
    1695     MYPOPAD
    1696 
    1697     mov     eax, VINF_SUCCESS
    1698 
    1699     popf
    1700     pop     rbp
    1701     add     rsp, 6 * xCB
    1702     ret
    1703 ENDPROC SVMR0VMRun
    1704 %endif ; RT_ARCH_AMD64
    1705 
Note: See TracChangeset for help on using the changeset viewer.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette