2013-02-25 14:08:37 -08:00
/*
* ChromeOS EC multi - function device
*
* Copyright ( C ) 2012 Google , Inc
*
* This software is licensed under the terms of the GNU General Public
* License version 2 , as published by the Free Software Foundation , and
* may be copied , distributed , and modified under those terms .
*
* 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 .
*
* The ChromeOS EC multi function device is used to mux all the requests
* to the EC device for its multiple features : keyboard controller ,
* battery charging and regulator control , firmware update .
*/
2015-05-20 11:31:28 +02:00
# include <linux/of_platform.h>
2013-02-25 14:08:37 -08:00
# include <linux/interrupt.h>
# include <linux/slab.h>
2013-03-20 09:46:15 +01:00
# include <linux/module.h>
2013-02-25 14:08:37 -08:00
# include <linux/mfd/core.h>
# include <linux/mfd/cros_ec.h>
2014-09-18 17:18:56 +02:00
2013-11-18 14:33:06 +01:00
static const struct mfd_cell cros_devs [ ] = {
2015-02-02 12:26:26 +01:00
{
. name = " cros-ec-ctl " ,
2015-05-20 11:31:28 +02:00
. id = PLATFORM_DEVID_AUTO ,
2015-02-02 12:26:26 +01:00
} ,
2013-02-25 14:08:37 -08:00
} ;
int cros_ec_register ( struct cros_ec_device * ec_dev )
{
struct device * dev = ec_dev - > dev ;
int err = 0 ;
2015-06-09 13:04:45 +02:00
ec_dev - > max_request = sizeof ( struct ec_params_hello ) ;
ec_dev - > max_response = sizeof ( struct ec_response_get_protocol_info ) ;
ec_dev - > max_passthru = 0 ;
ec_dev - > din = devm_kzalloc ( dev , ec_dev - > din_size , GFP_KERNEL ) ;
if ( ! ec_dev - > din )
return - ENOMEM ;
ec_dev - > dout = devm_kzalloc ( dev , ec_dev - > dout_size , GFP_KERNEL ) ;
if ( ! ec_dev - > dout )
return - ENOMEM ;
2013-02-25 14:08:37 -08:00
2014-09-18 17:18:57 +02:00
mutex_init ( & ec_dev - > lock ) ;
2015-06-09 13:04:45 +02:00
cros_ec_query_all ( ec_dev ) ;
2013-02-25 14:08:37 -08:00
err = mfd_add_devices ( dev , 0 , cros_devs ,
ARRAY_SIZE ( cros_devs ) ,
NULL , ec_dev - > irq , NULL ) ;
if ( err ) {
dev_err ( dev , " failed to add mfd devices \n " ) ;
2014-06-18 11:14:07 -07:00
return err ;
2013-02-25 14:08:37 -08:00
}
2015-05-20 11:31:28 +02:00
if ( IS_ENABLED ( CONFIG_OF ) & & dev - > of_node ) {
err = of_platform_populate ( dev - > of_node , NULL , NULL , dev ) ;
if ( err ) {
mfd_remove_devices ( dev ) ;
dev_err ( dev , " Failed to register sub-devices \n " ) ;
return err ;
}
}
2014-06-18 11:14:03 -07:00
dev_info ( dev , " Chrome EC device registered \n " ) ;
2013-02-25 14:08:37 -08:00
return 0 ;
}
2013-03-20 09:46:15 +01:00
EXPORT_SYMBOL ( cros_ec_register ) ;
2013-02-25 14:08:37 -08:00
int cros_ec_remove ( struct cros_ec_device * ec_dev )
{
mfd_remove_devices ( ec_dev - > dev ) ;
return 0 ;
}
2013-03-20 09:46:15 +01:00
EXPORT_SYMBOL ( cros_ec_remove ) ;
2013-02-25 14:08:37 -08:00
# ifdef CONFIG_PM_SLEEP
int cros_ec_suspend ( struct cros_ec_device * ec_dev )
{
struct device * dev = ec_dev - > dev ;
if ( device_may_wakeup ( dev ) )
ec_dev - > wake_enabled = ! enable_irq_wake ( ec_dev - > irq ) ;
disable_irq ( ec_dev - > irq ) ;
ec_dev - > was_wake_device = ec_dev - > wake_enabled ;
return 0 ;
}
2013-03-20 09:46:15 +01:00
EXPORT_SYMBOL ( cros_ec_suspend ) ;
2013-02-25 14:08:37 -08:00
int cros_ec_resume ( struct cros_ec_device * ec_dev )
{
enable_irq ( ec_dev - > irq ) ;
if ( ec_dev - > wake_enabled ) {
disable_irq_wake ( ec_dev - > irq ) ;
ec_dev - > wake_enabled = 0 ;
}
return 0 ;
}
2013-03-20 09:46:15 +01:00
EXPORT_SYMBOL ( cros_ec_resume ) ;
2013-02-25 14:08:37 -08:00
# endif
2014-04-17 12:32:17 -07:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " ChromeOS EC core driver " ) ;