2011-05-15 13:43:44 +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:44 +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/sched.h>
# include <linux/wait.h>
# include <linux/delay.h>
2012-12-25 19:06:03 +02:00
# include <linux/mei.h>
2011-05-15 13:43:44 +03:00
# include "mei_dev.h"
2013-03-11 18:27:03 +02:00
# include "hbm.h"
2013-01-08 23:07:14 +02:00
# include "client.h"
2011-05-15 13:43:44 +03:00
2012-08-07 00:03:56 +03:00
const char * mei_dev_state_str ( int state )
{
# define MEI_DEV_STATE(state) case MEI_DEV_##state: return #state
switch ( state ) {
MEI_DEV_STATE ( INITIALIZING ) ;
MEI_DEV_STATE ( INIT_CLIENTS ) ;
MEI_DEV_STATE ( ENABLED ) ;
MEI_DEV_STATE ( RESETING ) ;
MEI_DEV_STATE ( DISABLED ) ;
MEI_DEV_STATE ( POWER_DOWN ) ;
MEI_DEV_STATE ( POWER_UP ) ;
default :
return " unkown " ;
}
# undef MEI_DEV_STATE
}
2013-02-06 14:06:40 +02:00
void mei_device_init ( struct mei_device * dev )
2011-05-15 13:43:44 +03:00
{
/* setup our list array */
INIT_LIST_HEAD ( & dev - > file_list ) ;
mutex_init ( & dev - > device_lock ) ;
2013-03-11 18:27:03 +02:00
init_waitqueue_head ( & dev - > wait_hw_ready ) ;
2011-05-15 13:43:44 +03:00
init_waitqueue_head ( & dev - > wait_recvd_msg ) ;
init_waitqueue_head ( & dev - > wait_stop_wd ) ;
2012-08-07 00:03:56 +03:00
dev - > dev_state = MEI_DEV_INITIALIZING ;
2011-06-06 10:44:34 +03:00
mei_io_list_init ( & dev - > read_list ) ;
mei_io_list_init ( & dev - > write_list ) ;
mei_io_list_init ( & dev - > write_waiting_list ) ;
mei_io_list_init ( & dev - > ctrl_wr_list ) ;
mei_io_list_init ( & dev - > ctrl_rd_list ) ;
2011-05-15 13:43:44 +03:00
}
/**
* mei_hw_init - initializes host and fw to start work .
*
* @ dev : the device structure
*
* returns 0 on success , < 0 on failure .
*/
int mei_hw_init ( struct mei_device * dev )
{
2013-01-08 23:07:31 +02:00
int ret = 0 ;
2011-05-15 13:43:44 +03:00
mutex_lock ( & dev - > device_lock ) ;
/* acknowledge interrupt and stop interupts */
2012-12-25 19:06:06 +02:00
mei_clear_interrupts ( dev ) ;
2011-05-15 13:43:44 +03:00
2013-01-08 23:07:31 +02:00
mei_hw_config ( dev ) ;
2012-06-25 23:46:27 +03:00
2011-05-25 17:28:22 +03:00
dev - > recvd_msg = false ;
2011-05-15 13:43:44 +03:00
dev_dbg ( & dev - > pdev - > dev , " reset in start the mei device. \n " ) ;
mei_reset ( dev , 1 ) ;
/* wait for ME to turn on ME_RDY */
if ( ! dev - > recvd_msg ) {
mutex_unlock ( & dev - > device_lock ) ;
2013-01-08 23:07:31 +02:00
ret = wait_event_interruptible_timeout ( dev - > wait_recvd_msg ,
2012-11-01 21:17:14 +02:00
dev - > recvd_msg ,
mei_secs_to_jiffies ( MEI_INTEROP_TIMEOUT ) ) ;
2011-05-15 13:43:44 +03:00
mutex_lock ( & dev - > device_lock ) ;
}
2013-01-08 23:07:31 +02:00
if ( ret < = 0 & & ! dev - > recvd_msg ) {
2012-08-07 00:03:56 +03:00
dev - > dev_state = MEI_DEV_DISABLED ;
2011-05-15 13:43:44 +03:00
dev_dbg ( & dev - > pdev - > dev ,
" wait_event_interruptible_timeout failed "
" on wait for ME to turn on ME_RDY. \n " ) ;
2013-01-08 23:07:31 +02:00
goto err ;
2011-05-15 13:43:44 +03:00
}
2013-01-08 23:07:31 +02:00
if ( ! mei_host_is_ready ( dev ) ) {
dev_err ( & dev - > pdev - > dev , " host is not ready. \n " ) ;
goto err ;
}
2011-05-15 13:43:44 +03:00
2013-02-06 14:06:41 +02:00
if ( ! mei_hw_is_ready ( dev ) ) {
2013-01-08 23:07:31 +02:00
dev_err ( & dev - > pdev - > dev , " ME is not ready. \n " ) ;
goto err ;
2011-05-15 13:43:44 +03:00
}
if ( dev - > version . major_version ! = HBM_MAJOR_VERSION | |
dev - > version . minor_version ! = HBM_MINOR_VERSION ) {
dev_dbg ( & dev - > pdev - > dev , " MEI start failed. \n " ) ;
2013-01-08 23:07:31 +02:00
goto err ;
2011-05-15 13:43:44 +03:00
}
2011-05-25 17:28:22 +03:00
dev - > recvd_msg = false ;
2011-05-15 13:43:44 +03:00
dev_dbg ( & dev - > pdev - > dev , " link layer has been established. \n " ) ;
mutex_unlock ( & dev - > device_lock ) ;
2013-01-08 23:07:31 +02:00
return 0 ;
err :
dev_err ( & dev - > pdev - > dev , " link layer initialization failed. \n " ) ;
dev - > dev_state = MEI_DEV_DISABLED ;
mutex_unlock ( & dev - > device_lock ) ;
return - ENODEV ;
2011-05-15 13:43:44 +03:00
}
/**
* mei_reset - resets host and fw .
*
* @ dev : the device structure
* @ interrupts_enabled : if interrupt should be enabled after reset .
*/
void mei_reset ( struct mei_device * dev , int interrupts_enabled )
{
bool unexpected ;
2012-08-07 00:03:56 +03:00
unexpected = ( dev - > dev_state ! = MEI_DEV_INITIALIZING & &
dev - > dev_state ! = MEI_DEV_DISABLED & &
dev - > dev_state ! = MEI_DEV_POWER_DOWN & &
dev - > dev_state ! = MEI_DEV_POWER_UP ) ;
2011-05-15 13:43:44 +03:00
mei_hw_reset ( dev , interrupts_enabled ) ;
2012-08-07 00:03:56 +03:00
if ( dev - > dev_state ! = MEI_DEV_INITIALIZING ) {
if ( dev - > dev_state ! = MEI_DEV_DISABLED & &
dev - > dev_state ! = MEI_DEV_POWER_DOWN )
dev - > dev_state = MEI_DEV_RESETING ;
2011-05-15 13:43:44 +03:00
2013-02-06 14:06:44 +02:00
mei_cl_all_disconnect ( dev ) ;
2011-05-15 13:43:44 +03:00
/* remove entry if already in list */
2012-11-11 17:38:03 +02:00
dev_dbg ( & dev - > pdev - > dev , " remove iamthif and wd from the file list. \n " ) ;
2013-01-08 23:07:14 +02:00
mei_cl_unlink ( & dev - > wd_cl ) ;
2013-01-08 23:07:22 +02:00
if ( dev - > open_handle_count > 0 )
dev - > open_handle_count - - ;
2013-01-08 23:07:14 +02:00
mei_cl_unlink ( & dev - > iamthif_cl ) ;
2013-01-08 23:07:22 +02:00
if ( dev - > open_handle_count > 0 )
dev - > open_handle_count - - ;
2011-05-15 13:43:44 +03:00
2012-11-01 21:17:15 +02:00
mei_amthif_reset_params ( dev ) ;
2012-11-18 15:13:15 +02:00
memset ( & dev - > wr_ext_msg , 0 , sizeof ( dev - > wr_ext_msg ) ) ;
2011-05-15 13:43:44 +03:00
}
2011-06-06 10:44:33 +03:00
dev - > me_clients_num = 0 ;
2011-05-15 13:43:44 +03:00
dev - > rd_msg_hdr = 0 ;
2011-05-25 17:28:22 +03:00
dev - > wd_pending = false ;
2011-05-15 13:43:44 +03:00
if ( unexpected )
2012-08-07 00:03:56 +03:00
dev_warn ( & dev - > pdev - > dev , " unexpected reset: dev_state = %s \n " ,
mei_dev_state_str ( dev - > dev_state ) ) ;
2011-05-15 13:43:44 +03:00
2013-03-11 18:27:03 +02:00
if ( ! interrupts_enabled ) {
dev_dbg ( & dev - > pdev - > dev , " intr not enabled end of reset \n " ) ;
return ;
}
mei_hw_start ( dev ) ;
dev_dbg ( & dev - > pdev - > dev , " link is established start sending messages. \n " ) ;
/* link is established * start sending messages. */
dev - > dev_state = MEI_DEV_INIT_CLIENTS ;
mei_hbm_start_req ( dev ) ;
2013-02-06 14:06:44 +02:00
/* wake up all readings so they can be interrupted */
mei_cl_all_read_wakeup ( dev ) ;
2011-05-15 13:43:44 +03:00
/* remove all waiting requests */
2013-02-06 14:06:44 +02:00
mei_cl_all_write_clear ( dev ) ;
2011-05-15 13:43:44 +03:00
}
2013-03-10 13:56:08 +02:00
void mei_stop ( struct mei_device * dev )
{
dev_dbg ( & dev - > pdev - > dev , " stopping the device. \n " ) ;
mutex_lock ( & dev - > device_lock ) ;
cancel_delayed_work ( & dev - > timer_work ) ;
mei_wd_stop ( dev ) ;
dev - > dev_state = MEI_DEV_POWER_DOWN ;
mei_reset ( dev , 0 ) ;
mutex_unlock ( & dev - > device_lock ) ;
flush_scheduled_work ( ) ;
2013-03-27 16:58:26 +02:00
mei_watchdog_unregister ( dev ) ;
2013-03-10 13:56:08 +02:00
}
2011-05-15 13:43:44 +03:00
2012-11-18 15:13:20 +02:00
2011-05-15 13:43:44 +03:00