Merge branch 'hinic-add-SR-IOV-support'
Luo bin says: ==================== hinic: add SR-IOV support patch #1 adds mailbox channel support and vf can communicate with pf or hw through it. patch #2 adds support for enabling vf and tx/rx capabilities based on vf. patch #3 adds support for vf's basic configurations. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
06b439de5f
@ -4,4 +4,4 @@ obj-$(CONFIG_HINIC) += hinic.o
|
||||
hinic-y := hinic_main.o hinic_tx.o hinic_rx.o hinic_port.o hinic_hw_dev.o \
|
||||
hinic_hw_io.o hinic_hw_qp.o hinic_hw_cmdq.o hinic_hw_wq.o \
|
||||
hinic_hw_mgmt.o hinic_hw_api_cmd.o hinic_hw_eqs.o hinic_hw_if.o \
|
||||
hinic_common.o hinic_ethtool.o
|
||||
hinic_common.o hinic_ethtool.o hinic_hw_mbox.o hinic_sriov.o
|
||||
|
@ -16,6 +16,7 @@
|
||||
#include "hinic_hw_dev.h"
|
||||
#include "hinic_tx.h"
|
||||
#include "hinic_rx.h"
|
||||
#include "hinic_sriov.h"
|
||||
|
||||
#define HINIC_DRV_NAME "hinic"
|
||||
|
||||
@ -23,6 +24,7 @@ enum hinic_flags {
|
||||
HINIC_LINK_UP = BIT(0),
|
||||
HINIC_INTF_UP = BIT(1),
|
||||
HINIC_RSS_ENABLE = BIT(2),
|
||||
HINIC_LINK_DOWN = BIT(3),
|
||||
};
|
||||
|
||||
struct hinic_rx_mode_work {
|
||||
@ -78,6 +80,7 @@ struct hinic_dev {
|
||||
struct hinic_rss_type rss_type;
|
||||
u8 *rss_hkey_user;
|
||||
s32 *rss_indir_user;
|
||||
struct hinic_sriov_info sriov_info;
|
||||
};
|
||||
|
||||
#endif
|
||||
|
@ -64,7 +64,7 @@
|
||||
#define CMDQ_WQE_SIZE 64
|
||||
#define CMDQ_DEPTH SZ_4K
|
||||
|
||||
#define CMDQ_WQ_PAGE_SIZE SZ_4K
|
||||
#define CMDQ_WQ_PAGE_SIZE SZ_256K
|
||||
|
||||
#define WQE_LCMD_SIZE 64
|
||||
#define WQE_SCMD_SIZE 64
|
||||
@ -705,7 +705,7 @@ static void cmdq_init_queue_ctxt(struct hinic_cmdq_ctxt *cmdq_ctxt,
|
||||
/* The data in the HW is in Big Endian Format */
|
||||
wq_first_page_paddr = be64_to_cpu(*wq->block_vaddr);
|
||||
|
||||
pfn = CMDQ_PFN(wq_first_page_paddr, wq->wq_page_size);
|
||||
pfn = CMDQ_PFN(wq_first_page_paddr, SZ_4K);
|
||||
|
||||
ctxt_info->curr_wqe_page_pfn =
|
||||
HINIC_CMDQ_CTXT_PAGE_INFO_SET(pfn, CURR_WQE_PAGE_PFN) |
|
||||
@ -714,16 +714,19 @@ static void cmdq_init_queue_ctxt(struct hinic_cmdq_ctxt *cmdq_ctxt,
|
||||
HINIC_CMDQ_CTXT_PAGE_INFO_SET(1, CEQ_EN) |
|
||||
HINIC_CMDQ_CTXT_PAGE_INFO_SET(cmdq->wrapped, WRAPPED);
|
||||
|
||||
/* block PFN - Read Modify Write */
|
||||
cmdq_first_block_paddr = cmdq_pages->page_paddr;
|
||||
if (wq->num_q_pages != 1) {
|
||||
/* block PFN - Read Modify Write */
|
||||
cmdq_first_block_paddr = cmdq_pages->page_paddr;
|
||||
|
||||
pfn = CMDQ_PFN(cmdq_first_block_paddr, wq->wq_page_size);
|
||||
pfn = CMDQ_PFN(cmdq_first_block_paddr, wq->wq_page_size);
|
||||
}
|
||||
|
||||
ctxt_info->wq_block_pfn =
|
||||
HINIC_CMDQ_CTXT_BLOCK_INFO_SET(pfn, WQ_BLOCK_PFN) |
|
||||
HINIC_CMDQ_CTXT_BLOCK_INFO_SET(atomic_read(&wq->cons_idx), CI);
|
||||
|
||||
cmdq_ctxt->func_idx = HINIC_HWIF_FUNC_IDX(cmdqs->hwif);
|
||||
cmdq_ctxt->ppf_idx = HINIC_HWIF_PPF_IDX(cmdqs->hwif);
|
||||
cmdq_ctxt->cmdq_type = cmdq->cmdq_type;
|
||||
}
|
||||
|
||||
@ -795,11 +798,6 @@ static int init_cmdqs_ctxt(struct hinic_hwdev *hwdev,
|
||||
size_t cmdq_ctxts_size;
|
||||
int err;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
cmdq_ctxts_size = HINIC_MAX_CMDQ_TYPES * sizeof(*cmdq_ctxts);
|
||||
cmdq_ctxts = devm_kzalloc(&pdev->dev, cmdq_ctxts_size, GFP_KERNEL);
|
||||
if (!cmdq_ctxts)
|
||||
|
@ -122,7 +122,7 @@ struct hinic_cmdq_ctxt {
|
||||
|
||||
u16 func_idx;
|
||||
u8 cmdq_type;
|
||||
u8 rsvd1[1];
|
||||
u8 ppf_idx;
|
||||
|
||||
u8 rsvd2[4];
|
||||
|
||||
|
@ -10,7 +10,7 @@
|
||||
/* HW interface registers */
|
||||
#define HINIC_CSR_FUNC_ATTR0_ADDR 0x0
|
||||
#define HINIC_CSR_FUNC_ATTR1_ADDR 0x4
|
||||
|
||||
#define HINIC_CSR_FUNC_ATTR2_ADDR 0x8
|
||||
#define HINIC_CSR_FUNC_ATTR4_ADDR 0x10
|
||||
#define HINIC_CSR_FUNC_ATTR5_ADDR 0x14
|
||||
|
||||
|
@ -15,7 +15,9 @@
|
||||
#include <linux/jiffies.h>
|
||||
#include <linux/log2.h>
|
||||
#include <linux/err.h>
|
||||
#include <linux/netdevice.h>
|
||||
|
||||
#include "hinic_sriov.h"
|
||||
#include "hinic_hw_if.h"
|
||||
#include "hinic_hw_eqs.h"
|
||||
#include "hinic_hw_mgmt.h"
|
||||
@ -46,20 +48,6 @@ enum hw_ioctxt_set_cmdq_depth {
|
||||
HW_IOCTXT_SET_CMDQ_DEPTH_DEFAULT,
|
||||
};
|
||||
|
||||
/* HW struct */
|
||||
struct hinic_dev_cap {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u8 rsvd1[5];
|
||||
u8 intr_type;
|
||||
u8 rsvd2[66];
|
||||
u16 max_sqs;
|
||||
u16 max_rqs;
|
||||
u8 rsvd3[208];
|
||||
};
|
||||
|
||||
/**
|
||||
* get_capability - convert device capabilities to NIC capabilities
|
||||
* @hwdev: the HW device to set and convert device capabilities for
|
||||
@ -67,16 +55,13 @@ struct hinic_dev_cap {
|
||||
*
|
||||
* Return 0 - Success, negative - Failure
|
||||
**/
|
||||
static int get_capability(struct hinic_hwdev *hwdev,
|
||||
struct hinic_dev_cap *dev_cap)
|
||||
static int parse_capability(struct hinic_hwdev *hwdev,
|
||||
struct hinic_dev_cap *dev_cap)
|
||||
{
|
||||
struct hinic_cap *nic_cap = &hwdev->nic_cap;
|
||||
int num_aeqs, num_ceqs, num_irqs;
|
||||
|
||||
if (!HINIC_IS_PF(hwdev->hwif) && !HINIC_IS_PPF(hwdev->hwif))
|
||||
return -EINVAL;
|
||||
|
||||
if (dev_cap->intr_type != INTR_MSIX_TYPE)
|
||||
if (!HINIC_IS_VF(hwdev->hwif) && dev_cap->intr_type != INTR_MSIX_TYPE)
|
||||
return -EFAULT;
|
||||
|
||||
num_aeqs = HINIC_HWIF_NUM_AEQS(hwdev->hwif);
|
||||
@ -89,13 +74,19 @@ static int get_capability(struct hinic_hwdev *hwdev,
|
||||
if (nic_cap->num_qps > HINIC_Q_CTXT_MAX)
|
||||
nic_cap->num_qps = HINIC_Q_CTXT_MAX;
|
||||
|
||||
nic_cap->max_qps = dev_cap->max_sqs + 1;
|
||||
if (nic_cap->max_qps != (dev_cap->max_rqs + 1))
|
||||
return -EFAULT;
|
||||
if (!HINIC_IS_VF(hwdev->hwif))
|
||||
nic_cap->max_qps = dev_cap->max_sqs + 1;
|
||||
else
|
||||
nic_cap->max_qps = dev_cap->max_sqs;
|
||||
|
||||
if (nic_cap->num_qps > nic_cap->max_qps)
|
||||
nic_cap->num_qps = nic_cap->max_qps;
|
||||
|
||||
if (!HINIC_IS_VF(hwdev->hwif)) {
|
||||
nic_cap->max_vf = dev_cap->max_vf;
|
||||
nic_cap->max_vf_qps = dev_cap->max_vf_sqs + 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -105,27 +96,26 @@ static int get_capability(struct hinic_hwdev *hwdev,
|
||||
*
|
||||
* Return 0 - Success, negative - Failure
|
||||
**/
|
||||
static int get_cap_from_fw(struct hinic_pfhwdev *pfhwdev)
|
||||
static int get_capability(struct hinic_pfhwdev *pfhwdev)
|
||||
{
|
||||
struct hinic_hwdev *hwdev = &pfhwdev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_dev_cap dev_cap;
|
||||
u16 in_len, out_len;
|
||||
u16 out_len;
|
||||
int err;
|
||||
|
||||
in_len = 0;
|
||||
out_len = sizeof(dev_cap);
|
||||
|
||||
err = hinic_msg_to_mgmt(&pfhwdev->pf_to_mgmt, HINIC_MOD_CFGM,
|
||||
HINIC_CFG_NIC_CAP, &dev_cap, in_len, &dev_cap,
|
||||
&out_len, HINIC_MGMT_MSG_SYNC);
|
||||
HINIC_CFG_NIC_CAP, &dev_cap, sizeof(dev_cap),
|
||||
&dev_cap, &out_len, HINIC_MGMT_MSG_SYNC);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to get capability from FW\n");
|
||||
return err;
|
||||
}
|
||||
|
||||
return get_capability(hwdev, &dev_cap);
|
||||
return parse_capability(hwdev, &dev_cap);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -144,15 +134,14 @@ static int get_dev_cap(struct hinic_hwdev *hwdev)
|
||||
switch (HINIC_FUNC_TYPE(hwif)) {
|
||||
case HINIC_PPF:
|
||||
case HINIC_PF:
|
||||
case HINIC_VF:
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
err = get_cap_from_fw(pfhwdev);
|
||||
err = get_capability(pfhwdev);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to get capability from FW\n");
|
||||
dev_err(&pdev->dev, "Failed to get capability\n");
|
||||
return err;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
@ -225,15 +214,8 @@ static void disable_msix(struct hinic_hwdev *hwdev)
|
||||
int hinic_port_msg_cmd(struct hinic_hwdev *hwdev, enum hinic_port_cmd cmd,
|
||||
void *buf_in, u16 in_size, void *buf_out, u16 *out_size)
|
||||
{
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
return hinic_msg_to_mgmt(&pfhwdev->pf_to_mgmt, HINIC_MOD_L2NIC, cmd,
|
||||
@ -252,14 +234,9 @@ static int init_fw_ctxt(struct hinic_hwdev *hwdev)
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_cmd_fw_ctxt fw_ctxt;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(fw_ctxt);
|
||||
int err;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
fw_ctxt.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
fw_ctxt.rx_buf_sz = HINIC_RX_BUF_SZ;
|
||||
|
||||
@ -288,14 +265,8 @@ static int set_hw_ioctxt(struct hinic_hwdev *hwdev, unsigned int rq_depth,
|
||||
{
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct hinic_cmd_hw_ioctxt hw_ioctxt;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
hw_ioctxt.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
hw_ioctxt.ppf_idx = HINIC_HWIF_PPF_IDX(hwif);
|
||||
|
||||
@ -374,11 +345,6 @@ static int clear_io_resources(struct hinic_hwdev *hwdev)
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
int err;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* sleep 100ms to wait for firmware stopping I/O */
|
||||
msleep(100);
|
||||
|
||||
@ -410,14 +376,8 @@ static int set_resources_state(struct hinic_hwdev *hwdev,
|
||||
{
|
||||
struct hinic_cmd_set_res_state res_state;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
res_state.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
res_state.state = state;
|
||||
|
||||
@ -441,8 +401,8 @@ static int get_base_qpn(struct hinic_hwdev *hwdev, u16 *base_qpn)
|
||||
{
|
||||
struct hinic_cmd_base_qpn cmd_base_qpn;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(cmd_base_qpn);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
cmd_base_qpn.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -488,7 +448,7 @@ int hinic_hwdev_ifup(struct hinic_hwdev *hwdev)
|
||||
num_ceqs = HINIC_HWIF_NUM_CEQS(hwif);
|
||||
|
||||
ceq_msix_entries = &hwdev->msix_entries[num_aeqs];
|
||||
|
||||
func_to_io->hwdev = hwdev;
|
||||
err = hinic_io_init(func_to_io, hwif, nic_cap->max_qps, num_ceqs,
|
||||
ceq_msix_entries);
|
||||
if (err) {
|
||||
@ -558,17 +518,10 @@ void hinic_hwdev_cb_register(struct hinic_hwdev *hwdev,
|
||||
u16 in_size, void *buf_out,
|
||||
u16 *out_size))
|
||||
{
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
struct hinic_nic_cb *nic_cb;
|
||||
u8 cmd_cb;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "unsupported PCI Function type\n");
|
||||
return;
|
||||
}
|
||||
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
cmd_cb = cmd - HINIC_MGMT_MSG_CMD_BASE;
|
||||
@ -588,15 +541,12 @@ void hinic_hwdev_cb_unregister(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mgmt_msg_cmd cmd)
|
||||
{
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
struct hinic_nic_cb *nic_cb;
|
||||
u8 cmd_cb;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "unsupported PCI Function type\n");
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif))
|
||||
return;
|
||||
}
|
||||
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
@ -676,10 +626,23 @@ static int init_pfhwdev(struct hinic_pfhwdev *pfhwdev)
|
||||
return err;
|
||||
}
|
||||
|
||||
hinic_register_mgmt_msg_cb(&pfhwdev->pf_to_mgmt, HINIC_MOD_L2NIC,
|
||||
pfhwdev, nic_mgmt_msg_handler);
|
||||
err = hinic_func_to_func_init(hwdev);
|
||||
if (err) {
|
||||
dev_err(&hwif->pdev->dev, "Failed to init mailbox\n");
|
||||
hinic_pf_to_mgmt_free(&pfhwdev->pf_to_mgmt);
|
||||
return err;
|
||||
}
|
||||
|
||||
if (!HINIC_IS_VF(hwif))
|
||||
hinic_register_mgmt_msg_cb(&pfhwdev->pf_to_mgmt,
|
||||
HINIC_MOD_L2NIC, pfhwdev,
|
||||
nic_mgmt_msg_handler);
|
||||
else
|
||||
hinic_register_vf_mbox_cb(hwdev, HINIC_MOD_L2NIC,
|
||||
nic_mgmt_msg_handler);
|
||||
|
||||
hinic_set_pf_action(hwif, HINIC_PF_MGMT_ACTIVE);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -693,7 +656,13 @@ static void free_pfhwdev(struct hinic_pfhwdev *pfhwdev)
|
||||
|
||||
hinic_set_pf_action(hwdev->hwif, HINIC_PF_MGMT_INIT);
|
||||
|
||||
hinic_unregister_mgmt_msg_cb(&pfhwdev->pf_to_mgmt, HINIC_MOD_L2NIC);
|
||||
if (!HINIC_IS_VF(hwdev->hwif))
|
||||
hinic_unregister_mgmt_msg_cb(&pfhwdev->pf_to_mgmt,
|
||||
HINIC_MOD_L2NIC);
|
||||
else
|
||||
hinic_unregister_vf_mbox_cb(hwdev, HINIC_MOD_L2NIC);
|
||||
|
||||
hinic_func_to_func_free(hwdev);
|
||||
|
||||
hinic_pf_to_mgmt_free(&pfhwdev->pf_to_mgmt);
|
||||
}
|
||||
@ -723,12 +692,6 @@ struct hinic_hwdev *hinic_init_hwdev(struct pci_dev *pdev)
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
err = -EFAULT;
|
||||
goto err_func_type;
|
||||
}
|
||||
|
||||
pfhwdev = devm_kzalloc(&pdev->dev, sizeof(*pfhwdev), GFP_KERNEL);
|
||||
if (!pfhwdev) {
|
||||
err = -ENOMEM;
|
||||
@ -772,6 +735,12 @@ struct hinic_hwdev *hinic_init_hwdev(struct pci_dev *pdev)
|
||||
goto err_dev_cap;
|
||||
}
|
||||
|
||||
err = hinic_vf_func_init(hwdev);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to init nic mbox\n");
|
||||
goto err_vf_func_init;
|
||||
}
|
||||
|
||||
err = init_fw_ctxt(hwdev);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to init function table\n");
|
||||
@ -788,6 +757,8 @@ struct hinic_hwdev *hinic_init_hwdev(struct pci_dev *pdev)
|
||||
|
||||
err_resources_state:
|
||||
err_init_fw_ctxt:
|
||||
hinic_vf_func_free(hwdev);
|
||||
err_vf_func_init:
|
||||
err_dev_cap:
|
||||
free_pfhwdev(pfhwdev);
|
||||
|
||||
@ -799,7 +770,6 @@ err_aeqs_init:
|
||||
|
||||
err_init_msix:
|
||||
err_pfhwdev_alloc:
|
||||
err_func_type:
|
||||
hinic_free_hwif(hwif);
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
@ -930,15 +900,9 @@ int hinic_hwdev_hw_ci_addr_set(struct hinic_hwdev *hwdev, struct hinic_sq *sq,
|
||||
{
|
||||
struct hinic_qp *qp = container_of(sq, struct hinic_qp, sq);
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
struct hinic_cmd_hw_ci hw_ci;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "Unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
hw_ci.dma_attr_off = 0;
|
||||
hw_ci.pending_limit = pending_limit;
|
||||
hw_ci.coalesc_timer = coalesc_timer;
|
||||
|
@ -16,18 +16,27 @@
|
||||
#include "hinic_hw_mgmt.h"
|
||||
#include "hinic_hw_qp.h"
|
||||
#include "hinic_hw_io.h"
|
||||
#include "hinic_hw_mbox.h"
|
||||
|
||||
#define HINIC_MAX_QPS 32
|
||||
|
||||
#define HINIC_MGMT_NUM_MSG_CMD (HINIC_MGMT_MSG_CMD_MAX - \
|
||||
HINIC_MGMT_MSG_CMD_BASE)
|
||||
|
||||
#define HINIC_PF_SET_VF_ALREADY 0x4
|
||||
#define HINIC_MGMT_STATUS_EXIST 0x6
|
||||
|
||||
struct hinic_cap {
|
||||
u16 max_qps;
|
||||
u16 num_qps;
|
||||
u8 max_vf;
|
||||
u16 max_vf_qps;
|
||||
};
|
||||
|
||||
enum hinic_port_cmd {
|
||||
HINIC_PORT_CMD_VF_REGISTER = 0x0,
|
||||
HINIC_PORT_CMD_VF_UNREGISTER = 0x1,
|
||||
|
||||
HINIC_PORT_CMD_CHANGE_MTU = 2,
|
||||
|
||||
HINIC_PORT_CMD_ADD_VLAN = 3,
|
||||
@ -83,10 +92,18 @@ enum hinic_port_cmd {
|
||||
|
||||
HINIC_PORT_CMD_GET_GLOBAL_QPN = 102,
|
||||
|
||||
HINIC_PORT_CMD_SET_VF_VLAN = 106,
|
||||
|
||||
HINIC_PORT_CMD_CLR_VF_VLAN,
|
||||
|
||||
HINIC_PORT_CMD_SET_TSO = 112,
|
||||
|
||||
HINIC_PORT_CMD_SET_RQ_IQ_MAP = 115,
|
||||
|
||||
HINIC_PORT_CMD_LINK_STATUS_REPORT = 160,
|
||||
|
||||
HINIC_PORT_CMD_UPDATE_MAC = 164,
|
||||
|
||||
HINIC_PORT_CMD_GET_CAP = 170,
|
||||
|
||||
HINIC_PORT_CMD_SET_LRO_TIMER = 244,
|
||||
@ -191,6 +208,17 @@ struct hinic_cmd_set_res_state {
|
||||
u32 rsvd2;
|
||||
};
|
||||
|
||||
struct hinic_ceq_ctrl_reg {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u16 func_id;
|
||||
u16 q_id;
|
||||
u32 ctrl0;
|
||||
u32 ctrl1;
|
||||
};
|
||||
|
||||
struct hinic_cmd_base_qpn {
|
||||
u8 status;
|
||||
u8 version;
|
||||
@ -225,6 +253,7 @@ struct hinic_hwdev {
|
||||
|
||||
struct hinic_aeqs aeqs;
|
||||
struct hinic_func_to_io func_to_io;
|
||||
struct hinic_mbox_func_to_func *func_to_func;
|
||||
|
||||
struct hinic_cap nic_cap;
|
||||
};
|
||||
@ -246,6 +275,25 @@ struct hinic_pfhwdev {
|
||||
struct hinic_nic_cb nic_cb[HINIC_MGMT_NUM_MSG_CMD];
|
||||
};
|
||||
|
||||
struct hinic_dev_cap {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u8 rsvd1[5];
|
||||
u8 intr_type;
|
||||
u8 max_cos_id;
|
||||
u8 er_id;
|
||||
u8 port_id;
|
||||
u8 max_vf;
|
||||
u8 rsvd2[62];
|
||||
u16 max_sqs;
|
||||
u16 max_rqs;
|
||||
u16 max_vf_sqs;
|
||||
u16 max_vf_rqs;
|
||||
u8 rsvd3[204];
|
||||
};
|
||||
|
||||
void hinic_hwdev_cb_register(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mgmt_msg_cmd cmd, void *handle,
|
||||
void (*handler)(void *handle, void *buf_in,
|
||||
|
@ -17,6 +17,7 @@
|
||||
#include <asm/byteorder.h>
|
||||
#include <asm/barrier.h>
|
||||
|
||||
#include "hinic_hw_dev.h"
|
||||
#include "hinic_hw_csr.h"
|
||||
#include "hinic_hw_if.h"
|
||||
#include "hinic_hw_eqs.h"
|
||||
@ -416,11 +417,11 @@ static irqreturn_t ceq_interrupt(int irq, void *data)
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static void set_ctrl0(struct hinic_eq *eq)
|
||||
static u32 get_ctrl0_val(struct hinic_eq *eq, u32 addr)
|
||||
{
|
||||
struct msix_entry *msix_entry = &eq->msix_entry;
|
||||
enum hinic_eq_type type = eq->type;
|
||||
u32 addr, val, ctrl0;
|
||||
u32 val, ctrl0;
|
||||
|
||||
if (type == HINIC_AEQ) {
|
||||
/* RMW Ctrl0 */
|
||||
@ -440,9 +441,7 @@ static void set_ctrl0(struct hinic_eq *eq)
|
||||
HINIC_AEQ_CTRL_0_SET(EQ_INT_MODE_ARMED, INT_MODE);
|
||||
|
||||
val |= ctrl0;
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
} else if (type == HINIC_CEQ) {
|
||||
} else {
|
||||
/* RMW Ctrl0 */
|
||||
addr = HINIC_CSR_CEQ_CTRL_0_ADDR(eq->q_id);
|
||||
|
||||
@ -462,16 +461,28 @@ static void set_ctrl0(struct hinic_eq *eq)
|
||||
HINIC_CEQ_CTRL_0_SET(EQ_INT_MODE_ARMED, INTR_MODE);
|
||||
|
||||
val |= ctrl0;
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
static void set_ctrl1(struct hinic_eq *eq)
|
||||
static void set_ctrl0(struct hinic_eq *eq)
|
||||
{
|
||||
u32 val, addr;
|
||||
|
||||
if (eq->type == HINIC_AEQ)
|
||||
addr = HINIC_CSR_AEQ_CTRL_0_ADDR(eq->q_id);
|
||||
else
|
||||
addr = HINIC_CSR_CEQ_CTRL_0_ADDR(eq->q_id);
|
||||
|
||||
val = get_ctrl0_val(eq, addr);
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
}
|
||||
|
||||
static u32 get_ctrl1_val(struct hinic_eq *eq, u32 addr)
|
||||
{
|
||||
u32 page_size_val, elem_size, val, ctrl1;
|
||||
enum hinic_eq_type type = eq->type;
|
||||
u32 page_size_val, elem_size;
|
||||
u32 addr, val, ctrl1;
|
||||
|
||||
if (type == HINIC_AEQ) {
|
||||
/* RMW Ctrl1 */
|
||||
@ -491,9 +502,7 @@ static void set_ctrl1(struct hinic_eq *eq)
|
||||
HINIC_AEQ_CTRL_1_SET(page_size_val, PAGE_SIZE);
|
||||
|
||||
val |= ctrl1;
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
} else if (type == HINIC_CEQ) {
|
||||
} else {
|
||||
/* RMW Ctrl1 */
|
||||
addr = HINIC_CSR_CEQ_CTRL_1_ADDR(eq->q_id);
|
||||
|
||||
@ -508,19 +517,70 @@ static void set_ctrl1(struct hinic_eq *eq)
|
||||
HINIC_CEQ_CTRL_1_SET(page_size_val, PAGE_SIZE);
|
||||
|
||||
val |= ctrl1;
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
static void set_ctrl1(struct hinic_eq *eq)
|
||||
{
|
||||
u32 addr, val;
|
||||
|
||||
if (eq->type == HINIC_AEQ)
|
||||
addr = HINIC_CSR_AEQ_CTRL_1_ADDR(eq->q_id);
|
||||
else
|
||||
addr = HINIC_CSR_CEQ_CTRL_1_ADDR(eq->q_id);
|
||||
|
||||
val = get_ctrl1_val(eq, addr);
|
||||
|
||||
hinic_hwif_write_reg(eq->hwif, addr, val);
|
||||
}
|
||||
|
||||
static int set_ceq_ctrl_reg(struct hinic_eq *eq)
|
||||
{
|
||||
struct hinic_ceq_ctrl_reg ceq_ctrl = {0};
|
||||
struct hinic_hwdev *hwdev = eq->hwdev;
|
||||
u16 out_size = sizeof(ceq_ctrl);
|
||||
u16 in_size = sizeof(ceq_ctrl);
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
u32 addr;
|
||||
int err;
|
||||
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
addr = HINIC_CSR_CEQ_CTRL_0_ADDR(eq->q_id);
|
||||
ceq_ctrl.ctrl0 = get_ctrl0_val(eq, addr);
|
||||
addr = HINIC_CSR_CEQ_CTRL_1_ADDR(eq->q_id);
|
||||
ceq_ctrl.ctrl1 = get_ctrl1_val(eq, addr);
|
||||
|
||||
ceq_ctrl.func_id = HINIC_HWIF_FUNC_IDX(hwdev->hwif);
|
||||
ceq_ctrl.q_id = eq->q_id;
|
||||
|
||||
err = hinic_msg_to_mgmt(&pfhwdev->pf_to_mgmt, HINIC_MOD_COMM,
|
||||
HINIC_COMM_CMD_CEQ_CTRL_REG_WR_BY_UP,
|
||||
&ceq_ctrl, in_size,
|
||||
&ceq_ctrl, &out_size, HINIC_MGMT_MSG_SYNC);
|
||||
if (err || !out_size || ceq_ctrl.status) {
|
||||
dev_err(&hwdev->hwif->pdev->dev,
|
||||
"Failed to set ceq %d ctrl reg, err: %d status: 0x%x, out_size: 0x%x\n",
|
||||
eq->q_id, err, ceq_ctrl.status, out_size);
|
||||
return -EFAULT;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* set_eq_ctrls - setting eq's ctrl registers
|
||||
* @eq: the Event Queue for setting
|
||||
**/
|
||||
static void set_eq_ctrls(struct hinic_eq *eq)
|
||||
static int set_eq_ctrls(struct hinic_eq *eq)
|
||||
{
|
||||
if (HINIC_IS_VF(eq->hwif) && eq->type == HINIC_CEQ)
|
||||
return set_ceq_ctrl_reg(eq);
|
||||
|
||||
set_ctrl0(eq);
|
||||
set_ctrl1(eq);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -703,7 +763,12 @@ static int init_eq(struct hinic_eq *eq, struct hinic_hwif *hwif,
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
set_eq_ctrls(eq);
|
||||
err = set_eq_ctrls(eq);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to set eq ctrls\n");
|
||||
return err;
|
||||
}
|
||||
|
||||
eq_update_ci(eq, EQ_ARMED);
|
||||
|
||||
err = alloc_eq_pages(eq);
|
||||
@ -859,6 +924,7 @@ int hinic_ceqs_init(struct hinic_ceqs *ceqs, struct hinic_hwif *hwif,
|
||||
ceqs->num_ceqs = num_ceqs;
|
||||
|
||||
for (q_id = 0; q_id < num_ceqs; q_id++) {
|
||||
ceqs->ceq[q_id].hwdev = ceqs->hwdev;
|
||||
err = init_eq(&ceqs->ceq[q_id], hwif, HINIC_CEQ, q_id, q_len,
|
||||
page_size, msix_entries[q_id]);
|
||||
if (err) {
|
||||
|
@ -143,8 +143,9 @@ enum hinic_eq_type {
|
||||
};
|
||||
|
||||
enum hinic_aeq_type {
|
||||
HINIC_MBX_FROM_FUNC = 1,
|
||||
HINIC_MSG_FROM_MGMT_CPU = 2,
|
||||
|
||||
HINIC_MBX_SEND_RSLT = 5,
|
||||
HINIC_MAX_AEQ_EVENTS,
|
||||
};
|
||||
|
||||
@ -171,7 +172,7 @@ struct hinic_eq_work {
|
||||
|
||||
struct hinic_eq {
|
||||
struct hinic_hwif *hwif;
|
||||
|
||||
struct hinic_hwdev *hwdev;
|
||||
enum hinic_eq_type type;
|
||||
int q_id;
|
||||
u32 q_len;
|
||||
@ -219,7 +220,7 @@ struct hinic_ceq_cb {
|
||||
|
||||
struct hinic_ceqs {
|
||||
struct hinic_hwif *hwif;
|
||||
|
||||
struct hinic_hwdev *hwdev;
|
||||
struct hinic_eq ceq[HINIC_MAX_CEQS];
|
||||
int num_ceqs;
|
||||
|
||||
|
@ -115,8 +115,12 @@ int hinic_msix_attr_cnt_clear(struct hinic_hwif *hwif, u16 msix_index)
|
||||
**/
|
||||
void hinic_set_pf_action(struct hinic_hwif *hwif, enum hinic_pf_action action)
|
||||
{
|
||||
u32 attr5 = hinic_hwif_read_reg(hwif, HINIC_CSR_FUNC_ATTR5_ADDR);
|
||||
u32 attr5;
|
||||
|
||||
if (HINIC_IS_VF(hwif))
|
||||
return;
|
||||
|
||||
attr5 = hinic_hwif_read_reg(hwif, HINIC_CSR_FUNC_ATTR5_ADDR);
|
||||
attr5 = HINIC_FA5_CLEAR(attr5, PF_ACTION);
|
||||
attr5 |= HINIC_FA5_SET(action, PF_ACTION);
|
||||
|
||||
@ -203,7 +207,8 @@ static int hwif_ready(struct hinic_hwif *hwif)
|
||||
* @attr0: the first attribute that was read from the hw
|
||||
* @attr1: the second attribute that was read from the hw
|
||||
**/
|
||||
static void set_hwif_attr(struct hinic_hwif *hwif, u32 attr0, u32 attr1)
|
||||
static void set_hwif_attr(struct hinic_hwif *hwif, u32 attr0, u32 attr1,
|
||||
u32 attr2)
|
||||
{
|
||||
hwif->attr.func_idx = HINIC_FA0_GET(attr0, FUNC_IDX);
|
||||
hwif->attr.pf_idx = HINIC_FA0_GET(attr0, PF_IDX);
|
||||
@ -214,6 +219,8 @@ static void set_hwif_attr(struct hinic_hwif *hwif, u32 attr0, u32 attr1)
|
||||
hwif->attr.num_ceqs = BIT(HINIC_FA1_GET(attr1, CEQS_PER_FUNC));
|
||||
hwif->attr.num_irqs = BIT(HINIC_FA1_GET(attr1, IRQS_PER_FUNC));
|
||||
hwif->attr.num_dma_attr = BIT(HINIC_FA1_GET(attr1, DMA_ATTR_PER_FUNC));
|
||||
hwif->attr.global_vf_id_of_pf = HINIC_FA2_GET(attr2,
|
||||
GLOBAL_VF_ID_OF_PF);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -222,7 +229,7 @@ static void set_hwif_attr(struct hinic_hwif *hwif, u32 attr0, u32 attr1)
|
||||
**/
|
||||
static void read_hwif_attr(struct hinic_hwif *hwif)
|
||||
{
|
||||
u32 addr, attr0, attr1;
|
||||
u32 addr, attr0, attr1, attr2;
|
||||
|
||||
addr = HINIC_CSR_FUNC_ATTR0_ADDR;
|
||||
attr0 = hinic_hwif_read_reg(hwif, addr);
|
||||
@ -230,7 +237,10 @@ static void read_hwif_attr(struct hinic_hwif *hwif)
|
||||
addr = HINIC_CSR_FUNC_ATTR1_ADDR;
|
||||
attr1 = hinic_hwif_read_reg(hwif, addr);
|
||||
|
||||
set_hwif_attr(hwif, attr0, attr1);
|
||||
addr = HINIC_CSR_FUNC_ATTR2_ADDR;
|
||||
attr2 = hinic_hwif_read_reg(hwif, addr);
|
||||
|
||||
set_hwif_attr(hwif, attr0, attr1, attr2);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -309,6 +319,34 @@ static void dma_attr_init(struct hinic_hwif *hwif)
|
||||
HINIC_PCIE_SNOOP, HINIC_PCIE_TPH_DISABLE);
|
||||
}
|
||||
|
||||
u16 hinic_glb_pf_vf_offset(struct hinic_hwif *hwif)
|
||||
{
|
||||
if (!hwif)
|
||||
return 0;
|
||||
|
||||
return hwif->attr.global_vf_id_of_pf;
|
||||
}
|
||||
|
||||
u16 hinic_global_func_id_hw(struct hinic_hwif *hwif)
|
||||
{
|
||||
u32 addr, attr0;
|
||||
|
||||
addr = HINIC_CSR_FUNC_ATTR0_ADDR;
|
||||
attr0 = hinic_hwif_read_reg(hwif, addr);
|
||||
|
||||
return HINIC_FA0_GET(attr0, FUNC_IDX);
|
||||
}
|
||||
|
||||
u16 hinic_pf_id_of_vf_hw(struct hinic_hwif *hwif)
|
||||
{
|
||||
u32 addr, attr0;
|
||||
|
||||
addr = HINIC_CSR_FUNC_ATTR0_ADDR;
|
||||
attr0 = hinic_hwif_read_reg(hwif, addr);
|
||||
|
||||
return HINIC_FA0_GET(attr0, PF_IDX);
|
||||
}
|
||||
|
||||
/**
|
||||
* hinic_init_hwif - initialize the hw interface
|
||||
* @hwif: the HW interface of a pci function device
|
||||
|
@ -35,6 +35,7 @@
|
||||
#define HINIC_FA0_FUNC_IDX_SHIFT 0
|
||||
#define HINIC_FA0_PF_IDX_SHIFT 10
|
||||
#define HINIC_FA0_PCI_INTF_IDX_SHIFT 14
|
||||
#define HINIC_FA0_VF_IN_PF_SHIFT 16
|
||||
/* reserved members - off 16 */
|
||||
#define HINIC_FA0_FUNC_TYPE_SHIFT 24
|
||||
|
||||
@ -42,6 +43,7 @@
|
||||
#define HINIC_FA0_PF_IDX_MASK 0xF
|
||||
#define HINIC_FA0_PCI_INTF_IDX_MASK 0x3
|
||||
#define HINIC_FA0_FUNC_TYPE_MASK 0x1
|
||||
#define HINIC_FA0_VF_IN_PF_MASK 0xFF
|
||||
|
||||
#define HINIC_FA0_GET(val, member) \
|
||||
(((val) >> HINIC_FA0_##member##_SHIFT) & HINIC_FA0_##member##_MASK)
|
||||
@ -64,6 +66,12 @@
|
||||
#define HINIC_FA1_GET(val, member) \
|
||||
(((val) >> HINIC_FA1_##member##_SHIFT) & HINIC_FA1_##member##_MASK)
|
||||
|
||||
#define HINIC_FA2_GLOBAL_VF_ID_OF_PF_SHIFT 16
|
||||
#define HINIC_FA2_GLOBAL_VF_ID_OF_PF_MASK 0x3FF
|
||||
|
||||
#define HINIC_FA2_GET(val, member) \
|
||||
(((val) >> HINIC_FA2_##member##_SHIFT) & HINIC_FA2_##member##_MASK)
|
||||
|
||||
#define HINIC_FA4_OUTBOUND_STATE_SHIFT 0
|
||||
#define HINIC_FA4_DB_STATE_SHIFT 1
|
||||
|
||||
@ -140,6 +148,7 @@
|
||||
#define HINIC_HWIF_PPF_IDX(hwif) ((hwif)->attr.ppf_idx)
|
||||
|
||||
#define HINIC_FUNC_TYPE(hwif) ((hwif)->attr.func_type)
|
||||
#define HINIC_IS_VF(hwif) (HINIC_FUNC_TYPE(hwif) == HINIC_VF)
|
||||
#define HINIC_IS_PF(hwif) (HINIC_FUNC_TYPE(hwif) == HINIC_PF)
|
||||
#define HINIC_IS_PPF(hwif) (HINIC_FUNC_TYPE(hwif) == HINIC_PPF)
|
||||
|
||||
@ -173,6 +182,7 @@ enum hinic_pcie_tph {
|
||||
|
||||
enum hinic_func_type {
|
||||
HINIC_PF = 0,
|
||||
HINIC_VF = 1,
|
||||
HINIC_PPF = 2,
|
||||
};
|
||||
|
||||
@ -223,6 +233,8 @@ struct hinic_func_attr {
|
||||
u8 num_ceqs;
|
||||
|
||||
u8 num_dma_attr;
|
||||
|
||||
u16 global_vf_id_of_pf;
|
||||
};
|
||||
|
||||
struct hinic_hwif {
|
||||
@ -271,6 +283,12 @@ enum hinic_db_state hinic_db_state_get(struct hinic_hwif *hwif);
|
||||
void hinic_db_state_set(struct hinic_hwif *hwif,
|
||||
enum hinic_db_state db_state);
|
||||
|
||||
u16 hinic_glb_pf_vf_offset(struct hinic_hwif *hwif);
|
||||
|
||||
u16 hinic_global_func_id_hw(struct hinic_hwif *hwif);
|
||||
|
||||
u16 hinic_pf_id_of_vf_hw(struct hinic_hwif *hwif);
|
||||
|
||||
int hinic_init_hwif(struct hinic_hwif *hwif, struct pci_dev *pdev);
|
||||
|
||||
void hinic_free_hwif(struct hinic_hwif *hwif);
|
||||
|
@ -15,6 +15,7 @@
|
||||
#include <linux/io.h>
|
||||
#include <linux/err.h>
|
||||
|
||||
#include "hinic_hw_dev.h"
|
||||
#include "hinic_hw_if.h"
|
||||
#include "hinic_hw_eqs.h"
|
||||
#include "hinic_hw_wqe.h"
|
||||
@ -34,6 +35,8 @@
|
||||
#define DB_IDX(db, db_base) \
|
||||
(((unsigned long)(db) - (unsigned long)(db_base)) / HINIC_DB_PAGE_SIZE)
|
||||
|
||||
#define HINIC_PAGE_SIZE_HW(pg_size) ((u8)ilog2((u32)((pg_size) >> 12)))
|
||||
|
||||
enum io_cmd {
|
||||
IO_CMD_MODIFY_QUEUE_CTXT = 0,
|
||||
IO_CMD_CLEAN_QUEUE_CTXT,
|
||||
@ -484,6 +487,33 @@ void hinic_io_destroy_qps(struct hinic_func_to_io *func_to_io, int num_qps)
|
||||
devm_kfree(&pdev->dev, func_to_io->qps);
|
||||
}
|
||||
|
||||
int hinic_set_wq_page_size(struct hinic_hwdev *hwdev, u16 func_idx,
|
||||
u32 page_size)
|
||||
{
|
||||
struct hinic_wq_page_size page_size_info = {0};
|
||||
u16 out_size = sizeof(page_size_info);
|
||||
struct hinic_pfhwdev *pfhwdev;
|
||||
int err;
|
||||
|
||||
pfhwdev = container_of(hwdev, struct hinic_pfhwdev, hwdev);
|
||||
|
||||
page_size_info.func_idx = func_idx;
|
||||
page_size_info.ppf_idx = HINIC_HWIF_PPF_IDX(hwdev->hwif);
|
||||
page_size_info.page_size = HINIC_PAGE_SIZE_HW(page_size);
|
||||
|
||||
err = hinic_msg_to_mgmt(&pfhwdev->pf_to_mgmt, HINIC_MOD_COMM,
|
||||
HINIC_COMM_CMD_PAGESIZE_SET, &page_size_info,
|
||||
sizeof(page_size_info), &page_size_info,
|
||||
&out_size, HINIC_MGMT_MSG_SYNC);
|
||||
if (err || !out_size || page_size_info.status) {
|
||||
dev_err(&hwdev->hwif->pdev->dev, "Failed to set wq page size, err: %d, status: 0x%x, out_size: 0x%0x\n",
|
||||
err, page_size_info.status, out_size);
|
||||
return -EFAULT;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* hinic_io_init - Initialize the IO components
|
||||
* @func_to_io: func to io channel that holds the IO components
|
||||
@ -506,6 +536,7 @@ int hinic_io_init(struct hinic_func_to_io *func_to_io,
|
||||
func_to_io->hwif = hwif;
|
||||
func_to_io->qps = NULL;
|
||||
func_to_io->max_qps = max_qps;
|
||||
func_to_io->ceqs.hwdev = func_to_io->hwdev;
|
||||
|
||||
err = hinic_ceqs_init(&func_to_io->ceqs, hwif, num_ceqs,
|
||||
HINIC_DEFAULT_CEQ_LEN, HINIC_EQ_PAGE_SIZE,
|
||||
@ -541,6 +572,14 @@ int hinic_io_init(struct hinic_func_to_io *func_to_io,
|
||||
func_to_io->cmdq_db_area[cmdq] = db_area;
|
||||
}
|
||||
|
||||
err = hinic_set_wq_page_size(func_to_io->hwdev,
|
||||
HINIC_HWIF_FUNC_IDX(hwif),
|
||||
HINIC_DEFAULT_WQ_PAGE_SIZE);
|
||||
if (err) {
|
||||
dev_err(&func_to_io->hwif->pdev->dev, "Failed to set wq page size\n");
|
||||
goto init_wq_pg_size_err;
|
||||
}
|
||||
|
||||
err = hinic_init_cmdqs(&func_to_io->cmdqs, hwif,
|
||||
func_to_io->cmdq_db_area);
|
||||
if (err) {
|
||||
@ -551,6 +590,11 @@ int hinic_io_init(struct hinic_func_to_io *func_to_io,
|
||||
return 0;
|
||||
|
||||
err_init_cmdqs:
|
||||
if (!HINIC_IS_VF(func_to_io->hwif))
|
||||
hinic_set_wq_page_size(func_to_io->hwdev,
|
||||
HINIC_HWIF_FUNC_IDX(hwif),
|
||||
HINIC_HW_WQ_PAGE_SIZE);
|
||||
init_wq_pg_size_err:
|
||||
err_db_area:
|
||||
for (type = HINIC_CMDQ_SYNC; type < cmdq; type++)
|
||||
return_db_area(func_to_io, func_to_io->cmdq_db_area[type]);
|
||||
@ -575,6 +619,11 @@ void hinic_io_free(struct hinic_func_to_io *func_to_io)
|
||||
|
||||
hinic_free_cmdqs(&func_to_io->cmdqs);
|
||||
|
||||
if (!HINIC_IS_VF(func_to_io->hwif))
|
||||
hinic_set_wq_page_size(func_to_io->hwdev,
|
||||
HINIC_HWIF_FUNC_IDX(func_to_io->hwif),
|
||||
HINIC_HW_WQ_PAGE_SIZE);
|
||||
|
||||
for (cmdq = HINIC_CMDQ_SYNC; cmdq < HINIC_MAX_CMDQ_TYPES; cmdq++)
|
||||
return_db_area(func_to_io, func_to_io->cmdq_db_area[cmdq]);
|
||||
|
||||
|
@ -20,6 +20,8 @@
|
||||
|
||||
#define HINIC_DB_PAGE_SIZE SZ_4K
|
||||
#define HINIC_DB_SIZE SZ_4M
|
||||
#define HINIC_HW_WQ_PAGE_SIZE SZ_4K
|
||||
#define HINIC_DEFAULT_WQ_PAGE_SIZE SZ_256K
|
||||
|
||||
#define HINIC_DB_MAX_AREAS (HINIC_DB_SIZE / HINIC_DB_PAGE_SIZE)
|
||||
|
||||
@ -47,7 +49,7 @@ struct hinic_free_db_area {
|
||||
|
||||
struct hinic_func_to_io {
|
||||
struct hinic_hwif *hwif;
|
||||
|
||||
struct hinic_hwdev *hwdev;
|
||||
struct hinic_ceqs ceqs;
|
||||
|
||||
struct hinic_wqs wqs;
|
||||
@ -69,8 +71,27 @@ struct hinic_func_to_io {
|
||||
void __iomem *cmdq_db_area[HINIC_MAX_CMDQ_TYPES];
|
||||
|
||||
struct hinic_cmdqs cmdqs;
|
||||
|
||||
u16 max_vfs;
|
||||
struct vf_data_storage *vf_infos;
|
||||
u8 link_status;
|
||||
};
|
||||
|
||||
struct hinic_wq_page_size {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u16 func_idx;
|
||||
u8 ppf_idx;
|
||||
u8 page_size;
|
||||
|
||||
u32 rsvd1;
|
||||
};
|
||||
|
||||
int hinic_set_wq_page_size(struct hinic_hwdev *hwdev, u16 func_idx,
|
||||
u32 page_size);
|
||||
|
||||
int hinic_io_create_qps(struct hinic_func_to_io *func_to_io,
|
||||
u16 base_qpn, int num_qps,
|
||||
struct msix_entry *sq_msix_entries,
|
||||
|
1213
drivers/net/ethernet/huawei/hinic/hinic_hw_mbox.c
Normal file
1213
drivers/net/ethernet/huawei/hinic/hinic_hw_mbox.c
Normal file
File diff suppressed because it is too large
Load Diff
154
drivers/net/ethernet/huawei/hinic/hinic_hw_mbox.h
Normal file
154
drivers/net/ethernet/huawei/hinic/hinic_hw_mbox.h
Normal file
@ -0,0 +1,154 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/* Huawei HiNIC PCI Express Linux driver
|
||||
* Copyright(c) 2017 Huawei Technologies Co., Ltd
|
||||
*/
|
||||
|
||||
#ifndef HINIC_MBOX_H_
|
||||
#define HINIC_MBOX_H_
|
||||
|
||||
#define HINIC_MBOX_PF_SEND_ERR 0x1
|
||||
#define HINIC_MBOX_PF_BUSY_ACTIVE_FW 0x2
|
||||
#define HINIC_MBOX_VF_CMD_ERROR 0x3
|
||||
|
||||
#define HINIC_MAX_FUNCTIONS 512
|
||||
|
||||
#define HINIC_MAX_PF_FUNCS 16
|
||||
|
||||
#define HINIC_MBOX_WQ_NAME "hinic_mbox"
|
||||
|
||||
#define HINIC_FUNC_CSR_MAILBOX_DATA_OFF 0x80
|
||||
#define HINIC_FUNC_CSR_MAILBOX_CONTROL_OFF 0x0100
|
||||
#define HINIC_FUNC_CSR_MAILBOX_INT_OFFSET_OFF 0x0104
|
||||
#define HINIC_FUNC_CSR_MAILBOX_RESULT_H_OFF 0x0108
|
||||
#define HINIC_FUNC_CSR_MAILBOX_RESULT_L_OFF 0x010C
|
||||
|
||||
enum hinic_mbox_ack_type {
|
||||
MBOX_ACK,
|
||||
MBOX_NO_ACK,
|
||||
};
|
||||
|
||||
struct mbox_msg_info {
|
||||
u8 msg_id;
|
||||
u8 status;
|
||||
};
|
||||
|
||||
struct hinic_recv_mbox {
|
||||
struct completion recv_done;
|
||||
void *mbox;
|
||||
u8 cmd;
|
||||
enum hinic_mod_type mod;
|
||||
u16 mbox_len;
|
||||
void *buf_out;
|
||||
enum hinic_mbox_ack_type ack_type;
|
||||
struct mbox_msg_info msg_info;
|
||||
u8 seq_id;
|
||||
atomic_t msg_cnt;
|
||||
};
|
||||
|
||||
struct hinic_send_mbox {
|
||||
struct completion send_done;
|
||||
u8 *data;
|
||||
|
||||
u64 *wb_status;
|
||||
void *wb_vaddr;
|
||||
dma_addr_t wb_paddr;
|
||||
};
|
||||
|
||||
typedef void (*hinic_vf_mbox_cb)(void *handle, u8 cmd, void *buf_in,
|
||||
u16 in_size, void *buf_out, u16 *out_size);
|
||||
typedef int (*hinic_pf_mbox_cb)(void *handle, u16 vf_id, u8 cmd, void *buf_in,
|
||||
u16 in_size, void *buf_out, u16 *out_size);
|
||||
|
||||
enum mbox_event_state {
|
||||
EVENT_START = 0,
|
||||
EVENT_FAIL,
|
||||
EVENT_TIMEOUT,
|
||||
EVENT_END,
|
||||
};
|
||||
|
||||
enum hinic_mbox_cb_state {
|
||||
HINIC_VF_MBOX_CB_REG = 0,
|
||||
HINIC_VF_MBOX_CB_RUNNING,
|
||||
HINIC_PF_MBOX_CB_REG,
|
||||
HINIC_PF_MBOX_CB_RUNNING,
|
||||
HINIC_PPF_MBOX_CB_REG,
|
||||
HINIC_PPF_MBOX_CB_RUNNING,
|
||||
HINIC_PPF_TO_PF_MBOX_CB_REG,
|
||||
HINIC_PPF_TO_PF_MBOX_CB_RUNNIG,
|
||||
};
|
||||
|
||||
struct hinic_mbox_func_to_func {
|
||||
struct hinic_hwdev *hwdev;
|
||||
struct hinic_hwif *hwif;
|
||||
|
||||
struct semaphore mbox_send_sem;
|
||||
struct semaphore msg_send_sem;
|
||||
struct hinic_send_mbox send_mbox;
|
||||
|
||||
struct workqueue_struct *workq;
|
||||
|
||||
struct hinic_recv_mbox mbox_resp[HINIC_MAX_FUNCTIONS];
|
||||
struct hinic_recv_mbox mbox_send[HINIC_MAX_FUNCTIONS];
|
||||
|
||||
hinic_vf_mbox_cb vf_mbox_cb[HINIC_MOD_MAX];
|
||||
hinic_pf_mbox_cb pf_mbox_cb[HINIC_MOD_MAX];
|
||||
unsigned long pf_mbox_cb_state[HINIC_MOD_MAX];
|
||||
unsigned long vf_mbox_cb_state[HINIC_MOD_MAX];
|
||||
|
||||
u8 send_msg_id;
|
||||
enum mbox_event_state event_flag;
|
||||
|
||||
/* lock for mbox event flag */
|
||||
spinlock_t mbox_lock;
|
||||
};
|
||||
|
||||
struct hinic_mbox_work {
|
||||
struct work_struct work;
|
||||
u16 src_func_idx;
|
||||
struct hinic_mbox_func_to_func *func_to_func;
|
||||
struct hinic_recv_mbox *recv_mbox;
|
||||
};
|
||||
|
||||
struct vf_cmd_msg_handle {
|
||||
u8 cmd;
|
||||
int (*cmd_msg_handler)(void *hwdev, u16 vf_id,
|
||||
void *buf_in, u16 in_size,
|
||||
void *buf_out, u16 *out_size);
|
||||
};
|
||||
|
||||
int hinic_register_pf_mbox_cb(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mod_type mod,
|
||||
hinic_pf_mbox_cb callback);
|
||||
|
||||
int hinic_register_vf_mbox_cb(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mod_type mod,
|
||||
hinic_vf_mbox_cb callback);
|
||||
|
||||
void hinic_unregister_pf_mbox_cb(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mod_type mod);
|
||||
|
||||
void hinic_unregister_vf_mbox_cb(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mod_type mod);
|
||||
|
||||
void hinic_mbox_func_aeqe_handler(void *handle, void *header, u8 size);
|
||||
|
||||
void hinic_mbox_self_aeqe_handler(void *handle, void *header, u8 size);
|
||||
|
||||
int hinic_func_to_func_init(struct hinic_hwdev *hwdev);
|
||||
|
||||
void hinic_func_to_func_free(struct hinic_hwdev *hwdev);
|
||||
|
||||
int hinic_mbox_to_pf(struct hinic_hwdev *hwdev, enum hinic_mod_type mod,
|
||||
u8 cmd, void *buf_in, u16 in_size, void *buf_out,
|
||||
u16 *out_size, u32 timeout);
|
||||
|
||||
int hinic_mbox_to_func(struct hinic_mbox_func_to_func *func_to_func,
|
||||
enum hinic_mod_type mod, u16 cmd, u16 dst_func,
|
||||
void *buf_in, u16 in_size, void *buf_out,
|
||||
u16 *out_size, u32 timeout);
|
||||
|
||||
int hinic_mbox_to_vf(struct hinic_hwdev *hwdev,
|
||||
enum hinic_mod_type mod, u16 vf_id, u8 cmd, void *buf_in,
|
||||
u16 in_size, void *buf_out, u16 *out_size, u32 timeout);
|
||||
|
||||
#endif
|
@ -353,7 +353,11 @@ int hinic_msg_to_mgmt(struct hinic_pf_to_mgmt *pf_to_mgmt,
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return msg_to_mgmt_sync(pf_to_mgmt, mod, cmd, buf_in, in_size,
|
||||
if (HINIC_IS_VF(hwif))
|
||||
return hinic_mbox_to_pf(pf_to_mgmt->hwdev, mod, cmd, buf_in,
|
||||
in_size, buf_out, out_size, 0);
|
||||
else
|
||||
return msg_to_mgmt_sync(pf_to_mgmt, mod, cmd, buf_in, in_size,
|
||||
buf_out, out_size, MGMT_DIRECT_SEND,
|
||||
MSG_NOT_RESP);
|
||||
}
|
||||
@ -390,8 +394,8 @@ static void mgmt_recv_msg_handler(struct hinic_pf_to_mgmt *pf_to_mgmt,
|
||||
recv_msg->msg, recv_msg->msg_len,
|
||||
buf_out, &out_size);
|
||||
else
|
||||
dev_err(&pdev->dev, "No MGMT msg handler, mod = %d\n",
|
||||
recv_msg->mod);
|
||||
dev_err(&pdev->dev, "No MGMT msg handler, mod: %d, cmd: %d\n",
|
||||
recv_msg->mod, recv_msg->cmd);
|
||||
|
||||
mgmt_cb->state &= ~HINIC_MGMT_CB_RUNNING;
|
||||
|
||||
@ -553,6 +557,10 @@ int hinic_pf_to_mgmt_init(struct hinic_pf_to_mgmt *pf_to_mgmt,
|
||||
int err;
|
||||
|
||||
pf_to_mgmt->hwif = hwif;
|
||||
pf_to_mgmt->hwdev = hwdev;
|
||||
|
||||
if (HINIC_IS_VF(hwif))
|
||||
return 0;
|
||||
|
||||
sema_init(&pf_to_mgmt->sync_msg_lock, 1);
|
||||
pf_to_mgmt->sync_msg_id = 0;
|
||||
@ -584,6 +592,9 @@ void hinic_pf_to_mgmt_free(struct hinic_pf_to_mgmt *pf_to_mgmt)
|
||||
struct hinic_pfhwdev *pfhwdev = mgmt_to_pfhwdev(pf_to_mgmt);
|
||||
struct hinic_hwdev *hwdev = &pfhwdev->hwdev;
|
||||
|
||||
if (HINIC_IS_VF(hwdev->hwif))
|
||||
return;
|
||||
|
||||
hinic_aeq_unregister_hw_cb(&hwdev->aeqs, HINIC_MSG_FROM_MGMT_CPU);
|
||||
hinic_api_cmd_free(pf_to_mgmt->cmd_chain);
|
||||
}
|
||||
|
@ -60,7 +60,9 @@ enum hinic_cfg_cmd {
|
||||
};
|
||||
|
||||
enum hinic_comm_cmd {
|
||||
HINIC_COMM_CMD_START_FLR = 0x1,
|
||||
HINIC_COMM_CMD_IO_STATUS_GET = 0x3,
|
||||
HINIC_COMM_CMD_DMA_ATTR_SET = 0x4,
|
||||
|
||||
HINIC_COMM_CMD_CMDQ_CTXT_SET = 0x10,
|
||||
HINIC_COMM_CMD_CMDQ_CTXT_GET = 0x11,
|
||||
@ -74,7 +76,13 @@ enum hinic_comm_cmd {
|
||||
|
||||
HINIC_COMM_CMD_IO_RES_CLEAR = 0x29,
|
||||
|
||||
HINIC_COMM_CMD_MAX = 0x32,
|
||||
HINIC_COMM_CMD_CEQ_CTRL_REG_WR_BY_UP = 0x33,
|
||||
|
||||
HINIC_COMM_CMD_L2NIC_RESET = 0x4b,
|
||||
|
||||
HINIC_COMM_CMD_PAGESIZE_SET = 0x50,
|
||||
|
||||
HINIC_COMM_CMD_MAX = 0x51,
|
||||
};
|
||||
|
||||
enum hinic_mgmt_cb_state {
|
||||
@ -107,7 +115,7 @@ struct hinic_mgmt_cb {
|
||||
|
||||
struct hinic_pf_to_mgmt {
|
||||
struct hinic_hwif *hwif;
|
||||
|
||||
struct hinic_hwdev *hwdev;
|
||||
struct semaphore sync_msg_lock;
|
||||
u16 sync_msg_id;
|
||||
u8 *sync_msg_buf;
|
||||
|
@ -108,7 +108,12 @@ void hinic_sq_prepare_ctxt(struct hinic_sq_ctxt *sq_ctxt,
|
||||
wq_page_pfn_hi = upper_32_bits(wq_page_pfn);
|
||||
wq_page_pfn_lo = lower_32_bits(wq_page_pfn);
|
||||
|
||||
wq_block_pfn = HINIC_WQ_BLOCK_PFN(wq->block_paddr);
|
||||
/* If only one page, use 0-level CLA */
|
||||
if (wq->num_q_pages == 1)
|
||||
wq_block_pfn = HINIC_WQ_BLOCK_PFN(wq_page_addr);
|
||||
else
|
||||
wq_block_pfn = HINIC_WQ_BLOCK_PFN(wq->block_paddr);
|
||||
|
||||
wq_block_pfn_hi = upper_32_bits(wq_block_pfn);
|
||||
wq_block_pfn_lo = lower_32_bits(wq_block_pfn);
|
||||
|
||||
|
@ -38,8 +38,8 @@
|
||||
#define HINIC_SQ_WQEBB_SIZE 64
|
||||
#define HINIC_RQ_WQEBB_SIZE 32
|
||||
|
||||
#define HINIC_SQ_PAGE_SIZE SZ_4K
|
||||
#define HINIC_RQ_PAGE_SIZE SZ_4K
|
||||
#define HINIC_SQ_PAGE_SIZE SZ_256K
|
||||
#define HINIC_RQ_PAGE_SIZE SZ_256K
|
||||
|
||||
#define HINIC_SQ_DEPTH SZ_4K
|
||||
#define HINIC_RQ_DEPTH SZ_4K
|
||||
|
@ -503,7 +503,7 @@ err_alloc_wq_pages:
|
||||
* Return 0 - Success, negative - Failure
|
||||
**/
|
||||
int hinic_wq_allocate(struct hinic_wqs *wqs, struct hinic_wq *wq,
|
||||
u16 wqebb_size, u16 wq_page_size, u16 q_depth,
|
||||
u16 wqebb_size, u32 wq_page_size, u16 q_depth,
|
||||
u16 max_wqe_size)
|
||||
{
|
||||
struct hinic_hwif *hwif = wqs->hwif;
|
||||
@ -600,7 +600,7 @@ void hinic_wq_free(struct hinic_wqs *wqs, struct hinic_wq *wq)
|
||||
**/
|
||||
int hinic_wqs_cmdq_alloc(struct hinic_cmdq_pages *cmdq_pages,
|
||||
struct hinic_wq *wq, struct hinic_hwif *hwif,
|
||||
int cmdq_blocks, u16 wqebb_size, u16 wq_page_size,
|
||||
int cmdq_blocks, u16 wqebb_size, u32 wq_page_size,
|
||||
u16 q_depth, u16 max_wqe_size)
|
||||
{
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
@ -768,7 +768,10 @@ struct hinic_hw_wqe *hinic_get_wqe(struct hinic_wq *wq, unsigned int wqe_size,
|
||||
|
||||
*prod_idx = curr_prod_idx;
|
||||
|
||||
if (curr_pg != end_pg) {
|
||||
/* If we only have one page, still need to get shadown wqe when
|
||||
* wqe rolling-over page
|
||||
*/
|
||||
if (curr_pg != end_pg || MASKED_WQE_IDX(wq, end_prod_idx) < *prod_idx) {
|
||||
void *shadow_addr = &wq->shadow_wqe[curr_pg * wq->max_wqe_size];
|
||||
|
||||
copy_wqe_to_shadow(wq, shadow_addr, num_wqebbs, *prod_idx);
|
||||
|
@ -26,7 +26,7 @@ struct hinic_wq {
|
||||
int block_idx;
|
||||
|
||||
u16 wqebb_size;
|
||||
u16 wq_page_size;
|
||||
u32 wq_page_size;
|
||||
u16 q_depth;
|
||||
u16 max_wqe_size;
|
||||
u16 num_wqebbs_per_page;
|
||||
@ -76,7 +76,7 @@ struct hinic_cmdq_pages {
|
||||
|
||||
int hinic_wqs_cmdq_alloc(struct hinic_cmdq_pages *cmdq_pages,
|
||||
struct hinic_wq *wq, struct hinic_hwif *hwif,
|
||||
int cmdq_blocks, u16 wqebb_size, u16 wq_page_size,
|
||||
int cmdq_blocks, u16 wqebb_size, u32 wq_page_size,
|
||||
u16 q_depth, u16 max_wqe_size);
|
||||
|
||||
void hinic_wqs_cmdq_free(struct hinic_cmdq_pages *cmdq_pages,
|
||||
@ -88,7 +88,7 @@ int hinic_wqs_alloc(struct hinic_wqs *wqs, int num_wqs,
|
||||
void hinic_wqs_free(struct hinic_wqs *wqs);
|
||||
|
||||
int hinic_wq_allocate(struct hinic_wqs *wqs, struct hinic_wq *wq,
|
||||
u16 wqebb_size, u16 wq_page_size, u16 q_depth,
|
||||
u16 wqebb_size, u32 wq_page_size, u16 q_depth,
|
||||
u16 max_wqe_size);
|
||||
|
||||
void hinic_wq_free(struct hinic_wqs *wqs, struct hinic_wq *wq);
|
||||
|
@ -29,6 +29,7 @@
|
||||
#include "hinic_tx.h"
|
||||
#include "hinic_rx.h"
|
||||
#include "hinic_dev.h"
|
||||
#include "hinic_sriov.h"
|
||||
|
||||
MODULE_AUTHOR("Huawei Technologies CO., Ltd");
|
||||
MODULE_DESCRIPTION("Huawei Intelligent NIC driver");
|
||||
@ -46,6 +47,7 @@ MODULE_PARM_DESC(rx_weight, "Number Rx packets for NAPI budget (default=64)");
|
||||
#define HINIC_DEV_ID_DUAL_PORT_100GE 0x0200
|
||||
#define HINIC_DEV_ID_DUAL_PORT_100GE_MEZZ 0x0205
|
||||
#define HINIC_DEV_ID_QUAD_PORT_25GE_MEZZ 0x0210
|
||||
#define HINIC_DEV_ID_VF 0x375e
|
||||
|
||||
#define HINIC_WQ_NAME "hinic_dev"
|
||||
|
||||
@ -65,6 +67,8 @@ MODULE_PARM_DESC(rx_weight, "Number Rx packets for NAPI budget (default=64)");
|
||||
#define rx_mode_work_to_nic_dev(rx_mode_work) \
|
||||
container_of(rx_mode_work, struct hinic_dev, rx_mode_work)
|
||||
|
||||
#define HINIC_WAIT_SRIOV_CFG_TIMEOUT 15000
|
||||
|
||||
static int change_mac_addr(struct net_device *netdev, const u8 *addr);
|
||||
|
||||
static int set_features(struct hinic_dev *nic_dev,
|
||||
@ -423,8 +427,9 @@ static int hinic_open(struct net_device *netdev)
|
||||
goto err_func_port_state;
|
||||
}
|
||||
|
||||
/* Wait up to 3 sec between port enable to link state */
|
||||
msleep(3000);
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
/* Wait up to 3 sec between port enable to link state */
|
||||
msleep(3000);
|
||||
|
||||
down(&nic_dev->mgmt_lock);
|
||||
|
||||
@ -434,6 +439,9 @@ static int hinic_open(struct net_device *netdev)
|
||||
goto err_port_link;
|
||||
}
|
||||
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_notify_all_vfs_link_changed(nic_dev->hwdev, link_state);
|
||||
|
||||
if (link_state == HINIC_LINK_STATE_UP)
|
||||
nic_dev->flags |= HINIC_LINK_UP;
|
||||
|
||||
@ -497,6 +505,9 @@ static int hinic_close(struct net_device *netdev)
|
||||
|
||||
up(&nic_dev->mgmt_lock);
|
||||
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_notify_all_vfs_link_changed(nic_dev->hwdev, 0);
|
||||
|
||||
err = hinic_port_set_func_state(nic_dev, HINIC_FUNC_PORT_DISABLE);
|
||||
if (err) {
|
||||
netif_err(nic_dev, drv, netdev,
|
||||
@ -685,7 +696,7 @@ static int hinic_vlan_rx_add_vid(struct net_device *netdev,
|
||||
}
|
||||
|
||||
err = hinic_port_add_mac(nic_dev, netdev->dev_addr, vid);
|
||||
if (err) {
|
||||
if (err && err != HINIC_PF_SET_VF_ALREADY) {
|
||||
netif_err(nic_dev, drv, netdev, "Failed to set mac\n");
|
||||
goto err_add_mac;
|
||||
}
|
||||
@ -737,8 +748,6 @@ static void set_rx_mode(struct work_struct *work)
|
||||
struct hinic_rx_mode_work *rx_mode_work = work_to_rx_mode_work(work);
|
||||
struct hinic_dev *nic_dev = rx_mode_work_to_nic_dev(rx_mode_work);
|
||||
|
||||
netif_info(nic_dev, drv, nic_dev->netdev, "set rx mode work\n");
|
||||
|
||||
hinic_port_set_rx_mode(nic_dev, rx_mode_work->rx_mode);
|
||||
|
||||
__dev_uc_sync(nic_dev->netdev, add_mac_addr, remove_mac_addr);
|
||||
@ -770,8 +779,26 @@ static void hinic_set_rx_mode(struct net_device *netdev)
|
||||
static void hinic_tx_timeout(struct net_device *netdev, unsigned int txqueue)
|
||||
{
|
||||
struct hinic_dev *nic_dev = netdev_priv(netdev);
|
||||
u16 sw_pi, hw_ci, sw_ci;
|
||||
struct hinic_sq *sq;
|
||||
u16 num_sqs, q_id;
|
||||
|
||||
num_sqs = hinic_hwdev_num_qps(nic_dev->hwdev);
|
||||
|
||||
netif_err(nic_dev, drv, netdev, "Tx timeout\n");
|
||||
|
||||
for (q_id = 0; q_id < num_sqs; q_id++) {
|
||||
if (!netif_xmit_stopped(netdev_get_tx_queue(netdev, q_id)))
|
||||
continue;
|
||||
|
||||
sq = hinic_hwdev_get_sq(nic_dev->hwdev, q_id);
|
||||
sw_pi = atomic_read(&sq->wq->prod_idx) & sq->wq->mask;
|
||||
hw_ci = be16_to_cpu(*(u16 *)(sq->hw_ci_addr)) & sq->wq->mask;
|
||||
sw_ci = atomic_read(&sq->wq->cons_idx) & sq->wq->mask;
|
||||
netif_err(nic_dev, drv, netdev, "Txq%d: sw_pi: %d, hw_ci: %d, sw_ci: %d, napi->state: 0x%lx\n",
|
||||
q_id, sw_pi, hw_ci, sw_ci,
|
||||
nic_dev->txqs[q_id].napi.state);
|
||||
}
|
||||
}
|
||||
|
||||
static void hinic_get_stats64(struct net_device *netdev,
|
||||
@ -837,6 +864,26 @@ static const struct net_device_ops hinic_netdev_ops = {
|
||||
.ndo_get_stats64 = hinic_get_stats64,
|
||||
.ndo_fix_features = hinic_fix_features,
|
||||
.ndo_set_features = hinic_set_features,
|
||||
.ndo_set_vf_mac = hinic_ndo_set_vf_mac,
|
||||
.ndo_set_vf_vlan = hinic_ndo_set_vf_vlan,
|
||||
.ndo_get_vf_config = hinic_ndo_get_vf_config,
|
||||
.ndo_set_vf_trust = hinic_ndo_set_vf_trust,
|
||||
};
|
||||
|
||||
static const struct net_device_ops hinicvf_netdev_ops = {
|
||||
.ndo_open = hinic_open,
|
||||
.ndo_stop = hinic_close,
|
||||
.ndo_change_mtu = hinic_change_mtu,
|
||||
.ndo_set_mac_address = hinic_set_mac_addr,
|
||||
.ndo_validate_addr = eth_validate_addr,
|
||||
.ndo_vlan_rx_add_vid = hinic_vlan_rx_add_vid,
|
||||
.ndo_vlan_rx_kill_vid = hinic_vlan_rx_kill_vid,
|
||||
.ndo_set_rx_mode = hinic_set_rx_mode,
|
||||
.ndo_start_xmit = hinic_xmit_frame,
|
||||
.ndo_tx_timeout = hinic_tx_timeout,
|
||||
.ndo_get_stats64 = hinic_get_stats64,
|
||||
.ndo_fix_features = hinic_fix_features,
|
||||
.ndo_set_features = hinic_set_features,
|
||||
};
|
||||
|
||||
static void netdev_features_init(struct net_device *netdev)
|
||||
@ -896,6 +943,10 @@ static void link_status_event_handler(void *handle, void *buf_in, u16 in_size,
|
||||
netif_info(nic_dev, drv, nic_dev->netdev, "HINIC_Link is DOWN\n");
|
||||
}
|
||||
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_notify_all_vfs_link_changed(nic_dev->hwdev,
|
||||
link_status->link);
|
||||
|
||||
ret_link_status = buf_out;
|
||||
ret_link_status->status = 0;
|
||||
|
||||
@ -969,7 +1020,12 @@ static int nic_dev_init(struct pci_dev *pdev)
|
||||
}
|
||||
|
||||
hinic_set_ethtool_ops(netdev);
|
||||
netdev->netdev_ops = &hinic_netdev_ops;
|
||||
|
||||
if (!HINIC_IS_VF(hwdev->hwif))
|
||||
netdev->netdev_ops = &hinic_netdev_ops;
|
||||
else
|
||||
netdev->netdev_ops = &hinicvf_netdev_ops;
|
||||
|
||||
netdev->max_mtu = ETH_MAX_MTU;
|
||||
|
||||
nic_dev = netdev_priv(netdev);
|
||||
@ -981,6 +1037,8 @@ static int nic_dev_init(struct pci_dev *pdev)
|
||||
nic_dev->rxqs = NULL;
|
||||
nic_dev->tx_weight = tx_weight;
|
||||
nic_dev->rx_weight = rx_weight;
|
||||
nic_dev->sriov_info.hwdev = hwdev;
|
||||
nic_dev->sriov_info.pdev = pdev;
|
||||
|
||||
sema_init(&nic_dev->mgmt_lock, 1);
|
||||
|
||||
@ -1007,11 +1065,25 @@ static int nic_dev_init(struct pci_dev *pdev)
|
||||
pci_set_drvdata(pdev, netdev);
|
||||
|
||||
err = hinic_port_get_mac(nic_dev, netdev->dev_addr);
|
||||
if (err)
|
||||
dev_warn(&pdev->dev, "Failed to get mac address\n");
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "Failed to get mac address\n");
|
||||
goto err_get_mac;
|
||||
}
|
||||
|
||||
if (!is_valid_ether_addr(netdev->dev_addr)) {
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif)) {
|
||||
dev_err(&pdev->dev, "Invalid MAC address\n");
|
||||
err = -EIO;
|
||||
goto err_add_mac;
|
||||
}
|
||||
|
||||
dev_info(&pdev->dev, "Invalid MAC address %pM, using random\n",
|
||||
netdev->dev_addr);
|
||||
eth_hw_addr_random(netdev);
|
||||
}
|
||||
|
||||
err = hinic_port_add_mac(nic_dev, netdev->dev_addr, 0);
|
||||
if (err) {
|
||||
if (err && err != HINIC_PF_SET_VF_ALREADY) {
|
||||
dev_err(&pdev->dev, "Failed to add mac\n");
|
||||
goto err_add_mac;
|
||||
}
|
||||
@ -1053,6 +1125,7 @@ err_set_features:
|
||||
cancel_work_sync(&rx_mode_work->work);
|
||||
|
||||
err_set_mtu:
|
||||
err_get_mac:
|
||||
err_add_mac:
|
||||
pci_set_drvdata(pdev, NULL);
|
||||
destroy_workqueue(nic_dev->workq);
|
||||
@ -1126,12 +1199,37 @@ err_pci_regions:
|
||||
return err;
|
||||
}
|
||||
|
||||
#define HINIC_WAIT_SRIOV_CFG_TIMEOUT 15000
|
||||
|
||||
static void wait_sriov_cfg_complete(struct hinic_dev *nic_dev)
|
||||
{
|
||||
struct hinic_sriov_info *sriov_info = &nic_dev->sriov_info;
|
||||
u32 loop_cnt = 0;
|
||||
|
||||
set_bit(HINIC_FUNC_REMOVE, &sriov_info->state);
|
||||
usleep_range(9900, 10000);
|
||||
|
||||
while (loop_cnt < HINIC_WAIT_SRIOV_CFG_TIMEOUT) {
|
||||
if (!test_bit(HINIC_SRIOV_ENABLE, &sriov_info->state) &&
|
||||
!test_bit(HINIC_SRIOV_DISABLE, &sriov_info->state))
|
||||
return;
|
||||
|
||||
usleep_range(9900, 10000);
|
||||
loop_cnt++;
|
||||
}
|
||||
}
|
||||
|
||||
static void hinic_remove(struct pci_dev *pdev)
|
||||
{
|
||||
struct net_device *netdev = pci_get_drvdata(pdev);
|
||||
struct hinic_dev *nic_dev = netdev_priv(netdev);
|
||||
struct hinic_rx_mode_work *rx_mode_work;
|
||||
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif)) {
|
||||
wait_sriov_cfg_complete(nic_dev);
|
||||
hinic_pci_sriov_disable(pdev);
|
||||
}
|
||||
|
||||
unregister_netdev(netdev);
|
||||
|
||||
hinic_hwdev_cb_unregister(nic_dev->hwdev,
|
||||
@ -1144,6 +1242,8 @@ static void hinic_remove(struct pci_dev *pdev)
|
||||
|
||||
destroy_workqueue(nic_dev->workq);
|
||||
|
||||
hinic_vf_func_free(nic_dev->hwdev);
|
||||
|
||||
hinic_free_hwdev(nic_dev->hwdev);
|
||||
|
||||
free_netdev(netdev);
|
||||
@ -1164,6 +1264,7 @@ static const struct pci_device_id hinic_pci_table[] = {
|
||||
{ PCI_VDEVICE(HUAWEI, HINIC_DEV_ID_DUAL_PORT_100GE), 0},
|
||||
{ PCI_VDEVICE(HUAWEI, HINIC_DEV_ID_DUAL_PORT_100GE_MEZZ), 0},
|
||||
{ PCI_VDEVICE(HUAWEI, HINIC_DEV_ID_QUAD_PORT_25GE_MEZZ), 0},
|
||||
{ PCI_VDEVICE(HUAWEI, HINIC_DEV_ID_VF), 0},
|
||||
{ 0, 0}
|
||||
};
|
||||
MODULE_DEVICE_TABLE(pci, hinic_pci_table);
|
||||
@ -1174,6 +1275,7 @@ static struct pci_driver hinic_driver = {
|
||||
.probe = hinic_probe,
|
||||
.remove = hinic_remove,
|
||||
.shutdown = hinic_shutdown,
|
||||
.sriov_configure = hinic_pci_sriov_configure,
|
||||
};
|
||||
|
||||
module_pci_driver(hinic_driver);
|
||||
|
@ -37,20 +37,14 @@ enum mac_op {
|
||||
static int change_mac(struct hinic_dev *nic_dev, const u8 *addr,
|
||||
u16 vlan_id, enum mac_op op)
|
||||
{
|
||||
struct net_device *netdev = nic_dev->netdev;
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_port_mac_cmd port_mac_cmd;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(port_mac_cmd);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
enum hinic_port_cmd cmd;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
if (vlan_id >= VLAN_N_VID) {
|
||||
netif_err(nic_dev, drv, netdev, "Invalid VLAN number\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (op == MAC_SET)
|
||||
cmd = HINIC_PORT_CMD_SET_MAC;
|
||||
else
|
||||
@ -63,12 +57,25 @@ static int change_mac(struct hinic_dev *nic_dev, const u8 *addr,
|
||||
err = hinic_port_msg_cmd(hwdev, cmd, &port_mac_cmd,
|
||||
sizeof(port_mac_cmd),
|
||||
&port_mac_cmd, &out_size);
|
||||
if (err || (out_size != sizeof(port_mac_cmd)) || port_mac_cmd.status) {
|
||||
if (err || out_size != sizeof(port_mac_cmd) ||
|
||||
(port_mac_cmd.status &&
|
||||
port_mac_cmd.status != HINIC_PF_SET_VF_ALREADY &&
|
||||
port_mac_cmd.status != HINIC_MGMT_STATUS_EXIST)) {
|
||||
dev_err(&pdev->dev, "Failed to change MAC, ret = %d\n",
|
||||
port_mac_cmd.status);
|
||||
return -EFAULT;
|
||||
}
|
||||
|
||||
if (cmd == HINIC_PORT_CMD_SET_MAC && port_mac_cmd.status ==
|
||||
HINIC_PF_SET_VF_ALREADY) {
|
||||
dev_warn(&pdev->dev, "PF has already set VF mac, Ignore set operation\n");
|
||||
return HINIC_PF_SET_VF_ALREADY;
|
||||
}
|
||||
|
||||
if (cmd == HINIC_PORT_CMD_SET_MAC && port_mac_cmd.status ==
|
||||
HINIC_MGMT_STATUS_EXIST)
|
||||
dev_warn(&pdev->dev, "MAC is repeated. Ignore set operation\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -112,8 +119,8 @@ int hinic_port_get_mac(struct hinic_dev *nic_dev, u8 *addr)
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_port_mac_cmd port_mac_cmd;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(port_mac_cmd);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
port_mac_cmd.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -144,9 +151,9 @@ int hinic_port_set_mtu(struct hinic_dev *nic_dev, int new_mtu)
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_port_mtu_cmd port_mtu_cmd;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(port_mtu_cmd);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
int err, max_frame;
|
||||
u16 out_size;
|
||||
|
||||
if (new_mtu < HINIC_MIN_MTU_SIZE) {
|
||||
netif_err(nic_dev, drv, netdev, "mtu < MIN MTU size");
|
||||
@ -248,14 +255,9 @@ int hinic_port_link_state(struct hinic_dev *nic_dev,
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct hinic_port_link_cmd link_cmd;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(link_cmd);
|
||||
int err;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
link_cmd.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
|
||||
err = hinic_port_msg_cmd(hwdev, HINIC_PORT_CMD_GET_LINK_STATE,
|
||||
@ -284,13 +286,11 @@ int hinic_port_set_state(struct hinic_dev *nic_dev, enum hinic_port_state state)
|
||||
struct hinic_port_state_cmd port_state;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(port_state);
|
||||
int err;
|
||||
|
||||
if (!HINIC_IS_PF(hwif) && !HINIC_IS_PPF(hwif)) {
|
||||
dev_err(&pdev->dev, "unsupported PCI Function type\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
if (HINIC_IS_VF(hwdev->hwif))
|
||||
return 0;
|
||||
|
||||
port_state.state = state;
|
||||
|
||||
@ -320,7 +320,7 @@ int hinic_port_set_func_state(struct hinic_dev *nic_dev,
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(func_state);
|
||||
int err;
|
||||
|
||||
func_state.func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -351,7 +351,7 @@ int hinic_port_get_cap(struct hinic_dev *nic_dev,
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(*port_cap);
|
||||
int err;
|
||||
|
||||
port_cap->func_idx = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -382,7 +382,7 @@ int hinic_port_set_tso(struct hinic_dev *nic_dev, enum hinic_tso_state state)
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct hinic_tso_config tso_cfg = {0};
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(tso_cfg);
|
||||
int err;
|
||||
|
||||
tso_cfg.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -405,9 +405,9 @@ int hinic_set_rx_csum_offload(struct hinic_dev *nic_dev, u32 en)
|
||||
{
|
||||
struct hinic_checksum_offload rx_csum_cfg = {0};
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
u16 out_size = sizeof(rx_csum_cfg);
|
||||
struct hinic_hwif *hwif;
|
||||
struct pci_dev *pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
if (!hwdev)
|
||||
@ -443,6 +443,7 @@ int hinic_set_rx_vlan_offload(struct hinic_dev *nic_dev, u8 en)
|
||||
if (!hwdev)
|
||||
return -EINVAL;
|
||||
|
||||
out_size = sizeof(vlan_cfg);
|
||||
hwif = hwdev->hwif;
|
||||
pdev = hwif->pdev;
|
||||
vlan_cfg.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -465,8 +466,8 @@ int hinic_set_max_qnum(struct hinic_dev *nic_dev, u8 num_rqs)
|
||||
{
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
struct hinic_rq_num rq_num = { 0 };
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size = sizeof(rq_num);
|
||||
int err;
|
||||
|
||||
@ -491,8 +492,8 @@ static int hinic_set_rx_lro(struct hinic_dev *nic_dev, u8 ipv4_en, u8 ipv6_en,
|
||||
u8 max_wqe_num)
|
||||
{
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct hinic_lro_config lro_cfg = { 0 };
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size = sizeof(lro_cfg);
|
||||
int err;
|
||||
@ -568,6 +569,9 @@ int hinic_set_rx_lro_state(struct hinic_dev *nic_dev, u8 lro_en,
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
if (HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
return 0;
|
||||
|
||||
err = hinic_set_rx_lro_timer(nic_dev, lro_timer);
|
||||
if (err)
|
||||
return err;
|
||||
@ -741,9 +745,9 @@ int hinic_get_rss_type(struct hinic_dev *nic_dev, u32 tmpl_idx,
|
||||
{
|
||||
struct hinic_rss_context_table ctx_tbl = { 0 };
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
u16 out_size = sizeof(ctx_tbl);
|
||||
struct hinic_hwif *hwif;
|
||||
struct pci_dev *pdev;
|
||||
u16 out_size = sizeof(ctx_tbl);
|
||||
int err;
|
||||
|
||||
if (!hwdev || !rss_type)
|
||||
@ -784,7 +788,7 @@ int hinic_rss_set_template_tbl(struct hinic_dev *nic_dev, u32 template_id,
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct hinic_rss_key rss_key = { 0 };
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(rss_key);
|
||||
int err;
|
||||
|
||||
rss_key.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -809,9 +813,9 @@ int hinic_rss_get_template_tbl(struct hinic_dev *nic_dev, u32 tmpl_idx,
|
||||
{
|
||||
struct hinic_rss_template_key temp_key = { 0 };
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
u16 out_size = sizeof(temp_key);
|
||||
struct hinic_hwif *hwif;
|
||||
struct pci_dev *pdev;
|
||||
u16 out_size = sizeof(temp_key);
|
||||
int err;
|
||||
|
||||
if (!hwdev || !temp)
|
||||
@ -844,7 +848,7 @@ int hinic_rss_set_hash_engine(struct hinic_dev *nic_dev, u8 template_id,
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(rss_engine);
|
||||
int err;
|
||||
|
||||
rss_engine.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -868,9 +872,9 @@ int hinic_rss_get_hash_engine(struct hinic_dev *nic_dev, u8 tmpl_idx, u8 *type)
|
||||
{
|
||||
struct hinic_rss_engine_type hash_type = { 0 };
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
u16 out_size = sizeof(hash_type);
|
||||
struct hinic_hwif *hwif;
|
||||
struct pci_dev *pdev;
|
||||
u16 out_size = sizeof(hash_type);
|
||||
int err;
|
||||
|
||||
if (!hwdev || !type)
|
||||
@ -901,7 +905,7 @@ int hinic_rss_cfg(struct hinic_dev *nic_dev, u8 rss_en, u8 template_id)
|
||||
struct hinic_rss_config rss_cfg = { 0 };
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
u16 out_size = sizeof(rss_cfg);
|
||||
int err;
|
||||
|
||||
rss_cfg.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -927,8 +931,8 @@ int hinic_rss_template_alloc(struct hinic_dev *nic_dev, u8 *tmpl_idx)
|
||||
struct hinic_rss_template_mgmt template_mgmt = { 0 };
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(template_mgmt);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
template_mgmt.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -953,8 +957,8 @@ int hinic_rss_template_free(struct hinic_dev *nic_dev, u8 tmpl_idx)
|
||||
struct hinic_rss_template_mgmt template_mgmt = { 0 };
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_hwif *hwif = hwdev->hwif;
|
||||
u16 out_size = sizeof(template_mgmt);
|
||||
struct pci_dev *pdev = hwif->pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
template_mgmt.func_id = HINIC_HWIF_FUNC_IDX(hwif);
|
||||
@ -1043,9 +1047,9 @@ int hinic_get_mgmt_version(struct hinic_dev *nic_dev, u8 *mgmt_ver)
|
||||
{
|
||||
struct hinic_hwdev *hwdev = nic_dev->hwdev;
|
||||
struct hinic_version_info up_ver = {0};
|
||||
u16 out_size = sizeof(up_ver);
|
||||
struct hinic_hwif *hwif;
|
||||
struct pci_dev *pdev;
|
||||
u16 out_size;
|
||||
int err;
|
||||
|
||||
if (!hwdev)
|
||||
|
@ -148,9 +148,9 @@ struct hinic_port_link_status {
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u16 rsvd1;
|
||||
u16 func_id;
|
||||
u8 link;
|
||||
u8 rsvd2;
|
||||
u8 port_id;
|
||||
};
|
||||
|
||||
struct hinic_port_func_state_cmd {
|
||||
|
@ -432,9 +432,11 @@ static int rx_poll(struct napi_struct *napi, int budget)
|
||||
return budget;
|
||||
|
||||
napi_complete(napi);
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
rq->msix_entry,
|
||||
HINIC_MSIX_ENABLE);
|
||||
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
rq->msix_entry,
|
||||
HINIC_MSIX_ENABLE);
|
||||
|
||||
return pkts;
|
||||
}
|
||||
@ -461,9 +463,10 @@ static irqreturn_t rx_irq(int irq, void *data)
|
||||
|
||||
/* Disable the interrupt until napi will be completed */
|
||||
nic_dev = netdev_priv(rxq->netdev);
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
rq->msix_entry,
|
||||
HINIC_MSIX_DISABLE);
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
rq->msix_entry,
|
||||
HINIC_MSIX_DISABLE);
|
||||
|
||||
nic_dev = netdev_priv(rxq->netdev);
|
||||
hinic_hwdev_msix_cnt_set(nic_dev->hwdev, rq->msix_entry);
|
||||
|
1016
drivers/net/ethernet/huawei/hinic/hinic_sriov.c
Normal file
1016
drivers/net/ethernet/huawei/hinic/hinic_sriov.c
Normal file
File diff suppressed because it is too large
Load Diff
102
drivers/net/ethernet/huawei/hinic/hinic_sriov.h
Normal file
102
drivers/net/ethernet/huawei/hinic/hinic_sriov.h
Normal file
@ -0,0 +1,102 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-only */
|
||||
/* Huawei HiNIC PCI Express Linux driver
|
||||
* Copyright(c) 2017 Huawei Technologies Co., Ltd
|
||||
*/
|
||||
|
||||
#ifndef HINIC_SRIOV_H
|
||||
#define HINIC_SRIOV_H
|
||||
|
||||
#include "hinic_hw_dev.h"
|
||||
|
||||
#define OS_VF_ID_TO_HW(os_vf_id) ((os_vf_id) + 1)
|
||||
#define HW_VF_ID_TO_OS(hw_vf_id) ((hw_vf_id) - 1)
|
||||
|
||||
enum hinic_sriov_state {
|
||||
HINIC_SRIOV_DISABLE,
|
||||
HINIC_SRIOV_ENABLE,
|
||||
HINIC_FUNC_REMOVE,
|
||||
};
|
||||
|
||||
enum {
|
||||
HINIC_IFLA_VF_LINK_STATE_AUTO, /* link state of the uplink */
|
||||
HINIC_IFLA_VF_LINK_STATE_ENABLE, /* link always up */
|
||||
HINIC_IFLA_VF_LINK_STATE_DISABLE, /* link always down */
|
||||
};
|
||||
|
||||
struct hinic_sriov_info {
|
||||
struct pci_dev *pdev;
|
||||
struct hinic_hwdev *hwdev;
|
||||
bool sriov_enabled;
|
||||
unsigned int num_vfs;
|
||||
unsigned long state;
|
||||
};
|
||||
|
||||
struct vf_data_storage {
|
||||
u8 vf_mac_addr[ETH_ALEN];
|
||||
bool registered;
|
||||
bool pf_set_mac;
|
||||
u16 pf_vlan;
|
||||
u8 pf_qos;
|
||||
u32 max_rate;
|
||||
u32 min_rate;
|
||||
|
||||
bool link_forced;
|
||||
bool link_up; /* only valid if VF link is forced */
|
||||
bool spoofchk;
|
||||
bool trust;
|
||||
};
|
||||
|
||||
struct hinic_register_vf {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
};
|
||||
|
||||
struct hinic_port_mac_update {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u16 func_id;
|
||||
u16 vlan_id;
|
||||
u16 rsvd1;
|
||||
u8 old_mac[ETH_ALEN];
|
||||
u16 rsvd2;
|
||||
u8 new_mac[ETH_ALEN];
|
||||
};
|
||||
|
||||
struct hinic_vf_vlan_config {
|
||||
u8 status;
|
||||
u8 version;
|
||||
u8 rsvd0[6];
|
||||
|
||||
u16 func_id;
|
||||
u16 vlan_id;
|
||||
u8 qos;
|
||||
u8 rsvd1[7];
|
||||
};
|
||||
|
||||
int hinic_ndo_set_vf_mac(struct net_device *netdev, int vf, u8 *mac);
|
||||
|
||||
int hinic_ndo_set_vf_vlan(struct net_device *netdev, int vf, u16 vlan, u8 qos,
|
||||
__be16 vlan_proto);
|
||||
|
||||
int hinic_ndo_get_vf_config(struct net_device *netdev,
|
||||
int vf, struct ifla_vf_info *ivi);
|
||||
|
||||
int hinic_ndo_set_vf_trust(struct net_device *netdev, int vf, bool setting);
|
||||
|
||||
void hinic_notify_all_vfs_link_changed(struct hinic_hwdev *hwdev,
|
||||
u8 link_status);
|
||||
|
||||
int hinic_pci_sriov_disable(struct pci_dev *dev);
|
||||
|
||||
int hinic_pci_sriov_enable(struct pci_dev *dev, int num_vfs);
|
||||
|
||||
int hinic_vf_func_init(struct hinic_hwdev *hwdev);
|
||||
|
||||
void hinic_vf_func_free(struct hinic_hwdev *hwdev);
|
||||
|
||||
int hinic_pci_sriov_configure(struct pci_dev *dev, int num_vfs);
|
||||
|
||||
#endif
|
@ -673,9 +673,11 @@ static int free_tx_poll(struct napi_struct *napi, int budget)
|
||||
|
||||
if (pkts < budget) {
|
||||
napi_complete(napi);
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
sq->msix_entry,
|
||||
HINIC_MSIX_ENABLE);
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
sq->msix_entry,
|
||||
HINIC_MSIX_ENABLE);
|
||||
|
||||
return pkts;
|
||||
}
|
||||
|
||||
@ -701,10 +703,11 @@ static irqreturn_t tx_irq(int irq, void *data)
|
||||
|
||||
nic_dev = netdev_priv(txq->netdev);
|
||||
|
||||
/* Disable the interrupt until napi will be completed */
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
txq->sq->msix_entry,
|
||||
HINIC_MSIX_DISABLE);
|
||||
if (!HINIC_IS_VF(nic_dev->hwdev->hwif))
|
||||
/* Disable the interrupt until napi will be completed */
|
||||
hinic_hwdev_set_msix_state(nic_dev->hwdev,
|
||||
txq->sq->msix_entry,
|
||||
HINIC_MSIX_DISABLE);
|
||||
|
||||
hinic_hwdev_msix_cnt_set(nic_dev->hwdev, txq->sq->msix_entry);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user