2006-05-28 10:05:54 +04:00
/*
* xfrm4_mode_tunnel . c - Tunnel mode encapsulation for IPv4 .
*
* Copyright ( c ) 2004 - 2006 Herbert Xu < herbert @ gondor . apana . org . au >
*/
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/skbuff.h>
# include <linux/stringify.h>
# include <net/dst.h>
# include <net/inet_ecn.h>
# include <net/ip.h>
# include <net/xfrm.h>
static inline void ipip_ecn_decapsulate ( struct sk_buff * skb )
{
2007-04-21 09:47:35 +04:00
struct iphdr * outer_iph = ip_hdr ( skb ) ;
2006-05-28 10:05:54 +04:00
struct iphdr * inner_iph = skb - > h . ipiph ;
if ( INET_ECN_is_ce ( outer_iph - > tos ) )
IP_ECN_set_ce ( inner_iph ) ;
}
2007-02-07 01:27:02 +03:00
static inline void ipip6_ecn_decapsulate ( struct iphdr * iph , struct sk_buff * skb )
{
if ( INET_ECN_is_ce ( iph - > tos ) )
2007-04-26 04:54:47 +04:00
IP6_ECN_set_ce ( ipv6_hdr ( skb ) ) ;
2007-02-07 01:27:02 +03:00
}
2006-05-28 10:05:54 +04:00
/* Add encapsulation header.
*
* The top IP header will be constructed per RFC 2401. The following fields
* in it shall be filled in by x - > type - > output :
* tot_len
* check
*
* On exit , skb - > h will be set to the start of the payload to be processed
* by x - > type - > output and skb - > nh will be set to the top IP header .
*/
2006-09-01 04:42:59 +04:00
static int xfrm4_tunnel_output ( struct xfrm_state * x , struct sk_buff * skb )
2006-05-28 10:05:54 +04:00
{
struct dst_entry * dst = skb - > dst ;
2007-02-07 01:27:02 +03:00
struct xfrm_dst * xdst = ( struct xfrm_dst * ) dst ;
2006-05-28 10:05:54 +04:00
struct iphdr * iph , * top_iph ;
int flags ;
2007-04-21 09:47:35 +04:00
iph = ip_hdr ( skb ) ;
2006-05-28 10:05:54 +04:00
skb - > h . ipiph = iph ;
2007-04-11 07:46:21 +04:00
skb_push ( skb , x - > props . header_len ) ;
skb_reset_network_header ( skb ) ;
2007-04-21 09:47:35 +04:00
top_iph = ip_hdr ( skb ) ;
2006-05-28 10:05:54 +04:00
top_iph - > ihl = 5 ;
top_iph - > version = 4 ;
2007-02-07 01:27:02 +03:00
flags = x - > props . flags ;
2006-05-28 10:05:54 +04:00
/* DS disclosed */
2007-02-07 01:27:02 +03:00
if ( xdst - > route - > ops - > family = = AF_INET ) {
top_iph - > protocol = IPPROTO_IPIP ;
top_iph - > tos = INET_ECN_encapsulate ( iph - > tos , iph - > tos ) ;
top_iph - > frag_off = ( flags & XFRM_STATE_NOPMTUDISC ) ?
0 : ( iph - > frag_off & htons ( IP_DF ) ) ;
}
# if defined(CONFIG_IPV6) || defined (CONFIG_IPV6_MODULE)
else {
struct ipv6hdr * ipv6h = ( struct ipv6hdr * ) iph ;
top_iph - > protocol = IPPROTO_IPV6 ;
top_iph - > tos = INET_ECN_encapsulate ( iph - > tos , ipv6_get_dsfield ( ipv6h ) ) ;
top_iph - > frag_off = 0 ;
}
# endif
2006-05-28 10:05:54 +04:00
if ( flags & XFRM_STATE_NOECN )
IP_ECN_clear ( top_iph ) ;
if ( ! top_iph - > frag_off )
__ip_select_ident ( top_iph , dst - > child , 0 ) ;
top_iph - > ttl = dst_metric ( dst - > child , RTAX_HOPLIMIT ) ;
top_iph - > saddr = x - > props . saddr . a4 ;
top_iph - > daddr = x - > id . daddr . a4 ;
memset ( & ( IPCB ( skb ) - > opt ) , 0 , sizeof ( struct ip_options ) ) ;
return 0 ;
}
static int xfrm4_tunnel_input ( struct xfrm_state * x , struct sk_buff * skb )
{
2007-04-21 09:47:35 +04:00
struct iphdr * iph = ip_hdr ( skb ) ;
2007-03-10 18:40:27 +03:00
const unsigned char * old_mac ;
2006-05-28 10:05:54 +04:00
int err = - EINVAL ;
2007-03-09 07:44:43 +03:00
switch ( iph - > protocol ) {
2007-02-07 01:27:02 +03:00
case IPPROTO_IPIP :
2007-02-13 07:27:10 +03:00
break ;
2007-02-07 01:27:02 +03:00
# if defined(CONFIG_IPV6) || defined (CONFIG_IPV6_MODULE)
case IPPROTO_IPV6 :
break ;
# endif
default :
goto out ;
}
2006-05-28 10:05:54 +04:00
if ( ! pskb_may_pull ( skb , sizeof ( struct iphdr ) ) )
goto out ;
if ( skb_cloned ( skb ) & &
( err = pskb_expand_head ( skb , 0 , 0 , GFP_ATOMIC ) ) )
goto out ;
2007-04-21 09:47:35 +04:00
iph = ip_hdr ( skb ) ;
2007-02-07 01:27:02 +03:00
if ( iph - > protocol = = IPPROTO_IPIP ) {
if ( x - > props . flags & XFRM_STATE_DECAP_DSCP )
ipv4_copy_dscp ( iph , skb - > h . ipiph ) ;
if ( ! ( x - > props . flags & XFRM_STATE_NOECN ) )
ipip_ecn_decapsulate ( skb ) ;
}
# if defined(CONFIG_IPV6) || defined (CONFIG_IPV6_MODULE)
else {
if ( ! ( x - > props . flags & XFRM_STATE_NOECN ) )
ipip6_ecn_decapsulate ( iph , skb ) ;
skb - > protocol = htons ( ETH_P_IPV6 ) ;
}
# endif
2007-03-20 01:33:04 +03:00
old_mac = skb_mac_header ( skb ) ;
2007-03-10 18:40:27 +03:00
skb_set_mac_header ( skb , - skb - > mac_len ) ;
2007-03-20 01:33:04 +03:00
memmove ( skb_mac_header ( skb ) , old_mac , skb - > mac_len ) ;
2007-04-11 07:45:18 +04:00
skb_reset_network_header ( skb ) ;
2006-05-28 10:05:54 +04:00
err = 0 ;
out :
return err ;
}
static struct xfrm_mode xfrm4_tunnel_mode = {
. input = xfrm4_tunnel_input ,
. output = xfrm4_tunnel_output ,
. owner = THIS_MODULE ,
. encap = XFRM_MODE_TUNNEL ,
} ;
static int __init xfrm4_tunnel_init ( void )
{
return xfrm_register_mode ( & xfrm4_tunnel_mode , AF_INET ) ;
}
static void __exit xfrm4_tunnel_exit ( void )
{
int err ;
err = xfrm_unregister_mode ( & xfrm4_tunnel_mode , AF_INET ) ;
BUG_ON ( err ) ;
}
module_init ( xfrm4_tunnel_init ) ;
module_exit ( xfrm4_tunnel_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_XFRM_MODE ( AF_INET , XFRM_MODE_TUNNEL ) ;