2011-05-15 14:43:47 +04:00
/*
*
* Intel Management Engine Interface ( Intel MEI ) Linux driver
2012-02-09 21:25:53 +04:00
* Copyright ( c ) 2003 - 2012 , Intel Corporation .
2011-05-15 14:43:47 +04: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/kernel.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/device.h>
# include <linux/pci.h>
# include <linux/sched.h>
2011-09-07 10:03:09 +04:00
# include <linux/watchdog.h>
2011-05-15 14:43:47 +04:00
# include "mei_dev.h"
# include "hw.h"
# include "interface.h"
2012-05-09 17:38:59 +04:00
# include <linux/mei.h>
2011-05-15 14:43:47 +04:00
static const u8 mei_start_wd_params [ ] = { 0x02 , 0x12 , 0x13 , 0x10 } ;
static const u8 mei_stop_wd_params [ ] = { 0x02 , 0x02 , 0x14 , 0x10 } ;
const u8 mei_wd_state_independence_msg [ 3 ] [ 4 ] = {
{ 0x05 , 0x02 , 0x51 , 0x10 } ,
{ 0x05 , 0x02 , 0x52 , 0x10 } ,
{ 0x07 , 0x02 , 0x01 , 0x10 }
} ;
2011-12-22 20:50:50 +04:00
/*
* AMT Watchdog Device
*/
# define INTEL_AMT_WATCHDOG_ID "INTCAMT"
2011-05-15 14:43:47 +04:00
/* UUIDs for AMT F/W clients */
const uuid_le mei_wd_guid = UUID_LE ( 0x05B79A6F , 0x4628 , 0x4D7F , 0x89 ,
0x9D , 0xA9 , 0x15 , 0x14 , 0xCB ,
0x32 , 0xAB ) ;
2012-03-19 19:58:43 +04:00
static void mei_wd_set_start_timeout ( struct mei_device * dev , u16 timeout )
2011-05-15 14:43:47 +04:00
{
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev , " wd: set timeout=%d. \n " , timeout ) ;
2012-08-16 20:39:42 +04:00
memcpy ( dev - > wd_data , mei_start_wd_params , MEI_WD_HDR_SIZE ) ;
memcpy ( dev - > wd_data + MEI_WD_HDR_SIZE , & timeout , sizeof ( u16 ) ) ;
2011-05-15 14:43:47 +04:00
}
/**
2012-07-04 20:24:51 +04:00
* mei_wd_host_init - connect to the watchdog client
2011-05-15 14:43:47 +04:00
*
* @ dev : the device structure
2012-04-04 00:34:58 +04:00
* returns - ENENT if wd client cannot be found
* - EIO if write has failed
2012-07-04 20:24:51 +04:00
* 0 on success
2011-05-15 14:43:47 +04:00
*/
2012-04-04 00:34:58 +04:00
int mei_wd_host_init ( struct mei_device * dev )
2011-05-15 14:43:47 +04:00
{
2011-05-25 18:28:21 +04:00
mei_cl_init ( & dev - > wd_cl , dev ) ;
2011-05-15 14:43:47 +04:00
/* look for WD client and connect to it */
dev - > wd_cl . state = MEI_FILE_DISCONNECTED ;
2012-08-16 20:39:42 +04:00
dev - > wd_timeout = MEI_WD_DEFAULT_TIMEOUT ;
2011-05-15 14:43:47 +04:00
2011-09-07 10:03:15 +04:00
/* find ME WD client */
2012-07-23 15:05:39 +04:00
mei_me_cl_update_filext ( dev , & dev - > wd_cl ,
2011-09-07 10:03:15 +04:00
& mei_wd_guid , MEI_WD_HOST_CLIENT_ID ) ;
2011-05-15 14:43:47 +04:00
2012-04-04 00:34:58 +04:00
dev_dbg ( & dev - > pdev - > dev , " wd: check client \n " ) ;
if ( MEI_FILE_CONNECTING ! = dev - > wd_cl . state ) {
dev_info ( & dev - > pdev - > dev , " wd: failed to find the client \n " ) ;
return - ENOENT ;
}
if ( mei_connect ( dev , & dev - > wd_cl ) ) {
dev_err ( & dev - > pdev - > dev , " wd: failed to connect to the client \n " ) ;
dev - > wd_cl . state = MEI_FILE_DISCONNECTED ;
dev - > wd_cl . host_client_id = 0 ;
return - EIO ;
2011-05-15 14:43:47 +04:00
}
2012-04-04 00:34:58 +04:00
dev - > wd_cl . timer_count = CONNECT_TIMEOUT ;
2011-09-07 10:03:07 +04:00
2012-04-04 00:34:58 +04:00
return 0 ;
2011-05-15 14:43:47 +04:00
}
/**
* mei_wd_send - sends watch dog message to fw .
*
* @ dev : the device structure
*
* returns 0 if success ,
* - EIO when message send fails
* - EINVAL when invalid message is to be sent
*/
int mei_wd_send ( struct mei_device * dev )
{
struct mei_msg_hdr * mei_hdr ;
mei_hdr = ( struct mei_msg_hdr * ) & dev - > wr_msg_buf [ 0 ] ;
mei_hdr - > host_addr = dev - > wd_cl . host_client_id ;
mei_hdr - > me_addr = dev - > wd_cl . me_client_id ;
mei_hdr - > msg_complete = 1 ;
mei_hdr - > reserved = 0 ;
2012-08-16 20:39:42 +04:00
if ( ! memcmp ( dev - > wd_data , mei_start_wd_params , MEI_WD_HDR_SIZE ) )
mei_hdr - > length = MEI_WD_START_MSG_SIZE ;
else if ( ! memcmp ( dev - > wd_data , mei_stop_wd_params , MEI_WD_HDR_SIZE ) )
mei_hdr - > length = MEI_WD_STOP_MSG_SIZE ;
2011-05-15 14:43:47 +04:00
else
return - EINVAL ;
2012-03-14 16:39:42 +04:00
return mei_write_message ( dev , mei_hdr , dev - > wd_data , mei_hdr - > length ) ;
2011-05-15 14:43:47 +04:00
}
2011-09-07 10:03:17 +04:00
/**
* mei_wd_stop - sends watchdog stop message to fw .
*
* @ dev : the device structure
* @ preserve : indicate if to keep the timeout value
*
* returns 0 if success ,
* - EIO when message send fails
* - EINVAL when invalid message is to be sent
*/
2011-05-15 14:43:47 +04:00
int mei_wd_stop ( struct mei_device * dev , bool preserve )
{
int ret ;
u16 wd_timeout = dev - > wd_timeout ;
2011-09-07 10:03:13 +04:00
cancel_delayed_work ( & dev - > timer_work ) ;
2011-05-15 14:43:47 +04:00
if ( dev - > wd_cl . state ! = MEI_FILE_CONNECTED | | ! dev - > wd_timeout )
return 0 ;
dev - > wd_timeout = 0 ;
2012-08-16 20:39:42 +04:00
memcpy ( dev - > wd_data , mei_stop_wd_params , MEI_WD_STOP_MSG_SIZE ) ;
2011-05-25 18:28:22 +04:00
dev - > stop = true ;
2011-05-15 14:43:47 +04:00
ret = mei_flow_ctrl_creds ( dev , & dev - > wd_cl ) ;
if ( ret < 0 )
goto out ;
if ( ret & & dev - > mei_host_buffer_is_empty ) {
ret = 0 ;
2011-05-25 18:28:22 +04:00
dev - > mei_host_buffer_is_empty = false ;
2011-05-15 14:43:47 +04:00
if ( ! mei_wd_send ( dev ) ) {
ret = mei_flow_ctrl_reduce ( dev , & dev - > wd_cl ) ;
if ( ret )
goto out ;
} else {
2012-04-04 00:34:59 +04:00
dev_err ( & dev - > pdev - > dev , " wd: send stop failed \n " ) ;
2011-05-15 14:43:47 +04:00
}
2011-05-25 18:28:22 +04:00
dev - > wd_pending = false ;
2011-05-15 14:43:47 +04:00
} else {
2011-05-25 18:28:22 +04:00
dev - > wd_pending = true ;
2011-05-15 14:43:47 +04:00
}
2011-05-25 18:28:22 +04:00
dev - > wd_stopped = false ;
2011-05-15 14:43:47 +04:00
mutex_unlock ( & dev - > device_lock ) ;
ret = wait_event_interruptible_timeout ( dev - > wait_stop_wd ,
dev - > wd_stopped , 10 * HZ ) ;
mutex_lock ( & dev - > device_lock ) ;
2011-06-13 17:39:31 +04:00
if ( dev - > wd_stopped ) {
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev , " wd: stop completed ret=%d. \n " , ret ) ;
2011-06-13 17:39:31 +04:00
ret = 0 ;
} else {
if ( ! ret )
ret = - ETIMEDOUT ;
dev_warn ( & dev - > pdev - > dev ,
2012-04-04 00:34:59 +04:00
" wd: stop failed to complete ret=%d. \n " , ret ) ;
2011-06-13 17:39:31 +04:00
}
2011-05-15 14:43:47 +04:00
if ( preserve )
dev - > wd_timeout = wd_timeout ;
out :
return ret ;
}
2011-09-07 10:03:10 +04:00
/*
* mei_wd_ops_start - wd start command from the watchdog core .
*
* @ wd_dev - watchdog device struct
*
* returns 0 if success , negative errno code for failure
*/
static int mei_wd_ops_start ( struct watchdog_device * wd_dev )
{
int err = - ENODEV ;
struct mei_device * dev ;
dev = pci_get_drvdata ( mei_device ) ;
if ( ! dev )
return - ENODEV ;
mutex_lock ( & dev - > device_lock ) ;
2012-08-07 01:03:56 +04:00
if ( dev - > dev_state ! = MEI_DEV_ENABLED ) {
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev ,
2012-08-07 01:03:56 +04:00
" wd: dev_state != MEI_DEV_ENABLED dev_state = %s \n " ,
mei_dev_state_str ( dev - > dev_state ) ) ;
2011-09-07 10:03:10 +04:00
goto end_unlock ;
}
if ( dev - > wd_cl . state ! = MEI_FILE_CONNECTED ) {
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev ,
" MEI Driver is not connected to Watchdog Client \n " ) ;
2011-09-07 10:03:10 +04:00
goto end_unlock ;
}
2011-09-07 10:03:12 +04:00
mei_wd_set_start_timeout ( dev , dev - > wd_timeout ) ;
2011-09-07 10:03:10 +04:00
err = 0 ;
end_unlock :
mutex_unlock ( & dev - > device_lock ) ;
return err ;
}
/*
* mei_wd_ops_stop - wd stop command from the watchdog core .
*
* @ wd_dev - watchdog device struct
*
* returns 0 if success , negative errno code for failure
*/
static int mei_wd_ops_stop ( struct watchdog_device * wd_dev )
{
struct mei_device * dev ;
dev = pci_get_drvdata ( mei_device ) ;
if ( ! dev )
return - ENODEV ;
mutex_lock ( & dev - > device_lock ) ;
mei_wd_stop ( dev , false ) ;
mutex_unlock ( & dev - > device_lock ) ;
return 0 ;
}
2011-09-07 10:03:11 +04:00
/*
* mei_wd_ops_ping - wd ping command from the watchdog core .
*
* @ wd_dev - watchdog device struct
*
* returns 0 if success , negative errno code for failure
*/
static int mei_wd_ops_ping ( struct watchdog_device * wd_dev )
{
int ret = 0 ;
struct mei_device * dev ;
dev = pci_get_drvdata ( mei_device ) ;
if ( ! dev )
return - ENODEV ;
mutex_lock ( & dev - > device_lock ) ;
if ( dev - > wd_cl . state ! = MEI_FILE_CONNECTED ) {
2012-04-04 00:34:59 +04:00
dev_err ( & dev - > pdev - > dev , " wd: not connected. \n " ) ;
2011-09-07 10:03:11 +04:00
ret = - ENODEV ;
goto end ;
}
/* Check if we can send the ping to HW*/
if ( dev - > mei_host_buffer_is_empty & &
mei_flow_ctrl_creds ( dev , & dev - > wd_cl ) > 0 ) {
dev - > mei_host_buffer_is_empty = false ;
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev , " wd: sending ping \n " ) ;
2011-09-07 10:03:11 +04:00
if ( mei_wd_send ( dev ) ) {
2012-04-04 00:34:59 +04:00
dev_err ( & dev - > pdev - > dev , " wd: send failed. \n " ) ;
2011-09-07 10:03:11 +04:00
ret = - EIO ;
goto end ;
}
if ( mei_flow_ctrl_reduce ( dev , & dev - > wd_cl ) ) {
2012-04-04 00:34:59 +04:00
dev_err ( & dev - > pdev - > dev ,
" wd: mei_flow_ctrl_reduce() failed. \n " ) ;
2011-09-07 10:03:11 +04:00
ret = - EIO ;
goto end ;
}
} else {
dev - > wd_pending = true ;
}
end :
mutex_unlock ( & dev - > device_lock ) ;
return ret ;
}
2011-09-07 10:03:12 +04:00
/*
* mei_wd_ops_set_timeout - wd set timeout command from the watchdog core .
*
* @ wd_dev - watchdog device struct
* @ timeout - timeout value to set
*
* returns 0 if success , negative errno code for failure
*/
static int mei_wd_ops_set_timeout ( struct watchdog_device * wd_dev , unsigned int timeout )
{
struct mei_device * dev ;
dev = pci_get_drvdata ( mei_device ) ;
if ( ! dev )
return - ENODEV ;
/* Check Timeout value */
2012-08-16 20:39:42 +04:00
if ( timeout < MEI_WD_MIN_TIMEOUT | | timeout > MEI_WD_MAX_TIMEOUT )
2011-09-07 10:03:12 +04:00
return - EINVAL ;
mutex_lock ( & dev - > device_lock ) ;
dev - > wd_timeout = timeout ;
2012-02-29 23:20:58 +04:00
wd_dev - > timeout = timeout ;
2011-09-07 10:03:12 +04:00
mei_wd_set_start_timeout ( dev , dev - > wd_timeout ) ;
mutex_unlock ( & dev - > device_lock ) ;
return 0 ;
}
2011-09-07 10:03:10 +04:00
/*
* Watchdog Device structs
*/
2011-10-23 20:30:39 +04:00
static const struct watchdog_ops wd_ops = {
2011-09-07 10:03:10 +04:00
. owner = THIS_MODULE ,
. start = mei_wd_ops_start ,
. stop = mei_wd_ops_stop ,
2011-09-07 10:03:11 +04:00
. ping = mei_wd_ops_ping ,
2011-09-07 10:03:12 +04:00
. set_timeout = mei_wd_ops_set_timeout ,
2011-09-07 10:03:10 +04:00
} ;
2011-10-23 20:30:39 +04:00
static const struct watchdog_info wd_info = {
2011-09-07 10:03:10 +04:00
. identity = INTEL_AMT_WATCHDOG_ID ,
2012-08-16 20:39:41 +04:00
. options = WDIOF_KEEPALIVEPING |
WDIOF_SETTIMEOUT |
WDIOF_ALARMONLY ,
2011-09-07 10:03:10 +04:00
} ;
2012-04-02 21:32:38 +04:00
static struct watchdog_device amt_wd_dev = {
2011-09-07 10:03:10 +04:00
. info = & wd_info ,
. ops = & wd_ops ,
2012-08-16 20:39:42 +04:00
. timeout = MEI_WD_DEFAULT_TIMEOUT ,
. min_timeout = MEI_WD_MIN_TIMEOUT ,
. max_timeout = MEI_WD_MAX_TIMEOUT ,
2011-09-07 10:03:10 +04:00
} ;
2011-12-22 20:50:50 +04:00
void mei_watchdog_register ( struct mei_device * dev )
{
dev_dbg ( & dev - > pdev - > dev , " dev->wd_timeout =%d. \n " , dev - > wd_timeout ) ;
if ( watchdog_register_device ( & amt_wd_dev ) ) {
2012-04-04 00:34:59 +04:00
dev_err ( & dev - > pdev - > dev ,
" wd: unable to register watchdog device. \n " ) ;
2011-12-22 20:50:50 +04:00
dev - > wd_interface_reg = false ;
} else {
2012-04-04 00:34:59 +04:00
dev_dbg ( & dev - > pdev - > dev ,
" wd: successfully register watchdog interface. \n " ) ;
2011-12-22 20:50:50 +04:00
dev - > wd_interface_reg = true ;
}
}
void mei_watchdog_unregister ( struct mei_device * dev )
{
if ( dev - > wd_interface_reg )
watchdog_unregister_device ( & amt_wd_dev ) ;
dev - > wd_interface_reg = false ;
}