VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/bootsectors/bs3kit/bs3-cmn-TrapDefaultHandler.c@ 66456

Last change on this file since 66456 was 66456, checked in by vboxsync, 8 years ago

bs3kit: Mark system traps with '* GURU *' to distinguish it from other register contexts being printed.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.6 KB
Line 
1/* $Id: bs3-cmn-TrapDefaultHandler.c 66456 2017-04-06 10:43:27Z vboxsync $ */
2/** @file
3 * BS3Kit - Bs3TrapDefaultHandler
4 */
5
6/*
7 * Copyright (C) 2007-2016 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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include "bs3kit-template-header.h"
32#if TMPL_BITS != 64
33# include <VBox/VMMDevTesting.h>
34# include <iprt/asm-amd64-x86.h>
35#endif
36
37
38/*********************************************************************************************************************************
39* Global Variables *
40*********************************************************************************************************************************/
41#define g_pBs3TrapSetJmpFrame BS3_DATA_NM(g_pBs3TrapSetJmpFrame)
42extern uint32_t g_pBs3TrapSetJmpFrame;
43
44#define g_Bs3TrapSetJmpCtx BS3_DATA_NM(g_Bs3TrapSetJmpCtx)
45extern BS3REGCTX g_Bs3TrapSetJmpCtx;
46
47
48#if TMPL_BITS != 64
49static void bs3TrapDefaultHandlerV8086Syscall(PBS3TRAPFRAME pTrapFrame)
50{
51 /* Minimal syscall. */
52 if (pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_PRINT_CHR)
53 Bs3PrintChr(pTrapFrame->Ctx.rcx.u8);
54 else if (pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_PRINT_STR)
55 Bs3PrintStrN(Bs3XptrFlatToCurrent(((uint32_t)pTrapFrame->Ctx.rcx.u16 << 4) + pTrapFrame->Ctx.rsi.u16),
56 pTrapFrame->Ctx.rdx.u16);
57 else if (pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_RESTORE_CTX)
58 Bs3RegCtxRestore(Bs3XptrFlatToCurrent(((uint32_t)pTrapFrame->Ctx.rcx.u16 << 4) + pTrapFrame->Ctx.rsi.u16),
59 pTrapFrame->Ctx.rdx.u16);
60 else if ( pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_TO_RING0
61 || pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_TO_RING1
62 || pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_TO_RING2
63 || pTrapFrame->Ctx.rax.u16 == BS3_SYSCALL_TO_RING3)
64 {
65 Bs3RegCtxConvertToRingX(&pTrapFrame->Ctx, pTrapFrame->Ctx.rax.u16 - BS3_SYSCALL_TO_RING0);
66 }
67 else
68 Bs3Panic();
69}
70#endif
71
72#undef Bs3TrapDefaultHandler
73BS3_CMN_DEF(void, Bs3TrapDefaultHandler,(PBS3TRAPFRAME pTrapFrame))
74{
75#if TMPL_BITS != 64
76 /*
77 * v8086 VMM tasks.
78 */
79 if (pTrapFrame->Ctx.rflags.u32 & X86_EFL_VM)
80 {
81 bool fHandled = true;
82 uint8_t cBitsOpcode = 16;
83 uint8_t bOpCode;
84 uint8_t const BS3_FAR *pbCodeStart;
85 uint8_t const BS3_FAR *pbCode;
86 uint16_t BS3_FAR *pusStack;
87
88 pusStack = (uint16_t BS3_FAR *)BS3_MAKE_PROT_R0PTR_FROM_REAL(pTrapFrame->Ctx.ss, pTrapFrame->Ctx.rsp.u16);
89 pbCode = (uint8_t const BS3_FAR *)BS3_MAKE_PROT_R0PTR_FROM_REAL(pTrapFrame->Ctx.cs, pTrapFrame->Ctx.rip.u16);
90 pbCodeStart = pbCode;
91
92 /*
93 * Deal with GPs in V8086 mode.
94 */
95 if (pTrapFrame->bXcpt == X86_XCPT_GP)
96 {
97 bOpCode = *pbCode++;
98 if (bOpCode == 0x66)
99 {
100 cBitsOpcode = 32;
101 bOpCode = *pbCode++;
102 }
103
104 /* INT xx: Real mode behaviour, but intercepting and implementing most of our syscall interface. */
105 if (bOpCode == 0xcd)
106 {
107 uint8_t bVector = *pbCode++;
108 if (bVector == BS3_TRAP_SYSCALL)
109 bs3TrapDefaultHandlerV8086Syscall(pTrapFrame);
110 else
111 {
112 /* Real mode behaviour. */
113 uint16_t BS3_FAR *pusIvte = (uint16_t BS3_FAR *)BS3_MAKE_PROT_R0PTR_FROM_REAL(0, 0);
114 pusIvte += (uint16_t)bVector *2;
115
116 pusStack[0] = pTrapFrame->Ctx.rflags.u16;
117 pusStack[1] = pTrapFrame->Ctx.cs;
118 pusStack[2] = pTrapFrame->Ctx.rip.u16 + (uint16_t)(pbCode - pbCodeStart);
119
120 pTrapFrame->Ctx.rip.u16 = pusIvte[0];
121 pTrapFrame->Ctx.cs = pusIvte[1];
122 pTrapFrame->Ctx.rflags.u16 &= ~X86_EFL_IF; /** @todo this isn't all, but it'll do for now, I hope. */
123 Bs3RegCtxRestore(&pTrapFrame->Ctx, 0/*fFlags*/); /* does not return. */
124 }
125 }
126 /* PUSHF: Real mode behaviour. */
127 else if (bOpCode == 0x9c)
128 {
129 if (cBitsOpcode == 32)
130 *pusStack++ = pTrapFrame->Ctx.rflags.au16[1] & ~(X86_EFL_VM | X86_EFL_RF);
131 *pusStack++ = pTrapFrame->Ctx.rflags.u16;
132 pTrapFrame->Ctx.rsp.u16 += cBitsOpcode / 8;
133 }
134 /* POPF: Real mode behaviour. */
135 else if (bOpCode == 0x9d)
136 {
137 if (cBitsOpcode == 32)
138 {
139 pTrapFrame->Ctx.rflags.u32 &= ~X86_EFL_POPF_BITS;
140 pTrapFrame->Ctx.rflags.u32 |= X86_EFL_POPF_BITS & *(uint32_t const *)pusStack;
141 }
142 else
143 {
144 pTrapFrame->Ctx.rflags.u32 &= ~(X86_EFL_POPF_BITS | UINT32_C(0xffff0000)) & ~X86_EFL_RF;
145 pTrapFrame->Ctx.rflags.u16 |= (uint16_t)X86_EFL_POPF_BITS & *pusStack;
146 }
147 pTrapFrame->Ctx.rsp.u16 -= cBitsOpcode / 8;
148 }
149 /* CLI: Real mode behaviour. */
150 else if (bOpCode == 0xfa)
151 pTrapFrame->Ctx.rflags.u16 &= ~X86_EFL_IF;
152 /* STI: Real mode behaviour. */
153 else if (bOpCode == 0xfb)
154 pTrapFrame->Ctx.rflags.u16 |= X86_EFL_IF;
155 /* OUT: byte I/O to VMMDev. */
156 else if ( bOpCode == 0xee
157 && ((unsigned)(pTrapFrame->Ctx.rdx.u16 - VMMDEV_TESTING_IOPORT_BASE) < (unsigned)VMMDEV_TESTING_IOPORT_COUNT))
158 ASMOutU8(pTrapFrame->Ctx.rdx.u16, pTrapFrame->Ctx.rax.u8);
159 /* OUT: [d]word I/O to VMMDev. */
160 else if ( bOpCode == 0xef
161 && ((unsigned)(pTrapFrame->Ctx.rdx.u16 - VMMDEV_TESTING_IOPORT_BASE) < (unsigned)VMMDEV_TESTING_IOPORT_COUNT))
162 {
163 if (cBitsOpcode != 32)
164 ASMOutU16(pTrapFrame->Ctx.rdx.u16, pTrapFrame->Ctx.rax.u16);
165 else
166 ASMOutU32(pTrapFrame->Ctx.rdx.u16, pTrapFrame->Ctx.rax.u32);
167 }
168 /* IN: byte I/O to VMMDev. */
169 else if ( bOpCode == 0xec
170 && ((unsigned)(pTrapFrame->Ctx.rdx.u16 - VMMDEV_TESTING_IOPORT_BASE) < (unsigned)VMMDEV_TESTING_IOPORT_COUNT))
171 pTrapFrame->Ctx.rax.u8 = ASMInU8(pTrapFrame->Ctx.rdx.u16);
172 /* IN: [d]word I/O to VMMDev. */
173 else if ( bOpCode == 0xed
174 && ((unsigned)(pTrapFrame->Ctx.rdx.u16 - VMMDEV_TESTING_IOPORT_BASE) < (unsigned)VMMDEV_TESTING_IOPORT_COUNT))
175 {
176 if (cBitsOpcode != 32)
177 pTrapFrame->Ctx.rax.u16 = ASMInU16(pTrapFrame->Ctx.rdx.u16);
178 else
179 pTrapFrame->Ctx.rax.u32 = ASMInU32(pTrapFrame->Ctx.rdx.u32);
180 }
181 /* Unexpected. */
182 else
183 fHandled = false;
184 }
185 /*
186 * Deal with lock prefixed int xxh syscall in v8086 mode.
187 */
188 else if ( pTrapFrame->bXcpt == X86_XCPT_UD
189 && pbCode[0] == 0xf0
190 && pbCode[1] == 0xcd
191 && pbCode[2] == BS3_TRAP_SYSCALL
192 && pTrapFrame->Ctx.cs == BS3_SEL_TEXT16)
193 {
194 pbCode += 3;
195 bs3TrapDefaultHandlerV8086Syscall(pTrapFrame);
196 }
197 else
198 fHandled = false;
199 if (fHandled)
200 {
201 pTrapFrame->Ctx.rip.u16 += (uint16_t)(pbCode - pbCodeStart);
202# if 0
203 Bs3Printf("Calling Bs3RegCtxRestore\n");
204 Bs3RegCtxPrint(&pTrapFrame->Ctx);
205# endif
206 Bs3RegCtxRestore(&pTrapFrame->Ctx, 0 /*fFlags*/); /* does not return. */
207 return;
208 }
209 }
210#endif
211
212 /*
213 * Any pending setjmp?
214 */
215 if (g_pBs3TrapSetJmpFrame != 0)
216 {
217 PBS3TRAPFRAME pSetJmpFrame = (PBS3TRAPFRAME)Bs3XptrFlatToCurrent(g_pBs3TrapSetJmpFrame);
218 //Bs3Printf("Calling longjmp: pSetJmpFrame=%p (%#lx)\n", pSetJmpFrame, g_pBs3TrapSetJmpFrame);
219 g_pBs3TrapSetJmpFrame = 0;
220 Bs3MemCpy(pSetJmpFrame, pTrapFrame, sizeof(*pSetJmpFrame));
221 //Bs3RegCtxPrint(&g_Bs3TrapSetJmpCtx);
222 Bs3RegCtxRestore(&g_Bs3TrapSetJmpCtx, 0 /*fFlags*/);
223 }
224
225 /*
226 * Fatal.
227 */
228 Bs3TestPrintf("*** GURU ***\n");
229 Bs3TrapPrintFrame(pTrapFrame);
230 Bs3Panic();
231}
232
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