aboutsummaryrefslogtreecommitdiffstats
path: root/net/openvswitch
diff options
context:
space:
mode:
authorGreg Rose <gvrose8192@gmail.com>2019-08-27 10:58:09 -0400
committerDavid S. Miller <davem@davemloft.net>2019-08-28 17:53:51 -0400
commitad06a566e118e57b852cab5933dbbbaebb141de3 (patch)
treec51c3fae651607226cc0dd2feb462008846340d7 /net/openvswitch
parenta84d016479896b5526a2cc54784e6ffc41c9d6f6 (diff)
openvswitch: Properly set L4 keys on "later" IP fragments
When IP fragments are reassembled before being sent to conntrack, the key from the last fragment is used. Unless there are reordering issues, the last fragment received will not contain the L4 ports, so the key for the reassembled datagram won't contain them. This patch updates the key once we have a reassembled datagram. The handle_fragments() function works on L3 headers so we pull the L3/L4 flow key update code from key_extract into a new function 'key_extract_l3l4'. Then we add a another new function ovs_flow_key_update_l3l4() and export it so that it is accessible by handle_fragments() for conntrack packet reassembly. Co-authored-by: Justin Pettit <jpettit@ovn.org> Signed-off-by: Greg Rose <gvrose8192@gmail.com> Acked-by: Pravin B Shelar <pshelar@ovn.org> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/openvswitch')
-rw-r--r--net/openvswitch/conntrack.c5
-rw-r--r--net/openvswitch/flow.c155
-rw-r--r--net/openvswitch/flow.h1
3 files changed, 95 insertions, 66 deletions
diff --git a/net/openvswitch/conntrack.c b/net/openvswitch/conntrack.c
index d8da6477d6be..05249eb45082 100644
--- a/net/openvswitch/conntrack.c
+++ b/net/openvswitch/conntrack.c
@@ -525,6 +525,11 @@ static int handle_fragments(struct net *net, struct sw_flow_key *key,
525 return -EPFNOSUPPORT; 525 return -EPFNOSUPPORT;
526 } 526 }
527 527
528 /* The key extracted from the fragment that completed this datagram
529 * likely didn't have an L4 header, so regenerate it.
530 */
531 ovs_flow_key_update_l3l4(skb, key);
532
528 key->ip.frag = OVS_FRAG_TYPE_NONE; 533 key->ip.frag = OVS_FRAG_TYPE_NONE;
529 skb_clear_hash(skb); 534 skb_clear_hash(skb);
530 skb->ignore_df = 1; 535 skb->ignore_df = 1;
diff --git a/net/openvswitch/flow.c b/net/openvswitch/flow.c
index bc89e16e0505..005f7622edac 100644
--- a/net/openvswitch/flow.c
+++ b/net/openvswitch/flow.c
@@ -523,78 +523,15 @@ static int parse_nsh(struct sk_buff *skb, struct sw_flow_key *key)
523} 523}
524 524
525/** 525/**
526 * key_extract - extracts a flow key from an Ethernet frame. 526 * key_extract_l3l4 - extracts L3/L4 header information.
527 * @skb: sk_buff that contains the frame, with skb->data pointing to the 527 * @skb: sk_buff that contains the frame, with skb->data pointing to the
528 * Ethernet header 528 * L3 header
529 * @key: output flow key 529 * @key: output flow key
530 * 530 *
531 * The caller must ensure that skb->len >= ETH_HLEN.
532 *
533 * Returns 0 if successful, otherwise a negative errno value.
534 *
535 * Initializes @skb header fields as follows:
536 *
537 * - skb->mac_header: the L2 header.
538 *
539 * - skb->network_header: just past the L2 header, or just past the
540 * VLAN header, to the first byte of the L2 payload.
541 *
542 * - skb->transport_header: If key->eth.type is ETH_P_IP or ETH_P_IPV6
543 * on output, then just past the IP header, if one is present and
544 * of a correct length, otherwise the same as skb->network_header.
545 * For other key->eth.type values it is left untouched.
546 *
547 * - skb->protocol: the type of the data starting at skb->network_header.
548 * Equals to key->eth.type.
549 */ 531 */
550static int key_extract(struct sk_buff *skb, struct sw_flow_key *key) 532static int key_extract_l3l4(struct sk_buff *skb, struct sw_flow_key *key)
551{ 533{
552 int error; 534 int error;
553 struct ethhdr *eth;
554
555 /* Flags are always used as part of stats */
556 key->tp.flags = 0;
557
558 skb_reset_mac_header(skb);
559
560 /* Link layer. */
561 clear_vlan(key);
562 if (ovs_key_mac_proto(key) == MAC_PROTO_NONE) {
563 if (unlikely(eth_type_vlan(skb->protocol)))
564 return -EINVAL;
565
566 skb_reset_network_header(skb);
567 key->eth.type = skb->protocol;
568 } else {
569 eth = eth_hdr(skb);
570 ether_addr_copy(key->eth.src, eth->h_source);
571 ether_addr_copy(key->eth.dst, eth->h_dest);
572
573 __skb_pull(skb, 2 * ETH_ALEN);
574 /* We are going to push all headers that we pull, so no need to
575 * update skb->csum here.
576 */
577
578 if (unlikely(parse_vlan(skb, key)))
579 return -ENOMEM;
580
581 key->eth.type = parse_ethertype(skb);
582 if (unlikely(key->eth.type == htons(0)))
583 return -ENOMEM;
584
585 /* Multiple tagged packets need to retain TPID to satisfy
586 * skb_vlan_pop(), which will later shift the ethertype into
587 * skb->protocol.
588 */
589 if (key->eth.cvlan.tci & htons(VLAN_CFI_MASK))
590 skb->protocol = key->eth.cvlan.tpid;
591 else
592 skb->protocol = key->eth.type;
593
594 skb_reset_network_header(skb);
595 __skb_push(skb, skb->data - skb_mac_header(skb));
596 }
597 skb_reset_mac_len(skb);
598 535
599 /* Network layer. */ 536 /* Network layer. */
600 if (key->eth.type == htons(ETH_P_IP)) { 537 if (key->eth.type == htons(ETH_P_IP)) {
@@ -788,6 +725,92 @@ static int key_extract(struct sk_buff *skb, struct sw_flow_key *key)
788 return 0; 725 return 0;
789} 726}
790 727
728/**
729 * key_extract - extracts a flow key from an Ethernet frame.
730 * @skb: sk_buff that contains the frame, with skb->data pointing to the
731 * Ethernet header
732 * @key: output flow key
733 *
734 * The caller must ensure that skb->len >= ETH_HLEN.
735 *
736 * Returns 0 if successful, otherwise a negative errno value.
737 *
738 * Initializes @skb header fields as follows:
739 *
740 * - skb->mac_header: the L2 header.
741 *
742 * - skb->network_header: just past the L2 header, or just past the
743 * VLAN header, to the first byte of the L2 payload.
744 *
745 * - skb->transport_header: If key->eth.type is ETH_P_IP or ETH_P_IPV6
746 * on output, then just past the IP header, if one is present and
747 * of a correct length, otherwise the same as skb->network_header.
748 * For other key->eth.type values it is left untouched.
749 *
750 * - skb->protocol: the type of the data starting at skb->network_header.
751 * Equals to key->eth.type.
752 */
753static int key_extract(struct sk_buff *skb, struct sw_flow_key *key)
754{
755 struct ethhdr *eth;
756
757 /* Flags are always used as part of stats */
758 key->tp.flags = 0;
759
760 skb_reset_mac_header(skb);
761
762 /* Link layer. */
763 clear_vlan(key);
764 if (ovs_key_mac_proto(key) == MAC_PROTO_NONE) {
765 if (unlikely(eth_type_vlan(skb->protocol)))
766 return -EINVAL;
767
768 skb_reset_network_header(skb);
769 key->eth.type = skb->protocol;
770 } else {
771 eth = eth_hdr(skb);
772 ether_addr_copy(key->eth.src, eth->h_source);
773 ether_addr_copy(key->eth.dst, eth->h_dest);
774
775 __skb_pull(skb, 2 * ETH_ALEN);
776 /* We are going to push all headers that we pull, so no need to
777 * update skb->csum here.
778 */
779
780 if (unlikely(parse_vlan(skb, key)))
781 return -ENOMEM;
782
783 key->eth.type = parse_ethertype(skb);
784 if (unlikely(key->eth.type == htons(0)))
785 return -ENOMEM;
786
787 /* Multiple tagged packets need to retain TPID to satisfy
788 * skb_vlan_pop(), which will later shift the ethertype into
789 * skb->protocol.
790 */
791 if (key->eth.cvlan.tci & htons(VLAN_CFI_MASK))
792 skb->protocol = key->eth.cvlan.tpid;
793 else
794 skb->protocol = key->eth.type;
795
796 skb_reset_network_header(skb);
797 __skb_push(skb, skb->data - skb_mac_header(skb));
798 }
799
800 skb_reset_mac_len(skb);
801
802 /* Fill out L3/L4 key info, if any */
803 return key_extract_l3l4(skb, key);
804}
805
806/* In the case of conntrack fragment handling it expects L3 headers,
807 * add a helper.
808 */
809int ovs_flow_key_update_l3l4(struct sk_buff *skb, struct sw_flow_key *key)
810{
811 return key_extract_l3l4(skb, key);
812}
813
791int ovs_flow_key_update(struct sk_buff *skb, struct sw_flow_key *key) 814int ovs_flow_key_update(struct sk_buff *skb, struct sw_flow_key *key)
792{ 815{
793 int res; 816 int res;
diff --git a/net/openvswitch/flow.h b/net/openvswitch/flow.h
index a5506e2d4b7a..b830d5ff7af4 100644
--- a/net/openvswitch/flow.h
+++ b/net/openvswitch/flow.h
@@ -270,6 +270,7 @@ void ovs_flow_stats_clear(struct sw_flow *);
270u64 ovs_flow_used_time(unsigned long flow_jiffies); 270u64 ovs_flow_used_time(unsigned long flow_jiffies);
271 271
272int ovs_flow_key_update(struct sk_buff *skb, struct sw_flow_key *key); 272int ovs_flow_key_update(struct sk_buff *skb, struct sw_flow_key *key);
273int ovs_flow_key_update_l3l4(struct sk_buff *skb, struct sw_flow_key *key);
273int ovs_flow_key_extract(const struct ip_tunnel_info *tun_info, 274int ovs_flow_key_extract(const struct ip_tunnel_info *tun_info,
274 struct sk_buff *skb, 275 struct sk_buff *skb,
275 struct sw_flow_key *key); 276 struct sw_flow_key *key);