mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2024-11-10 15:16:34 +00:00
214 lines
8.4 KiB
C++
214 lines
8.4 KiB
C++
/*
|
|
* Copyright (c) 2018-2019 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>
|
|
|
|
class ProcessList;
|
|
|
|
#define LAUNCHFLAGS_NOTIFYWHENEXITED(flags) (flags & 1)
|
|
#define LAUNCHFLAGS_STARTSUSPENDED(flags) (flags & (GetRuntimeFirmwareVersion() >= FirmwareVersion_500 ? 0x10 : 0x2))
|
|
#define LAUNCHFLAGS_ARGLOW(flags) (GetRuntimeFirmwareVersion() >= FirmwareVersion_500 ? ((flags & 0x14) != 0x10) : (GetRuntimeFirmwareVersion() >= FirmwareVersion_200 ? ((flags & 0x6) != 0x2) : ((flags >> 2) & 1)))
|
|
#define LAUNCHFLAGS_ARGHIGH(flags) ((flags & (GetRuntimeFirmwareVersion() >= FirmwareVersion_500 ? 0x20 : 0x8)) ? 2 : 0)
|
|
#define LAUNCHFLAGS_NOTIFYDEBUGEVENTS(flags) (flags & (GetRuntimeFirmwareVersion() >= FirmwareVersion_500 ? 0x8 : 0x10))
|
|
#define LAUNCHFLAGS_NOTIYDEBUGSPECIAL(flags) (flags & (GetRuntimeFirmwareVersion() >= FirmwareVersion_500 ? 0x2 : (GetRuntimeFirmwareVersion() >= FirmwareVersion_200 ? 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;
|
|
};
|
|
private:
|
|
static Result LaunchProcess(u64 *out_pid, const TidSid tid_sid, const u64 launch_flags);
|
|
public:
|
|
/* TODO: Better evaluate public vs private API. */
|
|
static void InitializeSystemResources();
|
|
static IWaitable *GetProcessLaunchStartEvent();
|
|
static ProcessList &GetProcessList();
|
|
static Result ProcessLaunchStartCallback(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 Handle GetBootFinishedEventHandle();
|
|
|
|
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 HasLaunchedTitle(u64 title_id);
|
|
|
|
static void SignalBootFinished();
|
|
|
|
static bool HasApplicationProcess(std::shared_ptr<Process> *out = nullptr);
|
|
};
|
|
|
|
#include "pm_process_wait.hpp"
|