Преглед изворни кода

Improve InputCEC to not block the main thread.

Turns out that CEC::FindAdapters() could block the main thread and
introduce frame drops.

libcec have a new API to combat this called DetectAdapters() which
contains a quick scan. This commit changes to that API but also moves
the CEC operations to a separate thread. This is just to make sure
nothing else blocks since it's not very well documented what calls that
block and not.

Split the class into InputCEC and InputCECWorker using the same thread
model we are using for InputSDL.

Related to #129
Tobias Hieta пре 9 година
родитељ
комит
c9beb1474c
2 измењених фајлова са 98 додато и 75 уклоњено
  1. 59 54
      src/input/InputCEC.cpp
  2. 39 21
      src/input/InputCEC.h

+ 59 - 54
src/input/InputCEC.cpp

@@ -49,19 +49,27 @@ static QMap<int, KeyAction> cecKeyMap   { \
 //////////////////////////////////////////////////////////////////////////////////////////////////
 InputCEC::InputCEC(QObject *parent) : InputBase(parent)
 {
-  m_configuration.Clear();
-  m_callbacks.Clear();
-  m_adapter = NULL;
-  m_verboseLogging = false;
+  m_cecThread = new QThread(this);
+  m_cecThread->setObjectName("InputCEC");
 
-  // setup adapter check timer
-  m_timer.setInterval(1000);
-  m_timer.setSingleShot(false);
-  connect(&m_timer, &QTimer::timeout, this, &InputCEC::checkAdapter);
+  m_cecWorker = new InputCECWorker(NULL);
+  m_cecWorker->moveToThread(m_cecThread);
+
+  m_cecThread->start(QThread::LowPriority);
+  connect(m_cecWorker, &InputCECWorker::receivedInput, this, &InputCEC::receivedInput);
 }
 
 //////////////////////////////////////////////////////////////////////////////////////////////////
 bool InputCEC::initInput()
+{
+  bool retVal;
+  QMetaObject::invokeMethod(m_cecWorker, "init", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, retVal));
+
+  return retVal;
+}
+
+/////////////////////////////////////////////////////////////////////////////////////////
+bool InputCECWorker::init()
 {
   m_configuration.Clear();
   m_callbacks.Clear();
@@ -81,8 +89,7 @@ bool InputCEC::initInput()
   m_configuration.bAutodetectAddress =  CEC_DEFAULT_SETTING_AUTODETECT_ADDRESS;
   m_configuration.iPhysicalAddress = CEC_PHYSICAL_ADDRESS_TV;
   m_configuration.baseDevice = CECDEVICE_AUDIOSYSTEM;
-  m_configuration.bActivateSource = SettingsComponent::Get().value(SETTINGS_SECTION_CEC, "activatesource").toBool();
-
+  m_configuration.bActivateSource = (uint8_t)SettingsComponent::Get().value(SETTINGS_SECTION_CEC, "activatesource").toBool();
   m_configuration.iHDMIPort = (quint8)SettingsComponent::Get().value(SETTINGS_SECTION_CEC, "hdmiport").toInt();
 
   // open libcec
@@ -99,20 +106,27 @@ bool InputCEC::initInput()
   // init video on targets that need this
   m_adapter->InitVideoStandalone();
 
-  // start the adapter check timer
-  m_timer.start();
+  // check for attached adapters
+  checkAdapter();
+
+  // Start a timer to keep track of attached/removed adapters
+  m_timer = new QTimer(this);
+  m_timer->setInterval(10 * 1000);
+  connect(m_timer, &QTimer::timeout, this, &InputCECWorker::checkAdapter);
+  m_timer->start();
+
   return true;
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-InputCEC::~InputCEC()
+InputCECWorker::~InputCECWorker()
 {
-  m_timer.stop();
-  close();
+  m_timer->stop();
+  closeCec();
 }
 
 //////////////////////////////////////////////////////////////////////////////////////////////////
-void InputCEC::close()
+void InputCECWorker::closeCec()
 {
   if (m_adapter)
   {
@@ -123,54 +137,43 @@ void InputCEC::close()
 }
 
 //////////////////////////////////////////////////////////////////////////////////////////////////
-bool InputCEC::openAdapter()
+bool InputCECWorker::openAdapter()
 {
   bool ret = false;
-  m_lock.lock();
 
   // try to find devices
-  cec_adapter devices[10];
-  int devicesCount = m_adapter->FindAdapters(devices, 10, NULL);
+  cec_adapter_descriptor devices[10];
+  int devicesCount = m_adapter->DetectAdapters(devices, 10, NULL, false);
   if (devicesCount > 0)
   {
     // list devices
     QLOG_INFO() << "libCEC found" << devicesCount << "CEC adapters.";
 
     // open first adapter
-    m_adapterPort = devices[0].comm;
+    m_adapterPort = devices[0].strComPath;
     if (m_adapter->Open(m_adapterPort.toStdString().c_str()))
     {
-      QLOG_INFO() << "Device " << devices[0].path << "was successfully openned";
+      QLOG_INFO() << "Device " << devices[0].strComName << "was successfully openned";
       ret = true;
     }
     else
     {
-      QLOG_ERROR() << "Opening device" << devices[0].path << "failed";
+      QLOG_ERROR() << "Opening device" << devices[0].strComName << "failed";
       ret = false;
     }
   }
 
-  m_lock.unlock();
   return ret;
 }
 
 //////////////////////////////////////////////////////////////////////////////////////////////////
-void InputCEC::closeAdapter()
+void InputCECWorker::closeAdapter()
 {
-  m_lock.lock();
-
-  m_adapterPort = "";
-  m_lock.unlock();
-}
-
-//////////////////////////////////////////////////////////////////////////////////////////////////
-void InputCEC::reopenAdapter()
-{
-  closeAdapter();
+  m_adapterPort.clear();
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-void InputCEC::checkAdapter()
+void InputCECWorker::checkAdapter()
 {
   if (m_adapterPort.isEmpty())
   {    
@@ -182,13 +185,13 @@ void InputCEC::checkAdapter()
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-void InputCEC::sendReceivedInput(const QString &source, const QString &keycode, float amount)
+void InputCECWorker::sendReceivedInput(const QString &source, const QString &keycode, float amount)
 {
   emit receivedInput(source, keycode, amount);
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-QString InputCEC::getCommandString(cec_user_control_code code, unsigned int duration)
+QString InputCECWorker::getCommandString(cec_user_control_code code, unsigned int duration)
 {
   QString key;
 
@@ -207,9 +210,12 @@ QString InputCEC::getCommandString(cec_user_control_code code, unsigned int dura
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-int InputCEC::CecLogMessage(void* cbParam, const cec_log_message message)
+int InputCECWorker::CecLogMessage(void* cbParam, const cec_log_message message)
 {
-  InputCEC *cec = (InputCEC*)cbParam;
+  auto *cec = static_cast<InputCECWorker*>(cbParam);
+
+  Q_ASSERT(cec);
+
   switch (message.level)
   {
     case CEC_LOG_ERROR:
@@ -242,11 +248,14 @@ int InputCEC::CecLogMessage(void* cbParam, const cec_log_message message)
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-int InputCEC::CecKeyPress(void *cbParam, const cec_keypress key)
+int InputCECWorker::CecKeyPress(void *cbParam, const cec_keypress key)
 {
   static int lastKeyPressed = -1;
 
-  InputCEC *cec = (InputCEC*)cbParam;
+  auto *cec = static_cast<InputCECWorker*>(cbParam);
+
+  Q_ASSERT(cec);
+
   if (cec->m_verboseLogging)
   {
     QLOG_DEBUG() << QString("CecKeyPress : Got key %1 (%2), last was %3").arg(key.keycode).arg(key.duration).arg(lastKeyPressed);
@@ -272,7 +281,7 @@ int InputCEC::CecKeyPress(void *cbParam, const cec_keypress key)
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-QString InputCEC::getCommandParamsList(cec_command command)
+QString InputCECWorker::getCommandParamsList(cec_command command)
 {
   QString output = QString("%1 parameter(s) :").arg(command.parameters.size);
 
@@ -286,9 +295,10 @@ QString InputCEC::getCommandParamsList(cec_command command)
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-int InputCEC::CecCommand(void *cbParam, const cec_command command)
+int InputCECWorker::CecCommand(void *cbParam, const cec_command command)
 {
-  InputCEC *cec = (InputCEC*)cbParam;
+  auto cec = static_cast<InputCECWorker*>(cbParam);
+  Q_ASSERT(cec);
 
   if (cec->m_verboseLogging)
   {
@@ -369,7 +379,7 @@ int InputCEC::CecCommand(void *cbParam, const cec_command command)
 }
 
 ///////////////////////////////////////////////////////////////////////////////////////////////////
-int InputCEC::CecAlert(void *cbParam, const libcec_alert type, const libcec_parameter param)
+int InputCECWorker::CecAlert(void *cbParam, const libcec_alert type, const libcec_parameter param)
 {
   bool reopen = false;
 
@@ -401,15 +411,10 @@ int InputCEC::CecAlert(void *cbParam, const libcec_alert type, const libcec_para
   if (reopen)
   {
     QLOG_DEBUG() << "libCEC : Reopenning adapter";
-    InputCEC *cec = (InputCEC*)cbParam;
+    auto cec = static_cast<InputCECWorker*>(cbParam);
     if (cec)
-      cec->reopenAdapter();
+      cec->closeAdapter();
   }
 
   return 0;
-}
-
-
-
-
-
+}

+ 39 - 21
src/input/InputCEC.h

@@ -12,45 +12,63 @@ using namespace CEC;
 
 #define CEC_INPUT_NAME "CEC"
 
+class InputCECWorker;
+
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 class InputCEC : public InputBase
 {
 public:
   InputCEC(QObject* parent);
-  ~InputCEC();
 
-  virtual const char* inputName() { return CEC_INPUT_NAME; }
+  virtual const char* inputName()
+  { return CEC_INPUT_NAME; }
+
   virtual bool initInput();
-  void close();
 
 private:
-  libcec_configuration m_configuration;
-  ICECCallbacks m_callbacks;
-  ICECAdapter*  m_adapter;
-  QString m_adapterPort;
-  QMutex m_lock;
-  QTimer m_timer;
-  bool m_verboseLogging;
+  QThread* m_cecThread;
+  InputCECWorker* m_cecWorker;
+};
+
+/////////////////////////////////////////////////////////////////////////////////////////
+class InputCECWorker : public QObject
+{
+Q_OBJECT
+public:
+  InputCECWorker(QObject* parent = 0) : QObject(parent), m_adapter(0), m_adapterPort("")
+  {
+  }
+
+  ~InputCECWorker();
+
+  Q_SLOT bool init();
+  Q_SIGNAL void receivedInput(const QString& source, const QString& keycode, float amount);
+
+public slots:
+  void checkAdapter();
+
+private:
+  void closeCec();
 
   bool openAdapter();
   void closeAdapter();
+
   QString getCommandString(cec_user_control_code code, unsigned int duration);
   void sendReceivedInput(const QString& source, const QString& keycode, float amount = 1.0);
   QString getCommandParamsList(cec_command command);
 
-
   // libcec callbacks
-  static int CecLogMessage(void *cbParam, const cec_log_message message);
-  static int CecKeyPress(void *cbParam, const cec_keypress key);
-  static int CecCommand(void *cbParam, const cec_command command);
-  static int CecAlert(void *cbParam, const libcec_alert type, const libcec_parameter param);
-
-public:
-  void reopenAdapter();
-
-public slots:
-  void checkAdapter();
+  static int CecLogMessage(void* cbParam, const cec_log_message message);
+  static int CecKeyPress(void* cbParam, const cec_keypress key);
+  static int CecCommand(void* cbParam, const cec_command command);
+  static int CecAlert(void* cbParam, const libcec_alert type, const libcec_parameter param);
 
+  libcec_configuration m_configuration;
+  ICECCallbacks m_callbacks;
+  ICECAdapter* m_adapter;
+  QString m_adapterPort;
+  QTimer* m_timer;
+  bool m_verboseLogging;
 };
 
 #endif // INPUTCEC_H