2018-06-09 19:33:22 -06:00
|
|
|
#include <switch.h>
|
|
|
|
#include "fsmitm_service.hpp"
|
2018-06-10 11:11:05 -06:00
|
|
|
#include "fs_shim.h"
|
|
|
|
|
|
|
|
#include "fsmitm_worker.hpp"
|
|
|
|
#include "fsmitm_utils.hpp"
|
|
|
|
#include "fsmitm_romstorage.hpp"
|
2018-06-14 17:50:01 -06:00
|
|
|
#include "fsmitm_layeredrom.hpp"
|
2018-06-10 11:11:05 -06:00
|
|
|
|
2018-06-14 17:50:01 -06:00
|
|
|
#include "mitm_query_service.hpp"
|
2018-06-10 11:11:05 -06:00
|
|
|
#include "debug.hpp"
|
2018-06-09 19:33:22 -06:00
|
|
|
|
|
|
|
Result FsMitMService::dispatch(IpcParsedCommand &r, IpcCommand &out_c, u64 cmd_id, u8 *pointer_buffer, size_t pointer_buffer_size) {
|
|
|
|
Result rc = 0xF601;
|
2018-06-14 17:50:01 -06:00
|
|
|
if (this->has_initialized) {
|
|
|
|
switch (cmd_id) {
|
|
|
|
case FspSrv_Cmd_OpenDataStorageByCurrentProcess:
|
|
|
|
rc = WrapIpcCommandImpl<&FsMitMService::open_data_storage_by_current_process>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
case FspSrv_Cmd_OpenDataStorageByDataId:
|
|
|
|
rc = WrapIpcCommandImpl<&FsMitMService::open_data_storage_by_data_id>(this, r, out_c, pointer_buffer, pointer_buffer_size);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cmd_id == FspSrv_Cmd_SetCurrentProcess) {
|
|
|
|
if (r.HasPid) {
|
|
|
|
this->init_pid = r.Pid;
|
2018-06-10 03:06:50 -06:00
|
|
|
}
|
2018-06-14 17:50:01 -06:00
|
|
|
}
|
2018-06-10 03:06:50 -06:00
|
|
|
}
|
2018-06-10 01:17:00 -06:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2018-06-12 16:00:09 -06:00
|
|
|
void FsMitMService::postprocess(IpcParsedCommand &r, IpcCommand &out_c, u64 cmd_id, u8 *pointer_buffer, size_t pointer_buffer_size) {
|
2018-06-10 01:19:21 -06:00
|
|
|
struct {
|
|
|
|
u64 magic;
|
|
|
|
u64 result;
|
|
|
|
} *resp = (decltype(resp))r.Raw;
|
|
|
|
|
2018-06-12 16:00:09 -06:00
|
|
|
u64 *tls = (u64 *)armGetTls();
|
|
|
|
u64 backup_tls[0x100/sizeof(u64)];
|
|
|
|
for (unsigned int i = 0; i < sizeof(backup_tls)/sizeof(u64); i++) {
|
|
|
|
backup_tls[i] = tls[i];
|
|
|
|
}
|
|
|
|
|
2018-06-10 01:19:21 -06:00
|
|
|
Result rc = (Result)resp->result;
|
2018-06-10 03:06:50 -06:00
|
|
|
switch (cmd_id) {
|
|
|
|
case FspSrv_Cmd_SetCurrentProcess:
|
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
|
|
this->has_initialized = true;
|
2018-06-12 16:00:09 -06:00
|
|
|
}
|
2018-06-14 17:50:01 -06:00
|
|
|
this->process_id = this->init_pid;
|
|
|
|
this->title_id = this->process_id;
|
|
|
|
if (R_FAILED(MitMQueryUtils::get_associated_tid_for_pid(this->process_id, &this->title_id))) {
|
|
|
|
/* Log here, if desired. */
|
|
|
|
}
|
2018-06-12 16:00:09 -06:00
|
|
|
for (unsigned int i = 0; i < sizeof(backup_tls)/sizeof(u64); i++) {
|
|
|
|
tls[i] = backup_tls[i];
|
|
|
|
}
|
2018-06-10 03:06:50 -06:00
|
|
|
break;
|
|
|
|
}
|
2018-06-12 16:00:09 -06:00
|
|
|
resp->result = rc;
|
2018-06-09 19:33:22 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
Result FsMitMService::handle_deferred() {
|
|
|
|
/* This service is never deferrable. */
|
|
|
|
return 0;
|
2018-06-10 11:11:05 -06:00
|
|
|
}
|
|
|
|
|
2018-06-14 17:50:01 -06:00
|
|
|
/* Add redirection for RomFS to the SD card. */
|
|
|
|
std::tuple<Result, OutSession<IStorageInterface>> FsMitMService::open_data_storage_by_current_process() {
|
|
|
|
IPCSession<IStorageInterface> *out_session = NULL;
|
|
|
|
FsStorage data_storage;
|
|
|
|
FsFile data_file;
|
|
|
|
u32 out_domain_id = 0;
|
|
|
|
Result rc;
|
|
|
|
if (this->get_owner() == NULL) {
|
|
|
|
rc = fsOpenDataStorageByCurrentProcessFwd(this->forward_service, &data_storage);
|
|
|
|
} else {
|
|
|
|
rc = fsOpenDataStorageByCurrentProcessFromDomainFwd(this->forward_service, &out_domain_id);
|
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
|
|
rc = ipcCopyFromDomain(this->forward_service->handle, out_domain_id, &data_storage.s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Log(armGetTls(), 0x100);
|
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
|
|
/* TODO: Is there a sensible path that ends in ".romfs" we can use?" */
|
|
|
|
if (R_SUCCEEDED(Utils::OpenSdFileForAtmosphere(this->title_id, "romfs.bin", FS_OPEN_READ, &data_file))) {
|
|
|
|
out_session = new IPCSession<IStorageInterface>(std::make_shared<IStorageInterface>(new LayeredRomFS(std::make_shared<RomInterfaceStorage>(data_storage), std::make_shared<RomFileStorage>(data_file), this->title_id)));
|
|
|
|
} else {
|
|
|
|
out_session = new IPCSession<IStorageInterface>(std::make_shared<IStorageInterface>(new LayeredRomFS(std::make_shared<RomInterfaceStorage>(data_storage), nullptr, this->title_id)));
|
|
|
|
}
|
|
|
|
if (this->get_owner() == NULL) {
|
|
|
|
FsMitMWorker::AddWaitable(out_session);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
OutSession out_s = OutSession(out_session);
|
|
|
|
out_s.domain_id = out_domain_id;
|
|
|
|
return {rc, out_s};
|
|
|
|
}
|
|
|
|
|
2018-06-10 11:11:05 -06:00
|
|
|
/* Add redirection for System Data Archives to the SD card. */
|
2018-06-12 16:00:09 -06:00
|
|
|
std::tuple<Result, OutSession<IStorageInterface>> FsMitMService::open_data_storage_by_data_id(u64 sid, u64 data_id) {
|
|
|
|
FsStorageId storage_id = (FsStorageId)sid;
|
|
|
|
IPCSession<IStorageInterface> *out_session = NULL;
|
2018-06-10 11:11:05 -06:00
|
|
|
FsStorage data_storage;
|
|
|
|
FsFile data_file;
|
2018-06-14 17:50:01 -06:00
|
|
|
u32 out_domain_id = 0;
|
2018-06-12 16:00:09 -06:00
|
|
|
Result rc;
|
|
|
|
if (this->get_owner() == NULL) {
|
|
|
|
rc = fsOpenDataStorageByDataId(this->forward_service, storage_id, data_id, &data_storage);
|
|
|
|
} else {
|
|
|
|
rc = fsOpenDataStorageByDataIdFromDomain(this->forward_service, storage_id, data_id, &out_domain_id);
|
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
|
|
rc = ipcCopyFromDomain(this->forward_service->handle, out_domain_id, &data_storage.s);
|
|
|
|
}
|
|
|
|
}
|
2018-06-10 11:11:05 -06:00
|
|
|
if (R_SUCCEEDED(rc)) {
|
|
|
|
/* TODO: Is there a sensible path that ends in ".romfs" we can use?" */
|
2018-06-14 17:50:01 -06:00
|
|
|
if (R_SUCCEEDED(Utils::OpenSdFileForAtmosphere(data_id, "romfs.bin", FS_OPEN_READ, &data_file))) {
|
|
|
|
out_session = new IPCSession<IStorageInterface>(std::make_shared<IStorageInterface>(new LayeredRomFS(std::make_shared<RomInterfaceStorage>(data_storage), std::make_shared<RomFileStorage>(data_file), data_id)));
|
|
|
|
} else {
|
|
|
|
out_session = new IPCSession<IStorageInterface>(std::make_shared<IStorageInterface>(new LayeredRomFS(std::make_shared<RomInterfaceStorage>(data_storage), nullptr, data_id)));
|
2018-06-10 11:11:05 -06:00
|
|
|
}
|
2018-06-12 16:00:09 -06:00
|
|
|
if (this->get_owner() == NULL) {
|
2018-06-14 17:50:01 -06:00
|
|
|
FsMitMWorker::AddWaitable(out_session);
|
2018-06-12 16:00:09 -06:00
|
|
|
}
|
2018-06-10 11:11:05 -06:00
|
|
|
}
|
2018-06-12 16:00:09 -06:00
|
|
|
|
|
|
|
OutSession out_s = OutSession(out_session);
|
|
|
|
out_s.domain_id = out_domain_id;
|
|
|
|
return {rc, out_s};
|
2018-06-09 19:33:22 -06:00
|
|
|
}
|