VirtualBox

source: vbox/trunk/src/VBox/VMM/IOMInternal.h@ 7088

Last change on this file since 7088 was 7053, checked in by vboxsync, 17 years ago

Realigning after RTGCPHYS change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 18.7 KB
Line 
1/* $Id: IOMInternal.h 7053 2008-02-20 16:26:06Z vboxsync $ */
2/** @file
3 * IOM - Internal header file.
4 */
5
6/*
7 * Copyright (C) 2006-2007 innotek GmbH
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#ifndef ___IOMInternal_h
19#define ___IOMInternal_h
20
21#include <VBox/cdefs.h>
22#include <VBox/types.h>
23#include <VBox/iom.h>
24#include <VBox/stam.h>
25#include <iprt/avl.h>
26
27#if !defined(IN_IOM_R3) && !defined(IN_IOM_R0) && !defined(IN_IOM_GC)
28# error "Not in IOM! This is an internal header!"
29#endif
30
31
32/** @defgroup grp_iom_int Internals
33 * @ingroup grp_iom
34 * @internal
35 * @{
36 */
37
38/**
39 * MMIO range descriptor, R3 version.
40 */
41typedef struct IOMMMIORANGER3
42{
43 /** Avl node core with GCPhys as Key and GCPhys + cbSize - 1 as KeyLast. */
44 AVLROGCPHYSNODECORE Core;
45 /** Start physical address. */
46 RTGCPHYS GCPhys;
47 /** Size of the range. */
48 RTUINT cbSize;
49 /** Alignment. */
50 uint32_t u32Alignment;
51 /** Pointer to user argument. */
52 RTR3PTR pvUser;
53 /** Pointer to device instance. */
54 R3PTRTYPE(PPDMDEVINS) pDevIns;
55 /** Pointer to write callback function. */
56 R3PTRTYPE(PFNIOMMMIOWRITE) pfnWriteCallback;
57 /** Pointer to read callback function. */
58 R3PTRTYPE(PFNIOMMMIOREAD) pfnReadCallback;
59 /** Pointer to fill (memset) callback function. */
60 R3PTRTYPE(PFNIOMMMIOFILL) pfnFillCallback;
61 /** Description / Name. For easing debugging. */
62 R3PTRTYPE(const char *) pszDesc;
63} IOMMMIORANGER3;
64/** Pointer to a MMIO range descriptor, R3 version. */
65typedef struct IOMMMIORANGER3 *PIOMMMIORANGER3;
66
67/**
68 * MMIO range descriptor, R0 version.
69 */
70typedef struct IOMMMIORANGER0
71{
72 /** Avl node core with GCPhys as Key and GCPhys + cbSize - 1 as KeyLast. */
73 AVLROGCPHYSNODECORE Core;
74 /** Start physical address. */
75 RTGCPHYS GCPhys;
76 /** Size of the range. */
77 uint32_t cbSize;
78 /** Alignment. */
79 uint32_t u32Alignment;
80 /** Pointer to user argument. */
81 RTR0PTR pvUser;
82 /** Pointer to device instance. */
83 R0PTRTYPE(PPDMDEVINS) pDevIns;
84 /** Pointer to write callback function. */
85 R0PTRTYPE(PFNIOMMMIOWRITE) pfnWriteCallback;
86 /** Pointer to read callback function. */
87 R0PTRTYPE(PFNIOMMMIOREAD) pfnReadCallback;
88 /** Pointer to fill (memset) callback function. */
89 R0PTRTYPE(PFNIOMMMIOFILL) pfnFillCallback;
90 /** Description / Name. For easing debugging. */
91 R3PTRTYPE(const char *) pszDesc;
92} IOMMMIORANGER0;
93/** Pointer to a MMIO range descriptor, R0 version. */
94typedef struct IOMMMIORANGER0 *PIOMMMIORANGER0;
95
96/**
97 * MMIO range descriptor, GC version.
98 */
99typedef struct IOMMMIORANGEGC
100{
101 /** Avl node core with GCPhys as Key and GCPhys + cbSize - 1 as KeyLast. */
102 AVLROGCPHYSNODECORE Core;
103 /** Start physical address. */
104 RTGCPHYS GCPhys;
105 /** Size of the range. */
106 RTUINT cbSize;
107 /** Pointer to user argument. */
108 RTGCPTR pvUser;
109 /** Pointer to device instance. */
110 GCPTRTYPE(PPDMDEVINS) pDevIns;
111 /** Pointer to write callback function. */
112 GCPTRTYPE(PFNIOMMMIOWRITE) pfnWriteCallback;
113 /** Pointer to read callback function. */
114 GCPTRTYPE(PFNIOMMMIOREAD) pfnReadCallback;
115 /** Pointer to fill (memset) callback function. */
116 GCPTRTYPE(PFNIOMMMIOFILL) pfnFillCallback;
117 /** Description / Name. For easing debugging. */
118 R3PTRTYPE(const char *) pszDesc;
119} IOMMMIORANGEGC;
120/** Pointer to a MMIO range descriptor, GC version. */
121typedef struct IOMMMIORANGEGC *PIOMMMIORANGEGC;
122
123
124/**
125 * MMIO address statistics. (one address)
126 *
127 * This is a simple way of making on demand statistics, however it's a
128 * bit free with the hypervisor heap memory..
129 */
130typedef struct IOMMMIOSTATS
131{
132 /** Avl node core with the address as Key. */
133 AVLOGCPHYSNODECORE Core;
134 /** Number of reads to this address from R3. */
135 STAMCOUNTER ReadR3;
136 /** Number of writes to this address from R3. */
137 STAMCOUNTER WriteR3;
138 /** Number of reads to this address from R0. */
139 STAMCOUNTER ReadR0;
140 /** Number of writes to this address from R0. */
141 STAMCOUNTER WriteR0;
142 /** Number of reads to this address from GC. */
143 STAMCOUNTER ReadGC;
144 /** Number of writes to this address from GC. */
145 STAMCOUNTER WriteGC;
146 /** Profiling read handler overhead in R3. */
147 STAMPROFILEADV ProfReadR3;
148 /** Profiling write handler overhead in R3. */
149 STAMPROFILEADV ProfWriteR3;
150 /** Profiling read handler overhead in R0. */
151 STAMPROFILEADV ProfReadR0;
152 /** Profiling write handler overhead in R0. */
153 STAMPROFILEADV ProfWriteR0;
154 /** Profiling read handler overhead in GC. */
155 STAMPROFILEADV ProfReadGC;
156 /** Profiling write handler overhead in GC. */
157 STAMPROFILEADV ProfWriteGC;
158 /** Number of reads to this address from R0 which was serviced in R3. */
159 STAMCOUNTER ReadR0ToR3;
160 /** Number of writes to this address from R0 which was serviced in R3. */
161 STAMCOUNTER WriteR0ToR3;
162 /** Number of reads to this address from GC which was serviced in R3. */
163 STAMCOUNTER ReadGCToR3;
164 /** Number of writes to this address from GC which was serviced in R3. */
165 STAMCOUNTER WriteGCToR3;
166} IOMMMIOSTATS;
167/** Pointer to I/O port statistics. */
168typedef IOMMMIOSTATS *PIOMMMIOSTATS;
169
170
171/**
172 * I/O port range descriptor, R3 version.
173 */
174typedef struct IOMIOPORTRANGER3
175{
176 /** Avl node core with Port as Key and Port + cPorts - 1 as KeyLast. */
177 AVLROIOPORTNODECORE Core;
178#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32 && !defined(RT_OS_WINDOWS)
179 uint32_t u32Alignment; /**< The sizeof(Core) differs. */
180#endif
181 /** Start I/O port address. */
182 RTIOPORT Port;
183 /** Size of the range. */
184 uint16_t cPorts;
185 /** Pointer to user argument. */
186 RTR3PTR pvUser;
187 /** Pointer to the associated device instance. */
188 R3PTRTYPE(PPDMDEVINS) pDevIns;
189 /** Pointer to OUT callback function. */
190 R3PTRTYPE(PFNIOMIOPORTOUT) pfnOutCallback;
191 /** Pointer to IN callback function. */
192 R3PTRTYPE(PFNIOMIOPORTIN) pfnInCallback;
193 /** Pointer to string OUT callback function. */
194 R3PTRTYPE(PFNIOMIOPORTOUTSTRING) pfnOutStrCallback;
195 /** Pointer to string IN callback function. */
196 R3PTRTYPE(PFNIOMIOPORTINSTRING) pfnInStrCallback;
197 /** Description / Name. For easing debugging. */
198 R3PTRTYPE(const char *) pszDesc;
199} IOMIOPORTRANGER3;
200/** Pointer to I/O port range descriptor, R3 version. */
201typedef IOMIOPORTRANGER3 *PIOMIOPORTRANGER3;
202
203/**
204 * I/O port range descriptor, R0 version.
205 */
206typedef struct IOMIOPORTRANGER0
207{
208 /** Avl node core with Port as Key and Port + cPorts - 1 as KeyLast. */
209 AVLROIOPORTNODECORE Core;
210#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32 && !defined(RT_OS_WINDOWS)
211 uint32_t u32Alignment; /**< The sizeof(Core) differs. */
212#endif
213 /** Start I/O port address. */
214 RTIOPORT Port;
215 /** Size of the range. */
216 uint16_t cPorts;
217 /** Pointer to user argument. */
218 RTR0PTR pvUser;
219 /** Pointer to the associated device instance. */
220 R0PTRTYPE(PPDMDEVINS) pDevIns;
221 /** Pointer to OUT callback function. */
222 R0PTRTYPE(PFNIOMIOPORTOUT) pfnOutCallback;
223 /** Pointer to IN callback function. */
224 R0PTRTYPE(PFNIOMIOPORTIN) pfnInCallback;
225 /** Pointer to string OUT callback function. */
226 R0PTRTYPE(PFNIOMIOPORTOUTSTRING) pfnOutStrCallback;
227 /** Pointer to string IN callback function. */
228 R0PTRTYPE(PFNIOMIOPORTINSTRING) pfnInStrCallback;
229 /** Description / Name. For easing debugging. */
230 R3PTRTYPE(const char *) pszDesc;
231} IOMIOPORTRANGER0;
232/** Pointer to I/O port range descriptor, R0 version. */
233typedef IOMIOPORTRANGER0 *PIOMIOPORTRANGER0;
234
235/**
236 * I/O port range descriptor.
237 */
238typedef struct IOMIOPORTRANGEGC
239{
240 /** Avl node core with Port as Key and Port + cPorts - 1 as KeyLast. */
241 AVLROIOPORTNODECORE Core;
242 /** Start I/O port address. */
243 RTIOPORT Port;
244 /** Size of the range. */
245 uint16_t cPorts;
246 /** Pointer to user argument. */
247 RTGCPTR pvUser;
248 /** Pointer to the associated device instance. */
249 GCPTRTYPE(PPDMDEVINS) pDevIns;
250 /** Pointer to OUT callback function. */
251 GCPTRTYPE(PFNIOMIOPORTOUT) pfnOutCallback;
252 /** Pointer to IN callback function. */
253 GCPTRTYPE(PFNIOMIOPORTIN) pfnInCallback;
254 /** Pointer to string OUT callback function. */
255 GCPTRTYPE(PFNIOMIOPORTOUTSTRING) pfnOutStrCallback;
256 /** Pointer to string IN callback function. */
257 GCPTRTYPE(PFNIOMIOPORTINSTRING) pfnInStrCallback;
258#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32
259 RTGCPTR GCPtrAlignment; /**< pszDesc is 8 byte aligned. */
260#endif
261 /** Description / Name. For easing debugging. */
262 R3PTRTYPE(const char *) pszDesc;
263} IOMIOPORTRANGEGC;
264/** Pointer to I/O port range descriptor, GC version. */
265typedef IOMIOPORTRANGEGC *PIOMIOPORTRANGEGC;
266
267
268/**
269 * I/O port statistics. (one I/O port)
270 *
271 * This is a simple way of making on demand statistics, however it's a
272 * bit free with the hypervisor heap memory..
273 */
274typedef struct IOMIOPORTSTATS
275{
276 /** Avl node core with the port as Key. */
277 AVLOIOPORTNODECORE Core;
278#if HC_ARCH_BITS == 64 && GC_ARCH_BITS == 32 && !defined(RT_OS_WINDOWS)
279 uint32_t u32Alignment; /**< The sizeof(Core) differs. */
280#endif
281 /** Number of INs to this port from R3. */
282 STAMCOUNTER InR3;
283 /** Number of OUTs to this port from R3. */
284 STAMCOUNTER OutR3;
285 /** Number of INs to this port from R0. */
286 STAMCOUNTER InR0;
287 /** Number of OUTs to this port from R0. */
288 STAMCOUNTER OutR0;
289 /** Number of INs to this port from GC. */
290 STAMCOUNTER InGC;
291 /** Number of OUTs to this port from GC. */
292 STAMCOUNTER OutGC;
293 /** Profiling IN handler overhead in R3. */
294 STAMPROFILEADV ProfInR3;
295 /** Profiling OUT handler overhead in R3. */
296 STAMPROFILEADV ProfOutR3;
297 /** Profiling IN handler overhead in R0. */
298 STAMPROFILEADV ProfInR0;
299 /** Profiling OUT handler overhead in R0. */
300 STAMPROFILEADV ProfOutR0;
301 /** Profiling IN handler overhead in GC. */
302 STAMPROFILEADV ProfInGC;
303 /** Profiling OUT handler overhead in GC. */
304 STAMPROFILEADV ProfOutGC;
305 /** Number of INs to this port from R0 which was serviced in R3. */
306 STAMCOUNTER InR0ToR3;
307 /** Number of OUTs to this port from R0 which was serviced in R3. */
308 STAMCOUNTER OutR0ToR3;
309 /** Number of INs to this port from GC which was serviced in R3. */
310 STAMCOUNTER InGCToR3;
311 /** Number of OUTs to this port from GC which was serviced in R3. */
312 STAMCOUNTER OutGCToR3;
313} IOMIOPORTSTATS;
314/** Pointer to I/O port statistics. */
315typedef IOMIOPORTSTATS *PIOMIOPORTSTATS;
316
317
318/**
319 * The IOM trees.
320 * These are offset based the nodes and root must be in the same
321 * memory block in HC. The locations of IOM structure and the hypervisor heap
322 * are quite different in HC and GC.
323 */
324typedef struct IOMTREES
325{
326 /** Tree containing I/O port range descriptors registered for HC (IOMIOPORTRANGEHC). */
327 AVLROIOPORTTREE IOPortTreeR3;
328 /** Tree containing I/O port range descriptors registered for R0 (IOMIOPORTRANGER0). */
329 AVLROIOPORTTREE IOPortTreeR0;
330 /** Tree containing I/O port range descriptors registered for GC (IOMIOPORTRANGEGC). */
331 AVLROIOPORTTREE IOPortTreeGC;
332
333 /** Tree containing MMIO range descriptors registered for HC (IOMMMIORANGEHC). */
334 AVLROGCPHYSTREE MMIOTreeR3;
335 /** Tree containing MMIO range descriptors registered for R0 (IOMMMIORANGER0). */
336 AVLROGCPHYSTREE MMIOTreeR0;
337 /** Tree containing MMIO range descriptors registered for GC (IOMMMIORANGEGC). */
338 AVLROGCPHYSTREE MMIOTreeGC;
339
340 /** Tree containing I/O port statistics (IOMIOPORTSTATS). */
341 AVLOIOPORTTREE IOPortStatTree;
342 /** Tree containing MMIO statistics (IOMMMIOSTATS). */
343 AVLOGCPHYSTREE MMIOStatTree;
344} IOMTREES;
345/** Pointer to the IOM trees. */
346typedef IOMTREES *PIOMTREES;
347
348
349/**
350 * Converts an IOM pointer into a VM pointer.
351 * @returns Pointer to the VM structure the PGM is part of.
352 * @param pIOM Pointer to IOM instance data.
353 */
354#define IOM2VM(pIOM) ( (PVM)((char*)pIOM - pIOM->offVM) )
355
356/**
357 * IOM Data (part of VM)
358 */
359typedef struct IOM
360{
361 /** Offset to the VM structure. */
362 RTINT offVM;
363
364 /** Pointer to the trees - GC ptr. */
365 GCPTRTYPE(PIOMTREES) pTreesGC;
366 /** Pointer to the trees - HC ptr. */
367 R3R0PTRTYPE(PIOMTREES) pTreesHC;
368
369
370 /** @name Caching of I/O Port ranges and statistics.
371 * (Saves quite some time in rep outs/ins instruction emulation.)
372 * @{ */
373 R3PTRTYPE(PIOMIOPORTRANGER3) pRangeLastReadR3;
374 R3PTRTYPE(PIOMIOPORTRANGER3) pRangeLastWriteR3;
375 R3PTRTYPE(PIOMIOPORTSTATS) pStatsLastReadR3;
376 R3PTRTYPE(PIOMIOPORTSTATS) pStatsLastWriteR3;
377
378 R3R0PTRTYPE(PIOMIOPORTRANGER0) pRangeLastReadR0;
379 R3R0PTRTYPE(PIOMIOPORTRANGER0) pRangeLastWriteR0;
380 R3R0PTRTYPE(PIOMIOPORTSTATS) pStatsLastReadR0;
381 R3R0PTRTYPE(PIOMIOPORTSTATS) pStatsLastWriteR0;
382
383 GCPTRTYPE(PIOMIOPORTRANGEGC) pRangeLastReadGC;
384 GCPTRTYPE(PIOMIOPORTRANGEGC) pRangeLastWriteGC;
385 GCPTRTYPE(PIOMIOPORTSTATS) pStatsLastReadGC;
386 GCPTRTYPE(PIOMIOPORTSTATS) pStatsLastWriteGC;
387 /** @} */
388
389 /** @name I/O Port statistics.
390 * @{ */
391 STAMPROFILE StatGCIOPortHandler;
392
393 STAMCOUNTER StatGCInstIn;
394 STAMCOUNTER StatGCInstOut;
395 STAMCOUNTER StatGCInstIns;
396 STAMCOUNTER StatGCInstOuts;
397 /** @} */
398
399 /** @name MMIO statistics.
400 * @{ */
401 STAMPROFILE StatGCMMIOHandler;
402 STAMCOUNTER StatGCMMIOFailures;
403
404 STAMPROFILE StatGCInstMov;
405 STAMPROFILE StatGCInstCmp;
406 STAMPROFILE StatGCInstAnd;
407 STAMPROFILE StatGCInstTest;
408 STAMPROFILE StatGCInstXchg;
409 STAMPROFILE StatGCInstStos;
410 STAMPROFILE StatGCInstLods;
411 STAMPROFILE StatGCInstMovs;
412 STAMPROFILE StatGCInstMovsToMMIO;
413 STAMPROFILE StatGCInstMovsFromMMIO;
414 STAMPROFILE StatGCInstMovsMMIO;
415 STAMCOUNTER StatGCInstOther;
416
417 STAMCOUNTER StatGCMMIO1Byte;
418 STAMCOUNTER StatGCMMIO2Bytes;
419 STAMCOUNTER StatGCMMIO4Bytes;
420
421 RTUINT cMovsMaxBytes;
422 RTUINT cStosMaxBytes;
423 /** @} */
424
425} IOM;
426/** Pointer to IOM instance data. */
427typedef IOM *PIOM;
428
429
430__BEGIN_DECLS
431
432#ifdef IN_IOM_R3
433PIOMIOPORTSTATS iomr3IOPortStatsCreate(PVM pVM, RTIOPORT Port, const char *pszDesc);
434PIOMMMIOSTATS iomR3MMIOStatsCreate(PVM pVM, RTGCPHYS GCPhys, const char *pszDesc);
435#endif /* IN_IOM_R3 */
436
437/**
438 * \#PF Handler callback for MMIO ranges.
439 *
440 * @returns VBox status code (appropriate for GC return).
441 *
442 * @param pVM VM Handle.
443 * @param uErrorCode CPU Error code.
444 * @param pRegFrame Trap register frame.
445 * @param pvFault The fault address (cr2).
446 * @param GCPhysFault The GC physical address corresponding to pvFault.
447 * @param pvUser Pointer to the MMIO range entry.
448 */
449IOMDECL(int) IOMMMIOHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, void *pvFault, RTGCPHYS GCPhysFault, void *pvUser);
450
451/**
452 * Gets the I/O port range for the specified I/O port in the current context.
453 *
454 * @returns Pointer to I/O port range.
455 * @returns NULL if no port registered.
456 *
457 * @param pIOM IOM instance data.
458 * @param Port Port to lookup.
459 */
460inline CTXALLSUFF(PIOMIOPORTRANGE) iomIOPortGetRange(PIOM pIOM, RTIOPORT Port)
461{
462 CTXALLSUFF(PIOMIOPORTRANGE) pRange = (CTXALLSUFF(PIOMIOPORTRANGE))RTAvlroIOPortRangeGet(&pIOM->CTXSUFF(pTrees)->CTXALLSUFF(IOPortTree), Port);
463 return pRange;
464}
465
466/**
467 * Gets the I/O port range for the specified I/O port in the HC.
468 *
469 * @returns Pointer to I/O port range.
470 * @returns NULL if no port registered.
471 *
472 * @param pIOM IOM instance data.
473 * @param Port Port to lookup.
474 */
475inline PIOMIOPORTRANGER3 iomIOPortGetRangeHC(PIOM pIOM, RTIOPORT Port)
476{
477 PIOMIOPORTRANGER3 pRange = (PIOMIOPORTRANGER3)RTAvlroIOPortRangeGet(&pIOM->CTXSUFF(pTrees)->IOPortTreeR3, Port);
478 return pRange;
479}
480
481
482/**
483 * Gets the MMIO range for the specified physical address in the current context.
484 *
485 * @returns Pointer to MMIO range.
486 * @returns NULL if address not in a MMIO range.
487 *
488 * @param pIOM IOM instance data.
489 * @param GCPhys Physical address to lookup.
490 */
491inline CTXALLSUFF(PIOMMMIORANGE) iomMMIOGetRange(PIOM pIOM, RTGCPHYS GCPhys)
492{
493 CTXALLSUFF(PIOMMMIORANGE) pRange = (CTXALLSUFF(PIOMMMIORANGE))RTAvlroGCPhysRangeGet(&pIOM->CTXSUFF(pTrees)->CTXALLSUFF(MMIOTree), GCPhys);
494 return pRange;
495}
496
497
498/**
499 * Gets the MMIO range for the specified physical address in the current context.
500 *
501 * @returns Pointer to MMIO range.
502 * @returns NULL if address not in a MMIO range.
503 *
504 * @param pIOM IOM instance data.
505 * @param GCPhys Physical address to lookup.
506 */
507inline PIOMMMIORANGER3 iomMMIOGetRangeHC(PIOM pIOM, RTGCPHYS GCPhys)
508{
509 PIOMMMIORANGER3 pRange = (PIOMMMIORANGER3)RTAvlroGCPhysRangeGet(&pIOM->CTXSUFF(pTrees)->MMIOTreeR3, GCPhys);
510 return pRange;
511}
512
513#ifdef VBOX_WITH_STATISTICS
514/**
515 * Gets the MMIO statistics record.
516 * @returns Pointer to MMIO stats.
517 * @returns NULL if not found.
518 *
519 * @param pIOM IOM instance data.
520 * @param GCPhys Physical address to lookup.
521 */
522inline PIOMMMIOSTATS iomMMIOGetStats(PIOM pIOM, RTGCPHYS GCPhys)
523{
524 PIOMMMIOSTATS pStats = (PIOMMMIOSTATS)RTAvloGCPhysGet(&pIOM->CTXSUFF(pTrees)->MMIOStatTree, GCPhys);
525 return pStats;
526}
527#endif
528
529__END_DECLS
530
531#ifdef IN_RING3
532
533#endif
534
535/** @} */
536
537#endif /* ___IOMInternal_h */
Note: See TracBrowser for help on using the repository browser.

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