- Timestamp:
- Apr 20, 2018 10:54:01 AM (7 years ago)
- Location:
- trunk
- Files:
-
- 5 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/include/VBox/vmm/hm_svm.h
r71929 r71933 1016 1016 * SVM nested-guest VMCB cache. 1017 1017 * 1018 * A state structure for holding information across AMD-V VMRUN/\#VMEXIT1019 * operation during execution of the nested-guest, restored on \#VMEXIT.1018 * Contains VMCB fields from the nested-guest VMCB before they're modified by 1019 * SVM R0 code for hardware-assisted SVM execution of a nested-guest. 1020 1020 * 1021 * @remarks Please update hmR3Info NstGuestVmcbCache() when changes are made to1022 * 1021 * @remarks Please update hmR3InfoSvmNstGstVmcbCache() when changes are made to 1022 * this structure. 1023 1023 */ 1024 1024 #pragma pack(1) 1025 1025 typedef struct SVMNESTEDVMCBCACHE 1026 1026 { 1027 /** @name Nested-guest VMCB controls.1028 * @{ */1029 1027 /** Cache of CRX read intercepts. */ 1030 1028 uint16_t u16InterceptRdCRx; … … 1035 1033 /** Cache of DRX write intercepts. */ 1036 1034 uint16_t u16InterceptWrDRx; 1035 1037 1036 /** Cache of the pause-filter threshold. */ 1038 1037 uint16_t u16PauseFilterThreshold; 1039 1038 /** Cache of the pause-filter count. */ 1040 1039 uint16_t u16PauseFilterCount; 1040 1041 1041 /** Cache of exception intercepts. */ 1042 1042 uint32_t u32InterceptXcpt; 1043 1044 1043 /** Cache of control intercepts. */ 1045 1044 uint64_t u64InterceptCtrl; 1046 /** Cache of IOPM nested-guest physical address. */ 1047 uint64_t u64IOPMPhysAddr; 1048 /** Cache of MSRPM nested-guest physical address. */ 1049 uint64_t u64MSRPMPhysAddr; 1045 1050 1046 /** Cache of the TSC offset. */ 1051 1047 uint64_t u64TSCOffset; 1052 /** Cache of the VMCB clean bits. */ 1053 uint32_t u32VmcbCleanBits; 1054 uint32_t u32Reserved0; 1055 /** Cache of the TLB control. */ 1056 SVMTLBCTRL TLBCtrl; 1057 /** Cache of the nested-paging control. */ 1058 uint32_t u1NestedPaging : 1; 1059 /** Cache of the LBR virtualization control. */ 1060 uint32_t u1LbrVirt : 1; 1061 uint32_t u30Reserved : 30; 1062 uint32_t u32Reserved1; 1063 /** @} */ 1064 1065 /** @name Nested-guest VMCB guest state. 1066 * @{ */ 1067 /** Cache of CR0. */ 1068 uint64_t u64CR0; 1069 /** Cache of CR3. */ 1070 uint64_t u64CR3; 1071 /** Cache of CR4. */ 1072 uint64_t u64CR4; 1073 /** Cache of EFER. */ 1074 uint64_t u64EFER; 1075 /** Cache of DBGCTL. */ 1076 uint64_t u64DBGCTL; 1077 /** Cache of the PAT MSR. */ 1078 uint64_t u64PAT; 1079 /** @} */ 1080 1081 /** @name Other miscellaneous state. 1082 * @{ */ 1048 1083 1049 /** Cache of V_INTR_MASKING bit. */ 1084 1050 bool fVIntrMasking; 1051 /** Cache of the nested-paging bit. */ 1052 bool fNestedPaging; 1053 /** Cache of the LBR virtualization bit. */ 1054 bool fLbrVirt; 1085 1055 /** Alignment. */ 1086 bool afPadding0[7]; 1087 /** @} */ 1056 bool afPadding0[5]; 1088 1057 } SVMNESTEDVMCBCACHE; 1089 1058 #pragma pack() -
trunk/src/VBox/VMM/VMMAll/HMSVMAll.cpp
r71927 r71933 132 132 * in IEM). 133 133 * 134 * @param pVCpu The cross context virtual CPU structure.135 * @param pCtx Pointer to the guest-CPU context.134 * @param pVCpu The cross context virtual CPU structure. 135 * @param pCtx Pointer to the guest-CPU context. 136 136 * 137 137 * @sa hmR0SvmVmRunCacheVmcb. … … 139 139 VMM_INT_DECL(void) HMSvmNstGstVmExitNotify(PVMCPU pVCpu, PCPUMCTX pCtx) 140 140 { 141 /*142 * Restore the nested-guest VMCB fields which have been modified for executing143 * the nested-guest under SVM R0.144 */145 141 if (pCtx->hwvirt.svm.fHMCachedVmcb) 146 142 { 147 PSVMVMCB pVmcbNstGst = pCtx->hwvirt.svm.CTX_SUFF(pVmcb); 148 PSVMVMCBCTRL pVmcbNstGstCtrl = &pVmcbNstGst->ctrl; 149 PSVMVMCBSTATESAVE pVmcbNstGstState = &pVmcbNstGst->guest; 150 PSVMNESTEDVMCBCACHE pNstGstVmcbCache = &pVCpu->hm.s.svm.NstGstVmcbCache; 143 PSVMVMCBCTRL pVmcbNstGstCtrl = &pCtx->hwvirt.svm.CTX_SUFF(pVmcb)->ctrl; 144 PSVMNESTEDVMCBCACHE pVmcbNstGstCache = &pVCpu->hm.s.svm.NstGstVmcbCache; 151 145 152 146 /* 153 * The fields that are guaranteed to be read-only during SVM guest execution 154 * can safely be restored from our VMCB cache. Other fields like control registers 155 * are already updated by hardware-assisted SVM or by IEM. We only restore those 156 * fields that are potentially modified by hardware-assisted SVM. 147 * Restore fields as our own code might look at the VMCB controls as part 148 * of the #VMEXIT handling. Otherwise, we don't need to restore the current 149 * fields because none of them are written by a physical CPU on #VMEXIT. 157 150 */ 158 pVmcbNstGstCtrl->u16InterceptRdCRx = pNstGstVmcbCache->u16InterceptRdCRx; 159 pVmcbNstGstCtrl->u16InterceptWrCRx = pNstGstVmcbCache->u16InterceptWrCRx; 160 pVmcbNstGstCtrl->u16InterceptRdDRx = pNstGstVmcbCache->u16InterceptRdDRx; 161 pVmcbNstGstCtrl->u16InterceptWrDRx = pNstGstVmcbCache->u16InterceptWrDRx; 162 pVmcbNstGstCtrl->u16PauseFilterCount = pNstGstVmcbCache->u16PauseFilterCount; 163 pVmcbNstGstCtrl->u16PauseFilterThreshold = pNstGstVmcbCache->u16PauseFilterThreshold; 164 pVmcbNstGstCtrl->u32InterceptXcpt = pNstGstVmcbCache->u32InterceptXcpt; 165 pVmcbNstGstCtrl->u64InterceptCtrl = pNstGstVmcbCache->u64InterceptCtrl; 166 pVmcbNstGstState->u64DBGCTL = pNstGstVmcbCache->u64DBGCTL; 167 pVmcbNstGstCtrl->u32VmcbCleanBits = pNstGstVmcbCache->u32VmcbCleanBits; 168 pVmcbNstGstCtrl->u64IOPMPhysAddr = pNstGstVmcbCache->u64IOPMPhysAddr; 169 pVmcbNstGstCtrl->u64MSRPMPhysAddr = pNstGstVmcbCache->u64MSRPMPhysAddr; 170 pVmcbNstGstCtrl->u64TSCOffset = pNstGstVmcbCache->u64TSCOffset; 171 pVmcbNstGstCtrl->IntCtrl.n.u1VIntrMasking = pNstGstVmcbCache->fVIntrMasking; 172 pVmcbNstGstCtrl->TLBCtrl = pNstGstVmcbCache->TLBCtrl; 173 174 /* 175 * If the nested-hypervisor isn't using nested-paging (and thus shadow paging 176 * is used by HM), we restore the original PAT MSR from the nested-guest VMCB. 177 * Otherwise, the nested-guest-CPU PAT MSR would've already been saved here by 178 * hardware-assisted SVM or by IEM. 179 */ 180 if (!pNstGstVmcbCache->u1NestedPaging) 181 pVmcbNstGstState->u64PAT = pNstGstVmcbCache->u64PAT; 182 183 pVmcbNstGstCtrl->NestedPagingCtrl.n.u1NestedPaging = pNstGstVmcbCache->u1NestedPaging; 184 pVmcbNstGstCtrl->LbrVirt.n.u1LbrVirt = pNstGstVmcbCache->u1LbrVirt; 151 pVmcbNstGstCtrl->u16InterceptRdCRx = pVmcbNstGstCache->u16InterceptRdCRx; 152 pVmcbNstGstCtrl->u16InterceptWrCRx = pVmcbNstGstCache->u16InterceptWrCRx; 153 pVmcbNstGstCtrl->u16InterceptRdDRx = pVmcbNstGstCache->u16InterceptRdDRx; 154 pVmcbNstGstCtrl->u16InterceptWrDRx = pVmcbNstGstCache->u16InterceptWrDRx; 155 pVmcbNstGstCtrl->u16PauseFilterThreshold = pVmcbNstGstCache->u16PauseFilterThreshold; 156 pVmcbNstGstCtrl->u16PauseFilterCount = pVmcbNstGstCache->u16PauseFilterCount; 157 pVmcbNstGstCtrl->u32InterceptXcpt = pVmcbNstGstCache->u32InterceptXcpt; 158 pVmcbNstGstCtrl->u64InterceptCtrl = pVmcbNstGstCache->u64InterceptCtrl; 159 pVmcbNstGstCtrl->u64TSCOffset = pVmcbNstGstCache->u64TSCOffset; 160 pVmcbNstGstCtrl->IntCtrl.n.u1VIntrMasking = pVmcbNstGstCache->fVIntrMasking; 161 pVmcbNstGstCtrl->NestedPagingCtrl.n.u1NestedPaging = pVmcbNstGstCache->fNestedPaging; 162 pVmcbNstGstCtrl->LbrVirt.n.u1LbrVirt = pVmcbNstGstCache->fLbrVirt; 185 163 pCtx->hwvirt.svm.fHMCachedVmcb = false; 186 164 } … … 580 558 Assert(pCtx->hwvirt.svm.fHMCachedVmcb); NOREF(pCtx); 581 559 PCSVMNESTEDVMCBCACHE pVmcbNstGstCache = &pVCpu->hm.s.svm.NstGstVmcbCache; 582 return RT_BOOL(pVmcbNstGstCache->u1NestedPaging);560 return pVmcbNstGstCache->fNestedPaging; 583 561 } 584 562 -
trunk/src/VBox/VMM/VMMAll/IEMAllCImplSvmInstr.cpp.h
r71867 r71933 124 124 125 125 /* 126 * Save the nested-guest state into the VMCB state-save area. 127 */ 128 PSVMVMCB pVmcbNstGst = pCtx->hwvirt.svm.CTX_SUFF(pVmcb); 129 PSVMVMCBCTRL pVmcbNstGstCtrl = &pVmcbNstGst->ctrl; 130 PSVMVMCBSTATESAVE pVmcbNstGstState = &pVmcbNstGst->guest; 131 132 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbNstGstState, ES, es); 133 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbNstGstState, CS, cs); 134 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbNstGstState, SS, ss); 135 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbNstGstState, DS, ds); 136 pVmcbNstGstState->GDTR.u32Limit = pCtx->gdtr.cbGdt; 137 pVmcbNstGstState->GDTR.u64Base = pCtx->gdtr.pGdt; 138 pVmcbNstGstState->IDTR.u32Limit = pCtx->idtr.cbIdt; 139 pVmcbNstGstState->IDTR.u64Base = pCtx->idtr.pIdt; 140 pVmcbNstGstState->u64EFER = pCtx->msrEFER; 141 pVmcbNstGstState->u64CR4 = pCtx->cr4; 142 pVmcbNstGstState->u64CR3 = pCtx->cr3; 143 pVmcbNstGstState->u64CR2 = pCtx->cr2; 144 pVmcbNstGstState->u64CR0 = pCtx->cr0; 145 /** @todo Nested paging. */ 146 pVmcbNstGstState->u64RFlags = pCtx->rflags.u64; 147 pVmcbNstGstState->u64RIP = pCtx->rip; 148 pVmcbNstGstState->u64RSP = pCtx->rsp; 149 pVmcbNstGstState->u64RAX = pCtx->rax; 150 pVmcbNstGstState->u64DR7 = pCtx->dr[7]; 151 pVmcbNstGstState->u64DR6 = pCtx->dr[6]; 152 pVmcbNstGstState->u8CPL = pCtx->ss.Attr.n.u2Dpl; /* See comment in CPUMGetGuestCPL(). */ 153 Assert(CPUMGetGuestCPL(pVCpu) == pCtx->ss.Attr.n.u2Dpl); 154 if (CPUMIsGuestSvmNestedPagingEnabled(pVCpu, pCtx)) 155 pVmcbNstGstState->u64PAT = pCtx->msrPAT; 156 157 PSVMVMCBCTRL pVmcbCtrl = &pCtx->hwvirt.svm.CTX_SUFF(pVmcb)->ctrl; 158 159 /* 160 * Save additional state and intercept information. 126 * Map the nested-guest VMCB from its location in guest memory. 127 * Write exactly what the CPU does on #VMEXIT thereby preserving most other bits in the 128 * guest's VMCB in memory, see @bugref{7243#c113} and related comment on iemSvmVmrun(). 161 129 * 162 * - Interrupt shadow: Tracked using VMCPU_FF_INHIBIT_INTERRUPTS and RIP. 163 * - V_TPR: Already updated by iemCImpl_load_CrX or by the physical CPU for 164 * hardware-assisted SVM execution. 165 * - V_IRQ: Tracked using VMCPU_FF_INTERRUPT_NESTED_GUEST force-flag and updated below. 166 */ 167 if ( VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS) 168 && EMGetInhibitInterruptsPC(pVCpu) == pCtx->rip) 169 { 170 pVmcbCtrl->IntShadow.n.u1IntShadow = 1; 171 172 /* Clear the inhibit-interrupt force-flag so as to not affect the outer guest. */ 173 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS); 174 LogFlow(("iemSvmVmexit: Interrupt shadow till %#RX64\n", pCtx->rip)); 175 } 176 177 if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_INTERRUPT_NESTED_GUEST)) 178 { 179 Assert(pVmcbCtrl->IntCtrl.n.u1VIrqPending); 180 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_NESTED_GUEST); 181 } 182 else 183 pVmcbCtrl->IntCtrl.n.u1VIrqPending = 0; 184 185 /* Save exit information. */ 186 pVmcbCtrl->u64ExitCode = uExitCode; 187 pVmcbCtrl->u64ExitInfo1 = uExitInfo1; 188 pVmcbCtrl->u64ExitInfo2 = uExitInfo2; 189 190 /* 191 * Update the exit interrupt-information field if this #VMEXIT happened as a result 192 * of delivering an event through IEM. 193 * 194 * Don't update the exit interrupt-information field if the event wasn't being injected 195 * through IEM, as it may have been updated by real hardware if the nested-guest was 196 * executed using hardware-assisted SVM. 197 */ 198 { 199 uint8_t uExitIntVector; 200 uint32_t uExitIntErr; 201 uint32_t fExitIntFlags; 202 bool const fRaisingEvent = IEMGetCurrentXcpt(pVCpu, &uExitIntVector, &fExitIntFlags, &uExitIntErr, 203 NULL /* uExitIntCr2 */); 204 if (fRaisingEvent) 130 */ 131 PSVMVMCB pVmcbMem; 132 PGMPAGEMAPLOCK PgLockMem; 133 PSVMVMCBCTRL pVmcbCtrl = &pCtx->hwvirt.svm.CTX_SUFF(pVmcb)->ctrl; 134 rcStrict = iemMemPageMap(pVCpu, pCtx->hwvirt.svm.GCPhysVmcb, IEM_ACCESS_DATA_RW, (void **)&pVmcbMem, &PgLockMem); 135 if (rcStrict == VINF_SUCCESS) 136 { 137 /* 138 * Notify HM in case the nested-guest was executed using hardware-assisted SVM (which 139 * would have modified some VMCB state) that might need to be restored on #VMEXIT before 140 * writing the VMCB back to guest memory. 141 */ 142 HMSvmNstGstVmExitNotify(pVCpu, pCtx); 143 144 /* 145 * Save the nested-guest state into the VMCB state-save area. 146 */ 147 PSVMVMCBSTATESAVE pVmcbMemState = &pVmcbMem->guest; 148 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbMemState, ES, es); 149 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbMemState, CS, cs); 150 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbMemState, SS, ss); 151 HMSVM_SEG_REG_COPY_TO_VMCB(pCtx, pVmcbMemState, DS, ds); 152 pVmcbMemState->GDTR.u32Limit = pCtx->gdtr.cbGdt; 153 pVmcbMemState->GDTR.u64Base = pCtx->gdtr.pGdt; 154 pVmcbMemState->IDTR.u32Limit = pCtx->idtr.cbIdt; 155 pVmcbMemState->IDTR.u64Base = pCtx->idtr.pIdt; 156 pVmcbMemState->u64EFER = pCtx->msrEFER; 157 pVmcbMemState->u64CR4 = pCtx->cr4; 158 pVmcbMemState->u64CR3 = pCtx->cr3; 159 pVmcbMemState->u64CR2 = pCtx->cr2; 160 pVmcbMemState->u64CR0 = pCtx->cr0; 161 /** @todo Nested paging. */ 162 pVmcbMemState->u64RFlags = pCtx->rflags.u64; 163 pVmcbMemState->u64RIP = pCtx->rip; 164 pVmcbMemState->u64RSP = pCtx->rsp; 165 pVmcbMemState->u64RAX = pCtx->rax; 166 pVmcbMemState->u64DR7 = pCtx->dr[7]; 167 pVmcbMemState->u64DR6 = pCtx->dr[6]; 168 pVmcbMemState->u8CPL = pCtx->ss.Attr.n.u2Dpl; /* See comment in CPUMGetGuestCPL(). */ 169 Assert(CPUMGetGuestCPL(pVCpu) == pCtx->ss.Attr.n.u2Dpl); 170 if (CPUMIsGuestSvmNestedPagingEnabled(pVCpu, pCtx)) 171 pVmcbMemState->u64PAT = pCtx->msrPAT; 172 173 /* 174 * Save additional state and intercept information. 175 * 176 * - V_IRQ: Tracked using VMCPU_FF_INTERRUPT_NESTED_GUEST force-flag and updated below. 177 * - V_TPR: Already updated by iemCImpl_load_CrX or by the physical CPU for 178 * hardware-assisted SVM execution. 179 * - Interrupt shadow: Tracked using VMCPU_FF_INHIBIT_INTERRUPTS and RIP. 180 */ 181 PSVMVMCBCTRL pVmcbMemCtrl = &pVmcbMem->ctrl; 182 if (VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_INTERRUPT_NESTED_GUEST)) /* V_IRQ. */ 205 183 { 206 pVmcbCtrl->ExitIntInfo.n.u1Valid = 1; 207 pVmcbCtrl->ExitIntInfo.n.u8Vector = uExitIntVector; 208 pVmcbCtrl->ExitIntInfo.n.u3Type = iemGetSvmEventType(uExitIntVector, fExitIntFlags); 209 if (fExitIntFlags & IEM_XCPT_FLAGS_ERR) 184 Assert(pVmcbCtrl->IntCtrl.n.u1VIrqPending); 185 pVmcbMemCtrl->IntCtrl.n.u1VIrqPending = 1; 186 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_NESTED_GUEST); 187 } 188 else 189 pVmcbMemCtrl->IntCtrl.n.u1VIrqPending = 0; 190 191 pVmcbMemCtrl->IntCtrl.n.u8VTPR = pVmcbCtrl->IntCtrl.n.u8VTPR; /* V_TPR. */ 192 193 if ( VMCPU_FF_IS_PENDING(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS) /* Interrupt shadow. */ 194 && EMGetInhibitInterruptsPC(pVCpu) == pCtx->rip) 195 { 196 pVmcbMemCtrl->IntShadow.n.u1IntShadow = 1; 197 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS); 198 LogFlow(("iemSvmVmexit: Interrupt shadow till %#RX64\n", pCtx->rip)); 199 } 200 else 201 pVmcbMemCtrl->IntShadow.n.u1IntShadow = 0; 202 203 /* 204 * Save nRIP, instruction length and byte fields. 205 */ 206 pVmcbMemCtrl->u64NextRIP = pVmcbCtrl->u64NextRIP; 207 pVmcbMemCtrl->cbInstrFetched = pVmcbCtrl->cbInstrFetched; 208 memcpy(&pVmcbMemCtrl->abInstr[0], &pVmcbCtrl->abInstr[0], sizeof(pVmcbMemCtrl->abInstr)); 209 210 /* 211 * Save exit information. 212 */ 213 pVmcbMemCtrl->u64ExitCode = uExitCode; 214 pVmcbMemCtrl->u64ExitInfo1 = uExitInfo1; 215 pVmcbMemCtrl->u64ExitInfo2 = uExitInfo2; 216 217 /* 218 * Update the exit interrupt-information field if this #VMEXIT happened as a result 219 * of delivering an event through IEM. 220 * 221 * Don't update the exit interrupt-information field if the event wasn't being injected 222 * through IEM, as it would have been updated by real hardware if the nested-guest was 223 * executed using hardware-assisted SVM. 224 */ 225 { 226 uint8_t uExitIntVector; 227 uint32_t uExitIntErr; 228 uint32_t fExitIntFlags; 229 bool const fRaisingEvent = IEMGetCurrentXcpt(pVCpu, &uExitIntVector, &fExitIntFlags, &uExitIntErr, 230 NULL /* uExitIntCr2 */); 231 if (fRaisingEvent) 210 232 { 211 pVmcbCtrl->ExitIntInfo.n.u1ErrorCodeValid = true; 212 pVmcbCtrl->ExitIntInfo.n.u32ErrorCode = uExitIntErr; 233 pVmcbCtrl->ExitIntInfo.n.u1Valid = 1; 234 pVmcbCtrl->ExitIntInfo.n.u8Vector = uExitIntVector; 235 pVmcbCtrl->ExitIntInfo.n.u3Type = iemGetSvmEventType(uExitIntVector, fExitIntFlags); 236 if (fExitIntFlags & IEM_XCPT_FLAGS_ERR) 237 { 238 pVmcbCtrl->ExitIntInfo.n.u1ErrorCodeValid = true; 239 pVmcbCtrl->ExitIntInfo.n.u32ErrorCode = uExitIntErr; 240 } 213 241 } 214 242 } 215 } 216 217 /* 218 * Clear event injection in the VMCB. 219 */ 220 pVmcbCtrl->EventInject.n.u1Valid = 0; 221 222 /* 223 * Notify HM in case the nested-guest was executed using hardware-assisted SVM (which 224 * would have modified some VMCB state) that need to be restored on #VMEXIT before 225 * writing the VMCB back to guest memory. 226 */ 227 HMSvmNstGstVmExitNotify(pVCpu, pCtx); 228 229 /* 230 * Write back the nested-guest's VMCB to its guest physical memory location. 231 */ 232 rcStrict = PGMPhysSimpleWriteGCPhys(pVCpu->CTX_SUFF(pVM), pCtx->hwvirt.svm.GCPhysVmcb, pVmcbNstGst, sizeof(*pVmcbNstGst)); 243 244 /* 245 * Save the exit interrupt-information field. 246 * We choose to write the whole field including reserved bits as it was observed on an 247 * AMD Ryzen 5 Pro 1500 that the CPU does not preserve reserved bits in EXITINTINFO. 248 */ 249 pVmcbMemCtrl->ExitIntInfo = pVmcbCtrl->ExitIntInfo; 250 251 /* 252 * Clear event injection. 253 */ 254 pVmcbMemCtrl->EventInject.n.u1Valid = 0; 255 256 iemMemPageUnmap(pVCpu, pCtx->hwvirt.svm.GCPhysVmcb, IEM_ACCESS_DATA_RW, pVmcbMem, &PgLockMem); 257 } 233 258 234 259 /* … … 239 264 * the nested-guest or not. 240 265 */ 241 memset(pVmcb NstGstCtrl, 0, sizeof(*pVmcbNstGstCtrl));266 memset(pVmcbCtrl, 0, sizeof(*pVmcbCtrl)); 242 267 Assert(!CPUMIsGuestInSvmNestedHwVirtMode(pCtx)); 243 268 … … 251 276 } 252 277 253 if ( RT_SUCCESS(rcStrict))278 if (rcStrict == VINF_SUCCESS) 254 279 { 255 280 /** @todo Nested paging. */ … … 278 303 else 279 304 { 280 LogFlow(("iemSvmVmexit: Writing VMCB at %#RGp failed. rc=%Rrc\n", pCtx->hwvirt.svm.GCPhysVmcb, 281 VBOXSTRICTRC_VAL(rcStrict))); 305 LogFlow(("iemSvmVmexit: Mapping VMCB at %#RGp failed. rc=%Rrc\n", pCtx->hwvirt.svm.GCPhysVmcb, VBOXSTRICTRC_VAL(rcStrict))); 282 306 rcStrict = VERR_SVM_VMEXIT_FAILED; 283 307 } … … 340 364 341 365 /* 342 * Read the guest VMCB state.366 * Read the guest VMCB. 343 367 */ 344 368 PVM pVM = pVCpu->CTX_SUFF(pVM); … … 346 370 if (RT_SUCCESS(rc)) 347 371 { 372 /* 373 * AMD-V seems to preserve reserved fields and only writes back selected, recognized 374 * fields on #VMEXIT. However, not all reserved bits are preserved (e.g, EXITINTINFO) 375 * but in our implementation we try to preserve as much as we possibly can. 376 * 377 * We could read the entire page here and only write back the relevant fields on 378 * #VMEXIT but since our internal VMCB is also being used by HM during hardware-assisted 379 * SVM execution, it creates a potential for a nested-hypervisor to set bits that are 380 * currently reserved but may be recognized as features bits in future CPUs causing 381 * unexpected & undesired results. Hence, we zero out unrecognized fields here as we 382 * typically enter hardware-assisted SVM soon anyway, see @bugref{7243#c113}. 383 */ 348 384 PSVMVMCBCTRL pVmcbCtrl = &pCtx->hwvirt.svm.CTX_SUFF(pVmcb)->ctrl; 349 385 PSVMVMCBSTATESAVE pVmcbNstGst = &pCtx->hwvirt.svm.CTX_SUFF(pVmcb)->guest; 386 387 RT_ZERO(pVmcbCtrl->u8Reserved0); 388 RT_ZERO(pVmcbCtrl->u8Reserved1); 389 RT_ZERO(pVmcbCtrl->u8Reserved2); 390 RT_ZERO(pVmcbNstGst->u8Reserved0); 391 RT_ZERO(pVmcbNstGst->u8Reserved1); 392 RT_ZERO(pVmcbNstGst->u8Reserved2); 393 RT_ZERO(pVmcbNstGst->u8Reserved3); 394 RT_ZERO(pVmcbNstGst->u8Reserved4); 395 RT_ZERO(pVmcbNstGst->u8Reserved5); 396 pVmcbCtrl->u32Reserved0 = 0; 397 pVmcbCtrl->TLBCtrl.n.u24Reserved = 0; 398 pVmcbCtrl->IntCtrl.n.u6Reserved = 0; 399 pVmcbCtrl->IntCtrl.n.u3Reserved = 0; 400 pVmcbCtrl->IntCtrl.n.u5Reserved = 0; 401 pVmcbCtrl->IntCtrl.n.u24Reserved = 0; 402 pVmcbCtrl->IntShadow.n.u30Reserved = 0; 403 pVmcbCtrl->ExitIntInfo.n.u19Reserved = 0; 404 pVmcbCtrl->NestedPagingCtrl.n.u29Reserved = 0; 405 pVmcbCtrl->EventInject.n.u19Reserved = 0; 406 pVmcbCtrl->LbrVirt.n.u30Reserved = 0; 350 407 351 408 /* … … 363 420 && !pVM->cpum.ro.GuestFeatures.fSvmNestedPaging) 364 421 { 365 Log(("iemSvmVmrun: Nested paging not supported -> #VMEXIT\n"));366 return iemSvmVmexit(pVCpu, pCtx, SVM_EXIT_INVALID, 0 /* uExitInfo1 */, 0 /* uExitInfo2 */);422 Log(("iemSvmVmrun: Nested paging not supported -> Disabling\n")); 423 pVmcbCtrl->NestedPagingCtrl.n.u1NestedPaging = 0; 367 424 } 368 425 -
trunk/src/VBox/VMM/VMMR0/HMSVMR0.cpp
r71927 r71933 2484 2484 static bool hmR0SvmVmRunCacheVmcb(PVMCPU pVCpu, PCPUMCTX pCtx) 2485 2485 { 2486 PSVMVMCB pVmcbNstGst = pCtx->hwvirt.svm.CTX_SUFF(pVmcb);2487 PCSVMVMCBCTRL pVmcbNstGstCtrl = &pVmcbNstGst->ctrl;2488 PCSVMVMCBSTATESAVE pVmcbNstGstState = &pVmcbNstGst->guest;2489 PSVMNESTEDVMCBCACHE pVmcbNstGstCache = &pVCpu->hm.s.svm.NstGstVmcbCache;2490 2491 2486 /* 2492 2487 * Cache the nested-guest programmed VMCB fields if we have not cached it yet. … … 2499 2494 if (!fWasCached) 2500 2495 { 2496 PSVMVMCB pVmcbNstGst = pCtx->hwvirt.svm.CTX_SUFF(pVmcb); 2497 PCSVMVMCBCTRL pVmcbNstGstCtrl = &pVmcbNstGst->ctrl; 2498 PSVMNESTEDVMCBCACHE pVmcbNstGstCache = &pVCpu->hm.s.svm.NstGstVmcbCache; 2499 2501 2500 pVmcbNstGstCache->u16InterceptRdCRx = pVmcbNstGstCtrl->u16InterceptRdCRx; 2502 2501 pVmcbNstGstCache->u16InterceptWrCRx = pVmcbNstGstCtrl->u16InterceptWrCRx; 2503 2502 pVmcbNstGstCache->u16InterceptRdDRx = pVmcbNstGstCtrl->u16InterceptRdDRx; 2504 2503 pVmcbNstGstCache->u16InterceptWrDRx = pVmcbNstGstCtrl->u16InterceptWrDRx; 2504 pVmcbNstGstCache->u16PauseFilterThreshold = pVmcbNstGstCtrl->u16PauseFilterThreshold; 2505 2505 pVmcbNstGstCache->u16PauseFilterCount = pVmcbNstGstCtrl->u16PauseFilterCount; 2506 pVmcbNstGstCache->u16PauseFilterThreshold = pVmcbNstGstCtrl->u16PauseFilterThreshold;2507 2506 pVmcbNstGstCache->u32InterceptXcpt = pVmcbNstGstCtrl->u32InterceptXcpt; 2508 2507 pVmcbNstGstCache->u64InterceptCtrl = pVmcbNstGstCtrl->u64InterceptCtrl; 2509 pVmcbNstGstCache->u64CR0 = pVmcbNstGstState->u64CR0;2510 pVmcbNstGstCache->u64CR3 = pVmcbNstGstState->u64CR3;2511 pVmcbNstGstCache->u64CR4 = pVmcbNstGstState->u64CR4;2512 pVmcbNstGstCache->u64EFER = pVmcbNstGstState->u64EFER;2513 pVmcbNstGstCache->u64PAT = pVmcbNstGstState->u64PAT;2514 pVmcbNstGstCache->u64DBGCTL = pVmcbNstGstState->u64DBGCTL;2515 pVmcbNstGstCache->u64IOPMPhysAddr = pVmcbNstGstCtrl->u64IOPMPhysAddr;2516 pVmcbNstGstCache->u64MSRPMPhysAddr = pVmcbNstGstCtrl->u64MSRPMPhysAddr;2517 2508 pVmcbNstGstCache->u64TSCOffset = pVmcbNstGstCtrl->u64TSCOffset; 2518 pVmcbNstGstCache->u32VmcbCleanBits = pVmcbNstGstCtrl->u32VmcbCleanBits;2519 2509 pVmcbNstGstCache->fVIntrMasking = pVmcbNstGstCtrl->IntCtrl.n.u1VIntrMasking; 2520 pVmcbNstGstCache->TLBCtrl = pVmcbNstGstCtrl->TLBCtrl; 2521 pVmcbNstGstCache->u1NestedPaging = pVmcbNstGstCtrl->NestedPagingCtrl.n.u1NestedPaging; 2522 pVmcbNstGstCache->u1LbrVirt = pVmcbNstGstCtrl->LbrVirt.n.u1LbrVirt; 2510 pVmcbNstGstCache->fNestedPaging = pVmcbNstGstCtrl->NestedPagingCtrl.n.u1NestedPaging; 2511 pVmcbNstGstCache->fLbrVirt = pVmcbNstGstCtrl->LbrVirt.n.u1LbrVirt; 2523 2512 pCtx->hwvirt.svm.fHMCachedVmcb = true; 2524 2513 Log4(("hmR0SvmVmRunCacheVmcb: Cached VMCB fields\n")); -
trunk/src/VBox/VMM/VMMR3/HM.cpp
r71856 r71933 3774 3774 PCSVMNESTEDVMCBCACHE pVmcbNstGstCache = &pVCpu->hm.s.svm.NstGstVmcbCache; 3775 3775 pHlp->pfnPrintf(pHlp, "CPU[%u]: HM SVM nested-guest VMCB cache\n", pVCpu->idCpu); 3776 pHlp->pfnPrintf(pHlp, " fHMCachedVmcb = %#RTbool\n", pCtx->hwvirt.svm.fHMCachedVmcb); 3777 pHlp->pfnPrintf(pHlp, " u16InterceptRdCRx = %#RX16\n", pVmcbNstGstCache->u16InterceptRdCRx); 3778 pHlp->pfnPrintf(pHlp, " u16InterceptWrCRx = %#RX16\n", pVmcbNstGstCache->u16InterceptWrCRx); 3779 pHlp->pfnPrintf(pHlp, " u16InterceptRdDRx = %#RX16\n", pVmcbNstGstCache->u16InterceptRdDRx); 3780 pHlp->pfnPrintf(pHlp, " u16InterceptWrDRx = %#RX16\n", pVmcbNstGstCache->u16InterceptWrDRx); 3781 pHlp->pfnPrintf(pHlp, " u32InterceptXcpt = %#RX32\n", pVmcbNstGstCache->u32InterceptXcpt); 3782 pHlp->pfnPrintf(pHlp, " u64InterceptCtrl = %#RX64\n", pVmcbNstGstCache->u64InterceptCtrl); 3783 pHlp->pfnPrintf(pHlp, " u64IOPMPhysAddr = %#RX64\n", pVmcbNstGstCache->u64IOPMPhysAddr); 3784 pHlp->pfnPrintf(pHlp, " u64MSRPMPhysAddr = %#RX64\n", pVmcbNstGstCache->u64MSRPMPhysAddr); 3785 pHlp->pfnPrintf(pHlp, " u64TSCOffset = %#RX64\n", pVmcbNstGstCache->u64TSCOffset); 3786 pHlp->pfnPrintf(pHlp, " u32VmcbCleanBits = %#RX32\n", pVmcbNstGstCache->u32VmcbCleanBits); 3787 pHlp->pfnPrintf(pHlp, " TLBCtrl = %#RX64\n", pVmcbNstGstCache->TLBCtrl); 3788 pHlp->pfnPrintf(pHlp, " u32ASID = %#RX64\n", pVmcbNstGstCache->TLBCtrl.n.u32ASID); 3789 pHlp->pfnPrintf(pHlp, " u8TLBFlush = %#RX64\n", pVmcbNstGstCache->TLBCtrl.n.u8TLBFlush); 3790 pHlp->pfnPrintf(pHlp, " u1NestedPaging = %RTbool\n", pVmcbNstGstCache->u1NestedPaging); 3791 pHlp->pfnPrintf(pHlp, " u1LbrVirt = %RTbool\n", pVmcbNstGstCache->u1LbrVirt); 3792 pHlp->pfnPrintf(pHlp, " u64CR0 = %#RX64\n", pVmcbNstGstCache->u64CR0); 3793 pHlp->pfnPrintf(pHlp, " u64CR3 = %#RX64\n", pVmcbNstGstCache->u64CR3); 3794 pHlp->pfnPrintf(pHlp, " u64CR4 = %#RX64\n", pVmcbNstGstCache->u64CR4); 3795 pHlp->pfnPrintf(pHlp, " u64EFER = %#RX64\n", pVmcbNstGstCache->u64EFER); 3796 pHlp->pfnPrintf(pHlp, " u64DBGCTL = %#RX64\n", pVmcbNstGstCache->u64DBGCTL); 3797 pHlp->pfnPrintf(pHlp, " fVIntrMasking = %RTbool\n", pVmcbNstGstCache->fVIntrMasking); 3776 pHlp->pfnPrintf(pHlp, " fHMCachedVmcb = %#RTbool\n", pCtx->hwvirt.svm.fHMCachedVmcb); 3777 pHlp->pfnPrintf(pHlp, " u16InterceptRdCRx = %#RX16\n", pVmcbNstGstCache->u16InterceptRdCRx); 3778 pHlp->pfnPrintf(pHlp, " u16InterceptWrCRx = %#RX16\n", pVmcbNstGstCache->u16InterceptWrCRx); 3779 pHlp->pfnPrintf(pHlp, " u16InterceptRdDRx = %#RX16\n", pVmcbNstGstCache->u16InterceptRdDRx); 3780 pHlp->pfnPrintf(pHlp, " u16InterceptWrDRx = %#RX16\n", pVmcbNstGstCache->u16InterceptWrDRx); 3781 pHlp->pfnPrintf(pHlp, " u16PauseFilterThreshold = %#RX16\n", pVmcbNstGstCache->u16PauseFilterThreshold); 3782 pHlp->pfnPrintf(pHlp, " u16PauseFilterCount = %#RX16\n", pVmcbNstGstCache->u16PauseFilterCount); 3783 pHlp->pfnPrintf(pHlp, " u32InterceptXcpt = %#RX32\n", pVmcbNstGstCache->u32InterceptXcpt); 3784 pHlp->pfnPrintf(pHlp, " u64InterceptCtrl = %#RX64\n", pVmcbNstGstCache->u64InterceptCtrl); 3785 pHlp->pfnPrintf(pHlp, " u64TSCOffset = %#RX64\n", pVmcbNstGstCache->u64TSCOffset); 3786 pHlp->pfnPrintf(pHlp, " fVIntrMasking = %RTbool\n", pVmcbNstGstCache->fVIntrMasking); 3787 pHlp->pfnPrintf(pHlp, " fNestedPaging = %RTbool\n", pVmcbNstGstCache->fNestedPaging); 3788 pHlp->pfnPrintf(pHlp, " fLbrVirt = %RTbool\n", pVmcbNstGstCache->fLbrVirt); 3798 3789 } 3799 3790 else
Note:
See TracChangeset
for help on using the changeset viewer.