2020-10-30 22:36:11 +00:00
|
|
|
/*
|
2021-10-04 19:59:10 +00:00
|
|
|
* Copyright (c) Atmosphère-NX
|
2020-10-30 22:36:11 +00:00
|
|
|
*
|
|
|
|
* 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 <stratosphere.hpp>
|
|
|
|
|
|
|
|
namespace ams::ddsf {
|
|
|
|
|
|
|
|
Result DeviceCodeEntryManager::Add(DeviceCode device_code, IDevice *device) {
|
|
|
|
/* Check pre-conditions. */
|
|
|
|
AMS_ASSERT(device != nullptr);
|
|
|
|
AMS_ASSERT(device->IsDriverAttached());
|
|
|
|
|
|
|
|
/* Acquire exclusive access to the manager. */
|
|
|
|
std::scoped_lock lk(this->entry_list_lock);
|
|
|
|
|
|
|
|
/* Check that we don't already have an entry with the code. */
|
|
|
|
for (const auto &holder : this->entry_list) {
|
|
|
|
AMS_ASSERT(holder.IsConstructed());
|
|
|
|
AMS_ASSERT(holder.Get().GetDeviceCode() != device_code);
|
2021-09-30 04:32:40 +00:00
|
|
|
AMS_UNUSED(holder);
|
2020-10-30 22:36:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate memory for a new device code entry holder. */
|
|
|
|
void *holder_storage = this->memory_resource->Allocate(sizeof(DeviceCodeEntryHolder));
|
|
|
|
R_UNLESS(holder_storage != nullptr, ddsf::ResultOutOfResource());
|
|
|
|
|
|
|
|
/* Initialize the new holder. */
|
2021-03-22 03:30:40 +00:00
|
|
|
auto *holder = std::construct_at(static_cast<DeviceCodeEntryHolder *>(holder_storage));
|
2020-10-30 22:36:11 +00:00
|
|
|
holder->Construct(device_code, device);
|
|
|
|
|
|
|
|
/* Link the new holder. */
|
|
|
|
holder->AddTo(this->entry_list);
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DeviceCodeEntryManager::Remove(DeviceCode device_code) {
|
|
|
|
/* Acquire exclusive access to the manager. */
|
|
|
|
std::scoped_lock lk(this->entry_list_lock);
|
|
|
|
|
|
|
|
/* Find and erase the entry. */
|
|
|
|
bool erased = false;
|
|
|
|
for (auto it = this->entry_list.begin(); it != this->entry_list.end(); /* ... */) {
|
|
|
|
/* Get the current entry, and advance the iterator. */
|
|
|
|
DeviceCodeEntryHolder *cur = std::addressof(*(it++));
|
|
|
|
|
|
|
|
/* If the entry matches the device code, remove it. */
|
|
|
|
AMS_ASSERT(cur->IsConstructed());
|
|
|
|
if (cur->Get().GetDeviceCode() == device_code) {
|
|
|
|
/* Destroy and deallocate the holder. */
|
|
|
|
cur->Destroy();
|
2021-03-22 03:30:40 +00:00
|
|
|
std::destroy_at(cur);
|
2020-10-30 22:36:11 +00:00
|
|
|
this->memory_resource->Deallocate(cur, sizeof(*cur));
|
|
|
|
|
|
|
|
erased = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return erased;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result DeviceCodeEntryManager::FindDeviceCodeEntry(DeviceCodeEntry **out, DeviceCode device_code) {
|
|
|
|
/* Check arguments. */
|
|
|
|
AMS_ASSERT(out != nullptr);
|
|
|
|
R_UNLESS(out != nullptr, ddsf::ResultInvalidArgument());
|
|
|
|
|
|
|
|
/* Find the device. */
|
|
|
|
bool found = false;
|
|
|
|
this->ForEachEntry([&](DeviceCodeEntry &entry) -> bool {
|
|
|
|
if (entry.GetDeviceCode() == device_code) {
|
|
|
|
found = true;
|
|
|
|
*out = std::addressof(entry);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
|
|
|
/* Check that we found the device. */
|
|
|
|
R_UNLESS(found, ddsf::ResultDeviceCodeNotFound());
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result DeviceCodeEntryManager::FindDeviceCodeEntry(const DeviceCodeEntry **out, DeviceCode device_code) const {
|
|
|
|
/* Check arguments. */
|
|
|
|
AMS_ASSERT(out != nullptr);
|
|
|
|
R_UNLESS(out != nullptr, ddsf::ResultInvalidArgument());
|
|
|
|
|
|
|
|
/* Find the device. */
|
|
|
|
bool found = false;
|
|
|
|
this->ForEachEntry([&](const DeviceCodeEntry &entry) -> bool {
|
|
|
|
if (entry.GetDeviceCode() == device_code) {
|
|
|
|
found = true;
|
|
|
|
*out = std::addressof(entry);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
|
|
|
/* Check that we found the device. */
|
|
|
|
R_UNLESS(found, ddsf::ResultDeviceCodeNotFound());
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result DeviceCodeEntryManager::FindDevice(IDevice **out, DeviceCode device_code) {
|
|
|
|
/* Check pre-conditions. */
|
|
|
|
AMS_ASSERT(out != nullptr);
|
|
|
|
|
|
|
|
/* Find the entry. */
|
|
|
|
DeviceCodeEntry *entry;
|
|
|
|
R_TRY(this->FindDeviceCodeEntry(std::addressof(entry), device_code));
|
|
|
|
|
|
|
|
/* Set the output. */
|
|
|
|
if (out != nullptr) {
|
|
|
|
*out = std::addressof(entry->GetDevice());
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result DeviceCodeEntryManager::FindDevice(const IDevice **out, DeviceCode device_code) const {
|
|
|
|
/* Check pre-conditions. */
|
|
|
|
AMS_ASSERT(out != nullptr);
|
|
|
|
|
|
|
|
/* Find the entry. */
|
|
|
|
const DeviceCodeEntry *entry;
|
|
|
|
R_TRY(this->FindDeviceCodeEntry(std::addressof(entry), device_code));
|
|
|
|
|
|
|
|
/* Set the output. */
|
|
|
|
if (out != nullptr) {
|
|
|
|
*out = std::addressof(entry->GetDevice());
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|