Commit fc48b7a6 authored by Yuval Mintz's avatar Yuval Mintz Committed by David S. Miller

qed/qede: use 8.7.3.0 FW.

This patch moves the qed* driver into utilizing the 8.7.3.0 FW.
This new FW is required for a lot of new SW features, including:
  - Vlan filtering offload
  - Encapsulation offload support
  - HW ingress aggregations
As well as paving the way for the possibility of adding storage protocols
in the future.

V2:
 - Fix kbuild test robot error/warnings.
Signed-off-by: default avatarYuval Mintz <Yuval.Mintz@qlogic.com>
Signed-off-by: default avatarSudarsana Reddy Kalluru <Sudarsana.Kalluru@qlogic.com>
Signed-off-by: default avatarManish Chopra <manish.chopra@qlogic.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 7530e44c
...@@ -70,8 +70,8 @@ struct qed_sb_sp_info; ...@@ -70,8 +70,8 @@ struct qed_sb_sp_info;
struct qed_mcp_info; struct qed_mcp_info;
struct qed_rt_data { struct qed_rt_data {
u32 init_val; u32 *init_val;
bool b_valid; bool *b_valid;
}; };
/* The PCI personality is not quite synonymous to protocol ID: /* The PCI personality is not quite synonymous to protocol ID:
...@@ -120,6 +120,10 @@ enum QED_PORT_MODE { ...@@ -120,6 +120,10 @@ enum QED_PORT_MODE {
QED_PORT_MODE_DE_1X25G QED_PORT_MODE_DE_1X25G
}; };
enum qed_dev_cap {
QED_DEV_CAP_ETH,
};
struct qed_hw_info { struct qed_hw_info {
/* PCI personality */ /* PCI personality */
enum qed_pci_personality personality; enum qed_pci_personality personality;
...@@ -151,6 +155,7 @@ struct qed_hw_info { ...@@ -151,6 +155,7 @@ struct qed_hw_info {
u32 port_mode; u32 port_mode;
u32 hw_mode; u32 hw_mode;
unsigned long device_capabilities;
}; };
struct qed_hw_cid_data { struct qed_hw_cid_data {
...@@ -267,7 +272,7 @@ struct qed_hwfn { ...@@ -267,7 +272,7 @@ struct qed_hwfn {
struct qed_hw_info hw_info; struct qed_hw_info hw_info;
/* rt_array (for init-tool) */ /* rt_array (for init-tool) */
struct qed_rt_data *rt_data; struct qed_rt_data rt_data;
/* SPQ */ /* SPQ */
struct qed_spq *p_spq; struct qed_spq *p_spq;
...@@ -350,9 +355,20 @@ struct qed_dev { ...@@ -350,9 +355,20 @@ struct qed_dev {
char name[NAME_SIZE]; char name[NAME_SIZE];
u8 type; u8 type;
#define QED_DEV_TYPE_BB_A0 (0 << 0) #define QED_DEV_TYPE_BB (0 << 0)
#define QED_DEV_TYPE_MASK (0x3) #define QED_DEV_TYPE_AH BIT(0)
#define QED_DEV_TYPE_SHIFT (0) /* Translate type/revision combo into the proper conditions */
#define QED_IS_BB(dev) ((dev)->type == QED_DEV_TYPE_BB)
#define QED_IS_BB_A0(dev) (QED_IS_BB(dev) && \
CHIP_REV_IS_A0(dev))
#define QED_IS_BB_B0(dev) (QED_IS_BB(dev) && \
CHIP_REV_IS_B0(dev))
#define QED_GET_TYPE(dev) (QED_IS_BB_A0(dev) ? CHIP_BB_A0 : \
QED_IS_BB_B0(dev) ? CHIP_BB_B0 : CHIP_K2)
u16 vendor_id;
u16 device_id;
u16 chip_num; u16 chip_num;
#define CHIP_NUM_MASK 0xffff #define CHIP_NUM_MASK 0xffff
...@@ -361,6 +377,8 @@ struct qed_dev { ...@@ -361,6 +377,8 @@ struct qed_dev {
u16 chip_rev; u16 chip_rev;
#define CHIP_REV_MASK 0xf #define CHIP_REV_MASK 0xf
#define CHIP_REV_SHIFT 12 #define CHIP_REV_SHIFT 12
#define CHIP_REV_IS_A0(_cdev) (!(_cdev)->chip_rev)
#define CHIP_REV_IS_B0(_cdev) ((_cdev)->chip_rev == 1)
u16 chip_metal; u16 chip_metal;
#define CHIP_METAL_MASK 0xff #define CHIP_METAL_MASK 0xff
...@@ -375,10 +393,10 @@ struct qed_dev { ...@@ -375,10 +393,10 @@ struct qed_dev {
u8 num_funcs_in_port; u8 num_funcs_in_port;
u8 path_id; u8 path_id;
enum mf_mode mf_mode; enum qed_mf_mode mf_mode;
#define IS_MF(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode != SF) #define IS_MF_DEFAULT(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode == QED_MF_DEFAULT)
#define IS_MF_SI(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode == MF_NPAR) #define IS_MF_SI(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode == QED_MF_NPAR)
#define IS_MF_SD(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode == MF_OVLAN) #define IS_MF_SD(_p_hwfn) (((_p_hwfn)->cdev)->mf_mode == QED_MF_OVLAN)
int pcie_width; int pcie_width;
int pcie_speed; int pcie_speed;
...@@ -441,11 +459,6 @@ struct qed_dev { ...@@ -441,11 +459,6 @@ struct qed_dev {
const struct firmware *firmware; const struct firmware *firmware;
}; };
#define QED_GET_TYPE(dev) (((dev)->type & QED_DEV_TYPE_MASK) >> \
QED_DEV_TYPE_SHIFT)
#define QED_IS_BB_A0(dev) (QED_GET_TYPE(dev) == QED_DEV_TYPE_BB_A0)
#define QED_IS_BB(dev) (QED_IS_BB_A0(dev))
#define NUM_OF_SBS(dev) MAX_SB_PER_PATH_BB #define NUM_OF_SBS(dev) MAX_SB_PER_PATH_BB
#define NUM_OF_ENG_PFS(dev) MAX_NUM_PFS_BB #define NUM_OF_ENG_PFS(dev) MAX_NUM_PFS_BB
......
...@@ -581,7 +581,8 @@ void qed_qm_init_pf(struct qed_hwfn *p_hwfn) ...@@ -581,7 +581,8 @@ void qed_qm_init_pf(struct qed_hwfn *p_hwfn)
params.num_pf_cids = iids.cids; params.num_pf_cids = iids.cids;
params.start_pq = qm_info->start_pq; params.start_pq = qm_info->start_pq;
params.num_pf_pqs = qm_info->num_pqs; params.num_pf_pqs = qm_info->num_pqs;
params.start_vport = qm_info->num_vports; params.start_vport = qm_info->start_vport;
params.num_vports = qm_info->num_vports;
params.pf_wfq = qm_info->pf_wfq; params.pf_wfq = qm_info->pf_wfq;
params.pf_rl = qm_info->pf_rl; params.pf_rl = qm_info->pf_rl;
params.pq_params = qm_info->qm_pq_params; params.pq_params = qm_info->qm_pq_params;
......
...@@ -341,11 +341,6 @@ void qed_resc_setup(struct qed_dev *cdev) ...@@ -341,11 +341,6 @@ void qed_resc_setup(struct qed_dev *cdev)
} }
} }
#define FINAL_CLEANUP_CMD_OFFSET (0)
#define FINAL_CLEANUP_CMD (0x1)
#define FINAL_CLEANUP_VALID_OFFSET (6)
#define FINAL_CLEANUP_VFPF_ID_SHIFT (7)
#define FINAL_CLEANUP_COMP (0x2)
#define FINAL_CLEANUP_POLL_CNT (100) #define FINAL_CLEANUP_POLL_CNT (100)
#define FINAL_CLEANUP_POLL_TIME (10) #define FINAL_CLEANUP_POLL_TIME (10)
int qed_final_cleanup(struct qed_hwfn *p_hwfn, int qed_final_cleanup(struct qed_hwfn *p_hwfn,
...@@ -355,12 +350,14 @@ int qed_final_cleanup(struct qed_hwfn *p_hwfn, ...@@ -355,12 +350,14 @@ int qed_final_cleanup(struct qed_hwfn *p_hwfn,
u32 command = 0, addr, count = FINAL_CLEANUP_POLL_CNT; u32 command = 0, addr, count = FINAL_CLEANUP_POLL_CNT;
int rc = -EBUSY; int rc = -EBUSY;
addr = GTT_BAR0_MAP_REG_USDM_RAM + USTORM_FLR_FINAL_ACK_OFFSET; addr = GTT_BAR0_MAP_REG_USDM_RAM +
USTORM_FLR_FINAL_ACK_OFFSET(p_hwfn->rel_pf_id);
command |= FINAL_CLEANUP_CMD << FINAL_CLEANUP_CMD_OFFSET; command |= X_FINAL_CLEANUP_AGG_INT <<
command |= 1 << FINAL_CLEANUP_VALID_OFFSET; SDM_AGG_INT_COMP_PARAMS_AGG_INT_INDEX_SHIFT;
command |= id << FINAL_CLEANUP_VFPF_ID_SHIFT; command |= 1 << SDM_AGG_INT_COMP_PARAMS_AGG_VECTOR_ENABLE_SHIFT;
command |= FINAL_CLEANUP_COMP << SDM_OP_GEN_COMP_TYPE_SHIFT; command |= id << SDM_AGG_INT_COMP_PARAMS_AGG_VECTOR_BIT_SHIFT;
command |= SDM_COMP_TYPE_AGG_INT << SDM_OP_GEN_COMP_TYPE_SHIFT;
/* Make sure notification is not set before initiating final cleanup */ /* Make sure notification is not set before initiating final cleanup */
if (REG_RD(p_hwfn, addr)) { if (REG_RD(p_hwfn, addr)) {
...@@ -415,18 +412,16 @@ static void qed_calc_hw_mode(struct qed_hwfn *p_hwfn) ...@@ -415,18 +412,16 @@ static void qed_calc_hw_mode(struct qed_hwfn *p_hwfn)
} }
switch (p_hwfn->cdev->mf_mode) { switch (p_hwfn->cdev->mf_mode) {
case SF: case QED_MF_DEFAULT:
hw_mode |= 1 << MODE_SF; case QED_MF_NPAR:
hw_mode |= 1 << MODE_MF_SI;
break; break;
case MF_OVLAN: case QED_MF_OVLAN:
hw_mode |= 1 << MODE_MF_SD; hw_mode |= 1 << MODE_MF_SD;
break; break;
case MF_NPAR:
hw_mode |= 1 << MODE_MF_SI;
break;
default: default:
DP_NOTICE(p_hwfn, "Unsupported MF mode, init as SF\n"); DP_NOTICE(p_hwfn, "Unsupported MF mode, init as DEFAULT\n");
hw_mode |= 1 << MODE_SF; hw_mode |= 1 << MODE_MF_SI;
} }
hw_mode |= 1 << MODE_ASIC; hw_mode |= 1 << MODE_ASIC;
...@@ -1018,8 +1013,7 @@ static void qed_hw_get_resc(struct qed_hwfn *p_hwfn) ...@@ -1018,8 +1013,7 @@ static void qed_hw_get_resc(struct qed_hwfn *p_hwfn)
u32 *resc_num = p_hwfn->hw_info.resc_num; u32 *resc_num = p_hwfn->hw_info.resc_num;
int num_funcs, i; int num_funcs, i;
num_funcs = IS_MF(p_hwfn) ? MAX_NUM_PFS_BB num_funcs = MAX_NUM_PFS_BB;
: p_hwfn->cdev->num_ports_in_engines;
resc_num[QED_SB] = min_t(u32, resc_num[QED_SB] = min_t(u32,
(MAX_SB_PER_PATH_BB / num_funcs), (MAX_SB_PER_PATH_BB / num_funcs),
...@@ -1071,7 +1065,7 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn, ...@@ -1071,7 +1065,7 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn,
struct qed_ptt *p_ptt) struct qed_ptt *p_ptt)
{ {
u32 nvm_cfg1_offset, mf_mode, addr, generic_cont0, core_cfg; u32 nvm_cfg1_offset, mf_mode, addr, generic_cont0, core_cfg;
u32 port_cfg_addr, link_temp, val, nvm_cfg_addr; u32 port_cfg_addr, link_temp, nvm_cfg_addr, device_capabilities;
struct qed_mcp_link_params *link; struct qed_mcp_link_params *link;
/* Read global nvm_cfg address */ /* Read global nvm_cfg address */
...@@ -1134,21 +1128,6 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn, ...@@ -1134,21 +1128,6 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn,
break; break;
} }
addr = MCP_REG_SCRATCH + nvm_cfg1_offset +
offsetof(struct nvm_cfg1, func[MCP_PF_ID(p_hwfn)]) +
offsetof(struct nvm_cfg1_func, device_id);
val = qed_rd(p_hwfn, p_ptt, addr);
if (IS_MF(p_hwfn)) {
p_hwfn->hw_info.device_id =
(val & NVM_CFG1_FUNC_MF_VENDOR_DEVICE_ID_MASK) >>
NVM_CFG1_FUNC_MF_VENDOR_DEVICE_ID_OFFSET;
} else {
p_hwfn->hw_info.device_id =
(val & NVM_CFG1_FUNC_VENDOR_DEVICE_ID_MASK) >>
NVM_CFG1_FUNC_VENDOR_DEVICE_ID_OFFSET;
}
/* Read default link configuration */ /* Read default link configuration */
link = &p_hwfn->mcp_info->link_input; link = &p_hwfn->mcp_info->link_input;
port_cfg_addr = MCP_REG_SCRATCH + nvm_cfg1_offset + port_cfg_addr = MCP_REG_SCRATCH + nvm_cfg1_offset +
...@@ -1220,18 +1199,28 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn, ...@@ -1220,18 +1199,28 @@ static int qed_hw_get_nvm_info(struct qed_hwfn *p_hwfn,
switch (mf_mode) { switch (mf_mode) {
case NVM_CFG1_GLOB_MF_MODE_MF_ALLOWED: case NVM_CFG1_GLOB_MF_MODE_MF_ALLOWED:
p_hwfn->cdev->mf_mode = MF_OVLAN; p_hwfn->cdev->mf_mode = QED_MF_OVLAN;
break; break;
case NVM_CFG1_GLOB_MF_MODE_NPAR1_0: case NVM_CFG1_GLOB_MF_MODE_NPAR1_0:
p_hwfn->cdev->mf_mode = MF_NPAR; p_hwfn->cdev->mf_mode = QED_MF_NPAR;
break; break;
case NVM_CFG1_GLOB_MF_MODE_FORCED_SF: case NVM_CFG1_GLOB_MF_MODE_DEFAULT:
p_hwfn->cdev->mf_mode = SF; p_hwfn->cdev->mf_mode = QED_MF_DEFAULT;
break; break;
} }
DP_INFO(p_hwfn, "Multi function mode is %08x\n", DP_INFO(p_hwfn, "Multi function mode is %08x\n",
p_hwfn->cdev->mf_mode); p_hwfn->cdev->mf_mode);
/* Read Multi-function information from shmem */
addr = MCP_REG_SCRATCH + nvm_cfg1_offset +
offsetof(struct nvm_cfg1, glob) +
offsetof(struct nvm_cfg1_glob, device_capabilities);
device_capabilities = qed_rd(p_hwfn, p_ptt, addr);
if (device_capabilities & NVM_CFG1_GLOB_DEVICE_CAPABILITIES_ETHERNET)
__set_bit(QED_DEV_CAP_ETH,
&p_hwfn->hw_info.device_capabilities);
return qed_mcp_fill_shmem_func_info(p_hwfn, p_ptt); return qed_mcp_fill_shmem_func_info(p_hwfn, p_ptt);
} }
...@@ -1293,29 +1282,36 @@ qed_get_hw_info(struct qed_hwfn *p_hwfn, ...@@ -1293,29 +1282,36 @@ qed_get_hw_info(struct qed_hwfn *p_hwfn,
static void qed_get_dev_info(struct qed_dev *cdev) static void qed_get_dev_info(struct qed_dev *cdev)
{ {
struct qed_hwfn *p_hwfn = QED_LEADING_HWFN(cdev);
u32 tmp; u32 tmp;
cdev->chip_num = (u16)qed_rd(cdev->hwfns, cdev->hwfns[0].p_main_ptt, /* Read Vendor Id / Device Id */
pci_read_config_word(cdev->pdev, PCI_VENDOR_ID,
&cdev->vendor_id);
pci_read_config_word(cdev->pdev, PCI_DEVICE_ID,
&cdev->device_id);
cdev->chip_num = (u16)qed_rd(p_hwfn, p_hwfn->p_main_ptt,
MISCS_REG_CHIP_NUM); MISCS_REG_CHIP_NUM);
cdev->chip_rev = (u16)qed_rd(cdev->hwfns, cdev->hwfns[0].p_main_ptt, cdev->chip_rev = (u16)qed_rd(p_hwfn, p_hwfn->p_main_ptt,
MISCS_REG_CHIP_REV); MISCS_REG_CHIP_REV);
MASK_FIELD(CHIP_REV, cdev->chip_rev); MASK_FIELD(CHIP_REV, cdev->chip_rev);
cdev->type = QED_DEV_TYPE_BB;
/* Learn number of HW-functions */ /* Learn number of HW-functions */
tmp = qed_rd(cdev->hwfns, cdev->hwfns[0].p_main_ptt, tmp = qed_rd(p_hwfn, p_hwfn->p_main_ptt,
MISCS_REG_CMT_ENABLED_FOR_PAIR); MISCS_REG_CMT_ENABLED_FOR_PAIR);
if (tmp & (1 << cdev->hwfns[0].rel_pf_id)) { if (tmp & (1 << p_hwfn->rel_pf_id)) {
DP_NOTICE(cdev->hwfns, "device in CMT mode\n"); DP_NOTICE(cdev->hwfns, "device in CMT mode\n");
cdev->num_hwfns = 2; cdev->num_hwfns = 2;
} else { } else {
cdev->num_hwfns = 1; cdev->num_hwfns = 1;
} }
cdev->chip_bond_id = qed_rd(cdev->hwfns, cdev->hwfns[0].p_main_ptt, cdev->chip_bond_id = qed_rd(p_hwfn, p_hwfn->p_main_ptt,
MISCS_REG_CHIP_TEST_REG) >> 4; MISCS_REG_CHIP_TEST_REG) >> 4;
MASK_FIELD(CHIP_BOND_ID, cdev->chip_bond_id); MASK_FIELD(CHIP_BOND_ID, cdev->chip_bond_id);
cdev->chip_metal = (u16)qed_rd(cdev->hwfns, cdev->hwfns[0].p_main_ptt, cdev->chip_metal = (u16)qed_rd(p_hwfn, p_hwfn->p_main_ptt,
MISCS_REG_CHIP_METAL); MISCS_REG_CHIP_METAL);
MASK_FIELD(CHIP_METAL, cdev->chip_metal); MASK_FIELD(CHIP_METAL, cdev->chip_metal);
......
This source diff could not be displayed because it is too large. You can view the blob instead.
...@@ -513,17 +513,14 @@ static int qed_pf_rl_rt_init(struct qed_hwfn *p_hwfn, ...@@ -513,17 +513,14 @@ static int qed_pf_rl_rt_init(struct qed_hwfn *p_hwfn,
* Return -1 on error. * Return -1 on error.
*/ */
static int qed_vp_wfq_rt_init(struct qed_hwfn *p_hwfn, static int qed_vp_wfq_rt_init(struct qed_hwfn *p_hwfn,
u8 start_vport,
u8 num_vports, u8 num_vports,
struct init_qm_vport_params *vport_params) struct init_qm_vport_params *vport_params)
{ {
u8 tc, i, vport_id;
u32 inc_val; u32 inc_val;
u8 tc, i;
/* go over all PF VPORTs */ /* go over all PF VPORTs */
for (i = 0, vport_id = start_vport; i < num_vports; i++, vport_id++) { for (i = 0; i < num_vports; i++) {
u32 temp = QM_REG_WFQVPUPPERBOUND_RT_OFFSET;
u16 *pq_ids = &vport_params[i].first_tx_pq_id[0];
if (!vport_params[i].vport_wfq) if (!vport_params[i].vport_wfq)
continue; continue;
...@@ -539,20 +536,16 @@ static int qed_vp_wfq_rt_init(struct qed_hwfn *p_hwfn, ...@@ -539,20 +536,16 @@ static int qed_vp_wfq_rt_init(struct qed_hwfn *p_hwfn,
* different TCs * different TCs
*/ */
for (tc = 0; tc < NUM_OF_TCS; tc++) { for (tc = 0; tc < NUM_OF_TCS; tc++) {
u16 vport_pq_id = pq_ids[tc]; u16 vport_pq_id = vport_params[i].first_tx_pq_id[tc];
if (vport_pq_id != QM_INVALID_PQ_ID) { if (vport_pq_id != QM_INVALID_PQ_ID) {
STORE_RT_REG(p_hwfn,
QM_REG_WFQVPWEIGHT_RT_OFFSET +
vport_pq_id, inc_val);
STORE_RT_REG(p_hwfn, temp + vport_pq_id,
QM_WFQ_UPPER_BOUND |
QM_WFQ_CRD_REG_SIGN_BIT);
STORE_RT_REG(p_hwfn, STORE_RT_REG(p_hwfn,
QM_REG_WFQVPCRD_RT_OFFSET + QM_REG_WFQVPCRD_RT_OFFSET +
vport_pq_id, vport_pq_id,
QM_WFQ_INIT_CRD(inc_val) |
QM_WFQ_CRD_REG_SIGN_BIT); QM_WFQ_CRD_REG_SIGN_BIT);
STORE_RT_REG(p_hwfn,
QM_REG_WFQVPWEIGHT_RT_OFFSET +
vport_pq_id, inc_val);
} }
} }
} }
...@@ -709,8 +702,7 @@ int qed_qm_pf_rt_init(struct qed_hwfn *p_hwfn, ...@@ -709,8 +702,7 @@ int qed_qm_pf_rt_init(struct qed_hwfn *p_hwfn,
if (qed_pf_rl_rt_init(p_hwfn, p_params->pf_id, p_params->pf_rl)) if (qed_pf_rl_rt_init(p_hwfn, p_params->pf_id, p_params->pf_rl))
return -1; return -1;
if (qed_vp_wfq_rt_init(p_hwfn, p_params->start_vport, if (qed_vp_wfq_rt_init(p_hwfn, p_params->num_vports, vport_params))
p_params->num_vports, vport_params))
return -1; return -1;
if (qed_vport_rl_rt_init(p_hwfn, p_params->start_vport, if (qed_vport_rl_rt_init(p_hwfn, p_params->start_vport,
......
...@@ -55,63 +55,98 @@ void qed_init_clear_rt_data(struct qed_hwfn *p_hwfn) ...@@ -55,63 +55,98 @@ void qed_init_clear_rt_data(struct qed_hwfn *p_hwfn)
int i; int i;
for (i = 0; i < RUNTIME_ARRAY_SIZE; i++) for (i = 0; i < RUNTIME_ARRAY_SIZE; i++)
p_hwfn->rt_data[i].b_valid = false; p_hwfn->rt_data.b_valid[i] = false;
} }
void qed_init_store_rt_reg(struct qed_hwfn *p_hwfn, void qed_init_store_rt_reg(struct qed_hwfn *p_hwfn,
u32 rt_offset, u32 rt_offset,
u32 val) u32 val)
{ {
p_hwfn->rt_data[rt_offset].init_val = val; p_hwfn->rt_data.init_val[rt_offset] = val;
p_hwfn->rt_data[rt_offset].b_valid = true; p_hwfn->rt_data.b_valid[rt_offset] = true;
} }
void qed_init_store_rt_agg(struct qed_hwfn *p_hwfn, void qed_init_store_rt_agg(struct qed_hwfn *p_hwfn,
u32 rt_offset, u32 rt_offset, u32 *p_val,
u32 *val,
size_t size) size_t size)
{ {
size_t i; size_t i;
for (i = 0; i < size / sizeof(u32); i++) { for (i = 0; i < size / sizeof(u32); i++) {
p_hwfn->rt_data[rt_offset + i].init_val = val[i]; p_hwfn->rt_data.init_val[rt_offset + i] = p_val[i];
p_hwfn->rt_data[rt_offset + i].b_valid = true; p_hwfn->rt_data.b_valid[rt_offset + i] = true;
} }
} }
static void qed_init_rt(struct qed_hwfn *p_hwfn, static int qed_init_rt(struct qed_hwfn *p_hwfn,
struct qed_ptt *p_ptt, struct qed_ptt *p_ptt,
u32 addr, u32 addr,
u32 rt_offset, u16 rt_offset,
u32 size) u16 size,
bool b_must_dmae)
{ {
struct qed_rt_data *rt_data = p_hwfn->rt_data + rt_offset; u32 *p_init_val = &p_hwfn->rt_data.init_val[rt_offset];
u32 i; bool *p_valid = &p_hwfn->rt_data.b_valid[rt_offset];
u16 i, segment;
int rc = 0;
/* Since not all RT entries are initialized, go over the RT and
* for each segment of initialized values use DMA.
*/
for (i = 0; i < size; i++) { for (i = 0; i < size; i++) {
if (!rt_data[i].b_valid) if (!p_valid[i])
continue;
/* In case there isn't any wide-bus configuration here,
* simply write the data instead of using dmae.
*/
if (!b_must_dmae) {
qed_wr(p_hwfn, p_ptt, addr + (i << 2),
p_init_val[i]);
continue; continue;
qed_wr(p_hwfn, p_ptt, addr + (i << 2), rt_data[i].init_val);
} }
/* Start of a new segment */
for (segment = 1; i + segment < size; segment++)
if (!p_valid[i + segment])
break;
rc = qed_dmae_host2grc(p_hwfn, p_ptt,
(uintptr_t)(p_init_val + i),
addr + (i << 2), segment, 0);
if (rc != 0)
return rc;
/* Jump over the entire segment, including invalid entry */
i += segment;
}
return rc;
} }
int qed_init_alloc(struct qed_hwfn *p_hwfn) int qed_init_alloc(struct qed_hwfn *p_hwfn)
{ {
struct qed_rt_data *rt_data; struct qed_rt_data *rt_data = &p_hwfn->rt_data;
rt_data = kzalloc(sizeof(*rt_data) * RUNTIME_ARRAY_SIZE, GFP_ATOMIC); rt_data->b_valid = kzalloc(sizeof(bool) * RUNTIME_ARRAY_SIZE,
if (!rt_data) GFP_KERNEL);
if (!rt_data->b_valid)
return -ENOMEM; return -ENOMEM;
p_hwfn->rt_data = rt_data; rt_data->init_val = kzalloc(sizeof(u32) * RUNTIME_ARRAY_SIZE,
GFP_KERNEL);
if (!rt_data->init_val) {
kfree(rt_data->b_valid);
return -ENOMEM;
}
return 0; return 0;
} }
void qed_init_free(struct qed_hwfn *p_hwfn) void qed_init_free(struct qed_hwfn *p_hwfn)
{ {
kfree(p_hwfn->rt_data); kfree(p_hwfn->rt_data.init_val);
p_hwfn->rt_data = NULL; kfree(p_hwfn->rt_data.b_valid);
} }
static int qed_init_array_dmae(struct qed_hwfn *p_hwfn, static int qed_init_array_dmae(struct qed_hwfn *p_hwfn,
...@@ -289,7 +324,8 @@ static int qed_init_cmd_wr(struct qed_hwfn *p_hwfn, ...@@ -289,7 +324,8 @@ static int qed_init_cmd_wr(struct qed_hwfn *p_hwfn,
case INIT_SRC_RUNTIME: case INIT_SRC_RUNTIME:
qed_init_rt(p_hwfn, p_ptt, addr, qed_init_rt(p_hwfn, p_ptt, addr,
le16_to_cpu(arg->runtime.offset), le16_to_cpu(arg->runtime.offset),
le16_to_cpu(arg->runtime.size)); le16_to_cpu(arg->runtime.size),
b_must_dmae);
break; break;
} }
...@@ -316,49 +352,50 @@ static void qed_init_cmd_rd(struct qed_hwfn *p_hwfn, ...@@ -316,49 +352,50 @@ static void qed_init_cmd_rd(struct qed_hwfn *p_hwfn,
struct qed_ptt *p_ptt, struct qed_ptt *p_ptt,
struct init_read_op *cmd) struct init_read_op *cmd)
{ {
u32 data = le32_to_cpu(cmd->op_data); bool (*comp_check)(u32 val, u32 expected_val);
u32 addr = GET_FIELD(data, INIT_READ_OP_ADDRESS) << 2;
bool (*comp_check)(u32 val,
u32 expected_val);
u32 delay = QED_INIT_POLL_PERIOD_US, val; u32 delay = QED_INIT_POLL_PERIOD_US, val;
u32 data, addr, poll;
int i;
data = le32_to_cpu(cmd->op_data);
addr = GET_FIELD(data, INIT_READ_OP_ADDRESS) << 2;
poll = GET_FIELD(data, INIT_READ_OP_POLL_TYPE);
val = qed_rd(p_hwfn, p_ptt, addr); val = qed_rd(p_hwfn, p_ptt, addr);
data = le32_to_cpu(cmd->op_data); if (poll == INIT_POLL_NONE)
if (GET_FIELD(data, INIT_READ_OP_POLL)) { return;
int i;
switch (GET_FIELD(data, INIT_READ_OP_POLL_COMP)) { switch (poll) {
case INIT_COMPARISON_EQ: case INIT_POLL_EQ:
comp_check = comp_eq; comp_check = comp_eq;
break; break;
case INIT_COMPARISON_OR: case INIT_POLL_OR:
comp_check = comp_or; comp_check = comp_or;
break; break;
case INIT_COMPARISON_AND: case INIT_POLL_AND:
comp_check = comp_and; comp_check = comp_and;
break; break;
default: default:
comp_check = NULL;
DP_ERR(p_hwfn, "Invalid poll comparison type %08x\n", DP_ERR(p_hwfn, "Invalid poll comparison type %08x\n",
data); cmd->op_data);
return; return;
} }
data = le32_to_cpu(cmd->expected_val);
for (i = 0; for (i = 0;
i < QED_INIT_MAX_POLL_COUNT && i < QED_INIT_MAX_POLL_COUNT && !comp_check(val, data);
!comp_check(val, le32_to_cpu(cmd->expected_val));
i++) { i++) {
udelay(delay); udelay(delay);
val = qed_rd(p_hwfn, p_ptt, addr); val = qed_rd(p_hwfn, p_ptt, addr);
} }
if (i == QED_INIT_MAX_POLL_COUNT) if (i == QED_INIT_MAX_POLL_COUNT) {
DP_ERR(p_hwfn, DP_ERR(p_hwfn,
"Timeout when polling reg: 0x%08x [ Waiting-for: %08x Got: %08x (comparsion %08x)]\n", "Timeout when polling reg: 0x%08x [ Waiting-for: %08x Got: %08x (comparsion %08x)]\n",
addr, le32_to_cpu(cmd->expected_val), addr, le32_to_cpu(cmd->expected_val),
val, data); val, le32_to_cpu(cmd->op_data));
} }
} }
......
...@@ -714,7 +714,6 @@ qed_sp_eth_txq_start_ramrod(struct qed_hwfn *p_hwfn, ...@@ -714,7 +714,6 @@ qed_sp_eth_txq_start_ramrod(struct qed_hwfn *p_hwfn,
p_ramrod->sb_id = cpu_to_le16(p_params->sb); p_ramrod->sb_id = cpu_to_le16(p_params->sb);
p_ramrod->sb_index = p_params->sb_idx; p_ramrod->sb_index = p_params->sb_idx;
p_ramrod->stats_counter_id = stats_id; p_ramrod->stats_counter_id = stats_id;
p_ramrod->tc = p_pq_params->eth.tc;
p_ramrod->pbl_size = cpu_to_le16(pbl_size); p_ramrod->pbl_size = cpu_to_le16(pbl_size);
p_ramrod->pbl_base_addr.hi = DMA_HI_LE(pbl_addr); p_ramrod->pbl_base_addr.hi = DMA_HI_LE(pbl_addr);
...@@ -821,9 +820,8 @@ qed_filter_action(enum qed_filter_opcode opcode) ...@@ -821,9 +820,8 @@ qed_filter_action(enum qed_filter_opcode opcode)
case QED_FILTER_REMOVE: case QED_FILTER_REMOVE:
action = ETH_FILTER_ACTION_REMOVE; action = ETH_FILTER_ACTION_REMOVE;
break; break;
case QED_FILTER_REPLACE:
case QED_FILTER_FLUSH: case QED_FILTER_FLUSH:
action = ETH_FILTER_ACTION_REPLACE; action = ETH_FILTER_ACTION_REMOVE_ALL;
break; break;
default: default:
action = MAX_ETH_FILTER_ACTION; action = MAX_ETH_FILTER_ACTION;
...@@ -892,8 +890,7 @@ qed_filter_ucast_common(struct qed_hwfn *p_hwfn, ...@@ -892,8 +890,7 @@ qed_filter_ucast_common(struct qed_hwfn *p_hwfn,
p_ramrod->filter_cmd_hdr.tx = p_filter_cmd->is_tx_filter ? 1 : 0; p_ramrod->filter_cmd_hdr.tx = p_filter_cmd->is_tx_filter ? 1 : 0;
switch (p_filter_cmd->opcode) { switch (p_filter_cmd->opcode) {
case QED_FILTER_FLUSH: case QED_FILTER_REPLACE:
p_ramrod->filter_cmd_hdr.cmd_cnt = 0; break;
case QED_FILTER_MOVE: case QED_FILTER_MOVE:
p_ramrod->filter_cmd_hdr.cmd_cnt = 2; break; p_ramrod->filter_cmd_hdr.cmd_cnt = 2; break;
default: default:
...@@ -962,6 +959,12 @@ qed_filter_ucast_common(struct qed_hwfn *p_hwfn, ...@@ -962,6 +959,12 @@ qed_filter_ucast_common(struct qed_hwfn *p_hwfn,
p_second_filter->action = ETH_FILTER_ACTION_ADD; p_second_filter->action = ETH_FILTER_ACTION_ADD;
p_second_filter->vport_id = vport_to_add_to; p_second_filter->vport_id = vport_to_add_to;
} else if (p_filter_cmd->opcode == QED_FILTER_REPLACE) {
p_first_filter->vport_id = vport_to_add_to;
memcpy(p_second_filter, p_first_filter,
sizeof(*p_second_filter));
p_first_filter->action = ETH_FILTER_ACTION_REMOVE_ALL;
p_second_filter->action = ETH_FILTER_ACTION_ADD;
} else { } else {
action = qed_filter_action(p_filter_cmd->opcode); action = qed_filter_action(p_filter_cmd->opcode);
......
...@@ -190,7 +190,7 @@ int qed_fill_dev_info(struct qed_dev *cdev, ...@@ -190,7 +190,7 @@ int qed_fill_dev_info(struct qed_dev *cdev,
dev_info->pci_mem_start = cdev->pci_params.mem_start; dev_info->pci_mem_start = cdev->pci_params.mem_start;
dev_info->pci_mem_end = cdev->pci_params.mem_end; dev_info->pci_mem_end = cdev->pci_params.mem_end;
dev_info->pci_irq = cdev->pci_params.irq; dev_info->pci_irq = cdev->pci_params.irq;
dev_info->is_mf = IS_MF(&cdev->hwfns[0]); dev_info->is_mf_default = IS_MF_DEFAULT(&cdev->hwfns[0]);
ether_addr_copy(dev_info->hw_mac, cdev->hwfns[0].hw_info.hw_mac_addr); ether_addr_copy(dev_info->hw_mac, cdev->hwfns[0].hw_info.hw_mac_addr);
dev_info->fw_major = FW_MAJOR_VERSION; dev_info->fw_major = FW_MAJOR_VERSION;
......
...@@ -720,7 +720,7 @@ int qed_mcp_fill_shmem_func_info(struct qed_hwfn *p_hwfn, ...@@ -720,7 +720,7 @@ int qed_mcp_fill_shmem_func_info(struct qed_hwfn *p_hwfn,
return -EINVAL; return -EINVAL;
} }
if (p_hwfn->cdev->mf_mode != SF) {
info->bandwidth_min = (shmem_info.config & info->bandwidth_min = (shmem_info.config &
FUNC_MF_CFG_MIN_BW_MASK) >> FUNC_MF_CFG_MIN_BW_MASK) >>
FUNC_MF_CFG_MIN_BW_SHIFT; FUNC_MF_CFG_MIN_BW_SHIFT;
...@@ -740,7 +740,6 @@ int qed_mcp_fill_shmem_func_info(struct qed_hwfn *p_hwfn, ...@@ -740,7 +740,6 @@ int qed_mcp_fill_shmem_func_info(struct qed_hwfn *p_hwfn,
info->bandwidth_max); info->bandwidth_max);
info->bandwidth_max = 100; info->bandwidth_max = 100;
} }
}
if (shmem_info.mac_upper || shmem_info.mac_lower) { if (shmem_info.mac_upper || shmem_info.mac_lower) {
info->mac[0] = (u8)(shmem_info.mac_upper >> 8); info->mac[0] = (u8)(shmem_info.mac_upper >> 8);
......
...@@ -343,7 +343,7 @@ int qed_sp_init_request(struct qed_hwfn *p_hwfn, ...@@ -343,7 +343,7 @@ int qed_sp_init_request(struct qed_hwfn *p_hwfn,
*/ */
int qed_sp_pf_start(struct qed_hwfn *p_hwfn, int qed_sp_pf_start(struct qed_hwfn *p_hwfn,
enum mf_mode mode); enum qed_mf_mode mode);
/** /**
* @brief qed_sp_pf_stop - PF Function Stop Ramrod * @brief qed_sp_pf_stop - PF Function Stop Ramrod
......
...@@ -90,7 +90,7 @@ int qed_sp_init_request(struct qed_hwfn *p_hwfn, ...@@ -90,7 +90,7 @@ int qed_sp_init_request(struct qed_hwfn *p_hwfn,
} }
int qed_sp_pf_start(struct qed_hwfn *p_hwfn, int qed_sp_pf_start(struct qed_hwfn *p_hwfn,
enum mf_mode mode) enum qed_mf_mode mode)
{ {
struct qed_sp_init_request_params params; struct qed_sp_init_request_params params;
struct pf_start_ramrod_data *p_ramrod = NULL; struct pf_start_ramrod_data *p_ramrod = NULL;
...@@ -125,6 +125,18 @@ int qed_sp_pf_start(struct qed_hwfn *p_hwfn, ...@@ -125,6 +125,18 @@ int qed_sp_pf_start(struct qed_hwfn *p_hwfn,
p_ramrod->dont_log_ramrods = 0; p_ramrod->dont_log_ramrods = 0;
p_ramrod->log_type_mask = cpu_to_le16(0xf); p_ramrod->log_type_mask = cpu_to_le16(0xf);
p_ramrod->mf_mode = mode; p_ramrod->mf_mode = mode;
switch (mode) {
case QED_MF_DEFAULT:
case QED_MF_NPAR:
p_ramrod->mf_mode = MF_NPAR;
break;
case QED_MF_OVLAN:
p_ramrod->mf_mode = MF_OVLAN;
break;
default:
DP_NOTICE(p_hwfn, "Unsupported MF mode, init as DEFAULT\n");
p_ramrod->mf_mode = MF_NPAR;
}
p_ramrod->outer_tag = p_hwfn->hw_info.ovlan; p_ramrod->outer_tag = p_hwfn->hw_info.ovlan;
/* Place EQ address in RAMROD */ /* Place EQ address in RAMROD */
...@@ -142,9 +154,8 @@ int qed_sp_pf_start(struct qed_hwfn *p_hwfn, ...@@ -142,9 +154,8 @@ int qed_sp_pf_start(struct qed_hwfn *p_hwfn,
p_hwfn->hw_info.personality = PERSONALITY_ETH; p_hwfn->hw_info.personality = PERSONALITY_ETH;
DP_VERBOSE(p_hwfn, QED_MSG_SPQ, DP_VERBOSE(p_hwfn, QED_MSG_SPQ,
"Setting event_ring_sb [id %04x index %02x], mf [%s] outer_tag [%d]\n", "Setting event_ring_sb [id %04x index %02x], outer_tag [%d]\n",
sb, sb_index, sb, sb_index,
(p_ramrod->mf_mode == SF) ? "SF" : "Multi-Pf",
p_ramrod->outer_tag); p_ramrod->outer_tag);
return qed_spq_post(p_hwfn, p_ent, NULL); return qed_spq_post(p_hwfn, p_ent, NULL);
......
...@@ -173,9 +173,9 @@ enum QEDE_STATE { ...@@ -173,9 +173,9 @@ enum QEDE_STATE {
* skb are built only after the frame was DMA-ed. * skb are built only after the frame was DMA-ed.
*/ */
struct sw_rx_data { struct sw_rx_data {
u8 *data; struct page *data;
dma_addr_t mapping;
DEFINE_DMA_UNMAP_ADDR(mapping); unsigned int page_offset;
}; };
struct qede_rx_queue { struct qede_rx_queue {
...@@ -188,6 +188,7 @@ struct qede_rx_queue { ...@@ -188,6 +188,7 @@ struct qede_rx_queue {
void __iomem *hw_rxq_prod_addr; void __iomem *hw_rxq_prod_addr;
int rx_buf_size; int rx_buf_size;
unsigned int rx_buf_seg_size;
u16 num_rx_buffers; u16 num_rx_buffers;
u16 rxq_id; u16 rxq_id;
...@@ -281,6 +282,7 @@ void qede_fill_by_demand_stats(struct qede_dev *edev); ...@@ -281,6 +282,7 @@ void qede_fill_by_demand_stats(struct qede_dev *edev);
#define NUM_TX_BDS_MIN 128 #define NUM_TX_BDS_MIN 128
#define NUM_TX_BDS_DEF NUM_TX_BDS_MAX #define NUM_TX_BDS_DEF NUM_TX_BDS_MAX
#define QEDE_RX_HDR_SIZE 256
#define for_each_rss(i) for (i = 0; i < edev->num_rss; i++) #define for_each_rss(i) for (i = 0; i < edev->num_rss; i++)
#endif /* _QEDE_H_ */ #endif /* _QEDE_H_ */
...@@ -217,9 +217,9 @@ static int qede_set_settings(struct net_device *dev, struct ethtool_cmd *cmd) ...@@ -217,9 +217,9 @@ static int qede_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
struct qed_link_params params; struct qed_link_params params;
u32 speed; u32 speed;
if (edev->dev_info.common.is_mf) { if (!edev->dev_info.common.is_mf_default) {
DP_INFO(edev, DP_INFO(edev,
"Link parameters can not be changed in MF mode\n"); "Link parameters can not be changed in non-default mode\n");
return -EOPNOTSUPP; return -EOPNOTSUPP;
} }
...@@ -428,7 +428,7 @@ static int qede_set_pauseparam(struct net_device *dev, ...@@ -428,7 +428,7 @@ static int qede_set_pauseparam(struct net_device *dev,
struct qed_link_params params; struct qed_link_params params;
struct qed_link_output current_link; struct qed_link_output current_link;
if (!edev->dev_info.common.is_mf) { if (!edev->dev_info.common.is_mf_default) {
DP_INFO(edev, DP_INFO(edev,
"Pause parameters can not be updated in non-default mode\n"); "Pause parameters can not be updated in non-default mode\n");
return -EOPNOTSUPP; return -EOPNOTSUPP;
......
...@@ -330,15 +330,15 @@ static void qede_set_params_for_ipv6_ext(struct sk_buff *skb, ...@@ -330,15 +330,15 @@ static void qede_set_params_for_ipv6_ext(struct sk_buff *skb,
struct eth_tx_3rd_bd *third_bd) struct eth_tx_3rd_bd *third_bd)
{ {
u8 l4_proto; u8 l4_proto;
u16 bd2_bits = 0, bd2_bits2 = 0; u16 bd2_bits1 = 0, bd2_bits2 = 0;
bd2_bits2 |= (1 << ETH_TX_DATA_2ND_BD_IPV6_EXT_SHIFT); bd2_bits1 |= (1 << ETH_TX_DATA_2ND_BD_IPV6_EXT_SHIFT);
bd2_bits |= ((((u8 *)skb_transport_header(skb) - skb->data) >> 1) & bd2_bits2 |= ((((u8 *)skb_transport_header(skb) - skb->data) >> 1) &
ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_MASK) ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_MASK)
<< ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_SHIFT; << ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_SHIFT;
bd2_bits2 |= (ETH_L4_PSEUDO_CSUM_CORRECT_LENGTH << bd2_bits1 |= (ETH_L4_PSEUDO_CSUM_CORRECT_LENGTH <<
ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_SHIFT); ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_SHIFT);
if (vlan_get_protocol(skb) == htons(ETH_P_IPV6)) if (vlan_get_protocol(skb) == htons(ETH_P_IPV6))
...@@ -347,16 +347,15 @@ static void qede_set_params_for_ipv6_ext(struct sk_buff *skb, ...@@ -347,16 +347,15 @@ static void qede_set_params_for_ipv6_ext(struct sk_buff *skb,
l4_proto = ip_hdr(skb)->protocol; l4_proto = ip_hdr(skb)->protocol;
if (l4_proto == IPPROTO_UDP) if (l4_proto == IPPROTO_UDP)
bd2_bits2 |= 1 << ETH_TX_DATA_2ND_BD_L4_UDP_SHIFT; bd2_bits1 |= 1 << ETH_TX_DATA_2ND_BD_L4_UDP_SHIFT;
if (third_bd) { if (third_bd)
third_bd->data.bitfields |= third_bd->data.bitfields |=
((tcp_hdrlen(skb) / 4) & cpu_to_le16(((tcp_hdrlen(skb) / 4) &
ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_MASK) << ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_MASK) <<
ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_SHIFT; ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_SHIFT);
}
second_bd->data.bitfields = cpu_to_le16(bd2_bits); second_bd->data.bitfields1 = cpu_to_le16(bd2_bits1);
second_bd->data.bitfields2 = cpu_to_le16(bd2_bits2); second_bd->data.bitfields2 = cpu_to_le16(bd2_bits2);
} }
...@@ -464,12 +463,16 @@ netdev_tx_t qede_start_xmit(struct sk_buff *skb, ...@@ -464,12 +463,16 @@ netdev_tx_t qede_start_xmit(struct sk_buff *skb,
/* Fill the parsing flags & params according to the requested offload */ /* Fill the parsing flags & params according to the requested offload */
if (xmit_type & XMIT_L4_CSUM) { if (xmit_type & XMIT_L4_CSUM) {
u16 temp = 1 << ETH_TX_DATA_1ST_BD_TUNN_CFG_OVERRIDE_SHIFT;
/* We don't re-calculate IP checksum as it is already done by /* We don't re-calculate IP checksum as it is already done by
* the upper stack * the upper stack
*/ */
first_bd->data.bd_flags.bitfields |= first_bd->data.bd_flags.bitfields |=
1 << ETH_TX_1ST_BD_FLAGS_L4_CSUM_SHIFT; 1 << ETH_TX_1ST_BD_FLAGS_L4_CSUM_SHIFT;
first_bd->data.bitfields |= cpu_to_le16(temp);
/* If the packet is IPv6 with extension header, indicate that /* If the packet is IPv6 with extension header, indicate that
* to FW and pass few params, since the device cracker doesn't * to FW and pass few params, since the device cracker doesn't
* support parsing IPv6 with extension header/s. * support parsing IPv6 with extension header/s.
...@@ -491,7 +494,7 @@ netdev_tx_t qede_start_xmit(struct sk_buff *skb, ...@@ -491,7 +494,7 @@ netdev_tx_t qede_start_xmit(struct sk_buff *skb,
/* @@@TBD - if will not be removed need to check */ /* @@@TBD - if will not be removed need to check */
third_bd->data.bitfields |= third_bd->data.bitfields |=
(1 << ETH_TX_DATA_3RD_BD_HDR_NBD_SHIFT); cpu_to_le16((1 << ETH_TX_DATA_3RD_BD_HDR_NBD_SHIFT));
/* Make life easier for FW guys who can't deal with header and /* Make life easier for FW guys who can't deal with header and
* data on same BD. If we need to split, use the second bd... * data on same BD. If we need to split, use the second bd...
...@@ -719,26 +722,52 @@ static bool qede_has_tx_work(struct qede_fastpath *fp) ...@@ -719,26 +722,52 @@ static bool qede_has_tx_work(struct qede_fastpath *fp)
return false; return false;
} }
/* This function copies the Rx buffer from the CONS position to the PROD /* This function reuses the buffer(from an offset) from
* position, since we failed to allocate a new Rx buffer. * consumer index to producer index in the bd ring
*/ */
static void qede_reuse_rx_data(struct qede_rx_queue *rxq) static inline void qede_reuse_page(struct qede_dev *edev,
struct qede_rx_queue *rxq,
struct sw_rx_data *curr_cons)
{ {
struct eth_rx_bd *rx_bd_cons = qed_chain_consume(&rxq->rx_bd_ring);
struct eth_rx_bd *rx_bd_prod = qed_chain_produce(&rxq->rx_bd_ring); struct eth_rx_bd *rx_bd_prod = qed_chain_produce(&rxq->rx_bd_ring);
struct sw_rx_data *sw_rx_data_cons = struct sw_rx_data *curr_prod;
&rxq->sw_rx_ring[rxq->sw_rx_cons & NUM_RX_BDS_MAX]; dma_addr_t new_mapping;
struct sw_rx_data *sw_rx_data_prod =
&rxq->sw_rx_ring[rxq->sw_rx_prod & NUM_RX_BDS_MAX];
dma_unmap_addr_set(sw_rx_data_prod, mapping, curr_prod = &rxq->sw_rx_ring[rxq->sw_rx_prod & NUM_RX_BDS_MAX];
dma_unmap_addr(sw_rx_data_cons, mapping)); *curr_prod = *curr_cons;
sw_rx_data_prod->data = sw_rx_data_cons->data; new_mapping = curr_prod->mapping + curr_prod->page_offset;
memcpy(rx_bd_prod, rx_bd_cons, sizeof(struct eth_rx_bd));
rx_bd_prod->addr.hi = cpu_to_le32(upper_32_bits(new_mapping));
rx_bd_prod->addr.lo = cpu_to_le32(lower_32_bits(new_mapping));
rxq->sw_rx_cons++;
rxq->sw_rx_prod++; rxq->sw_rx_prod++;
curr_cons->data = NULL;
}
static inline int qede_realloc_rx_buffer(struct qede_dev *edev,
struct qede_rx_queue *rxq,
struct sw_rx_data *curr_cons)
{
/* Move to the next segment in the page */
curr_cons->page_offset += rxq->rx_buf_seg_size;
if (curr_cons->page_offset == PAGE_SIZE) {
if (unlikely(qede_alloc_rx_buffer(edev, rxq)))
return -ENOMEM;
dma_unmap_page(&edev->pdev->dev, curr_cons->mapping,
PAGE_SIZE, DMA_FROM_DEVICE);
} else {
/* Increment refcount of the page as we don't want
* network stack to take the ownership of the page
* which can be recycled multiple times by the driver.
*/
atomic_inc(&curr_cons->data->_count);
qede_reuse_page(edev, rxq, curr_cons);
}
return 0;
} }
static inline void qede_update_rx_prod(struct qede_dev *edev, static inline void qede_update_rx_prod(struct qede_dev *edev,
...@@ -857,9 +886,10 @@ static int qede_rx_int(struct qede_fastpath *fp, int budget) ...@@ -857,9 +886,10 @@ static int qede_rx_int(struct qede_fastpath *fp, int budget)
struct sw_rx_data *sw_rx_data; struct sw_rx_data *sw_rx_data;
union eth_rx_cqe *cqe; union eth_rx_cqe *cqe;
struct sk_buff *skb; struct sk_buff *skb;
struct page *data;
__le16 flags;
u16 len, pad; u16 len, pad;
u32 rx_hash; u32 rx_hash;
u8 *data;
/* Get the CQE from the completion ring */ /* Get the CQE from the completion ring */
cqe = (union eth_rx_cqe *) cqe = (union eth_rx_cqe *)
...@@ -879,56 +909,110 @@ static int qede_rx_int(struct qede_fastpath *fp, int budget) ...@@ -879,56 +909,110 @@ static int qede_rx_int(struct qede_fastpath *fp, int budget)
data = sw_rx_data->data; data = sw_rx_data->data;
fp_cqe = &cqe->fast_path_regular; fp_cqe = &cqe->fast_path_regular;
len = le16_to_cpu(fp_cqe->pkt_len); len = le16_to_cpu(fp_cqe->len_on_first_bd);
pad = fp_cqe->placement_offset; pad = fp_cqe->placement_offset;
flags = cqe->fast_path_regular.pars_flags.flags;
/* For every Rx BD consumed, we allocate a new BD so the BD ring
* is always with a fixed size. If allocation fails, we take the
* consumed BD and return it to the ring in the PROD position.
* The packet that was received on that BD will be dropped (and
* not passed to the upper stack).
*/
if (likely(qede_alloc_rx_buffer(edev, rxq) == 0)) {
dma_unmap_single(&edev->pdev->dev,
dma_unmap_addr(sw_rx_data, mapping),
rxq->rx_buf_size, DMA_FROM_DEVICE);
/* If this is an error packet then drop it */ /* If this is an error packet then drop it */
parse_flag = parse_flag = le16_to_cpu(flags);
le16_to_cpu(cqe->fast_path_regular.pars_flags.flags);
csum_flag = qede_check_csum(parse_flag); csum_flag = qede_check_csum(parse_flag);
if (csum_flag == QEDE_CSUM_ERROR) { if (unlikely(csum_flag == QEDE_CSUM_ERROR)) {
DP_NOTICE(edev, DP_NOTICE(edev,
"CQE in CONS = %u has error, flags = %x, dropping incoming packet\n", "CQE in CONS = %u has error, flags = %x, dropping incoming packet\n",
sw_comp_cons, parse_flag); sw_comp_cons, parse_flag);
rxq->rx_hw_errors++; rxq->rx_hw_errors++;
kfree(data); qede_reuse_page(edev, rxq, sw_rx_data);
goto next_rx; goto next_rx;
} }
skb = build_skb(data, 0); skb = netdev_alloc_skb(edev->ndev, QEDE_RX_HDR_SIZE);
if (unlikely(!skb)) { if (unlikely(!skb)) {
DP_NOTICE(edev, DP_NOTICE(edev,
"Build_skb failed, dropping incoming packet\n"); "Build_skb failed, dropping incoming packet\n");
kfree(data); qede_reuse_page(edev, rxq, sw_rx_data);
rxq->rx_alloc_errors++; rxq->rx_alloc_errors++;
goto next_rx; goto next_rx;
} }
skb_reserve(skb, pad); /* Copy data into SKB */
if (len + pad <= QEDE_RX_HDR_SIZE) {
memcpy(skb_put(skb, len),
page_address(data) + pad +
sw_rx_data->page_offset, len);
qede_reuse_page(edev, rxq, sw_rx_data);
} else { } else {
DP_NOTICE(edev, struct skb_frag_struct *frag;
"New buffer allocation failed, dropping incoming packet and reusing its buffer\n"); unsigned int pull_len;
qede_reuse_rx_data(rxq); unsigned char *va;
frag = &skb_shinfo(skb)->frags[0];
skb_add_rx_frag(skb, skb_shinfo(skb)->nr_frags, data,
pad + sw_rx_data->page_offset,
len, rxq->rx_buf_seg_size);
va = skb_frag_address(frag);
pull_len = eth_get_headlen(va, QEDE_RX_HDR_SIZE);
/* Align the pull_len to optimize memcpy */
memcpy(skb->data, va, ALIGN(pull_len, sizeof(long)));
skb_frag_size_sub(frag, pull_len);
frag->page_offset += pull_len;
skb->data_len -= pull_len;
skb->tail += pull_len;
if (unlikely(qede_realloc_rx_buffer(edev, rxq,
sw_rx_data))) {
DP_ERR(edev, "Failed to allocate rx buffer\n");
rxq->rx_alloc_errors++; rxq->rx_alloc_errors++;
goto next_cqe; goto next_cqe;
} }
}
if (fp_cqe->bd_num != 1) {
u16 pkt_len = le16_to_cpu(fp_cqe->pkt_len);
u8 num_frags;
pkt_len -= len;
for (num_frags = fp_cqe->bd_num - 1; num_frags > 0;
num_frags--) {
u16 cur_size = pkt_len > rxq->rx_buf_size ?
rxq->rx_buf_size : pkt_len;
WARN_ONCE(!cur_size,
"Still got %d BDs for mapping jumbo, but length became 0\n",
num_frags);
if (unlikely(qede_alloc_rx_buffer(edev, rxq)))
goto next_cqe;
rxq->sw_rx_cons++;
sw_rx_index = rxq->sw_rx_cons & NUM_RX_BDS_MAX;
sw_rx_data = &rxq->sw_rx_ring[sw_rx_index];
qed_chain_consume(&rxq->rx_bd_ring);
dma_unmap_page(&edev->pdev->dev,
sw_rx_data->mapping,
PAGE_SIZE, DMA_FROM_DEVICE);
skb_fill_page_desc(skb,
skb_shinfo(skb)->nr_frags++,
sw_rx_data->data, 0,
cur_size);
sw_rx_data->data = NULL; skb->truesize += PAGE_SIZE;
skb->data_len += cur_size;
skb->len += cur_size;
pkt_len -= cur_size;
}
skb_put(skb, len); if (pkt_len)
DP_ERR(edev,
"Mapped all BDs of jumbo, but still have %d bytes\n",
pkt_len);
}
skb->protocol = eth_type_trans(skb, edev->ndev); skb->protocol = eth_type_trans(skb, edev->ndev);
...@@ -1566,17 +1650,17 @@ static void qede_free_rx_buffers(struct qede_dev *edev, ...@@ -1566,17 +1650,17 @@ static void qede_free_rx_buffers(struct qede_dev *edev,
for (i = rxq->sw_rx_cons; i != rxq->sw_rx_prod; i++) { for (i = rxq->sw_rx_cons; i != rxq->sw_rx_prod; i++) {
struct sw_rx_data *rx_buf; struct sw_rx_data *rx_buf;
u8 *data; struct page *data;
rx_buf = &rxq->sw_rx_ring[i & NUM_RX_BDS_MAX]; rx_buf = &rxq->sw_rx_ring[i & NUM_RX_BDS_MAX];
data = rx_buf->data; data = rx_buf->data;
dma_unmap_single(&edev->pdev->dev, dma_unmap_page(&edev->pdev->dev,
dma_unmap_addr(rx_buf, mapping), rx_buf->mapping,
rxq->rx_buf_size, DMA_FROM_DEVICE); PAGE_SIZE, DMA_FROM_DEVICE);
rx_buf->data = NULL; rx_buf->data = NULL;
kfree(data); __free_page(data);
} }
} }
...@@ -1600,29 +1684,32 @@ static int qede_alloc_rx_buffer(struct qede_dev *edev, ...@@ -1600,29 +1684,32 @@ static int qede_alloc_rx_buffer(struct qede_dev *edev,
struct sw_rx_data *sw_rx_data; struct sw_rx_data *sw_rx_data;
struct eth_rx_bd *rx_bd; struct eth_rx_bd *rx_bd;
dma_addr_t mapping; dma_addr_t mapping;
struct page *data;
u16 rx_buf_size; u16 rx_buf_size;
u8 *data;
rx_buf_size = rxq->rx_buf_size; rx_buf_size = rxq->rx_buf_size;
data = kmalloc(rx_buf_size, GFP_ATOMIC); data = alloc_pages(GFP_ATOMIC, 0);
if (unlikely(!data)) { if (unlikely(!data)) {
DP_NOTICE(edev, "Failed to allocate Rx data\n"); DP_NOTICE(edev, "Failed to allocate Rx data [page]\n");
return -ENOMEM; return -ENOMEM;
} }
mapping = dma_map_single(&edev->pdev->dev, data, /* Map the entire page as it would be used
rx_buf_size, DMA_FROM_DEVICE); * for multiple RX buffer segment size mapping.
*/
mapping = dma_map_page(&edev->pdev->dev, data, 0,
PAGE_SIZE, DMA_FROM_DEVICE);
if (unlikely(dma_mapping_error(&edev->pdev->dev, mapping))) { if (unlikely(dma_mapping_error(&edev->pdev->dev, mapping))) {
kfree(data); __free_page(data);
DP_NOTICE(edev, "Failed to map Rx buffer\n"); DP_NOTICE(edev, "Failed to map Rx buffer\n");
return -ENOMEM; return -ENOMEM;
} }
sw_rx_data = &rxq->sw_rx_ring[rxq->sw_rx_prod & NUM_RX_BDS_MAX]; sw_rx_data = &rxq->sw_rx_ring[rxq->sw_rx_prod & NUM_RX_BDS_MAX];
sw_rx_data->page_offset = 0;
sw_rx_data->data = data; sw_rx_data->data = data;
sw_rx_data->mapping = mapping;
dma_unmap_addr_set(sw_rx_data, mapping, mapping);
/* Advance PROD and get BD pointer */ /* Advance PROD and get BD pointer */
rx_bd = (struct eth_rx_bd *)qed_chain_produce(&rxq->rx_bd_ring); rx_bd = (struct eth_rx_bd *)qed_chain_produce(&rxq->rx_bd_ring);
...@@ -1643,13 +1730,16 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev, ...@@ -1643,13 +1730,16 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev,
rxq->num_rx_buffers = edev->q_num_rx_buffers; rxq->num_rx_buffers = edev->q_num_rx_buffers;
rxq->rx_buf_size = NET_IP_ALIGN + rxq->rx_buf_size = NET_IP_ALIGN + ETH_OVERHEAD +
ETH_OVERHEAD + edev->ndev->mtu;
edev->ndev->mtu + if (rxq->rx_buf_size > PAGE_SIZE)
QEDE_FW_RX_ALIGN_END; rxq->rx_buf_size = PAGE_SIZE;
/* Segment size to spilt a page in multiple equal parts */
rxq->rx_buf_seg_size = roundup_pow_of_two(rxq->rx_buf_size);
/* Allocate the parallel driver ring for Rx buffers */ /* Allocate the parallel driver ring for Rx buffers */
size = sizeof(*rxq->sw_rx_ring) * NUM_RX_BDS_MAX; size = sizeof(*rxq->sw_rx_ring) * RX_RING_SIZE;
rxq->sw_rx_ring = kzalloc(size, GFP_KERNEL); rxq->sw_rx_ring = kzalloc(size, GFP_KERNEL);
if (!rxq->sw_rx_ring) { if (!rxq->sw_rx_ring) {
DP_ERR(edev, "Rx buffers ring allocation failed\n"); DP_ERR(edev, "Rx buffers ring allocation failed\n");
...@@ -1660,7 +1750,7 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev, ...@@ -1660,7 +1750,7 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev,
rc = edev->ops->common->chain_alloc(edev->cdev, rc = edev->ops->common->chain_alloc(edev->cdev,
QED_CHAIN_USE_TO_CONSUME_PRODUCE, QED_CHAIN_USE_TO_CONSUME_PRODUCE,
QED_CHAIN_MODE_NEXT_PTR, QED_CHAIN_MODE_NEXT_PTR,
NUM_RX_BDS_MAX, RX_RING_SIZE,
sizeof(struct eth_rx_bd), sizeof(struct eth_rx_bd),
&rxq->rx_bd_ring); &rxq->rx_bd_ring);
...@@ -1671,7 +1761,7 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev, ...@@ -1671,7 +1761,7 @@ static int qede_alloc_mem_rxq(struct qede_dev *edev,
rc = edev->ops->common->chain_alloc(edev->cdev, rc = edev->ops->common->chain_alloc(edev->cdev,
QED_CHAIN_USE_TO_CONSUME, QED_CHAIN_USE_TO_CONSUME,
QED_CHAIN_MODE_PBL, QED_CHAIN_MODE_PBL,
NUM_RX_BDS_MAX, RX_RING_SIZE,
sizeof(union eth_rx_cqe), sizeof(union eth_rx_cqe),
&rxq->rx_comp_ring); &rxq->rx_comp_ring);
if (rc) if (rc)
......
...@@ -11,9 +11,11 @@ ...@@ -11,9 +11,11 @@
#define CORE_SPQE_PAGE_SIZE_BYTES 4096 #define CORE_SPQE_PAGE_SIZE_BYTES 4096
#define X_FINAL_CLEANUP_AGG_INT 1
#define FW_MAJOR_VERSION 8 #define FW_MAJOR_VERSION 8
#define FW_MINOR_VERSION 4 #define FW_MINOR_VERSION 7
#define FW_REVISION_VERSION 2 #define FW_REVISION_VERSION 3
#define FW_ENGINEERING_VERSION 0 #define FW_ENGINEERING_VERSION 0
/***********************/ /***********************/
...@@ -152,6 +154,9 @@ ...@@ -152,6 +154,9 @@
/* number of queues in a PF queue group */ /* number of queues in a PF queue group */
#define QM_PF_QUEUE_GROUP_SIZE 8 #define QM_PF_QUEUE_GROUP_SIZE 8
/* the size of a single queue element in bytes */
#define QM_PQ_ELEMENT_SIZE 4
/* base number of Tx PQs in the CM PQ representation. /* base number of Tx PQs in the CM PQ representation.
* should be used when storing PQ IDs in CM PQ registers and context * should be used when storing PQ IDs in CM PQ registers and context
*/ */
...@@ -285,6 +290,16 @@ ...@@ -285,6 +290,16 @@
#define PXP_NUM_ILT_RECORDS_K2 11000 #define PXP_NUM_ILT_RECORDS_K2 11000
#define MAX_NUM_ILT_RECORDS MAX(PXP_NUM_ILT_RECORDS_BB, PXP_NUM_ILT_RECORDS_K2) #define MAX_NUM_ILT_RECORDS MAX(PXP_NUM_ILT_RECORDS_BB, PXP_NUM_ILT_RECORDS_K2)
#define SDM_COMP_TYPE_NONE 0
#define SDM_COMP_TYPE_WAKE_THREAD 1
#define SDM_COMP_TYPE_AGG_INT 2
#define SDM_COMP_TYPE_CM 3
#define SDM_COMP_TYPE_LOADER 4
#define SDM_COMP_TYPE_PXP 5
#define SDM_COMP_TYPE_INDICATE_ERROR 6
#define SDM_COMP_TYPE_RELEASE_THREAD 7
#define SDM_COMP_TYPE_RAM 8
/******************/ /******************/
/* PBF CONSTANTS */ /* PBF CONSTANTS */
/******************/ /******************/
...@@ -335,7 +350,7 @@ struct event_ring_entry { ...@@ -335,7 +350,7 @@ struct event_ring_entry {
/* Multi function mode */ /* Multi function mode */
enum mf_mode { enum mf_mode {
SF, ERROR_MODE /* Unsupported mode */,
MF_OVLAN, MF_OVLAN,
MF_NPAR, MF_NPAR,
MAX_MF_MODE MAX_MF_MODE
...@@ -606,4 +621,19 @@ struct status_block { ...@@ -606,4 +621,19 @@ struct status_block {
#define STATUS_BLOCK_ZERO_PAD3_SHIFT 24 #define STATUS_BLOCK_ZERO_PAD3_SHIFT 24
}; };
struct tunnel_parsing_flags {
u8 flags;
#define TUNNEL_PARSING_FLAGS_TYPE_MASK 0x3
#define TUNNEL_PARSING_FLAGS_TYPE_SHIFT 0
#define TUNNEL_PARSING_FLAGS_TENNANT_ID_EXIST_MASK 0x1
#define TUNNEL_PARSING_FLAGS_TENNANT_ID_EXIST_SHIFT 2
#define TUNNEL_PARSING_FLAGS_NEXT_PROTOCOL_MASK 0x3
#define TUNNEL_PARSING_FLAGS_NEXT_PROTOCOL_SHIFT 3
#define TUNNEL_PARSING_FLAGS_FIRSTHDRIPMATCH_MASK 0x1
#define TUNNEL_PARSING_FLAGS_FIRSTHDRIPMATCH_SHIFT 5
#define TUNNEL_PARSING_FLAGS_IPV4_FRAGMENT_MASK 0x1
#define TUNNEL_PARSING_FLAGS_IPV4_FRAGMENT_SHIFT 6
#define TUNNEL_PARSING_FLAGS_IPV4_OPTIONS_MASK 0x1
#define TUNNEL_PARSING_FLAGS_IPV4_OPTIONS_SHIFT 7
};
#endif /* __COMMON_HSI__ */ #endif /* __COMMON_HSI__ */
...@@ -17,10 +17,8 @@ ...@@ -17,10 +17,8 @@
#define ETH_MAX_RAMROD_PER_CON 8 #define ETH_MAX_RAMROD_PER_CON 8
#define ETH_TX_BD_PAGE_SIZE_BYTES 4096 #define ETH_TX_BD_PAGE_SIZE_BYTES 4096
#define ETH_RX_BD_PAGE_SIZE_BYTES 4096 #define ETH_RX_BD_PAGE_SIZE_BYTES 4096
#define ETH_RX_SGE_PAGE_SIZE_BYTES 4096
#define ETH_RX_CQE_PAGE_SIZE_BYTES 4096 #define ETH_RX_CQE_PAGE_SIZE_BYTES 4096
#define ETH_RX_NUM_NEXT_PAGE_BDS 2 #define ETH_RX_NUM_NEXT_PAGE_BDS 2
#define ETH_RX_NUM_NEXT_PAGE_SGES 2
#define ETH_TX_MIN_BDS_PER_NON_LSO_PKT 1 #define ETH_TX_MIN_BDS_PER_NON_LSO_PKT 1
#define ETH_TX_MAX_BDS_PER_NON_LSO_PACKET 18 #define ETH_TX_MAX_BDS_PER_NON_LSO_PACKET 18
...@@ -34,7 +32,8 @@ ...@@ -34,7 +32,8 @@
#define ETH_NUM_STATISTIC_COUNTERS MAX_NUM_VPORTS #define ETH_NUM_STATISTIC_COUNTERS MAX_NUM_VPORTS
#define ETH_REG_CQE_PBL_SIZE 3 /* Maximum number of buffers, used for RX packet placement */
#define ETH_RX_MAX_BUFF_PER_PKT 5
/* num of MAC/VLAN filters */ /* num of MAC/VLAN filters */
#define ETH_NUM_MAC_FILTERS 512 #define ETH_NUM_MAC_FILTERS 512
...@@ -54,9 +53,9 @@ ...@@ -54,9 +53,9 @@
/* TPA constants */ /* TPA constants */
#define ETH_TPA_MAX_AGGS_NUM 64 #define ETH_TPA_MAX_AGGS_NUM 64
#define ETH_TPA_CQE_START_SGL_SIZE 3 #define ETH_TPA_CQE_START_LEN_LIST_SIZE ETH_RX_MAX_BUFF_PER_PKT
#define ETH_TPA_CQE_CONT_SGL_SIZE 6 #define ETH_TPA_CQE_CONT_LEN_LIST_SIZE 6
#define ETH_TPA_CQE_END_SGL_SIZE 4 #define ETH_TPA_CQE_END_LEN_LIST_SIZE 4
/* Queue Zone sizes */ /* Queue Zone sizes */
#define TSTORM_QZONE_SIZE 0 #define TSTORM_QZONE_SIZE 0
...@@ -74,18 +73,18 @@ struct coalescing_timeset { ...@@ -74,18 +73,18 @@ struct coalescing_timeset {
struct eth_tx_1st_bd_flags { struct eth_tx_1st_bd_flags {
u8 bitfields; u8 bitfields;
#define ETH_TX_1ST_BD_FLAGS_START_BD_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_START_BD_SHIFT 0
#define ETH_TX_1ST_BD_FLAGS_FORCE_VLAN_MODE_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_FORCE_VLAN_MODE_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_FORCE_VLAN_MODE_SHIFT 0 #define ETH_TX_1ST_BD_FLAGS_FORCE_VLAN_MODE_SHIFT 1
#define ETH_TX_1ST_BD_FLAGS_IP_CSUM_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_IP_CSUM_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_IP_CSUM_SHIFT 1 #define ETH_TX_1ST_BD_FLAGS_IP_CSUM_SHIFT 2
#define ETH_TX_1ST_BD_FLAGS_L4_CSUM_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_L4_CSUM_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_L4_CSUM_SHIFT 2 #define ETH_TX_1ST_BD_FLAGS_L4_CSUM_SHIFT 3
#define ETH_TX_1ST_BD_FLAGS_VLAN_INSERTION_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_VLAN_INSERTION_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_VLAN_INSERTION_SHIFT 3 #define ETH_TX_1ST_BD_FLAGS_VLAN_INSERTION_SHIFT 4
#define ETH_TX_1ST_BD_FLAGS_LSO_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_LSO_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_LSO_SHIFT 4 #define ETH_TX_1ST_BD_FLAGS_LSO_SHIFT 5
#define ETH_TX_1ST_BD_FLAGS_START_BD_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_START_BD_SHIFT 5
#define ETH_TX_1ST_BD_FLAGS_TUNN_IP_CSUM_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_TUNN_IP_CSUM_MASK 0x1
#define ETH_TX_1ST_BD_FLAGS_TUNN_IP_CSUM_SHIFT 6 #define ETH_TX_1ST_BD_FLAGS_TUNN_IP_CSUM_SHIFT 6
#define ETH_TX_1ST_BD_FLAGS_TUNN_L4_CSUM_MASK 0x1 #define ETH_TX_1ST_BD_FLAGS_TUNN_L4_CSUM_MASK 0x1
...@@ -97,38 +96,44 @@ struct eth_tx_data_1st_bd { ...@@ -97,38 +96,44 @@ struct eth_tx_data_1st_bd {
__le16 vlan; __le16 vlan;
u8 nbds; u8 nbds;
struct eth_tx_1st_bd_flags bd_flags; struct eth_tx_1st_bd_flags bd_flags;
__le16 fw_use_only; __le16 bitfields;
#define ETH_TX_DATA_1ST_BD_TUNN_CFG_OVERRIDE_MASK 0x1
#define ETH_TX_DATA_1ST_BD_TUNN_CFG_OVERRIDE_SHIFT 0
#define ETH_TX_DATA_1ST_BD_RESERVED0_MASK 0x1
#define ETH_TX_DATA_1ST_BD_RESERVED0_SHIFT 1
#define ETH_TX_DATA_1ST_BD_FW_USE_ONLY_MASK 0x3FFF
#define ETH_TX_DATA_1ST_BD_FW_USE_ONLY_SHIFT 2
}; };
/* The parsing information data for the second tx bd of a given packet. */ /* The parsing information data for the second tx bd of a given packet. */
struct eth_tx_data_2nd_bd { struct eth_tx_data_2nd_bd {
__le16 tunn_ip_size; __le16 tunn_ip_size;
__le16 bitfields; __le16 bitfields1;
#define ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_MASK 0x1FFF
#define ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_SHIFT 0
#define ETH_TX_DATA_2ND_BD_RESERVED0_MASK 0x7
#define ETH_TX_DATA_2ND_BD_RESERVED0_SHIFT 13
__le16 bitfields2;
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_L2_HDR_SIZE_W_MASK 0xF #define ETH_TX_DATA_2ND_BD_TUNN_INNER_L2_HDR_SIZE_W_MASK 0xF
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_L2_HDR_SIZE_W_SHIFT 0 #define ETH_TX_DATA_2ND_BD_TUNN_INNER_L2_HDR_SIZE_W_SHIFT 0
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_ETH_TYPE_MASK 0x3 #define ETH_TX_DATA_2ND_BD_TUNN_INNER_ETH_TYPE_MASK 0x3
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_ETH_TYPE_SHIFT 4 #define ETH_TX_DATA_2ND_BD_TUNN_INNER_ETH_TYPE_SHIFT 4
#define ETH_TX_DATA_2ND_BD_DEST_PORT_MODE_MASK 0x3 #define ETH_TX_DATA_2ND_BD_DEST_PORT_MODE_MASK 0x3
#define ETH_TX_DATA_2ND_BD_DEST_PORT_MODE_SHIFT 6 #define ETH_TX_DATA_2ND_BD_DEST_PORT_MODE_SHIFT 6
#define ETH_TX_DATA_2ND_BD_START_BD_MASK 0x1
#define ETH_TX_DATA_2ND_BD_START_BD_SHIFT 8
#define ETH_TX_DATA_2ND_BD_TUNN_TYPE_MASK 0x3 #define ETH_TX_DATA_2ND_BD_TUNN_TYPE_MASK 0x3
#define ETH_TX_DATA_2ND_BD_TUNN_TYPE_SHIFT 8 #define ETH_TX_DATA_2ND_BD_TUNN_TYPE_SHIFT 9
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_IPV6_MASK 0x1 #define ETH_TX_DATA_2ND_BD_TUNN_INNER_IPV6_MASK 0x1
#define ETH_TX_DATA_2ND_BD_TUNN_INNER_IPV6_SHIFT 10 #define ETH_TX_DATA_2ND_BD_TUNN_INNER_IPV6_SHIFT 11
#define ETH_TX_DATA_2ND_BD_IPV6_EXT_MASK 0x1 #define ETH_TX_DATA_2ND_BD_IPV6_EXT_MASK 0x1
#define ETH_TX_DATA_2ND_BD_IPV6_EXT_SHIFT 11 #define ETH_TX_DATA_2ND_BD_IPV6_EXT_SHIFT 12
#define ETH_TX_DATA_2ND_BD_TUNN_IPV6_EXT_MASK 0x1 #define ETH_TX_DATA_2ND_BD_TUNN_IPV6_EXT_MASK 0x1
#define ETH_TX_DATA_2ND_BD_TUNN_IPV6_EXT_SHIFT 12 #define ETH_TX_DATA_2ND_BD_TUNN_IPV6_EXT_SHIFT 13
#define ETH_TX_DATA_2ND_BD_L4_UDP_MASK 0x1 #define ETH_TX_DATA_2ND_BD_L4_UDP_MASK 0x1
#define ETH_TX_DATA_2ND_BD_L4_UDP_SHIFT 13 #define ETH_TX_DATA_2ND_BD_L4_UDP_SHIFT 14
#define ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_MASK 0x1 #define ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_MASK 0x1
#define ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_SHIFT 14 #define ETH_TX_DATA_2ND_BD_L4_PSEUDO_CSUM_MODE_SHIFT 15
#define ETH_TX_DATA_2ND_BD_RESERVED1_MASK 0x1 __le16 bitfields2;
#define ETH_TX_DATA_2ND_BD_RESERVED1_SHIFT 15 #define ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_MASK 0x1FFF
#define ETH_TX_DATA_2ND_BD_L4_HDR_START_OFFSET_W_SHIFT 0
#define ETH_TX_DATA_2ND_BD_RESERVED0_MASK 0x7
#define ETH_TX_DATA_2ND_BD_RESERVED0_SHIFT 13
}; };
/* Regular ETH Rx FP CQE. */ /* Regular ETH Rx FP CQE. */
...@@ -145,11 +150,68 @@ struct eth_fast_path_rx_reg_cqe { ...@@ -145,11 +150,68 @@ struct eth_fast_path_rx_reg_cqe {
struct parsing_and_err_flags pars_flags; struct parsing_and_err_flags pars_flags;
__le16 vlan_tag; __le16 vlan_tag;
__le32 rss_hash; __le32 rss_hash;
__le16 len_on_bd; __le16 len_on_first_bd;
u8 placement_offset; u8 placement_offset;
u8 reserved; struct tunnel_parsing_flags tunnel_pars_flags;
__le16 pbl[ETH_REG_CQE_PBL_SIZE]; u8 bd_num;
u8 reserved1[10]; u8 reserved[7];
u32 fw_debug;
u8 reserved1[3];
u8 flags;
#define ETH_FAST_PATH_RX_REG_CQE_VALID_MASK 0x1
#define ETH_FAST_PATH_RX_REG_CQE_VALID_SHIFT 0
#define ETH_FAST_PATH_RX_REG_CQE_VALID_TOGGLE_MASK 0x1
#define ETH_FAST_PATH_RX_REG_CQE_VALID_TOGGLE_SHIFT 1
#define ETH_FAST_PATH_RX_REG_CQE_RESERVED2_MASK 0x3F
#define ETH_FAST_PATH_RX_REG_CQE_RESERVED2_SHIFT 2
};
/* TPA-continue ETH Rx FP CQE. */
struct eth_fast_path_rx_tpa_cont_cqe {
u8 type;
u8 tpa_agg_index;
__le16 len_list[ETH_TPA_CQE_CONT_LEN_LIST_SIZE];
u8 reserved[5];
u8 reserved1;
__le16 reserved2[ETH_TPA_CQE_CONT_LEN_LIST_SIZE];
};
/* TPA-end ETH Rx FP CQE. */
struct eth_fast_path_rx_tpa_end_cqe {
u8 type;
u8 tpa_agg_index;
__le16 total_packet_len;
u8 num_of_bds;
u8 end_reason;
__le16 num_of_coalesced_segs;
__le32 ts_delta;
__le16 len_list[ETH_TPA_CQE_END_LEN_LIST_SIZE];
u8 reserved1[3];
u8 reserved2;
__le16 reserved3[ETH_TPA_CQE_END_LEN_LIST_SIZE];
};
/* TPA-start ETH Rx FP CQE. */
struct eth_fast_path_rx_tpa_start_cqe {
u8 type;
u8 bitfields;
#define ETH_FAST_PATH_RX_TPA_START_CQE_RSS_HASH_TYPE_MASK 0x7
#define ETH_FAST_PATH_RX_TPA_START_CQE_RSS_HASH_TYPE_SHIFT 0
#define ETH_FAST_PATH_RX_TPA_START_CQE_TC_MASK 0xF
#define ETH_FAST_PATH_RX_TPA_START_CQE_TC_SHIFT 3
#define ETH_FAST_PATH_RX_TPA_START_CQE_RESERVED0_MASK 0x1
#define ETH_FAST_PATH_RX_TPA_START_CQE_RESERVED0_SHIFT 7
__le16 seg_len;
struct parsing_and_err_flags pars_flags;
__le16 vlan_tag;
__le32 rss_hash;
__le16 len_on_first_bd;
u8 placement_offset;
struct tunnel_parsing_flags tunnel_pars_flags;
u8 tpa_agg_index;
u8 header_len;
__le16 ext_bd_len_list[ETH_TPA_CQE_START_LEN_LIST_SIZE];
u32 fw_debug;
}; };
/* The L4 pseudo checksum mode for Ethernet */ /* The L4 pseudo checksum mode for Ethernet */
...@@ -168,13 +230,26 @@ struct eth_slow_path_rx_cqe { ...@@ -168,13 +230,26 @@ struct eth_slow_path_rx_cqe {
u8 type; u8 type;
u8 ramrod_cmd_id; u8 ramrod_cmd_id;
u8 error_flag; u8 error_flag;
u8 reserved[27]; u8 reserved[25];
__le16 echo; __le16 echo;
u8 reserved1;
u8 flags;
/* for PMD mode - valid indication */
#define ETH_SLOW_PATH_RX_CQE_VALID_MASK 0x1
#define ETH_SLOW_PATH_RX_CQE_VALID_SHIFT 0
/* for PMD mode - valid toggle indication */
#define ETH_SLOW_PATH_RX_CQE_VALID_TOGGLE_MASK 0x1
#define ETH_SLOW_PATH_RX_CQE_VALID_TOGGLE_SHIFT 1
#define ETH_SLOW_PATH_RX_CQE_RESERVED2_MASK 0x3F
#define ETH_SLOW_PATH_RX_CQE_RESERVED2_SHIFT 2
}; };
/* union for all ETH Rx CQE types */ /* union for all ETH Rx CQE types */
union eth_rx_cqe { union eth_rx_cqe {
struct eth_fast_path_rx_reg_cqe fast_path_regular; struct eth_fast_path_rx_reg_cqe fast_path_regular;
struct eth_fast_path_rx_tpa_start_cqe fast_path_tpa_start;
struct eth_fast_path_rx_tpa_cont_cqe fast_path_tpa_cont;
struct eth_fast_path_rx_tpa_end_cqe fast_path_tpa_end;
struct eth_slow_path_rx_cqe slow_path; struct eth_slow_path_rx_cqe slow_path;
}; };
...@@ -183,15 +258,18 @@ enum eth_rx_cqe_type { ...@@ -183,15 +258,18 @@ enum eth_rx_cqe_type {
ETH_RX_CQE_TYPE_UNUSED, ETH_RX_CQE_TYPE_UNUSED,
ETH_RX_CQE_TYPE_REGULAR, ETH_RX_CQE_TYPE_REGULAR,
ETH_RX_CQE_TYPE_SLOW_PATH, ETH_RX_CQE_TYPE_SLOW_PATH,
ETH_RX_CQE_TYPE_TPA_START,
ETH_RX_CQE_TYPE_TPA_CONT,
ETH_RX_CQE_TYPE_TPA_END,
MAX_ETH_RX_CQE_TYPE MAX_ETH_RX_CQE_TYPE
}; };
/* ETH Rx producers data */ /* ETH Rx producers data */
struct eth_rx_prod_data { struct eth_rx_prod_data {
__le16 bd_prod; __le16 bd_prod;
__le16 sge_prod;
__le16 cqe_prod; __le16 cqe_prod;
__le16 reserved; __le16 reserved;
__le16 reserved1;
}; };
/* The first tx bd of a given packet */ /* The first tx bd of a given packet */
...@@ -211,12 +289,17 @@ struct eth_tx_2nd_bd { ...@@ -211,12 +289,17 @@ struct eth_tx_2nd_bd {
/* The parsing information data for the third tx bd of a given packet. */ /* The parsing information data for the third tx bd of a given packet. */
struct eth_tx_data_3rd_bd { struct eth_tx_data_3rd_bd {
__le16 lso_mss; __le16 lso_mss;
u8 bitfields; __le16 bitfields;
#define ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_MASK 0xF #define ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_MASK 0xF
#define ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_SHIFT 0 #define ETH_TX_DATA_3RD_BD_TCP_HDR_LEN_DW_SHIFT 0
#define ETH_TX_DATA_3RD_BD_HDR_NBD_MASK 0xF #define ETH_TX_DATA_3RD_BD_HDR_NBD_MASK 0xF
#define ETH_TX_DATA_3RD_BD_HDR_NBD_SHIFT 4 #define ETH_TX_DATA_3RD_BD_HDR_NBD_SHIFT 4
u8 resereved0[3]; #define ETH_TX_DATA_3RD_BD_START_BD_MASK 0x1
#define ETH_TX_DATA_3RD_BD_START_BD_SHIFT 8
#define ETH_TX_DATA_3RD_BD_RESERVED0_MASK 0x7F
#define ETH_TX_DATA_3RD_BD_RESERVED0_SHIFT 9
u8 tunn_l4_hdr_start_offset_w;
u8 tunn_hdr_size_w;
}; };
/* The third tx bd of a given packet */ /* The third tx bd of a given packet */
...@@ -226,12 +309,24 @@ struct eth_tx_3rd_bd { ...@@ -226,12 +309,24 @@ struct eth_tx_3rd_bd {
struct eth_tx_data_3rd_bd data; struct eth_tx_data_3rd_bd data;
}; };
/* Complementary information for the regular tx bd of a given packet. */
struct eth_tx_data_bd {
__le16 reserved0;
__le16 bitfields;
#define ETH_TX_DATA_BD_RESERVED1_MASK 0xFF
#define ETH_TX_DATA_BD_RESERVED1_SHIFT 0
#define ETH_TX_DATA_BD_START_BD_MASK 0x1
#define ETH_TX_DATA_BD_START_BD_SHIFT 8
#define ETH_TX_DATA_BD_RESERVED2_MASK 0x7F
#define ETH_TX_DATA_BD_RESERVED2_SHIFT 9
__le16 reserved3;
};
/* The common non-special TX BD ring element */ /* The common non-special TX BD ring element */
struct eth_tx_bd { struct eth_tx_bd {
struct regpair addr; struct regpair addr;
__le16 nbytes; __le16 nbytes;
__le16 reserved0; struct eth_tx_data_bd data;
__le32 reserved1;
}; };
union eth_tx_bd_types { union eth_tx_bd_types {
......
...@@ -80,7 +80,7 @@ struct qed_dev_info { ...@@ -80,7 +80,7 @@ struct qed_dev_info {
u8 num_hwfns; u8 num_hwfns;
u8 hw_mac[ETH_ALEN]; u8 hw_mac[ETH_ALEN];
bool is_mf; bool is_mf_default;
/* FW version */ /* FW version */
u16 fw_major; u16 fw_major;
...@@ -360,6 +360,12 @@ enum DP_MODULE { ...@@ -360,6 +360,12 @@ enum DP_MODULE {
/* to be added...up to 0x8000000 */ /* to be added...up to 0x8000000 */
}; };
enum qed_mf_mode {
QED_MF_DEFAULT,
QED_MF_OVLAN,
QED_MF_NPAR,
};
struct qed_eth_stats { struct qed_eth_stats {
u64 no_buff_discards; u64 no_buff_discards;
u64 packet_too_big_discard; u64 packet_too_big_discard;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment