2005-04-16 15:20:36 -07:00
/*
* INET An implementation of the TCP / IP protocol suite for the LINUX
* operating system . INET is implemented using the BSD Socket
* interface as the means of communication with the user level .
*
* Definitions for the IP module .
*
* Version : @ ( # ) ip . h 1.0 .2 05 / 07 / 93
*
2005-05-05 16:16:16 -07:00
* Authors : Ross Biro
2005-04-16 15:20:36 -07:00
* Fred N . van Kempen , < waltje @ uWalt . NL . Mugnet . ORG >
* Alan Cox , < gw4pts @ gw4pts . ampr . org >
*
* Changes :
* Mike McLagan : Routing by source
*
* 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 .
*/
# ifndef _IP_H
# define _IP_H
# include <linux/types.h>
# include <linux/ip.h>
# include <linux/in.h>
2007-03-12 20:09:15 -03:00
# include <linux/skbuff.h>
2005-12-27 02:43:12 -02:00
# include <net/inet_sock.h>
2005-04-16 15:20:36 -07:00
# include <net/snmp.h>
2008-10-01 07:44:42 -07:00
# include <net/flow.h>
2005-04-16 15:20:36 -07:00
struct sock ;
2009-11-03 03:26:03 +00:00
struct inet_skb_parm {
2005-04-16 15:20:36 -07:00
struct ip_options opt ; /* Compiled IP options */
unsigned char flags ;
2006-01-06 23:04:01 -08:00
# define IPSKB_FORWARDED 1
# define IPSKB_XFRM_TUNNEL_SIZE 2
2006-01-06 23:04:54 -08:00
# define IPSKB_XFRM_TRANSFORMED 4
# define IPSKB_FRAG_COMPLETE 8
2006-02-15 15:10:22 -08:00
# define IPSKB_REROUTED 16
2005-04-16 15:20:36 -07:00
} ;
2007-03-12 20:09:15 -03:00
static inline unsigned int ip_hdrlen ( const struct sk_buff * skb )
{
2007-04-20 22:47:35 -07:00
return ip_hdr ( skb ) - > ihl * 4 ;
2007-03-12 20:09:15 -03:00
}
2009-11-03 03:26:03 +00:00
struct ipcm_cookie {
2006-09-27 18:28:28 -07:00
__be32 addr ;
2005-04-16 15:20:36 -07:00
int oif ;
struct ip_options * opt ;
2009-02-12 05:03:39 +00:00
union skb_shared_tx shtx ;
2005-04-16 15:20:36 -07:00
} ;
# define IPCB(skb) ((struct inet_skb_parm*)((skb)->cb))
2009-11-03 03:26:03 +00:00
struct ip_ra_chain {
2005-04-16 15:20:36 -07:00
struct ip_ra_chain * next ;
struct sock * sk ;
void ( * destructor ) ( struct sock * ) ;
} ;
extern struct ip_ra_chain * ip_ra_chain ;
extern rwlock_t ip_ra_lock ;
/* IP flags. */
# define IP_CE 0x8000 /* Flag: "Congestion" */
# define IP_DF 0x4000 /* Flag: "Don't Fragment" */
# define IP_MF 0x2000 /* Flag: "More Fragments" */
# define IP_OFFSET 0x1FFF /* "Fragment Offset" part */
# define IP_FRAG_TIME (30 * HZ) /* fragment lifetime */
2005-12-27 02:43:12 -02:00
struct msghdr ;
struct net_device ;
struct packet_type ;
struct rtable ;
struct sockaddr ;
2005-04-16 15:20:36 -07:00
extern int igmp_mc_proc_init ( void ) ;
/*
* Functions provided by ip . c
*/
extern int ip_build_and_send_pkt ( struct sk_buff * skb , struct sock * sk ,
2006-09-26 22:27:30 -07:00
__be32 saddr , __be32 daddr ,
2005-04-16 15:20:36 -07:00
struct ip_options * opt ) ;
extern int ip_rcv ( struct sk_buff * skb , struct net_device * dev ,
2005-08-09 19:34:12 -07:00
struct packet_type * pt , struct net_device * orig_dev ) ;
2005-04-16 15:20:36 -07:00
extern int ip_local_deliver ( struct sk_buff * skb ) ;
extern int ip_mr_input ( struct sk_buff * skb ) ;
extern int ip_output ( struct sk_buff * skb ) ;
extern int ip_mc_output ( struct sk_buff * skb ) ;
2006-04-04 13:42:35 -07:00
extern int ip_fragment ( struct sk_buff * skb , int ( * output ) ( struct sk_buff * ) ) ;
2005-04-16 15:20:36 -07:00
extern int ip_do_nat ( struct sk_buff * skb ) ;
extern void ip_send_check ( struct iphdr * ip ) ;
2008-01-11 19:14:00 -08:00
extern int __ip_local_out ( struct sk_buff * skb ) ;
extern int ip_local_out ( struct sk_buff * skb ) ;
2010-04-15 16:43:08 +00:00
extern int ip_queue_xmit ( struct sk_buff * skb ) ;
2005-04-16 15:20:36 -07:00
extern void ip_init ( void ) ;
extern int ip_append_data ( struct sock * sk ,
int getfrag ( void * from , char * to , int offset , int len ,
int odd , struct sk_buff * skb ) ,
void * from , int len , int protolen ,
struct ipcm_cookie * ipc ,
2008-11-24 15:52:46 -08:00
struct rtable * * rt ,
2005-04-16 15:20:36 -07:00
unsigned int flags ) ;
extern int ip_generic_getfrag ( void * from , char * to , int offset , int len , int odd , struct sk_buff * skb ) ;
extern ssize_t ip_append_page ( struct sock * sk , struct page * page ,
int offset , size_t size , int flags ) ;
extern int ip_push_pending_frames ( struct sock * sk ) ;
extern void ip_flush_pending_frames ( struct sock * sk ) ;
/* datagram.c */
extern int ip4_datagram_connect ( struct sock * sk ,
struct sockaddr * uaddr , int addr_len ) ;
/*
* Map a multicast IP onto multicast MAC for type Token Ring .
* This conforms to RFC1469 Option 2 Multicasting i . e .
* using a functional address to transmit / receive
* multicast packets .
*/
2006-11-14 20:51:49 -08:00
static inline void ip_tr_mc_map ( __be32 addr , char * buf )
2005-04-16 15:20:36 -07:00
{
buf [ 0 ] = 0xC0 ;
buf [ 1 ] = 0x00 ;
buf [ 2 ] = 0x00 ;
buf [ 3 ] = 0x04 ;
buf [ 4 ] = 0x00 ;
buf [ 5 ] = 0x00 ;
}
struct ip_reply_arg {
struct kvec iov [ 1 ] ;
2008-10-01 07:41:00 -07:00
int flags ;
2006-11-14 21:26:08 -08:00
__wsum csum ;
2005-04-16 15:20:36 -07:00
int csumoffset ; /* u16 offset of csum in iov[0].iov_base */
/* -1 if not needed */
2007-06-04 21:32:46 -07:00
int bound_dev_if ;
2005-04-16 15:20:36 -07:00
} ;
2008-10-01 07:41:00 -07:00
# define IP_REPLY_ARG_NOSRCCHECK 1
2008-10-01 07:44:42 -07:00
static inline __u8 ip_reply_arg_flowi_flags ( const struct ip_reply_arg * arg )
{
return ( arg - > flags & IP_REPLY_ARG_NOSRCCHECK ) ? FLOWI_FLAG_ANYSRC : 0 ;
}
2005-04-16 15:20:36 -07:00
void ip_send_reply ( struct sock * sk , struct sk_buff * skb , struct ip_reply_arg * arg ,
unsigned int len ) ;
2009-11-03 03:26:03 +00:00
struct ipv4_config {
2005-04-16 15:20:36 -07:00
int log_martians ;
int no_pmtu_disc ;
} ;
extern struct ipv4_config ipv4_config ;
2008-07-18 04:02:42 -07:00
# define IP_INC_STATS(net, field) SNMP_INC_STATS((net)->mib.ip_statistics, field)
# define IP_INC_STATS_BH(net, field) SNMP_INC_STATS_BH((net)->mib.ip_statistics, field)
2009-04-27 02:45:02 -07:00
# define IP_ADD_STATS(net, field, val) SNMP_ADD_STATS((net)->mib.ip_statistics, field, val)
2008-07-18 04:02:42 -07:00
# define IP_ADD_STATS_BH(net, field, val) SNMP_ADD_STATS_BH((net)->mib.ip_statistics, field, val)
2009-04-27 02:45:02 -07:00
# define IP_UPD_PO_STATS(net, field, val) SNMP_UPD_PO_STATS((net)->mib.ip_statistics, field, val)
# define IP_UPD_PO_STATS_BH(net, field, val) SNMP_UPD_PO_STATS_BH((net)->mib.ip_statistics, field, val)
2008-07-18 04:03:08 -07:00
# define NET_INC_STATS(net, field) SNMP_INC_STATS((net)->mib.net_statistics, field)
# define NET_INC_STATS_BH(net, field) SNMP_INC_STATS_BH((net)->mib.net_statistics, field)
# define NET_INC_STATS_USER(net, field) SNMP_INC_STATS_USER((net)->mib.net_statistics, field)
# define NET_ADD_STATS_BH(net, field, adnd) SNMP_ADD_STATS_BH((net)->mib.net_statistics, field, adnd)
# define NET_ADD_STATS_USER(net, field, adnd) SNMP_ADD_STATS_USER((net)->mib.net_statistics, field, adnd)
2005-04-16 15:20:36 -07:00
2010-02-16 15:20:26 +00:00
extern unsigned long snmp_fold_field ( void __percpu * mib [ ] , int offt ) ;
extern int snmp_mib_init ( void __percpu * ptr [ 2 ] , size_t mibsize ) ;
extern void snmp_mib_free ( void __percpu * ptr [ 2 ] ) ;
2007-04-20 15:57:15 -07:00
2008-10-08 14:18:04 -07:00
extern struct local_ports {
seqlock_t lock ;
int range [ 2 ] ;
} sysctl_local_ports ;
2007-10-10 17:30:46 -07:00
extern void inet_get_local_port_range ( int * low , int * high ) ;
2005-04-16 15:20:36 -07:00
extern int sysctl_ip_default_ttl ;
2005-06-13 15:12:33 -07:00
extern int sysctl_ip_nonlocal_bind ;
2005-04-16 15:20:36 -07:00
2008-11-25 18:00:48 -08:00
extern struct ctl_path net_core_path [ ] ;
2008-01-09 00:33:11 -08:00
extern struct ctl_path net_ipv4_ctl_path [ ] ;
2005-08-16 02:18:02 -03:00
/* From inetpeer.c */
extern int inet_peer_threshold ;
extern int inet_peer_minttl ;
extern int inet_peer_maxttl ;
extern int inet_peer_gc_mintime ;
extern int inet_peer_gc_maxtime ;
/* From ip_output.c */
extern int sysctl_ip_dynaddr ;
extern void ipfrag_init ( void ) ;
2008-07-15 16:00:59 -04:00
extern void ip_static_sysctl_init ( void ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_INET
2005-12-27 02:43:12 -02:00
# include <net/dst.h>
2005-04-16 15:20:36 -07:00
/* The function in 2.2 was invalid, producing wrong result for
* check = 0xFEFF . It was noticed by Arthur Skawina _year_ ago . - - ANK ( 000625 ) */
static inline
int ip_decrease_ttl ( struct iphdr * iph )
{
2006-11-14 21:42:26 -08:00
u32 check = ( __force u32 ) iph - > check ;
check + = ( __force u32 ) htons ( 0x0100 ) ;
iph - > check = ( __force __sum16 ) ( check + ( check > = 0xFFFF ) ) ;
2005-04-16 15:20:36 -07:00
return - - iph - > ttl ;
}
static inline
int ip_dont_fragment ( struct sock * sk , struct dst_entry * dst )
{
return ( inet_sk ( sk ) - > pmtudisc = = IP_PMTUDISC_DO | |
( inet_sk ( sk ) - > pmtudisc = = IP_PMTUDISC_WANT & &
2008-05-04 22:12:43 -07:00
! ( dst_metric_locked ( dst , RTAX_MTU ) ) ) ) ;
2005-04-16 15:20:36 -07:00
}
extern void __ip_select_ident ( struct iphdr * iph , struct dst_entry * dst , int more ) ;
static inline void ip_select_ident ( struct iphdr * iph , struct dst_entry * dst , struct sock * sk )
{
if ( iph - > frag_off & htons ( IP_DF ) ) {
/* This is only to work around buggy Windows95/2000
* VJ compression implementations . If the ID field
* does not change , they drop every other packet in
* a TCP stream using header compression .
*/
2009-10-15 06:30:45 +00:00
iph - > id = ( sk & & inet_sk ( sk ) - > inet_daddr ) ?
htons ( inet_sk ( sk ) - > inet_id + + ) : 0 ;
2005-04-16 15:20:36 -07:00
} else
__ip_select_ident ( iph , dst , 0 ) ;
}
static inline void ip_select_ident_more ( struct iphdr * iph , struct dst_entry * dst , struct sock * sk , int more )
{
if ( iph - > frag_off & htons ( IP_DF ) ) {
2009-10-15 06:30:45 +00:00
if ( sk & & inet_sk ( sk ) - > inet_daddr ) {
iph - > id = htons ( inet_sk ( sk ) - > inet_id ) ;
inet_sk ( sk ) - > inet_id + = 1 + more ;
2005-04-16 15:20:36 -07:00
} else
iph - > id = 0 ;
} else
__ip_select_ident ( iph , dst , more ) ;
}
/*
* Map a multicast IP onto multicast MAC for type ethernet .
*/
2006-11-14 20:51:49 -08:00
static inline void ip_eth_mc_map ( __be32 naddr , char * buf )
2005-04-16 15:20:36 -07:00
{
2006-11-14 20:51:49 -08:00
__u32 addr = ntohl ( naddr ) ;
2005-04-16 15:20:36 -07:00
buf [ 0 ] = 0x01 ;
buf [ 1 ] = 0x00 ;
buf [ 2 ] = 0x5e ;
buf [ 5 ] = addr & 0xFF ;
addr > > = 8 ;
buf [ 4 ] = addr & 0xFF ;
addr > > = 8 ;
buf [ 3 ] = addr & 0x7F ;
}
/*
* Map a multicast IP onto multicast MAC for type IP - over - InfiniBand .
* Leave P_Key as 0 to be filled in by driver .
*/
2007-12-10 13:38:41 -07:00
static inline void ip_ib_mc_map ( __be32 naddr , const unsigned char * broadcast , char * buf )
2005-04-16 15:20:36 -07:00
{
2006-11-14 20:51:49 -08:00
__u32 addr ;
2007-12-10 13:38:41 -07:00
unsigned char scope = broadcast [ 5 ] & 0xF ;
2005-04-16 15:20:36 -07:00
buf [ 0 ] = 0 ; /* Reserved */
buf [ 1 ] = 0xff ; /* Multicast QPN */
buf [ 2 ] = 0xff ;
buf [ 3 ] = 0xff ;
2006-11-14 20:51:49 -08:00
addr = ntohl ( naddr ) ;
2005-04-16 15:20:36 -07:00
buf [ 4 ] = 0xff ;
2007-12-10 13:38:41 -07:00
buf [ 5 ] = 0x10 | scope ; /* scope from broadcast address */
2005-04-16 15:20:36 -07:00
buf [ 6 ] = 0x40 ; /* IPv4 signature */
buf [ 7 ] = 0x1b ;
2007-12-10 13:38:41 -07:00
buf [ 8 ] = broadcast [ 8 ] ; /* P_Key */
buf [ 9 ] = broadcast [ 9 ] ;
2005-04-16 15:20:36 -07:00
buf [ 10 ] = 0 ;
buf [ 11 ] = 0 ;
buf [ 12 ] = 0 ;
buf [ 13 ] = 0 ;
buf [ 14 ] = 0 ;
buf [ 15 ] = 0 ;
buf [ 19 ] = addr & 0xff ;
addr > > = 8 ;
buf [ 18 ] = addr & 0xff ;
addr > > = 8 ;
buf [ 17 ] = addr & 0xff ;
addr > > = 8 ;
buf [ 16 ] = addr & 0x0f ;
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
# include <linux/ipv6.h>
# endif
static __inline__ void inet_reset_saddr ( struct sock * sk )
{
2009-10-15 06:30:45 +00:00
inet_sk ( sk ) - > inet_rcv_saddr = inet_sk ( sk ) - > inet_saddr = 0 ;
2005-04-16 15:20:36 -07:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
if ( sk - > sk_family = = PF_INET6 ) {
struct ipv6_pinfo * np = inet6_sk ( sk ) ;
memset ( & np - > saddr , 0 , sizeof ( np - > saddr ) ) ;
memset ( & np - > rcv_saddr , 0 , sizeof ( np - > rcv_saddr ) ) ;
}
# endif
}
# endif
2010-01-06 20:37:01 -08:00
static inline int sk_mc_loop ( struct sock * sk )
{
if ( ! sk )
return 1 ;
switch ( sk - > sk_family ) {
case AF_INET :
return inet_sk ( sk ) - > mc_loop ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
case AF_INET6 :
return inet6_sk ( sk ) - > mc_loop ;
# endif
}
2010-01-13 18:10:36 -08:00
WARN_ON ( 1 ) ;
2010-01-06 20:37:01 -08:00
return 1 ;
}
2005-04-16 15:20:36 -07:00
extern int ip_call_ra_chain ( struct sk_buff * skb ) ;
/*
2008-01-01 21:13:09 -08:00
* Functions provided by ip_fragment . c
2005-04-16 15:20:36 -07:00
*/
2009-11-03 03:26:03 +00:00
enum ip_defrag_users {
2005-04-16 15:20:36 -07:00
IP_DEFRAG_LOCAL_DELIVER ,
IP_DEFRAG_CALL_RA_CHAIN ,
IP_DEFRAG_CONNTRACK_IN ,
2010-02-15 18:13:33 +01:00
__IP_DEFRAG_CONNTRACK_IN_END = IP_DEFRAG_CONNTRACK_IN + USHORT_MAX ,
2005-04-16 15:20:36 -07:00
IP_DEFRAG_CONNTRACK_OUT ,
2010-02-15 18:13:33 +01:00
__IP_DEFRAG_CONNTRACK_OUT_END = IP_DEFRAG_CONNTRACK_OUT + USHORT_MAX ,
2009-12-15 16:59:59 +01:00
IP_DEFRAG_CONNTRACK_BRIDGE_IN ,
2010-02-15 18:13:33 +01:00
__IP_DEFRAG_CONNTRACK_BRIDGE_IN = IP_DEFRAG_CONNTRACK_BRIDGE_IN + USHORT_MAX ,
2005-04-16 15:20:36 -07:00
IP_DEFRAG_VS_IN ,
IP_DEFRAG_VS_OUT ,
IP_DEFRAG_VS_FWD
} ;
2007-10-14 00:38:32 -07:00
int ip_defrag ( struct sk_buff * skb , u32 user ) ;
2008-01-22 06:07:25 -08:00
int ip_frag_mem ( struct net * net ) ;
2008-01-22 06:06:23 -08:00
int ip_frag_nqueues ( struct net * net ) ;
2005-04-16 15:20:36 -07:00
/*
* Functions provided by ip_forward . c
*/
extern int ip_forward ( struct sk_buff * skb ) ;
/*
* Functions provided by ip_options . c
*/
2006-09-26 22:27:05 -07:00
extern void ip_options_build ( struct sk_buff * skb , struct ip_options * opt , __be32 daddr , struct rtable * rt , int is_frag ) ;
2005-04-16 15:20:36 -07:00
extern int ip_options_echo ( struct ip_options * dopt , struct sk_buff * skb ) ;
extern void ip_options_fragment ( struct sk_buff * skb ) ;
2008-03-24 15:29:23 -07:00
extern int ip_options_compile ( struct net * net ,
struct ip_options * opt , struct sk_buff * skb ) ;
2008-03-24 15:29:55 -07:00
extern int ip_options_get ( struct net * net , struct ip_options * * optp ,
2005-08-16 19:46:48 -03:00
unsigned char * data , int optlen ) ;
2008-03-24 15:29:55 -07:00
extern int ip_options_get_from_user ( struct net * net , struct ip_options * * optp ,
2005-08-16 19:46:48 -03:00
unsigned char __user * data , int optlen ) ;
2005-04-16 15:20:36 -07:00
extern void ip_options_undo ( struct ip_options * opt ) ;
extern void ip_forward_options ( struct sk_buff * skb ) ;
extern int ip_options_rcv_srr ( struct sk_buff * skb ) ;
/*
* Functions provided by ip_sockglue . c
*/
2010-04-28 15:31:51 -07:00
extern int ip_queue_rcv_skb ( struct sock * sk , struct sk_buff * skb ) ;
2005-04-16 15:20:36 -07:00
extern void ip_cmsg_recv ( struct msghdr * msg , struct sk_buff * skb ) ;
2008-03-24 15:30:27 -07:00
extern int ip_cmsg_send ( struct net * net ,
struct msghdr * msg , struct ipcm_cookie * ipc ) ;
2009-09-30 16:12:20 -07:00
extern int ip_setsockopt ( struct sock * sk , int level , int optname , char __user * optval , unsigned int optlen ) ;
2005-04-16 15:20:36 -07:00
extern int ip_getsockopt ( struct sock * sk , int level , int optname , char __user * optval , int __user * optlen ) ;
2006-03-20 22:45:21 -08:00
extern int compat_ip_setsockopt ( struct sock * sk , int level ,
2009-09-30 16:12:20 -07:00
int optname , char __user * optval , unsigned int optlen ) ;
2006-03-20 22:45:21 -08:00
extern int compat_ip_getsockopt ( struct sock * sk , int level ,
int optname , char __user * optval , int __user * optlen ) ;
2005-04-16 15:20:36 -07:00
extern int ip_ra_control ( struct sock * sk , unsigned char on , void ( * destructor ) ( struct sock * ) ) ;
extern int ip_recv_error ( struct sock * sk , struct msghdr * msg , int len ) ;
extern void ip_icmp_error ( struct sock * sk , struct sk_buff * skb , int err ,
2006-09-27 18:34:21 -07:00
__be16 port , u32 info , u8 * payload ) ;
2006-09-27 18:33:40 -07:00
extern void ip_local_error ( struct sock * sk , int err , __be32 daddr , __be16 dport ,
2005-04-16 15:20:36 -07:00
u32 info ) ;
/* sysctl helpers - any sysctl which holds a value that ends up being
* fed into the routing cache should use these handlers .
*/
int ipv4_doint_and_flush ( ctl_table * ctl , int write ,
2009-09-23 15:57:19 -07:00
void __user * buffer ,
2005-04-16 15:20:36 -07:00
size_t * lenp , loff_t * ppos ) ;
2008-10-15 22:04:23 -07:00
int ipv4_doint_and_flush_strategy ( ctl_table * table ,
2005-04-16 15:20:36 -07:00
void __user * oldval , size_t __user * oldlenp ,
2006-12-10 02:19:10 -08:00
void __user * newval , size_t newlen ) ;
2005-08-16 02:18:02 -03:00
# ifdef CONFIG_PROC_FS
extern int ip_misc_proc_init ( void ) ;
# endif
2005-04-16 15:20:36 -07:00
# endif /* _IP_H */