2016-02-04 14:37:35 +01:00
|
|
|
/*
|
2017-02-06 18:26:45 +01:00
|
|
|
* Copyright (C) 2015-2017 Simon Brummer
|
2016-02-04 14:37:35 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
2017-02-20 21:42:59 +01:00
|
|
|
|
2016-02-04 14:37:35 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include "net/af.h"
|
|
|
|
#include "random.h"
|
|
|
|
#include "net/gnrc/ipv6.h"
|
|
|
|
#include "net/gnrc/tcp.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* Number of possible parallel connections */
|
|
|
|
#ifndef CONNS
|
2017-02-04 10:19:59 +01:00
|
|
|
#define CONNS (1)
|
2016-02-04 14:37:35 +01:00
|
|
|
#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)
|
|
|
|
{
|
|
|
|
printf("\nStarting Client Threads. TARGET_ADDR=%s, TARGET_PORT=%d, ", TARGET_ADDR, TARGET_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) {
|
2017-02-04 10:19:59 +01:00
|
|
|
thread_create((char *) stacks[i], sizeof(stacks[i]), THREAD_PRIORITY_MAIN, 0, cli_thread,
|
|
|
|
(void *) i, NULL);
|
2016-02-04 14:37:35 +01:00
|
|
|
}
|
|
|
|
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;
|
|
|
|
|
|
|
|
/* Target Peer Address Information */
|
|
|
|
ipv6_addr_t target_addr;
|
|
|
|
uint16_t target_port;
|
|
|
|
|
|
|
|
/* Initialize Target Information */
|
|
|
|
ipv6_addr_from_str(&target_addr, TARGET_ADDR);
|
|
|
|
target_port = TARGET_PORT;
|
|
|
|
|
|
|
|
printf("Client running: TID=%d\n", tid);
|
|
|
|
while (cycles < CYCLES) {
|
|
|
|
/* Initialize tcb struct */
|
|
|
|
gnrc_tcp_tcb_init(&tcb);
|
|
|
|
|
|
|
|
/* Connect to Peer */
|
|
|
|
int ret = gnrc_tcp_open_active(&tcb, AF_INET6, (uint8_t *) &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:
|
2017-02-04 10:19:59 +01:00
|
|
|
printf("TID=%d : gnrc_tcp_open_active() : -ETIMEDOUT : retry after 10sec\n", tid);
|
2016-02-04 14:37:35 +01:00
|
|
|
xtimer_sleep(10);
|
|
|
|
continue;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("TID=%d : gnrc_tcp_open_active() : %d\n", tid, ret);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill Buffer with a test pattern */
|
2017-02-04 10:19:59 +01:00
|
|
|
for (size_t i = 0; i < sizeof(bufs[tid]); ++i){
|
2016-02-04 14:37:35 +01:00
|
|
|
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) {
|
2017-02-04 10:19:59 +01:00
|
|
|
ret = gnrc_tcp_recv(&tcb, (void *) (bufs[tid] + rcvd), sizeof(bufs[tid]) - rcvd,
|
2016-02-04 14:37:35 +01:00
|
|
|
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 */
|
2017-02-04 10:19:59 +01:00
|
|
|
for (size_t i = 0; i < sizeof(bufs[tid]); ++i) {
|
2016-02-04 14:37:35 +01:00
|
|
|
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;
|
2017-02-04 10:19:59 +01:00
|
|
|
if (ret >= 0) {
|
2016-02-04 14:37:35 +01:00
|
|
|
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;
|
|
|
|
}
|