2  * net/sched/mirred.c   packet mirroring and redirect actions
 
   4  *              This program is free software; you can redistribute it and/or
 
   5  *              modify it under the terms of the GNU General Public License
 
   6  *              as published by the Free Software Foundation; either version
 
   7  *              2 of the License, or (at your option) any later version.
 
   9  * Authors:     Jamal Hadi Salim (2002-4)
 
  11  * TODO: Add ingress support (and socket redirect support)
 
  15 #include <asm/uaccess.h>
 
  16 #include <asm/system.h>
 
  17 #include <asm/bitops.h>
 
  18 #include <linux/types.h>
 
  19 #include <linux/kernel.h>
 
  20 #include <linux/sched.h>
 
  21 #include <linux/string.h>
 
  23 #include <linux/socket.h>
 
  24 #include <linux/sockios.h>
 
  26 #include <linux/errno.h>
 
  27 #include <linux/interrupt.h>
 
  28 #include <linux/netdevice.h>
 
  29 #include <linux/skbuff.h>
 
  30 #include <linux/rtnetlink.h>
 
  31 #include <linux/module.h>
 
  32 #include <linux/init.h>
 
  33 #include <linux/proc_fs.h>
 
  35 #include <net/pkt_sched.h>
 
  36 #include <linux/tc_act/tc_mirred.h>
 
  37 #include <net/tc_act/tc_mirred.h>
 
  39 #include <linux/etherdevice.h>
 
  40 #include <linux/if_arp.h>
 
  43 /* use generic hash table */
 
  45 #define MY_TAB_MASK     (MY_TAB_SIZE - 1)
 
  47 static struct tcf_mirred *tcf_mirred_ht[MY_TAB_SIZE];
 
  48 static DEFINE_RWLOCK(mirred_lock);
 
  50 /* ovewrride the defaults */
 
  51 #define tcf_st          tcf_mirred
 
  52 #define tc_st           tc_mirred
 
  53 #define tcf_t_lock      mirred_lock
 
  54 #define tcf_ht          tcf_mirred_ht
 
  56 #define CONFIG_NET_ACT_INIT 1
 
  57 #include <net/pkt_act.h>
 
  60 tcf_mirred_release(struct tcf_mirred *p, int bind)
 
  66                 if(!p->bindcnt && p->refcnt <= 0) {
 
  76 tcf_mirred_init(struct rtattr *rta, struct rtattr *est, struct tc_action *a,
 
  79         struct rtattr *tb[TCA_MIRRED_MAX];
 
  80         struct tc_mirred *parm;
 
  82         struct net_device *dev = NULL;
 
  86         if (rta == NULL || rtattr_parse_nested(tb, TCA_MIRRED_MAX, rta) < 0)
 
  89         if (tb[TCA_MIRRED_PARMS-1] == NULL ||
 
  90             RTA_PAYLOAD(tb[TCA_MIRRED_PARMS-1]) < sizeof(*parm))
 
  92         parm = RTA_DATA(tb[TCA_MIRRED_PARMS-1]);
 
  95                 dev = __dev_get_by_index(parm->ifindex);
 
 113         p = tcf_hash_check(parm->index, a, ovr, bind);
 
 117                 p = tcf_hash_create(parm->index, est, a, sizeof(*p), ovr, bind);
 
 123                         tcf_mirred_release(p, bind);
 
 128         spin_lock_bh(&p->lock);
 
 129         p->action = parm->action;
 
 130         p->eaction = parm->eaction;
 
 132                 p->ifindex = parm->ifindex;
 
 133                 if (ret != ACT_P_CREATED)
 
 137                 p->ok_push = ok_push;
 
 139         spin_unlock_bh(&p->lock);
 
 140         if (ret == ACT_P_CREATED)
 
 143         DPRINTK("tcf_mirred_init index %d action %d eaction %d device %s "
 
 144                 "ifindex %d\n", parm->index, parm->action, parm->eaction,
 
 145                 dev->name, parm->ifindex);
 
 150 tcf_mirred_cleanup(struct tc_action *a, int bind)
 
 152         struct tcf_mirred *p = PRIV(a, mirred);
 
 155                 return tcf_mirred_release(p, bind);
 
 160 tcf_mirred(struct sk_buff *skb, struct tc_action *a, struct tcf_result *res)
 
 162         struct tcf_mirred *p = PRIV(a, mirred);
 
 163         struct net_device *dev;
 
 164         struct sk_buff *skb2 = NULL;
 
 165         u32 at = G_TC_AT(skb->tc_verd);
 
 170         p->tm.lastuse = jiffies;
 
 172         if (!(dev->flags&IFF_UP) ) {
 
 174                         printk("mirred to Houston: device %s is gone!\n",
 
 179                 p->qstats.overlimits++;
 
 180                 p->bstats.bytes += skb->len;
 
 182                 spin_unlock(&p->lock);
 
 183                 /* should we be asking for packet to be dropped?
 
 184                  * may make sense for redirect case only
 
 189         skb2 = skb_clone(skb, GFP_ATOMIC);
 
 192         if (p->eaction != TCA_EGRESS_MIRROR && p->eaction != TCA_EGRESS_REDIR) {
 
 194                         printk("tcf_mirred unknown action %d\n", p->eaction);
 
 198         p->bstats.bytes += skb2->len;
 
 200         if (!(at & AT_EGRESS))
 
 202                         skb_push(skb2, skb2->dev->hard_header_len);
 
 204         /* mirror is always swallowed */
 
 205         if (p->eaction != TCA_EGRESS_MIRROR)
 
 206                 skb2->tc_verd = SET_TC_FROM(skb2->tc_verd, at);
 
 209         skb2->input_dev = skb->dev;
 
 210         dev_queue_xmit(skb2);
 
 211         spin_unlock(&p->lock);
 
 216 tcf_mirred_dump(struct sk_buff *skb, struct tc_action *a, int bind, int ref)
 
 218         unsigned char *b = skb->tail;
 
 219         struct tc_mirred opt;
 
 220         struct tcf_mirred *p = PRIV(a, mirred);
 
 223         opt.index = p->index;
 
 224         opt.action = p->action;
 
 225         opt.refcnt = p->refcnt - ref;
 
 226         opt.bindcnt = p->bindcnt - bind;
 
 227         opt.eaction = p->eaction;
 
 228         opt.ifindex = p->ifindex;
 
 229         DPRINTK("tcf_mirred_dump index %d action %d eaction %d ifindex %d\n",
 
 230                  p->index, p->action, p->eaction, p->ifindex);
 
 231         RTA_PUT(skb, TCA_MIRRED_PARMS, sizeof(opt), &opt);
 
 232         t.install = jiffies_to_clock_t(jiffies - p->tm.install);
 
 233         t.lastuse = jiffies_to_clock_t(jiffies - p->tm.lastuse);
 
 234         t.expires = jiffies_to_clock_t(p->tm.expires);
 
 235         RTA_PUT(skb, TCA_MIRRED_TM, sizeof(t), &t);
 
 239         skb_trim(skb, b - skb->data);
 
 243 static struct tc_action_ops act_mirred_ops = {
 
 245         .type           =       TCA_ACT_MIRRED,
 
 246         .capab          =       TCA_CAP_NONE,
 
 247         .owner          =       THIS_MODULE,
 
 249         .dump           =       tcf_mirred_dump,
 
 250         .cleanup        =       tcf_mirred_cleanup,
 
 251         .lookup         =       tcf_hash_search,
 
 252         .init           =       tcf_mirred_init,
 
 253         .walk           =       tcf_generic_walker
 
 256 MODULE_AUTHOR("Jamal Hadi Salim(2002)");
 
 257 MODULE_DESCRIPTION("Device Mirror/redirect actions");
 
 258 MODULE_LICENSE("GPL");
 
 261 mirred_init_module(void)
 
 263         printk("Mirror/redirect action on\n");
 
 264         return tcf_register_action(&act_mirred_ops);
 
 268 mirred_cleanup_module(void)
 
 270         tcf_unregister_action(&act_mirred_ops);
 
 273 module_init(mirred_init_module);
 
 274 module_exit(mirred_cleanup_module);