aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/net_namespace.h3
-rw-r--r--net/unix/af_unix.c9
-rw-r--r--net/unix/sysctl_net_unix.c27
3 files changed, 31 insertions, 8 deletions
diff --git a/include/net/net_namespace.h b/include/net/net_namespace.h
index b0cf07519b81..f97b2a4469ae 100644
--- a/include/net/net_namespace.h
+++ b/include/net/net_namespace.h
@@ -11,6 +11,8 @@
11struct proc_dir_entry; 11struct proc_dir_entry;
12struct net_device; 12struct net_device;
13struct sock; 13struct sock;
14struct ctl_table_header;
15
14struct net { 16struct net {
15 atomic_t count; /* To decided when the network 17 atomic_t count; /* To decided when the network
16 * namespace should be freed. 18 * namespace should be freed.
@@ -41,6 +43,7 @@ struct net {
41 43
42 /* unix sockets */ 44 /* unix sockets */
43 int sysctl_unix_max_dgram_qlen; 45 int sysctl_unix_max_dgram_qlen;
46 struct ctl_table_header *unix_ctl;
44}; 47};
45 48
46#ifdef CONFIG_NET 49#ifdef CONFIG_NET
diff --git a/net/unix/af_unix.c b/net/unix/af_unix.c
index 73620d61762b..b8a2189fb5c0 100644
--- a/net/unix/af_unix.c
+++ b/net/unix/af_unix.c
@@ -2139,10 +2139,14 @@ static int unix_net_init(struct net *net)
2139 int error = -ENOMEM; 2139 int error = -ENOMEM;
2140 2140
2141 net->sysctl_unix_max_dgram_qlen = 10; 2141 net->sysctl_unix_max_dgram_qlen = 10;
2142 if (unix_sysctl_register(net))
2143 goto out;
2142 2144
2143#ifdef CONFIG_PROC_FS 2145#ifdef CONFIG_PROC_FS
2144 if (!proc_net_fops_create(net, "unix", 0, &unix_seq_fops)) 2146 if (!proc_net_fops_create(net, "unix", 0, &unix_seq_fops)) {
2147 unix_sysctl_unregister(net);
2145 goto out; 2148 goto out;
2149 }
2146#endif 2150#endif
2147 error = 0; 2151 error = 0;
2148out: 2152out:
@@ -2151,6 +2155,7 @@ out:
2151 2155
2152static void unix_net_exit(struct net *net) 2156static void unix_net_exit(struct net *net)
2153{ 2157{
2158 unix_sysctl_unregister(net);
2154 proc_net_remove(net, "unix"); 2159 proc_net_remove(net, "unix");
2155} 2160}
2156 2161
@@ -2175,7 +2180,6 @@ static int __init af_unix_init(void)
2175 2180
2176 sock_register(&unix_family_ops); 2181 sock_register(&unix_family_ops);
2177 register_pernet_subsys(&unix_net_ops); 2182 register_pernet_subsys(&unix_net_ops);
2178 unix_sysctl_register(&init_net);
2179out: 2183out:
2180 return rc; 2184 return rc;
2181} 2185}
@@ -2183,7 +2187,6 @@ out:
2183static void __exit af_unix_exit(void) 2187static void __exit af_unix_exit(void)
2184{ 2188{
2185 sock_unregister(PF_UNIX); 2189 sock_unregister(PF_UNIX);
2186 unix_sysctl_unregister(&init_net);
2187 proto_unregister(&unix_proto); 2190 proto_unregister(&unix_proto);
2188 unregister_pernet_subsys(&unix_net_ops); 2191 unregister_pernet_subsys(&unix_net_ops);
2189} 2192}
diff --git a/net/unix/sysctl_net_unix.c b/net/unix/sysctl_net_unix.c
index 43dd356730fd..553ef6a487d0 100644
--- a/net/unix/sysctl_net_unix.c
+++ b/net/unix/sysctl_net_unix.c
@@ -32,16 +32,33 @@ static struct ctl_path unix_path[] = {
32 { }, 32 { },
33}; 33};
34 34
35static struct ctl_table_header * unix_sysctl_header;
36
37int unix_sysctl_register(struct net *net) 35int unix_sysctl_register(struct net *net)
38{ 36{
39 unix_sysctl_header = register_sysctl_paths(unix_path, unix_table); 37 struct ctl_table *table;
40 return unix_sysctl_header == NULL ? -ENOMEM : 0; 38
39 table = kmemdup(unix_table, sizeof(unix_table), GFP_KERNEL);
40 if (table == NULL)
41 goto err_alloc;
42
43 table[0].data = &net->sysctl_unix_max_dgram_qlen;
44 net->unix_ctl = register_net_sysctl_table(net, unix_path, table);
45 if (net->unix_ctl == NULL)
46 goto err_reg;
47
48 return 0;
49
50err_reg:
51 kfree(table);
52err_alloc:
53 return -ENOMEM;
41} 54}
42 55
43void unix_sysctl_unregister(struct net *net) 56void unix_sysctl_unregister(struct net *net)
44{ 57{
45 unregister_sysctl_table(unix_sysctl_header); 58 struct ctl_table *table;
59
60 table = net->unix_ctl->ctl_table_arg;
61 unregister_sysctl_table(net->unix_ctl);
62 kfree(table);
46} 63}
47 64