X-Git-Url: https://git.piment-noir.org/?a=blobdiff_plain;f=src%2Flib%2Fimplementations%2FSLCommandHandler.cpp;h=f1e3d7e229539980e1b34e9c71e74f32661c25ec;hb=96b887ec283ed6ede1db692baf792e214d8946c8;hp=2475e14ae28c786b8648f8288cd3b778f44eeca1;hpb=199034ab9274a19316150586306e39fa26bbb73c;p=deb_libcec.git diff --git a/src/lib/implementations/SLCommandHandler.cpp b/src/lib/implementations/SLCommandHandler.cpp index 2475e14..f1e3d7e 100644 --- a/src/lib/implementations/SLCommandHandler.cpp +++ b/src/lib/implementations/SLCommandHandler.cpp @@ -34,6 +34,7 @@ #include "../devices/CECBusDevice.h" #include "../devices/CECPlaybackDevice.h" #include "../CECProcessor.h" +#include "../LibCEC.h" using namespace CEC; @@ -44,12 +45,13 @@ 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(); @@ -57,7 +59,6 @@ CSLCommandHandler::CSLCommandHandler(CCECBusDevice *busDevice) : /* imitate LG devices */ if (primary && m_busDevice->GetLogicalAddress() != primary->GetLogicalAddress()) primary->SetVendorId(CEC_VENDOR_LG); - SetLGDeckStatus(); /* LG TVs don't always reply to CEC version requests, so just set it to 1.3a */ if (m_busDevice->GetLogicalAddress() == CECDEVICE_TV) @@ -79,7 +80,7 @@ bool CSLCommandHandler::InitHandler(void) /* reply with LGs vendor id */ CCECBusDevice *primary = m_processor->GetPrimaryDevice(); if (m_busDevice->GetLogicalAddress() != primary->GetLogicalAddress()) - primary->TransmitVendorID(CECDEVICE_TV, false); + primary->TransmitVendorID(CECDEVICE_BROADCAST, false); primary->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON); return true; @@ -87,6 +88,13 @@ bool CSLCommandHandler::InitHandler(void) bool CSLCommandHandler::ActivateSource(void) { + if (!m_bSLEnabled) + return true; + + if (m_bActiveSourceSent) + return true; + m_bActiveSourceSent = true; + CCECBusDevice *primary = m_processor->GetPrimaryDevice(); primary->SetActiveSource(); primary->TransmitActiveSource(); @@ -110,21 +118,13 @@ bool CSLCommandHandler::HandleDeviceVendorId(const cec_command &command) { 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) @@ -133,7 +133,7 @@ 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; @@ -171,6 +171,7 @@ bool CSLCommandHandler::HandleVendorCommand(const cec_command &command) void CSLCommandHandler::HandleVendorCommand01(const cec_command &command) { + m_processor->GetPrimaryDevice()->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON); TransmitVendorCommand0205(command.destination, command.initiator); } @@ -195,7 +196,6 @@ void CSLCommandHandler::HandleVendorCommandPowerOn(const cec_command &command) device->TransmitPowerState(command.initiator); device->SetPowerStatus(CEC_POWER_STATUS_ON); - SetLGDeckStatus(); device->SetActiveSource(); TransmitImageViewOn(device->GetLogicalAddress(), command.initiator); } @@ -214,32 +214,122 @@ void CSLCommandHandler::HandleVendorCommandPowerOnStatus(const cec_command &comm void CSLCommandHandler::HandleVendorCommandSLConnect(const cec_command &command) { m_bSLEnabled = true; - SetLGDeckStatus(); + TransmitVendorCommandSetDeviceMode(m_processor->GetLogicalAddress(), command.initiator, CEC_DEVICE_TYPE_RECORDING_DEVICE); - CCECBusDevice *primary = m_processor->GetPrimaryDevice(); - - primary->SetActiveSource(); - TransmitImageViewOn(primary->GetLogicalAddress(), command.initiator); - TransmitVendorCommand05(primary->GetLogicalAddress(), command.initiator); + 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::SetLGDeckStatus(void) +bool CSLCommandHandler::HandleGiveDeckStatus(const cec_command &command) { - /* LG TVs only route keypresses when the deck status is set to 0x20 */ - CCECBusDevice *device = m_processor->GetDeviceByType(CEC_DEVICE_TYPE_PLAYBACK_DEVICE); - if (device) - ((CCECPlaybackDevice *)device)->SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG); + 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); + 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) + { + bReturn = device->TransmitPowerState(command.initiator); + device->SetPowerStatus(CEC_POWER_STATUS_ON); + } + else + { + if (!m_bActiveSourceSent) + { + device->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON); + bReturn = device->TransmitPowerState(command.initiator); + ActivateSource(); + } + else if (m_resetPowerState.IsSet() && m_resetPowerState.TimeLeft() > 0) + { + /* 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); + m_resetPowerState.Init(5000); + } + else + { + bReturn = device->TransmitPowerState(command.initiator); + m_resetPowerState.Init(5000); + } + } + } + + 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; +} + +bool CSLCommandHandler::HandleFeatureAbort(const cec_command &command) +{ + if (command.parameters.size == 0 && m_processor->GetPrimaryDevice()->GetPowerStatus() == CEC_POWER_STATUS_ON && !m_bSLEnabled) + { + m_processor->GetPrimaryDevice()->TransmitPowerState(command.initiator); + m_processor->GetPrimaryDevice()->TransmitVendorID(CECDEVICE_BROADCAST, false); + } + + return CCECCommandHandler::HandleFeatureAbort(command); +} + +bool CSLCommandHandler::HandleStandby(const cec_command &command) +{ + if (command.initiator == CECDEVICE_TV) + { + m_bSLEnabled = false; + m_bPowerStateReset = false; + m_bActiveSourceSent = false; + } + + CCECBusDevice *device = GetDevice(command.initiator); if (device) - ((CCECPlaybackDevice *)device)->SetDeckStatus(CEC_DECK_INFO_OTHER_STATUS_LG); + device->SetPowerStatus(CEC_POWER_STATUS_STANDBY); + + return true; }