2008-01-11 17:57:09 +03:00
/* SCTP kernel implementation
2005-04-17 02:20:36 +04:00
* ( C ) Copyright IBM Corp . 2001 , 2003
* Copyright ( c ) Cisco 1999 , 2000
* Copyright ( c ) Motorola 1999 , 2000 , 2001
* Copyright ( c ) La Monte H . P . Yarroll 2001
*
2008-01-11 17:57:09 +03:00
* This file is part of the SCTP kernel implementation .
2005-04-17 02:20:36 +04:00
*
* A collection class to handle the storage of transport addresses .
*
2008-01-11 17:57:09 +03:00
* This SCTP implementation is free software ;
2005-04-17 02:20:36 +04:00
* 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 , or ( at your option )
* any later version .
*
2008-01-11 17:57:09 +03:00
* This SCTP implementation is distributed in the hope that it
2005-04-17 02:20:36 +04:00
* will be useful , but WITHOUT ANY WARRANTY ; without even the implied
* * * * * * * * * * * * * * * * * * * * * * * * *
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
* See the GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with GNU CC ; see the file COPYING . If not , write to
* the Free Software Foundation , 59 Temple Place - Suite 330 ,
* Boston , MA 02111 - 1307 , USA .
*
* Please send any bug reports or fixes you make to the
* email address ( es ) :
* lksctp developers < lksctp - developers @ lists . sourceforge . net >
*
* Or submit a bug report through the following website :
* http : //www.sf.net/projects/lksctp
*
* Written or modified by :
* La Monte H . P . Yarroll < piggy @ acm . org >
* Karl Knutson < karl @ athena . chicago . il . us >
* Jon Grimm < jgrimm @ us . ibm . com >
* Daisy Chang < daisyc @ us . ibm . com >
*
* Any bugs reported given to us we will try to fix . . . any fixes shared will
* be incorporated into the next SCTP release .
*/
# include <linux/types.h>
# include <linux/in.h>
# include <net/sock.h>
# include <net/ipv6.h>
# include <net/if_inet6.h>
# include <net/sctp/sctp.h>
# include <net/sctp/sm.h>
/* Forward declarations for internal helpers. */
static int sctp_copy_one_addr ( struct sctp_bind_addr * , union sctp_addr * ,
2005-10-07 10:46:04 +04:00
sctp_scope_t scope , gfp_t gfp ,
2005-07-12 07:57:47 +04:00
int flags ) ;
2005-04-17 02:20:36 +04:00
static void sctp_bind_addr_clean ( struct sctp_bind_addr * ) ;
/* First Level Abstractions. */
/* Copy 'src' to 'dest' taking 'scope' into account. Omit addresses
* in ' src ' which have a broader scope than ' scope ' .
*/
2007-02-09 17:25:18 +03:00
int sctp_bind_addr_copy ( struct sctp_bind_addr * dest ,
2005-04-17 02:20:36 +04:00
const struct sctp_bind_addr * src ,
2005-10-07 10:46:04 +04:00
sctp_scope_t scope , gfp_t gfp ,
2005-07-12 07:57:47 +04:00
int flags )
2005-04-17 02:20:36 +04:00
{
struct sctp_sockaddr_entry * addr ;
int error = 0 ;
/* All addresses share the same port. */
dest - > port = src - > port ;
/* Extract the addresses which are relevant for this scope. */
2008-04-13 05:54:24 +04:00
list_for_each_entry ( addr , & src - > address_list , list ) {
2006-11-21 04:12:07 +03:00
error = sctp_copy_one_addr ( dest , & addr - > a , scope ,
2005-04-17 02:20:36 +04:00
gfp , flags ) ;
if ( error < 0 )
goto out ;
}
/* If there are no addresses matching the scope and
* this is global scope , try to get a link scope address , with
* the assumption that we must be sitting behind a NAT .
*/
if ( list_empty ( & dest - > address_list ) & & ( SCTP_SCOPE_GLOBAL = = scope ) ) {
2008-04-13 05:54:24 +04:00
list_for_each_entry ( addr , & src - > address_list , list ) {
2006-11-21 04:12:07 +03:00
error = sctp_copy_one_addr ( dest , & addr - > a ,
2005-04-17 02:20:36 +04:00
SCTP_SCOPE_LINK , gfp ,
flags ) ;
if ( error < 0 )
goto out ;
}
}
out :
if ( error )
sctp_bind_addr_clean ( dest ) ;
return error ;
}
2007-12-07 09:50:54 +03:00
/* Exactly duplicate the address lists. This is necessary when doing
* peer - offs and accepts . We don ' t want to put all the current system
* addresses into the endpoint . That ' s useless . But we do want duplicat
* the list of bound addresses that the older endpoint used .
*/
int sctp_bind_addr_dup ( struct sctp_bind_addr * dest ,
const struct sctp_bind_addr * src ,
gfp_t gfp )
{
struct sctp_sockaddr_entry * addr ;
int error = 0 ;
/* All addresses share the same port. */
dest - > port = src - > port ;
2008-04-13 05:54:24 +04:00
list_for_each_entry ( addr , & src - > address_list , list ) {
2007-12-07 09:50:54 +03:00
error = sctp_add_bind_addr ( dest , & addr - > a , 1 , gfp ) ;
if ( error < 0 )
break ;
}
return error ;
}
2005-04-17 02:20:36 +04:00
/* Initialize the SCTP_bind_addr structure for either an endpoint or
* an association .
*/
void sctp_bind_addr_init ( struct sctp_bind_addr * bp , __u16 port )
{
bp - > malloced = 0 ;
INIT_LIST_HEAD ( & bp - > address_list ) ;
bp - > port = port ;
}
/* Dispose of the address list. */
static void sctp_bind_addr_clean ( struct sctp_bind_addr * bp )
{
struct sctp_sockaddr_entry * addr ;
struct list_head * pos , * temp ;
/* Empty the bind address list. */
list_for_each_safe ( pos , temp , & bp - > address_list ) {
addr = list_entry ( pos , struct sctp_sockaddr_entry , list ) ;
list_del ( pos ) ;
kfree ( addr ) ;
SCTP_DBG_OBJCNT_DEC ( addr ) ;
}
}
/* Dispose of an SCTP_bind_addr structure */
void sctp_bind_addr_free ( struct sctp_bind_addr * bp )
{
/* Empty the bind address list. */
sctp_bind_addr_clean ( bp ) ;
if ( bp - > malloced ) {
kfree ( bp ) ;
SCTP_DBG_OBJCNT_DEC ( bind_addr ) ;
}
}
/* Add an address to the bind address list in the SCTP_bind_addr structure. */
int sctp_add_bind_addr ( struct sctp_bind_addr * bp , union sctp_addr * new ,
2007-12-21 01:12:24 +03:00
__u8 addr_state , gfp_t gfp )
2005-04-17 02:20:36 +04:00
{
struct sctp_sockaddr_entry * addr ;
/* Add the address to the bind address list. */
addr = t_new ( struct sctp_sockaddr_entry , gfp ) ;
if ( ! addr )
return - ENOMEM ;
2006-11-21 04:10:38 +03:00
memcpy ( & addr - > a , new , sizeof ( * new ) ) ;
2005-04-17 02:20:36 +04:00
/* Fix up the port if it has not yet been set.
* Both v4 and v6 have the port at the same offset .
*/
2006-11-21 04:10:38 +03:00
if ( ! addr - > a . v4 . sin_port )
addr - > a . v4 . sin_port = htons ( bp - > port ) ;
2005-04-17 02:20:36 +04:00
2007-12-21 01:12:24 +03:00
addr - > state = addr_state ;
2007-09-17 03:02:12 +04:00
addr - > valid = 1 ;
2006-07-22 01:49:25 +04:00
2005-04-17 02:20:36 +04:00
INIT_LIST_HEAD ( & addr - > list ) ;
2007-09-17 03:02:12 +04:00
INIT_RCU_HEAD ( & addr - > rcu ) ;
2007-09-17 03:03:28 +04:00
/* We always hold a socket lock when calling this function,
* and that acts as a writer synchronizing lock .
*/
list_add_tail_rcu ( & addr - > list , & bp - > address_list ) ;
2005-04-17 02:20:36 +04:00
SCTP_DBG_OBJCNT_INC ( addr ) ;
return 0 ;
}
/* Delete an address from the bind address list in the SCTP_bind_addr
* structure .
*/
2007-10-25 00:10:00 +04:00
int sctp_del_bind_addr ( struct sctp_bind_addr * bp , union sctp_addr * del_addr )
2005-04-17 02:20:36 +04:00
{
2007-09-17 03:03:28 +04:00
struct sctp_sockaddr_entry * addr , * temp ;
2008-03-12 04:05:02 +03:00
int found = 0 ;
2005-04-17 02:20:36 +04:00
2007-09-17 03:03:28 +04:00
/* We hold the socket lock when calling this function,
* and that acts as a writer synchronizing lock .
*/
list_for_each_entry_safe ( addr , temp , & bp - > address_list , list ) {
2006-11-21 04:07:48 +03:00
if ( sctp_cmp_addr_exact ( & addr - > a , del_addr ) ) {
2005-04-17 02:20:36 +04:00
/* Found the exact match. */
2008-03-12 04:05:02 +03:00
found = 1 ;
2007-09-17 03:03:28 +04:00
addr - > valid = 0 ;
list_del_rcu ( & addr - > list ) ;
break ;
2005-04-17 02:20:36 +04:00
}
}
2008-03-12 04:05:02 +03:00
if ( found ) {
2007-10-25 00:10:00 +04:00
call_rcu ( & addr - > rcu , sctp_local_addr_free ) ;
2007-09-17 03:03:28 +04:00
SCTP_DBG_OBJCNT_DEC ( addr ) ;
2007-10-25 00:10:00 +04:00
return 0 ;
2007-09-17 03:03:28 +04:00
}
2005-04-17 02:20:36 +04:00
return - EINVAL ;
}
/* Create a network byte-order representation of all the addresses
* formated as SCTP parameters .
*
* The second argument is the return value for the length .
*/
union sctp_params sctp_bind_addrs_to_raw ( const struct sctp_bind_addr * bp ,
2005-07-12 07:57:47 +04:00
int * addrs_len ,
2005-10-07 10:46:04 +04:00
gfp_t gfp )
2005-04-17 02:20:36 +04:00
{
union sctp_params addrparms ;
union sctp_params retval ;
int addrparms_len ;
union sctp_addr_param rawaddr ;
int len ;
struct sctp_sockaddr_entry * addr ;
struct list_head * pos ;
struct sctp_af * af ;
addrparms_len = 0 ;
len = 0 ;
/* Allocate enough memory at once. */
list_for_each ( pos , & bp - > address_list ) {
len + = sizeof ( union sctp_addr_param ) ;
}
/* Don't even bother embedding an address if there
* is only one .
*/
if ( len = = sizeof ( union sctp_addr_param ) ) {
retval . v = NULL ;
goto end_raw ;
}
retval . v = kmalloc ( len , gfp ) ;
if ( ! retval . v )
goto end_raw ;
addrparms = retval ;
2008-04-13 05:54:24 +04:00
list_for_each_entry ( addr , & bp - > address_list , list ) {
2006-11-21 04:21:44 +03:00
af = sctp_get_af_specific ( addr - > a . v4 . sin_family ) ;
len = af - > to_addr_param ( & addr - > a , & rawaddr ) ;
2005-04-17 02:20:36 +04:00
memcpy ( addrparms . v , & rawaddr , len ) ;
addrparms . v + = len ;
addrparms_len + = len ;
}
end_raw :
* addrs_len = addrparms_len ;
return retval ;
}
/*
* Create an address list out of the raw address list format ( IPv4 and IPv6
* address parameters ) .
*/
int sctp_raw_to_bind_addrs ( struct sctp_bind_addr * bp , __u8 * raw_addr_list ,
2005-10-07 10:46:04 +04:00
int addrs_len , __u16 port , gfp_t gfp )
2005-04-17 02:20:36 +04:00
{
union sctp_addr_param * rawaddr ;
struct sctp_paramhdr * param ;
union sctp_addr addr ;
int retval = 0 ;
int len ;
struct sctp_af * af ;
/* Convert the raw address to standard address format */
while ( addrs_len ) {
param = ( struct sctp_paramhdr * ) raw_addr_list ;
rawaddr = ( union sctp_addr_param * ) raw_addr_list ;
af = sctp_get_af_specific ( param_type2af ( param - > type ) ) ;
if ( unlikely ( ! af ) ) {
retval = - EINVAL ;
sctp_bind_addr_clean ( bp ) ;
break ;
}
2006-11-21 04:11:13 +03:00
af - > from_addr_param ( & addr , rawaddr , htons ( port ) , 0 ) ;
2007-12-21 01:12:24 +03:00
retval = sctp_add_bind_addr ( bp , & addr , SCTP_ADDR_SRC , gfp ) ;
2005-04-17 02:20:36 +04:00
if ( retval ) {
/* Can't finish building the list, clean up. */
sctp_bind_addr_clean ( bp ) ;
break ;
}
len = ntohs ( param - > length ) ;
addrs_len - = len ;
raw_addr_list + = len ;
}
return retval ;
}
/********************************************************************
* 2 nd Level Abstractions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* Does this contain a specified address? Allow wildcarding. */
2007-02-09 17:25:18 +03:00
int sctp_bind_addr_match ( struct sctp_bind_addr * bp ,
2005-04-17 02:20:36 +04:00
const union sctp_addr * addr ,
struct sctp_sock * opt )
{
struct sctp_sockaddr_entry * laddr ;
2007-09-17 03:03:28 +04:00
int match = 0 ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( laddr , & bp - > address_list , list ) {
if ( ! laddr - > valid )
continue ;
if ( opt - > pf - > cmp_addr ( & laddr - > a , addr , opt ) ) {
match = 1 ;
break ;
}
2005-04-17 02:20:36 +04:00
}
2007-09-17 03:03:28 +04:00
rcu_read_unlock ( ) ;
2005-04-17 02:20:36 +04:00
2007-09-17 03:03:28 +04:00
return match ;
2005-04-17 02:20:36 +04:00
}
2008-07-19 10:05:40 +04:00
/* Does the address 'addr' conflict with any addresses in
* the bp .
*/
int sctp_bind_addr_conflict ( struct sctp_bind_addr * bp ,
const union sctp_addr * addr ,
struct sctp_sock * bp_sp ,
struct sctp_sock * addr_sp )
{
struct sctp_sockaddr_entry * laddr ;
int conflict = 0 ;
struct sctp_sock * sp ;
/* Pick the IPv6 socket as the basis of comparison
* since it ' s usually a superset of the IPv4 .
* If there is no IPv6 socket , then default to bind_addr .
*/
if ( sctp_opt2sk ( bp_sp ) - > sk_family = = AF_INET6 )
sp = bp_sp ;
else if ( sctp_opt2sk ( addr_sp ) - > sk_family = = AF_INET6 )
sp = addr_sp ;
else
sp = bp_sp ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( laddr , & bp - > address_list , list ) {
if ( ! laddr - > valid )
continue ;
conflict = sp - > pf - > cmp_addr ( & laddr - > a , addr , sp ) ;
if ( conflict )
break ;
}
rcu_read_unlock ( ) ;
return conflict ;
}
2007-12-21 01:12:59 +03:00
/* Get the state of the entry in the bind_addr_list */
int sctp_bind_addr_state ( const struct sctp_bind_addr * bp ,
const union sctp_addr * addr )
{
struct sctp_sockaddr_entry * laddr ;
struct sctp_af * af ;
int state = - 1 ;
af = sctp_get_af_specific ( addr - > sa . sa_family ) ;
if ( unlikely ( ! af ) )
return state ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( laddr , & bp - > address_list , list ) {
if ( ! laddr - > valid )
continue ;
if ( af - > cmp_addr ( & laddr - > a , addr ) ) {
state = laddr - > state ;
break ;
}
}
rcu_read_unlock ( ) ;
return state ;
}
2005-04-17 02:20:36 +04:00
/* Find the first address in the bind address list that is not present in
* the addrs packed array .
*/
union sctp_addr * sctp_find_unmatch_addr ( struct sctp_bind_addr * bp ,
const union sctp_addr * addrs ,
int addrcnt ,
struct sctp_sock * opt )
{
struct sctp_sockaddr_entry * laddr ;
union sctp_addr * addr ;
void * addr_buf ;
struct sctp_af * af ;
int i ;
2007-09-17 03:03:28 +04:00
/* This is only called sctp_send_asconf_del_ip() and we hold
* the socket lock in that code patch , so that address list
* can ' t change .
*/
list_for_each_entry ( laddr , & bp - > address_list , list ) {
2005-04-17 02:20:36 +04:00
addr_buf = ( union sctp_addr * ) addrs ;
for ( i = 0 ; i < addrcnt ; i + + ) {
addr = ( union sctp_addr * ) addr_buf ;
af = sctp_get_af_specific ( addr - > v4 . sin_family ) ;
2007-02-09 17:25:18 +03:00
if ( ! af )
2007-09-17 03:03:28 +04:00
break ;
2005-04-17 02:20:36 +04:00
2006-11-21 04:05:23 +03:00
if ( opt - > pf - > cmp_addr ( & laddr - > a , addr , opt ) )
2005-04-17 02:20:36 +04:00
break ;
addr_buf + = af - > sockaddr_len ;
}
if ( i = = addrcnt )
2006-11-21 04:22:08 +03:00
return & laddr - > a ;
2005-04-17 02:20:36 +04:00
}
return NULL ;
}
/* Copy out addresses from the global local address list. */
2007-02-09 17:25:18 +03:00
static int sctp_copy_one_addr ( struct sctp_bind_addr * dest ,
2005-04-17 02:20:36 +04:00
union sctp_addr * addr ,
2005-10-07 10:46:04 +04:00
sctp_scope_t scope , gfp_t gfp ,
2005-07-12 07:57:47 +04:00
int flags )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
if ( sctp_is_any ( addr ) ) {
error = sctp_copy_local_addr_list ( dest , scope , gfp , flags ) ;
} else if ( sctp_in_scope ( addr , scope ) ) {
/* Now that the address is in scope, check to see if
* the address type is supported by local sock as
* well as the remote peer .
*/
if ( ( ( ( AF_INET = = addr - > sa . sa_family ) & &
( flags & SCTP_ADDR4_PEERSUPP ) ) ) | |
( ( ( AF_INET6 = = addr - > sa . sa_family ) & &
( flags & SCTP_ADDR6_ALLOWED ) & &
( flags & SCTP_ADDR6_PEERSUPP ) ) ) )
2007-12-21 01:12:24 +03:00
error = sctp_add_bind_addr ( dest , addr , SCTP_ADDR_SRC ,
gfp ) ;
2005-04-17 02:20:36 +04:00
}
return error ;
}
/* Is this a wildcard address? */
int sctp_is_any ( const union sctp_addr * addr )
{
struct sctp_af * af = sctp_get_af_specific ( addr - > sa . sa_family ) ;
if ( ! af )
return 0 ;
return af - > is_any ( addr ) ;
}
/* Is 'addr' valid for 'scope'? */
int sctp_in_scope ( const union sctp_addr * addr , sctp_scope_t scope )
{
sctp_scope_t addr_scope = sctp_scope ( addr ) ;
/* The unusable SCTP addresses will not be considered with
* any defined scopes .
*/
if ( SCTP_SCOPE_UNUSABLE = = addr_scope )
return 0 ;
/*
* For INIT and INIT - ACK address list , let L be the level of
* of requested destination address , sender and receiver
* SHOULD include all of its addresses with level greater
* than or equal to L .
*/
if ( addr_scope < = scope )
return 1 ;
return 0 ;
}
/********************************************************************
* 3 rd Level Abstractions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* What is the scope of 'addr'? */
sctp_scope_t sctp_scope ( const union sctp_addr * addr )
{
struct sctp_af * af ;
af = sctp_get_af_specific ( addr - > sa . sa_family ) ;
if ( ! af )
return SCTP_SCOPE_UNUSABLE ;
return af - > scope ( ( union sctp_addr * ) addr ) ;
}