2010-07-06 16:08:46 +02:00
/*
* leds - ns2 . c - Driver for the Network Space v2 ( and parents ) dual - GPIO LED
*
* Copyright ( C ) 2010 LaCie
*
* Author : Simon Guinot < sguinot @ lacie . com >
*
* Based on leds - gpio . c by Raphael Assenat < raph @ 8 d . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/kernel.h>
# include <linux/platform_device.h>
# include <linux/slab.h>
# include <linux/gpio.h>
# include <linux/leds.h>
2011-07-03 13:56:03 -04:00
# include <linux/module.h>
2012-08-24 15:21:54 +02:00
# include <linux/platform_data/leds-kirkwood-ns2.h>
2013-09-28 04:38:30 -07:00
# include <linux/of.h>
2012-10-17 12:09:03 +02:00
# include <linux/of_gpio.h>
2015-07-02 19:56:42 +02:00
# include "leds.h"
2010-07-06 16:08:46 +02:00
/*
2015-07-02 19:56:40 +02:00
* The Network Space v2 dual - GPIO LED is wired to a CPLD . Three different LED
* modes are available : off , on and SATA activity blinking . The LED modes are
* controlled through two GPIOs ( command and slow ) : each combination of values
* for the command / slow GPIOs corresponds to a LED mode .
2010-07-06 16:08:46 +02:00
*/
struct ns2_led_data {
struct led_classdev cdev ;
unsigned cmd ;
unsigned slow ;
2015-07-02 19:56:42 +02:00
bool can_sleep ;
2010-07-06 16:08:46 +02:00
unsigned char sata ; /* True when SATA mode active. */
rwlock_t rw_lock ; /* Lock GPIOs. */
2015-07-02 19:56:40 +02:00
int num_modes ;
struct ns2_led_modval * modval ;
2010-07-06 16:08:46 +02:00
} ;
static int ns2_led_get_mode ( struct ns2_led_data * led_dat ,
enum ns2_led_modes * mode )
{
int i ;
int ret = - EINVAL ;
int cmd_level ;
int slow_level ;
2015-07-02 19:56:42 +02:00
cmd_level = gpio_get_value_cansleep ( led_dat - > cmd ) ;
slow_level = gpio_get_value_cansleep ( led_dat - > slow ) ;
2010-07-06 16:08:46 +02:00
2015-07-02 19:56:40 +02:00
for ( i = 0 ; i < led_dat - > num_modes ; i + + ) {
if ( cmd_level = = led_dat - > modval [ i ] . cmd_level & &
slow_level = = led_dat - > modval [ i ] . slow_level ) {
* mode = led_dat - > modval [ i ] . mode ;
2010-07-06 16:08:46 +02:00
ret = 0 ;
break ;
}
}
return ret ;
}
static void ns2_led_set_mode ( struct ns2_led_data * led_dat ,
enum ns2_led_modes mode )
{
int i ;
2015-07-02 19:56:42 +02:00
bool found = false ;
2010-09-19 15:30:59 +02:00
unsigned long flags ;
2010-07-06 16:08:46 +02:00
2015-07-02 19:56:42 +02:00
for ( i = 0 ; i < led_dat - > num_modes ; i + + )
2015-07-02 19:56:40 +02:00
if ( mode = = led_dat - > modval [ i ] . mode ) {
2015-07-02 19:56:42 +02:00
found = true ;
break ;
2010-07-06 16:08:46 +02:00
}
2015-07-02 19:56:42 +02:00
if ( ! found )
return ;
write_lock_irqsave ( & led_dat - > rw_lock , flags ) ;
if ( ! led_dat - > can_sleep ) {
gpio_set_value ( led_dat - > cmd ,
led_dat - > modval [ i ] . cmd_level ) ;
gpio_set_value ( led_dat - > slow ,
led_dat - > modval [ i ] . slow_level ) ;
goto exit_unlock ;
2010-07-06 16:08:46 +02:00
}
2015-11-20 11:39:41 +01:00
gpio_set_value_cansleep ( led_dat - > cmd , led_dat - > modval [ i ] . cmd_level ) ;
gpio_set_value_cansleep ( led_dat - > slow , led_dat - > modval [ i ] . slow_level ) ;
2015-07-02 19:56:42 +02:00
exit_unlock :
2010-09-19 15:30:59 +02:00
write_unlock_irqrestore ( & led_dat - > rw_lock , flags ) ;
2010-07-06 16:08:46 +02:00
}
static void ns2_led_set ( struct led_classdev * led_cdev ,
enum led_brightness value )
{
struct ns2_led_data * led_dat =
container_of ( led_cdev , struct ns2_led_data , cdev ) ;
enum ns2_led_modes mode ;
if ( value = = LED_OFF )
mode = NS_V2_LED_OFF ;
else if ( led_dat - > sata )
mode = NS_V2_LED_SATA ;
else
mode = NS_V2_LED_ON ;
ns2_led_set_mode ( led_dat , mode ) ;
}
2015-11-20 11:39:41 +01:00
static int ns2_led_set_blocking ( struct led_classdev * led_cdev ,
enum led_brightness value )
{
ns2_led_set ( led_cdev , value ) ;
return 0 ;
}
2010-07-06 16:08:46 +02:00
static ssize_t ns2_led_sata_store ( struct device * dev ,
struct device_attribute * attr ,
const char * buff , size_t count )
{
2010-10-07 16:35:40 +02:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
struct ns2_led_data * led_dat =
container_of ( led_cdev , struct ns2_led_data , cdev ) ;
2010-07-06 16:08:46 +02:00
int ret ;
unsigned long enable ;
2012-10-23 05:25:35 -07:00
ret = kstrtoul ( buff , 10 , & enable ) ;
2010-07-06 16:08:46 +02:00
if ( ret < 0 )
return ret ;
enable = ! ! enable ;
if ( led_dat - > sata = = enable )
2015-07-02 19:56:42 +02:00
goto exit ;
2010-07-06 16:08:46 +02:00
2015-07-02 19:56:42 +02:00
led_dat - > sata = enable ;
if ( ! led_get_brightness ( led_cdev ) )
goto exit ;
2010-07-06 16:08:46 +02:00
2015-07-02 19:56:42 +02:00
if ( enable )
2010-07-06 16:08:46 +02:00
ns2_led_set_mode ( led_dat , NS_V2_LED_SATA ) ;
2015-07-02 19:56:42 +02:00
else
2010-07-06 16:08:46 +02:00
ns2_led_set_mode ( led_dat , NS_V2_LED_ON ) ;
2015-07-02 19:56:42 +02:00
exit :
2010-07-06 16:08:46 +02:00
return count ;
}
static ssize_t ns2_led_sata_show ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-10-07 16:35:40 +02:00
struct led_classdev * led_cdev = dev_get_drvdata ( dev ) ;
struct ns2_led_data * led_dat =
container_of ( led_cdev , struct ns2_led_data , cdev ) ;
2010-07-06 16:08:46 +02:00
return sprintf ( buf , " %d \n " , led_dat - > sata ) ;
}
static DEVICE_ATTR ( sata , 0644 , ns2_led_sata_show , ns2_led_sata_store ) ;
2014-06-25 10:08:51 -07:00
static struct attribute * ns2_led_attrs [ ] = {
& dev_attr_sata . attr ,
NULL
} ;
ATTRIBUTE_GROUPS ( ns2_led ) ;
2012-11-19 13:23:02 -05:00
static int
2010-07-06 16:08:46 +02:00
create_ns2_led ( struct platform_device * pdev , struct ns2_led_data * led_dat ,
const struct ns2_led * template )
{
int ret ;
enum ns2_led_modes mode ;
2012-11-25 10:28:10 +05:30
ret = devm_gpio_request_one ( & pdev - > dev , template - > cmd ,
2015-07-02 19:56:42 +02:00
gpio_get_value_cansleep ( template - > cmd ) ?
2013-03-07 18:38:26 -08:00
GPIOF_OUT_INIT_HIGH : GPIOF_OUT_INIT_LOW ,
2012-10-23 05:18:21 -07:00
template - > name ) ;
2010-07-06 16:08:46 +02:00
if ( ret ) {
dev_err ( & pdev - > dev , " %s: failed to setup command GPIO \n " ,
template - > name ) ;
2012-10-23 05:18:21 -07:00
return ret ;
2010-07-06 16:08:46 +02:00
}
2012-11-25 10:28:10 +05:30
ret = devm_gpio_request_one ( & pdev - > dev , template - > slow ,
2015-07-02 19:56:42 +02:00
gpio_get_value_cansleep ( template - > slow ) ?
2013-03-07 18:38:26 -08:00
GPIOF_OUT_INIT_HIGH : GPIOF_OUT_INIT_LOW ,
2012-10-23 05:18:21 -07:00
template - > name ) ;
2010-07-06 16:08:46 +02:00
if ( ret ) {
dev_err ( & pdev - > dev , " %s: failed to setup slow GPIO \n " ,
template - > name ) ;
2012-11-25 10:28:10 +05:30
return ret ;
2010-07-06 16:08:46 +02:00
}
rwlock_init ( & led_dat - > rw_lock ) ;
led_dat - > cdev . name = template - > name ;
led_dat - > cdev . default_trigger = template - > default_trigger ;
led_dat - > cdev . blink_set = NULL ;
led_dat - > cdev . flags | = LED_CORE_SUSPENDRESUME ;
2014-06-25 10:08:51 -07:00
led_dat - > cdev . groups = ns2_led_groups ;
2010-07-06 16:08:46 +02:00
led_dat - > cmd = template - > cmd ;
led_dat - > slow = template - > slow ;
2015-07-02 19:56:42 +02:00
led_dat - > can_sleep = gpio_cansleep ( led_dat - > cmd ) |
gpio_cansleep ( led_dat - > slow ) ;
2015-11-20 11:39:41 +01:00
if ( led_dat - > can_sleep )
led_dat - > cdev . brightness_set_blocking = ns2_led_set_blocking ;
else
led_dat - > cdev . brightness_set = ns2_led_set ;
2015-07-02 19:56:40 +02:00
led_dat - > modval = template - > modval ;
led_dat - > num_modes = template - > num_modes ;
2010-07-06 16:08:46 +02:00
ret = ns2_led_get_mode ( led_dat , & mode ) ;
if ( ret < 0 )
2012-11-25 10:28:10 +05:30
return ret ;
2010-07-06 16:08:46 +02:00
/* Set LED initial state. */
led_dat - > sata = ( mode = = NS_V2_LED_SATA ) ? 1 : 0 ;
led_dat - > cdev . brightness =
( mode = = NS_V2_LED_OFF ) ? LED_OFF : LED_FULL ;
ret = led_classdev_register ( & pdev - > dev , & led_dat - > cdev ) ;
if ( ret < 0 )
2012-11-25 10:28:10 +05:30
return ret ;
2010-07-06 16:08:46 +02:00
return 0 ;
}
2012-05-10 13:01:46 -07:00
static void delete_ns2_led ( struct ns2_led_data * led_dat )
2010-07-06 16:08:46 +02:00
{
led_classdev_unregister ( & led_dat - > cdev ) ;
}
2012-10-17 12:09:03 +02:00
# ifdef CONFIG_OF_GPIO
/*
* Translate OpenFirmware node properties into platform_data .
*/
2012-12-12 12:14:06 -08:00
static int
2012-10-17 12:09:03 +02:00
ns2_leds_get_of_pdata ( struct device * dev , struct ns2_led_platform_data * pdata )
{
struct device_node * np = dev - > of_node ;
struct device_node * child ;
2015-07-02 19:56:40 +02:00
struct ns2_led * led , * leds ;
2012-10-17 12:09:03 +02:00
int num_leds = 0 ;
num_leds = of_get_child_count ( np ) ;
if ( ! num_leds )
return - ENODEV ;
leds = devm_kzalloc ( dev , num_leds * sizeof ( struct ns2_led ) ,
GFP_KERNEL ) ;
if ( ! leds )
return - ENOMEM ;
2015-07-02 19:56:40 +02:00
led = leds ;
2012-10-17 12:09:03 +02:00
for_each_child_of_node ( np , child ) {
const char * string ;
2015-07-02 19:56:40 +02:00
int ret , i , num_modes ;
struct ns2_led_modval * modval ;
2012-10-17 12:09:03 +02:00
ret = of_get_named_gpio ( child , " cmd-gpio " , 0 ) ;
if ( ret < 0 )
return ret ;
2015-07-02 19:56:40 +02:00
led - > cmd = ret ;
2012-10-17 12:09:03 +02:00
ret = of_get_named_gpio ( child , " slow-gpio " , 0 ) ;
if ( ret < 0 )
return ret ;
2015-07-02 19:56:40 +02:00
led - > slow = ret ;
2012-10-17 12:09:03 +02:00
ret = of_property_read_string ( child , " label " , & string ) ;
2015-07-02 19:56:40 +02:00
led - > name = ( ret = = 0 ) ? string : child - > name ;
2012-10-17 12:09:03 +02:00
ret = of_property_read_string ( child , " linux,default-trigger " ,
& string ) ;
if ( ret = = 0 )
2015-07-02 19:56:40 +02:00
led - > default_trigger = string ;
ret = of_property_count_u32_elems ( child , " modes-map " ) ;
if ( ret < 0 | | ret % 3 ) {
dev_err ( dev ,
" Missing or malformed modes-map property \n " ) ;
return - EINVAL ;
}
num_modes = ret / 3 ;
modval = devm_kzalloc ( dev ,
num_modes * sizeof ( struct ns2_led_modval ) ,
GFP_KERNEL ) ;
if ( ! modval )
return - ENOMEM ;
for ( i = 0 ; i < num_modes ; i + + ) {
of_property_read_u32_index ( child ,
" modes-map " , 3 * i ,
( u32 * ) & modval [ i ] . mode ) ;
of_property_read_u32_index ( child ,
" modes-map " , 3 * i + 1 ,
( u32 * ) & modval [ i ] . cmd_level ) ;
of_property_read_u32_index ( child ,
" modes-map " , 3 * i + 2 ,
( u32 * ) & modval [ i ] . slow_level ) ;
}
led - > num_modes = num_modes ;
led - > modval = modval ;
2012-10-17 12:09:03 +02:00
2015-07-02 19:56:40 +02:00
led + + ;
2012-10-17 12:09:03 +02:00
}
pdata - > leds = leds ;
pdata - > num_leds = num_leds ;
return 0 ;
}
static const struct of_device_id of_ns2_leds_match [ ] = {
{ . compatible = " lacie,ns2-leds " , } ,
{ } ,
} ;
2015-09-01 23:36:59 +02:00
MODULE_DEVICE_TABLE ( of , of_ns2_leds_match ) ;
2012-10-17 12:09:03 +02:00
# endif /* CONFIG_OF_GPIO */
2013-03-19 11:07:29 -07:00
struct ns2_led_priv {
int num_leds ;
struct ns2_led_data leds_data [ ] ;
} ;
static inline int sizeof_ns2_led_priv ( int num_leds )
{
return sizeof ( struct ns2_led_priv ) +
( sizeof ( struct ns2_led_data ) * num_leds ) ;
}
2012-11-19 13:23:02 -05:00
static int ns2_led_probe ( struct platform_device * pdev )
2010-07-06 16:08:46 +02:00
{
2013-07-30 01:07:35 -07:00
struct ns2_led_platform_data * pdata = dev_get_platdata ( & pdev - > dev ) ;
2013-03-19 11:07:29 -07:00
struct ns2_led_priv * priv ;
2010-07-06 16:08:46 +02:00
int i ;
int ret ;
2012-10-17 12:09:03 +02:00
# ifdef CONFIG_OF_GPIO
if ( ! pdata ) {
pdata = devm_kzalloc ( & pdev - > dev ,
sizeof ( struct ns2_led_platform_data ) ,
GFP_KERNEL ) ;
if ( ! pdata )
return - ENOMEM ;
ret = ns2_leds_get_of_pdata ( & pdev - > dev , pdata ) ;
if ( ret )
return ret ;
}
# else
2010-07-06 16:08:46 +02:00
if ( ! pdata )
return - EINVAL ;
2012-10-17 12:09:03 +02:00
# endif /* CONFIG_OF_GPIO */
2010-07-06 16:08:46 +02:00
2013-03-19 11:07:29 -07:00
priv = devm_kzalloc ( & pdev - > dev ,
sizeof_ns2_led_priv ( pdata - > num_leds ) , GFP_KERNEL ) ;
if ( ! priv )
2010-07-06 16:08:46 +02:00
return - ENOMEM ;
2013-03-19 11:07:29 -07:00
priv - > num_leds = pdata - > num_leds ;
2010-07-06 16:08:46 +02:00
2013-03-19 11:07:29 -07:00
for ( i = 0 ; i < priv - > num_leds ; i + + ) {
ret = create_ns2_led ( pdev , & priv - > leds_data [ i ] ,
& pdata - > leds [ i ] ) ;
2012-07-04 12:30:50 +08:00
if ( ret < 0 ) {
for ( i = i - 1 ; i > = 0 ; i - - )
2013-03-19 11:07:29 -07:00
delete_ns2_led ( & priv - > leds_data [ i ] ) ;
2012-07-04 12:30:50 +08:00
return ret ;
}
2010-07-06 16:08:46 +02:00
}
2013-03-19 11:07:29 -07:00
platform_set_drvdata ( pdev , priv ) ;
2010-07-06 16:08:46 +02:00
return 0 ;
}
2012-11-19 13:26:00 -05:00
static int ns2_led_remove ( struct platform_device * pdev )
2010-07-06 16:08:46 +02:00
{
int i ;
2013-03-19 11:07:29 -07:00
struct ns2_led_priv * priv ;
2010-07-06 16:08:46 +02:00
2013-03-19 11:07:29 -07:00
priv = platform_get_drvdata ( pdev ) ;
2010-07-06 16:08:46 +02:00
2013-03-19 11:07:29 -07:00
for ( i = 0 ; i < priv - > num_leds ; i + + )
delete_ns2_led ( & priv - > leds_data [ i ] ) ;
2010-07-06 16:08:46 +02:00
return 0 ;
}
static struct platform_driver ns2_led_driver = {
. probe = ns2_led_probe ,
2012-11-19 13:20:20 -05:00
. remove = ns2_led_remove ,
2010-07-06 16:08:46 +02:00
. driver = {
2012-10-17 12:09:03 +02:00
. name = " leds-ns2 " ,
. of_match_table = of_match_ptr ( of_ns2_leds_match ) ,
2010-07-06 16:08:46 +02:00
} ,
} ;
2012-01-10 15:09:24 -08:00
module_platform_driver ( ns2_led_driver ) ;
2010-07-06 16:08:46 +02:00
MODULE_AUTHOR ( " Simon Guinot <sguinot@lacie.com> " ) ;
MODULE_DESCRIPTION ( " Network Space v2 LED driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
2012-01-10 15:09:24 -08:00
MODULE_ALIAS ( " platform:leds-ns2 " ) ;