using namespace CEC;
using namespace std;
-#define CEC_TEST_CLIENT_VERSION 8
+#define CEC_TEST_CLIENT_VERSION 1
#include <cecloader.h>
int g_cecLogLevel = CEC_LOG_ALL;
-int g_iLogicalAddress = CECDEVICE_PLAYBACKDEVICE1;
ofstream g_logOutput;
bool g_bShortLog = false;
CStdString g_strPort;
return true;
}
-void flush_log(ICECAdapter *cecParser)
+void FlushLog(ICECAdapter *cecParser)
{
cec_log_message message;
while (cecParser && cecParser->GetNextLogMessage(&message))
}
}
-void list_devices(ICECAdapter *parser)
+void ListDevices(ICECAdapter *parser)
{
cec_adapter *devices = new cec_adapter[10];
uint8_t iDevicesFound = parser->FindAdapters(devices, 10, NULL);
}
}
-void show_help(const char* strExec)
+void ShowHelpCommandLine(const char* strExec)
{
cout << endl <<
strExec << " {-h|--help|-l|--list-devices|[COM PORT]}" << endl <<
"parameters:" << endl <<
" -h --help Shows this help text" << endl <<
" -l --list-devices List all devices on this system" << endl <<
- " -la --logical-address {a} The logical address to use." << endl <<
+ " -t --type {p|r|t|a} The device type to use. More than one is possible." << endl <<
" -f --log-file {file} Writes all libCEC log message to a file" << endl <<
" -sf --short-log-file {file} Writes all libCEC log message without timestamps" << endl <<
" and log levels to a file." << endl <<
" -d --log-level {level} Sets the log level. See cectypes.h for values." << endl <<
+ " -s --single-command Execute a single command and exit. Does not power" << endl <<
+ " on devices on startup and power them off on exit." << endl <<
" [COM PORT] The com port to connect to. If no COM" << endl <<
" port is given, the client tries to connect to the" << endl <<
" first device that is detected." << endl <<
"available commands" << endl;
}
-void show_console_help(void)
+ICECAdapter *CreateParser(cec_device_type_list typeList)
+{
+ ICECAdapter *parser = LibCecInit("CECTester", typeList);
+ if (!parser || parser->GetMinLibVersion() > CEC_TEST_CLIENT_VERSION)
+ {
+ #ifdef __WINDOWS__
+ cout << "Cannot load libcec.dll" << endl;
+ #else
+ cout << "Cannot load libcec.so" << endl;
+ #endif
+ return NULL;
+ }
+
+ CStdString strLog;
+ strLog.Format("CEC Parser created - libcec version %d.%d", parser->GetLibVersionMajor(), parser->GetLibVersionMinor());
+ cout << strLog.c_str() << endl;
+
+ return parser;
+}
+
+void ShowHelpConsole(void)
{
cout << endl <<
"================================================================================" << endl <<
"pow {addr} get the power status of the specified device." << endl <<
"[pow 0] get the power status of the TV" << endl <<
endl <<
+ "poll {addr} poll the specified device." << endl <<
+ "[poll 0] sends a poll message to the TV" << endl <<
+ endl <<
"[mon] {1|0} enable or disable CEC bus monitoring." << endl <<
"[log] {1 - 31} change the log level. see cectypes.h for values." << endl <<
"[ping] send a ping command to the CEC adapter." << endl <<
int main (int argc, char *argv[])
{
- ICECAdapter *parser = LoadLibCec("CECTester");
- if (!parser || parser->GetMinVersion() > CEC_TEST_CLIENT_VERSION)
- {
-#ifdef __WINDOWS__
- cout << "Cannot load libcec.dll" << endl;
-#else
- cout << "Cannot load libcec.so" << endl;
-#endif
- return 1;
- }
- CStdString strLog;
- strLog.Format("CEC Parser created - libcec version %d.%d", parser->GetLibVersion(), parser->GetLibVersionMinor());
- cout << strLog.c_str() << endl;
-
- //make stdin non-blocking
-#ifndef __WINDOWS__
- int flags = fcntl(0, F_GETFL, 0);
- flags |= O_NONBLOCK;
- fcntl(0, F_SETFL, flags);
-#endif
+ cec_device_type_list typeList;
+ typeList.Clear();
int iArgPtr = 1;
+ bool bSingleCommand(false);
while (iArgPtr < argc)
{
if (argc >= iArgPtr + 1)
if (iNewLevel >= CEC_LOG_ERROR && iNewLevel <= CEC_LOG_ALL)
{
g_cecLogLevel = iNewLevel;
- cout << "log level set to " << argv[iArgPtr + 1] << endl;
+ if (!bSingleCommand)
+ cout << "log level set to " << argv[iArgPtr + 1] << endl;
}
else
{
++iArgPtr;
}
}
- else if (!strcmp(argv[iArgPtr], "-la") ||
- !strcmp(argv[iArgPtr], "--logical-address"))
+ else if (!strcmp(argv[iArgPtr], "-t") ||
+ !strcmp(argv[iArgPtr], "--type"))
{
if (argc >= iArgPtr + 2)
{
- int iNewAddress = atoi(argv[iArgPtr + 1]);
- if (iNewAddress >= 0 && iNewAddress <= 15)
+ if (!strcmp(argv[iArgPtr + 1], "p"))
{
- g_iLogicalAddress = iNewAddress;
- cout << "logical address set to " << argv[iArgPtr + 1] << endl;
+ if (!bSingleCommand)
+ cout << "== using device type 'playback device'" << endl;
+ typeList.Add(CEC_DEVICE_TYPE_PLAYBACK_DEVICE);
+ }
+ else if (!strcmp(argv[iArgPtr + 1], "r"))
+ {
+ if (!bSingleCommand)
+ cout << "== using device type 'recording device'" << endl;
+ typeList.Add(CEC_DEVICE_TYPE_RECORDING_DEVICE);
+ }
+ else if (!strcmp(argv[iArgPtr + 1], "t"))
+ {
+ if (!bSingleCommand)
+ cout << "== using device type 'tuner'" << endl;
+ typeList.Add(CEC_DEVICE_TYPE_TUNER);
+ }
+ else if (!strcmp(argv[iArgPtr + 1], "a"))
+ {
+ if (!bSingleCommand)
+ cout << "== using device type 'audio system'" << endl;
+ typeList.Add(CEC_DEVICE_TYPE_AUDIO_SYSTEM);
}
else
{
- cout << "== skipped logical-address parameter: invalid address '" << argv[iArgPtr + 1] << "' ==" << endl;
+ cout << "== skipped invalid device type '" << argv[iArgPtr + 1] << "'" << endl;
}
- iArgPtr += 2;
- }
- else
- {
- cout << "== skipped logical-address parameter: no address given ==" << endl;
++iArgPtr;
}
+ ++iArgPtr;
}
else if (!strcmp(argv[iArgPtr], "--list-devices") ||
!strcmp(argv[iArgPtr], "-l"))
{
- list_devices(parser);
- UnloadLibCec(parser);
+ ICECAdapter *parser = CreateParser(typeList);
+ if (parser)
+ {
+ ListDevices(parser);
+ UnloadLibCec(parser);
+ }
return 0;
}
+ else if (!strcmp(argv[iArgPtr], "--single-command") ||
+ !strcmp(argv[iArgPtr], "-s"))
+ {
+ bSingleCommand = true;
+ ++iArgPtr;
+ }
else if (!strcmp(argv[iArgPtr], "--help") ||
!strcmp(argv[iArgPtr], "-h"))
{
- show_help(argv[0]);
- UnloadLibCec(parser);
+ ShowHelpCommandLine(argv[0]);
return 0;
}
else
}
}
+ if (typeList.IsEmpty())
+ {
+ if (!bSingleCommand)
+ cout << "No device type given. Using 'playback device'" << endl;
+ typeList.Add(CEC_DEVICE_TYPE_PLAYBACK_DEVICE);
+ }
+
+ ICECAdapter *parser = LibCecInit("CECTester", typeList);
+ if (!parser || parser->GetMinLibVersion() > CEC_TEST_CLIENT_VERSION)
+ {
+#ifdef __WINDOWS__
+ cout << "Cannot load libcec.dll" << endl;
+#else
+ cout << "Cannot load libcec.so" << endl;
+#endif
+ return 1;
+ }
+
+ if (!bSingleCommand)
+ {
+ CStdString strLog;
+ strLog.Format("CEC Parser created - libcec version %d.%d", parser->GetLibVersionMajor(), parser->GetLibVersionMinor());
+ cout << strLog.c_str() << endl;
+
+ //make stdin non-blocking
+ #ifndef __WINDOWS__
+ int flags = fcntl(0, F_GETFL, 0);
+ flags |= O_NONBLOCK;
+ fcntl(0, F_SETFL, flags);
+ #endif
+ }
+
if (g_strPort.IsEmpty())
{
- cout << "no serial port given. trying autodetect: ";
+ if (!bSingleCommand)
+ cout << "no serial port given. trying autodetect: ";
cec_adapter devices[10];
uint8_t iDevicesFound = parser->FindAdapters(devices, 10, NULL);
if (iDevicesFound <= 0)
{
+ if (bSingleCommand)
+ cout << "autodetect ";
cout << "FAILED" << endl;
UnloadLibCec(parser);
return 1;
}
else
{
- cout << endl << " path: " << devices[0].path << endl <<
- " com port: " << devices[0].comm << endl << endl;
+ if (!bSingleCommand)
+ {
+ cout << endl << " path: " << devices[0].path << endl <<
+ " com port: " << devices[0].comm << endl << endl;
+ }
g_strPort = devices[0].comm;
}
}
- parser->SetLogicalAddress((cec_logical_address) g_iLogicalAddress);
-
if (!parser->Open(g_strPort.c_str()))
{
cout << "unable to open the device on port " << g_strPort << endl;
- flush_log(parser);
+ FlushLog(parser);
UnloadLibCec(parser);
return 1;
}
- cout << "cec device opened" << endl;
+ if (!bSingleCommand)
+ {
+ cout << "cec device opened" << endl;
- parser->PowerOnDevices(CECDEVICE_TV);
- flush_log(parser);
+ parser->PowerOnDevices(CECDEVICE_TV);
+ FlushLog(parser);
- parser->SetActiveView();
- flush_log(parser);
+ parser->SetActiveSource();
+ FlushLog(parser);
+
+ cout << "waiting for input" << endl;
+ }
bool bContinue(true);
- cout << "waiting for input" << endl;
while (bContinue)
{
- flush_log(parser);
+ if (bSingleCommand)
+ bContinue = false;
+
+ FlushLog(parser);
/* just ignore the command buffer and clear it */
cec_command dummy;
string strvalue;
uint8_t ivalue;
cec_command bytes;
- bytes.clear();
+ bytes.Clear();
while (GetWord(input, strvalue) && HexStrToInt(strvalue, ivalue))
- bytes.push_back(ivalue);
+ bytes.PushBack(ivalue);
if (command == "txn")
bytes.transmit_timeout = 0;
cout << "invalid destination" << endl;
}
}
+ else if (command == "poll")
+ {
+ string strValue;
+ uint8_t iValue = 0;
+ if (GetWord(input, strValue) && HexStrToInt(strValue, iValue) && iValue <= 0xF)
+ {
+ if (parser->PollDevice((cec_logical_address) iValue))
+ cout << "POLL message sent" << endl;
+ else
+ cout << "POLL message not sent" << endl;
+ }
+ else
+ {
+ cout << "invalid destination" << endl;
+ }
+ }
else if (command == "la")
{
string strvalue;
{
cout << "closing the connection" << endl;
parser->Close();
- flush_log(parser);
+ FlushLog(parser);
cout << "opening a new connection" << endl;
parser->Open(g_strPort.c_str());
- flush_log(parser);
+ FlushLog(parser);
- cout << "setting active view" << endl;
- parser->SetActiveView();
+ cout << "setting active source" << endl;
+ parser->SetActiveSource();
}
else if (command == "h" || command == "help")
{
- show_console_help();
+ ShowHelpConsole();
}
else if (command == "q" || command == "quit")
{
if (bContinue)
cout << "waiting for input" << endl;
}
- CCondition::Sleep(50);
+
+ if (bContinue)
+ CCondition::Sleep(50);
}
- parser->StandbyDevices(CECDEVICE_BROADCAST);
+ if (!bSingleCommand)
+ parser->StandbyDevices(CECDEVICE_BROADCAST);
+
parser->Close();
- flush_log(parser);
+ FlushLog(parser);
UnloadLibCec(parser);
if (g_logOutput.is_open())