12874c5fdSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later 21da177e4SLinus Torvalds /* 31da177e4SLinus Torvalds * IPv6 input 41da177e4SLinus Torvalds * Linux INET6 implementation 51da177e4SLinus Torvalds * 61da177e4SLinus Torvalds * Authors: 71da177e4SLinus Torvalds * Pedro Roque <roque@di.fc.ul.pt> 81da177e4SLinus Torvalds * Ian P. Morris <I.P.Morris@soton.ac.uk> 91da177e4SLinus Torvalds * 101da177e4SLinus Torvalds * Based in linux/net/ipv4/ip_input.c 111da177e4SLinus Torvalds */ 121da177e4SLinus Torvalds /* Changes 131da177e4SLinus Torvalds * 141da177e4SLinus Torvalds * Mitsuru KANDA @USAGI and 151da177e4SLinus Torvalds * YOSHIFUJI Hideaki @USAGI: Remove ipv6_parse_exthdrs(). 161da177e4SLinus Torvalds */ 171da177e4SLinus Torvalds 181da177e4SLinus Torvalds #include <linux/errno.h> 191da177e4SLinus Torvalds #include <linux/types.h> 201da177e4SLinus Torvalds #include <linux/socket.h> 211da177e4SLinus Torvalds #include <linux/sockios.h> 221da177e4SLinus Torvalds #include <linux/net.h> 231da177e4SLinus Torvalds #include <linux/netdevice.h> 241da177e4SLinus Torvalds #include <linux/in6.h> 251da177e4SLinus Torvalds #include <linux/icmpv6.h> 267bc570c8SYOSHIFUJI Hideaki #include <linux/mroute6.h> 275a0e3ad6STejun Heo #include <linux/slab.h> 280e219ae4SPaolo Abeni #include <linux/indirect_call_wrapper.h> 291da177e4SLinus Torvalds 301da177e4SLinus Torvalds #include <linux/netfilter.h> 311da177e4SLinus Torvalds #include <linux/netfilter_ipv6.h> 321da177e4SLinus Torvalds 331da177e4SLinus Torvalds #include <net/sock.h> 341da177e4SLinus Torvalds #include <net/snmp.h> 35f9a4719cSLeon Romanovsky #include <net/udp.h> 361da177e4SLinus Torvalds 371da177e4SLinus Torvalds #include <net/ipv6.h> 381da177e4SLinus Torvalds #include <net/protocol.h> 391da177e4SLinus Torvalds #include <net/transp_v6.h> 401da177e4SLinus Torvalds #include <net/rawv6.h> 411da177e4SLinus Torvalds #include <net/ndisc.h> 421da177e4SLinus Torvalds #include <net/ip6_route.h> 431da177e4SLinus Torvalds #include <net/addrconf.h> 441da177e4SLinus Torvalds #include <net/xfrm.h> 451f07d03eSEric Dumazet #include <net/inet_ecn.h> 4648fb6b55SWei-Chun Chao #include <net/dst_metadata.h> 471da177e4SLinus Torvalds 4897ff7ffbSPaolo Abeni INDIRECT_CALLABLE_DECLARE(void tcp_v6_early_demux(struct sk_buff *)); 49d8269e2cSEdward Cree static void ip6_rcv_finish_core(struct net *net, struct sock *sk, 50d8269e2cSEdward Cree struct sk_buff *skb) 511da177e4SLinus Torvalds { 52dddb64bcSsubashab@codeaurora.org void (*edemux)(struct sk_buff *skb); 53dddb64bcSsubashab@codeaurora.org 54e21145a9SNikolay Borisov if (net->ipv4.sysctl_ip_early_demux && !skb_dst(skb) && skb->sk == NULL) { 55c7109986SEric Dumazet const struct inet6_protocol *ipprot; 56c7109986SEric Dumazet 57c7109986SEric Dumazet ipprot = rcu_dereference(inet6_protos[ipv6_hdr(skb)->nexthdr]); 58dddb64bcSsubashab@codeaurora.org if (ipprot && (edemux = READ_ONCE(ipprot->early_demux))) 5997ff7ffbSPaolo Abeni INDIRECT_CALL_2(edemux, tcp_v6_early_demux, 6097ff7ffbSPaolo Abeni udp_v6_early_demux, skb); 61c7109986SEric Dumazet } 6248fb6b55SWei-Chun Chao if (!skb_valid_dst(skb)) 631da177e4SLinus Torvalds ip6_route_input(skb); 64d8269e2cSEdward Cree } 65d8269e2cSEdward Cree 66d8269e2cSEdward Cree int ip6_rcv_finish(struct net *net, struct sock *sk, struct sk_buff *skb) 67d8269e2cSEdward Cree { 68d8269e2cSEdward Cree /* if ingress device is enslaved to an L3 master device pass the 69d8269e2cSEdward Cree * skb to its handler for processing 70d8269e2cSEdward Cree */ 71d8269e2cSEdward Cree skb = l3mdev_ip6_rcv(skb); 72d8269e2cSEdward Cree if (!skb) 73d8269e2cSEdward Cree return NET_RX_SUCCESS; 74d8269e2cSEdward Cree ip6_rcv_finish_core(net, sk, skb); 751da177e4SLinus Torvalds 761da177e4SLinus Torvalds return dst_input(skb); 771da177e4SLinus Torvalds } 781da177e4SLinus Torvalds 79d8269e2cSEdward Cree static void ip6_sublist_rcv_finish(struct list_head *head) 80d8269e2cSEdward Cree { 81d8269e2cSEdward Cree struct sk_buff *skb, *next; 82d8269e2cSEdward Cree 83c7a42eb4SXin Long list_for_each_entry_safe(skb, next, head, list) { 84c7a42eb4SXin Long skb_list_del_init(skb); 85d8269e2cSEdward Cree dst_input(skb); 86d8269e2cSEdward Cree } 87c7a42eb4SXin Long } 88d8269e2cSEdward Cree 89197dbf24SPaolo Abeni static bool ip6_can_use_hint(const struct sk_buff *skb, 90197dbf24SPaolo Abeni const struct sk_buff *hint) 91197dbf24SPaolo Abeni { 92197dbf24SPaolo Abeni return hint && !skb_dst(skb) && 93197dbf24SPaolo Abeni ipv6_addr_equal(&ipv6_hdr(hint)->daddr, &ipv6_hdr(skb)->daddr); 94197dbf24SPaolo Abeni } 95197dbf24SPaolo Abeni 96197dbf24SPaolo Abeni static struct sk_buff *ip6_extract_route_hint(const struct net *net, 97197dbf24SPaolo Abeni struct sk_buff *skb) 98197dbf24SPaolo Abeni { 99197dbf24SPaolo Abeni if (fib6_routes_require_src(net) || fib6_has_custom_rules(net)) 100197dbf24SPaolo Abeni return NULL; 101197dbf24SPaolo Abeni 102197dbf24SPaolo Abeni return skb; 103197dbf24SPaolo Abeni } 104197dbf24SPaolo Abeni 105d8269e2cSEdward Cree static void ip6_list_rcv_finish(struct net *net, struct sock *sk, 106d8269e2cSEdward Cree struct list_head *head) 107d8269e2cSEdward Cree { 108197dbf24SPaolo Abeni struct sk_buff *skb, *next, *hint = NULL; 109d8269e2cSEdward Cree struct dst_entry *curr_dst = NULL; 110d8269e2cSEdward Cree struct list_head sublist; 111d8269e2cSEdward Cree 112d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 113d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 114d8269e2cSEdward Cree struct dst_entry *dst; 115d8269e2cSEdward Cree 11622f6bbb7SEdward Cree skb_list_del_init(skb); 117d8269e2cSEdward Cree /* if ingress device is enslaved to an L3 master device pass the 118d8269e2cSEdward Cree * skb to its handler for processing 119d8269e2cSEdward Cree */ 120d8269e2cSEdward Cree skb = l3mdev_ip6_rcv(skb); 121d8269e2cSEdward Cree if (!skb) 122d8269e2cSEdward Cree continue; 123197dbf24SPaolo Abeni 124197dbf24SPaolo Abeni if (ip6_can_use_hint(skb, hint)) 125197dbf24SPaolo Abeni skb_dst_copy(skb, hint); 126197dbf24SPaolo Abeni else 127d8269e2cSEdward Cree ip6_rcv_finish_core(net, sk, skb); 128d8269e2cSEdward Cree dst = skb_dst(skb); 129d8269e2cSEdward Cree if (curr_dst != dst) { 130197dbf24SPaolo Abeni hint = ip6_extract_route_hint(net, skb); 131197dbf24SPaolo Abeni 132d8269e2cSEdward Cree /* dispatch old sublist */ 133d8269e2cSEdward Cree if (!list_empty(&sublist)) 134d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 135d8269e2cSEdward Cree /* start new sublist */ 136d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 137d8269e2cSEdward Cree curr_dst = dst; 138d8269e2cSEdward Cree } 139d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 140d8269e2cSEdward Cree } 141d8269e2cSEdward Cree /* dispatch final sublist */ 142d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 143d8269e2cSEdward Cree } 144d8269e2cSEdward Cree 145d8269e2cSEdward Cree static struct sk_buff *ip6_rcv_core(struct sk_buff *skb, struct net_device *dev, 146d8269e2cSEdward Cree struct net *net) 1471da177e4SLinus Torvalds { 148*4daf841aSMenglong Dong enum skb_drop_reason reason; 149b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 1501da177e4SLinus Torvalds u32 pkt_len; 151a11d206dSYOSHIFUJI Hideaki struct inet6_dev *idev; 1521da177e4SLinus Torvalds 153a11d206dSYOSHIFUJI Hideaki if (skb->pkt_type == PACKET_OTHERHOST) { 154794c24e9SJeffrey Ji dev_core_stats_rx_otherhost_dropped_inc(skb->dev); 155*4daf841aSMenglong Dong kfree_skb_reason(skb, SKB_DROP_REASON_OTHERHOST); 156d8269e2cSEdward Cree return NULL; 157a11d206dSYOSHIFUJI Hideaki } 1581da177e4SLinus Torvalds 159a11d206dSYOSHIFUJI Hideaki rcu_read_lock(); 160a11d206dSYOSHIFUJI Hideaki 161a11d206dSYOSHIFUJI Hideaki idev = __in6_dev_get(skb->dev); 162a11d206dSYOSHIFUJI Hideaki 163c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_IN, skb->len); 1641da177e4SLinus Torvalds 165*4daf841aSMenglong Dong SKB_DR_SET(reason, NOT_SPECIFIED); 166778d80beSYOSHIFUJI Hideaki if ((skb = skb_share_check(skb, GFP_ATOMIC)) == NULL || 167778d80beSYOSHIFUJI Hideaki !idev || unlikely(idev->cnf.disable_ipv6)) { 1681d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 169*4daf841aSMenglong Dong if (unlikely(idev->cnf.disable_ipv6)) 170*4daf841aSMenglong Dong SKB_DR_SET(reason, IPV6DISABLED); 17171f6f6dfSJesper Nilsson goto drop; 1721da177e4SLinus Torvalds } 1731da177e4SLinus Torvalds 1746b7fdc3aSGuillaume Chazarain memset(IP6CB(skb), 0, sizeof(struct inet6_skb_parm)); 1756b7fdc3aSGuillaume Chazarain 1761da177e4SLinus Torvalds /* 1771da177e4SLinus Torvalds * Store incoming device index. When the packet will 1781da177e4SLinus Torvalds * be queued, we cannot refer to skb->dev anymore. 1791da177e4SLinus Torvalds * 1801da177e4SLinus Torvalds * BTW, when we send a packet for our own local address on a 1811da177e4SLinus Torvalds * non-loopback interface (e.g. ethX), it is being delivered 182de3cb747SDaniel Lezcano * via the loopback interface (lo) here; skb->dev = loopback_dev. 1831da177e4SLinus Torvalds * It, however, should be considered as if it is being 1841da177e4SLinus Torvalds * arrived via the sending interface (ethX), because of the 1851da177e4SLinus Torvalds * nature of scoping architecture. --yoshfuji 1861da177e4SLinus Torvalds */ 18748fb6b55SWei-Chun Chao IP6CB(skb)->iif = skb_valid_dst(skb) ? ip6_dst_idev(skb_dst(skb))->dev->ifindex : dev->ifindex; 1881da177e4SLinus Torvalds 1892889139aSHerbert Xu if (unlikely(!pskb_may_pull(skb, sizeof(*hdr)))) 1901da177e4SLinus Torvalds goto err; 1911da177e4SLinus Torvalds 1920660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 1931da177e4SLinus Torvalds 194*4daf841aSMenglong Dong if (hdr->version != 6) { 195*4daf841aSMenglong Dong SKB_DR_SET(reason, UNHANDLED_PROTO); 1961da177e4SLinus Torvalds goto err; 197*4daf841aSMenglong Dong } 1981da177e4SLinus Torvalds 1991d015503SEric Dumazet __IP6_ADD_STATS(net, idev, 2001f07d03eSEric Dumazet IPSTATS_MIB_NOECTPKTS + 2011f07d03eSEric Dumazet (ipv6_get_dsfield(hdr) & INET_ECN_MASK), 2021f07d03eSEric Dumazet max_t(unsigned short, 1, skb_shinfo(skb)->gso_segs)); 203f630e43aSYOSHIFUJI Hideaki /* 204f630e43aSYOSHIFUJI Hideaki * RFC4291 2.5.3 2050aa8c13eSFlorian Westphal * The loopback address must not be used as the source address in IPv6 2060aa8c13eSFlorian Westphal * packets that are sent outside of a single node. [..] 207f630e43aSYOSHIFUJI Hideaki * A packet received on an interface with a destination address 208f630e43aSYOSHIFUJI Hideaki * of loopback must be dropped. 209f630e43aSYOSHIFUJI Hideaki */ 2100aa8c13eSFlorian Westphal if ((ipv6_addr_loopback(&hdr->saddr) || 2110aa8c13eSFlorian Westphal ipv6_addr_loopback(&hdr->daddr)) && 2123ede0bbcSRobert Shearman !(dev->flags & IFF_LOOPBACK) && 2133ede0bbcSRobert Shearman !netif_is_l3_master(dev)) 214f630e43aSYOSHIFUJI Hideaki goto err; 215f630e43aSYOSHIFUJI Hideaki 2161c4a154eSHannes Frederic Sowa /* RFC4291 Errata ID: 3480 2171c4a154eSHannes Frederic Sowa * Interface-Local scope spans only a single interface on a 2181c4a154eSHannes Frederic Sowa * node and is useful only for loopback transmission of 2191c4a154eSHannes Frederic Sowa * multicast. Packets with interface-local scope received 2201c4a154eSHannes Frederic Sowa * from another node must be discarded. 2211c4a154eSHannes Frederic Sowa */ 2221c4a154eSHannes Frederic Sowa if (!(skb->pkt_type == PACKET_LOOPBACK || 2231c4a154eSHannes Frederic Sowa dev->flags & IFF_LOOPBACK) && 2241c4a154eSHannes Frederic Sowa ipv6_addr_is_multicast(&hdr->daddr) && 2251c4a154eSHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 1) 2261c4a154eSHannes Frederic Sowa goto err; 2271c4a154eSHannes Frederic Sowa 228abbc3043SJohannes Berg /* If enabled, drop unicast packets that were encapsulated in link-layer 229abbc3043SJohannes Berg * multicast or broadcast to protected against the so-called "hole-196" 230abbc3043SJohannes Berg * attack in 802.11 wireless. 231abbc3043SJohannes Berg */ 232abbc3043SJohannes Berg if (!ipv6_addr_is_multicast(&hdr->daddr) && 233abbc3043SJohannes Berg (skb->pkt_type == PACKET_BROADCAST || 234abbc3043SJohannes Berg skb->pkt_type == PACKET_MULTICAST) && 235*4daf841aSMenglong Dong idev->cnf.drop_unicast_in_l2_multicast) { 236*4daf841aSMenglong Dong SKB_DR_SET(reason, UNICAST_IN_L2_MULTICAST); 237abbc3043SJohannes Berg goto err; 238*4daf841aSMenglong Dong } 239abbc3043SJohannes Berg 24020314092SHannes Frederic Sowa /* RFC4291 2.7 24120314092SHannes Frederic Sowa * Nodes must not originate a packet to a multicast address whose scope 24220314092SHannes Frederic Sowa * field contains the reserved value 0; if such a packet is received, it 24320314092SHannes Frederic Sowa * must be silently dropped. 24420314092SHannes Frederic Sowa */ 24520314092SHannes Frederic Sowa if (ipv6_addr_is_multicast(&hdr->daddr) && 24620314092SHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 0) 24720314092SHannes Frederic Sowa goto err; 24820314092SHannes Frederic Sowa 249c457338dSBrian Haley /* 250c457338dSBrian Haley * RFC4291 2.7 251c457338dSBrian Haley * Multicast addresses must not be used as source addresses in IPv6 252c457338dSBrian Haley * packets or appear in any Routing header. 253c457338dSBrian Haley */ 254c457338dSBrian Haley if (ipv6_addr_is_multicast(&hdr->saddr)) 255c457338dSBrian Haley goto err; 256c457338dSBrian Haley 257b0e380b1SArnaldo Carvalho de Melo skb->transport_header = skb->network_header + sizeof(*hdr); 258951dbc8aSPatrick McHardy IP6CB(skb)->nhoff = offsetof(struct ipv6hdr, nexthdr); 259951dbc8aSPatrick McHardy 2601da177e4SLinus Torvalds pkt_len = ntohs(hdr->payload_len); 2611da177e4SLinus Torvalds 2621da177e4SLinus Torvalds /* pkt_len may be zero if Jumbo payload option is present */ 2631da177e4SLinus Torvalds if (pkt_len || hdr->nexthdr != NEXTHDR_HOP) { 26460e5c166SMitsuru Chinen if (pkt_len + sizeof(struct ipv6hdr) > skb->len) { 2651d015503SEric Dumazet __IP6_INC_STATS(net, 266483a47d2SDenis V. Lunev idev, IPSTATS_MIB_INTRUNCATEDPKTS); 267*4daf841aSMenglong Dong SKB_DR_SET(reason, PKT_TOO_SMALL); 26860e5c166SMitsuru Chinen goto drop; 26960e5c166SMitsuru Chinen } 270*4daf841aSMenglong Dong if (pskb_trim_rcsum(skb, pkt_len + sizeof(struct ipv6hdr))) 271*4daf841aSMenglong Dong goto err; 2720660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 2731da177e4SLinus Torvalds } 2741da177e4SLinus Torvalds 2751da177e4SLinus Torvalds if (hdr->nexthdr == NEXTHDR_HOP) { 276e5bbef20SHerbert Xu if (ipv6_parse_hopopts(skb) < 0) { 2771d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 278a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 279d8269e2cSEdward Cree return NULL; 2801da177e4SLinus Torvalds } 2811da177e4SLinus Torvalds } 2821da177e4SLinus Torvalds 283a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 284a11d206dSYOSHIFUJI Hideaki 28571f9dacdSHerbert Xu /* Must drop socket now because of tproxy. */ 286cf7fbe66SJoe Stringer if (!skb_sk_is_prefetched(skb)) 28771f9dacdSHerbert Xu skb_orphan(skb); 28871f9dacdSHerbert Xu 289d8269e2cSEdward Cree return skb; 2901da177e4SLinus Torvalds err: 2911d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 292*4daf841aSMenglong Dong SKB_DR_OR(reason, IP_INHDR); 2931da177e4SLinus Torvalds drop: 294a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 295*4daf841aSMenglong Dong kfree_skb_reason(skb, reason); 296d8269e2cSEdward Cree return NULL; 297d8269e2cSEdward Cree } 298d8269e2cSEdward Cree 299d8269e2cSEdward Cree int ipv6_rcv(struct sk_buff *skb, struct net_device *dev, struct packet_type *pt, struct net_device *orig_dev) 300d8269e2cSEdward Cree { 301d8269e2cSEdward Cree struct net *net = dev_net(skb->dev); 302d8269e2cSEdward Cree 303d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 304d8269e2cSEdward Cree if (skb == NULL) 3055c91faceSMark Smith return NET_RX_DROP; 306d8269e2cSEdward Cree return NF_HOOK(NFPROTO_IPV6, NF_INET_PRE_ROUTING, 307d8269e2cSEdward Cree net, NULL, skb, dev, NULL, 308d8269e2cSEdward Cree ip6_rcv_finish); 309d8269e2cSEdward Cree } 310d8269e2cSEdward Cree 311d8269e2cSEdward Cree static void ip6_sublist_rcv(struct list_head *head, struct net_device *dev, 312d8269e2cSEdward Cree struct net *net) 313d8269e2cSEdward Cree { 314d8269e2cSEdward Cree NF_HOOK_LIST(NFPROTO_IPV6, NF_INET_PRE_ROUTING, net, NULL, 315d8269e2cSEdward Cree head, dev, NULL, ip6_rcv_finish); 316d8269e2cSEdward Cree ip6_list_rcv_finish(net, NULL, head); 317d8269e2cSEdward Cree } 318d8269e2cSEdward Cree 319d8269e2cSEdward Cree /* Receive a list of IPv6 packets */ 320d8269e2cSEdward Cree void ipv6_list_rcv(struct list_head *head, struct packet_type *pt, 321d8269e2cSEdward Cree struct net_device *orig_dev) 322d8269e2cSEdward Cree { 323d8269e2cSEdward Cree struct net_device *curr_dev = NULL; 324d8269e2cSEdward Cree struct net *curr_net = NULL; 325d8269e2cSEdward Cree struct sk_buff *skb, *next; 326d8269e2cSEdward Cree struct list_head sublist; 327d8269e2cSEdward Cree 328d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 329d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 330d8269e2cSEdward Cree struct net_device *dev = skb->dev; 331d8269e2cSEdward Cree struct net *net = dev_net(dev); 332d8269e2cSEdward Cree 33322f6bbb7SEdward Cree skb_list_del_init(skb); 334d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 335d8269e2cSEdward Cree if (skb == NULL) 336d8269e2cSEdward Cree continue; 337d8269e2cSEdward Cree 338d8269e2cSEdward Cree if (curr_dev != dev || curr_net != net) { 339d8269e2cSEdward Cree /* dispatch old sublist */ 340d8269e2cSEdward Cree if (!list_empty(&sublist)) 341d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 342d8269e2cSEdward Cree /* start new sublist */ 343d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 344d8269e2cSEdward Cree curr_dev = dev; 345d8269e2cSEdward Cree curr_net = net; 346d8269e2cSEdward Cree } 347d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 348d8269e2cSEdward Cree } 349d8269e2cSEdward Cree /* dispatch final sublist */ 35051210ad5SFlorian Westphal if (!list_empty(&sublist)) 351d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 3521da177e4SLinus Torvalds } 3531da177e4SLinus Torvalds 3540e219ae4SPaolo Abeni INDIRECT_CALLABLE_DECLARE(int tcp_v6_rcv(struct sk_buff *)); 3550e219ae4SPaolo Abeni 3561da177e4SLinus Torvalds /* 3571da177e4SLinus Torvalds * Deliver the packet to the host 3581da177e4SLinus Torvalds */ 35980bde363SPaolo Abeni void ip6_protocol_deliver_rcu(struct net *net, struct sk_buff *skb, int nexthdr, 36080bde363SPaolo Abeni bool have_final) 3611da177e4SLinus Torvalds { 36241135cc8SAlexey Dobriyan const struct inet6_protocol *ipprot; 363f9242b6bSDavid S. Miller struct inet6_dev *idev; 3641da177e4SLinus Torvalds unsigned int nhoff; 365a50feda5SEric Dumazet bool raw; 3661da177e4SLinus Torvalds 3671da177e4SLinus Torvalds /* 3681da177e4SLinus Torvalds * Parse extension headers 3691da177e4SLinus Torvalds */ 3701da177e4SLinus Torvalds 3711b0ccfe5SDavid S. Miller resubmit: 372adf30907SEric Dumazet idev = ip6_dst_idev(skb_dst(skb)); 37380bde363SPaolo Abeni nhoff = IP6CB(skb)->nhoff; 37480bde363SPaolo Abeni if (!have_final) { 375ea2ae17dSArnaldo Carvalho de Melo if (!pskb_pull(skb, skb_transport_offset(skb))) 3761da177e4SLinus Torvalds goto discard; 377d56f90a7SArnaldo Carvalho de Melo nexthdr = skb_network_header(skb)[nhoff]; 37880bde363SPaolo Abeni } 3791da177e4SLinus Torvalds 3804c64242aSTom Herbert resubmit_final: 38169d6da0bSPavel Emelyanov raw = raw6_local_deliver(skb, nexthdr); 382e5d08d71SIan Morris ipprot = rcu_dereference(inet6_protos[nexthdr]); 38353b24b8fSIan Morris if (ipprot) { 3841da177e4SLinus Torvalds int ret; 3851da177e4SLinus Torvalds 3861da44f9cSTom Herbert if (have_final) { 3871da44f9cSTom Herbert if (!(ipprot->flags & INET6_PROTO_FINAL)) { 3881da44f9cSTom Herbert /* Once we've seen a final protocol don't 3891da44f9cSTom Herbert * allow encapsulation on any non-final 3901da44f9cSTom Herbert * ones. This allows foo in UDP encapsulation 3911da44f9cSTom Herbert * to work. 3921da44f9cSTom Herbert */ 3931da44f9cSTom Herbert goto discard; 3941da44f9cSTom Herbert } 3951da44f9cSTom Herbert } else if (ipprot->flags & INET6_PROTO_FINAL) { 396b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 3975226b6a9SMike Manning int sdif = inet6_sdif(skb); 3985226b6a9SMike Manning struct net_device *dev; 3991da177e4SLinus Torvalds 4001da44f9cSTom Herbert /* Only do this once for first final protocol */ 4011da44f9cSTom Herbert have_final = true; 4021da44f9cSTom Herbert 4039fb9cbb1SYasuyuki Kozakai /* Free reference early: we don't need it any more, 4049fb9cbb1SYasuyuki Kozakai and it may hold ip_conntrack module loaded 4059fb9cbb1SYasuyuki Kozakai indefinitely. */ 406895b5c9fSFlorian Westphal nf_reset_ct(skb); 4079fb9cbb1SYasuyuki Kozakai 408d56f90a7SArnaldo Carvalho de Melo skb_postpull_rcsum(skb, skb_network_header(skb), 409cfe1fc77SArnaldo Carvalho de Melo skb_network_header_len(skb)); 4100660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 4115226b6a9SMike Manning 4125226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 4135226b6a9SMike Manning if (sdif) { 4145226b6a9SMike Manning dev = dev_get_by_index_rcu(net, sdif); 4155226b6a9SMike Manning if (!dev) 4165226b6a9SMike Manning goto discard; 4175226b6a9SMike Manning } else { 4185226b6a9SMike Manning dev = skb->dev; 4195226b6a9SMike Manning } 4205226b6a9SMike Manning 4211da177e4SLinus Torvalds if (ipv6_addr_is_multicast(&hdr->daddr) && 4225226b6a9SMike Manning !ipv6_chk_mcast_addr(dev, &hdr->daddr, 4231da177e4SLinus Torvalds &hdr->saddr) && 424daad1512SYOSHIFUJI Hideaki / 吉藤英明 !ipv6_is_mld(skb, nexthdr, skb_network_header_len(skb))) 4251da177e4SLinus Torvalds goto discard; 4261da177e4SLinus Torvalds } 4271da177e4SLinus Torvalds if (!(ipprot->flags & INET6_PROTO_NOPOLICY) && 4281da177e4SLinus Torvalds !xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) 4291da177e4SLinus Torvalds goto discard; 4301da177e4SLinus Torvalds 4310e219ae4SPaolo Abeni ret = INDIRECT_CALL_2(ipprot->handler, tcp_v6_rcv, udpv6_rcv, 4320e219ae4SPaolo Abeni skb); 4334c64242aSTom Herbert if (ret > 0) { 4344c64242aSTom Herbert if (ipprot->flags & INET6_PROTO_FINAL) { 4354c64242aSTom Herbert /* Not an extension header, most likely UDP 4364c64242aSTom Herbert * encapsulation. Use return value as nexthdr 4374c64242aSTom Herbert * protocol not nhoff (which presumably is 4384c64242aSTom Herbert * not set by handler). 4394c64242aSTom Herbert */ 4404c64242aSTom Herbert nexthdr = ret; 4414c64242aSTom Herbert goto resubmit_final; 4424c64242aSTom Herbert } else { 4431da177e4SLinus Torvalds goto resubmit; 4444c64242aSTom Herbert } 4454c64242aSTom Herbert } else if (ret == 0) { 4461d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 4474c64242aSTom Herbert } 4481da177e4SLinus Torvalds } else { 44969d6da0bSPavel Emelyanov if (!raw) { 4501da177e4SLinus Torvalds if (xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) { 4511d015503SEric Dumazet __IP6_INC_STATS(net, idev, 452483a47d2SDenis V. Lunev IPSTATS_MIB_INUNKNOWNPROTOS); 453fad87acaSPatrick McHardy icmpv6_send(skb, ICMPV6_PARAMPROB, 4543ffe533cSAlexey Dobriyan ICMPV6_UNK_NEXTHDR, nhoff); 4551da177e4SLinus Torvalds } 4561da177e4SLinus Torvalds kfree_skb(skb); 457d8c6f4b9SNeil Horman } else { 4581d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 459d8c6f4b9SNeil Horman consume_skb(skb); 460d8c6f4b9SNeil Horman } 4611da177e4SLinus Torvalds } 46280bde363SPaolo Abeni return; 4631da177e4SLinus Torvalds 4641da177e4SLinus Torvalds discard: 4651d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 4661da177e4SLinus Torvalds kfree_skb(skb); 46780bde363SPaolo Abeni } 46880bde363SPaolo Abeni 46980bde363SPaolo Abeni static int ip6_input_finish(struct net *net, struct sock *sk, struct sk_buff *skb) 47080bde363SPaolo Abeni { 471cd14e9b7SMartin KaFai Lau skb_clear_delivery_time(skb); 47280bde363SPaolo Abeni rcu_read_lock(); 47380bde363SPaolo Abeni ip6_protocol_deliver_rcu(net, skb, 0, false); 47480bde363SPaolo Abeni rcu_read_unlock(); 47580bde363SPaolo Abeni 4761da177e4SLinus Torvalds return 0; 4771da177e4SLinus Torvalds } 4781da177e4SLinus Torvalds 4791da177e4SLinus Torvalds 4801da177e4SLinus Torvalds int ip6_input(struct sk_buff *skb) 4811da177e4SLinus Torvalds { 48229a26a56SEric W. Biederman return NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_IN, 48329a26a56SEric W. Biederman dev_net(skb->dev), NULL, skb, skb->dev, NULL, 4846e23ae2aSPatrick McHardy ip6_input_finish); 4851da177e4SLinus Torvalds } 486b4869aa2SDavid Ahern EXPORT_SYMBOL_GPL(ip6_input); 4871da177e4SLinus Torvalds 4881da177e4SLinus Torvalds int ip6_mc_input(struct sk_buff *skb) 4891da177e4SLinus Torvalds { 4905226b6a9SMike Manning int sdif = inet6_sdif(skb); 491b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 4925226b6a9SMike Manning struct net_device *dev; 493a50feda5SEric Dumazet bool deliver; 4941da177e4SLinus Torvalds 495c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(dev_net(skb_dst(skb)->dev), 496bdb7cc64SStephen Suryaputra __in6_dev_get_safely(skb->dev), IPSTATS_MIB_INMCAST, 497edf391ffSNeil Horman skb->len); 4981da177e4SLinus Torvalds 4995226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 5005226b6a9SMike Manning if (sdif) { 5015226b6a9SMike Manning rcu_read_lock(); 5025226b6a9SMike Manning dev = dev_get_by_index_rcu(dev_net(skb->dev), sdif); 5035226b6a9SMike Manning if (!dev) { 5045226b6a9SMike Manning rcu_read_unlock(); 5055226b6a9SMike Manning kfree_skb(skb); 5065226b6a9SMike Manning return -ENODEV; 5075226b6a9SMike Manning } 5085226b6a9SMike Manning } else { 5095226b6a9SMike Manning dev = skb->dev; 5105226b6a9SMike Manning } 5115226b6a9SMike Manning 5120660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 5135226b6a9SMike Manning deliver = ipv6_chk_mcast_addr(dev, &hdr->daddr, NULL); 5145226b6a9SMike Manning if (sdif) 5155226b6a9SMike Manning rcu_read_unlock(); 5161da177e4SLinus Torvalds 5177bc570c8SYOSHIFUJI Hideaki #ifdef CONFIG_IPV6_MROUTE 5181da177e4SLinus Torvalds /* 5197bc570c8SYOSHIFUJI Hideaki * IPv6 multicast router mode is now supported ;) 5201da177e4SLinus Torvalds */ 521145c7a79SEric Dumazet if (atomic_read(&dev_net(skb->dev)->ipv6.devconf_all->mc_forwarding) && 522ddf64354SHannes Frederic Sowa !(ipv6_addr_type(&hdr->daddr) & 523ddf64354SHannes Frederic Sowa (IPV6_ADDR_LOOPBACK|IPV6_ADDR_LINKLOCAL)) && 5247bc570c8SYOSHIFUJI Hideaki likely(!(IP6CB(skb)->flags & IP6SKB_FORWARDED))) { 5257bc570c8SYOSHIFUJI Hideaki /* 5267bc570c8SYOSHIFUJI Hideaki * Okay, we try to forward - split and duplicate 5277bc570c8SYOSHIFUJI Hideaki * packets. 5287bc570c8SYOSHIFUJI Hideaki */ 5291da177e4SLinus Torvalds struct sk_buff *skb2; 5307bc570c8SYOSHIFUJI Hideaki struct inet6_skb_parm *opt = IP6CB(skb); 5311da177e4SLinus Torvalds 5327bc570c8SYOSHIFUJI Hideaki /* Check for MLD */ 533dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (unlikely(opt->flags & IP6SKB_ROUTERALERT)) { 5347bc570c8SYOSHIFUJI Hideaki /* Check if this is a mld message */ 5357bc570c8SYOSHIFUJI Hideaki u8 nexthdr = hdr->nexthdr; 53675f2811cSJesse Gross __be16 frag_off; 5377bc570c8SYOSHIFUJI Hideaki int offset; 5381da177e4SLinus Torvalds 5397bc570c8SYOSHIFUJI Hideaki /* Check if the value of Router Alert 5407bc570c8SYOSHIFUJI Hideaki * is for MLD (0x0000). 5417bc570c8SYOSHIFUJI Hideaki */ 542dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (opt->ra == htons(IPV6_OPT_ROUTERALERT_MLD)) { 543a50feda5SEric Dumazet deliver = false; 544aba6096bSYOSHIFUJI Hideaki 5457bc570c8SYOSHIFUJI Hideaki if (!ipv6_ext_hdr(nexthdr)) { 5467bc570c8SYOSHIFUJI Hideaki /* BUG */ 547aba6096bSYOSHIFUJI Hideaki goto out; 5481da177e4SLinus Torvalds } 5497bc570c8SYOSHIFUJI Hideaki offset = ipv6_skip_exthdr(skb, sizeof(*hdr), 55075f2811cSJesse Gross &nexthdr, &frag_off); 5517bc570c8SYOSHIFUJI Hideaki if (offset < 0) 552aba6096bSYOSHIFUJI Hideaki goto out; 5537bc570c8SYOSHIFUJI Hideaki 5544c938d22SAngga if (ipv6_is_mld(skb, nexthdr, offset)) 555a50feda5SEric Dumazet deliver = true; 5564c938d22SAngga 5574c938d22SAngga goto out; 5587bc570c8SYOSHIFUJI Hideaki } 5597bc570c8SYOSHIFUJI Hideaki /* unknown RA - process it normally */ 5607bc570c8SYOSHIFUJI Hideaki } 5617bc570c8SYOSHIFUJI Hideaki 5627bc570c8SYOSHIFUJI Hideaki if (deliver) 5637bc570c8SYOSHIFUJI Hideaki skb2 = skb_clone(skb, GFP_ATOMIC); 5647bc570c8SYOSHIFUJI Hideaki else { 5657bc570c8SYOSHIFUJI Hideaki skb2 = skb; 5667bc570c8SYOSHIFUJI Hideaki skb = NULL; 5677bc570c8SYOSHIFUJI Hideaki } 5687bc570c8SYOSHIFUJI Hideaki 5697bc570c8SYOSHIFUJI Hideaki if (skb2) { 5707bc570c8SYOSHIFUJI Hideaki ip6_mr_input(skb2); 5711da177e4SLinus Torvalds } 5721da177e4SLinus Torvalds } 5737bc570c8SYOSHIFUJI Hideaki out: 574aba6096bSYOSHIFUJI Hideaki #endif 575aba6096bSYOSHIFUJI Hideaki if (likely(deliver)) 5761da177e4SLinus Torvalds ip6_input(skb); 577aba6096bSYOSHIFUJI Hideaki else { 5781da177e4SLinus Torvalds /* discard */ 5791da177e4SLinus Torvalds kfree_skb(skb); 580aba6096bSYOSHIFUJI Hideaki } 5811da177e4SLinus Torvalds 5821da177e4SLinus Torvalds return 0; 5831da177e4SLinus Torvalds } 584