diff options
author | stephen hemminger <shemminger@vyatta.com> | 2011-12-31 08:26:46 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2012-01-12 14:35:44 -0500 |
commit | 81aaa36dde03db855fb233382c7667f98ac12659 (patch) | |
tree | a032daae8934d50399ef623786c5433facfa469d | |
parent | 2643bcef53a08c09975861ebbed6d687571a2751 (diff) |
bonding: fix error handling if slave is busy (v2)
commit f7d9821a6a9c83450ac35e76d3709e32fd38b76f upstream.
If slave device already has a receive handler registered, then the
error unwind of bonding device enslave function is broken.
The following will leave a pointer to freed memory in the slave
device list, causing a later kernel panic.
# modprobe dummy
# ip li add dummy0-1 link dummy0 type macvlan
# modprobe bonding
# echo +dummy0 >/sys/class/net/bond0/bonding/slaves
The fix is to detach the slave (which removes it from the list)
in the unwind path.
Signed-off-by: Stephen Hemminger <shemminger@vyatta.com>
Reviewed-by: Nicolas de Pesloüan <nicolas.2p.debian@free.fr>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
-rw-r--r-- | drivers/net/bonding/bond_main.c | 9 |
1 files changed, 7 insertions, 2 deletions
diff --git a/drivers/net/bonding/bond_main.c b/drivers/net/bonding/bond_main.c index 2065cb4002b..0b65c5fccc3 100644 --- a/drivers/net/bonding/bond_main.c +++ b/drivers/net/bonding/bond_main.c | |||
@@ -1905,7 +1905,7 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev) | |||
1905 | "but new slave device does not support netpoll.\n", | 1905 | "but new slave device does not support netpoll.\n", |
1906 | bond_dev->name); | 1906 | bond_dev->name); |
1907 | res = -EBUSY; | 1907 | res = -EBUSY; |
1908 | goto err_close; | 1908 | goto err_detach; |
1909 | } | 1909 | } |
1910 | } | 1910 | } |
1911 | #endif | 1911 | #endif |
@@ -1914,7 +1914,7 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev) | |||
1914 | 1914 | ||
1915 | res = bond_create_slave_symlinks(bond_dev, slave_dev); | 1915 | res = bond_create_slave_symlinks(bond_dev, slave_dev); |
1916 | if (res) | 1916 | if (res) |
1917 | goto err_close; | 1917 | goto err_detach; |
1918 | 1918 | ||
1919 | res = netdev_rx_handler_register(slave_dev, bond_handle_frame, | 1919 | res = netdev_rx_handler_register(slave_dev, bond_handle_frame, |
1920 | new_slave); | 1920 | new_slave); |
@@ -1935,6 +1935,11 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev) | |||
1935 | err_dest_symlinks: | 1935 | err_dest_symlinks: |
1936 | bond_destroy_slave_symlinks(bond_dev, slave_dev); | 1936 | bond_destroy_slave_symlinks(bond_dev, slave_dev); |
1937 | 1937 | ||
1938 | err_detach: | ||
1939 | write_lock_bh(&bond->lock); | ||
1940 | bond_detach_slave(bond, new_slave); | ||
1941 | write_unlock_bh(&bond->lock); | ||
1942 | |||
1938 | err_close: | 1943 | err_close: |
1939 | dev_close(slave_dev); | 1944 | dev_close(slave_dev); |
1940 | 1945 | ||