2019-07-12 01:13:43 +00:00
|
|
|
|
using Ryujinx.HLE.Exceptions;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
using Ryujinx.HLE.HOS.Ipc;
|
|
|
|
|
using Ryujinx.HLE.HOS.Kernel.Common;
|
|
|
|
|
using Ryujinx.HLE.HOS.Kernel.Threading;
|
|
|
|
|
using Ryujinx.HLE.HOS.Services.Hid;
|
2019-09-19 00:45:11 +00:00
|
|
|
|
using Ryujinx.HLE.HOS.Services.Hid.HidServer;
|
|
|
|
|
using Ryujinx.HLE.HOS.Services.Nfc.Nfp.UserManager;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
|
|
namespace Ryujinx.HLE.HOS.Services.Nfc.Nfp
|
|
|
|
|
{
|
|
|
|
|
class IUser : IpcService
|
|
|
|
|
{
|
|
|
|
|
private State _state = State.NonInitialized;
|
|
|
|
|
|
|
|
|
|
private KEvent _availabilityChangeEvent;
|
|
|
|
|
private int _availabilityChangeEventHandle = 0;
|
|
|
|
|
|
|
|
|
|
private List<Device> _devices = new List<Device>();
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
public IUser() { }
|
2019-06-28 11:08:23 +00:00
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(0)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Initialize(u64, u64, pid, buffer<unknown, 5>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Initialize(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
long appletResourceUserId = context.RequestData.ReadInt64();
|
|
|
|
|
long mcuVersionData = context.RequestData.ReadInt64();
|
|
|
|
|
|
|
|
|
|
long inputPosition = context.Request.SendBuff[0].Position;
|
|
|
|
|
long inputSize = context.Request.SendBuff[0].Size;
|
|
|
|
|
|
2020-05-03 22:54:50 +00:00
|
|
|
|
byte[] unknownBuffer = new byte[inputSize];
|
|
|
|
|
|
|
|
|
|
context.Memory.Read((ulong)inputPosition, unknownBuffer);
|
2019-06-28 11:08:23 +00:00
|
|
|
|
|
|
|
|
|
// NOTE: appletResourceUserId, mcuVersionData and the buffer are stored inside an internal struct.
|
|
|
|
|
// The buffer seems to contains entries with a size of 0x40 bytes each.
|
|
|
|
|
// Sadly, this internal struct doesn't seems to be used in retail.
|
|
|
|
|
|
|
|
|
|
// TODO: Add an instance of nn::nfc::server::Manager when it will be implemented.
|
|
|
|
|
// Add an instance of nn::nfc::server::SaveData when it will be implemented.
|
|
|
|
|
|
|
|
|
|
// TODO: When we will be able to add multiple controllers add one entry by controller here.
|
|
|
|
|
Device device1 = new Device
|
|
|
|
|
{
|
2020-04-03 00:10:02 +00:00
|
|
|
|
NpadIdType = NpadIdType.Player1,
|
|
|
|
|
Handle = HidUtils.GetIndexFromNpadIdType(NpadIdType.Player1),
|
2019-06-28 11:08:23 +00:00
|
|
|
|
State = DeviceState.Initialized
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
_devices.Add(device1);
|
|
|
|
|
|
|
|
|
|
_state = State.Initialized;
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(1)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Finalize()
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Finalize(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
// TODO: Call StopDetection() and Unmount() when they will be implemented.
|
|
|
|
|
// Remove the instance of nn::nfc::server::Manager when it will be implemented.
|
|
|
|
|
// Remove the instance of nn::nfc::server::SaveData when it will be implemented.
|
|
|
|
|
|
|
|
|
|
_devices.Clear();
|
|
|
|
|
|
|
|
|
|
_state = State.NonInitialized;
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(2)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// ListDevices() -> (u32, buffer<unknown, 0xa>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode ListDevices(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
if (context.Request.RecvListBuff.Count == 0)
|
|
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DevicesBufferIsNull;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
long outputPosition = context.Request.RecvListBuff[0].Position;
|
|
|
|
|
long outputSize = context.Request.RecvListBuff[0].Size;
|
|
|
|
|
|
|
|
|
|
if (_devices.Count == 0)
|
|
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DeviceNotFound;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < _devices.Count; i++)
|
|
|
|
|
{
|
2020-05-03 22:54:50 +00:00
|
|
|
|
context.Memory.Write((ulong)(outputPosition + (i * sizeof(long))), (uint)_devices[i].Handle);
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.ResponseData.Write(_devices.Count);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(3)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// StartDetection(bytes<8, 4>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode StartDetection(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(4)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// StopDetection(bytes<8, 4>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode StopDetection(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(5)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Mount(bytes<8, 4>, u32, u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Mount(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(6)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Unmount(bytes<8, 4>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Unmount(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(7)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// OpenApplicationArea(bytes<8, 4>, u32)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode OpenApplicationArea(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(8)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetApplicationArea(bytes<8, 4>) -> (u32, buffer<unknown, 6>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetApplicationArea(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(9)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// SetApplicationArea(bytes<8, 4>, buffer<unknown, 5>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode SetApplicationArea(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(10)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Flush(bytes<8, 4>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Flush(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(11)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// Restore(bytes<8, 4>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode Restore(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(12)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// CreateApplicationArea(bytes<8, 4>, u32, buffer<unknown, 5>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode CreateApplicationArea(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(13)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetTagInfo(bytes<8, 4>) -> buffer<unknown<0x58>, 0x1a>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetTagInfo(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(14)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetRegisterInfo(bytes<8, 4>) -> buffer<unknown<0x100>, 0x1a>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetRegisterInfo(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(15)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetCommonInfo(bytes<8, 4>) -> buffer<unknown<0x40>, 0x1a>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetCommonInfo(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(16)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetModelInfo(bytes<8, 4>) -> buffer<unknown<0x40>, 0x1a>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetModelInfo(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(17)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// AttachActivateEvent(bytes<8, 4>) -> handle<copy>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode AttachActivateEvent(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
uint deviceHandle = context.RequestData.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < _devices.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((uint)_devices[i].Handle == deviceHandle)
|
|
|
|
|
{
|
|
|
|
|
if (_devices[i].ActivateEventHandle == 0)
|
|
|
|
|
{
|
2020-05-04 03:41:29 +00:00
|
|
|
|
_devices[i].ActivateEvent = new KEvent(context.Device.System.KernelContext);
|
2019-06-28 11:08:23 +00:00
|
|
|
|
|
|
|
|
|
if (context.Process.HandleTable.GenerateHandle(_devices[i].ActivateEvent.ReadableEvent, out _devices[i].ActivateEventHandle) != KernelResult.Success)
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.Response.HandleDesc = IpcHandleDesc.MakeCopy(_devices[i].ActivateEventHandle);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DeviceNotFound;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(18)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// AttachDeactivateEvent(bytes<8, 4>) -> handle<copy>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode AttachDeactivateEvent(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
uint deviceHandle = context.RequestData.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < _devices.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((uint)_devices[i].Handle == deviceHandle)
|
|
|
|
|
{
|
|
|
|
|
if (_devices[i].DeactivateEventHandle == 0)
|
|
|
|
|
{
|
2020-05-04 03:41:29 +00:00
|
|
|
|
_devices[i].DeactivateEvent = new KEvent(context.Device.System.KernelContext);
|
2019-06-28 11:08:23 +00:00
|
|
|
|
|
|
|
|
|
if (context.Process.HandleTable.GenerateHandle(_devices[i].DeactivateEvent.ReadableEvent, out _devices[i].DeactivateEventHandle) != KernelResult.Success)
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.Response.HandleDesc = IpcHandleDesc.MakeCopy(_devices[i].DeactivateEventHandle);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DeviceNotFound;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(19)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetState() -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetState(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
context.ResponseData.Write((int)_state);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(20)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetDeviceState(bytes<8, 4>) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetDeviceState(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
uint deviceHandle = context.RequestData.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < _devices.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((uint)_devices[i].Handle == deviceHandle)
|
|
|
|
|
{
|
|
|
|
|
context.ResponseData.Write((uint)_devices[i].State);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.ResponseData.Write((uint)DeviceState.Unavailable);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DeviceNotFound;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(21)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetNpadId(bytes<8, 4>) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetNpadId(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
uint deviceHandle = context.RequestData.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < _devices.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((uint)_devices[i].Handle == deviceHandle)
|
|
|
|
|
{
|
|
|
|
|
context.ResponseData.Write((uint)HidUtils.GetNpadIdTypeFromIndex(_devices[i].Handle));
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.DeviceNotFound;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(22)]
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// GetApplicationAreaSize(bytes<8, 4>) -> u32
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode GetApplicationAreaSize(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(23)] // 3.0.0+
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// AttachAvailabilityChangeEvent() -> handle<copy>
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode AttachAvailabilityChangeEvent(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
if (_availabilityChangeEventHandle == 0)
|
|
|
|
|
{
|
2020-05-04 03:41:29 +00:00
|
|
|
|
_availabilityChangeEvent = new KEvent(context.Device.System.KernelContext);
|
2019-06-28 11:08:23 +00:00
|
|
|
|
|
|
|
|
|
if (context.Process.HandleTable.GenerateHandle(_availabilityChangeEvent.ReadableEvent, out _availabilityChangeEventHandle) != KernelResult.Success)
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.Response.HandleDesc = IpcHandleDesc.MakeCopy(_availabilityChangeEventHandle);
|
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
|
return ResultCode.Success;
|
2019-06-28 11:08:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-12 01:13:43 +00:00
|
|
|
|
[Command(24)] // 3.0.0+
|
2019-06-28 11:08:23 +00:00
|
|
|
|
// RecreateApplicationArea(bytes<8, 4>, u32, buffer<unknown, 5>)
|
2019-07-14 19:04:38 +00:00
|
|
|
|
public ResultCode RecreateApplicationArea(ServiceCtx context)
|
2019-06-28 11:08:23 +00:00
|
|
|
|
{
|
|
|
|
|
throw new ServiceNotImplementedException(context);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|