mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-25 17:55:48 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			205 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018 Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
|  
 | |
| #pragma once
 | |
| #include <switch.h>
 | |
| #include <stratosphere.hpp>
 | |
| #include <memory>
 | |
| #include <mutex>
 | |
| 
 | |
| #define LAUNCHFLAGS_NOTIFYWHENEXITED(flags) (flags & 1)
 | |
| #define LAUNCHFLAGS_STARTSUSPENDED(flags) (flags & (kernelAbove500() ? 0x10 : 0x2))
 | |
| #define LAUNCHFLAGS_ARGLOW(flags) (kernelAbove500() ? ((flags & 0x14) != 0x10) : (kernelAbove200() ? ((flags & 0x6) != 0x2) : ((flags >> 2) & 1)))
 | |
| #define LAUNCHFLAGS_ARGHIGH(flags) ((flags & (kernelAbove500() ? 0x20 : 0x8)) ? 2 : 0)
 | |
| #define LAUNCHFLAGS_NOTIFYDEBUGEVENTS(flags) (flags & (kernelAbove500() ? 0x8 : 0x10))
 | |
| #define LAUNCHFLAGS_NOTIYDEBUGSPECIAL(flags) (flags & (kernelAbove500() ? 0x2 : (kernelAbove200() ? 0x20 : 0x0)))
 | |
| 
 | |
| // none of these names are official or authoritative in any way
 | |
| enum {
 | |
|     /*
 | |
|         set in HandleProcessLaunch when
 | |
|             - launch_flags has LAUNCHFLAGS_NOTIFYWHENEXITED set
 | |
|         signals g_process_event in HandleSignaledProcess when
 | |
|             - process enters Exited state
 | |
|             - we're below 5.0.0
 | |
|             (finalizes dead process when not set)
 | |
|         adds to dead process list in FinalizeExitedProcess when
 | |
|             - we're 5.0.0+
 | |
|             (also signals g_process_event)
 | |
|         [5.0.0+] causes ProcessEventType 2
 | |
|      */
 | |
|     PROCESSFLAGS_NOTIFYWHENEXITED = 0x001,
 | |
| 
 | |
|     /*
 | |
|         set in HandleSignaledProcess when
 | |
|             - process crashes
 | |
|         causes ProcessEventType 1 ([5.0.0+] 3) (persistent)
 | |
|      */
 | |
|     PROCESSFLAGS_CRASHED = 0x002,
 | |
| 
 | |
|     /*
 | |
|         cleared in HandleSignaledProcess when
 | |
|             - process goes from CRASHED to any other state
 | |
|         set in HandleSignaledProcess when
 | |
|             - process crashes
 | |
|         adds process to GetDebugProcessIds
 | |
|      */
 | |
|     PROCESSFLAGS_CRASH_DEBUG = 0x004,
 | |
| 
 | |
|     /*
 | |
|         set in HandleProcessLaunch when
 | |
|             - launch_flags has LAUNCHFLAGS_NOTIFYDEBUGEVENTS set
 | |
|             - and (we're 1.0.0 or program_info.application_type & 4) (is this because 1.0.0 doesn't check?)
 | |
|         signals g_process_event in HandleSignaledProcess when
 | |
|             - process enters DebugDetached state
 | |
|         signals g_process_event in HandleSignaledProcess when
 | |
|             - process enters Running state
 | |
|      */
 | |
|     PROCESSFLAGS_NOTIFYDEBUGEVENTS = 0x008,
 | |
| 
 | |
|     /*
 | |
|         set in HandleSignaledProcess when
 | |
|             - process enters DebugDetached state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|         set in HandleSignaledProcess when
 | |
|             - process enters Running state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|         set in HandleSignaledProcess when
 | |
|             - process enters DebugSuspended state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|         causes some complicated ProcessEvent (one-shot)
 | |
|      */
 | |
|     PROCESSFLAGS_DEBUGEVENTPENDING = 0x010,
 | |
| 
 | |
|     /*
 | |
|         cleared in HandleSignaledProcess when
 | |
|             - process enters DebugDetached state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|         cleared in HandleSignaledProcess when
 | |
|             - process enters Running state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|         set in HandleSignaledProcess when
 | |
|             - process enters DebugSuspended state
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGEVENTS is set
 | |
|      */
 | |
|     PROCESSFLAGS_DEBUGSUSPENDED = 0x020,
 | |
|     
 | |
|     /*
 | |
|         set in HandleProcessLaunch when
 | |
|             - program_info.application_type & 1
 | |
|         signals g_debug_application_event in HandleProcessLaunch when
 | |
|             - g_debug_next_application is set (unsets g_debug_next_application)
 | |
|         causes HasApplicationProcess to return true
 | |
|         meaning?
 | |
|             application
 | |
|     */
 | |
|     PROCESSFLAGS_APPLICATION = 0x040,
 | |
| 
 | |
|     /*
 | |
|         set in HandleProcessLaunch when
 | |
|             - we're above 2.0.0 (creport related?)
 | |
|             - and launch_flags has LAUNCHFLAGS_NOTIFYDEBUGSPECIAL set
 | |
|             - and program_info.application_type & 4
 | |
|         tested in HandleSignaledProcess when
 | |
|             - process enters DebugDetached state
 | |
|             - and we're above 2.0.0
 | |
|             causes
 | |
|             - clearing of PROCESSFLAGS_NOTIFYDEBUGSPECIAL (one-shot?)
 | |
|             - setting of PROCESSFLAGS_DEBUGDETACHED
 | |
|      */
 | |
|     PROCESSFLAGS_NOTIFYDEBUGSPECIAL = 0x080,
 | |
| 
 | |
|     /*
 | |
|         set in HandleSignaledProcess when
 | |
|             - process enters DebugDetached state
 | |
|             - and we're above 2.0.0
 | |
|             - and PROCESSFLAGS_NOTIFYDEBUGSPECIAL was set
 | |
|         causes ProcessEventType 5 ([5.0.0+] 2) (one-shot)
 | |
|      */
 | |
|     PROCESSFLAGS_DEBUGDETACHED = 0x100,
 | |
| };
 | |
| 
 | |
| enum {
 | |
|     PROCESSEVENTTYPE_CRASH = 1,
 | |
|     PROCESSEVENTTYPE_EXIT = 2, // only fired once, when process enters DebugDetached state (likely creport related)
 | |
|     PROCESSEVENTTYPE_RUNNING = 3, // debug detached or running
 | |
|     PROCESSEVENTTYPE_SUSPENDED = 4, // debug suspended
 | |
|     PROCESSEVENTTYPE_DEBUGDETACHED = 5,
 | |
| 
 | |
| 
 | |
|     PROCESSEVENTTYPE_500_EXIT = 1,
 | |
|     PROCESSEVENTTYPE_500_DEBUGDETACHED = 2, // only fired once, when process enters DebugDetached state (likely creport related)
 | |
|     PROCESSEVENTTYPE_500_CRASH = 3,
 | |
|     PROCESSEVENTTYPE_500_RUNNING = 4, // debug detached or running
 | |
|     PROCESSEVENTTYPE_500_SUSPENDED = 5, // debug suspended
 | |
| };
 | |
|     
 | |
| class Registration {
 | |
|     public:
 | |
|         struct TidSid {
 | |
|             u64 title_id;
 | |
|             FsStorageId storage_id;
 | |
|         };
 | |
|         struct Process {
 | |
|             Handle handle;
 | |
|             u64 pid;
 | |
|             u64 ldr_queue_index;
 | |
|             Registration::TidSid tid_sid;
 | |
|             ProcessState state;
 | |
|             u32 flags;
 | |
|         };
 | |
|         
 | |
|         struct ProcessLaunchState {
 | |
|             TidSid tid_sid;
 | |
|             u64 launch_flags;
 | |
|             u64* out_pid;
 | |
|             Result result;
 | |
|         };
 | |
| 
 | |
|         static void InitializeSystemResources();
 | |
|         static IWaitable *GetProcessLaunchStartEvent();
 | |
|         static std::unique_lock<HosRecursiveMutex> GetProcessListUniqueLock();
 | |
|         static void SetProcessListManager(WaitableManager *m);
 | |
|         static Result ProcessLaunchStartCallback(void *arg, Handle *handles, size_t num_handles, u64 timeout);
 | |
|         
 | |
|         static Result HandleSignaledProcess(std::shared_ptr<Process> process);
 | |
|         static void FinalizeExitedProcess(std::shared_ptr<Process> process);
 | |
|         
 | |
|         static void AddProcessToList(std::shared_ptr<Process> process);
 | |
|         static void RemoveProcessFromList(u64 pid);
 | |
|         static void SetProcessState(u64 pid, ProcessState new_state);
 | |
|         
 | |
|         static std::shared_ptr<Registration::Process> GetProcess(u64 pid);
 | |
|         static std::shared_ptr<Registration::Process> GetProcessByTitleId(u64 tid);
 | |
|         static Result GetDebugProcessIds(u64 *out_pids, u32 max_out, u32 *num_out);
 | |
|         static Handle GetProcessEventHandle();
 | |
|         static void GetProcessEventType(u64 *out_pid, u64 *out_type);
 | |
|         static Result EnableDebugForTitleId(u64 tid, Handle *out);
 | |
|         static Result EnableDebugForApplication(Handle *out);
 | |
|         static Result DisableDebug(u32 which);
 | |
|         static Handle GetDebugTitleEventHandle();
 | |
|         static Handle GetDebugApplicationEventHandle();
 | |
|         
 | |
|         static void HandleProcessLaunch();
 | |
|         static Result LaunchDebugProcess(u64 pid);
 | |
|         static void SignalFinishLaunchProcess();
 | |
|         static Result LaunchProcess(u64 title_id, FsStorageId storage_id, u64 launch_flags, u64 *out_pid);
 | |
|         static Result LaunchProcessByTidSid(TidSid tid_sid, u64 launch_flags, u64 *out_pid);
 | |
|         
 | |
|         static bool HasApplicationProcess(std::shared_ptr<Process> *out);
 | |
| };
 | |
| 
 |