2009-02-10 16:12:29 +03:00
/*
* Battery and Power Management code for the Sharp SL - 5 x00
*
* Copyright ( C ) 2009 Thomas Kunze
*
* based on tosa_battery . c
*
* 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/kernel.h>
# include <linux/module.h>
# include <linux/power_supply.h>
# include <linux/delay.h>
# include <linux/spinlock.h>
# include <linux/interrupt.h>
# include <linux/gpio.h>
# include <linux/mfd/ucb1x00.h>
# include <asm/mach/sharpsl_param.h>
# include <asm/mach-types.h>
# include <mach/collie.h>
static DEFINE_MUTEX ( bat_lock ) ; /* protects gpio pins */
static struct work_struct bat_work ;
static struct ucb1x00 * ucb ;
struct collie_bat {
int status ;
struct power_supply psy ;
int full_chrg ;
struct mutex work_lock ; /* protects data */
bool ( * is_present ) ( struct collie_bat * bat ) ;
int gpio_full ;
int gpio_charge_on ;
int technology ;
int gpio_bat ;
int adc_bat ;
int adc_bat_divider ;
int bat_max ;
int bat_min ;
int gpio_temp ;
int adc_temp ;
int adc_temp_divider ;
} ;
static struct collie_bat collie_bat_main ;
static unsigned long collie_read_bat ( struct collie_bat * bat )
{
unsigned long value = 0 ;
if ( bat - > gpio_bat < 0 | | bat - > adc_bat < 0 )
return 0 ;
mutex_lock ( & bat_lock ) ;
gpio_set_value ( bat - > gpio_bat , 1 ) ;
msleep ( 5 ) ;
ucb1x00_adc_enable ( ucb ) ;
value = ucb1x00_adc_read ( ucb , bat - > adc_bat , UCB_SYNC ) ;
ucb1x00_adc_disable ( ucb ) ;
gpio_set_value ( bat - > gpio_bat , 0 ) ;
mutex_unlock ( & bat_lock ) ;
value = value * 1000000 / bat - > adc_bat_divider ;
return value ;
}
static unsigned long collie_read_temp ( struct collie_bat * bat )
{
unsigned long value = 0 ;
if ( bat - > gpio_temp < 0 | | bat - > adc_temp < 0 )
return 0 ;
mutex_lock ( & bat_lock ) ;
gpio_set_value ( bat - > gpio_temp , 1 ) ;
msleep ( 5 ) ;
ucb1x00_adc_enable ( ucb ) ;
value = ucb1x00_adc_read ( ucb , bat - > adc_temp , UCB_SYNC ) ;
ucb1x00_adc_disable ( ucb ) ;
gpio_set_value ( bat - > gpio_temp , 0 ) ;
mutex_unlock ( & bat_lock ) ;
value = value * 10000 / bat - > adc_temp_divider ;
return value ;
}
static int collie_bat_get_property ( struct power_supply * psy ,
enum power_supply_property psp ,
union power_supply_propval * val )
{
int ret = 0 ;
struct collie_bat * bat = container_of ( psy , struct collie_bat , psy ) ;
if ( bat - > is_present & & ! bat - > is_present ( bat )
& & psp ! = POWER_SUPPLY_PROP_PRESENT ) {
return - ENODEV ;
}
switch ( psp ) {
case POWER_SUPPLY_PROP_STATUS :
val - > intval = bat - > status ;
break ;
case POWER_SUPPLY_PROP_TECHNOLOGY :
val - > intval = bat - > technology ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_NOW :
val - > intval = collie_read_bat ( bat ) ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_MAX :
if ( bat - > full_chrg = = - 1 )
val - > intval = bat - > bat_max ;
else
val - > intval = bat - > full_chrg ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN :
val - > intval = bat - > bat_max ;
break ;
case POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN :
val - > intval = bat - > bat_min ;
break ;
case POWER_SUPPLY_PROP_TEMP :
val - > intval = collie_read_temp ( bat ) ;
break ;
case POWER_SUPPLY_PROP_PRESENT :
val - > intval = bat - > is_present ? bat - > is_present ( bat ) : 1 ;
break ;
default :
ret = - EINVAL ;
break ;
}
return ret ;
}
static void collie_bat_external_power_changed ( struct power_supply * psy )
{
schedule_work ( & bat_work ) ;
}
static irqreturn_t collie_bat_gpio_isr ( int irq , void * data )
{
pr_info ( " collie_bat_gpio irq: %d \n " , gpio_get_value ( irq_to_gpio ( irq ) ) ) ;
schedule_work ( & bat_work ) ;
return IRQ_HANDLED ;
}
static void collie_bat_update ( struct collie_bat * bat )
{
int old ;
struct power_supply * psy = & bat - > psy ;
mutex_lock ( & bat - > work_lock ) ;
old = bat - > status ;
if ( bat - > is_present & & ! bat - > is_present ( bat ) ) {
printk ( KERN_NOTICE " %s not present \n " , psy - > name ) ;
bat - > status = POWER_SUPPLY_STATUS_UNKNOWN ;
bat - > full_chrg = - 1 ;
} else if ( power_supply_am_i_supplied ( psy ) ) {
if ( bat - > status = = POWER_SUPPLY_STATUS_DISCHARGING ) {
gpio_set_value ( bat - > gpio_charge_on , 1 ) ;
mdelay ( 15 ) ;
}
if ( gpio_get_value ( bat - > gpio_full ) ) {
if ( old = = POWER_SUPPLY_STATUS_CHARGING | |
bat - > full_chrg = = - 1 )
bat - > full_chrg = collie_read_bat ( bat ) ;
gpio_set_value ( bat - > gpio_charge_on , 0 ) ;
bat - > status = POWER_SUPPLY_STATUS_FULL ;
} else {
gpio_set_value ( bat - > gpio_charge_on , 1 ) ;
bat - > status = POWER_SUPPLY_STATUS_CHARGING ;
}
} else {
gpio_set_value ( bat - > gpio_charge_on , 0 ) ;
bat - > status = POWER_SUPPLY_STATUS_DISCHARGING ;
}
if ( old ! = bat - > status )
power_supply_changed ( psy ) ;
mutex_unlock ( & bat - > work_lock ) ;
}
static void collie_bat_work ( struct work_struct * work )
{
collie_bat_update ( & collie_bat_main ) ;
}
static enum power_supply_property collie_bat_main_props [ ] = {
POWER_SUPPLY_PROP_STATUS ,
POWER_SUPPLY_PROP_TECHNOLOGY ,
POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN ,
POWER_SUPPLY_PROP_VOLTAGE_NOW ,
POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN ,
POWER_SUPPLY_PROP_VOLTAGE_MAX ,
POWER_SUPPLY_PROP_PRESENT ,
POWER_SUPPLY_PROP_TEMP ,
} ;
static enum power_supply_property collie_bat_bu_props [ ] = {
POWER_SUPPLY_PROP_STATUS ,
POWER_SUPPLY_PROP_TECHNOLOGY ,
POWER_SUPPLY_PROP_VOLTAGE_MIN_DESIGN ,
POWER_SUPPLY_PROP_VOLTAGE_NOW ,
POWER_SUPPLY_PROP_VOLTAGE_MAX_DESIGN ,
POWER_SUPPLY_PROP_VOLTAGE_MAX ,
POWER_SUPPLY_PROP_PRESENT ,
} ;
static struct collie_bat collie_bat_main = {
. status = POWER_SUPPLY_STATUS_DISCHARGING ,
. full_chrg = - 1 ,
. psy = {
. name = " main-battery " ,
. type = POWER_SUPPLY_TYPE_BATTERY ,
. properties = collie_bat_main_props ,
. num_properties = ARRAY_SIZE ( collie_bat_main_props ) ,
. get_property = collie_bat_get_property ,
. external_power_changed = collie_bat_external_power_changed ,
. use_for_apm = 1 ,
} ,
. gpio_full = COLLIE_GPIO_CO ,
. gpio_charge_on = COLLIE_GPIO_CHARGE_ON ,
. technology = POWER_SUPPLY_TECHNOLOGY_LIPO ,
. gpio_bat = COLLIE_GPIO_MBAT_ON ,
. adc_bat = UCB_ADC_INP_AD1 ,
. adc_bat_divider = 155 ,
. bat_max = 4310000 ,
. bat_min = 1551 * 1000000 / 414 ,
. gpio_temp = COLLIE_GPIO_TMP_ON ,
. adc_temp = UCB_ADC_INP_AD0 ,
. adc_temp_divider = 10000 ,
} ;
static struct collie_bat collie_bat_bu = {
. status = POWER_SUPPLY_STATUS_UNKNOWN ,
. full_chrg = - 1 ,
. psy = {
. name = " backup-battery " ,
. type = POWER_SUPPLY_TYPE_BATTERY ,
. properties = collie_bat_bu_props ,
. num_properties = ARRAY_SIZE ( collie_bat_bu_props ) ,
. get_property = collie_bat_get_property ,
. external_power_changed = collie_bat_external_power_changed ,
} ,
. gpio_full = - 1 ,
. gpio_charge_on = - 1 ,
. technology = POWER_SUPPLY_TECHNOLOGY_LiMn ,
. gpio_bat = COLLIE_GPIO_BBAT_ON ,
. adc_bat = UCB_ADC_INP_AD1 ,
. adc_bat_divider = 155 ,
. bat_max = 3000000 ,
. bat_min = 1900000 ,
. gpio_temp = - 1 ,
. adc_temp = - 1 ,
. adc_temp_divider = - 1 ,
} ;
static struct {
int gpio ;
char * name ;
bool output ;
int value ;
} gpios [ ] = {
{ COLLIE_GPIO_CO , " main battery full " , 0 , 0 } ,
{ COLLIE_GPIO_MAIN_BAT_LOW , " main battery low " , 0 , 0 } ,
{ COLLIE_GPIO_CHARGE_ON , " main charge on " , 1 , 0 } ,
{ COLLIE_GPIO_MBAT_ON , " main battery " , 1 , 0 } ,
{ COLLIE_GPIO_TMP_ON , " main battery temp " , 1 , 0 } ,
{ COLLIE_GPIO_BBAT_ON , " backup battery " , 1 , 0 } ,
} ;
# ifdef CONFIG_PM
static int collie_bat_suspend ( struct ucb1x00_dev * dev , pm_message_t state )
{
/* flush all pending status updates */
2010-12-11 19:51:45 +03:00
flush_work_sync ( & bat_work ) ;
2009-02-10 16:12:29 +03:00
return 0 ;
}
static int collie_bat_resume ( struct ucb1x00_dev * dev )
{
/* things may have changed while we were away */
schedule_work ( & bat_work ) ;
return 0 ;
}
# else
# define collie_bat_suspend NULL
# define collie_bat_resume NULL
# endif
static int __devinit collie_bat_probe ( struct ucb1x00_dev * dev )
{
int ret ;
int i ;
if ( ! machine_is_collie ( ) )
return - ENODEV ;
ucb = dev - > ucb ;
for ( i = 0 ; i < ARRAY_SIZE ( gpios ) ; i + + ) {
ret = gpio_request ( gpios [ i ] . gpio , gpios [ i ] . name ) ;
if ( ret ) {
i - - ;
goto err_gpio ;
}
if ( gpios [ i ] . output )
ret = gpio_direction_output ( gpios [ i ] . gpio ,
gpios [ i ] . value ) ;
else
ret = gpio_direction_input ( gpios [ i ] . gpio ) ;
if ( ret )
goto err_gpio ;
}
mutex_init ( & collie_bat_main . work_lock ) ;
INIT_WORK ( & bat_work , collie_bat_work ) ;
ret = power_supply_register ( & dev - > ucb - > dev , & collie_bat_main . psy ) ;
if ( ret )
goto err_psy_reg_main ;
ret = power_supply_register ( & dev - > ucb - > dev , & collie_bat_bu . psy ) ;
if ( ret )
goto err_psy_reg_bu ;
ret = request_irq ( gpio_to_irq ( COLLIE_GPIO_CO ) ,
collie_bat_gpio_isr ,
IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING ,
" main full " , & collie_bat_main ) ;
if ( ! ret ) {
schedule_work ( & bat_work ) ;
return 0 ;
}
power_supply_unregister ( & collie_bat_bu . psy ) ;
err_psy_reg_bu :
power_supply_unregister ( & collie_bat_main . psy ) ;
err_psy_reg_main :
/* see comment in collie_bat_remove */
2010-12-11 19:51:45 +03:00
cancel_work_sync ( & bat_work ) ;
2009-02-10 16:12:29 +03:00
i - - ;
err_gpio :
for ( ; i > = 0 ; i - - )
gpio_free ( gpios [ i ] . gpio ) ;
return ret ;
}
static void __devexit collie_bat_remove ( struct ucb1x00_dev * dev )
{
int i ;
free_irq ( gpio_to_irq ( COLLIE_GPIO_CO ) , & collie_bat_main ) ;
power_supply_unregister ( & collie_bat_bu . psy ) ;
power_supply_unregister ( & collie_bat_main . psy ) ;
/*
2010-12-11 19:51:45 +03:00
* Now cancel the bat_work . We won ' t get any more schedules ,
* since all sources ( isr and external_power_changed ) are
* unregistered now .
2009-02-10 16:12:29 +03:00
*/
2010-12-11 19:51:45 +03:00
cancel_work_sync ( & bat_work ) ;
2009-02-10 16:12:29 +03:00
for ( i = ARRAY_SIZE ( gpios ) - 1 ; i > = 0 ; i - - )
gpio_free ( gpios [ i ] . gpio ) ;
}
static struct ucb1x00_driver collie_bat_driver = {
. add = collie_bat_probe ,
. remove = __devexit_p ( collie_bat_remove ) ,
. suspend = collie_bat_suspend ,
. resume = collie_bat_resume ,
} ;
static int __init collie_bat_init ( void )
{
return ucb1x00_register_driver ( & collie_bat_driver ) ;
}
static void __exit collie_bat_exit ( void )
{
ucb1x00_unregister_driver ( & collie_bat_driver ) ;
}
module_init ( collie_bat_init ) ;
module_exit ( collie_bat_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Thomas Kunze " ) ;
MODULE_DESCRIPTION ( " Collie battery driver " ) ;