[PATCH v2] scsi: tcmu: avoid cmd/qfull timers updated whenever a new cmd comes

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



From: Xiubo Li <xiubli@xxxxxxxxxx>

Currently there has one cmd timeout timer and one qfull timer for
each udev, and whenever there has any new coming cmd it will update
the cmd timer or qfull timer. And for some corner case the timers
are always working only for the ringbuffer's and full queue's newest
cmd. That's to say the timer won't be fired even there has one cmd
stuck for a very long time and the deadline is reached.

This fix will keep the cmd/qfull timers to be pended for the oldest
cmd in ringbuffer and full queue, and will update them with the next
cmd's deadline only when the old cmd's deadline is reached or removed
from the ringbuffer and full queue.

Signed-off-by: Xiubo Li <xiubli@xxxxxxxxxx>
---

Changed in V2:
- Add qfull timer
- Addressed all the comments from Mike, thanks.

 drivers/target/target_core_user.c | 52 ++++++++++++++++++++++++++-----
 1 file changed, 44 insertions(+), 8 deletions(-)

diff --git a/drivers/target/target_core_user.c b/drivers/target/target_core_user.c
index 9cd404acdb82..99e73dfb0d2e 100644
--- a/drivers/target/target_core_user.c
+++ b/drivers/target/target_core_user.c
@@ -159,9 +159,11 @@ struct tcmu_dev {
 
 	struct timer_list cmd_timer;
 	unsigned int cmd_time_out;
+	unsigned long cmd_next_deadline;
 
 	struct timer_list qfull_timer;
 	int qfull_time_out;
+	unsigned long qfull_next_deadline;
 
 	struct list_head timedout_entry;
 
@@ -915,7 +917,9 @@ static int tcmu_setup_cmd_timer(struct tcmu_cmd *tcmu_cmd, unsigned int tmo,
 		return 0;
 
 	tcmu_cmd->deadline = round_jiffies_up(jiffies + msecs_to_jiffies(tmo));
-	mod_timer(timer, tcmu_cmd->deadline);
+	if (!timer_pending(timer))
+		mod_timer(timer, tcmu_cmd->deadline);
+
 	return 0;
 }
 
@@ -1197,6 +1201,7 @@ static void tcmu_handle_completion(struct tcmu_cmd *cmd, struct tcmu_cmd_entry *
 static unsigned int tcmu_handle_completions(struct tcmu_dev *udev)
 {
 	struct tcmu_mailbox *mb;
+	struct tcmu_cmd *cmd;
 	int handled = 0;
 
 	if (test_bit(TCMU_DEV_BIT_BROKEN, &udev->flags)) {
@@ -1210,7 +1215,6 @@ static unsigned int tcmu_handle_completions(struct tcmu_dev *udev)
 	while (udev->cmdr_last_cleaned != READ_ONCE(mb->cmd_tail)) {
 
 		struct tcmu_cmd_entry *entry = (void *) mb + CMDR_OFF + udev->cmdr_last_cleaned;
-		struct tcmu_cmd *cmd;
 
 		tcmu_flush_dcache_range(entry, sizeof(*entry));
 
@@ -1252,6 +1256,22 @@ static unsigned int tcmu_handle_completions(struct tcmu_dev *udev)
 			    tcmu_global_max_blocks)
 				schedule_delayed_work(&tcmu_unmap_work, 0);
 		}
+	} else if (udev->cmd_time_out) {
+		unsigned long deadline;
+		bool is_running;
+		int i;
+
+		deadline = round_jiffies_up(jiffies + msecs_to_jiffies(UINT_MAX));
+		udev->cmd_next_deadline = deadline;
+		idr_for_each_entry(&udev->commands, cmd, i) {
+			is_running = list_empty(&cmd->cmdr_queue_entry);
+			if (is_running && udev->cmd_next_deadline > cmd->deadline)
+				udev->cmd_next_deadline = cmd->deadline;
+		}
+		if (udev->cmd_next_deadline != deadline)
+			mod_timer(&udev->cmd_timer, udev->cmd_next_deadline);
+		else
+			del_timer(&udev->cmd_timer);
 	}
 
 	return handled;
@@ -1261,18 +1281,23 @@ static int tcmu_check_expired_cmd(int id, void *p, void *data)
 {
 	struct tcmu_cmd *cmd = p;
 	struct tcmu_dev *udev = cmd->tcmu_dev;
+	bool is_running = list_empty(&cmd->cmdr_queue_entry);
+	struct se_cmd *se_cmd = cmd->se_cmd;
 	u8 scsi_status;
-	struct se_cmd *se_cmd;
-	bool is_running;
 
 	if (test_bit(TCMU_CMD_BIT_EXPIRED, &cmd->flags))
 		return 0;
 
-	if (!time_after(jiffies, cmd->deadline))
+	if (!time_after(jiffies, cmd->deadline)) {
+		if (is_running) {
+			if (udev->cmd_next_deadline > cmd->deadline)
+				udev->cmd_next_deadline = cmd->deadline;
+		} else {
+			if (udev->qfull_next_deadline > cmd->deadline)
+				udev->qfull_next_deadline = cmd->deadline;
+		}
 		return 0;
-
-	is_running = list_empty(&cmd->cmdr_queue_entry);
-	se_cmd = cmd->se_cmd;
+	}
 
 	if (is_running) {
 		/*
@@ -2661,11 +2686,22 @@ static void check_timedout_devices(void)
 	list_splice_init(&timed_out_udevs, &devs);
 
 	list_for_each_entry_safe(udev, tmp_dev, &devs, timedout_entry) {
+		unsigned long max_deadline;
+
 		list_del_init(&udev->timedout_entry);
 		spin_unlock_bh(&timed_out_udevs_lock);
 
 		mutex_lock(&udev->cmdr_lock);
+		max_deadline = round_jiffies_up(jiffies + msecs_to_jiffies(UINT_MAX));
+		udev->cmd_next_deadline = max_deadline;
+		udev->qfull_next_deadline = max_deadline;
 		idr_for_each(&udev->commands, tcmu_check_expired_cmd, NULL);
+
+		if (udev->cmd_next_deadline != max_deadline)
+			mod_timer(&udev->cmd_timer, udev->cmd_next_deadline);
+		if (udev->qfull_next_deadline != max_deadline)
+			mod_timer(&udev->qfull_timer, udev->qfull_next_deadline);
+
 		mutex_unlock(&udev->cmdr_lock);
 
 		spin_lock_bh(&timed_out_udevs_lock);
-- 
2.19.0.216.g2d3b1c576




[Index of Archives]     [Linux SCSI]     [Kernel Newbies]     [Linux SCSI Target Infrastructure]     [Share Photos]     [IDE]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux ATA RAID]     [Linux IIO]     [Device Mapper]

  Powered by Linux