Merge pull request #7291 from VinDuv/gc-adapter-error-report

GC USB Adapter configuration improvements
This commit is contained in:
Léo Lam 2019-05-30 13:54:01 +02:00 committed by GitHub
commit 8d8ed37e27
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 106 additions and 36 deletions

View File

@ -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;
} }

View File

@ -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]);

View File

@ -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;

View File

@ -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

View File

@ -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();

View File

@ -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;