Merge pull request #7291 from VinDuv/gc-adapter-error-report
GC USB Adapter configuration improvements
This commit is contained in:
commit
8d8ed37e27
|
@ -391,9 +391,9 @@ void DolphinAnalytics::MakePerGameBuilder()
|
||||||
// We grab enough to tell what percentage of our users are playing with keyboard/mouse, some kind
|
// We grab enough to tell what percentage of our users are playing with keyboard/mouse, some kind
|
||||||
// of gamepad
|
// of gamepad
|
||||||
// or the official gamecube adapter.
|
// or the official gamecube adapter.
|
||||||
builder.AddData("gcadapter-detected", GCAdapter::IsDetected());
|
builder.AddData("gcadapter-detected", GCAdapter::IsDetected(nullptr));
|
||||||
builder.AddData("has-controller", Pad::GetConfig()->IsControllerControlledByGamepadDevice(0) ||
|
builder.AddData("has-controller", Pad::GetConfig()->IsControllerControlledByGamepadDevice(0) ||
|
||||||
GCAdapter::IsDetected());
|
GCAdapter::IsDetected(nullptr));
|
||||||
|
|
||||||
m_per_game_builder = builder;
|
m_per_game_builder = builder;
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
#include <QVBoxLayout>
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
|
#include "DolphinQt/QtUtils/QueueOnObject.h"
|
||||||
|
|
||||||
#include "InputCommon/GCAdapter.h"
|
#include "InputCommon/GCAdapter.h"
|
||||||
|
|
||||||
|
@ -24,28 +25,31 @@ GCPadWiiUConfigDialog::GCPadWiiUConfigDialog(int port, QWidget* parent)
|
||||||
ConnectWidgets();
|
ConnectWidgets();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GCPadWiiUConfigDialog::~GCPadWiiUConfigDialog()
|
||||||
|
{
|
||||||
|
GCAdapter::SetAdapterCallback(nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
void GCPadWiiUConfigDialog::CreateLayout()
|
void GCPadWiiUConfigDialog::CreateLayout()
|
||||||
{
|
{
|
||||||
setWindowTitle(tr("GameCube Adapter for Wii U at Port %1").arg(m_port + 1));
|
setWindowTitle(tr("GameCube Adapter for Wii U at Port %1").arg(m_port + 1));
|
||||||
|
|
||||||
const bool detected = GCAdapter::IsDetected();
|
|
||||||
m_layout = new QVBoxLayout();
|
m_layout = new QVBoxLayout();
|
||||||
m_status_label = new QLabel(detected ? tr("Adapter Detected") : tr("No Adapter Detected"));
|
m_status_label = new QLabel();
|
||||||
m_rumble = new QCheckBox(tr("Enable Rumble"));
|
m_rumble = new QCheckBox(tr("Enable Rumble"));
|
||||||
m_simulate_bongos = new QCheckBox(tr("Simulate DK Bongos"));
|
m_simulate_bongos = new QCheckBox(tr("Simulate DK Bongos"));
|
||||||
m_button_box = new QDialogButtonBox(QDialogButtonBox::Ok);
|
m_button_box = new QDialogButtonBox(QDialogButtonBox::Ok);
|
||||||
|
|
||||||
|
UpdateAdapterStatus();
|
||||||
|
|
||||||
|
auto callback = [this] { QueueOnObject(this, &GCPadWiiUConfigDialog::UpdateAdapterStatus); };
|
||||||
|
GCAdapter::SetAdapterCallback(callback);
|
||||||
|
|
||||||
m_layout->addWidget(m_status_label);
|
m_layout->addWidget(m_status_label);
|
||||||
m_layout->addWidget(m_rumble);
|
m_layout->addWidget(m_rumble);
|
||||||
m_layout->addWidget(m_simulate_bongos);
|
m_layout->addWidget(m_simulate_bongos);
|
||||||
m_layout->addWidget(m_button_box);
|
m_layout->addWidget(m_button_box);
|
||||||
|
|
||||||
if (!detected)
|
|
||||||
{
|
|
||||||
m_rumble->setEnabled(false);
|
|
||||||
m_simulate_bongos->setEnabled(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
setLayout(m_layout);
|
setLayout(m_layout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,6 +60,31 @@ void GCPadWiiUConfigDialog::ConnectWidgets()
|
||||||
connect(m_button_box, &QDialogButtonBox::accepted, this, &GCPadWiiUConfigDialog::accept);
|
connect(m_button_box, &QDialogButtonBox::accepted, this, &GCPadWiiUConfigDialog::accept);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GCPadWiiUConfigDialog::UpdateAdapterStatus()
|
||||||
|
{
|
||||||
|
const char* error_message = nullptr;
|
||||||
|
const bool detected = GCAdapter::IsDetected(&error_message);
|
||||||
|
QString status_text;
|
||||||
|
|
||||||
|
if (detected)
|
||||||
|
{
|
||||||
|
status_text = tr("Adapter Detected");
|
||||||
|
}
|
||||||
|
else if (error_message)
|
||||||
|
{
|
||||||
|
status_text = tr("Error Opening Adapter: %1").arg(QString::fromUtf8(error_message));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
status_text = tr("No Adapter Detected");
|
||||||
|
}
|
||||||
|
|
||||||
|
m_status_label->setText(status_text);
|
||||||
|
|
||||||
|
m_rumble->setEnabled(detected);
|
||||||
|
m_simulate_bongos->setEnabled(detected);
|
||||||
|
}
|
||||||
|
|
||||||
void GCPadWiiUConfigDialog::LoadSettings()
|
void GCPadWiiUConfigDialog::LoadSettings()
|
||||||
{
|
{
|
||||||
m_rumble->setChecked(SConfig::GetInstance().m_AdapterRumble[m_port]);
|
m_rumble->setChecked(SConfig::GetInstance().m_AdapterRumble[m_port]);
|
||||||
|
|
|
@ -16,6 +16,7 @@ class GCPadWiiUConfigDialog final : public QDialog
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit GCPadWiiUConfigDialog(int port, QWidget* parent = nullptr);
|
explicit GCPadWiiUConfigDialog(int port, QWidget* parent = nullptr);
|
||||||
|
~GCPadWiiUConfigDialog();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void LoadSettings();
|
void LoadSettings();
|
||||||
|
@ -24,6 +25,9 @@ private:
|
||||||
void CreateLayout();
|
void CreateLayout();
|
||||||
void ConnectWidgets();
|
void ConnectWidgets();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void UpdateAdapterStatus();
|
||||||
|
|
||||||
int m_port;
|
int m_port;
|
||||||
|
|
||||||
QVBoxLayout* m_layout;
|
QVBoxLayout* m_layout;
|
||||||
|
|
|
@ -9,6 +9,7 @@
|
||||||
#include "Common/Event.h"
|
#include "Common/Event.h"
|
||||||
#include "Common/Flag.h"
|
#include "Common/Flag.h"
|
||||||
#include "Common/Logging/Log.h"
|
#include "Common/Logging/Log.h"
|
||||||
|
#include "Common/ScopeGuard.h"
|
||||||
#include "Common/Thread.h"
|
#include "Common/Thread.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/Core.h"
|
#include "Core/Core.h"
|
||||||
|
@ -29,7 +30,14 @@ static void ResetRumbleLockNeeded();
|
||||||
static void Reset();
|
static void Reset();
|
||||||
static void Setup();
|
static void Setup();
|
||||||
|
|
||||||
static bool s_detected = false;
|
enum
|
||||||
|
{
|
||||||
|
NO_ADAPTER_DETECTED = 0,
|
||||||
|
ADAPTER_DETECTED = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Current adapter status: detected/not detected/in error (holds the error code)
|
||||||
|
static int s_status = NO_ADAPTER_DETECTED;
|
||||||
static libusb_device_handle* s_handle = nullptr;
|
static libusb_device_handle* s_handle = nullptr;
|
||||||
static u8 s_controller_type[SerialInterface::MAX_SI_CHANNELS] = {
|
static u8 s_controller_type[SerialInterface::MAX_SI_CHANNELS] = {
|
||||||
ControllerTypes::CONTROLLER_NONE, ControllerTypes::CONTROLLER_NONE,
|
ControllerTypes::CONTROLLER_NONE, ControllerTypes::CONTROLLER_NONE,
|
||||||
|
@ -54,7 +62,6 @@ static Common::Flag s_adapter_detect_thread_running;
|
||||||
|
|
||||||
static std::function<void(void)> s_detect_callback;
|
static std::function<void(void)> s_detect_callback;
|
||||||
|
|
||||||
static bool s_libusb_driver_not_supported = false;
|
|
||||||
#if defined(__FreeBSD__) && __FreeBSD__ >= 11
|
#if defined(__FreeBSD__) && __FreeBSD__ >= 11
|
||||||
static bool s_libusb_hotplug_enabled = true;
|
static bool s_libusb_hotplug_enabled = true;
|
||||||
#else
|
#else
|
||||||
|
@ -115,11 +122,23 @@ static int HotplugCallback(libusb_context* ctx, libusb_device* dev, libusb_hotpl
|
||||||
std::lock_guard<std::mutex> lk(s_init_mutex);
|
std::lock_guard<std::mutex> lk(s_init_mutex);
|
||||||
AddGCAdapter(dev);
|
AddGCAdapter(dev);
|
||||||
}
|
}
|
||||||
|
else if (s_status < 0 && s_detect_callback != nullptr)
|
||||||
|
{
|
||||||
|
s_detect_callback();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (event == LIBUSB_HOTPLUG_EVENT_DEVICE_LEFT)
|
else if (event == LIBUSB_HOTPLUG_EVENT_DEVICE_LEFT)
|
||||||
{
|
{
|
||||||
if (s_handle != nullptr && libusb_get_device(s_handle) == dev)
|
if (s_handle != nullptr && libusb_get_device(s_handle) == dev)
|
||||||
Reset();
|
Reset();
|
||||||
|
|
||||||
|
// Reset a potential error status now that the adapter is unplugged
|
||||||
|
if (s_status < 0)
|
||||||
|
{
|
||||||
|
s_status = NO_ADAPTER_DETECTED;
|
||||||
|
if (s_detect_callback != nullptr)
|
||||||
|
s_detect_callback();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -158,8 +177,6 @@ static void ScanThreadFunc()
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lk(s_init_mutex);
|
std::lock_guard<std::mutex> lk(s_init_mutex);
|
||||||
Setup();
|
Setup();
|
||||||
if (s_detected && s_detect_callback != nullptr)
|
|
||||||
s_detect_callback();
|
|
||||||
}
|
}
|
||||||
Common::SleepCurrentThread(500);
|
Common::SleepCurrentThread(500);
|
||||||
}
|
}
|
||||||
|
@ -184,7 +201,7 @@ void Init()
|
||||||
s_last_init = CoreTiming::GetTicks();
|
s_last_init = CoreTiming::GetTicks();
|
||||||
}
|
}
|
||||||
|
|
||||||
s_libusb_driver_not_supported = false;
|
s_status = NO_ADAPTER_DETECTED;
|
||||||
|
|
||||||
if (UseAdapter())
|
if (UseAdapter())
|
||||||
StartScanThread();
|
StartScanThread();
|
||||||
|
@ -210,6 +227,12 @@ void StopScanThread()
|
||||||
|
|
||||||
static void Setup()
|
static void Setup()
|
||||||
{
|
{
|
||||||
|
int prev_status = s_status;
|
||||||
|
|
||||||
|
// Reset the error status in case the adapter gets unplugged
|
||||||
|
if (s_status < 0)
|
||||||
|
s_status = NO_ADAPTER_DETECTED;
|
||||||
|
|
||||||
for (int i = 0; i < SerialInterface::MAX_SI_CHANNELS; i++)
|
for (int i = 0; i < SerialInterface::MAX_SI_CHANNELS; i++)
|
||||||
{
|
{
|
||||||
s_controller_type[i] = ControllerTypes::CONTROLLER_NONE;
|
s_controller_type[i] = ControllerTypes::CONTROLLER_NONE;
|
||||||
|
@ -225,6 +248,9 @@ static void Setup()
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (s_status != ADAPTER_DETECTED && prev_status != s_status && s_detect_callback != nullptr)
|
||||||
|
s_detect_callback();
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool CheckDeviceAccess(libusb_device* device)
|
static bool CheckDeviceAccess(libusb_device* device)
|
||||||
|
@ -247,6 +273,9 @@ static bool CheckDeviceAccess(libusb_device* device)
|
||||||
NOTICE_LOG(SERIALINTERFACE, "Found GC Adapter with Vendor: %X Product: %X Devnum: %d",
|
NOTICE_LOG(SERIALINTERFACE, "Found GC Adapter with Vendor: %X Product: %X Devnum: %d",
|
||||||
desc.idVendor, desc.idProduct, 1);
|
desc.idVendor, desc.idProduct, 1);
|
||||||
|
|
||||||
|
// In case of failure, capture the libusb error code into the adapter status
|
||||||
|
Common::ScopeGuard status_guard([&ret] { s_status = ret; });
|
||||||
|
|
||||||
u8 bus = libusb_get_bus_number(device);
|
u8 bus = libusb_get_bus_number(device);
|
||||||
u8 port = libusb_get_device_address(device);
|
u8 port = libusb_get_device_address(device);
|
||||||
ret = libusb_open(device, &s_handle);
|
ret = libusb_open(device, &s_handle);
|
||||||
|
@ -260,8 +289,6 @@ static bool CheckDeviceAccess(libusb_device* device)
|
||||||
if (ret)
|
if (ret)
|
||||||
{
|
{
|
||||||
ERROR_LOG(SERIALINTERFACE, "libusb_open failed to open device with error = %d", ret);
|
ERROR_LOG(SERIALINTERFACE, "libusb_open failed to open device with error = %d", ret);
|
||||||
if (ret == LIBUSB_ERROR_NOT_SUPPORTED)
|
|
||||||
s_libusb_driver_not_supported = true;
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -276,15 +303,24 @@ static bool CheckDeviceAccess(libusb_device* device)
|
||||||
// this split is needed so that we don't avoid claiming the interface when
|
// this split is needed so that we don't avoid claiming the interface when
|
||||||
// detaching the kernel driver is successful
|
// detaching the kernel driver is successful
|
||||||
if (ret != 0 && ret != LIBUSB_ERROR_NOT_SUPPORTED)
|
if (ret != 0 && ret != LIBUSB_ERROR_NOT_SUPPORTED)
|
||||||
|
{
|
||||||
|
libusb_close(s_handle);
|
||||||
|
s_handle = nullptr;
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
ret = libusb_claim_interface(s_handle, 0);
|
ret = libusb_claim_interface(s_handle, 0);
|
||||||
if (ret)
|
if (ret)
|
||||||
{
|
{
|
||||||
ERROR_LOG(SERIALINTERFACE, "libusb_claim_interface failed with error: %d", ret);
|
ERROR_LOG(SERIALINTERFACE, "libusb_claim_interface failed with error: %d", ret);
|
||||||
|
libusb_close(s_handle);
|
||||||
|
s_handle = nullptr;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Updating the adapter status will be done in AddGCAdapter
|
||||||
|
status_guard.Dismiss();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -317,7 +353,7 @@ static void AddGCAdapter(libusb_device* device)
|
||||||
s_adapter_input_thread = std::thread(Read);
|
s_adapter_input_thread = std::thread(Read);
|
||||||
s_adapter_output_thread = std::thread(Write);
|
s_adapter_output_thread = std::thread(Write);
|
||||||
|
|
||||||
s_detected = true;
|
s_status = ADAPTER_DETECTED;
|
||||||
if (s_detect_callback != nullptr)
|
if (s_detect_callback != nullptr)
|
||||||
s_detect_callback();
|
s_detect_callback();
|
||||||
ResetRumbleLockNeeded();
|
ResetRumbleLockNeeded();
|
||||||
|
@ -332,7 +368,7 @@ void Shutdown()
|
||||||
#endif
|
#endif
|
||||||
Reset();
|
Reset();
|
||||||
|
|
||||||
s_libusb_driver_not_supported = false;
|
s_status = NO_ADAPTER_DETECTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void Reset()
|
static void Reset()
|
||||||
|
@ -340,7 +376,7 @@ static void Reset()
|
||||||
std::unique_lock<std::mutex> lock(s_init_mutex, std::defer_lock);
|
std::unique_lock<std::mutex> lock(s_init_mutex, std::defer_lock);
|
||||||
if (!lock.try_lock())
|
if (!lock.try_lock())
|
||||||
return;
|
return;
|
||||||
if (!s_detected)
|
if (s_status != ADAPTER_DETECTED)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (s_adapter_thread_running.TestAndClear())
|
if (s_adapter_thread_running.TestAndClear())
|
||||||
|
@ -353,7 +389,7 @@ static void Reset()
|
||||||
for (int i = 0; i < SerialInterface::MAX_SI_CHANNELS; i++)
|
for (int i = 0; i < SerialInterface::MAX_SI_CHANNELS; i++)
|
||||||
s_controller_type[i] = ControllerTypes::CONTROLLER_NONE;
|
s_controller_type[i] = ControllerTypes::CONTROLLER_NONE;
|
||||||
|
|
||||||
s_detected = false;
|
s_status = NO_ADAPTER_DETECTED;
|
||||||
|
|
||||||
if (s_handle)
|
if (s_handle)
|
||||||
{
|
{
|
||||||
|
@ -371,7 +407,7 @@ GCPadStatus Input(int chan)
|
||||||
if (!UseAdapter())
|
if (!UseAdapter())
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
if (s_handle == nullptr || !s_detected)
|
if (s_handle == nullptr || s_status != ADAPTER_DETECTED)
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
int payload_size = 0;
|
int payload_size = 0;
|
||||||
|
@ -491,7 +527,7 @@ void ResetRumble()
|
||||||
// being called while the libusb state is being reset
|
// being called while the libusb state is being reset
|
||||||
static void ResetRumbleLockNeeded()
|
static void ResetRumbleLockNeeded()
|
||||||
{
|
{
|
||||||
if (!UseAdapter() || (s_handle == nullptr || !s_detected))
|
if (!UseAdapter() || (s_handle == nullptr || s_status != ADAPTER_DETECTED))
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -521,14 +557,20 @@ void Output(int chan, u8 rumble_command)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsDetected()
|
bool IsDetected(const char** error_message)
|
||||||
{
|
{
|
||||||
return s_detected;
|
if (s_status >= 0)
|
||||||
}
|
{
|
||||||
|
if (error_message)
|
||||||
|
*error_message = nullptr;
|
||||||
|
|
||||||
bool IsDriverDetected()
|
return s_status == ADAPTER_DETECTED;
|
||||||
{
|
}
|
||||||
return !s_libusb_driver_not_supported;
|
|
||||||
|
if (error_message)
|
||||||
|
*error_message = libusb_strerror(static_cast<libusb_error>(s_status));
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // end of namespace GCAdapter
|
} // end of namespace GCAdapter
|
||||||
|
|
|
@ -26,8 +26,7 @@ void StartScanThread();
|
||||||
void StopScanThread();
|
void StopScanThread();
|
||||||
GCPadStatus Input(int chan);
|
GCPadStatus Input(int chan);
|
||||||
void Output(int chan, u8 rumble_command);
|
void Output(int chan, u8 rumble_command);
|
||||||
bool IsDetected();
|
bool IsDetected(const char** error_message);
|
||||||
bool IsDriverDetected();
|
|
||||||
bool DeviceConnected(int chan);
|
bool DeviceConnected(int chan);
|
||||||
void ResetDeviceType(int chan);
|
void ResetDeviceType(int chan);
|
||||||
bool UseAdapter();
|
bool UseAdapter();
|
||||||
|
|
|
@ -369,14 +369,10 @@ void Output(int chan, u8 rumble_command)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsDetected()
|
bool IsDetected(const char** error_message)
|
||||||
{
|
{
|
||||||
return s_detected;
|
return s_detected;
|
||||||
}
|
}
|
||||||
bool IsDriverDetected()
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
bool DeviceConnected(int chan)
|
bool DeviceConnected(int chan)
|
||||||
{
|
{
|
||||||
return s_controller_type[chan] != ControllerTypes::CONTROLLER_NONE;
|
return s_controller_type[chan] != ControllerTypes::CONTROLLER_NONE;
|
||||||
|
|
Loading…
Reference in New Issue