Skip site navigation (1)Skip section navigation (2)
Date:      Wed, 7 Mar 2018 14:41:29 +0000 (UTC)
From:      Hans Petter Selasky <hselasky@FreeBSD.org>
To:        src-committers@freebsd.org, svn-src-all@freebsd.org, svn-src-head@freebsd.org
Subject:   svn commit: r330600 - in head/sys/dev/mlx5: . mlx5_core
Message-ID:  <201803071441.w27EfTIm049941@repo.freebsd.org>

next in thread | raw e-mail | index | archive | help
Author: hselasky
Date: Wed Mar  7 14:41:29 2018
New Revision: 330600
URL: https://svnweb.freebsd.org/changeset/base/330600

Log:
  Add timeout handle to commands with callback in mlx5core.
  
  The current implementation does not handle timeout in case of command
  with callback request, and this can lead to deadlock if the command
  doesn't get firmware response. Add delayed callback timeout work
  before posting the command to firmware. In case of real firmware
  command completion we will cancel the delayed work. In case of
  firmware command timeout the callback timeout handler will be called
  and it will simulate firmware completion with timeout error.
  
  linux commit 65ee67084589c1783a74b4a4a5db38d7264ec8b5
  
  MFC after:	1 week
  Sponsored by:	Mellanox Technologies

Modified:
  head/sys/dev/mlx5/driver.h
  head/sys/dev/mlx5/mlx5_core/mlx5_cmd.c

Modified: head/sys/dev/mlx5/driver.h
==============================================================================
--- head/sys/dev/mlx5/driver.h	Wed Mar  7 14:35:28 2018	(r330599)
+++ head/sys/dev/mlx5/driver.h	Wed Mar  7 14:41:29 2018	(r330600)
@@ -727,6 +727,7 @@ struct mlx5_cmd_work_ent {
 	void		       *uout;
 	int			uout_size;
 	mlx5_cmd_cbk_t		callback;
+        struct delayed_work     cb_timeout_work;
 	void		       *context;
 	int			idx;
 	struct completion	done;

Modified: head/sys/dev/mlx5/mlx5_core/mlx5_cmd.c
==============================================================================
--- head/sys/dev/mlx5/mlx5_core/mlx5_cmd.c	Wed Mar  7 14:35:28 2018	(r330599)
+++ head/sys/dev/mlx5/mlx5_core/mlx5_cmd.c	Wed Mar  7 14:41:29 2018	(r330600)
@@ -246,6 +246,7 @@ static void poll_timeout(struct mlx5_cmd_work_ent *ent
 
 static void free_cmd(struct mlx5_cmd_work_ent *ent)
 {
+        cancel_delayed_work_sync(&ent->cb_timeout_work);
 	kfree(ent);
 }
 
@@ -499,6 +500,30 @@ static void dump_command(struct mlx5_core_dev *dev,
 		pr_debug("\n");
 }
 
+static u16 msg_to_opcode(struct mlx5_cmd_msg *in)
+{
+        struct mlx5_inbox_hdr *hdr = (struct mlx5_inbox_hdr *)(in->first.data);
+
+        return be16_to_cpu(hdr->opcode);
+}
+
+static void cb_timeout_handler(struct work_struct *work)
+{
+        struct delayed_work *dwork = container_of(work, struct delayed_work,
+                                                  work);
+        struct mlx5_cmd_work_ent *ent = container_of(dwork,
+                                                     struct mlx5_cmd_work_ent,
+                                                     cb_timeout_work);
+        struct mlx5_core_dev *dev = container_of(ent->cmd, struct mlx5_core_dev,
+                                                 cmd);
+
+        ent->ret = -ETIMEDOUT;
+        mlx5_core_warn(dev, "%s(0x%x) timeout. Will cause a leak of a command resource\n",
+                       mlx5_command_str(msg_to_opcode(ent->in)),
+                       msg_to_opcode(ent->in));
+        mlx5_cmd_comp_handler(dev, 1UL << ent->idx);
+}
+
 static int set_internal_err_outbox(struct mlx5_core_dev *dev, u16 opcode,
 				   struct mlx5_outbox_hdr *hdr)
 {
@@ -731,6 +756,7 @@ static void cmd_work_handler(struct work_struct *work)
 	struct mlx5_cmd_work_ent *ent = container_of(work, struct mlx5_cmd_work_ent, work);
 	struct mlx5_cmd *cmd = ent->cmd;
 	struct mlx5_core_dev *dev = container_of(cmd, struct mlx5_core_dev, cmd);
+        unsigned long cb_timeout = msecs_to_jiffies(MLX5_CMD_TIMEOUT_MSEC);
 	struct mlx5_cmd_layout *lay;
 	struct semaphore *sem;
 
@@ -761,6 +787,9 @@ static void cmd_work_handler(struct work_struct *work)
 	dump_command(dev, ent, 1);
 	ent->ts1 = ktime_get_ns();
 	ent->busy = 0;
+        if (ent->callback)
+                schedule_delayed_work(&ent->cb_timeout_work, cb_timeout);
+
 	/* ring doorbell after the descriptor is valid */
 	mlx5_core_dbg(dev, "writing 0x%x to command doorbell\n", 1 << ent->idx);
 	/* make sure data is written to RAM */
@@ -805,13 +834,6 @@ static const char *deliv_status_to_str(u8 status)
 	}
 }
 
-static u16 msg_to_opcode(struct mlx5_cmd_msg *in)
-{
-	struct mlx5_inbox_hdr *hdr = (struct mlx5_inbox_hdr *)(in->first.data);
-
-	return be16_to_cpu(hdr->opcode);
-}
-
 static int wait_func(struct mlx5_core_dev *dev, struct mlx5_cmd_work_ent *ent)
 {
 	int timeout = msecs_to_jiffies(MLX5_CMD_TIMEOUT_MSEC);
@@ -867,6 +889,7 @@ static int mlx5_cmd_invoke(struct mlx5_core_dev *dev, 
 	if (!callback)
 		init_completion(&ent->done);
 
+        INIT_DELAYED_WORK(&ent->cb_timeout_work, cb_timeout_handler);
 	INIT_WORK(&ent->work, cmd_work_handler);
 	if (page_queue) {
 		cmd_work_handler(&ent->work);
@@ -1065,6 +1088,8 @@ void mlx5_cmd_comp_handler(struct mlx5_core_dev *dev, 
 		i = ffs(vector) - 1;
 		vector &= ~(1U << i);
 		ent = cmd->ent_arr[i];
+                if (ent->callback)
+                        cancel_delayed_work(&ent->cb_timeout_work);
 		ent->ts2 = ktime_get_ns();
 		memcpy(ent->out->first.data, ent->lay->out,
 		       sizeof(ent->lay->out));



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?201803071441.w27EfTIm049941>