diff options
Diffstat (limited to 'drivers/media/video/saa7134/saa7134-input.c')
-rw-r--r-- | drivers/media/video/saa7134/saa7134-input.c | 202 |
1 files changed, 200 insertions, 2 deletions
diff --git a/drivers/media/video/saa7134/saa7134-input.c b/drivers/media/video/saa7134/saa7134-input.c index f99dbb729555..6413a6e09873 100644 --- a/drivers/media/video/saa7134/saa7134-input.c +++ b/drivers/media/video/saa7134/saa7134-input.c | |||
@@ -39,6 +39,8 @@ MODULE_PARM_DESC(ir_debug,"enable debug messages [IR]"); | |||
39 | 39 | ||
40 | #define dprintk(fmt, arg...) if (ir_debug) \ | 40 | #define dprintk(fmt, arg...) if (ir_debug) \ |
41 | printk(KERN_DEBUG "%s/ir: " fmt, dev->name , ## arg) | 41 | printk(KERN_DEBUG "%s/ir: " fmt, dev->name , ## arg) |
42 | #define i2cdprintk(fmt, arg...) if (ir_debug) \ | ||
43 | printk(KERN_DEBUG "%s/ir: " fmt, ir->c.name , ## arg) | ||
42 | 44 | ||
43 | /* ---------------------------------------------------------------------- */ | 45 | /* ---------------------------------------------------------------------- */ |
44 | 46 | ||
@@ -443,7 +445,105 @@ static IR_KEYTAB_TYPE gotview7135_codes[IR_KEYTAB_SIZE] = { | |||
443 | [ 112] = KEY_F24, /* NORMAL TIMESHIFT */ | 445 | [ 112] = KEY_F24, /* NORMAL TIMESHIFT */ |
444 | }; | 446 | }; |
445 | 447 | ||
446 | /* ---------------------------------------------------------------------- */ | 448 | static IR_KEYTAB_TYPE ir_codes_purpletv[IR_KEYTAB_SIZE] = { |
449 | [ 0x3 ] = KEY_POWER, | ||
450 | [ 0x6f ] = KEY_MUTE, | ||
451 | [ 0x10 ] = KEY_BACKSPACE, /* Recall */ | ||
452 | |||
453 | [ 0x11 ] = KEY_KP0, | ||
454 | [ 0x4 ] = KEY_KP1, | ||
455 | [ 0x5 ] = KEY_KP2, | ||
456 | [ 0x6 ] = KEY_KP3, | ||
457 | [ 0x8 ] = KEY_KP4, | ||
458 | [ 0x9 ] = KEY_KP5, | ||
459 | [ 0xa ] = KEY_KP6, | ||
460 | [ 0xc ] = KEY_KP7, | ||
461 | [ 0xd ] = KEY_KP8, | ||
462 | [ 0xe ] = KEY_KP9, | ||
463 | [ 0x12 ] = KEY_KPDOT, /* 100+ */ | ||
464 | |||
465 | [ 0x7 ] = KEY_VOLUMEUP, | ||
466 | [ 0xb ] = KEY_VOLUMEDOWN, | ||
467 | [ 0x1a ] = KEY_KPPLUS, | ||
468 | [ 0x18 ] = KEY_KPMINUS, | ||
469 | [ 0x15 ] = KEY_UP, | ||
470 | [ 0x1d ] = KEY_DOWN, | ||
471 | [ 0xf ] = KEY_CHANNELUP, | ||
472 | [ 0x13 ] = KEY_CHANNELDOWN, | ||
473 | [ 0x48 ] = KEY_ZOOM, | ||
474 | |||
475 | [ 0x1b ] = KEY_VIDEO, /* Video source */ | ||
476 | [ 0x49 ] = KEY_LANGUAGE, /* MTS Select */ | ||
477 | [ 0x19 ] = KEY_SEARCH, /* Auto Scan */ | ||
478 | |||
479 | [ 0x4b ] = KEY_RECORD, | ||
480 | [ 0x46 ] = KEY_PLAY, | ||
481 | [ 0x45 ] = KEY_PAUSE, /* Pause */ | ||
482 | [ 0x44 ] = KEY_STOP, | ||
483 | [ 0x40 ] = KEY_FORWARD, /* Forward ? */ | ||
484 | [ 0x42 ] = KEY_REWIND, /* Backward ? */ | ||
485 | |||
486 | }; | ||
487 | |||
488 | static IR_KEYTAB_TYPE ir_codes_pinnacle[IR_KEYTAB_SIZE] = { | ||
489 | [ 0x59 ] = KEY_MUTE, | ||
490 | [ 0x4a ] = KEY_POWER, | ||
491 | |||
492 | [ 0x18 ] = KEY_TEXT, | ||
493 | [ 0x26 ] = KEY_TV, | ||
494 | [ 0x3d ] = KEY_PRINT, | ||
495 | |||
496 | [ 0x48 ] = KEY_RED, | ||
497 | [ 0x04 ] = KEY_GREEN, | ||
498 | [ 0x11 ] = KEY_YELLOW, | ||
499 | [ 0x00 ] = KEY_BLUE, | ||
500 | |||
501 | [ 0x2d ] = KEY_VOLUMEUP, | ||
502 | [ 0x1e ] = KEY_VOLUMEDOWN, | ||
503 | |||
504 | [ 0x49 ] = KEY_MENU, | ||
505 | |||
506 | [ 0x16 ] = KEY_CHANNELUP, | ||
507 | [ 0x17 ] = KEY_CHANNELDOWN, | ||
508 | |||
509 | [ 0x20 ] = KEY_UP, | ||
510 | [ 0x21 ] = KEY_DOWN, | ||
511 | [ 0x22 ] = KEY_LEFT, | ||
512 | [ 0x23 ] = KEY_RIGHT, | ||
513 | [ 0x0d ] = KEY_SELECT, | ||
514 | |||
515 | |||
516 | |||
517 | [ 0x08 ] = KEY_BACK, | ||
518 | [ 0x07 ] = KEY_REFRESH, | ||
519 | |||
520 | [ 0x2f ] = KEY_ZOOM, | ||
521 | [ 0x29 ] = KEY_RECORD, | ||
522 | |||
523 | [ 0x4b ] = KEY_PAUSE, | ||
524 | [ 0x4d ] = KEY_REWIND, | ||
525 | [ 0x2e ] = KEY_PLAY, | ||
526 | [ 0x4e ] = KEY_FORWARD, | ||
527 | [ 0x53 ] = KEY_PREVIOUS, | ||
528 | [ 0x4c ] = KEY_STOP, | ||
529 | [ 0x54 ] = KEY_NEXT, | ||
530 | |||
531 | [ 0x69 ] = KEY_KP0, | ||
532 | [ 0x6a ] = KEY_KP1, | ||
533 | [ 0x6b ] = KEY_KP2, | ||
534 | [ 0x6c ] = KEY_KP3, | ||
535 | [ 0x6d ] = KEY_KP4, | ||
536 | [ 0x6e ] = KEY_KP5, | ||
537 | [ 0x6f ] = KEY_KP6, | ||
538 | [ 0x70 ] = KEY_KP7, | ||
539 | [ 0x71 ] = KEY_KP8, | ||
540 | [ 0x72 ] = KEY_KP9, | ||
541 | |||
542 | [ 0x74 ] = KEY_CHANNEL, | ||
543 | [ 0x0a ] = KEY_BACKSPACE, | ||
544 | }; | ||
545 | |||
546 | /* -------------------- GPIO generic keycode builder -------------------- */ | ||
447 | 547 | ||
448 | static int build_key(struct saa7134_dev *dev) | 548 | static int build_key(struct saa7134_dev *dev) |
449 | { | 549 | { |
@@ -474,7 +574,81 @@ static int build_key(struct saa7134_dev *dev) | |||
474 | return 0; | 574 | return 0; |
475 | } | 575 | } |
476 | 576 | ||
477 | /* ---------------------------------------------------------------------- */ | 577 | /* --------------------- Chip specific I2C key builders ----------------- */ |
578 | |||
579 | static int get_key_purpletv(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | ||
580 | { | ||
581 | unsigned char b; | ||
582 | |||
583 | /* poll IR chip */ | ||
584 | if (1 != i2c_master_recv(&ir->c,&b,1)) { | ||
585 | i2cdprintk("read error\n"); | ||
586 | return -EIO; | ||
587 | } | ||
588 | |||
589 | /* no button press */ | ||
590 | if (b==0) | ||
591 | return 0; | ||
592 | |||
593 | /* repeating */ | ||
594 | if (b & 0x80) | ||
595 | return 1; | ||
596 | |||
597 | *ir_key = b; | ||
598 | *ir_raw = b; | ||
599 | return 1; | ||
600 | } | ||
601 | |||
602 | /* The new pinnacle PCTV remote (with the colored buttons) | ||
603 | * | ||
604 | * Ricardo Cerqueira <v4l@cerqueira.org> | ||
605 | */ | ||
606 | |||
607 | static int get_key_pinnacle(struct IR_i2c *ir, u32 *ir_key, u32 *ir_raw) | ||
608 | { | ||
609 | unsigned char b[4]; | ||
610 | unsigned int start = 0,parity = 0,code = 0; | ||
611 | |||
612 | /* poll IR chip */ | ||
613 | if (4 != i2c_master_recv(&ir->c,b,4)) { | ||
614 | i2cdprintk("read error\n"); | ||
615 | return -EIO; | ||
616 | } | ||
617 | |||
618 | for (start = 0; start<4; start++) { | ||
619 | if (b[start] == 0x80) { | ||
620 | code=b[(start+3)%4]; | ||
621 | parity=b[(start+2)%4]; | ||
622 | } | ||
623 | } | ||
624 | |||
625 | /* Empty Request */ | ||
626 | if (parity==0) | ||
627 | return 0; | ||
628 | |||
629 | /* Repeating... */ | ||
630 | if (ir->old == parity) | ||
631 | return 0; | ||
632 | |||
633 | |||
634 | ir->old = parity; | ||
635 | |||
636 | /* Reduce code value to fit inside IR_KEYTAB_SIZE | ||
637 | * | ||
638 | * this is the only value that results in 42 unique | ||
639 | * codes < 128 | ||
640 | */ | ||
641 | |||
642 | code %= 0x88; | ||
643 | |||
644 | *ir_raw = code; | ||
645 | *ir_key = code; | ||
646 | |||
647 | i2cdprintk("Pinnacle PCTV key %02x\n", code); | ||
648 | |||
649 | return 1; | ||
650 | } | ||
651 | |||
478 | 652 | ||
479 | void saa7134_input_irq(struct saa7134_dev *dev) | 653 | void saa7134_input_irq(struct saa7134_dev *dev) |
480 | { | 654 | { |
@@ -658,6 +832,30 @@ void saa7134_input_fini(struct saa7134_dev *dev) | |||
658 | dev->remote = NULL; | 832 | dev->remote = NULL; |
659 | } | 833 | } |
660 | 834 | ||
835 | void saa7134_set_i2c_ir(struct saa7134_dev *dev, struct IR_i2c *ir) | ||
836 | { | ||
837 | if (disable_ir) { | ||
838 | ir->get_key=NULL; | ||
839 | return; | ||
840 | } | ||
841 | |||
842 | switch (dev->board) { | ||
843 | case SAA7134_BOARD_PINNACLE_PCTV_110i: | ||
844 | snprintf(ir->c.name, sizeof(ir->c.name), "Pinnacle PCTV"); | ||
845 | ir->get_key = get_key_pinnacle; | ||
846 | ir->ir_codes = ir_codes_pinnacle; | ||
847 | break; | ||
848 | case SAA7134_BOARD_UPMOST_PURPLE_TV: | ||
849 | snprintf(ir->c.name, sizeof(ir->c.name), "Purple TV"); | ||
850 | ir->get_key = get_key_purpletv; | ||
851 | ir->ir_codes = ir_codes_purpletv; | ||
852 | break; | ||
853 | default: | ||
854 | dprintk("Shouldn't get here: Unknown board %x for I2C IR?\n",dev->board); | ||
855 | break; | ||
856 | } | ||
857 | |||
858 | } | ||
661 | /* ---------------------------------------------------------------------- | 859 | /* ---------------------------------------------------------------------- |
662 | * Local variables: | 860 | * Local variables: |
663 | * c-basic-offset: 8 | 861 | * c-basic-offset: 8 |