@@ -76,8 +76,9 @@ pedit - generic packet editor action
.BR clear " | " invert " | " set
.IR VAL " | "
.BR add
.IR VAL " | "
+.BR decrement " | "
.BR preserve " } [ " retain
.IR RVAL " ]"
.ti -8
@@ -95,9 +96,9 @@ chosen automatically based on the header field size.
.TP
.B ex
Use extended pedit.
.I EXTENDED_LAYERED_OP
-and the add
+and the add/decrement
.I CMD_SPEC
are allowed only in this mode.
.TP
.BI offset " OFFSET " "\fR{ \fBu32 \fR| \fBu16 \fR| \fBu8 \fR}"
@@ -287,8 +288,13 @@ Add the addressed data by a specific value. The size of
is defined by the size of the addressed header field in
.IR EXTENDED_LAYERED_OP .
This operation is supported only for extended layered op.
.TP
+.BI decrement
+Decrement the addressed data by one.
+This operation is supported only for
+.BR ip " " ttl " and " ip6 " " hoplimit "."
+.TP
.B preserve
Keep the addressed data as is.
.TP
.BI retain " RVAL"
@@ -339,8 +339,11 @@ int parse_cmd(int *argc_p, char ***argv_p, __u32 len, int type, __u32 retain,
__u32 o = 0xFF;
int res = -1;
int argc = *argc_p;
char **argv = *argv_p;
+ int flags = type;
+
+ type &= 0xff; /* strip flags */
if (argc <= 0)
return -1;
@@ -359,17 +362,26 @@ int parse_cmd(int *argc_p, char ***argv_p, __u32 len, int type, __u32 retain,
matches(*argv, "add") == 0) {
if (matches(*argv, "add") == 0)
tkey->cmd = TCA_PEDIT_KEY_EX_CMD_ADD;
- if (!sel->extended && tkey->cmd) {
- fprintf(stderr,
- "Non extended mode. only 'set' command is supported\n");
- return -1;
- }
+ if (!sel->extended && tkey->cmd)
+ goto non_ext_only_set_cmd;
NEXT_ARG();
if (parse_val(&argc, &argv, val, type))
return -1;
+ } else if (matches(*argv, "decrement") == 0) {
+ if ((flags & TFLAG_ALLOW_DEC) == 0) {
+ fprintf(stderr,
+ "decrement command is not supported for this field\n");
+ return -1;
+ }
+
+ if (!sel->extended)
+ goto non_ext_only_set_cmd;
+
+ tkey->cmd = TCA_PEDIT_KEY_EX_CMD_ADD;
+ *v = retain; /* decrement by overflow */
} else if (matches(*argv, "preserve") == 0) {
retain = 0;
} else {
if (matches(*argv, "clear") != 0)
@@ -422,16 +434,21 @@ int parse_cmd(int *argc_p, char ***argv_p, __u32 len, int type, __u32 retain,
goto done;
}
return -1;
+
done:
if (pedit_debug)
printf("parse_cmd done argc %d %s offset %d length %d\n",
argc, *argv, tkey->off, len);
*argc_p = argc;
*argv_p = argv;
return res;
+non_ext_only_set_cmd:
+ fprintf(stderr,
+ "Non extended mode. only 'set' command is supported\n");
+ return -1;
}
static int parse_offset(int *argc_p, char ***argv_p, struct m_pedit_sel *sel,
struct m_pedit_key *tkey)
@@ -32,8 +32,10 @@
#define TINT 3
#define TU32 4
#define TMAC 5
+#define TFLAG_ALLOW_DEC (1<<8)
+
#define RU32 0xFFFFFFFF
#define RU16 0xFFFF
#define RU8 0xFF
@@ -67,9 +67,9 @@ parse_ip(int *argc_p, char ***argv_p,
}
if (strcmp(*argv, "ttl") == 0) {
NEXT_ARG();
tkey->off = 8;
- res = parse_cmd(&argc, &argv, 1, TU32, RU8, sel, tkey);
+ res = parse_cmd(&argc, &argv, 1, TU32 | TFLAG_ALLOW_DEC, RU8, sel, tkey);
goto done;
}
if (strcmp(*argv, "protocol") == 0) {
NEXT_ARG();
@@ -70,9 +70,9 @@ parse_ip6(int *argc_p, char ***argv_p,
}
if (strcmp(*argv, "hoplimit") == 0) {
NEXT_ARG();
tkey->off = 7;
- res = parse_cmd(&argc, &argv, 1, TU32, RU8, sel, tkey);
+ res = parse_cmd(&argc, &argv, 1, TU32 | TFLAG_ALLOW_DEC, RU8, sel, tkey);
goto done;
}
if (strcmp(*argv, "traffic_class") == 0) {
NEXT_ARG();
Implement a decrement operation for ttl and hoplimit. Since this is just syntactic sugar, it goes that: tc filter add ... action pedit ex munge ip ttl dec ... tc filter add ... action pedit ex munge ip6 hoplimit dec ... is just a more readable version of this: tc filter add ... action pedit ex munge ip ttl add 0xff ... tc filter add ... action pedit ex munge ip6 hoplimit add 0xff ... This feature was suggested by some pseudo tc examples in Mellanox's documentation[1], but wasn't present in neither their mlnx-iproute2 nor iproute2. In order to avoid adding an extra parameter to parse_cmd(), I have re-used the `int type` parameter to also carry flags. Tested with skip_sw on Mellanox ConnectX-6 Dx. [1] https://docs.mellanox.com/pages/viewpage.action?pageId=47033989 Signed-off-by: Asbjørn Sloth Tønnesen <asbjorn@asbjorn.st> --- man/man8/tc-pedit.8 | 8 +++++++- tc/m_pedit.c | 27 ++++++++++++++++++++++----- tc/m_pedit.h | 2 ++ tc/p_ip.c | 2 +- tc/p_ip6.c | 2 +- 5 files changed, 33 insertions(+), 8 deletions(-)