2018-04-19 05:00:10 +00:00
|
|
|
#include <switch.h>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cstdio>
|
|
|
|
#include "ldr_registration.hpp"
|
|
|
|
|
|
|
|
static Registration::List g_registration_list = {0};
|
|
|
|
static u64 g_num_registered = 1;
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
Registration::Process *Registration::GetFreeProcess() {
|
2018-04-27 02:50:27 +00:00
|
|
|
for (unsigned int i = 0; i < REGISTRATION_LIST_MAX; i++) {
|
2018-04-19 05:00:10 +00:00
|
|
|
if (!g_registration_list.processes[i].in_use) {
|
|
|
|
return &g_registration_list.processes[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
Registration::Process *Registration::GetProcess(u64 index) {
|
2018-04-27 02:50:27 +00:00
|
|
|
for (unsigned int i = 0; i < REGISTRATION_LIST_MAX; i++) {
|
|
|
|
if (g_registration_list.processes[i].in_use && g_registration_list.processes[i].index == index) {
|
|
|
|
return &g_registration_list.processes[i];
|
|
|
|
}
|
2018-04-19 05:00:10 +00:00
|
|
|
}
|
2018-04-27 02:50:27 +00:00
|
|
|
return NULL;
|
2018-04-19 05:00:10 +00:00
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
Registration::Process *Registration::GetProcessByProcessId(u64 pid) {
|
2018-04-27 02:50:27 +00:00
|
|
|
for (unsigned int i = 0; i < REGISTRATION_LIST_MAX; i++) {
|
|
|
|
if (g_registration_list.processes[i].in_use && g_registration_list.processes[i].process_id == pid) {
|
|
|
|
return &g_registration_list.processes[i];
|
|
|
|
}
|
2018-04-21 01:34:29 +00:00
|
|
|
}
|
2018-04-27 02:50:27 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
Registration::Process *Registration::GetProcessByRoService(void *service) {
|
|
|
|
for (unsigned int i = 0; i < REGISTRATION_LIST_MAX; i++) {
|
|
|
|
if (g_registration_list.processes[i].in_use && g_registration_list.processes[i].owner_ro_service == service) {
|
|
|
|
return &g_registration_list.processes[i];
|
|
|
|
}
|
2018-04-19 22:14:48 +00:00
|
|
|
}
|
2018-04-27 02:50:27 +00:00
|
|
|
return NULL;
|
2018-04-19 22:14:48 +00:00
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
bool Registration::RegisterTidSid(const TidSid *tid_sid, u64 *out_index) {
|
|
|
|
Registration::Process *free_process = GetFreeProcess();
|
2018-04-19 05:00:10 +00:00
|
|
|
if (free_process == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reset the process. */
|
|
|
|
*free_process = {0};
|
|
|
|
free_process->tid_sid = *tid_sid;
|
|
|
|
free_process->in_use = true;
|
|
|
|
free_process->index = g_num_registered++;
|
|
|
|
*out_index = free_process->index;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
bool Registration::UnregisterIndex(u64 index) {
|
|
|
|
Registration::Process *target_process = GetProcess(index);
|
2018-04-19 05:00:10 +00:00
|
|
|
if (target_process == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reset the process. */
|
|
|
|
*target_process = {0};
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-22 01:52:49 +00:00
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
Result Registration::GetRegisteredTidSid(u64 index, Registration::TidSid *out) {
|
|
|
|
Registration::Process *target_process = GetProcess(index);
|
2018-04-22 01:52:49 +00:00
|
|
|
if (target_process == NULL) {
|
|
|
|
return 0x1009;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = target_process->tid_sid;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
void Registration::SetProcessIdTidMinAndIs64BitAddressSpace(u64 index, u64 process_id, u64 tid_min, bool is_64_bit_addspace) {
|
|
|
|
Registration::Process *target_process = GetProcess(index);
|
2018-04-19 05:00:10 +00:00
|
|
|
if (target_process == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
target_process->process_id = process_id;
|
|
|
|
target_process->title_id_min = tid_min;
|
2018-04-26 20:53:33 +00:00
|
|
|
target_process->is_64_bit_addspace = is_64_bit_addspace;
|
2018-04-19 05:00:10 +00:00
|
|
|
}
|
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
void Registration::AddNsoInfo(u64 index, u64 base_address, u64 size, const unsigned char *build_id) {
|
|
|
|
Registration::Process *target_process = GetProcess(index);
|
2018-04-19 05:00:10 +00:00
|
|
|
if (target_process == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < NSO_INFO_MAX; i++) {
|
|
|
|
if (!target_process->nso_infos[i].in_use) {
|
2018-04-19 22:14:48 +00:00
|
|
|
target_process->nso_infos[i].info.base_address = base_address;
|
|
|
|
target_process->nso_infos[i].info.size = size;
|
|
|
|
std::copy(build_id, build_id + sizeof(target_process->nso_infos[i].info.build_id), target_process->nso_infos[i].info.build_id);
|
2018-04-19 05:00:10 +00:00
|
|
|
target_process->nso_infos[i].in_use = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2018-04-19 22:14:48 +00:00
|
|
|
}
|
|
|
|
|
2018-04-27 02:50:27 +00:00
|
|
|
void Registration::CloseRoService(void *service, Handle process_h) {
|
|
|
|
Registration::Process *target_process = GetProcessByRoService(service);
|
|
|
|
if (target_process == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (unsigned int i = 0; i < NRR_INFO_MAX; i++) {
|
|
|
|
if (target_process->nrr_infos[i].IsActive() && target_process->nrr_infos[i].process_handle == process_h) {
|
|
|
|
target_process->nrr_infos[i].Close();
|
|
|
|
}
|
|
|
|
}
|
2018-04-27 02:51:12 +00:00
|
|
|
target_process->owner_ro_service = NULL;
|
2018-04-27 02:50:27 +00:00
|
|
|
}
|
2018-04-19 22:14:48 +00:00
|
|
|
|
2018-04-27 02:27:52 +00:00
|
|
|
Result Registration::AddNrrInfo(u64 index, MappedCodeMemory *nrr_info) {
|
2018-04-27 01:13:55 +00:00
|
|
|
Registration::Process *target_process = GetProcess(index);
|
|
|
|
if (target_process == NULL) {
|
|
|
|
/* TODO: panic() */
|
|
|
|
return 0x7009;
|
|
|
|
}
|
|
|
|
|
2018-04-27 02:37:38 +00:00
|
|
|
for (unsigned int i = 0; i < NRR_INFO_MAX; i++) {
|
2018-04-27 02:27:52 +00:00
|
|
|
if (!target_process->nrr_infos[i].IsActive()) {
|
|
|
|
target_process->nrr_infos[i] = *nrr_info;
|
2018-04-27 01:13:55 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0x7009;
|
|
|
|
}
|
|
|
|
|
2018-04-27 02:37:38 +00:00
|
|
|
Result Registration::RemoveNrrInfo(u64 index, u64 base_address) {
|
|
|
|
Registration::Process *target_process = GetProcess(index);
|
|
|
|
if (target_process == NULL) {
|
|
|
|
/* Despite the fact that this should really be a panic condition, Nintendo returns 0x1009 in this case. */
|
|
|
|
return 0x1009;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < NRR_INFO_MAX; i++) {
|
|
|
|
if (target_process->nrr_infos[i].IsActive() && target_process->nrr_infos[i].base_address == base_address) {
|
|
|
|
target_process->nrr_infos[i].Close();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0xAA09;
|
|
|
|
}
|
2018-04-27 01:13:55 +00:00
|
|
|
|
2018-04-26 23:03:10 +00:00
|
|
|
Result Registration::GetNsoInfosForProcessId(Registration::NsoInfo *out, u32 max_out, u64 process_id, u32 *num_written) {
|
|
|
|
Registration::Process *target_process = GetProcessByProcessId(process_id);
|
2018-04-19 22:14:48 +00:00
|
|
|
if (target_process == NULL) {
|
|
|
|
return 0x1009;
|
|
|
|
}
|
|
|
|
u32 cur = 0;
|
|
|
|
|
|
|
|
if (max_out > 0) {
|
|
|
|
for (unsigned int i = 0; i < NSO_INFO_MAX && cur < max_out; i++) {
|
|
|
|
if (target_process->nso_infos[i].in_use) {
|
|
|
|
out[cur++] = target_process->nso_infos[i].info;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*num_written = cur;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|