diff --git a/gtests/net/packetdrill/lexer.l b/gtests/net/packetdrill/lexer.l
index d452e25dc515b249501508be61899e7d4613ca56..0fa52409cf7d383407124cc5ef743cb4f4d0d0e9 100644
--- a/gtests/net/packetdrill/lexer.l
+++ b/gtests/net/packetdrill/lexer.l
@@ -557,6 +557,7 @@ I-DATA				return I_DATA;
 PAD				return PAD;
 RECONFIG			return RECONFIG;
 FORWARD_TSN                     return FORWARD_TSN;
+I_FORWARD_TSN                   return I_FORWARD_TSN;
 type				return TYPE;
 flgs				return FLAGS;
 len				return LEN;
diff --git a/gtests/net/packetdrill/packet_to_string_test.c b/gtests/net/packetdrill/packet_to_string_test.c
index 136a075272bdb4413bd6c340c8178727dbfd72f9..4774f7ed30dab34395652e13d702877500aef44e 100644
--- a/gtests/net/packetdrill/packet_to_string_test.c
+++ b/gtests/net/packetdrill/packet_to_string_test.c
@@ -103,7 +103,7 @@ static void test_sctp_ipv6_packet_to_string(void)
 	/* An IPv6/SCTP packet. */
 	u8 data[] = {
 		/* IPv6 Base Header: */
-		0x60, 0x00, 0x00, 0x00, 0x02, 0x0c, 0x84, 0xff,
+		0x60, 0x00, 0x00, 0x00, 0x02, 0x1c, 0x84, 0xff,
 		0x00, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 		0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x22, 0x22,
 		0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
@@ -111,7 +111,7 @@ static void test_sctp_ipv6_packet_to_string(void)
 		/* SCTP Common Header: */
 		0x04, 0xd2, 0x1f, 0x90,
 		0x01, 0x02, 0x03, 0x04,
-		0xd8, 0x6a, 0x11, 0x71,
+		0xc9, 0xc7, 0x27, 0x40,
 		/* SCTP DATA Chunk */
 		0x00, 0x0f, 0x00, 0x13,
 		0x01, 0x02, 0x03, 0x04,
@@ -242,6 +242,11 @@ static void test_sctp_ipv6_packet_to_string(void)
 		0xb5, 0xaa, 0xaf, 0x0f,
 		0x00, 0x01, 0x00, 0x02,
 		0x00, 0x03, 0x00, 0x04,
+		/* I_FORWARD_TSN Chunk*/
+		0xc2, 0x00, 0x00, 0x10,
+		0xb5, 0xaa, 0xaf, 0x0f,
+		0x00, 0x01, 0x00, 0x01,
+		0x00, 0x00, 0x01, 0x00,
 		/* SCTP I-DATA Chunk */
 		0x40, 0x0f, 0x00, 0x17,
 		0x00, 0x00, 0x00, 0x04,
@@ -341,6 +346,7 @@ static void test_sctp_ipv6_packet_to_string(void)
 		"CWR[flgs=0x00, tsn=16909060]; "
 		"SHUTDOWN_COMPLETE[flgs=T]; "
 		"FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,2},{3,4}]]; "
+		"I_FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,1,256}]]; "
 		"I-DATA[flgs=IUBE, len=23, tsn=4, sid=255, mid=1, ppid=0]; "
 		"I-DATA[flgs=IUE, len=23, tsn=4, sid=255, mid=2, fsn=1]; "
 		"PAD[flgs=0x00, len=16, val=...]";
@@ -403,6 +409,7 @@ static void test_sctp_ipv6_packet_to_string(void)
 		"CWR[flgs=0x00, tsn=16909060]; "
 		"SHUTDOWN_COMPLETE[flgs=T]; "
 		"FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,2},{3,4}]]; "
+		"I_FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,1,256}]]; "
 		"I-DATA[flgs=IUBE, len=23, tsn=4, sid=255, mid=1, ppid=0]; "
 		"I-DATA[flgs=IUE, len=23, tsn=4, sid=255, mid=2, fsn=1]; "
 		"PAD[flgs=0x00, len=16, val=...]";
@@ -465,14 +472,15 @@ static void test_sctp_ipv6_packet_to_string(void)
 		"CWR[flgs=0x00, tsn=16909060]; "
 		"SHUTDOWN_COMPLETE[flgs=T]; "
 		"FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,2},{3,4}]]; "
+		"I_FORWARD_TSN[flgs=0x00, len=16, cum_tsn=3047862031, ids=[{1,1,256}]]; "
 		"I-DATA[flgs=IUBE, len=23, tsn=4, sid=255, mid=1, ppid=0]; "
 		"I-DATA[flgs=IUE, len=23, tsn=4, sid=255, mid=2, fsn=1]; "
 		"PAD[flgs=0x00, len=16, val=...]"
 		"\n"
-		"0x0000: 60 00 00 00 02 0c 84 ff 00 02 00 00 00 00 00 00 " "\n"
+		"0x0000: 60 00 00 00 02 1c 84 ff 00 02 00 00 00 00 00 00 " "\n"
 		"0x0010: 00 00 00 00 00 00 22 22 00 01 00 00 00 00 00 00 " "\n"
 		"0x0020: 00 00 00 00 00 00 11 11 04 d2 1f 90 01 02 03 04 " "\n"
-		"0x0030: d8 6a 11 71 00 0f 00 13 01 02 03 04 00 ff 01 00 " "\n"
+		"0x0030: c9 c7 27 40 00 0f 00 13 01 02 03 04 00 ff 01 00 " "\n"
 		"0x0040: 00 00 00 00 00 01 02 00 01 00 00 68 00 00 00 01 " "\n"
 		"0x0050: 00 01 00 00 00 0f 00 0f 01 02 03 04 00 05 00 08 " "\n"
 		"0x0060: 01 02 03 04 00 06 00 14 00 00 00 00 00 00 00 00 " "\n"
@@ -500,11 +508,13 @@ static void test_sctp_ipv6_packet_to_string(void)
 		"0x01c0: 09 00 00 04 0a 00 00 05 45 00 00 00 0b 00 00 04 " "\n"
 		"0x01d0: 0c 00 00 08 01 02 03 04 0d 00 00 08 01 02 03 04 " "\n"
 		"0x01e0: 0e 01 00 04 c0 00 00 10 b5 aa af 0f 00 01 00 02 " "\n"
-		"0x01f0: 00 03 00 04 40 0f 00 17 00 00 00 04 00 ff 00 00 " "\n"
-		"0x0200: 00 00 00 01 00 00 00 00 00 01 02 00 40 0d 00 17 " "\n"
-		"0x0210: 00 00 00 04 00 ff 00 00 00 00 00 02 00 00 00 01 " "\n"
-		"0x0220: 00 01 02 00 84 00 00 10 50 50 50 50 50 50 50 50 " "\n"
-		"0x0230: 50 50 50 50 " "\n";
+		"0x01f0: 00 03 00 04 c2 00 00 10 b5 aa af 0f 00 01 00 01 " "\n"
+		"0x0200: 00 00 01 00 40 0f 00 17 00 00 00 04 00 ff 00 00 " "\n"
+		"0x0210: 00 00 00 01 00 00 00 00 00 01 02 00 40 0d 00 17 " "\n"
+		"0x0220: 00 00 00 04 00 ff 00 00 00 00 00 02 00 00 00 01 " "\n"
+		"0x0230: 00 01 02 00 84 00 00 10 50 50 50 50 50 50 50 50 " "\n"
+		"0x0240: 50 50 50 50 " "\n";
+
 	printf("expected = '%s'\n", expected);
 	assert(strcmp(dump, expected) == 0);
 	free(dump);
diff --git a/gtests/net/packetdrill/parser.y b/gtests/net/packetdrill/parser.y
index f2bc2b49151d17f13ebb88284f7051f9a5397e20..51843f8d1efe7a8c3367c671f90a7fb9f44d6b75 100644
--- a/gtests/net/packetdrill/parser.y
+++ b/gtests/net/packetdrill/parser.y
@@ -474,6 +474,8 @@ static struct tcp_option *new_tcp_fast_open_option(const char *cookie_string,
 	struct sctp_sack_block_list *sack_block_list;
 	struct sctp_forward_tsn_ids_list *forward_tsn_ids_list;
 	struct sctp_forward_tsn_ids_list_item  *forward_tsn_ids_list_item;
+	struct sctp_i_forward_tsn_ids_list *i_forward_tsn_ids_list;
+	struct sctp_i_forward_tsn_ids_list_item  *i_forward_tsn_ids_list_item;
 	struct sctp_address_type_list_item *address_type_list_item;
 	struct sctp_address_type_list *address_type_list;
 	struct sctp_parameter_type_list_item *parameter_type_list_item;
@@ -521,7 +523,7 @@ static struct tcp_option *new_tcp_fast_open_option(const char *cookie_string,
 %token <reserved> SPINFO_MTU GAUTH_ASSOC_ID GAUTH_NUMBER_OF_CHUNKS GAUTH_CHUNKS
 %token <reserved> CHUNK DATA INIT INIT_ACK HEARTBEAT HEARTBEAT_ACK ABORT
 %token <reserved> SHUTDOWN SHUTDOWN_ACK ERROR COOKIE_ECHO COOKIE_ACK ECNE CWR
-%token <reserved> SHUTDOWN_COMPLETE I_DATA PAD RECONFIG FORWARD_TSN
+%token <reserved> SHUTDOWN_COMPLETE I_DATA PAD RECONFIG FORWARD_TSN I_FORWARD_TSN
 %token <reserved> TYPE FLAGS LEN
 %token <reserved> TAG A_RWND OS IS TSN SID SSN MID PPID FSN CUM_TSN GAPS NR_GAPS DUPS
 %token <reserved> PARAMETER HEARTBEAT_INFORMATION IPV4_ADDRESS IPV6_ADDRESS
@@ -677,7 +679,7 @@ static struct tcp_option *new_tcp_fast_open_option(const char *cookie_string,
 %type <chunk_list_item> sctp_shutdown_complete_chunk_spec
 %type <chunk_list_item> sctp_i_data_chunk_spec
 %type <chunk_list_item> sctp_pad_chunk_spec sctp_reconfig_chunk_spec
-%type <chunk_list_item> sctp_forward_tsn_spec
+%type <chunk_list_item> sctp_forward_tsn_spec sctp_i_forward_tsn_spec
 %type <parameter_list> opt_parameter_list_spec sctp_parameter_list_spec
 %type <parameter_list_item> sctp_parameter_spec
 %type <parameter_list_item> sctp_generic_parameter_spec
@@ -726,8 +728,10 @@ static struct tcp_option *new_tcp_fast_open_option(const char *cookie_string,
 %type <u16_item> u16_item
 %type <sack_block_list> opt_gaps opt_nr_gaps gap_list opt_dups dup_list
 %type <sack_block_list_item> gap dup
-%type <forward_tsn_ids_list> opt_stream_identifier ids_list
+%type <forward_tsn_ids_list> opt_stream_identifier  ids_list
 %type <forward_tsn_ids_list_item> id
+%type <i_forward_tsn_ids_list> opt_i_forward_tsn_stream_identifier i_forward_tsn_ids_list
+%type <i_forward_tsn_ids_list_item> i_forward_tsn_id
 %type <address_type_list> address_types_list
 %type <address_type_list_item> address_type
 %type <parameter_type_list> parameter_types_list
@@ -1061,6 +1065,7 @@ sctp_chunk_spec
 | sctp_pad_chunk_spec               { $$ = $1; }
 | sctp_reconfig_chunk_spec          { $$ = $1; }
 | sctp_forward_tsn_spec             { $$ = $1; }
+| sctp_i_forward_tsn_spec           { $$ = $1; }
 ;
 
 chunk_type
@@ -1610,6 +1615,35 @@ id
 }
 ;
 
+opt_i_forward_tsn_stream_identifier
+: IDS '=' ELLIPSIS         { $$ = NULL; }
+| IDS '=' '[' ELLIPSIS ']' { $$ = NULL; }
+| IDS '=' '[' i_forward_tsn_ids_list ']' { $$ = $4; }
+;
+
+i_forward_tsn_ids_list
+:                  { $$ = sctp_i_forward_tsn_ids_list_new(); }
+| i_forward_tsn_id { $$ =sctp_i_forward_tsn_ids_list_new();
+                     sctp_i_forward_tsn_ids_list_append($$, $1); }
+| i_forward_tsn_ids_list ',' i_forward_tsn_id { $$ = $1;
+                     sctp_i_forward_tsn_ids_list_append($1, $3); }
+;
+
+i_forward_tsn_id
+: '{' INTEGER ',' INTEGER ',' INTEGER '}' {
+	if (!is_valid_u16($2)) {
+		semantic_error("stream identifier out of range");
+	}
+	if (!is_valid_u16($4)) {
+		semantic_error("reserved out of range");
+	}
+	if (!is_valid_u32($6)) {
+		semantic_error("message identifier number out of range");
+	}
+	$$ = sctp_i_forward_tsn_ids_list_item_new($2, $4, $6);
+}
+;
+
 sctp_generic_chunk_spec
 : CHUNK '[' opt_chunk_type ',' opt_flags ',' opt_len ',' opt_val ']' {
 	if (($7 != -1) &&
@@ -1752,6 +1786,11 @@ sctp_forward_tsn_spec
     $$ = sctp_forward_tsn_chunk_new($3, $5);
 }
 
+sctp_i_forward_tsn_spec
+: I_FORWARD_TSN '[' opt_cum_tsn ',' opt_i_forward_tsn_stream_identifier']' {
+    $$ = sctp_i_forward_tsn_chunk_new($3, $5);
+}
+
 opt_req_sn
 : REQ_SN '=' INTEGER {
 	if (!is_valid_u32($3)) {
diff --git a/gtests/net/packetdrill/run_packet.c b/gtests/net/packetdrill/run_packet.c
index 65f2cc7fa920898d13eaf364ab48fe5ad5d54468..a1459d72f9cb0bc2c7c18b960b0452e5e62204b3 100644
--- a/gtests/net/packetdrill/run_packet.c
+++ b/gtests/net/packetdrill/run_packet.c
@@ -655,6 +655,7 @@ static int map_inbound_sctp_packet(
 	struct sctp_i_data_chunk *i_data;
 	struct sctp_reconfig_chunk *reconfig;
 	struct sctp_forward_tsn_chunk *forward_tsn;
+	struct sctp_i_forward_tsn_chunk *i_forward_tsn;
 	
 	u32 local_diff, remote_diff;
 	u32 v_tag;
@@ -761,6 +762,10 @@ static int map_inbound_sctp_packet(
 			forward_tsn = (struct sctp_forward_tsn_chunk *) chunk;
 			forward_tsn->cum_tsn = htonl(ntohl(forward_tsn->cum_tsn) + local_diff);
 			break;
+		case SCTP_I_FORWARD_TSN_CHUNK_TYPE:
+			i_forward_tsn = (struct sctp_i_forward_tsn_chunk *) chunk;
+			i_forward_tsn->cum_tsn = htonl(ntohl(i_forward_tsn->cum_tsn) + local_diff);
+			break;
 		case SCTP_RECONFIG_CHUNK_TYPE:
 			reconfig = (struct sctp_reconfig_chunk *)chunk;
 			if (htons(reconfig->length) >= sizeof(struct sctp_reconfig_chunk) + 4) {
@@ -973,6 +978,7 @@ static int map_outbound_live_sctp_packet(
 	struct sctp_i_data_chunk *i_data;
 	struct sctp_reconfig_chunk *reconfig;
 	struct sctp_forward_tsn_chunk *forward_tsn;
+	struct sctp_i_forward_tsn_chunk *i_forward_tsn;
 	u32 local_diff, remote_diff;
 	u16 nr_gap_blocks, nr_dup_tsns, number_of_nr_gap_blocks, i;
 
@@ -1048,6 +1054,10 @@ static int map_outbound_live_sctp_packet(
 			forward_tsn = (struct sctp_forward_tsn_chunk *) chunk;
 			forward_tsn->cum_tsn = htonl(ntohl(forward_tsn->cum_tsn) + local_diff);
 			break;
+		case SCTP_I_FORWARD_TSN_CHUNK_TYPE:
+			i_forward_tsn = (struct sctp_i_forward_tsn_chunk *) chunk;
+			i_forward_tsn->cum_tsn = htonl(ntohl(i_forward_tsn->cum_tsn) + local_diff);
+			break;
 		case SCTP_RECONFIG_CHUNK_TYPE:
 			reconfig = (struct sctp_reconfig_chunk *)chunk;
 			if (reconfig->length > sizeof(struct sctp_reconfig_chunk)) {
@@ -2214,6 +2224,58 @@ static int verify_forward_tsn_chunk(struct sctp_forward_tsn_chunk *actual_chunk,
 	return STATUS_OK;
 }
 
+static u16 get_num_id_blocks_for_i_forward_tsn (u16 packet_length) {
+	return (packet_length - sizeof(struct sctp_i_forward_tsn_chunk)) / sizeof(struct sctp_i_forward_tsn_identifier_block);
+}
+
+static int verify_i_forward_tsn_chunk(struct sctp_i_forward_tsn_chunk *actual_chunk,
+				 struct sctp_i_forward_tsn_chunk *script_chunk,
+				 u32 flags, char **error) {
+	u16 actual_packet_length = ntohs(script_chunk->length);
+	u16 script_packet_length = ntohs(script_chunk->length);
+	u16 actual_nr_id_blocks = get_num_id_blocks_for_i_forward_tsn(actual_packet_length);
+	u16 script_nr_id_blocks = get_num_id_blocks_for_i_forward_tsn(script_packet_length);
+	u16 i;
+	
+	if ((flags & FLAG_I_FORWARD_TSN_CHUNK_CUM_TSN_NOCHECK) == 0) {
+		if (check_field("sctp_i_forward_tsn_cum_tsn",
+				 ntohl(script_chunk->cum_tsn),
+				 ntohl(actual_chunk->cum_tsn),
+				 error) == STATUS_ERR) {
+			return STATUS_ERR;
+		}
+	}
+	
+	if ((flags & FLAG_I_FORWARD_TSN_CHUNK_IDS_NOCHECK) == 0) {
+		if (check_field("nr_id_blocks",
+				 actual_nr_id_blocks,
+				 script_nr_id_blocks,
+				 error) == STATUS_ERR) {
+			return STATUS_ERR;
+		}
+		
+		for (i = 0; i < script_nr_id_blocks; i++) {
+			if (check_field("sctp_i_forward_tsn_stream_identifier",
+		                        ntohs(script_chunk->stream_identifier_blocks[i].stream_identifier),
+		                        ntohs(actual_chunk->stream_identifier_blocks[i].stream_identifier),
+		                        error) == STATUS_ERR ||
+		            check_field("sctp_i_forward_tsn_u_bit",
+		                        ntohs(script_chunk->stream_identifier_blocks[i].reserved),
+		                        ntohs(actual_chunk->stream_identifier_blocks[i].reserved),
+		                        error) == STATUS_ERR ||
+			    check_field("sctp_i_forward_tsn_message_identifier",
+		                        ntohs(script_chunk->stream_identifier_blocks[i].message_identifier),
+		                        ntohs(actual_chunk->stream_identifier_blocks[i].message_identifier),
+		                        error) == STATUS_ERR 
+				) {
+				return STATUS_ERR;
+			}
+		}
+	}
+	
+	return STATUS_OK;
+}
+
 /* Verify that required actual SCTP packet fields are as the script expected. */
 static int verify_sctp(
 	const struct packet *actual_packet,
@@ -2369,6 +2431,11 @@ static int verify_sctp(
 			                               (struct sctp_forward_tsn_chunk *)script_chunk,
 			                               flags, error);
 			break;
+		case SCTP_I_FORWARD_TSN_CHUNK_TYPE:
+			result = verify_i_forward_tsn_chunk((struct sctp_i_forward_tsn_chunk *)actual_chunk,
+			                               (struct sctp_i_forward_tsn_chunk *)script_chunk,
+			                               flags, error);
+			break;
 		default:
 			result = STATUS_ERR;
 			assert(!"unsupported SCTP chunk type");
diff --git a/gtests/net/packetdrill/sctp.h b/gtests/net/packetdrill/sctp.h
index 6a17a416edbba0903362c603adca544e56c051c4..e8d0650b5cdd220bb22306feecd143fc7bde5a64 100644
--- a/gtests/net/packetdrill/sctp.h
+++ b/gtests/net/packetdrill/sctp.h
@@ -57,6 +57,8 @@ struct sctp_common_header {
 #define SCTP_RECONFIG_CHUNK_TYPE			0x82
 #define SCTP_PAD_CHUNK_TYPE				0x84
 #define SCTP_FORWARD_TSN_CHUNK_TYPE                     0xc0
+#define SCTP_I_FORWARD_TSN_CHUNK_TYPE                   0xc2
+
 
 #define MAX_SCTP_CHUNK_BYTES	0xffff
 
@@ -281,6 +283,20 @@ struct sctp_forward_tsn_chunk {
 	struct sctp_stream_identifier_block stream_identifier_blocks[];
 } __packed;
 
+struct sctp_i_forward_tsn_identifier_block {
+	__u16 stream_identifier;
+	__u16 reserved;
+	__u32 message_identifier;
+};
+
+struct sctp_i_forward_tsn_chunk {
+	__u8 type;
+	__u8 flags;
+	__be16 length;
+	__be32 cum_tsn;
+	struct sctp_i_forward_tsn_identifier_block stream_identifier_blocks[];
+} __packed;
+
 #define SCTP_HEARTBEAT_INFORMATION_PARAMETER_TYPE	0x0001
 #define SCTP_IPV4_ADDRESS_PARAMETER_TYPE		0x0005
 #define SCTP_IPV6_ADDRESS_PARAMETER_TYPE		0x0006
diff --git a/gtests/net/packetdrill/sctp_chunk_to_string.c b/gtests/net/packetdrill/sctp_chunk_to_string.c
index 386cab745171ad3cdfeb011dcb8fd398d074da59..51b8c217640d55267c1ac379e6b9cac104ca075a 100644
--- a/gtests/net/packetdrill/sctp_chunk_to_string.c
+++ b/gtests/net/packetdrill/sctp_chunk_to_string.c
@@ -1742,6 +1742,46 @@ static int sctp_forward_tsn_chunk_to_string(
 	return STATUS_OK;
 }
 
+static u16 get_num_id_blocks_for_i_forward_tsn (u16 packet_length) {
+	return (packet_length - sizeof(struct sctp_i_forward_tsn_chunk)) / sizeof(struct sctp_i_forward_tsn_identifier_block);
+}
+
+static int sctp_i_forward_tsn_chunk_to_string(
+	FILE *s,
+	struct sctp_i_forward_tsn_chunk *chunk,
+	char **error)
+{
+	u16 length, i;
+	length = ntohs(chunk->length);
+	u16 num_id_blocks = get_num_id_blocks_for_i_forward_tsn(length);
+	
+	if (length < sizeof(struct sctp_i_forward_tsn_chunk)) {
+		asprintf(error, "I_FORWARD_TSN chunk too short (length=%u)", length);
+		return STATUS_ERR;
+	}
+	
+	fputs("I_FORWARD_TSN[", s);
+	fprintf(s, "flgs=0x%02x, ", chunk->flags);
+	fprintf(s, "len=%u, ", length);
+	fprintf(s, "cum_tsn=%u, ", ntohl(chunk->cum_tsn));
+	
+	fprintf(s, "ids=[");
+	
+	for (i = 0; i < num_id_blocks; i++) {
+		fprintf(s, "{%u,%u,%u}",  
+			ntohs(chunk->stream_identifier_blocks[i].stream_identifier), 
+			ntohs(chunk->stream_identifier_blocks[i].reserved),
+			ntohl(chunk->stream_identifier_blocks[i].message_identifier));
+		if (i != num_id_blocks-1) {
+			fprintf(s, ",");
+		}
+	}
+	
+	fputs("]]", s);
+	
+	return STATUS_OK;
+}
+
 static int sctp_unknown_chunk_to_string(FILE *s,
 					struct sctp_chunk *chunk,
 					char **error)
@@ -1846,6 +1886,10 @@ int sctp_chunk_to_string(FILE *s, struct sctp_chunk *chunk, char **error)
 		result = sctp_forward_tsn_chunk_to_string(s,
 			(struct sctp_forward_tsn_chunk *)chunk, error);
 		break;
+	case SCTP_I_FORWARD_TSN_CHUNK_TYPE:
+		result = sctp_i_forward_tsn_chunk_to_string(s,
+			(struct sctp_i_forward_tsn_chunk *)chunk, error);
+		break;
 	default:
 		result = sctp_unknown_chunk_to_string(s, chunk, error);
 		break;
diff --git a/gtests/net/packetdrill/sctp_packet.c b/gtests/net/packetdrill/sctp_packet.c
index ad7048918e205345628b099d9d2390e607a1a85d..8be43c6e81cd68a191467c364eec22f7dc3184db 100644
--- a/gtests/net/packetdrill/sctp_packet.c
+++ b/gtests/net/packetdrill/sctp_packet.c
@@ -293,6 +293,66 @@ sctp_forward_tsn_ids_list_item_new(u16 stream_identifier, u16 stream_sequence_nu
 	return item;
 }
 
+struct sctp_i_forward_tsn_ids_list *
+sctp_i_forward_tsn_ids_list_new () {
+	struct sctp_i_forward_tsn_ids_list *list;
+
+	list = malloc(sizeof(struct sctp_i_forward_tsn_ids_list));
+	assert(list != NULL);
+	list->first = NULL;
+	list->last = NULL;
+	list->nr_entries = 0;
+	return list;
+}
+
+void
+sctp_i_forward_tsn_ids_list_append(struct sctp_i_forward_tsn_ids_list *list,
+			          struct sctp_i_forward_tsn_ids_list_item *item) {
+	assert(item->next == NULL);
+	if (list->last == NULL) {
+		assert(list->first == NULL);
+		assert(list->nr_entries == 0);
+		list->first = item;
+	} else {
+		assert(list->first != NULL);
+		list->last->next = item;
+	}
+	list->last = item;
+	list->nr_entries++;
+}
+
+void sctp_i_forward_tsn_ids_list_free (struct sctp_i_forward_tsn_ids_list *list) {
+	struct sctp_i_forward_tsn_ids_list_item *current_item, *next_item;
+
+	if (list == NULL) {
+		return;
+	}
+	current_item = list->first;
+	while (current_item != NULL) {
+		assert(list->nr_entries > 0);
+		next_item = current_item->next;
+		assert(next_item != NULL || current_item == list->last);
+		free(current_item);
+		current_item = next_item;
+		list->nr_entries--;
+	}
+	assert(list->nr_entries == 0);
+	free(list);
+}
+
+struct sctp_i_forward_tsn_ids_list_item *
+sctp_i_forward_tsn_ids_list_item_new(u16 stream_identifier, u16 reserved, u32 message_identifier) {
+	struct sctp_i_forward_tsn_ids_list_item *item;
+
+	item = malloc(sizeof(struct sctp_i_forward_tsn_ids_list_item));
+	assert(item != NULL);
+	item->next = NULL;
+	item->stream_identifier = stream_identifier;
+	item->reserved = reserved;
+	item->message_identifier = message_identifier;
+	return item;
+}
+
 struct sctp_address_type_list *
 sctp_address_type_list_new(void)
 {
@@ -786,6 +846,8 @@ sctp_sack_chunk_new(s64 flgs, s64 cum_tsn, s64 a_rwnd,
 			chunk->block[i].gap.end = htons(item->block.gap.end);
 		}
 		assert((i == nr_gaps) && (item == NULL));
+		sctp_sack_block_list_free(gaps);
+		
 	}
 	if (dups != NULL) {
 		for (i = 0, item = dups->first;
@@ -794,6 +856,7 @@ sctp_sack_chunk_new(s64 flgs, s64 cum_tsn, s64 a_rwnd,
 			chunk->block[i + nr_gaps].tsn= htonl(item->block.tsn);
 		}
 		assert((i == nr_dups) && (item == NULL));
+		sctp_sack_block_list_free(dups);
 	}
 	return sctp_chunk_list_item_new((struct sctp_chunk *)chunk,
 	                                length, flags,
@@ -874,6 +937,7 @@ sctp_nr_sack_chunk_new(s64 flgs, s64 cum_tsn, s64 a_rwnd,
 			chunk->block[i].gap.end = htons(item->block.gap.end);
 		}
 		assert((i == nr_gaps) && (item == NULL));
+		sctp_sack_block_list_free(gaps);
 	}
 	if (nr_gaps_list != NULL) {
 		for (i = 0, item = nr_gaps_list->first;
@@ -883,6 +947,7 @@ sctp_nr_sack_chunk_new(s64 flgs, s64 cum_tsn, s64 a_rwnd,
 			chunk->block[i + nr_gaps].gap.end = htons(item->block.gap.end);
 		}
 		assert((i == number_of_nr_gaps) && (item == NULL));
+		sctp_sack_block_list_free(nr_gaps_list);
 	}
 	if (dups != NULL) {
 		for (i = 0, item = dups->first;
@@ -891,6 +956,7 @@ sctp_nr_sack_chunk_new(s64 flgs, s64 cum_tsn, s64 a_rwnd,
 			chunk->block[i + nr_gaps + number_of_nr_gaps].tsn= htonl(item->block.tsn);
 		}
 		assert((i == nr_dups) && (item == NULL));
+		sctp_sack_block_list_free(dups);
 	}
 	return sctp_chunk_list_item_new((struct sctp_chunk *)chunk,
 	                                length, flags,
@@ -1440,11 +1506,11 @@ sctp_pad_chunk_new(s64 flgs, s64 len, u8* padding)
 }
 
 struct sctp_chunk_list_item *
-sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *sids) {
+sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *ids_list) {
 	struct sctp_forward_tsn_chunk *chunk;
 	struct sctp_forward_tsn_ids_list_item *item;
 	
-	DEBUGP("sctp_forward_tsn_chunk_new called with cum_tsn = %d and sids_list = %p", cum_tsn, sids);
+	DEBUGP("sctp_forward_tsn_chunk_new called with cum_tsn = %d and sids_list = %p", cum_tsn, ids_list);
 	
 	u32 flags;
 	u32 length;
@@ -1452,12 +1518,12 @@ sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *sids)
 
 	flags = 0;
 	length = sizeof(struct sctp_forward_tsn_chunk);
-	if (sids == NULL) {
+	if (ids_list == NULL) {
 		nr_sids = 0;
 		flags |= FLAG_CHUNK_LENGTH_NOCHECK;
 		flags |= FLAG_FORWARD_TSN_CHUNK_IDS_NOCHECK;
 	} else {
-		nr_sids = sids->nr_entries;
+		nr_sids = ids_list->nr_entries;
 		length += nr_sids * sizeof(struct sctp_stream_identifier_block);
 	}
 	
@@ -1475,12 +1541,12 @@ sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *sids)
 		chunk->cum_tsn = htonl((u32)cum_tsn);
 	}
 	
-	if (nr_sids == 0 || sids == NULL) {
+	if (nr_sids == 0 || ids_list == NULL) {
 		flags |= FLAG_FORWARD_TSN_CHUNK_IDS_NOCHECK;
 	}
 
-	if (sids != NULL) {
-		for (i = 0, item = sids->first;
+	if (ids_list != NULL) {
+		for (i = 0, item = ids_list->first;
 		     (i < nr_sids) && (item != NULL);
 		     i++, item = item->next) {
 			chunk->stream_identifier_blocks[i].stream= htons(item->stream_identifier);
@@ -1488,7 +1554,68 @@ sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *sids)
 		}
 		
 		assert((i == nr_sids) && (item == NULL));
+		sctp_forward_tsn_ids_list_free(ids_list);
+	}
+	
+	return sctp_chunk_list_item_new((struct sctp_chunk *)chunk,
+	                                length, flags,
+	                                sctp_parameter_list_new(),
+	                                sctp_cause_list_new());
+}
+
+struct sctp_chunk_list_item *
+sctp_i_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_i_forward_tsn_ids_list *ids_list) {
+	struct sctp_i_forward_tsn_chunk *chunk;
+	struct sctp_i_forward_tsn_ids_list_item *item;
+	
+	DEBUGP("sctp_i_forward_tsn_chunk_new called with cum_tsn = %d and sids_list = %p", cum_tsn, ids_list);
+	
+	u32 flags;
+	u32 length;
+	u16 i, nr_ids;
+
+	flags = 0;
+	length = sizeof(struct sctp_i_forward_tsn_chunk);
+	if (ids_list == NULL) {
+		nr_ids = 0;
+		flags |= FLAG_CHUNK_LENGTH_NOCHECK;
+		flags |= FLAG_I_FORWARD_TSN_CHUNK_IDS_NOCHECK;
+	} else {
+		nr_ids = ids_list->nr_entries;
+		length += nr_ids * sizeof(struct sctp_i_forward_tsn_identifier_block);
 	}
+	
+	assert(is_valid_u16(length));
+	assert(length % 4 == 0);
+	chunk = malloc(length);
+	assert(chunk != NULL);
+	chunk->type = SCTP_I_FORWARD_TSN_CHUNK_TYPE;
+	chunk->flags = 0;
+	chunk->length = htons(length);
+	if (cum_tsn == -1) {
+		chunk->cum_tsn = htonl(0);
+		flags |= FLAG_I_FORWARD_TSN_CHUNK_CUM_TSN_NOCHECK;
+	} else {
+		chunk->cum_tsn = htonl((u32)cum_tsn);
+	}
+	
+	if (nr_ids == 0 || ids_list == NULL) {
+		flags |= FLAG_I_FORWARD_TSN_CHUNK_IDS_NOCHECK;
+	}
+
+	if (ids_list != NULL) {
+		for (i = 0, item = ids_list->first;
+		     (i < nr_ids) && (item != NULL);
+		     i++, item = item->next) {
+			chunk->stream_identifier_blocks[i].stream_identifier= htons(item->stream_identifier);
+			chunk->stream_identifier_blocks[i].reserved = htons(item->reserved); 
+			chunk->stream_identifier_blocks[i].message_identifier = htonl(item->message_identifier);
+		}
+		
+		assert((i == nr_ids) && (item == NULL));
+		sctp_i_forward_tsn_ids_list_free(ids_list);
+	}
+	
 	return sctp_chunk_list_item_new((struct sctp_chunk *)chunk,
 	                                length, flags,
 	                                sctp_parameter_list_new(),
diff --git a/gtests/net/packetdrill/sctp_packet.h b/gtests/net/packetdrill/sctp_packet.h
index f41521fd274e2ddf4f8d12a56443eaff59e9e68e..6fece3a3f5de69163eadb8e49f379df1eb8462db 100644
--- a/gtests/net/packetdrill/sctp_packet.h
+++ b/gtests/net/packetdrill/sctp_packet.h
@@ -123,13 +123,37 @@ void
 sctp_forward_tsn_ids_list_append(struct sctp_forward_tsn_ids_list *list,
 			          struct sctp_forward_tsn_ids_list_item *item);
 
-// TODO: where to call this freeing method... sctp_sack_block_list_free and sctp_byte_list_free are unused...?
 void sctp_forward_tsn_ids_list_free (struct sctp_forward_tsn_ids_list *list);
 
 struct sctp_forward_tsn_ids_list_item *
 sctp_forward_tsn_ids_list_item_new(u16 stream_identifier, u16 stream_sequence_number);
 
 
+struct sctp_i_forward_tsn_ids_list_item {
+	struct sctp_i_forward_tsn_ids_list_item *next;
+	u16 stream_identifier;
+        u16 reserved;
+	u32 message_identifier;
+}; 
+
+struct sctp_i_forward_tsn_ids_list {
+	struct sctp_i_forward_tsn_ids_list_item *first;
+	struct sctp_i_forward_tsn_ids_list_item *last;
+	u16 nr_entries;
+};
+
+struct sctp_i_forward_tsn_ids_list *
+sctp_i_forward_tsn_ids_list_new ();
+
+void
+sctp_i_forward_tsn_ids_list_append(struct sctp_i_forward_tsn_ids_list *list,
+			          struct sctp_i_forward_tsn_ids_list_item *item);
+
+void sctp_i_forward_tsn_ids_list_free (struct sctp_i_forward_tsn_ids_list *list);
+
+struct sctp_i_forward_tsn_ids_list_item *
+sctp_i_forward_tsn_ids_list_item_new(u16 stream_identifier, u16 reserved, u32 message_identifier);
+
 struct sctp_address_type_list_item {
 	struct sctp_address_type_list_item *next;
 	u16 address_type;
@@ -357,6 +381,12 @@ sctp_pad_chunk_new(s64 flgs, s64 len, u8* padding);
 struct sctp_chunk_list_item *
 sctp_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_forward_tsn_ids_list *sids_list);
 
+#define FLAG_I_FORWARD_TSN_CHUNK_CUM_TSN_NOCHECK  0x00000100
+#define FLAG_I_FORWARD_TSN_CHUNK_IDS_NOCHECK      0x00000200
+
+struct sctp_chunk_list_item *
+sctp_i_forward_tsn_chunk_new(u32 cum_tsn, struct sctp_i_forward_tsn_ids_list *ids_list);
+
 struct sctp_chunk_list_item *
 sctp_reconfig_chunk_new(s64 flgs, struct sctp_parameter_list *parameters);