route.c 73.4 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5
/*
 *	Linux INET6 implementation
 *	FIB front-end.
 *
 *	Authors:
6
 *	Pedro Roque		<roque@di.fc.ul.pt>
L
Linus Torvalds 已提交
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 *
 *	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.
 */

/*	Changes:
 *
 *	YOSHIFUJI Hideaki @USAGI
 *		reworked default router selection.
 *		- respect outgoing interface
 *		- select from (probably) reachable routers (i.e.
 *		routers in REACHABLE, STALE, DELAY or PROBE states).
 *		- always select the same router if it is (probably)
 *		reachable.  otherwise, round-robin the list.
23 24
 *	Ville Nuorvala
 *		Fixed routing subtrees.
L
Linus Torvalds 已提交
25 26
 */

27 28
#define pr_fmt(fmt) "IPv6: " fmt

29
#include <linux/capability.h>
L
Linus Torvalds 已提交
30
#include <linux/errno.h>
31
#include <linux/export.h>
L
Linus Torvalds 已提交
32 33 34 35 36 37 38 39
#include <linux/types.h>
#include <linux/times.h>
#include <linux/socket.h>
#include <linux/sockios.h>
#include <linux/net.h>
#include <linux/route.h>
#include <linux/netdevice.h>
#include <linux/in6.h>
40
#include <linux/mroute6.h>
L
Linus Torvalds 已提交
41 42 43 44
#include <linux/init.h>
#include <linux/if_arp.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
45
#include <linux/nsproxy.h>
46
#include <linux/slab.h>
47
#include <net/net_namespace.h>
L
Linus Torvalds 已提交
48 49 50 51 52 53 54 55 56 57
#include <net/snmp.h>
#include <net/ipv6.h>
#include <net/ip6_fib.h>
#include <net/ip6_route.h>
#include <net/ndisc.h>
#include <net/addrconf.h>
#include <net/tcp.h>
#include <linux/rtnetlink.h>
#include <net/dst.h>
#include <net/xfrm.h>
58
#include <net/netevent.h>
59
#include <net/netlink.h>
L
Linus Torvalds 已提交
60 61 62 63 64 65 66

#include <asm/uaccess.h>

#ifdef CONFIG_SYSCTL
#include <linux/sysctl.h>
#endif

67
static struct rt6_info *ip6_rt_copy(struct rt6_info *ort,
E
Eric Dumazet 已提交
68
				    const struct in6_addr *dest);
L
Linus Torvalds 已提交
69
static struct dst_entry	*ip6_dst_check(struct dst_entry *dst, u32 cookie);
70
static unsigned int	 ip6_default_advmss(const struct dst_entry *dst);
71
static unsigned int	 ip6_mtu(const struct dst_entry *dst);
L
Linus Torvalds 已提交
72 73 74 75
static struct dst_entry *ip6_negative_advice(struct dst_entry *);
static void		ip6_dst_destroy(struct dst_entry *);
static void		ip6_dst_ifdown(struct dst_entry *,
				       struct net_device *dev, int how);
76
static int		 ip6_dst_gc(struct dst_ops *ops);
L
Linus Torvalds 已提交
77 78 79 80

static int		ip6_pkt_discard(struct sk_buff *skb);
static int		ip6_pkt_discard_out(struct sk_buff *skb);
static void		ip6_link_failure(struct sk_buff *skb);
81 82 83 84
static void		ip6_rt_update_pmtu(struct dst_entry *dst, struct sock *sk,
					   struct sk_buff *skb, u32 mtu);
static void		rt6_do_redirect(struct dst_entry *dst, struct sock *sk,
					struct sk_buff *skb);
L
Linus Torvalds 已提交
85

86
#ifdef CONFIG_IPV6_ROUTE_INFO
87
static struct rt6_info *rt6_add_route_info(struct net *net,
88 89
					   const struct in6_addr *prefix, int prefixlen,
					   const struct in6_addr *gwaddr, int ifindex,
90
					   unsigned int pref);
91
static struct rt6_info *rt6_get_route_info(struct net *net,
92 93
					   const struct in6_addr *prefix, int prefixlen,
					   const struct in6_addr *gwaddr, int ifindex);
94 95
#endif

96 97 98 99 100 101
static u32 *ipv6_cow_metrics(struct dst_entry *dst, unsigned long old)
{
	struct rt6_info *rt = (struct rt6_info *) dst;
	struct inet_peer *peer;
	u32 *p = NULL;

102 103 104
	if (!(rt->dst.flags & DST_HOST))
		return NULL;

105
	peer = rt6_get_peer_create(rt);
106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
	if (peer) {
		u32 *old_p = __DST_METRICS_PTR(old);
		unsigned long prev, new;

		p = peer->metrics;
		if (inet_metrics_new(peer))
			memcpy(p, old_p, sizeof(u32) * RTAX_MAX);

		new = (unsigned long) p;
		prev = cmpxchg(&dst->_metrics, old, new);

		if (prev != old) {
			p = __DST_METRICS_PTR(prev);
			if (prev & DST_METRICS_READ_ONLY)
				p = NULL;
		}
	}
	return p;
}

126 127 128
static inline const void *choose_neigh_daddr(struct rt6_info *rt,
					     struct sk_buff *skb,
					     const void *daddr)
129 130 131
{
	struct in6_addr *p = &rt->rt6i_gateway;

D
David S. Miller 已提交
132
	if (!ipv6_addr_any(p))
133
		return (const void *) p;
134 135
	else if (skb)
		return &ipv6_hdr(skb)->daddr;
136 137 138
	return daddr;
}

139 140 141
static struct neighbour *ip6_neigh_lookup(const struct dst_entry *dst,
					  struct sk_buff *skb,
					  const void *daddr)
142
{
143 144 145
	struct rt6_info *rt = (struct rt6_info *) dst;
	struct neighbour *n;

146
	daddr = choose_neigh_daddr(rt, skb, daddr);
147
	n = __ipv6_neigh_lookup(&nd_tbl, dst->dev, daddr);
148 149 150 151 152
	if (n)
		return n;
	return neigh_create(&nd_tbl, daddr, dst->dev);
}

153
static int rt6_bind_neighbour(struct rt6_info *rt, struct net_device *dev)
154
{
155 156 157 158 159 160
	struct neighbour *n = __ipv6_neigh_lookup(&nd_tbl, dev, &rt->rt6i_gateway);
	if (!n) {
		n = neigh_create(&nd_tbl, &rt->rt6i_gateway, dev);
		if (IS_ERR(n))
			return PTR_ERR(n);
	}
161
	rt->n = n;
162 163

	return 0;
164 165
}

166
static struct dst_ops ip6_dst_ops_template = {
L
Linus Torvalds 已提交
167
	.family			=	AF_INET6,
168
	.protocol		=	cpu_to_be16(ETH_P_IPV6),
L
Linus Torvalds 已提交
169 170 171
	.gc			=	ip6_dst_gc,
	.gc_thresh		=	1024,
	.check			=	ip6_dst_check,
172
	.default_advmss		=	ip6_default_advmss,
173
	.mtu			=	ip6_mtu,
174
	.cow_metrics		=	ipv6_cow_metrics,
L
Linus Torvalds 已提交
175 176 177 178 179
	.destroy		=	ip6_dst_destroy,
	.ifdown			=	ip6_dst_ifdown,
	.negative_advice	=	ip6_negative_advice,
	.link_failure		=	ip6_link_failure,
	.update_pmtu		=	ip6_rt_update_pmtu,
180
	.redirect		=	rt6_do_redirect,
181
	.local_out		=	__ip6_local_out,
182
	.neigh_lookup		=	ip6_neigh_lookup,
L
Linus Torvalds 已提交
183 184
};

185
static unsigned int ip6_blackhole_mtu(const struct dst_entry *dst)
186
{
187 188 189
	unsigned int mtu = dst_metric_raw(dst, RTAX_MTU);

	return mtu ? : dst->dev->mtu;
190 191
}

192 193
static void ip6_rt_blackhole_update_pmtu(struct dst_entry *dst, struct sock *sk,
					 struct sk_buff *skb, u32 mtu)
194 195 196
{
}

197 198
static void ip6_rt_blackhole_redirect(struct dst_entry *dst, struct sock *sk,
				      struct sk_buff *skb)
199 200 201
{
}

202 203 204 205 206 207
static u32 *ip6_rt_blackhole_cow_metrics(struct dst_entry *dst,
					 unsigned long old)
{
	return NULL;
}

208 209
static struct dst_ops ip6_dst_blackhole_ops = {
	.family			=	AF_INET6,
210
	.protocol		=	cpu_to_be16(ETH_P_IPV6),
211 212
	.destroy		=	ip6_dst_destroy,
	.check			=	ip6_dst_check,
213
	.mtu			=	ip6_blackhole_mtu,
214
	.default_advmss		=	ip6_default_advmss,
215
	.update_pmtu		=	ip6_rt_blackhole_update_pmtu,
216
	.redirect		=	ip6_rt_blackhole_redirect,
217
	.cow_metrics		=	ip6_rt_blackhole_cow_metrics,
218
	.neigh_lookup		=	ip6_neigh_lookup,
219 220
};

221 222 223 224
static const u32 ip6_template_metrics[RTAX_MAX] = {
	[RTAX_HOPLIMIT - 1] = 255,
};

225
static struct rt6_info ip6_null_entry_template = {
226 227 228 229 230 231 232
	.dst = {
		.__refcnt	= ATOMIC_INIT(1),
		.__use		= 1,
		.obsolete	= -1,
		.error		= -ENETUNREACH,
		.input		= ip6_pkt_discard,
		.output		= ip6_pkt_discard_out,
L
Linus Torvalds 已提交
233 234
	},
	.rt6i_flags	= (RTF_REJECT | RTF_NONEXTHOP),
235
	.rt6i_protocol  = RTPROT_KERNEL,
L
Linus Torvalds 已提交
236 237 238 239
	.rt6i_metric	= ~(u32) 0,
	.rt6i_ref	= ATOMIC_INIT(1),
};

T
Thomas Graf 已提交
240 241
#ifdef CONFIG_IPV6_MULTIPLE_TABLES

242 243 244
static int ip6_pkt_prohibit(struct sk_buff *skb);
static int ip6_pkt_prohibit_out(struct sk_buff *skb);

245
static struct rt6_info ip6_prohibit_entry_template = {
246 247 248 249 250 251 252
	.dst = {
		.__refcnt	= ATOMIC_INIT(1),
		.__use		= 1,
		.obsolete	= -1,
		.error		= -EACCES,
		.input		= ip6_pkt_prohibit,
		.output		= ip6_pkt_prohibit_out,
T
Thomas Graf 已提交
253 254
	},
	.rt6i_flags	= (RTF_REJECT | RTF_NONEXTHOP),
255
	.rt6i_protocol  = RTPROT_KERNEL,
T
Thomas Graf 已提交
256 257 258 259
	.rt6i_metric	= ~(u32) 0,
	.rt6i_ref	= ATOMIC_INIT(1),
};

260
static struct rt6_info ip6_blk_hole_entry_template = {
261 262 263 264 265 266 267
	.dst = {
		.__refcnt	= ATOMIC_INIT(1),
		.__use		= 1,
		.obsolete	= -1,
		.error		= -EINVAL,
		.input		= dst_discard,
		.output		= dst_discard,
T
Thomas Graf 已提交
268 269
	},
	.rt6i_flags	= (RTF_REJECT | RTF_NONEXTHOP),
270
	.rt6i_protocol  = RTPROT_KERNEL,
T
Thomas Graf 已提交
271 272 273 274 275 276
	.rt6i_metric	= ~(u32) 0,
	.rt6i_ref	= ATOMIC_INIT(1),
};

#endif

L
Linus Torvalds 已提交
277
/* allocate dst with ip6_dst_ops */
278
static inline struct rt6_info *ip6_dst_alloc(struct net *net,
279
					     struct net_device *dev,
280 281
					     int flags,
					     struct fib6_table *table)
L
Linus Torvalds 已提交
282
{
283
	struct rt6_info *rt = dst_alloc(&net->ipv6.ip6_dst_ops, dev,
284
					0, DST_OBSOLETE_NONE, flags);
285

286
	if (rt) {
287 288 289
		struct dst_entry *dst = &rt->dst;

		memset(dst + 1, 0, sizeof(*rt) - sizeof(*dst));
290
		rt6_init_peer(rt, table ? &table->tb6_peers : net->ipv6.peers);
291
	}
292
	return rt;
L
Linus Torvalds 已提交
293 294 295 296 297 298 299
}

static void ip6_dst_destroy(struct dst_entry *dst)
{
	struct rt6_info *rt = (struct rt6_info *)dst;
	struct inet6_dev *idev = rt->rt6i_idev;

300 301 302
	if (rt->n)
		neigh_release(rt->n);

303 304 305
	if (!(rt->dst.flags & DST_HOST))
		dst_destroy_metrics_generic(dst);

306
	if (idev) {
L
Linus Torvalds 已提交
307 308
		rt->rt6i_idev = NULL;
		in6_dev_put(idev);
309
	}
310 311 312 313

	if (!(rt->rt6i_flags & RTF_EXPIRES) && dst->from)
		dst_release(dst->from);

314 315
	if (rt6_has_peer(rt)) {
		struct inet_peer *peer = rt6_peer_ptr(rt);
316 317 318 319
		inet_putpeer(peer);
	}
}

320 321 322 323 324 325 326
static atomic_t __rt6_peer_genid = ATOMIC_INIT(0);

static u32 rt6_peer_genid(void)
{
	return atomic_read(&__rt6_peer_genid);
}

327 328
void rt6_bind_peer(struct rt6_info *rt, int create)
{
329
	struct inet_peer_base *base;
330 331
	struct inet_peer *peer;

332 333 334 335 336
	base = inetpeer_base_ptr(rt->_rt6i_peer);
	if (!base)
		return;

	peer = inet_getpeer_v6(base, &rt->rt6i_dst.addr, create);
337 338 339 340 341 342
	if (peer) {
		if (!rt6_set_peer(rt, peer))
			inet_putpeer(peer);
		else
			rt->rt6i_peer_genid = rt6_peer_genid();
	}
L
Linus Torvalds 已提交
343 344 345 346 347 348 349
}

static void ip6_dst_ifdown(struct dst_entry *dst, struct net_device *dev,
			   int how)
{
	struct rt6_info *rt = (struct rt6_info *)dst;
	struct inet6_dev *idev = rt->rt6i_idev;
350
	struct net_device *loopback_dev =
351
		dev_net(dev)->loopback_dev;
L
Linus Torvalds 已提交
352

353 354 355 356 357 358 359 360 361 362 363 364 365
	if (dev != loopback_dev) {
		if (idev && idev->dev == dev) {
			struct inet6_dev *loopback_idev =
				in6_dev_get(loopback_dev);
			if (loopback_idev) {
				rt->rt6i_idev = loopback_idev;
				in6_dev_put(idev);
			}
		}
		if (rt->n && rt->n->dev == dev) {
			rt->n->dev = loopback_dev;
			dev_hold(loopback_dev);
			dev_put(dev);
L
Linus Torvalds 已提交
366 367 368 369
		}
	}
}

370
static bool rt6_check_expired(const struct rt6_info *rt)
L
Linus Torvalds 已提交
371
{
372 373 374 375
	struct rt6_info *ort = NULL;

	if (rt->rt6i_flags & RTF_EXPIRES) {
		if (time_after(jiffies, rt->dst.expires))
376
			return true;
377 378 379 380 381
	} else if (rt->dst.from) {
		ort = (struct rt6_info *) rt->dst.from;
		return (ort->rt6i_flags & RTF_EXPIRES) &&
			time_after(jiffies, ort->dst.expires);
	}
382
	return false;
L
Linus Torvalds 已提交
383 384
}

385
static bool rt6_need_strict(const struct in6_addr *daddr)
T
Thomas Graf 已提交
386
{
E
Eric Dumazet 已提交
387 388
	return ipv6_addr_type(daddr) &
		(IPV6_ADDR_MULTICAST | IPV6_ADDR_LINKLOCAL | IPV6_ADDR_LOOPBACK);
T
Thomas Graf 已提交
389 390
}

L
Linus Torvalds 已提交
391
/*
T
Thomas Graf 已提交
392
 *	Route lookup. Any table->tb6_lock is implied.
L
Linus Torvalds 已提交
393 394
 */

395 396
static inline struct rt6_info *rt6_device_match(struct net *net,
						    struct rt6_info *rt,
397
						    const struct in6_addr *saddr,
L
Linus Torvalds 已提交
398
						    int oif,
399
						    int flags)
L
Linus Torvalds 已提交
400 401 402 403
{
	struct rt6_info *local = NULL;
	struct rt6_info *sprt;

404 405 406
	if (!oif && ipv6_addr_any(saddr))
		goto out;

407
	for (sprt = rt; sprt; sprt = sprt->dst.rt6_next) {
408
		struct net_device *dev = sprt->dst.dev;
409 410

		if (oif) {
L
Linus Torvalds 已提交
411 412 413
			if (dev->ifindex == oif)
				return sprt;
			if (dev->flags & IFF_LOOPBACK) {
414
				if (!sprt->rt6i_idev ||
L
Linus Torvalds 已提交
415
				    sprt->rt6i_idev->dev->ifindex != oif) {
416
					if (flags & RT6_LOOKUP_F_IFACE && oif)
L
Linus Torvalds 已提交
417
						continue;
418
					if (local && (!oif ||
L
Linus Torvalds 已提交
419 420 421 422 423
						      local->rt6i_idev->dev->ifindex == oif))
						continue;
				}
				local = sprt;
			}
424 425 426 427
		} else {
			if (ipv6_chk_addr(net, saddr, dev,
					  flags & RT6_LOOKUP_F_IFACE))
				return sprt;
L
Linus Torvalds 已提交
428
		}
429
	}
L
Linus Torvalds 已提交
430

431
	if (oif) {
L
Linus Torvalds 已提交
432 433 434
		if (local)
			return local;

435
		if (flags & RT6_LOOKUP_F_IFACE)
436
			return net->ipv6.ip6_null_entry;
L
Linus Torvalds 已提交
437
	}
438
out:
L
Linus Torvalds 已提交
439 440 441
	return rt;
}

442 443 444
#ifdef CONFIG_IPV6_ROUTER_PREF
static void rt6_probe(struct rt6_info *rt)
{
445
	struct neighbour *neigh;
446 447 448 449 450 451 452 453
	/*
	 * Okay, this does not seem to be appropriate
	 * for now, however, we need to check if it
	 * is really so; aka Router Reachability Probing.
	 *
	 * Router Reachability Probe MUST be rate-limited
	 * to no more than one per minute.
	 */
454
	neigh = rt ? rt->n : NULL;
455
	if (!neigh || (neigh->nud_state & NUD_VALID))
456
		return;
457 458
	read_lock_bh(&neigh->lock);
	if (!(neigh->nud_state & NUD_VALID) &&
459
	    time_after(jiffies, neigh->updated + rt->rt6i_idev->cnf.rtr_probe_interval)) {
460 461 462 463 464 465 466 467
		struct in6_addr mcaddr;
		struct in6_addr *target;

		neigh->updated = jiffies;
		read_unlock_bh(&neigh->lock);

		target = (struct in6_addr *)&neigh->primary_key;
		addrconf_addr_solict_mult(target, &mcaddr);
468
		ndisc_send_ns(rt->dst.dev, NULL, target, &mcaddr, NULL);
469
	} else {
470
		read_unlock_bh(&neigh->lock);
471
	}
472 473 474 475 476 477 478
}
#else
static inline void rt6_probe(struct rt6_info *rt)
{
}
#endif

L
Linus Torvalds 已提交
479
/*
480
 * Default Router Selection (RFC 2461 6.3.6)
L
Linus Torvalds 已提交
481
 */
D
Dave Jones 已提交
482
static inline int rt6_check_dev(struct rt6_info *rt, int oif)
483
{
484
	struct net_device *dev = rt->dst.dev;
485
	if (!oif || dev->ifindex == oif)
486
		return 2;
487 488 489 490
	if ((dev->flags & IFF_LOOPBACK) &&
	    rt->rt6i_idev && rt->rt6i_idev->dev->ifindex == oif)
		return 1;
	return 0;
491
}
L
Linus Torvalds 已提交
492

D
Dave Jones 已提交
493
static inline int rt6_check_neigh(struct rt6_info *rt)
L
Linus Torvalds 已提交
494
{
495
	struct neighbour *neigh;
496
	int m;
497

498
	neigh = rt->n;
499 500 501 502
	if (rt->rt6i_flags & RTF_NONEXTHOP ||
	    !(rt->rt6i_flags & RTF_GATEWAY))
		m = 1;
	else if (neigh) {
503 504
		read_lock_bh(&neigh->lock);
		if (neigh->nud_state & NUD_VALID)
505
			m = 2;
506 507 508 509 510
#ifdef CONFIG_IPV6_ROUTER_PREF
		else if (neigh->nud_state & NUD_FAILED)
			m = 0;
#endif
		else
511
			m = 1;
512
		read_unlock_bh(&neigh->lock);
513 514
	} else
		m = 0;
515
	return m;
L
Linus Torvalds 已提交
516 517
}

518 519
static int rt6_score_route(struct rt6_info *rt, int oif,
			   int strict)
L
Linus Torvalds 已提交
520
{
521
	int m, n;
522

523
	m = rt6_check_dev(rt, oif);
524
	if (!m && (strict & RT6_LOOKUP_F_IFACE))
525
		return -1;
526 527 528
#ifdef CONFIG_IPV6_ROUTER_PREF
	m |= IPV6_DECODE_PREF(IPV6_EXTRACT_PREF(rt->rt6i_flags)) << 2;
#endif
529
	n = rt6_check_neigh(rt);
530
	if (!n && (strict & RT6_LOOKUP_F_REACHABLE))
531 532 533 534
		return -1;
	return m;
}

535 536
static struct rt6_info *find_match(struct rt6_info *rt, int oif, int strict,
				   int *mpri, struct rt6_info *match)
537
{
538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564
	int m;

	if (rt6_check_expired(rt))
		goto out;

	m = rt6_score_route(rt, oif, strict);
	if (m < 0)
		goto out;

	if (m > *mpri) {
		if (strict & RT6_LOOKUP_F_REACHABLE)
			rt6_probe(match);
		*mpri = m;
		match = rt;
	} else if (strict & RT6_LOOKUP_F_REACHABLE) {
		rt6_probe(rt);
	}

out:
	return match;
}

static struct rt6_info *find_rr_leaf(struct fib6_node *fn,
				     struct rt6_info *rr_head,
				     u32 metric, int oif, int strict)
{
	struct rt6_info *rt, *match;
565
	int mpri = -1;
L
Linus Torvalds 已提交
566

567 568
	match = NULL;
	for (rt = rr_head; rt && rt->rt6i_metric == metric;
569
	     rt = rt->dst.rt6_next)
570 571
		match = find_match(rt, oif, strict, &mpri, match);
	for (rt = fn->leaf; rt && rt != rr_head && rt->rt6i_metric == metric;
572
	     rt = rt->dst.rt6_next)
573
		match = find_match(rt, oif, strict, &mpri, match);
L
Linus Torvalds 已提交
574

575 576
	return match;
}
L
Linus Torvalds 已提交
577

578 579 580
static struct rt6_info *rt6_select(struct fib6_node *fn, int oif, int strict)
{
	struct rt6_info *match, *rt0;
581
	struct net *net;
L
Linus Torvalds 已提交
582

583 584 585
	rt0 = fn->rr_ptr;
	if (!rt0)
		fn->rr_ptr = rt0 = fn->leaf;
L
Linus Torvalds 已提交
586

587
	match = find_rr_leaf(fn, rt0, rt0->rt6i_metric, oif, strict);
L
Linus Torvalds 已提交
588

589
	if (!match &&
590
	    (strict & RT6_LOOKUP_F_REACHABLE)) {
591
		struct rt6_info *next = rt0->dst.rt6_next;
592

593
		/* no entries matched; do round-robin */
594 595 596 597 598
		if (!next || next->rt6i_metric != rt0->rt6i_metric)
			next = fn->leaf;

		if (next != rt0)
			fn->rr_ptr = next;
L
Linus Torvalds 已提交
599 600
	}

601
	net = dev_net(rt0->dst.dev);
E
Eric Dumazet 已提交
602
	return match ? match : net->ipv6.ip6_null_entry;
L
Linus Torvalds 已提交
603 604
}

605 606
#ifdef CONFIG_IPV6_ROUTE_INFO
int rt6_route_rcv(struct net_device *dev, u8 *opt, int len,
607
		  const struct in6_addr *gwaddr)
608
{
609
	struct net *net = dev_net(dev);
610 611 612
	struct route_info *rinfo = (struct route_info *) opt;
	struct in6_addr prefix_buf, *prefix;
	unsigned int pref;
613
	unsigned long lifetime;
614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636
	struct rt6_info *rt;

	if (len < sizeof(struct route_info)) {
		return -EINVAL;
	}

	/* Sanity check for prefix_len and length */
	if (rinfo->length > 3) {
		return -EINVAL;
	} else if (rinfo->prefix_len > 128) {
		return -EINVAL;
	} else if (rinfo->prefix_len > 64) {
		if (rinfo->length < 2) {
			return -EINVAL;
		}
	} else if (rinfo->prefix_len > 0) {
		if (rinfo->length < 1) {
			return -EINVAL;
		}
	}

	pref = rinfo->route_pref;
	if (pref == ICMPV6_ROUTER_PREF_INVALID)
637
		return -EINVAL;
638

639
	lifetime = addrconf_timeout_fixup(ntohl(rinfo->lifetime), HZ);
640 641 642 643 644 645 646 647 648 649 650

	if (rinfo->length == 3)
		prefix = (struct in6_addr *)rinfo->prefix;
	else {
		/* this function is safe */
		ipv6_addr_prefix(&prefix_buf,
				 (struct in6_addr *)rinfo->prefix,
				 rinfo->prefix_len);
		prefix = &prefix_buf;
	}

651 652
	rt = rt6_get_route_info(net, prefix, rinfo->prefix_len, gwaddr,
				dev->ifindex);
653 654

	if (rt && !lifetime) {
655
		ip6_del_rt(rt);
656 657 658 659
		rt = NULL;
	}

	if (!rt && lifetime)
660
		rt = rt6_add_route_info(net, prefix, rinfo->prefix_len, gwaddr, dev->ifindex,
661 662 663 664 665 666
					pref);
	else if (rt)
		rt->rt6i_flags = RTF_ROUTEINFO |
				 (rt->rt6i_flags & ~RTF_PREF_MASK) | RTF_PREF(pref);

	if (rt) {
667 668 669 670 671
		if (!addrconf_finite_timeout(lifetime))
			rt6_clean_expires(rt);
		else
			rt6_set_expires(rt, jiffies + HZ * lifetime);

672
		dst_release(&rt->dst);
673 674 675 676 677
	}
	return 0;
}
#endif

678
#define BACKTRACK(__net, saddr)			\
679
do { \
680
	if (rt == __net->ipv6.ip6_null_entry) {	\
681
		struct fib6_node *pn; \
V
Ville Nuorvala 已提交
682
		while (1) { \
683 684 685 686
			if (fn->fn_flags & RTN_TL_ROOT) \
				goto out; \
			pn = fn->parent; \
			if (FIB6_SUBTREE(pn) && FIB6_SUBTREE(pn) != fn) \
687
				fn = fib6_lookup(FIB6_SUBTREE(pn), NULL, saddr); \
688 689 690 691
			else \
				fn = pn; \
			if (fn->fn_flags & RTN_RTINFO) \
				goto restart; \
T
Thomas Graf 已提交
692 693
		} \
	} \
694
} while (0)
T
Thomas Graf 已提交
695

696 697
static struct rt6_info *ip6_pol_route_lookup(struct net *net,
					     struct fib6_table *table,
698
					     struct flowi6 *fl6, int flags)
L
Linus Torvalds 已提交
699 700 701 702
{
	struct fib6_node *fn;
	struct rt6_info *rt;

T
Thomas Graf 已提交
703
	read_lock_bh(&table->tb6_lock);
704
	fn = fib6_lookup(&table->tb6_root, &fl6->daddr, &fl6->saddr);
T
Thomas Graf 已提交
705 706
restart:
	rt = fn->leaf;
707 708
	rt = rt6_device_match(net, rt, &fl6->saddr, fl6->flowi6_oif, flags);
	BACKTRACK(net, &fl6->saddr);
T
Thomas Graf 已提交
709
out:
710
	dst_use(&rt->dst, jiffies);
T
Thomas Graf 已提交
711 712 713 714 715
	read_unlock_bh(&table->tb6_lock);
	return rt;

}

F
Florian Westphal 已提交
716 717 718 719 720 721 722
struct dst_entry * ip6_route_lookup(struct net *net, struct flowi6 *fl6,
				    int flags)
{
	return fib6_rule_lookup(net, fl6, flags, ip6_pol_route_lookup);
}
EXPORT_SYMBOL_GPL(ip6_route_lookup);

723 724
struct rt6_info *rt6_lookup(struct net *net, const struct in6_addr *daddr,
			    const struct in6_addr *saddr, int oif, int strict)
T
Thomas Graf 已提交
725
{
726 727 728
	struct flowi6 fl6 = {
		.flowi6_oif = oif,
		.daddr = *daddr,
T
Thomas Graf 已提交
729 730
	};
	struct dst_entry *dst;
731
	int flags = strict ? RT6_LOOKUP_F_IFACE : 0;
T
Thomas Graf 已提交
732

733
	if (saddr) {
734
		memcpy(&fl6.saddr, saddr, sizeof(*saddr));
735 736 737
		flags |= RT6_LOOKUP_F_HAS_SADDR;
	}

738
	dst = fib6_rule_lookup(net, &fl6, flags, ip6_pol_route_lookup);
T
Thomas Graf 已提交
739 740 741 742 743
	if (dst->error == 0)
		return (struct rt6_info *) dst;

	dst_release(dst);

L
Linus Torvalds 已提交
744 745 746
	return NULL;
}

747 748
EXPORT_SYMBOL(rt6_lookup);

T
Thomas Graf 已提交
749
/* ip6_ins_rt is called with FREE table->tb6_lock.
L
Linus Torvalds 已提交
750 751 752 753 754
   It takes new route entry, the addition fails by any reason the
   route is freed. In any case, if caller does not hold it, it may
   be destroyed.
 */

755
static int __ip6_ins_rt(struct rt6_info *rt, struct nl_info *info)
L
Linus Torvalds 已提交
756 757
{
	int err;
T
Thomas Graf 已提交
758
	struct fib6_table *table;
L
Linus Torvalds 已提交
759

T
Thomas Graf 已提交
760 761
	table = rt->rt6i_table;
	write_lock_bh(&table->tb6_lock);
762
	err = fib6_add(&table->tb6_root, rt, info);
T
Thomas Graf 已提交
763
	write_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
764 765 766 767

	return err;
}

768 769
int ip6_ins_rt(struct rt6_info *rt)
{
770
	struct nl_info info = {
771
		.nl_net = dev_net(rt->dst.dev),
772
	};
773
	return __ip6_ins_rt(rt, &info);
774 775
}

776
static struct rt6_info *rt6_alloc_cow(struct rt6_info *ort,
E
Eric Dumazet 已提交
777
				      const struct in6_addr *daddr,
778
				      const struct in6_addr *saddr)
L
Linus Torvalds 已提交
779 780 781 782 783 784 785
{
	struct rt6_info *rt;

	/*
	 *	Clone the route.
	 */

E
Eric Dumazet 已提交
786
	rt = ip6_rt_copy(ort, daddr);
L
Linus Torvalds 已提交
787 788

	if (rt) {
789 790
		int attempts = !in_softirq();

791
		if (!(rt->rt6i_flags & RTF_GATEWAY)) {
792
			if (ort->rt6i_dst.plen != 128 &&
E
Eric Dumazet 已提交
793
			    ipv6_addr_equal(&ort->rt6i_dst.addr, daddr))
794
				rt->rt6i_flags |= RTF_ANYCAST;
A
Alexey Dobriyan 已提交
795
			rt->rt6i_gateway = *daddr;
796
		}
L
Linus Torvalds 已提交
797 798 799 800 801

		rt->rt6i_flags |= RTF_CACHE;

#ifdef CONFIG_IPV6_SUBTREES
		if (rt->rt6i_src.plen && saddr) {
A
Alexey Dobriyan 已提交
802
			rt->rt6i_src.addr = *saddr;
L
Linus Torvalds 已提交
803 804 805 806
			rt->rt6i_src.plen = 128;
		}
#endif

807
	retry:
808
		if (rt6_bind_neighbour(rt, rt->dst.dev)) {
809
			struct net *net = dev_net(rt->dst.dev);
810 811 812 813 814 815 816 817 818
			int saved_rt_min_interval =
				net->ipv6.sysctl.ip6_rt_gc_min_interval;
			int saved_rt_elasticity =
				net->ipv6.sysctl.ip6_rt_gc_elasticity;

			if (attempts-- > 0) {
				net->ipv6.sysctl.ip6_rt_gc_elasticity = 1;
				net->ipv6.sysctl.ip6_rt_gc_min_interval = 0;

819
				ip6_dst_gc(&net->ipv6.ip6_dst_ops);
820 821 822 823 824 825 826 827

				net->ipv6.sysctl.ip6_rt_gc_elasticity =
					saved_rt_elasticity;
				net->ipv6.sysctl.ip6_rt_gc_min_interval =
					saved_rt_min_interval;
				goto retry;
			}

828
			net_warn_ratelimited("Neighbour table overflow\n");
829
			dst_free(&rt->dst);
830 831
			return NULL;
		}
832
	}
L
Linus Torvalds 已提交
833

834 835
	return rt;
}
L
Linus Torvalds 已提交
836

E
Eric Dumazet 已提交
837 838
static struct rt6_info *rt6_alloc_clone(struct rt6_info *ort,
					const struct in6_addr *daddr)
839
{
E
Eric Dumazet 已提交
840 841
	struct rt6_info *rt = ip6_rt_copy(ort, daddr);

842 843
	if (rt) {
		rt->rt6i_flags |= RTF_CACHE;
844
		rt->n = neigh_clone(ort->n);
845 846 847 848
	}
	return rt;
}

849
static struct rt6_info *ip6_pol_route(struct net *net, struct fib6_table *table, int oif,
850
				      struct flowi6 *fl6, int flags)
L
Linus Torvalds 已提交
851 852
{
	struct fib6_node *fn;
853
	struct rt6_info *rt, *nrt;
T
Thomas Graf 已提交
854
	int strict = 0;
L
Linus Torvalds 已提交
855
	int attempts = 3;
856
	int err;
857
	int reachable = net->ipv6.devconf_all->forwarding ? 0 : RT6_LOOKUP_F_REACHABLE;
L
Linus Torvalds 已提交
858

859
	strict |= flags & RT6_LOOKUP_F_IFACE;
L
Linus Torvalds 已提交
860 861

relookup:
T
Thomas Graf 已提交
862
	read_lock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
863

864
restart_2:
865
	fn = fib6_lookup(&table->tb6_root, &fl6->daddr, &fl6->saddr);
L
Linus Torvalds 已提交
866 867

restart:
868
	rt = rt6_select(fn, oif, strict | reachable);
869

870
	BACKTRACK(net, &fl6->saddr);
871
	if (rt == net->ipv6.ip6_null_entry ||
872
	    rt->rt6i_flags & RTF_CACHE)
873
		goto out;
L
Linus Torvalds 已提交
874

875
	dst_hold(&rt->dst);
T
Thomas Graf 已提交
876
	read_unlock_bh(&table->tb6_lock);
877

878
	if (!rt->n && !(rt->rt6i_flags & RTF_NONEXTHOP))
879
		nrt = rt6_alloc_cow(rt, &fl6->daddr, &fl6->saddr);
880
	else if (!(rt->dst.flags & DST_HOST))
881
		nrt = rt6_alloc_clone(rt, &fl6->daddr);
882 883
	else
		goto out2;
884

885
	dst_release(&rt->dst);
886
	rt = nrt ? : net->ipv6.ip6_null_entry;
L
Linus Torvalds 已提交
887

888
	dst_hold(&rt->dst);
889
	if (nrt) {
890
		err = ip6_ins_rt(nrt);
891
		if (!err)
L
Linus Torvalds 已提交
892 893 894
			goto out2;
	}

895 896 897 898
	if (--attempts <= 0)
		goto out2;

	/*
T
Thomas Graf 已提交
899
	 * Race condition! In the gap, when table->tb6_lock was
900 901
	 * released someone could insert this route.  Relookup.
	 */
902
	dst_release(&rt->dst);
903 904 905
	goto relookup;

out:
906 907 908 909
	if (reachable) {
		reachable = 0;
		goto restart_2;
	}
910
	dst_hold(&rt->dst);
T
Thomas Graf 已提交
911
	read_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
912
out2:
913 914
	rt->dst.lastuse = jiffies;
	rt->dst.__use++;
T
Thomas Graf 已提交
915 916

	return rt;
L
Linus Torvalds 已提交
917 918
}

919
static struct rt6_info *ip6_pol_route_input(struct net *net, struct fib6_table *table,
920
					    struct flowi6 *fl6, int flags)
921
{
922
	return ip6_pol_route(net, table, fl6->flowi6_iif, fl6, flags);
923 924
}

925 926 927 928 929 930 931 932 933 934
static struct dst_entry *ip6_route_input_lookup(struct net *net,
						struct net_device *dev,
						struct flowi6 *fl6, int flags)
{
	if (rt6_need_strict(&fl6->daddr) && dev->type != ARPHRD_PIMREG)
		flags |= RT6_LOOKUP_F_IFACE;

	return fib6_rule_lookup(net, fl6, flags, ip6_pol_route_input);
}

T
Thomas Graf 已提交
935 936
void ip6_route_input(struct sk_buff *skb)
{
937
	const struct ipv6hdr *iph = ipv6_hdr(skb);
938
	struct net *net = dev_net(skb->dev);
939
	int flags = RT6_LOOKUP_F_HAS_SADDR;
940 941 942 943
	struct flowi6 fl6 = {
		.flowi6_iif = skb->dev->ifindex,
		.daddr = iph->daddr,
		.saddr = iph->saddr,
944
		.flowlabel = (* (__be32 *) iph) & IPV6_FLOWINFO_MASK,
945 946
		.flowi6_mark = skb->mark,
		.flowi6_proto = iph->nexthdr,
T
Thomas Graf 已提交
947
	};
948

949
	skb_dst_set(skb, ip6_route_input_lookup(net, skb->dev, &fl6, flags));
T
Thomas Graf 已提交
950 951
}

952
static struct rt6_info *ip6_pol_route_output(struct net *net, struct fib6_table *table,
953
					     struct flowi6 *fl6, int flags)
L
Linus Torvalds 已提交
954
{
955
	return ip6_pol_route(net, table, fl6->flowi6_oif, fl6, flags);
T
Thomas Graf 已提交
956 957
}

958
struct dst_entry * ip6_route_output(struct net *net, const struct sock *sk,
959
				    struct flowi6 *fl6)
T
Thomas Graf 已提交
960 961 962
{
	int flags = 0;

963
	fl6->flowi6_iif = LOOPBACK_IFINDEX;
964

965
	if ((sk && sk->sk_bound_dev_if) || rt6_need_strict(&fl6->daddr))
966
		flags |= RT6_LOOKUP_F_IFACE;
T
Thomas Graf 已提交
967

968
	if (!ipv6_addr_any(&fl6->saddr))
969
		flags |= RT6_LOOKUP_F_HAS_SADDR;
970 971
	else if (sk)
		flags |= rt6_srcprefs2flags(inet6_sk(sk)->srcprefs);
972

973
	return fib6_rule_lookup(net, fl6, flags, ip6_pol_route_output);
L
Linus Torvalds 已提交
974 975
}

976
EXPORT_SYMBOL(ip6_route_output);
L
Linus Torvalds 已提交
977

978
struct dst_entry *ip6_blackhole_route(struct net *net, struct dst_entry *dst_orig)
979
{
980
	struct rt6_info *rt, *ort = (struct rt6_info *) dst_orig;
981 982
	struct dst_entry *new = NULL;

983
	rt = dst_alloc(&ip6_dst_blackhole_ops, ort->dst.dev, 1, DST_OBSOLETE_NONE, 0);
984
	if (rt) {
985
		new = &rt->dst;
986

987 988 989
		memset(new + 1, 0, sizeof(*rt) - sizeof(*new));
		rt6_init_peer(rt, net->ipv6.peers);

990
		new->__use = 1;
991 992
		new->input = dst_discard;
		new->output = dst_discard;
993

E
Eric Dumazet 已提交
994 995 996 997
		if (dst_metrics_read_only(&ort->dst))
			new->_metrics = ort->dst._metrics;
		else
			dst_copy_metrics(new, &ort->dst);
998 999 1000 1001
		rt->rt6i_idev = ort->rt6i_idev;
		if (rt->rt6i_idev)
			in6_dev_hold(rt->rt6i_idev);

A
Alexey Dobriyan 已提交
1002
		rt->rt6i_gateway = ort->rt6i_gateway;
1003 1004
		rt->rt6i_flags = ort->rt6i_flags;
		rt6_clean_expires(rt);
1005 1006 1007 1008 1009 1010 1011 1012 1013 1014
		rt->rt6i_metric = 0;

		memcpy(&rt->rt6i_dst, &ort->rt6i_dst, sizeof(struct rt6key));
#ifdef CONFIG_IPV6_SUBTREES
		memcpy(&rt->rt6i_src, &ort->rt6i_src, sizeof(struct rt6key));
#endif

		dst_free(new);
	}

1015 1016
	dst_release(dst_orig);
	return new ? new : ERR_PTR(-ENOMEM);
1017 1018
}

L
Linus Torvalds 已提交
1019 1020 1021 1022 1023 1024 1025 1026 1027 1028
/*
 *	Destination cache support functions
 */

static struct dst_entry *ip6_dst_check(struct dst_entry *dst, u32 cookie)
{
	struct rt6_info *rt;

	rt = (struct rt6_info *) dst;

1029 1030
	if (rt->rt6i_node && (rt->rt6i_node->fn_sernum == cookie)) {
		if (rt->rt6i_peer_genid != rt6_peer_genid()) {
1031
			if (!rt6_has_peer(rt))
1032 1033 1034
				rt6_bind_peer(rt, 0);
			rt->rt6i_peer_genid = rt6_peer_genid();
		}
L
Linus Torvalds 已提交
1035
		return dst;
1036
	}
L
Linus Torvalds 已提交
1037 1038 1039 1040 1041 1042 1043 1044
	return NULL;
}

static struct dst_entry *ip6_negative_advice(struct dst_entry *dst)
{
	struct rt6_info *rt = (struct rt6_info *) dst;

	if (rt) {
1045 1046 1047 1048 1049 1050
		if (rt->rt6i_flags & RTF_CACHE) {
			if (rt6_check_expired(rt)) {
				ip6_del_rt(rt);
				dst = NULL;
			}
		} else {
L
Linus Torvalds 已提交
1051
			dst_release(dst);
1052 1053
			dst = NULL;
		}
L
Linus Torvalds 已提交
1054
	}
1055
	return dst;
L
Linus Torvalds 已提交
1056 1057 1058 1059 1060 1061
}

static void ip6_link_failure(struct sk_buff *skb)
{
	struct rt6_info *rt;

1062
	icmpv6_send(skb, ICMPV6_DEST_UNREACH, ICMPV6_ADDR_UNREACH, 0);
L
Linus Torvalds 已提交
1063

E
Eric Dumazet 已提交
1064
	rt = (struct rt6_info *) skb_dst(skb);
L
Linus Torvalds 已提交
1065
	if (rt) {
1066 1067 1068
		if (rt->rt6i_flags & RTF_CACHE)
			rt6_update_expires(rt, 0);
		else if (rt->rt6i_node && (rt->rt6i_flags & RTF_DEFAULT))
L
Linus Torvalds 已提交
1069 1070 1071 1072
			rt->rt6i_node->fn_sernum = -1;
	}
}

1073 1074
static void ip6_rt_update_pmtu(struct dst_entry *dst, struct sock *sk,
			       struct sk_buff *skb, u32 mtu)
L
Linus Torvalds 已提交
1075 1076 1077
{
	struct rt6_info *rt6 = (struct rt6_info*)dst;

1078
	dst_confirm(dst);
L
Linus Torvalds 已提交
1079
	if (mtu < dst_mtu(dst) && rt6->rt6i_dst.plen == 128) {
1080 1081
		struct net *net = dev_net(dst->dev);

L
Linus Torvalds 已提交
1082 1083
		rt6->rt6i_flags |= RTF_MODIFIED;
		if (mtu < IPV6_MIN_MTU) {
1084
			u32 features = dst_metric(dst, RTAX_FEATURES);
L
Linus Torvalds 已提交
1085
			mtu = IPV6_MIN_MTU;
1086 1087
			features |= RTAX_FEATURE_ALLFRAG;
			dst_metric_set(dst, RTAX_FEATURES, features);
L
Linus Torvalds 已提交
1088
		}
1089
		dst_metric_set(dst, RTAX_MTU, mtu);
1090
		rt6_update_expires(rt6, net->ipv6.sysctl.ip6_rt_mtu_expires);
L
Linus Torvalds 已提交
1091 1092 1093
	}
}

1094 1095
void ip6_update_pmtu(struct sk_buff *skb, struct net *net, __be32 mtu,
		     int oif, u32 mark)
1096 1097 1098 1099 1100 1101 1102 1103
{
	const struct ipv6hdr *iph = (struct ipv6hdr *) skb->data;
	struct dst_entry *dst;
	struct flowi6 fl6;

	memset(&fl6, 0, sizeof(fl6));
	fl6.flowi6_oif = oif;
	fl6.flowi6_mark = mark;
1104
	fl6.flowi6_flags = 0;
1105 1106 1107 1108 1109 1110
	fl6.daddr = iph->daddr;
	fl6.saddr = iph->saddr;
	fl6.flowlabel = (*(__be32 *) iph) & IPV6_FLOWINFO_MASK;

	dst = ip6_route_output(net, NULL, &fl6);
	if (!dst->error)
1111
		ip6_rt_update_pmtu(dst, NULL, skb, ntohl(mtu));
1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122
	dst_release(dst);
}
EXPORT_SYMBOL_GPL(ip6_update_pmtu);

void ip6_sk_update_pmtu(struct sk_buff *skb, struct sock *sk, __be32 mtu)
{
	ip6_update_pmtu(skb, sock_net(sk), mtu,
			sk->sk_bound_dev_if, sk->sk_mark);
}
EXPORT_SYMBOL_GPL(ip6_sk_update_pmtu);

1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138
void ip6_redirect(struct sk_buff *skb, struct net *net, int oif, u32 mark)
{
	const struct ipv6hdr *iph = (struct ipv6hdr *) skb->data;
	struct dst_entry *dst;
	struct flowi6 fl6;

	memset(&fl6, 0, sizeof(fl6));
	fl6.flowi6_oif = oif;
	fl6.flowi6_mark = mark;
	fl6.flowi6_flags = 0;
	fl6.daddr = iph->daddr;
	fl6.saddr = iph->saddr;
	fl6.flowlabel = (*(__be32 *) iph) & IPV6_FLOWINFO_MASK;

	dst = ip6_route_output(net, NULL, &fl6);
	if (!dst->error)
1139
		rt6_do_redirect(dst, NULL, skb);
1140 1141 1142 1143 1144 1145 1146 1147 1148 1149
	dst_release(dst);
}
EXPORT_SYMBOL_GPL(ip6_redirect);

void ip6_sk_redirect(struct sk_buff *skb, struct sock *sk)
{
	ip6_redirect(skb, sock_net(sk), sk->sk_bound_dev_if, sk->sk_mark);
}
EXPORT_SYMBOL_GPL(ip6_sk_redirect);

1150
static unsigned int ip6_default_advmss(const struct dst_entry *dst)
L
Linus Torvalds 已提交
1151
{
1152 1153 1154 1155
	struct net_device *dev = dst->dev;
	unsigned int mtu = dst_mtu(dst);
	struct net *net = dev_net(dev);

L
Linus Torvalds 已提交
1156 1157
	mtu -= sizeof(struct ipv6hdr) + sizeof(struct tcphdr);

1158 1159
	if (mtu < net->ipv6.sysctl.ip6_rt_min_advmss)
		mtu = net->ipv6.sysctl.ip6_rt_min_advmss;
L
Linus Torvalds 已提交
1160 1161

	/*
1162 1163 1164
	 * Maximal non-jumbo IPv6 payload is IPV6_MAXPLEN and
	 * corresponding MSS is IPV6_MAXPLEN - tcp_header_size.
	 * IPV6_MAXPLEN is also valid and means: "any MSS,
L
Linus Torvalds 已提交
1165 1166 1167 1168 1169 1170 1171
	 * rely only on pmtu discovery"
	 */
	if (mtu > IPV6_MAXPLEN - sizeof(struct tcphdr))
		mtu = IPV6_MAXPLEN;
	return mtu;
}

1172
static unsigned int ip6_mtu(const struct dst_entry *dst)
1173 1174
{
	struct inet6_dev *idev;
1175 1176 1177 1178 1179 1180
	unsigned int mtu = dst_metric_raw(dst, RTAX_MTU);

	if (mtu)
		return mtu;

	mtu = IPV6_MIN_MTU;
1181 1182 1183 1184 1185 1186 1187 1188 1189 1190

	rcu_read_lock();
	idev = __in6_dev_get(dst->dev);
	if (idev)
		mtu = idev->cnf.mtu6;
	rcu_read_unlock();

	return mtu;
}

1191 1192
static struct dst_entry *icmp6_dst_gc_list;
static DEFINE_SPINLOCK(icmp6_dst_lock);
1193

1194
struct dst_entry *icmp6_dst_alloc(struct net_device *dev,
L
Linus Torvalds 已提交
1195
				  struct neighbour *neigh,
1196
				  struct flowi6 *fl6)
L
Linus Torvalds 已提交
1197
{
1198
	struct dst_entry *dst;
L
Linus Torvalds 已提交
1199 1200
	struct rt6_info *rt;
	struct inet6_dev *idev = in6_dev_get(dev);
1201
	struct net *net = dev_net(dev);
L
Linus Torvalds 已提交
1202

1203
	if (unlikely(!idev))
E
Eric Dumazet 已提交
1204
		return ERR_PTR(-ENODEV);
L
Linus Torvalds 已提交
1205

1206
	rt = ip6_dst_alloc(net, dev, 0, NULL);
1207
	if (unlikely(!rt)) {
L
Linus Torvalds 已提交
1208
		in6_dev_put(idev);
1209
		dst = ERR_PTR(-ENOMEM);
L
Linus Torvalds 已提交
1210 1211 1212 1213 1214
		goto out;
	}

	if (neigh)
		neigh_hold(neigh);
1215
	else {
1216
		neigh = ip6_neigh_lookup(&rt->dst, NULL, &fl6->daddr);
1217
		if (IS_ERR(neigh)) {
1218
			in6_dev_put(idev);
1219 1220 1221
			dst_free(&rt->dst);
			return ERR_CAST(neigh);
		}
1222
	}
L
Linus Torvalds 已提交
1223

1224 1225
	rt->dst.flags |= DST_HOST;
	rt->dst.output  = ip6_output;
1226
	rt->n = neigh;
1227
	atomic_set(&rt->dst.__refcnt, 1);
1228
	rt->rt6i_dst.addr = fl6->daddr;
1229 1230
	rt->rt6i_dst.plen = 128;
	rt->rt6i_idev     = idev;
1231
	dst_metric_set(&rt->dst, RTAX_HOPLIMIT, 255);
L
Linus Torvalds 已提交
1232

1233
	spin_lock_bh(&icmp6_dst_lock);
1234 1235
	rt->dst.next = icmp6_dst_gc_list;
	icmp6_dst_gc_list = &rt->dst;
1236
	spin_unlock_bh(&icmp6_dst_lock);
L
Linus Torvalds 已提交
1237

1238
	fib6_force_start_gc(net);
L
Linus Torvalds 已提交
1239

1240 1241
	dst = xfrm_lookup(net, &rt->dst, flowi6_to_flowi(fl6), NULL, 0);

L
Linus Torvalds 已提交
1242
out:
1243
	return dst;
L
Linus Torvalds 已提交
1244 1245
}

1246
int icmp6_dst_gc(void)
L
Linus Torvalds 已提交
1247
{
1248
	struct dst_entry *dst, **pprev;
1249
	int more = 0;
L
Linus Torvalds 已提交
1250

1251 1252
	spin_lock_bh(&icmp6_dst_lock);
	pprev = &icmp6_dst_gc_list;
1253

L
Linus Torvalds 已提交
1254 1255 1256 1257 1258 1259
	while ((dst = *pprev) != NULL) {
		if (!atomic_read(&dst->__refcnt)) {
			*pprev = dst->next;
			dst_free(dst);
		} else {
			pprev = &dst->next;
1260
			++more;
L
Linus Torvalds 已提交
1261 1262 1263
		}
	}

1264
	spin_unlock_bh(&icmp6_dst_lock);
1265

1266
	return more;
L
Linus Torvalds 已提交
1267 1268
}

1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287
static void icmp6_clean_all(int (*func)(struct rt6_info *rt, void *arg),
			    void *arg)
{
	struct dst_entry *dst, **pprev;

	spin_lock_bh(&icmp6_dst_lock);
	pprev = &icmp6_dst_gc_list;
	while ((dst = *pprev) != NULL) {
		struct rt6_info *rt = (struct rt6_info *) dst;
		if (func(rt, arg)) {
			*pprev = dst->next;
			dst_free(dst);
		} else {
			pprev = &dst->next;
		}
	}
	spin_unlock_bh(&icmp6_dst_lock);
}

1288
static int ip6_dst_gc(struct dst_ops *ops)
L
Linus Torvalds 已提交
1289 1290
{
	unsigned long now = jiffies;
1291
	struct net *net = container_of(ops, struct net, ipv6.ip6_dst_ops);
1292 1293 1294 1295 1296
	int rt_min_interval = net->ipv6.sysctl.ip6_rt_gc_min_interval;
	int rt_max_size = net->ipv6.sysctl.ip6_rt_max_size;
	int rt_elasticity = net->ipv6.sysctl.ip6_rt_gc_elasticity;
	int rt_gc_timeout = net->ipv6.sysctl.ip6_rt_gc_timeout;
	unsigned long rt_last_gc = net->ipv6.ip6_rt_last_gc;
1297
	int entries;
1298

1299
	entries = dst_entries_get_fast(ops);
1300
	if (time_after(rt_last_gc + rt_min_interval, now) &&
1301
	    entries <= rt_max_size)
L
Linus Torvalds 已提交
1302 1303
		goto out;

1304 1305 1306
	net->ipv6.ip6_rt_gc_expire++;
	fib6_run_gc(net->ipv6.ip6_rt_gc_expire, net);
	net->ipv6.ip6_rt_last_gc = now;
1307 1308
	entries = dst_entries_get_slow(ops);
	if (entries < ops->gc_thresh)
1309
		net->ipv6.ip6_rt_gc_expire = rt_gc_timeout>>1;
L
Linus Torvalds 已提交
1310
out:
1311
	net->ipv6.ip6_rt_gc_expire -= net->ipv6.ip6_rt_gc_expire>>rt_elasticity;
1312
	return entries > rt_max_size;
L
Linus Torvalds 已提交
1313 1314 1315 1316 1317 1318 1319 1320
}

/* Clean host part of a prefix. Not necessary in radix tree,
   but results in cleaner routing tables.

   Remove it only when all the things will work!
 */

1321
int ip6_dst_hoplimit(struct dst_entry *dst)
L
Linus Torvalds 已提交
1322
{
1323
	int hoplimit = dst_metric_raw(dst, RTAX_HOPLIMIT);
1324
	if (hoplimit == 0) {
1325
		struct net_device *dev = dst->dev;
1326 1327 1328 1329 1330
		struct inet6_dev *idev;

		rcu_read_lock();
		idev = __in6_dev_get(dev);
		if (idev)
1331
			hoplimit = idev->cnf.hop_limit;
1332
		else
1333
			hoplimit = dev_net(dev)->ipv6.devconf_all->hop_limit;
1334
		rcu_read_unlock();
L
Linus Torvalds 已提交
1335 1336 1337
	}
	return hoplimit;
}
1338
EXPORT_SYMBOL(ip6_dst_hoplimit);
L
Linus Torvalds 已提交
1339 1340 1341 1342 1343

/*
 *
 */

1344
int ip6_route_add(struct fib6_config *cfg)
L
Linus Torvalds 已提交
1345 1346
{
	int err;
1347
	struct net *net = cfg->fc_nlinfo.nl_net;
L
Linus Torvalds 已提交
1348 1349 1350
	struct rt6_info *rt = NULL;
	struct net_device *dev = NULL;
	struct inet6_dev *idev = NULL;
T
Thomas Graf 已提交
1351
	struct fib6_table *table;
L
Linus Torvalds 已提交
1352 1353
	int addr_type;

1354
	if (cfg->fc_dst_len > 128 || cfg->fc_src_len > 128)
L
Linus Torvalds 已提交
1355 1356
		return -EINVAL;
#ifndef CONFIG_IPV6_SUBTREES
1357
	if (cfg->fc_src_len)
L
Linus Torvalds 已提交
1358 1359
		return -EINVAL;
#endif
1360
	if (cfg->fc_ifindex) {
L
Linus Torvalds 已提交
1361
		err = -ENODEV;
1362
		dev = dev_get_by_index(net, cfg->fc_ifindex);
L
Linus Torvalds 已提交
1363 1364 1365 1366 1367 1368 1369
		if (!dev)
			goto out;
		idev = in6_dev_get(dev);
		if (!idev)
			goto out;
	}

1370 1371
	if (cfg->fc_metric == 0)
		cfg->fc_metric = IP6_RT_PRIO_USER;
L
Linus Torvalds 已提交
1372

1373
	err = -ENOBUFS;
1374 1375
	if (cfg->fc_nlinfo.nlh &&
	    !(cfg->fc_nlinfo.nlh->nlmsg_flags & NLM_F_CREATE)) {
1376
		table = fib6_get_table(net, cfg->fc_table);
1377
		if (!table) {
1378
			pr_warn("NLM_F_CREATE should be specified when creating new route\n");
1379 1380 1381 1382 1383
			table = fib6_new_table(net, cfg->fc_table);
		}
	} else {
		table = fib6_new_table(net, cfg->fc_table);
	}
1384 1385

	if (!table)
T
Thomas Graf 已提交
1386 1387
		goto out;

1388
	rt = ip6_dst_alloc(net, NULL, DST_NOCOUNT, table);
L
Linus Torvalds 已提交
1389

1390
	if (!rt) {
L
Linus Torvalds 已提交
1391 1392 1393 1394
		err = -ENOMEM;
		goto out;
	}

1395
	rt->dst.obsolete = -1;
1396 1397 1398 1399 1400 1401

	if (cfg->fc_flags & RTF_EXPIRES)
		rt6_set_expires(rt, jiffies +
				clock_t_to_jiffies(cfg->fc_expires));
	else
		rt6_clean_expires(rt);
L
Linus Torvalds 已提交
1402

1403 1404 1405 1406 1407
	if (cfg->fc_protocol == RTPROT_UNSPEC)
		cfg->fc_protocol = RTPROT_BOOT;
	rt->rt6i_protocol = cfg->fc_protocol;

	addr_type = ipv6_addr_type(&cfg->fc_dst);
L
Linus Torvalds 已提交
1408 1409

	if (addr_type & IPV6_ADDR_MULTICAST)
1410
		rt->dst.input = ip6_mc_input;
1411 1412
	else if (cfg->fc_flags & RTF_LOCAL)
		rt->dst.input = ip6_input;
L
Linus Torvalds 已提交
1413
	else
1414
		rt->dst.input = ip6_forward;
L
Linus Torvalds 已提交
1415

1416
	rt->dst.output = ip6_output;
L
Linus Torvalds 已提交
1417

1418 1419
	ipv6_addr_prefix(&rt->rt6i_dst.addr, &cfg->fc_dst, cfg->fc_dst_len);
	rt->rt6i_dst.plen = cfg->fc_dst_len;
L
Linus Torvalds 已提交
1420
	if (rt->rt6i_dst.plen == 128)
1421
	       rt->dst.flags |= DST_HOST;
L
Linus Torvalds 已提交
1422

1423 1424 1425 1426 1427 1428 1429 1430
	if (!(rt->dst.flags & DST_HOST) && cfg->fc_mx) {
		u32 *metrics = kzalloc(sizeof(u32) * RTAX_MAX, GFP_KERNEL);
		if (!metrics) {
			err = -ENOMEM;
			goto out;
		}
		dst_init_metrics(&rt->dst, metrics, 0);
	}
L
Linus Torvalds 已提交
1431
#ifdef CONFIG_IPV6_SUBTREES
1432 1433
	ipv6_addr_prefix(&rt->rt6i_src.addr, &cfg->fc_src, cfg->fc_src_len);
	rt->rt6i_src.plen = cfg->fc_src_len;
L
Linus Torvalds 已提交
1434 1435
#endif

1436
	rt->rt6i_metric = cfg->fc_metric;
L
Linus Torvalds 已提交
1437 1438 1439 1440

	/* We cannot add true routes via loopback here,
	   they would result in kernel looping; promote them to reject routes
	 */
1441
	if ((cfg->fc_flags & RTF_REJECT) ||
1442 1443 1444
	    (dev && (dev->flags & IFF_LOOPBACK) &&
	     !(addr_type & IPV6_ADDR_LOOPBACK) &&
	     !(cfg->fc_flags & RTF_LOCAL))) {
L
Linus Torvalds 已提交
1445
		/* hold loopback dev/idev if we haven't done so. */
1446
		if (dev != net->loopback_dev) {
L
Linus Torvalds 已提交
1447 1448 1449 1450
			if (dev) {
				dev_put(dev);
				in6_dev_put(idev);
			}
1451
			dev = net->loopback_dev;
L
Linus Torvalds 已提交
1452 1453 1454 1455 1456 1457 1458
			dev_hold(dev);
			idev = in6_dev_get(dev);
			if (!idev) {
				err = -ENODEV;
				goto out;
			}
		}
1459 1460
		rt->dst.output = ip6_pkt_discard_out;
		rt->dst.input = ip6_pkt_discard;
L
Linus Torvalds 已提交
1461
		rt->rt6i_flags = RTF_REJECT|RTF_NONEXTHOP;
1462 1463 1464 1465 1466 1467 1468
		switch (cfg->fc_type) {
		case RTN_BLACKHOLE:
			rt->dst.error = -EINVAL;
			break;
		case RTN_PROHIBIT:
			rt->dst.error = -EACCES;
			break;
1469 1470 1471
		case RTN_THROW:
			rt->dst.error = -EAGAIN;
			break;
1472 1473 1474 1475
		default:
			rt->dst.error = -ENETUNREACH;
			break;
		}
L
Linus Torvalds 已提交
1476 1477 1478
		goto install_route;
	}

1479
	if (cfg->fc_flags & RTF_GATEWAY) {
1480
		const struct in6_addr *gw_addr;
L
Linus Torvalds 已提交
1481 1482
		int gwa_type;

1483
		gw_addr = &cfg->fc_gateway;
A
Alexey Dobriyan 已提交
1484
		rt->rt6i_gateway = *gw_addr;
L
Linus Torvalds 已提交
1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497
		gwa_type = ipv6_addr_type(gw_addr);

		if (gwa_type != (IPV6_ADDR_LINKLOCAL|IPV6_ADDR_UNICAST)) {
			struct rt6_info *grt;

			/* IPv6 strictly inhibits using not link-local
			   addresses as nexthop address.
			   Otherwise, router will not able to send redirects.
			   It is very good, but in some (rare!) circumstances
			   (SIT, PtP, NBMA NOARP links) it is handy to allow
			   some exceptions. --ANK
			 */
			err = -EINVAL;
1498
			if (!(gwa_type & IPV6_ADDR_UNICAST))
L
Linus Torvalds 已提交
1499 1500
				goto out;

1501
			grt = rt6_lookup(net, gw_addr, NULL, cfg->fc_ifindex, 1);
L
Linus Torvalds 已提交
1502 1503

			err = -EHOSTUNREACH;
1504
			if (!grt)
L
Linus Torvalds 已提交
1505 1506
				goto out;
			if (dev) {
1507
				if (dev != grt->dst.dev) {
1508
					dst_release(&grt->dst);
L
Linus Torvalds 已提交
1509 1510 1511
					goto out;
				}
			} else {
1512
				dev = grt->dst.dev;
L
Linus Torvalds 已提交
1513 1514 1515 1516
				idev = grt->rt6i_idev;
				dev_hold(dev);
				in6_dev_hold(grt->rt6i_idev);
			}
1517
			if (!(grt->rt6i_flags & RTF_GATEWAY))
L
Linus Torvalds 已提交
1518
				err = 0;
1519
			dst_release(&grt->dst);
L
Linus Torvalds 已提交
1520 1521 1522 1523 1524

			if (err)
				goto out;
		}
		err = -EINVAL;
1525
		if (!dev || (dev->flags & IFF_LOOPBACK))
L
Linus Torvalds 已提交
1526 1527 1528 1529
			goto out;
	}

	err = -ENODEV;
1530
	if (!dev)
L
Linus Torvalds 已提交
1531 1532
		goto out;

1533 1534 1535 1536 1537
	if (!ipv6_addr_any(&cfg->fc_prefsrc)) {
		if (!ipv6_chk_addr(net, &cfg->fc_prefsrc, dev, 0)) {
			err = -EINVAL;
			goto out;
		}
A
Alexey Dobriyan 已提交
1538
		rt->rt6i_prefsrc.addr = cfg->fc_prefsrc;
1539 1540 1541 1542
		rt->rt6i_prefsrc.plen = 128;
	} else
		rt->rt6i_prefsrc.plen = 0;

1543
	if (cfg->fc_flags & (RTF_GATEWAY | RTF_NONEXTHOP)) {
1544
		err = rt6_bind_neighbour(rt, dev);
1545
		if (err)
L
Linus Torvalds 已提交
1546 1547 1548
			goto out;
	}

1549
	rt->rt6i_flags = cfg->fc_flags;
L
Linus Torvalds 已提交
1550 1551

install_route:
1552 1553 1554 1555 1556
	if (cfg->fc_mx) {
		struct nlattr *nla;
		int remaining;

		nla_for_each_attr(nla, cfg->fc_mx, cfg->fc_mx_len, remaining) {
1557
			int type = nla_type(nla);
1558 1559 1560

			if (type) {
				if (type > RTAX_MAX) {
L
Linus Torvalds 已提交
1561 1562 1563
					err = -EINVAL;
					goto out;
				}
1564

1565
				dst_metric_set(&rt->dst, type, nla_get_u32(nla));
L
Linus Torvalds 已提交
1566 1567 1568 1569
			}
		}
	}

1570
	rt->dst.dev = dev;
L
Linus Torvalds 已提交
1571
	rt->rt6i_idev = idev;
T
Thomas Graf 已提交
1572
	rt->rt6i_table = table;
1573

1574
	cfg->fc_nlinfo.nl_net = dev_net(dev);
1575

1576
	return __ip6_ins_rt(rt, &cfg->fc_nlinfo);
L
Linus Torvalds 已提交
1577 1578 1579 1580 1581 1582 1583

out:
	if (dev)
		dev_put(dev);
	if (idev)
		in6_dev_put(idev);
	if (rt)
1584
		dst_free(&rt->dst);
L
Linus Torvalds 已提交
1585 1586 1587
	return err;
}

1588
static int __ip6_del_rt(struct rt6_info *rt, struct nl_info *info)
L
Linus Torvalds 已提交
1589 1590
{
	int err;
T
Thomas Graf 已提交
1591
	struct fib6_table *table;
1592
	struct net *net = dev_net(rt->dst.dev);
L
Linus Torvalds 已提交
1593

1594
	if (rt == net->ipv6.ip6_null_entry)
1595 1596
		return -ENOENT;

T
Thomas Graf 已提交
1597 1598
	table = rt->rt6i_table;
	write_lock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1599

1600
	err = fib6_del(rt, info);
1601
	dst_release(&rt->dst);
L
Linus Torvalds 已提交
1602

T
Thomas Graf 已提交
1603
	write_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1604 1605 1606 1607

	return err;
}

1608 1609
int ip6_del_rt(struct rt6_info *rt)
{
1610
	struct nl_info info = {
1611
		.nl_net = dev_net(rt->dst.dev),
1612
	};
1613
	return __ip6_del_rt(rt, &info);
1614 1615
}

1616
static int ip6_route_del(struct fib6_config *cfg)
L
Linus Torvalds 已提交
1617
{
T
Thomas Graf 已提交
1618
	struct fib6_table *table;
L
Linus Torvalds 已提交
1619 1620 1621 1622
	struct fib6_node *fn;
	struct rt6_info *rt;
	int err = -ESRCH;

1623
	table = fib6_get_table(cfg->fc_nlinfo.nl_net, cfg->fc_table);
1624
	if (!table)
T
Thomas Graf 已提交
1625 1626 1627
		return err;

	read_lock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1628

T
Thomas Graf 已提交
1629
	fn = fib6_locate(&table->tb6_root,
1630 1631
			 &cfg->fc_dst, cfg->fc_dst_len,
			 &cfg->fc_src, cfg->fc_src_len);
1632

L
Linus Torvalds 已提交
1633
	if (fn) {
1634
		for (rt = fn->leaf; rt; rt = rt->dst.rt6_next) {
1635
			if (cfg->fc_ifindex &&
1636 1637
			    (!rt->dst.dev ||
			     rt->dst.dev->ifindex != cfg->fc_ifindex))
L
Linus Torvalds 已提交
1638
				continue;
1639 1640
			if (cfg->fc_flags & RTF_GATEWAY &&
			    !ipv6_addr_equal(&cfg->fc_gateway, &rt->rt6i_gateway))
L
Linus Torvalds 已提交
1641
				continue;
1642
			if (cfg->fc_metric && cfg->fc_metric != rt->rt6i_metric)
L
Linus Torvalds 已提交
1643
				continue;
1644
			dst_hold(&rt->dst);
T
Thomas Graf 已提交
1645
			read_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1646

1647
			return __ip6_del_rt(rt, &cfg->fc_nlinfo);
L
Linus Torvalds 已提交
1648 1649
		}
	}
T
Thomas Graf 已提交
1650
	read_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1651 1652 1653 1654

	return err;
}

1655
static void rt6_do_redirect(struct dst_entry *dst, struct sock *sk, struct sk_buff *skb)
1656
{
1657
	struct net *net = dev_net(skb->dev);
1658
	struct netevent_redirect netevent;
1659 1660 1661
	struct rt6_info *rt, *nrt = NULL;
	const struct in6_addr *target;
	struct ndisc_options ndopts;
1662 1663
	const struct in6_addr *dest;
	struct neighbour *old_neigh;
1664 1665 1666
	struct inet6_dev *in6_dev;
	struct neighbour *neigh;
	struct icmp6hdr *icmph;
1667 1668
	int optlen, on_link;
	u8 *lladdr;
1669 1670 1671 1672 1673

	optlen = skb->tail - skb->transport_header;
	optlen -= sizeof(struct icmp6hdr) + 2 * sizeof(struct in6_addr);

	if (optlen < 0) {
1674
		net_dbg_ratelimited("rt6_do_redirect: packet too short\n");
1675 1676 1677 1678 1679 1680 1681 1682
		return;
	}

	icmph = icmp6_hdr(skb);
	target = (const struct in6_addr *) (icmph + 1);
	dest = target + 1;

	if (ipv6_addr_is_multicast(dest)) {
1683
		net_dbg_ratelimited("rt6_do_redirect: destination address is multicast\n");
1684 1685 1686
		return;
	}

1687
	on_link = 0;
1688 1689 1690 1691
	if (ipv6_addr_equal(dest, target)) {
		on_link = 1;
	} else if (ipv6_addr_type(target) !=
		   (IPV6_ADDR_UNICAST|IPV6_ADDR_LINKLOCAL)) {
1692
		net_dbg_ratelimited("rt6_do_redirect: target address is not link-local unicast\n");
1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710
		return;
	}

	in6_dev = __in6_dev_get(skb->dev);
	if (!in6_dev)
		return;
	if (in6_dev->cnf.forwarding || !in6_dev->cnf.accept_redirects)
		return;

	/* RFC2461 8.1:
	 *	The IP source address of the Redirect MUST be the same as the current
	 *	first-hop router for the specified ICMP Destination Address.
	 */

	if (!ndisc_parse_options((u8*)(dest + 1), optlen, &ndopts)) {
		net_dbg_ratelimited("rt6_redirect: invalid ND options\n");
		return;
	}
1711 1712

	lladdr = NULL;
1713 1714 1715 1716 1717 1718 1719 1720 1721
	if (ndopts.nd_opts_tgt_lladdr) {
		lladdr = ndisc_opt_addr_data(ndopts.nd_opts_tgt_lladdr,
					     skb->dev);
		if (!lladdr) {
			net_dbg_ratelimited("rt6_redirect: invalid link-layer address length\n");
			return;
		}
	}

1722 1723 1724
	rt = (struct rt6_info *) dst;
	if (rt == net->ipv6.ip6_null_entry) {
		net_dbg_ratelimited("rt6_redirect: source isn't a valid nexthop for redirect target\n");
1725
		return;
1726
	}
1727

1728 1729 1730 1731 1732
	/* Redirect received -> path was valid.
	 * Look, redirects are sent only in response to data packets,
	 * so that this nexthop apparently is reachable. --ANK
	 */
	dst_confirm(&rt->dst);
1733

1734 1735 1736
	neigh = __neigh_lookup(&nd_tbl, target, skb->dev, 1);
	if (!neigh)
		return;
1737

1738 1739 1740
	/* Duplicate redirect: silently ignore. */
	old_neigh = rt->n;
	if (neigh == old_neigh)
1741
		goto out;
L
Linus Torvalds 已提交
1742 1743 1744 1745 1746

	/*
	 *	We have finally decided to accept it.
	 */

1747
	neigh_update(neigh, lladdr, NUD_STALE,
L
Linus Torvalds 已提交
1748 1749 1750 1751 1752 1753
		     NEIGH_UPDATE_F_WEAK_OVERRIDE|
		     NEIGH_UPDATE_F_OVERRIDE|
		     (on_link ? 0 : (NEIGH_UPDATE_F_OVERRIDE_ISROUTER|
				     NEIGH_UPDATE_F_ISROUTER))
		     );

E
Eric Dumazet 已提交
1754
	nrt = ip6_rt_copy(rt, dest);
1755
	if (!nrt)
L
Linus Torvalds 已提交
1756 1757 1758 1759 1760 1761
		goto out;

	nrt->rt6i_flags = RTF_GATEWAY|RTF_UP|RTF_DYNAMIC|RTF_CACHE;
	if (on_link)
		nrt->rt6i_flags &= ~RTF_GATEWAY;

A
Alexey Dobriyan 已提交
1762
	nrt->rt6i_gateway = *(struct in6_addr *)neigh->primary_key;
1763
	nrt->n = neigh_clone(neigh);
L
Linus Torvalds 已提交
1764

1765
	if (ip6_ins_rt(nrt))
L
Linus Torvalds 已提交
1766 1767
		goto out;

1768
	netevent.old = &rt->dst;
1769
	netevent.old_neigh = old_neigh;
1770
	netevent.new = &nrt->dst;
1771 1772
	netevent.new_neigh = neigh;
	netevent.daddr = dest;
1773 1774
	call_netevent_notifiers(NETEVENT_REDIRECT, &netevent);

1775
	if (rt->rt6i_flags & RTF_CACHE) {
1776
		rt = (struct rt6_info *) dst_clone(&rt->dst);
1777
		ip6_del_rt(rt);
L
Linus Torvalds 已提交
1778 1779 1780
	}

out:
1781
	neigh_release(neigh);
1782 1783
}

L
Linus Torvalds 已提交
1784 1785 1786 1787
/*
 *	Misc support functions
 */

1788
static struct rt6_info *ip6_rt_copy(struct rt6_info *ort,
E
Eric Dumazet 已提交
1789
				    const struct in6_addr *dest)
L
Linus Torvalds 已提交
1790
{
1791
	struct net *net = dev_net(ort->dst.dev);
1792 1793
	struct rt6_info *rt = ip6_dst_alloc(net, ort->dst.dev, 0,
					    ort->rt6i_table);
L
Linus Torvalds 已提交
1794 1795

	if (rt) {
1796 1797
		rt->dst.input = ort->dst.input;
		rt->dst.output = ort->dst.output;
1798
		rt->dst.flags |= DST_HOST;
1799

A
Alexey Dobriyan 已提交
1800
		rt->rt6i_dst.addr = *dest;
1801
		rt->rt6i_dst.plen = 128;
1802
		dst_copy_metrics(&rt->dst, &ort->dst);
1803
		rt->dst.error = ort->dst.error;
L
Linus Torvalds 已提交
1804 1805 1806
		rt->rt6i_idev = ort->rt6i_idev;
		if (rt->rt6i_idev)
			in6_dev_hold(rt->rt6i_idev);
1807
		rt->dst.lastuse = jiffies;
L
Linus Torvalds 已提交
1808

A
Alexey Dobriyan 已提交
1809
		rt->rt6i_gateway = ort->rt6i_gateway;
1810 1811 1812 1813 1814 1815
		rt->rt6i_flags = ort->rt6i_flags;
		if ((ort->rt6i_flags & (RTF_DEFAULT | RTF_ADDRCONF)) ==
		    (RTF_DEFAULT | RTF_ADDRCONF))
			rt6_set_from(rt, ort);
		else
			rt6_clean_expires(rt);
L
Linus Torvalds 已提交
1816 1817 1818 1819 1820
		rt->rt6i_metric = 0;

#ifdef CONFIG_IPV6_SUBTREES
		memcpy(&rt->rt6i_src, &ort->rt6i_src, sizeof(struct rt6key));
#endif
1821
		memcpy(&rt->rt6i_prefsrc, &ort->rt6i_prefsrc, sizeof(struct rt6key));
T
Thomas Graf 已提交
1822
		rt->rt6i_table = ort->rt6i_table;
L
Linus Torvalds 已提交
1823 1824 1825 1826
	}
	return rt;
}

1827
#ifdef CONFIG_IPV6_ROUTE_INFO
1828
static struct rt6_info *rt6_get_route_info(struct net *net,
1829 1830
					   const struct in6_addr *prefix, int prefixlen,
					   const struct in6_addr *gwaddr, int ifindex)
1831 1832 1833
{
	struct fib6_node *fn;
	struct rt6_info *rt = NULL;
T
Thomas Graf 已提交
1834 1835
	struct fib6_table *table;

1836
	table = fib6_get_table(net, RT6_TABLE_INFO);
1837
	if (!table)
T
Thomas Graf 已提交
1838
		return NULL;
1839

T
Thomas Graf 已提交
1840 1841
	write_lock_bh(&table->tb6_lock);
	fn = fib6_locate(&table->tb6_root, prefix ,prefixlen, NULL, 0);
1842 1843 1844
	if (!fn)
		goto out;

1845
	for (rt = fn->leaf; rt; rt = rt->dst.rt6_next) {
1846
		if (rt->dst.dev->ifindex != ifindex)
1847 1848 1849 1850 1851
			continue;
		if ((rt->rt6i_flags & (RTF_ROUTEINFO|RTF_GATEWAY)) != (RTF_ROUTEINFO|RTF_GATEWAY))
			continue;
		if (!ipv6_addr_equal(&rt->rt6i_gateway, gwaddr))
			continue;
1852
		dst_hold(&rt->dst);
1853 1854 1855
		break;
	}
out:
T
Thomas Graf 已提交
1856
	write_unlock_bh(&table->tb6_lock);
1857 1858 1859
	return rt;
}

1860
static struct rt6_info *rt6_add_route_info(struct net *net,
1861 1862
					   const struct in6_addr *prefix, int prefixlen,
					   const struct in6_addr *gwaddr, int ifindex,
1863
					   unsigned int pref)
1864
{
1865 1866
	struct fib6_config cfg = {
		.fc_table	= RT6_TABLE_INFO,
1867
		.fc_metric	= IP6_RT_PRIO_USER,
1868 1869 1870 1871
		.fc_ifindex	= ifindex,
		.fc_dst_len	= prefixlen,
		.fc_flags	= RTF_GATEWAY | RTF_ADDRCONF | RTF_ROUTEINFO |
				  RTF_UP | RTF_PREF(pref),
1872
		.fc_nlinfo.portid = 0,
1873 1874
		.fc_nlinfo.nlh = NULL,
		.fc_nlinfo.nl_net = net,
1875 1876
	};

A
Alexey Dobriyan 已提交
1877 1878
	cfg.fc_dst = *prefix;
	cfg.fc_gateway = *gwaddr;
1879

1880 1881
	/* We should treat it as a default route if prefix length is 0. */
	if (!prefixlen)
1882
		cfg.fc_flags |= RTF_DEFAULT;
1883

1884
	ip6_route_add(&cfg);
1885

1886
	return rt6_get_route_info(net, prefix, prefixlen, gwaddr, ifindex);
1887 1888 1889
}
#endif

1890
struct rt6_info *rt6_get_dflt_router(const struct in6_addr *addr, struct net_device *dev)
1891
{
L
Linus Torvalds 已提交
1892
	struct rt6_info *rt;
T
Thomas Graf 已提交
1893
	struct fib6_table *table;
L
Linus Torvalds 已提交
1894

1895
	table = fib6_get_table(dev_net(dev), RT6_TABLE_DFLT);
1896
	if (!table)
T
Thomas Graf 已提交
1897
		return NULL;
L
Linus Torvalds 已提交
1898

T
Thomas Graf 已提交
1899
	write_lock_bh(&table->tb6_lock);
1900
	for (rt = table->tb6_root.leaf; rt; rt=rt->dst.rt6_next) {
1901
		if (dev == rt->dst.dev &&
1902
		    ((rt->rt6i_flags & (RTF_ADDRCONF | RTF_DEFAULT)) == (RTF_ADDRCONF | RTF_DEFAULT)) &&
L
Linus Torvalds 已提交
1903 1904 1905 1906
		    ipv6_addr_equal(&rt->rt6i_gateway, addr))
			break;
	}
	if (rt)
1907
		dst_hold(&rt->dst);
T
Thomas Graf 已提交
1908
	write_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1909 1910 1911
	return rt;
}

1912
struct rt6_info *rt6_add_dflt_router(const struct in6_addr *gwaddr,
1913 1914
				     struct net_device *dev,
				     unsigned int pref)
L
Linus Torvalds 已提交
1915
{
1916 1917
	struct fib6_config cfg = {
		.fc_table	= RT6_TABLE_DFLT,
1918
		.fc_metric	= IP6_RT_PRIO_USER,
1919 1920 1921
		.fc_ifindex	= dev->ifindex,
		.fc_flags	= RTF_GATEWAY | RTF_ADDRCONF | RTF_DEFAULT |
				  RTF_UP | RTF_EXPIRES | RTF_PREF(pref),
1922
		.fc_nlinfo.portid = 0,
1923
		.fc_nlinfo.nlh = NULL,
1924
		.fc_nlinfo.nl_net = dev_net(dev),
1925
	};
L
Linus Torvalds 已提交
1926

A
Alexey Dobriyan 已提交
1927
	cfg.fc_gateway = *gwaddr;
L
Linus Torvalds 已提交
1928

1929
	ip6_route_add(&cfg);
L
Linus Torvalds 已提交
1930 1931 1932 1933

	return rt6_get_dflt_router(gwaddr, dev);
}

1934
void rt6_purge_dflt_routers(struct net *net)
L
Linus Torvalds 已提交
1935 1936
{
	struct rt6_info *rt;
T
Thomas Graf 已提交
1937 1938 1939
	struct fib6_table *table;

	/* NOTE: Keep consistent with rt6_get_dflt_router */
1940
	table = fib6_get_table(net, RT6_TABLE_DFLT);
1941
	if (!table)
T
Thomas Graf 已提交
1942
		return;
L
Linus Torvalds 已提交
1943 1944

restart:
T
Thomas Graf 已提交
1945
	read_lock_bh(&table->tb6_lock);
1946
	for (rt = table->tb6_root.leaf; rt; rt = rt->dst.rt6_next) {
L
Linus Torvalds 已提交
1947
		if (rt->rt6i_flags & (RTF_DEFAULT | RTF_ADDRCONF)) {
1948
			dst_hold(&rt->dst);
T
Thomas Graf 已提交
1949
			read_unlock_bh(&table->tb6_lock);
1950
			ip6_del_rt(rt);
L
Linus Torvalds 已提交
1951 1952 1953
			goto restart;
		}
	}
T
Thomas Graf 已提交
1954
	read_unlock_bh(&table->tb6_lock);
L
Linus Torvalds 已提交
1955 1956
}

1957 1958
static void rtmsg_to_fib6_config(struct net *net,
				 struct in6_rtmsg *rtmsg,
1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970
				 struct fib6_config *cfg)
{
	memset(cfg, 0, sizeof(*cfg));

	cfg->fc_table = RT6_TABLE_MAIN;
	cfg->fc_ifindex = rtmsg->rtmsg_ifindex;
	cfg->fc_metric = rtmsg->rtmsg_metric;
	cfg->fc_expires = rtmsg->rtmsg_info;
	cfg->fc_dst_len = rtmsg->rtmsg_dst_len;
	cfg->fc_src_len = rtmsg->rtmsg_src_len;
	cfg->fc_flags = rtmsg->rtmsg_flags;

1971
	cfg->fc_nlinfo.nl_net = net;
1972

A
Alexey Dobriyan 已提交
1973 1974 1975
	cfg->fc_dst = rtmsg->rtmsg_dst;
	cfg->fc_src = rtmsg->rtmsg_src;
	cfg->fc_gateway = rtmsg->rtmsg_gateway;
1976 1977
}

1978
int ipv6_route_ioctl(struct net *net, unsigned int cmd, void __user *arg)
L
Linus Torvalds 已提交
1979
{
1980
	struct fib6_config cfg;
L
Linus Torvalds 已提交
1981 1982 1983 1984 1985 1986 1987 1988 1989 1990 1991 1992
	struct in6_rtmsg rtmsg;
	int err;

	switch(cmd) {
	case SIOCADDRT:		/* Add a route */
	case SIOCDELRT:		/* Delete a route */
		if (!capable(CAP_NET_ADMIN))
			return -EPERM;
		err = copy_from_user(&rtmsg, arg,
				     sizeof(struct in6_rtmsg));
		if (err)
			return -EFAULT;
1993

1994
		rtmsg_to_fib6_config(net, &rtmsg, &cfg);
1995

L
Linus Torvalds 已提交
1996 1997 1998
		rtnl_lock();
		switch (cmd) {
		case SIOCADDRT:
1999
			err = ip6_route_add(&cfg);
L
Linus Torvalds 已提交
2000 2001
			break;
		case SIOCDELRT:
2002
			err = ip6_route_del(&cfg);
L
Linus Torvalds 已提交
2003 2004 2005 2006 2007 2008 2009
			break;
		default:
			err = -EINVAL;
		}
		rtnl_unlock();

		return err;
2010
	}
L
Linus Torvalds 已提交
2011 2012 2013 2014 2015 2016 2017 2018

	return -EINVAL;
}

/*
 *	Drop the packet on the floor
 */

2019
static int ip6_pkt_drop(struct sk_buff *skb, u8 code, int ipstats_mib_noroutes)
L
Linus Torvalds 已提交
2020
{
2021
	int type;
E
Eric Dumazet 已提交
2022
	struct dst_entry *dst = skb_dst(skb);
2023 2024
	switch (ipstats_mib_noroutes) {
	case IPSTATS_MIB_INNOROUTES:
2025
		type = ipv6_addr_type(&ipv6_hdr(skb)->daddr);
U
Ulrich Weber 已提交
2026
		if (type == IPV6_ADDR_ANY) {
2027 2028
			IP6_INC_STATS(dev_net(dst->dev), ip6_dst_idev(dst),
				      IPSTATS_MIB_INADDRERRORS);
2029 2030 2031 2032
			break;
		}
		/* FALLTHROUGH */
	case IPSTATS_MIB_OUTNOROUTES:
2033 2034
		IP6_INC_STATS(dev_net(dst->dev), ip6_dst_idev(dst),
			      ipstats_mib_noroutes);
2035 2036
		break;
	}
2037
	icmpv6_send(skb, ICMPV6_DEST_UNREACH, code, 0);
L
Linus Torvalds 已提交
2038 2039 2040 2041
	kfree_skb(skb);
	return 0;
}

2042 2043
static int ip6_pkt_discard(struct sk_buff *skb)
{
2044
	return ip6_pkt_drop(skb, ICMPV6_NOROUTE, IPSTATS_MIB_INNOROUTES);
2045 2046
}

2047
static int ip6_pkt_discard_out(struct sk_buff *skb)
L
Linus Torvalds 已提交
2048
{
E
Eric Dumazet 已提交
2049
	skb->dev = skb_dst(skb)->dev;
2050
	return ip6_pkt_drop(skb, ICMPV6_NOROUTE, IPSTATS_MIB_OUTNOROUTES);
L
Linus Torvalds 已提交
2051 2052
}

2053 2054
#ifdef CONFIG_IPV6_MULTIPLE_TABLES

2055 2056
static int ip6_pkt_prohibit(struct sk_buff *skb)
{
2057
	return ip6_pkt_drop(skb, ICMPV6_ADM_PROHIBITED, IPSTATS_MIB_INNOROUTES);
2058 2059 2060 2061
}

static int ip6_pkt_prohibit_out(struct sk_buff *skb)
{
E
Eric Dumazet 已提交
2062
	skb->dev = skb_dst(skb)->dev;
2063
	return ip6_pkt_drop(skb, ICMPV6_ADM_PROHIBITED, IPSTATS_MIB_OUTNOROUTES);
2064 2065
}

2066 2067
#endif

L
Linus Torvalds 已提交
2068 2069 2070 2071 2072 2073
/*
 *	Allocate a dst for local (unicast / anycast) address.
 */

struct rt6_info *addrconf_dst_alloc(struct inet6_dev *idev,
				    const struct in6_addr *addr,
2074
				    bool anycast)
L
Linus Torvalds 已提交
2075
{
2076
	struct net *net = dev_net(idev->dev);
2077
	struct rt6_info *rt = ip6_dst_alloc(net, net->loopback_dev, 0, NULL);
2078
	int err;
L
Linus Torvalds 已提交
2079

2080
	if (!rt) {
2081
		net_warn_ratelimited("Maximum number of routes reached, consider increasing route/max_size\n");
L
Linus Torvalds 已提交
2082
		return ERR_PTR(-ENOMEM);
2083
	}
L
Linus Torvalds 已提交
2084 2085 2086

	in6_dev_hold(idev);

2087
	rt->dst.flags |= DST_HOST;
2088 2089
	rt->dst.input = ip6_input;
	rt->dst.output = ip6_output;
L
Linus Torvalds 已提交
2090
	rt->rt6i_idev = idev;
2091
	rt->dst.obsolete = -1;
L
Linus Torvalds 已提交
2092 2093

	rt->rt6i_flags = RTF_UP | RTF_NONEXTHOP;
2094 2095 2096
	if (anycast)
		rt->rt6i_flags |= RTF_ANYCAST;
	else
L
Linus Torvalds 已提交
2097
		rt->rt6i_flags |= RTF_LOCAL;
2098
	err = rt6_bind_neighbour(rt, rt->dst.dev);
2099
	if (err) {
2100
		dst_free(&rt->dst);
2101
		return ERR_PTR(err);
L
Linus Torvalds 已提交
2102 2103
	}

A
Alexey Dobriyan 已提交
2104
	rt->rt6i_dst.addr = *addr;
L
Linus Torvalds 已提交
2105
	rt->rt6i_dst.plen = 128;
2106
	rt->rt6i_table = fib6_get_table(net, RT6_TABLE_LOCAL);
L
Linus Torvalds 已提交
2107

2108
	atomic_set(&rt->dst.__refcnt, 1);
L
Linus Torvalds 已提交
2109 2110 2111 2112

	return rt;
}

2113 2114
int ip6_route_get_saddr(struct net *net,
			struct rt6_info *rt,
2115
			const struct in6_addr *daddr,
2116 2117 2118 2119 2120 2121
			unsigned int prefs,
			struct in6_addr *saddr)
{
	struct inet6_dev *idev = ip6_dst_idev((struct dst_entry*)rt);
	int err = 0;
	if (rt->rt6i_prefsrc.plen)
A
Alexey Dobriyan 已提交
2122
		*saddr = rt->rt6i_prefsrc.addr;
2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141
	else
		err = ipv6_dev_get_saddr(net, idev ? idev->dev : NULL,
					 daddr, prefs, saddr);
	return err;
}

/* remove deleted ip from prefsrc entries */
struct arg_dev_net_ip {
	struct net_device *dev;
	struct net *net;
	struct in6_addr *addr;
};

static int fib6_remove_prefsrc(struct rt6_info *rt, void *arg)
{
	struct net_device *dev = ((struct arg_dev_net_ip *)arg)->dev;
	struct net *net = ((struct arg_dev_net_ip *)arg)->net;
	struct in6_addr *addr = ((struct arg_dev_net_ip *)arg)->addr;

2142
	if (((void *)rt->dst.dev == dev || !dev) &&
2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161
	    rt != net->ipv6.ip6_null_entry &&
	    ipv6_addr_equal(addr, &rt->rt6i_prefsrc.addr)) {
		/* remove prefsrc entry */
		rt->rt6i_prefsrc.plen = 0;
	}
	return 0;
}

void rt6_remove_prefsrc(struct inet6_ifaddr *ifp)
{
	struct net *net = dev_net(ifp->idev->dev);
	struct arg_dev_net_ip adni = {
		.dev = ifp->idev->dev,
		.net = net,
		.addr = &ifp->addr,
	};
	fib6_clean_all(net, fib6_remove_prefsrc, 0, &adni);
}

2162 2163 2164 2165 2166
struct arg_dev_net {
	struct net_device *dev;
	struct net *net;
};

L
Linus Torvalds 已提交
2167 2168
static int fib6_ifdown(struct rt6_info *rt, void *arg)
{
S
stephen hemminger 已提交
2169 2170
	const struct arg_dev_net *adn = arg;
	const struct net_device *dev = adn->dev;
2171

2172
	if ((rt->dst.dev == dev || !dev) &&
2173
	    rt != adn->net->ipv6.ip6_null_entry)
L
Linus Torvalds 已提交
2174
		return -1;
2175

L
Linus Torvalds 已提交
2176 2177 2178
	return 0;
}

2179
void rt6_ifdown(struct net *net, struct net_device *dev)
L
Linus Torvalds 已提交
2180
{
2181 2182 2183 2184 2185 2186
	struct arg_dev_net adn = {
		.dev = dev,
		.net = net,
	};

	fib6_clean_all(net, fib6_ifdown, 0, &adn);
2187
	icmp6_clean_all(fib6_ifdown, &adn);
L
Linus Torvalds 已提交
2188 2189
}

2190
struct rt6_mtu_change_arg {
L
Linus Torvalds 已提交
2191
	struct net_device *dev;
2192
	unsigned int mtu;
L
Linus Torvalds 已提交
2193 2194 2195 2196 2197 2198 2199 2200 2201 2202 2203 2204 2205 2206
};

static int rt6_mtu_change_route(struct rt6_info *rt, void *p_arg)
{
	struct rt6_mtu_change_arg *arg = (struct rt6_mtu_change_arg *) p_arg;
	struct inet6_dev *idev;

	/* In IPv6 pmtu discovery is not optional,
	   so that RTAX_MTU lock cannot disable it.
	   We still use this lock to block changes
	   caused by addrconf/ndisc.
	*/

	idev = __in6_dev_get(arg->dev);
2207
	if (!idev)
L
Linus Torvalds 已提交
2208 2209 2210 2211 2212 2213 2214 2215 2216 2217 2218 2219 2220 2221 2222 2223
		return 0;

	/* For administrative MTU increase, there is no way to discover
	   IPv6 PMTU increase, so PMTU increase should be updated here.
	   Since RFC 1981 doesn't include administrative MTU increase
	   update PMTU increase is a MUST. (i.e. jumbo frame)
	 */
	/*
	   If new MTU is less than route PMTU, this new MTU will be the
	   lowest MTU in the path, update the route PMTU to reflect PMTU
	   decreases; if new MTU is greater than route PMTU, and the
	   old MTU is the lowest MTU in the path, update the route PMTU
	   to reflect the increase. In this case if the other nodes' MTU
	   also have the lowest MTU, TOO BIG MESSAGE will be lead to
	   PMTU discouvery.
	 */
2224
	if (rt->dst.dev == arg->dev &&
2225 2226 2227 2228
	    !dst_metric_locked(&rt->dst, RTAX_MTU) &&
	    (dst_mtu(&rt->dst) >= arg->mtu ||
	     (dst_mtu(&rt->dst) < arg->mtu &&
	      dst_mtu(&rt->dst) == idev->cnf.mtu6))) {
2229
		dst_metric_set(&rt->dst, RTAX_MTU, arg->mtu);
2230
	}
L
Linus Torvalds 已提交
2231 2232 2233
	return 0;
}

2234
void rt6_mtu_change(struct net_device *dev, unsigned int mtu)
L
Linus Torvalds 已提交
2235
{
T
Thomas Graf 已提交
2236 2237 2238 2239
	struct rt6_mtu_change_arg arg = {
		.dev = dev,
		.mtu = mtu,
	};
L
Linus Torvalds 已提交
2240

2241
	fib6_clean_all(dev_net(dev), rt6_mtu_change_route, 0, &arg);
L
Linus Torvalds 已提交
2242 2243
}

2244
static const struct nla_policy rtm_ipv6_policy[RTA_MAX+1] = {
2245
	[RTA_GATEWAY]           = { .len = sizeof(struct in6_addr) },
2246
	[RTA_OIF]               = { .type = NLA_U32 },
2247
	[RTA_IIF]		= { .type = NLA_U32 },
2248 2249 2250 2251 2252 2253
	[RTA_PRIORITY]          = { .type = NLA_U32 },
	[RTA_METRICS]           = { .type = NLA_NESTED },
};

static int rtm_to_fib6_config(struct sk_buff *skb, struct nlmsghdr *nlh,
			      struct fib6_config *cfg)
L
Linus Torvalds 已提交
2254
{
2255 2256 2257
	struct rtmsg *rtm;
	struct nlattr *tb[RTA_MAX+1];
	int err;
L
Linus Torvalds 已提交
2258

2259 2260 2261
	err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv6_policy);
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
2262

2263 2264 2265 2266 2267 2268 2269 2270 2271
	err = -EINVAL;
	rtm = nlmsg_data(nlh);
	memset(cfg, 0, sizeof(*cfg));

	cfg->fc_table = rtm->rtm_table;
	cfg->fc_dst_len = rtm->rtm_dst_len;
	cfg->fc_src_len = rtm->rtm_src_len;
	cfg->fc_flags = RTF_UP;
	cfg->fc_protocol = rtm->rtm_protocol;
2272
	cfg->fc_type = rtm->rtm_type;
2273

2274 2275
	if (rtm->rtm_type == RTN_UNREACHABLE ||
	    rtm->rtm_type == RTN_BLACKHOLE ||
2276 2277
	    rtm->rtm_type == RTN_PROHIBIT ||
	    rtm->rtm_type == RTN_THROW)
2278 2279
		cfg->fc_flags |= RTF_REJECT;

2280 2281 2282
	if (rtm->rtm_type == RTN_LOCAL)
		cfg->fc_flags |= RTF_LOCAL;

2283
	cfg->fc_nlinfo.portid = NETLINK_CB(skb).portid;
2284
	cfg->fc_nlinfo.nlh = nlh;
2285
	cfg->fc_nlinfo.nl_net = sock_net(skb->sk);
2286 2287 2288 2289

	if (tb[RTA_GATEWAY]) {
		nla_memcpy(&cfg->fc_gateway, tb[RTA_GATEWAY], 16);
		cfg->fc_flags |= RTF_GATEWAY;
L
Linus Torvalds 已提交
2290
	}
2291 2292 2293 2294 2295 2296 2297 2298

	if (tb[RTA_DST]) {
		int plen = (rtm->rtm_dst_len + 7) >> 3;

		if (nla_len(tb[RTA_DST]) < plen)
			goto errout;

		nla_memcpy(&cfg->fc_dst, tb[RTA_DST], plen);
L
Linus Torvalds 已提交
2299
	}
2300 2301 2302 2303 2304 2305 2306 2307

	if (tb[RTA_SRC]) {
		int plen = (rtm->rtm_src_len + 7) >> 3;

		if (nla_len(tb[RTA_SRC]) < plen)
			goto errout;

		nla_memcpy(&cfg->fc_src, tb[RTA_SRC], plen);
L
Linus Torvalds 已提交
2308
	}
2309

2310 2311 2312
	if (tb[RTA_PREFSRC])
		nla_memcpy(&cfg->fc_prefsrc, tb[RTA_PREFSRC], 16);

2313 2314 2315 2316 2317 2318 2319 2320 2321
	if (tb[RTA_OIF])
		cfg->fc_ifindex = nla_get_u32(tb[RTA_OIF]);

	if (tb[RTA_PRIORITY])
		cfg->fc_metric = nla_get_u32(tb[RTA_PRIORITY]);

	if (tb[RTA_METRICS]) {
		cfg->fc_mx = nla_data(tb[RTA_METRICS]);
		cfg->fc_mx_len = nla_len(tb[RTA_METRICS]);
L
Linus Torvalds 已提交
2322
	}
2323 2324 2325 2326 2327 2328 2329

	if (tb[RTA_TABLE])
		cfg->fc_table = nla_get_u32(tb[RTA_TABLE]);

	err = 0;
errout:
	return err;
L
Linus Torvalds 已提交
2330 2331
}

2332
static int inet6_rtm_delroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
L
Linus Torvalds 已提交
2333
{
2334 2335
	struct fib6_config cfg;
	int err;
L
Linus Torvalds 已提交
2336

2337 2338 2339 2340 2341
	err = rtm_to_fib6_config(skb, nlh, &cfg);
	if (err < 0)
		return err;

	return ip6_route_del(&cfg);
L
Linus Torvalds 已提交
2342 2343
}

2344
static int inet6_rtm_newroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
L
Linus Torvalds 已提交
2345
{
2346 2347
	struct fib6_config cfg;
	int err;
L
Linus Torvalds 已提交
2348

2349 2350 2351 2352 2353
	err = rtm_to_fib6_config(skb, nlh, &cfg);
	if (err < 0)
		return err;

	return ip6_route_add(&cfg);
L
Linus Torvalds 已提交
2354 2355
}

2356 2357 2358 2359 2360 2361 2362 2363 2364 2365 2366
static inline size_t rt6_nlmsg_size(void)
{
	return NLMSG_ALIGN(sizeof(struct rtmsg))
	       + nla_total_size(16) /* RTA_SRC */
	       + nla_total_size(16) /* RTA_DST */
	       + nla_total_size(16) /* RTA_GATEWAY */
	       + nla_total_size(16) /* RTA_PREFSRC */
	       + nla_total_size(4) /* RTA_TABLE */
	       + nla_total_size(4) /* RTA_IIF */
	       + nla_total_size(4) /* RTA_OIF */
	       + nla_total_size(4) /* RTA_PRIORITY */
2367
	       + RTAX_MAX * nla_total_size(4) /* RTA_METRICS */
2368 2369 2370
	       + nla_total_size(sizeof(struct rta_cacheinfo));
}

2371 2372
static int rt6_fill_node(struct net *net,
			 struct sk_buff *skb, struct rt6_info *rt,
2373
			 struct in6_addr *dst, struct in6_addr *src,
2374
			 int iif, int type, u32 portid, u32 seq,
2375
			 int prefix, int nowait, unsigned int flags)
L
Linus Torvalds 已提交
2376 2377
{
	struct rtmsg *rtm;
2378
	struct nlmsghdr *nlh;
2379
	long expires;
2380
	u32 table;
2381
	struct neighbour *n;
L
Linus Torvalds 已提交
2382 2383 2384 2385 2386 2387 2388 2389

	if (prefix) {	/* user wants prefix routes only */
		if (!(rt->rt6i_flags & RTF_PREFIX_RT)) {
			/* success since this is not a prefix route */
			return 1;
		}
	}

2390
	nlh = nlmsg_put(skb, portid, seq, type, sizeof(*rtm), flags);
2391
	if (!nlh)
2392
		return -EMSGSIZE;
2393 2394

	rtm = nlmsg_data(nlh);
L
Linus Torvalds 已提交
2395 2396 2397 2398
	rtm->rtm_family = AF_INET6;
	rtm->rtm_dst_len = rt->rt6i_dst.plen;
	rtm->rtm_src_len = rt->rt6i_src.plen;
	rtm->rtm_tos = 0;
T
Thomas Graf 已提交
2399
	if (rt->rt6i_table)
2400
		table = rt->rt6i_table->tb6_id;
T
Thomas Graf 已提交
2401
	else
2402 2403
		table = RT6_TABLE_UNSPEC;
	rtm->rtm_table = table;
D
David S. Miller 已提交
2404 2405
	if (nla_put_u32(skb, RTA_TABLE, table))
		goto nla_put_failure;
2406 2407 2408 2409 2410 2411 2412 2413
	if (rt->rt6i_flags & RTF_REJECT) {
		switch (rt->dst.error) {
		case -EINVAL:
			rtm->rtm_type = RTN_BLACKHOLE;
			break;
		case -EACCES:
			rtm->rtm_type = RTN_PROHIBIT;
			break;
2414 2415 2416
		case -EAGAIN:
			rtm->rtm_type = RTN_THROW;
			break;
2417 2418 2419 2420 2421
		default:
			rtm->rtm_type = RTN_UNREACHABLE;
			break;
		}
	}
2422
	else if (rt->rt6i_flags & RTF_LOCAL)
2423
		rtm->rtm_type = RTN_LOCAL;
2424
	else if (rt->dst.dev && (rt->dst.dev->flags & IFF_LOOPBACK))
L
Linus Torvalds 已提交
2425 2426 2427 2428 2429 2430
		rtm->rtm_type = RTN_LOCAL;
	else
		rtm->rtm_type = RTN_UNICAST;
	rtm->rtm_flags = 0;
	rtm->rtm_scope = RT_SCOPE_UNIVERSE;
	rtm->rtm_protocol = rt->rt6i_protocol;
2431
	if (rt->rt6i_flags & RTF_DYNAMIC)
L
Linus Torvalds 已提交
2432
		rtm->rtm_protocol = RTPROT_REDIRECT;
2433 2434 2435 2436 2437 2438
	else if (rt->rt6i_flags & RTF_ADDRCONF) {
		if (rt->rt6i_flags & (RTF_DEFAULT | RTF_ROUTEINFO))
			rtm->rtm_protocol = RTPROT_RA;
		else
			rtm->rtm_protocol = RTPROT_KERNEL;
	}
L
Linus Torvalds 已提交
2439

2440
	if (rt->rt6i_flags & RTF_CACHE)
L
Linus Torvalds 已提交
2441 2442 2443
		rtm->rtm_flags |= RTM_F_CLONED;

	if (dst) {
D
David S. Miller 已提交
2444 2445
		if (nla_put(skb, RTA_DST, 16, dst))
			goto nla_put_failure;
2446
		rtm->rtm_dst_len = 128;
L
Linus Torvalds 已提交
2447
	} else if (rtm->rtm_dst_len)
D
David S. Miller 已提交
2448 2449
		if (nla_put(skb, RTA_DST, 16, &rt->rt6i_dst.addr))
			goto nla_put_failure;
L
Linus Torvalds 已提交
2450 2451
#ifdef CONFIG_IPV6_SUBTREES
	if (src) {
D
David S. Miller 已提交
2452 2453
		if (nla_put(skb, RTA_SRC, 16, src))
			goto nla_put_failure;
2454
		rtm->rtm_src_len = 128;
D
David S. Miller 已提交
2455 2456 2457
	} else if (rtm->rtm_src_len &&
		   nla_put(skb, RTA_SRC, 16, &rt->rt6i_src.addr))
		goto nla_put_failure;
L
Linus Torvalds 已提交
2458
#endif
2459 2460 2461
	if (iif) {
#ifdef CONFIG_IPV6_MROUTE
		if (ipv6_addr_is_multicast(&rt->rt6i_dst.addr)) {
2462
			int err = ip6mr_get_route(net, skb, rtm, nowait);
2463 2464 2465 2466 2467 2468 2469 2470 2471 2472 2473 2474
			if (err <= 0) {
				if (!nowait) {
					if (err == 0)
						return 0;
					goto nla_put_failure;
				} else {
					if (err == -EMSGSIZE)
						goto nla_put_failure;
				}
			}
		} else
#endif
D
David S. Miller 已提交
2475 2476
			if (nla_put_u32(skb, RTA_IIF, iif))
				goto nla_put_failure;
2477
	} else if (dst) {
L
Linus Torvalds 已提交
2478
		struct in6_addr saddr_buf;
D
David S. Miller 已提交
2479 2480 2481
		if (ip6_route_get_saddr(net, rt, dst, 0, &saddr_buf) == 0 &&
		    nla_put(skb, RTA_PREFSRC, 16, &saddr_buf))
			goto nla_put_failure;
L
Linus Torvalds 已提交
2482
	}
2483

2484 2485
	if (rt->rt6i_prefsrc.plen) {
		struct in6_addr saddr_buf;
A
Alexey Dobriyan 已提交
2486
		saddr_buf = rt->rt6i_prefsrc.addr;
D
David S. Miller 已提交
2487 2488
		if (nla_put(skb, RTA_PREFSRC, 16, &saddr_buf))
			goto nla_put_failure;
2489 2490
	}

2491
	if (rtnetlink_put_metrics(skb, dst_metrics_ptr(&rt->dst)) < 0)
2492 2493
		goto nla_put_failure;

2494
	n = rt->n;
2495
	if (n) {
2496
		if (nla_put(skb, RTA_GATEWAY, 16, &n->primary_key) < 0)
2497 2498
			goto nla_put_failure;
	}
2499

D
David S. Miller 已提交
2500 2501 2502 2503 2504
	if (rt->dst.dev &&
	    nla_put_u32(skb, RTA_OIF, rt->dst.dev->ifindex))
		goto nla_put_failure;
	if (nla_put_u32(skb, RTA_PRIORITY, rt->rt6i_metric))
		goto nla_put_failure;
2505 2506

	expires = (rt->rt6i_flags & RTF_EXPIRES) ? rt->dst.expires - jiffies : 0;
2507

2508
	if (rtnl_put_cacheinfo(skb, &rt->dst, 0, expires, rt->dst.error) < 0)
2509
		goto nla_put_failure;
2510 2511 2512 2513

	return nlmsg_end(skb, nlh);

nla_put_failure:
2514 2515
	nlmsg_cancel(skb, nlh);
	return -EMSGSIZE;
L
Linus Torvalds 已提交
2516 2517
}

2518
int rt6_dump_route(struct rt6_info *rt, void *p_arg)
L
Linus Torvalds 已提交
2519 2520 2521 2522
{
	struct rt6_rtnl_dump_arg *arg = (struct rt6_rtnl_dump_arg *) p_arg;
	int prefix;

2523 2524
	if (nlmsg_len(arg->cb->nlh) >= sizeof(struct rtmsg)) {
		struct rtmsg *rtm = nlmsg_data(arg->cb->nlh);
L
Linus Torvalds 已提交
2525 2526 2527 2528
		prefix = (rtm->rtm_flags & RTM_F_PREFIX) != 0;
	} else
		prefix = 0;

2529 2530
	return rt6_fill_node(arg->net,
		     arg->skb, rt, NULL, NULL, 0, RTM_NEWROUTE,
2531
		     NETLINK_CB(arg->cb->skb).portid, arg->cb->nlh->nlmsg_seq,
2532
		     prefix, 0, NLM_F_MULTI);
L
Linus Torvalds 已提交
2533 2534
}

2535
static int inet6_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void *arg)
L
Linus Torvalds 已提交
2536
{
2537
	struct net *net = sock_net(in_skb->sk);
2538 2539
	struct nlattr *tb[RTA_MAX+1];
	struct rt6_info *rt;
L
Linus Torvalds 已提交
2540
	struct sk_buff *skb;
2541
	struct rtmsg *rtm;
2542
	struct flowi6 fl6;
2543
	int err, iif = 0, oif = 0;
L
Linus Torvalds 已提交
2544

2545 2546 2547
	err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv6_policy);
	if (err < 0)
		goto errout;
L
Linus Torvalds 已提交
2548

2549
	err = -EINVAL;
2550
	memset(&fl6, 0, sizeof(fl6));
L
Linus Torvalds 已提交
2551

2552 2553 2554 2555
	if (tb[RTA_SRC]) {
		if (nla_len(tb[RTA_SRC]) < sizeof(struct in6_addr))
			goto errout;

A
Alexey Dobriyan 已提交
2556
		fl6.saddr = *(struct in6_addr *)nla_data(tb[RTA_SRC]);
2557 2558 2559 2560 2561 2562
	}

	if (tb[RTA_DST]) {
		if (nla_len(tb[RTA_DST]) < sizeof(struct in6_addr))
			goto errout;

A
Alexey Dobriyan 已提交
2563
		fl6.daddr = *(struct in6_addr *)nla_data(tb[RTA_DST]);
2564 2565 2566 2567 2568 2569
	}

	if (tb[RTA_IIF])
		iif = nla_get_u32(tb[RTA_IIF]);

	if (tb[RTA_OIF])
2570
		oif = nla_get_u32(tb[RTA_OIF]);
L
Linus Torvalds 已提交
2571 2572 2573

	if (iif) {
		struct net_device *dev;
2574 2575
		int flags = 0;

2576
		dev = __dev_get_by_index(net, iif);
L
Linus Torvalds 已提交
2577 2578
		if (!dev) {
			err = -ENODEV;
2579
			goto errout;
L
Linus Torvalds 已提交
2580
		}
2581 2582 2583 2584 2585 2586 2587 2588 2589 2590 2591 2592

		fl6.flowi6_iif = iif;

		if (!ipv6_addr_any(&fl6.saddr))
			flags |= RT6_LOOKUP_F_HAS_SADDR;

		rt = (struct rt6_info *)ip6_route_input_lookup(net, dev, &fl6,
							       flags);
	} else {
		fl6.flowi6_oif = oif;

		rt = (struct rt6_info *)ip6_route_output(net, NULL, &fl6);
L
Linus Torvalds 已提交
2593 2594
	}

2595
	skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL);
2596
	if (!skb) {
2597
		dst_release(&rt->dst);
2598 2599 2600
		err = -ENOBUFS;
		goto errout;
	}
L
Linus Torvalds 已提交
2601

2602 2603 2604
	/* Reserve room for dummy headers, this skb can pass
	   through good chunk of routing engine.
	 */
2605
	skb_reset_mac_header(skb);
2606
	skb_reserve(skb, MAX_HEADER + sizeof(struct ipv6hdr));
L
Linus Torvalds 已提交
2607

2608
	skb_dst_set(skb, &rt->dst);
L
Linus Torvalds 已提交
2609

2610
	err = rt6_fill_node(net, skb, rt, &fl6.daddr, &fl6.saddr, iif,
2611
			    RTM_NEWROUTE, NETLINK_CB(in_skb).portid,
2612
			    nlh->nlmsg_seq, 0, 0, 0);
L
Linus Torvalds 已提交
2613
	if (err < 0) {
2614 2615
		kfree_skb(skb);
		goto errout;
L
Linus Torvalds 已提交
2616 2617
	}

2618
	err = rtnl_unicast(skb, net, NETLINK_CB(in_skb).portid);
2619
errout:
L
Linus Torvalds 已提交
2620 2621 2622
	return err;
}

2623
void inet6_rt_notify(int event, struct rt6_info *rt, struct nl_info *info)
L
Linus Torvalds 已提交
2624 2625
{
	struct sk_buff *skb;
2626
	struct net *net = info->nl_net;
2627 2628 2629 2630
	u32 seq;
	int err;

	err = -ENOBUFS;
2631
	seq = info->nlh ? info->nlh->nlmsg_seq : 0;
2632

2633
	skb = nlmsg_new(rt6_nlmsg_size(), gfp_any());
2634
	if (!skb)
2635 2636
		goto errout;

2637
	err = rt6_fill_node(net, skb, rt, NULL, NULL, 0,
2638
				event, info->portid, seq, 0, 0, 0);
2639 2640 2641 2642 2643 2644
	if (err < 0) {
		/* -EMSGSIZE implies BUG in rt6_nlmsg_size() */
		WARN_ON(err == -EMSGSIZE);
		kfree_skb(skb);
		goto errout;
	}
2645
	rtnl_notify(skb, net, info->portid, RTNLGRP_IPV6_ROUTE,
2646 2647
		    info->nlh, gfp_any());
	return;
2648 2649
errout:
	if (err < 0)
2650
		rtnl_set_sk_err(net, RTNLGRP_IPV6_ROUTE, err);
L
Linus Torvalds 已提交
2651 2652
}

2653 2654 2655 2656
static int ip6_route_dev_notify(struct notifier_block *this,
				unsigned long event, void *data)
{
	struct net_device *dev = (struct net_device *)data;
2657
	struct net *net = dev_net(dev);
2658 2659

	if (event == NETDEV_REGISTER && (dev->flags & IFF_LOOPBACK)) {
2660
		net->ipv6.ip6_null_entry->dst.dev = dev;
2661 2662
		net->ipv6.ip6_null_entry->rt6i_idev = in6_dev_get(dev);
#ifdef CONFIG_IPV6_MULTIPLE_TABLES
2663
		net->ipv6.ip6_prohibit_entry->dst.dev = dev;
2664
		net->ipv6.ip6_prohibit_entry->rt6i_idev = in6_dev_get(dev);
2665
		net->ipv6.ip6_blk_hole_entry->dst.dev = dev;
2666 2667 2668 2669 2670 2671 2672
		net->ipv6.ip6_blk_hole_entry->rt6i_idev = in6_dev_get(dev);
#endif
	}

	return NOTIFY_OK;
}

L
Linus Torvalds 已提交
2673 2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689
/*
 *	/proc
 */

#ifdef CONFIG_PROC_FS

struct rt6_proc_arg
{
	char *buffer;
	int offset;
	int length;
	int skip;
	int len;
};

static int rt6_info_route(struct rt6_info *rt, void *p_arg)
{
2690
	struct seq_file *m = p_arg;
2691
	struct neighbour *n;
L
Linus Torvalds 已提交
2692

2693
	seq_printf(m, "%pi6 %02x ", &rt->rt6i_dst.addr, rt->rt6i_dst.plen);
L
Linus Torvalds 已提交
2694 2695

#ifdef CONFIG_IPV6_SUBTREES
2696
	seq_printf(m, "%pi6 %02x ", &rt->rt6i_src.addr, rt->rt6i_src.plen);
L
Linus Torvalds 已提交
2697
#else
2698
	seq_puts(m, "00000000000000000000000000000000 00 ");
L
Linus Torvalds 已提交
2699
#endif
2700
	n = rt->n;
2701 2702
	if (n) {
		seq_printf(m, "%pi6", n->primary_key);
L
Linus Torvalds 已提交
2703
	} else {
2704
		seq_puts(m, "00000000000000000000000000000000");
L
Linus Torvalds 已提交
2705
	}
2706
	seq_printf(m, " %08x %08x %08x %08x %8s\n",
2707 2708
		   rt->rt6i_metric, atomic_read(&rt->dst.__refcnt),
		   rt->dst.__use, rt->rt6i_flags,
2709
		   rt->dst.dev ? rt->dst.dev->name : "");
L
Linus Torvalds 已提交
2710 2711 2712
	return 0;
}

2713
static int ipv6_route_show(struct seq_file *m, void *v)
L
Linus Torvalds 已提交
2714
{
2715
	struct net *net = (struct net *)m->private;
2716
	fib6_clean_all_ro(net, rt6_info_route, 0, m);
2717 2718
	return 0;
}
L
Linus Torvalds 已提交
2719

2720 2721
static int ipv6_route_open(struct inode *inode, struct file *file)
{
2722
	return single_open_net(inode, file, ipv6_route_show);
2723 2724
}

2725 2726 2727 2728 2729
static const struct file_operations ipv6_route_proc_fops = {
	.owner		= THIS_MODULE,
	.open		= ipv6_route_open,
	.read		= seq_read,
	.llseek		= seq_lseek,
2730
	.release	= single_release_net,
2731 2732
};

L
Linus Torvalds 已提交
2733 2734
static int rt6_stats_seq_show(struct seq_file *seq, void *v)
{
2735
	struct net *net = (struct net *)seq->private;
L
Linus Torvalds 已提交
2736
	seq_printf(seq, "%04x %04x %04x %04x %04x %04x %04x\n",
2737 2738 2739 2740 2741
		   net->ipv6.rt6_stats->fib_nodes,
		   net->ipv6.rt6_stats->fib_route_nodes,
		   net->ipv6.rt6_stats->fib_rt_alloc,
		   net->ipv6.rt6_stats->fib_rt_entries,
		   net->ipv6.rt6_stats->fib_rt_cache,
2742
		   dst_entries_get_slow(&net->ipv6.ip6_dst_ops),
2743
		   net->ipv6.rt6_stats->fib_discarded_routes);
L
Linus Torvalds 已提交
2744 2745 2746 2747 2748 2749

	return 0;
}

static int rt6_stats_seq_open(struct inode *inode, struct file *file)
{
2750
	return single_open_net(inode, file, rt6_stats_seq_show);
2751 2752
}

2753
static const struct file_operations rt6_stats_seq_fops = {
L
Linus Torvalds 已提交
2754 2755 2756 2757
	.owner	 = THIS_MODULE,
	.open	 = rt6_stats_seq_open,
	.read	 = seq_read,
	.llseek	 = seq_lseek,
2758
	.release = single_release_net,
L
Linus Torvalds 已提交
2759 2760 2761 2762 2763 2764
};
#endif	/* CONFIG_PROC_FS */

#ifdef CONFIG_SYSCTL

static
2765
int ipv6_sysctl_rtcache_flush(ctl_table *ctl, int write,
L
Linus Torvalds 已提交
2766 2767
			      void __user *buffer, size_t *lenp, loff_t *ppos)
{
2768 2769 2770
	struct net *net;
	int delay;
	if (!write)
L
Linus Torvalds 已提交
2771
		return -EINVAL;
2772 2773 2774 2775 2776 2777

	net = (struct net *)ctl->extra1;
	delay = net->ipv6.sysctl.flush_delay;
	proc_dointvec(ctl, write, buffer, lenp, ppos);
	fib6_run_gc(delay <= 0 ? ~0UL : (unsigned long)delay, net);
	return 0;
L
Linus Torvalds 已提交
2778 2779
}

2780
ctl_table ipv6_route_table_template[] = {
2781
	{
L
Linus Torvalds 已提交
2782
		.procname	=	"flush",
2783
		.data		=	&init_net.ipv6.sysctl.flush_delay,
L
Linus Torvalds 已提交
2784
		.maxlen		=	sizeof(int),
2785
		.mode		=	0200,
A
Alexey Dobriyan 已提交
2786
		.proc_handler	=	ipv6_sysctl_rtcache_flush
L
Linus Torvalds 已提交
2787 2788 2789
	},
	{
		.procname	=	"gc_thresh",
2790
		.data		=	&ip6_dst_ops_template.gc_thresh,
L
Linus Torvalds 已提交
2791 2792
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2793
		.proc_handler	=	proc_dointvec,
L
Linus Torvalds 已提交
2794 2795 2796
	},
	{
		.procname	=	"max_size",
2797
		.data		=	&init_net.ipv6.sysctl.ip6_rt_max_size,
L
Linus Torvalds 已提交
2798 2799
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2800
		.proc_handler	=	proc_dointvec,
L
Linus Torvalds 已提交
2801 2802 2803
	},
	{
		.procname	=	"gc_min_interval",
2804
		.data		=	&init_net.ipv6.sysctl.ip6_rt_gc_min_interval,
L
Linus Torvalds 已提交
2805 2806
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2807
		.proc_handler	=	proc_dointvec_jiffies,
L
Linus Torvalds 已提交
2808 2809 2810
	},
	{
		.procname	=	"gc_timeout",
2811
		.data		=	&init_net.ipv6.sysctl.ip6_rt_gc_timeout,
L
Linus Torvalds 已提交
2812 2813
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2814
		.proc_handler	=	proc_dointvec_jiffies,
L
Linus Torvalds 已提交
2815 2816 2817
	},
	{
		.procname	=	"gc_interval",
2818
		.data		=	&init_net.ipv6.sysctl.ip6_rt_gc_interval,
L
Linus Torvalds 已提交
2819 2820
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2821
		.proc_handler	=	proc_dointvec_jiffies,
L
Linus Torvalds 已提交
2822 2823 2824
	},
	{
		.procname	=	"gc_elasticity",
2825
		.data		=	&init_net.ipv6.sysctl.ip6_rt_gc_elasticity,
L
Linus Torvalds 已提交
2826 2827
		.maxlen		=	sizeof(int),
		.mode		=	0644,
2828
		.proc_handler	=	proc_dointvec,
L
Linus Torvalds 已提交
2829 2830 2831
	},
	{
		.procname	=	"mtu_expires",
2832
		.data		=	&init_net.ipv6.sysctl.ip6_rt_mtu_expires,
L
Linus Torvalds 已提交
2833 2834
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2835
		.proc_handler	=	proc_dointvec_jiffies,
L
Linus Torvalds 已提交
2836 2837 2838
	},
	{
		.procname	=	"min_adv_mss",
2839
		.data		=	&init_net.ipv6.sysctl.ip6_rt_min_advmss,
L
Linus Torvalds 已提交
2840 2841
		.maxlen		=	sizeof(int),
		.mode		=	0644,
2842
		.proc_handler	=	proc_dointvec,
L
Linus Torvalds 已提交
2843 2844 2845
	},
	{
		.procname	=	"gc_min_interval_ms",
2846
		.data		=	&init_net.ipv6.sysctl.ip6_rt_gc_min_interval,
L
Linus Torvalds 已提交
2847 2848
		.maxlen		=	sizeof(int),
		.mode		=	0644,
A
Alexey Dobriyan 已提交
2849
		.proc_handler	=	proc_dointvec_ms_jiffies,
L
Linus Torvalds 已提交
2850
	},
2851
	{ }
L
Linus Torvalds 已提交
2852 2853
};

2854
struct ctl_table * __net_init ipv6_route_sysctl_init(struct net *net)
2855 2856 2857 2858 2859 2860
{
	struct ctl_table *table;

	table = kmemdup(ipv6_route_table_template,
			sizeof(ipv6_route_table_template),
			GFP_KERNEL);
2861 2862 2863

	if (table) {
		table[0].data = &net->ipv6.sysctl.flush_delay;
2864
		table[0].extra1 = net;
2865
		table[1].data = &net->ipv6.ip6_dst_ops.gc_thresh;
2866 2867 2868 2869 2870 2871 2872
		table[2].data = &net->ipv6.sysctl.ip6_rt_max_size;
		table[3].data = &net->ipv6.sysctl.ip6_rt_gc_min_interval;
		table[4].data = &net->ipv6.sysctl.ip6_rt_gc_timeout;
		table[5].data = &net->ipv6.sysctl.ip6_rt_gc_interval;
		table[6].data = &net->ipv6.sysctl.ip6_rt_gc_elasticity;
		table[7].data = &net->ipv6.sysctl.ip6_rt_mtu_expires;
		table[8].data = &net->ipv6.sysctl.ip6_rt_min_advmss;
2873
		table[9].data = &net->ipv6.sysctl.ip6_rt_gc_min_interval;
2874 2875
	}

2876 2877
	return table;
}
L
Linus Torvalds 已提交
2878 2879
#endif

2880
static int __net_init ip6_route_net_init(struct net *net)
2881
{
2882
	int ret = -ENOMEM;
2883

2884 2885
	memcpy(&net->ipv6.ip6_dst_ops, &ip6_dst_ops_template,
	       sizeof(net->ipv6.ip6_dst_ops));
2886

2887 2888 2889
	if (dst_entries_init(&net->ipv6.ip6_dst_ops) < 0)
		goto out_ip6_dst_ops;

2890 2891 2892 2893
	net->ipv6.ip6_null_entry = kmemdup(&ip6_null_entry_template,
					   sizeof(*net->ipv6.ip6_null_entry),
					   GFP_KERNEL);
	if (!net->ipv6.ip6_null_entry)
2894
		goto out_ip6_dst_entries;
2895
	net->ipv6.ip6_null_entry->dst.path =
2896
		(struct dst_entry *)net->ipv6.ip6_null_entry;
2897
	net->ipv6.ip6_null_entry->dst.ops = &net->ipv6.ip6_dst_ops;
2898 2899
	dst_init_metrics(&net->ipv6.ip6_null_entry->dst,
			 ip6_template_metrics, true);
2900 2901 2902 2903 2904

#ifdef CONFIG_IPV6_MULTIPLE_TABLES
	net->ipv6.ip6_prohibit_entry = kmemdup(&ip6_prohibit_entry_template,
					       sizeof(*net->ipv6.ip6_prohibit_entry),
					       GFP_KERNEL);
2905 2906
	if (!net->ipv6.ip6_prohibit_entry)
		goto out_ip6_null_entry;
2907
	net->ipv6.ip6_prohibit_entry->dst.path =
2908
		(struct dst_entry *)net->ipv6.ip6_prohibit_entry;
2909
	net->ipv6.ip6_prohibit_entry->dst.ops = &net->ipv6.ip6_dst_ops;
2910 2911
	dst_init_metrics(&net->ipv6.ip6_prohibit_entry->dst,
			 ip6_template_metrics, true);
2912 2913 2914 2915

	net->ipv6.ip6_blk_hole_entry = kmemdup(&ip6_blk_hole_entry_template,
					       sizeof(*net->ipv6.ip6_blk_hole_entry),
					       GFP_KERNEL);
2916 2917
	if (!net->ipv6.ip6_blk_hole_entry)
		goto out_ip6_prohibit_entry;
2918
	net->ipv6.ip6_blk_hole_entry->dst.path =
2919
		(struct dst_entry *)net->ipv6.ip6_blk_hole_entry;
2920
	net->ipv6.ip6_blk_hole_entry->dst.ops = &net->ipv6.ip6_dst_ops;
2921 2922
	dst_init_metrics(&net->ipv6.ip6_blk_hole_entry->dst,
			 ip6_template_metrics, true);
2923 2924
#endif

2925 2926 2927 2928 2929 2930 2931 2932 2933
	net->ipv6.sysctl.flush_delay = 0;
	net->ipv6.sysctl.ip6_rt_max_size = 4096;
	net->ipv6.sysctl.ip6_rt_gc_min_interval = HZ / 2;
	net->ipv6.sysctl.ip6_rt_gc_timeout = 60*HZ;
	net->ipv6.sysctl.ip6_rt_gc_interval = 30*HZ;
	net->ipv6.sysctl.ip6_rt_gc_elasticity = 9;
	net->ipv6.sysctl.ip6_rt_mtu_expires = 10*60*HZ;
	net->ipv6.sysctl.ip6_rt_min_advmss = IPV6_MIN_MTU - 20 - 40;

2934 2935
	net->ipv6.ip6_rt_gc_expire = 30*HZ;

2936 2937 2938
	ret = 0;
out:
	return ret;
2939

2940 2941 2942 2943 2944 2945
#ifdef CONFIG_IPV6_MULTIPLE_TABLES
out_ip6_prohibit_entry:
	kfree(net->ipv6.ip6_prohibit_entry);
out_ip6_null_entry:
	kfree(net->ipv6.ip6_null_entry);
#endif
2946 2947
out_ip6_dst_entries:
	dst_entries_destroy(&net->ipv6.ip6_dst_ops);
2948 2949
out_ip6_dst_ops:
	goto out;
2950 2951
}

2952
static void __net_exit ip6_route_net_exit(struct net *net)
2953
{
2954 2955 2956 2957 2958
	kfree(net->ipv6.ip6_null_entry);
#ifdef CONFIG_IPV6_MULTIPLE_TABLES
	kfree(net->ipv6.ip6_prohibit_entry);
	kfree(net->ipv6.ip6_blk_hole_entry);
#endif
2959
	dst_entries_destroy(&net->ipv6.ip6_dst_ops);
2960 2961
}

2962 2963 2964 2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975 2976 2977 2978
static int __net_init ip6_route_net_init_late(struct net *net)
{
#ifdef CONFIG_PROC_FS
	proc_net_fops_create(net, "ipv6_route", 0, &ipv6_route_proc_fops);
	proc_net_fops_create(net, "rt6_stats", S_IRUGO, &rt6_stats_seq_fops);
#endif
	return 0;
}

static void __net_exit ip6_route_net_exit_late(struct net *net)
{
#ifdef CONFIG_PROC_FS
	proc_net_remove(net, "ipv6_route");
	proc_net_remove(net, "rt6_stats");
#endif
}

2979 2980 2981 2982 2983
static struct pernet_operations ip6_route_net_ops = {
	.init = ip6_route_net_init,
	.exit = ip6_route_net_exit,
};

2984 2985 2986 2987 2988 2989 2990 2991 2992 2993 2994 2995 2996 2997 2998 2999
static int __net_init ipv6_inetpeer_init(struct net *net)
{
	struct inet_peer_base *bp = kmalloc(sizeof(*bp), GFP_KERNEL);

	if (!bp)
		return -ENOMEM;
	inet_peer_base_init(bp);
	net->ipv6.peers = bp;
	return 0;
}

static void __net_exit ipv6_inetpeer_exit(struct net *net)
{
	struct inet_peer_base *bp = net->ipv6.peers;

	net->ipv6.peers = NULL;
3000
	inetpeer_invalidate_tree(bp);
3001 3002 3003
	kfree(bp);
}

3004
static struct pernet_operations ipv6_inetpeer_ops = {
3005 3006 3007 3008
	.init	=	ipv6_inetpeer_init,
	.exit	=	ipv6_inetpeer_exit,
};

3009 3010 3011 3012 3013
static struct pernet_operations ip6_route_net_late_ops = {
	.init = ip6_route_net_init_late,
	.exit = ip6_route_net_exit_late,
};

3014 3015 3016 3017 3018
static struct notifier_block ip6_route_dev_notifier = {
	.notifier_call = ip6_route_dev_notify,
	.priority = 0,
};

3019
int __init ip6_route_init(void)
L
Linus Torvalds 已提交
3020
{
3021 3022
	int ret;

3023 3024
	ret = -ENOMEM;
	ip6_dst_ops_template.kmem_cachep =
A
Alexey Dobriyan 已提交
3025
		kmem_cache_create("ip6_dst_cache", sizeof(struct rt6_info), 0,
3026
				  SLAB_HWCACHE_ALIGN, NULL);
3027
	if (!ip6_dst_ops_template.kmem_cachep)
3028
		goto out;
3029

3030
	ret = dst_entries_init(&ip6_dst_blackhole_ops);
3031
	if (ret)
3032 3033
		goto out_kmem_cache;

3034 3035
	ret = register_pernet_subsys(&ipv6_inetpeer_ops);
	if (ret)
3036
		goto out_dst_entries;
3037

3038 3039 3040
	ret = register_pernet_subsys(&ip6_route_net_ops);
	if (ret)
		goto out_register_inetpeer;
3041

3042 3043
	ip6_dst_blackhole_ops.kmem_cachep = ip6_dst_ops_template.kmem_cachep;

3044 3045 3046
	/* Registering of the loopback is done before this portion of code,
	 * the loopback reference in rt6_info will not be taken, do it
	 * manually for init_net */
3047
	init_net.ipv6.ip6_null_entry->dst.dev = init_net.loopback_dev;
3048 3049
	init_net.ipv6.ip6_null_entry->rt6i_idev = in6_dev_get(init_net.loopback_dev);
  #ifdef CONFIG_IPV6_MULTIPLE_TABLES
3050
	init_net.ipv6.ip6_prohibit_entry->dst.dev = init_net.loopback_dev;
3051
	init_net.ipv6.ip6_prohibit_entry->rt6i_idev = in6_dev_get(init_net.loopback_dev);
3052
	init_net.ipv6.ip6_blk_hole_entry->dst.dev = init_net.loopback_dev;
3053 3054
	init_net.ipv6.ip6_blk_hole_entry->rt6i_idev = in6_dev_get(init_net.loopback_dev);
  #endif
3055
	ret = fib6_init();
3056
	if (ret)
3057
		goto out_register_subsys;
3058 3059 3060

	ret = xfrm6_init();
	if (ret)
3061
		goto out_fib6_init;
3062

3063 3064 3065
	ret = fib6_rules_init();
	if (ret)
		goto xfrm6_init;
3066

3067 3068 3069 3070
	ret = register_pernet_subsys(&ip6_route_net_late_ops);
	if (ret)
		goto fib6_rules_init;

3071
	ret = -ENOBUFS;
3072 3073 3074
	if (__rtnl_register(PF_INET6, RTM_NEWROUTE, inet6_rtm_newroute, NULL, NULL) ||
	    __rtnl_register(PF_INET6, RTM_DELROUTE, inet6_rtm_delroute, NULL, NULL) ||
	    __rtnl_register(PF_INET6, RTM_GETROUTE, inet6_rtm_getroute, NULL, NULL))
3075
		goto out_register_late_subsys;
3076

3077
	ret = register_netdevice_notifier(&ip6_route_dev_notifier);
3078
	if (ret)
3079
		goto out_register_late_subsys;
3080

3081 3082 3083
out:
	return ret;

3084 3085
out_register_late_subsys:
	unregister_pernet_subsys(&ip6_route_net_late_ops);
3086 3087 3088 3089
fib6_rules_init:
	fib6_rules_cleanup();
xfrm6_init:
	xfrm6_fini();
3090 3091
out_fib6_init:
	fib6_gc_cleanup();
3092 3093
out_register_subsys:
	unregister_pernet_subsys(&ip6_route_net_ops);
3094 3095
out_register_inetpeer:
	unregister_pernet_subsys(&ipv6_inetpeer_ops);
3096 3097
out_dst_entries:
	dst_entries_destroy(&ip6_dst_blackhole_ops);
3098
out_kmem_cache:
3099
	kmem_cache_destroy(ip6_dst_ops_template.kmem_cachep);
3100
	goto out;
L
Linus Torvalds 已提交
3101 3102 3103 3104
}

void ip6_route_cleanup(void)
{
3105
	unregister_netdevice_notifier(&ip6_route_dev_notifier);
3106
	unregister_pernet_subsys(&ip6_route_net_late_ops);
T
Thomas Graf 已提交
3107
	fib6_rules_cleanup();
L
Linus Torvalds 已提交
3108 3109
	xfrm6_fini();
	fib6_gc_cleanup();
3110
	unregister_pernet_subsys(&ipv6_inetpeer_ops);
3111
	unregister_pernet_subsys(&ip6_route_net_ops);
3112
	dst_entries_destroy(&ip6_dst_blackhole_ops);
3113
	kmem_cache_destroy(ip6_dst_ops_template.kmem_cachep);
L
Linus Torvalds 已提交
3114
}