2014-08-16 23:34:04 +00:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
|
|
******************************************************************************
|
|
|
|
* Copyright 2014 Ben Vanik. All rights reserved. *
|
|
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
|
|
******************************************************************************
|
|
|
|
*/
|
|
|
|
|
2015-05-02 10:42:51 +00:00
|
|
|
#include "xenia/base/main.h"
|
2014-08-16 23:34:04 +00:00
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <io.h>
|
|
|
|
|
2016-01-01 20:46:26 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
|
2015-12-27 19:53:37 +00:00
|
|
|
// Autogenerated by `xb premake`.
|
|
|
|
#include "build/version.h"
|
|
|
|
|
2019-04-17 19:49:29 +00:00
|
|
|
#include "xenia/base/filesystem.h"
|
2015-08-30 01:06:30 +00:00
|
|
|
#include "xenia/base/logging.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
|
|
|
|
2018-12-16 05:36:07 +00:00
|
|
|
#include "third_party/xbyak/xbyak/xbyak_util.h"
|
|
|
|
|
2018-05-22 21:41:05 +00:00
|
|
|
#include <bcrypt.h>
|
2019-04-17 19:49:29 +00:00
|
|
|
#include "xenia/base/cvar.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");
|
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() {
|
|
|
|
bool has_console = ::AttachConsole(ATTACH_PARENT_PROCESS) == TRUE;
|
|
|
|
if (!has_console) {
|
|
|
|
// We weren't launched from a console, so just return.
|
|
|
|
// We could alloc our own console, but meh:
|
|
|
|
// has_console = AllocConsole() == TRUE;
|
|
|
|
has_console_attached_ = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
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");
|
|
|
|
*stdout = *fp;
|
2018-05-31 21:46:03 +00:00
|
|
|
setvbuf(stdout, nullptr, _IONBF, 0);
|
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");
|
|
|
|
*stderr = *fp;
|
2014-08-17 04:19:21 +00:00
|
|
|
setvbuf(stderr, nullptr, _IONBF, 0);
|
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(
|
|
|
|
GetModuleHandle(L"ntdll.dll"), "NtQueryTimerResolution");
|
|
|
|
NtSetTimerResolution = (decltype(NtSetTimerResolution))GetProcAddress(
|
|
|
|
GetModuleHandle(L"ntdll.dll"), "NtSetTimerResolution");
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2015-08-01 06:48:24 +00:00
|
|
|
int Main() {
|
2015-05-02 10:42:51 +00:00
|
|
|
auto entry_info = xe::GetEntryInfo();
|
2014-08-16 23:34:04 +00:00
|
|
|
|
2015-08-01 06:48:24 +00:00
|
|
|
// Convert command line to an argv-like format so we can share code/use
|
|
|
|
// gflags.
|
|
|
|
auto command_line = GetCommandLineW();
|
|
|
|
int argc;
|
|
|
|
wchar_t** argv = CommandLineToArgvW(command_line, &argc);
|
|
|
|
if (!argv) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-08-16 23:34:04 +00:00
|
|
|
// Convert all args to narrow, as gflags doesn't support wchar.
|
|
|
|
int argca = argc;
|
2015-08-07 03:17:01 +00:00
|
|
|
char** argva = reinterpret_cast<char**>(alloca(sizeof(char*) * argca));
|
2014-08-16 23:34:04 +00:00
|
|
|
for (int n = 0; n < argca; n++) {
|
2017-05-14 21:29:45 +00:00
|
|
|
size_t len = std::wcstombs(nullptr, argv[n], 0);
|
|
|
|
argva[n] = reinterpret_cast<char*>(alloca(sizeof(char) * (len + 1)));
|
2016-01-01 20:46:26 +00:00
|
|
|
std::wcstombs(argva[n], argv[n], len + 1);
|
2014-08-16 23:34:04 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 19:49:29 +00:00
|
|
|
cvar::ParseLaunchArguments(argca, argva);
|
2014-08-16 23:34:04 +00:00
|
|
|
|
|
|
|
// Widen all remaining flags and convert to usable strings.
|
|
|
|
std::vector<std::wstring> args;
|
|
|
|
for (int n = 0; n < argc; n++) {
|
2017-05-14 21:29:45 +00:00
|
|
|
size_t len = std::mbstowcs(nullptr, argva[n], 0);
|
|
|
|
auto argvw =
|
|
|
|
reinterpret_cast<wchar_t*>(alloca(sizeof(wchar_t) * (len + 1)));
|
|
|
|
std::mbstowcs(argvw, argva[n], len + 1);
|
|
|
|
args.push_back(std::wstring(argvw));
|
2014-08-16 23:34:04 +00:00
|
|
|
}
|
|
|
|
|
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.
|
|
|
|
XELOGI("Build: %s / %s on %s", XE_BUILD_BRANCH, XE_BUILD_COMMIT,
|
|
|
|
XE_BUILD_DATE);
|
|
|
|
|
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();
|
2015-08-01 06:48:24 +00:00
|
|
|
LocalFree(argv);
|
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) {
|
|
|
|
// 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();
|
|
|
|
|
|
|
|
// 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
|