Commit de2eb4d5 authored by Jeff Skirvin's avatar Jeff Skirvin Committed by Dan Williams

isci: End the RNC resumption wait when the RNC is destroyed.

While the RNC is suspended for I/O cleanup, the remote device can be
stopped and the RNC setup for destruction.  These changes accomodate that
case in the abort path.
Signed-off-by: default avatarJeff Skirvin <jeffrey.d.skirvin@intel.com>
Signed-off-by: default avatarDan Williams <dan.j.williams@intel.com>
parent 6c6aacbb
...@@ -340,6 +340,11 @@ static inline struct isci_host *dev_to_ihost(struct domain_device *dev) ...@@ -340,6 +340,11 @@ static inline struct isci_host *dev_to_ihost(struct domain_device *dev)
return dev->port->ha->lldd_ha; return dev->port->ha->lldd_ha;
} }
static inline struct isci_host *idev_to_ihost(struct isci_remote_device *idev)
{
return dev_to_ihost(idev->domain_dev);
}
/* we always use protocol engine group zero */ /* we always use protocol engine group zero */
#define ISCI_PEG 0 #define ISCI_PEG 0
......
...@@ -1368,27 +1368,40 @@ static void isci_remote_device_resume_from_abort_complete(void *cbparam) ...@@ -1368,27 +1368,40 @@ static void isci_remote_device_resume_from_abort_complete(void *cbparam)
wake_up(&ihost->eventq); wake_up(&ihost->eventq);
} }
static bool isci_remote_device_test_resume_done(
struct isci_host *ihost,
struct isci_remote_device *idev)
{
unsigned long flags;
bool done;
spin_lock_irqsave(&ihost->scic_lock, flags);
done = !test_bit(IDEV_ABORT_PATH_RESUME_PENDING, &idev->flags)
|| test_bit(IDEV_STOP_PENDING, &idev->flags)
|| sci_remote_node_context_is_being_destroyed(&idev->rnc);
spin_unlock_irqrestore(&ihost->scic_lock, flags);
return done;
}
void isci_remote_device_wait_for_resume_from_abort( void isci_remote_device_wait_for_resume_from_abort(
struct isci_host *ihost, struct isci_host *ihost,
struct isci_remote_device *idev) struct isci_remote_device *idev)
{ {
dev_dbg(scirdev_to_dev(idev), "%s: starting resume wait: %p\n", dev_dbg(&ihost->pdev->dev, "%s: starting resume wait: %p\n",
__func__, idev); __func__, idev);
#define MAX_RESUME_MSECS 10000 #define MAX_RESUME_MSECS 10000
if (!wait_event_timeout(ihost->eventq, if (!wait_event_timeout(ihost->eventq,
(!test_bit(IDEV_ABORT_PATH_RESUME_PENDING, isci_remote_device_test_resume_done(ihost, idev),
&idev->flags) msecs_to_jiffies(MAX_RESUME_MSECS))) {
|| test_bit(IDEV_STOP_PENDING, &idev->flags)),
msecs_to_jiffies(MAX_RESUME_MSECS))) {
dev_warn(scirdev_to_dev(idev), "%s: #### Timeout waiting for " dev_warn(&ihost->pdev->dev, "%s: #### Timeout waiting for "
"resume: %p\n", __func__, idev); "resume: %p\n", __func__, idev);
} }
clear_bit(IDEV_ABORT_PATH_RESUME_PENDING, &idev->flags); clear_bit(IDEV_ABORT_PATH_RESUME_PENDING, &idev->flags);
dev_dbg(scirdev_to_dev(idev), "%s: resume wait done: %p\n", dev_dbg(&ihost->pdev->dev, "%s: resume wait done: %p\n",
__func__, idev); __func__, idev);
} }
...@@ -1414,7 +1427,7 @@ enum sci_status isci_remote_device_resume_from_abort( ...@@ -1414,7 +1427,7 @@ enum sci_status isci_remote_device_resume_from_abort(
idev, isci_remote_device_resume_from_abort_complete, idev, isci_remote_device_resume_from_abort_complete,
idev); idev);
spin_unlock_irqrestore(&ihost->scic_lock, flags); spin_unlock_irqrestore(&ihost->scic_lock, flags);
if (!destroyed) if (!destroyed && (status == SCI_SUCCESS))
isci_remote_device_wait_for_resume_from_abort(ihost, idev); isci_remote_device_wait_for_resume_from_abort(ihost, idev);
else else
clear_bit(IDEV_ABORT_PATH_RESUME_PENDING, &idev->flags); clear_bit(IDEV_ABORT_PATH_RESUME_PENDING, &idev->flags);
......
...@@ -190,9 +190,13 @@ static void sci_remote_node_context_setup_to_destroy( ...@@ -190,9 +190,13 @@ static void sci_remote_node_context_setup_to_destroy(
scics_sds_remote_node_context_callback callback, scics_sds_remote_node_context_callback callback,
void *callback_parameter) void *callback_parameter)
{ {
struct isci_host *ihost = idev_to_ihost(rnc_to_dev(sci_rnc));
sci_rnc->destination_state = RNC_DEST_FINAL; sci_rnc->destination_state = RNC_DEST_FINAL;
sci_rnc->user_callback = callback; sci_rnc->user_callback = callback;
sci_rnc->user_cookie = callback_parameter; sci_rnc->user_cookie = callback_parameter;
wake_up(&ihost->eventq);
} }
/** /**
......
...@@ -229,8 +229,8 @@ int sci_remote_node_context_is_safe_to_abort( ...@@ -229,8 +229,8 @@ int sci_remote_node_context_is_safe_to_abort(
static inline bool sci_remote_node_context_is_being_destroyed( static inline bool sci_remote_node_context_is_being_destroyed(
struct sci_remote_node_context *sci_rnc) struct sci_remote_node_context *sci_rnc)
{ {
return ((sci_rnc->sm.current_state_id == SCI_RNC_INVALIDATING) return (sci_rnc->destination_state == RNC_DEST_FINAL)
&& (sci_rnc->destination_state == RNC_DEST_FINAL)) || ((sci_rnc->sm.current_state_id == SCI_RNC_INITIAL)
|| (sci_rnc->sm.current_state_id == SCI_RNC_INITIAL); && (sci_rnc->destination_state == RNC_DEST_UNSPECIFIED));
} }
#endif /* _SCIC_SDS_REMOTE_NODE_CONTEXT_H_ */ #endif /* _SCIC_SDS_REMOTE_NODE_CONTEXT_H_ */
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