VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/nt/initterm-r0drv-nt.cpp@ 55998

Last change on this file since 55998 was 53720, checked in by vboxsync, 10 years ago

IPRT/r0drv/nt: More RTMpPokeCpu work, this time for 32-bit windows 7 and later.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 17.9 KB
Line 
1/* $Id: initterm-r0drv-nt.cpp 53720 2015-01-03 05:57:05Z vboxsync $ */
2/** @file
3 * IPRT - Initialization & Termination, R0 Driver, NT.
4 */
5
6/*
7 * Copyright (C) 2006-2013 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* Header Files *
29*******************************************************************************/
30#include "the-nt-kernel.h"
31#include <iprt/asm-amd64-x86.h>
32#include <iprt/assert.h>
33#include <iprt/err.h>
34#include <iprt/mp.h>
35#include <iprt/string.h>
36#include "internal/initterm.h"
37#include "internal-r0drv-nt.h"
38#include "symdb.h"
39#include "symdbdata.h"
40
41
42/*******************************************************************************
43* Global Variables *
44*******************************************************************************/
45/** The NT CPU set.
46 * KeQueryActiveProcssors() cannot be called at all IRQLs and therefore we'll
47 * have to cache it. Fortunately, Nt doesn't really support taking CPUs offline
48 * or online. It's first with W2K8 that support for CPU hotplugging was added.
49 * Once we start caring about this, we'll simply let the native MP event callback
50 * and update this variable as CPUs comes online. (The code is done already.)
51 */
52RTCPUSET g_rtMpNtCpuSet;
53
54/** ExSetTimerResolution, introduced in W2K. */
55PFNMYEXSETTIMERRESOLUTION g_pfnrtNtExSetTimerResolution;
56/** KeFlushQueuedDpcs, introduced in XP. */
57PFNMYKEFLUSHQUEUEDDPCS g_pfnrtNtKeFlushQueuedDpcs;
58/** HalRequestIpi, version introduced with windows 7. */
59PFNHALREQUESTIPI_W7PLUS g_pfnrtHalRequestIpiW7Plus;
60/** HalRequestIpi, version valid up to windows vista?? */
61PFNHALREQUESTIPI_PRE_W7 g_pfnrtHalRequestIpiPreW7;
62/** HalSendSoftwareInterrupt, introduced in AMD64 version of W2K3. */
63PFNHALSENDSOFTWAREINTERRUPT g_pfnrtNtHalSendSoftwareInterrupt;
64/** Worker for RTMpPokeCpu. */
65PFNRTSENDIPI g_pfnrtMpPokeCpuWorker;
66/** KeIpiGenericCall - Introduced in Windows Server 2003. */
67PFNRTKEIPIGENERICCALL g_pfnrtKeIpiGenericCall;
68/** KeInitializeAffinityEx - Introducted in Windows 7. */
69PFNKEINITIALIZEAFFINITYEX g_pfnrtKeInitializeAffinityEx;
70/** KeAddProcessorAffinityEx - Introducted in Windows 7. */
71PFNKEADDPROCESSORAFFINITYEX g_pfnrtKeAddProcessorAffinityEx;
72/** KeGetProcessorIndexFromNumber - Introducted in Windows 7. */
73PFNKEGETPROCESSORINDEXFROMNUMBER g_pfnrtKeGetProcessorIndexFromNumber;
74/** RtlGetVersion, introduced in ??. */
75PFNRTRTLGETVERSION g_pfnrtRtlGetVersion;
76#ifndef RT_ARCH_AMD64
77/** KeQueryInterruptTime - exported/new in Windows 2000. */
78PFNRTKEQUERYINTERRUPTTIME g_pfnrtKeQueryInterruptTime;
79/** KeQuerySystemTime - exported/new in Windows 2000. */
80PFNRTKEQUERYSYSTEMTIME g_pfnrtKeQuerySystemTime;
81#endif
82/** KeQueryInterruptTimePrecise - new in Windows 8. */
83PFNRTKEQUERYINTERRUPTTIMEPRECISE g_pfnrtKeQueryInterruptTimePrecise;
84/** KeQuerySystemTimePrecise - new in Windows 8. */
85PFNRTKEQUERYSYSTEMTIMEPRECISE g_pfnrtKeQuerySystemTimePrecise;
86
87/** Offset of the _KPRCB::QuantumEnd field. 0 if not found. */
88uint32_t g_offrtNtPbQuantumEnd;
89/** Size of the _KPRCB::QuantumEnd field. 0 if not found. */
90uint32_t g_cbrtNtPbQuantumEnd;
91/** Offset of the _KPRCB::DpcQueueDepth field. 0 if not found. */
92uint32_t g_offrtNtPbDpcQueueDepth;
93
94
95/**
96 * Determines the NT kernel verison information.
97 *
98 * @param pOsVerInfo Where to return the version information.
99 *
100 * @remarks pOsVerInfo->fSmp is only definitive if @c true.
101 * @remarks pOsVerInfo->uCsdNo is set to MY_NIL_CSD if it cannot be determined.
102 */
103static void rtR0NtGetOsVersionInfo(PRTNTSDBOSVER pOsVerInfo)
104{
105 ULONG ulMajorVersion = 0;
106 ULONG ulMinorVersion = 0;
107 ULONG ulBuildNumber = 0;
108
109 pOsVerInfo->fChecked = PsGetVersion(&ulMajorVersion, &ulMinorVersion, &ulBuildNumber, NULL) == TRUE;
110 pOsVerInfo->uMajorVer = (uint8_t)ulMajorVersion;
111 pOsVerInfo->uMinorVer = (uint8_t)ulMinorVersion;
112 pOsVerInfo->uBuildNo = ulBuildNumber;
113#define MY_NIL_CSD 0x3f
114 pOsVerInfo->uCsdNo = MY_NIL_CSD;
115
116 if (g_pfnrtRtlGetVersion)
117 {
118 RTL_OSVERSIONINFOEXW VerInfo;
119 RT_ZERO(VerInfo);
120 VerInfo.dwOSVersionInfoSize = sizeof(VerInfo);
121
122 NTSTATUS rcNt = g_pfnrtRtlGetVersion(&VerInfo);
123 if (NT_SUCCESS(rcNt))
124 pOsVerInfo->uCsdNo = VerInfo.wServicePackMajor;
125 }
126
127 /* Note! We cannot quite say if something is MP or UNI. So, fSmp is
128 redefined to indicate that it must be MP. */
129 pOsVerInfo->fSmp = RTMpGetCount() > 1
130 || ulMajorVersion >= 6; /* Vista and later has no UNI kernel AFAIK. */
131}
132
133
134/**
135 * Tries a set against the current kernel.
136 *
137 * @retval @c true if it matched up, global variables are updated.
138 * @retval @c false otherwise (no globals updated).
139 * @param pSet The data set.
140 * @param pbPrcb Pointer to the processor control block.
141 * @param pszVendor Pointer to the processor vendor string.
142 * @param pOsVerInfo The OS version info.
143 */
144static bool rtR0NtTryMatchSymSet(PCRTNTSDBSET pSet, uint8_t *pbPrcb, const char *pszVendor, PCRTNTSDBOSVER pOsVerInfo)
145{
146 /*
147 * Don't bother trying stuff where the NT kernel version number differs, or
148 * if the build type or SMPness doesn't match up.
149 */
150 if ( pSet->OsVerInfo.uMajorVer != pOsVerInfo->uMajorVer
151 || pSet->OsVerInfo.uMinorVer != pOsVerInfo->uMinorVer
152 || pSet->OsVerInfo.fChecked != pOsVerInfo->fChecked
153 || (!pSet->OsVerInfo.fSmp && pOsVerInfo->fSmp /*must-be-smp*/) )
154 {
155 //DbgPrint("IPRT: #%d Version/type mismatch.\n", pSet - &g_artNtSdbSets[0]);
156 return false;
157 }
158
159 /*
160 * Do the CPU vendor test.
161 *
162 * Note! The MmIsAddressValid call is the real #PF security here as the
163 * __try/__except has limited/no ability to catch everything we need.
164 */
165 char *pszPrcbVendorString = (char *)&pbPrcb[pSet->KPRCB.offVendorString];
166 if (!MmIsAddressValid(&pszPrcbVendorString[4 * 3 - 1]))
167 {
168 //DbgPrint("IPRT: #%d invalid vendor string address.\n", pSet - &g_artNtSdbSets[0]);
169 return false;
170 }
171 __try
172 {
173 if (memcmp(pszPrcbVendorString, pszVendor, RT_MIN(4 * 3, pSet->KPRCB.cbVendorString)) != 0)
174 {
175 //DbgPrint("IPRT: #%d Vendor string mismatch.\n", pSet - &g_artNtSdbSets[0]);
176 return false;
177 }
178 }
179 __except(EXCEPTION_EXECUTE_HANDLER)
180 {
181 DbgPrint("IPRT: %#d Exception\n", pSet - &g_artNtSdbSets[0]);
182 return false;
183 }
184
185 /*
186 * Got a match, update the global variables and report succcess.
187 */
188 g_offrtNtPbQuantumEnd = pSet->KPRCB.offQuantumEnd;
189 g_cbrtNtPbQuantumEnd = pSet->KPRCB.cbQuantumEnd;
190 g_offrtNtPbDpcQueueDepth = pSet->KPRCB.offDpcQueueDepth;
191
192#if 0
193 DbgPrint("IPRT: Using data set #%u for %u.%usp%u build %u %s %s.\n",
194 pSet - &g_artNtSdbSets[0],
195 pSet->OsVerInfo.uMajorVer,
196 pSet->OsVerInfo.uMinorVer,
197 pSet->OsVerInfo.uCsdNo,
198 pSet->OsVerInfo.uBuildNo,
199 pSet->OsVerInfo.fSmp ? "smp" : "uni",
200 pSet->OsVerInfo.fChecked ? "checked" : "free");
201#endif
202 return true;
203}
204
205
206DECLHIDDEN(int) rtR0InitNative(void)
207{
208 /*
209 * Init the Nt cpu set.
210 */
211#ifdef IPRT_TARGET_NT4
212 KAFFINITY ActiveProcessors = (UINT64_C(1) << KeNumberProcessors) - UINT64_C(1);
213#else
214 KAFFINITY ActiveProcessors = KeQueryActiveProcessors();
215#endif
216 RTCpuSetEmpty(&g_rtMpNtCpuSet);
217 RTCpuSetFromU64(&g_rtMpNtCpuSet, ActiveProcessors);
218/** @todo Port to W2K8 with > 64 cpus/threads. */
219
220 /*
221 * Initialize the function pointers.
222 */
223#ifdef IPRT_TARGET_NT4
224 g_pfnrtNtExSetTimerResolution = NULL;
225 g_pfnrtNtKeFlushQueuedDpcs = NULL;
226 g_pfnrtHalRequestIpiW7Plus = NULL;
227 g_pfnrtHalRequestIpiPreW7 = NULL;
228 g_pfnrtNtHalSendSoftwareInterrupt = NULL;
229 g_pfnrtKeIpiGenericCall = NULL;
230 g_pfnrtKeInitializeAffinityEx = NULL;
231 g_pfnrtKeAddProcessorAffinityEx = NULL;
232 g_pfnrtKeGetProcessorIndexFromNumber = NULL;
233 g_pfnrtRtlGetVersion = NULL;
234 g_pfnrtKeQueryInterruptTime = NULL;
235 g_pfnrtKeQueryInterruptTimePrecise = NULL;
236 g_pfnrtKeQuerySystemTime = NULL;
237 g_pfnrtKeQuerySystemTimePrecise = NULL;
238#else
239 UNICODE_STRING RoutineName;
240 RtlInitUnicodeString(&RoutineName, L"ExSetTimerResolution");
241 g_pfnrtNtExSetTimerResolution = (PFNMYEXSETTIMERRESOLUTION)MmGetSystemRoutineAddress(&RoutineName);
242
243 RtlInitUnicodeString(&RoutineName, L"KeFlushQueuedDpcs");
244 g_pfnrtNtKeFlushQueuedDpcs = (PFNMYKEFLUSHQUEUEDDPCS)MmGetSystemRoutineAddress(&RoutineName);
245
246 RtlInitUnicodeString(&RoutineName, L"HalRequestIpi");
247 g_pfnrtHalRequestIpiW7Plus = (PFNHALREQUESTIPI_W7PLUS)MmGetSystemRoutineAddress(&RoutineName);
248 g_pfnrtHalRequestIpiPreW7 = (PFNHALREQUESTIPI_PRE_W7)g_pfnrtHalRequestIpiW7Plus;
249
250 RtlInitUnicodeString(&RoutineName, L"HalSendSoftwareInterrupt");
251 g_pfnrtNtHalSendSoftwareInterrupt = (PFNHALSENDSOFTWAREINTERRUPT)MmGetSystemRoutineAddress(&RoutineName);
252
253 RtlInitUnicodeString(&RoutineName, L"KeIpiGenericCall");
254 g_pfnrtKeIpiGenericCall = (PFNRTKEIPIGENERICCALL)MmGetSystemRoutineAddress(&RoutineName);
255
256 RtlInitUnicodeString(&RoutineName, L"KeInitializeAffinityEx");
257 g_pfnrtKeInitializeAffinityEx = (PFNKEINITIALIZEAFFINITYEX)MmGetSystemRoutineAddress(&RoutineName);
258
259 RtlInitUnicodeString(&RoutineName, L"KeAddProcessorAffinityEx");
260 g_pfnrtKeAddProcessorAffinityEx = (PFNKEADDPROCESSORAFFINITYEX)MmGetSystemRoutineAddress(&RoutineName);
261
262 RtlInitUnicodeString(&RoutineName, L"KeGetProcessorIndexFromNumber");
263 g_pfnrtKeGetProcessorIndexFromNumber = (PFNKEGETPROCESSORINDEXFROMNUMBER)MmGetSystemRoutineAddress(&RoutineName);
264
265 RtlInitUnicodeString(&RoutineName, L"RtlGetVersion");
266 g_pfnrtRtlGetVersion = (PFNRTRTLGETVERSION)MmGetSystemRoutineAddress(&RoutineName);
267# ifndef RT_ARCH_AMD64
268 RtlInitUnicodeString(&RoutineName, L"KeQueryInterruptTime");
269 g_pfnrtKeQueryInterruptTime = (PFNRTKEQUERYINTERRUPTTIME)MmGetSystemRoutineAddress(&RoutineName);
270
271 RtlInitUnicodeString(&RoutineName, L"KeQuerySystemTime");
272 g_pfnrtKeQuerySystemTime = (PFNRTKEQUERYSYSTEMTIME)MmGetSystemRoutineAddress(&RoutineName);
273# endif
274 RtlInitUnicodeString(&RoutineName, L"KeQueryInterruptTimePrecise");
275 g_pfnrtKeQueryInterruptTimePrecise = (PFNRTKEQUERYINTERRUPTTIMEPRECISE)MmGetSystemRoutineAddress(&RoutineName);
276
277 RtlInitUnicodeString(&RoutineName, L"KeQuerySystemTimePrecise");
278 g_pfnrtKeQuerySystemTimePrecise = (PFNRTKEQUERYSYSTEMTIMEPRECISE)MmGetSystemRoutineAddress(&RoutineName);
279#endif
280
281 /*
282 * HACK ALERT! (and déjà vu warning - remember win32k.sys?)
283 *
284 * Try find _KPRCB::QuantumEnd and _KPRCB::[DpcData.]DpcQueueDepth.
285 * For purpose of verification we use the VendorString member (12+1 chars).
286 *
287 * The offsets was initially derived by poking around with windbg
288 * (dt _KPRCB, !prcb ++, and such like). Systematic harvesting was then
289 * planned using dia2dump, grep and the symbol pack in a manner like this:
290 * dia2dump -type _KDPC_DATA -type _KPRCB EXE\ntkrnlmp.pdb | grep -wE "QuantumEnd|DpcData|DpcQueueDepth|VendorString"
291 *
292 * The final solution ended up using a custom harvester program called
293 * ntBldSymDb that recursively searches thru unpacked symbol packages for
294 * the desired structure offsets. The program assumes that the packages
295 * are unpacked into directories with the same name as the package, with
296 * exception of some of the w2k packages which requires a 'w2k' prefix to
297 * be distinguishable from another.
298 */
299
300 RTNTSDBOSVER OsVerInfo;
301 rtR0NtGetOsVersionInfo(&OsVerInfo);
302
303 /*
304 * Gather consistent CPU vendor string and PRCB pointers.
305 */
306 KIRQL OldIrql;
307 KeRaiseIrql(DISPATCH_LEVEL, &OldIrql); /* make sure we stay on the same cpu */
308
309 union
310 {
311 uint32_t auRegs[4];
312 char szVendor[4*3+1];
313 } u;
314 ASMCpuId(0, &u.auRegs[3], &u.auRegs[0], &u.auRegs[2], &u.auRegs[1]);
315 u.szVendor[4*3] = '\0';
316
317 uint8_t *pbPrcb;
318 __try /* Warning. This try/except statement may provide some false safety. */
319 {
320#if defined(RT_ARCH_X86)
321 PKPCR pPcr = (PKPCR)__readfsdword(RT_OFFSETOF(KPCR,SelfPcr));
322 pbPrcb = (uint8_t *)pPcr->Prcb;
323#elif defined(RT_ARCH_AMD64)
324 PKPCR pPcr = (PKPCR)__readgsqword(RT_OFFSETOF(KPCR,Self));
325 pbPrcb = (uint8_t *)pPcr->CurrentPrcb;
326#else
327# error "port me"
328 pbPrcb = NULL;
329#endif
330 }
331 __except(EXCEPTION_EXECUTE_HANDLER)
332 {
333 pbPrcb = NULL;
334 }
335
336 /*
337 * Search the database
338 */
339 if (pbPrcb)
340 {
341 /* Find the best matching kernel version based on build number. */
342 uint32_t iBest = UINT32_MAX;
343 int32_t iBestDelta = INT32_MAX;
344 for (uint32_t i = 0; i < RT_ELEMENTS(g_artNtSdbSets); i++)
345 {
346 if (g_artNtSdbSets[i].OsVerInfo.fChecked != OsVerInfo.fChecked)
347 continue;
348 if (OsVerInfo.fSmp /*must-be-smp*/ && !g_artNtSdbSets[i].OsVerInfo.fSmp)
349 continue;
350
351 int32_t iDelta = RT_ABS((int32_t)OsVerInfo.uBuildNo - (int32_t)g_artNtSdbSets[i].OsVerInfo.uBuildNo);
352 if ( iDelta == 0
353 && (g_artNtSdbSets[i].OsVerInfo.uCsdNo == OsVerInfo.uCsdNo || OsVerInfo.uCsdNo == MY_NIL_CSD))
354 {
355 /* prefect */
356 iBestDelta = iDelta;
357 iBest = i;
358 break;
359 }
360 if ( iDelta < iBestDelta
361 || iBest == UINT32_MAX
362 || ( iDelta == iBestDelta
363 && OsVerInfo.uCsdNo != MY_NIL_CSD
364 && RT_ABS(g_artNtSdbSets[i ].OsVerInfo.uCsdNo - (int32_t)OsVerInfo.uCsdNo)
365 < RT_ABS(g_artNtSdbSets[iBest].OsVerInfo.uCsdNo - (int32_t)OsVerInfo.uCsdNo)
366 )
367 )
368 {
369 iBestDelta = iDelta;
370 iBest = i;
371 }
372 }
373 if (iBest < RT_ELEMENTS(g_artNtSdbSets))
374 {
375 /* Try all sets: iBest -> End; iBest -> Start. */
376 bool fDone = false;
377 int32_t i = iBest;
378 while ( i < RT_ELEMENTS(g_artNtSdbSets)
379 && !(fDone = rtR0NtTryMatchSymSet(&g_artNtSdbSets[i], pbPrcb, u.szVendor, &OsVerInfo)))
380 i++;
381 if (!fDone)
382 {
383 i = (int32_t)iBest - 1;
384 while ( i >= 0
385 && !(fDone = rtR0NtTryMatchSymSet(&g_artNtSdbSets[i], pbPrcb, u.szVendor, &OsVerInfo)))
386 i--;
387 }
388 }
389 else
390 DbgPrint("IPRT: Failed to locate data set.\n");
391 }
392 else
393 DbgPrint("IPRT: Failed to get PCBR pointer.\n");
394
395 KeLowerIrql(OldIrql); /* Lowering the IRQL early in the hope that we may catch exceptions below. */
396
397#ifndef IN_GUEST
398 if (!g_offrtNtPbQuantumEnd && !g_offrtNtPbDpcQueueDepth)
399 DbgPrint("IPRT: Neither _KPRCB::QuantumEnd nor _KPRCB::DpcQueueDepth was not found! Kernel %u.%u %u %s\n",
400 OsVerInfo.uMajorVer, OsVerInfo.uMinorVer, OsVerInfo.uBuildNo, OsVerInfo.fChecked ? "checked" : "free");
401# ifdef DEBUG
402 else
403 DbgPrint("IPRT: _KPRCB:{.QuantumEnd=%x/%d, .DpcQueueDepth=%x/%d} Kernel %u.%u %u %s\n",
404 g_offrtNtPbQuantumEnd, g_cbrtNtPbQuantumEnd, g_offrtNtPbDpcQueueDepth,
405 OsVerInfo.uMajorVer, OsVerInfo.uMinorVer, OsVerInfo.uBuildNo, OsVerInfo.fChecked ? "checked" : "free");
406# endif
407#endif
408
409 /*
410 * Special IPI fun for RTMpPokeCpu.
411 *
412 * On Vista and later the DPC method doesn't seem to reliably send IPIs,
413 * so we have to use alternative methods. The NtHalSendSoftwareInterrupt
414 * is preferrable, but it's AMD64 only. The NalRequestIpip method changed
415 * in Windows 7 with the lots-of-processors-support, but it's the only
416 * targeted IPI game in town if we cannot use KeInsertQueueDpc. Worst case
417 * we use broadcast IPIs.
418 */
419 if ( OsVerInfo.uMajorVer > 6
420 || (OsVerInfo.uMajorVer == 6 && OsVerInfo.uMinorVer > 0))
421 g_pfnrtHalRequestIpiPreW7 = NULL;
422 else
423 g_pfnrtHalRequestIpiW7Plus = NULL;
424
425 g_pfnrtMpPokeCpuWorker = rtMpPokeCpuUsingDpc;
426#ifndef IPRT_TARGET_NT4
427 if (g_pfnrtNtHalSendSoftwareInterrupt)
428 g_pfnrtMpPokeCpuWorker = rtMpPokeCpuUsingHalSendSoftwareInterrupt;
429 else if ( g_pfnrtHalRequestIpiW7Plus
430 && g_pfnrtKeInitializeAffinityEx
431 && g_pfnrtKeAddProcessorAffinityEx
432 && g_pfnrtKeGetProcessorIndexFromNumber)
433 g_pfnrtMpPokeCpuWorker = rtMpPokeCpuUsingHalReqestIpiW7Plus;
434 else if (OsVerInfo.uMajorVer >= 6 && g_pfnrtKeIpiGenericCall)
435 g_pfnrtMpPokeCpuWorker = rtMpPokeCpuUsingBroadcastIpi;
436 /* else: Windows XP should send always send an IPI -> VERIFY */
437#endif
438
439 return VINF_SUCCESS;
440}
441
442
443DECLHIDDEN(void) rtR0TermNative(void)
444{
445}
446
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