2012-11-15 12:49:20 +04: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>
# include <net/ipv6.h>
# include <net/ip6_fib.h>
2013-08-31 09:44:28 +04:00
# include <net/addrconf.h>
2014-03-30 20:28:03 +04:00
# include <net/secure_seq.h>
2012-11-15 12:49:20 +04:00
void ipv6_select_ident ( struct frag_hdr * fhdr , struct rt6_info * rt )
{
static atomic_t ipv6_fragmentation_id ;
2014-03-30 20:28:03 +04:00
struct in6_addr addr ;
2012-11-15 12:49:20 +04:00
int old , new ;
# if IS_ENABLED(CONFIG_IPV6)
2014-03-30 20:28:03 +04:00
struct inet_peer * peer ;
struct net * net ;
2012-11-15 12:49:20 +04:00
2014-03-30 20:28:03 +04:00
net = dev_net ( rt - > dst . dev ) ;
peer = inet_getpeer_v6 ( net - > ipv6 . peers , & rt - > rt6i_dst . addr , 1 ) ;
if ( peer ) {
fhdr - > identification = htonl ( inet_getid ( peer , 0 ) ) ;
inet_putpeer ( peer ) ;
return ;
2012-11-15 12:49:20 +04:00
}
# endif
do {
old = atomic_read ( & ipv6_fragmentation_id ) ;
new = old + 1 ;
if ( ! new )
new = 1 ;
} while ( atomic_cmpxchg ( & ipv6_fragmentation_id , old , new ) ! = old ) ;
2014-03-30 20:28:03 +04:00
addr = rt - > rt6i_dst . addr ;
addr . s6_addr32 [ 0 ] ^ = ( __force __be32 ) new ;
fhdr - > identification = htonl ( secure_ipv6_id ( addr . s6_addr32 ) ) ;
2012-11-15 12:49:20 +04:00
}
EXPORT_SYMBOL ( ipv6_select_ident ) ;
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-29 00:34:26 +04:00
unsigned int packet_len = skb_tail_pointer ( skb ) -
skb_network_header ( skb ) ;
2012-11-15 12:49:20 +04: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 ;
default :
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 09:44:28 +04: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 09:44:29 +04: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 ) ;
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 ) ;