dlink-dir685-touchkeys.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. /*
  2. * D-Link DIR-685 router I2C-based Touchkeys input driver
  3. * Copyright (C) 2017 Linus Walleij <linus.walleij@linaro.org>
  4. *
  5. * This is a one-off touchkey controller based on the Cypress Semiconductor
  6. * CY8C214 MCU with some firmware in its internal 8KB flash. The circuit
  7. * board inside the router is named E119921
  8. */
  9. #include <linux/module.h>
  10. #include <linux/i2c.h>
  11. #include <linux/interrupt.h>
  12. #include <linux/delay.h>
  13. #include <linux/input.h>
  14. #include <linux/slab.h>
  15. #include <linux/bitops.h>
  16. struct dir685_touchkeys {
  17. struct device *dev;
  18. struct i2c_client *client;
  19. struct input_dev *input;
  20. unsigned long cur_key;
  21. u16 codes[7];
  22. };
  23. static irqreturn_t dir685_tk_irq_thread(int irq, void *data)
  24. {
  25. struct dir685_touchkeys *tk = data;
  26. const int num_bits = min_t(int, ARRAY_SIZE(tk->codes), 16);
  27. unsigned long changed;
  28. u8 buf[6];
  29. unsigned long key;
  30. int i;
  31. int err;
  32. memset(buf, 0, sizeof(buf));
  33. err = i2c_master_recv(tk->client, buf, sizeof(buf));
  34. if (err != sizeof(buf)) {
  35. dev_err(tk->dev, "short read %d\n", err);
  36. return IRQ_HANDLED;
  37. }
  38. dev_dbg(tk->dev, "IN: %*ph\n", (int)sizeof(buf), buf);
  39. key = be16_to_cpup((__be16 *) &buf[4]);
  40. /* Figure out if any bits went high or low since last message */
  41. changed = tk->cur_key ^ key;
  42. for_each_set_bit(i, &changed, num_bits) {
  43. dev_dbg(tk->dev, "key %d is %s\n", i,
  44. test_bit(i, &key) ? "down" : "up");
  45. input_report_key(tk->input, tk->codes[i], test_bit(i, &key));
  46. }
  47. /* Store currently down keys */
  48. tk->cur_key = key;
  49. input_sync(tk->input);
  50. return IRQ_HANDLED;
  51. }
  52. static int dir685_tk_probe(struct i2c_client *client,
  53. const struct i2c_device_id *id)
  54. {
  55. struct dir685_touchkeys *tk;
  56. struct device *dev = &client->dev;
  57. u8 bl_data[] = { 0xa7, 0x40 };
  58. int err;
  59. int i;
  60. tk = devm_kzalloc(&client->dev, sizeof(*tk), GFP_KERNEL);
  61. if (!tk)
  62. return -ENOMEM;
  63. tk->input = devm_input_allocate_device(dev);
  64. if (!tk->input)
  65. return -ENOMEM;
  66. tk->client = client;
  67. tk->dev = dev;
  68. tk->input->keycodesize = sizeof(u16);
  69. tk->input->keycodemax = ARRAY_SIZE(tk->codes);
  70. tk->input->keycode = tk->codes;
  71. tk->codes[0] = KEY_UP;
  72. tk->codes[1] = KEY_DOWN;
  73. tk->codes[2] = KEY_LEFT;
  74. tk->codes[3] = KEY_RIGHT;
  75. tk->codes[4] = KEY_ENTER;
  76. tk->codes[5] = KEY_WPS_BUTTON;
  77. /*
  78. * This key appears in the vendor driver, but I have
  79. * not been able to activate it.
  80. */
  81. tk->codes[6] = KEY_RESERVED;
  82. __set_bit(EV_KEY, tk->input->evbit);
  83. for (i = 0; i < ARRAY_SIZE(tk->codes); i++)
  84. __set_bit(tk->codes[i], tk->input->keybit);
  85. __clear_bit(KEY_RESERVED, tk->input->keybit);
  86. tk->input->name = "D-Link DIR-685 touchkeys";
  87. tk->input->id.bustype = BUS_I2C;
  88. err = input_register_device(tk->input);
  89. if (err)
  90. return err;
  91. /* Set the brightness to max level */
  92. err = i2c_master_send(client, bl_data, sizeof(bl_data));
  93. if (err != sizeof(bl_data))
  94. dev_warn(tk->dev, "error setting brightness level\n");
  95. if (!client->irq) {
  96. dev_err(dev, "no IRQ on the I2C device\n");
  97. return -ENODEV;
  98. }
  99. err = devm_request_threaded_irq(dev, client->irq,
  100. NULL, dir685_tk_irq_thread,
  101. IRQF_ONESHOT,
  102. "dir685-tk", tk);
  103. if (err) {
  104. dev_err(dev, "can't request IRQ\n");
  105. return err;
  106. }
  107. return 0;
  108. }
  109. static const struct i2c_device_id dir685_tk_id[] = {
  110. { "dir685tk", 0 },
  111. { }
  112. };
  113. MODULE_DEVICE_TABLE(i2c, dir685_tk_id);
  114. #ifdef CONFIG_OF
  115. static const struct of_device_id dir685_tk_of_match[] = {
  116. { .compatible = "dlink,dir685-touchkeys" },
  117. {},
  118. };
  119. MODULE_DEVICE_TABLE(of, dir685_tk_of_match);
  120. #endif
  121. static struct i2c_driver dir685_tk_i2c_driver = {
  122. .driver = {
  123. .name = "dlink-dir685-touchkeys",
  124. .of_match_table = of_match_ptr(dir685_tk_of_match),
  125. },
  126. .probe = dir685_tk_probe,
  127. .id_table = dir685_tk_id,
  128. };
  129. module_i2c_driver(dir685_tk_i2c_driver);
  130. MODULE_AUTHOR("Linus Walleij <linus.walleij@linaro.org>");
  131. MODULE_DESCRIPTION("D-Link DIR-685 touchkeys driver");
  132. MODULE_LICENSE("GPL");