Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 192897
b: refs/heads/master
c: 9f15478
h: refs/heads/master
i:
  192895: 96171b0
v: v3
  • Loading branch information
Mauro Carvalho Chehab committed May 18, 2010
1 parent bf53430 commit bb73534
Show file tree
Hide file tree
Showing 6 changed files with 118 additions and 66 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: ada39630c758c5c3098f4fc1361103ea2bc1afe0
refs/heads/master: 9f1547829a6f39fe6b2da22653dff40502f3d568
1 change: 1 addition & 0 deletions trunk/drivers/media/IR/ir-keytable.c
Original file line number Diff line number Diff line change
Expand Up @@ -405,6 +405,7 @@ void ir_keyup(struct input_dev *dev)
if (!ir->keypressed)
return;

IR_dprintk(1, "keyup key 0x%04x\n", ir->keycode);
input_report_key(dev, ir->keycode, 0);
input_sync(dev);
ir->keypressed = 0;
Expand Down
122 changes: 83 additions & 39 deletions trunk/drivers/media/IR/ir-nec-decoder.c
Original file line number Diff line number Diff line change
Expand Up @@ -14,71 +14,111 @@

#include <media/ir-core.h>

/* Start time: 4.5 ms */
#define MIN_START_TIME 3900000
#define MAX_START_TIME 5100000
/* Start time: 4.5 ms + 560 us of the next pulse */
#define MIN_START_TIME (3900000 + 560000)
#define MAX_START_TIME (5100000 + 560000)

/* Pulse time: 560 us */
#define MIN_PULSE_TIME 460000
#define MAX_PULSE_TIME 660000
/* Bit 1 time: 2.25ms us */
#define MIN_BIT1_TIME 2050000
#define MAX_BIT1_TIME 2450000

/* Bit 1 space time: 2.25ms-560 us */
#define MIN_BIT1_TIME 1490000
#define MAX_BIT1_TIME 1890000
/* Bit 0 time: 1.12ms us */
#define MIN_BIT0_TIME 920000
#define MAX_BIT0_TIME 1320000

/* Bit 0 space time: 1.12ms-560 us */
#define MIN_BIT0_TIME 360000
#define MAX_BIT0_TIME 760000
/* Total IR code is 110 ms, including the 9 ms for the start pulse */
#define MAX_NEC_TIME 4000000

/* Total IR code is 110 ms, including the 9 ms for the start pulse */
#define MIN_REPEAT_TIME 99000000
#define MAX_REPEAT_TIME 112000000

/* Repeat time: 2.25ms us */
#define MIN_REPEAT_START_TIME 2050000
#define MAX_REPEAT_START_TIME 3000000

#define REPEAT_TIME 240 /* ms */

/** is_repeat - Check if it is a NEC repeat event
* @input_dev: the struct input_dev descriptor of the device
* @pos: the position of the first event
* @len: the length of the buffer
*/
static int is_repeat(struct ir_raw_event *evs, int len, int pos)
{
if ((evs[pos].delta.tv_nsec < MIN_REPEAT_START_TIME) ||
(evs[pos].delta.tv_nsec > MAX_REPEAT_START_TIME))
return 0;

if (++pos >= len)
return 0;

if ((evs[pos].delta.tv_nsec < MIN_REPEAT_TIME) ||
(evs[pos].delta.tv_nsec > MAX_REPEAT_TIME))
return 0;

return 1;
}

/**
* __ir_nec_decode() - Decode one NEC pulsecode
* @input_dev: the struct input_dev descriptor of the device
* @evs: event array with type/duration of pulse/space
* @len: length of the array
* @pos: position to start seeking for a code
* This function returns the decoded ircode or -EINVAL if no pulse got decoded
* This function returns -EINVAL if no pulse got decoded,
* 0 if buffer is empty and 1 if one keycode were handled.
*/
static int __ir_nec_decode(struct input_dev *input_dev,
struct ir_raw_event *evs,
int len, int *pos)
{
struct ir_input_dev *ir = input_get_drvdata(input_dev);
int count = -1;
int ircode = 0, not_code = 0;

/* Be sure that the first event is an start one and is a pulse */
for (; *pos < len; (*pos)++) {
if (evs[*pos].type & (IR_START_EVENT | IR_PULSE))
/* Very long delays are considered as start events */
if (evs[*pos].delta.tv_nsec > MAX_NEC_TIME)
break;
}
(*pos)++; /* First event doesn't contain data */
if (evs[*pos].type & IR_START_EVENT)
break;
IR_dprintk(1, "%luus: Spurious NEC %s\n",
(evs[*pos].delta.tv_nsec + 500) / 1000,
(evs[*pos].type & IR_SPACE) ? "space" : "pulse");

}
if (*pos >= len)
return 0;

/* First space should have 4.5 ms otherwise is not NEC protocol */
if ((evs[*pos].delta.tv_nsec < MIN_START_TIME) |
(evs[*pos].delta.tv_nsec > MAX_START_TIME) |
(evs[*pos].type != IR_SPACE))
(*pos)++; /* First event doesn't contain data */

if (evs[*pos].type != IR_PULSE)
goto err;

/*
* FIXME: need to implement the repeat sequence
*/
/* Check if it is a NEC repeat event */
if (is_repeat(evs, len, *pos)) {
*pos += 2;
if (ir->keypressed) {
mod_timer(&ir->raw->timer_keyup,
jiffies + msecs_to_jiffies(REPEAT_TIME));
IR_dprintk(1, "NEC repeat event\n");
return 1;
} else {
IR_dprintk(1, "missing NEC repeat event\n");
return 0;
}
}

/* First space should have 4.5 ms otherwise is not NEC protocol */
if ((evs[*pos].delta.tv_nsec < MIN_START_TIME) ||
(evs[*pos].delta.tv_nsec > MAX_START_TIME))
goto err;

count = 0;
for ((*pos)++; *pos < len; (*pos)++) {
int bit;

if ((evs[*pos].delta.tv_nsec < MIN_PULSE_TIME) |
(evs[*pos].delta.tv_nsec > MAX_PULSE_TIME) |
(evs[*pos].type != IR_PULSE))
goto err;

if (++*pos >= len)
goto err;
if (evs[*pos].type != IR_SPACE)
goto err;

if ((evs[*pos].delta.tv_nsec > MIN_BIT1_TIME) &&
(evs[*pos].delta.tv_nsec < MAX_BIT1_TIME))
bit = 1;
Expand Down Expand Up @@ -107,6 +147,7 @@ static int __ir_nec_decode(struct input_dev *input_dev,
if (++count == 32)
break;
}
*pos++;

/*
* Fixme: may need to accept Extended NEC protocol?
Expand All @@ -119,12 +160,15 @@ static int __ir_nec_decode(struct input_dev *input_dev,

IR_dprintk(1, "NEC scancode 0x%04x\n", ircode);
ir_keydown(input_dev, ircode);
ir_keyup(input_dev);
mod_timer(&ir->raw->timer_keyup,
jiffies + msecs_to_jiffies(REPEAT_TIME));

return ircode;
return 1;
err:
IR_dprintk(1, "NEC decoded failed at bit %d while decoding %luus time\n",
count, (evs[*pos].delta.tv_nsec + 500) / 1000);
IR_dprintk(1, "NEC decoded failed at bit %d (%s) while decoding %luus time\n",
count,
(evs[*pos].type & IR_SPACE) ? "space" : "pulse",
(evs[*pos].delta.tv_nsec + 500) / 1000);

return -EINVAL;
}
Expand All @@ -145,7 +189,7 @@ int ir_nec_decode(struct input_dev *input_dev,
int rc = 0;

while (pos < len) {
if (__ir_nec_decode(input_dev, evs, len, &pos) >= 0)
if (__ir_nec_decode(input_dev, evs, len, &pos) > 0)
rc++;
}

Expand Down
14 changes: 14 additions & 0 deletions trunk/drivers/media/IR/ir-raw-event.c
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,13 @@
/* Define the max number of bit transitions per IR keycode */
#define MAX_IR_EVENT_SIZE 256

static void ir_keyup_timer(unsigned long data)
{
struct input_dev *input_dev = (struct input_dev *)data;

ir_keyup(input_dev);
}

int ir_raw_event_register(struct input_dev *input_dev)
{
struct ir_input_dev *ir = input_get_drvdata(input_dev);
Expand All @@ -27,6 +34,11 @@ int ir_raw_event_register(struct input_dev *input_dev)
size = sizeof(struct ir_raw_event) * MAX_IR_EVENT_SIZE * 2;
size = roundup_pow_of_two(size);

init_timer(&ir->raw->timer_keyup);
ir->raw->timer_keyup.function = ir_keyup_timer;
ir->raw->timer_keyup.data = (unsigned long)input_dev;
set_bit(EV_REP, input_dev->evbit);

rc = kfifo_alloc(&ir->raw->kfifo, size, GFP_KERNEL);

return rc;
Expand All @@ -40,6 +52,8 @@ void ir_raw_event_unregister(struct input_dev *input_dev)
if (!ir->raw)
return;

del_timer_sync(&ir->raw->timer_keyup);

kfifo_free(&ir->raw->kfifo);
kfree(ir->raw);
ir->raw = NULL;
Expand Down
43 changes: 17 additions & 26 deletions trunk/drivers/media/video/saa7134/saa7134-input.c
Original file line number Diff line number Diff line change
Expand Up @@ -425,8 +425,11 @@ static void saa7134_input_timer(unsigned long data)
void ir_raw_decode_timer_end(unsigned long data)
{
struct saa7134_dev *dev = (struct saa7134_dev *)data;
struct card_ir *ir = dev->remote;

ir_raw_event_handle(dev->remote->dev);

ir->active = 0;
}

void saa7134_ir_start(struct saa7134_dev *dev, struct card_ir *ir)
Expand Down Expand Up @@ -462,6 +465,7 @@ void saa7134_ir_start(struct saa7134_dev *dev, struct card_ir *ir)
init_timer(&ir->timer_end);
ir->timer_end.function = ir_raw_decode_timer_end;
ir->timer_end.data = (unsigned long)dev;
ir->active = 0;
}
}

Expand All @@ -477,8 +481,10 @@ void saa7134_ir_stop(struct saa7134_dev *dev)
del_timer_sync(&ir->timer_end);
else if (ir->nec_gpio)
tasklet_kill(&ir->tlet);
else if (ir->raw_decode)
else if (ir->raw_decode) {
del_timer_sync(&ir->timer_end);
ir->active = 0;
}

ir->running = 0;
}
Expand Down Expand Up @@ -951,38 +957,23 @@ static int saa7134_raw_decode_irq(struct saa7134_dev *dev)
unsigned long timeout;
int count, pulse, oldpulse;

/* Disable IR IRQ line */
saa_clearl(SAA7134_IRQ2, SAA7134_IRQ2_INTE_GPIO18);

/* Generate initial event */
saa_clearb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
saa_setb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
pulse = saa_readl(SAA7134_GPIO_GPSTATUS0 >> 2) & ir->mask_keydown;
ir_raw_event_store(dev->remote->dev, pulse? IR_PULSE : IR_SPACE);

#if 1
/* Wait up to 10 ms for event change */
oldpulse = pulse;
for (count = 0; count < 1000; count++) {
udelay(10);
/* rising SAA7134_GPIO_GPRESCAN reads the status */
saa_clearb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
saa_setb(SAA7134_GPIO_GPMODE3, SAA7134_GPIO_GPRESCAN);
pulse = saa_readl(SAA7134_GPIO_GPSTATUS0 >> 2)
& ir->mask_keydown;
if (pulse != oldpulse)
break;
}

/* Store final event */
ir_raw_event_store(dev->remote->dev, pulse? IR_PULSE : IR_SPACE);
#endif
/* Wait 15 ms before deciding to do something else */
timeout = jiffies + jiffies_to_msecs(15);
mod_timer(&ir->timer_end, timeout);

/* Enable IR IRQ line */
saa_setl(SAA7134_IRQ2, SAA7134_IRQ2_INTE_GPIO18);
/*
* Wait 15 ms from the start of the first IR event before processing
* the event. This time is enough for NEC protocol. May need adjustments
* to work with other protocols.
*/
if (!ir->active) {
timeout = jiffies + jiffies_to_msecs(15);
mod_timer(&ir->timer_end, timeout);
ir->active = 1;
}

return 1;
}
Expand Down
2 changes: 2 additions & 0 deletions trunk/include/media/ir-core.h
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
#include <linux/spinlock.h>
#include <linux/kfifo.h>
#include <linux/time.h>
#include <linux/timer.h>

extern int ir_core_debug;
#define IR_dprintk(level, fmt, arg...) if (ir_core_debug >= level) \
Expand Down Expand Up @@ -63,6 +64,7 @@ struct ir_raw_event {
struct ir_raw_event_ctrl {
struct kfifo kfifo; /* fifo for the pulse/space events */
struct timespec last_event; /* when last event occurred */
struct timer_list timer_keyup; /* timer for key release */
};

struct ir_input_dev {
Expand Down

0 comments on commit bb73534

Please sign in to comment.