2013-02-26 02:08:37 +04: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 .
*/
# include <linux/interrupt.h>
# include <linux/slab.h>
2013-03-20 12:46:15 +04:00
# include <linux/module.h>
2013-02-26 02:08:37 +04:00
# include <linux/mfd/core.h>
# include <linux/mfd/cros_ec.h>
# include <linux/mfd/cros_ec_commands.h>
int cros_ec_prepare_tx ( struct cros_ec_device * ec_dev ,
2014-06-18 22:14:02 +04:00
struct cros_ec_command * msg )
2013-02-26 02:08:37 +04:00
{
uint8_t * out ;
int csum , i ;
2014-06-18 22:14:02 +04:00
BUG_ON ( msg - > outsize > EC_PROTO2_MAX_PARAM_SIZE ) ;
2013-02-26 02:08:37 +04:00
out = ec_dev - > dout ;
out [ 0 ] = EC_CMD_VERSION0 + msg - > version ;
2014-06-18 22:14:02 +04:00
out [ 1 ] = msg - > command ;
out [ 2 ] = msg - > outsize ;
2013-02-26 02:08:37 +04:00
csum = out [ 0 ] + out [ 1 ] + out [ 2 ] ;
2014-06-18 22:14:02 +04:00
for ( i = 0 ; i < msg - > outsize ; i + + )
csum + = out [ EC_MSG_TX_HEADER_BYTES + i ] = msg - > outdata [ i ] ;
out [ EC_MSG_TX_HEADER_BYTES + msg - > outsize ] = ( uint8_t ) ( csum & 0xff ) ;
2013-02-26 02:08:37 +04:00
2014-06-18 22:14:02 +04:00
return EC_MSG_TX_PROTO_BYTES + msg - > outsize ;
2013-02-26 02:08:37 +04:00
}
2013-03-20 12:46:15 +04:00
EXPORT_SYMBOL ( cros_ec_prepare_tx ) ;
2013-02-26 02:08:37 +04:00
static irqreturn_t ec_irq_thread ( int irq , void * data )
{
struct cros_ec_device * ec_dev = data ;
if ( device_may_wakeup ( ec_dev - > dev ) )
pm_wakeup_event ( ec_dev - > dev , 0 ) ;
blocking_notifier_call_chain ( & ec_dev - > event_notifier , 1 , ec_dev ) ;
return IRQ_HANDLED ;
}
2013-11-18 17:33:06 +04:00
static const struct mfd_cell cros_devs [ ] = {
2013-02-26 02:08:37 +04:00
{
. name = " cros-ec-keyb " ,
. id = 1 ,
. of_compatible = " google,cros-ec-keyb " ,
} ,
2014-04-30 21:44:09 +04:00
{
. name = " cros-ec-i2c-tunnel " ,
. id = 2 ,
. of_compatible = " google,cros-ec-i2c-tunnel " ,
} ,
2013-02-26 02:08:37 +04:00
} ;
int cros_ec_register ( struct cros_ec_device * ec_dev )
{
struct device * dev = ec_dev - > dev ;
int err = 0 ;
BLOCKING_INIT_NOTIFIER_HEAD ( & ec_dev - > event_notifier ) ;
if ( ec_dev - > din_size ) {
2013-05-23 19:25:10 +04:00
ec_dev - > din = devm_kzalloc ( dev , ec_dev - > din_size , GFP_KERNEL ) ;
if ( ! ec_dev - > din )
return - ENOMEM ;
2013-02-26 02:08:37 +04:00
}
if ( ec_dev - > dout_size ) {
2013-05-23 19:25:10 +04:00
ec_dev - > dout = devm_kzalloc ( dev , ec_dev - > dout_size , GFP_KERNEL ) ;
if ( ! ec_dev - > dout )
return - ENOMEM ;
2013-02-26 02:08:37 +04:00
}
if ( ! ec_dev - > irq ) {
dev_dbg ( dev , " no valid IRQ: %d \n " , ec_dev - > irq ) ;
2013-05-23 19:25:10 +04:00
return err ;
2013-02-26 02:08:37 +04:00
}
err = request_threaded_irq ( ec_dev - > irq , NULL , ec_irq_thread ,
IRQF_TRIGGER_LOW | IRQF_ONESHOT ,
" chromeos-ec " , ec_dev ) ;
if ( err ) {
dev_err ( dev , " request irq %d: error %d \n " , ec_dev - > irq , err ) ;
2013-05-23 19:25:10 +04:00
return err ;
2013-02-26 02:08:37 +04: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 " ) ;
goto fail_mfd ;
}
2014-06-18 22:14:03 +04:00
dev_info ( dev , " Chrome EC device registered \n " ) ;
2013-02-26 02:08:37 +04:00
return 0 ;
fail_mfd :
free_irq ( ec_dev - > irq , ec_dev ) ;
2013-05-23 19:25:10 +04:00
2013-02-26 02:08:37 +04:00
return err ;
}
2013-03-20 12:46:15 +04:00
EXPORT_SYMBOL ( cros_ec_register ) ;
2013-02-26 02:08:37 +04:00
int cros_ec_remove ( struct cros_ec_device * ec_dev )
{
mfd_remove_devices ( ec_dev - > dev ) ;
free_irq ( ec_dev - > irq , ec_dev ) ;
return 0 ;
}
2013-03-20 12:46:15 +04:00
EXPORT_SYMBOL ( cros_ec_remove ) ;
2013-02-26 02:08:37 +04: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 12:46:15 +04:00
EXPORT_SYMBOL ( cros_ec_suspend ) ;
2013-02-26 02:08:37 +04: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 12:46:15 +04:00
EXPORT_SYMBOL ( cros_ec_resume ) ;
2013-02-26 02:08:37 +04:00
# endif
2014-04-17 23:32:17 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " ChromeOS EC core driver " ) ;