VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMRC/PDMRCDevice.cpp@ 36261

Last change on this file since 36261 was 35738, checked in by vboxsync, 14 years ago

r=bird: Clean up of r69600 (PCIRAW), left a few review notes behind.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 26.3 KB
Line 
1/* $Id: PDMRCDevice.cpp 35738 2011-01-27 14:17:41Z vboxsync $ */
2/** @file
3 * PDM - Pluggable Device and Driver Manager, RC Device parts.
4 */
5
6/*
7 * Copyright (C) 2006-2011 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/mm.h>
27#include <VBox/vmm/vm.h>
28#include <VBox/vmm/vmm.h>
29#include <VBox/vmm/patm.h>
30
31#include <VBox/log.h>
32#include <VBox/err.h>
33#include <iprt/asm.h>
34#include <iprt/assert.h>
35#include <iprt/string.h>
36
37
38/*******************************************************************************
39* Global Variables *
40*******************************************************************************/
41RT_C_DECLS_BEGIN
42extern DECLEXPORT(const PDMDEVHLPRC) g_pdmRCDevHlp;
43extern DECLEXPORT(const PDMPICHLPRC) g_pdmRCPicHlp;
44extern DECLEXPORT(const PDMAPICHLPRC) g_pdmRCApicHlp;
45extern DECLEXPORT(const PDMIOAPICHLPRC) g_pdmRCIoApicHlp;
46extern DECLEXPORT(const PDMPCIHLPRC) g_pdmRCPciHlp;
47extern DECLEXPORT(const PDMHPETHLPRC) g_pdmRCHpetHlp;
48extern DECLEXPORT(const PDMDRVHLPRC) g_pdmRCDrvHlp;
49/** @todo missing PDMPCIRAWHLPRC */
50RT_C_DECLS_END
51
52
53/*******************************************************************************
54* Internal Functions *
55*******************************************************************************/
56static void pdmRCIsaSetIrq(PVM pVM, int iIrq, int iLevel);
57static void pdmRCIoApicSetIrq(PVM pVM, int iIrq, int iLevel);
58static void pdmRCIoApicSendMsi(PVM pVM, RTGCPHYS GCAddr, uint32_t uValue);
59
60/** @name Raw-Mode Context Device Helpers
61 * @{
62 */
63
64/** @interface_method_impl{PDMDEVHLPRC,pfnPCISetIrq} */
65static DECLCALLBACK(void) pdmRCDevHlp_PCISetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
66{
67 PDMDEV_ASSERT_DEVINS(pDevIns);
68 LogFlow(("pdmRCDevHlp_PCISetIrq: caller=%p/%d: iIrq=%d iLevel=%d\n", pDevIns, pDevIns->iInstance, iIrq, iLevel));
69
70 PVM pVM = pDevIns->Internal.s.pVMRC;
71 PPCIDEVICE pPciDev = pDevIns->Internal.s.pPciDeviceRC;
72 PPDMPCIBUS pPciBus = pDevIns->Internal.s.pPciBusRC;
73 if ( pPciDev
74 && pPciBus
75 && pPciBus->pDevInsRC)
76 {
77 pdmLock(pVM);
78 pPciBus->pfnSetIrqRC(pPciBus->pDevInsRC, pPciDev, iIrq, iLevel);
79 pdmUnlock(pVM);
80 }
81 else
82 {
83 /* queue for ring-3 execution. */
84 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)PDMQueueAlloc(pVM->pdm.s.pDevHlpQueueRC);
85 if (pTask)
86 {
87 pTask->enmOp = PDMDEVHLPTASKOP_PCI_SET_IRQ;
88 pTask->pDevInsR3 = PDMDEVINS_2_R3PTR(pDevIns);
89 pTask->u.SetIRQ.iIrq = iIrq;
90 pTask->u.SetIRQ.iLevel = iLevel;
91
92 PDMQueueInsertEx(pVM->pdm.s.pDevHlpQueueRC, &pTask->Core, 0);
93 }
94 else
95 AssertMsgFailed(("We're out of devhlp queue items!!!\n"));
96 }
97
98 LogFlow(("pdmRCDevHlp_PCISetIrq: caller=%p/%d: returns void\n", pDevIns, pDevIns->iInstance));
99}
100
101
102/** @interface_method_impl{PDMDRVHLPRC,pfnPCISetIrq} */
103static DECLCALLBACK(void) pdmRCDevHlp_ISASetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
104{
105 PDMDEV_ASSERT_DEVINS(pDevIns);
106 LogFlow(("pdmRCDevHlp_ISASetIrq: caller=%p/%d: iIrq=%d iLevel=%d\n", pDevIns, pDevIns->iInstance, iIrq, iLevel));
107
108 pdmRCIsaSetIrq(pDevIns->Internal.s.pVMRC, iIrq, iLevel);
109
110 LogFlow(("pdmRCDevHlp_ISASetIrq: caller=%p/%d: returns void\n", pDevIns, pDevIns->iInstance));
111}
112
113
114/** @interface_method_impl{PDMDEVHLPRC,pfnPhysRead} */
115static DECLCALLBACK(int) pdmRCDevHlp_PhysRead(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, void *pvBuf, size_t cbRead)
116{
117 PDMDEV_ASSERT_DEVINS(pDevIns);
118 LogFlow(("pdmRCDevHlp_PhysRead: caller=%p/%d: GCPhys=%RGp pvBuf=%p cbRead=%#x\n",
119 pDevIns, pDevIns->iInstance, GCPhys, pvBuf, cbRead));
120
121 int rc = PGMPhysRead(pDevIns->Internal.s.pVMRC, GCPhys, pvBuf, cbRead);
122 AssertRC(rc); /** @todo track down the users for this bugger. */
123
124 Log(("pdmRCDevHlp_PhysRead: caller=%p/%d: returns %Rrc\n", pDevIns, pDevIns->iInstance, rc));
125 return rc;
126}
127
128
129/** @interface_method_impl{PDMDEVHLPRC,pfnPhysWrite} */
130static DECLCALLBACK(int) pdmRCDevHlp_PhysWrite(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, const void *pvBuf, size_t cbWrite)
131{
132 PDMDEV_ASSERT_DEVINS(pDevIns);
133 LogFlow(("pdmRCDevHlp_PhysWrite: caller=%p/%d: GCPhys=%RGp pvBuf=%p cbWrite=%#x\n",
134 pDevIns, pDevIns->iInstance, GCPhys, pvBuf, cbWrite));
135
136 int rc = PGMPhysWrite(pDevIns->Internal.s.pVMRC, GCPhys, pvBuf, cbWrite);
137 AssertRC(rc); /** @todo track down the users for this bugger. */
138
139 Log(("pdmRCDevHlp_PhysWrite: caller=%p/%d: returns %Rrc\n", pDevIns, pDevIns->iInstance, rc));
140 return rc;
141}
142
143
144/** @interface_method_impl{PDMDEVHLPRC,pfnA20IsEnabled} */
145static DECLCALLBACK(bool) pdmRCDevHlp_A20IsEnabled(PPDMDEVINS pDevIns)
146{
147 PDMDEV_ASSERT_DEVINS(pDevIns);
148 LogFlow(("pdmRCDevHlp_A20IsEnabled: caller=%p/%d:\n", pDevIns, pDevIns->iInstance));
149
150 bool fEnabled = PGMPhysIsA20Enabled(VMMGetCpu0(pDevIns->Internal.s.pVMRC));
151
152 Log(("pdmRCDevHlp_A20IsEnabled: caller=%p/%d: returns %RTbool\n", pDevIns, pDevIns->iInstance, fEnabled));
153 return fEnabled;
154}
155
156
157/** @interface_method_impl{PDMDEVHLPRC,pfnVMState} */
158static DECLCALLBACK(VMSTATE) pdmRCDevHlp_VMState(PPDMDEVINS pDevIns)
159{
160 PDMDEV_ASSERT_DEVINS(pDevIns);
161
162 VMSTATE enmVMState = pDevIns->Internal.s.pVMRC->enmVMState;
163
164 LogFlow(("pdmRCDevHlp_VMState: caller=%p/%d: returns %d\n", pDevIns, pDevIns->iInstance, enmVMState));
165 return enmVMState;
166}
167
168
169/** @interface_method_impl{PDMDEVHLPRC,pfnVMSetError} */
170static DECLCALLBACK(int) pdmRCDevHlp_VMSetError(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
171{
172 PDMDEV_ASSERT_DEVINS(pDevIns);
173 va_list args;
174 va_start(args, pszFormat);
175 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMRC, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc2 == rc); NOREF(rc2);
176 va_end(args);
177 return rc;
178}
179
180
181/** @interface_method_impl{PDMDEVHLPRC,pfnVMSetErrorV} */
182static DECLCALLBACK(int) pdmRCDevHlp_VMSetErrorV(PPDMDEVINS pDevIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
183{
184 PDMDEV_ASSERT_DEVINS(pDevIns);
185 int rc2 = VMSetErrorV(pDevIns->Internal.s.pVMRC, rc, RT_SRC_POS_ARGS, pszFormat, va); Assert(rc2 == rc); NOREF(rc2);
186 return rc;
187}
188
189
190/** @interface_method_impl{PDMDEVHLPRC,pfnVMSetRuntimeError} */
191static DECLCALLBACK(int) pdmRCDevHlp_VMSetRuntimeError(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
192{
193 PDMDEV_ASSERT_DEVINS(pDevIns);
194 va_list va;
195 va_start(va, pszFormat);
196 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMRC, fFlags, pszErrorId, pszFormat, va);
197 va_end(va);
198 return rc;
199}
200
201
202/** @interface_method_impl{PDMDEVHLPRC,pfnVMSetErrorV} */
203static DECLCALLBACK(int) pdmRCDevHlp_VMSetRuntimeErrorV(PPDMDEVINS pDevIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
204{
205 PDMDEV_ASSERT_DEVINS(pDevIns);
206 int rc = VMSetRuntimeErrorV(pDevIns->Internal.s.pVMRC, fFlags, pszErrorId, pszFormat, va);
207 return rc;
208}
209
210
211/** @interface_method_impl{PDMDEVHLPRC,pfnPATMSetMMIOPatchInfo} */
212static DECLCALLBACK(int) pdmRCDevHlp_PATMSetMMIOPatchInfo(PPDMDEVINS pDevIns, RTGCPHYS GCPhys, RTGCPTR pCachedData)
213{
214 PDMDEV_ASSERT_DEVINS(pDevIns);
215 LogFlow(("pdmRCDevHlp_PATMSetMMIOPatchInfo: caller=%p/%d:\n", pDevIns, pDevIns->iInstance));
216
217 return PATMSetMMIOPatchInfo(pDevIns->Internal.s.pVMRC, GCPhys, (RTRCPTR)(uintptr_t)pCachedData);
218}
219
220
221/** @interface_method_impl{PDMDEVHLPRC,pfnGetVM} */
222static DECLCALLBACK(PVM) pdmRCDevHlp_GetVM(PPDMDEVINS pDevIns)
223{
224 PDMDEV_ASSERT_DEVINS(pDevIns);
225 LogFlow(("pdmRCDevHlp_GetVM: caller='%p'/%d\n", pDevIns, pDevIns->iInstance));
226 return pDevIns->Internal.s.pVMRC;
227}
228
229
230/** @interface_method_impl{PDMDEVHLPRC,pfnGetVMCPU} */
231static DECLCALLBACK(PVMCPU) pdmRCDevHlp_GetVMCPU(PPDMDEVINS pDevIns)
232{
233 PDMDEV_ASSERT_DEVINS(pDevIns);
234 LogFlow(("pdmRCDevHlp_GetVMCPU: caller='%p'/%d\n", pDevIns, pDevIns->iInstance));
235 return VMMGetCpu(pDevIns->Internal.s.pVMRC);
236}
237
238
239/** @interface_method_impl{PDMDEVHLPRC,pfnTMTimeVirtGet} */
240static DECLCALLBACK(uint64_t) pdmRCDevHlp_TMTimeVirtGet(PPDMDEVINS pDevIns)
241{
242 PDMDEV_ASSERT_DEVINS(pDevIns);
243 LogFlow(("pdmRCDevHlp_TMTimeVirtGet: caller='%p'/%d\n", pDevIns, pDevIns->iInstance));
244 return TMVirtualGet(pDevIns->Internal.s.pVMRC);
245}
246
247
248/** @interface_method_impl{PDMDEVHLPRC,pfnTMTimeVirtGetFreq} */
249static DECLCALLBACK(uint64_t) pdmRCDevHlp_TMTimeVirtGetFreq(PPDMDEVINS pDevIns)
250{
251 PDMDEV_ASSERT_DEVINS(pDevIns);
252 LogFlow(("pdmRCDevHlp_TMTimeVirtGetFreq: caller='%p'/%d\n", pDevIns, pDevIns->iInstance));
253 return TMVirtualGetFreq(pDevIns->Internal.s.pVMRC);
254}
255
256
257/** @interface_method_impl{PDMDEVHLPRC,pfnTMTimeVirtGetNano} */
258static DECLCALLBACK(uint64_t) pdmRCDevHlp_TMTimeVirtGetNano(PPDMDEVINS pDevIns)
259{
260 PDMDEV_ASSERT_DEVINS(pDevIns);
261 LogFlow(("pdmRCDevHlp_TMTimeVirtGetNano: caller='%p'/%d\n", pDevIns, pDevIns->iInstance));
262 return TMVirtualToNano(pDevIns->Internal.s.pVMRC, TMVirtualGet(pDevIns->Internal.s.pVMRC));
263}
264
265
266/**
267 * The Raw-Mode Context Device Helper Callbacks.
268 */
269extern DECLEXPORT(const PDMDEVHLPRC) g_pdmRCDevHlp =
270{
271 PDM_DEVHLPRC_VERSION,
272 pdmRCDevHlp_PCISetIrq,
273 pdmRCDevHlp_ISASetIrq,
274 pdmRCDevHlp_PhysRead,
275 pdmRCDevHlp_PhysWrite,
276 pdmRCDevHlp_A20IsEnabled,
277 pdmRCDevHlp_VMState,
278 pdmRCDevHlp_VMSetError,
279 pdmRCDevHlp_VMSetErrorV,
280 pdmRCDevHlp_VMSetRuntimeError,
281 pdmRCDevHlp_VMSetRuntimeErrorV,
282 pdmRCDevHlp_PATMSetMMIOPatchInfo,
283 pdmRCDevHlp_GetVM,
284 pdmRCDevHlp_GetVMCPU,
285 pdmRCDevHlp_TMTimeVirtGet,
286 pdmRCDevHlp_TMTimeVirtGetFreq,
287 pdmRCDevHlp_TMTimeVirtGetNano,
288 PDM_DEVHLPRC_VERSION
289};
290
291/** @} */
292
293
294
295
296/** @name PIC RC Helpers
297 * @{
298 */
299
300/** @interface_method_impl{PDMPICHLPGC,pfnSetInterruptFF} */
301static DECLCALLBACK(void) pdmRCPicHlp_SetInterruptFF(PPDMDEVINS pDevIns)
302{
303 PDMDEV_ASSERT_DEVINS(pDevIns);
304 PVM pVM = pDevIns->Internal.s.pVMRC;
305
306 if (pVM->pdm.s.Apic.pfnLocalInterruptRC)
307 {
308 LogFlow(("pdmRCPicHlp_SetInterruptFF: caller='%p'/%d: Setting local interrupt on LAPIC\n",
309 pDevIns, pDevIns->iInstance));
310 /* Raise the LAPIC's LINT0 line instead of signaling the CPU directly. */
311 pVM->pdm.s.Apic.pfnLocalInterruptRC(pVM->pdm.s.Apic.pDevInsRC, 0, 1);
312 return;
313 }
314
315 PVMCPU pVCpu = &pVM->aCpus[0]; /* for PIC we always deliver to CPU 0, MP use APIC */
316
317 LogFlow(("pdmRCPicHlp_SetInterruptFF: caller=%p/%d: VMMCPU_FF_INTERRUPT_PIC %d -> 1\n",
318 pDevIns, pDevIns->iInstance, VMCPU_FF_ISSET(pVCpu, VMCPU_FF_INTERRUPT_PIC)));
319
320 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC);
321}
322
323
324/** @interface_method_impl{PDMPICHLPGC,pfnClearInterruptFF} */
325static DECLCALLBACK(void) pdmRCPicHlp_ClearInterruptFF(PPDMDEVINS pDevIns)
326{
327 PDMDEV_ASSERT_DEVINS(pDevIns);
328 PVM pVM = pDevIns->Internal.s.CTX_SUFF(pVM);
329
330 if (pVM->pdm.s.Apic.pfnLocalInterruptRC)
331 {
332 /* Raise the LAPIC's LINT0 line instead of signaling the CPU directly. */
333 LogFlow(("pdmRCPicHlp_ClearInterruptFF: caller='%s'/%d: Clearing local interrupt on LAPIC\n",
334 pDevIns, pDevIns->iInstance));
335 /* Lower the LAPIC's LINT0 line instead of signaling the CPU directly. */
336 pVM->pdm.s.Apic.pfnLocalInterruptRC(pVM->pdm.s.Apic.pDevInsRC, 0, 0);
337 return;
338 }
339
340 PVMCPU pVCpu = &pVM->aCpus[0]; /* for PIC we always deliver to CPU 0, MP use APIC */
341
342 LogFlow(("pdmRCPicHlp_ClearInterruptFF: caller=%p/%d: VMCPU_FF_INTERRUPT_PIC %d -> 0\n",
343 pDevIns, pDevIns->iInstance, VMCPU_FF_ISSET(pVCpu, VMCPU_FF_INTERRUPT_PIC)));
344
345 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_PIC);
346}
347
348
349/** @interface_method_impl{PDMPICHLPGC,pfnLock} */
350static DECLCALLBACK(int) pdmRCPicHlp_Lock(PPDMDEVINS pDevIns, int rc)
351{
352 PDMDEV_ASSERT_DEVINS(pDevIns);
353 return pdmLockEx(pDevIns->Internal.s.pVMRC, rc);
354}
355
356
357/** @interface_method_impl{PDMPICHLPGC,pfnUnlock} */
358static DECLCALLBACK(void) pdmRCPicHlp_Unlock(PPDMDEVINS pDevIns)
359{
360 PDMDEV_ASSERT_DEVINS(pDevIns);
361 pdmUnlock(pDevIns->Internal.s.pVMRC);
362}
363
364
365/**
366 * The Raw-Mode Context PIC Helper Callbacks.
367 */
368extern DECLEXPORT(const PDMPICHLPRC) g_pdmRCPicHlp =
369{
370 PDM_PICHLPRC_VERSION,
371 pdmRCPicHlp_SetInterruptFF,
372 pdmRCPicHlp_ClearInterruptFF,
373 pdmRCPicHlp_Lock,
374 pdmRCPicHlp_Unlock,
375 PDM_PICHLPRC_VERSION
376};
377
378/** @} */
379
380
381
382
383/** @name APIC RC Helpers
384 * @{
385 */
386
387/** @interface_method_impl{PDMAPICHLPRC,pfnSetInterruptFF} */
388static DECLCALLBACK(void) pdmRCApicHlp_SetInterruptFF(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu)
389{
390 PDMDEV_ASSERT_DEVINS(pDevIns);
391 PVM pVM = pDevIns->Internal.s.pVMRC;
392 PVMCPU pVCpu = &pVM->aCpus[idCpu];
393
394 AssertReturnVoid(idCpu < pVM->cCpus);
395
396 LogFlow(("pdmRCApicHlp_SetInterruptFF: caller=%p/%d: VM_FF_INTERRUPT %d -> 1\n",
397 pDevIns, pDevIns->iInstance, VMCPU_FF_ISSET(pVCpu, VMCPU_FF_INTERRUPT_APIC)));
398 switch (enmType)
399 {
400 case PDMAPICIRQ_HARDWARE:
401 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_APIC);
402 break;
403 case PDMAPICIRQ_NMI:
404 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_NMI);
405 break;
406 case PDMAPICIRQ_SMI:
407 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_SMI);
408 break;
409 case PDMAPICIRQ_EXTINT:
410 VMCPU_FF_SET(pVCpu, VMCPU_FF_INTERRUPT_PIC);
411 break;
412 default:
413 AssertMsgFailed(("enmType=%d\n", enmType));
414 break;
415 }
416}
417
418
419/** @interface_method_impl{PDMAPICHLPRC,pfnClearInterruptFF} */
420static DECLCALLBACK(void) pdmRCApicHlp_ClearInterruptFF(PPDMDEVINS pDevIns, PDMAPICIRQ enmType, VMCPUID idCpu)
421{
422 PDMDEV_ASSERT_DEVINS(pDevIns);
423 PVM pVM = pDevIns->Internal.s.pVMRC;
424 PVMCPU pVCpu = &pVM->aCpus[idCpu];
425
426 AssertReturnVoid(idCpu < pVM->cCpus);
427
428 LogFlow(("pdmRCApicHlp_ClearInterruptFF: caller=%p/%d: VM_FF_INTERRUPT %d -> 0\n",
429 pDevIns, pDevIns->iInstance, VMCPU_FF_ISSET(pVCpu, VMCPU_FF_INTERRUPT_APIC)));
430
431 /* Note: NMI/SMI can't be cleared. */
432 switch (enmType)
433 {
434 case PDMAPICIRQ_HARDWARE:
435 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_APIC);
436 break;
437 case PDMAPICIRQ_EXTINT:
438 VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_PIC);
439 break;
440 default:
441 AssertMsgFailed(("enmType=%d\n", enmType));
442 break;
443 }
444}
445
446
447/** @interface_method_impl{PDMAPICHLPRC,pfnChangeFeature} */
448static DECLCALLBACK(void) pdmRCApicHlp_ChangeFeature(PPDMDEVINS pDevIns, PDMAPICVERSION enmVersion)
449{
450 PDMDEV_ASSERT_DEVINS(pDevIns);
451 LogFlow(("pdmRCApicHlp_ChangeFeature: caller=%p/%d: version=%d\n", pDevIns, pDevIns->iInstance, (int)enmVersion));
452 switch (enmVersion)
453 {
454 case PDMAPICVERSION_NONE:
455 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_APIC);
456 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_X2APIC);
457 break;
458 case PDMAPICVERSION_APIC:
459 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_APIC);
460 CPUMClearGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_X2APIC);
461 break;
462 case PDMAPICVERSION_X2APIC:
463 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_X2APIC);
464 CPUMSetGuestCpuIdFeature(pDevIns->Internal.s.pVMRC, CPUMCPUIDFEATURE_APIC);
465 break;
466 default:
467 AssertMsgFailed(("Unknown APIC version: %d\n", (int)enmVersion));
468 }
469}
470
471
472/** @interface_method_impl{PDMAPICHLPRC,pfnLock} */
473static DECLCALLBACK(int) pdmRCApicHlp_Lock(PPDMDEVINS pDevIns, int rc)
474{
475 PDMDEV_ASSERT_DEVINS(pDevIns);
476 return pdmLockEx(pDevIns->Internal.s.pVMRC, rc);
477}
478
479
480/** @interface_method_impl{PDMAPICHLPRC,pfnUnlock} */
481static DECLCALLBACK(void) pdmRCApicHlp_Unlock(PPDMDEVINS pDevIns)
482{
483 PDMDEV_ASSERT_DEVINS(pDevIns);
484 pdmUnlock(pDevIns->Internal.s.pVMRC);
485}
486
487
488/** @interface_method_impl{PDMAPICHLPRC,pfnGetCpuId} */
489static DECLCALLBACK(VMCPUID) pdmRCApicHlp_GetCpuId(PPDMDEVINS pDevIns)
490{
491 PDMDEV_ASSERT_DEVINS(pDevIns);
492 return VMMGetCpuId(pDevIns->Internal.s.pVMRC);
493}
494
495
496/**
497 * The Raw-Mode Context APIC Helper Callbacks.
498 */
499extern DECLEXPORT(const PDMAPICHLPRC) g_pdmRCApicHlp =
500{
501 PDM_APICHLPRC_VERSION,
502 pdmRCApicHlp_SetInterruptFF,
503 pdmRCApicHlp_ClearInterruptFF,
504 pdmRCApicHlp_ChangeFeature,
505 pdmRCApicHlp_Lock,
506 pdmRCApicHlp_Unlock,
507 pdmRCApicHlp_GetCpuId,
508 PDM_APICHLPRC_VERSION
509};
510
511/** @} */
512
513
514
515
516/** @name I/O APIC RC Helpers
517 * @{
518 */
519
520/** @interface_method_impl{PDMIOAPICHLPRC,pfnApicBusDeliver} */
521static DECLCALLBACK(int) pdmRCIoApicHlp_ApicBusDeliver(PPDMDEVINS pDevIns, uint8_t u8Dest, uint8_t u8DestMode, uint8_t u8DeliveryMode,
522 uint8_t iVector, uint8_t u8Polarity, uint8_t u8TriggerMode)
523{
524 PDMDEV_ASSERT_DEVINS(pDevIns);
525 PVM pVM = pDevIns->Internal.s.pVMRC;
526 LogFlow(("pdmRCIoApicHlp_ApicBusDeliver: caller=%p/%d: u8Dest=%RX8 u8DestMode=%RX8 u8DeliveryMode=%RX8 iVector=%RX8 u8Polarity=%RX8 u8TriggerMode=%RX8\n",
527 pDevIns, pDevIns->iInstance, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode));
528 if (pVM->pdm.s.Apic.pfnBusDeliverRC)
529 return pVM->pdm.s.Apic.pfnBusDeliverRC(pVM->pdm.s.Apic.pDevInsRC, u8Dest, u8DestMode, u8DeliveryMode, iVector, u8Polarity, u8TriggerMode);
530 return VINF_SUCCESS;
531}
532
533
534/** @interface_method_impl{PDMIOAPICHLPRC,pfnLock} */
535static DECLCALLBACK(int) pdmRCIoApicHlp_Lock(PPDMDEVINS pDevIns, int rc)
536{
537 PDMDEV_ASSERT_DEVINS(pDevIns);
538 return pdmLockEx(pDevIns->Internal.s.pVMRC, rc);
539}
540
541
542/** @interface_method_impl{PDMIOAPICHLPRC,pfnUnlock} */
543static DECLCALLBACK(void) pdmRCIoApicHlp_Unlock(PPDMDEVINS pDevIns)
544{
545 PDMDEV_ASSERT_DEVINS(pDevIns);
546 pdmUnlock(pDevIns->Internal.s.pVMRC);
547}
548
549
550/**
551 * The Raw-Mode Context I/O APIC Helper Callbacks.
552 */
553extern DECLEXPORT(const PDMIOAPICHLPRC) g_pdmRCIoApicHlp =
554{
555 PDM_IOAPICHLPRC_VERSION,
556 pdmRCIoApicHlp_ApicBusDeliver,
557 pdmRCIoApicHlp_Lock,
558 pdmRCIoApicHlp_Unlock,
559 PDM_IOAPICHLPRC_VERSION
560};
561
562/** @} */
563
564
565
566
567/** @name PCI Bus RC Helpers
568 * @{
569 */
570
571/** @interface_method_impl{PDMPCIHLPRC,pfnIsaSetIrq} */
572static DECLCALLBACK(void) pdmRCPciHlp_IsaSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
573{
574 PDMDEV_ASSERT_DEVINS(pDevIns);
575 Log4(("pdmRCPciHlp_IsaSetIrq: iIrq=%d iLevel=%d\n", iIrq, iLevel));
576 pdmRCIsaSetIrq(pDevIns->Internal.s.pVMRC, iIrq, iLevel);
577}
578
579
580/** @interface_method_impl{PDMPCIHLPRC,pfnIoApicSetIrq} */
581static DECLCALLBACK(void) pdmRCPciHlp_IoApicSetIrq(PPDMDEVINS pDevIns, int iIrq, int iLevel)
582{
583 PDMDEV_ASSERT_DEVINS(pDevIns);
584 Log4(("pdmRCPciHlp_IoApicSetIrq: iIrq=%d iLevel=%d\n", iIrq, iLevel));
585 pdmRCIoApicSetIrq(pDevIns->Internal.s.pVMRC, iIrq, iLevel);
586}
587
588/** @interface_method_impl{PDMPCIHLPRC,pfnIoApicSendMsi} */
589static DECLCALLBACK(void) pdmRCPciHlp_IoApicSendMsi(PPDMDEVINS pDevIns, RTGCPHYS GCAddr, uint32_t uValue)
590{
591 PDMDEV_ASSERT_DEVINS(pDevIns);
592 Log4(("pdmRCPciHlp_IoApicSendMsi: Address=%p Value=%d\n", GCAddr, uValue));
593 pdmRCIoApicSendMsi(pDevIns->Internal.s.pVMRC, GCAddr, uValue);
594}
595
596
597/** @interface_method_impl{PDMPCIHLPRC,pfnLock} */
598static DECLCALLBACK(int) pdmRCPciHlp_Lock(PPDMDEVINS pDevIns, int rc)
599{
600 PDMDEV_ASSERT_DEVINS(pDevIns);
601 return pdmLockEx(pDevIns->Internal.s.pVMRC, rc);
602}
603
604
605/** @interface_method_impl{PDMPCIHLPRC,pfnUnlock} */
606static DECLCALLBACK(void) pdmRCPciHlp_Unlock(PPDMDEVINS pDevIns)
607{
608 PDMDEV_ASSERT_DEVINS(pDevIns);
609 pdmUnlock(pDevIns->Internal.s.pVMRC);
610}
611
612
613/**
614 * The Raw-Mode Context PCI Bus Helper Callbacks.
615 */
616extern DECLEXPORT(const PDMPCIHLPRC) g_pdmRCPciHlp =
617{
618 PDM_PCIHLPRC_VERSION,
619 pdmRCPciHlp_IsaSetIrq,
620 pdmRCPciHlp_IoApicSetIrq,
621 pdmRCPciHlp_IoApicSendMsi,
622 pdmRCPciHlp_Lock,
623 pdmRCPciHlp_Unlock,
624 PDM_PCIHLPRC_VERSION, /* the end */
625};
626
627/** @} */
628
629
630
631
632/** @name HPET RC Helpers
633 * @{
634 */
635
636
637/**
638 * The Raw-Mode Context HPET Helper Callbacks.
639 */
640extern DECLEXPORT(const PDMHPETHLPRC) g_pdmRCHpetHlp =
641{
642 PDM_HPETHLPRC_VERSION,
643 PDM_HPETHLPRC_VERSION, /* the end */
644};
645
646/** @} */
647
648
649
650
651/** @name Raw-Mode Context Driver Helpers
652 * @{
653 */
654
655/** @interface_method_impl{PDMDRVHLPRC,pfnVMSetError} */
656static DECLCALLBACK(int) pdmRCDrvHlp_VMSetError(PPDMDRVINS pDrvIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, ...)
657{
658 PDMDRV_ASSERT_DRVINS(pDrvIns);
659 va_list args;
660 va_start(args, pszFormat);
661 int rc2 = VMSetErrorV(pDrvIns->Internal.s.pVMRC, rc, RT_SRC_POS_ARGS, pszFormat, args); Assert(rc2 == rc); NOREF(rc2);
662 va_end(args);
663 return rc;
664}
665
666
667/** @interface_method_impl{PDMDRVHLPRC,pfnVMSetErrorV} */
668static DECLCALLBACK(int) pdmRCDrvHlp_VMSetErrorV(PPDMDRVINS pDrvIns, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
669{
670 PDMDRV_ASSERT_DRVINS(pDrvIns);
671 int rc2 = VMSetErrorV(pDrvIns->Internal.s.pVMRC, rc, RT_SRC_POS_ARGS, pszFormat, va); Assert(rc2 == rc); NOREF(rc2);
672 return rc;
673}
674
675
676/** @interface_method_impl{PDMDRVHLPRC,pfnVMSetRuntimeError} */
677static DECLCALLBACK(int) pdmRCDrvHlp_VMSetRuntimeError(PPDMDRVINS pDrvIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
678{
679 PDMDRV_ASSERT_DRVINS(pDrvIns);
680 va_list va;
681 va_start(va, pszFormat);
682 int rc = VMSetRuntimeErrorV(pDrvIns->Internal.s.pVMRC, fFlags, pszErrorId, pszFormat, va);
683 va_end(va);
684 return rc;
685}
686
687
688/** @interface_method_impl{PDMDRVHLPRC,pfnVMSetErrorV} */
689static DECLCALLBACK(int) pdmRCDrvHlp_VMSetRuntimeErrorV(PPDMDRVINS pDrvIns, uint32_t fFlags, const char *pszErrorId, const char *pszFormat, va_list va)
690{
691 PDMDRV_ASSERT_DRVINS(pDrvIns);
692 int rc = VMSetRuntimeErrorV(pDrvIns->Internal.s.pVMRC, fFlags, pszErrorId, pszFormat, va);
693 return rc;
694}
695
696
697/** @interface_method_impl{PDMDRVHLPRC,pfnAssertEMT} */
698static DECLCALLBACK(bool) pdmRCDrvHlp_AssertEMT(PPDMDRVINS pDrvIns, const char *pszFile, unsigned iLine, const char *pszFunction)
699{
700 PDMDRV_ASSERT_DRVINS(pDrvIns);
701 if (VM_IS_EMT(pDrvIns->Internal.s.pVMRC))
702 return true;
703
704 RTAssertMsg1Weak("AssertEMT", iLine, pszFile, pszFunction);
705 RTAssertPanic();
706 return false;
707}
708
709
710/** @interface_method_impl{PDMDRVHLPRC,pfnAssertOther} */
711static DECLCALLBACK(bool) pdmRCDrvHlp_AssertOther(PPDMDRVINS pDrvIns, const char *pszFile, unsigned iLine, const char *pszFunction)
712{
713 PDMDRV_ASSERT_DRVINS(pDrvIns);
714 if (!VM_IS_EMT(pDrvIns->Internal.s.pVMRC))
715 return true;
716
717 /* Note: While we don't have any other threads but EMT(0) in RC, might
718 still have drive code compiled in which it shouldn't execute. */
719 RTAssertMsg1Weak("AssertOther", iLine, pszFile, pszFunction);
720 RTAssertPanic();
721 return false;
722}
723
724
725/** @interface_method_impl{PDMDRVHLPRC,pfnFTSetCheckpoint} */
726static DECLCALLBACK(int) pdmRCDrvHlp_FTSetCheckpoint(PPDMDRVINS pDrvIns, FTMCHECKPOINTTYPE enmType)
727{
728 PDMDRV_ASSERT_DRVINS(pDrvIns);
729 return FTMSetCheckpoint(pDrvIns->Internal.s.pVMRC, enmType);
730}
731
732
733/**
734 * The Raw-Mode Context Driver Helper Callbacks.
735 */
736extern DECLEXPORT(const PDMDRVHLPRC) g_pdmRCDrvHlp =
737{
738 PDM_DRVHLPRC_VERSION,
739 pdmRCDrvHlp_VMSetError,
740 pdmRCDrvHlp_VMSetErrorV,
741 pdmRCDrvHlp_VMSetRuntimeError,
742 pdmRCDrvHlp_VMSetRuntimeErrorV,
743 pdmRCDrvHlp_AssertEMT,
744 pdmRCDrvHlp_AssertOther,
745 pdmRCDrvHlp_FTSetCheckpoint,
746 PDM_DRVHLPRC_VERSION
747};
748
749/** @} */
750
751
752
753
754/**
755 * Sets an irq on the I/O APIC.
756 *
757 * @param pVM The VM handle.
758 * @param iIrq The irq.
759 * @param iLevel The new level.
760 */
761static void pdmRCIsaSetIrq(PVM pVM, int iIrq, int iLevel)
762{
763 if ( ( pVM->pdm.s.IoApic.pDevInsRC
764 || !pVM->pdm.s.IoApic.pDevInsR3)
765 && ( pVM->pdm.s.Pic.pDevInsRC
766 || !pVM->pdm.s.Pic.pDevInsR3))
767 {
768 pdmLock(pVM);
769 if (pVM->pdm.s.Pic.pDevInsRC)
770 pVM->pdm.s.Pic.pfnSetIrqRC(pVM->pdm.s.Pic.pDevInsRC, iIrq, iLevel);
771 if (pVM->pdm.s.IoApic.pDevInsRC)
772 pVM->pdm.s.IoApic.pfnSetIrqRC(pVM->pdm.s.IoApic.pDevInsRC, iIrq, iLevel);
773 pdmUnlock(pVM);
774 }
775 else
776 {
777 /* queue for ring-3 execution. */
778 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)PDMQueueAlloc(pVM->pdm.s.pDevHlpQueueRC);
779 if (pTask)
780 {
781 pTask->enmOp = PDMDEVHLPTASKOP_ISA_SET_IRQ;
782 pTask->pDevInsR3 = NIL_RTR3PTR; /* not required */
783 pTask->u.SetIRQ.iIrq = iIrq;
784 pTask->u.SetIRQ.iLevel = iLevel;
785
786 PDMQueueInsertEx(pVM->pdm.s.pDevHlpQueueRC, &pTask->Core, 0);
787 }
788 else
789 AssertMsgFailed(("We're out of devhlp queue items!!!\n"));
790 }
791}
792
793
794/**
795 * Sets an irq on the I/O APIC.
796 *
797 * @param pVM The VM handle.
798 * @param iIrq The irq.
799 * @param iLevel The new level.
800 */
801static void pdmRCIoApicSetIrq(PVM pVM, int iIrq, int iLevel)
802{
803 if (pVM->pdm.s.IoApic.pDevInsRC)
804 {
805 pdmLock(pVM);
806 pVM->pdm.s.IoApic.pfnSetIrqRC(pVM->pdm.s.IoApic.pDevInsRC, iIrq, iLevel);
807 pdmUnlock(pVM);
808 }
809 else if (pVM->pdm.s.IoApic.pDevInsR3)
810 {
811 /* queue for ring-3 execution. */
812 PPDMDEVHLPTASK pTask = (PPDMDEVHLPTASK)PDMQueueAlloc(pVM->pdm.s.pDevHlpQueueRC);
813 if (pTask)
814 {
815 pTask->enmOp = PDMDEVHLPTASKOP_IOAPIC_SET_IRQ;
816 pTask->pDevInsR3 = NIL_RTR3PTR; /* not required */
817 pTask->u.SetIRQ.iIrq = iIrq;
818 pTask->u.SetIRQ.iLevel = iLevel;
819
820 PDMQueueInsertEx(pVM->pdm.s.pDevHlpQueueRC, &pTask->Core, 0);
821 }
822 else
823 AssertMsgFailed(("We're out of devhlp queue items!!!\n"));
824 }
825}
826
827
828/**
829 * Sends an MSI to I/O APIC.
830 *
831 * @param pVM The VM handle.
832 * @param GCAddr Address of the message.
833 * @param uValue Value of the message.
834 */
835static void pdmRCIoApicSendMsi(PVM pVM, RTGCPHYS GCAddr, uint32_t uValue)
836{
837 if (pVM->pdm.s.IoApic.pDevInsRC)
838 {
839 pdmLock(pVM);
840 pVM->pdm.s.IoApic.pfnSendMsiRC(pVM->pdm.s.IoApic.pDevInsRC, GCAddr, uValue);
841 pdmUnlock(pVM);
842 }
843}
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