VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/IOMAllMMIO.cpp@ 7932

Last change on this file since 7932 was 7815, checked in by vboxsync, 17 years ago

Fixed wrong test in IOMMMIOHandler that caused assertions/bad stuff later on if only the ring-3 handler was registered.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 59.6 KB
Line 
1/* $Id: IOMAllMMIO.cpp 7815 2008-04-08 19:57:44Z vboxsync $ */
2/** @file
3 * IOM - Input / Output Monitor - Guest Context.
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
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_IOM
23#include <VBox/iom.h>
24#include <VBox/cpum.h>
25#include <VBox/pgm.h>
26#include <VBox/selm.h>
27#include <VBox/mm.h>
28#include <VBox/em.h>
29#include <VBox/pgm.h>
30#include <VBox/trpm.h>
31#include "IOMInternal.h"
32#include <VBox/vm.h>
33
34#include <VBox/dis.h>
35#include <VBox/disopcode.h>
36#include <VBox/param.h>
37#include <VBox/err.h>
38#include <iprt/assert.h>
39#include <VBox/log.h>
40#include <iprt/asm.h>
41#include <iprt/string.h>
42
43/*******************************************************************************
44* Internal Functions *
45*******************************************************************************/
46static bool iomGetRegImmData(PDISCPUSTATE pCpu, PCOP_PARAMETER pParam, PCPUMCTXCORE pRegFrame, uint32_t *pu32Data, unsigned *pcbSize);
47static bool iomSaveDataToReg(PDISCPUSTATE pCpu, PCOP_PARAMETER pParam, PCPUMCTXCORE pRegFrame, uint32_t u32Data);
48
49
50/*******************************************************************************
51* Global Variables *
52*******************************************************************************/
53/**
54 * Array for accessing 32-bit general registers in VMMREGFRAME structure
55 * by register's index from disasm.
56 */
57static unsigned g_aReg32Index[] =
58{
59 RT_OFFSETOF(CPUMCTXCORE, eax), /* USE_REG_EAX */
60 RT_OFFSETOF(CPUMCTXCORE, ecx), /* USE_REG_ECX */
61 RT_OFFSETOF(CPUMCTXCORE, edx), /* USE_REG_EDX */
62 RT_OFFSETOF(CPUMCTXCORE, ebx), /* USE_REG_EBX */
63 RT_OFFSETOF(CPUMCTXCORE, esp), /* USE_REG_ESP */
64 RT_OFFSETOF(CPUMCTXCORE, ebp), /* USE_REG_EBP */
65 RT_OFFSETOF(CPUMCTXCORE, esi), /* USE_REG_ESI */
66 RT_OFFSETOF(CPUMCTXCORE, edi) /* USE_REG_EDI */
67};
68
69/**
70 * Macro for accessing 32-bit general purpose registers in CPUMCTXCORE structure.
71 */
72#define ACCESS_REG32(p, idx) (*((uint32_t *)((char *)(p) + g_aReg32Index[idx])))
73
74/**
75 * Array for accessing 16-bit general registers in CPUMCTXCORE structure
76 * by register's index from disasm.
77 */
78static unsigned g_aReg16Index[] =
79{
80 RT_OFFSETOF(CPUMCTXCORE, eax), /* USE_REG_AX */
81 RT_OFFSETOF(CPUMCTXCORE, ecx), /* USE_REG_CX */
82 RT_OFFSETOF(CPUMCTXCORE, edx), /* USE_REG_DX */
83 RT_OFFSETOF(CPUMCTXCORE, ebx), /* USE_REG_BX */
84 RT_OFFSETOF(CPUMCTXCORE, esp), /* USE_REG_SP */
85 RT_OFFSETOF(CPUMCTXCORE, ebp), /* USE_REG_BP */
86 RT_OFFSETOF(CPUMCTXCORE, esi), /* USE_REG_SI */
87 RT_OFFSETOF(CPUMCTXCORE, edi) /* USE_REG_DI */
88};
89
90/**
91 * Macro for accessing 16-bit general purpose registers in CPUMCTXCORE structure.
92 */
93#define ACCESS_REG16(p, idx) (*((uint16_t *)((char *)(p) + g_aReg16Index[idx])))
94
95/**
96 * Array for accessing 8-bit general registers in CPUMCTXCORE structure
97 * by register's index from disasm.
98 */
99static unsigned g_aReg8Index[] =
100{
101 RT_OFFSETOF(CPUMCTXCORE, eax), /* USE_REG_AL */
102 RT_OFFSETOF(CPUMCTXCORE, ecx), /* USE_REG_CL */
103 RT_OFFSETOF(CPUMCTXCORE, edx), /* USE_REG_DL */
104 RT_OFFSETOF(CPUMCTXCORE, ebx), /* USE_REG_BL */
105 RT_OFFSETOF(CPUMCTXCORE, eax) + 1, /* USE_REG_AH */
106 RT_OFFSETOF(CPUMCTXCORE, ecx) + 1, /* USE_REG_CH */
107 RT_OFFSETOF(CPUMCTXCORE, edx) + 1, /* USE_REG_DH */
108 RT_OFFSETOF(CPUMCTXCORE, ebx) + 1 /* USE_REG_BH */
109};
110
111/**
112 * Macro for accessing 8-bit general purpose registers in CPUMCTXCORE structure.
113 */
114#define ACCESS_REG8(p, idx) (*((uint8_t *)((char *)(p) + g_aReg8Index[idx])))
115
116/**
117 * Array for accessing segment registers in CPUMCTXCORE structure
118 * by register's index from disasm.
119 */
120static unsigned g_aRegSegIndex[] =
121{
122 RT_OFFSETOF(CPUMCTXCORE, es), /* USE_REG_ES */
123 RT_OFFSETOF(CPUMCTXCORE, cs), /* USE_REG_CS */
124 RT_OFFSETOF(CPUMCTXCORE, ss), /* USE_REG_SS */
125 RT_OFFSETOF(CPUMCTXCORE, ds), /* USE_REG_DS */
126 RT_OFFSETOF(CPUMCTXCORE, fs), /* USE_REG_FS */
127 RT_OFFSETOF(CPUMCTXCORE, gs) /* USE_REG_GS */
128};
129
130/**
131 * Macro for accessing segment registers in CPUMCTXCORE structure.
132 */
133#define ACCESS_REGSEG(p, idx) (*((uint16_t *)((char *)(p) + g_aRegSegIndex[idx])))
134
135/**
136 * Array for fast recode of the operand size (1/2/4/8 bytes) to bit shift value.
137 */
138static const unsigned g_aSize2Shift[] =
139{
140 ~0, /* 0 - invalid */
141 0, /* *1 == 2^0 */
142 1, /* *2 == 2^1 */
143 ~0, /* 3 - invalid */
144 2, /* *4 == 2^2 */
145 ~0, /* 5 - invalid */
146 ~0, /* 6 - invalid */
147 ~0, /* 7 - invalid */
148 3 /* *8 == 2^3 */
149};
150
151/**
152 * Macro for fast recode of the operand size (1/2/4/8 bytes) to bit shift value.
153 */
154#define SIZE_2_SHIFT(cb) (g_aSize2Shift[cb])
155
156
157/**
158 * Wrapper which does the write and updates range statistics when such are enabled.
159 * @warning VBOX_SUCCESS(rc=VINF_IOM_HC_MMIO_WRITE) is TRUE!
160 */
161DECLINLINE(int) iomMMIODoWrite(PVM pVM, PIOMMMIORANGE pRange, RTGCPHYS GCPhysFault, const void *pvData, unsigned cb)
162{
163#ifdef VBOX_WITH_STATISTICS
164 PIOMMMIOSTATS pStats = iomMMIOGetStats(&pVM->iom.s, GCPhysFault, pRange);
165 Assert(pStats);
166#endif
167
168 int rc;
169 if (RT_LIKELY(pRange->CTXALLSUFF(pfnWriteCallback)))
170 rc = pRange->CTXALLSUFF(pfnWriteCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), GCPhysFault, (void *)pvData, cb); /* @todo fix const!! */
171 else
172 rc = VINF_SUCCESS;
173 if (rc != VINF_IOM_HC_MMIO_WRITE)
174 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Write));
175 return rc;
176}
177
178/**
179 * Wrapper which does the read and updates range statistics when such are enabled.
180 */
181DECLINLINE(int) iomMMIODoRead(PVM pVM, PIOMMMIORANGE pRange, RTGCPHYS GCPhysFault, void *pvData, unsigned cb)
182{
183#ifdef VBOX_WITH_STATISTICS
184 PIOMMMIOSTATS pStats = iomMMIOGetStats(&pVM->iom.s, GCPhysFault, pRange);
185 Assert(pStats);
186#endif
187
188 int rc;
189 if (RT_LIKELY(pRange->CTXALLSUFF(pfnReadCallback)))
190 rc = pRange->CTXALLSUFF(pfnReadCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), GCPhysFault, pvData, cb);
191 else
192 {
193 switch (cb)
194 {
195 case 1: *(uint8_t *)pvData = 0; break;
196 case 2: *(uint16_t *)pvData = 0; break;
197 case 4: *(uint32_t *)pvData = 0; break;
198 case 8: *(uint64_t *)pvData = 0; break;
199 default:
200 memset(pvData, 0, cb);
201 break;
202 }
203 rc = VINF_SUCCESS;
204 }
205 if (rc != VINF_IOM_HC_MMIO_READ)
206 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Read));
207 return rc;
208}
209
210
211/**
212 * Returns the contents of register or immediate data of instruction's parameter.
213 *
214 * @returns true on success.
215 *
216 * @param pCpu Pointer to current disassembler context.
217 * @param pParam Pointer to parameter of instruction to proccess.
218 * @param pRegFrame Pointer to CPUMCTXCORE guest structure.
219 * @param pu32Data Where to store retrieved data.
220 * @param pcbSize Where to store the size of data (1, 2, 4).
221 */
222static bool iomGetRegImmData(PDISCPUSTATE pCpu, PCOP_PARAMETER pParam, PCPUMCTXCORE pRegFrame, uint32_t *pu32Data, unsigned *pcbSize)
223{
224 if (pParam->flags & (USE_BASE | USE_INDEX | USE_SCALE | USE_DISPLACEMENT8 | USE_DISPLACEMENT16 | USE_DISPLACEMENT32))
225 {
226 *pcbSize = 0;
227 *pu32Data = 0;
228 return false;
229 }
230
231 if (pParam->flags & USE_REG_GEN32)
232 {
233 *pcbSize = 4;
234 *pu32Data = ACCESS_REG32(pRegFrame, pParam->base.reg_gen32);
235 return true;
236 }
237
238 if (pParam->flags & USE_REG_GEN16)
239 {
240 *pcbSize = 2;
241 *pu32Data = ACCESS_REG16(pRegFrame, pParam->base.reg_gen16);
242 return true;
243 }
244
245 if (pParam->flags & USE_REG_GEN8)
246 {
247 *pcbSize = 1;
248 *pu32Data = ACCESS_REG8(pRegFrame, pParam->base.reg_gen8);
249 return true;
250 }
251
252 if (pParam->flags & (USE_IMMEDIATE32|USE_IMMEDIATE32_SX8))
253 {
254 *pcbSize = 4;
255 *pu32Data = (uint32_t)pParam->parval;
256 return true;
257 }
258
259 if (pParam->flags & (USE_IMMEDIATE16|USE_IMMEDIATE16_SX8))
260 {
261 *pcbSize = 2;
262 *pu32Data = (uint16_t)pParam->parval;
263 return true;
264 }
265
266 if (pParam->flags & USE_IMMEDIATE8)
267 {
268 *pcbSize = 1;
269 *pu32Data = (uint8_t)pParam->parval;
270 return true;
271 }
272
273 if (pParam->flags & USE_REG_SEG)
274 {
275 *pcbSize = 2;
276 *pu32Data = ACCESS_REGSEG(pRegFrame, pParam->base.reg_seg);
277 return true;
278 } /* Else - error. */
279
280 *pcbSize = 0;
281 *pu32Data = 0;
282 return false;
283}
284
285
286/**
287 * Saves data to 8/16/32 general purpose or segment register defined by
288 * instruction's parameter.
289 *
290 * @returns true on success.
291 * @param pCpu Pointer to current disassembler context.
292 * @param pParam Pointer to parameter of instruction to proccess.
293 * @param pRegFrame Pointer to CPUMCTXCORE guest structure.
294 * @param u32Data 8/16/32 bit data to store.
295 */
296static bool iomSaveDataToReg(PDISCPUSTATE pCpu, PCOP_PARAMETER pParam, PCPUMCTXCORE pRegFrame, unsigned u32Data)
297{
298 if (pParam->flags & (USE_BASE | USE_INDEX | USE_SCALE | USE_DISPLACEMENT8 | USE_DISPLACEMENT16 | USE_DISPLACEMENT32 | USE_IMMEDIATE8 | USE_IMMEDIATE16 | USE_IMMEDIATE32 | USE_IMMEDIATE32_SX8 | USE_IMMEDIATE16_SX8))
299 {
300 return false;
301 }
302
303 if (pParam->flags & USE_REG_GEN32)
304 {
305 ACCESS_REG32(pRegFrame, pParam->base.reg_gen32) = u32Data;
306 return true;
307 }
308
309 if (pParam->flags & USE_REG_GEN16)
310 {
311 ACCESS_REG16(pRegFrame, pParam->base.reg_gen16) = (uint16_t)u32Data;
312 return true;
313 }
314
315 if (pParam->flags & USE_REG_GEN8)
316 {
317 ACCESS_REG8(pRegFrame, pParam->base.reg_gen8) = (uint8_t)u32Data;
318 return true;
319 }
320
321 if (pParam->flags & USE_REG_SEG)
322 {
323 ACCESS_REGSEG(pRegFrame, pParam->base.reg_seg) = (uint16_t)u32Data;
324 return true;
325 }
326
327 /* Else - error. */
328 return false;
329}
330
331
332/*
333 * Internal - statistics only.
334 */
335DECLINLINE(void) iomMMIOStatLength(PVM pVM, unsigned cb)
336{
337#ifdef VBOX_WITH_STATISTICS
338 switch (cb)
339 {
340 case 1:
341 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIO1Byte);
342 break;
343 case 2:
344 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIO2Bytes);
345 break;
346 case 4:
347 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIO4Bytes);
348 break;
349 default:
350 /* No way. */
351 AssertMsgFailed(("Invalid data length %d\n", cb));
352 break;
353 }
354#else
355 NOREF(pVM); NOREF(cb);
356#endif
357}
358
359
360/**
361 * MOV reg, mem (read)
362 * MOVZX reg, mem (read)
363 * MOVSX reg, mem (read)
364 *
365 * @returns VBox status code.
366 *
367 * @param pVM The virtual machine (GC pointer ofcourse).
368 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
369 * @param pCpu Disassembler CPU state.
370 * @param pRange Pointer MMIO range.
371 * @param GCPhysFault The GC physical address corresponding to pvFault.
372 */
373static int iomInterpretMOVxXRead(PVM pVM, PCPUMCTXCORE pRegFrame, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange, RTGCPHYS GCPhysFault)
374{
375 Assert(pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3);
376
377 /*
378 * Get the data size from parameter 2,
379 * and call the handler function to get the data.
380 */
381 unsigned cb = DISGetParamSize(pCpu, &pCpu->param2);
382 AssertMsg(cb > 0 && cb <= sizeof(uint32_t), ("cb=%d\n", cb));
383
384 uint32_t u32Data = 0;
385 int rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &u32Data, cb);
386 if (rc == VINF_SUCCESS)
387 {
388 /*
389 * Do sign extension for MOVSX.
390 */
391 /** @todo checkup MOVSX implementation! */
392 if (pCpu->pCurInstr->opcode == OP_MOVSX)
393 {
394 if (cb == 1)
395 {
396 /* DWORD <- BYTE */
397 int32_t iData = (int8_t)u32Data;
398 u32Data = (uint32_t)iData;
399 }
400 else
401 {
402 /* DWORD <- WORD */
403 int32_t iData = (int16_t)u32Data;
404 u32Data = (uint32_t)iData;
405 }
406 }
407
408 /*
409 * Store the result to register (parameter 1).
410 */
411 bool fRc = iomSaveDataToReg(pCpu, &pCpu->param1, pRegFrame, u32Data);
412 AssertMsg(fRc, ("Failed to store register value!\n")); NOREF(fRc);
413 }
414
415 if (rc == VINF_SUCCESS)
416 iomMMIOStatLength(pVM, cb);
417 return rc;
418}
419
420
421/**
422 * MOV mem, reg|imm (write)
423 *
424 * @returns VBox status code.
425 *
426 * @param pVM The virtual machine (GC pointer ofcourse).
427 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
428 * @param pCpu Disassembler CPU state.
429 * @param pRange Pointer MMIO range.
430 * @param GCPhysFault The GC physical address corresponding to pvFault.
431 */
432static int iomInterpretMOVxXWrite(PVM pVM, PCPUMCTXCORE pRegFrame, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange, RTGCPHYS GCPhysFault)
433{
434 Assert(pRange->CTXALLSUFF(pfnWriteCallback) || !pRange->pfnWriteCallbackR3);
435
436 /*
437 * Get data to write from second parameter,
438 * and call the callback to write it.
439 */
440 unsigned cb = 0;
441 uint32_t u32Data = 0;
442 bool fRc = iomGetRegImmData(pCpu, &pCpu->param2, pRegFrame, &u32Data, &cb);
443 AssertMsg(fRc, ("Failed to get reg/imm port number!\n")); NOREF(fRc);
444
445 int rc = iomMMIODoWrite(pVM, pRange, GCPhysFault, &u32Data, cb);
446 if (rc == VINF_SUCCESS)
447 iomMMIOStatLength(pVM, cb);
448 return rc;
449}
450
451
452/** Wrapper for reading virtual memory. */
453DECLINLINE(int) iomRamRead(PVM pVM, void *pDest, RTGCPTR GCSrc, uint32_t cb)
454{
455#ifdef IN_GC
456 return MMGCRamReadNoTrapHandler(pDest, GCSrc, cb);
457#else
458 return PGMPhysReadGCPtrSafe(pVM, pDest, GCSrc, cb);
459#endif
460}
461
462
463/** Wrapper for writing virtual memory. */
464DECLINLINE(int) iomRamWrite(PVM pVM, RTGCPTR GCDest, void *pSrc, uint32_t cb)
465{
466#ifdef IN_GC
467 return MMGCRamWriteNoTrapHandler(GCDest, pSrc, cb);
468#else
469 return PGMPhysWriteGCPtrSafe(pVM, GCDest, pSrc, cb);
470#endif
471}
472
473
474#ifdef iom_MOVS_SUPPORT
475/**
476 * [REP] MOVSB
477 * [REP] MOVSW
478 * [REP] MOVSD
479 *
480 * Restricted implementation.
481 *
482 *
483 * @returns VBox status code.
484 *
485 * @param pVM The virtual machine (GC pointer ofcourse).
486 * @param uErrorCode CPU Error code.
487 * @param pRegFrame Trap register frame.
488 * @param GCPhysFault The GC physical address corresponding to pvFault.
489 * @param pCpu Disassembler CPU state.
490 * @param pRange Pointer MMIO range.
491 */
492static int iomInterpretMOVS(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
493{
494 /*
495 * We do not support segment prefixes or REPNE.
496 */
497 if (pCpu->prefix & (PREFIX_SEG | PREFIX_REPNE))
498 return VINF_IOM_HC_MMIO_READ_WRITE; /** @todo -> interpret whatever. */
499
500
501 /*
502 * Get bytes/words/dwords count to copy.
503 */
504 uint32_t cTransfers = 1;
505 if (pCpu->prefix & PREFIX_REP)
506 {
507 cTransfers = pRegFrame->ecx;
508 if (!SELMIsSelector32Bit(pVM, pRegFrame->eflags, pRegFrame->cs, &pRegFrame->csHid))
509 cTransfers &= 0xffff;
510
511 if (!cTransfers)
512 return VINF_SUCCESS;
513 }
514
515 /* Get the current privilege level. */
516 uint32_t cpl = CPUMGetGuestCPL(pVM, pRegFrame);
517
518 /*
519 * Get data size.
520 */
521 unsigned cb = DISGetParamSize(pCpu, &pCpu->param1);
522 Assert(cb);
523 int offIncrement = pRegFrame->eflags.Bits.u1DF ? -(signed)cb : (signed)cb;
524
525#ifdef VBOX_WITH_STATISTICS
526 if (pVM->iom.s.cMovsMaxBytes < (cTransfers << SIZE_2_SHIFT(cb)))
527 pVM->iom.s.cMovsMaxBytes = cTransfers << SIZE_2_SHIFT(cb);
528#endif
529
530/** @todo re-evaluate on page boundraries. */
531
532 RTGCPHYS Phys = GCPhysFault;
533 int rc;
534 if (uErrorCode & X86_TRAP_PF_RW)
535 {
536 /*
537 * Write operation: [Mem] -> [MMIO]
538 * ds:esi (Virt Src) -> es:edi (Phys Dst)
539 */
540 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMovsToMMIO, a2);
541
542 /* Check callback. */
543 if (!pRange->CTXALLSUFF(pfnWriteCallback))
544 {
545 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsToMMIO, a2);
546 return VINF_IOM_HC_MMIO_WRITE;
547 }
548
549 /* Convert source address ds:esi. */
550 RTGCUINTPTR pu8Virt;
551 rc = SELMToFlatEx(pVM, pRegFrame->eflags, pRegFrame->ds, (RTGCPTR)pRegFrame->esi, &pRegFrame->dsHid,
552 SELMTOFLAT_FLAGS_HYPER | SELMTOFLAT_FLAGS_NO_PL,
553 (PRTGCPTR)&pu8Virt, NULL);
554 if (VBOX_SUCCESS(rc))
555 {
556
557 /* Access verification first; we currently can't recover properly from traps inside this instruction */
558 rc = PGMVerifyAccess(pVM, pu8Virt, cTransfers * cb, (cpl == 3) ? X86_PTE_US : 0);
559 if (rc != VINF_SUCCESS)
560 {
561 Log(("MOVS will generate a trap -> recompiler, rc=%d\n", rc));
562 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsToMMIO, a2);
563 return VINF_EM_RAW_EMULATE_INSTR;
564 }
565
566#ifdef IN_GC
567 MMGCRamRegisterTrapHandler(pVM);
568#endif
569
570 /* copy loop. */
571 while (cTransfers)
572 {
573 uint32_t u32Data = 0;
574 rc = iomRamRead(pVM, &u32Data, (RTGCPTR)pu8Virt, cb);
575 if (rc != VINF_SUCCESS)
576 break;
577 rc = iomMMIODoWrite(pVM, pRange, Phys, &u32Data, cb);
578 if (rc != VINF_SUCCESS)
579 break;
580
581 pu8Virt += offIncrement;
582 Phys += offIncrement;
583 pRegFrame->esi += offIncrement;
584 pRegFrame->edi += offIncrement;
585 cTransfers--;
586 }
587#ifdef IN_GC
588 MMGCRamDeregisterTrapHandler(pVM);
589#endif
590 /* Update ecx. */
591 if (pCpu->prefix & PREFIX_REP)
592 pRegFrame->ecx = cTransfers;
593 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsToMMIO, a2);
594 }
595 else
596 rc = VINF_IOM_HC_MMIO_READ_WRITE;
597 }
598 else
599 {
600 /*
601 * Read operation: [MMIO] -> [mem] or [MMIO] -> [MMIO]
602 * ds:[eSI] (Phys Src) -> es:[eDI] (Virt Dst)
603 */
604 /* Check callback. */
605 if (!pRange->pfnReadCallback)
606 return VINF_IOM_HC_MMIO_READ;
607
608 /* Convert destination address. */
609 RTGCUINTPTR pu8Virt;
610 rc = SELMToFlatEx(pVM, pRegFrame->eflags, pRegFrame->es, (RTGCPTR)pRegFrame->edi, &pRegFrame->esHid,
611 SELMTOFLAT_FLAGS_HYPER | SELMTOFLAT_FLAGS_NO_PL,
612 (RTGCPTR *)&pu8Virt, NULL);
613 if (VBOX_FAILURE(rc))
614 return VINF_EM_RAW_GUEST_TRAP;
615
616 /* Check if destination address is MMIO. */
617 PIOMMMIORANGE pMMIODst;
618 RTGCPHYS PhysDst;
619 rc = PGMGstGetPage(pVM, (RTGCPTR)pu8Virt, NULL, &PhysDst);
620 PhysDst |= (RTGCUINTPTR)pu8Virt & PAGE_OFFSET_MASK;
621 if ( VBOX_SUCCESS(rc)
622 && (pMMIODst = iomMMIOGetRange(&pVM->iom.s, PhysDst)))
623 {
624 /*
625 * Extra: [MMIO] -> [MMIO]
626 */
627 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMovsMMIO, d);
628 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
629
630 if (!pMMIODst->CTXALLSUFF(pfnWriteCallback) && pMMIODst->pfnWriteCallbackR3)
631 {
632 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsMMIO, d);
633 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
634 return VINF_IOM_HC_MMIO_READ_WRITE;
635 }
636
637 /* copy loop. */
638 while (cTransfers)
639 {
640 uint32_t u32Data;
641 rc = iomMMIODoRead(pVM, pRange, Phys, &u32Data, cb);
642 if (rc != VINF_SUCCESS)
643 break;
644 rc = iomMMIODoWrite(pVM, pMMIODst, PhysDst, &u32Data, cb);
645 if (rc != VINF_SUCCESS)
646 break;
647
648 Phys += offIncrement;
649 PhysDst += offIncrement;
650 pRegFrame->esi += offIncrement;
651 pRegFrame->edi += offIncrement;
652 cTransfers--;
653 }
654 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsMMIO, d);
655 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
656 }
657 else
658 {
659 /*
660 * Normal: [MMIO] -> [Mem]
661 */
662 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
663
664 /* Access verification first; we currently can't recover properly from traps inside this instruction */
665 rc = PGMVerifyAccess(pVM, pu8Virt, cTransfers * cb, X86_PTE_RW | ((cpl == 3) ? X86_PTE_US : 0));
666 if (rc != VINF_SUCCESS)
667 {
668 Log(("MOVS will generate a trap -> recompiler, rc=%d\n", rc));
669 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
670 return VINF_EM_RAW_EMULATE_INSTR;
671 }
672
673 /* copy loop. */
674#ifdef IN_GC
675 MMGCRamRegisterTrapHandler(pVM);
676#endif
677 while (cTransfers)
678 {
679 uint32_t u32Data;
680 rc = iomMMIODoRead(pVM, pRange, Phys, &u32Data, cb);
681 if (rc != VINF_SUCCESS)
682 break;
683 rc = iomRamWrite(pVM, (RTGCPTR)pu8Virt, &u32Data, cb);
684 if (rc != VINF_SUCCESS)
685 {
686 Log(("iomRamWrite %08X size=%d failed with %d\n", pu8Virt, cb, rc));
687 break;
688 }
689
690 pu8Virt += offIncrement;
691 Phys += offIncrement;
692 pRegFrame->esi += offIncrement;
693 pRegFrame->edi += offIncrement;
694 cTransfers--;
695 }
696#ifdef IN_GC
697 MMGCRamDeregisterTrapHandler(pVM);
698#endif
699 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovsFromMMIO, c);
700 }
701
702 /* Update ecx on exit. */
703 if (pCpu->prefix & PREFIX_REP)
704 pRegFrame->ecx = cTransfers;
705 }
706
707 /* work statistics. */
708 if (rc == VINF_SUCCESS)
709 {
710 iomMMIOStatLength(pVM, cb);
711 }
712 return rc;
713}
714#endif
715
716
717
718/**
719 * [REP] STOSB
720 * [REP] STOSW
721 * [REP] STOSD
722 *
723 * Restricted implementation.
724 *
725 *
726 * @returns VBox status code.
727 *
728 * @param pVM The virtual machine (GC pointer ofcourse).
729 * @param pRegFrame Trap register frame.
730 * @param GCPhysFault The GC physical address corresponding to pvFault.
731 * @param pCpu Disassembler CPU state.
732 * @param pRange Pointer MMIO range.
733 */
734static int iomInterpretSTOS(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
735{
736 /*
737 * We do not support segment prefixes or REPNE..
738 */
739 if (pCpu->prefix & (PREFIX_SEG | PREFIX_REPNE))
740 return VINF_IOM_HC_MMIO_READ_WRITE; /** @todo -> REM instead of HC */
741
742 /*
743 * Get bytes/words/dwords count to copy.
744 */
745 uint32_t cTransfers = 1;
746 if (pCpu->prefix & PREFIX_REP)
747 {
748 cTransfers = pRegFrame->ecx;
749 if (!SELMIsSelector32Bit(pVM, pRegFrame->eflags, pRegFrame->cs, &pRegFrame->csHid))
750 cTransfers &= 0xffff;
751
752 if (!cTransfers)
753 return VINF_SUCCESS;
754 }
755
756 /*
757 * Get data size.
758 */
759 unsigned cb = DISGetParamSize(pCpu, &pCpu->param1);
760 Assert(cb);
761 int offIncrement = pRegFrame->eflags.Bits.u1DF ? -(signed)cb : (signed)cb;
762
763#ifdef VBOX_WITH_STATISTICS
764 if (pVM->iom.s.cStosMaxBytes < (cTransfers << SIZE_2_SHIFT(cb)))
765 pVM->iom.s.cStosMaxBytes = cTransfers << SIZE_2_SHIFT(cb);
766#endif
767
768
769 RTGCPHYS Phys = GCPhysFault;
770 uint32_t u32Data = pRegFrame->eax;
771 int rc;
772 if (pRange->CTXALLSUFF(pfnFillCallback))
773 {
774 /*
775 * Use the fill callback.
776 */
777 /** @todo pfnFillCallback must return number of bytes successfully written!!! */
778 if (offIncrement > 0)
779 {
780 /* addr++ variant. */
781 rc = pRange->CTXALLSUFF(pfnFillCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), Phys, u32Data, cb, cTransfers);
782 if (rc == VINF_SUCCESS)
783 {
784 /* Update registers. */
785 pRegFrame->edi += cTransfers << SIZE_2_SHIFT(cb);
786 if (pCpu->prefix & PREFIX_REP)
787 pRegFrame->ecx = 0;
788 }
789 }
790 else
791 {
792 /* addr-- variant. */
793 rc = pRange->CTXALLSUFF(pfnFillCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), (Phys - (cTransfers - 1)) << SIZE_2_SHIFT(cb), u32Data, cb, cTransfers);
794 if (rc == VINF_SUCCESS)
795 {
796 /* Update registers. */
797 pRegFrame->edi -= cTransfers << SIZE_2_SHIFT(cb);
798 if (pCpu->prefix & PREFIX_REP)
799 pRegFrame->ecx = 0;
800 }
801 }
802 }
803 else
804 {
805 /*
806 * Use the write callback.
807 */
808 Assert(pRange->CTXALLSUFF(pfnWriteCallback) || !pRange->pfnWriteCallbackR3);
809
810 /* fill loop. */
811 do
812 {
813 rc = iomMMIODoWrite(pVM, pRange, Phys, &u32Data, cb);
814 if (rc != VINF_SUCCESS)
815 break;
816
817 Phys += offIncrement;
818 pRegFrame->edi += offIncrement;
819 cTransfers--;
820 } while (cTransfers);
821
822 /* Update ecx on exit. */
823 if (pCpu->prefix & PREFIX_REP)
824 pRegFrame->ecx = cTransfers;
825 }
826
827 /*
828 * Work statistics and return.
829 */
830 if (rc == VINF_SUCCESS)
831 iomMMIOStatLength(pVM, cb);
832 return rc;
833}
834
835
836/**
837 * [REP] LODSB
838 * [REP] LODSW
839 * [REP] LODSD
840 *
841 * Restricted implementation.
842 *
843 *
844 * @returns VBox status code.
845 *
846 * @param pVM The virtual machine (GC pointer ofcourse).
847 * @param pRegFrame Trap register frame.
848 * @param GCPhysFault The GC physical address corresponding to pvFault.
849 * @param pCpu Disassembler CPU state.
850 * @param pRange Pointer MMIO range.
851 */
852static int iomInterpretLODS(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
853{
854 Assert(pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3);
855
856 /*
857 * We do not support segment prefixes or REP*.
858 */
859 if (pCpu->prefix & (PREFIX_SEG | PREFIX_REP | PREFIX_REPNE))
860 return VINF_IOM_HC_MMIO_READ_WRITE; /** @todo -> REM instead of HC */
861
862 /*
863 * Get data size.
864 */
865 unsigned cb = DISGetParamSize(pCpu, &pCpu->param2);
866 Assert(cb);
867 int offIncrement = pRegFrame->eflags.Bits.u1DF ? -(signed)cb : (signed)cb;
868
869 /*
870 * Perform read.
871 */
872 int rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &pRegFrame->eax, cb);
873 if (rc == VINF_SUCCESS)
874 pRegFrame->esi += offIncrement;
875
876 /*
877 * Work statistics and return.
878 */
879 if (rc == VINF_SUCCESS)
880 iomMMIOStatLength(pVM, cb);
881 return rc;
882}
883
884
885/**
886 * CMP [MMIO], reg|imm
887 * CMP reg|imm, [MMIO]
888 *
889 * Restricted implementation.
890 *
891 *
892 * @returns VBox status code.
893 *
894 * @param pVM The virtual machine (GC pointer ofcourse).
895 * @param pRegFrame Trap register frame.
896 * @param GCPhysFault The GC physical address corresponding to pvFault.
897 * @param pCpu Disassembler CPU state.
898 * @param pRange Pointer MMIO range.
899 */
900static int iomInterpretCMP(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
901{
902 Assert(pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3);
903
904 /*
905 * Get the operands.
906 */
907 unsigned cb = 0;
908 uint32_t uData1;
909 uint32_t uData2;
910 int rc;
911 if (iomGetRegImmData(pCpu, &pCpu->param1, pRegFrame, &uData1, &cb))
912 /* cmp reg, [MMIO]. */
913 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData2, cb);
914 else if (iomGetRegImmData(pCpu, &pCpu->param2, pRegFrame, &uData2, &cb))
915 /* cmp [MMIO], reg|imm. */
916 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData1, cb);
917 else
918 {
919 AssertMsgFailed(("Disassember CMP problem..\n"));
920 rc = VERR_IOM_MMIO_HANDLER_DISASM_ERROR;
921 }
922
923 if (rc == VINF_SUCCESS)
924 {
925 /* Emulate CMP and update guest flags. */
926 uint32_t eflags = EMEmulateCmp(uData1, uData2, cb);
927 pRegFrame->eflags.u32 = (pRegFrame->eflags.u32 & ~(X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF))
928 | (eflags & (X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF));
929 iomMMIOStatLength(pVM, cb);
930 }
931
932 return rc;
933}
934
935
936/**
937 * AND [MMIO], reg|imm
938 * AND reg, [MMIO]
939 *
940 * Restricted implementation.
941 *
942 *
943 * @returns VBox status code.
944 *
945 * @param pVM The virtual machine (GC pointer ofcourse).
946 * @param pRegFrame Trap register frame.
947 * @param GCPhysFault The GC physical address corresponding to pvFault.
948 * @param pCpu Disassembler CPU state.
949 * @param pRange Pointer MMIO range.
950 */
951static int iomInterpretAND(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
952{
953 unsigned cb = 0;
954 uint32_t uData1;
955 uint32_t uData2;
956 bool fAndWrite;
957 int rc;
958 if (iomGetRegImmData(pCpu, &pCpu->param1, pRegFrame, &uData1, &cb))
959 {
960 /* and reg, [MMIO]. */
961 Assert(pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3);
962 fAndWrite = false;
963 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData2, cb);
964 }
965 else if (iomGetRegImmData(pCpu, &pCpu->param2, pRegFrame, &uData2, &cb))
966 {
967 /* and [MMIO], reg|imm. */
968 fAndWrite = true;
969 if ( (pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3)
970 && (pRange->CTXALLSUFF(pfnWriteCallback) || !pRange->pfnWriteCallbackR3))
971 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData1, cb);
972 else
973 rc = VINF_IOM_HC_MMIO_READ_WRITE;
974 }
975 else
976 {
977 AssertMsgFailed(("Disassember AND problem..\n"));
978 return VERR_IOM_MMIO_HANDLER_DISASM_ERROR;
979 }
980
981 if (rc == VINF_SUCCESS)
982 {
983 /* Emulate AND and update guest flags. */
984 uint32_t eflags = EMEmulateAnd(&uData1, uData2, cb);
985 if (fAndWrite)
986 /* Store result to MMIO. */
987 rc = iomMMIODoWrite(pVM, pRange, GCPhysFault, &uData1, cb);
988 else
989 {
990 /* Store result to register. */
991 bool fRc = iomSaveDataToReg(pCpu, &pCpu->param1, pRegFrame, uData1);
992 AssertMsg(fRc, ("Failed to store register value!\n")); NOREF(fRc);
993 }
994 if (rc == VINF_SUCCESS)
995 {
996 /* Update guest's eflags and finish. */
997 pRegFrame->eflags.u32 = (pRegFrame->eflags.u32 & ~(X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF))
998 | (eflags & (X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF));
999 iomMMIOStatLength(pVM, cb);
1000 }
1001 }
1002
1003 return rc;
1004}
1005
1006
1007
1008/**
1009 * TEST [MMIO], reg|imm
1010 * TEST reg, [MMIO]
1011 *
1012 * Restricted implementation.
1013 *
1014 *
1015 * @returns VBox status code.
1016 *
1017 * @param pVM The virtual machine (GC pointer ofcourse).
1018 * @param pRegFrame Trap register frame.
1019 * @param GCPhysFault The GC physical address corresponding to pvFault.
1020 * @param pCpu Disassembler CPU state.
1021 * @param pRange Pointer MMIO range.
1022 */
1023static int iomInterpretTEST(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
1024{
1025 Assert(pRange->CTXALLSUFF(pfnReadCallback) || !pRange->pfnReadCallbackR3);
1026
1027 unsigned cb = 0;
1028 uint32_t uData1;
1029 uint32_t uData2;
1030 int rc;
1031
1032 if (iomGetRegImmData(pCpu, &pCpu->param1, pRegFrame, &uData1, &cb))
1033 {
1034 /* and test, [MMIO]. */
1035 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData2, cb);
1036 }
1037 else if (iomGetRegImmData(pCpu, &pCpu->param2, pRegFrame, &uData2, &cb))
1038 {
1039 /* test [MMIO], reg|imm. */
1040 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData1, cb);
1041 }
1042 else
1043 {
1044 AssertMsgFailed(("Disassember TEST problem..\n"));
1045 return VERR_IOM_MMIO_HANDLER_DISASM_ERROR;
1046 }
1047
1048 if (rc == VINF_SUCCESS)
1049 {
1050 /* Emulate TEST (=AND without write back) and update guest EFLAGS. */
1051 uint32_t eflags = EMEmulateAnd(&uData1, uData2, cb);
1052 pRegFrame->eflags.u32 = (pRegFrame->eflags.u32 & ~(X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF))
1053 | (eflags & (X86_EFL_CF | X86_EFL_PF | X86_EFL_AF | X86_EFL_ZF | X86_EFL_SF | X86_EFL_OF));
1054 iomMMIOStatLength(pVM, cb);
1055 }
1056
1057 return rc;
1058}
1059
1060/**
1061 * XCHG [MMIO], reg
1062 * XCHG reg, [MMIO]
1063 *
1064 * Restricted implementation.
1065 *
1066 *
1067 * @returns VBox status code.
1068 *
1069 * @param pVM The virtual machine (GC pointer ofcourse).
1070 * @param pRegFrame Trap register frame.
1071 * @param GCPhysFault The GC physical address corresponding to pvFault.
1072 * @param pCpu Disassembler CPU state.
1073 * @param pRange Pointer MMIO range.
1074 */
1075static int iomInterpretXCHG(PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPHYS GCPhysFault, PDISCPUSTATE pCpu, PIOMMMIORANGE pRange)
1076{
1077 /* Check for read & write handlers since IOMMMIOHandler doesn't cover this. */
1078 if ( (!pRange->CTXALLSUFF(pfnReadCallback) && pRange->pfnReadCallbackR3)
1079 || (!pRange->CTXALLSUFF(pfnWriteCallback) && pRange->pfnWriteCallbackR3))
1080 return VINF_IOM_HC_MMIO_READ_WRITE;
1081
1082 int rc;
1083 unsigned cb = 0;
1084 uint32_t uData1;
1085 uint32_t uData2;
1086 if (iomGetRegImmData(pCpu, &pCpu->param1, pRegFrame, &uData1, &cb))
1087 {
1088 /* xchg reg, [MMIO]. */
1089 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData2, cb);
1090 if (rc == VINF_SUCCESS)
1091 {
1092 /* Store result to MMIO. */
1093 rc = iomMMIODoWrite(pVM, pRange, GCPhysFault, &uData1, cb);
1094
1095 if (rc == VINF_SUCCESS)
1096 {
1097 /* Store result to register. */
1098 bool fRc = iomSaveDataToReg(pCpu, &pCpu->param1, pRegFrame, uData2);
1099 AssertMsg(fRc, ("Failed to store register value!\n")); NOREF(fRc);
1100 }
1101 else
1102 Assert(rc == VINF_IOM_HC_MMIO_WRITE || rc == VINF_PATM_HC_MMIO_PATCH_WRITE);
1103 }
1104 else
1105 Assert(rc == VINF_IOM_HC_MMIO_READ || rc == VINF_PATM_HC_MMIO_PATCH_READ);
1106 }
1107 else if (iomGetRegImmData(pCpu, &pCpu->param2, pRegFrame, &uData2, &cb))
1108 {
1109 /* xchg [MMIO], reg. */
1110 rc = iomMMIODoRead(pVM, pRange, GCPhysFault, &uData1, cb);
1111 if (rc == VINF_SUCCESS)
1112 {
1113 /* Store result to MMIO. */
1114 rc = iomMMIODoWrite(pVM, pRange, GCPhysFault, &uData2, cb);
1115 if (rc == VINF_SUCCESS)
1116 {
1117 /* Store result to register. */
1118 bool fRc = iomSaveDataToReg(pCpu, &pCpu->param2, pRegFrame, uData1);
1119 AssertMsg(fRc, ("Failed to store register value!\n")); NOREF(fRc);
1120 }
1121 else
1122 Assert(rc == VINF_IOM_HC_MMIO_WRITE || rc == VINF_PATM_HC_MMIO_PATCH_WRITE);
1123 }
1124 else
1125 Assert(rc == VINF_IOM_HC_MMIO_READ || rc == VINF_PATM_HC_MMIO_PATCH_READ);
1126 }
1127 else
1128 {
1129 AssertMsgFailed(("Disassember XCHG problem..\n"));
1130 rc = VERR_IOM_MMIO_HANDLER_DISASM_ERROR;
1131 }
1132 return rc;
1133}
1134
1135
1136/**
1137 * \#PF Handler callback for MMIO ranges.
1138 * Note: we are on ring0 in Hypervisor and interrupts are disabled.
1139 *
1140 * @returns VBox status code (appropriate for GC return).
1141 * @param pVM VM Handle.
1142 * @param uErrorCode CPU Error code.
1143 * @param pCtxCore Trap register frame.
1144 * @param pvFault The fault address (cr2).
1145 * @param GCPhysFault The GC physical address corresponding to pvFault.
1146 * @param pvUser Pointer to the MMIO ring-3 range entry.
1147 */
1148IOMDECL(int) IOMMMIOHandler(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pCtxCore, void *pvFault, RTGCPHYS GCPhysFault, void *pvUser)
1149{
1150 STAM_PROFILE_START(&pVM->iom.s.StatGCMMIOHandler, a);
1151 Log3(("IOMMMIOHandler: GCPhys=%RGp uErr=%#x pvFault=%p eip=%RGv\n",
1152 GCPhysFault, uErrorCode, pvFault, pCtxCore->eip));
1153
1154 PIOMMMIORANGE pRange = (PIOMMMIORANGE)pvUser;
1155 Assert(pRange);
1156 Assert(pRange == iomMMIOGetRange(&pVM->iom.s, GCPhysFault));
1157
1158#ifdef VBOX_WITH_STATISTICS
1159 /*
1160 * Locate the statistics, if > PAGE_SIZE we'll use the first byte for everything.
1161 */
1162 PIOMMMIOSTATS pStats = iomMMIOGetStats(&pVM->iom.s, GCPhysFault, pRange);
1163 if (!pStats)
1164 {
1165# ifdef IN_RING3
1166 return VERR_NO_MEMORY;
1167# else
1168 STAM_PROFILE_STOP(&pVM->iom.s.StatGCMMIOHandler, a);
1169 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIOFailures);
1170 return uErrorCode & X86_TRAP_PF_RW ? VINF_IOM_HC_MMIO_WRITE : VINF_IOM_HC_MMIO_READ;
1171# endif
1172 }
1173#endif
1174
1175#ifndef IN_RING3
1176 /*
1177 * Should we defer the request right away?
1178 */
1179 if (uErrorCode & X86_TRAP_PF_RW
1180 ? !pRange->CTXALLSUFF(pfnWriteCallback) && pRange->pfnWriteCallbackR3
1181 : !pRange->CTXALLSUFF(pfnReadCallback) && pRange->pfnReadCallbackR3)
1182 {
1183# ifdef VBOX_WITH_STATISTICS
1184 if (uErrorCode & X86_TRAP_PF_RW)
1185 STAM_COUNTER_INC(&pStats->CTXALLMID(Write,ToR3));
1186 else
1187 STAM_COUNTER_INC(&pStats->CTXALLMID(Read,ToR3));
1188# endif
1189
1190 STAM_PROFILE_STOP(&pVM->iom.s.StatGCMMIOHandler, a);
1191 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIOFailures);
1192 return uErrorCode & X86_TRAP_PF_RW ? VINF_IOM_HC_MMIO_WRITE : VINF_IOM_HC_MMIO_READ;
1193 }
1194#endif /* !IN_RING3 */
1195
1196 /*
1197 * Disassemble the instruction and interprete it.
1198 */
1199 DISCPUSTATE Cpu;
1200 unsigned cbOp;
1201 int rc = EMInterpretDisasOne(pVM, pCtxCore, &Cpu, &cbOp);
1202 AssertRCReturn(rc, rc);
1203 switch (Cpu.pCurInstr->opcode)
1204 {
1205 case OP_MOV:
1206 case OP_MOVZX:
1207 case OP_MOVSX:
1208 {
1209 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMov, b);
1210 if (uErrorCode & X86_TRAP_PF_RW)
1211 rc = iomInterpretMOVxXWrite(pVM, pCtxCore, &Cpu, pRange, GCPhysFault);
1212 else
1213 rc = iomInterpretMOVxXRead(pVM, pCtxCore, &Cpu, pRange, GCPhysFault);
1214 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMov, b);
1215 break;
1216 }
1217
1218
1219#ifdef iom_MOVS_SUPPORT
1220 case OP_MOVSB:
1221 case OP_MOVSWD:
1222 STAM_PROFILE_START(&pVM->iom.s.StatGCInstMovs, c);
1223 rc = iomInterpretMOVS(pVM, uErrorCode, pCtxCore, GCPhysFault, &Cpu, pRange);
1224 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstMovs, c);
1225 break;
1226#endif
1227
1228 case OP_STOSB:
1229 case OP_STOSWD:
1230 Assert(uErrorCode & X86_TRAP_PF_RW);
1231 STAM_PROFILE_START(&pVM->iom.s.StatGCInstStos, d);
1232 rc = iomInterpretSTOS(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1233 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstStos, d);
1234 break;
1235
1236 case OP_LODSB:
1237 case OP_LODSWD:
1238 Assert(!(uErrorCode & X86_TRAP_PF_RW));
1239 STAM_PROFILE_START(&pVM->iom.s.StatGCInstLods, e);
1240 rc = iomInterpretLODS(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1241 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstLods, e);
1242 break;
1243
1244 case OP_CMP:
1245 Assert(!(uErrorCode & X86_TRAP_PF_RW));
1246 STAM_PROFILE_START(&pVM->iom.s.StatGCInstCmp, f);
1247 rc = iomInterpretCMP(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1248 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstCmp, f);
1249 break;
1250
1251 case OP_AND:
1252 STAM_PROFILE_START(&pVM->iom.s.StatGCInstAnd, g);
1253 rc = iomInterpretAND(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1254 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstAnd, g);
1255 break;
1256
1257 case OP_TEST:
1258 Assert(!(uErrorCode & X86_TRAP_PF_RW));
1259 STAM_PROFILE_START(&pVM->iom.s.StatGCInstTest, h);
1260 rc = iomInterpretTEST(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1261 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstTest, h);
1262 break;
1263
1264 case OP_XCHG:
1265 STAM_PROFILE_START(&pVM->iom.s.StatGCInstXchg, i);
1266 rc = iomInterpretXCHG(pVM, pCtxCore, GCPhysFault, &Cpu, pRange);
1267 STAM_PROFILE_STOP(&pVM->iom.s.StatGCInstXchg, i);
1268 break;
1269
1270
1271 /*
1272 * The instruction isn't supported. Hand it on to ring-3.
1273 */
1274 default:
1275 STAM_COUNTER_INC(&pVM->iom.s.StatGCInstOther);
1276 rc = (uErrorCode & X86_TRAP_PF_RW) ? VINF_IOM_HC_MMIO_WRITE : VINF_IOM_HC_MMIO_READ;
1277 break;
1278 }
1279
1280 /*
1281 * On success advance EIP.
1282 */
1283 if (rc == VINF_SUCCESS)
1284 pCtxCore->eip += cbOp;
1285 else
1286 {
1287 STAM_COUNTER_INC(&pVM->iom.s.StatGCMMIOFailures);
1288#if defined(VBOX_WITH_STATISTICS) && !defined(IN_RING3)
1289 switch (rc)
1290 {
1291 case VINF_IOM_HC_MMIO_READ:
1292 case VINF_IOM_HC_MMIO_READ_WRITE:
1293 STAM_COUNTER_INC(&pStats->CTXALLMID(Read,ToR3));
1294 break;
1295 case VINF_IOM_HC_MMIO_WRITE:
1296 STAM_COUNTER_INC(&pStats->CTXALLMID(Write,ToR3));
1297 break;
1298 }
1299#endif
1300 }
1301
1302 STAM_PROFILE_STOP(&pVM->iom.s.StatGCMMIOHandler, a);
1303 return rc;
1304}
1305
1306
1307/**
1308 * Reads a MMIO register.
1309 *
1310 * @returns VBox status code.
1311 *
1312 * @param pVM VM handle.
1313 * @param GCPhys The physical address to read.
1314 * @param pu32Value Where to store the value read.
1315 * @param cbValue The size of the register to read in bytes. 1, 2 or 4 bytes.
1316 */
1317IOMDECL(int) IOMMMIORead(PVM pVM, RTGCPHYS GCPhys, uint32_t *pu32Value, size_t cbValue)
1318{
1319 /*
1320 * Lookup the current context range node and statistics.
1321 */
1322 PIOMMMIORANGE pRange = iomMMIOGetRange(&pVM->iom.s, GCPhys);
1323 AssertMsgReturn(pRange,
1324 ("Handlers and page tables are out of sync or something! GCPhys=%VGp cbValue=%d\n", GCPhys, cbValue),
1325 VERR_INTERNAL_ERROR);
1326#ifdef VBOX_WITH_STATISTICS
1327 PIOMMMIOSTATS pStats = iomMMIOGetStats(&pVM->iom.s, GCPhys, pRange);
1328 if (!pStats)
1329# ifdef IN_RING3
1330 return VERR_NO_MEMORY;
1331# else
1332 return VINF_IOM_HC_MMIO_READ;
1333# endif
1334#endif /* VBOX_WITH_STATISTICS */
1335 if (pRange->CTXALLSUFF(pfnReadCallback))
1336 {
1337 /*
1338 * Perform the read and deal with the result.
1339 */
1340#ifdef VBOX_WITH_STATISTICS
1341 if (pStats)
1342 STAM_PROFILE_ADV_START(&pStats->CTXALLSUFF(ProfRead), a);
1343#endif
1344 int rc = pRange->CTXALLSUFF(pfnReadCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), GCPhys, pu32Value, cbValue);
1345#ifdef VBOX_WITH_STATISTICS
1346 if (pStats)
1347 STAM_PROFILE_ADV_STOP(&pStats->CTXALLSUFF(ProfRead), a);
1348 if (pStats && rc != VINF_IOM_HC_MMIO_READ)
1349 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Read));
1350#endif
1351 switch (rc)
1352 {
1353 case VINF_SUCCESS:
1354 default:
1355 Log4(("IOMMMIORead: GCPhys=%RGp *pu32=%08RX32 cb=%d rc=%Vrc\n", GCPhys, *pu32Value, cbValue, rc));
1356 return rc;
1357
1358 case VINF_IOM_MMIO_UNUSED_00:
1359 switch (cbValue)
1360 {
1361 case 1: *(uint8_t *)pu32Value = 0x00; break;
1362 case 2: *(uint16_t *)pu32Value = 0x0000; break;
1363 case 4: *(uint32_t *)pu32Value = 0x00000000; break;
1364 default: AssertReleaseMsgFailed(("cbValue=%d GCPhys=%VGp\n", cbValue, GCPhys)); break;
1365 }
1366 Log4(("IOMMMIORead: GCPhys=%RGp *pu32=%08RX32 cb=%d rc=%Vrc\n", GCPhys, *pu32Value, cbValue, rc));
1367 return VINF_SUCCESS;
1368
1369 case VINF_IOM_MMIO_UNUSED_FF:
1370 switch (cbValue)
1371 {
1372 case 1: *(uint8_t *)pu32Value = 0xff; break;
1373 case 2: *(uint16_t *)pu32Value = 0xffff; break;
1374 case 4: *(uint32_t *)pu32Value = 0xffffffff; break;
1375 default: AssertReleaseMsgFailed(("cbValue=%d GCPhys=%VGp\n", cbValue, GCPhys)); break;
1376 }
1377 Log4(("IOMMMIORead: GCPhys=%RGp *pu32=%08RX32 cb=%d rc=%Vrc\n", GCPhys, *pu32Value, cbValue, rc));
1378 return VINF_SUCCESS;
1379 }
1380 }
1381#ifndef IN_RING3
1382 if (pRange->pfnReadCallbackR3)
1383 {
1384 STAM_COUNTER_INC(&pStats->CTXALLMID(Read,ToR3));
1385 return VINF_IOM_HC_MMIO_READ;
1386 }
1387#endif
1388
1389 /*
1390 * Lookup the ring-3 range.
1391 */
1392#ifdef VBOX_WITH_STATISTICS
1393 if (pStats)
1394 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Read));
1395#endif
1396 *pu32Value = 0;
1397 Log4(("IOMMMIORead: GCPhys=%RGp *pu32=%08RX32 cb=%d rc=VINF_SUCCESS\n", GCPhys, *pu32Value, cbValue));
1398 return VINF_SUCCESS;
1399}
1400
1401
1402/**
1403 * Writes to a MMIO register.
1404 *
1405 * @returns VBox status code.
1406 *
1407 * @param pVM VM handle.
1408 * @param GCPhys The physical address to write to.
1409 * @param u32Value The value to write.
1410 * @param cbValue The size of the register to read in bytes. 1, 2 or 4 bytes.
1411 */
1412IOMDECL(int) IOMMMIOWrite(PVM pVM, RTGCPHYS GCPhys, uint32_t u32Value, size_t cbValue)
1413{
1414 /*
1415 * Lookup the current context range node.
1416 */
1417 PIOMMMIORANGE pRange = iomMMIOGetRange(&pVM->iom.s, GCPhys);
1418 AssertMsgReturn(pRange,
1419 ("Handlers and page tables are out of sync or something! GCPhys=%VGp cbValue=%d\n", GCPhys, cbValue),
1420 VERR_INTERNAL_ERROR);
1421#ifdef VBOX_WITH_STATISTICS
1422 PIOMMMIOSTATS pStats = iomMMIOGetStats(&pVM->iom.s, GCPhys, pRange);
1423 if (!pStats)
1424# ifdef IN_RING3
1425 return VERR_NO_MEMORY;
1426# else
1427 return VINF_IOM_HC_MMIO_WRITE;
1428# endif
1429#endif /* VBOX_WITH_STATISTICS */
1430
1431 /*
1432 * Perform the write if there's a write handler. R0/GC may have
1433 * to defer it to ring-3.
1434 */
1435 if (pRange->CTXALLSUFF(pfnWriteCallback))
1436 {
1437#ifdef VBOX_WITH_STATISTICS
1438 if (pStats)
1439 STAM_PROFILE_ADV_START(&pStats->CTXALLSUFF(ProfWrite), a);
1440#endif
1441 int rc = pRange->CTXALLSUFF(pfnWriteCallback)(pRange->CTXALLSUFF(pDevIns), pRange->CTXALLSUFF(pvUser), GCPhys, &u32Value, cbValue);
1442#ifdef VBOX_WITH_STATISTICS
1443 if (pStats)
1444 STAM_PROFILE_ADV_STOP(&pStats->CTXALLSUFF(ProfWrite), a);
1445 if (pStats && rc != VINF_IOM_HC_MMIO_WRITE)
1446 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Write));
1447#endif
1448 Log4(("IOMMMIOWrite: GCPhys=%RGp u32=%08RX32 cb=%d rc=%Vrc\n", GCPhys, u32Value, cbValue, rc));
1449 return rc;
1450 }
1451#ifndef IN_RING3
1452 if (pRange->pfnWriteCallbackR3)
1453 {
1454 STAM_COUNTER_INC(&pStats->CTXALLMID(Write,ToR3));
1455 return VINF_IOM_HC_MMIO_WRITE;
1456 }
1457#endif
1458
1459 /*
1460 * No write handler, nothing to do.
1461 */
1462#ifdef VBOX_WITH_STATISTICS
1463 if (pStats)
1464 STAM_COUNTER_INC(&pStats->CTXALLSUFF(Write));
1465#endif
1466 Log4(("IOMMMIOWrite: GCPhys=%RGp u32=%08RX32 cb=%d rc=%Vrc\n", GCPhys, u32Value, cbValue, VINF_SUCCESS));
1467 return VINF_SUCCESS;
1468}
1469
1470
1471/**
1472 * [REP*] INSB/INSW/INSD
1473 * ES:EDI,DX[,ECX]
1474 *
1475 * @remark Assumes caller checked the access privileges (IOMInterpretCheckPortIOAccess)
1476 *
1477 * @returns Strict VBox status code. Informational status codes other than the one documented
1478 * here are to be treated as internal failure. Use IOM_SUCCESS() to check for success.
1479 * @retval VINF_SUCCESS Success.
1480 * @retval VINF_EM_FIRST-VINF_EM_LAST Success with some exceptions (see IOM_SUCCESS()), the
1481 * status code must be passed on to EM.
1482 * @retval VINF_IOM_HC_IOPORT_READ Defer the read to ring-3. (R0/GC only)
1483 * @retval VINF_EM_RAW_EMULATE_INSTR Defer the read to the REM.
1484 * @retval VINF_EM_RAW_GUEST_TRAP The exception was left pending. (TRPMRaiseXcptErr)
1485 * @retval VINF_TRPM_XCPT_DISPATCHED The exception was raised and dispatched for raw-mode execution. (TRPMRaiseXcptErr)
1486 * @retval VINF_EM_RESCHEDULE_REM The exception was dispatched and cannot be executed in raw-mode. (TRPMRaiseXcptErr)
1487 *
1488 * @param pVM The virtual machine (GC pointer ofcourse).
1489 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
1490 * @param uPort IO Port
1491 * @param uPrefix IO instruction prefix
1492 * @param cbTransfer Size of transfer unit
1493 */
1494IOMDECL(int) IOMInterpretINSEx(PVM pVM, PCPUMCTXCORE pRegFrame, uint32_t uPort, uint32_t uPrefix, uint32_t cbTransfer)
1495{
1496#ifdef VBOX_WITH_STATISTICS
1497 STAM_COUNTER_INC(&pVM->iom.s.StatGCInstIns);
1498#endif
1499
1500 /*
1501 * We do not support REPNE or decrementing destination
1502 * pointer. Segment prefixes are deliberately ignored, as per the instruction specification.
1503 */
1504 if ( (uPrefix & PREFIX_REPNE)
1505 || pRegFrame->eflags.Bits.u1DF)
1506 return VINF_EM_RAW_EMULATE_INSTR;
1507
1508 /*
1509 * Get bytes/words/dwords count to transfer.
1510 */
1511 RTGCUINTREG cTransfers = 1;
1512 if (uPrefix & PREFIX_REP)
1513 {
1514 cTransfers = pRegFrame->ecx;
1515
1516 if (!SELMIsSelector32Bit(pVM, pRegFrame->eflags, pRegFrame->cs, &pRegFrame->csHid))
1517 cTransfers &= 0xffff;
1518
1519 if (!cTransfers)
1520 return VINF_SUCCESS;
1521 }
1522
1523 /* Convert destination address es:edi. */
1524 RTGCPTR GCPtrDst;
1525 int rc = SELMToFlatEx(pVM, pRegFrame->eflags, pRegFrame->es, (RTGCPTR)pRegFrame->edi, &pRegFrame->esHid,
1526 SELMTOFLAT_FLAGS_HYPER | SELMTOFLAT_FLAGS_NO_PL,
1527 &GCPtrDst, NULL);
1528 if (VBOX_FAILURE(rc))
1529 {
1530 Log(("INS destination address conversion failed -> fallback, rc=%d\n", rc));
1531 return VINF_EM_RAW_EMULATE_INSTR;
1532 }
1533
1534 /* Access verification first; we can't recover from traps inside this instruction, as the port read cannot be repeated. */
1535 uint32_t cpl = CPUMGetGuestCPL(pVM, pRegFrame);
1536
1537 rc = PGMVerifyAccess(pVM, (RTGCUINTPTR)GCPtrDst, cTransfers * cbTransfer,
1538 X86_PTE_RW | ((cpl == 3) ? X86_PTE_US : 0));
1539 if (rc != VINF_SUCCESS)
1540 {
1541 Log(("INS will generate a trap -> fallback, rc=%d\n", rc));
1542 return VINF_EM_RAW_EMULATE_INSTR;
1543 }
1544
1545 Log(("IOM: rep ins%d port %#x count %d\n", cbTransfer * 8, uPort, cTransfers));
1546 if (cTransfers > 1)
1547 {
1548 /* If the device supports string transfers, ask it to do as
1549 * much as it wants. The rest is done with single-word transfers. */
1550 const RTGCUINTREG cTransfersOrg = cTransfers;
1551 rc = IOMIOPortReadString(pVM, uPort, &GCPtrDst, &cTransfers, cbTransfer);
1552 AssertRC(rc); Assert(cTransfers <= cTransfersOrg);
1553 pRegFrame->edi += (cTransfersOrg - cTransfers) * cbTransfer;
1554 }
1555
1556#ifdef IN_GC
1557 MMGCRamRegisterTrapHandler(pVM);
1558#endif
1559
1560 while (cTransfers && rc == VINF_SUCCESS)
1561 {
1562 uint32_t u32Value;
1563 rc = IOMIOPortRead(pVM, uPort, &u32Value, cbTransfer);
1564 if (!IOM_SUCCESS(rc))
1565 break;
1566 int rc2 = iomRamWrite(pVM, GCPtrDst, &u32Value, cbTransfer);
1567 Assert(rc2 == VINF_SUCCESS); NOREF(rc2);
1568 GCPtrDst = (RTGCPTR)((RTGCUINTPTR)GCPtrDst + cbTransfer);
1569 pRegFrame->edi += cbTransfer;
1570 cTransfers--;
1571 }
1572#ifdef IN_GC
1573 MMGCRamDeregisterTrapHandler(pVM);
1574#endif
1575
1576 /* Update ecx on exit. */
1577 if (uPrefix & PREFIX_REP)
1578 pRegFrame->ecx = cTransfers;
1579
1580 AssertMsg(rc == VINF_SUCCESS || rc == VINF_IOM_HC_IOPORT_READ || (rc >= VINF_EM_FIRST && rc <= VINF_EM_LAST) || VBOX_FAILURE(rc), ("%Vrc\n", rc));
1581 return rc;
1582}
1583
1584
1585/**
1586 * [REP*] INSB/INSW/INSD
1587 * ES:EDI,DX[,ECX]
1588 *
1589 * @returns Strict VBox status code. Informational status codes other than the one documented
1590 * here are to be treated as internal failure. Use IOM_SUCCESS() to check for success.
1591 * @retval VINF_SUCCESS Success.
1592 * @retval VINF_EM_FIRST-VINF_EM_LAST Success with some exceptions (see IOM_SUCCESS()), the
1593 * status code must be passed on to EM.
1594 * @retval VINF_IOM_HC_IOPORT_READ Defer the read to ring-3. (R0/GC only)
1595 * @retval VINF_EM_RAW_EMULATE_INSTR Defer the read to the REM.
1596 * @retval VINF_EM_RAW_GUEST_TRAP The exception was left pending. (TRPMRaiseXcptErr)
1597 * @retval VINF_TRPM_XCPT_DISPATCHED The exception was raised and dispatched for raw-mode execution. (TRPMRaiseXcptErr)
1598 * @retval VINF_EM_RESCHEDULE_REM The exception was dispatched and cannot be executed in raw-mode. (TRPMRaiseXcptErr)
1599 *
1600 * @param pVM The virtual machine (GC pointer ofcourse).
1601 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
1602 * @param pCpu Disassembler CPU state.
1603 */
1604IOMDECL(int) IOMInterpretINS(PVM pVM, PCPUMCTXCORE pRegFrame, PDISCPUSTATE pCpu)
1605{
1606 /*
1607 * Get port number directly from the register (no need to bother the
1608 * disassembler). And get the I/O register size from the opcode / prefix.
1609 */
1610 RTIOPORT Port = pRegFrame->edx & 0xffff;
1611 unsigned cb = 0;
1612 if (pCpu->pCurInstr->opcode == OP_INSB)
1613 cb = 1;
1614 else
1615 cb = pCpu->opmode == CPUMODE_32BIT ? 4 : 2;
1616
1617 int rc = IOMInterpretCheckPortIOAccess(pVM, pRegFrame, Port, cb);
1618 if (RT_UNLIKELY(rc != VINF_SUCCESS))
1619 {
1620 AssertMsg(rc == VINF_EM_RAW_GUEST_TRAP || rc == VINF_TRPM_XCPT_DISPATCHED || rc == VINF_TRPM_XCPT_DISPATCHED || VBOX_FAILURE(rc), ("%Vrc\n", rc));
1621 return rc;
1622 }
1623
1624 return IOMInterpretINSEx(pVM, pRegFrame, Port, pCpu->prefix, cb);
1625}
1626
1627
1628/**
1629 * [REP*] OUTSB/OUTSW/OUTSD
1630 * DS:ESI,DX[,ECX]
1631 *
1632 * @remark Assumes caller checked the access privileges (IOMInterpretCheckPortIOAccess)
1633 *
1634 * @returns Strict VBox status code. Informational status codes other than the one documented
1635 * here are to be treated as internal failure. Use IOM_SUCCESS() to check for success.
1636 * @retval VINF_SUCCESS Success.
1637 * @retval VINF_EM_FIRST-VINF_EM_LAST Success with some exceptions (see IOM_SUCCESS()), the
1638 * status code must be passed on to EM.
1639 * @retval VINF_IOM_HC_IOPORT_WRITE Defer the write to ring-3. (R0/GC only)
1640 * @retval VINF_EM_RAW_GUEST_TRAP The exception was left pending. (TRPMRaiseXcptErr)
1641 * @retval VINF_TRPM_XCPT_DISPATCHED The exception was raised and dispatched for raw-mode execution. (TRPMRaiseXcptErr)
1642 * @retval VINF_EM_RESCHEDULE_REM The exception was dispatched and cannot be executed in raw-mode. (TRPMRaiseXcptErr)
1643 *
1644 * @param pVM The virtual machine (GC pointer ofcourse).
1645 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
1646 * @param uPort IO Port
1647 * @param uPrefix IO instruction prefix
1648 * @param cbTransfer Size of transfer unit
1649 */
1650IOMDECL(int) IOMInterpretOUTSEx(PVM pVM, PCPUMCTXCORE pRegFrame, uint32_t uPort, uint32_t uPrefix, uint32_t cbTransfer)
1651{
1652#ifdef VBOX_WITH_STATISTICS
1653 STAM_COUNTER_INC(&pVM->iom.s.StatGCInstOuts);
1654#endif
1655
1656 /*
1657 * We do not support segment prefixes, REPNE or
1658 * decrementing source pointer.
1659 */
1660 if ( (uPrefix & (PREFIX_SEG | PREFIX_REPNE))
1661 || pRegFrame->eflags.Bits.u1DF)
1662 return VINF_EM_RAW_EMULATE_INSTR;
1663
1664 /*
1665 * Get bytes/words/dwords count to transfer.
1666 */
1667 RTGCUINTREG cTransfers = 1;
1668 if (uPrefix & PREFIX_REP)
1669 {
1670 cTransfers = pRegFrame->ecx;
1671 if (!SELMIsSelector32Bit(pVM, pRegFrame->eflags, pRegFrame->cs, &pRegFrame->csHid))
1672 cTransfers &= 0xffff;
1673
1674 if (!cTransfers)
1675 return VINF_SUCCESS;
1676 }
1677
1678 /* Convert source address ds:esi. */
1679 RTGCPTR GCPtrSrc;
1680 int rc = SELMToFlatEx(pVM, pRegFrame->eflags, pRegFrame->ds, (RTGCPTR)pRegFrame->esi, &pRegFrame->dsHid,
1681 SELMTOFLAT_FLAGS_HYPER | SELMTOFLAT_FLAGS_NO_PL,
1682 &GCPtrSrc, NULL);
1683 if (VBOX_FAILURE(rc))
1684 {
1685 Log(("OUTS source address conversion failed -> fallback, rc=%Vrc\n", rc));
1686 return VINF_EM_RAW_EMULATE_INSTR;
1687 }
1688
1689 /* Access verification first; we currently can't recover properly from traps inside this instruction */
1690 uint32_t cpl = CPUMGetGuestCPL(pVM, pRegFrame);
1691 rc = PGMVerifyAccess(pVM, (RTGCUINTPTR)GCPtrSrc, cTransfers * cbTransfer,
1692 (cpl == 3) ? X86_PTE_US : 0);
1693 if (rc != VINF_SUCCESS)
1694 {
1695 Log(("OUTS will generate a trap -> fallback, rc=%Vrc\n", rc));
1696 return VINF_EM_RAW_EMULATE_INSTR;
1697 }
1698
1699 Log(("IOM: rep outs%d port %#x count %d\n", cbTransfer * 8, uPort, cTransfers));
1700 if (cTransfers > 1)
1701 {
1702 /*
1703 * If the device supports string transfers, ask it to do as
1704 * much as it wants. The rest is done with single-word transfers.
1705 */
1706 const RTGCUINTREG cTransfersOrg = cTransfers;
1707 rc = IOMIOPortWriteString(pVM, uPort, &GCPtrSrc, &cTransfers, cbTransfer);
1708 AssertRC(rc); Assert(cTransfers <= cTransfersOrg);
1709 pRegFrame->esi += (cTransfersOrg - cTransfers) * cbTransfer;
1710 }
1711
1712#ifdef IN_GC
1713 MMGCRamRegisterTrapHandler(pVM);
1714#endif
1715
1716 while (cTransfers && rc == VINF_SUCCESS)
1717 {
1718 uint32_t u32Value;
1719 rc = iomRamRead(pVM, &u32Value, GCPtrSrc, cbTransfer);
1720 if (rc != VINF_SUCCESS)
1721 break;
1722 rc = IOMIOPortWrite(pVM, uPort, u32Value, cbTransfer);
1723 if (!IOM_SUCCESS(rc))
1724 break;
1725 GCPtrSrc = (RTGCPTR)((RTUINTPTR)GCPtrSrc + cbTransfer);
1726 pRegFrame->esi += cbTransfer;
1727 cTransfers--;
1728 }
1729
1730#ifdef IN_GC
1731 MMGCRamDeregisterTrapHandler(pVM);
1732#endif
1733
1734 /* Update ecx on exit. */
1735 if (uPrefix & PREFIX_REP)
1736 pRegFrame->ecx = cTransfers;
1737
1738 AssertMsg(rc == VINF_SUCCESS || rc == VINF_IOM_HC_IOPORT_WRITE || (rc >= VINF_EM_FIRST && rc <= VINF_EM_LAST) || VBOX_FAILURE(rc), ("%Vrc\n", rc));
1739 return rc;
1740}
1741
1742
1743/**
1744 * [REP*] OUTSB/OUTSW/OUTSD
1745 * DS:ESI,DX[,ECX]
1746 *
1747 * @returns Strict VBox status code. Informational status codes other than the one documented
1748 * here are to be treated as internal failure. Use IOM_SUCCESS() to check for success.
1749 * @retval VINF_SUCCESS Success.
1750 * @retval VINF_EM_FIRST-VINF_EM_LAST Success with some exceptions (see IOM_SUCCESS()), the
1751 * status code must be passed on to EM.
1752 * @retval VINF_IOM_HC_IOPORT_WRITE Defer the write to ring-3. (R0/GC only)
1753 * @retval VINF_EM_RAW_EMULATE_INSTR Defer the write to the REM.
1754 * @retval VINF_EM_RAW_GUEST_TRAP The exception was left pending. (TRPMRaiseXcptErr)
1755 * @retval VINF_TRPM_XCPT_DISPATCHED The exception was raised and dispatched for raw-mode execution. (TRPMRaiseXcptErr)
1756 * @retval VINF_EM_RESCHEDULE_REM The exception was dispatched and cannot be executed in raw-mode. (TRPMRaiseXcptErr)
1757 *
1758 * @param pVM The virtual machine (GC pointer ofcourse).
1759 * @param pRegFrame Pointer to CPUMCTXCORE guest registers structure.
1760 * @param pCpu Disassembler CPU state.
1761 */
1762IOMDECL(int) IOMInterpretOUTS(PVM pVM, PCPUMCTXCORE pRegFrame, PDISCPUSTATE pCpu)
1763{
1764 /*
1765 * Get port number from the first parameter.
1766 * And get the I/O register size from the opcode / prefix.
1767 */
1768 uint32_t Port = 0;
1769 unsigned cb = 0;
1770 bool fRc = iomGetRegImmData(pCpu, &pCpu->param1, pRegFrame, &Port, &cb);
1771 AssertMsg(fRc, ("Failed to get reg/imm port number!\n")); NOREF(fRc);
1772 if (pCpu->pCurInstr->opcode == OP_OUTSB)
1773 cb = 1;
1774 else
1775 cb = (pCpu->opmode == CPUMODE_32BIT) ? 4 : 2;
1776
1777 int rc = IOMInterpretCheckPortIOAccess(pVM, pRegFrame, Port, cb);
1778 if (RT_UNLIKELY(rc != VINF_SUCCESS))
1779 {
1780 AssertMsg(rc == VINF_EM_RAW_GUEST_TRAP || rc == VINF_TRPM_XCPT_DISPATCHED || rc == VINF_TRPM_XCPT_DISPATCHED || VBOX_FAILURE(rc), ("%Vrc\n", rc));
1781 return rc;
1782 }
1783
1784 return IOMInterpretOUTSEx(pVM, pRegFrame, Port, pCpu->prefix, cb);
1785}
1786
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