From: Paul Greenwalt <paul.greenw...@intel.com>

Tx Packet Pacing (TXPP) functionality is in addition to the Tx LAN queue
flow, so it has a separate Tx Time queue context.

Add Tx Time queue context data structures, and two new related AQCs.
- ice_aq_set_txtimeq (opcode 0x0C35) to set Tx Time queue(s) context
- ice_aq_operate_txtimeq (opcode 0x0C37) to enable/disable Tx Time
  queue(s)

Signed-off-by: Paul Greenwalt <paul.greenw...@intel.com>
Signed-off-by: Soumyadeep Hore <soumyadeep.h...@intel.com>
---
 drivers/net/ice/base/ice_adminq_cmd.h | 55 +++++++++++++++
 drivers/net/ice/base/ice_common.c     | 96 +++++++++++++++++++++++++++
 drivers/net/ice/base/ice_common.h     |  9 +++
 drivers/net/ice/base/ice_lan_tx_rx.h  | 37 +++++++++++
 4 files changed, 197 insertions(+)

diff --git a/drivers/net/ice/base/ice_adminq_cmd.h 
b/drivers/net/ice/base/ice_adminq_cmd.h
index 3ec207927b..1fa8bbad29 100644
--- a/drivers/net/ice/base/ice_adminq_cmd.h
+++ b/drivers/net/ice/base/ice_adminq_cmd.h
@@ -2975,6 +2975,55 @@ struct ice_aqc_move_txqs_data {
        struct ice_aqc_move_txqs_elem txqs[STRUCT_HACK_VAR_LEN];
 };
 
+/* Set Tx Time LAN Queue (indirect 0x0C35) */
+struct ice_aqc_set_txtimeqs {
+       __le16 q_id;
+       __le16 q_amount;
+       u8 reserved[4];
+       __le32 addr_high;
+       __le32 addr_low;
+};
+
+/* This is the descriptor of each queue entry for the Set Tx Time Queue
+ * command (0x0C35). Only used within struct ice_aqc_set_txtime_qgrp.
+ */
+struct ice_aqc_set_txtimeqs_perq {
+       u8 reserved[4];
+       u8 txtime_ctx[25];
+       u8 reserved1[3];
+};
+
+/* The format of the command buffer for Set Tx Time Queue (0x0C35)
+ * is an array of the following structs. Please note that the length of
+ * each struct ice_aqc_set_txtime_qgrp is variable due to the variable
+ * number of queues in each group!
+ */
+struct ice_aqc_set_txtime_qgrp {
+       u8 reserved[8];
+       struct ice_aqc_set_txtimeqs_perq txtimeqs[STRUCT_HACK_VAR_LEN];
+};
+
+/* Operate Tx Time Queue (indirect 0x0C37) */
+struct ice_aqc_ena_dis_txtimeqs {
+       __le16 q_id;
+       __le16 q_amount;
+       u8 cmd_type;
+#define ICE_AQC_TXTIME_CMD_TYPE_S      0
+#define ICE_AQC_TXTIME_CMD_TYPE_M      (0x1 << ICE_AQC_Q_CMD_TYPE_S)
+#define ICE_AQC_TXTIME_CMD_TYPE_Q_ENA  1
+       u8 reserved[3];
+       __le32 addr_high;
+       __le32 addr_low;
+};
+#pragma pack(1)
+
+struct ice_aqc_ena_dis_txtime_qgrp {
+       u8 reserved[5];
+       __le16 fail_txtime_q;
+       u8 reserved1[1];
+};
+#pragma pack()
+
 /* Download Package (indirect 0x0C40) */
 /* Also used for Update Package (indirect 0x0C41 and 0x0C42) */
 struct ice_aqc_download_pkg {
@@ -3297,6 +3346,8 @@ struct ice_aq_desc {
                struct ice_aqc_add_txqs add_txqs;
                struct ice_aqc_dis_txqs dis_txqs;
                struct ice_aqc_move_txqs move_txqs;
+               struct ice_aqc_set_txtimeqs set_txtimeqs;
+               struct ice_aqc_ena_dis_txtimeqs operate_txtimeqs;
                struct ice_aqc_txqs_cleanup txqs_cleanup;
                struct ice_aqc_add_get_update_free_vsi vsi_cmd;
                struct ice_aqc_add_update_free_vsi_resp add_update_free_vsi_res;
@@ -3576,6 +3627,10 @@ enum ice_adminq_opc {
        ice_aqc_opc_txqs_cleanup                        = 0x0C31,
        ice_aqc_opc_move_recfg_txqs                     = 0x0C32,
 
+       /* Tx Time queue commands */
+       ice_aqc_opc_set_txtimeqs                        = 0x0C35,
+       ice_aqc_opc_ena_dis_txtimeqs            = 0x0C37,
+
        /* package commands */
        ice_aqc_opc_download_pkg                        = 0x0C40,
        ice_aqc_opc_upload_section                      = 0x0C41,
diff --git a/drivers/net/ice/base/ice_common.c 
b/drivers/net/ice/base/ice_common.c
index c8047ca59f..fce9b070cf 100644
--- a/drivers/net/ice/base/ice_common.c
+++ b/drivers/net/ice/base/ice_common.c
@@ -1660,6 +1660,31 @@ ice_clear_tx_drbell_q_ctx(struct ice_hw *hw, u32 
tx_drbell_q_index)
        return 0;
 }
 
+/* Tx time Queue Context */
+const struct ice_ctx_ele ice_txtime_ctx_info[] = {
+                                   /* Field                    Width   LSB */
+       ICE_CTX_STORE(ice_txtime_ctx, base,                     57,     0),
+       ICE_CTX_STORE(ice_txtime_ctx, pf_num,                   3,      57),
+       ICE_CTX_STORE(ice_txtime_ctx, vmvf_num,                 10,     60),
+       ICE_CTX_STORE(ice_txtime_ctx, vmvf_type,                2,      70),
+       ICE_CTX_STORE(ice_txtime_ctx, src_vsi,                  10,     72),
+       ICE_CTX_STORE(ice_txtime_ctx, cpuid,                    8,      82),
+       ICE_CTX_STORE(ice_txtime_ctx, tphrd_desc,               1,      90),
+       ICE_CTX_STORE(ice_txtime_ctx, qlen,                     13,     91),
+       ICE_CTX_STORE(ice_txtime_ctx, timer_num,                3,      104),
+       ICE_CTX_STORE(ice_txtime_ctx, txtime_ena_q,             1,      107),
+       ICE_CTX_STORE(ice_txtime_ctx, drbell_mode_32,           1,      108),
+       ICE_CTX_STORE(ice_txtime_ctx, ts_res,                   4,      109),
+       ICE_CTX_STORE(ice_txtime_ctx, ts_round_type,            2,      113),
+       ICE_CTX_STORE(ice_txtime_ctx, ts_pacing_slot,           3,      115),
+       ICE_CTX_STORE(ice_txtime_ctx, merging_ena,              1,      118),
+       ICE_CTX_STORE(ice_txtime_ctx, ts_fetch_prof_id,         4,      119),
+       ICE_CTX_STORE(ice_txtime_ctx, ts_fetch_cache_line_aln_thld, 4,  123),
+       ICE_CTX_STORE(ice_txtime_ctx, tx_pipe_delay_mode,       1,      127),
+       ICE_CTX_STORE(ice_txtime_ctx, int_q_state,              70,     128),
+       { 0 }
+};
+
 /* Sideband Queue command wrappers */
 
 /**
@@ -4729,6 +4754,77 @@ ice_aq_move_recfg_lan_txq(struct ice_hw *hw, u8 num_qs, 
bool is_move,
        return status;
 }
 
+/**
+ * ice_aq_set_txtimeq - set Tx time queues
+ * @hw: pointer to the hardware structure
+ * @txtimeq: first Tx time queue id to configure
+ * @q_count: number of queues to configure
+ * @txtime_qg: queue group to be set
+ * @buf_size: size of buffer for indirect command
+ * @cd: pointer to command details structure or NULL
+ *
+ * Set Tx Time queue (0x0C35)
+ */
+enum ice_status
+ice_aq_set_txtimeq(struct ice_hw *hw, u16 txtimeq, u8 q_count,
+                  struct ice_aqc_set_txtime_qgrp *txtime_qg, u16 buf_size,
+                  struct ice_sq_cd *cd)
+{
+       struct ice_aqc_set_txtimeqs *cmd;
+       struct ice_aq_desc desc;
+       u16 size;
+       ice_debug(hw, ICE_DBG_TRACE, "%s\n", __func__);
+       cmd = &desc.params.set_txtimeqs;
+       ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_set_txtimeqs);
+       if (!txtime_qg)
+               return ICE_ERR_PARAM;
+       if (txtimeq > ICE_TXTIME_MAX_QUEUE || q_count < 1 ||
+           q_count > ICE_SET_TXTIME_MAX_Q_AMOUNT)
+               return ICE_ERR_PARAM;
+       size = ice_struct_size(txtime_qg, txtimeqs, q_count);
+       if (buf_size != size)
+               return ICE_ERR_PARAM;
+       desc.flags |= CPU_TO_LE16(ICE_AQ_FLAG_RD);
+       cmd->q_id = CPU_TO_LE16(txtimeq);
+       cmd->q_amount = CPU_TO_LE16(q_count);
+       return ice_aq_send_cmd(hw, &desc, txtime_qg, buf_size, cd);
+}
+
+/**
+ * ice_aq_ena_dis_txtimeq - enable/disable Tx time queue
+ * @hw: pointer to the hardware structure
+ * @txtimeq: first Tx time queue id to configure
+ * @q_count: number of queues to configure
+ * @q_ena: enable/disable Tx time queue
+ * @txtime_qg: holds the first Tx time queue that failed enable/disable on
+ * response
+ * @cd: pointer to command details structure or NULL
+ *
+ * Enable/disable Tx Time queue (0x0C37)
+ */
+enum ice_status
+ice_aq_ena_dis_txtimeq(struct ice_hw *hw, u16 txtimeq, u16 q_count, bool q_ena,
+                      struct ice_aqc_ena_dis_txtime_qgrp *txtime_qg,
+                      struct ice_sq_cd *cd)
+{
+       struct ice_aqc_ena_dis_txtimeqs *cmd;
+       struct ice_aq_desc desc;
+       ice_debug(hw, ICE_DBG_TRACE, "%s\n", __func__);
+       cmd = &desc.params.operate_txtimeqs;
+       ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_ena_dis_txtimeqs);
+       if (!txtime_qg)
+               return ICE_ERR_PARAM;
+       if (txtimeq > ICE_TXTIME_MAX_QUEUE || q_count < 1 ||
+           q_count > ICE_OP_TXTIME_MAX_Q_AMOUNT)
+               return ICE_ERR_PARAM;
+       desc.flags |= CPU_TO_LE16(ICE_AQ_FLAG_RD);
+       cmd->q_id = CPU_TO_LE16(txtimeq);
+       cmd->q_amount = CPU_TO_LE16(q_count);
+       if (q_ena)
+               cmd->cmd_type |= ICE_AQC_TXTIME_CMD_TYPE_Q_ENA;
+       return ice_aq_send_cmd(hw, &desc, txtime_qg, sizeof(*txtime_qg), cd);
+}
+
 /* End of FW Admin Queue command wrappers */
 
 /**
diff --git a/drivers/net/ice/base/ice_common.h 
b/drivers/net/ice/base/ice_common.h
index 71b1f34d0a..f7e7ff5aaf 100644
--- a/drivers/net/ice/base/ice_common.h
+++ b/drivers/net/ice/base/ice_common.h
@@ -132,6 +132,15 @@ ice_aq_move_recfg_lan_txq(struct ice_hw *hw, u8 num_qs, 
bool is_move,
                          u8 timeout, u32 *blocked_cgds,
                          struct ice_aqc_move_txqs_data *buf, u16 buf_size,
                          u8 *txqs_moved, struct ice_sq_cd *cd);
+enum ice_status
+ice_aq_set_txtimeq(struct ice_hw *hw, u16 txtimeq, u8 q_count,
+                  struct ice_aqc_set_txtime_qgrp *txtime_qg,
+                  u16 buf_size, struct ice_sq_cd *cd);
+enum ice_status
+ice_aq_ena_dis_txtimeq(struct ice_hw *hw, u16 txtimeq, u16 q_count, bool q_ena,
+                      struct ice_aqc_ena_dis_txtime_qgrp *txtime_qg,
+                      struct ice_sq_cd *cd);
+extern const struct ice_ctx_ele ice_txtime_ctx_info[];
 
 bool ice_check_sq_alive(struct ice_hw *hw, struct ice_ctl_q_info *cq);
 int ice_aq_q_shutdown(struct ice_hw *hw, bool unloading);
diff --git a/drivers/net/ice/base/ice_lan_tx_rx.h 
b/drivers/net/ice/base/ice_lan_tx_rx.h
index bcc6e9a716..f92382346f 100644
--- a/drivers/net/ice/base/ice_lan_tx_rx.h
+++ b/drivers/net/ice/base/ice_lan_tx_rx.h
@@ -1275,6 +1275,43 @@ struct ice_ts_desc {
 };
 #define ICE_TS_DESC(R, i) (&(((struct ice_ts_desc *)((R)->desc))[i]))
 
+#define ICE_TXTIME_MAX_QUEUE           2047
+#define ICE_SET_TXTIME_MAX_Q_AMOUNT    127
+#define ICE_OP_TXTIME_MAX_Q_AMOUNT     2047
+/* Tx Time queue context data
+ *
+ * The sizes of the variables may be larger than needed due to crossing byte
+ * boundaries. If we do not have the width of the variable set to the correct
+ * size then we could end up shifting bits off the top of the variable when the
+ * variable is at the top of a byte and crosses over into the next byte.
+ */
+struct ice_txtime_ctx {
+#define ICE_TXTIME_CTX_BASE_S          7
+       u64 base;                       /* base is defined in 128-byte units */
+       u8 pf_num;
+       u16 vmvf_num;
+       u8 vmvf_type;
+#define ICE_TXTIME_CTX_VMVF_TYPE_VF    0
+#define ICE_TXTIME_CTX_VMVF_TYPE_VMQ   1
+#define ICE_TXTIME_CTX_VMVF_TYPE_PF    2
+       u16 src_vsi;
+       u8 cpuid;
+       u8 tphrd_desc;
+       u32 qlen;               /* bigger than needed, see above for reason */
+       u8 timer_num;
+       u8 txtime_ena_q;
+       u8 drbell_mode_32;
+#define ICE_TXTIME_CTX_DRBELL_MODE_32  1
+       u8 ts_res;
+       u8 ts_round_type;
+       u8 ts_pacing_slot;
+       u8 merging_ena;
+       u8 ts_fetch_prof_id;
+       u8 ts_fetch_cache_line_aln_thld;
+       u8 tx_pipe_delay_mode;
+       u8 int_q_state; /* width not needed - internal - DO NOT WRITE!!! */
+};
+
 /* The ice_ptype_lkup table is used to convert from the 10-bit ptype in the
  * hardware to a bit-field that can be used by SW to more easily determine the
  * packet type.
-- 
2.43.0

Reply via email to