fib_frontend.c 27.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 * INET		An implementation of the TCP/IP protocol suite for the LINUX
 *		operating system.  INET is implemented using the  BSD Socket
 *		interface as the means of communication with the user level.
 *
 *		IPv4 Forwarding Information Base: FIB frontend.
 *
 * Authors:	Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
 *
 *		This program is free software; you can redistribute it and/or
 *		modify it under the terms of the GNU General Public License
 *		as published by the Free Software Foundation; either version
 *		2 of the License, or (at your option) any later version.
 */

#include <linux/module.h>
#include <asm/uaccess.h>
#include <linux/bitops.h>
19
#include <linux/capability.h>
L
Linus Torvalds 已提交
20 21 22 23 24 25 26 27 28
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/string.h>
#include <linux/socket.h>
#include <linux/sockios.h>
#include <linux/errno.h>
#include <linux/in.h>
#include <linux/inet.h>
29
#include <linux/inetdevice.h>
L
Linus Torvalds 已提交
30
#include <linux/netdevice.h>
31
#include <linux/if_addr.h>
L
Linus Torvalds 已提交
32 33
#include <linux/if_arp.h>
#include <linux/skbuff.h>
34
#include <linux/cache.h>
L
Linus Torvalds 已提交
35
#include <linux/init.h>
36
#include <linux/list.h>
37
#include <linux/slab.h>
L
Linus Torvalds 已提交
38 39 40 41 42 43 44 45

#include <net/ip.h>
#include <net/protocol.h>
#include <net/route.h>
#include <net/tcp.h>
#include <net/sock.h>
#include <net/arp.h>
#include <net/ip_fib.h>
46
#include <net/rtnetlink.h>
47
#include <net/xfrm.h>
L
Linus Torvalds 已提交
48 49 50

#ifndef CONFIG_IP_MULTIPLE_TABLES

51
static int __net_init fib4_rules_init(struct net *net)
52
{
53 54
	struct fib_table *local_table, *main_table;

55
	local_table = fib_trie_table(RT_TABLE_LOCAL);
56
	if (local_table == NULL)
57 58
		return -ENOMEM;

59
	main_table  = fib_trie_table(RT_TABLE_MAIN);
60
	if (main_table == NULL)
61 62
		goto fail;

63
	hlist_add_head_rcu(&local_table->tb_hlist,
64
				&net->ipv4.fib_table_hash[TABLE_LOCAL_INDEX]);
65
	hlist_add_head_rcu(&main_table->tb_hlist,
66
				&net->ipv4.fib_table_hash[TABLE_MAIN_INDEX]);
67 68 69
	return 0;

fail:
70
	kfree(local_table);
71
	return -ENOMEM;
72
}
73
#else
L
Linus Torvalds 已提交
74

75
struct fib_table *fib_new_table(struct net *net, u32 id)
L
Linus Torvalds 已提交
76 77
{
	struct fib_table *tb;
78
	unsigned int h;
L
Linus Torvalds 已提交
79

80 81
	if (id == 0)
		id = RT_TABLE_MAIN;
82
	tb = fib_get_table(net, id);
83 84
	if (tb)
		return tb;
85

86
	tb = fib_trie_table(id);
L
Linus Torvalds 已提交
87 88
	if (!tb)
		return NULL;
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106

	switch (id) {
	case RT_TABLE_LOCAL:
		net->ipv4.fib_local = tb;
		break;

	case RT_TABLE_MAIN:
		net->ipv4.fib_main = tb;
		break;

	case RT_TABLE_DEFAULT:
		net->ipv4.fib_default = tb;
		break;

	default:
		break;
	}

107
	h = id & (FIB_TABLE_HASHSZ - 1);
108
	hlist_add_head_rcu(&tb->tb_hlist, &net->ipv4.fib_table_hash[h]);
L
Linus Torvalds 已提交
109 110 111
	return tb;
}

112
struct fib_table *fib_get_table(struct net *net, u32 id)
113 114 115
{
	struct fib_table *tb;
	struct hlist_node *node;
116
	struct hlist_head *head;
117
	unsigned int h;
L
Linus Torvalds 已提交
118

119 120 121
	if (id == 0)
		id = RT_TABLE_MAIN;
	h = id & (FIB_TABLE_HASHSZ - 1);
122

123
	rcu_read_lock();
124 125
	head = &net->ipv4.fib_table_hash[h];
	hlist_for_each_entry_rcu(tb, node, head, tb_hlist) {
126 127 128 129 130 131 132 133
		if (tb->tb_id == id) {
			rcu_read_unlock();
			return tb;
		}
	}
	rcu_read_unlock();
	return NULL;
}
L
Linus Torvalds 已提交
134 135
#endif /* CONFIG_IP_MULTIPLE_TABLES */

136
static void fib_flush(struct net *net)
L
Linus Torvalds 已提交
137 138 139
{
	int flushed = 0;
	struct fib_table *tb;
140
	struct hlist_node *node;
141
	struct hlist_head *head;
142
	unsigned int h;
L
Linus Torvalds 已提交
143

144
	for (h = 0; h < FIB_TABLE_HASHSZ; h++) {
145 146
		head = &net->ipv4.fib_table_hash[h];
		hlist_for_each_entry(tb, node, head, tb_hlist)
147
			flushed += fib_table_flush(tb);
L
Linus Torvalds 已提交
148 149 150
	}

	if (flushed)
151
		rt_cache_flush(net);
L
Linus Torvalds 已提交
152 153
}

154 155 156 157
/*
 * Find address type as if only "dev" was present in the system. If
 * on_dev is NULL then all interfaces are taken into consideration.
 */
158 159 160
static inline unsigned int __inet_dev_addr_type(struct net *net,
						const struct net_device *dev,
						__be32 addr)
L
Linus Torvalds 已提交
161
{
D
David S. Miller 已提交
162
	struct flowi4		fl4 = { .daddr = addr };
L
Linus Torvalds 已提交
163
	struct fib_result	res;
164
	unsigned int ret = RTN_BROADCAST;
165
	struct fib_table *local_table;
L
Linus Torvalds 已提交
166

167
	if (ipv4_is_zeronet(addr) || ipv4_is_lbcast(addr))
L
Linus Torvalds 已提交
168
		return RTN_BROADCAST;
169
	if (ipv4_is_multicast(addr))
L
Linus Torvalds 已提交
170 171
		return RTN_MULTICAST;

172
	local_table = fib_get_table(net, RT_TABLE_LOCAL);
173
	if (local_table) {
L
Linus Torvalds 已提交
174
		ret = RTN_UNICAST;
E
Eric Dumazet 已提交
175
		rcu_read_lock();
D
David S. Miller 已提交
176
		if (!fib_table_lookup(local_table, &fl4, &res, FIB_LOOKUP_NOREF)) {
177 178
			if (!dev || dev == res.fi->fib_dev)
				ret = res.type;
L
Linus Torvalds 已提交
179
		}
E
Eric Dumazet 已提交
180
		rcu_read_unlock();
L
Linus Torvalds 已提交
181 182 183 184
	}
	return ret;
}

185
unsigned int inet_addr_type(struct net *net, __be32 addr)
186
{
187
	return __inet_dev_addr_type(net, NULL, addr);
188
}
E
Eric Dumazet 已提交
189
EXPORT_SYMBOL(inet_addr_type);
190

191 192
unsigned int inet_dev_addr_type(struct net *net, const struct net_device *dev,
				__be32 addr)
193
{
E
Eric Dumazet 已提交
194
	return __inet_dev_addr_type(net, dev, addr);
195
}
E
Eric Dumazet 已提交
196
EXPORT_SYMBOL(inet_dev_addr_type);
197

198 199 200 201 202
__be32 fib_compute_spec_dst(struct sk_buff *skb)
{
	struct net_device *dev = skb->dev;
	struct in_device *in_dev;
	struct fib_result res;
203
	struct rtable *rt;
204 205
	struct flowi4 fl4;
	struct net *net;
206
	int scope;
207

208
	rt = skb_rtable(skb);
209 210
	if ((rt->rt_flags & (RTCF_BROADCAST | RTCF_MULTICAST | RTCF_LOCAL)) ==
	    RTCF_LOCAL)
211 212 213 214 215 216
		return ip_hdr(skb)->daddr;

	in_dev = __in_dev_get_rcu(dev);
	BUG_ON(!in_dev);

	net = dev_net(dev);
217 218 219 220

	scope = RT_SCOPE_UNIVERSE;
	if (!ipv4_is_zeronet(ip_hdr(skb)->saddr)) {
		fl4.flowi4_oif = 0;
221
		fl4.flowi4_iif = LOOPBACK_IFINDEX;
222 223 224 225 226 227 228 229 230 231 232 233
		fl4.daddr = ip_hdr(skb)->saddr;
		fl4.saddr = 0;
		fl4.flowi4_tos = RT_TOS(ip_hdr(skb)->tos);
		fl4.flowi4_scope = scope;
		fl4.flowi4_mark = IN_DEV_SRC_VMARK(in_dev) ? skb->mark : 0;
		if (!fib_lookup(net, &fl4, &res))
			return FIB_RES_PREFSRC(net, res);
	} else {
		scope = RT_SCOPE_LINK;
	}

	return inet_select_addr(dev, ip_hdr(skb)->saddr, scope);
234 235
}

L
Linus Torvalds 已提交
236
/* Given (packet source, input interface) and optional (dst, oif, tos):
E
Eric Dumazet 已提交
237 238 239 240 241
 * - (main) check, that source is valid i.e. not broadcast or our local
 *   address.
 * - figure out what "logical" interface this packet arrived
 *   and calculate "specific destination" address.
 * - check, that packet arrived from expected physical interface.
E
Eric Dumazet 已提交
242
 * called with rcu_read_lock()
L
Linus Torvalds 已提交
243
 */
244 245 246
static int __fib_validate_source(struct sk_buff *skb, __be32 src, __be32 dst,
				 u8 tos, int oif, struct net_device *dev,
				 int rpf, struct in_device *idev, u32 *itag)
L
Linus Torvalds 已提交
247
{
248
	int ret, no_addr, accept_local;
L
Linus Torvalds 已提交
249
	struct fib_result res;
250
	struct flowi4 fl4;
251
	struct net *net;
252
	bool dev_match;
L
Linus Torvalds 已提交
253

D
David S. Miller 已提交
254 255 256 257 258 259
	fl4.flowi4_oif = 0;
	fl4.flowi4_iif = oif;
	fl4.daddr = src;
	fl4.saddr = dst;
	fl4.flowi4_tos = tos;
	fl4.flowi4_scope = RT_SCOPE_UNIVERSE;
260

261
	no_addr = idev->ifa_list == NULL;
262

263 264
	accept_local = IN_DEV_ACCEPT_LOCAL(idev);
	fl4.flowi4_mark = IN_DEV_SRC_VMARK(idev) ? skb->mark : 0;
L
Linus Torvalds 已提交
265

266
	net = dev_net(dev);
D
David S. Miller 已提交
267
	if (fib_lookup(net, &fl4, &res))
L
Linus Torvalds 已提交
268
		goto last_resort;
269 270
	if (res.type != RTN_UNICAST) {
		if (res.type != RTN_LOCAL || !accept_local)
E
Eric Dumazet 已提交
271
			goto e_inval;
272
	}
L
Linus Torvalds 已提交
273
	fib_combine_itag(itag, &res);
274 275
	dev_match = false;

L
Linus Torvalds 已提交
276
#ifdef CONFIG_IP_ROUTE_MULTIPATH
277 278 279 280 281 282 283 284
	for (ret = 0; ret < res.fi->fib_nhs; ret++) {
		struct fib_nh *nh = &res.fi->fib_nh[ret];

		if (nh->nh_dev == dev) {
			dev_match = true;
			break;
		}
	}
L
Linus Torvalds 已提交
285 286
#else
	if (FIB_RES_DEV(res) == dev)
287
		dev_match = true;
L
Linus Torvalds 已提交
288
#endif
289
	if (dev_match) {
L
Linus Torvalds 已提交
290 291 292 293 294
		ret = FIB_RES_NH(res).nh_scope >= RT_SCOPE_HOST;
		return ret;
	}
	if (no_addr)
		goto last_resort;
295
	if (rpf == 1)
296
		goto e_rpf;
D
David S. Miller 已提交
297
	fl4.flowi4_oif = dev->ifindex;
L
Linus Torvalds 已提交
298 299

	ret = 0;
D
David S. Miller 已提交
300
	if (fib_lookup(net, &fl4, &res) == 0) {
301
		if (res.type == RTN_UNICAST)
L
Linus Torvalds 已提交
302 303 304 305 306 307
			ret = FIB_RES_NH(res).nh_scope >= RT_SCOPE_HOST;
	}
	return ret;

last_resort:
	if (rpf)
308
		goto e_rpf;
L
Linus Torvalds 已提交
309 310 311 312 313
	*itag = 0;
	return 0;

e_inval:
	return -EINVAL;
314 315
e_rpf:
	return -EXDEV;
L
Linus Torvalds 已提交
316 317
}

318 319 320 321 322 323 324
/* Ignore rp_filter for packets protected by IPsec. */
int fib_validate_source(struct sk_buff *skb, __be32 src, __be32 dst,
			u8 tos, int oif, struct net_device *dev,
			struct in_device *idev, u32 *itag)
{
	int r = secpath_exists(skb) ? 0 : IN_DEV_RPFILTER(idev);

J
Julian Anastasov 已提交
325 326
	if (!r && !fib_num_tclassid_users(dev_net(dev)) &&
	    (dev->ifindex != oif || !IN_DEV_TX_REDIRECTS(idev))) {
327 328 329 330 331 332
		*itag = 0;
		return 0;
	}
	return __fib_validate_source(skb, src, dst, tos, oif, dev, r, idev, itag);
}

A
Al Viro 已提交
333
static inline __be32 sk_extract_addr(struct sockaddr *addr)
334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349
{
	return ((struct sockaddr_in *) addr)->sin_addr.s_addr;
}

static int put_rtax(struct nlattr *mx, int len, int type, u32 value)
{
	struct nlattr *nla;

	nla = (struct nlattr *) ((char *) mx + len);
	nla->nla_type = type;
	nla->nla_len = nla_attr_size(4);
	*(u32 *) nla_data(nla) = value;

	return len + nla_total_size(4);
}

350
static int rtentry_to_fib_config(struct net *net, int cmd, struct rtentry *rt,
351 352
				 struct fib_config *cfg)
{
353
	__be32 addr;
354 355 356
	int plen;

	memset(cfg, 0, sizeof(*cfg));
357
	cfg->fc_nlinfo.nl_net = net;
358 359 360 361 362 363 364 365 366 367 368 369 370 371 372

	if (rt->rt_dst.sa_family != AF_INET)
		return -EAFNOSUPPORT;

	/*
	 * Check mask for validity:
	 * a) it must be contiguous.
	 * b) destination must have all host bits clear.
	 * c) if application forgot to set correct family (AF_INET),
	 *    reject request unless it is absolutely clear i.e.
	 *    both family and mask are zero.
	 */
	plen = 32;
	addr = sk_extract_addr(&rt->rt_dst);
	if (!(rt->rt_flags & RTF_HOST)) {
A
Al Viro 已提交
373
		__be32 mask = sk_extract_addr(&rt->rt_genmask);
374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417

		if (rt->rt_genmask.sa_family != AF_INET) {
			if (mask || rt->rt_genmask.sa_family)
				return -EAFNOSUPPORT;
		}

		if (bad_mask(mask, addr))
			return -EINVAL;

		plen = inet_mask_len(mask);
	}

	cfg->fc_dst_len = plen;
	cfg->fc_dst = addr;

	if (cmd != SIOCDELRT) {
		cfg->fc_nlflags = NLM_F_CREATE;
		cfg->fc_protocol = RTPROT_BOOT;
	}

	if (rt->rt_metric)
		cfg->fc_priority = rt->rt_metric - 1;

	if (rt->rt_flags & RTF_REJECT) {
		cfg->fc_scope = RT_SCOPE_HOST;
		cfg->fc_type = RTN_UNREACHABLE;
		return 0;
	}

	cfg->fc_scope = RT_SCOPE_NOWHERE;
	cfg->fc_type = RTN_UNICAST;

	if (rt->rt_dev) {
		char *colon;
		struct net_device *dev;
		char devname[IFNAMSIZ];

		if (copy_from_user(devname, rt->rt_dev, IFNAMSIZ-1))
			return -EFAULT;

		devname[IFNAMSIZ-1] = 0;
		colon = strchr(devname, ':');
		if (colon)
			*colon = 0;
418
		dev = __dev_get_by_name(net, devname);
419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440
		if (!dev)
			return -ENODEV;
		cfg->fc_oif = dev->ifindex;
		if (colon) {
			struct in_ifaddr *ifa;
			struct in_device *in_dev = __in_dev_get_rtnl(dev);
			if (!in_dev)
				return -ENODEV;
			*colon = ':';
			for (ifa = in_dev->ifa_list; ifa; ifa = ifa->ifa_next)
				if (strcmp(ifa->ifa_label, devname) == 0)
					break;
			if (ifa == NULL)
				return -ENODEV;
			cfg->fc_prefsrc = ifa->ifa_local;
		}
	}

	addr = sk_extract_addr(&rt->rt_gateway);
	if (rt->rt_gateway.sa_family == AF_INET && addr) {
		cfg->fc_gw = addr;
		if (rt->rt_flags & RTF_GATEWAY &&
441
		    inet_addr_type(net, addr) == RTN_UNICAST)
442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458
			cfg->fc_scope = RT_SCOPE_UNIVERSE;
	}

	if (cmd == SIOCDELRT)
		return 0;

	if (rt->rt_flags & RTF_GATEWAY && !cfg->fc_gw)
		return -EINVAL;

	if (cfg->fc_scope == RT_SCOPE_NOWHERE)
		cfg->fc_scope = RT_SCOPE_LINK;

	if (rt->rt_flags & (RTF_MTU | RTF_WINDOW | RTF_IRTT)) {
		struct nlattr *mx;
		int len = 0;

		mx = kzalloc(3 * nla_total_size(4), GFP_KERNEL);
459
		if (mx == NULL)
460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477
			return -ENOMEM;

		if (rt->rt_flags & RTF_MTU)
			len = put_rtax(mx, len, RTAX_ADVMSS, rt->rt_mtu - 40);

		if (rt->rt_flags & RTF_WINDOW)
			len = put_rtax(mx, len, RTAX_WINDOW, rt->rt_window);

		if (rt->rt_flags & RTF_IRTT)
			len = put_rtax(mx, len, RTAX_RTT, rt->rt_irtt << 3);

		cfg->fc_mx = mx;
		cfg->fc_mx_len = len;
	}

	return 0;
}

L
Linus Torvalds 已提交
478
/*
E
Eric Dumazet 已提交
479 480
 * Handle IP routing ioctl calls.
 * These are used to manipulate the routing tables
L
Linus Torvalds 已提交
481
 */
482
int ip_rt_ioctl(struct net *net, unsigned int cmd, void __user *arg)
L
Linus Torvalds 已提交
483
{
484 485
	struct fib_config cfg;
	struct rtentry rt;
L
Linus Torvalds 已提交
486 487 488 489 490 491 492
	int err;

	switch (cmd) {
	case SIOCADDRT:		/* Add a route */
	case SIOCDELRT:		/* Delete a route */
		if (!capable(CAP_NET_ADMIN))
			return -EPERM;
493 494

		if (copy_from_user(&rt, arg, sizeof(rt)))
L
Linus Torvalds 已提交
495
			return -EFAULT;
496

L
Linus Torvalds 已提交
497
		rtnl_lock();
498
		err = rtentry_to_fib_config(net, cmd, &rt, &cfg);
L
Linus Torvalds 已提交
499
		if (err == 0) {
500 501
			struct fib_table *tb;

L
Linus Torvalds 已提交
502
			if (cmd == SIOCDELRT) {
503
				tb = fib_get_table(net, cfg.fc_table);
L
Linus Torvalds 已提交
504
				if (tb)
505
					err = fib_table_delete(tb, &cfg);
506 507
				else
					err = -ESRCH;
L
Linus Torvalds 已提交
508
			} else {
509
				tb = fib_new_table(net, cfg.fc_table);
L
Linus Torvalds 已提交
510
				if (tb)
511
					err = fib_table_insert(tb, &cfg);
512 513
				else
					err = -ENOBUFS;
L
Linus Torvalds 已提交
514
			}
515 516 517

			/* allocated by rtentry_to_fib_config() */
			kfree(cfg.fc_mx);
L
Linus Torvalds 已提交
518 519 520 521 522 523 524
		}
		rtnl_unlock();
		return err;
	}
	return -EINVAL;
}

E
Eric Dumazet 已提交
525
const struct nla_policy rtm_ipv4_policy[RTA_MAX + 1] = {
526 527 528 529 530 531 532 533
	[RTA_DST]		= { .type = NLA_U32 },
	[RTA_SRC]		= { .type = NLA_U32 },
	[RTA_IIF]		= { .type = NLA_U32 },
	[RTA_OIF]		= { .type = NLA_U32 },
	[RTA_GATEWAY]		= { .type = NLA_U32 },
	[RTA_PRIORITY]		= { .type = NLA_U32 },
	[RTA_PREFSRC]		= { .type = NLA_U32 },
	[RTA_METRICS]		= { .type = NLA_NESTED },
534
	[RTA_MULTIPATH]		= { .len = sizeof(struct rtnexthop) },
535 536 537
	[RTA_FLOW]		= { .type = NLA_U32 },
};

538
static int rtm_to_fib_config(struct net *net, struct sk_buff *skb,
E
Eric Dumazet 已提交
539
			     struct nlmsghdr *nlh, struct fib_config *cfg)
L
Linus Torvalds 已提交
540
{
541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560
	struct nlattr *attr;
	int err, remaining;
	struct rtmsg *rtm;

	err = nlmsg_validate(nlh, sizeof(*rtm), RTA_MAX, rtm_ipv4_policy);
	if (err < 0)
		goto errout;

	memset(cfg, 0, sizeof(*cfg));

	rtm = nlmsg_data(nlh);
	cfg->fc_dst_len = rtm->rtm_dst_len;
	cfg->fc_tos = rtm->rtm_tos;
	cfg->fc_table = rtm->rtm_table;
	cfg->fc_protocol = rtm->rtm_protocol;
	cfg->fc_scope = rtm->rtm_scope;
	cfg->fc_type = rtm->rtm_type;
	cfg->fc_flags = rtm->rtm_flags;
	cfg->fc_nlflags = nlh->nlmsg_flags;

561
	cfg->fc_nlinfo.portid = NETLINK_CB(skb).portid;
562
	cfg->fc_nlinfo.nlh = nlh;
563
	cfg->fc_nlinfo.nl_net = net;
564

565 566 567 568 569
	if (cfg->fc_type > RTN_MAX) {
		err = -EINVAL;
		goto errout;
	}

570
	nlmsg_for_each_attr(attr, nlh, sizeof(struct rtmsg), remaining) {
571
		switch (nla_type(attr)) {
572
		case RTA_DST:
573
			cfg->fc_dst = nla_get_be32(attr);
574 575 576 577 578
			break;
		case RTA_OIF:
			cfg->fc_oif = nla_get_u32(attr);
			break;
		case RTA_GATEWAY:
579
			cfg->fc_gw = nla_get_be32(attr);
580 581 582 583 584
			break;
		case RTA_PRIORITY:
			cfg->fc_priority = nla_get_u32(attr);
			break;
		case RTA_PREFSRC:
585
			cfg->fc_prefsrc = nla_get_be32(attr);
586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
			break;
		case RTA_METRICS:
			cfg->fc_mx = nla_data(attr);
			cfg->fc_mx_len = nla_len(attr);
			break;
		case RTA_MULTIPATH:
			cfg->fc_mp = nla_data(attr);
			cfg->fc_mp_len = nla_len(attr);
			break;
		case RTA_FLOW:
			cfg->fc_flow = nla_get_u32(attr);
			break;
		case RTA_TABLE:
			cfg->fc_table = nla_get_u32(attr);
			break;
L
Linus Torvalds 已提交
601 602
		}
	}
603

L
Linus Torvalds 已提交
604
	return 0;
605 606
errout:
	return err;
L
Linus Torvalds 已提交
607 608
}

609
static int inet_rtm_delroute(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
L
Linus Torvalds 已提交
610
{
611
	struct net *net = sock_net(skb->sk);
612 613 614
	struct fib_config cfg;
	struct fib_table *tb;
	int err;
L
Linus Torvalds 已提交
615

616 617 618
	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

619
	err = rtm_to_fib_config(net, skb, nlh, &cfg);
620 621
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
622

623
	tb = fib_get_table(net, cfg.fc_table);
624 625 626 627 628
	if (tb == NULL) {
		err = -ESRCH;
		goto errout;
	}

629
	err = fib_table_delete(tb, &cfg);
630 631
errout:
	return err;
L
Linus Torvalds 已提交
632 633
}

634
static int inet_rtm_newroute(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
L
Linus Torvalds 已提交
635
{
636
	struct net *net = sock_net(skb->sk);
637 638 639
	struct fib_config cfg;
	struct fib_table *tb;
	int err;
L
Linus Torvalds 已提交
640

641 642 643
	if (!capable(CAP_NET_ADMIN))
		return -EPERM;

644
	err = rtm_to_fib_config(net, skb, nlh, &cfg);
645 646
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
647

648
	tb = fib_new_table(net, cfg.fc_table);
649 650 651 652 653
	if (tb == NULL) {
		err = -ENOBUFS;
		goto errout;
	}

654
	err = fib_table_insert(tb, &cfg);
655 656
errout:
	return err;
L
Linus Torvalds 已提交
657 658
}

659
static int inet_dump_fib(struct sk_buff *skb, struct netlink_callback *cb)
L
Linus Torvalds 已提交
660
{
661
	struct net *net = sock_net(skb->sk);
662 663
	unsigned int h, s_h;
	unsigned int e = 0, s_e;
L
Linus Torvalds 已提交
664
	struct fib_table *tb;
665
	struct hlist_node *node;
666
	struct hlist_head *head;
667
	int dumped = 0;
L
Linus Torvalds 已提交
668

669 670
	if (nlmsg_len(cb->nlh) >= sizeof(struct rtmsg) &&
	    ((struct rtmsg *) nlmsg_data(cb->nlh))->rtm_flags & RTM_F_CLONED)
L
Linus Torvalds 已提交
671 672
		return ip_rt_dump(skb, cb);

673 674 675 676 677
	s_h = cb->args[0];
	s_e = cb->args[1];

	for (h = s_h; h < FIB_TABLE_HASHSZ; h++, s_e = 0) {
		e = 0;
678 679
		head = &net->ipv4.fib_table_hash[h];
		hlist_for_each_entry(tb, node, head, tb_hlist) {
680 681 682 683
			if (e < s_e)
				goto next;
			if (dumped)
				memset(&cb->args[2], 0, sizeof(cb->args) -
684
						 2 * sizeof(cb->args[0]));
685
			if (fib_table_dump(tb, skb, cb) < 0)
686 687 688 689 690
				goto out;
			dumped = 1;
next:
			e++;
		}
L
Linus Torvalds 已提交
691
	}
692 693 694
out:
	cb->args[1] = e;
	cb->args[0] = h;
L
Linus Torvalds 已提交
695 696 697 698 699

	return skb->len;
}

/* Prepare and feed intra-kernel routing request.
E
Eric Dumazet 已提交
700 701 702 703
 * Really, it should be netlink message, but :-( netlink
 * can be not configured, so that we feed it directly
 * to fib engine. It is legal, because all events occur
 * only when netlink is already locked.
L
Linus Torvalds 已提交
704
 */
A
Al Viro 已提交
705
static void fib_magic(int cmd, int type, __be32 dst, int dst_len, struct in_ifaddr *ifa)
L
Linus Torvalds 已提交
706
{
707
	struct net *net = dev_net(ifa->ifa_dev->dev);
708 709 710 711 712 713 714 715 716
	struct fib_table *tb;
	struct fib_config cfg = {
		.fc_protocol = RTPROT_KERNEL,
		.fc_type = type,
		.fc_dst = dst,
		.fc_dst_len = dst_len,
		.fc_prefsrc = ifa->ifa_local,
		.fc_oif = ifa->ifa_dev->dev->ifindex,
		.fc_nlflags = NLM_F_CREATE | NLM_F_APPEND,
717
		.fc_nlinfo = {
718
			.nl_net = net,
719
		},
720
	};
L
Linus Torvalds 已提交
721 722

	if (type == RTN_UNICAST)
723
		tb = fib_new_table(net, RT_TABLE_MAIN);
L
Linus Torvalds 已提交
724
	else
725
		tb = fib_new_table(net, RT_TABLE_LOCAL);
L
Linus Torvalds 已提交
726 727 728 729

	if (tb == NULL)
		return;

730
	cfg.fc_table = tb->tb_id;
L
Linus Torvalds 已提交
731

732 733 734 735
	if (type != RTN_LOCAL)
		cfg.fc_scope = RT_SCOPE_LINK;
	else
		cfg.fc_scope = RT_SCOPE_HOST;
L
Linus Torvalds 已提交
736 737

	if (cmd == RTM_NEWROUTE)
738
		fib_table_insert(tb, &cfg);
L
Linus Torvalds 已提交
739
	else
740
		fib_table_delete(tb, &cfg);
L
Linus Torvalds 已提交
741 742
}

743
void fib_add_ifaddr(struct in_ifaddr *ifa)
L
Linus Torvalds 已提交
744 745 746 747
{
	struct in_device *in_dev = ifa->ifa_dev;
	struct net_device *dev = in_dev->dev;
	struct in_ifaddr *prim = ifa;
A
Al Viro 已提交
748 749
	__be32 mask = ifa->ifa_mask;
	__be32 addr = ifa->ifa_local;
E
Eric Dumazet 已提交
750
	__be32 prefix = ifa->ifa_address & mask;
L
Linus Torvalds 已提交
751

E
Eric Dumazet 已提交
752
	if (ifa->ifa_flags & IFA_F_SECONDARY) {
L
Linus Torvalds 已提交
753 754
		prim = inet_ifa_byprefix(in_dev, prefix, mask);
		if (prim == NULL) {
J
Joe Perches 已提交
755
			pr_warn("%s: bug: prim == NULL\n", __func__);
L
Linus Torvalds 已提交
756 757 758 759 760 761
			return;
		}
	}

	fib_magic(RTM_NEWROUTE, RTN_LOCAL, addr, 32, prim);

E
Eric Dumazet 已提交
762
	if (!(dev->flags & IFF_UP))
L
Linus Torvalds 已提交
763 764 765
		return;

	/* Add broadcast address, if it is explicitly assigned. */
A
Al Viro 已提交
766
	if (ifa->ifa_broadcast && ifa->ifa_broadcast != htonl(0xFFFFFFFF))
L
Linus Torvalds 已提交
767 768
		fib_magic(RTM_NEWROUTE, RTN_BROADCAST, ifa->ifa_broadcast, 32, prim);

E
Eric Dumazet 已提交
769
	if (!ipv4_is_zeronet(prefix) && !(ifa->ifa_flags & IFA_F_SECONDARY) &&
L
Linus Torvalds 已提交
770
	    (prefix != addr || ifa->ifa_prefixlen < 32)) {
E
Eric Dumazet 已提交
771 772 773
		fib_magic(RTM_NEWROUTE,
			  dev->flags & IFF_LOOPBACK ? RTN_LOCAL : RTN_UNICAST,
			  prefix, ifa->ifa_prefixlen, prim);
L
Linus Torvalds 已提交
774 775 776 777

		/* Add network specific broadcasts, when it takes a sense */
		if (ifa->ifa_prefixlen < 31) {
			fib_magic(RTM_NEWROUTE, RTN_BROADCAST, prefix, 32, prim);
E
Eric Dumazet 已提交
778 779
			fib_magic(RTM_NEWROUTE, RTN_BROADCAST, prefix | ~mask,
				  32, prim);
L
Linus Torvalds 已提交
780 781 782 783
		}
	}
}

784 785 786 787 788 789
/* Delete primary or secondary address.
 * Optionally, on secondary address promotion consider the addresses
 * from subnet iprim as deleted, even if they are in device list.
 * In this case the secondary ifa can be in device list.
 */
void fib_del_ifaddr(struct in_ifaddr *ifa, struct in_ifaddr *iprim)
L
Linus Torvalds 已提交
790 791 792 793
{
	struct in_device *in_dev = ifa->ifa_dev;
	struct net_device *dev = in_dev->dev;
	struct in_ifaddr *ifa1;
794
	struct in_ifaddr *prim = ifa, *prim1 = NULL;
E
Eric Dumazet 已提交
795 796
	__be32 brd = ifa->ifa_address | ~ifa->ifa_mask;
	__be32 any = ifa->ifa_address & ifa->ifa_mask;
L
Linus Torvalds 已提交
797 798 799 800
#define LOCAL_OK	1
#define BRD_OK		2
#define BRD0_OK		4
#define BRD1_OK		8
801
	unsigned int ok = 0;
802 803 804
	int subnet = 0;		/* Primary network */
	int gone = 1;		/* Address is missing */
	int same_prefsrc = 0;	/* Another primary with same IP */
L
Linus Torvalds 已提交
805

806
	if (ifa->ifa_flags & IFA_F_SECONDARY) {
L
Linus Torvalds 已提交
807 808
		prim = inet_ifa_byprefix(in_dev, any, ifa->ifa_mask);
		if (prim == NULL) {
J
Joe Perches 已提交
809
			pr_warn("%s: bug: prim == NULL\n", __func__);
L
Linus Torvalds 已提交
810 811
			return;
		}
812
		if (iprim && iprim != prim) {
J
Joe Perches 已提交
813
			pr_warn("%s: bug: iprim != prim\n", __func__);
814 815 816 817 818 819 820 821
			return;
		}
	} else if (!ipv4_is_zeronet(any) &&
		   (any != ifa->ifa_local || ifa->ifa_prefixlen < 32)) {
		fib_magic(RTM_DELROUTE,
			  dev->flags & IFF_LOOPBACK ? RTN_LOCAL : RTN_UNICAST,
			  any, ifa->ifa_prefixlen, prim);
		subnet = 1;
L
Linus Torvalds 已提交
822 823 824
	}

	/* Deletion is more complicated than add.
E
Eric Dumazet 已提交
825 826 827
	 * We should take care of not to delete too much :-)
	 *
	 * Scan address list to be sure that addresses are really gone.
L
Linus Torvalds 已提交
828 829 830
	 */

	for (ifa1 = in_dev->ifa_list; ifa1; ifa1 = ifa1->ifa_next) {
831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873
		if (ifa1 == ifa) {
			/* promotion, keep the IP */
			gone = 0;
			continue;
		}
		/* Ignore IFAs from our subnet */
		if (iprim && ifa1->ifa_mask == iprim->ifa_mask &&
		    inet_ifa_match(ifa1->ifa_address, iprim))
			continue;

		/* Ignore ifa1 if it uses different primary IP (prefsrc) */
		if (ifa1->ifa_flags & IFA_F_SECONDARY) {
			/* Another address from our subnet? */
			if (ifa1->ifa_mask == prim->ifa_mask &&
			    inet_ifa_match(ifa1->ifa_address, prim))
				prim1 = prim;
			else {
				/* We reached the secondaries, so
				 * same_prefsrc should be determined.
				 */
				if (!same_prefsrc)
					continue;
				/* Search new prim1 if ifa1 is not
				 * using the current prim1
				 */
				if (!prim1 ||
				    ifa1->ifa_mask != prim1->ifa_mask ||
				    !inet_ifa_match(ifa1->ifa_address, prim1))
					prim1 = inet_ifa_byprefix(in_dev,
							ifa1->ifa_address,
							ifa1->ifa_mask);
				if (!prim1)
					continue;
				if (prim1->ifa_local != prim->ifa_local)
					continue;
			}
		} else {
			if (prim->ifa_local != ifa1->ifa_local)
				continue;
			prim1 = ifa1;
			if (prim != prim1)
				same_prefsrc = 1;
		}
L
Linus Torvalds 已提交
874 875 876 877 878 879 880 881
		if (ifa->ifa_local == ifa1->ifa_local)
			ok |= LOCAL_OK;
		if (ifa->ifa_broadcast == ifa1->ifa_broadcast)
			ok |= BRD_OK;
		if (brd == ifa1->ifa_broadcast)
			ok |= BRD1_OK;
		if (any == ifa1->ifa_broadcast)
			ok |= BRD0_OK;
882 883 884 885 886 887 888 889 890 891 892 893 894 895 896
		/* primary has network specific broadcasts */
		if (prim1 == ifa1 && ifa1->ifa_prefixlen < 31) {
			__be32 brd1 = ifa1->ifa_address | ~ifa1->ifa_mask;
			__be32 any1 = ifa1->ifa_address & ifa1->ifa_mask;

			if (!ipv4_is_zeronet(any1)) {
				if (ifa->ifa_broadcast == brd1 ||
				    ifa->ifa_broadcast == any1)
					ok |= BRD_OK;
				if (brd == brd1 || brd == any1)
					ok |= BRD1_OK;
				if (any == brd1 || any == any1)
					ok |= BRD0_OK;
			}
		}
L
Linus Torvalds 已提交
897 898
	}

E
Eric Dumazet 已提交
899
	if (!(ok & BRD_OK))
L
Linus Torvalds 已提交
900
		fib_magic(RTM_DELROUTE, RTN_BROADCAST, ifa->ifa_broadcast, 32, prim);
901 902 903 904 905 906
	if (subnet && ifa->ifa_prefixlen < 31) {
		if (!(ok & BRD1_OK))
			fib_magic(RTM_DELROUTE, RTN_BROADCAST, brd, 32, prim);
		if (!(ok & BRD0_OK))
			fib_magic(RTM_DELROUTE, RTN_BROADCAST, any, 32, prim);
	}
E
Eric Dumazet 已提交
907
	if (!(ok & LOCAL_OK)) {
L
Linus Torvalds 已提交
908 909 910
		fib_magic(RTM_DELROUTE, RTN_LOCAL, ifa->ifa_local, 32, prim);

		/* Check, that this local address finally disappeared. */
911 912
		if (gone &&
		    inet_addr_type(dev_net(dev), ifa->ifa_local) != RTN_LOCAL) {
L
Linus Torvalds 已提交
913
			/* And the last, but not the least thing.
E
Eric Dumazet 已提交
914 915 916 917 918
			 * We must flush stray FIB entries.
			 *
			 * First of all, we scan fib_info list searching
			 * for stray nexthop entries, then ignite fib_flush.
			 */
919 920
			if (fib_sync_down_addr(dev_net(dev), ifa->ifa_local))
				fib_flush(dev_net(dev));
L
Linus Torvalds 已提交
921 922 923 924 925 926 927 928
		}
	}
#undef LOCAL_OK
#undef BRD_OK
#undef BRD0_OK
#undef BRD1_OK
}

E
Eric Dumazet 已提交
929
static void nl_fib_lookup(struct fib_result_nl *frn, struct fib_table *tb)
930
{
931

932
	struct fib_result       res;
D
David S. Miller 已提交
933 934 935 936 937
	struct flowi4           fl4 = {
		.flowi4_mark = frn->fl_mark,
		.daddr = frn->fl_addr,
		.flowi4_tos = frn->fl_tos,
		.flowi4_scope = frn->fl_scope,
E
Eric Dumazet 已提交
938
	};
939 940

	frn->err = -ENOENT;
941 942 943 944
	if (tb) {
		local_bh_disable();

		frn->tb_id = tb->tb_id;
E
Eric Dumazet 已提交
945
		rcu_read_lock();
D
David S. Miller 已提交
946
		frn->err = fib_table_lookup(tb, &fl4, &res, FIB_LOOKUP_NOREF);
947 948 949 950 951 952 953

		if (!frn->err) {
			frn->prefixlen = res.prefixlen;
			frn->nh_sel = res.nh_sel;
			frn->type = res.type;
			frn->scope = res.scope;
		}
E
Eric Dumazet 已提交
954
		rcu_read_unlock();
955 956 957 958
		local_bh_enable();
	}
}

959
static void nl_fib_input(struct sk_buff *skb)
960
{
961
	struct net *net;
962
	struct fib_result_nl *frn;
963
	struct nlmsghdr *nlh;
964
	struct fib_table *tb;
965
	u32 portid;
966

967
	net = sock_net(skb->sk);
968
	nlh = nlmsg_hdr(skb);
969
	if (skb->len < NLMSG_SPACE(0) || skb->len < nlh->nlmsg_len ||
970
	    nlh->nlmsg_len < NLMSG_LENGTH(sizeof(*frn)))
971
		return;
972 973 974 975 976

	skb = skb_clone(skb, GFP_KERNEL);
	if (skb == NULL)
		return;
	nlh = nlmsg_hdr(skb);
977

978
	frn = (struct fib_result_nl *) NLMSG_DATA(nlh);
979
	tb = fib_get_table(net, frn->tb_id_in);
980 981

	nl_fib_lookup(frn, tb);
982

983 984
	portid = NETLINK_CB(skb).portid;      /* pid of sending process */
	NETLINK_CB(skb).portid = 0;        /* from kernel */
985
	NETLINK_CB(skb).dst_group = 0;  /* unicast */
986
	netlink_unicast(net->ipv4.fibnl, skb, portid, MSG_DONTWAIT);
987
}
988

989
static int __net_init nl_fib_lookup_init(struct net *net)
990
{
991
	struct sock *sk;
992 993 994 995
	struct netlink_kernel_cfg cfg = {
		.input	= nl_fib_input,
	};

996
	sk = netlink_kernel_create(net, NETLINK_FIB_LOOKUP, &cfg);
997
	if (sk == NULL)
998
		return -EAFNOSUPPORT;
999
	net->ipv4.fibnl = sk;
1000 1001 1002 1003 1004
	return 0;
}

static void nl_fib_lookup_exit(struct net *net)
{
1005
	netlink_kernel_release(net->ipv4.fibnl);
1006
	net->ipv4.fibnl = NULL;
1007 1008
}

1009
static void fib_disable_ip(struct net_device *dev, int force)
L
Linus Torvalds 已提交
1010
{
D
Denis V. Lunev 已提交
1011
	if (fib_sync_down_dev(dev, force))
1012
		fib_flush(dev_net(dev));
1013
	rt_cache_flush(dev_net(dev));
L
Linus Torvalds 已提交
1014 1015 1016 1017 1018
	arp_ifdown(dev);
}

static int fib_inetaddr_event(struct notifier_block *this, unsigned long event, void *ptr)
{
1019
	struct in_ifaddr *ifa = (struct in_ifaddr *)ptr;
1020
	struct net_device *dev = ifa->ifa_dev->dev;
1021
	struct net *net = dev_net(dev);
L
Linus Torvalds 已提交
1022 1023 1024 1025 1026

	switch (event) {
	case NETDEV_UP:
		fib_add_ifaddr(ifa);
#ifdef CONFIG_IP_ROUTE_MULTIPATH
1027
		fib_sync_up(dev);
L
Linus Torvalds 已提交
1028
#endif
1029
		atomic_inc(&net->ipv4.dev_addr_genid);
1030
		rt_cache_flush(dev_net(dev));
L
Linus Torvalds 已提交
1031 1032
		break;
	case NETDEV_DOWN:
1033
		fib_del_ifaddr(ifa, NULL);
1034
		atomic_inc(&net->ipv4.dev_addr_genid);
1035
		if (ifa->ifa_dev->ifa_list == NULL) {
L
Linus Torvalds 已提交
1036
			/* Last address was deleted from this interface.
E
Eric Dumazet 已提交
1037
			 * Disable IP.
L
Linus Torvalds 已提交
1038
			 */
1039
			fib_disable_ip(dev, 1);
L
Linus Torvalds 已提交
1040
		} else {
1041
			rt_cache_flush(dev_net(dev));
L
Linus Torvalds 已提交
1042 1043 1044 1045 1046 1047 1048 1049 1050
		}
		break;
	}
	return NOTIFY_DONE;
}

static int fib_netdev_event(struct notifier_block *this, unsigned long event, void *ptr)
{
	struct net_device *dev = ptr;
1051
	struct in_device *in_dev;
1052
	struct net *net = dev_net(dev);
L
Linus Torvalds 已提交
1053 1054

	if (event == NETDEV_UNREGISTER) {
1055
		fib_disable_ip(dev, 2);
1056
		rt_flush_dev(dev);
L
Linus Torvalds 已提交
1057 1058 1059
		return NOTIFY_DONE;
	}

1060 1061
	in_dev = __in_dev_get_rtnl(dev);

L
Linus Torvalds 已提交
1062 1063 1064 1065 1066 1067 1068 1069
	switch (event) {
	case NETDEV_UP:
		for_ifa(in_dev) {
			fib_add_ifaddr(ifa);
		} endfor_ifa(in_dev);
#ifdef CONFIG_IP_ROUTE_MULTIPATH
		fib_sync_up(dev);
#endif
1070
		atomic_inc(&net->ipv4.dev_addr_genid);
1071
		rt_cache_flush(net);
L
Linus Torvalds 已提交
1072 1073
		break;
	case NETDEV_DOWN:
1074
		fib_disable_ip(dev, 0);
L
Linus Torvalds 已提交
1075 1076 1077
		break;
	case NETDEV_CHANGEMTU:
	case NETDEV_CHANGE:
1078
		rt_cache_flush(net);
L
Linus Torvalds 已提交
1079 1080 1081 1082 1083 1084
		break;
	}
	return NOTIFY_DONE;
}

static struct notifier_block fib_inetaddr_notifier = {
1085
	.notifier_call = fib_inetaddr_event,
L
Linus Torvalds 已提交
1086 1087 1088
};

static struct notifier_block fib_netdev_notifier = {
1089
	.notifier_call = fib_netdev_event,
L
Linus Torvalds 已提交
1090 1091
};

1092
static int __net_init ip_fib_net_init(struct net *net)
L
Linus Torvalds 已提交
1093
{
1094
	int err;
1095 1096 1097 1098
	size_t size = sizeof(struct hlist_head) * FIB_TABLE_HASHSZ;

	/* Avoid false sharing : Use at least a full cache line */
	size = max_t(size_t, size, L1_CACHE_BYTES);
1099

1100
	net->ipv4.fib_table_hash = kzalloc(size, GFP_KERNEL);
1101 1102 1103
	if (net->ipv4.fib_table_hash == NULL)
		return -ENOMEM;

1104 1105 1106 1107 1108 1109 1110 1111
	err = fib4_rules_init(net);
	if (err < 0)
		goto fail;
	return 0;

fail:
	kfree(net->ipv4.fib_table_hash);
	return err;
1112
}
L
Linus Torvalds 已提交
1113

1114
static void ip_fib_net_exit(struct net *net)
1115 1116 1117 1118 1119 1120 1121
{
	unsigned int i;

#ifdef CONFIG_IP_MULTIPLE_TABLES
	fib4_rules_exit(net);
#endif

1122
	rtnl_lock();
1123 1124 1125 1126
	for (i = 0; i < FIB_TABLE_HASHSZ; i++) {
		struct fib_table *tb;
		struct hlist_head *head;
		struct hlist_node *node, *tmp;
1127

1128
		head = &net->ipv4.fib_table_hash[i];
1129 1130
		hlist_for_each_entry_safe(tb, node, tmp, head, tb_hlist) {
			hlist_del(node);
1131
			fib_table_flush(tb);
1132
			fib_free_table(tb);
1133 1134
		}
	}
1135
	rtnl_unlock();
1136
	kfree(net->ipv4.fib_table_hash);
1137 1138 1139 1140 1141 1142
}

static int __net_init fib_net_init(struct net *net)
{
	int error;

1143 1144 1145
#ifdef CONFIG_IP_ROUTE_CLASSID
	net->ipv4.fib_num_tclassid_users = 0;
#endif
1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178
	error = ip_fib_net_init(net);
	if (error < 0)
		goto out;
	error = nl_fib_lookup_init(net);
	if (error < 0)
		goto out_nlfl;
	error = fib_proc_init(net);
	if (error < 0)
		goto out_proc;
out:
	return error;

out_proc:
	nl_fib_lookup_exit(net);
out_nlfl:
	ip_fib_net_exit(net);
	goto out;
}

static void __net_exit fib_net_exit(struct net *net)
{
	fib_proc_exit(net);
	nl_fib_lookup_exit(net);
	ip_fib_net_exit(net);
}

static struct pernet_operations fib_net_ops = {
	.init = fib_net_init,
	.exit = fib_net_exit,
};

void __init ip_fib_init(void)
{
1179 1180 1181
	rtnl_register(PF_INET, RTM_NEWROUTE, inet_rtm_newroute, NULL, NULL);
	rtnl_register(PF_INET, RTM_DELROUTE, inet_rtm_delroute, NULL, NULL);
	rtnl_register(PF_INET, RTM_GETROUTE, NULL, inet_dump_fib, NULL);
1182 1183 1184 1185

	register_pernet_subsys(&fib_net_ops);
	register_netdevice_notifier(&fib_netdev_notifier);
	register_inetaddr_notifier(&fib_inetaddr_notifier);
1186

1187
	fib_trie_init();
L
Linus Torvalds 已提交
1188
}