2 * IPv6 library code, needed by static components when full IPv6 support is
3 * not configured or static. These functions are needed by GSO/GRO implementation.
5 #include <linux/export.h>
8 #include <net/ip6_fib.h>
9 #include <net/addrconf.h>
10 #include <net/secure_seq.h>
12 static u32
__ipv6_select_ident(struct net
*net
, u32 hashrnd
,
13 struct in6_addr
*dst
, struct in6_addr
*src
)
17 hash
= __ipv6_addr_jhash(dst
, hashrnd
);
18 hash
= __ipv6_addr_jhash(src
, hash
);
19 hash
^= net_hash_mix(net
);
21 /* Treat id of 0 as unset and if we get 0 back from ip_idents_reserve,
22 * set the hight order instead thus minimizing possible future
25 id
= ip_idents_reserve(hash
, 1);
32 /* This function exists only for tap drivers that must support broken
33 * clients requesting UFO without specifying an IPv6 fragment ID.
35 * This is similar to ipv6_select_ident() but we use an independent hash
36 * seed to limit information leakage.
38 * The network header must be set before calling this.
40 void ipv6_proxy_select_ident(struct net
*net
, struct sk_buff
*skb
)
42 static u32 ip6_proxy_idents_hashrnd __read_mostly
;
43 struct in6_addr buf
[2];
44 struct in6_addr
*addrs
;
47 addrs
= skb_header_pointer(skb
,
48 skb_network_offset(skb
) +
49 offsetof(struct ipv6hdr
, saddr
),
54 net_get_random_once(&ip6_proxy_idents_hashrnd
,
55 sizeof(ip6_proxy_idents_hashrnd
));
57 id
= __ipv6_select_ident(net
, ip6_proxy_idents_hashrnd
,
58 &addrs
[1], &addrs
[0]);
59 skb_shinfo(skb
)->ip6_frag_id
= htonl(id
);
61 EXPORT_SYMBOL_GPL(ipv6_proxy_select_ident
);
63 void ipv6_select_ident(struct net
*net
, struct frag_hdr
*fhdr
,
66 static u32 ip6_idents_hashrnd __read_mostly
;
69 net_get_random_once(&ip6_idents_hashrnd
, sizeof(ip6_idents_hashrnd
));
71 id
= __ipv6_select_ident(net
, ip6_idents_hashrnd
, &rt
->rt6i_dst
.addr
,
73 fhdr
->identification
= htonl(id
);
75 EXPORT_SYMBOL(ipv6_select_ident
);
77 int ip6_find_1stfragopt(struct sk_buff
*skb
, u8
**nexthdr
)
79 u16 offset
= sizeof(struct ipv6hdr
);
80 struct ipv6_opt_hdr
*exthdr
=
81 (struct ipv6_opt_hdr
*)(ipv6_hdr(skb
) + 1);
82 unsigned int packet_len
= skb_tail_pointer(skb
) -
83 skb_network_header(skb
);
85 *nexthdr
= &ipv6_hdr(skb
)->nexthdr
;
87 while (offset
+ 1 <= packet_len
) {
97 #if IS_ENABLED(CONFIG_IPV6_MIP6)
98 if (ipv6_find_tlv(skb
, offset
, IPV6_TLV_HAO
) >= 0)
108 offset
+= ipv6_optlen(exthdr
);
109 *nexthdr
= &exthdr
->nexthdr
;
110 exthdr
= (struct ipv6_opt_hdr
*)(skb_network_header(skb
) +
116 EXPORT_SYMBOL(ip6_find_1stfragopt
);
118 #if IS_ENABLED(CONFIG_IPV6)
119 int ip6_dst_hoplimit(struct dst_entry
*dst
)
121 int hoplimit
= dst_metric_raw(dst
, RTAX_HOPLIMIT
);
123 struct net_device
*dev
= dst
->dev
;
124 struct inet6_dev
*idev
;
127 idev
= __in6_dev_get(dev
);
129 hoplimit
= idev
->cnf
.hop_limit
;
131 hoplimit
= dev_net(dev
)->ipv6
.devconf_all
->hop_limit
;
136 EXPORT_SYMBOL(ip6_dst_hoplimit
);
139 static int __ip6_local_out_sk(struct sock
*sk
, struct sk_buff
*skb
)
143 len
= skb
->len
- sizeof(struct ipv6hdr
);
144 if (len
> IPV6_MAXPLEN
)
146 ipv6_hdr(skb
)->payload_len
= htons(len
);
147 IP6CB(skb
)->nhoff
= offsetof(struct ipv6hdr
, nexthdr
);
149 return nf_hook(NFPROTO_IPV6
, NF_INET_LOCAL_OUT
, sk
, skb
,
150 NULL
, skb_dst(skb
)->dev
, dst_output_sk
);
153 int __ip6_local_out(struct sk_buff
*skb
)
155 return __ip6_local_out_sk(skb
->sk
, skb
);
157 EXPORT_SYMBOL_GPL(__ip6_local_out
);
159 int ip6_local_out_sk(struct sock
*sk
, struct sk_buff
*skb
)
163 err
= __ip6_local_out_sk(sk
, skb
);
164 if (likely(err
== 1))
165 err
= dst_output_sk(sk
, skb
);
169 EXPORT_SYMBOL_GPL(ip6_local_out_sk
);
171 int ip6_local_out(struct sk_buff
*skb
)
173 return ip6_local_out_sk(skb
->sk
, skb
);
175 EXPORT_SYMBOL_GPL(ip6_local_out
);