mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2024-12-23 16:03:05 +00:00
339 lines
No EOL
12 KiB
C++
339 lines
No EOL
12 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/>.
|
|
*/
|
|
|
|
#include <switch.h>
|
|
#include <algorithm>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <functional>
|
|
#include "ldr_registration.hpp"
|
|
#include "ldr_nro.hpp"
|
|
|
|
static Registration::List g_registration_list = {0};
|
|
static u64 g_num_registered = 1;
|
|
|
|
Registration::Process *Registration::GetFreeProcess() {
|
|
auto process_it = std::find_if_not(g_registration_list.processes.begin(), g_registration_list.processes.end(), std::mem_fn(&Registration::Process::in_use));
|
|
if (process_it == g_registration_list.processes.end()) {
|
|
return nullptr;
|
|
}
|
|
return &*process_it;
|
|
}
|
|
|
|
Registration::Process *Registration::GetProcess(u64 index) {
|
|
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];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
Registration::Process *Registration::GetProcessByProcessId(u64 pid) {
|
|
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];
|
|
}
|
|
}
|
|
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];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool Registration::RegisterTidSid(const TidSid *tid_sid, u64 *out_index) {
|
|
Registration::Process *free_process = GetFreeProcess();
|
|
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;
|
|
}
|
|
|
|
bool Registration::UnregisterIndex(u64 index) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return false;
|
|
}
|
|
|
|
/* Reset the process. */
|
|
*target_process = {0};
|
|
return true;
|
|
}
|
|
|
|
|
|
Result Registration::GetRegisteredTidSid(u64 index, Registration::TidSid *out) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return 0x1009;
|
|
}
|
|
|
|
*out = target_process->tid_sid;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void Registration::SetProcessIdTidAndIs64BitAddressSpace(u64 index, u64 process_id, u64 tid, bool is_64_bit_addspace) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return;
|
|
}
|
|
|
|
target_process->process_id = process_id;
|
|
target_process->title_id = tid;
|
|
target_process->is_64_bit_addspace = is_64_bit_addspace;
|
|
}
|
|
|
|
void Registration::AddNsoInfo(u64 index, u64 base_address, u64 size, const unsigned char *build_id) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return;
|
|
}
|
|
|
|
auto nso_info_it = std::find_if_not(target_process->nso_infos.begin(), target_process->nso_infos.end(), std::mem_fn(&Registration::NsoInfoHolder::in_use));
|
|
if (nso_info_it != target_process->nso_infos.end()) {
|
|
nso_info_it->info.base_address = base_address;
|
|
nso_info_it->info.size = size;
|
|
std::copy(build_id, build_id + sizeof(nso_info_it->info.build_id), nso_info_it->info.build_id);
|
|
nso_info_it->in_use = true;
|
|
}
|
|
}
|
|
|
|
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();
|
|
}
|
|
}
|
|
target_process->owner_ro_service = NULL;
|
|
}
|
|
|
|
Result Registration::AddNrrInfo(u64 index, MappedCodeMemory *nrr_info) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
/* TODO: panic() */
|
|
return 0x7009;
|
|
}
|
|
|
|
auto nrr_info_it = std::find_if_not(target_process->nrr_infos.begin(), target_process->nrr_infos.end(), std::mem_fn(&MappedCodeMemory::IsActive));
|
|
if (nrr_info_it == target_process->nrr_infos.end()) {
|
|
return 0x7009;
|
|
}
|
|
*nrr_info_it = *nrr_info;
|
|
return 0;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
|
|
bool Registration::IsNroHashPresent(u64 index, u8 *nro_hash) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
/* TODO: panic */
|
|
return false;
|
|
}
|
|
|
|
for (unsigned int i = 0; i < NRR_INFO_MAX; i++) {
|
|
if (target_process->nrr_infos[i].IsActive()) {
|
|
NroUtils::NrrHeader *nrr = (NroUtils::NrrHeader *)target_process->nrr_infos[i].mapped_address;
|
|
/* Binary search. */
|
|
int low = 0, high = (int)(nrr->num_hashes - 1);
|
|
while (low <= high) {
|
|
int mid = (low + high) / 2;
|
|
u8 *hash_in_nrr = (u8 *)nrr + nrr->hash_offset + 0x20 * mid;
|
|
int ret = std::memcmp(hash_in_nrr, nro_hash, 0x20);
|
|
if (ret == 0) {
|
|
return true;
|
|
} else if (ret > 0) {
|
|
high = mid - 1;
|
|
} else {
|
|
low = mid + 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool Registration::IsNroAlreadyLoaded(u64 index, u8 *build_id) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
/* TODO: panic */
|
|
return true;
|
|
}
|
|
|
|
for (unsigned int i = 0; i < NRO_INFO_MAX; i++) {
|
|
if (target_process->nro_infos[i].in_use && std::equal(build_id, build_id + 0x20, target_process->nro_infos[i].build_id)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void Registration::AddNroToProcess(u64 index, MappedCodeMemory *nro, MappedCodeMemory *bss, u32 text_size, u32 ro_size, u32 rw_size, u8 *build_id) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
/* TODO: panic */
|
|
return;
|
|
}
|
|
|
|
auto nro_info_it = std::find_if_not(target_process->nro_infos.begin(), target_process->nro_infos.end(), std::mem_fn(&Registration::NroInfo::in_use));
|
|
if (nro_info_it != target_process->nro_infos.end()) {
|
|
nro_info_it->base_address = nro->code_memory_address;
|
|
nro_info_it->nro_heap_address = nro->base_address;
|
|
nro_info_it->nro_heap_size = nro->size;
|
|
nro_info_it->bss_heap_address = bss->base_address;
|
|
nro_info_it->bss_heap_size = bss->size;
|
|
nro_info_it->text_size = text_size;
|
|
nro_info_it->ro_size = ro_size;
|
|
nro_info_it->rw_size = rw_size;
|
|
std::copy(build_id, build_id + sizeof(nro_info_it->build_id), nro_info_it->build_id);
|
|
nro_info_it->in_use = true;
|
|
}
|
|
}
|
|
|
|
Result Registration::RemoveNroInfo(u64 index, Handle process_h, u64 nro_heap_address) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return 0xA809;
|
|
}
|
|
|
|
for (unsigned int i = 0; i < NRO_INFO_MAX; i++) {
|
|
if (target_process->nro_infos[i].in_use && target_process->nro_infos[i].nro_heap_address == nro_heap_address) {
|
|
NroInfo *info = &target_process->nro_infos[i];
|
|
Result rc = svcUnmapProcessCodeMemory(process_h, info->base_address + info->text_size + info->ro_size + info->rw_size, info->bss_heap_address, info->bss_heap_size);
|
|
if (R_SUCCEEDED(rc)) {
|
|
rc = svcUnmapProcessCodeMemory(process_h, info->base_address + info->text_size + info->ro_size, nro_heap_address + info->text_size + info->ro_size, info->rw_size);
|
|
if (R_SUCCEEDED(rc)) {
|
|
rc = svcUnmapProcessCodeMemory(process_h, info->base_address, nro_heap_address, info->text_size + info->ro_size);
|
|
}
|
|
}
|
|
target_process->nro_infos[i] = (const NroInfo ){0};
|
|
return rc;
|
|
}
|
|
}
|
|
return 0xA809;
|
|
}
|
|
|
|
Result Registration::GetNsoInfosForProcessId(Registration::NsoInfo *out, u32 max_out, u64 process_id, u32 *num_written) {
|
|
Registration::Process *target_process = GetProcessByProcessId(process_id);
|
|
if (target_process == NULL) {
|
|
return 0x1009;
|
|
}
|
|
u32 cur = 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;
|
|
}
|
|
|
|
void Registration::AssociatePidTidForMitM(u64 index) {
|
|
Registration::Process *target_process = GetProcess(index);
|
|
if (target_process == NULL) {
|
|
return;
|
|
}
|
|
|
|
Handle sm_hnd;
|
|
Result rc = svcConnectToNamedPort(&sm_hnd, "sm:");
|
|
if (R_SUCCEEDED(rc)) {
|
|
/* Initialize. */
|
|
{
|
|
IpcCommand c;
|
|
ipcInitialize(&c);
|
|
ipcSendPid(&c);
|
|
|
|
struct {
|
|
u64 magic;
|
|
u64 cmd_id;
|
|
u64 zero;
|
|
u64 reserved[2];
|
|
} *raw = (decltype(raw))ipcPrepareHeader(&c, sizeof(*raw));
|
|
|
|
raw->magic = SFCI_MAGIC;
|
|
raw->cmd_id = 0;
|
|
raw->zero = 0;
|
|
|
|
rc = ipcDispatch(sm_hnd);
|
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
IpcParsedCommand r;
|
|
ipcParse(&r);
|
|
|
|
struct {
|
|
u64 magic;
|
|
u64 result;
|
|
} *resp = (decltype(resp))r.Raw;
|
|
|
|
rc = resp->result;
|
|
}
|
|
}
|
|
/* Associate. */
|
|
if (R_SUCCEEDED(rc)) {
|
|
IpcCommand c;
|
|
ipcInitialize(&c);
|
|
struct {
|
|
u64 magic;
|
|
u64 cmd_id;
|
|
u64 process_id;
|
|
u64 title_id;
|
|
} *raw = (decltype(raw))ipcPrepareHeader(&c, sizeof(*raw));
|
|
|
|
raw->magic = SFCI_MAGIC;
|
|
raw->cmd_id = 65002;
|
|
raw->process_id = target_process->process_id;
|
|
raw->title_id = target_process->tid_sid.title_id;
|
|
|
|
ipcDispatch(sm_hnd);
|
|
}
|
|
svcCloseHandle(sm_hnd);
|
|
}
|
|
} |