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> 351da177e4SLinus Torvalds 361da177e4SLinus Torvalds #include <net/ipv6.h> 371da177e4SLinus Torvalds #include <net/protocol.h> 381da177e4SLinus Torvalds #include <net/transp_v6.h> 391da177e4SLinus Torvalds #include <net/rawv6.h> 401da177e4SLinus Torvalds #include <net/ndisc.h> 411da177e4SLinus Torvalds #include <net/ip6_route.h> 421da177e4SLinus Torvalds #include <net/addrconf.h> 431da177e4SLinus Torvalds #include <net/xfrm.h> 441f07d03eSEric Dumazet #include <net/inet_ecn.h> 4548fb6b55SWei-Chun Chao #include <net/dst_metadata.h> 461da177e4SLinus Torvalds 4797ff7ffbSPaolo Abeni INDIRECT_CALLABLE_DECLARE(void udp_v6_early_demux(struct sk_buff *)); 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 89d8269e2cSEdward Cree static void ip6_list_rcv_finish(struct net *net, struct sock *sk, 90d8269e2cSEdward Cree struct list_head *head) 91d8269e2cSEdward Cree { 92d8269e2cSEdward Cree struct dst_entry *curr_dst = NULL; 93d8269e2cSEdward Cree struct sk_buff *skb, *next; 94d8269e2cSEdward Cree struct list_head sublist; 95d8269e2cSEdward Cree 96d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 97d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 98d8269e2cSEdward Cree struct dst_entry *dst; 99d8269e2cSEdward Cree 10022f6bbb7SEdward Cree skb_list_del_init(skb); 101d8269e2cSEdward Cree /* if ingress device is enslaved to an L3 master device pass the 102d8269e2cSEdward Cree * skb to its handler for processing 103d8269e2cSEdward Cree */ 104d8269e2cSEdward Cree skb = l3mdev_ip6_rcv(skb); 105d8269e2cSEdward Cree if (!skb) 106d8269e2cSEdward Cree continue; 107d8269e2cSEdward Cree ip6_rcv_finish_core(net, sk, skb); 108d8269e2cSEdward Cree dst = skb_dst(skb); 109d8269e2cSEdward Cree if (curr_dst != dst) { 110d8269e2cSEdward Cree /* dispatch old sublist */ 111d8269e2cSEdward Cree if (!list_empty(&sublist)) 112d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 113d8269e2cSEdward Cree /* start new sublist */ 114d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 115d8269e2cSEdward Cree curr_dst = dst; 116d8269e2cSEdward Cree } 117d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 118d8269e2cSEdward Cree } 119d8269e2cSEdward Cree /* dispatch final sublist */ 120d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 121d8269e2cSEdward Cree } 122d8269e2cSEdward Cree 123d8269e2cSEdward Cree static struct sk_buff *ip6_rcv_core(struct sk_buff *skb, struct net_device *dev, 124d8269e2cSEdward Cree struct net *net) 1251da177e4SLinus Torvalds { 126b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 1271da177e4SLinus Torvalds u32 pkt_len; 128a11d206dSYOSHIFUJI Hideaki struct inet6_dev *idev; 1291da177e4SLinus Torvalds 130a11d206dSYOSHIFUJI Hideaki if (skb->pkt_type == PACKET_OTHERHOST) { 131a11d206dSYOSHIFUJI Hideaki kfree_skb(skb); 132d8269e2cSEdward Cree return NULL; 133a11d206dSYOSHIFUJI Hideaki } 1341da177e4SLinus Torvalds 135a11d206dSYOSHIFUJI Hideaki rcu_read_lock(); 136a11d206dSYOSHIFUJI Hideaki 137a11d206dSYOSHIFUJI Hideaki idev = __in6_dev_get(skb->dev); 138a11d206dSYOSHIFUJI Hideaki 139c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_IN, skb->len); 1401da177e4SLinus Torvalds 141778d80beSYOSHIFUJI Hideaki if ((skb = skb_share_check(skb, GFP_ATOMIC)) == NULL || 142778d80beSYOSHIFUJI Hideaki !idev || unlikely(idev->cnf.disable_ipv6)) { 1431d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 14471f6f6dfSJesper Nilsson goto drop; 1451da177e4SLinus Torvalds } 1461da177e4SLinus Torvalds 1476b7fdc3aSGuillaume Chazarain memset(IP6CB(skb), 0, sizeof(struct inet6_skb_parm)); 1486b7fdc3aSGuillaume Chazarain 1491da177e4SLinus Torvalds /* 1501da177e4SLinus Torvalds * Store incoming device index. When the packet will 1511da177e4SLinus Torvalds * be queued, we cannot refer to skb->dev anymore. 1521da177e4SLinus Torvalds * 1531da177e4SLinus Torvalds * BTW, when we send a packet for our own local address on a 1541da177e4SLinus Torvalds * non-loopback interface (e.g. ethX), it is being delivered 155de3cb747SDaniel Lezcano * via the loopback interface (lo) here; skb->dev = loopback_dev. 1561da177e4SLinus Torvalds * It, however, should be considered as if it is being 1571da177e4SLinus Torvalds * arrived via the sending interface (ethX), because of the 1581da177e4SLinus Torvalds * nature of scoping architecture. --yoshfuji 1591da177e4SLinus Torvalds */ 16048fb6b55SWei-Chun Chao IP6CB(skb)->iif = skb_valid_dst(skb) ? ip6_dst_idev(skb_dst(skb))->dev->ifindex : dev->ifindex; 1611da177e4SLinus Torvalds 1622889139aSHerbert Xu if (unlikely(!pskb_may_pull(skb, sizeof(*hdr)))) 1631da177e4SLinus Torvalds goto err; 1641da177e4SLinus Torvalds 1650660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 1661da177e4SLinus Torvalds 1671da177e4SLinus Torvalds if (hdr->version != 6) 1681da177e4SLinus Torvalds goto err; 1691da177e4SLinus Torvalds 1701d015503SEric Dumazet __IP6_ADD_STATS(net, idev, 1711f07d03eSEric Dumazet IPSTATS_MIB_NOECTPKTS + 1721f07d03eSEric Dumazet (ipv6_get_dsfield(hdr) & INET_ECN_MASK), 1731f07d03eSEric Dumazet max_t(unsigned short, 1, skb_shinfo(skb)->gso_segs)); 174f630e43aSYOSHIFUJI Hideaki /* 175f630e43aSYOSHIFUJI Hideaki * RFC4291 2.5.3 1760aa8c13eSFlorian Westphal * The loopback address must not be used as the source address in IPv6 1770aa8c13eSFlorian Westphal * packets that are sent outside of a single node. [..] 178f630e43aSYOSHIFUJI Hideaki * A packet received on an interface with a destination address 179f630e43aSYOSHIFUJI Hideaki * of loopback must be dropped. 180f630e43aSYOSHIFUJI Hideaki */ 1810aa8c13eSFlorian Westphal if ((ipv6_addr_loopback(&hdr->saddr) || 1820aa8c13eSFlorian Westphal ipv6_addr_loopback(&hdr->daddr)) && 1833ede0bbcSRobert Shearman !(dev->flags & IFF_LOOPBACK) && 1843ede0bbcSRobert Shearman !netif_is_l3_master(dev)) 185f630e43aSYOSHIFUJI Hideaki goto err; 186f630e43aSYOSHIFUJI Hideaki 1871c4a154eSHannes Frederic Sowa /* RFC4291 Errata ID: 3480 1881c4a154eSHannes Frederic Sowa * Interface-Local scope spans only a single interface on a 1891c4a154eSHannes Frederic Sowa * node and is useful only for loopback transmission of 1901c4a154eSHannes Frederic Sowa * multicast. Packets with interface-local scope received 1911c4a154eSHannes Frederic Sowa * from another node must be discarded. 1921c4a154eSHannes Frederic Sowa */ 1931c4a154eSHannes Frederic Sowa if (!(skb->pkt_type == PACKET_LOOPBACK || 1941c4a154eSHannes Frederic Sowa dev->flags & IFF_LOOPBACK) && 1951c4a154eSHannes Frederic Sowa ipv6_addr_is_multicast(&hdr->daddr) && 1961c4a154eSHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 1) 1971c4a154eSHannes Frederic Sowa goto err; 1981c4a154eSHannes Frederic Sowa 199abbc3043SJohannes Berg /* If enabled, drop unicast packets that were encapsulated in link-layer 200abbc3043SJohannes Berg * multicast or broadcast to protected against the so-called "hole-196" 201abbc3043SJohannes Berg * attack in 802.11 wireless. 202abbc3043SJohannes Berg */ 203abbc3043SJohannes Berg if (!ipv6_addr_is_multicast(&hdr->daddr) && 204abbc3043SJohannes Berg (skb->pkt_type == PACKET_BROADCAST || 205abbc3043SJohannes Berg skb->pkt_type == PACKET_MULTICAST) && 206abbc3043SJohannes Berg idev->cnf.drop_unicast_in_l2_multicast) 207abbc3043SJohannes Berg goto err; 208abbc3043SJohannes Berg 20920314092SHannes Frederic Sowa /* RFC4291 2.7 21020314092SHannes Frederic Sowa * Nodes must not originate a packet to a multicast address whose scope 21120314092SHannes Frederic Sowa * field contains the reserved value 0; if such a packet is received, it 21220314092SHannes Frederic Sowa * must be silently dropped. 21320314092SHannes Frederic Sowa */ 21420314092SHannes Frederic Sowa if (ipv6_addr_is_multicast(&hdr->daddr) && 21520314092SHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 0) 21620314092SHannes Frederic Sowa goto err; 21720314092SHannes Frederic Sowa 218c457338dSBrian Haley /* 219c457338dSBrian Haley * RFC4291 2.7 220c457338dSBrian Haley * Multicast addresses must not be used as source addresses in IPv6 221c457338dSBrian Haley * packets or appear in any Routing header. 222c457338dSBrian Haley */ 223c457338dSBrian Haley if (ipv6_addr_is_multicast(&hdr->saddr)) 224c457338dSBrian Haley goto err; 225c457338dSBrian Haley 226b0e380b1SArnaldo Carvalho de Melo skb->transport_header = skb->network_header + sizeof(*hdr); 227951dbc8aSPatrick McHardy IP6CB(skb)->nhoff = offsetof(struct ipv6hdr, nexthdr); 228951dbc8aSPatrick McHardy 2291da177e4SLinus Torvalds pkt_len = ntohs(hdr->payload_len); 2301da177e4SLinus Torvalds 2311da177e4SLinus Torvalds /* pkt_len may be zero if Jumbo payload option is present */ 2321da177e4SLinus Torvalds if (pkt_len || hdr->nexthdr != NEXTHDR_HOP) { 23360e5c166SMitsuru Chinen if (pkt_len + sizeof(struct ipv6hdr) > skb->len) { 2341d015503SEric Dumazet __IP6_INC_STATS(net, 235483a47d2SDenis V. Lunev idev, IPSTATS_MIB_INTRUNCATEDPKTS); 23660e5c166SMitsuru Chinen goto drop; 23760e5c166SMitsuru Chinen } 2381da177e4SLinus Torvalds if (pskb_trim_rcsum(skb, pkt_len + sizeof(struct ipv6hdr))) { 2391d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 2401da177e4SLinus Torvalds goto drop; 2411da177e4SLinus Torvalds } 2420660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 2431da177e4SLinus Torvalds } 2441da177e4SLinus Torvalds 2451da177e4SLinus Torvalds if (hdr->nexthdr == NEXTHDR_HOP) { 246e5bbef20SHerbert Xu if (ipv6_parse_hopopts(skb) < 0) { 2471d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 248a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 249d8269e2cSEdward Cree return NULL; 2501da177e4SLinus Torvalds } 2511da177e4SLinus Torvalds } 2521da177e4SLinus Torvalds 253a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 254a11d206dSYOSHIFUJI Hideaki 25571f9dacdSHerbert Xu /* Must drop socket now because of tproxy. */ 25671f9dacdSHerbert Xu skb_orphan(skb); 25771f9dacdSHerbert Xu 258d8269e2cSEdward Cree return skb; 2591da177e4SLinus Torvalds err: 2601d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 2611da177e4SLinus Torvalds drop: 262a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 2631da177e4SLinus Torvalds kfree_skb(skb); 264d8269e2cSEdward Cree return NULL; 265d8269e2cSEdward Cree } 266d8269e2cSEdward Cree 267d8269e2cSEdward Cree int ipv6_rcv(struct sk_buff *skb, struct net_device *dev, struct packet_type *pt, struct net_device *orig_dev) 268d8269e2cSEdward Cree { 269d8269e2cSEdward Cree struct net *net = dev_net(skb->dev); 270d8269e2cSEdward Cree 271d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 272d8269e2cSEdward Cree if (skb == NULL) 2735c91faceSMark Smith return NET_RX_DROP; 274d8269e2cSEdward Cree return NF_HOOK(NFPROTO_IPV6, NF_INET_PRE_ROUTING, 275d8269e2cSEdward Cree net, NULL, skb, dev, NULL, 276d8269e2cSEdward Cree ip6_rcv_finish); 277d8269e2cSEdward Cree } 278d8269e2cSEdward Cree 279d8269e2cSEdward Cree static void ip6_sublist_rcv(struct list_head *head, struct net_device *dev, 280d8269e2cSEdward Cree struct net *net) 281d8269e2cSEdward Cree { 282d8269e2cSEdward Cree NF_HOOK_LIST(NFPROTO_IPV6, NF_INET_PRE_ROUTING, net, NULL, 283d8269e2cSEdward Cree head, dev, NULL, ip6_rcv_finish); 284d8269e2cSEdward Cree ip6_list_rcv_finish(net, NULL, head); 285d8269e2cSEdward Cree } 286d8269e2cSEdward Cree 287d8269e2cSEdward Cree /* Receive a list of IPv6 packets */ 288d8269e2cSEdward Cree void ipv6_list_rcv(struct list_head *head, struct packet_type *pt, 289d8269e2cSEdward Cree struct net_device *orig_dev) 290d8269e2cSEdward Cree { 291d8269e2cSEdward Cree struct net_device *curr_dev = NULL; 292d8269e2cSEdward Cree struct net *curr_net = NULL; 293d8269e2cSEdward Cree struct sk_buff *skb, *next; 294d8269e2cSEdward Cree struct list_head sublist; 295d8269e2cSEdward Cree 296d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 297d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 298d8269e2cSEdward Cree struct net_device *dev = skb->dev; 299d8269e2cSEdward Cree struct net *net = dev_net(dev); 300d8269e2cSEdward Cree 30122f6bbb7SEdward Cree skb_list_del_init(skb); 302d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 303d8269e2cSEdward Cree if (skb == NULL) 304d8269e2cSEdward Cree continue; 305d8269e2cSEdward Cree 306d8269e2cSEdward Cree if (curr_dev != dev || curr_net != net) { 307d8269e2cSEdward Cree /* dispatch old sublist */ 308d8269e2cSEdward Cree if (!list_empty(&sublist)) 309d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 310d8269e2cSEdward Cree /* start new sublist */ 311d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 312d8269e2cSEdward Cree curr_dev = dev; 313d8269e2cSEdward Cree curr_net = net; 314d8269e2cSEdward Cree } 315d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 316d8269e2cSEdward Cree } 317d8269e2cSEdward Cree /* dispatch final sublist */ 318d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 3191da177e4SLinus Torvalds } 3201da177e4SLinus Torvalds 3210e219ae4SPaolo Abeni INDIRECT_CALLABLE_DECLARE(int udpv6_rcv(struct sk_buff *)); 3220e219ae4SPaolo Abeni INDIRECT_CALLABLE_DECLARE(int tcp_v6_rcv(struct sk_buff *)); 3230e219ae4SPaolo Abeni 3241da177e4SLinus Torvalds /* 3251da177e4SLinus Torvalds * Deliver the packet to the host 3261da177e4SLinus Torvalds */ 32780bde363SPaolo Abeni void ip6_protocol_deliver_rcu(struct net *net, struct sk_buff *skb, int nexthdr, 32880bde363SPaolo Abeni bool have_final) 3291da177e4SLinus Torvalds { 33041135cc8SAlexey Dobriyan const struct inet6_protocol *ipprot; 331f9242b6bSDavid S. Miller struct inet6_dev *idev; 3321da177e4SLinus Torvalds unsigned int nhoff; 333a50feda5SEric Dumazet bool raw; 3341da177e4SLinus Torvalds 3351da177e4SLinus Torvalds /* 3361da177e4SLinus Torvalds * Parse extension headers 3371da177e4SLinus Torvalds */ 3381da177e4SLinus Torvalds 3391b0ccfe5SDavid S. Miller resubmit: 340adf30907SEric Dumazet idev = ip6_dst_idev(skb_dst(skb)); 34180bde363SPaolo Abeni nhoff = IP6CB(skb)->nhoff; 34280bde363SPaolo Abeni if (!have_final) { 343ea2ae17dSArnaldo Carvalho de Melo if (!pskb_pull(skb, skb_transport_offset(skb))) 3441da177e4SLinus Torvalds goto discard; 345d56f90a7SArnaldo Carvalho de Melo nexthdr = skb_network_header(skb)[nhoff]; 34680bde363SPaolo Abeni } 3471da177e4SLinus Torvalds 3484c64242aSTom Herbert resubmit_final: 34969d6da0bSPavel Emelyanov raw = raw6_local_deliver(skb, nexthdr); 350e5d08d71SIan Morris ipprot = rcu_dereference(inet6_protos[nexthdr]); 35153b24b8fSIan Morris if (ipprot) { 3521da177e4SLinus Torvalds int ret; 3531da177e4SLinus Torvalds 3541da44f9cSTom Herbert if (have_final) { 3551da44f9cSTom Herbert if (!(ipprot->flags & INET6_PROTO_FINAL)) { 3561da44f9cSTom Herbert /* Once we've seen a final protocol don't 3571da44f9cSTom Herbert * allow encapsulation on any non-final 3581da44f9cSTom Herbert * ones. This allows foo in UDP encapsulation 3591da44f9cSTom Herbert * to work. 3601da44f9cSTom Herbert */ 3611da44f9cSTom Herbert goto discard; 3621da44f9cSTom Herbert } 3631da44f9cSTom Herbert } else if (ipprot->flags & INET6_PROTO_FINAL) { 364b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 3655226b6a9SMike Manning int sdif = inet6_sdif(skb); 3665226b6a9SMike Manning struct net_device *dev; 3671da177e4SLinus Torvalds 3681da44f9cSTom Herbert /* Only do this once for first final protocol */ 3691da44f9cSTom Herbert have_final = true; 3701da44f9cSTom Herbert 3719fb9cbb1SYasuyuki Kozakai /* Free reference early: we don't need it any more, 3729fb9cbb1SYasuyuki Kozakai and it may hold ip_conntrack module loaded 3739fb9cbb1SYasuyuki Kozakai indefinitely. */ 374895b5c9fSFlorian Westphal nf_reset_ct(skb); 3759fb9cbb1SYasuyuki Kozakai 376d56f90a7SArnaldo Carvalho de Melo skb_postpull_rcsum(skb, skb_network_header(skb), 377cfe1fc77SArnaldo Carvalho de Melo skb_network_header_len(skb)); 3780660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 3795226b6a9SMike Manning 3805226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 3815226b6a9SMike Manning if (sdif) { 3825226b6a9SMike Manning dev = dev_get_by_index_rcu(net, sdif); 3835226b6a9SMike Manning if (!dev) 3845226b6a9SMike Manning goto discard; 3855226b6a9SMike Manning } else { 3865226b6a9SMike Manning dev = skb->dev; 3875226b6a9SMike Manning } 3885226b6a9SMike Manning 3891da177e4SLinus Torvalds if (ipv6_addr_is_multicast(&hdr->daddr) && 3905226b6a9SMike Manning !ipv6_chk_mcast_addr(dev, &hdr->daddr, 3911da177e4SLinus Torvalds &hdr->saddr) && 392daad1512SYOSHIFUJI Hideaki / 吉藤英明 !ipv6_is_mld(skb, nexthdr, skb_network_header_len(skb))) 3931da177e4SLinus Torvalds goto discard; 3941da177e4SLinus Torvalds } 3951da177e4SLinus Torvalds if (!(ipprot->flags & INET6_PROTO_NOPOLICY) && 3961da177e4SLinus Torvalds !xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) 3971da177e4SLinus Torvalds goto discard; 3981da177e4SLinus Torvalds 3990e219ae4SPaolo Abeni ret = INDIRECT_CALL_2(ipprot->handler, tcp_v6_rcv, udpv6_rcv, 4000e219ae4SPaolo Abeni skb); 4014c64242aSTom Herbert if (ret > 0) { 4024c64242aSTom Herbert if (ipprot->flags & INET6_PROTO_FINAL) { 4034c64242aSTom Herbert /* Not an extension header, most likely UDP 4044c64242aSTom Herbert * encapsulation. Use return value as nexthdr 4054c64242aSTom Herbert * protocol not nhoff (which presumably is 4064c64242aSTom Herbert * not set by handler). 4074c64242aSTom Herbert */ 4084c64242aSTom Herbert nexthdr = ret; 4094c64242aSTom Herbert goto resubmit_final; 4104c64242aSTom Herbert } else { 4111da177e4SLinus Torvalds goto resubmit; 4124c64242aSTom Herbert } 4134c64242aSTom Herbert } else if (ret == 0) { 4141d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 4154c64242aSTom Herbert } 4161da177e4SLinus Torvalds } else { 41769d6da0bSPavel Emelyanov if (!raw) { 4181da177e4SLinus Torvalds if (xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) { 4191d015503SEric Dumazet __IP6_INC_STATS(net, idev, 420483a47d2SDenis V. Lunev IPSTATS_MIB_INUNKNOWNPROTOS); 421fad87acaSPatrick McHardy icmpv6_send(skb, ICMPV6_PARAMPROB, 4223ffe533cSAlexey Dobriyan ICMPV6_UNK_NEXTHDR, nhoff); 4231da177e4SLinus Torvalds } 4241da177e4SLinus Torvalds kfree_skb(skb); 425d8c6f4b9SNeil Horman } else { 4261d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 427d8c6f4b9SNeil Horman consume_skb(skb); 428d8c6f4b9SNeil Horman } 4291da177e4SLinus Torvalds } 43080bde363SPaolo Abeni return; 4311da177e4SLinus Torvalds 4321da177e4SLinus Torvalds discard: 4331d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 4341da177e4SLinus Torvalds kfree_skb(skb); 43580bde363SPaolo Abeni } 43680bde363SPaolo Abeni 43780bde363SPaolo Abeni static int ip6_input_finish(struct net *net, struct sock *sk, struct sk_buff *skb) 43880bde363SPaolo Abeni { 43980bde363SPaolo Abeni rcu_read_lock(); 44080bde363SPaolo Abeni ip6_protocol_deliver_rcu(net, skb, 0, false); 44180bde363SPaolo Abeni rcu_read_unlock(); 44280bde363SPaolo Abeni 4431da177e4SLinus Torvalds return 0; 4441da177e4SLinus Torvalds } 4451da177e4SLinus Torvalds 4461da177e4SLinus Torvalds 4471da177e4SLinus Torvalds int ip6_input(struct sk_buff *skb) 4481da177e4SLinus Torvalds { 44929a26a56SEric W. Biederman return NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_IN, 45029a26a56SEric W. Biederman dev_net(skb->dev), NULL, skb, skb->dev, NULL, 4516e23ae2aSPatrick McHardy ip6_input_finish); 4521da177e4SLinus Torvalds } 453b4869aa2SDavid Ahern EXPORT_SYMBOL_GPL(ip6_input); 4541da177e4SLinus Torvalds 4551da177e4SLinus Torvalds int ip6_mc_input(struct sk_buff *skb) 4561da177e4SLinus Torvalds { 4575226b6a9SMike Manning int sdif = inet6_sdif(skb); 458b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 4595226b6a9SMike Manning struct net_device *dev; 460a50feda5SEric Dumazet bool deliver; 4611da177e4SLinus Torvalds 462c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(dev_net(skb_dst(skb)->dev), 463bdb7cc64SStephen Suryaputra __in6_dev_get_safely(skb->dev), IPSTATS_MIB_INMCAST, 464edf391ffSNeil Horman skb->len); 4651da177e4SLinus Torvalds 4665226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 4675226b6a9SMike Manning if (sdif) { 4685226b6a9SMike Manning rcu_read_lock(); 4695226b6a9SMike Manning dev = dev_get_by_index_rcu(dev_net(skb->dev), sdif); 4705226b6a9SMike Manning if (!dev) { 4715226b6a9SMike Manning rcu_read_unlock(); 4725226b6a9SMike Manning kfree_skb(skb); 4735226b6a9SMike Manning return -ENODEV; 4745226b6a9SMike Manning } 4755226b6a9SMike Manning } else { 4765226b6a9SMike Manning dev = skb->dev; 4775226b6a9SMike Manning } 4785226b6a9SMike Manning 4790660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 4805226b6a9SMike Manning deliver = ipv6_chk_mcast_addr(dev, &hdr->daddr, NULL); 4815226b6a9SMike Manning if (sdif) 4825226b6a9SMike Manning rcu_read_unlock(); 4831da177e4SLinus Torvalds 4847bc570c8SYOSHIFUJI Hideaki #ifdef CONFIG_IPV6_MROUTE 4851da177e4SLinus Torvalds /* 4867bc570c8SYOSHIFUJI Hideaki * IPv6 multicast router mode is now supported ;) 4871da177e4SLinus Torvalds */ 48853b7997fSYOSHIFUJI Hideaki if (dev_net(skb->dev)->ipv6.devconf_all->mc_forwarding && 489ddf64354SHannes Frederic Sowa !(ipv6_addr_type(&hdr->daddr) & 490ddf64354SHannes Frederic Sowa (IPV6_ADDR_LOOPBACK|IPV6_ADDR_LINKLOCAL)) && 4917bc570c8SYOSHIFUJI Hideaki likely(!(IP6CB(skb)->flags & IP6SKB_FORWARDED))) { 4927bc570c8SYOSHIFUJI Hideaki /* 4937bc570c8SYOSHIFUJI Hideaki * Okay, we try to forward - split and duplicate 4947bc570c8SYOSHIFUJI Hideaki * packets. 4957bc570c8SYOSHIFUJI Hideaki */ 4961da177e4SLinus Torvalds struct sk_buff *skb2; 4977bc570c8SYOSHIFUJI Hideaki struct inet6_skb_parm *opt = IP6CB(skb); 4981da177e4SLinus Torvalds 4997bc570c8SYOSHIFUJI Hideaki /* Check for MLD */ 500dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (unlikely(opt->flags & IP6SKB_ROUTERALERT)) { 5017bc570c8SYOSHIFUJI Hideaki /* Check if this is a mld message */ 5027bc570c8SYOSHIFUJI Hideaki u8 nexthdr = hdr->nexthdr; 50375f2811cSJesse Gross __be16 frag_off; 5047bc570c8SYOSHIFUJI Hideaki int offset; 5051da177e4SLinus Torvalds 5067bc570c8SYOSHIFUJI Hideaki /* Check if the value of Router Alert 5077bc570c8SYOSHIFUJI Hideaki * is for MLD (0x0000). 5087bc570c8SYOSHIFUJI Hideaki */ 509dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (opt->ra == htons(IPV6_OPT_ROUTERALERT_MLD)) { 510a50feda5SEric Dumazet deliver = false; 511aba6096bSYOSHIFUJI Hideaki 5127bc570c8SYOSHIFUJI Hideaki if (!ipv6_ext_hdr(nexthdr)) { 5137bc570c8SYOSHIFUJI Hideaki /* BUG */ 514aba6096bSYOSHIFUJI Hideaki goto out; 5151da177e4SLinus Torvalds } 5167bc570c8SYOSHIFUJI Hideaki offset = ipv6_skip_exthdr(skb, sizeof(*hdr), 51775f2811cSJesse Gross &nexthdr, &frag_off); 5187bc570c8SYOSHIFUJI Hideaki if (offset < 0) 519aba6096bSYOSHIFUJI Hideaki goto out; 5207bc570c8SYOSHIFUJI Hideaki 5214c938d22SAngga if (ipv6_is_mld(skb, nexthdr, offset)) 522a50feda5SEric Dumazet deliver = true; 5234c938d22SAngga 5244c938d22SAngga goto out; 5257bc570c8SYOSHIFUJI Hideaki } 5267bc570c8SYOSHIFUJI Hideaki /* unknown RA - process it normally */ 5277bc570c8SYOSHIFUJI Hideaki } 5287bc570c8SYOSHIFUJI Hideaki 5297bc570c8SYOSHIFUJI Hideaki if (deliver) 5307bc570c8SYOSHIFUJI Hideaki skb2 = skb_clone(skb, GFP_ATOMIC); 5317bc570c8SYOSHIFUJI Hideaki else { 5327bc570c8SYOSHIFUJI Hideaki skb2 = skb; 5337bc570c8SYOSHIFUJI Hideaki skb = NULL; 5347bc570c8SYOSHIFUJI Hideaki } 5357bc570c8SYOSHIFUJI Hideaki 5367bc570c8SYOSHIFUJI Hideaki if (skb2) { 5377bc570c8SYOSHIFUJI Hideaki ip6_mr_input(skb2); 5381da177e4SLinus Torvalds } 5391da177e4SLinus Torvalds } 5407bc570c8SYOSHIFUJI Hideaki out: 541aba6096bSYOSHIFUJI Hideaki #endif 542aba6096bSYOSHIFUJI Hideaki if (likely(deliver)) 5431da177e4SLinus Torvalds ip6_input(skb); 544aba6096bSYOSHIFUJI Hideaki else { 5451da177e4SLinus Torvalds /* discard */ 5461da177e4SLinus Torvalds kfree_skb(skb); 547aba6096bSYOSHIFUJI Hideaki } 5481da177e4SLinus Torvalds 5491da177e4SLinus Torvalds return 0; 5501da177e4SLinus Torvalds } 551