2005-04-16 15:20:36 -07:00
/*
* xfrm6_output . c - Common IPsec encapsulation code for IPv6 .
* Copyright ( C ) 2002 USAGI / WIDE Project
* Copyright ( c ) 2004 Herbert Xu < herbert @ gondor . apana . org . au >
2007-02-09 23:24:49 +09:00
*
2005-04-16 15:20:36 -07:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
2007-10-08 17:16:30 -07:00
# include <linux/if_ether.h>
2007-11-13 21:40:52 -08:00
# include <linux/kernel.h>
# include <linux/module.h>
2005-04-16 15:20:36 -07:00
# include <linux/skbuff.h>
# include <linux/icmpv6.h>
2006-01-06 23:01:48 -08:00
# include <linux/netfilter_ipv6.h>
2007-11-13 21:40:52 -08:00
# include <net/dst.h>
2005-04-16 15:20:36 -07:00
# include <net/ipv6.h>
# include <net/xfrm.h>
2006-08-23 17:57:28 -07:00
int xfrm6_find_1stfragopt ( struct xfrm_state * x , struct sk_buff * skb ,
u8 * * prevhdr )
{
return ip6_find_1stfragopt ( skb , prevhdr ) ;
}
2007-02-22 22:05:40 +09:00
EXPORT_SYMBOL ( xfrm6_find_1stfragopt ) ;
2005-04-16 15:20:36 -07:00
static int xfrm6_tunnel_check_size ( struct sk_buff * skb )
{
int mtu , ret = 0 ;
2009-06-02 05:19:30 +00:00
struct dst_entry * dst = skb_dst ( skb ) ;
2005-04-16 15:20:36 -07:00
mtu = dst_mtu ( dst ) ;
if ( mtu < IPV6_MIN_MTU )
mtu = IPV6_MIN_MTU ;
2008-02-12 18:07:27 -08:00
if ( ! skb - > local_df & & skb - > len > mtu ) {
2005-05-23 13:11:07 -07:00
skb - > dev = dst - > dev ;
2010-02-18 08:25:24 +00:00
icmpv6_send ( skb , ICMPV6_PKT_TOOBIG , 0 , mtu ) ;
2005-04-16 15:20:36 -07:00
ret = - EMSGSIZE ;
}
return ret ;
}
2007-11-13 21:40:52 -08:00
int xfrm6_extract_output ( struct xfrm_state * x , struct sk_buff * skb )
{
int err ;
err = xfrm6_tunnel_check_size ( skb ) ;
if ( err )
return err ;
2007-11-19 18:47:58 -08:00
XFRM_MODE_SKB_CB ( skb ) - > protocol = ipv6_hdr ( skb ) - > nexthdr ;
2007-11-13 21:40:52 -08:00
return xfrm6_extract_header ( skb ) ;
}
int xfrm6_prepare_output ( struct xfrm_state * x , struct sk_buff * skb )
{
int err ;
2008-03-24 14:51:51 -07:00
err = xfrm_inner_extract_output ( x , skb ) ;
2007-11-13 21:40:52 -08:00
if ( err )
return err ;
memset ( IP6CB ( skb ) , 0 , sizeof ( * IP6CB ( skb ) ) ) ;
2007-11-13 21:43:11 -08:00
# ifdef CONFIG_NETFILTER
IP6CB ( skb ) - > flags | = IP6SKB_XFRM_TRANSFORMED ;
# endif
2007-11-13 21:40:52 -08:00
skb - > protocol = htons ( ETH_P_IPV6 ) ;
2009-04-06 03:58:50 +00:00
skb - > local_df = 1 ;
2007-11-13 21:40:52 -08:00
return x - > outer_mode - > output2 ( x , skb ) ;
}
EXPORT_SYMBOL ( xfrm6_prepare_output ) ;
2006-06-22 03:08:03 -07:00
static int xfrm6_output_finish ( struct sk_buff * skb )
{
2007-11-13 21:43:11 -08:00
# ifdef CONFIG_NETFILTER
IP6CB ( skb ) - > flags | = IP6SKB_XFRM_TRANSFORMED ;
# endif
2006-06-22 03:08:03 -07:00
2006-07-30 20:19:11 -07:00
skb - > protocol = htons ( ETH_P_IPV6 ) ;
2007-11-13 21:43:11 -08:00
return xfrm_output ( skb ) ;
2006-06-22 03:08:03 -07:00
}
2006-01-06 23:01:48 -08:00
int xfrm6_output ( struct sk_buff * skb )
{
2009-06-02 05:19:30 +00:00
return NF_HOOK ( PF_INET6 , NF_INET_POST_ROUTING , skb , NULL , skb_dst ( skb ) - > dev ,
2006-01-06 23:01:48 -08:00
xfrm6_output_finish ) ;
}