免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2323 | 回复: 3
打印 上一主题 下一主题

[硬件及驱动] 如何使得触摸按键驱动正常运行? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-04-26 01:11 |只看该作者 |倒序浏览

我的平台: Beagle Bone Black, Ubuntu, kernel: 3.14.29-ti-r46
触摸芯片: MPR121, connect to Beagle's i2c-2

我需要使触摸按键芯片MPR121 (Freescal) 工作在我的 Beagle Bone Black上. 内核驱动是现成的,但是它需要结合device tree才能工作。

我的做法:首先我创建一个设备树文件dts file 用于支持MPR121,如下:
  1. tomxue@ubuntu:~/Tom/Source_Code/BBB/Ubuntu/ti-linux-kernel-dev/KERNEL/arch/arm/boot/dts$ cat am335x-bone-i2c2-mpr121.dts
  2. /*
  3. * Copyright (C) 2012 Texas Instruments Incorporated -
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2 as
  7. * published by the Free Software Foundation.
  8. */
  9. /dts-v1/;

  10. #include "am33xx.dtsi"
  11. #include "am335x-bone-common.dtsi"
  12. #include "am335x-bone-common-pinmux.dtsi"

  13. / {
  14.         model = "TI AM335x BeagleBone Black";
  15.         compatible = "ti,am335x-bone-black", "ti,am335x-bone", "ti,am33xx";
  16. };

  17. &ocp {
  18.         /* i2c2 */
  19.         P9_19_pinmux {
  20.                 mode = "i2c";
  21.         };
  22.         P9_20_pinmux {
  23.                 mode = "i2c";
  24.         };
  25. };

  26. &i2c2 {
  27.         status = "okay";
  28.         pinctrl-names = "default";
  29.         pinctrl-0 = <>; /* pinctrl-0 = <&i2c2_pins>; */

  30.         clock-frequency = <100000>;

  31.         touchkey: mpr121@5a {
  32.                 compatible = "fsl,mpr121";
  33.                 reg = <0x5a>;
  34.         };
  35. };

  36. #include "am335x-bone-i2c2-cape-eeprom.dtsi"
复制代码
对应的Makefile(在dts目录内):
  1. dtb-$(CONFIG_ARCH_OMAP2PLUS) += omap2420-h4.dtb \
  2.     ...
  3.     am335x-base0033.dtb \
  4.     am335x-bone-i2c2-mpr121.dtb \
  5.     am3517-craneboard.dtb \
复制代码
然后我就重新编译内核+设备树,使用BeagleBone提供的脚本烧写内核镜像,模块和设备树。
重启之后,先看看内核版本:
  1. root@arm:/dev/input# uname -a
  2. Linux arm 3.14.29-ti-r46 #2 SMP PREEMPT Tue Apr 21 14:30:14 CST 2015 armv7l armv7l armv7l GNU/Linux
复制代码
加载相应的模块:
  1. root@arm:~# insmod mpr121_touchkey.ko
复制代码
可是dmesg中没有任何关于mpr121的信息。我又检查了一下i2c-2总线,mpr121已经连在上面了,地址是0x5a,和设备树中设置的一样:

  1. root@arm:~# i2cdetect -y -r 2
  2.      0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f
  3. 00:          -- -- -- -- -- -- -- -- -- -- -- -- --
  4. 10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
  5. 20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
  6. 30: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
  7. 40: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
  8. 50: -- -- -- -- UU UU UU UU -- -- 5a -- -- -- -- --
  9. 60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
  10. 70: -- -- -- -- -- -- -- --
复制代码
最后,我看了一下这个目录,按照内核模块源码显示,它应该在input目录下生成相应的设备文件,但是实际上找不到。
  1. root@arm:/dev/input# ls
  2. mice
复制代码
这到底是怎么回事?我需要怎么做才能把该驱动模块正常跑起来?
谢谢!


补充下,内核模块文件为mpr121_touchkey.c ,在这里:drivers/input/keyboard,其源码如下:
  1. /*
  2. * Touchkey driver for Freescale MPR121 Controllor
  3. *
  4. * Copyright (C) 2011 Freescale Semiconductor, Inc.
  5. * Author: Zhang Jiejing <jiejing.zhang@freescale.com>
  6. *
  7. * Based on mcs_touchkey.c
  8. *
  9. * This program is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License version 2 as
  11. * published by the Free Software Foundation.
  12. *
  13. */

  14. #include <linux/module.h>
  15. #include <linux/input.h>
  16. #include <linux/i2c.h>
  17. #include <linux/slab.h>
  18. #include <linux/delay.h>
  19. #include <linux/bitops.h>
  20. #include <linux/interrupt.h>
  21. #include <linux/i2c/mpr121_touchkey.h>

  22. /* Register definitions */
  23. #define ELE_TOUCH_STATUS_0_ADDR        0x0
  24. #define ELE_TOUCH_STATUS_1_ADDR        0X1
  25. #define MHD_RISING_ADDR                0x2b
  26. #define NHD_RISING_ADDR                0x2c
  27. #define NCL_RISING_ADDR                0x2d
  28. #define FDL_RISING_ADDR                0x2e
  29. #define MHD_FALLING_ADDR        0x2f
  30. #define NHD_FALLING_ADDR        0x30
  31. #define NCL_FALLING_ADDR        0x31
  32. #define FDL_FALLING_ADDR        0x32
  33. #define ELE0_TOUCH_THRESHOLD_ADDR        0x41
  34. #define ELE0_RELEASE_THRESHOLD_ADDR        0x42
  35. #define AFE_CONF_ADDR                        0x5c
  36. #define FILTER_CONF_ADDR                0x5d

  37. /*
  38. * ELECTRODE_CONF_ADDR: This register configures the number of
  39. * enabled capacitance sensing inputs and its run/suspend mode.
  40. */
  41. #define ELECTRODE_CONF_ADDR                0x5e
  42. #define ELECTRODE_CONF_QUICK_CHARGE        0x80
  43. #define AUTO_CONFIG_CTRL_ADDR                0x7b
  44. #define AUTO_CONFIG_USL_ADDR                0x7d
  45. #define AUTO_CONFIG_LSL_ADDR                0x7e
  46. #define AUTO_CONFIG_TL_ADDR                0x7f

  47. /* Threshold of touch/release trigger */
  48. #define TOUCH_THRESHOLD                        0x08
  49. #define RELEASE_THRESHOLD                0x05
  50. /* Masks for touch and release triggers */
  51. #define TOUCH_STATUS_MASK                0xfff
  52. /* MPR121 has 12 keys */
  53. #define MPR121_MAX_KEY_COUNT                12

  54. struct mpr121_touchkey {
  55.         struct i2c_client        *client;
  56.         struct input_dev        *input_dev;
  57.         unsigned int                key_val;
  58.         unsigned int                statusbits;
  59.         unsigned int                keycount;
  60.         u16                        keycodes[MPR121_MAX_KEY_COUNT];
  61. };

  62. struct mpr121_init_register {
  63.         int addr;
  64.         u8 val;
  65. };

  66. static const struct mpr121_init_register init_reg_table[] = {
  67.         { MHD_RISING_ADDR,        0x1 },
  68.         { NHD_RISING_ADDR,        0x1 },
  69.         { MHD_FALLING_ADDR,        0x1 },
  70.         { NHD_FALLING_ADDR,        0x1 },
  71.         { NCL_FALLING_ADDR,        0xff },
  72.         { FDL_FALLING_ADDR,        0x02 },
  73.         { FILTER_CONF_ADDR,        0x04 },
  74.         { AFE_CONF_ADDR,        0x0b },
  75.         { AUTO_CONFIG_CTRL_ADDR, 0x0b },
  76. };

  77. static irqreturn_t mpr_touchkey_interrupt(int irq, void *dev_id)
  78. {
  79.         struct mpr121_touchkey *mpr121 = dev_id;
  80.         struct i2c_client *client = mpr121->client;
  81.         struct input_dev *input = mpr121->input_dev;
  82.         unsigned int key_num, key_val, pressed;
  83.         int reg;

  84.         reg = i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_1_ADDR);
  85.         if (reg < 0) {
  86.                 dev_err(&client->dev, "i2c read error [%d]\n", reg);
  87.                 goto out;
  88.         }

  89.         reg <<= 8;
  90.         reg |= i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_0_ADDR);
  91.         if (reg < 0) {
  92.                 dev_err(&client->dev, "i2c read error [%d]\n", reg);
  93.                 goto out;
  94.         }

  95.         reg &= TOUCH_STATUS_MASK;
  96.         /* use old press bit to figure out which bit changed */
  97.         key_num = ffs(reg ^ mpr121->statusbits) - 1;
  98.         pressed = reg & (1 << key_num);
  99.         mpr121->statusbits = reg;

  100.         key_val = mpr121->keycodes[key_num];

  101.         input_event(input, EV_MSC, MSC_SCAN, key_num);
  102.         input_report_key(input, key_val, pressed);
  103.         input_sync(input);

  104.         dev_dbg(&client->dev, "key %d %d %s\n", key_num, key_val,
  105.                 pressed ? "pressed" : "released");

  106. out:
  107.         return IRQ_HANDLED;
  108. }

  109. static int mpr121_phys_init(const struct mpr121_platform_data *pdata,
  110.                                       struct mpr121_touchkey *mpr121,
  111.                                       struct i2c_client *client)
  112. {
  113.         const struct mpr121_init_register *reg;
  114.         unsigned char usl, lsl, tl, eleconf;
  115.         int i, t, vdd, ret;

  116.         /* Set up touch/release threshold for ele0-ele11 */
  117.         for (i = 0; i <= MPR121_MAX_KEY_COUNT; i++) {
  118.                 t = ELE0_TOUCH_THRESHOLD_ADDR + (i * 2);
  119.                 ret = i2c_smbus_write_byte_data(client, t, TOUCH_THRESHOLD);
  120.                 if (ret < 0)
  121.                         goto err_i2c_write;
  122.                 ret = i2c_smbus_write_byte_data(client, t + 1,
  123.                                                 RELEASE_THRESHOLD);
  124.                 if (ret < 0)
  125.                         goto err_i2c_write;
  126.         }

  127.         /* Set up init register */
  128.         for (i = 0; i < ARRAY_SIZE(init_reg_table); i++) {
  129.                 reg = &init_reg_table[i];
  130.                 ret = i2c_smbus_write_byte_data(client, reg->addr, reg->val);
  131.                 if (ret < 0)
  132.                         goto err_i2c_write;
  133.         }


  134.         /*
  135.          * Capacitance on sensing input varies and needs to be compensated.
  136.          * The internal MPR121-auto-configuration can do this if it's
  137.          * registers are set properly (based on pdata->vdd_uv).
  138.          */
  139.         vdd = pdata->vdd_uv / 1000;
  140.         usl = ((vdd - 700) * 256) / vdd;
  141.         lsl = (usl * 65) / 100;
  142.         tl = (usl * 90) / 100;
  143.         ret = i2c_smbus_write_byte_data(client, AUTO_CONFIG_USL_ADDR, usl);
  144.         ret |= i2c_smbus_write_byte_data(client, AUTO_CONFIG_LSL_ADDR, lsl);
  145.         ret |= i2c_smbus_write_byte_data(client, AUTO_CONFIG_TL_ADDR, tl);

  146.         /*
  147.          * Quick charge bit will let the capacitive charge to ready
  148.          * state quickly, or the buttons may not function after system
  149.          * boot.
  150.          */
  151.         eleconf = mpr121->keycount | ELECTRODE_CONF_QUICK_CHARGE;
  152.         ret |= i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR,
  153.                                          eleconf);
  154.         if (ret != 0)
  155.                 goto err_i2c_write;

  156.         dev_dbg(&client->dev, "set up with %x keys.\n", mpr121->keycount);

  157.         return 0;

  158. err_i2c_write:
  159.         dev_err(&client->dev, "i2c write error: %d\n", ret);
  160.         return ret;
  161. }

  162. static int mpr_touchkey_probe(struct i2c_client *client,
  163.                               const struct i2c_device_id *id)
  164. {
  165.         const struct mpr121_platform_data *pdata =
  166.                         dev_get_platdata(&client->dev);
  167.         struct mpr121_touchkey *mpr121;
  168.         struct input_dev *input_dev;
  169.         int error;
  170.         int i;

  171.         if (!pdata) {
  172.                 dev_err(&client->dev, "no platform data defined\n");
  173.                 return -EINVAL;
  174.         }

  175.         if (!pdata->keymap || !pdata->keymap_size) {
  176.                 dev_err(&client->dev, "missing keymap data\n");
  177.                 return -EINVAL;
  178.         }

  179.         if (pdata->keymap_size > MPR121_MAX_KEY_COUNT) {
  180.                 dev_err(&client->dev, "too many keys defined\n");
  181.                 return -EINVAL;
  182.         }

  183.         if (!client->irq) {
  184.                 dev_err(&client->dev, "irq number should not be zero\n");
  185.                 return -EINVAL;
  186.         }

  187.         mpr121 = kzalloc(sizeof(struct mpr121_touchkey), GFP_KERNEL);
  188.         input_dev = input_allocate_device();
  189.         if (!mpr121 || !input_dev) {
  190.                 dev_err(&client->dev, "Failed to allocate memory\n");
  191.                 error = -ENOMEM;
  192.                 goto err_free_mem;
  193.         }

  194.         mpr121->client = client;
  195.         mpr121->input_dev = input_dev;
  196.         mpr121->keycount = pdata->keymap_size;

  197.         input_dev->name = "Freescale MPR121 Touchkey";
  198.         input_dev->id.bustype = BUS_I2C;
  199.         input_dev->dev.parent = &client->dev;
  200.         input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REP);

  201.         input_dev->keycode = mpr121->keycodes;
  202.         input_dev->keycodesize = sizeof(mpr121->keycodes[0]);
  203.         input_dev->keycodemax = mpr121->keycount;

  204.         for (i = 0; i < pdata->keymap_size; i++) {
  205.                 input_set_capability(input_dev, EV_KEY, pdata->keymap[i]);
  206.                 mpr121->keycodes[i] = pdata->keymap[i];
  207.         }

  208.         error = mpr121_phys_init(pdata, mpr121, client);
  209.         if (error) {
  210.                 dev_err(&client->dev, "Failed to init register\n");
  211.                 goto err_free_mem;
  212.         }

  213.         error = request_threaded_irq(client->irq, NULL,
  214.                                      mpr_touchkey_interrupt,
  215.                                      IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
  216.                                      client->dev.driver->name, mpr121);
  217.         if (error) {
  218.                 dev_err(&client->dev, "Failed to register interrupt\n");
  219.                 goto err_free_mem;
  220.         }

  221.         error = input_register_device(input_dev);
  222.         if (error)
  223.                 goto err_free_irq;

  224.         i2c_set_clientdata(client, mpr121);
  225.         device_init_wakeup(&client->dev, pdata->wakeup);

  226.         return 0;

  227. err_free_irq:
  228.         free_irq(client->irq, mpr121);
  229. err_free_mem:
  230.         input_free_device(input_dev);
  231.         kfree(mpr121);
  232.         return error;
  233. }

  234. static int mpr_touchkey_remove(struct i2c_client *client)
  235. {
  236.         struct mpr121_touchkey *mpr121 = i2c_get_clientdata(client);

  237.         free_irq(client->irq, mpr121);
  238.         input_unregister_device(mpr121->input_dev);
  239.         kfree(mpr121);

  240.         return 0;
  241. }

  242. #ifdef CONFIG_PM_SLEEP
  243. static int mpr_suspend(struct device *dev)
  244. {
  245.         struct i2c_client *client = to_i2c_client(dev);

  246.         if (device_may_wakeup(&client->dev))
  247.                 enable_irq_wake(client->irq);

  248.         i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR, 0x00);

  249.         return 0;
  250. }

  251. static int mpr_resume(struct device *dev)
  252. {
  253.         struct i2c_client *client = to_i2c_client(dev);
  254.         struct mpr121_touchkey *mpr121 = i2c_get_clientdata(client);

  255.         if (device_may_wakeup(&client->dev))
  256.                 disable_irq_wake(client->irq);

  257.         i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR,
  258.                                   mpr121->keycount);

  259.         return 0;
  260. }
  261. #endif

  262. static SIMPLE_DEV_PM_OPS(mpr121_touchkey_pm_ops, mpr_suspend, mpr_resume);

  263. static const struct i2c_device_id mpr121_id[] = {
  264.         { "mpr121_touchkey", 0 },
  265.         { }
  266. };
  267. MODULE_DEVICE_TABLE(i2c, mpr121_id);

  268. static struct i2c_driver mpr_touchkey_driver = {
  269.         .driver = {
  270.                 .name        = "mpr121",
  271.                 .owner        = THIS_MODULE,
  272.                 .pm        = &mpr121_touchkey_pm_ops,
  273.         },
  274.         .id_table        = mpr121_id,
  275.         .probe                = mpr_touchkey_probe,
  276.         .remove                = mpr_touchkey_remove,
  277. };

  278. module_i2c_driver(mpr_touchkey_driver);

  279. MODULE_LICENSE("GPL");
  280. MODULE_AUTHOR("Zhang Jiejing <jiejing.zhang@freescale.com>");
  281. MODULE_DESCRIPTION("Touch Key driver for Freescale MPR121 Chip");
复制代码

论坛徽章:
1
拜羊年徽章
日期:2015-03-03 16:15:43
2 [报告]
发表于 2015-04-26 04:36 |只看该作者
本帖最后由 linuxfellow 于 2015-04-26 11:35 编辑

运行这个命令看看:
echo mpr121 0x5a > /sys/bus/i2c/devices/i2c-2/new_device

论坛徽章:
0
3 [报告]
发表于 2015-04-26 17:47 |只看该作者
本帖最后由 tomxue0126 于 2015-04-26 17:47 编辑

回复 2# linuxfellow

多谢!
执行上述命令,然后就会多出一个目录,如下:
  1. i2c-2# echo mpr121 0x5a > /sys/bus/i2c/devices/i2c
  2. root@arm:/sys/bus/i2c/devices/i2c-2# ls
  3. 2-0054  2-0056  2-005a         device   name        power      uevent
  4. 2-0055  2-0057  delete_device  i2c-dev  new_device  subsystem
  5. root@arm:/sys/bus/i2c/devices/i2c-2# cd 2-005a/
  6. root@arm:/sys/bus/i2c/devices/i2c-2/2-005a# ls
  7. modalias  name  power  subsystem  uevent
复制代码
然后呢?里面貌似找不到有用的信息,我的硬件端触摸按键触发的话,也找不到对应的驱动输出
而/dev/input/目录下,还是只有一个mice文件

接下来应该做什么?



   

论坛徽章:
1
拜羊年徽章
日期:2015-03-03 16:15:43
4 [报告]
发表于 2015-04-27 05:20 |只看该作者
本帖最后由 linuxfellow 于 2015-04-27 05:28 编辑

/sys/bus/i2c/devices/i2c-2/2-005a 应该就是你的device
从i2c角度,你的设备和驱动都有
你的系统内核没有配置input event, 添加后应该出现/dev/input/event*
然后你可以运行一个evtest程序,你触摸屏幕时,相应的event就会打印在console里
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP