VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/PDMAll.cpp@ 61568

Last change on this file since 61568 was 60804, checked in by vboxsync, 9 years ago

VMM: Introduced the new APIC update force-flag and relevant fixes, Win8.1 SMP boots now.
VMM/HMVMXR0: Fixed the HLT exit to use instruction length and also keep the INHIBIT force-flag
more in sync with the VT-x's guest-interruptibility state.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 16.4 KB
Line 
1/* $Id: PDMAll.cpp 60804 2016-05-03 14:13:51Z vboxsync $ */
2/** @file
3 * PDM Critical Sections
4 */
5
6/*
7 * Copyright (C) 2006-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_PDM
23#include "PDMInternal.h"
24#include <VBox/vmm/pdm.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/vm.h>
27#include <VBox/err.h>
28#ifdef VBOX_WITH_NEW_APIC
29# include <VBox/vmm/apic.h>
30#endif
31
32#include <VBox/log.h>
33#include <iprt/asm.h>
34#include <iprt/assert.h>
35
36#include "PDMInline.h"
37#include "dtrace/VBoxVMM.h"
38
39
40
41/**
42 * Gets the pending interrupt.
43 *
44 * @returns VBox status code.
45 * @retval VINF_SUCCESS on success.
46 * @retval VERR_APIC_INTR_MASKED_BY_TPR when an APIC interrupt is pending but
47 * can't be delivered due to TPR priority.
48 * @retval VERR_NO_DATA if there is no interrupt to be delivered (either APIC
49 * has been software-disabled since it flagged something was pending,
50 * or other reasons).
51 *
52 * @param pVCpu The cross context virtual CPU structure.
53 * @param pu8Interrupt Where to store the interrupt on success.
54 */
55VMMDECL(int) PDMGetInterrupt(PVMCPU pVCpu, uint8_t *pu8Interrupt)
56{
57 PVM pVM = pVCpu->CTX_SUFF(pVM);
58
59#ifndef VBOX_WITH_NEW_APIC
60 pdmLock(pVM);
61#endif
62
63 /*
64 * The local APIC has a higher priority than the PIC.
65 */
66 if (VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_APIC))
67 {
68 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_APIC);
69 Assert(pVM->pdm.s.Apic.CTX_SUFF(pDevIns));
70 Assert(pVM->pdm.s.Apic.CTX_SUFF(pfnGetInterrupt));
71 uint32_t uTagSrc;
72 uint8_t uVector;
73 int rc = pVM->pdm.s.Apic.CTX_SUFF(pfnGetInterrupt)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, &uVector, &uTagSrc);
74 if (RT_SUCCESS(rc))
75 {
76 *pu8Interrupt = uVector;
77 if (rc == VINF_SUCCESS)
78 VBOXVMM_PDM_IRQ_GET(pVCpu, RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc), uVector);
79#ifndef VBOX_WITH_NEW_APIC
80 pdmUnlock(pVM);
81#endif
82 return rc;
83 }
84 /* else if it's masked by TPR/PPR/whatever, go ahead checking the PIC. Such masked
85 interrupts shouldn't prevent ExtINT from being delivered. */
86 }
87
88#ifdef VBOX_WITH_NEW_APIC
89 pdmLock(pVM);
90#endif
91
92 /*
93 * Check the PIC.
94 */
95 if (VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC))
96 {
97 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_PIC);
98 Assert(pVM->pdm.s.Pic.CTX_SUFF(pDevIns));
99 Assert(pVM->pdm.s.Pic.CTX_SUFF(pfnGetInterrupt));
100 uint32_t uTagSrc;
101 int i = pVM->pdm.s.Pic.CTX_SUFF(pfnGetInterrupt)(pVM->pdm.s.Pic.CTX_SUFF(pDevIns), &uTagSrc);
102 AssertMsg(i <= 255 && i >= 0, ("i=%d\n", i));
103 if (i >= 0)
104 {
105 pdmUnlock(pVM);
106 *pu8Interrupt = (uint8_t)i;
107 VBOXVMM_PDM_IRQ_GET(pVCpu, RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc), i);
108 return VINF_SUCCESS;
109 }
110 }
111
112 /*
113 * One scenario where we may possibly get here is if the APIC signaled a pending interrupt,
114 * got an APIC MMIO/MSR VM-exit which disabled the APIC. We could, in theory, clear the APIC
115 * force-flag from all the places which disables the APIC but letting PDMGetInterrupt() fail
116 * without returning a valid interrupt still needs to be handled for the TPR masked case,
117 * so we shall just handle it here regardless if we choose to update the APIC code in the future.
118 */
119
120 pdmUnlock(pVM);
121 return VERR_NO_DATA;
122}
123
124
125/**
126 * Sets the pending interrupt coming from ISA source or HPET.
127 *
128 * @returns VBox status code.
129 * @param pVM The cross context VM structure.
130 * @param u8Irq The IRQ line.
131 * @param u8Level The new level.
132 * @param uTagSrc The IRQ tag and source tracer ID.
133 */
134VMMDECL(int) PDMIsaSetIrq(PVM pVM, uint8_t u8Irq, uint8_t u8Level, uint32_t uTagSrc)
135{
136 pdmLock(pVM);
137
138 /** @todo put the IRQ13 code elsewhere to avoid this unnecessary bloat. */
139 if (!uTagSrc && (u8Level & PDM_IRQ_LEVEL_HIGH)) /* FPU IRQ */
140 {
141 if (u8Level == PDM_IRQ_LEVEL_HIGH)
142 VBOXVMM_PDM_IRQ_HIGH(VMMGetCpu(pVM), 0, 0);
143 else
144 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), 0, 0);
145 }
146
147 int rc = VERR_PDM_NO_PIC_INSTANCE;
148 if (pVM->pdm.s.Pic.CTX_SUFF(pDevIns))
149 {
150 Assert(pVM->pdm.s.Pic.CTX_SUFF(pfnSetIrq));
151 pVM->pdm.s.Pic.CTX_SUFF(pfnSetIrq)(pVM->pdm.s.Pic.CTX_SUFF(pDevIns), u8Irq, u8Level, uTagSrc);
152 rc = VINF_SUCCESS;
153 }
154
155 if (pVM->pdm.s.IoApic.CTX_SUFF(pDevIns))
156 {
157 Assert(pVM->pdm.s.IoApic.CTX_SUFF(pfnSetIrq));
158
159 /*
160 * Apply Interrupt Source Override rules.
161 * See ACPI 4.0 specification 5.2.12.4 and 5.2.12.5 for details on
162 * interrupt source override.
163 * Shortly, ISA IRQ0 is electically connected to pin 2 on IO-APIC, and some OSes,
164 * notably recent OS X rely upon this configuration.
165 * If changing, also update override rules in MADT and MPS.
166 */
167 /* ISA IRQ0 routed to pin 2, all others ISA sources are identity mapped */
168 if (u8Irq == 0)
169 u8Irq = 2;
170
171 pVM->pdm.s.IoApic.CTX_SUFF(pfnSetIrq)(pVM->pdm.s.IoApic.CTX_SUFF(pDevIns), u8Irq, u8Level, uTagSrc);
172 rc = VINF_SUCCESS;
173 }
174
175 if (!uTagSrc && u8Level == PDM_IRQ_LEVEL_LOW)
176 VBOXVMM_PDM_IRQ_LOW(VMMGetCpu(pVM), 0, 0);
177 pdmUnlock(pVM);
178 return rc;
179}
180
181
182/**
183 * Sets the pending I/O APIC interrupt.
184 *
185 * @returns VBox status code.
186 * @param pVM The cross context VM structure.
187 * @param u8Irq The IRQ line.
188 * @param u8Level The new level.
189 * @param uTagSrc The IRQ tag and source tracer ID.
190 */
191VMM_INT_DECL(int) PDMIoApicSetIrq(PVM pVM, uint8_t u8Irq, uint8_t u8Level, uint32_t uTagSrc)
192{
193 if (pVM->pdm.s.IoApic.CTX_SUFF(pDevIns))
194 {
195 Assert(pVM->pdm.s.IoApic.CTX_SUFF(pfnSetIrq));
196 pdmLock(pVM);
197 pVM->pdm.s.IoApic.CTX_SUFF(pfnSetIrq)(pVM->pdm.s.IoApic.CTX_SUFF(pDevIns), u8Irq, u8Level, uTagSrc);
198 pdmUnlock(pVM);
199 return VINF_SUCCESS;
200 }
201 return VERR_PDM_NO_PIC_INSTANCE;
202}
203
204/**
205 * Send a MSI to an I/O APIC.
206 *
207 * @returns VBox status code.
208 * @param pVM The cross context VM structure.
209 * @param GCAddr Request address.
210 * @param uValue Request value.
211 * @param uTagSrc The IRQ tag and source tracer ID.
212 */
213VMM_INT_DECL(int) PDMIoApicSendMsi(PVM pVM, RTGCPHYS GCAddr, uint32_t uValue, uint32_t uTagSrc)
214{
215 if (pVM->pdm.s.IoApic.CTX_SUFF(pDevIns))
216 {
217 Assert(pVM->pdm.s.IoApic.CTX_SUFF(pfnSendMsi));
218 pdmLock(pVM);
219 pVM->pdm.s.IoApic.CTX_SUFF(pfnSendMsi)(pVM->pdm.s.IoApic.CTX_SUFF(pDevIns), GCAddr, uValue, uTagSrc);
220 pdmUnlock(pVM);
221 return VINF_SUCCESS;
222 }
223 return VERR_PDM_NO_PIC_INSTANCE;
224}
225
226
227
228/**
229 * Returns the presence of an IO-APIC.
230 *
231 * @returns VBox true if an IO-APIC is present.
232 * @param pVM The cross context VM structure.
233 */
234VMM_INT_DECL(bool) PDMHasIoApic(PVM pVM)
235{
236 return pVM->pdm.s.IoApic.CTX_SUFF(pDevIns) != NULL;
237}
238
239
240/**
241 * Returns the presence of a Local APIC.
242 *
243 * @returns VBox true if a Local APIC is present.
244 * @param pVM The cross context VM structure.
245 */
246VMM_INT_DECL(bool) PDMHasApic(PVM pVM)
247{
248 return pVM->pdm.s.Apic.CTX_SUFF(pDevIns) != NULL;
249}
250
251
252/**
253 * Set the APIC base.
254 *
255 * @returns Strict VBox status code.
256 * @param pVCpu The cross context virtual CPU structure.
257 * @param u64Base The new base.
258 */
259VMMDECL(VBOXSTRICTRC) PDMApicSetBaseMsr(PVMCPU pVCpu, uint64_t u64Base)
260{
261 PVM pVM = pVCpu->CTX_SUFF(pVM);
262 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
263 {
264 Assert(pVM->pdm.s.Apic.CTX_SUFF(pfnSetBaseMsr));
265#ifndef VBOX_WITH_NEW_APIC
266 pdmLock(pVM);
267#endif
268 VBOXSTRICTRC rcStrict = pVM->pdm.s.Apic.CTX_SUFF(pfnSetBaseMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, u64Base);
269
270 /* Update CPUM's copy of the APIC base. */
271 PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
272 Assert(pCtx);
273 pCtx->msrApicBase = pVM->pdm.s.Apic.CTX_SUFF(pfnGetBaseMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu);
274
275#ifndef VBOX_WITH_NEW_APIC
276 pdmUnlock(pVM);
277#endif
278 return rcStrict;
279 }
280
281#ifdef IN_RING3
282 LogRelMax(5, ("PDM: APIC%U: Writing APIC base MSR (%#x) invalid since there isn't an APIC -> #GP(0)\n", pVCpu->idCpu,
283 MSR_IA32_APICBASE));
284 return VERR_CPUM_RAISE_GP_0;
285#else
286 return VINF_CPUM_R3_MSR_WRITE;
287#endif
288}
289
290
291/**
292 * Get the APIC base MSR from the APIC device.
293 *
294 * @returns Strict VBox status code.
295 * @param pVCpu The cross context virtual CPU structure.
296 * @param pu64Base Where to store the APIC base.
297 * @param fIgnoreErrors Whether to ignore errors (i.e. not a real guest MSR
298 * access).
299 */
300VMMDECL(VBOXSTRICTRC) PDMApicGetBaseMsr(PVMCPU pVCpu, uint64_t *pu64Base, bool fIgnoreErrors)
301{
302 PVM pVM = pVCpu->CTX_SUFF(pVM);
303 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
304 {
305 Assert(pVM->pdm.s.Apic.CTX_SUFF(pfnGetBaseMsr));
306#ifdef VBOX_WITH_NEW_APIC
307 VMCPU_ASSERT_EMT_OR_NOT_RUNNING(pVCpu);
308 *pu64Base = pVM->pdm.s.Apic.CTX_SUFF(pfnGetBaseMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu);
309#else
310 pdmLock(pVM);
311 *pu64Base = pVM->pdm.s.Apic.CTX_SUFF(pfnGetBaseMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu);
312 pdmUnlock(pVM);
313#endif
314 return VINF_SUCCESS;
315 }
316
317 *pu64Base = 0;
318 if (fIgnoreErrors)
319 return VINF_SUCCESS;
320
321#ifdef IN_RING3
322 LogRelMax(5, ("PDM: APIC%u: Reading APIC base MSR (%#x) invalid without an APIC instance -> #GP(0)\n", pVCpu->idCpu,
323 MSR_IA32_APICBASE));
324 return VERR_CPUM_RAISE_GP_0;
325#else
326 return VINF_CPUM_R3_MSR_WRITE;
327#endif
328}
329
330
331/**
332 * Set the TPR (Task Priority Register).
333 *
334 * @returns VBox status code.
335 * @param pVCpu The cross context virtual CPU structure.
336 * @param u8TPR The new TPR.
337 */
338VMMDECL(int) PDMApicSetTPR(PVMCPU pVCpu, uint8_t u8TPR)
339{
340 PVM pVM = pVCpu->CTX_SUFF(pVM);
341 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
342 {
343 Assert(pVM->pdm.s.Apic.CTX_SUFF(pfnSetTpr));
344#ifdef VBOX_WITH_NEW_APIC
345 pVM->pdm.s.Apic.CTX_SUFF(pfnSetTpr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, u8TPR);
346#else
347 pdmLock(pVM);
348 pVM->pdm.s.Apic.CTX_SUFF(pfnSetTpr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, u8TPR);
349 pdmUnlock(pVM);
350#endif
351 return VINF_SUCCESS;
352 }
353 return VERR_PDM_NO_APIC_INSTANCE;
354}
355
356
357/**
358 * Get the TPR (Task Priority Register).
359 *
360 * @returns VINF_SUCCESS or VERR_PDM_NO_APIC_INSTANCE.
361 * @param pVCpu The cross context virtual CPU structure.
362 * @param pu8TPR Where to store the TRP.
363 * @param pfPending Where to store whether there is a pending interrupt
364 * (out, optional).
365 * @param pu8PendingIntr Where to store the highest-priority pending
366 * interrupt (out, optional).
367 *
368 * @remarks No-long-jump zone!!!
369 */
370VMMDECL(int) PDMApicGetTPR(PVMCPU pVCpu, uint8_t *pu8TPR, bool *pfPending, uint8_t *pu8PendingIntr)
371{
372 PVM pVM = pVCpu->CTX_SUFF(pVM);
373 PPDMDEVINS pApicIns = pVM->pdm.s.Apic.CTX_SUFF(pDevIns);
374 if (pApicIns)
375 {
376 /*
377 * Note! We don't acquire the PDM lock here as we're just reading
378 * information. Doing so causes massive contention as this
379 * function is called very often by each and every VCPU.
380 */
381 Assert(pVM->pdm.s.Apic.CTX_SUFF(pfnGetTpr));
382 *pu8TPR = pVM->pdm.s.Apic.CTX_SUFF(pfnGetTpr)(pApicIns, pVCpu, pfPending, pu8PendingIntr);
383 return VINF_SUCCESS;
384 }
385 *pu8TPR = 0;
386 return VERR_PDM_NO_APIC_INSTANCE;
387}
388
389
390/**
391 * Write a MSR in APIC range.
392 *
393 * @returns Strict VBox status code.
394 * @param pVCpu The cross context virtual CPU structure.
395 * @param u32Reg MSR to write.
396 * @param u64Value Value to write.
397 */
398VMM_INT_DECL(VBOXSTRICTRC) PDMApicWriteMsr(PVMCPU pVCpu, uint32_t u32Reg, uint64_t u64Value)
399{
400 PVM pVM = pVCpu->CTX_SUFF(pVM);
401 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
402 {
403 AssertPtr(pVM->pdm.s.Apic.CTX_SUFF(pfnWriteMsr));
404 return pVM->pdm.s.Apic.CTX_SUFF(pfnWriteMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, u32Reg, u64Value);
405 }
406 return VERR_CPUM_RAISE_GP_0;
407}
408
409
410/**
411 * Read a MSR in APIC range.
412 *
413 * @returns Strict VBox status code.
414 * @param pVCpu The cross context virtual CPU structure.
415 * @param u32Reg MSR to read.
416 * @param pu64Value Where to store the value read.
417 */
418VMM_INT_DECL(VBOXSTRICTRC) PDMApicReadMsr(PVMCPU pVCpu, uint32_t u32Reg, uint64_t *pu64Value)
419{
420 PVM pVM = pVCpu->CTX_SUFF(pVM);
421 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
422 {
423 AssertPtr(pVM->pdm.s.Apic.CTX_SUFF(pfnReadMsr));
424 return pVM->pdm.s.Apic.CTX_SUFF(pfnReadMsr)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns), pVCpu, u32Reg, pu64Value);
425 }
426 return VERR_CPUM_RAISE_GP_0;
427}
428
429
430/**
431 * Gets the frequency of the APIC timer.
432 *
433 * @returns VBox status code.
434 * @param pVM The cross context VM structure.
435 * @param pu64Value Where to store the frequency.
436 */
437VMM_INT_DECL(int) PDMApicGetTimerFreq(PVM pVM, uint64_t *pu64Value)
438{
439 if (pVM->pdm.s.Apic.CTX_SUFF(pDevIns))
440 {
441 AssertPtr(pVM->pdm.s.Apic.CTX_SUFF(pfnGetTimerFreq));
442 *pu64Value = pVM->pdm.s.Apic.CTX_SUFF(pfnGetTimerFreq)(pVM->pdm.s.Apic.CTX_SUFF(pDevIns));
443 return VINF_SUCCESS;
444 }
445 return VERR_PDM_NO_APIC_INSTANCE;
446}
447
448
449/**
450 * Locks PDM.
451 * This might call back to Ring-3 in order to deal with lock contention in GC and R3.
452 *
453 * @param pVM The cross context VM structure.
454 */
455void pdmLock(PVM pVM)
456{
457#ifdef IN_RING3
458 int rc = PDMCritSectEnter(&pVM->pdm.s.CritSect, VERR_IGNORED);
459#else
460 int rc = PDMCritSectEnter(&pVM->pdm.s.CritSect, VERR_GENERAL_FAILURE);
461 if (rc == VERR_GENERAL_FAILURE)
462 rc = VMMRZCallRing3NoCpu(pVM, VMMCALLRING3_PDM_LOCK, 0);
463#endif
464 AssertRC(rc);
465}
466
467
468/**
469 * Locks PDM but don't go to ring-3 if it's owned by someone.
470 *
471 * @returns VINF_SUCCESS on success.
472 * @returns rc if we're in GC or R0 and can't get the lock.
473 * @param pVM The cross context VM structure.
474 * @param rc The RC to return in GC or R0 when we can't get the lock.
475 */
476int pdmLockEx(PVM pVM, int rc)
477{
478 return PDMCritSectEnter(&pVM->pdm.s.CritSect, rc);
479}
480
481
482/**
483 * Unlocks PDM.
484 *
485 * @param pVM The cross context VM structure.
486 */
487void pdmUnlock(PVM pVM)
488{
489 PDMCritSectLeave(&pVM->pdm.s.CritSect);
490}
491
492
493/**
494 * Converts ring 3 VMM heap pointer to a guest physical address
495 *
496 * @returns VBox status code.
497 * @param pVM The cross context VM structure.
498 * @param pv Ring-3 pointer.
499 * @param pGCPhys GC phys address (out).
500 */
501VMM_INT_DECL(int) PDMVmmDevHeapR3ToGCPhys(PVM pVM, RTR3PTR pv, RTGCPHYS *pGCPhys)
502{
503 if (RT_LIKELY(pVM->pdm.s.GCPhysVMMDevHeap != NIL_RTGCPHYS))
504 {
505 RTR3UINTPTR const offHeap = (RTR3UINTPTR)pv - (RTR3UINTPTR)pVM->pdm.s.pvVMMDevHeap;
506 if (RT_LIKELY(offHeap < pVM->pdm.s.cbVMMDevHeap))
507 {
508 *pGCPhys = pVM->pdm.s.GCPhysVMMDevHeap + offHeap;
509 return VINF_SUCCESS;
510 }
511
512 /* Don't assert here as this is called before we can catch ring-0 assertions. */
513 Log(("PDMVmmDevHeapR3ToGCPhys: pv=%p pvVMMDevHeap=%p cbVMMDevHeap=%#x\n",
514 pv, pVM->pdm.s.pvVMMDevHeap, pVM->pdm.s.cbVMMDevHeap));
515 }
516 else
517 Log(("PDMVmmDevHeapR3ToGCPhys: GCPhysVMMDevHeap=%RGp (pv=%p)\n", pVM->pdm.s.GCPhysVMMDevHeap, pv));
518 return VERR_PDM_DEV_HEAP_R3_TO_GCPHYS;
519}
520
521
522/**
523 * Checks if the vmm device heap is enabled (== vmm device's pci region mapped)
524 *
525 * @returns dev heap enabled status (true/false)
526 * @param pVM The cross context VM structure.
527 */
528VMM_INT_DECL(bool) PDMVmmDevHeapIsEnabled(PVM pVM)
529{
530 return pVM->pdm.s.GCPhysVMMDevHeap != NIL_RTGCPHYS;
531}
Note: See TracBrowser for help on using the repository browser.

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