2014-11-19 14:05:03 +01:00
/*
* Copyright ( c ) 2014 Jiri Pirko < jiri @ resnulli . us >
*
* 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 .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/skbuff.h>
# include <linux/rtnetlink.h>
# include <linux/if_vlan.h>
# include <net/netlink.h>
# include <net/pkt_sched.h>
# include <linux/tc_act/tc_vlan.h>
# include <net/tc_act/tc_vlan.h>
# define VLAN_TAB_MASK 15
2016-02-22 15:57:53 -08:00
static int vlan_net_id ;
2016-07-25 16:09:41 -07:00
static struct tc_action_ops act_vlan_ops ;
2016-02-22 15:57:53 -08:00
2014-11-19 14:05:03 +01:00
static int tcf_vlan ( struct sk_buff * skb , const struct tc_action * a ,
struct tcf_result * res )
{
2016-07-25 16:09:41 -07:00
struct tcf_vlan * v = to_vlan ( a ) ;
2014-11-19 14:05:03 +01:00
int action ;
int err ;
2016-09-19 19:11:10 +03:00
u16 tci ;
2014-11-19 14:05:03 +01:00
spin_lock ( & v - > tcf_lock ) ;
2016-06-06 06:32:53 -04:00
tcf_lastuse_update ( & v - > tcf_tm ) ;
2014-11-19 14:05:03 +01:00
bstats_update ( & v - > tcf_bstats , skb ) ;
action = v - > tcf_action ;
2016-09-29 12:10:40 +03:00
/* Ensure 'data' points at mac_header prior calling vlan manipulating
* functions .
*/
if ( skb_at_tc_ingress ( skb ) )
skb_push_rcsum ( skb , skb - > mac_len ) ;
2014-11-19 14:05:03 +01:00
switch ( v - > tcfv_action ) {
case TCA_VLAN_ACT_POP :
err = skb_vlan_pop ( skb ) ;
if ( err )
goto drop ;
break ;
case TCA_VLAN_ACT_PUSH :
2016-08-17 13:36:14 +03:00
err = skb_vlan_push ( skb , v - > tcfv_push_proto , v - > tcfv_push_vid |
( v - > tcfv_push_prio < < VLAN_PRIO_SHIFT ) ) ;
2014-11-19 14:05:03 +01:00
if ( err )
goto drop ;
break ;
2016-09-19 19:11:10 +03:00
case TCA_VLAN_ACT_MODIFY :
/* No-op if no vlan tag (either hw-accel or in-payload) */
if ( ! skb_vlan_tagged ( skb ) )
goto unlock ;
/* extract existing tag (and guarantee no hw-accel tag) */
if ( skb_vlan_tag_present ( skb ) ) {
tci = skb_vlan_tag_get ( skb ) ;
skb - > vlan_tci = 0 ;
} else {
/* in-payload vlan tag, pop it */
err = __skb_vlan_pop ( skb , & tci ) ;
if ( err )
goto drop ;
}
/* replace the vid */
tci = ( tci & ~ VLAN_VID_MASK ) | v - > tcfv_push_vid ;
/* replace prio bits, if tcfv_push_prio specified */
if ( v - > tcfv_push_prio ) {
tci & = ~ VLAN_PRIO_MASK ;
tci | = v - > tcfv_push_prio < < VLAN_PRIO_SHIFT ;
}
/* put updated tci as hwaccel tag */
__vlan_hwaccel_put_tag ( skb , v - > tcfv_push_proto , tci ) ;
break ;
2014-11-19 14:05:03 +01:00
default :
BUG ( ) ;
}
goto unlock ;
drop :
action = TC_ACT_SHOT ;
v - > tcf_qstats . drops + + ;
unlock :
2016-09-29 12:10:40 +03:00
if ( skb_at_tc_ingress ( skb ) )
skb_pull_rcsum ( skb , skb - > mac_len ) ;
2014-11-19 14:05:03 +01:00
spin_unlock ( & v - > tcf_lock ) ;
return action ;
}
static const struct nla_policy vlan_policy [ TCA_VLAN_MAX + 1 ] = {
[ TCA_VLAN_PARMS ] = { . len = sizeof ( struct tc_vlan ) } ,
[ TCA_VLAN_PUSH_VLAN_ID ] = { . type = NLA_U16 } ,
[ TCA_VLAN_PUSH_VLAN_PROTOCOL ] = { . type = NLA_U16 } ,
2016-08-17 13:36:14 +03:00
[ TCA_VLAN_PUSH_VLAN_PRIORITY ] = { . type = NLA_U8 } ,
2014-11-19 14:05:03 +01:00
} ;
static int tcf_vlan_init ( struct net * net , struct nlattr * nla ,
2016-07-25 16:09:41 -07:00
struct nlattr * est , struct tc_action * * a ,
2014-11-19 14:05:03 +01:00
int ovr , int bind )
{
2016-02-22 15:57:53 -08:00
struct tc_action_net * tn = net_generic ( net , vlan_net_id ) ;
2014-11-19 14:05:03 +01:00
struct nlattr * tb [ TCA_VLAN_MAX + 1 ] ;
struct tc_vlan * parm ;
struct tcf_vlan * v ;
int action ;
__be16 push_vid = 0 ;
__be16 push_proto = 0 ;
2016-08-17 13:36:14 +03:00
u8 push_prio = 0 ;
2016-06-13 13:46:28 -07:00
bool exists = false ;
int ret = 0 , err ;
2014-11-19 14:05:03 +01:00
if ( ! nla )
return - EINVAL ;
err = nla_parse_nested ( tb , TCA_VLAN_MAX , nla , vlan_policy ) ;
if ( err < 0 )
return err ;
if ( ! tb [ TCA_VLAN_PARMS ] )
return - EINVAL ;
parm = nla_data ( tb [ TCA_VLAN_PARMS ] ) ;
2016-05-10 16:49:26 -04:00
exists = tcf_hash_check ( tn , parm - > index , a , bind ) ;
if ( exists & & bind )
return 0 ;
2014-11-19 14:05:03 +01:00
switch ( parm - > v_action ) {
case TCA_VLAN_ACT_POP :
break ;
case TCA_VLAN_ACT_PUSH :
2016-09-19 19:11:10 +03:00
case TCA_VLAN_ACT_MODIFY :
2016-05-10 16:49:26 -04:00
if ( ! tb [ TCA_VLAN_PUSH_VLAN_ID ] ) {
if ( exists )
2016-07-25 16:09:41 -07:00
tcf_hash_release ( * a , bind ) ;
2014-11-19 14:05:03 +01:00
return - EINVAL ;
2016-05-10 16:49:26 -04:00
}
2014-11-19 14:05:03 +01:00
push_vid = nla_get_u16 ( tb [ TCA_VLAN_PUSH_VLAN_ID ] ) ;
2016-05-10 16:49:26 -04:00
if ( push_vid > = VLAN_VID_MASK ) {
if ( exists )
2016-07-25 16:09:41 -07:00
tcf_hash_release ( * a , bind ) ;
2014-11-19 14:05:03 +01:00
return - ERANGE ;
2016-05-10 16:49:26 -04:00
}
2014-11-19 14:05:03 +01:00
if ( tb [ TCA_VLAN_PUSH_VLAN_PROTOCOL ] ) {
push_proto = nla_get_be16 ( tb [ TCA_VLAN_PUSH_VLAN_PROTOCOL ] ) ;
switch ( push_proto ) {
case htons ( ETH_P_8021Q ) :
case htons ( ETH_P_8021AD ) :
break ;
default :
return - EPROTONOSUPPORT ;
}
} else {
push_proto = htons ( ETH_P_8021Q ) ;
}
2016-08-17 13:36:14 +03:00
if ( tb [ TCA_VLAN_PUSH_VLAN_PRIORITY ] )
push_prio = nla_get_u8 ( tb [ TCA_VLAN_PUSH_VLAN_PRIORITY ] ) ;
2014-11-19 14:05:03 +01:00
break ;
default :
2016-05-10 16:49:26 -04:00
if ( exists )
2016-07-25 16:09:41 -07:00
tcf_hash_release ( * a , bind ) ;
2014-11-19 14:05:03 +01:00
return - EINVAL ;
}
action = parm - > v_action ;
2016-05-10 16:49:26 -04:00
if ( ! exists ) {
2016-02-22 15:57:53 -08:00
ret = tcf_hash_create ( tn , parm - > index , est , a ,
2016-07-25 16:09:41 -07:00
& act_vlan_ops , bind , false ) ;
2014-11-19 14:05:03 +01:00
if ( ret )
return ret ;
ret = ACT_P_CREATED ;
} else {
2016-07-25 16:09:41 -07:00
tcf_hash_release ( * a , bind ) ;
2014-11-19 14:05:03 +01:00
if ( ! ovr )
return - EEXIST ;
}
2016-07-25 16:09:41 -07:00
v = to_vlan ( * a ) ;
2014-11-19 14:05:03 +01:00
spin_lock_bh ( & v - > tcf_lock ) ;
v - > tcfv_action = action ;
v - > tcfv_push_vid = push_vid ;
2016-08-17 13:36:14 +03:00
v - > tcfv_push_prio = push_prio ;
2014-11-19 14:05:03 +01:00
v - > tcfv_push_proto = push_proto ;
v - > tcf_action = parm - > action ;
spin_unlock_bh ( & v - > tcf_lock ) ;
if ( ret = = ACT_P_CREATED )
2016-07-25 16:09:41 -07:00
tcf_hash_insert ( tn , * a ) ;
2014-11-19 14:05:03 +01:00
return ret ;
}
static int tcf_vlan_dump ( struct sk_buff * skb , struct tc_action * a ,
int bind , int ref )
{
unsigned char * b = skb_tail_pointer ( skb ) ;
2016-07-25 16:09:41 -07:00
struct tcf_vlan * v = to_vlan ( a ) ;
2014-11-19 14:05:03 +01:00
struct tc_vlan opt = {
. index = v - > tcf_index ,
. refcnt = v - > tcf_refcnt - ref ,
. bindcnt = v - > tcf_bindcnt - bind ,
. action = v - > tcf_action ,
. v_action = v - > tcfv_action ,
} ;
struct tcf_t t ;
if ( nla_put ( skb , TCA_VLAN_PARMS , sizeof ( opt ) , & opt ) )
goto nla_put_failure ;
2016-09-19 19:11:10 +03:00
if ( ( v - > tcfv_action = = TCA_VLAN_ACT_PUSH | |
v - > tcfv_action = = TCA_VLAN_ACT_MODIFY ) & &
2014-11-19 14:05:03 +01:00
( nla_put_u16 ( skb , TCA_VLAN_PUSH_VLAN_ID , v - > tcfv_push_vid ) | |
2016-06-05 10:41:32 -04:00
nla_put_be16 ( skb , TCA_VLAN_PUSH_VLAN_PROTOCOL ,
2016-08-17 13:36:14 +03:00
v - > tcfv_push_proto ) | |
( nla_put_u8 ( skb , TCA_VLAN_PUSH_VLAN_PRIORITY ,
v - > tcfv_push_prio ) ) ) )
2014-11-19 14:05:03 +01:00
goto nla_put_failure ;
2016-06-06 06:32:55 -04:00
tcf_tm_dump ( & t , & v - > tcf_tm ) ;
2016-04-26 10:06:18 +02:00
if ( nla_put_64bit ( skb , TCA_VLAN_TM , sizeof ( t ) , & t , TCA_VLAN_PAD ) )
2014-11-19 14:05:03 +01:00
goto nla_put_failure ;
return skb - > len ;
nla_put_failure :
nlmsg_trim ( skb , b ) ;
return - 1 ;
}
2016-02-22 15:57:53 -08:00
static int tcf_vlan_walker ( struct net * net , struct sk_buff * skb ,
struct netlink_callback * cb , int type ,
2016-07-25 16:09:41 -07:00
const struct tc_action_ops * ops )
2016-02-22 15:57:53 -08:00
{
struct tc_action_net * tn = net_generic ( net , vlan_net_id ) ;
2016-07-25 16:09:41 -07:00
return tcf_generic_walker ( tn , skb , cb , type , ops ) ;
2016-02-22 15:57:53 -08:00
}
2016-07-25 16:09:41 -07:00
static int tcf_vlan_search ( struct net * net , struct tc_action * * a , u32 index )
2016-02-22 15:57:53 -08:00
{
struct tc_action_net * tn = net_generic ( net , vlan_net_id ) ;
return tcf_hash_search ( tn , a , index ) ;
}
2014-11-19 14:05:03 +01:00
static struct tc_action_ops act_vlan_ops = {
. kind = " vlan " ,
. type = TCA_ACT_VLAN ,
. owner = THIS_MODULE ,
. act = tcf_vlan ,
. dump = tcf_vlan_dump ,
. init = tcf_vlan_init ,
2016-02-22 15:57:53 -08:00
. walk = tcf_vlan_walker ,
. lookup = tcf_vlan_search ,
2016-07-25 16:09:41 -07:00
. size = sizeof ( struct tcf_vlan ) ,
2016-02-22 15:57:53 -08:00
} ;
static __net_init int vlan_init_net ( struct net * net )
{
struct tc_action_net * tn = net_generic ( net , vlan_net_id ) ;
return tc_action_net_init ( tn , & act_vlan_ops , VLAN_TAB_MASK ) ;
}
static void __net_exit vlan_exit_net ( struct net * net )
{
struct tc_action_net * tn = net_generic ( net , vlan_net_id ) ;
tc_action_net_exit ( tn ) ;
}
static struct pernet_operations vlan_net_ops = {
. init = vlan_init_net ,
. exit = vlan_exit_net ,
. id = & vlan_net_id ,
. size = sizeof ( struct tc_action_net ) ,
2014-11-19 14:05:03 +01:00
} ;
static int __init vlan_init_module ( void )
{
2016-02-22 15:57:53 -08:00
return tcf_register_action ( & act_vlan_ops , & vlan_net_ops ) ;
2014-11-19 14:05:03 +01:00
}
static void __exit vlan_cleanup_module ( void )
{
2016-02-22 15:57:53 -08:00
tcf_unregister_action ( & act_vlan_ops , & vlan_net_ops ) ;
2014-11-19 14:05:03 +01:00
}
module_init ( vlan_init_module ) ;
module_exit ( vlan_cleanup_module ) ;
MODULE_AUTHOR ( " Jiri Pirko <jiri@resnulli.us> " ) ;
MODULE_DESCRIPTION ( " vlan manipulation actions " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;