2011-02-07 22:45:55 +03:00
/*
* Power button driver for Medfield .
*
* Copyright ( C ) 2010 Intel Corp
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License along
* with this program ; if not , write to the Free Software Foundation , Inc . ,
* 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/slab.h>
# include <linux/platform_device.h>
# include <linux/input.h>
2011-04-06 18:44:37 +04:00
2011-02-07 22:45:55 +03:00
# include <asm/intel_scu_ipc.h>
# define DRIVER_NAME "msic_power_btn"
# define MSIC_PB_STATUS 0x3f
2011-04-06 18:44:37 +04:00
# define MSIC_PB_LEVEL (1 << 3) /* 1 - release, 0 - press */
2011-02-07 22:45:55 +03:00
static irqreturn_t mfld_pb_isr ( int irq , void * dev_id )
{
2011-04-06 18:44:37 +04:00
struct input_dev * input = dev_id ;
2011-02-07 22:45:55 +03:00
int ret ;
u8 pbstat ;
ret = intel_scu_ipc_ioread8 ( MSIC_PB_STATUS , & pbstat ) ;
2011-04-06 18:44:37 +04:00
if ( ret < 0 ) {
dev_err ( input - > dev . parent , " Read error %d while reading "
" MSIC_PB_STATUS \n " , ret ) ;
} else {
input_event ( input , EV_KEY , KEY_POWER ,
! ( pbstat & MSIC_PB_LEVEL ) ) ;
input_sync ( input ) ;
}
2011-02-07 22:45:55 +03:00
return IRQ_HANDLED ;
}
static int __devinit mfld_pb_probe ( struct platform_device * pdev )
{
struct input_dev * input ;
2011-04-06 18:44:37 +04:00
int irq = platform_get_irq ( pdev , 0 ) ;
2011-02-07 22:45:55 +03:00
int error ;
if ( irq < 0 )
return - EINVAL ;
input = input_allocate_device ( ) ;
2011-04-06 18:44:37 +04:00
if ( ! input ) {
dev_err ( & pdev - > dev , " Input device allocation error \n " ) ;
return - ENOMEM ;
2011-02-07 22:45:55 +03:00
}
input - > name = pdev - > name ;
input - > phys = " power-button/input0 " ;
input - > id . bustype = BUS_HOST ;
input - > dev . parent = & pdev - > dev ;
input_set_capability ( input , EV_KEY , KEY_POWER ) ;
2011-04-06 18:44:37 +04:00
error = request_threaded_irq ( irq , NULL , mfld_pb_isr , 0 ,
DRIVER_NAME , input ) ;
2011-02-07 22:45:55 +03:00
if ( error ) {
2011-04-06 18:44:37 +04:00
dev_err ( & pdev - > dev , " Unable to request irq %d for mfld power "
" button \n " , irq ) ;
goto err_free_input ;
2011-02-07 22:45:55 +03:00
}
error = input_register_device ( input ) ;
if ( error ) {
2011-04-06 18:44:37 +04:00
dev_err ( & pdev - > dev , " Unable to register input dev, error "
" %d \n " , error ) ;
2011-02-07 22:45:55 +03:00
goto err_free_irq ;
}
2011-04-06 18:44:37 +04:00
platform_set_drvdata ( pdev , input ) ;
2011-02-07 22:45:55 +03:00
return 0 ;
err_free_irq :
2011-04-06 18:44:37 +04:00
free_irq ( irq , input ) ;
err_free_input :
2011-02-07 22:45:55 +03:00
input_free_device ( input ) ;
return error ;
}
static int __devexit mfld_pb_remove ( struct platform_device * pdev )
{
2011-04-06 18:44:37 +04:00
struct input_dev * input = platform_get_drvdata ( pdev ) ;
int irq = platform_get_irq ( pdev , 0 ) ;
2011-02-07 22:45:55 +03:00
2011-04-06 18:44:37 +04:00
free_irq ( irq , input ) ;
input_unregister_device ( input ) ;
2011-02-07 22:45:55 +03:00
platform_set_drvdata ( pdev , NULL ) ;
2011-04-06 18:44:37 +04:00
2011-02-07 22:45:55 +03:00
return 0 ;
}
static struct platform_driver mfld_pb_driver = {
. driver = {
. name = DRIVER_NAME ,
. owner = THIS_MODULE ,
} ,
. probe = mfld_pb_probe ,
. remove = __devexit_p ( mfld_pb_remove ) ,
} ;
static int __init mfld_pb_init ( void )
{
return platform_driver_register ( & mfld_pb_driver ) ;
}
module_init ( mfld_pb_init ) ;
static void __exit mfld_pb_exit ( void )
{
platform_driver_unregister ( & mfld_pb_driver ) ;
}
module_exit ( mfld_pb_exit ) ;
MODULE_AUTHOR ( " Hong Liu <hong.liu@intel.com> " ) ;
MODULE_DESCRIPTION ( " Intel Medfield Power Button Driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS ( " platform: " DRIVER_NAME ) ;