Commit 4665079c authored by Pavel Emelyanov's avatar Pavel Emelyanov Committed by David S. Miller

[NETNS]: Move some code into __init section when CONFIG_NET_NS=n

With the net namespaces many code leaved the __init section,
thus making the kernel occupy more memory than it did before.
Since we have a config option that prohibits the namespace
creation, the functions that initialize/finalize some netns
stuff are simply not needed and can be freed after the boot.

Currently, this is almost not noticeable, since few calls
are no longer in __init, but when the namespaces will be
merged it will be possible to free more code. I propose to
use the __net_init, __net_exit and __net_initdata "attributes"
for functions/variables that are not used if the CONFIG_NET_NS
is not set to save more space in memory.

The exiting functions cannot just reside in the __exit section,
as noticed by David, since the init section will have
references on it and the compilation will fail due to modpost
checks. These references can exist, since the init namespace
never dies and the exit callbacks are never called. So I
introduce the __exit_refok attribute just like it is already
done with the __init_refok.
Signed-off-by: default avatarPavel Emelyanov <xemul@openvz.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent d62a38d1
...@@ -250,7 +250,7 @@ static void loopback_setup(struct net_device *dev) ...@@ -250,7 +250,7 @@ static void loopback_setup(struct net_device *dev)
} }
/* Setup and register the loopback device. */ /* Setup and register the loopback device. */
static int loopback_net_init(struct net *net) static __net_init int loopback_net_init(struct net *net)
{ {
struct net_device *dev; struct net_device *dev;
int err; int err;
...@@ -278,14 +278,14 @@ out_free_netdev: ...@@ -278,14 +278,14 @@ out_free_netdev:
goto out; goto out;
} }
static void loopback_net_exit(struct net *net) static __net_exit void loopback_net_exit(struct net *net)
{ {
struct net_device *dev = net->loopback_dev; struct net_device *dev = net->loopback_dev;
unregister_netdev(dev); unregister_netdev(dev);
} }
static struct pernet_operations loopback_net_ops = { static struct pernet_operations __net_initdata loopback_net_ops = {
.init = loopback_net_init, .init = loopback_net_init,
.exit = loopback_net_exit, .exit = loopback_net_exit,
}; };
......
...@@ -140,7 +140,7 @@ static struct inode_operations proc_net_dir_inode_operations = { ...@@ -140,7 +140,7 @@ static struct inode_operations proc_net_dir_inode_operations = {
.setattr = proc_net_setattr, .setattr = proc_net_setattr,
}; };
static int proc_net_ns_init(struct net *net) static __net_init int proc_net_ns_init(struct net *net)
{ {
struct proc_dir_entry *root, *netd, *net_statd; struct proc_dir_entry *root, *netd, *net_statd;
int err; int err;
...@@ -178,19 +178,19 @@ free_root: ...@@ -178,19 +178,19 @@ free_root:
goto out; goto out;
} }
static void proc_net_ns_exit(struct net *net) static __net_exit void proc_net_ns_exit(struct net *net)
{ {
remove_proc_entry("stat", net->proc_net); remove_proc_entry("stat", net->proc_net);
remove_proc_entry("net", net->proc_net_root); remove_proc_entry("net", net->proc_net_root);
kfree(net->proc_net_root); kfree(net->proc_net_root);
} }
struct pernet_operations proc_net_ns_ops = { struct pernet_operations __net_initdata proc_net_ns_ops = {
.init = proc_net_ns_init, .init = proc_net_ns_init,
.exit = proc_net_ns_exit, .exit = proc_net_ns_exit,
}; };
int proc_net_init(void) int __init proc_net_init(void)
{ {
proc_net_shadow = proc_mkdir("net", NULL); proc_net_shadow = proc_mkdir("net", NULL);
proc_net_shadow->proc_iops = &proc_net_dir_inode_operations; proc_net_shadow->proc_iops = &proc_net_dir_inode_operations;
......
...@@ -57,6 +57,7 @@ ...@@ -57,6 +57,7 @@
* The markers follow same syntax rules as __init / __initdata. */ * The markers follow same syntax rules as __init / __initdata. */
#define __init_refok noinline __attribute__ ((__section__ (".text.init.refok"))) #define __init_refok noinline __attribute__ ((__section__ (".text.init.refok")))
#define __initdata_refok __attribute__ ((__section__ (".data.init.refok"))) #define __initdata_refok __attribute__ ((__section__ (".data.init.refok")))
#define __exit_refok noinline __attribute__ ((__section__ (".exit.text.refok")))
#ifdef MODULE #ifdef MODULE
#define __exit __attribute__ ((__section__(".exit.text"))) __cold #define __exit __attribute__ ((__section__(".exit.text"))) __cold
......
...@@ -99,6 +99,15 @@ static inline void release_net(struct net *net) ...@@ -99,6 +99,15 @@ static inline void release_net(struct net *net)
#define for_each_net(VAR) \ #define for_each_net(VAR) \
list_for_each_entry(VAR, &net_namespace_list, list) list_for_each_entry(VAR, &net_namespace_list, list)
#ifdef CONFIG_NET_NS
#define __net_init
#define __net_exit
#define __net_initdata
#else
#define __net_init __init
#define __net_exit __exit_refok
#define __net_initdata __initdata
#endif
struct pernet_operations { struct pernet_operations {
struct list_head list; struct list_head list;
......
...@@ -2611,7 +2611,7 @@ static const struct file_operations ptype_seq_fops = { ...@@ -2611,7 +2611,7 @@ static const struct file_operations ptype_seq_fops = {
}; };
static int dev_proc_net_init(struct net *net) static int __net_init dev_proc_net_init(struct net *net)
{ {
int rc = -ENOMEM; int rc = -ENOMEM;
...@@ -2636,7 +2636,7 @@ out_dev: ...@@ -2636,7 +2636,7 @@ out_dev:
goto out; goto out;
} }
static void dev_proc_net_exit(struct net *net) static void __net_exit dev_proc_net_exit(struct net *net)
{ {
wext_proc_exit(net); wext_proc_exit(net);
...@@ -2645,7 +2645,7 @@ static void dev_proc_net_exit(struct net *net) ...@@ -2645,7 +2645,7 @@ static void dev_proc_net_exit(struct net *net)
proc_net_remove(net, "dev"); proc_net_remove(net, "dev");
} }
static struct pernet_operations dev_proc_ops = { static struct pernet_operations __net_initdata dev_proc_ops = {
.init = dev_proc_net_init, .init = dev_proc_net_init,
.exit = dev_proc_net_exit, .exit = dev_proc_net_exit,
}; };
...@@ -4278,7 +4278,7 @@ static struct hlist_head *netdev_create_hash(void) ...@@ -4278,7 +4278,7 @@ static struct hlist_head *netdev_create_hash(void)
} }
/* Initialize per network namespace state */ /* Initialize per network namespace state */
static int netdev_init(struct net *net) static int __net_init netdev_init(struct net *net)
{ {
INIT_LIST_HEAD(&net->dev_base_head); INIT_LIST_HEAD(&net->dev_base_head);
rwlock_init(&dev_base_lock); rwlock_init(&dev_base_lock);
...@@ -4299,18 +4299,18 @@ err_name: ...@@ -4299,18 +4299,18 @@ err_name:
return -ENOMEM; return -ENOMEM;
} }
static void netdev_exit(struct net *net) static void __net_exit netdev_exit(struct net *net)
{ {
kfree(net->dev_name_head); kfree(net->dev_name_head);
kfree(net->dev_index_head); kfree(net->dev_index_head);
} }
static struct pernet_operations netdev_net_ops = { static struct pernet_operations __net_initdata netdev_net_ops = {
.init = netdev_init, .init = netdev_init,
.exit = netdev_exit, .exit = netdev_exit,
}; };
static void default_device_exit(struct net *net) static void __net_exit default_device_exit(struct net *net)
{ {
struct net_device *dev, *next; struct net_device *dev, *next;
/* /*
...@@ -4336,7 +4336,7 @@ static void default_device_exit(struct net *net) ...@@ -4336,7 +4336,7 @@ static void default_device_exit(struct net *net)
rtnl_unlock(); rtnl_unlock();
} }
static struct pernet_operations default_device_ops = { static struct pernet_operations __net_initdata default_device_ops = {
.exit = default_device_exit, .exit = default_device_exit,
}; };
......
...@@ -273,19 +273,19 @@ static const struct file_operations dev_mc_seq_fops = { ...@@ -273,19 +273,19 @@ static const struct file_operations dev_mc_seq_fops = {
#endif #endif
static int dev_mc_net_init(struct net *net) static int __net_init dev_mc_net_init(struct net *net)
{ {
if (!proc_net_fops_create(net, "dev_mcast", 0, &dev_mc_seq_fops)) if (!proc_net_fops_create(net, "dev_mcast", 0, &dev_mc_seq_fops))
return -ENOMEM; return -ENOMEM;
return 0; return 0;
} }
static void dev_mc_net_exit(struct net *net) static void __net_exit dev_mc_net_exit(struct net *net)
{ {
proc_net_remove(net, "dev_mcast"); proc_net_remove(net, "dev_mcast");
} }
static struct pernet_operations dev_mc_net_ops = { static struct pernet_operations __net_initdata dev_mc_net_ops = {
.init = dev_mc_net_init, .init = dev_mc_net_init,
.exit = dev_mc_net_exit, .exit = dev_mc_net_exit,
}; };
......
...@@ -1924,7 +1924,7 @@ static struct net_proto_family netlink_family_ops = { ...@@ -1924,7 +1924,7 @@ static struct net_proto_family netlink_family_ops = {
.owner = THIS_MODULE, /* for consistency 8) */ .owner = THIS_MODULE, /* for consistency 8) */
}; };
static int netlink_net_init(struct net *net) static int __net_init netlink_net_init(struct net *net)
{ {
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
if (!proc_net_fops_create(net, "netlink", 0, &netlink_seq_fops)) if (!proc_net_fops_create(net, "netlink", 0, &netlink_seq_fops))
...@@ -1933,14 +1933,14 @@ static int netlink_net_init(struct net *net) ...@@ -1933,14 +1933,14 @@ static int netlink_net_init(struct net *net)
return 0; return 0;
} }
static void netlink_net_exit(struct net *net) static void __net_exit netlink_net_exit(struct net *net)
{ {
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
proc_net_remove(net, "netlink"); proc_net_remove(net, "netlink");
#endif #endif
} }
static struct pernet_operations netlink_net_ops = { static struct pernet_operations __net_initdata netlink_net_ops = {
.init = netlink_net_init, .init = netlink_net_init,
.exit = netlink_net_exit, .exit = netlink_net_exit,
}; };
......
...@@ -709,6 +709,7 @@ static int secref_whitelist(const char *modname, const char *tosec, ...@@ -709,6 +709,7 @@ static int secref_whitelist(const char *modname, const char *tosec,
/* Check for pattern 0 */ /* Check for pattern 0 */
if ((strncmp(fromsec, ".text.init.refok", strlen(".text.init.refok")) == 0) || if ((strncmp(fromsec, ".text.init.refok", strlen(".text.init.refok")) == 0) ||
(strncmp(fromsec, ".exit.text.refok", strlen(".exit.text.refok")) == 0) ||
(strncmp(fromsec, ".data.init.refok", strlen(".data.init.refok")) == 0)) (strncmp(fromsec, ".data.init.refok", strlen(".data.init.refok")) == 0))
return 1; return 1;
......
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