2015-07-21 11:43:46 +03:00
# ifndef __NET_LWTUNNEL_H
# define __NET_LWTUNNEL_H 1
# include <linux/lwtunnel.h>
# include <linux/netdevice.h>
# include <linux/skbuff.h>
# include <linux/types.h>
# include <net/route.h>
# define LWTUNNEL_HASH_BITS 7
# define LWTUNNEL_HASH_SIZE (1 << LWTUNNEL_HASH_BITS)
/* lw tunnel state flags */
2015-08-17 23:42:24 +03:00
# define LWTUNNEL_STATE_OUTPUT_REDIRECT BIT(0)
# define LWTUNNEL_STATE_INPUT_REDIRECT BIT(1)
2015-07-21 11:43:46 +03:00
struct lwtunnel_state {
__u16 type ;
__u16 flags ;
atomic_t refcnt ;
2015-08-17 23:42:24 +03:00
int ( * orig_output ) ( struct sock * sk , struct sk_buff * skb ) ;
int ( * orig_input ) ( struct sk_buff * ) ;
2015-07-21 11:43:46 +03:00
int len ;
__u8 data [ 0 ] ;
} ;
struct lwtunnel_encap_ops {
int ( * build_state ) ( struct net_device * dev , struct nlattr * encap ,
struct lwtunnel_state * * ts ) ;
int ( * output ) ( struct sock * sk , struct sk_buff * skb ) ;
2015-08-17 23:42:24 +03:00
int ( * input ) ( struct sk_buff * skb ) ;
2015-07-21 11:43:46 +03:00
int ( * fill_encap ) ( struct sk_buff * skb ,
struct lwtunnel_state * lwtstate ) ;
int ( * get_encap_size ) ( struct lwtunnel_state * lwtstate ) ;
int ( * cmp_encap ) ( struct lwtunnel_state * a , struct lwtunnel_state * b ) ;
} ;
# ifdef CONFIG_LWTUNNEL
2015-08-18 19:41:13 +03:00
static inline void lwtstate_free ( struct lwtunnel_state * lws )
{
kfree ( lws ) ;
}
2015-07-24 13:28:36 +03:00
static inline struct lwtunnel_state *
lwtstate_get ( struct lwtunnel_state * lws )
2015-07-21 11:43:46 +03:00
{
2015-07-24 13:28:36 +03:00
if ( lws )
atomic_inc ( & lws - > refcnt ) ;
return lws ;
2015-07-21 11:43:46 +03:00
}
2015-07-24 13:28:36 +03:00
static inline void lwtstate_put ( struct lwtunnel_state * lws )
2015-07-21 11:43:46 +03:00
{
if ( ! lws )
return ;
if ( atomic_dec_and_test ( & lws - > refcnt ) )
2015-08-18 19:41:13 +03:00
lwtstate_free ( lws ) ;
2015-07-21 11:43:46 +03:00
}
static inline bool lwtunnel_output_redirect ( struct lwtunnel_state * lwtstate )
{
if ( lwtstate & & ( lwtstate - > flags & LWTUNNEL_STATE_OUTPUT_REDIRECT ) )
return true ;
return false ;
}
2015-08-17 23:42:24 +03:00
static inline bool lwtunnel_input_redirect ( struct lwtunnel_state * lwtstate )
{
if ( lwtstate & & ( lwtstate - > flags & LWTUNNEL_STATE_INPUT_REDIRECT ) )
return true ;
return false ;
}
2015-07-21 11:43:46 +03:00
int lwtunnel_encap_add_ops ( const struct lwtunnel_encap_ops * op ,
unsigned int num ) ;
int lwtunnel_encap_del_ops ( const struct lwtunnel_encap_ops * op ,
unsigned int num ) ;
int lwtunnel_build_state ( struct net_device * dev , u16 encap_type ,
struct nlattr * encap ,
struct lwtunnel_state * * lws ) ;
int lwtunnel_fill_encap ( struct sk_buff * skb ,
struct lwtunnel_state * lwtstate ) ;
int lwtunnel_get_encap_size ( struct lwtunnel_state * lwtstate ) ;
struct lwtunnel_state * lwtunnel_state_alloc ( int hdr_len ) ;
int lwtunnel_cmp_encap ( struct lwtunnel_state * a , struct lwtunnel_state * b ) ;
2015-07-21 11:43:49 +03:00
int lwtunnel_output ( struct sock * sk , struct sk_buff * skb ) ;
int lwtunnel_output6 ( struct sock * sk , struct sk_buff * skb ) ;
2015-08-17 23:42:24 +03:00
int lwtunnel_input ( struct sk_buff * skb ) ;
int lwtunnel_input6 ( struct sk_buff * skb ) ;
2015-07-21 11:43:46 +03:00
# else
2015-07-24 13:28:36 +03:00
static inline struct lwtunnel_state *
lwtstate_get ( struct lwtunnel_state * lws )
2015-07-21 11:43:46 +03:00
{
2015-07-24 13:28:36 +03:00
return lws ;
2015-07-21 11:43:46 +03:00
}
2015-07-24 13:28:36 +03:00
static inline void lwtstate_put ( struct lwtunnel_state * lws )
2015-07-21 11:43:46 +03:00
{
}
static inline bool lwtunnel_output_redirect ( struct lwtunnel_state * lwtstate )
{
return false ;
}
2015-08-17 23:42:24 +03:00
static inline bool lwtunnel_input_redirect ( struct lwtunnel_state * lwtstate )
{
return false ;
}
2015-07-21 11:43:46 +03:00
static inline int lwtunnel_encap_add_ops ( const struct lwtunnel_encap_ops * op ,
unsigned int num )
{
return - EOPNOTSUPP ;
}
static inline int lwtunnel_encap_del_ops ( const struct lwtunnel_encap_ops * op ,
unsigned int num )
{
return - EOPNOTSUPP ;
}
static inline int lwtunnel_build_state ( struct net_device * dev , u16 encap_type ,
struct nlattr * encap ,
struct lwtunnel_state * * lws )
{
return - EOPNOTSUPP ;
}
static inline int lwtunnel_fill_encap ( struct sk_buff * skb ,
struct lwtunnel_state * lwtstate )
{
return 0 ;
}
static inline int lwtunnel_get_encap_size ( struct lwtunnel_state * lwtstate )
{
return 0 ;
}
static inline struct lwtunnel_state * lwtunnel_state_alloc ( int hdr_len )
{
return NULL ;
}
static inline int lwtunnel_cmp_encap ( struct lwtunnel_state * a ,
struct lwtunnel_state * b )
{
return 0 ;
}
2015-07-21 11:43:49 +03:00
static inline int lwtunnel_output ( struct sock * sk , struct sk_buff * skb )
{
return - EOPNOTSUPP ;
}
static inline int lwtunnel_output6 ( struct sock * sk , struct sk_buff * skb )
{
return - EOPNOTSUPP ;
}
2015-08-17 23:42:24 +03:00
static inline int lwtunnel_input ( struct sk_buff * skb )
{
return - EOPNOTSUPP ;
}
static inline int lwtunnel_input6 ( struct sk_buff * skb )
{
return - EOPNOTSUPP ;
}
2015-07-21 11:43:46 +03:00
# endif
# endif /* __NET_LWTUNNEL_H */