2012-11-15 08:49:20 +00:00
/*
* IPv6 library code , needed by static components when full IPv6 support is
* not configured or static . These functions are needed by GSO / GRO implementation .
*/
# include <linux/export.h>
2014-10-30 18:27:17 +00:00
# include <net/ip.h>
2012-11-15 08:49:20 +00:00
# include <net/ipv6.h>
# include <net/ip6_fib.h>
2013-08-31 13:44:28 +08:00
# include <net/addrconf.h>
2014-03-30 18:28:03 +02:00
# include <net/secure_seq.h>
2012-11-15 08:49:20 +00:00
2014-10-30 18:27:17 +00:00
/* This function exists only for tap drivers that must support broken
* clients requesting UFO without specifying an IPv6 fragment ID .
*
* This is similar to ipv6_select_ident ( ) but we use an independent hash
* seed to limit information leakage .
*
* The network header must be set before calling this .
*/
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 ;
addrs = skb_header_pointer ( skb ,
skb_network_offset ( skb ) +
offsetof ( struct ipv6hdr , saddr ) ,
sizeof ( buf ) , buf ) ;
if ( ! addrs )
return ;
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 ) ;
skb_shinfo ( skb ) - > ip6_frag_id = htonl ( id ) ;
}
EXPORT_SYMBOL_GPL ( ipv6_proxy_select_ident ) ;
2012-11-15 08:49:20 +00:00
int ip6_find_1stfragopt ( struct sk_buff * skb , u8 * * nexthdr )
{
u16 offset = sizeof ( struct ipv6hdr ) ;
struct ipv6_opt_hdr * exthdr =
( struct ipv6_opt_hdr * ) ( ipv6_hdr ( skb ) + 1 ) ;
2013-05-28 20:34:26 +00:00
unsigned int packet_len = skb_tail_pointer ( skb ) -
skb_network_header ( skb ) ;
2012-11-15 08:49:20 +00:00
int found_rhdr = 0 ;
* nexthdr = & ipv6_hdr ( skb ) - > nexthdr ;
while ( offset + 1 < = packet_len ) {
switch ( * * nexthdr ) {
case NEXTHDR_HOP :
break ;
case NEXTHDR_ROUTING :
found_rhdr = 1 ;
break ;
case NEXTHDR_DEST :
# if IS_ENABLED(CONFIG_IPV6_MIP6)
if ( ipv6_find_tlv ( skb , offset , IPV6_TLV_HAO ) > = 0 )
break ;
# endif
if ( found_rhdr )
return offset ;
break ;
2014-08-24 21:53:10 +01:00
default :
2012-11-15 08:49:20 +00:00
return offset ;
}
offset + = ipv6_optlen ( exthdr ) ;
* nexthdr = & exthdr - > nexthdr ;
exthdr = ( struct ipv6_opt_hdr * ) ( skb_network_header ( skb ) +
offset ) ;
}
return offset ;
}
EXPORT_SYMBOL ( ip6_find_1stfragopt ) ;
2013-08-31 13:44:28 +08:00
# if IS_ENABLED(CONFIG_IPV6)
int ip6_dst_hoplimit ( struct dst_entry * dst )
{
int hoplimit = dst_metric_raw ( dst , RTAX_HOPLIMIT ) ;
if ( hoplimit = = 0 ) {
struct net_device * dev = dst - > dev ;
struct inet6_dev * idev ;
rcu_read_lock ( ) ;
idev = __in6_dev_get ( dev ) ;
if ( idev )
hoplimit = idev - > cnf . hop_limit ;
else
hoplimit = dev_net ( dev ) - > ipv6 . devconf_all - > hop_limit ;
rcu_read_unlock ( ) ;
}
return hoplimit ;
}
EXPORT_SYMBOL ( ip6_dst_hoplimit ) ;
# endif
2013-08-31 13:44:29 +08:00
int __ip6_local_out ( struct sk_buff * skb )
{
int len ;
len = skb - > len - sizeof ( struct ipv6hdr ) ;
if ( len > IPV6_MAXPLEN )
len = 0 ;
ipv6_hdr ( skb ) - > payload_len = htons ( len ) ;
2014-06-09 12:37:25 +08:00
IP6CB ( skb ) - > nhoff = offsetof ( struct ipv6hdr , nexthdr ) ;
2013-08-31 13:44:29 +08:00
return nf_hook ( NFPROTO_IPV6 , NF_INET_LOCAL_OUT , skb , NULL ,
skb_dst ( skb ) - > dev , dst_output ) ;
}
EXPORT_SYMBOL_GPL ( __ip6_local_out ) ;
int ip6_local_out ( struct sk_buff * skb )
{
int err ;
err = __ip6_local_out ( skb ) ;
if ( likely ( err = = 1 ) )
err = dst_output ( skb ) ;
return err ;
}
EXPORT_SYMBOL_GPL ( ip6_local_out ) ;