VirtualBox

source: vbox/trunk/src/VBox/VMM/tools/VBoxCpuReport.cpp@ 62776

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

VMM: warnings.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 223.6 KB
Line 
1/* $Id: VBoxCpuReport.cpp 62776 2016-07-31 21:58:30Z vboxsync $ */
2/** @file
3 * VBoxCpuReport - Produces the basis for a CPU DB entry.
4 */
5
6/*
7 * Copyright (C) 2013-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
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#include <iprt/asm.h>
23#include <iprt/asm-amd64-x86.h>
24#include <iprt/buildconfig.h>
25#include <iprt/ctype.h>
26#include <iprt/file.h>
27#include <iprt/getopt.h>
28#include <iprt/initterm.h>
29#include <iprt/message.h>
30#include <iprt/mem.h>
31#include <iprt/path.h>
32#include <iprt/string.h>
33#include <iprt/stream.h>
34#include <iprt/symlink.h>
35#include <iprt/thread.h>
36#include <iprt/time.h>
37
38#include <VBox/err.h>
39#include <VBox/vmm/cpum.h>
40#include <VBox/sup.h>
41
42
43/*******************************************************************************
44* Structures and Typedefs *
45*******************************************************************************/
46/** Write only register. */
47#define VBCPUREPMSR_F_WRITE_ONLY RT_BIT(0)
48
49typedef struct VBCPUREPMSR
50{
51 /** The first MSR register number. */
52 uint32_t uMsr;
53 /** Flags (MSRREPORT_F_XXX). */
54 uint32_t fFlags;
55 /** The value we read, unless write-only. */
56 uint64_t uValue;
57} VBCPUREPMSR;
58
59
60/*******************************************************************************
61* Global Variables *
62*******************************************************************************/
63/** The CPU vendor. Used by the MSR code. */
64static CPUMCPUVENDOR g_enmVendor = CPUMCPUVENDOR_INVALID;
65/** The CPU microarchitecture. Used by the MSR code. */
66static CPUMMICROARCH g_enmMicroarch = kCpumMicroarch_Invalid;
67/** Set if g_enmMicroarch indicates an Intel NetBurst CPU. */
68static bool g_fIntelNetBurst = false;
69/** The alternative report stream. */
70static PRTSTREAM g_pReportOut;
71/** The alternative debug stream. */
72static PRTSTREAM g_pDebugOut;
73
74/** Snooping info storage for vbCpuRepGuessScalableBusFrequencyName. */
75static uint64_t g_uMsrIntelP6FsbFrequency = UINT64_MAX;
76
77
78static void vbCpuRepDebug(const char *pszMsg, ...)
79{
80 va_list va;
81
82 /* Always print a copy of the report to standard error. */
83 va_start(va, pszMsg);
84 RTStrmPrintfV(g_pStdErr, pszMsg, va);
85 va_end(va);
86 RTStrmFlush(g_pStdErr);
87
88 /* Alternatively, also print to a log file. */
89 if (g_pDebugOut)
90 {
91 va_start(va, pszMsg);
92 RTStrmPrintfV(g_pDebugOut, pszMsg, va);
93 va_end(va);
94 RTStrmFlush(g_pDebugOut);
95 }
96
97 /* Give the output device a chance to write / display it. */
98 RTThreadSleep(1);
99}
100
101
102static void vbCpuRepPrintf(const char *pszMsg, ...)
103{
104 va_list va;
105
106 /* Output to report file, if requested. */
107 if (g_pReportOut)
108 {
109 va_start(va, pszMsg);
110 RTStrmPrintfV(g_pReportOut, pszMsg, va);
111 va_end(va);
112 RTStrmFlush(g_pReportOut);
113 }
114
115 /* Always print a copy of the report to standard out. */
116 va_start(va, pszMsg);
117 RTStrmPrintfV(g_pStdOut, pszMsg, va);
118 va_end(va);
119 RTStrmFlush(g_pStdOut);
120}
121
122
123
124static int vbCpuRepMsrsAddOne(VBCPUREPMSR **ppaMsrs, uint32_t *pcMsrs,
125 uint32_t uMsr, uint64_t uValue, uint32_t fFlags)
126{
127 /*
128 * Grow the array?
129 */
130 uint32_t cMsrs = *pcMsrs;
131 if ((cMsrs % 64) == 0)
132 {
133 void *pvNew = RTMemRealloc(*ppaMsrs, (cMsrs + 64) * sizeof(**ppaMsrs));
134 if (!pvNew)
135 {
136 RTMemFree(*ppaMsrs);
137 *ppaMsrs = NULL;
138 *pcMsrs = 0;
139 return VERR_NO_MEMORY;
140 }
141 *ppaMsrs = (VBCPUREPMSR *)pvNew;
142 }
143
144 /*
145 * Add it.
146 */
147 VBCPUREPMSR *pEntry = *ppaMsrs + cMsrs;
148 pEntry->uMsr = uMsr;
149 pEntry->fFlags = fFlags;
150 pEntry->uValue = uValue;
151 *pcMsrs = cMsrs + 1;
152
153 return VINF_SUCCESS;
154}
155
156
157/**
158 * Returns the max physical address width as a number of bits.
159 *
160 * @returns Bit count.
161 */
162static uint8_t vbCpuRepGetPhysAddrWidth(void)
163{
164 uint8_t cMaxWidth;
165 uint32_t cMaxExt = ASMCpuId_EAX(0x80000000);
166 if (!ASMHasCpuId())
167 cMaxWidth = 32;
168 else if (ASMIsValidExtRange(cMaxExt)&& cMaxExt >= 0x80000008)
169 cMaxWidth = ASMCpuId_EAX(0x80000008) & 0xff;
170 else if ( ASMIsValidStdRange(ASMCpuId_EAX(0))
171 && (ASMCpuId_EDX(1) & X86_CPUID_FEATURE_EDX_PSE36))
172 cMaxWidth = 36;
173 else
174 cMaxWidth = 32;
175 return cMaxWidth;
176}
177
178
179static bool vbCpuRepSupportsPae(void)
180{
181 return ASMHasCpuId()
182 && ASMIsValidStdRange(ASMCpuId_EAX(0))
183 && (ASMCpuId_EDX(1) & X86_CPUID_FEATURE_EDX_PAE);
184}
185
186
187static bool vbCpuRepSupportsLongMode(void)
188{
189 return ASMHasCpuId()
190 && ASMIsValidExtRange(ASMCpuId_EAX(0x80000000))
191 && (ASMCpuId_EDX(0x80000001) & X86_CPUID_EXT_FEATURE_EDX_LONG_MODE);
192}
193
194
195static bool vbCpuRepSupportsNX(void)
196{
197 return ASMHasCpuId()
198 && ASMIsValidExtRange(ASMCpuId_EAX(0x80000000))
199 && (ASMCpuId_EDX(0x80000001) & X86_CPUID_EXT_FEATURE_EDX_NX);
200}
201
202
203static bool vbCpuRepSupportsX2Apic(void)
204{
205 return ASMHasCpuId()
206 && ASMIsValidStdRange(ASMCpuId_EAX(0))
207 && (ASMCpuId_ECX(1) & X86_CPUID_FEATURE_ECX_X2APIC);
208}
209
210
211
212static bool msrProberWrite(uint32_t uMsr, uint64_t uValue)
213{
214 bool fGp;
215 int rc = SUPR3MsrProberWrite(uMsr, NIL_RTCPUID, uValue, &fGp);
216 AssertRC(rc);
217 return RT_SUCCESS(rc) && !fGp;
218}
219
220
221static bool msrProberRead(uint32_t uMsr, uint64_t *puValue)
222{
223 *puValue = 0;
224 bool fGp;
225 int rc = SUPR3MsrProberRead(uMsr, NIL_RTCPUID, puValue, &fGp);
226 AssertRC(rc);
227 return RT_SUCCESS(rc) && !fGp;
228}
229
230
231/** Tries to modify the register by writing the original value to it. */
232static bool msrProberModifyNoChange(uint32_t uMsr)
233{
234 SUPMSRPROBERMODIFYRESULT Result;
235 int rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, UINT64_MAX, 0, &Result);
236 return RT_SUCCESS(rc)
237 && !Result.fBeforeGp
238 && !Result.fModifyGp
239 && !Result.fAfterGp
240 && !Result.fRestoreGp;
241}
242
243
244/** Tries to modify the register by writing zero to it. */
245static bool msrProberModifyZero(uint32_t uMsr)
246{
247 SUPMSRPROBERMODIFYRESULT Result;
248 int rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, 0, 0, &Result);
249 return RT_SUCCESS(rc)
250 && !Result.fBeforeGp
251 && !Result.fModifyGp
252 && !Result.fAfterGp
253 && !Result.fRestoreGp;
254}
255
256
257/**
258 * Tries to modify each bit in the MSR and see if we can make it change.
259 *
260 * @returns VBox status code.
261 * @param uMsr The MSR.
262 * @param pfIgnMask The ignore mask to update.
263 * @param pfGpMask The GP mask to update.
264 * @param fSkipMask Mask of bits to skip.
265 */
266static int msrProberModifyBitChanges(uint32_t uMsr, uint64_t *pfIgnMask, uint64_t *pfGpMask, uint64_t fSkipMask)
267{
268 for (unsigned iBit = 0; iBit < 64; iBit++)
269 {
270 uint64_t fBitMask = RT_BIT_64(iBit);
271 if (fBitMask & fSkipMask)
272 continue;
273
274 /* Set it. */
275 SUPMSRPROBERMODIFYRESULT ResultSet;
276 int rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, ~fBitMask, fBitMask, &ResultSet);
277 if (RT_FAILURE(rc))
278 return RTMsgErrorRc(rc, "SUPR3MsrProberModify(%#x,,%#llx,%#llx,): %Rrc", uMsr, ~fBitMask, fBitMask, rc);
279
280 /* Clear it. */
281 SUPMSRPROBERMODIFYRESULT ResultClear;
282 rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, ~fBitMask, 0, &ResultClear);
283 if (RT_FAILURE(rc))
284 return RTMsgErrorRc(rc, "SUPR3MsrProberModify(%#x,,%#llx,%#llx,): %Rrc", uMsr, ~fBitMask, 0, rc);
285
286 if (ResultSet.fModifyGp || ResultClear.fModifyGp)
287 *pfGpMask |= fBitMask;
288 else if ( ( ((ResultSet.uBefore ^ ResultSet.uAfter) & fBitMask) == 0
289 && !ResultSet.fBeforeGp
290 && !ResultSet.fAfterGp)
291 && ( ((ResultClear.uBefore ^ ResultClear.uAfter) & fBitMask) == 0
292 && !ResultClear.fBeforeGp
293 && !ResultClear.fAfterGp) )
294 *pfIgnMask |= fBitMask;
295 }
296
297 return VINF_SUCCESS;
298}
299
300
301/**
302 * Tries to modify one bit.
303 *
304 * @retval -2 on API error.
305 * @retval -1 on \#GP.
306 * @retval 0 if ignored.
307 * @retval 1 if it changed.
308 *
309 * @param uMsr The MSR.
310 * @param iBit The bit to try modify.
311 */
312static int msrProberModifyBit(uint32_t uMsr, unsigned iBit)
313{
314 uint64_t fBitMask = RT_BIT_64(iBit);
315
316 /* Set it. */
317 SUPMSRPROBERMODIFYRESULT ResultSet;
318 int rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, ~fBitMask, fBitMask, &ResultSet);
319 if (RT_FAILURE(rc))
320 return RTMsgErrorRc(-2, "SUPR3MsrProberModify(%#x,,%#llx,%#llx,): %Rrc", uMsr, ~fBitMask, fBitMask, rc);
321
322 /* Clear it. */
323 SUPMSRPROBERMODIFYRESULT ResultClear;
324 rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, ~fBitMask, 0, &ResultClear);
325 if (RT_FAILURE(rc))
326 return RTMsgErrorRc(-2, "SUPR3MsrProberModify(%#x,,%#llx,%#llx,): %Rrc", uMsr, ~fBitMask, 0, rc);
327
328 if (ResultSet.fModifyGp || ResultClear.fModifyGp)
329 return -1;
330
331 if ( ( ((ResultSet.uBefore ^ ResultSet.uAfter) & fBitMask) != 0
332 && !ResultSet.fBeforeGp
333 && !ResultSet.fAfterGp)
334 || ( ((ResultClear.uBefore ^ ResultClear.uAfter) & fBitMask) != 0
335 && !ResultClear.fBeforeGp
336 && !ResultClear.fAfterGp) )
337 return 1;
338
339 return 0;
340}
341
342
343/**
344 * Tries to do a simple AND+OR change and see if we \#GP or not.
345 *
346 * @retval @c true if successfully modified.
347 * @retval @c false if \#GP or other error.
348 *
349 * @param uMsr The MSR.
350 * @param fAndMask The AND mask.
351 * @param fOrMask The OR mask.
352 */
353static bool msrProberModifySimpleGp(uint32_t uMsr, uint64_t fAndMask, uint64_t fOrMask)
354{
355 SUPMSRPROBERMODIFYRESULT Result;
356 int rc = SUPR3MsrProberModify(uMsr, NIL_RTCPUID, fAndMask, fOrMask, &Result);
357 if (RT_FAILURE(rc))
358 {
359 RTMsgError("SUPR3MsrProberModify(%#x,,%#llx,%#llx,): %Rrc", uMsr, fAndMask, fOrMask, rc);
360 return false;
361 }
362 return !Result.fBeforeGp
363 && !Result.fModifyGp
364 && !Result.fAfterGp
365 && !Result.fRestoreGp;
366}
367
368
369
370
371/**
372 * Combination of the basic tests.
373 *
374 * @returns VBox status code.
375 * @param uMsr The MSR.
376 * @param fSkipMask Mask of bits to skip.
377 * @param pfReadOnly Where to return read-only status.
378 * @param pfIgnMask Where to return the write ignore mask. Need not
379 * be initialized.
380 * @param pfGpMask Where to return the write GP mask. Need not
381 * be initialized.
382 */
383static int msrProberModifyBasicTests(uint32_t uMsr, uint64_t fSkipMask, bool *pfReadOnly, uint64_t *pfIgnMask, uint64_t *pfGpMask)
384{
385 if (msrProberModifyNoChange(uMsr))
386 {
387 *pfReadOnly = false;
388 *pfIgnMask = 0;
389 *pfGpMask = 0;
390 return msrProberModifyBitChanges(uMsr, pfIgnMask, pfGpMask, fSkipMask);
391 }
392
393 *pfReadOnly = true;
394 *pfIgnMask = 0;
395 *pfGpMask = UINT64_MAX;
396 return VINF_SUCCESS;
397}
398
399
400
401/**
402 * Determines for the MSR AND mask.
403 *
404 * Older CPUs doesn't necessiarly implement all bits of the MSR register number.
405 * So, we have to approximate how many are used so we don't get an overly large
406 * and confusing set of MSRs when probing.
407 *
408 * @returns The mask.
409 */
410static uint32_t determineMsrAndMask(void)
411{
412#define VBCPUREP_MASK_TEST_MSRS 7
413 static uint32_t const s_aMsrs[VBCPUREP_MASK_TEST_MSRS] =
414 {
415 /* Try a bunch of mostly read only registers: */
416 MSR_P5_MC_TYPE, MSR_IA32_PLATFORM_ID, MSR_IA32_MTRR_CAP, MSR_IA32_MCG_CAP, MSR_IA32_CR_PAT,
417 /* Then some which aren't supposed to be present on any CPU: */
418 0x00000015, 0x00000019,
419 };
420
421 /* Get the base values. */
422 uint64_t auBaseValues[VBCPUREP_MASK_TEST_MSRS];
423 for (unsigned i = 0; i < RT_ELEMENTS(s_aMsrs); i++)
424 {
425 if (!msrProberRead(s_aMsrs[i], &auBaseValues[i]))
426 auBaseValues[i] = UINT64_MAX;
427 //vbCpuRepDebug("Base: %#x -> %#llx\n", s_aMsrs[i], auBaseValues[i]);
428 }
429
430 /* Do the probing. */
431 unsigned iBit;
432 for (iBit = 31; iBit > 8; iBit--)
433 {
434 uint64_t fMsrOrMask = RT_BIT_64(iBit);
435 for (unsigned iTest = 0; iTest <= 64 && fMsrOrMask < UINT32_MAX; iTest++)
436 {
437 for (unsigned i = 0; i < RT_ELEMENTS(s_aMsrs); i++)
438 {
439 uint64_t uValue;
440 if (!msrProberRead(s_aMsrs[i] | fMsrOrMask, &uValue))
441 uValue = UINT64_MAX;
442 if (uValue != auBaseValues[i])
443 {
444 uint32_t fMsrMask = iBit >= 31 ? UINT32_MAX : RT_BIT_32(iBit + 1) - 1;
445 vbCpuRepDebug("MSR AND mask: quit on iBit=%u uMsr=%#x (%#x) %llx != %llx => fMsrMask=%#x\n",
446 iBit, s_aMsrs[i] | (uint32_t)fMsrOrMask, s_aMsrs[i], uValue, auBaseValues[i], fMsrMask);
447 return fMsrMask;
448 }
449 }
450
451 /* Advance. */
452 if (iBit <= 6)
453 fMsrOrMask += RT_BIT_64(iBit);
454 else if (iBit <= 11)
455 fMsrOrMask += RT_BIT_64(iBit) * 33;
456 else if (iBit <= 16)
457 fMsrOrMask += RT_BIT_64(iBit) * 1025;
458 else if (iBit <= 22)
459 fMsrOrMask += RT_BIT_64(iBit) * 65537;
460 else
461 fMsrOrMask += RT_BIT_64(iBit) * 262145;
462 }
463 }
464
465 uint32_t fMsrMask = RT_BIT_32(iBit + 1) - 1;
466 vbCpuRepDebug("MSR AND mask: less that %u bits that matters?!? => fMsrMask=%#x\n", iBit + 1, fMsrMask);
467 return fMsrMask;
468}
469
470
471static int findMsrs(VBCPUREPMSR **ppaMsrs, uint32_t *pcMsrs, uint32_t fMsrMask)
472{
473 /*
474 * Gather them.
475 */
476 static struct { uint32_t uFirst, cMsrs; } const s_aRanges[] =
477 {
478 { 0x00000000, 0x00042000 },
479 { 0x10000000, 0x00001000 },
480 { 0x20000000, 0x00001000 },
481 { 0x40000000, 0x00012000 },
482 { 0x80000000, 0x00012000 },
483 { 0xc0000000, 0x00022000 }, /* Had some trouble here on solaris with the tstVMM setup. */
484 };
485
486 *pcMsrs = 0;
487 *ppaMsrs = NULL;
488
489 for (unsigned i = 0; i < RT_ELEMENTS(s_aRanges); i++)
490 {
491 uint32_t uMsr = s_aRanges[i].uFirst;
492 if ((uMsr & fMsrMask) != uMsr)
493 continue;
494 uint32_t cLeft = s_aRanges[i].cMsrs;
495 while (cLeft-- > 0 && (uMsr & fMsrMask) == uMsr)
496 {
497 if ((uMsr & 0xfff) == 0)
498 {
499 vbCpuRepDebug("testing %#x...\n", uMsr);
500 RTThreadSleep(22);
501 }
502#if 0
503 else if (uMsr >= 0x00003170 && uMsr <= 0xc0000090)
504 {
505 vbCpuRepDebug("testing %#x...\n", uMsr);
506 RTThreadSleep(250);
507 }
508#endif
509 /* Skip 0xc0011012..13 as it seems to be bad for our health (Phenom II X6 1100T). */
510 /* Ditto for 0x0000002a (EBL_CR_POWERON) and 0x00000277 (MSR_IA32_CR_PAT) on Intel (Atom 330). */
511 /* And more of the same for 0x280 on Intel Pentium III. */
512 if ( ((uMsr >= 0xc0011012 && uMsr <= 0xc0011013) && g_enmVendor == CPUMCPUVENDOR_AMD)
513 || ( (uMsr == 0x2a || uMsr == 0x277)
514 && g_enmVendor == CPUMCPUVENDOR_INTEL
515 && g_enmMicroarch == kCpumMicroarch_Intel_Atom_Bonnell)
516 || ( (uMsr == 0x280)
517 && g_enmMicroarch == kCpumMicroarch_Intel_P6_III))
518 vbCpuRepDebug("Skipping %#x\n", uMsr);
519 else
520 {
521 /* Read probing normally does it. */
522 uint64_t uValue = 0;
523 bool fGp = true;
524 int rc = SUPR3MsrProberRead(uMsr, NIL_RTCPUID, &uValue, &fGp);
525 if (RT_FAILURE(rc))
526 {
527 RTMemFree(*ppaMsrs);
528 *ppaMsrs = NULL;
529 return RTMsgErrorRc(rc, "SUPR3MsrProberRead failed on %#x: %Rrc\n", uMsr, rc);
530 }
531
532 uint32_t fFlags;
533 if (!fGp)
534 fFlags = 0;
535 /* VIA HACK - writing to 0x0000317e on a quad core make the core unresponsive. */
536 else if (uMsr == 0x0000317e && g_enmVendor == CPUMCPUVENDOR_VIA)
537 {
538 uValue = 0;
539 fFlags = VBCPUREPMSR_F_WRITE_ONLY;
540 fGp = *pcMsrs == 0
541 || (*ppaMsrs)[*pcMsrs - 1].uMsr != 0x0000317d
542 || (*ppaMsrs)[*pcMsrs - 1].fFlags != VBCPUREPMSR_F_WRITE_ONLY;
543 }
544 else
545 {
546 /* Is it a write only register? */
547#if 0
548 if (uMsr >= 0x00003170 && uMsr <= 0xc0000090)
549 {
550 vbCpuRepDebug("test writing %#x...\n", uMsr);
551 RTThreadSleep(250);
552 }
553#endif
554 fGp = true;
555 rc = SUPR3MsrProberWrite(uMsr, NIL_RTCPUID, 0, &fGp);
556 if (RT_FAILURE(rc))
557 {
558 RTMemFree(*ppaMsrs);
559 *ppaMsrs = NULL;
560 return RTMsgErrorRc(rc, "SUPR3MsrProberWrite failed on %#x: %Rrc\n", uMsr, rc);
561 }
562 uValue = 0;
563 fFlags = VBCPUREPMSR_F_WRITE_ONLY;
564
565 /*
566 * Tweaks. On Intel CPUs we've got trouble detecting
567 * IA32_BIOS_UPDT_TRIG (0x00000079), so we have to add it manually here.
568 * Ditto on AMD with PATCH_LOADER (0xc0010020).
569 */
570 if ( uMsr == 0x00000079
571 && fGp
572 && g_enmMicroarch >= kCpumMicroarch_Intel_P6_Core_Atom_First
573 && g_enmMicroarch <= kCpumMicroarch_Intel_End)
574 fGp = false;
575 if ( uMsr == 0xc0010020
576 && fGp
577 && g_enmMicroarch >= kCpumMicroarch_AMD_K8_First
578 && g_enmMicroarch <= kCpumMicroarch_AMD_End)
579 fGp = false;
580 }
581
582 if (!fGp)
583 {
584 /* Add it. */
585 rc = vbCpuRepMsrsAddOne(ppaMsrs, pcMsrs, uMsr, uValue, fFlags);
586 if (RT_FAILURE(rc))
587 return RTMsgErrorRc(rc, "Out of memory (uMsr=%#x).\n", uMsr);
588 if ( g_enmVendor != CPUMCPUVENDOR_VIA
589 || uValue
590 || fFlags)
591 vbCpuRepDebug("%#010x: uValue=%#llx fFlags=%#x\n", uMsr, uValue, fFlags);
592 }
593 }
594
595 uMsr++;
596 }
597 }
598
599 return VINF_SUCCESS;
600}
601
602/**
603 * Get the name of the specified MSR, if we know it and can handle it.
604 *
605 * Do _NOT_ add any new names here without ALSO at the SAME TIME making sure it
606 * is handled correctly by the PROBING CODE and REPORTED correctly!!
607 *
608 * @returns Pointer to name if handled, NULL if not yet explored.
609 * @param uMsr The MSR in question.
610 */
611static const char *getMsrNameHandled(uint32_t uMsr)
612{
613 /** @todo figure out where NCU_EVENT_CORE_MASK might be... */
614 switch (uMsr)
615 {
616 case 0x00000000: return "IA32_P5_MC_ADDR";
617 case 0x00000001: return "IA32_P5_MC_TYPE";
618 case 0x00000006:
619 if (g_enmMicroarch >= kCpumMicroarch_Intel_First && g_enmMicroarch <= kCpumMicroarch_Intel_P6_Core_Atom_First)
620 return NULL; /* TR4 / cache tag on Pentium, but that's for later. */
621 return "IA32_MONITOR_FILTER_LINE_SIZE";
622 //case 0x0000000e: return "P?_TR12"; /* K6-III docs */
623 case 0x00000010: return "IA32_TIME_STAMP_COUNTER";
624 case 0x00000017: return "IA32_PLATFORM_ID";
625 case 0x00000018: return "P6_UNK_0000_0018"; /* P6_M_Dothan. */
626 case 0x0000001b: return "IA32_APIC_BASE";
627 case 0x00000021: return "C2_UNK_0000_0021"; /* Core2_Penryn */
628 case 0x0000002a: return g_fIntelNetBurst ? "P4_EBC_HARD_POWERON" : "EBL_CR_POWERON";
629 case 0x0000002b: return g_fIntelNetBurst ? "P4_EBC_SOFT_POWERON" : NULL;
630 case 0x0000002c: return g_fIntelNetBurst ? "P4_EBC_FREQUENCY_ID" : NULL;
631 case 0x0000002e: return "I7_UNK_0000_002e"; /* SandyBridge, IvyBridge. */
632 case 0x0000002f: return "P6_UNK_0000_002f"; /* P6_M_Dothan. */
633 case 0x00000032: return "P6_UNK_0000_0032"; /* P6_M_Dothan. */
634 case 0x00000033: return "TEST_CTL";
635 case 0x00000034: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch)
636 || CPUMMICROARCH_IS_INTEL_SILVERMONT_PLUS(g_enmMicroarch)
637 ? "MSR_SMI_COUNT" : "P6_UNK_0000_0034"; /* P6_M_Dothan. */
638 case 0x00000035: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch) ? "MSR_CORE_THREAD_COUNT" : "P6_UNK_0000_0035"; /* P6_M_Dothan. */
639 case 0x00000036: return "I7_UNK_0000_0036"; /* SandyBridge, IvyBridge. */
640 case 0x00000039: return "C2_UNK_0000_0039"; /* Core2_Penryn */
641 case 0x0000003a: return "IA32_FEATURE_CONTROL";
642 case 0x0000003b: return "P6_UNK_0000_003b"; /* P6_M_Dothan. */
643 case 0x0000003e: return "I7_UNK_0000_003e"; /* SandyBridge, IvyBridge. */
644 case 0x0000003f: return "P6_UNK_0000_003f"; /* P6_M_Dothan. */
645 case 0x00000040: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_0_FROM_IP" : "MSR_LASTBRANCH_0";
646 case 0x00000041: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_1_FROM_IP" : "MSR_LASTBRANCH_1";
647 case 0x00000042: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_2_FROM_IP" : "MSR_LASTBRANCH_2";
648 case 0x00000043: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_3_FROM_IP" : "MSR_LASTBRANCH_3";
649 case 0x00000044: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_4_FROM_IP" : "MSR_LASTBRANCH_4";
650 case 0x00000045: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_5_FROM_IP" : "MSR_LASTBRANCH_5";
651 case 0x00000046: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_6_FROM_IP" : "MSR_LASTBRANCH_6";
652 case 0x00000047: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_7_FROM_IP" : "MSR_LASTBRANCH_7";
653 case 0x00000048: return "MSR_LASTBRANCH_8"; /*??*/
654 case 0x00000049: return "MSR_LASTBRANCH_9"; /*??*/
655 case 0x0000004a: return "P6_UNK_0000_004a"; /* P6_M_Dothan. */
656 case 0x0000004b: return "P6_UNK_0000_004b"; /* P6_M_Dothan. */
657 case 0x0000004c: return "P6_UNK_0000_004c"; /* P6_M_Dothan. */
658 case 0x0000004d: return "P6_UNK_0000_004d"; /* P6_M_Dothan. */
659 case 0x0000004e: return "P6_UNK_0000_004e"; /* P6_M_Dothan. */
660 case 0x0000004f: return "P6_UNK_0000_004f"; /* P6_M_Dothan. */
661 case 0x00000050: return "P6_UNK_0000_0050"; /* P6_M_Dothan. */
662 case 0x00000051: return "P6_UNK_0000_0051"; /* P6_M_Dothan. */
663 case 0x00000052: return "P6_UNK_0000_0052"; /* P6_M_Dothan. */
664 case 0x00000053: return "P6_UNK_0000_0053"; /* P6_M_Dothan. */
665 case 0x00000054: return "P6_UNK_0000_0054"; /* P6_M_Dothan. */
666 case 0x00000060: return "MSR_LASTBRANCH_0_TO_IP"; /* Core2_Penryn */
667 case 0x00000061: return "MSR_LASTBRANCH_1_TO_IP"; /* Core2_Penryn */
668 case 0x00000062: return "MSR_LASTBRANCH_2_TO_IP"; /* Core2_Penryn */
669 case 0x00000063: return "MSR_LASTBRANCH_3_TO_IP"; /* Core2_Penryn */
670 case 0x00000064: return "MSR_LASTBRANCH_4_TO_IP"; /* Atom? */
671 case 0x00000065: return "MSR_LASTBRANCH_5_TO_IP";
672 case 0x00000066: return "MSR_LASTBRANCH_6_TO_IP";
673 case 0x00000067: return "MSR_LASTBRANCH_7_TO_IP";
674 case 0x0000006c: return "P6_UNK_0000_006c"; /* P6_M_Dothan. */
675 case 0x0000006d: return "P6_UNK_0000_006d"; /* P6_M_Dothan. */
676 case 0x0000006e: return "P6_UNK_0000_006e"; /* P6_M_Dothan. */
677 case 0x0000006f: return "P6_UNK_0000_006f"; /* P6_M_Dothan. */
678 case 0x00000079: return "IA32_BIOS_UPDT_TRIG";
679 case 0x00000080: return "P4_UNK_0000_0080";
680 case 0x00000088: return "BBL_CR_D0";
681 case 0x00000089: return "BBL_CR_D1";
682 case 0x0000008a: return "BBL_CR_D2";
683 case 0x0000008b: return g_enmVendor == CPUMCPUVENDOR_AMD ? "AMD_K8_PATCH_LEVEL"
684 : g_fIntelNetBurst ? "IA32_BIOS_SIGN_ID" : "BBL_CR_D3|BIOS_SIGN";
685 case 0x0000008c: return "P6_UNK_0000_008c"; /* P6_M_Dothan. */
686 case 0x0000008d: return "P6_UNK_0000_008d"; /* P6_M_Dothan. */
687 case 0x0000008e: return "P6_UNK_0000_008e"; /* P6_M_Dothan. */
688 case 0x0000008f: return "P6_UNK_0000_008f"; /* P6_M_Dothan. */
689 case 0x00000090: return "P6_UNK_0000_0090"; /* P6_M_Dothan. */
690 case 0x0000009b: return "IA32_SMM_MONITOR_CTL";
691 case 0x000000a8: return "C2_EMTTM_CR_TABLES_0";
692 case 0x000000a9: return "C2_EMTTM_CR_TABLES_1";
693 case 0x000000aa: return "C2_EMTTM_CR_TABLES_2";
694 case 0x000000ab: return "C2_EMTTM_CR_TABLES_3";
695 case 0x000000ac: return "C2_EMTTM_CR_TABLES_4";
696 case 0x000000ad: return "C2_EMTTM_CR_TABLES_5";
697 case 0x000000ae: return "P6_UNK_0000_00ae"; /* P6_M_Dothan. */
698 case 0x000000c1: return "IA32_PMC0";
699 case 0x000000c2: return "IA32_PMC1";
700 case 0x000000c3: return "IA32_PMC2";
701 case 0x000000c4: return "IA32_PMC3";
702 /* PMC4+ first seen on SandyBridge. The earlier cut off is just to be
703 on the safe side as we must avoid P6_M_Dothan and possibly others. */
704 case 0x000000c5: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? "IA32_PMC4" : NULL;
705 case 0x000000c6: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? "IA32_PMC5" : NULL;
706 case 0x000000c7: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? "IA32_PMC6" : "P6_UNK_0000_00c7"; /* P6_M_Dothan. */
707 case 0x000000c8: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? "IA32_PMC7" : NULL;
708 case 0x000000cd: return "MSR_FSB_FREQ"; /* P6_M_Dothan. */
709 case 0x000000ce: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? "IA32_PLATFORM_INFO" : "P6_UNK_0000_00ce"; /* P6_M_Dothan. */
710 case 0x000000cf: return "C2_UNK_0000_00cf"; /* Core2_Penryn. */
711 case 0x000000e0: return "C2_UNK_0000_00e0"; /* Core2_Penryn. */
712 case 0x000000e1: return "C2_UNK_0000_00e1"; /* Core2_Penryn. */
713 case 0x000000e2: return "MSR_PKG_CST_CONFIG_CONTROL";
714 case 0x000000e3: return "C2_SMM_CST_MISC_INFO"; /* Core2_Penryn. */
715 case 0x000000e4: return "MSR_PMG_IO_CAPTURE_BASE";
716 case 0x000000e5: return "C2_UNK_0000_00e5"; /* Core2_Penryn. */
717 case 0x000000e7: return "IA32_MPERF";
718 case 0x000000e8: return "IA32_APERF";
719 case 0x000000ee: return "C1_EXT_CONFIG"; /* Core2_Penryn. msrtool lists it for Core1 as well. */
720 case 0x000000fe: return "IA32_MTRRCAP";
721 case 0x00000102: return "I7_IB_UNK_0000_0102"; /* IvyBridge. */
722 case 0x00000103: return "I7_IB_UNK_0000_0103"; /* IvyBridge. */
723 case 0x00000104: return "I7_IB_UNK_0000_0104"; /* IvyBridge. */
724 case 0x00000116: return "BBL_CR_ADDR";
725 case 0x00000118: return "BBL_CR_DECC";
726 case 0x00000119: return "BBL_CR_CTL";
727 case 0x0000011a: return "BBL_CR_TRIG";
728 case 0x0000011b: return "P6_UNK_0000_011b"; /* P6_M_Dothan. */
729 case 0x0000011c: return "C2_UNK_0000_011c"; /* Core2_Penryn. */
730 case 0x0000011e: return "BBL_CR_CTL3";
731 case 0x00000120: return "SILV_UNK_0000_0120"; /* Silvermont */
732 case 0x00000130: return g_enmMicroarch == kCpumMicroarch_Intel_Core7_Westmere
733 || g_enmMicroarch == kCpumMicroarch_Intel_Core7_Nehalem
734 ? "CPUID1_FEATURE_MASK" : NULL;
735 case 0x00000131: return g_enmMicroarch == kCpumMicroarch_Intel_Core7_Westmere
736 || g_enmMicroarch == kCpumMicroarch_Intel_Core7_Nehalem
737 ? "CPUID80000001_FEATURE_MASK" : "P6_UNK_0000_0131" /* P6_M_Dothan. */;
738 case 0x00000132: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
739 ? "CPUID1_FEATURE_MASK" : NULL;
740 case 0x00000133: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
741 ? "CPUIDD_01_FEATURE_MASK" : NULL;
742 case 0x00000134: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
743 ? "CPUID80000001_FEATURE_MASK" : NULL;
744 case 0x0000013c: return "I7_SB_AES_NI_CTL"; /* SandyBridge. Bit 0 is lock bit, bit 1 disables AES-NI. */
745 case 0x00000140: return "I7_IB_UNK_0000_0140"; /* IvyBridge. */
746 case 0x00000142: return "I7_IB_UNK_0000_0142"; /* IvyBridge. */
747 case 0x0000014e: return "P6_UNK_0000_014e"; /* P6_M_Dothan. */
748 case 0x0000014f: return "P6_UNK_0000_014f"; /* P6_M_Dothan. */
749 case 0x00000150: return "P6_UNK_0000_0150"; /* P6_M_Dothan. */
750 case 0x00000151: return "P6_UNK_0000_0151"; /* P6_M_Dothan. */
751 case 0x00000154: return "P6_UNK_0000_0154"; /* P6_M_Dothan. */
752 case 0x0000015b: return "P6_UNK_0000_015b"; /* P6_M_Dothan. */
753 case 0x0000015e: return "C2_UNK_0000_015e"; /* Core2_Penryn. */
754 case 0x0000015f: return "C1_DTS_CAL_CTRL"; /* Core2_Penryn. msrtool only docs this for core1! */
755 case 0x00000174: return "IA32_SYSENTER_CS";
756 case 0x00000175: return "IA32_SYSENTER_ESP";
757 case 0x00000176: return "IA32_SYSENTER_EIP";
758 case 0x00000179: return "IA32_MCG_CAP";
759 case 0x0000017a: return "IA32_MCG_STATUS";
760 case 0x0000017b: return "IA32_MCG_CTL";
761 case 0x0000017f: return "I7_SB_ERROR_CONTROL"; /* SandyBridge. */
762 case 0x00000180: return g_fIntelNetBurst ? "MSR_MCG_RAX" : NULL;
763 case 0x00000181: return g_fIntelNetBurst ? "MSR_MCG_RBX" : NULL;
764 case 0x00000182: return g_fIntelNetBurst ? "MSR_MCG_RCX" : NULL;
765 case 0x00000183: return g_fIntelNetBurst ? "MSR_MCG_RDX" : NULL;
766 case 0x00000184: return g_fIntelNetBurst ? "MSR_MCG_RSI" : NULL;
767 case 0x00000185: return g_fIntelNetBurst ? "MSR_MCG_RDI" : NULL;
768 case 0x00000186: return g_fIntelNetBurst ? "MSR_MCG_RBP" : "IA32_PERFEVTSEL0";
769 case 0x00000187: return g_fIntelNetBurst ? "MSR_MCG_RSP" : "IA32_PERFEVTSEL1";
770 case 0x00000188: return g_fIntelNetBurst ? "MSR_MCG_RFLAGS" : "IA32_PERFEVTSEL2";
771 case 0x00000189: return g_fIntelNetBurst ? "MSR_MCG_RIP" : "IA32_PERFEVTSEL3";
772 case 0x0000018a: return g_fIntelNetBurst ? "MSR_MCG_MISC" : "IA32_PERFEVTSEL4";
773 case 0x0000018b: return g_fIntelNetBurst ? "MSR_MCG_RESERVED1" : "IA32_PERFEVTSEL5";
774 case 0x0000018c: return g_fIntelNetBurst ? "MSR_MCG_RESERVED2" : "IA32_PERFEVTSEL6";
775 case 0x0000018d: return g_fIntelNetBurst ? "MSR_MCG_RESERVED3" : "IA32_PERFEVTSEL7";
776 case 0x0000018e: return g_fIntelNetBurst ? "MSR_MCG_RESERVED4" : "IA32_PERFEVTSEL8";
777 case 0x0000018f: return g_fIntelNetBurst ? "MSR_MCG_RESERVED5" : "IA32_PERFEVTSEL9";
778 case 0x00000190: return g_fIntelNetBurst ? "MSR_MCG_R8" : NULL;
779 case 0x00000191: return g_fIntelNetBurst ? "MSR_MCG_R9" : NULL;
780 case 0x00000192: return g_fIntelNetBurst ? "MSR_MCG_R10" : NULL;
781 case 0x00000193: return g_fIntelNetBurst ? "MSR_MCG_R11" : "C2_UNK_0000_0193";
782 case 0x00000194: return g_fIntelNetBurst ? "MSR_MCG_R12" : "CLOCK_FLEX_MAX";
783 case 0x00000195: return g_fIntelNetBurst ? "MSR_MCG_R13" : NULL;
784 case 0x00000196: return g_fIntelNetBurst ? "MSR_MCG_R14" : NULL;
785 case 0x00000197: return g_fIntelNetBurst ? "MSR_MCG_R15" : NULL;
786 case 0x00000198: return "IA32_PERF_STATUS";
787 case 0x00000199: return "IA32_PERF_CTL";
788 case 0x0000019a: return "IA32_CLOCK_MODULATION";
789 case 0x0000019b: return "IA32_THERM_INTERRUPT";
790 case 0x0000019c: return "IA32_THERM_STATUS";
791 case 0x0000019d: return "IA32_THERM2_CTL";
792 case 0x0000019e: return "P6_UNK_0000_019e"; /* P6_M_Dothan. */
793 case 0x0000019f: return "P6_UNK_0000_019f"; /* P6_M_Dothan. */
794 case 0x000001a0: return "IA32_MISC_ENABLE";
795 case 0x000001a1: return g_fIntelNetBurst ? "MSR_PLATFORM_BRV" : "P6_UNK_0000_01a1" /* P6_M_Dothan. */;
796 case 0x000001a2: return g_fIntelNetBurst ? "P4_UNK_0000_01a2" : "I7_MSR_TEMPERATURE_TARGET" /* SandyBridge, IvyBridge. */;
797 case 0x000001a4: return "I7_UNK_0000_01a4"; /* SandyBridge, IvyBridge. */
798 case 0x000001a6: return "I7_MSR_OFFCORE_RSP_0";
799 case 0x000001a7: return "I7_MSR_OFFCORE_RSP_1";
800 case 0x000001a8: return "I7_UNK_0000_01a8"; /* SandyBridge, IvyBridge. */
801 case 0x000001aa: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch) ? "MSR_MISC_PWR_MGMT" : "P6_PIC_SENS_CFG" /* Pentium M. */;
802 case 0x000001ad: return "I7_MSR_TURBO_RATIO_LIMIT"; /* SandyBridge+, Silvermount+ */
803 case 0x000001ae: return "P6_UNK_0000_01ae"; /* P6_M_Dothan. */
804 case 0x000001af: return "P6_UNK_0000_01af"; /* P6_M_Dothan. */
805 case 0x000001b0: return "IA32_ENERGY_PERF_BIAS";
806 case 0x000001b1: return "IA32_PACKAGE_THERM_STATUS";
807 case 0x000001b2: return "IA32_PACKAGE_THERM_INTERRUPT";
808 case 0x000001bf: return "C2_UNK_0000_01bf"; /* Core2_Penryn. */
809 case 0x000001c6: return "I7_UNK_0000_01c6"; /* SandyBridge*/
810 case 0x000001c8: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_Nehalem ? "MSR_LBR_SELECT" : NULL;
811 case 0x000001c9: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah
812 && g_enmMicroarch <= kCpumMicroarch_Intel_P6_Core_Atom_End
813 ? "MSR_LASTBRANCH_TOS" : NULL /* Pentium M Dothan seems to have something else here. */;
814 case 0x000001d3: return "P6_UNK_0000_01d3"; /* P6_M_Dothan. */
815 case 0x000001d7: return g_fIntelNetBurst ? "MSR_LER_FROM_LIP" : NULL;
816 case 0x000001d8: return g_fIntelNetBurst ? "MSR_LER_TO_LIP" : NULL;
817 case 0x000001d9: return "IA32_DEBUGCTL";
818 case 0x000001da: return g_fIntelNetBurst ? "MSR_LASTBRANCH_TOS" : NULL;
819 case 0x000001db: return g_fIntelNetBurst ? "P6_LASTBRANCH_0" : "P6_LAST_BRANCH_FROM_IP"; /* Not exclusive to P6, also AMD. */
820 case 0x000001dc: return g_fIntelNetBurst ? "P6_LASTBRANCH_1" : "P6_LAST_BRANCH_TO_IP";
821 case 0x000001dd: return g_fIntelNetBurst ? "P6_LASTBRANCH_2" : "P6_LAST_INT_FROM_IP";
822 case 0x000001de: return g_fIntelNetBurst ? "P6_LASTBRANCH_3" : "P6_LAST_INT_TO_IP";
823 case 0x000001e0: return "MSR_ROB_CR_BKUPTMPDR6";
824 case 0x000001e1: return "I7_SB_UNK_0000_01e1";
825 case 0x000001ef: return "I7_SB_UNK_0000_01ef";
826 case 0x000001f0: return "I7_VLW_CAPABILITY"; /* SandyBridge. Bit 1 is A20M and was implemented incorrectly (AAJ49). */
827 case 0x000001f2: return "IA32_SMRR_PHYSBASE";
828 case 0x000001f3: return "IA32_SMRR_PHYSMASK";
829 case 0x000001f8: return "IA32_PLATFORM_DCA_CAP";
830 case 0x000001f9: return "IA32_CPU_DCA_CAP";
831 case 0x000001fa: return "IA32_DCA_0_CAP";
832 case 0x000001fc: return "I7_MSR_POWER_CTL";
833
834 case 0x00000200: return "IA32_MTRR_PHYS_BASE0";
835 case 0x00000202: return "IA32_MTRR_PHYS_BASE1";
836 case 0x00000204: return "IA32_MTRR_PHYS_BASE2";
837 case 0x00000206: return "IA32_MTRR_PHYS_BASE3";
838 case 0x00000208: return "IA32_MTRR_PHYS_BASE4";
839 case 0x0000020a: return "IA32_MTRR_PHYS_BASE5";
840 case 0x0000020c: return "IA32_MTRR_PHYS_BASE6";
841 case 0x0000020e: return "IA32_MTRR_PHYS_BASE7";
842 case 0x00000210: return "IA32_MTRR_PHYS_BASE8";
843 case 0x00000212: return "IA32_MTRR_PHYS_BASE9";
844 case 0x00000214: return "IA32_MTRR_PHYS_BASE10";
845 case 0x00000216: return "IA32_MTRR_PHYS_BASE11";
846 case 0x00000218: return "IA32_MTRR_PHYS_BASE12";
847 case 0x0000021a: return "IA32_MTRR_PHYS_BASE13";
848 case 0x0000021c: return "IA32_MTRR_PHYS_BASE14";
849 case 0x0000021e: return "IA32_MTRR_PHYS_BASE15";
850
851 case 0x00000201: return "IA32_MTRR_PHYS_MASK0";
852 case 0x00000203: return "IA32_MTRR_PHYS_MASK1";
853 case 0x00000205: return "IA32_MTRR_PHYS_MASK2";
854 case 0x00000207: return "IA32_MTRR_PHYS_MASK3";
855 case 0x00000209: return "IA32_MTRR_PHYS_MASK4";
856 case 0x0000020b: return "IA32_MTRR_PHYS_MASK5";
857 case 0x0000020d: return "IA32_MTRR_PHYS_MASK6";
858 case 0x0000020f: return "IA32_MTRR_PHYS_MASK7";
859 case 0x00000211: return "IA32_MTRR_PHYS_MASK8";
860 case 0x00000213: return "IA32_MTRR_PHYS_MASK9";
861 case 0x00000215: return "IA32_MTRR_PHYS_MASK10";
862 case 0x00000217: return "IA32_MTRR_PHYS_MASK11";
863 case 0x00000219: return "IA32_MTRR_PHYS_MASK12";
864 case 0x0000021b: return "IA32_MTRR_PHYS_MASK13";
865 case 0x0000021d: return "IA32_MTRR_PHYS_MASK14";
866 case 0x0000021f: return "IA32_MTRR_PHYS_MASK15";
867
868 case 0x00000250: return "IA32_MTRR_FIX64K_00000";
869 case 0x00000258: return "IA32_MTRR_FIX16K_80000";
870 case 0x00000259: return "IA32_MTRR_FIX16K_A0000";
871 case 0x00000268: return "IA32_MTRR_FIX4K_C0000";
872 case 0x00000269: return "IA32_MTRR_FIX4K_C8000";
873 case 0x0000026a: return "IA32_MTRR_FIX4K_D0000";
874 case 0x0000026b: return "IA32_MTRR_FIX4K_D8000";
875 case 0x0000026c: return "IA32_MTRR_FIX4K_E0000";
876 case 0x0000026d: return "IA32_MTRR_FIX4K_E8000";
877 case 0x0000026e: return "IA32_MTRR_FIX4K_F0000";
878 case 0x0000026f: return "IA32_MTRR_FIX4K_F8000";
879 case 0x00000277: return "IA32_PAT";
880 case 0x00000280: return "IA32_MC0_CTL2";
881 case 0x00000281: return "IA32_MC1_CTL2";
882 case 0x00000282: return "IA32_MC2_CTL2";
883 case 0x00000283: return "IA32_MC3_CTL2";
884 case 0x00000284: return "IA32_MC4_CTL2";
885 case 0x00000285: return "IA32_MC5_CTL2";
886 case 0x00000286: return "IA32_MC6_CTL2";
887 case 0x00000287: return "IA32_MC7_CTL2";
888 case 0x00000288: return "IA32_MC8_CTL2";
889 case 0x00000289: return "IA32_MC9_CTL2";
890 case 0x0000028a: return "IA32_MC10_CTL2";
891 case 0x0000028b: return "IA32_MC11_CTL2";
892 case 0x0000028c: return "IA32_MC12_CTL2";
893 case 0x0000028d: return "IA32_MC13_CTL2";
894 case 0x0000028e: return "IA32_MC14_CTL2";
895 case 0x0000028f: return "IA32_MC15_CTL2";
896 case 0x00000290: return "IA32_MC16_CTL2";
897 case 0x00000291: return "IA32_MC17_CTL2";
898 case 0x00000292: return "IA32_MC18_CTL2";
899 case 0x00000293: return "IA32_MC19_CTL2";
900 case 0x00000294: return "IA32_MC20_CTL2";
901 case 0x00000295: return "IA32_MC21_CTL2";
902 //case 0x00000296: return "IA32_MC22_CTL2";
903 //case 0x00000297: return "IA32_MC23_CTL2";
904 //case 0x00000298: return "IA32_MC24_CTL2";
905 //case 0x00000299: return "IA32_MC25_CTL2";
906 //case 0x0000029a: return "IA32_MC26_CTL2";
907 //case 0x0000029b: return "IA32_MC27_CTL2";
908 //case 0x0000029c: return "IA32_MC28_CTL2";
909 //case 0x0000029d: return "IA32_MC29_CTL2";
910 //case 0x0000029e: return "IA32_MC30_CTL2";
911 //case 0x0000029f: return "IA32_MC31_CTL2";
912 case 0x000002e0: return "I7_SB_NO_EVICT_MODE"; /* (Bits 1 & 0 are said to have something to do with no-evict cache mode used during early boot.) */
913 case 0x000002e6: return "I7_IB_UNK_0000_02e6"; /* IvyBridge */
914 case 0x000002e7: return "I7_IB_UNK_0000_02e7"; /* IvyBridge */
915 case 0x000002ff: return "IA32_MTRR_DEF_TYPE";
916 case 0x00000300: return g_fIntelNetBurst ? "P4_MSR_BPU_COUNTER0" : "I7_SB_UNK_0000_0300" /* SandyBridge */;
917 case 0x00000301: return g_fIntelNetBurst ? "P4_MSR_BPU_COUNTER1" : NULL;
918 case 0x00000302: return g_fIntelNetBurst ? "P4_MSR_BPU_COUNTER2" : NULL;
919 case 0x00000303: return g_fIntelNetBurst ? "P4_MSR_BPU_COUNTER3" : NULL;
920 case 0x00000304: return g_fIntelNetBurst ? "P4_MSR_MS_COUNTER0" : NULL;
921 case 0x00000305: return g_fIntelNetBurst ? "P4_MSR_MS_COUNTER1" : "I7_SB_UNK_0000_0305" /* SandyBridge, IvyBridge */;
922 case 0x00000306: return g_fIntelNetBurst ? "P4_MSR_MS_COUNTER2" : NULL;
923 case 0x00000307: return g_fIntelNetBurst ? "P4_MSR_MS_COUNTER3" : NULL;
924 case 0x00000308: return g_fIntelNetBurst ? "P4_MSR_FLAME_COUNTER0" : NULL;
925 case 0x00000309: return g_fIntelNetBurst ? "P4_MSR_FLAME_COUNTER1" : "IA32_FIXED_CTR0";
926 case 0x0000030a: return g_fIntelNetBurst ? "P4_MSR_FLAME_COUNTER2" : "IA32_FIXED_CTR1";
927 case 0x0000030b: return g_fIntelNetBurst ? "P4_MSR_FLAME_COUNTER3" : "IA32_FIXED_CTR2";
928 case 0x0000030c: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER0" : NULL;
929 case 0x0000030d: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER1" : NULL;
930 case 0x0000030e: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER2" : NULL;
931 case 0x0000030f: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER3" : NULL;
932 case 0x00000310: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER4" : NULL;
933 case 0x00000311: return g_fIntelNetBurst ? "P4_MSR_IQ_COUNTER5" : NULL;
934 case 0x00000345: return "IA32_PERF_CAPABILITIES";
935 case 0x00000360: return g_fIntelNetBurst ? "P4_MSR_BPU_CCCR0" : NULL;
936 case 0x00000361: return g_fIntelNetBurst ? "P4_MSR_BPU_CCCR1" : NULL;
937 case 0x00000362: return g_fIntelNetBurst ? "P4_MSR_BPU_CCCR2" : NULL;
938 case 0x00000363: return g_fIntelNetBurst ? "P4_MSR_BPU_CCCR3" : NULL;
939 case 0x00000364: return g_fIntelNetBurst ? "P4_MSR_MS_CCCR0" : NULL;
940 case 0x00000365: return g_fIntelNetBurst ? "P4_MSR_MS_CCCR1" : NULL;
941 case 0x00000366: return g_fIntelNetBurst ? "P4_MSR_MS_CCCR2" : NULL;
942 case 0x00000367: return g_fIntelNetBurst ? "P4_MSR_MS_CCCR3" : NULL;
943 case 0x00000368: return g_fIntelNetBurst ? "P4_MSR_FLAME_CCCR0" : NULL;
944 case 0x00000369: return g_fIntelNetBurst ? "P4_MSR_FLAME_CCCR1" : NULL;
945 case 0x0000036a: return g_fIntelNetBurst ? "P4_MSR_FLAME_CCCR2" : NULL;
946 case 0x0000036b: return g_fIntelNetBurst ? "P4_MSR_FLAME_CCCR3" : NULL;
947 case 0x0000036c: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR0" : NULL;
948 case 0x0000036d: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR1" : NULL;
949 case 0x0000036e: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR2" : NULL;
950 case 0x0000036f: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR3" : NULL;
951 case 0x00000370: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR4" : NULL;
952 case 0x00000371: return g_fIntelNetBurst ? "P4_MSR_IQ_CCCR5" : NULL;
953 case 0x0000038d: return "IA32_FIXED_CTR_CTRL";
954 case 0x0000038e: return "IA32_PERF_GLOBAL_STATUS";
955 case 0x0000038f: return "IA32_PERF_GLOBAL_CTRL";
956 case 0x00000390: return "IA32_PERF_GLOBAL_OVF_CTRL";
957 case 0x00000391: return "I7_UNC_PERF_GLOBAL_CTRL"; /* S,H,X */
958 case 0x00000392: return "I7_UNC_PERF_GLOBAL_STATUS"; /* S,H,X */
959 case 0x00000393: return "I7_UNC_PERF_GLOBAL_OVF_CTRL"; /* X. ASSUMING this is the same on sandybridge and later. */
960 case 0x00000394: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PERF_FIXED_CTR" /* X */ : "I7_UNC_PERF_FIXED_CTR_CTRL"; /* >= S,H */
961 case 0x00000395: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PERF_FIXED_CTR_CTRL" /* X*/ : "I7_UNC_PERF_FIXED_CTR"; /* >= S,H */
962 case 0x00000396: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_ADDR_OPCODE_MATCH" /* X */ : "I7_UNC_CBO_CONFIG"; /* >= S,H */
963 case 0x00000397: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? NULL : "I7_SB_UNK_0000_0397";
964 case 0x0000039c: return "I7_SB_MSR_PEBS_NUM_ALT";
965 case 0x000003a0: return g_fIntelNetBurst ? "P4_MSR_BSU_ESCR0" : NULL;
966 case 0x000003a1: return g_fIntelNetBurst ? "P4_MSR_BSU_ESCR1" : NULL;
967 case 0x000003a2: return g_fIntelNetBurst ? "P4_MSR_FSB_ESCR0" : NULL;
968 case 0x000003a3: return g_fIntelNetBurst ? "P4_MSR_FSB_ESCR1" : NULL;
969 case 0x000003a4: return g_fIntelNetBurst ? "P4_MSR_FIRM_ESCR0" : NULL;
970 case 0x000003a5: return g_fIntelNetBurst ? "P4_MSR_FIRM_ESCR1" : NULL;
971 case 0x000003a6: return g_fIntelNetBurst ? "P4_MSR_FLAME_ESCR0" : NULL;
972 case 0x000003a7: return g_fIntelNetBurst ? "P4_MSR_FLAME_ESCR1" : NULL;
973 case 0x000003a8: return g_fIntelNetBurst ? "P4_MSR_DAC_ESCR0" : NULL;
974 case 0x000003a9: return g_fIntelNetBurst ? "P4_MSR_DAC_ESCR1" : NULL;
975 case 0x000003aa: return g_fIntelNetBurst ? "P4_MSR_MOB_ESCR0" : NULL;
976 case 0x000003ab: return g_fIntelNetBurst ? "P4_MSR_MOB_ESCR1" : NULL;
977 case 0x000003ac: return g_fIntelNetBurst ? "P4_MSR_PMH_ESCR0" : NULL;
978 case 0x000003ad: return g_fIntelNetBurst ? "P4_MSR_PMH_ESCR1" : NULL;
979 case 0x000003ae: return g_fIntelNetBurst ? "P4_MSR_SAAT_ESCR0" : NULL;
980 case 0x000003af: return g_fIntelNetBurst ? "P4_MSR_SAAT_ESCR1" : NULL;
981 case 0x000003b0: return g_fIntelNetBurst ? "P4_MSR_U2L_ESCR0" : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PMC0" /* X */ : "I7_UNC_ARB_PERF_CTR0"; /* >= S,H */
982 case 0x000003b1: return g_fIntelNetBurst ? "P4_MSR_U2L_ESCR1" : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PMC1" /* X */ : "I7_UNC_ARB_PERF_CTR1"; /* >= S,H */
983 case 0x000003b2: return g_fIntelNetBurst ? "P4_MSR_BPU_ESCR0" : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PMC2" /* X */ : "I7_UNC_ARB_PERF_EVT_SEL0"; /* >= S,H */
984 case 0x000003b3: return g_fIntelNetBurst ? "P4_MSR_BPU_ESCR1" : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "I7_UNC_PMC3" /* X */ : "I7_UNC_ARB_PERF_EVT_SEL1"; /* >= S,H */
985 case 0x000003b4: return g_fIntelNetBurst ? "P4_MSR_IS_ESCR0" : "I7_UNC_PMC4";
986 case 0x000003b5: return g_fIntelNetBurst ? "P4_MSR_IS_ESCR1" : "I7_UNC_PMC5";
987 case 0x000003b6: return g_fIntelNetBurst ? "P4_MSR_ITLB_ESCR0" : "I7_UNC_PMC6";
988 case 0x000003b7: return g_fIntelNetBurst ? "P4_MSR_ITLB_ESCR1" : "I7_UNC_PMC7";
989 case 0x000003b8: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR0" : NULL;
990 case 0x000003b9: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR1" : NULL;
991 case 0x000003ba: return g_fIntelNetBurst ? "P4_MSR_IQ_ESCR0" : NULL;
992 case 0x000003bb: return g_fIntelNetBurst ? "P4_MSR_IQ_ESCR1" : NULL;
993 case 0x000003bc: return g_fIntelNetBurst ? "P4_MSR_RAT_ESCR0" : NULL;
994 case 0x000003bd: return g_fIntelNetBurst ? "P4_MSR_RAT_ESCR1" : NULL;
995 case 0x000003be: return g_fIntelNetBurst ? "P4_MSR_SSU_ESCR0" : NULL;
996 case 0x000003c0: return g_fIntelNetBurst ? "P4_MSR_MS_ESCR0" : "I7_UNC_PERF_EVT_SEL0";
997 case 0x000003c1: return g_fIntelNetBurst ? "P4_MSR_MS_ESCR1" : "I7_UNC_PERF_EVT_SEL1";
998 case 0x000003c2: return g_fIntelNetBurst ? "P4_MSR_TBPU_ESCR0" : "I7_UNC_PERF_EVT_SEL2";
999 case 0x000003c3: return g_fIntelNetBurst ? "P4_MSR_TBPU_ESCR1" : "I7_UNC_PERF_EVT_SEL3";
1000 case 0x000003c4: return g_fIntelNetBurst ? "P4_MSR_TC_ESCR0" : "I7_UNC_PERF_EVT_SEL4";
1001 case 0x000003c5: return g_fIntelNetBurst ? "P4_MSR_TC_ESCR1" : "I7_UNC_PERF_EVT_SEL5";
1002 case 0x000003c6: return g_fIntelNetBurst ? NULL : "I7_UNC_PERF_EVT_SEL6";
1003 case 0x000003c7: return g_fIntelNetBurst ? NULL : "I7_UNC_PERF_EVT_SEL7";
1004 case 0x000003c8: return g_fIntelNetBurst ? "P4_MSR_IX_ESCR0" : NULL;
1005 case 0x000003c9: return g_fIntelNetBurst ? "P4_MSR_IX_ESCR0" : NULL;
1006 case 0x000003ca: return g_fIntelNetBurst ? "P4_MSR_ALF_ESCR0" : NULL;
1007 case 0x000003cb: return g_fIntelNetBurst ? "P4_MSR_ALF_ESCR1" : NULL;
1008 case 0x000003cc: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR2" : NULL;
1009 case 0x000003cd: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR3" : NULL;
1010 case 0x000003e0: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR4" : NULL;
1011 case 0x000003e1: return g_fIntelNetBurst ? "P4_MSR_CRU_ESCR5" : NULL;
1012 case 0x000003f0: return g_fIntelNetBurst ? "P4_MSR_TC_PRECISE_EVENT" : NULL;
1013 case 0x000003f1: return "IA32_PEBS_ENABLE";
1014 case 0x000003f2: return g_fIntelNetBurst ? "P4_MSR_PEBS_MATRIX_VERT" : "IA32_PEBS_ENABLE";
1015 case 0x000003f3: return g_fIntelNetBurst ? "P4_UNK_0000_03f3" : NULL;
1016 case 0x000003f4: return g_fIntelNetBurst ? "P4_UNK_0000_03f4" : NULL;
1017 case 0x000003f5: return g_fIntelNetBurst ? "P4_UNK_0000_03f5" : NULL;
1018 case 0x000003f6: return g_fIntelNetBurst ? "P4_UNK_0000_03f6" : "I7_MSR_PEBS_LD_LAT";
1019 case 0x000003f7: return g_fIntelNetBurst ? "P4_UNK_0000_03f7" : "I7_MSR_PEBS_LD_LAT";
1020 case 0x000003f8: return g_fIntelNetBurst ? "P4_UNK_0000_03f8" : "I7_MSR_PKG_C3_RESIDENCY";
1021 case 0x000003f9: return "I7_MSR_PKG_C6_RESIDENCY";
1022 case 0x000003fa: return "I7_MSR_PKG_C7_RESIDENCY";
1023 case 0x000003fc: return "I7_MSR_CORE_C3_RESIDENCY";
1024 case 0x000003fd: return "I7_MSR_CORE_C6_RESIDENCY";
1025 case 0x000003fe: return "I7_MSR_CORE_C7_RESIDENCY";
1026 case 0x00000478: return g_enmMicroarch == kCpumMicroarch_Intel_Core2_Penryn ? "CPUID1_FEATURE_MASK" : NULL;
1027 case 0x00000480: return "IA32_VMX_BASIC";
1028 case 0x00000481: return "IA32_VMX_PINBASED_CTLS";
1029 case 0x00000482: return "IA32_VMX_PROCBASED_CTLS";
1030 case 0x00000483: return "IA32_VMX_EXIT_CTLS";
1031 case 0x00000484: return "IA32_VMX_ENTRY_CTLS";
1032 case 0x00000485: return "IA32_VMX_MISC";
1033 case 0x00000486: return "IA32_VMX_CR0_FIXED0";
1034 case 0x00000487: return "IA32_VMX_CR0_FIXED1";
1035 case 0x00000488: return "IA32_VMX_CR4_FIXED0";
1036 case 0x00000489: return "IA32_VMX_CR4_FIXED1";
1037 case 0x0000048a: return "IA32_VMX_VMCS_ENUM";
1038 case 0x0000048b: return "IA32_VMX_PROCBASED_CTLS2";
1039 case 0x0000048c: return "IA32_VMX_EPT_VPID_CAP";
1040 case 0x0000048d: return "IA32_VMX_TRUE_PINBASED_CTLS";
1041 case 0x0000048e: return "IA32_VMX_TRUE_PROCBASED_CTLS";
1042 case 0x0000048f: return "IA32_VMX_TRUE_EXIT_CTLS";
1043 case 0x00000490: return "IA32_VMX_TRUE_ENTRY_CTLS";
1044 case 0x00000491: return "IA32_VMX_VMFUNC";
1045 case 0x000004c1: return "IA32_A_PMC0";
1046 case 0x000004c2: return "IA32_A_PMC1";
1047 case 0x000004c3: return "IA32_A_PMC2";
1048 case 0x000004c4: return "IA32_A_PMC3";
1049 case 0x000004c5: return "IA32_A_PMC4";
1050 case 0x000004c6: return "IA32_A_PMC5";
1051 case 0x000004c7: return "IA32_A_PMC6";
1052 case 0x000004c8: return "IA32_A_PMC7";
1053 case 0x000004f8: return "C2_UNK_0000_04f8"; /* Core2_Penryn. */
1054 case 0x000004f9: return "C2_UNK_0000_04f9"; /* Core2_Penryn. */
1055 case 0x000004fa: return "C2_UNK_0000_04fa"; /* Core2_Penryn. */
1056 case 0x000004fb: return "C2_UNK_0000_04fb"; /* Core2_Penryn. */
1057 case 0x000004fc: return "C2_UNK_0000_04fc"; /* Core2_Penryn. */
1058 case 0x000004fd: return "C2_UNK_0000_04fd"; /* Core2_Penryn. */
1059 case 0x000004fe: return "C2_UNK_0000_04fe"; /* Core2_Penryn. */
1060 case 0x000004ff: return "C2_UNK_0000_04ff"; /* Core2_Penryn. */
1061 case 0x00000502: return "I7_SB_UNK_0000_0502";
1062 case 0x00000590: return "C2_UNK_0000_0590"; /* Core2_Penryn. */
1063 case 0x00000591: return "C2_UNK_0000_0591"; /* Core2_Penryn. */
1064 case 0x000005a0: return "C2_PECI_CTL"; /* Core2_Penryn. */
1065 case 0x000005a1: return "C2_UNK_0000_05a1"; /* Core2_Penryn. */
1066 case 0x00000600: return "IA32_DS_AREA";
1067 case 0x00000601: return "I7_SB_MSR_VR_CURRENT_CONFIG"; /* SandyBridge, IvyBridge. */
1068 case 0x00000602: return "I7_IB_UNK_0000_0602";
1069 case 0x00000603: return "I7_SB_MSR_VR_MISC_CONFIG"; /* SandyBridge, IvyBridge. */
1070 case 0x00000604: return "I7_IB_UNK_0000_0602";
1071 case 0x00000606: return "I7_SB_MSR_RAPL_POWER_UNIT"; /* SandyBridge, IvyBridge. */
1072 case 0x00000609: return "I7_SB_UNK_0000_0609"; /* SandyBridge (non EP). */
1073 case 0x0000060a: return "I7_SB_MSR_PKGC3_IRTL"; /* SandyBridge, IvyBridge. */
1074 case 0x0000060b: return "I7_SB_MSR_PKGC6_IRTL"; /* SandyBridge, IvyBridge. */
1075 case 0x0000060c: return "I7_SB_MSR_PKGC7_IRTL"; /* SandyBridge, IvyBridge. */
1076 case 0x0000060d: return "I7_SB_MSR_PKG_C2_RESIDENCY"; /* SandyBridge, IvyBridge. */
1077 case 0x00000610: return "I7_SB_MSR_PKG_POWER_LIMIT";
1078 case 0x00000611: return "I7_SB_MSR_PKG_ENERGY_STATUS";
1079 case 0x00000613: return "I7_SB_MSR_PKG_PERF_STATUS";
1080 case 0x00000614: return "I7_SB_MSR_PKG_POWER_INFO";
1081 case 0x00000618: return "I7_SB_MSR_DRAM_POWER_LIMIT";
1082 case 0x00000619: return "I7_SB_MSR_DRAM_ENERGY_STATUS";
1083 case 0x0000061b: return "I7_SB_MSR_DRAM_PERF_STATUS";
1084 case 0x0000061c: return "I7_SB_MSR_DRAM_POWER_INFO";
1085 case 0x00000638: return "I7_SB_MSR_PP0_POWER_LIMIT";
1086 case 0x00000639: return "I7_SB_MSR_PP0_ENERGY_STATUS";
1087 case 0x0000063a: return "I7_SB_MSR_PP0_POLICY";
1088 case 0x0000063b: return "I7_SB_MSR_PP0_PERF_STATUS";
1089 case 0x00000640: return "I7_HW_MSR_PP0_POWER_LIMIT";
1090 case 0x00000641: return "I7_HW_MSR_PP0_ENERGY_STATUS";
1091 case 0x00000642: return "I7_HW_MSR_PP0_POLICY";
1092 case 0x00000648: return "I7_IB_MSR_CONFIG_TDP_NOMINAL";
1093 case 0x00000649: return "I7_IB_MSR_CONFIG_TDP_LEVEL1";
1094 case 0x0000064a: return "I7_IB_MSR_CONFIG_TDP_LEVEL2";
1095 case 0x0000064b: return "I7_IB_MSR_CONFIG_TDP_CONTROL";
1096 case 0x0000064c: return "I7_IB_MSR_TURBO_ACTIVATION_RATIO";
1097 case 0x00000660: return "SILV_CORE_C1_RESIDENCY";
1098 case 0x00000661: return "SILV_UNK_0000_0661";
1099 case 0x00000662: return "SILV_UNK_0000_0662";
1100 case 0x00000663: return "SILV_UNK_0000_0663";
1101 case 0x00000664: return "SILV_UNK_0000_0664";
1102 case 0x00000665: return "SILV_UNK_0000_0665";
1103 case 0x00000666: return "SILV_UNK_0000_0666";
1104 case 0x00000667: return "SILV_UNK_0000_0667";
1105 case 0x00000668: return "SILV_UNK_0000_0668";
1106 case 0x00000669: return "SILV_UNK_0000_0669";
1107 case 0x0000066a: return "SILV_UNK_0000_066a";
1108 case 0x0000066b: return "SILV_UNK_0000_066b";
1109 case 0x0000066c: return "SILV_UNK_0000_066c";
1110 case 0x0000066d: return "SILV_UNK_0000_066d";
1111 case 0x0000066e: return "SILV_UNK_0000_066e";
1112 case 0x0000066f: return "SILV_UNK_0000_066f";
1113 case 0x00000670: return "SILV_UNK_0000_0670";
1114 case 0x00000671: return "SILV_UNK_0000_0671";
1115 case 0x00000672: return "SILV_UNK_0000_0672";
1116 case 0x00000673: return "SILV_UNK_0000_0673";
1117 case 0x00000674: return "SILV_UNK_0000_0674";
1118 case 0x00000675: return "SILV_UNK_0000_0675";
1119 case 0x00000676: return "SILV_UNK_0000_0676";
1120 case 0x00000677: return "SILV_UNK_0000_0677";
1121
1122 case 0x00000680: return "MSR_LASTBRANCH_0_FROM_IP";
1123 case 0x00000681: return "MSR_LASTBRANCH_1_FROM_IP";
1124 case 0x00000682: return "MSR_LASTBRANCH_2_FROM_IP";
1125 case 0x00000683: return "MSR_LASTBRANCH_3_FROM_IP";
1126 case 0x00000684: return "MSR_LASTBRANCH_4_FROM_IP";
1127 case 0x00000685: return "MSR_LASTBRANCH_5_FROM_IP";
1128 case 0x00000686: return "MSR_LASTBRANCH_6_FROM_IP";
1129 case 0x00000687: return "MSR_LASTBRANCH_7_FROM_IP";
1130 case 0x00000688: return "MSR_LASTBRANCH_8_FROM_IP";
1131 case 0x00000689: return "MSR_LASTBRANCH_9_FROM_IP";
1132 case 0x0000068a: return "MSR_LASTBRANCH_10_FROM_IP";
1133 case 0x0000068b: return "MSR_LASTBRANCH_11_FROM_IP";
1134 case 0x0000068c: return "MSR_LASTBRANCH_12_FROM_IP";
1135 case 0x0000068d: return "MSR_LASTBRANCH_13_FROM_IP";
1136 case 0x0000068e: return "MSR_LASTBRANCH_14_FROM_IP";
1137 case 0x0000068f: return "MSR_LASTBRANCH_15_FROM_IP";
1138 case 0x000006c0: return "MSR_LASTBRANCH_0_TO_IP";
1139 case 0x000006c1: return "MSR_LASTBRANCH_1_TO_IP";
1140 case 0x000006c2: return "MSR_LASTBRANCH_2_TO_IP";
1141 case 0x000006c3: return "MSR_LASTBRANCH_3_TO_IP";
1142 case 0x000006c4: return "MSR_LASTBRANCH_4_TO_IP";
1143 case 0x000006c5: return "MSR_LASTBRANCH_5_TO_IP";
1144 case 0x000006c6: return "MSR_LASTBRANCH_6_TO_IP";
1145 case 0x000006c7: return "MSR_LASTBRANCH_7_TO_IP";
1146 case 0x000006c8: return "MSR_LASTBRANCH_8_TO_IP";
1147 case 0x000006c9: return "MSR_LASTBRANCH_9_TO_IP";
1148 case 0x000006ca: return "MSR_LASTBRANCH_10_TO_IP";
1149 case 0x000006cb: return "MSR_LASTBRANCH_11_TO_IP";
1150 case 0x000006cc: return "MSR_LASTBRANCH_12_TO_IP";
1151 case 0x000006cd: return "MSR_LASTBRANCH_13_TO_IP";
1152 case 0x000006ce: return "MSR_LASTBRANCH_14_TO_IP";
1153 case 0x000006cf: return "MSR_LASTBRANCH_15_TO_IP";
1154 case 0x000006e0: return "IA32_TSC_DEADLINE";
1155
1156 case 0x00000768: return "SILV_UNK_0000_0768";
1157 case 0x00000769: return "SILV_UNK_0000_0769";
1158 case 0x0000076a: return "SILV_UNK_0000_076a";
1159 case 0x0000076b: return "SILV_UNK_0000_076b";
1160 case 0x0000076c: return "SILV_UNK_0000_076c";
1161 case 0x0000076d: return "SILV_UNK_0000_076d";
1162 case 0x0000076e: return "SILV_UNK_0000_076e";
1163
1164 case 0x00000c80: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_IvyBridge ? "IA32_DEBUG_INTERFACE" : NULL; /* Mentioned in an intel dataskit called 4th-gen-core-family-desktop-vol-1-datasheet.pdf. */
1165 case 0x00000c81: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_IvyBridge ? "I7_IB_UNK_0000_0c81" : NULL; /* Probably related to IA32_DEBUG_INTERFACE... */
1166 case 0x00000c82: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_IvyBridge ? "I7_IB_UNK_0000_0c82" : NULL; /* Probably related to IA32_DEBUG_INTERFACE... */
1167 case 0x00000c83: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_IvyBridge ? "I7_IB_UNK_0000_0c83" : NULL; /* Probably related to IA32_DEBUG_INTERFACE... */
1168
1169 /* 0x1000..0x1004 seems to have been used by IBM 386 and 486 clones too. */
1170 case 0x00001000: return "P6_DEBUG_REGISTER_0";
1171 case 0x00001001: return "P6_DEBUG_REGISTER_1";
1172 case 0x00001002: return "P6_DEBUG_REGISTER_2";
1173 case 0x00001003: return "P6_DEBUG_REGISTER_3";
1174 case 0x00001004: return "P6_DEBUG_REGISTER_4";
1175 case 0x00001005: return "P6_DEBUG_REGISTER_5";
1176 case 0x00001006: return "P6_DEBUG_REGISTER_6";
1177 case 0x00001007: return "P6_DEBUG_REGISTER_7";
1178 case 0x0000103f: return "P6_UNK_0000_103f"; /* P6_M_Dothan. */
1179 case 0x000010cd: return "P6_UNK_0000_10cd"; /* P6_M_Dothan. */
1180
1181 case 0x00001107: return "VIA_UNK_0000_1107";
1182 case 0x0000110f: return "VIA_UNK_0000_110f";
1183 case 0x00001153: return "VIA_UNK_0000_1153";
1184 case 0x00001200: return "VIA_UNK_0000_1200";
1185 case 0x00001201: return "VIA_UNK_0000_1201";
1186 case 0x00001202: return "VIA_UNK_0000_1202";
1187 case 0x00001203: return "VIA_UNK_0000_1203";
1188 case 0x00001204: return "VIA_UNK_0000_1204";
1189 case 0x00001205: return "VIA_UNK_0000_1205";
1190 case 0x00001206: return "VIA_ALT_VENDOR_EBX";
1191 case 0x00001207: return "VIA_ALT_VENDOR_ECDX";
1192 case 0x00001208: return "VIA_UNK_0000_1208";
1193 case 0x00001209: return "VIA_UNK_0000_1209";
1194 case 0x0000120a: return "VIA_UNK_0000_120a";
1195 case 0x0000120b: return "VIA_UNK_0000_120b";
1196 case 0x0000120c: return "VIA_UNK_0000_120c";
1197 case 0x0000120d: return "VIA_UNK_0000_120d";
1198 case 0x0000120e: return "VIA_UNK_0000_120e";
1199 case 0x0000120f: return "VIA_UNK_0000_120f";
1200 case 0x00001210: return "VIA_UNK_0000_1210";
1201 case 0x00001211: return "VIA_UNK_0000_1211";
1202 case 0x00001212: return "VIA_UNK_0000_1212";
1203 case 0x00001213: return "VIA_UNK_0000_1213";
1204 case 0x00001214: return "VIA_UNK_0000_1214";
1205 case 0x00001220: return "VIA_UNK_0000_1220";
1206 case 0x00001221: return "VIA_UNK_0000_1221";
1207 case 0x00001230: return "VIA_UNK_0000_1230";
1208 case 0x00001231: return "VIA_UNK_0000_1231";
1209 case 0x00001232: return "VIA_UNK_0000_1232";
1210 case 0x00001233: return "VIA_UNK_0000_1233";
1211 case 0x00001234: return "VIA_UNK_0000_1234";
1212 case 0x00001235: return "VIA_UNK_0000_1235";
1213 case 0x00001236: return "VIA_UNK_0000_1236";
1214 case 0x00001237: return "VIA_UNK_0000_1237";
1215 case 0x00001238: return "VIA_UNK_0000_1238";
1216 case 0x00001239: return "VIA_UNK_0000_1239";
1217 case 0x00001240: return "VIA_UNK_0000_1240";
1218 case 0x00001241: return "VIA_UNK_0000_1241";
1219 case 0x00001243: return "VIA_UNK_0000_1243";
1220 case 0x00001245: return "VIA_UNK_0000_1245";
1221 case 0x00001246: return "VIA_UNK_0000_1246";
1222 case 0x00001247: return "VIA_UNK_0000_1247";
1223 case 0x00001248: return "VIA_UNK_0000_1248";
1224 case 0x00001249: return "VIA_UNK_0000_1249";
1225 case 0x0000124a: return "VIA_UNK_0000_124a";
1226
1227 case 0x00001301: return "VIA_UNK_0000_1301";
1228 case 0x00001302: return "VIA_UNK_0000_1302";
1229 case 0x00001303: return "VIA_UNK_0000_1303";
1230 case 0x00001304: return "VIA_UNK_0000_1304";
1231 case 0x00001305: return "VIA_UNK_0000_1305";
1232 case 0x00001306: return "VIA_UNK_0000_1306";
1233 case 0x00001307: return "VIA_UNK_0000_1307";
1234 case 0x00001308: return "VIA_UNK_0000_1308";
1235 case 0x00001309: return "VIA_UNK_0000_1309";
1236 case 0x0000130d: return "VIA_UNK_0000_130d";
1237 case 0x0000130e: return "VIA_UNK_0000_130e";
1238 case 0x00001312: return "VIA_UNK_0000_1312";
1239 case 0x00001315: return "VIA_UNK_0000_1315";
1240 case 0x00001317: return "VIA_UNK_0000_1317";
1241 case 0x00001318: return "VIA_UNK_0000_1318";
1242 case 0x0000131a: return "VIA_UNK_0000_131a";
1243 case 0x0000131b: return "VIA_UNK_0000_131b";
1244 case 0x00001402: return "VIA_UNK_0000_1402";
1245 case 0x00001403: return "VIA_UNK_0000_1403";
1246 case 0x00001404: return "VIA_UNK_0000_1404";
1247 case 0x00001405: return "VIA_UNK_0000_1405";
1248 case 0x00001406: return "VIA_UNK_0000_1406";
1249 case 0x00001407: return "VIA_UNK_0000_1407";
1250 case 0x00001410: return "VIA_UNK_0000_1410";
1251 case 0x00001411: return "VIA_UNK_0000_1411";
1252 case 0x00001412: return "VIA_UNK_0000_1412";
1253 case 0x00001413: return "VIA_UNK_0000_1413";
1254 case 0x00001414: return "VIA_UNK_0000_1414";
1255 case 0x00001415: return "VIA_UNK_0000_1415";
1256 case 0x00001416: return "VIA_UNK_0000_1416";
1257 case 0x00001417: return "VIA_UNK_0000_1417";
1258 case 0x00001418: return "VIA_UNK_0000_1418";
1259 case 0x00001419: return "VIA_UNK_0000_1419";
1260 case 0x0000141a: return "VIA_UNK_0000_141a";
1261 case 0x0000141b: return "VIA_UNK_0000_141b";
1262 case 0x0000141c: return "VIA_UNK_0000_141c";
1263 case 0x0000141d: return "VIA_UNK_0000_141d";
1264 case 0x0000141e: return "VIA_UNK_0000_141e";
1265 case 0x0000141f: return "VIA_UNK_0000_141f";
1266 case 0x00001420: return "VIA_UNK_0000_1420";
1267 case 0x00001421: return "VIA_UNK_0000_1421";
1268 case 0x00001422: return "VIA_UNK_0000_1422";
1269 case 0x00001423: return "VIA_UNK_0000_1423";
1270 case 0x00001424: return "VIA_UNK_0000_1424";
1271 case 0x00001425: return "VIA_UNK_0000_1425";
1272 case 0x00001426: return "VIA_UNK_0000_1426";
1273 case 0x00001427: return "VIA_UNK_0000_1427";
1274 case 0x00001428: return "VIA_UNK_0000_1428";
1275 case 0x00001429: return "VIA_UNK_0000_1429";
1276 case 0x0000142a: return "VIA_UNK_0000_142a";
1277 case 0x0000142b: return "VIA_UNK_0000_142b";
1278 case 0x0000142c: return "VIA_UNK_0000_142c";
1279 case 0x0000142d: return "VIA_UNK_0000_142d";
1280 case 0x0000142e: return "VIA_UNK_0000_142e";
1281 case 0x0000142f: return "VIA_UNK_0000_142f";
1282 case 0x00001434: return "VIA_UNK_0000_1434";
1283 case 0x00001435: return "VIA_UNK_0000_1435";
1284 case 0x00001436: return "VIA_UNK_0000_1436";
1285 case 0x00001437: return "VIA_UNK_0000_1437";
1286 case 0x00001438: return "VIA_UNK_0000_1438";
1287 case 0x0000143a: return "VIA_UNK_0000_143a";
1288 case 0x0000143c: return "VIA_UNK_0000_143c";
1289 case 0x0000143d: return "VIA_UNK_0000_143d";
1290 case 0x00001440: return "VIA_UNK_0000_1440";
1291 case 0x00001441: return "VIA_UNK_0000_1441";
1292 case 0x00001442: return "VIA_UNK_0000_1442";
1293 case 0x00001449: return "VIA_UNK_0000_1449";
1294 case 0x00001450: return "VIA_UNK_0000_1450";
1295 case 0x00001451: return "VIA_UNK_0000_1451";
1296 case 0x00001452: return "VIA_UNK_0000_1452";
1297 case 0x00001453: return "VIA_UNK_0000_1453";
1298 case 0x00001460: return "VIA_UNK_0000_1460";
1299 case 0x00001461: return "VIA_UNK_0000_1461";
1300 case 0x00001462: return "VIA_UNK_0000_1462";
1301 case 0x00001463: return "VIA_UNK_0000_1463";
1302 case 0x00001465: return "VIA_UNK_0000_1465";
1303 case 0x00001466: return "VIA_UNK_0000_1466";
1304 case 0x00001470: return "VIA_UNK_0000_1470";
1305 case 0x00001471: return "VIA_UNK_0000_1471";
1306 case 0x00001480: return "VIA_UNK_0000_1480";
1307 case 0x00001481: return "VIA_UNK_0000_1481";
1308 case 0x00001482: return "VIA_UNK_0000_1482";
1309 case 0x00001483: return "VIA_UNK_0000_1483";
1310 case 0x00001484: return "VIA_UNK_0000_1484";
1311 case 0x00001485: return "VIA_UNK_0000_1485";
1312 case 0x00001486: return "VIA_UNK_0000_1486";
1313 case 0x00001490: return "VIA_UNK_0000_1490";
1314 case 0x00001491: return "VIA_UNK_0000_1491";
1315 case 0x00001492: return "VIA_UNK_0000_1492";
1316 case 0x00001493: return "VIA_UNK_0000_1493";
1317 case 0x00001494: return "VIA_UNK_0000_1494";
1318 case 0x00001495: return "VIA_UNK_0000_1495";
1319 case 0x00001496: return "VIA_UNK_0000_1496";
1320 case 0x00001497: return "VIA_UNK_0000_1497";
1321 case 0x00001498: return "VIA_UNK_0000_1498";
1322 case 0x00001499: return "VIA_UNK_0000_1499";
1323 case 0x0000149a: return "VIA_UNK_0000_149a";
1324 case 0x0000149b: return "VIA_UNK_0000_149b";
1325 case 0x0000149c: return "VIA_UNK_0000_149c";
1326 case 0x0000149f: return "VIA_UNK_0000_149f";
1327 case 0x00001523: return "VIA_UNK_0000_1523";
1328
1329 case 0x00002000: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_CR0" : NULL;
1330 case 0x00002002: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_CR2" : NULL;
1331 case 0x00002003: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_CR3" : NULL;
1332 case 0x00002004: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_CR4" : NULL;
1333 case 0x0000203f: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_UNK_0000_203f" /* P6_M_Dothan. */ : NULL;
1334 case 0x000020cd: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_UNK_0000_20cd" /* P6_M_Dothan. */ : NULL;
1335 case 0x0000303f: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_UNK_0000_303f" /* P6_M_Dothan. */ : NULL;
1336 case 0x000030cd: return g_enmVendor == CPUMCPUVENDOR_INTEL ? "P6_UNK_0000_30cd" /* P6_M_Dothan. */ : NULL;
1337
1338 case 0x0000317a: return "VIA_UNK_0000_317a";
1339 case 0x0000317b: return "VIA_UNK_0000_317b";
1340 case 0x0000317d: return "VIA_UNK_0000_317d";
1341 case 0x0000317e: return "VIA_UNK_0000_317e";
1342 case 0x0000317f: return "VIA_UNK_0000_317f";
1343 case 0x80000198: return "VIA_UNK_8000_0198";
1344
1345 case 0xc0000080: return "AMD64_EFER";
1346 case 0xc0000081: return "AMD64_STAR";
1347 case 0xc0000082: return "AMD64_STAR64";
1348 case 0xc0000083: return "AMD64_STARCOMPAT";
1349 case 0xc0000084: return "AMD64_SYSCALL_FLAG_MASK";
1350 case 0xc0000100: return "AMD64_FS_BASE";
1351 case 0xc0000101: return "AMD64_GS_BASE";
1352 case 0xc0000102: return "AMD64_KERNEL_GS_BASE";
1353 case 0xc0000103: return "AMD64_TSC_AUX";
1354 case 0xc0000104: return "AMD_15H_TSC_RATE";
1355 case 0xc0000105: return "AMD_15H_LWP_CFG"; /* Only Family 15h? */
1356 case 0xc0000106: return "AMD_15H_LWP_CBADDR"; /* Only Family 15h? */
1357 case 0xc0000408: return "AMD_10H_MC4_MISC1";
1358 case 0xc0000409: return "AMD_10H_MC4_MISC2";
1359 case 0xc000040a: return "AMD_10H_MC4_MISC3";
1360 case 0xc000040b: return "AMD_10H_MC4_MISC4";
1361 case 0xc000040c: return "AMD_10H_MC4_MISC5";
1362 case 0xc000040d: return "AMD_10H_MC4_MISC6";
1363 case 0xc000040e: return "AMD_10H_MC4_MISC7";
1364 case 0xc000040f: return "AMD_10H_MC4_MISC8";
1365 case 0xc0010000: return "AMD_K8_PERF_CTL_0";
1366 case 0xc0010001: return "AMD_K8_PERF_CTL_1";
1367 case 0xc0010002: return "AMD_K8_PERF_CTL_2";
1368 case 0xc0010003: return "AMD_K8_PERF_CTL_3";
1369 case 0xc0010004: return "AMD_K8_PERF_CTR_0";
1370 case 0xc0010005: return "AMD_K8_PERF_CTR_1";
1371 case 0xc0010006: return "AMD_K8_PERF_CTR_2";
1372 case 0xc0010007: return "AMD_K8_PERF_CTR_3";
1373 case 0xc0010010: return "AMD_K8_SYS_CFG";
1374 case 0xc0010015: return "AMD_K8_HW_CFG";
1375 case 0xc0010016: return "AMD_K8_IORR_BASE_0";
1376 case 0xc0010017: return "AMD_K8_IORR_MASK_0";
1377 case 0xc0010018: return "AMD_K8_IORR_BASE_1";
1378 case 0xc0010019: return "AMD_K8_IORR_MASK_1";
1379 case 0xc001001a: return "AMD_K8_TOP_MEM";
1380 case 0xc001001d: return "AMD_K8_TOP_MEM2";
1381 case 0xc001001e: return "AMD_K8_MANID";
1382 case 0xc001001f: return "AMD_K8_NB_CFG1";
1383 case 0xc0010020: return "AMD_K8_PATCH_LOADER";
1384 case 0xc0010021: return "AMD_K8_UNK_c001_0021";
1385 case 0xc0010022: return "AMD_K8_MC_XCPT_REDIR";
1386 case 0xc0010028: return "AMD_K8_UNK_c001_0028";
1387 case 0xc0010029: return "AMD_K8_UNK_c001_0029";
1388 case 0xc001002a: return "AMD_K8_UNK_c001_002a";
1389 case 0xc001002b: return "AMD_K8_UNK_c001_002b";
1390 case 0xc001002c: return "AMD_K8_UNK_c001_002c";
1391 case 0xc001002d: return "AMD_K8_UNK_c001_002d";
1392 case 0xc0010030: return "AMD_K8_CPU_NAME_0";
1393 case 0xc0010031: return "AMD_K8_CPU_NAME_1";
1394 case 0xc0010032: return "AMD_K8_CPU_NAME_2";
1395 case 0xc0010033: return "AMD_K8_CPU_NAME_3";
1396 case 0xc0010034: return "AMD_K8_CPU_NAME_4";
1397 case 0xc0010035: return "AMD_K8_CPU_NAME_5";
1398 case 0xc001003e: return "AMD_K8_HTC";
1399 case 0xc001003f: return "AMD_K8_STC";
1400 case 0xc0010041: return "AMD_K8_FIDVID_CTL";
1401 case 0xc0010042: return "AMD_K8_FIDVID_STATUS";
1402 case 0xc0010043: return "AMD_K8_THERMTRIP_STATUS"; /* BDKG says it was removed in K8 revision C.*/
1403 case 0xc0010044: return "AMD_K8_MC_CTL_MASK_0";
1404 case 0xc0010045: return "AMD_K8_MC_CTL_MASK_1";
1405 case 0xc0010046: return "AMD_K8_MC_CTL_MASK_2";
1406 case 0xc0010047: return "AMD_K8_MC_CTL_MASK_3";
1407 case 0xc0010048: return "AMD_K8_MC_CTL_MASK_4";
1408 case 0xc0010049: return "AMD_K8_MC_CTL_MASK_5";
1409 case 0xc001004a: return "AMD_K8_MC_CTL_MASK_6";
1410 //case 0xc001004b: return "AMD_K8_MC_CTL_MASK_7";
1411 case 0xc0010050: return "AMD_K8_SMI_ON_IO_TRAP_0";
1412 case 0xc0010051: return "AMD_K8_SMI_ON_IO_TRAP_1";
1413 case 0xc0010052: return "AMD_K8_SMI_ON_IO_TRAP_2";
1414 case 0xc0010053: return "AMD_K8_SMI_ON_IO_TRAP_3";
1415 case 0xc0010054: return "AMD_K8_SMI_ON_IO_TRAP_CTL_STS";
1416 case 0xc0010055: return "AMD_K8_INT_PENDING_MSG";
1417 case 0xc0010056: return "AMD_K8_SMI_TRIGGER_IO_CYCLE";
1418 case 0xc0010057: return "AMD_10H_UNK_c001_0057";
1419 case 0xc0010058: return "AMD_10H_MMIO_CFG_BASE_ADDR";
1420 case 0xc0010059: return "AMD_10H_TRAP_CTL?"; /* Undocumented, only one google hit. */
1421 case 0xc001005a: return "AMD_10H_UNK_c001_005a";
1422 case 0xc001005b: return "AMD_10H_UNK_c001_005b";
1423 case 0xc001005c: return "AMD_10H_UNK_c001_005c";
1424 case 0xc001005d: return "AMD_10H_UNK_c001_005d";
1425 case 0xc0010060: return "AMD_K8_BIST_RESULT"; /* BDKG says it as introduced with revision F. */
1426 case 0xc0010061: return "AMD_10H_P_ST_CUR_LIM";
1427 case 0xc0010062: return "AMD_10H_P_ST_CTL";
1428 case 0xc0010063: return "AMD_10H_P_ST_STS";
1429 case 0xc0010064: return "AMD_10H_P_ST_0";
1430 case 0xc0010065: return "AMD_10H_P_ST_1";
1431 case 0xc0010066: return "AMD_10H_P_ST_2";
1432 case 0xc0010067: return "AMD_10H_P_ST_3";
1433 case 0xc0010068: return "AMD_10H_P_ST_4";
1434 case 0xc0010069: return "AMD_10H_P_ST_5";
1435 case 0xc001006a: return "AMD_10H_P_ST_6";
1436 case 0xc001006b: return "AMD_10H_P_ST_7";
1437 case 0xc0010070: return "AMD_10H_COFVID_CTL";
1438 case 0xc0010071: return "AMD_10H_COFVID_STS";
1439 case 0xc0010073: return "AMD_10H_C_ST_IO_BASE_ADDR";
1440 case 0xc0010074: return "AMD_10H_CPU_WD_TMR_CFG";
1441 // case 0xc0010075: return "AMD_15H_APML_TDP_LIM";
1442 // case 0xc0010077: return "AMD_15H_CPU_PWR_IN_TDP";
1443 // case 0xc0010078: return "AMD_15H_PWR_AVG_PERIOD";
1444 // case 0xc0010079: return "AMD_15H_DRAM_CTR_CMD_THR";
1445 // case 0xc0010080: return "AMD_16H_FSFM_ACT_CNT_0";
1446 // case 0xc0010081: return "AMD_16H_FSFM_REF_CNT_0";
1447 case 0xc0010111: return "AMD_K8_SMM_BASE";
1448 case 0xc0010112: return "AMD_K8_SMM_ADDR";
1449 case 0xc0010113: return "AMD_K8_SMM_MASK";
1450 case 0xc0010114: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AMD_K8_VM_CR" : "AMD_K8_UNK_c001_0114";
1451 case 0xc0010115: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AMD_K8_IGNNE" : "AMD_K8_UNK_c001_0115";
1452 case 0xc0010116: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AMD_K8_SMM_CTL" : "AMD_K8_UNK_c001_0116";
1453 case 0xc0010117: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AMD_K8_VM_HSAVE_PA" : "AMD_K8_UNK_c001_0117";
1454 case 0xc0010118: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AMD_10H_VM_LOCK_KEY" : "AMD_K8_UNK_c001_0118";
1455 case 0xc0010119: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AMD_10H_SSM_LOCK_KEY" : "AMD_K8_UNK_c001_0119";
1456 case 0xc001011a: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AMD_10H_LOCAL_SMI_STS" : "AMD_K8_UNK_c001_011a";
1457 case 0xc001011b: return "AMD_K8_UNK_c001_011b";
1458 case 0xc001011c: return "AMD_K8_UNK_c001_011c";
1459 case 0xc0010140: return "AMD_10H_OSVW_ID_LEN";
1460 case 0xc0010141: return "AMD_10H_OSVW_STS";
1461 case 0xc0010200: return "AMD_K8_PERF_CTL_0";
1462 case 0xc0010202: return "AMD_K8_PERF_CTL_1";
1463 case 0xc0010204: return "AMD_K8_PERF_CTL_2";
1464 case 0xc0010206: return "AMD_K8_PERF_CTL_3";
1465 case 0xc0010208: return "AMD_K8_PERF_CTL_4";
1466 case 0xc001020a: return "AMD_K8_PERF_CTL_5";
1467 //case 0xc001020c: return "AMD_K8_PERF_CTL_6";
1468 //case 0xc001020e: return "AMD_K8_PERF_CTL_7";
1469 case 0xc0010201: return "AMD_K8_PERF_CTR_0";
1470 case 0xc0010203: return "AMD_K8_PERF_CTR_1";
1471 case 0xc0010205: return "AMD_K8_PERF_CTR_2";
1472 case 0xc0010207: return "AMD_K8_PERF_CTR_3";
1473 case 0xc0010209: return "AMD_K8_PERF_CTR_4";
1474 case 0xc001020b: return "AMD_K8_PERF_CTR_5";
1475 //case 0xc001020d: return "AMD_K8_PERF_CTR_6";
1476 //case 0xc001020f: return "AMD_K8_PERF_CTR_7";
1477 case 0xc0010230: return "AMD_16H_L2I_PERF_CTL_0";
1478 case 0xc0010232: return "AMD_16H_L2I_PERF_CTL_1";
1479 case 0xc0010234: return "AMD_16H_L2I_PERF_CTL_2";
1480 case 0xc0010236: return "AMD_16H_L2I_PERF_CTL_3";
1481 //case 0xc0010238: return "AMD_16H_L2I_PERF_CTL_4";
1482 //case 0xc001023a: return "AMD_16H_L2I_PERF_CTL_5";
1483 //case 0xc001030c: return "AMD_16H_L2I_PERF_CTL_6";
1484 //case 0xc001023e: return "AMD_16H_L2I_PERF_CTL_7";
1485 case 0xc0010231: return "AMD_16H_L2I_PERF_CTR_0";
1486 case 0xc0010233: return "AMD_16H_L2I_PERF_CTR_1";
1487 case 0xc0010235: return "AMD_16H_L2I_PERF_CTR_2";
1488 case 0xc0010237: return "AMD_16H_L2I_PERF_CTR_3";
1489 //case 0xc0010239: return "AMD_16H_L2I_PERF_CTR_4";
1490 //case 0xc001023b: return "AMD_16H_L2I_PERF_CTR_5";
1491 //case 0xc001023d: return "AMD_16H_L2I_PERF_CTR_6";
1492 //case 0xc001023f: return "AMD_16H_L2I_PERF_CTR_7";
1493 case 0xc0010240: return "AMD_15H_NB_PERF_CTL_0";
1494 case 0xc0010242: return "AMD_15H_NB_PERF_CTL_1";
1495 case 0xc0010244: return "AMD_15H_NB_PERF_CTL_2";
1496 case 0xc0010246: return "AMD_15H_NB_PERF_CTL_3";
1497 //case 0xc0010248: return "AMD_15H_NB_PERF_CTL_4";
1498 //case 0xc001024a: return "AMD_15H_NB_PERF_CTL_5";
1499 //case 0xc001024c: return "AMD_15H_NB_PERF_CTL_6";
1500 //case 0xc001024e: return "AMD_15H_NB_PERF_CTL_7";
1501 case 0xc0010241: return "AMD_15H_NB_PERF_CTR_0";
1502 case 0xc0010243: return "AMD_15H_NB_PERF_CTR_1";
1503 case 0xc0010245: return "AMD_15H_NB_PERF_CTR_2";
1504 case 0xc0010247: return "AMD_15H_NB_PERF_CTR_3";
1505 //case 0xc0010249: return "AMD_15H_NB_PERF_CTR_4";
1506 //case 0xc001024b: return "AMD_15H_NB_PERF_CTR_5";
1507 //case 0xc001024d: return "AMD_15H_NB_PERF_CTR_6";
1508 //case 0xc001024f: return "AMD_15H_NB_PERF_CTR_7";
1509 case 0xc0011000: return "AMD_K7_MCODE_CTL";
1510 case 0xc0011001: return "AMD_K7_APIC_CLUSTER_ID"; /* Mentioned in BKDG (r3.00) for fam16h when describing EBL_CR_POWERON. */
1511 case 0xc0011002: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_CPUID_CTL_STD07" : NULL;
1512 case 0xc0011003: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_CPUID_CTL_STD06" : NULL;
1513 case 0xc0011004: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_CPUID_CTL_STD01" : NULL;
1514 case 0xc0011005: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_CPUID_CTL_EXT01" : NULL;
1515 case 0xc0011006: return "AMD_K7_DEBUG_STS?";
1516 case 0xc0011007: return "AMD_K7_BH_TRACE_BASE?";
1517 case 0xc0011008: return "AMD_K7_BH_TRACE_PTR?";
1518 case 0xc0011009: return "AMD_K7_BH_TRACE_LIM?";
1519 case 0xc001100a: return "AMD_K7_HDT_CFG?";
1520 case 0xc001100b: return "AMD_K7_FAST_FLUSH_COUNT?";
1521 case 0xc001100c: return "AMD_K7_NODE_ID";
1522 case 0xc001100d: return "AMD_K8_LOGICAL_CPUS_NUM?";
1523 case 0xc001100e: return "AMD_K8_WRMSR_BP?";
1524 case 0xc001100f: return "AMD_K8_WRMSR_BP_MASK?";
1525 case 0xc0011010: return "AMD_K8_BH_TRACE_CTL?";
1526 case 0xc0011011: return "AMD_K8_BH_TRACE_USRD?";
1527 case 0xc0011012: return "AMD_K7_UNK_c001_1012";
1528 case 0xc0011013: return "AMD_K7_UNK_c001_1013";
1529 case 0xc0011014: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_XCPT_BP_RIP?" : "AMD_K7_MOBIL_DEBUG?";
1530 case 0xc0011015: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_XCPT_BP_RIP_MASK?" : NULL;
1531 case 0xc0011016: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_COND_HDT_VAL?" : NULL;
1532 case 0xc0011017: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_COND_HDT_VAL_MASK?" : NULL;
1533 case 0xc0011018: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_XCPT_BP_CTL?" : NULL;
1534 case 0xc0011019: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AMD_16H_DR1_ADDR_MASK" : NULL;
1535 case 0xc001101a: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AMD_16H_DR2_ADDR_MASK" : NULL;
1536 case 0xc001101b: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AMD_16H_DR3_ADDR_MASK" : NULL;
1537 case 0xc001101d: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_NB_BIST?" : NULL;
1538 case 0xc001101e: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_THERMTRIP_2?" : NULL;
1539 case 0xc001101f: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AMD_K8_NB_CFG?" : NULL;
1540 case 0xc0011020: return "AMD_K7_LS_CFG";
1541 case 0xc0011021: return "AMD_K7_IC_CFG";
1542 case 0xc0011022: return "AMD_K7_DC_CFG";
1543 case 0xc0011023: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AMD_15H_CU_CFG" : "AMD_K7_BU_CFG";
1544 case 0xc0011024: return "AMD_K7_DEBUG_CTL_2?";
1545 case 0xc0011025: return "AMD_K7_DR0_DATA_MATCH?";
1546 case 0xc0011026: return "AMD_K7_DR0_DATA_MATCH?";
1547 case 0xc0011027: return "AMD_K7_DR0_ADDR_MASK";
1548 case 0xc0011028: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_First ? "AMD_15H_FP_CFG"
1549 : CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch) ? "AMD_10H_UNK_c001_1028"
1550 : NULL;
1551 case 0xc0011029: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_First ? "AMD_15H_DC_CFG"
1552 : CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch) ? "AMD_10H_UNK_c001_1029"
1553 : NULL;
1554 case 0xc001102a: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AMD_15H_CU_CFG2"
1555 : CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch) || g_enmMicroarch > kCpumMicroarch_AMD_15h_End
1556 ? "AMD_10H_BU_CFG2" /* 10h & 16h */ : NULL;
1557 case 0xc001102b: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AMD_15H_CU_CFG3" : NULL;
1558 case 0xc001102c: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AMD_15H_EX_CFG" : NULL;
1559 case 0xc001102d: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AMD_15H_LS_CFG2" : NULL;
1560 case 0xc0011030: return "AMD_10H_IBS_FETCH_CTL";
1561 case 0xc0011031: return "AMD_10H_IBS_FETCH_LIN_ADDR";
1562 case 0xc0011032: return "AMD_10H_IBS_FETCH_PHYS_ADDR";
1563 case 0xc0011033: return "AMD_10H_IBS_OP_EXEC_CTL";
1564 case 0xc0011034: return "AMD_10H_IBS_OP_RIP";
1565 case 0xc0011035: return "AMD_10H_IBS_OP_DATA";
1566 case 0xc0011036: return "AMD_10H_IBS_OP_DATA2";
1567 case 0xc0011037: return "AMD_10H_IBS_OP_DATA3";
1568 case 0xc0011038: return "AMD_10H_IBS_DC_LIN_ADDR";
1569 case 0xc0011039: return "AMD_10H_IBS_DC_PHYS_ADDR";
1570 case 0xc001103a: return "AMD_10H_IBS_CTL";
1571 case 0xc001103b: return "AMD_14H_IBS_BR_TARGET";
1572
1573 case 0xc0011040: return "AMD_15H_UNK_c001_1040";
1574 case 0xc0011041: return "AMD_15H_UNK_c001_1041";
1575 case 0xc0011042: return "AMD_15H_UNK_c001_1042";
1576 case 0xc0011043: return "AMD_15H_UNK_c001_1043";
1577 case 0xc0011044: return "AMD_15H_UNK_c001_1044";
1578 case 0xc0011045: return "AMD_15H_UNK_c001_1045";
1579 case 0xc0011046: return "AMD_15H_UNK_c001_1046";
1580 case 0xc0011047: return "AMD_15H_UNK_c001_1047";
1581 case 0xc0011048: return "AMD_15H_UNK_c001_1048";
1582 case 0xc0011049: return "AMD_15H_UNK_c001_1049";
1583 case 0xc001104a: return "AMD_15H_UNK_c001_104a";
1584 case 0xc001104b: return "AMD_15H_UNK_c001_104b";
1585 case 0xc001104c: return "AMD_15H_UNK_c001_104c";
1586 case 0xc001104d: return "AMD_15H_UNK_c001_104d";
1587 case 0xc001104e: return "AMD_15H_UNK_c001_104e";
1588 case 0xc001104f: return "AMD_15H_UNK_c001_104f";
1589 case 0xc0011050: return "AMD_15H_UNK_c001_1050";
1590 case 0xc0011051: return "AMD_15H_UNK_c001_1051";
1591 case 0xc0011052: return "AMD_15H_UNK_c001_1052";
1592 case 0xc0011053: return "AMD_15H_UNK_c001_1053";
1593 case 0xc0011054: return "AMD_15H_UNK_c001_1054";
1594 case 0xc0011055: return "AMD_15H_UNK_c001_1055";
1595 case 0xc0011056: return "AMD_15H_UNK_c001_1056";
1596 case 0xc0011057: return "AMD_15H_UNK_c001_1057";
1597 case 0xc0011058: return "AMD_15H_UNK_c001_1058";
1598 case 0xc0011059: return "AMD_15H_UNK_c001_1059";
1599 case 0xc001105a: return "AMD_15H_UNK_c001_105a";
1600 case 0xc001105b: return "AMD_15H_UNK_c001_105b";
1601 case 0xc001105c: return "AMD_15H_UNK_c001_105c";
1602 case 0xc001105d: return "AMD_15H_UNK_c001_105d";
1603 case 0xc001105e: return "AMD_15H_UNK_c001_105e";
1604 case 0xc001105f: return "AMD_15H_UNK_c001_105f";
1605 case 0xc0011060: return "AMD_15H_UNK_c001_1060";
1606 case 0xc0011061: return "AMD_15H_UNK_c001_1061";
1607 case 0xc0011062: return "AMD_15H_UNK_c001_1062";
1608 case 0xc0011063: return "AMD_15H_UNK_c001_1063";
1609 case 0xc0011064: return "AMD_15H_UNK_c001_1064";
1610 case 0xc0011065: return "AMD_15H_UNK_c001_1065";
1611 case 0xc0011066: return "AMD_15H_UNK_c001_1066";
1612 case 0xc0011067: return "AMD_15H_UNK_c001_1067";
1613 case 0xc0011068: return "AMD_15H_UNK_c001_1068";
1614 case 0xc0011069: return "AMD_15H_UNK_c001_1069";
1615 case 0xc001106a: return "AMD_15H_UNK_c001_106a";
1616 case 0xc001106b: return "AMD_15H_UNK_c001_106b";
1617 case 0xc001106c: return "AMD_15H_UNK_c001_106c";
1618 case 0xc001106d: return "AMD_15H_UNK_c001_106d";
1619 case 0xc001106e: return "AMD_15H_UNK_c001_106e";
1620 case 0xc001106f: return "AMD_15H_UNK_c001_106f";
1621 case 0xc0011070: return "AMD_15H_UNK_c001_1070"; /* coreboot defines this, but with a numerical name. */
1622 case 0xc0011071: return "AMD_15H_UNK_c001_1071";
1623 case 0xc0011072: return "AMD_15H_UNK_c001_1072";
1624 case 0xc0011073: return "AMD_15H_UNK_c001_1073";
1625 case 0xc0011080: return "AMD_15H_UNK_c001_1080";
1626 }
1627
1628 /*
1629 * Uncore stuff on Sandy. Putting it here to avoid ugly microarch checks for each register.
1630 * Note! These are found on model 42 (2a) but not 45 (2d), the latter is the EP variant.
1631 */
1632 if (g_enmMicroarch == kCpumMicroarch_Intel_Core7_SandyBridge)
1633 switch (uMsr)
1634 {
1635 case 0x00000700: return "MSR_UNC_CBO_0_PERFEVTSEL0";
1636 case 0x00000701: return "MSR_UNC_CBO_0_PERFEVTSEL1";
1637 case 0x00000702: return "MSR_UNC_CBO_0_PERFEVTSEL2?";
1638 case 0x00000703: return "MSR_UNC_CBO_0_PERFEVTSEL3?";
1639 case 0x00000704: return "MSR_UNC_CBO_0_UNK_4";
1640 case 0x00000705: return "MSR_UNC_CBO_0_UNK_5";
1641 case 0x00000706: return "MSR_UNC_CBO_0_PER_CTR0";
1642 case 0x00000707: return "MSR_UNC_CBO_0_PER_CTR1";
1643 case 0x00000708: return "MSR_UNC_CBO_0_PER_CTR2?";
1644 case 0x00000709: return "MSR_UNC_CBO_0_PER_CTR3?";
1645 case 0x00000710: return "MSR_UNC_CBO_1_PERFEVTSEL0";
1646 case 0x00000711: return "MSR_UNC_CBO_1_PERFEVTSEL1";
1647 case 0x00000712: return "MSR_UNC_CBO_1_PERFEVTSEL2?";
1648 case 0x00000713: return "MSR_UNC_CBO_1_PERFEVTSEL3?";
1649 case 0x00000714: return "MSR_UNC_CBO_1_UNK_4";
1650 case 0x00000715: return "MSR_UNC_CBO_1_UNK_5";
1651 case 0x00000716: return "MSR_UNC_CBO_1_PER_CTR0";
1652 case 0x00000717: return "MSR_UNC_CBO_1_PER_CTR1";
1653 case 0x00000718: return "MSR_UNC_CBO_1_PER_CTR2?";
1654 case 0x00000719: return "MSR_UNC_CBO_1_PER_CTR3?";
1655 case 0x00000720: return "MSR_UNC_CBO_2_PERFEVTSEL0";
1656 case 0x00000721: return "MSR_UNC_CBO_2_PERFEVTSEL1";
1657 case 0x00000722: return "MSR_UNC_CBO_2_PERFEVTSEL2?";
1658 case 0x00000723: return "MSR_UNC_CBO_2_PERFEVTSEL3?";
1659 case 0x00000724: return "MSR_UNC_CBO_2_UNK_4";
1660 case 0x00000725: return "MSR_UNC_CBO_2_UNK_5";
1661 case 0x00000726: return "MSR_UNC_CBO_2_PER_CTR0";
1662 case 0x00000727: return "MSR_UNC_CBO_2_PER_CTR1";
1663 case 0x00000728: return "MSR_UNC_CBO_2_PER_CTR2?";
1664 case 0x00000729: return "MSR_UNC_CBO_2_PER_CTR3?";
1665 case 0x00000730: return "MSR_UNC_CBO_3_PERFEVTSEL0";
1666 case 0x00000731: return "MSR_UNC_CBO_3_PERFEVTSEL1";
1667 case 0x00000732: return "MSR_UNC_CBO_3_PERFEVTSEL2?";
1668 case 0x00000733: return "MSR_UNC_CBO_3_PERFEVTSEL3?";
1669 case 0x00000734: return "MSR_UNC_CBO_3_UNK_4";
1670 case 0x00000735: return "MSR_UNC_CBO_3_UNK_5";
1671 case 0x00000736: return "MSR_UNC_CBO_3_PER_CTR0";
1672 case 0x00000737: return "MSR_UNC_CBO_3_PER_CTR1";
1673 case 0x00000738: return "MSR_UNC_CBO_3_PER_CTR2?";
1674 case 0x00000739: return "MSR_UNC_CBO_3_PER_CTR3?";
1675 case 0x00000740: return "MSR_UNC_CBO_4_PERFEVTSEL0?";
1676 case 0x00000741: return "MSR_UNC_CBO_4_PERFEVTSEL1?";
1677 case 0x00000742: return "MSR_UNC_CBO_4_PERFEVTSEL2?";
1678 case 0x00000743: return "MSR_UNC_CBO_4_PERFEVTSEL3?";
1679 case 0x00000744: return "MSR_UNC_CBO_4_UNK_4";
1680 case 0x00000745: return "MSR_UNC_CBO_4_UNK_5";
1681 case 0x00000746: return "MSR_UNC_CBO_4_PER_CTR0?";
1682 case 0x00000747: return "MSR_UNC_CBO_4_PER_CTR1?";
1683 case 0x00000748: return "MSR_UNC_CBO_4_PER_CTR2?";
1684 case 0x00000749: return "MSR_UNC_CBO_4_PER_CTR3?";
1685
1686 }
1687
1688 /*
1689 * Bunch of unknown sandy bridge registers. They might seem like the
1690 * nehalem based xeon stuff, but the layout doesn't match. I bet it's the
1691 * same kind of registes though (i.e. uncore (UNC)).
1692 *
1693 * Kudos to Intel for keeping these a secret! Many thanks guys!!
1694 */
1695 if (g_enmMicroarch == kCpumMicroarch_Intel_Core7_SandyBridge)
1696 switch (uMsr)
1697 {
1698 case 0x00000a00: return "I7_SB_UNK_0000_0a00"; case 0x00000a01: return "I7_SB_UNK_0000_0a01";
1699 case 0x00000a02: return "I7_SB_UNK_0000_0a02";
1700 case 0x00000c00: return "I7_SB_UNK_0000_0c00"; case 0x00000c01: return "I7_SB_UNK_0000_0c01";
1701 case 0x00000c06: return "I7_SB_UNK_0000_0c06"; case 0x00000c08: return "I7_SB_UNK_0000_0c08";
1702 case 0x00000c09: return "I7_SB_UNK_0000_0c09"; case 0x00000c10: return "I7_SB_UNK_0000_0c10";
1703 case 0x00000c11: return "I7_SB_UNK_0000_0c11"; case 0x00000c14: return "I7_SB_UNK_0000_0c14";
1704 case 0x00000c15: return "I7_SB_UNK_0000_0c15"; case 0x00000c16: return "I7_SB_UNK_0000_0c16";
1705 case 0x00000c17: return "I7_SB_UNK_0000_0c17"; case 0x00000c24: return "I7_SB_UNK_0000_0c24";
1706 case 0x00000c30: return "I7_SB_UNK_0000_0c30"; case 0x00000c31: return "I7_SB_UNK_0000_0c31";
1707 case 0x00000c32: return "I7_SB_UNK_0000_0c32"; case 0x00000c33: return "I7_SB_UNK_0000_0c33";
1708 case 0x00000c34: return "I7_SB_UNK_0000_0c34"; case 0x00000c35: return "I7_SB_UNK_0000_0c35";
1709 case 0x00000c36: return "I7_SB_UNK_0000_0c36"; case 0x00000c37: return "I7_SB_UNK_0000_0c37";
1710 case 0x00000c38: return "I7_SB_UNK_0000_0c38"; case 0x00000c39: return "I7_SB_UNK_0000_0c39";
1711 case 0x00000d04: return "I7_SB_UNK_0000_0d04";
1712 case 0x00000d10: return "I7_SB_UNK_0000_0d10"; case 0x00000d11: return "I7_SB_UNK_0000_0d11";
1713 case 0x00000d12: return "I7_SB_UNK_0000_0d12"; case 0x00000d13: return "I7_SB_UNK_0000_0d13";
1714 case 0x00000d14: return "I7_SB_UNK_0000_0d14"; case 0x00000d15: return "I7_SB_UNK_0000_0d15";
1715 case 0x00000d16: return "I7_SB_UNK_0000_0d16"; case 0x00000d17: return "I7_SB_UNK_0000_0d17";
1716 case 0x00000d18: return "I7_SB_UNK_0000_0d18"; case 0x00000d19: return "I7_SB_UNK_0000_0d19";
1717 case 0x00000d24: return "I7_SB_UNK_0000_0d24";
1718 case 0x00000d30: return "I7_SB_UNK_0000_0d30"; case 0x00000d31: return "I7_SB_UNK_0000_0d31";
1719 case 0x00000d32: return "I7_SB_UNK_0000_0d32"; case 0x00000d33: return "I7_SB_UNK_0000_0d33";
1720 case 0x00000d34: return "I7_SB_UNK_0000_0d34"; case 0x00000d35: return "I7_SB_UNK_0000_0d35";
1721 case 0x00000d36: return "I7_SB_UNK_0000_0d36"; case 0x00000d37: return "I7_SB_UNK_0000_0d37";
1722 case 0x00000d38: return "I7_SB_UNK_0000_0d38"; case 0x00000d39: return "I7_SB_UNK_0000_0d39";
1723 case 0x00000d44: return "I7_SB_UNK_0000_0d44";
1724 case 0x00000d50: return "I7_SB_UNK_0000_0d50"; case 0x00000d51: return "I7_SB_UNK_0000_0d51";
1725 case 0x00000d52: return "I7_SB_UNK_0000_0d52"; case 0x00000d53: return "I7_SB_UNK_0000_0d53";
1726 case 0x00000d54: return "I7_SB_UNK_0000_0d54"; case 0x00000d55: return "I7_SB_UNK_0000_0d55";
1727 case 0x00000d56: return "I7_SB_UNK_0000_0d56"; case 0x00000d57: return "I7_SB_UNK_0000_0d57";
1728 case 0x00000d58: return "I7_SB_UNK_0000_0d58"; case 0x00000d59: return "I7_SB_UNK_0000_0d59";
1729 case 0x00000d64: return "I7_SB_UNK_0000_0d64";
1730 case 0x00000d70: return "I7_SB_UNK_0000_0d70"; case 0x00000d71: return "I7_SB_UNK_0000_0d71";
1731 case 0x00000d72: return "I7_SB_UNK_0000_0d72"; case 0x00000d73: return "I7_SB_UNK_0000_0d73";
1732 case 0x00000d74: return "I7_SB_UNK_0000_0d74"; case 0x00000d75: return "I7_SB_UNK_0000_0d75";
1733 case 0x00000d76: return "I7_SB_UNK_0000_0d76"; case 0x00000d77: return "I7_SB_UNK_0000_0d77";
1734 case 0x00000d78: return "I7_SB_UNK_0000_0d78"; case 0x00000d79: return "I7_SB_UNK_0000_0d79";
1735 case 0x00000d84: return "I7_SB_UNK_0000_0d84";
1736 case 0x00000d90: return "I7_SB_UNK_0000_0d90"; case 0x00000d91: return "I7_SB_UNK_0000_0d91";
1737 case 0x00000d92: return "I7_SB_UNK_0000_0d92"; case 0x00000d93: return "I7_SB_UNK_0000_0d93";
1738 case 0x00000d94: return "I7_SB_UNK_0000_0d94"; case 0x00000d95: return "I7_SB_UNK_0000_0d95";
1739 case 0x00000d96: return "I7_SB_UNK_0000_0d96"; case 0x00000d97: return "I7_SB_UNK_0000_0d97";
1740 case 0x00000d98: return "I7_SB_UNK_0000_0d98"; case 0x00000d99: return "I7_SB_UNK_0000_0d99";
1741 case 0x00000da4: return "I7_SB_UNK_0000_0da4";
1742 case 0x00000db0: return "I7_SB_UNK_0000_0db0"; case 0x00000db1: return "I7_SB_UNK_0000_0db1";
1743 case 0x00000db2: return "I7_SB_UNK_0000_0db2"; case 0x00000db3: return "I7_SB_UNK_0000_0db3";
1744 case 0x00000db4: return "I7_SB_UNK_0000_0db4"; case 0x00000db5: return "I7_SB_UNK_0000_0db5";
1745 case 0x00000db6: return "I7_SB_UNK_0000_0db6"; case 0x00000db7: return "I7_SB_UNK_0000_0db7";
1746 case 0x00000db8: return "I7_SB_UNK_0000_0db8"; case 0x00000db9: return "I7_SB_UNK_0000_0db9";
1747 }
1748
1749 /*
1750 * Ditto for ivy bridge (observed on the i5-3570). There are some haswell
1751 * and sandybridge related docs on registers in this ares, but either
1752 * things are different for ivy or they're very incomplete. Again, kudos
1753 * to intel!
1754 */
1755 if (g_enmMicroarch == kCpumMicroarch_Intel_Core7_IvyBridge)
1756 switch (uMsr)
1757 {
1758 case 0x00000700: return "I7_IB_UNK_0000_0700"; case 0x00000701: return "I7_IB_UNK_0000_0701";
1759 case 0x00000702: return "I7_IB_UNK_0000_0702"; case 0x00000703: return "I7_IB_UNK_0000_0703";
1760 case 0x00000704: return "I7_IB_UNK_0000_0704"; case 0x00000705: return "I7_IB_UNK_0000_0705";
1761 case 0x00000706: return "I7_IB_UNK_0000_0706"; case 0x00000707: return "I7_IB_UNK_0000_0707";
1762 case 0x00000708: return "I7_IB_UNK_0000_0708"; case 0x00000709: return "I7_IB_UNK_0000_0709";
1763 case 0x00000710: return "I7_IB_UNK_0000_0710"; case 0x00000711: return "I7_IB_UNK_0000_0711";
1764 case 0x00000712: return "I7_IB_UNK_0000_0712"; case 0x00000713: return "I7_IB_UNK_0000_0713";
1765 case 0x00000714: return "I7_IB_UNK_0000_0714"; case 0x00000715: return "I7_IB_UNK_0000_0715";
1766 case 0x00000716: return "I7_IB_UNK_0000_0716"; case 0x00000717: return "I7_IB_UNK_0000_0717";
1767 case 0x00000718: return "I7_IB_UNK_0000_0718"; case 0x00000719: return "I7_IB_UNK_0000_0719";
1768 case 0x00000720: return "I7_IB_UNK_0000_0720"; case 0x00000721: return "I7_IB_UNK_0000_0721";
1769 case 0x00000722: return "I7_IB_UNK_0000_0722"; case 0x00000723: return "I7_IB_UNK_0000_0723";
1770 case 0x00000724: return "I7_IB_UNK_0000_0724"; case 0x00000725: return "I7_IB_UNK_0000_0725";
1771 case 0x00000726: return "I7_IB_UNK_0000_0726"; case 0x00000727: return "I7_IB_UNK_0000_0727";
1772 case 0x00000728: return "I7_IB_UNK_0000_0728"; case 0x00000729: return "I7_IB_UNK_0000_0729";
1773 case 0x00000730: return "I7_IB_UNK_0000_0730"; case 0x00000731: return "I7_IB_UNK_0000_0731";
1774 case 0x00000732: return "I7_IB_UNK_0000_0732"; case 0x00000733: return "I7_IB_UNK_0000_0733";
1775 case 0x00000734: return "I7_IB_UNK_0000_0734"; case 0x00000735: return "I7_IB_UNK_0000_0735";
1776 case 0x00000736: return "I7_IB_UNK_0000_0736"; case 0x00000737: return "I7_IB_UNK_0000_0737";
1777 case 0x00000738: return "I7_IB_UNK_0000_0738"; case 0x00000739: return "I7_IB_UNK_0000_0739";
1778 case 0x00000740: return "I7_IB_UNK_0000_0740"; case 0x00000741: return "I7_IB_UNK_0000_0741";
1779 case 0x00000742: return "I7_IB_UNK_0000_0742"; case 0x00000743: return "I7_IB_UNK_0000_0743";
1780 case 0x00000744: return "I7_IB_UNK_0000_0744"; case 0x00000745: return "I7_IB_UNK_0000_0745";
1781 case 0x00000746: return "I7_IB_UNK_0000_0746"; case 0x00000747: return "I7_IB_UNK_0000_0747";
1782 case 0x00000748: return "I7_IB_UNK_0000_0748"; case 0x00000749: return "I7_IB_UNK_0000_0749";
1783
1784 }
1785 return NULL;
1786}
1787
1788
1789/**
1790 * Gets the name of an MSR.
1791 *
1792 * This may return a static buffer, so the content should only be considered
1793 * valid until the next time this function is called!.
1794 *
1795 * @returns MSR name.
1796 * @param uMsr The MSR in question.
1797 */
1798static const char *getMsrName(uint32_t uMsr)
1799{
1800 const char *pszReadOnly = getMsrNameHandled(uMsr);
1801 if (pszReadOnly)
1802 return pszReadOnly;
1803
1804 /*
1805 * This MSR needs looking into, return a TODO_XXXX_XXXX name.
1806 */
1807 static char s_szBuf[32];
1808 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "TODO_%04x_%04x", RT_HI_U16(uMsr), RT_LO_U16(uMsr));
1809 return s_szBuf;
1810}
1811
1812
1813
1814/**
1815 * Gets the name of an MSR range.
1816 *
1817 * This may return a static buffer, so the content should only be considered
1818 * valid until the next time this function is called!.
1819 *
1820 * @returns MSR name.
1821 * @param uMsr The first MSR in the range.
1822 */
1823static const char *getMsrRangeName(uint32_t uMsr)
1824{
1825 switch (uMsr)
1826 {
1827 case 0x00000040:
1828 return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah ? "MSR_LASTBRANCH_n_FROM_IP" : "MSR_LASTBRANCH_n";
1829 case 0x00000060:
1830 if (g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah)
1831 return "MSR_LASTBRANCH_n_TO_IP";
1832 break;
1833
1834 case 0x000003f8:
1835 case 0x000003f9:
1836 case 0x000003fa:
1837 return "I7_MSR_PKG_Cn_RESIDENCY";
1838 case 0x000003fc:
1839 case 0x000003fd:
1840 case 0x000003fe:
1841 return "I7_MSR_CORE_Cn_RESIDENCY";
1842
1843 case 0x00000400:
1844 return "IA32_MCi_CTL_STATUS_ADDR_MISC";
1845
1846 case 0x00000680:
1847 return "MSR_LASTBRANCH_n_FROM_IP";
1848 case 0x000006c0:
1849 return "MSR_LASTBRANCH_n_TO_IP";
1850
1851 case 0x00000800: case 0x00000801: case 0x00000802: case 0x00000803:
1852 case 0x00000804: case 0x00000805: case 0x00000806: case 0x00000807:
1853 case 0x00000808: case 0x00000809: case 0x0000080a: case 0x0000080b:
1854 case 0x0000080c: case 0x0000080d: case 0x0000080e: case 0x0000080f:
1855 return "IA32_X2APIC_n";
1856 }
1857
1858 static char s_szBuf[96];
1859 const char *pszReadOnly = getMsrNameHandled(uMsr);
1860 if (pszReadOnly)
1861 {
1862 /*
1863 * Replace the last char with 'n'.
1864 */
1865 RTStrCopy(s_szBuf, sizeof(s_szBuf), pszReadOnly);
1866 size_t off = strlen(s_szBuf);
1867 if (off > 0)
1868 off--;
1869 if (off + 1 < sizeof(s_szBuf))
1870 {
1871 s_szBuf[off] = 'n';
1872 s_szBuf[off + 1] = '\0';
1873 }
1874 }
1875 else
1876 {
1877 /*
1878 * This MSR needs looking into, return a TODO_XXXX_XXXX_n name.
1879 */
1880 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "TODO_%04x_%04x_n", RT_HI_U16(uMsr), RT_LO_U16(uMsr));
1881 }
1882 return s_szBuf;
1883}
1884
1885
1886/**
1887 * Returns the function name for MSRs that have one or two.
1888 *
1889 * @returns Function name if applicable, NULL if not.
1890 * @param uMsr The MSR in question.
1891 * @param pfTakesValue Whether this MSR function takes a value or not.
1892 * Optional.
1893 */
1894static const char *getMsrFnName(uint32_t uMsr, bool *pfTakesValue)
1895{
1896 bool fTmp;
1897 if (!pfTakesValue)
1898 pfTakesValue = &fTmp;
1899
1900 *pfTakesValue = false;
1901
1902 switch (uMsr)
1903 {
1904 case 0x00000000: return "Ia32P5McAddr";
1905 case 0x00000001: return "Ia32P5McType";
1906 case 0x00000006:
1907 if (g_enmMicroarch >= kCpumMicroarch_Intel_First && g_enmMicroarch <= kCpumMicroarch_Intel_P6_Core_Atom_First)
1908 return NULL; /* TR4 / cache tag on Pentium, but that's for later. */
1909 return "Ia32MonitorFilterLineSize";
1910 case 0x00000010: return "Ia32TimestampCounter";
1911 case 0x00000017: *pfTakesValue = true; return "Ia32PlatformId";
1912 case 0x0000001b: return "Ia32ApicBase";
1913 case 0x0000002a: *pfTakesValue = true; return g_fIntelNetBurst ? "IntelP4EbcHardPowerOn" : "IntelEblCrPowerOn";
1914 case 0x0000002b: *pfTakesValue = true; return g_fIntelNetBurst ? "IntelP4EbcSoftPowerOn" : NULL;
1915 case 0x0000002c: *pfTakesValue = true; return g_fIntelNetBurst ? "IntelP4EbcFrequencyId" : NULL;
1916 //case 0x00000033: return "IntelTestCtl";
1917 case 0x00000034: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch)
1918 || CPUMMICROARCH_IS_INTEL_SILVERMONT_PLUS(g_enmMicroarch)
1919 ? "IntelI7SmiCount" : NULL;
1920 case 0x00000035: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch) ? "IntelI7CoreThreadCount" : NULL;
1921 case 0x0000003a: return "Ia32FeatureControl";
1922
1923 case 0x00000040:
1924 case 0x00000041:
1925 case 0x00000042:
1926 case 0x00000043:
1927 case 0x00000044:
1928 case 0x00000045:
1929 case 0x00000046:
1930 case 0x00000047:
1931 return "IntelLastBranchFromToN";
1932
1933 case 0x0000008b: return g_enmVendor == CPUMCPUVENDOR_AMD ? "AmdK8PatchLevel" : "Ia32BiosSignId";
1934 case 0x0000009b: return "Ia32SmmMonitorCtl";
1935
1936 case 0x000000a8:
1937 case 0x000000a9:
1938 case 0x000000aa:
1939 case 0x000000ab:
1940 case 0x000000ac:
1941 case 0x000000ad:
1942 *pfTakesValue = true;
1943 return "IntelCore2EmttmCrTablesN";
1944
1945 case 0x000000c1:
1946 case 0x000000c2:
1947 case 0x000000c3:
1948 case 0x000000c4:
1949 return "Ia32PmcN";
1950 case 0x000000c5:
1951 case 0x000000c6:
1952 case 0x000000c7:
1953 case 0x000000c8:
1954 if (g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First)
1955 return "Ia32PmcN";
1956 return NULL;
1957
1958 case 0x000000cd: *pfTakesValue = true; return "IntelP6FsbFrequency";
1959 case 0x000000ce: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch) ? "IntelPlatformInfo" : NULL;
1960 case 0x000000e2: return "IntelPkgCStConfigControl";
1961 case 0x000000e3: return "IntelCore2SmmCStMiscInfo";
1962 case 0x000000e4: return "IntelPmgIoCaptureBase";
1963 case 0x000000e7: return "Ia32MPerf";
1964 case 0x000000e8: return "Ia32APerf";
1965 case 0x000000ee: return "IntelCore1ExtConfig";
1966 case 0x000000fe: *pfTakesValue = true; return "Ia32MtrrCap";
1967 case 0x00000119: *pfTakesValue = true; return "IntelBblCrCtl";
1968 case 0x0000011e: *pfTakesValue = true; return "IntelBblCrCtl3";
1969
1970 case 0x00000130: return g_enmMicroarch == kCpumMicroarch_Intel_Core7_Westmere
1971 || g_enmMicroarch == kCpumMicroarch_Intel_Core7_Nehalem
1972 ? "IntelCpuId1FeatureMaskEcdx" : NULL;
1973 case 0x00000131: return g_enmMicroarch == kCpumMicroarch_Intel_Core7_Westmere
1974 || g_enmMicroarch == kCpumMicroarch_Intel_Core7_Nehalem
1975 ? "IntelCpuId80000001FeatureMaskEcdx" : NULL;
1976 case 0x00000132: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
1977 ? "IntelCpuId1FeatureMaskEax" : NULL;
1978 case 0x00000133: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
1979 ? "IntelCpuId1FeatureMaskEcdx" : NULL;
1980 case 0x00000134: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge
1981 ? "IntelCpuId80000001FeatureMaskEcdx" : NULL;
1982 case 0x0000013c: return "IntelI7SandyAesNiCtl";
1983 case 0x0000015f: return "IntelCore1DtsCalControl";
1984 case 0x00000174: return "Ia32SysEnterCs";
1985 case 0x00000175: return "Ia32SysEnterEsp";
1986 case 0x00000176: return "Ia32SysEnterEip";
1987 case 0x00000179: *pfTakesValue = true; return "Ia32McgCap";
1988 case 0x0000017a: return "Ia32McgStatus";
1989 case 0x0000017b: return "Ia32McgCtl";
1990 case 0x0000017f: return "IntelI7SandyErrorControl"; /* SandyBridge. */
1991 case 0x00000186: return "Ia32PerfEvtSelN";
1992 case 0x00000187: return "Ia32PerfEvtSelN";
1993 case 0x00000193: return /*g_fIntelNetBurst ? NULL :*/ NULL /* Core2_Penryn. */;
1994 case 0x00000194: if (g_fIntelNetBurst) break; *pfTakesValue = true; return "IntelFlexRatio";
1995 case 0x00000198: *pfTakesValue = true; return "Ia32PerfStatus";
1996 case 0x00000199: *pfTakesValue = true; return "Ia32PerfCtl";
1997 case 0x0000019a: *pfTakesValue = true; return "Ia32ClockModulation";
1998 case 0x0000019b: *pfTakesValue = true; return "Ia32ThermInterrupt";
1999 case 0x0000019c: *pfTakesValue = true; return "Ia32ThermStatus";
2000 case 0x0000019d: *pfTakesValue = true; return "Ia32Therm2Ctl";
2001 case 0x000001a0: *pfTakesValue = true; return "Ia32MiscEnable";
2002 case 0x000001a2: *pfTakesValue = true; return "IntelI7TemperatureTarget";
2003 case 0x000001a6: return "IntelI7MsrOffCoreResponseN";
2004 case 0x000001a7: return "IntelI7MsrOffCoreResponseN";
2005 case 0x000001aa: return CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch) ? "IntelI7MiscPwrMgmt" : NULL /*"P6PicSensCfg"*/;
2006 case 0x000001ad: *pfTakesValue = true; return "IntelI7TurboRatioLimit"; /* SandyBridge+, Silvermount+ */
2007 case 0x000001c8: return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_Nehalem ? "IntelI7LbrSelect" : NULL;
2008 case 0x000001c9: return g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah
2009 && g_enmMicroarch <= kCpumMicroarch_Intel_P6_Core_Atom_End
2010 ? "IntelLastBranchTos" : NULL /* Pentium M Dothan seems to have something else here. */;
2011 case 0x000001d7: return g_fIntelNetBurst ? "P6LastIntFromIp" : NULL;
2012 case 0x000001d8: return g_fIntelNetBurst ? "P6LastIntToIp" : NULL;
2013 case 0x000001d9: return "Ia32DebugCtl";
2014 case 0x000001da: return g_fIntelNetBurst ? "IntelLastBranchTos" : NULL;
2015 case 0x000001db: return g_fIntelNetBurst ? "IntelLastBranchFromToN" : "P6LastBranchFromIp";
2016 case 0x000001dc: return g_fIntelNetBurst ? "IntelLastBranchFromToN" : "P6LastBranchToIp";
2017 case 0x000001dd: return g_fIntelNetBurst ? "IntelLastBranchFromToN" : "P6LastIntFromIp";
2018 case 0x000001de: return g_fIntelNetBurst ? "IntelLastBranchFromToN" : "P6LastIntToIp";
2019 case 0x000001f0: return "IntelI7VirtualLegacyWireCap"; /* SandyBridge. */
2020 case 0x000001f2: return "Ia32SmrrPhysBase";
2021 case 0x000001f3: return "Ia32SmrrPhysMask";
2022 case 0x000001f8: return "Ia32PlatformDcaCap";
2023 case 0x000001f9: return "Ia32CpuDcaCap";
2024 case 0x000001fa: return "Ia32Dca0Cap";
2025 case 0x000001fc: return "IntelI7PowerCtl";
2026
2027 case 0x00000200: case 0x00000202: case 0x00000204: case 0x00000206:
2028 case 0x00000208: case 0x0000020a: case 0x0000020c: case 0x0000020e:
2029 case 0x00000210: case 0x00000212: case 0x00000214: case 0x00000216:
2030 case 0x00000218: case 0x0000021a: case 0x0000021c: case 0x0000021e:
2031 return "Ia32MtrrPhysBaseN";
2032 case 0x00000201: case 0x00000203: case 0x00000205: case 0x00000207:
2033 case 0x00000209: case 0x0000020b: case 0x0000020d: case 0x0000020f:
2034 case 0x00000211: case 0x00000213: case 0x00000215: case 0x00000217:
2035 case 0x00000219: case 0x0000021b: case 0x0000021d: case 0x0000021f:
2036 return "Ia32MtrrPhysMaskN";
2037 case 0x00000250:
2038 case 0x00000258: case 0x00000259:
2039 case 0x00000268: case 0x00000269: case 0x0000026a: case 0x0000026b:
2040 case 0x0000026c: case 0x0000026d: case 0x0000026e: case 0x0000026f:
2041 return "Ia32MtrrFixed";
2042 case 0x00000277: *pfTakesValue = true; return "Ia32Pat";
2043
2044 case 0x00000280: case 0x00000281: case 0x00000282: case 0x00000283:
2045 case 0x00000284: case 0x00000285: case 0x00000286: case 0x00000287:
2046 case 0x00000288: case 0x00000289: case 0x0000028a: case 0x0000028b:
2047 case 0x0000028c: case 0x0000028d: case 0x0000028e: case 0x0000028f:
2048 case 0x00000290: case 0x00000291: case 0x00000292: case 0x00000293:
2049 case 0x00000294: case 0x00000295: //case 0x00000296: case 0x00000297:
2050 //case 0x00000298: case 0x00000299: case 0x0000029a: case 0x0000029b:
2051 //case 0x0000029c: case 0x0000029d: case 0x0000029e: case 0x0000029f:
2052 return "Ia32McNCtl2";
2053
2054 case 0x000002ff: return "Ia32MtrrDefType";
2055 //case 0x00000305: return g_fIntelNetBurst ? TODO : NULL;
2056 case 0x00000309: return g_fIntelNetBurst ? NULL /** @todo P4 */ : "Ia32FixedCtrN";
2057 case 0x0000030a: return g_fIntelNetBurst ? NULL /** @todo P4 */ : "Ia32FixedCtrN";
2058 case 0x0000030b: return g_fIntelNetBurst ? NULL /** @todo P4 */ : "Ia32FixedCtrN";
2059 case 0x00000345: *pfTakesValue = true; return "Ia32PerfCapabilities";
2060 /* Note! Lots of P4 MSR 0x00000360..0x00000371. */
2061 case 0x0000038d: return "Ia32FixedCtrCtrl";
2062 case 0x0000038e: *pfTakesValue = true; return "Ia32PerfGlobalStatus";
2063 case 0x0000038f: return "Ia32PerfGlobalCtrl";
2064 case 0x00000390: return "Ia32PerfGlobalOvfCtrl";
2065 case 0x00000391: return "IntelI7UncPerfGlobalCtrl"; /* S,H,X */
2066 case 0x00000392: return "IntelI7UncPerfGlobalStatus"; /* S,H,X */
2067 case 0x00000393: return "IntelI7UncPerfGlobalOvfCtrl"; /* X. ASSUMING this is the same on sandybridge and later. */
2068 case 0x00000394: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPerfFixedCtr" /* X */ : "IntelI7UncPerfFixedCtrCtrl"; /* >= S,H */
2069 case 0x00000395: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPerfFixedCtrCtrl" /* X*/ : "IntelI7UncPerfFixedCtr"; /* >= S,H */
2070 case 0x00000396: return g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncAddrOpcodeMatch" /* X */ : "IntelI7UncCBoxConfig"; /* >= S,H */
2071 case 0x0000039c: return "IntelI7SandyPebsNumAlt";
2072 /* Note! Lots of P4 MSR 0x000003a0..0x000003e1. */
2073 case 0x000003b0: return g_fIntelNetBurst ? NULL : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPmcN" /* X */ : "IntelI7UncArbPerfCtrN"; /* >= S,H */
2074 case 0x000003b1: return g_fIntelNetBurst ? NULL : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPmcN" /* X */ : "IntelI7UncArbPerfCtrN"; /* >= S,H */
2075 case 0x000003b2: return g_fIntelNetBurst ? NULL : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPmcN" /* X */ : "IntelI7UncArbPerfEvtSelN"; /* >= S,H */
2076 case 0x000003b3: return g_fIntelNetBurst ? NULL : g_enmMicroarch < kCpumMicroarch_Intel_Core7_SandyBridge ? "IntelI7UncPmcN" /* X */ : "IntelI7UncArbPerfEvtSelN"; /* >= S,H */
2077 case 0x000003b4: case 0x000003b5: case 0x000003b6: case 0x000003b7:
2078 return g_fIntelNetBurst ? NULL : "IntelI7UncPmcN";
2079 case 0x000003c0: case 0x000003c1: case 0x000003c2: case 0x000003c3:
2080 case 0x000003c4: case 0x000003c5: case 0x000003c6: case 0x000003c7:
2081 return g_fIntelNetBurst ? NULL : "IntelI7UncPerfEvtSelN";
2082 case 0x000003f1: return "Ia32PebsEnable";
2083 case 0x000003f6: return g_fIntelNetBurst ? NULL /*??*/ : "IntelI7PebsLdLat";
2084 case 0x000003f8: return g_fIntelNetBurst ? NULL : "IntelI7PkgCnResidencyN";
2085 case 0x000003f9: return "IntelI7PkgCnResidencyN";
2086 case 0x000003fa: return "IntelI7PkgCnResidencyN";
2087 case 0x000003fc: return "IntelI7CoreCnResidencyN";
2088 case 0x000003fd: return "IntelI7CoreCnResidencyN";
2089 case 0x000003fe: return "IntelI7CoreCnResidencyN";
2090
2091 case 0x00000478: return g_enmMicroarch == kCpumMicroarch_Intel_Core2_Penryn ? "IntelCpuId1FeatureMaskEcdx" : NULL;
2092 case 0x00000480: *pfTakesValue = true; return "Ia32VmxBase";
2093 case 0x00000481: *pfTakesValue = true; return "Ia32VmxPinbasedCtls";
2094 case 0x00000482: *pfTakesValue = true; return "Ia32VmxProcbasedCtls";
2095 case 0x00000483: *pfTakesValue = true; return "Ia32VmxExitCtls";
2096 case 0x00000484: *pfTakesValue = true; return "Ia32VmxEntryCtls";
2097 case 0x00000485: *pfTakesValue = true; return "Ia32VmxMisc";
2098 case 0x00000486: *pfTakesValue = true; return "Ia32VmxCr0Fixed0";
2099 case 0x00000487: *pfTakesValue = true; return "Ia32VmxCr0Fixed1";
2100 case 0x00000488: *pfTakesValue = true; return "Ia32VmxCr4Fixed0";
2101 case 0x00000489: *pfTakesValue = true; return "Ia32VmxCr4Fixed1";
2102 case 0x0000048a: *pfTakesValue = true; return "Ia32VmxVmcsEnum";
2103 case 0x0000048b: *pfTakesValue = true; return "Ia32VmxProcBasedCtls2";
2104 case 0x0000048c: *pfTakesValue = true; return "Ia32VmxEptVpidCap";
2105 case 0x0000048d: *pfTakesValue = true; return "Ia32VmxTruePinbasedCtls";
2106 case 0x0000048e: *pfTakesValue = true; return "Ia32VmxTrueProcbasedCtls";
2107 case 0x0000048f: *pfTakesValue = true; return "Ia32VmxTrueExitCtls";
2108 case 0x00000490: *pfTakesValue = true; return "Ia32VmxTrueEntryCtls";
2109 case 0x00000491: *pfTakesValue = true; return "Ia32VmxVmFunc";
2110
2111 case 0x000004c1:
2112 case 0x000004c2:
2113 case 0x000004c3:
2114 case 0x000004c4:
2115 case 0x000004c5:
2116 case 0x000004c6:
2117 case 0x000004c7:
2118 case 0x000004c8:
2119 return "Ia32PmcN";
2120
2121 case 0x000005a0: return "IntelCore2PeciControl"; /* Core2_Penryn. */
2122
2123 case 0x00000600: return "Ia32DsArea";
2124 case 0x00000601: *pfTakesValue = true; return "IntelI7SandyVrCurrentConfig";
2125 case 0x00000603: *pfTakesValue = true; return "IntelI7SandyVrMiscConfig";
2126 case 0x00000606: *pfTakesValue = true; return "IntelI7SandyRaplPowerUnit";
2127 case 0x0000060a: *pfTakesValue = true; return "IntelI7SandyPkgCnIrtlN";
2128 case 0x0000060b: *pfTakesValue = true; return "IntelI7SandyPkgCnIrtlN";
2129 case 0x0000060c: *pfTakesValue = true; return "IntelI7SandyPkgCnIrtlN";
2130 case 0x0000060d: *pfTakesValue = true; return "IntelI7SandyPkgC2Residency";
2131
2132 case 0x00000610: *pfTakesValue = true; return "IntelI7RaplPkgPowerLimit";
2133 case 0x00000611: *pfTakesValue = true; return "IntelI7RaplPkgEnergyStatus";
2134 case 0x00000613: *pfTakesValue = true; return "IntelI7RaplPkgPerfStatus";
2135 case 0x00000614: *pfTakesValue = true; return "IntelI7RaplPkgPowerInfo";
2136 case 0x00000618: *pfTakesValue = true; return "IntelI7RaplDramPowerLimit";
2137 case 0x00000619: *pfTakesValue = true; return "IntelI7RaplDramEnergyStatus";
2138 case 0x0000061b: *pfTakesValue = true; return "IntelI7RaplDramPerfStatus";
2139 case 0x0000061c: *pfTakesValue = true; return "IntelI7RaplDramPowerInfo";
2140 case 0x00000638: *pfTakesValue = true; return "IntelI7RaplPp0PowerLimit";
2141 case 0x00000639: *pfTakesValue = true; return "IntelI7RaplPp0EnergyStatus";
2142 case 0x0000063a: *pfTakesValue = true; return "IntelI7RaplPp0Policy";
2143 case 0x0000063b: *pfTakesValue = true; return "IntelI7RaplPp0PerfStatus";
2144 case 0x00000640: *pfTakesValue = true; return "IntelI7RaplPp1PowerLimit";
2145 case 0x00000641: *pfTakesValue = true; return "IntelI7RaplPp1EnergyStatus";
2146 case 0x00000642: *pfTakesValue = true; return "IntelI7RaplPp1Policy";
2147 case 0x00000648: *pfTakesValue = true; return "IntelI7IvyConfigTdpNominal";
2148 case 0x00000649: *pfTakesValue = true; return "IntelI7IvyConfigTdpLevel1";
2149 case 0x0000064a: *pfTakesValue = true; return "IntelI7IvyConfigTdpLevel2";
2150 case 0x0000064b: return "IntelI7IvyConfigTdpControl";
2151 case 0x0000064c: return "IntelI7IvyTurboActivationRatio";
2152
2153 case 0x00000660: return "IntelAtSilvCoreC1Recidency";
2154
2155 case 0x00000680: case 0x00000681: case 0x00000682: case 0x00000683:
2156 case 0x00000684: case 0x00000685: case 0x00000686: case 0x00000687:
2157 case 0x00000688: case 0x00000689: case 0x0000068a: case 0x0000068b:
2158 case 0x0000068c: case 0x0000068d: case 0x0000068e: case 0x0000068f:
2159 //case 0x00000690: case 0x00000691: case 0x00000692: case 0x00000693:
2160 //case 0x00000694: case 0x00000695: case 0x00000696: case 0x00000697:
2161 //case 0x00000698: case 0x00000699: case 0x0000069a: case 0x0000069b:
2162 //case 0x0000069c: case 0x0000069d: case 0x0000069e: case 0x0000069f:
2163 return "IntelLastBranchFromN";
2164 case 0x000006c0: case 0x000006c1: case 0x000006c2: case 0x000006c3:
2165 case 0x000006c4: case 0x000006c5: case 0x000006c6: case 0x000006c7:
2166 case 0x000006c8: case 0x000006c9: case 0x000006ca: case 0x000006cb:
2167 case 0x000006cc: case 0x000006cd: case 0x000006ce: case 0x000006cf:
2168 //case 0x000006d0: case 0x000006d1: case 0x000006d2: case 0x000006d3:
2169 //case 0x000006d4: case 0x000006d5: case 0x000006d6: case 0x000006d7:
2170 //case 0x000006d8: case 0x000006d9: case 0x000006da: case 0x000006db:
2171 //case 0x000006dc: case 0x000006dd: case 0x000006de: case 0x000006df:
2172 return "IntelLastBranchToN";
2173 case 0x000006e0: return "Ia32TscDeadline"; /** @todo detect this correctly! */
2174
2175 case 0x00000c80: return g_enmMicroarch > kCpumMicroarch_Intel_Core7_Nehalem ? "Ia32DebugInterface" : NULL;
2176
2177 case 0xc0000080: return "Amd64Efer";
2178 case 0xc0000081: return "Amd64SyscallTarget";
2179 case 0xc0000082: return "Amd64LongSyscallTarget";
2180 case 0xc0000083: return "Amd64CompSyscallTarget";
2181 case 0xc0000084: return "Amd64SyscallFlagMask";
2182 case 0xc0000100: return "Amd64FsBase";
2183 case 0xc0000101: return "Amd64GsBase";
2184 case 0xc0000102: return "Amd64KernelGsBase";
2185 case 0xc0000103: return "Amd64TscAux";
2186 case 0xc0000104: return "AmdFam15hTscRate";
2187 case 0xc0000105: return "AmdFam15hLwpCfg";
2188 case 0xc0000106: return "AmdFam15hLwpCbAddr";
2189 case 0xc0000408: return "AmdFam10hMc4MiscN";
2190 case 0xc0000409: return "AmdFam10hMc4MiscN";
2191 case 0xc000040a: return "AmdFam10hMc4MiscN";
2192 case 0xc000040b: return "AmdFam10hMc4MiscN";
2193 case 0xc000040c: return "AmdFam10hMc4MiscN";
2194 case 0xc000040d: return "AmdFam10hMc4MiscN";
2195 case 0xc000040e: return "AmdFam10hMc4MiscN";
2196 case 0xc000040f: return "AmdFam10hMc4MiscN";
2197 case 0xc0010000: return "AmdK8PerfCtlN";
2198 case 0xc0010001: return "AmdK8PerfCtlN";
2199 case 0xc0010002: return "AmdK8PerfCtlN";
2200 case 0xc0010003: return "AmdK8PerfCtlN";
2201 case 0xc0010004: return "AmdK8PerfCtrN";
2202 case 0xc0010005: return "AmdK8PerfCtrN";
2203 case 0xc0010006: return "AmdK8PerfCtrN";
2204 case 0xc0010007: return "AmdK8PerfCtrN";
2205 case 0xc0010010: *pfTakesValue = true; return "AmdK8SysCfg";
2206 case 0xc0010015: return "AmdK8HwCr";
2207 case 0xc0010016: case 0xc0010018: return "AmdK8IorrBaseN";
2208 case 0xc0010017: case 0xc0010019: return "AmdK8IorrMaskN";
2209 case 0xc001001a: case 0xc001001d: return "AmdK8TopOfMemN";
2210 case 0xc001001f: return "AmdK8NbCfg1";
2211 case 0xc0010020: return "AmdK8PatchLoader";
2212 case 0xc0010022: return "AmdK8McXcptRedir";
2213 case 0xc0010030: case 0xc0010031: case 0xc0010032:
2214 case 0xc0010033: case 0xc0010034: case 0xc0010035:
2215 return "AmdK8CpuNameN";
2216 case 0xc001003e: *pfTakesValue = true; return "AmdK8HwThermalCtrl";
2217 case 0xc001003f: return "AmdK8SwThermalCtrl";
2218 case 0xc0010041: *pfTakesValue = true; return "AmdK8FidVidControl";
2219 case 0xc0010042: *pfTakesValue = true; return "AmdK8FidVidStatus";
2220 case 0xc0010044: case 0xc0010045: case 0xc0010046: case 0xc0010047:
2221 case 0xc0010048: case 0xc0010049: case 0xc001004a: //case 0xc001004b:
2222 return "AmdK8McCtlMaskN";
2223 case 0xc0010050: case 0xc0010051: case 0xc0010052: case 0xc0010053:
2224 return "AmdK8SmiOnIoTrapN";
2225 case 0xc0010054: return "AmdK8SmiOnIoTrapCtlSts";
2226 case 0xc0010055: return "AmdK8IntPendingMessage";
2227 case 0xc0010056: return "AmdK8SmiTriggerIoCycle";
2228 case 0xc0010058: return "AmdFam10hMmioCfgBaseAddr";
2229 case 0xc0010059: return "AmdFam10hTrapCtlMaybe";
2230 case 0xc0010061: *pfTakesValue = true; return "AmdFam10hPStateCurLimit";
2231 case 0xc0010062: *pfTakesValue = true; return "AmdFam10hPStateControl";
2232 case 0xc0010063: *pfTakesValue = true; return "AmdFam10hPStateStatus";
2233 case 0xc0010064: case 0xc0010065: case 0xc0010066: case 0xc0010067:
2234 case 0xc0010068: case 0xc0010069: case 0xc001006a: case 0xc001006b:
2235 *pfTakesValue = true; return "AmdFam10hPStateN";
2236 case 0xc0010070: *pfTakesValue = true; return "AmdFam10hCofVidControl";
2237 case 0xc0010071: *pfTakesValue = true; return "AmdFam10hCofVidStatus";
2238 case 0xc0010073: return "AmdFam10hCStateIoBaseAddr";
2239 case 0xc0010074: return "AmdFam10hCpuWatchdogTimer";
2240 // case 0xc0010075: return "AmdFam15hApmlTdpLimit";
2241 // case 0xc0010077: return "AmdFam15hCpuPowerInTdp";
2242 // case 0xc0010078: return "AmdFam15hPowerAveragingPeriod";
2243 // case 0xc0010079: return "AmdFam15hDramCtrlCmdThrottle";
2244 // case 0xc0010080: return "AmdFam16hFreqSensFeedbackMonActCnt0";
2245 // case 0xc0010081: return "AmdFam16hFreqSensFeedbackMonRefCnt0";
2246 case 0xc0010111: return "AmdK8SmmBase"; /** @todo probably misdetected ign/gp due to locking */
2247 case 0xc0010112: return "AmdK8SmmAddr"; /** @todo probably misdetected ign/gp due to locking */
2248 case 0xc0010113: return "AmdK8SmmMask"; /** @todo probably misdetected ign/gp due to locking */
2249 case 0xc0010114: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AmdK8VmCr" : NULL; /** @todo probably misdetected due to locking */
2250 case 0xc0010115: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AmdK8IgnNe" : NULL;
2251 case 0xc0010116: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AmdK8SmmCtl" : NULL;
2252 case 0xc0010117: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AmdK8VmHSavePa" : NULL; /** @todo probably misdetected due to locking */
2253 case 0xc0010118: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV ? "AmdFam10hVmLockKey" : NULL;
2254 case 0xc0010119: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AmdFam10hSmmLockKey" : NULL; /* Not documented by BKDG, found in netbsd patch. */
2255 case 0xc001011a: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm ? "AmdFam10hLocalSmiStatus" : NULL;
2256 case 0xc0010140: *pfTakesValue = true; return "AmdFam10hOsVisWrkIdLength";
2257 case 0xc0010141: *pfTakesValue = true; return "AmdFam10hOsVisWrkStatus";
2258 case 0xc0010200: case 0xc0010202: case 0xc0010204: case 0xc0010206:
2259 case 0xc0010208: case 0xc001020a: //case 0xc001020c: case 0xc001020e:
2260 return "AmdK8PerfCtlN";
2261 case 0xc0010201: case 0xc0010203: case 0xc0010205: case 0xc0010207:
2262 case 0xc0010209: case 0xc001020b: //case 0xc001020d: case 0xc001020f:
2263 return "AmdK8PerfCtrN";
2264 case 0xc0010230: case 0xc0010232: case 0xc0010234: case 0xc0010236:
2265 //case 0xc0010238: case 0xc001023a: case 0xc001030c: case 0xc001023e:
2266 return "AmdFam16hL2IPerfCtlN";
2267 case 0xc0010231: case 0xc0010233: case 0xc0010235: case 0xc0010237:
2268 //case 0xc0010239: case 0xc001023b: case 0xc001023d: case 0xc001023f:
2269 return "AmdFam16hL2IPerfCtrN";
2270 case 0xc0010240: case 0xc0010242: case 0xc0010244: case 0xc0010246:
2271 //case 0xc0010248: case 0xc001024a: case 0xc001024c: case 0xc001024e:
2272 return "AmdFam15hNorthbridgePerfCtlN";
2273 case 0xc0010241: case 0xc0010243: case 0xc0010245: case 0xc0010247:
2274 //case 0xc0010249: case 0xc001024b: case 0xc001024d: case 0xc001024f:
2275 return "AmdFam15hNorthbridgePerfCtrN";
2276 case 0xc0011000: *pfTakesValue = true; return "AmdK7MicrocodeCtl";
2277 case 0xc0011001: *pfTakesValue = true; return "AmdK7ClusterIdMaybe";
2278 case 0xc0011002: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AmdK8CpuIdCtlStd07hEbax" : NULL;
2279 case 0xc0011003: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AmdK8CpuIdCtlStd06hEcx" : NULL;
2280 case 0xc0011004: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AmdK8CpuIdCtlStd01hEdcx" : NULL;
2281 case 0xc0011005: return g_enmMicroarch >= kCpumMicroarch_AMD_K8_First ? "AmdK8CpuIdCtlExt01hEdcx" : NULL;
2282 case 0xc0011006: return "AmdK7DebugStatusMaybe";
2283 case 0xc0011007: return "AmdK7BHTraceBaseMaybe";
2284 case 0xc0011008: return "AmdK7BHTracePtrMaybe";
2285 case 0xc0011009: return "AmdK7BHTraceLimitMaybe";
2286 case 0xc001100a: return "AmdK7HardwareDebugToolCfgMaybe";
2287 case 0xc001100b: return "AmdK7FastFlushCountMaybe";
2288 case 0xc001100c: return "AmdK7NodeId"; /** @todo dunno if this was there is K7 already. Kinda doubt it. */
2289 case 0xc0011019: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AmdK7DrXAddrMaskN" : NULL;
2290 case 0xc001101a: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AmdK7DrXAddrMaskN" : NULL;
2291 case 0xc001101b: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver ? "AmdK7DrXAddrMaskN" : NULL;
2292 case 0xc0011020: return "AmdK7LoadStoreCfg";
2293 case 0xc0011021: return "AmdK7InstrCacheCfg";
2294 case 0xc0011022: return "AmdK7DataCacheCfg";
2295 case 0xc0011023: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AmdFam15hCombUnitCfg" : "AmdK7BusUnitCfg";
2296 case 0xc0011024: return "AmdK7DebugCtl2Maybe";
2297 case 0xc0011025: return "AmdK7Dr0DataMatchMaybe";
2298 case 0xc0011026: return "AmdK7Dr0DataMaskMaybe";
2299 case 0xc0011027: return "AmdK7DrXAddrMaskN";
2300 case 0xc0011028: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_First ? "AmdFam15hFpuCfg" : NULL;
2301 case 0xc0011029: return g_enmMicroarch >= kCpumMicroarch_AMD_15h_First ? "AmdFam15hDecoderCfg" : NULL;
2302 case 0xc001102a: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AmdFam15hCombUnitCfg2"
2303 : CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch) || g_enmMicroarch > kCpumMicroarch_AMD_15h_End
2304 ? "AmdFam10hBusUnitCfg2" /* 10h & 16h */ : NULL;
2305 case 0xc001102b: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AmdFam15hCombUnitCfg3" : NULL;
2306 case 0xc001102c: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AmdFam15hExecUnitCfg" : NULL;
2307 case 0xc001102d: return CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch) ? "AmdFam15hLoadStoreCfg2" : NULL;
2308 case 0xc0011030: return "AmdFam10hIbsFetchCtl";
2309 case 0xc0011031: return "AmdFam10hIbsFetchLinAddr";
2310 case 0xc0011032: return "AmdFam10hIbsFetchPhysAddr";
2311 case 0xc0011033: return "AmdFam10hIbsOpExecCtl";
2312 case 0xc0011034: return "AmdFam10hIbsOpRip";
2313 case 0xc0011035: return "AmdFam10hIbsOpData";
2314 case 0xc0011036: return "AmdFam10hIbsOpData2";
2315 case 0xc0011037: return "AmdFam10hIbsOpData3";
2316 case 0xc0011038: return "AmdFam10hIbsDcLinAddr";
2317 case 0xc0011039: return "AmdFam10hIbsDcPhysAddr";
2318 case 0xc001103a: return "AmdFam10hIbsCtl";
2319 case 0xc001103b: return "AmdFam14hIbsBrTarget";
2320 }
2321 return NULL;
2322}
2323
2324
2325/**
2326 * Names CPUMCPU variables that MSRs corresponds to.
2327 *
2328 * @returns The variable name @a uMsr corresponds to, NULL if no variable.
2329 * @param uMsr The MSR in question.
2330 */
2331static const char *getMsrCpumCpuVarName(uint32_t uMsr)
2332{
2333 switch (uMsr)
2334 {
2335 case 0x00000250: return "GuestMsrs.msr.MtrrFix64K_00000";
2336 case 0x00000258: return "GuestMsrs.msr.MtrrFix16K_80000";
2337 case 0x00000259: return "GuestMsrs.msr.MtrrFix16K_A0000";
2338 case 0x00000268: return "GuestMsrs.msr.MtrrFix4K_C0000";
2339 case 0x00000269: return "GuestMsrs.msr.MtrrFix4K_C8000";
2340 case 0x0000026a: return "GuestMsrs.msr.MtrrFix4K_D0000";
2341 case 0x0000026b: return "GuestMsrs.msr.MtrrFix4K_D8000";
2342 case 0x0000026c: return "GuestMsrs.msr.MtrrFix4K_E0000";
2343 case 0x0000026d: return "GuestMsrs.msr.MtrrFix4K_E8000";
2344 case 0x0000026e: return "GuestMsrs.msr.MtrrFix4K_F0000";
2345 case 0x0000026f: return "GuestMsrs.msr.MtrrFix4K_F8000";
2346 case 0x00000277: return "Guest.msrPAT";
2347 case 0x000002ff: return "GuestMsrs.msr.MtrrDefType";
2348 }
2349 return NULL;
2350}
2351
2352
2353/**
2354 * Checks whether the MSR should read as zero for some reason.
2355 *
2356 * @returns true if the register should read as zero, false if not.
2357 * @param uMsr The MSR.
2358 */
2359static bool doesMsrReadAsZero(uint32_t uMsr)
2360{
2361 switch (uMsr)
2362 {
2363 case 0x00000088: return true; // "BBL_CR_D0" - RAZ until understood/needed.
2364 case 0x00000089: return true; // "BBL_CR_D1" - RAZ until understood/needed.
2365 case 0x0000008a: return true; // "BBL_CR_D2" - RAZ until understood/needed.
2366
2367 /* Non-zero, but unknown register. */
2368 case 0x0000004a:
2369 case 0x0000004b:
2370 case 0x0000004c:
2371 case 0x0000004d:
2372 case 0x0000004e:
2373 case 0x0000004f:
2374 case 0x00000050:
2375 case 0x00000051:
2376 case 0x00000052:
2377 case 0x00000053:
2378 case 0x00000054:
2379 case 0x0000008c:
2380 case 0x0000008d:
2381 case 0x0000008e:
2382 case 0x0000008f:
2383 case 0x00000090:
2384 case 0xc0011011:
2385 return true;
2386 }
2387
2388 return false;
2389}
2390
2391
2392/**
2393 * Gets the skip mask for the given MSR.
2394 *
2395 * @returns Skip mask (0 means skipping nothing).
2396 * @param uMsr The MSR.
2397 */
2398static uint64_t getGenericSkipMask(uint32_t uMsr)
2399{
2400 switch (uMsr)
2401 {
2402 case 0x0000013c: return 3; /* AES-NI lock bit ++. */
2403
2404 case 0x000001f2: return UINT64_C(0xfffff00f); /* Ia32SmrrPhysBase - Only writable in SMM. */
2405 case 0x000001f3: return UINT64_C(0xfffff800); /* Ia32SmrrPhysMask - Only writable in SMM. */
2406
2407 /* these two have lock bits. */
2408 case 0x0000064b: return UINT64_C(0x80000003);
2409 case 0x0000064c: return UINT64_C(0x800000ff);
2410
2411 case 0xc0010015: return 1; /* SmmLock bit */
2412
2413 /* SmmLock effect: */
2414 case 0xc0010111: return UINT32_MAX;
2415 case 0xc0010112: return UINT64_C(0xfffe0000) | ((RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & ~(uint64_t)UINT32_MAX);
2416 case 0xc0010113: return UINT64_C(0xfffe773f) | ((RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & ~(uint64_t)UINT32_MAX);
2417 case 0xc0010116: return 0x1f;
2418
2419 case 0xc0010114: return RT_BIT_64(3) /* SVM lock */ | RT_BIT_64(4) /* SvmeDisable */;
2420
2421 /* Canonical */
2422 case 0xc0011034:
2423 case 0xc0011038:
2424 case 0xc001103b:
2425 return UINT64_C(0xffff800000000000);
2426
2427 case 0x00000060: case 0x00000061: case 0x00000062: case 0x00000063:
2428 case 0x00000064: case 0x00000065: case 0x00000066: case 0x00000067:
2429 case 0x00000040: case 0x00000041: case 0x00000042: case 0x00000043:
2430 case 0x00000044: case 0x00000045: case 0x00000046: case 0x00000047:
2431 case 0x00000600:
2432 if (g_enmMicroarch >= kCpumMicroarch_Intel_Core2_First)
2433 return UINT64_C(0xffff800000000000);
2434 break;
2435
2436
2437 /* Write only bits. */
2438 case 0xc0010041: return RT_BIT_64(16); /* FIDVID_CTL.InitFidVid */
2439
2440 /* Time counters - fudge them to avoid incorrect ignore masks. */
2441 case 0x00000010:
2442 case 0x000000e7:
2443 case 0x000000e8:
2444 return RT_BIT_32(29) - 1;
2445 }
2446 return 0;
2447}
2448
2449
2450
2451
2452/** queryMsrWriteBadness return values. */
2453typedef enum
2454{
2455 /** . */
2456 VBCPUREPBADNESS_MOSTLY_HARMLESS = 0,
2457 /** Not a problem if accessed with care. */
2458 VBCPUREPBADNESS_MIGHT_BITE,
2459 /** Worse than a bad james bond villain. */
2460 VBCPUREPBADNESS_BOND_VILLAIN
2461} VBCPUREPBADNESS;
2462
2463
2464/**
2465 * Backlisting and graylisting of MSRs which may cause tripple faults.
2466 *
2467 * @returns Badness factor.
2468 * @param uMsr The MSR in question.
2469 */
2470static VBCPUREPBADNESS queryMsrWriteBadness(uint32_t uMsr)
2471{
2472 /** @todo Having trouble in the 0xc0010247,0xc0011006,?? region on Bulldozer. */
2473 /** @todo Having trouble in the 0xc001100f,0xc001100d,?? region on Opteron
2474 * 2384. */
2475
2476 switch (uMsr)
2477 {
2478 case 0x00000050:
2479 case 0x00000051:
2480 case 0x00000052:
2481 case 0x00000053:
2482 case 0x00000054:
2483
2484 case 0x00001006:
2485 case 0x00001007:
2486 return VBCPUREPBADNESS_BOND_VILLAIN;
2487
2488 case 0x0000120e:
2489 case 0x00001233:
2490 case 0x00001239:
2491 case 0x00001249:
2492 case 0x0000124a:
2493 case 0x00001404:
2494 case 0x00001405:
2495 case 0x00001413:
2496 case 0x0000142c: /* Caused rip to be set to 297 or some such weirdness... */
2497 case 0x0000142e:
2498 case 0x00001435:
2499 case 0x00001436:
2500 case 0x00001438:
2501 case 0x0000317f:
2502 if (g_enmVendor == CPUMCPUVENDOR_VIA)
2503 return VBCPUREPBADNESS_BOND_VILLAIN;
2504 break;
2505
2506 case 0xc0010010:
2507 case 0xc0010016:
2508 case 0xc0010017:
2509 case 0xc0010018:
2510 case 0xc0010019:
2511 case 0xc001001a:
2512 case 0xc001001d:
2513 case 0xc0010064: /* P-state fequency, voltage, ++. */
2514 case 0xc0010065: /* P-state fequency, voltage, ++. */
2515 case 0xc0010066: /* P-state fequency, voltage, ++. */
2516 case 0xc0010067: /* P-state fequency, voltage, ++. */
2517 case 0xc0010068: /* P-state fequency, voltage, ++. */
2518 case 0xc0010069: /* P-state fequency, voltage, ++. */
2519 case 0xc001006a: /* P-state fequency, voltage, ++. */
2520 case 0xc001006b: /* P-state fequency, voltage, ++. */
2521 case 0xc0010070: /* COFVID Control. */
2522 case 0xc001101e:
2523 case 0xc0011021: /* IC_CFG (instruction cache configuration) */
2524 case 0xc0011023: /* CU_CFG (combined unit configuration) */
2525 case 0xc001102c: /* EX_CFG (execution unit configuration) */
2526 return VBCPUREPBADNESS_BOND_VILLAIN;
2527
2528 case 0xc0011012:
2529 if (CPUMMICROARCH_IS_AMD_FAM_0FH(g_enmMicroarch))
2530 return VBCPUREPBADNESS_MIGHT_BITE;
2531 break;
2532
2533 case 0x000001a0: /* IA32_MISC_ENABLE */
2534 case 0x00000199: /* IA32_PERF_CTL */
2535 return VBCPUREPBADNESS_MIGHT_BITE;
2536 case 0x00002000: /* P6_CR0. */
2537 case 0x00002003: /* P6_CR3. */
2538 case 0x00002004: /* P6_CR4. */
2539 if (g_enmVendor == CPUMCPUVENDOR_INTEL)
2540 return VBCPUREPBADNESS_MIGHT_BITE;
2541 break;
2542 case 0xc0000080: /* MSR_K6_EFER */
2543 return VBCPUREPBADNESS_MIGHT_BITE;
2544 }
2545 return VBCPUREPBADNESS_MOSTLY_HARMLESS;
2546}
2547
2548
2549/**
2550 * Checks if this might be a VIA dummy register.
2551 *
2552 * @returns true if it's a dummy, false if it isn't.
2553 * @param uMsr The MSR.
2554 * @param uValue The value.
2555 * @param fFlags The flags.
2556 */
2557static bool isMsrViaDummy(uint32_t uMsr, uint64_t uValue, uint32_t fFlags)
2558{
2559 if (g_enmVendor != CPUMCPUVENDOR_VIA)
2560 return false;
2561
2562 if (uValue)
2563 return false;
2564
2565 if (fFlags)
2566 return false;
2567
2568 switch (uMsr)
2569 {
2570 case 0x00000010:
2571 case 0x0000001b:
2572 case 0x000000c1:
2573 case 0x000000c2:
2574 case 0x0000011e:
2575 case 0x00000186:
2576 case 0x00000187:
2577 //case 0x00000200 ... (mtrrs will be detected)
2578 return false;
2579
2580 case 0xc0000080:
2581 case 0xc0000081:
2582 case 0xc0000082:
2583 case 0xc0000083:
2584 if (vbCpuRepSupportsLongMode())
2585 return false;
2586 break;
2587 }
2588
2589 if (uMsr >= 0x00001200 && uMsr <= 0x00003fff && queryMsrWriteBadness(uMsr) != VBCPUREPBADNESS_MOSTLY_HARMLESS)
2590 return false;
2591
2592 if ( !msrProberModifyNoChange(uMsr)
2593 && !msrProberModifyZero(uMsr))
2594 return false;
2595
2596 uint64_t fIgnMask = 0;
2597 uint64_t fGpMask = 0;
2598 int rc = msrProberModifyBitChanges(uMsr, &fIgnMask, &fGpMask, 0);
2599 if (RT_FAILURE(rc))
2600 return false;
2601
2602 if (fIgnMask != UINT64_MAX)
2603 return false;
2604 if (fGpMask != 0)
2605 return false;
2606
2607 return true;
2608}
2609
2610
2611/**
2612 * Adjusts the ignore and GP masks for MSRs which contains canonical addresses.
2613 *
2614 * @param uMsr The MSR.
2615 * @param pfIgn Pointer to the ignore mask.
2616 * @param pfGp Pointer to the GP mask.
2617 */
2618static void adjustCanonicalIgnAndGpMasks(uint32_t uMsr, uint64_t *pfIgn, uint64_t *pfGp)
2619{
2620 RT_NOREF1(pfIgn);
2621 if (!vbCpuRepSupportsLongMode())
2622 return;
2623 switch (uMsr)
2624 {
2625 case 0x00000175:
2626 case 0x00000176:
2627 case 0x000001da:
2628 case 0x000001db:
2629 case 0x000001dc:
2630 case 0x000001de:
2631 case 0x00000600:
2632 if (*pfGp == UINT64_C(0xffff800000000000))
2633 *pfGp = 0;
2634 break;
2635 case 0x000001dd:
2636 if (*pfGp == UINT64_C(0x7fff800000000000) || *pfGp == UINT64_C(0xffff800000000000)) /* why is the top bit writable? */
2637 *pfGp = 0;
2638 break;
2639
2640 case 0xc0000082:
2641 case 0xc0000083:
2642 case 0xc0000100:
2643 case 0xc0000101:
2644 case 0xc0000102:
2645 *pfGp = 0;
2646 break;
2647 }
2648}
2649
2650
2651
2652/**
2653 * Prints a 64-bit value in the best way.
2654 *
2655 * @param uValue The value.
2656 */
2657static void printMsrValueU64(uint64_t uValue)
2658{
2659 if (uValue == 0)
2660 vbCpuRepPrintf(", 0");
2661 else if (uValue == UINT16_MAX)
2662 vbCpuRepPrintf(", UINT16_MAX");
2663 else if (uValue == UINT32_MAX)
2664 vbCpuRepPrintf(", UINT32_MAX");
2665 else if (uValue == UINT64_MAX)
2666 vbCpuRepPrintf(", UINT64_MAX");
2667 else if (uValue == UINT64_C(0xffffffff00000000))
2668 vbCpuRepPrintf(", ~(uint64_t)UINT32_MAX");
2669 else if (uValue <= (UINT32_MAX >> 1))
2670 vbCpuRepPrintf(", %#llx", uValue);
2671 else if (uValue <= UINT32_MAX)
2672 vbCpuRepPrintf(", UINT32_C(%#llx)", uValue);
2673 else
2674 vbCpuRepPrintf(", UINT64_C(%#llx)", uValue);
2675}
2676
2677
2678/**
2679 * Prints the newline after an MSR line has been printed.
2680 *
2681 * This is used as a hook to slow down the output and make sure the remote
2682 * terminal or/and output file has received the last update before we go and
2683 * crash probing the next MSR.
2684 */
2685static void printMsrNewLine(void)
2686{
2687 vbCpuRepPrintf("\n");
2688#if 1
2689 RTThreadSleep(8);
2690#endif
2691}
2692
2693static int printMsrWriteOnly(uint32_t uMsr, const char *pszWrFnName, const char *pszAnnotation)
2694{
2695 if (!pszWrFnName)
2696 pszWrFnName = "IgnoreWrite";
2697 vbCpuRepPrintf(pszAnnotation
2698 ? " MFN(%#010x, \"%s\", WriteOnly, %s), /* %s */"
2699 : " MFN(%#010x, \"%s\", WriteOnly, %s),",
2700 uMsr, getMsrName(uMsr), pszWrFnName, pszAnnotation);
2701 printMsrNewLine();
2702 return VINF_SUCCESS;
2703}
2704
2705
2706static int printMsrValueReadOnly(uint32_t uMsr, uint64_t uValue, const char *pszAnnotation)
2707{
2708 vbCpuRepPrintf(" MVO(%#010x, \"%s\"", uMsr, getMsrName(uMsr));
2709 printMsrValueU64(uValue);
2710 vbCpuRepPrintf("),");
2711 if (pszAnnotation)
2712 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2713 printMsrNewLine();
2714 return VINF_SUCCESS;
2715}
2716
2717
2718
2719static int printMsrValueIgnoreWritesNamed(uint32_t uMsr, uint64_t uValue, const char *pszName, const char *pszAnnotation)
2720{
2721 vbCpuRepPrintf(" MVI(%#010x, \"%s\"", uMsr, pszName);
2722 printMsrValueU64(uValue);
2723 vbCpuRepPrintf("),");
2724 if (pszAnnotation)
2725 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2726 printMsrNewLine();
2727 return VINF_SUCCESS;
2728}
2729
2730
2731static int printMsrValueIgnoreWrites(uint32_t uMsr, uint64_t uValue, const char *pszAnnotation)
2732{
2733 return printMsrValueIgnoreWritesNamed(uMsr, uValue, getMsrName(uMsr), pszAnnotation);
2734}
2735
2736
2737static int printMsrValueExtended(uint32_t uMsr, uint64_t uValue, uint64_t fIgnMask, uint64_t fGpMask,
2738 const char *pszAnnotation)
2739{
2740 vbCpuRepPrintf(" MVX(%#010x, \"%s\"", uMsr, getMsrName(uMsr));
2741 printMsrValueU64(uValue);
2742 printMsrValueU64(fIgnMask);
2743 printMsrValueU64(fGpMask);
2744 vbCpuRepPrintf("),");
2745 if (pszAnnotation)
2746 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2747 printMsrNewLine();
2748 return VINF_SUCCESS;
2749}
2750
2751
2752static int printMsrRangeValueReadOnly(uint32_t uMsr, uint32_t uLast, uint64_t uValue, const char *pszAnnotation)
2753{
2754 vbCpuRepPrintf(" RVO(%#010x, %#010x, \"%s\"", uMsr, uLast, getMsrRangeName(uMsr));
2755 printMsrValueU64(uValue);
2756 vbCpuRepPrintf("),");
2757 if (pszAnnotation)
2758 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2759 printMsrNewLine();
2760 return VINF_SUCCESS;
2761}
2762
2763
2764static int printMsrRangeValueIgnoreWritesNamed(uint32_t uMsr, uint32_t uLast, uint64_t uValue, const char *pszName, const char *pszAnnotation)
2765{
2766 vbCpuRepPrintf(" RVI(%#010x, %#010x, \"%s\"", uMsr, uLast, pszName);
2767 printMsrValueU64(uValue);
2768 vbCpuRepPrintf("),");
2769 if (pszAnnotation)
2770 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2771 printMsrNewLine();
2772 return VINF_SUCCESS;
2773}
2774
2775
2776static int printMsrRangeValueIgnoreWrites(uint32_t uMsr, uint32_t uLast, uint64_t uValue, const char *pszAnnotation)
2777{
2778 return printMsrRangeValueIgnoreWritesNamed(uMsr, uLast, uValue, getMsrRangeName(uMsr), pszAnnotation);
2779}
2780
2781
2782static int printMsrFunction(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName, const char *pszAnnotation)
2783{
2784 if (!pszRdFnName)
2785 pszRdFnName = getMsrFnName(uMsr, NULL);
2786 if (!pszWrFnName)
2787 pszWrFnName = pszRdFnName;
2788 vbCpuRepPrintf(" MFN(%#010x, \"%s\", %s, %s),", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName);
2789 if (pszAnnotation)
2790 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2791 printMsrNewLine();
2792 return VINF_SUCCESS;
2793}
2794
2795
2796static int printMsrFunctionReadOnly(uint32_t uMsr, const char *pszRdFnName, const char *pszAnnotation)
2797{
2798 if (!pszRdFnName)
2799 pszRdFnName = getMsrFnName(uMsr, NULL);
2800 vbCpuRepPrintf(" MFO(%#010x, \"%s\", %s),", uMsr, getMsrName(uMsr), pszRdFnName);
2801 if (pszAnnotation)
2802 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2803 printMsrNewLine();
2804 return VINF_SUCCESS;
2805}
2806
2807
2808static int printMsrFunctionIgnoreWrites(uint32_t uMsr, const char *pszRdFnName, const char *pszAnnotation)
2809{
2810 if (!pszRdFnName)
2811 pszRdFnName = getMsrFnName(uMsr, NULL);
2812 vbCpuRepPrintf(" MFI(%#010x, \"%s\", %s),", uMsr, getMsrName(uMsr), pszRdFnName);
2813 if (pszAnnotation)
2814 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2815 printMsrNewLine();
2816 return VINF_SUCCESS;
2817}
2818
2819
2820static int printMsrFunctionIgnoreMask(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName,
2821 uint64_t fIgnMask, const char *pszAnnotation)
2822{
2823 if (!pszRdFnName)
2824 pszRdFnName = getMsrFnName(uMsr, NULL);
2825 if (!pszWrFnName)
2826 pszWrFnName = pszRdFnName;
2827 vbCpuRepPrintf(" MFW(%#010x, \"%s\", %s, %s", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName);
2828 printMsrValueU64(fIgnMask);
2829 vbCpuRepPrintf("),");
2830 if (pszAnnotation)
2831 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2832 printMsrNewLine();
2833 return VINF_SUCCESS;
2834}
2835
2836
2837static int printMsrFunctionExtended(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName, uint64_t uValue,
2838 uint64_t fIgnMask, uint64_t fGpMask, const char *pszAnnotation)
2839{
2840 if (!pszRdFnName)
2841 pszRdFnName = getMsrFnName(uMsr, NULL);
2842 if (!pszWrFnName)
2843 pszWrFnName = pszRdFnName;
2844 vbCpuRepPrintf(" MFX(%#010x, \"%s\", %s, %s", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName);
2845 printMsrValueU64(uValue);
2846 printMsrValueU64(fIgnMask);
2847 printMsrValueU64(fGpMask);
2848 vbCpuRepPrintf("),");
2849 if (pszAnnotation)
2850 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2851 printMsrNewLine();
2852 return VINF_SUCCESS;
2853}
2854
2855
2856static int printMsrFunctionExtendedIdxVal(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName, uint64_t uValue,
2857 uint64_t fIgnMask, uint64_t fGpMask, const char *pszAnnotation)
2858{
2859 if (!pszRdFnName)
2860 pszRdFnName = getMsrFnName(uMsr, NULL);
2861 if (!pszWrFnName)
2862 pszWrFnName = pszRdFnName;
2863 vbCpuRepPrintf(" MFX(%#010x, \"%s\", %s, %s, %#x", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName, uValue);
2864 printMsrValueU64(fIgnMask);
2865 printMsrValueU64(fGpMask);
2866 vbCpuRepPrintf("),");
2867 if (pszAnnotation)
2868 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2869 printMsrNewLine();
2870 return VINF_SUCCESS;
2871}
2872
2873
2874static int printMsrFunctionCpumCpu(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName,
2875 const char *pszCpumCpuStorage, const char *pszAnnotation)
2876{
2877 if (!pszRdFnName)
2878 pszRdFnName = getMsrFnName(uMsr, NULL);
2879 if (!pszWrFnName)
2880 pszWrFnName = pszRdFnName;
2881 if (!pszCpumCpuStorage)
2882 pszCpumCpuStorage = getMsrCpumCpuVarName(uMsr);
2883 if (!pszCpumCpuStorage)
2884 return RTMsgErrorRc(VERR_NOT_FOUND, "Missing CPUMCPU member for %#s (%#x)\n", getMsrName(uMsr), uMsr);
2885 vbCpuRepPrintf(" MFS(%#010x, \"%s\", %s, %s, %s),", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName, pszCpumCpuStorage);
2886 if (pszAnnotation)
2887 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2888 printMsrNewLine();
2889 return VINF_SUCCESS;
2890}
2891
2892
2893static int printMsrFunctionCpumCpuEx(uint32_t uMsr, const char *pszRdFnName, const char *pszWrFnName,
2894 const char *pszCpumCpuStorage, uint64_t fIgnMask, uint64_t fGpMask,
2895 const char *pszAnnotation)
2896{
2897 if (!pszRdFnName)
2898 pszRdFnName = getMsrFnName(uMsr, NULL);
2899 if (!pszWrFnName)
2900 pszWrFnName = pszRdFnName;
2901 if (!pszCpumCpuStorage)
2902 pszCpumCpuStorage = getMsrCpumCpuVarName(uMsr);
2903 if (!pszCpumCpuStorage)
2904 return RTMsgErrorRc(VERR_NOT_FOUND, "Missing CPUMCPU member for %#s (%#x)\n", getMsrName(uMsr), uMsr);
2905 vbCpuRepPrintf(" MFZ(%#010x, \"%s\", %s, %s, %s", uMsr, getMsrName(uMsr), pszRdFnName, pszWrFnName, pszCpumCpuStorage);
2906 printMsrValueU64(fIgnMask);
2907 printMsrValueU64(fGpMask);
2908 vbCpuRepPrintf("),");
2909 if (pszAnnotation)
2910 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2911 printMsrNewLine();
2912 return VINF_SUCCESS;
2913}
2914
2915
2916static int printMsrRangeFunction(uint32_t uMsr, uint32_t uLast, const char *pszRdFnName, const char *pszWrFnName,
2917 const char *pszAnnotation)
2918{
2919 if (!pszRdFnName)
2920 pszRdFnName = getMsrFnName(uMsr, NULL);
2921 if (!pszWrFnName)
2922 pszWrFnName = pszRdFnName;
2923 vbCpuRepPrintf(" RFN(%#010x, %#010x, \"%s\", %s, %s),", uMsr, uLast, getMsrRangeName(uMsr), pszRdFnName, pszWrFnName);
2924 if (pszAnnotation)
2925 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2926 printMsrNewLine();
2927 return VINF_SUCCESS;
2928}
2929
2930
2931static int printMsrRangeFunctionEx(uint32_t uMsr, uint32_t uLast, const char *pszRdFnName, const char *pszWrFnName,
2932 uint64_t uValue, uint64_t fIgnMask, uint64_t fGpMask, const char *pszAnnotation)
2933{
2934 if (!pszRdFnName)
2935 pszRdFnName = getMsrFnName(uMsr, NULL);
2936 if (!pszWrFnName)
2937 pszWrFnName = pszRdFnName;
2938 vbCpuRepPrintf(" RSN(%#010x, %#010x, \"%s\", %s, %s", uMsr, uLast, getMsrRangeName(uMsr), pszRdFnName, pszWrFnName);
2939 printMsrValueU64(uValue);
2940 printMsrValueU64(fIgnMask);
2941 printMsrValueU64(fGpMask);
2942 vbCpuRepPrintf("),");
2943 if (pszAnnotation)
2944 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2945 printMsrNewLine();
2946 return VINF_SUCCESS;
2947}
2948
2949
2950static int printMsrRangeFunctionExIdxVal(uint32_t uMsr, uint32_t uLast, const char *pszRdFnName, const char *pszWrFnName,
2951 uint64_t uValue, uint64_t fIgnMask, uint64_t fGpMask, const char *pszAnnotation)
2952{
2953 if (!pszRdFnName)
2954 pszRdFnName = getMsrFnName(uMsr, NULL);
2955 if (!pszWrFnName)
2956 pszWrFnName = pszRdFnName;
2957 vbCpuRepPrintf(" RSN(%#010x, %#010x, \"%s\", %s, %s, %#x",
2958 uMsr, uLast, getMsrRangeName(uMsr), pszRdFnName, pszWrFnName, uValue);
2959 printMsrValueU64(fIgnMask);
2960 printMsrValueU64(fGpMask);
2961 vbCpuRepPrintf("),");
2962 if (pszAnnotation)
2963 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2964 printMsrNewLine();
2965 return VINF_SUCCESS;
2966}
2967
2968
2969static int printMsrAlias(uint32_t uMsr, uint32_t uTarget, const char *pszAnnotation)
2970{
2971 vbCpuRepPrintf(" MAL(%#010x, \"%s\", %#010x),", uMsr, getMsrName(uMsr), uTarget);
2972 if (pszAnnotation)
2973 vbCpuRepPrintf(" /* %s */", pszAnnotation);
2974 printMsrNewLine();
2975 return VINF_SUCCESS;
2976}
2977
2978
2979
2980static const char *annotateValue(uint64_t uValue)
2981{
2982 static char s_szBuf[40];
2983 if (uValue <= UINT32_MAX)
2984 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "value=%#llx", uValue);
2985 else
2986 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "value=%#x`%08x", RT_HI_U32(uValue), RT_LO_U32(uValue));
2987 return s_szBuf;
2988}
2989
2990
2991static const char *annotateValueExtra(const char *pszExtra, uint64_t uValue)
2992{
2993 static char s_szBuf[40];
2994 if (uValue <= UINT32_MAX)
2995 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "%s value=%#llx", pszExtra, uValue);
2996 else
2997 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "%s value=%#x`%08x", pszExtra, RT_HI_U32(uValue), RT_LO_U32(uValue));
2998 return s_szBuf;
2999}
3000
3001
3002static const char *annotateIfMissingBits(uint64_t uValue, uint64_t fBits)
3003{
3004 static char s_szBuf[80];
3005 if ((uValue & fBits) == fBits)
3006 return annotateValue(uValue);
3007 RTStrPrintf(s_szBuf, sizeof(s_szBuf), "XXX: Unexpected value %#llx - wanted bits %#llx to be set.", uValue, fBits);
3008 return s_szBuf;
3009}
3010
3011
3012static int reportMsr_Generic(uint32_t uMsr, uint32_t fFlags, uint64_t uValue)
3013{
3014 int rc;
3015 bool fTakesValue = false;
3016 const char *pszFnName = getMsrFnName(uMsr, &fTakesValue);
3017
3018 if (fFlags & VBCPUREPMSR_F_WRITE_ONLY)
3019 rc = printMsrWriteOnly(uMsr, pszFnName, NULL);
3020 else
3021 {
3022 bool fReadAsZero = doesMsrReadAsZero(uMsr);
3023 fTakesValue = fTakesValue && !fReadAsZero;
3024
3025
3026 switch (queryMsrWriteBadness(uMsr))
3027 {
3028 /* This is what we're here for... */
3029 case VBCPUREPBADNESS_MOSTLY_HARMLESS:
3030 {
3031 if ( msrProberModifyNoChange(uMsr)
3032 || msrProberModifyZero(uMsr))
3033 {
3034 uint64_t fSkipMask = getGenericSkipMask(uMsr);
3035 uint64_t fIgnMask = 0;
3036 uint64_t fGpMask = 0;
3037 rc = msrProberModifyBitChanges(uMsr, &fIgnMask, &fGpMask, fSkipMask);
3038 if (RT_FAILURE(rc))
3039 return rc;
3040 adjustCanonicalIgnAndGpMasks(uMsr, &fIgnMask, &fGpMask);
3041
3042 if (pszFnName)
3043 {
3044 if (fGpMask == 0 && fIgnMask == UINT64_MAX && !fTakesValue)
3045 rc = printMsrFunctionIgnoreWrites(uMsr, pszFnName, annotateValue(uValue));
3046 else if (fGpMask == 0 && fIgnMask == 0 && (!fTakesValue || uValue == 0))
3047 rc = printMsrFunction(uMsr, pszFnName, pszFnName, annotateValue(uValue));
3048 else
3049 rc = printMsrFunctionExtended(uMsr, pszFnName, pszFnName, fTakesValue ? uValue : 0,
3050 fIgnMask, fGpMask, annotateValue(uValue));
3051 }
3052 else if (fGpMask == 0 && fIgnMask == UINT64_MAX)
3053 rc = printMsrValueIgnoreWrites(uMsr, fReadAsZero ? 0 : uValue, fReadAsZero ? annotateValue(uValue) : NULL);
3054 else
3055 rc = printMsrValueExtended(uMsr, fReadAsZero ? 0 : uValue, fIgnMask, fGpMask,
3056 fReadAsZero ? annotateValue(uValue) : NULL);
3057 }
3058 /* Most likely read-only. */
3059 else if (pszFnName && !fTakesValue)
3060 rc = printMsrFunctionReadOnly(uMsr, pszFnName, annotateValue(uValue));
3061 else if (pszFnName)
3062 rc = printMsrFunctionExtended(uMsr, pszFnName, "ReadOnly", uValue, 0, 0, annotateValue(uValue));
3063 else if (fReadAsZero)
3064 rc = printMsrValueReadOnly(uMsr, 0, annotateValue(uValue));
3065 else
3066 rc = printMsrValueReadOnly(uMsr, uValue, NULL);
3067 break;
3068 }
3069
3070 /* These should have special handling, so just do a simple
3071 write back same value check to see if it's writable. */
3072 case VBCPUREPBADNESS_MIGHT_BITE:
3073 if (msrProberModifyNoChange(uMsr))
3074 {
3075 if (pszFnName && !fTakesValue)
3076 rc = printMsrFunction(uMsr, pszFnName, pszFnName, annotateValueExtra("Might bite.", uValue));
3077 else if (pszFnName)
3078 rc = printMsrFunctionExtended(uMsr, pszFnName, pszFnName, uValue, 0, 0,
3079 annotateValueExtra("Might bite.", uValue));
3080 else if (fReadAsZero)
3081 rc = printMsrValueIgnoreWrites(uMsr, 0, annotateValueExtra("Might bite.", uValue));
3082 else
3083 rc = printMsrValueIgnoreWrites(uMsr, uValue, "Might bite.");
3084 }
3085 else if (pszFnName && !fTakesValue)
3086 rc = printMsrFunctionReadOnly(uMsr, pszFnName, annotateValueExtra("Might bite.", uValue));
3087 else if (pszFnName)
3088 rc = printMsrFunctionExtended(uMsr, pszFnName, "ReadOnly", uValue, 0, UINT64_MAX,
3089 annotateValueExtra("Might bite.", uValue));
3090 else if (fReadAsZero)
3091 rc = printMsrValueReadOnly(uMsr, 0, annotateValueExtra("Might bite.", uValue));
3092 else
3093 rc = printMsrValueReadOnly(uMsr, uValue, "Might bite.");
3094 break;
3095
3096
3097 /* Don't try anything with these guys. */
3098 case VBCPUREPBADNESS_BOND_VILLAIN:
3099 default:
3100 if (pszFnName && !fTakesValue)
3101 rc = printMsrFunction(uMsr, pszFnName, pszFnName, annotateValueExtra("Villain?", uValue));
3102 else if (pszFnName)
3103 rc = printMsrFunctionExtended(uMsr, pszFnName, pszFnName, uValue, 0, 0,
3104 annotateValueExtra("Villain?", uValue));
3105 else if (fReadAsZero)
3106 rc = printMsrValueIgnoreWrites(uMsr, 0, annotateValueExtra("Villain?", uValue));
3107 else
3108 rc = printMsrValueIgnoreWrites(uMsr, uValue, "Villain?");
3109 break;
3110 }
3111 }
3112
3113 return rc;
3114}
3115
3116
3117static int reportMsr_GenRangeFunctionEx(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t cMax, const char *pszRdWrFnName,
3118 uint32_t uMsrBase, bool fEarlyEndOk, bool fNoIgnMask, uint64_t fSkipMask, uint32_t *pidxLoop)
3119{
3120 uint32_t uMsr = paMsrs[0].uMsr;
3121 uint32_t iRange = uMsr - uMsrBase;
3122 Assert(cMax > iRange);
3123 cMax -= iRange;
3124
3125 /* Resolve default function name. */
3126 if (!pszRdWrFnName)
3127 {
3128 pszRdWrFnName = getMsrFnName(uMsr, NULL);
3129 if (!pszRdWrFnName)
3130 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "uMsr=%#x no function name\n", uMsr);
3131 }
3132
3133 /* Figure the possible register count. */
3134 if (cMax > cMsrs)
3135 cMax = cMsrs;
3136 uint32_t cRegs = 1;
3137 while ( cRegs < cMax
3138 && paMsrs[cRegs].uMsr == uMsr + cRegs)
3139 cRegs++;
3140
3141 /* Probe the first register and check that the others exhibit
3142 the same characteristics. */
3143 bool fReadOnly0;
3144 uint64_t fIgnMask0, fGpMask0;
3145 int rc = msrProberModifyBasicTests(uMsr, fSkipMask, &fReadOnly0, &fIgnMask0, &fGpMask0);
3146 if (RT_FAILURE(rc))
3147 return rc;
3148
3149 const char *pszAnnotation = NULL;
3150 for (uint32_t i = 1; i < cRegs; i++)
3151 {
3152 bool fReadOnlyN;
3153 uint64_t fIgnMaskN, fGpMaskN;
3154 rc = msrProberModifyBasicTests(paMsrs[i].uMsr, fSkipMask, &fReadOnlyN, &fIgnMaskN, &fGpMaskN);
3155 if (RT_FAILURE(rc))
3156 return rc;
3157 if ( fReadOnlyN != fReadOnly0
3158 || (fIgnMaskN != fIgnMask0 && !fNoIgnMask)
3159 || fGpMaskN != fGpMask0)
3160 {
3161 if (!fEarlyEndOk && !isMsrViaDummy(uMsr, paMsrs[i].uValue, paMsrs[i].fFlags))
3162 {
3163 vbCpuRepDebug("MSR %s (%#x) range ended unexpectedly early on %#x: ro=%d ign=%#llx/%#llx gp=%#llx/%#llx [N/0]\n",
3164 getMsrNameHandled(uMsr), uMsr, paMsrs[i].uMsr,
3165 fReadOnlyN, fReadOnly0, fIgnMaskN, fIgnMask0, fGpMaskN, fGpMask0);
3166 pszAnnotation = "XXX: The range ended earlier than expected!";
3167 }
3168 cRegs = i;
3169 break;
3170 }
3171 }
3172
3173 /*
3174 * Report the range (or single MSR as it might be).
3175 */
3176 *pidxLoop += cRegs - 1;
3177
3178 if (fNoIgnMask)
3179 fIgnMask0 = 0;
3180 bool fSimple = fIgnMask0 == 0
3181 && (fGpMask0 == 0 || (fGpMask0 == UINT64_MAX && fReadOnly0))
3182 && iRange == 0;
3183 if (cRegs == 1)
3184 return printMsrFunctionExtendedIdxVal(uMsr, pszRdWrFnName, fReadOnly0 ? "ReadOnly" : pszRdWrFnName,
3185 iRange, fIgnMask0, fGpMask0,
3186 pszAnnotation ? pszAnnotation : annotateValue(paMsrs[0].uValue));
3187 if (fSimple)
3188 return printMsrRangeFunction(uMsr, uMsr + cRegs - 1,
3189 pszRdWrFnName, fReadOnly0 ? "ReadOnly" : pszRdWrFnName, pszAnnotation);
3190
3191 return printMsrRangeFunctionExIdxVal(uMsr, uMsr + cRegs - 1, pszRdWrFnName, fReadOnly0 ? "ReadOnly" : pszRdWrFnName,
3192 iRange /*uValue*/, fIgnMask0, fGpMask0, pszAnnotation);
3193}
3194
3195
3196static int reportMsr_GenRangeFunction(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t cMax, const char *pszRdWrFnName,
3197 uint32_t *pidxLoop)
3198{
3199 return reportMsr_GenRangeFunctionEx(paMsrs, cMsrs, cMax, pszRdWrFnName, paMsrs[0].uMsr, false /*fEarlyEndOk*/, false /*fNoIgnMask*/,
3200 getGenericSkipMask(paMsrs[0].uMsr), pidxLoop);
3201}
3202
3203
3204/**
3205 * Generic report for an MSR implemented by functions, extended version.
3206 *
3207 * @returns VBox status code.
3208 * @param uMsr The MSR.
3209 * @param pszRdWrFnName The read/write function name, optional.
3210 * @param uValue The MSR range value.
3211 * @param fSkipMask Mask of bits to skip.
3212 * @param fNoGpMask Mask of bits to remove from the GP mask after
3213 * probing
3214 * @param pszAnnotate Annotation.
3215 */
3216static int reportMsr_GenFunctionEx(uint32_t uMsr, const char *pszRdWrFnName, uint32_t uValue,
3217 uint64_t fSkipMask, uint64_t fNoGpMask, const char *pszAnnotate)
3218{
3219 /* Resolve default function name. */
3220 if (!pszRdWrFnName)
3221 {
3222 pszRdWrFnName = getMsrFnName(uMsr, NULL);
3223 if (!pszRdWrFnName)
3224 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "uMsr=%#x no function name\n", uMsr);
3225 }
3226
3227 /* Probe the register and report. */
3228 uint64_t fIgnMask = 0;
3229 uint64_t fGpMask = 0;
3230 int rc = msrProberModifyBitChanges(uMsr, &fIgnMask, &fGpMask, fSkipMask);
3231 if (RT_SUCCESS(rc))
3232 {
3233 fGpMask &= ~fNoGpMask;
3234
3235 if (fGpMask == UINT64_MAX && uValue == 0 && !msrProberModifyZero(uMsr))
3236 rc = printMsrFunctionReadOnly(uMsr, pszRdWrFnName, pszAnnotate);
3237 else if (fIgnMask == UINT64_MAX && fGpMask == 0 && uValue == 0)
3238 rc = printMsrFunctionIgnoreWrites(uMsr, pszRdWrFnName, pszAnnotate);
3239 else if (fIgnMask != 0 && fGpMask == 0 && uValue == 0)
3240 rc = printMsrFunctionIgnoreMask(uMsr, pszRdWrFnName, NULL, fIgnMask, pszAnnotate);
3241 else if (fIgnMask == 0 && fGpMask == 0 && uValue == 0)
3242 rc = printMsrFunction(uMsr, pszRdWrFnName, NULL, pszAnnotate);
3243 else
3244 rc = printMsrFunctionExtended(uMsr, pszRdWrFnName, NULL, uValue, fIgnMask, fGpMask, pszAnnotate);
3245 }
3246 return rc;
3247}
3248
3249
3250/**
3251 * Reports a VIA dummy range.
3252 *
3253 * @returns VBox status code.
3254 * @param paMsrs Pointer to the first MSR.
3255 * @param cMsrs The number of MSRs in the array @a paMsr.
3256 * @param pidxLoop Index variable that should be advanced to the
3257 * last MSR entry in the range.
3258 */
3259static int reportMsr_ViaDummyRange(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3260{
3261 /* Figure how many. */
3262 uint32_t uMsr = paMsrs[0].uMsr;
3263 uint32_t cRegs = 1;
3264 while ( cRegs < cMsrs
3265 && paMsrs[cRegs].uMsr == uMsr + cRegs
3266 && isMsrViaDummy(paMsrs[cRegs].uMsr, paMsrs[cRegs].uValue, paMsrs[cRegs].fFlags))
3267 {
3268 cRegs++;
3269 if (!(cRegs % 0x80))
3270 vbCpuRepDebug("VIA dummy detection %#llx..%#llx (%#x regs)...\n", uMsr, uMsr + cRegs - 1, cRegs);
3271 }
3272
3273 /* Advance. */
3274 *pidxLoop += cRegs - 1;
3275
3276 /* Report it/them. */
3277 char szName[80];
3278 if (cRegs == 1)
3279 {
3280 RTStrPrintf(szName, sizeof(szName), "ZERO_%04x_%04x", RT_HI_U16(uMsr), RT_LO_U16(uMsr));
3281 return printMsrValueIgnoreWritesNamed(uMsr, 0, szName, NULL);
3282 }
3283
3284 uint32_t uMsrLast = uMsr + cRegs - 1;
3285 RTStrPrintf(szName, sizeof(szName), "ZERO_%04x_%04x_THRU_%04x_%04x",
3286 RT_HI_U16(uMsr), RT_LO_U16(uMsr), RT_HI_U16(uMsrLast), RT_LO_U16(uMsrLast));
3287 return printMsrRangeValueIgnoreWritesNamed(uMsr, uMsrLast, 0, szName, NULL);
3288}
3289
3290
3291/**
3292 * Special function for reporting the IA32_APIC_BASE register, as it seems to be
3293 * causing trouble on newer systems.
3294 *
3295 * @returns
3296 * @param uMsr The MSR number.
3297 * @param uValue The value.
3298 */
3299static int reportMsr_Ia32ApicBase(uint32_t uMsr, uint64_t uValue)
3300{
3301 /* Trouble with the generic treatment of both the "APIC Global Enable" and
3302 "Enable x2APIC mode" bits on an i7-3820QM running OS X 10.8.5. */
3303 uint64_t fSkipMask = RT_BIT_64(11);
3304 if (vbCpuRepSupportsX2Apic())
3305 fSkipMask |= RT_BIT_64(10);
3306 /* For some reason, twiddling this bit kills a Tualatin PIII-S. */
3307 if (g_enmMicroarch == kCpumMicroarch_Intel_P6_III)
3308 fSkipMask |= RT_BIT(9);
3309 return reportMsr_GenFunctionEx(uMsr, "Ia32ApicBase", uValue, fSkipMask, 0, NULL);
3310}
3311
3312
3313/**
3314 * Special function for reporting the IA32_MISC_ENABLE register, as it seems to
3315 * be causing trouble on newer systems.
3316 *
3317 * @returns
3318 * @param uMsr The MSR number.
3319 * @param uValue The value.
3320 */
3321static int reportMsr_Ia32MiscEnable(uint32_t uMsr, uint64_t uValue)
3322{
3323 uint64_t fSkipMask = 0;
3324
3325 if ( ( g_enmMicroarch >= kCpumMicroarch_Intel_Core7_Broadwell
3326 && g_enmMicroarch <= kCpumMicroarch_Intel_Core7_End)
3327 || ( g_enmMicroarch >= kCpumMicroarch_Intel_Atom_Airmount
3328 && g_enmMicroarch <= kCpumMicroarch_Intel_Atom_End)
3329 )
3330 {
3331 vbCpuRepPrintf("WARNING: IA32_MISC_ENABLE probing needs hacking on this CPU!\n");
3332 RTThreadSleep(128);
3333 }
3334
3335 /* The no execute related flag is deadly if clear. */
3336 if ( !(uValue & MSR_IA32_MISC_ENABLE_XD_DISABLE)
3337 && ( g_enmMicroarch < kCpumMicroarch_Intel_First
3338 || g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah
3339 || vbCpuRepSupportsNX() ) )
3340 fSkipMask |= MSR_IA32_MISC_ENABLE_XD_DISABLE;
3341
3342 uint64_t fIgnMask = 0;
3343 uint64_t fGpMask = 0;
3344 int rc = msrProberModifyBitChanges(uMsr, &fIgnMask, &fGpMask, fSkipMask);
3345 if (RT_SUCCESS(rc))
3346 rc = printMsrFunctionExtended(uMsr, "Ia32MiscEnable", "Ia32MiscEnable", uValue,
3347 fIgnMask, fGpMask, annotateValue(uValue));
3348 return rc;
3349}
3350
3351
3352/**
3353 * Verifies that MTRR type field works correctly in the given MSR.
3354 *
3355 * @returns VBox status code (failure if bad MSR behavior).
3356 * @param uMsr The MSR.
3357 * @param iBit The first bit of the type field (8-bit wide).
3358 * @param cExpected The number of types expected - PAT=8, MTRR=7.
3359 */
3360static int msrVerifyMtrrTypeGPs(uint32_t uMsr, uint32_t iBit, uint32_t cExpected)
3361{
3362 uint32_t uEndTypes = 0;
3363 while (uEndTypes < 255)
3364 {
3365 bool fGp = !msrProberModifySimpleGp(uMsr, ~(UINT64_C(0xff) << iBit), (uint64_t)uEndTypes << iBit);
3366 if (!fGp && (uEndTypes == 2 || uEndTypes == 3))
3367 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "MTRR types %u does not cause a GP as it should. (msr %#x)\n",
3368 uEndTypes, uMsr);
3369 if (fGp && uEndTypes != 2 && uEndTypes != 3)
3370 break;
3371 uEndTypes++;
3372 }
3373 if (uEndTypes != cExpected)
3374 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "MTRR types detected to be %#x (msr %#x). Expected %#x.\n",
3375 uEndTypes, uMsr, cExpected);
3376 return VINF_SUCCESS;
3377}
3378
3379
3380/**
3381 * Deals with the variable MTRR MSRs.
3382 *
3383 * @returns VBox status code.
3384 * @param paMsrs Pointer to the first variable MTRR MSR (200h).
3385 * @param cMsrs The number of MSRs in the array @a paMsr.
3386 * @param pidxLoop Index variable that should be advanced to the
3387 * last MTRR MSR entry.
3388 */
3389static int reportMsr_Ia32MtrrPhysBaseMaskN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3390{
3391 uint32_t uMsr = paMsrs[0].uMsr;
3392
3393 /* Count them. */
3394 uint32_t cRegs = 1;
3395 while ( cRegs < cMsrs
3396 && paMsrs[cRegs].uMsr == uMsr + cRegs
3397 && !isMsrViaDummy(paMsrs[cRegs].uMsr, paMsrs[cRegs].uValue, paMsrs[cRegs].fFlags) )
3398 cRegs++;
3399 if (cRegs & 1)
3400 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "MTRR variable MSR range is odd: cRegs=%#x\n", cRegs);
3401 if (cRegs > 0x20)
3402 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "MTRR variable MSR range is too large: cRegs=%#x\n", cRegs);
3403
3404 /* Find a disabled register that we can play around with. */
3405 uint32_t iGuineaPig;
3406 for (iGuineaPig = 0; iGuineaPig < cRegs; iGuineaPig += 2)
3407 if (!(paMsrs[iGuineaPig + 1].uValue & RT_BIT_32(11)))
3408 break;
3409 if (iGuineaPig >= cRegs)
3410 iGuineaPig = cRegs - 2;
3411 vbCpuRepDebug("iGuineaPig=%#x -> %#x\n", iGuineaPig, uMsr + iGuineaPig);
3412
3413 /* Probe the base. */
3414 uint64_t fIgnBase = 0;
3415 uint64_t fGpBase = 0;
3416 int rc = msrProberModifyBitChanges(uMsr + iGuineaPig, &fIgnBase, &fGpBase, 0);
3417 if (RT_FAILURE(rc))
3418 return rc;
3419 rc = msrVerifyMtrrTypeGPs(uMsr + iGuineaPig, 0, 7);
3420 if (RT_FAILURE(rc))
3421 return rc;
3422 vbCpuRepDebug("fIgnBase=%#llx fGpBase=%#llx\n", fIgnBase, fGpBase);
3423
3424 /* Probing the mask is relatively straight forward. */
3425 uint64_t fIgnMask = 0;
3426 uint64_t fGpMask = 0;
3427 rc = msrProberModifyBitChanges(uMsr + iGuineaPig + 1, &fIgnMask, &fGpMask, 0x800); /* enabling it may cause trouble */
3428 if (RT_FAILURE(rc))
3429 return rc;
3430 vbCpuRepDebug("fIgnMask=%#llx fGpMask=%#llx\n", fIgnMask, fGpMask);
3431
3432 /* Validate that the whole range subscribes to the apprimately same GP rules. */
3433 for (uint32_t i = 0; i < cRegs; i += 2)
3434 {
3435 uint64_t fSkipBase = ~fGpBase;
3436 uint64_t fSkipMask = ~fGpMask;
3437 if (!(paMsrs[i + 1].uValue & RT_BIT_32(11)))
3438 fSkipBase = fSkipMask = 0;
3439 fSkipBase |= 0x7; /* Always skip the type. */
3440 fSkipMask |= RT_BIT_32(11); /* Always skip the enable bit. */
3441
3442 vbCpuRepDebug("i=%#x fSkipBase=%#llx fSkipMask=%#llx\n", i, fSkipBase, fSkipMask);
3443
3444 if (!(paMsrs[i + 1].uValue & RT_BIT_32(11)))
3445 {
3446 rc = msrVerifyMtrrTypeGPs(uMsr + iGuineaPig, 0, 7);
3447 if (RT_FAILURE(rc))
3448 return rc;
3449 }
3450
3451 uint64_t fIgnBaseN = 0;
3452 uint64_t fGpBaseN = 0;
3453 rc = msrProberModifyBitChanges(uMsr + i, &fIgnBaseN, &fGpBaseN, fSkipBase);
3454 if (RT_FAILURE(rc))
3455 return rc;
3456
3457 if ( fIgnBaseN != (fIgnBase & ~fSkipBase)
3458 || fGpBaseN != (fGpBase & ~fSkipBase) )
3459 return RTMsgErrorRc(VERR_INVALID_PARAMETER,
3460 "MTRR PHYS BASE register %#x behaves differently from %#x: ign=%#llx/%#llx gp=%#llx/%#llx (fSkipBase=%#llx)\n",
3461 uMsr + i, uMsr + iGuineaPig,
3462 fIgnBaseN, fIgnBase & ~fSkipBase, fGpBaseN, fGpBase & ~fSkipBase, fSkipBase);
3463
3464 uint64_t fIgnMaskN = 0;
3465 uint64_t fGpMaskN = 0;
3466 rc = msrProberModifyBitChanges(uMsr + i + 1, &fIgnMaskN, &fGpMaskN, fSkipMask);
3467 if (RT_FAILURE(rc))
3468 return rc;
3469 if ( fIgnMaskN != (fIgnMask & ~fSkipMask)
3470 || fGpMaskN != (fGpMask & ~fSkipMask) )
3471 return RTMsgErrorRc(VERR_INVALID_PARAMETER,
3472 "MTRR PHYS MASK register %#x behaves differently from %#x: ign=%#llx/%#llx gp=%#llx/%#llx (fSkipMask=%#llx)\n",
3473 uMsr + i + 1, uMsr + iGuineaPig + 1,
3474 fIgnMaskN, fIgnMask & ~fSkipMask, fGpMaskN, fGpMask & ~fSkipMask, fSkipMask);
3475 }
3476
3477 /* Print the whole range. */
3478 fGpBase &= ~(uint64_t)0x7; /* Valid type bits, see msrVerifyMtrrTypeGPs(). */
3479 for (uint32_t i = 0; i < cRegs; i += 2)
3480 {
3481 printMsrFunctionExtendedIdxVal(uMsr + i, "Ia32MtrrPhysBaseN", NULL, i / 2, fIgnBase, fGpBase,
3482 annotateValue(paMsrs[i].uValue));
3483 printMsrFunctionExtendedIdxVal(uMsr + i + 1, "Ia32MtrrPhysMaskN", NULL, i / 2, fIgnMask, fGpMask,
3484 annotateValue(paMsrs[i + 1].uValue));
3485 }
3486
3487 *pidxLoop += cRegs - 1;
3488 return VINF_SUCCESS;
3489}
3490
3491
3492/**
3493 * Deals with fixed MTRR and PAT MSRs, checking the 8 memory type fields.
3494 *
3495 * @returns VBox status code.
3496 * @param uMsr The MSR.
3497 */
3498static int reportMsr_Ia32MtrrFixedOrPat(uint32_t uMsr)
3499{
3500 /* Had a spot of trouble on an old macbook pro with core2 duo T9900 (penryn)
3501 running 64-bit win81pe. Not giving PAT such a scrutiny fixes it. */
3502 if ( uMsr != 0x00000277
3503 || ( g_enmVendor == CPUMCPUVENDOR_INTEL
3504 ? g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First
3505 : g_enmVendor == CPUMCPUVENDOR_AMD
3506 ? g_enmMicroarch != kCpumMicroarch_AMD_K8_90nm_AMDV
3507 : true) )
3508 {
3509 /* Every 8 bytes is a type, check the type ranges one by one. */
3510 for (uint32_t iBit = 0; iBit < 64; iBit += 8)
3511 {
3512 int rc = msrVerifyMtrrTypeGPs(uMsr, iBit, 7 + (uMsr == 0x00000277));
3513 if (RT_FAILURE(rc))
3514 return rc;
3515 }
3516 }
3517
3518 return printMsrFunctionCpumCpu(uMsr, NULL, NULL, NULL, NULL);
3519}
3520
3521
3522/**
3523 * Deals with IA32_MTRR_DEF_TYPE.
3524 *
3525 * @returns VBox status code.
3526 * @param uMsr The MSR.
3527 */
3528static int reportMsr_Ia32MtrrDefType(uint32_t uMsr)
3529{
3530 uint64_t fGpMask = 0;
3531 uint64_t fIgnMask = 0;
3532 if (g_enmMicroarch == kCpumMicroarch_AMD_K8_90nm_AMDV)
3533 {
3534 /* Problematic CPU! Fake it for now. */
3535 fGpMask = ~(uint64_t)0xc07;
3536 fIgnMask = 0;
3537 }
3538 else
3539 {
3540 int rc = msrVerifyMtrrTypeGPs(uMsr, 0, 7);
3541 if (RT_FAILURE(rc))
3542 return rc;
3543
3544 rc = msrProberModifyBitChanges(uMsr, &fIgnMask, &fGpMask, 0x7);
3545 if (RT_FAILURE(rc))
3546 return rc;
3547 Assert(!(fGpMask & 7)); Assert(!(fIgnMask & 7));
3548 }
3549
3550 return printMsrFunctionCpumCpuEx(uMsr, NULL, NULL, NULL, fIgnMask, fGpMask, NULL);
3551}
3552
3553
3554/**
3555 * Deals with the Machine Check (MC) MSRs in the 400h+ area.
3556 *
3557 * @returns VBox status code.
3558 * @param paMsrs Pointer to the first MC MSR (400h).
3559 * @param cMsrs The number of MSRs in the array @a paMsr.
3560 * @param pidxLoop Index variable that should be advanced to the
3561 * last MC MSR entry.
3562 */
3563static int reportMsr_Ia32McCtlStatusAddrMiscN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3564{
3565 uint32_t uMsr = paMsrs[0].uMsr;
3566
3567 /* Count them. */
3568 uint32_t cRegs = 1;
3569 uint32_t cDetectedRegs = 1;
3570 while ( cDetectedRegs < cMsrs
3571 && ( paMsrs[cDetectedRegs].uMsr == uMsr + cRegs
3572 || (cRegs & 3) == 2 /* ADDR may or may not be there, depends on STATUS and CPU. */
3573 || (cRegs & 3) == 3 /* MISC may or may not be there, depends on STATUS and CPU. */
3574 || cRegs == 0x13 /* MC4_MISC may not be there, depends on CPU. */
3575 || cRegs == 0x14 /* MC5_CTL may not be there, depends on CPU. */)
3576 && cRegs < 0x7f )
3577 {
3578 if (paMsrs[cDetectedRegs].uMsr == uMsr + cRegs)
3579 cDetectedRegs++;
3580 cRegs++;
3581 }
3582 if (cRegs & 3)
3583 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "MC MSR range is odd: cRegs=%#x\n", cRegs);
3584
3585 /* Just report them. We don't bother probing here as the CTL format
3586 and such seems to be a lot of work to test correctly and changes between
3587 cpu generations. */
3588 *pidxLoop += cDetectedRegs - 1;
3589 return printMsrRangeFunction(uMsr, uMsr + cRegs - 1, "Ia32McCtlStatusAddrMiscN", NULL, NULL);
3590}
3591
3592
3593
3594/**
3595 * Deals with the X2APIC msrs.
3596 *
3597 * @returns VBox status code.
3598 * @param paMsrs Pointer to the first X2APIC MSR.
3599 * @param cMsrs The number of MSRs in the array @a paMsr.
3600 * @param pidxLoop Index variable that should be advanced to the
3601 * last X2APIC MSR entry.
3602 */
3603static int reportMsr_GenX2Apic(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3604{
3605 /* Advance. */
3606 uint32_t cRegs = 1;
3607 while ( cRegs < cMsrs
3608 && paMsrs[cRegs].uMsr <= 0x8ff)
3609 cRegs++;
3610 *pidxLoop += cRegs - 1;
3611
3612 /* Just emit an X2APIC range. */
3613 return printMsrRangeFunction(0x800, 0x8ff, "Ia32X2ApicN", NULL, NULL);
3614}
3615
3616
3617/**
3618 * Deals carefully with the EFER register.
3619 *
3620 * @returns VBox status code.
3621 * @param uMsr The MSR number.
3622 * @param uValue The current value.
3623 */
3624static int reportMsr_Amd64Efer(uint32_t uMsr, uint64_t uValue)
3625{
3626 uint64_t fSkipMask = 0;
3627 if (vbCpuRepSupportsLongMode())
3628 fSkipMask |= MSR_K6_EFER_LME;
3629 if ( (uValue & MSR_K6_EFER_NXE)
3630 || vbCpuRepSupportsNX())
3631 fSkipMask |= MSR_K6_EFER_NXE;
3632
3633 /* NetBurst prescott 2MB (model 4) hung or triple faulted here. The extra
3634 sleep or something seemed to help for some screwed up reason. */
3635 if (g_fIntelNetBurst)
3636 {
3637 // This doesn't matter:
3638 //fSkipMask |= MSR_K6_EFER_SCE;
3639 //if (vbCpuRepSupportsLongMode())
3640 // fSkipMask |= MSR_K6_EFER_LMA;
3641 //vbCpuRepDebug("EFER - netburst workaround - ignore SCE & LMA (fSkipMask=%#llx)\n", fSkipMask);
3642
3643 vbCpuRepDebug("EFER - netburst sleep fudge - fSkipMask=%#llx\n", fSkipMask);
3644 RTThreadSleep(1000);
3645 }
3646
3647 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, MSR_K6_EFER_LMA, NULL);
3648}
3649
3650
3651/**
3652 * Deals with the MC4_MISCn (n >= 1) range and the following reserved MSRs.
3653 *
3654 * @returns VBox status code.
3655 * @param paMsrs Pointer to the first MSR.
3656 * @param cMsrs The number of MSRs in the array @a paMsr.
3657 * @param pidxLoop Index variable that should be advanced to the
3658 * last MSR entry in the range.
3659 */
3660static int reportMsr_AmdFam10hMc4MiscN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3661{
3662 /* Count registers. */
3663 uint32_t cRegs = 1;
3664 while ( cRegs < cMsrs
3665 && cRegs < 8
3666 && paMsrs[cRegs].uMsr == paMsrs[0].uMsr + cRegs)
3667 cRegs++;
3668
3669 /* Probe & report used MSRs. */
3670 uint64_t fIgnMask = 0;
3671 uint64_t fGpMask = 0;
3672 uint32_t cUsed = 0;
3673 while (cUsed < cRegs)
3674 {
3675 uint64_t fIgnMaskN = 0;
3676 uint64_t fGpMaskN = 0;
3677 int rc = msrProberModifyBitChanges(paMsrs[cUsed].uMsr, &fIgnMaskN, &fGpMaskN, 0);
3678 if (RT_FAILURE(rc))
3679 return rc;
3680 if (fIgnMaskN == UINT64_MAX || fGpMaskN == UINT64_MAX)
3681 break;
3682 if (cUsed == 0)
3683 {
3684 fIgnMask = fIgnMaskN;
3685 fGpMask = fGpMaskN;
3686 }
3687 else if ( fIgnMaskN != fIgnMask
3688 || fGpMaskN != fGpMask)
3689 return RTMsgErrorRc(VERR_NOT_EQUAL, "AmdFam16hMc4MiscN mismatch: fIgn=%#llx/%#llx fGp=%#llx/%#llx uMsr=%#x\n",
3690 fIgnMaskN, fIgnMask, fGpMaskN, fGpMask, paMsrs[cUsed].uMsr);
3691 cUsed++;
3692 }
3693 if (cUsed > 0)
3694 printMsrRangeFunctionEx(paMsrs[0].uMsr, paMsrs[cUsed - 1].uMsr, "AmdFam10hMc4MiscN", NULL, 0, fIgnMask, fGpMask, NULL);
3695
3696 /* Probe & report reserved MSRs. */
3697 uint32_t cReserved = 0;
3698 while (cUsed + cReserved < cRegs)
3699 {
3700 fIgnMask = fGpMask = 0;
3701 int rc = msrProberModifyBitChanges(paMsrs[cUsed + cReserved].uMsr, &fIgnMask, &fGpMask, 0);
3702 if (RT_FAILURE(rc))
3703 return rc;
3704 if ((fIgnMask != UINT64_MAX && fGpMask != UINT64_MAX) || paMsrs[cUsed + cReserved].uValue)
3705 return RTMsgErrorRc(VERR_NOT_EQUAL,
3706 "Unexpected reserved AmdFam16hMc4MiscN: fIgn=%#llx fGp=%#llx uMsr=%#x uValue=%#llx\n",
3707 fIgnMask, fGpMask, paMsrs[cUsed + cReserved].uMsr, paMsrs[cUsed + cReserved].uValue);
3708 cReserved++;
3709 }
3710 if (cReserved > 0 && fIgnMask == UINT64_MAX)
3711 printMsrRangeValueIgnoreWrites(paMsrs[cUsed].uMsr, paMsrs[cUsed + cReserved - 1].uMsr, 0, NULL);
3712 else if (cReserved > 0 && fGpMask == UINT64_MAX)
3713 printMsrRangeValueReadOnly(paMsrs[cUsed].uMsr, paMsrs[cUsed + cReserved - 1].uMsr, 0, NULL);
3714
3715 *pidxLoop += cRegs - 1;
3716 return VINF_SUCCESS;
3717}
3718
3719
3720/**
3721 * Deals with the AMD PERF_CTL range.
3722 *
3723 * @returns VBox status code.
3724 * @param paMsrs Pointer to the first MSR.
3725 * @param cMsrs The number of MSRs in the array @a paMsr.
3726 * @param pidxLoop Index variable that should be advanced to the
3727 * last MSR entry in the range.
3728 */
3729static int reportMsr_AmdK8PerfCtlN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3730{
3731 uint32_t uMsr = paMsrs[0].uMsr;
3732 Assert(uMsr == 0xc0010000);
3733
3734 /* Family 15h (bulldozer +) aliases these registers sparsely onto c001020x. */
3735 if (CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
3736 {
3737 for (uint32_t i = 0; i < 4; i++)
3738 printMsrAlias(uMsr + i, 0xc0010200 + i * 2, NULL);
3739 *pidxLoop += 3;
3740 }
3741 else
3742 return reportMsr_GenRangeFunction(paMsrs, cMsrs, 4, "AmdK8PerfCtlN", pidxLoop);
3743 return VINF_SUCCESS;
3744}
3745
3746
3747/**
3748 * Deals with the AMD PERF_CTR range.
3749 *
3750 * @returns VBox status code.
3751 * @param paMsrs Pointer to the first MSR.
3752 * @param cMsrs The number of MSRs in the array @a paMsr.
3753 * @param pidxLoop Index variable that should be advanced to the
3754 * last MSR entry in the range.
3755 */
3756static int reportMsr_AmdK8PerfCtrN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3757{
3758 uint32_t uMsr = paMsrs[0].uMsr;
3759 Assert(uMsr == 0xc0010004);
3760
3761 /* Family 15h (bulldozer +) aliases these registers sparsely onto c001020x. */
3762 if (CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
3763 {
3764 for (uint32_t i = 0; i < 4; i++)
3765 printMsrAlias(uMsr + i, 0xc0010201 + i * 2, NULL);
3766 *pidxLoop += 3;
3767 }
3768 else
3769 return reportMsr_GenRangeFunction(paMsrs, cMsrs, 4, "AmdK8PerfCtrN", pidxLoop);
3770 return VINF_SUCCESS;
3771}
3772
3773
3774/**
3775 * Deals carefully with the SYS_CFG register.
3776 *
3777 * @returns VBox status code.
3778 * @param uMsr The MSR number.
3779 * @param uValue The current value.
3780 */
3781static int reportMsr_AmdK8SysCfg(uint32_t uMsr, uint64_t uValue)
3782{
3783 uint64_t fSkipMask = 0;
3784
3785 /* Bit 21 (MtrrTom2En) is marked reserved in family 0fh, while in family
3786 10h BKDG this changes (as does the document style). Testing this bit
3787 causes bulldozer running win64 to restart, thus this special treatment. */
3788 if (g_enmMicroarch >= kCpumMicroarch_AMD_K10)
3789 fSkipMask |= RT_BIT(21);
3790
3791 /* Turns out there are more killer bits here, at least on Opteron 2384.
3792 Skipping all known bits. */
3793 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_90nm_AMDV /* Not sure when introduced - harmless? */)
3794 fSkipMask |= RT_BIT(22); /* Tom2ForceMemTypeWB */
3795 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3796 fSkipMask |= RT_BIT(21); /* MtrrTom2En */
3797 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3798 fSkipMask |= RT_BIT(20); /* MtrrVarDramEn*/
3799 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3800 fSkipMask |= RT_BIT(19); /* MtrrFixDramModEn */
3801 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3802 fSkipMask |= RT_BIT(18); /* MtrrFixDramEn */
3803 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3804 fSkipMask |= RT_BIT(17); /* SysUcLockEn */
3805 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
3806 fSkipMask |= RT_BIT(16); /* ChgToDirtyDis */
3807 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First && g_enmMicroarch < kCpumMicroarch_AMD_15h_First)
3808 fSkipMask |= RT_BIT(10); /* SetDirtyEnO */
3809 if (g_enmMicroarch >= kCpumMicroarch_AMD_K8_First && g_enmMicroarch < kCpumMicroarch_AMD_15h_First)
3810 fSkipMask |= RT_BIT(9); /* SetDirtyEnS */
3811 if ( CPUMMICROARCH_IS_AMD_FAM_0FH(g_enmMicroarch)
3812 || CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch))
3813 fSkipMask |= RT_BIT(8); /* SetDirtyEnE */
3814 if ( CPUMMICROARCH_IS_AMD_FAM_0FH(g_enmMicroarch)
3815 || CPUMMICROARCH_IS_AMD_FAM_11H(g_enmMicroarch) )
3816 fSkipMask |= RT_BIT(7) /* SysVicLimit */
3817 | RT_BIT(6) /* SysVicLimit */
3818 | RT_BIT(5) /* SysVicLimit */
3819 | RT_BIT(4) /* SysAckLimit */
3820 | RT_BIT(3) /* SysAckLimit */
3821 | RT_BIT(2) /* SysAckLimit */
3822 | RT_BIT(1) /* SysAckLimit */
3823 | RT_BIT(0) /* SysAckLimit */;
3824
3825 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
3826}
3827
3828
3829/**
3830 * Deals carefully with the HWCR register.
3831 *
3832 * @returns VBox status code.
3833 * @param uMsr The MSR number.
3834 * @param uValue The current value.
3835 */
3836static int reportMsr_AmdK8HwCr(uint32_t uMsr, uint64_t uValue)
3837{
3838 uint64_t fSkipMask = 0;
3839
3840 /* Trouble on Opteron 2384, skip some of the known bits. */
3841 if (g_enmMicroarch >= kCpumMicroarch_AMD_K10 && !CPUMMICROARCH_IS_AMD_FAM_11H(g_enmMicroarch))
3842 fSkipMask |= /*RT_BIT(10)*/ 0 /* MonMwaitUserEn */
3843 | RT_BIT(9); /* MonMwaitDis */
3844 fSkipMask |= RT_BIT(8); /* #IGNNE port emulation */
3845 if ( CPUMMICROARCH_IS_AMD_FAM_0FH(g_enmMicroarch)
3846 || CPUMMICROARCH_IS_AMD_FAM_11H(g_enmMicroarch) )
3847 fSkipMask |= RT_BIT(7) /* DisLock */
3848 | RT_BIT(6); /* FFDis (TLB flush filter) */
3849 fSkipMask |= RT_BIT(4); /* INVD to WBINVD */
3850 fSkipMask |= RT_BIT(3); /* TLBCACHEDIS */
3851 if ( CPUMMICROARCH_IS_AMD_FAM_0FH(g_enmMicroarch)
3852 || CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch)
3853 || CPUMMICROARCH_IS_AMD_FAM_11H(g_enmMicroarch) )
3854 fSkipMask |= RT_BIT(1); /* SLOWFENCE */
3855 fSkipMask |= RT_BIT(0); /* SMMLOCK */
3856
3857 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
3858}
3859
3860
3861/**
3862 * Deals carefully with a IORRBasei register.
3863 *
3864 * @returns VBox status code.
3865 * @param uMsr The MSR number.
3866 * @param uValue The current value.
3867 */
3868static int reportMsr_AmdK8IorrBaseN(uint32_t uMsr, uint64_t uValue)
3869{
3870 /* Skip know bits here, as harm seems to come from messing with them. */
3871 uint64_t fSkipMask = RT_BIT(4) | RT_BIT(3);
3872 fSkipMask |= (RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & X86_PAGE_4K_BASE_MASK;
3873 return reportMsr_GenFunctionEx(uMsr, NULL, (uMsr - 0xc0010016) / 2, fSkipMask, 0, annotateValue(uValue));
3874}
3875
3876
3877/**
3878 * Deals carefully with a IORRMaski register.
3879 *
3880 * @returns VBox status code.
3881 * @param uMsr The MSR number.
3882 * @param uValue The current value.
3883 */
3884static int reportMsr_AmdK8IorrMaskN(uint32_t uMsr, uint64_t uValue)
3885{
3886 /* Skip know bits here, as harm seems to come from messing with them. */
3887 uint64_t fSkipMask = RT_BIT(11);
3888 fSkipMask |= (RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & X86_PAGE_4K_BASE_MASK;
3889 return reportMsr_GenFunctionEx(uMsr, NULL, (uMsr - 0xc0010017) / 2, fSkipMask, 0, annotateValue(uValue));
3890}
3891
3892
3893/**
3894 * Deals carefully with a IORRMaski register.
3895 *
3896 * @returns VBox status code.
3897 * @param uMsr The MSR number.
3898 * @param uValue The current value.
3899 */
3900static int reportMsr_AmdK8TopMemN(uint32_t uMsr, uint64_t uValue)
3901{
3902 /* Skip know bits here, as harm seems to come from messing with them. */
3903 uint64_t fSkipMask = (RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & ~(RT_BIT_64(23) - 1);
3904 return reportMsr_GenFunctionEx(uMsr, NULL, uMsr == 0xc001001d, fSkipMask, 0, annotateValue(uValue));
3905}
3906
3907
3908/**
3909 * Deals with the AMD P-state config range.
3910 *
3911 * @returns VBox status code.
3912 * @param paMsrs Pointer to the first MSR.
3913 * @param cMsrs The number of MSRs in the array @a paMsr.
3914 * @param pidxLoop Index variable that should be advanced to the
3915 * last MSR entry in the range.
3916 */
3917static int reportMsr_AmdFam10hPStateN(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t *pidxLoop)
3918{
3919 uint32_t uMsr = paMsrs[0].uMsr;
3920 AssertRelease(uMsr == 0xc0010064);
3921
3922 /* Count them. */
3923 uint32_t cRegs = 1;
3924 while ( cRegs < 8
3925 && cRegs < cMsrs
3926 && paMsrs[cRegs].uMsr == uMsr + cRegs)
3927 cRegs++;
3928
3929 /* Figure out which bits we should skip when probing. This is based on
3930 specs and may need adjusting for real life when handy. */
3931 uint64_t fSkipMask = RT_BIT_64(63); /* PstateEn */
3932 fSkipMask |= RT_BIT_64(41) | RT_BIT_64(40); /* IddDiv */
3933 fSkipMask |= UINT64_C(0x000000ff00000000); /* IddValue */
3934 if (CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch))
3935 fSkipMask |= UINT32_C(0xfe000000); /* NbVid - Northbridge VID */
3936 if ( CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch)
3937 || CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
3938 fSkipMask |= RT_BIT_32(22); /* NbDid or NbPstate. */
3939 if (g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver) /* ?? - listed in 10-1Fh model BDKG as well asFam16h */
3940 fSkipMask |= RT_BIT_32(16); /* CpuVid[7] */
3941 fSkipMask |= UINT32_C(0x0000fe00); /* CpuVid[6:0] */
3942 fSkipMask |= UINT32_C(0x000001c0); /* CpuDid */
3943 fSkipMask |= UINT32_C(0x0000003f); /* CpuFid */
3944
3945 /* Probe and report them one by one since we're passing values instead of
3946 register indexes to the functions. */
3947 for (uint32_t i = 0; i < cRegs; i++)
3948 {
3949 uint64_t fIgnMask = 0;
3950 uint64_t fGpMask = 0;
3951 int rc = msrProberModifyBitChanges(uMsr + i, &fIgnMask, &fGpMask, fSkipMask);
3952 if (RT_FAILURE(rc))
3953 return rc;
3954 printMsrFunctionExtended(uMsr + i, "AmdFam10hPStateN", NULL, paMsrs[i].uValue, fIgnMask, fGpMask,
3955 annotateValue(paMsrs[i].uValue));
3956 }
3957
3958 /* Advance. */
3959 *pidxLoop += cRegs - 1;
3960 return VINF_SUCCESS;
3961}
3962
3963
3964/**
3965 * Deals carefully with a COFVID control register.
3966 *
3967 * @returns VBox status code.
3968 * @param uMsr The MSR number.
3969 * @param uValue The current value.
3970 */
3971static int reportMsr_AmdFam10hCofVidControl(uint32_t uMsr, uint64_t uValue)
3972{
3973 /* Skip know bits here, as harm seems to come from messing with them. */
3974 uint64_t fSkipMask = 0;
3975 if (CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch))
3976 fSkipMask |= UINT32_C(0xfe000000); /* NbVid - Northbridge VID */
3977 else if (g_enmMicroarch >= kCpumMicroarch_AMD_15h_First) /* Listed in preliminary Fam16h BDKG. */
3978 fSkipMask |= UINT32_C(0xff000000); /* NbVid - Northbridge VID - includes bit 24 for Fam15h and Fam16h. Odd... */
3979 if ( CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch)
3980 || g_enmMicroarch >= kCpumMicroarch_AMD_15h_First) /* Listed in preliminary Fam16h BDKG. */
3981 fSkipMask |= RT_BIT_32(22); /* NbDid or NbPstate. */
3982 if (g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver) /* ?? - listed in 10-1Fh model BDKG as well asFam16h */
3983 fSkipMask |= RT_BIT_32(20); /* CpuVid[7] */
3984 fSkipMask |= UINT32_C(0x00070000); /* PstatId */
3985 fSkipMask |= UINT32_C(0x0000fe00); /* CpuVid[6:0] */
3986 fSkipMask |= UINT32_C(0x000001c0); /* CpuDid */
3987 fSkipMask |= UINT32_C(0x0000003f); /* CpuFid */
3988
3989 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
3990}
3991
3992
3993/**
3994 * Deals with the AMD [|L2I_|NB_]PERF_CT[LR] mixed ranges.
3995 *
3996 * Mixed here refers to the control and counter being in mixed in pairs as
3997 * opposed to them being two separate parallel arrays like in the 0xc0010000
3998 * area.
3999 *
4000 * @returns VBox status code.
4001 * @param paMsrs Pointer to the first MSR.
4002 * @param cMsrs The number of MSRs in the array @a paMsr.
4003 * @param cMax The max number of MSRs (not counters).
4004 * @param pidxLoop Index variable that should be advanced to the
4005 * last MSR entry in the range.
4006 */
4007static int reportMsr_AmdGenPerfMixedRange(VBCPUREPMSR const *paMsrs, uint32_t cMsrs, uint32_t cMax, uint32_t *pidxLoop)
4008{
4009 uint32_t uMsr = paMsrs[0].uMsr;
4010
4011 /* Count them. */
4012 uint32_t cRegs = 1;
4013 while ( cRegs < cMax
4014 && cRegs < cMsrs
4015 && paMsrs[cRegs].uMsr == uMsr + cRegs)
4016 cRegs++;
4017 if (cRegs & 1)
4018 return RTMsgErrorRc(VERR_INVALID_PARAMETER, "PERF range at %#x is odd: cRegs=%#x\n", uMsr, cRegs);
4019
4020 /* Report them as individual entries, using default names and such. */
4021 for (uint32_t i = 0; i < cRegs; i++)
4022 {
4023 uint64_t fIgnMask = 0;
4024 uint64_t fGpMask = 0;
4025 int rc = msrProberModifyBitChanges(uMsr + i, &fIgnMask, &fGpMask, 0);
4026 if (RT_FAILURE(rc))
4027 return rc;
4028 printMsrFunctionExtendedIdxVal(uMsr + i, NULL, NULL, i / 2, fIgnMask, fGpMask, annotateValue(paMsrs[i].uValue));
4029 }
4030
4031 /* Advance. */
4032 *pidxLoop += cRegs - 1;
4033 return VINF_SUCCESS;
4034}
4035
4036
4037/**
4038 * Deals carefully with a LS_CFG register.
4039 *
4040 * @returns VBox status code.
4041 * @param uMsr The MSR number.
4042 * @param uValue The current value.
4043 */
4044static int reportMsr_AmdK7InstrCacheCfg(uint32_t uMsr, uint64_t uValue)
4045{
4046 /* Skip know bits here, as harm seems to come from messing with them. */
4047 uint64_t fSkipMask = RT_BIT_64(9) /* DIS_SPEC_TLB_RLD */;
4048 if (CPUMMICROARCH_IS_AMD_FAM_10H(g_enmMicroarch))
4049 fSkipMask |= RT_BIT_64(14); /* DIS_IND */
4050 if (CPUMMICROARCH_IS_AMD_FAM_16H(g_enmMicroarch))
4051 fSkipMask |= RT_BIT_64(26); /* DIS_WIDEREAD_PWR_SAVE */
4052 if (CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
4053 {
4054 fSkipMask |= 0x1e; /* DisIcWayFilter */
4055 fSkipMask |= RT_BIT_64(39); /* DisLoopPredictor */
4056 fSkipMask |= RT_BIT_64(27); /* Unknown killer bit, possibly applicable to other microarchs. */
4057 fSkipMask |= RT_BIT_64(28); /* Unknown killer bit, possibly applicable to other microarchs. */
4058 }
4059 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
4060}
4061
4062
4063/**
4064 * Deals carefully with a CU_CFG register.
4065 *
4066 * @returns VBox status code.
4067 * @param uMsr The MSR number.
4068 * @param uValue The current value.
4069 */
4070static int reportMsr_AmdFam15hCombUnitCfg(uint32_t uMsr, uint64_t uValue)
4071{
4072 /* Skip know bits here, as harm seems to come from messing with them. */
4073 uint64_t fSkipMask = RT_BIT_64(23) /* L2WayLock */
4074 | RT_BIT_64(22) /* L2FirstLockWay */
4075 | RT_BIT_64(21) /* L2FirstLockWay */
4076 | RT_BIT_64(20) /* L2FirstLockWay */
4077 | RT_BIT_64(19) /* L2FirstLockWay */
4078 | RT_BIT_64(10) /* DcacheAggressivePriority */;
4079 fSkipMask |= RT_BIT_64(46) | RT_BIT_64(45); /* Killer field. Seen bit 46 set, 45 clear. Messing with either means reboot/BSOD. */
4080 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
4081}
4082
4083
4084/**
4085 * Deals carefully with a EX_CFG register.
4086 *
4087 * @returns VBox status code.
4088 * @param uMsr The MSR number.
4089 * @param uValue The current value.
4090 */
4091static int reportMsr_AmdFam15hExecUnitCfg(uint32_t uMsr, uint64_t uValue)
4092{
4093 /* Skip know bits here, as harm seems to come from messing with them. */
4094 uint64_t fSkipMask = RT_BIT_64(54) /* LateSbzResync */;
4095 fSkipMask |= RT_BIT_64(35); /* Undocumented killer bit. */
4096 return reportMsr_GenFunctionEx(uMsr, NULL, uValue, fSkipMask, 0, annotateValue(uValue));
4097}
4098
4099
4100
4101static int produceMsrReport(VBCPUREPMSR *paMsrs, uint32_t cMsrs)
4102{
4103 vbCpuRepDebug("produceMsrReport\n");
4104 RTThreadSleep(500);
4105
4106 for (uint32_t i = 0; i < cMsrs; i++)
4107 {
4108 uint32_t uMsr = paMsrs[i].uMsr;
4109 uint32_t fFlags = paMsrs[i].fFlags;
4110 uint64_t uValue = paMsrs[i].uValue;
4111 int rc;
4112#if 0
4113 //if (uMsr < 0x00000000)
4114 // continue;
4115 if (uMsr >= 0x00000277)
4116 {
4117 vbCpuRepDebug("produceMsrReport: uMsr=%#x (%s)...\n", uMsr, getMsrNameHandled(uMsr));
4118 RTThreadSleep(1000);
4119 }
4120#endif
4121 /*
4122 * Deal with write only regs first to avoid having to avoid them all the time.
4123 */
4124 if (fFlags & VBCPUREPMSR_F_WRITE_ONLY)
4125 {
4126 if (uMsr == 0x00000079)
4127 rc = printMsrWriteOnly(uMsr, NULL, NULL);
4128 else
4129 rc = reportMsr_Generic(uMsr, fFlags, uValue);
4130 }
4131 /*
4132 * VIA implement MSRs in a interesting way, so we have to select what we
4133 * want to handle there to avoid making the code below unreadable.
4134 */
4135 else if (isMsrViaDummy(uMsr, uValue, fFlags))
4136 rc = reportMsr_ViaDummyRange(&paMsrs[i], cMsrs - i, &i);
4137 /*
4138 * This shall be sorted by uMsr as much as possible.
4139 */
4140 else if (uMsr == 0x00000000 && g_enmVendor == CPUMCPUVENDOR_AMD && g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
4141 rc = printMsrAlias(uMsr, 0x00000402, NULL);
4142 else if (uMsr == 0x00000001 && g_enmVendor == CPUMCPUVENDOR_AMD && g_enmMicroarch >= kCpumMicroarch_AMD_K8_First)
4143 rc = printMsrAlias(uMsr, 0x00000401, NULL); /** @todo not 101% correct on Fam15h and later, 0xc0010015[McstatusWrEn] effect differs. */
4144 else if (uMsr == 0x0000001b)
4145 rc = reportMsr_Ia32ApicBase(uMsr, uValue);
4146 else if (uMsr == 0x00000040 && g_enmMicroarch <= kCpumMicroarch_Intel_P6_M_Dothan)
4147 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 8 /*cMax*/, "IntelLastBranchFromToN", &i);
4148 else if (uMsr == 0x00000040)
4149 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 8 /*cMax*/, "IntelLastBranchToN", uMsr, false,
4150 true, getGenericSkipMask(uMsr), &i);
4151 else if (uMsr == 0x00000060 && g_enmMicroarch >= kCpumMicroarch_Intel_Core_Yonah)
4152 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 8 /*cMax*/, "IntelLastBranchFromN", uMsr, false,
4153 true, getGenericSkipMask(uMsr), &i);
4154 else if (uMsr == 0x000000c1)
4155 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i,
4156 g_enmMicroarch >= kCpumMicroarch_Intel_Core7_First ? 8 : 4 /*cMax*/,
4157 NULL, &i);
4158 else if (uMsr == 0x00000186 && !g_fIntelNetBurst)
4159 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 8 /*cMax*/, "Ia32PerfEvtSelN", &i);
4160 else if (uMsr == 0x000001a0)
4161 rc = reportMsr_Ia32MiscEnable(uMsr, uValue);
4162 else if (uMsr >= 0x000001a6 && uMsr <= 0x000001a7)
4163 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 2 /*cMax*/, "IntelI7MsrOffCoreResponseN", &i);
4164 else if (uMsr == 0x000001db && g_fIntelNetBurst)
4165 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 4 /*cMax*/, "IntelLastBranchFromToN", &i);
4166 else if (uMsr == 0x00000200)
4167 rc = reportMsr_Ia32MtrrPhysBaseMaskN(&paMsrs[i], cMsrs - i, &i);
4168 else if (uMsr >= 0x00000250 && uMsr <= 0x00000279)
4169 rc = reportMsr_Ia32MtrrFixedOrPat(uMsr);
4170 else if (uMsr >= 0x00000280 && uMsr <= 0x00000295)
4171 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 22 /*cMax*/, NULL, 0x00000280, true /*fEarlyEndOk*/, false, 0, &i);
4172 else if (uMsr == 0x000002ff)
4173 rc = reportMsr_Ia32MtrrDefType(uMsr);
4174 else if (uMsr >= 0x00000309 && uMsr <= 0x0000030b && !g_fIntelNetBurst)
4175 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 3 /*cMax*/, NULL, 0x00000309, true /*fEarlyEndOk*/, false, 0, &i);
4176 else if ((uMsr == 0x000003f8 || uMsr == 0x000003fc || uMsr == 0x0000060a) && !g_fIntelNetBurst)
4177 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 4, NULL, uMsr - 3, true, false, 0, &i);
4178 else if ((uMsr == 0x000003f9 || uMsr == 0x000003fd || uMsr == 0x0000060b) && !g_fIntelNetBurst)
4179 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 8, NULL, uMsr - 6, true, false, 0, &i);
4180 else if ((uMsr == 0x000003fa || uMsr == 0x000003fe || uMsr == 0x0000060c) && !g_fIntelNetBurst)
4181 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 8, NULL, uMsr - 7, true, false, 0, &i);
4182 else if (uMsr >= 0x00000400 && uMsr <= 0x00000477)
4183 rc = reportMsr_Ia32McCtlStatusAddrMiscN(&paMsrs[i], cMsrs - i, &i);
4184 else if (uMsr == 0x000004c1)
4185 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 8, NULL, &i);
4186 else if (uMsr == 0x00000680 || uMsr == 0x000006c0)
4187 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 16, NULL, uMsr, false, false,
4188 g_fIntelNetBurst
4189 ? UINT64_C(0xffffffffffffff00) /* kludge */
4190 : UINT64_C(0xffff800000000000), &i);
4191 else if (uMsr >= 0x00000800 && uMsr <= 0x000008ff)
4192 rc = reportMsr_GenX2Apic(&paMsrs[i], cMsrs - i, &i);
4193 else if (uMsr == 0x00002000 && g_enmVendor == CPUMCPUVENDOR_INTEL)
4194 rc = reportMsr_GenFunctionEx(uMsr, "IntelP6CrN", 0, X86_CR0_PE | X86_CR0_PG, 0,
4195 annotateIfMissingBits(uValue, X86_CR0_PE | X86_CR0_PE | X86_CR0_ET));
4196 else if (uMsr == 0x00002002 && g_enmVendor == CPUMCPUVENDOR_INTEL)
4197 rc = reportMsr_GenFunctionEx(uMsr, "IntelP6CrN", 2, 0, 0, annotateValue(uValue));
4198 else if (uMsr == 0x00002003 && g_enmVendor == CPUMCPUVENDOR_INTEL)
4199 {
4200 uint64_t fCr3Mask = (RT_BIT_64(vbCpuRepGetPhysAddrWidth()) - 1) & (X86_CR3_PAE_PAGE_MASK | X86_CR3_AMD64_PAGE_MASK);
4201 if (!vbCpuRepSupportsPae())
4202 fCr3Mask &= X86_CR3_PAGE_MASK | X86_CR3_AMD64_PAGE_MASK;
4203 rc = reportMsr_GenFunctionEx(uMsr, "IntelP6CrN", 3, fCr3Mask, 0, annotateValue(uValue));
4204 }
4205 else if (uMsr == 0x00002004 && g_enmVendor == CPUMCPUVENDOR_INTEL)
4206 rc = reportMsr_GenFunctionEx(uMsr, "IntelP6CrN", 4,
4207 X86_CR4_PSE | X86_CR4_PAE | X86_CR4_MCE | X86_CR4_SMXE, 0,
4208 annotateValue(uValue));
4209 else if (uMsr == 0xc0000080)
4210 rc = reportMsr_Amd64Efer(uMsr, uValue);
4211 else if (uMsr >= 0xc0000408 && uMsr <= 0xc000040f)
4212 rc = reportMsr_AmdFam10hMc4MiscN(&paMsrs[i], cMsrs - i, &i);
4213 else if (uMsr == 0xc0010000 && g_enmVendor == CPUMCPUVENDOR_AMD)
4214 rc = reportMsr_AmdK8PerfCtlN(&paMsrs[i], cMsrs - i, &i);
4215 else if (uMsr == 0xc0010004 && g_enmVendor == CPUMCPUVENDOR_AMD)
4216 rc = reportMsr_AmdK8PerfCtrN(&paMsrs[i], cMsrs - i, &i);
4217 else if (uMsr == 0xc0010010 && g_enmVendor == CPUMCPUVENDOR_AMD)
4218 rc = reportMsr_AmdK8SysCfg(uMsr, uValue);
4219 else if (uMsr == 0xc0010015 && g_enmVendor == CPUMCPUVENDOR_AMD)
4220 rc = reportMsr_AmdK8HwCr(uMsr, uValue);
4221 else if ((uMsr == 0xc0010016 || uMsr == 0xc0010018) && g_enmVendor == CPUMCPUVENDOR_AMD)
4222 rc = reportMsr_AmdK8IorrBaseN(uMsr, uValue);
4223 else if ((uMsr == 0xc0010017 || uMsr == 0xc0010019) && g_enmVendor == CPUMCPUVENDOR_AMD)
4224 rc = reportMsr_AmdK8IorrMaskN(uMsr, uValue);
4225 else if ((uMsr == 0xc001001a || uMsr == 0xc001001d) && g_enmVendor == CPUMCPUVENDOR_AMD)
4226 rc = reportMsr_AmdK8TopMemN(uMsr, uValue);
4227 else if (uMsr == 0xc0010030 && g_enmVendor == CPUMCPUVENDOR_AMD)
4228 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 6, "AmdK8CpuNameN", &i);
4229 else if (uMsr >= 0xc0010044 && uMsr <= 0xc001004a && g_enmVendor == CPUMCPUVENDOR_AMD)
4230 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 7, "AmdK8McCtlMaskN", 0xc0010044, true /*fEarlyEndOk*/, false, 0, &i);
4231 else if (uMsr == 0xc0010050 && g_enmVendor == CPUMCPUVENDOR_AMD)
4232 rc = reportMsr_GenRangeFunction(&paMsrs[i], cMsrs - i, 4, "AmdK8SmiOnIoTrapN", &i);
4233 else if (uMsr == 0xc0010064 && g_enmVendor == CPUMCPUVENDOR_AMD)
4234 rc = reportMsr_AmdFam10hPStateN(&paMsrs[i], cMsrs - i, &i);
4235 else if (uMsr == 0xc0010070 && g_enmVendor == CPUMCPUVENDOR_AMD)
4236 rc = reportMsr_AmdFam10hCofVidControl(uMsr, uValue);
4237 else if ((uMsr == 0xc0010118 || uMsr == 0xc0010119) && getMsrFnName(uMsr, NULL) && g_enmVendor == CPUMCPUVENDOR_AMD)
4238 rc = printMsrFunction(uMsr, NULL, NULL, annotateValue(uValue)); /* RAZ, write key. */
4239 else if (uMsr == 0xc0010200 && g_enmVendor == CPUMCPUVENDOR_AMD)
4240 rc = reportMsr_AmdGenPerfMixedRange(&paMsrs[i], cMsrs - i, 12, &i);
4241 else if (uMsr == 0xc0010230 && g_enmVendor == CPUMCPUVENDOR_AMD)
4242 rc = reportMsr_AmdGenPerfMixedRange(&paMsrs[i], cMsrs - i, 8, &i);
4243 else if (uMsr == 0xc0010240 && g_enmVendor == CPUMCPUVENDOR_AMD)
4244 rc = reportMsr_AmdGenPerfMixedRange(&paMsrs[i], cMsrs - i, 8, &i);
4245 else if (uMsr == 0xc0011019 && g_enmMicroarch >= kCpumMicroarch_AMD_15h_Piledriver && g_enmVendor == CPUMCPUVENDOR_AMD)
4246 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 3, "AmdK7DrXAddrMaskN", 0xc0011019 - 1,
4247 false /*fEarlyEndOk*/, false /*fNoIgnMask*/, 0, &i);
4248 else if (uMsr == 0xc0011021 && g_enmVendor == CPUMCPUVENDOR_AMD)
4249 rc = reportMsr_AmdK7InstrCacheCfg(uMsr, uValue);
4250 else if (uMsr == 0xc0011023 && CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
4251 rc = reportMsr_AmdFam15hCombUnitCfg(uMsr, uValue);
4252 else if (uMsr == 0xc0011027 && g_enmVendor == CPUMCPUVENDOR_AMD)
4253 rc = reportMsr_GenRangeFunctionEx(&paMsrs[i], cMsrs - i, 1, "AmdK7DrXAddrMaskN", 0xc0011027,
4254 false /*fEarlyEndOk*/, false /*fNoIgnMask*/, 0, &i);
4255 else if (uMsr == 0xc001102c && CPUMMICROARCH_IS_AMD_FAM_15H(g_enmMicroarch))
4256 rc = reportMsr_AmdFam15hExecUnitCfg(uMsr, uValue);
4257 /* generic handling. */
4258 else
4259 rc = reportMsr_Generic(uMsr, fFlags, uValue);
4260
4261 if (RT_FAILURE(rc))
4262 return rc;
4263
4264 /*
4265 * A little ugly snooping.
4266 */
4267 if (uMsr == 0x000000cd && !(fFlags & VBCPUREPMSR_F_WRITE_ONLY))
4268 g_uMsrIntelP6FsbFrequency = uValue;
4269 }
4270
4271 return VINF_SUCCESS;
4272}
4273
4274
4275/**
4276 * Custom MSR hacking & probing.
4277 *
4278 * Called when the '-d' option is given.
4279 *
4280 * @returns VBox status code.
4281 */
4282static int hackingMsrs(void)
4283{
4284#if 0
4285 vbCpuRepDebug("\nhackingMsrs:\n"); RTStrmFlush(g_pDebugOut); RTThreadSleep(2000);
4286
4287 uint32_t uMsr = 0xc0000081;
4288 vbCpuRepDebug("%#x: msrProberModifyNoChange -> %RTbool\n", uMsr, msrProberModifyNoChange(uMsr));
4289 RTThreadSleep(3000);
4290
4291 vbCpuRepDebug("%#x: msrProberModifyBit 30 -> %d\n", uMsr, msrProberModifyBit(uMsr, 30));
4292 RTThreadSleep(3000);
4293
4294 vbCpuRepDebug("%#x: msrProberModifyZero -> %RTbool\n", uMsr, msrProberModifyZero(uMsr));
4295 RTThreadSleep(3000);
4296
4297 for (uint32_t i = 0; i < 63; i++)
4298 {
4299 vbCpuRepDebug("%#x: bit=%02u -> %d\n", msrProberModifyBit(uMsr, i));
4300 RTThreadSleep(500);
4301 }
4302#else
4303
4304 uint32_t uMsr = 0xc0010010;
4305 uint64_t uValue = 0;
4306 msrProberRead(uMsr, &uValue);
4307 reportMsr_AmdK8SysCfg(uMsr, uValue);
4308#endif
4309 return VINF_SUCCESS;
4310}
4311
4312
4313static int probeMsrs(bool fHacking, const char *pszNameC, const char *pszCpuDesc,
4314 char *pszMsrMask, size_t cbMsrMask)
4315{
4316 /* Initialize the mask. */
4317 if (pszMsrMask && cbMsrMask)
4318 RTStrCopy(pszMsrMask, cbMsrMask, "UINT32_MAX /** @todo */");
4319
4320 /*
4321 * Are MSRs supported by the CPU?
4322 */
4323 if ( !ASMIsValidStdRange(ASMCpuId_EAX(0))
4324 || !(ASMCpuId_EDX(1) & X86_CPUID_FEATURE_EDX_MSR) )
4325 {
4326 vbCpuRepDebug("Skipping MSR probing, CPUID indicates there isn't any MSR support.\n");
4327 return VINF_SUCCESS;
4328 }
4329
4330 /*
4331 * Initialize the support library and check if we can read MSRs.
4332 */
4333 int rc = SUPR3Init(NULL);
4334 if (RT_FAILURE(rc))
4335 {
4336 vbCpuRepDebug("warning: Unable to initialize the support library (%Rrc), skipping MSR detection.\n", rc);
4337 return VINF_SUCCESS;
4338 }
4339 uint64_t uValue;
4340 bool fGp;
4341 rc = SUPR3MsrProberRead(MSR_IA32_TSC, NIL_RTCPUID, &uValue, &fGp);
4342 if (RT_FAILURE(rc))
4343 {
4344 vbCpuRepDebug("warning: MSR probing not supported by the support driver (%Rrc), skipping MSR detection.\n", rc);
4345 return VINF_SUCCESS;
4346 }
4347 vbCpuRepDebug("MSR_IA32_TSC: %#llx fGp=%RTbool\n", uValue, fGp);
4348 rc = SUPR3MsrProberRead(0xdeadface, NIL_RTCPUID, &uValue, &fGp);
4349 vbCpuRepDebug("0xdeadface: %#llx fGp=%RTbool rc=%Rrc\n", uValue, fGp, rc);
4350
4351 /*
4352 * Initialize globals we use.
4353 */
4354 uint32_t uEax, uEbx, uEcx, uEdx;
4355 ASMCpuIdExSlow(0, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4356 if (!ASMIsValidStdRange(uEax))
4357 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Invalid std CPUID range: %#x\n", uEax);
4358 g_enmVendor = CPUMR3CpuIdDetectVendorEx(uEax, uEbx, uEcx, uEdx);
4359
4360 ASMCpuIdExSlow(1, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4361 g_enmMicroarch = CPUMR3CpuIdDetermineMicroarchEx(g_enmVendor,
4362 ASMGetCpuFamily(uEax),
4363 ASMGetCpuModel(uEax, g_enmVendor == CPUMCPUVENDOR_INTEL),
4364 ASMGetCpuStepping(uEax));
4365 g_fIntelNetBurst = CPUMMICROARCH_IS_INTEL_NETBURST(g_enmMicroarch);
4366
4367 /*
4368 * Do the probing.
4369 */
4370 if (fHacking)
4371 rc = hackingMsrs();
4372 else
4373 {
4374 /* Determine the MSR mask. */
4375 uint32_t fMsrMask = determineMsrAndMask();
4376 if (fMsrMask == UINT32_MAX)
4377 RTStrCopy(pszMsrMask, cbMsrMask, "UINT32_MAX");
4378 else
4379 RTStrPrintf(pszMsrMask, cbMsrMask, "UINT32_C(%#x)", fMsrMask);
4380
4381 /* Detect MSR. */
4382 VBCPUREPMSR *paMsrs;
4383 uint32_t cMsrs;
4384 rc = findMsrs(&paMsrs, &cMsrs, fMsrMask);
4385 if (RT_FAILURE(rc))
4386 return rc;
4387
4388 /* Probe the MSRs and spit out the database table. */
4389 vbCpuRepPrintf("\n"
4390 "#ifndef CPUM_DB_STANDALONE\n"
4391 "/**\n"
4392 " * MSR ranges for %s.\n"
4393 " */\n"
4394 "static CPUMMSRRANGE const g_aMsrRanges_%s[] = \n{\n",
4395 pszCpuDesc,
4396 pszNameC);
4397 rc = produceMsrReport(paMsrs, cMsrs);
4398 vbCpuRepPrintf("};\n"
4399 "#endif /* !CPUM_DB_STANDALONE */\n"
4400 "\n"
4401 );
4402
4403 RTMemFree(paMsrs);
4404 paMsrs = NULL;
4405 }
4406 return rc;
4407}
4408
4409
4410static int produceCpuIdArray(const char *pszNameC, const char *pszCpuDesc)
4411{
4412 /*
4413 * Collect the data.
4414 */
4415 PCPUMCPUIDLEAF paLeaves;
4416 uint32_t cLeaves;
4417 int rc = CPUMR3CpuIdCollectLeaves(&paLeaves, &cLeaves);
4418 if (RT_FAILURE(rc))
4419 return RTMsgErrorRc(rc, "CPUMR3CollectCpuIdInfo failed: %Rrc\n", rc);
4420
4421 /*
4422 * Dump the array.
4423 */
4424 vbCpuRepPrintf("\n"
4425 "#ifndef CPUM_DB_STANDALONE\n"
4426 "/**\n"
4427 " * CPUID leaves for %s.\n"
4428 " */\n"
4429 "static CPUMCPUIDLEAF const g_aCpuIdLeaves_%s[] = \n{\n",
4430 pszCpuDesc,
4431 pszNameC);
4432 for (uint32_t i = 0; i < cLeaves; i++)
4433 {
4434 vbCpuRepPrintf(" { %#010x, %#010x, ", paLeaves[i].uLeaf, paLeaves[i].uSubLeaf);
4435 if (paLeaves[i].fSubLeafMask == UINT32_MAX)
4436 vbCpuRepPrintf("UINT32_MAX, ");
4437 else
4438 vbCpuRepPrintf("%#010x, ", paLeaves[i].fSubLeafMask);
4439 vbCpuRepPrintf("%#010x, %#010x, %#010x, %#010x, ",
4440 paLeaves[i].uEax, paLeaves[i].uEbx, paLeaves[i].uEcx, paLeaves[i].uEdx);
4441 if (paLeaves[i].fFlags == 0)
4442 vbCpuRepPrintf("0 },\n");
4443 else
4444 {
4445 vbCpuRepPrintf("0");
4446 uint32_t fFlags = paLeaves[i].fFlags;
4447 if (paLeaves[i].fFlags & CPUMCPUIDLEAF_F_INTEL_TOPOLOGY_SUBLEAVES)
4448 {
4449 vbCpuRepPrintf(" | CPUMCPUIDLEAF_F_INTEL_TOPOLOGY_SUBLEAVES");
4450 fFlags &= ~CPUMCPUIDLEAF_F_INTEL_TOPOLOGY_SUBLEAVES;
4451 }
4452 if (paLeaves[i].fFlags & CPUMCPUIDLEAF_F_CONTAINS_APIC_ID)
4453 {
4454 vbCpuRepPrintf(" | CPUMCPUIDLEAF_F_CONTAINS_APIC_ID");
4455 fFlags &= ~CPUMCPUIDLEAF_F_CONTAINS_APIC_ID;
4456 }
4457 if (paLeaves[i].fFlags & CPUMCPUIDLEAF_F_CONTAINS_APIC)
4458 {
4459 vbCpuRepPrintf(" | CPUMCPUIDLEAF_F_CONTAINS_APIC");
4460 fFlags &= ~CPUMCPUIDLEAF_F_CONTAINS_APIC;
4461 }
4462 if (fFlags)
4463 {
4464 RTMemFree(paLeaves);
4465 return RTMsgErrorRc(rc, "Unknown CPUID flags %#x\n", fFlags);
4466 }
4467 vbCpuRepPrintf(" },\n");
4468 }
4469 }
4470 vbCpuRepPrintf("};\n"
4471 "#endif /* !CPUM_DB_STANDALONE */\n"
4472 "\n");
4473 RTMemFree(paLeaves);
4474 return VINF_SUCCESS;
4475}
4476
4477
4478static const char *cpuVendorToString(CPUMCPUVENDOR enmCpuVendor)
4479{
4480 switch (enmCpuVendor)
4481 {
4482 case CPUMCPUVENDOR_INTEL: return "Intel";
4483 case CPUMCPUVENDOR_AMD: return "AMD";
4484 case CPUMCPUVENDOR_VIA: return "VIA";
4485 case CPUMCPUVENDOR_CYRIX: return "Cyrix";
4486 case CPUMCPUVENDOR_INVALID:
4487 case CPUMCPUVENDOR_UNKNOWN:
4488 case CPUMCPUVENDOR_32BIT_HACK:
4489 break;
4490 }
4491 return "invalid-cpu-vendor";
4492}
4493
4494
4495/**
4496 * Takes a shot a the bus frequency name (last part).
4497 *
4498 * @returns Name suffix.
4499 */
4500static const char *vbCpuRepGuessScalableBusFrequencyName(void)
4501{
4502 if (CPUMMICROARCH_IS_INTEL_CORE7(g_enmMicroarch))
4503 return g_enmMicroarch >= kCpumMicroarch_Intel_Core7_SandyBridge ? "100MHZ" : "133MHZ";
4504
4505 if (g_uMsrIntelP6FsbFrequency != UINT64_MAX)
4506 switch (g_uMsrIntelP6FsbFrequency & 0x7)
4507 {
4508 case 5: return "100MHZ";
4509 case 1: return "133MHZ";
4510 case 3: return "167MHZ";
4511 case 2: return "200MHZ";
4512 case 0: return "267MHZ";
4513 case 4: return "333MHZ";
4514 case 6: return "400MHZ";
4515 }
4516
4517 return "UNKNOWN";
4518}
4519
4520
4521static int produceCpuReport(void)
4522{
4523 /*
4524 * Figure the cpu vendor.
4525 */
4526 if (!ASMHasCpuId())
4527 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "No CPUID support.\n");
4528 uint32_t uEax, uEbx, uEcx, uEdx;
4529 ASMCpuIdExSlow(0, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4530 if (!ASMIsValidStdRange(uEax))
4531 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Invalid std CPUID range: %#x\n", uEax);
4532
4533 CPUMCPUVENDOR enmVendor = CPUMR3CpuIdDetectVendorEx(uEax, uEbx, uEcx, uEdx);
4534 if (enmVendor == CPUMCPUVENDOR_UNKNOWN)
4535 return RTMsgErrorRc(VERR_NOT_IMPLEMENTED, "Unknown CPU vendor: %.4s%.4s%.4s\n", &uEbx, &uEdx, &uEcx);
4536 vbCpuRepDebug("CPU Vendor: %s - %.4s%.4s%.4s\n", CPUMR3CpuVendorName(enmVendor), &uEbx, &uEdx, &uEcx);
4537
4538 /*
4539 * Determine the micro arch.
4540 */
4541 ASMCpuIdExSlow(1, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4542 CPUMMICROARCH enmMicroarch = CPUMR3CpuIdDetermineMicroarchEx(enmVendor,
4543 ASMGetCpuFamily(uEax),
4544 ASMGetCpuModel(uEax, enmVendor == CPUMCPUVENDOR_INTEL),
4545 ASMGetCpuStepping(uEax));
4546
4547 /*
4548 * Generate a name.
4549 */
4550 char szName[16*3+1];
4551 char szNameC[16*3+1];
4552 char szNameRaw[16*3+1];
4553 char *pszName = szName;
4554 char *pszCpuDesc = (char *)"";
4555
4556 ASMCpuIdExSlow(0x80000000, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4557 if (ASMIsValidExtRange(uEax) && uEax >= UINT32_C(0x80000004))
4558 {
4559 /* Get the raw name and strip leading spaces. */
4560 ASMCpuIdExSlow(0x80000002, 0, 0, 0, &szNameRaw[0 + 0], &szNameRaw[4 + 0], &szNameRaw[8 + 0], &szNameRaw[12 + 0]);
4561 ASMCpuIdExSlow(0x80000003, 0, 0, 0, &szNameRaw[0 + 16], &szNameRaw[4 + 16], &szNameRaw[8 + 16], &szNameRaw[12 + 16]);
4562 ASMCpuIdExSlow(0x80000004, 0, 0, 0, &szNameRaw[0 + 32], &szNameRaw[4 + 32], &szNameRaw[8 + 32], &szNameRaw[12 + 32]);
4563 szNameRaw[48] = '\0';
4564 pszCpuDesc = RTStrStrip(szNameRaw);
4565 vbCpuRepDebug("Name2: %s\n", pszCpuDesc);
4566
4567 /* Reduce the name. */
4568 pszName = strcpy(szName, pszCpuDesc);
4569
4570 static const char * const s_apszSuffixes[] =
4571 {
4572 "CPU @",
4573 };
4574 for (uint32_t i = 0; i < RT_ELEMENTS(s_apszSuffixes); i++)
4575 {
4576 char *pszHit = strstr(pszName, s_apszSuffixes[i]);
4577 if (pszHit)
4578 RT_BZERO(pszHit, strlen(pszHit));
4579 }
4580
4581 static const char * const s_apszWords[] =
4582 {
4583 "(TM)", "(tm)", "(R)", "(r)", "Processor", "CPU", "@",
4584 };
4585 for (uint32_t i = 0; i < RT_ELEMENTS(s_apszWords); i++)
4586 {
4587 const char *pszWord = s_apszWords[i];
4588 size_t cchWord = strlen(pszWord);
4589 char *pszHit;
4590 while ((pszHit = strstr(pszName, pszWord)) != NULL)
4591 memmove(pszHit, pszHit + cchWord, strlen(pszHit + cchWord) + 1);
4592 }
4593
4594 RTStrStripR(pszName);
4595 for (char *psz = pszName; *psz; psz++)
4596 if (RT_C_IS_BLANK(*psz))
4597 {
4598 size_t cchBlanks = 1;
4599 while (RT_C_IS_BLANK(psz[cchBlanks]))
4600 cchBlanks++;
4601 *psz = ' ';
4602 if (cchBlanks > 1)
4603 memmove(psz + 1, psz + cchBlanks, strlen(psz + cchBlanks) + 1);
4604 }
4605 pszName = RTStrStripL(pszName);
4606 vbCpuRepDebug("Name: %s\n", pszName);
4607
4608 /* Make it C/C++ acceptable. */
4609 strcpy(szNameC, pszName);
4610 unsigned offDst = 0;
4611 for (unsigned offSrc = 0; ; offSrc++)
4612 {
4613 char ch = szNameC[offSrc];
4614 if (!RT_C_IS_ALNUM(ch) && ch != '_' && ch != '\0')
4615 ch = '_';
4616 if (ch == '_' && offDst > 0 && szNameC[offDst - 1] == '_')
4617 offDst--;
4618 szNameC[offDst++] = ch;
4619 if (!ch)
4620 break;
4621 }
4622 while (offDst > 1 && szNameC[offDst - 1] == '_')
4623 szNameC[--offDst] = '\0';
4624
4625 vbCpuRepDebug("NameC: %s\n", szNameC);
4626 }
4627 else
4628 {
4629 ASMCpuIdExSlow(1, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4630 RTStrPrintf(szNameC, sizeof(szNameC), "%s_%u_%u_%u", cpuVendorToString(enmVendor), ASMGetCpuFamily(uEax),
4631 ASMGetCpuModel(uEax, enmVendor == CPUMCPUVENDOR_INTEL), ASMGetCpuStepping(uEax));
4632 pszCpuDesc = pszName = szNameC;
4633 vbCpuRepDebug("Name/NameC: %s\n", szNameC);
4634 }
4635
4636 /*
4637 * Print a file header, if we're not outputting to stdout (assumption being
4638 * that stdout is used while hacking the reporter and too much output is
4639 * unwanted).
4640 */
4641 if (g_pReportOut)
4642 {
4643 RTTIMESPEC Now;
4644 char szNow[64];
4645 RTTimeSpecToString(RTTimeNow(&Now), szNow, sizeof(szNow));
4646 char *pchDot = strchr(szNow, '.');
4647 if (pchDot)
4648 strcpy(pchDot, "Z");
4649
4650 vbCpuRepPrintf("/* $" "Id" "$ */\n"
4651 "/** @file\n"
4652 " * CPU database entry \"%s\".\n"
4653 " * Generated at %s by VBoxCpuReport v%sr%s on %s.%s.\n"
4654 " */\n"
4655 "\n"
4656 "/*\n"
4657 " * Copyright (C) 2013-2016 Oracle Corporation\n"
4658 " *\n"
4659 " * This file is part of VirtualBox Open Source Edition (OSE), as\n"
4660 " * available from http://www.virtualbox.org. This file is free software;\n"
4661 " * you can redistribute it and/or modify it under the terms of the GNU\n"
4662 " * General Public License (GPL) as published by the Free Software\n"
4663 " * Foundation, in version 2 as it comes in the \"COPYING\" file of the\n"
4664 " * VirtualBox OSE distribution. VirtualBox OSE is distributed in the\n"
4665 " * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.\n"
4666 " */\n"
4667 "\n"
4668 "#ifndef VBOX_CPUDB_%s\n"
4669 "#define VBOX_CPUDB_%s\n"
4670 "\n",
4671 pszName,
4672 szNow, RTBldCfgVersion(), RTBldCfgRevisionStr(), RTBldCfgTarget(), RTBldCfgTargetArch(),
4673 szNameC, szNameC);
4674 }
4675
4676 /*
4677 * Extract CPUID based data.
4678 */
4679 int rc = produceCpuIdArray(szNameC, pszCpuDesc);
4680 if (RT_FAILURE(rc))
4681 return rc;
4682
4683 CPUMUNKNOWNCPUID enmUnknownMethod;
4684 CPUMCPUID DefUnknown;
4685 rc = CPUMR3CpuIdDetectUnknownLeafMethod(&enmUnknownMethod, &DefUnknown);
4686 if (RT_FAILURE(rc))
4687 return RTMsgErrorRc(rc, "CPUMR3DetectCpuIdUnknownMethod failed: %Rrc\n", rc);
4688 vbCpuRepDebug("enmUnknownMethod=%s\n", CPUMR3CpuIdUnknownLeafMethodName(enmUnknownMethod));
4689
4690 /*
4691 * Do the MSRs, if we can.
4692 */
4693 char szMsrMask[64];
4694 probeMsrs(false /*fHacking*/, szNameC, pszCpuDesc, szMsrMask, sizeof(szMsrMask));
4695
4696 /*
4697 * Emit the CPUMDBENTRY record.
4698 */
4699 ASMCpuIdExSlow(1, 0, 0, 0, &uEax, &uEbx, &uEcx, &uEdx);
4700 vbCpuRepPrintf("\n"
4701 "/**\n"
4702 " * Database entry for %s.\n"
4703 " */\n"
4704 "static CPUMDBENTRY const g_Entry_%s = \n"
4705 "{\n"
4706 " /*.pszName = */ \"%s\",\n"
4707 " /*.pszFullName = */ \"%s\",\n"
4708 " /*.enmVendor = */ CPUMCPUVENDOR_%s,\n"
4709 " /*.uFamily = */ %u,\n"
4710 " /*.uModel = */ %u,\n"
4711 " /*.uStepping = */ %u,\n"
4712 " /*.enmMicroarch = */ kCpumMicroarch_%s,\n"
4713 " /*.uScalableBusFreq = */ CPUM_SBUSFREQ_%s,\n"
4714 " /*.fFlags = */ 0,\n"
4715 " /*.cMaxPhysAddrWidth= */ %u,\n"
4716 " /*.paCpuIdLeaves = */ NULL_ALONE(g_aCpuIdLeaves_%s),\n"
4717 " /*.cCpuIdLeaves = */ ZERO_ALONE(RT_ELEMENTS(g_aCpuIdLeaves_%s)),\n"
4718 " /*.enmUnknownCpuId = */ CPUMUNKNOWNCPUID_%s,\n"
4719 " /*.DefUnknownCpuId = */ { %#010x, %#010x, %#010x, %#010x },\n"
4720 " /*.fMsrMask = */ %s,\n"
4721 " /*.cMsrRanges = */ ZERO_ALONE(RT_ELEMENTS(g_aMsrRanges_%s)),\n"
4722 " /*.paMsrRanges = */ NULL_ALONE(g_aMsrRanges_%s),\n"
4723 "};\n"
4724 "\n"
4725 "#endif /* !VBOX_DB_%s */\n"
4726 "\n",
4727 pszCpuDesc,
4728 szNameC,
4729 pszName,
4730 pszCpuDesc,
4731 CPUMR3CpuVendorName(enmVendor),
4732 ASMGetCpuFamily(uEax),
4733 ASMGetCpuModel(uEax, enmVendor == CPUMCPUVENDOR_INTEL),
4734 ASMGetCpuStepping(uEax),
4735 CPUMR3MicroarchName(enmMicroarch),
4736 vbCpuRepGuessScalableBusFrequencyName(),
4737 vbCpuRepGetPhysAddrWidth(),
4738 szNameC,
4739 szNameC,
4740 CPUMR3CpuIdUnknownLeafMethodName(enmUnknownMethod),
4741 DefUnknown.uEax,
4742 DefUnknown.uEbx,
4743 DefUnknown.uEcx,
4744 DefUnknown.uEdx,
4745 szMsrMask,
4746 szNameC,
4747 szNameC,
4748 szNameC
4749 );
4750
4751 return VINF_SUCCESS;
4752}
4753
4754
4755int main(int argc, char **argv)
4756{
4757 int rc = RTR3InitExe(argc, &argv, 0 /*fFlags*/);
4758 if (RT_FAILURE(rc))
4759 return RTMsgInitFailure(rc);
4760
4761 /*
4762 * Argument parsing?
4763 */
4764 static const RTGETOPTDEF s_aOptions[] =
4765 {
4766 { "--msrs-only", 'm', RTGETOPT_REQ_NOTHING },
4767 { "--msrs-dev", 'd', RTGETOPT_REQ_NOTHING },
4768 { "--output", 'o', RTGETOPT_REQ_STRING },
4769 { "--log", 'l', RTGETOPT_REQ_STRING },
4770 };
4771 RTGETOPTSTATE State;
4772 RTGetOptInit(&State, argc, argv, &s_aOptions[0], RT_ELEMENTS(s_aOptions), 1, RTGETOPTINIT_FLAGS_OPTS_FIRST);
4773
4774 enum
4775 {
4776 kCpuReportOp_Normal,
4777 kCpuReportOp_MsrsOnly,
4778 kCpuReportOp_MsrsHacking
4779 } enmOp = kCpuReportOp_Normal;
4780 g_pReportOut = NULL;
4781 g_pDebugOut = NULL;
4782 const char *pszOutput = NULL;
4783 const char *pszDebugOut = NULL;
4784
4785 int iOpt;
4786 RTGETOPTUNION ValueUnion;
4787 while ((iOpt = RTGetOpt(&State, &ValueUnion)) != 0)
4788 {
4789 switch (iOpt)
4790 {
4791 case 'm':
4792 enmOp = kCpuReportOp_MsrsOnly;
4793 break;
4794
4795 case 'd':
4796 enmOp = kCpuReportOp_MsrsHacking;
4797 break;
4798
4799 case 'o':
4800 pszOutput = ValueUnion.psz;
4801 break;
4802
4803 case 'l':
4804 pszDebugOut = ValueUnion.psz;
4805 break;
4806
4807 case 'h':
4808 RTPrintf("Usage: VBoxCpuReport [-m|--msrs-only] [-d|--msrs-dev] [-h|--help] [-V|--version] [-o filename.h] [-l debug.log]\n");
4809 RTPrintf("Internal tool for gathering information to the VMM CPU database.\n");
4810 return RTEXITCODE_SUCCESS;
4811 case 'V':
4812 RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
4813 return RTEXITCODE_SUCCESS;
4814 default:
4815 return RTGetOptPrintError(iOpt, &ValueUnion);
4816 }
4817 }
4818
4819 /*
4820 * Open the alternative debug log stream.
4821 */
4822 if (pszDebugOut)
4823 {
4824 if (RTFileExists(pszDebugOut) && !RTSymlinkExists(pszDebugOut))
4825 {
4826 char szOld[RTPATH_MAX];
4827 rc = RTStrCopy(szOld, sizeof(szOld), pszDebugOut);
4828 if (RT_SUCCESS(rc))
4829 rc = RTStrCat(szOld, sizeof(szOld), ".old");
4830 if (RT_SUCCESS(rc))
4831 RTFileRename(pszDebugOut, szOld, RTFILEMOVE_FLAGS_REPLACE);
4832 }
4833 rc = RTStrmOpen(pszDebugOut, "w", &g_pDebugOut);
4834 if (RT_FAILURE(rc))
4835 {
4836 RTMsgError("Error opening '%s': %Rrc", pszDebugOut, rc);
4837 g_pDebugOut = NULL;
4838 }
4839 }
4840
4841 /*
4842 * Do the requested job.
4843 */
4844 rc = VERR_INTERNAL_ERROR;
4845 switch (enmOp)
4846 {
4847 case kCpuReportOp_Normal:
4848 /* switch output file. */
4849 if (pszOutput)
4850 {
4851 if (RTFileExists(pszOutput) && !RTSymlinkExists(pszOutput))
4852 {
4853 char szOld[RTPATH_MAX];
4854 rc = RTStrCopy(szOld, sizeof(szOld), pszOutput);
4855 if (RT_SUCCESS(rc))
4856 rc = RTStrCat(szOld, sizeof(szOld), ".old");
4857 if (RT_SUCCESS(rc))
4858 RTFileRename(pszOutput, szOld, RTFILEMOVE_FLAGS_REPLACE);
4859 }
4860 rc = RTStrmOpen(pszOutput, "w", &g_pReportOut);
4861 if (RT_FAILURE(rc))
4862 {
4863 RTMsgError("Error opening '%s': %Rrc", pszOutput, rc);
4864 break;
4865 }
4866 }
4867 rc = produceCpuReport();
4868 break;
4869 case kCpuReportOp_MsrsOnly:
4870 case kCpuReportOp_MsrsHacking:
4871 rc = probeMsrs(enmOp == kCpuReportOp_MsrsHacking, NULL, NULL, NULL, 0);
4872 break;
4873 }
4874
4875 /*
4876 * Close the output files.
4877 */
4878 if (g_pReportOut)
4879 {
4880 RTStrmClose(g_pReportOut);
4881 g_pReportOut = NULL;
4882 }
4883
4884 if (g_pDebugOut)
4885 {
4886 RTStrmClose(g_pDebugOut);
4887 g_pDebugOut = NULL;
4888 }
4889
4890 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
4891}
4892
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