2006-03-31 14:31:04 +04:00
/*
* LED Class Core
*
* Copyright ( C ) 2005 John Lenz < lenz @ cs . wisc . edu >
2007-07-09 02:19:31 +04:00
* Copyright ( C ) 2005 - 2007 Richard Purdie < rpurdie @ openedhand . com >
2006-03-31 14:31:04 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/list.h>
# include <linux/spinlock.h>
# include <linux/device.h>
# include <linux/sysdev.h>
# include <linux/timer.h>
# include <linux/err.h>
2006-05-15 20:44:17 +04:00
# include <linux/ctype.h>
2006-03-31 14:31:04 +04:00
# include <linux/leds.h>
# include "leds.h"
static struct class * leds_class ;
2008-03-18 12:47:48 +03:00
static void led_update_brightness ( struct led_classdev * led_cdev )
{
if ( led_cdev - > brightness_get )
led_cdev - > brightness = led_cdev - > brightness_get ( led_cdev ) ;
}
2007-07-09 02:19:31 +04:00
static ssize_t led_brightness_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
2006-03-31 14:31:04 +04:00
{
2007-07-09 02:19:31 +04:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
2006-03-31 14:31:04 +04:00
/* no lock needed for this */
2008-03-18 12:47:48 +03:00
led_update_brightness ( led_cdev ) ;
2006-03-31 14:31:04 +04:00
2008-10-13 13:41:39 +04:00
return sprintf ( buf , " %u \n " , led_cdev - > brightness ) ;
2006-03-31 14:31:04 +04:00
}
2007-07-09 02:19:31 +04:00
static ssize_t led_brightness_store ( struct device * dev ,
struct device_attribute * attr , const char * buf , size_t size )
2006-03-31 14:31:04 +04:00
{
2007-07-09 02:19:31 +04:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
2006-03-31 14:31:04 +04:00
ssize_t ret = - EINVAL ;
char * after ;
unsigned long state = simple_strtoul ( buf , & after , 10 ) ;
2006-05-15 20:44:17 +04:00
size_t count = after - buf ;
2006-03-31 14:31:04 +04:00
2009-12-15 05:01:06 +03:00
if ( isspace ( * after ) )
2006-05-15 20:44:17 +04:00
count + + ;
if ( count = = size ) {
ret = count ;
2008-03-09 23:54:37 +03:00
if ( state = = LED_OFF )
led_trigger_remove ( led_cdev ) ;
2006-03-31 14:31:04 +04:00
led_set_brightness ( led_cdev , state ) ;
}
return ret ;
}
2009-01-10 21:54:39 +03:00
static ssize_t led_max_brightness_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %u \n " , led_cdev - > max_brightness ) ;
}
2010-03-10 20:32:18 +03:00
static struct device_attribute led_class_attrs [ ] = {
__ATTR ( brightness , 0644 , led_brightness_show , led_brightness_store ) ,
2010-05-17 04:48:28 +04:00
__ATTR ( max_brightness , 0444 , led_max_brightness_show , NULL ) ,
2006-03-31 14:31:05 +04:00
# ifdef CONFIG_LEDS_TRIGGERS
2010-03-10 20:32:18 +03:00
__ATTR ( trigger , 0644 , led_trigger_show , led_trigger_store ) ,
2006-03-31 14:31:05 +04:00
# endif
2010-03-10 20:32:18 +03:00
__ATTR_NULL ,
} ;
2006-03-31 14:31:04 +04:00
2010-11-12 01:05:21 +03:00
static void led_timer_function ( unsigned long data )
{
struct led_classdev * led_cdev = ( void * ) data ;
unsigned long brightness ;
unsigned long delay ;
if ( ! led_cdev - > blink_delay_on | | ! led_cdev - > blink_delay_off ) {
led_set_brightness ( led_cdev , LED_OFF ) ;
return ;
}
brightness = led_get_brightness ( led_cdev ) ;
if ( ! brightness ) {
/* Time to switch the LED on. */
brightness = led_cdev - > blink_brightness ;
delay = led_cdev - > blink_delay_on ;
} else {
/* Store the current brightness value to be able
* to restore it when the delay_off period is over .
*/
led_cdev - > blink_brightness = brightness ;
brightness = LED_OFF ;
delay = led_cdev - > blink_delay_off ;
}
led_set_brightness ( led_cdev , brightness ) ;
mod_timer ( & led_cdev - > blink_timer , jiffies + msecs_to_jiffies ( delay ) ) ;
}
static void led_stop_software_blink ( struct led_classdev * led_cdev )
{
/* deactivate previous settings */
del_timer_sync ( & led_cdev - > blink_timer ) ;
led_cdev - > blink_delay_on = 0 ;
led_cdev - > blink_delay_off = 0 ;
}
static void led_set_software_blink ( struct led_classdev * led_cdev ,
unsigned long delay_on ,
unsigned long delay_off )
{
int current_brightness ;
current_brightness = led_get_brightness ( led_cdev ) ;
if ( current_brightness )
led_cdev - > blink_brightness = current_brightness ;
if ( ! led_cdev - > blink_brightness )
led_cdev - > blink_brightness = led_cdev - > max_brightness ;
2011-05-25 04:13:22 +04:00
if ( led_get_trigger_data ( led_cdev ) & &
delay_on = = led_cdev - > blink_delay_on & &
2010-11-12 01:05:21 +03:00
delay_off = = led_cdev - > blink_delay_off )
return ;
led_stop_software_blink ( led_cdev ) ;
led_cdev - > blink_delay_on = delay_on ;
led_cdev - > blink_delay_off = delay_off ;
/* never on - don't blink */
if ( ! delay_on )
return ;
/* never off - just set to brightness */
if ( ! delay_off ) {
led_set_brightness ( led_cdev , led_cdev - > blink_brightness ) ;
return ;
}
mod_timer ( & led_cdev - > blink_timer , jiffies + 1 ) ;
}
2006-03-31 14:31:04 +04:00
/**
* led_classdev_suspend - suspend an led_classdev .
* @ led_cdev : the led_classdev to suspend .
*/
void led_classdev_suspend ( struct led_classdev * led_cdev )
{
led_cdev - > flags | = LED_SUSPENDED ;
led_cdev - > brightness_set ( led_cdev , 0 ) ;
}
EXPORT_SYMBOL_GPL ( led_classdev_suspend ) ;
/**
* led_classdev_resume - resume an led_classdev .
* @ led_cdev : the led_classdev to resume .
*/
void led_classdev_resume ( struct led_classdev * led_cdev )
{
led_cdev - > brightness_set ( led_cdev , led_cdev - > brightness ) ;
led_cdev - > flags & = ~ LED_SUSPENDED ;
}
EXPORT_SYMBOL_GPL ( led_classdev_resume ) ;
2009-01-08 20:55:03 +03:00
static int led_suspend ( struct device * dev , pm_message_t state )
{
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
if ( led_cdev - > flags & LED_CORE_SUSPENDRESUME )
led_classdev_suspend ( led_cdev ) ;
return 0 ;
}
static int led_resume ( struct device * dev )
{
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
if ( led_cdev - > flags & LED_CORE_SUSPENDRESUME )
led_classdev_resume ( led_cdev ) ;
return 0 ;
}
2006-03-31 14:31:04 +04:00
/**
* led_classdev_register - register a new object of led_classdev class .
2008-12-03 11:04:57 +03:00
* @ parent : The device to register .
2006-03-31 14:31:04 +04:00
* @ led_cdev : the led_classdev structure for this device .
*/
int led_classdev_register ( struct device * parent , struct led_classdev * led_cdev )
{
2008-07-22 07:03:34 +04:00
led_cdev - > dev = device_create ( leds_class , parent , 0 , led_cdev ,
" %s " , led_cdev - > name ) ;
2008-04-29 12:03:09 +04:00
if ( IS_ERR ( led_cdev - > dev ) )
2007-07-09 02:19:31 +04:00
return PTR_ERR ( led_cdev - > dev ) ;
2006-03-31 14:31:04 +04:00
2008-10-21 02:16:17 +04:00
# ifdef CONFIG_LEDS_TRIGGERS
init_rwsem ( & led_cdev - > trigger_lock ) ;
# endif
2006-03-31 14:31:04 +04:00
/* add to the list of leds */
2008-01-01 02:09:44 +03:00
down_write ( & leds_list_lock ) ;
2006-03-31 14:31:04 +04:00
list_add_tail ( & led_cdev - > node , & leds_list ) ;
2008-01-01 02:09:44 +03:00
up_write ( & leds_list_lock ) ;
2006-03-31 14:31:04 +04:00
2009-01-10 21:54:39 +03:00
if ( ! led_cdev - > max_brightness )
led_cdev - > max_brightness = LED_FULL ;
2008-03-18 12:47:48 +03:00
led_update_brightness ( led_cdev ) ;
2010-11-12 01:05:21 +03:00
init_timer ( & led_cdev - > blink_timer ) ;
led_cdev - > blink_timer . function = led_timer_function ;
led_cdev - > blink_timer . data = ( unsigned long ) led_cdev ;
2006-03-31 14:31:05 +04:00
# ifdef CONFIG_LEDS_TRIGGERS
2006-10-17 11:10:20 +04:00
led_trigger_set_default ( led_cdev ) ;
2006-03-31 14:31:05 +04:00
# endif
2010-01-07 01:34:55 +03:00
printk ( KERN_DEBUG " Registered led device: %s \n " ,
2007-07-09 02:19:31 +04:00
led_cdev - > name ) ;
2006-03-31 14:31:04 +04:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( led_classdev_register ) ;
/**
2008-10-21 02:04:36 +04:00
* led_classdev_unregister - unregisters a object of led_properties class .
2006-10-04 01:31:30 +04:00
* @ led_cdev : the led device to unregister
2006-03-31 14:31:04 +04:00
*
* Unregisters a previously registered via led_classdev_register object .
*/
2008-03-23 22:28:24 +03:00
void led_classdev_unregister ( struct led_classdev * led_cdev )
2006-03-31 14:31:04 +04:00
{
2006-03-31 14:31:05 +04:00
# ifdef CONFIG_LEDS_TRIGGERS
2007-11-10 16:29:04 +03:00
down_write ( & led_cdev - > trigger_lock ) ;
2006-03-31 14:31:05 +04:00
if ( led_cdev - > trigger )
led_trigger_set ( led_cdev , NULL ) ;
2007-11-10 16:29:04 +03:00
up_write ( & led_cdev - > trigger_lock ) ;
2006-03-31 14:31:05 +04:00
# endif
2006-03-31 14:31:04 +04:00
2010-11-12 01:05:21 +03:00
/* Stop blinking */
led_brightness_set ( led_cdev , LED_OFF ) ;
2008-03-23 22:28:24 +03:00
device_unregister ( led_cdev - > dev ) ;
2006-03-31 14:31:04 +04:00
2008-01-01 02:09:44 +03:00
down_write ( & leds_list_lock ) ;
2006-03-31 14:31:04 +04:00
list_del ( & led_cdev - > node ) ;
2008-01-01 02:09:44 +03:00
up_write ( & leds_list_lock ) ;
2006-03-31 14:31:04 +04:00
}
2008-03-23 22:28:24 +03:00
EXPORT_SYMBOL_GPL ( led_classdev_unregister ) ;
2006-03-31 14:31:04 +04:00
2010-11-12 01:05:21 +03:00
void led_blink_set ( struct led_classdev * led_cdev ,
unsigned long * delay_on ,
unsigned long * delay_off )
{
if ( led_cdev - > blink_set & &
2010-12-22 04:24:28 +03:00
! led_cdev - > blink_set ( led_cdev , delay_on , delay_off ) )
2010-11-12 01:05:21 +03:00
return ;
/* blink with 1 Hz as default if nothing specified */
if ( ! * delay_on & & ! * delay_off )
* delay_on = * delay_off = 500 ;
led_set_software_blink ( led_cdev , * delay_on , * delay_off ) ;
}
EXPORT_SYMBOL ( led_blink_set ) ;
void led_brightness_set ( struct led_classdev * led_cdev ,
enum led_brightness brightness )
{
led_stop_software_blink ( led_cdev ) ;
led_cdev - > brightness_set ( led_cdev , brightness ) ;
}
EXPORT_SYMBOL ( led_brightness_set ) ;
2006-03-31 14:31:04 +04:00
static int __init leds_init ( void )
{
leds_class = class_create ( THIS_MODULE , " leds " ) ;
if ( IS_ERR ( leds_class ) )
return PTR_ERR ( leds_class ) ;
2009-01-08 20:55:03 +03:00
leds_class - > suspend = led_suspend ;
leds_class - > resume = led_resume ;
2010-03-10 20:32:18 +03:00
leds_class - > dev_attrs = led_class_attrs ;
2006-03-31 14:31:04 +04:00
return 0 ;
}
static void __exit leds_exit ( void )
{
class_destroy ( leds_class ) ;
}
subsys_initcall ( leds_init ) ;
module_exit ( leds_exit ) ;
MODULE_AUTHOR ( " John Lenz, Richard Purdie " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " LED Class Interface " ) ;