1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/gnrc_tcp/main.c
2021-07-13 20:32:14 +02:00

473 lines
12 KiB
C

/*
* Copyright (C) 2018 Simon Brummer <simon.brummer@posteo.de>
*
* 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 <stdio.h>
#include <string.h>
#include "shell.h"
#include "msg.h"
#include "net/af.h"
#include "net/gnrc/tcp.h"
#define MAIN_QUEUE_SIZE (8)
#define TCB_QUEUE_SIZE (1)
#define BUFFER_SIZE (2049)
static msg_t main_msg_queue[MAIN_QUEUE_SIZE];
static gnrc_tcp_tcb_t tcbs[TCB_QUEUE_SIZE];
static gnrc_tcp_tcb_t *tcb = tcbs;
static gnrc_tcp_tcb_queue_t queue = GNRC_TCP_TCB_QUEUE_INIT;
static char buffer[BUFFER_SIZE];
void dump_args(int argc, char **argv)
{
printf("%s: ", argv[0]);
printf("argc=%d", argc);
for (int i = 0; i < argc; ++i) {
printf(", argv[%d] = %s", i, argv[i]);
}
printf("\n");
}
/* API Export for test script */
int buffer_init_cmd(int argc, char **argv)
{
dump_args(argc, argv);
memset(buffer, '\0', sizeof(buffer));
return 0;
}
int buffer_get_max_size_cmd(int argc, char **argv)
{
dump_args(argc, argv);
printf("%s: returns %d\n", argv[0], BUFFER_SIZE - 1);
return 0;
}
int buffer_write_cmd(int argc, char **argv)
{
dump_args(argc, argv);
size_t offset = atol(argv[1]);
char *src = argv[2];
size_t src_len = strlen(src);
char *dst = buffer + offset;
memcpy(dst, src, src_len);
return 0;
}
int buffer_read_cmd(int argc, char **argv)
{
dump_args(argc, argv);
size_t offset = atol(argv[1]);
size_t size = atol(argv[2]);
/* Calculate Start and End of readout */
char *begin = buffer + offset;
char *end = begin + size;
/* Place temporary endmarker in buffer and print */
char tmp = *end;
*end = '\0';
printf("%s: <begin>%s<end>\n", argv[0], begin);
*end = tmp;
return 0;
}
int gnrc_tcp_ep_from_str_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_ep_from_str(&ep, argv[1]);
switch (err) {
case -EINVAL:
printf("%s: returns -EINVAL\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
if (err == 0) {
char addr_as_str[IPV6_ADDR_MAX_STR_LEN];
switch (ep.family) {
case AF_INET6:
printf("Family: AF_INET6\n");
ipv6_addr_to_str(addr_as_str, (ipv6_addr_t *) ep.addr.ipv6,
sizeof(addr_as_str));
printf("Addr: %s\n", addr_as_str);
break;
case AF_INET:
printf("Family: AF_INET\n");
break;
default:
printf("Family: %d\n", ep.family);
}
printf("Port: %d\n", ep.port);
printf("Netif: %d\n", ep.netif);
}
return err;
}
int gnrc_tcp_tcb_init_cmd(int argc, char **argv)
{
dump_args(argc, argv);
// Initialize all given TCBs
for (int i = 0; i < TCB_QUEUE_SIZE; ++i)
{
gnrc_tcp_tcb_init(&(tcbs[i]));
}
printf("%s: returns 0\n", argv[0]);
return 0;
}
int gnrc_tcp_open_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t remote;
gnrc_tcp_ep_from_str(&remote, argv[1]);
uint16_t local_port = atol(argv[2]);
int err = gnrc_tcp_open(tcb, &remote, local_port);
switch (err) {
case -EAFNOSUPPORT:
printf("%s: returns -EAFNOSUPPORT\n", argv[0]);
break;
case -EINVAL:
printf("%s: returns -EINVAL\n", argv[0]);
break;
case -EISCONN:
printf("%s: returns -EISCONN\n", argv[0]);
break;
case -ENOMEM:
printf("%s: returns -ENOMEM\n", argv[0]);
break;
case -EADDRINUSE:
printf("%s: returns -EADDRINUSE\n", argv[0]);
break;
case -ETIMEDOUT:
printf("%s: returns -ETIMEOUT\n", argv[0]);
break;
case -ECONNREFUSED:
printf("%s: returns -ECONNREFUSED\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return err;
}
int gnrc_tcp_listen_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t local;
gnrc_tcp_ep_from_str(&local, argv[1]);
int err = gnrc_tcp_listen(&queue, tcbs, ARRAY_SIZE(tcbs), &local);
switch (err) {
case -EAFNOSUPPORT:
printf("%s: returns -EAFNOSUPPORT\n", argv[0]);
break;
case -EINVAL:
printf("%s: returns -EINVAL\n", argv[0]);
break;
case -EISCONN:
printf("%s: returns -EISCONN\n", argv[0]);
break;
case -ENOMEM:
printf("%s: returns -ENOMEM\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return err;
}
int gnrc_tcp_accept_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_tcb_t *tmp = NULL;
int timeout = atol(argv[1]);
int err = gnrc_tcp_accept(&queue, &tmp, timeout);
switch (err) {
case -EINVAL:
printf("%s: returns -EINVAL\n", argv[0]);
break;
case -EAGAIN:
printf("%s: returns -EAGAIN\n", argv[0]);
break;
case -ENOMEM:
printf("%s: returns -ENOMEM\n", argv[0]);
break;
case -ETIMEDOUT:
printf("%s: returns -ETIMEDOUT\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
if (tmp) {
tcb = tmp;
}
return err;
}
int gnrc_tcp_send_cmd(int argc, char **argv)
{
dump_args(argc, argv);
int timeout = atol(argv[1]);
size_t to_send = strlen(buffer);
size_t sent = 0;
while (sent < to_send) {
int ret = gnrc_tcp_send(tcb, buffer + sent, to_send - sent, timeout);
switch (ret) {
case -ENOTCONN:
printf("%s: returns -ENOTCONN\n", argv[0]);
return ret;
case -ECONNRESET:
printf("%s: returns -ECONNRESET\n", argv[0]);
return ret;
case -ECONNABORTED:
printf("%s: returns -ECONNABORTED\n", argv[0]);
return ret;
case -ETIMEDOUT:
printf("%s: returns -ETIMEDOUT\n", argv[0]);
return ret;
}
sent += ret;
}
printf("%s: sent %u\n", argv[0], (unsigned)sent);
return sent;
}
int gnrc_tcp_recv_cmd(int argc, char **argv)
{
dump_args(argc, argv);
int timeout = atol(argv[1]);
size_t to_receive = atol(argv[2]);
size_t rcvd = 0;
while (rcvd < to_receive) {
int ret = gnrc_tcp_recv(tcb, buffer + rcvd, to_receive - rcvd,
timeout);
switch (ret) {
case 0:
printf("%s: returns 0\n", argv[0]);
return ret;
case -EAGAIN:
printf("%s: returns -EAGAIN\n", argv[0]);
continue;
case -ETIMEDOUT:
printf("%s: returns -ETIMEDOUT\n", argv[0]);
continue;
case -ENOTCONN:
printf("%s: returns -ENOTCONN\n", argv[0]);
return ret;
case -ECONNRESET:
printf("%s: returns -ECONNRESET\n", argv[0]);
return ret;
case -ECONNABORTED:
printf("%s: returns -ECONNABORTED\n", argv[0]);
return ret;
}
rcvd += ret;
}
printf("%s: received %u\n", argv[0], (unsigned)rcvd);
return 0;
}
int gnrc_tcp_close_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_close(tcb);
printf("%s: returns\n", argv[0]);
return 0;
}
int gnrc_tcp_abort_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_abort(tcb);
printf("%s: returns\n", argv[0]);
return 0;
}
int gnrc_tcp_stop_listen_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_stop_listen(&queue);
printf("%s: returns\n", argv[0]);
return 0;
}
int gnrc_tcp_get_local_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_get_local(tcb, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -EADDRNOTAVAIL:
printf("%s: returns -EADDRNOTAVAIL\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
int gnrc_tcp_get_remote_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_get_remote(tcb, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -ENOTCONN:
printf("%s: returns -ENOTCONN\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
int gnrc_tcp_queue_get_local_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_queue_get_local(&queue, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -EADDRNOTAVAIL:
printf("%s: returns -EADDRNOTAVAIL\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
/* Exporting GNRC TCP Api to for shell usage */
static const shell_command_t shell_commands[] = {
{ "gnrc_tcp_ep_from_str", "Build endpoint from string",
gnrc_tcp_ep_from_str_cmd },
{ "gnrc_tcp_tcb_init", "gnrc_tcp: init tcb",
gnrc_tcp_tcb_init_cmd },
{ "gnrc_tcp_open", "gnrc_tcp: open connection",
gnrc_tcp_open_cmd },
{ "gnrc_tcp_listen", "gnrc_tcp: listen for connection",
gnrc_tcp_listen_cmd },
{ "gnrc_tcp_accept", "gnrc_tcp: accept connection",
gnrc_tcp_accept_cmd },
{ "gnrc_tcp_send", "gnrc_tcp: send data to connected peer",
gnrc_tcp_send_cmd },
{ "gnrc_tcp_recv", "gnrc_tcp: recv data from connected peer",
gnrc_tcp_recv_cmd },
{ "gnrc_tcp_close", "gnrc_tcp: close connection gracefully",
gnrc_tcp_close_cmd },
{ "gnrc_tcp_abort", "gnrc_tcp: close connection forcefully",
gnrc_tcp_abort_cmd },
{ "gnrc_tcp_stop_listen", "gnrc_tcp: stop listening",
gnrc_tcp_stop_listen_cmd },
{ "gnrc_tcp_get_local", "gnrc_tcp: get local",
gnrc_tcp_get_local_cmd },
{ "gnrc_tcp_get_remote", "gnrc_tcp: get remote",
gnrc_tcp_get_remote_cmd },
{ "gnrc_tcp_queue_get_local", "gnrc_tcp: get queue local",
gnrc_tcp_queue_get_local_cmd },
{ "buffer_init", "init internal buffer",
buffer_init_cmd },
{ "buffer_get_max_size", "get max size of internal buffer",
buffer_get_max_size_cmd },
{ "buffer_write", "write data into internal buffer",
buffer_write_cmd },
{ "buffer_read", "read data from internal buffer",
buffer_read_cmd },
{ NULL, NULL, NULL }
};
int main(void)
{
/* we need a message queue for the thread running the shell in order to
* receive potentially fast incoming networking packets */
msg_init_queue(main_msg_queue, MAIN_QUEUE_SIZE);
printf("RIOT GNRC_TCP test application\n");
/* start shell */
char line_buf[SHELL_DEFAULT_BUFSIZE];
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
/* should be never reached */
return 0;
}