blob: b062c63822276d46bbc8bfa3b003ece4caf1a92c [file] [log] [blame]
Harald Welte2e7377d2002-02-17 19:54:42 +00001/* Shared library add-on to iptables for DSCP
2 *
3 * (C) 2000- 2002 by Matthew G. Marsh <mgm@paktronix.com>,
4 * Harald Welte <laforge@gnumonks.org>
5 *
6 * This program is distributed under the terms of GNU GPL v2, 1991
7 *
8 * libipt_DSCP.c borrowed heavily from libipt_TOS.c
9 *
Iain Barnesdf5e13f2002-04-11 10:48:53 +000010 * --set-class added by Iain Barnes
Harald Welte2e7377d2002-02-17 19:54:42 +000011 */
12#include <stdio.h>
13#include <string.h>
14#include <stdlib.h>
15#include <getopt.h>
16
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +000017#include <xtables.h>
18#include <linux/netfilter/x_tables.h>
19#include <linux/netfilter/xt_DSCP.h>
Harald Welte2e7377d2002-02-17 19:54:42 +000020
Iain Barnes0ddae8f2002-06-21 17:35:55 +000021/* This is evil, but it's my code - HW*/
22#include "libipt_dscp_helper.c"
Iain Barnesdf5e13f2002-04-11 10:48:53 +000023
Iain Barnesdf5e13f2002-04-11 10:48:53 +000024
Yasuyuki KOZAKAI193df8e2007-07-24 05:57:28 +000025static void init(struct xt_entry_target *t, unsigned int *nfcache)
Harald Welte2e7377d2002-02-17 19:54:42 +000026{
27}
28
29static void help(void)
30{
31 printf(
32"DSCP target options\n"
33" --set-dscp value Set DSCP field in packet header to value\n"
34" This value can be in decimal (ex: 32)\n"
35" or in hex (ex: 0x20)\n"
Harald Weltea49ded02002-08-07 09:55:37 +000036" --set-dscp-class class Set the DSCP field in packet header to the\n"
37" value represented by the DiffServ class value.\n"
Maciej Soltysiak920463d2004-03-04 00:14:03 +000038" This class may be EF,BE or any of the CSxx\n"
Iain Barnesdf5e13f2002-04-11 10:48:53 +000039" or AFxx classes.\n"
40"\n"
41" These two options are mutually exclusive !\n"
Harald Welte2e7377d2002-02-17 19:54:42 +000042);
43}
44
Jan Engelhardt661f1122007-07-30 14:46:51 +000045static const struct option opts[] = {
Harald Welte2e7377d2002-02-17 19:54:42 +000046 { "set-dscp", 1, 0, 'F' },
Harald Weltea49ded02002-08-07 09:55:37 +000047 { "set-dscp-class", 1, 0, 'G' },
Harald Welte2e7377d2002-02-17 19:54:42 +000048 { 0 }
49};
50
51static void
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +000052parse_dscp(const char *s, struct xt_DSCP_info *dinfo)
Harald Welte2e7377d2002-02-17 19:54:42 +000053{
54 unsigned int dscp;
55
56 if (string_to_number(s, 0, 255, &dscp) == -1)
57 exit_error(PARAMETER_PROBLEM,
58 "Invalid dscp `%s'\n", s);
59
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +000060 if (dscp > XT_DSCP_MAX)
Harald Welte2e7377d2002-02-17 19:54:42 +000061 exit_error(PARAMETER_PROBLEM,
62 "DSCP `%d` out of range\n", dscp);
63
Harald Welteed18bad2002-02-17 21:28:51 +000064 dinfo->dscp = (u_int8_t )dscp;
Harald Welte2e7377d2002-02-17 19:54:42 +000065 return;
66}
67
Iain Barnesdf5e13f2002-04-11 10:48:53 +000068
69static void
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +000070parse_class(const char *s, struct xt_DSCP_info *dinfo)
Iain Barnesdf5e13f2002-04-11 10:48:53 +000071{
Iain Barnes0ddae8f2002-06-21 17:35:55 +000072 unsigned int dscp = class_to_dscp(s);
Iain Barnesdf5e13f2002-04-11 10:48:53 +000073
Iain Barnes0ddae8f2002-06-21 17:35:55 +000074 /* Assign the value */
75 dinfo->dscp = (u_int8_t)dscp;
Iain Barnesdf5e13f2002-04-11 10:48:53 +000076}
77
78
Harald Welte2e7377d2002-02-17 19:54:42 +000079static int
80parse(int c, char **argv, int invert, unsigned int *flags,
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +000081 const void *entry,
Yasuyuki KOZAKAI193df8e2007-07-24 05:57:28 +000082 struct xt_entry_target **target)
Harald Welte2e7377d2002-02-17 19:54:42 +000083{
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +000084 struct xt_DSCP_info *dinfo
85 = (struct xt_DSCP_info *)(*target)->data;
Harald Welte2e7377d2002-02-17 19:54:42 +000086
87 switch (c) {
88 case 'F':
89 if (*flags)
90 exit_error(PARAMETER_PROBLEM,
91 "DSCP target: Only use --set-dscp ONCE!");
Harald Welteed18bad2002-02-17 21:28:51 +000092 parse_dscp(optarg, dinfo);
Harald Welte2e7377d2002-02-17 19:54:42 +000093 *flags = 1;
94 break;
Iain Barnesdf5e13f2002-04-11 10:48:53 +000095 case 'G':
96 if (*flags)
97 exit_error(PARAMETER_PROBLEM,
Harald Weltea49ded02002-08-07 09:55:37 +000098 "DSCP target: Only use --set-dscp-class ONCE!");
Iain Barnesdf5e13f2002-04-11 10:48:53 +000099 parse_class(optarg, dinfo);
100 *flags = 1;
101 break;
Harald Welte2e7377d2002-02-17 19:54:42 +0000102
103 default:
104 return 0;
105 }
106
107 return 1;
108}
109
110static void
111final_check(unsigned int flags)
112{
113 if (!flags)
114 exit_error(PARAMETER_PROBLEM,
115 "DSCP target: Parameter --set-dscp is required");
116}
117
118static void
Harald Welteed18bad2002-02-17 21:28:51 +0000119print_dscp(u_int8_t dscp, int numeric)
Harald Welte2e7377d2002-02-17 19:54:42 +0000120{
121 printf("0x%02x ", dscp);
122}
123
124/* Prints out the targinfo. */
125static void
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +0000126print(const void *ip,
Yasuyuki KOZAKAI193df8e2007-07-24 05:57:28 +0000127 const struct xt_entry_target *target,
Harald Welte2e7377d2002-02-17 19:54:42 +0000128 int numeric)
129{
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000130 const struct xt_DSCP_info *dinfo =
131 (const struct xt_DSCP_info *)target->data;
Harald Welte2e7377d2002-02-17 19:54:42 +0000132 printf("DSCP set ");
Harald Welteed18bad2002-02-17 21:28:51 +0000133 print_dscp(dinfo->dscp, numeric);
Harald Welte2e7377d2002-02-17 19:54:42 +0000134}
135
136/* Saves the union ipt_targinfo in parsable form to stdout. */
137static void
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +0000138save(const void *ip, const struct xt_entry_target *target)
Harald Welte2e7377d2002-02-17 19:54:42 +0000139{
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000140 const struct xt_DSCP_info *dinfo =
141 (const struct xt_DSCP_info *)target->data;
Harald Welte2e7377d2002-02-17 19:54:42 +0000142
Harald Welteed18bad2002-02-17 21:28:51 +0000143 printf("--set-dscp 0x%02x ", dinfo->dscp);
Harald Welte2e7377d2002-02-17 19:54:42 +0000144}
145
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000146static struct xtables_target dscp = {
147 .family = AF_INET,
Pablo Neira8caee8b2004-12-28 13:11:59 +0000148 .name = "DSCP",
149 .version = IPTABLES_VERSION,
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000150 .size = XT_ALIGN(sizeof(struct xt_DSCP_info)),
151 .userspacesize = XT_ALIGN(sizeof(struct xt_DSCP_info)),
Pablo Neira8caee8b2004-12-28 13:11:59 +0000152 .help = &help,
153 .init = &init,
154 .parse = &parse,
155 .final_check = &final_check,
156 .print = &print,
157 .save = &save,
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000158 .extra_opts = opts,
159};
160
161static struct xtables_target dscp6 = {
162 .family = AF_INET6,
163 .name = "DSCP",
164 .version = IPTABLES_VERSION,
165 .size = XT_ALIGN(sizeof(struct xt_DSCP_info)),
166 .userspacesize = XT_ALIGN(sizeof(struct xt_DSCP_info)),
167 .help = &help,
168 .init = &init,
169 .parse = &parse,
170 .final_check = &final_check,
171 .print = &print,
172 .save = &save,
173 .extra_opts = opts,
Harald Welte2e7377d2002-02-17 19:54:42 +0000174};
175
176void _init(void)
177{
Yasuyuki KOZAKAIa7bf6d02007-08-04 08:24:29 +0000178 xtables_register_target(&dscp);
179 xtables_register_target(&dscp6);
Harald Welte2e7377d2002-02-17 19:54:42 +0000180}