2006-05-27 23:05:54 -07:00
/*
* xfrm4_mode_transport . c - Transport 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/ip.h>
# include <net/xfrm.h>
/* Add encapsulation header.
*
* The IP header will be moved forward to make space for the encapsulation
* header .
*/
2006-08-31 17:42:59 -07:00
static int xfrm4_transport_output ( struct xfrm_state * x , struct sk_buff * skb )
2006-05-27 23:05:54 -07:00
{
2007-04-20 22:47:35 -07:00
struct iphdr * iph = ip_hdr ( skb ) ;
int ihl = iph - > ihl * 4 ;
2006-05-27 23:05:54 -07:00
2007-10-10 15:44:06 -07:00
skb_set_network_header ( skb , - x - > props . header_len ) ;
2007-10-10 15:44:44 -07:00
skb - > mac_header = skb - > network_header +
offsetof ( struct iphdr , protocol ) ;
skb - > transport_header = skb - > network_header + ihl ;
2007-10-10 15:44:06 -07:00
__skb_pull ( skb , ihl ) ;
2007-04-10 20:50:43 -07:00
memmove ( skb_network_header ( skb ) , iph , ihl ) ;
2006-05-27 23:05:54 -07:00
return 0 ;
}
2006-05-27 23:06:13 -07:00
/* Remove encapsulation header.
*
* The IP header will be moved over the top of the encapsulation header .
*
* On entry , skb - > h shall point to where the IP header should be and skb - > nh
* shall be set to where the IP header currently is . skb - > data shall point
* to the start of the payload .
*/
2006-05-27 23:05:54 -07:00
static int xfrm4_transport_input ( struct xfrm_state * x , struct sk_buff * skb )
{
2007-04-25 18:04:18 -07:00
int ihl = skb - > data - skb_transport_header ( skb ) ;
2006-05-27 23:06:13 -07:00
2007-04-10 21:21:55 -07:00
if ( skb - > transport_header ! = skb - > network_header ) {
2007-04-25 18:04:18 -07:00
memmove ( skb_transport_header ( skb ) ,
skb_network_header ( skb ) , ihl ) ;
2007-04-10 21:21:55 -07:00
skb - > network_header = skb - > transport_header ;
2007-03-10 19:59:16 -03:00
}
2007-04-20 22:47:35 -07:00
ip_hdr ( skb ) - > tot_len = htons ( skb - > len + ihl ) ;
2007-03-13 13:06:52 -03:00
skb_reset_transport_header ( skb ) ;
2006-05-27 23:05:54 -07:00
return 0 ;
}
static struct xfrm_mode xfrm4_transport_mode = {
. input = xfrm4_transport_input ,
. output = xfrm4_transport_output ,
. owner = THIS_MODULE ,
. encap = XFRM_MODE_TRANSPORT ,
} ;
static int __init xfrm4_transport_init ( void )
{
return xfrm_register_mode ( & xfrm4_transport_mode , AF_INET ) ;
}
static void __exit xfrm4_transport_exit ( void )
{
int err ;
err = xfrm_unregister_mode ( & xfrm4_transport_mode , AF_INET ) ;
BUG_ON ( err ) ;
}
module_init ( xfrm4_transport_init ) ;
module_exit ( xfrm4_transport_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_XFRM_MODE ( AF_INET , XFRM_MODE_TRANSPORT ) ;