| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | // This needs to be included before getopt.h because the latter #defines symbols used by it
 | 
					
						
							|  |  |  | #include "common/microprofile.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | #ifdef _MSC_VER
 | 
					
						
							|  |  |  | #include <getopt.h>
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #include <getopt.h>
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2016-12-03 21:08:02 +01:00
										 |  |  | #include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "citra/config.h"
 | 
					
						
							|  |  |  | #include "citra/emu_window/emu_window_sdl2.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include "common/logging/backend.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  | #include "common/logging/filter.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  | #include "common/scm_rev.h"
 | 
					
						
							| 
									
										
										
										
											2016-03-15 02:59:14 +00:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-08 13:53:41 +08:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-08 20:15:08 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2015-10-21 22:19:55 -04:00
										 |  |  | #include "core/gdbstub/gdbstub.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-16 18:03:13 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-18 21:21:33 -07:00
										 |  |  | #include "video_core/video_core.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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" | 
					
						
							|  |  |  |                  "-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() { | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  |     std::cout << "Citra " << Common::g_scm_branch << " " << Common::g_scm_desc << std::endl; | 
					
						
							| 
									
										
										
										
											2015-06-22 20:59:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |         LOG_CRITICAL(Frontend, "Failed to get command line arguments"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     std::string filepath; | 
					
						
							| 
									
										
										
										
											2016-04-06 07:01:00 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     static struct option long_options[] = { | 
					
						
							|  |  |  |         {"gdbport", required_argument, 0, 'g'}, | 
					
						
							|  |  |  |         {"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) { | 
					
						
							| 
									
										
										
										
											2016-04-19 02:24:21 +01:00
										 |  |  |         char arg = getopt_long(argc, argv, "g:hv", 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; | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  |     Log::Filter log_filter(Log::Level::Debug); | 
					
						
							| 
									
										
										
										
											2015-03-06 19:15:02 +01:00
										 |  |  |     Log::SetFilter(&log_filter); | 
					
						
							| 
									
										
										
										
											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()) { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02: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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  |     log_filter.ParseFilterString(Settings::values.log_filter); | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00: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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |     std::unique_ptr<EmuWindow_SDL2> emu_window{std::make_unique<EmuWindow_SDL2>()}; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							|  |  |  |         LOG_CRITICAL(Frontend, "Failed to obtain loader for %s!", filepath.c_str()); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     case Core::System::ResultStatus::ErrorLoader: | 
					
						
							|  |  |  |         LOG_CRITICAL(Frontend, "Failed to load ROM!"); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:25:55 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-04-07 00:53:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |