summaryrefslogtreecommitdiffstats
path: root/src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-06 03:01:46 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-06 03:01:46 +0000
commitf8fe689a81f906d1b91bb3220acde2a4ecb14c5b (patch)
tree26484e9d7e2c67806c2d1760196ff01aaa858e8c /src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp
parentInitial commit. (diff)
downloadvirtualbox-f8fe689a81f906d1b91bb3220acde2a4ecb14c5b.tar.xz
virtualbox-f8fe689a81f906d1b91bb3220acde2a4ecb14c5b.zip
Adding upstream version 6.0.4-dfsg.upstream/6.0.4-dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp')
-rw-r--r--src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp170
1 files changed, 170 insertions, 0 deletions
diff --git a/src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp b/src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp
new file mode 100644
index 00000000..cb4aeaff
--- /dev/null
+++ b/src/VBox/VMM/tools/VBoxCpuReportMsrLinux.cpp
@@ -0,0 +1,170 @@
+/* $Id: VBoxCpuReportMsrLinux.cpp $ */
+/** @file
+ * MsrLinux - Linux-specific MSR access.
+ */
+
+/*
+ * Copyright (C) 2013-2019 Oracle Corporation
+ *
+ * This file is part of VirtualBox Open Source Edition (OSE), as
+ * available from http://www.virtualbox.org. This file is free software;
+ * you can redistribute it and/or modify it under the terms of the GNU
+ * General Public License (GPL) as published by the Free Software
+ * Foundation, in version 2 as it comes in the "COPYING" file of the
+ * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
+ * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
+ */
+
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#include "VBoxCpuReport.h"
+
+#include <iprt/err.h>
+#include <iprt/file.h>
+#include <iprt/thread.h>
+
+#ifndef RT_OS_WINDOWS
+# include <unistd.h>
+#else /* RT_OS_WINDOWS: for test compiling this file on windows */
+# include <io.h>
+int pread(int, void *, size_t, off_t);
+int pwrite(int, void const *, size_t, off_t);
+#endif
+#include <fcntl.h>
+#include <errno.h>
+
+
+/*********************************************************************************************************************************
+* Defined Constants And Macros *
+*********************************************************************************************************************************/
+#define MSR_DEV_NAME "/dev/cpu/0/msr"
+
+
+/*********************************************************************************************************************************
+* Global Variables *
+*********************************************************************************************************************************/
+/** The /dev/xxx/msr file descriptor. */
+static int g_fdMsr;
+
+
+/**
+ * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberRead}
+ */
+static DECLCALLBACK(int) linuxMsrProberRead(uint32_t uMsr, RTCPUID idCpu, uint64_t *puValue, bool *pfGp)
+{
+ int rc = VINF_SUCCESS;
+
+ if (idCpu != NIL_RTCPUID)
+ return VERR_INVALID_PARAMETER;
+
+ if (g_fdMsr < 0)
+ return VERR_INVALID_STATE;
+
+ *pfGp = true;
+ if (pread(g_fdMsr, puValue, sizeof(*puValue), uMsr) != sizeof(*puValue))
+ rc = VERR_READ_ERROR;
+ else
+ *pfGp = false;
+
+ return RT_SUCCESS(rc) && !pfGp;
+}
+
+
+/**
+ * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberWrite}
+ */
+static DECLCALLBACK(int) linuxMsrProberWrite(uint32_t uMsr, RTCPUID idCpu, uint64_t uValue, bool *pfGp)
+{
+ int rc = VINF_SUCCESS;
+
+ if (idCpu != NIL_RTCPUID)
+ return VERR_INVALID_PARAMETER;
+
+ if (g_fdMsr < 0)
+ return VERR_INVALID_STATE;
+
+ *pfGp = true;
+ if (pwrite(g_fdMsr, &uValue, sizeof(uValue), uMsr) != sizeof(uValue))
+ rc = VERR_WRITE_ERROR;
+ else
+ *pfGp = false;
+
+ return RT_SUCCESS(rc) && !pfGp;
+}
+
+/**
+ * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnMsrProberModify}
+ */
+static DECLCALLBACK(int) linuxMsrProberModify(uint32_t uMsr, RTCPUID idCpu, uint64_t fAndMask, uint64_t fOrMask,
+ PSUPMSRPROBERMODIFYRESULT pResult)
+{
+ int rc = VINF_SUCCESS;
+ uint64_t uBefore, uWrite, uAfter;
+ int rcBefore, rcWrite, rcAfter, rcRestore;
+
+ if (idCpu != NIL_RTCPUID)
+ return VERR_INVALID_PARAMETER;
+
+ if (g_fdMsr < 0)
+ return VERR_INVALID_STATE;
+
+#if 0
+ vbCpuRepDebug("MSR %#x\n", uMsr);
+ RTThreadSleep(10);
+#endif
+ rcBefore = pread(g_fdMsr, &uBefore, sizeof(uBefore), uMsr);
+ uWrite = (uBefore & fAndMask) | fOrMask;
+ rcWrite = pwrite(g_fdMsr, &uWrite, sizeof(uWrite), uMsr);
+ rcAfter = pread(g_fdMsr, &uAfter, sizeof(uAfter), uMsr);
+ rcRestore = pwrite(g_fdMsr, &uBefore, sizeof(uBefore), uMsr);
+
+#if 0
+ vbCpuRepDebug("MSR: %#x, %#llx -> %#llx -> %#llx (%d/%d/%d/%d)\n",
+ uMsr, uBefore, uWrite, uAfter,
+ rcBefore, rcWrite != sizeof(uWrite), rcAfter, rcRestore);
+#endif
+ pResult->uBefore = uBefore;
+ pResult->uWritten = uWrite;
+ pResult->uAfter = uAfter;
+ pResult->fBeforeGp = rcBefore != sizeof(uBefore);
+ pResult->fModifyGp = rcWrite != sizeof(uWrite);
+ pResult->fAfterGp = rcAfter != sizeof(uAfter);
+ pResult->fRestoreGp = rcRestore != sizeof(uBefore);
+
+ return rc;
+}
+
+
+/**
+ * @interface_method_impl{VBCPUREPMSRACCESSORS,pfnTerm}
+ */
+static DECLCALLBACK(void) linuxMsrProberTerm(void)
+{
+ if (g_fdMsr >= 0)
+ {
+ close(g_fdMsr);
+ g_fdMsr = -1;
+ }
+}
+
+int VbCpuRepMsrProberInitPlatform(PVBCPUREPMSRACCESSORS pMsrAccessors)
+{
+ RTFILE hFile;
+ int rc = RTFileOpen(&hFile, MSR_DEV_NAME, RTFILE_O_READWRITE | RTFILE_O_DENY_NONE | RTFILE_O_OPEN);
+ if (RT_SUCCESS(rc))
+ {
+ g_fdMsr = RTFileToNative(hFile);
+ Assert(g_fdMsr != -1);
+
+ pMsrAccessors->fAtomic = false; /* Can't modify/restore MSRs without trip to R3. */
+ pMsrAccessors->pfnMsrProberRead = linuxMsrProberRead;
+ pMsrAccessors->pfnMsrProberWrite = linuxMsrProberWrite;
+ pMsrAccessors->pfnMsrProberModify = linuxMsrProberModify;
+ pMsrAccessors->pfnTerm = linuxMsrProberTerm;
+ return VINF_SUCCESS;
+ }
+ vbCpuRepDebug("warning: Failed to open " MSR_DEV_NAME ": %Rrc\n", rc);
+ return rc;
+}