2011-05-05 15:39:25 -03:00
/*
* drivers / media / video / adp1653 . c
*
* Copyright ( C ) 2008 - - 2011 Nokia Corporation
*
* Contact : Sakari Ailus < sakari . ailus @ maxwell . research . nokia . com >
*
* Contributors :
* Sakari Ailus < sakari . ailus @ maxwell . research . nokia . com >
* Tuukka Toivonen < tuukkat76 @ gmail . 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 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 . , 51 Franklin St , Fifth Floor , Boston , MA
* 02110 - 1301 USA
*
* TODO :
* - fault interrupt handling
* - hardware strobe
* - power doesn ' t need to be ON if all lights are off
*
*/
# include <linux/delay.h>
# include <linux/i2c.h>
2011-08-03 13:12:26 -03:00
# include <linux/module.h>
2011-05-05 15:39:25 -03:00
# include <linux/slab.h>
# include <linux/version.h>
# include <media/adp1653.h>
# include <media/v4l2-device.h>
# define TIMEOUT_MAX 820000
# define TIMEOUT_STEP 54600
# define TIMEOUT_MIN (TIMEOUT_MAX - ADP1653_REG_CONFIG_TMR_SET_MAX \
* TIMEOUT_STEP )
# define TIMEOUT_US_TO_CODE(t) ((TIMEOUT_MAX + (TIMEOUT_STEP / 2) - (t)) \
/ TIMEOUT_STEP )
# define TIMEOUT_CODE_TO_US(c) (TIMEOUT_MAX - (c) * TIMEOUT_STEP)
/* Write values into ADP1653 registers. */
static int adp1653_update_hw ( struct adp1653_flash * flash )
{
struct i2c_client * client = v4l2_get_subdevdata ( & flash - > subdev ) ;
u8 out_sel ;
u8 config = 0 ;
int rval ;
out_sel = ADP1653_INDICATOR_INTENSITY_uA_TO_REG (
flash - > indicator_intensity - > val )
< < ADP1653_REG_OUT_SEL_ILED_SHIFT ;
switch ( flash - > led_mode - > val ) {
case V4L2_FLASH_LED_MODE_NONE :
break ;
case V4L2_FLASH_LED_MODE_FLASH :
/* Flash mode, light on with strobe, duration from timer */
config = ADP1653_REG_CONFIG_TMR_CFG ;
config | = TIMEOUT_US_TO_CODE ( flash - > flash_timeout - > val )
< < ADP1653_REG_CONFIG_TMR_SET_SHIFT ;
break ;
case V4L2_FLASH_LED_MODE_TORCH :
/* Torch mode, light immediately on, duration indefinite */
out_sel | = ADP1653_FLASH_INTENSITY_mA_TO_REG (
flash - > torch_intensity - > val )
< < ADP1653_REG_OUT_SEL_HPLED_SHIFT ;
break ;
}
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_OUT_SEL , out_sel ) ;
if ( rval < 0 )
return rval ;
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_CONFIG , config ) ;
if ( rval < 0 )
return rval ;
return 0 ;
}
static int adp1653_get_fault ( struct adp1653_flash * flash )
{
struct i2c_client * client = v4l2_get_subdevdata ( & flash - > subdev ) ;
int fault ;
int rval ;
fault = i2c_smbus_read_byte_data ( client , ADP1653_REG_FAULT ) ;
if ( IS_ERR_VALUE ( fault ) )
return fault ;
flash - > fault | = fault ;
if ( ! flash - > fault )
return 0 ;
/* Clear faults. */
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_OUT_SEL , 0 ) ;
if ( IS_ERR_VALUE ( rval ) )
return rval ;
flash - > led_mode - > val = V4L2_FLASH_LED_MODE_NONE ;
rval = adp1653_update_hw ( flash ) ;
if ( IS_ERR_VALUE ( rval ) )
return rval ;
return flash - > fault ;
}
static int adp1653_strobe ( struct adp1653_flash * flash , int enable )
{
struct i2c_client * client = v4l2_get_subdevdata ( & flash - > subdev ) ;
u8 out_sel = ADP1653_INDICATOR_INTENSITY_uA_TO_REG (
flash - > indicator_intensity - > val )
< < ADP1653_REG_OUT_SEL_ILED_SHIFT ;
int rval ;
if ( flash - > led_mode - > val ! = V4L2_FLASH_LED_MODE_FLASH )
return - EBUSY ;
if ( ! enable )
return i2c_smbus_write_byte_data ( client , ADP1653_REG_OUT_SEL ,
out_sel ) ;
out_sel | = ADP1653_FLASH_INTENSITY_mA_TO_REG (
flash - > flash_intensity - > val )
< < ADP1653_REG_OUT_SEL_HPLED_SHIFT ;
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_OUT_SEL , out_sel ) ;
if ( rval )
return rval ;
/* Software strobe using i2c */
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_SW_STROBE ,
ADP1653_REG_SW_STROBE_SW_STROBE ) ;
if ( rval )
return rval ;
return i2c_smbus_write_byte_data ( client , ADP1653_REG_SW_STROBE , 0 ) ;
}
/* --------------------------------------------------------------------------
* V4L2 controls
*/
static int adp1653_get_ctrl ( struct v4l2_ctrl * ctrl )
{
struct adp1653_flash * flash =
container_of ( ctrl - > handler , struct adp1653_flash , ctrls ) ;
int rval ;
rval = adp1653_get_fault ( flash ) ;
if ( IS_ERR_VALUE ( rval ) )
return rval ;
ctrl - > cur . val = 0 ;
if ( flash - > fault & ADP1653_REG_FAULT_FLT_SCP )
ctrl - > cur . val | = V4L2_FLASH_FAULT_SHORT_CIRCUIT ;
if ( flash - > fault & ADP1653_REG_FAULT_FLT_OT )
ctrl - > cur . val | = V4L2_FLASH_FAULT_OVER_TEMPERATURE ;
if ( flash - > fault & ADP1653_REG_FAULT_FLT_TMR )
ctrl - > cur . val | = V4L2_FLASH_FAULT_TIMEOUT ;
if ( flash - > fault & ADP1653_REG_FAULT_FLT_OV )
ctrl - > cur . val | = V4L2_FLASH_FAULT_OVER_VOLTAGE ;
flash - > fault = 0 ;
return 0 ;
}
static int adp1653_set_ctrl ( struct v4l2_ctrl * ctrl )
{
struct adp1653_flash * flash =
container_of ( ctrl - > handler , struct adp1653_flash , ctrls ) ;
int rval ;
rval = adp1653_get_fault ( flash ) ;
if ( IS_ERR_VALUE ( rval ) )
return rval ;
if ( ( rval & ( ADP1653_REG_FAULT_FLT_SCP |
ADP1653_REG_FAULT_FLT_OT |
ADP1653_REG_FAULT_FLT_OV ) ) & &
( ctrl - > id = = V4L2_CID_FLASH_STROBE | |
ctrl - > id = = V4L2_CID_FLASH_TORCH_INTENSITY | |
ctrl - > id = = V4L2_CID_FLASH_LED_MODE ) )
return - EBUSY ;
switch ( ctrl - > id ) {
case V4L2_CID_FLASH_STROBE :
return adp1653_strobe ( flash , 1 ) ;
case V4L2_CID_FLASH_STROBE_STOP :
return adp1653_strobe ( flash , 0 ) ;
}
return adp1653_update_hw ( flash ) ;
}
static const struct v4l2_ctrl_ops adp1653_ctrl_ops = {
. g_volatile_ctrl = adp1653_get_ctrl ,
. s_ctrl = adp1653_set_ctrl ,
} ;
static int adp1653_init_controls ( struct adp1653_flash * flash )
{
struct v4l2_ctrl * fault ;
v4l2_ctrl_handler_init ( & flash - > ctrls , 9 ) ;
flash - > led_mode =
v4l2_ctrl_new_std_menu ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_LED_MODE ,
V4L2_FLASH_LED_MODE_TORCH , ~ 0x7 , 0 ) ;
v4l2_ctrl_new_std_menu ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_STROBE_SOURCE ,
V4L2_FLASH_STROBE_SOURCE_SOFTWARE , ~ 0x1 , 0 ) ;
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_STROBE , 0 , 0 , 0 , 0 ) ;
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_STROBE_STOP , 0 , 0 , 0 , 0 ) ;
flash - > flash_timeout =
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_TIMEOUT , TIMEOUT_MIN ,
flash - > platform_data - > max_flash_timeout ,
TIMEOUT_STEP ,
flash - > platform_data - > max_flash_timeout ) ;
flash - > flash_intensity =
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_INTENSITY ,
ADP1653_FLASH_INTENSITY_MIN ,
flash - > platform_data - > max_flash_intensity ,
1 , flash - > platform_data - > max_flash_intensity ) ;
flash - > torch_intensity =
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_TORCH_INTENSITY ,
ADP1653_TORCH_INTENSITY_MIN ,
flash - > platform_data - > max_torch_intensity ,
ADP1653_FLASH_INTENSITY_STEP ,
flash - > platform_data - > max_torch_intensity ) ;
flash - > indicator_intensity =
v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_INDICATOR_INTENSITY ,
ADP1653_INDICATOR_INTENSITY_MIN ,
flash - > platform_data - > max_indicator_intensity ,
ADP1653_INDICATOR_INTENSITY_STEP ,
ADP1653_INDICATOR_INTENSITY_MIN ) ;
fault = v4l2_ctrl_new_std ( & flash - > ctrls , & adp1653_ctrl_ops ,
V4L2_CID_FLASH_FAULT , 0 ,
V4L2_FLASH_FAULT_OVER_VOLTAGE
| V4L2_FLASH_FAULT_OVER_TEMPERATURE
| V4L2_FLASH_FAULT_SHORT_CIRCUIT , 0 , 0 ) ;
if ( flash - > ctrls . error )
return flash - > ctrls . error ;
2011-08-26 07:35:14 -03:00
fault - > flags | = V4L2_CTRL_FLAG_VOLATILE ;
2011-05-05 15:39:25 -03:00
flash - > subdev . ctrl_handler = & flash - > ctrls ;
return 0 ;
}
/* --------------------------------------------------------------------------
* V4L2 subdev operations
*/
static int
adp1653_init_device ( struct adp1653_flash * flash )
{
struct i2c_client * client = v4l2_get_subdevdata ( & flash - > subdev ) ;
int rval ;
/* Clear FAULT register by writing zero to OUT_SEL */
rval = i2c_smbus_write_byte_data ( client , ADP1653_REG_OUT_SEL , 0 ) ;
if ( rval < 0 ) {
dev_err ( & client - > dev , " failed writing fault register \n " ) ;
return - EIO ;
}
mutex_lock ( & flash - > ctrls . lock ) ;
/* Reset faults before reading new ones. */
flash - > fault = 0 ;
rval = adp1653_get_fault ( flash ) ;
mutex_unlock ( & flash - > ctrls . lock ) ;
if ( rval > 0 ) {
dev_err ( & client - > dev , " faults detected: 0x%1.1x \n " , rval ) ;
return - EIO ;
}
mutex_lock ( & flash - > ctrls . lock ) ;
rval = adp1653_update_hw ( flash ) ;
mutex_unlock ( & flash - > ctrls . lock ) ;
if ( rval ) {
dev_err ( & client - > dev ,
" adp1653_update_hw failed at %s \n " , __func__ ) ;
return - EIO ;
}
return 0 ;
}
static int
__adp1653_set_power ( struct adp1653_flash * flash , int on )
{
int ret ;
ret = flash - > platform_data - > power ( & flash - > subdev , on ) ;
if ( ret < 0 )
return ret ;
if ( ! on )
return 0 ;
ret = adp1653_init_device ( flash ) ;
if ( ret < 0 )
flash - > platform_data - > power ( & flash - > subdev , 0 ) ;
return ret ;
}
static int
adp1653_set_power ( struct v4l2_subdev * subdev , int on )
{
struct adp1653_flash * flash = to_adp1653_flash ( subdev ) ;
int ret = 0 ;
mutex_lock ( & flash - > power_lock ) ;
/* If the power count is modified from 0 to != 0 or from != 0 to 0,
* update the power state .
*/
if ( flash - > power_count = = ! on ) {
ret = __adp1653_set_power ( flash , ! ! on ) ;
if ( ret < 0 )
goto done ;
}
/* Update the power count. */
flash - > power_count + = on ? 1 : - 1 ;
WARN_ON ( flash - > power_count < 0 ) ;
done :
mutex_unlock ( & flash - > power_lock ) ;
return ret ;
}
static int adp1653_open ( struct v4l2_subdev * sd , struct v4l2_subdev_fh * fh )
{
return adp1653_set_power ( sd , 1 ) ;
}
static int adp1653_close ( struct v4l2_subdev * sd , struct v4l2_subdev_fh * fh )
{
return adp1653_set_power ( sd , 0 ) ;
}
static const struct v4l2_subdev_core_ops adp1653_core_ops = {
. s_power = adp1653_set_power ,
} ;
static const struct v4l2_subdev_ops adp1653_ops = {
. core = & adp1653_core_ops ,
} ;
static const struct v4l2_subdev_internal_ops adp1653_internal_ops = {
. open = adp1653_open ,
. close = adp1653_close ,
} ;
/* --------------------------------------------------------------------------
* I2C driver
*/
# ifdef CONFIG_PM
static int adp1653_suspend ( struct device * dev )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct v4l2_subdev * subdev = i2c_get_clientdata ( client ) ;
struct adp1653_flash * flash = to_adp1653_flash ( subdev ) ;
if ( ! flash - > power_count )
return 0 ;
return __adp1653_set_power ( flash , 0 ) ;
}
static int adp1653_resume ( struct device * dev )
{
struct i2c_client * client = to_i2c_client ( dev ) ;
struct v4l2_subdev * subdev = i2c_get_clientdata ( client ) ;
struct adp1653_flash * flash = to_adp1653_flash ( subdev ) ;
if ( ! flash - > power_count )
return 0 ;
return __adp1653_set_power ( flash , 1 ) ;
}
# else
# define adp1653_suspend NULL
# define adp1653_resume NULL
# endif /* CONFIG_PM */
static int adp1653_probe ( struct i2c_client * client ,
const struct i2c_device_id * devid )
{
struct adp1653_flash * flash ;
int ret ;
2011-07-25 11:16:41 -03:00
/* we couldn't work without platform data */
if ( client - > dev . platform_data = = NULL )
return - ENODEV ;
2011-05-05 15:39:25 -03:00
flash = kzalloc ( sizeof ( * flash ) , GFP_KERNEL ) ;
if ( flash = = NULL )
return - ENOMEM ;
flash - > platform_data = client - > dev . platform_data ;
mutex_init ( & flash - > power_lock ) ;
v4l2_i2c_subdev_init ( & flash - > subdev , client , & adp1653_ops ) ;
flash - > subdev . internal_ops = & adp1653_internal_ops ;
flash - > subdev . flags | = V4L2_SUBDEV_FL_HAS_DEVNODE ;
2011-07-28 04:59:38 -03:00
ret = adp1653_init_controls ( flash ) ;
if ( ret )
goto free_and_quit ;
2011-05-05 15:39:25 -03:00
ret = media_entity_init ( & flash - > subdev . entity , 0 , NULL , 0 ) ;
if ( ret < 0 )
2011-07-28 04:59:38 -03:00
goto free_and_quit ;
2011-05-05 15:39:25 -03:00
2011-08-11 08:35:04 -03:00
flash - > subdev . entity . type = MEDIA_ENT_T_V4L2_SUBDEV_FLASH ;
2011-07-28 04:59:38 -03:00
return 0 ;
free_and_quit :
v4l2_ctrl_handler_free ( & flash - > ctrls ) ;
kfree ( flash ) ;
2011-05-05 15:39:25 -03:00
return ret ;
}
static int __exit adp1653_remove ( struct i2c_client * client )
{
struct v4l2_subdev * subdev = i2c_get_clientdata ( client ) ;
struct adp1653_flash * flash = to_adp1653_flash ( subdev ) ;
v4l2_device_unregister_subdev ( & flash - > subdev ) ;
v4l2_ctrl_handler_free ( & flash - > ctrls ) ;
media_entity_cleanup ( & flash - > subdev . entity ) ;
kfree ( flash ) ;
return 0 ;
}
static const struct i2c_device_id adp1653_id_table [ ] = {
{ ADP1653_NAME , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , adp1653_id_table ) ;
static struct dev_pm_ops adp1653_pm_ops = {
. suspend = adp1653_suspend ,
. resume = adp1653_resume ,
} ;
static struct i2c_driver adp1653_i2c_driver = {
. driver = {
. name = ADP1653_NAME ,
. pm = & adp1653_pm_ops ,
} ,
. probe = adp1653_probe ,
. remove = __exit_p ( adp1653_remove ) ,
. id_table = adp1653_id_table ,
} ;
static int __init adp1653_init ( void )
{
int rval ;
rval = i2c_add_driver ( & adp1653_i2c_driver ) ;
if ( rval )
printk ( KERN_ALERT " %s: failed at i2c_add_driver \n " , __func__ ) ;
return rval ;
}
static void __exit adp1653_exit ( void )
{
i2c_del_driver ( & adp1653_i2c_driver ) ;
}
module_init ( adp1653_init ) ;
module_exit ( adp1653_exit ) ;
MODULE_AUTHOR ( " Sakari Ailus <sakari.ailus@nokia.com> " ) ;
MODULE_DESCRIPTION ( " Analog Devices ADP1653 LED flash driver " ) ;
MODULE_LICENSE ( " GPL " ) ;