net/ice: handle AdminQ command by DCF
authorHaiyue Wang <haiyue.wang@intel.com>
Fri, 27 Mar 2020 02:56:38 +0000 (10:56 +0800)
committerFerruh Yigit <ferruh.yigit@intel.com>
Tue, 21 Apr 2020 11:57:05 +0000 (13:57 +0200)
The DCF (Device Config Function) splits the AdminQ command into two
parts: one is the descriptor of AdminQ command, the other is the buffer
of AdminQ command (the descriptor has BUF flag set). When both of them
are received by the PF, the PF will handle them as one command.

And also, the filled descriptor and buffer of the response will be sent
back to DCF one by one through the virtchnl from PF.

Signed-off-by: Haiyue Wang <haiyue.wang@intel.com>
Acked-by: Qi Zhang <qi.z.zhang@intel.com>
drivers/net/ice/ice_dcf.c
drivers/net/ice/ice_dcf.h

index 6ea68fe..baef5b8 100644 (file)
@@ -396,6 +396,71 @@ ret:
        return err;
 }
 
+int
+ice_dcf_send_aq_cmd(void *dcf_hw, struct ice_aq_desc *desc,
+                   void *buf, uint16_t buf_size)
+{
+       struct dcf_virtchnl_cmd desc_cmd, buff_cmd;
+       struct ice_dcf_hw *hw = dcf_hw;
+       int err = 0;
+       int i = 0;
+
+       if ((buf && !buf_size) || (!buf && buf_size) ||
+           buf_size > ICE_DCF_AQ_BUF_SZ)
+               return -EINVAL;
+
+       desc_cmd.v_op = VIRTCHNL_OP_DCF_CMD_DESC;
+       desc_cmd.req_msglen = sizeof(*desc);
+       desc_cmd.req_msg = (uint8_t *)desc;
+       desc_cmd.rsp_buflen = sizeof(*desc);
+       desc_cmd.rsp_msgbuf = (uint8_t *)desc;
+
+       if (buf == NULL)
+               return ice_dcf_execute_virtchnl_cmd(hw, &desc_cmd);
+
+       desc->flags |= rte_cpu_to_le_16(ICE_AQ_FLAG_BUF);
+
+       buff_cmd.v_op = VIRTCHNL_OP_DCF_CMD_BUFF;
+       buff_cmd.req_msglen = buf_size;
+       buff_cmd.req_msg = buf;
+       buff_cmd.rsp_buflen = buf_size;
+       buff_cmd.rsp_msgbuf = buf;
+
+       rte_spinlock_lock(&hw->vc_cmd_send_lock);
+       ice_dcf_vc_cmd_set(hw, &desc_cmd);
+       ice_dcf_vc_cmd_set(hw, &buff_cmd);
+
+       if (ice_dcf_vc_cmd_send(hw, &desc_cmd) ||
+           ice_dcf_vc_cmd_send(hw, &buff_cmd)) {
+               err = -1;
+               PMD_DRV_LOG(ERR, "fail to send OP_DCF_CMD_DESC/BUFF");
+               goto ret;
+       }
+
+       do {
+               if ((!desc_cmd.pending && !buff_cmd.pending) ||
+                   (!desc_cmd.pending && desc_cmd.v_ret != IAVF_SUCCESS) ||
+                   (!buff_cmd.pending && buff_cmd.v_ret != IAVF_SUCCESS))
+                       break;
+
+               rte_delay_ms(ICE_DCF_ARQ_CHECK_TIME);
+       } while (i++ < ICE_DCF_ARQ_MAX_RETRIES);
+
+       if (desc_cmd.v_ret != IAVF_SUCCESS || buff_cmd.v_ret != IAVF_SUCCESS) {
+               err = -1;
+               PMD_DRV_LOG(ERR,
+                           "No response (%d times) or return failure (desc: %d / buff: %d)",
+                           i, desc_cmd.v_ret, buff_cmd.v_ret);
+       }
+
+ret:
+       ice_dcf_aq_cmd_clear(hw, &desc_cmd);
+       ice_dcf_aq_cmd_clear(hw, &buff_cmd);
+       rte_spinlock_unlock(&hw->vc_cmd_send_lock);
+
+       return err;
+}
+
 int
 ice_dcf_init_hw(struct rte_eth_dev *eth_dev, struct ice_dcf_hw *hw)
 {
index f44c09d..99bd53b 100644 (file)
@@ -11,6 +11,7 @@
 #include <iavf_adminq_cmd.h>
 #include <iavf_type.h>
 
+#include "base/ice_type.h"
 #include "ice_logs.h"
 
 struct dcf_virtchnl_cmd {
@@ -45,7 +46,8 @@ struct ice_dcf_hw {
 
 int ice_dcf_execute_virtchnl_cmd(struct ice_dcf_hw *hw,
                                 struct dcf_virtchnl_cmd *cmd);
-
+int ice_dcf_send_aq_cmd(void *dcf_hw, struct ice_aq_desc *desc,
+                       void *buf, uint16_t buf_size);
 int ice_dcf_init_hw(struct rte_eth_dev *eth_dev, struct ice_dcf_hw *hw);
 void ice_dcf_uninit_hw(struct rte_eth_dev *eth_dev, struct ice_dcf_hw *hw);