2019-06-26 23:46:19 +01:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
#include "ldr_ecs.hpp"
|
|
|
|
|
2019-10-24 10:30:10 +01:00
|
|
|
namespace ams::ldr::ecs {
|
2019-06-26 23:46:19 +01:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
/* Convenience definition. */
|
|
|
|
constexpr size_t DeviceNameSizeMax = 0x20;
|
|
|
|
constexpr size_t MaxExternalContentSourceCount = 0x10;
|
|
|
|
|
|
|
|
/* Types. */
|
|
|
|
class ExternalContentSource {
|
|
|
|
NON_COPYABLE(ExternalContentSource);
|
|
|
|
NON_MOVEABLE(ExternalContentSource);
|
|
|
|
private:
|
2019-12-13 06:56:50 +00:00
|
|
|
bool has_mounted = false;
|
2019-06-26 23:46:19 +01:00
|
|
|
char device_name[DeviceNameSizeMax];
|
2019-12-13 06:56:50 +00:00
|
|
|
os::ManagedHandle client;
|
|
|
|
|
|
|
|
Result Mount() {
|
|
|
|
/* Create service. */
|
|
|
|
Service srv;
|
|
|
|
serviceCreate(&srv, client.Move());
|
|
|
|
FsFileSystem fs = { srv };
|
|
|
|
auto fs_guard = SCOPE_GUARD { fsFsClose(&fs); };
|
|
|
|
|
|
|
|
/* Try to mount. */
|
|
|
|
R_UNLESS(fsdevMountDevice(device_name, fs) >= 0, fs::ResultMountNameAlreadyExists());
|
|
|
|
fs_guard.Cancel();
|
|
|
|
|
|
|
|
this->has_mounted = true;
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
2019-06-26 23:46:19 +01:00
|
|
|
public:
|
2019-12-13 06:56:50 +00:00
|
|
|
ExternalContentSource(const char *dn, os::ManagedHandle client) : client(std::move(client)) {
|
2019-10-15 05:40:05 +01:00
|
|
|
std::strncpy(this->device_name, dn, sizeof(this->device_name));
|
2019-06-26 23:46:19 +01:00
|
|
|
this->device_name[sizeof(this->device_name) - 1] = '\0';
|
|
|
|
}
|
|
|
|
|
2019-12-13 06:56:50 +00:00
|
|
|
Result EnsureMounted() {
|
|
|
|
if (!this->has_mounted) {
|
|
|
|
return Mount();
|
|
|
|
}
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2019-06-26 23:46:19 +01:00
|
|
|
~ExternalContentSource() {
|
2019-12-13 06:56:50 +00:00
|
|
|
if (this->has_mounted) {
|
|
|
|
fsdevUnmountDevice(this->device_name);
|
|
|
|
}
|
2019-06-26 23:46:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *GetDeviceName() const {
|
|
|
|
return this->device_name;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Global storage. */
|
|
|
|
std::unordered_map<u64, ExternalContentSource> g_map;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* API. */
|
2019-10-28 04:43:01 +00:00
|
|
|
const char *Get(ncm::ProgramId program_id) {
|
|
|
|
auto it = g_map.find(static_cast<u64>(program_id));
|
2019-06-26 23:46:19 +01:00
|
|
|
if (it == g_map.end()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2019-12-13 06:56:50 +00:00
|
|
|
if (R_FAILED(it->second.EnsureMounted())) {
|
|
|
|
g_map.erase(it);
|
|
|
|
return nullptr;
|
|
|
|
}
|
2019-06-26 23:46:19 +01:00
|
|
|
return it->second.GetDeviceName();
|
|
|
|
}
|
|
|
|
|
2019-10-28 04:43:01 +00:00
|
|
|
Result Set(Handle *out, ncm::ProgramId program_id) {
|
2019-10-24 09:40:44 +01:00
|
|
|
/* TODO: Is this an appropriate error? */
|
|
|
|
R_UNLESS(g_map.size() < MaxExternalContentSourceCount, ldr::ResultTooManyArguments());
|
2019-06-26 23:46:19 +01:00
|
|
|
|
|
|
|
/* Clear any sources. */
|
2019-10-28 04:43:01 +00:00
|
|
|
R_ASSERT(Clear(program_id));
|
2019-06-26 23:46:19 +01:00
|
|
|
|
|
|
|
/* Generate mountpoint. */
|
|
|
|
char device_name[DeviceNameSizeMax];
|
2019-10-28 04:43:01 +00:00
|
|
|
std::snprintf(device_name, DeviceNameSizeMax, "ecs-%016lx", static_cast<u64>(program_id));
|
2019-06-26 23:46:19 +01:00
|
|
|
|
|
|
|
/* Create session. */
|
2019-09-28 02:04:58 +01:00
|
|
|
os::ManagedHandle server, client;
|
2019-06-28 01:37:33 +01:00
|
|
|
R_TRY(svcCreateSession(server.GetPointer(), client.GetPointer(), 0, 0));
|
2019-06-26 23:46:19 +01:00
|
|
|
|
2019-12-13 06:56:50 +00:00
|
|
|
/* Do not create service yet. */
|
|
|
|
/* Defer until we've handed the server side back so we don't deadlock on querying pointer buffer size. */
|
2019-06-26 23:46:19 +01:00
|
|
|
|
|
|
|
/* Add to map. */
|
2019-12-13 06:56:50 +00:00
|
|
|
g_map.emplace(std::piecewise_construct,
|
|
|
|
std::make_tuple(static_cast<u64>(program_id)),
|
|
|
|
std::make_tuple(device_name, std::move(client)));
|
2019-06-26 23:46:19 +01:00
|
|
|
*out = server.Move();
|
2019-10-24 09:40:44 +01:00
|
|
|
return ResultSuccess();
|
2019-06-26 23:46:19 +01:00
|
|
|
}
|
|
|
|
|
2019-10-28 04:43:01 +00:00
|
|
|
Result Clear(ncm::ProgramId program_id) {
|
2019-06-26 23:46:19 +01:00
|
|
|
/* Delete if present. */
|
2019-10-28 04:43:01 +00:00
|
|
|
g_map.erase(static_cast<u64>(program_id));
|
2019-10-24 09:40:44 +01:00
|
|
|
return ResultSuccess();
|
2019-06-26 23:46:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|