2012-12-25 19:06:07 +02:00
/*
*
* Intel Management Engine Interface ( Intel MEI ) Linux driver
* Copyright ( c ) 2003 - 2012 , Intel Corporation .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope 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 .
*
*/
2014-03-18 22:51:55 +02:00
# include <linux/export.h>
2012-12-25 19:06:07 +02:00
# include <linux/sched.h>
# include <linux/wait.h>
2014-03-18 22:52:02 +02:00
# include <linux/pm_runtime.h>
2014-09-29 16:31:46 +03:00
# include <linux/slab.h>
# include <linux/mei.h>
2012-12-25 19:06:07 +02:00
# include "mei_dev.h"
2013-01-08 23:07:12 +02:00
# include "hbm.h"
2014-02-17 15:13:23 +02:00
# include "client.h"
2012-12-25 19:06:07 +02:00
2014-08-21 14:29:16 +03:00
static const char * mei_hbm_status_str ( enum mei_hbm_status status )
{
# define MEI_HBM_STATUS(status) case MEI_HBMS_##status: return #status
switch ( status ) {
MEI_HBM_STATUS ( SUCCESS ) ;
MEI_HBM_STATUS ( CLIENT_NOT_FOUND ) ;
MEI_HBM_STATUS ( ALREADY_EXISTS ) ;
MEI_HBM_STATUS ( REJECTED ) ;
MEI_HBM_STATUS ( INVALID_PARAMETER ) ;
MEI_HBM_STATUS ( NOT_ALLOWED ) ;
MEI_HBM_STATUS ( ALREADY_STARTED ) ;
MEI_HBM_STATUS ( NOT_STARTED ) ;
default : return " unknown " ;
}
# undef MEI_HBM_STATUS
} ;
2014-02-17 15:13:20 +02:00
static const char * mei_cl_conn_status_str ( enum mei_cl_connect_status status )
{
# define MEI_CL_CS(status) case MEI_CL_CONN_##status: return #status
switch ( status ) {
MEI_CL_CS ( SUCCESS ) ;
MEI_CL_CS ( NOT_FOUND ) ;
MEI_CL_CS ( ALREADY_STARTED ) ;
MEI_CL_CS ( OUT_OF_RESOURCES ) ;
MEI_CL_CS ( MESSAGE_SMALL ) ;
default : return " unknown " ;
}
# undef MEI_CL_CCS
}
2014-09-29 16:31:33 +03:00
const char * mei_hbm_state_str ( enum mei_hbm_state state )
{
# define MEI_HBM_STATE(state) case MEI_HBM_##state: return #state
switch ( state ) {
MEI_HBM_STATE ( IDLE ) ;
MEI_HBM_STATE ( STARTING ) ;
MEI_HBM_STATE ( STARTED ) ;
MEI_HBM_STATE ( ENUM_CLIENTS ) ;
MEI_HBM_STATE ( CLIENT_PROPERTIES ) ;
MEI_HBM_STATE ( STOPPED ) ;
default :
return " unknown " ;
}
# undef MEI_HBM_STATE
}
2014-02-17 15:13:20 +02:00
/**
* mei_cl_conn_status_to_errno - convert client connect response
* status to error code
*
* @ status : client connect response status
*
2014-09-29 16:31:49 +03:00
* Return : corresponding error code
2014-02-17 15:13:20 +02:00
*/
static int mei_cl_conn_status_to_errno ( enum mei_cl_connect_status status )
{
switch ( status ) {
case MEI_CL_CONN_SUCCESS : return 0 ;
case MEI_CL_CONN_NOT_FOUND : return - ENOTTY ;
case MEI_CL_CONN_ALREADY_STARTED : return - EBUSY ;
case MEI_CL_CONN_OUT_OF_RESOURCES : return - EBUSY ;
case MEI_CL_CONN_MESSAGE_SMALL : return - EINVAL ;
default : return - EINVAL ;
}
}
2014-05-07 16:51:28 +03:00
/**
* mei_hbm_idle - set hbm to idle state
*
* @ dev : the device structure
*/
void mei_hbm_idle ( struct mei_device * dev )
{
dev - > init_clients_timer = 0 ;
dev - > hbm_state = MEI_HBM_IDLE ;
}
/**
2014-08-21 14:29:14 +03:00
* mei_me_cl_remove_all - remove all me clients
2014-05-07 16:51:28 +03:00
*
* @ dev : the device structure
*/
2014-08-21 14:29:14 +03:00
static void mei_me_cl_remove_all ( struct mei_device * dev )
2014-05-07 16:51:28 +03:00
{
2014-08-21 14:29:13 +03:00
struct mei_me_client * me_cl , * next ;
2014-09-29 16:31:37 +03:00
2014-08-21 14:29:14 +03:00
list_for_each_entry_safe ( me_cl , next , & dev - > me_clients , list ) {
list_del ( & me_cl - > list ) ;
kfree ( me_cl ) ;
}
}
2014-08-21 14:29:13 +03:00
2014-08-21 14:29:14 +03:00
/**
* mei_hbm_reset - reset hbm counters and book keeping data structurs
*
* @ dev : the device structure
*/
void mei_hbm_reset ( struct mei_device * dev )
{
2014-05-07 16:51:28 +03:00
dev - > me_client_index = 0 ;
2014-08-21 14:29:14 +03:00
mei_me_cl_remove_all ( dev ) ;
2014-05-07 16:51:28 +03:00
mei_hbm_idle ( dev ) ;
}
2014-09-29 16:31:40 +03:00
/**
* mei_hbm_hdr - construct hbm header
*
* @ hdr : hbm header
* @ length : payload length
*/
static inline void mei_hbm_hdr ( struct mei_msg_hdr * hdr , size_t length )
{
hdr - > host_addr = 0 ;
hdr - > me_addr = 0 ;
hdr - > length = length ;
hdr - > msg_complete = 1 ;
hdr - > reserved = 0 ;
}
2012-12-25 19:06:09 +02:00
/**
* mei_hbm_cl_hdr - construct client hbm header
2013-04-05 01:05:05 +09:00
*
2014-08-21 14:29:11 +03:00
* @ cl : client
2012-12-25 19:06:09 +02:00
* @ hbm_cmd : host bus message command
* @ buf : buffer for cl header
* @ len : buffer length
*/
static inline
void mei_hbm_cl_hdr ( struct mei_cl * cl , u8 hbm_cmd , void * buf , size_t len )
{
struct mei_hbm_cl_cmd * cmd = buf ;
memset ( cmd , 0 , len ) ;
cmd - > hbm_cmd = hbm_cmd ;
cmd - > host_addr = cl - > host_client_id ;
cmd - > me_addr = cl - > me_client_id ;
}
2014-08-21 14:29:11 +03:00
/**
* mei_hbm_cl_write - write simple hbm client message
*
* @ dev : the device structure
* @ cl : client
* @ hbm_cmd : host bus message command
* @ len : buffer length
*/
static inline
int mei_hbm_cl_write ( struct mei_device * dev ,
struct mei_cl * cl , u8 hbm_cmd , size_t len )
{
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
mei_hbm_hdr ( mei_hdr , len ) ;
mei_hbm_cl_hdr ( cl , hbm_cmd , dev - > wr_msg . data , len ) ;
return mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
}
2012-12-25 19:06:09 +02:00
/**
2014-08-21 14:29:20 +03:00
* mei_hbm_cl_addr_equal - check if the client ' s and
* the message address match
2012-12-25 19:06:09 +02:00
*
2014-08-21 14:29:20 +03:00
* @ cl : client
* @ cmd : hbm client message
2012-12-25 19:06:09 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : true if addresses are the same
2012-12-25 19:06:09 +02:00
*/
static inline
2014-08-21 14:29:20 +03:00
bool mei_hbm_cl_addr_equal ( struct mei_cl * cl , struct mei_hbm_cl_cmd * cmd )
2012-12-25 19:06:09 +02:00
{
return cl - > host_client_id = = cmd - > host_addr & &
cl - > me_client_id = = cmd - > me_addr ;
}
2014-08-21 14:29:20 +03:00
/**
* mei_hbm_cl_find_by_cmd - find recipient client
*
* @ dev : the device structure
* @ buf : a buffer with hbm cl command
*
2014-09-29 16:31:49 +03:00
* Return : the recipient client or NULL if not found
2014-08-21 14:29:20 +03:00
*/
static inline
struct mei_cl * mei_hbm_cl_find_by_cmd ( struct mei_device * dev , void * buf )
{
struct mei_hbm_cl_cmd * cmd = ( struct mei_hbm_cl_cmd * ) buf ;
struct mei_cl * cl ;
list_for_each_entry ( cl , & dev - > file_list , link )
if ( mei_hbm_cl_addr_equal ( cl , cmd ) )
return cl ;
return NULL ;
}
2014-08-21 14:29:19 +03:00
/**
* mei_hbm_start_wait - wait for start response message .
*
* @ dev : the device structure
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and < 0 on failure
2014-08-21 14:29:19 +03:00
*/
2013-04-18 23:03:48 +03:00
int mei_hbm_start_wait ( struct mei_device * dev )
{
int ret ;
2014-08-21 14:29:19 +03:00
if ( dev - > hbm_state > MEI_HBM_STARTING )
2013-04-18 23:03:48 +03:00
return 0 ;
mutex_unlock ( & dev - > device_lock ) ;
2014-08-21 14:29:19 +03:00
ret = wait_event_timeout ( dev - > wait_hbm_start ,
dev - > hbm_state ! = MEI_HBM_STARTING ,
2014-01-14 23:10:10 +02:00
mei_secs_to_jiffies ( MEI_HBM_TIMEOUT ) ) ;
2013-04-18 23:03:48 +03:00
mutex_lock ( & dev - > device_lock ) ;
2014-08-21 14:29:19 +03:00
if ( ret = = 0 & & ( dev - > hbm_state < = MEI_HBM_STARTING ) ) {
2013-04-18 23:03:48 +03:00
dev - > hbm_state = MEI_HBM_IDLE ;
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " waiting for mei start failed \n " ) ;
2014-02-19 17:35:49 +02:00
return - ETIME ;
2013-04-18 23:03:48 +03:00
}
return 0 ;
}
2012-12-25 19:06:07 +02:00
/**
2012-12-25 19:06:11 +02:00
* mei_hbm_start_req - sends start request message .
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure
2014-01-08 20:19:21 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and < 0 on failure
2012-12-25 19:06:07 +02:00
*/
2013-04-18 23:03:48 +03:00
int mei_hbm_start_req ( struct mei_device * dev )
2012-12-25 19:06:07 +02:00
{
2012-12-25 19:06:10 +02:00
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
2012-12-25 19:06:07 +02:00
struct hbm_host_version_request * start_req ;
const size_t len = sizeof ( struct hbm_host_version_request ) ;
2014-01-08 20:19:21 +02:00
int ret ;
2012-12-25 19:06:07 +02:00
2014-08-21 14:29:13 +03:00
mei_hbm_reset ( dev ) ;
2012-12-25 19:06:10 +02:00
mei_hbm_hdr ( mei_hdr , len ) ;
2012-12-25 19:06:07 +02:00
/* host start message */
2012-12-25 19:06:10 +02:00
start_req = ( struct hbm_host_version_request * ) dev - > wr_msg . data ;
2012-12-25 19:06:07 +02:00
memset ( start_req , 0 , len ) ;
start_req - > hbm_cmd = HOST_START_REQ_CMD ;
start_req - > host_version . major_version = HBM_MAJOR_VERSION ;
start_req - > host_version . minor_version = HBM_MINOR_VERSION ;
2013-04-18 23:03:48 +03:00
dev - > hbm_state = MEI_HBM_IDLE ;
2014-01-08 20:19:21 +02:00
ret = mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
if ( ret ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " version message write failed: ret = %d \n " ,
2014-01-08 20:19:21 +02:00
ret ) ;
return ret ;
2012-12-25 19:06:07 +02:00
}
2014-01-08 20:19:21 +02:00
2014-08-21 14:29:19 +03:00
dev - > hbm_state = MEI_HBM_STARTING ;
2012-12-25 19:06:07 +02:00
dev - > init_clients_timer = MEI_CLIENTS_INIT_TIMEOUT ;
2013-04-18 23:03:48 +03:00
return 0 ;
2012-12-25 19:06:07 +02:00
}
2013-04-18 23:03:48 +03:00
/*
2012-12-25 19:06:11 +02:00
* mei_hbm_enum_clients_req - sends enumeration client request message .
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and < 0 on failure
2012-12-25 19:06:07 +02:00
*/
2014-01-08 20:19:21 +02:00
static int mei_hbm_enum_clients_req ( struct mei_device * dev )
2012-12-25 19:06:07 +02:00
{
2012-12-25 19:06:10 +02:00
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
2012-12-25 19:06:07 +02:00
struct hbm_host_enum_request * enum_req ;
const size_t len = sizeof ( struct hbm_host_enum_request ) ;
2014-01-08 20:19:21 +02:00
int ret ;
2012-12-25 19:06:07 +02:00
/* enumerate clients */
2012-12-25 19:06:10 +02:00
mei_hbm_hdr ( mei_hdr , len ) ;
2012-12-25 19:06:07 +02:00
2012-12-25 19:06:10 +02:00
enum_req = ( struct hbm_host_enum_request * ) dev - > wr_msg . data ;
memset ( enum_req , 0 , len ) ;
2012-12-25 19:06:07 +02:00
enum_req - > hbm_cmd = HOST_ENUM_REQ_CMD ;
2014-01-08 20:19:21 +02:00
ret = mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
if ( ret ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " enumeration request write failed: ret = %d. \n " ,
2014-01-08 20:19:21 +02:00
ret ) ;
return ret ;
2012-12-25 19:06:07 +02:00
}
2013-04-18 23:03:48 +03:00
dev - > hbm_state = MEI_HBM_ENUM_CLIENTS ;
2012-12-25 19:06:07 +02:00
dev - > init_clients_timer = MEI_CLIENTS_INIT_TIMEOUT ;
2014-01-08 20:19:21 +02:00
return 0 ;
2012-12-25 19:06:07 +02:00
}
2014-08-21 14:29:13 +03:00
/*
* mei_hbm_me_cl_add - add new me client to the list
*
* @ dev : the device structure
* @ res : hbm property response
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and - ENOMEM on allocation failure
2014-08-21 14:29:13 +03:00
*/
static int mei_hbm_me_cl_add ( struct mei_device * dev ,
struct hbm_props_response * res )
{
struct mei_me_client * me_cl ;
me_cl = kzalloc ( sizeof ( struct mei_me_client ) , GFP_KERNEL ) ;
if ( ! me_cl )
return - ENOMEM ;
me_cl - > props = res - > client_properties ;
me_cl - > client_id = res - > me_addr ;
me_cl - > mei_flow_ctrl_creds = 0 ;
list_add ( & me_cl - > list , & dev - > me_clients ) ;
return 0 ;
}
2012-12-25 19:06:11 +02:00
/**
2013-04-05 01:05:05 +09:00
* mei_hbm_prop_req - request property for a single client
2012-12-25 19:06:11 +02:00
*
* @ dev : the device structure
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and < 0 on failure
2012-12-25 19:06:11 +02:00
*/
2012-12-25 19:06:07 +02:00
2012-12-25 19:06:11 +02:00
static int mei_hbm_prop_req ( struct mei_device * dev )
2012-12-25 19:06:07 +02:00
{
2012-12-25 19:06:10 +02:00
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
2012-12-25 19:06:07 +02:00
struct hbm_props_request * prop_req ;
const size_t len = sizeof ( struct hbm_props_request ) ;
unsigned long next_client_index ;
2014-01-08 20:19:21 +02:00
int ret ;
2012-12-25 19:06:07 +02:00
next_client_index = find_next_bit ( dev - > me_clients_map , MEI_CLIENTS_MAX ,
dev - > me_client_index ) ;
/* We got all client properties */
if ( next_client_index = = MEI_CLIENTS_MAX ) {
2013-04-18 23:03:48 +03:00
dev - > hbm_state = MEI_HBM_STARTED ;
2012-12-25 19:06:07 +02:00
schedule_work ( & dev - > init_work ) ;
return 0 ;
}
2012-12-25 19:06:10 +02:00
mei_hbm_hdr ( mei_hdr , len ) ;
prop_req = ( struct hbm_props_request * ) dev - > wr_msg . data ;
2012-12-25 19:06:07 +02:00
memset ( prop_req , 0 , sizeof ( struct hbm_props_request ) ) ;
prop_req - > hbm_cmd = HOST_CLIENT_PROPERTIES_REQ_CMD ;
2014-08-21 14:29:10 +03:00
prop_req - > me_addr = next_client_index ;
2012-12-25 19:06:07 +02:00
2014-01-08 20:19:21 +02:00
ret = mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
if ( ret ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " properties request write failed: ret = %d \n " ,
2014-01-08 20:19:21 +02:00
ret ) ;
return ret ;
2012-12-25 19:06:07 +02:00
}
dev - > init_clients_timer = MEI_CLIENTS_INIT_TIMEOUT ;
dev - > me_client_index = next_client_index ;
return 0 ;
}
2014-03-18 22:51:55 +02:00
/*
* mei_hbm_pg - sends pg command
*
* @ dev : the device structure
* @ pg_cmd : the pg command code
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2014-08-21 14:29:21 +03:00
* - EOPNOTSUPP if the operation is not supported by the protocol
2014-03-18 22:51:55 +02:00
*/
int mei_hbm_pg ( struct mei_device * dev , u8 pg_cmd )
{
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
struct hbm_power_gate * req ;
const size_t len = sizeof ( struct hbm_power_gate ) ;
int ret ;
2014-08-21 14:29:21 +03:00
if ( ! dev - > hbm_f_pg_supported )
return - EOPNOTSUPP ;
2014-03-18 22:51:55 +02:00
mei_hbm_hdr ( mei_hdr , len ) ;
req = ( struct hbm_power_gate * ) dev - > wr_msg . data ;
memset ( req , 0 , len ) ;
req - > hbm_cmd = pg_cmd ;
ret = mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
if ( ret )
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " power gate command write failed. \n " ) ;
2014-03-18 22:51:55 +02:00
return ret ;
}
EXPORT_SYMBOL_GPL ( mei_hbm_pg ) ;
2012-12-25 19:06:10 +02:00
/**
2014-02-12 21:41:52 +02:00
* mei_hbm_stop_req - send stop request message
2012-12-25 19:06:10 +02:00
*
2014-09-29 16:31:49 +03:00
* @ dev : mei device
2014-02-12 21:41:52 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2012-12-25 19:06:10 +02:00
*/
2014-02-12 21:41:52 +02:00
static int mei_hbm_stop_req ( struct mei_device * dev )
2012-12-25 19:06:10 +02:00
{
2014-02-12 21:41:52 +02:00
struct mei_msg_hdr * mei_hdr = & dev - > wr_msg . hdr ;
2012-12-25 19:06:10 +02:00
struct hbm_host_stop_request * req =
2014-02-12 21:41:52 +02:00
( struct hbm_host_stop_request * ) dev - > wr_msg . data ;
2012-12-25 19:06:10 +02:00
const size_t len = sizeof ( struct hbm_host_stop_request ) ;
mei_hbm_hdr ( mei_hdr , len ) ;
memset ( req , 0 , len ) ;
req - > hbm_cmd = HOST_STOP_REQ_CMD ;
req - > reason = DRIVER_STOP_REQUEST ;
2014-02-12 21:41:52 +02:00
return mei_write_message ( dev , mei_hdr , dev - > wr_msg . data ) ;
2012-12-25 19:06:10 +02:00
}
2012-12-25 19:06:07 +02:00
/**
2014-01-08 22:31:46 +02:00
* mei_hbm_cl_flow_control_req - sends flow control request .
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure
2012-12-25 19:06:11 +02:00
* @ cl : client info
2012-12-25 19:06:07 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2012-12-25 19:06:07 +02:00
*/
2012-12-25 19:06:11 +02:00
int mei_hbm_cl_flow_control_req ( struct mei_device * dev , struct mei_cl * cl )
2012-12-25 19:06:07 +02:00
{
const size_t len = sizeof ( struct hbm_flow_control ) ;
2014-09-29 16:31:37 +03:00
2014-03-16 14:35:55 +02:00
cl_dbg ( dev , cl , " sending flow control \n " ) ;
2014-08-21 14:29:11 +03:00
return mei_hbm_cl_write ( dev , cl , MEI_FLOW_CONTROL_CMD , len ) ;
2012-12-25 19:06:07 +02:00
}
2012-12-25 19:06:12 +02:00
/**
2013-04-05 01:05:05 +09:00
* mei_hbm_add_single_flow_creds - adds single buffer credentials .
2012-12-25 19:06:12 +02:00
*
2013-04-05 01:05:05 +09:00
* @ dev : the device structure
2012-12-25 19:06:12 +02:00
* @ flow : flow control .
2014-02-17 15:13:23 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success , < 0 otherwise
2012-12-25 19:06:12 +02:00
*/
2014-02-17 15:13:23 +02:00
static int mei_hbm_add_single_flow_creds ( struct mei_device * dev ,
2012-12-25 19:06:12 +02:00
struct hbm_flow_control * flow )
{
2014-02-17 15:13:23 +02:00
struct mei_me_client * me_cl ;
2014-08-24 12:08:55 +03:00
me_cl = mei_me_cl_by_id ( dev , flow - > me_addr ) ;
if ( ! me_cl ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " no such me client %d \n " ,
2014-02-17 15:13:23 +02:00
flow - > me_addr ) ;
2014-08-24 12:08:55 +03:00
return - ENOENT ;
2014-02-17 15:13:23 +02:00
}
2014-08-24 12:08:55 +03:00
if ( WARN_ON ( me_cl - > props . single_recv_buf = = 0 ) )
return - EINVAL ;
me_cl - > mei_flow_ctrl_creds + + ;
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " recv flow ctrl msg ME %d (single) creds = %d. \n " ,
2014-08-24 12:08:55 +03:00
flow - > me_addr , me_cl - > mei_flow_ctrl_creds ) ;
2014-02-17 15:13:23 +02:00
return 0 ;
2012-12-25 19:06:12 +02:00
}
/**
* mei_hbm_cl_flow_control_res - flow control response from me
*
* @ dev : the device structure
* @ flow_control : flow control response bus message
*/
static void mei_hbm_cl_flow_control_res ( struct mei_device * dev ,
2014-08-21 14:29:20 +03:00
struct hbm_flow_control * flow_control )
2012-12-25 19:06:12 +02:00
{
2014-02-17 15:13:25 +02:00
struct mei_cl * cl ;
2012-12-25 19:06:12 +02:00
if ( ! flow_control - > host_addr ) {
/* single receive buffer */
mei_hbm_add_single_flow_creds ( dev , flow_control ) ;
return ;
}
2014-08-21 14:29:20 +03:00
cl = mei_hbm_cl_find_by_cmd ( dev , flow_control ) ;
if ( cl ) {
cl - > mei_flow_ctrl_creds + + ;
cl_dbg ( dev , cl , " flow control creds = %d. \n " ,
2014-08-21 14:29:13 +03:00
cl - > mei_flow_ctrl_creds ) ;
2012-12-25 19:06:12 +02:00
}
}
2012-12-25 19:06:07 +02:00
/**
2012-12-25 19:06:11 +02:00
* mei_hbm_cl_disconnect_req - sends disconnect message to fw .
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure
2012-12-25 19:06:11 +02:00
* @ cl : a client to disconnect from
2012-12-25 19:06:07 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2012-12-25 19:06:07 +02:00
*/
2012-12-25 19:06:11 +02:00
int mei_hbm_cl_disconnect_req ( struct mei_device * dev , struct mei_cl * cl )
2012-12-25 19:06:07 +02:00
{
const size_t len = sizeof ( struct hbm_client_connect_request ) ;
2014-09-29 16:31:37 +03:00
2014-08-21 14:29:11 +03:00
return mei_hbm_cl_write ( dev , cl , CLIENT_DISCONNECT_REQ_CMD , len ) ;
2012-12-25 19:06:07 +02:00
}
2014-02-12 21:41:52 +02:00
/**
* mei_hbm_cl_disconnect_rsp - sends disconnect respose to the FW
*
* @ dev : the device structure
* @ cl : a client to disconnect from
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2014-02-12 21:41:52 +02:00
*/
int mei_hbm_cl_disconnect_rsp ( struct mei_device * dev , struct mei_cl * cl )
{
const size_t len = sizeof ( struct hbm_client_connect_response ) ;
2014-09-29 16:31:37 +03:00
2014-08-21 14:29:11 +03:00
return mei_hbm_cl_write ( dev , cl , CLIENT_DISCONNECT_RES_CMD , len ) ;
2014-02-12 21:41:52 +02:00
}
2012-12-25 19:06:12 +02:00
/**
2014-08-21 14:29:18 +03:00
* mei_hbm_cl_disconnect_res - update the client state according
* disconnect response
2012-12-25 19:06:12 +02:00
*
2014-08-21 14:29:18 +03:00
* @ cl : mei host client
* @ cmd : disconnect client response host bus message
2012-12-25 19:06:12 +02:00
*/
2014-08-21 14:29:18 +03:00
static void mei_hbm_cl_disconnect_res ( struct mei_cl * cl ,
struct mei_hbm_cl_cmd * cmd )
2012-12-25 19:06:12 +02:00
{
2014-08-21 14:29:18 +03:00
struct hbm_client_connect_response * rs =
( struct hbm_client_connect_response * ) cmd ;
2012-12-25 19:06:12 +02:00
2014-09-29 16:31:42 +03:00
dev_dbg ( cl - > dev - > dev , " hbm: disconnect response cl:host=%02d me=%02d status=%d \n " ,
2014-02-17 15:13:20 +02:00
rs - > me_addr , rs - > host_addr , rs - > status ) ;
2012-12-25 19:06:12 +02:00
2014-08-21 14:29:18 +03:00
if ( rs - > status = = MEI_CL_DISCONN_SUCCESS )
cl - > state = MEI_FILE_DISCONNECTED ;
cl - > status = 0 ;
2012-12-25 19:06:12 +02:00
}
2012-12-25 19:06:07 +02:00
/**
2012-12-25 19:06:11 +02:00
* mei_hbm_cl_connect_req - send connection request to specific me client
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure
2012-12-25 19:06:11 +02:00
* @ cl : a client to connect to
2012-12-25 19:06:07 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : - EIO on write failure
2012-12-25 19:06:07 +02:00
*/
2012-12-25 19:06:11 +02:00
int mei_hbm_cl_connect_req ( struct mei_device * dev , struct mei_cl * cl )
2012-12-25 19:06:07 +02:00
{
const size_t len = sizeof ( struct hbm_client_connect_request ) ;
2014-09-29 16:31:37 +03:00
2014-08-21 14:29:11 +03:00
return mei_hbm_cl_write ( dev , cl , CLIENT_CONNECT_REQ_CMD , len ) ;
2012-12-25 19:06:07 +02:00
}
2012-12-25 19:06:12 +02:00
/**
2014-08-21 14:29:18 +03:00
* mei_hbm_cl_connect_res - update the client state according
* connection response
2012-12-25 19:06:12 +02:00
*
2014-08-21 14:29:18 +03:00
* @ cl : mei host client
* @ cmd : connect client response host bus message
2012-12-25 19:06:12 +02:00
*/
2014-08-21 14:29:18 +03:00
static void mei_hbm_cl_connect_res ( struct mei_cl * cl ,
struct mei_hbm_cl_cmd * cmd )
2012-12-25 19:06:12 +02:00
{
2014-08-21 14:29:18 +03:00
struct hbm_client_connect_response * rs =
( struct hbm_client_connect_response * ) cmd ;
2012-12-25 19:06:12 +02:00
2014-09-29 16:31:42 +03:00
dev_dbg ( cl - > dev - > dev , " hbm: connect response cl:host=%02d me=%02d status=%s \n " ,
2014-02-17 15:13:20 +02:00
rs - > me_addr , rs - > host_addr ,
mei_cl_conn_status_str ( rs - > status ) ) ;
2012-12-25 19:06:12 +02:00
2014-08-21 14:29:18 +03:00
if ( rs - > status = = MEI_CL_CONN_SUCCESS )
cl - > state = MEI_FILE_CONNECTED ;
else
cl - > state = MEI_FILE_DISCONNECTED ;
cl - > status = mei_cl_conn_status_to_errno ( rs - > status ) ;
}
/**
* mei_hbm_cl_res - process hbm response received on behalf
* an client
*
* @ dev : the device structure
* @ rs : hbm client message
* @ fop_type : file operation type
*/
static void mei_hbm_cl_res ( struct mei_device * dev ,
struct mei_hbm_cl_cmd * rs ,
enum mei_cb_file_ops fop_type )
{
struct mei_cl * cl ;
struct mei_cl_cb * cb , * next ;
2012-12-25 19:06:12 +02:00
2014-08-21 14:29:18 +03:00
cl = NULL ;
2014-02-17 15:13:21 +02:00
list_for_each_entry_safe ( cb , next , & dev - > ctrl_rd_list . list , list ) {
2012-12-25 19:06:12 +02:00
2014-02-17 15:13:21 +02:00
cl = cb - > cl ;
/* this should not happen */
if ( WARN_ON ( ! cl ) ) {
list_del_init ( & cb - > list ) ;
continue ;
}
2012-12-25 19:06:12 +02:00
2014-08-21 14:29:18 +03:00
if ( cb - > fop_type ! = fop_type )
2014-02-17 15:13:21 +02:00
continue ;
2012-12-25 19:06:12 +02:00
2014-02-17 15:13:21 +02:00
if ( mei_hbm_cl_addr_equal ( cl , rs ) ) {
list_del ( & cb - > list ) ;
break ;
2012-12-25 19:06:12 +02:00
}
}
2014-02-17 15:13:21 +02:00
if ( ! cl )
return ;
2014-08-21 14:29:18 +03:00
switch ( fop_type ) {
case MEI_FOP_CONNECT :
mei_hbm_cl_connect_res ( cl , rs ) ;
break ;
case MEI_FOP_DISCONNECT :
mei_hbm_cl_disconnect_res ( cl , rs ) ;
break ;
default :
return ;
}
2014-02-17 15:13:21 +02:00
cl - > timer_count = 0 ;
2014-08-21 14:29:18 +03:00
wake_up ( & cl - > wait ) ;
2012-12-25 19:06:12 +02:00
}
2012-12-25 19:06:07 +02:00
/**
2014-01-08 22:31:46 +02:00
* mei_hbm_fw_disconnect_req - disconnect request initiated by ME firmware
* host sends disconnect response
2012-12-25 19:06:07 +02:00
*
* @ dev : the device structure .
2012-12-25 19:06:11 +02:00
* @ disconnect_req : disconnect request bus message from the me
2014-02-12 21:41:52 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : - ENOMEM on allocation failure
2012-12-25 19:06:07 +02:00
*/
2014-02-12 21:41:52 +02:00
static int mei_hbm_fw_disconnect_req ( struct mei_device * dev ,
2012-12-25 19:06:07 +02:00
struct hbm_client_connect_request * disconnect_req )
{
2014-02-17 15:13:25 +02:00
struct mei_cl * cl ;
2014-02-12 21:41:52 +02:00
struct mei_cl_cb * cb ;
2012-12-25 19:06:07 +02:00
2014-08-21 14:29:20 +03:00
cl = mei_hbm_cl_find_by_cmd ( dev , disconnect_req ) ;
if ( cl ) {
cl_dbg ( dev , cl , " disconnect request received \n " ) ;
cl - > state = MEI_FILE_DISCONNECTED ;
cl - > timer_count = 0 ;
cb = mei_io_cb_init ( cl , NULL ) ;
if ( ! cb )
return - ENOMEM ;
cb - > fop_type = MEI_FOP_DISCONNECT_RSP ;
cl_dbg ( dev , cl , " add disconnect response as first \n " ) ;
list_add ( & cb - > list , & dev - > ctrl_wr_list . list ) ;
2012-12-25 19:06:07 +02:00
}
2014-02-12 21:41:52 +02:00
return 0 ;
2012-12-25 19:06:07 +02:00
}
2014-08-21 14:29:21 +03:00
/**
2014-09-29 16:31:49 +03:00
* mei_hbm_config_features - check what hbm features and commands
2014-08-21 14:29:21 +03:00
* are supported by the fw
*
* @ dev : the device structure
*/
static void mei_hbm_config_features ( struct mei_device * dev )
{
/* Power Gating Isolation Support */
dev - > hbm_f_pg_supported = 0 ;
if ( dev - > version . major_version > HBM_MAJOR_VERSION_PGI )
dev - > hbm_f_pg_supported = 1 ;
if ( dev - > version . major_version = = HBM_MAJOR_VERSION_PGI & &
dev - > version . minor_version > = HBM_MINOR_VERSION_PGI )
dev - > hbm_f_pg_supported = 1 ;
}
2012-12-25 19:06:07 +02:00
2013-06-16 09:16:31 +03:00
/**
* mei_hbm_version_is_supported - checks whether the driver can
* support the hbm version of the device
*
* @ dev : the device structure
2014-09-29 16:31:49 +03:00
* Return : true if driver can support hbm version of the device
2013-06-16 09:16:31 +03:00
*/
bool mei_hbm_version_is_supported ( struct mei_device * dev )
{
return ( dev - > version . major_version < HBM_MAJOR_VERSION ) | |
( dev - > version . major_version = = HBM_MAJOR_VERSION & &
dev - > version . minor_version < = HBM_MINOR_VERSION ) ;
}
2012-12-25 19:06:07 +02:00
/**
* mei_hbm_dispatch - bottom half read routine after ISR to
* handle the read bus message cmd processing .
*
* @ dev : the device structure
2014-09-29 16:31:49 +03:00
* @ hdr : header of bus message
2014-01-08 20:19:21 +02:00
*
2014-09-29 16:31:49 +03:00
* Return : 0 on success and < 0 on failure
2012-12-25 19:06:07 +02:00
*/
2014-01-08 20:19:21 +02:00
int mei_hbm_dispatch ( struct mei_device * dev , struct mei_msg_hdr * hdr )
2012-12-25 19:06:07 +02:00
{
struct mei_bus_message * mei_msg ;
struct hbm_host_version_response * version_res ;
struct hbm_props_response * props_res ;
struct hbm_host_enum_response * enum_res ;
2014-08-21 14:29:18 +03:00
struct mei_hbm_cl_cmd * cl_cmd ;
struct hbm_client_connect_request * disconnect_req ;
struct hbm_flow_control * flow_control ;
2012-12-25 19:06:07 +02:00
/* read the message to our buffer */
BUG_ON ( hdr - > length > = sizeof ( dev - > rd_msg_buf ) ) ;
mei_read_slots ( dev , dev - > rd_msg_buf , hdr - > length ) ;
mei_msg = ( struct mei_bus_message * ) dev - > rd_msg_buf ;
2014-08-21 14:29:18 +03:00
cl_cmd = ( struct mei_hbm_cl_cmd * ) mei_msg ;
2012-12-25 19:06:07 +02:00
2014-01-08 20:19:22 +02:00
/* ignore spurious message and prevent reset nesting
* hbm is put to idle during system reset
*/
if ( dev - > hbm_state = = MEI_HBM_IDLE ) {
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: state is idle ignore spurious messages \n " ) ;
2014-01-08 20:19:22 +02:00
return 0 ;
}
2012-12-25 19:06:07 +02:00
switch ( mei_msg - > hbm_cmd ) {
case HOST_START_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: start: response message received. \n " ) ;
2014-01-08 20:19:21 +02:00
dev - > init_clients_timer = 0 ;
2012-12-25 19:06:07 +02:00
version_res = ( struct hbm_host_version_response * ) mei_msg ;
2013-06-16 09:16:31 +03:00
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " HBM VERSION: DRIVER=%02d:%02d DEVICE=%02d:%02d \n " ,
2013-06-16 09:16:31 +03:00
HBM_MAJOR_VERSION , HBM_MINOR_VERSION ,
version_res - > me_max_version . major_version ,
version_res - > me_max_version . minor_version ) ;
if ( version_res - > host_version_supported ) {
dev - > version . major_version = HBM_MAJOR_VERSION ;
dev - > version . minor_version = HBM_MINOR_VERSION ;
} else {
dev - > version . major_version =
version_res - > me_max_version . major_version ;
dev - > version . minor_version =
version_res - > me_max_version . minor_version ;
}
if ( ! mei_hbm_version_is_supported ( dev ) ) {
2014-09-29 16:31:42 +03:00
dev_warn ( dev - > dev , " hbm: start: version mismatch - stopping the driver. \n " ) ;
2012-12-25 19:06:07 +02:00
2014-01-08 20:19:21 +02:00
dev - > hbm_state = MEI_HBM_STOPPED ;
2014-02-12 21:41:52 +02:00
if ( mei_hbm_stop_req ( dev ) ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: start: failed to send stop request \n " ) ;
2014-01-08 20:19:21 +02:00
return - EIO ;
}
break ;
}
2013-04-18 23:03:48 +03:00
2014-08-21 14:29:21 +03:00
mei_hbm_config_features ( dev ) ;
2014-01-08 20:19:21 +02:00
if ( dev - > dev_state ! = MEI_DEV_INIT_CLIENTS | |
2014-08-21 14:29:19 +03:00
dev - > hbm_state ! = MEI_HBM_STARTING ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: start: state mismatch, [%d, %d] \n " ,
2014-01-08 20:19:21 +02:00
dev - > dev_state , dev - > hbm_state ) ;
return - EPROTO ;
2012-12-25 19:06:10 +02:00
}
2012-12-25 19:06:07 +02:00
2014-01-08 20:19:21 +02:00
dev - > hbm_state = MEI_HBM_STARTED ;
if ( mei_hbm_enum_clients_req ( dev ) ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: start: failed to send enumeration request \n " ) ;
2014-01-08 20:19:21 +02:00
return - EIO ;
2012-12-25 19:06:07 +02:00
}
2014-08-21 14:29:19 +03:00
wake_up ( & dev - > wait_hbm_start ) ;
2012-12-25 19:06:07 +02:00
break ;
case CLIENT_CONNECT_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: client connect response: message received. \n " ) ;
2014-08-21 14:29:18 +03:00
mei_hbm_cl_res ( dev , cl_cmd , MEI_FOP_CONNECT ) ;
2012-12-25 19:06:07 +02:00
break ;
case CLIENT_DISCONNECT_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: client disconnect response: message received. \n " ) ;
2014-08-21 14:29:18 +03:00
mei_hbm_cl_res ( dev , cl_cmd , MEI_FOP_DISCONNECT ) ;
2012-12-25 19:06:07 +02:00
break ;
case MEI_FLOW_CONTROL_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: client flow control response: message received. \n " ) ;
2014-01-08 20:19:21 +02:00
2012-12-25 19:06:07 +02:00
flow_control = ( struct hbm_flow_control * ) mei_msg ;
2012-12-25 19:06:12 +02:00
mei_hbm_cl_flow_control_res ( dev , flow_control ) ;
2012-12-25 19:06:07 +02:00
break ;
2014-03-18 22:51:55 +02:00
case MEI_PG_ISOLATION_ENTRY_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " power gate isolation entry response received \n " ) ;
2014-03-18 22:52:00 +02:00
dev - > pg_event = MEI_PG_EVENT_RECEIVED ;
2014-03-18 22:51:55 +02:00
if ( waitqueue_active ( & dev - > wait_pg ) )
wake_up ( & dev - > wait_pg ) ;
break ;
case MEI_PG_ISOLATION_EXIT_REQ_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " power gate isolation exit request received \n " ) ;
2014-03-18 22:52:00 +02:00
dev - > pg_event = MEI_PG_EVENT_RECEIVED ;
2014-03-18 22:51:55 +02:00
if ( waitqueue_active ( & dev - > wait_pg ) )
wake_up ( & dev - > wait_pg ) ;
2014-03-18 22:52:02 +02:00
else
/*
* If the driver is not waiting on this then
* this is HW initiated exit from PG .
* Start runtime pm resume sequence to exit from PG .
*/
2014-09-29 16:31:42 +03:00
pm_request_resume ( dev - > dev ) ;
2014-03-18 22:51:55 +02:00
break ;
2012-12-25 19:06:07 +02:00
case HOST_CLIENT_PROPERTIES_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: properties response: message received. \n " ) ;
2014-01-08 20:19:21 +02:00
dev - > init_clients_timer = 0 ;
2014-08-21 14:29:13 +03:00
if ( dev - > dev_state ! = MEI_DEV_INIT_CLIENTS | |
dev - > hbm_state ! = MEI_HBM_CLIENT_PROPERTIES ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: properties response: state mismatch, [%d, %d] \n " ,
2014-08-21 14:29:13 +03:00
dev - > dev_state , dev - > hbm_state ) ;
2014-01-08 20:19:21 +02:00
return - EPROTO ;
}
2012-12-25 19:06:07 +02:00
props_res = ( struct hbm_props_response * ) mei_msg ;
2014-01-08 20:19:21 +02:00
if ( props_res - > status ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: properties response: wrong status = %d %s \n " ,
2014-08-21 14:29:16 +03:00
props_res - > status ,
mei_hbm_status_str ( props_res - > status ) ) ;
2014-01-08 20:19:21 +02:00
return - EPROTO ;
2012-12-25 19:06:07 +02:00
}
2014-08-21 14:29:13 +03:00
mei_hbm_me_cl_add ( dev , props_res ) ;
2012-12-25 19:06:07 +02:00
dev - > me_client_index + + ;
2012-12-25 19:06:11 +02:00
/* request property for the next client */
2014-01-08 20:19:21 +02:00
if ( mei_hbm_prop_req ( dev ) )
return - EIO ;
2012-12-25 19:06:07 +02:00
break ;
case HOST_ENUM_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: enumeration response: message received \n " ) ;
2014-01-08 20:19:21 +02:00
dev - > init_clients_timer = 0 ;
2012-12-25 19:06:07 +02:00
enum_res = ( struct hbm_host_enum_response * ) mei_msg ;
2013-09-02 03:11:01 +03:00
BUILD_BUG_ON ( sizeof ( dev - > me_clients_map )
< sizeof ( enum_res - > valid_addresses ) ) ;
memcpy ( dev - > me_clients_map , enum_res - > valid_addresses ,
2014-08-21 14:29:13 +03:00
sizeof ( enum_res - > valid_addresses ) ) ;
2014-01-08 20:19:21 +02:00
if ( dev - > dev_state ! = MEI_DEV_INIT_CLIENTS | |
dev - > hbm_state ! = MEI_HBM_ENUM_CLIENTS ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: enumeration response: state mismatch, [%d, %d] \n " ,
2014-01-08 20:19:21 +02:00
dev - > dev_state , dev - > hbm_state ) ;
return - EPROTO ;
2012-12-25 19:06:07 +02:00
}
2014-01-08 20:19:21 +02:00
dev - > hbm_state = MEI_HBM_CLIENT_PROPERTIES ;
/* first property request */
if ( mei_hbm_prop_req ( dev ) )
return - EIO ;
2012-12-25 19:06:07 +02:00
break ;
case HOST_STOP_RES_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: stop response: message received \n " ) ;
2014-01-08 20:19:21 +02:00
dev - > init_clients_timer = 0 ;
if ( dev - > hbm_state ! = MEI_HBM_STOPPED ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: stop response: state mismatch, [%d, %d] \n " ,
2014-01-08 20:19:21 +02:00
dev - > dev_state , dev - > hbm_state ) ;
return - EPROTO ;
}
2013-04-18 23:03:48 +03:00
2014-01-12 00:36:09 +02:00
dev - > dev_state = MEI_DEV_POWER_DOWN ;
2014-09-29 16:31:42 +03:00
dev_info ( dev - > dev , " hbm: stop response: resetting. \n " ) ;
2014-01-08 20:19:21 +02:00
/* force the reset */
return - EPROTO ;
2012-12-25 19:06:07 +02:00
break ;
case CLIENT_DISCONNECT_REQ_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: disconnect request: message received \n " ) ;
2014-01-08 20:19:21 +02:00
2012-12-25 19:06:07 +02:00
disconnect_req = ( struct hbm_client_connect_request * ) mei_msg ;
2012-12-25 19:06:11 +02:00
mei_hbm_fw_disconnect_req ( dev , disconnect_req ) ;
2012-12-25 19:06:07 +02:00
break ;
case ME_STOP_REQ_CMD :
2014-09-29 16:31:42 +03:00
dev_dbg ( dev - > dev , " hbm: stop request: message received \n " ) ;
2014-01-08 20:19:21 +02:00
dev - > hbm_state = MEI_HBM_STOPPED ;
2014-02-12 21:41:52 +02:00
if ( mei_hbm_stop_req ( dev ) ) {
2014-09-29 16:31:42 +03:00
dev_err ( dev - > dev , " hbm: stop request: failed to send stop request \n " ) ;
2014-02-12 21:41:52 +02:00
return - EIO ;
}
2012-12-25 19:06:07 +02:00
break ;
default :
BUG ( ) ;
break ;
}
2014-01-08 20:19:21 +02:00
return 0 ;
2012-12-25 19:06:07 +02:00
}