2007-02-10 01:29:00 -05:00
/*
* atlas_btns . c - Atlas Wallmount Touchscreen ACPI Extras
*
* Copyright ( C ) 2006 Jaya Kumar
* Based on Toshiba ACPI by John Belmonte and ASUS ACPI
* This work was sponsored by CIS ( M ) Sdn Bhd .
*
* 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 ; either version 2 of the License , or
* ( at your option ) any later version .
*
* 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
*
*/
2010-07-13 09:33:20 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2007-02-10 01:29:00 -05:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/input.h>
# include <linux/types.h>
# include <asm/uaccess.h>
# include <acpi/acpi_drivers.h>
2007-11-04 00:41:30 -04:00
# define ACPI_ATLAS_NAME "Atlas ACPI"
# define ACPI_ATLAS_CLASS "Atlas"
2007-02-10 01:29:00 -05:00
2007-11-04 00:41:30 -04:00
static unsigned short atlas_keymap [ 16 ] ;
2007-02-10 01:29:00 -05:00
static struct input_dev * input_dev ;
/* button handling code */
static acpi_status acpi_atlas_button_setup ( acpi_handle region_handle ,
u32 function , void * handler_context , void * * return_context )
{
* return_context =
( function ! = ACPI_REGION_DEACTIVATE ) ? handler_context : NULL ;
return AE_OK ;
}
static acpi_status acpi_atlas_button_handler ( u32 function ,
acpi_physical_address address ,
2010-01-28 10:53:19 +08:00
u32 bit_width , u64 * value ,
2007-02-10 01:29:00 -05:00
void * handler_context , void * region_context )
{
acpi_status status ;
if ( function = = ACPI_WRITE ) {
2007-11-04 00:41:30 -04:00
int code = address & 0x0f ;
int key_down = ! ( address & 0x10 ) ;
input_event ( input_dev , EV_MSC , MSC_SCAN , code ) ;
input_report_key ( input_dev , atlas_keymap [ code ] , key_down ) ;
2007-02-10 01:29:00 -05:00
input_sync ( input_dev ) ;
2007-11-04 00:41:30 -04:00
2010-07-13 09:25:12 -07:00
status = AE_OK ;
2007-02-10 01:29:00 -05:00
} else {
2010-07-13 09:33:20 -07:00
pr_warn ( " shrugged on unexpected function: function=%x,address=%lx,value=%x \n " ,
2007-02-10 01:29:00 -05:00
function , ( unsigned long ) address , ( u32 ) * value ) ;
2010-07-13 09:25:12 -07:00
status = AE_BAD_PARAMETER ;
2007-02-10 01:29:00 -05:00
}
return status ;
}
static int atlas_acpi_button_add ( struct acpi_device * device )
{
acpi_status status ;
2007-11-04 00:41:30 -04:00
int i ;
2007-02-10 01:29:00 -05:00
int err ;
input_dev = input_allocate_device ( ) ;
if ( ! input_dev ) {
2010-07-13 09:33:20 -07:00
pr_err ( " unable to allocate input device \n " ) ;
2007-02-10 01:29:00 -05:00
return - ENOMEM ;
}
input_dev - > name = " Atlas ACPI button driver " ;
input_dev - > phys = " ASIM0000/atlas/input0 " ;
input_dev - > id . bustype = BUS_HOST ;
2007-11-04 00:41:30 -04:00
input_dev - > keycode = atlas_keymap ;
input_dev - > keycodesize = sizeof ( unsigned short ) ;
input_dev - > keycodemax = ARRAY_SIZE ( atlas_keymap ) ;
input_set_capability ( input_dev , EV_MSC , MSC_SCAN ) ;
__set_bit ( EV_KEY , input_dev - > evbit ) ;
for ( i = 0 ; i < ARRAY_SIZE ( atlas_keymap ) ; i + + ) {
if ( i < 9 ) {
atlas_keymap [ i ] = KEY_F1 + i ;
__set_bit ( KEY_F1 + i , input_dev - > keybit ) ;
} else
atlas_keymap [ i ] = KEY_RESERVED ;
}
2007-02-10 01:29:00 -05:00
err = input_register_device ( input_dev ) ;
if ( err ) {
2010-07-13 09:33:20 -07:00
pr_err ( " couldn't register input device \n " ) ;
2007-02-10 01:29:00 -05:00
input_free_device ( input_dev ) ;
return err ;
}
/* hookup button handler */
status = acpi_install_address_space_handler ( device - > handle ,
0x81 , & acpi_atlas_button_handler ,
& acpi_atlas_button_setup , device ) ;
if ( ACPI_FAILURE ( status ) ) {
2010-07-13 09:33:20 -07:00
pr_err ( " error installing addr spc handler \n " ) ;
2007-02-10 01:29:00 -05:00
input_unregister_device ( input_dev ) ;
2010-07-13 09:25:12 -07:00
err = - EINVAL ;
2007-02-10 01:29:00 -05:00
}
2010-07-13 09:25:12 -07:00
return err ;
2007-02-10 01:29:00 -05:00
}
2013-01-24 00:24:48 +01:00
static int atlas_acpi_button_remove ( struct acpi_device * device )
2007-02-10 01:29:00 -05:00
{
acpi_status status ;
status = acpi_remove_address_space_handler ( device - > handle ,
0x81 , & acpi_atlas_button_handler ) ;
2010-07-13 09:25:12 -07:00
if ( ACPI_FAILURE ( status ) )
2010-07-13 09:33:20 -07:00
pr_err ( " error removing addr spc handler \n " ) ;
2007-02-10 01:29:00 -05:00
input_unregister_device ( input_dev ) ;
2010-07-13 09:25:12 -07:00
return 0 ;
2007-02-10 01:29:00 -05:00
}
2007-07-23 14:44:41 +02:00
static const struct acpi_device_id atlas_device_ids [ ] = {
{ " ASIM0000 " , 0 } ,
{ " " , 0 } ,
} ;
MODULE_DEVICE_TABLE ( acpi , atlas_device_ids ) ;
2007-02-10 01:29:00 -05:00
static struct acpi_driver atlas_acpi_driver = {
. name = ACPI_ATLAS_NAME ,
. class = ACPI_ATLAS_CLASS ,
2010-07-13 09:13:23 -07:00
. owner = THIS_MODULE ,
2007-07-23 14:44:41 +02:00
. ids = atlas_device_ids ,
2007-02-10 01:29:00 -05:00
. ops = {
. add = atlas_acpi_button_add ,
. remove = atlas_acpi_button_remove ,
} ,
} ;
2012-09-07 10:31:44 +03:00
module_acpi_driver ( atlas_acpi_driver ) ;
2007-02-10 01:29:00 -05:00
MODULE_AUTHOR ( " Jaya Kumar " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Atlas button driver " ) ;