bool CCECAdapterMessage::is_error(void) const
{
cec_adapter_messagecode code = message();
- return (code == MSGCODE_TIMEOUT_ERROR ||
- code == MSGCODE_HIGH_ERROR ||
+ return (code == MSGCODE_HIGH_ERROR ||
code == MSGCODE_LOW_ERROR ||
code == MSGCODE_RECEIVE_FAILED ||
code == MSGCODE_COMMAND_REJECTED ||
}
else if (buf == MSGSTART) //we found a msgstart before msgend, this is not right, remove
{
- m_controller->AddLog(CEC_LOG_WARNING, "received MSGSTART before MSGEND, removing previous buffer contents");
+ if (msg.size() > 0)
+ m_controller->AddLog(CEC_LOG_WARNING, "received MSGSTART before MSGEND, removing previous buffer contents");
msg.clear();
bGotStart = true;
}
m_bStarted(false),
m_iHDMIPort(CEC_DEFAULT_HDMI_PORT),
m_iBaseDevice((cec_logical_address)CEC_DEFAULT_BASE_DEVICE),
+ m_lastInitiator(CECDEVICE_UNKNOWN),
m_strDeviceName(strDeviceName),
m_communication(serComm),
m_controller(controller),
return bReturn;
}
+bool CCECProcessor::SetLineTimeout(uint8_t iTimeout)
+{
+ bool bReturn(false);
+ CCECAdapterMessage *output = new CCECAdapterMessage;
+
+ output->push_back(MSGSTART);
+ output->push_escaped(MSGCODE_TRANSMIT_IDLETIME);
+ output->push_escaped(iTimeout);
+ output->push_back(MSGEND);
+
+ if ((bReturn = Transmit(output)) == false)
+ m_controller->AddLog(CEC_LOG_ERROR, "could not set the idletime");
+ delete output;
+ return bReturn;
+}
+
void *CCECProcessor::Process(void)
{
bool bParseFrame(false);
}
else if (m_communication->IsOpen() && m_communication->Read(msg, 50))
{
- m_controller->AddLog(msg.is_error() ? CEC_LOG_WARNING : CEC_LOG_DEBUG, msg.ToString());
if ((bParseFrame = (ParseMessage(msg) && !IsStopped())) == true)
command = m_currentframe;
}
strLog.Format("== %s monitoring mode ==", bEnable ? "enabling" : "disabling");
m_controller->AddLog(CEC_LOG_NOTICE, strLog.c_str());
- m_bMonitor = bEnable;
+ {
+ CLockObject lock(&m_mutex);
+ m_bMonitor = bEnable;
+
+ if (bEnable)
+ {
+ if (!m_busScan)
+ {
+ m_busScan = new CCECBusScan(this);
+ m_busScan->CreateThread(true);
+ }
+ }
+ else
+ {
+ if (m_busScan)
+ {
+ m_busScan->StopThread();
+ delete m_busScan;
+ m_busScan = NULL;
+ }
+ }
+ }
+
if (bEnable)
return SetAckMask(0);
else
continue;
}
+ if (msg.message() == MSGCODE_FRAME_START && msg.ack())
+ {
+ m_busDevices[msg.initiator()]->GetHandler()->HandlePoll(msg.initiator(), msg.destination());
+ m_lastInitiator = msg.initiator();
+ iNow = GetTimeMs();
+ continue;
+ }
+
+ bError = msg.is_error();
+ if (msg.message() == MSGCODE_RECEIVE_FAILED &&
+ m_lastInitiator != CECDEVICE_UNKNOWN &&
+ !m_busDevices[m_lastInitiator]->GetHandler()->HandleReceiveFailed())
+ {
+ iNow = GetTimeMs();
+ continue;
+ }
+
if ((bError = msg.is_error()) == false)
{
m_controller->AddLog(bError ? CEC_LOG_WARNING : CEC_LOG_DEBUG, msg.ToString());
bool CCECProcessor::ParseMessage(const CCECAdapterMessage &msg)
{
- bool bEom = false;
+ bool bEom(false);
+ bool bIsError(msg.is_error());
if (msg.empty())
return bEom;
m_currentframe.ack = msg.ack();
m_currentframe.eom = msg.eom();
}
+ if (m_currentframe.ack == true)
+ {
+ m_lastInitiator = m_currentframe.initiator;
+ m_busDevices[m_lastInitiator]->GetHandler()->HandlePoll(m_currentframe.initiator, m_currentframe.destination);
+ }
+ }
+ break;
+ case MSGCODE_RECEIVE_FAILED:
+ {
+ if (m_lastInitiator != CECDEVICE_UNKNOWN)
+ bIsError = m_busDevices[m_lastInitiator]->GetHandler()->HandleReceiveFailed();
}
break;
case MSGCODE_FRAME_DATA:
break;
}
+ m_controller->AddLog(bIsError ? CEC_LOG_WARNING : CEC_LOG_DEBUG, msg.ToString());
return bEom;
}
virtual bool SendKeyRelease(cec_logical_address iDestination, bool bWait = false);
virtual bool EnablePhysicalAddressDetection(void) { return false; };
+ bool SetLineTimeout(uint8_t iTimeout);
+
const char *ToString(const cec_menu_state state);
const char *ToString(const cec_version version);
const char *ToString(const cec_power_status status);
cec_logical_address m_iBaseDevice;
cec_command m_currentframe;
cec_logical_addresses m_logicalAddresses;
+ cec_logical_address m_lastInitiator;
std::string m_strDeviceName;
cec_device_type_list m_types;
CMutex m_mutex;
strLog.Format(">> %s (%X) -> %s (%X): %s (%2X)", m_processor->ToString(command.initiator), command.initiator, m_processor->ToString(command.destination), command.destination, m_processor->ToString(command.opcode), command.opcode);
m_busDevice->AddLog(CEC_LOG_NOTICE, strLog);
+ m_processor->AddCommand(command);
+
switch(command.opcode)
{
case CEC_OPCODE_REPORT_POWER_STATUS:
HandleSystemAudioModeRequest(command);
break;
case CEC_OPCODE_REPORT_AUDIO_STATUS:
- HandleReportAudioStatus(command);//YYY
+ HandleReportAudioStatus(command);
break;
case CEC_OPCODE_SYSTEM_AUDIO_MODE_STATUS:
- HandleSystemAudioModeStatus(command);//YYY
+ HandleSystemAudioModeStatus(command);
break;
case CEC_OPCODE_SET_SYSTEM_AUDIO_MODE:
- HandleSetSystemAudioMode(command);//YYY
+ HandleSetSystemAudioMode(command);
break;
case CEC_OPCODE_REQUEST_ACTIVE_SOURCE:
HandleRequestActiveSource(command);
break;
}
- m_processor->AddCommand(command);
return bHandled;
}
}
}
}
+
+void CCECCommandHandler::HandlePoll(const cec_logical_address iInitiator, const cec_logical_address iDestination)
+{
+ CStdString strLog;
+ strLog.Format("<< POLL: %s (%x) -> %s (%x)", m_processor->ToString(iInitiator), iInitiator, m_processor->ToString(iDestination), iDestination);
+ m_processor->AddLog(CEC_LOG_DEBUG, strLog);
+}
+
+bool CCECCommandHandler::HandleReceiveFailed(void)
+{
+ /* default = error */
+ return true;
+}
virtual bool HandleCommand(const cec_command &command);
virtual cec_vendor_id GetVendorId(void) { return CEC_VENDOR_UNKNOWN; };
+ virtual void HandlePoll(const cec_logical_address iInitiator, const cec_logical_address iDestination);
+ virtual bool HandleReceiveFailed(void);
protected:
virtual bool HandleActiveSource(const cec_command &command);
using namespace CEC;
CSLCommandHandler::CSLCommandHandler(CCECBusDevice *busDevice) :
- CCECCommandHandler(busDevice)
+ CCECCommandHandler(busDevice),
+ m_bAwaitingReceiveFailed(false)
{
}
return bHandled;
}
+
+
+void CSLCommandHandler::HandlePoll(const cec_logical_address iInitiator, const cec_logical_address iDestination)
+{
+ CCECCommandHandler::HandlePoll(iInitiator, iDestination);
+ m_bAwaitingReceiveFailed = true;
+}
+
+bool CSLCommandHandler::HandleReceiveFailed(void)
+{
+ if (m_bAwaitingReceiveFailed)
+ {
+ m_bAwaitingReceiveFailed = false;
+ return false;
+ }
+
+ return true;
+}
virtual cec_vendor_id GetVendorId(void) { return CEC_VENDOR_LG; };
virtual bool HandleCommand(const cec_command &command);
+ virtual void HandlePoll(const cec_logical_address iInitiator, const cec_logical_address iDestination);
+ virtual bool HandleReceiveFailed(void);
protected:
virtual bool HandleGiveDeviceVendorId(const cec_command &command);
virtual bool HandleVendorCommand(const cec_command &command);
+
+ bool m_bAwaitingReceiveFailed;
};
};
CSerialPort::CSerialPort()
{
m_fd = -1;
+ m_tostdout = false;
}
CSerialPort::~CSerialPort()
}
//print what's written to stdout for debugging
-// if (m_tostdout)
-// {
-// printf("%s write:", m_name.c_str());
-// for (int i = 0; i < byteswritten; i++)
-// printf(" %02x", (unsigned int)data[i]);
-//
-// printf("\n");
-// }
+ if (m_tostdout)
+ {
+ printf("%s write:", m_name.c_str());
+ for (int i = 0; i < byteswritten; i++)
+ printf(" %02x", data->at(i));
+
+ printf("\n");
+ }
return byteswritten;
}
}
//print what's read to stdout for debugging
-// if (m_tostdout && bytesread > 0)
-// {
-// printf("%s read:", m_name.c_str());
-// for (int i = 0; i < bytesread; i++)
-// printf(" %02x", (unsigned int)data[i]);
-//
-// printf("\n");
-// }
+ if (m_tostdout && bytesread > 0)
+ {
+ printf("%s read:", m_name.c_str());
+ for (int i = 0; i < bytesread; i++)
+ printf(" %02x", data[i]);
+
+ printf("\n");
+ }
return bytesread;
}
std::string m_error;
std::string m_name;
CMutex m_mutex;
+ bool m_tostdout;
#ifdef __WINDOWS__
bool SetTimeouts(bool bBlocking);