Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

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

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
Loading
Loading
Loading
Loading
+5 −0
Original line number Original line Diff line number Diff line
@@ -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


+21 −8
Original line number Original line Diff line number Diff line
@@ -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)
				|| test_bit(IDEV_STOP_PENDING, &idev->flags)),
				msecs_to_jiffies(MAX_RESUME_MSECS))) {
				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(
			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);
+4 −0
Original line number Original line Diff line number Diff line
@@ -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);
}
}


/**
/**
+3 −3
Original line number Original line Diff line number Diff line
@@ -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_ */