2012-10-31 13:48:40 +04:00
/*
* Nokia RX - 51 battery driver
*
* Copyright ( C ) 2012 Pali Rohár < pali . rohar @ gmail . 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 . ,
* 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA .
*/
# include <linux/module.h>
# include <linux/param.h>
# include <linux/platform_device.h>
# include <linux/power_supply.h>
# include <linux/slab.h>
2014-03-02 00:22:44 +04:00
# include <linux/iio/consumer.h>
# include <linux/of.h>
2013-08-22 02:45:10 +04:00
2012-10-31 13:48:40 +04:00
struct rx51_device_info {
struct device * dev ;
2015-03-12 10:44:11 +03:00
struct power_supply * bat ;
struct power_supply_desc bat_desc ;
2014-03-02 00:22:44 +04:00
struct iio_channel * channel_temp ;
struct iio_channel * channel_bsi ;
struct iio_channel * channel_vbat ;
2012-10-31 13:48:40 +04:00
} ;
/*
* Read ADCIN channel value , code copied from maemo kernel
*/
2014-03-02 00:22:44 +04:00
static int rx51_battery_read_adc ( struct iio_channel * channel )
2012-10-31 13:48:40 +04:00
{
2014-03-02 00:22:44 +04:00
int val , err ;
err = iio_read_channel_average_raw ( channel , & val ) ;
if ( err < 0 )
return err ;
return val ;
2012-10-31 13:48:40 +04:00
}
/*
* Read ADCIN channel 12 ( voltage ) and convert RAW value to micro voltage
* This conversion formula was extracted from maemo program bsi - read
*/
static int rx51_battery_read_voltage ( struct rx51_device_info * di )
{
2014-03-02 00:22:44 +04:00
int voltage = rx51_battery_read_adc ( di - > channel_vbat ) ;
2012-10-31 13:48:40 +04:00
2014-03-02 00:22:44 +04:00
if ( voltage < 0 ) {
dev_err ( di - > dev , " Could not read ADC: %d \n " , voltage ) ;
2012-10-31 13:48:40 +04:00
return voltage ;
2014-03-02 00:22:44 +04:00
}
2012-10-31 13:48:40 +04:00
return 1000 * ( 10000 * voltage / 1705 ) ;
}
/*
* Temperature look - up tables
* TEMP = ( 1 / ( t1 + 1 / 298 ) - 273.15 )
* Where t1 = ( 1 / B ) * ln ( ( RAW_ADC_U * 2.5 ) / ( R * I * 255 ) )
* Formula is based on experimental data , RX - 51 CAL data , maemo program bme
* and formula from da9052 driver with values R = 100 , B = 3380 , I = 0.00671
*/
/*
* Table1 ( temperature for first 25 RAW values )
* Usage : TEMP = rx51_temp_table1 [ RAW ]
* RAW is between 1 and 24
* TEMP is between 201 C and 55 C
*/
static u8 rx51_temp_table1 [ ] = {
255 , 201 , 159 , 138 , 124 , 114 , 106 , 99 , 94 , 89 , 85 , 82 , 78 , 75 ,
73 , 70 , 68 , 66 , 64 , 62 , 61 , 59 , 57 , 56 , 55
} ;
/*
* Table2 ( lowest RAW value for temperature )
* Usage : RAW = rx51_temp_table2 [ TEMP - rx51_temp_table2_first ]
* TEMP is between 53 C and - 32 C
* RAW is between 25 and 993
*/
# define rx51_temp_table2_first 53
static u16 rx51_temp_table2 [ ] = {
25 , 26 , 27 , 28 , 29 , 30 , 31 , 32 , 33 , 34 , 35 , 36 , 37 , 39 ,
40 , 41 , 43 , 44 , 46 , 48 , 49 , 51 , 53 , 55 , 57 , 59 , 61 , 64 ,
66 , 69 , 71 , 74 , 77 , 80 , 83 , 86 , 90 , 94 , 97 , 101 , 106 , 110 ,
115 , 119 , 125 , 130 , 136 , 141 , 148 , 154 , 161 , 168 , 176 , 184 , 202 , 211 ,
221 , 231 , 242 , 254 , 266 , 279 , 293 , 308 , 323 , 340 , 357 , 375 , 395 , 415 ,
437 , 460 , 485 , 511 , 539 , 568 , 600 , 633 , 669 , 706 , 747 , 790 , 836 , 885 ,
937 , 993 , 1024
} ;
/*
* Read ADCIN channel 0 ( battery temp ) and convert value to tenths of Celsius
* Use Temperature look - up tables for conversation
*/
static int rx51_battery_read_temperature ( struct rx51_device_info * di )
{
int min = 0 ;
int max = ARRAY_SIZE ( rx51_temp_table2 ) - 1 ;
2014-03-02 00:22:44 +04:00
int raw = rx51_battery_read_adc ( di - > channel_temp ) ;
if ( raw < 0 )
dev_err ( di - > dev , " Could not read ADC: %d \n " , raw ) ;
2012-10-31 13:48:40 +04:00
/* Zero and negative values are undefined */
if ( raw < = 0 )
return INT_MAX ;
/* ADC channels are 10 bit, higher value are undefined */
if ( raw > = ( 1 < < 10 ) )
return INT_MIN ;
/* First check for temperature in first direct table */
if ( raw < ARRAY_SIZE ( rx51_temp_table1 ) )
2013-03-28 20:42:23 +04:00
return rx51_temp_table1 [ raw ] * 10 ;
2012-10-31 13:48:40 +04:00
/* Binary search RAW value in second inverse table */
while ( max - min > 1 ) {
int mid = ( max + min ) / 2 ;
if ( rx51_temp_table2 [ mid ] < = raw )
min = mid ;
else if ( rx51_temp_table2 [ mid ] > raw )
max = mid ;
if ( rx51_temp_table2 [ mid ] = = raw )
break ;
}
2013-03-28 20:42:23 +04:00
return ( rx51_temp_table2_first - min ) * 10 ;
2012-10-31 13:48:40 +04:00
}
/*
* Read ADCIN channel 4 ( BSI ) and convert RAW value to micro Ah
* This conversion formula was extracted from maemo program bsi - read
*/
static int rx51_battery_read_capacity ( struct rx51_device_info * di )
{
2014-03-02 00:22:44 +04:00
int capacity = rx51_battery_read_adc ( di - > channel_bsi ) ;
2012-10-31 13:48:40 +04:00
2014-03-02 00:22:44 +04:00
if ( capacity < 0 ) {
dev_err ( di - > dev , " Could not read ADC: %d \n " , capacity ) ;
2012-10-31 13:48:40 +04:00
return capacity ;
2014-03-02 00:22:44 +04:00
}
2012-10-31 13:48:40 +04:00
return 1280 * ( 1200 * capacity ) / ( 1024 - capacity ) ;
}
/*
* Return power_supply property
*/
static int rx51_battery_get_property ( struct power_supply * psy ,
enum power_supply_property psp ,
union power_supply_propval * val )
{
2015-03-12 10:44:11 +03:00
struct rx51_device_info * di = power_supply_get_drvdata ( psy ) ;
2012-10-31 13:48:40 +04:00
switch ( psp ) {
case POWER_SUPPLY_PROP_TECHNOLOGY :
val - > intval = POWER_SUPPLY_TECHNOLOGY_LION ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN :
val - > intval = 4200000 ;
break ;
case POWER_SUPPLY_PROP_PRESENT :
val - > intval = rx51_battery_read_voltage ( di ) ? 1 : 0 ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_NOW :
val - > intval = rx51_battery_read_voltage ( di ) ;
break ;
case POWER_SUPPLY_PROP_TEMP :
val - > intval = rx51_battery_read_temperature ( di ) ;
break ;
case POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN :
val - > intval = rx51_battery_read_capacity ( di ) ;
break ;
default :
return - EINVAL ;
}
if ( val - > intval = = INT_MAX | | val - > intval = = INT_MIN )
return - EINVAL ;
return 0 ;
}
static enum power_supply_property rx51_battery_props [ ] = {
POWER_SUPPLY_PROP_TECHNOLOGY ,
POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN ,
POWER_SUPPLY_PROP_PRESENT ,
POWER_SUPPLY_PROP_VOLTAGE_NOW ,
POWER_SUPPLY_PROP_TEMP ,
POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN ,
} ;
2012-12-22 03:04:54 +04:00
static int rx51_battery_probe ( struct platform_device * pdev )
2012-10-31 13:48:40 +04:00
{
2015-03-12 10:44:11 +03:00
struct power_supply_config psy_cfg = { } ;
2012-10-31 13:48:40 +04:00
struct rx51_device_info * di ;
int ret ;
2013-03-11 10:35:58 +04:00
di = devm_kzalloc ( & pdev - > dev , sizeof ( * di ) , GFP_KERNEL ) ;
2012-10-31 13:48:40 +04:00
if ( ! di )
return - ENOMEM ;
platform_set_drvdata ( pdev , di ) ;
2014-03-02 00:22:44 +04:00
di - > dev = & pdev - > dev ;
2015-07-28 21:44:41 +03:00
di - > bat_desc . name = " rx51-battery " ;
2015-03-12 10:44:11 +03:00
di - > bat_desc . type = POWER_SUPPLY_TYPE_BATTERY ;
di - > bat_desc . properties = rx51_battery_props ;
di - > bat_desc . num_properties = ARRAY_SIZE ( rx51_battery_props ) ;
di - > bat_desc . get_property = rx51_battery_get_property ;
psy_cfg . drv_data = di ;
2012-10-31 13:48:40 +04:00
2014-03-02 00:22:44 +04:00
di - > channel_temp = iio_channel_get ( di - > dev , " temp " ) ;
if ( IS_ERR ( di - > channel_temp ) ) {
ret = PTR_ERR ( di - > channel_temp ) ;
goto error ;
}
di - > channel_bsi = iio_channel_get ( di - > dev , " bsi " ) ;
if ( IS_ERR ( di - > channel_bsi ) ) {
ret = PTR_ERR ( di - > channel_bsi ) ;
goto error_channel_temp ;
}
di - > channel_vbat = iio_channel_get ( di - > dev , " vbat " ) ;
if ( IS_ERR ( di - > channel_vbat ) ) {
ret = PTR_ERR ( di - > channel_vbat ) ;
goto error_channel_bsi ;
}
2015-03-12 10:44:11 +03:00
di - > bat = power_supply_register ( di - > dev , & di - > bat_desc , & psy_cfg ) ;
if ( IS_ERR ( di - > bat ) ) {
ret = PTR_ERR ( di - > bat ) ;
2014-03-02 00:22:44 +04:00
goto error_channel_vbat ;
2015-03-12 10:44:11 +03:00
}
2012-10-31 13:48:40 +04:00
return 0 ;
2014-03-02 00:22:44 +04:00
error_channel_vbat :
iio_channel_release ( di - > channel_vbat ) ;
error_channel_bsi :
iio_channel_release ( di - > channel_bsi ) ;
error_channel_temp :
iio_channel_release ( di - > channel_temp ) ;
error :
return ret ;
2012-10-31 13:48:40 +04:00
}
2012-12-22 03:04:54 +04:00
static int rx51_battery_remove ( struct platform_device * pdev )
2012-10-31 13:48:40 +04:00
{
struct rx51_device_info * di = platform_get_drvdata ( pdev ) ;
2015-03-12 10:44:11 +03:00
power_supply_unregister ( di - > bat ) ;
2012-10-31 13:48:40 +04:00
2014-03-02 00:22:44 +04:00
iio_channel_release ( di - > channel_vbat ) ;
iio_channel_release ( di - > channel_bsi ) ;
iio_channel_release ( di - > channel_temp ) ;
2012-10-31 13:48:40 +04:00
return 0 ;
}
2014-03-02 00:22:44 +04:00
# ifdef CONFIG_OF
static const struct of_device_id n900_battery_of_match [ ] = {
{ . compatible = " nokia,n900-battery " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , n900_battery_of_match ) ;
# endif
2012-10-31 13:48:40 +04:00
static struct platform_driver rx51_battery_driver = {
. probe = rx51_battery_probe ,
2012-12-22 03:04:54 +04:00
. remove = rx51_battery_remove ,
2012-10-31 13:48:40 +04:00
. driver = {
. name = " rx51-battery " ,
2014-03-02 00:22:44 +04:00
. of_match_table = of_match_ptr ( n900_battery_of_match ) ,
2012-10-31 13:48:40 +04:00
} ,
} ;
module_platform_driver ( rx51_battery_driver ) ;
MODULE_ALIAS ( " platform:rx51-battery " ) ;
MODULE_AUTHOR ( " Pali Rohár <pali.rohar@gmail.com> " ) ;
MODULE_DESCRIPTION ( " Nokia RX-51 battery driver " ) ;
MODULE_LICENSE ( " GPL " ) ;