1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 08:32:43 +01:00
RIOT/sys/net/gnrc/transport_layer/tcp/gnrc_tcp.c

676 lines
22 KiB
C
Raw Normal View History

2016-02-04 14:37:35 +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.
*/
/**
* @ingroup net_gnrc
* @{
*
* @file
* @brief GNRC TCP API implementation
2016-02-04 14:37:35 +01:00
*
* @author Simon Brummer <simon.brummer@posteo.de>
2016-02-04 14:37:35 +01:00
* @}
*/
#include <errno.h>
2018-07-10 15:25:02 +02:00
#include <string.h>
2018-07-21 10:13:00 +02:00
#include <utlist.h>
2016-02-04 14:37:35 +01:00
#include "net/af.h"
2018-07-10 15:25:02 +02:00
#include "net/gnrc.h"
2016-02-04 14:37:35 +01:00
#include "net/gnrc/tcp.h"
#include "internal/common.h"
2016-02-04 14:37:35 +01:00
#include "internal/fsm.h"
#include "internal/pkt.h"
#include "internal/option.h"
#include "internal/eventloop.h"
#include "internal/rcvbuf.h"
#ifdef MODULE_GNRC_IPV6
#include "net/gnrc/ipv6.h"
#endif
#define ENABLE_DEBUG (0)
#include "debug.h"
/**
* @brief Allocate memory for GNRC TCP thread stack.
2016-02-04 14:37:35 +01:00
*/
#if ENABLE_DEBUG
static char _stack[TCP_EVENTLOOP_STACK_SIZE + THREAD_EXTRA_STACKSIZE_PRINTF];
2016-02-04 14:37:35 +01:00
#else
static char _stack[TCP_EVENTLOOP_STACK_SIZE];
2016-02-04 14:37:35 +01:00
#endif
/**
* @brief TCPs eventloop pid, declared externally.
2016-02-04 14:37:35 +01:00
*/
kernel_pid_t gnrc_tcp_pid = KERNEL_PID_UNDEF;
2016-02-04 14:37:35 +01:00
/**
* @brief Head of liked TCB list.
2016-02-04 14:37:35 +01:00
*/
gnrc_tcp_tcb_t *_list_tcb_head;
2016-02-04 14:37:35 +01:00
/**
* @brief Mutex for TCB list synchronization.
2016-02-04 14:37:35 +01:00
*/
mutex_t _list_tcb_lock;
2016-02-04 14:37:35 +01:00
/**
* @brief Helper struct, holding all argument data for_cb_mbox_put_msg.
*/
typedef struct _cb_arg {
uint32_t msg_type; /**< Message Type to Put into mbox behind mbox_ptr */
mbox_t *mbox_ptr; /**< Pointer to mbox */
} cb_arg_t;
/**
* @brief Callback for xtimer, puts a message in a mbox.
*
* @param[in] arg Ptr to cb_arg_t. Must not be NULL or anything else.
*/
static void _cb_mbox_put_msg(void *arg)
{
msg_t msg;
msg.type = ((cb_arg_t *) arg)->msg_type;
mbox_try_put(((cb_arg_t *) arg)->mbox_ptr, &msg);
}
/**
* @brief Setup timer with a callback function.
*
* @param[in/out] timer Ptr to timer, which should be set.
* @param[in] duration Duration after @p timer expires.
* @param[in] cb Function to be called after @p duration.
* @param[in] arg Arguments for @p cb.
*/
static void _setup_timeout(xtimer_t *timer, const uint32_t duration, const xtimer_callback_t cb,
cb_arg_t *arg)
{
timer->callback = cb;
timer->arg = arg;
xtimer_set(timer, duration);
}
/**
* @brief Establishes a new TCP connection
2016-02-04 14:37:35 +01:00
*
* @param[in,out] tcb TCB holding the connection information.
* @param[in] target_addr Target address to connect to, if this is a active connection.
* @param[in] target_port Target port to connect to, if this is a active connection.
* @param[in] local_addr Local address to bind on, if this is a passive connection.
* @param[in] local_port Local port to bind on, if this is a passive connection.
* @param[in] passive Flag to indicate if this is a active or passive open.
2016-02-04 14:37:35 +01:00
*
* @returns Zero on success.
* -EISCONN if TCB is already connected.
* -ENOMEM if the receive buffer for the TCB could not be allocated.
* -EADDRINUSE if @p local_port is already in use.
* -ETIMEDOUT if the connection opening timed out.
2019-10-23 21:16:23 +02:00
* -ECONNREFUSED if the connection was reset by the peer.
2016-02-04 14:37:35 +01:00
*/
static int _gnrc_tcp_open(gnrc_tcp_tcb_t *tcb, char *target_addr, uint16_t target_port,
const char *local_addr, uint16_t local_port, uint8_t passive)
2016-02-04 14:37:35 +01:00
{
msg_t msg;
xtimer_t connection_timeout;
cb_arg_t connection_timeout_arg = {MSG_TYPE_CONNECTION_TIMEOUT, &(tcb->mbox)};
int8_t ret = 0;
2016-02-04 14:37:35 +01:00
/* Lock the TCB for this function call */
2016-02-04 14:37:35 +01:00
mutex_lock(&(tcb->function_lock));
/* Connection is already connected: Return -EISCONN */
if (tcb->state != FSM_STATE_CLOSED) {
2016-02-04 14:37:35 +01:00
mutex_unlock(&(tcb->function_lock));
return -EISCONN;
}
/* Mark TCB as waiting for incoming messages */
tcb->status |= STATUS_WAIT_FOR_MSG;
/* 'Flush' mbox */
while (mbox_try_get(&(tcb->mbox), &msg) != 0) {
}
2016-02-04 14:37:35 +01:00
/* Setup passive connection */
if (passive) {
/* Mark connection as passive opend */
tcb->status |= STATUS_PASSIVE;
2016-02-04 14:37:35 +01:00
if (local_addr == NULL) {
tcb->status |= STATUS_ALLOW_ANY_ADDR;
2016-02-04 14:37:35 +01:00
}
#ifdef MODULE_GNRC_IPV6
/* If local address is specified: Copy it into TCB */
else if (tcb->address_family == AF_INET6) {
if (ipv6_addr_from_str((ipv6_addr_t *) tcb->local_addr, local_addr) == NULL) {
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : Invalid peer addr\n");
return -EINVAL;
}
2016-02-04 14:37:35 +01:00
}
2018-07-10 15:25:02 +02:00
#else
/* Suppress Compiler Warnings */
2018-07-10 15:25:02 +02:00
(void) target_addr;
#endif
/* Set port number to listen on */
2016-02-04 14:37:35 +01:00
tcb->local_port = local_port;
}
/* Setup active connection */
else {
/* Parse target address and port number into TCB */
2016-02-04 14:37:35 +01:00
#ifdef MODULE_GNRC_IPV6
if ((target_addr != NULL) && (tcb->address_family == AF_INET6)) {
/* Extract interface (optional) specifier from target address */
char *ll_iface = ipv6_addr_split_iface(target_addr);
if (ipv6_addr_from_str((ipv6_addr_t *) tcb->peer_addr, target_addr) == NULL) {
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : Invalid peer addr\n");
return -EINVAL;
}
/* In case the given address is link-local: Memorize the interface Id if existing. */
if ((ll_iface) && ipv6_addr_is_link_local((ipv6_addr_t *) tcb->peer_addr)) {
tcb->ll_iface = atoi(ll_iface);
}
2016-02-04 14:37:35 +01:00
}
#endif
/* Assign port numbers, verification happens in fsm */
2016-02-04 14:37:35 +01:00
tcb->local_port = local_port;
tcb->peer_port = target_port;
/* Setup connection timeout: Put timeout message in TCBs mbox on expiration */
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
2016-02-04 14:37:35 +01:00
}
/* Call FSM with event: CALL_OPEN */
ret = _fsm(tcb, FSM_EVENT_CALL_OPEN, NULL, NULL, 0);
2016-02-04 14:37:35 +01:00
if (ret == -ENOMEM) {
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : Out of receive buffers.\n");
}
else if(ret == -EADDRINUSE) {
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : local_port is already in use.\n");
2016-02-04 14:37:35 +01:00
}
/* Wait until a connection was established or closed */
while (ret >= 0 && tcb->state != FSM_STATE_CLOSED && tcb->state != FSM_STATE_ESTABLISHED &&
tcb->state != FSM_STATE_CLOSE_WAIT) {
mbox_get(&(tcb->mbox), &msg);
2016-02-04 14:37:35 +01:00
switch (msg.type) {
2019-02-04 20:57:30 +01:00
case MSG_TYPE_NOTIFY_USER:
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : MSG_TYPE_NOTIFY_USER\n");
/* Setup a timeout to be able to revert back to LISTEN state, in case the
* send SYN+ACK we received upon entering SYN_RCVD is never acknowledged
* by the peer. */
if ((tcb->state == FSM_STATE_SYN_RCVD) && (tcb->status & STATUS_PASSIVE)) {
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
}
break;
2016-02-04 14:37:35 +01:00
case MSG_TYPE_CONNECTION_TIMEOUT:
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : CONNECTION_TIMEOUT\n");
2019-02-04 20:57:30 +01:00
/* The connection establishment attempt timed out:
* 1) Active connections return -ETIMEOUT.
* 2) Passive connections stop the ongoing retransmissions and repeat the
* open call to wait for the next connection attempt. */
if (tcb->status & STATUS_PASSIVE) {
_fsm(tcb, FSM_EVENT_CLEAR_RETRANSMIT, NULL, NULL, 0);
_fsm(tcb, FSM_EVENT_CALL_OPEN, NULL, NULL, 0);
}
else {
_fsm(tcb, FSM_EVENT_TIMEOUT_CONNECTION, NULL, NULL, 0);
ret = -ETIMEDOUT;
}
2016-02-04 14:37:35 +01:00
break;
default:
DEBUG("gnrc_tcp.c : _gnrc_tcp_open() : other message type\n");
}
}
/* Cleanup */
xtimer_remove(&connection_timeout);
if (tcb->state == FSM_STATE_CLOSED && ret == 0) {
2016-02-04 14:37:35 +01:00
ret = -ECONNREFUSED;
}
tcb->status &= ~STATUS_WAIT_FOR_MSG;
2016-02-04 14:37:35 +01:00
mutex_unlock(&(tcb->function_lock));
return ret;
}
/* External GNRC TCP API */
2016-02-04 14:37:35 +01:00
int gnrc_tcp_init(void)
{
/* Guard: Check if thread is already running */
if (gnrc_tcp_pid != KERNEL_PID_UNDEF) {
2016-02-04 14:37:35 +01:00
return -1;
}
/* Initialize mutex for TCB list synchronization */
mutex_init(&(_list_tcb_lock));
2016-02-04 14:37:35 +01:00
/* Initialize TCB list */
_list_tcb_head = NULL;
2016-02-04 14:37:35 +01:00
_rcvbuf_init();
/* Start TCP processing thread */
return thread_create(_stack, sizeof(_stack), TCP_EVENTLOOP_PRIO,
THREAD_CREATE_STACKTEST, _event_loop, NULL,
"gnrc_tcp");
2016-02-04 14:37:35 +01:00
}
void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb)
2016-02-04 14:37:35 +01:00
{
memset(tcb, 0, sizeof(gnrc_tcp_tcb_t));
2016-02-04 14:37:35 +01:00
#ifdef MODULE_GNRC_IPV6
tcb->address_family = AF_INET6;
#else
DEBUG("gnrc_tcp.c : gnrc_tcp_tcb_init() : Address unspec, add netlayer module to makefile\n");
#endif
tcb->rtt_var = RTO_UNINITIALIZED;
tcb->srtt = RTO_UNINITIALIZED;
tcb->rto = RTO_UNINITIALIZED;
mbox_init(&(tcb->mbox), tcb->mbox_raw, GNRC_TCP_TCB_MBOX_SIZE);
2016-02-04 14:37:35 +01:00
mutex_init(&(tcb->fsm_lock));
mutex_init(&(tcb->function_lock));
}
int gnrc_tcp_open_active(gnrc_tcp_tcb_t *tcb, uint8_t address_family,
char *target_addr, uint16_t target_port,
uint16_t local_port)
2016-02-04 14:37:35 +01:00
{
assert(tcb != NULL);
assert(target_addr != NULL);
assert(target_port != PORT_UNSPEC);
2016-02-04 14:37:35 +01:00
/* Check if AF-Family of target_addr is supported */
2016-02-04 14:37:35 +01:00
#ifdef MODULE_GNRC_IPV6
if (address_family != AF_INET6) {
return -EAFNOSUPPORT;
2016-02-04 14:37:35 +01:00
}
#else
return -EAFNOSUPPORT;
#endif
/* Check if AF-Family for target address matches internally used AF-Family */
2016-02-04 14:37:35 +01:00
if (tcb->address_family != address_family) {
return -EINVAL;
}
/* Proceed with connection opening */
2016-02-04 14:37:35 +01:00
return _gnrc_tcp_open(tcb, target_addr, target_port, NULL, local_port, 0);
}
int gnrc_tcp_open_passive(gnrc_tcp_tcb_t *tcb, uint8_t address_family,
const char *local_addr, uint16_t local_port)
2016-02-04 14:37:35 +01:00
{
assert(tcb != NULL);
2017-05-12 09:35:02 +02:00
assert(local_port != PORT_UNSPEC);
2016-02-04 14:37:35 +01:00
/* Check AF-Family support if local address was supplied */
if (local_addr != NULL) {
#ifdef MODULE_GNRC_IPV6
if (address_family != AF_INET6) {
return -EAFNOSUPPORT;
2016-02-04 14:37:35 +01:00
}
#else
return -EAFNOSUPPORT;
#endif
2016-02-04 14:37:35 +01:00
/* Check if AF-Family matches internally used AF-Family */
if (tcb->address_family != address_family) {
return -EINVAL;
}
}
/* Proceed with connection opening */
2016-02-04 14:37:35 +01:00
return _gnrc_tcp_open(tcb, NULL, 0, local_addr, local_port, 1);
}
ssize_t gnrc_tcp_send(gnrc_tcp_tcb_t *tcb, const void *data, const size_t len,
const uint32_t timeout_duration_us)
{
assert(tcb != NULL);
assert(data != NULL);
msg_t msg;
xtimer_t connection_timeout;
cb_arg_t connection_timeout_arg = {MSG_TYPE_CONNECTION_TIMEOUT, &(tcb->mbox)};
xtimer_t user_timeout;
cb_arg_t user_timeout_arg = {MSG_TYPE_USER_SPEC_TIMEOUT, &(tcb->mbox)};
xtimer_t probe_timeout;
cb_arg_t probe_timeout_arg = {MSG_TYPE_PROBE_TIMEOUT, &(tcb->mbox)};
uint32_t probe_timeout_duration_us = 0;
ssize_t ret = 0;
bool probing_mode = false;
2016-02-04 14:37:35 +01:00
/* Lock the TCB for this function call */
2016-02-04 14:37:35 +01:00
mutex_lock(&(tcb->function_lock));
/* Check if connection is in a valid state */
if (tcb->state != FSM_STATE_ESTABLISHED && tcb->state != FSM_STATE_CLOSE_WAIT) {
mutex_unlock(&(tcb->function_lock));
return -ENOTCONN;
2016-02-04 14:37:35 +01:00
}
/* Mark TCB as waiting for incoming messages */
tcb->status |= STATUS_WAIT_FOR_MSG;
/* 'Flush' mbox */
while (mbox_try_get(&(tcb->mbox), &msg) != 0) {
}
2016-02-04 14:37:35 +01:00
/* Setup connection timeout: Put timeout message in tcb's mbox on expiration */
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
2016-02-04 14:37:35 +01:00
/* Setup user specified timeout if timeout_us is greater than zero */
2016-02-04 14:37:35 +01:00
if (timeout_duration_us > 0) {
_setup_timeout(&user_timeout, timeout_duration_us, _cb_mbox_put_msg, &user_timeout_arg);
2016-02-04 14:37:35 +01:00
}
/* Loop until something was sent and acked */
while (ret == 0 || tcb->pkt_retransmit != NULL) {
/* Check if the connections state is closed. If so, a reset was received */
if (tcb->state == FSM_STATE_CLOSED) {
ret = -ECONNRESET;
break;
2016-02-04 14:37:35 +01:00
}
/* If the send window is closed: Setup Probing */
if (tcb->snd_wnd <= 0) {
/* If this is the first probe: Setup probing duration */
if (!probing_mode) {
probing_mode = true;
2016-02-04 14:37:35 +01:00
probe_timeout_duration_us = tcb->rto;
}
/* Setup probe timeout */
_setup_timeout(&probe_timeout, probe_timeout_duration_us, _cb_mbox_put_msg,
&probe_timeout_arg);
2016-02-04 14:37:35 +01:00
}
/* Try to send data in case there nothing has been sent and we are not probing */
if (ret == 0 && !probing_mode) {
ret = _fsm(tcb, FSM_EVENT_CALL_SEND, NULL, (void *) data, len);
2016-02-04 14:37:35 +01:00
}
/* Wait for responses */
mbox_get(&(tcb->mbox), &msg);
2016-02-04 14:37:35 +01:00
switch (msg.type) {
case MSG_TYPE_CONNECTION_TIMEOUT:
DEBUG("gnrc_tcp.c : gnrc_tcp_send() : CONNECTION_TIMEOUT\n");
_fsm(tcb, FSM_EVENT_TIMEOUT_CONNECTION, NULL, NULL, 0);
2016-02-04 14:37:35 +01:00
ret = -ECONNABORTED;
break;
case MSG_TYPE_USER_SPEC_TIMEOUT:
DEBUG("gnrc_tcp.c : gnrc_tcp_send() : USER_SPEC_TIMEOUT\n");
_fsm(tcb, FSM_EVENT_CLEAR_RETRANSMIT, NULL, NULL, 0);
2016-02-04 14:37:35 +01:00
ret = -ETIMEDOUT;
break;
case MSG_TYPE_PROBE_TIMEOUT:
DEBUG("gnrc_tcp.c : gnrc_tcp_send() : PROBE_TIMEOUT\n");
/* Send probe */
_fsm(tcb, FSM_EVENT_SEND_PROBE, NULL, NULL, 0);
2016-02-04 14:37:35 +01:00
probe_timeout_duration_us += probe_timeout_duration_us;
/* Boundary check for time interval between probes */
2016-02-04 14:37:35 +01:00
if (probe_timeout_duration_us < GNRC_TCP_PROBE_LOWER_BOUND) {
probe_timeout_duration_us = GNRC_TCP_PROBE_LOWER_BOUND;
}
else if (probe_timeout_duration_us > GNRC_TCP_PROBE_UPPER_BOUND) {
probe_timeout_duration_us = GNRC_TCP_PROBE_UPPER_BOUND;
}
break;
case MSG_TYPE_NOTIFY_USER:
DEBUG("gnrc_tcp.c : gnrc_tcp_send() : NOTIFY_USER\n");
/* Connection is alive: Reset Connection Timeout */
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
2016-02-04 14:37:35 +01:00
/* If the window re-opened and we are probing: Stop it */
if (tcb->snd_wnd > 0 && probing_mode) {
probing_mode = false;
xtimer_remove(&probe_timeout);
2016-02-04 14:37:35 +01:00
}
break;
default:
DEBUG("gnrc_tcp.c : gnrc_tcp_send() : other message type\n");
}
}
/* Cleanup */
xtimer_remove(&probe_timeout);
xtimer_remove(&connection_timeout);
xtimer_remove(&user_timeout);
tcb->status &= ~STATUS_WAIT_FOR_MSG;
2016-02-04 14:37:35 +01:00
mutex_unlock(&(tcb->function_lock));
return ret;
}
ssize_t gnrc_tcp_recv(gnrc_tcp_tcb_t *tcb, void *data, const size_t max_len,
const uint32_t timeout_duration_us)
{
assert(tcb != NULL);
assert(data != NULL);
msg_t msg;
xtimer_t connection_timeout;
cb_arg_t connection_timeout_arg = {MSG_TYPE_CONNECTION_TIMEOUT, &(tcb->mbox)};
xtimer_t user_timeout;
cb_arg_t user_timeout_arg = {MSG_TYPE_USER_SPEC_TIMEOUT, &(tcb->mbox)};
ssize_t ret = 0;
2016-02-04 14:37:35 +01:00
/* Lock the TCB for this function call */
2016-02-04 14:37:35 +01:00
mutex_lock(&(tcb->function_lock));
/* Check if connection is in a valid state */
if (tcb->state != FSM_STATE_ESTABLISHED && tcb->state != FSM_STATE_FIN_WAIT_1 &&
tcb->state != FSM_STATE_FIN_WAIT_2 && tcb->state != FSM_STATE_CLOSE_WAIT) {
mutex_unlock(&(tcb->function_lock));
return -ENOTCONN;
2016-02-04 14:37:35 +01:00
}
/* If FIN was received (CLOSE_WAIT), no further data can be received. */
/* Copy received data into given buffer and return number of bytes. Can be zero. */
if (tcb->state == FSM_STATE_CLOSE_WAIT) {
ret = _fsm(tcb, FSM_EVENT_CALL_RECV, NULL, data, max_len);
mutex_unlock(&(tcb->function_lock));
return ret;
}
2016-02-04 14:37:35 +01:00
/* If this call is non-blocking (timeout_duration_us == 0): Try to read data and return */
if (timeout_duration_us == 0) {
ret = _fsm(tcb, FSM_EVENT_CALL_RECV, NULL, data, max_len);
if (ret == 0) {
2016-02-04 14:37:35 +01:00
ret = -EAGAIN;
}
mutex_unlock(&(tcb->function_lock));
return ret;
}
/* Mark TCB as waiting for incoming messages */
tcb->status |= STATUS_WAIT_FOR_MSG;
/* 'Flush' mbox */
while (mbox_try_get(&(tcb->mbox), &msg) != 0) {
}
2016-02-04 14:37:35 +01:00
/* Setup connection timeout: Put timeout message in tcb's mbox on expiration */
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
2016-02-04 14:37:35 +01:00
/* Setup user specified timeout */
_setup_timeout(&user_timeout, timeout_duration_us, _cb_mbox_put_msg, &user_timeout_arg);
2016-02-04 14:37:35 +01:00
/* Processing loop */
2016-02-04 14:37:35 +01:00
while (ret == 0) {
/* Check if the connections state is closed. If so, a reset was received */
if (tcb->state == FSM_STATE_CLOSED) {
ret = -ECONNRESET;
break;
2016-02-04 14:37:35 +01:00
}
/* Try to read available data */
ret = _fsm(tcb, FSM_EVENT_CALL_RECV, NULL, data, max_len);
2016-02-04 14:37:35 +01:00
/* If FIN was received (CLOSE_WAIT), no further data can be received. Leave event loop */
if (tcb->state == FSM_STATE_CLOSE_WAIT) {
break;
}
2016-02-04 14:37:35 +01:00
/* If there was no data: Wait for next packet or until the timeout fires */
if (ret <= 0) {
mbox_get(&(tcb->mbox), &msg);
2016-02-04 14:37:35 +01:00
switch (msg.type) {
case MSG_TYPE_CONNECTION_TIMEOUT:
DEBUG("gnrc_tcp.c : gnrc_tcp_recv() : CONNECTION_TIMEOUT\n");
_fsm(tcb, FSM_EVENT_TIMEOUT_CONNECTION, NULL, NULL, 0);
ret = -ECONNABORTED;
break;
case MSG_TYPE_USER_SPEC_TIMEOUT:
2019-10-03 15:54:36 +02:00
DEBUG("gnrc_tcp.c : gnrc_tcp_recv() : USER_SPEC_TIMEOUT\n");
_fsm(tcb, FSM_EVENT_CLEAR_RETRANSMIT, NULL, NULL, 0);
ret = -ETIMEDOUT;
break;
case MSG_TYPE_NOTIFY_USER:
DEBUG("gnrc_tcp.c : gnrc_tcp_recv() : NOTIFY_USER\n");
break;
default:
DEBUG("gnrc_tcp.c : gnrc_tcp_recv() : other message type\n");
2016-02-04 14:37:35 +01:00
}
}
}
/* Cleanup */
xtimer_remove(&connection_timeout);
xtimer_remove(&user_timeout);
tcb->status &= ~STATUS_WAIT_FOR_MSG;
2016-02-04 14:37:35 +01:00
mutex_unlock(&(tcb->function_lock));
return ret;
}
2017-05-03 14:37:51 +02:00
void gnrc_tcp_close(gnrc_tcp_tcb_t *tcb)
2016-02-04 14:37:35 +01:00
{
assert(tcb != NULL);
msg_t msg;
xtimer_t connection_timeout;
cb_arg_t connection_timeout_arg = {MSG_TYPE_CONNECTION_TIMEOUT, &(tcb->mbox)};
2016-02-04 14:37:35 +01:00
/* Lock the TCB for this function call */
2016-02-04 14:37:35 +01:00
mutex_lock(&(tcb->function_lock));
/* Return if connection is closed */
if (tcb->state == FSM_STATE_CLOSED) {
mutex_unlock(&(tcb->function_lock));
2017-05-17 18:39:38 +02:00
return;
}
2016-02-04 14:37:35 +01:00
/* Mark TCB as waiting for incoming messages */
tcb->status |= STATUS_WAIT_FOR_MSG;
2016-02-04 14:37:35 +01:00
/* 'Flush' mbox */
while (mbox_try_get(&(tcb->mbox), &msg) != 0) {
}
2016-02-04 14:37:35 +01:00
/* Setup connection timeout: Put timeout message in tcb's mbox on expiration */
_setup_timeout(&connection_timeout, GNRC_TCP_CONNECTION_TIMEOUT_DURATION,
_cb_mbox_put_msg, &connection_timeout_arg);
2016-02-04 14:37:35 +01:00
/* Start connection teardown sequence */
_fsm(tcb, FSM_EVENT_CALL_CLOSE, NULL, NULL, 0);
2016-02-04 14:37:35 +01:00
/* Loop until the connection has been closed */
while (tcb->state != FSM_STATE_CLOSED) {
mbox_get(&(tcb->mbox), &msg);
switch (msg.type) {
case MSG_TYPE_CONNECTION_TIMEOUT:
DEBUG("gnrc_tcp.c : gnrc_tcp_close() : CONNECTION_TIMEOUT\n");
_fsm(tcb, FSM_EVENT_TIMEOUT_CONNECTION, NULL, NULL, 0);
break;
case MSG_TYPE_NOTIFY_USER:
DEBUG("gnrc_tcp.c : gnrc_tcp_close() : NOTIFY_USER\n");
break;
default:
DEBUG("gnrc_tcp.c : gnrc_tcp_close() : other message type\n");
2016-02-04 14:37:35 +01:00
}
}
/* Cleanup */
xtimer_remove(&connection_timeout);
tcb->status &= ~STATUS_WAIT_FOR_MSG;
2016-02-04 14:37:35 +01:00
mutex_unlock(&(tcb->function_lock));
2017-05-03 14:37:51 +02:00
}
void gnrc_tcp_abort(gnrc_tcp_tcb_t *tcb)
{
assert(tcb != NULL);
/* Lock the TCB for this function call */
mutex_lock(&(tcb->function_lock));
if (tcb->state != FSM_STATE_CLOSED) {
/* Call FSM ABORT event */
_fsm(tcb, FSM_EVENT_CALL_ABORT, NULL, NULL, 0);
}
mutex_unlock(&(tcb->function_lock));
2016-02-04 14:37:35 +01:00
}
int gnrc_tcp_calc_csum(const gnrc_pktsnip_t *hdr, const gnrc_pktsnip_t *pseudo_hdr)
{
uint16_t csum;
if ((hdr == NULL) || (pseudo_hdr == NULL)) {
return -EFAULT;
}
if (hdr->type != GNRC_NETTYPE_TCP) {
return -EBADMSG;
}
csum = _pkt_calc_csum(hdr, pseudo_hdr, hdr->next);
if (csum == 0) {
return -ENOENT;
}
((tcp_hdr_t *)hdr->data)->checksum = byteorder_htons(csum);
return 0;
}
gnrc_pktsnip_t *gnrc_tcp_hdr_build(gnrc_pktsnip_t *payload, uint16_t src, uint16_t dst)
{
gnrc_pktsnip_t *res;
tcp_hdr_t *hdr;
/* Allocate header */
2016-02-04 14:37:35 +01:00
res = gnrc_pktbuf_add(payload, NULL, sizeof(tcp_hdr_t), GNRC_NETTYPE_TCP);
if (res == NULL) {
2019-10-03 15:54:36 +02:00
DEBUG("gnrc_tcp.c : gnrc_tcp_hdr_build() : No space left in packet buffer\n");
2016-02-04 14:37:35 +01:00
return NULL;
}
hdr = (tcp_hdr_t *) res->data;
/* Clear Header */
memset(hdr, 0, sizeof(tcp_hdr_t));
/* Initialize header with sane defaults */
2016-02-04 14:37:35 +01:00
hdr->src_port = byteorder_htons(src);
hdr->dst_port = byteorder_htons(dst);
hdr->checksum = byteorder_htons(0);
2017-01-31 19:22:43 +01:00
hdr->off_ctl = byteorder_htons(TCP_HDR_OFFSET_MIN);
2016-02-04 14:37:35 +01:00
return res;
}