VirtualBox

source: vbox/trunk/src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp@ 72969

Last change on this file since 72969 was 69661, checked in by vboxsync, 7 years ago

VBoxCpuReporter: cleanups [build fix]

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.6 KB
Line 
1/* $Id: VBoxCpuReportMsrLinux.cpp 69661 2017-11-12 15:39:02Z vboxsync $ */
2/** @file
3 * MsrLinux - Linux-specific MSR access.
4 */
5
6/*
7 * Copyright (C) 2013-2017 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 "VBoxCpuReport.h"
23
24#include <iprt/file.h>
25#include <iprt/thread.h>
26
27#ifndef RT_OS_WINDOWS
28# include <unistd.h>
29#else /* RT_OS_WINDOWS: for test compiling this file on windows */
30# include <io.h>
31int pread(int, void *, size_t, off_t);
32int pwrite(int, void const *, size_t, off_t);
33#endif
34#include <fcntl.h>
35#include <errno.h>
36
37
38/*********************************************************************************************************************************
39* Defined Constants And Macros *
40*********************************************************************************************************************************/
41#define MSR_DEV_NAME "/dev/cpu/0/msr"
42
43
44/*********************************************************************************************************************************
45* Global Variables *
46*********************************************************************************************************************************/
47/** The /dev/xxx/msr file descriptor. */
48static int g_fdMsr;
49
50
51/**
52 * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberRead}
53 */
54static DECLCALLBACK(int) linuxMsrProberRead(uint32_t uMsr, RTCPUID idCpu, uint64_t *puValue, bool *pfGp)
55{
56 int rc = VINF_SUCCESS;
57
58 if (idCpu != NIL_RTCPUID)
59 return VERR_INVALID_PARAMETER;
60
61 if (g_fdMsr < 0)
62 return VERR_INVALID_STATE;
63
64 *pfGp = true;
65 if (pread(g_fdMsr, puValue, sizeof(*puValue), uMsr) != sizeof(*puValue))
66 rc = VERR_READ_ERROR;
67 else
68 *pfGp = false;
69
70 return RT_SUCCESS(rc) && !pfGp;
71}
72
73
74/**
75 * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberWrite}
76 */
77static DECLCALLBACK(int) linuxMsrProberWrite(uint32_t uMsr, RTCPUID idCpu, uint64_t uValue, bool *pfGp)
78{
79 int rc = VINF_SUCCESS;
80
81 if (idCpu != NIL_RTCPUID)
82 return VERR_INVALID_PARAMETER;
83
84 if (g_fdMsr < 0)
85 return VERR_INVALID_STATE;
86
87 *pfGp = true;
88 if (pwrite(g_fdMsr, &uValue, sizeof(uValue), uMsr) != sizeof(uValue))
89 rc = VERR_WRITE_ERROR;
90 else
91 *pfGp = false;
92
93 return RT_SUCCESS(rc) && !pfGp;
94}
95
96/**
97 * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberModify}
98 */
99static DECLCALLBACK(int) linuxMsrProberModify(uint32_t uMsr, RTCPUID idCpu, uint64_t fAndMask, uint64_t fOrMask,
100 PSUPMSRPROBERMODIFYRESULT pResult)
101{
102 int rc = VINF_SUCCESS;
103 uint64_t uBefore, uWrite, uAfter;
104 int rcBefore, rcWrite, rcAfter, rcRestore;
105
106 if (idCpu != NIL_RTCPUID)
107 return VERR_INVALID_PARAMETER;
108
109 if (g_fdMsr < 0)
110 return VERR_INVALID_STATE;
111
112#if 0
113 vbCpuRepDebug("MSR %#x\n", uMsr);
114 RTThreadSleep(10);
115#endif
116 rcBefore = pread(g_fdMsr, &uBefore, sizeof(uBefore), uMsr);
117 uWrite = (uBefore & fAndMask) | fOrMask;
118 rcWrite = pwrite(g_fdMsr, &uWrite, sizeof(uWrite), uMsr);
119 rcAfter = pread(g_fdMsr, &uAfter, sizeof(uAfter), uMsr);
120 rcRestore = pwrite(g_fdMsr, &uBefore, sizeof(uBefore), uMsr);
121
122#if 0
123 vbCpuRepDebug("MSR: %#x, %#llx -> %#llx -> %#llx (%d/%d/%d/%d)\n",
124 uMsr, uBefore, uWrite, uAfter,
125 rcBefore, rcWrite != sizeof(uWrite), rcAfter, rcRestore);
126#endif
127 pResult->uBefore = uBefore;
128 pResult->uWritten = uWrite;
129 pResult->uAfter = uAfter;
130 pResult->fBeforeGp = rcBefore != sizeof(uBefore);
131 pResult->fModifyGp = rcWrite != sizeof(uWrite);
132 pResult->fAfterGp = rcAfter != sizeof(uAfter);
133 pResult->fRestoreGp = rcRestore != sizeof(uBefore);
134
135 return rc;
136}
137
138
139/**
140 * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnTerm}
141 */
142static DECLCALLBACK(void) linuxMsrProberTerm(void)
143{
144 if (g_fdMsr >= 0)
145 {
146 close(g_fdMsr);
147 g_fdMsr = -1;
148 }
149}
150
151int VbCpuRepMsrProberInitPlatform(PVBCPUREPMSRACCESSORS pMsrAccessors)
152{
153 RTFILE hFile;
154 int rc = RTFileOpen(&hFile, MSR_DEV_NAME, RTFILE_O_READWRITE | RTFILE_O_DENY_NONE | RTFILE_O_OPEN);
155 if (RT_SUCCESS(rc))
156 {
157 g_fdMsr = RTFileToNative(hFile);
158 Assert(g_fdMsr != -1);
159
160 pMsrAccessors->fAtomic = false; /* Can't modify/restore MSRs without trip to R3. */
161 pMsrAccessors->pfnMsrProberRead = linuxMsrProberRead;
162 pMsrAccessors->pfnMsrProberWrite = linuxMsrProberWrite;
163 pMsrAccessors->pfnMsrProberModify = linuxMsrProberModify;
164 pMsrAccessors->pfnTerm = linuxMsrProberTerm;
165 return VINF_SUCCESS;
166 }
167 vbCpuRepDebug("warning: Failed to open " MSR_DEV_NAME ": %Rrc\n", rc);
168 return rc;
169}
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