xref: /aosp_15_r20/external/iproute2/tc/f_matchall.c (revision de1e4e894b0c224df933550f0afdecc354b238c4)
1*de1e4e89SAndroid Build Coastguard Worker /*
2*de1e4e89SAndroid Build Coastguard Worker  * f_matchall.c		Match-all Classifier
3*de1e4e89SAndroid Build Coastguard Worker  *
4*de1e4e89SAndroid Build Coastguard Worker  *		This program is free software; you can distribute it and/or
5*de1e4e89SAndroid Build Coastguard Worker  *		modify it under the terms of the GNU General Public License
6*de1e4e89SAndroid Build Coastguard Worker  *		as published by the Free Software Foundation; either version
7*de1e4e89SAndroid Build Coastguard Worker  *		2 of the License, or (at your option) any later version.
8*de1e4e89SAndroid Build Coastguard Worker  *
9*de1e4e89SAndroid Build Coastguard Worker  * Authors:	Jiri Pirko <[email protected]>, Yotam Gigi <[email protected]>
10*de1e4e89SAndroid Build Coastguard Worker  *
11*de1e4e89SAndroid Build Coastguard Worker  */
12*de1e4e89SAndroid Build Coastguard Worker 
13*de1e4e89SAndroid Build Coastguard Worker #include <stdio.h>
14*de1e4e89SAndroid Build Coastguard Worker #include <stdlib.h>
15*de1e4e89SAndroid Build Coastguard Worker #include <unistd.h>
16*de1e4e89SAndroid Build Coastguard Worker #include <syslog.h>
17*de1e4e89SAndroid Build Coastguard Worker #include <fcntl.h>
18*de1e4e89SAndroid Build Coastguard Worker #include <sys/socket.h>
19*de1e4e89SAndroid Build Coastguard Worker #include <netinet/in.h>
20*de1e4e89SAndroid Build Coastguard Worker #include <arpa/inet.h>
21*de1e4e89SAndroid Build Coastguard Worker #include <string.h>
22*de1e4e89SAndroid Build Coastguard Worker #include <linux/if.h>
23*de1e4e89SAndroid Build Coastguard Worker 
24*de1e4e89SAndroid Build Coastguard Worker #include "utils.h"
25*de1e4e89SAndroid Build Coastguard Worker #include "tc_util.h"
26*de1e4e89SAndroid Build Coastguard Worker 
explain(void)27*de1e4e89SAndroid Build Coastguard Worker static void explain(void)
28*de1e4e89SAndroid Build Coastguard Worker {
29*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "Usage: ... matchall [skip_sw | skip_hw]\n");
30*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "                 [ action ACTION_SPEC ] [ classid CLASSID ]\n");
31*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "\n");
32*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "Where: SELECTOR := SAMPLE SAMPLE ...\n");
33*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "       FILTERID := X:Y:Z\n");
34*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "       ACTION_SPEC := ... look at individual actions\n");
35*de1e4e89SAndroid Build Coastguard Worker 	fprintf(stderr, "\nNOTE: CLASSID is parsed as hexadecimal input.\n");
36*de1e4e89SAndroid Build Coastguard Worker }
37*de1e4e89SAndroid Build Coastguard Worker 
matchall_parse_opt(struct filter_util * qu,char * handle,int argc,char ** argv,struct nlmsghdr * n)38*de1e4e89SAndroid Build Coastguard Worker static int matchall_parse_opt(struct filter_util *qu, char *handle,
39*de1e4e89SAndroid Build Coastguard Worker 			   int argc, char **argv, struct nlmsghdr *n)
40*de1e4e89SAndroid Build Coastguard Worker {
41*de1e4e89SAndroid Build Coastguard Worker 	struct tcmsg *t = NLMSG_DATA(n);
42*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *tail;
43*de1e4e89SAndroid Build Coastguard Worker 	__u32 flags = 0;
44*de1e4e89SAndroid Build Coastguard Worker 	long h = 0;
45*de1e4e89SAndroid Build Coastguard Worker 
46*de1e4e89SAndroid Build Coastguard Worker 	if (handle) {
47*de1e4e89SAndroid Build Coastguard Worker 		h = strtol(handle, NULL, 0);
48*de1e4e89SAndroid Build Coastguard Worker 		if (h == LONG_MIN || h == LONG_MAX) {
49*de1e4e89SAndroid Build Coastguard Worker 			fprintf(stderr, "Illegal handle \"%s\", must be numeric.\n",
50*de1e4e89SAndroid Build Coastguard Worker 			    handle);
51*de1e4e89SAndroid Build Coastguard Worker 			return -1;
52*de1e4e89SAndroid Build Coastguard Worker 		}
53*de1e4e89SAndroid Build Coastguard Worker 	}
54*de1e4e89SAndroid Build Coastguard Worker 	t->tcm_handle = h;
55*de1e4e89SAndroid Build Coastguard Worker 
56*de1e4e89SAndroid Build Coastguard Worker 	if (argc == 0)
57*de1e4e89SAndroid Build Coastguard Worker 		return 0;
58*de1e4e89SAndroid Build Coastguard Worker 
59*de1e4e89SAndroid Build Coastguard Worker 	tail = (struct rtattr *)(((void *)n)+NLMSG_ALIGN(n->nlmsg_len));
60*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, MAX_MSG, TCA_OPTIONS, NULL, 0);
61*de1e4e89SAndroid Build Coastguard Worker 
62*de1e4e89SAndroid Build Coastguard Worker 	while (argc > 0) {
63*de1e4e89SAndroid Build Coastguard Worker 		if (matches(*argv, "classid") == 0 ||
64*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "flowid") == 0) {
65*de1e4e89SAndroid Build Coastguard Worker 			unsigned int handle;
66*de1e4e89SAndroid Build Coastguard Worker 
67*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
68*de1e4e89SAndroid Build Coastguard Worker 			if (get_tc_classid(&handle, *argv)) {
69*de1e4e89SAndroid Build Coastguard Worker 				fprintf(stderr, "Illegal \"classid\"\n");
70*de1e4e89SAndroid Build Coastguard Worker 				return -1;
71*de1e4e89SAndroid Build Coastguard Worker 			}
72*de1e4e89SAndroid Build Coastguard Worker 			addattr_l(n, MAX_MSG, TCA_MATCHALL_CLASSID, &handle, 4);
73*de1e4e89SAndroid Build Coastguard Worker 		} else if (matches(*argv, "action") == 0) {
74*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
75*de1e4e89SAndroid Build Coastguard Worker 			if (parse_action(&argc, &argv, TCA_MATCHALL_ACT, n)) {
76*de1e4e89SAndroid Build Coastguard Worker 				fprintf(stderr, "Illegal \"action\"\n");
77*de1e4e89SAndroid Build Coastguard Worker 				return -1;
78*de1e4e89SAndroid Build Coastguard Worker 			}
79*de1e4e89SAndroid Build Coastguard Worker 			continue;
80*de1e4e89SAndroid Build Coastguard Worker 
81*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "skip_hw") == 0) {
82*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
83*de1e4e89SAndroid Build Coastguard Worker 			flags |= TCA_CLS_FLAGS_SKIP_HW;
84*de1e4e89SAndroid Build Coastguard Worker 			continue;
85*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "skip_sw") == 0) {
86*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
87*de1e4e89SAndroid Build Coastguard Worker 			flags |= TCA_CLS_FLAGS_SKIP_SW;
88*de1e4e89SAndroid Build Coastguard Worker 			continue;
89*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "help") == 0) {
90*de1e4e89SAndroid Build Coastguard Worker 			explain();
91*de1e4e89SAndroid Build Coastguard Worker 			return -1;
92*de1e4e89SAndroid Build Coastguard Worker 		} else {
93*de1e4e89SAndroid Build Coastguard Worker 			fprintf(stderr, "What is \"%s\"?\n", *argv);
94*de1e4e89SAndroid Build Coastguard Worker 			explain();
95*de1e4e89SAndroid Build Coastguard Worker 			return -1;
96*de1e4e89SAndroid Build Coastguard Worker 		}
97*de1e4e89SAndroid Build Coastguard Worker 		argc--; argv++;
98*de1e4e89SAndroid Build Coastguard Worker 	}
99*de1e4e89SAndroid Build Coastguard Worker 
100*de1e4e89SAndroid Build Coastguard Worker 	if (flags) {
101*de1e4e89SAndroid Build Coastguard Worker 		if (!(flags ^ (TCA_CLS_FLAGS_SKIP_HW |
102*de1e4e89SAndroid Build Coastguard Worker 			       TCA_CLS_FLAGS_SKIP_SW))) {
103*de1e4e89SAndroid Build Coastguard Worker 			fprintf(stderr,
104*de1e4e89SAndroid Build Coastguard Worker 				"skip_hw and skip_sw are mutually exclusive\n");
105*de1e4e89SAndroid Build Coastguard Worker 			return -1;
106*de1e4e89SAndroid Build Coastguard Worker 		}
107*de1e4e89SAndroid Build Coastguard Worker 		addattr_l(n, MAX_MSG, TCA_MATCHALL_FLAGS, &flags, 4);
108*de1e4e89SAndroid Build Coastguard Worker 	}
109*de1e4e89SAndroid Build Coastguard Worker 
110*de1e4e89SAndroid Build Coastguard Worker 	tail->rta_len = (((void *)n)+n->nlmsg_len) - (void *)tail;
111*de1e4e89SAndroid Build Coastguard Worker 	return 0;
112*de1e4e89SAndroid Build Coastguard Worker }
113*de1e4e89SAndroid Build Coastguard Worker 
matchall_print_opt(struct filter_util * qu,FILE * f,struct rtattr * opt,__u32 handle)114*de1e4e89SAndroid Build Coastguard Worker static int matchall_print_opt(struct filter_util *qu, FILE *f,
115*de1e4e89SAndroid Build Coastguard Worker 			   struct rtattr *opt, __u32 handle)
116*de1e4e89SAndroid Build Coastguard Worker {
117*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *tb[TCA_MATCHALL_MAX+1];
118*de1e4e89SAndroid Build Coastguard Worker 
119*de1e4e89SAndroid Build Coastguard Worker 	if (opt == NULL)
120*de1e4e89SAndroid Build Coastguard Worker 		return 0;
121*de1e4e89SAndroid Build Coastguard Worker 
122*de1e4e89SAndroid Build Coastguard Worker 	parse_rtattr_nested(tb, TCA_MATCHALL_MAX, opt);
123*de1e4e89SAndroid Build Coastguard Worker 
124*de1e4e89SAndroid Build Coastguard Worker 	if (handle)
125*de1e4e89SAndroid Build Coastguard Worker 		fprintf(f, "handle 0x%x ", handle);
126*de1e4e89SAndroid Build Coastguard Worker 
127*de1e4e89SAndroid Build Coastguard Worker 	if (tb[TCA_MATCHALL_CLASSID]) {
128*de1e4e89SAndroid Build Coastguard Worker 		SPRINT_BUF(b1);
129*de1e4e89SAndroid Build Coastguard Worker 		fprintf(f, "flowid %s ",
130*de1e4e89SAndroid Build Coastguard Worker 			sprint_tc_classid(rta_getattr_u32(tb[TCA_MATCHALL_CLASSID]), b1));
131*de1e4e89SAndroid Build Coastguard Worker 	}
132*de1e4e89SAndroid Build Coastguard Worker 
133*de1e4e89SAndroid Build Coastguard Worker 	if (tb[TCA_MATCHALL_FLAGS]) {
134*de1e4e89SAndroid Build Coastguard Worker 		__u32 flags = rta_getattr_u32(tb[TCA_MATCHALL_FLAGS]);
135*de1e4e89SAndroid Build Coastguard Worker 
136*de1e4e89SAndroid Build Coastguard Worker 		if (flags & TCA_CLS_FLAGS_SKIP_HW)
137*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "\n  skip_hw");
138*de1e4e89SAndroid Build Coastguard Worker 		if (flags & TCA_CLS_FLAGS_SKIP_SW)
139*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "\n  skip_sw");
140*de1e4e89SAndroid Build Coastguard Worker 
141*de1e4e89SAndroid Build Coastguard Worker 		if (flags & TCA_CLS_FLAGS_IN_HW)
142*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "\n  in_hw");
143*de1e4e89SAndroid Build Coastguard Worker 		else if (flags & TCA_CLS_FLAGS_NOT_IN_HW)
144*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "\n  not_in_hw");
145*de1e4e89SAndroid Build Coastguard Worker 	}
146*de1e4e89SAndroid Build Coastguard Worker 
147*de1e4e89SAndroid Build Coastguard Worker 	if (tb[TCA_MATCHALL_ACT])
148*de1e4e89SAndroid Build Coastguard Worker 		tc_print_action(f, tb[TCA_MATCHALL_ACT], 0);
149*de1e4e89SAndroid Build Coastguard Worker 
150*de1e4e89SAndroid Build Coastguard Worker 	return 0;
151*de1e4e89SAndroid Build Coastguard Worker }
152*de1e4e89SAndroid Build Coastguard Worker 
153*de1e4e89SAndroid Build Coastguard Worker struct filter_util matchall_filter_util = {
154*de1e4e89SAndroid Build Coastguard Worker 	.id = "matchall",
155*de1e4e89SAndroid Build Coastguard Worker 	.parse_fopt = matchall_parse_opt,
156*de1e4e89SAndroid Build Coastguard Worker 	.print_fopt = matchall_print_opt,
157*de1e4e89SAndroid Build Coastguard Worker };
158