2005-04-16 15:20:36 -07:00
/*
* xfrm6_input . c : based on net / ipv4 / xfrm4_input . c
*
* Authors :
* Mitsuru KANDA @ USAGI
* Kazunori MIYAZAWA @ USAGI
* Kunihiro Ishiguro < kunihiro @ ipinfusion . com >
* YOSHIFUJI Hideaki @ USAGI
* IPv6 support
*/
# include <linux/module.h>
# include <linux/string.h>
2006-01-06 23:03:34 -08:00
# include <linux/netfilter.h>
# include <linux/netfilter_ipv6.h>
2005-04-16 15:20:36 -07:00
# include <net/ipv6.h>
# include <net/xfrm.h>
2006-09-27 18:48:18 -07:00
int xfrm6_rcv_spi ( struct sk_buff * skb , __be32 spi )
2005-04-16 15:20:36 -07:00
{
int err ;
2006-09-27 18:47:59 -07:00
__be32 seq ;
2006-04-01 00:54:16 -08:00
struct xfrm_state * xfrm_vec [ XFRM_MAX_DEPTH ] ;
2005-04-16 15:20:36 -07:00
struct xfrm_state * x ;
int xfrm_nr = 0 ;
int decaps = 0 ;
int nexthdr ;
unsigned int nhoff ;
2006-01-06 23:02:34 -08:00
nhoff = IP6CB ( skb ) - > nhoff ;
2007-04-10 20:50:43 -07:00
nexthdr = skb_network_header ( skb ) [ nhoff ] ;
2005-04-16 15:20:36 -07:00
seq = 0 ;
if ( ! spi & & ( err = xfrm_parse_spi ( skb , nexthdr , & spi , & seq ) ) ! = 0 )
goto drop ;
2007-02-09 23:24:49 +09:00
2005-04-16 15:20:36 -07:00
do {
2007-04-25 17:54:47 -07:00
struct ipv6hdr * iph = ipv6_hdr ( skb ) ;
2005-04-16 15:20:36 -07:00
if ( xfrm_nr = = XFRM_MAX_DEPTH )
goto drop ;
2007-02-13 12:55:55 -08:00
x = xfrm_state_lookup ( ( xfrm_address_t * ) & iph - > daddr , spi ,
nexthdr ! = IPPROTO_IPIP ? nexthdr : IPPROTO_IPV6 , AF_INET6 ) ;
2005-04-16 15:20:36 -07:00
if ( x = = NULL )
goto drop ;
spin_lock ( & x - > lock ) ;
if ( unlikely ( x - > km . state ! = XFRM_STATE_VALID ) )
goto drop_unlock ;
if ( x - > props . replay_window & & xfrm_replay_check ( x , seq ) )
goto drop_unlock ;
if ( xfrm_state_check_expire ( x ) )
goto drop_unlock ;
2006-04-01 00:52:46 -08:00
nexthdr = x - > type - > input ( x , skb ) ;
2005-04-16 15:20:36 -07:00
if ( nexthdr < = 0 )
goto drop_unlock ;
2007-04-10 20:50:43 -07:00
skb_network_header ( skb ) [ nhoff ] = nexthdr ;
2005-04-16 15:20:36 -07:00
if ( x - > props . replay_window )
xfrm_replay_advance ( x , seq ) ;
x - > curlft . bytes + = skb - > len ;
x - > curlft . packets + + ;
spin_unlock ( & x - > lock ) ;
2006-04-01 00:54:16 -08:00
xfrm_vec [ xfrm_nr + + ] = x ;
2005-04-16 15:20:36 -07:00
2006-05-27 23:05:54 -07:00
if ( x - > mode - > input ( x , skb ) )
goto drop ;
2006-09-22 15:05:15 -07:00
if ( x - > props . mode = = XFRM_MODE_TUNNEL ) { /* XXX */
2005-04-16 15:20:36 -07:00
decaps = 1 ;
break ;
}
if ( ( err = xfrm_parse_spi ( skb , nexthdr , & spi , & seq ) ) < 0 )
goto drop ;
} while ( ! err ) ;
/* Allocate new secpath or COW existing one. */
if ( ! skb - > sp | | atomic_read ( & skb - > sp - > refcnt ) ! = 1 ) {
struct sec_path * sp ;
sp = secpath_dup ( skb - > sp ) ;
if ( ! sp )
goto drop ;
if ( skb - > sp )
secpath_put ( skb - > sp ) ;
skb - > sp = sp ;
}
if ( xfrm_nr + skb - > sp - > len > XFRM_MAX_DEPTH )
goto drop ;
2006-04-01 00:54:16 -08:00
memcpy ( skb - > sp - > xvec + skb - > sp - > len , xfrm_vec ,
xfrm_nr * sizeof ( xfrm_vec [ 0 ] ) ) ;
2005-04-16 15:20:36 -07:00
skb - > sp - > len + = xfrm_nr ;
skb - > ip_summed = CHECKSUM_NONE ;
2006-01-06 23:03:34 -08:00
nf_reset ( skb ) ;
2005-04-16 15:20:36 -07:00
if ( decaps ) {
if ( ! ( skb - > dev - > flags & IFF_LOOPBACK ) ) {
dst_release ( skb - > dst ) ;
skb - > dst = NULL ;
}
netif_rx ( skb ) ;
return - 1 ;
} else {
2006-01-06 23:03:34 -08:00
# ifdef CONFIG_NETFILTER
2007-04-25 17:54:47 -07:00
ipv6_hdr ( skb ) - > payload_len = htons ( skb - > len ) ;
2007-04-10 20:50:43 -07:00
__skb_push ( skb , skb - > data - skb_network_header ( skb ) ) ;
2006-01-06 23:03:34 -08:00
NF_HOOK ( PF_INET6 , NF_IP6_PRE_ROUTING , skb , skb - > dev , NULL ,
2007-02-09 23:24:49 +09:00
ip6_rcv_finish ) ;
2006-01-06 23:03:34 -08:00
return - 1 ;
# else
2005-04-16 15:20:36 -07:00
return 1 ;
2006-01-06 23:03:34 -08:00
# endif
2005-04-16 15:20:36 -07:00
}
drop_unlock :
spin_unlock ( & x - > lock ) ;
xfrm_state_put ( x ) ;
drop :
while ( - - xfrm_nr > = 0 )
2006-04-01 00:54:16 -08:00
xfrm_state_put ( xfrm_vec [ xfrm_nr ] ) ;
2005-04-16 15:20:36 -07:00
kfree_skb ( skb ) ;
return - 1 ;
}
EXPORT_SYMBOL ( xfrm6_rcv_spi ) ;
2006-01-06 23:02:34 -08:00
int xfrm6_rcv ( struct sk_buff * * pskb )
2005-04-16 15:20:36 -07:00
{
2006-03-28 01:12:13 -08:00
return xfrm6_rcv_spi ( * pskb , 0 ) ;
2005-04-16 15:20:36 -07:00
}
2006-08-23 18:08:21 -07:00
2007-02-22 22:05:40 +09:00
EXPORT_SYMBOL ( xfrm6_rcv ) ;
2006-08-23 18:08:21 -07:00
int xfrm6_input_addr ( struct sk_buff * skb , xfrm_address_t * daddr ,
xfrm_address_t * saddr , u8 proto )
{
2007-02-09 23:24:49 +09:00
struct xfrm_state * x = NULL ;
int wildcard = 0 ;
2006-08-23 18:08:21 -07:00
xfrm_address_t * xany ;
struct xfrm_state * xfrm_vec_one = NULL ;
2007-02-09 23:24:49 +09:00
int nh = 0 ;
2006-08-23 18:08:21 -07:00
int i = 0 ;
2007-04-24 20:44:50 +09:00
xany = ( xfrm_address_t * ) & in6addr_any ;
2006-08-23 18:08:21 -07:00
for ( i = 0 ; i < 3 ; i + + ) {
xfrm_address_t * dst , * src ;
switch ( i ) {
case 0 :
dst = daddr ;
src = saddr ;
break ;
case 1 :
/* lookup state with wild-card source address */
wildcard = 1 ;
dst = daddr ;
src = xany ;
break ;
case 2 :
default :
2007-02-09 23:24:49 +09:00
/* lookup state with wild-card addresses */
2006-08-23 18:08:21 -07:00
wildcard = 1 ; /* XXX */
dst = xany ;
src = xany ;
break ;
2007-02-09 23:24:49 +09:00
}
2006-08-23 18:08:21 -07:00
x = xfrm_state_lookup_byaddr ( dst , src , proto , AF_INET6 ) ;
if ( ! x )
continue ;
spin_lock ( & x - > lock ) ;
if ( wildcard ) {
if ( ( x - > props . flags & XFRM_STATE_WILDRECV ) = = 0 ) {
spin_unlock ( & x - > lock ) ;
xfrm_state_put ( x ) ;
x = NULL ;
continue ;
}
}
if ( unlikely ( x - > km . state ! = XFRM_STATE_VALID ) ) {
spin_unlock ( & x - > lock ) ;
xfrm_state_put ( x ) ;
2007-02-09 23:24:49 +09:00
x = NULL ;
continue ;
2006-08-23 18:08:21 -07:00
}
if ( xfrm_state_check_expire ( x ) ) {
spin_unlock ( & x - > lock ) ;
xfrm_state_put ( x ) ;
x = NULL ;
continue ;
}
nh = x - > type - > input ( x , skb ) ;
if ( nh < = 0 ) {
spin_unlock ( & x - > lock ) ;
xfrm_state_put ( x ) ;
x = NULL ;
continue ;
}
x - > curlft . bytes + = skb - > len ;
x - > curlft . packets + + ;
spin_unlock ( & x - > lock ) ;
xfrm_vec_one = x ;
break ;
}
if ( ! xfrm_vec_one )
goto drop ;
/* Allocate new secpath or COW existing one. */
if ( ! skb - > sp | | atomic_read ( & skb - > sp - > refcnt ) ! = 1 ) {
struct sec_path * sp ;
sp = secpath_dup ( skb - > sp ) ;
if ( ! sp )
goto drop ;
if ( skb - > sp )
secpath_put ( skb - > sp ) ;
skb - > sp = sp ;
}
if ( 1 + skb - > sp - > len > XFRM_MAX_DEPTH )
goto drop ;
skb - > sp - > xvec [ skb - > sp - > len ] = xfrm_vec_one ;
skb - > sp - > len + + ;
return 1 ;
drop :
if ( xfrm_vec_one )
xfrm_state_put ( xfrm_vec_one ) ;
return - 1 ;
}
2007-02-22 22:05:40 +09:00
EXPORT_SYMBOL ( xfrm6_input_addr ) ;