2005-12-27 02:43:12 -02: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 inet_sock
*
* Authors : Many , reorganised here by
* Arnaldo Carvalho de Melo < acme @ mandriva . com >
*
* 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 _INET_SOCK_H
# define _INET_SOCK_H
2008-09-09 06:43:12 +02:00
# include <linux/kmemcheck.h>
2005-12-27 02:43:12 -02:00
# include <linux/string.h>
# include <linux/types.h>
2007-03-23 11:40:27 -07:00
# include <linux/jhash.h>
2005-12-27 02:43:12 -02:00
# include <net/flow.h>
# include <net/sock.h>
# include <net/request_sock.h>
2008-06-16 17:14:11 -07:00
# include <net/netns/hash.h>
2005-12-27 02:43:12 -02:00
/** struct ip_options - IP Options
*
* @ faddr - Saved first hop address
* @ is_data - Options in __data , rather than skb
* @ is_strictroute - Strict source route
* @ srr_is_hit - Packet destination addr was our one
* @ is_changed - IP checksum more not valid
* @ rr_needaddr - Need to record addr of outgoing dev
* @ ts_needtime - Need to record timestamp
* @ ts_needaddr - Need to record addr of outgoing dev
*/
struct ip_options {
2006-09-27 18:28:07 -07:00
__be32 faddr ;
2005-12-27 02:43:12 -02:00
unsigned char optlen ;
unsigned char srr ;
unsigned char rr ;
unsigned char ts ;
2008-03-22 16:35:29 -07:00
unsigned char is_strictroute : 1 ,
2005-12-27 02:43:12 -02:00
srr_is_hit : 1 ,
is_changed : 1 ,
rr_needaddr : 1 ,
ts_needtime : 1 ,
ts_needaddr : 1 ;
unsigned char router_alert ;
2006-08-03 16:46:20 -07:00
unsigned char cipso ;
2005-12-27 02:43:12 -02:00
unsigned char __pad2 ;
unsigned char __data [ 0 ] ;
} ;
# define optlength(opt) (sizeof(struct ip_options) + opt->optlen)
struct inet_request_sock {
struct request_sock req ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
u16 inet6_rsk_offset ;
# endif
2008-10-01 07:46:49 -07:00
__be16 loc_port ;
2006-09-27 18:27:13 -07:00
__be32 loc_addr ;
__be32 rmt_addr ;
2006-09-27 18:35:29 -07:00
__be16 rmt_port ;
2008-09-09 06:43:12 +02:00
kmemcheck_bitfield_begin ( flags ) ;
u16 snd_wscale : 4 ,
rcv_wscale : 4 ,
2005-12-27 02:43:12 -02:00
tstamp_ok : 1 ,
sack_ok : 1 ,
wscale_ok : 1 ,
ecn_ok : 1 ,
2008-10-01 07:41:00 -07:00
acked : 1 ,
no_srccheck : 1 ;
2008-09-09 06:43:12 +02:00
kmemcheck_bitfield_end ( flags ) ;
2005-12-27 02:43:12 -02:00
struct ip_options * opt ;
} ;
static inline struct inet_request_sock * inet_rsk ( const struct request_sock * sk )
{
return ( struct inet_request_sock * ) sk ;
}
struct ip_mc_socklist ;
struct ipv6_pinfo ;
struct rtable ;
/** struct inet_sock - representation of INET sockets
*
* @ sk - ancestor class
* @ pinet6 - pointer to IPv6 control block
2009-10-15 06:30:45 +00:00
* @ inet_daddr - Foreign IPv4 addr
* @ inet_rcv_saddr - Bound local IPv4 addr
* @ inet_dport - Destination port
* @ inet_num - Local port
* @ inet_saddr - Sending source
2005-12-27 02:43:12 -02:00
* @ uc_ttl - Unicast TTL
2009-10-15 06:30:45 +00:00
* @ inet_sport - Source port
* @ inet_id - ID counter for DF pkts
2005-12-27 02:43:12 -02:00
* @ tos - TOS
* @ mc_ttl - Multicasting TTL
* @ is_icsk - is this an inet_connection_sock ?
* @ mc_index - Multicast device index
* @ mc_list - Group array
* @ cork - info to build ip hdr on each ip frag while socket is corked
*/
struct inet_sock {
/* sk and pinet6 has to be the first two members of inet_sock */
struct sock sk ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
struct ipv6_pinfo * pinet6 ;
# endif
/* Socket demultiplex comparisons on incoming packets. */
2009-10-15 06:30:45 +00:00
__be32 inet_daddr ;
__be32 inet_rcv_saddr ;
__be16 inet_dport ;
__u16 inet_num ;
__be32 inet_saddr ;
2005-12-27 02:43:12 -02:00
__s16 uc_ttl ;
__u16 cmsg_flags ;
2009-10-15 06:30:45 +00:00
__be16 inet_sport ;
__u16 inet_id ;
2010-01-11 16:28:01 -08:00
struct ip_options * opt ;
2005-12-27 02:43:12 -02:00
__u8 tos ;
2010-01-11 16:28:01 -08:00
__u8 min_ttl ;
2005-12-27 02:43:12 -02:00
__u8 mc_ttl ;
__u8 pmtudisc ;
__u8 recverr : 1 ,
is_icsk : 1 ,
freebind : 1 ,
hdrincl : 1 ,
2008-10-01 07:30:02 -07:00
mc_loop : 1 ,
2009-05-28 07:00:46 +00:00
transparent : 1 ,
mc_all : 1 ;
2005-12-27 02:43:12 -02:00
int mc_index ;
2006-09-26 21:27:35 -07:00
__be32 mc_addr ;
2005-12-27 02:43:12 -02:00
struct ip_mc_socklist * mc_list ;
struct {
unsigned int flags ;
unsigned int fragsize ;
struct ip_options * opt ;
2008-03-10 04:30:37 -04:00
struct dst_entry * dst ;
2005-12-27 02:43:12 -02:00
int length ; /* Total length of all frames */
2006-09-26 21:27:35 -07:00
__be32 addr ;
2005-12-27 02:43:12 -02:00
struct flowi fl ;
} cork ;
} ;
# define IPCORK_OPT 1 /* ip-options has been held in ipcork.opt */
# define IPCORK_ALLFRAG 2 /* always fragment (for ipv6 for now) */
static inline struct inet_sock * inet_sk ( const struct sock * sk )
{
return ( struct inet_sock * ) sk ;
}
static inline void __inet_sk_copy_descendant ( struct sock * sk_to ,
const struct sock * sk_from ,
const int ancestor_size )
{
memcpy ( inet_sk ( sk_to ) + 1 , inet_sk ( sk_from ) + 1 ,
sk_from - > sk_prot - > obj_size - ancestor_size ) ;
}
# if !(defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE))
static inline void inet_sk_copy_descendant ( struct sock * sk_to ,
const struct sock * sk_from )
{
__inet_sk_copy_descendant ( sk_to , sk_from , sizeof ( struct inet_sock ) ) ;
}
# endif
extern int inet_sk_rebuild_header ( struct sock * sk ) ;
2007-03-23 11:40:27 -07:00
extern u32 inet_ehash_secret ;
extern void build_ehash_secret ( void ) ;
2008-06-16 17:13:27 -07:00
static inline unsigned int inet_ehashfn ( struct net * net ,
const __be32 laddr , const __u16 lport ,
2006-09-27 18:43:33 -07:00
const __be32 faddr , const __be16 fport )
2005-12-27 02:43:12 -02:00
{
2008-03-04 14:28:41 -08:00
return jhash_3words ( ( __force __u32 ) laddr ,
( __force __u32 ) faddr ,
2007-03-23 11:40:27 -07:00
( ( __u32 ) lport ) < < 16 | ( __force __u32 ) fport ,
2008-06-16 17:14:11 -07:00
inet_ehash_secret + net_hash_mix ( net ) ) ;
2005-12-27 02:43:12 -02:00
}
static inline int inet_sk_ehashfn ( const struct sock * sk )
{
const struct inet_sock * inet = inet_sk ( sk ) ;
2009-10-15 06:30:45 +00:00
const __be32 laddr = inet - > inet_rcv_saddr ;
const __u16 lport = inet - > inet_num ;
const __be32 faddr = inet - > inet_daddr ;
const __be16 fport = inet - > inet_dport ;
2008-06-16 17:13:27 -07:00
struct net * net = sock_net ( sk ) ;
2005-12-27 02:43:12 -02:00
2008-06-16 17:13:27 -07:00
return inet_ehashfn ( net , laddr , lport , faddr , fport ) ;
2005-12-27 02:43:12 -02:00
}
2008-06-10 12:39:35 -07:00
static inline struct request_sock * inet_reqsk_alloc ( struct request_sock_ops * ops )
{
struct request_sock * req = reqsk_alloc ( ops ) ;
2008-09-09 06:43:12 +02:00
struct inet_request_sock * ireq = inet_rsk ( req ) ;
2008-06-10 12:39:35 -07:00
2008-09-09 06:43:12 +02:00
if ( req ! = NULL ) {
kmemcheck_annotate_bitfield ( ireq , flags ) ;
ireq - > opt = NULL ;
}
2008-06-10 12:39:35 -07:00
return req ;
}
2008-10-01 07:41:00 -07:00
static inline __u8 inet_sk_flowi_flags ( const struct sock * sk )
{
return inet_sk ( sk ) - > transparent ? FLOWI_FLAG_ANYSRC : 0 ;
}
2005-12-27 02:43:12 -02:00
# endif /* _INET_SOCK_H */