| 
									
										
										
										
											2014-04-08 20:25:53 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-08 20:25:53 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2013-08-29 23:35:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | #include <iostream>
 | 
					
						
							| 
									
										
										
										
											2016-04-05 13:29:55 +01:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 21:39:22 -04:00
										 |  |  | #include "common/common_paths.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-23 21:51:49 -04:00
										 |  |  | #include "common/logging/backend.h"
 | 
					
						
							|  |  |  | #include "common/logging/filter.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-23 21:51:49 -04:00
										 |  |  | #include "common/scm_rev.h"
 | 
					
						
							|  |  |  | #include "common/scope_exit.h"
 | 
					
						
							|  |  |  | #include "common/string_util.h"
 | 
					
						
							|  |  |  | #include "core/core.h"
 | 
					
						
							|  |  |  | #include "core/gdbstub/gdbstub.h"
 | 
					
						
							|  |  |  | #include "core/loader/loader.h"
 | 
					
						
							|  |  |  | #include "core/settings.h"
 | 
					
						
							|  |  |  | #include "yuzu_cmd/config.h"
 | 
					
						
							|  |  |  | #include "yuzu_cmd/emu_window/emu_window_sdl2.h"
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | #include <getopt.h>
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include "core/crypto/key_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-12 18:26:27 +02:00
										 |  |  | #ifndef _MSC_VER
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2017-06-25 02:38:16 +02:00
										 |  |  | // windows.h needs to be included before shellapi.h
 | 
					
						
							| 
									
										
										
										
											2016-12-03 21:08:02 +01:00
										 |  |  | #include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2017-06-25 02:38:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <shellapi.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:22:26 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | // tells Nvidia and AMD drivers to use the dedicated GPU by default on laptops with switchable
 | 
					
						
							|  |  |  | // graphics
 | 
					
						
							|  |  |  | __declspec(dllexport) unsigned long NvOptimusEnablement = 0x00000001; | 
					
						
							|  |  |  | __declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | static void PrintHelp(const char* argv0) { | 
					
						
							|  |  |  |     std::cout << "Usage: " << argv0 | 
					
						
							|  |  |  |               << " [options] <filename>\n" | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  |                  "-g, --gdbport=NUMBER  Enable gdb stub on port NUMBER\n" | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |                  "-f, --fullscreen     Start in fullscreen mode\n" | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  |                  "-h, --help            Display this help and exit\n" | 
					
						
							|  |  |  |                  "-v, --version         Output version information and exit\n"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | static void PrintVersion() { | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |     std::cout << "yuzu " << Common::g_scm_branch << " " << Common::g_scm_desc << std::endl; | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 15:52:42 -04:00
										 |  |  | static void InitializeLogging() { | 
					
						
							|  |  |  |     Log::Filter log_filter(Log::Level::Debug); | 
					
						
							|  |  |  |     log_filter.ParseFilterString(Settings::values.log_filter); | 
					
						
							|  |  |  |     Log::SetGlobalFilter(log_filter); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Log::AddBackend(std::make_unique<Log::ColorConsoleBackend>()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::string& log_dir = FileUtil::GetUserPath(FileUtil::UserPath::LogDir); | 
					
						
							|  |  |  |     FileUtil::CreateFullPath(log_dir); | 
					
						
							|  |  |  |     Log::AddBackend(std::make_unique<Log::FileBackend>(log_dir + LOG_FILE)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-29 23:35:09 -04:00
										 |  |  | /// Application entry point
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | int main(int argc, char** argv) { | 
					
						
							| 
									
										
										
										
											2016-04-06 22:27:28 -04:00
										 |  |  |     Config config; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  |     int option_index = 0; | 
					
						
							| 
									
										
										
										
											2016-04-06 22:27:28 -04:00
										 |  |  |     bool use_gdbstub = Settings::values.use_gdbstub; | 
					
						
							|  |  |  |     u32 gdb_port = static_cast<u32>(Settings::values.gdbstub_port); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     char* endarg; | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |     int argc_w; | 
					
						
							|  |  |  |     auto argv_w = CommandLineToArgvW(GetCommandLineW(), &argc_w); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (argv_w == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Failed to get command line arguments"); | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     std::string filepath; | 
					
						
							| 
									
										
										
										
											2016-04-06 07:01:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |     bool fullscreen = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     static struct option long_options[] = { | 
					
						
							|  |  |  |         {"gdbport", required_argument, 0, 'g'}, | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |         {"fullscreen", no_argument, 0, 'f'}, | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |         {"help", no_argument, 0, 'h'}, | 
					
						
							|  |  |  |         {"version", no_argument, 0, 'v'}, | 
					
						
							|  |  |  |         {0, 0, 0, 0}, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     while (optind < argc) { | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |         char arg = getopt_long(argc, argv, "g:fhv", long_options, &option_index); | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  |         if (arg != -1) { | 
					
						
							|  |  |  |             switch (arg) { | 
					
						
							| 
									
										
										
										
											2016-04-06 07:01:00 -04:00
										 |  |  |             case 'g': | 
					
						
							|  |  |  |                 errno = 0; | 
					
						
							|  |  |  |                 gdb_port = strtoul(optarg, &endarg, 0); | 
					
						
							| 
									
										
										
										
											2016-04-06 22:27:28 -04:00
										 |  |  |                 use_gdbstub = true; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 if (endarg == optarg) | 
					
						
							|  |  |  |                     errno = EINVAL; | 
					
						
							| 
									
										
										
										
											2016-04-06 07:01:00 -04:00
										 |  |  |                 if (errno != 0) { | 
					
						
							|  |  |  |                     perror("--gdbport"); | 
					
						
							|  |  |  |                     exit(1); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |             case 'f': | 
					
						
							|  |  |  |                 fullscreen = true; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |                 LOG_INFO(Frontend, "Starting in fullscreen mode..."); | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  |             case 'h': | 
					
						
							|  |  |  |                 PrintHelp(argv[0]); | 
					
						
							|  |  |  |                 return 0; | 
					
						
							|  |  |  |             case 'v': | 
					
						
							|  |  |  |                 PrintVersion(); | 
					
						
							|  |  |  |                 return 0; | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             filepath = Common::UTF16ToUTF8(argv_w[optind]); | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             filepath = argv[optind]; | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  |             optind++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |     LocalFree(argv_w); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 15:52:42 -04:00
										 |  |  |     InitializeLogging(); | 
					
						
							| 
									
										
										
										
											2013-09-17 22:57:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("EmuThread"); | 
					
						
							| 
									
										
										
										
											2016-03-15 02:59:14 +00:00
										 |  |  |     SCOPE_EXIT({ MicroProfileShutdown(); }); | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     if (filepath.empty()) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Failed to load ROM: No ROM specified"); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2014-05-04 15:47:42 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-17 11:59:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-11 15:20:05 +02:00
										 |  |  |     // Apply the command line arguments
 | 
					
						
							|  |  |  |     Settings::values.gdbstub_port = gdb_port; | 
					
						
							|  |  |  |     Settings::values.use_gdbstub = use_gdbstub; | 
					
						
							|  |  |  |     Settings::Apply(); | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-21 13:22:34 +05:30
										 |  |  |     std::unique_ptr<EmuWindow_SDL2> emu_window{std::make_unique<EmuWindow_SDL2>(fullscreen)}; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 14:10:16 -04:00
										 |  |  |     if (!Settings::values.use_multi_core) { | 
					
						
							|  |  |  |         // Single core mode must acquire OpenGL context for entire emulation session
 | 
					
						
							|  |  |  |         emu_window->MakeCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |     Core::System& system{Core::System::GetInstance()}; | 
					
						
							| 
									
										
										
										
											2016-11-26 23:13:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     SCOPE_EXIT({ system.Shutdown(); }); | 
					
						
							| 
									
										
										
										
											2016-11-19 20:40:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |     const Core::System::ResultStatus load_result{system.Load(emu_window.get(), filepath)}; | 
					
						
							| 
									
										
										
										
											2016-11-19 20:40:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     switch (load_result) { | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorGetLoader: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Failed to obtain loader for %s!", filepath.c_str()); | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorLoader: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Failed to load ROM!"); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |     case Core::System::ResultStatus::ErrorLoader_ErrorEncrypted: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "The game that you are trying to load must be decrypted before " | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                                "being used with yuzu. \n\n For more information on dumping and " | 
					
						
							|  |  |  |                                "decrypting games, please refer to: " | 
					
						
							|  |  |  |                                "https://yuzu-emu.org/wiki/dumping-game-cartridges/"); | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorLoader_ErrorInvalidFormat: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Error while loading ROM: The ROM format is not supported."); | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorNotInitialized: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "CPUCore not initialized"); | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorSystemMode: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Failed to determine system mode!"); | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorVideoCore: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "VideoCore not initialized"); | 
					
						
							| 
									
										
										
										
											2017-01-29 18:09:33 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::Success: | 
					
						
							|  |  |  |         break; // Expected case
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:25:55 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-04-07 00:53:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-22 22:47:56 -04:00
										 |  |  |     Core::Telemetry().AddField(Telemetry::FieldType::App, "Frontend", "SDL"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-15 18:48:02 -07:00
										 |  |  |     while (emu_window->IsOpen()) { | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |         system.RunLoop(); | 
					
						
							| 
									
										
										
										
											2014-08-29 23:24:32 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-08-29 23:35:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-17 11:59:18 -04:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2013-08-29 23:35:09 -04:00
										 |  |  | } |