I.MX6 gpio-keys driver hacking

简介: /**************************************************************************** * I.MX6 gpio-keys driver hacking * 说明: * 1. 本文解读gpio-keys驱动是如何注册,最终处理函数在哪里。
/****************************************************************************
 *                   I.MX6 gpio-keys driver hacking
 * 说明:
 *     1. 本文解读gpio-keys驱动是如何注册,最终处理函数在哪里。
 *     2. 从最后生成的设备节点来看,我们直接可以通过操作该设备节点来来让系统
 *         进行相关操作,譬如关机、挂起等操作。
 *
 *                                          2016-3-17 深圳 南山平山村 曾剑锋
 ***************************************************************************/

static struct platform_driver gpio_keys_device_driver = {     <----+
    .probe      = gpio_keys_probe,                        ---------*-------+
    .remove     = __devexit_p(gpio_keys_remove),                   |       |
    .driver     = {                                                |       |
        .name   = "gpio-keys",                                     |       |
        .owner  = THIS_MODULE,                                     |       |
#ifdef CONFIG_PM                                                   |       |
        .pm = &gpio_keys_pm_ops,                                   |       |
#endif                                                             |       |
    }                                                              |       |
};                                                                 |       |
                                                                   |       |
static int __init gpio_keys_init(void)            <------------+   |       |
{                                                              |   |       |
    return platform_driver_register(&gpio_keys_device_driver); | --+       |
}                                                              |           |
                                                               |           |
static void __exit gpio_keys_exit(void)                        |           |
{                                                              |           |
    platform_driver_unregister(&gpio_keys_device_driver);      |           |
}                                                              |           |
                                                               |           |
module_init(gpio_keys_init);                      -------------+           |
module_exit(gpio_keys_exit);                                               |
                                                                           |
MODULE_LICENSE("GPL");                                                     |
MODULE_AUTHOR("Phil Blundell <pb@handhelds.org>");                         |
MODULE_DESCRIPTION("Keyboard driver for CPU GPIOs");                       |
MODULE_ALIAS("platform:gpio-keys");                                        |
                                                                           |
static int __devinit gpio_keys_probe(struct platform_device *pdev)   <-----+
{
    struct gpio_keys_platform_data *pdata = pdev->dev.platform_data;
    struct gpio_keys_drvdata *ddata;
    struct device *dev = &pdev->dev;
    struct input_dev *input;
    int i, error;
    int wakeup = 0;

    ddata = kzalloc(sizeof(struct gpio_keys_drvdata) +
            pdata->nbuttons * sizeof(struct gpio_button_data),
            GFP_KERNEL);
    input = input_allocate_device();
    if (!ddata || !input) {
        dev_err(dev, "failed to allocate state\n");
        error = -ENOMEM;
        goto fail1;
    }

    ddata->input = input;
    ddata->n_buttons = pdata->nbuttons;
    ddata->enable = pdata->enable;
    ddata->disable = pdata->disable;
    mutex_init(&ddata->disable_lock);

    platform_set_drvdata(pdev, ddata);
    input_set_drvdata(input, ddata);

    input->name = pdata->name ? : pdev->name;
    input->phys = "gpio-keys/input0";
    input->dev.parent = &pdev->dev;
    input->open = gpio_keys_open;
    input->close = gpio_keys_close;

    input->id.bustype = BUS_HOST;
    input->id.vendor = 0x0001;
    input->id.product = 0x0001;
    input->id.version = 0x0100;

    /* Enable auto repeat feature of Linux input subsystem */
    if (pdata->rep)
        __set_bit(EV_REP, input->evbit);

    for (i = 0; i < pdata->nbuttons; i++) {
        struct gpio_keys_button *button = &pdata->buttons[i];
        struct gpio_button_data *bdata = &ddata->data[i];
        unsigned int type = button->type ?: EV_KEY;

        bdata->input = input;
        bdata->button = button;

        error = gpio_keys_setup_key(pdev, bdata, button);         -------+
        if (error)                                                       |
            goto fail2;                                                  |
                                                                         |
        if (button->wakeup)                                              |
            wakeup = 1;                                                  |
                                                                         |
        input_set_capability(input, type, button->code);                 |
    }                                                                    |
                                                                         |
    error = sysfs_create_group(&pdev->dev.kobj, &gpio_keys_attr_group);  |
    if (error) {                                                         |
        dev_err(dev, "Unable to export keys/switches, error: %d\n",      |
            error);                                                      |
        goto fail2;                                                      |
    }                                                                    |
                                                                         |
    error = input_register_device(input);                                |
    if (error) {                                                         |
        dev_err(dev, "Unable to register input device, error: %d\n",     |
            error);                                                      |
        goto fail3;                                                      |
    }                                                                    |
                                                                         |
    /* get current state of buttons */                                   |
    for (i = 0; i < pdata->nbuttons; i++)                                |
        gpio_keys_report_event(&ddata->data[i]);                         |
    input_sync(input);                                                   |
                                                                         |
    device_init_wakeup(&pdev->dev, wakeup);                              |
                                                                         |
    return 0;                                                            |
                                                                         |
 fail3:                                                                  |
    sysfs_remove_group(&pdev->dev.kobj, &gpio_keys_attr_group);          |
 fail2:                                                                  |
    while (--i >= 0) {                                                   |
        free_irq(gpio_to_irq(pdata->buttons[i].gpio), &ddata->data[i]);  |
        if (ddata->data[i].timer_debounce)                               |
            del_timer_sync(&ddata->data[i].timer);                       |
        cancel_work_sync(&ddata->data[i].work);                          |
        gpio_free(pdata->buttons[i].gpio);                               |
    }                                                                    |
                                                                         |
    platform_set_drvdata(pdev, NULL);                                    |
 fail1:                                                                  |
    input_free_device(input);                                            |
    kfree(ddata);                                                        |
                                 +---------------------------------------+
    return error;                |
}                                |
                                 V
static int __devinit gpio_keys_setup_key(struct platform_device *pdev,
                     struct gpio_button_data *bdata,
                     struct gpio_keys_button *button)
{
    const char *desc = button->desc ? button->desc : "gpio_keys";
    struct device *dev = &pdev->dev;
    unsigned long irqflags;
    int irq, error;

    setup_timer(&bdata->timer, gpio_keys_timer, (unsigned long)bdata);
    INIT_WORK(&bdata->work, gpio_keys_work_func);           --------------------+
                                                                                |
    error = gpio_request(button->gpio, desc);                                   |
    if (error < 0) {                                                            |
        dev_err(dev, "failed to request GPIO %d, error %d\n",                   |
            button->gpio, error);                                               |
        goto fail2;                                                             |
    }                                                                           |
                                                                                |
    error = gpio_direction_input(button->gpio);                                 |
    if (error < 0) {                                                            |
        dev_err(dev, "failed to configure"                                      |
            " direction for GPIO %d, error %d\n",                               |
            button->gpio, error);                                               |
        goto fail3;                                                             |
    }                                                                           |
                                                                                |
    if (button->debounce_interval) {                                            |
        error = gpio_set_debounce(button->gpio,                                 |
                      button->debounce_interval * 1000);                        |
        /* use timer if gpiolib doesn't provide debounce */                     |
        if (error < 0)                                                          |
            bdata->timer_debounce = button->debounce_interval;                  |
    }                                                                           |
                                                                                |
    irq = gpio_to_irq(button->gpio);                                            |
    if (irq < 0) {                                                              |
        error = irq;                                                            |
        dev_err(dev, "Unable to get irq number for GPIO %d, error %d\n",        |
            button->gpio, error);                                               |
        goto fail3;                                                             |
    }                                                                           |
                                                                                |
    irqflags = IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING;                      |
    /*                                                                          |
     * If platform has specified that the button can be disabled,               |
     * we don't want it to share the interrupt line.                            |
     */                                                                         |
    if (!button->can_disable)                                                   |
        irqflags |= IRQF_SHARED;                                                |
    /*                                                                          |
     * Resume power key early during syscore instead of at device               |
     * resume time.                                                             |
     * Some platform like Android need to konw the power key is pressed         |
     * then to reume the other devcies                                          |
     */                                                                         |
    if (button->wakeup)                                                         |
        irqflags |= IRQF_NO_SUSPEND | IRQF_EARLY_RESUME;                        |
                                                                                |
    error = request_any_context_irq(irq, gpio_keys_isr, irqflags, desc, bdata); |
    if (error < 0) {                                                            |
        dev_err(dev, "Unable to claim irq %d; error %d\n",                      |
            irq, error);                                                        |
        goto fail3;                                                             |
    }                                                                           |
                                                                                |
    return 0;                                                                   |
                                                                                |
fail3:                                                                          |
    gpio_free(button->gpio);                                                    |
fail2:                                                                          |
    return error;                                                               |
}                                                                               |
                                                                                |
static void gpio_keys_work_func(struct work_struct *work)         <-------------+
{
    struct gpio_button_data *bdata =
        container_of(work, struct gpio_button_data, work);

    gpio_keys_report_event(bdata);                                -----------+
}                                                                            |
                                                                             |
static void gpio_keys_report_event(struct gpio_button_data *bdata) <---------+
{
    struct gpio_keys_button *button = bdata->button;
    struct input_dev *input = bdata->input;
    unsigned int type = button->type ?: EV_KEY;
    int state = (gpio_get_value_cansleep(button->gpio) ? 1 : 0) ^ button->active_low;
    printk("zengjf check gpio-keys positon: %s in line %d.\n", __func__, _LINE__);

    if (type == EV_ABS) {
        if (state)
            input_event(input, type, button->code, button->value);
    } else {
        input_event(input, type, button->code, !!state);
    }
    input_sync(input);
}



/**
 * root@android:/ # cat /proc/bus/input/devices                                   
 *    I: Bus=0019 Vendor=0001 Product=0001 Version=0100
 *    N: Name="gpio-keys"
 *    P: Phys=gpio-keys/input0
 *    S: Sysfs=/devices/platform/gpio-keys/input/input0
 *    U: Uniq=
 *    H: Handlers=event0 
 *    B: PROP=0
 *    B: EV=3
 *    B: KEY=100000 0 0 0
 *    ......
 */

 

目录
相关文章
|
Linux
I.MX6 driver goto 使用
/************************************************************************** * I.MX6 driver goto 使用 * 说明: * 在绝大多数地方,我们都被告诉尽可能不要用goto,甚至都没学过goto,但 * 这种语法却在内核驱动中普遍使用。
937 0
|
芯片
I.MX6 PMU MMPF0100 driver porting
/************************************************************************** * I.MX6 MMPF0100 driver porting * 说明: * 虽然主板上有MMPF0100芯片,却没有注册设备并使用该PMU驱动,真是浪费, * 当然因为需要,所以将PMU的驱动注册起来。
986 0
|
Linux
I.MX6 U-boot GPIO hacking
/******************************************************************************* * I.MX6 U-boot GPIO hacking * 说明: * 本文主要记录I.MX6 U-boot是如何设置GPIO口输入输出的,主要是考虑到这个阶段 * 并没有像Linux内核中的gpio_request一系列函数使用。
1089 0
|
Linux
OK335xS LAN8710 phy driver hacking
/******************************************************************** * OK335xS LAN8710 phy driver hacking * 说明: * 本文主要是对OK335xS中的phy的驱动进行代码跟踪,并解决当前遇到 * LAN8710上电后插入网线,会导致LAN8710无法自动握手,Link灯不亮,内核 * 也检测不到LAN8710有状态发生了改变,最终问题定位于LAN8710的驱动初 * 始化部分,本文解决办法选择注释掉对应的内容就行了。
1126 0
|
芯片
I.MX6 bq27441 driver hacking
/************************************************************************* * I.MX6 bq27441 driver hacking * 声明: * 本文主要是记录对电池计量芯片bq27441芯片驱动注册过程进行代码跟踪。
795 0
I.MX6 Goodix GT9xx touchscreen driver porting
/************************************************************************ * I.MX6 Goodix GT9xx touchscreen driver porting * 声明: * 本文主要是记录GT9xx电容Touch移植过程中遇到的一些问题。
2411 0
|
Android开发 SoC
I.MX6 Power off register hacking
/*********************************************************************** * I.MX6 Power off register hacking * 声明: * 本文主要记录I.MX6DL中的Power off按键的注册过程。
693 0
|
Linux 芯片
I.MX6 bq27441 driver porting
/************************************************************************** * I.MX6 bq27441 driver porting * 声明: * 本文主要记录bq27441电池检测芯片驱动遇到的一些问题以及解决方法。
885 0
|
机器学习/深度学习 定位技术 Android开发
I.MX6 GPS JNI HAL register init hacking
/************************************************************************************ * I.
1018 0
|
Linux 测试技术
I.MX6 PWM buzzer driver hacking with Demo test
/***************************************************************************** * I.MX6 PWM buzzer driver hacking with Demo test * 声明: * 1. I.MX6和OK335xS实现PWM驱动函数是不一样的; * 2. 通过分析PWM驱动,了解有哪些驱动函数可以用; * 3. 使用I.MX6提供的PWM函数,编写测试用例buzzer驱动; * 4. 使用C编写测试程序。
1031 0