summaryrefslogtreecommitdiffstats
path: root/src/VBox/VMM/tools/VBoxVMMPreload.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/VBox/VMM/tools/VBoxVMMPreload.cpp')
-rw-r--r--src/VBox/VMM/tools/VBoxVMMPreload.cpp224
1 files changed, 224 insertions, 0 deletions
diff --git a/src/VBox/VMM/tools/VBoxVMMPreload.cpp b/src/VBox/VMM/tools/VBoxVMMPreload.cpp
new file mode 100644
index 00000000..33fcc372
--- /dev/null
+++ b/src/VBox/VMM/tools/VBoxVMMPreload.cpp
@@ -0,0 +1,224 @@
+/* $Id: VBoxVMMPreload.cpp $ */
+/** @file
+ * VBoxVMMPreload - Preload VBox the ring-0 modules.
+ */
+
+/*
+ * Copyright (C) 2012-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 <iprt/buildconfig.h>
+#include <iprt/getopt.h>
+#include <iprt/initterm.h>
+#include <iprt/message.h>
+#include <iprt/path.h>
+#include <iprt/stream.h>
+#include <iprt/string.h>
+#include <iprt/thread.h>
+
+#include <VBox/sup.h>
+#include <VBox/version.h>
+
+
+/*********************************************************************************************************************************
+* Global Variables *
+*********************************************************************************************************************************/
+/**
+ * Known modules and their associated data (there are only known modules!).
+ */
+static struct
+{
+ const char *pszName;
+ bool fPreload;
+ void *pvImageBase;
+} g_aModules[] =
+{
+ { "VMMR0.r0", true, NULL },
+ { "VBoxDDR0.r0", true, NULL },
+};
+
+static uint32_t g_cVerbose = 1;
+static bool g_fLockDown = false;
+
+
+/**
+ * Parses the options.
+ *
+ * @returns RTEXITCODE_SUCCESS on success.
+ * @param argc Argument count .
+ * @param argv Argument vector.
+ * @param pfExit Set to @c true if we should exit.
+ */
+static RTEXITCODE ParseOptions(int argc, char **argv, bool *pfExit)
+{
+ /*
+ * Parse arguments.
+ */
+ static const RTGETOPTDEF s_aOptions[] =
+ {
+ { "--only", 'o', RTGETOPT_REQ_STRING },
+ { "--quiet", 'q', RTGETOPT_REQ_NOTHING },
+ { "--lock" , 'l', RTGETOPT_REQ_NOTHING },
+ { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
+ };
+
+ bool fAll = true;
+
+ int ch;
+ RTGETOPTUNION ValueUnion;
+ RTGETOPTSTATE GetState;
+ RTGetOptInit(&GetState, argc, argv, s_aOptions, RT_ELEMENTS(s_aOptions), 1, 0 /* fFlags */);
+ while ((ch = RTGetOpt(&GetState, &ValueUnion)))
+ {
+ switch(ch)
+ {
+ case 'o':
+ {
+ uint32_t i;
+
+ if (fAll)
+ {
+ fAll = false;
+ for (i = 0; i < RT_ELEMENTS(g_aModules); i++)
+ g_aModules[i].fPreload = false;
+ }
+
+ i = RT_ELEMENTS(g_aModules);
+ while (i-- > 0)
+ if (!strcmp(ValueUnion.psz, g_aModules[i].pszName))
+ {
+ g_aModules[i].fPreload = true;
+ break;
+ }
+ if (i > RT_ELEMENTS(g_aModules))
+ return RTMsgErrorExit(RTEXITCODE_FAILURE, "No known module '%s'", ValueUnion.psz);
+ break;
+ }
+
+ case 'v':
+ g_cVerbose++;
+ break;
+
+ case 'q':
+ g_cVerbose = 0;
+ break;
+
+ case 'l':
+ g_fLockDown = true;
+ break;
+
+ case 'h':
+ RTPrintf(VBOX_PRODUCT " VMM ring-0 Module Preloader Version " VBOX_VERSION_STRING
+ "(C) 2005-" VBOX_C_YEAR " " VBOX_VENDOR "\n"
+ "All rights reserved.\n"
+ "\n"
+ "Usage: VBoxVMMPreload [-hlqvV] [-o|--only <mod>]\n"
+ "\n");
+ *pfExit = true;
+ return RTEXITCODE_SUCCESS;
+
+ case 'V':
+ RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
+ *pfExit = true;
+ return RTEXITCODE_SUCCESS;
+
+ default:
+ return RTGetOptPrintError(ch, &ValueUnion);
+ }
+ }
+ return RTEXITCODE_SUCCESS;
+}
+
+
+/**
+ * Loads the modules.
+ *
+ * @returns RTEXITCODE_SUCCESS on success.
+ */
+static RTEXITCODE LoadModules(void)
+{
+ RTERRINFOSTATIC ErrInfo;
+
+ for (uint32_t i = 0; i < RT_ELEMENTS(g_aModules); i++)
+ {
+ if (g_aModules[i].fPreload)
+ {
+ char szPath[RTPATH_MAX];
+ int rc = RTPathAppPrivateArch(szPath, sizeof(szPath));
+ if (RT_SUCCESS(rc))
+ rc = RTPathAppend(szPath, sizeof(szPath), g_aModules[i].pszName);
+ if (RT_FAILURE(rc))
+ return RTMsgErrorExit(RTEXITCODE_FAILURE, "RTPathAppPrivateArch or RTPathAppend returned %Rrc", rc);
+
+ RTErrInfoInitStatic(&ErrInfo);
+ rc = SUPR3LoadModule(szPath, g_aModules[i].pszName, &g_aModules[i].pvImageBase, &ErrInfo.Core);
+ if (RT_FAILURE(rc))
+ return RTMsgErrorExit(RTEXITCODE_FAILURE, "SUPR3LoadModule failed for %s (%s): %s (rc=%Rrc)",
+ g_aModules[i].pszName, szPath, ErrInfo.Core.pszMsg, rc);
+ if (g_cVerbose >= 1)
+ RTMsgInfo("Loaded '%s' ('%s') at %p\n", szPath, g_aModules[i].pszName, g_aModules[i].pvImageBase);
+ }
+ }
+
+ if (g_fLockDown)
+ {
+ RTErrInfoInitStatic(&ErrInfo);
+ int rc = SUPR3LockDownLoader(&ErrInfo.Core);
+ if (RT_FAILURE(rc))
+ return RTMsgErrorExit(RTEXITCODE_FAILURE, "SUPR3LockDownLoader failed: %s (rc=%Rrc)",
+ ErrInfo.Core.pszMsg, rc);
+ if (g_cVerbose >= 1)
+ RTMsgInfo("Locked down module loader interface!\n");
+ }
+
+ RTStrmFlush(g_pStdOut);
+ return RTEXITCODE_SUCCESS;
+}
+
+
+/**
+ * Entry point.
+ */
+extern "C" DECLEXPORT(int) TrustedMain(int argc, char **argv, char **envp)
+{
+ RT_NOREF1(envp);
+ bool fExit = false;
+ RTEXITCODE rcExit = ParseOptions(argc, argv, &fExit);
+ if (rcExit == RTEXITCODE_SUCCESS && !fExit)
+ {
+ rcExit = LoadModules();
+ if (rcExit == RTEXITCODE_SUCCESS)
+ {
+ for (;;)
+ RTThreadSleep(RT_INDEFINITE_WAIT);
+ }
+ }
+ return rcExit;
+}
+
+
+#ifndef VBOX_WITH_HARDENING
+/**
+ * Main entry point.
+ */
+int main(int argc, char **argv, char **envp)
+{
+ int rc = RTR3InitExe(argc, &argv, RTR3INIT_FLAGS_SUPLIB);
+ if (RT_FAILURE(rc))
+ return RTMsgInitFailure(rc);
+ return TrustedMain(argc, argv, envp);
+}
+#endif /* !VBOX_WITH_HARDENING */
+