2010-05-05 12:13:23 -05:00
/*
* Driver for the Cirrus EP93xx lcd backlight
*
* Copyright ( c ) 2010 H Hartley Sweeten < hsweeten @ visionengravers . 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 .
*
* This driver controls the pulse width modulated brightness control output ,
* BRIGHT , on the Cirrus EP9307 , EP9312 , and EP9315 processors .
*/
2011-08-25 15:59:14 -07:00
# include <linux/module.h>
2010-05-05 12:13:23 -05:00
# include <linux/platform_device.h>
# include <linux/io.h>
# include <linux/fb.h>
# include <linux/backlight.h>
# define EP93XX_MAX_COUNT 255
# define EP93XX_MAX_BRIGHT 255
# define EP93XX_DEF_BRIGHT 128
struct ep93xxbl {
void __iomem * mmio ;
int brightness ;
} ;
static int ep93xxbl_set ( struct backlight_device * bl , int brightness )
{
struct ep93xxbl * ep93xxbl = bl_get_data ( bl ) ;
2012-01-22 20:31:32 +11:00
writel ( ( brightness < < 8 ) | EP93XX_MAX_COUNT , ep93xxbl - > mmio ) ;
2010-05-05 12:13:23 -05:00
ep93xxbl - > brightness = brightness ;
return 0 ;
}
static int ep93xxbl_update_status ( struct backlight_device * bl )
{
int brightness = bl - > props . brightness ;
if ( bl - > props . power ! = FB_BLANK_UNBLANK | |
bl - > props . fb_blank ! = FB_BLANK_UNBLANK )
brightness = 0 ;
return ep93xxbl_set ( bl , brightness ) ;
}
static int ep93xxbl_get_brightness ( struct backlight_device * bl )
{
struct ep93xxbl * ep93xxbl = bl_get_data ( bl ) ;
return ep93xxbl - > brightness ;
}
static const struct backlight_ops ep93xxbl_ops = {
. update_status = ep93xxbl_update_status ,
. get_brightness = ep93xxbl_get_brightness ,
} ;
2013-04-29 16:17:27 -07:00
static int ep93xxbl_probe ( struct platform_device * dev )
2010-05-05 12:13:23 -05:00
{
struct ep93xxbl * ep93xxbl ;
struct backlight_device * bl ;
struct backlight_properties props ;
2012-01-22 20:31:32 +11:00
struct resource * res ;
2010-05-05 12:13:23 -05:00
ep93xxbl = devm_kzalloc ( & dev - > dev , sizeof ( * ep93xxbl ) , GFP_KERNEL ) ;
if ( ! ep93xxbl )
return - ENOMEM ;
2012-01-22 20:31:32 +11:00
res = platform_get_resource ( dev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - ENXIO ;
2010-05-05 12:13:23 -05:00
/*
2012-01-22 20:31:32 +11:00
* FIXME - We don ' t do a request_mem_region here because we are
* sharing the register space with the framebuffer driver ( see
* drivers / video / ep93xx - fb . c ) and doing so will cause the second
* loaded driver to return - EBUSY .
2010-05-05 12:13:23 -05:00
*
* NOTE : No locking is required ; the framebuffer does not touch
* this register .
*/
2012-01-22 20:31:32 +11:00
ep93xxbl - > mmio = devm_ioremap ( & dev - > dev , res - > start ,
resource_size ( res ) ) ;
if ( ! ep93xxbl - > mmio )
return - ENXIO ;
2010-05-05 12:13:23 -05:00
memset ( & props , 0 , sizeof ( struct backlight_properties ) ) ;
2011-03-22 16:30:21 -07:00
props . type = BACKLIGHT_RAW ;
2010-05-05 12:13:23 -05:00
props . max_brightness = EP93XX_MAX_BRIGHT ;
2013-11-12 15:09:14 -08:00
bl = devm_backlight_device_register ( & dev - > dev , dev - > name , & dev - > dev ,
ep93xxbl , & ep93xxbl_ops , & props ) ;
2010-05-05 12:13:23 -05:00
if ( IS_ERR ( bl ) )
return PTR_ERR ( bl ) ;
bl - > props . brightness = EP93XX_DEF_BRIGHT ;
platform_set_drvdata ( dev , bl ) ;
ep93xxbl_update_status ( bl ) ;
return 0 ;
}
2013-04-29 16:17:42 -07:00
# ifdef CONFIG_PM_SLEEP
static int ep93xxbl_suspend ( struct device * dev )
2010-05-05 12:13:23 -05:00
{
2013-04-29 16:17:42 -07:00
struct backlight_device * bl = dev_get_drvdata ( dev ) ;
2010-05-05 12:13:23 -05:00
return ep93xxbl_set ( bl , 0 ) ;
}
2013-04-29 16:17:42 -07:00
static int ep93xxbl_resume ( struct device * dev )
2010-05-05 12:13:23 -05:00
{
2013-04-29 16:17:42 -07:00
struct backlight_device * bl = dev_get_drvdata ( dev ) ;
2010-05-05 12:13:23 -05:00
backlight_update_status ( bl ) ;
return 0 ;
}
# endif
2013-04-29 16:17:42 -07:00
static SIMPLE_DEV_PM_OPS ( ep93xxbl_pm_ops , ep93xxbl_suspend , ep93xxbl_resume ) ;
2010-05-05 12:13:23 -05:00
static struct platform_driver ep93xxbl_driver = {
. driver = {
. name = " ep93xx-bl " ,
2013-04-29 16:17:42 -07:00
. pm = & ep93xxbl_pm_ops ,
2010-05-05 12:13:23 -05:00
} ,
. probe = ep93xxbl_probe ,
} ;
2012-01-10 15:09:11 -08:00
module_platform_driver ( ep93xxbl_driver ) ;
2010-05-05 12:13:23 -05:00
MODULE_DESCRIPTION ( " EP93xx Backlight Driver " ) ;
MODULE_AUTHOR ( " H Hartley Sweeten <hsweeten@visionengravers.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform:ep93xx-bl " ) ;