2005-04-17 02:20:36 +04:00
/*
* event . c - exporting ACPI events via procfs
*
* Copyright ( C ) 2001 , 2002 Andy Grover < andrew . grover @ intel . com >
* Copyright ( C ) 2001 , 2002 Paul Diefenbaugh < paul . s . diefenbaugh @ intel . com >
*
*/
# include <linux/spinlock.h>
# include <linux/proc_fs.h>
# include <linux/init.h>
# include <linux/poll.h>
# include <acpi/acpi_drivers.h>
# define _COMPONENT ACPI_SYSTEM_COMPONENT
2007-02-13 06:42:12 +03:00
ACPI_MODULE_NAME ( " event " ) ;
2005-04-17 02:20:36 +04:00
/* Global vars for handling event proc entry */
static DEFINE_SPINLOCK ( acpi_system_event_lock ) ;
2005-08-05 08:44:28 +04:00
int event_is_open = 0 ;
extern struct list_head acpi_bus_event_list ;
extern wait_queue_head_t acpi_bus_event_queue ;
2005-04-17 02:20:36 +04:00
2005-08-05 08:44:28 +04:00
static int acpi_system_open_event ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
2005-08-05 08:37:45 +04:00
spin_lock_irq ( & acpi_system_event_lock ) ;
2005-04-17 02:20:36 +04:00
2005-08-05 08:37:45 +04:00
if ( event_is_open )
2005-04-17 02:20:36 +04:00
goto out_busy ;
event_is_open = 1 ;
2005-08-05 08:37:45 +04:00
spin_unlock_irq ( & acpi_system_event_lock ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2005-08-05 08:44:28 +04:00
out_busy :
2005-08-05 08:37:45 +04:00
spin_unlock_irq ( & acpi_system_event_lock ) ;
2005-04-17 02:20:36 +04:00
return - EBUSY ;
}
static ssize_t
2005-08-05 08:44:28 +04:00
acpi_system_read_event ( struct file * file , char __user * buffer , size_t count ,
loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
2005-08-05 08:44:28 +04:00
int result = 0 ;
struct acpi_bus_event event ;
static char str [ ACPI_MAX_STRING ] ;
static int chars_remaining = 0 ;
static char * ptr ;
2005-04-17 02:20:36 +04:00
if ( ! chars_remaining ) {
memset ( & event , 0 , sizeof ( struct acpi_bus_event ) ) ;
if ( ( file - > f_flags & O_NONBLOCK )
& & ( list_empty ( & acpi_bus_event_list ) ) )
2006-06-27 08:41:40 +04:00
return - EAGAIN ;
2005-04-17 02:20:36 +04:00
result = acpi_bus_receive_event ( & event ) ;
2005-10-18 03:43:31 +04:00
if ( result )
2006-06-27 08:41:40 +04:00
return result ;
2005-04-17 02:20:36 +04:00
2005-08-05 08:44:28 +04:00
chars_remaining = sprintf ( str , " %s %s %08x %08x \n " ,
event . device_class ? event .
device_class : " <unknown> " ,
event . bus_id ? event .
bus_id : " <unknown> " , event . type ,
event . data ) ;
2005-04-17 02:20:36 +04:00
ptr = str ;
}
if ( chars_remaining < count ) {
count = chars_remaining ;
}
if ( copy_to_user ( buffer , ptr , count ) )
2006-06-27 08:41:40 +04:00
return - EFAULT ;
2005-04-17 02:20:36 +04:00
* ppos + = count ;
chars_remaining - = count ;
ptr + = count ;
2006-06-27 08:41:40 +04:00
return count ;
2005-04-17 02:20:36 +04:00
}
2005-08-05 08:44:28 +04:00
static int acpi_system_close_event ( struct inode * inode , struct file * file )
2005-04-17 02:20:36 +04:00
{
2005-08-05 08:44:28 +04:00
spin_lock_irq ( & acpi_system_event_lock ) ;
2005-04-17 02:20:36 +04:00
event_is_open = 0 ;
2005-08-05 08:44:28 +04:00
spin_unlock_irq ( & acpi_system_event_lock ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2005-08-05 08:44:28 +04:00
static unsigned int acpi_system_poll_event ( struct file * file , poll_table * wait )
2005-04-17 02:20:36 +04:00
{
poll_wait ( file , & acpi_bus_event_queue , wait ) ;
if ( ! list_empty ( & acpi_bus_event_list ) )
return POLLIN | POLLRDNORM ;
return 0 ;
}
2006-07-04 21:06:00 +04:00
static const struct file_operations acpi_system_event_ops = {
2005-08-05 08:44:28 +04:00
. open = acpi_system_open_event ,
. read = acpi_system_read_event ,
. release = acpi_system_close_event ,
. poll = acpi_system_poll_event ,
2005-04-17 02:20:36 +04:00
} ;
static int __init acpi_event_init ( void )
{
2005-08-05 08:44:28 +04:00
struct proc_dir_entry * entry ;
2005-04-17 02:20:36 +04:00
int error = 0 ;
if ( acpi_disabled )
2006-06-27 08:41:40 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
/* 'event' [R] */
entry = create_proc_entry ( " event " , S_IRUSR , acpi_root_dir ) ;
if ( entry )
entry - > proc_fops = & acpi_system_event_ops ;
else {
2006-06-27 07:58:43 +04:00
error = - ENODEV ;
2005-04-17 02:20:36 +04:00
}
2006-06-27 08:41:40 +04:00
return error ;
2005-04-17 02:20:36 +04:00
}
subsys_initcall ( acpi_event_init ) ;