4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
9 * Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
17 #include <sys/socket.h>
18 #include <netinet/in.h>
19 #include <arpa/inet.h>
21 #include <linux/if.h> /* IFNAMSIZ */
25 static void explain(void)
28 "Usage: ... fw [ classid CLASSID ] [ indev DEV ] [ action ACTION_SPEC ]\n");
30 " CLASSID := Push matching packets to the class identified by CLASSID with format X:Y\n");
32 " CLASSID is parsed as hexadecimal input.\n");
34 " DEV := specify device for incoming device classification.\n");
36 " ACTION_SPEC := Apply an action on matching packets.\n");
38 " NOTE: handle is represented as HANDLE[/FWMASK].\n");
39 fprintf(stderr
, " FWMASK is 0xffffffff by default.\n");
42 static int fw_parse_opt(struct filter_util
*qu
, char *handle
, int argc
, char **argv
, struct nlmsghdr
*n
)
44 struct tcmsg
*t
= NLMSG_DATA(n
);
52 if ((slash
= strchr(handle
, '/')) != NULL
)
54 if (get_u32(&t
->tcm_handle
, handle
, 0)) {
55 fprintf(stderr
, "Illegal \"handle\"\n");
59 if (get_u32(&mask
, slash
+1, 0)) {
60 fprintf(stderr
, "Illegal \"handle\" mask\n");
71 addattr_l(n
, 4096, TCA_OPTIONS
, NULL
, 0);
74 addattr32(n
, MAX_MSG
, TCA_FW_MASK
, mask
);
77 if (matches(*argv
, "classid") == 0 ||
78 matches(*argv
, "flowid") == 0) {
82 if (get_tc_classid(&handle
, *argv
)) {
83 fprintf(stderr
, "Illegal \"classid\"\n");
86 addattr_l(n
, 4096, TCA_FW_CLASSID
, &handle
, 4);
87 } else if (matches(*argv
, "police") == 0) {
89 if (parse_police(&argc
, &argv
, TCA_FW_POLICE
, n
)) {
90 fprintf(stderr
, "Illegal \"police\"\n");
94 } else if (matches(*argv
, "action") == 0) {
96 if (parse_action(&argc
, &argv
, TCA_FW_ACT
, n
)) {
97 fprintf(stderr
, "Illegal fw \"action\"\n");
101 } else if (strcmp(*argv
, "indev") == 0) {
102 char d
[IFNAMSIZ
+1] = {};
107 fprintf(stderr
, "Illegal indev\n");
110 strncpy(d
, *argv
, sizeof(d
) - 1);
111 addattr_l(n
, MAX_MSG
, TCA_FW_INDEV
, d
, strlen(d
) + 1);
112 } else if (strcmp(*argv
, "help") == 0) {
116 fprintf(stderr
, "What is \"%s\"?\n", *argv
);
122 tail
->rta_len
= (void *) NLMSG_TAIL(n
) - (void *) tail
;
126 static int fw_print_opt(struct filter_util
*qu
, FILE *f
, struct rtattr
*opt
, __u32 handle
)
128 struct rtattr
*tb
[TCA_FW_MAX
+1];
133 parse_rtattr_nested(tb
, TCA_FW_MAX
, opt
);
135 if (handle
|| tb
[TCA_FW_MASK
]) {
136 __u32 mark
= 0, mask
= 0;
140 if (tb
[TCA_FW_MASK
] &&
141 (mask
= rta_getattr_u32(tb
[TCA_FW_MASK
])) != 0xFFFFFFFF)
142 fprintf(f
, "handle 0x%x/0x%x ", mark
, mask
);
144 fprintf(f
, "handle 0x%x ", handle
);
147 if (tb
[TCA_FW_CLASSID
]) {
149 fprintf(f
, "classid %s ", sprint_tc_classid(rta_getattr_u32(tb
[TCA_FW_CLASSID
]), b1
));
152 if (tb
[TCA_FW_POLICE
])
153 tc_print_police(f
, tb
[TCA_FW_POLICE
]);
154 if (tb
[TCA_FW_INDEV
]) {
155 struct rtattr
*idev
= tb
[TCA_FW_INDEV
];
157 fprintf(f
, "input dev %s ", rta_getattr_str(idev
));
160 if (tb
[TCA_FW_ACT
]) {
162 tc_print_action(f
, tb
[TCA_FW_ACT
], 0);
167 struct filter_util fw_filter_util
= {
169 .parse_fopt
= fw_parse_opt
,
170 .print_fopt
= fw_print_opt
,