2006-03-31 02:31:04 -08:00
/*
* LED Class Core
*
* Copyright ( C ) 2005 John Lenz < lenz @ cs . wisc . edu >
2007-07-08 23:19:31 +01:00
* Copyright ( C ) 2005 - 2007 Richard Purdie < rpurdie @ openedhand . com >
2006-03-31 02:31:04 -08: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/timer.h>
# include <linux/err.h>
2006-05-15 09:44:17 -07:00
# include <linux/ctype.h>
2006-03-31 02:31:04 -08:00
# include <linux/leds.h>
# include "leds.h"
static struct class * leds_class ;
2008-03-18 09:47:48 +00: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 ) ;
}
2013-07-24 15:05:13 -07:00
static ssize_t brightness_show ( struct device * dev ,
2007-07-08 23:19:31 +01:00
struct device_attribute * attr , char * buf )
2006-03-31 02:31:04 -08:00
{
2007-07-08 23:19:31 +01:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
2006-03-31 02:31:04 -08:00
/* no lock needed for this */
2008-03-18 09:47:48 +00:00
led_update_brightness ( led_cdev ) ;
2006-03-31 02:31:04 -08:00
2008-10-13 10:41:39 +01:00
return sprintf ( buf , " %u \n " , led_cdev - > brightness ) ;
2006-03-31 02:31:04 -08:00
}
2013-07-24 15:05:13 -07:00
static ssize_t brightness_store ( struct device * dev ,
2007-07-08 23:19:31 +01:00
struct device_attribute * attr , const char * buf , size_t size )
2006-03-31 02:31:04 -08:00
{
2007-07-08 23:19:31 +01:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
2012-05-29 15:07:26 -07:00
unsigned long state ;
2006-03-31 02:31:04 -08:00
ssize_t ret = - EINVAL ;
2012-05-29 15:07:26 -07:00
ret = kstrtoul ( buf , 10 , & state ) ;
if ( ret )
return ret ;
2006-05-15 09:44:17 -07:00
2012-05-29 15:07:26 -07:00
if ( state = = LED_OFF )
led_trigger_remove ( led_cdev ) ;
2012-06-13 10:01:37 +08:00
__led_set_brightness ( led_cdev , state ) ;
2008-03-09 20:54:37 +00:00
2012-05-29 15:07:26 -07:00
return size ;
2006-03-31 02:31:04 -08:00
}
2013-07-24 15:05:13 -07:00
static DEVICE_ATTR_RW ( brightness ) ;
2006-03-31 02:31:04 -08:00
2009-01-10 18:54:39 +00: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 ) ;
}
2013-07-24 15:05:13 -07:00
static DEVICE_ATTR ( max_brightness , 0444 , led_max_brightness_show , NULL ) ;
2009-01-10 18:54:39 +00:00
2006-03-31 02:31:05 -08:00
# ifdef CONFIG_LEDS_TRIGGERS
2013-07-24 15:05:13 -07:00
static DEVICE_ATTR ( trigger , 0644 , led_trigger_show , led_trigger_store ) ;
static struct attribute * led_trigger_attrs [ ] = {
& dev_attr_trigger . attr ,
NULL ,
} ;
static const struct attribute_group led_trigger_group = {
. attrs = led_trigger_attrs ,
} ;
# endif
static struct attribute * led_class_attrs [ ] = {
& dev_attr_brightness . attr ,
& dev_attr_max_brightness . attr ,
NULL ,
} ;
static const struct attribute_group led_group = {
. attrs = led_class_attrs ,
} ;
static const struct attribute_group * led_groups [ ] = {
& led_group ,
# ifdef CONFIG_LEDS_TRIGGERS
& led_trigger_group ,
2006-03-31 02:31:05 -08:00
# endif
2013-07-24 15:05:13 -07:00
NULL ,
2010-03-10 18:32:18 +01:00
} ;
2006-03-31 02:31:04 -08:00
2010-11-11 14:05:21 -08: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 ) {
2012-06-13 10:01:37 +08:00
__led_set_brightness ( led_cdev , LED_OFF ) ;
2010-11-11 14:05:21 -08:00
return ;
}
2012-05-27 07:19:22 +08:00
if ( led_cdev - > flags & LED_BLINK_ONESHOT_STOP ) {
led_cdev - > flags & = ~ LED_BLINK_ONESHOT_STOP ;
return ;
}
2010-11-11 14:05:21 -08:00
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 ;
}
2012-06-13 10:01:37 +08:00
__led_set_brightness ( led_cdev , brightness ) ;
2010-11-11 14:05:21 -08:00
2012-05-27 07:19:22 +08:00
/* Return in next iteration if led is in one-shot mode and we are in
* the final blink state so that the led is toggled each delay_on +
* delay_off milliseconds in worst case .
*/
if ( led_cdev - > flags & LED_BLINK_ONESHOT ) {
if ( led_cdev - > flags & LED_BLINK_INVERT ) {
if ( brightness )
led_cdev - > flags | = LED_BLINK_ONESHOT_STOP ;
} else {
if ( ! brightness )
led_cdev - > flags | = LED_BLINK_ONESHOT_STOP ;
}
}
2010-11-11 14:05:21 -08:00
mod_timer ( & led_cdev - > blink_timer , jiffies + msecs_to_jiffies ( delay ) ) ;
}
2012-08-15 21:44:34 +08:00
static void set_brightness_delayed ( struct work_struct * ws )
{
struct led_classdev * led_cdev =
container_of ( ws , struct led_classdev , set_brightness_work ) ;
led_stop_software_blink ( led_cdev ) ;
__led_set_brightness ( led_cdev , led_cdev - > delayed_set_value ) ;
}
2006-03-31 02:31:04 -08: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 ) ;
2013-06-20 12:08:44 -07:00
static int led_suspend ( struct device * dev )
2009-01-08 17:55:03 +00:00
{
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 ;
}
2013-06-20 12:08:44 -07:00
static const struct dev_pm_ops leds_class_dev_pm_ops = {
. suspend = led_suspend ,
. resume = led_resume ,
} ;
2006-03-31 02:31:04 -08:00
/**
* led_classdev_register - register a new object of led_classdev class .
2008-12-03 08:04:57 +00:00
* @ parent : The device to register .
2006-03-31 02:31:04 -08:00
* @ led_cdev : the led_classdev structure for this device .
*/
int led_classdev_register ( struct device * parent , struct led_classdev * led_cdev )
{
2008-07-21 20:03:34 -07:00
led_cdev - > dev = device_create ( leds_class , parent , 0 , led_cdev ,
" %s " , led_cdev - > name ) ;
2008-04-29 01:03:09 -07:00
if ( IS_ERR ( led_cdev - > dev ) )
2007-07-08 23:19:31 +01:00
return PTR_ERR ( led_cdev - > dev ) ;
2006-03-31 02:31:04 -08:00
2008-10-20 23:16:17 +01:00
# ifdef CONFIG_LEDS_TRIGGERS
init_rwsem ( & led_cdev - > trigger_lock ) ;
# endif
2006-03-31 02:31:04 -08:00
/* add to the list of leds */
2007-12-31 23:09:44 +00:00
down_write ( & leds_list_lock ) ;
2006-03-31 02:31:04 -08:00
list_add_tail ( & led_cdev - > node , & leds_list ) ;
2007-12-31 23:09:44 +00:00
up_write ( & leds_list_lock ) ;
2006-03-31 02:31:04 -08:00
2009-01-10 18:54:39 +00:00
if ( ! led_cdev - > max_brightness )
led_cdev - > max_brightness = LED_FULL ;
2008-03-18 09:47:48 +00:00
led_update_brightness ( led_cdev ) ;
2012-08-15 21:44:34 +08:00
INIT_WORK ( & led_cdev - > set_brightness_work , set_brightness_delayed ) ;
2010-11-11 14:05:21 -08: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 02:31:05 -08:00
# ifdef CONFIG_LEDS_TRIGGERS
2006-10-17 00:10:20 -07:00
led_trigger_set_default ( led_cdev ) ;
2006-03-31 02:31:05 -08:00
# endif
2012-11-25 10:57:35 +05:30
dev_dbg ( parent , " Registered led device: %s \n " ,
2007-07-08 23:19:31 +01:00
led_cdev - > name ) ;
2006-03-31 02:31:04 -08:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( led_classdev_register ) ;
/**
2008-10-20 23:04:36 +01:00
* led_classdev_unregister - unregisters a object of led_properties class .
2006-10-03 23:31:30 +02:00
* @ led_cdev : the led device to unregister
2006-03-31 02:31:04 -08:00
*
* Unregisters a previously registered via led_classdev_register object .
*/
2008-03-23 20:28:24 +01:00
void led_classdev_unregister ( struct led_classdev * led_cdev )
2006-03-31 02:31:04 -08:00
{
2006-03-31 02:31:05 -08:00
# ifdef CONFIG_LEDS_TRIGGERS
2007-11-10 13:29:04 +00:00
down_write ( & led_cdev - > trigger_lock ) ;
2006-03-31 02:31:05 -08:00
if ( led_cdev - > trigger )
led_trigger_set ( led_cdev , NULL ) ;
2007-11-10 13:29:04 +00:00
up_write ( & led_cdev - > trigger_lock ) ;
2006-03-31 02:31:05 -08:00
# endif
2006-03-31 02:31:04 -08:00
2012-08-15 21:44:34 +08:00
cancel_work_sync ( & led_cdev - > set_brightness_work ) ;
2010-11-11 14:05:21 -08:00
/* Stop blinking */
2012-08-15 21:44:34 +08:00
led_stop_software_blink ( led_cdev ) ;
2012-06-14 04:34:30 +08:00
led_set_brightness ( led_cdev , LED_OFF ) ;
2010-11-11 14:05:21 -08:00
2008-03-23 20:28:24 +01:00
device_unregister ( led_cdev - > dev ) ;
2006-03-31 02:31:04 -08:00
2007-12-31 23:09:44 +00:00
down_write ( & leds_list_lock ) ;
2006-03-31 02:31:04 -08:00
list_del ( & led_cdev - > node ) ;
2007-12-31 23:09:44 +00:00
up_write ( & leds_list_lock ) ;
2006-03-31 02:31:04 -08:00
}
2008-03-23 20:28:24 +01:00
EXPORT_SYMBOL_GPL ( led_classdev_unregister ) ;
2006-03-31 02:31:04 -08:00
static int __init leds_init ( void )
{
leds_class = class_create ( THIS_MODULE , " leds " ) ;
if ( IS_ERR ( leds_class ) )
return PTR_ERR ( leds_class ) ;
2013-06-20 12:08:44 -07:00
leds_class - > pm = & leds_class_dev_pm_ops ;
2013-07-24 15:05:13 -07:00
leds_class - > dev_groups = led_groups ;
2006-03-31 02:31:04 -08: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 " ) ;