2017-05-14 16:17:46 +08:00
|
|
|
/*
|
|
|
|
* p_ip6.c packet editor: IPV6 header
|
|
|
|
*
|
|
|
|
* This program is free software; you can distribute 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: Amir Vadai <amir@vadai.me>
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "utils.h"
|
|
|
|
#include "tc_util.h"
|
|
|
|
#include "m_pedit.h"
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_ip6(int *argc_p, char ***argv_p,
|
|
|
|
struct m_pedit_sel *sel, struct m_pedit_key *tkey)
|
|
|
|
{
|
|
|
|
int res = -1;
|
|
|
|
int argc = *argc_p;
|
|
|
|
char **argv = *argv_p;
|
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!sel->extended)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
tkey->htype = TCA_PEDIT_KEY_EX_HDR_TYPE_IP6;
|
|
|
|
|
|
|
|
if (strcmp(*argv, "src") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 8;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 16, TIPV6, RU32, sel, tkey, 0);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (strcmp(*argv, "dst") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 24;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 16, TIPV6, RU32, sel, tkey, 0);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (strcmp(*argv, "flow_lbl") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 0;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 4, TU32, 0x0007ffff, sel, tkey, 0);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (strcmp(*argv, "payload_len") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 4;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 2, TU32, RU16, sel, tkey, 0);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (strcmp(*argv, "nexthdr") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 6;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 1, TU32, RU8, sel, tkey, 0);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (strcmp(*argv, "hoplimit") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 7;
|
2021-06-19 00:06:35 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 1, TU32, RU8, sel, tkey, PEDIT_ALLOW_DEC);
|
2017-05-14 16:17:46 +08:00
|
|
|
goto done;
|
|
|
|
}
|
2020-04-04 07:05:29 +08:00
|
|
|
if (strcmp(*argv, "traffic_class") == 0) {
|
|
|
|
NEXT_ARG();
|
|
|
|
tkey->off = 1;
|
2021-06-19 00:06:34 +08:00
|
|
|
res = parse_cmd(&argc, &argv, 1, TU32, RU8, sel, tkey, 0);
|
2020-04-04 07:05:29 +08:00
|
|
|
|
|
|
|
/* Shift the field by 4 bits on success. */
|
|
|
|
if (!res) {
|
|
|
|
int nkeys = sel->sel.nkeys;
|
2020-11-30 01:40:49 +08:00
|
|
|
struct tc_pedit_key *key = &sel->keys[nkeys - 1];
|
2020-04-04 07:05:29 +08:00
|
|
|
|
|
|
|
key->mask = htonl(ntohl(key->mask) << 4 | 0xf);
|
|
|
|
key->val = htonl(ntohl(key->val) << 4);
|
|
|
|
}
|
|
|
|
goto done;
|
|
|
|
}
|
2017-05-14 16:17:46 +08:00
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
done:
|
|
|
|
*argc_p = argc;
|
|
|
|
*argv_p = argv;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct m_pedit_util p_pedit_ip6 = {
|
2019-02-27 20:10:17 +08:00
|
|
|
.id = "ip6",
|
2018-11-16 06:36:39 +08:00
|
|
|
.parse_peopt = parse_ip6,
|
2017-05-14 16:17:46 +08:00
|
|
|
};
|