mei: limit the number of consecutive resets
authorTomas Winkler <tomas.winkler@intel.com>
Sat, 11 Jan 2014 22:36:10 +0000 (00:36 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 13 Jan 2014 22:57:21 +0000 (14:57 -0800)
give up reseting after 3 unsuccessful tries

Signed-off-by: Tomas Winkler <tomas.winkler@intel.com>
Signed-off-by: Alexander Usyskin <alexander.usyskin@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/misc/mei/client.c
drivers/misc/mei/init.c
drivers/misc/mei/mei_dev.h

index 4f268a37772034a3c6914ba0cc9300d71644405f..1ee2b9492a82527d508b4396647f75823956e28c 100644 (file)
@@ -370,6 +370,7 @@ void mei_host_client_init(struct work_struct *work)
        }
 
        dev->dev_state = MEI_DEV_ENABLED;
+       dev->reset_count = 0;
 
        mutex_unlock(&dev->device_lock);
 }
index 059133d8cacafc6b24aa2b0cf9b169c5bda46837..cdd31c2a2a2bf095e827758e3d71567d80557a5f 100644 (file)
@@ -89,6 +89,13 @@ int mei_reset(struct mei_device *dev)
        interrupts_enabled = state != MEI_DEV_POWER_DOWN;
        dev->dev_state = MEI_DEV_RESETTING;
 
+       dev->reset_count++;
+       if (dev->reset_count > MEI_MAX_CONSEC_RESET) {
+               dev_err(&dev->pdev->dev, "reset: reached maximal consecutive resets: disabling the device\n");
+               dev->dev_state = MEI_DEV_DISABLED;
+               return -ENODEV;
+       }
+
        ret = mei_hw_reset(dev, interrupts_enabled);
        /* fall through and remove the sw state even if hw reset has failed */
 
@@ -169,6 +176,7 @@ int mei_start(struct mei_device *dev)
        dev_dbg(&dev->pdev->dev, "reset in start the mei device.\n");
 
        dev->dev_state = MEI_DEV_INITIALIZING;
+       dev->reset_count = 0;
        mei_reset(dev);
 
        if (dev->dev_state == MEI_DEV_DISABLED) {
@@ -224,6 +232,7 @@ int mei_restart(struct mei_device *dev)
        mei_clear_interrupts(dev);
 
        dev->dev_state = MEI_DEV_POWER_UP;
+       dev->reset_count = 0;
 
        err = mei_reset(dev);
 
@@ -285,6 +294,7 @@ void mei_device_init(struct mei_device *dev)
        init_waitqueue_head(&dev->wait_recvd_msg);
        init_waitqueue_head(&dev->wait_stop_wd);
        dev->dev_state = MEI_DEV_INITIALIZING;
+       dev->reset_count = 0;
 
        mei_io_list_init(&dev->read_list);
        mei_io_list_init(&dev->write_list);
index a617c8494b7014f6716935368114b1bdfacaf8df..f7de95b4cdd907ebfbf241b46f73d95cdc9cb3d7 100644 (file)
@@ -60,6 +60,11 @@ extern const uuid_le mei_wd_guid;
  */
 #define MEI_CLIENTS_MAX 256
 
+/*
+ * maximum number of consecutive resets
+ */
+#define MEI_MAX_CONSEC_RESET  3
+
 /*
  * Number of File descriptors/handles
  * that can be opened to the driver.
@@ -327,6 +332,7 @@ struct mei_cl_device {
 /**
  * struct mei_device -  MEI private device struct
 
+ * @reset_count - limits the number of consecutive resets
  * @hbm_state - state of host bus message protocol
  * @mem_addr - mem mapped base register address
 
@@ -370,6 +376,7 @@ struct mei_device {
        /*
         * mei device  states
         */
+       unsigned long reset_count;
        enum mei_dev_state dev_state;
        enum mei_hbm_state hbm_state;
        u16 init_clients_timer;