mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Remote control data is now received in a separate thread, which makes things a lot smoother. - Repeat and release of remote control keys is now explicitly distinguished. - In replay mode the search forward/back and skip functions now have two modes: Pressing the key shortly and releasing it starts the function, and pressing it again stops it. Pressing and holding down the key starts the function and releasing the key stops it. - The '@' character that marks an "instant recording" can now be turned off in the "Setup" menu (thanks to Matthias Schniedermeyer). - Pressing the "Back" button while replaying now stops replaying and brings up the "Recordings" menu (suggested by Carsten Koch). This can be used to easily delete a recording after watching it, or to switch to a different recording. - The "Recordings" menu now places the cursor on the last replayed recording, if that file still exists. - The "Blue" button in the "Main" menu can now be used to "Resume" a previously stopped replay session (suggested by Martin Hammerschmid). - The low and high LNB frequencies can now be changed in the "Setup" menu.
494 lines
12 KiB
C
494 lines
12 KiB
C
/*
|
|
* remote.c: Interface to the Remote Control Unit
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* Ported to LIRC by Carsten Koch <Carsten.Koch@icem.de> 2000-06-16.
|
|
*
|
|
* $Id: remote.c 1.18 2000/10/08 16:49:41 kls Exp $
|
|
*/
|
|
|
|
#include "remote.h"
|
|
#include <fcntl.h>
|
|
#include <netinet/in.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <sys/time.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
|
|
#if defined REMOTE_LIRC
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/un.h>
|
|
#endif
|
|
|
|
#include "config.h"
|
|
#include "tools.h"
|
|
|
|
// --- cRcIoBase -------------------------------------------------------------
|
|
|
|
cRcIoBase::cRcIoBase(void)
|
|
{
|
|
t = 0;
|
|
}
|
|
|
|
cRcIoBase::~cRcIoBase()
|
|
{
|
|
}
|
|
|
|
// --- cRcIoKBD --------------------------------------------------------------
|
|
|
|
#if defined REMOTE_KBD
|
|
|
|
cRcIoKBD::cRcIoKBD(void)
|
|
{
|
|
f.Open(0); // stdin
|
|
}
|
|
|
|
cRcIoKBD::~cRcIoKBD()
|
|
{
|
|
}
|
|
|
|
void cRcIoKBD::Flush(int WaitMs)
|
|
{
|
|
int t0 = time_ms();
|
|
|
|
timeout(10);
|
|
for (;;) {
|
|
while (getch() > 0)
|
|
t0 = time_ms();
|
|
if (time_ms() - t0 >= WaitMs)
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool cRcIoKBD::InputAvailable(void)
|
|
{
|
|
return f.Ready(false);
|
|
}
|
|
|
|
bool cRcIoKBD::GetCommand(unsigned int *Command, bool *Repeat, bool *Release)
|
|
{
|
|
if (Command) {
|
|
*Command = getch();
|
|
return *Command > 0;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// --- cRcIoRCU --------------------------------------------------------------
|
|
|
|
#elif defined REMOTE_RCU
|
|
|
|
#define REPEATLIMIT 20 // ms
|
|
#define REPEATDELAY 350 // ms
|
|
|
|
cRcIoRCU::cRcIoRCU(char *DeviceName)
|
|
{
|
|
dp = 0;
|
|
mode = modeB;
|
|
code = 0;
|
|
address = 0xFFFF;
|
|
receivedAddress = 0;
|
|
receivedCommand = 0;
|
|
receivedData = receivedRepeat = receivedRelease = false;
|
|
lastNumber = 0;
|
|
if ((f = open(DeviceName, O_RDWR | O_NONBLOCK)) >= 0) {
|
|
struct termios t;
|
|
if (tcgetattr(f, &t) == 0) {
|
|
cfsetspeed(&t, B9600);
|
|
cfmakeraw(&t);
|
|
if (tcsetattr(f, TCSAFLUSH, &t) == 0) {
|
|
Start();
|
|
return;
|
|
}
|
|
}
|
|
LOG_ERROR_STR(DeviceName);
|
|
close(f);
|
|
}
|
|
else
|
|
LOG_ERROR_STR(DeviceName);
|
|
f = -1;
|
|
}
|
|
|
|
cRcIoRCU::~cRcIoRCU()
|
|
{
|
|
Stop();
|
|
}
|
|
|
|
void cRcIoRCU::Action(void)
|
|
{
|
|
#pragma pack(1)
|
|
union {
|
|
struct {
|
|
unsigned short address;
|
|
unsigned int command;
|
|
} data;
|
|
unsigned char raw[6];
|
|
} buffer;
|
|
#pragma pack()
|
|
|
|
dsyslog(LOG_INFO, "RCU remote control thread started (pid=%d)", getpid());
|
|
|
|
int FirstTime = 0;
|
|
unsigned int LastCommand = 0;
|
|
|
|
for (; f >= 0;) {
|
|
|
|
LOCK_THREAD;
|
|
|
|
if (ReceiveByte(REPEATLIMIT) == 'X') {
|
|
for (int i = 0; i < 6; i++) {
|
|
int b = ReceiveByte();
|
|
if (b >= 0) {
|
|
buffer.raw[i] = b;
|
|
if (i == 5) {
|
|
unsigned short Address = ntohs(buffer.data.address); // the PIC sends bytes in "network order"
|
|
unsigned int Command = ntohl(buffer.data.command);
|
|
if (code == 'B' && address == 0x0000 && Command == 0x00004000)
|
|
// Well, well, if it isn't the "d-box"...
|
|
// This remote control sends the above command before and after
|
|
// each keypress - let's just drop this:
|
|
break;
|
|
if (!receivedData) { // only accept new data the previous data has been fetched
|
|
int Now = time_ms();
|
|
if (Command != LastCommand) {
|
|
receivedAddress = Address;
|
|
receivedCommand = Command;
|
|
receivedData = true;
|
|
receivedRepeat = receivedRelease = false;
|
|
FirstTime = Now;
|
|
}
|
|
else {
|
|
if (Now - FirstTime < REPEATDELAY)
|
|
break; // repeat function kicks in after a short delay
|
|
receivedData = receivedRepeat = true;
|
|
}
|
|
LastCommand = Command;
|
|
WakeUp();
|
|
}
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
else if (receivedData) { // the last data before releasing the key hasn't been fetched yet
|
|
if (receivedRepeat) { // it was a repeat, so let's make it a release
|
|
receivedRepeat = false;
|
|
receivedRelease = true;
|
|
LastCommand = 0;
|
|
WakeUp();
|
|
}
|
|
}
|
|
else if (receivedRepeat) { // all data has already been fetched, but the last one was a repeat, so let's generate a release
|
|
receivedData = receivedRelease = true;
|
|
receivedRepeat = false;
|
|
LastCommand = 0;
|
|
WakeUp();
|
|
}
|
|
else
|
|
LastCommand = 0;
|
|
}
|
|
}
|
|
|
|
int cRcIoRCU::ReceiveByte(int TimeoutMs)
|
|
{
|
|
// Returns the byte if one was received within a timeout, -1 otherwise
|
|
if (cFile::FileReady(f, TimeoutMs)) {
|
|
unsigned char b;
|
|
if (read(f, &b, 1) == 1)
|
|
return b;
|
|
else
|
|
LOG_ERROR;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
bool cRcIoRCU::SendByteHandshake(unsigned char c)
|
|
{
|
|
if (f >= 0) {
|
|
int w = write(f, &c, 1);
|
|
if (w == 1) {
|
|
for (int reply = ReceiveByte(REPEATLIMIT); reply >= 0;) {
|
|
if (reply == c)
|
|
return true;
|
|
else if (reply == 'X') {
|
|
// skip any incoming RC code - it will come again
|
|
for (int i = 6; i--;) {
|
|
if (ReceiveByte() < 0)
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
}
|
|
LOG_ERROR;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cRcIoRCU::SendByte(unsigned char c)
|
|
{
|
|
LOCK_THREAD;
|
|
|
|
for (int retry = 5; retry--;) {
|
|
if (SendByteHandshake(c))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cRcIoRCU::SetCode(unsigned char Code, unsigned short Address)
|
|
{
|
|
code = Code;
|
|
address = Address;
|
|
return SendCommand(code);
|
|
}
|
|
|
|
bool cRcIoRCU::SetMode(unsigned char Mode)
|
|
{
|
|
mode = Mode;
|
|
return SendCommand(mode);
|
|
}
|
|
|
|
void cRcIoRCU::Flush(int WaitMs)
|
|
{
|
|
LOCK_THREAD;
|
|
|
|
int t0 = time_ms();
|
|
for (;;) {
|
|
while (ReceiveByte() >= 0)
|
|
t0 = time_ms();
|
|
if (time_ms() - t0 >= WaitMs)
|
|
break;
|
|
}
|
|
receivedData = receivedRepeat = false;
|
|
}
|
|
|
|
bool cRcIoRCU::GetCommand(unsigned int *Command, bool *Repeat, bool *Release)
|
|
{
|
|
if (receivedData) { // first we check the boolean flag without a lock, to avoid delays
|
|
|
|
LOCK_THREAD;
|
|
|
|
if (receivedData) { // need to check again, since the status might have changed while waiting for the lock
|
|
if (Command)
|
|
*Command = receivedCommand;
|
|
if (Repeat)
|
|
*Repeat = receivedRepeat;
|
|
if (Release)
|
|
*Release = receivedRelease;
|
|
receivedData = false;
|
|
return true;
|
|
}
|
|
}
|
|
if (time(NULL) - t > 60) {
|
|
SendCommand(code); // in case the PIC listens to the wrong code
|
|
t = time(NULL);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cRcIoRCU::SendCommand(unsigned char Cmd)
|
|
{
|
|
return SendByte(Cmd | 0x80);
|
|
}
|
|
|
|
bool cRcIoRCU::Digit(int n, int v)
|
|
{
|
|
return SendByte(((n & 0x03) << 5) | (v & 0x0F) | (((dp >> n) & 0x01) << 4));
|
|
}
|
|
|
|
bool cRcIoRCU::Number(int n, bool Hex)
|
|
{
|
|
LOCK_THREAD;
|
|
|
|
if (!Hex) {
|
|
char buf[8];
|
|
sprintf(buf, "%4d", n & 0xFFFF);
|
|
n = 0;
|
|
for (char *d = buf; *d; d++) {
|
|
if (*d == ' ')
|
|
*d = 0xF;
|
|
n = (n << 4) | ((*d - '0') & 0x0F);
|
|
}
|
|
}
|
|
lastNumber = n;
|
|
for (int i = 0; i < 4; i++) {
|
|
if (!Digit(i, n))
|
|
return false;
|
|
n >>= 4;
|
|
}
|
|
return SendCommand(mode);
|
|
}
|
|
|
|
bool cRcIoRCU::String(char *s)
|
|
{
|
|
LOCK_THREAD;
|
|
|
|
const char *chars = mode == modeH ? "0123456789ABCDEF" : "0123456789-EHLP ";
|
|
int n = 0;
|
|
|
|
for (int i = 0; *s && i < 4; s++, i++) {
|
|
n <<= 4;
|
|
for (const char *c = chars; *c; c++) {
|
|
if (*c == *s) {
|
|
n |= c - chars;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return Number(n, true);
|
|
}
|
|
|
|
void cRcIoRCU::SetPoints(unsigned char Dp, bool On)
|
|
{
|
|
if (On)
|
|
dp |= Dp;
|
|
else
|
|
dp &= ~Dp;
|
|
Number(lastNumber, true);
|
|
}
|
|
|
|
bool cRcIoRCU::DetectCode(unsigned char *Code, unsigned short *Address)
|
|
{
|
|
// Caller should initialize 'Code' to 0 and call DetectCode()
|
|
// until it returns true. Whenever DetectCode() returns false
|
|
// and 'Code' is not 0, the caller can use 'Code' to display
|
|
// a message like "Trying code '%c'". If false is returned and
|
|
// 'Code' is 0, all possible codes have been tried and the caller
|
|
// can either stop calling DetectCode() (and give some error
|
|
// message), or start all over again.
|
|
if (*Code < 'A' || *Code > 'D') {
|
|
*Code = 'A';
|
|
return false;
|
|
}
|
|
if (*Code <= 'D') {
|
|
SetMode(modeH);
|
|
char buf[5];
|
|
sprintf(buf, "C0D%c", *Code);
|
|
String(buf);
|
|
SetCode(*Code, 0);
|
|
delay_ms(REPEATDELAY);
|
|
receivedData = receivedRepeat = 0;
|
|
delay_ms(REPEATDELAY);
|
|
if (GetCommand()) {
|
|
*Address = receivedAddress;
|
|
SetMode(modeB);
|
|
String("----");
|
|
return true;
|
|
}
|
|
if (*Code < 'D') {
|
|
(*Code)++;
|
|
return false;
|
|
}
|
|
}
|
|
*Code = 0;
|
|
return false;
|
|
}
|
|
|
|
// --- cRcIoLIRC -------------------------------------------------------------
|
|
|
|
#elif defined REMOTE_LIRC
|
|
|
|
#define REPEATLIMIT 20 // ms
|
|
#define REPEATDELAY 350 // ms
|
|
|
|
cRcIoLIRC::cRcIoLIRC(char *DeviceName)
|
|
{
|
|
*keyName = 0;
|
|
receivedData = receivedRepeat = false;
|
|
struct sockaddr_un addr;
|
|
addr.sun_family = AF_UNIX;
|
|
strcpy(addr.sun_path, DeviceName);
|
|
if ((f = socket(AF_UNIX, SOCK_STREAM, 0)) >= 0) {
|
|
if (connect(f, (struct sockaddr *)&addr, sizeof(addr)) >= 0) {
|
|
Start();
|
|
return;
|
|
}
|
|
LOG_ERROR_STR(DeviceName);
|
|
close(f);
|
|
}
|
|
else
|
|
LOG_ERROR_STR(DeviceName);
|
|
f = -1;
|
|
}
|
|
|
|
cRcIoLIRC::~cRcIoLIRC()
|
|
{
|
|
Stop();
|
|
}
|
|
|
|
void cRcIoLIRC::Action(void)
|
|
{
|
|
dsyslog(LOG_INFO, "LIRC remote control thread started (pid=%d)", getpid());
|
|
|
|
int FirstTime = 0;
|
|
char buf[LIRC_BUFFER_SIZE];
|
|
char LastKeyName[LIRC_KEY_BUF];
|
|
|
|
for (; f >= 0;) {
|
|
|
|
LOCK_THREAD;
|
|
|
|
if (cFile::FileReady(f, REPEATLIMIT) && read(f, buf, sizeof(buf)) > 21) {
|
|
if (!receivedData) { // only accept new data the previous data has been fetched
|
|
int count;
|
|
sscanf(buf, "%*x %x %7s", &count, LastKeyName); // '7' in '%7s' is LIRC_KEY_BUF-1!
|
|
int Now = time_ms();
|
|
if (count == 0) {
|
|
strcpy(keyName, LastKeyName);
|
|
receivedData = true;
|
|
receivedRepeat = receivedRelease = false;
|
|
FirstTime = Now;
|
|
}
|
|
else {
|
|
if (Now - FirstTime < REPEATDELAY)
|
|
continue; // repeat function kicks in after a short delay
|
|
receivedData = receivedRepeat = true;
|
|
}
|
|
WakeUp();
|
|
}
|
|
}
|
|
else if (receivedData) { // the last data before releasing the key hasn't been fetched yet
|
|
if (receivedRepeat) { // it was a repeat, so let's make it a release
|
|
receivedRepeat = false;
|
|
receivedRelease = true;
|
|
WakeUp();
|
|
}
|
|
}
|
|
else if (receivedRepeat) { // all data has already been fetched, but the last one was a repeat, so let's generate a release
|
|
receivedData = receivedRelease = true;
|
|
receivedRepeat = false;
|
|
WakeUp();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool cRcIoLIRC::GetCommand(unsigned int *Command, bool *Repeat, bool *Release)
|
|
{
|
|
if (receivedData) { // first we check the boolean flag without a lock, to avoid delays
|
|
|
|
LOCK_THREAD;
|
|
|
|
if (receivedData) { // need to check again, since the status might have changed while waiting for the lock
|
|
if (Command)
|
|
*Command = Keys.Encode(keyName);
|
|
if (Repeat)
|
|
*Repeat = receivedRepeat;
|
|
if (Release)
|
|
*Release = receivedRelease;
|
|
receivedData = false;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
#endif
|
|
|