]> git.proxmox.com Git - mirror_ubuntu-bionic-kernel.git/commitdiff
bnx2x: Support of PF driver of a VF setup_q request
authorAriel Elior <ariele@broadcom.com>
Tue, 1 Jan 2013 05:22:37 +0000 (05:22 +0000)
committerDavid S. Miller <davem@davemloft.net>
Wed, 2 Jan 2013 09:45:07 +0000 (01:45 -0800)
Upon receiving a 'setup_q' request from the VF over the VF <-> PF
channel the PF driver will open a corresponding queue in the
device. The PF driver configures the queue with appropriate mac
address, vlan configuration, etc from the VF.

Signed-off-by: Ariel Elior <ariele@broadcom.com>
Signed-off-by: Eilon Greenstein <eilong@broadcom.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/broadcom/bnx2x/bnx2x.h
drivers/net/ethernet/broadcom/bnx2x/bnx2x_cmn.c
drivers/net/ethernet/broadcom/bnx2x/bnx2x_sriov.c
drivers/net/ethernet/broadcom/bnx2x/bnx2x_sriov.h
drivers/net/ethernet/broadcom/bnx2x/bnx2x_vfpf.c

index d096e3e8528744cccdb371c1aa36e32443b2aae8..fbe8be31f67357b24b82da97cf8b1a96549c0190 100644 (file)
@@ -969,6 +969,7 @@ extern struct workqueue_struct *bnx2x_wq;
 #define BNX2X_MAX_NUM_OF_VFS   64
 #define BNX2X_VF_CID_WND       0
 #define BNX2X_CIDS_PER_VF      (1 << BNX2X_VF_CID_WND)
+#define BNX2X_CLIENTS_PER_VF   1
 #define BNX2X_FIRST_VF_CID     256
 #define BNX2X_VF_CIDS          (BNX2X_MAX_NUM_OF_VFS * BNX2X_CIDS_PER_VF)
 #define BNX2X_VF_ID_INVALID    0xFF
index 022c491cd0fb29810a60defcca3a690930faf044..cdb073a6297e791d791b4175e3a959dcb4526d51 100644 (file)
@@ -2029,7 +2029,7 @@ static void bnx2x_free_fw_stats_mem(struct bnx2x *bp)
 
 static int bnx2x_alloc_fw_stats_mem(struct bnx2x *bp)
 {
-       int num_groups;
+       int num_groups, vf_headroom = 0;
        int is_fcoe_stats = NO_FCOE(bp) ? 0 : 1;
 
        /* number of queues for statistics is number of eth queues + FCoE */
@@ -2042,18 +2042,26 @@ static int bnx2x_alloc_fw_stats_mem(struct bnx2x *bp)
         */
        bp->fw_stats_num = 2 + is_fcoe_stats + num_queue_stats;
 
+       /* vf stats appear in the request list, but their data is allocated by
+        * the VFs themselves. We don't include them in the bp->fw_stats_num as
+        * it is used to determine where to place the vf stats queries in the
+        * request struct
+        */
+       if (IS_SRIOV(bp))
+               vf_headroom = bp->vfdb->sriov.nr_virtfn * BNX2X_CLIENTS_PER_VF;
+
        /* Request is built from stats_query_header and an array of
         * stats_query_cmd_group each of which contains
         * STATS_QUERY_CMD_COUNT rules. The real number or requests is
         * configured in the stats_query_header.
         */
        num_groups =
-               (((bp->fw_stats_num) / STATS_QUERY_CMD_COUNT) +
-                (((bp->fw_stats_num) % STATS_QUERY_CMD_COUNT) ?
+               (((bp->fw_stats_num + vf_headroom) / STATS_QUERY_CMD_COUNT) +
+                (((bp->fw_stats_num + vf_headroom) % STATS_QUERY_CMD_COUNT) ?
                 1 : 0));
 
-       DP(BNX2X_MSG_SP, "stats fw_stats_num %d, num_groups %d\n",
-          bp->fw_stats_num, num_groups);
+       DP(BNX2X_MSG_SP, "stats fw_stats_num %d, vf headroom %d, num_groups %d\n",
+          bp->fw_stats_num, vf_headroom, num_groups);
        bp->fw_stats_req_sz = sizeof(struct stats_query_header) +
                num_groups * sizeof(struct stats_query_cmd_group);
 
index 71e1c6fb205df1ab6afa3019ca46692fcf189ae8..5b47b0849a58d467915d3067191a28af3a42c335 100644 (file)
@@ -100,10 +100,233 @@ static void bnx2x_vf_igu_ack_sb(struct bnx2x *bp, struct bnx2x_virtf *vf,
        mmiowb();
        barrier();
 }
+/* VFOP - VF slow-path operation support */
+
+/* VFOP operations states */
+enum bnx2x_vfop_qctor_state {
+          BNX2X_VFOP_QCTOR_INIT,
+          BNX2X_VFOP_QCTOR_SETUP,
+          BNX2X_VFOP_QCTOR_INT_EN
+};
+
+enum bnx2x_vfop_vlan_mac_state {
+          BNX2X_VFOP_VLAN_MAC_CONFIG_SINGLE,
+          BNX2X_VFOP_VLAN_MAC_CLEAR,
+          BNX2X_VFOP_VLAN_MAC_CHK_DONE,
+          BNX2X_VFOP_MAC_CONFIG_LIST,
+          BNX2X_VFOP_VLAN_CONFIG_LIST,
+          BNX2X_VFOP_VLAN_CONFIG_LIST_0
+};
+
+enum bnx2x_vfop_qsetup_state {
+          BNX2X_VFOP_QSETUP_CTOR,
+          BNX2X_VFOP_QSETUP_VLAN0,
+          BNX2X_VFOP_QSETUP_DONE
+};
+
+#define bnx2x_vfop_reset_wq(vf)        atomic_set(&vf->op_in_progress, 0)
+
+void bnx2x_vfop_qctor_dump_tx(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                             struct bnx2x_queue_init_params *init_params,
+                             struct bnx2x_queue_setup_params *setup_params,
+                             u16 q_idx, u16 sb_idx)
+{
+       DP(BNX2X_MSG_IOV,
+          "VF[%d] Q_SETUP: txq[%d]-- vfsb=%d, sb-index=%d, hc-rate=%d, flags=0x%lx, traffic-type=%d",
+          vf->abs_vfid,
+          q_idx,
+          sb_idx,
+          init_params->tx.sb_cq_index,
+          init_params->tx.hc_rate,
+          setup_params->flags,
+          setup_params->txq_params.traffic_type);
+}
 
-static int bnx2x_ari_enabled(struct pci_dev *dev)
+void bnx2x_vfop_qctor_dump_rx(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                           struct bnx2x_queue_init_params *init_params,
+                           struct bnx2x_queue_setup_params *setup_params,
+                           u16 q_idx, u16 sb_idx)
 {
-       return dev->bus->self && dev->bus->self->ari_enabled;
+       struct bnx2x_rxq_setup_params *rxq_params = &setup_params->rxq_params;
+
+       DP(BNX2X_MSG_IOV, "VF[%d] Q_SETUP: rxq[%d]-- vfsb=%d, sb-index=%d, hc-rate=%d, mtu=%d, buf-size=%d\n"
+          "sge-size=%d, max_sge_pkt=%d, tpa-agg-size=%d, flags=0x%lx, drop-flags=0x%x, cache-log=%d\n",
+          vf->abs_vfid,
+          q_idx,
+          sb_idx,
+          init_params->rx.sb_cq_index,
+          init_params->rx.hc_rate,
+          setup_params->gen_params.mtu,
+          rxq_params->buf_sz,
+          rxq_params->sge_buf_sz,
+          rxq_params->max_sges_pkt,
+          rxq_params->tpa_agg_sz,
+          setup_params->flags,
+          rxq_params->drop_flags,
+          rxq_params->cache_line_log);
+}
+
+void bnx2x_vfop_qctor_prep(struct bnx2x *bp,
+                          struct bnx2x_virtf *vf,
+                          struct bnx2x_vf_queue *q,
+                          struct bnx2x_vfop_qctor_params *p,
+                          unsigned long q_type)
+{
+       struct bnx2x_queue_init_params *init_p = &p->qstate.params.init;
+       struct bnx2x_queue_setup_params *setup_p = &p->prep_qsetup;
+
+       /* INIT */
+
+       /* Enable host coalescing in the transition to INIT state */
+       if (test_bit(BNX2X_Q_FLG_HC, &init_p->rx.flags))
+               __set_bit(BNX2X_Q_FLG_HC_EN, &init_p->rx.flags);
+
+       if (test_bit(BNX2X_Q_FLG_HC, &init_p->tx.flags))
+               __set_bit(BNX2X_Q_FLG_HC_EN, &init_p->tx.flags);
+
+       /* FW SB ID */
+       init_p->rx.fw_sb_id = vf_igu_sb(vf, q->sb_idx);
+       init_p->tx.fw_sb_id = vf_igu_sb(vf, q->sb_idx);
+
+       /* context */
+       init_p->cxts[0] = q->cxt;
+
+       /* SETUP */
+
+       /* Setup-op general parameters */
+       setup_p->gen_params.spcl_id = vf->sp_cl_id;
+       setup_p->gen_params.stat_id = vfq_stat_id(vf, q);
+
+       /* Setup-op pause params:
+        * Nothing to do, the pause thresholds are set by default to 0 which
+        * effectively turns off the feature for this queue. We don't want
+        * one queue (VF) to interfering with another queue (another VF)
+        */
+       if (vf->cfg_flags & VF_CFG_FW_FC)
+               BNX2X_ERR("No support for pause to VFs (abs_vfid: %d)\n",
+                         vf->abs_vfid);
+       /* Setup-op flags:
+        * collect statistics, zero statistics, local-switching, security,
+        * OV for Flex10, RSS and MCAST for leading
+        */
+       if (test_bit(BNX2X_Q_FLG_STATS, &setup_p->flags))
+               __set_bit(BNX2X_Q_FLG_ZERO_STATS, &setup_p->flags);
+
+       /* for VFs, enable tx switching, bd coherency, and mac address
+        * anti-spoofing
+        */
+       __set_bit(BNX2X_Q_FLG_TX_SWITCH, &setup_p->flags);
+       __set_bit(BNX2X_Q_FLG_TX_SEC, &setup_p->flags);
+       __set_bit(BNX2X_Q_FLG_ANTI_SPOOF, &setup_p->flags);
+
+       if (vfq_is_leading(q)) {
+               __set_bit(BNX2X_Q_FLG_LEADING_RSS, &setup_p->flags);
+               __set_bit(BNX2X_Q_FLG_MCAST, &setup_p->flags);
+       }
+
+       /* Setup-op rx parameters */
+       if (test_bit(BNX2X_Q_TYPE_HAS_RX, &q_type)) {
+               struct bnx2x_rxq_setup_params *rxq_p = &setup_p->rxq_params;
+
+               rxq_p->cl_qzone_id = vfq_qzone_id(vf, q);
+               rxq_p->fw_sb_id = vf_igu_sb(vf, q->sb_idx);
+               rxq_p->rss_engine_id = FW_VF_HANDLE(vf->abs_vfid);
+
+               if (test_bit(BNX2X_Q_FLG_TPA, &setup_p->flags))
+                       rxq_p->max_tpa_queues = BNX2X_VF_MAX_TPA_AGG_QUEUES;
+       }
+
+       /* Setup-op tx parameters */
+       if (test_bit(BNX2X_Q_TYPE_HAS_TX, &q_type)) {
+               setup_p->txq_params.tss_leading_cl_id = vf->leading_rss;
+               setup_p->txq_params.fw_sb_id = vf_igu_sb(vf, q->sb_idx);
+       }
+}
+
+/* VFOP queue construction */
+static void bnx2x_vfop_qctor(struct bnx2x *bp, struct bnx2x_virtf *vf)
+{
+       struct bnx2x_vfop *vfop = bnx2x_vfop_cur(bp, vf);
+       struct bnx2x_vfop_args_qctor *args = &vfop->args.qctor;
+       struct bnx2x_queue_state_params *q_params = &vfop->op_p->qctor.qstate;
+       enum bnx2x_vfop_qctor_state state = vfop->state;
+
+       bnx2x_vfop_reset_wq(vf);
+
+       if (vfop->rc < 0)
+               goto op_err;
+
+       DP(BNX2X_MSG_IOV, "vf[%d] STATE: %d\n", vf->abs_vfid, state);
+
+       switch (state) {
+       case BNX2X_VFOP_QCTOR_INIT:
+
+               /* has this queue already been opened? */
+               if (bnx2x_get_q_logical_state(bp, q_params->q_obj) ==
+                   BNX2X_Q_LOGICAL_STATE_ACTIVE) {
+                       DP(BNX2X_MSG_IOV,
+                          "Entered qctor but queue was already up. Aborting gracefully\n");
+                       goto op_done;
+               }
+
+               /* next state */
+               vfop->state = BNX2X_VFOP_QCTOR_SETUP;
+
+               q_params->cmd = BNX2X_Q_CMD_INIT;
+               vfop->rc = bnx2x_queue_state_change(bp, q_params);
+
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_CONT);
+
+       case BNX2X_VFOP_QCTOR_SETUP:
+               /* next state */
+               vfop->state = BNX2X_VFOP_QCTOR_INT_EN;
+
+               /* copy pre-prepared setup params to the queue-state params */
+               vfop->op_p->qctor.qstate.params.setup =
+                       vfop->op_p->qctor.prep_qsetup;
+
+               q_params->cmd = BNX2X_Q_CMD_SETUP;
+               vfop->rc = bnx2x_queue_state_change(bp, q_params);
+
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_CONT);
+
+       case BNX2X_VFOP_QCTOR_INT_EN:
+
+               /* enable interrupts */
+               bnx2x_vf_igu_ack_sb(bp, vf, vf_igu_sb(vf, args->sb_idx),
+                                   USTORM_ID, 0, IGU_INT_ENABLE, 0);
+               goto op_done;
+       default:
+               bnx2x_vfop_default(state);
+       }
+op_err:
+       BNX2X_ERR("QCTOR[%d:%d] error: cmd %d, rc %d\n",
+                 vf->abs_vfid, args->qid, q_params->cmd, vfop->rc);
+op_done:
+       bnx2x_vfop_end(bp, vf, vfop);
+op_pending:
+       return;
+}
+
+static int bnx2x_vfop_qctor_cmd(struct bnx2x *bp,
+                               struct bnx2x_virtf *vf,
+                               struct bnx2x_vfop_cmd *cmd,
+                               int qid)
+{
+       struct bnx2x_vfop *vfop = bnx2x_vfop_add(bp, vf);
+
+       if (vfop) {
+               vf->op_params.qctor.qstate.q_obj = &bnx2x_vfq(vf, qid, sp_obj);
+
+               vfop->args.qctor.qid = qid;
+               vfop->args.qctor.sb_idx = bnx2x_vfq(vf, qid, sb_idx);
+
+               bnx2x_vfop_opset(BNX2X_VFOP_QCTOR_INIT,
+                                bnx2x_vfop_qctor, cmd->done);
+               return bnx2x_vfop_transition(bp, vf, bnx2x_vfop_qctor,
+                                            cmd->block);
+       }
+       return -ENOMEM;
 }
 
 static void
@@ -117,225 +340,342 @@ bnx2x_vf_set_igu_info(struct bnx2x *bp, u8 igu_sb_id, u8 abs_vfid)
        }
 }
 
-static void
-bnx2x_get_vf_igu_cam_info(struct bnx2x *bp)
+/* VFOP MAC/VLAN helpers */
+static inline void bnx2x_vfop_credit(struct bnx2x *bp,
+                                    struct bnx2x_vfop *vfop,
+                                    struct bnx2x_vlan_mac_obj *obj)
 {
-       int sb_id;
-       u32 val;
-       u8 fid;
+       struct bnx2x_vfop_args_filters *args = &vfop->args.filters;
 
-       /* IGU in normal mode - read CAM */
-       for (sb_id = 0; sb_id < IGU_REG_MAPPING_MEMORY_SIZE; sb_id++) {
-               val = REG_RD(bp, IGU_REG_MAPPING_MEMORY + sb_id * 4);
-               if (!(val & IGU_REG_MAPPING_MEMORY_VALID))
-                       continue;
-               fid = GET_FIELD((val), IGU_REG_MAPPING_MEMORY_FID);
-               if (!(fid & IGU_FID_ENCODE_IS_PF))
-                       bnx2x_vf_set_igu_info(bp, sb_id,
-                                             (fid & IGU_FID_VF_NUM_MASK));
+       /* update credit only if there is no error
+        * and a valid credit counter
+        */
+       if (!vfop->rc && args->credit) {
+               int cnt = 0;
+               struct list_head *pos;
 
-               DP(BNX2X_MSG_IOV, "%s[%d], igu_sb_id=%d, msix=%d\n",
-                  ((fid & IGU_FID_ENCODE_IS_PF) ? "PF" : "VF"),
-                  ((fid & IGU_FID_ENCODE_IS_PF) ? (fid & IGU_FID_PF_NUM_MASK) :
-                  (fid & IGU_FID_VF_NUM_MASK)), sb_id,
-                  GET_FIELD((val), IGU_REG_MAPPING_MEMORY_VECTOR));
+               list_for_each(pos, &obj->head)
+                       cnt++;
+
+               atomic_set(args->credit, cnt);
        }
 }
 
-static void __bnx2x_iov_free_vfdb(struct bnx2x *bp)
+static int bnx2x_vfop_set_user_req(struct bnx2x *bp,
+                                   struct bnx2x_vfop_filter *pos,
+                                   struct bnx2x_vlan_mac_data *user_req)
 {
-       if (bp->vfdb) {
-               kfree(bp->vfdb->vfqs);
-               kfree(bp->vfdb->vfs);
-               kfree(bp->vfdb);
+       user_req->cmd = pos->add ? BNX2X_VLAN_MAC_ADD :
+               BNX2X_VLAN_MAC_DEL;
+
+       switch (pos->type) {
+       case BNX2X_VFOP_FILTER_MAC:
+               memcpy(user_req->u.mac.mac, pos->mac, ETH_ALEN);
+               break;
+       case BNX2X_VFOP_FILTER_VLAN:
+               user_req->u.vlan.vlan = pos->vid;
+               break;
+       default:
+               BNX2X_ERR("Invalid filter type, skipping\n");
+               return 1;
        }
-       bp->vfdb = NULL;
+       return 0;
 }
 
-static int bnx2x_sriov_pci_cfg_info(struct bnx2x *bp, struct bnx2x_sriov *iov)
+static int
+bnx2x_vfop_config_vlan0(struct bnx2x *bp,
+                       struct bnx2x_vlan_mac_ramrod_params *vlan_mac,
+                       bool add)
 {
-       int pos;
-       struct pci_dev *dev = bp->pdev;
-
-       pos = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_SRIOV);
-       if (!pos) {
-               BNX2X_ERR("failed to find SRIOV capability in device\n");
-               return -ENODEV;
-       }
+       int rc;
 
-       iov->pos = pos;
-       DP(BNX2X_MSG_IOV, "sriov ext pos %d\n", pos);
-       pci_read_config_word(dev, pos + PCI_SRIOV_CTRL, &iov->ctrl);
-       pci_read_config_word(dev, pos + PCI_SRIOV_TOTAL_VF, &iov->total);
-       pci_read_config_word(dev, pos + PCI_SRIOV_INITIAL_VF, &iov->initial);
-       pci_read_config_word(dev, pos + PCI_SRIOV_VF_OFFSET, &iov->offset);
-       pci_read_config_word(dev, pos + PCI_SRIOV_VF_STRIDE, &iov->stride);
-       pci_read_config_dword(dev, pos + PCI_SRIOV_SUP_PGSIZE, &iov->pgsz);
-       pci_read_config_dword(dev, pos + PCI_SRIOV_CAP, &iov->cap);
-       pci_read_config_byte(dev, pos + PCI_SRIOV_FUNC_LINK, &iov->link);
+       vlan_mac->user_req.cmd = add ? BNX2X_VLAN_MAC_ADD :
+               BNX2X_VLAN_MAC_DEL;
+       vlan_mac->user_req.u.vlan.vlan = 0;
 
-       return 0;
+       rc = bnx2x_config_vlan_mac(bp, vlan_mac);
+       if (rc == -EEXIST)
+               rc = 0;
+       return rc;
 }
 
-static int bnx2x_sriov_info(struct bnx2x *bp, struct bnx2x_sriov *iov)
+static int bnx2x_vfop_config_list(struct bnx2x *bp,
+                                 struct bnx2x_vfop_filters *filters,
+                                 struct bnx2x_vlan_mac_ramrod_params *vlan_mac)
 {
-       u32 val;
-
-       /* read the SRIOV capability structure
-        * The fields can be read via configuration read or
-        * directly from the device (starting at offset PCICFG_OFFSET)
-        */
-       if (bnx2x_sriov_pci_cfg_info(bp, iov))
-               return -ENODEV;
+       struct bnx2x_vfop_filter *pos, *tmp;
+       struct list_head rollback_list, *filters_list = &filters->head;
+       struct bnx2x_vlan_mac_data *user_req = &vlan_mac->user_req;
+       int rc = 0, cnt = 0;
 
-       /* get the number of SRIOV bars */
-       iov->nres = 0;
+       INIT_LIST_HEAD(&rollback_list);
 
-       /* read the first_vfid */
-       val = REG_RD(bp, PCICFG_OFFSET + GRC_CONFIG_REG_PF_INIT_VF);
-       iov->first_vf_in_pf = ((val & GRC_CR_PF_INIT_VF_PF_FIRST_VF_NUM_MASK)
-                              * 8) - (BNX2X_MAX_NUM_OF_VFS * BP_PATH(bp));
+       list_for_each_entry_safe(pos, tmp, filters_list, link) {
+               if (bnx2x_vfop_set_user_req(bp, pos, user_req))
+                       continue;
 
-       DP(BNX2X_MSG_IOV,
-          "IOV info[%d]: first vf %d, nres %d, cap 0x%x, ctrl 0x%x, total %d, initial %d, num vfs %d, offset %d, stride %d, page size 0x%x\n",
-          BP_FUNC(bp),
-          iov->first_vf_in_pf, iov->nres, iov->cap, iov->ctrl, iov->total,
-          iov->initial, iov->nr_virtfn, iov->offset, iov->stride, iov->pgsz);
+               rc = bnx2x_config_vlan_mac(bp, vlan_mac);
+               if (rc >= 0) {
+                       cnt += pos->add ? 1 : -1;
+                       list_del(&pos->link);
+                       list_add(&pos->link, &rollback_list);
+                       rc = 0;
+               } else if (rc == -EEXIST) {
+                       rc = 0;
+               } else {
+                       BNX2X_ERR("Failed to add a new vlan_mac command\n");
+                       break;
+               }
+       }
 
-       return 0;
+       /* rollback if error or too many rules added */
+       if (rc || cnt > filters->add_cnt) {
+               BNX2X_ERR("error or too many rules added. Performing rollback\n");
+               list_for_each_entry_safe(pos, tmp, &rollback_list, link) {
+                       pos->add = !pos->add;   /* reverse op */
+                       bnx2x_vfop_set_user_req(bp, pos, user_req);
+                       bnx2x_config_vlan_mac(bp, vlan_mac);
+                       list_del(&pos->link);
+               }
+               cnt = 0;
+               if (!rc)
+                       rc = -EINVAL;
+       }
+       filters->add_cnt = cnt;
+       return rc;
 }
 
-static u8 bnx2x_iov_get_max_queue_count(struct bnx2x *bp)
+/* VFOP set VLAN/MAC */
+static void bnx2x_vfop_vlan_mac(struct bnx2x *bp, struct bnx2x_virtf *vf)
 {
-       int i;
-       u8 queue_count = 0;
+       struct bnx2x_vfop *vfop = bnx2x_vfop_cur(bp, vf);
+       struct bnx2x_vlan_mac_ramrod_params *vlan_mac = &vfop->op_p->vlan_mac;
+       struct bnx2x_vlan_mac_obj *obj = vlan_mac->vlan_mac_obj;
+       struct bnx2x_vfop_filters *filters = vfop->args.filters.multi_filter;
 
-       if (IS_SRIOV(bp))
-               for_each_vf(bp, i)
-                       queue_count += bnx2x_vf(bp, i, alloc_resc.num_sbs);
+       enum bnx2x_vfop_vlan_mac_state state = vfop->state;
 
-       return queue_count;
-}
+       if (vfop->rc < 0)
+               goto op_err;
 
-/* must be called after PF bars are mapped */
-int bnx2x_iov_init_one(struct bnx2x *bp, int int_mode_param,
-                                int num_vfs_param)
-{
-       int err, i, qcount;
-       struct bnx2x_sriov *iov;
-       struct pci_dev *dev = bp->pdev;
+       DP(BNX2X_MSG_IOV, "vf[%d] STATE: %d\n", vf->abs_vfid, state);
 
-       bp->vfdb = NULL;
+       bnx2x_vfop_reset_wq(vf);
 
-       /* verify sriov capability is present in configuration space */
-       if (!pci_find_ext_capability(dev, PCI_EXT_CAP_ID_SRIOV)) {
-               DP(BNX2X_MSG_IOV, "no sriov - capability not found\n");
-               return 0;
-       }
+       switch (state) {
+       case BNX2X_VFOP_VLAN_MAC_CLEAR:
+               /* next state */
+               vfop->state = BNX2X_VFOP_VLAN_MAC_CHK_DONE;
 
-       /* verify is pf */
-       if (IS_VF(bp))
-               return 0;
+               /* do delete */
+               vfop->rc = obj->delete_all(bp, obj,
+                                          &vlan_mac->user_req.vlan_mac_flags,
+                                          &vlan_mac->ramrod_flags);
 
-       /* verify chip revision */
-       if (CHIP_IS_E1x(bp))
-               return 0;
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_DONE);
 
-       /* check if SRIOV support is turned off */
-       if (!num_vfs_param)
-               return 0;
+       case BNX2X_VFOP_VLAN_MAC_CONFIG_SINGLE:
+               /* next state */
+               vfop->state = BNX2X_VFOP_VLAN_MAC_CHK_DONE;
 
-       /* SRIOV assumes that num of PF CIDs < BNX2X_FIRST_VF_CID */
-       if (BNX2X_L2_MAX_CID(bp) >= BNX2X_FIRST_VF_CID) {
-               BNX2X_ERR("PF cids %d are overspilling into vf space (starts at %d). Abort SRIOV\n",
-                         BNX2X_L2_MAX_CID(bp), BNX2X_FIRST_VF_CID);
-               return 0;
-       }
+               /* do config */
+               vfop->rc = bnx2x_config_vlan_mac(bp, vlan_mac);
+               if (vfop->rc == -EEXIST)
+                       vfop->rc = 0;
 
-       /* SRIOV can be enabled only with MSIX */
-       if (int_mode_param == BNX2X_INT_MODE_MSI ||
-           int_mode_param == BNX2X_INT_MODE_INTX) {
-               BNX2X_ERR("Forced MSI/INTx mode is incompatible with SRIOV\n");
-               return 0;
-       }
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_DONE);
 
-       /* verify ari is enabled */
-       if (!bnx2x_ari_enabled(bp->pdev)) {
-               BNX2X_ERR("ARI not supported, SRIOV can not be enabled\n");
-               return 0;
-       }
+       case BNX2X_VFOP_VLAN_MAC_CHK_DONE:
+               vfop->rc = !!obj->raw.check_pending(&obj->raw);
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_DONE);
 
-       /* verify igu is in normal mode */
-       if (CHIP_INT_MODE_IS_BC(bp)) {
-               BNX2X_ERR("IGU not normal mode,  SRIOV can not be enabled\n");
-               return 0;
-       }
+       case BNX2X_VFOP_MAC_CONFIG_LIST:
+               /* next state */
+               vfop->state = BNX2X_VFOP_VLAN_MAC_CHK_DONE;
 
-       /* allocate the vfs database */
-       bp->vfdb = kzalloc(sizeof(*(bp->vfdb)), GFP_KERNEL);
-       if (!bp->vfdb) {
-               BNX2X_ERR("failed to allocate vf database\n");
-               err = -ENOMEM;
-               goto failed;
-       }
+               /* do list config */
+               vfop->rc = bnx2x_vfop_config_list(bp, filters, vlan_mac);
+               if (vfop->rc)
+                       goto op_err;
 
-       /* get the sriov info - Linux already collected all the pertinent
-        * information, however the sriov structure is for the private use
-        * of the pci module. Also we want this information regardless
-        * of the hyper-visor.
-        */
-       iov = &(bp->vfdb->sriov);
-       err = bnx2x_sriov_info(bp, iov);
-       if (err)
-               goto failed;
+               set_bit(RAMROD_CONT, &vlan_mac->ramrod_flags);
+               vfop->rc = bnx2x_config_vlan_mac(bp, vlan_mac);
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_DONE);
 
-       /* SR-IOV capability was enabled but there are no VFs*/
-       if (iov->total == 0)
-               goto failed;
+       case BNX2X_VFOP_VLAN_CONFIG_LIST:
+               /* next state */
+               vfop->state = BNX2X_VFOP_VLAN_CONFIG_LIST_0;
 
-       /* calcuate the actual number of VFs */
-       iov->nr_virtfn = min_t(u16, iov->total, (u16)num_vfs_param);
+               /* remove vlan0 - could be no-op */
+               vfop->rc = bnx2x_vfop_config_vlan0(bp, vlan_mac, false);
+               if (vfop->rc)
+                       goto op_err;
 
-       /* allcate the vf array */
-       bp->vfdb->vfs = kzalloc(sizeof(struct bnx2x_virtf) *
-                               BNX2X_NR_VIRTFN(bp), GFP_KERNEL);
-       if (!bp->vfdb->vfs) {
-               BNX2X_ERR("failed to allocate vf array\n");
-               err = -ENOMEM;
-               goto failed;
+               /* Do vlan list config. if this operation fails we try to
+                * restore vlan0 to keep the queue is working order
+                */
+               vfop->rc = bnx2x_vfop_config_list(bp, filters, vlan_mac);
+               if (!vfop->rc) {
+                       set_bit(RAMROD_CONT, &vlan_mac->ramrod_flags);
+                       vfop->rc = bnx2x_config_vlan_mac(bp, vlan_mac);
+               }
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_CONT); /* fall-through */
+
+       case BNX2X_VFOP_VLAN_CONFIG_LIST_0:
+               /* next state */
+               vfop->state = BNX2X_VFOP_VLAN_MAC_CHK_DONE;
+
+               if (list_empty(&obj->head))
+                       /* add vlan0 */
+                       vfop->rc = bnx2x_vfop_config_vlan0(bp, vlan_mac, true);
+               bnx2x_vfop_finalize(vf, vfop->rc, VFOP_DONE);
+
+       default:
+               bnx2x_vfop_default(state);
        }
+op_err:
+       BNX2X_ERR("VLAN-MAC error: rc %d\n", vfop->rc);
+op_done:
+       kfree(filters);
+       bnx2x_vfop_credit(bp, vfop, obj);
+       bnx2x_vfop_end(bp, vf, vfop);
+op_pending:
+       return;
+}
 
-       /* Initial VF init - index and abs_vfid - nr_virtfn must be set */
-       for_each_vf(bp, i) {
-               bnx2x_vf(bp, i, index) = i;
-               bnx2x_vf(bp, i, abs_vfid) = iov->first_vf_in_pf + i;
-               bnx2x_vf(bp, i, state) = VF_FREE;
-               INIT_LIST_HEAD(&bnx2x_vf(bp, i, op_list_head));
-               mutex_init(&bnx2x_vf(bp, i, op_mutex));
-               bnx2x_vf(bp, i, op_current) = CHANNEL_TLV_NONE;
+struct bnx2x_vfop_vlan_mac_flags {
+       bool drv_only;
+       bool dont_consume;
+       bool single_cmd;
+       bool add;
+};
+
+static void
+bnx2x_vfop_vlan_mac_prep_ramrod(struct bnx2x_vlan_mac_ramrod_params *ramrod,
+                               struct bnx2x_vfop_vlan_mac_flags *flags)
+{
+       struct bnx2x_vlan_mac_data *ureq = &ramrod->user_req;
+
+       memset(ramrod, 0, sizeof(*ramrod));
+
+       /* ramrod flags */
+       if (flags->drv_only)
+               set_bit(RAMROD_DRV_CLR_ONLY, &ramrod->ramrod_flags);
+       if (flags->single_cmd)
+               set_bit(RAMROD_EXEC, &ramrod->ramrod_flags);
+
+       /* mac_vlan flags */
+       if (flags->dont_consume)
+               set_bit(BNX2X_DONT_CONSUME_CAM_CREDIT, &ureq->vlan_mac_flags);
+
+       /* cmd */
+       ureq->cmd = flags->add ? BNX2X_VLAN_MAC_ADD : BNX2X_VLAN_MAC_DEL;
+}
+
+int bnx2x_vfop_vlan_set_cmd(struct bnx2x *bp,
+                           struct bnx2x_virtf *vf,
+                           struct bnx2x_vfop_cmd *cmd,
+                           int qid, u16 vid, bool add)
+{
+       struct bnx2x_vfop *vfop = bnx2x_vfop_add(bp, vf);
+
+       if (vfop) {
+               struct bnx2x_vfop_args_filters filters = {
+                       .multi_filter = NULL, /* single command */
+                       .credit = &bnx2x_vfq(vf, qid, vlan_count),
+               };
+               struct bnx2x_vfop_vlan_mac_flags flags = {
+                       .drv_only = false,
+                       .dont_consume = (filters.credit != NULL),
+                       .single_cmd = true,
+                       .add = add,
+               };
+               struct bnx2x_vlan_mac_ramrod_params *ramrod =
+                       &vf->op_params.vlan_mac;
+
+               /* set ramrod params */
+               bnx2x_vfop_vlan_mac_prep_ramrod(ramrod, &flags);
+               ramrod->user_req.u.vlan.vlan = vid;
+
+               /* set object */
+               ramrod->vlan_mac_obj = &bnx2x_vfq(vf, qid, vlan_obj);
+
+               /* set extra args */
+               vfop->args.filters = filters;
+
+               bnx2x_vfop_opset(BNX2X_VFOP_VLAN_MAC_CONFIG_SINGLE,
+                                bnx2x_vfop_vlan_mac, cmd->done);
+               return bnx2x_vfop_transition(bp, vf, bnx2x_vfop_vlan_mac,
+                                            cmd->block);
        }
+       return -ENOMEM;
+}
 
-       /* re-read the IGU CAM for VFs - index and abs_vfid must be set */
-       bnx2x_get_vf_igu_cam_info(bp);
+/* VFOP queue setup (queue constructor + set vlan 0) */
+static void bnx2x_vfop_qsetup(struct bnx2x *bp, struct bnx2x_virtf *vf)
+{
+       struct bnx2x_vfop *vfop = bnx2x_vfop_cur(bp, vf);
+       int qid = vfop->args.qctor.qid;
+       enum bnx2x_vfop_qsetup_state state = vfop->state;
+       struct bnx2x_vfop_cmd cmd = {
+               .done = bnx2x_vfop_qsetup,
+               .block = false,
+       };
+
+       if (vfop->rc < 0)
+               goto op_err;
+
+       DP(BNX2X_MSG_IOV, "vf[%d] STATE: %d\n", vf->abs_vfid, state);
+
+       switch (state) {
+       case BNX2X_VFOP_QSETUP_CTOR:
+               /* init the queue ctor command */
+               vfop->state = BNX2X_VFOP_QSETUP_VLAN0;
+               vfop->rc = bnx2x_vfop_qctor_cmd(bp, vf, &cmd, qid);
+               if (vfop->rc)
+                       goto op_err;
+               return;
 
-       /* get the total queue count and allocate the global queue arrays */
-       qcount = bnx2x_iov_get_max_queue_count(bp);
+       case BNX2X_VFOP_QSETUP_VLAN0:
+               /* skip if non-leading or FPGA/EMU*/
+               if (qid)
+                       goto op_done;
 
-       /* allocate the queue arrays for all VFs */
-       bp->vfdb->vfqs = kzalloc(qcount * sizeof(struct bnx2x_vf_queue),
-                                GFP_KERNEL);
-       if (!bp->vfdb->vfqs) {
-               BNX2X_ERR("failed to allocate vf queue array\n");
-               err = -ENOMEM;
-               goto failed;
+               /* init the queue set-vlan command (for vlan 0) */
+               vfop->state = BNX2X_VFOP_QSETUP_DONE;
+               vfop->rc = bnx2x_vfop_vlan_set_cmd(bp, vf, &cmd, qid, 0, true);
+               if (vfop->rc)
+                       goto op_err;
+               return;
+op_err:
+       BNX2X_ERR("QSETUP[%d:%d] error: rc %d\n", vf->abs_vfid, qid, vfop->rc);
+op_done:
+       case BNX2X_VFOP_QSETUP_DONE:
+               bnx2x_vfop_end(bp, vf, vfop);
+               return;
+       default:
+               bnx2x_vfop_default(state);
        }
+}
 
-       return 0;
-failed:
-       DP(BNX2X_MSG_IOV, "Failed err=%d\n", err);
-       __bnx2x_iov_free_vfdb(bp);
-       return err;
+int bnx2x_vfop_qsetup_cmd(struct bnx2x *bp,
+                         struct bnx2x_virtf *vf,
+                         struct bnx2x_vfop_cmd *cmd,
+                         int qid)
+{
+       struct bnx2x_vfop *vfop = bnx2x_vfop_add(bp, vf);
+
+       if (vfop) {
+               vfop->args.qctor.qid = qid;
+
+               bnx2x_vfop_opset(BNX2X_VFOP_QSETUP_CTOR,
+                                bnx2x_vfop_qsetup, cmd->done);
+               return bnx2x_vfop_transition(bp, vf, bnx2x_vfop_qsetup,
+                                            cmd->block);
+       }
+       return -ENOMEM;
 }
+
 /* VF enable primitives
  * when pretend is required the caller is responsible
  * for calling pretend prior to calling these routines
@@ -610,6 +950,228 @@ static void bnx2x_vf_set_bars(struct bnx2x *bp, struct bnx2x_virtf *vf)
        }
 }
 
+static int bnx2x_ari_enabled(struct pci_dev *dev)
+{
+       return dev->bus->self && dev->bus->self->ari_enabled;
+}
+
+static void
+bnx2x_get_vf_igu_cam_info(struct bnx2x *bp)
+{
+       int sb_id;
+       u32 val;
+       u8 fid;
+
+       /* IGU in normal mode - read CAM */
+       for (sb_id = 0; sb_id < IGU_REG_MAPPING_MEMORY_SIZE; sb_id++) {
+               val = REG_RD(bp, IGU_REG_MAPPING_MEMORY + sb_id * 4);
+               if (!(val & IGU_REG_MAPPING_MEMORY_VALID))
+                       continue;
+               fid = GET_FIELD((val), IGU_REG_MAPPING_MEMORY_FID);
+               if (!(fid & IGU_FID_ENCODE_IS_PF))
+                       bnx2x_vf_set_igu_info(bp, sb_id,
+                                             (fid & IGU_FID_VF_NUM_MASK));
+
+               DP(BNX2X_MSG_IOV, "%s[%d], igu_sb_id=%d, msix=%d\n",
+                  ((fid & IGU_FID_ENCODE_IS_PF) ? "PF" : "VF"),
+                  ((fid & IGU_FID_ENCODE_IS_PF) ? (fid & IGU_FID_PF_NUM_MASK) :
+                  (fid & IGU_FID_VF_NUM_MASK)), sb_id,
+                  GET_FIELD((val), IGU_REG_MAPPING_MEMORY_VECTOR));
+       }
+}
+
+static void __bnx2x_iov_free_vfdb(struct bnx2x *bp)
+{
+       if (bp->vfdb) {
+               kfree(bp->vfdb->vfqs);
+               kfree(bp->vfdb->vfs);
+               kfree(bp->vfdb);
+       }
+       bp->vfdb = NULL;
+}
+
+static int bnx2x_sriov_pci_cfg_info(struct bnx2x *bp, struct bnx2x_sriov *iov)
+{
+       int pos;
+       struct pci_dev *dev = bp->pdev;
+
+       pos = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_SRIOV);
+       if (!pos) {
+               BNX2X_ERR("failed to find SRIOV capability in device\n");
+               return -ENODEV;
+       }
+
+       iov->pos = pos;
+       DP(BNX2X_MSG_IOV, "sriov ext pos %d\n", pos);
+       pci_read_config_word(dev, pos + PCI_SRIOV_CTRL, &iov->ctrl);
+       pci_read_config_word(dev, pos + PCI_SRIOV_TOTAL_VF, &iov->total);
+       pci_read_config_word(dev, pos + PCI_SRIOV_INITIAL_VF, &iov->initial);
+       pci_read_config_word(dev, pos + PCI_SRIOV_VF_OFFSET, &iov->offset);
+       pci_read_config_word(dev, pos + PCI_SRIOV_VF_STRIDE, &iov->stride);
+       pci_read_config_dword(dev, pos + PCI_SRIOV_SUP_PGSIZE, &iov->pgsz);
+       pci_read_config_dword(dev, pos + PCI_SRIOV_CAP, &iov->cap);
+       pci_read_config_byte(dev, pos + PCI_SRIOV_FUNC_LINK, &iov->link);
+
+       return 0;
+}
+
+static int bnx2x_sriov_info(struct bnx2x *bp, struct bnx2x_sriov *iov)
+{
+       u32 val;
+
+       /* read the SRIOV capability structure
+        * The fields can be read via configuration read or
+        * directly from the device (starting at offset PCICFG_OFFSET)
+        */
+       if (bnx2x_sriov_pci_cfg_info(bp, iov))
+               return -ENODEV;
+
+       /* get the number of SRIOV bars */
+       iov->nres = 0;
+
+       /* read the first_vfid */
+       val = REG_RD(bp, PCICFG_OFFSET + GRC_CONFIG_REG_PF_INIT_VF);
+       iov->first_vf_in_pf = ((val & GRC_CR_PF_INIT_VF_PF_FIRST_VF_NUM_MASK)
+                              * 8) - (BNX2X_MAX_NUM_OF_VFS * BP_PATH(bp));
+
+       DP(BNX2X_MSG_IOV,
+          "IOV info[%d]: first vf %d, nres %d, cap 0x%x, ctrl 0x%x, total %d, initial %d, num vfs %d, offset %d, stride %d, page size 0x%x\n",
+          BP_FUNC(bp),
+          iov->first_vf_in_pf, iov->nres, iov->cap, iov->ctrl, iov->total,
+          iov->initial, iov->nr_virtfn, iov->offset, iov->stride, iov->pgsz);
+
+       return 0;
+}
+
+static u8 bnx2x_iov_get_max_queue_count(struct bnx2x *bp)
+{
+       int i;
+       u8 queue_count = 0;
+
+       if (IS_SRIOV(bp))
+               for_each_vf(bp, i)
+                       queue_count += bnx2x_vf(bp, i, alloc_resc.num_sbs);
+
+       return queue_count;
+}
+
+/* must be called after PF bars are mapped */
+int bnx2x_iov_init_one(struct bnx2x *bp, int int_mode_param,
+                       int num_vfs_param)
+{
+       int err, i, qcount;
+       struct bnx2x_sriov *iov;
+       struct pci_dev *dev = bp->pdev;
+
+       bp->vfdb = NULL;
+
+       /* verify is pf */
+       if (IS_VF(bp))
+               return 0;
+
+       /* verify sriov capability is present in configuration space */
+       if (!pci_find_ext_capability(dev, PCI_EXT_CAP_ID_SRIOV))
+               return 0;
+
+       /* verify chip revision */
+       if (CHIP_IS_E1x(bp))
+               return 0;
+
+       /* check if SRIOV support is turned off */
+       if (!num_vfs_param)
+               return 0;
+
+       /* SRIOV assumes that num of PF CIDs < BNX2X_FIRST_VF_CID */
+       if (BNX2X_L2_MAX_CID(bp) >= BNX2X_FIRST_VF_CID) {
+               BNX2X_ERR("PF cids %d are overspilling into vf space (starts at %d). Abort SRIOV\n",
+                         BNX2X_L2_MAX_CID(bp), BNX2X_FIRST_VF_CID);
+               return 0;
+       }
+
+       /* SRIOV can be enabled only with MSIX */
+       if (int_mode_param == BNX2X_INT_MODE_MSI ||
+           int_mode_param == BNX2X_INT_MODE_INTX)
+               BNX2X_ERR("Forced MSI/INTx mode is incompatible with SRIOV\n");
+
+       err = -EIO;
+       /* verify ari is enabled */
+       if (!bnx2x_ari_enabled(bp->pdev)) {
+               BNX2X_ERR("ARI not supported, SRIOV can not be enabled\n");
+               return err;
+       }
+
+       /* verify igu is in normal mode */
+       if (CHIP_INT_MODE_IS_BC(bp)) {
+               BNX2X_ERR("IGU not normal mode,  SRIOV can not be enabled\n");
+               return err;
+       }
+
+       /* allocate the vfs database */
+       bp->vfdb = kzalloc(sizeof(*(bp->vfdb)), GFP_KERNEL);
+       if (!bp->vfdb) {
+               BNX2X_ERR("failed to allocate vf database\n");
+               err = -ENOMEM;
+               goto failed;
+       }
+
+       /* get the sriov info - Linux already collected all the pertinent
+        * information, however the sriov structure is for the private use
+        * of the pci module. Also we want this information regardless
+        * of the hyper-visor.
+        */
+       iov = &(bp->vfdb->sriov);
+       err = bnx2x_sriov_info(bp, iov);
+       if (err)
+               goto failed;
+
+       /* SR-IOV capability was enabled but there are no VFs*/
+       if (iov->total == 0)
+               goto failed;
+
+       /* calculate the actual number of VFs */
+       iov->nr_virtfn = min_t(u16, iov->total, (u16)num_vfs_param);
+
+       /* allocate the vf array */
+       bp->vfdb->vfs = kzalloc(sizeof(struct bnx2x_virtf) *
+                               BNX2X_NR_VIRTFN(bp), GFP_KERNEL);
+       if (!bp->vfdb->vfs) {
+               BNX2X_ERR("failed to allocate vf array\n");
+               err = -ENOMEM;
+               goto failed;
+       }
+
+       /* Initial VF init - index and abs_vfid - nr_virtfn must be set */
+       for_each_vf(bp, i) {
+               bnx2x_vf(bp, i, index) = i;
+               bnx2x_vf(bp, i, abs_vfid) = iov->first_vf_in_pf + i;
+               bnx2x_vf(bp, i, state) = VF_FREE;
+               INIT_LIST_HEAD(&bnx2x_vf(bp, i, op_list_head));
+               mutex_init(&bnx2x_vf(bp, i, op_mutex));
+               bnx2x_vf(bp, i, op_current) = CHANNEL_TLV_NONE;
+       }
+
+       /* re-read the IGU CAM for VFs - index and abs_vfid must be set */
+       bnx2x_get_vf_igu_cam_info(bp);
+
+       /* get the total queue count and allocate the global queue arrays */
+       qcount = bnx2x_iov_get_max_queue_count(bp);
+
+       /* allocate the queue arrays for all VFs */
+       bp->vfdb->vfqs = kzalloc(qcount * sizeof(struct bnx2x_vf_queue),
+                                GFP_KERNEL);
+       if (!bp->vfdb->vfqs) {
+               BNX2X_ERR("failed to allocate vf queue array\n");
+               err = -ENOMEM;
+               goto failed;
+       }
+
+       return 0;
+failed:
+       DP(BNX2X_MSG_IOV, "Failed err=%d\n", err);
+       __bnx2x_iov_free_vfdb(bp);
+       return err;
+}
+
 void bnx2x_iov_remove_one(struct bnx2x *bp)
 {
        /* if SRIOV is not enabled there's nothing to do */
index c3d27b5a713af532f76f538b9462c4412981fdcd..49d452e9117457b37da97c93359f9eb5aee6a254 100644 (file)
@@ -26,6 +26,8 @@
  * The VF array is indexed by the relative vfid.
  */
 #define BNX2X_VF_MAX_QUEUES            16
+#define BNX2X_VF_MAX_TPA_AGG_QUEUES    8
+
 struct bnx2x_sriov {
        u32 first_vf_in_pf;
 
@@ -91,6 +93,11 @@ struct bnx2x_virtf;
 /* VFOP definitions */
 typedef void (*vfop_handler_t)(struct bnx2x *bp, struct bnx2x_virtf *vf);
 
+struct bnx2x_vfop_cmd {
+       vfop_handler_t done;
+       bool block;
+};
+
 /* VFOP queue filters command additional arguments */
 struct bnx2x_vfop_filter {
        struct list_head link;
@@ -405,6 +412,11 @@ static u8 vfq_cl_id(struct bnx2x_virtf *vf, struct bnx2x_vf_queue *q)
        return vf->igu_base_id + q->index;
 }
 
+static inline u8 vfq_stat_id(struct bnx2x_virtf *vf, struct bnx2x_vf_queue *q)
+{
+       return vfq_cl_id(vf, q);
+}
+
 static inline u8 vfq_qzone_id(struct bnx2x_virtf *vf, struct bnx2x_vf_queue *q)
 {
        return vfq_cl_id(vf, q);
@@ -435,6 +447,45 @@ int bnx2x_vf_acquire(struct bnx2x *bp, struct bnx2x_virtf *vf,
 /* init */
 int bnx2x_vf_init(struct bnx2x *bp, struct bnx2x_virtf *vf,
                  dma_addr_t *sb_map);
+
+/* VFOP generic helpers */
+#define bnx2x_vfop_default(state) do {                         \
+               BNX2X_ERR("Bad state %d\n", (state));           \
+               vfop->rc = -EINVAL;                             \
+               goto op_err;                                    \
+       } while (0)
+
+enum {
+       VFOP_DONE,
+       VFOP_CONT,
+       VFOP_VERIFY_PEND,
+};
+
+#define bnx2x_vfop_finalize(vf, rc, next) do {                         \
+               if ((rc) < 0)                                           \
+                       goto op_err;                                    \
+               else if ((rc) > 0)                                      \
+                       goto op_pending;                                \
+               else if ((next) == VFOP_DONE)                           \
+                       goto op_done;                                   \
+               else if ((next) == VFOP_VERIFY_PEND)                    \
+                       BNX2X_ERR("expected pending\n");                \
+               else {                                                  \
+                       DP(BNX2X_MSG_IOV, "no ramrod. scheduling\n");   \
+                       atomic_set(&vf->op_in_progress, 1);             \
+                       queue_delayed_work(bnx2x_wq, &bp->sp_task, 0);  \
+                       return;                                         \
+               }                                                       \
+       } while (0)
+
+#define bnx2x_vfop_opset(first_state, trans_hndlr, done_hndlr)         \
+       do {                                                            \
+               vfop->state = first_state;                              \
+               vfop->op_p = &vf->op_params;                            \
+               vfop->transition = trans_hndlr;                         \
+               vfop->done = done_hndlr;                                \
+       } while (0)
+
 static inline struct bnx2x_vfop *bnx2x_vfop_cur(struct bnx2x *bp,
                                                struct bnx2x_virtf *vf)
 {
@@ -443,6 +494,132 @@ static inline struct bnx2x_vfop *bnx2x_vfop_cur(struct bnx2x *bp,
        return list_first_entry(&vf->op_list_head, struct bnx2x_vfop, link);
 }
 
+static inline struct bnx2x_vfop *bnx2x_vfop_add(struct bnx2x *bp,
+                                               struct bnx2x_virtf *vf)
+{
+       struct bnx2x_vfop *vfop = kzalloc(sizeof(*vfop), GFP_KERNEL);
+
+       WARN(!mutex_is_locked(&vf->op_mutex), "about to access vf op linked list but mutex was not locked!");
+       if (vfop) {
+               INIT_LIST_HEAD(&vfop->link);
+               list_add(&vfop->link, &vf->op_list_head);
+       }
+       return vfop;
+}
+
+static inline void bnx2x_vfop_end(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                                 struct bnx2x_vfop *vfop)
+{
+       /* rc < 0 - error, otherwise set to 0 */
+       DP(BNX2X_MSG_IOV, "rc was %d\n", vfop->rc);
+       if (vfop->rc >= 0)
+               vfop->rc = 0;
+       DP(BNX2X_MSG_IOV, "rc is now %d\n", vfop->rc);
+
+       /* unlink the current op context and propagate error code
+        * must be done before invoking the 'done()' handler
+        */
+       WARN(!mutex_is_locked(&vf->op_mutex),
+            "about to access vf op linked list but mutex was not locked!");
+       list_del(&vfop->link);
+
+       if (list_empty(&vf->op_list_head)) {
+               DP(BNX2X_MSG_IOV, "list was empty %d\n", vfop->rc);
+               vf->op_rc = vfop->rc;
+               DP(BNX2X_MSG_IOV, "copying rc vf->op_rc %d,  vfop->rc %d\n",
+                  vf->op_rc, vfop->rc);
+       } else {
+               struct bnx2x_vfop *cur_vfop;
+
+               DP(BNX2X_MSG_IOV, "list not empty %d\n", vfop->rc);
+               cur_vfop = bnx2x_vfop_cur(bp, vf);
+               cur_vfop->rc = vfop->rc;
+               DP(BNX2X_MSG_IOV, "copying rc vf->op_rc %d, vfop->rc %d\n",
+                  vf->op_rc, vfop->rc);
+       }
+
+       /* invoke done handler */
+       if (vfop->done) {
+               DP(BNX2X_MSG_IOV, "calling done handler\n");
+               vfop->done(bp, vf);
+       }
+
+       DP(BNX2X_MSG_IOV, "done handler complete. vf->op_rc %d, vfop->rc %d\n",
+          vf->op_rc, vfop->rc);
+
+       /* if this is the last nested op reset the wait_blocking flag
+        * to release any blocking wrappers, only after 'done()' is invoked
+        */
+       if (list_empty(&vf->op_list_head)) {
+               DP(BNX2X_MSG_IOV, "list was empty after done %d\n", vfop->rc);
+               vf->op_wait_blocking = false;
+       }
+
+       kfree(vfop);
+}
+
+static inline int bnx2x_vfop_wait_blocking(struct bnx2x *bp,
+                                          struct bnx2x_virtf *vf)
+{
+       /* can take a while if any port is running */
+       int cnt = 5000;
+
+       might_sleep();
+       while (cnt--) {
+               if (vf->op_wait_blocking == false) {
+#ifdef BNX2X_STOP_ON_ERROR
+                       DP(BNX2X_MSG_IOV, "exit  (cnt %d)\n", 5000 - cnt);
+#endif
+                       return 0;
+               }
+               usleep_range(1000, 2000);
+
+               if (bp->panic)
+                       return -EIO;
+       }
+
+       /* timeout! */
+#ifdef BNX2X_STOP_ON_ERROR
+       bnx2x_panic();
+#endif
+
+       return -EBUSY;
+}
+
+static inline int bnx2x_vfop_transition(struct bnx2x *bp,
+                                       struct bnx2x_virtf *vf,
+                                       vfop_handler_t transition,
+                                       bool block)
+{
+       if (block)
+               vf->op_wait_blocking = true;
+       transition(bp, vf);
+       if (block)
+               return bnx2x_vfop_wait_blocking(bp, vf);
+       return 0;
+}
+
+/* VFOP queue construction helpers */
+void bnx2x_vfop_qctor_dump_tx(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                           struct bnx2x_queue_init_params *init_params,
+                           struct bnx2x_queue_setup_params *setup_params,
+                           u16 q_idx, u16 sb_idx);
+
+void bnx2x_vfop_qctor_dump_rx(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                           struct bnx2x_queue_init_params *init_params,
+                           struct bnx2x_queue_setup_params *setup_params,
+                           u16 q_idx, u16 sb_idx);
+
+void bnx2x_vfop_qctor_prep(struct bnx2x *bp,
+                          struct bnx2x_virtf *vf,
+                          struct bnx2x_vf_queue *q,
+                          struct bnx2x_vfop_qctor_params *p,
+                          unsigned long q_type);
+int bnx2x_vfop_qsetup_cmd(struct bnx2x *bp,
+                         struct bnx2x_virtf *vf,
+                         struct bnx2x_vfop_cmd *cmd,
+                         int qid);
+
 int bnx2x_vf_idx_by_abs_fid(struct bnx2x *bp, u16 abs_vfid);
 u8 bnx2x_vf_max_queue_cnt(struct bnx2x *bp, struct bnx2x_virtf *vf);
 /* VF FLR helpers */
index 7aa0e4f5346ade725df142cbbf5ffa0c4aadae37..6605567e4b0c73879a6fef60d213332abf75b4fe 100644 (file)
@@ -370,6 +370,149 @@ static void bnx2x_vf_mbx_init_vf(struct bnx2x *bp, struct bnx2x_virtf *vf,
        bnx2x_vf_mbx_resp(bp, vf);
 }
 
+/* convert MBX queue-flags to standard SP queue-flags */
+static void bnx2x_vf_mbx_set_q_flags(u32 mbx_q_flags,
+                                    unsigned long *sp_q_flags)
+{
+       if (mbx_q_flags & VFPF_QUEUE_FLG_TPA)
+               __set_bit(BNX2X_Q_FLG_TPA, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_TPA_IPV6)
+               __set_bit(BNX2X_Q_FLG_TPA_IPV6, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_TPA_GRO)
+               __set_bit(BNX2X_Q_FLG_TPA_GRO, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_STATS)
+               __set_bit(BNX2X_Q_FLG_STATS, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_OV)
+               __set_bit(BNX2X_Q_FLG_OV, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_VLAN)
+               __set_bit(BNX2X_Q_FLG_VLAN, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_COS)
+               __set_bit(BNX2X_Q_FLG_COS, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_HC)
+               __set_bit(BNX2X_Q_FLG_HC, sp_q_flags);
+       if (mbx_q_flags & VFPF_QUEUE_FLG_DHC)
+               __set_bit(BNX2X_Q_FLG_DHC, sp_q_flags);
+}
+
+static void bnx2x_vf_mbx_setup_q(struct bnx2x *bp, struct bnx2x_virtf *vf,
+                                struct bnx2x_vf_mbx *mbx)
+{
+       struct vfpf_setup_q_tlv *setup_q = &mbx->msg->req.setup_q;
+       struct bnx2x_vfop_cmd cmd = {
+               .done = bnx2x_vf_mbx_resp,
+               .block = false,
+       };
+
+       /* verify vf_qid */
+       if (setup_q->vf_qid >= vf_rxq_count(vf)) {
+               BNX2X_ERR("vf_qid %d invalid, max queue count is %d\n",
+                         setup_q->vf_qid, vf_rxq_count(vf));
+               vf->op_rc = -EINVAL;
+               goto response;
+       }
+
+       /* tx queues must be setup alongside rx queues thus if the rx queue
+        * is not marked as valid there's nothing to do.
+        */
+       if (setup_q->param_valid & (VFPF_RXQ_VALID|VFPF_TXQ_VALID)) {
+               struct bnx2x_vf_queue *q = vfq_get(vf, setup_q->vf_qid);
+               unsigned long q_type = 0;
+
+               struct bnx2x_queue_init_params *init_p;
+               struct bnx2x_queue_setup_params *setup_p;
+
+               /* reinit the VF operation context */
+               memset(&vf->op_params.qctor, 0 , sizeof(vf->op_params.qctor));
+               setup_p = &vf->op_params.qctor.prep_qsetup;
+               init_p =  &vf->op_params.qctor.qstate.params.init;
+
+               /* activate immediately */
+               __set_bit(BNX2X_Q_FLG_ACTIVE, &setup_p->flags);
+
+               if (setup_q->param_valid & VFPF_TXQ_VALID) {
+                       struct bnx2x_txq_setup_params *txq_params =
+                               &setup_p->txq_params;
+
+                       __set_bit(BNX2X_Q_TYPE_HAS_TX, &q_type);
+
+                       /* save sb resource index */
+                       q->sb_idx = setup_q->txq.vf_sb;
+
+                       /* tx init */
+                       init_p->tx.hc_rate = setup_q->txq.hc_rate;
+                       init_p->tx.sb_cq_index = setup_q->txq.sb_index;
+
+                       bnx2x_vf_mbx_set_q_flags(setup_q->txq.flags,
+                                                &init_p->tx.flags);
+
+                       /* tx setup - flags */
+                       bnx2x_vf_mbx_set_q_flags(setup_q->txq.flags,
+                                                &setup_p->flags);
+
+                       /* tx setup - general, nothing */
+
+                       /* tx setup - tx */
+                       txq_params->dscr_map = setup_q->txq.txq_addr;
+                       txq_params->sb_cq_index = setup_q->txq.sb_index;
+                       txq_params->traffic_type = setup_q->txq.traffic_type;
+
+                       bnx2x_vfop_qctor_dump_tx(bp, vf, init_p, setup_p,
+                                                q->index, q->sb_idx);
+               }
+
+               if (setup_q->param_valid & VFPF_RXQ_VALID) {
+                       struct bnx2x_rxq_setup_params *rxq_params =
+                                                       &setup_p->rxq_params;
+
+                       __set_bit(BNX2X_Q_TYPE_HAS_RX, &q_type);
+
+                       /* Note: there is no support for different SBs
+                        * for TX and RX
+                        */
+                       q->sb_idx = setup_q->rxq.vf_sb;
+
+                       /* rx init */
+                       init_p->rx.hc_rate = setup_q->rxq.hc_rate;
+                       init_p->rx.sb_cq_index = setup_q->rxq.sb_index;
+                       bnx2x_vf_mbx_set_q_flags(setup_q->rxq.flags,
+                                                &init_p->rx.flags);
+
+                       /* rx setup - flags */
+                       bnx2x_vf_mbx_set_q_flags(setup_q->rxq.flags,
+                                                &setup_p->flags);
+
+                       /* rx setup - general */
+                       setup_p->gen_params.mtu = setup_q->rxq.mtu;
+
+                       /* rx setup - rx */
+                       rxq_params->drop_flags = setup_q->rxq.drop_flags;
+                       rxq_params->dscr_map = setup_q->rxq.rxq_addr;
+                       rxq_params->sge_map = setup_q->rxq.sge_addr;
+                       rxq_params->rcq_map = setup_q->rxq.rcq_addr;
+                       rxq_params->rcq_np_map = setup_q->rxq.rcq_np_addr;
+                       rxq_params->buf_sz = setup_q->rxq.buf_sz;
+                       rxq_params->tpa_agg_sz = setup_q->rxq.tpa_agg_sz;
+                       rxq_params->max_sges_pkt = setup_q->rxq.max_sge_pkt;
+                       rxq_params->sge_buf_sz = setup_q->rxq.sge_buf_sz;
+                       rxq_params->cache_line_log =
+                               setup_q->rxq.cache_line_log;
+                       rxq_params->sb_cq_index = setup_q->rxq.sb_index;
+
+                       bnx2x_vfop_qctor_dump_rx(bp, vf, init_p, setup_p,
+                                                q->index, q->sb_idx);
+               }
+               /* complete the preparations */
+               bnx2x_vfop_qctor_prep(bp, vf, q, &vf->op_params.qctor, q_type);
+
+               vf->op_rc = bnx2x_vfop_qsetup_cmd(bp, vf, &cmd, q->index);
+               if (vf->op_rc)
+                       goto response;
+               return;
+       }
+response:
+       bnx2x_vf_mbx_resp(bp, vf);
+}
+
 /* dispatch request */
 static void bnx2x_vf_mbx_request(struct bnx2x *bp, struct bnx2x_virtf *vf,
                                  struct bnx2x_vf_mbx *mbx)
@@ -391,6 +534,9 @@ static void bnx2x_vf_mbx_request(struct bnx2x *bp, struct bnx2x_virtf *vf,
                case CHANNEL_TLV_INIT:
                        bnx2x_vf_mbx_init_vf(bp, vf, mbx);
                        break;
+               case CHANNEL_TLV_SETUP_Q:
+                       bnx2x_vf_mbx_setup_q(bp, vf, mbx);
+                       break;
                }
        } else {
                /* unknown TLV - this may belong to a VF driver from the future