Commit 2ef813b8 authored by Herat Ramani's avatar Herat Ramani Committed by Jakub Kicinski

cxgb4: handle 4-tuple PEDIT to NAT mode translation

The 4-tuple NAT offload via PEDIT always overwrites all the 4-tuple
fields even if they had not been explicitly enabled. If any fields in
the 4-tuple are not enabled, then the hardware overwrites the
disabled fields with zeros, instead of ignoring them.

So, add a parser that can translate the enabled 4-tuple PEDIT fields
to one of the NAT mode combinations supported by the hardware and
hence avoid overwriting disabled fields to 0. Any rule with
unsupported NAT mode combination is rejected.
Signed-off-by: default avatarHerat Ramani <herat@chelsio.com>
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent f8ea4a19
...@@ -60,6 +60,89 @@ static struct ch_tc_pedit_fields pedits[] = { ...@@ -60,6 +60,89 @@ static struct ch_tc_pedit_fields pedits[] = {
PEDIT_FIELDS(IP6_, DST_127_96, 4, nat_lip, 12), PEDIT_FIELDS(IP6_, DST_127_96, 4, nat_lip, 12),
}; };
static const struct cxgb4_natmode_config cxgb4_natmode_config_array[] = {
/* Default supported NAT modes */
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_NONE,
.natmode = NAT_MODE_NONE,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP,
.natmode = NAT_MODE_DIP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_DPORT,
.natmode = NAT_MODE_DIP_DP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_DPORT |
CXGB4_ACTION_NATMODE_SIP,
.natmode = NAT_MODE_DIP_DP_SIP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_DPORT |
CXGB4_ACTION_NATMODE_SPORT,
.natmode = NAT_MODE_DIP_DP_SP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_SIP | CXGB4_ACTION_NATMODE_SPORT,
.natmode = NAT_MODE_SIP_SP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_SIP |
CXGB4_ACTION_NATMODE_SPORT,
.natmode = NAT_MODE_DIP_SIP_SP,
},
{
.chip = CHELSIO_T5,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_SIP |
CXGB4_ACTION_NATMODE_DPORT |
CXGB4_ACTION_NATMODE_SPORT,
.natmode = NAT_MODE_ALL,
},
/* T6+ can ignore L4 ports when they're disabled. */
{
.chip = CHELSIO_T6,
.flags = CXGB4_ACTION_NATMODE_SIP,
.natmode = NAT_MODE_SIP_SP,
},
{
.chip = CHELSIO_T6,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_SPORT,
.natmode = NAT_MODE_DIP_DP_SP,
},
{
.chip = CHELSIO_T6,
.flags = CXGB4_ACTION_NATMODE_DIP | CXGB4_ACTION_NATMODE_SIP,
.natmode = NAT_MODE_ALL,
},
};
static void cxgb4_action_natmode_tweak(struct ch_filter_specification *fs,
u8 natmode_flags)
{
u8 i = 0;
/* Translate the enabled NAT 4-tuple fields to one of the
* hardware supported NAT mode configurations. This ensures
* that we pick a valid combination, where the disabled fields
* do not get overwritten to 0.
*/
for (i = 0; i < ARRAY_SIZE(cxgb4_natmode_config_array); i++) {
if (cxgb4_natmode_config_array[i].flags == natmode_flags) {
fs->nat_mode = cxgb4_natmode_config_array[i].natmode;
return;
}
}
}
static struct ch_tc_flower_entry *allocate_flower_entry(void) static struct ch_tc_flower_entry *allocate_flower_entry(void)
{ {
struct ch_tc_flower_entry *new = kzalloc(sizeof(*new), GFP_KERNEL); struct ch_tc_flower_entry *new = kzalloc(sizeof(*new), GFP_KERNEL);
...@@ -289,7 +372,8 @@ static void offload_pedit(struct ch_filter_specification *fs, u32 val, u32 mask, ...@@ -289,7 +372,8 @@ static void offload_pedit(struct ch_filter_specification *fs, u32 val, u32 mask,
} }
static void process_pedit_field(struct ch_filter_specification *fs, u32 val, static void process_pedit_field(struct ch_filter_specification *fs, u32 val,
u32 mask, u32 offset, u8 htype) u32 mask, u32 offset, u8 htype,
u8 *natmode_flags)
{ {
switch (htype) { switch (htype) {
case FLOW_ACT_MANGLE_HDR_TYPE_ETH: case FLOW_ACT_MANGLE_HDR_TYPE_ETH:
...@@ -314,67 +398,102 @@ static void process_pedit_field(struct ch_filter_specification *fs, u32 val, ...@@ -314,67 +398,102 @@ static void process_pedit_field(struct ch_filter_specification *fs, u32 val,
switch (offset) { switch (offset) {
case PEDIT_IP4_SRC: case PEDIT_IP4_SRC:
offload_pedit(fs, val, mask, IP4_SRC); offload_pedit(fs, val, mask, IP4_SRC);
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break; break;
case PEDIT_IP4_DST: case PEDIT_IP4_DST:
offload_pedit(fs, val, mask, IP4_DST); offload_pedit(fs, val, mask, IP4_DST);
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
} }
fs->nat_mode = NAT_MODE_ALL;
break; break;
case FLOW_ACT_MANGLE_HDR_TYPE_IP6: case FLOW_ACT_MANGLE_HDR_TYPE_IP6:
switch (offset) { switch (offset) {
case PEDIT_IP6_SRC_31_0: case PEDIT_IP6_SRC_31_0:
offload_pedit(fs, val, mask, IP6_SRC_31_0); offload_pedit(fs, val, mask, IP6_SRC_31_0);
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break; break;
case PEDIT_IP6_SRC_63_32: case PEDIT_IP6_SRC_63_32:
offload_pedit(fs, val, mask, IP6_SRC_63_32); offload_pedit(fs, val, mask, IP6_SRC_63_32);
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break; break;
case PEDIT_IP6_SRC_95_64: case PEDIT_IP6_SRC_95_64:
offload_pedit(fs, val, mask, IP6_SRC_95_64); offload_pedit(fs, val, mask, IP6_SRC_95_64);
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break; break;
case PEDIT_IP6_SRC_127_96: case PEDIT_IP6_SRC_127_96:
offload_pedit(fs, val, mask, IP6_SRC_127_96); offload_pedit(fs, val, mask, IP6_SRC_127_96);
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break; break;
case PEDIT_IP6_DST_31_0: case PEDIT_IP6_DST_31_0:
offload_pedit(fs, val, mask, IP6_DST_31_0); offload_pedit(fs, val, mask, IP6_DST_31_0);
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
break; break;
case PEDIT_IP6_DST_63_32: case PEDIT_IP6_DST_63_32:
offload_pedit(fs, val, mask, IP6_DST_63_32); offload_pedit(fs, val, mask, IP6_DST_63_32);
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
break; break;
case PEDIT_IP6_DST_95_64: case PEDIT_IP6_DST_95_64:
offload_pedit(fs, val, mask, IP6_DST_95_64); offload_pedit(fs, val, mask, IP6_DST_95_64);
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
break; break;
case PEDIT_IP6_DST_127_96: case PEDIT_IP6_DST_127_96:
offload_pedit(fs, val, mask, IP6_DST_127_96); offload_pedit(fs, val, mask, IP6_DST_127_96);
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
} }
fs->nat_mode = NAT_MODE_ALL;
break; break;
case FLOW_ACT_MANGLE_HDR_TYPE_TCP: case FLOW_ACT_MANGLE_HDR_TYPE_TCP:
switch (offset) { switch (offset) {
case PEDIT_TCP_SPORT_DPORT: case PEDIT_TCP_SPORT_DPORT:
if (~mask & PEDIT_TCP_UDP_SPORT_MASK) if (~mask & PEDIT_TCP_UDP_SPORT_MASK) {
fs->nat_fport = val; fs->nat_fport = val;
else *natmode_flags |= CXGB4_ACTION_NATMODE_SPORT;
} else {
fs->nat_lport = val >> 16; fs->nat_lport = val >> 16;
*natmode_flags |= CXGB4_ACTION_NATMODE_DPORT;
}
} }
fs->nat_mode = NAT_MODE_ALL;
break; break;
case FLOW_ACT_MANGLE_HDR_TYPE_UDP: case FLOW_ACT_MANGLE_HDR_TYPE_UDP:
switch (offset) { switch (offset) {
case PEDIT_UDP_SPORT_DPORT: case PEDIT_UDP_SPORT_DPORT:
if (~mask & PEDIT_TCP_UDP_SPORT_MASK) if (~mask & PEDIT_TCP_UDP_SPORT_MASK) {
fs->nat_fport = val; fs->nat_fport = val;
else *natmode_flags |= CXGB4_ACTION_NATMODE_SPORT;
} else {
fs->nat_lport = val >> 16; fs->nat_lport = val >> 16;
*natmode_flags |= CXGB4_ACTION_NATMODE_DPORT;
}
} }
fs->nat_mode = NAT_MODE_ALL; break;
} }
} }
static int cxgb4_action_natmode_validate(struct adapter *adap, u8 natmode_flags,
struct netlink_ext_ack *extack)
{
u8 i = 0;
/* Extract the NAT mode to enable based on what 4-tuple fields
* are enabled to be overwritten. This ensures that the
* disabled fields don't get overwritten to 0.
*/
for (i = 0; i < ARRAY_SIZE(cxgb4_natmode_config_array); i++) {
const struct cxgb4_natmode_config *c;
c = &cxgb4_natmode_config_array[i];
if (CHELSIO_CHIP_VERSION(adap->params.chip) >= c->chip &&
natmode_flags == c->flags)
return 0;
}
NL_SET_ERR_MSG_MOD(extack, "Unsupported NAT mode 4-tuple combination");
return -EOPNOTSUPP;
}
void cxgb4_process_flow_actions(struct net_device *in, void cxgb4_process_flow_actions(struct net_device *in,
struct flow_action *actions, struct flow_action *actions,
struct ch_filter_specification *fs) struct ch_filter_specification *fs)
{ {
struct flow_action_entry *act; struct flow_action_entry *act;
u8 natmode_flags = 0;
int i; int i;
flow_action_for_each(i, act, actions) { flow_action_for_each(i, act, actions) {
...@@ -426,7 +545,8 @@ void cxgb4_process_flow_actions(struct net_device *in, ...@@ -426,7 +545,8 @@ void cxgb4_process_flow_actions(struct net_device *in,
val = act->mangle.val; val = act->mangle.val;
offset = act->mangle.offset; offset = act->mangle.offset;
process_pedit_field(fs, val, mask, offset, htype); process_pedit_field(fs, val, mask, offset, htype,
&natmode_flags);
} }
break; break;
case FLOW_ACTION_QUEUE: case FLOW_ACTION_QUEUE:
...@@ -438,6 +558,9 @@ void cxgb4_process_flow_actions(struct net_device *in, ...@@ -438,6 +558,9 @@ void cxgb4_process_flow_actions(struct net_device *in,
break; break;
} }
} }
if (natmode_flags)
cxgb4_action_natmode_tweak(fs, natmode_flags);
} }
static bool valid_l4_mask(u32 mask) static bool valid_l4_mask(u32 mask)
...@@ -454,7 +577,8 @@ static bool valid_l4_mask(u32 mask) ...@@ -454,7 +577,8 @@ static bool valid_l4_mask(u32 mask)
} }
static bool valid_pedit_action(struct net_device *dev, static bool valid_pedit_action(struct net_device *dev,
const struct flow_action_entry *act) const struct flow_action_entry *act,
u8 *natmode_flags)
{ {
u32 mask, offset; u32 mask, offset;
u8 htype; u8 htype;
...@@ -479,7 +603,10 @@ static bool valid_pedit_action(struct net_device *dev, ...@@ -479,7 +603,10 @@ static bool valid_pedit_action(struct net_device *dev,
case FLOW_ACT_MANGLE_HDR_TYPE_IP4: case FLOW_ACT_MANGLE_HDR_TYPE_IP4:
switch (offset) { switch (offset) {
case PEDIT_IP4_SRC: case PEDIT_IP4_SRC:
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break;
case PEDIT_IP4_DST: case PEDIT_IP4_DST:
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
break; break;
default: default:
netdev_err(dev, "%s: Unsupported pedit field\n", netdev_err(dev, "%s: Unsupported pedit field\n",
...@@ -493,10 +620,13 @@ static bool valid_pedit_action(struct net_device *dev, ...@@ -493,10 +620,13 @@ static bool valid_pedit_action(struct net_device *dev,
case PEDIT_IP6_SRC_63_32: case PEDIT_IP6_SRC_63_32:
case PEDIT_IP6_SRC_95_64: case PEDIT_IP6_SRC_95_64:
case PEDIT_IP6_SRC_127_96: case PEDIT_IP6_SRC_127_96:
*natmode_flags |= CXGB4_ACTION_NATMODE_SIP;
break;
case PEDIT_IP6_DST_31_0: case PEDIT_IP6_DST_31_0:
case PEDIT_IP6_DST_63_32: case PEDIT_IP6_DST_63_32:
case PEDIT_IP6_DST_95_64: case PEDIT_IP6_DST_95_64:
case PEDIT_IP6_DST_127_96: case PEDIT_IP6_DST_127_96:
*natmode_flags |= CXGB4_ACTION_NATMODE_DIP;
break; break;
default: default:
netdev_err(dev, "%s: Unsupported pedit field\n", netdev_err(dev, "%s: Unsupported pedit field\n",
...@@ -512,6 +642,10 @@ static bool valid_pedit_action(struct net_device *dev, ...@@ -512,6 +642,10 @@ static bool valid_pedit_action(struct net_device *dev,
__func__); __func__);
return false; return false;
} }
if (~mask & PEDIT_TCP_UDP_SPORT_MASK)
*natmode_flags |= CXGB4_ACTION_NATMODE_SPORT;
else
*natmode_flags |= CXGB4_ACTION_NATMODE_DPORT;
break; break;
default: default:
netdev_err(dev, "%s: Unsupported pedit field\n", netdev_err(dev, "%s: Unsupported pedit field\n",
...@@ -527,6 +661,10 @@ static bool valid_pedit_action(struct net_device *dev, ...@@ -527,6 +661,10 @@ static bool valid_pedit_action(struct net_device *dev,
__func__); __func__);
return false; return false;
} }
if (~mask & PEDIT_TCP_UDP_SPORT_MASK)
*natmode_flags |= CXGB4_ACTION_NATMODE_SPORT;
else
*natmode_flags |= CXGB4_ACTION_NATMODE_DPORT;
break; break;
default: default:
netdev_err(dev, "%s: Unsupported pedit field\n", netdev_err(dev, "%s: Unsupported pedit field\n",
...@@ -546,10 +684,12 @@ int cxgb4_validate_flow_actions(struct net_device *dev, ...@@ -546,10 +684,12 @@ int cxgb4_validate_flow_actions(struct net_device *dev,
struct netlink_ext_ack *extack, struct netlink_ext_ack *extack,
u8 matchall_filter) u8 matchall_filter)
{ {
struct adapter *adap = netdev2adap(dev);
struct flow_action_entry *act; struct flow_action_entry *act;
bool act_redir = false; bool act_redir = false;
bool act_pedit = false; bool act_pedit = false;
bool act_vlan = false; bool act_vlan = false;
u8 natmode_flags = 0;
int i; int i;
if (!flow_action_basic_hw_stats_check(actions, extack)) if (!flow_action_basic_hw_stats_check(actions, extack))
...@@ -563,7 +703,6 @@ int cxgb4_validate_flow_actions(struct net_device *dev, ...@@ -563,7 +703,6 @@ int cxgb4_validate_flow_actions(struct net_device *dev,
break; break;
case FLOW_ACTION_MIRRED: case FLOW_ACTION_MIRRED:
case FLOW_ACTION_REDIRECT: { case FLOW_ACTION_REDIRECT: {
struct adapter *adap = netdev2adap(dev);
struct net_device *n_dev, *target_dev; struct net_device *n_dev, *target_dev;
bool found = false; bool found = false;
unsigned int i; unsigned int i;
...@@ -620,7 +759,8 @@ int cxgb4_validate_flow_actions(struct net_device *dev, ...@@ -620,7 +759,8 @@ int cxgb4_validate_flow_actions(struct net_device *dev,
} }
break; break;
case FLOW_ACTION_MANGLE: { case FLOW_ACTION_MANGLE: {
bool pedit_valid = valid_pedit_action(dev, act); bool pedit_valid = valid_pedit_action(dev, act,
&natmode_flags);
if (!pedit_valid) if (!pedit_valid)
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -642,6 +782,15 @@ int cxgb4_validate_flow_actions(struct net_device *dev, ...@@ -642,6 +782,15 @@ int cxgb4_validate_flow_actions(struct net_device *dev,
return -EINVAL; return -EINVAL;
} }
if (act_pedit) {
int ret;
ret = cxgb4_action_natmode_validate(adap, natmode_flags,
extack);
if (ret)
return ret;
}
return 0; return 0;
} }
......
...@@ -108,6 +108,21 @@ struct ch_tc_pedit_fields { ...@@ -108,6 +108,21 @@ struct ch_tc_pedit_fields {
#define PEDIT_TCP_SPORT_DPORT 0x0 #define PEDIT_TCP_SPORT_DPORT 0x0
#define PEDIT_UDP_SPORT_DPORT 0x0 #define PEDIT_UDP_SPORT_DPORT 0x0
enum cxgb4_action_natmode_flags {
CXGB4_ACTION_NATMODE_NONE = 0,
CXGB4_ACTION_NATMODE_DIP = (1 << 0),
CXGB4_ACTION_NATMODE_SIP = (1 << 1),
CXGB4_ACTION_NATMODE_DPORT = (1 << 2),
CXGB4_ACTION_NATMODE_SPORT = (1 << 3),
};
/* TC PEDIT action to NATMODE translation entry */
struct cxgb4_natmode_config {
enum chip_type chip;
u8 flags;
u8 natmode;
};
void cxgb4_process_flow_actions(struct net_device *in, void cxgb4_process_flow_actions(struct net_device *in,
struct flow_action *actions, struct flow_action *actions,
struct ch_filter_specification *fs); struct ch_filter_specification *fs);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment