2013-07-04 21:13:25 +02:00
/*
* Sanyo LV5207LP LED Driver
*
* Copyright ( C ) 2013 Ideas on board SPRL
*
* Contact : Laurent Pinchart < laurent . pinchart @ ideasonboard . com >
*
* 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/backlight.h>
# include <linux/err.h>
# include <linux/fb.h>
# include <linux/i2c.h>
# include <linux/module.h>
# include <linux/platform_data/lv5207lp.h>
# include <linux/slab.h>
# define LV5207LP_CTRL1 0x00
# define LV5207LP_CPSW (1 << 7)
# define LV5207LP_SCTEN (1 << 6)
# define LV5207LP_C10 (1 << 5)
# define LV5207LP_CKSW (1 << 4)
# define LV5207LP_RSW (1 << 3)
# define LV5207LP_GSW (1 << 2)
# define LV5207LP_BSW (1 << 1)
# define LV5207LP_CTRL2 0x01
# define LV5207LP_MSW (1 << 7)
# define LV5207LP_MLED4 (1 << 6)
# define LV5207LP_RED 0x02
# define LV5207LP_GREEN 0x03
# define LV5207LP_BLUE 0x04
# define LV5207LP_MAX_BRIGHTNESS 32
struct lv5207lp {
struct i2c_client * client ;
struct backlight_device * backlight ;
struct lv5207lp_platform_data * pdata ;
} ;
static int lv5207lp_write ( struct lv5207lp * lv , u8 reg , u8 data )
{
return i2c_smbus_write_byte_data ( lv - > client , reg , data ) ;
}
static int lv5207lp_backlight_update_status ( struct backlight_device * backlight )
{
struct lv5207lp * lv = bl_get_data ( backlight ) ;
int brightness = backlight - > props . brightness ;
if ( backlight - > props . power ! = FB_BLANK_UNBLANK | |
backlight - > props . fb_blank ! = FB_BLANK_UNBLANK | |
backlight - > props . state & ( BL_CORE_SUSPENDED | BL_CORE_FBBLANK ) )
brightness = 0 ;
if ( brightness ) {
lv5207lp_write ( lv , LV5207LP_CTRL1 ,
LV5207LP_CPSW | LV5207LP_C10 | LV5207LP_CKSW ) ;
lv5207lp_write ( lv , LV5207LP_CTRL2 ,
LV5207LP_MSW | LV5207LP_MLED4 |
( brightness - 1 ) ) ;
} else {
lv5207lp_write ( lv , LV5207LP_CTRL1 , 0 ) ;
lv5207lp_write ( lv , LV5207LP_CTRL2 , 0 ) ;
}
return 0 ;
}
static int lv5207lp_backlight_check_fb ( struct backlight_device * backlight ,
struct fb_info * info )
{
struct lv5207lp * lv = bl_get_data ( backlight ) ;
return lv - > pdata - > fbdev = = NULL | | lv - > pdata - > fbdev = = info - > dev ;
}
static const struct backlight_ops lv5207lp_backlight_ops = {
. options = BL_CORE_SUSPENDRESUME ,
. update_status = lv5207lp_backlight_update_status ,
. check_fb = lv5207lp_backlight_check_fb ,
} ;
static int lv5207lp_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
2013-11-12 15:09:04 -08:00
struct lv5207lp_platform_data * pdata = dev_get_platdata ( & client - > dev ) ;
2013-07-04 21:13:25 +02:00
struct backlight_device * backlight ;
struct backlight_properties props ;
struct lv5207lp * lv ;
if ( pdata = = NULL ) {
dev_err ( & client - > dev , " No platform data supplied \n " ) ;
return - EINVAL ;
}
if ( ! i2c_check_functionality ( client - > adapter ,
I2C_FUNC_SMBUS_BYTE_DATA ) ) {
dev_warn ( & client - > dev ,
" I2C adapter doesn't support I2C_FUNC_SMBUS_BYTE \n " ) ;
return - EIO ;
}
lv = devm_kzalloc ( & client - > dev , sizeof ( * lv ) , GFP_KERNEL ) ;
if ( ! lv )
return - ENOMEM ;
lv - > client = client ;
lv - > pdata = pdata ;
memset ( & props , 0 , sizeof ( props ) ) ;
props . type = BACKLIGHT_RAW ;
props . max_brightness = min_t ( unsigned int , pdata - > max_value ,
LV5207LP_MAX_BRIGHTNESS ) ;
props . brightness = clamp_t ( unsigned int , pdata - > def_value , 0 ,
props . max_brightness ) ;
2013-11-12 15:09:20 -08:00
backlight = devm_backlight_device_register ( & client - > dev ,
dev_name ( & client - > dev ) , & lv - > client - > dev ,
lv , & lv5207lp_backlight_ops , & props ) ;
2013-07-04 21:13:25 +02:00
if ( IS_ERR ( backlight ) ) {
dev_err ( & client - > dev , " failed to register backlight \n " ) ;
return PTR_ERR ( backlight ) ;
}
backlight_update_status ( backlight ) ;
i2c_set_clientdata ( client , backlight ) ;
return 0 ;
}
static int lv5207lp_remove ( struct i2c_client * client )
{
struct backlight_device * backlight = i2c_get_clientdata ( client ) ;
backlight - > props . brightness = 0 ;
backlight_update_status ( backlight ) ;
return 0 ;
}
static const struct i2c_device_id lv5207lp_ids [ ] = {
{ " lv5207lp " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , lv5207lp_ids ) ;
static struct i2c_driver lv5207lp_driver = {
. driver = {
. name = " lv5207lp " ,
} ,
. probe = lv5207lp_probe ,
. remove = lv5207lp_remove ,
. id_table = lv5207lp_ids ,
} ;
module_i2c_driver ( lv5207lp_driver ) ;
MODULE_DESCRIPTION ( " Sanyo LV5207LP Backlight Driver " ) ;
MODULE_AUTHOR ( " Laurent Pinchart <laurent.pinchart@ideasonboard.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;