2005-03-18 18:45:35 -05:00
/*
* Link physical devices with ACPI devices support
*
* Copyright ( c ) 2005 David Shaohua Li < shaohua . li @ intel . com >
* Copyright ( c ) 2005 Intel Corp .
*
* This file is released under the GPLv2 .
*/
# include <linux/init.h>
# include <linux/list.h>
# include <linux/device.h>
# include <linux/rwsem.h>
# include <linux/acpi.h>
2009-07-28 16:45:54 -04:00
# include "internal.h"
2005-03-18 18:45:35 -05:00
# define ACPI_GLUE_DEBUG 0
# if ACPI_GLUE_DEBUG
# define DBG(x...) printk(PREFIX x)
# else
2007-07-09 11:33:14 -07:00
# define DBG(x...) do { } while(0)
2005-03-18 18:45:35 -05:00
# endif
static LIST_HEAD ( bus_type_list ) ;
static DECLARE_RWSEM ( bus_type_sem ) ;
int register_acpi_bus_type ( struct acpi_bus_type * type )
{
if ( acpi_disabled )
return - ENODEV ;
if ( type & & type - > bus & & type - > find_device ) {
down_write ( & bus_type_sem ) ;
list_add_tail ( & type - > list , & bus_type_list ) ;
up_write ( & bus_type_sem ) ;
2005-08-05 00:44:28 -04:00
printk ( KERN_INFO PREFIX " bus type %s registered \n " ,
type - > bus - > name ) ;
2005-03-18 18:45:35 -05:00
return 0 ;
}
return - ENODEV ;
}
int unregister_acpi_bus_type ( struct acpi_bus_type * type )
{
if ( acpi_disabled )
return 0 ;
if ( type ) {
down_write ( & bus_type_sem ) ;
list_del_init ( & type - > list ) ;
up_write ( & bus_type_sem ) ;
2005-08-05 00:44:28 -04:00
printk ( KERN_INFO PREFIX " ACPI bus type %s unregistered \n " ,
type - > bus - > name ) ;
2005-03-18 18:45:35 -05:00
return 0 ;
}
return - ENODEV ;
}
static struct acpi_bus_type * acpi_get_bus_type ( struct bus_type * type )
{
struct acpi_bus_type * tmp , * ret = NULL ;
down_read ( & bus_type_sem ) ;
list_for_each_entry ( tmp , & bus_type_list , list ) {
if ( tmp - > bus = = type ) {
ret = tmp ;
break ;
}
}
up_read ( & bus_type_sem ) ;
return ret ;
}
static int acpi_find_bridge_device ( struct device * dev , acpi_handle * handle )
{
struct acpi_bus_type * tmp ;
int ret = - ENODEV ;
down_read ( & bus_type_sem ) ;
list_for_each_entry ( tmp , & bus_type_list , list ) {
if ( tmp - > find_bridge & & ! tmp - > find_bridge ( dev , handle ) ) {
ret = 0 ;
break ;
}
}
up_read ( & bus_type_sem ) ;
return ret ;
}
/* Get device's handler per its address under its parent */
struct acpi_find_child {
acpi_handle handle ;
acpi_integer address ;
} ;
static acpi_status
do_acpi_find_child ( acpi_handle handle , u32 lvl , void * context , void * * rv )
{
acpi_status status ;
struct acpi_device_info * info ;
2006-10-01 00:28:50 +02:00
struct acpi_find_child * find = context ;
2005-03-18 18:45:35 -05:00
2009-06-29 13:39:29 +08:00
status = acpi_get_object_info ( handle , & info ) ;
2005-03-18 18:45:35 -05:00
if ( ACPI_SUCCESS ( status ) ) {
if ( info - > address = = find - > address )
find - > handle = handle ;
2009-06-29 13:39:29 +08:00
kfree ( info ) ;
2005-03-18 18:45:35 -05:00
}
return AE_OK ;
}
acpi_handle acpi_get_child ( acpi_handle parent , acpi_integer address )
{
struct acpi_find_child find = { NULL , address } ;
if ( ! parent )
return NULL ;
acpi_walk_namespace ( ACPI_TYPE_DEVICE , parent ,
1 , do_acpi_find_child , & find , NULL ) ;
return find . handle ;
}
EXPORT_SYMBOL ( acpi_get_child ) ;
/* Link ACPI devices with physical devices */
static void acpi_glue_data_handler ( acpi_handle handle ,
2009-06-29 13:43:27 +08:00
void * context )
2005-03-18 18:45:35 -05:00
{
/* we provide an empty handler */
}
/* Note: a success call will increase reference count by one */
struct device * acpi_get_physical_device ( acpi_handle handle )
{
acpi_status status ;
struct device * dev ;
status = acpi_get_data ( handle , acpi_glue_data_handler , ( void * * ) & dev ) ;
if ( ACPI_SUCCESS ( status ) )
return get_device ( dev ) ;
return NULL ;
}
EXPORT_SYMBOL ( acpi_get_physical_device ) ;
static int acpi_bind_one ( struct device * dev , acpi_handle handle )
{
2008-02-22 21:54:24 -08:00
struct acpi_device * acpi_dev ;
2005-03-18 18:45:35 -05:00
acpi_status status ;
2006-11-11 17:18:42 +11:00
if ( dev - > archdata . acpi_handle ) {
2008-05-02 06:02:41 +02:00
dev_warn ( dev , " Drivers changed 'acpi_handle' \n " ) ;
2005-03-18 18:45:35 -05:00
return - EINVAL ;
}
get_device ( dev ) ;
status = acpi_attach_data ( handle , acpi_glue_data_handler , dev ) ;
if ( ACPI_FAILURE ( status ) ) {
put_device ( dev ) ;
return - EINVAL ;
}
2006-11-11 17:18:42 +11:00
dev - > archdata . acpi_handle = handle ;
2005-03-18 18:45:35 -05:00
2008-02-22 21:54:24 -08:00
status = acpi_bus_get_device ( handle , & acpi_dev ) ;
if ( ! ACPI_FAILURE ( status ) ) {
int ret ;
ret = sysfs_create_link ( & dev - > kobj , & acpi_dev - > dev . kobj ,
" firmware_node " ) ;
ret = sysfs_create_link ( & acpi_dev - > dev . kobj , & dev - > kobj ,
" physical_node " ) ;
2008-10-03 15:23:49 -07:00
if ( acpi_dev - > wakeup . flags . valid ) {
2008-07-07 03:34:48 +02:00
device_set_wakeup_capable ( dev , true ) ;
2008-10-03 15:23:49 -07:00
device_set_wakeup_enable ( dev ,
acpi_dev - > wakeup . state . enabled ) ;
}
2008-02-22 21:54:24 -08:00
}
2005-03-18 18:45:35 -05:00
return 0 ;
}
static int acpi_unbind_one ( struct device * dev )
{
2006-11-11 17:18:42 +11:00
if ( ! dev - > archdata . acpi_handle )
2005-03-18 18:45:35 -05:00
return 0 ;
2006-11-11 17:18:42 +11:00
if ( dev = = acpi_get_physical_device ( dev - > archdata . acpi_handle ) ) {
2008-02-22 21:54:24 -08:00
struct acpi_device * acpi_dev ;
2005-03-18 18:45:35 -05:00
/* acpi_get_physical_device increase refcnt by one */
put_device ( dev ) ;
2008-02-22 21:54:24 -08:00
if ( ! acpi_bus_get_device ( dev - > archdata . acpi_handle ,
& acpi_dev ) ) {
sysfs_remove_link ( & dev - > kobj , " firmware_node " ) ;
sysfs_remove_link ( & acpi_dev - > dev . kobj , " physical_node " ) ;
}
2006-11-11 17:18:42 +11:00
acpi_detach_data ( dev - > archdata . acpi_handle ,
acpi_glue_data_handler ) ;
dev - > archdata . acpi_handle = NULL ;
2005-03-18 18:45:35 -05:00
/* acpi_bind_one increase refcnt by one */
put_device ( dev ) ;
} else {
2008-05-02 06:02:41 +02:00
dev_err ( dev , " Oops, 'acpi_handle' corrupt \n " ) ;
2005-03-18 18:45:35 -05:00
}
return 0 ;
}
static int acpi_platform_notify ( struct device * dev )
{
struct acpi_bus_type * type ;
acpi_handle handle ;
int ret = - EINVAL ;
if ( ! dev - > bus | | ! dev - > parent ) {
/* bridge devices genernally haven't bus or parent */
ret = acpi_find_bridge_device ( dev , & handle ) ;
goto end ;
}
type = acpi_get_bus_type ( dev - > bus ) ;
if ( ! type ) {
2009-01-25 23:40:56 +01:00
DBG ( " No ACPI bus support for %s \n " , dev_name ( dev ) ) ;
2005-03-18 18:45:35 -05:00
ret = - EINVAL ;
goto end ;
}
if ( ( ret = type - > find_device ( dev , & handle ) ) ! = 0 )
2009-01-25 23:40:56 +01:00
DBG ( " Can't get handler for %s \n " , dev_name ( dev ) ) ;
2005-03-18 18:45:35 -05:00
end :
if ( ! ret )
acpi_bind_one ( dev , handle ) ;
# if ACPI_GLUE_DEBUG
if ( ! ret ) {
struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER , NULL } ;
2006-11-11 17:18:42 +11:00
acpi_get_name ( dev - > archdata . acpi_handle ,
ACPI_FULL_PATHNAME , & buffer ) ;
2009-01-25 23:40:56 +01:00
DBG ( " Device %s -> %s \n " , dev_name ( dev ) , ( char * ) buffer . pointer ) ;
2006-06-30 03:19:10 -04:00
kfree ( buffer . pointer ) ;
2005-03-18 18:45:35 -05:00
} else
2009-01-25 23:40:56 +01:00
DBG ( " Device %s -> No ACPI support \n " , dev_name ( dev ) ) ;
2005-03-18 18:45:35 -05:00
# endif
return ret ;
}
static int acpi_platform_notify_remove ( struct device * dev )
{
acpi_unbind_one ( dev ) ;
return 0 ;
}
2009-03-24 16:50:09 -06:00
int __init init_acpi_device_notify ( void )
2005-03-18 18:45:35 -05:00
{
if ( platform_notify | | platform_notify_remove ) {
printk ( KERN_ERR PREFIX " Can't use platform_notify \n " ) ;
return 0 ;
}
platform_notify = acpi_platform_notify ;
platform_notify_remove = acpi_platform_notify_remove ;
return 0 ;
}