mirror of
https://github.com/tuxbox-fork-migrations/recycled-ni-neutrino.git
synced 2025-08-27 15:32:52 +02:00
Conflicts:
src/neutrino.cpp
src/zapit/src/femanager.cpp
src/zapit/src/getservices.cpp
src/zapit/src/transponder.cpp
Origin commit data
------------------
Branch: ni/coolstream
Commit: 026592e8fe
Author: [CST] Focus <focus.cst@gmail.com>
Date: 2013-02-28 (Thu, 28 Feb 2013)
------------------
This commit was generated by Migit
1639 lines
42 KiB
C++
1639 lines
42 KiB
C++
/*
|
|
Neutrino-GUI - DBoxII-Project
|
|
|
|
Copyright (C) 2001 Steffen Hehn 'McClean'
|
|
2003 thegoodguy
|
|
|
|
Copyright (C) 2008-2012 Stefan Seyfried
|
|
|
|
License: GPL
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <driver/rcinput.h>
|
|
#include <driver/abstime.h>
|
|
|
|
#include <stdio.h>
|
|
#include <asm/types.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/types.h>
|
|
#include <sys/time.h>
|
|
#include <utime.h>
|
|
#include <stdlib.h>
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
#include <termio.h>
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/un.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
|
|
#include <eventserver.h>
|
|
|
|
#include <global.h>
|
|
#include <driver/shutdown_count.h>
|
|
#include <neutrino.h>
|
|
#include <cs_api.h>
|
|
|
|
//#define RCDEBUG
|
|
//#define USE_GETTIMEOFDAY
|
|
|
|
#define ENABLE_REPEAT_CHECK
|
|
|
|
const char * const RC_EVENT_DEVICE[NUMBER_OF_EVENT_DEVICES] = {"/dev/input/nevis_ir"};
|
|
typedef struct input_event t_input_event;
|
|
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
static struct termio orig_termio;
|
|
static bool saved_orig_termio = false;
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
|
|
/*********************************************************************************
|
|
* Constructor - opens rc-input device, selects rc-hardware and starts threads
|
|
*
|
|
*********************************************************************************/
|
|
CRCInput::CRCInput()
|
|
{
|
|
timerid= 1;
|
|
|
|
// pipe for internal event-queue
|
|
// -----------------------------
|
|
if (pipe(fd_pipe_high_priority) < 0)
|
|
{
|
|
perror("fd_pipe_high_priority");
|
|
exit(-1);
|
|
}
|
|
|
|
fcntl(fd_pipe_high_priority[0], F_SETFL, O_NONBLOCK );
|
|
fcntl(fd_pipe_high_priority[1], F_SETFL, O_NONBLOCK );
|
|
|
|
if (pipe(fd_pipe_low_priority) < 0)
|
|
{
|
|
perror("fd_pipe_low_priority");
|
|
exit(-1);
|
|
}
|
|
|
|
fcntl(fd_pipe_low_priority[0], F_SETFL, O_NONBLOCK );
|
|
fcntl(fd_pipe_low_priority[1], F_SETFL, O_NONBLOCK );
|
|
|
|
|
|
// open event-library
|
|
// -----------------------------
|
|
fd_event = 0;
|
|
|
|
//network-setup
|
|
struct sockaddr_un servaddr;
|
|
int clilen;
|
|
memset(&servaddr, 0, sizeof(struct sockaddr_un));
|
|
servaddr.sun_family = AF_UNIX;
|
|
strcpy(servaddr.sun_path, NEUTRINO_UDS_NAME);
|
|
clilen = sizeof(servaddr.sun_family) + strlen(servaddr.sun_path);
|
|
unlink(NEUTRINO_UDS_NAME);
|
|
|
|
//network-setup
|
|
if ((fd_event = socket(AF_UNIX, SOCK_STREAM, 0)) < 0)
|
|
{
|
|
perror("[neutrino] socket\n");
|
|
}
|
|
|
|
if ( bind(fd_event, (struct sockaddr*) &servaddr, clilen) <0 )
|
|
{
|
|
perror("[neutrino] bind failed...\n");
|
|
exit(-1);
|
|
}
|
|
|
|
|
|
if (listen(fd_event, 15) !=0)
|
|
{
|
|
perror("[neutrino] listen failed...\n");
|
|
exit( -1 );
|
|
}
|
|
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++)
|
|
{
|
|
fd_rc[i] = -1;
|
|
}
|
|
clickfd = -1;
|
|
repeat_block = repeat_block_generic = 0;
|
|
open();
|
|
rc_last_key = KEY_MAX;
|
|
|
|
//select and setup remote control hardware
|
|
set_rc_hw();
|
|
}
|
|
|
|
void CRCInput::open()
|
|
{
|
|
close();
|
|
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++)
|
|
{
|
|
if ((fd_rc[i] = ::open(RC_EVENT_DEVICE[i], O_RDONLY)) == -1)
|
|
perror(RC_EVENT_DEVICE[i]);
|
|
else
|
|
{
|
|
fcntl(fd_rc[i], F_SETFL, O_NONBLOCK);
|
|
}
|
|
printf("CRCInput::open: %s fd %d\n", RC_EVENT_DEVICE[i], fd_rc[i]);
|
|
}
|
|
|
|
//+++++++++++++++++++++++++++++++++++++++
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
fd_keyb = STDIN_FILENO;
|
|
#else
|
|
fd_keyb = 0;
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
/*
|
|
::open("/dev/dbox/rc0", O_RDONLY);
|
|
if (fd_keyb<0)
|
|
{
|
|
perror("/dev/stdin");
|
|
exit(-1);
|
|
}
|
|
*/
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
::fcntl(fd_keyb, F_SETFL, O_NONBLOCK);
|
|
|
|
struct termio new_termio;
|
|
|
|
::ioctl(STDIN_FILENO, TCGETA, &orig_termio);
|
|
|
|
saved_orig_termio = true;
|
|
|
|
new_termio = orig_termio;
|
|
new_termio.c_lflag &= ~ICANON;
|
|
// new_termio.c_lflag &= ~(ICANON|ECHO);
|
|
new_termio.c_cc[VMIN ] = 1;
|
|
new_termio.c_cc[VTIME] = 0;
|
|
|
|
::ioctl(STDIN_FILENO, TCSETA, &new_termio);
|
|
|
|
#else
|
|
//fcntl(fd_keyb, F_SETFL, O_NONBLOCK );
|
|
|
|
//+++++++++++++++++++++++++++++++++++++++
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
|
|
open_click();
|
|
calculateMaxFd();
|
|
}
|
|
|
|
void CRCInput::close()
|
|
{
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++) {
|
|
if (fd_rc[i] != -1) {
|
|
::close(fd_rc[i]);
|
|
fd_rc[i] = -1;
|
|
}
|
|
}
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
if (saved_orig_termio)
|
|
{
|
|
::ioctl(STDIN_FILENO, TCSETA, &orig_termio);
|
|
printf("Original terminal settings restored.\n");
|
|
}
|
|
#else
|
|
/*
|
|
if(fd_keyb)
|
|
{
|
|
::close(fd_keyb);
|
|
}
|
|
*/
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
calculateMaxFd();
|
|
}
|
|
|
|
void CRCInput::calculateMaxFd()
|
|
{
|
|
fd_max = fd_event;
|
|
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++)
|
|
if (fd_rc[i] > fd_max)
|
|
fd_max = fd_rc[i];
|
|
|
|
if(fd_pipe_high_priority[0] > fd_max)
|
|
fd_max = fd_pipe_high_priority[0];
|
|
if(fd_pipe_low_priority[0] > fd_max)
|
|
fd_max = fd_pipe_low_priority[0];
|
|
}
|
|
|
|
/**************************************************************************
|
|
* Destructor - close the input-device
|
|
*
|
|
**************************************************************************/
|
|
CRCInput::~CRCInput()
|
|
{
|
|
close();
|
|
|
|
if(fd_pipe_high_priority[0])
|
|
::close(fd_pipe_high_priority[0]);
|
|
if(fd_pipe_high_priority[1])
|
|
::close(fd_pipe_high_priority[1]);
|
|
|
|
if(fd_pipe_low_priority[0])
|
|
::close(fd_pipe_low_priority[0]);
|
|
if(fd_pipe_low_priority[1])
|
|
::close(fd_pipe_low_priority[1]);
|
|
|
|
if(fd_event)
|
|
::close(fd_event);
|
|
close_click();
|
|
}
|
|
|
|
/**************************************************************************
|
|
* stopInput - stop reading rcin for plugins
|
|
*
|
|
**************************************************************************/
|
|
void CRCInput::stopInput()
|
|
{
|
|
close();
|
|
}
|
|
|
|
/**************************************************************************
|
|
* restartInput - restart reading rcin after calling plugins
|
|
*
|
|
**************************************************************************/
|
|
void CRCInput::restartInput()
|
|
{
|
|
close();
|
|
open();
|
|
}
|
|
#if 0
|
|
//never used
|
|
int CRCInput::messageLoop( bool anyKeyCancels, int timeout )
|
|
{
|
|
neutrino_msg_t msg;
|
|
neutrino_msg_data_t data;
|
|
|
|
int res = menu_return::RETURN_REPAINT;
|
|
|
|
bool doLoop = true;
|
|
|
|
if ( timeout == -1 )
|
|
timeout = g_settings.timing[SNeutrinoSettings::TIMING_MENU];
|
|
|
|
uint64_t timeoutEnd = CRCInput::calcTimeoutEnd( timeout == 0 ? 0xFFFF : timeout);
|
|
|
|
while (doLoop)
|
|
{
|
|
g_RCInput->getMsgAbsoluteTimeout( &msg, &data, &timeoutEnd );
|
|
|
|
if ( ( msg == CRCInput::RC_timeout ) ||
|
|
( msg == CRCInput::RC_home ) ||
|
|
( msg == CRCInput::RC_ok ) )
|
|
doLoop = false;
|
|
else if((msg == CRCInput::RC_sat) || (msg == CRCInput::RC_favorites)) {
|
|
}
|
|
else
|
|
{
|
|
int mr = CNeutrinoApp::getInstance()->handleMsg( msg, data );
|
|
|
|
if ( mr & messages_return::cancel_all )
|
|
{
|
|
res = menu_return::RETURN_EXIT_ALL;
|
|
doLoop = false;
|
|
}
|
|
else if ( mr & messages_return::unhandled )
|
|
{
|
|
if ((msg <= CRCInput::RC_MaxRC) &&
|
|
(data == 0)) /* <- button pressed */
|
|
{
|
|
if ( anyKeyCancels )
|
|
doLoop = false;
|
|
else
|
|
timeoutEnd = CRCInput::calcTimeoutEnd( timeout );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
return res;
|
|
}
|
|
#endif
|
|
|
|
int CRCInput::addTimer(uint64_t Interval, bool oneshot, bool correct_time )
|
|
{
|
|
#ifdef USE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
gettimeofday( &tv, NULL );
|
|
uint64_t timeNow = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t timeNow = time_monotonic_us();
|
|
#endif
|
|
|
|
timer _newtimer;
|
|
if (!oneshot)
|
|
_newtimer.interval = Interval;
|
|
else
|
|
_newtimer.interval = 0;
|
|
|
|
_newtimer.id = timerid++;
|
|
if ( correct_time )
|
|
_newtimer.times_out = timeNow+ Interval;
|
|
else
|
|
_newtimer.times_out = Interval;
|
|
|
|
_newtimer.correct_time = correct_time;
|
|
|
|
//printf("adding timer %d (0x%llx, 0x%llx)\n", _newtimer.id, _newtimer.times_out, Interval);
|
|
|
|
std::vector<timer>::iterator e;
|
|
for ( e= timers.begin(); e!= timers.end(); ++e )
|
|
if ( e->times_out> _newtimer.times_out )
|
|
break;
|
|
|
|
timers.insert(e, _newtimer);
|
|
return _newtimer.id;
|
|
}
|
|
|
|
#ifdef USE_GETTIMEOFDAY
|
|
int CRCInput::addTimer(struct timeval Timeout)
|
|
{
|
|
uint64_t timesout = (uint64_t) Timeout.tv_usec + (uint64_t)((uint64_t) Timeout.tv_sec * (uint64_t) 1000000);
|
|
return addTimer( timesout, true, false );
|
|
}
|
|
|
|
int CRCInput::addTimer(const time_t *Timeout)
|
|
{
|
|
return addTimer( (uint64_t)*Timeout* (uint64_t) 1000000, true, false );
|
|
}
|
|
#endif
|
|
|
|
void CRCInput::killTimer(uint32_t &id)
|
|
{
|
|
//printf("killing timer %d\n", id);
|
|
if(id == 0)
|
|
return;
|
|
|
|
std::vector<timer>::iterator e;
|
|
for ( e= timers.begin(); e!= timers.end(); ++e )
|
|
if ( e->id == id )
|
|
{
|
|
timers.erase(e);
|
|
break;
|
|
}
|
|
id = 0;
|
|
}
|
|
|
|
int CRCInput::checkTimers()
|
|
{
|
|
int _id = 0;
|
|
#ifdef USE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
gettimeofday( &tv, NULL );
|
|
uint64_t timeNow = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t timeNow = time_monotonic_us();
|
|
#endif
|
|
std::vector<timer>::iterator e;
|
|
for ( e= timers.begin(); e!= timers.end(); ++e )
|
|
if ( e->times_out< timeNow+ 2000 )
|
|
{
|
|
//printf("timeout timer %d %llx %llx\n",e->id,e->times_out,timeNow );
|
|
_id = e->id;
|
|
if ( e->interval != 0 )
|
|
{
|
|
timer _newtimer;
|
|
_newtimer.id = e->id;
|
|
_newtimer.interval = e->interval;
|
|
_newtimer.correct_time = e->correct_time;
|
|
if ( _newtimer.correct_time )
|
|
_newtimer.times_out = timeNow + e->interval;
|
|
else
|
|
_newtimer.times_out = e->times_out + e->interval;
|
|
|
|
timers.erase(e);
|
|
for ( e= timers.begin(); e!= timers.end(); ++e )
|
|
if ( e->times_out> _newtimer.times_out )
|
|
break;
|
|
|
|
timers.insert(e, _newtimer);
|
|
}
|
|
else
|
|
timers.erase(e);
|
|
|
|
break;
|
|
}
|
|
// else
|
|
// printf("skipped timer %d %llx %llx\n",e->id,e->times_out, timeNow );
|
|
//printf("checkTimers: return %d\n", _id);
|
|
return _id;
|
|
}
|
|
|
|
|
|
|
|
int64_t CRCInput::calcTimeoutEnd(const int timeout_in_seconds)
|
|
{
|
|
#ifdef USE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
gettimeofday(&tv, NULL);
|
|
return (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec + (uint64_t)timeout_in_seconds) * (uint64_t) 1000000;
|
|
#else
|
|
return time_monotonic_us() + ((uint64_t)timeout_in_seconds * (uint64_t) 1000000);
|
|
#endif
|
|
}
|
|
|
|
int64_t CRCInput::calcTimeoutEnd_MS(const int timeout_in_milliseconds)
|
|
{
|
|
#ifdef USE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
gettimeofday(&tv, NULL);
|
|
uint64_t timeNow = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t timeNow = time_monotonic_us();
|
|
#endif
|
|
return ( timeNow + timeout_in_milliseconds * 1000 );
|
|
}
|
|
|
|
void CRCInput::getMsgAbsoluteTimeout(neutrino_msg_t * msg, neutrino_msg_data_t * data, uint64_t *TimeoutEnd, bool bAllowRepeatLR)
|
|
{
|
|
#ifdef USE_GETTIMEOFDAY
|
|
struct timeval tv;
|
|
gettimeofday( &tv, NULL );
|
|
uint64_t timeNow = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t timeNow = time_monotonic_us();
|
|
#endif
|
|
uint64_t diff;
|
|
|
|
if ( *TimeoutEnd < timeNow+ 100 )
|
|
diff = 100; // Minimum Differenz...
|
|
else
|
|
diff = ( *TimeoutEnd - timeNow );
|
|
|
|
//printf("CRCInput::getMsgAbsoluteTimeout diff %llx TimeoutEnd %llx now %llx\n", diff, *TimeoutEnd, timeNow);
|
|
getMsg_us( msg, data, diff, bAllowRepeatLR );
|
|
#ifdef USE_GETTIMEOFDAY
|
|
if ( *msg == NeutrinoMessages::EVT_TIMESET )
|
|
{
|
|
// recalculate timeout....
|
|
//uint64_t ta= *TimeoutEnd;
|
|
*TimeoutEnd= *TimeoutEnd + *(int64_t*) *data;
|
|
|
|
//printf("[getMsgAbsoluteTimeout]: EVT_TIMESET - recalculate timeout\n%llx/%llx - %llx/%llx\n", timeNow, *(int64_t*) *data, *TimeoutEnd, ta );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void CRCInput::getMsg(neutrino_msg_t * msg, neutrino_msg_data_t * data, int Timeout, bool bAllowRepeatLR)
|
|
{
|
|
getMsg_us(msg, data, (uint64_t) Timeout * 100 * 1000, bAllowRepeatLR);
|
|
}
|
|
|
|
void CRCInput::getMsg_ms(neutrino_msg_t * msg, neutrino_msg_data_t * data, int Timeout, bool bAllowRepeatLR)
|
|
{
|
|
getMsg_us(msg, data, (uint64_t) Timeout * 1000, bAllowRepeatLR);
|
|
}
|
|
|
|
void CRCInput::getMsg_us(neutrino_msg_t * msg, neutrino_msg_data_t * data, uint64_t Timeout, bool bAllowRepeatLR)
|
|
{
|
|
static uint64_t last_keypress = 0ULL;
|
|
//uint64_t getKeyBegin;
|
|
|
|
//static __u16 rc_last_key = KEY_MAX;
|
|
static __u16 rc_last_repeat_key = KEY_MAX;
|
|
|
|
struct timeval tv;
|
|
struct timeval tvselect;
|
|
uint64_t InitialTimeout = Timeout;
|
|
int64_t targetTimeout;
|
|
|
|
int timer_id;
|
|
fd_set rfds;
|
|
t_input_event ev;
|
|
|
|
*data = 0;
|
|
|
|
// wiederholung reinmachen - dass wirklich die ganze zeit bis timeout gewartet wird!
|
|
#ifdef USE_GETTIMEOFDAY
|
|
gettimeofday( &tv, NULL );
|
|
uint64_t getKeyBegin = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t getKeyBegin = time_monotonic_us();
|
|
#endif
|
|
while(1) {
|
|
timer_id = 0;
|
|
if ( !timers.empty() )
|
|
{
|
|
#ifdef USE_GETTIMEOFDAY
|
|
gettimeofday( &tv, NULL );
|
|
uint64_t t_n= (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
#else
|
|
uint64_t t_n = time_monotonic_us();
|
|
#endif
|
|
if ( timers[0].times_out< t_n )
|
|
{
|
|
timer_id = checkTimers();
|
|
*msg = NeutrinoMessages::EVT_TIMER;
|
|
*data = timer_id;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
targetTimeout = timers[0].times_out - t_n;
|
|
if ( (uint64_t) targetTimeout> Timeout)
|
|
targetTimeout= Timeout;
|
|
else
|
|
timer_id = timers[0].id;
|
|
}
|
|
}
|
|
else
|
|
targetTimeout= Timeout;
|
|
|
|
tvselect.tv_sec = targetTimeout/1000000;
|
|
tvselect.tv_usec = targetTimeout%1000000;
|
|
|
|
FD_ZERO(&rfds);
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++)
|
|
{
|
|
if (fd_rc[i] != -1)
|
|
FD_SET(fd_rc[i], &rfds);
|
|
}
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
if (true)
|
|
#else
|
|
if (fd_keyb> 0)
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
FD_SET(fd_keyb, &rfds);
|
|
|
|
FD_SET(fd_event, &rfds);
|
|
FD_SET(fd_pipe_high_priority[0], &rfds);
|
|
FD_SET(fd_pipe_low_priority[0], &rfds);
|
|
|
|
int status = select(fd_max+1, &rfds, NULL, NULL, &tvselect);
|
|
|
|
if ( status == -1 )
|
|
{
|
|
perror("[neutrino - getMsg_us]: select returned ");
|
|
// in case of an error return timeout...?!
|
|
*msg = RC_timeout;
|
|
*data = 0;
|
|
return;
|
|
}
|
|
else if ( status == 0 ) // Timeout!
|
|
{
|
|
if ( timer_id != 0 )
|
|
{
|
|
timer_id = checkTimers();
|
|
if ( timer_id != 0 )
|
|
{
|
|
*msg = NeutrinoMessages::EVT_TIMER;
|
|
*data = timer_id;
|
|
return;
|
|
}
|
|
else
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
*msg = RC_timeout;
|
|
*data = 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
if(FD_ISSET(fd_pipe_high_priority[0], &rfds))
|
|
{
|
|
struct event buf;
|
|
|
|
read(fd_pipe_high_priority[0], &buf, sizeof(buf));
|
|
|
|
*msg = buf.msg;
|
|
*data = buf.data;
|
|
|
|
// printf("got event from high-pri pipe %x %x\n", *msg, *data );
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
#ifdef KEYBOARD_INSTEAD_OF_REMOTE_CONTROL
|
|
if (FD_ISSET(fd_keyb, &rfds))
|
|
{
|
|
int trkey;
|
|
char key = 0;
|
|
read(fd_keyb, &key, sizeof(key));
|
|
|
|
switch(key)
|
|
{
|
|
case 27: // <- Esc
|
|
trkey = KEY_HOME;
|
|
break;
|
|
case 10: // <- Return
|
|
case 'o':
|
|
trkey = KEY_OK;
|
|
break;
|
|
case 'p':
|
|
trkey = KEY_POWER;
|
|
break;
|
|
case 's':
|
|
trkey = KEY_SETUP;
|
|
break;
|
|
case 'h':
|
|
trkey = KEY_HELP;
|
|
break;
|
|
case 'i':
|
|
trkey = KEY_UP;
|
|
break;
|
|
case 'm':
|
|
trkey = KEY_DOWN;
|
|
break;
|
|
case 'j':
|
|
trkey = KEY_LEFT;
|
|
break;
|
|
case 'k':
|
|
trkey = KEY_RIGHT;
|
|
break;
|
|
case 'r':
|
|
trkey = KEY_RED;
|
|
break;
|
|
case 'g':
|
|
trkey = KEY_GREEN;
|
|
break;
|
|
case 'y':
|
|
trkey = KEY_YELLOW;
|
|
break;
|
|
case 'b':
|
|
trkey = KEY_BLUE;
|
|
break;
|
|
case '0':
|
|
trkey = RC_0;
|
|
break;
|
|
case '1':
|
|
trkey = RC_1;
|
|
break;
|
|
case '2':
|
|
trkey = RC_2;
|
|
break;
|
|
case '3':
|
|
trkey = RC_3;
|
|
break;
|
|
case '4':
|
|
trkey = RC_4;
|
|
break;
|
|
case '5':
|
|
trkey = RC_5;
|
|
break;
|
|
case '6':
|
|
trkey = RC_6;
|
|
break;
|
|
case '7':
|
|
trkey = RC_7;
|
|
break;
|
|
case '8':
|
|
trkey = RC_8;
|
|
break;
|
|
case '9':
|
|
trkey = RC_9;
|
|
break;
|
|
case '+':
|
|
trkey = RC_plus;
|
|
break;
|
|
case '-':
|
|
trkey = RC_minus;
|
|
break;
|
|
case 'a':
|
|
trkey = KEY_A;
|
|
break;
|
|
case 'u':
|
|
trkey = KEY_U;
|
|
break;
|
|
case '/':
|
|
trkey = KEY_SLASH;
|
|
break;
|
|
case '\\':
|
|
trkey = KEY_BACKSLASH;
|
|
break;
|
|
default:
|
|
trkey = RC_nokey;
|
|
}
|
|
if (trkey != RC_nokey)
|
|
{
|
|
*msg = trkey;
|
|
*data = 0; /* <- button pressed */
|
|
return;
|
|
}
|
|
}
|
|
#else
|
|
/*
|
|
if(FD_ISSET(fd_keyb, &rfds))
|
|
{
|
|
char key = 0;
|
|
read(fd_keyb, &key, sizeof(key));
|
|
printf("keyboard: %d\n", rc_key);
|
|
}
|
|
*/
|
|
#endif /* KEYBOARD_INSTEAD_OF_REMOTE_CONTROL */
|
|
|
|
if(FD_ISSET(fd_event, &rfds)) {
|
|
//printf("[neutrino] event - accept!\n");
|
|
socklen_t clilen;
|
|
struct sockaddr_in cliaddr;
|
|
clilen = sizeof(cliaddr);
|
|
int fd_eventclient = accept(fd_event, (struct sockaddr *) &cliaddr, &clilen);
|
|
|
|
*msg = RC_nokey;
|
|
//printf("[neutrino] network event - read!\n");
|
|
CEventServer::eventHead emsg;
|
|
int read_bytes= recv(fd_eventclient, &emsg, sizeof(emsg), MSG_WAITALL);
|
|
//printf("[neutrino] event read %d bytes - following %d bytes\n", read_bytes, emsg.dataSize );
|
|
if ( read_bytes == sizeof(emsg) ) {
|
|
bool dont_delete_p = false;
|
|
|
|
unsigned char* p;
|
|
p= new unsigned char[ emsg.dataSize + 1 ];
|
|
if ( p!=NULL )
|
|
{
|
|
read_bytes= recv(fd_eventclient, p, emsg.dataSize, MSG_WAITALL);
|
|
//printf("[neutrino] eventbody read %d bytes - initiator %x\n", read_bytes, emsg.initiatorID );
|
|
|
|
#if 0
|
|
if ( emsg.initiatorID == CEventServer::INITID_CONTROLD )
|
|
{
|
|
switch(emsg.eventID)
|
|
{
|
|
case CControldClient::EVT_VOLUMECHANGED :
|
|
*msg = NeutrinoMessages::EVT_VOLCHANGED;
|
|
*data = 0;
|
|
break;
|
|
case CControldClient::EVT_MUTECHANGED :
|
|
*msg = NeutrinoMessages::EVT_MUTECHANGED;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CControldClient::EVT_VCRCHANGED :
|
|
*msg = NeutrinoMessages::EVT_VCRCHANGED;
|
|
*data = *(int*) p;
|
|
break;
|
|
case CControldClient::EVT_MODECHANGED :
|
|
*msg = NeutrinoMessages::EVT_MODECHANGED;
|
|
*data = *(int*) p;
|
|
break;
|
|
default:
|
|
printf("[neutrino] event INITID_CONTROLD - unknown eventID 0x%x\n", emsg.eventID );
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
if ( emsg.initiatorID == CEventServer::INITID_HTTPD )
|
|
{
|
|
switch(emsg.eventID)
|
|
{
|
|
case NeutrinoMessages::SHUTDOWN :
|
|
*msg = NeutrinoMessages::SHUTDOWN;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::EVT_POPUP :
|
|
*msg = NeutrinoMessages::EVT_POPUP;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case NeutrinoMessages::EVT_EXTMSG :
|
|
*msg = NeutrinoMessages::EVT_EXTMSG;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case NeutrinoMessages::CHANGEMODE : // Change
|
|
*msg = NeutrinoMessages::CHANGEMODE;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case NeutrinoMessages::STANDBY_TOGGLE :
|
|
*msg = NeutrinoMessages::STANDBY_TOGGLE;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::STANDBY_ON :
|
|
*msg = NeutrinoMessages::STANDBY_ON;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::STANDBY_OFF :
|
|
*msg = NeutrinoMessages::STANDBY_OFF;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::EVT_START_PLUGIN :
|
|
*msg = NeutrinoMessages::EVT_START_PLUGIN;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case NeutrinoMessages::LOCK_RC :
|
|
*msg = NeutrinoMessages::LOCK_RC;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::UNLOCK_RC :
|
|
*msg = NeutrinoMessages::UNLOCK_RC;
|
|
*data = 0;
|
|
break;
|
|
case NeutrinoMessages::RELOAD_SETUP :
|
|
*msg = NeutrinoMessages::RELOAD_SETUP;
|
|
*data = 0;
|
|
break;
|
|
default:
|
|
printf("[neutrino] event INITID_HTTPD - unknown eventID 0x%x\n", emsg.eventID );
|
|
}
|
|
}
|
|
else if ( emsg.initiatorID == CEventServer::INITID_SECTIONSD )
|
|
{
|
|
//printf("[neutrino] event - from SECTIONSD %x %x\n", emsg.eventID, *(unsigned*) p);
|
|
switch(emsg.eventID)
|
|
{
|
|
case CSectionsdClient::EVT_TIMESET:
|
|
{
|
|
#if 0
|
|
struct timeval ltv;
|
|
gettimeofday(<v, NULL);
|
|
int64_t timeOld = ltv.tv_usec + ltv.tv_sec * (int64_t)1000000;
|
|
time_t dvbtime = *((time_t*)p);
|
|
if (dvbtime) {
|
|
printf("[neutrino] timeset event. ");
|
|
time_t difftime = dvbtime - ltv.tv_sec;
|
|
if (abs(difftime) > 120)
|
|
{
|
|
printf("difference is %ld s, stepping...\n", difftime);
|
|
if (stime(&dvbtime))
|
|
perror("stime");
|
|
} else if (difftime != 0) {
|
|
struct timeval oldd;
|
|
ltv.tv_sec = difftime;
|
|
ltv.tv_usec = 0;
|
|
if (adjtime(<v, &oldd))
|
|
perror("adjtime");
|
|
int64_t t = oldd.tv_sec * 1000000LL + oldd.tv_usec;
|
|
printf("difference is %ld s, using adjtime(). oldd: %lld us\n", difftime, t);
|
|
} else
|
|
printf("difference is 0 s, nothing to do...\n");
|
|
}
|
|
gettimeofday( <v, NULL );
|
|
int64_t timeNew = ltv.tv_usec + ltv.tv_sec * (int64_t)1000000;
|
|
|
|
delete[] p;//new [] delete []
|
|
p = new unsigned char[sizeof(int64_t)];
|
|
*(int64_t*) p = timeNew - timeOld;
|
|
#endif
|
|
printf("[neutrino] CSectionsdClient::EVT_TIMESET: timediff %" PRId64 "\n", *(int64_t*) p);
|
|
/* FIXME what this code really do ? */
|
|
if ((int64_t)last_keypress > *(int64_t*)p)
|
|
last_keypress += *(int64_t *)p;
|
|
|
|
#ifdef USE_GETTIMEOFDAY
|
|
// Timer anpassen
|
|
for(std::vector<timer>::iterator e = timers.begin(); e != timers.end(); ++e)
|
|
if (e->correct_time)
|
|
e->times_out+= *(int64_t*) p;
|
|
#endif
|
|
*msg = NeutrinoMessages::EVT_TIMESET;
|
|
*data = (neutrino_msg_data_t) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
}
|
|
case CSectionsdClient::EVT_GOT_CN_EPG:
|
|
printf("[neutrino] CSectionsdClient::EVT_GOT_CN_EPG\n");
|
|
*msg = NeutrinoMessages::EVT_CURRENTNEXT_EPG;
|
|
*data = (neutrino_msg_data_t) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CSectionsdClient::EVT_WRITE_SI_FINISHED:
|
|
*msg = NeutrinoMessages::EVT_SI_FINISHED;
|
|
*data = 0;
|
|
break;
|
|
#if 0
|
|
case CSectionsdClient::EVT_SERVICES_UPDATE:
|
|
*msg = NeutrinoMessages::EVT_SERVICES_UPD;
|
|
*data = 0;
|
|
break;
|
|
case CSectionsdClient::EVT_BOUQUETS_UPDATE:
|
|
break;
|
|
#endif
|
|
default:
|
|
printf("[neutrino] event INITID_SECTIONSD - unknown eventID 0x%x\n", emsg.eventID );
|
|
}
|
|
}
|
|
else if ( emsg.initiatorID == CEventServer::INITID_ZAPIT )
|
|
{
|
|
//printf("[neutrino] event - from ZAPIT %x %x\n", emsg.eventID, *(unsigned*) p);
|
|
switch(emsg.eventID)
|
|
{
|
|
case CZapitClient::EVT_RECORDMODE_ACTIVATED:
|
|
*msg = NeutrinoMessages::EVT_RECORDMODE;
|
|
*data = true;
|
|
break;
|
|
case CZapitClient::EVT_RECORDMODE_DEACTIVATED:
|
|
*msg = NeutrinoMessages::EVT_RECORDMODE;
|
|
*data = false;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_COMPLETE:
|
|
*msg = NeutrinoMessages::EVT_ZAP_COMPLETE;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_FAILED:
|
|
*msg = NeutrinoMessages::EVT_ZAP_FAILED;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_SUB_FAILED:
|
|
*msg = NeutrinoMessages::EVT_ZAP_SUB_FAILED;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_COMPLETE_IS_NVOD:
|
|
*msg = NeutrinoMessages::EVT_ZAP_ISNVOD;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_SUB_COMPLETE:
|
|
*msg = NeutrinoMessages::EVT_ZAP_SUB_COMPLETE;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_COMPLETE:
|
|
*msg = NeutrinoMessages::EVT_SCAN_COMPLETE;
|
|
*data = 0;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_NUM_TRANSPONDERS:
|
|
*msg = NeutrinoMessages::EVT_SCAN_NUM_TRANSPONDERS;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_REPORT_NUM_SCANNED_TRANSPONDERS:
|
|
*msg = NeutrinoMessages::EVT_SCAN_REPORT_NUM_SCANNED_TRANSPONDERS;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_REPORT_FREQUENCY:
|
|
*msg = NeutrinoMessages::EVT_SCAN_REPORT_FREQUENCY;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_FOUND_A_CHAN:
|
|
*msg = NeutrinoMessages::EVT_SCAN_FOUND_A_CHAN;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_SERVICENAME:
|
|
*msg = NeutrinoMessages::EVT_SCAN_SERVICENAME;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_FOUND_TV_CHAN:
|
|
*msg = NeutrinoMessages::EVT_SCAN_FOUND_TV_CHAN;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_FOUND_RADIO_CHAN:
|
|
*msg = NeutrinoMessages::EVT_SCAN_FOUND_RADIO_CHAN;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_FOUND_DATA_CHAN:
|
|
*msg = NeutrinoMessages::EVT_SCAN_FOUND_DATA_CHAN;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_REPORT_FREQUENCYP:
|
|
*msg = NeutrinoMessages::EVT_SCAN_REPORT_FREQUENCYP;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_NUM_CHANNELS:
|
|
*msg = NeutrinoMessages::EVT_SCAN_NUM_CHANNELS;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_PROVIDER:
|
|
*msg = NeutrinoMessages::EVT_SCAN_PROVIDER;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_SATELLITE:
|
|
*msg = NeutrinoMessages::EVT_SCAN_SATELLITE;
|
|
break;
|
|
case CZapitClient::EVT_BOUQUETS_CHANGED:
|
|
*msg = NeutrinoMessages::EVT_BOUQUETSCHANGED;
|
|
*data = 0;
|
|
break;
|
|
case CZapitClient::EVT_SERVICES_CHANGED:
|
|
*msg = NeutrinoMessages::EVT_SERVICESCHANGED;
|
|
*data = 0;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_CA_CLEAR:
|
|
*msg = NeutrinoMessages::EVT_ZAP_CA_CLEAR;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_CA_LOCK:
|
|
*msg = NeutrinoMessages::EVT_ZAP_CA_LOCK;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_CA_FTA:
|
|
*msg = NeutrinoMessages::EVT_ZAP_CA_FTA;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_CA_ID :
|
|
*msg = NeutrinoMessages::EVT_ZAP_CA_ID;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SCAN_FAILED:
|
|
*msg = NeutrinoMessages::EVT_SCAN_FAILED;
|
|
*data = 0;
|
|
break;
|
|
case CZapitClient::EVT_ZAP_MOTOR:
|
|
*msg = NeutrinoMessages::EVT_ZAP_MOTOR;
|
|
*data = *(unsigned*) p;
|
|
break;
|
|
case CZapitClient::EVT_SDT_CHANGED:
|
|
*msg = NeutrinoMessages::EVT_SERVICES_UPD;
|
|
*data = 0;
|
|
break;
|
|
case CZapitClient::EVT_PMT_CHANGED:
|
|
*msg = NeutrinoMessages::EVT_PMT_CHANGED;
|
|
*data = (neutrino_msg_data_t) p;
|
|
break;
|
|
case CZapitClient::EVT_TUNE_COMPLETE:
|
|
*msg = NeutrinoMessages::EVT_TUNE_COMPLETE;
|
|
*data = (neutrino_msg_data_t) p;
|
|
break;
|
|
default:
|
|
printf("[neutrino] event INITID_ZAPIT - unknown eventID 0x%x\n", emsg.eventID );
|
|
}
|
|
if (((*msg) >= CRCInput::RC_WithData) && ((*msg) < CRCInput::RC_WithData + 0x10000000))
|
|
{
|
|
*data = (neutrino_msg_data_t) p;
|
|
dont_delete_p = true;
|
|
}
|
|
}
|
|
else if ( emsg.initiatorID == CEventServer::INITID_TIMERD )
|
|
{
|
|
/*
|
|
if (emsg.eventID==CTimerdClient::EVT_ANNOUNCE_NEXTPROGRAM)
|
|
{
|
|
}
|
|
|
|
if (emsg.eventID==CTimerdClient::EVT_NEXTPROGRAM)
|
|
{
|
|
*msg = NeutrinoMessages::EVT_NEXTPROGRAM;
|
|
*data = (neutrino_msg_data_t) p;
|
|
dont_delete_p = true;
|
|
}
|
|
*/
|
|
switch(emsg.eventID)
|
|
{
|
|
case CTimerdClient::EVT_ANNOUNCE_RECORD :
|
|
*msg = NeutrinoMessages::ANNOUNCE_RECORD;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_ANNOUNCE_ZAPTO :
|
|
*msg = NeutrinoMessages::ANNOUNCE_ZAPTO;
|
|
*data = (neutrino_msg_data_t)p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_ANNOUNCE_SHUTDOWN :
|
|
*msg = NeutrinoMessages::ANNOUNCE_SHUTDOWN;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_ANNOUNCE_SLEEPTIMER :
|
|
*msg = NeutrinoMessages::ANNOUNCE_SLEEPTIMER;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_SLEEPTIMER :
|
|
*msg = NeutrinoMessages::SLEEPTIMER;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_RECORD_START :
|
|
*msg = NeutrinoMessages::RECORD_START;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_RECORD_STOP :
|
|
*msg = NeutrinoMessages::RECORD_STOP;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_ZAPTO :
|
|
*msg = NeutrinoMessages::ZAPTO;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_SHUTDOWN :
|
|
*msg = NeutrinoMessages::SHUTDOWN;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_STANDBY_ON :
|
|
*msg = NeutrinoMessages::STANDBY_ON;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_STANDBY_OFF :
|
|
*msg = NeutrinoMessages::STANDBY_OFF;
|
|
*data = 0;
|
|
break;
|
|
case CTimerdClient::EVT_REMIND :
|
|
*msg = NeutrinoMessages::REMIND;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
case CTimerdClient::EVT_EXEC_PLUGIN :
|
|
*msg = NeutrinoMessages::EVT_START_PLUGIN;
|
|
*data = (unsigned) p;
|
|
dont_delete_p = true;
|
|
break;
|
|
|
|
default :
|
|
printf("[neutrino] event INITID_TIMERD - unknown eventID 0x%x\n", emsg.eventID );
|
|
|
|
}
|
|
}
|
|
else if (emsg.initiatorID == CEventServer::INITID_NEUTRINO)
|
|
{
|
|
#if 0
|
|
if ((emsg.eventID == NeutrinoMessages::EVT_RECORDING_ENDED) &&
|
|
(read_bytes == sizeof(stream2file_status2_t)))
|
|
{
|
|
*msg = NeutrinoMessages::EVT_RECORDING_ENDED;
|
|
*data = (neutrino_msg_data_t) p;
|
|
dont_delete_p = true;
|
|
}
|
|
#endif
|
|
}
|
|
else if (emsg.initiatorID == CEventServer::INITID_GENERIC_INPUT_EVENT_PROVIDER)
|
|
{
|
|
if (read_bytes == sizeof(int))
|
|
{
|
|
*msg = *(int *)p;
|
|
*data = emsg.eventID;
|
|
}
|
|
}
|
|
else
|
|
printf("[neutrino] event - unknown initiatorID 0x%x\n", emsg.initiatorID);
|
|
if ( !dont_delete_p )
|
|
{
|
|
delete[] p;//new [] delete []
|
|
p= NULL;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
printf("[neutrino] event - read failed!\n");
|
|
}
|
|
|
|
::close(fd_eventclient);
|
|
|
|
if ( *msg != RC_nokey )
|
|
{
|
|
// raus hier :)
|
|
//printf("[neutrino] event 0x%x\n", *msg);
|
|
return;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++) {
|
|
if ((fd_rc[i] != -1) && (FD_ISSET(fd_rc[i], &rfds))) {
|
|
int ret;
|
|
ret = read(fd_rc[i], &ev, sizeof(t_input_event));
|
|
|
|
if(ret != sizeof(t_input_event))
|
|
continue;
|
|
SHTDCNT::getInstance()->resetSleepTimer();
|
|
uint32_t trkey = translate(ev.code, i);
|
|
#ifdef DEBUG
|
|
printf("key: %04x value %d, translate: %04x -%s-\n", ev.code, ev.value, trkey, getKeyName(trkey).c_str());
|
|
#endif
|
|
if (trkey == RC_nokey)
|
|
continue;
|
|
if (ev.value) {
|
|
#ifdef RCDEBUG
|
|
printf("got keydown native key: %04x %04x, translate: %04x -%s-\n", ev.code, ev.code&0x1f, translate(ev.code, 0), getKeyName(translate(ev.code, 0)).c_str());
|
|
printf("rc_last_key %04x rc_last_repeat_key %04x\n\n", rc_last_key, rc_last_repeat_key);
|
|
#endif
|
|
uint64_t now_pressed;
|
|
bool keyok = true;
|
|
|
|
tv = ev.time;
|
|
now_pressed = (uint64_t) tv.tv_usec + (uint64_t)((uint64_t) tv.tv_sec * (uint64_t) 1000000);
|
|
if (ev.code == rc_last_key) {
|
|
/* only allow selected keys to be repeated */
|
|
/* (why?) */
|
|
if( (trkey == RC_up) || (trkey == RC_down ) ||
|
|
(trkey == RC_plus ) || (trkey == RC_minus ) ||
|
|
(trkey == RC_page_down ) || (trkey == RC_page_up ) ||
|
|
((bAllowRepeatLR) && ((trkey == RC_left ) || (trkey == RC_right))) ||
|
|
(g_settings.shutdown_real_rcdelay && ((trkey == RC_standby) && (cs_get_revision() > 7))) )
|
|
{
|
|
#ifdef ENABLE_REPEAT_CHECK
|
|
if (rc_last_repeat_key != ev.code) {
|
|
if ((now_pressed > last_keypress + repeat_block) ||
|
|
/* accept all keys after time discontinuity: */
|
|
(now_pressed < last_keypress))
|
|
rc_last_repeat_key = ev.code;
|
|
else
|
|
keyok = false;
|
|
}
|
|
#endif
|
|
}
|
|
else
|
|
keyok = false;
|
|
}
|
|
else
|
|
rc_last_repeat_key = KEY_MAX;
|
|
|
|
rc_last_key = ev.code;
|
|
|
|
if (keyok) {
|
|
#ifdef ENABLE_REPEAT_CHECK
|
|
if ((now_pressed > last_keypress + repeat_block_generic) ||
|
|
/* accept all keys after time discontinuity: */
|
|
(now_pressed < last_keypress))
|
|
#endif
|
|
{
|
|
last_keypress = now_pressed;
|
|
|
|
*msg = trkey;
|
|
*data = 0; /* <- button pressed */
|
|
if(g_settings.key_click)
|
|
play_click();
|
|
return;
|
|
}
|
|
} /*if keyok */
|
|
} /* if (ev.value) */
|
|
else {
|
|
// clear rc_last_key on keyup event
|
|
#ifdef RCDEBUG
|
|
printf("got keyup native key: %04x %04x, translate: %04x -%s-\n", ev.code, ev.code&0x1f, translate(ev.code, 0), getKeyName(translate(ev.code, 0)).c_str() );
|
|
#endif
|
|
rc_last_key = KEY_MAX;
|
|
if (trkey == RC_standby) {
|
|
*msg = RC_standby;
|
|
*data = 1; /* <- button released */
|
|
return;
|
|
}
|
|
}
|
|
}/* if FDSET */
|
|
} /* for NUMBER_OF_EVENT_DEVICES */
|
|
|
|
if(FD_ISSET(fd_pipe_low_priority[0], &rfds))
|
|
{
|
|
struct event buf;
|
|
|
|
read(fd_pipe_low_priority[0], &buf, sizeof(buf));
|
|
|
|
*msg = buf.msg;
|
|
*data = buf.data;
|
|
|
|
// printf("got event from low-pri pipe %x %x\n", *msg, *data );
|
|
|
|
return;
|
|
}
|
|
|
|
if ( InitialTimeout == 0 )
|
|
{
|
|
//nicht warten wenn kein key da ist
|
|
*msg = RC_timeout;
|
|
*data = 0;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
//timeout neu kalkulieren
|
|
#ifdef USE_GETTIMEOFDAY
|
|
gettimeofday( &tv, NULL );
|
|
int64_t getKeyNow = (int64_t) tv.tv_usec + (int64_t)((int64_t) tv.tv_sec * (int64_t) 1000000);
|
|
#else
|
|
int64_t getKeyNow = time_monotonic_us();
|
|
#endif
|
|
int64_t diff = (getKeyNow - getKeyBegin);
|
|
if( Timeout <= (uint64_t) diff )
|
|
{
|
|
*msg = RC_timeout;
|
|
*data = 0;
|
|
return;
|
|
}
|
|
else
|
|
Timeout -= diff;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CRCInput::postMsg(const neutrino_msg_t msg, const neutrino_msg_data_t data, const bool Priority)
|
|
{
|
|
// printf("postMsg %x %x %d\n", msg, data, Priority );
|
|
|
|
struct event buf;
|
|
buf.msg = msg;
|
|
buf.data = data;
|
|
|
|
if (Priority)
|
|
write(fd_pipe_high_priority[1], &buf, sizeof(buf));
|
|
else
|
|
write(fd_pipe_low_priority[1], &buf, sizeof(buf));
|
|
}
|
|
|
|
|
|
void CRCInput::clearRCMsg()
|
|
{
|
|
t_input_event ev;
|
|
|
|
for (int i = 0; i < NUMBER_OF_EVENT_DEVICES; i++)
|
|
{
|
|
if (fd_rc[i] != -1)
|
|
{
|
|
while (read(fd_rc[i], &ev, sizeof(t_input_event)) == sizeof(t_input_event))
|
|
;
|
|
}
|
|
}
|
|
rc_last_key = KEY_MAX;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* isNumeric - test if key is 0..9
|
|
*
|
|
**************************************************************************/
|
|
bool CRCInput::isNumeric(const neutrino_msg_t key)
|
|
{
|
|
return ((key == RC_0) || ((key >= RC_1) && (key <= RC_9)));
|
|
}
|
|
|
|
/**************************************************************************
|
|
* getNumericValue - return numeric value of the key or -1
|
|
*
|
|
**************************************************************************/
|
|
int CRCInput::getNumericValue(const neutrino_msg_t key)
|
|
{
|
|
return ((key == RC_0) ? (int)0 : (((key >= RC_1) && (key <= RC_9)) ? (int)(key - RC_1 + 1) : (int)-1));
|
|
}
|
|
|
|
/**************************************************************************
|
|
* convertDigitToKey - return key representing digit or RC_nokey
|
|
*
|
|
**************************************************************************/
|
|
static const unsigned int digit_to_key[10] = {CRCInput::RC_0, CRCInput::RC_1, CRCInput::RC_2, CRCInput::RC_3, CRCInput::RC_4, CRCInput::RC_5, CRCInput::RC_6, CRCInput::RC_7, CRCInput::RC_8, CRCInput::RC_9};
|
|
|
|
unsigned int CRCInput::convertDigitToKey(const unsigned int digit)
|
|
{
|
|
return (digit < 10) ? digit_to_key[digit] : RC_nokey;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* getUnicodeValue - return unicode value of the key or -1
|
|
*
|
|
**************************************************************************/
|
|
#define UNICODE_VALUE_SIZE 58
|
|
static const int unicode_value[UNICODE_VALUE_SIZE] = {-1 , -1 , '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', -1 , -1 ,
|
|
'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', -1 , -1 , 'A', 'S',
|
|
'D', 'F', 'G', 'H', 'J', 'K', 'L', ';', -1 /* FIXME */, -1 /* FIXME */, -1 , '\\', 'Z', 'X', 'C', 'V',
|
|
'B', 'N', 'M', ',', '.', '/', -1, -1, -1, ' '};
|
|
|
|
int CRCInput::getUnicodeValue(const neutrino_msg_t key)
|
|
{
|
|
if (key < UNICODE_VALUE_SIZE)
|
|
return unicode_value[key];
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* transforms the rc-key to const char *
|
|
*
|
|
**************************************************************************/
|
|
const char * CRCInput::getSpecialKeyName(const unsigned int key)
|
|
{
|
|
switch(key)
|
|
{
|
|
case RC_standby:
|
|
return "standby";
|
|
case RC_home:
|
|
return "home";
|
|
case RC_setup:
|
|
return "setup";
|
|
case RC_red:
|
|
return "red button";
|
|
case RC_green:
|
|
return "green button";
|
|
case RC_yellow:
|
|
return "yellow button";
|
|
case RC_blue:
|
|
return "blue button";
|
|
case RC_page_up:
|
|
return "page up";
|
|
case RC_page_down:
|
|
return "page down";
|
|
case RC_up:
|
|
return "cursor up";
|
|
case RC_down:
|
|
return "cursor down";
|
|
case RC_left:
|
|
return "cursor left";
|
|
case RC_right:
|
|
return "cursor right";
|
|
case RC_ok:
|
|
return "ok";
|
|
case RC_plus:
|
|
return "vol. inc";
|
|
case RC_minus:
|
|
return "vol. dec";
|
|
case RC_spkr:
|
|
return "mute";
|
|
case RC_help:
|
|
return "help";
|
|
case RC_info:
|
|
return "info";
|
|
case RC_topleft:
|
|
return "topleft";
|
|
case RC_topright:
|
|
return "topright";
|
|
case RC_audio:
|
|
return "audio";
|
|
case RC_video:
|
|
return "video";
|
|
case RC_tv:
|
|
return "tv";
|
|
case RC_radio:
|
|
return "radio";
|
|
case RC_text:
|
|
return "text";
|
|
#if 0
|
|
case RC_shift_red:
|
|
return "shift-red";
|
|
case RC_shift_green:
|
|
return "shift-green";
|
|
case RC_shift_yellow:
|
|
return "shift-yellow";
|
|
case RC_shift_blue:
|
|
return "shift-blue";
|
|
case RC_shift_tv:
|
|
return "shift-tv";
|
|
case RC_shift_radio:
|
|
return "shift-radio";
|
|
#endif
|
|
case RC_epg:
|
|
return "epg";
|
|
case RC_recall:
|
|
return "recall";
|
|
case RC_favorites:
|
|
return "favorites";
|
|
case RC_sat:
|
|
return "sat";
|
|
case RC_sat2:
|
|
return "sat2";
|
|
case RC_timeout:
|
|
return "timeout";
|
|
case RC_play:
|
|
return "play";
|
|
case RC_stop:
|
|
return "stop";
|
|
case RC_forward:
|
|
return "forward";
|
|
case RC_rewind:
|
|
return "rewind";
|
|
case RC_timeshift:
|
|
return "timeshift";
|
|
case RC_mode:
|
|
return "mode";
|
|
case RC_record:
|
|
return "record";
|
|
case RC_pause:
|
|
return "pause";
|
|
case RC_games:
|
|
return "games";
|
|
case RC_next:
|
|
return "next";
|
|
case RC_prev:
|
|
return "prev";
|
|
case RC_nokey:
|
|
return "none";
|
|
case RC_power_on:
|
|
return "power on";
|
|
case RC_power_off:
|
|
return "power off";
|
|
case RC_standby_on:
|
|
return "standby on";
|
|
case RC_standby_off:
|
|
return "standby off";
|
|
case RC_mute_on:
|
|
return "mute on";
|
|
case RC_mute_off:
|
|
return "mute off";
|
|
case RC_analog_on:
|
|
return "analog on";
|
|
case RC_analog_off:
|
|
return "analog off";
|
|
case RC_www:
|
|
return "window print";
|
|
default:
|
|
printf("unknown key: %d (0x%x) \n", key, key);
|
|
return "unknown";
|
|
}
|
|
}
|
|
|
|
std::string CRCInput::getKeyName(const unsigned int key)
|
|
{
|
|
int lunicode_value = getUnicodeValue(key);
|
|
if (lunicode_value == -1)
|
|
return getSpecialKeyName(key);
|
|
else
|
|
{
|
|
char tmp[2];
|
|
tmp[0] = lunicode_value;
|
|
tmp[1] = 0;
|
|
return std::string(tmp);
|
|
}
|
|
}
|
|
|
|
/**************************************************************************
|
|
* transforms the rc-key to generic - internal use only!
|
|
*
|
|
**************************************************************************/
|
|
int CRCInput::translate(int code, int /*num*/)
|
|
{
|
|
if(code == 0x100) code = RC_up;
|
|
else if(code == 0x101) code = RC_down;
|
|
if ((code >= 0) && (code <= KEY_MAX))
|
|
return code;
|
|
else
|
|
return RC_nokey;
|
|
}
|
|
|
|
void CRCInput::close_click()
|
|
{
|
|
}
|
|
|
|
void CRCInput::open_click()
|
|
{
|
|
}
|
|
#if 0
|
|
//never used
|
|
void CRCInput::reset_dsp(int /*rate*/)
|
|
{
|
|
}
|
|
|
|
void CRCInput::set_dsp()
|
|
{
|
|
}
|
|
#endif
|
|
void CRCInput::play_click()
|
|
{
|
|
}
|
|
|
|
|
|
#ifdef IOC_IR_SET_PRI_PROTOCOL
|
|
// hint: ir_protocol_t and other useful things are defined in cs_ir_generic.h
|
|
void CRCInput::set_rc_hw(ir_protocol_t ir_protocol, unsigned int ir_address)
|
|
{
|
|
int ioctl_ret = -1;
|
|
|
|
//fixme?: for now fd_rc[] is hardcoded to 0 since only fd_rc[0] is used at the moment
|
|
ioctl_ret = ::ioctl(fd_rc[0], IOC_IR_SET_PRI_PROTOCOL, ir_protocol);
|
|
if(ioctl_ret < 0)
|
|
perror("IOC_IR_SET_PRI_PROTOCOL");
|
|
else
|
|
printf("CRCInput::set_rc_hw: Set IOCTRL : IOC_IR_SET_PRI_PROTOCOL, %05X\n", ir_protocol);
|
|
|
|
//bypass setting of IR Address with ir_address=0
|
|
if(ir_address > 0)
|
|
{
|
|
//fixme?: for now fd_rc[] is hardcoded to 0 since only fd_rc[0] is used at the moment
|
|
ioctl_ret = ::ioctl(fd_rc[0], IOC_IR_SET_PRI_ADDRESS, ir_address);
|
|
if(ioctl_ret < 0)
|
|
perror("IOC_IR_SET_PRI_ADDRESS");
|
|
else
|
|
printf("CRCInput::set_rc_hw: Set IOCTRL : IOC_IR_SET_PRI_ADDRESS, %05X\n", ir_address);
|
|
}
|
|
}
|
|
|
|
// hint: ir_protocol_t and other useful things are defined in cs_ir_generic.h
|
|
void CRCInput::set_rc_hw(void)
|
|
{
|
|
ir_protocol_t ir_protocol = IR_PROTOCOL_UNKNOWN;
|
|
unsigned int ir_address = 0x00;
|
|
|
|
switch(g_settings.remote_control_hardware)
|
|
{
|
|
case RC_HW_COOLSTREAM:
|
|
ir_protocol = IR_PROTOCOL_NECE;
|
|
ir_address = 0xFF80;
|
|
break;
|
|
case RC_HW_DBOX:
|
|
ir_protocol = IR_PROTOCOL_NRC17;
|
|
ir_address = 0x00C5;
|
|
break;
|
|
case RC_HW_PHILIPS:
|
|
ir_protocol = IR_PROTOCOL_RC5;
|
|
ir_address = 0x000A;
|
|
break;
|
|
case RC_HW_TRIPLEDRAGON:
|
|
ir_protocol = IR_PROTOCOL_RMAP_E;
|
|
ir_address = 0x000A; // with device id 0
|
|
// ir_address = 0x100A; // with device id 1
|
|
// ir_address = 0x200A; // with device id 2
|
|
// ir_address = 0x300A; // with device id 3
|
|
// ir_address = 0x400A; // with device id 4
|
|
// ir_address = 0x500A; // with device id 5
|
|
// ir_address = 0x600A; // with device id 6
|
|
// ir_address = 0x700A; // with device id 7
|
|
break;
|
|
default:
|
|
ir_protocol = IR_PROTOCOL_NECE;
|
|
ir_address = 0xFF80;
|
|
}
|
|
|
|
set_rc_hw(ir_protocol, ir_address);
|
|
}
|
|
#else
|
|
void CRCInput::set_rc_hw(void)
|
|
{
|
|
}
|
|
#endif
|