mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-10 10:15:12 +00:00
4b1bed1b05
This was tested against HW with https://github.com/Xpl0itR/Input-Test/tree/master/Input-Test and a few changes to record the minimum value axis value when the stick buttons were marked as pressed.
112 lines
4.1 KiB
C#
112 lines
4.1 KiB
C#
using Ryujinx.Common;
|
|
using Ryujinx.HLE.Exceptions;
|
|
using Ryujinx.Common.Configuration.Hid;
|
|
using System.Collections.Generic;
|
|
using System.Runtime.CompilerServices;
|
|
|
|
namespace Ryujinx.HLE.HOS.Services.Hid
|
|
{
|
|
public class Hid
|
|
{
|
|
private readonly Switch _device;
|
|
|
|
private readonly ulong _hidMemoryAddress;
|
|
|
|
internal ref HidSharedMemory SharedMemory => ref _device.Memory.GetRef<HidSharedMemory>(_hidMemoryAddress);
|
|
|
|
internal const int SharedMemEntryCount = 17;
|
|
|
|
public DebugPadDevice DebugPad;
|
|
public TouchDevice Touchscreen;
|
|
public MouseDevice Mouse;
|
|
public KeyboardDevice Keyboard;
|
|
public NpadDevices Npads;
|
|
|
|
static Hid()
|
|
{
|
|
if (Unsafe.SizeOf<ShMemDebugPad>() != 0x400)
|
|
{
|
|
throw new InvalidStructLayoutException<ShMemDebugPad>(0x400);
|
|
}
|
|
if (Unsafe.SizeOf<ShMemTouchScreen>() != 0x3000)
|
|
{
|
|
throw new InvalidStructLayoutException<ShMemTouchScreen>(0x3000);
|
|
}
|
|
if (Unsafe.SizeOf<ShMemKeyboard>() != 0x400)
|
|
{
|
|
throw new InvalidStructLayoutException<ShMemKeyboard>(0x400);
|
|
}
|
|
if (Unsafe.SizeOf<ShMemMouse>() != 0x400)
|
|
{
|
|
throw new InvalidStructLayoutException<ShMemMouse>(0x400);
|
|
}
|
|
if (Unsafe.SizeOf<ShMemNpad>() != 0x5000)
|
|
{
|
|
throw new InvalidStructLayoutException<ShMemNpad>(0x5000);
|
|
}
|
|
if (Unsafe.SizeOf<HidSharedMemory>() != Horizon.HidSize)
|
|
{
|
|
throw new InvalidStructLayoutException<HidSharedMemory>(Horizon.HidSize);
|
|
}
|
|
}
|
|
|
|
public Hid(in Switch device, ulong sharedHidMemoryAddress)
|
|
{
|
|
_device = device;
|
|
_hidMemoryAddress = sharedHidMemoryAddress;
|
|
|
|
device.Memory.ZeroFill(sharedHidMemoryAddress, Horizon.HidSize);
|
|
}
|
|
|
|
public void InitDevices()
|
|
{
|
|
DebugPad = new DebugPadDevice(_device, true);
|
|
Touchscreen = new TouchDevice(_device, true);
|
|
Mouse = new MouseDevice(_device, false);
|
|
Keyboard = new KeyboardDevice(_device, false);
|
|
Npads = new NpadDevices(_device, true);
|
|
}
|
|
|
|
internal void RefreshInputConfig(List<InputConfig> inputConfig)
|
|
{
|
|
ControllerConfig[] npadConfig = new ControllerConfig[inputConfig.Count];
|
|
|
|
for (int i = 0; i < npadConfig.Length; ++i)
|
|
{
|
|
npadConfig[i].Player = (PlayerIndex)inputConfig[i].PlayerIndex;
|
|
npadConfig[i].Type = (ControllerType)inputConfig[i].ControllerType;
|
|
}
|
|
|
|
_device.Hid.Npads.Configure(npadConfig);
|
|
}
|
|
|
|
internal void RefreshInputConfigEvent(object _, ReactiveEventArgs<List<InputConfig>> args)
|
|
{
|
|
RefreshInputConfig(args.NewValue);
|
|
}
|
|
|
|
public ControllerKeys UpdateStickButtons(JoystickPosition leftStick, JoystickPosition rightStick)
|
|
{
|
|
const int stickButtonThreshold = short.MaxValue / 2;
|
|
ControllerKeys result = 0;
|
|
|
|
result |= (leftStick.Dx < -stickButtonThreshold) ? ControllerKeys.LStickLeft : result;
|
|
result |= (leftStick.Dx > stickButtonThreshold) ? ControllerKeys.LStickRight : result;
|
|
result |= (leftStick.Dy < -stickButtonThreshold) ? ControllerKeys.LStickDown : result;
|
|
result |= (leftStick.Dy > stickButtonThreshold) ? ControllerKeys.LStickUp : result;
|
|
|
|
result |= (rightStick.Dx < -stickButtonThreshold) ? ControllerKeys.RStickLeft : result;
|
|
result |= (rightStick.Dx > stickButtonThreshold) ? ControllerKeys.RStickRight : result;
|
|
result |= (rightStick.Dy < -stickButtonThreshold) ? ControllerKeys.RStickDown : result;
|
|
result |= (rightStick.Dy > stickButtonThreshold) ? ControllerKeys.RStickUp : result;
|
|
|
|
return result;
|
|
}
|
|
|
|
internal static ulong GetTimestampTicks()
|
|
{
|
|
return (ulong)PerformanceCounter.ElapsedMilliseconds * 19200;
|
|
}
|
|
}
|
|
}
|