#include "../devices/CECBusDevice.h"
#include "../devices/CECPlaybackDevice.h"
#include "../CECProcessor.h"
+#include "../LibCEC.h"
using namespace CEC;
#define SL_COMMAND_REQUEST_POWER_STATUS 0xa0
#define SL_COMMAND_POWER_ON 0x03
#define SL_COMMAND_CONNECT_REQUEST 0x04
-#define SL_COMMAND_CONNECT_ACCEPT 0x05
+#define SL_COMMAND_SET_DEVICE_MODE 0x05
CSLCommandHandler::CSLCommandHandler(CCECBusDevice *busDevice) :
CCECCommandHandler(busDevice),
m_bSLEnabled(false),
- m_bPowerStateReset(false)
+ m_bPowerStateReset(false),
+ m_bActiveSourceSent(false)
{
m_vendorId = CEC_VENDOR_LG;
CCECBusDevice *primary = m_processor->GetPrimaryDevice();
/* imitate LG devices */
if (primary && m_busDevice->GetLogicalAddress() != primary->GetLogicalAddress())
primary->SetVendorId(CEC_VENDOR_LG);
- SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG);
/* LG TVs don't always reply to CEC version requests, so just set it to 1.3a */
if (m_busDevice->GetLogicalAddress() == CECDEVICE_TV)
bool CSLCommandHandler::ActivateSource(void)
{
+ if (m_bActiveSourceSent)
+ return false;
+ m_bActiveSourceSent = true;
+
CCECBusDevice *primary = m_processor->GetPrimaryDevice();
primary->SetActiveSource();
- primary->TransmitImageViewOn();
primary->TransmitActiveSource();
return true;
}
{
SetVendorId(command);
- cec_command response;
- cec_command::Format(response, m_processor->GetLogicalAddress(), command.initiator, CEC_OPCODE_FEATURE_ABORT);
- return Transmit(response);
-}
-
-bool CSLCommandHandler::HandleFeatureAbort(const cec_command &command)
-{
- CCECBusDevice *primary = m_processor->GetPrimaryDevice();
- if (primary->GetPowerStatus(false) == CEC_POWER_STATUS_ON && !m_bPowerStateReset && !m_bSLEnabled)
+ if (!m_bSLEnabled)
{
- m_bPowerStateReset = true;
- primary->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON);
+ cec_command response;
+ cec_command::Format(response, m_processor->GetLogicalAddress(), command.initiator, CEC_OPCODE_FEATURE_ABORT);
+ return Transmit(response);
}
-
- return CCECCommandHandler::HandleFeatureAbort(command);
+ return true;
}
bool CSLCommandHandler::HandleGivePhysicalAddress(const cec_command &command)
{
CCECBusDevice *device = GetDevice(command.destination);
if (device)
- return device->TransmitPhysicalAddress();
+ return device->TransmitPhysicalAddress(); // only the physical address, don't send image view on
}
return false;
void CSLCommandHandler::HandleVendorCommand01(const cec_command &command)
{
+ m_processor->GetPrimaryDevice()->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON);
TransmitVendorCommand0205(command.destination, command.initiator);
}
device->TransmitPowerState(command.initiator);
device->SetPowerStatus(CEC_POWER_STATUS_ON);
- SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG);
device->SetActiveSource();
TransmitImageViewOn(device->GetLogicalAddress(), command.initiator);
}
void CSLCommandHandler::HandleVendorCommandSLConnect(const cec_command &command)
{
m_bSLEnabled = true;
- CCECBusDevice *primary = m_processor->GetPrimaryDevice();
-
- TransmitVendorCommand05(primary->GetLogicalAddress(), command.initiator);
-
- CCECPlaybackDevice *playback = (primary->GetType() == CEC_DEVICE_TYPE_PLAYBACK_DEVICE || primary->GetType() == CEC_DEVICE_TYPE_RECORDING_DEVICE) ?
- (CCECPlaybackDevice *)primary : NULL;
- if (playback)
- {
- SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS);
- playback->TransmitDeckStatus(CECDEVICE_TV);
- PLATFORM::CEvent::Sleep(2000);
- }
-
- primary->SetActiveSource();
- primary->TransmitImageViewOn();
+ TransmitVendorCommandSetDeviceMode(m_processor->GetLogicalAddress(), command.initiator, CEC_DEVICE_TYPE_RECORDING_DEVICE);
- SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG);
- if (playback)
- playback->TransmitDeckStatus(CECDEVICE_TV);
+ ActivateSource();
}
-void CSLCommandHandler::TransmitVendorCommand05(const cec_logical_address iSource, const cec_logical_address iDestination)
+void CSLCommandHandler::TransmitVendorCommandSetDeviceMode(const cec_logical_address iSource, const cec_logical_address iDestination, const cec_device_type type)
{
cec_command response;
cec_command::Format(response, iSource, iDestination, CEC_OPCODE_VENDOR_COMMAND);
- response.PushBack(SL_COMMAND_CONNECT_ACCEPT);
- response.PushBack((uint8_t)m_processor->m_busDevices[iSource]->GetType());
+ response.PushBack(SL_COMMAND_SET_DEVICE_MODE);
+ response.PushBack((uint8_t)type);
Transmit(response, false);
}
-void CSLCommandHandler::SetDeckStatus(cec_deck_info deckStatus)
+bool CSLCommandHandler::HandleGiveDeckStatus(const cec_command &command)
{
- CCECBusDevice *device = m_processor->GetDeviceByType(CEC_DEVICE_TYPE_PLAYBACK_DEVICE);
- if (device)
- ((CCECPlaybackDevice *)device)->SetDeckStatus(deckStatus);
+ if (m_processor->IsRunning() && m_busDevice->MyLogicalAddressContains(command.destination))
+ {
+ CCECBusDevice *device = GetDevice(command.destination);
+ if (device && (device->GetType() == CEC_DEVICE_TYPE_PLAYBACK_DEVICE || device->GetType() == CEC_DEVICE_TYPE_RECORDING_DEVICE))
+ {
+ if (command.parameters.size > 0)
+ {
+ if (command.parameters[0] == CEC_STATUS_REQUEST_ON)
+ {
+ ((CCECPlaybackDevice *) device)->SetDeckStatus(CEC_DECK_INFO_STOP);
+ return ((CCECPlaybackDevice *) device)->TransmitDeckStatus(command.initiator) &&
+ device->TransmitImageViewOn() &&
+ device->TransmitPhysicalAddress();
+ }
+ else if (command.parameters[0] == CEC_STATUS_REQUEST_ONCE)
+ {
+ ((CCECPlaybackDevice *) device)->SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG);
+ return ((CCECPlaybackDevice *) device)->TransmitDeckStatus(command.initiator);
+ }
+ }
+ }
+ return CCECCommandHandler::HandleGiveDeckStatus(command);
+ }
- device = m_processor->GetDeviceByType(CEC_DEVICE_TYPE_RECORDING_DEVICE);
- if (device)
- ((CCECPlaybackDevice *)device)->SetDeckStatus(deckStatus);
+ return false;
}
-
bool CSLCommandHandler::HandleGiveDevicePowerStatus(const cec_command &command)
{
+ bool bReturn(false);
if (m_processor->IsRunning() && m_busDevice->MyLogicalAddressContains(command.destination))
{
CCECBusDevice *device = GetDevice(command.destination);
if (device && device->GetPowerStatus(false) != CEC_POWER_STATUS_ON)
- return device->TransmitPowerState(command.initiator);
+ {
+ bReturn = device->TransmitPowerState(command.initiator);
+ device->SetPowerStatus(CEC_POWER_STATUS_ON);
+ }
+ else if (!ActivateSource())
+ {
+ /* assume that we've bugged out */
+ CLibCEC::AddLog(CEC_LOG_NOTICE, "LG seems to have bugged out. resetting to 'in transition standby to on'");
+ m_bActiveSourceSent = false;
+ device->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON);
+ bReturn = device->TransmitPowerState(command.initiator);
+ device->SetPowerStatus(CEC_POWER_STATUS_ON);
+ }
+ else
+ {
+ bReturn = true;
+ }
}
+ return bReturn;
+}
+
+bool CSLCommandHandler::HandleRequestActiveSource(const cec_command &command)
+{
+ if (m_processor->IsRunning())
+ {
+ CLibCEC::AddLog(CEC_LOG_DEBUG, ">> %i requests active source, ignored", (uint8_t) command.initiator);
+ return true;
+ }
return false;
}