aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net
diff options
context:
space:
mode:
authorEric W. Biederman <ebiederm@aristanetworks.com>2009-10-29 10:18:25 -0400
committerDavid S. Miller <davem@davemloft.net>2009-10-30 15:41:21 -0400
commit88ead977109da926a03068e277869ea8fedd170d (patch)
tree81f1ef8acbb6e6aa49c020e6c1d6e7ed5f018341 /drivers/net
parentc67dfb299e05a132154b9bfaae4a83de478ffaa9 (diff)
bond: Implement a basic set of rtnl link ops
This implements a basic set of rtnl link ops and takes advantage of the fact that rtnl_link_unregister kills all of the surviving devices to all us to kill bond_free_all. A module alias is added so ip link add can pull in the bonding module. Signed-off-by: Eric W. Biederman <ebiederm@aristanetworks.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net')
-rw-r--r--drivers/net/bonding/bond_main.c55
1 files changed, 30 insertions, 25 deletions
diff --git a/drivers/net/bonding/bond_main.c b/drivers/net/bonding/bond_main.c
index db9640b43d01..405971374fe2 100644
--- a/drivers/net/bonding/bond_main.c
+++ b/drivers/net/bonding/bond_main.c
@@ -4612,22 +4612,6 @@ static void bond_uninit(struct net_device *bond_dev)
4612 netif_addr_unlock_bh(bond_dev); 4612 netif_addr_unlock_bh(bond_dev);
4613} 4613}
4614 4614
4615/* Unregister and free all bond devices.
4616 * Caller must hold rtnl_lock.
4617 */
4618static void bond_free_all(void)
4619{
4620 struct bonding *bond, *nxt;
4621
4622 list_for_each_entry_safe(bond, nxt, &bond_dev_list, bond_list) {
4623 struct net_device *bond_dev = bond->dev;
4624
4625 unregister_netdevice(bond_dev);
4626 }
4627
4628 bond_destroy_proc_dir();
4629}
4630
4631/*------------------------- Module initialization ---------------------------*/ 4615/*------------------------- Module initialization ---------------------------*/
4632 4616
4633/* 4617/*
@@ -5065,6 +5049,23 @@ static int bond_init(struct net_device *bond_dev)
5065 return 0; 5049 return 0;
5066} 5050}
5067 5051
5052static int bond_validate(struct nlattr *tb[], struct nlattr *data[])
5053{
5054 if (tb[IFLA_ADDRESS]) {
5055 if (nla_len(tb[IFLA_ADDRESS]) != ETH_ALEN)
5056 return -EINVAL;
5057 if (!is_valid_ether_addr(nla_data(tb[IFLA_ADDRESS])))
5058 return -EADDRNOTAVAIL;
5059 }
5060 return 0;
5061}
5062
5063static struct rtnl_link_ops bond_link_ops __read_mostly = {
5064 .kind = "bond",
5065 .setup = bond_setup,
5066 .validate = bond_validate,
5067};
5068
5068/* Create a new bond based on the specified name and bonding parameters. 5069/* Create a new bond based on the specified name and bonding parameters.
5069 * If name is NULL, obtain a suitable "bond%d" name for us. 5070 * If name is NULL, obtain a suitable "bond%d" name for us.
5070 * Caller must NOT hold rtnl_lock; we need to release it here before we 5071 * Caller must NOT hold rtnl_lock; we need to release it here before we
@@ -5086,6 +5087,8 @@ int bond_create(const char *name)
5086 goto out; 5087 goto out;
5087 } 5088 }
5088 5089
5090 bond_dev->rtnl_link_ops = &bond_link_ops;
5091
5089 if (!name) { 5092 if (!name) {
5090 res = dev_alloc_name(bond_dev, "bond%d"); 5093 res = dev_alloc_name(bond_dev, "bond%d");
5091 if (res < 0) 5094 if (res < 0)
@@ -5115,6 +5118,10 @@ static int __init bonding_init(void)
5115 5118
5116 bond_create_proc_dir(); 5119 bond_create_proc_dir();
5117 5120
5121 res = rtnl_link_register(&bond_link_ops);
5122 if (res)
5123 goto err;
5124
5118 for (i = 0; i < max_bonds; i++) { 5125 for (i = 0; i < max_bonds; i++) {
5119 res = bond_create(NULL); 5126 res = bond_create(NULL);
5120 if (res) 5127 if (res)
@@ -5128,14 +5135,12 @@ static int __init bonding_init(void)
5128 register_netdevice_notifier(&bond_netdev_notifier); 5135 register_netdevice_notifier(&bond_netdev_notifier);
5129 register_inetaddr_notifier(&bond_inetaddr_notifier); 5136 register_inetaddr_notifier(&bond_inetaddr_notifier);
5130 bond_register_ipv6_notifier(); 5137 bond_register_ipv6_notifier();
5131
5132 goto out;
5133err:
5134 rtnl_lock();
5135 bond_free_all();
5136 rtnl_unlock();
5137out: 5138out:
5138 return res; 5139 return res;
5140err:
5141 rtnl_link_unregister(&bond_link_ops);
5142 bond_destroy_proc_dir();
5143 goto out;
5139 5144
5140} 5145}
5141 5146
@@ -5147,9 +5152,8 @@ static void __exit bonding_exit(void)
5147 5152
5148 bond_destroy_sysfs(); 5153 bond_destroy_sysfs();
5149 5154
5150 rtnl_lock(); 5155 rtnl_link_unregister(&bond_link_ops);
5151 bond_free_all(); 5156 bond_destroy_proc_dir();
5152 rtnl_unlock();
5153} 5157}
5154 5158
5155module_init(bonding_init); 5159module_init(bonding_init);
@@ -5158,3 +5162,4 @@ MODULE_LICENSE("GPL");
5158MODULE_VERSION(DRV_VERSION); 5162MODULE_VERSION(DRV_VERSION);
5159MODULE_DESCRIPTION(DRV_DESCRIPTION ", v" DRV_VERSION); 5163MODULE_DESCRIPTION(DRV_DESCRIPTION ", v" DRV_VERSION);
5160MODULE_AUTHOR("Thomas Davis, tadavis@lbl.gov and many others"); 5164MODULE_AUTHOR("Thomas Davis, tadavis@lbl.gov and many others");
5165MODULE_ALIAS_RTNL_LINK("bond");