nfp: flower: ensure ip protocol is specified for L4 matches
authorJohn Hurley <john.hurley@netronome.com>
Wed, 10 Jul 2019 18:30:30 +0000 (19:30 +0100)
committerDavid S. Miller <davem@davemloft.net>
Fri, 12 Jul 2019 22:31:55 +0000 (15:31 -0700)
Flower rules on the NFP firmware are able to match on an IP protocol
field. When parsing rules in the driver, unknown IP protocols are only
rejected when further matches are to be carried out on layer 4 fields, as
the firmware will not be able to extract such fields from packets.

L4 protocol dissectors such as FLOW_DISSECTOR_KEY_PORTS are only parsed if
an IP protocol is specified. This leaves a loophole whereby a rule that
attempts to match on transport layer information such as port numbers but
does not explicitly give an IP protocol type can be incorrectly offloaded
(in this case with wildcard port numbers matches).

Fix this by rejecting the offload of flows that attempt to match on L4
information, not only when matching on an unknown IP protocol type, but
also when the protocol is wildcarded.

Fixes: 2a04784594f6 ("nfp: flower: check L4 matches on unknown IP protocols")
Signed-off-by: John Hurley <john.hurley@netronome.com>
Reviewed-by: Jakub Kicinski <jakub.kicinski@netronome.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/netronome/nfp/flower/offload.c

index 885f96887150052908511877456bbc5766a869f6..faa8ba012a374f9c623c292b8470e46967f7df2f 100644 (file)
@@ -386,18 +386,15 @@ nfp_flower_calculate_key_layers(struct nfp_app *app,
                        key_layer |= NFP_FLOWER_LAYER_TP;
                        key_size += sizeof(struct nfp_flower_tp_ports);
                        break;
-               default:
-                       /* Other ip proto - we need check the masks for the
-                        * remainder of the key to ensure we can offload.
-                        */
-                       if (nfp_flower_check_higher_than_l3(flow)) {
-                               NL_SET_ERR_MSG_MOD(extack, "unsupported offload: unknown IP protocol with L4 matches not supported");
-                               return -EOPNOTSUPP;
-                       }
-                       break;
                }
        }
 
+       if (!(key_layer & NFP_FLOWER_LAYER_TP) &&
+           nfp_flower_check_higher_than_l3(flow)) {
+               NL_SET_ERR_MSG_MOD(extack, "unsupported offload: cannot match on L4 information without specified IP protocol type");
+               return -EOPNOTSUPP;
+       }
+
        if (flow_rule_match_key(rule, FLOW_DISSECTOR_KEY_TCP)) {
                struct flow_match_tcp tcp;
                u32 tcp_flags;