mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
473 lines
12 KiB
C
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;
|
|
}
|