Changeset 73885 in vbox
- Timestamp:
- Aug 25, 2018 4:00:00 AM (6 years ago)
- Location:
- trunk
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/include/VBox/vmm/hm_vmx.h
r73868 r73885 2862 2862 kVmxVInstrDiag_Vmptrst_Cpl, 2863 2863 kVmxVInstrDiag_Vmptrst_PtrMap, 2864 kVmxVInstrDiag_Vmptrst_Success, 2864 2865 /* VMCLEAR. */ 2865 2866 kVmxVInstrDiag_Vmclear_Cpl, 2867 kVmxVInstrDiag_Vmclear_PtrAbnormal, 2868 kVmxVInstrDiag_Vmclear_PtrAlign, 2869 kVmxVInstrDiag_Vmclear_PtrMap, 2870 kVmxVInstrDiag_Vmclear_PtrReadPhys, 2871 kVmxVInstrDiag_Vmclear_PtrVmxon, 2872 kVmxVInstrDiag_Vmclear_PtrWidth, 2873 kVmxVInstrDiag_Vmclear_Success, 2866 2874 /* Last member for determining array index limit. */ 2867 2875 kVmxVInstrDiag_Last 2868 2876 } VMXVINSTRDIAG; 2869 2877 AssertCompileSize(VMXVINSTRDIAG, 4); 2878 2879 /** @name VMX_V_VMCS_STATE_XXX - Virtual VMCS state. 2880 * @{ */ 2881 /** VMCS state clear. */ 2882 #define VMX_V_VMCS_STATE_CLEAR RT_BIT(0) 2883 /** VMCS state launched. */ 2884 #define VMX_V_VMCS_STATE_LAUNCHED RT_BIT(1) 2885 /** @} */ 2870 2886 2871 2887 /** … … 2876 2892 * The first 8 bytes are as per Intel spec. 24.2 "Format of the VMCS Region". 2877 2893 * 2894 * The offset and size of the VMCS state field (fVmcsState) is also fixed as we use 2895 * it to offset into guest memory. 2896 * 2878 2897 * Although the guest is supposed to access the VMCS only through the execution of 2879 2898 * VMX instructions (VMREAD, VMWRITE etc.), since the VMCS may reside in guest … … 2889 2908 /** 0x4 - VMX-abort indicator. */ 2890 2909 uint32_t u32VmxAbortId; 2891 /** 0x8 - Reserved for future. */ 2892 uint32_t au32Reserved0[8]; 2910 /** 0x8 - VMCS state, see VMX_V_VMCS_STATE_XXX. */ 2911 uint8_t fVmcsState; 2912 /** 0x9 - Reserved for future. */ 2913 uint8_t au8Padding0[3]; 2914 /** 0xc - Reserved for future. */ 2915 uint32_t au32Reserved0[7]; 2893 2916 2894 2917 /** @name 16-bit control fields. … … 3300 3323 typedef const VMXVVMCS *PCVMXVVMCS; 3301 3324 AssertCompileSize(VMXVVMCS, X86_PAGE_4K_SIZE); 3325 AssertCompileMemberSize(VMXVVMCS, fVmcsState, sizeof(uint8_t)); 3302 3326 AssertCompileMemberOffset(VMXVVMCS, u32VmxAbortId, 0x004); 3327 AssertCompileMemberOffset(VMXVVMCS, fVmcsState, 0x008); 3303 3328 AssertCompileMemberOffset(VMXVVMCS, u16Vpid, 0x028); 3304 3329 AssertCompileMemberOffset(VMXVVMCS, GuestEs, 0x03e); -
trunk/src/VBox/VMM/VMMAll/HMVMXAll.cpp
r73756 r73885 85 85 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmptrst_Cpl , "Cpl" ), 86 86 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmptrst_PtrMap , "PtrMap" ), 87 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmptrst_Success , "Success" ), 87 88 /* VMCLEAR. */ 88 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_Cpl , "Cpl" ) 89 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_Cpl , "Cpl" ), 90 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrAbnormal , "PtrAbnormal" ), 91 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrAlign , "PtrAlign" ), 92 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrMap , "PtrMap" ), 93 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrReadPhys , "PtrReadPhys" ), 94 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrVmxon , "PtrVmxon" ), 95 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_PtrWidth , "PtrWidth" ), 96 VMX_INSTR_DIAG_DESC(kVmxVInstrDiag_Vmclear_Success , "Success" ) 89 97 /* kVmxVInstrDiag_Last */ 90 98 }; -
trunk/src/VBox/VMM/VMMAll/IEMAllCImplVmxInstr.cpp.h
r73870 r73885 55 55 /* VMX_VMCS_ENC_WIDTH_16BIT | VMX_VMCS_ENC_TYPE_GUEST_STATE: */ 56 56 { 57 /* 0 */ RT_OFFSET (VMXVVMCS, GuestEs),58 /* 1 */ RT_OFFSET (VMXVVMCS, GuestCs),59 /* 2 */ RT_OFFSET (VMXVVMCS, GuestSs),60 /* 3 */ RT_OFFSET (VMXVVMCS, GuestDs),61 /* 4 */ RT_OFFSET (VMXVVMCS, GuestFs),62 /* 5 */ RT_OFFSET (VMXVVMCS, GuestGs),63 /* 6 */ RT_OFFSET (VMXVVMCS, GuestLdtr),64 /* 7 */ RT_OFFSET (VMXVVMCS, GuestTr),65 /* 8 */ RT_OFFSET (VMXVVMCS, u16GuestIntStatus),66 /* 9 */ RT_OFFSET (VMXVVMCS, u16PmlIndex),57 /* 0 */ RT_OFFSETOF(VMXVVMCS, GuestEs), 58 /* 1 */ RT_OFFSETOF(VMXVVMCS, GuestCs), 59 /* 2 */ RT_OFFSETOF(VMXVVMCS, GuestSs), 60 /* 3 */ RT_OFFSETOF(VMXVVMCS, GuestDs), 61 /* 4 */ RT_OFFSETOF(VMXVVMCS, GuestFs), 62 /* 5 */ RT_OFFSETOF(VMXVVMCS, GuestGs), 63 /* 6 */ RT_OFFSETOF(VMXVVMCS, GuestLdtr), 64 /* 7 */ RT_OFFSETOF(VMXVVMCS, GuestTr), 65 /* 8 */ RT_OFFSETOF(VMXVVMCS, u16GuestIntStatus), 66 /* 9 */ RT_OFFSETOF(VMXVVMCS, u16PmlIndex), 67 67 /* 10-17 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 68 68 /* 18-25 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX … … 70 70 /* VMX_VMCS_ENC_WIDTH_16BIT | VMX_VMCS_ENC_TYPE_HOST_STATE: */ 71 71 { 72 /* 0 */ RT_OFFSET (VMXVVMCS, HostEs),73 /* 1 */ RT_OFFSET (VMXVVMCS, HostCs),74 /* 2 */ RT_OFFSET (VMXVVMCS, HostSs),75 /* 3 */ RT_OFFSET (VMXVVMCS, HostDs),76 /* 4 */ RT_OFFSET (VMXVVMCS, HostFs),77 /* 5 */ RT_OFFSET (VMXVVMCS, HostGs),78 /* 6 */ RT_OFFSET (VMXVVMCS, HostTr),72 /* 0 */ RT_OFFSETOF(VMXVVMCS, HostEs), 73 /* 1 */ RT_OFFSETOF(VMXVVMCS, HostCs), 74 /* 2 */ RT_OFFSETOF(VMXVVMCS, HostSs), 75 /* 3 */ RT_OFFSETOF(VMXVVMCS, HostDs), 76 /* 4 */ RT_OFFSETOF(VMXVVMCS, HostFs), 77 /* 5 */ RT_OFFSETOF(VMXVVMCS, HostGs), 78 /* 6 */ RT_OFFSETOF(VMXVVMCS, HostTr), 79 79 /* 7-14 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 80 80 /* 15-22 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 112 112 /* VMX_VMCS_ENC_WIDTH_64BIT | VMX_VMCS_ENC_TYPE_VMEXIT_INFO: */ 113 113 { 114 /* 0 */ RT_OFFSET (VMXVVMCS, u64GuestPhysAddr),115 /* 1-8 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 114 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64GuestPhysAddr), 115 /* 1-8 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 116 116 /* 9-16 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 117 117 /* 17-24 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 120 120 /* VMX_VMCS_ENC_WIDTH_64BIT | VMX_VMCS_ENC_TYPE_GUEST_STATE: */ 121 121 { 122 /* 0 */ RT_OFFSET (VMXVVMCS, u64VmcsLinkPtr),123 /* 1 */ RT_OFFSET (VMXVVMCS, u64GuestDebugCtlMsr),124 /* 2 */ RT_OFFSET (VMXVVMCS, u64GuestPatMsr),125 /* 3 */ RT_OFFSET (VMXVVMCS, u64GuestEferMsr),126 /* 4 */ RT_OFFSET (VMXVVMCS, u64GuestPerfGlobalCtlMsr),127 /* 5 */ RT_OFFSET (VMXVVMCS, u64GuestPdpte0),128 /* 6 */ RT_OFFSET (VMXVVMCS, u64GuestPdpte1),129 /* 7 */ RT_OFFSET (VMXVVMCS, u64GuestPdpte2),130 /* 8 */ RT_OFFSET (VMXVVMCS, u64GuestPdpte3),131 /* 9 */ RT_OFFSET (VMXVVMCS, u64GuestBndcfgsMsr),132 /* 10-17 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 122 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64VmcsLinkPtr), 123 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64GuestDebugCtlMsr), 124 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64GuestPatMsr), 125 /* 3 */ RT_OFFSETOF(VMXVVMCS, u64GuestEferMsr), 126 /* 4 */ RT_OFFSETOF(VMXVVMCS, u64GuestPerfGlobalCtlMsr), 127 /* 5 */ RT_OFFSETOF(VMXVVMCS, u64GuestPdpte0), 128 /* 6 */ RT_OFFSETOF(VMXVVMCS, u64GuestPdpte1), 129 /* 7 */ RT_OFFSETOF(VMXVVMCS, u64GuestPdpte2), 130 /* 8 */ RT_OFFSETOF(VMXVVMCS, u64GuestPdpte3), 131 /* 9 */ RT_OFFSETOF(VMXVVMCS, u64GuestBndcfgsMsr), 132 /* 10-17 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 133 133 /* 18-25 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 134 134 }, 135 135 /* VMX_VMCS_ENC_WIDTH_64BIT | VMX_VMCS_ENC_TYPE_HOST_STATE: */ 136 136 { 137 /* 0 */ RT_OFFSET (VMXVVMCS, u64HostPatMsr),138 /* 1 */ RT_OFFSET (VMXVVMCS, u64HostEferMsr),139 /* 2 */ RT_OFFSET (VMXVVMCS, u64HostPerfGlobalCtlMsr),137 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64HostPatMsr), 138 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64HostEferMsr), 139 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64HostPerfGlobalCtlMsr), 140 140 /* 3-10 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 141 141 /* 11-18 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 144 144 /* VMX_VMCS_ENC_WIDTH_32BIT | VMX_VMCS_ENC_TYPE_CONTROL: */ 145 145 { 146 /* 0 */ RT_OFFSET (VMXVVMCS, u32PinCtls),147 /* 1 */ RT_OFFSET (VMXVVMCS, u32ProcCtls),148 /* 2 */ RT_OFFSET (VMXVVMCS, u32XcptBitmap),149 /* 3 */ RT_OFFSET (VMXVVMCS, u32XcptPFMask),150 /* 4 */ RT_OFFSET (VMXVVMCS, u32XcptPFMatch),151 /* 5 */ RT_OFFSET (VMXVVMCS, u32Cr3TargetCount),152 /* 6 */ RT_OFFSET (VMXVVMCS, u32ExitCtls),153 /* 7 */ RT_OFFSET (VMXVVMCS, u32ExitMsrStoreCount),154 /* 8 */ RT_OFFSET (VMXVVMCS, u32ExitMsrLoadCount),155 /* 9 */ RT_OFFSET (VMXVVMCS, u32EntryCtls),156 /* 10 */ RT_OFFSET (VMXVVMCS, u32EntryMsrLoadCount),157 /* 11 */ RT_OFFSET (VMXVVMCS, u32EntryIntInfo),158 /* 12 */ RT_OFFSET (VMXVVMCS, u32EntryXcptErrCode),159 /* 13 */ RT_OFFSET (VMXVVMCS, u32EntryInstrLen),160 /* 14 */ RT_OFFSET (VMXVVMCS, u32TprTreshold),161 /* 15 */ RT_OFFSET (VMXVVMCS, u32ProcCtls2),162 /* 16 */ RT_OFFSET (VMXVVMCS, u32PleGap),163 /* 17 */ RT_OFFSET (VMXVVMCS, u32PleWindow),146 /* 0 */ RT_OFFSETOF(VMXVVMCS, u32PinCtls), 147 /* 1 */ RT_OFFSETOF(VMXVVMCS, u32ProcCtls), 148 /* 2 */ RT_OFFSETOF(VMXVVMCS, u32XcptBitmap), 149 /* 3 */ RT_OFFSETOF(VMXVVMCS, u32XcptPFMask), 150 /* 4 */ RT_OFFSETOF(VMXVVMCS, u32XcptPFMatch), 151 /* 5 */ RT_OFFSETOF(VMXVVMCS, u32Cr3TargetCount), 152 /* 6 */ RT_OFFSETOF(VMXVVMCS, u32ExitCtls), 153 /* 7 */ RT_OFFSETOF(VMXVVMCS, u32ExitMsrStoreCount), 154 /* 8 */ RT_OFFSETOF(VMXVVMCS, u32ExitMsrLoadCount), 155 /* 9 */ RT_OFFSETOF(VMXVVMCS, u32EntryCtls), 156 /* 10 */ RT_OFFSETOF(VMXVVMCS, u32EntryMsrLoadCount), 157 /* 11 */ RT_OFFSETOF(VMXVVMCS, u32EntryIntInfo), 158 /* 12 */ RT_OFFSETOF(VMXVVMCS, u32EntryXcptErrCode), 159 /* 13 */ RT_OFFSETOF(VMXVVMCS, u32EntryInstrLen), 160 /* 14 */ RT_OFFSETOF(VMXVVMCS, u32TprTreshold), 161 /* 15 */ RT_OFFSETOF(VMXVVMCS, u32ProcCtls2), 162 /* 16 */ RT_OFFSETOF(VMXVVMCS, u32PleGap), 163 /* 17 */ RT_OFFSETOF(VMXVVMCS, u32PleWindow), 164 164 /* 18-25 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 165 165 }, 166 166 /* VMX_VMCS_ENC_WIDTH_32BIT | VMX_VMCS_ENC_TYPE_VMEXIT_INFO: */ 167 167 { 168 /* 0 */ RT_OFFSET (VMXVVMCS, u32RoVmInstrError),169 /* 1 */ RT_OFFSET (VMXVVMCS, u32RoVmExitReason),170 /* 2 */ RT_OFFSET (VMXVVMCS, u32RoVmExitIntInfo),171 /* 3 */ RT_OFFSET (VMXVVMCS, u32RoVmExitErrCode),172 /* 4 */ RT_OFFSET (VMXVVMCS, u32RoIdtVectoringInfo),173 /* 5 */ RT_OFFSET (VMXVVMCS, u32RoIdtVectoringErrCode),174 /* 6 */ RT_OFFSET (VMXVVMCS, u32RoVmExitInstrLen),175 /* 7 */ RT_OFFSET (VMXVVMCS, u32RoVmExitInstrInfo),168 /* 0 */ RT_OFFSETOF(VMXVVMCS, u32RoVmInstrError), 169 /* 1 */ RT_OFFSETOF(VMXVVMCS, u32RoVmExitReason), 170 /* 2 */ RT_OFFSETOF(VMXVVMCS, u32RoVmExitIntInfo), 171 /* 3 */ RT_OFFSETOF(VMXVVMCS, u32RoVmExitErrCode), 172 /* 4 */ RT_OFFSETOF(VMXVVMCS, u32RoIdtVectoringInfo), 173 /* 5 */ RT_OFFSETOF(VMXVVMCS, u32RoIdtVectoringErrCode), 174 /* 6 */ RT_OFFSETOF(VMXVVMCS, u32RoVmExitInstrLen), 175 /* 7 */ RT_OFFSETOF(VMXVVMCS, u32RoVmExitInstrInfo), 176 176 /* 8-15 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 177 177 /* 16-23 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 180 180 /* VMX_VMCS_ENC_WIDTH_32BIT | VMX_VMCS_ENC_TYPE_GUEST_STATE: */ 181 181 { 182 /* 0 */ RT_OFFSET (VMXVVMCS, u32GuestEsLimit),183 /* 1 */ RT_OFFSET (VMXVVMCS, u32GuestCsLimit),184 /* 2 */ RT_OFFSET (VMXVVMCS, u32GuestSsLimit),185 /* 3 */ RT_OFFSET (VMXVVMCS, u32GuestDsLimit),186 /* 4 */ RT_OFFSET (VMXVVMCS, u32GuestEsLimit),187 /* 5 */ RT_OFFSET (VMXVVMCS, u32GuestFsLimit),188 /* 6 */ RT_OFFSET (VMXVVMCS, u32GuestGsLimit),189 /* 7 */ RT_OFFSET (VMXVVMCS, u32GuestLdtrLimit),190 /* 8 */ RT_OFFSET (VMXVVMCS, u32GuestTrLimit),191 /* 9 */ RT_OFFSET (VMXVVMCS, u32GuestGdtrLimit),192 /* 10 */ RT_OFFSET (VMXVVMCS, u32GuestIdtrLimit),193 /* 11 */ RT_OFFSET (VMXVVMCS, u32GuestEsAttr),194 /* 12 */ RT_OFFSET (VMXVVMCS, u32GuestCsAttr),195 /* 13 */ RT_OFFSET (VMXVVMCS, u32GuestSsAttr),196 /* 14 */ RT_OFFSET (VMXVVMCS, u32GuestDsAttr),197 /* 15 */ RT_OFFSET (VMXVVMCS, u32GuestFsAttr),198 /* 16 */ RT_OFFSET (VMXVVMCS, u32GuestGsAttr),199 /* 17 */ RT_OFFSET (VMXVVMCS, u32GuestLdtrAttr),200 /* 18 */ RT_OFFSET (VMXVVMCS, u32GuestTrAttr),201 /* 19 */ RT_OFFSET (VMXVVMCS, u32GuestIntrState),202 /* 20 */ RT_OFFSET (VMXVVMCS, u32GuestActivityState),203 /* 21 */ RT_OFFSET (VMXVVMCS, u32GuestSmBase),204 /* 22 */ RT_OFFSET (VMXVVMCS, u32GuestSysenterCS),205 /* 23 */ RT_OFFSET (VMXVVMCS, u32PreemptTimer),182 /* 0 */ RT_OFFSETOF(VMXVVMCS, u32GuestEsLimit), 183 /* 1 */ RT_OFFSETOF(VMXVVMCS, u32GuestCsLimit), 184 /* 2 */ RT_OFFSETOF(VMXVVMCS, u32GuestSsLimit), 185 /* 3 */ RT_OFFSETOF(VMXVVMCS, u32GuestDsLimit), 186 /* 4 */ RT_OFFSETOF(VMXVVMCS, u32GuestEsLimit), 187 /* 5 */ RT_OFFSETOF(VMXVVMCS, u32GuestFsLimit), 188 /* 6 */ RT_OFFSETOF(VMXVVMCS, u32GuestGsLimit), 189 /* 7 */ RT_OFFSETOF(VMXVVMCS, u32GuestLdtrLimit), 190 /* 8 */ RT_OFFSETOF(VMXVVMCS, u32GuestTrLimit), 191 /* 9 */ RT_OFFSETOF(VMXVVMCS, u32GuestGdtrLimit), 192 /* 10 */ RT_OFFSETOF(VMXVVMCS, u32GuestIdtrLimit), 193 /* 11 */ RT_OFFSETOF(VMXVVMCS, u32GuestEsAttr), 194 /* 12 */ RT_OFFSETOF(VMXVVMCS, u32GuestCsAttr), 195 /* 13 */ RT_OFFSETOF(VMXVVMCS, u32GuestSsAttr), 196 /* 14 */ RT_OFFSETOF(VMXVVMCS, u32GuestDsAttr), 197 /* 15 */ RT_OFFSETOF(VMXVVMCS, u32GuestFsAttr), 198 /* 16 */ RT_OFFSETOF(VMXVVMCS, u32GuestGsAttr), 199 /* 17 */ RT_OFFSETOF(VMXVVMCS, u32GuestLdtrAttr), 200 /* 18 */ RT_OFFSETOF(VMXVVMCS, u32GuestTrAttr), 201 /* 19 */ RT_OFFSETOF(VMXVVMCS, u32GuestIntrState), 202 /* 20 */ RT_OFFSETOF(VMXVVMCS, u32GuestActivityState), 203 /* 21 */ RT_OFFSETOF(VMXVVMCS, u32GuestSmBase), 204 /* 22 */ RT_OFFSETOF(VMXVVMCS, u32GuestSysenterCS), 205 /* 23 */ RT_OFFSETOF(VMXVVMCS, u32PreemptTimer), 206 206 /* 24-25 */ UINT16_MAX, UINT16_MAX 207 207 }, 208 208 /* VMX_VMCS_ENC_WIDTH_32BIT | VMX_VMCS_ENC_TYPE_HOST_STATE: */ 209 209 { 210 /* 0 */ RT_OFFSET (VMXVVMCS, u32HostSysenterCs),211 /* 1-8 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 210 /* 0 */ RT_OFFSETOF(VMXVVMCS, u32HostSysenterCs), 211 /* 1-8 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 212 212 /* 9-16 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 213 213 /* 17-24 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 216 216 /* VMX_VMCS_ENC_WIDTH_NATURAL | VMX_VMCS_ENC_TYPE_CONTROL: */ 217 217 { 218 /* 0 */ RT_OFFSET (VMXVVMCS, u64Cr0Mask),219 /* 1 */ RT_OFFSET (VMXVVMCS, u64Cr4Mask),220 /* 2 */ RT_OFFSET (VMXVVMCS, u64Cr0ReadShadow),221 /* 3 */ RT_OFFSET (VMXVVMCS, u64Cr4ReadShadow),222 /* 4 */ RT_OFFSET (VMXVVMCS, u64Cr3Target0),223 /* 5 */ RT_OFFSET (VMXVVMCS, u64Cr3Target1),224 /* 6 */ RT_OFFSET (VMXVVMCS, u64Cr3Target2),225 /* 7 */ RT_OFFSET (VMXVVMCS, u64Cr3Target3),218 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64Cr0Mask), 219 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64Cr4Mask), 220 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64Cr0ReadShadow), 221 /* 3 */ RT_OFFSETOF(VMXVVMCS, u64Cr4ReadShadow), 222 /* 4 */ RT_OFFSETOF(VMXVVMCS, u64Cr3Target0), 223 /* 5 */ RT_OFFSETOF(VMXVVMCS, u64Cr3Target1), 224 /* 6 */ RT_OFFSETOF(VMXVVMCS, u64Cr3Target2), 225 /* 7 */ RT_OFFSETOF(VMXVVMCS, u64Cr3Target3), 226 226 /* 8-15 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 227 227 /* 16-23 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 230 230 /* VMX_VMCS_ENC_WIDTH_NATURAL | VMX_VMCS_ENC_TYPE_VMEXIT_INFO: */ 231 231 { 232 /* 0 */ RT_OFFSET (VMXVVMCS, u64ExitQual),233 /* 1 */ RT_OFFSET (VMXVVMCS, u64IoRcx),234 /* 2 */ RT_OFFSET (VMXVVMCS, u64IoRsi),235 /* 3 */ RT_OFFSET (VMXVVMCS, u64IoRdi),236 /* 4 */ RT_OFFSET (VMXVVMCS, u64IoRip),237 /* 5 */ RT_OFFSET (VMXVVMCS, u64GuestLinearAddr),232 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64ExitQual), 233 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64IoRcx), 234 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64IoRsi), 235 /* 3 */ RT_OFFSETOF(VMXVVMCS, u64IoRdi), 236 /* 4 */ RT_OFFSETOF(VMXVVMCS, u64IoRip), 237 /* 5 */ RT_OFFSETOF(VMXVVMCS, u64GuestLinearAddr), 238 238 /* 6-13 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 239 239 /* 14-21 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, … … 242 242 /* VMX_VMCS_ENC_WIDTH_NATURAL | VMX_VMCS_ENC_TYPE_GUEST_STATE: */ 243 243 { 244 /* 0 */ RT_OFFSET (VMXVVMCS, u64GuestCr0),245 /* 1 */ RT_OFFSET (VMXVVMCS, u64GuestCr3),246 /* 2 */ RT_OFFSET (VMXVVMCS, u64GuestCr4),247 /* 3 */ RT_OFFSET (VMXVVMCS, u64GuestEsBase),248 /* 4 */ RT_OFFSET (VMXVVMCS, u64GuestCsBase),249 /* 5 */ RT_OFFSET (VMXVVMCS, u64GuestSsBase),250 /* 6 */ RT_OFFSET (VMXVVMCS, u64GuestDsBase),251 /* 7 */ RT_OFFSET (VMXVVMCS, u64GuestFsBase),252 /* 8 */ RT_OFFSET (VMXVVMCS, u64GuestGsBase),253 /* 9 */ RT_OFFSET (VMXVVMCS, u64GuestLdtrBase),254 /* 10 */ RT_OFFSET (VMXVVMCS, u64GuestTrBase),255 /* 11 */ RT_OFFSET (VMXVVMCS, u64GuestGdtrBase),256 /* 12 */ RT_OFFSET (VMXVVMCS, u64GuestIdtrBase),257 /* 13 */ RT_OFFSET (VMXVVMCS, u64GuestDr7),258 /* 14 */ RT_OFFSET (VMXVVMCS, u64GuestRsp),259 /* 15 */ RT_OFFSET (VMXVVMCS, u64GuestRip),260 /* 16 */ RT_OFFSET (VMXVVMCS, u64GuestRFlags),261 /* 17 */ RT_OFFSET (VMXVVMCS, u64GuestPendingDbgXcpt),262 /* 18 */ RT_OFFSET (VMXVVMCS, u64GuestSysenterEsp),263 /* 19 */ RT_OFFSET (VMXVVMCS, u64GuestSysenterEip),244 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64GuestCr0), 245 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64GuestCr3), 246 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64GuestCr4), 247 /* 3 */ RT_OFFSETOF(VMXVVMCS, u64GuestEsBase), 248 /* 4 */ RT_OFFSETOF(VMXVVMCS, u64GuestCsBase), 249 /* 5 */ RT_OFFSETOF(VMXVVMCS, u64GuestSsBase), 250 /* 6 */ RT_OFFSETOF(VMXVVMCS, u64GuestDsBase), 251 /* 7 */ RT_OFFSETOF(VMXVVMCS, u64GuestFsBase), 252 /* 8 */ RT_OFFSETOF(VMXVVMCS, u64GuestGsBase), 253 /* 9 */ RT_OFFSETOF(VMXVVMCS, u64GuestLdtrBase), 254 /* 10 */ RT_OFFSETOF(VMXVVMCS, u64GuestTrBase), 255 /* 11 */ RT_OFFSETOF(VMXVVMCS, u64GuestGdtrBase), 256 /* 12 */ RT_OFFSETOF(VMXVVMCS, u64GuestIdtrBase), 257 /* 13 */ RT_OFFSETOF(VMXVVMCS, u64GuestDr7), 258 /* 14 */ RT_OFFSETOF(VMXVVMCS, u64GuestRsp), 259 /* 15 */ RT_OFFSETOF(VMXVVMCS, u64GuestRip), 260 /* 16 */ RT_OFFSETOF(VMXVVMCS, u64GuestRFlags), 261 /* 17 */ RT_OFFSETOF(VMXVVMCS, u64GuestPendingDbgXcpt), 262 /* 18 */ RT_OFFSETOF(VMXVVMCS, u64GuestSysenterEsp), 263 /* 19 */ RT_OFFSETOF(VMXVVMCS, u64GuestSysenterEip), 264 264 /* 20-25 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX 265 265 }, 266 266 /* VMX_VMCS_ENC_WIDTH_NATURAL | VMX_VMCS_ENC_TYPE_HOST_STATE: */ 267 267 { 268 /* 0 */ RT_OFFSET (VMXVVMCS, u64HostCr0),269 /* 1 */ RT_OFFSET (VMXVVMCS, u64HostCr3),270 /* 2 */ RT_OFFSET (VMXVVMCS, u64HostCr4),271 /* 3 */ RT_OFFSET (VMXVVMCS, u64HostFsBase),272 /* 4 */ RT_OFFSET (VMXVVMCS, u64HostGsBase),273 /* 5 */ RT_OFFSET (VMXVVMCS, u64HostTrBase),274 /* 6 */ RT_OFFSET (VMXVVMCS, u64HostGdtrBase),275 /* 7 */ RT_OFFSET (VMXVVMCS, u64HostIdtrBase),276 /* 8 */ RT_OFFSET (VMXVVMCS, u64HostSysenterEsp),277 /* 9 */ RT_OFFSET (VMXVVMCS, u64HostSysenterEip),278 /* 10 */ RT_OFFSET (VMXVVMCS, u64HostRsp),279 /* 11 */ RT_OFFSET (VMXVVMCS, u64HostRip),268 /* 0 */ RT_OFFSETOF(VMXVVMCS, u64HostCr0), 269 /* 1 */ RT_OFFSETOF(VMXVVMCS, u64HostCr3), 270 /* 2 */ RT_OFFSETOF(VMXVVMCS, u64HostCr4), 271 /* 3 */ RT_OFFSETOF(VMXVVMCS, u64HostFsBase), 272 /* 4 */ RT_OFFSETOF(VMXVVMCS, u64HostGsBase), 273 /* 5 */ RT_OFFSETOF(VMXVVMCS, u64HostTrBase), 274 /* 6 */ RT_OFFSETOF(VMXVVMCS, u64HostGdtrBase), 275 /* 7 */ RT_OFFSETOF(VMXVVMCS, u64HostIdtrBase), 276 /* 8 */ RT_OFFSETOF(VMXVVMCS, u64HostSysenterEsp), 277 /* 9 */ RT_OFFSETOF(VMXVVMCS, u64HostSysenterEip), 278 /* 10 */ RT_OFFSETOF(VMXVVMCS, u64HostRsp), 279 /* 11 */ RT_OFFSETOF(VMXVVMCS, u64HostRip), 280 280 /* 12-19 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 281 281 /* 20-25 */ UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX … … 360 360 } while (0) 361 361 # endif /* !IEM_WITH_CODE_TLB */ 362 363 /** Whether a current VMCS is present for the given VCPU. */ 364 #define IEM_VMX_HAS_CURRENT_VMCS(a_pVCpu) RT_BOOL((a_pVCpu)->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs != NIL_RTGCPHYS) 365 366 /** Gets the current VMCS for the given VCPU. */ 367 #define IEM_VMX_GET_CURRENT_VMCS(a_pVCpu) ((a_pVCpu)->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs) 368 369 /** Sets a new VMCS as the current VMCS for the given VCPU. */ 370 #define IEM_VMX_SET_CURRENT_VMCS(a_pVCpu, a_GCPhysVmcs) \ 371 do \ 372 { \ 373 Assert((a_GCPhysVmcs) != NIL_RTGCPHYS); \ 374 (a_pVCpu)->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs = (a_GCPhysVmcs); \ 375 } while (0) 376 377 /** Clears any current VMCS for the given VCPU. */ 378 #define IEM_VMX_CLEAR_CURRENT_VMCS(a_pVCpu) \ 379 do \ 380 { \ 381 (a_pVCpu)->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs = NIL_RTGCPHYS; \ 382 } while (0) 362 383 363 384 … … 966 987 DECLINLINE(void) iemVmxVmFailValid(PVMCPU pVCpu, VMXINSTRERR enmInsErr) 967 988 { 968 if ( pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs != NIL_RTGCPHYS)989 if (IEM_VMX_HAS_CURRENT_VMCS(pVCpu)) 969 990 { 970 991 pVCpu->cpum.GstCtx.eflags.u32 &= ~(X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF); … … 984 1005 DECLINLINE(void) iemVmxVmFail(PVMCPU pVCpu, VMXINSTRERR enmInsErr) 985 1006 { 986 if ( pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs != NIL_RTGCPHYS)1007 if (IEM_VMX_HAS_CURRENT_VMCS(pVCpu)) 987 1008 { 988 1009 iemVmxVmFailValid(pVCpu, enmInsErr); … … 991 1012 else 992 1013 iemVmxVmFailInvalid(pVCpu); 1014 } 1015 1016 1017 /** 1018 * Flushes the current VMCS contents back to guest memory. 1019 * 1020 * @returns VBox status code. 1021 * @param pVCpu The cross context virtual CPU structure. 1022 */ 1023 DECLINLINE(int) iemVmxCommitCurrentVmcsToMemory(PVMCPU pVCpu) 1024 { 1025 Assert(IEM_VMX_HAS_CURRENT_VMCS(pVCpu)); 1026 int rc = PGMPhysSimpleWriteGCPhys(pVCpu->CTX_SUFF(pVM), IEM_VMX_GET_CURRENT_VMCS(pVCpu), 1027 pVCpu->cpum.GstCtx.hwvirt.vmx.CTX_SUFF(pVmcs), sizeof(VMXVVMCS)); 1028 IEM_VMX_CLEAR_CURRENT_VMCS(pVCpu); 1029 return rc; 993 1030 } 994 1031 … … 1024 1061 } 1025 1062 1026 /** @todo NSTVMX: VMCLEAR impl. */ 1027 RT_NOREF(GCPtrVmcs); RT_NOREF(pExitInstrInfo); RT_NOREF(cbInstr); 1028 return VINF_SUCCESS; 1063 /* Get the VMCS pointer from the location specified by the source memory operand. */ 1064 RTGCPHYS GCPhysVmcs; 1065 VBOXSTRICTRC rcStrict = iemMemFetchDataU64(pVCpu, &GCPhysVmcs, pExitInstrInfo->VmxXsave.iSegReg, GCPtrVmcs); 1066 if (RT_UNLIKELY(rcStrict != VINF_SUCCESS)) 1067 { 1068 Log(("vmclear: Failed to read VMCS physaddr from %#RGv, rc=%Rrc\n", GCPtrVmcs, VBOXSTRICTRC_VAL(rcStrict))); 1069 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmptrld_PtrMap; 1070 return rcStrict; 1071 } 1072 1073 /* VMCS pointer alignment. */ 1074 if (GCPhysVmcs & X86_PAGE_4K_OFFSET_MASK) 1075 { 1076 Log(("vmclear: VMCS pointer not page-aligned -> VMFail()\n")); 1077 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmclear_PtrAlign; 1078 iemVmxVmFail(pVCpu, VMXINSTRERR_VMCLEAR_INVALID_PHYSADDR); 1079 iemRegAddToRipAndClearRF(pVCpu, cbInstr); 1080 return VINF_SUCCESS; 1081 } 1082 1083 /* VMCS physical-address width limits. */ 1084 Assert(!VMX_V_VMCS_PHYSADDR_4G_LIMIT); 1085 if (GCPhysVmcs >> IEM_GET_GUEST_CPU_FEATURES(pVCpu)->cMaxPhysAddrWidth) 1086 { 1087 Log(("vmclear: VMCS pointer extends beyond physical-address width -> VMFail()\n")); 1088 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmclear_PtrWidth; 1089 iemVmxVmFail(pVCpu, VMXINSTRERR_VMCLEAR_INVALID_PHYSADDR); 1090 iemRegAddToRipAndClearRF(pVCpu, cbInstr); 1091 return VINF_SUCCESS; 1092 } 1093 1094 /* VMCS is not the VMXON region. */ 1095 if (GCPhysVmcs == pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmxon) 1096 { 1097 Log(("vmclear: VMCS pointer cannot be identical to VMXON region pointer -> VMFail()\n")); 1098 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmclear_PtrVmxon; 1099 iemVmxVmFail(pVCpu, VMXINSTRERR_VMCLEAR_VMXON_PTR); 1100 iemRegAddToRipAndClearRF(pVCpu, cbInstr); 1101 return VINF_SUCCESS; 1102 } 1103 1104 /* Ensure VMCS is not MMIO, ROM etc. This is not an Intel requirement but a 1105 restriction imposed by our implementation. */ 1106 if (!PGMPhysIsGCPhysNormal(pVCpu->CTX_SUFF(pVM), GCPhysVmcs)) 1107 { 1108 Log(("vmclear: VMCS not normal memory -> VMFail()\n")); 1109 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmclear_PtrAbnormal; 1110 iemVmxVmFail(pVCpu, VMXINSTRERR_VMCLEAR_INVALID_PHYSADDR); 1111 iemRegAddToRipAndClearRF(pVCpu, cbInstr); 1112 return VINF_SUCCESS; 1113 } 1114 1115 /* 1116 * VMCLEAR allows committing and clearing any valid VMCS pointer. 1117 * 1118 * If the current VMCS is the one being cleared, set its state to 'clear' and commit 1119 * to guest memory. Otherwise, set the state of the VMCS referenced in guest memory 1120 * to 'clear'. 1121 */ 1122 uint8_t const fVmcsStateClear = VMX_V_VMCS_STATE_CLEAR; 1123 if (IEM_VMX_GET_CURRENT_VMCS(pVCpu) == GCPhysVmcs) 1124 { 1125 Assert(GCPhysVmcs != NIL_RTGCPHYS); /* Paranoia. */ 1126 pVCpu->cpum.GstCtx.hwvirt.vmx.CTX_SUFF(pVmcs)->fVmcsState = fVmcsStateClear; 1127 iemVmxCommitCurrentVmcsToMemory(pVCpu); 1128 Assert(!IEM_VMX_HAS_CURRENT_VMCS(pVCpu)); 1129 } 1130 else 1131 { 1132 rcStrict = PGMPhysSimpleWriteGCPhys(pVCpu->CTX_SUFF(pVM), GCPtrVmcs + RT_OFFSETOF(VMXVVMCS, fVmcsState), 1133 (const void *)&fVmcsStateClear, sizeof(fVmcsStateClear)); 1134 } 1135 1136 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmclear_Success; 1137 iemVmxVmSucceed(pVCpu); 1138 iemRegAddToRipAndClearRF(pVCpu, cbInstr); 1139 return rcStrict; 1029 1140 } 1030 1141 … … 1062 1173 1063 1174 /* Set the VMCS pointer to the location specified by the destination memory operand. */ 1064 Assert (NIL_RTGCPHYS == ~(RTGCPHYS)0U);1175 AssertCompile(NIL_RTGCPHYS == ~(RTGCPHYS)0U); 1065 1176 VBOXSTRICTRC rcStrict = iemMemStoreDataU64(pVCpu, pExitInstrInfo->VmxXsave.iSegReg, GCPtrVmcs, 1066 pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs);1177 IEM_VMX_GET_CURRENT_VMCS(pVCpu)); 1067 1178 if (RT_LIKELY(rcStrict == VINF_SUCCESS)) 1068 1179 { 1180 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmptrst_Success; 1069 1181 iemVmxVmSucceed(pVCpu); 1070 1182 iemRegAddToRipAndClearRF(pVCpu, cbInstr); … … 1072 1184 } 1073 1185 1074 Log(("vmptr ld: Failed to store VMCS pointer to memory at destination operand %#Rrc\n", VBOXSTRICTRC_VAL(rcStrict)));1186 Log(("vmptrst: Failed to store VMCS pointer to memory at destination operand %#Rrc\n", VBOXSTRICTRC_VAL(rcStrict))); 1075 1187 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmptrst_PtrMap; 1076 1188 return rcStrict; … … 1193 1305 } 1194 1306 1195 pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs = GCPhysVmcs; 1307 /* 1308 * We only maintain only the current VMCS in our virtual CPU context (CPUMCTX). Therefore, 1309 * VMPTRLD shall always flush any existing current VMCS back to guest memory before loading 1310 * a new VMCS as current. 1311 */ 1312 if (IEM_VMX_GET_CURRENT_VMCS(pVCpu) != GCPhysVmcs) 1313 { 1314 iemVmxCommitCurrentVmcsToMemory(pVCpu); 1315 IEM_VMX_SET_CURRENT_VMCS(pVCpu, GCPhysVmcs); 1316 } 1196 1317 pVCpu->cpum.GstCtx.hwvirt.vmx.enmInstrDiag = kVmxVInstrDiag_Vmptrld_Success; 1197 1318 iemVmxVmSucceed(pVCpu); … … 1343 1464 */ 1344 1465 pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmxon = GCPhysVmxon; 1345 pVCpu->cpum.GstCtx.hwvirt.vmx.GCPhysVmcs = NIL_RTGCPHYS;1466 IEM_VMX_CLEAR_CURRENT_VMCS(pVCpu); 1346 1467 pVCpu->cpum.GstCtx.hwvirt.vmx.fInVmxRootMode = true; 1347 1468 /** @todo NSTVMX: clear address-range monitoring. */
Note:
See TracChangeset
for help on using the changeset viewer.