mirror of
https://github.com/cemu-project/DS4Windows.git
synced 2025-01-19 19:51:18 +01:00
275 lines
11 KiB
C#
275 lines
11 KiB
C#
using System;
|
|
|
|
namespace DS4Windows
|
|
{
|
|
public delegate void SixAxisHandler<TEventArgs>(DS4SixAxis sender, TEventArgs args);
|
|
|
|
public class SixAxisEventArgs : EventArgs
|
|
{
|
|
public readonly SixAxis sixAxis;
|
|
public readonly DateTime timeStamp;
|
|
public SixAxisEventArgs(DateTime utcTimestamp, SixAxis sa)
|
|
{
|
|
sixAxis = sa;
|
|
timeStamp = utcTimestamp;
|
|
}
|
|
}
|
|
|
|
public class SixAxis
|
|
{
|
|
public const int ACC_RES_PER_G = 8192;
|
|
private const float F_ACC_RES_PER_G = ACC_RES_PER_G;
|
|
public const int GYRO_RES_IN_DEG_SEC = 16;
|
|
private const float F_GYRO_RES_IN_DEG_SEC = GYRO_RES_IN_DEG_SEC;
|
|
|
|
public int gyroYaw, gyroPitch, gyroRoll, accelX, accelY, accelZ;
|
|
public int outputAccelX, outputAccelY, outputAccelZ;
|
|
public double accelXG, accelYG, accelZG;
|
|
public double angVelYaw, angVelPitch, angVelRoll;
|
|
public int gyroYawFull, gyroPitchFull, gyroRollFull;
|
|
public int accelXFull, accelYFull, accelZFull;
|
|
public double elapsed;
|
|
public SixAxis previousAxis = null;
|
|
|
|
private double tempDouble = 0d;
|
|
|
|
public SixAxis(int X, int Y, int Z,
|
|
int aX, int aY, int aZ,
|
|
double elapsedDelta, SixAxis prevAxis = null)
|
|
{
|
|
populate(X, Y, Z, aX, aY, aZ, elapsedDelta, prevAxis);
|
|
}
|
|
|
|
public void copy(SixAxis src)
|
|
{
|
|
gyroYaw = src.gyroYaw;
|
|
gyroPitch = src.gyroPitch;
|
|
gyroRoll = src.gyroRoll;
|
|
|
|
gyroYawFull = src.gyroYawFull;
|
|
accelXFull = src.accelXFull; accelYFull = src.accelYFull; accelZFull = src.accelZFull;
|
|
|
|
angVelYaw = src.angVelYaw;
|
|
angVelPitch = src.angVelPitch;
|
|
angVelRoll = src.angVelRoll;
|
|
|
|
accelXG = src.accelXG;
|
|
accelYG = src.accelYG;
|
|
accelZG = src.accelZG;
|
|
|
|
// Put accel ranges between 0 - 128 abs
|
|
accelX = src.accelX;
|
|
accelY = src.accelY;
|
|
accelZ = src.accelZ;
|
|
outputAccelX = accelX;
|
|
outputAccelY = accelY;
|
|
outputAccelZ = accelZ;
|
|
|
|
elapsed = src.elapsed;
|
|
previousAxis = src.previousAxis;
|
|
}
|
|
|
|
public void populate(int X, int Y, int Z,
|
|
int aX, int aY, int aZ,
|
|
double elapsedDelta, SixAxis prevAxis = null)
|
|
{
|
|
gyroYaw = -X / 256;
|
|
gyroPitch = Y / 256;
|
|
gyroRoll = -Z / 256;
|
|
|
|
gyroYawFull = -X; gyroPitchFull = Y; gyroRollFull = -Z;
|
|
accelXFull = -aX; accelYFull = -aY; accelZFull = aZ;
|
|
|
|
angVelYaw = gyroYawFull / F_GYRO_RES_IN_DEG_SEC;
|
|
angVelPitch = gyroPitchFull / F_GYRO_RES_IN_DEG_SEC;
|
|
angVelRoll = gyroRollFull / F_GYRO_RES_IN_DEG_SEC;
|
|
|
|
accelXG = tempDouble = accelXFull / F_ACC_RES_PER_G;
|
|
accelYG = tempDouble = accelYFull / F_ACC_RES_PER_G;
|
|
accelZG = tempDouble = accelZFull / F_ACC_RES_PER_G;
|
|
|
|
// Put accel ranges between 0 - 128 abs
|
|
accelX = -aX / 64;
|
|
accelY = -aY / 64;
|
|
accelZ = aZ / 64;
|
|
outputAccelX = accelX;
|
|
outputAccelY = accelY;
|
|
outputAccelZ = accelZ;
|
|
|
|
elapsed = elapsedDelta;
|
|
previousAxis = prevAxis;
|
|
}
|
|
}
|
|
|
|
internal class CalibData
|
|
{
|
|
public int bias;
|
|
public int sensNumer;
|
|
public int sensDenom;
|
|
public const int GyroPitchIdx = 0, GyroYawIdx = 1, GyroRollIdx = 2,
|
|
AccelXIdx = 3, AccelYIdx = 4, AccelZIdx = 5;
|
|
}
|
|
|
|
public class DS4SixAxis
|
|
{
|
|
//public event EventHandler<SixAxisEventArgs> SixAccelMoved = null;
|
|
public event SixAxisHandler<SixAxisEventArgs> SixAccelMoved = null;
|
|
private SixAxis sPrev = null, now = null;
|
|
private CalibData[] calibrationData = new CalibData[6] { new CalibData(), new CalibData(),
|
|
new CalibData(), new CalibData(), new CalibData(), new CalibData()
|
|
};
|
|
private bool calibrationDone = false;
|
|
|
|
public DS4SixAxis()
|
|
{
|
|
sPrev = new SixAxis(0, 0, 0, 0, 0, 0, 0.0);
|
|
now = new SixAxis(0, 0, 0, 0, 0, 0, 0.0);
|
|
}
|
|
|
|
int temInt = 0;
|
|
public void setCalibrationData(ref byte[] calibData, bool fromUSB)
|
|
{
|
|
int pitchPlus, pitchMinus, yawPlus, yawMinus, rollPlus, rollMinus,
|
|
accelXPlus, accelXMinus, accelYPlus, accelYMinus, accelZPlus, accelZMinus,
|
|
gyroSpeedPlus, gyroSpeedMinus;
|
|
|
|
calibrationData[0].bias = (short)((ushort)(calibData[2] << 8) | calibData[1]);
|
|
calibrationData[1].bias = (short)((ushort)(calibData[4] << 8) | calibData[3]);
|
|
calibrationData[2].bias = (short)((ushort)(calibData[6] << 8) | calibData[5]);
|
|
|
|
if (!fromUSB)
|
|
{
|
|
pitchPlus = temInt = (short)((ushort)(calibData[8] << 8) | calibData[7]);
|
|
yawPlus = temInt = (short)((ushort)(calibData[10] << 8) | calibData[9]);
|
|
rollPlus = temInt = (short)((ushort)(calibData[12] << 8) | calibData[11]);
|
|
pitchMinus = temInt = (short)((ushort)(calibData[14] << 8) | calibData[13]);
|
|
yawMinus = temInt = (short)((ushort)(calibData[16] << 8) | calibData[15]);
|
|
rollMinus = temInt = (short)((ushort)(calibData[18] << 8) | calibData[17]);
|
|
}
|
|
else
|
|
{
|
|
pitchPlus = temInt = (short)((ushort)(calibData[8] << 8) | calibData[7]);
|
|
pitchMinus = temInt = (short)((ushort)(calibData[10] << 8) | calibData[9]);
|
|
yawPlus = temInt = (short)((ushort)(calibData[12] << 8) | calibData[11]);
|
|
yawMinus = temInt = (short)((ushort)(calibData[14] << 8) | calibData[13]);
|
|
rollPlus = temInt = (short)((ushort)(calibData[16] << 8) | calibData[15]);
|
|
rollMinus = temInt = (short)((ushort)(calibData[18] << 8) | calibData[17]);
|
|
}
|
|
|
|
gyroSpeedPlus = temInt = (short)((ushort)(calibData[20] << 8) | calibData[19]);
|
|
gyroSpeedMinus = temInt = (short)((ushort)(calibData[22] << 8) | calibData[21]);
|
|
accelXPlus = temInt = (short)((ushort)(calibData[24] << 8) | calibData[23]);
|
|
accelXMinus = temInt = (short)((ushort)(calibData[26] << 8) | calibData[25]);
|
|
|
|
accelYPlus = temInt = (short)((ushort)(calibData[28] << 8) | calibData[27]);
|
|
accelYMinus = temInt = (short)((ushort)(calibData[30] << 8) | calibData[29]);
|
|
|
|
accelZPlus = temInt = (short)((ushort)(calibData[32] << 8) | calibData[31]);
|
|
accelZMinus = temInt = (short)((ushort)(calibData[34] << 8) | calibData[33]);
|
|
|
|
int gyroSpeed2x = temInt = (gyroSpeedPlus + gyroSpeedMinus);
|
|
calibrationData[0].sensNumer = gyroSpeed2x* SixAxis.GYRO_RES_IN_DEG_SEC;
|
|
calibrationData[0].sensDenom = pitchPlus - pitchMinus;
|
|
|
|
calibrationData[1].sensNumer = gyroSpeed2x* SixAxis.GYRO_RES_IN_DEG_SEC;
|
|
calibrationData[1].sensDenom = yawPlus - yawMinus;
|
|
|
|
calibrationData[2].sensNumer = gyroSpeed2x* SixAxis.GYRO_RES_IN_DEG_SEC;
|
|
calibrationData[2].sensDenom = rollPlus - rollMinus;
|
|
|
|
int accelRange = temInt = accelXPlus - accelXMinus;
|
|
calibrationData[3].bias = accelXPlus - accelRange / 2;
|
|
calibrationData[3].sensNumer = 2 * SixAxis.ACC_RES_PER_G;
|
|
calibrationData[3].sensDenom = accelRange;
|
|
|
|
accelRange = temInt = accelYPlus - accelYMinus;
|
|
calibrationData[4].bias = accelYPlus - accelRange / 2;
|
|
calibrationData[4].sensNumer = 2 * SixAxis.ACC_RES_PER_G;
|
|
calibrationData[4].sensDenom = accelRange;
|
|
|
|
accelRange = temInt = accelZPlus - accelZMinus;
|
|
calibrationData[5].bias = accelZPlus - accelRange / 2;
|
|
calibrationData[5].sensNumer = 2 * SixAxis.ACC_RES_PER_G;
|
|
calibrationData[5].sensDenom = accelRange;
|
|
|
|
// Check that denom will not be zero.
|
|
calibrationDone = calibrationData[0].sensDenom != 0 &&
|
|
calibrationData[1].sensDenom != 0 &&
|
|
calibrationData[2].sensDenom != 0 &&
|
|
accelRange != 0;
|
|
}
|
|
|
|
private void applyCalibs(ref int yaw, ref int pitch, ref int roll,
|
|
ref int accelX, ref int accelY, ref int accelZ)
|
|
{
|
|
CalibData current = calibrationData[0];
|
|
temInt = pitch - current.bias;
|
|
pitch = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
|
|
current = calibrationData[1];
|
|
temInt = yaw - current.bias;
|
|
yaw = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
|
|
current = calibrationData[2];
|
|
temInt = roll - current.bias;
|
|
roll = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
|
|
current = calibrationData[3];
|
|
temInt = accelX - current.bias;
|
|
accelX = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
|
|
current = calibrationData[4];
|
|
temInt = accelY - current.bias;
|
|
accelY = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
|
|
current = calibrationData[5];
|
|
temInt = accelZ - current.bias;
|
|
accelZ = temInt = (int)(temInt * (current.sensNumer / (float)current.sensDenom));
|
|
}
|
|
|
|
public unsafe void handleSixaxis(byte* gyro, byte* accel, DS4State state,
|
|
double elapsedDelta)
|
|
{
|
|
int currentYaw = (short)((ushort)(gyro[3] << 8) | gyro[2]);
|
|
int currentPitch = (short)((ushort)(gyro[1] << 8) | gyro[0]);
|
|
int currentRoll = (short)((ushort)(gyro[5] << 8) | gyro[4]);
|
|
int AccelX = (short)((ushort)(accel[1] << 8) | accel[0]);
|
|
int AccelY = (short)((ushort)(accel[3] << 8) | accel[2]);
|
|
int AccelZ = (short)((ushort)(accel[5] << 8) | accel[4]);
|
|
|
|
if (calibrationDone)
|
|
applyCalibs(ref currentYaw, ref currentPitch, ref currentRoll, ref AccelX, ref AccelY, ref AccelZ);
|
|
|
|
SixAxisEventArgs args = null;
|
|
if (AccelX != 0 || AccelY != 0 || AccelZ != 0)
|
|
{
|
|
if (SixAccelMoved != null)
|
|
{
|
|
sPrev.copy(now);
|
|
now.populate(currentYaw, currentPitch, currentRoll,
|
|
AccelX, AccelY, AccelZ, elapsedDelta, sPrev);
|
|
|
|
args = new SixAxisEventArgs(state.ReportTimeStamp, now);
|
|
state.Motion = now;
|
|
SixAccelMoved(this, args);
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool fixupInvertedGyroAxis()
|
|
{
|
|
bool result = false;
|
|
// Some, not all, DS4 rev1 gamepads have an inverted YAW gyro axis calibration value (sensNumber>0 but at the same time sensDenom value is <0 while other two axies have both attributes >0).
|
|
// If this gamepad has YAW calibration with weird mixed values then fix it automatically to workaround inverted YAW axis problem.
|
|
if (calibrationData[1].sensNumer > 0 && calibrationData[1].sensDenom < 0 &&
|
|
calibrationData[0].sensDenom > 0 && calibrationData[2].sensDenom > 0)
|
|
{
|
|
calibrationData[1].sensDenom *= -1;
|
|
result = true; // Fixed inverted axis
|
|
}
|
|
return result;
|
|
}
|
|
|
|
}
|
|
}
|