| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944 | #include "test_tcp_oos.h"#include "lwip/tcp_impl.h"#include "lwip/stats.h"#include "tcp_helper.h"#if !LWIP_STATS || !TCP_STATS || !MEMP_STATS#error "This tests needs TCP- and MEMP-statistics enabled"#endif#if !TCP_QUEUE_OOSEQ#error "This tests needs TCP_QUEUE_OOSEQ enabled"#endif/** CHECK_SEGMENTS_ON_OOSEQ: * 1: check count, seqno and len of segments on pcb->ooseq (strict) * 0: only check that bytes are received in correct order (less strict) */#define CHECK_SEGMENTS_ON_OOSEQ 1#if CHECK_SEGMENTS_ON_OOSEQ#define EXPECT_OOSEQ(x) EXPECT(x)#else#define EXPECT_OOSEQ(x)#endif/* helper functions *//** Get the numbers of segments on the ooseq list */static int tcp_oos_count(struct tcp_pcb* pcb){  int num = 0;  struct tcp_seg* seg = pcb->ooseq;  while(seg != NULL) {    num++;    seg = seg->next;  }  return num;}/** Get the numbers of pbufs on the ooseq list */static int tcp_oos_pbuf_count(struct tcp_pcb* pcb){  int num = 0;  struct tcp_seg* seg = pcb->ooseq;  while(seg != NULL) {    num += pbuf_clen(seg->p);    seg = seg->next;  }  return num;}/** Get the seqno of a segment (by index) on the ooseq list * * @param pcb the pcb to check for ooseq segments * @param seg_index index of the segment on the ooseq list * @return seqno of the segment */static u32_ttcp_oos_seg_seqno(struct tcp_pcb* pcb, int seg_index){  int num = 0;  struct tcp_seg* seg = pcb->ooseq;  /* then check the actual segment */  while(seg != NULL) {    if(num == seg_index) {      return seg->tcphdr->seqno;    }    num++;    seg = seg->next;  }  fail();  return 0;}/** Get the tcplen (datalen + SYN/FIN) of a segment (by index) on the ooseq list * * @param pcb the pcb to check for ooseq segments * @param seg_index index of the segment on the ooseq list * @return tcplen of the segment */static inttcp_oos_seg_tcplen(struct tcp_pcb* pcb, int seg_index){  int num = 0;  struct tcp_seg* seg = pcb->ooseq;  /* then check the actual segment */  while(seg != NULL) {    if(num == seg_index) {      return TCP_TCPLEN(seg);    }    num++;    seg = seg->next;  }  fail();  return -1;}/** Get the tcplen (datalen + SYN/FIN) of all segments on the ooseq list * * @param pcb the pcb to check for ooseq segments * @return tcplen of all segment */static inttcp_oos_tcplen(struct tcp_pcb* pcb){  int len = 0;  struct tcp_seg* seg = pcb->ooseq;  /* then check the actual segment */  while(seg != NULL) {    len += TCP_TCPLEN(seg);    seg = seg->next;  }  return len;}/* Setup/teardown functions */static voidtcp_oos_setup(void){  tcp_remove_all();}static voidtcp_oos_teardown(void){  tcp_remove_all();}/* Test functions *//** create multiple segments and pass them to tcp_input in a wrong * order to see if ooseq-caching works correctly * FIN is received in out-of-sequence segments only */START_TEST(test_tcp_recv_ooseq_FIN_OOSEQ){  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *p_8_9, *p_4_8, *p_4_10, *p_2_14, *p_fin, *pinseq;  char data[] = {     1,  2,  3,  4,     5,  6,  7,  8,     9, 10, 11, 12,    13, 14, 15, 16};  ip_addr_t remote_ip, local_ip;  u16_t data_len;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  LWIP_UNUSED_ARG(_i);  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  data_len = sizeof(data);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = data_len;  counters.expected_data = data;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  /* create segments */  /* pinseq is sent as last segment! */  pinseq = tcp_create_rx_segment(pcb, &data[0],  4, 0, 0, TCP_ACK);  /* p1: 8 bytes before FIN */  /*     seqno: 8..16 */  p_8_9  = tcp_create_rx_segment(pcb, &data[8],  8, 8, 0, TCP_ACK|TCP_FIN);  /* p2: 4 bytes before p1, including the first 4 bytes of p1 (partly duplicate) */  /*     seqno: 4..11 */  p_4_8  = tcp_create_rx_segment(pcb, &data[4],  8, 4, 0, TCP_ACK);  /* p3: same as p2 but 2 bytes longer */  /*     seqno: 4..13 */  p_4_10 = tcp_create_rx_segment(pcb, &data[4], 10, 4, 0, TCP_ACK);  /* p4: 14 bytes before FIN, includes data from p1 and p2, plus partly from pinseq */  /*     seqno: 2..15 */  p_2_14 = tcp_create_rx_segment(pcb, &data[2], 14, 2, 0, TCP_ACK);  /* FIN, seqno 16 */  p_fin  = tcp_create_rx_segment(pcb,     NULL,  0,16, 0, TCP_ACK|TCP_FIN);  EXPECT(pinseq != NULL);  EXPECT(p_8_9 != NULL);  EXPECT(p_4_8 != NULL);  EXPECT(p_4_10 != NULL);  EXPECT(p_2_14 != NULL);  EXPECT(p_fin != NULL);  if ((pinseq != NULL) && (p_8_9 != NULL) && (p_4_8 != NULL) && (p_4_10 != NULL) && (p_2_14 != NULL) && (p_fin != NULL)) {    /* pass the segment to tcp_input */    test_tcp_input(p_8_9, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 8);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 9); /* includes FIN */    /* pass the segment to tcp_input */    test_tcp_input(p_4_8, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 4);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 4);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 1) == 8);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 1) == 9); /* includes FIN */    /* pass the segment to tcp_input */    test_tcp_input(p_4_10, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* ooseq queue: unchanged */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 4);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 4);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 1) == 8);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 1) == 9); /* includes FIN */    /* pass the segment to tcp_input */    test_tcp_input(p_2_14, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 2);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 15); /* includes FIN */    /* pass the segment to tcp_input */    test_tcp_input(p_fin, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* ooseq queue: unchanged */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 2);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 15); /* includes FIN */    /* pass the segment to tcp_input */    test_tcp_input(pinseq, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 1);    EXPECT(counters.recv_calls == 1);    EXPECT(counters.recved_bytes == data_len);    EXPECT(counters.err_calls == 0);    EXPECT(pcb->ooseq == NULL);  }  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);}END_TEST/** create multiple segments and pass them to tcp_input in a wrong * order to see if ooseq-caching works correctly * FIN is received IN-SEQUENCE at the end */START_TEST(test_tcp_recv_ooseq_FIN_INSEQ){  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *p_1_2, *p_4_8, *p_3_11, *p_2_12, *p_15_1, *p_15_1a, *pinseq, *pinseqFIN;  char data[] = {     1,  2,  3,  4,     5,  6,  7,  8,     9, 10, 11, 12,    13, 14, 15, 16};  ip_addr_t remote_ip, local_ip;  u16_t data_len;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  LWIP_UNUSED_ARG(_i);  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  data_len = sizeof(data);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = data_len;  counters.expected_data = data;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  /* create segments */  /* p1: 7 bytes - 2 before FIN */  /*     seqno: 1..2 */  p_1_2  = tcp_create_rx_segment(pcb, &data[1],  2, 1, 0, TCP_ACK);  /* p2: 4 bytes before p1, including the first 4 bytes of p1 (partly duplicate) */  /*     seqno: 4..11 */  p_4_8  = tcp_create_rx_segment(pcb, &data[4],  8, 4, 0, TCP_ACK);  /* p3: same as p2 but 2 bytes longer and one byte more at the front */  /*     seqno: 3..13 */  p_3_11 = tcp_create_rx_segment(pcb, &data[3], 11, 3, 0, TCP_ACK);  /* p4: 13 bytes - 2 before FIN - should be ignored as contained in p1 and p3 */  /*     seqno: 2..13 */  p_2_12 = tcp_create_rx_segment(pcb, &data[2], 12, 2, 0, TCP_ACK);  /* pinseq is the first segment that is held back to create ooseq! */  /*     seqno: 0..3 */  pinseq = tcp_create_rx_segment(pcb, &data[0],  4, 0, 0, TCP_ACK);  /* p5: last byte before FIN */  /*     seqno: 15 */  p_15_1 = tcp_create_rx_segment(pcb, &data[15], 1, 15, 0, TCP_ACK);  /* p6: same as p5, should be ignored */  p_15_1a= tcp_create_rx_segment(pcb, &data[15], 1, 15, 0, TCP_ACK);  /* pinseqFIN: last 2 bytes plus FIN */  /*     only segment containing seqno 14 and FIN */  pinseqFIN = tcp_create_rx_segment(pcb,  &data[14], 2, 14, 0, TCP_ACK|TCP_FIN);  EXPECT(pinseq != NULL);  EXPECT(p_1_2 != NULL);  EXPECT(p_4_8 != NULL);  EXPECT(p_3_11 != NULL);  EXPECT(p_2_12 != NULL);  EXPECT(p_15_1 != NULL);  EXPECT(p_15_1a != NULL);  EXPECT(pinseqFIN != NULL);  if ((pinseq != NULL) && (p_1_2 != NULL) && (p_4_8 != NULL) && (p_3_11 != NULL) && (p_2_12 != NULL)    && (p_15_1 != NULL) && (p_15_1a != NULL) && (pinseqFIN != NULL)) {    /* pass the segment to tcp_input */    test_tcp_input(p_1_2, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 1);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 2);    /* pass the segment to tcp_input */    test_tcp_input(p_4_8, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 1);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 1) == 4);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 1) == 8);    /* pass the segment to tcp_input */    test_tcp_input(p_3_11, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 1);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 2);    /* p_3_11 has removed p_4_8 from ooseq */    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 1) == 3);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 1) == 11);    /* pass the segment to tcp_input */    test_tcp_input(p_2_12, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 2);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 1);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 1) == 2);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 1) == 12);    /* pass the segment to tcp_input */    test_tcp_input(pinseq, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 1);    EXPECT(counters.recved_bytes == 14);    EXPECT(counters.err_calls == 0);    EXPECT(pcb->ooseq == NULL);    /* pass the segment to tcp_input */    test_tcp_input(p_15_1, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 1);    EXPECT(counters.recved_bytes == 14);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 15);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 1);    /* pass the segment to tcp_input */    test_tcp_input(p_15_1a, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 1);    EXPECT(counters.recved_bytes == 14);    EXPECT(counters.err_calls == 0);    /* check ooseq queue: unchanged */    EXPECT_OOSEQ(tcp_oos_count(pcb) == 1);    EXPECT_OOSEQ(tcp_oos_seg_seqno(pcb, 0) == 15);    EXPECT_OOSEQ(tcp_oos_seg_tcplen(pcb, 0) == 1);    /* pass the segment to tcp_input */    test_tcp_input(pinseqFIN, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 1);    EXPECT(counters.recv_calls == 2);    EXPECT(counters.recved_bytes == data_len);    EXPECT(counters.err_calls == 0);    EXPECT(pcb->ooseq == NULL);  }  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);}END_TESTstatic char data_full_wnd[TCP_WND];/** create multiple segments and pass them to tcp_input with the first segment missing * to simulate overruning the rxwin with ooseq queueing enabled */START_TEST(test_tcp_recv_ooseq_overrun_rxwin){#if !TCP_OOSEQ_MAX_BYTES && !TCP_OOSEQ_MAX_PBUFS  int i, k;  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *pinseq, *p_ovr;  ip_addr_t remote_ip, local_ip;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  int datalen = 0;  int datalen2;  for(i = 0; i < sizeof(data_full_wnd); i++) {    data_full_wnd[i] = (char)i;  }  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = TCP_WND;  counters.expected_data = data_full_wnd;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  pcb->rcv_nxt = 0x8000;  /* create segments */  /* pinseq is sent as last segment! */  pinseq = tcp_create_rx_segment(pcb, &data_full_wnd[0],  TCP_MSS, 0, 0, TCP_ACK);  for(i = TCP_MSS, k = 0; i < TCP_WND; i += TCP_MSS, k++) {    int count, expected_datalen;    struct pbuf *p = tcp_create_rx_segment(pcb, &data_full_wnd[TCP_MSS*(k+1)],                                           TCP_MSS, TCP_MSS*(k+1), 0, TCP_ACK);    EXPECT_RET(p != NULL);    /* pass the segment to tcp_input */    test_tcp_input(p, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    count = tcp_oos_count(pcb);    EXPECT_OOSEQ(count == k+1);    datalen = tcp_oos_tcplen(pcb);    if (i + TCP_MSS < TCP_WND) {      expected_datalen = (k+1)*TCP_MSS;    } else {      expected_datalen = TCP_WND - TCP_MSS;    }    if (datalen != expected_datalen) {      EXPECT_OOSEQ(datalen == expected_datalen);    }  }  /* pass in one more segment, cleary overrunning the rxwin */  p_ovr = tcp_create_rx_segment(pcb, &data_full_wnd[TCP_MSS*(k+1)], TCP_MSS, TCP_MSS*(k+1), 0, TCP_ACK);  EXPECT_RET(p_ovr != NULL);  /* pass the segment to tcp_input */  test_tcp_input(p_ovr, &netif);  /* check if counters are as expected */  EXPECT(counters.close_calls == 0);  EXPECT(counters.recv_calls == 0);  EXPECT(counters.recved_bytes == 0);  EXPECT(counters.err_calls == 0);  /* check ooseq queue */  EXPECT_OOSEQ(tcp_oos_count(pcb) == k);  datalen2 = tcp_oos_tcplen(pcb);  EXPECT_OOSEQ(datalen == datalen2);  /* now pass inseq */  test_tcp_input(pinseq, &netif);  EXPECT(pcb->ooseq == NULL);  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);#endif /* !TCP_OOSEQ_MAX_BYTES && !TCP_OOSEQ_MAX_PBUFS */  LWIP_UNUSED_ARG(_i);}END_TESTSTART_TEST(test_tcp_recv_ooseq_max_bytes){#if TCP_OOSEQ_MAX_BYTES && (TCP_OOSEQ_MAX_BYTES < (TCP_WND + 1)) && (PBUF_POOL_BUFSIZE >= (TCP_MSS + PBUF_LINK_HLEN + PBUF_IP_HLEN + PBUF_TRANSPORT_HLEN))  int i, k;  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *p_ovr;  ip_addr_t remote_ip, local_ip;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  int datalen = 0;  int datalen2;  for(i = 0; i < sizeof(data_full_wnd); i++) {    data_full_wnd[i] = (char)i;  }  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = TCP_WND;  counters.expected_data = data_full_wnd;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  pcb->rcv_nxt = 0x8000;  /* don't 'recv' the first segment (1 byte) so that all other segments will be ooseq */  /* create segments and 'recv' them */  for(k = 1, i = 1; k < TCP_OOSEQ_MAX_BYTES; k += TCP_MSS, i++) {    int count;    struct pbuf *p = tcp_create_rx_segment(pcb, &data_full_wnd[k],                                           TCP_MSS, k, 0, TCP_ACK);    EXPECT_RET(p != NULL);    EXPECT_RET(p->next == NULL);    /* pass the segment to tcp_input */    test_tcp_input(p, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    count = tcp_oos_pbuf_count(pcb);    EXPECT_OOSEQ(count == i);    datalen = tcp_oos_tcplen(pcb);    EXPECT_OOSEQ(datalen == (i * TCP_MSS));  }  /* pass in one more segment, overrunning the limit */  p_ovr = tcp_create_rx_segment(pcb, &data_full_wnd[k+1], 1, k+1, 0, TCP_ACK);  EXPECT_RET(p_ovr != NULL);  /* pass the segment to tcp_input */  test_tcp_input(p_ovr, &netif);  /* check if counters are as expected */  EXPECT(counters.close_calls == 0);  EXPECT(counters.recv_calls == 0);  EXPECT(counters.recved_bytes == 0);  EXPECT(counters.err_calls == 0);  /* check ooseq queue (ensure the new segment was not accepted) */  EXPECT_OOSEQ(tcp_oos_count(pcb) == (i-1));  datalen2 = tcp_oos_tcplen(pcb);  EXPECT_OOSEQ(datalen2 == ((i-1) * TCP_MSS));  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);#endif /* TCP_OOSEQ_MAX_BYTES && (TCP_OOSEQ_MAX_BYTES < (TCP_WND + 1)) && (PBUF_POOL_BUFSIZE >= (TCP_MSS + PBUF_LINK_HLEN + PBUF_IP_HLEN + PBUF_TRANSPORT_HLEN)) */  LWIP_UNUSED_ARG(_i);}END_TESTSTART_TEST(test_tcp_recv_ooseq_max_pbufs){#if TCP_OOSEQ_MAX_PBUFS && (TCP_OOSEQ_MAX_PBUFS < ((TCP_WND / TCP_MSS) + 1)) && (PBUF_POOL_BUFSIZE >= (TCP_MSS + PBUF_LINK_HLEN + PBUF_IP_HLEN + PBUF_TRANSPORT_HLEN))  int i;  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *p_ovr;  ip_addr_t remote_ip, local_ip;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  int datalen = 0;  int datalen2;  for(i = 0; i < sizeof(data_full_wnd); i++) {    data_full_wnd[i] = (char)i;  }  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = TCP_WND;  counters.expected_data = data_full_wnd;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  pcb->rcv_nxt = 0x8000;  /* don't 'recv' the first segment (1 byte) so that all other segments will be ooseq */  /* create segments and 'recv' them */  for(i = 1; i <= TCP_OOSEQ_MAX_PBUFS; i++) {    int count;    struct pbuf *p = tcp_create_rx_segment(pcb, &data_full_wnd[i],                                           1, i, 0, TCP_ACK);    EXPECT_RET(p != NULL);    EXPECT_RET(p->next == NULL);    /* pass the segment to tcp_input */    test_tcp_input(p, &netif);    /* check if counters are as expected */    EXPECT(counters.close_calls == 0);    EXPECT(counters.recv_calls == 0);    EXPECT(counters.recved_bytes == 0);    EXPECT(counters.err_calls == 0);    /* check ooseq queue */    count = tcp_oos_pbuf_count(pcb);    EXPECT_OOSEQ(count == i);    datalen = tcp_oos_tcplen(pcb);    EXPECT_OOSEQ(datalen == i);  }  /* pass in one more segment, overrunning the limit */  p_ovr = tcp_create_rx_segment(pcb, &data_full_wnd[i+1], 1, i+1, 0, TCP_ACK);  EXPECT_RET(p_ovr != NULL);  /* pass the segment to tcp_input */  test_tcp_input(p_ovr, &netif);  /* check if counters are as expected */  EXPECT(counters.close_calls == 0);  EXPECT(counters.recv_calls == 0);  EXPECT(counters.recved_bytes == 0);  EXPECT(counters.err_calls == 0);  /* check ooseq queue (ensure the new segment was not accepted) */  EXPECT_OOSEQ(tcp_oos_count(pcb) == (i-1));  datalen2 = tcp_oos_tcplen(pcb);  EXPECT_OOSEQ(datalen2 == (i-1));  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);#endif /* TCP_OOSEQ_MAX_PBUFS && (TCP_OOSEQ_MAX_BYTES < (TCP_WND + 1)) && (PBUF_POOL_BUFSIZE >= (TCP_MSS + PBUF_LINK_HLEN + PBUF_IP_HLEN + PBUF_TRANSPORT_HLEN)) */  LWIP_UNUSED_ARG(_i);}END_TESTstatic voidcheck_rx_counters(struct tcp_pcb *pcb, struct test_tcp_counters *counters, u32_t exp_close_calls, u32_t exp_rx_calls,                  u32_t exp_rx_bytes, u32_t exp_err_calls, int exp_oos_count, int exp_oos_len){  int oos_len;  EXPECT(counters->close_calls == exp_close_calls);  EXPECT(counters->recv_calls == exp_rx_calls);  EXPECT(counters->recved_bytes == exp_rx_bytes);  EXPECT(counters->err_calls == exp_err_calls);  /* check that pbuf is queued in ooseq */  EXPECT_OOSEQ(tcp_oos_count(pcb) == exp_oos_count);  oos_len = tcp_oos_tcplen(pcb);  EXPECT_OOSEQ(exp_oos_len == oos_len);}/* this test uses 4 packets: * - data (len=TCP_MSS) * - FIN * - data after FIN (len=1) (invalid) * - 2nd FIN (invalid) * * the parameter 'delay_packet' is a bitmask that choses which on these packets is ooseq */static void test_tcp_recv_ooseq_double_FINs(int delay_packet){  int i, k;  struct test_tcp_counters counters;  struct tcp_pcb* pcb;  struct pbuf *p_normal_fin, *p_data_after_fin, *p, *p_2nd_fin_ooseq;  ip_addr_t remote_ip, local_ip;  u16_t remote_port = 0x100, local_port = 0x101;  struct netif netif;  u32_t exp_rx_calls = 0, exp_rx_bytes = 0, exp_close_calls = 0, exp_oos_pbufs = 0, exp_oos_tcplen = 0;  int first_dropped = 0xff;  int last_dropped = 0;  for(i = 0; i < sizeof(data_full_wnd); i++) {    data_full_wnd[i] = (char)i;  }  /* initialize local vars */  memset(&netif, 0, sizeof(netif));  IP4_ADDR(&local_ip, 192, 168, 1, 1);  IP4_ADDR(&remote_ip, 192, 168, 1, 2);  /* initialize counter struct */  memset(&counters, 0, sizeof(counters));  counters.expected_data_len = TCP_WND;  counters.expected_data = data_full_wnd;  /* create and initialize the pcb */  pcb = test_tcp_new_counters_pcb(&counters);  EXPECT_RET(pcb != NULL);  tcp_set_state(pcb, ESTABLISHED, &local_ip, &remote_ip, local_port, remote_port);  pcb->rcv_nxt = 0x8000;  /* create segments */  p = tcp_create_rx_segment(pcb, &data_full_wnd[0], TCP_MSS, 0, 0, TCP_ACK);  p_normal_fin = tcp_create_rx_segment(pcb, NULL, 0, TCP_MSS, 0, TCP_ACK|TCP_FIN);  k = 1;  p_data_after_fin = tcp_create_rx_segment(pcb, &data_full_wnd[TCP_MSS+1], k, TCP_MSS+1, 0, TCP_ACK);  p_2nd_fin_ooseq = tcp_create_rx_segment(pcb, NULL, 0, TCP_MSS+1+k, 0, TCP_ACK|TCP_FIN);  if(delay_packet & 1) {    /* drop normal data */    first_dropped = 1;    last_dropped = 1;  } else {    /* send normal data */    test_tcp_input(p, &netif);    exp_rx_calls++;    exp_rx_bytes += TCP_MSS;  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 2) {    /* drop FIN */    if(first_dropped > 2) {      first_dropped = 2;    }    last_dropped = 2;  } else {    /* send FIN */    test_tcp_input(p_normal_fin, &netif);    if (first_dropped < 2) {      /* already dropped packets, this one is ooseq */      exp_oos_pbufs++;      exp_oos_tcplen++;    } else {      /* inseq */      exp_close_calls++;    }  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 4) {    /* drop data-after-FIN */    if(first_dropped > 3) {      first_dropped = 3;    }    last_dropped = 3;  } else {    /* send data-after-FIN */    test_tcp_input(p_data_after_fin, &netif);    if (first_dropped < 3) {      /* already dropped packets, this one is ooseq */      if (delay_packet & 2) {        /* correct FIN was ooseq */        exp_oos_pbufs++;        exp_oos_tcplen += k;      }    } else {      /* inseq: no change */    }  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 8) {    /* drop 2nd-FIN */    if(first_dropped > 4) {      first_dropped = 4;    }    last_dropped = 4;  } else {    /* send 2nd-FIN */    test_tcp_input(p_2nd_fin_ooseq, &netif);    if (first_dropped < 3) {      /* already dropped packets, this one is ooseq */      if (delay_packet & 2) {        /* correct FIN was ooseq */        exp_oos_pbufs++;        exp_oos_tcplen++;      }    } else {      /* inseq: no change */    }  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 1) {    /* dropped normal data before */    test_tcp_input(p, &netif);    exp_rx_calls++;    exp_rx_bytes += TCP_MSS;    if((delay_packet & 2) == 0) {      /* normal FIN was NOT delayed */      exp_close_calls++;      exp_oos_pbufs = exp_oos_tcplen = 0;    }  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 2) {    /* dropped normal FIN before */    test_tcp_input(p_normal_fin, &netif);    exp_close_calls++;    exp_oos_pbufs = exp_oos_tcplen = 0;  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 4) {    /* dropped data-after-FIN before */    test_tcp_input(p_data_after_fin, &netif);  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  if(delay_packet & 8) {    /* dropped 2nd-FIN before */    test_tcp_input(p_2nd_fin_ooseq, &netif);  }  /* check if counters are as expected */  check_rx_counters(pcb, &counters, exp_close_calls, exp_rx_calls, exp_rx_bytes, 0, exp_oos_pbufs, exp_oos_tcplen);  /* check that ooseq data has been dumped */  EXPECT(pcb->ooseq == NULL);  /* make sure the pcb is freed */  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 1);  tcp_abort(pcb);  EXPECT(lwip_stats.memp[MEMP_TCP_PCB].used == 0);}/** create multiple segments and pass them to tcp_input with the first segment missing * to simulate overruning the rxwin with ooseq queueing enabled */#define FIN_TEST(name, num) \  START_TEST(name) \  { \    LWIP_UNUSED_ARG(_i); \    test_tcp_recv_ooseq_double_FINs(num); \  } \  END_TESTFIN_TEST(test_tcp_recv_ooseq_double_FIN_0, 0)FIN_TEST(test_tcp_recv_ooseq_double_FIN_1, 1)FIN_TEST(test_tcp_recv_ooseq_double_FIN_2, 2)FIN_TEST(test_tcp_recv_ooseq_double_FIN_3, 3)FIN_TEST(test_tcp_recv_ooseq_double_FIN_4, 4)FIN_TEST(test_tcp_recv_ooseq_double_FIN_5, 5)FIN_TEST(test_tcp_recv_ooseq_double_FIN_6, 6)FIN_TEST(test_tcp_recv_ooseq_double_FIN_7, 7)FIN_TEST(test_tcp_recv_ooseq_double_FIN_8, 8)FIN_TEST(test_tcp_recv_ooseq_double_FIN_9, 9)FIN_TEST(test_tcp_recv_ooseq_double_FIN_10, 10)FIN_TEST(test_tcp_recv_ooseq_double_FIN_11, 11)FIN_TEST(test_tcp_recv_ooseq_double_FIN_12, 12)FIN_TEST(test_tcp_recv_ooseq_double_FIN_13, 13)FIN_TEST(test_tcp_recv_ooseq_double_FIN_14, 14)FIN_TEST(test_tcp_recv_ooseq_double_FIN_15, 15)/** Create the suite including all tests for this module */Suite *tcp_oos_suite(void){  TFun tests[] = {    test_tcp_recv_ooseq_FIN_OOSEQ,    test_tcp_recv_ooseq_FIN_INSEQ,    test_tcp_recv_ooseq_overrun_rxwin,    test_tcp_recv_ooseq_max_bytes,    test_tcp_recv_ooseq_max_pbufs,    test_tcp_recv_ooseq_double_FIN_0,    test_tcp_recv_ooseq_double_FIN_1,    test_tcp_recv_ooseq_double_FIN_2,    test_tcp_recv_ooseq_double_FIN_3,    test_tcp_recv_ooseq_double_FIN_4,    test_tcp_recv_ooseq_double_FIN_5,    test_tcp_recv_ooseq_double_FIN_6,    test_tcp_recv_ooseq_double_FIN_7,    test_tcp_recv_ooseq_double_FIN_8,    test_tcp_recv_ooseq_double_FIN_9,    test_tcp_recv_ooseq_double_FIN_10,    test_tcp_recv_ooseq_double_FIN_11,    test_tcp_recv_ooseq_double_FIN_12,    test_tcp_recv_ooseq_double_FIN_13,    test_tcp_recv_ooseq_double_FIN_14,    test_tcp_recv_ooseq_double_FIN_15  };  return create_suite("TCP_OOS", tests, sizeof(tests)/sizeof(TFun), tcp_oos_setup, tcp_oos_teardown);}
 |