mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-15 17:43:59 +00:00
flow_offload: add flow_rule and flow_match structures and use them
This patch wraps the dissector key and mask - that flower uses to represent the matching side - around the flow_match structure. To avoid a follow up patch that would edit the same LoCs in the drivers, this patch also wraps this new flow match structure around the flow rule object. This new structure will also contain the flow actions in follow up patches. This introduces two new interfaces: bool flow_rule_match_key(rule, dissector_id) that returns true if a given matching key is set on, and: flow_rule_match_XYZ(rule, &match); To fetch the matching side XYZ into the match container structure, to retrieve the key and the mask with one single call. Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org> Acked-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
d9b5a67522
commit
8f2566225a
@ -177,18 +177,12 @@ static int bnxt_tc_parse_actions(struct bnxt *bp,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define GET_KEY(flow_cmd, key_type) \
|
|
||||||
skb_flow_dissector_target((flow_cmd)->dissector, key_type,\
|
|
||||||
(flow_cmd)->key)
|
|
||||||
#define GET_MASK(flow_cmd, key_type) \
|
|
||||||
skb_flow_dissector_target((flow_cmd)->dissector, key_type,\
|
|
||||||
(flow_cmd)->mask)
|
|
||||||
|
|
||||||
static int bnxt_tc_parse_flow(struct bnxt *bp,
|
static int bnxt_tc_parse_flow(struct bnxt *bp,
|
||||||
struct tc_cls_flower_offload *tc_flow_cmd,
|
struct tc_cls_flower_offload *tc_flow_cmd,
|
||||||
struct bnxt_tc_flow *flow)
|
struct bnxt_tc_flow *flow)
|
||||||
{
|
{
|
||||||
struct flow_dissector *dissector = tc_flow_cmd->dissector;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(tc_flow_cmd);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
|
|
||||||
/* KEY_CONTROL and KEY_BASIC are needed for forming a meaningful key */
|
/* KEY_CONTROL and KEY_BASIC are needed for forming a meaningful key */
|
||||||
if ((dissector->used_keys & BIT(FLOW_DISSECTOR_KEY_CONTROL)) == 0 ||
|
if ((dissector->used_keys & BIT(FLOW_DISSECTOR_KEY_CONTROL)) == 0 ||
|
||||||
@ -198,140 +192,120 @@ static int bnxt_tc_parse_flow(struct bnxt *bp,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_BASIC);
|
|
||||||
struct flow_dissector_key_basic *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_BASIC);
|
|
||||||
|
|
||||||
flow->l2_key.ether_type = key->n_proto;
|
flow_rule_match_basic(rule, &match);
|
||||||
flow->l2_mask.ether_type = mask->n_proto;
|
flow->l2_key.ether_type = match.key->n_proto;
|
||||||
|
flow->l2_mask.ether_type = match.mask->n_proto;
|
||||||
|
|
||||||
if (key->n_proto == htons(ETH_P_IP) ||
|
if (match.key->n_proto == htons(ETH_P_IP) ||
|
||||||
key->n_proto == htons(ETH_P_IPV6)) {
|
match.key->n_proto == htons(ETH_P_IPV6)) {
|
||||||
flow->l4_key.ip_proto = key->ip_proto;
|
flow->l4_key.ip_proto = match.key->ip_proto;
|
||||||
flow->l4_mask.ip_proto = mask->ip_proto;
|
flow->l4_mask.ip_proto = match.mask->ip_proto;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key =
|
struct flow_match_eth_addrs match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_ETH_ADDRS);
|
|
||||||
struct flow_dissector_key_eth_addrs *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_ETH_ADDRS);
|
|
||||||
|
|
||||||
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_ETH_ADDRS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_ETH_ADDRS;
|
||||||
ether_addr_copy(flow->l2_key.dmac, key->dst);
|
ether_addr_copy(flow->l2_key.dmac, match.key->dst);
|
||||||
ether_addr_copy(flow->l2_mask.dmac, mask->dst);
|
ether_addr_copy(flow->l2_mask.dmac, match.mask->dst);
|
||||||
ether_addr_copy(flow->l2_key.smac, key->src);
|
ether_addr_copy(flow->l2_key.smac, match.key->src);
|
||||||
ether_addr_copy(flow->l2_mask.smac, mask->src);
|
ether_addr_copy(flow->l2_mask.smac, match.mask->src);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_VLAN);
|
|
||||||
struct flow_dissector_key_vlan *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_VLAN);
|
|
||||||
|
|
||||||
|
flow_rule_match_vlan(rule, &match);
|
||||||
flow->l2_key.inner_vlan_tci =
|
flow->l2_key.inner_vlan_tci =
|
||||||
cpu_to_be16(VLAN_TCI(key->vlan_id, key->vlan_priority));
|
cpu_to_be16(VLAN_TCI(match.key->vlan_id,
|
||||||
|
match.key->vlan_priority));
|
||||||
flow->l2_mask.inner_vlan_tci =
|
flow->l2_mask.inner_vlan_tci =
|
||||||
cpu_to_be16((VLAN_TCI(mask->vlan_id, mask->vlan_priority)));
|
cpu_to_be16((VLAN_TCI(match.mask->vlan_id,
|
||||||
|
match.mask->vlan_priority)));
|
||||||
flow->l2_key.inner_vlan_tpid = htons(ETH_P_8021Q);
|
flow->l2_key.inner_vlan_tpid = htons(ETH_P_8021Q);
|
||||||
flow->l2_mask.inner_vlan_tpid = htons(0xffff);
|
flow->l2_mask.inner_vlan_tpid = htons(0xffff);
|
||||||
flow->l2_key.num_vlans = 1;
|
flow->l2_key.num_vlans = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_IPV4_ADDRS);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_IPV4_ADDRS);
|
|
||||||
|
|
||||||
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_IPV4_ADDRS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_IPV4_ADDRS;
|
||||||
flow->l3_key.ipv4.daddr.s_addr = key->dst;
|
flow->l3_key.ipv4.daddr.s_addr = match.key->dst;
|
||||||
flow->l3_mask.ipv4.daddr.s_addr = mask->dst;
|
flow->l3_mask.ipv4.daddr.s_addr = match.mask->dst;
|
||||||
flow->l3_key.ipv4.saddr.s_addr = key->src;
|
flow->l3_key.ipv4.saddr.s_addr = match.key->src;
|
||||||
flow->l3_mask.ipv4.saddr.s_addr = mask->src;
|
flow->l3_mask.ipv4.saddr.s_addr = match.mask->src;
|
||||||
} else if (dissector_uses_key(dissector,
|
} else if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
struct flow_match_ipv6_addrs match;
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_IPV6_ADDRS);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_IPV6_ADDRS);
|
|
||||||
|
|
||||||
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_IPV6_ADDRS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_IPV6_ADDRS;
|
||||||
flow->l3_key.ipv6.daddr = key->dst;
|
flow->l3_key.ipv6.daddr = match.key->dst;
|
||||||
flow->l3_mask.ipv6.daddr = mask->dst;
|
flow->l3_mask.ipv6.daddr = match.mask->dst;
|
||||||
flow->l3_key.ipv6.saddr = key->src;
|
flow->l3_key.ipv6.saddr = match.key->src;
|
||||||
flow->l3_mask.ipv6.saddr = mask->src;
|
flow->l3_mask.ipv6.saddr = match.mask->src;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key =
|
struct flow_match_ports match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_PORTS);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_PORTS);
|
|
||||||
|
|
||||||
|
flow_rule_match_ports(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_PORTS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_PORTS;
|
||||||
flow->l4_key.ports.dport = key->dst;
|
flow->l4_key.ports.dport = match.key->dst;
|
||||||
flow->l4_mask.ports.dport = mask->dst;
|
flow->l4_mask.ports.dport = match.mask->dst;
|
||||||
flow->l4_key.ports.sport = key->src;
|
flow->l4_key.ports.sport = match.key->src;
|
||||||
flow->l4_mask.ports.sport = mask->src;
|
flow->l4_mask.ports.sport = match.mask->src;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_ICMP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ICMP)) {
|
||||||
struct flow_dissector_key_icmp *key =
|
struct flow_match_icmp match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_ICMP);
|
|
||||||
struct flow_dissector_key_icmp *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_ICMP);
|
|
||||||
|
|
||||||
|
flow_rule_match_icmp(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_ICMP;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_ICMP;
|
||||||
flow->l4_key.icmp.type = key->type;
|
flow->l4_key.icmp.type = match.key->type;
|
||||||
flow->l4_key.icmp.code = key->code;
|
flow->l4_key.icmp.code = match.key->code;
|
||||||
flow->l4_mask.icmp.type = mask->type;
|
flow->l4_mask.icmp.type = match.mask->type;
|
||||||
flow->l4_mask.icmp.code = mask->code;
|
flow->l4_mask.icmp.code = match.mask->code;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS)) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
GET_MASK(tc_flow_cmd,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS);
|
|
||||||
|
|
||||||
|
flow_rule_match_enc_ipv4_addrs(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_IPV4_ADDRS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_IPV4_ADDRS;
|
||||||
flow->tun_key.u.ipv4.dst = key->dst;
|
flow->tun_key.u.ipv4.dst = match.key->dst;
|
||||||
flow->tun_mask.u.ipv4.dst = mask->dst;
|
flow->tun_mask.u.ipv4.dst = match.mask->dst;
|
||||||
flow->tun_key.u.ipv4.src = key->src;
|
flow->tun_key.u.ipv4.src = match.key->src;
|
||||||
flow->tun_mask.u.ipv4.src = mask->src;
|
flow->tun_mask.u.ipv4.src = match.mask->src;
|
||||||
} else if (dissector_uses_key(dissector,
|
} else if (flow_rule_match_key(rule,
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV6_ADDRS)) {
|
FLOW_DISSECTOR_KEY_ENC_IPV6_ADDRS)) {
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *key =
|
struct flow_match_enc_keyid match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_ENC_KEYID);
|
|
||||||
struct flow_dissector_key_keyid *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_ENC_KEYID);
|
|
||||||
|
|
||||||
|
flow_rule_match_enc_keyid(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_ID;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_ID;
|
||||||
flow->tun_key.tun_id = key32_to_tunnel_id(key->keyid);
|
flow->tun_key.tun_id = key32_to_tunnel_id(match.key->keyid);
|
||||||
flow->tun_mask.tun_id = key32_to_tunnel_id(mask->keyid);
|
flow->tun_mask.tun_id = key32_to_tunnel_id(match.mask->keyid);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(dissector, FLOW_DISSECTOR_KEY_ENC_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key =
|
struct flow_match_ports match;
|
||||||
GET_KEY(tc_flow_cmd, FLOW_DISSECTOR_KEY_ENC_PORTS);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
GET_MASK(tc_flow_cmd, FLOW_DISSECTOR_KEY_ENC_PORTS);
|
|
||||||
|
|
||||||
|
flow_rule_match_enc_ports(rule, &match);
|
||||||
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_PORTS;
|
flow->flags |= BNXT_TC_FLOW_FLAGS_TUNL_PORTS;
|
||||||
flow->tun_key.tp_dst = key->dst;
|
flow->tun_key.tp_dst = match.key->dst;
|
||||||
flow->tun_mask.tp_dst = mask->dst;
|
flow->tun_mask.tp_dst = match.mask->dst;
|
||||||
flow->tun_key.tp_src = key->src;
|
flow->tun_key.tp_src = match.key->src;
|
||||||
flow->tun_mask.tp_src = mask->src;
|
flow->tun_mask.tp_src = match.mask->src;
|
||||||
}
|
}
|
||||||
|
|
||||||
return bnxt_tc_parse_actions(bp, &flow->actions, tc_flow_cmd->exts);
|
return bnxt_tc_parse_actions(bp, &flow->actions, tc_flow_cmd->exts);
|
||||||
|
@ -83,28 +83,23 @@ static void cxgb4_process_flow_match(struct net_device *dev,
|
|||||||
struct tc_cls_flower_offload *cls,
|
struct tc_cls_flower_offload *cls,
|
||||||
struct ch_filter_specification *fs)
|
struct ch_filter_specification *fs)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(cls);
|
||||||
u16 addr_type = 0;
|
u16 addr_type = 0;
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key =
|
struct flow_match_control match;
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
|
||||||
cls->key);
|
|
||||||
|
|
||||||
addr_type = key->addr_type;
|
flow_rule_match_control(rule, &match);
|
||||||
|
addr_type = match.key->addr_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(cls->dissector,
|
u16 ethtype_key, ethtype_mask;
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
cls->key);
|
flow_rule_match_basic(rule, &match);
|
||||||
struct flow_dissector_key_basic *mask =
|
ethtype_key = ntohs(match.key->n_proto);
|
||||||
skb_flow_dissector_target(cls->dissector,
|
ethtype_mask = ntohs(match.mask->n_proto);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
cls->mask);
|
|
||||||
u16 ethtype_key = ntohs(key->n_proto);
|
|
||||||
u16 ethtype_mask = ntohs(mask->n_proto);
|
|
||||||
|
|
||||||
if (ethtype_key == ETH_P_ALL) {
|
if (ethtype_key == ETH_P_ALL) {
|
||||||
ethtype_key = 0;
|
ethtype_key = 0;
|
||||||
@ -116,115 +111,89 @@ static void cxgb4_process_flow_match(struct net_device *dev,
|
|||||||
|
|
||||||
fs->val.ethtype = ethtype_key;
|
fs->val.ethtype = ethtype_key;
|
||||||
fs->mask.ethtype = ethtype_mask;
|
fs->mask.ethtype = ethtype_mask;
|
||||||
fs->val.proto = key->ip_proto;
|
fs->val.proto = match.key->ip_proto;
|
||||||
fs->mask.proto = mask->ip_proto;
|
fs->mask.proto = match.mask->ip_proto;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
cls->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
cls->mask);
|
|
||||||
fs->type = 0;
|
fs->type = 0;
|
||||||
memcpy(&fs->val.lip[0], &key->dst, sizeof(key->dst));
|
memcpy(&fs->val.lip[0], &match.key->dst, sizeof(match.key->dst));
|
||||||
memcpy(&fs->val.fip[0], &key->src, sizeof(key->src));
|
memcpy(&fs->val.fip[0], &match.key->src, sizeof(match.key->src));
|
||||||
memcpy(&fs->mask.lip[0], &mask->dst, sizeof(mask->dst));
|
memcpy(&fs->mask.lip[0], &match.mask->dst, sizeof(match.mask->dst));
|
||||||
memcpy(&fs->mask.fip[0], &mask->src, sizeof(mask->src));
|
memcpy(&fs->mask.fip[0], &match.mask->src, sizeof(match.mask->src));
|
||||||
|
|
||||||
/* also initialize nat_lip/fip to same values */
|
/* also initialize nat_lip/fip to same values */
|
||||||
memcpy(&fs->nat_lip[0], &key->dst, sizeof(key->dst));
|
memcpy(&fs->nat_lip[0], &match.key->dst, sizeof(match.key->dst));
|
||||||
memcpy(&fs->nat_fip[0], &key->src, sizeof(key->src));
|
memcpy(&fs->nat_fip[0], &match.key->src, sizeof(match.key->src));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
cls->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
cls->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
fs->type = 1;
|
fs->type = 1;
|
||||||
memcpy(&fs->val.lip[0], key->dst.s6_addr, sizeof(key->dst));
|
memcpy(&fs->val.lip[0], match.key->dst.s6_addr,
|
||||||
memcpy(&fs->val.fip[0], key->src.s6_addr, sizeof(key->src));
|
sizeof(match.key->dst));
|
||||||
memcpy(&fs->mask.lip[0], mask->dst.s6_addr, sizeof(mask->dst));
|
memcpy(&fs->val.fip[0], match.key->src.s6_addr,
|
||||||
memcpy(&fs->mask.fip[0], mask->src.s6_addr, sizeof(mask->src));
|
sizeof(match.key->src));
|
||||||
|
memcpy(&fs->mask.lip[0], match.mask->dst.s6_addr,
|
||||||
|
sizeof(match.mask->dst));
|
||||||
|
memcpy(&fs->mask.fip[0], match.mask->src.s6_addr,
|
||||||
|
sizeof(match.mask->src));
|
||||||
|
|
||||||
/* also initialize nat_lip/fip to same values */
|
/* also initialize nat_lip/fip to same values */
|
||||||
memcpy(&fs->nat_lip[0], key->dst.s6_addr, sizeof(key->dst));
|
memcpy(&fs->nat_lip[0], match.key->dst.s6_addr,
|
||||||
memcpy(&fs->nat_fip[0], key->src.s6_addr, sizeof(key->src));
|
sizeof(match.key->dst));
|
||||||
|
memcpy(&fs->nat_fip[0], match.key->src.s6_addr,
|
||||||
|
sizeof(match.key->src));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key, *mask;
|
struct flow_match_ports match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(cls->dissector,
|
flow_rule_match_ports(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
fs->val.lport = cpu_to_be16(match.key->dst);
|
||||||
cls->key);
|
fs->mask.lport = cpu_to_be16(match.mask->dst);
|
||||||
mask = skb_flow_dissector_target(cls->dissector,
|
fs->val.fport = cpu_to_be16(match.key->src);
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
fs->mask.fport = cpu_to_be16(match.mask->src);
|
||||||
cls->mask);
|
|
||||||
fs->val.lport = cpu_to_be16(key->dst);
|
|
||||||
fs->mask.lport = cpu_to_be16(mask->dst);
|
|
||||||
fs->val.fport = cpu_to_be16(key->src);
|
|
||||||
fs->mask.fport = cpu_to_be16(mask->src);
|
|
||||||
|
|
||||||
/* also initialize nat_lport/fport to same values */
|
/* also initialize nat_lport/fport to same values */
|
||||||
fs->nat_lport = cpu_to_be16(key->dst);
|
fs->nat_lport = cpu_to_be16(match.key->dst);
|
||||||
fs->nat_fport = cpu_to_be16(key->src);
|
fs->nat_fport = cpu_to_be16(match.key->src);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IP)) {
|
||||||
struct flow_dissector_key_ip *key, *mask;
|
struct flow_match_ip match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(cls->dissector,
|
flow_rule_match_ip(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
fs->val.tos = match.key->tos;
|
||||||
cls->key);
|
fs->mask.tos = match.mask->tos;
|
||||||
mask = skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
|
||||||
cls->mask);
|
|
||||||
fs->val.tos = key->tos;
|
|
||||||
fs->mask.tos = mask->tos;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *key, *mask;
|
struct flow_match_enc_keyid match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(cls->dissector,
|
flow_rule_match_enc_keyid(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
fs->val.vni = be32_to_cpu(match.key->keyid);
|
||||||
cls->key);
|
fs->mask.vni = be32_to_cpu(match.mask->keyid);
|
||||||
mask = skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
cls->mask);
|
|
||||||
fs->val.vni = be32_to_cpu(key->keyid);
|
|
||||||
fs->mask.vni = be32_to_cpu(mask->keyid);
|
|
||||||
if (fs->mask.vni) {
|
if (fs->mask.vni) {
|
||||||
fs->val.encap_vld = 1;
|
fs->val.encap_vld = 1;
|
||||||
fs->mask.encap_vld = 1;
|
fs->mask.encap_vld = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key, *mask;
|
struct flow_match_vlan match;
|
||||||
u16 vlan_tci, vlan_tci_mask;
|
u16 vlan_tci, vlan_tci_mask;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(cls->dissector,
|
flow_rule_match_vlan(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
vlan_tci = match.key->vlan_id | (match.key->vlan_priority <<
|
||||||
cls->key);
|
VLAN_PRIO_SHIFT);
|
||||||
mask = skb_flow_dissector_target(cls->dissector,
|
vlan_tci_mask = match.mask->vlan_id | (match.mask->vlan_priority <<
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
VLAN_PRIO_SHIFT);
|
||||||
cls->mask);
|
|
||||||
vlan_tci = key->vlan_id | (key->vlan_priority <<
|
|
||||||
VLAN_PRIO_SHIFT);
|
|
||||||
vlan_tci_mask = mask->vlan_id | (mask->vlan_priority <<
|
|
||||||
VLAN_PRIO_SHIFT);
|
|
||||||
fs->val.ivlan = vlan_tci;
|
fs->val.ivlan = vlan_tci;
|
||||||
fs->mask.ivlan = vlan_tci_mask;
|
fs->mask.ivlan = vlan_tci_mask;
|
||||||
|
|
||||||
@ -255,10 +224,12 @@ static void cxgb4_process_flow_match(struct net_device *dev,
|
|||||||
static int cxgb4_validate_flow_match(struct net_device *dev,
|
static int cxgb4_validate_flow_match(struct net_device *dev,
|
||||||
struct tc_cls_flower_offload *cls)
|
struct tc_cls_flower_offload *cls)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(cls);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
u16 ethtype_mask = 0;
|
u16 ethtype_mask = 0;
|
||||||
u16 ethtype_key = 0;
|
u16 ethtype_key = 0;
|
||||||
|
|
||||||
if (cls->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS) |
|
||||||
@ -268,36 +239,29 @@ static int cxgb4_validate_flow_match(struct net_device *dev,
|
|||||||
BIT(FLOW_DISSECTOR_KEY_VLAN) |
|
BIT(FLOW_DISSECTOR_KEY_VLAN) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_IP))) {
|
BIT(FLOW_DISSECTOR_KEY_IP))) {
|
||||||
netdev_warn(dev, "Unsupported key used: 0x%x\n",
|
netdev_warn(dev, "Unsupported key used: 0x%x\n",
|
||||||
cls->dissector->used_keys);
|
dissector->used_keys);
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
flow_rule_match_basic(rule, &match);
|
||||||
cls->key);
|
ethtype_key = ntohs(match.key->n_proto);
|
||||||
struct flow_dissector_key_basic *mask =
|
ethtype_mask = ntohs(match.mask->n_proto);
|
||||||
skb_flow_dissector_target(cls->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
cls->mask);
|
|
||||||
ethtype_key = ntohs(key->n_proto);
|
|
||||||
ethtype_mask = ntohs(mask->n_proto);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(cls->dissector, FLOW_DISSECTOR_KEY_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IP)) {
|
||||||
u16 eth_ip_type = ethtype_key & ethtype_mask;
|
u16 eth_ip_type = ethtype_key & ethtype_mask;
|
||||||
struct flow_dissector_key_ip *mask;
|
struct flow_match_ip match;
|
||||||
|
|
||||||
if (eth_ip_type != ETH_P_IP && eth_ip_type != ETH_P_IPV6) {
|
if (eth_ip_type != ETH_P_IP && eth_ip_type != ETH_P_IPV6) {
|
||||||
netdev_err(dev, "IP Key supported only with IPv4/v6");
|
netdev_err(dev, "IP Key supported only with IPv4/v6");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
mask = skb_flow_dissector_target(cls->dissector,
|
flow_rule_match_ip(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
if (match.mask->ttl) {
|
||||||
cls->mask);
|
|
||||||
if (mask->ttl) {
|
|
||||||
netdev_warn(dev, "ttl match unsupported for offload");
|
netdev_warn(dev, "ttl match unsupported for offload");
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
@ -7169,11 +7169,13 @@ static int i40e_parse_cls_flower(struct i40e_vsi *vsi,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct i40e_cloud_filter *filter)
|
struct i40e_cloud_filter *filter)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
u16 n_proto_mask = 0, n_proto_key = 0, addr_type = 0;
|
u16 n_proto_mask = 0, n_proto_key = 0, addr_type = 0;
|
||||||
struct i40e_pf *pf = vsi->back;
|
struct i40e_pf *pf = vsi->back;
|
||||||
u8 field_flags = 0;
|
u8 field_flags = 0;
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
||||||
@ -7183,143 +7185,109 @@ static int i40e_parse_cls_flower(struct i40e_vsi *vsi,
|
|||||||
BIT(FLOW_DISSECTOR_KEY_PORTS) |
|
BIT(FLOW_DISSECTOR_KEY_PORTS) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ENC_KEYID))) {
|
BIT(FLOW_DISSECTOR_KEY_ENC_KEYID))) {
|
||||||
dev_err(&pf->pdev->dev, "Unsupported key used: 0x%x\n",
|
dev_err(&pf->pdev->dev, "Unsupported key used: 0x%x\n",
|
||||||
f->dissector->used_keys);
|
dissector->used_keys);
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *key =
|
struct flow_match_enc_keyid match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_keyid *mask =
|
flow_rule_match_enc_keyid(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
if (match.mask->keyid != 0)
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->keyid != 0)
|
|
||||||
field_flags |= I40E_CLOUD_FIELD_TEN_ID;
|
field_flags |= I40E_CLOUD_FIELD_TEN_ID;
|
||||||
|
|
||||||
filter->tenant_id = be32_to_cpu(key->keyid);
|
filter->tenant_id = be32_to_cpu(match.key->keyid);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_basic *mask =
|
flow_rule_match_basic(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
n_proto_key = ntohs(match.key->n_proto);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
n_proto_mask = ntohs(match.mask->n_proto);
|
||||||
f->mask);
|
|
||||||
|
|
||||||
n_proto_key = ntohs(key->n_proto);
|
|
||||||
n_proto_mask = ntohs(mask->n_proto);
|
|
||||||
|
|
||||||
if (n_proto_key == ETH_P_ALL) {
|
if (n_proto_key == ETH_P_ALL) {
|
||||||
n_proto_key = 0;
|
n_proto_key = 0;
|
||||||
n_proto_mask = 0;
|
n_proto_mask = 0;
|
||||||
}
|
}
|
||||||
filter->n_proto = n_proto_key & n_proto_mask;
|
filter->n_proto = n_proto_key & n_proto_mask;
|
||||||
filter->ip_proto = key->ip_proto;
|
filter->ip_proto = match.key->ip_proto;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key =
|
struct flow_match_eth_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_eth_addrs *mask =
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
/* use is_broadcast and is_zero to check for all 0xf or 0 */
|
/* use is_broadcast and is_zero to check for all 0xf or 0 */
|
||||||
if (!is_zero_ether_addr(mask->dst)) {
|
if (!is_zero_ether_addr(match.mask->dst)) {
|
||||||
if (is_broadcast_ether_addr(mask->dst)) {
|
if (is_broadcast_ether_addr(match.mask->dst)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_OMAC;
|
field_flags |= I40E_CLOUD_FIELD_OMAC;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad ether dest mask %pM\n",
|
dev_err(&pf->pdev->dev, "Bad ether dest mask %pM\n",
|
||||||
mask->dst);
|
match.mask->dst);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_zero_ether_addr(mask->src)) {
|
if (!is_zero_ether_addr(match.mask->src)) {
|
||||||
if (is_broadcast_ether_addr(mask->src)) {
|
if (is_broadcast_ether_addr(match.mask->src)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IMAC;
|
field_flags |= I40E_CLOUD_FIELD_IMAC;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad ether src mask %pM\n",
|
dev_err(&pf->pdev->dev, "Bad ether src mask %pM\n",
|
||||||
mask->src);
|
match.mask->src);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ether_addr_copy(filter->dst_mac, key->dst);
|
ether_addr_copy(filter->dst_mac, match.key->dst);
|
||||||
ether_addr_copy(filter->src_mac, key->src);
|
ether_addr_copy(filter->src_mac, match.key->src);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_vlan *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->vlan_id) {
|
flow_rule_match_vlan(rule, &match);
|
||||||
if (mask->vlan_id == VLAN_VID_MASK) {
|
if (match.mask->vlan_id) {
|
||||||
|
if (match.mask->vlan_id == VLAN_VID_MASK) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IVLAN;
|
field_flags |= I40E_CLOUD_FIELD_IVLAN;
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad vlan mask 0x%04x\n",
|
dev_err(&pf->pdev->dev, "Bad vlan mask 0x%04x\n",
|
||||||
mask->vlan_id);
|
match.mask->vlan_id);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
filter->vlan_id = cpu_to_be16(key->vlan_id);
|
filter->vlan_id = cpu_to_be16(match.key->vlan_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key =
|
struct flow_match_control match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
addr_type = key->addr_type;
|
flow_rule_match_control(rule, &match);
|
||||||
|
addr_type = match.key->addr_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->dst) {
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
if (mask->dst == cpu_to_be32(0xffffffff)) {
|
if (match.mask->dst) {
|
||||||
|
if (match.mask->dst == cpu_to_be32(0xffffffff)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IIP;
|
field_flags |= I40E_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad ip dst mask %pI4b\n",
|
dev_err(&pf->pdev->dev, "Bad ip dst mask %pI4b\n",
|
||||||
&mask->dst);
|
&match.mask->dst);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->src) {
|
if (match.mask->src) {
|
||||||
if (mask->src == cpu_to_be32(0xffffffff)) {
|
if (match.mask->src == cpu_to_be32(0xffffffff)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IIP;
|
field_flags |= I40E_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad ip src mask %pI4b\n",
|
dev_err(&pf->pdev->dev, "Bad ip src mask %pI4b\n",
|
||||||
&mask->src);
|
&match.mask->src);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -7328,70 +7296,60 @@ static int i40e_parse_cls_flower(struct i40e_vsi *vsi,
|
|||||||
dev_err(&pf->pdev->dev, "Tenant id not allowed for ip filter\n");
|
dev_err(&pf->pdev->dev, "Tenant id not allowed for ip filter\n");
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
filter->dst_ipv4 = key->dst;
|
filter->dst_ipv4 = match.key->dst;
|
||||||
filter->src_ipv4 = key->src;
|
filter->src_ipv4 = match.key->src;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
/* src and dest IPV6 address should not be LOOPBACK
|
/* src and dest IPV6 address should not be LOOPBACK
|
||||||
* (0:0:0:0:0:0:0:1), which can be represented as ::1
|
* (0:0:0:0:0:0:0:1), which can be represented as ::1
|
||||||
*/
|
*/
|
||||||
if (ipv6_addr_loopback(&key->dst) ||
|
if (ipv6_addr_loopback(&match.key->dst) ||
|
||||||
ipv6_addr_loopback(&key->src)) {
|
ipv6_addr_loopback(&match.key->src)) {
|
||||||
dev_err(&pf->pdev->dev,
|
dev_err(&pf->pdev->dev,
|
||||||
"Bad ipv6, addr is LOOPBACK\n");
|
"Bad ipv6, addr is LOOPBACK\n");
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
if (!ipv6_addr_any(&mask->dst) || !ipv6_addr_any(&mask->src))
|
if (!ipv6_addr_any(&match.mask->dst) ||
|
||||||
|
!ipv6_addr_any(&match.mask->src))
|
||||||
field_flags |= I40E_CLOUD_FIELD_IIP;
|
field_flags |= I40E_CLOUD_FIELD_IIP;
|
||||||
|
|
||||||
memcpy(&filter->src_ipv6, &key->src.s6_addr32,
|
memcpy(&filter->src_ipv6, &match.key->src.s6_addr32,
|
||||||
sizeof(filter->src_ipv6));
|
sizeof(filter->src_ipv6));
|
||||||
memcpy(&filter->dst_ipv6, &key->dst.s6_addr32,
|
memcpy(&filter->dst_ipv6, &match.key->dst.s6_addr32,
|
||||||
sizeof(filter->dst_ipv6));
|
sizeof(filter->dst_ipv6));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key =
|
struct flow_match_ports match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->src) {
|
flow_rule_match_ports(rule, &match);
|
||||||
if (mask->src == cpu_to_be16(0xffff)) {
|
if (match.mask->src) {
|
||||||
|
if (match.mask->src == cpu_to_be16(0xffff)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IIP;
|
field_flags |= I40E_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad src port mask 0x%04x\n",
|
dev_err(&pf->pdev->dev, "Bad src port mask 0x%04x\n",
|
||||||
be16_to_cpu(mask->src));
|
be16_to_cpu(match.mask->src));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->dst) {
|
if (match.mask->dst) {
|
||||||
if (mask->dst == cpu_to_be16(0xffff)) {
|
if (match.mask->dst == cpu_to_be16(0xffff)) {
|
||||||
field_flags |= I40E_CLOUD_FIELD_IIP;
|
field_flags |= I40E_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&pf->pdev->dev, "Bad dst port mask 0x%04x\n",
|
dev_err(&pf->pdev->dev, "Bad dst port mask 0x%04x\n",
|
||||||
be16_to_cpu(mask->dst));
|
be16_to_cpu(match.mask->dst));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
filter->dst_port = key->dst;
|
filter->dst_port = match.key->dst;
|
||||||
filter->src_port = key->src;
|
filter->src_port = match.key->src;
|
||||||
|
|
||||||
switch (filter->ip_proto) {
|
switch (filter->ip_proto) {
|
||||||
case IPPROTO_TCP:
|
case IPPROTO_TCP:
|
||||||
|
@ -2439,6 +2439,8 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct iavf_cloud_filter *filter)
|
struct iavf_cloud_filter *filter)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
u16 n_proto_mask = 0;
|
u16 n_proto_mask = 0;
|
||||||
u16 n_proto_key = 0;
|
u16 n_proto_key = 0;
|
||||||
u8 field_flags = 0;
|
u8 field_flags = 0;
|
||||||
@ -2447,7 +2449,7 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
int i = 0;
|
int i = 0;
|
||||||
struct virtchnl_filter *vf = &filter->f;
|
struct virtchnl_filter *vf = &filter->f;
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
||||||
@ -2457,32 +2459,24 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
BIT(FLOW_DISSECTOR_KEY_PORTS) |
|
BIT(FLOW_DISSECTOR_KEY_PORTS) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ENC_KEYID))) {
|
BIT(FLOW_DISSECTOR_KEY_ENC_KEYID))) {
|
||||||
dev_err(&adapter->pdev->dev, "Unsupported key used: 0x%x\n",
|
dev_err(&adapter->pdev->dev, "Unsupported key used: 0x%x\n",
|
||||||
f->dissector->used_keys);
|
dissector->used_keys);
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *mask =
|
struct flow_match_enc_keyid match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->keyid != 0)
|
flow_rule_match_enc_keyid(rule, &match);
|
||||||
|
if (match.mask->keyid != 0)
|
||||||
field_flags |= IAVF_CLOUD_FIELD_TEN_ID;
|
field_flags |= IAVF_CLOUD_FIELD_TEN_ID;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_basic *mask =
|
flow_rule_match_basic(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
n_proto_key = ntohs(match.key->n_proto);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
n_proto_mask = ntohs(match.mask->n_proto);
|
||||||
f->mask);
|
|
||||||
n_proto_key = ntohs(key->n_proto);
|
|
||||||
n_proto_mask = ntohs(mask->n_proto);
|
|
||||||
|
|
||||||
if (n_proto_key == ETH_P_ALL) {
|
if (n_proto_key == ETH_P_ALL) {
|
||||||
n_proto_key = 0;
|
n_proto_key = 0;
|
||||||
@ -2496,122 +2490,103 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
vf->flow_type = VIRTCHNL_TCP_V6_FLOW;
|
vf->flow_type = VIRTCHNL_TCP_V6_FLOW;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (key->ip_proto != IPPROTO_TCP) {
|
if (match.key->ip_proto != IPPROTO_TCP) {
|
||||||
dev_info(&adapter->pdev->dev, "Only TCP transport is supported\n");
|
dev_info(&adapter->pdev->dev, "Only TCP transport is supported\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key =
|
struct flow_match_eth_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_eth_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
/* use is_broadcast and is_zero to check for all 0xf or 0 */
|
/* use is_broadcast and is_zero to check for all 0xf or 0 */
|
||||||
if (!is_zero_ether_addr(mask->dst)) {
|
if (!is_zero_ether_addr(match.mask->dst)) {
|
||||||
if (is_broadcast_ether_addr(mask->dst)) {
|
if (is_broadcast_ether_addr(match.mask->dst)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_OMAC;
|
field_flags |= IAVF_CLOUD_FIELD_OMAC;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad ether dest mask %pM\n",
|
dev_err(&adapter->pdev->dev, "Bad ether dest mask %pM\n",
|
||||||
mask->dst);
|
match.mask->dst);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_zero_ether_addr(mask->src)) {
|
if (!is_zero_ether_addr(match.mask->src)) {
|
||||||
if (is_broadcast_ether_addr(mask->src)) {
|
if (is_broadcast_ether_addr(match.mask->src)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IMAC;
|
field_flags |= IAVF_CLOUD_FIELD_IMAC;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad ether src mask %pM\n",
|
dev_err(&adapter->pdev->dev, "Bad ether src mask %pM\n",
|
||||||
mask->src);
|
match.mask->src);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_zero_ether_addr(key->dst))
|
if (!is_zero_ether_addr(match.key->dst))
|
||||||
if (is_valid_ether_addr(key->dst) ||
|
if (is_valid_ether_addr(match.key->dst) ||
|
||||||
is_multicast_ether_addr(key->dst)) {
|
is_multicast_ether_addr(match.key->dst)) {
|
||||||
/* set the mask if a valid dst_mac address */
|
/* set the mask if a valid dst_mac address */
|
||||||
for (i = 0; i < ETH_ALEN; i++)
|
for (i = 0; i < ETH_ALEN; i++)
|
||||||
vf->mask.tcp_spec.dst_mac[i] |= 0xff;
|
vf->mask.tcp_spec.dst_mac[i] |= 0xff;
|
||||||
ether_addr_copy(vf->data.tcp_spec.dst_mac,
|
ether_addr_copy(vf->data.tcp_spec.dst_mac,
|
||||||
key->dst);
|
match.key->dst);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_zero_ether_addr(key->src))
|
if (!is_zero_ether_addr(match.key->src))
|
||||||
if (is_valid_ether_addr(key->src) ||
|
if (is_valid_ether_addr(match.key->src) ||
|
||||||
is_multicast_ether_addr(key->src)) {
|
is_multicast_ether_addr(match.key->src)) {
|
||||||
/* set the mask if a valid dst_mac address */
|
/* set the mask if a valid dst_mac address */
|
||||||
for (i = 0; i < ETH_ALEN; i++)
|
for (i = 0; i < ETH_ALEN; i++)
|
||||||
vf->mask.tcp_spec.src_mac[i] |= 0xff;
|
vf->mask.tcp_spec.src_mac[i] |= 0xff;
|
||||||
ether_addr_copy(vf->data.tcp_spec.src_mac,
|
ether_addr_copy(vf->data.tcp_spec.src_mac,
|
||||||
key->src);
|
match.key->src);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_vlan *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->vlan_id) {
|
flow_rule_match_vlan(rule, &match);
|
||||||
if (mask->vlan_id == VLAN_VID_MASK) {
|
if (match.mask->vlan_id) {
|
||||||
|
if (match.mask->vlan_id == VLAN_VID_MASK) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IVLAN;
|
field_flags |= IAVF_CLOUD_FIELD_IVLAN;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad vlan mask %u\n",
|
dev_err(&adapter->pdev->dev, "Bad vlan mask %u\n",
|
||||||
mask->vlan_id);
|
match.mask->vlan_id);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
vf->mask.tcp_spec.vlan_id |= cpu_to_be16(0xffff);
|
vf->mask.tcp_spec.vlan_id |= cpu_to_be16(0xffff);
|
||||||
vf->data.tcp_spec.vlan_id = cpu_to_be16(key->vlan_id);
|
vf->data.tcp_spec.vlan_id = cpu_to_be16(match.key->vlan_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key =
|
struct flow_match_control match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
addr_type = key->addr_type;
|
flow_rule_match_control(rule, &match);
|
||||||
|
addr_type = match.key->addr_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->dst) {
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
if (mask->dst == cpu_to_be32(0xffffffff)) {
|
if (match.mask->dst) {
|
||||||
|
if (match.mask->dst == cpu_to_be32(0xffffffff)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad ip dst mask 0x%08x\n",
|
dev_err(&adapter->pdev->dev, "Bad ip dst mask 0x%08x\n",
|
||||||
be32_to_cpu(mask->dst));
|
be32_to_cpu(match.mask->dst));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->src) {
|
if (match.mask->src) {
|
||||||
if (mask->src == cpu_to_be32(0xffffffff)) {
|
if (match.mask->src == cpu_to_be32(0xffffffff)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad ip src mask 0x%08x\n",
|
dev_err(&adapter->pdev->dev, "Bad ip src mask 0x%08x\n",
|
||||||
be32_to_cpu(mask->dst));
|
be32_to_cpu(match.mask->dst));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2620,28 +2595,23 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
dev_info(&adapter->pdev->dev, "Tenant id not allowed for ip filter\n");
|
dev_info(&adapter->pdev->dev, "Tenant id not allowed for ip filter\n");
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
if (key->dst) {
|
if (match.key->dst) {
|
||||||
vf->mask.tcp_spec.dst_ip[0] |= cpu_to_be32(0xffffffff);
|
vf->mask.tcp_spec.dst_ip[0] |= cpu_to_be32(0xffffffff);
|
||||||
vf->data.tcp_spec.dst_ip[0] = key->dst;
|
vf->data.tcp_spec.dst_ip[0] = match.key->dst;
|
||||||
}
|
}
|
||||||
if (key->src) {
|
if (match.key->src) {
|
||||||
vf->mask.tcp_spec.src_ip[0] |= cpu_to_be32(0xffffffff);
|
vf->mask.tcp_spec.src_ip[0] |= cpu_to_be32(0xffffffff);
|
||||||
vf->data.tcp_spec.src_ip[0] = key->src;
|
vf->data.tcp_spec.src_ip[0] = match.key->src;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
/* validate mask, make sure it is not IPV6_ADDR_ANY */
|
/* validate mask, make sure it is not IPV6_ADDR_ANY */
|
||||||
if (ipv6_addr_any(&mask->dst)) {
|
if (ipv6_addr_any(&match.mask->dst)) {
|
||||||
dev_err(&adapter->pdev->dev, "Bad ipv6 dst mask 0x%02x\n",
|
dev_err(&adapter->pdev->dev, "Bad ipv6 dst mask 0x%02x\n",
|
||||||
IPV6_ADDR_ANY);
|
IPV6_ADDR_ANY);
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
@ -2650,61 +2620,56 @@ static int iavf_parse_cls_flower(struct iavf_adapter *adapter,
|
|||||||
/* src and dest IPv6 address should not be LOOPBACK
|
/* src and dest IPv6 address should not be LOOPBACK
|
||||||
* (0:0:0:0:0:0:0:1) which can be represented as ::1
|
* (0:0:0:0:0:0:0:1) which can be represented as ::1
|
||||||
*/
|
*/
|
||||||
if (ipv6_addr_loopback(&key->dst) ||
|
if (ipv6_addr_loopback(&match.key->dst) ||
|
||||||
ipv6_addr_loopback(&key->src)) {
|
ipv6_addr_loopback(&match.key->src)) {
|
||||||
dev_err(&adapter->pdev->dev,
|
dev_err(&adapter->pdev->dev,
|
||||||
"ipv6 addr should not be loopback\n");
|
"ipv6 addr should not be loopback\n");
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
if (!ipv6_addr_any(&mask->dst) || !ipv6_addr_any(&mask->src))
|
if (!ipv6_addr_any(&match.mask->dst) ||
|
||||||
|
!ipv6_addr_any(&match.mask->src))
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
||||||
|
|
||||||
for (i = 0; i < 4; i++)
|
for (i = 0; i < 4; i++)
|
||||||
vf->mask.tcp_spec.dst_ip[i] |= cpu_to_be32(0xffffffff);
|
vf->mask.tcp_spec.dst_ip[i] |= cpu_to_be32(0xffffffff);
|
||||||
memcpy(&vf->data.tcp_spec.dst_ip, &key->dst.s6_addr32,
|
memcpy(&vf->data.tcp_spec.dst_ip, &match.key->dst.s6_addr32,
|
||||||
sizeof(vf->data.tcp_spec.dst_ip));
|
sizeof(vf->data.tcp_spec.dst_ip));
|
||||||
for (i = 0; i < 4; i++)
|
for (i = 0; i < 4; i++)
|
||||||
vf->mask.tcp_spec.src_ip[i] |= cpu_to_be32(0xffffffff);
|
vf->mask.tcp_spec.src_ip[i] |= cpu_to_be32(0xffffffff);
|
||||||
memcpy(&vf->data.tcp_spec.src_ip, &key->src.s6_addr32,
|
memcpy(&vf->data.tcp_spec.src_ip, &match.key->src.s6_addr32,
|
||||||
sizeof(vf->data.tcp_spec.src_ip));
|
sizeof(vf->data.tcp_spec.src_ip));
|
||||||
}
|
}
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key =
|
struct flow_match_ports match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->src) {
|
flow_rule_match_ports(rule, &match);
|
||||||
if (mask->src == cpu_to_be16(0xffff)) {
|
if (match.mask->src) {
|
||||||
|
if (match.mask->src == cpu_to_be16(0xffff)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad src port mask %u\n",
|
dev_err(&adapter->pdev->dev, "Bad src port mask %u\n",
|
||||||
be16_to_cpu(mask->src));
|
be16_to_cpu(match.mask->src));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->dst) {
|
if (match.mask->dst) {
|
||||||
if (mask->dst == cpu_to_be16(0xffff)) {
|
if (match.mask->dst == cpu_to_be16(0xffff)) {
|
||||||
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
field_flags |= IAVF_CLOUD_FIELD_IIP;
|
||||||
} else {
|
} else {
|
||||||
dev_err(&adapter->pdev->dev, "Bad dst port mask %u\n",
|
dev_err(&adapter->pdev->dev, "Bad dst port mask %u\n",
|
||||||
be16_to_cpu(mask->dst));
|
be16_to_cpu(match.mask->dst));
|
||||||
return I40E_ERR_CONFIG;
|
return I40E_ERR_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (key->dst) {
|
if (match.key->dst) {
|
||||||
vf->mask.tcp_spec.dst_port |= cpu_to_be16(0xffff);
|
vf->mask.tcp_spec.dst_port |= cpu_to_be16(0xffff);
|
||||||
vf->data.tcp_spec.dst_port = key->dst;
|
vf->data.tcp_spec.dst_port = match.key->dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (key->src) {
|
if (match.key->src) {
|
||||||
vf->mask.tcp_spec.src_port |= cpu_to_be16(0xffff);
|
vf->mask.tcp_spec.src_port |= cpu_to_be16(0xffff);
|
||||||
vf->data.tcp_spec.src_port = key->src;
|
vf->data.tcp_spec.src_port = match.key->src;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
vf->field_flags = field_flags;
|
vf->field_flags = field_flags;
|
||||||
|
@ -2581,9 +2581,11 @@ static int igb_parse_cls_flower(struct igb_adapter *adapter,
|
|||||||
int traffic_class,
|
int traffic_class,
|
||||||
struct igb_nfc_filter *input)
|
struct igb_nfc_filter *input)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
struct netlink_ext_ack *extack = f->common.extack;
|
struct netlink_ext_ack *extack = f->common.extack;
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
~(BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
||||||
@ -2593,78 +2595,60 @@ static int igb_parse_cls_flower(struct igb_adapter *adapter,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key, *mask;
|
struct flow_match_eth_addrs match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
if (!is_zero_ether_addr(match.mask->dst)) {
|
||||||
f->key);
|
if (!is_broadcast_ether_addr(match.mask->dst)) {
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (!is_zero_ether_addr(mask->dst)) {
|
|
||||||
if (!is_broadcast_ether_addr(mask->dst)) {
|
|
||||||
NL_SET_ERR_MSG_MOD(extack, "Only full masks are supported for destination MAC address");
|
NL_SET_ERR_MSG_MOD(extack, "Only full masks are supported for destination MAC address");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
input->filter.match_flags |=
|
input->filter.match_flags |=
|
||||||
IGB_FILTER_FLAG_DST_MAC_ADDR;
|
IGB_FILTER_FLAG_DST_MAC_ADDR;
|
||||||
ether_addr_copy(input->filter.dst_addr, key->dst);
|
ether_addr_copy(input->filter.dst_addr, match.key->dst);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_zero_ether_addr(mask->src)) {
|
if (!is_zero_ether_addr(match.mask->src)) {
|
||||||
if (!is_broadcast_ether_addr(mask->src)) {
|
if (!is_broadcast_ether_addr(match.mask->src)) {
|
||||||
NL_SET_ERR_MSG_MOD(extack, "Only full masks are supported for source MAC address");
|
NL_SET_ERR_MSG_MOD(extack, "Only full masks are supported for source MAC address");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
input->filter.match_flags |=
|
input->filter.match_flags |=
|
||||||
IGB_FILTER_FLAG_SRC_MAC_ADDR;
|
IGB_FILTER_FLAG_SRC_MAC_ADDR;
|
||||||
ether_addr_copy(input->filter.src_addr, key->src);
|
ether_addr_copy(input->filter.src_addr, match.key->src);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key, *mask;
|
struct flow_match_basic match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_basic(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
if (match.mask->n_proto) {
|
||||||
f->key);
|
if (match.mask->n_proto != ETHER_TYPE_FULL_MASK) {
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->n_proto) {
|
|
||||||
if (mask->n_proto != ETHER_TYPE_FULL_MASK) {
|
|
||||||
NL_SET_ERR_MSG_MOD(extack, "Only full mask is supported for EtherType filter");
|
NL_SET_ERR_MSG_MOD(extack, "Only full mask is supported for EtherType filter");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
input->filter.match_flags |= IGB_FILTER_FLAG_ETHER_TYPE;
|
input->filter.match_flags |= IGB_FILTER_FLAG_ETHER_TYPE;
|
||||||
input->filter.etype = key->n_proto;
|
input->filter.etype = match.key->n_proto;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key, *mask;
|
struct flow_match_vlan match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_vlan(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
if (match.mask->vlan_priority) {
|
||||||
f->key);
|
if (match.mask->vlan_priority != VLAN_PRIO_FULL_MASK) {
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if (mask->vlan_priority) {
|
|
||||||
if (mask->vlan_priority != VLAN_PRIO_FULL_MASK) {
|
|
||||||
NL_SET_ERR_MSG_MOD(extack, "Only full mask is supported for VLAN priority");
|
NL_SET_ERR_MSG_MOD(extack, "Only full mask is supported for VLAN priority");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
input->filter.match_flags |= IGB_FILTER_FLAG_VLAN_TCI;
|
input->filter.match_flags |= IGB_FILTER_FLAG_VLAN_TCI;
|
||||||
input->filter.vlan_tci = key->vlan_priority;
|
input->filter.vlan_tci = match.key->vlan_priority;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -496,25 +496,21 @@ static int mlx5e_tc_tun_parse_vxlan(struct mlx5e_priv *priv,
|
|||||||
void *headers_c,
|
void *headers_c,
|
||||||
void *headers_v)
|
void *headers_v)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
struct netlink_ext_ack *extack = f->common.extack;
|
struct netlink_ext_ack *extack = f->common.extack;
|
||||||
struct flow_dissector_key_ports *key =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_PORTS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_PORTS,
|
|
||||||
f->mask);
|
|
||||||
void *misc_c = MLX5_ADDR_OF(fte_match_param,
|
void *misc_c = MLX5_ADDR_OF(fte_match_param,
|
||||||
spec->match_criteria,
|
spec->match_criteria,
|
||||||
misc_parameters);
|
misc_parameters);
|
||||||
void *misc_v = MLX5_ADDR_OF(fte_match_param,
|
void *misc_v = MLX5_ADDR_OF(fte_match_param,
|
||||||
spec->match_value,
|
spec->match_value,
|
||||||
misc_parameters);
|
misc_parameters);
|
||||||
|
struct flow_match_ports enc_ports;
|
||||||
|
|
||||||
|
flow_rule_match_enc_ports(rule, &enc_ports);
|
||||||
|
|
||||||
/* Full udp dst port must be given */
|
/* Full udp dst port must be given */
|
||||||
if (!dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_PORTS) ||
|
if (!flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_PORTS) ||
|
||||||
memchr_inv(&mask->dst, 0xff, sizeof(mask->dst))) {
|
memchr_inv(&enc_ports.mask->dst, 0xff, sizeof(enc_ports.mask->dst))) {
|
||||||
NL_SET_ERR_MSG_MOD(extack,
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
"VXLAN decap filter must include enc_dst_port condition");
|
"VXLAN decap filter must include enc_dst_port condition");
|
||||||
netdev_warn(priv->netdev,
|
netdev_warn(priv->netdev,
|
||||||
@ -523,12 +519,12 @@ static int mlx5e_tc_tun_parse_vxlan(struct mlx5e_priv *priv,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* udp dst port must be knonwn as a VXLAN port */
|
/* udp dst port must be knonwn as a VXLAN port */
|
||||||
if (!mlx5_vxlan_lookup_port(priv->mdev->vxlan, be16_to_cpu(key->dst))) {
|
if (!mlx5_vxlan_lookup_port(priv->mdev->vxlan, be16_to_cpu(enc_ports.key->dst))) {
|
||||||
NL_SET_ERR_MSG_MOD(extack,
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
"Matched UDP port is not registered as a VXLAN port");
|
"Matched UDP port is not registered as a VXLAN port");
|
||||||
netdev_warn(priv->netdev,
|
netdev_warn(priv->netdev,
|
||||||
"UDP port %d is not registered as a VXLAN port\n",
|
"UDP port %d is not registered as a VXLAN port\n",
|
||||||
be16_to_cpu(key->dst));
|
be16_to_cpu(enc_ports.key->dst));
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -536,26 +532,26 @@ static int mlx5e_tc_tun_parse_vxlan(struct mlx5e_priv *priv,
|
|||||||
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ip_protocol);
|
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ip_protocol);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_protocol, IPPROTO_UDP);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_protocol, IPPROTO_UDP);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, udp_dport, ntohs(mask->dst));
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, udp_dport,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, udp_dport, ntohs(key->dst));
|
ntohs(enc_ports.mask->dst));
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, udp_dport,
|
||||||
|
ntohs(enc_ports.key->dst));
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, udp_sport, ntohs(mask->src));
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, udp_sport,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, udp_sport, ntohs(key->src));
|
ntohs(enc_ports.mask->src));
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, udp_sport,
|
||||||
|
ntohs(enc_ports.key->src));
|
||||||
|
|
||||||
/* match on VNI */
|
/* match on VNI */
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *key =
|
struct flow_match_enc_keyid enc_keyid;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
flow_rule_match_enc_keyid(rule, &enc_keyid);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_keyid *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->mask);
|
|
||||||
MLX5_SET(fte_match_set_misc, misc_c, vxlan_vni,
|
MLX5_SET(fte_match_set_misc, misc_c, vxlan_vni,
|
||||||
be32_to_cpu(mask->keyid));
|
be32_to_cpu(enc_keyid.mask->keyid));
|
||||||
MLX5_SET(fte_match_set_misc, misc_v, vxlan_vni,
|
MLX5_SET(fte_match_set_misc, misc_v, vxlan_vni,
|
||||||
be32_to_cpu(key->keyid));
|
be32_to_cpu(enc_keyid.key->keyid));
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -570,6 +566,7 @@ static int mlx5e_tc_tun_parse_gretap(struct mlx5e_priv *priv,
|
|||||||
misc_parameters);
|
misc_parameters);
|
||||||
void *misc_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
void *misc_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
||||||
misc_parameters);
|
misc_parameters);
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
|
||||||
if (!MLX5_CAP_ESW(priv->mdev, nvgre_encap_decap)) {
|
if (!MLX5_CAP_ESW(priv->mdev, nvgre_encap_decap)) {
|
||||||
NL_SET_ERR_MSG_MOD(f->common.extack,
|
NL_SET_ERR_MSG_MOD(f->common.extack,
|
||||||
@ -587,21 +584,14 @@ static int mlx5e_tc_tun_parse_gretap(struct mlx5e_priv *priv,
|
|||||||
MLX5_SET(fte_match_set_misc, misc_v, gre_protocol, ETH_P_TEB);
|
MLX5_SET(fte_match_set_misc, misc_v, gre_protocol, ETH_P_TEB);
|
||||||
|
|
||||||
/* gre key */
|
/* gre key */
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
struct flow_dissector_key_keyid *mask = NULL;
|
struct flow_match_enc_keyid enc_keyid;
|
||||||
struct flow_dissector_key_keyid *key = NULL;
|
|
||||||
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_enc_keyid(rule, &enc_keyid);
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->mask);
|
|
||||||
MLX5_SET(fte_match_set_misc, misc_c,
|
MLX5_SET(fte_match_set_misc, misc_c,
|
||||||
gre_key.key, be32_to_cpu(mask->keyid));
|
gre_key.key, be32_to_cpu(enc_keyid.mask->keyid));
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
|
||||||
f->key);
|
|
||||||
MLX5_SET(fte_match_set_misc, misc_v,
|
MLX5_SET(fte_match_set_misc, misc_v,
|
||||||
gre_key.key, be32_to_cpu(key->keyid));
|
gre_key.key, be32_to_cpu(enc_keyid.key->keyid));
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1309,12 +1309,9 @@ static int parse_tunnel_attr(struct mlx5e_priv *priv,
|
|||||||
outer_headers);
|
outer_headers);
|
||||||
void *headers_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
void *headers_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
||||||
outer_headers);
|
outer_headers);
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
struct flow_dissector_key_control *enc_control =
|
struct flow_match_control enc_control;
|
||||||
skb_flow_dissector_target(f->dissector,
|
int err;
|
||||||
FLOW_DISSECTOR_KEY_ENC_CONTROL,
|
|
||||||
f->key);
|
|
||||||
int err = 0;
|
|
||||||
|
|
||||||
err = mlx5e_tc_tun_parse(filter_dev, priv, spec, f,
|
err = mlx5e_tc_tun_parse(filter_dev, priv, spec, f,
|
||||||
headers_c, headers_v);
|
headers_c, headers_v);
|
||||||
@ -1324,79 +1321,70 @@ static int parse_tunnel_attr(struct mlx5e_priv *priv,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (enc_control->addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
flow_rule_match_enc_control(rule, &enc_control);
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
if (enc_control.key->addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS,
|
struct flow_match_ipv4_addrs match;
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
flow_rule_match_enc_ipv4_addrs(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
src_ipv4_src_ipv6.ipv4_layout.ipv4,
|
src_ipv4_src_ipv6.ipv4_layout.ipv4,
|
||||||
ntohl(mask->src));
|
ntohl(match.mask->src));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
src_ipv4_src_ipv6.ipv4_layout.ipv4,
|
src_ipv4_src_ipv6.ipv4_layout.ipv4,
|
||||||
ntohl(key->src));
|
ntohl(match.key->src));
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
dst_ipv4_dst_ipv6.ipv4_layout.ipv4,
|
dst_ipv4_dst_ipv6.ipv4_layout.ipv4,
|
||||||
ntohl(mask->dst));
|
ntohl(match.mask->dst));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
dst_ipv4_dst_ipv6.ipv4_layout.ipv4,
|
dst_ipv4_dst_ipv6.ipv4_layout.ipv4,
|
||||||
ntohl(key->dst));
|
ntohl(match.key->dst));
|
||||||
|
|
||||||
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ethertype);
|
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ethertype);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype, ETH_P_IP);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype, ETH_P_IP);
|
||||||
} else if (enc_control->addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
} else if (enc_control.key->addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV6_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV6_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_enc_ipv6_addrs(rule, &match);
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
||||||
&mask->src, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
&match.mask->src, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
||||||
&key->src, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
&match.key->src, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
||||||
|
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
||||||
&mask->dst, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
&match.mask->dst, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
||||||
&key->dst, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
&match.key->dst, MLX5_FLD_SZ_BYTES(ipv6_layout, ipv6));
|
||||||
|
|
||||||
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ethertype);
|
MLX5_SET_TO_ONES(fte_match_set_lyr_2_4, headers_c, ethertype);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype, ETH_P_IPV6);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype, ETH_P_IPV6);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_IP)) {
|
||||||
struct flow_dissector_key_ip *key =
|
struct flow_match_ip match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IP,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ip *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_IP,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_ecn, mask->tos & 0x3);
|
flow_rule_match_enc_ip(rule, &match);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_ecn, key->tos & 0x3);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_ecn,
|
||||||
|
match.mask->tos & 0x3);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_ecn,
|
||||||
|
match.key->tos & 0x3);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_dscp, mask->tos >> 2);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_dscp,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_dscp, key->tos >> 2);
|
match.mask->tos >> 2);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_dscp,
|
||||||
|
match.key->tos >> 2);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ttl_hoplimit, mask->ttl);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ttl_hoplimit,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ttl_hoplimit, key->ttl);
|
match.mask->ttl);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ttl_hoplimit,
|
||||||
|
match.key->ttl);
|
||||||
|
|
||||||
if (mask->ttl &&
|
if (match.mask->ttl &&
|
||||||
!MLX5_CAP_ESW_FLOWTABLE_FDB
|
!MLX5_CAP_ESW_FLOWTABLE_FDB
|
||||||
(priv->mdev,
|
(priv->mdev,
|
||||||
ft_field_support.outer_ipv4_ttl)) {
|
ft_field_support.outer_ipv4_ttl)) {
|
||||||
@ -1437,12 +1425,14 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
misc_parameters);
|
misc_parameters);
|
||||||
void *misc_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
void *misc_v = MLX5_ADDR_OF(fte_match_param, spec->match_value,
|
||||||
misc_parameters);
|
misc_parameters);
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
u16 addr_type = 0;
|
u16 addr_type = 0;
|
||||||
u8 ip_proto = 0;
|
u8 ip_proto = 0;
|
||||||
|
|
||||||
*match_level = MLX5_MATCH_NONE;
|
*match_level = MLX5_MATCH_NONE;
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
||||||
@ -1461,20 +1451,18 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
BIT(FLOW_DISSECTOR_KEY_ENC_IP))) {
|
BIT(FLOW_DISSECTOR_KEY_ENC_IP))) {
|
||||||
NL_SET_ERR_MSG_MOD(extack, "Unsupported key");
|
NL_SET_ERR_MSG_MOD(extack, "Unsupported key");
|
||||||
netdev_warn(priv->netdev, "Unsupported key used: 0x%x\n",
|
netdev_warn(priv->netdev, "Unsupported key used: 0x%x\n",
|
||||||
f->dissector->used_keys);
|
dissector->used_keys);
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((dissector_uses_key(f->dissector,
|
if ((flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS) ||
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS) ||
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID) ||
|
||||||
dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_KEYID) ||
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_PORTS)) &&
|
||||||
dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_PORTS)) &&
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_CONTROL)) {
|
||||||
dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ENC_CONTROL)) {
|
struct flow_match_control match;
|
||||||
struct flow_dissector_key_control *key =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
flow_rule_match_enc_control(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ENC_CONTROL,
|
switch (match.key->addr_type) {
|
||||||
f->key);
|
|
||||||
switch (key->addr_type) {
|
|
||||||
case FLOW_DISSECTOR_KEY_IPV4_ADDRS:
|
case FLOW_DISSECTOR_KEY_IPV4_ADDRS:
|
||||||
case FLOW_DISSECTOR_KEY_IPV6_ADDRS:
|
case FLOW_DISSECTOR_KEY_IPV6_ADDRS:
|
||||||
if (parse_tunnel_attr(priv, spec, f, filter_dev))
|
if (parse_tunnel_attr(priv, spec, f, filter_dev))
|
||||||
@ -1493,35 +1481,27 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
inner_headers);
|
inner_headers);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_basic *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->mask);
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ethertype,
|
|
||||||
ntohs(mask->n_proto));
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype,
|
|
||||||
ntohs(key->n_proto));
|
|
||||||
|
|
||||||
if (mask->n_proto)
|
flow_rule_match_basic(rule, &match);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ethertype,
|
||||||
|
ntohs(match.mask->n_proto));
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ethertype,
|
||||||
|
ntohs(match.key->n_proto));
|
||||||
|
|
||||||
|
if (match.mask->n_proto)
|
||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
flow_rule_match_vlan(rule, &match);
|
||||||
f->key);
|
if (match.mask->vlan_id ||
|
||||||
struct flow_dissector_key_vlan *mask =
|
match.mask->vlan_priority ||
|
||||||
skb_flow_dissector_target(f->dissector,
|
match.mask->vlan_tpid) {
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
if (match.key->vlan_tpid == htons(ETH_P_8021AD)) {
|
||||||
f->mask);
|
|
||||||
if (mask->vlan_id || mask->vlan_priority || mask->vlan_tpid) {
|
|
||||||
if (key->vlan_tpid == htons(ETH_P_8021AD)) {
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
svlan_tag, 1);
|
svlan_tag, 1);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
@ -1533,11 +1513,15 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
cvlan_tag, 1);
|
cvlan_tag, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, first_vid, mask->vlan_id);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, first_vid,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, first_vid, key->vlan_id);
|
match.mask->vlan_id);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, first_vid,
|
||||||
|
match.key->vlan_id);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, first_prio, mask->vlan_priority);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, first_prio,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, first_prio, key->vlan_priority);
|
match.mask->vlan_priority);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, first_prio,
|
||||||
|
match.key->vlan_priority);
|
||||||
|
|
||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
}
|
}
|
||||||
@ -1547,17 +1531,14 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_CVLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CVLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CVLAN,
|
flow_rule_match_vlan(rule, &match);
|
||||||
f->key);
|
if (match.mask->vlan_id ||
|
||||||
struct flow_dissector_key_vlan *mask =
|
match.mask->vlan_priority ||
|
||||||
skb_flow_dissector_target(f->dissector,
|
match.mask->vlan_tpid) {
|
||||||
FLOW_DISSECTOR_KEY_CVLAN,
|
if (match.key->vlan_tpid == htons(ETH_P_8021AD)) {
|
||||||
f->mask);
|
|
||||||
if (mask->vlan_id || mask->vlan_priority || mask->vlan_tpid) {
|
|
||||||
if (key->vlan_tpid == htons(ETH_P_8021AD)) {
|
|
||||||
MLX5_SET(fte_match_set_misc, misc_c,
|
MLX5_SET(fte_match_set_misc, misc_c,
|
||||||
outer_second_svlan_tag, 1);
|
outer_second_svlan_tag, 1);
|
||||||
MLX5_SET(fte_match_set_misc, misc_v,
|
MLX5_SET(fte_match_set_misc, misc_v,
|
||||||
@ -1570,69 +1551,58 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
}
|
}
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_misc, misc_c, outer_second_vid,
|
MLX5_SET(fte_match_set_misc, misc_c, outer_second_vid,
|
||||||
mask->vlan_id);
|
match.mask->vlan_id);
|
||||||
MLX5_SET(fte_match_set_misc, misc_v, outer_second_vid,
|
MLX5_SET(fte_match_set_misc, misc_v, outer_second_vid,
|
||||||
key->vlan_id);
|
match.key->vlan_id);
|
||||||
MLX5_SET(fte_match_set_misc, misc_c, outer_second_prio,
|
MLX5_SET(fte_match_set_misc, misc_c, outer_second_prio,
|
||||||
mask->vlan_priority);
|
match.mask->vlan_priority);
|
||||||
MLX5_SET(fte_match_set_misc, misc_v, outer_second_prio,
|
MLX5_SET(fte_match_set_misc, misc_v, outer_second_prio,
|
||||||
key->vlan_priority);
|
match.key->vlan_priority);
|
||||||
|
|
||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key =
|
struct flow_match_eth_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_eth_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
dmac_47_16),
|
dmac_47_16),
|
||||||
mask->dst);
|
match.mask->dst);
|
||||||
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
dmac_47_16),
|
dmac_47_16),
|
||||||
key->dst);
|
match.key->dst);
|
||||||
|
|
||||||
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
smac_47_16),
|
smac_47_16),
|
||||||
mask->src);
|
match.mask->src);
|
||||||
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
ether_addr_copy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
smac_47_16),
|
smac_47_16),
|
||||||
key->src);
|
match.key->src);
|
||||||
|
|
||||||
if (!is_zero_ether_addr(mask->src) || !is_zero_ether_addr(mask->dst))
|
if (!is_zero_ether_addr(match.mask->src) ||
|
||||||
|
!is_zero_ether_addr(match.mask->dst))
|
||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key =
|
struct flow_match_control match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
|
||||||
f->key);
|
|
||||||
|
|
||||||
struct flow_dissector_key_control *mask =
|
flow_rule_match_control(rule, &match);
|
||||||
skb_flow_dissector_target(f->dissector,
|
addr_type = match.key->addr_type;
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
|
||||||
f->mask);
|
|
||||||
addr_type = key->addr_type;
|
|
||||||
|
|
||||||
/* the HW doesn't support frag first/later */
|
/* the HW doesn't support frag first/later */
|
||||||
if (mask->flags & FLOW_DIS_FIRST_FRAG)
|
if (match.mask->flags & FLOW_DIS_FIRST_FRAG)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (mask->flags & FLOW_DIS_IS_FRAGMENT) {
|
if (match.mask->flags & FLOW_DIS_IS_FRAGMENT) {
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, frag, 1);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, frag, 1);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, frag,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, frag,
|
||||||
key->flags & FLOW_DIS_IS_FRAGMENT);
|
match.key->flags & FLOW_DIS_IS_FRAGMENT);
|
||||||
|
|
||||||
/* the HW doesn't need L3 inline to match on frag=no */
|
/* the HW doesn't need L3 inline to match on frag=no */
|
||||||
if (!(key->flags & FLOW_DIS_IS_FRAGMENT))
|
if (!(match.key->flags & FLOW_DIS_IS_FRAGMENT))
|
||||||
*match_level = MLX5_MATCH_L2;
|
*match_level = MLX5_MATCH_L2;
|
||||||
/* *** L2 attributes parsing up to here *** */
|
/* *** L2 attributes parsing up to here *** */
|
||||||
else
|
else
|
||||||
@ -1640,102 +1610,85 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
flow_rule_match_basic(rule, &match);
|
||||||
f->key);
|
ip_proto = match.key->ip_proto;
|
||||||
struct flow_dissector_key_basic *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->mask);
|
|
||||||
ip_proto = key->ip_proto;
|
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_protocol,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_protocol,
|
||||||
mask->ip_proto);
|
match.mask->ip_proto);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_protocol,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_protocol,
|
||||||
key->ip_proto);
|
match.key->ip_proto);
|
||||||
|
|
||||||
if (mask->ip_proto)
|
if (match.mask->ip_proto)
|
||||||
*match_level = MLX5_MATCH_L3;
|
*match_level = MLX5_MATCH_L3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS) {
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
src_ipv4_src_ipv6.ipv4_layout.ipv4),
|
src_ipv4_src_ipv6.ipv4_layout.ipv4),
|
||||||
&mask->src, sizeof(mask->src));
|
&match.mask->src, sizeof(match.mask->src));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
src_ipv4_src_ipv6.ipv4_layout.ipv4),
|
src_ipv4_src_ipv6.ipv4_layout.ipv4),
|
||||||
&key->src, sizeof(key->src));
|
&match.key->src, sizeof(match.key->src));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
dst_ipv4_dst_ipv6.ipv4_layout.ipv4),
|
dst_ipv4_dst_ipv6.ipv4_layout.ipv4),
|
||||||
&mask->dst, sizeof(mask->dst));
|
&match.mask->dst, sizeof(match.mask->dst));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
dst_ipv4_dst_ipv6.ipv4_layout.ipv4),
|
dst_ipv4_dst_ipv6.ipv4_layout.ipv4),
|
||||||
&key->dst, sizeof(key->dst));
|
&match.key->dst, sizeof(match.key->dst));
|
||||||
|
|
||||||
if (mask->src || mask->dst)
|
if (match.mask->src || match.mask->dst)
|
||||||
*match_level = MLX5_MATCH_L3;
|
*match_level = MLX5_MATCH_L3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV6_ADDRS) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
||||||
&mask->src, sizeof(mask->src));
|
&match.mask->src, sizeof(match.mask->src));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
src_ipv4_src_ipv6.ipv6_layout.ipv6),
|
||||||
&key->src, sizeof(key->src));
|
&match.key->src, sizeof(match.key->src));
|
||||||
|
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_c,
|
||||||
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
||||||
&mask->dst, sizeof(mask->dst));
|
&match.mask->dst, sizeof(match.mask->dst));
|
||||||
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
memcpy(MLX5_ADDR_OF(fte_match_set_lyr_2_4, headers_v,
|
||||||
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
dst_ipv4_dst_ipv6.ipv6_layout.ipv6),
|
||||||
&key->dst, sizeof(key->dst));
|
&match.key->dst, sizeof(match.key->dst));
|
||||||
|
|
||||||
if (ipv6_addr_type(&mask->src) != IPV6_ADDR_ANY ||
|
if (ipv6_addr_type(&match.mask->src) != IPV6_ADDR_ANY ||
|
||||||
ipv6_addr_type(&mask->dst) != IPV6_ADDR_ANY)
|
ipv6_addr_type(&match.mask->dst) != IPV6_ADDR_ANY)
|
||||||
*match_level = MLX5_MATCH_L3;
|
*match_level = MLX5_MATCH_L3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IP)) {
|
||||||
struct flow_dissector_key_ip *key =
|
struct flow_match_ip match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ip *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_ecn, mask->tos & 0x3);
|
flow_rule_match_ip(rule, &match);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_ecn, key->tos & 0x3);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_ecn,
|
||||||
|
match.mask->tos & 0x3);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_ecn,
|
||||||
|
match.key->tos & 0x3);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_dscp, mask->tos >> 2);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ip_dscp,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_dscp, key->tos >> 2);
|
match.mask->tos >> 2);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ip_dscp,
|
||||||
|
match.key->tos >> 2);
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ttl_hoplimit, mask->ttl);
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, ttl_hoplimit,
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ttl_hoplimit, key->ttl);
|
match.mask->ttl);
|
||||||
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, ttl_hoplimit,
|
||||||
|
match.key->ttl);
|
||||||
|
|
||||||
if (mask->ttl &&
|
if (match.mask->ttl &&
|
||||||
!MLX5_CAP_ESW_FLOWTABLE_FDB(priv->mdev,
|
!MLX5_CAP_ESW_FLOWTABLE_FDB(priv->mdev,
|
||||||
ft_field_support.outer_ipv4_ttl)) {
|
ft_field_support.outer_ipv4_ttl)) {
|
||||||
NL_SET_ERR_MSG_MOD(extack,
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
@ -1743,44 +1696,39 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->tos || mask->ttl)
|
if (match.mask->tos || match.mask->ttl)
|
||||||
*match_level = MLX5_MATCH_L3;
|
*match_level = MLX5_MATCH_L3;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* *** L3 attributes parsing up to here *** */
|
/* *** L3 attributes parsing up to here *** */
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
struct flow_dissector_key_ports *key =
|
struct flow_match_ports match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
flow_rule_match_ports(rule, &match);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ports *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->mask);
|
|
||||||
switch (ip_proto) {
|
switch (ip_proto) {
|
||||||
case IPPROTO_TCP:
|
case IPPROTO_TCP:
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
tcp_sport, ntohs(mask->src));
|
tcp_sport, ntohs(match.mask->src));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
tcp_sport, ntohs(key->src));
|
tcp_sport, ntohs(match.key->src));
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
tcp_dport, ntohs(mask->dst));
|
tcp_dport, ntohs(match.mask->dst));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
tcp_dport, ntohs(key->dst));
|
tcp_dport, ntohs(match.key->dst));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case IPPROTO_UDP:
|
case IPPROTO_UDP:
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
udp_sport, ntohs(mask->src));
|
udp_sport, ntohs(match.mask->src));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
udp_sport, ntohs(key->src));
|
udp_sport, ntohs(match.key->src));
|
||||||
|
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c,
|
||||||
udp_dport, ntohs(mask->dst));
|
udp_dport, ntohs(match.mask->dst));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v,
|
||||||
udp_dport, ntohs(key->dst));
|
udp_dport, ntohs(match.key->dst));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
NL_SET_ERR_MSG_MOD(extack,
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
@ -1790,26 +1738,20 @@ static int __parse_cls_flower(struct mlx5e_priv *priv,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask->src || mask->dst)
|
if (match.mask->src || match.mask->dst)
|
||||||
*match_level = MLX5_MATCH_L4;
|
*match_level = MLX5_MATCH_L4;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_TCP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_TCP)) {
|
||||||
struct flow_dissector_key_tcp *key =
|
struct flow_match_tcp match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_TCP,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_tcp *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_TCP,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_tcp(rule, &match);
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_c, tcp_flags,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_c, tcp_flags,
|
||||||
ntohs(mask->flags));
|
ntohs(match.mask->flags));
|
||||||
MLX5_SET(fte_match_set_lyr_2_4, headers_v, tcp_flags,
|
MLX5_SET(fte_match_set_lyr_2_4, headers_v, tcp_flags,
|
||||||
ntohs(key->flags));
|
ntohs(match.key->flags));
|
||||||
|
|
||||||
if (mask->flags)
|
if (match.mask->flags)
|
||||||
*match_level = MLX5_MATCH_L4;
|
*match_level = MLX5_MATCH_L4;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -113,59 +113,49 @@ static int mlxsw_sp_flower_parse_actions(struct mlxsw_sp *mlxsw_sp,
|
|||||||
static void mlxsw_sp_flower_parse_ipv4(struct mlxsw_sp_acl_rule_info *rulei,
|
static void mlxsw_sp_flower_parse_ipv4(struct mlxsw_sp_acl_rule_info *rulei,
|
||||||
struct tc_cls_flower_offload *f)
|
struct tc_cls_flower_offload *f)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_ipv4_addrs *key =
|
struct flow_match_ipv4_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
flow_rule_match_ipv4_addrs(f->rule, &match);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv4_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_0_31,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_0_31,
|
||||||
(char *) &key->src,
|
(char *) &match.key->src,
|
||||||
(char *) &mask->src, 4);
|
(char *) &match.mask->src, 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_0_31,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_0_31,
|
||||||
(char *) &key->dst,
|
(char *) &match.key->dst,
|
||||||
(char *) &mask->dst, 4);
|
(char *) &match.mask->dst, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mlxsw_sp_flower_parse_ipv6(struct mlxsw_sp_acl_rule_info *rulei,
|
static void mlxsw_sp_flower_parse_ipv6(struct mlxsw_sp_acl_rule_info *rulei,
|
||||||
struct tc_cls_flower_offload *f)
|
struct tc_cls_flower_offload *f)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_ipv6_addrs *key =
|
struct flow_match_ipv6_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
flow_rule_match_ipv6_addrs(f->rule, &match);
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_ipv6_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_96_127,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_96_127,
|
||||||
&key->src.s6_addr[0x0],
|
&match.key->src.s6_addr[0x0],
|
||||||
&mask->src.s6_addr[0x0], 4);
|
&match.mask->src.s6_addr[0x0], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_64_95,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_64_95,
|
||||||
&key->src.s6_addr[0x4],
|
&match.key->src.s6_addr[0x4],
|
||||||
&mask->src.s6_addr[0x4], 4);
|
&match.mask->src.s6_addr[0x4], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_32_63,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_32_63,
|
||||||
&key->src.s6_addr[0x8],
|
&match.key->src.s6_addr[0x8],
|
||||||
&mask->src.s6_addr[0x8], 4);
|
&match.mask->src.s6_addr[0x8], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_0_31,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_SRC_IP_0_31,
|
||||||
&key->src.s6_addr[0xC],
|
&match.key->src.s6_addr[0xC],
|
||||||
&mask->src.s6_addr[0xC], 4);
|
&match.mask->src.s6_addr[0xC], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_96_127,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_96_127,
|
||||||
&key->dst.s6_addr[0x0],
|
&match.key->dst.s6_addr[0x0],
|
||||||
&mask->dst.s6_addr[0x0], 4);
|
&match.mask->dst.s6_addr[0x0], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_64_95,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_64_95,
|
||||||
&key->dst.s6_addr[0x4],
|
&match.key->dst.s6_addr[0x4],
|
||||||
&mask->dst.s6_addr[0x4], 4);
|
&match.mask->dst.s6_addr[0x4], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_32_63,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_32_63,
|
||||||
&key->dst.s6_addr[0x8],
|
&match.key->dst.s6_addr[0x8],
|
||||||
&mask->dst.s6_addr[0x8], 4);
|
&match.mask->dst.s6_addr[0x8], 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_0_31,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei, MLXSW_AFK_ELEMENT_DST_IP_0_31,
|
||||||
&key->dst.s6_addr[0xC],
|
&match.key->dst.s6_addr[0xC],
|
||||||
&mask->dst.s6_addr[0xC], 4);
|
&match.mask->dst.s6_addr[0xC], 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlxsw_sp_flower_parse_ports(struct mlxsw_sp *mlxsw_sp,
|
static int mlxsw_sp_flower_parse_ports(struct mlxsw_sp *mlxsw_sp,
|
||||||
@ -173,9 +163,10 @@ static int mlxsw_sp_flower_parse_ports(struct mlxsw_sp *mlxsw_sp,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
u8 ip_proto)
|
u8 ip_proto)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_ports *key, *mask;
|
const struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_match_ports match;
|
||||||
|
|
||||||
if (!dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_PORTS))
|
if (!flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (ip_proto != IPPROTO_TCP && ip_proto != IPPROTO_UDP) {
|
if (ip_proto != IPPROTO_TCP && ip_proto != IPPROTO_UDP) {
|
||||||
@ -184,16 +175,13 @@ static int mlxsw_sp_flower_parse_ports(struct mlxsw_sp *mlxsw_sp,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_ports(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->key);
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->mask);
|
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_DST_L4_PORT,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_DST_L4_PORT,
|
||||||
ntohs(key->dst), ntohs(mask->dst));
|
ntohs(match.key->dst),
|
||||||
|
ntohs(match.mask->dst));
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_SRC_L4_PORT,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_SRC_L4_PORT,
|
||||||
ntohs(key->src), ntohs(mask->src));
|
ntohs(match.key->src),
|
||||||
|
ntohs(match.mask->src));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -202,9 +190,10 @@ static int mlxsw_sp_flower_parse_tcp(struct mlxsw_sp *mlxsw_sp,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
u8 ip_proto)
|
u8 ip_proto)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_tcp *key, *mask;
|
const struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_match_tcp match;
|
||||||
|
|
||||||
if (!dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_TCP))
|
if (!flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_TCP))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (ip_proto != IPPROTO_TCP) {
|
if (ip_proto != IPPROTO_TCP) {
|
||||||
@ -213,14 +202,11 @@ static int mlxsw_sp_flower_parse_tcp(struct mlxsw_sp *mlxsw_sp,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_tcp(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_TCP,
|
|
||||||
f->key);
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_TCP,
|
|
||||||
f->mask);
|
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_TCP_FLAGS,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_TCP_FLAGS,
|
||||||
ntohs(key->flags), ntohs(mask->flags));
|
ntohs(match.key->flags),
|
||||||
|
ntohs(match.mask->flags));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -229,9 +215,10 @@ static int mlxsw_sp_flower_parse_ip(struct mlxsw_sp *mlxsw_sp,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
u16 n_proto)
|
u16 n_proto)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_ip *key, *mask;
|
const struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_match_ip match;
|
||||||
|
|
||||||
if (!dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_IP))
|
if (!flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IP))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (n_proto != ETH_P_IP && n_proto != ETH_P_IPV6) {
|
if (n_proto != ETH_P_IP && n_proto != ETH_P_IPV6) {
|
||||||
@ -240,20 +227,18 @@ static int mlxsw_sp_flower_parse_ip(struct mlxsw_sp *mlxsw_sp,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_ip(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
|
||||||
f->key);
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
|
||||||
f->mask);
|
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_TTL_,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_TTL_,
|
||||||
key->ttl, mask->ttl);
|
match.key->ttl, match.mask->ttl);
|
||||||
|
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_ECN,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_ECN,
|
||||||
key->tos & 0x3, mask->tos & 0x3);
|
match.key->tos & 0x3,
|
||||||
|
match.mask->tos & 0x3);
|
||||||
|
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_DSCP,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei, MLXSW_AFK_ELEMENT_IP_DSCP,
|
||||||
key->tos >> 6, mask->tos >> 6);
|
match.key->tos >> 6,
|
||||||
|
match.mask->tos >> 6);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -263,13 +248,15 @@ static int mlxsw_sp_flower_parse(struct mlxsw_sp *mlxsw_sp,
|
|||||||
struct mlxsw_sp_acl_rule_info *rulei,
|
struct mlxsw_sp_acl_rule_info *rulei,
|
||||||
struct tc_cls_flower_offload *f)
|
struct tc_cls_flower_offload *f)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
u16 n_proto_mask = 0;
|
u16 n_proto_mask = 0;
|
||||||
u16 n_proto_key = 0;
|
u16 n_proto_key = 0;
|
||||||
u16 addr_type = 0;
|
u16 addr_type = 0;
|
||||||
u8 ip_proto = 0;
|
u8 ip_proto = 0;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_ETH_ADDRS) |
|
||||||
@ -286,25 +273,19 @@ static int mlxsw_sp_flower_parse(struct mlxsw_sp *mlxsw_sp,
|
|||||||
|
|
||||||
mlxsw_sp_acl_rulei_priority(rulei, f->common.prio);
|
mlxsw_sp_acl_rulei_priority(rulei, f->common.prio);
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key =
|
struct flow_match_control match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
flow_rule_match_control(rule, &match);
|
||||||
f->key);
|
addr_type = match.key->addr_type;
|
||||||
addr_type = key->addr_type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key =
|
struct flow_match_basic match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
flow_rule_match_basic(rule, &match);
|
||||||
f->key);
|
n_proto_key = ntohs(match.key->n_proto);
|
||||||
struct flow_dissector_key_basic *mask =
|
n_proto_mask = ntohs(match.mask->n_proto);
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
f->mask);
|
|
||||||
n_proto_key = ntohs(key->n_proto);
|
|
||||||
n_proto_mask = ntohs(mask->n_proto);
|
|
||||||
|
|
||||||
if (n_proto_key == ETH_P_ALL) {
|
if (n_proto_key == ETH_P_ALL) {
|
||||||
n_proto_key = 0;
|
n_proto_key = 0;
|
||||||
@ -314,60 +295,53 @@ static int mlxsw_sp_flower_parse(struct mlxsw_sp *mlxsw_sp,
|
|||||||
MLXSW_AFK_ELEMENT_ETHERTYPE,
|
MLXSW_AFK_ELEMENT_ETHERTYPE,
|
||||||
n_proto_key, n_proto_mask);
|
n_proto_key, n_proto_mask);
|
||||||
|
|
||||||
ip_proto = key->ip_proto;
|
ip_proto = match.key->ip_proto;
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
||||||
MLXSW_AFK_ELEMENT_IP_PROTO,
|
MLXSW_AFK_ELEMENT_IP_PROTO,
|
||||||
key->ip_proto, mask->ip_proto);
|
match.key->ip_proto,
|
||||||
|
match.mask->ip_proto);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
struct flow_dissector_key_eth_addrs *key =
|
struct flow_match_eth_addrs match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_eth_addrs *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
||||||
MLXSW_AFK_ELEMENT_DMAC_32_47,
|
MLXSW_AFK_ELEMENT_DMAC_32_47,
|
||||||
key->dst, mask->dst, 2);
|
match.key->dst,
|
||||||
|
match.mask->dst, 2);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
||||||
MLXSW_AFK_ELEMENT_DMAC_0_31,
|
MLXSW_AFK_ELEMENT_DMAC_0_31,
|
||||||
key->dst + 2, mask->dst + 2, 4);
|
match.key->dst + 2,
|
||||||
|
match.mask->dst + 2, 4);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
||||||
MLXSW_AFK_ELEMENT_SMAC_32_47,
|
MLXSW_AFK_ELEMENT_SMAC_32_47,
|
||||||
key->src, mask->src, 2);
|
match.key->src,
|
||||||
|
match.mask->src, 2);
|
||||||
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
mlxsw_sp_acl_rulei_keymask_buf(rulei,
|
||||||
MLXSW_AFK_ELEMENT_SMAC_0_31,
|
MLXSW_AFK_ELEMENT_SMAC_0_31,
|
||||||
key->src + 2, mask->src + 2, 4);
|
match.key->src + 2,
|
||||||
|
match.mask->src + 2, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *key =
|
struct flow_match_vlan match;
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->key);
|
|
||||||
struct flow_dissector_key_vlan *mask =
|
|
||||||
skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
|
flow_rule_match_vlan(rule, &match);
|
||||||
if (mlxsw_sp_acl_block_is_egress_bound(block)) {
|
if (mlxsw_sp_acl_block_is_egress_bound(block)) {
|
||||||
NL_SET_ERR_MSG_MOD(f->common.extack, "vlan_id key is not supported on egress");
|
NL_SET_ERR_MSG_MOD(f->common.extack, "vlan_id key is not supported on egress");
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
if (mask->vlan_id != 0)
|
if (match.mask->vlan_id != 0)
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
||||||
MLXSW_AFK_ELEMENT_VID,
|
MLXSW_AFK_ELEMENT_VID,
|
||||||
key->vlan_id,
|
match.key->vlan_id,
|
||||||
mask->vlan_id);
|
match.mask->vlan_id);
|
||||||
if (mask->vlan_priority != 0)
|
if (match.mask->vlan_priority != 0)
|
||||||
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
mlxsw_sp_acl_rulei_keymask_u32(rulei,
|
||||||
MLXSW_AFK_ELEMENT_PCP,
|
MLXSW_AFK_ELEMENT_PCP,
|
||||||
key->vlan_priority,
|
match.key->vlan_priority,
|
||||||
mask->vlan_priority);
|
match.mask->vlan_priority);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS)
|
if (addr_type == FLOW_DISSECTOR_KEY_IPV4_ADDRS)
|
||||||
|
@ -587,6 +587,7 @@ static int
|
|||||||
nfp_fl_pedit(const struct tc_action *action, struct tc_cls_flower_offload *flow,
|
nfp_fl_pedit(const struct tc_action *action, struct tc_cls_flower_offload *flow,
|
||||||
char *nfp_action, int *a_len, u32 *csum_updated)
|
char *nfp_action, int *a_len, u32 *csum_updated)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct nfp_fl_set_ipv6_addr set_ip6_dst, set_ip6_src;
|
struct nfp_fl_set_ipv6_addr set_ip6_dst, set_ip6_src;
|
||||||
struct nfp_fl_set_ipv6_tc_hl_fl set_ip6_tc_hl_fl;
|
struct nfp_fl_set_ipv6_tc_hl_fl set_ip6_tc_hl_fl;
|
||||||
struct nfp_fl_set_ip4_ttl_tos set_ip_ttl_tos;
|
struct nfp_fl_set_ip4_ttl_tos set_ip_ttl_tos;
|
||||||
@ -643,13 +644,11 @@ nfp_fl_pedit(const struct tc_action *action, struct tc_cls_flower_offload *flow,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *basic;
|
struct flow_match_basic match;
|
||||||
|
|
||||||
basic = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_basic(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
ip_proto = match.key->ip_proto;
|
||||||
flow->key);
|
|
||||||
ip_proto = basic->ip_proto;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (set_eth.head.len_lw) {
|
if (set_eth.head.len_lw) {
|
||||||
|
@ -8,31 +8,41 @@
|
|||||||
#include "main.h"
|
#include "main.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_meta_tci(struct nfp_flower_meta_tci *frame,
|
nfp_flower_compile_meta_tci(struct nfp_flower_meta_tci *ext,
|
||||||
struct tc_cls_flower_offload *flow, u8 key_type,
|
struct nfp_flower_meta_tci *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow, u8 key_type)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_vlan *flow_vlan;
|
|
||||||
u16 tmp_tci;
|
u16 tmp_tci;
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_meta_tci));
|
memset(ext, 0, sizeof(struct nfp_flower_meta_tci));
|
||||||
/* Populate the metadata frame. */
|
memset(msk, 0, sizeof(struct nfp_flower_meta_tci));
|
||||||
frame->nfp_flow_key_layer = key_type;
|
|
||||||
frame->mask_id = ~0;
|
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
/* Populate the metadata frame. */
|
||||||
flow_vlan = skb_flow_dissector_target(flow->dissector,
|
ext->nfp_flow_key_layer = key_type;
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
ext->mask_id = ~0;
|
||||||
target);
|
|
||||||
|
msk->nfp_flow_key_layer = key_type;
|
||||||
|
msk->mask_id = ~0;
|
||||||
|
|
||||||
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
|
struct flow_match_vlan match;
|
||||||
|
|
||||||
|
flow_rule_match_vlan(rule, &match);
|
||||||
/* Populate the tci field. */
|
/* Populate the tci field. */
|
||||||
if (flow_vlan->vlan_id || flow_vlan->vlan_priority) {
|
if (match.key->vlan_id || match.key->vlan_priority) {
|
||||||
tmp_tci = FIELD_PREP(NFP_FLOWER_MASK_VLAN_PRIO,
|
tmp_tci = FIELD_PREP(NFP_FLOWER_MASK_VLAN_PRIO,
|
||||||
flow_vlan->vlan_priority) |
|
match.key->vlan_priority) |
|
||||||
FIELD_PREP(NFP_FLOWER_MASK_VLAN_VID,
|
FIELD_PREP(NFP_FLOWER_MASK_VLAN_VID,
|
||||||
flow_vlan->vlan_id) |
|
match.key->vlan_id) |
|
||||||
NFP_FLOWER_MASK_VLAN_CFI;
|
NFP_FLOWER_MASK_VLAN_CFI;
|
||||||
frame->tci = cpu_to_be16(tmp_tci);
|
ext->tci = cpu_to_be16(tmp_tci);
|
||||||
|
tmp_tci = FIELD_PREP(NFP_FLOWER_MASK_VLAN_PRIO,
|
||||||
|
match.mask->vlan_priority) |
|
||||||
|
FIELD_PREP(NFP_FLOWER_MASK_VLAN_VID,
|
||||||
|
match.mask->vlan_id) |
|
||||||
|
NFP_FLOWER_MASK_VLAN_CFI;
|
||||||
|
msk->tci = cpu_to_be16(tmp_tci);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -64,231 +74,244 @@ nfp_flower_compile_port(struct nfp_flower_in_port *frame, u32 cmsg_port,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_mac(struct nfp_flower_mac_mpls *frame,
|
nfp_flower_compile_mac(struct nfp_flower_mac_mpls *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_mac_mpls *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_eth_addrs *addr;
|
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_mac_mpls));
|
memset(ext, 0, sizeof(struct nfp_flower_mac_mpls));
|
||||||
|
memset(msk, 0, sizeof(struct nfp_flower_mac_mpls));
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS)) {
|
||||||
addr = skb_flow_dissector_target(flow->dissector,
|
struct flow_match_eth_addrs match;
|
||||||
FLOW_DISSECTOR_KEY_ETH_ADDRS,
|
|
||||||
target);
|
flow_rule_match_eth_addrs(rule, &match);
|
||||||
/* Populate mac frame. */
|
/* Populate mac frame. */
|
||||||
ether_addr_copy(frame->mac_dst, &addr->dst[0]);
|
ether_addr_copy(ext->mac_dst, &match.key->dst[0]);
|
||||||
ether_addr_copy(frame->mac_src, &addr->src[0]);
|
ether_addr_copy(ext->mac_src, &match.key->src[0]);
|
||||||
|
ether_addr_copy(msk->mac_dst, &match.mask->dst[0]);
|
||||||
|
ether_addr_copy(msk->mac_src, &match.mask->src[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_MPLS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_MPLS)) {
|
||||||
struct flow_dissector_key_mpls *mpls;
|
struct flow_match_mpls match;
|
||||||
u32 t_mpls;
|
u32 t_mpls;
|
||||||
|
|
||||||
mpls = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_mpls(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_MPLS,
|
t_mpls = FIELD_PREP(NFP_FLOWER_MASK_MPLS_LB, match.key->mpls_label) |
|
||||||
target);
|
FIELD_PREP(NFP_FLOWER_MASK_MPLS_TC, match.key->mpls_tc) |
|
||||||
|
FIELD_PREP(NFP_FLOWER_MASK_MPLS_BOS, match.key->mpls_bos) |
|
||||||
t_mpls = FIELD_PREP(NFP_FLOWER_MASK_MPLS_LB, mpls->mpls_label) |
|
|
||||||
FIELD_PREP(NFP_FLOWER_MASK_MPLS_TC, mpls->mpls_tc) |
|
|
||||||
FIELD_PREP(NFP_FLOWER_MASK_MPLS_BOS, mpls->mpls_bos) |
|
|
||||||
NFP_FLOWER_MASK_MPLS_Q;
|
NFP_FLOWER_MASK_MPLS_Q;
|
||||||
|
ext->mpls_lse = cpu_to_be32(t_mpls);
|
||||||
frame->mpls_lse = cpu_to_be32(t_mpls);
|
t_mpls = FIELD_PREP(NFP_FLOWER_MASK_MPLS_LB, match.mask->mpls_label) |
|
||||||
} else if (dissector_uses_key(flow->dissector,
|
FIELD_PREP(NFP_FLOWER_MASK_MPLS_TC, match.mask->mpls_tc) |
|
||||||
FLOW_DISSECTOR_KEY_BASIC)) {
|
FIELD_PREP(NFP_FLOWER_MASK_MPLS_BOS, match.mask->mpls_bos) |
|
||||||
|
NFP_FLOWER_MASK_MPLS_Q;
|
||||||
|
msk->mpls_lse = cpu_to_be32(t_mpls);
|
||||||
|
} else if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
/* Check for mpls ether type and set NFP_FLOWER_MASK_MPLS_Q
|
/* Check for mpls ether type and set NFP_FLOWER_MASK_MPLS_Q
|
||||||
* bit, which indicates an mpls ether type but without any
|
* bit, which indicates an mpls ether type but without any
|
||||||
* mpls fields.
|
* mpls fields.
|
||||||
*/
|
*/
|
||||||
struct flow_dissector_key_basic *key_basic;
|
struct flow_match_basic match;
|
||||||
|
|
||||||
key_basic = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_basic(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
if (match.key->n_proto == cpu_to_be16(ETH_P_MPLS_UC) ||
|
||||||
flow->key);
|
match.key->n_proto == cpu_to_be16(ETH_P_MPLS_MC)) {
|
||||||
if (key_basic->n_proto == cpu_to_be16(ETH_P_MPLS_UC) ||
|
ext->mpls_lse = cpu_to_be32(NFP_FLOWER_MASK_MPLS_Q);
|
||||||
key_basic->n_proto == cpu_to_be16(ETH_P_MPLS_MC))
|
msk->mpls_lse = cpu_to_be32(NFP_FLOWER_MASK_MPLS_Q);
|
||||||
frame->mpls_lse = cpu_to_be32(NFP_FLOWER_MASK_MPLS_Q);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_tport(struct nfp_flower_tp_ports *frame,
|
nfp_flower_compile_tport(struct nfp_flower_tp_ports *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_tp_ports *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_ports *tp;
|
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_tp_ports));
|
memset(ext, 0, sizeof(struct nfp_flower_tp_ports));
|
||||||
|
memset(msk, 0, sizeof(struct nfp_flower_tp_ports));
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
tp = skb_flow_dissector_target(flow->dissector,
|
struct flow_match_ports match;
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
target);
|
flow_rule_match_ports(rule, &match);
|
||||||
frame->port_src = tp->src;
|
ext->port_src = match.key->src;
|
||||||
frame->port_dst = tp->dst;
|
ext->port_dst = match.key->dst;
|
||||||
|
msk->port_src = match.mask->src;
|
||||||
|
msk->port_dst = match.mask->dst;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_ip_ext(struct nfp_flower_ip_ext *frame,
|
nfp_flower_compile_ip_ext(struct nfp_flower_ip_ext *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_ip_ext *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *basic;
|
struct flow_match_basic match;
|
||||||
|
|
||||||
basic = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_basic(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
ext->proto = match.key->ip_proto;
|
||||||
target);
|
msk->proto = match.mask->ip_proto;
|
||||||
frame->proto = basic->ip_proto;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IP)) {
|
||||||
struct flow_dissector_key_ip *flow_ip;
|
struct flow_match_ip match;
|
||||||
|
|
||||||
flow_ip = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_ip(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_IP,
|
ext->tos = match.key->tos;
|
||||||
target);
|
ext->ttl = match.key->ttl;
|
||||||
frame->tos = flow_ip->tos;
|
msk->tos = match.mask->tos;
|
||||||
frame->ttl = flow_ip->ttl;
|
msk->ttl = match.mask->ttl;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_TCP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_TCP)) {
|
||||||
struct flow_dissector_key_tcp *tcp;
|
struct flow_match_tcp match;
|
||||||
u32 tcp_flags;
|
u16 tcp_flags;
|
||||||
|
|
||||||
tcp = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_tcp(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_TCP, target);
|
tcp_flags = be16_to_cpu(match.key->flags);
|
||||||
tcp_flags = be16_to_cpu(tcp->flags);
|
|
||||||
|
|
||||||
if (tcp_flags & TCPHDR_FIN)
|
if (tcp_flags & TCPHDR_FIN) {
|
||||||
frame->flags |= NFP_FL_TCP_FLAG_FIN;
|
ext->flags |= NFP_FL_TCP_FLAG_FIN;
|
||||||
if (tcp_flags & TCPHDR_SYN)
|
msk->flags |= NFP_FL_TCP_FLAG_FIN;
|
||||||
frame->flags |= NFP_FL_TCP_FLAG_SYN;
|
}
|
||||||
if (tcp_flags & TCPHDR_RST)
|
if (tcp_flags & TCPHDR_SYN) {
|
||||||
frame->flags |= NFP_FL_TCP_FLAG_RST;
|
ext->flags |= NFP_FL_TCP_FLAG_SYN;
|
||||||
if (tcp_flags & TCPHDR_PSH)
|
msk->flags |= NFP_FL_TCP_FLAG_SYN;
|
||||||
frame->flags |= NFP_FL_TCP_FLAG_PSH;
|
}
|
||||||
if (tcp_flags & TCPHDR_URG)
|
if (tcp_flags & TCPHDR_RST) {
|
||||||
frame->flags |= NFP_FL_TCP_FLAG_URG;
|
ext->flags |= NFP_FL_TCP_FLAG_RST;
|
||||||
|
msk->flags |= NFP_FL_TCP_FLAG_RST;
|
||||||
|
}
|
||||||
|
if (tcp_flags & TCPHDR_PSH) {
|
||||||
|
ext->flags |= NFP_FL_TCP_FLAG_PSH;
|
||||||
|
msk->flags |= NFP_FL_TCP_FLAG_PSH;
|
||||||
|
}
|
||||||
|
if (tcp_flags & TCPHDR_URG) {
|
||||||
|
ext->flags |= NFP_FL_TCP_FLAG_URG;
|
||||||
|
msk->flags |= NFP_FL_TCP_FLAG_URG;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key;
|
struct flow_match_control match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_control(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
if (match.key->flags & FLOW_DIS_IS_FRAGMENT) {
|
||||||
target);
|
ext->flags |= NFP_FL_IP_FRAGMENTED;
|
||||||
if (key->flags & FLOW_DIS_IS_FRAGMENT)
|
msk->flags |= NFP_FL_IP_FRAGMENTED;
|
||||||
frame->flags |= NFP_FL_IP_FRAGMENTED;
|
}
|
||||||
if (key->flags & FLOW_DIS_FIRST_FRAG)
|
if (match.key->flags & FLOW_DIS_FIRST_FRAG) {
|
||||||
frame->flags |= NFP_FL_IP_FRAG_FIRST;
|
ext->flags |= NFP_FL_IP_FRAG_FIRST;
|
||||||
|
msk->flags |= NFP_FL_IP_FRAG_FIRST;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_ipv4(struct nfp_flower_ipv4 *frame,
|
nfp_flower_compile_ipv4(struct nfp_flower_ipv4 *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_ipv4 *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_ipv4_addrs *addr;
|
struct flow_match_ipv4_addrs match;
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_ipv4));
|
memset(ext, 0, sizeof(struct nfp_flower_ipv4));
|
||||||
|
memset(msk, 0, sizeof(struct nfp_flower_ipv4));
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
addr = skb_flow_dissector_target(flow->dissector,
|
ext->ipv4_src = match.key->src;
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
ext->ipv4_dst = match.key->dst;
|
||||||
target);
|
msk->ipv4_src = match.mask->src;
|
||||||
frame->ipv4_src = addr->src;
|
msk->ipv4_dst = match.mask->dst;
|
||||||
frame->ipv4_dst = addr->dst;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nfp_flower_compile_ip_ext(&frame->ip_ext, flow, mask_version);
|
nfp_flower_compile_ip_ext(&ext->ip_ext, &msk->ip_ext, flow);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_ipv6(struct nfp_flower_ipv6 *frame,
|
nfp_flower_compile_ipv6(struct nfp_flower_ipv6 *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_ipv6 *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_ipv6_addrs *addr;
|
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_ipv6));
|
memset(ext, 0, sizeof(struct nfp_flower_ipv6));
|
||||||
|
memset(msk, 0, sizeof(struct nfp_flower_ipv6));
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
struct flow_match_ipv6_addrs match;
|
||||||
addr = skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
target);
|
ext->ipv6_src = match.key->src;
|
||||||
frame->ipv6_src = addr->src;
|
ext->ipv6_dst = match.key->dst;
|
||||||
frame->ipv6_dst = addr->dst;
|
msk->ipv6_src = match.mask->src;
|
||||||
|
msk->ipv6_dst = match.mask->dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
nfp_flower_compile_ip_ext(&frame->ip_ext, flow, mask_version);
|
nfp_flower_compile_ip_ext(&ext->ip_ext, &msk->ip_ext, flow);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
nfp_flower_compile_geneve_opt(void *key_buf, struct tc_cls_flower_offload *flow,
|
nfp_flower_compile_geneve_opt(void *ext, void *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_match_enc_opts match;
|
||||||
struct flow_dissector_key_enc_opts *opts;
|
|
||||||
|
|
||||||
opts = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_enc_opts(flow->rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ENC_OPTS,
|
memcpy(ext, match.key->data, match.key->len);
|
||||||
target);
|
memcpy(msk, match.mask->data, match.mask->len);
|
||||||
memcpy(key_buf, opts->data, opts->len);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
nfp_flower_compile_ipv4_udp_tun(struct nfp_flower_ipv4_udp_tun *frame,
|
nfp_flower_compile_ipv4_udp_tun(struct nfp_flower_ipv4_udp_tun *ext,
|
||||||
struct tc_cls_flower_offload *flow,
|
struct nfp_flower_ipv4_udp_tun *msk,
|
||||||
bool mask_version)
|
struct tc_cls_flower_offload *flow)
|
||||||
{
|
{
|
||||||
struct fl_flow_key *target = mask_version ? flow->mask : flow->key;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_ipv4_addrs *tun_ips;
|
|
||||||
struct flow_dissector_key_keyid *vni;
|
|
||||||
struct flow_dissector_key_ip *ip;
|
|
||||||
|
|
||||||
memset(frame, 0, sizeof(struct nfp_flower_ipv4_udp_tun));
|
memset(ext, 0, sizeof(struct nfp_flower_ipv4_udp_tun));
|
||||||
|
memset(msk, 0, sizeof(struct nfp_flower_ipv4_udp_tun));
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID)) {
|
struct flow_match_enc_keyid match;
|
||||||
u32 temp_vni;
|
u32 temp_vni;
|
||||||
|
|
||||||
vni = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_enc_keyid(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ENC_KEYID,
|
temp_vni = be32_to_cpu(match.key->keyid) << NFP_FL_TUN_VNI_OFFSET;
|
||||||
target);
|
ext->tun_id = cpu_to_be32(temp_vni);
|
||||||
temp_vni = be32_to_cpu(vni->keyid) << NFP_FL_TUN_VNI_OFFSET;
|
temp_vni = be32_to_cpu(match.mask->keyid) << NFP_FL_TUN_VNI_OFFSET;
|
||||||
frame->tun_id = cpu_to_be32(temp_vni);
|
msk->tun_id = cpu_to_be32(temp_vni);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS)) {
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS)) {
|
struct flow_match_ipv4_addrs match;
|
||||||
tun_ips =
|
|
||||||
skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_enc_ipv4_addrs(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS,
|
ext->ip_src = match.key->src;
|
||||||
target);
|
ext->ip_dst = match.key->dst;
|
||||||
frame->ip_src = tun_ips->src;
|
msk->ip_src = match.mask->src;
|
||||||
frame->ip_dst = tun_ips->dst;
|
msk->ip_dst = match.mask->dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_ENC_IP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_IP)) {
|
||||||
ip = skb_flow_dissector_target(flow->dissector,
|
struct flow_match_ip match;
|
||||||
FLOW_DISSECTOR_KEY_ENC_IP,
|
|
||||||
target);
|
flow_rule_match_enc_ip(rule, &match);
|
||||||
frame->tos = ip->tos;
|
ext->tos = match.key->tos;
|
||||||
frame->ttl = ip->ttl;
|
ext->ttl = match.key->ttl;
|
||||||
|
msk->tos = match.mask->tos;
|
||||||
|
msk->ttl = match.mask->ttl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,12 +336,9 @@ int nfp_flower_compile_flow_match(struct nfp_app *app,
|
|||||||
ext = nfp_flow->unmasked_data;
|
ext = nfp_flow->unmasked_data;
|
||||||
msk = nfp_flow->mask_data;
|
msk = nfp_flow->mask_data;
|
||||||
|
|
||||||
/* Populate Exact Metadata. */
|
|
||||||
nfp_flower_compile_meta_tci((struct nfp_flower_meta_tci *)ext,
|
nfp_flower_compile_meta_tci((struct nfp_flower_meta_tci *)ext,
|
||||||
flow, key_ls->key_layer, false);
|
(struct nfp_flower_meta_tci *)msk,
|
||||||
/* Populate Mask Metadata. */
|
flow, key_ls->key_layer);
|
||||||
nfp_flower_compile_meta_tci((struct nfp_flower_meta_tci *)msk,
|
|
||||||
flow, key_ls->key_layer, true);
|
|
||||||
ext += sizeof(struct nfp_flower_meta_tci);
|
ext += sizeof(struct nfp_flower_meta_tci);
|
||||||
msk += sizeof(struct nfp_flower_meta_tci);
|
msk += sizeof(struct nfp_flower_meta_tci);
|
||||||
|
|
||||||
@ -348,45 +368,33 @@ int nfp_flower_compile_flow_match(struct nfp_app *app,
|
|||||||
msk += sizeof(struct nfp_flower_in_port);
|
msk += sizeof(struct nfp_flower_in_port);
|
||||||
|
|
||||||
if (NFP_FLOWER_LAYER_MAC & key_ls->key_layer) {
|
if (NFP_FLOWER_LAYER_MAC & key_ls->key_layer) {
|
||||||
/* Populate Exact MAC Data. */
|
|
||||||
nfp_flower_compile_mac((struct nfp_flower_mac_mpls *)ext,
|
nfp_flower_compile_mac((struct nfp_flower_mac_mpls *)ext,
|
||||||
flow, false);
|
(struct nfp_flower_mac_mpls *)msk,
|
||||||
/* Populate Mask MAC Data. */
|
flow);
|
||||||
nfp_flower_compile_mac((struct nfp_flower_mac_mpls *)msk,
|
|
||||||
flow, true);
|
|
||||||
ext += sizeof(struct nfp_flower_mac_mpls);
|
ext += sizeof(struct nfp_flower_mac_mpls);
|
||||||
msk += sizeof(struct nfp_flower_mac_mpls);
|
msk += sizeof(struct nfp_flower_mac_mpls);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (NFP_FLOWER_LAYER_TP & key_ls->key_layer) {
|
if (NFP_FLOWER_LAYER_TP & key_ls->key_layer) {
|
||||||
/* Populate Exact TP Data. */
|
|
||||||
nfp_flower_compile_tport((struct nfp_flower_tp_ports *)ext,
|
nfp_flower_compile_tport((struct nfp_flower_tp_ports *)ext,
|
||||||
flow, false);
|
(struct nfp_flower_tp_ports *)msk,
|
||||||
/* Populate Mask TP Data. */
|
flow);
|
||||||
nfp_flower_compile_tport((struct nfp_flower_tp_ports *)msk,
|
|
||||||
flow, true);
|
|
||||||
ext += sizeof(struct nfp_flower_tp_ports);
|
ext += sizeof(struct nfp_flower_tp_ports);
|
||||||
msk += sizeof(struct nfp_flower_tp_ports);
|
msk += sizeof(struct nfp_flower_tp_ports);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (NFP_FLOWER_LAYER_IPV4 & key_ls->key_layer) {
|
if (NFP_FLOWER_LAYER_IPV4 & key_ls->key_layer) {
|
||||||
/* Populate Exact IPv4 Data. */
|
|
||||||
nfp_flower_compile_ipv4((struct nfp_flower_ipv4 *)ext,
|
nfp_flower_compile_ipv4((struct nfp_flower_ipv4 *)ext,
|
||||||
flow, false);
|
(struct nfp_flower_ipv4 *)msk,
|
||||||
/* Populate Mask IPv4 Data. */
|
flow);
|
||||||
nfp_flower_compile_ipv4((struct nfp_flower_ipv4 *)msk,
|
|
||||||
flow, true);
|
|
||||||
ext += sizeof(struct nfp_flower_ipv4);
|
ext += sizeof(struct nfp_flower_ipv4);
|
||||||
msk += sizeof(struct nfp_flower_ipv4);
|
msk += sizeof(struct nfp_flower_ipv4);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (NFP_FLOWER_LAYER_IPV6 & key_ls->key_layer) {
|
if (NFP_FLOWER_LAYER_IPV6 & key_ls->key_layer) {
|
||||||
/* Populate Exact IPv4 Data. */
|
|
||||||
nfp_flower_compile_ipv6((struct nfp_flower_ipv6 *)ext,
|
nfp_flower_compile_ipv6((struct nfp_flower_ipv6 *)ext,
|
||||||
flow, false);
|
(struct nfp_flower_ipv6 *)msk,
|
||||||
/* Populate Mask IPv4 Data. */
|
flow);
|
||||||
nfp_flower_compile_ipv6((struct nfp_flower_ipv6 *)msk,
|
|
||||||
flow, true);
|
|
||||||
ext += sizeof(struct nfp_flower_ipv6);
|
ext += sizeof(struct nfp_flower_ipv6);
|
||||||
msk += sizeof(struct nfp_flower_ipv6);
|
msk += sizeof(struct nfp_flower_ipv6);
|
||||||
}
|
}
|
||||||
@ -395,10 +403,7 @@ int nfp_flower_compile_flow_match(struct nfp_app *app,
|
|||||||
key_ls->key_layer_two & NFP_FLOWER_LAYER2_GENEVE) {
|
key_ls->key_layer_two & NFP_FLOWER_LAYER2_GENEVE) {
|
||||||
__be32 tun_dst;
|
__be32 tun_dst;
|
||||||
|
|
||||||
/* Populate Exact VXLAN Data. */
|
nfp_flower_compile_ipv4_udp_tun((void *)ext, (void *)msk, flow);
|
||||||
nfp_flower_compile_ipv4_udp_tun((void *)ext, flow, false);
|
|
||||||
/* Populate Mask VXLAN Data. */
|
|
||||||
nfp_flower_compile_ipv4_udp_tun((void *)msk, flow, true);
|
|
||||||
tun_dst = ((struct nfp_flower_ipv4_udp_tun *)ext)->ip_dst;
|
tun_dst = ((struct nfp_flower_ipv4_udp_tun *)ext)->ip_dst;
|
||||||
ext += sizeof(struct nfp_flower_ipv4_udp_tun);
|
ext += sizeof(struct nfp_flower_ipv4_udp_tun);
|
||||||
msk += sizeof(struct nfp_flower_ipv4_udp_tun);
|
msk += sizeof(struct nfp_flower_ipv4_udp_tun);
|
||||||
@ -410,11 +415,7 @@ int nfp_flower_compile_flow_match(struct nfp_app *app,
|
|||||||
nfp_tunnel_add_ipv4_off(app, tun_dst);
|
nfp_tunnel_add_ipv4_off(app, tun_dst);
|
||||||
|
|
||||||
if (key_ls->key_layer_two & NFP_FLOWER_LAYER2_GENEVE_OP) {
|
if (key_ls->key_layer_two & NFP_FLOWER_LAYER2_GENEVE_OP) {
|
||||||
err = nfp_flower_compile_geneve_opt(ext, flow, false);
|
err = nfp_flower_compile_geneve_opt(ext, msk, flow);
|
||||||
if (err)
|
|
||||||
return err;
|
|
||||||
|
|
||||||
err = nfp_flower_compile_geneve_opt(msk, flow, true);
|
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@ -102,23 +102,22 @@ nfp_flower_xmit_flow(struct nfp_app *app, struct nfp_fl_payload *nfp_flow,
|
|||||||
|
|
||||||
static bool nfp_flower_check_higher_than_mac(struct tc_cls_flower_offload *f)
|
static bool nfp_flower_check_higher_than_mac(struct tc_cls_flower_offload *f)
|
||||||
{
|
{
|
||||||
return dissector_uses_key(f->dissector,
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS) ||
|
|
||||||
dissector_uses_key(f->dissector,
|
return flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV4_ADDRS) ||
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS) ||
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV6_ADDRS) ||
|
||||||
dissector_uses_key(f->dissector,
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS) ||
|
||||||
FLOW_DISSECTOR_KEY_PORTS) ||
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ICMP);
|
||||||
dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_ICMP);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
nfp_flower_calc_opt_layer(struct flow_dissector_key_enc_opts *enc_opts,
|
nfp_flower_calc_opt_layer(struct flow_match_enc_opts *enc_opts,
|
||||||
u32 *key_layer_two, int *key_size)
|
u32 *key_layer_two, int *key_size)
|
||||||
{
|
{
|
||||||
if (enc_opts->len > NFP_FL_MAX_GENEVE_OPT_KEY)
|
if (enc_opts->key->len > NFP_FL_MAX_GENEVE_OPT_KEY)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (enc_opts->len > 0) {
|
if (enc_opts->key->len > 0) {
|
||||||
*key_layer_two |= NFP_FLOWER_LAYER2_GENEVE_OP;
|
*key_layer_two |= NFP_FLOWER_LAYER2_GENEVE_OP;
|
||||||
*key_size += sizeof(struct nfp_flower_geneve_options);
|
*key_size += sizeof(struct nfp_flower_geneve_options);
|
||||||
}
|
}
|
||||||
@ -133,20 +132,21 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
struct tc_cls_flower_offload *flow,
|
struct tc_cls_flower_offload *flow,
|
||||||
enum nfp_flower_tun_type *tun_type)
|
enum nfp_flower_tun_type *tun_type)
|
||||||
{
|
{
|
||||||
struct flow_dissector_key_basic *mask_basic = NULL;
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(flow);
|
||||||
struct flow_dissector_key_basic *key_basic = NULL;
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
|
struct flow_match_basic basic = { NULL, NULL};
|
||||||
struct nfp_flower_priv *priv = app->priv;
|
struct nfp_flower_priv *priv = app->priv;
|
||||||
u32 key_layer_two;
|
u32 key_layer_two;
|
||||||
u8 key_layer;
|
u8 key_layer;
|
||||||
int key_size;
|
int key_size;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (flow->dissector->used_keys & ~NFP_FLOWER_WHITELIST_DISSECTOR)
|
if (dissector->used_keys & ~NFP_FLOWER_WHITELIST_DISSECTOR)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
/* If any tun dissector is used then the required set must be used. */
|
/* If any tun dissector is used then the required set must be used. */
|
||||||
if (flow->dissector->used_keys & NFP_FLOWER_WHITELIST_TUN_DISSECTOR &&
|
if (dissector->used_keys & NFP_FLOWER_WHITELIST_TUN_DISSECTOR &&
|
||||||
(flow->dissector->used_keys & NFP_FLOWER_WHITELIST_TUN_DISSECTOR_R)
|
(dissector->used_keys & NFP_FLOWER_WHITELIST_TUN_DISSECTOR_R)
|
||||||
!= NFP_FLOWER_WHITELIST_TUN_DISSECTOR_R)
|
!= NFP_FLOWER_WHITELIST_TUN_DISSECTOR_R)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
@ -155,76 +155,53 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
key_size = sizeof(struct nfp_flower_meta_tci) +
|
key_size = sizeof(struct nfp_flower_meta_tci) +
|
||||||
sizeof(struct nfp_flower_in_port);
|
sizeof(struct nfp_flower_in_port);
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_ETH_ADDRS) ||
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS) ||
|
||||||
dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_MPLS)) {
|
flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_MPLS)) {
|
||||||
key_layer |= NFP_FLOWER_LAYER_MAC;
|
key_layer |= NFP_FLOWER_LAYER_MAC;
|
||||||
key_size += sizeof(struct nfp_flower_mac_mpls);
|
key_size += sizeof(struct nfp_flower_mac_mpls);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_VLAN)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_VLAN)) {
|
||||||
struct flow_dissector_key_vlan *flow_vlan;
|
struct flow_match_vlan vlan;
|
||||||
|
|
||||||
flow_vlan = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_vlan(rule, &vlan);
|
||||||
FLOW_DISSECTOR_KEY_VLAN,
|
|
||||||
flow->mask);
|
|
||||||
if (!(priv->flower_ext_feats & NFP_FL_FEATS_VLAN_PCP) &&
|
if (!(priv->flower_ext_feats & NFP_FL_FEATS_VLAN_PCP) &&
|
||||||
flow_vlan->vlan_priority)
|
vlan.key->vlan_priority)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_CONTROL)) {
|
||||||
FLOW_DISSECTOR_KEY_ENC_CONTROL)) {
|
struct flow_match_enc_opts enc_op = { NULL, NULL };
|
||||||
struct flow_dissector_key_ipv4_addrs *mask_ipv4 = NULL;
|
struct flow_match_ipv4_addrs ipv4_addrs;
|
||||||
struct flow_dissector_key_ports *mask_enc_ports = NULL;
|
struct flow_match_control enc_ctl;
|
||||||
struct flow_dissector_key_enc_opts *enc_op = NULL;
|
struct flow_match_ports enc_ports;
|
||||||
struct flow_dissector_key_ports *enc_ports = NULL;
|
|
||||||
struct flow_dissector_key_control *mask_enc_ctl =
|
|
||||||
skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_CONTROL,
|
|
||||||
flow->mask);
|
|
||||||
struct flow_dissector_key_control *enc_ctl =
|
|
||||||
skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_CONTROL,
|
|
||||||
flow->key);
|
|
||||||
|
|
||||||
if (mask_enc_ctl->addr_type != 0xffff ||
|
flow_rule_match_enc_control(rule, &enc_ctl);
|
||||||
enc_ctl->addr_type != FLOW_DISSECTOR_KEY_IPV4_ADDRS)
|
|
||||||
|
if (enc_ctl.mask->addr_type != 0xffff ||
|
||||||
|
enc_ctl.key->addr_type != FLOW_DISSECTOR_KEY_IPV4_ADDRS)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
/* These fields are already verified as used. */
|
/* These fields are already verified as used. */
|
||||||
mask_ipv4 =
|
flow_rule_match_enc_ipv4_addrs(rule, &ipv4_addrs);
|
||||||
skb_flow_dissector_target(flow->dissector,
|
if (ipv4_addrs.mask->dst != cpu_to_be32(~0))
|
||||||
FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS,
|
|
||||||
flow->mask);
|
|
||||||
if (mask_ipv4->dst != cpu_to_be32(~0))
|
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
mask_enc_ports =
|
|
||||||
skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_PORTS,
|
|
||||||
flow->mask);
|
|
||||||
enc_ports =
|
|
||||||
skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_PORTS,
|
|
||||||
flow->key);
|
|
||||||
|
|
||||||
if (mask_enc_ports->dst != cpu_to_be16(~0))
|
flow_rule_match_enc_ports(rule, &enc_ports);
|
||||||
|
if (enc_ports.mask->dst != cpu_to_be16(~0))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_ENC_OPTS))
|
||||||
FLOW_DISSECTOR_KEY_ENC_OPTS)) {
|
flow_rule_match_enc_opts(rule, &enc_op);
|
||||||
enc_op = skb_flow_dissector_target(flow->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_ENC_OPTS,
|
|
||||||
flow->key);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (enc_ports->dst) {
|
switch (enc_ports.key->dst) {
|
||||||
case htons(NFP_FL_VXLAN_PORT):
|
case htons(NFP_FL_VXLAN_PORT):
|
||||||
*tun_type = NFP_FL_TUNNEL_VXLAN;
|
*tun_type = NFP_FL_TUNNEL_VXLAN;
|
||||||
key_layer |= NFP_FLOWER_LAYER_VXLAN;
|
key_layer |= NFP_FLOWER_LAYER_VXLAN;
|
||||||
key_size += sizeof(struct nfp_flower_ipv4_udp_tun);
|
key_size += sizeof(struct nfp_flower_ipv4_udp_tun);
|
||||||
|
|
||||||
if (enc_op)
|
if (enc_op.key)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
break;
|
break;
|
||||||
case htons(NFP_FL_GENEVE_PORT):
|
case htons(NFP_FL_GENEVE_PORT):
|
||||||
@ -236,11 +213,11 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
key_layer_two |= NFP_FLOWER_LAYER2_GENEVE;
|
key_layer_two |= NFP_FLOWER_LAYER2_GENEVE;
|
||||||
key_size += sizeof(struct nfp_flower_ipv4_udp_tun);
|
key_size += sizeof(struct nfp_flower_ipv4_udp_tun);
|
||||||
|
|
||||||
if (!enc_op)
|
if (!enc_op.key)
|
||||||
break;
|
break;
|
||||||
if (!(priv->flower_ext_feats & NFP_FL_FEATS_GENEVE_OPT))
|
if (!(priv->flower_ext_feats & NFP_FL_FEATS_GENEVE_OPT))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
err = nfp_flower_calc_opt_layer(enc_op, &key_layer_two,
|
err = nfp_flower_calc_opt_layer(&enc_op, &key_layer_two,
|
||||||
&key_size);
|
&key_size);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
@ -254,19 +231,12 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC))
|
||||||
mask_basic = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_basic(rule, &basic);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
flow->mask);
|
|
||||||
|
|
||||||
key_basic = skb_flow_dissector_target(flow->dissector,
|
if (basic.mask && basic.mask->n_proto) {
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
|
||||||
flow->key);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mask_basic && mask_basic->n_proto) {
|
|
||||||
/* Ethernet type is present in the key. */
|
/* Ethernet type is present in the key. */
|
||||||
switch (key_basic->n_proto) {
|
switch (basic.key->n_proto) {
|
||||||
case cpu_to_be16(ETH_P_IP):
|
case cpu_to_be16(ETH_P_IP):
|
||||||
key_layer |= NFP_FLOWER_LAYER_IPV4;
|
key_layer |= NFP_FLOWER_LAYER_IPV4;
|
||||||
key_size += sizeof(struct nfp_flower_ipv4);
|
key_size += sizeof(struct nfp_flower_ipv4);
|
||||||
@ -305,9 +275,9 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mask_basic && mask_basic->ip_proto) {
|
if (basic.mask && basic.mask->ip_proto) {
|
||||||
/* Ethernet type is present in the key. */
|
/* Ethernet type is present in the key. */
|
||||||
switch (key_basic->ip_proto) {
|
switch (basic.key->ip_proto) {
|
||||||
case IPPROTO_TCP:
|
case IPPROTO_TCP:
|
||||||
case IPPROTO_UDP:
|
case IPPROTO_UDP:
|
||||||
case IPPROTO_SCTP:
|
case IPPROTO_SCTP:
|
||||||
@ -324,14 +294,12 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_TCP)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_TCP)) {
|
||||||
struct flow_dissector_key_tcp *tcp;
|
struct flow_match_tcp tcp;
|
||||||
u32 tcp_flags;
|
u32 tcp_flags;
|
||||||
|
|
||||||
tcp = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_tcp(rule, &tcp);
|
||||||
FLOW_DISSECTOR_KEY_TCP,
|
tcp_flags = be16_to_cpu(tcp.key->flags);
|
||||||
flow->key);
|
|
||||||
tcp_flags = be16_to_cpu(tcp->flags);
|
|
||||||
|
|
||||||
if (tcp_flags & ~NFP_FLOWER_SUPPORTED_TCPFLAGS)
|
if (tcp_flags & ~NFP_FLOWER_SUPPORTED_TCPFLAGS)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
@ -347,12 +315,12 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
* space, thus we need to ensure we include a IPv4/IPv6 key
|
* space, thus we need to ensure we include a IPv4/IPv6 key
|
||||||
* layer if we have not done so already.
|
* layer if we have not done so already.
|
||||||
*/
|
*/
|
||||||
if (!key_basic)
|
if (!basic.key)
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (!(key_layer & NFP_FLOWER_LAYER_IPV4) &&
|
if (!(key_layer & NFP_FLOWER_LAYER_IPV4) &&
|
||||||
!(key_layer & NFP_FLOWER_LAYER_IPV6)) {
|
!(key_layer & NFP_FLOWER_LAYER_IPV6)) {
|
||||||
switch (key_basic->n_proto) {
|
switch (basic.key->n_proto) {
|
||||||
case cpu_to_be16(ETH_P_IP):
|
case cpu_to_be16(ETH_P_IP):
|
||||||
key_layer |= NFP_FLOWER_LAYER_IPV4;
|
key_layer |= NFP_FLOWER_LAYER_IPV4;
|
||||||
key_size += sizeof(struct nfp_flower_ipv4);
|
key_size += sizeof(struct nfp_flower_ipv4);
|
||||||
@ -369,14 +337,11 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(flow->dissector, FLOW_DISSECTOR_KEY_CONTROL)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_CONTROL)) {
|
||||||
struct flow_dissector_key_control *key_ctl;
|
struct flow_match_control ctl;
|
||||||
|
|
||||||
key_ctl = skb_flow_dissector_target(flow->dissector,
|
flow_rule_match_control(rule, &ctl);
|
||||||
FLOW_DISSECTOR_KEY_CONTROL,
|
if (ctl.key->flags & ~NFP_FLOWER_SUPPORTED_CTLFLAGS)
|
||||||
flow->key);
|
|
||||||
|
|
||||||
if (key_ctl->flags & ~NFP_FLOWER_SUPPORTED_CTLFLAGS)
|
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2033,24 +2033,20 @@ qede_tc_parse_ports(struct qede_dev *edev,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct qede_arfs_tuple *t)
|
struct qede_arfs_tuple *t)
|
||||||
{
|
{
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_PORTS)) {
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
struct flow_dissector_key_ports *key, *mask;
|
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_PORTS)) {
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
struct flow_match_ports match;
|
||||||
f->key);
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_PORTS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if ((key->src && mask->src != U16_MAX) ||
|
flow_rule_match_ports(rule, &match);
|
||||||
(key->dst && mask->dst != U16_MAX)) {
|
if ((match.key->src && match.mask->src != U16_MAX) ||
|
||||||
|
(match.key->dst && match.mask->dst != U16_MAX)) {
|
||||||
DP_NOTICE(edev, "Do not support ports masks\n");
|
DP_NOTICE(edev, "Do not support ports masks\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
t->src_port = key->src;
|
t->src_port = match.key->src;
|
||||||
t->dst_port = key->dst;
|
t->dst_port = match.key->dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -2061,32 +2057,27 @@ qede_tc_parse_v6_common(struct qede_dev *edev,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct qede_arfs_tuple *t)
|
struct qede_arfs_tuple *t)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
struct in6_addr zero_addr, addr;
|
struct in6_addr zero_addr, addr;
|
||||||
|
|
||||||
memset(&zero_addr, 0, sizeof(addr));
|
memset(&zero_addr, 0, sizeof(addr));
|
||||||
memset(&addr, 0xff, sizeof(addr));
|
memset(&addr, 0xff, sizeof(addr));
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV6_ADDRS)) {
|
||||||
struct flow_dissector_key_ipv6_addrs *key, *mask;
|
struct flow_match_ipv6_addrs match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_ipv6_addrs(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
if ((memcmp(&match.key->src, &zero_addr, sizeof(addr)) &&
|
||||||
f->key);
|
memcmp(&match.mask->src, &addr, sizeof(addr))) ||
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
(memcmp(&match.key->dst, &zero_addr, sizeof(addr)) &&
|
||||||
FLOW_DISSECTOR_KEY_IPV6_ADDRS,
|
memcmp(&match.mask->dst, &addr, sizeof(addr)))) {
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if ((memcmp(&key->src, &zero_addr, sizeof(addr)) &&
|
|
||||||
memcmp(&mask->src, &addr, sizeof(addr))) ||
|
|
||||||
(memcmp(&key->dst, &zero_addr, sizeof(addr)) &&
|
|
||||||
memcmp(&mask->dst, &addr, sizeof(addr)))) {
|
|
||||||
DP_NOTICE(edev,
|
DP_NOTICE(edev,
|
||||||
"Do not support IPv6 address prefix/mask\n");
|
"Do not support IPv6 address prefix/mask\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(&t->src_ipv6, &key->src, sizeof(addr));
|
memcpy(&t->src_ipv6, &match.key->src, sizeof(addr));
|
||||||
memcpy(&t->dst_ipv6, &key->dst, sizeof(addr));
|
memcpy(&t->dst_ipv6, &match.key->dst, sizeof(addr));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (qede_tc_parse_ports(edev, f, t))
|
if (qede_tc_parse_ports(edev, f, t))
|
||||||
@ -2100,24 +2091,20 @@ qede_tc_parse_v4_common(struct qede_dev *edev,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct qede_arfs_tuple *t)
|
struct qede_arfs_tuple *t)
|
||||||
{
|
{
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
struct flow_dissector_key_ipv4_addrs *key, *mask;
|
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_IPV4_ADDRS)) {
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
struct flow_match_ipv4_addrs match;
|
||||||
f->key);
|
|
||||||
mask = skb_flow_dissector_target(f->dissector,
|
|
||||||
FLOW_DISSECTOR_KEY_IPV4_ADDRS,
|
|
||||||
f->mask);
|
|
||||||
|
|
||||||
if ((key->src && mask->src != U32_MAX) ||
|
flow_rule_match_ipv4_addrs(rule, &match);
|
||||||
(key->dst && mask->dst != U32_MAX)) {
|
if ((match.key->src && match.mask->src != U32_MAX) ||
|
||||||
|
(match.key->dst && match.mask->dst != U32_MAX)) {
|
||||||
DP_NOTICE(edev, "Do not support ipv4 prefix/masks\n");
|
DP_NOTICE(edev, "Do not support ipv4 prefix/masks\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
t->src_ipv4 = key->src;
|
t->src_ipv4 = match.key->src;
|
||||||
t->dst_ipv4 = key->dst;
|
t->dst_ipv4 = match.key->dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (qede_tc_parse_ports(edev, f, t))
|
if (qede_tc_parse_ports(edev, f, t))
|
||||||
@ -2175,19 +2162,21 @@ qede_parse_flower_attr(struct qede_dev *edev, __be16 proto,
|
|||||||
struct tc_cls_flower_offload *f,
|
struct tc_cls_flower_offload *f,
|
||||||
struct qede_arfs_tuple *tuple)
|
struct qede_arfs_tuple *tuple)
|
||||||
{
|
{
|
||||||
|
struct flow_rule *rule = tc_cls_flower_offload_flow_rule(f);
|
||||||
|
struct flow_dissector *dissector = rule->match.dissector;
|
||||||
int rc = -EINVAL;
|
int rc = -EINVAL;
|
||||||
u8 ip_proto = 0;
|
u8 ip_proto = 0;
|
||||||
|
|
||||||
memset(tuple, 0, sizeof(*tuple));
|
memset(tuple, 0, sizeof(*tuple));
|
||||||
|
|
||||||
if (f->dissector->used_keys &
|
if (dissector->used_keys &
|
||||||
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
~(BIT(FLOW_DISSECTOR_KEY_CONTROL) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
BIT(FLOW_DISSECTOR_KEY_BASIC) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_IPV6_ADDRS) |
|
BIT(FLOW_DISSECTOR_KEY_IPV6_ADDRS) |
|
||||||
BIT(FLOW_DISSECTOR_KEY_PORTS))) {
|
BIT(FLOW_DISSECTOR_KEY_PORTS))) {
|
||||||
DP_NOTICE(edev, "Unsupported key set:0x%x\n",
|
DP_NOTICE(edev, "Unsupported key set:0x%x\n",
|
||||||
f->dissector->used_keys);
|
dissector->used_keys);
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2197,13 +2186,11 @@ qede_parse_flower_attr(struct qede_dev *edev, __be16 proto,
|
|||||||
return -EPROTONOSUPPORT;
|
return -EPROTONOSUPPORT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dissector_uses_key(f->dissector, FLOW_DISSECTOR_KEY_BASIC)) {
|
if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_BASIC)) {
|
||||||
struct flow_dissector_key_basic *key;
|
struct flow_match_basic match;
|
||||||
|
|
||||||
key = skb_flow_dissector_target(f->dissector,
|
flow_rule_match_basic(rule, &match);
|
||||||
FLOW_DISSECTOR_KEY_BASIC,
|
ip_proto = match.key->ip_proto;
|
||||||
f->key);
|
|
||||||
ip_proto = key->ip_proto;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ip_proto == IPPROTO_TCP && proto == htons(ETH_P_IP))
|
if (ip_proto == IPPROTO_TCP && proto == htons(ETH_P_IP))
|
||||||
|
115
include/net/flow_offload.h
Normal file
115
include/net/flow_offload.h
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
#ifndef _NET_FLOW_OFFLOAD_H
|
||||||
|
#define _NET_FLOW_OFFLOAD_H
|
||||||
|
|
||||||
|
#include <net/flow_dissector.h>
|
||||||
|
|
||||||
|
struct flow_match {
|
||||||
|
struct flow_dissector *dissector;
|
||||||
|
void *mask;
|
||||||
|
void *key;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_basic {
|
||||||
|
struct flow_dissector_key_basic *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_control {
|
||||||
|
struct flow_dissector_key_control *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_eth_addrs {
|
||||||
|
struct flow_dissector_key_eth_addrs *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_vlan {
|
||||||
|
struct flow_dissector_key_vlan *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_ipv4_addrs {
|
||||||
|
struct flow_dissector_key_ipv4_addrs *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_ipv6_addrs {
|
||||||
|
struct flow_dissector_key_ipv6_addrs *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_ip {
|
||||||
|
struct flow_dissector_key_ip *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_ports {
|
||||||
|
struct flow_dissector_key_ports *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_icmp {
|
||||||
|
struct flow_dissector_key_icmp *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_tcp {
|
||||||
|
struct flow_dissector_key_tcp *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_mpls {
|
||||||
|
struct flow_dissector_key_mpls *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_enc_keyid {
|
||||||
|
struct flow_dissector_key_keyid *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_match_enc_opts {
|
||||||
|
struct flow_dissector_key_enc_opts *key, *mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_rule;
|
||||||
|
|
||||||
|
void flow_rule_match_basic(const struct flow_rule *rule,
|
||||||
|
struct flow_match_basic *out);
|
||||||
|
void flow_rule_match_control(const struct flow_rule *rule,
|
||||||
|
struct flow_match_control *out);
|
||||||
|
void flow_rule_match_eth_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_eth_addrs *out);
|
||||||
|
void flow_rule_match_vlan(const struct flow_rule *rule,
|
||||||
|
struct flow_match_vlan *out);
|
||||||
|
void flow_rule_match_ipv4_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv4_addrs *out);
|
||||||
|
void flow_rule_match_ipv6_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv6_addrs *out);
|
||||||
|
void flow_rule_match_ip(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ip *out);
|
||||||
|
void flow_rule_match_ports(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ports *out);
|
||||||
|
void flow_rule_match_tcp(const struct flow_rule *rule,
|
||||||
|
struct flow_match_tcp *out);
|
||||||
|
void flow_rule_match_icmp(const struct flow_rule *rule,
|
||||||
|
struct flow_match_icmp *out);
|
||||||
|
void flow_rule_match_mpls(const struct flow_rule *rule,
|
||||||
|
struct flow_match_mpls *out);
|
||||||
|
void flow_rule_match_enc_control(const struct flow_rule *rule,
|
||||||
|
struct flow_match_control *out);
|
||||||
|
void flow_rule_match_enc_ipv4_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv4_addrs *out);
|
||||||
|
void flow_rule_match_enc_ipv6_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv6_addrs *out);
|
||||||
|
void flow_rule_match_enc_ip(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ip *out);
|
||||||
|
void flow_rule_match_enc_ports(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ports *out);
|
||||||
|
void flow_rule_match_enc_keyid(const struct flow_rule *rule,
|
||||||
|
struct flow_match_enc_keyid *out);
|
||||||
|
void flow_rule_match_enc_opts(const struct flow_rule *rule,
|
||||||
|
struct flow_match_enc_opts *out);
|
||||||
|
|
||||||
|
struct flow_rule {
|
||||||
|
struct flow_match match;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct flow_rule *flow_rule_alloc(void);
|
||||||
|
|
||||||
|
static inline bool flow_rule_match_key(const struct flow_rule *rule,
|
||||||
|
enum flow_dissector_key_id key)
|
||||||
|
{
|
||||||
|
return dissector_uses_key(rule->match.dissector, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* _NET_FLOW_OFFLOAD_H */
|
@ -6,6 +6,7 @@
|
|||||||
#include <linux/workqueue.h>
|
#include <linux/workqueue.h>
|
||||||
#include <net/sch_generic.h>
|
#include <net/sch_generic.h>
|
||||||
#include <net/act_api.h>
|
#include <net/act_api.h>
|
||||||
|
#include <net/flow_offload.h>
|
||||||
|
|
||||||
/* TC action not accessible from user space */
|
/* TC action not accessible from user space */
|
||||||
#define TC_ACT_REINSERT (TC_ACT_VALUE_MAX + 1)
|
#define TC_ACT_REINSERT (TC_ACT_VALUE_MAX + 1)
|
||||||
@ -760,13 +761,17 @@ struct tc_cls_flower_offload {
|
|||||||
struct tc_cls_common_offload common;
|
struct tc_cls_common_offload common;
|
||||||
enum tc_fl_command command;
|
enum tc_fl_command command;
|
||||||
unsigned long cookie;
|
unsigned long cookie;
|
||||||
struct flow_dissector *dissector;
|
struct flow_rule *rule;
|
||||||
struct fl_flow_key *mask;
|
|
||||||
struct fl_flow_key *key;
|
|
||||||
struct tcf_exts *exts;
|
struct tcf_exts *exts;
|
||||||
u32 classid;
|
u32 classid;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static inline struct flow_rule *
|
||||||
|
tc_cls_flower_offload_flow_rule(struct tc_cls_flower_offload *tc_flow_cmd)
|
||||||
|
{
|
||||||
|
return tc_flow_cmd->rule;
|
||||||
|
}
|
||||||
|
|
||||||
enum tc_matchall_command {
|
enum tc_matchall_command {
|
||||||
TC_CLSMATCHALL_REPLACE,
|
TC_CLSMATCHALL_REPLACE,
|
||||||
TC_CLSMATCHALL_DESTROY,
|
TC_CLSMATCHALL_DESTROY,
|
||||||
|
@ -11,7 +11,7 @@ obj-$(CONFIG_SYSCTL) += sysctl_net_core.o
|
|||||||
obj-y += dev.o ethtool.o dev_addr_lists.o dst.o netevent.o \
|
obj-y += dev.o ethtool.o dev_addr_lists.o dst.o netevent.o \
|
||||||
neighbour.o rtnetlink.o utils.o link_watch.o filter.o \
|
neighbour.o rtnetlink.o utils.o link_watch.o filter.o \
|
||||||
sock_diag.o dev_ioctl.o tso.o sock_reuseport.o \
|
sock_diag.o dev_ioctl.o tso.o sock_reuseport.o \
|
||||||
fib_notifier.o xdp.o
|
fib_notifier.o xdp.o flow_offload.o
|
||||||
|
|
||||||
obj-y += net-sysfs.o
|
obj-y += net-sysfs.o
|
||||||
obj-$(CONFIG_PAGE_POOL) += page_pool.o
|
obj-$(CONFIG_PAGE_POOL) += page_pool.o
|
||||||
|
143
net/core/flow_offload.c
Normal file
143
net/core/flow_offload.c
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
/* SPDX-License-Identifier: GPL-2.0 */
|
||||||
|
#include <linux/kernel.h>
|
||||||
|
#include <linux/slab.h>
|
||||||
|
#include <net/flow_offload.h>
|
||||||
|
|
||||||
|
struct flow_rule *flow_rule_alloc(void)
|
||||||
|
{
|
||||||
|
return kzalloc(sizeof(struct flow_rule), GFP_KERNEL);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_alloc);
|
||||||
|
|
||||||
|
#define FLOW_DISSECTOR_MATCH(__rule, __type, __out) \
|
||||||
|
const struct flow_match *__m = &(__rule)->match; \
|
||||||
|
struct flow_dissector *__d = (__m)->dissector; \
|
||||||
|
\
|
||||||
|
(__out)->key = skb_flow_dissector_target(__d, __type, (__m)->key); \
|
||||||
|
(__out)->mask = skb_flow_dissector_target(__d, __type, (__m)->mask); \
|
||||||
|
|
||||||
|
void flow_rule_match_basic(const struct flow_rule *rule,
|
||||||
|
struct flow_match_basic *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_BASIC, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_basic);
|
||||||
|
|
||||||
|
void flow_rule_match_control(const struct flow_rule *rule,
|
||||||
|
struct flow_match_control *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_CONTROL, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_control);
|
||||||
|
|
||||||
|
void flow_rule_match_eth_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_eth_addrs *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ETH_ADDRS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_eth_addrs);
|
||||||
|
|
||||||
|
void flow_rule_match_vlan(const struct flow_rule *rule,
|
||||||
|
struct flow_match_vlan *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_VLAN, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_vlan);
|
||||||
|
|
||||||
|
void flow_rule_match_ipv4_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv4_addrs *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_IPV4_ADDRS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_ipv4_addrs);
|
||||||
|
|
||||||
|
void flow_rule_match_ipv6_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv6_addrs *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_IPV6_ADDRS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_ipv6_addrs);
|
||||||
|
|
||||||
|
void flow_rule_match_ip(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ip *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_IP, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_ip);
|
||||||
|
|
||||||
|
void flow_rule_match_ports(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ports *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_PORTS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_ports);
|
||||||
|
|
||||||
|
void flow_rule_match_tcp(const struct flow_rule *rule,
|
||||||
|
struct flow_match_tcp *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_TCP, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_tcp);
|
||||||
|
|
||||||
|
void flow_rule_match_icmp(const struct flow_rule *rule,
|
||||||
|
struct flow_match_icmp *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ICMP, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_icmp);
|
||||||
|
|
||||||
|
void flow_rule_match_mpls(const struct flow_rule *rule,
|
||||||
|
struct flow_match_mpls *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_MPLS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_mpls);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_control(const struct flow_rule *rule,
|
||||||
|
struct flow_match_control *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_CONTROL, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_control);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_ipv4_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv4_addrs *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_IPV4_ADDRS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_ipv4_addrs);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_ipv6_addrs(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ipv6_addrs *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_IPV6_ADDRS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_ipv6_addrs);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_ip(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ip *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_IP, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_ip);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_ports(const struct flow_rule *rule,
|
||||||
|
struct flow_match_ports *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_PORTS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_ports);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_keyid(const struct flow_rule *rule,
|
||||||
|
struct flow_match_enc_keyid *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_KEYID, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_keyid);
|
||||||
|
|
||||||
|
void flow_rule_match_enc_opts(const struct flow_rule *rule,
|
||||||
|
struct flow_match_enc_opts *out)
|
||||||
|
{
|
||||||
|
FLOW_DISSECTOR_MATCH(rule, FLOW_DISSECTOR_KEY_ENC_OPTS, out);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(flow_rule_match_enc_opts);
|
@ -381,16 +381,22 @@ static int fl_hw_replace_filter(struct tcf_proto *tp,
|
|||||||
bool skip_sw = tc_skip_sw(f->flags);
|
bool skip_sw = tc_skip_sw(f->flags);
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
|
cls_flower.rule = flow_rule_alloc();
|
||||||
|
if (!cls_flower.rule)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
tc_cls_common_offload_init(&cls_flower.common, tp, f->flags, extack);
|
tc_cls_common_offload_init(&cls_flower.common, tp, f->flags, extack);
|
||||||
cls_flower.command = TC_CLSFLOWER_REPLACE;
|
cls_flower.command = TC_CLSFLOWER_REPLACE;
|
||||||
cls_flower.cookie = (unsigned long) f;
|
cls_flower.cookie = (unsigned long) f;
|
||||||
cls_flower.dissector = &f->mask->dissector;
|
cls_flower.rule->match.dissector = &f->mask->dissector;
|
||||||
cls_flower.mask = &f->mask->key;
|
cls_flower.rule->match.mask = &f->mask->key;
|
||||||
cls_flower.key = &f->mkey;
|
cls_flower.rule->match.key = &f->mkey;
|
||||||
cls_flower.exts = &f->exts;
|
cls_flower.exts = &f->exts;
|
||||||
cls_flower.classid = f->res.classid;
|
cls_flower.classid = f->res.classid;
|
||||||
|
|
||||||
err = tc_setup_cb_call(block, TC_SETUP_CLSFLOWER, &cls_flower, skip_sw);
|
err = tc_setup_cb_call(block, TC_SETUP_CLSFLOWER, &cls_flower, skip_sw);
|
||||||
|
kfree(cls_flower.rule);
|
||||||
|
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
fl_hw_destroy_filter(tp, f, NULL);
|
fl_hw_destroy_filter(tp, f, NULL);
|
||||||
return err;
|
return err;
|
||||||
@ -1463,18 +1469,24 @@ static int fl_reoffload(struct tcf_proto *tp, bool add, tc_setup_cb_t *cb,
|
|||||||
if (tc_skip_hw(f->flags))
|
if (tc_skip_hw(f->flags))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
cls_flower.rule = flow_rule_alloc();
|
||||||
|
if (!cls_flower.rule)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
tc_cls_common_offload_init(&cls_flower.common, tp,
|
tc_cls_common_offload_init(&cls_flower.common, tp,
|
||||||
f->flags, extack);
|
f->flags, extack);
|
||||||
cls_flower.command = add ?
|
cls_flower.command = add ?
|
||||||
TC_CLSFLOWER_REPLACE : TC_CLSFLOWER_DESTROY;
|
TC_CLSFLOWER_REPLACE : TC_CLSFLOWER_DESTROY;
|
||||||
cls_flower.cookie = (unsigned long)f;
|
cls_flower.cookie = (unsigned long)f;
|
||||||
cls_flower.dissector = &mask->dissector;
|
cls_flower.rule->match.dissector = &mask->dissector;
|
||||||
cls_flower.mask = &mask->key;
|
cls_flower.rule->match.mask = &mask->key;
|
||||||
cls_flower.key = &f->mkey;
|
cls_flower.rule->match.key = &f->mkey;
|
||||||
cls_flower.exts = &f->exts;
|
cls_flower.exts = &f->exts;
|
||||||
cls_flower.classid = f->res.classid;
|
cls_flower.classid = f->res.classid;
|
||||||
|
|
||||||
err = cb(TC_SETUP_CLSFLOWER, &cls_flower, cb_priv);
|
err = cb(TC_SETUP_CLSFLOWER, &cls_flower, cb_priv);
|
||||||
|
kfree(cls_flower.rule);
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
if (add && tc_skip_sw(f->flags))
|
if (add && tc_skip_sw(f->flags))
|
||||||
return err;
|
return err;
|
||||||
@ -1489,25 +1501,32 @@ static int fl_reoffload(struct tcf_proto *tp, bool add, tc_setup_cb_t *cb,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void fl_hw_create_tmplt(struct tcf_chain *chain,
|
static int fl_hw_create_tmplt(struct tcf_chain *chain,
|
||||||
struct fl_flow_tmplt *tmplt)
|
struct fl_flow_tmplt *tmplt)
|
||||||
{
|
{
|
||||||
struct tc_cls_flower_offload cls_flower = {};
|
struct tc_cls_flower_offload cls_flower = {};
|
||||||
struct tcf_block *block = chain->block;
|
struct tcf_block *block = chain->block;
|
||||||
struct tcf_exts dummy_exts = { 0, };
|
struct tcf_exts dummy_exts = { 0, };
|
||||||
|
|
||||||
|
cls_flower.rule = flow_rule_alloc();
|
||||||
|
if (!cls_flower.rule)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
cls_flower.common.chain_index = chain->index;
|
cls_flower.common.chain_index = chain->index;
|
||||||
cls_flower.command = TC_CLSFLOWER_TMPLT_CREATE;
|
cls_flower.command = TC_CLSFLOWER_TMPLT_CREATE;
|
||||||
cls_flower.cookie = (unsigned long) tmplt;
|
cls_flower.cookie = (unsigned long) tmplt;
|
||||||
cls_flower.dissector = &tmplt->dissector;
|
cls_flower.rule->match.dissector = &tmplt->dissector;
|
||||||
cls_flower.mask = &tmplt->mask;
|
cls_flower.rule->match.mask = &tmplt->mask;
|
||||||
cls_flower.key = &tmplt->dummy_key;
|
cls_flower.rule->match.key = &tmplt->dummy_key;
|
||||||
cls_flower.exts = &dummy_exts;
|
cls_flower.exts = &dummy_exts;
|
||||||
|
|
||||||
/* We don't care if driver (any of them) fails to handle this
|
/* We don't care if driver (any of them) fails to handle this
|
||||||
* call. It serves just as a hint for it.
|
* call. It serves just as a hint for it.
|
||||||
*/
|
*/
|
||||||
tc_setup_cb_call(block, TC_SETUP_CLSFLOWER, &cls_flower, false);
|
tc_setup_cb_call(block, TC_SETUP_CLSFLOWER, &cls_flower, false);
|
||||||
|
kfree(cls_flower.rule);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void fl_hw_destroy_tmplt(struct tcf_chain *chain,
|
static void fl_hw_destroy_tmplt(struct tcf_chain *chain,
|
||||||
@ -1551,12 +1570,14 @@ static void *fl_tmplt_create(struct net *net, struct tcf_chain *chain,
|
|||||||
err = fl_set_key(net, tb, &tmplt->dummy_key, &tmplt->mask, extack);
|
err = fl_set_key(net, tb, &tmplt->dummy_key, &tmplt->mask, extack);
|
||||||
if (err)
|
if (err)
|
||||||
goto errout_tmplt;
|
goto errout_tmplt;
|
||||||
kfree(tb);
|
|
||||||
|
|
||||||
fl_init_dissector(&tmplt->dissector, &tmplt->mask);
|
fl_init_dissector(&tmplt->dissector, &tmplt->mask);
|
||||||
|
|
||||||
fl_hw_create_tmplt(chain, tmplt);
|
err = fl_hw_create_tmplt(chain, tmplt);
|
||||||
|
if (err)
|
||||||
|
goto errout_tmplt;
|
||||||
|
|
||||||
|
kfree(tb);
|
||||||
return tmplt;
|
return tmplt;
|
||||||
|
|
||||||
errout_tmplt:
|
errout_tmplt:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user