Blame SOURCES/0059-exthdr-Implement-SCTP-Chunk-matching.patch

4e0227
From 5a8d6197929e30520bb3839c9165d89930888daf Mon Sep 17 00:00:00 2001
4e0227
From: Phil Sutter <psutter@redhat.com>
4e0227
Date: Tue, 13 Jul 2021 13:54:42 +0200
4e0227
Subject: [PATCH] exthdr: Implement SCTP Chunk matching
4e0227
4e0227
Bugzilla: https://bugzilla.redhat.com/show_bug.cgi?id=1979334
4e0227
Upstream Status: nftables commit 0e3871cfd9a1e
4e0227
Conflicts:
4e0227
 * Context changes due to missing other scopes.
4e0227
 * Context change due to missing commit 6e6ef00028f1c
4e0227
   ("exthdr: remove tcp dependency for tcp option matching").
4e0227
4e0227
commit 0e3871cfd9a1e32a4ac041ce87a8057b11a89924
4e0227
Author: Phil Sutter <phil@nwl.cc>
4e0227
Date:   Tue May 4 13:41:38 2021 +0200
4e0227
4e0227
    exthdr: Implement SCTP Chunk matching
4e0227
4e0227
    Extend exthdr expression to support scanning through SCTP packet chunks
4e0227
    and matching on fixed fields' values.
4e0227
4e0227
    Signed-off-by: Phil Sutter <phil@nwl.cc>
4e0227
    Acked-by: Florian Westphal <fw@strlen.de>
4e0227
---
4e0227
 doc/libnftables-json.adoc           |  13 +
4e0227
 doc/payload-expression.txt          |  53 +++
4e0227
 include/linux/netfilter/nf_tables.h |   2 +
4e0227
 include/parser.h                    |   1 +
4e0227
 include/sctp_chunk.h                |  87 +++++
4e0227
 src/Makefile.am                     |   1 +
4e0227
 src/evaluate.c                      |   2 +
4e0227
 src/exthdr.c                        |   8 +
4e0227
 src/json.c                          |   2 +
4e0227
 src/parser_bison.y                  | 148 ++++++++-
4e0227
 src/parser_json.c                   |  49 +++
4e0227
 src/scanner.l                       |  38 +++
4e0227
 src/sctp_chunk.c                    | 261 +++++++++++++++
4e0227
 tests/py/inet/sctp.t                |  37 +++
4e0227
 tests/py/inet/sctp.t.json           | 478 ++++++++++++++++++++++++++++
4e0227
 tests/py/inet/sctp.t.payload        | 155 +++++++++
4e0227
 16 files changed, 1333 insertions(+), 2 deletions(-)
4e0227
 create mode 100644 include/sctp_chunk.h
4e0227
 create mode 100644 src/sctp_chunk.c
4e0227
4e0227
diff --git a/doc/libnftables-json.adoc b/doc/libnftables-json.adoc
4e0227
index 858abbf..fba4cb0 100644
4e0227
--- a/doc/libnftables-json.adoc
4e0227
+++ b/doc/libnftables-json.adoc
4e0227
@@ -1200,6 +1200,19 @@ Create a reference to a field (*field*) of a TCP option header (*name*).
4e0227
 If the *field* property is not given, the expression is to be used as a TCP option
4e0227
 existence check in a *match* statement with a boolean on the right hand side.
4e0227
 
4e0227
+=== SCTP CHUNK
4e0227
+[verse]
4e0227
+*{ "sctp chunk": {
4e0227
+	"name":* 'STRING'*,
4e0227
+	"field":* 'STRING'
4e0227
+*}}*
4e0227
+
4e0227
+Create a reference to a field (*field*) of an SCTP chunk (*name*).
4e0227
+
4e0227
+If the *field* property is not given, the expression is to be used as an SCTP
4e0227
+chunk existence check in a *match* statement with a boolean on the right hand
4e0227
+side.
4e0227
+
4e0227
 === META
4e0227
 [verse]
4e0227
 ____
4e0227
diff --git a/doc/payload-expression.txt b/doc/payload-expression.txt
4e0227
index b6d2a28..bd03ca2 100644
4e0227
--- a/doc/payload-expression.txt
4e0227
+++ b/doc/payload-expression.txt
4e0227
@@ -369,7 +369,33 @@ integer (16 bit)
4e0227
 SCTP HEADER EXPRESSION
4e0227
 ~~~~~~~~~~~~~~~~~~~~~~~
4e0227
 [verse]
4e0227
+____
4e0227
 *sctp* {*sport* | *dport* | *vtag* | *checksum*}
4e0227
+*sctp chunk* 'CHUNK' [ 'FIELD' ]
4e0227
+
4e0227
+'CHUNK' := *data* | *init* | *init-ack* | *sack* | *heartbeat* |
4e0227
+	   *heartbeat-ack* | *abort* | *shutdown* | *shutdown-ack* | *error* |
4e0227
+	   *cookie-echo* | *cookie-ack* | *ecne* | *cwr* | *shutdown-complete*
4e0227
+	   | *asconf-ack* | *forward-tsn* | *asconf*
4e0227
+
4e0227
+'FIELD' := 'COMMON_FIELD' | 'DATA_FIELD' | 'INIT_FIELD' | 'INIT_ACK_FIELD' |
4e0227
+	   'SACK_FIELD' | 'SHUTDOWN_FIELD' | 'ECNE_FIELD' | 'CWR_FIELD' |
4e0227
+	   'ASCONF_ACK_FIELD' | 'FORWARD_TSN_FIELD' | 'ASCONF_FIELD'
4e0227
+
4e0227
+'COMMON_FIELD' := *type* | *flags* | *length*
4e0227
+'DATA_FIELD' := *tsn* | *stream* | *ssn* | *ppid*
4e0227
+'INIT_FIELD' := *init-tag* | *a-rwnd* | *num-outbound-streams* |
4e0227
+		*num-inbound-streams* | *initial-tsn*
4e0227
+'INIT_ACK_FIELD' := 'INIT_FIELD'
4e0227
+'SACK_FIELD' := *cum-tsn-ack* | *a-rwnd* | *num-gap-ack-blocks* |
4e0227
+		*num-dup-tsns*
4e0227
+'SHUTDOWN_FIELD' := *cum-tsn-ack*
4e0227
+'ECNE_FIELD' := *lowest-tsn*
4e0227
+'CWR_FIELD' := *lowest-tsn*
4e0227
+'ASCONF_ACK_FIELD' := *seqno*
4e0227
+'FORWARD_TSN_FIELD' := *new-cum-tsn*
4e0227
+'ASCONF_FIELD' := *seqno*
4e0227
+____
4e0227
 
4e0227
 .SCTP header expression
4e0227
 [options="header"]
4e0227
@@ -387,8 +413,35 @@ integer (32 bit)
4e0227
 |checksum|
4e0227
 Checksum|
4e0227
 integer (32 bit)
4e0227
+|chunk|
4e0227
+Search chunk in packet|
4e0227
+without 'FIELD', boolean indicating existence
4e0227
 |================
4e0227
 
4e0227
+.SCTP chunk fields
4e0227
+[options="header"]
4e0227
+|==================
4e0227
+|Name| Width in bits | Chunk | Notes
4e0227
+|type| 8 | all | not useful, defined by chunk type
4e0227
+|flags| 8 | all | semantics defined on per-chunk basis
4e0227
+|length| 16 | all | length of this chunk in bytes excluding padding
4e0227
+|tsn| 32 | data | transmission sequence number
4e0227
+|stream| 16 | data | stream identifier
4e0227
+|ssn| 16 | data | stream sequence number
4e0227
+|ppid| 32 | data | payload protocol identifier
4e0227
+|init-tag| 32 | init, init-ack | initiate tag
4e0227
+|a-rwnd| 32 | init, init-ack, sack | advertised receiver window credit
4e0227
+|num-outbound-streams| 16 | init, init-ack | number of outbound streams
4e0227
+|num-inbound-streams| 16 | init, init-ack | number of inbound streams
4e0227
+|initial-tsn| 32 | init, init-ack | initial transmit sequence number
4e0227
+|cum-tsn-ack| 32 | sack, shutdown | cumulative transmission sequence number acknowledged
4e0227
+|num-gap-ack-blocks| 16 | sack | number of Gap Ack Blocks included
4e0227
+|num-dup-tsns| 16 | sack | number of duplicate transmission sequence numbers received
4e0227
+|lowest-tsn| 32 | ecne, cwr | lowest transmission sequence number
4e0227
+|seqno| 32 | asconf-ack, asconf | sequence number
4e0227
+|new-cum-tsn| 32 | forward-tsn | new cumulative transmission sequence number
4e0227
+|==================
4e0227
+
4e0227
 DCCP HEADER EXPRESSION
4e0227
 ~~~~~~~~~~~~~~~~~~~~~~
4e0227
 [verse]
4e0227
diff --git a/include/linux/netfilter/nf_tables.h b/include/linux/netfilter/nf_tables.h
4e0227
index 1328b8e..960a5b4 100644
4e0227
--- a/include/linux/netfilter/nf_tables.h
4e0227
+++ b/include/linux/netfilter/nf_tables.h
4e0227
@@ -755,11 +755,13 @@ enum nft_exthdr_flags {
4e0227
  * @NFT_EXTHDR_OP_IPV6: match against ipv6 extension headers
4e0227
  * @NFT_EXTHDR_OP_TCP: match against tcp options
4e0227
  * @NFT_EXTHDR_OP_IPV4: match against ipv4 options
4e0227
+ * @NFT_EXTHDR_OP_SCTP: match against sctp chunks
4e0227
  */
4e0227
 enum nft_exthdr_op {
4e0227
 	NFT_EXTHDR_OP_IPV6,
4e0227
 	NFT_EXTHDR_OP_TCPOPT,
4e0227
 	NFT_EXTHDR_OP_IPV4,
4e0227
+	NFT_EXTHDR_OP_SCTP,
4e0227
 	__NFT_EXTHDR_OP_MAX
4e0227
 };
4e0227
 #define NFT_EXTHDR_OP_MAX	(__NFT_EXTHDR_OP_MAX - 1)
4e0227
diff --git a/include/parser.h b/include/parser.h
4e0227
index 2e6ef4d..99bed3c 100644
4e0227
--- a/include/parser.h
4e0227
+++ b/include/parser.h
4e0227
@@ -33,6 +33,7 @@ enum startcond_type {
4e0227
 	PARSER_SC_SCTP,
4e0227
 	PARSER_SC_EXPR_HASH,
4e0227
 	PARSER_SC_EXPR_NUMGEN,
4e0227
+	PARSER_SC_EXPR_SCTP_CHUNK,
4e0227
 };
4e0227
 
4e0227
 struct mnl_socket;
4e0227
diff --git a/include/sctp_chunk.h b/include/sctp_chunk.h
4e0227
new file mode 100644
4e0227
index 0000000..3819200
4e0227
--- /dev/null
4e0227
+++ b/include/sctp_chunk.h
4e0227
@@ -0,0 +1,87 @@
4e0227
+/*
4e0227
+ * Copyright Red Hat
4e0227
+ *
4e0227
+ * This program is free software; you can redistribute it and/or modify
4e0227
+ * it under the terms of the GNU General Public License version 2 (or any
4e0227
+ * later) as published by the Free Software Foundation.
4e0227
+ */
4e0227
+
4e0227
+#ifndef NFTABLES_SCTP_CHUNK_H
4e0227
+#define NFTABLES_SCTP_CHUNK_H
4e0227
+
4e0227
+/* SCTP chunk types used on wire */
4e0227
+enum sctp_hdr_chunk_types {
4e0227
+	SCTP_CHUNK_TYPE_DATA			= 0,
4e0227
+	SCTP_CHUNK_TYPE_INIT			= 1,
4e0227
+	SCTP_CHUNK_TYPE_INIT_ACK		= 2,
4e0227
+	SCTP_CHUNK_TYPE_SACK			= 3,
4e0227
+	SCTP_CHUNK_TYPE_HEARTBEAT		= 4,
4e0227
+	SCTP_CHUNK_TYPE_HEARTBEAT_ACK		= 5,
4e0227
+	SCTP_CHUNK_TYPE_ABORT			= 6,
4e0227
+	SCTP_CHUNK_TYPE_SHUTDOWN		= 7,
4e0227
+	SCTP_CHUNK_TYPE_SHUTDOWN_ACK		= 8,
4e0227
+	SCTP_CHUNK_TYPE_ERROR			= 9,
4e0227
+	SCTP_CHUNK_TYPE_COOKIE_ECHO		= 10,
4e0227
+	SCTP_CHUNK_TYPE_COOKIE_ACK		= 11,
4e0227
+	SCTP_CHUNK_TYPE_ECNE			= 12,
4e0227
+	SCTP_CHUNK_TYPE_CWR			= 13,
4e0227
+	SCTP_CHUNK_TYPE_SHUTDOWN_COMPLETE	= 14,
4e0227
+	SCTP_CHUNK_TYPE_ASCONF_ACK		= 128,
4e0227
+	SCTP_CHUNK_TYPE_FORWARD_TSN		= 192,
4e0227
+	SCTP_CHUNK_TYPE_ASCONF			= 193,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_common_fields {
4e0227
+	SCTP_CHUNK_COMMON_TYPE,
4e0227
+	SCTP_CHUNK_COMMON_FLAGS,
4e0227
+	SCTP_CHUNK_COMMON_LENGTH,
4e0227
+	__SCTP_CHUNK_COMMON_MAX,
4e0227
+};
4e0227
+
4e0227
+#define SCTP_CHUNK_START_INDEX	__SCTP_CHUNK_COMMON_MAX
4e0227
+
4e0227
+enum sctp_hdr_chunk_data_fields {
4e0227
+	SCTP_CHUNK_DATA_TSN = SCTP_CHUNK_START_INDEX,
4e0227
+	SCTP_CHUNK_DATA_STREAM,
4e0227
+	SCTP_CHUNK_DATA_SSN,
4e0227
+	SCTP_CHUNK_DATA_PPID,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_init_fields {
4e0227
+	SCTP_CHUNK_INIT_TAG = SCTP_CHUNK_START_INDEX,
4e0227
+	SCTP_CHUNK_INIT_RWND,
4e0227
+	SCTP_CHUNK_INIT_OSTREAMS,
4e0227
+	SCTP_CHUNK_INIT_ISTREAMS,
4e0227
+	SCTP_CHUNK_INIT_TSN,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_sack_fields {
4e0227
+	SCTP_CHUNK_SACK_CTSN_ACK = SCTP_CHUNK_START_INDEX,
4e0227
+	SCTP_CHUNK_SACK_RWND,
4e0227
+	SCTP_CHUNK_SACK_GACK_BLOCKS,
4e0227
+	SCTP_CHUNK_SACK_DUP_TSNS,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_shutdown_fields {
4e0227
+	SCTP_CHUNK_SHUTDOWN_CTSN_ACK = SCTP_CHUNK_START_INDEX,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_ecne_cwr_fields {
4e0227
+	SCTP_CHUNK_ECNE_CWR_MIN_TSN = SCTP_CHUNK_START_INDEX,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_asconf_fields {
4e0227
+	SCTP_CHUNK_ASCONF_SEQNO = SCTP_CHUNK_START_INDEX,
4e0227
+};
4e0227
+
4e0227
+enum sctp_hdr_chunk_fwd_tsn_fields {
4e0227
+	SCTP_CHUNK_FORWARD_TSN_NCTSN = SCTP_CHUNK_START_INDEX,
4e0227
+};
4e0227
+
4e0227
+struct expr *sctp_chunk_expr_alloc(const struct location *loc,
4e0227
+				   unsigned int type, unsigned int field);
4e0227
+void sctp_chunk_init_raw(struct expr *expr, uint8_t type, unsigned int off,
4e0227
+			 unsigned int len, uint32_t flags);
4e0227
+const struct exthdr_desc *sctp_chunk_protocol_find(const char *name);
4e0227
+
4e0227
+#endif /* NFTABLES_SCTP_CHUNK_H */
4e0227
diff --git a/src/Makefile.am b/src/Makefile.am
4e0227
index 740c21f..366820b 100644
4e0227
--- a/src/Makefile.am
4e0227
+++ b/src/Makefile.am
4e0227
@@ -64,6 +64,7 @@ libnftables_la_SOURCES =			\
4e0227
 		tcpopt.c			\
4e0227
 		socket.c			\
4e0227
 		print.c				\
4e0227
+		sctp_chunk.c			\
4e0227
 		libnftables.c			\
4e0227
 		libnftables.map
4e0227
 
4e0227
diff --git a/src/evaluate.c b/src/evaluate.c
4e0227
index 99a66c2..00ec20b 100644
4e0227
--- a/src/evaluate.c
4e0227
+++ b/src/evaluate.c
4e0227
@@ -579,6 +579,8 @@ static int expr_evaluate_exthdr(struct eval_ctx *ctx, struct expr **exprp)
4e0227
 		dependency = &proto_tcp;
4e0227
 		pb = PROTO_BASE_TRANSPORT_HDR;
4e0227
 		break;
4e0227
+	case NFT_EXTHDR_OP_SCTP:
4e0227
+		return __expr_evaluate_exthdr(ctx, exprp);
4e0227
 	case NFT_EXTHDR_OP_IPV4:
4e0227
 		dependency = &proto_ip;
4e0227
 		break;
4e0227
diff --git a/src/exthdr.c b/src/exthdr.c
4e0227
index 5c75720..f5689e7 100644
4e0227
--- a/src/exthdr.c
4e0227
+++ b/src/exthdr.c
4e0227
@@ -22,6 +22,7 @@
4e0227
 #include <headers.h>
4e0227
 #include <expression.h>
4e0227
 #include <statement.h>
4e0227
+#include <sctp_chunk.h>
4e0227
 
4e0227
 static void exthdr_expr_print(const struct expr *expr, struct output_ctx *octx)
4e0227
 {
4e0227
@@ -55,6 +56,11 @@ static void exthdr_expr_print(const struct expr *expr, struct output_ctx *octx)
4e0227
 		if (expr->exthdr.flags & NFT_EXTHDR_F_PRESENT)
4e0227
 			return;
4e0227
 		nft_print(octx, " %s", expr->exthdr.tmpl->token);
4e0227
+	} else if (expr->exthdr.op == NFT_EXTHDR_OP_SCTP) {
4e0227
+		nft_print(octx, "sctp chunk %s", expr->exthdr.desc->name);
4e0227
+		if (expr->exthdr.flags & NFT_EXTHDR_F_PRESENT)
4e0227
+			return;
4e0227
+		nft_print(octx, " %s", expr->exthdr.tmpl->token);
4e0227
 	} else {
4e0227
 		if (expr->exthdr.flags & NFT_EXTHDR_F_PRESENT)
4e0227
 			nft_print(octx, "exthdr %s", expr->exthdr.desc->name);
4e0227
@@ -197,6 +203,8 @@ void exthdr_init_raw(struct expr *expr, uint8_t type,
4e0227
 		return tcpopt_init_raw(expr, type, offset, len, flags);
4e0227
 	if (op == NFT_EXTHDR_OP_IPV4)
4e0227
 		return ipopt_init_raw(expr, type, offset, len, flags, true);
4e0227
+	if (op == NFT_EXTHDR_OP_SCTP)
4e0227
+		return sctp_chunk_init_raw(expr, type, offset, len, flags);
4e0227
 
4e0227
 	expr->len = len;
4e0227
 	expr->exthdr.flags = flags;
4e0227
diff --git a/src/json.c b/src/json.c
4e0227
index a6d0716..dfc9031 100644
4e0227
--- a/src/json.c
4e0227
+++ b/src/json.c
4e0227
@@ -692,6 +692,8 @@ json_t *exthdr_expr_json(const struct expr *expr, struct output_ctx *octx)
4e0227
 	switch (expr->exthdr.op) {
4e0227
 	case NFT_EXTHDR_OP_IPV4:
4e0227
 		return json_pack("{s:o}", "ip option", root);
4e0227
+	case NFT_EXTHDR_OP_SCTP:
4e0227
+		return json_pack("{s:o}", "sctp chunk", root);
4e0227
 	default:
4e0227
 		return json_pack("{s:o}", "exthdr", root);
4e0227
 	}
4e0227
diff --git a/src/parser_bison.y b/src/parser_bison.y
4e0227
index beb5995..5ab5744 100644
4e0227
--- a/src/parser_bison.y
4e0227
+++ b/src/parser_bison.y
4e0227
@@ -38,6 +38,7 @@
4e0227
 #include <utils.h>
4e0227
 #include <parser.h>
4e0227
 #include <erec.h>
4e0227
+#include <sctp_chunk.h>
4e0227
 
4e0227
 #include "parser_bison.h"
4e0227
 
4e0227
@@ -402,6 +403,40 @@ int nft_lex(void *, void *, void *);
4e0227
 %token DCCP			"dccp"
4e0227
 
4e0227
 %token SCTP			"sctp"
4e0227
+%token CHUNK			"chunk"
4e0227
+%token DATA			"data"
4e0227
+%token INIT			"init"
4e0227
+%token INIT_ACK			"init-ack"
4e0227
+%token HEARTBEAT		"heartbeat"
4e0227
+%token HEARTBEAT_ACK		"heartbeat-ack"
4e0227
+%token ABORT			"abort"
4e0227
+%token SHUTDOWN			"shutdown"
4e0227
+%token SHUTDOWN_ACK		"shutdown-ack"
4e0227
+%token ERROR			"error"
4e0227
+%token COOKIE_ECHO		"cookie-echo"
4e0227
+%token COOKIE_ACK		"cookie-ack"
4e0227
+%token ECNE			"ecne"
4e0227
+%token CWR			"cwr"
4e0227
+%token SHUTDOWN_COMPLETE	"shutdown-complete"
4e0227
+%token ASCONF_ACK		"asconf-ack"
4e0227
+%token FORWARD_TSN		"forward-tsn"
4e0227
+%token ASCONF			"asconf"
4e0227
+%token TSN			"tsn"
4e0227
+%token STREAM			"stream"
4e0227
+%token SSN			"ssn"
4e0227
+%token PPID			"ppid"
4e0227
+%token INIT_TAG			"init-tag"
4e0227
+%token A_RWND			"a-rwnd"
4e0227
+%token NUM_OSTREAMS		"num-outbound-streams"
4e0227
+%token NUM_ISTREAMS		"num-inbound-streams"
4e0227
+%token INIT_TSN			"initial-tsn"
4e0227
+%token CUM_TSN_ACK		"cum-tsn-ack"
4e0227
+%token NUM_GACK_BLOCKS		"num-gap-ack-blocks"
4e0227
+%token NUM_DUP_TSNS		"num-dup-tsns"
4e0227
+%token LOWEST_TSN		"lowest-tsn"
4e0227
+%token SEQNO			"seqno"
4e0227
+%token NEW_CUM_TSN		"new-cum-tsn"
4e0227
+
4e0227
 %token VTAG			"vtag"
4e0227
 
4e0227
 %token RT			"rt"
4e0227
@@ -746,9 +781,12 @@ int nft_lex(void *, void *, void *);
4e0227
 %type <expr>			udp_hdr_expr	udplite_hdr_expr
4e0227
 %destructor { expr_free($$); }	udp_hdr_expr	udplite_hdr_expr
4e0227
 %type <val>			udp_hdr_field	udplite_hdr_field
4e0227
-%type <expr>			dccp_hdr_expr	sctp_hdr_expr
4e0227
-%destructor { expr_free($$); }	dccp_hdr_expr	sctp_hdr_expr
4e0227
+%type <expr>			dccp_hdr_expr	sctp_hdr_expr sctp_chunk_alloc
4e0227
+%destructor { expr_free($$); }	dccp_hdr_expr	sctp_hdr_expr sctp_chunk_alloc
4e0227
 %type <val>			dccp_hdr_field	sctp_hdr_field
4e0227
+%type <val>			sctp_chunk_type sctp_chunk_common_field
4e0227
+%type <val>			sctp_chunk_data_field sctp_chunk_init_field
4e0227
+%type <val>			sctp_chunk_sack_field
4e0227
 %type <expr>			th_hdr_expr
4e0227
 %destructor { expr_free($$); }	th_hdr_expr
4e0227
 %type <val>			th_hdr_field
4e0227
@@ -843,6 +881,7 @@ opt_newline		:	NEWLINE
4e0227
 close_scope_hash	: { scanner_pop_start_cond(nft->scanner, PARSER_SC_EXPR_HASH); };
4e0227
 close_scope_numgen	: { scanner_pop_start_cond(nft->scanner, PARSER_SC_EXPR_NUMGEN); };
4e0227
 close_scope_sctp	: { scanner_pop_start_cond(nft->scanner, PARSER_SC_SCTP); };
4e0227
+close_scope_sctp_chunk	: { scanner_pop_start_cond(nft->scanner, PARSER_SC_EXPR_SCTP_CHUNK); };
4e0227
 
4e0227
 common_block		:	INCLUDE		QUOTED_STRING	stmt_separator
4e0227
 			{
4e0227
@@ -4783,10 +4822,115 @@ dccp_hdr_field		:	SPORT		{ $$ = DCCPHDR_SPORT; }
4e0227
 			|	TYPE		{ $$ = DCCPHDR_TYPE; }
4e0227
 			;
4e0227
 
4e0227
+sctp_chunk_type		:	DATA		{ $$ = SCTP_CHUNK_TYPE_DATA; }
4e0227
+			|	INIT		{ $$ = SCTP_CHUNK_TYPE_INIT; }
4e0227
+			|	INIT_ACK	{ $$ = SCTP_CHUNK_TYPE_INIT_ACK; }
4e0227
+			|	SACK		{ $$ = SCTP_CHUNK_TYPE_SACK; }
4e0227
+			|	HEARTBEAT	{ $$ = SCTP_CHUNK_TYPE_HEARTBEAT; }
4e0227
+			|	HEARTBEAT_ACK	{ $$ = SCTP_CHUNK_TYPE_HEARTBEAT_ACK; }
4e0227
+			|	ABORT		{ $$ = SCTP_CHUNK_TYPE_ABORT; }
4e0227
+			|	SHUTDOWN	{ $$ = SCTP_CHUNK_TYPE_SHUTDOWN; }
4e0227
+			|	SHUTDOWN_ACK	{ $$ = SCTP_CHUNK_TYPE_SHUTDOWN_ACK; }
4e0227
+			|	ERROR		{ $$ = SCTP_CHUNK_TYPE_ERROR; }
4e0227
+			|	COOKIE_ECHO	{ $$ = SCTP_CHUNK_TYPE_COOKIE_ECHO; }
4e0227
+			|	COOKIE_ACK	{ $$ = SCTP_CHUNK_TYPE_COOKIE_ACK; }
4e0227
+			|	ECNE		{ $$ = SCTP_CHUNK_TYPE_ECNE; }
4e0227
+			|	CWR		{ $$ = SCTP_CHUNK_TYPE_CWR; }
4e0227
+			|	SHUTDOWN_COMPLETE { $$ = SCTP_CHUNK_TYPE_SHUTDOWN_COMPLETE; }
4e0227
+			|	ASCONF_ACK	{ $$ = SCTP_CHUNK_TYPE_ASCONF_ACK; }
4e0227
+			|	FORWARD_TSN	{ $$ = SCTP_CHUNK_TYPE_FORWARD_TSN; }
4e0227
+			|	ASCONF		{ $$ = SCTP_CHUNK_TYPE_ASCONF; }
4e0227
+			;
4e0227
+
4e0227
+sctp_chunk_common_field	:	TYPE	{ $$ = SCTP_CHUNK_COMMON_TYPE; }
4e0227
+			|	FLAGS	{ $$ = SCTP_CHUNK_COMMON_FLAGS; }
4e0227
+			|	LENGTH	{ $$ = SCTP_CHUNK_COMMON_LENGTH; }
4e0227
+			;
4e0227
+
4e0227
+sctp_chunk_data_field	:	TSN	{ $$ = SCTP_CHUNK_DATA_TSN; }
4e0227
+			|	STREAM	{ $$ = SCTP_CHUNK_DATA_STREAM; }
4e0227
+			|	SSN	{ $$ = SCTP_CHUNK_DATA_SSN; }
4e0227
+			|	PPID	{ $$ = SCTP_CHUNK_DATA_PPID; }
4e0227
+			;
4e0227
+
4e0227
+sctp_chunk_init_field	:	INIT_TAG	{ $$ = SCTP_CHUNK_INIT_TAG; }
4e0227
+			|	A_RWND		{ $$ = SCTP_CHUNK_INIT_RWND; }
4e0227
+			|	NUM_OSTREAMS	{ $$ = SCTP_CHUNK_INIT_OSTREAMS; }
4e0227
+			|	NUM_ISTREAMS	{ $$ = SCTP_CHUNK_INIT_ISTREAMS; }
4e0227
+			|	INIT_TSN	{ $$ = SCTP_CHUNK_INIT_TSN; }
4e0227
+			;
4e0227
+
4e0227
+sctp_chunk_sack_field	:	CUM_TSN_ACK	{ $$ = SCTP_CHUNK_SACK_CTSN_ACK; }
4e0227
+			|	A_RWND		{ $$ = SCTP_CHUNK_SACK_RWND; }
4e0227
+			|	NUM_GACK_BLOCKS	{ $$ = SCTP_CHUNK_SACK_GACK_BLOCKS; }
4e0227
+			|	NUM_DUP_TSNS	{ $$ = SCTP_CHUNK_SACK_DUP_TSNS; }
4e0227
+			;
4e0227
+
4e0227
+sctp_chunk_alloc	:	sctp_chunk_type
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, $1, SCTP_CHUNK_COMMON_TYPE);
4e0227
+				$$->exthdr.flags = NFT_EXTHDR_F_PRESENT;
4e0227
+			}
4e0227
+			|	sctp_chunk_type	sctp_chunk_common_field
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, $1, $2);
4e0227
+			}
4e0227
+			|	DATA	sctp_chunk_data_field
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_DATA, $2);
4e0227
+			}
4e0227
+			|	INIT	sctp_chunk_init_field
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_INIT, $2);
4e0227
+			}
4e0227
+			|	INIT_ACK	sctp_chunk_init_field
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_INIT_ACK, $2);
4e0227
+			}
4e0227
+			|	SACK	sctp_chunk_sack_field
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_SACK, $2);
4e0227
+			}
4e0227
+			|	SHUTDOWN	CUM_TSN_ACK
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_SHUTDOWN,
4e0227
+							   SCTP_CHUNK_SHUTDOWN_CTSN_ACK);
4e0227
+			}
4e0227
+			|	ECNE	LOWEST_TSN
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_ECNE,
4e0227
+							   SCTP_CHUNK_ECNE_CWR_MIN_TSN);
4e0227
+			}
4e0227
+			|	CWR	LOWEST_TSN
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_CWR,
4e0227
+							   SCTP_CHUNK_ECNE_CWR_MIN_TSN);
4e0227
+			}
4e0227
+			|	ASCONF_ACK	SEQNO
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_ASCONF_ACK,
4e0227
+							   SCTP_CHUNK_ASCONF_SEQNO);
4e0227
+			}
4e0227
+			|	FORWARD_TSN	NEW_CUM_TSN
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_FORWARD_TSN,
4e0227
+							   SCTP_CHUNK_FORWARD_TSN_NCTSN);
4e0227
+			}
4e0227
+			|	ASCONF	SEQNO
4e0227
+			{
4e0227
+				$$ = sctp_chunk_expr_alloc(&@$, SCTP_CHUNK_TYPE_ASCONF,
4e0227
+							   SCTP_CHUNK_ASCONF_SEQNO);
4e0227
+			}
4e0227
+			;
4e0227
+
4e0227
 sctp_hdr_expr		:	SCTP	sctp_hdr_field	close_scope_sctp
4e0227
 			{
4e0227
 				$$ = payload_expr_alloc(&@$, &proto_sctp, $2);
4e0227
 			}
4e0227
+			|	SCTP	CHUNK	sctp_chunk_alloc close_scope_sctp_chunk close_scope_sctp
4e0227
+			{
4e0227
+				$$ = $3;
4e0227
+			}
4e0227
 			;
4e0227
 
4e0227
 sctp_hdr_field		:	SPORT		{ $$ = SCTPHDR_SPORT; }
4e0227
diff --git a/src/parser_json.c b/src/parser_json.c
4e0227
index fbf7db5..a069a89 100644
4e0227
--- a/src/parser_json.c
4e0227
+++ b/src/parser_json.c
4e0227
@@ -11,6 +11,7 @@
4e0227
 #include <netlink.h>
4e0227
 #include <parser.h>
4e0227
 #include <rule.h>
4e0227
+#include <sctp_chunk.h>
4e0227
 #include <socket.h>
4e0227
 
4e0227
 #include <netdb.h>
4e0227
@@ -705,6 +706,53 @@ static struct expr *json_parse_ip_option_expr(struct json_ctx *ctx,
4e0227
 	return ipopt_expr_alloc(int_loc, descval, fieldval, 0);
4e0227
 }
4e0227
 
4e0227
+static int json_parse_sctp_chunk_field(const struct exthdr_desc *desc,
4e0227
+				       const char *name, int *val)
4e0227
+{
4e0227
+	unsigned int i;
4e0227
+
4e0227
+	for (i = 0; i < array_size(desc->templates); i++) {
4e0227
+		if (desc->templates[i].token &&
4e0227
+		    !strcmp(desc->templates[i].token, name)) {
4e0227
+			if (val)
4e0227
+				*val = i;
4e0227
+			return 0;
4e0227
+		}
4e0227
+	}
4e0227
+	return 1;
4e0227
+}
4e0227
+
4e0227
+static struct expr *json_parse_sctp_chunk_expr(struct json_ctx *ctx,
4e0227
+					       const char *type, json_t *root)
4e0227
+{
4e0227
+	const struct exthdr_desc *desc;
4e0227
+	const char *name, *field;
4e0227
+	struct expr *expr;
4e0227
+	int fieldval;
4e0227
+
4e0227
+	if (json_unpack_err(ctx, root, "{s:s}", "name", &name))
4e0227
+		return NULL;
4e0227
+
4e0227
+	desc = sctp_chunk_protocol_find(name);
4e0227
+	if (!desc) {
4e0227
+		json_error(ctx, "Unknown sctp chunk name '%s'.", name);
4e0227
+		return NULL;
4e0227
+	}
4e0227
+
4e0227
+	if (json_unpack(root, "{s:s}", "field", &field)) {
4e0227
+		expr = sctp_chunk_expr_alloc(int_loc, desc->type,
4e0227
+					     SCTP_CHUNK_COMMON_TYPE);
4e0227
+		expr->exthdr.flags = NFT_EXTHDR_F_PRESENT;
4e0227
+
4e0227
+		return expr;
4e0227
+	}
4e0227
+	if (json_parse_sctp_chunk_field(desc, field, &fieldval)) {
4e0227
+		json_error(ctx, "Unknown sctp chunk field '%s'.", field);
4e0227
+		return NULL;
4e0227
+	}
4e0227
+	return sctp_chunk_expr_alloc(int_loc, desc->type, fieldval);
4e0227
+}
4e0227
+
4e0227
 static const struct exthdr_desc *exthdr_lookup_byname(const char *name)
4e0227
 {
4e0227
 	const struct exthdr_desc *exthdr_tbl[] = {
4e0227
@@ -1410,6 +1458,7 @@ static struct expr *json_parse_expr(struct json_ctx *ctx, json_t *root)
4e0227
 		{ "exthdr", json_parse_exthdr_expr, CTX_F_PRIMARY | CTX_F_SET_RHS | CTX_F_SES | CTX_F_MAP | CTX_F_CONCAT },
4e0227
 		{ "tcp option", json_parse_tcp_option_expr, CTX_F_PRIMARY | CTX_F_SET_RHS | CTX_F_MANGLE | CTX_F_SES | CTX_F_CONCAT },
4e0227
 		{ "ip option", json_parse_ip_option_expr, CTX_F_PRIMARY | CTX_F_SET_RHS | CTX_F_MANGLE | CTX_F_SES | CTX_F_CONCAT },
4e0227
+		{ "sctp chunk", json_parse_sctp_chunk_expr, CTX_F_PRIMARY | CTX_F_SET_RHS | CTX_F_MANGLE | CTX_F_SES | CTX_F_CONCAT },
4e0227
 		{ "meta", json_parse_meta_expr, CTX_F_STMT | CTX_F_PRIMARY | CTX_F_SET_RHS | CTX_F_MANGLE | CTX_F_SES | CTX_F_MAP | CTX_F_CONCAT },
4e0227
 		{ "osf", json_parse_osf_expr, CTX_F_STMT | CTX_F_PRIMARY | CTX_F_MAP | CTX_F_CONCAT },
4e0227
 		{ "ipsec", json_parse_xfrm_expr, CTX_F_PRIMARY | CTX_F_MAP | CTX_F_CONCAT },
4e0227
diff --git a/src/scanner.l b/src/scanner.l
4e0227
index b79ae55..fe1b8ad 100644
4e0227
--- a/src/scanner.l
4e0227
+++ b/src/scanner.l
4e0227
@@ -199,6 +199,7 @@ addrstring	({macaddr}|{ip4addr}|{ip6addr})
4e0227
 %s SCANSTATE_SCTP
4e0227
 %s SCANSTATE_EXPR_HASH
4e0227
 %s SCANSTATE_EXPR_NUMGEN
4e0227
+%s SCANSTATE_EXPR_SCTP_CHUNK
4e0227
 
4e0227
 %%
4e0227
 
4e0227
@@ -492,9 +493,46 @@ addrstring	({macaddr}|{ip4addr}|{ip6addr})
4e0227
 "sctp"			{ scanner_push_start_cond(yyscanner, SCANSTATE_SCTP); return SCTP; }
4e0227
 
4e0227
 <SCANSTATE_SCTP>{
4e0227
+	"chunk"			{ scanner_push_start_cond(yyscanner, SCANSTATE_EXPR_SCTP_CHUNK); return CHUNK; }
4e0227
 	"vtag"			{ return VTAG; }
4e0227
 }
4e0227
 
4e0227
+<SCANSTATE_EXPR_SCTP_CHUNK>{
4e0227
+	"data"			{ return DATA; }
4e0227
+	"init"			{ return INIT; }
4e0227
+	"init-ack"		{ return INIT_ACK; }
4e0227
+	"heartbeat"		{ return HEARTBEAT; }
4e0227
+	"heartbeat-ack"		{ return HEARTBEAT_ACK; }
4e0227
+	"abort"			{ return ABORT; }
4e0227
+	"shutdown"		{ return SHUTDOWN; }
4e0227
+	"shutdown-ack"		{ return SHUTDOWN_ACK; }
4e0227
+	"error"			{ return ERROR; }
4e0227
+	"cookie-echo"		{ return COOKIE_ECHO; }
4e0227
+	"cookie-ack"		{ return COOKIE_ACK; }
4e0227
+	"ecne"			{ return ECNE; }
4e0227
+	"cwr"			{ return CWR; }
4e0227
+	"shutdown-complete"	{ return SHUTDOWN_COMPLETE; }
4e0227
+	"asconf-ack"		{ return ASCONF_ACK; }
4e0227
+	"forward-tsn"		{ return FORWARD_TSN; }
4e0227
+	"asconf"		{ return ASCONF; }
4e0227
+
4e0227
+	"tsn"			{ return TSN; }
4e0227
+	"stream"		{ return STREAM; }
4e0227
+	"ssn"			{ return SSN; }
4e0227
+	"ppid"			{ return PPID; }
4e0227
+	"init-tag"		{ return INIT_TAG; }
4e0227
+	"a-rwnd"		{ return A_RWND; }
4e0227
+	"num-outbound-streams"	{ return NUM_OSTREAMS; }
4e0227
+	"num-inbound-streams"	{ return NUM_ISTREAMS; }
4e0227
+	"initial-tsn"		{ return INIT_TSN; }
4e0227
+	"cum-tsn-ack"		{ return CUM_TSN_ACK; }
4e0227
+	"num-gap-ack-blocks"	{ return NUM_GACK_BLOCKS; }
4e0227
+	"num-dup-tsns"		{ return NUM_DUP_TSNS; }
4e0227
+	"lowest-tsn"		{ return LOWEST_TSN; }
4e0227
+	"seqno"			{ return SEQNO; }
4e0227
+	"new-cum-tsn"		{ return NEW_CUM_TSN; }
4e0227
+}
4e0227
+
4e0227
 "rt"			{ return RT; }
4e0227
 "rt0"			{ return RT0; }
4e0227
 "rt2"			{ return RT2; }
4e0227
diff --git a/src/sctp_chunk.c b/src/sctp_chunk.c
4e0227
new file mode 100644
4e0227
index 0000000..6e73e72
4e0227
--- /dev/null
4e0227
+++ b/src/sctp_chunk.c
4e0227
@@ -0,0 +1,261 @@
4e0227
+/*
4e0227
+ * Copyright Red Hat
4e0227
+ *
4e0227
+ * This program is free software; you can redistribute it and/or modify
4e0227
+ * it under the terms of the GNU General Public License version 2 (or any
4e0227
+ * later) as published by the Free Software Foundation.
4e0227
+ */
4e0227
+
4e0227
+#include <exthdr.h>
4e0227
+#include <sctp_chunk.h>
4e0227
+
4e0227
+#include <string.h>
4e0227
+
4e0227
+#define PHT(__token, __offset, __len) \
4e0227
+	PROTO_HDR_TEMPLATE(__token, &integer_type, BYTEORDER_BIG_ENDIAN, \
4e0227
+			   __offset, __len)
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_data = {
4e0227
+	.name	= "data",
4e0227
+	.type	= SCTP_CHUNK_TYPE_DATA,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_DATA_TSN]		= PHT("tsn", 32, 32),
4e0227
+		[SCTP_CHUNK_DATA_STREAM]	= PHT("stream", 64, 16),
4e0227
+		[SCTP_CHUNK_DATA_SSN]		= PHT("ssn", 80, 16),
4e0227
+		[SCTP_CHUNK_DATA_PPID]		= PHT("ppid", 96, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_init = {
4e0227
+	.name	= "init",
4e0227
+	.type	= SCTP_CHUNK_TYPE_INIT,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_INIT_TAG]		= PHT("init-tag", 32, 32),
4e0227
+		[SCTP_CHUNK_INIT_RWND]		= PHT("a-rwnd", 64, 32),
4e0227
+		[SCTP_CHUNK_INIT_OSTREAMS]	= PHT("num-outbound-streams", 96, 16),
4e0227
+		[SCTP_CHUNK_INIT_ISTREAMS]	= PHT("num-inbound-streams", 112, 16),
4e0227
+		[SCTP_CHUNK_INIT_TSN]		= PHT("initial-tsn", 128, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_init_ack = {
4e0227
+	.name	= "init-ack",
4e0227
+	.type	= SCTP_CHUNK_TYPE_INIT_ACK,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_INIT_TAG]		= PHT("init-tag", 32, 32),
4e0227
+		[SCTP_CHUNK_INIT_RWND]		= PHT("a-rwnd", 64, 32),
4e0227
+		[SCTP_CHUNK_INIT_OSTREAMS]	= PHT("num-outbound-streams", 96, 16),
4e0227
+		[SCTP_CHUNK_INIT_ISTREAMS]	= PHT("num-inbound-streams", 112, 16),
4e0227
+		[SCTP_CHUNK_INIT_TSN]		= PHT("initial-tsn", 128, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_sack = {
4e0227
+	.name	= "sack",
4e0227
+	.type	= SCTP_CHUNK_TYPE_SACK,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_SACK_CTSN_ACK]	= PHT("cum-tsn-ack", 32, 32),
4e0227
+		[SCTP_CHUNK_SACK_RWND]		= PHT("a-rwnd", 64, 32),
4e0227
+		[SCTP_CHUNK_SACK_GACK_BLOCKS]	= PHT("num-gap-ack-blocks", 96, 16),
4e0227
+		[SCTP_CHUNK_SACK_DUP_TSNS]	= PHT("num-dup-tsns", 112, 16),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_shutdown = {
4e0227
+	.name	= "shutdown",
4e0227
+	.type	= SCTP_CHUNK_TYPE_SHUTDOWN,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_SHUTDOWN_CTSN_ACK]	= PHT("cum-tsn-ack", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_ecne = {
4e0227
+	.name	= "ecne",
4e0227
+	.type	= SCTP_CHUNK_TYPE_ECNE,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_ECNE_CWR_MIN_TSN]	= PHT("lowest-tsn", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_cwr = {
4e0227
+	.name	= "cwr",
4e0227
+	.type	= SCTP_CHUNK_TYPE_CWR,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_ECNE_CWR_MIN_TSN]	= PHT("lowest-tsn", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_asconf_ack = {
4e0227
+	.name	= "asconf-ack",
4e0227
+	.type	= SCTP_CHUNK_TYPE_ASCONF_ACK,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_ASCONF_SEQNO]	= PHT("seqno", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_forward_tsn = {
4e0227
+	.name	= "forward-tsn",
4e0227
+	.type	= SCTP_CHUNK_TYPE_FORWARD_TSN,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_FORWARD_TSN_NCTSN]	= PHT("new-cum-tsn", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+static const struct exthdr_desc sctp_chunk_asconf = {
4e0227
+	.name	= "asconf",
4e0227
+	.type	= SCTP_CHUNK_TYPE_ASCONF,
4e0227
+	.templates = {
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),
4e0227
+		[SCTP_CHUNK_ASCONF_SEQNO]	= PHT("seqno", 32, 32),
4e0227
+	},
4e0227
+};
4e0227
+
4e0227
+#define SCTP_CHUNK_DESC_GENERATOR(descname, hname, desctype)		\
4e0227
+static const struct exthdr_desc sctp_chunk_##descname = {		\
4e0227
+	.name	= #hname,						\
4e0227
+	.type	= SCTP_CHUNK_TYPE_##desctype,				\
4e0227
+	.templates = {							\
4e0227
+		[SCTP_CHUNK_COMMON_TYPE]	= PHT("type", 0, 8),	\
4e0227
+		[SCTP_CHUNK_COMMON_FLAGS]	= PHT("flags", 8, 8),	\
4e0227
+		[SCTP_CHUNK_COMMON_LENGTH]	= PHT("length", 16, 16),\
4e0227
+	},								\
4e0227
+};
4e0227
+
4e0227
+SCTP_CHUNK_DESC_GENERATOR(heartbeat, heartbeat, HEARTBEAT)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(heartbeat_ack, heartbeat-ack, HEARTBEAT_ACK)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(abort, abort, ABORT)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(shutdown_ack, shutdown-ack, SHUTDOWN_ACK)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(error, error, ERROR)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(cookie_echo, cookie-echo, COOKIE_ECHO)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(cookie_ack, cookie-ack, COOKIE_ACK)
4e0227
+SCTP_CHUNK_DESC_GENERATOR(shutdown_complete, shutdown-complete, SHUTDOWN_COMPLETE)
4e0227
+
4e0227
+#undef SCTP_CHUNK_DESC_GENERATOR
4e0227
+
4e0227
+static const struct exthdr_desc *sctp_chunk_protocols[] = {
4e0227
+	[SCTP_CHUNK_TYPE_DATA]			= &sctp_chunk_data,
4e0227
+	[SCTP_CHUNK_TYPE_INIT]			= &sctp_chunk_init,
4e0227
+	[SCTP_CHUNK_TYPE_INIT_ACK]		= &sctp_chunk_init_ack,
4e0227
+	[SCTP_CHUNK_TYPE_SACK]			= &sctp_chunk_sack,
4e0227
+	[SCTP_CHUNK_TYPE_HEARTBEAT]		= &sctp_chunk_heartbeat,
4e0227
+	[SCTP_CHUNK_TYPE_HEARTBEAT_ACK]		= &sctp_chunk_heartbeat_ack,
4e0227
+	[SCTP_CHUNK_TYPE_ABORT]			= &sctp_chunk_abort,
4e0227
+	[SCTP_CHUNK_TYPE_SHUTDOWN]		= &sctp_chunk_shutdown,
4e0227
+	[SCTP_CHUNK_TYPE_SHUTDOWN_ACK]		= &sctp_chunk_shutdown_ack,
4e0227
+	[SCTP_CHUNK_TYPE_ERROR]			= &sctp_chunk_error,
4e0227
+	[SCTP_CHUNK_TYPE_COOKIE_ECHO]		= &sctp_chunk_cookie_echo,
4e0227
+	[SCTP_CHUNK_TYPE_COOKIE_ACK]		= &sctp_chunk_cookie_ack,
4e0227
+	[SCTP_CHUNK_TYPE_ECNE]			= &sctp_chunk_ecne,
4e0227
+	[SCTP_CHUNK_TYPE_CWR]			= &sctp_chunk_cwr,
4e0227
+	[SCTP_CHUNK_TYPE_SHUTDOWN_COMPLETE]	= &sctp_chunk_shutdown_complete,
4e0227
+	[SCTP_CHUNK_TYPE_ASCONF_ACK]		= &sctp_chunk_asconf_ack,
4e0227
+	[SCTP_CHUNK_TYPE_FORWARD_TSN]		= &sctp_chunk_forward_tsn,
4e0227
+	[SCTP_CHUNK_TYPE_ASCONF]		= &sctp_chunk_asconf,
4e0227
+};
4e0227
+
4e0227
+const struct exthdr_desc *sctp_chunk_protocol_find(const char *name)
4e0227
+{
4e0227
+	unsigned int i;
4e0227
+
4e0227
+	for (i = 0; i < array_size(sctp_chunk_protocols); i++) {
4e0227
+		if (sctp_chunk_protocols[i] &&
4e0227
+		    !strcmp(sctp_chunk_protocols[i]->name, name))
4e0227
+			return sctp_chunk_protocols[i];
4e0227
+	}
4e0227
+	return NULL;
4e0227
+}
4e0227
+
4e0227
+struct expr *sctp_chunk_expr_alloc(const struct location *loc,
4e0227
+				   unsigned int type, unsigned int field)
4e0227
+{
4e0227
+	const struct proto_hdr_template *tmpl;
4e0227
+	const struct exthdr_desc *desc = NULL;
4e0227
+	struct expr *expr;
4e0227
+
4e0227
+	if (type < array_size(sctp_chunk_protocols))
4e0227
+		desc = sctp_chunk_protocols[type];
4e0227
+
4e0227
+	if (!desc)
4e0227
+		return NULL;
4e0227
+
4e0227
+	tmpl = &desc->templates[field];
4e0227
+	if (!tmpl)
4e0227
+		return NULL;
4e0227
+
4e0227
+	expr = expr_alloc(loc, EXPR_EXTHDR, tmpl->dtype,
4e0227
+			  BYTEORDER_BIG_ENDIAN, tmpl->len);
4e0227
+	expr->exthdr.desc	= desc;
4e0227
+	expr->exthdr.tmpl	= tmpl;
4e0227
+	expr->exthdr.op		= NFT_EXTHDR_OP_SCTP;
4e0227
+	expr->exthdr.raw_type	= desc->type;
4e0227
+	expr->exthdr.offset	= tmpl->offset;
4e0227
+
4e0227
+	return expr;
4e0227
+}
4e0227
+
4e0227
+void sctp_chunk_init_raw(struct expr *expr, uint8_t type, unsigned int off,
4e0227
+			 unsigned int len, uint32_t flags)
4e0227
+{
4e0227
+	const struct proto_hdr_template *tmpl;
4e0227
+	unsigned int i;
4e0227
+
4e0227
+	assert(expr->etype == EXPR_EXTHDR);
4e0227
+
4e0227
+	expr->len = len;
4e0227
+	expr->exthdr.flags = flags;
4e0227
+	expr->exthdr.offset = off;
4e0227
+	expr->exthdr.op = NFT_EXTHDR_OP_SCTP;
4e0227
+
4e0227
+	if (flags & NFT_EXTHDR_F_PRESENT)
4e0227
+		datatype_set(expr, &boolean_type);
4e0227
+	else
4e0227
+		datatype_set(expr, &integer_type);
4e0227
+
4e0227
+	if (type >= array_size(sctp_chunk_protocols))
4e0227
+		return;
4e0227
+
4e0227
+	expr->exthdr.desc = sctp_chunk_protocols[type];
4e0227
+	expr->exthdr.flags = flags;
4e0227
+	assert(expr->exthdr.desc != NULL);
4e0227
+
4e0227
+	for (i = 0; i < array_size(expr->exthdr.desc->templates); ++i) {
4e0227
+		tmpl = &expr->exthdr.desc->templates[i];
4e0227
+		if (tmpl->offset != off || tmpl->len != len)
4e0227
+			continue;
4e0227
+
4e0227
+		if ((flags & NFT_EXTHDR_F_PRESENT) == 0)
4e0227
+			datatype_set(expr, tmpl->dtype);
4e0227
+
4e0227
+		expr->exthdr.tmpl = tmpl;
4e0227
+		break;
4e0227
+	}
4e0227
+}
4e0227
diff --git a/tests/py/inet/sctp.t b/tests/py/inet/sctp.t
4e0227
index 5188b57..3d1c2fd 100644
4e0227
--- a/tests/py/inet/sctp.t
4e0227
+++ b/tests/py/inet/sctp.t
4e0227
@@ -41,3 +41,40 @@ sctp vtag {33, 55, 67, 88};ok
4e0227
 sctp vtag != {33, 55, 67, 88};ok
4e0227
 sctp vtag { 33-55};ok
4e0227
 sctp vtag != { 33-55};ok
4e0227
+
4e0227
+# assert all chunk types are recognized
4e0227
+sctp chunk data exists;ok
4e0227
+sctp chunk init exists;ok
4e0227
+sctp chunk init-ack exists;ok
4e0227
+sctp chunk sack exists;ok
4e0227
+sctp chunk heartbeat exists;ok
4e0227
+sctp chunk heartbeat-ack exists;ok
4e0227
+sctp chunk abort exists;ok
4e0227
+sctp chunk shutdown exists;ok
4e0227
+sctp chunk shutdown-ack exists;ok
4e0227
+sctp chunk error exists;ok
4e0227
+sctp chunk cookie-echo exists;ok
4e0227
+sctp chunk cookie-ack exists;ok
4e0227
+sctp chunk ecne exists;ok
4e0227
+sctp chunk cwr exists;ok
4e0227
+sctp chunk shutdown-complete exists;ok
4e0227
+sctp chunk asconf-ack exists;ok
4e0227
+sctp chunk forward-tsn exists;ok
4e0227
+sctp chunk asconf exists;ok
4e0227
+
4e0227
+# test common header fields in random chunk types
4e0227
+sctp chunk data type 0;ok
4e0227
+sctp chunk init flags 23;ok
4e0227
+sctp chunk init-ack length 42;ok
4e0227
+
4e0227
+# test one custom field in every applicable chunk type
4e0227
+sctp chunk data stream 1337;ok
4e0227
+sctp chunk init initial-tsn 5;ok
4e0227
+sctp chunk init-ack num-outbound-streams 3;ok
4e0227
+sctp chunk sack a-rwnd 1;ok
4e0227
+sctp chunk shutdown cum-tsn-ack 65535;ok
4e0227
+sctp chunk ecne lowest-tsn 5;ok
4e0227
+sctp chunk cwr lowest-tsn 8;ok
4e0227
+sctp chunk asconf-ack seqno 12345;ok
4e0227
+sctp chunk forward-tsn new-cum-tsn 31337;ok
4e0227
+sctp chunk asconf seqno 12345;ok
4e0227
diff --git a/tests/py/inet/sctp.t.json b/tests/py/inet/sctp.t.json
4e0227
index 2684b03..8135686 100644
4e0227
--- a/tests/py/inet/sctp.t.json
4e0227
+++ b/tests/py/inet/sctp.t.json
4e0227
@@ -608,3 +608,481 @@
4e0227
     }
4e0227
 ]
4e0227
 
4e0227
+# sctp chunk data exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "data"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "init"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init-ack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "init-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk sack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "sack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk heartbeat exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "heartbeat"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk heartbeat-ack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "heartbeat-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk abort exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "abort"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk shutdown exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "shutdown"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk shutdown-ack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "shutdown-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk error exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "error"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk cookie-echo exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "cookie-echo"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk cookie-ack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "cookie-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk ecne exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "ecne"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk cwr exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "cwr"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk shutdown-complete exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "shutdown-complete"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk asconf-ack exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "asconf-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk forward-tsn exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "forward-tsn"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk asconf exists
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "name": "asconf"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": true
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk data type 0
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "type",
4e0227
+                    "name": "data"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 0
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init flags 23
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "flags",
4e0227
+                    "name": "init"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 23
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init-ack length 42
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "length",
4e0227
+                    "name": "init-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 42
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk data stream 1337
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "stream",
4e0227
+                    "name": "data"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 1337
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init initial-tsn 5
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "initial-tsn",
4e0227
+                    "name": "init"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 5
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk init-ack num-outbound-streams 3
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "num-outbound-streams",
4e0227
+                    "name": "init-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 3
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk sack a-rwnd 1
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "a-rwnd",
4e0227
+                    "name": "sack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 1
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk shutdown cum-tsn-ack 65535
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "cum-tsn-ack",
4e0227
+                    "name": "shutdown"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 65535
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk ecne lowest-tsn 5
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "lowest-tsn",
4e0227
+                    "name": "ecne"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 5
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk cwr lowest-tsn 8
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "lowest-tsn",
4e0227
+                    "name": "cwr"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 8
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk asconf-ack seqno 12345
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "seqno",
4e0227
+                    "name": "asconf-ack"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 12345
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk forward-tsn new-cum-tsn 31337
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "new-cum-tsn",
4e0227
+                    "name": "forward-tsn"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 31337
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
+# sctp chunk asconf seqno 12345
4e0227
+[
4e0227
+    {
4e0227
+        "match": {
4e0227
+            "left": {
4e0227
+                "sctp chunk": {
4e0227
+                    "field": "seqno",
4e0227
+                    "name": "asconf"
4e0227
+                }
4e0227
+            },
4e0227
+            "op": "==",
4e0227
+            "right": 12345
4e0227
+        }
4e0227
+    }
4e0227
+]
4e0227
+
4e0227
diff --git a/tests/py/inet/sctp.t.payload b/tests/py/inet/sctp.t.payload
4e0227
index ecfcc72..9c4854c 100644
4e0227
--- a/tests/py/inet/sctp.t.payload
4e0227
+++ b/tests/py/inet/sctp.t.payload
4e0227
@@ -274,3 +274,158 @@ inet test-inet input
4e0227
   [ payload load 4b @ transport header + 4 => reg 1 ]
4e0227
   [ lookup reg 1 set __set%d 0x1 ]
4e0227
 
4e0227
+# sctp chunk data exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 0 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk init exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 1 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk init-ack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 2 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk sack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 3 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk heartbeat exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 4 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk heartbeat-ack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 5 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk abort exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 6 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk shutdown exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 7 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk shutdown-ack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 8 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk error exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 9 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk cookie-echo exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 10 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk cookie-ack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 11 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk ecne exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 12 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk cwr exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 13 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk shutdown-complete exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 14 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk asconf-ack exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 128 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk forward-tsn exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 192 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk asconf exists
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 193 + 0 present => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000001 ]
4e0227
+
4e0227
+# sctp chunk data type 0
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 0 + 0 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000000 ]
4e0227
+
4e0227
+# sctp chunk init flags 23
4e0227
+ip
4e0227
+  [ exthdr load 1b @ 1 + 1 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000017 ]
4e0227
+
4e0227
+# sctp chunk init-ack length 42
4e0227
+ip
4e0227
+  [ exthdr load 2b @ 2 + 2 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00002a00 ]
4e0227
+
4e0227
+# sctp chunk data stream 1337
4e0227
+ip
4e0227
+  [ exthdr load 2b @ 0 + 8 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00003905 ]
4e0227
+
4e0227
+# sctp chunk init initial-tsn 5
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 1 + 16 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x05000000 ]
4e0227
+
4e0227
+# sctp chunk init-ack num-outbound-streams 3
4e0227
+ip
4e0227
+  [ exthdr load 2b @ 2 + 12 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x00000300 ]
4e0227
+
4e0227
+# sctp chunk sack a-rwnd 1
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 3 + 8 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x01000000 ]
4e0227
+
4e0227
+# sctp chunk shutdown cum-tsn-ack 65535
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 7 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0xffff0000 ]
4e0227
+
4e0227
+# sctp chunk ecne lowest-tsn 5
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 12 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x05000000 ]
4e0227
+
4e0227
+# sctp chunk cwr lowest-tsn 8
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 13 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x08000000 ]
4e0227
+
4e0227
+# sctp chunk asconf-ack seqno 12345
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 128 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x39300000 ]
4e0227
+
4e0227
+# sctp chunk forward-tsn new-cum-tsn 31337
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 192 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x697a0000 ]
4e0227
+
4e0227
+# sctp chunk asconf seqno 12345
4e0227
+ip
4e0227
+  [ exthdr load 4b @ 193 + 4 => reg 1 ]
4e0227
+  [ cmp eq reg 1 0x39300000 ]
4e0227
+
4e0227
-- 
4e0227
2.31.1
4e0227