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 48d8269e2cSEdward Cree static void ip6_rcv_finish_core(struct net *net, struct sock *sk, 49d8269e2cSEdward Cree struct sk_buff *skb) 501da177e4SLinus Torvalds { 51*11052589SKuniyuki Iwashima if (READ_ONCE(net->ipv4.sysctl_ip_early_demux) && 52*11052589SKuniyuki Iwashima !skb_dst(skb) && !skb->sk) { 53*11052589SKuniyuki Iwashima switch (ipv6_hdr(skb)->nexthdr) { 54*11052589SKuniyuki Iwashima case IPPROTO_TCP: 55*11052589SKuniyuki Iwashima if (READ_ONCE(net->ipv4.sysctl_tcp_early_demux)) 56*11052589SKuniyuki Iwashima tcp_v6_early_demux(skb); 57*11052589SKuniyuki Iwashima break; 58*11052589SKuniyuki Iwashima case IPPROTO_UDP: 59*11052589SKuniyuki Iwashima if (READ_ONCE(net->ipv4.sysctl_udp_early_demux)) 60*11052589SKuniyuki Iwashima udp_v6_early_demux(skb); 61*11052589SKuniyuki Iwashima break; 62c7109986SEric Dumazet } 63*11052589SKuniyuki Iwashima } 64*11052589SKuniyuki Iwashima 6548fb6b55SWei-Chun Chao if (!skb_valid_dst(skb)) 661da177e4SLinus Torvalds ip6_route_input(skb); 67d8269e2cSEdward Cree } 68d8269e2cSEdward Cree 69d8269e2cSEdward Cree int ip6_rcv_finish(struct net *net, struct sock *sk, struct sk_buff *skb) 70d8269e2cSEdward Cree { 71d8269e2cSEdward Cree /* if ingress device is enslaved to an L3 master device pass the 72d8269e2cSEdward Cree * skb to its handler for processing 73d8269e2cSEdward Cree */ 74d8269e2cSEdward Cree skb = l3mdev_ip6_rcv(skb); 75d8269e2cSEdward Cree if (!skb) 76d8269e2cSEdward Cree return NET_RX_SUCCESS; 77d8269e2cSEdward Cree ip6_rcv_finish_core(net, sk, skb); 781da177e4SLinus Torvalds 791da177e4SLinus Torvalds return dst_input(skb); 801da177e4SLinus Torvalds } 811da177e4SLinus Torvalds 82d8269e2cSEdward Cree static void ip6_sublist_rcv_finish(struct list_head *head) 83d8269e2cSEdward Cree { 84d8269e2cSEdward Cree struct sk_buff *skb, *next; 85d8269e2cSEdward Cree 86c7a42eb4SXin Long list_for_each_entry_safe(skb, next, head, list) { 87c7a42eb4SXin Long skb_list_del_init(skb); 88d8269e2cSEdward Cree dst_input(skb); 89d8269e2cSEdward Cree } 90c7a42eb4SXin Long } 91d8269e2cSEdward Cree 92197dbf24SPaolo Abeni static bool ip6_can_use_hint(const struct sk_buff *skb, 93197dbf24SPaolo Abeni const struct sk_buff *hint) 94197dbf24SPaolo Abeni { 95197dbf24SPaolo Abeni return hint && !skb_dst(skb) && 96197dbf24SPaolo Abeni ipv6_addr_equal(&ipv6_hdr(hint)->daddr, &ipv6_hdr(skb)->daddr); 97197dbf24SPaolo Abeni } 98197dbf24SPaolo Abeni 99197dbf24SPaolo Abeni static struct sk_buff *ip6_extract_route_hint(const struct net *net, 100197dbf24SPaolo Abeni struct sk_buff *skb) 101197dbf24SPaolo Abeni { 102197dbf24SPaolo Abeni if (fib6_routes_require_src(net) || fib6_has_custom_rules(net)) 103197dbf24SPaolo Abeni return NULL; 104197dbf24SPaolo Abeni 105197dbf24SPaolo Abeni return skb; 106197dbf24SPaolo Abeni } 107197dbf24SPaolo Abeni 108d8269e2cSEdward Cree static void ip6_list_rcv_finish(struct net *net, struct sock *sk, 109d8269e2cSEdward Cree struct list_head *head) 110d8269e2cSEdward Cree { 111197dbf24SPaolo Abeni struct sk_buff *skb, *next, *hint = NULL; 112d8269e2cSEdward Cree struct dst_entry *curr_dst = NULL; 113d8269e2cSEdward Cree struct list_head sublist; 114d8269e2cSEdward Cree 115d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 116d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 117d8269e2cSEdward Cree struct dst_entry *dst; 118d8269e2cSEdward Cree 11922f6bbb7SEdward Cree skb_list_del_init(skb); 120d8269e2cSEdward Cree /* if ingress device is enslaved to an L3 master device pass the 121d8269e2cSEdward Cree * skb to its handler for processing 122d8269e2cSEdward Cree */ 123d8269e2cSEdward Cree skb = l3mdev_ip6_rcv(skb); 124d8269e2cSEdward Cree if (!skb) 125d8269e2cSEdward Cree continue; 126197dbf24SPaolo Abeni 127197dbf24SPaolo Abeni if (ip6_can_use_hint(skb, hint)) 128197dbf24SPaolo Abeni skb_dst_copy(skb, hint); 129197dbf24SPaolo Abeni else 130d8269e2cSEdward Cree ip6_rcv_finish_core(net, sk, skb); 131d8269e2cSEdward Cree dst = skb_dst(skb); 132d8269e2cSEdward Cree if (curr_dst != dst) { 133197dbf24SPaolo Abeni hint = ip6_extract_route_hint(net, skb); 134197dbf24SPaolo Abeni 135d8269e2cSEdward Cree /* dispatch old sublist */ 136d8269e2cSEdward Cree if (!list_empty(&sublist)) 137d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 138d8269e2cSEdward Cree /* start new sublist */ 139d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 140d8269e2cSEdward Cree curr_dst = dst; 141d8269e2cSEdward Cree } 142d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 143d8269e2cSEdward Cree } 144d8269e2cSEdward Cree /* dispatch final sublist */ 145d8269e2cSEdward Cree ip6_sublist_rcv_finish(&sublist); 146d8269e2cSEdward Cree } 147d8269e2cSEdward Cree 148d8269e2cSEdward Cree static struct sk_buff *ip6_rcv_core(struct sk_buff *skb, struct net_device *dev, 149d8269e2cSEdward Cree struct net *net) 1501da177e4SLinus Torvalds { 1514daf841aSMenglong Dong enum skb_drop_reason reason; 152b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 1531da177e4SLinus Torvalds u32 pkt_len; 154a11d206dSYOSHIFUJI Hideaki struct inet6_dev *idev; 1551da177e4SLinus Torvalds 156a11d206dSYOSHIFUJI Hideaki if (skb->pkt_type == PACKET_OTHERHOST) { 157794c24e9SJeffrey Ji dev_core_stats_rx_otherhost_dropped_inc(skb->dev); 1584daf841aSMenglong Dong kfree_skb_reason(skb, SKB_DROP_REASON_OTHERHOST); 159d8269e2cSEdward Cree return NULL; 160a11d206dSYOSHIFUJI Hideaki } 1611da177e4SLinus Torvalds 162a11d206dSYOSHIFUJI Hideaki rcu_read_lock(); 163a11d206dSYOSHIFUJI Hideaki 164a11d206dSYOSHIFUJI Hideaki idev = __in6_dev_get(skb->dev); 165a11d206dSYOSHIFUJI Hideaki 166c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(net, idev, IPSTATS_MIB_IN, skb->len); 1671da177e4SLinus Torvalds 1684daf841aSMenglong Dong SKB_DR_SET(reason, NOT_SPECIFIED); 169778d80beSYOSHIFUJI Hideaki if ((skb = skb_share_check(skb, GFP_ATOMIC)) == NULL || 170778d80beSYOSHIFUJI Hideaki !idev || unlikely(idev->cnf.disable_ipv6)) { 1711d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 1720339d25aSEric Dumazet if (idev && unlikely(idev->cnf.disable_ipv6)) 1734daf841aSMenglong Dong SKB_DR_SET(reason, IPV6DISABLED); 17471f6f6dfSJesper Nilsson goto drop; 1751da177e4SLinus Torvalds } 1761da177e4SLinus Torvalds 1776b7fdc3aSGuillaume Chazarain memset(IP6CB(skb), 0, sizeof(struct inet6_skb_parm)); 1786b7fdc3aSGuillaume Chazarain 1791da177e4SLinus Torvalds /* 1801da177e4SLinus Torvalds * Store incoming device index. When the packet will 1811da177e4SLinus Torvalds * be queued, we cannot refer to skb->dev anymore. 1821da177e4SLinus Torvalds * 1831da177e4SLinus Torvalds * BTW, when we send a packet for our own local address on a 1841da177e4SLinus Torvalds * non-loopback interface (e.g. ethX), it is being delivered 185de3cb747SDaniel Lezcano * via the loopback interface (lo) here; skb->dev = loopback_dev. 1861da177e4SLinus Torvalds * It, however, should be considered as if it is being 1871da177e4SLinus Torvalds * arrived via the sending interface (ethX), because of the 1881da177e4SLinus Torvalds * nature of scoping architecture. --yoshfuji 1891da177e4SLinus Torvalds */ 19048fb6b55SWei-Chun Chao IP6CB(skb)->iif = skb_valid_dst(skb) ? ip6_dst_idev(skb_dst(skb))->dev->ifindex : dev->ifindex; 1911da177e4SLinus Torvalds 1922889139aSHerbert Xu if (unlikely(!pskb_may_pull(skb, sizeof(*hdr)))) 1931da177e4SLinus Torvalds goto err; 1941da177e4SLinus Torvalds 1950660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 1961da177e4SLinus Torvalds 1974daf841aSMenglong Dong if (hdr->version != 6) { 1984daf841aSMenglong Dong SKB_DR_SET(reason, UNHANDLED_PROTO); 1991da177e4SLinus Torvalds goto err; 2004daf841aSMenglong Dong } 2011da177e4SLinus Torvalds 2021d015503SEric Dumazet __IP6_ADD_STATS(net, idev, 2031f07d03eSEric Dumazet IPSTATS_MIB_NOECTPKTS + 2041f07d03eSEric Dumazet (ipv6_get_dsfield(hdr) & INET_ECN_MASK), 2051f07d03eSEric Dumazet max_t(unsigned short, 1, skb_shinfo(skb)->gso_segs)); 206f630e43aSYOSHIFUJI Hideaki /* 207f630e43aSYOSHIFUJI Hideaki * RFC4291 2.5.3 2080aa8c13eSFlorian Westphal * The loopback address must not be used as the source address in IPv6 2090aa8c13eSFlorian Westphal * packets that are sent outside of a single node. [..] 210f630e43aSYOSHIFUJI Hideaki * A packet received on an interface with a destination address 211f630e43aSYOSHIFUJI Hideaki * of loopback must be dropped. 212f630e43aSYOSHIFUJI Hideaki */ 2130aa8c13eSFlorian Westphal if ((ipv6_addr_loopback(&hdr->saddr) || 2140aa8c13eSFlorian Westphal ipv6_addr_loopback(&hdr->daddr)) && 2153ede0bbcSRobert Shearman !(dev->flags & IFF_LOOPBACK) && 2163ede0bbcSRobert Shearman !netif_is_l3_master(dev)) 217f630e43aSYOSHIFUJI Hideaki goto err; 218f630e43aSYOSHIFUJI Hideaki 2191c4a154eSHannes Frederic Sowa /* RFC4291 Errata ID: 3480 2201c4a154eSHannes Frederic Sowa * Interface-Local scope spans only a single interface on a 2211c4a154eSHannes Frederic Sowa * node and is useful only for loopback transmission of 2221c4a154eSHannes Frederic Sowa * multicast. Packets with interface-local scope received 2231c4a154eSHannes Frederic Sowa * from another node must be discarded. 2241c4a154eSHannes Frederic Sowa */ 2251c4a154eSHannes Frederic Sowa if (!(skb->pkt_type == PACKET_LOOPBACK || 2261c4a154eSHannes Frederic Sowa dev->flags & IFF_LOOPBACK) && 2271c4a154eSHannes Frederic Sowa ipv6_addr_is_multicast(&hdr->daddr) && 2281c4a154eSHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 1) 2291c4a154eSHannes Frederic Sowa goto err; 2301c4a154eSHannes Frederic Sowa 231abbc3043SJohannes Berg /* If enabled, drop unicast packets that were encapsulated in link-layer 232abbc3043SJohannes Berg * multicast or broadcast to protected against the so-called "hole-196" 233abbc3043SJohannes Berg * attack in 802.11 wireless. 234abbc3043SJohannes Berg */ 235abbc3043SJohannes Berg if (!ipv6_addr_is_multicast(&hdr->daddr) && 236abbc3043SJohannes Berg (skb->pkt_type == PACKET_BROADCAST || 237abbc3043SJohannes Berg skb->pkt_type == PACKET_MULTICAST) && 2384daf841aSMenglong Dong idev->cnf.drop_unicast_in_l2_multicast) { 2394daf841aSMenglong Dong SKB_DR_SET(reason, UNICAST_IN_L2_MULTICAST); 240abbc3043SJohannes Berg goto err; 2414daf841aSMenglong Dong } 242abbc3043SJohannes Berg 24320314092SHannes Frederic Sowa /* RFC4291 2.7 24420314092SHannes Frederic Sowa * Nodes must not originate a packet to a multicast address whose scope 24520314092SHannes Frederic Sowa * field contains the reserved value 0; if such a packet is received, it 24620314092SHannes Frederic Sowa * must be silently dropped. 24720314092SHannes Frederic Sowa */ 24820314092SHannes Frederic Sowa if (ipv6_addr_is_multicast(&hdr->daddr) && 24920314092SHannes Frederic Sowa IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 0) 25020314092SHannes Frederic Sowa goto err; 25120314092SHannes Frederic Sowa 252c457338dSBrian Haley /* 253c457338dSBrian Haley * RFC4291 2.7 254c457338dSBrian Haley * Multicast addresses must not be used as source addresses in IPv6 255c457338dSBrian Haley * packets or appear in any Routing header. 256c457338dSBrian Haley */ 257c457338dSBrian Haley if (ipv6_addr_is_multicast(&hdr->saddr)) 258c457338dSBrian Haley goto err; 259c457338dSBrian Haley 260b0e380b1SArnaldo Carvalho de Melo skb->transport_header = skb->network_header + sizeof(*hdr); 261951dbc8aSPatrick McHardy IP6CB(skb)->nhoff = offsetof(struct ipv6hdr, nexthdr); 262951dbc8aSPatrick McHardy 2631da177e4SLinus Torvalds pkt_len = ntohs(hdr->payload_len); 2641da177e4SLinus Torvalds 2651da177e4SLinus Torvalds /* pkt_len may be zero if Jumbo payload option is present */ 2661da177e4SLinus Torvalds if (pkt_len || hdr->nexthdr != NEXTHDR_HOP) { 26760e5c166SMitsuru Chinen if (pkt_len + sizeof(struct ipv6hdr) > skb->len) { 2681d015503SEric Dumazet __IP6_INC_STATS(net, 269483a47d2SDenis V. Lunev idev, IPSTATS_MIB_INTRUNCATEDPKTS); 2704daf841aSMenglong Dong SKB_DR_SET(reason, PKT_TOO_SMALL); 27160e5c166SMitsuru Chinen goto drop; 27260e5c166SMitsuru Chinen } 2734daf841aSMenglong Dong if (pskb_trim_rcsum(skb, pkt_len + sizeof(struct ipv6hdr))) 2744daf841aSMenglong Dong goto err; 2750660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 2761da177e4SLinus Torvalds } 2771da177e4SLinus Torvalds 2781da177e4SLinus Torvalds if (hdr->nexthdr == NEXTHDR_HOP) { 279e5bbef20SHerbert Xu if (ipv6_parse_hopopts(skb) < 0) { 2801d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 281a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 282d8269e2cSEdward Cree return NULL; 2831da177e4SLinus Torvalds } 2841da177e4SLinus Torvalds } 2851da177e4SLinus Torvalds 286a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 287a11d206dSYOSHIFUJI Hideaki 28871f9dacdSHerbert Xu /* Must drop socket now because of tproxy. */ 289cf7fbe66SJoe Stringer if (!skb_sk_is_prefetched(skb)) 29071f9dacdSHerbert Xu skb_orphan(skb); 29171f9dacdSHerbert Xu 292d8269e2cSEdward Cree return skb; 2931da177e4SLinus Torvalds err: 2941d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INHDRERRORS); 2954daf841aSMenglong Dong SKB_DR_OR(reason, IP_INHDR); 2961da177e4SLinus Torvalds drop: 297a11d206dSYOSHIFUJI Hideaki rcu_read_unlock(); 2984daf841aSMenglong Dong kfree_skb_reason(skb, reason); 299d8269e2cSEdward Cree return NULL; 300d8269e2cSEdward Cree } 301d8269e2cSEdward Cree 302d8269e2cSEdward Cree int ipv6_rcv(struct sk_buff *skb, struct net_device *dev, struct packet_type *pt, struct net_device *orig_dev) 303d8269e2cSEdward Cree { 304d8269e2cSEdward Cree struct net *net = dev_net(skb->dev); 305d8269e2cSEdward Cree 306d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 307d8269e2cSEdward Cree if (skb == NULL) 3085c91faceSMark Smith return NET_RX_DROP; 309d8269e2cSEdward Cree return NF_HOOK(NFPROTO_IPV6, NF_INET_PRE_ROUTING, 310d8269e2cSEdward Cree net, NULL, skb, dev, NULL, 311d8269e2cSEdward Cree ip6_rcv_finish); 312d8269e2cSEdward Cree } 313d8269e2cSEdward Cree 314d8269e2cSEdward Cree static void ip6_sublist_rcv(struct list_head *head, struct net_device *dev, 315d8269e2cSEdward Cree struct net *net) 316d8269e2cSEdward Cree { 317d8269e2cSEdward Cree NF_HOOK_LIST(NFPROTO_IPV6, NF_INET_PRE_ROUTING, net, NULL, 318d8269e2cSEdward Cree head, dev, NULL, ip6_rcv_finish); 319d8269e2cSEdward Cree ip6_list_rcv_finish(net, NULL, head); 320d8269e2cSEdward Cree } 321d8269e2cSEdward Cree 322d8269e2cSEdward Cree /* Receive a list of IPv6 packets */ 323d8269e2cSEdward Cree void ipv6_list_rcv(struct list_head *head, struct packet_type *pt, 324d8269e2cSEdward Cree struct net_device *orig_dev) 325d8269e2cSEdward Cree { 326d8269e2cSEdward Cree struct net_device *curr_dev = NULL; 327d8269e2cSEdward Cree struct net *curr_net = NULL; 328d8269e2cSEdward Cree struct sk_buff *skb, *next; 329d8269e2cSEdward Cree struct list_head sublist; 330d8269e2cSEdward Cree 331d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 332d8269e2cSEdward Cree list_for_each_entry_safe(skb, next, head, list) { 333d8269e2cSEdward Cree struct net_device *dev = skb->dev; 334d8269e2cSEdward Cree struct net *net = dev_net(dev); 335d8269e2cSEdward Cree 33622f6bbb7SEdward Cree skb_list_del_init(skb); 337d8269e2cSEdward Cree skb = ip6_rcv_core(skb, dev, net); 338d8269e2cSEdward Cree if (skb == NULL) 339d8269e2cSEdward Cree continue; 340d8269e2cSEdward Cree 341d8269e2cSEdward Cree if (curr_dev != dev || curr_net != net) { 342d8269e2cSEdward Cree /* dispatch old sublist */ 343d8269e2cSEdward Cree if (!list_empty(&sublist)) 344d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 345d8269e2cSEdward Cree /* start new sublist */ 346d8269e2cSEdward Cree INIT_LIST_HEAD(&sublist); 347d8269e2cSEdward Cree curr_dev = dev; 348d8269e2cSEdward Cree curr_net = net; 349d8269e2cSEdward Cree } 350d8269e2cSEdward Cree list_add_tail(&skb->list, &sublist); 351d8269e2cSEdward Cree } 352d8269e2cSEdward Cree /* dispatch final sublist */ 35351210ad5SFlorian Westphal if (!list_empty(&sublist)) 354d8269e2cSEdward Cree ip6_sublist_rcv(&sublist, curr_dev, curr_net); 3551da177e4SLinus Torvalds } 3561da177e4SLinus Torvalds 3570e219ae4SPaolo Abeni INDIRECT_CALLABLE_DECLARE(int tcp_v6_rcv(struct sk_buff *)); 3580e219ae4SPaolo Abeni 3591da177e4SLinus Torvalds /* 3601da177e4SLinus Torvalds * Deliver the packet to the host 3611da177e4SLinus Torvalds */ 36280bde363SPaolo Abeni void ip6_protocol_deliver_rcu(struct net *net, struct sk_buff *skb, int nexthdr, 36380bde363SPaolo Abeni bool have_final) 3641da177e4SLinus Torvalds { 36541135cc8SAlexey Dobriyan const struct inet6_protocol *ipprot; 366f9242b6bSDavid S. Miller struct inet6_dev *idev; 3671da177e4SLinus Torvalds unsigned int nhoff; 368eeab7e7fSMenglong Dong SKB_DR(reason); 369a50feda5SEric Dumazet bool raw; 3701da177e4SLinus Torvalds 3711da177e4SLinus Torvalds /* 3721da177e4SLinus Torvalds * Parse extension headers 3731da177e4SLinus Torvalds */ 3741da177e4SLinus Torvalds 3751b0ccfe5SDavid S. Miller resubmit: 376adf30907SEric Dumazet idev = ip6_dst_idev(skb_dst(skb)); 37780bde363SPaolo Abeni nhoff = IP6CB(skb)->nhoff; 37880bde363SPaolo Abeni if (!have_final) { 379ea2ae17dSArnaldo Carvalho de Melo if (!pskb_pull(skb, skb_transport_offset(skb))) 3801da177e4SLinus Torvalds goto discard; 381d56f90a7SArnaldo Carvalho de Melo nexthdr = skb_network_header(skb)[nhoff]; 38280bde363SPaolo Abeni } 3831da177e4SLinus Torvalds 3844c64242aSTom Herbert resubmit_final: 38569d6da0bSPavel Emelyanov raw = raw6_local_deliver(skb, nexthdr); 386e5d08d71SIan Morris ipprot = rcu_dereference(inet6_protos[nexthdr]); 38753b24b8fSIan Morris if (ipprot) { 3881da177e4SLinus Torvalds int ret; 3891da177e4SLinus Torvalds 3901da44f9cSTom Herbert if (have_final) { 3911da44f9cSTom Herbert if (!(ipprot->flags & INET6_PROTO_FINAL)) { 3921da44f9cSTom Herbert /* Once we've seen a final protocol don't 3931da44f9cSTom Herbert * allow encapsulation on any non-final 3941da44f9cSTom Herbert * ones. This allows foo in UDP encapsulation 3951da44f9cSTom Herbert * to work. 3961da44f9cSTom Herbert */ 3971da44f9cSTom Herbert goto discard; 3981da44f9cSTom Herbert } 3991da44f9cSTom Herbert } else if (ipprot->flags & INET6_PROTO_FINAL) { 400b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 4015226b6a9SMike Manning int sdif = inet6_sdif(skb); 4025226b6a9SMike Manning struct net_device *dev; 4031da177e4SLinus Torvalds 4041da44f9cSTom Herbert /* Only do this once for first final protocol */ 4051da44f9cSTom Herbert have_final = true; 4061da44f9cSTom Herbert 4079fb9cbb1SYasuyuki Kozakai /* Free reference early: we don't need it any more, 4089fb9cbb1SYasuyuki Kozakai and it may hold ip_conntrack module loaded 4099fb9cbb1SYasuyuki Kozakai indefinitely. */ 410895b5c9fSFlorian Westphal nf_reset_ct(skb); 4119fb9cbb1SYasuyuki Kozakai 412d56f90a7SArnaldo Carvalho de Melo skb_postpull_rcsum(skb, skb_network_header(skb), 413cfe1fc77SArnaldo Carvalho de Melo skb_network_header_len(skb)); 4140660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 4155226b6a9SMike Manning 4165226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 4175226b6a9SMike Manning if (sdif) { 4185226b6a9SMike Manning dev = dev_get_by_index_rcu(net, sdif); 4195226b6a9SMike Manning if (!dev) 4205226b6a9SMike Manning goto discard; 4215226b6a9SMike Manning } else { 4225226b6a9SMike Manning dev = skb->dev; 4235226b6a9SMike Manning } 4245226b6a9SMike Manning 4251da177e4SLinus Torvalds if (ipv6_addr_is_multicast(&hdr->daddr) && 4265226b6a9SMike Manning !ipv6_chk_mcast_addr(dev, &hdr->daddr, 4271da177e4SLinus Torvalds &hdr->saddr) && 428eeab7e7fSMenglong Dong !ipv6_is_mld(skb, nexthdr, skb_network_header_len(skb))) { 429eeab7e7fSMenglong Dong SKB_DR_SET(reason, IP_INADDRERRORS); 4301da177e4SLinus Torvalds goto discard; 4311da177e4SLinus Torvalds } 432eeab7e7fSMenglong Dong } 4331da177e4SLinus Torvalds if (!(ipprot->flags & INET6_PROTO_NOPOLICY) && 434eeab7e7fSMenglong Dong !xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) { 435eeab7e7fSMenglong Dong SKB_DR_SET(reason, XFRM_POLICY); 4361da177e4SLinus Torvalds goto discard; 437eeab7e7fSMenglong Dong } 4381da177e4SLinus Torvalds 4390e219ae4SPaolo Abeni ret = INDIRECT_CALL_2(ipprot->handler, tcp_v6_rcv, udpv6_rcv, 4400e219ae4SPaolo Abeni skb); 4414c64242aSTom Herbert if (ret > 0) { 4424c64242aSTom Herbert if (ipprot->flags & INET6_PROTO_FINAL) { 4434c64242aSTom Herbert /* Not an extension header, most likely UDP 4444c64242aSTom Herbert * encapsulation. Use return value as nexthdr 4454c64242aSTom Herbert * protocol not nhoff (which presumably is 4464c64242aSTom Herbert * not set by handler). 4474c64242aSTom Herbert */ 4484c64242aSTom Herbert nexthdr = ret; 4494c64242aSTom Herbert goto resubmit_final; 4504c64242aSTom Herbert } else { 4511da177e4SLinus Torvalds goto resubmit; 4524c64242aSTom Herbert } 4534c64242aSTom Herbert } else if (ret == 0) { 4541d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 4554c64242aSTom Herbert } 4561da177e4SLinus Torvalds } else { 45769d6da0bSPavel Emelyanov if (!raw) { 4581da177e4SLinus Torvalds if (xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) { 4591d015503SEric Dumazet __IP6_INC_STATS(net, idev, 460483a47d2SDenis V. Lunev IPSTATS_MIB_INUNKNOWNPROTOS); 461fad87acaSPatrick McHardy icmpv6_send(skb, ICMPV6_PARAMPROB, 4623ffe533cSAlexey Dobriyan ICMPV6_UNK_NEXTHDR, nhoff); 463eeab7e7fSMenglong Dong SKB_DR_SET(reason, IP_NOPROTO); 464eeab7e7fSMenglong Dong } else { 465eeab7e7fSMenglong Dong SKB_DR_SET(reason, XFRM_POLICY); 4661da177e4SLinus Torvalds } 467eeab7e7fSMenglong Dong kfree_skb_reason(skb, reason); 468d8c6f4b9SNeil Horman } else { 4691d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDELIVERS); 470d8c6f4b9SNeil Horman consume_skb(skb); 471d8c6f4b9SNeil Horman } 4721da177e4SLinus Torvalds } 47380bde363SPaolo Abeni return; 4741da177e4SLinus Torvalds 4751da177e4SLinus Torvalds discard: 4761d015503SEric Dumazet __IP6_INC_STATS(net, idev, IPSTATS_MIB_INDISCARDS); 477eeab7e7fSMenglong Dong kfree_skb_reason(skb, reason); 47880bde363SPaolo Abeni } 47980bde363SPaolo Abeni 48080bde363SPaolo Abeni static int ip6_input_finish(struct net *net, struct sock *sk, struct sk_buff *skb) 48180bde363SPaolo Abeni { 482cd14e9b7SMartin KaFai Lau skb_clear_delivery_time(skb); 48380bde363SPaolo Abeni rcu_read_lock(); 48480bde363SPaolo Abeni ip6_protocol_deliver_rcu(net, skb, 0, false); 48580bde363SPaolo Abeni rcu_read_unlock(); 48680bde363SPaolo Abeni 4871da177e4SLinus Torvalds return 0; 4881da177e4SLinus Torvalds } 4891da177e4SLinus Torvalds 4901da177e4SLinus Torvalds 4911da177e4SLinus Torvalds int ip6_input(struct sk_buff *skb) 4921da177e4SLinus Torvalds { 49329a26a56SEric W. Biederman return NF_HOOK(NFPROTO_IPV6, NF_INET_LOCAL_IN, 49429a26a56SEric W. Biederman dev_net(skb->dev), NULL, skb, skb->dev, NULL, 4956e23ae2aSPatrick McHardy ip6_input_finish); 4961da177e4SLinus Torvalds } 497b4869aa2SDavid Ahern EXPORT_SYMBOL_GPL(ip6_input); 4981da177e4SLinus Torvalds 4991da177e4SLinus Torvalds int ip6_mc_input(struct sk_buff *skb) 5001da177e4SLinus Torvalds { 5015226b6a9SMike Manning int sdif = inet6_sdif(skb); 502b71d1d42SEric Dumazet const struct ipv6hdr *hdr; 5035226b6a9SMike Manning struct net_device *dev; 504a50feda5SEric Dumazet bool deliver; 5051da177e4SLinus Torvalds 506c2005eb0SEric Dumazet __IP6_UPD_PO_STATS(dev_net(skb_dst(skb)->dev), 507bdb7cc64SStephen Suryaputra __in6_dev_get_safely(skb->dev), IPSTATS_MIB_INMCAST, 508edf391ffSNeil Horman skb->len); 5091da177e4SLinus Torvalds 5105226b6a9SMike Manning /* skb->dev passed may be master dev for vrfs. */ 5115226b6a9SMike Manning if (sdif) { 5125226b6a9SMike Manning rcu_read_lock(); 5135226b6a9SMike Manning dev = dev_get_by_index_rcu(dev_net(skb->dev), sdif); 5145226b6a9SMike Manning if (!dev) { 5155226b6a9SMike Manning rcu_read_unlock(); 5165226b6a9SMike Manning kfree_skb(skb); 5175226b6a9SMike Manning return -ENODEV; 5185226b6a9SMike Manning } 5195226b6a9SMike Manning } else { 5205226b6a9SMike Manning dev = skb->dev; 5215226b6a9SMike Manning } 5225226b6a9SMike Manning 5230660e03fSArnaldo Carvalho de Melo hdr = ipv6_hdr(skb); 5245226b6a9SMike Manning deliver = ipv6_chk_mcast_addr(dev, &hdr->daddr, NULL); 5255226b6a9SMike Manning if (sdif) 5265226b6a9SMike Manning rcu_read_unlock(); 5271da177e4SLinus Torvalds 5287bc570c8SYOSHIFUJI Hideaki #ifdef CONFIG_IPV6_MROUTE 5291da177e4SLinus Torvalds /* 5307bc570c8SYOSHIFUJI Hideaki * IPv6 multicast router mode is now supported ;) 5311da177e4SLinus Torvalds */ 532145c7a79SEric Dumazet if (atomic_read(&dev_net(skb->dev)->ipv6.devconf_all->mc_forwarding) && 533ddf64354SHannes Frederic Sowa !(ipv6_addr_type(&hdr->daddr) & 534ddf64354SHannes Frederic Sowa (IPV6_ADDR_LOOPBACK|IPV6_ADDR_LINKLOCAL)) && 5357bc570c8SYOSHIFUJI Hideaki likely(!(IP6CB(skb)->flags & IP6SKB_FORWARDED))) { 5367bc570c8SYOSHIFUJI Hideaki /* 5377bc570c8SYOSHIFUJI Hideaki * Okay, we try to forward - split and duplicate 5387bc570c8SYOSHIFUJI Hideaki * packets. 5397bc570c8SYOSHIFUJI Hideaki */ 5401da177e4SLinus Torvalds struct sk_buff *skb2; 5417bc570c8SYOSHIFUJI Hideaki struct inet6_skb_parm *opt = IP6CB(skb); 5421da177e4SLinus Torvalds 5437bc570c8SYOSHIFUJI Hideaki /* Check for MLD */ 544dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (unlikely(opt->flags & IP6SKB_ROUTERALERT)) { 5457bc570c8SYOSHIFUJI Hideaki /* Check if this is a mld message */ 5467bc570c8SYOSHIFUJI Hideaki u8 nexthdr = hdr->nexthdr; 54775f2811cSJesse Gross __be16 frag_off; 5487bc570c8SYOSHIFUJI Hideaki int offset; 5491da177e4SLinus Torvalds 5507bc570c8SYOSHIFUJI Hideaki /* Check if the value of Router Alert 5517bc570c8SYOSHIFUJI Hideaki * is for MLD (0x0000). 5527bc570c8SYOSHIFUJI Hideaki */ 553dd3332bfSYOSHIFUJI Hideaki / 吉藤英明 if (opt->ra == htons(IPV6_OPT_ROUTERALERT_MLD)) { 554a50feda5SEric Dumazet deliver = false; 555aba6096bSYOSHIFUJI Hideaki 5567bc570c8SYOSHIFUJI Hideaki if (!ipv6_ext_hdr(nexthdr)) { 5577bc570c8SYOSHIFUJI Hideaki /* BUG */ 558aba6096bSYOSHIFUJI Hideaki goto out; 5591da177e4SLinus Torvalds } 5607bc570c8SYOSHIFUJI Hideaki offset = ipv6_skip_exthdr(skb, sizeof(*hdr), 56175f2811cSJesse Gross &nexthdr, &frag_off); 5627bc570c8SYOSHIFUJI Hideaki if (offset < 0) 563aba6096bSYOSHIFUJI Hideaki goto out; 5647bc570c8SYOSHIFUJI Hideaki 5654c938d22SAngga if (ipv6_is_mld(skb, nexthdr, offset)) 566a50feda5SEric Dumazet deliver = true; 5674c938d22SAngga 5684c938d22SAngga goto out; 5697bc570c8SYOSHIFUJI Hideaki } 5707bc570c8SYOSHIFUJI Hideaki /* unknown RA - process it normally */ 5717bc570c8SYOSHIFUJI Hideaki } 5727bc570c8SYOSHIFUJI Hideaki 5737bc570c8SYOSHIFUJI Hideaki if (deliver) 5747bc570c8SYOSHIFUJI Hideaki skb2 = skb_clone(skb, GFP_ATOMIC); 5757bc570c8SYOSHIFUJI Hideaki else { 5767bc570c8SYOSHIFUJI Hideaki skb2 = skb; 5777bc570c8SYOSHIFUJI Hideaki skb = NULL; 5787bc570c8SYOSHIFUJI Hideaki } 5797bc570c8SYOSHIFUJI Hideaki 5807bc570c8SYOSHIFUJI Hideaki if (skb2) { 5817bc570c8SYOSHIFUJI Hideaki ip6_mr_input(skb2); 5821da177e4SLinus Torvalds } 5831da177e4SLinus Torvalds } 5847bc570c8SYOSHIFUJI Hideaki out: 585aba6096bSYOSHIFUJI Hideaki #endif 586aba6096bSYOSHIFUJI Hideaki if (likely(deliver)) 5871da177e4SLinus Torvalds ip6_input(skb); 588aba6096bSYOSHIFUJI Hideaki else { 5891da177e4SLinus Torvalds /* discard */ 5901da177e4SLinus Torvalds kfree_skb(skb); 591aba6096bSYOSHIFUJI Hideaki } 5921da177e4SLinus Torvalds 5931da177e4SLinus Torvalds return 0; 5941da177e4SLinus Torvalds } 595