| 
									
										
										
										
											2014-04-08 19:15:46 -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 19:15:46 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2013-09-05 18:33:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-08 19:15:46 -04:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2013-09-05 18:33:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-29 18:03:08 -05:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:30:09 -02:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:30:09 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | class EmuWindow; | 
					
						
							| 
									
										
										
										
											2014-12-22 04:30:09 -02:00
										 |  |  | class ARM_Interface; | 
					
						
							| 
									
										
										
										
											2014-04-03 21:22:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | namespace Loader { | 
					
						
							|  |  |  | class AppLoader; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-09-05 18:33:46 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-22 04:30:09 -02:00
										 |  |  | struct ThreadContext { | 
					
						
							|  |  |  |     u32 cpu_registers[13]; | 
					
						
							|  |  |  |     u32 sp; | 
					
						
							|  |  |  |     u32 lr; | 
					
						
							|  |  |  |     u32 pc; | 
					
						
							|  |  |  |     u32 cpsr; | 
					
						
							| 
									
										
										
										
											2015-11-29 13:06:45 -05:00
										 |  |  |     u32 fpu_registers[64]; | 
					
						
							| 
									
										
										
										
											2014-12-22 04:30:09 -02:00
										 |  |  |     u32 fpscr; | 
					
						
							|  |  |  |     u32 fpexc; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | class System { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Gets the instance of the System singleton class. | 
					
						
							|  |  |  |      * @returns Reference to the instance of the System singleton class. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     static System& GetInstance() { | 
					
						
							|  |  |  |         return s_instance; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Enumeration representing the return values of the System Initialize and Load process.
 | 
					
						
							|  |  |  |     enum class ResultStatus : u32 { | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |         Success,                    ///< Succeeded
 | 
					
						
							|  |  |  |         ErrorNotInitialized,        ///< Error trying to use core prior to initialization
 | 
					
						
							|  |  |  |         ErrorGetLoader,             ///< Error finding the correct application loader
 | 
					
						
							|  |  |  |         ErrorSystemMode,            ///< Error determining the system mode
 | 
					
						
							|  |  |  |         ErrorLoader,                ///< Error loading the specified application
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |         ErrorLoader_ErrorEncrypted, ///< Error loading the specified application due to encryption
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |         ErrorLoader_ErrorInvalidFormat, ///< Error loading the specified application due to an
 | 
					
						
							|  |  |  |                                         /// invalid format
 | 
					
						
							|  |  |  |         ErrorVideoCore,                 ///< Error in the video core
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Run the core CPU loop | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |      * This function runs the core for the specified number of CPU instructions before trying to | 
					
						
							|  |  |  |      * update hardware. This is much faster than SingleStep (and should be equivalent), as the CPU | 
					
						
							|  |  |  |      * is not required to do a full dispatch with each instruction. NOTE: the number of instructions | 
					
						
							|  |  |  |      * requested is not guaranteed to run, as this will be interrupted preemptively if a hardware | 
					
						
							|  |  |  |      * update is requested (e.g. on a thread switch). | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |      * @param tight_loop Number of instructions to execute. | 
					
						
							|  |  |  |      * @return Result status, indicating whethor or not the operation succeeded. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultStatus RunLoop(int tight_loop = 1000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Step the CPU one instruction | 
					
						
							|  |  |  |      * @return Result status, indicating whethor or not the operation succeeded. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultStatus SingleStep(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Shutdown the emulated system.
 | 
					
						
							|  |  |  |     void Shutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Load an executable application. | 
					
						
							|  |  |  |      * @param emu_window Pointer to the host-system window used for video output and keyboard input. | 
					
						
							|  |  |  |      * @param filepath String path to the executable application to load on the host file system. | 
					
						
							|  |  |  |      * @returns ResultStatus code, indicating if the operation succeeded. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultStatus Load(EmuWindow* emu_window, const std::string& filepath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Indicates if the emulated system is powered on (all subsystems initialized and able to run an | 
					
						
							|  |  |  |      * application). | 
					
						
							|  |  |  |      * @returns True if the emulated system is powered on, otherwise false. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     bool IsPoweredOn() const { | 
					
						
							|  |  |  |         return app_core != nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-16 00:37:38 -05:00
										 |  |  |     /// Prepare the core emulation for a reschedule
 | 
					
						
							|  |  |  |     void PrepareReschedule(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Gets a reference to the emulated AppCore CPU. | 
					
						
							|  |  |  |      * @returns A reference to the emulated AppCore CPU. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ARM_Interface& AppCore() { | 
					
						
							|  |  |  |         return *app_core; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2016-12-16 00:37:38 -05:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Initialize the emulated system. | 
					
						
							|  |  |  |      * @param emu_window Pointer to the host-system window used for video output and keyboard input. | 
					
						
							|  |  |  |      * @param system_mode The system mode. | 
					
						
							|  |  |  |      * @return ResultStatus code, indicating if the operation succeeded. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ResultStatus Init(EmuWindow* emu_window, u32 system_mode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Reschedule the core emulation
 | 
					
						
							|  |  |  |     void Reschedule(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |     /// AppLoader used to load the current executing application
 | 
					
						
							|  |  |  |     std::unique_ptr<Loader::AppLoader> app_loader; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ///< ARM11 application core
 | 
					
						
							|  |  |  |     std::unique_ptr<ARM_Interface> app_core; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-16 00:37:38 -05:00
										 |  |  |     /// When true, signals that a reschedule should happen
 | 
					
						
							|  |  |  |     bool reschedule_pending{}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |     static System s_instance; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-09-05 18:33:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | static ARM_Interface& AppCore() { | 
					
						
							|  |  |  |     return System::GetInstance().AppCore(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-04-05 00:01:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  | } // namespace Core
 |