2013-08-15 10:17:15 +02:00
|
|
|
/**
|
2014-07-31 17:53:46 +02:00
|
|
|
* TCP timer implementation
|
2013-08-15 10:17:15 +02:00
|
|
|
*
|
|
|
|
* Copyright (C) 2013 INRIA.
|
|
|
|
*
|
2014-07-31 19:45:27 +02: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.
|
2013-08-15 10:17:15 +02:00
|
|
|
*
|
2014-07-31 17:53:46 +02:00
|
|
|
* @ingroup transport_layer
|
2013-08-15 10:17:15 +02:00
|
|
|
* @{
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2013-09-30 13:28:19 +02:00
|
|
|
* @brief TCP timer
|
2013-08-15 10:17:15 +02:00
|
|
|
* @author Oliver Gesch <oliver.gesch@googlemail.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2013-09-30 13:19:19 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "sixlowpan.h"
|
2013-08-15 10:17:15 +02:00
|
|
|
#include "thread.h"
|
2013-09-30 13:19:19 +02:00
|
|
|
#include "vtimer.h"
|
|
|
|
|
|
|
|
#include "msg_help.h"
|
2013-09-23 12:40:16 +02:00
|
|
|
#include "socket.h"
|
|
|
|
|
2014-07-31 17:53:46 +02:00
|
|
|
#include "tcp.h"
|
2013-09-30 13:19:19 +02:00
|
|
|
#include "tcp_timer.h"
|
|
|
|
|
2013-08-15 10:17:15 +02:00
|
|
|
void handle_synchro_timeout(socket_internal_t *current_socket)
|
|
|
|
{
|
|
|
|
msg_t send;
|
|
|
|
|
|
|
|
if (thread_getstatus(current_socket->recv_pid) == STATUS_RECEIVE_BLOCKED) {
|
|
|
|
timex_t now;
|
|
|
|
vtimer_now(&now);
|
|
|
|
|
|
|
|
if ((current_socket->socket_values.tcp_control.no_of_retries == 0) &&
|
2013-12-25 17:35:08 +01:00
|
|
|
(timex_uint64(timex_sub(now, current_socket->socket_values.tcp_control.last_packet_time)) > TCP_SYN_INITIAL_TIMEOUT)) {
|
2013-08-15 10:17:15 +02:00
|
|
|
current_socket->socket_values.tcp_control.no_of_retries++;
|
2014-07-31 17:53:46 +02:00
|
|
|
socket_base_net_msg_send(&send, current_socket->recv_pid, 0, TCP_RETRY);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
|
|
|
else if ((current_socket->socket_values.tcp_control.no_of_retries > 0) &&
|
2013-12-25 17:35:08 +01:00
|
|
|
(timex_uint64(timex_sub(now, current_socket->socket_values.tcp_control.last_packet_time)) >
|
2013-09-30 13:28:19 +02:00
|
|
|
(current_socket->socket_values.tcp_control.no_of_retries *
|
|
|
|
TCP_SYN_TIMEOUT + TCP_SYN_INITIAL_TIMEOUT))) {
|
2013-08-15 10:17:15 +02:00
|
|
|
current_socket->socket_values.tcp_control.no_of_retries++;
|
|
|
|
|
|
|
|
if (current_socket->socket_values.tcp_control.no_of_retries >
|
2013-09-30 13:28:19 +02:00
|
|
|
TCP_MAX_SYN_RETRIES) {
|
2014-07-31 17:53:46 +02:00
|
|
|
socket_base_net_msg_send(&send, current_socket->recv_pid, 0, TCP_TIMEOUT);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
|
|
|
else {
|
2014-07-31 17:53:46 +02:00
|
|
|
socket_base_net_msg_send(&send, current_socket->recv_pid, 0, TCP_RETRY);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void handle_established(socket_internal_t *current_socket)
|
|
|
|
{
|
|
|
|
msg_t send;
|
|
|
|
double current_timeout = current_socket->socket_values.tcp_control.rto;
|
|
|
|
|
|
|
|
if (current_timeout < SECOND) {
|
|
|
|
current_timeout = SECOND;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ((current_socket->socket_values.tcp_control.send_nxt >
|
2013-09-30 13:28:19 +02:00
|
|
|
current_socket->socket_values.tcp_control.send_una) &&
|
|
|
|
(thread_getstatus(current_socket->send_pid) == STATUS_RECEIVE_BLOCKED)) {
|
2014-09-12 15:46:17 +02:00
|
|
|
for (uint8_t i = 0; i < current_socket->socket_values.tcp_control.no_of_retries;
|
2013-09-30 13:28:19 +02:00
|
|
|
i++) {
|
2013-08-15 10:17:15 +02:00
|
|
|
current_timeout *= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
timex_t now;
|
|
|
|
vtimer_now(&now);
|
|
|
|
|
|
|
|
if (current_timeout > TCP_ACK_MAX_TIMEOUT) {
|
2014-07-31 17:53:46 +02:00
|
|
|
socket_base_net_msg_send(&send, current_socket->send_pid, 0, TCP_TIMEOUT);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
2013-12-25 17:35:08 +01:00
|
|
|
else if (timex_uint64(timex_sub(now, current_socket->socket_values.tcp_control.last_packet_time)) >
|
2013-09-30 13:28:19 +02:00
|
|
|
current_timeout) {
|
2013-08-15 10:17:15 +02:00
|
|
|
current_socket->socket_values.tcp_control.no_of_retries++;
|
2014-07-31 17:53:46 +02:00
|
|
|
socket_base_net_msg_send(&send, current_socket->send_pid, 0, TCP_RETRY);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void check_sockets(void)
|
|
|
|
{
|
|
|
|
uint8_t i = 1;
|
|
|
|
|
|
|
|
while (i < MAX_SOCKETS + 1) {
|
2014-09-12 15:46:17 +02:00
|
|
|
socket_internal_t *current_socket = socket_base_get_socket(i);
|
2013-08-15 10:17:15 +02:00
|
|
|
|
2014-07-31 17:53:46 +02:00
|
|
|
if (tcp_socket_compliancy(i)) {
|
2013-09-30 13:28:19 +02:00
|
|
|
switch (current_socket->socket_values.tcp_control.state) {
|
2014-03-17 19:36:30 +01:00
|
|
|
case TCP_ESTABLISHED: {
|
2013-08-15 10:17:15 +02:00
|
|
|
handle_established(current_socket);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-03-17 19:36:30 +01:00
|
|
|
case TCP_SYN_SENT: {
|
2013-08-15 10:17:15 +02:00
|
|
|
handle_synchro_timeout(current_socket);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-03-17 19:36:30 +01:00
|
|
|
case TCP_SYN_RCVD: {
|
2013-08-15 10:17:15 +02:00
|
|
|
handle_synchro_timeout(current_socket);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void inc_global_variables(void)
|
|
|
|
{
|
2014-03-18 12:03:03 +01:00
|
|
|
mutex_lock(&global_sequence_counter_mutex);
|
2013-08-15 10:17:15 +02:00
|
|
|
global_sequence_counter += rand();
|
2014-03-18 12:03:03 +01:00
|
|
|
mutex_unlock(&global_sequence_counter_mutex);
|
2013-08-15 10:17:15 +02:00
|
|
|
#ifdef TCP_HC
|
|
|
|
mutex_lock(&global_context_counter_mutex);
|
|
|
|
global_context_counter += rand();
|
|
|
|
mutex_unlock(&global_context_counter_mutex);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
void *tcp_general_timer(void *arg)
|
2013-08-15 10:17:15 +02:00
|
|
|
{
|
2014-03-04 20:20:01 +01:00
|
|
|
(void) arg;
|
|
|
|
|
2013-08-15 10:17:15 +02:00
|
|
|
vtimer_t tcp_vtimer;
|
|
|
|
timex_t interval = timex_set(0, TCP_TIMER_RESOLUTION);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
inc_global_variables();
|
|
|
|
check_sockets();
|
2014-01-18 01:20:40 +01:00
|
|
|
|
2013-08-15 10:17:15 +02:00
|
|
|
vtimer_set_wakeup(&tcp_vtimer, interval, thread_getpid());
|
|
|
|
thread_sleep();
|
2014-01-18 01:20:40 +01:00
|
|
|
|
|
|
|
vtimer_remove(&tcp_vtimer);
|
2013-08-15 10:17:15 +02:00
|
|
|
}
|
|
|
|
}
|