WGInput: use c++/winrt
This commit is contained in:
parent
76d2e47444
commit
6bc8ab7001
|
@ -4,41 +4,27 @@
|
||||||
#include "InputCommon/ControllerInterface/WGInput/WGInput.h"
|
#include "InputCommon/ControllerInterface/WGInput/WGInput.h"
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
|
#include <map>
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
|
|
||||||
#include <fmt/format.h>
|
#include <fmt/format.h>
|
||||||
|
|
||||||
#include <roapi.h>
|
// TODO winrt translates com failures to c++ exceptions, so we must use try/catch in this file to
|
||||||
#include <windows.gaming.input.h>
|
// prevent possible errors from terminating Dolphin.
|
||||||
#include <wrl.h>
|
#include <winrt/windows.devices.haptics.h>
|
||||||
|
#include <winrt/windows.devices.power.h>
|
||||||
|
#include <winrt/windows.foundation.collections.h>
|
||||||
|
#include <winrt/windows.gaming.input.h>
|
||||||
|
#include <winrt/windows.system.power.h>
|
||||||
|
#pragma comment(lib, "windowsapp")
|
||||||
|
|
||||||
#include "Common/DynamicLibrary.h"
|
|
||||||
#include "Common/HRWrap.h"
|
|
||||||
#include "Common/Logging/Log.h"
|
#include "Common/Logging/Log.h"
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
||||||
|
|
||||||
using PIsApiSetImplemented = BOOL(APIENTRY*)(PCSTR Contract);
|
namespace WGI = winrt::Windows::Gaming::Input;
|
||||||
using PRoInitialize = decltype(&RoInitialize);
|
namespace Haptics = winrt::Windows::Devices::Haptics;
|
||||||
using PRoUninitialize = decltype(&RoUninitialize);
|
using winrt::Windows::Foundation::Collections::IVectorView;
|
||||||
using PRoGetActivationFactory = decltype(&RoGetActivationFactory);
|
|
||||||
using PWindowsCreateStringReference = decltype(&WindowsCreateStringReference);
|
|
||||||
using PWindowsGetStringRawBuffer = decltype(&WindowsGetStringRawBuffer);
|
|
||||||
|
|
||||||
namespace WGI = ABI::Windows::Gaming::Input;
|
|
||||||
using ABI::Windows::Foundation::Collections::IVectorView;
|
|
||||||
using Microsoft::WRL::ComPtr;
|
|
||||||
|
|
||||||
namespace
|
|
||||||
{
|
|
||||||
bool g_runtime_initialized = false;
|
|
||||||
bool g_runtime_needs_deinit = false;
|
|
||||||
PRoInitialize g_RoInitialize_address = nullptr;
|
|
||||||
PRoUninitialize g_RoUninitialize_address = nullptr;
|
|
||||||
PRoGetActivationFactory g_RoGetActivationFactory_address = nullptr;
|
|
||||||
PWindowsCreateStringReference g_WindowsCreateStringReference_address = nullptr;
|
|
||||||
PWindowsGetStringRawBuffer g_WindowsGetStringRawBuffer_address = nullptr;
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
namespace ciface::WGInput
|
namespace ciface::WGInput
|
||||||
{
|
{
|
||||||
|
@ -47,7 +33,7 @@ static constexpr std::string_view SOURCE_NAME = "WGInput";
|
||||||
// These names correspond to the values of the GameControllerButtonLabel enum.
|
// These names correspond to the values of the GameControllerButtonLabel enum.
|
||||||
// "None" is not used.
|
// "None" is not used.
|
||||||
// There are some overlapping names assuming no device exposes both
|
// There are some overlapping names assuming no device exposes both
|
||||||
// GameControllerButtonLabel_XboxLeftBumper and GameControllerButtonLabel_LeftBumper.
|
// GameControllerButtonLabel::XboxLeftBumper and GameControllerButtonLabel::LeftBumper.
|
||||||
// If needed we can prepend "Xbox" to relevant input names on conflict in the future.
|
// If needed we can prepend "Xbox" to relevant input names on conflict in the future.
|
||||||
static constexpr std::array wgi_button_names = {
|
static constexpr std::array wgi_button_names = {
|
||||||
"None", "Back", "Start", "Menu", "View", "Pad N",
|
"None", "Back", "Start", "Menu", "View", "Pad N",
|
||||||
|
@ -90,7 +76,7 @@ static constexpr MemberName<WGI::GamepadVibration, double> gamepad_motor_names[]
|
||||||
class Device : public Core::Device
|
class Device : public Core::Device
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Device(std::string name, ComPtr<WGI::IRawGameController> raw_controller, WGI::IGamepad* gamepad)
|
Device(std::string name, WGI::RawGameController raw_controller, WGI::Gamepad gamepad)
|
||||||
: m_name(std::move(name)), m_raw_controller(raw_controller), m_gamepad(gamepad)
|
: m_name(std::move(name)), m_raw_controller(raw_controller), m_gamepad(gamepad)
|
||||||
{
|
{
|
||||||
// Buttons:
|
// Buttons:
|
||||||
|
@ -123,9 +109,7 @@ public:
|
||||||
if (use_raw_controller_axes)
|
if (use_raw_controller_axes)
|
||||||
{
|
{
|
||||||
// Axes:
|
// Axes:
|
||||||
INT32 axis_count = 0;
|
m_axes.resize(m_raw_controller.AxisCount());
|
||||||
if (SUCCEEDED(m_raw_controller->get_AxisCount(&axis_count)))
|
|
||||||
m_axes.resize(axis_count);
|
|
||||||
|
|
||||||
u32 i = 0;
|
u32 i = 0;
|
||||||
for (auto& axis : m_axes)
|
for (auto& axis : m_axes)
|
||||||
|
@ -144,26 +128,23 @@ public:
|
||||||
// Switches (Hats):
|
// Switches (Hats):
|
||||||
if (use_raw_controller_switches)
|
if (use_raw_controller_switches)
|
||||||
{
|
{
|
||||||
INT32 switch_count = 0;
|
m_switches.resize(m_raw_controller.SwitchCount());
|
||||||
if (SUCCEEDED(m_raw_controller->get_SwitchCount(&switch_count)))
|
|
||||||
m_switches.resize(switch_count);
|
|
||||||
|
|
||||||
u32 i = 0;
|
u32 i = 0;
|
||||||
for (auto& swtch : m_switches)
|
for (auto& swtch : m_switches)
|
||||||
{
|
{
|
||||||
using gcsp = WGI::GameControllerSwitchPosition;
|
using gcsp = WGI::GameControllerSwitchPosition;
|
||||||
|
|
||||||
WGI::GameControllerSwitchKind switch_kind;
|
WGI::GameControllerSwitchKind switch_kind = m_raw_controller.GetSwitchKind(i);
|
||||||
m_raw_controller->GetSwitchKind(i, &switch_kind);
|
|
||||||
|
|
||||||
AddInput(new IndexedSwitch(&swtch, i, gcsp::GameControllerSwitchPosition_Up));
|
AddInput(new IndexedSwitch(&swtch, i, gcsp::Up));
|
||||||
AddInput(new IndexedSwitch(&swtch, i, gcsp::GameControllerSwitchPosition_Down));
|
AddInput(new IndexedSwitch(&swtch, i, gcsp::Down));
|
||||||
|
|
||||||
if (switch_kind != WGI::GameControllerSwitchKind_TwoWay)
|
if (switch_kind != WGI::GameControllerSwitchKind::TwoWay)
|
||||||
{
|
{
|
||||||
// If it's not a "two-way" switch (up/down only) then add the left/right inputs.
|
// If it's not a "two-way" switch (up/down only) then add the left/right inputs.
|
||||||
AddInput(new IndexedSwitch(&swtch, i, gcsp::GameControllerSwitchPosition_Left));
|
AddInput(new IndexedSwitch(&swtch, i, gcsp::Left));
|
||||||
AddInput(new IndexedSwitch(&swtch, i, gcsp::GameControllerSwitchPosition_Right));
|
AddInput(new IndexedSwitch(&swtch, i, gcsp::Right));
|
||||||
}
|
}
|
||||||
|
|
||||||
++i;
|
++i;
|
||||||
|
@ -174,21 +155,16 @@ public:
|
||||||
PopulateHaptics();
|
PopulateHaptics();
|
||||||
|
|
||||||
// Battery:
|
// Battery:
|
||||||
if (SUCCEEDED(m_raw_controller->QueryInterface(&m_controller_battery)) && m_controller_battery)
|
if (UpdateBatteryLevel())
|
||||||
{
|
AddInput(new Battery(&m_battery_level));
|
||||||
// It seems many controllers provide IGameControllerBatteryInfo with no battery info.
|
|
||||||
if (UpdateBatteryLevel())
|
|
||||||
AddInput(new Battery(&m_battery_level));
|
|
||||||
else
|
|
||||||
m_controller_battery = nullptr;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int GetSortPriority() const override { return -1; }
|
int GetSortPriority() const override { return -1; }
|
||||||
|
|
||||||
|
const WGI::RawGameController GetRawGameController() const { return m_raw_controller; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// `boolean` comes from Windows API. (typedef of unsigned char)
|
using ButtonValueType = u8;
|
||||||
using ButtonValueType = boolean;
|
|
||||||
|
|
||||||
class Button : public Input
|
class Button : public Input
|
||||||
{
|
{
|
||||||
|
@ -312,7 +288,7 @@ private:
|
||||||
public:
|
public:
|
||||||
IndexedSwitch(const WGI::GameControllerSwitchPosition* swtch, u32 index,
|
IndexedSwitch(const WGI::GameControllerSwitchPosition* swtch, u32 index,
|
||||||
WGI::GameControllerSwitchPosition direction)
|
WGI::GameControllerSwitchPosition direction)
|
||||||
: m_switch(*swtch), m_index(index), m_direction(direction)
|
: m_switch(*swtch), m_index(index), m_direction(static_cast<int32_t>(direction))
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
std::string GetName() const override
|
std::string GetName() const override
|
||||||
|
@ -321,20 +297,20 @@ private:
|
||||||
}
|
}
|
||||||
ControlState GetState() const override
|
ControlState GetState() const override
|
||||||
{
|
{
|
||||||
if (m_switch == WGI::GameControllerSwitchPosition_Center)
|
if (m_switch == WGI::GameControllerSwitchPosition::Center)
|
||||||
return 0.0;
|
return 0.0;
|
||||||
|
|
||||||
// All of the "inbetween" states (e.g. Up-Right) are one-off from the four cardinal
|
// All of the "inbetween" states (e.g. Up-Right) are one-off from the four cardinal
|
||||||
// directions. This tests that the current switch state value is within 1 of the desired
|
// directions. This tests that the current switch state value is within 1 of the desired
|
||||||
// state.
|
// state.
|
||||||
const auto direction_diff = std::abs(m_switch - m_direction);
|
const auto direction_diff = std::abs(static_cast<int32_t>(m_switch) - m_direction);
|
||||||
return ControlState(direction_diff <= 1 || direction_diff == 7);
|
return ControlState(direction_diff <= 1 || direction_diff == 7);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
const WGI::GameControllerSwitchPosition& m_switch;
|
const WGI::GameControllerSwitchPosition& m_switch;
|
||||||
const u32 m_index;
|
const u32 m_index;
|
||||||
const WGI::GameControllerSwitchPosition m_direction;
|
const int32_t m_direction;
|
||||||
};
|
};
|
||||||
|
|
||||||
class Battery : public Input
|
class Battery : public Input
|
||||||
|
@ -355,9 +331,8 @@ private:
|
||||||
class SimpleHaptics : public Output
|
class SimpleHaptics : public Output
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
SimpleHaptics(ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsController> haptics,
|
SimpleHaptics(Haptics::SimpleHapticsController haptics,
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsControllerFeedback> feedback,
|
Haptics::SimpleHapticsControllerFeedback feedback, u32 haptics_index)
|
||||||
u32 haptics_index)
|
|
||||||
: m_haptics(haptics), m_feedback(feedback), m_haptics_index(haptics_index)
|
: m_haptics(haptics), m_feedback(feedback), m_haptics_index(haptics_index)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -370,17 +345,17 @@ private:
|
||||||
m_current_state = state;
|
m_current_state = state;
|
||||||
|
|
||||||
if (state)
|
if (state)
|
||||||
m_haptics->SendHapticFeedbackWithIntensity(m_feedback.Get(), state);
|
m_haptics.SendHapticFeedback(m_feedback, state);
|
||||||
else
|
else
|
||||||
m_haptics->StopFeedback();
|
m_haptics.StopFeedback();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
u32 GetHapticsIndex() const { return m_haptics_index; }
|
u32 GetHapticsIndex() const { return m_haptics_index; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsController> m_haptics;
|
Haptics::SimpleHapticsController m_haptics;
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsControllerFeedback> m_feedback;
|
Haptics::SimpleHapticsControllerFeedback m_feedback;
|
||||||
const u32 m_haptics_index;
|
const u32 m_haptics_index;
|
||||||
ControlState m_current_state = 0;
|
ControlState m_current_state = 0;
|
||||||
};
|
};
|
||||||
|
@ -388,9 +363,9 @@ private:
|
||||||
class NamedFeedback final : public SimpleHaptics
|
class NamedFeedback final : public SimpleHaptics
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
NamedFeedback(ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsController> haptics,
|
NamedFeedback(Haptics::SimpleHapticsController haptics,
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsControllerFeedback> feedback,
|
Haptics::SimpleHapticsControllerFeedback feedback, u32 haptics_index,
|
||||||
u32 haptics_index, std::string_view feedback_name)
|
std::string_view feedback_name)
|
||||||
: SimpleHaptics(haptics, feedback, haptics_index), m_feedback_name(feedback_name)
|
: SimpleHaptics(haptics, feedback, haptics_index), m_feedback_name(feedback_name)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -406,18 +381,15 @@ private:
|
||||||
void PopulateButtons()
|
void PopulateButtons()
|
||||||
{
|
{
|
||||||
// Using RawGameController for buttons because it gives us a nice array instead of a bitmask.
|
// Using RawGameController for buttons because it gives us a nice array instead of a bitmask.
|
||||||
INT32 button_count = 0;
|
m_buttons.resize(m_raw_controller.ButtonCount());
|
||||||
if (SUCCEEDED(m_raw_controller->get_ButtonCount(&button_count)))
|
|
||||||
m_buttons.resize(button_count);
|
|
||||||
|
|
||||||
u32 i = 0;
|
u32 i = 0;
|
||||||
for (auto& button : m_buttons)
|
for (const auto& button : m_buttons)
|
||||||
{
|
{
|
||||||
WGI::GameControllerButtonLabel lbl = WGI::GameControllerButtonLabel_None;
|
const WGI::GameControllerButtonLabel lbl = m_raw_controller.GetButtonLabel(i);
|
||||||
m_raw_controller->GetButtonLabel(i, &lbl);
|
const int32_t button_name_idx = static_cast<int32_t>(lbl);
|
||||||
|
if (lbl != WGI::GameControllerButtonLabel::None && button_name_idx < wgi_button_names.size())
|
||||||
if (lbl != WGI::GameControllerButtonLabel_None && lbl < wgi_button_names.size())
|
AddInput(new NamedButton(&button, wgi_button_names[button_name_idx]));
|
||||||
AddInput(new NamedButton(&button, wgi_button_names[lbl]));
|
|
||||||
else
|
else
|
||||||
AddInput(new IndexedButton(&button, i));
|
AddInput(new IndexedButton(&button, i));
|
||||||
|
|
||||||
|
@ -427,68 +399,31 @@ private:
|
||||||
|
|
||||||
void PopulateHaptics()
|
void PopulateHaptics()
|
||||||
{
|
{
|
||||||
WGI::IRawGameController2* rgc2 = nullptr;
|
static const std::map<uint16_t, std::string> waveform_name_map{
|
||||||
if (FAILED(m_raw_controller->QueryInterface(&rgc2)) || !rgc2)
|
{Haptics::KnownSimpleHapticsControllerWaveforms::Click(), "Click"},
|
||||||
return;
|
{Haptics::KnownSimpleHapticsControllerWaveforms::BuzzContinuous(), "Buzz"},
|
||||||
|
{Haptics::KnownSimpleHapticsControllerWaveforms::RumbleContinuous(), "Rumble"},
|
||||||
|
};
|
||||||
|
|
||||||
IVectorView<ABI::Windows::Devices::Haptics::SimpleHapticsController*>* haptics = nullptr;
|
// SimpleHapticsControllers is available from Windows 1709.
|
||||||
if (FAILED(rgc2->get_SimpleHapticsControllers(&haptics)) || !haptics)
|
u32 haptics_index = 0;
|
||||||
return;
|
for (auto haptics_controller : m_raw_controller.SimpleHapticsControllers())
|
||||||
|
|
||||||
unsigned int haptic_count = 0;
|
|
||||||
if (FAILED(haptics->get_Size(&haptic_count)))
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (unsigned int h = 0; h != haptic_count; ++h)
|
|
||||||
{
|
{
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsController> haptic;
|
for (Haptics::SimpleHapticsControllerFeedback feedback :
|
||||||
if (FAILED(haptics->GetAt(h, &haptic)))
|
haptics_controller.SupportedFeedback())
|
||||||
continue;
|
|
||||||
|
|
||||||
IVectorView<ABI::Windows::Devices::Haptics::SimpleHapticsControllerFeedback*>* feedbacks =
|
|
||||||
nullptr;
|
|
||||||
if (FAILED(haptic->get_SupportedFeedback(&feedbacks)) || !feedbacks)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
unsigned int feedback_count = 0;
|
|
||||||
if (FAILED(haptics->get_Size(&feedback_count)))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (unsigned int f = 0; f != feedback_count; ++f)
|
|
||||||
{
|
{
|
||||||
ComPtr<ABI::Windows::Devices::Haptics::ISimpleHapticsControllerFeedback> feedback;
|
const uint16_t waveform = feedback.Waveform();
|
||||||
if (FAILED(feedbacks->GetAt(f, &feedback)))
|
auto waveform_name_it = waveform_name_map.find(waveform);
|
||||||
continue;
|
if (waveform_name_it == waveform_name_map.end())
|
||||||
|
|
||||||
UINT16 waveform = 0;
|
|
||||||
if (FAILED(feedback->get_Waveform(&waveform)))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
std::string_view waveform_name{};
|
|
||||||
|
|
||||||
// Haptic Usage Page from HID spec.
|
|
||||||
switch (waveform)
|
|
||||||
{
|
{
|
||||||
case 0x1003:
|
WARN_LOG_FMT(CONTROLLERINTERFACE,
|
||||||
waveform_name = "Click";
|
"WGInput: Unhandled haptics feedback waveform: 0x{:04x}.", waveform);
|
||||||
break;
|
|
||||||
case 0x1004:
|
|
||||||
waveform_name = "Buzz";
|
|
||||||
break;
|
|
||||||
case 0x1005:
|
|
||||||
waveform_name = "Rumble";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!waveform_name.data())
|
|
||||||
{
|
|
||||||
WARN_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Unknown haptics feedback waveform: 0x{:04x}.",
|
|
||||||
waveform);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
AddOutput(new NamedFeedback(haptics_controller, feedback, haptics_index,
|
||||||
AddOutput(new NamedFeedback(haptic, feedback, h, waveform_name));
|
waveform_name_it->second));
|
||||||
}
|
}
|
||||||
|
++haptics_index;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -499,60 +434,50 @@ private:
|
||||||
void UpdateInput() override
|
void UpdateInput() override
|
||||||
{
|
{
|
||||||
// IRawGameController:
|
// IRawGameController:
|
||||||
const auto button_count = UINT32(m_buttons.size());
|
static_assert(sizeof(bool) == sizeof(ButtonValueType));
|
||||||
const auto switch_count = UINT32(m_switches.size());
|
// This cludge is needed to workaround GetCurrentReading wanting array_view<bool>, while
|
||||||
const auto axis_count = UINT32(m_axes.size());
|
// using std::vector<bool> would create a bit-packed array, which isn't wanted. So, we keep
|
||||||
UINT64 timestamp = 0;
|
// vector<u8> and view it as array<bool>.
|
||||||
const HRESULT raw_result = m_raw_controller->GetCurrentReading(
|
auto buttons =
|
||||||
button_count, m_buttons.data(), switch_count, m_switches.data(), axis_count, m_axes.data(),
|
winrt::array_view<bool>(reinterpret_cast<winrt::array_view<bool>::pointer>(&m_buttons[0]),
|
||||||
×tamp);
|
static_cast<winrt::array_view<bool>::size_type>(m_buttons.size()));
|
||||||
if (FAILED(raw_result))
|
m_raw_controller.GetCurrentReading(buttons, m_switches, m_axes);
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE,
|
|
||||||
"WGInput: IRawGameController::GetCurrentReading failed: {}",
|
|
||||||
Common::HRWrap(raw_result));
|
|
||||||
}
|
|
||||||
|
|
||||||
// IGamepad:
|
// IGamepad:
|
||||||
if (m_gamepad)
|
if (m_gamepad)
|
||||||
{
|
m_gamepad_reading = m_gamepad.GetCurrentReading();
|
||||||
const HRESULT gamepad_result = m_gamepad->GetCurrentReading(&m_gamepad_reading);
|
|
||||||
if (FAILED(gamepad_result))
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: IGamepad::GetCurrentReading failed: {}",
|
|
||||||
Common::HRWrap(gamepad_result));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// IGameControllerBatteryInfo:
|
// IGameControllerBatteryInfo:
|
||||||
if (m_controller_battery && !UpdateBatteryLevel())
|
if (!UpdateBatteryLevel())
|
||||||
{
|
|
||||||
DEBUG_LOG_FMT(CONTROLLERINTERFACE, "WGInput: UpdateBatteryLevel failed.");
|
DEBUG_LOG_FMT(CONTROLLERINTERFACE, "WGInput: UpdateBatteryLevel failed.");
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void UpdateMotors() { m_gamepad->put_Vibration(m_state_out); }
|
void UpdateMotors() { m_gamepad.Vibration(m_state_out); }
|
||||||
|
|
||||||
bool UpdateBatteryLevel()
|
bool UpdateBatteryLevel()
|
||||||
{
|
{
|
||||||
ABI::Windows::Devices::Power::IBatteryReport* report = nullptr;
|
const winrt::Windows::Devices::Power::BatteryReport report =
|
||||||
|
m_raw_controller.TryGetBatteryReport();
|
||||||
if (FAILED(m_controller_battery->TryGetBatteryReport(&report)) || !report)
|
if (!report)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
using ABI::Windows::System::Power::BatteryStatus;
|
// TryGetBatteryReport causes a memleak of 0x58 bytes each time it is called, up to at
|
||||||
BatteryStatus status;
|
// least Windows 21H2. This is a big hack to "fix" the refcount.
|
||||||
if (FAILED(report->get_Status(&status)))
|
auto report_ = *(uintptr_t***)&report;
|
||||||
return false;
|
auto rc = &report_[0x40 / 8][0x30 / 8];
|
||||||
|
if (*rc == 2)
|
||||||
|
(*rc)--;
|
||||||
|
|
||||||
|
using winrt::Windows::System::Power::BatteryStatus;
|
||||||
|
const BatteryStatus status = report.Status();
|
||||||
switch (status)
|
switch (status)
|
||||||
{
|
{
|
||||||
case BatteryStatus::BatteryStatus_NotPresent:
|
case BatteryStatus::NotPresent:
|
||||||
m_battery_level = 0;
|
m_battery_level = 0;
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
case BatteryStatus::BatteryStatus_Idle:
|
case BatteryStatus::Idle:
|
||||||
case BatteryStatus::BatteryStatus_Charging:
|
case BatteryStatus::Charging:
|
||||||
m_battery_level = BATTERY_INPUT_MAX_VALUE;
|
m_battery_level = BATTERY_INPUT_MAX_VALUE;
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
@ -560,156 +485,113 @@ private:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ABI::Windows::Foundation::IReference<int>*i_remaining = nullptr, *i_full = nullptr;
|
const int32_t full_value = report.FullChargeCapacityInMilliwattHours().GetInt32();
|
||||||
int remaining_value = 0;
|
const int32_t remaining_value = report.RemainingCapacityInMilliwattHours().GetInt32();
|
||||||
int full_value = 0;
|
m_battery_level = BATTERY_INPUT_MAX_VALUE * remaining_value / full_value;
|
||||||
|
return true;
|
||||||
if (report && SUCCEEDED(report->get_RemainingCapacityInMilliwattHours(&i_remaining)) &&
|
|
||||||
i_remaining && SUCCEEDED(i_remaining->get_Value(&remaining_value)) &&
|
|
||||||
SUCCEEDED(report->get_FullChargeCapacityInMilliwattHours(&i_full)) && i_full &&
|
|
||||||
SUCCEEDED(i_full->get_Value(&full_value)))
|
|
||||||
{
|
|
||||||
m_battery_level = BATTERY_INPUT_MAX_VALUE * remaining_value / full_value;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string m_name;
|
const std::string m_name;
|
||||||
|
|
||||||
ComPtr<WGI::IRawGameController> const m_raw_controller;
|
const WGI::RawGameController m_raw_controller;
|
||||||
std::vector<ButtonValueType> m_buttons;
|
std::vector<ButtonValueType> m_buttons;
|
||||||
std::vector<WGI::GameControllerSwitchPosition> m_switches;
|
std::vector<WGI::GameControllerSwitchPosition> m_switches;
|
||||||
std::vector<double> m_axes;
|
std::vector<double> m_axes;
|
||||||
|
|
||||||
WGI::IGamepad* m_gamepad = nullptr;
|
WGI::Gamepad m_gamepad = nullptr;
|
||||||
WGI::GamepadReading m_gamepad_reading{};
|
WGI::GamepadReading m_gamepad_reading{};
|
||||||
WGI::GamepadVibration m_state_out{};
|
WGI::GamepadVibration m_state_out{};
|
||||||
|
|
||||||
WGI::IGameControllerBatteryInfo* m_controller_battery = nullptr;
|
|
||||||
ControlState m_battery_level = 0;
|
ControlState m_battery_level = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
static bool LoadFunctionPointers()
|
static thread_local bool s_initialized_winrt;
|
||||||
|
static winrt::event_token s_event_added, s_event_removed;
|
||||||
|
|
||||||
|
static void AddDevice(const WGI::RawGameController& raw_game_controller)
|
||||||
{
|
{
|
||||||
static Common::DynamicLibrary winrt_l1_1_0_handle;
|
// Get user-facing device name if available. Otherwise generate a name from vid/pid.
|
||||||
static Common::DynamicLibrary winrt_string_l1_1_0_handle;
|
auto device_name =
|
||||||
if (!winrt_l1_1_0_handle.IsOpen() && !winrt_string_l1_1_0_handle.IsOpen())
|
std::string(StripWhitespace(WStringToUTF8(raw_game_controller.DisplayName().c_str())));
|
||||||
|
if (device_name.empty())
|
||||||
{
|
{
|
||||||
Common::DynamicLibrary kernelBase("KernelBase.dll");
|
const u16 vid = raw_game_controller.HardwareVendorId();
|
||||||
if (!kernelBase.IsOpen())
|
const u16 pid = raw_game_controller.HardwareProductId();
|
||||||
return false;
|
device_name = fmt::format("Device_{:04x}:{:04x}", vid, pid);
|
||||||
|
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to get device name, using {}", device_name);
|
||||||
void* const IsApiSetImplemented_address = kernelBase.GetSymbolAddress("IsApiSetImplemented");
|
|
||||||
if (!IsApiSetImplemented_address)
|
|
||||||
return false;
|
|
||||||
if (!static_cast<PIsApiSetImplemented>(IsApiSetImplemented_address)(
|
|
||||||
"api-ms-win-core-winrt-l1-1-0"))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (!static_cast<PIsApiSetImplemented>(IsApiSetImplemented_address)(
|
|
||||||
"api-ms-win-core-winrt-string-l1-1-0"))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
winrt_l1_1_0_handle.Open("api-ms-win-core-winrt-l1-1-0.dll");
|
|
||||||
winrt_string_l1_1_0_handle.Open("api-ms-win-core-winrt-string-l1-1-0.dll");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!winrt_l1_1_0_handle.IsOpen() || !winrt_string_l1_1_0_handle.IsOpen())
|
WGI::Gamepad gamepad = WGI::Gamepad::FromGameController(raw_game_controller);
|
||||||
return false;
|
// Note that gamepad may be nullptr here. The Device class will deal with this.
|
||||||
|
auto dev = std::make_shared<Device>(std::move(device_name), raw_game_controller, gamepad);
|
||||||
|
|
||||||
g_RoInitialize_address =
|
// Only add if it has some inputs/outputs.
|
||||||
static_cast<PRoInitialize>(winrt_l1_1_0_handle.GetSymbolAddress("RoInitialize"));
|
if (dev->Inputs().size() || dev->Outputs().size())
|
||||||
if (!g_RoInitialize_address)
|
g_controller_interface.AddDevice(std::move(dev));
|
||||||
return false;
|
|
||||||
|
|
||||||
g_RoUninitialize_address =
|
|
||||||
static_cast<PRoUninitialize>(winrt_l1_1_0_handle.GetSymbolAddress("RoUninitialize"));
|
|
||||||
if (!g_RoUninitialize_address)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
g_RoGetActivationFactory_address = static_cast<PRoGetActivationFactory>(
|
|
||||||
winrt_l1_1_0_handle.GetSymbolAddress("RoGetActivationFactory"));
|
|
||||||
if (!g_RoGetActivationFactory_address)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
g_WindowsCreateStringReference_address = static_cast<PWindowsCreateStringReference>(
|
|
||||||
winrt_string_l1_1_0_handle.GetSymbolAddress("WindowsCreateStringReference"));
|
|
||||||
if (!g_WindowsCreateStringReference_address)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
g_WindowsGetStringRawBuffer_address = static_cast<PWindowsGetStringRawBuffer>(
|
|
||||||
winrt_string_l1_1_0_handle.GetSymbolAddress("WindowsGetStringRawBuffer"));
|
|
||||||
if (!g_WindowsGetStringRawBuffer_address)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void RemoveDevice(const WGI::RawGameController& raw_game_controller)
|
||||||
|
{
|
||||||
|
g_controller_interface.RemoveDevice([&](const auto* dev) {
|
||||||
|
if (dev->GetSource() != SOURCE_NAME)
|
||||||
|
return false;
|
||||||
|
return static_cast<const Device*>(dev)->GetRawGameController() == raw_game_controller;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma warning(push)
|
||||||
|
// 'winrt::impl::implements_delegate<winrt::Windows::Foundation::EventHandler<winrt::Windows::Gaming::Input::RawGameController>,H>':
|
||||||
|
// class has virtual functions, but its non-trivial destructor is not virtual; instances of this
|
||||||
|
// class may not be destructed correctly
|
||||||
|
// (H is the lambda)
|
||||||
|
#pragma warning(disable : 4265)
|
||||||
|
|
||||||
void Init()
|
void Init()
|
||||||
{
|
{
|
||||||
if (g_runtime_initialized)
|
try
|
||||||
return;
|
|
||||||
|
|
||||||
if (!LoadFunctionPointers())
|
|
||||||
{
|
{
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: System lacks support, skipping init.");
|
// TODO currently, com gets initialized in STA previously on the thread that calls this
|
||||||
return;
|
// function. need to ensure Devices in g_controller_interface only get accessed by threads that
|
||||||
|
// have had com initialized.
|
||||||
|
// winrt::init_apartment();
|
||||||
|
// s_initialized_winrt = true;
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
s_initialized_winrt = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const HRESULT hr = g_RoInitialize_address(RO_INIT_MULTITHREADED);
|
// XXX If SDL is enabled, these get broken after the first one is fired.
|
||||||
switch (hr)
|
|
||||||
{
|
|
||||||
case S_OK:
|
|
||||||
g_runtime_initialized = true;
|
|
||||||
g_runtime_needs_deinit = true;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case S_FALSE:
|
// These events will be invoked from WGI-managed threadpool.
|
||||||
case RPC_E_CHANGED_MODE:
|
s_event_added = WGI::RawGameController::RawGameControllerAdded(
|
||||||
g_runtime_initialized = true;
|
[](auto&&, const WGI::RawGameController raw_game_controller) {
|
||||||
break;
|
RemoveDevice(raw_game_controller);
|
||||||
|
AddDevice(raw_game_controller);
|
||||||
|
});
|
||||||
|
|
||||||
default:
|
s_event_removed = WGI::RawGameController::RawGameControllerRemoved(
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: RoInitialize failed: {}", Common::HRWrap(hr));
|
[](auto&&, const WGI::RawGameController raw_game_controller) {
|
||||||
break;
|
RemoveDevice(raw_game_controller);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pragma warning(pop)
|
||||||
|
|
||||||
void DeInit()
|
void DeInit()
|
||||||
{
|
{
|
||||||
if (!g_runtime_initialized)
|
WGI::RawGameController::RawGameControllerAdded(s_event_added);
|
||||||
return;
|
WGI::RawGameController::RawGameControllerRemoved(s_event_removed);
|
||||||
|
|
||||||
if (g_runtime_needs_deinit)
|
if (s_initialized_winrt)
|
||||||
{
|
{
|
||||||
g_RoUninitialize_address();
|
winrt::uninit_apartment();
|
||||||
g_runtime_needs_deinit = false;
|
s_initialized_winrt = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_runtime_initialized = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <unsigned int size>
|
|
||||||
static HRESULT WindowsCreateStringReferenceAutoSizeWrapper(wchar_t const (&source_string)[size],
|
|
||||||
HSTRING_HEADER* hstring_header,
|
|
||||||
HSTRING* hstring)
|
|
||||||
{
|
|
||||||
return g_WindowsCreateStringReference_address(source_string, size - 1, hstring_header, hstring);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void PopulateDevices()
|
void PopulateDevices()
|
||||||
{
|
{
|
||||||
if (!g_runtime_initialized)
|
|
||||||
return;
|
|
||||||
|
|
||||||
g_controller_interface.RemoveDevice(
|
|
||||||
[](const auto* dev) { return dev->GetSource() == SOURCE_NAME; });
|
|
||||||
|
|
||||||
// WGI Interfaces to potentially use:
|
// WGI Interfaces to potentially use:
|
||||||
// Gamepad: Buttons, 2x Sticks and 2x Triggers, 4x Vibration Motors
|
// Gamepad: Buttons, 2x Sticks and 2x Triggers, 4x Vibration Motors
|
||||||
// RawGameController: Buttons, Switches (Hats), Axes, Haptics
|
// RawGameController: Buttons, Switches (Hats), Axes, Haptics
|
||||||
|
@ -719,95 +601,14 @@ void PopulateDevices()
|
||||||
// RacingWheel: Buttons, Clutch, Handbrake, PatternShifterGear, Throttle, Wheel, WheelMotor
|
// RacingWheel: Buttons, Clutch, Handbrake, PatternShifterGear, Throttle, Wheel, WheelMotor
|
||||||
// UINavigationController: Directions, Scrolling, etc.
|
// UINavigationController: Directions, Scrolling, etc.
|
||||||
|
|
||||||
HSTRING_HEADER header_raw_game_controller;
|
// RawGameController available from Windows 15063.
|
||||||
HSTRING hstr_raw_game_controller;
|
// properties added in 1709: DisplayName, NonRoamableId, SimpleHapticsControllers
|
||||||
HRESULT hr;
|
|
||||||
if (FAILED(hr = WindowsCreateStringReferenceAutoSizeWrapper(
|
for (const WGI::RawGameController& raw_game_controller :
|
||||||
L"Windows.Gaming.Input.RawGameController", &header_raw_game_controller,
|
WGI::RawGameController::RawGameControllers())
|
||||||
&hstr_raw_game_controller)))
|
|
||||||
{
|
{
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to create string reference: {}",
|
RemoveDevice(raw_game_controller);
|
||||||
Common::HRWrap(hr));
|
AddDevice(raw_game_controller);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
HSTRING_HEADER header_gamepad;
|
|
||||||
HSTRING hstr_gamepad;
|
|
||||||
if (FAILED(hr = WindowsCreateStringReferenceAutoSizeWrapper(L"Windows.Gaming.Input.Gamepad",
|
|
||||||
&header_gamepad, &hstr_gamepad)))
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to create string reference: {}",
|
|
||||||
Common::HRWrap(hr));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ComPtr<WGI::IRawGameControllerStatics> raw_stats;
|
|
||||||
if (FAILED(hr = g_RoGetActivationFactory_address(
|
|
||||||
hstr_raw_game_controller, __uuidof(WGI::IRawGameControllerStatics), &raw_stats)))
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to get IRawGameControllerStatics: {}",
|
|
||||||
Common::HRWrap(hr));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ComPtr<WGI::IGamepadStatics2> gamepad_stats;
|
|
||||||
if (FAILED(hr = g_RoGetActivationFactory_address(hstr_gamepad, __uuidof(WGI::IGamepadStatics2),
|
|
||||||
&gamepad_stats)))
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to get IGamepadStatics2: {}",
|
|
||||||
Common::HRWrap(hr));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
IVectorView<WGI::RawGameController*>* raw_controllers;
|
|
||||||
if (FAILED(hr = raw_stats->get_RawGameControllers(&raw_controllers)))
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: get_RawGameControllers failed: {}",
|
|
||||||
Common::HRWrap(hr));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned int raw_count = 0;
|
|
||||||
raw_controllers->get_Size(&raw_count);
|
|
||||||
for (unsigned i = 0; i != raw_count; ++i)
|
|
||||||
{
|
|
||||||
ComPtr<WGI::IRawGameController> raw_controller;
|
|
||||||
if (SUCCEEDED(raw_controllers->GetAt(i, &raw_controller)) && raw_controller)
|
|
||||||
{
|
|
||||||
std::string device_name;
|
|
||||||
|
|
||||||
// Attempt to get the controller's name.
|
|
||||||
WGI::IRawGameController2* rgc2 = nullptr;
|
|
||||||
if (SUCCEEDED(hr = raw_controller->QueryInterface(&rgc2)) && rgc2)
|
|
||||||
{
|
|
||||||
HSTRING hstr = {};
|
|
||||||
if (SUCCEEDED(hr = rgc2->get_DisplayName(&hstr)) && hstr)
|
|
||||||
{
|
|
||||||
device_name =
|
|
||||||
StripWhitespace(WStringToUTF8(g_WindowsGetStringRawBuffer_address(hstr, nullptr)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (device_name.empty())
|
|
||||||
{
|
|
||||||
ERROR_LOG_FMT(CONTROLLERINTERFACE, "WGInput: Failed to get device name: {}",
|
|
||||||
Common::HRWrap(hr));
|
|
||||||
// Set a default name if we couldn't query the name or it was empty.
|
|
||||||
device_name = "Device";
|
|
||||||
}
|
|
||||||
|
|
||||||
WGI::IGameController* gamecontroller = nullptr;
|
|
||||||
WGI::IGamepad* gamepad = nullptr;
|
|
||||||
if (SUCCEEDED(raw_controller->QueryInterface(&gamecontroller)) && gamecontroller)
|
|
||||||
gamepad_stats->FromGameController(gamecontroller, &gamepad);
|
|
||||||
|
|
||||||
// Note that gamepad may be nullptr here. The Device class will deal with this.
|
|
||||||
auto dev = std::make_shared<Device>(std::move(device_name), raw_controller, gamepad);
|
|
||||||
|
|
||||||
// Only add if it has some inputs/outputs.
|
|
||||||
if (dev->Inputs().size() || dev->Outputs().size())
|
|
||||||
g_controller_interface.AddDevice(std::move(dev));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -46,7 +46,6 @@ _Pre_satisfies_(EventDataSize >= sizeof(CM_NOTIFY_EVENT_DATA)) static DWORD CALL
|
||||||
g_controller_interface.PlatformPopulateDevices([] {
|
g_controller_interface.PlatformPopulateDevices([] {
|
||||||
ciface::DInput::PopulateDevices(s_hwnd);
|
ciface::DInput::PopulateDevices(s_hwnd);
|
||||||
ciface::XInput::PopulateDevices();
|
ciface::XInput::PopulateDevices();
|
||||||
ciface::WGInput::PopulateDevices();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue