2019-02-02 12:50:43 +01:00
# ifndef _NET_FLOW_OFFLOAD_H
# define _NET_FLOW_OFFLOAD_H
2019-05-31 22:47:21 +01:00
# include <linux/kernel.h>
2019-07-19 18:20:15 +02:00
# include <linux/list.h>
2019-02-02 12:50:43 +01:00
# include <net/flow_dissector.h>
2019-08-07 09:13:52 +08:00
# include <linux/rhashtable.h>
2019-02-02 12:50:43 +01:00
struct flow_match {
struct flow_dissector * dissector ;
void * mask ;
void * key ;
} ;
2019-06-19 09:41:04 +03:00
struct flow_match_meta {
struct flow_dissector_key_meta * key , * mask ;
} ;
2019-02-02 12:50:43 +01:00
struct flow_match_basic {
struct flow_dissector_key_basic * key , * mask ;
} ;
struct flow_match_control {
struct flow_dissector_key_control * key , * mask ;
} ;
struct flow_match_eth_addrs {
struct flow_dissector_key_eth_addrs * key , * mask ;
} ;
struct flow_match_vlan {
struct flow_dissector_key_vlan * key , * mask ;
} ;
struct flow_match_ipv4_addrs {
struct flow_dissector_key_ipv4_addrs * key , * mask ;
} ;
struct flow_match_ipv6_addrs {
struct flow_dissector_key_ipv6_addrs * key , * mask ;
} ;
struct flow_match_ip {
struct flow_dissector_key_ip * key , * mask ;
} ;
struct flow_match_ports {
struct flow_dissector_key_ports * key , * mask ;
} ;
struct flow_match_icmp {
struct flow_dissector_key_icmp * key , * mask ;
} ;
struct flow_match_tcp {
struct flow_dissector_key_tcp * key , * mask ;
} ;
struct flow_match_mpls {
struct flow_dissector_key_mpls * key , * mask ;
} ;
struct flow_match_enc_keyid {
struct flow_dissector_key_keyid * key , * mask ;
} ;
struct flow_match_enc_opts {
struct flow_dissector_key_enc_opts * key , * mask ;
} ;
struct flow_rule ;
2019-06-19 09:41:04 +03:00
void flow_rule_match_meta ( const struct flow_rule * rule ,
struct flow_match_meta * out ) ;
2019-02-02 12:50:43 +01:00
void flow_rule_match_basic ( const struct flow_rule * rule ,
struct flow_match_basic * out ) ;
void flow_rule_match_control ( const struct flow_rule * rule ,
struct flow_match_control * out ) ;
void flow_rule_match_eth_addrs ( const struct flow_rule * rule ,
struct flow_match_eth_addrs * out ) ;
void flow_rule_match_vlan ( const struct flow_rule * rule ,
struct flow_match_vlan * out ) ;
2019-05-14 21:18:12 +01:00
void flow_rule_match_cvlan ( const struct flow_rule * rule ,
struct flow_match_vlan * out ) ;
2019-02-02 12:50:43 +01:00
void flow_rule_match_ipv4_addrs ( const struct flow_rule * rule ,
struct flow_match_ipv4_addrs * out ) ;
void flow_rule_match_ipv6_addrs ( const struct flow_rule * rule ,
struct flow_match_ipv6_addrs * out ) ;
void flow_rule_match_ip ( const struct flow_rule * rule ,
struct flow_match_ip * out ) ;
void flow_rule_match_ports ( const struct flow_rule * rule ,
struct flow_match_ports * out ) ;
void flow_rule_match_tcp ( const struct flow_rule * rule ,
struct flow_match_tcp * out ) ;
void flow_rule_match_icmp ( const struct flow_rule * rule ,
struct flow_match_icmp * out ) ;
void flow_rule_match_mpls ( const struct flow_rule * rule ,
struct flow_match_mpls * out ) ;
void flow_rule_match_enc_control ( const struct flow_rule * rule ,
struct flow_match_control * out ) ;
void flow_rule_match_enc_ipv4_addrs ( const struct flow_rule * rule ,
struct flow_match_ipv4_addrs * out ) ;
void flow_rule_match_enc_ipv6_addrs ( const struct flow_rule * rule ,
struct flow_match_ipv6_addrs * out ) ;
void flow_rule_match_enc_ip ( const struct flow_rule * rule ,
struct flow_match_ip * out ) ;
void flow_rule_match_enc_ports ( const struct flow_rule * rule ,
struct flow_match_ports * out ) ;
void flow_rule_match_enc_keyid ( const struct flow_rule * rule ,
struct flow_match_enc_keyid * out ) ;
void flow_rule_match_enc_opts ( const struct flow_rule * rule ,
struct flow_match_enc_opts * out ) ;
2019-02-02 12:50:45 +01:00
enum flow_action_id {
FLOW_ACTION_ACCEPT = 0 ,
FLOW_ACTION_DROP ,
FLOW_ACTION_TRAP ,
FLOW_ACTION_GOTO ,
FLOW_ACTION_REDIRECT ,
FLOW_ACTION_MIRRED ,
2019-08-04 16:09:06 +01:00
FLOW_ACTION_REDIRECT_INGRESS ,
FLOW_ACTION_MIRRED_INGRESS ,
2019-02-02 12:50:45 +01:00
FLOW_ACTION_VLAN_PUSH ,
FLOW_ACTION_VLAN_POP ,
FLOW_ACTION_VLAN_MANGLE ,
FLOW_ACTION_TUNNEL_ENCAP ,
FLOW_ACTION_TUNNEL_DECAP ,
FLOW_ACTION_MANGLE ,
FLOW_ACTION_ADD ,
FLOW_ACTION_CSUM ,
FLOW_ACTION_MARK ,
2019-08-04 16:09:04 +01:00
FLOW_ACTION_PTYPE ,
2019-02-02 12:50:50 +01:00
FLOW_ACTION_WAKE ,
FLOW_ACTION_QUEUE ,
2019-05-04 04:46:16 -07:00
FLOW_ACTION_SAMPLE ,
2019-05-04 04:46:22 -07:00
FLOW_ACTION_POLICE ,
2019-07-09 10:30:48 +03:00
FLOW_ACTION_CT ,
2019-07-23 15:33:59 +01:00
FLOW_ACTION_MPLS_PUSH ,
FLOW_ACTION_MPLS_POP ,
FLOW_ACTION_MPLS_MANGLE ,
2019-06-27 10:55:02 +00:00
NUM_FLOW_ACTIONS ,
2019-02-02 12:50:45 +01:00
} ;
/* This is mirroring enum pedit_header_type definition for easy mapping between
* tc pedit action . Legacy TCA_PEDIT_KEY_EX_HDR_TYPE_NETWORK is mapped to
* FLOW_ACT_MANGLE_UNSPEC , which is supported by no driver .
*/
enum flow_action_mangle_base {
FLOW_ACT_MANGLE_UNSPEC = 0 ,
FLOW_ACT_MANGLE_HDR_TYPE_ETH ,
FLOW_ACT_MANGLE_HDR_TYPE_IP4 ,
FLOW_ACT_MANGLE_HDR_TYPE_IP6 ,
FLOW_ACT_MANGLE_HDR_TYPE_TCP ,
FLOW_ACT_MANGLE_HDR_TYPE_UDP ,
} ;
2019-09-13 18:28:39 +03:00
typedef void ( * action_destr ) ( void * priv ) ;
2019-02-02 12:50:45 +01:00
struct flow_action_entry {
enum flow_action_id id ;
2019-09-13 18:28:39 +03:00
action_destr destructor ;
void * destructor_priv ;
2019-02-02 12:50:45 +01:00
union {
u32 chain_index ; /* FLOW_ACTION_GOTO */
struct net_device * dev ; /* FLOW_ACTION_REDIRECT */
struct { /* FLOW_ACTION_VLAN */
u16 vid ;
__be16 proto ;
u8 prio ;
} vlan ;
struct { /* FLOW_ACTION_PACKET_EDIT */
enum flow_action_mangle_base htype ;
u32 offset ;
u32 mask ;
u32 val ;
} mangle ;
2019-09-13 18:28:39 +03:00
struct ip_tunnel_info * tunnel ; /* FLOW_ACTION_TUNNEL_ENCAP */
2019-02-02 12:50:45 +01:00
u32 csum_flags ; /* FLOW_ACTION_CSUM */
u32 mark ; /* FLOW_ACTION_MARK */
2019-08-04 16:09:04 +01:00
u16 ptype ; /* FLOW_ACTION_PTYPE */
2019-02-02 12:50:50 +01:00
struct { /* FLOW_ACTION_QUEUE */
u32 ctx ;
u32 index ;
u8 vf ;
} queue ;
2019-05-04 04:46:16 -07:00
struct { /* FLOW_ACTION_SAMPLE */
struct psample_group * psample_group ;
u32 rate ;
u32 trunc_size ;
bool truncate ;
} sample ;
2019-05-04 04:46:22 -07:00
struct { /* FLOW_ACTION_POLICE */
s64 burst ;
u64 rate_bytes_ps ;
} police ;
2019-07-09 10:30:48 +03:00
struct { /* FLOW_ACTION_CT */
int action ;
u16 zone ;
} ct ;
2019-07-23 15:33:59 +01:00
struct { /* FLOW_ACTION_MPLS_PUSH */
u32 label ;
__be16 proto ;
u8 tc ;
u8 bos ;
u8 ttl ;
} mpls_push ;
struct { /* FLOW_ACTION_MPLS_POP */
__be16 proto ;
} mpls_pop ;
struct { /* FLOW_ACTION_MPLS_MANGLE */
u32 label ;
u8 tc ;
u8 bos ;
u8 ttl ;
} mpls_mangle ;
2019-02-02 12:50:45 +01:00
} ;
} ;
struct flow_action {
unsigned int num_entries ;
struct flow_action_entry entries [ 0 ] ;
} ;
static inline bool flow_action_has_entries ( const struct flow_action * action )
{
return action - > num_entries ;
}
2019-05-04 04:46:18 -07:00
/**
* flow_action_has_one_action ( ) - check if exactly one action is present
* @ action : tc filter flow offload action
*
* Returns true if exactly one action is present .
*/
static inline bool flow_offload_has_one_action ( const struct flow_action * action )
{
return action - > num_entries = = 1 ;
}
2019-02-02 12:50:45 +01:00
# define flow_action_for_each(__i, __act, __actions) \
2019-02-11 09:52:59 +02:00
for ( __i = 0 , __act = & ( __actions ) - > entries [ 0 ] ; __i < ( __actions ) - > num_entries ; __act = & ( __actions ) - > entries [ + + __i ] )
2019-02-02 12:50:45 +01:00
2019-02-02 12:50:43 +01:00
struct flow_rule {
struct flow_match match ;
2019-02-02 12:50:45 +01:00
struct flow_action action ;
2019-02-02 12:50:43 +01:00
} ;
2019-02-02 12:50:45 +01:00
struct flow_rule * flow_rule_alloc ( unsigned int num_actions ) ;
2019-02-02 12:50:43 +01:00
static inline bool flow_rule_match_key ( const struct flow_rule * rule ,
enum flow_dissector_key_id key )
{
return dissector_uses_key ( rule - > match . dissector , key ) ;
}
2019-02-02 12:50:47 +01:00
struct flow_stats {
u64 pkts ;
u64 bytes ;
u64 lastused ;
} ;
static inline void flow_stats_update ( struct flow_stats * flow_stats ,
u64 bytes , u64 pkts , u64 lastused )
{
2019-02-13 00:23:52 +00:00
flow_stats - > pkts + = pkts ;
flow_stats - > bytes + = bytes ;
flow_stats - > lastused = max_t ( u64 , flow_stats - > lastused , lastused ) ;
2019-02-02 12:50:47 +01:00
}
2019-07-09 22:55:39 +02:00
enum flow_block_command {
2019-07-09 22:55:40 +02:00
FLOW_BLOCK_BIND ,
FLOW_BLOCK_UNBIND ,
2019-07-09 22:55:39 +02:00
} ;
enum flow_block_binder_type {
2019-07-09 22:55:41 +02:00
FLOW_BLOCK_BINDER_TYPE_UNSPEC ,
FLOW_BLOCK_BINDER_TYPE_CLSACT_INGRESS ,
FLOW_BLOCK_BINDER_TYPE_CLSACT_EGRESS ,
2019-07-09 22:55:39 +02:00
} ;
2019-07-19 18:20:16 +02:00
struct flow_block {
struct list_head cb_list ;
} ;
2019-07-09 22:55:39 +02:00
struct netlink_ext_ack ;
struct flow_block_offload {
enum flow_block_command command ;
enum flow_block_binder_type binder_type ;
2019-07-09 22:55:46 +02:00
bool block_shared ;
2019-08-26 16:45:01 +03:00
bool unlocked_driver_cb ;
2019-07-09 22:55:43 +02:00
struct net * net ;
2019-07-19 18:20:16 +02:00
struct flow_block * block ;
2019-07-09 22:55:43 +02:00
struct list_head cb_list ;
2019-07-09 22:55:39 +02:00
struct list_head * driver_block_list ;
struct netlink_ext_ack * extack ;
} ;
2019-07-19 18:20:15 +02:00
enum tc_setup_type ;
typedef int flow_setup_cb_t ( enum tc_setup_type type , void * type_data ,
void * cb_priv ) ;
2019-07-09 22:55:42 +02:00
struct flow_block_cb {
2019-07-09 22:55:43 +02:00
struct list_head driver_list ;
2019-07-09 22:55:42 +02:00
struct list_head list ;
2019-07-19 18:20:15 +02:00
flow_setup_cb_t * cb ;
2019-07-09 22:55:42 +02:00
void * cb_ident ;
void * cb_priv ;
void ( * release ) ( void * cb_priv ) ;
unsigned int refcnt ;
} ;
2019-07-19 18:20:15 +02:00
struct flow_block_cb * flow_block_cb_alloc ( flow_setup_cb_t * cb ,
2019-07-09 22:55:42 +02:00
void * cb_ident , void * cb_priv ,
void ( * release ) ( void * cb_priv ) ) ;
void flow_block_cb_free ( struct flow_block_cb * block_cb ) ;
2019-07-19 18:20:16 +02:00
struct flow_block_cb * flow_block_cb_lookup ( struct flow_block * block ,
2019-07-19 18:20:15 +02:00
flow_setup_cb_t * cb , void * cb_ident ) ;
2019-07-09 22:55:43 +02:00
2019-07-09 22:55:44 +02:00
void * flow_block_cb_priv ( struct flow_block_cb * block_cb ) ;
void flow_block_cb_incref ( struct flow_block_cb * block_cb ) ;
unsigned int flow_block_cb_decref ( struct flow_block_cb * block_cb ) ;
2019-07-09 22:55:43 +02:00
static inline void flow_block_cb_add ( struct flow_block_cb * block_cb ,
struct flow_block_offload * offload )
{
list_add_tail ( & block_cb - > list , & offload - > cb_list ) ;
}
static inline void flow_block_cb_remove ( struct flow_block_cb * block_cb ,
struct flow_block_offload * offload )
{
list_move ( & block_cb - > list , & offload - > cb_list ) ;
}
2019-07-19 18:20:15 +02:00
bool flow_block_cb_is_busy ( flow_setup_cb_t * cb , void * cb_ident ,
2019-07-09 22:55:48 +02:00
struct list_head * driver_block_list ) ;
2019-07-09 22:55:39 +02:00
int flow_block_cb_setup_simple ( struct flow_block_offload * f ,
2019-07-19 18:20:15 +02:00
struct list_head * driver_list ,
flow_setup_cb_t * cb ,
2019-07-09 22:55:39 +02:00
void * cb_ident , void * cb_priv , bool ingress_only ) ;
2019-07-09 22:55:49 +02:00
enum flow_cls_command {
FLOW_CLS_REPLACE ,
FLOW_CLS_DESTROY ,
FLOW_CLS_STATS ,
FLOW_CLS_TMPLT_CREATE ,
FLOW_CLS_TMPLT_DESTROY ,
} ;
struct flow_cls_common_offload {
u32 chain_index ;
__be16 protocol ;
u32 prio ;
struct netlink_ext_ack * extack ;
} ;
struct flow_cls_offload {
struct flow_cls_common_offload common ;
enum flow_cls_command command ;
unsigned long cookie ;
struct flow_rule * rule ;
struct flow_stats stats ;
u32 classid ;
} ;
static inline struct flow_rule *
flow_cls_offload_flow_rule ( struct flow_cls_offload * flow_cmd )
{
return flow_cmd - > rule ;
}
2019-07-19 18:20:16 +02:00
static inline void flow_block_init ( struct flow_block * flow_block )
{
INIT_LIST_HEAD ( & flow_block - > cb_list ) ;
}
2019-08-07 09:13:52 +08:00
typedef int flow_indr_block_bind_cb_t ( struct net_device * dev , void * cb_priv ,
enum tc_setup_type type , void * type_data ) ;
typedef void flow_indr_block_ing_cmd_t ( struct net_device * dev ,
flow_indr_block_bind_cb_t * cb ,
void * cb_priv ,
enum flow_block_command command ) ;
2019-08-07 09:13:53 +08:00
struct flow_indr_block_ing_entry {
flow_indr_block_ing_cmd_t * cb ;
struct list_head list ;
} ;
void flow_indr_add_block_ing_cb ( struct flow_indr_block_ing_entry * entry ) ;
void flow_indr_del_block_ing_cb ( struct flow_indr_block_ing_entry * entry ) ;
2019-08-07 09:13:52 +08:00
int __flow_indr_block_cb_register ( struct net_device * dev , void * cb_priv ,
flow_indr_block_bind_cb_t * cb ,
void * cb_ident ) ;
void __flow_indr_block_cb_unregister ( struct net_device * dev ,
flow_indr_block_bind_cb_t * cb ,
void * cb_ident ) ;
int flow_indr_block_cb_register ( struct net_device * dev , void * cb_priv ,
flow_indr_block_bind_cb_t * cb , void * cb_ident ) ;
void flow_indr_block_cb_unregister ( struct net_device * dev ,
flow_indr_block_bind_cb_t * cb ,
void * cb_ident ) ;
void flow_indr_block_call ( struct net_device * dev ,
struct flow_block_offload * bo ,
enum flow_block_command command ) ;
2019-02-02 12:50:43 +01:00
# endif /* _NET_FLOW_OFFLOAD_H */