Merge tag 'media/v4.9-1' of git://git.kernel.org/pub/scm/linux/kernel/git/mchehab...
[cascardo/linux.git] / drivers / media / i2c / ir-kbd-i2c.c
index bf82726..f95a6bc 100644 (file)
@@ -35,6 +35,7 @@
  *
  */
 
+#include <asm/unaligned.h>
 #include <linux/module.h>
 #include <linux/init.h>
 #include <linux/kernel.h>
@@ -63,51 +64,80 @@ module_param(debug, int, 0644);    /* debug level (0,1,2) */
 /* ----------------------------------------------------------------------- */
 
 static int get_key_haup_common(struct IR_i2c *ir, enum rc_type *protocol,
-                              u32 *scancode, u8 *ptoggle, int size, int offset)
+                                       u32 *scancode, u8 *ptoggle, int size)
 {
        unsigned char buf[6];
-       int start, range, toggle, dev, code, ircode;
+       int start, range, toggle, dev, code, ircode, vendor;
 
        /* poll IR chip */
        if (size != i2c_master_recv(ir->c, buf, size))
                return -EIO;
 
-       /* split rc5 data block ... */
-       start  = (buf[offset] >> 7) &    1;
-       range  = (buf[offset] >> 6) &    1;
-       toggle = (buf[offset] >> 5) &    1;
-       dev    =  buf[offset]       & 0x1f;
-       code   = (buf[offset+1] >> 2) & 0x3f;
+       if (buf[0] & 0x80) {
+               int offset = (size == 6) ? 3 : 0;
 
-       /* rc5 has two start bits
-        * the first bit must be one
-        * the second bit defines the command range (1 = 0-63, 0 = 64 - 127)
-        */
-       if (!start)
-               /* no key pressed */
-               return 0;
+               /* split rc5 data block ... */
+               start  = (buf[offset] >> 7) &    1;
+               range  = (buf[offset] >> 6) &    1;
+               toggle = (buf[offset] >> 5) &    1;
+               dev    =  buf[offset]       & 0x1f;
+               code   = (buf[offset+1] >> 2) & 0x3f;
 
-       /* filter out invalid key presses */
-       ircode = (start << 12) | (toggle << 11) | (dev << 6) | code;
-       if ((ircode & 0x1fff) == 0x1fff)
-               return 0;
+               /* rc5 has two start bits
+                * the first bit must be one
+                * the second bit defines the command range:
+                * 1 = 0-63, 0 = 64 - 127
+                */
+               if (!start)
+                       /* no key pressed */
+                       return 0;
 
-       if (!range)
-               code += 64;
+               /* filter out invalid key presses */
+               ircode = (start << 12) | (toggle << 11) | (dev << 6) | code;
+               if ((ircode & 0x1fff) == 0x1fff)
+                       return 0;
 
-       dprintk(1,"ir hauppauge (rc5): s%d r%d t%d dev=%d code=%d\n",
-               start, range, toggle, dev, code);
+               if (!range)
+                       code += 64;
 
-       *protocol = RC_TYPE_RC5;
-       *scancode = RC_SCANCODE_RC5(dev, code);
-       *ptoggle = toggle;
-       return 1;
+               dprintk(1, "ir hauppauge (rc5): s%d r%d t%d dev=%d code=%d\n",
+                       start, range, toggle, dev, code);
+
+               *protocol = RC_TYPE_RC5;
+               *scancode = RC_SCANCODE_RC5(dev, code);
+               *ptoggle = toggle;
+
+               return 1;
+       } else if (size == 6 && (buf[0] & 0x40)) {
+               code = buf[4];
+               dev = buf[3];
+               vendor = get_unaligned_be16(buf + 1);
+
+               if (vendor == 0x800f) {
+                       *ptoggle = (dev & 0x80) != 0;
+                       *protocol = RC_TYPE_RC6_MCE;
+                       dev &= 0x7f;
+                       dprintk(1, "ir hauppauge (rc6-mce): t%d vendor=%d dev=%d code=%d\n",
+                                               toggle, vendor, dev, code);
+               } else {
+                       *ptoggle = 0;
+                       *protocol = RC_TYPE_RC6_6A_32;
+                       dprintk(1, "ir hauppauge (rc6-6a-32): vendor=%d dev=%d code=%d\n",
+                                                       vendor, dev, code);
+               }
+
+               *scancode = RC_SCANCODE_RC6_6A(vendor, dev, code);
+
+               return 1;
+       }
+
+       return 0;
 }
 
 static int get_key_haup(struct IR_i2c *ir, enum rc_type *protocol,
                        u32 *scancode, u8 *toggle)
 {
-       return get_key_haup_common (ir, protocol, scancode, toggle, 3, 0);
+       return get_key_haup_common(ir, protocol, scancode, toggle, 3);
 }
 
 static int get_key_haup_xvr(struct IR_i2c *ir, enum rc_type *protocol,
@@ -126,7 +156,7 @@ static int get_key_haup_xvr(struct IR_i2c *ir, enum rc_type *protocol,
        if (ret != 1)
                return (ret < 0) ? ret : -EINVAL;
 
-       return get_key_haup_common(ir, protocol, scancode, toggle, 6, 3);
+       return get_key_haup_common(ir, protocol, scancode, toggle, 6);
 }
 
 static int get_key_pixelview(struct IR_i2c *ir, enum rc_type *protocol,
@@ -347,7 +377,7 @@ static int ir_probe(struct i2c_client *client, const struct i2c_device_id *id)
        case 0x71:
                name        = "Hauppauge/Zilog Z8";
                ir->get_key = get_key_haup_xvr;
-               rc_type     = RC_BIT_RC5;
+               rc_type     = RC_BIT_RC5 | RC_BIT_RC6_MCE | RC_BIT_RC6_6A_32;
                ir_codes    = RC_MAP_HAUPPAUGE;
                break;
        }