亚洲av成人无遮挡网站在线观看,少妇性bbb搡bbb爽爽爽,亚洲av日韩精品久久久久久,兔费看少妇性l交大片免费,无码少妇一区二区三区

  免費注冊 查看新帖 |

Chinaunix

  平臺 論壇 博客 文庫
最近訪問板塊 發(fā)新帖
查看: 2360 | 回復: 3
打印 上一主題 下一主題

[硬件及驅動] 如何使得觸摸按鍵驅動正常運行? [復制鏈接]

論壇徽章:
0
跳轉到指定樓層
1 [收藏(0)] [報告]
發(fā)表于 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才能工作。

我的做法:首先我創(chuàng)建一個設備樹文件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 [報告]
發(fā)表于 2015-04-26 04:36 |只看該作者
本帖最后由 linuxfellow 于 2015-04-26 11:35 編輯

運行這個命令看看:
echo mpr121 0x5a > /sys/bus/i2c/devices/i2c-2/new_device

論壇徽章:
0
3 [報告]
發(fā)表于 2015-04-26 17:47 |只看該作者
本帖最后由 tomxue0126 于 2015-04-26 17:47 編輯

回復 2# linuxfellow

多謝!
執(zhí)行上述命令,然后就會多出一個目錄,如下:
  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
復制代碼
然后呢?里面貌似找不到有用的信息,我的硬件端觸摸按鍵觸發(fā)的話,也找不到對應的驅動輸出
而/dev/input/目錄下,還是只有一個mice文件

接下來應該做什么?



   

論壇徽章:
1
拜羊年徽章
日期:2015-03-03 16:15:43
4 [報告]
發(fā)表于 2015-04-27 05:20 |只看該作者
本帖最后由 linuxfellow 于 2015-04-27 05:28 編輯

/sys/bus/i2c/devices/i2c-2/2-005a 應該就是你的device
從i2c角度,你的設備和驅動都有
你的系統(tǒng)內核沒有配置input event, 添加后應該出現/dev/input/event*
然后你可以運行一個evtest程序,你觸摸屏幕時,相應的event就會打印在console里
您需要登錄后才可以回帖 登錄 | 注冊

本版積分規(guī)則 發(fā)表回復

  

北京盛拓優(yōu)訊信息技術有限公司. 版權所有 京ICP備16024965號-6 北京市公安局海淀分局網監(jiān)中心備案編號:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年舉報專區(qū)
中國互聯網協會會員  聯系我們:huangweiwei@itpub.net
感謝所有關心和支持過ChinaUnix的朋友們 轉載本站內容請注明原作者名及出處

清除 Cookies - ChinaUnix - Archiver - WAP - TOP