diff options
Diffstat (limited to 'net/ipv6/output_core.c')
| -rw-r--r-- | net/ipv6/output_core.c | 40 | 
1 files changed, 35 insertions, 5 deletions
| diff --git a/net/ipv6/output_core.c b/net/ipv6/output_core.c index 97f41a3e68d9..74581f706c4d 100644 --- a/net/ipv6/output_core.c +++ b/net/ipv6/output_core.c @@ -9,6 +9,25 @@  #include <net/addrconf.h>  #include <net/secure_seq.h> +static u32 __ipv6_select_ident(u32 hashrnd, struct in6_addr *dst, +			       struct in6_addr *src) +{ +	u32 hash, id; + +	hash = __ipv6_addr_jhash(dst, hashrnd); +	hash = __ipv6_addr_jhash(src, hash); + +	/* Treat id of 0 as unset and if we get 0 back from ip_idents_reserve, +	 * set the hight order instead thus minimizing possible future +	 * collisions. +	 */ +	id = ip_idents_reserve(hash, 1); +	if (unlikely(!id)) +		id = 1 << 31; + +	return id; +} +  /* This function exists only for tap drivers that must support broken   * clients requesting UFO without specifying an IPv6 fragment ID.   * @@ -22,7 +41,7 @@ void ipv6_proxy_select_ident(struct sk_buff *skb)  	static u32 ip6_proxy_idents_hashrnd __read_mostly;  	struct in6_addr buf[2];  	struct in6_addr *addrs; -	u32 hash, id; +	u32 id;  	addrs = skb_header_pointer(skb,  				   skb_network_offset(skb) + @@ -34,14 +53,25 @@ void ipv6_proxy_select_ident(struct sk_buff *skb)  	net_get_random_once(&ip6_proxy_idents_hashrnd,  			    sizeof(ip6_proxy_idents_hashrnd)); -	hash = __ipv6_addr_jhash(&addrs[1], ip6_proxy_idents_hashrnd); -	hash = __ipv6_addr_jhash(&addrs[0], hash); - -	id = ip_idents_reserve(hash, 1); +	id = __ipv6_select_ident(ip6_proxy_idents_hashrnd, +				 &addrs[1], &addrs[0]);  	skb_shinfo(skb)->ip6_frag_id = htonl(id);  }  EXPORT_SYMBOL_GPL(ipv6_proxy_select_ident); +void ipv6_select_ident(struct frag_hdr *fhdr, struct rt6_info *rt) +{ +	static u32 ip6_idents_hashrnd __read_mostly; +	u32 id; + +	net_get_random_once(&ip6_idents_hashrnd, sizeof(ip6_idents_hashrnd)); + +	id = __ipv6_select_ident(ip6_idents_hashrnd, &rt->rt6i_dst.addr, +				 &rt->rt6i_src.addr); +	fhdr->identification = htonl(id); +} +EXPORT_SYMBOL(ipv6_select_ident); +  int ip6_find_1stfragopt(struct sk_buff *skb, u8 **nexthdr)  {  	u16 offset = sizeof(struct ipv6hdr); |