VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/PDMDevMiscHlp.cpp@ 60650

Last change on this file since 60650 was 60404, checked in by vboxsync, 9 years ago

VMM,Devices,Main: Implemented soft/warm reset for shutdown status codes 05h, 09h and 0Ah.

This is a shot at adjusting our VM reset handling to handle the ancient way of
getting a 286 out of protected mode and back to real mode. Our exiting reset
code (XXXR3Reset, PDMDEVREG::pfnReset, and so on) is doing a cold reset of the
system and then some additional device & memory initialization that the firmware
is usually responsible for doing. When the guest triggers a reset via the
keyboard controller, system control port A, CPU triple fault, and possibly ACPI,
only the CPU is supposed to be reset. The BIOS would then decide whether memory
and devices needed resetting as well, or if the resetter justed wanted to get out
protected mode and resume executing some real mode code pointed to by 467h.

  • New states SOFT_RESETTING and SOFT_RESETTING_LS. The latter returns to RUNNING_LS, not SUSPENDED_LS like for hard reset.
  • Added a firmware interface so the VMM/PDM can ask it whether we're supposed to do a hard reset or a soft(/warm) one.
  • Implemented firmware interface for the PC BIOS (but not EFI). It indicates soft(/warm) reset when CMOS[0xf] is 5, 9 or 10.
  • Moved the CMOS[0xf] resetting from the RTC device to the PC BIOS since it's firmware thing, not RTC.
  • Added a flag parameter to PDMDevHlpVMReset for specifying the source of the reset operation. One class of sources (GIM) will always trigger hard resets, whereas the others will check with the firmware first.
  • Added PDMR3GetResetInfo for query the flags passed to PDMDevHlpVMReset and for asking the firmware whether it's a hard or soft reset. The latter, however, is only done if only CPU 0 is active. Systems with more than one CPU in a state other than EMSTATE_WAIT_SIPI status will always be hard reset.
  • Added internal VMR3ResetFF and VMR3ResetTripleFault APIs for handling the VM_FF_RESET and VINF_EM_TRIPLE_FAULT conditions.
  • Added PMDR3ResetSoft and had it call pfnSoftReset (which is now defined).

Warning! Major PDM_DEVHLPR3_VERSION change, minor PDM_DEVREG_VERSION change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 27.0 KB
Line 
1/* $Id: PDMDevMiscHlp.cpp 60404 2016-04-09 23:45:55Z vboxsync $ */
2/** @file
3 * PDM - Pluggable Device and Driver Manager, Misc. Device Helpers.
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_DEVICE
23#include "PDMInternal.h"
24#include <VBox/vmm/pdm.h>
25#include <VBox/vmm/pgm.h>
26#include <VBox/vmm/hm.h>
27#ifdef VBOX_WITH_REM
28# include <VBox/vmm/rem.h>
29#endif
30#include <VBox/vmm/vm.h>
31#include <VBox/vmm/vmm.h>
32
33#include <VBox/log.h>
34#include <VBox/err.h>
35#include <iprt/asm.h>
36#include <iprt/assert.h>
37#include <iprt/thread.h>
38
39
40#include "PDMInline.h"
41#include "dtrace/VBoxVMM.h"
42
43
44
45/** @name Ring-3 PIC Helpers
46 * @{
47 */
48
49/** @interface_method_impl{PDMPICHLPR3,pfnSetInterruptFF} */
50static DECLCALLBACK(void) pdmR3PicHlp_SetInterruptFF(PPDMDEVINS pDevIns)
51{
52 PDMDEV_ASSERT_DEVINS(pDevIns);
53 PVM pVM = pDevIns->Internal.s.pVMR3;
54 PVMCPU pVCpu = &pVM->aCpus[0]; /* for PIC we always deliver to CPU 0, MP use APIC */
55
56 if (pVM->pdm.s.Apic.pfnLocalInterruptR3)
57 {
58 LogFlow(("pdmR3PicHlp_SetInterruptFF: caller='%s'/%d: Setting local interrupt on LAPIC\n",
59 pDevIns->pReg->szName, pDevIns->iInstance));
60
61 /* Raise the LAPIC's LINT0 line instead of signaling the CPU directly. */
62 /** @todo 'rcRZ' propagation to pfnLocalInterrupt from caller. */
63 pVM->pdm.s.Apic.pfnLocalInterruptR3(pVM->pdm.s.Apic.pDevInsR3, pVCpu, 0 /* u8Pin */, 1 /* u8Level */,
64 VINF_SUCCESS /* rcRZ */);
65 return;
66 }
67
68 LogFlow(("pdmR3PicHlp_SetInterruptFF: caller='%s'/%d: VMCPU_FF_INTERRUPT_PIC %d -> 1\n",
69 pDevIns->pReg->szName, pDevIns->iInstance, VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC)));
70
71 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC);
72#ifdef VBOX_WITH_REM
73 REMR3NotifyInterruptSet(pVM, pVCpu);
74#endif
75 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM | VMNOTIFYFF_FLAGS_POKE);
76}
77
78
79/** @interface_method_impl{PDMPICHLPR3,pfnClearInterruptFF} */
80static DECLCALLBACK(void) pdmR3PicHlp_ClearInterruptFF(PPDMDEVINS pDevIns)
81{
82 PDMDEV_ASSERT_DEVINS(pDevIns);
83 PVM pVM = pDevIns->Internal.s.pVMR3;
84 PVMCPU pVCpu = &pVM->aCpus[0]; /* for PIC we always deliver to CPU 0, MP use APIC */
85
86 if (pVM->pdm.s.Apic.pfnLocalInterruptR3)
87 {
88 /* Raise the LAPIC's LINT0 line instead of signaling the CPU directly. */
89 LogFlow(("pdmR3PicHlp_ClearInterruptFF: caller='%s'/%d: Clearing local interrupt on LAPIC\n",
90 pDevIns->pReg->szName, pDevIns->iInstance));
91
92 /* Lower the LAPIC's LINT0 line instead of signaling the CPU directly. */
93 /** @todo 'rcRZ' propagation to pfnLocalInterrupt from caller. */
94 pVM->pdm.s.Apic.pfnLocalInterruptR3(pVM->pdm.s.Apic.pDevInsR3, pVCpu, 0 /* u8Pin */, 0 /* u8Level */,
95 VINF_SUCCESS /* rcRZ */);
96 return;
97 }
98
99 LogFlow(("pdmR3PicHlp_ClearInterruptFF: caller='%s'/%d: VMCPU_FF_INTERRUPT_PIC %d -> 0\n",
100 pDevIns->pReg->szName, pDevIns->iInstance, VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC)));
101
102 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_PIC);
103#ifdef VBOX_WITH_REM
104 REMR3NotifyInterruptClear(pVM, pVCpu);
105#endif
106}
107
108
109/** @interface_method_impl{PDMPICHLPR3,pfnLock} */
110static DECLCALLBACK(int) pdmR3PicHlp_Lock(PPDMDEVINS pDevIns, int rc)
111{
112 PDMDEV_ASSERT_DEVINS(pDevIns);
113 return pdmLockEx(pDevIns->Internal.s.pVMR3, rc);
114}
115
116
117/** @interface_method_impl{PDMPICHLPR3,pfnUnlock} */
118static DECLCALLBACK(void) pdmR3PicHlp_Unlock(PPDMDEVINS pDevIns)
119{
120 PDMDEV_ASSERT_DEVINS(pDevIns);
121 pdmUnlock(pDevIns->Internal.s.pVMR3);
122}
123
124
125/** @interface_method_impl{PDMPICHLPR3,pfnGetRCHelpers} */
126static DECLCALLBACK(PCPDMPICHLPRC) pdmR3PicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
127{
128 PDMDEV_ASSERT_DEVINS(pDevIns);
129 PVM pVM = pDevIns->Internal.s.pVMR3;
130 VM_ASSERT_EMT(pVM);
131
132 RTRCPTR pRCHelpers = NIL_RTRCPTR;
133 if (!HMIsEnabled(pVM))
134 {
135 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCPicHlp", &pRCHelpers);
136 AssertReleaseRC(rc);
137 AssertRelease(pRCHelpers);
138 }
139
140 LogFlow(("pdmR3PicHlp_GetRCHelpers: caller='%s'/%d: returns %RRv\n",
141 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
142 return pRCHelpers;
143}
144
145
146/** @interface_method_impl{PDMPICHLPR3,pfnGetR0Helpers} */
147static DECLCALLBACK(PCPDMPICHLPR0) pdmR3PicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
148{
149 PDMDEV_ASSERT_DEVINS(pDevIns);
150 PVM pVM = pDevIns->Internal.s.pVMR3;
151 VM_ASSERT_EMT(pVM);
152 PCPDMPICHLPR0 pR0Helpers = 0;
153 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0PicHlp", &pR0Helpers);
154 AssertReleaseRC(rc);
155 AssertRelease(pR0Helpers);
156 LogFlow(("pdmR3PicHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
157 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
158 return pR0Helpers;
159}
160
161
162/**
163 * PIC Device Helpers.
164 */
165const PDMPICHLPR3 g_pdmR3DevPicHlp =
166{
167 PDM_PICHLPR3_VERSION,
168 pdmR3PicHlp_SetInterruptFF,
169 pdmR3PicHlp_ClearInterruptFF,
170 pdmR3PicHlp_Lock,
171 pdmR3PicHlp_Unlock,
172 pdmR3PicHlp_GetRCHelpers,
173 pdmR3PicHlp_GetR0Helpers,
174 PDM_PICHLPR3_VERSION /* the end */
175};
176
177/** @} */
178
179
180
181
182/** @name R3 APIC Helpers
183 * @{
184 */
185
186/** @interface_method_impl{PDMAPICHLPR3,pfnSetInterruptFF} */
187static DECLCALLBACK(void) pdmR3ApicHlp_SetInterruptFF(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu)
188{
189 PDMDEV_ASSERT_DEVINS(pDevIns);
190 PVM pVM = pDevIns->Internal.s.pVMR3;
191 PVMCPU pVCpu = &pVM->aCpus[idCpu];
192
193 AssertReturnVoid(idCpu < pVM->cCpus);
194
195 LogFlow(("pdmR3ApicHlp_SetInterruptFF: caller='%s'/%d: VMCPU_FF_INTERRUPT_APIC(%d) %d -> 1\n",
196 pDevIns->pReg->szName, pDevIns->iInstance, idCpu, VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_APIC)));
197
198 switch (enmType)
199 {
200 case PDMAPICIRQ_HARDWARE:
201 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_APIC);
202 break;
203 case PDMAPICIRQ_NMI:
204 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_NMI);
205 break;
206 case PDMAPICIRQ_SMI:
207 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_SMI);
208 break;
209 case PDMAPICIRQ_EXTINT:
210 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC);
211 break;
212 default:
213 AssertMsgFailed(("enmType=%d\n", enmType));
214 break;
215 }
216#ifdef VBOX_WITH_REM
217 REMR3NotifyInterruptSet(pVM, pVCpu);
218#endif
219 VMR3NotifyCpuFFU(pVCpu->pUVCpu, VMNOTIFYFF_FLAGS_DONE_REM | VMNOTIFYFF_FLAGS_POKE);
220}
221
222
223/** @interface_method_impl{PDMAPICHLPR3,pfnClearInterruptFF} */
224static DECLCALLBACK(void) pdmR3ApicHlp_ClearInterruptFF(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu)
225{
226 PDMDEV_ASSERT_DEVINS(pDevIns);
227 PVM pVM = pDevIns->Internal.s.pVMR3;
228 PVMCPU pVCpu = &pVM->aCpus[idCpu];
229
230 AssertReturnVoid(idCpu < pVM->cCpus);
231
232 LogFlow(("pdmR3ApicHlp_ClearInterruptFF: caller='%s'/%d: VMCPU_FF_INTERRUPT_APIC(%d) %d -> 0\n",
233 pDevIns->pReg->szName, pDevIns->iInstance, idCpu, VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INTERRUPT_APIC)));
234
235 /* Note: NMI/SMI can't be cleared. */
236 switch (enmType)
237 {
238 case PDMAPICIRQ_HARDWARE:
239 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_APIC);
240 break;
241 case PDMAPICIRQ_EXTINT:
242 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_PIC);
243 break;
244 default:
245 AssertMsgFailed(("enmType=%d\n", enmType));
246 break;
247 }
248#ifdef VBOX_WITH_REM
249 REMR3NotifyInterruptClear(pVM, pVCpu);
250#endif
251}
252
253
254/** @interface_method_impl{PDMAPICHLPR3,pfnCalcIrqTag} */
255static DECLCALLBACK(uint32_t) pdmR3ApicHlp_CalcIrqTag(PPDMDEVINS pDevIns, uint8_t u8Level)
256{
257 PDMDEV_ASSERT_DEVINS(pDevIns);
258 PVM pVM = pDevIns->Internal.s.pVMR3;
259 Assert(u8Level == PDM_IRQ_LEVEL_HIGH || u8Level == PDM_IRQ_LEVEL_FLIP_FLOP);
260
261 pdmLock(pVM);
262
263 uint32_t uTagSrc = pdmCalcIrqTag(pVM, pDevIns->idTracing);
264 if (u8Level == PDM_IRQ_LEVEL_HIGH)
265 VBOXVMM_PDM_IRQ_HIGH(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
266 else
267 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
268
269
270 pdmUnlock(pVM);
271 LogFlow(("pdmR3ApicHlp_CalcIrqTag: caller='%s'/%d: returns %#x (u8Level=%d)\n",
272 pDevIns->pReg->szName, pDevIns->iInstance, uTagSrc, u8Level));
273 return uTagSrc;
274}
275
276
277/** @interface_method_impl{PDMAPICHLPR3,pfnChangeFeature} */
278static DECLCALLBACK(void) pdmR3ApicHlp_ChangeFeature(PPDMDEVINS pDevIns, PDMAPICMODE enmMode)
279{
280#ifdef VBOX_WITH_NEW_APIC
281 /*
282 * The old code is also most likely incorrect with regards to changing the CPUID bits,
283 * see @bugref{8245#c32}.
284 *
285 * The new code should directly invoke APICUpdateCpuIdForMode() instead of using this
286 * indirect helper.
287 */
288 AssertMsgFailed(("pdmR3ApicHlp_ChangeFeature unsupported in VBOX_WITH_NEW_APIC!"));
289#else
290 PDMDEV_ASSERT_DEVINS(pDevIns);
291 LogFlow(("pdmR3ApicHlp_ChangeFeature: caller='%s'/%d: mode=%d\n",
292 pDevIns->pReg->szName, pDevIns->iInstance, (int)enmMode));
293 switch (enmMode)
294 {
295 case PDMAPICMODE_NONE:
296 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_APIC);
297 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_X2APIC);
298 break;
299 case PDMAPICMODE_APIC:
300 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_APIC);
301 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_X2APIC);
302 break;
303 case PDMAPICMODE_X2APIC:
304 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_X2APIC);
305 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMR3, CPUMCPUIDFEATURE_APIC);
306 break;
307 default:
308 AssertMsgFailed(("Unknown APIC mode: %d\n", (int)enmMode));
309 }
310#endif
311}
312
313/** @interface_method_impl{PDMAPICHLPR3,pfnGetCpuId} */
314static DECLCALLBACK(VMCPUID) pdmR3ApicHlp_GetCpuId(PPDMDEVINS pDevIns)
315{
316 PDMDEV_ASSERT_DEVINS(pDevIns);
317 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
318 return VMMGetCpuId(pDevIns->Internal.s.pVMR3);
319}
320
321
322/** @interface_method_impl{PDMAPICHLPR3,pfnSendStartupIpi} */
323static DECLCALLBACK(void) pdmR3ApicHlp_SendStartupIpi(PPDMDEVINS pDevIns, VMCPUID idCpu, uint32_t uVector)
324{
325 PDMDEV_ASSERT_DEVINS(pDevIns);
326 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
327 VMMR3SendStartupIpi(pDevIns->Internal.s.pVMR3, idCpu, uVector);
328}
329
330
331/** @interface_method_impl{PDMAPICHLPR3,pfnSendInitIpi} */
332static DECLCALLBACK(void) pdmR3ApicHlp_SendInitIpi(PPDMDEVINS pDevIns, VMCPUID idCpu)
333{
334 PDMDEV_ASSERT_DEVINS(pDevIns);
335 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
336 VMMR3SendInitIpi(pDevIns->Internal.s.pVMR3, idCpu);
337}
338
339
340/** @interface_method_impl{PDMAPICHLPR3,pfnGetRCHelpers} */
341static DECLCALLBACK(PCPDMAPICHLPRC) pdmR3ApicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
342{
343 PDMDEV_ASSERT_DEVINS(pDevIns);
344 PVM pVM = pDevIns->Internal.s.pVMR3;
345 VM_ASSERT_EMT(pVM);
346
347 RTRCPTR pRCHelpers = NIL_RTRCPTR;
348 if (!HMIsEnabled(pVM))
349 {
350 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCApicHlp", &pRCHelpers);
351 AssertReleaseRC(rc);
352 AssertRelease(pRCHelpers);
353 }
354
355 LogFlow(("pdmR3ApicHlp_GetRCHelpers: caller='%s'/%d: returns %RRv\n",
356 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
357 return pRCHelpers;
358}
359
360
361/** @interface_method_impl{PDMAPICHLPR3,pfnGetR0Helpers} */
362static DECLCALLBACK(PCPDMAPICHLPR0) pdmR3ApicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
363{
364 PDMDEV_ASSERT_DEVINS(pDevIns);
365 PVM pVM = pDevIns->Internal.s.pVMR3;
366 VM_ASSERT_EMT(pVM);
367 PCPDMAPICHLPR0 pR0Helpers = 0;
368 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0ApicHlp", &pR0Helpers);
369 AssertReleaseRC(rc);
370 AssertRelease(pR0Helpers);
371 LogFlow(("pdmR3ApicHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
372 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
373 return pR0Helpers;
374}
375
376
377/** @interface_method_impl{PDMAPICHLPR3,pfnGetR3CritSect} */
378static DECLCALLBACK(R3PTRTYPE(PPDMCRITSECT)) pdmR3ApicHlp_GetR3CritSect(PPDMDEVINS pDevIns)
379{
380 PDMDEV_ASSERT_DEVINS(pDevIns);
381 LogFlow(("pdmR3ApicHlp_Lock: caller='%s'/%d\n", pDevIns->pReg->szName, pDevIns->iInstance));
382 return &pDevIns->Internal.s.pVMR3->pdm.s.CritSect;
383}
384
385
386/** @interface_method_impl{PDMAPICHLPR3,pfnGetRCCritSect} */
387static DECLCALLBACK(RCPTRTYPE(PPDMCRITSECT)) pdmR3ApicHlp_GetRCCritSect(PPDMDEVINS pDevIns)
388{
389 PDMDEV_ASSERT_DEVINS(pDevIns);
390 PVM pVM = pDevIns->Internal.s.pVMR3;
391 RTRCPTR RCPtr = MMHyperCCToRC(pVM, &pVM->pdm.s.CritSect);
392 LogFlow(("pdmR3ApicHlp_GetR0CritSect: caller='%s'/%d: return %RRv\n", pDevIns->pReg->szName, pDevIns->iInstance, RCPtr));
393 return RCPtr;
394}
395
396
397/** @interface_method_impl{PDMAPICHLPR3,pfnGetR3CritSect} */
398static DECLCALLBACK(R0PTRTYPE(PPDMCRITSECT)) pdmR3ApicHlp_GetR0CritSect(PPDMDEVINS pDevIns)
399{
400 PDMDEV_ASSERT_DEVINS(pDevIns);
401 PVM pVM = pDevIns->Internal.s.pVMR3;
402 RTR0PTR R0Ptr = MMHyperCCToR0(pVM, &pVM->pdm.s.CritSect);
403 LogFlow(("pdmR3ApicHlp_GetR0CritSect: caller='%s'/%d: return %RHv\n", pDevIns->pReg->szName, pDevIns->iInstance, R0Ptr));
404 return R0Ptr;
405}
406
407
408
409/**
410 * APIC Device Helpers.
411 */
412const PDMAPICHLPR3 g_pdmR3DevApicHlp =
413{
414 PDM_APICHLPR3_VERSION,
415 pdmR3ApicHlp_SetInterruptFF,
416 pdmR3ApicHlp_ClearInterruptFF,
417 pdmR3ApicHlp_CalcIrqTag,
418 pdmR3ApicHlp_ChangeFeature,
419 pdmR3ApicHlp_GetCpuId,
420 pdmR3ApicHlp_SendStartupIpi,
421 pdmR3ApicHlp_SendInitIpi,
422 pdmR3ApicHlp_GetRCHelpers,
423 pdmR3ApicHlp_GetR0Helpers,
424 pdmR3ApicHlp_GetR3CritSect,
425 pdmR3ApicHlp_GetRCCritSect,
426 pdmR3ApicHlp_GetR0CritSect,
427 PDM_APICHLPR3_VERSION /* the end */
428};
429
430/** @} */
431
432
433
434
435/** @name Ring-3 I/O APIC Helpers
436 * @{
437 */
438
439/** @interface_method_impl{PDMIOAPICHLPR3,pfnApicBusDeliver} */
440static DECLCALLBACK(int) pdmR3IoApicHlp_ApicBusDeliver(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
441 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode, uint32_t uTagSrc)
442{
443 PDMDEV_ASSERT_DEVINS(pDevIns);
444 PVM pVM = pDevIns->Internal.s.pVMR3;
445 LogFlow(("pdmR3IoApicHlp_ApicBusDeliver: caller='%s'/%d: u8Dest=%RX8 u8DestMode=%RX8 u8DeliveryMode=%RX8 iVector=%RX8 u8Polarity=%RX8 u8TriggerMode=%RX8 uTagSrc=%#x\n",
446 pDevIns->pReg->szName, pDevIns->iInstance, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode, uTagSrc));
447 if (pVM->pdm.s.Apic.pfnBusDeliverR3)
448 return pVM->pdm.s.Apic.pfnBusDeliverR3(pVM->pdm.s.Apic.pDevInsR3, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode, uTagSrc);
449 return VINF_SUCCESS;
450}
451
452
453/** @interface_method_impl{PDMIOAPICHLPR3,pfnLock} */
454static DECLCALLBACK(int) pdmR3IoApicHlp_Lock(PPDMDEVINS pDevIns, int rc)
455{
456 PDMDEV_ASSERT_DEVINS(pDevIns);
457 LogFlow(("pdmR3IoApicHlp_Lock: caller='%s'/%d: rc=%Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
458 return pdmLockEx(pDevIns->Internal.s.pVMR3, rc);
459}
460
461
462/** @interface_method_impl{PDMIOAPICHLPR3,pfnUnlock} */
463static DECLCALLBACK(void) pdmR3IoApicHlp_Unlock(PPDMDEVINS pDevIns)
464{
465 PDMDEV_ASSERT_DEVINS(pDevIns);
466 LogFlow(("pdmR3IoApicHlp_Unlock: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
467 pdmUnlock(pDevIns->Internal.s.pVMR3);
468}
469
470
471/** @interface_method_impl{PDMIOAPICHLPR3,pfnGetRCHelpers} */
472static DECLCALLBACK(PCPDMIOAPICHLPRC) pdmR3IoApicHlp_GetRCHelpers(PPDMDEVINS pDevIns)
473{
474 PDMDEV_ASSERT_DEVINS(pDevIns);
475 PVM pVM = pDevIns->Internal.s.pVMR3;
476 VM_ASSERT_EMT(pVM);
477
478 RTRCPTR pRCHelpers = NIL_RTRCPTR;
479 if (!HMIsEnabled(pVM))
480 {
481 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCIoApicHlp", &pRCHelpers);
482 AssertReleaseRC(rc);
483 AssertRelease(pRCHelpers);
484 }
485
486 LogFlow(("pdmR3IoApicHlp_GetRCHelpers: caller='%s'/%d: returns %RRv\n",
487 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
488 return pRCHelpers;
489}
490
491
492/** @interface_method_impl{PDMIOAPICHLPR3,pfnGetR0Helpers} */
493static DECLCALLBACK(PCPDMIOAPICHLPR0) pdmR3IoApicHlp_GetR0Helpers(PPDMDEVINS pDevIns)
494{
495 PDMDEV_ASSERT_DEVINS(pDevIns);
496 PVM pVM = pDevIns->Internal.s.pVMR3;
497 VM_ASSERT_EMT(pVM);
498 PCPDMIOAPICHLPR0 pR0Helpers = 0;
499 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0IoApicHlp", &pR0Helpers);
500 AssertReleaseRC(rc);
501 AssertRelease(pR0Helpers);
502 LogFlow(("pdmR3IoApicHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
503 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
504 return pR0Helpers;
505}
506
507
508/**
509 * I/O APIC Device Helpers.
510 */
511const PDMIOAPICHLPR3 g_pdmR3DevIoApicHlp =
512{
513 PDM_IOAPICHLPR3_VERSION,
514 pdmR3IoApicHlp_ApicBusDeliver,
515 pdmR3IoApicHlp_Lock,
516 pdmR3IoApicHlp_Unlock,
517 pdmR3IoApicHlp_GetRCHelpers,
518 pdmR3IoApicHlp_GetR0Helpers,
519 PDM_IOAPICHLPR3_VERSION /* the end */
520};
521
522/** @} */
523
524
525
526
527/** @name Ring-3 PCI Bus Helpers
528 * @{
529 */
530
531/** @interface_method_impl{PDMPCIHLPR3,pfnIsaSetIrq} */
532static DECLCALLBACK(void) pdmR3PciHlp_IsaSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc)
533{
534 PDMDEV_ASSERT_DEVINS(pDevIns);
535 Log4(("pdmR3PciHlp_IsaSetIrq: iIrq=%d iLevel=%d uTagSrc=%#x\n", iIrq, iLevel, uTagSrc));
536 PVM pVM = pDevIns->Internal.s.pVMR3;
537 PDMIsaSetIrq(pDevIns->Internal.s.pVMR3, iIrq, iLevel, uTagSrc);
538}
539
540/** @interface_method_impl{PDMPCIHLPR3,pfnIoApicSetIrq} */
541static DECLCALLBACK(void) pdmR3PciHlp_IoApicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel, uint32_t uTagSrc)
542{
543 PDMDEV_ASSERT_DEVINS(pDevIns);
544 Log4(("pdmR3PciHlp_IoApicSetIrq: iIrq=%d iLevel=%d uTagSrc=%#x\n", iIrq, iLevel, uTagSrc));
545 PDMIoApicSetIrq(pDevIns->Internal.s.pVMR3, iIrq, iLevel, uTagSrc);
546}
547
548/** @interface_method_impl{PDMPCIHLPR3,pfnIoApicSendMsi} */
549static DECLCALLBACK(void) pdmR3PciHlp_IoApicSendMsi(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, uint32_t uValue, uint32_t uTagSrc)
550{
551 PDMDEV_ASSERT_DEVINS(pDevIns);
552 Log4(("pdmR3PciHlp_IoApicSendMsi: address=%p value=%x uTagSrc=%#x\n", GCPhys, uValue, uTagSrc));
553 PDMIoApicSendMsi(pDevIns->Internal.s.pVMR3, GCPhys, uValue, uTagSrc);
554}
555
556/** @interface_method_impl{PDMPCIHLPR3,pfnIsMMIO2Base} */
557static DECLCALLBACK(bool) pdmR3PciHlp_IsMMIO2Base(PPDMDEVINS pDevIns, PPDMDEVINS pOwner, RTGCPHYS GCPhys)
558{
559 PDMDEV_ASSERT_DEVINS(pDevIns);
560 VM_ASSERT_EMT(pDevIns->Internal.s.pVMR3);
561 bool fRc = PGMR3PhysMMIO2IsBase(pDevIns->Internal.s.pVMR3, pOwner, GCPhys);
562 Log4(("pdmR3PciHlp_IsMMIO2Base: pOwner=%p GCPhys=%RGp -> %RTbool\n", pOwner, GCPhys, fRc));
563 return fRc;
564}
565
566
567/** @interface_method_impl{PDMPCIHLPR3,pfnLock} */
568static DECLCALLBACK(int) pdmR3PciHlp_Lock(PPDMDEVINS pDevIns, int rc)
569{
570 PDMDEV_ASSERT_DEVINS(pDevIns);
571 LogFlow(("pdmR3PciHlp_Lock: caller='%s'/%d: rc=%Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
572 return pdmLockEx(pDevIns->Internal.s.pVMR3, rc);
573}
574
575
576/** @interface_method_impl{PDMPCIHLPR3,pfnUnlock} */
577static DECLCALLBACK(void) pdmR3PciHlp_Unlock(PPDMDEVINS pDevIns)
578{
579 PDMDEV_ASSERT_DEVINS(pDevIns);
580 LogFlow(("pdmR3PciHlp_Unlock: caller='%s'/%d:\n", pDevIns->pReg->szName, pDevIns->iInstance));
581 pdmUnlock(pDevIns->Internal.s.pVMR3);
582}
583
584
585/** @interface_method_impl{PDMPCIHLPR3,pfnGetRCHelpers} */
586static DECLCALLBACK(PCPDMPCIHLPRC) pdmR3PciHlp_GetRCHelpers(PPDMDEVINS pDevIns)
587{
588 PDMDEV_ASSERT_DEVINS(pDevIns);
589 PVM pVM = pDevIns->Internal.s.pVMR3;
590 VM_ASSERT_EMT(pVM);
591
592 RTRCPTR pRCHelpers = NIL_RTRCPTR;
593 if (!HMIsEnabled(pVM))
594 {
595 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCPciHlp", &pRCHelpers);
596 AssertReleaseRC(rc);
597 AssertRelease(pRCHelpers);
598 }
599
600 LogFlow(("pdmR3IoApicHlp_GetGCHelpers: caller='%s'/%d: returns %RRv\n",
601 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
602 return pRCHelpers;
603}
604
605
606/** @interface_method_impl{PDMPCIHLPR3,pfnGetR0Helpers} */
607static DECLCALLBACK(PCPDMPCIHLPR0) pdmR3PciHlp_GetR0Helpers(PPDMDEVINS pDevIns)
608{
609 PDMDEV_ASSERT_DEVINS(pDevIns);
610 PVM pVM = pDevIns->Internal.s.pVMR3;
611 VM_ASSERT_EMT(pVM);
612 PCPDMPCIHLPR0 pR0Helpers = 0;
613 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0PciHlp", &pR0Helpers);
614 AssertReleaseRC(rc);
615 AssertRelease(pR0Helpers);
616 LogFlow(("pdmR3IoApicHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
617 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
618 return pR0Helpers;
619}
620
621
622/**
623 * PCI Bus Device Helpers.
624 */
625const PDMPCIHLPR3 g_pdmR3DevPciHlp =
626{
627 PDM_PCIHLPR3_VERSION,
628 pdmR3PciHlp_IsaSetIrq,
629 pdmR3PciHlp_IoApicSetIrq,
630 pdmR3PciHlp_IoApicSendMsi,
631 pdmR3PciHlp_IsMMIO2Base,
632 pdmR3PciHlp_GetRCHelpers,
633 pdmR3PciHlp_GetR0Helpers,
634 pdmR3PciHlp_Lock,
635 pdmR3PciHlp_Unlock,
636 PDM_PCIHLPR3_VERSION, /* the end */
637};
638
639/** @} */
640
641
642
643
644/** @name Ring-3 HPET Helpers
645 * {@
646 */
647
648/** @interface_method_impl{PDMHPETHLPR3,pfnSetLegacyMode} */
649static DECLCALLBACK(int) pdmR3HpetHlp_SetLegacyMode(PPDMDEVINS pDevIns, bool fActivated)
650{
651 PDMDEV_ASSERT_DEVINS(pDevIns);
652 LogFlow(("pdmR3HpetHlp_SetLegacyMode: caller='%s'/%d: fActivated=%RTbool\n", pDevIns->pReg->szName, pDevIns->iInstance, fActivated));
653
654 size_t i;
655 int rc = VINF_SUCCESS;
656 static const char * const s_apszDevsToNotify[] =
657 {
658 "i8254",
659 "mc146818"
660 };
661 for (i = 0; i < RT_ELEMENTS(s_apszDevsToNotify); i++)
662 {
663 PPDMIBASE pBase;
664 rc = PDMR3QueryDevice(pDevIns->Internal.s.pVMR3->pUVM, "i8254", 0, &pBase);
665 if (RT_SUCCESS(rc))
666 {
667 PPDMIHPETLEGACYNOTIFY pPort = PDMIBASE_QUERY_INTERFACE(pBase, PDMIHPETLEGACYNOTIFY);
668 AssertLogRelMsgBreakStmt(pPort, ("%s\n", s_apszDevsToNotify[i]), rc = VERR_PDM_HPET_LEGACY_NOTIFY_MISSING);
669 pPort->pfnModeChanged(pPort, fActivated);
670 }
671 else if ( rc == VERR_PDM_DEVICE_NOT_FOUND
672 || rc == VERR_PDM_DEVICE_INSTANCE_NOT_FOUND)
673 rc = VINF_SUCCESS; /* the device isn't configured, ignore. */
674 else
675 AssertLogRelMsgFailedBreak(("%s -> %Rrc\n", s_apszDevsToNotify[i], rc));
676 }
677
678 /* Don't bother cleaning up, any failure here will cause a guru meditation. */
679
680 LogFlow(("pdmR3HpetHlp_SetLegacyMode: caller='%s'/%d: returns %Rrc\n", pDevIns->pReg->szName, pDevIns->iInstance, rc));
681 return rc;
682}
683
684
685/** @interface_method_impl{PDMHPETHLPR3,pfnSetIrq} */
686static DECLCALLBACK(int) pdmR3HpetHlp_SetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
687{
688 PDMDEV_ASSERT_DEVINS(pDevIns);
689 LogFlow(("pdmR3HpetHlp_SetIrq: caller='%s'/%d: iIrq=%d iLevel=%d\n", pDevIns->pReg->szName, pDevIns->iInstance, iIrq, iLevel));
690 PVM pVM = pDevIns->Internal.s.pVMR3;
691
692 pdmLock(pVM);
693 uint32_t uTagSrc;
694 if (iLevel & PDM_IRQ_LEVEL_HIGH)
695 {
696 pDevIns->Internal.s.uLastIrqTag = uTagSrc = pdmCalcIrqTag(pVM, pDevIns->idTracing);
697 if (iLevel == PDM_IRQ_LEVEL_HIGH)
698 VBOXVMM_PDM_IRQ_HIGH(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
699 else
700 VBOXVMM_PDM_IRQ_HILO(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
701 }
702 else
703 uTagSrc = pDevIns->Internal.s.uLastIrqTag;
704
705 PDMIsaSetIrq(pVM, iIrq, iLevel, uTagSrc); /* (The API takes the lock recursively.) */
706
707 if (iLevel == PDM_IRQ_LEVEL_LOW)
708 VBOXVMM_PDM_IRQ_LOW(VMMGetCpu(pVM), RT_LOWORD(uTagSrc), RT_HIWORD(uTagSrc));
709 pdmUnlock(pVM);
710 return 0;
711}
712
713
714/** @interface_method_impl{PDMHPETHLPR3,pfnGetRCHelpers} */
715static DECLCALLBACK(PCPDMHPETHLPRC) pdmR3HpetHlp_GetRCHelpers(PPDMDEVINS pDevIns)
716{
717 PDMDEV_ASSERT_DEVINS(pDevIns);
718 PVM pVM = pDevIns->Internal.s.pVMR3;
719 VM_ASSERT_EMT(pVM);
720
721 RTRCPTR pRCHelpers = NIL_RTRCPTR;
722 if (!HMIsEnabled(pVM))
723 {
724 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCHpetHlp", &pRCHelpers);
725 AssertReleaseRC(rc);
726 AssertRelease(pRCHelpers);
727 }
728
729 LogFlow(("pdmR3HpetHlp_GetGCHelpers: caller='%s'/%d: returns %RRv\n",
730 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
731 return pRCHelpers;
732}
733
734
735/** @interface_method_impl{PDMHPETHLPR3,pfnGetR0Helpers} */
736static DECLCALLBACK(PCPDMHPETHLPR0) pdmR3HpetHlp_GetR0Helpers(PPDMDEVINS pDevIns)
737{
738 PDMDEV_ASSERT_DEVINS(pDevIns);
739 PVM pVM = pDevIns->Internal.s.pVMR3;
740 VM_ASSERT_EMT(pVM);
741 PCPDMHPETHLPR0 pR0Helpers = 0;
742 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0HpetHlp", &pR0Helpers);
743 AssertReleaseRC(rc);
744 AssertRelease(pR0Helpers);
745 LogFlow(("pdmR3HpetHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
746 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
747 return pR0Helpers;
748}
749
750
751/**
752 * HPET Device Helpers.
753 */
754const PDMHPETHLPR3 g_pdmR3DevHpetHlp =
755{
756 PDM_HPETHLPR3_VERSION,
757 pdmR3HpetHlp_GetRCHelpers,
758 pdmR3HpetHlp_GetR0Helpers,
759 pdmR3HpetHlp_SetLegacyMode,
760 pdmR3HpetHlp_SetIrq,
761 PDM_HPETHLPR3_VERSION, /* the end */
762};
763
764/** @} */
765
766
767/** @name Ring-3 Raw PCI Device Helpers
768 * {@
769 */
770
771/** @interface_method_impl{PDMPCIRAWHLPR3,pfnGetRCHelpers} */
772static DECLCALLBACK(PCPDMPCIRAWHLPRC) pdmR3PciRawHlp_GetRCHelpers(PPDMDEVINS pDevIns)
773{
774 PDMDEV_ASSERT_DEVINS(pDevIns);
775 PVM pVM = pDevIns->Internal.s.pVMR3;
776 VM_ASSERT_EMT(pVM);
777
778 RTRCPTR pRCHelpers = NIL_RTRCPTR;
779 if (!HMIsEnabled(pVM))
780 {
781 int rc = PDMR3LdrGetSymbolRC(pVM, NULL, "g_pdmRCPciRawHlp", &pRCHelpers);
782 AssertReleaseRC(rc);
783 AssertRelease(pRCHelpers);
784 }
785
786 LogFlow(("pdmR3PciRawHlp_GetGCHelpers: caller='%s'/%d: returns %RRv\n",
787 pDevIns->pReg->szName, pDevIns->iInstance, pRCHelpers));
788 return pRCHelpers;
789}
790
791
792/** @interface_method_impl{PDMPCIRAWHLPR3,pfnGetR0Helpers} */
793static DECLCALLBACK(PCPDMPCIRAWHLPR0) pdmR3PciRawHlp_GetR0Helpers(PPDMDEVINS pDevIns)
794{
795 PDMDEV_ASSERT_DEVINS(pDevIns);
796 PVM pVM = pDevIns->Internal.s.pVMR3;
797 VM_ASSERT_EMT(pVM);
798 PCPDMHPETHLPR0 pR0Helpers = NIL_RTR0PTR;
799 int rc = PDMR3LdrGetSymbolR0(pVM, NULL, "g_pdmR0PciRawHlp", &pR0Helpers);
800 AssertReleaseRC(rc);
801 AssertRelease(pR0Helpers);
802 LogFlow(("pdmR3PciRawHlp_GetR0Helpers: caller='%s'/%d: returns %RHv\n",
803 pDevIns->pReg->szName, pDevIns->iInstance, pR0Helpers));
804 return pR0Helpers;
805}
806
807
808/**
809 * Raw PCI Device Helpers.
810 */
811const PDMPCIRAWHLPR3 g_pdmR3DevPciRawHlp =
812{
813 PDM_PCIRAWHLPR3_VERSION,
814 pdmR3PciRawHlp_GetRCHelpers,
815 pdmR3PciRawHlp_GetR0Helpers,
816 PDM_PCIRAWHLPR3_VERSION, /* the end */
817};
818
819/** @} */
820
821
822/* none yet */
823
824/**
825 * Firmware Device Helpers.
826 */
827const PDMFWHLPR3 g_pdmR3DevFirmwareHlp =
828{
829 PDM_FWHLPR3_VERSION,
830 PDM_FWHLPR3_VERSION
831};
832
833/**
834 * DMAC Device Helpers.
835 */
836const PDMDMACHLP g_pdmR3DevDmacHlp =
837{
838 PDM_DMACHLP_VERSION
839};
840
841
842
843
844/* none yet */
845
846/**
847 * RTC Device Helpers.
848 */
849const PDMRTCHLP g_pdmR3DevRtcHlp =
850{
851 PDM_RTCHLP_VERSION
852};
853
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