2007-02-27 22:49:53 +03:00
/*
* LEDs driver for GPIOs
*
* Copyright ( C ) 2007 8 D Technologies inc .
* Raphael Assenat < raph @ 8 d . com >
2009-01-10 20:26:01 +03:00
* Copyright ( C ) 2008 Freescale Semiconductor , Inc .
2007-02-27 22:49:53 +03: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/kernel.h>
# include <linux/init.h>
# include <linux/platform_device.h>
# include <linux/leds.h>
2007-05-10 13:51:41 +04:00
# include <linux/workqueue.h>
2007-02-27 22:49:53 +03:00
# include <asm/gpio.h>
struct gpio_led_data {
struct led_classdev cdev ;
unsigned gpio ;
2007-05-10 13:51:41 +04:00
struct work_struct work ;
u8 new_level ;
u8 can_sleep ;
2007-02-27 22:49:53 +03:00
u8 active_low ;
2008-03-10 02:48:25 +03:00
int ( * platform_gpio_blink_set ) ( unsigned gpio ,
unsigned long * delay_on , unsigned long * delay_off ) ;
2007-02-27 22:49:53 +03:00
} ;
2007-05-10 13:51:41 +04:00
static void gpio_led_work ( struct work_struct * work )
{
struct gpio_led_data * led_dat =
container_of ( work , struct gpio_led_data , work ) ;
gpio_set_value_cansleep ( led_dat - > gpio , led_dat - > new_level ) ;
}
2007-02-27 22:49:53 +03:00
static void gpio_led_set ( struct led_classdev * led_cdev ,
enum led_brightness value )
{
struct gpio_led_data * led_dat =
container_of ( led_cdev , struct gpio_led_data , cdev ) ;
int level ;
if ( value = = LED_OFF )
level = 0 ;
else
level = 1 ;
if ( led_dat - > active_low )
level = ! level ;
2008-03-27 03:59:02 +03:00
/* Setting GPIOs with I2C/etc requires a task context, and we don't
* seem to have a reliable way to know if we ' re already in one ; so
* let ' s just assume the worst .
*/
2007-05-10 13:51:41 +04:00
if ( led_dat - > can_sleep ) {
2008-03-27 03:59:02 +03:00
led_dat - > new_level = level ;
schedule_work ( & led_dat - > work ) ;
2007-05-10 13:51:41 +04:00
} else
gpio_set_value ( led_dat - > gpio , level ) ;
2007-02-27 22:49:53 +03:00
}
2008-03-10 02:48:25 +03:00
static int gpio_blink_set ( struct led_classdev * led_cdev ,
unsigned long * delay_on , unsigned long * delay_off )
{
struct gpio_led_data * led_dat =
container_of ( led_cdev , struct gpio_led_data , cdev ) ;
return led_dat - > platform_gpio_blink_set ( led_dat - > gpio , delay_on , delay_off ) ;
}
2009-01-10 20:26:01 +03:00
static int __devinit create_gpio_led ( const struct gpio_led * template ,
struct gpio_led_data * led_dat , struct device * parent ,
int ( * blink_set ) ( unsigned , unsigned long * , unsigned long * ) )
{
int ret ;
2009-03-06 03:46:44 +03:00
/* skip leds that aren't available */
if ( ! gpio_is_valid ( template - > gpio ) ) {
printk ( KERN_INFO " Skipping unavilable LED gpio %d (%s) \n " ,
template - > gpio , template - > name ) ;
2009-04-08 04:51:49 +04:00
return 0 ;
2009-03-06 03:46:44 +03:00
}
2009-01-10 20:26:01 +03:00
ret = gpio_request ( template - > gpio , template - > name ) ;
if ( ret < 0 )
return ret ;
led_dat - > cdev . name = template - > name ;
led_dat - > cdev . default_trigger = template - > default_trigger ;
led_dat - > gpio = template - > gpio ;
led_dat - > can_sleep = gpio_cansleep ( template - > gpio ) ;
led_dat - > active_low = template - > active_low ;
if ( blink_set ) {
led_dat - > platform_gpio_blink_set = blink_set ;
led_dat - > cdev . blink_set = gpio_blink_set ;
}
led_dat - > cdev . brightness_set = gpio_led_set ;
led_dat - > cdev . brightness = LED_OFF ;
2009-02-17 18:04:07 +03:00
if ( ! template - > retain_state_suspended )
led_dat - > cdev . flags | = LED_CORE_SUSPENDRESUME ;
2009-01-10 20:26:01 +03:00
ret = gpio_direction_output ( led_dat - > gpio , led_dat - > active_low ) ;
if ( ret < 0 )
goto err ;
INIT_WORK ( & led_dat - > work , gpio_led_work ) ;
ret = led_classdev_register ( parent , & led_dat - > cdev ) ;
if ( ret < 0 )
goto err ;
return 0 ;
err :
gpio_free ( led_dat - > gpio ) ;
return ret ;
}
static void delete_gpio_led ( struct gpio_led_data * led )
{
2009-03-06 03:46:44 +03:00
if ( ! gpio_is_valid ( led - > gpio ) )
return ;
2009-01-10 20:26:01 +03:00
led_classdev_unregister ( & led - > cdev ) ;
cancel_work_sync ( & led - > work ) ;
gpio_free ( led - > gpio ) ;
}
# ifdef CONFIG_LEDS_GPIO_PLATFORM
2007-10-31 12:37:37 +03:00
static int gpio_led_probe ( struct platform_device * pdev )
2007-02-27 22:49:53 +03:00
{
struct gpio_led_platform_data * pdata = pdev - > dev . platform_data ;
2009-01-10 20:26:01 +03:00
struct gpio_led_data * leds_data ;
2007-02-27 22:49:53 +03:00
int i , ret = 0 ;
if ( ! pdata )
return - EBUSY ;
leds_data = kzalloc ( sizeof ( struct gpio_led_data ) * pdata - > num_leds ,
GFP_KERNEL ) ;
if ( ! leds_data )
return - ENOMEM ;
for ( i = 0 ; i < pdata - > num_leds ; i + + ) {
2009-01-10 20:26:01 +03:00
ret = create_gpio_led ( & pdata - > leds [ i ] , & leds_data [ i ] ,
& pdev - > dev , pdata - > gpio_blink_set ) ;
2008-03-09 23:42:27 +03:00
if ( ret < 0 )
goto err ;
2007-02-27 22:49:53 +03:00
}
platform_set_drvdata ( pdev , leds_data ) ;
return 0 ;
err :
2009-01-10 20:26:01 +03:00
for ( i = i - 1 ; i > = 0 ; i - - )
delete_gpio_led ( & leds_data [ i ] ) ;
2007-05-10 13:51:41 +04:00
2007-02-27 22:49:53 +03:00
kfree ( leds_data ) ;
return ret ;
}
2007-10-31 12:37:37 +03:00
static int __devexit gpio_led_remove ( struct platform_device * pdev )
2007-02-27 22:49:53 +03:00
{
int i ;
struct gpio_led_platform_data * pdata = pdev - > dev . platform_data ;
struct gpio_led_data * leds_data ;
leds_data = platform_get_drvdata ( pdev ) ;
2009-01-10 20:26:01 +03:00
for ( i = 0 ; i < pdata - > num_leds ; i + + )
delete_gpio_led ( & leds_data [ i ] ) ;
2007-10-31 12:37:37 +03:00
2007-02-27 22:49:53 +03:00
kfree ( leds_data ) ;
return 0 ;
}
static struct platform_driver gpio_led_driver = {
2007-10-31 12:37:37 +03:00
. probe = gpio_led_probe ,
. remove = __devexit_p ( gpio_led_remove ) ,
2007-02-27 22:49:53 +03:00
. driver = {
. name = " leds-gpio " ,
. owner = THIS_MODULE ,
} ,
} ;
2009-01-10 20:26:01 +03:00
MODULE_ALIAS ( " platform:leds-gpio " ) ;
# endif /* CONFIG_LEDS_GPIO_PLATFORM */
/* Code to create from OpenFirmware platform devices */
# ifdef CONFIG_LEDS_GPIO_OF
# include <linux/of_platform.h>
# include <linux/of_gpio.h>
struct gpio_led_of_platform_data {
int num_leds ;
struct gpio_led_data led_data [ ] ;
} ;
static int __devinit of_gpio_leds_probe ( struct of_device * ofdev ,
const struct of_device_id * match )
{
struct device_node * np = ofdev - > node , * child ;
struct gpio_led led ;
struct gpio_led_of_platform_data * pdata ;
int count = 0 , ret ;
/* count LEDs defined by this device, so we know how much to allocate */
for_each_child_of_node ( np , child )
count + + ;
if ( ! count )
return 0 ; /* or ENODEV? */
pdata = kzalloc ( sizeof ( * pdata ) + sizeof ( struct gpio_led_data ) * count ,
GFP_KERNEL ) ;
if ( ! pdata )
return - ENOMEM ;
memset ( & led , 0 , sizeof ( led ) ) ;
for_each_child_of_node ( np , child ) {
enum of_gpio_flags flags ;
led . gpio = of_get_gpio_flags ( child , 0 , & flags ) ;
led . active_low = flags & OF_GPIO_ACTIVE_LOW ;
led . name = of_get_property ( child , " label " , NULL ) ? : child - > name ;
led . default_trigger =
of_get_property ( child , " linux,default-trigger " , NULL ) ;
ret = create_gpio_led ( & led , & pdata - > led_data [ pdata - > num_leds + + ] ,
& ofdev - > dev , NULL ) ;
if ( ret < 0 ) {
of_node_put ( child ) ;
goto err ;
}
}
dev_set_drvdata ( & ofdev - > dev , pdata ) ;
return 0 ;
err :
for ( count = pdata - > num_leds - 2 ; count > = 0 ; count - - )
delete_gpio_led ( & pdata - > led_data [ count ] ) ;
kfree ( pdata ) ;
return ret ;
}
static int __devexit of_gpio_leds_remove ( struct of_device * ofdev )
{
struct gpio_led_of_platform_data * pdata = dev_get_drvdata ( & ofdev - > dev ) ;
int i ;
for ( i = 0 ; i < pdata - > num_leds ; i + + )
delete_gpio_led ( & pdata - > led_data [ i ] ) ;
kfree ( pdata ) ;
dev_set_drvdata ( & ofdev - > dev , NULL ) ;
return 0 ;
}
static const struct of_device_id of_gpio_leds_match [ ] = {
{ . compatible = " gpio-leds " , } ,
{ } ,
} ;
static struct of_platform_driver of_gpio_leds_driver = {
. driver = {
. name = " of_gpio_leds " ,
. owner = THIS_MODULE ,
} ,
. match_table = of_gpio_leds_match ,
. probe = of_gpio_leds_probe ,
. remove = __devexit_p ( of_gpio_leds_remove ) ,
} ;
2009-02-03 02:04:42 +03:00
# endif
2009-01-10 20:26:01 +03:00
2009-02-03 02:04:42 +03:00
static int __init gpio_led_init ( void )
2009-01-10 20:26:01 +03:00
{
2009-02-03 02:04:42 +03:00
int ret ;
# ifdef CONFIG_LEDS_GPIO_PLATFORM
ret = platform_driver_register ( & gpio_led_driver ) ;
if ( ret )
return ret ;
# endif
# ifdef CONFIG_LEDS_GPIO_OF
ret = of_register_platform_driver ( & of_gpio_leds_driver ) ;
# endif
# ifdef CONFIG_LEDS_GPIO_PLATFORM
if ( ret )
platform_driver_unregister ( & gpio_led_driver ) ;
# endif
return ret ;
2009-01-10 20:26:01 +03:00
}
2009-02-03 02:04:42 +03:00
static void __exit gpio_led_exit ( void )
2009-01-10 20:26:01 +03:00
{
2009-02-03 02:04:42 +03:00
# ifdef CONFIG_LEDS_GPIO_PLATFORM
platform_driver_unregister ( & gpio_led_driver ) ;
# endif
# ifdef CONFIG_LEDS_GPIO_OF
2009-01-10 20:26:01 +03:00
of_unregister_platform_driver ( & of_gpio_leds_driver ) ;
# endif
2009-02-03 02:04:42 +03:00
}
module_init ( gpio_led_init ) ;
module_exit ( gpio_led_exit ) ;
2009-01-10 20:26:01 +03:00
MODULE_AUTHOR ( " Raphael Assenat <raph@8d.com>, Trent Piepho <tpiepho@freescale.com> " ) ;
2007-02-27 22:49:53 +03:00
MODULE_DESCRIPTION ( " GPIO LED driver " ) ;
MODULE_LICENSE ( " GPL " ) ;