2005-04-17 02:20:36 +04:00
/*
* Extension Header handling for IPv6
* Linux INET6 implementation
*
* Authors :
* Pedro Roque < roque @ di . fc . ul . pt >
* Andi Kleen < ak @ muc . de >
* Alexey Kuznetsov < kuznet @ ms2 . inr . ac . ru >
*
* 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 .
*/
/* Changes:
2007-02-09 17:24:49 +03:00
* yoshfuji : ensure not to overrun while parsing
2005-04-17 02:20:36 +04:00
* tlv options .
* Mitsuru KANDA @ USAGI and : Remove ipv6_parse_exthdrs ( ) .
* YOSHIFUJI Hideaki @ USAGI Register inbound extension header
* handlers as inet6_protocol { } .
*/
# include <linux/errno.h>
# include <linux/types.h>
# include <linux/socket.h>
# include <linux/sockios.h>
# include <linux/net.h>
# include <linux/netdevice.h>
# include <linux/in6.h>
# include <linux/icmpv6.h>
2007-11-14 08:34:06 +03:00
# include <net/dst.h>
2005-04-17 02:20:36 +04:00
# include <net/sock.h>
# include <net/snmp.h>
# include <net/ipv6.h>
# include <net/protocol.h>
# include <net/transp_v6.h>
# include <net/rawv6.h>
# include <net/ndisc.h>
# include <net/ip6_route.h>
# include <net/addrconf.h>
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:16:22 +04:00
# include <net/xfrm.h>
# endif
2005-04-17 02:20:36 +04:00
# include <asm/uaccess.h>
2006-08-24 06:18:35 +04:00
int ipv6_find_tlv ( struct sk_buff * skb , int offset , int type )
{
2007-04-11 07:50:43 +04:00
const unsigned char * nh = skb_network_header ( skb ) ;
2007-04-20 07:29:13 +04:00
int packet_len = skb - > tail - skb - > network_header ;
2006-08-24 06:18:35 +04:00
struct ipv6_opt_hdr * hdr ;
int len ;
if ( offset + 2 > packet_len )
goto bad ;
2007-04-11 07:50:43 +04:00
hdr = ( struct ipv6_opt_hdr * ) ( nh + offset ) ;
2006-08-24 06:18:35 +04:00
len = ( ( hdr - > hdrlen + 1 ) < < 3 ) ;
if ( offset + len > packet_len )
goto bad ;
offset + = 2 ;
len - = 2 ;
while ( len > 0 ) {
2007-04-11 07:50:43 +04:00
int opttype = nh [ offset ] ;
2006-08-24 06:18:35 +04:00
int optlen ;
if ( opttype = = type )
return offset ;
switch ( opttype ) {
case IPV6_TLV_PAD0 :
optlen = 1 ;
break ;
default :
2007-04-11 07:50:43 +04:00
optlen = nh [ offset + 1 ] + 2 ;
2006-08-24 06:18:35 +04:00
if ( optlen > len )
goto bad ;
break ;
}
offset + = optlen ;
len - = optlen ;
}
/* not_found */
bad :
return - 1 ;
}
2007-06-27 10:56:32 +04:00
EXPORT_SYMBOL_GPL ( ipv6_find_tlv ) ;
2006-08-24 06:18:35 +04:00
2005-04-17 02:20:36 +04:00
/*
* Parsing tlv encoded headers .
*
* Parsing function " func " returns 1 , if parsing succeed
* and 0 , if it failed .
* It MUST NOT touch skb - > h .
*/
struct tlvtype_proc {
int type ;
2007-10-15 23:50:28 +04:00
int ( * func ) ( struct sk_buff * skb , int offset ) ;
2005-04-17 02:20:36 +04:00
} ;
/*********************
Generic functions
* * * * * * * * * * * * * * * * * * * * */
/* An unknown option is detected, decide what to do */
2007-10-15 23:50:28 +04:00
static int ip6_tlvopt_unknown ( struct sk_buff * skb , int optoff )
2005-04-17 02:20:36 +04:00
{
2007-04-11 07:50:43 +04:00
switch ( ( skb_network_header ( skb ) [ optoff ] & 0xC0 ) > > 6 ) {
2005-04-17 02:20:36 +04:00
case 0 : /* ignore */
return 1 ;
case 1 : /* drop packet */
break ;
case 3 : /* Send ICMP if not a multicast address and drop packet */
/* Actually, it is redundant check. icmp_send
will recheck in any case .
*/
2007-04-26 04:54:47 +04:00
if ( ipv6_addr_is_multicast ( & ipv6_hdr ( skb ) - > daddr ) )
2005-04-17 02:20:36 +04:00
break ;
case 2 : /* send ICMP PARM PROB regardless and drop packet */
icmpv6_param_prob ( skb , ICMPV6_UNK_OPTION , optoff ) ;
return 0 ;
2007-04-21 04:09:22 +04:00
}
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return 0 ;
}
/* Parse tlv encoded option header (hop-by-hop or destination) */
2007-10-15 23:50:28 +04:00
static int ip6_parse_tlv ( struct tlvtype_proc * procs , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
struct tlvtype_proc * curr ;
2007-04-11 07:50:43 +04:00
const unsigned char * nh = skb_network_header ( skb ) ;
2007-03-16 23:26:39 +03:00
int off = skb_network_header_len ( skb ) ;
2007-04-26 05:04:18 +04:00
int len = ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ;
2005-04-17 02:20:36 +04:00
2007-04-26 04:55:53 +04:00
if ( skb_transport_offset ( skb ) + len > skb_headlen ( skb ) )
2005-04-17 02:20:36 +04:00
goto bad ;
off + = 2 ;
len - = 2 ;
while ( len > 0 ) {
2007-04-11 07:50:43 +04:00
int optlen = nh [ off + 1 ] + 2 ;
2005-04-17 02:20:36 +04:00
2007-04-11 07:50:43 +04:00
switch ( nh [ off ] ) {
2005-04-17 02:20:36 +04:00
case IPV6_TLV_PAD0 :
optlen = 1 ;
break ;
case IPV6_TLV_PADN :
break ;
default : /* Other TLV code so scan list */
if ( optlen > len )
goto bad ;
for ( curr = procs ; curr - > type > = 0 ; curr + + ) {
2007-04-11 07:50:43 +04:00
if ( curr - > type = = nh [ off ] ) {
2007-02-09 17:24:49 +03:00
/* type specific length/alignment
checks will be performed in the
2005-04-17 02:20:36 +04:00
func ( ) . */
2007-10-15 23:50:28 +04:00
if ( curr - > func ( skb , off ) = = 0 )
2005-04-17 02:20:36 +04:00
return 0 ;
break ;
}
}
if ( curr - > type < 0 ) {
2007-10-15 23:50:28 +04:00
if ( ip6_tlvopt_unknown ( skb , off ) = = 0 )
2005-04-17 02:20:36 +04:00
return 0 ;
}
break ;
}
off + = optlen ;
len - = optlen ;
}
if ( len = = 0 )
return 1 ;
bad :
kfree_skb ( skb ) ;
return 0 ;
}
/*****************************
Destination options header .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2007-10-15 23:50:28 +04:00
static int ipv6_dest_hao ( struct sk_buff * skb , int optoff )
2006-08-24 06:24:48 +04:00
{
struct ipv6_destopt_hao * hao ;
struct inet6_skb_parm * opt = IP6CB ( skb ) ;
2007-04-26 04:54:47 +04:00
struct ipv6hdr * ipv6h = ipv6_hdr ( skb ) ;
2006-08-24 06:24:48 +04:00
struct in6_addr tmp_addr ;
int ret ;
if ( opt - > dsthao ) {
LIMIT_NETDEBUG ( KERN_DEBUG " hao duplicated \n " ) ;
goto discard ;
}
opt - > dsthao = opt - > dst1 ;
opt - > dst1 = 0 ;
2007-04-11 07:50:43 +04:00
hao = ( struct ipv6_destopt_hao * ) ( skb_network_header ( skb ) + optoff ) ;
2006-08-24 06:24:48 +04:00
if ( hao - > length ! = 16 ) {
LIMIT_NETDEBUG (
KERN_DEBUG " hao invalid option length = %d \n " , hao - > length ) ;
goto discard ;
}
if ( ! ( ipv6_addr_type ( & hao - > addr ) & IPV6_ADDR_UNICAST ) ) {
LIMIT_NETDEBUG (
KERN_DEBUG " hao is not an unicast addr: " NIP6_FMT " \n " , NIP6 ( hao - > addr ) ) ;
goto discard ;
}
ret = xfrm6_input_addr ( skb , ( xfrm_address_t * ) & ipv6h - > daddr ,
( xfrm_address_t * ) & hao - > addr , IPPROTO_DSTOPTS ) ;
if ( unlikely ( ret < 0 ) )
goto discard ;
if ( skb_cloned ( skb ) ) {
2007-10-15 12:29:10 +04:00
if ( pskb_expand_head ( skb , 0 , 0 , GFP_ATOMIC ) )
2006-08-24 06:24:48 +04:00
goto discard ;
/* update all variable using below by copied skbuff */
2007-10-15 12:29:10 +04:00
hao = ( struct ipv6_destopt_hao * ) ( skb_network_header ( skb ) +
2007-04-11 07:50:43 +04:00
optoff ) ;
2007-10-15 12:29:10 +04:00
ipv6h = ipv6_hdr ( skb ) ;
2006-08-24 06:24:48 +04:00
}
if ( skb - > ip_summed = = CHECKSUM_COMPLETE )
skb - > ip_summed = CHECKSUM_NONE ;
ipv6_addr_copy ( & tmp_addr , & ipv6h - > saddr ) ;
ipv6_addr_copy ( & ipv6h - > saddr , & hao - > addr ) ;
ipv6_addr_copy ( & hao - > addr , & tmp_addr ) ;
2007-04-20 03:16:32 +04:00
if ( skb - > tstamp . tv64 = = 0 )
2006-08-24 06:24:48 +04:00
__net_timestamp ( skb ) ;
return 1 ;
discard :
kfree_skb ( skb ) ;
return 0 ;
}
# endif
2005-04-17 02:20:36 +04:00
static struct tlvtype_proc tlvprocdestopt_lst [ ] = {
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:24:48 +04:00
{
. type = IPV6_TLV_HAO ,
. func = ipv6_dest_hao ,
} ,
# endif
2005-04-17 02:20:36 +04:00
{ - 1 , NULL }
} ;
2007-10-15 23:50:28 +04:00
static int ipv6_destopt_rcv ( struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
struct inet6_skb_parm * opt = IP6CB ( skb ) ;
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:24:48 +04:00
__u16 dstbuf ;
# endif
2006-11-04 14:11:37 +03:00
struct dst_entry * dst ;
2005-04-17 02:20:36 +04:00
2007-04-26 04:55:53 +04:00
if ( ! pskb_may_pull ( skb , skb_transport_offset ( skb ) + 8 ) | |
! pskb_may_pull ( skb , ( skb_transport_offset ( skb ) +
2007-04-26 05:04:18 +04:00
( ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ) ) ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( dev_net ( skb - > dst - > dev ) , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
2007-03-16 23:26:39 +03:00
opt - > lastopt = opt - > dst1 = skb_network_header_len ( skb ) ;
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:24:48 +04:00
dstbuf = opt - > dst1 ;
# endif
2005-04-17 02:20:36 +04:00
2006-11-04 14:11:37 +03:00
dst = dst_clone ( skb - > dst ) ;
2007-10-15 23:50:28 +04:00
if ( ip6_parse_tlv ( tlvprocdestopt_lst , skb ) ) {
2006-11-04 14:11:37 +03:00
dst_release ( dst ) ;
2007-04-11 08:21:55 +04:00
skb - > transport_header + = ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ;
2006-09-01 02:18:49 +04:00
opt = IP6CB ( skb ) ;
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:24:48 +04:00
opt - > nhoff = dstbuf ;
# else
2006-01-07 10:02:34 +03:00
opt - > nhoff = opt - > dst1 ;
2006-08-24 06:24:48 +04:00
# endif
2005-04-17 02:20:36 +04:00
return 1 ;
}
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( dev_net ( dst - > dev ) ,
ip6_dst_idev ( dst ) , IPSTATS_MIB_INHDRERRORS ) ;
2006-11-04 14:11:37 +03:00
dst_release ( dst ) ;
2005-04-17 02:20:36 +04:00
return - 1 ;
}
/********************************
Routing header .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-15 23:50:28 +04:00
static int ipv6_rthdr_rcv ( struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
struct inet6_skb_parm * opt = IP6CB ( skb ) ;
2006-08-24 06:16:22 +04:00
struct in6_addr * addr = NULL ;
2005-04-17 02:20:36 +04:00
struct in6_addr daddr ;
2007-04-25 01:58:30 +04:00
struct inet6_dev * idev ;
2005-04-17 02:20:36 +04:00
int n , i ;
struct ipv6_rt_hdr * hdr ;
struct rt0_hdr * rthdr ;
2008-10-08 22:09:27 +04:00
struct net * net = dev_net ( skb - > dev ) ;
int accept_source_route = net - > ipv6 . devconf_all - > accept_source_route ;
2007-04-25 01:58:30 +04:00
2007-07-11 09:47:58 +04:00
idev = in6_dev_get ( skb - > dev ) ;
if ( idev ) {
if ( accept_source_route > idev - > cnf . accept_source_route )
accept_source_route = idev - > cnf . accept_source_route ;
2007-04-25 01:58:30 +04:00
in6_dev_put ( idev ) ;
}
2007-04-26 04:55:53 +04:00
if ( ! pskb_may_pull ( skb , skb_transport_offset ( skb ) + 8 ) | |
! pskb_may_pull ( skb , ( skb_transport_offset ( skb ) +
2007-04-26 05:04:18 +04:00
( ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ) ) ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
2007-04-26 05:04:18 +04:00
hdr = ( struct ipv6_rt_hdr * ) skb_transport_header ( skb ) ;
2005-04-17 02:20:36 +04:00
2007-04-26 04:54:47 +04:00
if ( ipv6_addr_is_multicast ( & ipv6_hdr ( skb ) - > daddr ) | |
2005-04-17 02:20:36 +04:00
skb - > pkt_type ! = PACKET_HOST ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INADDRERRORS ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
looped_back :
if ( hdr - > segments_left = = 0 ) {
2006-08-24 06:16:22 +04:00
switch ( hdr - > type ) {
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:16:22 +04:00
case IPV6_SRCRT_TYPE_2 :
/* Silently discard type 2 header unless it was
* processed by own
*/
if ( ! addr ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INADDRERRORS ) ;
2006-08-24 06:16:22 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
break ;
# endif
default :
break ;
}
2007-03-16 23:26:39 +03:00
opt - > lastopt = opt - > srcrt = skb_network_header_len ( skb ) ;
2007-04-11 08:21:55 +04:00
skb - > transport_header + = ( hdr - > hdrlen + 1 ) < < 3 ;
2005-04-17 02:20:36 +04:00
opt - > dst0 = opt - > dst1 ;
opt - > dst1 = 0 ;
2007-04-11 07:50:43 +04:00
opt - > nhoff = ( & hdr - > nexthdr ) - skb_network_header ( skb ) ;
2005-04-17 02:20:36 +04:00
return 1 ;
}
2006-08-24 06:16:22 +04:00
switch ( hdr - > type ) {
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:16:22 +04:00
case IPV6_SRCRT_TYPE_2 :
2007-07-11 09:47:58 +04:00
if ( accept_source_route < 0 )
goto unknown_rh ;
2006-08-24 06:16:22 +04:00
/* Silently discard invalid RTH type 2 */
if ( hdr - > hdrlen ! = 2 | | hdr - > segments_left ! = 1 ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2006-08-24 06:16:22 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
break ;
# endif
2007-07-11 09:47:58 +04:00
default :
goto unknown_rh ;
2005-04-17 02:20:36 +04:00
}
/*
* This is the routing header forwarding algorithm from
* RFC 2460 , page 16.
*/
n = hdr - > hdrlen > > 1 ;
if ( hdr - > segments_left > n ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2007-04-11 07:50:43 +04:00
icmpv6_param_prob ( skb , ICMPV6_HDR_FIELD ,
( ( & hdr - > segments_left ) -
skb_network_header ( skb ) ) ) ;
2005-04-17 02:20:36 +04:00
return - 1 ;
}
/* We are about to mangle packet header. Be careful!
Do not damage packets queued somewhere .
*/
if ( skb_cloned ( skb ) ) {
/* the copy is a forwarded packet */
2007-10-15 12:29:10 +04:00
if ( pskb_expand_head ( skb , 0 , 0 , GFP_ATOMIC ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_OUTDISCARDS ) ;
kfree_skb ( skb ) ;
2005-04-17 02:20:36 +04:00
return - 1 ;
}
2007-10-15 12:29:10 +04:00
hdr = ( struct ipv6_rt_hdr * ) skb_transport_header ( skb ) ;
2005-04-17 02:20:36 +04:00
}
2006-08-30 03:44:56 +04:00
if ( skb - > ip_summed = = CHECKSUM_COMPLETE )
2005-04-17 02:20:36 +04:00
skb - > ip_summed = CHECKSUM_NONE ;
i = n - - - hdr - > segments_left ;
rthdr = ( struct rt0_hdr * ) hdr ;
addr = rthdr - > addr ;
addr + = i - 1 ;
2006-08-24 06:16:22 +04:00
switch ( hdr - > type ) {
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2006-08-24 06:16:22 +04:00
case IPV6_SRCRT_TYPE_2 :
if ( xfrm6_input_addr ( skb , ( xfrm_address_t * ) addr ,
2007-04-26 04:54:47 +04:00
( xfrm_address_t * ) & ipv6_hdr ( skb ) - > saddr ,
2006-08-24 06:16:22 +04:00
IPPROTO_ROUTING ) < 0 ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INADDRERRORS ) ;
2006-08-24 06:16:22 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
2008-07-11 03:54:50 +04:00
if ( ! ipv6_chk_home_addr ( dev_net ( skb - > dst - > dev ) , addr ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INADDRERRORS ) ;
2006-08-24 06:16:22 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
break ;
# endif
default :
break ;
}
2005-04-17 02:20:36 +04:00
if ( ipv6_addr_is_multicast ( addr ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INADDRERRORS ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
ipv6_addr_copy ( & daddr , addr ) ;
2007-04-26 04:54:47 +04:00
ipv6_addr_copy ( addr , & ipv6_hdr ( skb ) - > daddr ) ;
ipv6_addr_copy ( & ipv6_hdr ( skb ) - > daddr , & daddr ) ;
2005-04-17 02:20:36 +04:00
dst_release ( xchg ( & skb - > dst , NULL ) ) ;
ip6_route_input ( skb ) ;
if ( skb - > dst - > error ) {
2007-04-11 07:50:43 +04:00
skb_push ( skb , skb - > data - skb_network_header ( skb ) ) ;
2005-04-17 02:20:36 +04:00
dst_input ( skb ) ;
return - 1 ;
}
if ( skb - > dst - > dev - > flags & IFF_LOOPBACK ) {
2007-04-26 04:54:47 +04:00
if ( ipv6_hdr ( skb ) - > hop_limit < = 1 ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
icmpv6_send ( skb , ICMPV6_TIME_EXCEED , ICMPV6_EXC_HOPLIMIT ,
0 , skb - > dev ) ;
kfree_skb ( skb ) ;
return - 1 ;
}
2007-04-26 04:54:47 +04:00
ipv6_hdr ( skb ) - > hop_limit - - ;
2005-04-17 02:20:36 +04:00
goto looped_back ;
}
2007-04-11 07:50:43 +04:00
skb_push ( skb , skb - > data - skb_network_header ( skb ) ) ;
2005-04-17 02:20:36 +04:00
dst_input ( skb ) ;
return - 1 ;
2007-07-11 09:47:58 +04:00
unknown_rh :
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ip6_dst_idev ( skb - > dst ) , IPSTATS_MIB_INHDRERRORS ) ;
2007-07-11 09:47:58 +04:00
icmpv6_param_prob ( skb , ICMPV6_HDR_FIELD ,
( & hdr - > type ) - skb_network_header ( skb ) ) ;
return - 1 ;
2005-04-17 02:20:36 +04:00
}
static struct inet6_protocol rthdr_protocol = {
. handler = ipv6_rthdr_rcv ,
2006-07-01 00:36:15 +04:00
. flags = INET6_PROTO_NOPOLICY | INET6_PROTO_GSO_EXTHDR ,
2005-04-17 02:20:36 +04:00
} ;
2007-12-11 13:23:54 +03:00
static struct inet6_protocol destopt_protocol = {
. handler = ipv6_destopt_rcv ,
. flags = INET6_PROTO_NOPOLICY | INET6_PROTO_GSO_EXTHDR ,
} ;
static struct inet6_protocol nodata_protocol = {
. handler = dst_discard ,
. flags = INET6_PROTO_NOPOLICY ,
} ;
int __init ipv6_exthdrs_init ( void )
2005-04-17 02:20:36 +04:00
{
2007-12-11 13:23:54 +03:00
int ret ;
ret = inet6_add_protocol ( & rthdr_protocol , IPPROTO_ROUTING ) ;
if ( ret )
goto out ;
ret = inet6_add_protocol ( & destopt_protocol , IPPROTO_DSTOPTS ) ;
if ( ret )
goto out_rthdr ;
ret = inet6_add_protocol ( & nodata_protocol , IPPROTO_NONE ) ;
if ( ret )
goto out_destopt ;
out :
return ret ;
out_rthdr :
inet6_del_protocol ( & rthdr_protocol , IPPROTO_ROUTING ) ;
out_destopt :
inet6_del_protocol ( & destopt_protocol , IPPROTO_DSTOPTS ) ;
goto out ;
2005-04-17 02:20:36 +04:00
} ;
2007-12-11 13:23:54 +03:00
void ipv6_exthdrs_exit ( void )
{
inet6_del_protocol ( & nodata_protocol , IPPROTO_NONE ) ;
inet6_del_protocol ( & destopt_protocol , IPPROTO_DSTOPTS ) ;
inet6_del_protocol ( & rthdr_protocol , IPPROTO_ROUTING ) ;
}
2005-04-17 02:20:36 +04:00
/**********************************
Hop - by - hop options .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-05-10 01:01:59 +04:00
/*
* Note : we cannot rely on skb - > dst before we assign it in ip6_route_input ( ) .
*/
static inline struct inet6_dev * ipv6_skb_idev ( struct sk_buff * skb )
{
return skb - > dst ? ip6_dst_idev ( skb - > dst ) : __in6_dev_get ( skb - > dev ) ;
}
2005-04-17 02:20:36 +04:00
/* Router Alert as of RFC 2711 */
2007-10-15 23:50:28 +04:00
static int ipv6_hop_ra ( struct sk_buff * skb , int optoff )
2005-04-17 02:20:36 +04:00
{
2007-04-11 07:50:43 +04:00
const unsigned char * nh = skb_network_header ( skb ) ;
2006-08-24 06:19:50 +04:00
2007-04-11 07:50:43 +04:00
if ( nh [ optoff + 1 ] = = 2 ) {
2005-04-17 02:20:36 +04:00
IP6CB ( skb ) - > ra = optoff ;
return 1 ;
}
2005-08-10 07:50:53 +04:00
LIMIT_NETDEBUG ( KERN_DEBUG " ipv6_hop_ra: wrong RA length %d \n " ,
2007-04-11 07:50:43 +04:00
nh [ optoff + 1 ] ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return 0 ;
}
/* Jumbo payload */
2007-10-15 23:50:28 +04:00
static int ipv6_hop_jumbo ( struct sk_buff * skb , int optoff )
2005-04-17 02:20:36 +04:00
{
2007-04-11 07:50:43 +04:00
const unsigned char * nh = skb_network_header ( skb ) ;
2005-04-17 02:20:36 +04:00
u32 pkt_len ;
2008-10-08 22:09:27 +04:00
struct net * net = dev_net ( skb - > dst - > dev ) ;
2005-04-17 02:20:36 +04:00
2007-04-11 07:50:43 +04:00
if ( nh [ optoff + 1 ] ! = 4 | | ( optoff & 3 ) ! = 2 ) {
2005-08-10 07:50:53 +04:00
LIMIT_NETDEBUG ( KERN_DEBUG " ipv6_hop_jumbo: wrong jumbo opt length/alignment %d \n " ,
2007-04-11 07:50:43 +04:00
nh [ optoff + 1 ] ) ;
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ipv6_skb_idev ( skb ) ,
2006-11-04 14:11:37 +03:00
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
goto drop ;
}
2007-04-11 07:50:43 +04:00
pkt_len = ntohl ( * ( __be32 * ) ( nh + optoff + 2 ) ) ;
2005-04-17 02:20:36 +04:00
if ( pkt_len < = IPV6_MAXPLEN ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ipv6_skb_idev ( skb ) ,
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
icmpv6_param_prob ( skb , ICMPV6_HDR_FIELD , optoff + 2 ) ;
return 0 ;
}
2007-04-26 04:54:47 +04:00
if ( ipv6_hdr ( skb ) - > payload_len ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ipv6_skb_idev ( skb ) ,
IPSTATS_MIB_INHDRERRORS ) ;
2005-04-17 02:20:36 +04:00
icmpv6_param_prob ( skb , ICMPV6_HDR_FIELD , optoff ) ;
return 0 ;
}
if ( pkt_len > skb - > len - sizeof ( struct ipv6hdr ) ) {
2008-10-08 22:09:27 +04:00
IP6_INC_STATS_BH ( net , ipv6_skb_idev ( skb ) ,
IPSTATS_MIB_INTRUNCATEDPKTS ) ;
2005-04-17 02:20:36 +04:00
goto drop ;
}
2005-09-08 23:57:43 +04:00
if ( pskb_trim_rcsum ( skb , pkt_len + sizeof ( struct ipv6hdr ) ) )
goto drop ;
2005-04-17 02:20:36 +04:00
return 1 ;
drop :
kfree_skb ( skb ) ;
return 0 ;
}
static struct tlvtype_proc tlvprochopopt_lst [ ] = {
{
. type = IPV6_TLV_ROUTERALERT ,
. func = ipv6_hop_ra ,
} ,
{
. type = IPV6_TLV_JUMBO ,
. func = ipv6_hop_jumbo ,
} ,
{ - 1 , }
} ;
2007-10-15 23:50:28 +04:00
int ipv6_parse_hopopts ( struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2006-01-07 10:02:34 +03:00
struct inet6_skb_parm * opt = IP6CB ( skb ) ;
2006-04-19 01:46:26 +04:00
/*
2007-04-11 07:50:43 +04:00
* skb_network_header ( skb ) is equal to skb - > data , and
2007-03-16 23:26:39 +03:00
* skb_network_header_len ( skb ) is always equal to
2006-04-19 01:46:26 +04:00
* sizeof ( struct ipv6hdr ) by definition of
* hop - by - hop options .
*/
if ( ! pskb_may_pull ( skb , sizeof ( struct ipv6hdr ) + 8 ) | |
2007-04-26 05:04:18 +04:00
! pskb_may_pull ( skb , ( sizeof ( struct ipv6hdr ) +
( ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ) ) ) ) {
2006-04-19 01:46:26 +04:00
kfree_skb ( skb ) ;
return - 1 ;
}
2006-01-07 10:02:34 +03:00
opt - > hop = sizeof ( struct ipv6hdr ) ;
2007-10-15 23:50:28 +04:00
if ( ip6_parse_tlv ( tlvprochopopt_lst , skb ) ) {
2007-04-11 08:21:55 +04:00
skb - > transport_header + = ( skb_transport_header ( skb ) [ 1 ] + 1 ) < < 3 ;
2006-09-01 02:18:49 +04:00
opt = IP6CB ( skb ) ;
2006-01-07 10:02:34 +03:00
opt - > nhoff = sizeof ( struct ipv6hdr ) ;
2006-04-19 01:48:45 +04:00
return 1 ;
2006-01-07 10:02:34 +03:00
}
2005-04-17 02:20:36 +04:00
return - 1 ;
}
/*
* Creating outbound headers .
*
* " build " functions work when skb is filled from head to tail ( datagram )
* " push " functions work when headers are added from tail to head ( tcp )
*
* In both cases we assume , that caller reserved enough room
* for headers .
*/
static void ipv6_push_rthdr ( struct sk_buff * skb , u8 * proto ,
struct ipv6_rt_hdr * opt ,
struct in6_addr * * addr_p )
{
struct rt0_hdr * phdr , * ihdr ;
int hops ;
ihdr = ( struct rt0_hdr * ) opt ;
2007-02-09 17:24:49 +03:00
2005-04-17 02:20:36 +04:00
phdr = ( struct rt0_hdr * ) skb_push ( skb , ( ihdr - > rt_hdr . hdrlen + 1 ) < < 3 ) ;
memcpy ( phdr , ihdr , sizeof ( struct rt0_hdr ) ) ;
hops = ihdr - > rt_hdr . hdrlen > > 1 ;
if ( hops > 1 )
memcpy ( phdr - > addr , ihdr - > addr + 1 ,
( hops - 1 ) * sizeof ( struct in6_addr ) ) ;
ipv6_addr_copy ( phdr - > addr + ( hops - 1 ) , * addr_p ) ;
* addr_p = ihdr - > addr ;
phdr - > rt_hdr . nexthdr = * proto ;
* proto = NEXTHDR_ROUTING ;
}
static void ipv6_push_exthdr ( struct sk_buff * skb , u8 * proto , u8 type , struct ipv6_opt_hdr * opt )
{
struct ipv6_opt_hdr * h = ( struct ipv6_opt_hdr * ) skb_push ( skb , ipv6_optlen ( opt ) ) ;
memcpy ( h , opt , ipv6_optlen ( opt ) ) ;
h - > nexthdr = * proto ;
* proto = type ;
}
void ipv6_push_nfrag_opts ( struct sk_buff * skb , struct ipv6_txoptions * opt ,
u8 * proto ,
struct in6_addr * * daddr )
{
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
if ( opt - > srcrt ) {
2005-04-17 02:20:36 +04:00
ipv6_push_rthdr ( skb , proto , opt - > srcrt , daddr ) ;
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
/*
* IPV6_RTHDRDSTOPTS is ignored
* unless IPV6_RTHDR is set ( RFC3542 ) .
*/
if ( opt - > dst0opt )
ipv6_push_exthdr ( skb , proto , NEXTHDR_DEST , opt - > dst0opt ) ;
}
2005-04-17 02:20:36 +04:00
if ( opt - > hopopt )
ipv6_push_exthdr ( skb , proto , NEXTHDR_HOP , opt - > hopopt ) ;
}
2007-02-22 16:05:40 +03:00
EXPORT_SYMBOL ( ipv6_push_nfrag_opts ) ;
2005-04-17 02:20:36 +04:00
void ipv6_push_frag_opts ( struct sk_buff * skb , struct ipv6_txoptions * opt , u8 * proto )
{
if ( opt - > dst1opt )
ipv6_push_exthdr ( skb , proto , NEXTHDR_DEST , opt - > dst1opt ) ;
}
struct ipv6_txoptions *
ipv6_dup_options ( struct sock * sk , struct ipv6_txoptions * opt )
{
struct ipv6_txoptions * opt2 ;
opt2 = sock_kmalloc ( sk , opt - > tot_len , GFP_ATOMIC ) ;
if ( opt2 ) {
long dif = ( char * ) opt2 - ( char * ) opt ;
memcpy ( opt2 , opt , opt - > tot_len ) ;
if ( opt2 - > hopopt )
* ( ( char * * ) & opt2 - > hopopt ) + = dif ;
if ( opt2 - > dst0opt )
* ( ( char * * ) & opt2 - > dst0opt ) + = dif ;
if ( opt2 - > dst1opt )
* ( ( char * * ) & opt2 - > dst1opt ) + = dif ;
if ( opt2 - > srcrt )
* ( ( char * * ) & opt2 - > srcrt ) + = dif ;
}
return opt2 ;
}
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
2005-12-14 10:23:20 +03:00
EXPORT_SYMBOL_GPL ( ipv6_dup_options ) ;
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
static int ipv6_renew_option ( void * ohdr ,
struct ipv6_opt_hdr __user * newopt , int newoptlen ,
int inherit ,
struct ipv6_opt_hdr * * hdr ,
char * * p )
{
if ( inherit ) {
if ( ohdr ) {
memcpy ( * p , ohdr , ipv6_optlen ( ( struct ipv6_opt_hdr * ) ohdr ) ) ;
* hdr = ( struct ipv6_opt_hdr * ) * p ;
* p + = CMSG_ALIGN ( ipv6_optlen ( * ( struct ipv6_opt_hdr * * ) hdr ) ) ;
}
} else {
if ( newopt ) {
if ( copy_from_user ( * p , newopt , newoptlen ) )
return - EFAULT ;
* hdr = ( struct ipv6_opt_hdr * ) * p ;
if ( ipv6_optlen ( * ( struct ipv6_opt_hdr * * ) hdr ) > newoptlen )
return - EINVAL ;
* p + = CMSG_ALIGN ( newoptlen ) ;
}
}
return 0 ;
}
struct ipv6_txoptions *
ipv6_renew_options ( struct sock * sk , struct ipv6_txoptions * opt ,
int newtype ,
struct ipv6_opt_hdr __user * newopt , int newoptlen )
{
int tot_len = 0 ;
char * p ;
struct ipv6_txoptions * opt2 ;
int err ;
2006-09-01 01:52:17 +04:00
if ( opt ) {
if ( newtype ! = IPV6_HOPOPTS & & opt - > hopopt )
tot_len + = CMSG_ALIGN ( ipv6_optlen ( opt - > hopopt ) ) ;
if ( newtype ! = IPV6_RTHDRDSTOPTS & & opt - > dst0opt )
tot_len + = CMSG_ALIGN ( ipv6_optlen ( opt - > dst0opt ) ) ;
if ( newtype ! = IPV6_RTHDR & & opt - > srcrt )
tot_len + = CMSG_ALIGN ( ipv6_optlen ( opt - > srcrt ) ) ;
if ( newtype ! = IPV6_DSTOPTS & & opt - > dst1opt )
tot_len + = CMSG_ALIGN ( ipv6_optlen ( opt - > dst1opt ) ) ;
}
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
if ( newopt & & newoptlen )
tot_len + = CMSG_ALIGN ( newoptlen ) ;
if ( ! tot_len )
return NULL ;
2005-11-20 06:18:17 +03:00
tot_len + = sizeof ( * opt2 ) ;
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
opt2 = sock_kmalloc ( sk , tot_len , GFP_ATOMIC ) ;
if ( ! opt2 )
return ERR_PTR ( - ENOBUFS ) ;
memset ( opt2 , 0 , tot_len ) ;
opt2 - > tot_len = tot_len ;
p = ( char * ) ( opt2 + 1 ) ;
2006-09-01 01:52:17 +04:00
err = ipv6_renew_option ( opt ? opt - > hopopt : NULL , newopt , newoptlen ,
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
newtype ! = IPV6_HOPOPTS ,
& opt2 - > hopopt , & p ) ;
if ( err )
goto out ;
2006-09-01 01:52:17 +04:00
err = ipv6_renew_option ( opt ? opt - > dst0opt : NULL , newopt , newoptlen ,
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
newtype ! = IPV6_RTHDRDSTOPTS ,
& opt2 - > dst0opt , & p ) ;
if ( err )
goto out ;
2006-09-01 01:52:17 +04:00
err = ipv6_renew_option ( opt ? opt - > srcrt : NULL , newopt , newoptlen ,
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
newtype ! = IPV6_RTHDR ,
2006-09-01 01:52:17 +04:00
( struct ipv6_opt_hdr * * ) & opt2 - > srcrt , & p ) ;
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
if ( err )
goto out ;
2006-09-01 01:52:17 +04:00
err = ipv6_renew_option ( opt ? opt - > dst1opt : NULL , newopt , newoptlen ,
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
newtype ! = IPV6_DSTOPTS ,
& opt2 - > dst1opt , & p ) ;
if ( err )
goto out ;
opt2 - > opt_nflen = ( opt2 - > hopopt ? ipv6_optlen ( opt2 - > hopopt ) : 0 ) +
( opt2 - > dst0opt ? ipv6_optlen ( opt2 - > dst0opt ) : 0 ) +
( opt2 - > srcrt ? ipv6_optlen ( opt2 - > srcrt ) : 0 ) ;
opt2 - > opt_flen = ( opt2 - > dst1opt ? ipv6_optlen ( opt2 - > dst1opt ) : 0 ) ;
return opt2 ;
out :
2005-11-20 06:18:17 +03:00
sock_kfree_s ( sk , opt2 , opt2 - > tot_len ) ;
[IPV6]: Support several new sockopt / ancillary data in Advanced API (RFC3542).
Support several new socket options / ancillary data:
IPV6_RECVPKTINFO, IPV6_PKTINFO,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS,
IPV6_RECVDSTOPTS, IPV6_DSTOPTS, IPV6_RTHDRDSTOPTS,
IPV6_RECVRTHDR, IPV6_RTHDR,
IPV6_RECVHOPOPTS, IPV6_HOPOPTS
Old semantics are preserved as IPV6_2292xxxx so that
we can maintain backward compatibility.
Signed-off-by: YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
2005-09-08 04:59:17 +04:00
return ERR_PTR ( err ) ;
}
2005-11-20 06:23:18 +03:00
struct ipv6_txoptions * ipv6_fixup_options ( struct ipv6_txoptions * opt_space ,
struct ipv6_txoptions * opt )
{
/*
* ignore the dest before srcrt unless srcrt is being included .
* - - yoshfuji
*/
if ( opt & & opt - > dst0opt & & ! opt - > srcrt ) {
if ( opt_space ! = opt ) {
memcpy ( opt_space , opt , sizeof ( * opt_space ) ) ;
opt = opt_space ;
}
opt - > opt_nflen - = ipv6_optlen ( opt - > dst0opt ) ;
opt - > dst0opt = NULL ;
}
return opt ;
}