Commit 9f1179fb authored by Adrian Moreno's avatar Adrian Moreno Committed by Paolo Abeni

selftests: openvswitch: support key masks

The default value for the mask actually depends on the value (e.g: if
the value is non-null, the default is full-mask), so change the convert
functions to accept the full, possibly masked string and let them figure
out how to parse the different values.

Also, implement size-aware int parsing.

With this patch we can now express flows such as the following:
"eth(src=0a:ca:fe:ca:fe:0a/ff:ff:00:00:ff:00)"
"eth(src=0a:ca:fe:ca:fe:0a)" -> mask = ff:ff:ff:ff:ff:ff
"ipv4(src=192.168.1.1)" -> mask = 255.255.255.255
"ipv4(src=192.168.1.1/24)"
"ipv4(src=192.168.1.1/255.255.255.0)"
"tcp(src=8080)" -> mask = 0xffff
"tcp(src=8080/0xf0f0)"
Signed-off-by: default avatarAdrian Moreno <amorenoz@redhat.com>
Acked-by: default avatarAaron Conole <aconole@redhat.com>
Reviewed-by: default avatarSimon Horman <horms@kernel.org>
Signed-off-by: default avatarPaolo Abeni <pabeni@redhat.com>
parent 918423fd
...@@ -160,25 +160,45 @@ def parse_ct_state(statestr): ...@@ -160,25 +160,45 @@ def parse_ct_state(statestr):
return parse_flags(statestr, ct_flags) return parse_flags(statestr, ct_flags)
def convert_mac(mac_str, mask=False): def convert_mac(data):
if mac_str is None or mac_str == "": def to_bytes(mac):
mac_str = "00:00:00:00:00:00" mac_split = mac.split(":")
if mask is True and mac_str != "00:00:00:00:00:00": ret = bytearray([int(i, 16) for i in mac_split])
mac_str = "FF:FF:FF:FF:FF:FF" return bytes(ret)
mac_split = mac_str.split(":")
ret = bytearray([int(i, 16) for i in mac_split])
return bytes(ret)
mac_str, _, mask_str = data.partition('/')
def convert_ipv4(ip, mask=False): if not mac_str:
if ip is None: mac_str = mask_str = "00:00:00:00:00:00"
ip = 0 elif not mask_str:
if mask is True: mask_str = "FF:FF:FF:FF:FF:FF"
if ip != 0:
ip = int(ipaddress.IPv4Address(ip)) & 0xFFFFFFFF
return int(ipaddress.IPv4Address(ip)) return to_bytes(mac_str), to_bytes(mask_str)
def convert_ipv4(data):
ip, _, mask = data.partition('/')
if not ip:
ip = mask = 0
elif not mask:
mask = 0xFFFFFFFF
elif mask.isdigit():
mask = (0xFFFFFFFF << (32 - int(mask))) & 0xFFFFFFFF
return int(ipaddress.IPv4Address(ip)), int(ipaddress.IPv4Address(mask))
def convert_int(size):
def convert_int_sized(data):
value, _, mask = data.partition('/')
if not value:
return 0, 0
elif not mask:
return int(value, 0), pow(2, size) - 1
else:
return int(value, 0), int(mask, 0)
return convert_int_sized
def parse_starts_block(block_str, scanstr, returnskipped, scanregex=False): def parse_starts_block(block_str, scanstr, returnskipped, scanregex=False):
if scanregex: if scanregex:
...@@ -525,8 +545,10 @@ class ovskey(nla): ...@@ -525,8 +545,10 @@ class ovskey(nla):
) )
fields_map = ( fields_map = (
("src", "src", "%d", lambda x: int(x) if x is not None else 0), ("src", "src", "%d", lambda x: int(x) if x else 0,
("dst", "dst", "%d", lambda x: int(x) if x is not None else 0), convert_int(16)),
("dst", "dst", "%d", lambda x: int(x) if x else 0,
convert_int(16)),
) )
def __init__( def __init__(
...@@ -575,17 +597,13 @@ class ovskey(nla): ...@@ -575,17 +597,13 @@ class ovskey(nla):
data = flowstr[:splitchar] data = flowstr[:splitchar]
flowstr = flowstr[splitchar:] flowstr = flowstr[splitchar:]
else: else:
data = None data = ""
if len(f) > 4: if len(f) > 4:
func = f[4] k[f[0]], m[f[0]] = f[4](data)
else:
func = f[3]
k[f[0]] = func(data)
if len(f) > 4:
m[f[0]] = func(data, True)
else: else:
m[f[0]] = func(data) k[f[0]] = f[3](data)
m[f[0]] = f[3](data)
flowstr = flowstr[strspn(flowstr, ", ") :] flowstr = flowstr[strspn(flowstr, ", ") :]
if len(flowstr) == 0: if len(flowstr) == 0:
...@@ -689,10 +707,14 @@ class ovskey(nla): ...@@ -689,10 +707,14 @@ class ovskey(nla):
int, int,
convert_ipv4, convert_ipv4,
), ),
("proto", "proto", "%d", lambda x: int(x) if x is not None else 0), ("proto", "proto", "%d", lambda x: int(x) if x else 0,
("tos", "tos", "%d", lambda x: int(x) if x is not None else 0), convert_int(8)),
("ttl", "ttl", "%d", lambda x: int(x) if x is not None else 0), ("tos", "tos", "%d", lambda x: int(x) if x else 0,
("frag", "frag", "%d", lambda x: int(x) if x is not None else 0), convert_int(8)),
("ttl", "ttl", "%d", lambda x: int(x) if x else 0,
convert_int(8)),
("frag", "frag", "%d", lambda x: int(x) if x else 0,
convert_int(8)),
) )
def __init__( def __init__(
...@@ -828,8 +850,8 @@ class ovskey(nla): ...@@ -828,8 +850,8 @@ class ovskey(nla):
) )
fields_map = ( fields_map = (
("type", "type", "%d", int), ("type", "type", "%d", lambda x: int(x) if x else 0),
("code", "code", "%d", int), ("code", "code", "%d", lambda x: int(x) if x else 0),
) )
def __init__( def __init__(
...@@ -894,7 +916,7 @@ class ovskey(nla): ...@@ -894,7 +916,7 @@ class ovskey(nla):
int, int,
convert_ipv4, convert_ipv4,
), ),
("op", "op", "%d", lambda x: int(x) if x is not None else 0), ("op", "op", "%d", lambda x: int(x) if x else 0),
( (
"sha", "sha",
"sha", "sha",
...@@ -1098,6 +1120,16 @@ class ovskey(nla): ...@@ -1098,6 +1120,16 @@ class ovskey(nla):
"tcp", "tcp",
ovskey.ovs_key_tcp, ovskey.ovs_key_tcp,
), ),
(
"OVS_KEY_ATTR_UDP",
"udp",
ovskey.ovs_key_udp,
),
(
"OVS_KEY_ATTR_ICMP",
"icmp",
ovskey.ovs_key_icmp,
),
( (
"OVS_KEY_ATTR_TCP_FLAGS", "OVS_KEY_ATTR_TCP_FLAGS",
"tcp_flags", "tcp_flags",
......
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