1 // SPDX-License-Identifier: GPL-2.0-only
3 * IPv6 library code, needed by static components when full IPv6 support is
4 * not configured or static. These functions are needed by GSO/GRO implementation.
6 #include <linux/export.h>
9 #include <net/ip6_fib.h>
10 #include <net/addrconf.h>
11 #include <net/secure_seq.h>
12 #include <linux/netfilter.h>
14 static u32
__ipv6_select_ident(struct net
*net
,
15 const struct in6_addr
*dst
,
16 const struct in6_addr
*src
)
21 } __aligned(SIPHASH_ALIGNMENT
) combined
= {
27 /* Note the following code is not safe, but this is okay. */
28 if (unlikely(siphash_key_is_zero(&net
->ipv4
.ip_id_key
)))
29 get_random_bytes(&net
->ipv4
.ip_id_key
,
30 sizeof(net
->ipv4
.ip_id_key
));
32 hash
= siphash(&combined
, sizeof(combined
), &net
->ipv4
.ip_id_key
);
34 /* Treat id of 0 as unset and if we get 0 back from ip_idents_reserve,
35 * set the hight order instead thus minimizing possible future
38 id
= ip_idents_reserve(hash
, 1);
45 /* This function exists only for tap drivers that must support broken
46 * clients requesting UFO without specifying an IPv6 fragment ID.
48 * This is similar to ipv6_select_ident() but we use an independent hash
49 * seed to limit information leakage.
51 * The network header must be set before calling this.
53 __be32
ipv6_proxy_select_ident(struct net
*net
, struct sk_buff
*skb
)
55 struct in6_addr buf
[2];
56 struct in6_addr
*addrs
;
59 addrs
= skb_header_pointer(skb
,
60 skb_network_offset(skb
) +
61 offsetof(struct ipv6hdr
, saddr
),
66 id
= __ipv6_select_ident(net
, &addrs
[1], &addrs
[0]);
69 EXPORT_SYMBOL_GPL(ipv6_proxy_select_ident
);
71 __be32
ipv6_select_ident(struct net
*net
,
72 const struct in6_addr
*daddr
,
73 const struct in6_addr
*saddr
)
77 id
= __ipv6_select_ident(net
, daddr
, saddr
);
80 EXPORT_SYMBOL(ipv6_select_ident
);
82 int ip6_find_1stfragopt(struct sk_buff
*skb
, u8
**nexthdr
)
84 unsigned int offset
= sizeof(struct ipv6hdr
);
85 unsigned int packet_len
= skb_tail_pointer(skb
) -
86 skb_network_header(skb
);
88 *nexthdr
= &ipv6_hdr(skb
)->nexthdr
;
90 while (offset
<= packet_len
) {
91 struct ipv6_opt_hdr
*exthdr
;
101 #if IS_ENABLED(CONFIG_IPV6_MIP6)
102 if (ipv6_find_tlv(skb
, offset
, IPV6_TLV_HAO
) >= 0)
112 if (offset
+ sizeof(struct ipv6_opt_hdr
) > packet_len
)
115 exthdr
= (struct ipv6_opt_hdr
*)(skb_network_header(skb
) +
117 offset
+= ipv6_optlen(exthdr
);
118 if (offset
> IPV6_MAXPLEN
)
120 *nexthdr
= &exthdr
->nexthdr
;
125 EXPORT_SYMBOL(ip6_find_1stfragopt
);
127 #if IS_ENABLED(CONFIG_IPV6)
128 int ip6_dst_hoplimit(struct dst_entry
*dst
)
130 int hoplimit
= dst_metric_raw(dst
, RTAX_HOPLIMIT
);
132 struct net_device
*dev
= dst
->dev
;
133 struct inet6_dev
*idev
;
136 idev
= __in6_dev_get(dev
);
138 hoplimit
= idev
->cnf
.hop_limit
;
140 hoplimit
= dev_net(dev
)->ipv6
.devconf_all
->hop_limit
;
145 EXPORT_SYMBOL(ip6_dst_hoplimit
);
148 int __ip6_local_out(struct net
*net
, struct sock
*sk
, struct sk_buff
*skb
)
152 len
= skb
->len
- sizeof(struct ipv6hdr
);
153 if (len
> IPV6_MAXPLEN
)
155 ipv6_hdr(skb
)->payload_len
= htons(len
);
156 IP6CB(skb
)->nhoff
= offsetof(struct ipv6hdr
, nexthdr
);
158 /* if egress device is enslaved to an L3 master device pass the
159 * skb to its handler for processing
161 skb
= l3mdev_ip6_out(sk
, skb
);
165 skb
->protocol
= htons(ETH_P_IPV6
);
167 return nf_hook(NFPROTO_IPV6
, NF_INET_LOCAL_OUT
,
168 net
, sk
, skb
, NULL
, skb_dst(skb
)->dev
,
171 EXPORT_SYMBOL_GPL(__ip6_local_out
);
173 int ip6_local_out(struct net
*net
, struct sock
*sk
, struct sk_buff
*skb
)
177 err
= __ip6_local_out(net
, sk
, skb
);
178 if (likely(err
== 1))
179 err
= dst_output(net
, sk
, skb
);
183 EXPORT_SYMBOL_GPL(ip6_local_out
);