aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media
diff options
context:
space:
mode:
authorHermann Pitton <hermann-pitton@arcor.de>2006-12-07 19:45:28 -0500
committerMauro Carvalho Chehab <mchehab@infradead.org>2007-02-21 10:34:12 -0500
commit9160723ed620f31bf38332dee02041b1cb4c9967 (patch)
tree1e2bbbb78b98d9c21a769634f908874f79cbc7ba /drivers/media
parentc8f71b01a50597e298dc3214a2f2be7b8d31170c (diff)
V4L/DVB (4961): Add support for the ASUS P7131 remote control
Besides adding the board specific code, this patch moves the RC5 decoding code from bt8xx to ir-functions.c to make it available for all drivers. Signed-off-by: Marc Fargas <telenieko.telenieko.com> Signed-off-by: Hermann Pitton <hermann-pitton@arcor.de> Signed-off-by: Hartmut Hackmann <hartmut.hackmann@t-online.de> Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Diffstat (limited to 'drivers/media')
-rw-r--r--drivers/media/common/ir-functions.c110
-rw-r--r--drivers/media/common/ir-keymaps.c54
-rw-r--r--drivers/media/video/bt8xx/bttv-input.c143
-rw-r--r--drivers/media/video/bt8xx/bttv.h27
-rw-r--r--drivers/media/video/bt8xx/bttvp.h2
-rw-r--r--drivers/media/video/saa7134/saa7134-cards.c1
-rw-r--r--drivers/media/video/saa7134/saa7134-input.c86
-rw-r--r--drivers/media/video/saa7134/saa7134.h17
8 files changed, 268 insertions, 172 deletions
diff --git a/drivers/media/common/ir-functions.c b/drivers/media/common/ir-functions.c
index 9a8dd8764c9..cbf7c056488 100644
--- a/drivers/media/common/ir-functions.c
+++ b/drivers/media/common/ir-functions.c
@@ -256,6 +256,112 @@ int ir_decode_biphase(u32 *samples, int count, int low, int high)
256 return value; 256 return value;
257} 257}
258 258
259/* RC5 decoding stuff, moved from bttv-input.c to share it with
260 * saa7134 */
261
262/* decode raw bit pattern to RC5 code */
263u32 ir_rc5_decode(unsigned int code)
264{
265 unsigned int org_code = code;
266 unsigned int pair;
267 unsigned int rc5 = 0;
268 int i;
269
270 for (i = 0; i < 14; ++i) {
271 pair = code & 0x3;
272 code >>= 2;
273
274 rc5 <<= 1;
275 switch (pair) {
276 case 0:
277 case 2:
278 break;
279 case 1:
280 rc5 |= 1;
281 break;
282 case 3:
283 dprintk(1, "ir-common: ir_rc5_decode(%x) bad code\n", org_code);
284 return 0;
285 }
286 }
287 dprintk(1, "ir-common: code=%x, rc5=%x, start=%x, toggle=%x, address=%x, "
288 "instr=%x\n", rc5, org_code, RC5_START(rc5),
289 RC5_TOGGLE(rc5), RC5_ADDR(rc5), RC5_INSTR(rc5));
290 return rc5;
291}
292
293void ir_rc5_timer_end(unsigned long data)
294{
295 struct card_ir *ir = (struct card_ir *)data;
296 struct timeval tv;
297 unsigned long current_jiffies, timeout;
298 u32 gap;
299 u32 rc5 = 0;
300
301 /* get time */
302 current_jiffies = jiffies;
303 do_gettimeofday(&tv);
304
305 /* avoid overflow with gap >1s */
306 if (tv.tv_sec - ir->base_time.tv_sec > 1) {
307 gap = 200000;
308 } else {
309 gap = 1000000 * (tv.tv_sec - ir->base_time.tv_sec) +
310 tv.tv_usec - ir->base_time.tv_usec;
311 }
312
313 /* Allow some timmer jitter (RC5 is ~24ms anyway so this is ok) */
314 if (gap < 28000) {
315 dprintk(1, "ir-common: spurious timer_end\n");
316 return;
317 }
318
319 ir->active = 0;
320 if (ir->last_bit < 20) {
321 /* ignore spurious codes (caused by light/other remotes) */
322 dprintk(1, "ir-common: short code: %x\n", ir->code);
323 } else {
324 ir->code = (ir->code << ir->shift_by) | 1;
325 rc5 = ir_rc5_decode(ir->code);
326
327 /* two start bits? */
328 if (RC5_START(rc5) != ir->start) {
329 dprintk(1, "ir-common: rc5 start bits invalid: %u\n", RC5_START(rc5));
330
331 /* right address? */
332 } else if (RC5_ADDR(rc5) == ir->addr) {
333 u32 toggle = RC5_TOGGLE(rc5);
334 u32 instr = RC5_INSTR(rc5);
335
336 /* Good code, decide if repeat/repress */
337 if (toggle != RC5_TOGGLE(ir->last_rc5) ||
338 instr != RC5_INSTR(ir->last_rc5)) {
339 dprintk(1, "ir-common: instruction %x, toggle %x\n", instr,
340 toggle);
341 ir_input_nokey(ir->dev, &ir->ir);
342 ir_input_keydown(ir->dev, &ir->ir, instr,
343 instr);
344 }
345
346 /* Set/reset key-up timer */
347 timeout = current_jiffies + (500 + ir->rc5_key_timeout
348 * HZ) / 1000;
349 mod_timer(&ir->timer_keyup, timeout);
350
351 /* Save code for repeat test */
352 ir->last_rc5 = rc5;
353 }
354 }
355}
356
357void ir_rc5_timer_keyup(unsigned long data)
358{
359 struct card_ir *ir = (struct card_ir *)data;
360
361 dprintk(1, "ir-common: key released\n");
362 ir_input_nokey(ir->dev, &ir->ir);
363}
364
259EXPORT_SYMBOL_GPL(ir_input_init); 365EXPORT_SYMBOL_GPL(ir_input_init);
260EXPORT_SYMBOL_GPL(ir_input_nokey); 366EXPORT_SYMBOL_GPL(ir_input_nokey);
261EXPORT_SYMBOL_GPL(ir_input_keydown); 367EXPORT_SYMBOL_GPL(ir_input_keydown);
@@ -265,6 +371,10 @@ EXPORT_SYMBOL_GPL(ir_dump_samples);
265EXPORT_SYMBOL_GPL(ir_decode_biphase); 371EXPORT_SYMBOL_GPL(ir_decode_biphase);
266EXPORT_SYMBOL_GPL(ir_decode_pulsedistance); 372EXPORT_SYMBOL_GPL(ir_decode_pulsedistance);
267 373
374EXPORT_SYMBOL_GPL(ir_rc5_decode);
375EXPORT_SYMBOL_GPL(ir_rc5_timer_end);
376EXPORT_SYMBOL_GPL(ir_rc5_timer_keyup);
377
268/* 378/*
269 * Local variables: 379 * Local variables:
270 * c-basic-offset: 8 380 * c-basic-offset: 8
diff --git a/drivers/media/common/ir-keymaps.c b/drivers/media/common/ir-keymaps.c
index 0e948a5c5a0..4a54a559d81 100644
--- a/drivers/media/common/ir-keymaps.c
+++ b/drivers/media/common/ir-keymaps.c
@@ -1606,3 +1606,57 @@ IR_KEYTAB_TYPE ir_codes_budget_ci_old[IR_KEYTAB_SIZE] = {
1606}; 1606};
1607 1607
1608EXPORT_SYMBOL_GPL(ir_codes_budget_ci_old); 1608EXPORT_SYMBOL_GPL(ir_codes_budget_ci_old);
1609
1610/*
1611 * Marc Fargas <telenieko@telenieko.com>
1612 * this is the remote control that comes with the asus p7131
1613 * which has a label saying is "Model PC-39"
1614 */
1615IR_KEYTAB_TYPE ir_codes_asus_pc39[IR_KEYTAB_SIZE] = {
1616 /* Keys 0 to 9 */
1617 [ 0x15 ] = KEY_0,
1618 [ 0x29 ] = KEY_1,
1619 [ 0x2d ] = KEY_2,
1620 [ 0x2b ] = KEY_3,
1621 [ 0x09 ] = KEY_4,
1622 [ 0x0d ] = KEY_5,
1623 [ 0x0b ] = KEY_6,
1624 [ 0x31 ] = KEY_7,
1625 [ 0x35 ] = KEY_8,
1626 [ 0x33 ] = KEY_9,
1627
1628 [ 0x3e ] = KEY_RADIO, /* radio */
1629 [ 0x03 ] = KEY_MENU, /* dvd/menu */
1630 [ 0x2a ] = KEY_VOLUMEUP,
1631 [ 0x19 ] = KEY_VOLUMEDOWN,
1632 [ 0x37 ] = KEY_UP,
1633 [ 0x3b ] = KEY_DOWN,
1634 [ 0x27 ] = KEY_LEFT,
1635 [ 0x2f ] = KEY_RIGHT,
1636 [ 0x25 ] = KEY_VIDEO, /* video */
1637 [ 0x39 ] = KEY_AUDIO, /* music */
1638
1639 [ 0x21 ] = KEY_TV, /* tv */
1640 [ 0x1d ] = KEY_EXIT, /* back */
1641 [ 0x0a ] = KEY_CHANNELUP, /* channel / program + */
1642 [ 0x1b ] = KEY_CHANNELDOWN, /* channel / program - */
1643 [ 0x1a ] = KEY_ENTER, /* enter */
1644
1645 [ 0x06 ] = KEY_PAUSE, /* play/pause */
1646 [ 0x1e ] = KEY_PREVIOUS, /* rew */
1647 [ 0x26 ] = KEY_NEXT, /* forward */
1648 [ 0x0e ] = KEY_REWIND, /* backward << */
1649 [ 0x3a ] = KEY_FASTFORWARD, /* forward >> */
1650 [ 0x36 ] = KEY_STOP,
1651 [ 0x2e ] = KEY_RECORD, /* recording */
1652 [ 0x16 ] = KEY_POWER, /* the button that reads "close" */
1653
1654 [ 0x11 ] = KEY_ZOOM, /* full screen */
1655 [ 0x13 ] = KEY_MACRO, /* recall */
1656 [ 0x23 ] = KEY_HOME, /* home */
1657 [ 0x05 ] = KEY_PVR, /* picture */
1658 [ 0x3d ] = KEY_MUTE, /* mute */
1659 [ 0x01 ] = KEY_DVD, /* dvd */
1660};
1661
1662EXPORT_SYMBOL_GPL(ir_codes_asus_pc39);
diff --git a/drivers/media/video/bt8xx/bttv-input.c b/drivers/media/video/bt8xx/bttv-input.c
index cbc012f71f5..14c07c60663 100644
--- a/drivers/media/video/bt8xx/bttv-input.c
+++ b/drivers/media/video/bt8xx/bttv-input.c
@@ -36,13 +36,18 @@ module_param(repeat_delay, int, 0644);
36static int repeat_period = 33; 36static int repeat_period = 33;
37module_param(repeat_period, int, 0644); 37module_param(repeat_period, int, 0644);
38 38
39int ir_rc5_remote_gap = 885;
40module_param(ir_rc5_remote_gap, int, 0644);
41int ir_rc5_key_timeout = 200;
42module_param(ir_rc5_key_timeout, int, 0644);
43
39#define DEVNAME "bttv-input" 44#define DEVNAME "bttv-input"
40 45
41/* ---------------------------------------------------------------------- */ 46/* ---------------------------------------------------------------------- */
42 47
43static void ir_handle_key(struct bttv *btv) 48static void ir_handle_key(struct bttv *btv)
44{ 49{
45 struct bttv_ir *ir = btv->remote; 50 struct card_ir *ir = btv->remote;
46 u32 gpio,data; 51 u32 gpio,data;
47 52
48 /* read gpio value */ 53 /* read gpio value */
@@ -72,7 +77,7 @@ static void ir_handle_key(struct bttv *btv)
72 77
73void bttv_input_irq(struct bttv *btv) 78void bttv_input_irq(struct bttv *btv)
74{ 79{
75 struct bttv_ir *ir = btv->remote; 80 struct card_ir *ir = btv->remote;
76 81
77 if (!ir->polling) 82 if (!ir->polling)
78 ir_handle_key(btv); 83 ir_handle_key(btv);
@@ -81,7 +86,7 @@ void bttv_input_irq(struct bttv *btv)
81static void bttv_input_timer(unsigned long data) 86static void bttv_input_timer(unsigned long data)
82{ 87{
83 struct bttv *btv = (struct bttv*)data; 88 struct bttv *btv = (struct bttv*)data;
84 struct bttv_ir *ir = btv->remote; 89 struct card_ir *ir = btv->remote;
85 unsigned long timeout; 90 unsigned long timeout;
86 91
87 ir_handle_key(btv); 92 ir_handle_key(btv);
@@ -91,51 +96,9 @@ static void bttv_input_timer(unsigned long data)
91 96
92/* ---------------------------------------------------------------*/ 97/* ---------------------------------------------------------------*/
93 98
94static int rc5_remote_gap = 885;
95module_param(rc5_remote_gap, int, 0644);
96static int rc5_key_timeout = 200;
97module_param(rc5_key_timeout, int, 0644);
98
99#define RC5_START(x) (((x)>>12)&3)
100#define RC5_TOGGLE(x) (((x)>>11)&1)
101#define RC5_ADDR(x) (((x)>>6)&31)
102#define RC5_INSTR(x) ((x)&63)
103
104/* decode raw bit pattern to RC5 code */
105static u32 rc5_decode(unsigned int code)
106{
107 unsigned int org_code = code;
108 unsigned int pair;
109 unsigned int rc5 = 0;
110 int i;
111
112 code = (code << 1) | 1;
113 for (i = 0; i < 14; ++i) {
114 pair = code & 0x3;
115 code >>= 2;
116
117 rc5 <<= 1;
118 switch (pair) {
119 case 0:
120 case 2:
121 break;
122 case 1:
123 rc5 |= 1;
124 break;
125 case 3:
126 dprintk(KERN_WARNING "bad code: %x\n", org_code);
127 return 0;
128 }
129 }
130 dprintk(KERN_WARNING "code=%x, rc5=%x, start=%x, toggle=%x, address=%x, "
131 "instr=%x\n", rc5, org_code, RC5_START(rc5),
132 RC5_TOGGLE(rc5), RC5_ADDR(rc5), RC5_INSTR(rc5));
133 return rc5;
134}
135
136static int bttv_rc5_irq(struct bttv *btv) 99static int bttv_rc5_irq(struct bttv *btv)
137{ 100{
138 struct bttv_ir *ir = btv->remote; 101 struct card_ir *ir = btv->remote;
139 struct timeval tv; 102 struct timeval tv;
140 u32 gpio; 103 u32 gpio;
141 u32 gap; 104 u32 gap;
@@ -165,8 +128,8 @@ static int bttv_rc5_irq(struct bttv *btv)
165 /* only if in the code (otherwise spurious IRQ or timer 128 /* only if in the code (otherwise spurious IRQ or timer
166 late) */ 129 late) */
167 if (ir->last_bit < 28) { 130 if (ir->last_bit < 28) {
168 ir->last_bit = (gap - rc5_remote_gap / 2) / 131 ir->last_bit = (gap - ir_rc5_remote_gap / 2) /
169 rc5_remote_gap; 132 ir_rc5_remote_gap;
170 ir->code |= 1 << ir->last_bit; 133 ir->code |= 1 << ir->last_bit;
171 } 134 }
172 /* starting new code */ 135 /* starting new code */
@@ -186,80 +149,9 @@ static int bttv_rc5_irq(struct bttv *btv)
186 return 1; 149 return 1;
187} 150}
188 151
189
190static void bttv_rc5_timer_end(unsigned long data)
191{
192 struct bttv_ir *ir = (struct bttv_ir *)data;
193 struct timeval tv;
194 unsigned long current_jiffies, timeout;
195 u32 gap;
196
197 /* get time */
198 current_jiffies = jiffies;
199 do_gettimeofday(&tv);
200
201 /* avoid overflow with gap >1s */
202 if (tv.tv_sec - ir->base_time.tv_sec > 1) {
203 gap = 200000;
204 } else {
205 gap = 1000000 * (tv.tv_sec - ir->base_time.tv_sec) +
206 tv.tv_usec - ir->base_time.tv_usec;
207 }
208
209 /* Allow some timmer jitter (RC5 is ~24ms anyway so this is ok) */
210 if (gap < 28000) {
211 dprintk(KERN_WARNING "spurious timer_end\n");
212 return;
213 }
214
215 ir->active = 0;
216 if (ir->last_bit < 20) {
217 /* ignore spurious codes (caused by light/other remotes) */
218 dprintk(KERN_WARNING "short code: %x\n", ir->code);
219 } else {
220 u32 rc5 = rc5_decode(ir->code);
221
222 /* two start bits? */
223 if (RC5_START(rc5) != 3) {
224 dprintk(KERN_WARNING "rc5 start bits invalid: %u\n", RC5_START(rc5));
225
226 /* right address? */
227 } else if (RC5_ADDR(rc5) == 0x0) {
228 u32 toggle = RC5_TOGGLE(rc5);
229 u32 instr = RC5_INSTR(rc5);
230
231 /* Good code, decide if repeat/repress */
232 if (toggle != RC5_TOGGLE(ir->last_rc5) ||
233 instr != RC5_INSTR(ir->last_rc5)) {
234 dprintk(KERN_WARNING "instruction %x, toggle %x\n", instr,
235 toggle);
236 ir_input_nokey(ir->dev, &ir->ir);
237 ir_input_keydown(ir->dev, &ir->ir, instr,
238 instr);
239 }
240
241 /* Set/reset key-up timer */
242 timeout = current_jiffies + (500 + rc5_key_timeout
243 * HZ) / 1000;
244 mod_timer(&ir->timer_keyup, timeout);
245
246 /* Save code for repeat test */
247 ir->last_rc5 = rc5;
248 }
249 }
250}
251
252static void bttv_rc5_timer_keyup(unsigned long data)
253{
254 struct bttv_ir *ir = (struct bttv_ir *)data;
255
256 dprintk(KERN_DEBUG "key released\n");
257 ir_input_nokey(ir->dev, &ir->ir);
258}
259
260/* ---------------------------------------------------------------------- */ 152/* ---------------------------------------------------------------------- */
261 153
262static void bttv_ir_start(struct bttv *btv, struct bttv_ir *ir) 154static void bttv_ir_start(struct bttv *btv, struct card_ir *ir)
263{ 155{
264 if (ir->polling) { 156 if (ir->polling) {
265 init_timer(&ir->timer); 157 init_timer(&ir->timer);
@@ -270,12 +162,17 @@ static void bttv_ir_start(struct bttv *btv, struct bttv_ir *ir)
270 } else if (ir->rc5_gpio) { 162 } else if (ir->rc5_gpio) {
271 /* set timer_end for code completion */ 163 /* set timer_end for code completion */
272 init_timer(&ir->timer_end); 164 init_timer(&ir->timer_end);
273 ir->timer_end.function = bttv_rc5_timer_end; 165 ir->timer_end.function = ir_rc5_timer_end;
274 ir->timer_end.data = (unsigned long)ir; 166 ir->timer_end.data = (unsigned long)ir;
275 167
276 init_timer(&ir->timer_keyup); 168 init_timer(&ir->timer_keyup);
277 ir->timer_keyup.function = bttv_rc5_timer_keyup; 169 ir->timer_keyup.function = ir_rc5_timer_keyup;
278 ir->timer_keyup.data = (unsigned long)ir; 170 ir->timer_keyup.data = (unsigned long)ir;
171 ir->shift_by = 1;
172 ir->start = 3;
173 ir->addr = 0x0;
174 ir->rc5_key_timeout = ir_rc5_key_timeout;
175 ir->rc5_remote_gap = ir_rc5_remote_gap;
279 } 176 }
280} 177}
281 178
@@ -299,7 +196,7 @@ static void bttv_ir_stop(struct bttv *btv)
299 196
300int bttv_input_init(struct bttv *btv) 197int bttv_input_init(struct bttv *btv)
301{ 198{
302 struct bttv_ir *ir; 199 struct card_ir *ir;
303 IR_KEYTAB_TYPE *ir_codes = NULL; 200 IR_KEYTAB_TYPE *ir_codes = NULL;
304 struct input_dev *input_dev; 201 struct input_dev *input_dev;
305 int ir_type = IR_TYPE_OTHER; 202 int ir_type = IR_TYPE_OTHER;
diff --git a/drivers/media/video/bt8xx/bttv.h b/drivers/media/video/bt8xx/bttv.h
index f9c9e3c4d11..5491acbdaf6 100644
--- a/drivers/media/video/bt8xx/bttv.h
+++ b/drivers/media/video/bt8xx/bttv.h
@@ -197,33 +197,6 @@ struct bttv_core {
197struct bttv; 197struct bttv;
198 198
199 199
200struct bttv_ir {
201 struct input_dev *dev;
202 struct ir_input_state ir;
203 char name[32];
204 char phys[32];
205
206 /* Usual gpio signalling */
207
208 u32 mask_keycode;
209 u32 mask_keydown;
210 u32 mask_keyup;
211 u32 polling;
212 u32 last_gpio;
213 struct work_struct work;
214 struct timer_list timer;
215
216 /* RC5 gpio */
217 u32 rc5_gpio;
218 struct timer_list timer_end; /* timer_end for code completion */
219 struct timer_list timer_keyup; /* timer_end for key release */
220 u32 last_rc5; /* last good rc5 code */
221 u32 last_bit; /* last raw bit seen */
222 u32 code; /* raw code under construction */
223 struct timeval base_time; /* time of last seen code */
224 int active; /* building raw code */
225};
226
227struct tvcard 200struct tvcard
228{ 201{
229 char *name; 202 char *name;
diff --git a/drivers/media/video/bt8xx/bttvp.h b/drivers/media/video/bt8xx/bttvp.h
index 311c4c541e0..3802cafc9e4 100644
--- a/drivers/media/video/bt8xx/bttvp.h
+++ b/drivers/media/video/bt8xx/bttvp.h
@@ -308,7 +308,7 @@ struct bttv {
308 308
309 /* infrared remote */ 309 /* infrared remote */
310 int has_remote; 310 int has_remote;
311 struct bttv_ir *remote; 311 struct card_ir *remote;
312 312
313 /* locking */ 313 /* locking */
314 spinlock_t s_lock; 314 spinlock_t s_lock;
diff --git a/drivers/media/video/saa7134/saa7134-cards.c b/drivers/media/video/saa7134/saa7134-cards.c
index ae984bbe36b..dc45b2374f5 100644
--- a/drivers/media/video/saa7134/saa7134-cards.c
+++ b/drivers/media/video/saa7134/saa7134-cards.c
@@ -3926,6 +3926,7 @@ int saa7134_board_init1(struct saa7134_dev *dev)
3926 case SAA7134_BOARD_KWORLD_TERMINATOR: 3926 case SAA7134_BOARD_KWORLD_TERMINATOR:
3927 case SAA7134_BOARD_SEDNA_PC_TV_CARDBUS: 3927 case SAA7134_BOARD_SEDNA_PC_TV_CARDBUS:
3928 case SAA7134_BOARD_FLYDVBT_LR301: 3928 case SAA7134_BOARD_FLYDVBT_LR301:
3929 case SAA7134_BOARD_ASUSTeK_P7131_DUAL:
3929 case SAA7134_BOARD_FLYDVBTDUO: 3930 case SAA7134_BOARD_FLYDVBTDUO:
3930 case SAA7134_BOARD_PROTEUS_2309: 3931 case SAA7134_BOARD_PROTEUS_2309:
3931 case SAA7134_BOARD_AVERMEDIA_A16AR: 3932 case SAA7134_BOARD_AVERMEDIA_A16AR:
diff --git a/drivers/media/video/saa7134/saa7134-input.c b/drivers/media/video/saa7134/saa7134-input.c
index e4252683a59..08848ebc682 100644
--- a/drivers/media/video/saa7134/saa7134-input.c
+++ b/drivers/media/video/saa7134/saa7134-input.c
@@ -40,16 +40,24 @@ static int pinnacle_remote = 0;
40module_param(pinnacle_remote, int, 0644); /* Choose Pinnacle PCTV remote */ 40module_param(pinnacle_remote, int, 0644); /* Choose Pinnacle PCTV remote */
41MODULE_PARM_DESC(pinnacle_remote, "Specify Pinnacle PCTV remote: 0=coloured, 1=grey (defaults to 0)"); 41MODULE_PARM_DESC(pinnacle_remote, "Specify Pinnacle PCTV remote: 0=coloured, 1=grey (defaults to 0)");
42 42
43int ir_rc5_remote_gap = 885;
44module_param(ir_rc5_remote_gap, int, 0644);
45int ir_rc5_key_timeout = 115;
46module_param(ir_rc5_key_timeout, int, 0644);
47
43#define dprintk(fmt, arg...) if (ir_debug) \ 48#define dprintk(fmt, arg...) if (ir_debug) \
44 printk(KERN_DEBUG "%s/ir: " fmt, dev->name , ## arg) 49 printk(KERN_DEBUG "%s/ir: " fmt, dev->name , ## arg)
45#define i2cdprintk(fmt, arg...) if (ir_debug) \ 50#define i2cdprintk(fmt, arg...) if (ir_debug) \
46 printk(KERN_DEBUG "%s/ir: " fmt, ir->c.name , ## arg) 51 printk(KERN_DEBUG "%s/ir: " fmt, ir->c.name , ## arg)
47 52
53/** rc5 functions */
54static int saa7134_rc5_irq(struct saa7134_dev *dev);
55
48/* -------------------- GPIO generic keycode builder -------------------- */ 56/* -------------------- GPIO generic keycode builder -------------------- */
49 57
50static int build_key(struct saa7134_dev *dev) 58static int build_key(struct saa7134_dev *dev)
51{ 59{
52 struct saa7134_ir *ir = dev->remote; 60 struct card_ir *ir = dev->remote;
53 u32 gpio, data; 61 u32 gpio, data;
54 62
55 /* rising SAA7134_GPIO_GPRESCAN reads the status */ 63 /* rising SAA7134_GPIO_GPRESCAN reads the status */
@@ -134,16 +142,19 @@ static int get_key_hvr1110(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw)
134 142
135void saa7134_input_irq(struct saa7134_dev *dev) 143void saa7134_input_irq(struct saa7134_dev *dev)
136{ 144{
137 struct saa7134_ir *ir = dev->remote; 145 struct card_ir *ir = dev->remote;
138 146
139 if (!ir->polling) 147 if (!ir->polling && !ir->rc5_gpio) {
140 build_key(dev); 148 build_key(dev);
149 } else if (ir->rc5_gpio) {
150 saa7134_rc5_irq(dev);
151 }
141} 152}
142 153
143static void saa7134_input_timer(unsigned long data) 154static void saa7134_input_timer(unsigned long data)
144{ 155{
145 struct saa7134_dev *dev = (struct saa7134_dev*)data; 156 struct saa7134_dev *dev = (struct saa7134_dev*)data;
146 struct saa7134_ir *ir = dev->remote; 157 struct card_ir *ir = dev->remote;
147 unsigned long timeout; 158 unsigned long timeout;
148 159
149 build_key(dev); 160 build_key(dev);
@@ -151,7 +162,7 @@ static void saa7134_input_timer(unsigned long data)
151 mod_timer(&ir->timer, timeout); 162 mod_timer(&ir->timer, timeout);
152} 163}
153 164
154static void saa7134_ir_start(struct saa7134_dev *dev, struct saa7134_ir *ir) 165static void saa7134_ir_start(struct saa7134_dev *dev, struct card_ir *ir)
155{ 166{
156 if (ir->polling) { 167 if (ir->polling) {
157 init_timer(&ir->timer); 168 init_timer(&ir->timer);
@@ -159,6 +170,19 @@ static void saa7134_ir_start(struct saa7134_dev *dev, struct saa7134_ir *ir)
159 ir->timer.data = (unsigned long)dev; 170 ir->timer.data = (unsigned long)dev;
160 ir->timer.expires = jiffies + HZ; 171 ir->timer.expires = jiffies + HZ;
161 add_timer(&ir->timer); 172 add_timer(&ir->timer);
173 } else if (ir->rc5_gpio) {
174 /* set timer_end for code completion */
175 init_timer(&ir->timer_end);
176 ir->timer_end.function = ir_rc5_timer_end;
177 ir->timer_end.data = (unsigned long)ir;
178 init_timer(&ir->timer_keyup);
179 ir->timer_keyup.function = ir_rc5_timer_keyup;
180 ir->timer_keyup.data = (unsigned long)ir;
181 ir->shift_by = 2;
182 ir->start = 0x2;
183 ir->addr = 0x17;
184 ir->rc5_key_timeout = ir_rc5_key_timeout;
185 ir->rc5_remote_gap = ir_rc5_remote_gap;
162 } 186 }
163} 187}
164 188
@@ -170,13 +194,14 @@ static void saa7134_ir_stop(struct saa7134_dev *dev)
170 194
171int saa7134_input_init1(struct saa7134_dev *dev) 195int saa7134_input_init1(struct saa7134_dev *dev)
172{ 196{
173 struct saa7134_ir *ir; 197 struct card_ir *ir;
174 struct input_dev *input_dev; 198 struct input_dev *input_dev;
175 IR_KEYTAB_TYPE *ir_codes = NULL; 199 IR_KEYTAB_TYPE *ir_codes = NULL;
176 u32 mask_keycode = 0; 200 u32 mask_keycode = 0;
177 u32 mask_keydown = 0; 201 u32 mask_keydown = 0;
178 u32 mask_keyup = 0; 202 u32 mask_keyup = 0;
179 int polling = 0; 203 int polling = 0;
204 int rc5_gpio = 0;
180 int ir_type = IR_TYPE_OTHER; 205 int ir_type = IR_TYPE_OTHER;
181 int err; 206 int err;
182 207
@@ -295,6 +320,11 @@ int saa7134_input_init1(struct saa7134_dev *dev)
295 mask_keycode = 0x0001F00; 320 mask_keycode = 0x0001F00;
296 mask_keydown = 0x0040000; 321 mask_keydown = 0x0040000;
297 break; 322 break;
323 case SAA7134_BOARD_ASUSTeK_P7131_DUAL:
324 ir_codes = ir_codes_asus_pc39;
325 mask_keydown = 0x0040000;
326 rc5_gpio = 1;
327 break;
298 } 328 }
299 if (NULL == ir_codes) { 329 if (NULL == ir_codes) {
300 printk("%s: Oops: IR config error [card=%d]\n", 330 printk("%s: Oops: IR config error [card=%d]\n",
@@ -316,6 +346,7 @@ int saa7134_input_init1(struct saa7134_dev *dev)
316 ir->mask_keydown = mask_keydown; 346 ir->mask_keydown = mask_keydown;
317 ir->mask_keyup = mask_keyup; 347 ir->mask_keyup = mask_keyup;
318 ir->polling = polling; 348 ir->polling = polling;
349 ir->rc5_gpio = rc5_gpio;
319 350
320 /* init input device */ 351 /* init input device */
321 snprintf(ir->name, sizeof(ir->name), "saa7134 IR (%s)", 352 snprintf(ir->name, sizeof(ir->name), "saa7134 IR (%s)",
@@ -402,6 +433,49 @@ void saa7134_set_i2c_ir(struct saa7134_dev *dev, struct IR_i2c *ir)
402 } 433 }
403 434
404} 435}
436
437static int saa7134_rc5_irq(struct saa7134_dev *dev)
438{
439 struct card_ir *ir = dev->remote;
440 struct timeval tv;
441 u32 gap;
442 unsigned long current_jiffies, timeout;
443
444 /* get time of bit */
445 current_jiffies = jiffies;
446 do_gettimeofday(&tv);
447
448 /* avoid overflow with gap >1s */
449 if (tv.tv_sec - ir->base_time.tv_sec > 1) {
450 gap = 200000;
451 } else {
452 gap = 1000000 * (tv.tv_sec - ir->base_time.tv_sec) +
453 tv.tv_usec - ir->base_time.tv_usec;
454 }
455
456 /* active code => add bit */
457 if (ir->active) {
458 /* only if in the code (otherwise spurious IRQ or timer
459 late) */
460 if (ir->last_bit < 28) {
461 ir->last_bit = (gap - ir_rc5_remote_gap / 2) /
462 ir_rc5_remote_gap;
463 ir->code |= 1 << ir->last_bit;
464 }
465 /* starting new code */
466 } else {
467 ir->active = 1;
468 ir->code = 0;
469 ir->base_time = tv;
470 ir->last_bit = 0;
471
472 timeout = current_jiffies + (500 + 30 * HZ) / 1000;
473 mod_timer(&ir->timer_end, timeout);
474 }
475
476 return 1;
477}
478
405/* ---------------------------------------------------------------------- 479/* ----------------------------------------------------------------------
406 * Local variables: 480 * Local variables:
407 * c-basic-offset: 8 481 * c-basic-offset: 8
diff --git a/drivers/media/video/saa7134/saa7134.h b/drivers/media/video/saa7134/saa7134.h
index 88cd1297df1..73b16b2044f 100644
--- a/drivers/media/video/saa7134/saa7134.h
+++ b/drivers/media/video/saa7134/saa7134.h
@@ -411,20 +411,6 @@ struct saa7134_dmasound {
411 struct snd_pcm_substream *substream; 411 struct snd_pcm_substream *substream;
412}; 412};
413 413
414/* IR input */
415struct saa7134_ir {
416 struct input_dev *dev;
417 struct ir_input_state ir;
418 char name[32];
419 char phys[32];
420 u32 mask_keycode;
421 u32 mask_keydown;
422 u32 mask_keyup;
423 int polling;
424 u32 last_gpio;
425 struct timer_list timer;
426};
427
428/* ts/mpeg status */ 414/* ts/mpeg status */
429struct saa7134_ts { 415struct saa7134_ts {
430 /* TS capture */ 416 /* TS capture */
@@ -463,7 +449,7 @@ struct saa7134_dev {
463 449
464 /* infrared remote */ 450 /* infrared remote */
465 int has_remote; 451 int has_remote;
466 struct saa7134_ir *remote; 452 struct card_ir *remote;
467 453
468 /* pci i/o */ 454 /* pci i/o */
469 char name[32]; 455 char name[32];
@@ -698,6 +684,7 @@ void saa7134_input_fini(struct saa7134_dev *dev);
698void saa7134_input_irq(struct saa7134_dev *dev); 684void saa7134_input_irq(struct saa7134_dev *dev);
699void saa7134_set_i2c_ir(struct saa7134_dev *dev, struct IR_i2c *ir); 685void saa7134_set_i2c_ir(struct saa7134_dev *dev, struct IR_i2c *ir);
700 686
687
701/* 688/*
702 * Local variables: 689 * Local variables:
703 * c-basic-offset: 8 690 * c-basic-offset: 8