2019-05-19 16:51:43 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2015-02-02 14:26:25 +03:00
/*
* cros_ec_dev - expose the Chrome OS Embedded Controller to user - space
*
* Copyright ( C ) 2014 Google , Inc .
*/
2016-08-01 12:54:38 +03:00
# include <linux/mfd/core.h>
2019-09-02 12:53:02 +03:00
# include <linux/mfd/cros_ec.h>
2015-02-02 14:26:25 +03:00
# include <linux/module.h>
2018-06-20 08:47:28 +03:00
# include <linux/mod_devicetable.h>
2018-12-12 20:34:01 +03:00
# include <linux/of_platform.h>
2015-02-02 14:26:25 +03:00
# include <linux/platform_device.h>
2019-09-02 12:53:03 +03:00
# include <linux/platform_data/cros_ec_chardev.h>
2019-09-02 12:53:05 +03:00
# include <linux/platform_data/cros_ec_commands.h>
# include <linux/platform_data/cros_ec_proto.h>
2015-06-09 14:04:42 +03:00
# include <linux/slab.h>
2015-02-02 14:26:25 +03:00
2017-11-20 19:15:25 +03:00
# define DRV_NAME "cros-ec-dev"
2015-06-09 14:04:47 +03:00
static struct class cros_class = {
. owner = THIS_MODULE ,
. name = " chromeos " ,
} ;
2019-09-02 12:53:07 +03:00
/**
* cros_feature_to_name - CrOS feature id to name / short description .
* @ id : The feature identifier .
* @ name : Device name associated with the feature id .
* @ desc : Short name that will be displayed .
*/
struct cros_feature_to_name {
unsigned int id ;
const char * name ;
const char * desc ;
} ;
2019-09-02 12:53:08 +03:00
/**
* cros_feature_to_cells - CrOS feature id to mfd cells association .
* @ id : The feature identifier .
* @ mfd_cells : Pointer to the array of mfd cells that needs to be added .
* @ num_cells : Number of mfd cells into the array .
*/
struct cros_feature_to_cells {
unsigned int id ;
const struct mfd_cell * mfd_cells ;
unsigned int num_cells ;
} ;
2019-09-02 12:53:07 +03:00
static const struct cros_feature_to_name cros_mcu_devices [ ] = {
{
. id = EC_FEATURE_FINGERPRINT ,
. name = CROS_EC_DEV_FP_NAME ,
. desc = " Fingerprint " ,
} ,
{
. id = EC_FEATURE_ISH ,
. name = CROS_EC_DEV_ISH_NAME ,
. desc = " Integrated Sensor Hub " ,
} ,
{
. id = EC_FEATURE_SCP ,
. name = CROS_EC_DEV_SCP_NAME ,
. desc = " System Control Processor " ,
} ,
{
. id = EC_FEATURE_TOUCHPAD ,
. name = CROS_EC_DEV_TP_NAME ,
. desc = " Touchpad " ,
} ,
} ;
2019-09-02 12:53:08 +03:00
static const struct mfd_cell cros_ec_cec_cells [ ] = {
{ . name = " cros-ec-cec " , } ,
} ;
static const struct mfd_cell cros_ec_rtc_cells [ ] = {
{ . name = " cros-ec-rtc " , } ,
} ;
static const struct mfd_cell cros_usbpd_charger_cells [ ] = {
{ . name = " cros-usbpd-charger " , } ,
{ . name = " cros-usbpd-logger " , } ,
} ;
static const struct cros_feature_to_cells cros_subdevices [ ] = {
{
. id = EC_FEATURE_CEC ,
. mfd_cells = cros_ec_cec_cells ,
. num_cells = ARRAY_SIZE ( cros_ec_cec_cells ) ,
} ,
{
. id = EC_FEATURE_RTC ,
. mfd_cells = cros_ec_rtc_cells ,
. num_cells = ARRAY_SIZE ( cros_ec_rtc_cells ) ,
} ,
{
. id = EC_FEATURE_USB_PD ,
. mfd_cells = cros_usbpd_charger_cells ,
. num_cells = ARRAY_SIZE ( cros_usbpd_charger_cells ) ,
} ,
} ;
static const struct mfd_cell cros_ec_platform_cells [ ] = {
{ . name = " cros-ec-chardev " , } ,
{ . name = " cros-ec-debugfs " , } ,
{ . name = " cros-ec-lightbar " , } ,
{ . name = " cros-ec-sysfs " , } ,
} ;
static const struct mfd_cell cros_ec_vbc_cells [ ] = {
{ . name = " cros-ec-vbc " , }
} ;
2016-08-01 12:54:37 +03:00
static int cros_ec_check_features ( struct cros_ec_dev * ec , int feature )
{
struct cros_ec_command * msg ;
int ret ;
if ( ec - > features [ 0 ] = = - 1U & & ec - > features [ 1 ] = = - 1U ) {
/* features bitmap not read yet */
2019-09-02 12:53:06 +03:00
msg = kzalloc ( sizeof ( * msg ) + sizeof ( ec - > features ) , GFP_KERNEL ) ;
2016-08-01 12:54:37 +03:00
if ( ! msg )
return - ENOMEM ;
msg - > command = EC_CMD_GET_FEATURES + ec - > cmd_offset ;
msg - > insize = sizeof ( ec - > features ) ;
2019-09-02 12:53:06 +03:00
ret = cros_ec_cmd_xfer_status ( ec - > ec_dev , msg ) ;
if ( ret < 0 ) {
2016-08-01 12:54:37 +03:00
dev_warn ( ec - > dev , " cannot get EC features: %d/%d \n " ,
ret , msg - > result ) ;
memset ( ec - > features , 0 , sizeof ( ec - > features ) ) ;
2018-05-31 09:23:43 +03:00
} else {
memcpy ( ec - > features , msg - > data , sizeof ( ec - > features ) ) ;
2016-08-01 12:54:37 +03:00
}
dev_dbg ( ec - > dev , " EC features %08x %08x \n " ,
ec - > features [ 0 ] , ec - > features [ 1 ] ) ;
kfree ( msg ) ;
}
return ec - > features [ feature / 32 ] & EC_FEATURE_MASK_0 ( feature ) ;
}
2018-12-04 18:58:43 +03:00
static void cros_ec_class_release ( struct device * dev )
{
kfree ( to_cros_ec_dev ( dev ) ) ;
}
2016-08-01 12:54:38 +03:00
static void cros_ec_sensors_register ( struct cros_ec_dev * ec )
{
/*
* Issue a command to get the number of sensor reported .
* Build an array of sensors driver and register them all .
*/
int ret , i , id , sensor_num ;
struct mfd_cell * sensor_cells ;
struct cros_ec_sensor_platform * sensor_platforms ;
int sensor_type [ MOTIONSENSE_TYPE_MAX ] ;
struct ec_params_motion_sense * params ;
struct ec_response_motion_sense * resp ;
struct cros_ec_command * msg ;
msg = kzalloc ( sizeof ( struct cros_ec_command ) +
max ( sizeof ( * params ) , sizeof ( * resp ) ) , GFP_KERNEL ) ;
if ( msg = = NULL )
return ;
msg - > version = 2 ;
msg - > command = EC_CMD_MOTION_SENSE_CMD + ec - > cmd_offset ;
msg - > outsize = sizeof ( * params ) ;
msg - > insize = sizeof ( * resp ) ;
params = ( struct ec_params_motion_sense * ) msg - > data ;
params - > cmd = MOTIONSENSE_CMD_DUMP ;
2019-09-02 12:53:06 +03:00
ret = cros_ec_cmd_xfer_status ( ec - > ec_dev , msg ) ;
if ( ret < 0 ) {
2016-08-01 12:54:38 +03:00
dev_warn ( ec - > dev , " cannot get EC sensor information: %d/%d \n " ,
ret , msg - > result ) ;
goto error ;
}
resp = ( struct ec_response_motion_sense * ) msg - > data ;
sensor_num = resp - > dump . sensor_count ;
2019-05-18 02:38:55 +03:00
/*
* Allocate 2 extra sensors if lid angle sensor and / or FIFO are needed .
*/
sensor_cells = kcalloc ( sensor_num + 2 , sizeof ( struct mfd_cell ) ,
2016-08-01 12:54:38 +03:00
GFP_KERNEL ) ;
if ( sensor_cells = = NULL )
goto error ;
2019-05-18 02:38:55 +03:00
sensor_platforms = kcalloc ( sensor_num ,
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-13 00:03:40 +03:00
sizeof ( struct cros_ec_sensor_platform ) ,
GFP_KERNEL ) ;
2016-08-01 12:54:38 +03:00
if ( sensor_platforms = = NULL )
goto error_platforms ;
memset ( sensor_type , 0 , sizeof ( sensor_type ) ) ;
id = 0 ;
for ( i = 0 ; i < sensor_num ; i + + ) {
params - > cmd = MOTIONSENSE_CMD_INFO ;
params - > info . sensor_num = i ;
2019-09-02 12:53:06 +03:00
ret = cros_ec_cmd_xfer_status ( ec - > ec_dev , msg ) ;
if ( ret < 0 ) {
2016-08-01 12:54:38 +03:00
dev_warn ( ec - > dev , " no info for EC sensor %d : %d/%d \n " ,
i , ret , msg - > result ) ;
continue ;
}
switch ( resp - > info . type ) {
case MOTIONSENSE_TYPE_ACCEL :
sensor_cells [ id ] . name = " cros-ec-accel " ;
break ;
2017-01-24 16:41:41 +03:00
case MOTIONSENSE_TYPE_BARO :
sensor_cells [ id ] . name = " cros-ec-baro " ;
break ;
2016-08-01 12:54:38 +03:00
case MOTIONSENSE_TYPE_GYRO :
sensor_cells [ id ] . name = " cros-ec-gyro " ;
break ;
case MOTIONSENSE_TYPE_MAG :
sensor_cells [ id ] . name = " cros-ec-mag " ;
break ;
case MOTIONSENSE_TYPE_PROX :
sensor_cells [ id ] . name = " cros-ec-prox " ;
break ;
case MOTIONSENSE_TYPE_LIGHT :
sensor_cells [ id ] . name = " cros-ec-light " ;
break ;
case MOTIONSENSE_TYPE_ACTIVITY :
sensor_cells [ id ] . name = " cros-ec-activity " ;
break ;
default :
dev_warn ( ec - > dev , " unknown type %d \n " , resp - > info . type ) ;
continue ;
}
sensor_platforms [ id ] . sensor_num = i ;
sensor_cells [ id ] . id = sensor_type [ resp - > info . type ] ;
sensor_cells [ id ] . platform_data = & sensor_platforms [ id ] ;
sensor_cells [ id ] . pdata_size =
sizeof ( struct cros_ec_sensor_platform ) ;
sensor_type [ resp - > info . type ] + + ;
id + + ;
}
2018-03-23 20:42:47 +03:00
if ( sensor_type [ MOTIONSENSE_TYPE_ACCEL ] > = 2 )
ec - > has_kb_wake_angle = true ;
2016-08-01 12:54:38 +03:00
if ( cros_ec_check_features ( ec , EC_FEATURE_MOTION_SENSE_FIFO ) ) {
sensor_cells [ id ] . name = " cros-ec-ring " ;
id + + ;
}
2019-05-18 02:38:55 +03:00
if ( cros_ec_check_features ( ec ,
EC_FEATURE_REFINED_TABLET_MODE_HYSTERESIS ) ) {
sensor_cells [ id ] . name = " cros-ec-lid-angle " ;
id + + ;
}
2016-08-01 12:54:38 +03:00
ret = mfd_add_devices ( ec - > dev , 0 , sensor_cells , id ,
NULL , 0 , NULL ) ;
if ( ret )
dev_err ( ec - > dev , " failed to add EC sensors \n " ) ;
kfree ( sensor_platforms ) ;
error_platforms :
kfree ( sensor_cells ) ;
error :
kfree ( msg ) ;
}
2019-06-11 10:12:36 +03:00
static struct cros_ec_sensor_platform sensor_platforms [ ] = {
{ . sensor_num = 0 } ,
{ . sensor_num = 1 }
} ;
static const struct mfd_cell cros_ec_accel_legacy_cells [ ] = {
{
. name = " cros-ec-accel-legacy " ,
. platform_data = & sensor_platforms [ 0 ] ,
. pdata_size = sizeof ( struct cros_ec_sensor_platform ) ,
} ,
{
. name = " cros-ec-accel-legacy " ,
. platform_data = & sensor_platforms [ 1 ] ,
. pdata_size = sizeof ( struct cros_ec_sensor_platform ) ,
}
} ;
static void cros_ec_accel_legacy_register ( struct cros_ec_dev * ec )
{
struct cros_ec_device * ec_dev = ec - > ec_dev ;
u8 status ;
int ret ;
/*
* ECs that need legacy support are the main EC , directly connected to
* the AP .
*/
if ( ec - > cmd_offset ! = 0 )
return ;
/*
* Check if EC supports direct memory reads and if EC has
* accelerometers .
*/
if ( ec_dev - > cmd_readmem ) {
ret = ec_dev - > cmd_readmem ( ec_dev , EC_MEMMAP_ACC_STATUS , 1 ,
& status ) ;
if ( ret < 0 ) {
dev_warn ( ec - > dev , " EC direct read error. \n " ) ;
return ;
}
/* Check if EC has accelerometers. */
if ( ! ( status & EC_MEMMAP_ACC_STATUS_PRESENCE_BIT ) ) {
dev_info ( ec - > dev , " EC does not have accelerometers. \n " ) ;
return ;
}
}
/*
* The device may still support accelerometers :
* it would be an older ARM based device that do not suppor the
* EC_CMD_GET_FEATURES command .
*
* Register 2 accelerometers , we will fail in the IIO driver if there
* are no sensors .
*/
2019-09-02 12:53:09 +03:00
ret = mfd_add_hotplug_devices ( ec - > dev , cros_ec_accel_legacy_cells ,
ARRAY_SIZE ( cros_ec_accel_legacy_cells ) ) ;
2019-06-11 10:12:36 +03:00
if ( ret )
dev_err ( ec_dev - > dev , " failed to add EC sensors \n " ) ;
}
2015-02-02 14:26:25 +03:00
static int ec_device_probe ( struct platform_device * pdev )
{
2015-06-09 14:04:47 +03:00
int retval = - ENOMEM ;
2018-12-12 20:34:01 +03:00
struct device_node * node ;
2015-06-09 14:04:47 +03:00
struct device * dev = & pdev - > dev ;
struct cros_ec_platform * ec_platform = dev_get_platdata ( dev ) ;
2018-12-04 18:58:43 +03:00
struct cros_ec_dev * ec = kzalloc ( sizeof ( * ec ) , GFP_KERNEL ) ;
2019-09-02 12:53:07 +03:00
int i ;
2015-06-09 14:04:47 +03:00
if ( ! ec )
return retval ;
2015-02-02 14:26:25 +03:00
2015-06-09 14:04:47 +03:00
dev_set_drvdata ( dev , ec ) ;
ec - > ec_dev = dev_get_drvdata ( dev - > parent ) ;
ec - > dev = dev ;
ec - > cmd_offset = ec_platform - > cmd_offset ;
2016-08-01 12:54:37 +03:00
ec - > features [ 0 ] = - 1U ; /* Not cached yet */
ec - > features [ 1 ] = - 1U ; /* Not cached yet */
2015-06-09 14:04:47 +03:00
device_initialize ( & ec - > class_dev ) ;
2015-02-02 14:26:25 +03:00
2019-09-02 12:53:07 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( cros_mcu_devices ) ; i + + ) {
2019-05-08 12:19:55 +03:00
/*
2019-09-02 12:53:07 +03:00
* Check whether this is actually a dedicated MCU rather
* than an standard EC .
2019-05-08 12:19:55 +03:00
*/
2019-09-02 12:53:07 +03:00
if ( cros_ec_check_features ( ec , cros_mcu_devices [ i ] . id ) ) {
dev_info ( dev , " CrOS %s MCU detected \n " ,
cros_mcu_devices [ i ] . desc ) ;
/*
* Help userspace differentiating ECs from other MCU ,
* regardless of the probing order .
*/
ec_platform - > ec_name = cros_mcu_devices [ i ] . name ;
break ;
}
2019-06-03 06:45:11 +03:00
}
2015-06-09 14:04:47 +03:00
/*
* Add the class device
*/
ec - > class_dev . class = & cros_class ;
ec - > class_dev . parent = dev ;
2018-12-04 18:58:43 +03:00
ec - > class_dev . release = cros_ec_class_release ;
2015-06-09 14:04:47 +03:00
retval = dev_set_name ( & ec - > class_dev , " %s " , ec_platform - > ec_name ) ;
if ( retval ) {
dev_err ( dev , " dev_set_name failed => %d \n " , retval ) ;
2017-03-17 21:48:14 +03:00
goto failed ;
2015-02-02 14:26:25 +03:00
}
2019-09-02 12:53:03 +03:00
retval = device_add ( & ec - > class_dev ) ;
if ( retval )
goto failed ;
2018-03-23 20:42:47 +03:00
/* check whether this EC is a sensor hub. */
if ( cros_ec_check_features ( ec , EC_FEATURE_MOTION_SENSE ) )
cros_ec_sensors_register ( ec ) ;
2019-06-11 10:12:36 +03:00
else
/* Workaroud for older EC firmware */
cros_ec_accel_legacy_register ( ec ) ;
2018-03-23 20:42:47 +03:00
2019-09-02 12:53:08 +03:00
/*
* The following subdevices can be detected by sending the
* EC_FEATURE_GET_CMD Embedded Controller device .
*/
for ( i = 0 ; i < ARRAY_SIZE ( cros_subdevices ) ; i + + ) {
if ( cros_ec_check_features ( ec , cros_subdevices [ i ] . id ) ) {
retval = mfd_add_hotplug_devices ( ec - > dev ,
cros_subdevices [ i ] . mfd_cells ,
cros_subdevices [ i ] . num_cells ) ;
if ( retval )
dev_err ( ec - > dev ,
" failed to add %s subdevice: %d \n " ,
cros_subdevices [ i ] . mfd_cells - > name ,
retval ) ;
}
2018-05-02 18:44:18 +03:00
}
2019-09-02 12:53:08 +03:00
/*
* The following subdevices cannot be detected by sending the
* EC_FEATURE_GET_CMD to the Embedded Controller device .
*/
2019-09-02 12:53:09 +03:00
retval = mfd_add_hotplug_devices ( ec - > dev , cros_ec_platform_cells ,
ARRAY_SIZE ( cros_ec_platform_cells ) ) ;
2018-12-12 20:33:57 +03:00
if ( retval )
dev_warn ( ec - > dev ,
" failed to add cros-ec platform devices: %d \n " ,
retval ) ;
2018-12-12 20:34:01 +03:00
/* Check whether this EC instance has a VBC NVRAM */
node = ec - > ec_dev - > dev - > of_node ;
if ( of_property_read_bool ( node , " google,has-vbc-nvram " ) ) {
2019-09-02 12:53:09 +03:00
retval = mfd_add_hotplug_devices ( ec - > dev , cros_ec_vbc_cells ,
ARRAY_SIZE ( cros_ec_vbc_cells ) ) ;
2018-12-12 20:34:01 +03:00
if ( retval )
dev_warn ( ec - > dev , " failed to add VBC devices: %d \n " ,
retval ) ;
}
2015-02-02 14:26:25 +03:00
return 0 ;
2015-06-09 14:04:47 +03:00
2017-03-17 21:48:14 +03:00
failed :
put_device ( & ec - > class_dev ) ;
2015-06-09 14:04:47 +03:00
return retval ;
2015-02-02 14:26:25 +03:00
}
static int ec_device_remove ( struct platform_device * pdev )
{
2015-06-09 14:04:47 +03:00
struct cros_ec_dev * ec = dev_get_drvdata ( & pdev - > dev ) ;
2017-05-16 18:46:48 +03:00
2018-12-10 21:00:02 +03:00
mfd_remove_devices ( ec - > dev ) ;
2015-06-09 14:04:47 +03:00
device_unregister ( & ec - > class_dev ) ;
2015-02-02 14:26:25 +03:00
return 0 ;
}
2015-06-22 09:27:20 +03:00
static const struct platform_device_id cros_ec_id [ ] = {
2017-11-20 19:15:25 +03:00
{ DRV_NAME , 0 } ,
2018-04-18 13:24:03 +03:00
{ /* sentinel */ }
2015-06-22 09:27:20 +03:00
} ;
MODULE_DEVICE_TABLE ( platform , cros_ec_id ) ;
2015-02-02 14:26:25 +03:00
static struct platform_driver cros_ec_dev_driver = {
. driver = {
2017-11-20 19:15:25 +03:00
. name = DRV_NAME ,
2015-02-02 14:26:25 +03:00
} ,
2018-09-27 06:33:17 +03:00
. id_table = cros_ec_id ,
2015-02-02 14:26:25 +03:00
. probe = ec_device_probe ,
. remove = ec_device_remove ,
} ;
static int __init cros_ec_dev_init ( void )
{
int ret ;
2015-06-09 14:04:47 +03:00
ret = class_register ( & cros_class ) ;
if ( ret ) {
2015-02-02 14:26:25 +03:00
pr_err ( CROS_EC_DEV_NAME " : failed to register device class \n " ) ;
2015-06-09 14:04:47 +03:00
return ret ;
2015-02-02 14:26:25 +03:00
}
/* Register the driver */
ret = platform_driver_register ( & cros_ec_dev_driver ) ;
if ( ret < 0 ) {
pr_warn ( CROS_EC_DEV_NAME " : can't register driver: %d \n " , ret ) ;
goto failed_devreg ;
}
return 0 ;
failed_devreg :
2015-06-09 14:04:47 +03:00
class_unregister ( & cros_class ) ;
2015-02-02 14:26:25 +03:00
return ret ;
}
static void __exit cros_ec_dev_exit ( void )
{
platform_driver_unregister ( & cros_ec_dev_driver ) ;
2015-06-09 14:04:47 +03:00
class_unregister ( & cros_class ) ;
2015-02-02 14:26:25 +03:00
}
module_init ( cros_ec_dev_init ) ;
module_exit ( cros_ec_dev_exit ) ;
2017-11-20 19:15:25 +03:00
MODULE_ALIAS ( " platform: " DRV_NAME ) ;
2015-02-02 14:26:25 +03:00
MODULE_AUTHOR ( " Bill Richardson <wfrichar@chromium.org> " ) ;
MODULE_DESCRIPTION ( " Userspace interface to the Chrome OS Embedded Controller " ) ;
MODULE_VERSION ( " 1.0 " ) ;
MODULE_LICENSE ( " GPL " ) ;