mirror of
https://git.suyu.dev/suyu/suyu
synced 2025-01-09 16:03:21 +00:00
service: nfp: Implement debug Interface
This commit is contained in:
parent
00800d5289
commit
307371e01d
6 changed files with 444 additions and 8 deletions
|
@ -90,8 +90,8 @@ public:
|
||||||
explicit IDebug(Core::System& system_) : Interface(system_, "NFP:IDebug") {
|
explicit IDebug(Core::System& system_) : Interface(system_, "NFP:IDebug") {
|
||||||
// clang-format off
|
// clang-format off
|
||||||
static const FunctionInfo functions[] = {
|
static const FunctionInfo functions[] = {
|
||||||
{0, nullptr, "InitializeDebug"},
|
{0, &IDebug::InitializeDebug, "InitializeDebug"},
|
||||||
{1, nullptr, "FinalizeDebug"},
|
{1, &IDebug::FinalizeDebug, "FinalizeDebug"},
|
||||||
{2, &IDebug::ListDevices, "ListDevices"},
|
{2, &IDebug::ListDevices, "ListDevices"},
|
||||||
{3, &IDebug::StartDetection, "StartDetection"},
|
{3, &IDebug::StartDetection, "StartDetection"},
|
||||||
{4, &IDebug::StopDetection, "StopDetection"},
|
{4, &IDebug::StopDetection, "StopDetection"},
|
||||||
|
@ -122,10 +122,10 @@ public:
|
||||||
{104, &IDebug::DeleteRegisterInfo, "DeleteRegisterInfo"},
|
{104, &IDebug::DeleteRegisterInfo, "DeleteRegisterInfo"},
|
||||||
{105, &IDebug::DeleteApplicationArea, "DeleteApplicationArea"},
|
{105, &IDebug::DeleteApplicationArea, "DeleteApplicationArea"},
|
||||||
{106, &IDebug::ExistsApplicationArea, "ExistsApplicationArea"},
|
{106, &IDebug::ExistsApplicationArea, "ExistsApplicationArea"},
|
||||||
{200, nullptr, "GetAll"},
|
{200, &IDebug::GetAll, "GetAll"},
|
||||||
{201, nullptr, "SetAll"},
|
{201, &IDebug::SetAll, "SetAll"},
|
||||||
{202, nullptr, "FlushDebug"},
|
{202, &IDebug::FlushDebug, "FlushDebug"},
|
||||||
{203, nullptr, "BreakTag"},
|
{203, &IDebug::BreakTag, "BreakTag"},
|
||||||
{204, nullptr, "ReadBackupData"},
|
{204, nullptr, "ReadBackupData"},
|
||||||
{205, nullptr, "WriteBackupData"},
|
{205, nullptr, "WriteBackupData"},
|
||||||
{206, nullptr, "WriteNtf"},
|
{206, nullptr, "WriteNtf"},
|
||||||
|
|
|
@ -240,6 +240,42 @@ Result NfpDevice::Flush() {
|
||||||
|
|
||||||
tag_data.write_counter++;
|
tag_data.write_counter++;
|
||||||
|
|
||||||
|
FlushWithBreak(BreakType::Normal);
|
||||||
|
|
||||||
|
is_data_moddified = false;
|
||||||
|
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::FlushDebug() {
|
||||||
|
if (device_state != DeviceState::TagMounted) {
|
||||||
|
LOG_ERROR(Service_NFC, "Wrong device state {}", device_state);
|
||||||
|
if (device_state == DeviceState::TagRemoved) {
|
||||||
|
return TagRemoved;
|
||||||
|
}
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) {
|
||||||
|
LOG_ERROR(Service_NFC, "Amiibo is read only", device_state);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
tag_data.write_counter++;
|
||||||
|
|
||||||
|
FlushWithBreak(BreakType::Normal);
|
||||||
|
|
||||||
|
is_data_moddified = false;
|
||||||
|
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::FlushWithBreak(BreakType break_type) {
|
||||||
|
if (break_type != BreakType::Normal) {
|
||||||
|
LOG_ERROR(Service_NFC, "Break type not implemented {}", break_type);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<u8> data(sizeof(EncryptedNTAG215File));
|
std::vector<u8> data(sizeof(EncryptedNTAG215File));
|
||||||
if (is_plain_amiibo) {
|
if (is_plain_amiibo) {
|
||||||
memcpy(data.data(), &tag_data, sizeof(tag_data));
|
memcpy(data.data(), &tag_data, sizeof(tag_data));
|
||||||
|
@ -257,8 +293,6 @@ Result NfpDevice::Flush() {
|
||||||
return WriteAmiiboFailed;
|
return WriteAmiiboFailed;
|
||||||
}
|
}
|
||||||
|
|
||||||
is_data_moddified = false;
|
|
||||||
|
|
||||||
return ResultSuccess;
|
return ResultSuccess;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -857,6 +891,140 @@ Result NfpDevice::ExistApplicationArea(bool& has_application_area) {
|
||||||
return ResultSuccess;
|
return ResultSuccess;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::GetAll(NfpData& data) const {
|
||||||
|
if (device_state != DeviceState::TagMounted) {
|
||||||
|
LOG_ERROR(Service_NFC, "Wrong device state {}", device_state);
|
||||||
|
if (device_state == DeviceState::TagRemoved) {
|
||||||
|
return TagRemoved;
|
||||||
|
}
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) {
|
||||||
|
LOG_ERROR(Service_NFC, "Amiibo is read only", device_state);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
CommonInfo common_info{};
|
||||||
|
Service::Mii::MiiManager manager;
|
||||||
|
const u64 application_id = tag_data.application_id;
|
||||||
|
|
||||||
|
GetCommonInfo(common_info);
|
||||||
|
|
||||||
|
data = {
|
||||||
|
.magic = tag_data.constant_value,
|
||||||
|
.write_counter = tag_data.write_counter,
|
||||||
|
.settings_crc = tag_data.settings.crc,
|
||||||
|
.common_info = common_info,
|
||||||
|
.mii_char_info = tag_data.owner_mii,
|
||||||
|
.mii_store_data_extension = tag_data.mii_extension,
|
||||||
|
.creation_date = tag_data.settings.init_date.GetWriteDate(),
|
||||||
|
.amiibo_name = tag_data.settings.amiibo_name,
|
||||||
|
.amiibo_name_null_terminated = 0,
|
||||||
|
.settings = tag_data.settings.settings,
|
||||||
|
.unknown1 = tag_data.unknown,
|
||||||
|
.register_info_crc = tag_data.register_info_crc,
|
||||||
|
.unknown2 = tag_data.unknown2,
|
||||||
|
.application_id = application_id,
|
||||||
|
.access_id = tag_data.application_area_id,
|
||||||
|
.settings_crc_counter = tag_data.settings.crc_counter,
|
||||||
|
.font_region = tag_data.settings.settings.font_region,
|
||||||
|
.tag_type = PackedTagType::Type2,
|
||||||
|
.console_type =
|
||||||
|
static_cast<AppAreaVersion>(application_id >> application_id_version_offset & 0xf),
|
||||||
|
.application_id_byte = tag_data.application_id_byte,
|
||||||
|
.application_area = tag_data.application_area,
|
||||||
|
};
|
||||||
|
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::SetAll(const NfpData& data) {
|
||||||
|
if (device_state != DeviceState::TagMounted) {
|
||||||
|
LOG_ERROR(Service_NFC, "Wrong device state {}", device_state);
|
||||||
|
if (device_state == DeviceState::TagRemoved) {
|
||||||
|
return TagRemoved;
|
||||||
|
}
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) {
|
||||||
|
LOG_ERROR(Service_NFC, "Amiibo is read only", device_state);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
tag_data.constant_value = data.magic;
|
||||||
|
tag_data.write_counter = data.write_counter;
|
||||||
|
tag_data.settings.crc = data.settings_crc;
|
||||||
|
tag_data.settings.write_date.SetWriteDate(data.common_info.last_write_date);
|
||||||
|
tag_data.write_counter = data.common_info.write_counter;
|
||||||
|
tag_data.amiibo_version = data.common_info.version;
|
||||||
|
tag_data.owner_mii = data.mii_char_info;
|
||||||
|
tag_data.mii_extension = data.mii_store_data_extension;
|
||||||
|
tag_data.settings.init_date.SetWriteDate(data.creation_date);
|
||||||
|
tag_data.settings.amiibo_name = data.amiibo_name;
|
||||||
|
tag_data.settings.settings = data.settings;
|
||||||
|
tag_data.unknown = data.unknown1;
|
||||||
|
tag_data.register_info_crc = data.register_info_crc;
|
||||||
|
tag_data.unknown2 = data.unknown2;
|
||||||
|
tag_data.application_id = data.application_id;
|
||||||
|
tag_data.application_area_id = data.access_id;
|
||||||
|
tag_data.settings.crc_counter = data.settings_crc_counter;
|
||||||
|
tag_data.settings.settings.font_region.Assign(data.font_region);
|
||||||
|
tag_data.application_id_byte = data.application_id_byte;
|
||||||
|
tag_data.application_area = data.application_area;
|
||||||
|
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::BreakTag(BreakType break_type) {
|
||||||
|
if (device_state != DeviceState::TagMounted) {
|
||||||
|
LOG_ERROR(Service_NFC, "Wrong device state {}", device_state);
|
||||||
|
if (device_state == DeviceState::TagRemoved) {
|
||||||
|
return TagRemoved;
|
||||||
|
}
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) {
|
||||||
|
LOG_ERROR(Service_NFC, "Amiibo is read only", device_state);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Complete this implementation
|
||||||
|
|
||||||
|
return FlushWithBreak(break_type);
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::ReadBackupData() {
|
||||||
|
// Not implemented
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::WriteBackupData() {
|
||||||
|
// Not implemented
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result NfpDevice::WriteNtf() {
|
||||||
|
if (device_state != DeviceState::TagMounted) {
|
||||||
|
LOG_ERROR(Service_NFC, "Wrong device state {}", device_state);
|
||||||
|
if (device_state == DeviceState::TagRemoved) {
|
||||||
|
return TagRemoved;
|
||||||
|
}
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) {
|
||||||
|
LOG_ERROR(Service_NFC, "Amiibo is read only", device_state);
|
||||||
|
return WrongDeviceState;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not implemented
|
||||||
|
|
||||||
|
return ResultSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
u64 NfpDevice::GetHandle() const {
|
u64 NfpDevice::GetHandle() const {
|
||||||
// Generate a handle based of the npad id
|
// Generate a handle based of the npad id
|
||||||
return static_cast<u64>(npad_id);
|
return static_cast<u64>(npad_id);
|
||||||
|
|
|
@ -41,7 +41,10 @@ public:
|
||||||
Result StopDetection();
|
Result StopDetection();
|
||||||
Result Mount(MountTarget mount_target);
|
Result Mount(MountTarget mount_target);
|
||||||
Result Unmount();
|
Result Unmount();
|
||||||
|
|
||||||
Result Flush();
|
Result Flush();
|
||||||
|
Result FlushDebug();
|
||||||
|
Result FlushWithBreak(BreakType break_type);
|
||||||
|
|
||||||
Result GetTagInfo(TagInfo& tag_info) const;
|
Result GetTagInfo(TagInfo& tag_info) const;
|
||||||
Result GetCommonInfo(CommonInfo& common_info) const;
|
Result GetCommonInfo(CommonInfo& common_info) const;
|
||||||
|
@ -64,6 +67,13 @@ public:
|
||||||
Result DeleteApplicationArea();
|
Result DeleteApplicationArea();
|
||||||
Result ExistApplicationArea(bool& has_application_area);
|
Result ExistApplicationArea(bool& has_application_area);
|
||||||
|
|
||||||
|
Result GetAll(NfpData& data) const;
|
||||||
|
Result SetAll(const NfpData& data);
|
||||||
|
Result BreakTag(BreakType break_type);
|
||||||
|
Result ReadBackupData();
|
||||||
|
Result WriteBackupData();
|
||||||
|
Result WriteNtf();
|
||||||
|
|
||||||
u64 GetHandle() const;
|
u64 GetHandle() const;
|
||||||
u32 GetApplicationAreaSize() const;
|
u32 GetApplicationAreaSize() const;
|
||||||
DeviceState GetCurrentState() const;
|
DeviceState GetCurrentState() const;
|
||||||
|
|
|
@ -53,6 +53,19 @@ void Interface::InitializeSystem(HLERequestContext& ctx) {
|
||||||
rb.Push(ResultSuccess);
|
rb.Push(ResultSuccess);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Interface::InitializeDebug(HLERequestContext& ctx) {
|
||||||
|
LOG_INFO(Service_NFP, "called");
|
||||||
|
|
||||||
|
state = State::Initialized;
|
||||||
|
|
||||||
|
for (auto& device : devices) {
|
||||||
|
device->Initialize();
|
||||||
|
}
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
void Interface::Finalize(HLERequestContext& ctx) {
|
void Interface::Finalize(HLERequestContext& ctx) {
|
||||||
LOG_INFO(Service_NFP, "called");
|
LOG_INFO(Service_NFP, "called");
|
||||||
|
|
||||||
|
@ -79,6 +92,19 @@ void Interface::FinalizeSystem(HLERequestContext& ctx) {
|
||||||
rb.Push(ResultSuccess);
|
rb.Push(ResultSuccess);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Interface::FinalizeDebug(HLERequestContext& ctx) {
|
||||||
|
LOG_INFO(Service_NFP, "called");
|
||||||
|
|
||||||
|
state = State::NonInitialized;
|
||||||
|
|
||||||
|
for (auto& device : devices) {
|
||||||
|
device->Finalize();
|
||||||
|
}
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
void Interface::ListDevices(HLERequestContext& ctx) {
|
void Interface::ListDevices(HLERequestContext& ctx) {
|
||||||
LOG_DEBUG(Service_NFP, "called");
|
LOG_DEBUG(Service_NFP, "called");
|
||||||
|
|
||||||
|
@ -833,6 +859,184 @@ void Interface::ExistsApplicationArea(HLERequestContext& ctx) {
|
||||||
rb.Push(has_application_area);
|
rb.Push(has_application_area);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Interface::GetAll(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
NfpData data{};
|
||||||
|
const auto result = device.value()->GetAll(data);
|
||||||
|
|
||||||
|
ctx.WriteBuffer(data);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::SetAll(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
const auto nfp_data{ctx.ReadBuffer()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
NfpData data{};
|
||||||
|
memcpy(&data, nfp_data.data(), sizeof(NfpData));
|
||||||
|
|
||||||
|
const auto result = device.value()->SetAll(data);
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::FlushDebug(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto result = device.value()->FlushDebug();
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::BreakTag(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
const auto break_type{rp.PopEnum<BreakType>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}, break_type={}", device_handle, break_type);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto result = device.value()->BreakTag(break_type);
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::ReadBackupData(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto result = device.value()->ReadBackupData();
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::WriteBackupData(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto result = device.value()->WriteBackupData();
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Interface::WriteNtf(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto device_handle{rp.Pop<u64>()};
|
||||||
|
LOG_DEBUG(Service_NFP, "called, device_handle={}", device_handle);
|
||||||
|
|
||||||
|
if (state == State::NonInitialized) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(NfcDisabled);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto device = GetNfpDevice(device_handle);
|
||||||
|
|
||||||
|
if (!device.has_value()) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(DeviceNotFound);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto result = device.value()->WriteNtf();
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(result);
|
||||||
|
}
|
||||||
|
|
||||||
std::optional<std::shared_ptr<NfpDevice>> Interface::GetNfpDevice(u64 handle) {
|
std::optional<std::shared_ptr<NfpDevice>> Interface::GetNfpDevice(u64 handle) {
|
||||||
for (auto& device : devices) {
|
for (auto& device : devices) {
|
||||||
if (device->GetHandle() == handle) {
|
if (device->GetHandle() == handle) {
|
||||||
|
|
|
@ -20,8 +20,10 @@ public:
|
||||||
|
|
||||||
void Initialize(HLERequestContext& ctx);
|
void Initialize(HLERequestContext& ctx);
|
||||||
void InitializeSystem(HLERequestContext& ctx);
|
void InitializeSystem(HLERequestContext& ctx);
|
||||||
|
void InitializeDebug(HLERequestContext& ctx);
|
||||||
void Finalize(HLERequestContext& ctx);
|
void Finalize(HLERequestContext& ctx);
|
||||||
void FinalizeSystem(HLERequestContext& ctx);
|
void FinalizeSystem(HLERequestContext& ctx);
|
||||||
|
void FinalizeDebug(HLERequestContext& ctx);
|
||||||
void ListDevices(HLERequestContext& ctx);
|
void ListDevices(HLERequestContext& ctx);
|
||||||
void StartDetection(HLERequestContext& ctx);
|
void StartDetection(HLERequestContext& ctx);
|
||||||
void StopDetection(HLERequestContext& ctx);
|
void StopDetection(HLERequestContext& ctx);
|
||||||
|
@ -52,6 +54,13 @@ public:
|
||||||
void DeleteRegisterInfo(HLERequestContext& ctx);
|
void DeleteRegisterInfo(HLERequestContext& ctx);
|
||||||
void DeleteApplicationArea(HLERequestContext& ctx);
|
void DeleteApplicationArea(HLERequestContext& ctx);
|
||||||
void ExistsApplicationArea(HLERequestContext& ctx);
|
void ExistsApplicationArea(HLERequestContext& ctx);
|
||||||
|
void GetAll(HLERequestContext& ctx);
|
||||||
|
void SetAll(HLERequestContext& ctx);
|
||||||
|
void FlushDebug(HLERequestContext& ctx);
|
||||||
|
void BreakTag(HLERequestContext& ctx);
|
||||||
|
void ReadBackupData(HLERequestContext& ctx);
|
||||||
|
void WriteBackupData(HLERequestContext& ctx);
|
||||||
|
void WriteNtf(HLERequestContext& ctx);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
enum class State : u32 {
|
enum class State : u32 {
|
||||||
|
|
|
@ -109,6 +109,12 @@ enum class AppAreaVersion : u8 {
|
||||||
NotSet = 0xFF,
|
NotSet = 0xFF,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum class BreakType : u32 {
|
||||||
|
Normal,
|
||||||
|
Unknown1,
|
||||||
|
Unknown2,
|
||||||
|
};
|
||||||
|
|
||||||
enum class CabinetMode : u8 {
|
enum class CabinetMode : u8 {
|
||||||
StartNicknameAndOwnerSettings,
|
StartNicknameAndOwnerSettings,
|
||||||
StartGameDataEraser,
|
StartGameDataEraser,
|
||||||
|
@ -181,6 +187,12 @@ struct AmiiboDate {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SetWriteDate(const WriteDate& write_date) {
|
||||||
|
SetYear(write_date.year);
|
||||||
|
SetMonth(write_date.month);
|
||||||
|
SetDay(write_date.day);
|
||||||
|
}
|
||||||
|
|
||||||
void SetYear(u16 year) {
|
void SetYear(u16 year) {
|
||||||
const u16 year_converted = static_cast<u16>((year - 2000) << 9);
|
const u16 year_converted = static_cast<u16>((year - 2000) << 9);
|
||||||
raw_date = Common::swap16((GetValue() & ~0xFE00) | year_converted);
|
raw_date = Common::swap16((GetValue() & ~0xFE00) | year_converted);
|
||||||
|
@ -375,6 +387,39 @@ struct AdminInfo {
|
||||||
};
|
};
|
||||||
static_assert(sizeof(AdminInfo) == 0x40, "AdminInfo is an invalid size");
|
static_assert(sizeof(AdminInfo) == 0x40, "AdminInfo is an invalid size");
|
||||||
|
|
||||||
|
#pragma pack(1)
|
||||||
|
// This is nn::nfp::NfpData
|
||||||
|
struct NfpData {
|
||||||
|
u8 magic;
|
||||||
|
INSERT_PADDING_BYTES(0x1);
|
||||||
|
u8 write_counter;
|
||||||
|
INSERT_PADDING_BYTES(0x1);
|
||||||
|
u32 settings_crc;
|
||||||
|
INSERT_PADDING_BYTES(0x38);
|
||||||
|
CommonInfo common_info;
|
||||||
|
Service::Mii::Ver3StoreData mii_char_info;
|
||||||
|
Service::Mii::NfpStoreDataExtension mii_store_data_extension;
|
||||||
|
WriteDate creation_date;
|
||||||
|
std::array<u16_be, amiibo_name_length> amiibo_name;
|
||||||
|
u16 amiibo_name_null_terminated;
|
||||||
|
Settings settings;
|
||||||
|
u8 unknown1;
|
||||||
|
u32 register_info_crc;
|
||||||
|
std::array<u32, 5> unknown2;
|
||||||
|
INSERT_PADDING_BYTES(0x64);
|
||||||
|
u64 application_id;
|
||||||
|
u32 access_id;
|
||||||
|
u16 settings_crc_counter;
|
||||||
|
u8 font_region;
|
||||||
|
PackedTagType tag_type;
|
||||||
|
AppAreaVersion console_type;
|
||||||
|
u8 application_id_byte;
|
||||||
|
INSERT_PADDING_BYTES(0x2E);
|
||||||
|
ApplicationArea application_area;
|
||||||
|
};
|
||||||
|
static_assert(sizeof(NfpData) == 0x298, "NfpData is an invalid size");
|
||||||
|
#pragma pack()
|
||||||
|
|
||||||
struct SectorKey {
|
struct SectorKey {
|
||||||
MifareCmd command;
|
MifareCmd command;
|
||||||
u8 unknown; // Usually 1
|
u8 unknown; // Usually 1
|
||||||
|
|
Loading…
Reference in a new issue