aboutsummaryrefslogtreecommitdiffstats
path: root/net/ipv4/ipmr.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/ipv4/ipmr.c')
-rw-r--r--net/ipv4/ipmr.c50
1 files changed, 35 insertions, 15 deletions
diff --git a/net/ipv4/ipmr.c b/net/ipv4/ipmr.c
index 99508d66a642..ec19a890c9a0 100644
--- a/net/ipv4/ipmr.c
+++ b/net/ipv4/ipmr.c
@@ -47,6 +47,7 @@
47#include <linux/mroute.h> 47#include <linux/mroute.h>
48#include <linux/init.h> 48#include <linux/init.h>
49#include <linux/if_ether.h> 49#include <linux/if_ether.h>
50#include <linux/slab.h>
50#include <net/net_namespace.h> 51#include <net/net_namespace.h>
51#include <net/ip.h> 52#include <net/ip.h>
52#include <net/protocol.h> 53#include <net/protocol.h>
@@ -275,7 +276,8 @@ failure:
275 * @notify: Set to 1, if the caller is a notifier_call 276 * @notify: Set to 1, if the caller is a notifier_call
276 */ 277 */
277 278
278static int vif_delete(struct net *net, int vifi, int notify) 279static int vif_delete(struct net *net, int vifi, int notify,
280 struct list_head *head)
279{ 281{
280 struct vif_device *v; 282 struct vif_device *v;
281 struct net_device *dev; 283 struct net_device *dev;
@@ -319,7 +321,7 @@ static int vif_delete(struct net *net, int vifi, int notify)
319 } 321 }
320 322
321 if (v->flags&(VIFF_TUNNEL|VIFF_REGISTER) && !notify) 323 if (v->flags&(VIFF_TUNNEL|VIFF_REGISTER) && !notify)
322 unregister_netdevice(dev); 324 unregister_netdevice_queue(dev, head);
323 325
324 dev_put(dev); 326 dev_put(dev);
325 return 0; 327 return 0;
@@ -469,8 +471,18 @@ static int vif_add(struct net *net, struct vifctl *vifc, int mrtsock)
469 return err; 471 return err;
470 } 472 }
471 break; 473 break;
474
475 case VIFF_USE_IFINDEX:
472 case 0: 476 case 0:
473 dev = ip_dev_find(net, vifc->vifc_lcl_addr.s_addr); 477 if (vifc->vifc_flags == VIFF_USE_IFINDEX) {
478 dev = dev_get_by_index(net, vifc->vifc_lcl_ifindex);
479 if (dev && dev->ip_ptr == NULL) {
480 dev_put(dev);
481 return -EADDRNOTAVAIL;
482 }
483 } else
484 dev = ip_dev_find(net, vifc->vifc_lcl_addr.s_addr);
485
474 if (!dev) 486 if (!dev)
475 return -EADDRNOTAVAIL; 487 return -EADDRNOTAVAIL;
476 err = dev_set_allmulti(dev, 1); 488 err = dev_set_allmulti(dev, 1);
@@ -742,7 +754,8 @@ ipmr_cache_unresolved(struct net *net, vifi_t vifi, struct sk_buff *skb)
742 c->next = mfc_unres_queue; 754 c->next = mfc_unres_queue;
743 mfc_unres_queue = c; 755 mfc_unres_queue = c;
744 756
745 mod_timer(&ipmr_expire_timer, c->mfc_un.unres.expires); 757 if (atomic_read(&net->ipv4.cache_resolve_queue_len) == 1)
758 mod_timer(&ipmr_expire_timer, c->mfc_un.unres.expires);
746 } 759 }
747 760
748 /* 761 /*
@@ -791,6 +804,9 @@ static int ipmr_mfc_add(struct net *net, struct mfcctl *mfc, int mrtsock)
791 int line; 804 int line;
792 struct mfc_cache *uc, *c, **cp; 805 struct mfc_cache *uc, *c, **cp;
793 806
807 if (mfc->mfcc_parent >= MAXVIFS)
808 return -ENFILE;
809
794 line = MFC_HASH(mfc->mfcc_mcastgrp.s_addr, mfc->mfcc_origin.s_addr); 810 line = MFC_HASH(mfc->mfcc_mcastgrp.s_addr, mfc->mfcc_origin.s_addr);
795 811
796 for (cp = &net->ipv4.mfc_cache_array[line]; 812 for (cp = &net->ipv4.mfc_cache_array[line];
@@ -862,14 +878,16 @@ static int ipmr_mfc_add(struct net *net, struct mfcctl *mfc, int mrtsock)
862static void mroute_clean_tables(struct net *net) 878static void mroute_clean_tables(struct net *net)
863{ 879{
864 int i; 880 int i;
881 LIST_HEAD(list);
865 882
866 /* 883 /*
867 * Shut down all active vif entries 884 * Shut down all active vif entries
868 */ 885 */
869 for (i = 0; i < net->ipv4.maxvif; i++) { 886 for (i = 0; i < net->ipv4.maxvif; i++) {
870 if (!(net->ipv4.vif_table[i].flags&VIFF_STATIC)) 887 if (!(net->ipv4.vif_table[i].flags&VIFF_STATIC))
871 vif_delete(net, i, 0); 888 vif_delete(net, i, 0, &list);
872 } 889 }
890 unregister_netdevice_many(&list);
873 891
874 /* 892 /*
875 * Wipe the cache 893 * Wipe the cache
@@ -948,7 +966,7 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
948 switch (optname) { 966 switch (optname) {
949 case MRT_INIT: 967 case MRT_INIT:
950 if (sk->sk_type != SOCK_RAW || 968 if (sk->sk_type != SOCK_RAW ||
951 inet_sk(sk)->num != IPPROTO_IGMP) 969 inet_sk(sk)->inet_num != IPPROTO_IGMP)
952 return -EOPNOTSUPP; 970 return -EOPNOTSUPP;
953 if (optlen != sizeof(int)) 971 if (optlen != sizeof(int))
954 return -ENOPROTOOPT; 972 return -ENOPROTOOPT;
@@ -985,7 +1003,7 @@ int ip_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsi
985 if (optname == MRT_ADD_VIF) { 1003 if (optname == MRT_ADD_VIF) {
986 ret = vif_add(net, &vif, sk == net->ipv4.mroute_sk); 1004 ret = vif_add(net, &vif, sk == net->ipv4.mroute_sk);
987 } else { 1005 } else {
988 ret = vif_delete(net, vif.vifc_vifi, 0); 1006 ret = vif_delete(net, vif.vifc_vifi, 0, NULL);
989 } 1007 }
990 rtnl_unlock(); 1008 rtnl_unlock();
991 return ret; 1009 return ret;
@@ -1148,17 +1166,16 @@ static int ipmr_device_event(struct notifier_block *this, unsigned long event, v
1148 struct net *net = dev_net(dev); 1166 struct net *net = dev_net(dev);
1149 struct vif_device *v; 1167 struct vif_device *v;
1150 int ct; 1168 int ct;
1151 1169 LIST_HEAD(list);
1152 if (!net_eq(dev_net(dev), net))
1153 return NOTIFY_DONE;
1154 1170
1155 if (event != NETDEV_UNREGISTER) 1171 if (event != NETDEV_UNREGISTER)
1156 return NOTIFY_DONE; 1172 return NOTIFY_DONE;
1157 v = &net->ipv4.vif_table[0]; 1173 v = &net->ipv4.vif_table[0];
1158 for (ct = 0; ct < net->ipv4.maxvif; ct++, v++) { 1174 for (ct = 0; ct < net->ipv4.maxvif; ct++, v++) {
1159 if (v->dev == dev) 1175 if (v->dev == dev)
1160 vif_delete(net, ct, 1); 1176 vif_delete(net, ct, 1, &list);
1161 } 1177 }
1178 unregister_netdevice_many(&list);
1162 return NOTIFY_DONE; 1179 return NOTIFY_DONE;
1163} 1180}
1164 1181
@@ -1601,17 +1618,20 @@ ipmr_fill_mroute(struct sk_buff *skb, struct mfc_cache *c, struct rtmsg *rtm)
1601 int ct; 1618 int ct;
1602 struct rtnexthop *nhp; 1619 struct rtnexthop *nhp;
1603 struct net *net = mfc_net(c); 1620 struct net *net = mfc_net(c);
1604 struct net_device *dev = net->ipv4.vif_table[c->mfc_parent].dev;
1605 u8 *b = skb_tail_pointer(skb); 1621 u8 *b = skb_tail_pointer(skb);
1606 struct rtattr *mp_head; 1622 struct rtattr *mp_head;
1607 1623
1608 if (dev) 1624 /* If cache is unresolved, don't try to parse IIF and OIF */
1609 RTA_PUT(skb, RTA_IIF, 4, &dev->ifindex); 1625 if (c->mfc_parent > MAXVIFS)
1626 return -ENOENT;
1627
1628 if (VIF_EXISTS(net, c->mfc_parent))
1629 RTA_PUT(skb, RTA_IIF, 4, &net->ipv4.vif_table[c->mfc_parent].dev->ifindex);
1610 1630
1611 mp_head = (struct rtattr *)skb_put(skb, RTA_LENGTH(0)); 1631 mp_head = (struct rtattr *)skb_put(skb, RTA_LENGTH(0));
1612 1632
1613 for (ct = c->mfc_un.res.minvif; ct < c->mfc_un.res.maxvif; ct++) { 1633 for (ct = c->mfc_un.res.minvif; ct < c->mfc_un.res.maxvif; ct++) {
1614 if (c->mfc_un.res.ttls[ct] < 255) { 1634 if (VIF_EXISTS(net, ct) && c->mfc_un.res.ttls[ct] < 255) {
1615 if (skb_tailroom(skb) < RTA_ALIGN(RTA_ALIGN(sizeof(*nhp)) + 4)) 1635 if (skb_tailroom(skb) < RTA_ALIGN(RTA_ALIGN(sizeof(*nhp)) + 4))
1616 goto rtattr_failure; 1636 goto rtattr_failure;
1617 nhp = (struct rtnexthop *)skb_put(skb, RTA_ALIGN(sizeof(*nhp))); 1637 nhp = (struct rtnexthop *)skb_put(skb, RTA_ALIGN(sizeof(*nhp)));