2011-03-29 05:10:16 +04:00
/*
* max8903_charger . c - Maxim 8903 USB / Adapter Charger Driver
*
* Copyright ( C ) 2011 Samsung Electronics
* MyungJoo Ham < myungjoo . ham @ samsung . 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
# include <linux/gpio.h>
# include <linux/interrupt.h>
2011-07-03 23:28:29 +04:00
# include <linux/module.h>
2011-03-29 05:10:16 +04:00
# include <linux/slab.h>
# include <linux/power_supply.h>
# include <linux/platform_device.h>
# include <linux/power/max8903_charger.h>
struct max8903_data {
2011-06-30 05:09:40 +04:00
struct max8903_pdata pdata ;
2011-03-29 05:10:16 +04:00
struct device * dev ;
2015-03-12 10:44:11 +03:00
struct power_supply * psy ;
struct power_supply_desc psy_desc ;
2011-03-29 05:10:16 +04:00
bool fault ;
bool usb_in ;
bool ta_in ;
} ;
static enum power_supply_property max8903_charger_props [ ] = {
POWER_SUPPLY_PROP_STATUS , /* Charger status output */
POWER_SUPPLY_PROP_ONLINE , /* External power source */
POWER_SUPPLY_PROP_HEALTH , /* Fault or OK */
} ;
static int max8903_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 max8903_data * data = power_supply_get_drvdata ( psy ) ;
2011-03-29 05:10:16 +04:00
switch ( psp ) {
case POWER_SUPPLY_PROP_STATUS :
val - > intval = POWER_SUPPLY_STATUS_UNKNOWN ;
2011-06-30 05:09:40 +04:00
if ( data - > pdata . chg ) {
if ( gpio_get_value ( data - > pdata . chg ) = = 0 )
2011-03-29 05:10:16 +04:00
val - > intval = POWER_SUPPLY_STATUS_CHARGING ;
else if ( data - > usb_in | | data - > ta_in )
val - > intval = POWER_SUPPLY_STATUS_NOT_CHARGING ;
else
val - > intval = POWER_SUPPLY_STATUS_DISCHARGING ;
}
break ;
case POWER_SUPPLY_PROP_ONLINE :
val - > intval = 0 ;
if ( data - > usb_in | | data - > ta_in )
val - > intval = 1 ;
break ;
case POWER_SUPPLY_PROP_HEALTH :
val - > intval = POWER_SUPPLY_HEALTH_GOOD ;
if ( data - > fault )
val - > intval = POWER_SUPPLY_HEALTH_UNSPEC_FAILURE ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
static irqreturn_t max8903_dcin ( int irq , void * _data )
{
struct max8903_data * data = _data ;
2011-06-30 05:09:40 +04:00
struct max8903_pdata * pdata = & data - > pdata ;
2011-03-29 05:10:16 +04:00
bool ta_in ;
enum power_supply_type old_type ;
ta_in = gpio_get_value ( pdata - > dok ) ? false : true ;
if ( ta_in = = data - > ta_in )
return IRQ_HANDLED ;
data - > ta_in = ta_in ;
/* Set Current-Limit-Mode 1:DC 0:USB */
if ( pdata - > dcm )
gpio_set_value ( pdata - > dcm , ta_in ? 1 : 0 ) ;
/* Charger Enable / Disable (cen is negated) */
if ( pdata - > cen )
gpio_set_value ( pdata - > cen , ta_in ? 0 :
( data - > usb_in ? 0 : 1 ) ) ;
dev_dbg ( data - > dev , " TA(DC-IN) Charger %s. \n " , ta_in ?
" Connected " : " Disconnected " ) ;
2015-03-12 10:44:11 +03:00
old_type = data - > psy_desc . type ;
2011-03-29 05:10:16 +04:00
if ( data - > ta_in )
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_MAINS ;
2011-03-29 05:10:16 +04:00
else if ( data - > usb_in )
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_USB ;
2011-03-29 05:10:16 +04:00
else
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_BATTERY ;
2011-03-29 05:10:16 +04:00
2015-03-12 10:44:11 +03:00
if ( old_type ! = data - > psy_desc . type )
power_supply_changed ( data - > psy ) ;
2011-03-29 05:10:16 +04:00
return IRQ_HANDLED ;
}
static irqreturn_t max8903_usbin ( int irq , void * _data )
{
struct max8903_data * data = _data ;
2011-06-30 05:09:40 +04:00
struct max8903_pdata * pdata = & data - > pdata ;
2011-03-29 05:10:16 +04:00
bool usb_in ;
enum power_supply_type old_type ;
usb_in = gpio_get_value ( pdata - > uok ) ? false : true ;
if ( usb_in = = data - > usb_in )
return IRQ_HANDLED ;
data - > usb_in = usb_in ;
/* Do not touch Current-Limit-Mode */
/* Charger Enable / Disable (cen is negated) */
if ( pdata - > cen )
gpio_set_value ( pdata - > cen , usb_in ? 0 :
( data - > ta_in ? 0 : 1 ) ) ;
dev_dbg ( data - > dev , " USB Charger %s. \n " , usb_in ?
" Connected " : " Disconnected " ) ;
2015-03-12 10:44:11 +03:00
old_type = data - > psy_desc . type ;
2011-03-29 05:10:16 +04:00
if ( data - > ta_in )
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_MAINS ;
2011-03-29 05:10:16 +04:00
else if ( data - > usb_in )
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_USB ;
2011-03-29 05:10:16 +04:00
else
2015-03-12 10:44:11 +03:00
data - > psy_desc . type = POWER_SUPPLY_TYPE_BATTERY ;
2011-03-29 05:10:16 +04:00
2015-03-12 10:44:11 +03:00
if ( old_type ! = data - > psy_desc . type )
power_supply_changed ( data - > psy ) ;
2011-03-29 05:10:16 +04:00
return IRQ_HANDLED ;
}
static irqreturn_t max8903_fault ( int irq , void * _data )
{
struct max8903_data * data = _data ;
2011-06-30 05:09:40 +04:00
struct max8903_pdata * pdata = & data - > pdata ;
2011-03-29 05:10:16 +04:00
bool fault ;
fault = gpio_get_value ( pdata - > flt ) ? false : true ;
if ( fault = = data - > fault )
return IRQ_HANDLED ;
data - > fault = fault ;
if ( fault )
dev_err ( data - > dev , " Charger suffers a fault and stops. \n " ) ;
else
dev_err ( data - > dev , " Charger recovered from a fault. \n " ) ;
return IRQ_HANDLED ;
}
2012-11-19 22:22:23 +04:00
static int max8903_probe ( struct platform_device * pdev )
2011-03-29 05:10:16 +04:00
{
struct max8903_data * data ;
struct device * dev = & pdev - > dev ;
struct max8903_pdata * pdata = pdev - > dev . platform_data ;
2015-03-12 10:44:11 +03:00
struct power_supply_config psy_cfg = { } ;
2011-03-29 05:10:16 +04:00
int ret = 0 ;
int gpio ;
int ta_in = 0 ;
int usb_in = 0 ;
2013-03-11 10:34:35 +04:00
data = devm_kzalloc ( dev , sizeof ( struct max8903_data ) , GFP_KERNEL ) ;
2011-03-29 05:10:16 +04:00
if ( data = = NULL ) {
dev_err ( dev , " Cannot allocate memory. \n " ) ;
return - ENOMEM ;
}
2011-06-30 05:09:40 +04:00
memcpy ( & data - > pdata , pdata , sizeof ( struct max8903_pdata ) ) ;
2011-03-29 05:10:16 +04:00
data - > dev = dev ;
platform_set_drvdata ( pdev , data ) ;
if ( pdata - > dc_valid = = false & & pdata - > usb_valid = = false ) {
dev_err ( dev , " No valid power sources. \n " ) ;
ret = - EINVAL ;
goto err ;
}
if ( pdata - > dc_valid ) {
if ( pdata - > dok & & gpio_is_valid ( pdata - > dok ) & &
pdata - > dcm & & gpio_is_valid ( pdata - > dcm ) ) {
gpio = pdata - > dok ; /* PULL_UPed Interrupt */
ta_in = gpio_get_value ( gpio ) ? 0 : 1 ;
gpio = pdata - > dcm ; /* Output */
gpio_set_value ( gpio , ta_in ) ;
} else {
dev_err ( dev , " When DC is wired, DOK and DCM should "
" be wired as well. \n " ) ;
ret = - EINVAL ;
goto err ;
}
} else {
if ( pdata - > dcm ) {
if ( gpio_is_valid ( pdata - > dcm ) )
gpio_set_value ( pdata - > dcm , 0 ) ;
else {
dev_err ( dev , " Invalid pin: dcm. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
}
if ( pdata - > usb_valid ) {
if ( pdata - > uok & & gpio_is_valid ( pdata - > uok ) ) {
gpio = pdata - > uok ;
usb_in = gpio_get_value ( gpio ) ? 0 : 1 ;
} else {
dev_err ( dev , " When USB is wired, UOK should be wired. "
" as well. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
if ( pdata - > cen ) {
if ( gpio_is_valid ( pdata - > cen ) ) {
gpio_set_value ( pdata - > cen , ( ta_in | | usb_in ) ? 0 : 1 ) ;
} else {
dev_err ( dev , " Invalid pin: cen. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
if ( pdata - > chg ) {
if ( ! gpio_is_valid ( pdata - > chg ) ) {
dev_err ( dev , " Invalid pin: chg. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
if ( pdata - > flt ) {
if ( ! gpio_is_valid ( pdata - > flt ) ) {
dev_err ( dev , " Invalid pin: flt. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
if ( pdata - > usus ) {
if ( ! gpio_is_valid ( pdata - > usus ) ) {
dev_err ( dev , " Invalid pin: usus. \n " ) ;
ret = - EINVAL ;
goto err ;
}
}
data - > fault = false ;
data - > ta_in = ta_in ;
data - > usb_in = usb_in ;
2015-03-12 10:44:11 +03:00
data - > psy_desc . name = " max8903_charger " ;
data - > psy_desc . type = ( ta_in ) ? POWER_SUPPLY_TYPE_MAINS :
2011-03-29 05:10:16 +04:00
( ( usb_in ) ? POWER_SUPPLY_TYPE_USB :
POWER_SUPPLY_TYPE_BATTERY ) ;
2015-03-12 10:44:11 +03:00
data - > psy_desc . get_property = max8903_get_property ;
data - > psy_desc . properties = max8903_charger_props ;
data - > psy_desc . num_properties = ARRAY_SIZE ( max8903_charger_props ) ;
2011-03-29 05:10:16 +04:00
2015-03-12 10:44:11 +03:00
psy_cfg . drv_data = data ;
data - > psy = power_supply_register ( dev , & data - > psy_desc , & psy_cfg ) ;
if ( IS_ERR ( data - > psy ) ) {
2011-03-29 05:10:16 +04:00
dev_err ( dev , " failed: power supply register. \n " ) ;
2015-03-12 10:44:11 +03:00
ret = PTR_ERR ( data - > psy ) ;
2011-03-29 05:10:16 +04:00
goto err ;
}
if ( pdata - > dc_valid ) {
ret = request_threaded_irq ( gpio_to_irq ( pdata - > dok ) ,
NULL , max8903_dcin ,
IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING ,
" MAX8903 DC IN " , data ) ;
if ( ret ) {
dev_err ( dev , " Cannot request irq %d for DC (%d) \n " ,
gpio_to_irq ( pdata - > dok ) , ret ) ;
goto err_psy ;
}
}
if ( pdata - > usb_valid ) {
ret = request_threaded_irq ( gpio_to_irq ( pdata - > uok ) ,
NULL , max8903_usbin ,
IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING ,
" MAX8903 USB IN " , data ) ;
if ( ret ) {
dev_err ( dev , " Cannot request irq %d for USB (%d) \n " ,
gpio_to_irq ( pdata - > uok ) , ret ) ;
goto err_dc_irq ;
}
}
if ( pdata - > flt ) {
ret = request_threaded_irq ( gpio_to_irq ( pdata - > flt ) ,
NULL , max8903_fault ,
IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING ,
" MAX8903 Fault " , data ) ;
if ( ret ) {
dev_err ( dev , " Cannot request irq %d for Fault (%d) \n " ,
gpio_to_irq ( pdata - > flt ) , ret ) ;
goto err_usb_irq ;
}
}
return 0 ;
err_usb_irq :
if ( pdata - > usb_valid )
free_irq ( gpio_to_irq ( pdata - > uok ) , data ) ;
err_dc_irq :
if ( pdata - > dc_valid )
free_irq ( gpio_to_irq ( pdata - > dok ) , data ) ;
err_psy :
2015-03-12 10:44:11 +03:00
power_supply_unregister ( data - > psy ) ;
2011-03-29 05:10:16 +04:00
err :
return ret ;
}
2012-11-19 22:26:07 +04:00
static int max8903_remove ( struct platform_device * pdev )
2011-03-29 05:10:16 +04:00
{
struct max8903_data * data = platform_get_drvdata ( pdev ) ;
if ( data ) {
2011-06-30 05:09:40 +04:00
struct max8903_pdata * pdata = & data - > pdata ;
2011-03-29 05:10:16 +04:00
if ( pdata - > flt )
free_irq ( gpio_to_irq ( pdata - > flt ) , data ) ;
if ( pdata - > usb_valid )
free_irq ( gpio_to_irq ( pdata - > uok ) , data ) ;
if ( pdata - > dc_valid )
free_irq ( gpio_to_irq ( pdata - > dok ) , data ) ;
2015-03-12 10:44:11 +03:00
power_supply_unregister ( data - > psy ) ;
2011-03-29 05:10:16 +04:00
}
return 0 ;
}
static struct platform_driver max8903_driver = {
. probe = max8903_probe ,
2012-11-19 22:20:40 +04:00
. remove = max8903_remove ,
2011-03-29 05:10:16 +04:00
. driver = {
. name = " max8903-charger " ,
} ,
} ;
2011-11-26 08:01:10 +04:00
module_platform_driver ( max8903_driver ) ;
2011-03-29 05:10:16 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " MAX8903 Charger Driver " ) ;
MODULE_AUTHOR ( " MyungJoo Ham <myungjoo.ham@samsung.com> " ) ;
2011-06-24 18:26:36 +04:00
MODULE_ALIAS ( " platform:max8903-charger " ) ;