aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/net_namespace.h9
-rw-r--r--net/sysctl_net.c57
2 files changed, 66 insertions, 0 deletions
diff --git a/include/net/net_namespace.h b/include/net/net_namespace.h
index 4d0d6349aa7e..235214c4c231 100644
--- a/include/net/net_namespace.h
+++ b/include/net/net_namespace.h
@@ -25,6 +25,8 @@ struct net {
25 struct proc_dir_entry *proc_net_stat; 25 struct proc_dir_entry *proc_net_stat;
26 struct proc_dir_entry *proc_net_root; 26 struct proc_dir_entry *proc_net_root;
27 27
28 struct list_head sysctl_table_headers;
29
28 struct net_device *loopback_dev; /* The loopback */ 30 struct net_device *loopback_dev; /* The loopback */
29 31
30 struct list_head dev_base_head; 32 struct list_head dev_base_head;
@@ -144,4 +146,11 @@ extern void unregister_pernet_subsys(struct pernet_operations *);
144extern int register_pernet_device(struct pernet_operations *); 146extern int register_pernet_device(struct pernet_operations *);
145extern void unregister_pernet_device(struct pernet_operations *); 147extern void unregister_pernet_device(struct pernet_operations *);
146 148
149struct ctl_path;
150struct ctl_table;
151struct ctl_table_header;
152extern struct ctl_table_header *register_net_sysctl_table(struct net *net,
153 const struct ctl_path *path, struct ctl_table *table);
154extern void unregister_net_sysctl_table(struct ctl_table_header *header);
155
147#endif /* __NET_NET_NAMESPACE_H */ 156#endif /* __NET_NET_NAMESPACE_H */
diff --git a/net/sysctl_net.c b/net/sysctl_net.c
index cd4eafbab1b8..c50c793aa7f0 100644
--- a/net/sysctl_net.c
+++ b/net/sysctl_net.c
@@ -14,6 +14,7 @@
14 14
15#include <linux/mm.h> 15#include <linux/mm.h>
16#include <linux/sysctl.h> 16#include <linux/sysctl.h>
17#include <linux/nsproxy.h>
17 18
18#include <net/sock.h> 19#include <net/sock.h>
19 20
@@ -54,3 +55,59 @@ struct ctl_table net_table[] = {
54#endif 55#endif
55 { 0 }, 56 { 0 },
56}; 57};
58
59static struct list_head *
60net_ctl_header_lookup(struct ctl_table_root *root, struct nsproxy *namespaces)
61{
62 return &namespaces->net_ns->sysctl_table_headers;
63}
64
65static struct ctl_table_root net_sysctl_root = {
66 .lookup = net_ctl_header_lookup,
67};
68
69static int sysctl_net_init(struct net *net)
70{
71 INIT_LIST_HEAD(&net->sysctl_table_headers);
72 return 0;
73}
74
75static void sysctl_net_exit(struct net *net)
76{
77 WARN_ON(!list_empty(&net->sysctl_table_headers));
78 return;
79}
80
81static struct pernet_operations sysctl_pernet_ops = {
82 .init = sysctl_net_init,
83 .exit = sysctl_net_exit,
84};
85
86static __init int sysctl_init(void)
87{
88 int ret;
89 ret = register_pernet_subsys(&sysctl_pernet_ops);
90 if (ret)
91 goto out;
92 register_sysctl_root(&net_sysctl_root);
93out:
94 return ret;
95}
96subsys_initcall(sysctl_init);
97
98struct ctl_table_header *register_net_sysctl_table(struct net *net,
99 const struct ctl_path *path, struct ctl_table *table)
100{
101 struct nsproxy namespaces;
102 namespaces = *current->nsproxy;
103 namespaces.net_ns = net;
104 return __register_sysctl_paths(&net_sysctl_root,
105 &namespaces, path, table);
106}
107EXPORT_SYMBOL_GPL(register_net_sysctl_table);
108
109void unregister_net_sysctl_table(struct ctl_table_header *header)
110{
111 return unregister_sysctl_table(header);
112}
113EXPORT_SYMBOL_GPL(unregister_net_sysctl_table);