cls_rsvp.h 15.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
/*
 * net/sched/cls_rsvp.h	Template file for RSVPv[46] classifiers.
 *
 *		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.
 *
 * Authors:	Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
 */

/*
   Comparing to general packet classification problem,
   RSVP needs only sevaral relatively simple rules:

   * (dst, protocol) are always specified,
     so that we are able to hash them.
   * src may be exact, or may be wildcard, so that
     we can keep a hash table plus one wildcard entry.
   * source port (or flow label) is important only if src is given.

   IMPLEMENTATION.

   We use a two level hash table: The top level is keyed by
   destination address and protocol ID, every bucket contains a list
   of "rsvp sessions", identified by destination address, protocol and
   DPI(="Destination Port ID"): triple (key, mask, offset).

   Every bucket has a smaller hash table keyed by source address
   (cf. RSVP flowspec) and one wildcard entry for wildcard reservations.
   Every bucket is again a list of "RSVP flows", selected by
   source address and SPI(="Source Port ID" here rather than
   "security parameter index"): triple (key, mask, offset).


   NOTE 1. All the packets with IPv6 extension headers (but AH and ESP)
   and all fragmented packets go to the best-effort traffic class.


   NOTE 2. Two "port id"'s seems to be redundant, rfc2207 requires
   only one "Generalized Port Identifier". So that for classic
   ah, esp (and udp,tcp) both *pi should coincide or one of them
   should be wildcard.

   At first sight, this redundancy is just a waste of CPU
   resources. But DPI and SPI add the possibility to assign different
   priorities to GPIs. Look also at note 4 about tunnels below.


   NOTE 3. One complication is the case of tunneled packets.
   We implement it as following: if the first lookup
   matches a special session with "tunnelhdr" value not zero,
   flowid doesn't contain the true flow ID, but the tunnel ID (1...255).
   In this case, we pull tunnelhdr bytes and restart lookup
   with tunnel ID added to the list of keys. Simple and stupid 8)8)
   It's enough for PIMREG and IPIP.


   NOTE 4. Two GPIs make it possible to parse even GRE packets.
   F.e. DPI can select ETH_P_IP (and necessary flags to make
   tunnelhdr correct) in GRE protocol field and SPI matches
   GRE key. Is it not nice? 8)8)


   Well, as result, despite its simplicity, we get a pretty
   powerful classification engine.  */


E
Eric Dumazet 已提交
69
struct rsvp_head {
L
Linus Torvalds 已提交
70 71 72 73 74 75
	u32			tmap[256/32];
	u32			hgenerator;
	u8			tgenerator;
	struct rsvp_session	*ht[256];
};

E
Eric Dumazet 已提交
76
struct rsvp_session {
L
Linus Torvalds 已提交
77
	struct rsvp_session	*next;
A
Al Viro 已提交
78
	__be32			dst[RSVP_DST_LEN];
L
Linus Torvalds 已提交
79 80 81 82
	struct tc_rsvp_gpi 	dpi;
	u8			protocol;
	u8			tunnelid;
	/* 16 (src,sport) hash slots, and one wildcard source slot */
E
Eric Dumazet 已提交
83
	struct rsvp_filter	*ht[16 + 1];
L
Linus Torvalds 已提交
84 85 86
};


E
Eric Dumazet 已提交
87
struct rsvp_filter {
L
Linus Torvalds 已提交
88
	struct rsvp_filter	*next;
A
Al Viro 已提交
89
	__be32			src[RSVP_DST_LEN];
L
Linus Torvalds 已提交
90 91 92 93 94 95 96 97 98 99
	struct tc_rsvp_gpi	spi;
	u8			tunnelhdr;

	struct tcf_result	res;
	struct tcf_exts		exts;

	u32			handle;
	struct rsvp_session	*sess;
};

E
Eric Dumazet 已提交
100
static inline unsigned int hash_dst(__be32 *dst, u8 protocol, u8 tunnelid)
L
Linus Torvalds 已提交
101
{
E
Eric Dumazet 已提交
102 103
	unsigned int h = (__force __u32)dst[RSVP_DST_LEN - 1];

L
Linus Torvalds 已提交
104 105 106 107 108
	h ^= h>>16;
	h ^= h>>8;
	return (h ^ protocol ^ tunnelid) & 0xFF;
}

E
Eric Dumazet 已提交
109
static inline unsigned int hash_src(__be32 *src)
L
Linus Torvalds 已提交
110
{
E
Eric Dumazet 已提交
111 112
	unsigned int h = (__force __u32)src[RSVP_DST_LEN-1];

L
Linus Torvalds 已提交
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
	h ^= h>>16;
	h ^= h>>8;
	h ^= h>>4;
	return h & 0xF;
}

static struct tcf_ext_map rsvp_ext_map = {
	.police = TCA_RSVP_POLICE,
	.action = TCA_RSVP_ACT
};

#define RSVP_APPLY_RESULT()				\
{							\
	int r = tcf_exts_exec(skb, &f->exts, res);	\
	if (r < 0)					\
		continue;				\
	else if (r > 0)					\
		return r;				\
}
132

133
static int rsvp_classify(struct sk_buff *skb, const struct tcf_proto *tp,
L
Linus Torvalds 已提交
134 135
			 struct tcf_result *res)
{
E
Eric Dumazet 已提交
136
	struct rsvp_session **sht = ((struct rsvp_head *)tp->root)->ht;
L
Linus Torvalds 已提交
137 138
	struct rsvp_session *s;
	struct rsvp_filter *f;
E
Eric Dumazet 已提交
139
	unsigned int h1, h2;
A
Al Viro 已提交
140
	__be32 *dst, *src;
L
Linus Torvalds 已提交
141 142 143 144
	u8 protocol;
	u8 tunnelid = 0;
	u8 *xprt;
#if RSVP_DST_LEN == 4
145 146 147 148 149
	struct ipv6hdr *nhptr;

	if (!pskb_network_may_pull(skb, sizeof(*nhptr)))
		return -1;
	nhptr = ipv6_hdr(skb);
L
Linus Torvalds 已提交
150
#else
151 152 153 154 155
	struct iphdr *nhptr;

	if (!pskb_network_may_pull(skb, sizeof(*nhptr)))
		return -1;
	nhptr = ip_hdr(skb);
L
Linus Torvalds 已提交
156 157 158 159 160 161 162 163
#endif

restart:

#if RSVP_DST_LEN == 4
	src = &nhptr->saddr.s6_addr32[0];
	dst = &nhptr->daddr.s6_addr32[0];
	protocol = nhptr->nexthdr;
E
Eric Dumazet 已提交
164
	xprt = ((u8 *)nhptr) + sizeof(struct ipv6hdr);
L
Linus Torvalds 已提交
165 166 167 168
#else
	src = &nhptr->saddr;
	dst = &nhptr->daddr;
	protocol = nhptr->protocol;
E
Eric Dumazet 已提交
169
	xprt = ((u8 *)nhptr) + (nhptr->ihl<<2);
170
	if (ip_is_fragment(nhptr))
L
Linus Torvalds 已提交
171 172 173 174 175 176 177
		return -1;
#endif

	h1 = hash_dst(dst, protocol, tunnelid);
	h2 = hash_src(src);

	for (s = sht[h1]; s; s = s->next) {
E
Eric Dumazet 已提交
178
		if (dst[RSVP_DST_LEN-1] == s->dst[RSVP_DST_LEN - 1] &&
L
Linus Torvalds 已提交
179
		    protocol == s->protocol &&
180
		    !(s->dpi.mask &
E
Eric Dumazet 已提交
181
		      (*(u32 *)(xprt + s->dpi.offset) ^ s->dpi.key)) &&
L
Linus Torvalds 已提交
182
#if RSVP_DST_LEN == 4
183 184 185
		    dst[0] == s->dst[0] &&
		    dst[1] == s->dst[1] &&
		    dst[2] == s->dst[2] &&
L
Linus Torvalds 已提交
186
#endif
187
		    tunnelid == s->tunnelid) {
L
Linus Torvalds 已提交
188 189

			for (f = s->ht[h2]; f; f = f->next) {
E
Eric Dumazet 已提交
190 191
				if (src[RSVP_DST_LEN-1] == f->src[RSVP_DST_LEN - 1] &&
				    !(f->spi.mask & (*(u32 *)(xprt + f->spi.offset) ^ f->spi.key))
L
Linus Torvalds 已提交
192
#if RSVP_DST_LEN == 4
193 194 195 196
				    &&
				    src[0] == f->src[0] &&
				    src[1] == f->src[1] &&
				    src[2] == f->src[2]
L
Linus Torvalds 已提交
197 198 199 200 201 202 203 204 205 206
#endif
				    ) {
					*res = f->res;
					RSVP_APPLY_RESULT();

matched:
					if (f->tunnelhdr == 0)
						return 0;

					tunnelid = f->res.classid;
E
Eric Dumazet 已提交
207
					nhptr = (void *)(xprt + f->tunnelhdr - sizeof(*nhptr));
L
Linus Torvalds 已提交
208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
					goto restart;
				}
			}

			/* And wildcard bucket... */
			for (f = s->ht[16]; f; f = f->next) {
				*res = f->res;
				RSVP_APPLY_RESULT();
				goto matched;
			}
			return -1;
		}
	}
	return -1;
}

static unsigned long rsvp_get(struct tcf_proto *tp, u32 handle)
{
E
Eric Dumazet 已提交
226
	struct rsvp_session **sht = ((struct rsvp_head *)tp->root)->ht;
L
Linus Torvalds 已提交
227 228
	struct rsvp_session *s;
	struct rsvp_filter *f;
E
Eric Dumazet 已提交
229 230
	unsigned int h1 = handle & 0xFF;
	unsigned int h2 = (handle >> 8) & 0xFF;
L
Linus Torvalds 已提交
231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251

	if (h2 > 16)
		return 0;

	for (s = sht[h1]; s; s = s->next) {
		for (f = s->ht[h2]; f; f = f->next) {
			if (f->handle == handle)
				return (unsigned long)f;
		}
	}
	return 0;
}

static void rsvp_put(struct tcf_proto *tp, unsigned long f)
{
}

static int rsvp_init(struct tcf_proto *tp)
{
	struct rsvp_head *data;

252
	data = kzalloc(sizeof(struct rsvp_head), GFP_KERNEL);
L
Linus Torvalds 已提交
253 254 255 256 257 258 259
	if (data) {
		tp->root = data;
		return 0;
	}
	return -ENOBUFS;
}

E
Eric Dumazet 已提交
260
static void
L
Linus Torvalds 已提交
261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
rsvp_delete_filter(struct tcf_proto *tp, struct rsvp_filter *f)
{
	tcf_unbind_filter(tp, &f->res);
	tcf_exts_destroy(tp, &f->exts);
	kfree(f);
}

static void rsvp_destroy(struct tcf_proto *tp)
{
	struct rsvp_head *data = xchg(&tp->root, NULL);
	struct rsvp_session **sht;
	int h1, h2;

	if (data == NULL)
		return;

	sht = data->ht;

E
Eric Dumazet 已提交
279
	for (h1 = 0; h1 < 256; h1++) {
L
Linus Torvalds 已提交
280 281 282 283 284
		struct rsvp_session *s;

		while ((s = sht[h1]) != NULL) {
			sht[h1] = s->next;

E
Eric Dumazet 已提交
285
			for (h2 = 0; h2 <= 16; h2++) {
L
Linus Torvalds 已提交
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300
				struct rsvp_filter *f;

				while ((f = s->ht[h2]) != NULL) {
					s->ht[h2] = f->next;
					rsvp_delete_filter(tp, f);
				}
			}
			kfree(s);
		}
	}
	kfree(data);
}

static int rsvp_delete(struct tcf_proto *tp, unsigned long arg)
{
E
Eric Dumazet 已提交
301 302
	struct rsvp_filter **fp, *f = (struct rsvp_filter *)arg;
	unsigned int h = f->handle;
L
Linus Torvalds 已提交
303 304 305 306
	struct rsvp_session **sp;
	struct rsvp_session *s = f->sess;
	int i;

E
Eric Dumazet 已提交
307
	for (fp = &s->ht[(h >> 8) & 0xFF]; *fp; fp = &(*fp)->next) {
L
Linus Torvalds 已提交
308 309 310 311 312 313 314 315
		if (*fp == f) {
			tcf_tree_lock(tp);
			*fp = f->next;
			tcf_tree_unlock(tp);
			rsvp_delete_filter(tp, f);

			/* Strip tree */

E
Eric Dumazet 已提交
316
			for (i = 0; i <= 16; i++)
L
Linus Torvalds 已提交
317 318 319 320
				if (s->ht[i])
					return 0;

			/* OK, session has no flows */
E
Eric Dumazet 已提交
321
			for (sp = &((struct rsvp_head *)tp->root)->ht[h & 0xFF];
L
Linus Torvalds 已提交
322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338
			     *sp; sp = &(*sp)->next) {
				if (*sp == s) {
					tcf_tree_lock(tp);
					*sp = s->next;
					tcf_tree_unlock(tp);

					kfree(s);
					return 0;
				}
			}

			return 0;
		}
	}
	return 0;
}

E
Eric Dumazet 已提交
339
static unsigned int gen_handle(struct tcf_proto *tp, unsigned salt)
L
Linus Torvalds 已提交
340 341 342 343 344 345
{
	struct rsvp_head *data = tp->root;
	int i = 0xFFFF;

	while (i-- > 0) {
		u32 h;
E
Eric Dumazet 已提交
346

L
Linus Torvalds 已提交
347 348 349 350 351 352 353 354 355 356 357
		if ((data->hgenerator += 0x10000) == 0)
			data->hgenerator = 0x10000;
		h = data->hgenerator|salt;
		if (rsvp_get(tp, h) == 0)
			return h;
	}
	return 0;
}

static int tunnel_bts(struct rsvp_head *data)
{
E
Eric Dumazet 已提交
358 359
	int n = data->tgenerator >> 5;
	u32 b = 1 << (data->tgenerator & 0x1F);
360

E
Eric Dumazet 已提交
361
	if (data->tmap[n] & b)
L
Linus Torvalds 已提交
362 363 364 365 366 367 368 369 370 371 372 373 374
		return 0;
	data->tmap[n] |= b;
	return 1;
}

static void tunnel_recycle(struct rsvp_head *data)
{
	struct rsvp_session **sht = data->ht;
	u32 tmap[256/32];
	int h1, h2;

	memset(tmap, 0, sizeof(tmap));

E
Eric Dumazet 已提交
375
	for (h1 = 0; h1 < 256; h1++) {
L
Linus Torvalds 已提交
376 377
		struct rsvp_session *s;
		for (s = sht[h1]; s; s = s->next) {
E
Eric Dumazet 已提交
378
			for (h2 = 0; h2 <= 16; h2++) {
L
Linus Torvalds 已提交
379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397
				struct rsvp_filter *f;

				for (f = s->ht[h2]; f; f = f->next) {
					if (f->tunnelhdr == 0)
						continue;
					data->tgenerator = f->res.classid;
					tunnel_bts(data);
				}
			}
		}
	}

	memcpy(data->tmap, tmap, sizeof(tmap));
}

static u32 gen_tunnel(struct rsvp_head *data)
{
	int i, k;

E
Eric Dumazet 已提交
398 399
	for (k = 0; k < 2; k++) {
		for (i = 255; i > 0; i--) {
L
Linus Torvalds 已提交
400 401 402 403 404 405 406 407 408 409
			if (++data->tgenerator == 0)
				data->tgenerator = 1;
			if (tunnel_bts(data))
				return data->tgenerator;
		}
		tunnel_recycle(data);
	}
	return 0;
}

410 411 412 413 414 415 416 417 418
static const struct nla_policy rsvp_policy[TCA_RSVP_MAX + 1] = {
	[TCA_RSVP_CLASSID]	= { .type = NLA_U32 },
	[TCA_RSVP_DST]		= { .type = NLA_BINARY,
				    .len = RSVP_DST_LEN * sizeof(u32) },
	[TCA_RSVP_SRC]		= { .type = NLA_BINARY,
				    .len = RSVP_DST_LEN * sizeof(u32) },
	[TCA_RSVP_PINFO]	= { .len = sizeof(struct tc_rsvp_pinfo) },
};

419
static int rsvp_change(struct net *net, struct sk_buff *in_skb,
420
		       struct tcf_proto *tp, unsigned long base,
L
Linus Torvalds 已提交
421
		       u32 handle,
422
		       struct nlattr **tca,
L
Linus Torvalds 已提交
423 424 425 426 427 428
		       unsigned long *arg)
{
	struct rsvp_head *data = tp->root;
	struct rsvp_filter *f, **fp;
	struct rsvp_session *s, **sp;
	struct tc_rsvp_pinfo *pinfo = NULL;
429
	struct nlattr *opt = tca[TCA_OPTIONS];
430
	struct nlattr *tb[TCA_RSVP_MAX + 1];
L
Linus Torvalds 已提交
431
	struct tcf_exts e;
E
Eric Dumazet 已提交
432
	unsigned int h1, h2;
A
Al Viro 已提交
433
	__be32 *dst;
L
Linus Torvalds 已提交
434 435 436 437 438
	int err;

	if (opt == NULL)
		return handle ? -EINVAL : 0;

439
	err = nla_parse_nested(tb, TCA_RSVP_MAX, opt, rsvp_policy);
440 441
	if (err < 0)
		return err;
L
Linus Torvalds 已提交
442

443
	err = tcf_exts_validate(net, tp, tb, tca[TCA_RATE], &e, &rsvp_ext_map);
L
Linus Torvalds 已提交
444 445 446
	if (err < 0)
		return err;

E
Eric Dumazet 已提交
447 448
	f = (struct rsvp_filter *)*arg;
	if (f) {
L
Linus Torvalds 已提交
449 450 451 452
		/* Node exists: adjust only classid */

		if (f->handle != handle && handle)
			goto errout2;
453 454
		if (tb[TCA_RSVP_CLASSID]) {
			f->res.classid = nla_get_u32(tb[TCA_RSVP_CLASSID]);
L
Linus Torvalds 已提交
455 456 457 458 459 460 461 462 463 464 465
			tcf_bind_filter(tp, &f->res, base);
		}

		tcf_exts_change(tp, &f->exts, &e);
		return 0;
	}

	/* Now more serious part... */
	err = -EINVAL;
	if (handle)
		goto errout2;
466
	if (tb[TCA_RSVP_DST] == NULL)
L
Linus Torvalds 已提交
467 468 469
		goto errout2;

	err = -ENOBUFS;
470
	f = kzalloc(sizeof(struct rsvp_filter), GFP_KERNEL);
L
Linus Torvalds 已提交
471 472 473
	if (f == NULL)
		goto errout2;

474
	tcf_exts_init(&f->exts);
L
Linus Torvalds 已提交
475
	h2 = 16;
476 477
	if (tb[TCA_RSVP_SRC]) {
		memcpy(f->src, nla_data(tb[TCA_RSVP_SRC]), sizeof(f->src));
L
Linus Torvalds 已提交
478 479
		h2 = hash_src(f->src);
	}
480 481
	if (tb[TCA_RSVP_PINFO]) {
		pinfo = nla_data(tb[TCA_RSVP_PINFO]);
L
Linus Torvalds 已提交
482 483 484
		f->spi = pinfo->spi;
		f->tunnelhdr = pinfo->tunnelhdr;
	}
485 486
	if (tb[TCA_RSVP_CLASSID])
		f->res.classid = nla_get_u32(tb[TCA_RSVP_CLASSID]);
L
Linus Torvalds 已提交
487

488
	dst = nla_data(tb[TCA_RSVP_DST]);
L
Linus Torvalds 已提交
489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505
	h1 = hash_dst(dst, pinfo ? pinfo->protocol : 0, pinfo ? pinfo->tunnelid : 0);

	err = -ENOMEM;
	if ((f->handle = gen_handle(tp, h1 | (h2<<8))) == 0)
		goto errout;

	if (f->tunnelhdr) {
		err = -EINVAL;
		if (f->res.classid > 255)
			goto errout;

		err = -ENOMEM;
		if (f->res.classid == 0 &&
		    (f->res.classid = gen_tunnel(data)) == 0)
			goto errout;
	}

E
Eric Dumazet 已提交
506
	for (sp = &data->ht[h1]; (s = *sp) != NULL; sp = &s->next) {
L
Linus Torvalds 已提交
507 508
		if (dst[RSVP_DST_LEN-1] == s->dst[RSVP_DST_LEN-1] &&
		    pinfo && pinfo->protocol == s->protocol &&
509
		    memcmp(&pinfo->dpi, &s->dpi, sizeof(s->dpi)) == 0 &&
L
Linus Torvalds 已提交
510
#if RSVP_DST_LEN == 4
511 512 513
		    dst[0] == s->dst[0] &&
		    dst[1] == s->dst[1] &&
		    dst[2] == s->dst[2] &&
L
Linus Torvalds 已提交
514
#endif
515
		    pinfo->tunnelid == s->tunnelid) {
L
Linus Torvalds 已提交
516 517 518 519 520 521 522 523 524 525 526 527 528

insert:
			/* OK, we found appropriate session */

			fp = &s->ht[h2];

			f->sess = s;
			if (f->tunnelhdr == 0)
				tcf_bind_filter(tp, &f->res, base);

			tcf_exts_change(tp, &f->exts, &e);

			for (fp = &s->ht[h2]; *fp; fp = &(*fp)->next)
E
Eric Dumazet 已提交
529
				if (((*fp)->spi.mask & f->spi.mask) != f->spi.mask)
L
Linus Torvalds 已提交
530 531 532 533 534 535 536 537 538 539 540 541 542
					break;
			f->next = *fp;
			wmb();
			*fp = f;

			*arg = (unsigned long)f;
			return 0;
		}
	}

	/* No session found. Create new one. */

	err = -ENOBUFS;
543
	s = kzalloc(sizeof(struct rsvp_session), GFP_KERNEL);
L
Linus Torvalds 已提交
544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559
	if (s == NULL)
		goto errout;
	memcpy(s->dst, dst, sizeof(s->dst));

	if (pinfo) {
		s->dpi = pinfo->dpi;
		s->protocol = pinfo->protocol;
		s->tunnelid = pinfo->tunnelid;
	}
	for (sp = &data->ht[h1]; *sp; sp = &(*sp)->next) {
		if (((*sp)->dpi.mask&s->dpi.mask) != s->dpi.mask)
			break;
	}
	s->next = *sp;
	wmb();
	*sp = s;
560

L
Linus Torvalds 已提交
561 562 563
	goto insert;

errout:
J
Jesper Juhl 已提交
564
	kfree(f);
L
Linus Torvalds 已提交
565 566 567 568 569 570 571 572
errout2:
	tcf_exts_destroy(tp, &e);
	return err;
}

static void rsvp_walk(struct tcf_proto *tp, struct tcf_walker *arg)
{
	struct rsvp_head *head = tp->root;
E
Eric Dumazet 已提交
573
	unsigned int h, h1;
L
Linus Torvalds 已提交
574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603

	if (arg->stop)
		return;

	for (h = 0; h < 256; h++) {
		struct rsvp_session *s;

		for (s = head->ht[h]; s; s = s->next) {
			for (h1 = 0; h1 <= 16; h1++) {
				struct rsvp_filter *f;

				for (f = s->ht[h1]; f; f = f->next) {
					if (arg->count < arg->skip) {
						arg->count++;
						continue;
					}
					if (arg->fn(tp, (unsigned long)f, arg) < 0) {
						arg->stop = 1;
						return;
					}
					arg->count++;
				}
			}
		}
	}
}

static int rsvp_dump(struct tcf_proto *tp, unsigned long fh,
		     struct sk_buff *skb, struct tcmsg *t)
{
E
Eric Dumazet 已提交
604
	struct rsvp_filter *f = (struct rsvp_filter *)fh;
L
Linus Torvalds 已提交
605
	struct rsvp_session *s;
606
	unsigned char *b = skb_tail_pointer(skb);
607
	struct nlattr *nest;
L
Linus Torvalds 已提交
608 609 610 611 612 613 614 615
	struct tc_rsvp_pinfo pinfo;

	if (f == NULL)
		return skb->len;
	s = f->sess;

	t->tcm_handle = f->handle;

616 617 618
	nest = nla_nest_start(skb, TCA_OPTIONS);
	if (nest == NULL)
		goto nla_put_failure;
L
Linus Torvalds 已提交
619

620 621
	if (nla_put(skb, TCA_RSVP_DST, sizeof(s->dst), &s->dst))
		goto nla_put_failure;
L
Linus Torvalds 已提交
622 623 624 625 626
	pinfo.dpi = s->dpi;
	pinfo.spi = f->spi;
	pinfo.protocol = s->protocol;
	pinfo.tunnelid = s->tunnelid;
	pinfo.tunnelhdr = f->tunnelhdr;
627
	pinfo.pad = 0;
628 629 630 631 632 633 634 635
	if (nla_put(skb, TCA_RSVP_PINFO, sizeof(pinfo), &pinfo))
		goto nla_put_failure;
	if (f->res.classid &&
	    nla_put_u32(skb, TCA_RSVP_CLASSID, f->res.classid))
		goto nla_put_failure;
	if (((f->handle >> 8) & 0xFF) != 16 &&
	    nla_put(skb, TCA_RSVP_SRC, sizeof(f->src), f->src))
		goto nla_put_failure;
L
Linus Torvalds 已提交
636 637

	if (tcf_exts_dump(skb, &f->exts, &rsvp_ext_map) < 0)
638
		goto nla_put_failure;
L
Linus Torvalds 已提交
639

640
	nla_nest_end(skb, nest);
L
Linus Torvalds 已提交
641 642

	if (tcf_exts_dump_stats(skb, &f->exts, &rsvp_ext_map) < 0)
643
		goto nla_put_failure;
L
Linus Torvalds 已提交
644 645
	return skb->len;

646
nla_put_failure:
647
	nlmsg_trim(skb, b);
L
Linus Torvalds 已提交
648 649 650
	return -1;
}

651
static struct tcf_proto_ops RSVP_OPS __read_mostly = {
L
Linus Torvalds 已提交
652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669
	.kind		=	RSVP_ID,
	.classify	=	rsvp_classify,
	.init		=	rsvp_init,
	.destroy	=	rsvp_destroy,
	.get		=	rsvp_get,
	.put		=	rsvp_put,
	.change		=	rsvp_change,
	.delete		=	rsvp_delete,
	.walk		=	rsvp_walk,
	.dump		=	rsvp_dump,
	.owner		=	THIS_MODULE,
};

static int __init init_rsvp(void)
{
	return register_tcf_proto_ops(&RSVP_OPS);
}

670
static void __exit exit_rsvp(void)
L
Linus Torvalds 已提交
671 672 673 674 675 676
{
	unregister_tcf_proto_ops(&RSVP_OPS);
}

module_init(init_rsvp)
module_exit(exit_rsvp)