X-Git-Url: https://git.piment-noir.org/?a=blobdiff_plain;f=src%2Flib%2Fimplementations%2FSLCommandHandler.cpp;h=5a76d286749cd4f19e2068e17ab75cab023a2c01;hb=7e5a449a3115e69adca77af01fb5daf2c04896ad;hp=7a357d319f7d003c5ab086f191448c460016006a;hpb=2ffc8cddcb866da58a4a45923823ad9fb1c296eb;p=deb_libcec.git diff --git a/src/lib/implementations/SLCommandHandler.cpp b/src/lib/implementations/SLCommandHandler.cpp index 7a357d3..5a76d28 100644 --- a/src/lib/implementations/SLCommandHandler.cpp +++ b/src/lib/implementations/SLCommandHandler.cpp @@ -1,7 +1,7 @@ /* * This file is part of the libCEC(R) library. * - * libCEC(R) is Copyright (C) 2011 Pulse-Eight Limited. All rights reserved. + * libCEC(R) is Copyright (C) 2011-2012 Pulse-Eight Limited. All rights reserved. * libCEC(R) is an original work, containing original code. * * libCEC(R) is a trademark of Pulse-Eight Limited. @@ -34,8 +34,7 @@ #include "../devices/CECBusDevice.h" #include "../devices/CECPlaybackDevice.h" #include "../CECProcessor.h" -#include "../platform/timeutils.h" -#include "../platform/threads.h" +#include "../LibCEC.h" using namespace CEC; @@ -46,21 +45,20 @@ 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_bAwaitingReceiveFailed(false), 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 (m_busDevice->GetLogicalAddress() != primary->GetLogicalAddress()) + 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) @@ -73,24 +71,6 @@ CSLCommandHandler::CSLCommandHandler(CCECBusDevice *busDevice) : m_busDevice->SetMenuLanguage(lang); } - -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; -} - bool CSLCommandHandler::InitHandler(void) { if (m_bHandlerInited) @@ -100,14 +80,18 @@ 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_STANDBY); + primary->SetPowerStatus(CEC_POWER_STATUS_IN_TRANSITION_STANDBY_TO_ON); return true; } bool CSLCommandHandler::ActivateSource(void) { + if (m_bActiveSourceSent) + return false; + m_bActiveSourceSent = true; + CCECBusDevice *primary = m_processor->GetPrimaryDevice(); primary->SetActiveSource(); primary->TransmitActiveSource(); @@ -127,25 +111,26 @@ bool CSLCommandHandler::HandleActiveSource(const cec_command &command) return true; } -bool CSLCommandHandler::HandleFeatureAbort(const cec_command &command) +bool CSLCommandHandler::HandleDeviceVendorId(const cec_command &command) { - CCECBusDevice *primary = m_processor->GetPrimaryDevice(); - if (primary->GetPowerStatus(false) == CEC_POWER_STATUS_ON && !m_bPowerStateReset && !m_bSLEnabled) + SetVendorId(command); + + if (!m_bSLEnabled) { - m_bPowerStateReset = true; - primary->SetPowerStatus(CEC_POWER_STATUS_STANDBY); + 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) { - if (m_processor->IsStarted() && m_busDevice->MyLogicalAddressContains(command.destination)) + if (m_processor->IsRunning() && m_busDevice->MyLogicalAddressContains(command.destination)) { 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; @@ -183,6 +168,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); } @@ -207,7 +193,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); } @@ -226,32 +211,111 @@ void CSLCommandHandler::HandleVendorCommandPowerOnStatus(const cec_command &comm void CSLCommandHandler::HandleVendorCommandSLConnect(const cec_command &command) { m_bSLEnabled = true; - SetLGDeckStatus(); - - CCECBusDevice *primary = m_processor->GetPrimaryDevice(); + TransmitVendorCommandSetDeviceMode(m_processor->GetLogicalAddress(), command.initiator, CEC_DEVICE_TYPE_RECORDING_DEVICE); - 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)iSource); + 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 (!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; +} + +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; }