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

1170 lines
28 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(C) 2021 Marvell.
*/
#include "roc_api.h"
#include "roc_priv.h"
const struct roc_npc_item_info *
npc_parse_skip_void_and_any_items(const struct roc_npc_item_info *pattern)
{
while ((pattern->type == ROC_NPC_ITEM_TYPE_VOID) ||
(pattern->type == ROC_NPC_ITEM_TYPE_ANY))
pattern++;
return pattern;
}
int
npc_parse_meta_items(struct npc_parse_state *pst)
{
PLT_SET_USED(pst);
return 0;
}
int
npc_parse_mark_item(struct npc_parse_state *pst)
{
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_MARK) {
if (pst->flow->nix_intf != NIX_INTF_RX)
return -EINVAL;
pst->is_second_pass_rule = true;
pst->pattern++;
}
return 0;
}
static int
npc_flow_raw_item_prepare(const struct roc_npc_flow_item_raw *raw_spec,
const struct roc_npc_flow_item_raw *raw_mask,
struct npc_parse_item_info *info, uint8_t *spec_buf,
uint8_t *mask_buf)
{
memset(spec_buf, 0, NPC_MAX_RAW_ITEM_LEN);
memset(mask_buf, 0, NPC_MAX_RAW_ITEM_LEN);
memcpy(spec_buf + raw_spec->offset, raw_spec->pattern,
raw_spec->length);
if (raw_mask && raw_mask->pattern) {
memcpy(mask_buf + raw_spec->offset, raw_mask->pattern,
raw_spec->length);
} else {
memset(mask_buf + raw_spec->offset, 0xFF, raw_spec->length);
}
info->len = NPC_MAX_RAW_ITEM_LEN;
info->spec = spec_buf;
info->mask = mask_buf;
return 0;
}
int
npc_parse_pre_l2(struct npc_parse_state *pst)
{
uint8_t raw_spec_buf[NPC_MAX_RAW_ITEM_LEN] = {0};
uint8_t raw_mask_buf[NPC_MAX_RAW_ITEM_LEN] = {0};
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN] = {0};
const struct roc_npc_flow_item_raw *raw_spec;
struct npc_parse_item_info info;
int lid, lt, len;
int rc;
if (pst->npc->switch_header_type != ROC_PRIV_FLAGS_PRE_L2)
return 0;
/* Identify the pattern type into lid, lt */
if (pst->pattern->type != ROC_NPC_ITEM_TYPE_RAW)
return 0;
lid = NPC_LID_LA;
lt = NPC_LT_LA_CUSTOM_PRE_L2_ETHER;
info.hw_hdr_len = 0;
raw_spec = pst->pattern->spec;
len = raw_spec->length + raw_spec->offset;
if (len > NPC_MAX_RAW_ITEM_LEN)
return -EINVAL;
if (raw_spec->relative == 0 || raw_spec->search || raw_spec->limit ||
raw_spec->offset < 0)
return -EINVAL;
npc_flow_raw_item_prepare(
(const struct roc_npc_flow_item_raw *)pst->pattern->spec,
(const struct roc_npc_flow_item_raw *)pst->pattern->mask, &info,
raw_spec_buf, raw_mask_buf);
info.def_mask = NULL;
info.hw_mask = &hw_mask;
npc_get_hw_supp_mask(pst, &info, lid, lt);
/* Basic validation of item parameters */
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc)
return rc;
/* Update pst if not validate only? clash check? */
return npc_update_parse_state(pst, &info, lid, lt, 0);
}
int
npc_parse_cpt_hdr(struct npc_parse_state *pst)
{
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt;
int rc;
/* Identify the pattern type into lid, lt */
if (pst->pattern->type != ROC_NPC_ITEM_TYPE_CPT_HDR)
return 0;
lid = NPC_LID_LA;
lt = NPC_LT_LA_CPT_HDR;
info.hw_hdr_len = 0;
/* Prepare for parsing the item */
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.len = pst->pattern->size;
npc_get_hw_supp_mask(pst, &info, lid, lt);
info.spec = NULL;
info.mask = NULL;
/* Basic validation of item parameters */
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc)
return rc;
/* Update pst if not validate only? clash check? */
return npc_update_parse_state(pst, &info, lid, lt, 0);
}
int
npc_parse_higig2_hdr(struct npc_parse_state *pst)
{
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt;
int rc;
/* Identify the pattern type into lid, lt */
if (pst->pattern->type != ROC_NPC_ITEM_TYPE_HIGIG2)
return 0;
lid = NPC_LID_LA;
lt = NPC_LT_LA_HIGIG2_ETHER;
info.hw_hdr_len = 0;
if (pst->flow->nix_intf == NIX_INTF_TX) {
lt = NPC_LT_LA_IH_NIX_HIGIG2_ETHER;
info.hw_hdr_len = NPC_IH_LENGTH;
}
/* Prepare for parsing the item */
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.len = pst->pattern->size;
npc_get_hw_supp_mask(pst, &info, lid, lt);
info.spec = NULL;
info.mask = NULL;
/* Basic validation of item parameters */
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc)
return rc;
/* Update pst if not validate only? clash check? */
return npc_update_parse_state(pst, &info, lid, lt, 0);
}
int
npc_parse_la(struct npc_parse_state *pst)
{
const struct roc_npc_flow_item_eth *eth_item;
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt;
int rc;
/* Identify the pattern type into lid, lt */
if (pst->pattern->type != ROC_NPC_ITEM_TYPE_ETH)
return 0;
pst->has_eth_type = true;
eth_item = pst->pattern->spec;
lid = NPC_LID_LA;
lt = NPC_LT_LA_ETHER;
info.hw_hdr_len = 0;
if (pst->flow->nix_intf == NIX_INTF_TX) {
lt = NPC_LT_LA_IH_NIX_ETHER;
info.hw_hdr_len = NPC_IH_LENGTH;
if (pst->npc->switch_header_type == ROC_PRIV_FLAGS_HIGIG) {
lt = NPC_LT_LA_IH_NIX_HIGIG2_ETHER;
info.hw_hdr_len += NPC_HIGIG2_LENGTH;
}
} else {
if (pst->npc->switch_header_type == ROC_PRIV_FLAGS_HIGIG) {
lt = NPC_LT_LA_HIGIG2_ETHER;
info.hw_hdr_len = NPC_HIGIG2_LENGTH;
}
}
/* Prepare for parsing the item */
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.len = sizeof(eth_item->hdr);
npc_get_hw_supp_mask(pst, &info, lid, lt);
info.spec = NULL;
info.mask = NULL;
/* Basic validation of item parameters */
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc)
return rc;
rc = npc_update_parse_state(pst, &info, lid, lt, 0);
if (rc)
return rc;
if (eth_item && eth_item->has_vlan)
pst->set_vlan_ltype_mask = true;
return 0;
}
#define NPC_MAX_SUPPORTED_VLANS 3
static int
npc_parse_vlan_count(const struct roc_npc_item_info *pattern,
const struct roc_npc_item_info **pattern_list,
const struct roc_npc_flow_item_vlan **vlan_items, int *vlan_count)
{
*vlan_count = 0;
while (pattern->type == ROC_NPC_ITEM_TYPE_VLAN) {
if (*vlan_count > NPC_MAX_SUPPORTED_VLANS - 1)
return NPC_ERR_PATTERN_NOTSUP;
/* Don't support ranges */
if (pattern->last != NULL)
return NPC_ERR_INVALID_RANGE;
/* If spec is NULL, both mask and last must be NULL, this
* makes it to match ANY value (eq to mask = 0).
* Setting either mask or last without spec is an error
*/
if (pattern->spec == NULL) {
if (pattern->last != NULL && pattern->mask != NULL)
return NPC_ERR_INVALID_SPEC;
}
pattern_list[*vlan_count] = pattern;
vlan_items[*vlan_count] = pattern->spec;
(*vlan_count)++;
pattern++;
pattern = npc_parse_skip_void_and_any_items(pattern);
}
return 0;
}
static int
npc_parse_vlan_ltype_get(struct npc_parse_state *pst,
const struct roc_npc_flow_item_vlan **vlan_item, int vlan_count,
int *ltype, int *lflags)
{
switch (vlan_count) {
case 1:
*ltype = NPC_LT_LB_CTAG;
if (vlan_item[0] && vlan_item[0]->has_more_vlan)
*ltype = NPC_LT_LB_STAG_QINQ;
break;
case 2:
if (vlan_item[1] && vlan_item[1]->has_more_vlan) {
if (!(pst->npc->keyx_supp_nmask[pst->nix_intf] &
0x3ULL << NPC_LFLAG_LB_OFFSET))
return NPC_ERR_PATTERN_NOTSUP;
/* This lflag value will match either one of
* NPC_F_LB_L_WITH_STAG_STAG,
* NPC_F_LB_L_WITH_QINQ_CTAG,
* NPC_F_LB_L_WITH_QINQ_QINQ and
* NPC_F_LB_L_WITH_ITAG (0b0100 to 0b0111). For
* NPC_F_LB_L_WITH_ITAG, ltype is NPC_LT_LB_ETAG
* hence will not match.
*/
*lflags = NPC_F_LB_L_WITH_QINQ_CTAG & NPC_F_LB_L_WITH_QINQ_QINQ &
NPC_F_LB_L_WITH_STAG_STAG;
}
*ltype = NPC_LT_LB_STAG_QINQ;
break;
case 3:
if (vlan_item[2] && vlan_item[2]->has_more_vlan)
return NPC_ERR_PATTERN_NOTSUP;
if (!(pst->npc->keyx_supp_nmask[pst->nix_intf] & 0x3ULL << NPC_LFLAG_LB_OFFSET))
return NPC_ERR_PATTERN_NOTSUP;
*ltype = NPC_LT_LB_STAG_QINQ;
*lflags = NPC_F_STAG_STAG_CTAG;
break;
default:
return NPC_ERR_PATTERN_NOTSUP;
}
return 0;
}
static int
npc_update_vlan_parse_state(struct npc_parse_state *pst, const struct roc_npc_item_info *pattern,
int lid, int lt, uint8_t lflags, int vlan_count)
{
uint8_t vlan_spec[NPC_MAX_SUPPORTED_VLANS * sizeof(struct roc_vlan_hdr)];
uint8_t vlan_mask[NPC_MAX_SUPPORTED_VLANS * sizeof(struct roc_vlan_hdr)];
int rc = 0, i, offset = NPC_TPID_LENGTH;
struct npc_parse_item_info parse_info;
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
memset(vlan_spec, 0, sizeof(struct roc_vlan_hdr) * NPC_MAX_SUPPORTED_VLANS);
memset(vlan_mask, 0, sizeof(struct roc_vlan_hdr) * NPC_MAX_SUPPORTED_VLANS);
memset(&parse_info, 0, sizeof(parse_info));
if (vlan_count > 2)
vlan_count = 2;
for (i = 0; i < vlan_count; i++) {
if (pattern[i].spec)
memcpy(vlan_spec + offset, pattern[i].spec, sizeof(struct roc_vlan_hdr));
if (pattern[i].mask)
memcpy(vlan_mask + offset, pattern[i].mask, sizeof(struct roc_vlan_hdr));
offset += 4;
}
parse_info.def_mask = NULL;
parse_info.spec = vlan_spec;
parse_info.mask = vlan_mask;
parse_info.def_mask = NULL;
parse_info.hw_hdr_len = 0;
lid = NPC_LID_LB;
parse_info.hw_mask = hw_mask;
if (lt == NPC_LT_LB_CTAG)
parse_info.len = sizeof(struct roc_vlan_hdr) + NPC_TPID_LENGTH;
if (lt == NPC_LT_LB_STAG_QINQ)
parse_info.len = sizeof(struct roc_vlan_hdr) * 2 + NPC_TPID_LENGTH;
memset(hw_mask, 0, sizeof(hw_mask));
parse_info.hw_mask = &hw_mask;
npc_get_hw_supp_mask(pst, &parse_info, lid, lt);
rc = npc_mask_is_supported(parse_info.mask, parse_info.hw_mask, parse_info.len);
if (!rc)
return NPC_ERR_INVALID_MASK;
/* Point pattern to last item consumed */
pst->pattern = pattern;
return npc_update_parse_state(pst, &parse_info, lid, lt, lflags);
}
static int
npc_parse_lb_vlan(struct npc_parse_state *pst)
{
const struct roc_npc_flow_item_vlan *vlan_items[NPC_MAX_SUPPORTED_VLANS];
const struct roc_npc_item_info *pattern_list[NPC_MAX_SUPPORTED_VLANS];
const struct roc_npc_item_info *last_pattern;
int vlan_count = 0, rc = 0;
int lid, lt, lflags;
lid = NPC_LID_LB;
lflags = 0;
last_pattern = pst->pattern;
rc = npc_parse_vlan_count(pst->pattern, pattern_list, vlan_items, &vlan_count);
if (rc)
return rc;
rc = npc_parse_vlan_ltype_get(pst, vlan_items, vlan_count, &lt, &lflags);
if (rc)
return rc;
if (vlan_count == 3) {
if (pattern_list[2]->spec != NULL && pattern_list[2]->mask != NULL &&
pattern_list[2]->last != NULL)
return NPC_ERR_PATTERN_NOTSUP;
/* Matching can be done only for two tags. */
vlan_count = 2;
last_pattern++;
}
rc = npc_update_vlan_parse_state(pst, pattern_list[0], lid, lt, lflags, vlan_count);
if (rc)
return rc;
if (vlan_count > 1)
pst->pattern = last_pattern + vlan_count;
return 0;
}
int
npc_parse_lb(struct npc_parse_state *pst)
{
const struct roc_npc_item_info *pattern = pst->pattern;
const struct roc_npc_item_info *last_pattern;
const struct roc_npc_flow_item_raw *raw_spec;
uint8_t raw_spec_buf[NPC_MAX_RAW_ITEM_LEN];
uint8_t raw_mask_buf[NPC_MAX_RAW_ITEM_LEN];
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt, lflags, len = 0;
int rc;
info.def_mask = NULL;
info.spec = NULL;
info.mask = NULL;
info.def_mask = NULL;
info.hw_hdr_len = NPC_TPID_LENGTH;
lid = NPC_LID_LB;
lflags = 0;
last_pattern = pattern;
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_VLAN) {
/* RTE vlan is either 802.1q or 802.1ad,
* this maps to either CTAG/STAG. We need to decide
* based on number of VLANS present. Matching is
* supported on first two tags.
*/
return npc_parse_lb_vlan(pst);
} else if (pst->pattern->type == ROC_NPC_ITEM_TYPE_E_TAG) {
/* we can support ETAG and match a subsequent CTAG
* without any matching support.
*/
lt = NPC_LT_LB_ETAG;
lflags = 0;
last_pattern = pst->pattern;
pattern = npc_parse_skip_void_and_any_items(pst->pattern + 1);
if (pattern->type == ROC_NPC_ITEM_TYPE_VLAN) {
/* set supported mask to NULL for vlan tag */
info.hw_mask = NULL;
info.len = pattern->size;
rc = npc_parse_item_basic(pattern, &info);
if (rc != 0)
return rc;
lflags = NPC_F_ETAG_CTAG;
last_pattern = pattern;
}
info.len = pattern->size;
} else if (pst->pattern->type == ROC_NPC_ITEM_TYPE_QINQ) {
info.hw_mask = NULL;
info.len = pattern->size;
lt = NPC_LT_LB_STAG_QINQ;
lflags = NPC_F_STAG_CTAG;
} else if (pst->pattern->type == ROC_NPC_ITEM_TYPE_RAW) {
raw_spec = pst->pattern->spec;
if (raw_spec->relative)
return 0;
len = raw_spec->length + raw_spec->offset;
if (len > NPC_MAX_RAW_ITEM_LEN)
return -EINVAL;
if (pst->npc->switch_header_type == ROC_PRIV_FLAGS_VLAN_EXDSA) {
lt = NPC_LT_LB_VLAN_EXDSA;
} else if (pst->npc->switch_header_type ==
ROC_PRIV_FLAGS_EXDSA) {
lt = NPC_LT_LB_EXDSA;
} else {
return -EINVAL;
}
npc_flow_raw_item_prepare((const struct roc_npc_flow_item_raw *)
pst->pattern->spec,
(const struct roc_npc_flow_item_raw *)
pst->pattern->mask,
&info, raw_spec_buf, raw_mask_buf);
info.hw_hdr_len = 0;
} else {
return 0;
}
info.hw_mask = &hw_mask;
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
/* Point pattern to last item consumed */
pst->pattern = last_pattern;
return npc_update_parse_state(pst, &info, lid, lt, lflags);
}
static int
npc_parse_mpls_label_stack(struct npc_parse_state *pst, int *flag)
{
uint8_t flag_list[] = {0, NPC_F_MPLS_2_LABELS, NPC_F_MPLS_3_LABELS,
NPC_F_MPLS_4_LABELS};
const struct roc_npc_item_info *pattern = pst->pattern;
struct npc_parse_item_info info;
int nr_labels = 0;
int rc;
/*
* pst->pattern points to first MPLS label. We only check
* that subsequent labels do not have anything to match.
*/
info.def_mask = NULL;
info.hw_mask = NULL;
info.len = pattern->size;
info.spec = NULL;
info.mask = NULL;
info.hw_hdr_len = 0;
while (pattern->type == ROC_NPC_ITEM_TYPE_MPLS) {
nr_labels++;
/* Basic validation of Second/Third/Fourth mpls item */
if (nr_labels > 1) {
rc = npc_parse_item_basic(pattern, &info);
if (rc != 0)
return rc;
}
pst->last_pattern = pattern;
pattern++;
pattern = npc_parse_skip_void_and_any_items(pattern);
}
if (nr_labels < 1 || nr_labels > 4)
return NPC_ERR_PATTERN_NOTSUP;
*flag = flag_list[nr_labels - 1];
return 0;
}
static int
npc_parse_mpls(struct npc_parse_state *pst, int lid)
{
/* Find number of MPLS labels */
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lt, lflags;
int rc;
lflags = 0;
if (lid == NPC_LID_LC)
lt = NPC_LT_LC_MPLS;
else if (lid == NPC_LID_LD)
lt = NPC_LT_LD_TU_MPLS_IN_IP;
else
lt = NPC_LT_LE_TU_MPLS_IN_UDP;
/* Prepare for parsing the first item */
info.hw_mask = &hw_mask;
info.len = pst->pattern->size;
info.spec = NULL;
info.mask = NULL;
info.def_mask = NULL;
info.hw_hdr_len = 0;
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
/*
* Parse for more labels.
* This sets lflags and pst->last_pattern correctly.
*/
rc = npc_parse_mpls_label_stack(pst, &lflags);
if (rc != 0)
return rc;
pst->tunnel = 1;
pst->pattern = pst->last_pattern;
return npc_update_parse_state(pst, &info, lid, lt, lflags);
}
static inline void
npc_check_lc_ip_tunnel(struct npc_parse_state *pst)
{
const struct roc_npc_item_info *pattern = pst->pattern + 1;
pattern = npc_parse_skip_void_and_any_items(pattern);
if (pattern->type == ROC_NPC_ITEM_TYPE_MPLS ||
pattern->type == ROC_NPC_ITEM_TYPE_IPV4 ||
pattern->type == ROC_NPC_ITEM_TYPE_IPV6)
pst->tunnel = 1;
}
static int
npc_handle_ipv6ext_attr(const struct roc_npc_flow_item_ipv6 *ipv6_spec,
struct npc_parse_state *pst, uint8_t *flags)
{
int flags_count = 0;
if (ipv6_spec->has_hop_ext) {
*flags = NPC_F_LC_L_EXT_HOP;
flags_count++;
}
if (ipv6_spec->has_route_ext) {
*flags = NPC_F_LC_L_EXT_ROUT;
flags_count++;
}
if (ipv6_spec->has_frag_ext) {
*flags = NPC_F_LC_U_IP6_FRAG;
flags_count++;
}
if (ipv6_spec->has_dest_ext) {
*flags = NPC_F_LC_L_EXT_DEST;
flags_count++;
}
if (ipv6_spec->has_mobil_ext) {
*flags = NPC_F_LC_L_EXT_MOBILITY;
flags_count++;
}
if (ipv6_spec->has_hip_ext) {
*flags = NPC_F_LC_L_EXT_HOSTID;
flags_count++;
}
if (ipv6_spec->has_shim6_ext) {
*flags = NPC_F_LC_L_EXT_SHIM6;
flags_count++;
}
if (ipv6_spec->has_auth_ext) {
pst->lt[NPC_LID_LD] = NPC_LT_LD_AH;
flags_count++;
}
if (ipv6_spec->has_esp_ext) {
pst->lt[NPC_LID_LE] = NPC_LT_LE_ESP;
flags_count++;
}
if (flags_count > 1)
return -EINVAL;
if (flags_count)
pst->set_ipv6ext_ltype_mask = true;
return 0;
}
static int
npc_process_ipv6_item(struct npc_parse_state *pst)
{
uint8_t ipv6_hdr_mask[sizeof(struct roc_ipv6_hdr) + sizeof(struct roc_ipv6_fragment_ext)];
uint8_t ipv6_hdr_buf[sizeof(struct roc_ipv6_hdr) + sizeof(struct roc_ipv6_fragment_ext)];
const struct roc_npc_flow_item_ipv6 *ipv6_spec, *ipv6_mask;
const struct roc_npc_item_info *pattern = pst->pattern;
int offset = 0, rc = 0, lid, item_count = 0;
struct npc_parse_item_info parse_info;
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
uint8_t flags = 0, ltype;
memset(ipv6_hdr_buf, 0, sizeof(ipv6_hdr_buf));
memset(ipv6_hdr_mask, 0, sizeof(ipv6_hdr_mask));
ipv6_spec = pst->pattern->spec;
ipv6_mask = pst->pattern->mask;
parse_info.def_mask = NULL;
parse_info.spec = ipv6_hdr_buf;
parse_info.mask = ipv6_hdr_mask;
parse_info.def_mask = NULL;
parse_info.hw_hdr_len = 0;
parse_info.len = sizeof(ipv6_spec->hdr);
pst->set_ipv6ext_ltype_mask = true;
lid = NPC_LID_LC;
ltype = NPC_LT_LC_IP6;
if (pattern->type == ROC_NPC_ITEM_TYPE_IPV6) {
item_count++;
if (ipv6_spec) {
memcpy(ipv6_hdr_buf, &ipv6_spec->hdr, sizeof(struct roc_ipv6_hdr));
rc = npc_handle_ipv6ext_attr(ipv6_spec, pst, &flags);
if (rc)
return rc;
}
if (ipv6_mask)
memcpy(ipv6_hdr_mask, &ipv6_mask->hdr, sizeof(struct roc_ipv6_hdr));
}
offset = sizeof(struct roc_ipv6_hdr);
while (pattern->type != ROC_NPC_ITEM_TYPE_END) {
/* Don't support ranges */
if (pattern->last != NULL)
return NPC_ERR_INVALID_RANGE;
/* If spec is NULL, both mask and last must be NULL, this
* makes it to match ANY value (eq to mask = 0).
* Setting either mask or last without spec is
* an error
*/
if (pattern->spec == NULL) {
if (pattern->last != NULL && pattern->mask != NULL)
return NPC_ERR_INVALID_SPEC;
}
/* Either one ROC_NPC_ITEM_TYPE_IPV6_EXT or
* one ROC_NPC_ITEM_TYPE_IPV6_FRAG_EXT is supported
* following an ROC_NPC_ITEM_TYPE_IPV6 item.
*/
if (pattern->type == ROC_NPC_ITEM_TYPE_IPV6_EXT) {
item_count++;
ltype = NPC_LT_LC_IP6_EXT;
parse_info.len =
sizeof(struct roc_ipv6_hdr) + sizeof(struct roc_flow_item_ipv6_ext);
if (pattern->spec)
memcpy(ipv6_hdr_buf + offset, pattern->spec,
sizeof(struct roc_flow_item_ipv6_ext));
if (pattern->mask)
memcpy(ipv6_hdr_mask + offset, pattern->mask,
sizeof(struct roc_flow_item_ipv6_ext));
break;
} else if (pattern->type == ROC_NPC_ITEM_TYPE_IPV6_FRAG_EXT) {
item_count++;
ltype = NPC_LT_LC_IP6_EXT;
flags = NPC_F_LC_U_IP6_FRAG;
parse_info.len =
sizeof(struct roc_ipv6_hdr) + sizeof(struct roc_ipv6_fragment_ext);
if (pattern->spec)
memcpy(ipv6_hdr_buf + offset, pattern->spec,
sizeof(struct roc_ipv6_fragment_ext));
if (pattern->mask)
memcpy(ipv6_hdr_mask + offset, pattern->mask,
sizeof(struct roc_ipv6_fragment_ext));
break;
}
pattern++;
pattern = npc_parse_skip_void_and_any_items(pattern);
}
memset(hw_mask, 0, sizeof(hw_mask));
parse_info.hw_mask = &hw_mask;
npc_get_hw_supp_mask(pst, &parse_info, lid, ltype);
rc = npc_mask_is_supported(parse_info.mask, parse_info.hw_mask, parse_info.len);
if (!rc)
return NPC_ERR_INVALID_MASK;
rc = npc_update_parse_state(pst, &parse_info, lid, ltype, flags);
if (rc)
return rc;
/* npc_update_parse_state() increments pattern once.
* Check if additional increment is required.
*/
if (item_count == 2)
pst->pattern++;
return 0;
}
int
npc_parse_lc(struct npc_parse_state *pst)
{
const struct roc_npc_flow_item_raw *raw_spec;
uint8_t raw_spec_buf[NPC_MAX_RAW_ITEM_LEN];
uint8_t raw_mask_buf[NPC_MAX_RAW_ITEM_LEN];
uint8_t hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int rc, lid, lt, len = 0;
uint8_t flags = 0;
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_MPLS)
return npc_parse_mpls(pst, NPC_LID_LC);
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.spec = NULL;
info.mask = NULL;
info.hw_hdr_len = 0;
lid = NPC_LID_LC;
switch (pst->pattern->type) {
case ROC_NPC_ITEM_TYPE_IPV4:
lt = NPC_LT_LC_IP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_IPV6:
case ROC_NPC_ITEM_TYPE_IPV6_EXT:
case ROC_NPC_ITEM_TYPE_IPV6_FRAG_EXT:
return npc_process_ipv6_item(pst);
case ROC_NPC_ITEM_TYPE_ARP_ETH_IPV4:
lt = NPC_LT_LC_ARP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_L3_CUSTOM:
lt = NPC_LT_LC_CUSTOM0;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_RAW:
raw_spec = pst->pattern->spec;
if (!raw_spec->relative)
return 0;
len = raw_spec->length + raw_spec->offset;
if (len > NPC_MAX_RAW_ITEM_LEN)
return -EINVAL;
npc_flow_raw_item_prepare((const struct roc_npc_flow_item_raw *)
pst->pattern->spec,
(const struct roc_npc_flow_item_raw *)
pst->pattern->mask,
&info, raw_spec_buf, raw_mask_buf);
lid = NPC_LID_LC;
lt = NPC_LT_LC_NGIO;
info.hw_mask = &hw_mask;
npc_get_hw_supp_mask(pst, &info, lid, lt);
break;
default:
/* No match at this layer */
return 0;
}
/* Identify if IP tunnels MPLS or IPv4/v6 */
npc_check_lc_ip_tunnel(pst);
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
return npc_update_parse_state(pst, &info, lid, lt, flags);
}
int
npc_parse_ld(struct npc_parse_state *pst)
{
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt, lflags;
int rc;
if (pst->tunnel) {
/* We have already parsed MPLS or IPv4/v6 followed
* by MPLS or IPv4/v6. Subsequent TCP/UDP etc
* would be parsed as tunneled versions. Skip
* this layer, except for tunneled MPLS. If LC is
* MPLS, we have anyway skipped all stacked MPLS
* labels.
*/
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_MPLS)
return npc_parse_mpls(pst, NPC_LID_LD);
return 0;
}
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.spec = NULL;
info.mask = NULL;
info.len = 0;
info.hw_hdr_len = 0;
lid = NPC_LID_LD;
lflags = 0;
switch (pst->pattern->type) {
case ROC_NPC_ITEM_TYPE_ICMP:
if (pst->lt[NPC_LID_LC] == NPC_LT_LC_IP6)
lt = NPC_LT_LD_ICMP6;
else
lt = NPC_LT_LD_ICMP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_UDP:
lt = NPC_LT_LD_UDP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_IGMP:
lt = NPC_LT_LD_IGMP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_TCP:
lt = NPC_LT_LD_TCP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_SCTP:
lt = NPC_LT_LD_SCTP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_GRE:
lt = NPC_LT_LD_GRE;
info.len = pst->pattern->size;
pst->tunnel = 1;
break;
case ROC_NPC_ITEM_TYPE_GRE_KEY:
lt = NPC_LT_LD_GRE;
info.len = pst->pattern->size;
info.hw_hdr_len = 4;
pst->tunnel = 1;
break;
case ROC_NPC_ITEM_TYPE_NVGRE:
lt = NPC_LT_LD_NVGRE;
lflags = NPC_F_GRE_NVGRE;
info.len = pst->pattern->size;
/* Further IP/Ethernet are parsed as tunneled */
pst->tunnel = 1;
break;
default:
return 0;
}
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
return npc_update_parse_state(pst, &info, lid, lt, lflags);
}
int
npc_parse_le(struct npc_parse_state *pst)
{
const struct roc_npc_item_info *pattern = pst->pattern;
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt, lflags;
int rc;
if (pst->tunnel)
return 0;
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_MPLS)
return npc_parse_mpls(pst, NPC_LID_LE);
info.spec = NULL;
info.mask = NULL;
info.hw_mask = NULL;
info.def_mask = NULL;
info.len = 0;
info.hw_hdr_len = 0;
lid = NPC_LID_LE;
lflags = 0;
/* Ensure we are not matching anything in UDP */
rc = npc_parse_item_basic(pattern, &info);
if (rc)
return rc;
info.hw_mask = &hw_mask;
pattern = npc_parse_skip_void_and_any_items(pattern);
switch (pattern->type) {
case ROC_NPC_ITEM_TYPE_VXLAN:
lflags = NPC_F_UDP_VXLAN;
info.len = pattern->size;
lt = NPC_LT_LE_VXLAN;
break;
case ROC_NPC_ITEM_TYPE_GTPC:
lflags = NPC_F_UDP_GTP_GTPC;
info.len = pattern->size;
lt = NPC_LT_LE_GTPC;
break;
case ROC_NPC_ITEM_TYPE_GTPU:
lflags = NPC_F_UDP_GTP_GTPU_G_PDU;
info.len = pattern->size;
lt = NPC_LT_LE_GTPU;
break;
case ROC_NPC_ITEM_TYPE_GENEVE:
lflags = NPC_F_UDP_GENEVE;
info.len = pattern->size;
lt = NPC_LT_LE_GENEVE;
break;
case ROC_NPC_ITEM_TYPE_VXLAN_GPE:
lflags = NPC_F_UDP_VXLANGPE;
info.len = pattern->size;
lt = NPC_LT_LE_VXLANGPE;
break;
case ROC_NPC_ITEM_TYPE_ESP:
lt = NPC_LT_LE_ESP;
info.len = pst->pattern->size;
break;
default:
return 0;
}
pst->tunnel = 1;
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pattern, &info);
if (rc != 0)
return rc;
return npc_update_parse_state(pst, &info, lid, lt, lflags);
}
int
npc_parse_lf(struct npc_parse_state *pst)
{
const struct roc_npc_item_info *pattern, *last_pattern;
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt, lflags;
int nr_vlans = 0;
int rc;
/* We hit this layer if there is a tunneling protocol */
if (!pst->tunnel)
return 0;
if (pst->pattern->type != ROC_NPC_ITEM_TYPE_ETH)
return 0;
lid = NPC_LID_LF;
lt = NPC_LT_LF_TU_ETHER;
lflags = 0;
/* No match support for vlan tags */
info.def_mask = NULL;
info.hw_mask = NULL;
info.len = pst->pattern->size;
info.spec = NULL;
info.mask = NULL;
info.hw_hdr_len = 0;
/* Look ahead and find out any VLAN tags. These can be
* detected but no data matching is available.
*/
last_pattern = pst->pattern;
pattern = pst->pattern + 1;
pattern = npc_parse_skip_void_and_any_items(pattern);
while (pattern->type == ROC_NPC_ITEM_TYPE_VLAN) {
nr_vlans++;
last_pattern = pattern;
pattern++;
pattern = npc_parse_skip_void_and_any_items(pattern);
}
switch (nr_vlans) {
case 0:
break;
case 1:
lflags = NPC_F_TU_ETHER_CTAG;
break;
case 2:
lflags = NPC_F_TU_ETHER_STAG_CTAG;
break;
default:
return NPC_ERR_PATTERN_NOTSUP;
}
info.hw_mask = &hw_mask;
info.len = pst->pattern->size;
info.hw_hdr_len = 0;
npc_get_hw_supp_mask(pst, &info, lid, lt);
info.spec = NULL;
info.mask = NULL;
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
pst->pattern = last_pattern;
return npc_update_parse_state(pst, &info, lid, lt, lflags);
}
int
npc_parse_lg(struct npc_parse_state *pst)
{
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt;
int rc;
if (!pst->tunnel)
return 0;
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.spec = NULL;
info.mask = NULL;
info.hw_hdr_len = 0;
lid = NPC_LID_LG;
if (pst->pattern->type == ROC_NPC_ITEM_TYPE_IPV4) {
lt = NPC_LT_LG_TU_IP;
info.len = pst->pattern->size;
} else if (pst->pattern->type == ROC_NPC_ITEM_TYPE_IPV6) {
lt = NPC_LT_LG_TU_IP6;
info.len = pst->pattern->size;
} else {
/* There is no tunneled IP header */
return 0;
}
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
return npc_update_parse_state(pst, &info, lid, lt, 0);
}
int
npc_parse_lh(struct npc_parse_state *pst)
{
char hw_mask[NPC_MAX_EXTRACT_HW_LEN];
struct npc_parse_item_info info;
int lid, lt;
int rc;
if (!pst->tunnel)
return 0;
info.def_mask = NULL;
info.hw_mask = &hw_mask;
info.spec = NULL;
info.mask = NULL;
info.hw_hdr_len = 0;
lid = NPC_LID_LH;
switch (pst->pattern->type) {
case ROC_NPC_ITEM_TYPE_UDP:
lt = NPC_LT_LH_TU_UDP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_TCP:
lt = NPC_LT_LH_TU_TCP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_SCTP:
lt = NPC_LT_LH_TU_SCTP;
info.len = pst->pattern->size;
break;
case ROC_NPC_ITEM_TYPE_ESP:
lt = NPC_LT_LH_TU_ESP;
info.len = pst->pattern->size;
break;
default:
return 0;
}
npc_get_hw_supp_mask(pst, &info, lid, lt);
rc = npc_parse_item_basic(pst->pattern, &info);
if (rc != 0)
return rc;
return npc_update_parse_state(pst, &info, lid, lt, 0);
}