2005-04-17 02:20:36 +04:00
# ifndef _ADDRCONF_H
# define _ADDRCONF_H
# define RETRANS_TIMER HZ
# define MAX_RTR_SOLICITATIONS 3
# define RTR_SOLICITATION_INTERVAL (4*HZ)
# define MIN_VALID_LIFETIME (2*3600) /* 2 hours */
# define TEMP_VALID_LIFETIME (7*86400)
# define TEMP_PREFERRED_LIFETIME (86400)
# define REGEN_MAX_RETRY (5)
# define MAX_DESYNC_FACTOR (600)
# define ADDR_CHECK_FREQUENCY (120*HZ)
# define IPV6_MAX_ADDRESSES 16
2007-11-29 14:11:40 +03:00
# include <linux/in.h>
2005-05-04 09:17:18 +04:00
# include <linux/in6.h>
2005-04-17 02:20:36 +04:00
struct prefix_info {
__u8 type ;
__u8 length ;
__u8 prefix_len ;
# if defined(__BIG_ENDIAN_BITFIELD)
__u8 onlink : 1 ,
autoconf : 1 ,
reserved : 6 ;
# elif defined(__LITTLE_ENDIAN_BITFIELD)
__u8 reserved : 6 ,
autoconf : 1 ,
onlink : 1 ;
# else
# error "Please fix <asm / byteorder.h>"
# endif
2006-11-15 07:56:00 +03:00
__be32 valid ;
__be32 prefered ;
__be32 reserved2 ;
2005-04-17 02:20:36 +04:00
struct in6_addr prefix ;
} ;
# ifdef __KERNEL__
# include <linux/netdevice.h>
# include <net/if_inet6.h>
2005-05-04 01:25:13 +04:00
# include <net/ipv6.h>
2005-04-17 02:20:36 +04:00
# define IN6_ADDR_HSIZE 16
extern int addrconf_init ( void ) ;
extern void addrconf_cleanup ( void ) ;
extern int addrconf_add_ifaddr ( void __user * arg ) ;
extern int addrconf_del_ifaddr ( void __user * arg ) ;
extern int addrconf_set_dstaddr ( void __user * arg ) ;
2008-01-11 09:43:18 +03:00
extern int ipv6_chk_addr ( struct net * net ,
struct in6_addr * addr ,
2005-04-17 02:20:36 +04:00
struct net_device * dev ,
int strict ) ;
2008-01-11 09:43:18 +03:00
2007-06-27 10:56:32 +04:00
# if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
2008-01-11 09:44:40 +03:00
extern int ipv6_chk_home_addr ( struct net * net ,
struct in6_addr * addr ) ;
2006-08-24 06:16:22 +04:00
# endif
2008-01-11 09:44:09 +03:00
extern struct inet6_ifaddr * ipv6_get_ifaddr ( struct net * net ,
struct in6_addr * addr ,
struct net_device * dev ,
int strict ) ;
2005-04-17 02:20:36 +04:00
extern int ipv6_get_saddr ( struct dst_entry * dst ,
struct in6_addr * daddr ,
struct in6_addr * saddr ) ;
extern int ipv6_dev_get_saddr ( struct net_device * dev ,
struct in6_addr * daddr ,
struct in6_addr * saddr ) ;
2007-04-26 04:08:10 +04:00
extern int ipv6_get_lladdr ( struct net_device * dev ,
struct in6_addr * addr ,
unsigned char banned_flags ) ;
2005-04-17 02:20:36 +04:00
extern int ipv6_rcv_saddr_equal ( const struct sock * sk ,
const struct sock * sk2 ) ;
extern void addrconf_join_solict ( struct net_device * dev ,
struct in6_addr * addr ) ;
extern void addrconf_leave_solict ( struct inet6_dev * idev ,
struct in6_addr * addr ) ;
2007-11-14 09:56:23 +03:00
/*
* IPv6 Address Label subsystem ( addrlabel . c )
*/
extern int ipv6_addr_label_init ( void ) ;
extern void ipv6_addr_label_rtnl_register ( void ) ;
extern u32 ipv6_addr_label ( const struct in6_addr * addr ,
int type , int ifindex ) ;
2005-04-17 02:20:36 +04:00
/*
* multicast prototypes ( mcast . c )
*/
extern int ipv6_sock_mc_join ( struct sock * sk , int ifindex ,
struct in6_addr * addr ) ;
extern int ipv6_sock_mc_drop ( struct sock * sk , int ifindex ,
struct in6_addr * addr ) ;
extern void ipv6_sock_mc_close ( struct sock * sk ) ;
extern int inet6_mc_check ( struct sock * sk , struct in6_addr * mc_addr ,
struct in6_addr * src_addr ) ;
extern int ipv6_dev_mc_inc ( struct net_device * dev , struct in6_addr * addr ) ;
extern int __ipv6_dev_mc_dec ( struct inet6_dev * idev , struct in6_addr * addr ) ;
extern int ipv6_dev_mc_dec ( struct net_device * dev , struct in6_addr * addr ) ;
extern void ipv6_mc_up ( struct inet6_dev * idev ) ;
extern void ipv6_mc_down ( struct inet6_dev * idev ) ;
extern void ipv6_mc_init_dev ( struct inet6_dev * idev ) ;
extern void ipv6_mc_destroy_dev ( struct inet6_dev * idev ) ;
extern void addrconf_dad_failure ( struct inet6_ifaddr * ifp ) ;
extern int ipv6_chk_mcast_addr ( struct net_device * dev , struct in6_addr * group ,
struct in6_addr * src_addr ) ;
extern int ipv6_is_mld ( struct sk_buff * skb , int nexthdr ) ;
extern void addrconf_prefix_rcv ( struct net_device * dev , u8 * opt , int len ) ;
extern int ipv6_get_hoplimit ( struct net_device * dev ) ;
/*
* anycast prototypes ( anycast . c )
*/
extern int ipv6_sock_ac_join ( struct sock * sk , int ifindex , struct in6_addr * addr ) ;
extern int ipv6_sock_ac_drop ( struct sock * sk , int ifindex , struct in6_addr * addr ) ;
extern void ipv6_sock_ac_close ( struct sock * sk ) ;
extern int inet6_ac_check ( struct sock * sk , struct in6_addr * addr , int ifindex ) ;
extern int ipv6_dev_ac_inc ( struct net_device * dev , struct in6_addr * addr ) ;
extern int __ipv6_dev_ac_dec ( struct inet6_dev * idev , struct in6_addr * addr ) ;
extern int ipv6_chk_acast_addr ( struct net_device * dev , struct in6_addr * addr ) ;
/* Device notifier */
extern int register_inet6addr_notifier ( struct notifier_block * nb ) ;
extern int unregister_inet6addr_notifier ( struct notifier_block * nb ) ;
static inline struct inet6_dev *
__in6_dev_get ( struct net_device * dev )
{
2006-09-23 01:44:24 +04:00
return rcu_dereference ( dev - > ip6_ptr ) ;
2005-04-17 02:20:36 +04:00
}
static inline struct inet6_dev *
in6_dev_get ( struct net_device * dev )
{
struct inet6_dev * idev = NULL ;
2006-09-23 01:44:24 +04:00
rcu_read_lock ( ) ;
idev = __in6_dev_get ( dev ) ;
2005-04-17 02:20:36 +04:00
if ( idev )
atomic_inc ( & idev - > refcnt ) ;
2006-09-23 01:44:24 +04:00
rcu_read_unlock ( ) ;
2005-04-17 02:20:36 +04:00
return idev ;
}
extern void in6_dev_finish_destroy ( struct inet6_dev * idev ) ;
static inline void
in6_dev_put ( struct inet6_dev * idev )
{
if ( atomic_dec_and_test ( & idev - > refcnt ) )
in6_dev_finish_destroy ( idev ) ;
}
# define __in6_dev_put(idev) atomic_dec(&(idev)->refcnt)
# define in6_dev_hold(idev) atomic_inc(&(idev)->refcnt)
extern void inet6_ifa_finish_destroy ( struct inet6_ifaddr * ifp ) ;
static inline void in6_ifa_put ( struct inet6_ifaddr * ifp )
{
if ( atomic_dec_and_test ( & ifp - > refcnt ) )
inet6_ifa_finish_destroy ( ifp ) ;
}
# define __in6_ifa_put(ifp) atomic_dec(&(ifp)->refcnt)
# define in6_ifa_hold(ifp) atomic_inc(&(ifp)->refcnt)
extern void addrconf_forwarding_on ( void ) ;
/*
* Hash function taken from net_alias . c
*/
static __inline__ u8 ipv6_addr_hash ( const struct in6_addr * addr )
{
__u32 word ;
/*
* We perform the hash function over the last 64 bits of the address
* This will include the IEEE address token on links that support it .
*/
2006-11-15 07:56:00 +03:00
word = ( __force u32 ) ( addr - > s6_addr32 [ 2 ] ^ addr - > s6_addr32 [ 3 ] ) ;
2005-04-17 02:20:36 +04:00
word ^ = ( word > > 16 ) ;
word ^ = ( word > > 8 ) ;
return ( ( word ^ ( word > > 4 ) ) & 0x0f ) ;
}
/*
* compute link - local solicited - node multicast address
*/
static inline void addrconf_addr_solict_mult ( const struct in6_addr * addr ,
struct in6_addr * solicited )
{
ipv6_addr_set ( solicited ,
__constant_htonl ( 0xFF020000 ) , 0 ,
__constant_htonl ( 0x1 ) ,
__constant_htonl ( 0xFF000000 ) | addr - > s6_addr32 [ 3 ] ) ;
}
static inline void ipv6_addr_all_nodes ( struct in6_addr * addr )
{
ipv6_addr_set ( addr ,
__constant_htonl ( 0xFF020000 ) , 0 , 0 ,
__constant_htonl ( 0x1 ) ) ;
}
static inline void ipv6_addr_all_routers ( struct in6_addr * addr )
{
ipv6_addr_set ( addr ,
__constant_htonl ( 0xFF020000 ) , 0 , 0 ,
__constant_htonl ( 0x2 ) ) ;
}
static inline int ipv6_addr_is_multicast ( const struct in6_addr * addr )
{
return ( addr - > s6_addr32 [ 0 ] & __constant_htonl ( 0xFF000000 ) ) = = __constant_htonl ( 0xFF000000 ) ;
}
static inline int ipv6_addr_is_ll_all_nodes ( const struct in6_addr * addr )
{
return ( addr - > s6_addr32 [ 0 ] = = htonl ( 0xff020000 ) & &
addr - > s6_addr32 [ 1 ] = = 0 & &
addr - > s6_addr32 [ 2 ] = = 0 & &
addr - > s6_addr32 [ 3 ] = = htonl ( 0x00000001 ) ) ;
}
static inline int ipv6_addr_is_ll_all_routers ( const struct in6_addr * addr )
{
return ( addr - > s6_addr32 [ 0 ] = = htonl ( 0xff020000 ) & &
addr - > s6_addr32 [ 1 ] = = 0 & &
addr - > s6_addr32 [ 2 ] = = 0 & &
addr - > s6_addr32 [ 3 ] = = htonl ( 0x00000002 ) ) ;
}
2007-11-29 14:11:40 +03:00
static inline int ipv6_isatap_eui64 ( u8 * eui , __be32 addr )
{
2007-12-17 00:43:24 +03:00
eui [ 0 ] = ( ipv4_is_zeronet ( addr ) | | ipv4_is_private_10 ( addr ) | |
ipv4_is_loopback ( addr ) | | ipv4_is_linklocal_169 ( addr ) | |
ipv4_is_private_172 ( addr ) | | ipv4_is_test_192 ( addr ) | |
ipv4_is_anycast_6to4 ( addr ) | | ipv4_is_private_192 ( addr ) | |
ipv4_is_test_198 ( addr ) | | ipv4_is_multicast ( addr ) | |
2008-01-21 14:18:08 +03:00
ipv4_is_lbcast ( addr ) ) ? 0x00 : 0x02 ;
2007-11-29 14:11:40 +03:00
eui [ 1 ] = 0 ;
eui [ 2 ] = 0x5E ;
eui [ 3 ] = 0xFE ;
memcpy ( eui + 4 , & addr , 4 ) ;
return 0 ;
}
static inline int ipv6_addr_is_isatap ( const struct in6_addr * addr )
{
return ( ( addr - > s6_addr32 [ 2 ] | htonl ( 0x02000000 ) ) = = htonl ( 0x02005EFE ) ) ;
}
2005-08-16 09:18:02 +04:00
# ifdef CONFIG_PROC_FS
extern int if6_proc_init ( void ) ;
extern void if6_proc_exit ( void ) ;
# endif
2005-04-17 02:20:36 +04:00
# endif
# endif