WiimoteEmu: Add serialization for DesiredWiimoteState.
This commit is contained in:
parent
a2dadbb2f0
commit
f8518b2ff6
|
@ -279,6 +279,7 @@ add_library(core
|
||||||
HW/WiimoteCommon/WiimoteReport.h
|
HW/WiimoteCommon/WiimoteReport.h
|
||||||
HW/WiimoteEmu/Camera.cpp
|
HW/WiimoteEmu/Camera.cpp
|
||||||
HW/WiimoteEmu/Camera.h
|
HW/WiimoteEmu/Camera.h
|
||||||
|
HW/WiimoteEmu/DesiredWiimoteState.cpp
|
||||||
HW/WiimoteEmu/DesiredWiimoteState.h
|
HW/WiimoteEmu/DesiredWiimoteState.h
|
||||||
HW/WiimoteEmu/Dynamics.cpp
|
HW/WiimoteEmu/Dynamics.cpp
|
||||||
HW/WiimoteEmu/Dynamics.h
|
HW/WiimoteEmu/Dynamics.h
|
||||||
|
|
|
@ -0,0 +1,226 @@
|
||||||
|
// Copyright 2022 Dolphin Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstring>
|
||||||
|
#include <optional>
|
||||||
|
#include <type_traits>
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
|
#include "Common/CommonTypes.h"
|
||||||
|
|
||||||
|
#include "Core/HW/WiimoteEmu/DesiredWiimoteState.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Classic.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/DrawsomeTablet.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Drums.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Guitar.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Nunchuk.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Shinkansen.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/TaTaCon.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/Turntable.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/Extension/UDrawTablet.h"
|
||||||
|
#include "Core/HW/WiimoteEmu/MotionPlus.h"
|
||||||
|
|
||||||
|
namespace WiimoteEmu
|
||||||
|
{
|
||||||
|
SerializedWiimoteState SerializeDesiredState(const DesiredWiimoteState& state)
|
||||||
|
{
|
||||||
|
// If no motion or extension data is present, we can serialize into a much smaller representation,
|
||||||
|
// which reduces data sent over the network.
|
||||||
|
const bool is_buttons_only =
|
||||||
|
(state.acceleration == DesiredWiimoteState::DEFAULT_ACCELERATION &&
|
||||||
|
state.camera_points == DesiredWiimoteState::DEFAULT_CAMERA &&
|
||||||
|
!state.motion_plus.has_value() && state.extension.data.index() == ExtensionNumber::NONE);
|
||||||
|
|
||||||
|
SerializedWiimoteState s;
|
||||||
|
s.length = 0;
|
||||||
|
s.data[s.length++] =
|
||||||
|
u8((state.buttons.a) | (state.buttons.b << 1) | (state.buttons.plus << 2) |
|
||||||
|
(state.buttons.minus << 3) | (state.buttons.one << 4) | (state.buttons.two << 5) |
|
||||||
|
(state.buttons.home << 6) | ((is_buttons_only ? 0 : 1) << 7));
|
||||||
|
const u8 dpad = u8((state.buttons.up) | (state.buttons.down << 1) | (state.buttons.left << 2) |
|
||||||
|
(state.buttons.right << 3));
|
||||||
|
if (is_buttons_only)
|
||||||
|
{
|
||||||
|
// There's four unused bits here, which could be used to further optimize for common situations.
|
||||||
|
s.data[s.length++] = dpad;
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
const u16 accel_x = state.acceleration.value.x; // 10 bits
|
||||||
|
const u16 accel_y = state.acceleration.value.y; // 10 bits
|
||||||
|
const u16 accel_z = state.acceleration.value.z; // 10 bits
|
||||||
|
const u16 camera_p0_x = state.camera_points[0].position.x; // 10 bits
|
||||||
|
const u16 camera_p0_y = state.camera_points[0].position.y; // 10 bits
|
||||||
|
const u16 camera_p1_x = state.camera_points[1].position.x; // 10 bits
|
||||||
|
const u16 camera_p1_y = state.camera_points[1].position.y; // 10 bits
|
||||||
|
const u8 camera_p0_size = state.camera_points[0].size; // 4 bits
|
||||||
|
const u8 camera_p1_size = state.camera_points[1].size; // 4 bits
|
||||||
|
static_assert(std::variant_size_v<DesiredExtensionState::ExtensionData> < (1 << 5));
|
||||||
|
const u8 extension = u8(state.extension.data.index()); // maximum of 5 bits
|
||||||
|
const u8 motion_plus = state.motion_plus.has_value() ? 1 : 0;
|
||||||
|
s.data[s.length++] = u8(dpad | (camera_p0_size << 4));
|
||||||
|
s.data[s.length++] = u8(camera_p0_x);
|
||||||
|
s.data[s.length++] = u8(camera_p0_y);
|
||||||
|
s.data[s.length++] = u8(camera_p1_x);
|
||||||
|
s.data[s.length++] = u8(camera_p1_y);
|
||||||
|
s.data[s.length++] = u8(((camera_p0_x >> 8) & 3) | (((camera_p0_y >> 8) & 3) << 2) |
|
||||||
|
(((camera_p1_x >> 8) & 3) << 4) | (((camera_p1_y >> 8) & 3) << 6));
|
||||||
|
s.data[s.length++] =
|
||||||
|
u8(((accel_x >> 8) & 3) | (((accel_y >> 8) & 3) << 2) | (camera_p1_size << 4));
|
||||||
|
s.data[s.length++] = u8(accel_x);
|
||||||
|
s.data[s.length++] = u8(accel_y);
|
||||||
|
s.data[s.length++] = u8(accel_z);
|
||||||
|
s.data[s.length++] = u8(((accel_z >> 8) & 3) | (motion_plus << 2) | (extension << 3));
|
||||||
|
|
||||||
|
if (motion_plus)
|
||||||
|
{
|
||||||
|
const u16 pitch_slow = state.motion_plus->is_slow.x ? 1 : 0;
|
||||||
|
const u16 roll_slow = state.motion_plus->is_slow.y ? 1 : 0;
|
||||||
|
const u16 yaw_slow = state.motion_plus->is_slow.z ? 1 : 0;
|
||||||
|
const u16 pitch_value = state.motion_plus->gyro.value.x; // 14 bits
|
||||||
|
const u16 roll_value = state.motion_plus->gyro.value.y; // 14 bits
|
||||||
|
const u16 yaw_value = state.motion_plus->gyro.value.z; // 14 bits
|
||||||
|
s.data[s.length++] = u8(pitch_value);
|
||||||
|
s.data[s.length++] = u8((pitch_value >> 8) | (pitch_slow << 7));
|
||||||
|
s.data[s.length++] = u8(roll_value);
|
||||||
|
s.data[s.length++] = u8((roll_value >> 8) | (roll_slow << 7));
|
||||||
|
s.data[s.length++] = u8(yaw_value);
|
||||||
|
s.data[s.length++] = u8((yaw_value >> 8) | (yaw_slow << 7));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (extension)
|
||||||
|
{
|
||||||
|
std::visit(
|
||||||
|
[&s](const auto& arg) {
|
||||||
|
using T = std::decay_t<decltype(arg)>;
|
||||||
|
if constexpr (!std::is_same_v<std::monostate, T>)
|
||||||
|
{
|
||||||
|
static_assert(sizeof(arg) <= 6);
|
||||||
|
static_assert(std::is_trivially_copyable_v<T>);
|
||||||
|
std::memcpy(&s.data[s.length], &arg, sizeof(arg));
|
||||||
|
s.length += sizeof(arg);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
state.extension.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
static bool DeserializeExtensionState(DesiredWiimoteState* state,
|
||||||
|
const SerializedWiimoteState& serialized, size_t offset)
|
||||||
|
{
|
||||||
|
if (serialized.length < offset + sizeof(T))
|
||||||
|
return false;
|
||||||
|
auto& e = state->extension.data.emplace<T>();
|
||||||
|
static_assert(std::is_trivially_copyable_v<T>);
|
||||||
|
std::memcpy(&e, &serialized.data[offset], sizeof(T));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DeserializeDesiredState(DesiredWiimoteState* state, const SerializedWiimoteState& serialized)
|
||||||
|
{
|
||||||
|
// clear state
|
||||||
|
state->buttons.hex = 0;
|
||||||
|
state->acceleration = DesiredWiimoteState::DEFAULT_ACCELERATION;
|
||||||
|
state->camera_points = DesiredWiimoteState::DEFAULT_CAMERA;
|
||||||
|
state->motion_plus = std::nullopt;
|
||||||
|
state->extension.data = std::monostate();
|
||||||
|
|
||||||
|
if (serialized.length < 2)
|
||||||
|
{
|
||||||
|
// can't be valid
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto& d = serialized.data;
|
||||||
|
state->buttons.a = d[0] & 1;
|
||||||
|
state->buttons.b = (d[0] >> 1) & 1;
|
||||||
|
state->buttons.plus = (d[0] >> 2) & 1;
|
||||||
|
state->buttons.minus = (d[0] >> 3) & 1;
|
||||||
|
state->buttons.one = (d[0] >> 4) & 1;
|
||||||
|
state->buttons.two = (d[0] >> 5) & 1;
|
||||||
|
state->buttons.home = (d[0] >> 6) & 1;
|
||||||
|
const u8 has_motion_or_extension_data = (d[0] >> 7) & 1;
|
||||||
|
state->buttons.up = d[1] & 1;
|
||||||
|
state->buttons.down = (d[1] >> 1) & 1;
|
||||||
|
state->buttons.left = (d[1] >> 2) & 1;
|
||||||
|
state->buttons.right = (d[1] >> 3) & 1;
|
||||||
|
|
||||||
|
if (!has_motion_or_extension_data)
|
||||||
|
{
|
||||||
|
// is button-only state, we're done
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (serialized.length < 12)
|
||||||
|
{
|
||||||
|
// if it's not a button-only state it needs to have at least 12 bytes for the basic Wiimote data
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
state->camera_points[0].size = d[1] >> 4;
|
||||||
|
state->camera_points[0].position.x = d[2] | ((d[6] & 3) << 8);
|
||||||
|
state->camera_points[0].position.y = d[3] | (((d[6] >> 2) & 3) << 8);
|
||||||
|
state->camera_points[1].position.x = d[4] | (((d[6] >> 4) & 3) << 8);
|
||||||
|
state->camera_points[1].position.y = d[5] | (((d[6] >> 6) & 3) << 8);
|
||||||
|
state->camera_points[1].size = d[7] >> 4;
|
||||||
|
state->acceleration.value.x = d[8] | ((d[7] & 3) << 8);
|
||||||
|
state->acceleration.value.y = d[9] | (((d[7] >> 2) & 3) << 8);
|
||||||
|
state->acceleration.value.z = d[10] | ((d[11] & 3) << 8);
|
||||||
|
const u8 has_motion_plus = (d[11] >> 2) & 1;
|
||||||
|
const u8 extension = d[11] >> 3;
|
||||||
|
|
||||||
|
if (has_motion_plus && serialized.length < 18)
|
||||||
|
{
|
||||||
|
// can't hold motion plus data
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (has_motion_plus)
|
||||||
|
{
|
||||||
|
const u16 pitch_value = d[12] | ((d[13] & 0x3f) << 8);
|
||||||
|
const u16 roll_value = d[14] | ((d[15] & 0x3f) << 8);
|
||||||
|
const u16 yaw_value = d[16] | ((d[17] & 0x3f) << 8);
|
||||||
|
const bool pitch_slow = (d[13] & 0x80) != 0;
|
||||||
|
const bool roll_slow = (d[15] & 0x80) != 0;
|
||||||
|
const bool yaw_slow = (d[17] & 0x80) != 0;
|
||||||
|
state->motion_plus = MotionPlus::DataFormat::Data{
|
||||||
|
MotionPlus::DataFormat::GyroRawValue{
|
||||||
|
MotionPlus::DataFormat::GyroType(pitch_value, roll_value, yaw_value)},
|
||||||
|
MotionPlus::DataFormat::SlowType(pitch_slow, roll_slow, yaw_slow)};
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t offset = has_motion_plus ? 18 : 12;
|
||||||
|
switch (extension)
|
||||||
|
{
|
||||||
|
case ExtensionNumber::NONE:
|
||||||
|
return true;
|
||||||
|
case ExtensionNumber::NUNCHUK:
|
||||||
|
return DeserializeExtensionState<Nunchuk::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::CLASSIC:
|
||||||
|
return DeserializeExtensionState<Classic::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::GUITAR:
|
||||||
|
return DeserializeExtensionState<Guitar::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::DRUMS:
|
||||||
|
return DeserializeExtensionState<Drums::DesiredState>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::TURNTABLE:
|
||||||
|
return DeserializeExtensionState<Turntable::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::UDRAW_TABLET:
|
||||||
|
return DeserializeExtensionState<UDrawTablet::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::DRAWSOME_TABLET:
|
||||||
|
return DeserializeExtensionState<DrawsomeTablet::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::TATACON:
|
||||||
|
return DeserializeExtensionState<TaTaCon::DataFormat>(state, serialized, offset);
|
||||||
|
case ExtensionNumber::SHINKANSEN:
|
||||||
|
return DeserializeExtensionState<Shinkansen::DesiredState>(state, serialized, offset);
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} // namespace WiimoteEmu
|
|
@ -29,4 +29,14 @@ struct DesiredWiimoteState
|
||||||
std::optional<MotionPlus::DataFormat::Data> motion_plus = std::nullopt;
|
std::optional<MotionPlus::DataFormat::Data> motion_plus = std::nullopt;
|
||||||
DesiredExtensionState extension;
|
DesiredExtensionState extension;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// For Netplay.
|
||||||
|
struct SerializedWiimoteState
|
||||||
|
{
|
||||||
|
u8 length;
|
||||||
|
std::array<u8, 24> data; // 12 bytes Wiimote, 6 bytes MotionPlus, 6 bytes Extension
|
||||||
|
};
|
||||||
|
|
||||||
|
SerializedWiimoteState SerializeDesiredState(const DesiredWiimoteState& state);
|
||||||
|
bool DeserializeDesiredState(DesiredWiimoteState* state, const SerializedWiimoteState& serialized);
|
||||||
} // namespace WiimoteEmu
|
} // namespace WiimoteEmu
|
||||||
|
|
|
@ -933,6 +933,7 @@
|
||||||
<ClCompile Include="Core\HW\Wiimote.cpp" />
|
<ClCompile Include="Core\HW\Wiimote.cpp" />
|
||||||
<ClCompile Include="Core\HW\WiimoteCommon\DataReport.cpp" />
|
<ClCompile Include="Core\HW\WiimoteCommon\DataReport.cpp" />
|
||||||
<ClCompile Include="Core\HW\WiimoteEmu\Camera.cpp" />
|
<ClCompile Include="Core\HW\WiimoteEmu\Camera.cpp" />
|
||||||
|
<ClCompile Include="Core\HW\WiimoteEmu\DesiredWiimoteState.cpp" />
|
||||||
<ClCompile Include="Core\HW\WiimoteEmu\Dynamics.cpp" />
|
<ClCompile Include="Core\HW\WiimoteEmu\Dynamics.cpp" />
|
||||||
<ClCompile Include="Core\HW\WiimoteEmu\EmuSubroutines.cpp" />
|
<ClCompile Include="Core\HW\WiimoteEmu\EmuSubroutines.cpp" />
|
||||||
<ClCompile Include="Core\HW\WiimoteEmu\Encryption.cpp" />
|
<ClCompile Include="Core\HW\WiimoteEmu\Encryption.cpp" />
|
||||||
|
|
Loading…
Reference in New Issue