2011-05-15 13:43:42 +03:00
/*
*
* Intel Management Engine Interface ( Intel MEI ) Linux driver
2012-02-09 19:25:53 +02:00
* Copyright ( c ) 2003 - 2012 , Intel Corporation .
2011-05-15 13:43:42 +03:00
*
* 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 .
*
*/
# include <linux/pci.h>
# include <linux/kthread.h>
# include <linux/interrupt.h>
# include <linux/fs.h>
# include <linux/jiffies.h>
2012-05-09 16:38:59 +03:00
# include <linux/mei.h>
2012-12-25 19:06:03 +02:00
# include "mei_dev.h"
2013-01-08 23:07:12 +02:00
# include "hbm.h"
2013-01-08 23:07:17 +02:00
# include "hw-me.h"
2013-01-08 23:07:14 +02:00
# include "client.h"
2011-05-15 13:43:42 +03:00
/**
2013-03-17 11:41:20 +02:00
* mei_cl_complete_handler - processes completed operation for a client
2011-05-15 13:43:42 +03:00
*
* @ cl : private data of the file object .
2013-03-17 11:41:20 +02:00
* @ cb : callback block .
2011-05-15 13:43:42 +03:00
*/
2013-03-17 11:41:20 +02:00
static void mei_cl_complete_handler ( struct mei_cl * cl , struct mei_cl_cb * cb )
2011-05-15 13:43:42 +03:00
{
2013-03-17 11:41:20 +02:00
if ( cb - > fop_type = = MEI_FOP_WRITE ) {
mei_io_cb_free ( cb ) ;
cb = NULL ;
2011-05-15 13:43:42 +03:00
cl - > writing_state = MEI_WRITE_COMPLETE ;
if ( waitqueue_active ( & cl - > tx_wait ) )
wake_up_interruptible ( & cl - > tx_wait ) ;
2013-03-17 11:41:20 +02:00
} else if ( cb - > fop_type = = MEI_FOP_READ & &
2011-05-15 13:43:42 +03:00
MEI_READING = = cl - > reading_state ) {
cl - > reading_state = MEI_READ_COMPLETE ;
if ( waitqueue_active ( & cl - > rx_wait ) )
wake_up_interruptible ( & cl - > rx_wait ) ;
}
}
2013-03-17 11:41:20 +02:00
/**
* mei_irq_compl_handler - dispatch complete handelers
* for the completed callbacks
*
* @ dev - mei device
* @ compl_list - list of completed cbs
*/
void mei_irq_compl_handler ( struct mei_device * dev , struct mei_cl_cb * compl_list )
{
struct mei_cl_cb * cb , * next ;
struct mei_cl * cl ;
list_for_each_entry_safe ( cb , next , & compl_list - > list , list ) {
cl = cb - > cl ;
list_del ( & cb - > list ) ;
if ( ! cl )
continue ;
dev_dbg ( & dev - > pdev - > dev , " completing call back. \n " ) ;
if ( cl = = & dev - > iamthif_cl )
mei_amthif_complete ( dev , cb ) ;
else
mei_cl_complete_handler ( cl , cb ) ;
}
}
2011-05-15 13:43:42 +03:00
/**
* _mei_irq_thread_state_ok - checks if mei header matches file private data
*
* @ cl : private data of the file object
* @ mei_hdr : header of mei client message
*
* returns ! = 0 if matches , 0 if no match .
*/
static int _mei_irq_thread_state_ok ( struct mei_cl * cl ,
struct mei_msg_hdr * mei_hdr )
{
return ( cl - > host_client_id = = mei_hdr - > host_addr & &
cl - > me_client_id = = mei_hdr - > me_addr & &
cl - > state = = MEI_FILE_CONNECTED & &
MEI_READ_COMPLETE ! = cl - > reading_state ) ;
}
/**
* mei_irq_thread_read_client_message - bottom half read routine after ISR to
* handle the read mei client message data processing .
*
* @ complete_list : An instance of our list structure
* @ dev : the device structure
* @ mei_hdr : header of mei client message
*
* returns 0 on success , < 0 on failure .
*/
2012-10-15 12:06:48 +02:00
static int mei_irq_thread_read_client_message ( struct mei_cl_cb * complete_list ,
2011-05-15 13:43:42 +03:00
struct mei_device * dev ,
struct mei_msg_hdr * mei_hdr )
{
struct mei_cl * cl ;
struct mei_cl_cb * cb_pos = NULL , * cb_next = NULL ;
2011-06-16 00:46:03 +03:00
unsigned char * buffer = NULL ;
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " start client msg \n " ) ;
2012-10-15 12:06:48 +02:00
if ( list_empty ( & dev - > read_list . list ) )
2011-05-15 13:43:42 +03:00
goto quit ;
2012-10-15 12:06:48 +02:00
list_for_each_entry_safe ( cb_pos , cb_next , & dev - > read_list . list , list ) {
2012-11-11 17:37:59 +02:00
cl = cb_pos - > cl ;
2011-05-15 13:43:42 +03:00
if ( cl & & _mei_irq_thread_state_ok ( cl , mei_hdr ) ) {
cl - > reading_state = MEI_READING ;
2012-10-09 16:50:16 +02:00
buffer = cb_pos - > response_buffer . data + cb_pos - > buf_idx ;
2011-05-15 13:43:42 +03:00
if ( cb_pos - > response_buffer . size <
2012-10-09 16:50:16 +02:00
mei_hdr - > length + cb_pos - > buf_idx ) {
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " message overflow. \n " ) ;
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2011-05-15 13:43:42 +03:00
return - ENOMEM ;
}
if ( buffer )
mei_read_slots ( dev , buffer , mei_hdr - > length ) ;
2012-10-09 16:50:16 +02:00
cb_pos - > buf_idx + = mei_hdr - > length ;
2011-05-15 13:43:42 +03:00
if ( mei_hdr - > msg_complete ) {
cl - > status = 0 ;
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev ,
2012-09-11 00:43:22 +03:00
" completed read H cl = %d, ME cl = %d, length = %lu \n " ,
2011-05-15 13:43:42 +03:00
cl - > host_client_id ,
cl - > me_client_id ,
2012-10-09 16:50:16 +02:00
cb_pos - > buf_idx ) ;
2012-10-15 12:06:48 +02:00
list_add_tail ( & cb_pos - > list ,
& complete_list - > list ) ;
2011-05-15 13:43:42 +03:00
}
break ;
}
}
quit :
dev_dbg ( & dev - > pdev - > dev , " message read \n " ) ;
if ( ! buffer ) {
2012-02-09 19:25:54 +02:00
mei_read_slots ( dev , dev - > rd_msg_buf , mei_hdr - > length ) ;
2012-12-25 19:06:00 +02:00
dev_dbg ( & dev - > pdev - > dev , " discarding message " MEI_HDR_FMT " \n " ,
MEI_HDR_PRM ( mei_hdr ) ) ;
2011-05-15 13:43:42 +03:00
}
return 0 ;
}
/**
* _mei_irq_thread_close - processes close related operation .
*
* @ dev : the device structure .
* @ slots : free slots .
* @ cb_pos : callback block .
* @ cl : private data of the file object .
* @ cmpl_list : complete list .
*
* returns 0 , OK ; otherwise , error .
*/
static int _mei_irq_thread_close ( struct mei_device * dev , s32 * slots ,
struct mei_cl_cb * cb_pos ,
struct mei_cl * cl ,
2012-10-15 12:06:48 +02:00
struct mei_cl_cb * cmpl_list )
2011-05-15 13:43:42 +03:00
{
2013-03-11 18:27:02 +02:00
u32 msg_slots =
mei_data2slots ( sizeof ( struct hbm_client_connect_request ) ) ;
2011-05-15 13:43:42 +03:00
2013-03-11 18:27:02 +02:00
if ( * slots < msg_slots )
return - EMSGSIZE ;
* slots - = msg_slots ;
2012-07-04 19:24:53 +03:00
2012-12-25 19:06:11 +02:00
if ( mei_hbm_cl_disconnect_req ( dev , cl ) ) {
2012-07-04 19:24:53 +03:00
cl - > status = 0 ;
2012-10-09 16:50:16 +02:00
cb_pos - > buf_idx = 0 ;
2012-10-15 12:06:48 +02:00
list_move_tail ( & cb_pos - > list , & cmpl_list - > list ) ;
2013-03-11 18:27:02 +02:00
return - EIO ;
2011-05-15 13:43:42 +03:00
}
2013-03-11 18:27:02 +02:00
cl - > state = MEI_FILE_DISCONNECTING ;
cl - > status = 0 ;
cb_pos - > buf_idx = 0 ;
list_move_tail ( & cb_pos - > list , & dev - > ctrl_rd_list . list ) ;
cl - > timer_count = MEI_CONNECT_TIMEOUT ;
2011-05-15 13:43:42 +03:00
return 0 ;
}
/**
* _mei_hb_read - processes read related operation .
*
* @ dev : the device structure .
* @ slots : free slots .
* @ cb_pos : callback block .
* @ cl : private data of the file object .
* @ cmpl_list : complete list .
*
* returns 0 , OK ; otherwise , error .
*/
static int _mei_irq_thread_read ( struct mei_device * dev , s32 * slots ,
struct mei_cl_cb * cb_pos ,
struct mei_cl * cl ,
2012-10-15 12:06:48 +02:00
struct mei_cl_cb * cmpl_list )
2011-05-15 13:43:42 +03:00
{
2013-03-11 18:27:02 +02:00
u32 msg_slots = mei_data2slots ( sizeof ( struct hbm_flow_control ) ) ;
if ( * slots < msg_slots ) {
2011-05-15 13:43:42 +03:00
/* return the cancel routine */
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2013-03-11 18:27:02 +02:00
return - EMSGSIZE ;
2011-05-15 13:43:42 +03:00
}
2013-03-11 18:27:02 +02:00
* slots - = msg_slots ;
2012-07-04 19:24:52 +03:00
2012-12-25 19:06:11 +02:00
if ( mei_hbm_cl_flow_control_req ( dev , cl ) ) {
2012-03-14 14:39:42 +02:00
cl - > status = - ENODEV ;
2012-10-09 16:50:16 +02:00
cb_pos - > buf_idx = 0 ;
2012-10-15 12:06:48 +02:00
list_move_tail ( & cb_pos - > list , & cmpl_list - > list ) ;
2012-03-14 14:39:42 +02:00
return - ENODEV ;
}
2012-10-15 12:06:48 +02:00
list_move_tail ( & cb_pos - > list , & dev - > read_list . list ) ;
2012-03-14 14:39:42 +02:00
2011-05-15 13:43:42 +03:00
return 0 ;
}
/**
* _mei_irq_thread_ioctl - processes ioctl related operation .
*
* @ dev : the device structure .
* @ slots : free slots .
* @ cb_pos : callback block .
* @ cl : private data of the file object .
* @ cmpl_list : complete list .
*
* returns 0 , OK ; otherwise , error .
*/
static int _mei_irq_thread_ioctl ( struct mei_device * dev , s32 * slots ,
struct mei_cl_cb * cb_pos ,
struct mei_cl * cl ,
2012-10-15 12:06:48 +02:00
struct mei_cl_cb * cmpl_list )
2011-05-15 13:43:42 +03:00
{
2013-03-11 18:27:02 +02:00
u32 msg_slots =
mei_data2slots ( sizeof ( struct hbm_client_connect_request ) ) ;
if ( * slots < msg_slots ) {
2011-05-15 13:43:42 +03:00
/* return the cancel routine */
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2013-03-11 18:27:02 +02:00
return - EMSGSIZE ;
2011-05-15 13:43:42 +03:00
}
2013-03-11 18:27:02 +02:00
* slots - = msg_slots ;
2012-07-04 19:24:53 +03:00
cl - > state = MEI_FILE_CONNECTING ;
2013-03-11 18:27:02 +02:00
2012-12-25 19:06:11 +02:00
if ( mei_hbm_cl_connect_req ( dev , cl ) ) {
2012-07-04 19:24:53 +03:00
cl - > status = - ENODEV ;
2012-10-09 16:50:16 +02:00
cb_pos - > buf_idx = 0 ;
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2012-07-04 19:24:53 +03:00
return - ENODEV ;
} else {
2012-10-15 12:06:48 +02:00
list_move_tail ( & cb_pos - > list , & dev - > ctrl_rd_list . list ) ;
2012-07-04 19:24:53 +03:00
cl - > timer_count = MEI_CONNECT_TIMEOUT ;
}
2011-05-15 13:43:42 +03:00
return 0 ;
}
/**
2012-11-18 15:13:16 +02:00
* mei_irq_thread_write_complete - write messages to device .
2011-05-15 13:43:42 +03:00
*
* @ dev : the device structure .
* @ slots : free slots .
2012-11-18 15:13:16 +02:00
* @ cb : callback block .
2011-05-15 13:43:42 +03:00
* @ cmpl_list : complete list .
*
* returns 0 , OK ; otherwise , error .
*/
2012-11-18 15:13:16 +02:00
static int mei_irq_thread_write_complete ( struct mei_device * dev , s32 * slots ,
struct mei_cl_cb * cb , struct mei_cl_cb * cmpl_list )
2011-05-15 13:43:42 +03:00
{
2012-12-25 19:06:10 +02:00
struct mei_msg_hdr mei_hdr ;
2012-11-18 15:13:16 +02:00
struct mei_cl * cl = cb - > cl ;
size_t len = cb - > request_buffer . size - cb - > buf_idx ;
2013-03-11 18:27:02 +02:00
u32 msg_slots = mei_data2slots ( len ) ;
2012-11-18 15:13:16 +02:00
2012-12-25 19:06:10 +02:00
mei_hdr . host_addr = cl - > host_client_id ;
mei_hdr . me_addr = cl - > me_client_id ;
mei_hdr . reserved = 0 ;
2011-05-15 13:43:42 +03:00
2012-11-18 15:13:16 +02:00
if ( * slots > = msg_slots ) {
2012-12-25 19:06:10 +02:00
mei_hdr . length = len ;
mei_hdr . msg_complete = 1 ;
2012-11-18 15:13:16 +02:00
/* Split the message only if we can write the whole host buffer */
2012-06-25 23:46:27 +03:00
} else if ( * slots = = dev - > hbuf_depth ) {
2012-11-18 15:13:16 +02:00
msg_slots = * slots ;
len = ( * slots * sizeof ( u32 ) ) - sizeof ( struct mei_msg_hdr ) ;
2012-12-25 19:06:10 +02:00
mei_hdr . length = len ;
mei_hdr . msg_complete = 0 ;
2011-05-15 13:43:42 +03:00
} else {
2012-11-18 15:13:16 +02:00
/* wait for next time the host buffer is empty */
return 0 ;
2011-05-15 13:43:42 +03:00
}
2012-11-18 15:13:16 +02:00
dev_dbg ( & dev - > pdev - > dev , " buf: size = %d idx = %lu \n " ,
cb - > request_buffer . size , cb - > buf_idx ) ;
2012-12-25 19:06:10 +02:00
dev_dbg ( & dev - > pdev - > dev , MEI_HDR_FMT , MEI_HDR_PRM ( & mei_hdr ) ) ;
2012-11-18 15:13:16 +02:00
* slots - = msg_slots ;
2012-12-25 19:06:10 +02:00
if ( mei_write_message ( dev , & mei_hdr ,
2012-12-25 19:05:59 +02:00
cb - > request_buffer . data + cb - > buf_idx ) ) {
2012-11-18 15:13:16 +02:00
cl - > status = - ENODEV ;
list_move_tail ( & cb - > list , & cmpl_list - > list ) ;
return - ENODEV ;
}
2013-01-08 23:07:14 +02:00
if ( mei_cl_flow_ctrl_reduce ( cl ) )
2012-11-18 15:13:16 +02:00
return - ENODEV ;
cl - > status = 0 ;
2012-12-25 19:06:10 +02:00
cb - > buf_idx + = mei_hdr . length ;
if ( mei_hdr . msg_complete )
2012-11-18 15:13:16 +02:00
list_move_tail ( & cb - > list , & dev - > write_waiting_list . list ) ;
2011-05-15 13:43:42 +03:00
return 0 ;
}
/**
* mei_irq_thread_read_handler - bottom half read routine after ISR to
* handle the read processing .
*
* @ dev : the device structure
2013-02-06 14:06:42 +02:00
* @ cmpl_list : An instance of our list structure
2011-05-15 13:43:42 +03:00
* @ slots : slots to read .
*
* returns 0 on success , < 0 on failure .
*/
2013-02-06 14:06:42 +02:00
int mei_irq_read_handler ( struct mei_device * dev ,
struct mei_cl_cb * cmpl_list , s32 * slots )
2011-05-15 13:43:42 +03:00
{
struct mei_msg_hdr * mei_hdr ;
struct mei_cl * cl_pos = NULL ;
struct mei_cl * cl_next = NULL ;
int ret = 0 ;
if ( ! dev - > rd_msg_hdr ) {
2013-02-06 14:06:41 +02:00
dev - > rd_msg_hdr = mei_read_hdr ( dev ) ;
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " slots =%08x. \n " , * slots ) ;
( * slots ) - - ;
dev_dbg ( & dev - > pdev - > dev , " slots =%08x. \n " , * slots ) ;
}
mei_hdr = ( struct mei_msg_hdr * ) & dev - > rd_msg_hdr ;
2012-12-25 19:06:00 +02:00
dev_dbg ( & dev - > pdev - > dev , MEI_HDR_FMT , MEI_HDR_PRM ( mei_hdr ) ) ;
2011-05-15 13:43:42 +03:00
if ( mei_hdr - > reserved | | ! dev - > rd_msg_hdr ) {
dev_dbg ( & dev - > pdev - > dev , " corrupted message header. \n " ) ;
ret = - EBADMSG ;
goto end ;
}
if ( mei_hdr - > host_addr | | mei_hdr - > me_addr ) {
list_for_each_entry_safe ( cl_pos , cl_next ,
& dev - > file_list , link ) {
dev_dbg ( & dev - > pdev - > dev ,
" list_for_each_entry_safe read host "
" client = %d, ME client = %d \n " ,
cl_pos - > host_client_id ,
cl_pos - > me_client_id ) ;
if ( cl_pos - > host_client_id = = mei_hdr - > host_addr & &
cl_pos - > me_client_id = = mei_hdr - > me_addr )
break ;
}
if ( & cl_pos - > link = = & dev - > file_list ) {
dev_dbg ( & dev - > pdev - > dev , " corrupted message header \n " ) ;
ret = - EBADMSG ;
goto end ;
}
}
if ( ( ( * slots ) * sizeof ( u32 ) ) < mei_hdr - > length ) {
dev_dbg ( & dev - > pdev - > dev ,
" we can't read the message slots =%08x. \n " ,
* slots ) ;
/* we can't read the message */
ret = - ERANGE ;
goto end ;
}
/* decide where to read the message too */
if ( ! mei_hdr - > host_addr ) {
dev_dbg ( & dev - > pdev - > dev , " call mei_irq_thread_read_bus_message. \n " ) ;
2012-12-25 19:06:07 +02:00
mei_hbm_dispatch ( dev , mei_hdr ) ;
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " end mei_irq_thread_read_bus_message. \n " ) ;
} else if ( mei_hdr - > host_addr = = dev - > iamthif_cl . host_client_id & &
( MEI_FILE_CONNECTED = = dev - > iamthif_cl . state ) & &
( dev - > iamthif_state = = MEI_IAMTHIF_READING ) ) {
dev_dbg ( & dev - > pdev - > dev , " call mei_irq_thread_read_iamthif_message. \n " ) ;
2012-12-25 19:06:00 +02:00
dev_dbg ( & dev - > pdev - > dev , MEI_HDR_FMT , MEI_HDR_PRM ( mei_hdr ) ) ;
2012-11-01 21:17:15 +02:00
ret = mei_amthif_irq_read_message ( cmpl_list , dev , mei_hdr ) ;
2011-05-15 13:43:42 +03:00
if ( ret )
goto end ;
} else {
dev_dbg ( & dev - > pdev - > dev , " call mei_irq_thread_read_client_message. \n " ) ;
ret = mei_irq_thread_read_client_message ( cmpl_list ,
dev , mei_hdr ) ;
if ( ret )
goto end ;
}
/* reset the number of slots and header */
* slots = mei_count_full_read_slots ( dev ) ;
dev - > rd_msg_hdr = 0 ;
if ( * slots = = - EOVERFLOW ) {
/* overflow - reset */
dev_dbg ( & dev - > pdev - > dev , " resetting due to slots overflow. \n " ) ;
/* set the event since message has been read */
ret = - ERANGE ;
goto end ;
}
end :
return ret ;
}
/**
2013-02-06 14:06:42 +02:00
* mei_irq_write_handler - dispatch write requests
* after irq received
2011-05-15 13:43:42 +03:00
*
* @ dev : the device structure
2012-11-18 15:13:18 +02:00
* @ cmpl_list : An instance of our list structure
2011-05-15 13:43:42 +03:00
*
* returns 0 on success , < 0 on failure .
*/
2013-03-11 18:27:02 +02:00
int mei_irq_write_handler ( struct mei_device * dev , struct mei_cl_cb * cmpl_list )
2011-05-15 13:43:42 +03:00
{
struct mei_cl * cl ;
2011-11-27 21:43:34 +02:00
struct mei_cl_cb * pos = NULL , * next = NULL ;
2012-10-15 12:06:48 +02:00
struct mei_cl_cb * list ;
2012-11-18 15:13:18 +02:00
s32 slots ;
2011-05-15 13:43:42 +03:00
int ret ;
2013-02-06 14:06:41 +02:00
if ( ! mei_hbuf_is_ready ( dev ) ) {
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " host buffer is not empty. \n " ) ;
return 0 ;
}
2012-11-18 15:13:18 +02:00
slots = mei_hbuf_empty_slots ( dev ) ;
if ( slots < = 0 )
2012-06-19 09:13:36 +03:00
return - EMSGSIZE ;
2011-05-15 13:43:42 +03:00
/* complete all waiting for write CB */
dev_dbg ( & dev - > pdev - > dev , " complete all waiting for write cb. \n " ) ;
list = & dev - > write_waiting_list ;
2012-10-15 12:06:48 +02:00
list_for_each_entry_safe ( pos , next , & list - > list , list ) {
2012-11-11 17:37:59 +02:00
cl = pos - > cl ;
2011-11-27 21:43:34 +02:00
if ( cl = = NULL )
continue ;
cl - > status = 0 ;
2012-10-15 12:06:48 +02:00
list_del ( & pos - > list ) ;
2011-11-27 21:43:34 +02:00
if ( MEI_WRITING = = cl - > writing_state & &
2012-11-11 17:38:00 +02:00
pos - > fop_type = = MEI_FOP_WRITE & &
cl ! = & dev - > iamthif_cl ) {
2012-07-04 19:24:54 +03:00
dev_dbg ( & dev - > pdev - > dev , " MEI WRITE COMPLETE \n " ) ;
2011-11-27 21:43:34 +02:00
cl - > writing_state = MEI_WRITE_COMPLETE ;
2012-10-15 12:06:48 +02:00
list_add_tail ( & pos - > list , & cmpl_list - > list ) ;
2011-11-27 21:43:34 +02:00
}
if ( cl = = & dev - > iamthif_cl ) {
dev_dbg ( & dev - > pdev - > dev , " check iamthif flow control. \n " ) ;
if ( dev - > iamthif_flow_control_pending ) {
2012-11-18 15:13:18 +02:00
ret = mei_amthif_irq_read ( dev , & slots ) ;
2011-11-27 21:43:34 +02:00
if ( ret )
return ret ;
2011-05-15 13:43:42 +03:00
}
}
}
2012-08-16 19:39:43 +03:00
if ( dev - > wd_state = = MEI_WD_STOPPING ) {
dev - > wd_state = MEI_WD_IDLE ;
2011-05-15 13:43:42 +03:00
wake_up_interruptible ( & dev - > wait_stop_wd ) ;
}
2012-11-18 15:13:15 +02:00
if ( dev - > wr_ext_msg . hdr . length ) {
mei_write_message ( dev , & dev - > wr_ext_msg . hdr ,
2012-12-25 19:05:59 +02:00
dev - > wr_ext_msg . data ) ;
2012-11-18 15:13:18 +02:00
slots - = mei_data2slots ( dev - > wr_ext_msg . hdr . length ) ;
2012-11-18 15:13:15 +02:00
dev - > wr_ext_msg . hdr . length = 0 ;
2011-05-15 13:43:42 +03:00
}
2012-08-07 00:03:56 +03:00
if ( dev - > dev_state = = MEI_DEV_ENABLED ) {
2011-05-15 13:43:42 +03:00
if ( dev - > wd_pending & &
2013-01-08 23:07:14 +02:00
mei_cl_flow_ctrl_creds ( & dev - > wd_cl ) > 0 ) {
2011-05-15 13:43:42 +03:00
if ( mei_wd_send ( dev ) )
dev_dbg ( & dev - > pdev - > dev , " wd send failed. \n " ) ;
2013-01-08 23:07:14 +02:00
else if ( mei_cl_flow_ctrl_reduce ( & dev - > wd_cl ) )
2012-07-04 19:24:54 +03:00
return - ENODEV ;
2011-05-15 13:43:42 +03:00
2011-05-25 17:28:22 +03:00
dev - > wd_pending = false ;
2011-05-15 13:43:42 +03:00
2012-08-16 19:39:43 +03:00
if ( dev - > wd_state = = MEI_WD_RUNNING )
2012-11-18 15:13:18 +02:00
slots - = mei_data2slots ( MEI_WD_START_MSG_SIZE ) ;
2012-07-04 19:24:50 +03:00
else
2012-11-18 15:13:18 +02:00
slots - = mei_data2slots ( MEI_WD_STOP_MSG_SIZE ) ;
2011-05-15 13:43:42 +03:00
}
}
/* complete control write list CB */
2011-11-27 21:43:33 +02:00
dev_dbg ( & dev - > pdev - > dev , " complete control write list cb. \n " ) ;
2012-10-15 12:06:48 +02:00
list_for_each_entry_safe ( pos , next , & dev - > ctrl_wr_list . list , list ) {
2012-11-11 17:37:59 +02:00
cl = pos - > cl ;
2011-11-27 21:43:33 +02:00
if ( ! cl ) {
2012-10-15 12:06:48 +02:00
list_del ( & pos - > list ) ;
2011-11-27 21:43:33 +02:00
return - ENODEV ;
}
2012-11-11 17:38:00 +02:00
switch ( pos - > fop_type ) {
case MEI_FOP_CLOSE :
2011-11-27 21:43:33 +02:00
/* send disconnect message */
2012-11-18 15:13:18 +02:00
ret = _mei_irq_thread_close ( dev , & slots , pos ,
cl , cmpl_list ) ;
2011-11-27 21:43:33 +02:00
if ( ret )
return ret ;
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:33 +02:00
break ;
2012-11-11 17:38:00 +02:00
case MEI_FOP_READ :
2011-11-27 21:43:33 +02:00
/* send flow control message */
2012-11-18 15:13:18 +02:00
ret = _mei_irq_thread_read ( dev , & slots , pos ,
cl , cmpl_list ) ;
2011-11-27 21:43:33 +02:00
if ( ret )
return ret ;
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:33 +02:00
break ;
2012-11-11 17:38:00 +02:00
case MEI_FOP_IOCTL :
2011-11-27 21:43:33 +02:00
/* connect message */
2013-01-08 23:07:14 +02:00
if ( mei_cl_is_other_connecting ( cl ) )
2011-11-27 21:43:33 +02:00
continue ;
2012-11-18 15:13:18 +02:00
ret = _mei_irq_thread_ioctl ( dev , & slots , pos ,
cl , cmpl_list ) ;
2011-11-27 21:43:33 +02:00
if ( ret )
return ret ;
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:33 +02:00
break ;
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:33 +02:00
default :
BUG ( ) ;
2011-05-15 13:43:42 +03:00
}
2011-11-27 21:43:33 +02:00
2011-05-15 13:43:42 +03:00
}
/* complete write list CB */
2011-11-27 21:43:34 +02:00
dev_dbg ( & dev - > pdev - > dev , " complete write list cb. \n " ) ;
2012-10-15 12:06:48 +02:00
list_for_each_entry_safe ( pos , next , & dev - > write_list . list , list ) {
2012-11-11 17:37:59 +02:00
cl = pos - > cl ;
2011-11-27 21:43:34 +02:00
if ( cl = = NULL )
continue ;
2013-01-08 23:07:14 +02:00
if ( mei_cl_flow_ctrl_creds ( cl ) < = 0 ) {
2012-11-18 15:13:19 +02:00
dev_dbg ( & dev - > pdev - > dev ,
" No flow control credentials for client %d, not sending. \n " ,
cl - > host_client_id ) ;
continue ;
}
2011-11-27 21:43:34 +02:00
2012-11-18 15:13:19 +02:00
if ( cl = = & dev - > iamthif_cl )
2012-11-18 15:13:18 +02:00
ret = mei_amthif_irq_write_complete ( dev , & slots ,
2012-11-18 15:13:17 +02:00
pos , cmpl_list ) ;
2012-11-18 15:13:19 +02:00
else
ret = mei_irq_thread_write_complete ( dev , & slots , pos ,
cmpl_list ) ;
if ( ret )
return ret ;
2011-11-27 21:43:34 +02:00
2011-05-15 13:43:42 +03:00
}
return 0 ;
}
/**
* mei_timer - timer function .
*
* @ work : pointer to the work_struct structure
*
* NOTE : This function is called by timer interrupt work
*/
2011-09-07 09:03:13 +03:00
void mei_timer ( struct work_struct * work )
2011-05-15 13:43:42 +03:00
{
unsigned long timeout ;
struct mei_cl * cl_pos = NULL ;
struct mei_cl * cl_next = NULL ;
struct mei_cl_cb * cb_pos = NULL ;
struct mei_cl_cb * cb_next = NULL ;
struct mei_device * dev = container_of ( work ,
2011-09-07 09:03:13 +03:00
struct mei_device , timer_work . work ) ;
2011-05-15 13:43:42 +03:00
mutex_lock ( & dev - > device_lock ) ;
2012-08-07 00:03:56 +03:00
if ( dev - > dev_state ! = MEI_DEV_ENABLED ) {
if ( dev - > dev_state = = MEI_DEV_INIT_CLIENTS ) {
2011-05-15 13:43:42 +03:00
if ( dev - > init_clients_timer ) {
if ( - - dev - > init_clients_timer = = 0 ) {
dev_dbg ( & dev - > pdev - > dev , " IMEI reset due to init clients timeout ,init clients state = %d. \n " ,
dev - > init_clients_state ) ;
mei_reset ( dev , 1 ) ;
}
}
}
goto out ;
}
/*** connect/disconnect timeouts ***/
list_for_each_entry_safe ( cl_pos , cl_next , & dev - > file_list , link ) {
if ( cl_pos - > timer_count ) {
if ( - - cl_pos - > timer_count = = 0 ) {
dev_dbg ( & dev - > pdev - > dev , " HECI reset due to connect/disconnect timeout. \n " ) ;
mei_reset ( dev , 1 ) ;
goto out ;
}
}
}
if ( dev - > iamthif_stall_timer ) {
if ( - - dev - > iamthif_stall_timer = = 0 ) {
2012-01-25 23:14:56 +09:00
dev_dbg ( & dev - > pdev - > dev , " resetting because of hang to amthi. \n " ) ;
2011-05-15 13:43:42 +03:00
mei_reset ( dev , 1 ) ;
dev - > iamthif_msg_buf_size = 0 ;
dev - > iamthif_msg_buf_index = 0 ;
2011-05-25 17:28:22 +03:00
dev - > iamthif_canceled = false ;
dev - > iamthif_ioctl = true ;
2011-05-15 13:43:42 +03:00
dev - > iamthif_state = MEI_IAMTHIF_IDLE ;
dev - > iamthif_timer = 0 ;
2012-10-09 16:50:20 +02:00
mei_io_cb_free ( dev - > iamthif_current_cb ) ;
dev - > iamthif_current_cb = NULL ;
2011-05-15 13:43:42 +03:00
dev - > iamthif_file_object = NULL ;
2012-11-01 21:17:15 +02:00
mei_amthif_run_next_cmd ( dev ) ;
2011-05-15 13:43:42 +03:00
}
}
if ( dev - > iamthif_timer ) {
timeout = dev - > iamthif_timer +
2012-11-01 21:17:14 +02:00
mei_secs_to_jiffies ( MEI_IAMTHIF_READ_TIMER ) ;
2011-05-15 13:43:42 +03:00
dev_dbg ( & dev - > pdev - > dev , " dev->iamthif_timer = %ld \n " ,
dev - > iamthif_timer ) ;
dev_dbg ( & dev - > pdev - > dev , " timeout = %ld \n " , timeout ) ;
dev_dbg ( & dev - > pdev - > dev , " jiffies = %ld \n " , jiffies ) ;
if ( time_after ( jiffies , timeout ) ) {
/*
* User didn ' t read the AMTHI data on time ( 15 sec )
* freeing AMTHI for other requests
*/
dev_dbg ( & dev - > pdev - > dev , " freeing AMTHI for other requests \n " ) ;
2012-11-11 17:37:58 +02:00
list_for_each_entry_safe ( cb_pos , cb_next ,
& dev - > amthif_rd_complete_list . list , list ) {
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:34 +02:00
cl_pos = cb_pos - > file_object - > private_data ;
2011-05-15 13:43:42 +03:00
2011-11-27 21:43:34 +02:00
/* Finding the AMTHI entry. */
if ( cl_pos = = & dev - > iamthif_cl )
2012-10-15 12:06:48 +02:00
list_del ( & cb_pos - > list ) ;
2011-05-15 13:43:42 +03:00
}
2012-10-09 16:50:20 +02:00
mei_io_cb_free ( dev - > iamthif_current_cb ) ;
dev - > iamthif_current_cb = NULL ;
2011-05-15 13:43:42 +03:00
dev - > iamthif_file_object - > private_data = NULL ;
dev - > iamthif_file_object = NULL ;
dev - > iamthif_timer = 0 ;
2012-11-01 21:17:15 +02:00
mei_amthif_run_next_cmd ( dev ) ;
2011-05-15 13:43:42 +03:00
}
}
out :
2011-12-13 23:39:34 +02:00
schedule_delayed_work ( & dev - > timer_work , 2 * HZ ) ;
mutex_unlock ( & dev - > device_lock ) ;
2011-05-15 13:43:42 +03:00
}