Commit a1b3f594 authored by Eric W. Biederman's avatar Eric W. Biederman Committed by Greg Kroah-Hartman

net: Expose all network devices in a namespaces in sysfs

This reverts commit aaf8cdc3.

Drivers like the ipw2100 call device_create_group when they
are initialized and device_remove_group when they are shutdown.
Moving them between namespaces deletes their sysfs groups early.

In particular the following call chain results.
netdev_unregister_kobject -> device_del -> kobject_del -> sysfs_remove_dir
With sysfs_remove_dir recursively deleting all of it's subdirectories,
and nothing adding them back.

Ouch!

Therefore we need to call something that ultimate calls sysfs_mv_dir
as that sysfs function can move sysfs directories between namespaces
without deleting their subdirectories or their contents.   Allowing
us to avoid placing extra boiler plate into every driver that does
something interesting with sysfs.

Currently the function that provides that capability is device_rename.
That is the code works without nasty side effects as originally written.

So remove the misguided fix for moving devices between namespaces.  The
bug in the kobject layer that inspired it has now been recognized and
fixed.
Signed-off-by: default avatarEric W. Biederman <ebiederm@xmission.com>
Acked-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 417daa1e
...@@ -1002,15 +1002,10 @@ int dev_change_name(struct net_device *dev, const char *newname) ...@@ -1002,15 +1002,10 @@ int dev_change_name(struct net_device *dev, const char *newname)
return err; return err;
rollback: rollback:
/* For now only devices in the initial network namespace ret = device_rename(&dev->dev, dev->name);
* are in sysfs. if (ret) {
*/ memcpy(dev->name, oldname, IFNAMSIZ);
if (net_eq(net, &init_net)) { return ret;
ret = device_rename(&dev->dev, dev->name);
if (ret) {
memcpy(dev->name, oldname, IFNAMSIZ);
return ret;
}
} }
write_lock_bh(&dev_base_lock); write_lock_bh(&dev_base_lock);
...@@ -4994,8 +4989,6 @@ int register_netdevice(struct net_device *dev) ...@@ -4994,8 +4989,6 @@ int register_netdevice(struct net_device *dev)
if (dev->features & NETIF_F_SG) if (dev->features & NETIF_F_SG)
dev->features |= NETIF_F_GSO; dev->features |= NETIF_F_GSO;
netdev_initialize_kobject(dev);
ret = call_netdevice_notifiers(NETDEV_POST_INIT, dev); ret = call_netdevice_notifiers(NETDEV_POST_INIT, dev);
ret = notifier_to_errno(ret); ret = notifier_to_errno(ret);
if (ret) if (ret)
...@@ -5547,15 +5540,6 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char ...@@ -5547,15 +5540,6 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char
if (dev->features & NETIF_F_NETNS_LOCAL) if (dev->features & NETIF_F_NETNS_LOCAL)
goto out; goto out;
#ifdef CONFIG_SYSFS
/* Don't allow real devices to be moved when sysfs
* is enabled.
*/
err = -EINVAL;
if (dev->dev.parent)
goto out;
#endif
/* Ensure the device has been registrered */ /* Ensure the device has been registrered */
err = -EINVAL; err = -EINVAL;
if (dev->reg_state != NETREG_REGISTERED) if (dev->reg_state != NETREG_REGISTERED)
...@@ -5606,8 +5590,6 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char ...@@ -5606,8 +5590,6 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char
dev_uc_flush(dev); dev_uc_flush(dev);
dev_mc_flush(dev); dev_mc_flush(dev);
netdev_unregister_kobject(dev);
/* Actually switch the network namespace */ /* Actually switch the network namespace */
dev_net_set(dev, net); dev_net_set(dev, net);
...@@ -5620,7 +5602,7 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char ...@@ -5620,7 +5602,7 @@ int dev_change_net_namespace(struct net_device *dev, struct net *net, const char
} }
/* Fixup kobjects */ /* Fixup kobjects */
err = netdev_register_kobject(dev); err = device_rename(&dev->dev, dev->name);
WARN_ON(err); WARN_ON(err);
/* Add the device back in the hashes */ /* Add the device back in the hashes */
......
...@@ -808,9 +808,6 @@ static int netdev_uevent(struct device *d, struct kobj_uevent_env *env) ...@@ -808,9 +808,6 @@ static int netdev_uevent(struct device *d, struct kobj_uevent_env *env)
struct net_device *dev = to_net_dev(d); struct net_device *dev = to_net_dev(d);
int retval; int retval;
if (!net_eq(dev_net(dev), &init_net))
return 0;
/* pass interface to uevent. */ /* pass interface to uevent. */
retval = add_uevent_var(env, "INTERFACE=%s", dev->name); retval = add_uevent_var(env, "INTERFACE=%s", dev->name);
if (retval) if (retval)
...@@ -869,9 +866,6 @@ void netdev_unregister_kobject(struct net_device * net) ...@@ -869,9 +866,6 @@ void netdev_unregister_kobject(struct net_device * net)
kobject_get(&dev->kobj); kobject_get(&dev->kobj);
if (!net_eq(dev_net(net), &init_net))
return;
#ifdef CONFIG_RPS #ifdef CONFIG_RPS
rx_queue_remove_kobjects(net); rx_queue_remove_kobjects(net);
#endif #endif
...@@ -886,6 +880,7 @@ int netdev_register_kobject(struct net_device *net) ...@@ -886,6 +880,7 @@ int netdev_register_kobject(struct net_device *net)
const struct attribute_group **groups = net->sysfs_groups; const struct attribute_group **groups = net->sysfs_groups;
int error = 0; int error = 0;
device_initialize(dev);
dev->class = &net_class; dev->class = &net_class;
dev->platform_data = net; dev->platform_data = net;
dev->groups = groups; dev->groups = groups;
...@@ -908,9 +903,6 @@ int netdev_register_kobject(struct net_device *net) ...@@ -908,9 +903,6 @@ int netdev_register_kobject(struct net_device *net)
#endif #endif
#endif /* CONFIG_SYSFS */ #endif /* CONFIG_SYSFS */
if (!net_eq(dev_net(net), &init_net))
return 0;
error = device_add(dev); error = device_add(dev);
if (error) if (error)
return error; return error;
...@@ -939,12 +931,6 @@ void netdev_class_remove_file(struct class_attribute *class_attr) ...@@ -939,12 +931,6 @@ void netdev_class_remove_file(struct class_attribute *class_attr)
EXPORT_SYMBOL(netdev_class_create_file); EXPORT_SYMBOL(netdev_class_create_file);
EXPORT_SYMBOL(netdev_class_remove_file); EXPORT_SYMBOL(netdev_class_remove_file);
void netdev_initialize_kobject(struct net_device *net)
{
struct device *device = &(net->dev);
device_initialize(device);
}
int netdev_kobject_init(void) int netdev_kobject_init(void)
{ {
kobj_ns_type_register(&net_ns_type_operations); kobj_ns_type_register(&net_ns_type_operations);
......
...@@ -4,5 +4,4 @@ ...@@ -4,5 +4,4 @@
int netdev_kobject_init(void); int netdev_kobject_init(void);
int netdev_register_kobject(struct net_device *); int netdev_register_kobject(struct net_device *);
void netdev_unregister_kobject(struct net_device *); void netdev_unregister_kobject(struct net_device *);
void netdev_initialize_kobject(struct net_device *);
#endif #endif
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment