mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-12-28 10:15:32 -05:00
mei: retry connect if interrupted by link reset
When device is in D3cold the connect message will wake device and cause link reset. Link reset flow cleans all queues and wakes all waiters. Retry the connect flow if connect is failed and link reset is detected. Signed-off-by: Alexander Usyskin <alexander.usyskin@intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Link: https://lore.kernel.org/r/20250918130435.3327400-4-alexander.usyskin@intel.com
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
bb29fc32ae
commit
2b5c4cb2c0
@@ -27,6 +27,8 @@
|
||||
#define MKHI_RCV_TIMEOUT 500 /* receive timeout in msec */
|
||||
#define MKHI_RCV_TIMEOUT_SLOW 10000 /* receive timeout in msec, slow FW */
|
||||
|
||||
#define MEI_LINK_RESET_WAIT_TIMEOUT_MSEC 500 /* Max wait timeout for link reset, in msec */
|
||||
|
||||
/*
|
||||
* FW page size for DMA allocations
|
||||
*/
|
||||
|
||||
@@ -400,6 +400,7 @@ void mei_device_init(struct mei_device *dev,
|
||||
init_waitqueue_head(&dev->wait_pg);
|
||||
init_waitqueue_head(&dev->wait_hbm_start);
|
||||
dev->dev_state = MEI_DEV_UNINITIALIZED;
|
||||
init_waitqueue_head(&dev->wait_dev_state);
|
||||
dev->reset_count = 0;
|
||||
|
||||
INIT_LIST_HEAD(&dev->write_list);
|
||||
@@ -442,5 +443,6 @@ void mei_device_init(struct mei_device *dev,
|
||||
dev->timeouts.hbm = mei_secs_to_jiffies(MEI_HBM_TIMEOUT);
|
||||
dev->timeouts.mkhi_recv = msecs_to_jiffies(MKHI_RCV_TIMEOUT);
|
||||
}
|
||||
dev->timeouts.link_reset_wait = msecs_to_jiffies(MEI_LINK_RESET_WAIT_TIMEOUT_MSEC);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mei_device_init);
|
||||
|
||||
@@ -423,6 +423,7 @@ static int mei_ioctl_connect_client(struct file *file,
|
||||
cl->state != MEI_FILE_DISCONNECTED)
|
||||
return -EBUSY;
|
||||
|
||||
retry:
|
||||
/* find ME client we're trying to connect to */
|
||||
me_cl = mei_me_cl_by_uuid(dev, in_client_uuid);
|
||||
if (!me_cl) {
|
||||
@@ -454,6 +455,28 @@ static int mei_ioctl_connect_client(struct file *file,
|
||||
|
||||
rets = mei_cl_connect(cl, me_cl, file);
|
||||
|
||||
if (rets && cl->status == -EFAULT &&
|
||||
(dev->dev_state == MEI_DEV_RESETTING ||
|
||||
dev->dev_state == MEI_DEV_INIT_CLIENTS)) {
|
||||
/* in link reset, wait for it completion */
|
||||
mutex_unlock(&dev->device_lock);
|
||||
rets = wait_event_interruptible_timeout(dev->wait_dev_state,
|
||||
dev->dev_state == MEI_DEV_ENABLED,
|
||||
dev->timeouts.link_reset_wait);
|
||||
mutex_lock(&dev->device_lock);
|
||||
if (rets < 0) {
|
||||
if (signal_pending(current))
|
||||
rets = -EINTR;
|
||||
goto end;
|
||||
}
|
||||
if (dev->dev_state != MEI_DEV_ENABLED) {
|
||||
rets = -ETIME;
|
||||
goto end;
|
||||
}
|
||||
mei_me_cl_put(me_cl);
|
||||
goto retry;
|
||||
}
|
||||
|
||||
end:
|
||||
mei_me_cl_put(me_cl);
|
||||
return rets;
|
||||
@@ -1120,6 +1143,8 @@ void mei_set_devstate(struct mei_device *dev, enum mei_dev_state state)
|
||||
|
||||
dev->dev_state = state;
|
||||
|
||||
wake_up_interruptible_all(&dev->wait_dev_state);
|
||||
|
||||
if (!dev->cdev)
|
||||
return;
|
||||
|
||||
|
||||
@@ -466,6 +466,7 @@ struct mei_dev_timeouts {
|
||||
unsigned int d0i3; /* D0i3 set/unset max response time, in jiffies */
|
||||
unsigned long hbm; /* HBM operation timeout, in jiffies */
|
||||
unsigned long mkhi_recv; /* receive timeout, in jiffies */
|
||||
unsigned long link_reset_wait; /* link reset wait timeout, in jiffies */
|
||||
};
|
||||
|
||||
/**
|
||||
@@ -496,6 +497,7 @@ struct mei_dev_timeouts {
|
||||
*
|
||||
* @reset_count : number of consecutive resets
|
||||
* @dev_state : device state
|
||||
* @wait_dev_state: wait queue for device state change
|
||||
* @hbm_state : state of host bus message protocol
|
||||
* @pxp_mode : PXP device mode
|
||||
* @init_clients_timer : HBM init handshake timeout
|
||||
@@ -588,6 +590,7 @@ struct mei_device {
|
||||
*/
|
||||
unsigned long reset_count;
|
||||
enum mei_dev_state dev_state;
|
||||
wait_queue_head_t wait_dev_state;
|
||||
enum mei_hbm_state hbm_state;
|
||||
enum mei_dev_pxp_mode pxp_mode;
|
||||
u16 init_clients_timer;
|
||||
|
||||
Reference in New Issue
Block a user