cls_rsvp.h 14.8 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

L
Linus Torvalds 已提交
133 134 135
static int rsvp_classify(struct sk_buff *skb, struct tcf_proto *tp,
			 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 170
	xprt = ((u8 *)nhptr) + (nhptr->ihl<<2);
	if (nhptr->frag_off & htons(IP_MF | IP_OFFSET))
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) },
};

L
Linus Torvalds 已提交
419 420
static int rsvp_change(struct tcf_proto *tp, unsigned long base,
		       u32 handle,
421
		       struct nlattr **tca,
L
Linus Torvalds 已提交
422 423 424 425 426 427
		       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;
428 429
	struct nlattr *opt = tca[TCA_OPTIONS-1];
	struct nlattr *tb[TCA_RSVP_MAX + 1];
L
Linus Torvalds 已提交
430
	struct tcf_exts e;
E
Eric Dumazet 已提交
431
	unsigned int h1, h2;
A
Al Viro 已提交
432
	__be32 *dst;
L
Linus Torvalds 已提交
433 434 435 436 437
	int err;

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

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

	err = tcf_exts_validate(tp, tb, tca[TCA_RATE-1], &e, &rsvp_ext_map);
	if (err < 0)
		return err;

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

		if (f->handle != handle && handle)
			goto errout2;
		if (tb[TCA_RSVP_CLASSID-1]) {
453
			f->res.classid = nla_get_u32(tb[TCA_RSVP_CLASSID-1]);
L
Linus Torvalds 已提交
454 455 456 457 458 459 460 461 462 463 464 465 466 467 468
			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;
	if (tb[TCA_RSVP_DST-1] == NULL)
		goto errout2;

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

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

486
	dst = nla_data(tb[TCA_RSVP_DST-1]);
L
Linus Torvalds 已提交
487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503
	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 已提交
504
	for (sp = &data->ht[h1]; (s = *sp) != NULL; sp = &s->next) {
L
Linus Torvalds 已提交
505 506
		if (dst[RSVP_DST_LEN-1] == s->dst[RSVP_DST_LEN-1] &&
		    pinfo && pinfo->protocol == s->protocol &&
507
		    memcmp(&pinfo->dpi, &s->dpi, sizeof(s->dpi)) == 0 &&
L
Linus Torvalds 已提交
508
#if RSVP_DST_LEN == 4
509 510 511
		    dst[0] == s->dst[0] &&
		    dst[1] == s->dst[1] &&
		    dst[2] == s->dst[2] &&
L
Linus Torvalds 已提交
512
#endif
513
		    pinfo->tunnelid == s->tunnelid) {
L
Linus Torvalds 已提交
514 515 516 517 518 519 520 521 522 523 524 525 526

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 已提交
527
				if (((*fp)->spi.mask & f->spi.mask) != f->spi.mask)
L
Linus Torvalds 已提交
528 529 530 531 532 533 534 535 536 537 538 539 540
					break;
			f->next = *fp;
			wmb();
			*fp = f;

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

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

	err = -ENOBUFS;
541
	s = kzalloc(sizeof(struct rsvp_session), GFP_KERNEL);
L
Linus Torvalds 已提交
542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557
	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;
558

L
Linus Torvalds 已提交
559 560 561
	goto insert;

errout:
J
Jesper Juhl 已提交
562
	kfree(f);
L
Linus Torvalds 已提交
563 564 565 566 567 568 569 570
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 已提交
571
	unsigned int h, h1;
L
Linus Torvalds 已提交
572 573 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

	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 已提交
602
	struct rsvp_filter *f = (struct rsvp_filter *)fh;
L
Linus Torvalds 已提交
603
	struct rsvp_session *s;
604
	unsigned char *b = skb_tail_pointer(skb);
605
	struct nlattr *nest;
L
Linus Torvalds 已提交
606 607 608 609 610 611 612 613
	struct tc_rsvp_pinfo pinfo;

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

	t->tcm_handle = f->handle;

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

618
	NLA_PUT(skb, TCA_RSVP_DST, sizeof(s->dst), &s->dst);
L
Linus Torvalds 已提交
619 620 621 622 623
	pinfo.dpi = s->dpi;
	pinfo.spi = f->spi;
	pinfo.protocol = s->protocol;
	pinfo.tunnelid = s->tunnelid;
	pinfo.tunnelhdr = f->tunnelhdr;
624
	pinfo.pad = 0;
625
	NLA_PUT(skb, TCA_RSVP_PINFO, sizeof(pinfo), &pinfo);
L
Linus Torvalds 已提交
626
	if (f->res.classid)
627
		NLA_PUT_U32(skb, TCA_RSVP_CLASSID, f->res.classid);
E
Eric Dumazet 已提交
628
	if (((f->handle >> 8) & 0xFF) != 16)
629
		NLA_PUT(skb, TCA_RSVP_SRC, sizeof(f->src), f->src);
L
Linus Torvalds 已提交
630 631

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

634
	nla_nest_end(skb, nest);
L
Linus Torvalds 已提交
635 636

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

640
nla_put_failure:
641
	nlmsg_trim(skb, b);
L
Linus Torvalds 已提交
642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664
	return -1;
}

static struct tcf_proto_ops RSVP_OPS = {
	.next		=	NULL,
	.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);
}

665
static void __exit exit_rsvp(void)
L
Linus Torvalds 已提交
666 667 668 669 670 671
{
	unregister_tcf_proto_ops(&RSVP_OPS);
}

module_init(init_rsvp)
module_exit(exit_rsvp)