2005-04-17 02:20:36 +04:00
/*
* xfrm4_output . c - Common IPsec encapsulation code for IPv4 .
* Copyright ( c ) 2004 Herbert Xu < herbert @ gondor . apana . org . au >
2007-02-09 17:24:47 +03:00
*
2005-04-17 02:20:36 +04: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 .
*/
2006-01-07 10:01:48 +03:00
# include <linux/compiler.h>
2006-06-22 14:08:03 +04:00
# include <linux/if_ether.h>
# include <linux/kernel.h>
2005-04-17 02:20:36 +04:00
# include <linux/skbuff.h>
2006-01-07 10:01:48 +03:00
# include <linux/netfilter_ipv4.h>
2005-04-17 02:20:36 +04:00
# include <net/ip.h>
# include <net/xfrm.h>
# include <net/icmp.h>
static int xfrm4_tunnel_check_size ( struct sk_buff * skb )
{
int mtu , ret = 0 ;
struct dst_entry * dst ;
if ( IPCB ( skb ) - > flags & IPSKB_XFRM_TUNNEL_SIZE )
goto out ;
IPCB ( skb ) - > flags | = IPSKB_XFRM_TUNNEL_SIZE ;
2007-02-09 17:24:47 +03:00
2007-04-21 09:47:35 +04:00
if ( ! ( ip_hdr ( skb ) - > frag_off & htons ( IP_DF ) ) | | skb - > local_df )
2005-04-17 02:20:36 +04:00
goto out ;
dst = skb - > dst ;
mtu = dst_mtu ( dst ) ;
if ( skb - > len > mtu ) {
icmp_send ( skb , ICMP_DEST_UNREACH , ICMP_FRAG_NEEDED , htonl ( mtu ) ) ;
ret = - EMSGSIZE ;
}
out :
return ret ;
}
2007-10-09 04:16:30 +04:00
static inline int xfrm4_output_one ( struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
struct dst_entry * dst = skb - > dst ;
struct xfrm_state * x = dst - > xfrm ;
2007-10-11 02:45:52 +04:00
struct iphdr * iph ;
2005-04-17 02:20:36 +04:00
int err ;
2007-02-09 17:24:47 +03:00
2006-09-23 02:05:15 +04:00
if ( x - > props . mode = = XFRM_MODE_TUNNEL ) {
2005-04-17 02:20:36 +04:00
err = xfrm4_tunnel_check_size ( skb ) ;
if ( err )
goto error_nolock ;
}
2007-10-09 04:16:30 +04:00
err = xfrm_output ( skb ) ;
if ( err )
goto error_nolock ;
2006-01-07 10:01:48 +03:00
2007-10-11 02:45:52 +04:00
iph = ip_hdr ( skb ) ;
iph - > tot_len = htons ( skb - > len ) ;
ip_send_check ( iph ) ;
2006-01-07 10:04:54 +03:00
IPCB ( skb ) - > flags | = IPSKB_XFRM_TRANSFORMED ;
2006-01-07 10:01:48 +03:00
err = 0 ;
2005-04-17 02:20:36 +04:00
out_exit :
return err ;
error_nolock :
kfree_skb ( skb ) ;
goto out_exit ;
}
2006-01-07 10:01:48 +03:00
2006-06-22 14:08:03 +04:00
static int xfrm4_output_finish2 ( struct sk_buff * skb )
2006-01-07 10:01:48 +03:00
{
int err ;
while ( likely ( ( err = xfrm4_output_one ( skb ) ) = = 0 ) ) {
nf_reset ( skb ) ;
2007-10-15 11:53:15 +04:00
err = nf_hook ( PF_INET , NF_IP_LOCAL_OUT , skb , NULL ,
2006-01-07 10:01:48 +03:00
skb - > dst - > dev , dst_output ) ;
if ( unlikely ( err ! = 1 ) )
break ;
if ( ! skb - > dst - > xfrm )
return dst_output ( skb ) ;
2007-10-15 11:53:15 +04:00
err = nf_hook ( PF_INET , NF_IP_POST_ROUTING , skb , NULL ,
2006-06-22 14:08:03 +04:00
skb - > dst - > dev , xfrm4_output_finish2 ) ;
2006-01-07 10:01:48 +03:00
if ( unlikely ( err ! = 1 ) )
break ;
}
return err ;
}
2006-06-22 14:08:03 +04:00
static int xfrm4_output_finish ( struct sk_buff * skb )
{
struct sk_buff * segs ;
# ifdef CONFIG_NETFILTER
if ( ! skb - > dst - > xfrm ) {
IPCB ( skb ) - > flags | = IPSKB_REROUTED ;
return dst_output ( skb ) ;
}
# endif
2006-07-09 00:34:32 +04:00
if ( ! skb_is_gso ( skb ) )
2006-06-22 14:08:03 +04:00
return xfrm4_output_finish2 ( skb ) ;
skb - > protocol = htons ( ETH_P_IP ) ;
segs = skb_gso_segment ( skb , 0 ) ;
kfree_skb ( skb ) ;
if ( unlikely ( IS_ERR ( segs ) ) )
return PTR_ERR ( segs ) ;
do {
struct sk_buff * nskb = segs - > next ;
int err ;
segs - > next = NULL ;
err = xfrm4_output_finish2 ( segs ) ;
if ( unlikely ( err ) ) {
while ( ( segs = nskb ) ) {
nskb = segs - > next ;
segs - > next = NULL ;
kfree_skb ( segs ) ;
}
return err ;
}
segs = nskb ;
} while ( segs ) ;
return 0 ;
}
2006-01-07 10:01:48 +03:00
int xfrm4_output ( struct sk_buff * skb )
{
2006-02-16 02:10:22 +03:00
return NF_HOOK_COND ( PF_INET , NF_IP_POST_ROUTING , skb , NULL , skb - > dst - > dev ,
xfrm4_output_finish ,
! ( IPCB ( skb ) - > flags & IPSKB_REROUTED ) ) ;
2006-01-07 10:01:48 +03:00
}