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
	int err;

	switch (cmd) {
	case SIOCADDRT:		/* Add a route */
	case SIOCDELRT:		/* Delete a route */
491
		if (!ns_capable(net->user_ns, CAP_NET_ADMIN))
L
Linus Torvalds 已提交
492
			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
	err = rtm_to_fib_config(net, skb, nlh, &cfg);
617 618
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
619

620
	tb = fib_get_table(net, cfg.fc_table);
621 622 623 624 625
	if (tb == NULL) {
		err = -ESRCH;
		goto errout;
	}

626
	err = fib_table_delete(tb, &cfg);
627 628
errout:
	return err;
L
Linus Torvalds 已提交
629 630
}

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

638
	err = rtm_to_fib_config(net, skb, nlh, &cfg);
639 640
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
641

642
	tb = fib_new_table(net, cfg.fc_table);
643 644 645 646 647
	if (tb == NULL) {
		err = -ENOBUFS;
		goto errout;
	}

648
	err = fib_table_insert(tb, &cfg);
649 650
errout:
	return err;
L
Linus Torvalds 已提交
651 652
}

653
static int inet_dump_fib(struct sk_buff *skb, struct netlink_callback *cb)
L
Linus Torvalds 已提交
654
{
655
	struct net *net = sock_net(skb->sk);
656 657
	unsigned int h, s_h;
	unsigned int e = 0, s_e;
L
Linus Torvalds 已提交
658
	struct fib_table *tb;
659
	struct hlist_node *node;
660
	struct hlist_head *head;
661
	int dumped = 0;
L
Linus Torvalds 已提交
662

663 664
	if (nlmsg_len(cb->nlh) >= sizeof(struct rtmsg) &&
	    ((struct rtmsg *) nlmsg_data(cb->nlh))->rtm_flags & RTM_F_CLONED)
L
Linus Torvalds 已提交
665 666
		return ip_rt_dump(skb, cb);

667 668 669 670 671
	s_h = cb->args[0];
	s_e = cb->args[1];

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

	return skb->len;
}

/* Prepare and feed intra-kernel routing request.
E
Eric Dumazet 已提交
694 695 696 697
 * 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 已提交
698
 */
A
Al Viro 已提交
699
static void fib_magic(int cmd, int type, __be32 dst, int dst_len, struct in_ifaddr *ifa)
L
Linus Torvalds 已提交
700
{
701
	struct net *net = dev_net(ifa->ifa_dev->dev);
702 703 704 705 706 707 708 709 710
	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,
711
		.fc_nlinfo = {
712
			.nl_net = net,
713
		},
714
	};
L
Linus Torvalds 已提交
715 716

	if (type == RTN_UNICAST)
717
		tb = fib_new_table(net, RT_TABLE_MAIN);
L
Linus Torvalds 已提交
718
	else
719
		tb = fib_new_table(net, RT_TABLE_LOCAL);
L
Linus Torvalds 已提交
720 721 722 723

	if (tb == NULL)
		return;

724
	cfg.fc_table = tb->tb_id;
L
Linus Torvalds 已提交
725

726 727 728 729
	if (type != RTN_LOCAL)
		cfg.fc_scope = RT_SCOPE_LINK;
	else
		cfg.fc_scope = RT_SCOPE_HOST;
L
Linus Torvalds 已提交
730 731

	if (cmd == RTM_NEWROUTE)
732
		fib_table_insert(tb, &cfg);
L
Linus Torvalds 已提交
733
	else
734
		fib_table_delete(tb, &cfg);
L
Linus Torvalds 已提交
735 736
}

737
void fib_add_ifaddr(struct in_ifaddr *ifa)
L
Linus Torvalds 已提交
738 739 740 741
{
	struct in_device *in_dev = ifa->ifa_dev;
	struct net_device *dev = in_dev->dev;
	struct in_ifaddr *prim = ifa;
A
Al Viro 已提交
742 743
	__be32 mask = ifa->ifa_mask;
	__be32 addr = ifa->ifa_local;
E
Eric Dumazet 已提交
744
	__be32 prefix = ifa->ifa_address & mask;
L
Linus Torvalds 已提交
745

E
Eric Dumazet 已提交
746
	if (ifa->ifa_flags & IFA_F_SECONDARY) {
L
Linus Torvalds 已提交
747 748
		prim = inet_ifa_byprefix(in_dev, prefix, mask);
		if (prim == NULL) {
J
Joe Perches 已提交
749
			pr_warn("%s: bug: prim == NULL\n", __func__);
L
Linus Torvalds 已提交
750 751 752 753 754 755
			return;
		}
	}

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

E
Eric Dumazet 已提交
756
	if (!(dev->flags & IFF_UP))
L
Linus Torvalds 已提交
757 758 759
		return;

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

E
Eric Dumazet 已提交
763
	if (!ipv4_is_zeronet(prefix) && !(ifa->ifa_flags & IFA_F_SECONDARY) &&
L
Linus Torvalds 已提交
764
	    (prefix != addr || ifa->ifa_prefixlen < 32)) {
E
Eric Dumazet 已提交
765 766 767
		fib_magic(RTM_NEWROUTE,
			  dev->flags & IFF_LOOPBACK ? RTN_LOCAL : RTN_UNICAST,
			  prefix, ifa->ifa_prefixlen, prim);
L
Linus Torvalds 已提交
768 769 770 771

		/* 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 已提交
772 773
			fib_magic(RTM_NEWROUTE, RTN_BROADCAST, prefix | ~mask,
				  32, prim);
L
Linus Torvalds 已提交
774 775 776 777
		}
	}
}

778 779 780 781 782 783
/* 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 已提交
784 785 786 787
{
	struct in_device *in_dev = ifa->ifa_dev;
	struct net_device *dev = in_dev->dev;
	struct in_ifaddr *ifa1;
788
	struct in_ifaddr *prim = ifa, *prim1 = NULL;
E
Eric Dumazet 已提交
789 790
	__be32 brd = ifa->ifa_address | ~ifa->ifa_mask;
	__be32 any = ifa->ifa_address & ifa->ifa_mask;
L
Linus Torvalds 已提交
791 792 793 794
#define LOCAL_OK	1
#define BRD_OK		2
#define BRD0_OK		4
#define BRD1_OK		8
795
	unsigned int ok = 0;
796 797 798
	int subnet = 0;		/* Primary network */
	int gone = 1;		/* Address is missing */
	int same_prefsrc = 0;	/* Another primary with same IP */
L
Linus Torvalds 已提交
799

800
	if (ifa->ifa_flags & IFA_F_SECONDARY) {
L
Linus Torvalds 已提交
801 802
		prim = inet_ifa_byprefix(in_dev, any, ifa->ifa_mask);
		if (prim == NULL) {
J
Joe Perches 已提交
803
			pr_warn("%s: bug: prim == NULL\n", __func__);
L
Linus Torvalds 已提交
804 805
			return;
		}
806
		if (iprim && iprim != prim) {
J
Joe Perches 已提交
807
			pr_warn("%s: bug: iprim != prim\n", __func__);
808 809 810 811 812 813 814 815
			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 已提交
816 817 818
	}

	/* Deletion is more complicated than add.
E
Eric Dumazet 已提交
819 820 821
	 * We should take care of not to delete too much :-)
	 *
	 * Scan address list to be sure that addresses are really gone.
L
Linus Torvalds 已提交
822 823 824
	 */

	for (ifa1 = in_dev->ifa_list; ifa1; ifa1 = ifa1->ifa_next) {
825 826 827 828 829 830 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
		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 已提交
868 869 870 871 872 873 874 875
		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;
876 877 878 879 880 881 882 883 884 885 886 887 888 889 890
		/* 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 已提交
891 892
	}

E
Eric Dumazet 已提交
893
	if (!(ok & BRD_OK))
L
Linus Torvalds 已提交
894
		fib_magic(RTM_DELROUTE, RTN_BROADCAST, ifa->ifa_broadcast, 32, prim);
895 896 897 898 899 900
	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 已提交
901
	if (!(ok & LOCAL_OK)) {
L
Linus Torvalds 已提交
902 903 904
		fib_magic(RTM_DELROUTE, RTN_LOCAL, ifa->ifa_local, 32, prim);

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

E
Eric Dumazet 已提交
923
static void nl_fib_lookup(struct fib_result_nl *frn, struct fib_table *tb)
924
{
925

926
	struct fib_result       res;
D
David S. Miller 已提交
927 928 929 930 931
	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 已提交
932
	};
933 934

	frn->err = -ENOENT;
935 936 937 938
	if (tb) {
		local_bh_disable();

		frn->tb_id = tb->tb_id;
E
Eric Dumazet 已提交
939
		rcu_read_lock();
D
David S. Miller 已提交
940
		frn->err = fib_table_lookup(tb, &fl4, &res, FIB_LOOKUP_NOREF);
941 942 943 944 945 946 947

		if (!frn->err) {
			frn->prefixlen = res.prefixlen;
			frn->nh_sel = res.nh_sel;
			frn->type = res.type;
			frn->scope = res.scope;
		}
E
Eric Dumazet 已提交
948
		rcu_read_unlock();
949 950 951 952
		local_bh_enable();
	}
}

953
static void nl_fib_input(struct sk_buff *skb)
954
{
955
	struct net *net;
956
	struct fib_result_nl *frn;
957
	struct nlmsghdr *nlh;
958
	struct fib_table *tb;
959
	u32 portid;
960

961
	net = sock_net(skb->sk);
962
	nlh = nlmsg_hdr(skb);
963
	if (skb->len < NLMSG_SPACE(0) || skb->len < nlh->nlmsg_len ||
964
	    nlh->nlmsg_len < NLMSG_LENGTH(sizeof(*frn)))
965
		return;
966 967 968 969 970

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

972
	frn = (struct fib_result_nl *) NLMSG_DATA(nlh);
973
	tb = fib_get_table(net, frn->tb_id_in);
974 975

	nl_fib_lookup(frn, tb);
976

977 978
	portid = NETLINK_CB(skb).portid;      /* pid of sending process */
	NETLINK_CB(skb).portid = 0;        /* from kernel */
979
	NETLINK_CB(skb).dst_group = 0;  /* unicast */
980
	netlink_unicast(net->ipv4.fibnl, skb, portid, MSG_DONTWAIT);
981
}
982

983
static int __net_init nl_fib_lookup_init(struct net *net)
984
{
985
	struct sock *sk;
986 987 988 989
	struct netlink_kernel_cfg cfg = {
		.input	= nl_fib_input,
	};

990
	sk = netlink_kernel_create(net, NETLINK_FIB_LOOKUP, &cfg);
991
	if (sk == NULL)
992
		return -EAFNOSUPPORT;
993
	net->ipv4.fibnl = sk;
994 995 996 997 998
	return 0;
}

static void nl_fib_lookup_exit(struct net *net)
{
999
	netlink_kernel_release(net->ipv4.fibnl);
1000
	net->ipv4.fibnl = NULL;
1001 1002
}

1003
static void fib_disable_ip(struct net_device *dev, int force)
L
Linus Torvalds 已提交
1004
{
D
Denis V. Lunev 已提交
1005
	if (fib_sync_down_dev(dev, force))
1006
		fib_flush(dev_net(dev));
1007
	rt_cache_flush(dev_net(dev));
L
Linus Torvalds 已提交
1008 1009 1010 1011 1012
	arp_ifdown(dev);
}

static int fib_inetaddr_event(struct notifier_block *this, unsigned long event, void *ptr)
{
1013
	struct in_ifaddr *ifa = (struct in_ifaddr *)ptr;
1014
	struct net_device *dev = ifa->ifa_dev->dev;
1015
	struct net *net = dev_net(dev);
L
Linus Torvalds 已提交
1016 1017 1018 1019 1020

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

static int fib_netdev_event(struct notifier_block *this, unsigned long event, void *ptr)
{
	struct net_device *dev = ptr;
1045
	struct in_device *in_dev;
1046
	struct net *net = dev_net(dev);
L
Linus Torvalds 已提交
1047 1048

	if (event == NETDEV_UNREGISTER) {
1049
		fib_disable_ip(dev, 2);
1050
		rt_flush_dev(dev);
L
Linus Torvalds 已提交
1051 1052 1053
		return NOTIFY_DONE;
	}

1054 1055
	in_dev = __in_dev_get_rtnl(dev);

L
Linus Torvalds 已提交
1056 1057 1058 1059 1060 1061 1062 1063
	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
1064
		atomic_inc(&net->ipv4.dev_addr_genid);
1065
		rt_cache_flush(net);
L
Linus Torvalds 已提交
1066 1067
		break;
	case NETDEV_DOWN:
1068
		fib_disable_ip(dev, 0);
L
Linus Torvalds 已提交
1069 1070 1071
		break;
	case NETDEV_CHANGEMTU:
	case NETDEV_CHANGE:
1072
		rt_cache_flush(net);
L
Linus Torvalds 已提交
1073 1074 1075 1076 1077 1078
		break;
	}
	return NOTIFY_DONE;
}

static struct notifier_block fib_inetaddr_notifier = {
1079
	.notifier_call = fib_inetaddr_event,
L
Linus Torvalds 已提交
1080 1081 1082
};

static struct notifier_block fib_netdev_notifier = {
1083
	.notifier_call = fib_netdev_event,
L
Linus Torvalds 已提交
1084 1085
};

1086
static int __net_init ip_fib_net_init(struct net *net)
L
Linus Torvalds 已提交
1087
{
1088
	int err;
1089 1090 1091 1092
	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);
1093

1094
	net->ipv4.fib_table_hash = kzalloc(size, GFP_KERNEL);
1095 1096 1097
	if (net->ipv4.fib_table_hash == NULL)
		return -ENOMEM;

1098 1099 1100 1101 1102 1103 1104 1105
	err = fib4_rules_init(net);
	if (err < 0)
		goto fail;
	return 0;

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

1108
static void ip_fib_net_exit(struct net *net)
1109 1110 1111 1112 1113 1114 1115
{
	unsigned int i;

#ifdef CONFIG_IP_MULTIPLE_TABLES
	fib4_rules_exit(net);
#endif

1116
	rtnl_lock();
1117 1118 1119 1120
	for (i = 0; i < FIB_TABLE_HASHSZ; i++) {
		struct fib_table *tb;
		struct hlist_head *head;
		struct hlist_node *node, *tmp;
1121

1122
		head = &net->ipv4.fib_table_hash[i];
1123 1124
		hlist_for_each_entry_safe(tb, node, tmp, head, tb_hlist) {
			hlist_del(node);
1125
			fib_table_flush(tb);
1126
			fib_free_table(tb);
1127 1128
		}
	}
1129
	rtnl_unlock();
1130
	kfree(net->ipv4.fib_table_hash);
1131 1132 1133 1134 1135 1136
}

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

1137 1138 1139
#ifdef CONFIG_IP_ROUTE_CLASSID
	net->ipv4.fib_num_tclassid_users = 0;
#endif
1140 1141 1142 1143 1144 1145 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
	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)
{
1173 1174 1175
	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);
1176 1177 1178 1179

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

1181
	fib_trie_init();
L
Linus Torvalds 已提交
1182
}