2010-07-20 22:32:00 +01:00
/*
* Driver for the 8 user LEDs found on the RealViews and Versatiles
* Based on DaVinci ' s DM365 board code
*
* License terms : GNU General Public License ( GPL ) version 2
* Author : Linus Walleij < triad @ df . lth . se >
*/
# include <linux/kernel.h>
# include <linux/init.h>
2014-02-27 14:29:22 +01:00
# include <linux/module.h>
2010-07-20 22:32:00 +01:00
# include <linux/io.h>
# include <linux/slab.h>
# include <linux/leds.h>
2014-02-27 14:29:22 +01:00
# include <linux/platform_device.h>
2010-07-20 22:32:00 +01:00
struct versatile_led {
2014-02-27 14:29:22 +01:00
void __iomem * base ;
2010-07-20 22:32:00 +01:00
struct led_classdev cdev ;
u8 mask ;
} ;
/*
* The triggers lines up below will only be used if the
* LED triggers are compiled in .
*/
static const struct {
const char * name ;
const char * trigger ;
} versatile_leds [ ] = {
{ " versatile:0 " , " heartbeat " , } ,
{ " versatile:1 " , " mmc0 " , } ,
2012-03-13 18:06:36 +08:00
{ " versatile:2 " , " cpu0 " } ,
{ " versatile:3 " , " cpu1 " } ,
{ " versatile:4 " , " cpu2 " } ,
{ " versatile:5 " , " cpu3 " } ,
2010-07-20 22:32:00 +01:00
{ " versatile:6 " , } ,
{ " versatile:7 " , } ,
} ;
static void versatile_led_set ( struct led_classdev * cdev ,
enum led_brightness b )
{
struct versatile_led * led = container_of ( cdev ,
struct versatile_led , cdev ) ;
2014-02-27 14:29:22 +01:00
u32 reg = readl ( led - > base ) ;
2010-07-20 22:32:00 +01:00
if ( b ! = LED_OFF )
reg | = led - > mask ;
else
reg & = ~ led - > mask ;
2014-02-27 14:29:22 +01:00
writel ( reg , led - > base ) ;
2010-07-20 22:32:00 +01:00
}
static enum led_brightness versatile_led_get ( struct led_classdev * cdev )
{
struct versatile_led * led = container_of ( cdev ,
struct versatile_led , cdev ) ;
2014-02-27 14:29:22 +01:00
u32 reg = readl ( led - > base ) ;
2010-07-20 22:32:00 +01:00
return ( reg & led - > mask ) ? LED_FULL : LED_OFF ;
}
2014-02-27 14:29:22 +01:00
static int versatile_leds_probe ( struct platform_device * dev )
2010-07-20 22:32:00 +01:00
{
int i ;
2014-02-27 14:29:22 +01:00
struct resource * res ;
void __iomem * base ;
res = platform_get_resource ( dev , IORESOURCE_MEM , 0 ) ;
base = devm_ioremap_resource ( & dev - > dev , res ) ;
if ( IS_ERR ( base ) )
return PTR_ERR ( base ) ;
2010-07-20 22:32:00 +01:00
2013-11-26 19:44:18 +00:00
/* All off */
2014-02-27 14:29:22 +01:00
writel ( 0 , base ) ;
2010-07-20 22:32:00 +01:00
for ( i = 0 ; i < ARRAY_SIZE ( versatile_leds ) ; i + + ) {
struct versatile_led * led ;
led = kzalloc ( sizeof ( * led ) , GFP_KERNEL ) ;
if ( ! led )
break ;
2014-02-27 14:29:22 +01:00
led - > base = base ;
2010-07-20 22:32:00 +01:00
led - > cdev . name = versatile_leds [ i ] . name ;
led - > cdev . brightness_set = versatile_led_set ;
led - > cdev . brightness_get = versatile_led_get ;
led - > cdev . default_trigger = versatile_leds [ i ] . trigger ;
led - > mask = BIT ( i ) ;
if ( led_classdev_register ( NULL , & led - > cdev ) < 0 ) {
kfree ( led ) ;
break ;
}
}
return 0 ;
}
2014-02-27 14:29:22 +01:00
static struct platform_driver versatile_leds_driver = {
. driver = {
. name = " versatile-leds " ,
} ,
. probe = versatile_leds_probe ,
} ;
module_platform_driver ( versatile_leds_driver ) ;
MODULE_AUTHOR ( " Linus Walleij <linus.walleij@linaro.org> " ) ;
MODULE_DESCRIPTION ( " ARM Versatile LED driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;