2014-08-16 23:34:04 +00:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
|
|
******************************************************************************
|
2021-05-25 00:21:42 +00:00
|
|
|
* Copyright 2021 Ben Vanik. All rights reserved. *
|
2014-08-16 23:34:04 +00:00
|
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
|
|
******************************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <io.h>
|
|
|
|
|
2016-01-01 20:46:26 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
#include "xenia/base/cvar.h"
|
2015-08-30 01:06:30 +00:00
|
|
|
#include "xenia/base/logging.h"
|
2020-03-02 15:37:11 +00:00
|
|
|
#include "xenia/base/main.h"
|
2015-07-14 03:49:29 +00:00
|
|
|
#include "xenia/base/platform_win.h"
|
2015-05-02 10:42:51 +00:00
|
|
|
#include "xenia/base/string.h"
|
2014-08-16 23:34:04 +00:00
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
// Autogenerated by `xb premake`.
|
|
|
|
#include "build/version.h"
|
2018-12-16 05:36:07 +00:00
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
// For RequestHighPerformance.
|
|
|
|
#include <winternl.h>
|
|
|
|
|
|
|
|
// Includes Windows headers, so it goes here.
|
|
|
|
#include "third_party/xbyak/xbyak/xbyak_util.h"
|
2018-05-22 21:41:05 +00:00
|
|
|
|
2018-05-22 22:31:30 +00:00
|
|
|
DEFINE_bool(win32_high_freq, true,
|
2019-04-17 19:49:29 +00:00
|
|
|
"Requests high performance from the NT kernel", "Kernel");
|
2020-09-07 11:22:23 +00:00
|
|
|
DEFINE_bool(enable_console, false, "Open a console window with the main window",
|
|
|
|
"General");
|
2018-05-22 22:31:30 +00:00
|
|
|
|
2015-05-02 10:42:51 +00:00
|
|
|
namespace xe {
|
2014-08-16 23:34:04 +00:00
|
|
|
|
|
|
|
bool has_console_attached_ = true;
|
|
|
|
|
|
|
|
bool has_console_attached() { return has_console_attached_; }
|
|
|
|
|
|
|
|
void AttachConsole() {
|
2020-09-07 11:22:23 +00:00
|
|
|
if (!cvars::enable_console) {
|
2014-08-16 23:34:04 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-09-07 11:22:23 +00:00
|
|
|
|
|
|
|
AllocConsole();
|
|
|
|
|
2014-08-16 23:34:04 +00:00
|
|
|
has_console_attached_ = true;
|
|
|
|
|
2018-05-31 22:22:50 +00:00
|
|
|
auto std_handle = (intptr_t)GetStdHandle(STD_OUTPUT_HANDLE);
|
|
|
|
auto con_handle = _open_osfhandle(std_handle, _O_TEXT);
|
|
|
|
auto fp = _fdopen(con_handle, "w");
|
2020-09-07 11:22:23 +00:00
|
|
|
freopen_s(&fp, "CONOUT$", "w", stdout);
|
2018-05-31 22:22:50 +00:00
|
|
|
|
|
|
|
std_handle = (intptr_t)GetStdHandle(STD_ERROR_HANDLE);
|
|
|
|
con_handle = _open_osfhandle(std_handle, _O_TEXT);
|
|
|
|
fp = _fdopen(con_handle, "w");
|
2020-09-07 11:22:23 +00:00
|
|
|
freopen_s(&fp, "CONOUT$", "w", stderr);
|
2014-08-16 23:34:04 +00:00
|
|
|
}
|
|
|
|
|
2018-05-22 21:41:05 +00:00
|
|
|
static void RequestHighPerformance() {
|
|
|
|
#if XE_PLATFORM_WIN32
|
|
|
|
NTSTATUS(*NtQueryTimerResolution)
|
|
|
|
(OUT PULONG MinimumResolution, OUT PULONG MaximumResolution,
|
|
|
|
OUT PULONG CurrentResolution);
|
|
|
|
|
|
|
|
NTSTATUS(*NtSetTimerResolution)
|
|
|
|
(IN ULONG DesiredResolution, IN BOOLEAN SetResolution,
|
|
|
|
OUT PULONG CurrentResolution);
|
|
|
|
|
|
|
|
NtQueryTimerResolution = (decltype(NtQueryTimerResolution))GetProcAddress(
|
2020-03-02 15:37:11 +00:00
|
|
|
GetModuleHandleW(L"ntdll.dll"), "NtQueryTimerResolution");
|
2018-05-22 21:41:05 +00:00
|
|
|
NtSetTimerResolution = (decltype(NtSetTimerResolution))GetProcAddress(
|
2020-03-02 15:37:11 +00:00
|
|
|
GetModuleHandleW(L"ntdll.dll"), "NtSetTimerResolution");
|
2018-05-22 21:41:05 +00:00
|
|
|
if (!NtQueryTimerResolution || !NtSetTimerResolution) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG minimum_resolution, maximum_resolution, current_resolution;
|
|
|
|
NtQueryTimerResolution(&minimum_resolution, &maximum_resolution,
|
|
|
|
¤t_resolution);
|
2018-05-23 03:20:34 +00:00
|
|
|
NtSetTimerResolution(maximum_resolution, TRUE, ¤t_resolution);
|
2018-05-22 21:41:05 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
static bool parse_launch_arguments(const xe::EntryInfo& entry_info,
|
|
|
|
std::vector<std::string>& args) {
|
2015-08-01 06:48:24 +00:00
|
|
|
auto command_line = GetCommandLineW();
|
2020-03-02 15:37:11 +00:00
|
|
|
|
|
|
|
int wargc;
|
|
|
|
wchar_t** wargv = CommandLineToArgvW(command_line, &wargc);
|
|
|
|
if (!wargv) {
|
|
|
|
return false;
|
2015-08-01 06:48:24 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 22:08:59 +00:00
|
|
|
// Convert all args to narrow, as cxxopts doesn't support wchar.
|
2020-03-02 15:37:11 +00:00
|
|
|
int argc = wargc;
|
|
|
|
char** argv = reinterpret_cast<char**>(alloca(sizeof(char*) * argc));
|
|
|
|
for (int n = 0; n < argc; n++) {
|
|
|
|
size_t len = std::wcstombs(nullptr, wargv[n], 0);
|
|
|
|
argv[n] = reinterpret_cast<char*>(alloca(sizeof(char) * (len + 1)));
|
|
|
|
std::wcstombs(argv[n], wargv[n], len + 1);
|
2014-08-16 23:34:04 +00:00
|
|
|
}
|
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
LocalFree(wargv);
|
|
|
|
|
2021-05-25 00:21:42 +00:00
|
|
|
if (!entry_info.transparent_options) {
|
|
|
|
cvar::ParseLaunchArguments(argc, argv, entry_info.positional_usage.value(),
|
|
|
|
entry_info.positional_options.value());
|
|
|
|
}
|
2014-08-16 23:34:04 +00:00
|
|
|
|
2020-03-02 15:37:11 +00:00
|
|
|
args.clear();
|
2014-08-16 23:34:04 +00:00
|
|
|
for (int n = 0; n < argc; n++) {
|
2020-03-02 15:37:11 +00:00
|
|
|
args.push_back(std::string(argv[n]));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Main() {
|
|
|
|
auto entry_info = xe::GetEntryInfo();
|
|
|
|
|
|
|
|
std::vector<std::string> args;
|
|
|
|
if (!parse_launch_arguments(entry_info, args)) {
|
|
|
|
return 1;
|
2014-08-16 23:34:04 +00:00
|
|
|
}
|
|
|
|
|
2020-09-07 11:22:23 +00:00
|
|
|
// Attach a console so we can write output to stdout. If the user hasn't
|
|
|
|
// redirected output themselves it'll pop up a window.
|
|
|
|
xe::AttachConsole();
|
|
|
|
|
2014-08-17 04:19:21 +00:00
|
|
|
// Setup COM on the main thread.
|
|
|
|
// NOTE: this may fail if COM has already been initialized - that's OK.
|
2019-04-19 09:02:32 +00:00
|
|
|
#pragma warning(suppress : 6031)
|
2014-08-17 04:19:21 +00:00
|
|
|
CoInitializeEx(nullptr, COINIT_MULTITHREADED);
|
|
|
|
|
2015-08-30 01:06:30 +00:00
|
|
|
// Initialize logging. Needs parsed FLAGS.
|
|
|
|
xe::InitializeLogging(entry_info.name);
|
|
|
|
|
2019-04-20 06:31:38 +00:00
|
|
|
Xbyak::util::Cpu cpu;
|
|
|
|
if (!cpu.has(Xbyak::util::Cpu::tAVX)) {
|
|
|
|
xe::FatalError(
|
|
|
|
"Your CPU does not support AVX, which is required by Xenia. See the "
|
|
|
|
"FAQ for system requirements at https://xenia.jp");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-12-27 19:53:37 +00:00
|
|
|
// Print version info.
|
2020-02-28 20:30:48 +00:00
|
|
|
XELOGI("Build: " XE_BUILD_BRANCH " / " XE_BUILD_COMMIT " on " XE_BUILD_DATE);
|
2015-12-27 19:53:37 +00:00
|
|
|
|
2018-12-16 04:56:36 +00:00
|
|
|
// Request high performance timing.
|
2019-04-17 19:49:29 +00:00
|
|
|
if (cvars::win32_high_freq) {
|
2018-12-16 04:56:36 +00:00
|
|
|
RequestHighPerformance();
|
|
|
|
}
|
|
|
|
|
2014-08-16 23:34:04 +00:00
|
|
|
// Call app-provided entry point.
|
|
|
|
int result = entry_info.entry_point(args);
|
|
|
|
|
2018-03-04 01:00:04 +00:00
|
|
|
xe::ShutdownLogging();
|
2014-08-16 23:34:04 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-08-01 06:48:24 +00:00
|
|
|
} // namespace xe
|
2019-04-20 06:31:38 +00:00
|
|
|
|
|
|
|
// Used in console mode apps; automatically picked based on subsystem.
|
|
|
|
int main(int argc_ignored, char** argv_ignored) { return xe::Main(); }
|
|
|
|
|
|
|
|
// Used in windowed apps; automatically picked based on subsystem.
|
|
|
|
int WINAPI wWinMain(HINSTANCE, HINSTANCE, LPWSTR command_line, int) {
|
|
|
|
// Run normal entry point.
|
|
|
|
return xe::Main();
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined _M_IX86
|
|
|
|
#pragma comment( \
|
|
|
|
linker, \
|
|
|
|
"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='x86' publicKeyToken='6595b64144ccf1df' language='*'\"") // NOLINT(whitespace/line_length)
|
|
|
|
#elif defined _M_IA64
|
|
|
|
#pragma comment( \
|
|
|
|
linker, \
|
|
|
|
"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='ia64' publicKeyToken='6595b64144ccf1df' language='*'\"") // NOLINT(whitespace/line_length)
|
|
|
|
#elif defined _M_X64
|
|
|
|
#pragma comment( \
|
|
|
|
linker, \
|
|
|
|
"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='amd64' publicKeyToken='6595b64144ccf1df' language='*'\"") // NOLINT(whitespace/line_length)
|
|
|
|
#else
|
|
|
|
#pragma comment( \
|
|
|
|
linker, \
|
|
|
|
"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"") // NOLINT(whitespace/line_length)
|
|
|
|
#endif
|