sysctl_net.c 3.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/* -*- linux-c -*-
 * sysctl_net.c: sysctl interface to net subsystem.
 *
 * Begun April 1, 1996, Mike Shaver.
 * Added /proc/sys/net directories for each protocol family. [MS]
 *
 * Revision 1.2  1996/05/08  20:24:40  shaver
 * Added bits for NET_BRIDGE and the NET_IPV4_ARP stuff and
 * NET_IPV4_IP_FORWARD.
 *
 *
 */

#include <linux/mm.h>
#include <linux/sysctl.h>
16
#include <linux/nsproxy.h>
L
Linus Torvalds 已提交
17

18 19
#include <net/sock.h>

L
Linus Torvalds 已提交
20
#ifdef CONFIG_INET
21
#include <net/ip.h>
L
Linus Torvalds 已提交
22 23 24
#endif

#ifdef CONFIG_NET
25
#include <linux/if_ether.h>
L
Linus Torvalds 已提交
26 27 28
#endif

#ifdef CONFIG_TR
29
#include <linux/if_tr.h>
L
Linus Torvalds 已提交
30 31
#endif

32
static struct ctl_table_set *
33 34
net_ctl_header_lookup(struct ctl_table_root *root, struct nsproxy *namespaces)
{
35 36 37 38 39 40
	return &namespaces->net_ns->sysctls;
}

static int is_seen(struct ctl_table_set *set)
{
	return &current->nsproxy->net_ns->sysctls == set;
41 42
}

43 44 45 46 47 48 49 50 51 52 53 54 55
/* Return standard mode bits for table entry. */
static int net_ctl_permissions(struct ctl_table_root *root,
			       struct nsproxy *nsproxy,
			       struct ctl_table *table)
{
	/* Allow network administrator to have same access as root. */
	if (capable(CAP_NET_ADMIN)) {
		int mode = (table->mode >> 6) & 7;
		return (mode << 6) | (mode << 3) | mode;
	}
	return table->mode;
}

56 57
static struct ctl_table_root net_sysctl_root = {
	.lookup = net_ctl_header_lookup,
58
	.permissions = net_ctl_permissions,
59 60
};

61 62 63 64 65 66 67 68 69 70 71 72 73
static int net_ctl_ro_header_perms(struct ctl_table_root *root,
		struct nsproxy *namespaces, struct ctl_table *table)
{
	if (namespaces->net_ns == &init_net)
		return table->mode;
	else
		return table->mode & ~0222;
}

static struct ctl_table_root net_sysctl_ro_root = {
	.permissions = net_ctl_ro_header_perms,
};

74 75
static int sysctl_net_init(struct net *net)
{
76
	setup_sysctl_set(&net->sysctls, NULL, is_seen);
77 78 79 80 81
	return 0;
}

static void sysctl_net_exit(struct net *net)
{
82
	WARN_ON(!list_empty(&net->sysctls.list));
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
	return;
}

static struct pernet_operations sysctl_pernet_ops = {
	.init = sysctl_net_init,
	.exit = sysctl_net_exit,
};

static __init int sysctl_init(void)
{
	int ret;
	ret = register_pernet_subsys(&sysctl_pernet_ops);
	if (ret)
		goto out;
	register_sysctl_root(&net_sysctl_root);
98
	setup_sysctl_set(&net_sysctl_ro_root.default_set, NULL, NULL);
99
	register_sysctl_root(&net_sysctl_ro_root);
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
out:
	return ret;
}
subsys_initcall(sysctl_init);

struct ctl_table_header *register_net_sysctl_table(struct net *net,
	const struct ctl_path *path, struct ctl_table *table)
{
	struct nsproxy namespaces;
	namespaces = *current->nsproxy;
	namespaces.net_ns = net;
	return __register_sysctl_paths(&net_sysctl_root,
					&namespaces, path, table);
}
EXPORT_SYMBOL_GPL(register_net_sysctl_table);

116 117 118 119 120 121 122 123
struct ctl_table_header *register_net_sysctl_rotable(const
		struct ctl_path *path, struct ctl_table *table)
{
	return __register_sysctl_paths(&net_sysctl_ro_root,
			&init_nsproxy, path, table);
}
EXPORT_SYMBOL_GPL(register_net_sysctl_rotable);

124 125
void unregister_net_sysctl_table(struct ctl_table_header *header)
{
126
	unregister_sysctl_table(header);
127 128
}
EXPORT_SYMBOL_GPL(unregister_net_sysctl_table);