Commit 8d52af67 authored by Tomas Winkler's avatar Tomas Winkler Committed by Greg Kroah-Hartman

mei: speed up the power down flow

When mei driver is powering down due to suspend or shutdown
it will iterate over the mei client bus and disconnect
each client device attached in turn.
The power down flow consist of the link rest, which causes all clients
get disconnected at once, hence the individual disconnection
can be omitted and significantly reduce power down flow.
Signed-off-by: default avatarTomas Winkler <tomas.winkler@intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 2fc10246
...@@ -543,14 +543,20 @@ int mei_cldev_disable(struct mei_cl_device *cldev) ...@@ -543,14 +543,20 @@ int mei_cldev_disable(struct mei_cl_device *cldev)
mutex_lock(&bus->device_lock); mutex_lock(&bus->device_lock);
if (!mei_cl_is_connected(cl)) { if (!mei_cl_is_connected(cl)) {
dev_dbg(bus->dev, "Already disconnected"); dev_dbg(bus->dev, "Already disconnected\n");
err = 0;
goto out;
}
if (bus->dev_state == MEI_DEV_POWER_DOWN) {
dev_dbg(bus->dev, "Device is powering down don't botther with disconnection\n");
err = 0; err = 0;
goto out; goto out;
} }
err = mei_cl_disconnect(cl); err = mei_cl_disconnect(cl);
if (err < 0) if (err < 0)
dev_err(bus->dev, "Could not disconnect from the ME client"); dev_err(bus->dev, "Could not disconnect from the ME client\n");
out: out:
/* Flush queues and remove any pending read */ /* Flush queues and remove any pending read */
......
...@@ -1260,7 +1260,9 @@ irqreturn_t mei_me_irq_thread_handler(int irq, void *dev_id) ...@@ -1260,7 +1260,9 @@ irqreturn_t mei_me_irq_thread_handler(int irq, void *dev_id)
if (rets == -ENODATA) if (rets == -ENODATA)
break; break;
if (rets && dev->dev_state != MEI_DEV_RESETTING) { if (rets &&
(dev->dev_state != MEI_DEV_RESETTING ||
dev->dev_state != MEI_DEV_POWER_DOWN)) {
dev_err(dev->dev, "mei_irq_read_handler ret = %d.\n", dev_err(dev->dev, "mei_irq_read_handler ret = %d.\n",
rets); rets);
schedule_work(&dev->reset_work); schedule_work(&dev->reset_work);
......
...@@ -1127,7 +1127,9 @@ irqreturn_t mei_txe_irq_thread_handler(int irq, void *dev_id) ...@@ -1127,7 +1127,9 @@ irqreturn_t mei_txe_irq_thread_handler(int irq, void *dev_id)
if (test_and_clear_bit(TXE_INTR_OUT_DB_BIT, &hw->intr_cause)) { if (test_and_clear_bit(TXE_INTR_OUT_DB_BIT, &hw->intr_cause)) {
/* Read from TXE */ /* Read from TXE */
rets = mei_irq_read_handler(dev, &cmpl_list, &slots); rets = mei_irq_read_handler(dev, &cmpl_list, &slots);
if (rets && dev->dev_state != MEI_DEV_RESETTING) { if (rets &&
(dev->dev_state != MEI_DEV_RESETTING ||
dev->dev_state != MEI_DEV_POWER_DOWN)) {
dev_err(dev->dev, dev_err(dev->dev,
"mei_irq_read_handler ret = %d.\n", rets); "mei_irq_read_handler ret = %d.\n", rets);
......
...@@ -310,6 +310,9 @@ void mei_stop(struct mei_device *dev) ...@@ -310,6 +310,9 @@ void mei_stop(struct mei_device *dev)
{ {
dev_dbg(dev->dev, "stopping the device.\n"); dev_dbg(dev->dev, "stopping the device.\n");
mutex_lock(&dev->device_lock);
dev->dev_state = MEI_DEV_POWER_DOWN;
mutex_unlock(&dev->device_lock);
mei_cl_bus_remove_devices(dev); mei_cl_bus_remove_devices(dev);
mei_cancel_work(dev); mei_cancel_work(dev);
...@@ -319,7 +322,6 @@ void mei_stop(struct mei_device *dev) ...@@ -319,7 +322,6 @@ void mei_stop(struct mei_device *dev)
mutex_lock(&dev->device_lock); mutex_lock(&dev->device_lock);
dev->dev_state = MEI_DEV_POWER_DOWN;
mei_reset(dev); mei_reset(dev);
/* move device to disabled state unconditionally */ /* move device to disabled state unconditionally */
dev->dev_state = MEI_DEV_DISABLED; dev->dev_state = MEI_DEV_DISABLED;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment