Linux Audio

Check our new training course

Loading...
v6.2
  1// SPDX-License-Identifier: GPL-2.0+
  2/*
  3 * Author: Justin Iurman (justin.iurman@uliege.be)
  4 *
  5 * IOAM tester for IPv6, see ioam6.sh for details on each test case.
  6 */
  7#include <arpa/inet.h>
  8#include <errno.h>
  9#include <limits.h>
 10#include <linux/const.h>
 11#include <linux/if_ether.h>
 12#include <linux/ioam6.h>
 13#include <linux/ipv6.h>
 
 14#include <stdlib.h>
 15#include <string.h>
 16#include <unistd.h>
 17
 18struct ioam_config {
 19	__u32 id;
 20	__u64 wide;
 21	__u16 ingr_id;
 22	__u16 egr_id;
 23	__u32 ingr_wide;
 24	__u32 egr_wide;
 25	__u32 ns_data;
 26	__u64 ns_wide;
 27	__u32 sc_id;
 28	__u8 hlim;
 29	char *sc_data;
 30};
 31
 32/*
 33 * Be careful if you modify structs below - everything MUST be kept synchronized
 34 * with configurations inside ioam6.sh and always reflect the same.
 35 */
 36
 37static struct ioam_config node1 = {
 38	.id = 1,
 39	.wide = 11111111,
 40	.ingr_id = 0xffff, /* default value */
 41	.egr_id = 101,
 42	.ingr_wide = 0xffffffff, /* default value */
 43	.egr_wide = 101101,
 44	.ns_data = 0xdeadbee0,
 45	.ns_wide = 0xcafec0caf00dc0de,
 46	.sc_id = 777,
 47	.sc_data = "something that will be 4n-aligned",
 48	.hlim = 64,
 49};
 50
 51static struct ioam_config node2 = {
 52	.id = 2,
 53	.wide = 22222222,
 54	.ingr_id = 201,
 55	.egr_id = 202,
 56	.ingr_wide = 201201,
 57	.egr_wide = 202202,
 58	.ns_data = 0xdeadbee1,
 59	.ns_wide = 0xcafec0caf11dc0de,
 60	.sc_id = 666,
 61	.sc_data = "Hello there -Obi",
 62	.hlim = 63,
 63};
 64
 65static struct ioam_config node3 = {
 66	.id = 3,
 67	.wide = 33333333,
 68	.ingr_id = 301,
 69	.egr_id = 0xffff, /* default value */
 70	.ingr_wide = 301301,
 71	.egr_wide = 0xffffffff, /* default value */
 72	.ns_data = 0xdeadbee2,
 73	.ns_wide = 0xcafec0caf22dc0de,
 74	.sc_id = 0xffffff, /* default value */
 75	.sc_data = NULL,
 76	.hlim = 62,
 77};
 78
 79enum {
 80	/**********
 81	 * OUTPUT *
 82	 **********/
 
 
 83	TEST_OUT_UNDEF_NS,
 84	TEST_OUT_NO_ROOM,
 
 85	TEST_OUT_BIT0,
 86	TEST_OUT_BIT1,
 87	TEST_OUT_BIT2,
 88	TEST_OUT_BIT3,
 89	TEST_OUT_BIT4,
 90	TEST_OUT_BIT5,
 91	TEST_OUT_BIT6,
 92	TEST_OUT_BIT7,
 93	TEST_OUT_BIT8,
 94	TEST_OUT_BIT9,
 95	TEST_OUT_BIT10,
 96	TEST_OUT_BIT11,
 97	TEST_OUT_BIT22,
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 98	TEST_OUT_FULL_SUPP_TRACE,
 99
 
 
100	/*********
101	 * INPUT *
102	 *********/
 
 
103	TEST_IN_UNDEF_NS,
104	TEST_IN_NO_ROOM,
 
 
105	TEST_IN_OFLAG,
106	TEST_IN_BIT0,
107	TEST_IN_BIT1,
108	TEST_IN_BIT2,
109	TEST_IN_BIT3,
110	TEST_IN_BIT4,
111	TEST_IN_BIT5,
112	TEST_IN_BIT6,
113	TEST_IN_BIT7,
114	TEST_IN_BIT8,
115	TEST_IN_BIT9,
116	TEST_IN_BIT10,
117	TEST_IN_BIT11,
118	TEST_IN_BIT22,
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
119	TEST_IN_FULL_SUPP_TRACE,
120
121	/**********
122	 * GLOBAL *
123	 **********/
124	TEST_FWD_FULL_SUPP_TRACE,
125
126	__TEST_MAX,
127};
128
129static int check_ioam_header(int tid, struct ioam6_trace_hdr *ioam6h,
130			     __u32 trace_type, __u16 ioam_ns)
131{
132	if (__be16_to_cpu(ioam6h->namespace_id) != ioam_ns ||
133	    __be32_to_cpu(ioam6h->type_be32) != (trace_type << 8))
134		return 1;
135
136	switch (tid) {
137	case TEST_OUT_UNDEF_NS:
138	case TEST_IN_UNDEF_NS:
139		return ioam6h->overflow ||
140		       ioam6h->nodelen != 1 ||
141		       ioam6h->remlen != 1;
 
142
143	case TEST_OUT_NO_ROOM:
144	case TEST_IN_NO_ROOM:
145	case TEST_IN_OFLAG:
146		return !ioam6h->overflow ||
147		       ioam6h->nodelen != 2 ||
148		       ioam6h->remlen != 1;
 
 
 
 
 
 
 
 
 
 
 
 
149
150	case TEST_OUT_BIT0:
151	case TEST_IN_BIT0:
152	case TEST_OUT_BIT1:
153	case TEST_IN_BIT1:
154	case TEST_OUT_BIT2:
155	case TEST_IN_BIT2:
156	case TEST_OUT_BIT3:
157	case TEST_IN_BIT3:
158	case TEST_OUT_BIT4:
159	case TEST_IN_BIT4:
160	case TEST_OUT_BIT5:
161	case TEST_IN_BIT5:
162	case TEST_OUT_BIT6:
163	case TEST_IN_BIT6:
164	case TEST_OUT_BIT7:
165	case TEST_IN_BIT7:
166	case TEST_OUT_BIT11:
167	case TEST_IN_BIT11:
168		return ioam6h->overflow ||
169		       ioam6h->nodelen != 1 ||
170		       ioam6h->remlen;
171
172	case TEST_OUT_BIT8:
173	case TEST_IN_BIT8:
174	case TEST_OUT_BIT9:
175	case TEST_IN_BIT9:
176	case TEST_OUT_BIT10:
177	case TEST_IN_BIT10:
178		return ioam6h->overflow ||
179		       ioam6h->nodelen != 2 ||
180		       ioam6h->remlen;
181
182	case TEST_OUT_BIT22:
183	case TEST_IN_BIT22:
184		return ioam6h->overflow ||
185		       ioam6h->nodelen ||
186		       ioam6h->remlen;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
187
188	case TEST_OUT_FULL_SUPP_TRACE:
189	case TEST_IN_FULL_SUPP_TRACE:
190	case TEST_FWD_FULL_SUPP_TRACE:
191		return ioam6h->overflow ||
192		       ioam6h->nodelen != 15 ||
193		       ioam6h->remlen;
194
195	default:
196		break;
197	}
198
199	return 1;
200}
201
202static int check_ioam6_data(__u8 **p, struct ioam6_trace_hdr *ioam6h,
203			    const struct ioam_config cnf)
204{
205	unsigned int len;
206	__u8 aligned;
207	__u64 raw64;
208	__u32 raw32;
 
209
210	if (ioam6h->type.bit0) {
211		raw32 = __be32_to_cpu(*((__u32 *)*p));
212		if (cnf.hlim != (raw32 >> 24) || cnf.id != (raw32 & 0xffffff))
213			return 1;
214		*p += sizeof(__u32);
215	}
216
217	if (ioam6h->type.bit1) {
218		raw32 = __be32_to_cpu(*((__u32 *)*p));
219		if (cnf.ingr_id != (raw32 >> 16) ||
220		    cnf.egr_id != (raw32 & 0xffff))
221			return 1;
222		*p += sizeof(__u32);
223	}
224
225	if (ioam6h->type.bit2)
226		*p += sizeof(__u32);
227
228	if (ioam6h->type.bit3)
229		*p += sizeof(__u32);
230
231	if (ioam6h->type.bit4) {
232		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
233			return 1;
234		*p += sizeof(__u32);
235	}
236
237	if (ioam6h->type.bit5) {
238		if (__be32_to_cpu(*((__u32 *)*p)) != cnf.ns_data)
239			return 1;
240		*p += sizeof(__u32);
241	}
242
243	if (ioam6h->type.bit6)
244		*p += sizeof(__u32);
245
246	if (ioam6h->type.bit7) {
247		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
248			return 1;
249		*p += sizeof(__u32);
250	}
251
252	if (ioam6h->type.bit8) {
253		raw64 = __be64_to_cpu(*((__u64 *)*p));
254		if (cnf.hlim != (raw64 >> 56) ||
255		    cnf.wide != (raw64 & 0xffffffffffffff))
256			return 1;
257		*p += sizeof(__u64);
258	}
259
260	if (ioam6h->type.bit9) {
261		if (__be32_to_cpu(*((__u32 *)*p)) != cnf.ingr_wide)
262			return 1;
263		*p += sizeof(__u32);
264
265		if (__be32_to_cpu(*((__u32 *)*p)) != cnf.egr_wide)
 
 
266			return 1;
267		*p += sizeof(__u32);
268	}
269
270	if (ioam6h->type.bit10) {
271		if (__be64_to_cpu(*((__u64 *)*p)) != cnf.ns_wide)
 
 
272			return 1;
273		*p += sizeof(__u64);
274	}
275
276	if (ioam6h->type.bit11) {
277		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
 
 
278			return 1;
279		*p += sizeof(__u32);
280	}
281
282	if (ioam6h->type.bit12) {
283		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
 
 
284			return 1;
285		*p += sizeof(__u32);
286	}
287
288	if (ioam6h->type.bit13) {
289		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
290			return 1;
291		*p += sizeof(__u32);
292	}
293
294	if (ioam6h->type.bit14) {
295		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
296			return 1;
297		*p += sizeof(__u32);
298	}
299
300	if (ioam6h->type.bit15) {
301		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
302			return 1;
303		*p += sizeof(__u32);
304	}
305
306	if (ioam6h->type.bit16) {
307		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
308			return 1;
309		*p += sizeof(__u32);
310	}
311
312	if (ioam6h->type.bit17) {
313		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
 
 
314			return 1;
315		*p += sizeof(__u32);
316	}
317
318	if (ioam6h->type.bit18) {
319		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
320			return 1;
321		*p += sizeof(__u32);
322	}
323
324	if (ioam6h->type.bit19) {
325		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
326			return 1;
327		*p += sizeof(__u32);
328	}
329
330	if (ioam6h->type.bit20) {
331		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
332			return 1;
333		*p += sizeof(__u32);
334	}
335
336	if (ioam6h->type.bit21) {
337		if (__be32_to_cpu(*((__u32 *)*p)) != 0xffffffff)
338			return 1;
339		*p += sizeof(__u32);
340	}
341
342	if (ioam6h->type.bit22) {
343		len = cnf.sc_data ? strlen(cnf.sc_data) : 0;
344		aligned = cnf.sc_data ? __ALIGN_KERNEL(len, 4) : 0;
345
346		raw32 = __be32_to_cpu(*((__u32 *)*p));
347		if (aligned != (raw32 >> 24) * 4 ||
348		    cnf.sc_id != (raw32 & 0xffffff))
349			return 1;
350		*p += sizeof(__u32);
351
352		if (cnf.sc_data) {
353			if (strncmp((char *)*p, cnf.sc_data, len))
354				return 1;
355
356			*p += len;
357			aligned -= len;
358
359			while (aligned--) {
360				if (**p != '\0')
361					return 1;
362				*p += sizeof(__u8);
363			}
364		}
365	}
366
367	return 0;
368}
369
370static int check_ioam_header_and_data(int tid, struct ioam6_trace_hdr *ioam6h,
371				      __u32 trace_type, __u16 ioam_ns)
372{
373	__u8 *p;
374
375	if (check_ioam_header(tid, ioam6h, trace_type, ioam_ns))
376		return 1;
377
378	p = ioam6h->data + ioam6h->remlen * 4;
 
379
380	switch (tid) {
381	case TEST_OUT_BIT0:
382	case TEST_OUT_BIT1:
383	case TEST_OUT_BIT2:
384	case TEST_OUT_BIT3:
385	case TEST_OUT_BIT4:
386	case TEST_OUT_BIT5:
387	case TEST_OUT_BIT6:
388	case TEST_OUT_BIT7:
389	case TEST_OUT_BIT8:
390	case TEST_OUT_BIT9:
391	case TEST_OUT_BIT10:
392	case TEST_OUT_BIT11:
393	case TEST_OUT_BIT22:
394	case TEST_OUT_FULL_SUPP_TRACE:
395		return check_ioam6_data(&p, ioam6h, node1);
396
397	case TEST_IN_BIT0:
398	case TEST_IN_BIT1:
399	case TEST_IN_BIT2:
400	case TEST_IN_BIT3:
401	case TEST_IN_BIT4:
402	case TEST_IN_BIT5:
403	case TEST_IN_BIT6:
404	case TEST_IN_BIT7:
405	case TEST_IN_BIT8:
406	case TEST_IN_BIT9:
407	case TEST_IN_BIT10:
408	case TEST_IN_BIT11:
409	case TEST_IN_BIT22:
410	case TEST_IN_FULL_SUPP_TRACE:
411	{
412		__u32 tmp32 = node2.egr_wide;
413		__u16 tmp16 = node2.egr_id;
414		int res;
415
416		node2.egr_id = 0xffff;
417		node2.egr_wide = 0xffffffff;
418
419		res = check_ioam6_data(&p, ioam6h, node2);
420
421		node2.egr_id = tmp16;
422		node2.egr_wide = tmp32;
423
424		return res;
425	}
426
427	case TEST_FWD_FULL_SUPP_TRACE:
428		if (check_ioam6_data(&p, ioam6h, node3))
429			return 1;
430		if (check_ioam6_data(&p, ioam6h, node2))
431			return 1;
432		return check_ioam6_data(&p, ioam6h, node1);
433
434	default:
435		break;
436	}
437
438	return 1;
439}
440
441static int str2id(const char *tname)
442{
443	if (!strcmp("out_undef_ns", tname))
444		return TEST_OUT_UNDEF_NS;
445	if (!strcmp("out_no_room", tname))
446		return TEST_OUT_NO_ROOM;
447	if (!strcmp("out_bit0", tname))
 
 
448		return TEST_OUT_BIT0;
449	if (!strcmp("out_bit1", tname))
450		return TEST_OUT_BIT1;
451	if (!strcmp("out_bit2", tname))
452		return TEST_OUT_BIT2;
453	if (!strcmp("out_bit3", tname))
454		return TEST_OUT_BIT3;
455	if (!strcmp("out_bit4", tname))
456		return TEST_OUT_BIT4;
457	if (!strcmp("out_bit5", tname))
458		return TEST_OUT_BIT5;
459	if (!strcmp("out_bit6", tname))
460		return TEST_OUT_BIT6;
461	if (!strcmp("out_bit7", tname))
462		return TEST_OUT_BIT7;
463	if (!strcmp("out_bit8", tname))
464		return TEST_OUT_BIT8;
465	if (!strcmp("out_bit9", tname))
466		return TEST_OUT_BIT9;
467	if (!strcmp("out_bit10", tname))
468		return TEST_OUT_BIT10;
469	if (!strcmp("out_bit11", tname))
470		return TEST_OUT_BIT11;
471	if (!strcmp("out_bit22", tname))
472		return TEST_OUT_BIT22;
473	if (!strcmp("out_full_supp_trace", tname))
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
474		return TEST_OUT_FULL_SUPP_TRACE;
475	if (!strcmp("in_undef_ns", tname))
476		return TEST_IN_UNDEF_NS;
477	if (!strcmp("in_no_room", tname))
478		return TEST_IN_NO_ROOM;
479	if (!strcmp("in_oflag", tname))
 
 
 
 
480		return TEST_IN_OFLAG;
481	if (!strcmp("in_bit0", tname))
482		return TEST_IN_BIT0;
483	if (!strcmp("in_bit1", tname))
484		return TEST_IN_BIT1;
485	if (!strcmp("in_bit2", tname))
486		return TEST_IN_BIT2;
487	if (!strcmp("in_bit3", tname))
488		return TEST_IN_BIT3;
489	if (!strcmp("in_bit4", tname))
490		return TEST_IN_BIT4;
491	if (!strcmp("in_bit5", tname))
492		return TEST_IN_BIT5;
493	if (!strcmp("in_bit6", tname))
494		return TEST_IN_BIT6;
495	if (!strcmp("in_bit7", tname))
496		return TEST_IN_BIT7;
497	if (!strcmp("in_bit8", tname))
498		return TEST_IN_BIT8;
499	if (!strcmp("in_bit9", tname))
500		return TEST_IN_BIT9;
501	if (!strcmp("in_bit10", tname))
502		return TEST_IN_BIT10;
503	if (!strcmp("in_bit11", tname))
504		return TEST_IN_BIT11;
505	if (!strcmp("in_bit22", tname))
506		return TEST_IN_BIT22;
507	if (!strcmp("in_full_supp_trace", tname))
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
508		return TEST_IN_FULL_SUPP_TRACE;
509	if (!strcmp("fwd_full_supp_trace", tname))
510		return TEST_FWD_FULL_SUPP_TRACE;
511
512	return -1;
513}
514
515static int ipv6_addr_equal(const struct in6_addr *a1, const struct in6_addr *a2)
516{
517	return ((a1->s6_addr32[0] ^ a2->s6_addr32[0]) |
518		(a1->s6_addr32[1] ^ a2->s6_addr32[1]) |
519		(a1->s6_addr32[2] ^ a2->s6_addr32[2]) |
520		(a1->s6_addr32[3] ^ a2->s6_addr32[3])) == 0;
521}
522
523static int get_u32(__u32 *val, const char *arg, int base)
524{
525	unsigned long res;
526	char *ptr;
527
528	if (!arg || !*arg)
529		return -1;
530	res = strtoul(arg, &ptr, base);
531
532	if (!ptr || ptr == arg || *ptr)
533		return -1;
534
535	if (res == ULONG_MAX && errno == ERANGE)
536		return -1;
537
538	if (res > 0xFFFFFFFFUL)
539		return -1;
540
541	*val = res;
542	return 0;
543}
544
545static int get_u16(__u16 *val, const char *arg, int base)
546{
547	unsigned long res;
548	char *ptr;
549
550	if (!arg || !*arg)
551		return -1;
552	res = strtoul(arg, &ptr, base);
553
554	if (!ptr || ptr == arg || *ptr)
555		return -1;
556
557	if (res == ULONG_MAX && errno == ERANGE)
558		return -1;
559
560	if (res > 0xFFFFUL)
561		return -1;
562
563	*val = res;
564	return 0;
565}
566
567static int (*func[__TEST_MAX])(int, struct ioam6_trace_hdr *, __u32, __u16) = {
568	[TEST_OUT_UNDEF_NS]		= check_ioam_header,
569	[TEST_OUT_NO_ROOM]		= check_ioam_header,
570	[TEST_OUT_BIT0]		= check_ioam_header_and_data,
571	[TEST_OUT_BIT1]		= check_ioam_header_and_data,
572	[TEST_OUT_BIT2]		= check_ioam_header_and_data,
573	[TEST_OUT_BIT3]		= check_ioam_header_and_data,
574	[TEST_OUT_BIT4]		= check_ioam_header_and_data,
575	[TEST_OUT_BIT5]		= check_ioam_header_and_data,
576	[TEST_OUT_BIT6]		= check_ioam_header_and_data,
577	[TEST_OUT_BIT7]		= check_ioam_header_and_data,
578	[TEST_OUT_BIT8]		= check_ioam_header_and_data,
579	[TEST_OUT_BIT9]		= check_ioam_header_and_data,
580	[TEST_OUT_BIT10]		= check_ioam_header_and_data,
581	[TEST_OUT_BIT11]		= check_ioam_header_and_data,
582	[TEST_OUT_BIT22]		= check_ioam_header_and_data,
583	[TEST_OUT_FULL_SUPP_TRACE]	= check_ioam_header_and_data,
584	[TEST_IN_UNDEF_NS]		= check_ioam_header,
585	[TEST_IN_NO_ROOM]		= check_ioam_header,
586	[TEST_IN_OFLAG]		= check_ioam_header,
587	[TEST_IN_BIT0]			= check_ioam_header_and_data,
588	[TEST_IN_BIT1]			= check_ioam_header_and_data,
589	[TEST_IN_BIT2]			= check_ioam_header_and_data,
590	[TEST_IN_BIT3]			= check_ioam_header_and_data,
591	[TEST_IN_BIT4]			= check_ioam_header_and_data,
592	[TEST_IN_BIT5]			= check_ioam_header_and_data,
593	[TEST_IN_BIT6]			= check_ioam_header_and_data,
594	[TEST_IN_BIT7]			= check_ioam_header_and_data,
595	[TEST_IN_BIT8]			= check_ioam_header_and_data,
596	[TEST_IN_BIT9]			= check_ioam_header_and_data,
597	[TEST_IN_BIT10]		= check_ioam_header_and_data,
598	[TEST_IN_BIT11]		= check_ioam_header_and_data,
599	[TEST_IN_BIT22]		= check_ioam_header_and_data,
600	[TEST_IN_FULL_SUPP_TRACE]	= check_ioam_header_and_data,
601	[TEST_FWD_FULL_SUPP_TRACE]	= check_ioam_header_and_data,
602};
603
604int main(int argc, char **argv)
605{
606	int fd, size, hoplen, tid, ret = 1;
 
 
 
 
607	struct in6_addr src, dst;
608	struct ioam6_hdr *opt;
609	struct ipv6hdr *ip6h;
610	__u8 buffer[400], *p;
611	__u16 ioam_ns;
612	__u32 tr_type;
 
613
614	if (argc != 7)
615		goto out;
616
617	tid = str2id(argv[2]);
618	if (tid < 0 || !func[tid])
619		goto out;
620
621	if (inet_pton(AF_INET6, argv[3], &src) != 1 ||
622	    inet_pton(AF_INET6, argv[4], &dst) != 1)
 
 
 
 
623		goto out;
624
625	if (get_u32(&tr_type, argv[5], 16) ||
626	    get_u16(&ioam_ns, argv[6], 0))
627		goto out;
 
 
 
 
 
628
629	fd = socket(AF_PACKET, SOCK_DGRAM, __cpu_to_be16(ETH_P_IPV6));
630	if (!fd)
631		goto out;
632
633	if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE,
634		       argv[1], strlen(argv[1])))
635		goto close;
636
 
 
 
 
 
637recv:
638	size = recv(fd, buffer, sizeof(buffer), 0);
639	if (size <= 0)
640		goto close;
641
642	ip6h = (struct ipv6hdr *)buffer;
643
644	if (!ipv6_addr_equal(&ip6h->saddr, &src) ||
645	    !ipv6_addr_equal(&ip6h->daddr, &dst))
 
 
 
 
646		goto recv;
647
648	if (ip6h->nexthdr != IPPROTO_HOPOPTS)
 
 
649		goto close;
650
651	p = buffer + sizeof(*ip6h);
652	hoplen = (p[1] + 1) << 3;
653	p += sizeof(struct ipv6_hopopt_hdr);
654
655	while (hoplen > 0) {
656		opt = (struct ioam6_hdr *)p;
657
658		if (opt->opt_type == IPV6_TLV_IOAM &&
659		    opt->type == IOAM6_TYPE_PREALLOC) {
660			p += sizeof(*opt);
661			ret = func[tid](tid, (struct ioam6_trace_hdr *)p,
662					   tr_type, ioam_ns);
663			break;
664		}
665
666		p += opt->opt_len + 2;
667		hoplen -= opt->opt_len + 2;
668	}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
669close:
670	close(fd);
671out:
672	return ret;
673}
v6.13.7
   1// SPDX-License-Identifier: GPL-2.0+
   2/*
   3 * Author: Justin Iurman (justin.iurman@uliege.be)
   4 *
   5 * IOAM tester for IPv6, see ioam6.sh for details on each test case.
   6 */
   7#include <arpa/inet.h>
   8#include <errno.h>
   9#include <limits.h>
  10#include <linux/const.h>
  11#include <linux/if_ether.h>
  12#include <linux/ioam6.h>
  13#include <linux/ipv6.h>
  14#include <stdbool.h>
  15#include <stdlib.h>
  16#include <string.h>
  17#include <unistd.h>
  18
  19struct ioam_config {
  20	__u32 id;
  21	__u64 wide;
  22	__u16 ingr_id;
  23	__u16 egr_id;
  24	__u32 ingr_wide;
  25	__u32 egr_wide;
  26	__u32 ns_data;
  27	__u64 ns_wide;
  28	__u32 sc_id;
  29	__u8 hlim;
  30	char *sc_data;
  31};
  32
  33/*
  34 * Be careful if you modify structs below - everything MUST be kept synchronized
  35 * with configurations inside ioam6.sh and always reflect the same.
  36 */
  37
  38static struct ioam_config node1 = {
  39	.id = 1,
  40	.wide = 11111111,
  41	.ingr_id = 0xffff, /* default value */
  42	.egr_id = 101,
  43	.ingr_wide = 0xffffffff, /* default value */
  44	.egr_wide = 101101,
  45	.ns_data = 0xdeadbeef,
  46	.ns_wide = 0xcafec0caf00dc0de,
  47	.sc_id = 777,
  48	.sc_data = "something that will be 4n-aligned",
  49	.hlim = 64,
  50};
  51
  52static struct ioam_config node2 = {
  53	.id = 2,
  54	.wide = 22222222,
  55	.ingr_id = 201,
  56	.egr_id = 202,
  57	.ingr_wide = 201201,
  58	.egr_wide = 202202,
  59	.ns_data = 0xffffffff, /* default value */
  60	.ns_wide = 0xffffffffffffffff, /* default value */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
  61	.sc_id = 0xffffff, /* default value */
  62	.sc_data = NULL,
  63	.hlim = 63,
  64};
  65
  66enum {
  67	/**********
  68	 * OUTPUT *
  69	 **********/
  70	__TEST_OUT_MIN,
  71
  72	TEST_OUT_UNDEF_NS,
  73	TEST_OUT_NO_ROOM,
  74	TEST_OUT_NO_ROOM_OSS,
  75	TEST_OUT_BIT0,
  76	TEST_OUT_BIT1,
  77	TEST_OUT_BIT2,
  78	TEST_OUT_BIT3,
  79	TEST_OUT_BIT4,
  80	TEST_OUT_BIT5,
  81	TEST_OUT_BIT6,
  82	TEST_OUT_BIT7,
  83	TEST_OUT_BIT8,
  84	TEST_OUT_BIT9,
  85	TEST_OUT_BIT10,
  86	TEST_OUT_BIT11,
  87	TEST_OUT_BIT22,
  88	TEST_OUT_SIZE4,
  89	TEST_OUT_SIZE8,
  90	TEST_OUT_SIZE12,
  91	TEST_OUT_SIZE16,
  92	TEST_OUT_SIZE20,
  93	TEST_OUT_SIZE24,
  94	TEST_OUT_SIZE28,
  95	TEST_OUT_SIZE32,
  96	TEST_OUT_SIZE36,
  97	TEST_OUT_SIZE40,
  98	TEST_OUT_SIZE44,
  99	TEST_OUT_SIZE48,
 100	TEST_OUT_SIZE52,
 101	TEST_OUT_SIZE56,
 102	TEST_OUT_SIZE60,
 103	TEST_OUT_SIZE64,
 104	TEST_OUT_SIZE68,
 105	TEST_OUT_SIZE72,
 106	TEST_OUT_SIZE76,
 107	TEST_OUT_SIZE80,
 108	TEST_OUT_SIZE84,
 109	TEST_OUT_SIZE88,
 110	TEST_OUT_SIZE92,
 111	TEST_OUT_SIZE96,
 112	TEST_OUT_SIZE100,
 113	TEST_OUT_SIZE104,
 114	TEST_OUT_SIZE108,
 115	TEST_OUT_SIZE112,
 116	TEST_OUT_SIZE116,
 117	TEST_OUT_SIZE120,
 118	TEST_OUT_SIZE124,
 119	TEST_OUT_SIZE128,
 120	TEST_OUT_SIZE132,
 121	TEST_OUT_SIZE136,
 122	TEST_OUT_SIZE140,
 123	TEST_OUT_SIZE144,
 124	TEST_OUT_SIZE148,
 125	TEST_OUT_SIZE152,
 126	TEST_OUT_SIZE156,
 127	TEST_OUT_SIZE160,
 128	TEST_OUT_SIZE164,
 129	TEST_OUT_SIZE168,
 130	TEST_OUT_SIZE172,
 131	TEST_OUT_SIZE176,
 132	TEST_OUT_SIZE180,
 133	TEST_OUT_SIZE184,
 134	TEST_OUT_SIZE188,
 135	TEST_OUT_SIZE192,
 136	TEST_OUT_SIZE196,
 137	TEST_OUT_SIZE200,
 138	TEST_OUT_SIZE204,
 139	TEST_OUT_SIZE208,
 140	TEST_OUT_SIZE212,
 141	TEST_OUT_SIZE216,
 142	TEST_OUT_SIZE220,
 143	TEST_OUT_SIZE224,
 144	TEST_OUT_SIZE228,
 145	TEST_OUT_SIZE232,
 146	TEST_OUT_SIZE236,
 147	TEST_OUT_SIZE240,
 148	TEST_OUT_SIZE244,
 149	TEST_OUT_FULL_SUPP_TRACE,
 150
 151	__TEST_OUT_MAX,
 152
 153	/*********
 154	 * INPUT *
 155	 *********/
 156	__TEST_IN_MIN,
 157
 158	TEST_IN_UNDEF_NS,
 159	TEST_IN_NO_ROOM,
 160	TEST_IN_NO_ROOM_OSS,
 161	TEST_IN_DISABLED,
 162	TEST_IN_OFLAG,
 163	TEST_IN_BIT0,
 164	TEST_IN_BIT1,
 165	TEST_IN_BIT2,
 166	TEST_IN_BIT3,
 167	TEST_IN_BIT4,
 168	TEST_IN_BIT5,
 169	TEST_IN_BIT6,
 170	TEST_IN_BIT7,
 171	TEST_IN_BIT8,
 172	TEST_IN_BIT9,
 173	TEST_IN_BIT10,
 174	TEST_IN_BIT11,
 175	TEST_IN_BIT22,
 176	TEST_IN_SIZE4,
 177	TEST_IN_SIZE8,
 178	TEST_IN_SIZE12,
 179	TEST_IN_SIZE16,
 180	TEST_IN_SIZE20,
 181	TEST_IN_SIZE24,
 182	TEST_IN_SIZE28,
 183	TEST_IN_SIZE32,
 184	TEST_IN_SIZE36,
 185	TEST_IN_SIZE40,
 186	TEST_IN_SIZE44,
 187	TEST_IN_SIZE48,
 188	TEST_IN_SIZE52,
 189	TEST_IN_SIZE56,
 190	TEST_IN_SIZE60,
 191	TEST_IN_SIZE64,
 192	TEST_IN_SIZE68,
 193	TEST_IN_SIZE72,
 194	TEST_IN_SIZE76,
 195	TEST_IN_SIZE80,
 196	TEST_IN_SIZE84,
 197	TEST_IN_SIZE88,
 198	TEST_IN_SIZE92,
 199	TEST_IN_SIZE96,
 200	TEST_IN_SIZE100,
 201	TEST_IN_SIZE104,
 202	TEST_IN_SIZE108,
 203	TEST_IN_SIZE112,
 204	TEST_IN_SIZE116,
 205	TEST_IN_SIZE120,
 206	TEST_IN_SIZE124,
 207	TEST_IN_SIZE128,
 208	TEST_IN_SIZE132,
 209	TEST_IN_SIZE136,
 210	TEST_IN_SIZE140,
 211	TEST_IN_SIZE144,
 212	TEST_IN_SIZE148,
 213	TEST_IN_SIZE152,
 214	TEST_IN_SIZE156,
 215	TEST_IN_SIZE160,
 216	TEST_IN_SIZE164,
 217	TEST_IN_SIZE168,
 218	TEST_IN_SIZE172,
 219	TEST_IN_SIZE176,
 220	TEST_IN_SIZE180,
 221	TEST_IN_SIZE184,
 222	TEST_IN_SIZE188,
 223	TEST_IN_SIZE192,
 224	TEST_IN_SIZE196,
 225	TEST_IN_SIZE200,
 226	TEST_IN_SIZE204,
 227	TEST_IN_SIZE208,
 228	TEST_IN_SIZE212,
 229	TEST_IN_SIZE216,
 230	TEST_IN_SIZE220,
 231	TEST_IN_SIZE224,
 232	TEST_IN_SIZE228,
 233	TEST_IN_SIZE232,
 234	TEST_IN_SIZE236,
 235	TEST_IN_SIZE240,
 236	TEST_IN_SIZE244,
 237	TEST_IN_FULL_SUPP_TRACE,
 238
 239	__TEST_IN_MAX,
 
 
 
 240
 241	__TEST_MAX,
 242};
 243
 244static int check_header(int tid, struct ioam6_trace_hdr *trace,
 245			__u32 trace_type, __u8 trace_size, __u16 ioam_ns)
 246{
 247	if (__be16_to_cpu(trace->namespace_id) != ioam_ns ||
 248	    __be32_to_cpu(trace->type_be32) != (trace_type << 8))
 249		return 1;
 250
 251	switch (tid) {
 252	case TEST_OUT_UNDEF_NS:
 253	case TEST_IN_UNDEF_NS:
 254	case TEST_IN_DISABLED:
 255		return trace->overflow == 1 ||
 256		       trace->nodelen != 1 ||
 257		       trace->remlen != 1;
 258
 259	case TEST_OUT_NO_ROOM:
 260	case TEST_IN_NO_ROOM:
 261	case TEST_IN_OFLAG:
 262		return trace->overflow == 0 ||
 263		       trace->nodelen != 2 ||
 264		       trace->remlen != 1;
 265
 266	case TEST_OUT_NO_ROOM_OSS:
 267		return trace->overflow == 0 ||
 268		       trace->nodelen != 0 ||
 269		       trace->remlen != 1;
 270
 271	case TEST_IN_NO_ROOM_OSS:
 272	case TEST_OUT_BIT22:
 273	case TEST_IN_BIT22:
 274		return trace->overflow == 1 ||
 275		       trace->nodelen != 0 ||
 276		       trace->remlen != 0;
 277
 278	case TEST_OUT_BIT0:
 279	case TEST_IN_BIT0:
 280	case TEST_OUT_BIT1:
 281	case TEST_IN_BIT1:
 282	case TEST_OUT_BIT2:
 283	case TEST_IN_BIT2:
 284	case TEST_OUT_BIT3:
 285	case TEST_IN_BIT3:
 286	case TEST_OUT_BIT4:
 287	case TEST_IN_BIT4:
 288	case TEST_OUT_BIT5:
 289	case TEST_IN_BIT5:
 290	case TEST_OUT_BIT6:
 291	case TEST_IN_BIT6:
 292	case TEST_OUT_BIT7:
 293	case TEST_IN_BIT7:
 294	case TEST_OUT_BIT11:
 295	case TEST_IN_BIT11:
 296		return trace->overflow == 1 ||
 297		       trace->nodelen != 1 ||
 298		       trace->remlen != 0;
 299
 300	case TEST_OUT_BIT8:
 301	case TEST_IN_BIT8:
 302	case TEST_OUT_BIT9:
 303	case TEST_IN_BIT9:
 304	case TEST_OUT_BIT10:
 305	case TEST_IN_BIT10:
 306		return trace->overflow == 1 ||
 307		       trace->nodelen != 2 ||
 308		       trace->remlen != 0;
 309
 310	case TEST_OUT_SIZE4:
 311	case TEST_OUT_SIZE8:
 312	case TEST_OUT_SIZE12:
 313	case TEST_OUT_SIZE16:
 314	case TEST_OUT_SIZE20:
 315	case TEST_OUT_SIZE24:
 316	case TEST_OUT_SIZE28:
 317	case TEST_OUT_SIZE32:
 318	case TEST_OUT_SIZE36:
 319	case TEST_OUT_SIZE40:
 320	case TEST_OUT_SIZE44:
 321	case TEST_OUT_SIZE48:
 322	case TEST_OUT_SIZE52:
 323	case TEST_OUT_SIZE56:
 324	case TEST_OUT_SIZE60:
 325	case TEST_OUT_SIZE64:
 326	case TEST_OUT_SIZE68:
 327	case TEST_OUT_SIZE72:
 328	case TEST_OUT_SIZE76:
 329	case TEST_OUT_SIZE80:
 330	case TEST_OUT_SIZE84:
 331	case TEST_OUT_SIZE88:
 332	case TEST_OUT_SIZE92:
 333	case TEST_OUT_SIZE96:
 334	case TEST_OUT_SIZE100:
 335	case TEST_OUT_SIZE104:
 336	case TEST_OUT_SIZE108:
 337	case TEST_OUT_SIZE112:
 338	case TEST_OUT_SIZE116:
 339	case TEST_OUT_SIZE120:
 340	case TEST_OUT_SIZE124:
 341	case TEST_OUT_SIZE128:
 342	case TEST_OUT_SIZE132:
 343	case TEST_OUT_SIZE136:
 344	case TEST_OUT_SIZE140:
 345	case TEST_OUT_SIZE144:
 346	case TEST_OUT_SIZE148:
 347	case TEST_OUT_SIZE152:
 348	case TEST_OUT_SIZE156:
 349	case TEST_OUT_SIZE160:
 350	case TEST_OUT_SIZE164:
 351	case TEST_OUT_SIZE168:
 352	case TEST_OUT_SIZE172:
 353	case TEST_OUT_SIZE176:
 354	case TEST_OUT_SIZE180:
 355	case TEST_OUT_SIZE184:
 356	case TEST_OUT_SIZE188:
 357	case TEST_OUT_SIZE192:
 358	case TEST_OUT_SIZE196:
 359	case TEST_OUT_SIZE200:
 360	case TEST_OUT_SIZE204:
 361	case TEST_OUT_SIZE208:
 362	case TEST_OUT_SIZE212:
 363	case TEST_OUT_SIZE216:
 364	case TEST_OUT_SIZE220:
 365	case TEST_OUT_SIZE224:
 366	case TEST_OUT_SIZE228:
 367	case TEST_OUT_SIZE232:
 368	case TEST_OUT_SIZE236:
 369	case TEST_OUT_SIZE240:
 370	case TEST_OUT_SIZE244:
 371		return trace->overflow == 1 ||
 372		       trace->nodelen != 1 ||
 373		       trace->remlen != trace_size / 4;
 374
 375	case TEST_IN_SIZE4:
 376	case TEST_IN_SIZE8:
 377	case TEST_IN_SIZE12:
 378	case TEST_IN_SIZE16:
 379	case TEST_IN_SIZE20:
 380	case TEST_IN_SIZE24:
 381	case TEST_IN_SIZE28:
 382	case TEST_IN_SIZE32:
 383	case TEST_IN_SIZE36:
 384	case TEST_IN_SIZE40:
 385	case TEST_IN_SIZE44:
 386	case TEST_IN_SIZE48:
 387	case TEST_IN_SIZE52:
 388	case TEST_IN_SIZE56:
 389	case TEST_IN_SIZE60:
 390	case TEST_IN_SIZE64:
 391	case TEST_IN_SIZE68:
 392	case TEST_IN_SIZE72:
 393	case TEST_IN_SIZE76:
 394	case TEST_IN_SIZE80:
 395	case TEST_IN_SIZE84:
 396	case TEST_IN_SIZE88:
 397	case TEST_IN_SIZE92:
 398	case TEST_IN_SIZE96:
 399	case TEST_IN_SIZE100:
 400	case TEST_IN_SIZE104:
 401	case TEST_IN_SIZE108:
 402	case TEST_IN_SIZE112:
 403	case TEST_IN_SIZE116:
 404	case TEST_IN_SIZE120:
 405	case TEST_IN_SIZE124:
 406	case TEST_IN_SIZE128:
 407	case TEST_IN_SIZE132:
 408	case TEST_IN_SIZE136:
 409	case TEST_IN_SIZE140:
 410	case TEST_IN_SIZE144:
 411	case TEST_IN_SIZE148:
 412	case TEST_IN_SIZE152:
 413	case TEST_IN_SIZE156:
 414	case TEST_IN_SIZE160:
 415	case TEST_IN_SIZE164:
 416	case TEST_IN_SIZE168:
 417	case TEST_IN_SIZE172:
 418	case TEST_IN_SIZE176:
 419	case TEST_IN_SIZE180:
 420	case TEST_IN_SIZE184:
 421	case TEST_IN_SIZE188:
 422	case TEST_IN_SIZE192:
 423	case TEST_IN_SIZE196:
 424	case TEST_IN_SIZE200:
 425	case TEST_IN_SIZE204:
 426	case TEST_IN_SIZE208:
 427	case TEST_IN_SIZE212:
 428	case TEST_IN_SIZE216:
 429	case TEST_IN_SIZE220:
 430	case TEST_IN_SIZE224:
 431	case TEST_IN_SIZE228:
 432	case TEST_IN_SIZE232:
 433	case TEST_IN_SIZE236:
 434	case TEST_IN_SIZE240:
 435	case TEST_IN_SIZE244:
 436		return trace->overflow == 1 ||
 437		       trace->nodelen != 1 ||
 438		       trace->remlen != (trace_size / 4) - trace->nodelen;
 439
 440	case TEST_OUT_FULL_SUPP_TRACE:
 441	case TEST_IN_FULL_SUPP_TRACE:
 442		return trace->overflow == 1 ||
 443		       trace->nodelen != 15 ||
 444		       trace->remlen != 0;
 
 445
 446	default:
 447		break;
 448	}
 449
 450	return 1;
 451}
 452
 453static int check_data(struct ioam6_trace_hdr *trace, __u8 trace_size,
 454		      const struct ioam_config cnf, bool is_output)
 455{
 456	unsigned int len, i;
 457	__u8 aligned;
 458	__u64 raw64;
 459	__u32 raw32;
 460	__u8 *p;
 461
 462	if (trace->type.bit12 | trace->type.bit13 | trace->type.bit14 |
 463	    trace->type.bit15 | trace->type.bit16 | trace->type.bit17 |
 464	    trace->type.bit18 | trace->type.bit19 | trace->type.bit20 |
 465	    trace->type.bit21 | trace->type.bit23)
 466		return 1;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 467
 468	for (i = 0; i < trace->remlen * 4; i++) {
 469		if (trace->data[i] != 0)
 470			return 1;
 
 471	}
 472
 473	if (trace->remlen * 4 == trace_size)
 474		return 0;
 
 
 
 
 
 475
 476	p = trace->data + trace->remlen * 4;
 
 
 
 477
 478	if (trace->type.bit0) {
 479		raw32 = __be32_to_cpu(*((__u32 *)p));
 480		if (cnf.hlim != (raw32 >> 24) || cnf.id != (raw32 & 0xffffff))
 481			return 1;
 482		p += sizeof(__u32);
 483	}
 484
 485	if (trace->type.bit1) {
 486		raw32 = __be32_to_cpu(*((__u32 *)p));
 487		if (cnf.ingr_id != (raw32 >> 16) ||
 488		    cnf.egr_id != (raw32 & 0xffff))
 489			return 1;
 490		p += sizeof(__u32);
 491	}
 492
 493	if (trace->type.bit2) {
 494		raw32 = __be32_to_cpu(*((__u32 *)p));
 495		if ((is_output && raw32 != 0xffffffff) ||
 496		    (!is_output && (raw32 == 0 || raw32 == 0xffffffff)))
 497			return 1;
 498		p += sizeof(__u32);
 499	}
 500
 501	if (trace->type.bit3) {
 502		raw32 = __be32_to_cpu(*((__u32 *)p));
 503		if ((is_output && raw32 != 0xffffffff) ||
 504		    (!is_output && (raw32 == 0 || raw32 == 0xffffffff)))
 505			return 1;
 506		p += sizeof(__u32);
 507	}
 508
 509	if (trace->type.bit4) {
 510		if (__be32_to_cpu(*((__u32 *)p)) != 0xffffffff)
 511			return 1;
 512		p += sizeof(__u32);
 513	}
 514
 515	if (trace->type.bit5) {
 516		if (__be32_to_cpu(*((__u32 *)p)) != cnf.ns_data)
 517			return 1;
 518		p += sizeof(__u32);
 519	}
 520
 521	if (trace->type.bit6) {
 522		if (__be32_to_cpu(*((__u32 *)p)) == 0xffffffff)
 523			return 1;
 524		p += sizeof(__u32);
 525	}
 526
 527	if (trace->type.bit7) {
 528		if (__be32_to_cpu(*((__u32 *)p)) != 0xffffffff)
 529			return 1;
 530		p += sizeof(__u32);
 531	}
 532
 533	if (trace->type.bit8) {
 534		raw64 = __be64_to_cpu(*((__u64 *)p));
 535		if (cnf.hlim != (raw64 >> 56) ||
 536		    cnf.wide != (raw64 & 0xffffffffffffff))
 537			return 1;
 538		p += sizeof(__u64);
 539	}
 540
 541	if (trace->type.bit9) {
 542		if (__be32_to_cpu(*((__u32 *)p)) != cnf.ingr_wide)
 543			return 1;
 544		p += sizeof(__u32);
 
 545
 546		if (__be32_to_cpu(*((__u32 *)p)) != cnf.egr_wide)
 
 547			return 1;
 548		p += sizeof(__u32);
 549	}
 550
 551	if (trace->type.bit10) {
 552		if (__be64_to_cpu(*((__u64 *)p)) != cnf.ns_wide)
 553			return 1;
 554		p += sizeof(__u64);
 555	}
 556
 557	if (trace->type.bit11) {
 558		if (__be32_to_cpu(*((__u32 *)p)) != 0xffffffff)
 559			return 1;
 560		p += sizeof(__u32);
 561	}
 562
 563	if (trace->type.bit22) {
 564		len = cnf.sc_data ? strlen(cnf.sc_data) : 0;
 565		aligned = cnf.sc_data ? __ALIGN_KERNEL(len, 4) : 0;
 566
 567		raw32 = __be32_to_cpu(*((__u32 *)p));
 568		if (aligned != (raw32 >> 24) * 4 ||
 569		    cnf.sc_id != (raw32 & 0xffffff))
 570			return 1;
 571		p += sizeof(__u32);
 572
 573		if (cnf.sc_data) {
 574			if (strncmp((char *)p, cnf.sc_data, len))
 575				return 1;
 576
 577			p += len;
 578			aligned -= len;
 579
 580			while (aligned--) {
 581				if (*p != '\0')
 582					return 1;
 583				p += sizeof(__u8);
 584			}
 585		}
 586	}
 587
 588	return 0;
 589}
 590
 591static int check_ioam_trace(int tid, struct ioam6_trace_hdr *trace,
 592			    __u32 trace_type, __u8 trace_size, __u16 ioam_ns)
 593{
 594	if (check_header(tid, trace, trace_type, trace_size, ioam_ns))
 
 
 595		return 1;
 596
 597	if (tid > __TEST_OUT_MIN && tid < __TEST_OUT_MAX)
 598		return check_data(trace, trace_size, node1, true);
 599
 600	if (tid > __TEST_IN_MIN && tid < __TEST_IN_MAX)
 601		return check_data(trace, trace_size, node2, false);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 602
 603	return 1;
 604}
 605
 606static int str2id(const char *tname)
 607{
 608	if (!strcmp("output_undef_ns", tname))
 609		return TEST_OUT_UNDEF_NS;
 610	if (!strcmp("output_no_room", tname))
 611		return TEST_OUT_NO_ROOM;
 612	if (!strcmp("output_no_room_oss", tname))
 613		return TEST_OUT_NO_ROOM_OSS;
 614	if (!strcmp("output_bit0", tname))
 615		return TEST_OUT_BIT0;
 616	if (!strcmp("output_bit1", tname))
 617		return TEST_OUT_BIT1;
 618	if (!strcmp("output_bit2", tname))
 619		return TEST_OUT_BIT2;
 620	if (!strcmp("output_bit3", tname))
 621		return TEST_OUT_BIT3;
 622	if (!strcmp("output_bit4", tname))
 623		return TEST_OUT_BIT4;
 624	if (!strcmp("output_bit5", tname))
 625		return TEST_OUT_BIT5;
 626	if (!strcmp("output_bit6", tname))
 627		return TEST_OUT_BIT6;
 628	if (!strcmp("output_bit7", tname))
 629		return TEST_OUT_BIT7;
 630	if (!strcmp("output_bit8", tname))
 631		return TEST_OUT_BIT8;
 632	if (!strcmp("output_bit9", tname))
 633		return TEST_OUT_BIT9;
 634	if (!strcmp("output_bit10", tname))
 635		return TEST_OUT_BIT10;
 636	if (!strcmp("output_bit11", tname))
 637		return TEST_OUT_BIT11;
 638	if (!strcmp("output_bit22", tname))
 639		return TEST_OUT_BIT22;
 640	if (!strcmp("output_size4", tname))
 641		return TEST_OUT_SIZE4;
 642	if (!strcmp("output_size8", tname))
 643		return TEST_OUT_SIZE8;
 644	if (!strcmp("output_size12", tname))
 645		return TEST_OUT_SIZE12;
 646	if (!strcmp("output_size16", tname))
 647		return TEST_OUT_SIZE16;
 648	if (!strcmp("output_size20", tname))
 649		return TEST_OUT_SIZE20;
 650	if (!strcmp("output_size24", tname))
 651		return TEST_OUT_SIZE24;
 652	if (!strcmp("output_size28", tname))
 653		return TEST_OUT_SIZE28;
 654	if (!strcmp("output_size32", tname))
 655		return TEST_OUT_SIZE32;
 656	if (!strcmp("output_size36", tname))
 657		return TEST_OUT_SIZE36;
 658	if (!strcmp("output_size40", tname))
 659		return TEST_OUT_SIZE40;
 660	if (!strcmp("output_size44", tname))
 661		return TEST_OUT_SIZE44;
 662	if (!strcmp("output_size48", tname))
 663		return TEST_OUT_SIZE48;
 664	if (!strcmp("output_size52", tname))
 665		return TEST_OUT_SIZE52;
 666	if (!strcmp("output_size56", tname))
 667		return TEST_OUT_SIZE56;
 668	if (!strcmp("output_size60", tname))
 669		return TEST_OUT_SIZE60;
 670	if (!strcmp("output_size64", tname))
 671		return TEST_OUT_SIZE64;
 672	if (!strcmp("output_size68", tname))
 673		return TEST_OUT_SIZE68;
 674	if (!strcmp("output_size72", tname))
 675		return TEST_OUT_SIZE72;
 676	if (!strcmp("output_size76", tname))
 677		return TEST_OUT_SIZE76;
 678	if (!strcmp("output_size80", tname))
 679		return TEST_OUT_SIZE80;
 680	if (!strcmp("output_size84", tname))
 681		return TEST_OUT_SIZE84;
 682	if (!strcmp("output_size88", tname))
 683		return TEST_OUT_SIZE88;
 684	if (!strcmp("output_size92", tname))
 685		return TEST_OUT_SIZE92;
 686	if (!strcmp("output_size96", tname))
 687		return TEST_OUT_SIZE96;
 688	if (!strcmp("output_size100", tname))
 689		return TEST_OUT_SIZE100;
 690	if (!strcmp("output_size104", tname))
 691		return TEST_OUT_SIZE104;
 692	if (!strcmp("output_size108", tname))
 693		return TEST_OUT_SIZE108;
 694	if (!strcmp("output_size112", tname))
 695		return TEST_OUT_SIZE112;
 696	if (!strcmp("output_size116", tname))
 697		return TEST_OUT_SIZE116;
 698	if (!strcmp("output_size120", tname))
 699		return TEST_OUT_SIZE120;
 700	if (!strcmp("output_size124", tname))
 701		return TEST_OUT_SIZE124;
 702	if (!strcmp("output_size128", tname))
 703		return TEST_OUT_SIZE128;
 704	if (!strcmp("output_size132", tname))
 705		return TEST_OUT_SIZE132;
 706	if (!strcmp("output_size136", tname))
 707		return TEST_OUT_SIZE136;
 708	if (!strcmp("output_size140", tname))
 709		return TEST_OUT_SIZE140;
 710	if (!strcmp("output_size144", tname))
 711		return TEST_OUT_SIZE144;
 712	if (!strcmp("output_size148", tname))
 713		return TEST_OUT_SIZE148;
 714	if (!strcmp("output_size152", tname))
 715		return TEST_OUT_SIZE152;
 716	if (!strcmp("output_size156", tname))
 717		return TEST_OUT_SIZE156;
 718	if (!strcmp("output_size160", tname))
 719		return TEST_OUT_SIZE160;
 720	if (!strcmp("output_size164", tname))
 721		return TEST_OUT_SIZE164;
 722	if (!strcmp("output_size168", tname))
 723		return TEST_OUT_SIZE168;
 724	if (!strcmp("output_size172", tname))
 725		return TEST_OUT_SIZE172;
 726	if (!strcmp("output_size176", tname))
 727		return TEST_OUT_SIZE176;
 728	if (!strcmp("output_size180", tname))
 729		return TEST_OUT_SIZE180;
 730	if (!strcmp("output_size184", tname))
 731		return TEST_OUT_SIZE184;
 732	if (!strcmp("output_size188", tname))
 733		return TEST_OUT_SIZE188;
 734	if (!strcmp("output_size192", tname))
 735		return TEST_OUT_SIZE192;
 736	if (!strcmp("output_size196", tname))
 737		return TEST_OUT_SIZE196;
 738	if (!strcmp("output_size200", tname))
 739		return TEST_OUT_SIZE200;
 740	if (!strcmp("output_size204", tname))
 741		return TEST_OUT_SIZE204;
 742	if (!strcmp("output_size208", tname))
 743		return TEST_OUT_SIZE208;
 744	if (!strcmp("output_size212", tname))
 745		return TEST_OUT_SIZE212;
 746	if (!strcmp("output_size216", tname))
 747		return TEST_OUT_SIZE216;
 748	if (!strcmp("output_size220", tname))
 749		return TEST_OUT_SIZE220;
 750	if (!strcmp("output_size224", tname))
 751		return TEST_OUT_SIZE224;
 752	if (!strcmp("output_size228", tname))
 753		return TEST_OUT_SIZE228;
 754	if (!strcmp("output_size232", tname))
 755		return TEST_OUT_SIZE232;
 756	if (!strcmp("output_size236", tname))
 757		return TEST_OUT_SIZE236;
 758	if (!strcmp("output_size240", tname))
 759		return TEST_OUT_SIZE240;
 760	if (!strcmp("output_size244", tname))
 761		return TEST_OUT_SIZE244;
 762	if (!strcmp("output_full_supp_trace", tname))
 763		return TEST_OUT_FULL_SUPP_TRACE;
 764	if (!strcmp("input_undef_ns", tname))
 765		return TEST_IN_UNDEF_NS;
 766	if (!strcmp("input_no_room", tname))
 767		return TEST_IN_NO_ROOM;
 768	if (!strcmp("input_no_room_oss", tname))
 769		return TEST_IN_NO_ROOM_OSS;
 770	if (!strcmp("input_disabled", tname))
 771		return TEST_IN_DISABLED;
 772	if (!strcmp("input_oflag", tname))
 773		return TEST_IN_OFLAG;
 774	if (!strcmp("input_bit0", tname))
 775		return TEST_IN_BIT0;
 776	if (!strcmp("input_bit1", tname))
 777		return TEST_IN_BIT1;
 778	if (!strcmp("input_bit2", tname))
 779		return TEST_IN_BIT2;
 780	if (!strcmp("input_bit3", tname))
 781		return TEST_IN_BIT3;
 782	if (!strcmp("input_bit4", tname))
 783		return TEST_IN_BIT4;
 784	if (!strcmp("input_bit5", tname))
 785		return TEST_IN_BIT5;
 786	if (!strcmp("input_bit6", tname))
 787		return TEST_IN_BIT6;
 788	if (!strcmp("input_bit7", tname))
 789		return TEST_IN_BIT7;
 790	if (!strcmp("input_bit8", tname))
 791		return TEST_IN_BIT8;
 792	if (!strcmp("input_bit9", tname))
 793		return TEST_IN_BIT9;
 794	if (!strcmp("input_bit10", tname))
 795		return TEST_IN_BIT10;
 796	if (!strcmp("input_bit11", tname))
 797		return TEST_IN_BIT11;
 798	if (!strcmp("input_bit22", tname))
 799		return TEST_IN_BIT22;
 800	if (!strcmp("input_size4", tname))
 801		return TEST_IN_SIZE4;
 802	if (!strcmp("input_size8", tname))
 803		return TEST_IN_SIZE8;
 804	if (!strcmp("input_size12", tname))
 805		return TEST_IN_SIZE12;
 806	if (!strcmp("input_size16", tname))
 807		return TEST_IN_SIZE16;
 808	if (!strcmp("input_size20", tname))
 809		return TEST_IN_SIZE20;
 810	if (!strcmp("input_size24", tname))
 811		return TEST_IN_SIZE24;
 812	if (!strcmp("input_size28", tname))
 813		return TEST_IN_SIZE28;
 814	if (!strcmp("input_size32", tname))
 815		return TEST_IN_SIZE32;
 816	if (!strcmp("input_size36", tname))
 817		return TEST_IN_SIZE36;
 818	if (!strcmp("input_size40", tname))
 819		return TEST_IN_SIZE40;
 820	if (!strcmp("input_size44", tname))
 821		return TEST_IN_SIZE44;
 822	if (!strcmp("input_size48", tname))
 823		return TEST_IN_SIZE48;
 824	if (!strcmp("input_size52", tname))
 825		return TEST_IN_SIZE52;
 826	if (!strcmp("input_size56", tname))
 827		return TEST_IN_SIZE56;
 828	if (!strcmp("input_size60", tname))
 829		return TEST_IN_SIZE60;
 830	if (!strcmp("input_size64", tname))
 831		return TEST_IN_SIZE64;
 832	if (!strcmp("input_size68", tname))
 833		return TEST_IN_SIZE68;
 834	if (!strcmp("input_size72", tname))
 835		return TEST_IN_SIZE72;
 836	if (!strcmp("input_size76", tname))
 837		return TEST_IN_SIZE76;
 838	if (!strcmp("input_size80", tname))
 839		return TEST_IN_SIZE80;
 840	if (!strcmp("input_size84", tname))
 841		return TEST_IN_SIZE84;
 842	if (!strcmp("input_size88", tname))
 843		return TEST_IN_SIZE88;
 844	if (!strcmp("input_size92", tname))
 845		return TEST_IN_SIZE92;
 846	if (!strcmp("input_size96", tname))
 847		return TEST_IN_SIZE96;
 848	if (!strcmp("input_size100", tname))
 849		return TEST_IN_SIZE100;
 850	if (!strcmp("input_size104", tname))
 851		return TEST_IN_SIZE104;
 852	if (!strcmp("input_size108", tname))
 853		return TEST_IN_SIZE108;
 854	if (!strcmp("input_size112", tname))
 855		return TEST_IN_SIZE112;
 856	if (!strcmp("input_size116", tname))
 857		return TEST_IN_SIZE116;
 858	if (!strcmp("input_size120", tname))
 859		return TEST_IN_SIZE120;
 860	if (!strcmp("input_size124", tname))
 861		return TEST_IN_SIZE124;
 862	if (!strcmp("input_size128", tname))
 863		return TEST_IN_SIZE128;
 864	if (!strcmp("input_size132", tname))
 865		return TEST_IN_SIZE132;
 866	if (!strcmp("input_size136", tname))
 867		return TEST_IN_SIZE136;
 868	if (!strcmp("input_size140", tname))
 869		return TEST_IN_SIZE140;
 870	if (!strcmp("input_size144", tname))
 871		return TEST_IN_SIZE144;
 872	if (!strcmp("input_size148", tname))
 873		return TEST_IN_SIZE148;
 874	if (!strcmp("input_size152", tname))
 875		return TEST_IN_SIZE152;
 876	if (!strcmp("input_size156", tname))
 877		return TEST_IN_SIZE156;
 878	if (!strcmp("input_size160", tname))
 879		return TEST_IN_SIZE160;
 880	if (!strcmp("input_size164", tname))
 881		return TEST_IN_SIZE164;
 882	if (!strcmp("input_size168", tname))
 883		return TEST_IN_SIZE168;
 884	if (!strcmp("input_size172", tname))
 885		return TEST_IN_SIZE172;
 886	if (!strcmp("input_size176", tname))
 887		return TEST_IN_SIZE176;
 888	if (!strcmp("input_size180", tname))
 889		return TEST_IN_SIZE180;
 890	if (!strcmp("input_size184", tname))
 891		return TEST_IN_SIZE184;
 892	if (!strcmp("input_size188", tname))
 893		return TEST_IN_SIZE188;
 894	if (!strcmp("input_size192", tname))
 895		return TEST_IN_SIZE192;
 896	if (!strcmp("input_size196", tname))
 897		return TEST_IN_SIZE196;
 898	if (!strcmp("input_size200", tname))
 899		return TEST_IN_SIZE200;
 900	if (!strcmp("input_size204", tname))
 901		return TEST_IN_SIZE204;
 902	if (!strcmp("input_size208", tname))
 903		return TEST_IN_SIZE208;
 904	if (!strcmp("input_size212", tname))
 905		return TEST_IN_SIZE212;
 906	if (!strcmp("input_size216", tname))
 907		return TEST_IN_SIZE216;
 908	if (!strcmp("input_size220", tname))
 909		return TEST_IN_SIZE220;
 910	if (!strcmp("input_size224", tname))
 911		return TEST_IN_SIZE224;
 912	if (!strcmp("input_size228", tname))
 913		return TEST_IN_SIZE228;
 914	if (!strcmp("input_size232", tname))
 915		return TEST_IN_SIZE232;
 916	if (!strcmp("input_size236", tname))
 917		return TEST_IN_SIZE236;
 918	if (!strcmp("input_size240", tname))
 919		return TEST_IN_SIZE240;
 920	if (!strcmp("input_size244", tname))
 921		return TEST_IN_SIZE244;
 922	if (!strcmp("input_full_supp_trace", tname))
 923		return TEST_IN_FULL_SUPP_TRACE;
 
 
 924
 925	return -1;
 926}
 927
 928static int ipv6_addr_equal(const struct in6_addr *a1, const struct in6_addr *a2)
 929{
 930	return ((a1->s6_addr32[0] ^ a2->s6_addr32[0]) |
 931		(a1->s6_addr32[1] ^ a2->s6_addr32[1]) |
 932		(a1->s6_addr32[2] ^ a2->s6_addr32[2]) |
 933		(a1->s6_addr32[3] ^ a2->s6_addr32[3])) == 0;
 934}
 935
 936static int get_u32(__u32 *val, const char *arg, int base)
 937{
 938	unsigned long res;
 939	char *ptr;
 940
 941	if (!arg || !*arg)
 942		return -1;
 943	res = strtoul(arg, &ptr, base);
 944
 945	if (!ptr || ptr == arg || *ptr)
 946		return -1;
 947
 948	if (res == ULONG_MAX && errno == ERANGE)
 949		return -1;
 950
 951	if (res > 0xFFFFFFFFUL)
 952		return -1;
 953
 954	*val = res;
 955	return 0;
 956}
 957
 958static int get_u16(__u16 *val, const char *arg, int base)
 959{
 960	unsigned long res;
 961	char *ptr;
 962
 963	if (!arg || !*arg)
 964		return -1;
 965	res = strtoul(arg, &ptr, base);
 966
 967	if (!ptr || ptr == arg || *ptr)
 968		return -1;
 969
 970	if (res == ULONG_MAX && errno == ERANGE)
 971		return -1;
 972
 973	if (res > 0xFFFFUL)
 974		return -1;
 975
 976	*val = res;
 977	return 0;
 978}
 979
 980static int get_u8(__u8 *val, const char *arg, int base)
 981{
 982	unsigned long res;
 983	char *ptr;
 984
 985	if (!arg || !*arg)
 986		return -1;
 987	res = strtoul(arg, &ptr, base);
 988
 989	if (!ptr || ptr == arg || *ptr)
 990		return -1;
 991
 992	if (res == ULONG_MAX && errno == ERANGE)
 993		return -1;
 994
 995	if (res > 0xFFUL)
 996		return -1;
 997
 998	*val = res;
 999	return 0;
1000}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1001
1002int main(int argc, char **argv)
1003{
1004	__u8 buffer[512], *ptr, nexthdr, tr_size;
1005	struct ioam6_trace_hdr *trace;
1006	unsigned int hoplen, ret = 1;
1007	struct ipv6_hopopt_hdr *hbh;
1008	int fd, size, testname_id;
1009	struct in6_addr src, dst;
1010	struct ioam6_hdr *ioam6;
1011	struct timeval timeout;
1012	struct ipv6hdr *ipv6;
 
1013	__u32 tr_type;
1014	__u16 ioam_ns;
1015
1016	if (argc != 9)
1017		goto out;
1018
1019	testname_id = str2id(argv[2]);
 
 
1020
1021	if (testname_id < 0 ||
1022	    inet_pton(AF_INET6, argv[3], &src) != 1 ||
1023	    inet_pton(AF_INET6, argv[4], &dst) != 1 ||
1024	    get_u32(&tr_type, argv[5], 16) ||
1025	    get_u8(&tr_size, argv[6], 0) ||
1026	    get_u16(&ioam_ns, argv[7], 0))
1027		goto out;
1028
1029	nexthdr = (!strcmp(argv[8], "encap") ? IPPROTO_IPV6 : IPPROTO_ICMPV6);
1030
1031	hoplen = sizeof(*hbh);
1032	hoplen += 2; // 2-byte padding for alignment
1033	hoplen += sizeof(*ioam6); // IOAM option header
1034	hoplen += sizeof(*trace); // IOAM trace header
1035	hoplen += tr_size; // IOAM trace size
1036	hoplen += (tr_size % 8); // optional padding
1037
1038	fd = socket(AF_PACKET, SOCK_DGRAM, __cpu_to_be16(ETH_P_IPV6));
1039	if (fd < 0)
1040		goto out;
1041
1042	if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE,
1043		       argv[1], strlen(argv[1])))
1044		goto close;
1045
1046	timeout.tv_sec = 1;
1047	timeout.tv_usec = 0;
1048	if (setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO,
1049		       (const char *)&timeout, sizeof(timeout)))
1050		goto close;
1051recv:
1052	size = recv(fd, buffer, sizeof(buffer), 0);
1053	if (size <= 0)
1054		goto close;
1055
1056	ipv6 = (struct ipv6hdr *)buffer;
1057
1058	/* Skip packets that do not have the expected src/dst address or that
1059	 * do not have a Hop-by-hop.
1060	 */
1061	if (!ipv6_addr_equal(&ipv6->saddr, &src) ||
1062	    !ipv6_addr_equal(&ipv6->daddr, &dst) ||
1063	    ipv6->nexthdr != IPPROTO_HOPOPTS)
1064		goto recv;
1065
1066	/* Check Hbh's Next Header and Size. */
1067	hbh = (struct ipv6_hopopt_hdr *)(buffer + sizeof(*ipv6));
1068	if (hbh->nexthdr != nexthdr || hbh->hdrlen != (hoplen >> 3) - 1)
1069		goto close;
1070
1071	/* Check we have a 2-byte padding for alignment. */
1072	ptr = (__u8 *)hbh + sizeof(*hbh);
1073	if (ptr[0] != IPV6_TLV_PADN && ptr[1] != 0)
1074		goto close;
 
 
 
 
 
 
 
 
 
 
1075
1076	/* Check we now have the IOAM option. */
1077	ptr += 2;
1078	if (ptr[0] != IPV6_TLV_IOAM)
1079		goto close;
1080
1081	/* Check its size and the IOAM option type. */
1082	ioam6 = (struct ioam6_hdr *)ptr;
1083	if (ioam6->opt_len != sizeof(*ioam6) - 2 + sizeof(*trace) + tr_size ||
1084	    ioam6->type != IOAM6_TYPE_PREALLOC)
1085		goto close;
1086
1087	trace = (struct ioam6_trace_hdr *)(ptr + sizeof(*ioam6));
1088
1089	/* Check the trailing 4-byte padding (potentially). */
1090	ptr = (__u8 *)trace + sizeof(*trace) + tr_size;
1091	if (tr_size % 8 && ptr[0] != IPV6_TLV_PADN && ptr[1] != 2 &&
1092	    ptr[2] != 0 && ptr[3] != 0)
1093		goto close;
1094
1095	/* Check the IOAM header and data. */
1096	ret = check_ioam_trace(testname_id, trace, tr_type, tr_size, ioam_ns);
1097close:
1098	close(fd);
1099out:
1100	return ret;
1101}