f-stack/dpdk/drivers/common/cnxk/roc_nix_ops.c

502 lines
12 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(C) 2021 Marvell.
*/
#include "roc_api.h"
#include "roc_priv.h"
static inline struct mbox *
get_mbox(struct roc_nix *roc_nix)
{
struct nix *nix = roc_nix_to_nix_priv(roc_nix);
struct dev *dev = &nix->dev;
return dev->mbox;
}
static void
nix_lso_tcp(struct nix_lso_format_cfg *req, bool v4)
{
__io struct nix_lso_format *field;
/* Format works only with TCP packet marked by OL3/OL4 */
field = (__io struct nix_lso_format *)&req->fields[0];
req->field_mask = NIX_LSO_FIELD_MASK;
/* Outer IPv4/IPv6 */
field->layer = NIX_TXLAYER_OL3;
field->offset = v4 ? 2 : 4;
field->sizem1 = 1; /* 2B */
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
if (v4) {
/* IPID field */
field->layer = NIX_TXLAYER_OL3;
field->offset = 4;
field->sizem1 = 1;
/* Incremented linearly per segment */
field->alg = NIX_LSOALG_ADD_SEGNUM;
field++;
}
/* TCP sequence number update */
field->layer = NIX_TXLAYER_OL4;
field->offset = 4;
field->sizem1 = 3; /* 4 bytes */
field->alg = NIX_LSOALG_ADD_OFFSET;
field++;
/* TCP flags field */
field->layer = NIX_TXLAYER_OL4;
field->offset = 12;
field->sizem1 = 1;
field->alg = NIX_LSOALG_TCP_FLAGS;
field++;
}
static void
nix_lso_udp_tun_tcp(struct nix_lso_format_cfg *req, bool outer_v4,
bool inner_v4)
{
__io struct nix_lso_format *field;
field = (__io struct nix_lso_format *)&req->fields[0];
req->field_mask = NIX_LSO_FIELD_MASK;
/* Outer IPv4/IPv6 len */
field->layer = NIX_TXLAYER_OL3;
field->offset = outer_v4 ? 2 : 4;
field->sizem1 = 1; /* 2B */
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
if (outer_v4) {
/* IPID */
field->layer = NIX_TXLAYER_OL3;
field->offset = 4;
field->sizem1 = 1;
/* Incremented linearly per segment */
field->alg = NIX_LSOALG_ADD_SEGNUM;
field++;
}
/* Outer UDP length */
field->layer = NIX_TXLAYER_OL4;
field->offset = 4;
field->sizem1 = 1;
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
/* Inner IPv4/IPv6 */
field->layer = NIX_TXLAYER_IL3;
field->offset = inner_v4 ? 2 : 4;
field->sizem1 = 1; /* 2B */
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
if (inner_v4) {
/* IPID field */
field->layer = NIX_TXLAYER_IL3;
field->offset = 4;
field->sizem1 = 1;
/* Incremented linearly per segment */
field->alg = NIX_LSOALG_ADD_SEGNUM;
field++;
}
/* TCP sequence number update */
field->layer = NIX_TXLAYER_IL4;
field->offset = 4;
field->sizem1 = 3; /* 4 bytes */
field->alg = NIX_LSOALG_ADD_OFFSET;
field++;
/* TCP flags field */
field->layer = NIX_TXLAYER_IL4;
field->offset = 12;
field->sizem1 = 1;
field->alg = NIX_LSOALG_TCP_FLAGS;
field++;
}
static void
nix_lso_tun_tcp(struct nix_lso_format_cfg *req, bool outer_v4, bool inner_v4)
{
__io struct nix_lso_format *field;
field = (__io struct nix_lso_format *)&req->fields[0];
req->field_mask = NIX_LSO_FIELD_MASK;
/* Outer IPv4/IPv6 len */
field->layer = NIX_TXLAYER_OL3;
field->offset = outer_v4 ? 2 : 4;
field->sizem1 = 1; /* 2B */
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
if (outer_v4) {
/* IPID */
field->layer = NIX_TXLAYER_OL3;
field->offset = 4;
field->sizem1 = 1;
/* Incremented linearly per segment */
field->alg = NIX_LSOALG_ADD_SEGNUM;
field++;
}
/* Inner IPv4/IPv6 */
field->layer = NIX_TXLAYER_IL3;
field->offset = inner_v4 ? 2 : 4;
field->sizem1 = 1; /* 2B */
field->alg = NIX_LSOALG_ADD_PAYLEN;
field++;
if (inner_v4) {
/* IPID field */
field->layer = NIX_TXLAYER_IL3;
field->offset = 4;
field->sizem1 = 1;
/* Incremented linearly per segment */
field->alg = NIX_LSOALG_ADD_SEGNUM;
field++;
}
/* TCP sequence number update */
field->layer = NIX_TXLAYER_IL4;
field->offset = 4;
field->sizem1 = 3; /* 4 bytes */
field->alg = NIX_LSOALG_ADD_OFFSET;
field++;
/* TCP flags field */
field->layer = NIX_TXLAYER_IL4;
field->offset = 12;
field->sizem1 = 1;
field->alg = NIX_LSOALG_TCP_FLAGS;
field++;
}
int
roc_nix_lso_custom_fmt_setup(struct roc_nix *roc_nix,
struct nix_lso_format *fields, uint16_t nb_fields)
{
struct mbox *mbox = get_mbox(roc_nix);
struct nix_lso_format_cfg_rsp *rsp;
struct nix_lso_format_cfg *req;
int rc = -ENOSPC;
if (nb_fields > NIX_LSO_FIELD_MAX)
return -EINVAL;
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return rc;
req->field_mask = NIX_LSO_FIELD_MASK;
mbox_memcpy(req->fields, fields,
sizeof(struct nix_lso_format) * nb_fields);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
plt_nix_dbg("Setup custom format %u", rsp->lso_format_idx);
return rsp->lso_format_idx;
}
int
roc_nix_lso_fmt_setup(struct roc_nix *roc_nix)
{
struct nix *nix = roc_nix_to_nix_priv(roc_nix);
struct mbox *mbox = get_mbox(roc_nix);
struct nix_lso_format_cfg_rsp *rsp;
struct nix_lso_format_cfg *req;
int rc = -ENOSPC;
/*
* IPv4/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return rc;
nix_lso_tcp(req, true);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
if (rsp->lso_format_idx != NIX_LSO_FORMAT_IDX_TSOV4)
return NIX_ERR_INTERNAL;
plt_nix_dbg("tcpv4 lso fmt=%u\n", rsp->lso_format_idx);
/*
* IPv6/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_tcp(req, false);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
if (rsp->lso_format_idx != NIX_LSO_FORMAT_IDX_TSOV6)
return NIX_ERR_INTERNAL;
plt_nix_dbg("tcpv6 lso fmt=%u\n", rsp->lso_format_idx);
/*
* IPv4/UDP/TUN HDR/IPv4/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_udp_tun_tcp(req, true, true);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_udp_tun_idx[ROC_NIX_LSO_TUN_V4V4] = rsp->lso_format_idx;
plt_nix_dbg("udp tun v4v4 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv4/UDP/TUN HDR/IPv6/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_udp_tun_tcp(req, true, false);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_udp_tun_idx[ROC_NIX_LSO_TUN_V4V6] = rsp->lso_format_idx;
plt_nix_dbg("udp tun v4v6 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv6/UDP/TUN HDR/IPv4/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_udp_tun_tcp(req, false, true);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_udp_tun_idx[ROC_NIX_LSO_TUN_V6V4] = rsp->lso_format_idx;
plt_nix_dbg("udp tun v6v4 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv6/UDP/TUN HDR/IPv6/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_udp_tun_tcp(req, false, false);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_udp_tun_idx[ROC_NIX_LSO_TUN_V6V6] = rsp->lso_format_idx;
plt_nix_dbg("udp tun v6v6 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv4/TUN HDR/IPv4/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_tun_tcp(req, true, true);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_tun_idx[ROC_NIX_LSO_TUN_V4V4] = rsp->lso_format_idx;
plt_nix_dbg("tun v4v4 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv4/TUN HDR/IPv6/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_tun_tcp(req, true, false);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_tun_idx[ROC_NIX_LSO_TUN_V4V6] = rsp->lso_format_idx;
plt_nix_dbg("tun v4v6 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv6/TUN HDR/IPv4/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_tun_tcp(req, false, true);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_tun_idx[ROC_NIX_LSO_TUN_V6V4] = rsp->lso_format_idx;
plt_nix_dbg("tun v6v4 fmt=%u\n", rsp->lso_format_idx);
/*
* IPv6/TUN HDR/IPv6/TCP LSO
*/
req = mbox_alloc_msg_nix_lso_format_cfg(mbox);
if (req == NULL)
return -ENOSPC;
nix_lso_tun_tcp(req, false, false);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
nix->lso_tun_idx[ROC_NIX_LSO_TUN_V6V6] = rsp->lso_format_idx;
plt_nix_dbg("tun v6v6 fmt=%u\n", rsp->lso_format_idx);
return 0;
}
int
roc_nix_lso_fmt_get(struct roc_nix *roc_nix,
uint8_t udp_tun[ROC_NIX_LSO_TUN_MAX],
uint8_t tun[ROC_NIX_LSO_TUN_MAX])
{
struct nix *nix = roc_nix_to_nix_priv(roc_nix);
memcpy(udp_tun, nix->lso_udp_tun_idx, ROC_NIX_LSO_TUN_MAX);
memcpy(tun, nix->lso_tun_idx, ROC_NIX_LSO_TUN_MAX);
return 0;
}
int
roc_nix_switch_hdr_set(struct roc_nix *roc_nix, uint64_t switch_header_type,
uint8_t pre_l2_size_offset,
uint8_t pre_l2_size_offset_mask,
uint8_t pre_l2_size_shift_dir)
{
struct mbox *mbox = get_mbox(roc_nix);
struct npc_set_pkind *req;
struct msg_resp *rsp;
int rc = -ENOSPC;
if (switch_header_type == 0)
switch_header_type = ROC_PRIV_FLAGS_DEFAULT;
if (switch_header_type != ROC_PRIV_FLAGS_DEFAULT &&
switch_header_type != ROC_PRIV_FLAGS_EDSA &&
switch_header_type != ROC_PRIV_FLAGS_HIGIG &&
switch_header_type != ROC_PRIV_FLAGS_LEN_90B &&
switch_header_type != ROC_PRIV_FLAGS_EXDSA &&
switch_header_type != ROC_PRIV_FLAGS_VLAN_EXDSA &&
switch_header_type != ROC_PRIV_FLAGS_PRE_L2 &&
switch_header_type != ROC_PRIV_FLAGS_CUSTOM) {
plt_err("switch header type is not supported");
return NIX_ERR_PARAM;
}
if (switch_header_type == ROC_PRIV_FLAGS_LEN_90B &&
!roc_nix_is_sdp(roc_nix)) {
plt_err("chlen90b is not supported on non-SDP device");
return NIX_ERR_PARAM;
}
if (switch_header_type == ROC_PRIV_FLAGS_HIGIG &&
roc_nix_is_vf_or_sdp(roc_nix)) {
plt_err("higig2 is supported on PF devices only");
return NIX_ERR_PARAM;
}
req = mbox_alloc_msg_npc_set_pkind(mbox);
if (req == NULL)
return rc;
req->mode = switch_header_type;
if (switch_header_type == ROC_PRIV_FLAGS_LEN_90B) {
req->mode = ROC_PRIV_FLAGS_CUSTOM;
req->pkind = NPC_RX_CHLEN90B_PKIND;
} else if (switch_header_type == ROC_PRIV_FLAGS_EXDSA) {
req->mode = ROC_PRIV_FLAGS_CUSTOM;
req->pkind = NPC_RX_EXDSA_PKIND;
} else if (switch_header_type == ROC_PRIV_FLAGS_VLAN_EXDSA) {
req->mode = ROC_PRIV_FLAGS_CUSTOM;
req->pkind = NPC_RX_VLAN_EXDSA_PKIND;
} else if (switch_header_type == ROC_PRIV_FLAGS_PRE_L2) {
req->mode = ROC_PRIV_FLAGS_CUSTOM;
req->pkind = NPC_RX_CUSTOM_PRE_L2_PKIND;
req->var_len_off = pre_l2_size_offset;
req->var_len_off_mask = pre_l2_size_offset_mask;
req->shift_dir = pre_l2_size_shift_dir;
}
req->dir = PKIND_RX;
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return rc;
req = mbox_alloc_msg_npc_set_pkind(mbox);
if (req == NULL)
return -ENOSPC;
req->mode = switch_header_type;
req->dir = PKIND_TX;
return mbox_process_msg(mbox, (void *)&rsp);
}
int
roc_nix_eeprom_info_get(struct roc_nix *roc_nix,
struct roc_nix_eeprom_info *info)
{
struct mbox *mbox = get_mbox(roc_nix);
struct cgx_fw_data *rsp = NULL;
int rc;
if (!info) {
plt_err("Input buffer is NULL");
return NIX_ERR_PARAM;
}
mbox_alloc_msg_cgx_get_aux_link_info(mbox);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc) {
plt_err("Failed to get fw data: %d", rc);
return rc;
}
info->sff_id = rsp->fwdata.sfp_eeprom.sff_id;
mbox_memcpy(info->buf, rsp->fwdata.sfp_eeprom.buf, SFP_EEPROM_SIZE);
return 0;
}
int
roc_nix_rx_drop_re_set(struct roc_nix *roc_nix, bool ena)
{
struct nix *nix = roc_nix_to_nix_priv(roc_nix);
struct mbox *mbox = get_mbox(roc_nix);
struct nix_rx_cfg *req;
int rc = -EIO;
/* No-op if no change */
if (ena == !!(nix->rx_cfg & ROC_NIX_LF_RX_CFG_DROP_RE))
return 0;
req = mbox_alloc_msg_nix_set_rx_cfg(mbox);
if (req == NULL)
return rc;
if (ena)
req->len_verify |= NIX_RX_DROP_RE;
/* Keep other flags intact */
if (nix->rx_cfg & ROC_NIX_LF_RX_CFG_LEN_OL3)
req->len_verify |= NIX_RX_OL3_VERIFY;
if (nix->rx_cfg & ROC_NIX_LF_RX_CFG_LEN_OL4)
req->len_verify |= NIX_RX_OL4_VERIFY;
if (nix->rx_cfg & ROC_NIX_LF_RX_CFG_CSUM_OL4)
req->csum_verify |= NIX_RX_CSUM_OL4_VERIFY;
rc = mbox_process(mbox);
if (rc)
return rc;
if (ena)
nix->rx_cfg |= ROC_NIX_LF_RX_CFG_DROP_RE;
else
nix->rx_cfg &= ~ROC_NIX_LF_RX_CFG_DROP_RE;
return 0;
}