ip6mr.c 56.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/*
 *	Linux IPv6 multicast routing support for BSD pim6sd
 *	Based on net/ipv4/ipmr.c.
 *
 *	(c) 2004 Mickael Hoerdt, <hoerdt@clarinet.u-strasbg.fr>
 *		LSIIT Laboratory, Strasbourg, France
 *	(c) 2004 Jean-Philippe Andriot, <jean-philippe.andriot@6WIND.com>
 *		6WIND, Paris, France
 *	Copyright (C)2007,2008 USAGI/WIDE Project
 *		YOSHIFUJI Hideaki <yoshfuji@linux-ipv6.org>
 *
 *	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.
 *
 */

19
#include <linux/uaccess.h>
20 21 22 23 24 25 26 27 28 29 30 31 32 33
#include <linux/types.h>
#include <linux/sched.h>
#include <linux/errno.h>
#include <linux/mm.h>
#include <linux/kernel.h>
#include <linux/fcntl.h>
#include <linux/stat.h>
#include <linux/socket.h>
#include <linux/inet.h>
#include <linux/netdevice.h>
#include <linux/inetdevice.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/init.h>
34
#include <linux/compat.h>
35 36 37 38 39 40 41
#include <net/protocol.h>
#include <linux/skbuff.h>
#include <net/raw.h>
#include <linux/notifier.h>
#include <linux/if_arp.h>
#include <net/checksum.h>
#include <net/netlink.h>
42
#include <net/fib_rules.h>
43 44 45 46

#include <net/ipv6.h>
#include <net/ip6_route.h>
#include <linux/mroute6.h>
47
#include <linux/pim.h>
48 49
#include <net/addrconf.h>
#include <linux/netfilter_ipv6.h>
50
#include <linux/export.h>
D
Dave Jones 已提交
51
#include <net/ip6_checksum.h>
52
#include <linux/netconf.h>
53

54 55 56 57 58
struct ip6mr_rule {
	struct fib_rule		common;
};

struct ip6mr_result {
59
	struct mr_table	*mrt;
60 61
};

62 63 64 65 66 67
/* Big lock, protecting vif table, mrt cache and mroute socket state.
   Note that the changes are semaphored via rtnl_lock.
 */

static DEFINE_RWLOCK(mrt_lock);

68
/* Multicast router control variables */
69 70 71 72 73 74 75 76 77 78 79 80 81 82

/* Special spinlock for queue of unresolved entries */
static DEFINE_SPINLOCK(mfc_unres_lock);

/* We return to original Alan's scheme. Hash table of resolved
   entries is changed only in process context and protected
   with weak lock mrt_lock. Queue of unresolved entries is protected
   with strong spinlock mfc_unres_lock.

   In this case data path is free of exclusive locks at all.
 */

static struct kmem_cache *mrt_cachep __read_mostly;

83 84
static struct mr_table *ip6mr_new_table(struct net *net, u32 id);
static void ip6mr_free_table(struct mr_table *mrt);
85

86
static void ip6_mr_forward(struct net *net, struct mr_table *mrt,
87
			   struct sk_buff *skb, struct mfc6_cache *cache);
88
static int ip6mr_cache_report(struct mr_table *mrt, struct sk_buff *pkt,
89
			      mifi_t mifi, int assert);
90
static void mr6_netlink_event(struct mr_table *mrt, struct mfc6_cache *mfc,
91
			      int cmd);
92
static void mrt6msg_netlink_event(struct mr_table *mrt, struct sk_buff *pkt);
93 94
static int ip6mr_rtm_dumproute(struct sk_buff *skb,
			       struct netlink_callback *cb);
95
static void mroute_clean_tables(struct mr_table *mrt, bool all);
96
static void ipmr_expire_process(struct timer_list *t);
97 98

#ifdef CONFIG_IPV6_MROUTE_MULTIPLE_TABLES
99
#define ip6mr_for_each_table(mrt, net) \
100 101
	list_for_each_entry_rcu(mrt, &net->ipv6.mr6_tables, list)

Y
Yuval Mintz 已提交
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
static struct mr_table *ip6mr_mr_table_iter(struct net *net,
					    struct mr_table *mrt)
{
	struct mr_table *ret;

	if (!mrt)
		ret = list_entry_rcu(net->ipv6.mr6_tables.next,
				     struct mr_table, list);
	else
		ret = list_entry_rcu(mrt->list.next,
				     struct mr_table, list);

	if (&ret->list == &net->ipv6.mr6_tables)
		return NULL;
	return ret;
}

119
static struct mr_table *ip6mr_get_table(struct net *net, u32 id)
120
{
121
	struct mr_table *mrt;
122 123 124 125 126 127 128 129

	ip6mr_for_each_table(mrt, net) {
		if (mrt->id == id)
			return mrt;
	}
	return NULL;
}

130
static int ip6mr_fib_lookup(struct net *net, struct flowi6 *flp6,
131
			    struct mr_table **mrt)
132 133
{
	int err;
134 135 136 137 138
	struct ip6mr_result res;
	struct fib_lookup_arg arg = {
		.result = &res,
		.flags = FIB_LOOKUP_NOREF,
	};
139

140 141
	err = fib_rules_lookup(net->ipv6.mr6_rules_ops,
			       flowi6_to_flowi(flp6), 0, &arg);
142 143 144 145 146 147 148 149 150 151
	if (err < 0)
		return err;
	*mrt = res.mrt;
	return 0;
}

static int ip6mr_rule_action(struct fib_rule *rule, struct flowi *flp,
			     int flags, struct fib_lookup_arg *arg)
{
	struct ip6mr_result *res = arg->result;
152
	struct mr_table *mrt;
153 154 155 156 157 158 159 160 161 162 163 164 165 166

	switch (rule->action) {
	case FR_ACT_TO_TBL:
		break;
	case FR_ACT_UNREACHABLE:
		return -ENETUNREACH;
	case FR_ACT_PROHIBIT:
		return -EACCES;
	case FR_ACT_BLACKHOLE:
	default:
		return -EINVAL;
	}

	mrt = ip6mr_get_table(rule->fr_net, rule->table);
167
	if (!mrt)
168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
		return -EAGAIN;
	res->mrt = mrt;
	return 0;
}

static int ip6mr_rule_match(struct fib_rule *rule, struct flowi *flp, int flags)
{
	return 1;
}

static const struct nla_policy ip6mr_rule_policy[FRA_MAX + 1] = {
	FRA_GENERIC_POLICY,
};

static int ip6mr_rule_configure(struct fib_rule *rule, struct sk_buff *skb,
				struct fib_rule_hdr *frh, struct nlattr **tb)
{
	return 0;
}

static int ip6mr_rule_compare(struct fib_rule *rule, struct fib_rule_hdr *frh,
			      struct nlattr **tb)
{
	return 1;
}

static int ip6mr_rule_fill(struct fib_rule *rule, struct sk_buff *skb,
			   struct fib_rule_hdr *frh)
{
	frh->dst_len = 0;
	frh->src_len = 0;
	frh->tos     = 0;
	return 0;
}

203
static const struct fib_rules_ops __net_initconst ip6mr_rules_ops_template = {
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
	.family		= RTNL_FAMILY_IP6MR,
	.rule_size	= sizeof(struct ip6mr_rule),
	.addr_size	= sizeof(struct in6_addr),
	.action		= ip6mr_rule_action,
	.match		= ip6mr_rule_match,
	.configure	= ip6mr_rule_configure,
	.compare	= ip6mr_rule_compare,
	.fill		= ip6mr_rule_fill,
	.nlgroup	= RTNLGRP_IPV6_RULE,
	.policy		= ip6mr_rule_policy,
	.owner		= THIS_MODULE,
};

static int __net_init ip6mr_rules_init(struct net *net)
{
	struct fib_rules_ops *ops;
220
	struct mr_table *mrt;
221 222 223 224 225 226 227 228 229
	int err;

	ops = fib_rules_register(&ip6mr_rules_ops_template, net);
	if (IS_ERR(ops))
		return PTR_ERR(ops);

	INIT_LIST_HEAD(&net->ipv6.mr6_tables);

	mrt = ip6mr_new_table(net, RT6_TABLE_DFLT);
230
	if (!mrt) {
231 232 233 234 235 236 237 238 239 240 241 242
		err = -ENOMEM;
		goto err1;
	}

	err = fib_default_rule_add(ops, 0x7fff, RT6_TABLE_DFLT, 0);
	if (err < 0)
		goto err2;

	net->ipv6.mr6_rules_ops = ops;
	return 0;

err2:
243
	ip6mr_free_table(mrt);
244 245 246 247 248 249 250
err1:
	fib_rules_unregister(ops);
	return err;
}

static void __net_exit ip6mr_rules_exit(struct net *net)
{
251
	struct mr_table *mrt, *next;
252

253
	rtnl_lock();
E
Eric Dumazet 已提交
254 255
	list_for_each_entry_safe(mrt, next, &net->ipv6.mr6_tables, list) {
		list_del(&mrt->list);
256
		ip6mr_free_table(mrt);
E
Eric Dumazet 已提交
257
	}
258
	fib_rules_unregister(net->ipv6.mr6_rules_ops);
259
	rtnl_unlock();
260
}
Y
Yuval Mintz 已提交
261 262 263 264 265 266 267 268 269 270

static int ip6mr_rules_dump(struct net *net, struct notifier_block *nb)
{
	return fib_rules_dump(net, nb, RTNL_FAMILY_IP6MR);
}

static unsigned int ip6mr_rules_seq_read(struct net *net)
{
	return fib_rules_seq_read(net, RTNL_FAMILY_IP6MR);
}
271 272 273 274
#else
#define ip6mr_for_each_table(mrt, net) \
	for (mrt = net->ipv6.mrt6; mrt; mrt = NULL)

Y
Yuval Mintz 已提交
275 276 277 278 279 280 281 282
static struct mr_table *ip6mr_mr_table_iter(struct net *net,
					    struct mr_table *mrt)
{
	if (!mrt)
		return net->ipv6.mrt6;
	return NULL;
}

283
static struct mr_table *ip6mr_get_table(struct net *net, u32 id)
284 285 286 287
{
	return net->ipv6.mrt6;
}

288
static int ip6mr_fib_lookup(struct net *net, struct flowi6 *flp6,
289
			    struct mr_table **mrt)
290 291 292 293 294 295 296 297 298 299 300 301 302
{
	*mrt = net->ipv6.mrt6;
	return 0;
}

static int __net_init ip6mr_rules_init(struct net *net)
{
	net->ipv6.mrt6 = ip6mr_new_table(net, RT6_TABLE_DFLT);
	return net->ipv6.mrt6 ? 0 : -ENOMEM;
}

static void __net_exit ip6mr_rules_exit(struct net *net)
{
303
	rtnl_lock();
304
	ip6mr_free_table(net->ipv6.mrt6);
305 306
	net->ipv6.mrt6 = NULL;
	rtnl_unlock();
307
}
Y
Yuval Mintz 已提交
308 309 310 311 312 313 314 315 316 317

static int ip6mr_rules_dump(struct net *net, struct notifier_block *nb)
{
	return 0;
}

static unsigned int ip6mr_rules_seq_read(struct net *net)
{
	return 0;
}
318 319
#endif

320 321 322 323 324 325 326 327 328 329 330
static int ip6mr_hash_cmp(struct rhashtable_compare_arg *arg,
			  const void *ptr)
{
	const struct mfc6_cache_cmp_arg *cmparg = arg->key;
	struct mfc6_cache *c = (struct mfc6_cache *)ptr;

	return !ipv6_addr_equal(&c->mf6c_mcastgrp, &cmparg->mf6c_mcastgrp) ||
	       !ipv6_addr_equal(&c->mf6c_origin, &cmparg->mf6c_origin);
}

static const struct rhashtable_params ip6mr_rht_params = {
331
	.head_offset = offsetof(struct mr_mfc, mnode),
332 333 334 335 336 337 338 339
	.key_offset = offsetof(struct mfc6_cache, cmparg),
	.key_len = sizeof(struct mfc6_cache_cmp_arg),
	.nelem_hint = 3,
	.locks_mul = 1,
	.obj_cmpfn = ip6mr_hash_cmp,
	.automatic_shrinking = true,
};

340 341 342 343 344 345 346 347
static void ip6mr_new_table_set(struct mr_table *mrt,
				struct net *net)
{
#ifdef CONFIG_IPV6_MROUTE_MULTIPLE_TABLES
	list_add_tail_rcu(&mrt->list, &net->ipv6.mr6_tables);
#endif
}

348 349 350 351 352 353 354 355 356 357
static struct mfc6_cache_cmp_arg ip6mr_mr_table_ops_cmparg_any = {
	.mf6c_origin = IN6ADDR_ANY_INIT,
	.mf6c_mcastgrp = IN6ADDR_ANY_INIT,
};

static struct mr_table_ops ip6mr_mr_table_ops = {
	.rht_params = &ip6mr_rht_params,
	.cmparg_any = &ip6mr_mr_table_ops_cmparg_any,
};

358
static struct mr_table *ip6mr_new_table(struct net *net, u32 id)
359
{
360
	struct mr_table *mrt;
361 362

	mrt = ip6mr_get_table(net, id);
363
	if (mrt)
364 365
		return mrt;

366
	return mr_table_alloc(net, id, &ip6mr_mr_table_ops,
367
			      ipmr_expire_process, ip6mr_new_table_set);
368
}
369

370
static void ip6mr_free_table(struct mr_table *mrt)
371
{
372
	del_timer_sync(&mrt->ipmr_expire_timer);
373
	mroute_clean_tables(mrt, true);
374
	rhltable_destroy(&mrt->mfc_hash);
375 376
	kfree(mrt);
}
377 378

#ifdef CONFIG_PROC_FS
Y
Yuval Mintz 已提交
379 380
/* The /proc interfaces to multicast routing
 * /proc/ip6_mr_cache /proc/ip6_mr_vif
381 382 383 384 385
 */

static void *ip6mr_vif_seq_start(struct seq_file *seq, loff_t *pos)
	__acquires(mrt_lock)
{
386
	struct mr_vif_iter *iter = seq->private;
387
	struct net *net = seq_file_net(seq);
388
	struct mr_table *mrt;
389 390

	mrt = ip6mr_get_table(net, RT6_TABLE_DFLT);
391
	if (!mrt)
392 393 394
		return ERR_PTR(-ENOENT);

	iter->mrt = mrt;
395

396
	read_lock(&mrt_lock);
397
	return mr_vif_seq_start(seq, pos);
398 399 400 401 402 403 404 405 406 407
}

static void ip6mr_vif_seq_stop(struct seq_file *seq, void *v)
	__releases(mrt_lock)
{
	read_unlock(&mrt_lock);
}

static int ip6mr_vif_seq_show(struct seq_file *seq, void *v)
{
408
	struct mr_vif_iter *iter = seq->private;
409
	struct mr_table *mrt = iter->mrt;
410

411 412 413 414
	if (v == SEQ_START_TOKEN) {
		seq_puts(seq,
			 "Interface      BytesIn  PktsIn  BytesOut PktsOut Flags\n");
	} else {
415
		const struct vif_device *vif = v;
416 417 418
		const char *name = vif->dev ? vif->dev->name : "none";

		seq_printf(seq,
A
Al Viro 已提交
419
			   "%2td %-10s %8ld %7ld  %8ld %7ld %05X\n",
420
			   vif - mrt->vif_table,
421 422 423 424 425 426 427
			   name, vif->bytes_in, vif->pkt_in,
			   vif->bytes_out, vif->pkt_out,
			   vif->flags);
	}
	return 0;
}

428
static const struct seq_operations ip6mr_vif_seq_ops = {
429
	.start = ip6mr_vif_seq_start,
430
	.next  = mr_vif_seq_next,
431 432 433 434 435 436
	.stop  = ip6mr_vif_seq_stop,
	.show  = ip6mr_vif_seq_show,
};

static int ip6mr_vif_open(struct inode *inode, struct file *file)
{
437
	return seq_open_net(inode, file, &ip6mr_vif_seq_ops,
438
			    sizeof(struct mr_vif_iter));
439 440
}

441
static const struct file_operations ip6mr_vif_fops = {
442 443 444
	.open    = ip6mr_vif_open,
	.read    = seq_read,
	.llseek  = seq_lseek,
445
	.release = seq_release_net,
446 447 448 449
};

static void *ipmr_mfc_seq_start(struct seq_file *seq, loff_t *pos)
{
450
	struct net *net = seq_file_net(seq);
451
	struct mr_table *mrt;
452

453
	mrt = ip6mr_get_table(net, RT6_TABLE_DFLT);
454
	if (!mrt)
455 456
		return ERR_PTR(-ENOENT);

Y
Yuval Mintz 已提交
457
	return mr_mfc_seq_start(seq, pos, mrt, &mfc_unres_lock);
458 459 460 461 462 463 464 465 466 467 468 469 470
}

static int ipmr_mfc_seq_show(struct seq_file *seq, void *v)
{
	int n;

	if (v == SEQ_START_TOKEN) {
		seq_puts(seq,
			 "Group                            "
			 "Origin                           "
			 "Iif      Pkts  Bytes     Wrong  Oifs\n");
	} else {
		const struct mfc6_cache *mfc = v;
Y
Yuval Mintz 已提交
471
		const struct mr_mfc_iter *it = seq->private;
472
		struct mr_table *mrt = it->mrt;
473

474
		seq_printf(seq, "%pI6 %pI6 %-3hd",
475
			   &mfc->mf6c_mcastgrp, &mfc->mf6c_origin,
476
			   mfc->_c.mfc_parent);
477

478
		if (it->cache != &mrt->mfc_unres_queue) {
479
			seq_printf(seq, " %8lu %8lu %8lu",
480 481 482 483 484
				   mfc->_c.mfc_un.res.pkt,
				   mfc->_c.mfc_un.res.bytes,
				   mfc->_c.mfc_un.res.wrong_if);
			for (n = mfc->_c.mfc_un.res.minvif;
			     n < mfc->_c.mfc_un.res.maxvif; n++) {
485
				if (VIF_EXISTS(mrt, n) &&
486
				    mfc->_c.mfc_un.res.ttls[n] < 255)
487
					seq_printf(seq,
488 489
						   " %2d:%-3d", n,
						   mfc->_c.mfc_un.res.ttls[n]);
490
			}
491 492 493 494 495
		} else {
			/* unresolved mfc_caches don't contain
			 * pkt, bytes and wrong_if values
			 */
			seq_printf(seq, " %8lu %8lu %8lu", 0ul, 0ul, 0ul);
496 497 498 499 500 501
		}
		seq_putc(seq, '\n');
	}
	return 0;
}

J
James Morris 已提交
502
static const struct seq_operations ipmr_mfc_seq_ops = {
503
	.start = ipmr_mfc_seq_start,
Y
Yuval Mintz 已提交
504 505
	.next  = mr_mfc_seq_next,
	.stop  = mr_mfc_seq_stop,
506 507 508 509 510
	.show  = ipmr_mfc_seq_show,
};

static int ipmr_mfc_open(struct inode *inode, struct file *file)
{
511
	return seq_open_net(inode, file, &ipmr_mfc_seq_ops,
Y
Yuval Mintz 已提交
512
			    sizeof(struct mr_mfc_iter));
513 514
}

515
static const struct file_operations ip6mr_mfc_fops = {
516 517 518
	.open    = ipmr_mfc_open,
	.read    = seq_read,
	.llseek  = seq_lseek,
519
	.release = seq_release_net,
520 521 522
};
#endif

523 524 525 526 527 528 529
#ifdef CONFIG_IPV6_PIMSM_V2

static int pim6_rcv(struct sk_buff *skb)
{
	struct pimreghdr *pim;
	struct ipv6hdr   *encap;
	struct net_device  *reg_dev = NULL;
530
	struct net *net = dev_net(skb->dev);
531
	struct mr_table *mrt;
532 533 534
	struct flowi6 fl6 = {
		.flowi6_iif	= skb->dev->ifindex,
		.flowi6_mark	= skb->mark,
535 536
	};
	int reg_vif_num;
537 538 539 540 541

	if (!pskb_may_pull(skb, sizeof(*pim) + sizeof(*encap)))
		goto drop;

	pim = (struct pimreghdr *)skb_transport_header(skb);
542
	if (pim->type != ((PIM_VERSION << 4) | PIM_TYPE_REGISTER) ||
543
	    (pim->flags & PIM_NULL_REGISTER) ||
T
Thomas Goff 已提交
544 545 546
	    (csum_ipv6_magic(&ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr,
			     sizeof(*pim), IPPROTO_PIM,
			     csum_partial((void *)pim, sizeof(*pim), 0)) &&
547
	     csum_fold(skb_checksum(skb, 0, skb->len, 0))))
548 549 550 551 552 553 554 555 556 557 558
		goto drop;

	/* check if the inner packet is destined to mcast group */
	encap = (struct ipv6hdr *)(skb_transport_header(skb) +
				   sizeof(*pim));

	if (!ipv6_addr_is_multicast(&encap->daddr) ||
	    encap->payload_len == 0 ||
	    ntohs(encap->payload_len) + sizeof(*pim) > skb->len)
		goto drop;

559
	if (ip6mr_fib_lookup(net, &fl6, &mrt) < 0)
560 561 562
		goto drop;
	reg_vif_num = mrt->mroute_reg_vif_num;

563 564
	read_lock(&mrt_lock);
	if (reg_vif_num >= 0)
565
		reg_dev = mrt->vif_table[reg_vif_num].dev;
566 567 568 569
	if (reg_dev)
		dev_hold(reg_dev);
	read_unlock(&mrt_lock);

570
	if (!reg_dev)
571 572 573 574 575
		goto drop;

	skb->mac_header = skb->network_header;
	skb_pull(skb, (u8 *)encap - skb->data);
	skb_reset_network_header(skb);
T
Thomas Goff 已提交
576
	skb->protocol = htons(ETH_P_IPV6);
577
	skb->ip_summed = CHECKSUM_NONE;
578

579
	skb_tunnel_rx(skb, reg_dev, dev_net(reg_dev));
580

581
	netif_rx(skb);
E
Eric Dumazet 已提交
582

583 584 585 586 587 588 589
	dev_put(reg_dev);
	return 0;
 drop:
	kfree_skb(skb);
	return 0;
}

590
static const struct inet6_protocol pim6_protocol = {
591 592 593 594 595
	.handler	=	pim6_rcv,
};

/* Service routines creating virtual interfaces: PIMREG */

596 597
static netdev_tx_t reg_vif_xmit(struct sk_buff *skb,
				      struct net_device *dev)
598
{
599
	struct net *net = dev_net(dev);
600
	struct mr_table *mrt;
601 602
	struct flowi6 fl6 = {
		.flowi6_oif	= dev->ifindex,
603
		.flowi6_iif	= skb->skb_iif ? : LOOPBACK_IFINDEX,
604
		.flowi6_mark	= skb->mark,
605 606 607
	};
	int err;

608
	err = ip6mr_fib_lookup(net, &fl6, &mrt);
609 610
	if (err < 0) {
		kfree_skb(skb);
611
		return err;
612
	}
613

614
	read_lock(&mrt_lock);
615 616
	dev->stats.tx_bytes += skb->len;
	dev->stats.tx_packets++;
617
	ip6mr_cache_report(mrt, skb, mrt->mroute_reg_vif_num, MRT6MSG_WHOLEPKT);
618 619
	read_unlock(&mrt_lock);
	kfree_skb(skb);
620
	return NETDEV_TX_OK;
621 622
}

623 624 625 626 627
static int reg_vif_get_iflink(const struct net_device *dev)
{
	return 0;
}

628 629
static const struct net_device_ops reg_vif_netdev_ops = {
	.ndo_start_xmit	= reg_vif_xmit,
630
	.ndo_get_iflink = reg_vif_get_iflink,
631 632
};

633 634 635 636 637
static void reg_vif_setup(struct net_device *dev)
{
	dev->type		= ARPHRD_PIMREG;
	dev->mtu		= 1500 - sizeof(struct ipv6hdr) - 8;
	dev->flags		= IFF_NOARP;
638
	dev->netdev_ops		= &reg_vif_netdev_ops;
639
	dev->needs_free_netdev	= true;
T
Tom Goff 已提交
640
	dev->features		|= NETIF_F_NETNS_LOCAL;
641 642
}

643
static struct net_device *ip6mr_reg_vif(struct net *net, struct mr_table *mrt)
644 645
{
	struct net_device *dev;
646 647 648 649 650 651
	char name[IFNAMSIZ];

	if (mrt->id == RT6_TABLE_DFLT)
		sprintf(name, "pim6reg");
	else
		sprintf(name, "pim6reg%u", mrt->id);
652

653
	dev = alloc_netdev(0, name, NET_NAME_UNKNOWN, reg_vif_setup);
654
	if (!dev)
655 656
		return NULL;

657 658
	dev_net_set(dev, net);

659 660 661 662 663 664 665 666
	if (register_netdevice(dev)) {
		free_netdev(dev);
		return NULL;
	}

	if (dev_open(dev))
		goto failure;

W
Wang Chen 已提交
667
	dev_hold(dev);
668 669 670 671 672 673 674 675
	return dev;

failure:
	unregister_netdevice(dev);
	return NULL;
}
#endif

Y
Yuval Mintz 已提交
676 677 678 679 680 681 682 683 684
static int call_ip6mr_vif_entry_notifiers(struct net *net,
					  enum fib_event_type event_type,
					  struct vif_device *vif,
					  mifi_t vif_index, u32 tb_id)
{
	return mr_call_vif_notifiers(net, RTNL_FAMILY_IP6MR, event_type,
				     vif, vif_index, tb_id,
				     &net->ipv6.ipmr_seq);
}
685

Y
Yuval Mintz 已提交
686 687 688 689 690 691 692 693 694
static int call_ip6mr_mfc_entry_notifiers(struct net *net,
					  enum fib_event_type event_type,
					  struct mfc6_cache *mfc, u32 tb_id)
{
	return mr_call_mfc_notifiers(net, RTNL_FAMILY_IP6MR, event_type,
				     &mfc->_c, tb_id, &net->ipv6.ipmr_seq);
}

/* Delete a VIF entry */
695
static int mif6_delete(struct mr_table *mrt, int vifi, int notify,
696
		       struct list_head *head)
697
{
698
	struct vif_device *v;
699
	struct net_device *dev;
T
Thomas Goff 已提交
700
	struct inet6_dev *in6_dev;
701 702

	if (vifi < 0 || vifi >= mrt->maxvif)
703 704
		return -EADDRNOTAVAIL;

705
	v = &mrt->vif_table[vifi];
706

Y
Yuval Mintz 已提交
707 708 709 710 711
	if (VIF_EXISTS(mrt, vifi))
		call_ip6mr_vif_entry_notifiers(read_pnet(&mrt->net),
					       FIB_EVENT_VIF_DEL, v, vifi,
					       mrt->id);

712 713 714 715 716 717 718 719 720
	write_lock_bh(&mrt_lock);
	dev = v->dev;
	v->dev = NULL;

	if (!dev) {
		write_unlock_bh(&mrt_lock);
		return -EADDRNOTAVAIL;
	}

721
#ifdef CONFIG_IPV6_PIMSM_V2
722 723
	if (vifi == mrt->mroute_reg_vif_num)
		mrt->mroute_reg_vif_num = -1;
724 725
#endif

726
	if (vifi + 1 == mrt->maxvif) {
727 728
		int tmp;
		for (tmp = vifi - 1; tmp >= 0; tmp--) {
729
			if (VIF_EXISTS(mrt, tmp))
730 731
				break;
		}
732
		mrt->maxvif = tmp + 1;
733 734 735 736 737 738
	}

	write_unlock_bh(&mrt_lock);

	dev_set_allmulti(dev, -1);

T
Thomas Goff 已提交
739
	in6_dev = __in6_dev_get(dev);
740
	if (in6_dev) {
T
Thomas Goff 已提交
741
		in6_dev->cnf.mc_forwarding--;
742
		inet6_netconf_notify_devconf(dev_net(dev), RTM_NEWNETCONF,
743 744 745
					     NETCONFA_MC_FORWARDING,
					     dev->ifindex, &in6_dev->cnf);
	}
T
Thomas Goff 已提交
746

747
	if ((v->flags & MIFF_REGISTER) && !notify)
748
		unregister_netdevice_queue(dev, head);
749 750 751 752 753

	dev_put(dev);
	return 0;
}

754
static inline void ip6mr_cache_free_rcu(struct rcu_head *head)
755
{
756
	struct mr_mfc *c = container_of(head, struct mr_mfc, rcu);
757

758
	kmem_cache_free(mrt_cachep, (struct mfc6_cache *)c);
759 760
}

761 762
static inline void ip6mr_cache_free(struct mfc6_cache *c)
{
763
	call_rcu(&c->_c.rcu, ip6mr_cache_free_rcu);
764 765
}

766 767 768 769
/* Destroy an unresolved cache entry, killing queued skbs
   and reporting error to netlink readers.
 */

770
static void ip6mr_destroy_unres(struct mr_table *mrt, struct mfc6_cache *c)
771
{
772
	struct net *net = read_pnet(&mrt->net);
773 774
	struct sk_buff *skb;

775
	atomic_dec(&mrt->cache_resolve_queue_len);
776

777
	while ((skb = skb_dequeue(&c->_c.mfc_un.unres.unresolved)) != NULL) {
778
		if (ipv6_hdr(skb)->version == 0) {
779 780
			struct nlmsghdr *nlh = skb_pull(skb,
							sizeof(struct ipv6hdr));
781
			nlh->nlmsg_type = NLMSG_ERROR;
782
			nlh->nlmsg_len = nlmsg_msg_size(sizeof(struct nlmsgerr));
783
			skb_trim(skb, nlh->nlmsg_len);
784
			((struct nlmsgerr *)nlmsg_data(nlh))->error = -ETIMEDOUT;
785
			rtnl_unicast(skb, net, NETLINK_CB(skb).portid);
786 787 788 789
		} else
			kfree_skb(skb);
	}

790
	ip6mr_cache_free(c);
791 792 793
}


794
/* Timer process for all the unresolved queue. */
795

796
static void ipmr_do_expire_process(struct mr_table *mrt)
797 798 799
{
	unsigned long now = jiffies;
	unsigned long expires = 10 * HZ;
800
	struct mr_mfc *c, *next;
801

802
	list_for_each_entry_safe(c, next, &mrt->mfc_unres_queue, list) {
803 804 805 806 807 808 809 810
		if (time_after(c->mfc_un.unres.expires, now)) {
			/* not yet... */
			unsigned long interval = c->mfc_un.unres.expires - now;
			if (interval < expires)
				expires = interval;
			continue;
		}

811
		list_del(&c->list);
812 813
		mr6_netlink_event(mrt, (struct mfc6_cache *)c, RTM_DELROUTE);
		ip6mr_destroy_unres(mrt, (struct mfc6_cache *)c);
814 815
	}

816
	if (!list_empty(&mrt->mfc_unres_queue))
817
		mod_timer(&mrt->ipmr_expire_timer, jiffies + expires);
818 819
}

820
static void ipmr_expire_process(struct timer_list *t)
821
{
822
	struct mr_table *mrt = from_timer(mrt, t, ipmr_expire_timer);
823

824
	if (!spin_trylock(&mfc_unres_lock)) {
825
		mod_timer(&mrt->ipmr_expire_timer, jiffies + 1);
826 827 828
		return;
	}

829
	if (!list_empty(&mrt->mfc_unres_queue))
830
		ipmr_do_expire_process(mrt);
831 832 833 834 835 836

	spin_unlock(&mfc_unres_lock);
}

/* Fill oifs list. It is called under write locked mrt_lock. */

837
static void ip6mr_update_thresholds(struct mr_table *mrt,
838
				    struct mr_mfc *cache,
839
				    unsigned char *ttls)
840 841 842
{
	int vifi;

843
	cache->mfc_un.res.minvif = MAXMIFS;
844
	cache->mfc_un.res.maxvif = 0;
845
	memset(cache->mfc_un.res.ttls, 255, MAXMIFS);
846

847
	for (vifi = 0; vifi < mrt->maxvif; vifi++) {
848
		if (VIF_EXISTS(mrt, vifi) &&
849
		    ttls[vifi] && ttls[vifi] < 255) {
850 851 852 853 854 855 856
			cache->mfc_un.res.ttls[vifi] = ttls[vifi];
			if (cache->mfc_un.res.minvif > vifi)
				cache->mfc_un.res.minvif = vifi;
			if (cache->mfc_un.res.maxvif <= vifi)
				cache->mfc_un.res.maxvif = vifi + 1;
		}
	}
857
	cache->mfc_un.res.lastuse = jiffies;
858 859
}

860
static int mif6_add(struct net *net, struct mr_table *mrt,
861
		    struct mif6ctl *vifc, int mrtsock)
862 863
{
	int vifi = vifc->mif6c_mifi;
864
	struct vif_device *v = &mrt->vif_table[vifi];
865
	struct net_device *dev;
T
Thomas Goff 已提交
866
	struct inet6_dev *in6_dev;
867
	int err;
868 869

	/* Is vif busy ? */
870
	if (VIF_EXISTS(mrt, vifi))
871 872 873
		return -EADDRINUSE;

	switch (vifc->mif6c_flags) {
874 875 876 877 878 879
#ifdef CONFIG_IPV6_PIMSM_V2
	case MIFF_REGISTER:
		/*
		 * Special Purpose VIF in PIM
		 * All the packets will be sent to the daemon
		 */
880
		if (mrt->mroute_reg_vif_num >= 0)
881
			return -EADDRINUSE;
882
		dev = ip6mr_reg_vif(net, mrt);
883 884
		if (!dev)
			return -ENOBUFS;
885 886 887
		err = dev_set_allmulti(dev, 1);
		if (err) {
			unregister_netdevice(dev);
W
Wang Chen 已提交
888
			dev_put(dev);
889 890
			return err;
		}
891 892
		break;
#endif
893
	case 0:
894
		dev = dev_get_by_index(net, vifc->mif6c_pifi);
895 896
		if (!dev)
			return -EADDRNOTAVAIL;
897
		err = dev_set_allmulti(dev, 1);
W
Wang Chen 已提交
898 899
		if (err) {
			dev_put(dev);
900
			return err;
W
Wang Chen 已提交
901
		}
902 903 904 905 906
		break;
	default:
		return -EINVAL;
	}

T
Thomas Goff 已提交
907
	in6_dev = __in6_dev_get(dev);
908
	if (in6_dev) {
T
Thomas Goff 已提交
909
		in6_dev->cnf.mc_forwarding++;
910
		inet6_netconf_notify_devconf(dev_net(dev), RTM_NEWNETCONF,
911 912 913
					     NETCONFA_MC_FORWARDING,
					     dev->ifindex, &in6_dev->cnf);
	}
T
Thomas Goff 已提交
914

915 916 917 918
	/* Fill in the VIF structures */
	vif_device_init(v, dev, vifc->vifc_rate_limit, vifc->vifc_threshold,
			vifc->mif6c_flags | (!mrtsock ? VIFF_STATIC : 0),
			MIFF_REGISTER);
919 920 921 922

	/* And finish update writing critical data */
	write_lock_bh(&mrt_lock);
	v->dev = dev;
923 924
#ifdef CONFIG_IPV6_PIMSM_V2
	if (v->flags & MIFF_REGISTER)
925
		mrt->mroute_reg_vif_num = vifi;
926
#endif
927 928
	if (vifi + 1 > mrt->maxvif)
		mrt->maxvif = vifi + 1;
929
	write_unlock_bh(&mrt_lock);
Y
Yuval Mintz 已提交
930 931
	call_ip6mr_vif_entry_notifiers(net, FIB_EVENT_VIF_ADD,
				       v, vifi, mrt->id);
932 933 934
	return 0;
}

935
static struct mfc6_cache *ip6mr_cache_find(struct mr_table *mrt,
936 937
					   const struct in6_addr *origin,
					   const struct in6_addr *mcastgrp)
938
{
939 940 941 942 943
	struct mfc6_cache_cmp_arg arg = {
		.mf6c_origin = *origin,
		.mf6c_mcastgrp = *mcastgrp,
	};

944
	return mr_mfc_find(mrt, &arg);
945 946 947
}

/* Look for a (*,G) entry */
948
static struct mfc6_cache *ip6mr_cache_find_any(struct mr_table *mrt,
949 950 951
					       struct in6_addr *mcastgrp,
					       mifi_t mifi)
{
952 953 954 955
	struct mfc6_cache_cmp_arg arg = {
		.mf6c_origin = in6addr_any,
		.mf6c_mcastgrp = *mcastgrp,
	};
956 957

	if (ipv6_addr_any(mcastgrp))
958 959
		return mr_mfc_find_any_parent(mrt, mifi);
	return mr_mfc_find_any(mrt, mifi, &arg);
960 961
}

962 963
/* Look for a (S,G,iif) entry if parent != -1 */
static struct mfc6_cache *
964
ip6mr_cache_find_parent(struct mr_table *mrt,
965 966 967 968 969 970 971 972 973
			const struct in6_addr *origin,
			const struct in6_addr *mcastgrp,
			int parent)
{
	struct mfc6_cache_cmp_arg arg = {
		.mf6c_origin = *origin,
		.mf6c_mcastgrp = *mcastgrp,
	};

974
	return mr_mfc_find_parent(mrt, &arg, parent);
975 976
}

977
/* Allocate a multicast cache entry */
978
static struct mfc6_cache *ip6mr_cache_alloc(void)
979
{
980
	struct mfc6_cache *c = kmem_cache_zalloc(mrt_cachep, GFP_KERNEL);
981
	if (!c)
982
		return NULL;
983 984
	c->_c.mfc_un.res.last_assert = jiffies - MFC_ASSERT_THRESH - 1;
	c->_c.mfc_un.res.minvif = MAXMIFS;
985 986 987
	return c;
}

988
static struct mfc6_cache *ip6mr_cache_alloc_unres(void)
989
{
990
	struct mfc6_cache *c = kmem_cache_zalloc(mrt_cachep, GFP_ATOMIC);
991
	if (!c)
992
		return NULL;
993 994
	skb_queue_head_init(&c->_c.mfc_un.unres.unresolved);
	c->_c.mfc_un.unres.expires = jiffies + 10 * HZ;
995 996 997 998 999 1000 1001
	return c;
}

/*
 *	A cache entry has gone into a resolved state from queued
 */

1002
static void ip6mr_cache_resolve(struct net *net, struct mr_table *mrt,
1003
				struct mfc6_cache *uc, struct mfc6_cache *c)
1004 1005 1006 1007 1008 1009 1010
{
	struct sk_buff *skb;

	/*
	 *	Play the pending entries through our router
	 */

1011
	while ((skb = __skb_dequeue(&uc->_c.mfc_un.unres.unresolved))) {
1012
		if (ipv6_hdr(skb)->version == 0) {
1013 1014
			struct nlmsghdr *nlh = skb_pull(skb,
							sizeof(struct ipv6hdr));
1015

Y
Yuval Mintz 已提交
1016 1017
			if (mr_fill_mroute(mrt, skb, &c->_c,
					   nlmsg_data(nlh)) > 0) {
1018
				nlh->nlmsg_len = skb_tail_pointer(skb) - (u8 *)nlh;
1019 1020
			} else {
				nlh->nlmsg_type = NLMSG_ERROR;
1021
				nlh->nlmsg_len = nlmsg_msg_size(sizeof(struct nlmsgerr));
1022
				skb_trim(skb, nlh->nlmsg_len);
1023
				((struct nlmsgerr *)nlmsg_data(nlh))->error = -EMSGSIZE;
1024
			}
1025
			rtnl_unicast(skb, net, NETLINK_CB(skb).portid);
1026
		} else
1027
			ip6_mr_forward(net, mrt, skb, c);
1028 1029 1030 1031
	}
}

/*
1032
 *	Bounce a cache query up to pim6sd and netlink.
1033 1034 1035 1036
 *
 *	Called under mrt_lock.
 */

1037
static int ip6mr_cache_report(struct mr_table *mrt, struct sk_buff *pkt,
1038
			      mifi_t mifi, int assert)
1039
{
Y
Yuval Mintz 已提交
1040
	struct sock *mroute6_sk;
1041 1042 1043 1044
	struct sk_buff *skb;
	struct mrt6msg *msg;
	int ret;

1045 1046 1047 1048 1049 1050 1051
#ifdef CONFIG_IPV6_PIMSM_V2
	if (assert == MRT6MSG_WHOLEPKT)
		skb = skb_realloc_headroom(pkt, -skb_network_offset(pkt)
						+sizeof(*msg));
	else
#endif
		skb = alloc_skb(sizeof(struct ipv6hdr) + sizeof(*msg), GFP_ATOMIC);
1052 1053 1054 1055 1056 1057 1058 1059 1060

	if (!skb)
		return -ENOBUFS;

	/* I suppose that internal messages
	 * do not require checksums */

	skb->ip_summed = CHECKSUM_UNNECESSARY;

1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074
#ifdef CONFIG_IPV6_PIMSM_V2
	if (assert == MRT6MSG_WHOLEPKT) {
		/* Ugly, but we have no choice with this interface.
		   Duplicate old header, fix length etc.
		   And all this only to mangle msg->im6_msgtype and
		   to set msg->im6_mbz to "mbz" :-)
		 */
		skb_push(skb, -skb_network_offset(pkt));

		skb_push(skb, sizeof(*msg));
		skb_reset_transport_header(skb);
		msg = (struct mrt6msg *)skb_transport_header(skb);
		msg->im6_mbz = 0;
		msg->im6_msgtype = MRT6MSG_WHOLEPKT;
1075
		msg->im6_mif = mrt->mroute_reg_vif_num;
1076
		msg->im6_pad = 0;
A
Alexey Dobriyan 已提交
1077 1078
		msg->im6_src = ipv6_hdr(pkt)->saddr;
		msg->im6_dst = ipv6_hdr(pkt)->daddr;
1079 1080 1081 1082 1083

		skb->ip_summed = CHECKSUM_UNNECESSARY;
	} else
#endif
	{
1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100
	/*
	 *	Copy the IP header
	 */

	skb_put(skb, sizeof(struct ipv6hdr));
	skb_reset_network_header(skb);
	skb_copy_to_linear_data(skb, ipv6_hdr(pkt), sizeof(struct ipv6hdr));

	/*
	 *	Add our header
	 */
	skb_put(skb, sizeof(*msg));
	skb_reset_transport_header(skb);
	msg = (struct mrt6msg *)skb_transport_header(skb);

	msg->im6_mbz = 0;
	msg->im6_msgtype = assert;
1101
	msg->im6_mif = mifi;
1102
	msg->im6_pad = 0;
A
Alexey Dobriyan 已提交
1103 1104
	msg->im6_src = ipv6_hdr(pkt)->saddr;
	msg->im6_dst = ipv6_hdr(pkt)->daddr;
1105

E
Eric Dumazet 已提交
1106
	skb_dst_set(skb, dst_clone(skb_dst(pkt)));
1107
	skb->ip_summed = CHECKSUM_UNNECESSARY;
1108
	}
1109

Y
Yuval Mintz 已提交
1110
	rcu_read_lock();
1111
	mroute6_sk = rcu_dereference(mrt->mroute_sk);
Y
Yuval Mintz 已提交
1112 1113
	if (!mroute6_sk) {
		rcu_read_unlock();
1114 1115 1116 1117
		kfree_skb(skb);
		return -EINVAL;
	}

1118 1119
	mrt6msg_netlink_event(mrt, skb);

Y
Yuval Mintz 已提交
1120 1121 1122
	/* Deliver to user space multicast routing algorithms */
	ret = sock_queue_rcv_skb(mroute6_sk, skb);
	rcu_read_unlock();
1123
	if (ret < 0) {
1124
		net_warn_ratelimited("mroute6: pending queue full, dropping entries\n");
1125 1126 1127 1128 1129 1130
		kfree_skb(skb);
	}

	return ret;
}

1131 1132 1133
/* Queue a packet for resolution. It gets locked cache entry! */
static int ip6mr_cache_unresolved(struct mr_table *mrt, mifi_t mifi,
				  struct sk_buff *skb)
1134
{
1135
	struct mfc6_cache *c;
1136
	bool found = false;
1137 1138 1139
	int err;

	spin_lock_bh(&mfc_unres_lock);
1140
	list_for_each_entry(c, &mrt->mfc_unres_queue, _c.list) {
1141
		if (ipv6_addr_equal(&c->mf6c_mcastgrp, &ipv6_hdr(skb)->daddr) &&
1142 1143
		    ipv6_addr_equal(&c->mf6c_origin, &ipv6_hdr(skb)->saddr)) {
			found = true;
1144
			break;
1145
		}
1146 1147
	}

1148
	if (!found) {
1149 1150 1151 1152
		/*
		 *	Create a new entry if allowable
		 */

1153
		if (atomic_read(&mrt->cache_resolve_queue_len) >= 10 ||
1154
		    (c = ip6mr_cache_alloc_unres()) == NULL) {
1155 1156 1157 1158 1159 1160
			spin_unlock_bh(&mfc_unres_lock);

			kfree_skb(skb);
			return -ENOBUFS;
		}

1161 1162
		/* Fill in the new cache entry */
		c->_c.mfc_parent = -1;
1163 1164 1165 1166 1167 1168
		c->mf6c_origin = ipv6_hdr(skb)->saddr;
		c->mf6c_mcastgrp = ipv6_hdr(skb)->daddr;

		/*
		 *	Reflect first query at pim6sd
		 */
1169
		err = ip6mr_cache_report(mrt, skb, mifi, MRT6MSG_NOCACHE);
1170
		if (err < 0) {
1171 1172 1173 1174 1175
			/* If the report failed throw the cache entry
			   out - Brad Parker
			 */
			spin_unlock_bh(&mfc_unres_lock);

1176
			ip6mr_cache_free(c);
1177 1178 1179 1180
			kfree_skb(skb);
			return err;
		}

1181
		atomic_inc(&mrt->cache_resolve_queue_len);
1182
		list_add(&c->_c.list, &mrt->mfc_unres_queue);
1183
		mr6_netlink_event(mrt, c, RTM_NEWROUTE);
1184

1185
		ipmr_do_expire_process(mrt);
1186 1187
	}

1188 1189
	/* See if we can append the packet */
	if (c->_c.mfc_un.unres.unresolved.qlen > 3) {
1190 1191 1192
		kfree_skb(skb);
		err = -ENOBUFS;
	} else {
1193
		skb_queue_tail(&c->_c.mfc_un.unres.unresolved, skb);
1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204
		err = 0;
	}

	spin_unlock_bh(&mfc_unres_lock);
	return err;
}

/*
 *	MFC6 cache manipulation by user space
 */

1205
static int ip6mr_mfc_delete(struct mr_table *mrt, struct mf6cctl *mfc,
1206
			    int parent)
1207
{
1208
	struct mfc6_cache *c;
1209

1210 1211 1212 1213 1214 1215 1216
	/* The entries are added/deleted only under RTNL */
	rcu_read_lock();
	c = ip6mr_cache_find_parent(mrt, &mfc->mf6cc_origin.sin6_addr,
				    &mfc->mf6cc_mcastgrp.sin6_addr, parent);
	rcu_read_unlock();
	if (!c)
		return -ENOENT;
1217 1218
	rhltable_remove(&mrt->mfc_hash, &c->_c.mnode, ip6mr_rht_params);
	list_del_rcu(&c->_c.list);
1219

Y
Yuval Mintz 已提交
1220 1221
	call_ip6mr_mfc_entry_notifiers(read_pnet(&mrt->net),
				       FIB_EVENT_ENTRY_DEL, c, mrt->id);
1222 1223 1224
	mr6_netlink_event(mrt, c, RTM_DELROUTE);
	ip6mr_cache_free(c);
	return 0;
1225 1226 1227 1228 1229
}

static int ip6mr_device_event(struct notifier_block *this,
			      unsigned long event, void *ptr)
{
1230
	struct net_device *dev = netdev_notifier_info_to_dev(ptr);
1231
	struct net *net = dev_net(dev);
1232
	struct mr_table *mrt;
1233
	struct vif_device *v;
1234 1235 1236 1237 1238
	int ct;

	if (event != NETDEV_UNREGISTER)
		return NOTIFY_DONE;

1239
	ip6mr_for_each_table(mrt, net) {
1240
		v = &mrt->vif_table[0];
1241 1242
		for (ct = 0; ct < mrt->maxvif; ct++, v++) {
			if (v->dev == dev)
1243
				mif6_delete(mrt, ct, 1, NULL);
1244
		}
1245
	}
1246

1247 1248 1249
	return NOTIFY_DONE;
}

Y
Yuval Mintz 已提交
1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262
static unsigned int ip6mr_seq_read(struct net *net)
{
	ASSERT_RTNL();

	return net->ipv6.ipmr_seq + ip6mr_rules_seq_read(net);
}

static int ip6mr_dump(struct net *net, struct notifier_block *nb)
{
	return mr_dump(net, nb, RTNL_FAMILY_IP6MR, ip6mr_rules_dump,
		       ip6mr_mr_table_iter, &mrt_lock);
}

1263 1264 1265 1266
static struct notifier_block ip6_mr_notifier = {
	.notifier_call = ip6mr_device_event
};

Y
Yuval Mintz 已提交
1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278
static const struct fib_notifier_ops ip6mr_notifier_ops_template = {
	.family		= RTNL_FAMILY_IP6MR,
	.fib_seq_read	= ip6mr_seq_read,
	.fib_dump	= ip6mr_dump,
	.owner		= THIS_MODULE,
};

static int __net_init ip6mr_notifier_init(struct net *net)
{
	struct fib_notifier_ops *ops;

	net->ipv6.ipmr_seq = 0;
1279

Y
Yuval Mintz 已提交
1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295
	ops = fib_notifier_ops_register(&ip6mr_notifier_ops_template, net);
	if (IS_ERR(ops))
		return PTR_ERR(ops);

	net->ipv6.ip6mr_notifier_ops = ops;

	return 0;
}

static void __net_exit ip6mr_notifier_exit(struct net *net)
{
	fib_notifier_ops_unregister(net->ipv6.ip6mr_notifier_ops);
	net->ipv6.ip6mr_notifier_ops = NULL;
}

/* Setup for IP multicast routing */
1296 1297
static int __net_init ip6mr_net_init(struct net *net)
{
1298
	int err;
1299

Y
Yuval Mintz 已提交
1300 1301 1302 1303
	err = ip6mr_notifier_init(net);
	if (err)
		return err;

1304 1305
	err = ip6mr_rules_init(net);
	if (err < 0)
Y
Yuval Mintz 已提交
1306
		goto ip6mr_rules_fail;
1307 1308 1309

#ifdef CONFIG_PROC_FS
	err = -ENOMEM;
1310
	if (!proc_create("ip6_mr_vif", 0, net->proc_net, &ip6mr_vif_fops))
1311
		goto proc_vif_fail;
1312
	if (!proc_create("ip6_mr_cache", 0, net->proc_net, &ip6mr_mfc_fops))
1313 1314
		goto proc_cache_fail;
#endif
1315

1316 1317
	return 0;

1318 1319
#ifdef CONFIG_PROC_FS
proc_cache_fail:
1320
	remove_proc_entry("ip6_mr_vif", net->proc_net);
1321
proc_vif_fail:
1322
	ip6mr_rules_exit(net);
1323
#endif
Y
Yuval Mintz 已提交
1324 1325
ip6mr_rules_fail:
	ip6mr_notifier_exit(net);
1326 1327 1328 1329 1330
	return err;
}

static void __net_exit ip6mr_net_exit(struct net *net)
{
1331
#ifdef CONFIG_PROC_FS
1332 1333
	remove_proc_entry("ip6_mr_cache", net->proc_net);
	remove_proc_entry("ip6_mr_vif", net->proc_net);
1334
#endif
1335
	ip6mr_rules_exit(net);
Y
Yuval Mintz 已提交
1336
	ip6mr_notifier_exit(net);
1337 1338 1339 1340 1341
}

static struct pernet_operations ip6mr_net_ops = {
	.init = ip6mr_net_init,
	.exit = ip6mr_net_exit,
K
Kirill Tkhai 已提交
1342
	.async = true,
1343 1344
};

W
Wang Chen 已提交
1345
int __init ip6_mr_init(void)
1346
{
W
Wang Chen 已提交
1347 1348
	int err;

1349 1350 1351 1352 1353
	mrt_cachep = kmem_cache_create("ip6_mrt_cache",
				       sizeof(struct mfc6_cache),
				       0, SLAB_HWCACHE_ALIGN,
				       NULL);
	if (!mrt_cachep)
W
Wang Chen 已提交
1354
		return -ENOMEM;
1355

1356 1357 1358 1359
	err = register_pernet_subsys(&ip6mr_net_ops);
	if (err)
		goto reg_pernet_fail;

W
Wang Chen 已提交
1360 1361 1362
	err = register_netdevice_notifier(&ip6_mr_notifier);
	if (err)
		goto reg_notif_fail;
T
Tom Goff 已提交
1363 1364
#ifdef CONFIG_IPV6_PIMSM_V2
	if (inet6_add_protocol(&pim6_protocol, IPPROTO_PIM) < 0) {
1365
		pr_err("%s: can't add PIM protocol\n", __func__);
T
Tom Goff 已提交
1366 1367 1368 1369
		err = -EAGAIN;
		goto add_proto_fail;
	}
#endif
1370 1371 1372 1373 1374
	err = rtnl_register_module(THIS_MODULE, RTNL_FAMILY_IP6MR, RTM_GETROUTE,
				   NULL, ip6mr_rtm_dumproute, 0);
	if (err == 0)
		return 0;

T
Tom Goff 已提交
1375
#ifdef CONFIG_IPV6_PIMSM_V2
1376
	inet6_del_protocol(&pim6_protocol, IPPROTO_PIM);
T
Tom Goff 已提交
1377 1378 1379
add_proto_fail:
	unregister_netdevice_notifier(&ip6_mr_notifier);
#endif
B
Benjamin Thery 已提交
1380
reg_notif_fail:
1381 1382
	unregister_pernet_subsys(&ip6mr_net_ops);
reg_pernet_fail:
B
Benjamin Thery 已提交
1383
	kmem_cache_destroy(mrt_cachep);
W
Wang Chen 已提交
1384
	return err;
1385 1386
}

W
Wang Chen 已提交
1387 1388
void ip6_mr_cleanup(void)
{
1389 1390 1391 1392
	rtnl_unregister(RTNL_FAMILY_IP6MR, RTM_GETROUTE);
#ifdef CONFIG_IPV6_PIMSM_V2
	inet6_del_protocol(&pim6_protocol, IPPROTO_PIM);
#endif
W
Wang Chen 已提交
1393
	unregister_netdevice_notifier(&ip6_mr_notifier);
1394
	unregister_pernet_subsys(&ip6mr_net_ops);
W
Wang Chen 已提交
1395 1396
	kmem_cache_destroy(mrt_cachep);
}
1397

1398
static int ip6mr_mfc_add(struct net *net, struct mr_table *mrt,
1399
			 struct mf6cctl *mfc, int mrtsock, int parent)
1400
{
1401
	unsigned char ttls[MAXMIFS];
1402
	struct mfc6_cache *uc, *c;
1403
	struct mr_mfc *_uc;
1404 1405
	bool found;
	int i, err;
1406

1407 1408 1409
	if (mfc->mf6cc_parent >= MAXMIFS)
		return -ENFILE;

1410 1411
	memset(ttls, 255, MAXMIFS);
	for (i = 0; i < MAXMIFS; i++) {
1412 1413 1414 1415
		if (IF_ISSET(i, &mfc->mf6cc_ifset))
			ttls[i] = 1;
	}

1416 1417 1418 1419 1420 1421
	/* The entries are added/deleted only under RTNL */
	rcu_read_lock();
	c = ip6mr_cache_find_parent(mrt, &mfc->mf6cc_origin.sin6_addr,
				    &mfc->mf6cc_mcastgrp.sin6_addr, parent);
	rcu_read_unlock();
	if (c) {
1422
		write_lock_bh(&mrt_lock);
1423 1424
		c->_c.mfc_parent = mfc->mf6cc_parent;
		ip6mr_update_thresholds(mrt, &c->_c, ttls);
1425
		if (!mrtsock)
1426
			c->_c.mfc_flags |= MFC_STATIC;
1427
		write_unlock_bh(&mrt_lock);
Y
Yuval Mintz 已提交
1428 1429
		call_ip6mr_mfc_entry_notifiers(net, FIB_EVENT_ENTRY_REPLACE,
					       c, mrt->id);
1430
		mr6_netlink_event(mrt, c, RTM_NEWROUTE);
1431 1432 1433
		return 0;
	}

1434 1435
	if (!ipv6_addr_any(&mfc->mf6cc_mcastgrp.sin6_addr) &&
	    !ipv6_addr_is_multicast(&mfc->mf6cc_mcastgrp.sin6_addr))
1436 1437
		return -EINVAL;

1438
	c = ip6mr_cache_alloc();
1439
	if (!c)
1440 1441 1442 1443
		return -ENOMEM;

	c->mf6c_origin = mfc->mf6cc_origin.sin6_addr;
	c->mf6c_mcastgrp = mfc->mf6cc_mcastgrp.sin6_addr;
1444 1445
	c->_c.mfc_parent = mfc->mf6cc_parent;
	ip6mr_update_thresholds(mrt, &c->_c, ttls);
1446
	if (!mrtsock)
1447
		c->_c.mfc_flags |= MFC_STATIC;
1448

1449
	err = rhltable_insert_key(&mrt->mfc_hash, &c->cmparg, &c->_c.mnode,
1450 1451 1452 1453 1454 1455
				  ip6mr_rht_params);
	if (err) {
		pr_err("ip6mr: rhtable insert error %d\n", err);
		ip6mr_cache_free(c);
		return err;
	}
1456
	list_add_tail_rcu(&c->_c.list, &mrt->mfc_cache_list);
1457

1458 1459
	/* Check to see if we resolved a queued list. If so we
	 * need to send on the frames and tidy up.
1460
	 */
1461
	found = false;
1462
	spin_lock_bh(&mfc_unres_lock);
1463 1464
	list_for_each_entry(_uc, &mrt->mfc_unres_queue, list) {
		uc = (struct mfc6_cache *)_uc;
1465
		if (ipv6_addr_equal(&uc->mf6c_origin, &c->mf6c_origin) &&
1466
		    ipv6_addr_equal(&uc->mf6c_mcastgrp, &c->mf6c_mcastgrp)) {
1467
			list_del(&_uc->list);
1468
			atomic_dec(&mrt->cache_resolve_queue_len);
1469
			found = true;
1470 1471 1472
			break;
		}
	}
1473
	if (list_empty(&mrt->mfc_unres_queue))
1474
		del_timer(&mrt->ipmr_expire_timer);
1475 1476
	spin_unlock_bh(&mfc_unres_lock);

1477
	if (found) {
1478
		ip6mr_cache_resolve(net, mrt, uc, c);
1479
		ip6mr_cache_free(uc);
1480
	}
Y
Yuval Mintz 已提交
1481 1482
	call_ip6mr_mfc_entry_notifiers(net, FIB_EVENT_ENTRY_ADD,
				       c, mrt->id);
1483
	mr6_netlink_event(mrt, c, RTM_NEWROUTE);
1484 1485 1486 1487 1488 1489 1490
	return 0;
}

/*
 *	Close the multicast socket, and clear the vif tables etc
 */

1491
static void mroute_clean_tables(struct mr_table *mrt, bool all)
1492
{
1493
	struct mr_mfc *c, *tmp;
1494
	LIST_HEAD(list);
1495
	int i;
1496

1497
	/* Shut down all active vif entries */
1498
	for (i = 0; i < mrt->maxvif; i++) {
1499
		if (!all && (mrt->vif_table[i].flags & VIFF_STATIC))
1500
			continue;
1501
		mif6_delete(mrt, i, 0, &list);
1502
	}
1503
	unregister_netdevice_many(&list);
1504

1505
	/* Wipe the cache */
1506
	list_for_each_entry_safe(c, tmp, &mrt->mfc_cache_list, list) {
1507 1508
		if (!all && (c->mfc_flags & MFC_STATIC))
			continue;
1509
		rhltable_remove(&mrt->mfc_hash, &c->mnode, ip6mr_rht_params);
1510
		list_del_rcu(&c->list);
1511 1512
		mr6_netlink_event(mrt, (struct mfc6_cache *)c, RTM_DELROUTE);
		ip6mr_cache_free((struct mfc6_cache *)c);
1513 1514
	}

1515
	if (atomic_read(&mrt->cache_resolve_queue_len) != 0) {
1516
		spin_lock_bh(&mfc_unres_lock);
1517
		list_for_each_entry_safe(c, tmp, &mrt->mfc_unres_queue, list) {
1518
			list_del(&c->list);
Y
Yuval Mintz 已提交
1519 1520 1521 1522
			call_ip6mr_mfc_entry_notifiers(read_pnet(&mrt->net),
						       FIB_EVENT_ENTRY_DEL,
						       (struct mfc6_cache *)c,
						       mrt->id);
1523 1524 1525
			mr6_netlink_event(mrt, (struct mfc6_cache *)c,
					  RTM_DELROUTE);
			ip6mr_destroy_unres(mrt, (struct mfc6_cache *)c);
1526 1527 1528 1529 1530
		}
		spin_unlock_bh(&mfc_unres_lock);
	}
}

1531
static int ip6mr_sk_init(struct mr_table *mrt, struct sock *sk)
1532 1533
{
	int err = 0;
1534
	struct net *net = sock_net(sk);
1535 1536 1537

	rtnl_lock();
	write_lock_bh(&mrt_lock);
1538
	if (rtnl_dereference(mrt->mroute_sk)) {
1539
		err = -EADDRINUSE;
Y
Yuval Mintz 已提交
1540
	} else {
1541
		rcu_assign_pointer(mrt->mroute_sk, sk);
1542
		sock_set_flag(sk, SOCK_RCU_FREE);
Y
Yuval Mintz 已提交
1543
		net->ipv6.devconf_all->mc_forwarding++;
1544
	}
1545 1546
	write_unlock_bh(&mrt_lock);

1547
	if (!err)
1548 1549
		inet6_netconf_notify_devconf(net, RTM_NEWNETCONF,
					     NETCONFA_MC_FORWARDING,
1550 1551
					     NETCONFA_IFINDEX_ALL,
					     net->ipv6.devconf_all);
1552 1553 1554 1555 1556 1557 1558
	rtnl_unlock();

	return err;
}

int ip6mr_sk_done(struct sock *sk)
{
1559
	int err = -EACCES;
1560
	struct net *net = sock_net(sk);
1561
	struct mr_table *mrt;
1562

1563 1564 1565 1566
	if (sk->sk_type != SOCK_RAW ||
	    inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
		return err;

1567
	rtnl_lock();
1568
	ip6mr_for_each_table(mrt, net) {
1569
		if (sk == rtnl_dereference(mrt->mroute_sk)) {
1570
			write_lock_bh(&mrt_lock);
1571
			RCU_INIT_POINTER(mrt->mroute_sk, NULL);
1572 1573 1574 1575
			/* Note that mroute_sk had SOCK_RCU_FREE set,
			 * so the RCU grace period before sk freeing
			 * is guaranteed by sk_destruct()
			 */
1576
			net->ipv6.devconf_all->mc_forwarding--;
1577
			write_unlock_bh(&mrt_lock);
1578
			inet6_netconf_notify_devconf(net, RTM_NEWNETCONF,
1579 1580 1581
						     NETCONFA_MC_FORWARDING,
						     NETCONFA_IFINDEX_ALL,
						     net->ipv6.devconf_all);
1582

1583
			mroute_clean_tables(mrt, false);
1584 1585 1586 1587
			err = 0;
			break;
		}
	}
1588 1589 1590 1591 1592
	rtnl_unlock();

	return err;
}

Y
Yuval Mintz 已提交
1593
bool mroute6_is_socket(struct net *net, struct sk_buff *skb)
1594
{
1595
	struct mr_table *mrt;
1596
	struct flowi6 fl6 = {
1597
		.flowi6_iif	= skb->skb_iif ? : LOOPBACK_IFINDEX,
1598 1599
		.flowi6_oif	= skb->dev->ifindex,
		.flowi6_mark	= skb->mark,
1600 1601
	};

1602
	if (ip6mr_fib_lookup(net, &fl6, &mrt) < 0)
1603
		return NULL;
1604

1605
	return rcu_access_pointer(mrt->mroute_sk);
1606
}
Y
Yuval Mintz 已提交
1607
EXPORT_SYMBOL(mroute6_is_socket);
1608

1609 1610 1611 1612 1613 1614 1615
/*
 *	Socket options and virtual interface manipulation. The whole
 *	virtual interface system is a complete heap, but unfortunately
 *	that's how BSD mrouted happens to think. Maybe one day with a proper
 *	MOSPF/PIM router set up we can clean this up.
 */

1616
int ip6_mroute_setsockopt(struct sock *sk, int optname, char __user *optval, unsigned int optlen)
1617
{
1618
	int ret, parent = 0;
1619 1620 1621
	struct mif6ctl vif;
	struct mf6cctl mfc;
	mifi_t mifi;
1622
	struct net *net = sock_net(sk);
1623
	struct mr_table *mrt;
1624

1625 1626 1627 1628
	if (sk->sk_type != SOCK_RAW ||
	    inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
		return -EOPNOTSUPP;

1629
	mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
1630
	if (!mrt)
1631
		return -ENOENT;
1632 1633

	if (optname != MRT6_INIT) {
1634
		if (sk != rcu_access_pointer(mrt->mroute_sk) &&
Y
Yuval Mintz 已提交
1635
		    !ns_capable(net->user_ns, CAP_NET_ADMIN))
1636 1637 1638 1639 1640 1641 1642 1643
			return -EACCES;
	}

	switch (optname) {
	case MRT6_INIT:
		if (optlen < sizeof(int))
			return -EINVAL;

1644
		return ip6mr_sk_init(mrt, sk);
1645 1646 1647 1648 1649 1650 1651 1652 1653

	case MRT6_DONE:
		return ip6mr_sk_done(sk);

	case MRT6_ADD_MIF:
		if (optlen < sizeof(vif))
			return -EINVAL;
		if (copy_from_user(&vif, optval, sizeof(vif)))
			return -EFAULT;
1654
		if (vif.mif6c_mifi >= MAXMIFS)
1655 1656
			return -ENFILE;
		rtnl_lock();
Y
Yuval Mintz 已提交
1657
		ret = mif6_add(net, mrt, &vif,
1658
			       sk == rtnl_dereference(mrt->mroute_sk));
1659 1660 1661 1662 1663 1664 1665 1666 1667
		rtnl_unlock();
		return ret;

	case MRT6_DEL_MIF:
		if (optlen < sizeof(mifi_t))
			return -EINVAL;
		if (copy_from_user(&mifi, optval, sizeof(mifi_t)))
			return -EFAULT;
		rtnl_lock();
1668
		ret = mif6_delete(mrt, mifi, 0, NULL);
1669 1670 1671 1672 1673 1674 1675 1676 1677
		rtnl_unlock();
		return ret;

	/*
	 *	Manipulate the forwarding caches. These live
	 *	in a sort of kernel/user symbiosis.
	 */
	case MRT6_ADD_MFC:
	case MRT6_DEL_MFC:
1678
		parent = -1;
1679
		/* fall through */
1680 1681
	case MRT6_ADD_MFC_PROXY:
	case MRT6_DEL_MFC_PROXY:
1682 1683 1684 1685
		if (optlen < sizeof(mfc))
			return -EINVAL;
		if (copy_from_user(&mfc, optval, sizeof(mfc)))
			return -EFAULT;
1686 1687
		if (parent == 0)
			parent = mfc.mf6cc_parent;
1688
		rtnl_lock();
1689 1690
		if (optname == MRT6_DEL_MFC || optname == MRT6_DEL_MFC_PROXY)
			ret = ip6mr_mfc_delete(mrt, &mfc, parent);
1691
		else
1692
			ret = ip6mr_mfc_add(net, mrt, &mfc,
Y
Yuval Mintz 已提交
1693
					    sk ==
1694
					    rtnl_dereference(mrt->mroute_sk),
Y
Yuval Mintz 已提交
1695
					    parent);
1696 1697 1698
		rtnl_unlock();
		return ret;

1699 1700 1701 1702 1703 1704
	/*
	 *	Control PIM assert (to activate pim will activate assert)
	 */
	case MRT6_ASSERT:
	{
		int v;
1705 1706 1707

		if (optlen != sizeof(v))
			return -EINVAL;
1708 1709
		if (get_user(v, (int __user *)optval))
			return -EFAULT;
1710
		mrt->mroute_do_assert = v;
1711 1712 1713 1714 1715 1716
		return 0;
	}

#ifdef CONFIG_IPV6_PIMSM_V2
	case MRT6_PIM:
	{
1717
		int v;
1718 1719 1720

		if (optlen != sizeof(v))
			return -EINVAL;
1721 1722 1723 1724 1725
		if (get_user(v, (int __user *)optval))
			return -EFAULT;
		v = !!v;
		rtnl_lock();
		ret = 0;
1726 1727 1728
		if (v != mrt->mroute_do_pim) {
			mrt->mroute_do_pim = v;
			mrt->mroute_do_assert = v;
1729 1730 1731 1732 1733
		}
		rtnl_unlock();
		return ret;
	}

1734 1735 1736 1737 1738 1739 1740 1741 1742 1743
#endif
#ifdef CONFIG_IPV6_MROUTE_MULTIPLE_TABLES
	case MRT6_TABLE:
	{
		u32 v;

		if (optlen != sizeof(u32))
			return -EINVAL;
		if (get_user(v, (u32 __user *)optval))
			return -EFAULT;
1744 1745 1746
		/* "pim6reg%u" should not exceed 16 bytes (IFNAMSIZ) */
		if (v != RT_TABLE_DEFAULT && v >= 100000000)
			return -EINVAL;
1747
		if (sk == rcu_access_pointer(mrt->mroute_sk))
1748 1749 1750 1751 1752 1753 1754 1755 1756 1757
			return -EBUSY;

		rtnl_lock();
		ret = 0;
		if (!ip6mr_new_table(net, v))
			ret = -ENOMEM;
		raw6_sk(sk)->ip6mr_table = v;
		rtnl_unlock();
		return ret;
	}
1758
#endif
1759
	/*
1760
	 *	Spurious command, or MRT6_VERSION which you cannot
1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776
	 *	set.
	 */
	default:
		return -ENOPROTOOPT;
	}
}

/*
 *	Getsock opt support for the multicast routing system.
 */

int ip6_mroute_getsockopt(struct sock *sk, int optname, char __user *optval,
			  int __user *optlen)
{
	int olr;
	int val;
1777
	struct net *net = sock_net(sk);
1778
	struct mr_table *mrt;
1779

1780 1781 1782 1783
	if (sk->sk_type != SOCK_RAW ||
	    inet_sk(sk)->inet_num != IPPROTO_ICMPV6)
		return -EOPNOTSUPP;

1784
	mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
1785
	if (!mrt)
1786
		return -ENOENT;
1787 1788 1789 1790 1791

	switch (optname) {
	case MRT6_VERSION:
		val = 0x0305;
		break;
1792 1793
#ifdef CONFIG_IPV6_PIMSM_V2
	case MRT6_PIM:
1794
		val = mrt->mroute_do_pim;
1795 1796 1797
		break;
#endif
	case MRT6_ASSERT:
1798
		val = mrt->mroute_do_assert;
1799
		break;
1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825
	default:
		return -ENOPROTOOPT;
	}

	if (get_user(olr, optlen))
		return -EFAULT;

	olr = min_t(int, olr, sizeof(int));
	if (olr < 0)
		return -EINVAL;

	if (put_user(olr, optlen))
		return -EFAULT;
	if (copy_to_user(optval, &val, olr))
		return -EFAULT;
	return 0;
}

/*
 *	The IP multicast ioctl support routines.
 */

int ip6mr_ioctl(struct sock *sk, int cmd, void __user *arg)
{
	struct sioc_sg_req6 sr;
	struct sioc_mif_req6 vr;
1826
	struct vif_device *vif;
1827
	struct mfc6_cache *c;
1828
	struct net *net = sock_net(sk);
1829
	struct mr_table *mrt;
1830 1831

	mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
1832
	if (!mrt)
1833
		return -ENOENT;
1834 1835 1836 1837 1838

	switch (cmd) {
	case SIOCGETMIFCNT_IN6:
		if (copy_from_user(&vr, arg, sizeof(vr)))
			return -EFAULT;
1839
		if (vr.mifi >= mrt->maxvif)
1840 1841
			return -EINVAL;
		read_lock(&mrt_lock);
1842 1843
		vif = &mrt->vif_table[vr.mifi];
		if (VIF_EXISTS(mrt, vr.mifi)) {
1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859
			vr.icount = vif->pkt_in;
			vr.ocount = vif->pkt_out;
			vr.ibytes = vif->bytes_in;
			vr.obytes = vif->bytes_out;
			read_unlock(&mrt_lock);

			if (copy_to_user(arg, &vr, sizeof(vr)))
				return -EFAULT;
			return 0;
		}
		read_unlock(&mrt_lock);
		return -EADDRNOTAVAIL;
	case SIOCGETSGCNT_IN6:
		if (copy_from_user(&sr, arg, sizeof(sr)))
			return -EFAULT;

1860
		rcu_read_lock();
1861
		c = ip6mr_cache_find(mrt, &sr.src.sin6_addr, &sr.grp.sin6_addr);
1862
		if (c) {
1863 1864 1865
			sr.pktcnt = c->_c.mfc_un.res.pkt;
			sr.bytecnt = c->_c.mfc_un.res.bytes;
			sr.wrong_if = c->_c.mfc_un.res.wrong_if;
1866
			rcu_read_unlock();
1867 1868 1869 1870 1871

			if (copy_to_user(arg, &sr, sizeof(sr)))
				return -EFAULT;
			return 0;
		}
1872
		rcu_read_unlock();
1873 1874 1875 1876 1877 1878
		return -EADDRNOTAVAIL;
	default:
		return -ENOIOCTLCMD;
	}
}

1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899
#ifdef CONFIG_COMPAT
struct compat_sioc_sg_req6 {
	struct sockaddr_in6 src;
	struct sockaddr_in6 grp;
	compat_ulong_t pktcnt;
	compat_ulong_t bytecnt;
	compat_ulong_t wrong_if;
};

struct compat_sioc_mif_req6 {
	mifi_t	mifi;
	compat_ulong_t icount;
	compat_ulong_t ocount;
	compat_ulong_t ibytes;
	compat_ulong_t obytes;
};

int ip6mr_compat_ioctl(struct sock *sk, unsigned int cmd, void __user *arg)
{
	struct compat_sioc_sg_req6 sr;
	struct compat_sioc_mif_req6 vr;
1900
	struct vif_device *vif;
1901 1902
	struct mfc6_cache *c;
	struct net *net = sock_net(sk);
1903
	struct mr_table *mrt;
1904 1905

	mrt = ip6mr_get_table(net, raw6_sk(sk)->ip6mr_table ? : RT6_TABLE_DFLT);
1906
	if (!mrt)
1907 1908 1909 1910 1911 1912 1913 1914 1915
		return -ENOENT;

	switch (cmd) {
	case SIOCGETMIFCNT_IN6:
		if (copy_from_user(&vr, arg, sizeof(vr)))
			return -EFAULT;
		if (vr.mifi >= mrt->maxvif)
			return -EINVAL;
		read_lock(&mrt_lock);
1916 1917
		vif = &mrt->vif_table[vr.mifi];
		if (VIF_EXISTS(mrt, vr.mifi)) {
1918 1919 1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933
			vr.icount = vif->pkt_in;
			vr.ocount = vif->pkt_out;
			vr.ibytes = vif->bytes_in;
			vr.obytes = vif->bytes_out;
			read_unlock(&mrt_lock);

			if (copy_to_user(arg, &vr, sizeof(vr)))
				return -EFAULT;
			return 0;
		}
		read_unlock(&mrt_lock);
		return -EADDRNOTAVAIL;
	case SIOCGETSGCNT_IN6:
		if (copy_from_user(&sr, arg, sizeof(sr)))
			return -EFAULT;

1934
		rcu_read_lock();
1935 1936
		c = ip6mr_cache_find(mrt, &sr.src.sin6_addr, &sr.grp.sin6_addr);
		if (c) {
1937 1938 1939
			sr.pktcnt = c->_c.mfc_un.res.pkt;
			sr.bytecnt = c->_c.mfc_un.res.bytes;
			sr.wrong_if = c->_c.mfc_un.res.wrong_if;
1940
			rcu_read_unlock();
1941 1942 1943 1944 1945

			if (copy_to_user(arg, &sr, sizeof(sr)))
				return -EFAULT;
			return 0;
		}
1946
		rcu_read_unlock();
1947 1948 1949 1950 1951 1952
		return -EADDRNOTAVAIL;
	default:
		return -ENOIOCTLCMD;
	}
}
#endif
1953

1954
static inline int ip6mr_forward2_finish(struct net *net, struct sock *sk, struct sk_buff *skb)
1955
{
E
Eric Dumazet 已提交
1956 1957 1958 1959
	__IP6_INC_STATS(net, ip6_dst_idev(skb_dst(skb)),
			IPSTATS_MIB_OUTFORWDATAGRAMS);
	__IP6_ADD_STATS(net, ip6_dst_idev(skb_dst(skb)),
			IPSTATS_MIB_OUTOCTETS, skb->len);
1960
	return dst_output(net, sk, skb);
1961 1962 1963 1964 1965 1966
}

/*
 *	Processing handlers for ip6mr_forward
 */

1967
static int ip6mr_forward2(struct net *net, struct mr_table *mrt,
1968
			  struct sk_buff *skb, struct mfc6_cache *c, int vifi)
1969 1970
{
	struct ipv6hdr *ipv6h;
1971
	struct vif_device *vif = &mrt->vif_table[vifi];
1972 1973
	struct net_device *dev;
	struct dst_entry *dst;
1974
	struct flowi6 fl6;
1975

1976
	if (!vif->dev)
1977 1978
		goto out_free;

1979 1980 1981 1982
#ifdef CONFIG_IPV6_PIMSM_V2
	if (vif->flags & MIFF_REGISTER) {
		vif->pkt_out++;
		vif->bytes_out += skb->len;
1983 1984
		vif->dev->stats.tx_bytes += skb->len;
		vif->dev->stats.tx_packets++;
1985
		ip6mr_cache_report(mrt, skb, vifi, MRT6MSG_WHOLEPKT);
1986
		goto out_free;
1987 1988 1989
	}
#endif

1990 1991
	ipv6h = ipv6_hdr(skb);

1992 1993 1994
	fl6 = (struct flowi6) {
		.flowi6_oif = vif->link,
		.daddr = ipv6h->daddr,
1995 1996
	};

1997
	dst = ip6_route_output(net, NULL, &fl6);
1998 1999
	if (dst->error) {
		dst_release(dst);
2000
		goto out_free;
2001
	}
2002

E
Eric Dumazet 已提交
2003 2004
	skb_dst_drop(skb);
	skb_dst_set(skb, dst);
2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031

	/*
	 * RFC1584 teaches, that DVMRP/PIM router must deliver packets locally
	 * not only before forwarding, but after forwarding on all output
	 * interfaces. It is clear, if mrouter runs a multicasting
	 * program, it should receive packets not depending to what interface
	 * program is joined.
	 * If we will not make it, the program will have to join on all
	 * interfaces. On the other hand, multihoming host (or router, but
	 * not mrouter) cannot join to more than one interface - it will
	 * result in receiving multiple packets.
	 */
	dev = vif->dev;
	skb->dev = dev;
	vif->pkt_out++;
	vif->bytes_out += skb->len;

	/* We are about to write */
	/* XXX: extension headers? */
	if (skb_cow(skb, sizeof(*ipv6h) + LL_RESERVED_SPACE(dev)))
		goto out_free;

	ipv6h = ipv6_hdr(skb);
	ipv6h->hop_limit--;

	IP6CB(skb)->flags |= IP6SKB_FORWARDED;

2032 2033
	return NF_HOOK(NFPROTO_IPV6, NF_INET_FORWARD,
		       net, NULL, skb, skb->dev, dev,
2034 2035 2036 2037 2038 2039 2040
		       ip6mr_forward2_finish);

out_free:
	kfree_skb(skb);
	return 0;
}

2041
static int ip6mr_find_vif(struct mr_table *mrt, struct net_device *dev)
2042 2043
{
	int ct;
2044 2045

	for (ct = mrt->maxvif - 1; ct >= 0; ct--) {
2046
		if (mrt->vif_table[ct].dev == dev)
2047 2048 2049 2050 2051
			break;
	}
	return ct;
}

2052
static void ip6_mr_forward(struct net *net, struct mr_table *mrt,
2053
			   struct sk_buff *skb, struct mfc6_cache *c)
2054 2055 2056
{
	int psend = -1;
	int vif, ct;
2057
	int true_vifi = ip6mr_find_vif(mrt, skb->dev);
2058

2059 2060 2061 2062
	vif = c->_c.mfc_parent;
	c->_c.mfc_un.res.pkt++;
	c->_c.mfc_un.res.bytes += skb->len;
	c->_c.mfc_un.res.lastuse = jiffies;
2063

2064
	if (ipv6_addr_any(&c->mf6c_origin) && true_vifi >= 0) {
2065 2066
		struct mfc6_cache *cache_proxy;

2067
		/* For an (*,G) entry, we only check that the incoming
2068 2069
		 * interface is part of the static tree.
		 */
2070
		rcu_read_lock();
2071
		cache_proxy = mr_mfc_find_any_parent(mrt, vif);
2072
		if (cache_proxy &&
2073
		    cache_proxy->_c.mfc_un.res.ttls[true_vifi] < 255) {
2074
			rcu_read_unlock();
2075
			goto forward;
2076 2077
		}
		rcu_read_unlock();
2078 2079
	}

2080 2081 2082
	/*
	 * Wrong interface: drop packet and (maybe) send PIM assert.
	 */
2083
	if (mrt->vif_table[vif].dev != skb->dev) {
2084
		c->_c.mfc_un.res.wrong_if++;
2085

2086
		if (true_vifi >= 0 && mrt->mroute_do_assert &&
2087 2088 2089 2090 2091
		    /* pimsm uses asserts, when switching from RPT to SPT,
		       so that we cannot check that packet arrived on an oif.
		       It is bad, but otherwise we would need to move pretty
		       large chunk of pimd to kernel. Ough... --ANK
		     */
2092
		    (mrt->mroute_do_pim ||
2093
		     c->_c.mfc_un.res.ttls[true_vifi] < 255) &&
2094
		    time_after(jiffies,
2095 2096 2097
			       c->_c.mfc_un.res.last_assert +
			       MFC_ASSERT_THRESH)) {
			c->_c.mfc_un.res.last_assert = jiffies;
2098
			ip6mr_cache_report(mrt, skb, true_vifi, MRT6MSG_WRONGMIF);
2099 2100 2101 2102
		}
		goto dont_forward;
	}

2103
forward:
2104 2105
	mrt->vif_table[vif].pkt_in++;
	mrt->vif_table[vif].bytes_in += skb->len;
2106 2107 2108 2109

	/*
	 *	Forward the frame
	 */
2110 2111
	if (ipv6_addr_any(&c->mf6c_origin) &&
	    ipv6_addr_any(&c->mf6c_mcastgrp)) {
2112
		if (true_vifi >= 0 &&
2113
		    true_vifi != c->_c.mfc_parent &&
2114
		    ipv6_hdr(skb)->hop_limit >
2115
				c->_c.mfc_un.res.ttls[c->_c.mfc_parent]) {
2116 2117 2118 2119
			/* It's an (*,*) entry and the packet is not coming from
			 * the upstream: forward the packet to the upstream
			 * only.
			 */
2120
			psend = c->_c.mfc_parent;
2121 2122 2123 2124
			goto last_forward;
		}
		goto dont_forward;
	}
2125 2126
	for (ct = c->_c.mfc_un.res.maxvif - 1;
	     ct >= c->_c.mfc_un.res.minvif; ct--) {
2127
		/* For (*,G) entry, don't forward to the incoming interface */
2128 2129
		if ((!ipv6_addr_any(&c->mf6c_origin) || ct != true_vifi) &&
		    ipv6_hdr(skb)->hop_limit > c->_c.mfc_un.res.ttls[ct]) {
2130 2131 2132
			if (psend != -1) {
				struct sk_buff *skb2 = skb_clone(skb, GFP_ATOMIC);
				if (skb2)
2133 2134
					ip6mr_forward2(net, mrt, skb2,
						       c, psend);
2135 2136 2137 2138
			}
			psend = ct;
		}
	}
2139
last_forward:
2140
	if (psend != -1) {
2141
		ip6mr_forward2(net, mrt, skb, c, psend);
2142
		return;
2143 2144
	}

2145
dont_forward:
2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156
	kfree_skb(skb);
}


/*
 *	Multicast packets for forwarding arrive here
 */

int ip6_mr_input(struct sk_buff *skb)
{
	struct mfc6_cache *cache;
2157
	struct net *net = dev_net(skb->dev);
2158
	struct mr_table *mrt;
2159 2160 2161
	struct flowi6 fl6 = {
		.flowi6_iif	= skb->dev->ifindex,
		.flowi6_mark	= skb->mark,
2162 2163 2164
	};
	int err;

2165
	err = ip6mr_fib_lookup(net, &fl6, &mrt);
2166 2167
	if (err < 0) {
		kfree_skb(skb);
2168
		return err;
2169
	}
2170 2171

	read_lock(&mrt_lock);
2172
	cache = ip6mr_cache_find(mrt,
2173
				 &ipv6_hdr(skb)->saddr, &ipv6_hdr(skb)->daddr);
2174
	if (!cache) {
2175 2176 2177 2178 2179 2180 2181
		int vif = ip6mr_find_vif(mrt, skb->dev);

		if (vif >= 0)
			cache = ip6mr_cache_find_any(mrt,
						     &ipv6_hdr(skb)->daddr,
						     vif);
	}
2182 2183 2184 2185

	/*
	 *	No usable cache entry
	 */
2186
	if (!cache) {
2187 2188
		int vif;

2189
		vif = ip6mr_find_vif(mrt, skb->dev);
2190
		if (vif >= 0) {
2191
			int err = ip6mr_cache_unresolved(mrt, vif, skb);
2192 2193 2194 2195 2196 2197 2198 2199 2200
			read_unlock(&mrt_lock);

			return err;
		}
		read_unlock(&mrt_lock);
		kfree_skb(skb);
		return -ENODEV;
	}

2201
	ip6_mr_forward(net, mrt, skb, cache);
2202 2203 2204 2205 2206 2207

	read_unlock(&mrt_lock);

	return 0;
}

2208
int ip6mr_get_route(struct net *net, struct sk_buff *skb, struct rtmsg *rtm,
2209
		    u32 portid)
2210 2211
{
	int err;
2212
	struct mr_table *mrt;
2213
	struct mfc6_cache *cache;
E
Eric Dumazet 已提交
2214
	struct rt6_info *rt = (struct rt6_info *)skb_dst(skb);
2215

2216
	mrt = ip6mr_get_table(net, RT6_TABLE_DFLT);
2217
	if (!mrt)
2218 2219
		return -ENOENT;

2220
	read_lock(&mrt_lock);
2221
	cache = ip6mr_cache_find(mrt, &rt->rt6i_src.addr, &rt->rt6i_dst.addr);
2222 2223 2224 2225 2226 2227 2228
	if (!cache && skb->dev) {
		int vif = ip6mr_find_vif(mrt, skb->dev);

		if (vif >= 0)
			cache = ip6mr_cache_find_any(mrt, &rt->rt6i_dst.addr,
						     vif);
	}
2229 2230 2231 2232 2233 2234 2235 2236

	if (!cache) {
		struct sk_buff *skb2;
		struct ipv6hdr *iph;
		struct net_device *dev;
		int vif;

		dev = skb->dev;
2237
		if (!dev || (vif = ip6mr_find_vif(mrt, dev)) < 0) {
2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248
			read_unlock(&mrt_lock);
			return -ENODEV;
		}

		/* really correct? */
		skb2 = alloc_skb(sizeof(struct ipv6hdr), GFP_ATOMIC);
		if (!skb2) {
			read_unlock(&mrt_lock);
			return -ENOMEM;
		}

2249
		NETLINK_CB(skb2).portid = portid;
2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263
		skb_reset_transport_header(skb2);

		skb_put(skb2, sizeof(struct ipv6hdr));
		skb_reset_network_header(skb2);

		iph = ipv6_hdr(skb2);
		iph->version = 0;
		iph->priority = 0;
		iph->flow_lbl[0] = 0;
		iph->flow_lbl[1] = 0;
		iph->flow_lbl[2] = 0;
		iph->payload_len = 0;
		iph->nexthdr = IPPROTO_NONE;
		iph->hop_limit = 0;
A
Alexey Dobriyan 已提交
2264 2265
		iph->saddr = rt->rt6i_src.addr;
		iph->daddr = rt->rt6i_dst.addr;
2266

2267
		err = ip6mr_cache_unresolved(mrt, vif, skb2);
2268 2269 2270 2271 2272
		read_unlock(&mrt_lock);

		return err;
	}

Y
Yuval Mintz 已提交
2273
	err = mr_fill_mroute(mrt, skb, &cache->_c, rtm);
2274 2275 2276 2277
	read_unlock(&mrt_lock);
	return err;
}

2278
static int ip6mr_fill_mroute(struct mr_table *mrt, struct sk_buff *skb,
2279 2280
			     u32 portid, u32 seq, struct mfc6_cache *c, int cmd,
			     int flags)
2281 2282 2283
{
	struct nlmsghdr *nlh;
	struct rtmsg *rtm;
2284
	int err;
2285

2286
	nlh = nlmsg_put(skb, portid, seq, cmd, sizeof(*rtm), flags);
2287
	if (!nlh)
2288 2289 2290
		return -EMSGSIZE;

	rtm = nlmsg_data(nlh);
2291
	rtm->rtm_family   = RTNL_FAMILY_IP6MR;
2292 2293 2294 2295
	rtm->rtm_dst_len  = 128;
	rtm->rtm_src_len  = 128;
	rtm->rtm_tos      = 0;
	rtm->rtm_table    = mrt->id;
D
David S. Miller 已提交
2296 2297
	if (nla_put_u32(skb, RTA_TABLE, mrt->id))
		goto nla_put_failure;
2298
	rtm->rtm_type = RTN_MULTICAST;
2299
	rtm->rtm_scope    = RT_SCOPE_UNIVERSE;
2300
	if (c->_c.mfc_flags & MFC_STATIC)
2301 2302 2303
		rtm->rtm_protocol = RTPROT_STATIC;
	else
		rtm->rtm_protocol = RTPROT_MROUTED;
2304 2305
	rtm->rtm_flags    = 0;

2306 2307
	if (nla_put_in6_addr(skb, RTA_SRC, &c->mf6c_origin) ||
	    nla_put_in6_addr(skb, RTA_DST, &c->mf6c_mcastgrp))
D
David S. Miller 已提交
2308
		goto nla_put_failure;
Y
Yuval Mintz 已提交
2309
	err = mr_fill_mroute(mrt, skb, &c->_c, rtm);
2310 2311
	/* do not break the dump if cache is unresolved */
	if (err < 0 && err != -ENOENT)
2312 2313
		goto nla_put_failure;

2314 2315
	nlmsg_end(skb, nlh);
	return 0;
2316 2317 2318 2319 2320 2321

nla_put_failure:
	nlmsg_cancel(skb, nlh);
	return -EMSGSIZE;
}

Y
Yuval Mintz 已提交
2322 2323 2324 2325 2326 2327 2328 2329
static int _ip6mr_fill_mroute(struct mr_table *mrt, struct sk_buff *skb,
			      u32 portid, u32 seq, struct mr_mfc *c,
			      int cmd, int flags)
{
	return ip6mr_fill_mroute(mrt, skb, portid, seq, (struct mfc6_cache *)c,
				 cmd, flags);
}

2330 2331 2332 2333 2334 2335 2336 2337 2338 2339 2340 2341 2342 2343 2344
static int mr6_msgsize(bool unresolved, int maxvif)
{
	size_t len =
		NLMSG_ALIGN(sizeof(struct rtmsg))
		+ nla_total_size(4)	/* RTA_TABLE */
		+ nla_total_size(sizeof(struct in6_addr))	/* RTA_SRC */
		+ nla_total_size(sizeof(struct in6_addr))	/* RTA_DST */
		;

	if (!unresolved)
		len = len
		      + nla_total_size(4)	/* RTA_IIF */
		      + nla_total_size(0)	/* RTA_MULTIPATH */
		      + maxvif * NLA_ALIGN(sizeof(struct rtnexthop))
						/* RTA_MFC_STATS */
2345
		      + nla_total_size_64bit(sizeof(struct rta_mfc_stats))
2346 2347 2348 2349 2350
		;

	return len;
}

2351
static void mr6_netlink_event(struct mr_table *mrt, struct mfc6_cache *mfc,
2352 2353 2354 2355 2356 2357
			      int cmd)
{
	struct net *net = read_pnet(&mrt->net);
	struct sk_buff *skb;
	int err = -ENOBUFS;

2358
	skb = nlmsg_new(mr6_msgsize(mfc->_c.mfc_parent >= MAXMIFS, mrt->maxvif),
2359
			GFP_ATOMIC);
2360
	if (!skb)
2361 2362
		goto errout;

2363
	err = ip6mr_fill_mroute(mrt, skb, 0, 0, mfc, cmd, 0);
2364 2365 2366 2367 2368 2369 2370 2371 2372 2373 2374 2375
	if (err < 0)
		goto errout;

	rtnl_notify(skb, net, 0, RTNLGRP_IPV6_MROUTE, NULL, GFP_ATOMIC);
	return;

errout:
	kfree_skb(skb);
	if (err < 0)
		rtnl_set_sk_err(net, RTNLGRP_IPV6_MROUTE, err);
}

2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392
static size_t mrt6msg_netlink_msgsize(size_t payloadlen)
{
	size_t len =
		NLMSG_ALIGN(sizeof(struct rtgenmsg))
		+ nla_total_size(1)	/* IP6MRA_CREPORT_MSGTYPE */
		+ nla_total_size(4)	/* IP6MRA_CREPORT_MIF_ID */
					/* IP6MRA_CREPORT_SRC_ADDR */
		+ nla_total_size(sizeof(struct in6_addr))
					/* IP6MRA_CREPORT_DST_ADDR */
		+ nla_total_size(sizeof(struct in6_addr))
					/* IP6MRA_CREPORT_PKT */
		+ nla_total_size(payloadlen)
		;

	return len;
}

2393
static void mrt6msg_netlink_event(struct mr_table *mrt, struct sk_buff *pkt)
2394 2395 2396 2397 2398 2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425 2426 2427 2428 2429 2430 2431 2432 2433 2434 2435 2436 2437 2438 2439 2440
{
	struct net *net = read_pnet(&mrt->net);
	struct nlmsghdr *nlh;
	struct rtgenmsg *rtgenm;
	struct mrt6msg *msg;
	struct sk_buff *skb;
	struct nlattr *nla;
	int payloadlen;

	payloadlen = pkt->len - sizeof(struct mrt6msg);
	msg = (struct mrt6msg *)skb_transport_header(pkt);

	skb = nlmsg_new(mrt6msg_netlink_msgsize(payloadlen), GFP_ATOMIC);
	if (!skb)
		goto errout;

	nlh = nlmsg_put(skb, 0, 0, RTM_NEWCACHEREPORT,
			sizeof(struct rtgenmsg), 0);
	if (!nlh)
		goto errout;
	rtgenm = nlmsg_data(nlh);
	rtgenm->rtgen_family = RTNL_FAMILY_IP6MR;
	if (nla_put_u8(skb, IP6MRA_CREPORT_MSGTYPE, msg->im6_msgtype) ||
	    nla_put_u32(skb, IP6MRA_CREPORT_MIF_ID, msg->im6_mif) ||
	    nla_put_in6_addr(skb, IP6MRA_CREPORT_SRC_ADDR,
			     &msg->im6_src) ||
	    nla_put_in6_addr(skb, IP6MRA_CREPORT_DST_ADDR,
			     &msg->im6_dst))
		goto nla_put_failure;

	nla = nla_reserve(skb, IP6MRA_CREPORT_PKT, payloadlen);
	if (!nla || skb_copy_bits(pkt, sizeof(struct mrt6msg),
				  nla_data(nla), payloadlen))
		goto nla_put_failure;

	nlmsg_end(skb, nlh);

	rtnl_notify(skb, net, 0, RTNLGRP_IPV6_MROUTE_R, NULL, GFP_ATOMIC);
	return;

nla_put_failure:
	nlmsg_cancel(skb, nlh);
errout:
	kfree_skb(skb);
	rtnl_set_sk_err(net, RTNLGRP_IPV6_MROUTE_R, -ENOBUFS);
}

2441 2442
static int ip6mr_rtm_dumproute(struct sk_buff *skb, struct netlink_callback *cb)
{
Y
Yuval Mintz 已提交
2443 2444
	return mr_rtm_dumproute(skb, cb, ip6mr_mr_table_iter,
				_ip6mr_fill_mroute, &mfc_unres_lock);
2445
}