Path: blob/master/drivers/input/misc/ab8500-ponkey.c
15112 views
/*1* Copyright (C) ST-Ericsson SA 20102*3* License Terms: GNU General Public License v24* Author: Sundar Iyer <[email protected]> for ST-Ericsson5*6* AB8500 Power-On Key handler7*/89#include <linux/kernel.h>10#include <linux/module.h>11#include <linux/platform_device.h>12#include <linux/input.h>13#include <linux/interrupt.h>14#include <linux/mfd/ab8500.h>15#include <linux/slab.h>1617/**18* struct ab8500_ponkey - ab8500 ponkey information19* @input_dev: pointer to input device20* @ab8500: ab8500 parent21* @irq_dbf: irq number for falling transition22* @irq_dbr: irq number for rising transition23*/24struct ab8500_ponkey {25struct input_dev *idev;26struct ab8500 *ab8500;27int irq_dbf;28int irq_dbr;29};3031/* AB8500 gives us an interrupt when ONKEY is held */32static irqreturn_t ab8500_ponkey_handler(int irq, void *data)33{34struct ab8500_ponkey *ponkey = data;3536if (irq == ponkey->irq_dbf)37input_report_key(ponkey->idev, KEY_POWER, true);38else if (irq == ponkey->irq_dbr)39input_report_key(ponkey->idev, KEY_POWER, false);4041input_sync(ponkey->idev);4243return IRQ_HANDLED;44}4546static int __devinit ab8500_ponkey_probe(struct platform_device *pdev)47{48struct ab8500 *ab8500 = dev_get_drvdata(pdev->dev.parent);49struct ab8500_ponkey *ponkey;50struct input_dev *input;51int irq_dbf, irq_dbr;52int error;5354irq_dbf = platform_get_irq_byname(pdev, "ONKEY_DBF");55if (irq_dbf < 0) {56dev_err(&pdev->dev, "No IRQ for ONKEY_DBF, error=%d\n", irq_dbf);57return irq_dbf;58}5960irq_dbr = platform_get_irq_byname(pdev, "ONKEY_DBR");61if (irq_dbr < 0) {62dev_err(&pdev->dev, "No IRQ for ONKEY_DBR, error=%d\n", irq_dbr);63return irq_dbr;64}6566ponkey = kzalloc(sizeof(struct ab8500_ponkey), GFP_KERNEL);67input = input_allocate_device();68if (!ponkey || !input) {69error = -ENOMEM;70goto err_free_mem;71}7273ponkey->idev = input;74ponkey->ab8500 = ab8500;75ponkey->irq_dbf = irq_dbf;76ponkey->irq_dbr = irq_dbr;7778input->name = "AB8500 POn(PowerOn) Key";79input->dev.parent = &pdev->dev;8081input_set_capability(input, EV_KEY, KEY_POWER);8283error = request_any_context_irq(ponkey->irq_dbf, ab8500_ponkey_handler,840, "ab8500-ponkey-dbf", ponkey);85if (error < 0) {86dev_err(ab8500->dev, "Failed to request dbf IRQ#%d: %d\n",87ponkey->irq_dbf, error);88goto err_free_mem;89}9091error = request_any_context_irq(ponkey->irq_dbr, ab8500_ponkey_handler,920, "ab8500-ponkey-dbr", ponkey);93if (error < 0) {94dev_err(ab8500->dev, "Failed to request dbr IRQ#%d: %d\n",95ponkey->irq_dbr, error);96goto err_free_dbf_irq;97}9899error = input_register_device(ponkey->idev);100if (error) {101dev_err(ab8500->dev, "Can't register input device: %d\n", error);102goto err_free_dbr_irq;103}104105platform_set_drvdata(pdev, ponkey);106return 0;107108err_free_dbr_irq:109free_irq(ponkey->irq_dbr, ponkey);110err_free_dbf_irq:111free_irq(ponkey->irq_dbf, ponkey);112err_free_mem:113input_free_device(input);114kfree(ponkey);115116return error;117}118119static int __devexit ab8500_ponkey_remove(struct platform_device *pdev)120{121struct ab8500_ponkey *ponkey = platform_get_drvdata(pdev);122123free_irq(ponkey->irq_dbf, ponkey);124free_irq(ponkey->irq_dbr, ponkey);125input_unregister_device(ponkey->idev);126kfree(ponkey);127128platform_set_drvdata(pdev, NULL);129130return 0;131}132133static struct platform_driver ab8500_ponkey_driver = {134.driver = {135.name = "ab8500-poweron-key",136.owner = THIS_MODULE,137},138.probe = ab8500_ponkey_probe,139.remove = __devexit_p(ab8500_ponkey_remove),140};141142static int __init ab8500_ponkey_init(void)143{144return platform_driver_register(&ab8500_ponkey_driver);145}146module_init(ab8500_ponkey_init);147148static void __exit ab8500_ponkey_exit(void)149{150platform_driver_unregister(&ab8500_ponkey_driver);151}152module_exit(ab8500_ponkey_exit);153154MODULE_LICENSE("GPL v2");155MODULE_AUTHOR("Sundar Iyer <[email protected]>");156MODULE_DESCRIPTION("ST-Ericsson AB8500 Power-ON(Pon) Key driver");157158159