/* * Copyright (C) 2015-2017 Simon Brummer * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ #include #include #include "thread.h" #include "net/af.h" #include "net/gnrc/ipv6.h" #include "net/gnrc/netif.h" #include "net/gnrc/tcp.h" #define ENABLE_DEBUG (0) #include "debug.h" /* Number of possible parallel connections */ #ifndef CONNS #define CONNS (1) #endif /* Amount of data to transmit */ #ifndef NBYTE #define NBYTE (2048) #endif /* Test pattern used by client application */ #ifndef TEST_PATERN_CLI #define TEST_PATERN_CLI (0xF0) #endif /* Test pattern used by server application */ #ifndef TEST_PATERN_SRV #define TEST_PATERN_SRV (0xA7) #endif uint8_t bufs[CONNS][NBYTE]; uint8_t stacks[CONNS][THREAD_STACKSIZE_DEFAULT + THREAD_EXTRA_STACKSIZE_PRINTF]; void *cli_thread(void *arg); int main(void) { gnrc_netif_t *netif; ipv6_addr_t addr; if (!(netif = gnrc_netif_iter(NULL))) { printf("No valid network interface found\n"); return -1; } if (ipv6_addr_from_str(&addr, CLIENT_ADDR) == NULL) { printf("Can't convert given string to IPv6 Address\n"); return -1; } if (gnrc_netif_ipv6_addr_add(netif, &addr, 64, GNRC_NETIF_IPV6_ADDRS_FLAGS_STATE_VALID) < 0) { printf("Can't assign given IPv6 Address\n"); return -1; } printf("\nStarting Client Threads. SERVER_ADDR=%s, SERVER_PORT=%d, ", SERVER_ADDR, SERVER_PORT); printf("CONNS=%d, NBYTE=%d, CYCLES=%d\n\n", CONNS, NBYTE, CYCLES); /* Start connection handling threads */ for (int i = 0; i < CONNS; i += 1) { thread_create((char *) stacks[i], sizeof(stacks[i]), THREAD_PRIORITY_MAIN, 0, cli_thread, (void *) i, NULL); } return 0; } void *cli_thread(void *arg) { /* Test program variables */ int tid = (int) arg; uint32_t cycles = 0; uint32_t cycles_ok = 0; uint32_t failed_payload_verifications = 0; /* Transmission control block */ gnrc_tcp_tcb_t tcb; printf("Client running: TID=%d\n", tid); while (cycles < CYCLES) { /* Copy peer address information. NOTE: This test uses link-local addresses * -> The Device identifier is removed from target_addr in each iteration! */ char target_addr[] = SERVER_ADDR; uint16_t target_port = SERVER_PORT; /* Initialize TCB */ gnrc_tcp_tcb_init(&tcb); /* Connect to peer */ int ret = gnrc_tcp_open_active(&tcb, AF_INET6, target_addr, target_port, 0); switch (ret) { case 0: DEBUG("TID=%d : gnrc_tcp_open_active() : 0 : ok\n", tid); break; case -EISCONN: printf("TID=%d : gnrc_tcp_open_active() : -EISCONN\n", tid); return 0; case -EINVAL: printf("TID=%d : gnrc_tcp_open_active() : -EINVAL\n", tid); return 0; case -EAFNOSUPPORT: printf("TID=%d : gnrc_tcp_open_active() : -EAFNOSUPPORT\n", tid); return 0; case -EADDRINUSE: printf("TID=%d : gnrc_tcp_open_active() : -EADDRINUSE\n", tid); return 0; case -ECONNREFUSED: printf("TID=%d : gnrc_tcp_open_active() : -ECONNREFUSED : retry after 10sec\n", tid); xtimer_sleep(10); continue; case -ENOMEM: printf("TID=%d : gnrc_tcp_open_active() : -ENOMEM\n", tid); return 0; case -ETIMEDOUT: printf("TID=%d : gnrc_tcp_open_active() : -ETIMEDOUT : retry after 10sec\n", tid); xtimer_sleep(10); continue; default: printf("TID=%d : gnrc_tcp_open_active() : %d\n", tid, ret); return 0; } /* Fill buffer with a test pattern */ for (size_t i = 0; i < sizeof(bufs[tid]); ++i){ bufs[tid][i] = TEST_PATERN_CLI; } /* Send data, stop if errors were found */ for (size_t sent = 0; sent < sizeof(bufs[tid]) && ret >= 0; sent += ret) { ret = gnrc_tcp_send(&tcb, bufs[tid] + sent, sizeof(bufs[tid]) - sent, 0); switch (ret) { case -ENOTCONN: printf("TID=%d : gnrc_tcp_send() : -ENOTCONN\n", tid); break; case -ECONNABORTED: printf("TID=%d : gnrc_tcp_send() : -ECONNABORTED\n", tid); break; case -ETIMEDOUT: printf("TID=%d : gnrc_tcp_send() : -ETIMEDOUT\n", tid); break; case -ECONNRESET: printf("TID=%d : gnrc_tcp_send() : -ECONNRESET\n", tid); break; default: if (ret >= 0) { DEBUG("TID=%d : gnrc_tcp_send() : %d Bytes sent\n", tid, ret); } else { printf("TID=%d : gnrc_tcp_send() : %d\n", tid, ret); return 0; } } } /* Receive data, stop if errors were found */ for (size_t rcvd = 0; rcvd < sizeof(bufs[tid]) && ret >= 0; rcvd += ret) { ret = gnrc_tcp_recv(&tcb, (void *) (bufs[tid] + rcvd), sizeof(bufs[tid]) - rcvd, GNRC_TCP_CONNECTION_TIMEOUT_DURATION); switch (ret) { case -ENOTCONN: printf("TID=%d : gnrc_tcp_rcvd() : -ENOTCONN\n", tid); break; case -EAGAIN: printf("TID=%d : gnrc_tcp_rcvd() : -EAGAIN : retry after 10sec\n", tid); ret = 0; xtimer_sleep(10); break; case -ECONNABORTED: printf("TID=%d : gnrc_tcp_rcvd() : -ECONNABORTED\n", tid); break; case -ECONNRESET: printf("TID=%d : gnrc_tcp_rcvd() : -ECONNRESET\n", tid); break; case -ETIMEDOUT: printf("TID=%d : gnrc_tcp_rcvd() : -ETIMEDOUT\n", tid); break; default: if (ret >= 0) { DEBUG("TID=%d : gnrc_tcp_rcvd() : %d Bytes read.\n", tid, ret); } else { printf("TID=%d : gnrc_tcp_rcvd() : %d\n", tid, ret); return 0; } } } /* If there was no error: Check received pattern */ for (size_t i = 0; i < sizeof(bufs[tid]); ++i) { if (bufs[tid][i] != TEST_PATERN_SRV) { printf("TID=%d : Payload verfication failed\n", tid); failed_payload_verifications += 1; break; } } /* Close connection */ gnrc_tcp_close(&tcb); /* Gather data */ cycles += 1; if (ret >= 0) { cycles_ok += 1; } printf("TID=%d : %"PRIi32" test cycles completed. %"PRIi32" ok, %"PRIi32" faulty", tid, cycles, cycles_ok, cycles - cycles_ok); printf(", %"PRIi32" failed payload verifications\n", failed_payload_verifications); } printf("client thread terminating: TID=%d\n", tid); return 0; }