VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/Support/testcase/tstInt.cpp@ 6412

Last change on this file since 6412 was 5999, checked in by vboxsync, 17 years ago

The Giant CDDL Dual-License Header Change.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.1 KB
Line 
1/** $Id: tstInt.cpp 5999 2007-12-07 15:05:06Z vboxsync $ */
2/** @file
3 * Testcase: Test the interrupt gate feature of the support library.
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 * 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 <VBox/sup.h>
32#include <VBox/vm.h>
33#include <VBox/vmm.h>
34#include <VBox/err.h>
35#include <VBox/param.h>
36#include <iprt/runtime.h>
37#include <iprt/stream.h>
38#include <iprt/string.h>
39
40
41/**
42 * Makes a path to a file in the executable directory.
43 */
44static char *ExeDirFile(char *pszFile, const char *pszArgv0, const char *pszFilename)
45{
46 char *psz;
47 char *psz2;
48
49 strcpy(pszFile, pszArgv0);
50 psz = strrchr(pszFile, '/');
51 psz2 = strrchr(pszFile, '\\');
52 if (psz < psz2)
53 psz = psz2;
54 if (!psz)
55 psz = strrchr(pszFile, ':');
56 if (!psz)
57 {
58 strcpy(pszFile, "./");
59 psz = &pszFile[1];
60 }
61 strcpy(psz + 1, "VMMR0.r0");
62 return pszFile;
63}
64
65int main(int argc, char **argv)
66{
67 int rcRet = 0;
68 int i;
69 int rc;
70 int cIterations = argc > 1 ? RTStrToUInt32(argv[1]) : 32;
71 if (cIterations == 0)
72 cIterations = 64;
73
74 /*
75 * Init.
76 */
77 RTR3Init();
78 PSUPDRVSESSION pSession;
79 rc = SUPInit(&pSession);
80 rcRet += rc != 0;
81 RTPrintf("tstInt: SUPInit -> rc=%Vrc\n", rc);
82 if (!rc)
83 {
84 /*
85 * Load VMM code.
86 */
87 char szFile[RTPATH_MAX];
88 rc = SUPLoadVMM(ExeDirFile(szFile, argv[0], "VMMR0.r0"));
89 if (!rc)
90 {
91 /*
92 * Create a fake 'VM'.
93 */
94 PVMR0 pVMR0 = NIL_RTR0PTR;
95 PVM pVM = NULL;
96 const unsigned cPages = RT_ALIGN_Z(sizeof(*pVM), PAGE_SIZE) >> PAGE_SHIFT;
97 PSUPPAGE paPages = (PSUPPAGE)RTMemAllocZ(cPages * sizeof(SUPPAGE));
98 if (paPages)
99 rc = SUPLowAlloc(cPages, (void **)&pVM, &pVMR0, &paPages[0]);
100 else
101 rc = VERR_NO_MEMORY;
102 if (VBOX_SUCCESS(rc))
103 {
104 pVM->pVMGC = 0;
105 pVM->pVMR3 = pVM;
106 pVM->pVMR0 = pVMR0;
107 pVM->paVMPagesR3 = paPages;
108 pVM->pSession = pSession;
109
110 rc = SUPSetVMForFastIOCtl(pVMR0);
111 if (!rc)
112 {
113
114 /*
115 * Call VMM code with invalid function.
116 */
117 for (i = cIterations; i > 0; i--)
118 {
119 rc = SUPCallVMMR0(pVMR0, VMMR0_DO_NOP, NULL);
120 if (rc != VINF_SUCCESS)
121 {
122 RTPrintf("tstInt: SUPCallVMMR0 -> rc=%Vrc i=%d Expected VINF_SUCCESS!\n", rc, i);
123 rcRet++;
124 break;
125 }
126 }
127 RTPrintf("tstInt: Performed SUPCallVMMR0 %d times (rc=%Vrc)\n", cIterations, rc);
128
129 /*
130 * Profile it.
131 */
132 if (!rc)
133 {
134 RTTimeNanoTS();
135 uint64_t StartTS = RTTimeNanoTS();
136 uint64_t StartTick = ASMReadTSC();
137 uint64_t MinTicks = UINT64_MAX;
138 for (i = 0; i < 1000000; i++)
139 {
140 uint64_t OneStartTick = ASMReadTSC();
141 rc = SUPCallVMMR0(pVMR0, VMMR0_DO_NOP, NULL);
142 uint64_t Ticks = ASMReadTSC() - OneStartTick;
143 if (Ticks < MinTicks)
144 MinTicks = Ticks;
145
146 if (RT_UNLIKELY(rc != VINF_SUCCESS))
147 {
148 RTPrintf("tstInt: SUPCallVMMR0 -> rc=%Vrc i=%d Expected VINF_SUCCESS!\n", rc, i);
149 rcRet++;
150 break;
151 }
152 }
153 uint64_t Ticks = ASMReadTSC() - StartTick;
154 uint64_t NanoSecs = RTTimeNanoTS() - StartTS;
155
156 RTPrintf("tstInt: SUPCallVMMR0 - %d iterations in %llu ns / %llu ticks. %llu ns / %#llu ticks per iteration. Min %llu ticks.\n",
157 i, NanoSecs, Ticks, NanoSecs / i, Ticks / i, MinTicks);
158
159#ifdef VBOX_WITH_IDT_PATCHING
160 /*
161 * The fast path.
162 */
163 RTTimeNanoTS();
164 StartTS = RTTimeNanoTS();
165 StartTick = ASMReadTSC();
166 MinTicks = UINT64_MAX;
167 for (i = 0; i < 1000000; i++)
168 {
169 uint64_t OneStartTick = ASMReadTSC();
170 rc = SUPCallVMMR0Fast(pVMR0, VMMR0_DO_NOP);
171 uint64_t Ticks = ASMReadTSC() - OneStartTick;
172 if (Ticks < MinTicks)
173 MinTicks = Ticks;
174
175 if (RT_UNLIKELY(rc != VINF_SUCCESS))
176 {
177 RTPrintf("tstInt: SUPCallVMMR0 -> rc=%Vrc i=%d Expected VINF_SUCCESS!\n", rc, i);
178 rcRet++;
179 break;
180 }
181 }
182 Ticks = ASMReadTSC() - StartTick;
183 NanoSecs = RTTimeNanoTS() - StartTS;
184
185 RTPrintf("tstInt: SUPCallVMMR0Fast - %d iterations in %llu ns / %llu ticks. %llu ns / %#llu ticks per iteration. Min %llu ticks.\n",
186 i, NanoSecs, Ticks, NanoSecs / i, Ticks / i, MinTicks);
187#endif /* VBOX_WITH_IDT_PATCHING */
188
189 /*
190 * The ordinary path.
191 */
192 RTTimeNanoTS();
193 StartTS = RTTimeNanoTS();
194 StartTick = ASMReadTSC();
195 MinTicks = UINT64_MAX;
196 for (i = 0; i < 1000000; i++)
197 {
198 uint64_t OneStartTick = ASMReadTSC();
199 rc = SUPCallVMMR0Ex(pVMR0, VMMR0_DO_NOP, 0, NULL);
200 uint64_t Ticks = ASMReadTSC() - OneStartTick;
201 if (Ticks < MinTicks)
202 MinTicks = Ticks;
203
204 if (RT_UNLIKELY(rc != VINF_SUCCESS))
205 {
206 RTPrintf("tstInt: SUPCallVMMR0 -> rc=%Vrc i=%d Expected VINF_SUCCESS!\n", rc, i);
207 rcRet++;
208 break;
209 }
210 }
211 Ticks = ASMReadTSC() - StartTick;
212 NanoSecs = RTTimeNanoTS() - StartTS;
213
214 RTPrintf("tstInt: SUPCallVMMR0Ex - %d iterations in %llu ns / %llu ticks. %llu ns / %#llu ticks per iteration. Min %llu ticks.\n",
215 i, NanoSecs, Ticks, NanoSecs / i, Ticks / i, MinTicks);
216 }
217 }
218 else
219 {
220 RTPrintf("tstInt: SUPSetVMForFastIOCtl failed: %Vrc\n", rc);
221 rcRet++;
222 }
223 }
224 else
225 {
226 RTPrintf("tstInt: SUPContAlloc2(%#zx,,) failed\n", sizeof(*pVM));
227 rcRet++;
228 }
229
230 /*
231 * Unload VMM.
232 */
233 rc = SUPUnloadVMM();
234 if (rc)
235 {
236 RTPrintf("tstInt: SUPUnloadVMM failed with rc=%Vrc\n", rc);
237 rcRet++;
238 }
239 }
240 else
241 {
242 RTPrintf("tstInt: SUPLoadVMM failed with rc=%Vrc\n", rc);
243 rcRet++;
244 }
245
246 /*
247 * Terminate.
248 */
249 rc = SUPTerm();
250 rcRet += rc != 0;
251 RTPrintf("tstInt: SUPTerm -> rc=%Vrc\n", rc);
252 }
253
254 return !!rc;
255}
256
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