|
|
|
@ -112,7 +112,7 @@ static void update_range(struct sw_flow_match *match,
|
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
static bool match_validate(const struct sw_flow_match *match,
|
|
|
|
|
u64 key_attrs, u64 mask_attrs)
|
|
|
|
|
u64 key_attrs, u64 mask_attrs, bool log)
|
|
|
|
|
{
|
|
|
|
|
u64 key_expected = 1 << OVS_KEY_ATTR_ETHERNET;
|
|
|
|
|
u64 mask_allowed = key_attrs; /* At most allow all key attributes */
|
|
|
|
@ -230,15 +230,17 @@ static bool match_validate(const struct sw_flow_match *match,
|
|
|
|
|
|
|
|
|
|
if ((key_attrs & key_expected) != key_expected) {
|
|
|
|
|
/* Key attributes check failed. */
|
|
|
|
|
OVS_NLERR("Missing expected key attributes (key_attrs=%llx, expected=%llx).\n",
|
|
|
|
|
(unsigned long long)key_attrs, (unsigned long long)key_expected);
|
|
|
|
|
OVS_NLERR(log, "Missing key (keys=%llx, expected=%llx)",
|
|
|
|
|
(unsigned long long)key_attrs,
|
|
|
|
|
(unsigned long long)key_expected);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((mask_attrs & mask_allowed) != mask_attrs) {
|
|
|
|
|
/* Mask attributes check failed. */
|
|
|
|
|
OVS_NLERR("Contain more than allowed mask fields (mask_attrs=%llx, mask_allowed=%llx).\n",
|
|
|
|
|
(unsigned long long)mask_attrs, (unsigned long long)mask_allowed);
|
|
|
|
|
OVS_NLERR(log, "Unexpected mask (mask=%llx, allowed=%llx)",
|
|
|
|
|
(unsigned long long)mask_attrs,
|
|
|
|
|
(unsigned long long)mask_allowed);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -328,7 +330,7 @@ static bool is_all_zero(const u8 *fp, size_t size)
|
|
|
|
|
|
|
|
|
|
static int __parse_flow_nlattrs(const struct nlattr *attr,
|
|
|
|
|
const struct nlattr *a[],
|
|
|
|
|
u64 *attrsp, bool nz)
|
|
|
|
|
u64 *attrsp, bool log, bool nz)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *nla;
|
|
|
|
|
u64 attrs;
|
|
|
|
@ -340,21 +342,20 @@ static int __parse_flow_nlattrs(const struct nlattr *attr,
|
|
|
|
|
int expected_len;
|
|
|
|
|
|
|
|
|
|
if (type > OVS_KEY_ATTR_MAX) {
|
|
|
|
|
OVS_NLERR("Unknown key attribute (type=%d, max=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Key type %d is out of range max %d",
|
|
|
|
|
type, OVS_KEY_ATTR_MAX);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attrs & (1 << type)) {
|
|
|
|
|
OVS_NLERR("Duplicate key attribute (type %d).\n", type);
|
|
|
|
|
OVS_NLERR(log, "Duplicate key (type %d).", type);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
expected_len = ovs_key_lens[type];
|
|
|
|
|
if (nla_len(nla) != expected_len && expected_len != -1) {
|
|
|
|
|
OVS_NLERR("Key attribute has unexpected length (type=%d"
|
|
|
|
|
", length=%d, expected=%d).\n", type,
|
|
|
|
|
nla_len(nla), expected_len);
|
|
|
|
|
OVS_NLERR(log, "Key %d has unexpected len %d expected %d",
|
|
|
|
|
type, nla_len(nla), expected_len);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -364,7 +365,7 @@ static int __parse_flow_nlattrs(const struct nlattr *attr,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rem) {
|
|
|
|
|
OVS_NLERR("Message has %d unknown bytes.\n", rem);
|
|
|
|
|
OVS_NLERR(log, "Message has %d unknown bytes.", rem);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -373,28 +374,84 @@ static int __parse_flow_nlattrs(const struct nlattr *attr,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int parse_flow_mask_nlattrs(const struct nlattr *attr,
|
|
|
|
|
const struct nlattr *a[], u64 *attrsp)
|
|
|
|
|
const struct nlattr *a[], u64 *attrsp,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
return __parse_flow_nlattrs(attr, a, attrsp, true);
|
|
|
|
|
return __parse_flow_nlattrs(attr, a, attrsp, log, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int parse_flow_nlattrs(const struct nlattr *attr,
|
|
|
|
|
const struct nlattr *a[], u64 *attrsp)
|
|
|
|
|
const struct nlattr *a[], u64 *attrsp,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
return __parse_flow_nlattrs(attr, a, attrsp, false);
|
|
|
|
|
return __parse_flow_nlattrs(attr, a, attrsp, log, false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int genev_tun_opt_from_nlattr(const struct nlattr *a,
|
|
|
|
|
struct sw_flow_match *match, bool is_mask,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
unsigned long opt_key_offset;
|
|
|
|
|
|
|
|
|
|
if (nla_len(a) > sizeof(match->key->tun_opts)) {
|
|
|
|
|
OVS_NLERR(log, "Geneve option length err (len %d, max %zu).",
|
|
|
|
|
nla_len(a), sizeof(match->key->tun_opts));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (nla_len(a) % 4 != 0) {
|
|
|
|
|
OVS_NLERR(log, "Geneve opt len %d is not a multiple of 4.",
|
|
|
|
|
nla_len(a));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* We need to record the length of the options passed
|
|
|
|
|
* down, otherwise packets with the same format but
|
|
|
|
|
* additional options will be silently matched.
|
|
|
|
|
*/
|
|
|
|
|
if (!is_mask) {
|
|
|
|
|
SW_FLOW_KEY_PUT(match, tun_opts_len, nla_len(a),
|
|
|
|
|
false);
|
|
|
|
|
} else {
|
|
|
|
|
/* This is somewhat unusual because it looks at
|
|
|
|
|
* both the key and mask while parsing the
|
|
|
|
|
* attributes (and by extension assumes the key
|
|
|
|
|
* is parsed first). Normally, we would verify
|
|
|
|
|
* that each is the correct length and that the
|
|
|
|
|
* attributes line up in the validate function.
|
|
|
|
|
* However, that is difficult because this is
|
|
|
|
|
* variable length and we won't have the
|
|
|
|
|
* information later.
|
|
|
|
|
*/
|
|
|
|
|
if (match->key->tun_opts_len != nla_len(a)) {
|
|
|
|
|
OVS_NLERR(log, "Geneve option len %d != mask len %d",
|
|
|
|
|
match->key->tun_opts_len, nla_len(a));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SW_FLOW_KEY_PUT(match, tun_opts_len, 0xff, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opt_key_offset = (unsigned long)GENEVE_OPTS((struct sw_flow_key *)0,
|
|
|
|
|
nla_len(a));
|
|
|
|
|
SW_FLOW_KEY_MEMCPY_OFFSET(match, opt_key_offset, nla_data(a),
|
|
|
|
|
nla_len(a), is_mask);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int ipv4_tun_from_nlattr(const struct nlattr *attr,
|
|
|
|
|
struct sw_flow_match *match, bool is_mask)
|
|
|
|
|
struct sw_flow_match *match, bool is_mask,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
struct nlattr *a;
|
|
|
|
|
int rem;
|
|
|
|
|
bool ttl = false;
|
|
|
|
|
__be16 tun_flags = 0;
|
|
|
|
|
unsigned long opt_key_offset;
|
|
|
|
|
|
|
|
|
|
nla_for_each_nested(a, attr, rem) {
|
|
|
|
|
int type = nla_type(a);
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
static const u32 ovs_tunnel_key_lens[OVS_TUNNEL_KEY_ATTR_MAX + 1] = {
|
|
|
|
|
[OVS_TUNNEL_KEY_ATTR_ID] = sizeof(u64),
|
|
|
|
|
[OVS_TUNNEL_KEY_ATTR_IPV4_SRC] = sizeof(u32),
|
|
|
|
@ -410,15 +467,14 @@ static int ipv4_tun_from_nlattr(const struct nlattr *attr,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (type > OVS_TUNNEL_KEY_ATTR_MAX) {
|
|
|
|
|
OVS_NLERR("Unknown IPv4 tunnel attribute (type=%d, max=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Tunnel attr %d out of range max %d",
|
|
|
|
|
type, OVS_TUNNEL_KEY_ATTR_MAX);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ovs_tunnel_key_lens[type] != nla_len(a) &&
|
|
|
|
|
ovs_tunnel_key_lens[type] != -1) {
|
|
|
|
|
OVS_NLERR("IPv4 tunnel attribute type has unexpected "
|
|
|
|
|
" length (type=%d, length=%d, expected=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Tunnel attr %d has unexpected len %d expected %d",
|
|
|
|
|
type, nla_len(a), ovs_tunnel_key_lens[type]);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -464,58 +520,14 @@ static int ipv4_tun_from_nlattr(const struct nlattr *attr,
|
|
|
|
|
tun_flags |= TUNNEL_OAM;
|
|
|
|
|
break;
|
|
|
|
|
case OVS_TUNNEL_KEY_ATTR_GENEVE_OPTS:
|
|
|
|
|
err = genev_tun_opt_from_nlattr(a, match, is_mask, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
tun_flags |= TUNNEL_OPTIONS_PRESENT;
|
|
|
|
|
if (nla_len(a) > sizeof(match->key->tun_opts)) {
|
|
|
|
|
OVS_NLERR("Geneve option length exceeds maximum size (len %d, max %zu).\n",
|
|
|
|
|
nla_len(a),
|
|
|
|
|
sizeof(match->key->tun_opts));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (nla_len(a) % 4 != 0) {
|
|
|
|
|
OVS_NLERR("Geneve option length is not a multiple of 4 (len %d).\n",
|
|
|
|
|
nla_len(a));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* We need to record the length of the options passed
|
|
|
|
|
* down, otherwise packets with the same format but
|
|
|
|
|
* additional options will be silently matched.
|
|
|
|
|
*/
|
|
|
|
|
if (!is_mask) {
|
|
|
|
|
SW_FLOW_KEY_PUT(match, tun_opts_len, nla_len(a),
|
|
|
|
|
false);
|
|
|
|
|
} else {
|
|
|
|
|
/* This is somewhat unusual because it looks at
|
|
|
|
|
* both the key and mask while parsing the
|
|
|
|
|
* attributes (and by extension assumes the key
|
|
|
|
|
* is parsed first). Normally, we would verify
|
|
|
|
|
* that each is the correct length and that the
|
|
|
|
|
* attributes line up in the validate function.
|
|
|
|
|
* However, that is difficult because this is
|
|
|
|
|
* variable length and we won't have the
|
|
|
|
|
* information later.
|
|
|
|
|
*/
|
|
|
|
|
if (match->key->tun_opts_len != nla_len(a)) {
|
|
|
|
|
OVS_NLERR("Geneve option key length (%d) is different from mask length (%d).",
|
|
|
|
|
match->key->tun_opts_len,
|
|
|
|
|
nla_len(a));
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SW_FLOW_KEY_PUT(match, tun_opts_len, 0xff,
|
|
|
|
|
true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opt_key_offset = (unsigned long)GENEVE_OPTS(
|
|
|
|
|
(struct sw_flow_key *)0,
|
|
|
|
|
nla_len(a));
|
|
|
|
|
SW_FLOW_KEY_MEMCPY_OFFSET(match, opt_key_offset,
|
|
|
|
|
nla_data(a), nla_len(a),
|
|
|
|
|
is_mask);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
OVS_NLERR("Unknown IPv4 tunnel attribute (%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Unknown IPv4 tunnel attribute %d",
|
|
|
|
|
type);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -524,18 +536,19 @@ static int ipv4_tun_from_nlattr(const struct nlattr *attr,
|
|
|
|
|
SW_FLOW_KEY_PUT(match, tun_key.tun_flags, tun_flags, is_mask);
|
|
|
|
|
|
|
|
|
|
if (rem > 0) {
|
|
|
|
|
OVS_NLERR("IPv4 tunnel attribute has %d unknown bytes.\n", rem);
|
|
|
|
|
OVS_NLERR(log, "IPv4 tunnel attribute has %d unknown bytes.",
|
|
|
|
|
rem);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!is_mask) {
|
|
|
|
|
if (!match->key->tun_key.ipv4_dst) {
|
|
|
|
|
OVS_NLERR("IPv4 tunnel destination address is zero.\n");
|
|
|
|
|
OVS_NLERR(log, "IPv4 tunnel dst address is zero");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!ttl) {
|
|
|
|
|
OVS_NLERR("IPv4 tunnel TTL not specified.\n");
|
|
|
|
|
OVS_NLERR(log, "IPv4 tunnel TTL not specified.");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -614,7 +627,8 @@ int ovs_nla_put_egress_tunnel_key(struct sk_buff *skb,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int metadata_from_nlattrs(struct sw_flow_match *match, u64 *attrs,
|
|
|
|
|
const struct nlattr **a, bool is_mask)
|
|
|
|
|
const struct nlattr **a, bool is_mask,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
if (*attrs & (1 << OVS_KEY_ATTR_DP_HASH)) {
|
|
|
|
|
u32 hash_val = nla_get_u32(a[OVS_KEY_ATTR_DP_HASH]);
|
|
|
|
@ -642,7 +656,7 @@ static int metadata_from_nlattrs(struct sw_flow_match *match, u64 *attrs,
|
|
|
|
|
if (is_mask) {
|
|
|
|
|
in_port = 0xffffffff; /* Always exact match in_port. */
|
|
|
|
|
} else if (in_port >= DP_MAX_PORTS) {
|
|
|
|
|
OVS_NLERR("Port (%d) exceeds maximum allowable (%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Port %d exceeds max allowable %d",
|
|
|
|
|
in_port, DP_MAX_PORTS);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -661,7 +675,7 @@ static int metadata_from_nlattrs(struct sw_flow_match *match, u64 *attrs,
|
|
|
|
|
}
|
|
|
|
|
if (*attrs & (1 << OVS_KEY_ATTR_TUNNEL)) {
|
|
|
|
|
if (ipv4_tun_from_nlattr(a[OVS_KEY_ATTR_TUNNEL], match,
|
|
|
|
|
is_mask))
|
|
|
|
|
is_mask, log))
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
*attrs &= ~(1 << OVS_KEY_ATTR_TUNNEL);
|
|
|
|
|
}
|
|
|
|
@ -669,11 +683,12 @@ static int metadata_from_nlattrs(struct sw_flow_match *match, u64 *attrs,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
const struct nlattr **a, bool is_mask)
|
|
|
|
|
const struct nlattr **a, bool is_mask,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = metadata_from_nlattrs(match, &attrs, a, is_mask);
|
|
|
|
|
err = metadata_from_nlattrs(match, &attrs, a, is_mask, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -694,9 +709,9 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
tci = nla_get_be16(a[OVS_KEY_ATTR_VLAN]);
|
|
|
|
|
if (!(tci & htons(VLAN_TAG_PRESENT))) {
|
|
|
|
|
if (is_mask)
|
|
|
|
|
OVS_NLERR("VLAN TCI mask does not have exact match for VLAN_TAG_PRESENT bit.\n");
|
|
|
|
|
OVS_NLERR(log, "VLAN TCI mask does not have exact match for VLAN_TAG_PRESENT bit.");
|
|
|
|
|
else
|
|
|
|
|
OVS_NLERR("VLAN TCI does not have VLAN_TAG_PRESENT bit set.\n");
|
|
|
|
|
OVS_NLERR(log, "VLAN TCI does not have VLAN_TAG_PRESENT bit set.");
|
|
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -713,7 +728,7 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
/* Always exact match EtherType. */
|
|
|
|
|
eth_type = htons(0xffff);
|
|
|
|
|
} else if (ntohs(eth_type) < ETH_P_802_3_MIN) {
|
|
|
|
|
OVS_NLERR("EtherType is less than minimum (type=%x, min=%x).\n",
|
|
|
|
|
OVS_NLERR(log, "EtherType %x is less than min %x",
|
|
|
|
|
ntohs(eth_type), ETH_P_802_3_MIN);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -729,7 +744,7 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
|
|
|
|
|
ipv4_key = nla_data(a[OVS_KEY_ATTR_IPV4]);
|
|
|
|
|
if (!is_mask && ipv4_key->ipv4_frag > OVS_FRAG_TYPE_MAX) {
|
|
|
|
|
OVS_NLERR("Unknown IPv4 fragment type (value=%d, max=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "IPv4 frag type %d is out of range max %d",
|
|
|
|
|
ipv4_key->ipv4_frag, OVS_FRAG_TYPE_MAX);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -753,7 +768,7 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
|
|
|
|
|
ipv6_key = nla_data(a[OVS_KEY_ATTR_IPV6]);
|
|
|
|
|
if (!is_mask && ipv6_key->ipv6_frag > OVS_FRAG_TYPE_MAX) {
|
|
|
|
|
OVS_NLERR("Unknown IPv6 fragment type (value=%d, max=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "IPv6 frag type %d is out of range max %d",
|
|
|
|
|
ipv6_key->ipv6_frag, OVS_FRAG_TYPE_MAX);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -784,7 +799,7 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
|
|
|
|
|
arp_key = nla_data(a[OVS_KEY_ATTR_ARP]);
|
|
|
|
|
if (!is_mask && (arp_key->arp_op & htons(0xff00))) {
|
|
|
|
|
OVS_NLERR("Unknown ARP opcode (opcode=%d).\n",
|
|
|
|
|
OVS_NLERR(log, "Unknown ARP opcode (opcode=%d).",
|
|
|
|
|
arp_key->arp_op);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -885,7 +900,7 @@ static int ovs_key_from_nlattrs(struct sw_flow_match *match, u64 attrs,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attrs != 0) {
|
|
|
|
|
OVS_NLERR("Unknown key attributes (%llx).\n",
|
|
|
|
|
OVS_NLERR(log, "Unknown key attributes %llx",
|
|
|
|
|
(unsigned long long)attrs);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
@ -926,10 +941,14 @@ static void mask_set_nlattr(struct nlattr *attr, u8 val)
|
|
|
|
|
* of this flow.
|
|
|
|
|
* @mask: Optional. Netlink attribute holding nested %OVS_KEY_ATTR_* Netlink
|
|
|
|
|
* attribute specifies the mask field of the wildcarded flow.
|
|
|
|
|
* @log: Boolean to allow kernel error logging. Normally true, but when
|
|
|
|
|
* probing for feature compatibility this should be passed in as false to
|
|
|
|
|
* suppress unnecessary error logging.
|
|
|
|
|
*/
|
|
|
|
|
int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
const struct nlattr *nla_key,
|
|
|
|
|
const struct nlattr *nla_mask)
|
|
|
|
|
const struct nlattr *nla_mask,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *a[OVS_KEY_ATTR_MAX + 1];
|
|
|
|
|
const struct nlattr *encap;
|
|
|
|
@ -939,7 +958,7 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
bool encap_valid = false;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = parse_flow_nlattrs(nla_key, a, &key_attrs);
|
|
|
|
|
err = parse_flow_nlattrs(nla_key, a, &key_attrs, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -950,7 +969,7 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
|
|
|
|
|
if (!((key_attrs & (1 << OVS_KEY_ATTR_VLAN)) &&
|
|
|
|
|
(key_attrs & (1 << OVS_KEY_ATTR_ENCAP)))) {
|
|
|
|
|
OVS_NLERR("Invalid Vlan frame.\n");
|
|
|
|
|
OVS_NLERR(log, "Invalid Vlan frame.");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -961,22 +980,22 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
encap_valid = true;
|
|
|
|
|
|
|
|
|
|
if (tci & htons(VLAN_TAG_PRESENT)) {
|
|
|
|
|
err = parse_flow_nlattrs(encap, a, &key_attrs);
|
|
|
|
|
err = parse_flow_nlattrs(encap, a, &key_attrs, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
} else if (!tci) {
|
|
|
|
|
/* Corner case for truncated 802.1Q header. */
|
|
|
|
|
if (nla_len(encap)) {
|
|
|
|
|
OVS_NLERR("Truncated 802.1Q header has non-zero encap attribute.\n");
|
|
|
|
|
OVS_NLERR(log, "Truncated 802.1Q header has non-zero encap attribute.");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
OVS_NLERR("Encap attribute is set for a non-VLAN frame.\n");
|
|
|
|
|
OVS_NLERR(log, "Encap attr is set for non-VLAN frame");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = ovs_key_from_nlattrs(match, key_attrs, a, false);
|
|
|
|
|
err = ovs_key_from_nlattrs(match, key_attrs, a, false, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -1010,7 +1029,7 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
nla_mask = newmask;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = parse_flow_mask_nlattrs(nla_mask, a, &mask_attrs);
|
|
|
|
|
err = parse_flow_mask_nlattrs(nla_mask, a, &mask_attrs, log);
|
|
|
|
|
if (err)
|
|
|
|
|
goto free_newmask;
|
|
|
|
|
|
|
|
|
@ -1022,7 +1041,7 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
__be16 tci = 0;
|
|
|
|
|
|
|
|
|
|
if (!encap_valid) {
|
|
|
|
|
OVS_NLERR("Encap mask attribute is set for non-VLAN frame.\n");
|
|
|
|
|
OVS_NLERR(log, "Encap mask attribute is set for non-VLAN frame.");
|
|
|
|
|
err = -EINVAL;
|
|
|
|
|
goto free_newmask;
|
|
|
|
|
}
|
|
|
|
@ -1034,11 +1053,12 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
if (eth_type == htons(0xffff)) {
|
|
|
|
|
mask_attrs &= ~(1 << OVS_KEY_ATTR_ETHERTYPE);
|
|
|
|
|
encap = a[OVS_KEY_ATTR_ENCAP];
|
|
|
|
|
err = parse_flow_mask_nlattrs(encap, a, &mask_attrs);
|
|
|
|
|
err = parse_flow_mask_nlattrs(encap, a,
|
|
|
|
|
&mask_attrs, log);
|
|
|
|
|
if (err)
|
|
|
|
|
goto free_newmask;
|
|
|
|
|
} else {
|
|
|
|
|
OVS_NLERR("VLAN frames must have an exact match on the TPID (mask=%x).\n",
|
|
|
|
|
OVS_NLERR(log, "VLAN frames must have an exact match on the TPID (mask=%x).",
|
|
|
|
|
ntohs(eth_type));
|
|
|
|
|
err = -EINVAL;
|
|
|
|
|
goto free_newmask;
|
|
|
|
@ -1048,18 +1068,19 @@ int ovs_nla_get_match(struct sw_flow_match *match,
|
|
|
|
|
tci = nla_get_be16(a[OVS_KEY_ATTR_VLAN]);
|
|
|
|
|
|
|
|
|
|
if (!(tci & htons(VLAN_TAG_PRESENT))) {
|
|
|
|
|
OVS_NLERR("VLAN tag present bit must have an exact match (tci_mask=%x).\n", ntohs(tci));
|
|
|
|
|
OVS_NLERR(log, "VLAN tag present bit must have an exact match (tci_mask=%x).",
|
|
|
|
|
ntohs(tci));
|
|
|
|
|
err = -EINVAL;
|
|
|
|
|
goto free_newmask;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = ovs_key_from_nlattrs(match, mask_attrs, a, true);
|
|
|
|
|
err = ovs_key_from_nlattrs(match, mask_attrs, a, true, log);
|
|
|
|
|
if (err)
|
|
|
|
|
goto free_newmask;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!match_validate(match, key_attrs, mask_attrs))
|
|
|
|
|
if (!match_validate(match, key_attrs, mask_attrs, log))
|
|
|
|
|
err = -EINVAL;
|
|
|
|
|
|
|
|
|
|
free_newmask:
|
|
|
|
@ -1072,6 +1093,9 @@ free_newmask:
|
|
|
|
|
* @key: Receives extracted in_port, priority, tun_key and skb_mark.
|
|
|
|
|
* @attr: Netlink attribute holding nested %OVS_KEY_ATTR_* Netlink attribute
|
|
|
|
|
* sequence.
|
|
|
|
|
* @log: Boolean to allow kernel error logging. Normally true, but when
|
|
|
|
|
* probing for feature compatibility this should be passed in as false to
|
|
|
|
|
* suppress unnecessary error logging.
|
|
|
|
|
*
|
|
|
|
|
* This parses a series of Netlink attributes that form a flow key, which must
|
|
|
|
|
* take the same form accepted by flow_from_nlattrs(), but only enough of it to
|
|
|
|
@ -1080,14 +1104,15 @@ free_newmask:
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
int ovs_nla_get_flow_metadata(const struct nlattr *attr,
|
|
|
|
|
struct sw_flow_key *key)
|
|
|
|
|
struct sw_flow_key *key,
|
|
|
|
|
bool log)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *a[OVS_KEY_ATTR_MAX + 1];
|
|
|
|
|
struct sw_flow_match match;
|
|
|
|
|
u64 attrs = 0;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = parse_flow_nlattrs(attr, a, &attrs);
|
|
|
|
|
err = parse_flow_nlattrs(attr, a, &attrs, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
|
@ -1096,7 +1121,7 @@ int ovs_nla_get_flow_metadata(const struct nlattr *attr,
|
|
|
|
|
|
|
|
|
|
key->phy.in_port = DP_MAX_PORTS;
|
|
|
|
|
|
|
|
|
|
return metadata_from_nlattrs(&match, &attrs, a, false);
|
|
|
|
|
return metadata_from_nlattrs(&match, &attrs, a, false, log);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ovs_nla_put_flow(const struct sw_flow_key *swkey,
|
|
|
|
@ -1316,12 +1341,12 @@ nla_put_failure:
|
|
|
|
|
|
|
|
|
|
#define MAX_ACTIONS_BUFSIZE (32 * 1024)
|
|
|
|
|
|
|
|
|
|
static struct sw_flow_actions *nla_alloc_flow_actions(int size)
|
|
|
|
|
static struct sw_flow_actions *nla_alloc_flow_actions(int size, bool log)
|
|
|
|
|
{
|
|
|
|
|
struct sw_flow_actions *sfa;
|
|
|
|
|
|
|
|
|
|
if (size > MAX_ACTIONS_BUFSIZE) {
|
|
|
|
|
OVS_NLERR("Flow action size (%u bytes) exceeds maximum", size);
|
|
|
|
|
OVS_NLERR(log, "Flow action size %u bytes exceeds max", size);
|
|
|
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -1341,7 +1366,7 @@ void ovs_nla_free_flow_actions(struct sw_flow_actions *sf_acts)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct nlattr *reserve_sfa_size(struct sw_flow_actions **sfa,
|
|
|
|
|
int attr_len)
|
|
|
|
|
int attr_len, bool log)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
struct sw_flow_actions *acts;
|
|
|
|
@ -1361,7 +1386,7 @@ static struct nlattr *reserve_sfa_size(struct sw_flow_actions **sfa,
|
|
|
|
|
new_acts_size = MAX_ACTIONS_BUFSIZE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
acts = nla_alloc_flow_actions(new_acts_size);
|
|
|
|
|
acts = nla_alloc_flow_actions(new_acts_size, log);
|
|
|
|
|
if (IS_ERR(acts))
|
|
|
|
|
return (void *)acts;
|
|
|
|
|
|
|
|
|
@ -1376,11 +1401,11 @@ out:
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static struct nlattr *__add_action(struct sw_flow_actions **sfa,
|
|
|
|
|
int attrtype, void *data, int len)
|
|
|
|
|
int attrtype, void *data, int len, bool log)
|
|
|
|
|
{
|
|
|
|
|
struct nlattr *a;
|
|
|
|
|
|
|
|
|
|
a = reserve_sfa_size(sfa, nla_attr_size(len));
|
|
|
|
|
a = reserve_sfa_size(sfa, nla_attr_size(len), log);
|
|
|
|
|
if (IS_ERR(a))
|
|
|
|
|
return a;
|
|
|
|
|
|
|
|
|
@ -1395,11 +1420,11 @@ static struct nlattr *__add_action(struct sw_flow_actions **sfa,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int add_action(struct sw_flow_actions **sfa, int attrtype,
|
|
|
|
|
void *data, int len)
|
|
|
|
|
void *data, int len, bool log)
|
|
|
|
|
{
|
|
|
|
|
struct nlattr *a;
|
|
|
|
|
|
|
|
|
|
a = __add_action(sfa, attrtype, data, len);
|
|
|
|
|
a = __add_action(sfa, attrtype, data, len, log);
|
|
|
|
|
if (IS_ERR(a))
|
|
|
|
|
return PTR_ERR(a);
|
|
|
|
|
|
|
|
|
@ -1407,12 +1432,12 @@ static int add_action(struct sw_flow_actions **sfa, int attrtype,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline int add_nested_action_start(struct sw_flow_actions **sfa,
|
|
|
|
|
int attrtype)
|
|
|
|
|
int attrtype, bool log)
|
|
|
|
|
{
|
|
|
|
|
int used = (*sfa)->actions_len;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = add_action(sfa, attrtype, NULL, 0);
|
|
|
|
|
err = add_action(sfa, attrtype, NULL, 0, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -1431,12 +1456,12 @@ static inline void add_nested_action_end(struct sw_flow_actions *sfa,
|
|
|
|
|
static int __ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
const struct sw_flow_key *key,
|
|
|
|
|
int depth, struct sw_flow_actions **sfa,
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci);
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci, bool log);
|
|
|
|
|
|
|
|
|
|
static int validate_and_copy_sample(const struct nlattr *attr,
|
|
|
|
|
const struct sw_flow_key *key, int depth,
|
|
|
|
|
struct sw_flow_actions **sfa,
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci)
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci, bool log)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *attrs[OVS_SAMPLE_ATTR_MAX + 1];
|
|
|
|
|
const struct nlattr *probability, *actions;
|
|
|
|
@ -1462,19 +1487,19 @@ static int validate_and_copy_sample(const struct nlattr *attr,
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
|
|
/* validation done, copy sample action. */
|
|
|
|
|
start = add_nested_action_start(sfa, OVS_ACTION_ATTR_SAMPLE);
|
|
|
|
|
start = add_nested_action_start(sfa, OVS_ACTION_ATTR_SAMPLE, log);
|
|
|
|
|
if (start < 0)
|
|
|
|
|
return start;
|
|
|
|
|
err = add_action(sfa, OVS_SAMPLE_ATTR_PROBABILITY,
|
|
|
|
|
nla_data(probability), sizeof(u32));
|
|
|
|
|
nla_data(probability), sizeof(u32), log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
st_acts = add_nested_action_start(sfa, OVS_SAMPLE_ATTR_ACTIONS);
|
|
|
|
|
st_acts = add_nested_action_start(sfa, OVS_SAMPLE_ATTR_ACTIONS, log);
|
|
|
|
|
if (st_acts < 0)
|
|
|
|
|
return st_acts;
|
|
|
|
|
|
|
|
|
|
err = __ovs_nla_copy_actions(actions, key, depth + 1, sfa,
|
|
|
|
|
eth_type, vlan_tci);
|
|
|
|
|
eth_type, vlan_tci, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -1511,7 +1536,7 @@ void ovs_match_init(struct sw_flow_match *match,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int validate_and_copy_set_tun(const struct nlattr *attr,
|
|
|
|
|
struct sw_flow_actions **sfa)
|
|
|
|
|
struct sw_flow_actions **sfa, bool log)
|
|
|
|
|
{
|
|
|
|
|
struct sw_flow_match match;
|
|
|
|
|
struct sw_flow_key key;
|
|
|
|
@ -1520,7 +1545,7 @@ static int validate_and_copy_set_tun(const struct nlattr *attr,
|
|
|
|
|
int err, start;
|
|
|
|
|
|
|
|
|
|
ovs_match_init(&match, &key, NULL);
|
|
|
|
|
err = ipv4_tun_from_nlattr(nla_data(attr), &match, false);
|
|
|
|
|
err = ipv4_tun_from_nlattr(nla_data(attr), &match, false, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -1549,12 +1574,12 @@ static int validate_and_copy_set_tun(const struct nlattr *attr,
|
|
|
|
|
key.tun_key.tun_flags |= crit_opt ? TUNNEL_CRIT_OPT : 0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
start = add_nested_action_start(sfa, OVS_ACTION_ATTR_SET);
|
|
|
|
|
start = add_nested_action_start(sfa, OVS_ACTION_ATTR_SET, log);
|
|
|
|
|
if (start < 0)
|
|
|
|
|
return start;
|
|
|
|
|
|
|
|
|
|
a = __add_action(sfa, OVS_KEY_ATTR_TUNNEL_INFO, NULL,
|
|
|
|
|
sizeof(*tun_info) + key.tun_opts_len);
|
|
|
|
|
sizeof(*tun_info) + key.tun_opts_len, log);
|
|
|
|
|
if (IS_ERR(a))
|
|
|
|
|
return PTR_ERR(a);
|
|
|
|
|
|
|
|
|
@ -1582,7 +1607,7 @@ static int validate_and_copy_set_tun(const struct nlattr *attr,
|
|
|
|
|
static int validate_set(const struct nlattr *a,
|
|
|
|
|
const struct sw_flow_key *flow_key,
|
|
|
|
|
struct sw_flow_actions **sfa,
|
|
|
|
|
bool *set_tun, __be16 eth_type)
|
|
|
|
|
bool *set_tun, __be16 eth_type, bool log)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *ovs_key = nla_data(a);
|
|
|
|
|
int key_type = nla_type(ovs_key);
|
|
|
|
@ -1611,7 +1636,7 @@ static int validate_set(const struct nlattr *a,
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
|
|
*set_tun = true;
|
|
|
|
|
err = validate_and_copy_set_tun(a, sfa);
|
|
|
|
|
err = validate_and_copy_set_tun(a, sfa, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
break;
|
|
|
|
@ -1704,12 +1729,12 @@ static int validate_userspace(const struct nlattr *attr)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int copy_action(const struct nlattr *from,
|
|
|
|
|
struct sw_flow_actions **sfa)
|
|
|
|
|
struct sw_flow_actions **sfa, bool log)
|
|
|
|
|
{
|
|
|
|
|
int totlen = NLA_ALIGN(from->nla_len);
|
|
|
|
|
struct nlattr *to;
|
|
|
|
|
|
|
|
|
|
to = reserve_sfa_size(sfa, from->nla_len);
|
|
|
|
|
to = reserve_sfa_size(sfa, from->nla_len, log);
|
|
|
|
|
if (IS_ERR(to))
|
|
|
|
|
return PTR_ERR(to);
|
|
|
|
|
|
|
|
|
@ -1720,7 +1745,7 @@ static int copy_action(const struct nlattr *from,
|
|
|
|
|
static int __ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
const struct sw_flow_key *key,
|
|
|
|
|
int depth, struct sw_flow_actions **sfa,
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci)
|
|
|
|
|
__be16 eth_type, __be16 vlan_tci, bool log)
|
|
|
|
|
{
|
|
|
|
|
const struct nlattr *a;
|
|
|
|
|
bool out_tnl_port = false;
|
|
|
|
@ -1843,7 +1868,7 @@ static int __ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
|
|
|
|
|
case OVS_ACTION_ATTR_SET:
|
|
|
|
|
err = validate_set(a, key, sfa,
|
|
|
|
|
&out_tnl_port, eth_type);
|
|
|
|
|
&out_tnl_port, eth_type, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
@ -1852,18 +1877,18 @@ static int __ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
|
|
|
|
|
case OVS_ACTION_ATTR_SAMPLE:
|
|
|
|
|
err = validate_and_copy_sample(a, key, depth, sfa,
|
|
|
|
|
eth_type, vlan_tci);
|
|
|
|
|
eth_type, vlan_tci, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
skip_copy = true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
OVS_NLERR("Unknown tunnel attribute (%d).\n", type);
|
|
|
|
|
OVS_NLERR(log, "Unknown Action type %d", type);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (!skip_copy) {
|
|
|
|
|
err = copy_action(a, sfa);
|
|
|
|
|
err = copy_action(a, sfa, log);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
@ -1877,16 +1902,16 @@ static int __ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
|
|
|
|
|
int ovs_nla_copy_actions(const struct nlattr *attr,
|
|
|
|
|
const struct sw_flow_key *key,
|
|
|
|
|
struct sw_flow_actions **sfa)
|
|
|
|
|
struct sw_flow_actions **sfa, bool log)
|
|
|
|
|
{
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
*sfa = nla_alloc_flow_actions(nla_len(attr));
|
|
|
|
|
*sfa = nla_alloc_flow_actions(nla_len(attr), log);
|
|
|
|
|
if (IS_ERR(*sfa))
|
|
|
|
|
return PTR_ERR(*sfa);
|
|
|
|
|
|
|
|
|
|
err = __ovs_nla_copy_actions(attr, key, 0, sfa, key->eth.type,
|
|
|
|
|
key->eth.tci);
|
|
|
|
|
key->eth.tci, log);
|
|
|
|
|
if (err)
|
|
|
|
|
kfree(*sfa);
|
|
|
|
|
|
|
|
|
|