mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
1423 lines
51 KiB
C
1423 lines
51 KiB
C
/**
|
|
* TCP implementation
|
|
*
|
|
* Copyright (C) 2013 INRIA.
|
|
*
|
|
* 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 tcp
|
|
* @{
|
|
* @file
|
|
* @brief TCP implementation
|
|
* @author Oliver Gesch <oliver.gesch@googlemail.com>
|
|
* @author Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
* @}
|
|
*/
|
|
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "sixlowpan.h"
|
|
#include "thread.h"
|
|
#include "vtimer.h"
|
|
|
|
#include "socket_base/in.h"
|
|
|
|
#include "net_help.h"
|
|
|
|
#include "msg_help.h"
|
|
#include "socket.h"
|
|
#include "tcp_hc.h"
|
|
#include "tcp_timer.h"
|
|
|
|
#include "tcp.h"
|
|
|
|
#ifdef TCP_HC
|
|
mutex_t global_context_counter_mutex;
|
|
uint8_t global_context_counter;
|
|
#endif
|
|
|
|
mutex_t global_sequence_counter_mutex;
|
|
uint32_t global_sequence_counter;
|
|
|
|
char tcp_stack_buffer[TCP_STACK_SIZE];
|
|
char tcp_timer_stack[TCP_TIMER_STACKSIZE];
|
|
|
|
void set_socket_address(sockaddr6_t *sockaddr, uint8_t sin6_family,
|
|
uint16_t sin6_port, uint32_t sin6_flowinfo, ipv6_addr_t *sin6_addr)
|
|
{
|
|
sockaddr->sin6_family = sin6_family;
|
|
sockaddr->sin6_port = sin6_port;
|
|
sockaddr->sin6_flowinfo = sin6_flowinfo;
|
|
memcpy(&sockaddr->sin6_addr, sin6_addr, 16);
|
|
}
|
|
|
|
void printTCPHeader(tcp_hdr_t *tcp_header)
|
|
{
|
|
printf("\nBEGIN: TCP HEADER\n");
|
|
printf("ack_nr: %" PRIu32 "\n", tcp_header->ack_nr);
|
|
printf("checksum: %i\n", tcp_header->checksum);
|
|
printf("data_offset: %i\n", tcp_header->data_offset);
|
|
printf("dst_port: %i\n", tcp_header->dst_port);
|
|
printf("reserved_flags: %i\n", tcp_header->reserved_flags);
|
|
printf("seq_nr: %" PRIu32 "\n", tcp_header->seq_nr);
|
|
printf("src_port: %i\n", tcp_header->src_port);
|
|
printf("urg_pointer: %i\n", tcp_header->urg_pointer);
|
|
printf("window: %i\n", tcp_header->window);
|
|
printf("END: TCP HEADER\n");
|
|
}
|
|
|
|
void print_tcp_flags(tcp_hdr_t *tcp_header)
|
|
{
|
|
printf("FLAGS: ");
|
|
|
|
switch (tcp_header->reserved_flags) {
|
|
case TCP_ACK: {
|
|
printf("ACK ");
|
|
break;
|
|
}
|
|
|
|
case TCP_RST: {
|
|
printf("RST ");
|
|
break;
|
|
}
|
|
|
|
case TCP_SYN: {
|
|
printf("SYN ");
|
|
break;
|
|
}
|
|
|
|
case TCP_FIN: {
|
|
printf("FIN ");
|
|
break;
|
|
}
|
|
|
|
case TCP_URG_PSH: {
|
|
printf("URG PSH ");
|
|
break;
|
|
}
|
|
|
|
case TCP_SYN_ACK: {
|
|
printf("SYN ACK ");
|
|
break;
|
|
}
|
|
|
|
case TCP_FIN_ACK: {
|
|
printf("FIN ACK ");
|
|
break;
|
|
}
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
void print_tcp_cb(tcp_cb_t *cb)
|
|
{
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
printf("Send_ISS: %" PRIu32 "\nSend_UNA: %" PRIu32 "\nSend_NXT: %" PRIu32 "\nSend_WND: %u\n",
|
|
cb->send_iss, cb->send_una, cb->send_nxt, cb->send_wnd);
|
|
printf("Rcv_IRS: %" PRIu32 "\nRcv_NXT: %" PRIu32 "\nRcv_WND: %u\n",
|
|
cb->rcv_irs, cb->rcv_nxt, cb->rcv_wnd);
|
|
printf("Time difference: %" PRIu64 ", No_of_retries: %u, State: %u\n\n",
|
|
timex_uint64(timex_sub(now, cb->last_packet_time)), cb->no_of_retries, cb->state);
|
|
}
|
|
|
|
void print_tcp_status(int in_or_out, ipv6_hdr_t *ipv6_header,
|
|
tcp_hdr_t *tcp_header, socket_t *tcp_socket)
|
|
{
|
|
char addr_str[IPV6_MAX_ADDR_STR_LEN];
|
|
printf("--- %s TCP packet: ---\n",
|
|
(in_or_out == INC_PACKET ? "Incoming" : "Outgoing"));
|
|
printf("IPv6 Source: %s\n",
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
&ipv6_header->srcaddr));
|
|
printf("IPv6 Dest: %s\n",
|
|
ipv6_addr_to_str(addr_str, IPV6_MAX_ADDR_STR_LEN,
|
|
&ipv6_header->destaddr));
|
|
printf("TCP Length: %x\n", NTOHS(ipv6_header->length) - TCP_HDR_LEN);
|
|
printf("Source Port: %x, Dest. Port: %x\n",
|
|
NTOHS(tcp_header->src_port), NTOHS(tcp_header->dst_port));
|
|
printf("Source Port: %u, Dest. Port: %u\n",
|
|
NTOHS(tcp_header->src_port), NTOHS(tcp_header->dst_port));
|
|
printf("ACK: %" PRIu32 ", SEQ: %" PRIu32 ", Window: %x\n",
|
|
tcp_header->ack_nr, tcp_header->seq_nr, tcp_header->window);
|
|
printf("ACK: %" PRIu32 ", SEQ: %" PRIu32 ", Window: %u\n",
|
|
tcp_header->ack_nr, tcp_header->seq_nr, tcp_header->window);
|
|
print_tcp_flags(tcp_header);
|
|
print_tcp_cb(&tcp_socket->tcp_control);
|
|
#ifdef TCP_HC
|
|
printf_tcp_context(&tcp_socket->tcp_control.tcp_context);
|
|
#endif
|
|
}
|
|
|
|
uint16_t tcp_csum(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header)
|
|
{
|
|
uint16_t sum;
|
|
uint16_t len = NTOHS(ipv6_header->length);
|
|
|
|
sum = len + IPPROTO_TCP;
|
|
sum = csum(sum, (uint8_t *)&ipv6_header->srcaddr, 2 * sizeof(ipv6_addr_t));
|
|
sum = csum(sum, (uint8_t *)tcp_header, len);
|
|
return (sum == 0) ? 0xffff : HTONS(sum);
|
|
}
|
|
|
|
/* Check for consistent ACK and SEQ number */
|
|
int check_tcp_consistency(socket_t *current_tcp_socket, tcp_hdr_t *tcp_header, uint8_t tcp_payload_len)
|
|
{
|
|
if (tcp_payload_len == 0) {
|
|
if (tcp_header->ack_nr > (current_tcp_socket->tcp_control.send_nxt)) {
|
|
/* ACK of not yet sent byte, discard */
|
|
return ACK_NO_TOO_BIG;
|
|
}
|
|
else if (tcp_header->ack_nr <= (current_tcp_socket->tcp_control.send_una)) {
|
|
/* ACK of previous segments, maybe dropped? */
|
|
return ACK_NO_TOO_SMALL;
|
|
}
|
|
}
|
|
else if ((current_tcp_socket->tcp_control.rcv_nxt > 0) && (tcp_header->seq_nr < current_tcp_socket->tcp_control.rcv_nxt)) {
|
|
/* segment repetition, maybe ACK got lost? */
|
|
return SEQ_NO_TOO_SMALL;
|
|
}
|
|
|
|
return PACKET_OK;
|
|
}
|
|
|
|
void set_tcp_cb(tcp_cb_t *tcp_control, uint32_t rcv_nxt, uint16_t rcv_wnd,
|
|
uint32_t send_nxt, uint32_t send_una, uint16_t send_wnd)
|
|
{
|
|
tcp_control->rcv_nxt = rcv_nxt;
|
|
tcp_control->rcv_wnd = rcv_wnd;
|
|
tcp_control->send_nxt = send_nxt;
|
|
tcp_control->send_una = send_una;
|
|
tcp_control->send_wnd = send_wnd;
|
|
}
|
|
|
|
void switch_tcp_packet_byte_order(tcp_hdr_t *current_tcp_packet)
|
|
{
|
|
if (current_tcp_packet->data_offset * 4 > TCP_HDR_LEN) {
|
|
if (*(((uint8_t *)current_tcp_packet) + TCP_HDR_LEN) == TCP_MSS_OPTION) {
|
|
uint8_t *packet_pointer = (uint8_t *)current_tcp_packet;
|
|
packet_pointer += (TCP_HDR_LEN + 2);
|
|
uint8_t mss1 = *packet_pointer;
|
|
uint8_t mss2 = *(packet_pointer + 1);
|
|
*packet_pointer = mss2;
|
|
*(packet_pointer + 1) = mss1;
|
|
}
|
|
|
|
if (*(((uint8_t *)current_tcp_packet) + TCP_HDR_LEN) == TCP_TS_OPTION) {
|
|
/* TODO: Timestamp option not implemented */
|
|
}
|
|
}
|
|
|
|
current_tcp_packet->seq_nr = HTONL(current_tcp_packet->seq_nr);
|
|
current_tcp_packet->ack_nr = HTONL(current_tcp_packet->ack_nr);
|
|
current_tcp_packet->window = HTONS(current_tcp_packet->window);
|
|
current_tcp_packet->urg_pointer = HTONS(current_tcp_packet->urg_pointer);
|
|
}
|
|
|
|
void set_tcp_packet(tcp_hdr_t *tcp_hdr, uint16_t src_port, uint16_t dst_port,
|
|
uint32_t seq_nr, uint32_t ack_nr, uint8_t data_offset,
|
|
uint8_t reserved_flags, uint16_t window, uint16_t checksum,
|
|
uint16_t urg_pointer)
|
|
{
|
|
tcp_hdr->ack_nr = ack_nr;
|
|
tcp_hdr->checksum = checksum;
|
|
tcp_hdr->data_offset = data_offset;
|
|
tcp_hdr->dst_port = dst_port;
|
|
tcp_hdr->reserved_flags = reserved_flags;
|
|
tcp_hdr->reserved = 0;
|
|
tcp_hdr->flag_ns = 0;
|
|
tcp_hdr->seq_nr = seq_nr;
|
|
tcp_hdr->src_port = src_port;
|
|
tcp_hdr->urg_pointer = urg_pointer;
|
|
tcp_hdr->window = window;
|
|
}
|
|
|
|
int send_tcp(socket_internal_t *current_socket, tcp_hdr_t *current_tcp_packet,
|
|
ipv6_hdr_t *temp_ipv6_header, uint8_t flags, uint8_t payload_length)
|
|
{
|
|
socket_t *current_tcp_socket = ¤t_socket->socket_values;
|
|
uint8_t header_length = TCP_HDR_LEN / 4;
|
|
|
|
if (IS_TCP_SYN(flags) || IS_TCP_SYN_ACK(flags)) {
|
|
tcp_mss_option_t current_mss_option;
|
|
header_length += sizeof(tcp_mss_option_t) / 4;
|
|
|
|
current_mss_option.kind = TCP_MSS_OPTION;
|
|
current_mss_option.len = sizeof(tcp_mss_option_t);
|
|
current_mss_option.mss = TRANSPORT_LAYER_SOCKET_STATIC_MSS;
|
|
memcpy(((uint8_t *)current_tcp_packet) + TCP_HDR_LEN,
|
|
¤t_mss_option, sizeof(tcp_mss_option_t));
|
|
}
|
|
|
|
set_tcp_packet(current_tcp_packet, current_tcp_socket->local_address.sin6_port,
|
|
current_tcp_socket->foreign_address.sin6_port,
|
|
current_tcp_socket->tcp_control.send_una,
|
|
(IS_TCP_ACK(flags) ? current_tcp_socket->tcp_control.rcv_nxt : 0x00), header_length, flags,
|
|
current_tcp_socket->tcp_control.rcv_wnd, 0, 0);
|
|
|
|
/* Fill IPv6 Header */
|
|
memcpy(&(temp_ipv6_header->destaddr),
|
|
¤t_tcp_socket->foreign_address.sin6_addr, 16);
|
|
memcpy(&(temp_ipv6_header->srcaddr),
|
|
¤t_tcp_socket->local_address.sin6_addr, 16);
|
|
temp_ipv6_header->length = HTONS(header_length * 4 + payload_length);
|
|
|
|
current_tcp_packet->checksum = ~tcp_csum(temp_ipv6_header, current_tcp_packet);
|
|
|
|
#ifdef TCP_HC
|
|
uint16_t compressed_size;
|
|
|
|
compressed_size = compress_tcp_packet(current_socket,
|
|
(uint8_t *)current_tcp_packet,
|
|
temp_ipv6_header, flags,
|
|
payload_length);
|
|
|
|
if (compressed_size == 0) {
|
|
/* Error in compressing tcp packet header */
|
|
return -1;
|
|
}
|
|
|
|
return ipv6_sendto(¤t_tcp_socket->foreign_address.sin6_addr,
|
|
IPPROTO_TCP, (uint8_t *)(current_tcp_packet),
|
|
compressed_size);
|
|
#else
|
|
switch_tcp_packet_byte_order(current_tcp_packet);
|
|
return ipv6_sendto(¤t_tcp_socket->foreign_address.sin6_addr,
|
|
IPPROTO_TCP, (uint8_t *)(current_tcp_packet),
|
|
header_length * 4 + payload_length, NULL);
|
|
#endif
|
|
}
|
|
|
|
bool is_four_touple(socket_internal_t *current_socket, ipv6_hdr_t *ipv6_header,
|
|
tcp_hdr_t *tcp_header)
|
|
{
|
|
return (ipv6_addr_is_equal(¤t_socket->socket_values.local_address.sin6_addr,
|
|
&ipv6_header->destaddr) &&
|
|
(current_socket->socket_values.local_address.sin6_port == tcp_header->dst_port) &&
|
|
ipv6_addr_is_equal(¤t_socket->socket_values.foreign_address.sin6_addr,
|
|
&ipv6_header->srcaddr) &&
|
|
(current_socket->socket_values.foreign_address.sin6_port == tcp_header->src_port));
|
|
}
|
|
|
|
socket_internal_t *get_waiting_connection_socket(int socket,
|
|
ipv6_hdr_t *ipv6_header,
|
|
tcp_hdr_t *tcp_header)
|
|
{
|
|
int i;
|
|
socket_internal_t *listening_socket = socket_base_get_socket(socket);
|
|
|
|
for (i = 1; i < MAX_SOCKETS + 1; i++) {
|
|
socket_internal_t *current_socket = socket_base_get_socket(i);
|
|
|
|
if (!current_socket) {
|
|
continue;
|
|
}
|
|
|
|
/* Connection establishment ACK, Check for 4 touple and state */
|
|
if ((ipv6_header != NULL) && (tcp_header != NULL)) {
|
|
if (is_four_touple(current_socket, ipv6_header, tcp_header) &&
|
|
(current_socket->socket_values.tcp_control.state == TCP_SYN_RCVD)) {
|
|
return current_socket;
|
|
}
|
|
}
|
|
/* Connection establishment SYN ACK, check only for port and state */
|
|
else {
|
|
if ((current_socket->socket_values.tcp_control.state == TCP_SYN_RCVD) &&
|
|
(current_socket->socket_values.local_address.sin6_port ==
|
|
listening_socket->socket_values.local_address.sin6_port)) {
|
|
return current_socket;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
socket_internal_t *new_tcp_queued_socket(ipv6_hdr_t *ipv6_header,
|
|
tcp_hdr_t *tcp_header)
|
|
{
|
|
int queued_socket_id;
|
|
|
|
queued_socket_id = socket_base_socket(PF_INET6, SOCK_STREAM, IPPROTO_TCP);
|
|
socket_internal_t *current_queued_socket = socket_base_get_socket(queued_socket_id);
|
|
|
|
/* Foreign address */
|
|
set_socket_address(¤t_queued_socket->socket_values.foreign_address,
|
|
AF_INET6, tcp_header->src_port, ipv6_header->flowlabel,
|
|
&ipv6_header->srcaddr);
|
|
|
|
/* Local address */
|
|
set_socket_address(¤t_queued_socket->socket_values.local_address,
|
|
AF_INET6, tcp_header->dst_port, 0,
|
|
&ipv6_header->destaddr);
|
|
|
|
/* Foreign TCP information */
|
|
if ((tcp_header->data_offset * 4 > TCP_HDR_LEN) &&
|
|
(*(((uint8_t *)tcp_header) + TCP_HDR_LEN) == TCP_MSS_OPTION)) {
|
|
current_queued_socket->socket_values.tcp_control.mss =
|
|
*((uint16_t *)(((uint8_t *)tcp_header) + TCP_HDR_LEN + 2));
|
|
}
|
|
else {
|
|
current_queued_socket->socket_values.tcp_control.mss = TRANSPORT_LAYER_SOCKET_STATIC_MSS;
|
|
}
|
|
|
|
current_queued_socket->socket_values.tcp_control.rcv_irs =
|
|
tcp_header->seq_nr;
|
|
mutex_lock(&global_sequence_counter_mutex);
|
|
current_queued_socket->socket_values.tcp_control.send_iss =
|
|
global_sequence_counter;
|
|
mutex_unlock(&global_sequence_counter_mutex);
|
|
current_queued_socket->socket_values.tcp_control.state = TCP_SYN_RCVD;
|
|
set_tcp_cb(¤t_queued_socket->socket_values.tcp_control,
|
|
tcp_header->seq_nr + 1, TRANSPORT_LAYER_SOCKET_STATIC_WINDOW,
|
|
current_queued_socket->socket_values.tcp_control.send_iss + 1,
|
|
current_queued_socket->socket_values.tcp_control.send_iss,
|
|
tcp_header->window);
|
|
|
|
return current_queued_socket;
|
|
}
|
|
|
|
socket_internal_t *get_tcp_socket(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header)
|
|
{
|
|
uint8_t i = 1;
|
|
socket_internal_t *current_socket = NULL;
|
|
socket_internal_t *listening_socket = NULL;
|
|
uint8_t compare[16];
|
|
memset(compare, 0, 16);
|
|
|
|
while (i < MAX_SOCKETS + 1) {
|
|
current_socket = socket_base_get_socket(i);
|
|
|
|
/* Check for matching 4 touple, TCP_ESTABLISHED connection */
|
|
if (tcp_socket_compliancy(i) && is_four_touple(current_socket, ipv6_header,
|
|
tcp_header)) {
|
|
return current_socket;
|
|
}
|
|
/* Sockets in TCP_LISTEN and TCP_SYN_RCVD state should only be tested on local TCP values */
|
|
else if (tcp_socket_compliancy(i) &&
|
|
((current_socket->socket_values.tcp_control.state == TCP_LISTEN) ||
|
|
(current_socket->socket_values.tcp_control.state == TCP_SYN_RCVD)) &&
|
|
(current_socket->socket_values.local_address.sin6_addr.uint8[15] ==
|
|
ipv6_header->destaddr.uint8[15]) &&
|
|
(current_socket->socket_values.local_address.sin6_port ==
|
|
tcp_header->dst_port) &&
|
|
(current_socket->socket_values.foreign_address.sin6_addr.uint8[15] ==
|
|
0x00) &&
|
|
(current_socket->socket_values.foreign_address.sin6_port == 0)) {
|
|
listening_socket = current_socket;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
/* Return either NULL if nothing was matched or the listening 2 touple socket */
|
|
return listening_socket;
|
|
}
|
|
|
|
uint8_t handle_payload(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket, uint8_t *payload)
|
|
{
|
|
(void) tcp_header;
|
|
|
|
msg_t m_send_tcp, m_recv_tcp;
|
|
uint8_t tcp_payload_len = NTOHS(ipv6_header->length) - TCP_HDR_LEN;
|
|
uint8_t acknowledged_bytes = 0;
|
|
|
|
if (tcp_payload_len > tcp_socket->socket_values.tcp_control.rcv_wnd) {
|
|
mutex_lock(&tcp_socket->tcp_buffer_mutex);
|
|
memcpy(tcp_socket->tcp_input_buffer, payload,
|
|
tcp_socket->socket_values.tcp_control.rcv_wnd);
|
|
acknowledged_bytes = tcp_socket->socket_values.tcp_control.rcv_wnd;
|
|
tcp_socket->socket_values.tcp_control.rcv_wnd = 0;
|
|
tcp_socket->tcp_input_buffer_end = tcp_socket->tcp_input_buffer_end +
|
|
tcp_socket->socket_values.tcp_control.rcv_wnd;
|
|
mutex_unlock(&tcp_socket->tcp_buffer_mutex);
|
|
}
|
|
else {
|
|
mutex_lock(&tcp_socket->tcp_buffer_mutex);
|
|
memcpy(tcp_socket->tcp_input_buffer, payload, tcp_payload_len);
|
|
tcp_socket->socket_values.tcp_control.rcv_wnd =
|
|
tcp_socket->socket_values.tcp_control.rcv_wnd - tcp_payload_len;
|
|
acknowledged_bytes = tcp_payload_len;
|
|
tcp_socket->tcp_input_buffer_end = tcp_socket->tcp_input_buffer_end +
|
|
tcp_payload_len;
|
|
mutex_unlock(&tcp_socket->tcp_buffer_mutex);
|
|
}
|
|
|
|
if (thread_getstatus(tcp_socket->recv_pid) == STATUS_RECEIVE_BLOCKED) {
|
|
socket_base_net_msg_send_recv(&m_send_tcp, &m_recv_tcp, tcp_socket->recv_pid, UNDEFINED);
|
|
}
|
|
|
|
return acknowledged_bytes;
|
|
}
|
|
|
|
void handle_tcp_ack_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
msg_t m_recv_tcp, m_send_tcp;
|
|
|
|
if (tcp_socket->socket_values.tcp_control.state == TCP_LAST_ACK) {
|
|
uint8_t target_pid = tcp_socket->recv_pid;
|
|
memset(tcp_socket, 0, sizeof(socket_internal_t));
|
|
msg_try_send(&m_send_tcp, target_pid);
|
|
return;
|
|
}
|
|
else if (tcp_socket->socket_values.tcp_control.state == TCP_CLOSING) {
|
|
msg_try_send(&m_send_tcp, tcp_socket->recv_pid);
|
|
msg_try_send(&m_send_tcp, tcp_socket->send_pid);
|
|
return;
|
|
}
|
|
else if (get_waiting_connection_socket(tcp_socket->socket_id, ipv6_header,
|
|
tcp_header) != NULL) {
|
|
m_send_tcp.content.ptr = (char *)tcp_header;
|
|
socket_base_net_msg_send_recv(&m_send_tcp, &m_recv_tcp, tcp_socket->recv_pid, TCP_ACK);
|
|
return;
|
|
}
|
|
else if (tcp_socket->socket_values.tcp_control.state == TCP_ESTABLISHED) {
|
|
if (check_tcp_consistency(&tcp_socket->socket_values, tcp_header, 0) == PACKET_OK) {
|
|
m_send_tcp.content.ptr = (char *)tcp_header;
|
|
socket_base_net_msg_send(&m_send_tcp, tcp_socket->send_pid, 0, TCP_ACK);
|
|
return;
|
|
}
|
|
}
|
|
|
|
printf("NO WAY OF HANDLING THIS ACK!\n");
|
|
}
|
|
|
|
void handle_tcp_rst_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
(void) ipv6_header;
|
|
(void) tcp_header;
|
|
(void) tcp_socket;
|
|
|
|
/* TODO: Reset connection */
|
|
}
|
|
|
|
void handle_tcp_syn_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
msg_t m_send_tcp;
|
|
|
|
if (tcp_socket->socket_values.tcp_control.state == TCP_LISTEN) {
|
|
socket_internal_t *new_socket = new_tcp_queued_socket(ipv6_header,
|
|
tcp_header);
|
|
|
|
if (new_socket != NULL) {
|
|
#ifdef TCP_HC
|
|
update_tcp_hc_context(true, new_socket, tcp_header);
|
|
#endif
|
|
/* notify socket function tcp_accept(..) that a new
|
|
* connection request has arrived. No need to wait for an answer
|
|
* because the server tcp_accept() function isnt reading
|
|
* from anything other than the queued sockets */
|
|
socket_base_net_msg_send(&m_send_tcp, tcp_socket->recv_pid, 0, TCP_SYN);
|
|
}
|
|
else {
|
|
printf("Dropped TCP SYN Message because an error occured while "\
|
|
"requesting a new queued socket!\n");
|
|
}
|
|
}
|
|
else {
|
|
printf("Dropped TCP SYN Message because socket was not in state TCP_LISTEN!");
|
|
}
|
|
}
|
|
|
|
void handle_tcp_syn_ack_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
(void) ipv6_header;
|
|
|
|
msg_t m_send_tcp;
|
|
|
|
if (tcp_socket->socket_values.tcp_control.state == TCP_SYN_SENT) {
|
|
m_send_tcp.content.ptr = (char *) tcp_header;
|
|
socket_base_net_msg_send(&m_send_tcp, tcp_socket->recv_pid, 0, TCP_SYN_ACK);
|
|
}
|
|
else {
|
|
printf("Socket not in state TCP_SYN_SENT, dropping SYN-ACK-packet!");
|
|
}
|
|
}
|
|
|
|
void handle_tcp_fin_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
(void) ipv6_header;
|
|
|
|
msg_t m_send;
|
|
socket_t *current_tcp_socket = &tcp_socket->socket_values;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
set_tcp_cb(¤t_tcp_socket->tcp_control, tcp_header->seq_nr + 1,
|
|
current_tcp_socket->tcp_control.send_wnd, tcp_header->ack_nr + 1,
|
|
tcp_header->ack_nr, tcp_header->window);
|
|
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = COMPRESSED_HEADER;
|
|
#endif
|
|
|
|
if (current_tcp_socket->tcp_control.state == TCP_FIN_WAIT_1) {
|
|
current_tcp_socket->tcp_control.state = TCP_CLOSING;
|
|
|
|
send_tcp(tcp_socket, current_tcp_packet, temp_ipv6_header, TCP_FIN_ACK, 0);
|
|
}
|
|
else {
|
|
current_tcp_socket->tcp_control.state = TCP_LAST_ACK;
|
|
|
|
send_tcp(tcp_socket, current_tcp_packet, temp_ipv6_header, TCP_FIN_ACK, 0);
|
|
}
|
|
|
|
socket_base_net_msg_send(&m_send, tcp_socket->recv_pid, 0, CLOSE_CONN);
|
|
}
|
|
|
|
void handle_tcp_fin_ack_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket)
|
|
{
|
|
(void) ipv6_header;
|
|
|
|
msg_t m_send;
|
|
socket_t *current_tcp_socket = &tcp_socket->socket_values;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
current_tcp_socket->tcp_control.state = TCP_CLOSED;
|
|
|
|
set_tcp_cb(¤t_tcp_socket->tcp_control, tcp_header->seq_nr + 1,
|
|
current_tcp_socket->tcp_control.send_wnd, tcp_header->ack_nr,
|
|
tcp_header->ack_nr, tcp_header->window);
|
|
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = COMPRESSED_HEADER;
|
|
#endif
|
|
|
|
send_tcp(tcp_socket, current_tcp_packet, temp_ipv6_header, TCP_ACK, 0);
|
|
|
|
msg_try_send(&m_send, tcp_socket->send_pid);
|
|
msg_try_send(&m_send, tcp_socket->recv_pid);
|
|
}
|
|
|
|
void handle_tcp_no_flags_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header,
|
|
socket_internal_t *tcp_socket, uint8_t *payload, uint8_t tcp_payload_len)
|
|
{
|
|
socket_t *current_tcp_socket = &tcp_socket->socket_values;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
if (check_tcp_consistency(current_tcp_socket, tcp_header, tcp_payload_len) == PACKET_OK) {
|
|
uint8_t read_bytes = handle_payload(ipv6_header, tcp_header, tcp_socket, payload);
|
|
|
|
/* Refresh TCP status values */
|
|
current_tcp_socket->tcp_control.state = TCP_ESTABLISHED;
|
|
|
|
set_tcp_cb(¤t_tcp_socket->tcp_control,
|
|
tcp_header->seq_nr + read_bytes,
|
|
current_tcp_socket->tcp_control.rcv_wnd,
|
|
current_tcp_socket->tcp_control.send_nxt,
|
|
current_tcp_socket->tcp_control.send_una,
|
|
current_tcp_socket->tcp_control.send_wnd);
|
|
|
|
/* Send packet */
|
|
// block_continue_thread();
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = COMPRESSED_HEADER;
|
|
#endif
|
|
send_tcp(tcp_socket, current_tcp_packet, temp_ipv6_header, TCP_ACK, 0);
|
|
}
|
|
/* ACK packet probably got lost */
|
|
else {
|
|
// block_continue_thread();
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = FULL_HEADER;
|
|
#endif
|
|
send_tcp(tcp_socket, current_tcp_packet, temp_ipv6_header, TCP_ACK, 0);
|
|
}
|
|
}
|
|
|
|
void *tcp_packet_handler(void *arg)
|
|
{
|
|
(void) arg;
|
|
|
|
msg_t m_recv_ip, m_send_ip;
|
|
socket_internal_t *tcp_socket = NULL;
|
|
|
|
while (1) {
|
|
msg_receive(&m_recv_ip);
|
|
|
|
ipv6_hdr_t *ipv6_header = ((ipv6_hdr_t *)m_recv_ip.content.ptr);
|
|
tcp_hdr_t *tcp_header = ((tcp_hdr_t *)(m_recv_ip.content.ptr + IPV6_HDR_LEN));
|
|
#ifdef TCP_HC
|
|
tcp_socket = decompress_tcp_packet(ipv6_header);
|
|
#else
|
|
switch_tcp_packet_byte_order(tcp_header);
|
|
tcp_socket = get_tcp_socket(ipv6_header, tcp_header);
|
|
#endif
|
|
uint16_t chksum = tcp_csum(ipv6_header, tcp_header);
|
|
|
|
uint8_t *payload = (uint8_t *)(m_recv_ip.content.ptr + IPV6_HDR_LEN + tcp_header->data_offset * 4);
|
|
|
|
if ((chksum == 0xffff) && (tcp_socket != NULL)) {
|
|
#ifdef TCP_HC
|
|
update_tcp_hc_context(true, tcp_socket, tcp_header);
|
|
#endif
|
|
/* Remove reserved bits from tcp flags field */
|
|
switch (tcp_header->reserved_flags) {
|
|
case TCP_ACK: {
|
|
/* only ACK Bit set */
|
|
uint8_t tcp_payload_len = NTOHS(ipv6_header->length) - TCP_HDR_LEN;
|
|
uint8_t state = tcp_socket->socket_values.tcp_control.state;
|
|
|
|
if ((tcp_payload_len > 0) && (state == TCP_ESTABLISHED)) {
|
|
/* handle data segments only when the connection was established successfully */
|
|
handle_tcp_no_flags_packet(ipv6_header, tcp_header, tcp_socket, payload, tcp_payload_len);
|
|
}
|
|
else if (tcp_payload_len == 0
|
|
&& (state == TCP_ESTABLISHED || state == TCP_SYN_RCVD
|
|
|| state == TCP_CLOSING || state == TCP_LAST_ACK)) {
|
|
/* no payload, acknowledging data only */
|
|
handle_tcp_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case TCP_RST: {
|
|
printf("RST Bit set!\n");
|
|
/* only RST Bit set */
|
|
handle_tcp_rst_packet(ipv6_header, tcp_header, tcp_socket);
|
|
break;
|
|
}
|
|
|
|
case TCP_SYN: {
|
|
/* only SYN Bit set, look for matching, listening socket
|
|
* and request new queued socket */
|
|
printf("SYN Bit set!\n");
|
|
handle_tcp_syn_packet(ipv6_header, tcp_header, tcp_socket);
|
|
break;
|
|
}
|
|
|
|
case TCP_SYN_ACK: {
|
|
/* only SYN and ACK Bit set, complete three way handshake
|
|
* when socket in state TCP_SYN_SENT */
|
|
handle_tcp_syn_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
|
break;
|
|
}
|
|
|
|
case TCP_FIN_ACK: {
|
|
if (tcp_socket->socket_values.tcp_control.state == TCP_ESTABLISHED) {
|
|
/* this is the first FIN */
|
|
printf("FIN ACK Bit set!\n");
|
|
handle_tcp_fin_packet(ipv6_header, tcp_header, tcp_socket);
|
|
}
|
|
else {
|
|
/* this is the response to FIN */
|
|
handle_tcp_fin_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
|
}
|
|
break;
|
|
}
|
|
|
|
default: {
|
|
printf("Unable to process the incoming segment!\n");
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
printf("Wrong checksum (%x) or no corresponding socket found!\n",
|
|
chksum);
|
|
print_tcp_status(INC_PACKET, ipv6_header, tcp_header,
|
|
&tcp_socket->socket_values);
|
|
}
|
|
|
|
msg_reply(&m_recv_ip, &m_send_ip);
|
|
}
|
|
}
|
|
|
|
void printf_tcp_context(tcp_hc_context_t *current_tcp_context)
|
|
{
|
|
printf("Context: %u\n", current_tcp_context->context_id);
|
|
printf("Rcv Seq: %" PRIu32 " Rcv Ack: %" PRIu32 ", Rcv Wnd: %u\n", current_tcp_context->seq_rcv, current_tcp_context->ack_rcv, current_tcp_context->wnd_rcv);
|
|
printf("Snd Seq: %" PRIu32 " Snd Ack: %" PRIu32 ", Snd Wnd: %u\n", current_tcp_context->seq_snd, current_tcp_context->ack_snd, current_tcp_context->wnd_snd);
|
|
}
|
|
|
|
bool tcp_socket_compliancy(int s)
|
|
{
|
|
if ((socket_base_exists_socket(s)) &&
|
|
(socket_base_get_socket(s)->socket_values.domain == PF_INET6) &&
|
|
(socket_base_get_socket(s)->socket_values.type == SOCK_STREAM) &&
|
|
((socket_base_get_socket(s)->socket_values.protocol == IPPROTO_TCP) ||
|
|
(socket_base_get_socket(s)->socket_values.protocol == 0))) {
|
|
return true;
|
|
}
|
|
else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
int tcp_bind_socket(int s, sockaddr6_t *name, int namelen, uint8_t pid)
|
|
{
|
|
int i;
|
|
|
|
socket_internal_t *sock = socket_base_get_socket(s);
|
|
|
|
if (!sock) {
|
|
return -1;
|
|
}
|
|
|
|
for (i = 1; i < MAX_SOCKETS + 1; i++) {
|
|
if (tcp_socket_compliancy(i) &&
|
|
(socket_base_get_socket(i)->socket_values.local_address.sin6_port == name->sin6_port)) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
(void) namelen;
|
|
sock->socket_values.local_address = *name;
|
|
sock->socket_values.tcp_control.rto = TCP_INITIAL_ACK_TIMEOUT;
|
|
sock->recv_pid = pid;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void calculate_rto(tcp_cb_t *tcp_control, timex_t current_time)
|
|
{
|
|
double rtt = (double) timex_uint64(timex_sub(current_time, tcp_control->last_packet_time));
|
|
double srtt = tcp_control->srtt;
|
|
double rttvar = tcp_control->rttvar;
|
|
double rto = tcp_control->rto;
|
|
|
|
if ((srtt == 0) && (rttvar == 0) && (rto == TCP_INITIAL_ACK_TIMEOUT)) {
|
|
/* First calculation */
|
|
srtt = rtt;
|
|
rttvar = 0.5 * rtt;
|
|
rto = rtt + (((4 * rttvar) < TCP_TIMER_RESOLUTION) ?
|
|
(TCP_TIMER_RESOLUTION) : (4 * rttvar));
|
|
}
|
|
else {
|
|
/* every other calculation */
|
|
srtt = (1 - TCP_ALPHA) * srtt + TCP_ALPHA * rtt;
|
|
rttvar = (1 - TCP_BETA) * rttvar + TCP_BETA * abs(srtt - rtt);
|
|
rto = srtt + (((4 * rttvar) < TCP_TIMER_RESOLUTION) ?
|
|
(TCP_TIMER_RESOLUTION) : (4 * rttvar));
|
|
}
|
|
|
|
if (rto < SECOND) {
|
|
rto = SECOND;
|
|
}
|
|
|
|
tcp_control->srtt = srtt;
|
|
tcp_control->rttvar = rttvar;
|
|
tcp_control->rto = rto;
|
|
}
|
|
|
|
int handle_new_tcp_connection(socket_internal_t *current_queued_int_socket,
|
|
socket_internal_t *server_socket, uint8_t pid)
|
|
{
|
|
(void) pid;
|
|
|
|
msg_t msg_recv_client_ack, msg_send_client_ack;
|
|
socket_t *current_queued_socket = ¤t_queued_int_socket->socket_values;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *syn_ack_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
current_queued_int_socket->recv_pid = thread_getpid();
|
|
#ifdef TCP_HC
|
|
current_queued_int_socket->socket_values.tcp_control.tcp_context.hc_type =
|
|
FULL_HEADER;
|
|
memcpy(¤t_queued_int_socket->socket_values.tcp_control.tcp_context.context_id,
|
|
&server_socket->socket_values.tcp_control.tcp_context.context_id,
|
|
sizeof(server_socket->socket_values.tcp_control.tcp_context.context_id));
|
|
#endif
|
|
/* Remember current time */
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
current_queued_int_socket->socket_values.tcp_control.last_packet_time = now;
|
|
|
|
current_queued_int_socket->socket_values.tcp_control.no_of_retries = 0;
|
|
|
|
/* Set message type to Retry for while loop */
|
|
msg_recv_client_ack.type = TCP_RETRY;
|
|
|
|
while (msg_recv_client_ack.type == TCP_RETRY) {
|
|
/* Send packet */
|
|
send_tcp(current_queued_int_socket, syn_ack_packet, temp_ipv6_header,
|
|
TCP_SYN_ACK, 0);
|
|
|
|
/* wait for ACK from Client */
|
|
msg_receive(&msg_recv_client_ack);
|
|
|
|
if (msg_recv_client_ack.type == TCP_TIMEOUT) {
|
|
/* Set status of internal socket back to TCP_LISTEN */
|
|
server_socket->socket_values.tcp_control.state = TCP_LISTEN;
|
|
memset(current_queued_int_socket, 0, sizeof(socket_internal_t));
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
tcp_hdr_t *tcp_header;
|
|
|
|
tcp_header = ((tcp_hdr_t *)(msg_recv_client_ack.content.ptr));
|
|
|
|
/* Check for consistency */
|
|
if (tcp_header->ack_nr != current_queued_socket->tcp_control.send_nxt) {
|
|
printf("TCP packets not consistent!\n");
|
|
}
|
|
|
|
/* Got ack, connection established, refresh local and foreign tcp socket
|
|
* status */
|
|
set_tcp_cb(¤t_queued_socket->tcp_control, tcp_header->seq_nr,
|
|
current_queued_socket->tcp_control.rcv_wnd, tcp_header->ack_nr,
|
|
tcp_header->ack_nr, tcp_header->window);
|
|
|
|
#ifdef TCP_HC
|
|
/* Copy TCP context information into new socket */
|
|
memset(&server_socket->socket_values.tcp_control.tcp_context, 0,
|
|
sizeof(tcp_hc_context_t));
|
|
#endif
|
|
|
|
/* Update connection status information */
|
|
current_queued_socket->tcp_control.state = TCP_ESTABLISHED;
|
|
|
|
/* Set status of internal socket back to TCP_LISTEN */
|
|
server_socket->socket_values.tcp_control.state = TCP_LISTEN;
|
|
|
|
/* send a reply to the TCP handler after processing every information from
|
|
* the TCP ACK packet */
|
|
msg_reply(&msg_recv_client_ack, &msg_send_client_ack);
|
|
|
|
/* Reset PID to an unlikely value */
|
|
current_queued_int_socket->recv_pid = 255;
|
|
|
|
/* Waiting for Clients ACK waiting period to time out */
|
|
vtimer_usleep(TCP_SYN_INITIAL_TIMEOUT / 2);
|
|
|
|
socket_base_print_sockets();
|
|
|
|
return current_queued_int_socket->socket_id;
|
|
}
|
|
|
|
int32_t tcp_send(int s, const void *buf, uint32_t len, int flags)
|
|
{
|
|
(void) flags;
|
|
|
|
/* Variables */
|
|
msg_t recv_msg;
|
|
int32_t sent_bytes = 0;
|
|
uint32_t total_sent_bytes = 0;
|
|
socket_internal_t *current_int_tcp_socket;
|
|
socket_t *current_tcp_socket;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
memset(send_buffer, 0, BUFFER_SIZE);
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
/* Check if socket exists and is TCP socket */
|
|
if (!tcp_socket_compliancy(s)) {
|
|
return -1;
|
|
}
|
|
|
|
current_int_tcp_socket = socket_base_get_socket(s);
|
|
current_tcp_socket = ¤t_int_tcp_socket->socket_values;
|
|
|
|
/* Check for TCP_ESTABLISHED STATE */
|
|
if (current_tcp_socket->tcp_control.state != TCP_ESTABLISHED) {
|
|
return -1;
|
|
}
|
|
|
|
/* Add thread PID */
|
|
current_int_tcp_socket->send_pid = thread_getpid();
|
|
|
|
recv_msg.type = UNDEFINED;
|
|
|
|
while (1) {
|
|
current_tcp_socket->tcp_control.no_of_retries = 0;
|
|
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = COMPRESSED_HEADER;
|
|
/* Remember TCP Context for possible TCP_RETRY */
|
|
tcp_hc_context_t saved_tcp_context;
|
|
memcpy(&saved_tcp_context, ¤t_tcp_socket->tcp_control.tcp_context,
|
|
sizeof(tcp_hc_context_t) - 1);
|
|
#endif
|
|
|
|
while (recv_msg.type != TCP_ACK) {
|
|
/* Add packet data */
|
|
if (current_tcp_socket->tcp_control.send_wnd >
|
|
current_tcp_socket->tcp_control.mss) {
|
|
/* Window size > Maximum Segment Size */
|
|
if ((len - total_sent_bytes) > current_tcp_socket->tcp_control.mss) {
|
|
memcpy(&send_buffer[IPV6_HDR_LEN + TCP_HDR_LEN], buf,
|
|
current_tcp_socket->tcp_control.mss);
|
|
sent_bytes = current_tcp_socket->tcp_control.mss;
|
|
total_sent_bytes += sent_bytes;
|
|
}
|
|
else {
|
|
memcpy(&send_buffer[IPV6_HDR_LEN + TCP_HDR_LEN],
|
|
(uint8_t *) buf + total_sent_bytes, len - total_sent_bytes);
|
|
sent_bytes = len - total_sent_bytes;
|
|
total_sent_bytes = len;
|
|
}
|
|
}
|
|
else {
|
|
/* Window size <= Maximum Segment Size */
|
|
if ((len - total_sent_bytes) > current_tcp_socket->tcp_control.send_wnd) {
|
|
memcpy(&send_buffer[IPV6_HDR_LEN + TCP_HDR_LEN], buf,
|
|
current_tcp_socket->tcp_control.send_wnd);
|
|
sent_bytes = current_tcp_socket->tcp_control.send_wnd;
|
|
total_sent_bytes += sent_bytes;
|
|
}
|
|
else {
|
|
memcpy(&send_buffer[IPV6_HDR_LEN + TCP_HDR_LEN],
|
|
(uint8_t *) buf + total_sent_bytes, len - total_sent_bytes);
|
|
sent_bytes = len - total_sent_bytes;
|
|
total_sent_bytes = len;
|
|
}
|
|
}
|
|
|
|
current_tcp_socket->tcp_control.send_nxt += sent_bytes;
|
|
current_tcp_socket->tcp_control.send_wnd -= sent_bytes;
|
|
|
|
if (send_tcp(current_int_tcp_socket, current_tcp_packet,
|
|
temp_ipv6_header, TCP_ACK, sent_bytes) < 0) {
|
|
/* Error while sending tcp data */
|
|
current_tcp_socket->tcp_control.send_nxt -= sent_bytes;
|
|
current_tcp_socket->tcp_control.send_wnd += sent_bytes;
|
|
#ifdef TCP_HC
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type =
|
|
COMPRESSED_HEADER;
|
|
#endif
|
|
printf("Error while sending, returning to application thread!\n");
|
|
return -1;
|
|
}
|
|
|
|
/* Remember current time */
|
|
vtimer_now(¤t_tcp_socket->tcp_control.last_packet_time);
|
|
socket_base_net_msg_receive(&recv_msg);
|
|
|
|
switch (recv_msg.type) {
|
|
case TCP_ACK: {
|
|
if (current_tcp_socket->tcp_control.no_of_retries == 0) {
|
|
calculate_rto(¤t_tcp_socket->tcp_control,
|
|
current_tcp_socket->tcp_control.last_packet_time);
|
|
}
|
|
|
|
tcp_hdr_t *tcp_header = ((tcp_hdr_t *)(recv_msg.content.ptr));
|
|
|
|
if ((current_tcp_socket->tcp_control.send_nxt ==
|
|
tcp_header->ack_nr) && (total_sent_bytes == len)) {
|
|
current_tcp_socket->tcp_control.send_una = tcp_header->ack_nr;
|
|
current_tcp_socket->tcp_control.send_nxt = tcp_header->ack_nr;
|
|
current_tcp_socket->tcp_control.send_wnd = tcp_header->window;
|
|
/* Got ACK for every sent byte */
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type =
|
|
COMPRESSED_HEADER;
|
|
#endif
|
|
return sent_bytes;
|
|
}
|
|
else if ((current_tcp_socket->tcp_control.send_nxt ==
|
|
tcp_header->ack_nr) && (total_sent_bytes != len)) {
|
|
current_tcp_socket->tcp_control.send_una = tcp_header->ack_nr;
|
|
current_tcp_socket->tcp_control.send_nxt = tcp_header->ack_nr;
|
|
current_tcp_socket->tcp_control.send_wnd = tcp_header->window;
|
|
/* Got ACK for every sent byte */
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type =
|
|
COMPRESSED_HEADER;
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
/* else {
|
|
* TODO: If window size > MSS, ACK was valid only for
|
|
* a few segments, handle retransmit of missing
|
|
* segments
|
|
* break;
|
|
* } */
|
|
break;
|
|
}
|
|
|
|
case TCP_RETRY: {
|
|
current_tcp_socket->tcp_control.send_nxt -= sent_bytes;
|
|
current_tcp_socket->tcp_control.send_wnd += sent_bytes;
|
|
total_sent_bytes -= sent_bytes;
|
|
#ifdef TCP_HC
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
$&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type =
|
|
MOSTLY_COMPRESSED_HEADER;
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
case TCP_TIMEOUT: {
|
|
current_tcp_socket->tcp_control.send_nxt -= sent_bytes;
|
|
current_tcp_socket->tcp_control.send_wnd += sent_bytes;
|
|
#ifdef TCP_HC
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type =
|
|
COMPRESSED_HEADER;
|
|
#endif
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return sent_bytes;
|
|
}
|
|
|
|
int tcp_accept(int s, sockaddr6_t *addr, uint32_t *addrlen)
|
|
{
|
|
(void) addr;
|
|
(void) addrlen;
|
|
|
|
socket_internal_t *server_socket = socket_base_get_socket(s);
|
|
|
|
if (tcp_socket_compliancy(s) && (server_socket->socket_values.tcp_control.state == TCP_LISTEN)) {
|
|
socket_internal_t *current_queued_socket =
|
|
get_waiting_connection_socket(s, NULL, NULL);
|
|
|
|
if (current_queued_socket != NULL) {
|
|
return handle_new_tcp_connection(current_queued_socket,
|
|
server_socket, thread_getpid());
|
|
}
|
|
else {
|
|
/* No waiting connections, waiting for message from TCP Layer */
|
|
msg_t msg_recv_client_syn;
|
|
msg_recv_client_syn.type = UNDEFINED;
|
|
|
|
while (msg_recv_client_syn.type != TCP_SYN) {
|
|
msg_receive(&msg_recv_client_syn);
|
|
}
|
|
|
|
current_queued_socket = get_waiting_connection_socket(s, NULL, NULL);
|
|
|
|
return handle_new_tcp_connection(current_queued_socket,
|
|
server_socket, thread_getpid());
|
|
}
|
|
}
|
|
else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int tcp_connect(int socket, sockaddr6_t *addr, uint32_t addrlen)
|
|
{
|
|
(void) addrlen;
|
|
|
|
/* Variables */
|
|
ipv6_addr_t src_addr;
|
|
socket_internal_t *current_int_tcp_socket;
|
|
socket_t *current_tcp_socket;
|
|
msg_t msg_from_server;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
/* Check if socket exists */
|
|
current_int_tcp_socket = socket_base_get_socket(socket);
|
|
|
|
if (current_int_tcp_socket == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
current_tcp_socket = ¤t_int_tcp_socket->socket_values;
|
|
|
|
current_int_tcp_socket->recv_pid = thread_getpid();
|
|
|
|
/* Local address information */
|
|
ipv6_net_if_get_best_src_addr(&src_addr, &addr->sin6_addr);
|
|
set_socket_address(¤t_tcp_socket->local_address, PF_INET6,
|
|
HTONS(socket_base_get_free_source_port(IPPROTO_TCP)), 0, &src_addr);
|
|
|
|
/* Foreign address information */
|
|
set_socket_address(¤t_tcp_socket->foreign_address, addr->sin6_family,
|
|
addr->sin6_port, addr->sin6_flowinfo, &addr->sin6_addr);
|
|
|
|
/* Fill lcoal TCP socket information */
|
|
srand(addr->sin6_port);
|
|
|
|
current_tcp_socket->tcp_control.rcv_irs = 0;
|
|
mutex_lock(&global_sequence_counter_mutex);
|
|
current_tcp_socket->tcp_control.send_iss = global_sequence_counter;
|
|
mutex_unlock(&global_sequence_counter_mutex);
|
|
current_tcp_socket->tcp_control.state = TCP_SYN_SENT;
|
|
|
|
#ifdef TCP_HC
|
|
/* Choosing random number Context ID */
|
|
mutex_lock(&global_context_counter_mutex);
|
|
current_tcp_socket->tcp_control.tcp_context.context_id = global_context_counter;
|
|
mutex_unlock(&global_context_counter_mutex);
|
|
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = FULL_HEADER;
|
|
|
|
/* Remember TCP Context for possible TCP_RETRY */
|
|
tcp_hc_context_t saved_tcp_context;
|
|
memcpy(&saved_tcp_context, ¤t_tcp_socket->tcp_control.tcp_context,
|
|
sizeof(tcp_hc_context_t));
|
|
#endif
|
|
|
|
set_tcp_cb(¤t_tcp_socket->tcp_control, 0, TRANSPORT_LAYER_SOCKET_STATIC_WINDOW,
|
|
current_tcp_socket->tcp_control.send_iss + 1,
|
|
current_tcp_socket->tcp_control.send_iss, 0);
|
|
|
|
/* Remember current time */
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
current_tcp_socket->tcp_control.last_packet_time = now;
|
|
current_tcp_socket->tcp_control.no_of_retries = 0;
|
|
|
|
msg_from_server.type = TCP_RETRY;
|
|
|
|
while (msg_from_server.type == TCP_RETRY) {
|
|
/* Send packet */
|
|
send_tcp(current_int_tcp_socket, current_tcp_packet, temp_ipv6_header,
|
|
TCP_SYN, 0);
|
|
|
|
/* wait for SYN ACK or RETRY */
|
|
msg_receive(&msg_from_server);
|
|
|
|
if (msg_from_server.type == TCP_TIMEOUT) {
|
|
#ifdef TCP_HC
|
|
/* We did not send anything successful so restore last context */
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
#endif
|
|
return -1;
|
|
}
|
|
|
|
#ifdef TCP_HC
|
|
else if (msg_from_server.type == TCP_RETRY) {
|
|
/* We retry sending a packet so set everything to last values again */
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
/* Read packet content */
|
|
tcp_hdr_t *tcp_header = ((tcp_hdr_t *)(msg_from_server.content.ptr));
|
|
|
|
/* Check for consistency */
|
|
if (tcp_header->ack_nr != current_tcp_socket->tcp_control.send_nxt) {
|
|
printf("TCP packets not consistent!\n");
|
|
}
|
|
|
|
/* Got SYN ACK from Server */
|
|
/* Refresh foreign TCP socket information */
|
|
if ((tcp_header->data_offset * 4 > TCP_HDR_LEN) &&
|
|
(*(((uint8_t *)tcp_header) + TCP_HDR_LEN) == TCP_MSS_OPTION)) {
|
|
current_tcp_socket->tcp_control.mss =
|
|
*((uint16_t *)(((uint8_t *)tcp_header) + TCP_HDR_LEN + 2));
|
|
}
|
|
else {
|
|
current_tcp_socket->tcp_control.mss = TRANSPORT_LAYER_SOCKET_STATIC_MSS;
|
|
}
|
|
|
|
current_tcp_socket->tcp_control.rcv_irs = tcp_header->seq_nr;
|
|
set_tcp_cb(¤t_tcp_socket->tcp_control, tcp_header->seq_nr + 1,
|
|
current_tcp_socket->tcp_control.rcv_wnd,
|
|
current_tcp_socket->tcp_control.send_una + 1,
|
|
tcp_header->ack_nr,
|
|
tcp_header->window);
|
|
|
|
msg_from_server.type = UNDEFINED;
|
|
|
|
/* Remember current time */
|
|
vtimer_now(&now);
|
|
current_tcp_socket->tcp_control.last_packet_time = now;
|
|
current_tcp_socket->tcp_control.no_of_retries = 0;
|
|
|
|
#ifdef TCP_HC
|
|
current_tcp_socket->tcp_control.tcp_context.hc_type = FULL_HEADER;
|
|
/* Remember TCP Context for possible TCP_RETRY */
|
|
memcpy(&saved_tcp_context, ¤t_tcp_socket->tcp_control.tcp_context,
|
|
sizeof(tcp_hc_context_t));
|
|
#endif
|
|
|
|
while (msg_from_server.type != TCP_RETRY) {
|
|
/* Send packet */
|
|
send_tcp(current_int_tcp_socket, current_tcp_packet, temp_ipv6_header,
|
|
TCP_ACK, 0);
|
|
|
|
msg_receive(&msg_from_server);
|
|
#ifdef TCP_HC
|
|
|
|
if (msg_from_server.type == TCP_SYN_ACK) {
|
|
/* TCP_SYN_ACK from server arrived again, copy old context and
|
|
* send TCP_ACK again */
|
|
memcpy(¤t_tcp_socket->tcp_control.tcp_context,
|
|
&saved_tcp_context, sizeof(tcp_hc_context_t));
|
|
}
|
|
else if (msg_from_server.type == TCP_RETRY) {
|
|
/* We waited for RTT, no TCP_SYN_ACK received, so we assume the
|
|
* TCP_ACK packet arrived safely */
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
current_tcp_socket->tcp_control.state = TCP_ESTABLISHED;
|
|
|
|
current_int_tcp_socket->recv_pid = 255;
|
|
|
|
socket_base_print_sockets();
|
|
return 0;
|
|
}
|
|
|
|
uint8_t read_from_socket(socket_internal_t *current_int_tcp_socket,
|
|
void *buf, int len)
|
|
{
|
|
if (len >= current_int_tcp_socket->tcp_input_buffer_end) {
|
|
mutex_lock(¤t_int_tcp_socket->tcp_buffer_mutex);
|
|
uint8_t read_bytes = current_int_tcp_socket->tcp_input_buffer_end;
|
|
memcpy(buf, current_int_tcp_socket->tcp_input_buffer,
|
|
current_int_tcp_socket->tcp_input_buffer_end);
|
|
current_int_tcp_socket->tcp_input_buffer_end = 0;
|
|
current_int_tcp_socket->socket_values.tcp_control.rcv_wnd += read_bytes;
|
|
mutex_unlock(¤t_int_tcp_socket->tcp_buffer_mutex);
|
|
return read_bytes;
|
|
}
|
|
else {
|
|
mutex_lock(¤t_int_tcp_socket->tcp_buffer_mutex);
|
|
memcpy(buf, current_int_tcp_socket->tcp_input_buffer, len);
|
|
memmove(current_int_tcp_socket->tcp_input_buffer,
|
|
(current_int_tcp_socket->tcp_input_buffer + len),
|
|
current_int_tcp_socket->tcp_input_buffer_end - len);
|
|
current_int_tcp_socket->tcp_input_buffer_end =
|
|
current_int_tcp_socket->tcp_input_buffer_end - len;
|
|
current_int_tcp_socket->socket_values.tcp_control.rcv_wnd += len;
|
|
mutex_unlock(¤t_int_tcp_socket->tcp_buffer_mutex);
|
|
return len;
|
|
}
|
|
}
|
|
|
|
int32_t tcp_recv(int s, void *buf, uint32_t len, int flags)
|
|
{
|
|
(void) flags;
|
|
|
|
/* Variables */
|
|
msg_t m_recv, m_send;
|
|
socket_internal_t *current_int_tcp_socket;
|
|
|
|
/* Check if socket exists */
|
|
if (!tcp_socket_compliancy(s)) {
|
|
printf("INFO: NO TCP SOCKET!\n");
|
|
return -1;
|
|
}
|
|
|
|
current_int_tcp_socket = socket_base_get_socket(s);
|
|
|
|
/* Setting Thread PID */
|
|
current_int_tcp_socket->recv_pid = thread_getpid();
|
|
|
|
if (current_int_tcp_socket->tcp_input_buffer_end > 0) {
|
|
return read_from_socket(current_int_tcp_socket, buf, len);
|
|
}
|
|
|
|
msg_receive(&m_recv);
|
|
|
|
if ((socket_base_exists_socket(s)) && (current_int_tcp_socket->tcp_input_buffer_end > 0)) {
|
|
uint8_t read_bytes = read_from_socket(current_int_tcp_socket, buf, len);
|
|
socket_base_net_msg_reply(&m_recv, &m_send, UNDEFINED);
|
|
return read_bytes;
|
|
}
|
|
|
|
/* Received FIN */
|
|
if (m_recv.type == CLOSE_CONN) {
|
|
/* Sent FIN_ACK, wait for ACK */
|
|
msg_receive(&m_recv);
|
|
/* Received ACK, return with closed socket!*/
|
|
return -1;
|
|
}
|
|
|
|
/* Received Last ACK (connection closed) or no data to read yet */
|
|
return -1;
|
|
}
|
|
|
|
int tcp_listen(int s, int backlog)
|
|
{
|
|
(void) backlog;
|
|
|
|
if (tcp_socket_compliancy(s) && socket_base_get_socket(s)->socket_values.tcp_control.state == TCP_CLOSED) {
|
|
socket_internal_t *current_socket = socket_base_get_socket(s);
|
|
current_socket->socket_values.tcp_control.state = TCP_LISTEN;
|
|
return 0;
|
|
}
|
|
else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int tcp_teardown(socket_internal_t *current_socket)
|
|
{
|
|
/* Variables */
|
|
msg_t m_recv;
|
|
uint8_t send_buffer[BUFFER_SIZE];
|
|
ipv6_hdr_t *temp_ipv6_header = ((ipv6_hdr_t *)(&send_buffer));
|
|
tcp_hdr_t *current_tcp_packet = ((tcp_hdr_t *)(&send_buffer[IPV6_HDR_LEN]));
|
|
|
|
/* Check for TCP_ESTABLISHED STATE */
|
|
if (current_socket->socket_values.tcp_control.state != TCP_ESTABLISHED) {
|
|
memset(current_socket, 0, sizeof(socket_internal_t));
|
|
return 0;
|
|
}
|
|
|
|
current_socket->send_pid = thread_getpid();
|
|
|
|
/* Refresh local TCP socket information */
|
|
current_socket->socket_values.tcp_control.state = TCP_FIN_WAIT_1;
|
|
#ifdef TCP_HC
|
|
current_socket->socket_values.tcp_control.tcp_context.hc_type =
|
|
COMPRESSED_HEADER;
|
|
#endif
|
|
|
|
send_tcp(current_socket, current_tcp_packet, temp_ipv6_header,
|
|
TCP_FIN_ACK, 0);
|
|
msg_receive(&m_recv);
|
|
memset(current_socket, 0, sizeof(socket_internal_t));
|
|
return 1;
|
|
}
|
|
|
|
int tcp_init_transport_layer(void)
|
|
{
|
|
printf("Initializing transport layer protocol: tcp\n");
|
|
timex_t now;
|
|
vtimer_now(&now);
|
|
srand(timex_uint64(now));
|
|
#ifdef TCP_HC
|
|
printf("TCP_HC enabled!\n");
|
|
global_context_counter = rand();
|
|
#endif
|
|
global_sequence_counter = rand();
|
|
|
|
int tcp_thread_pid = thread_create(tcp_stack_buffer, TCP_STACK_SIZE,
|
|
THREAD_PRIORITY_MAIN, CREATE_STACKTEST, tcp_packet_handler, NULL, "tcp_packet_handler");
|
|
|
|
if (tcp_thread_pid < 0) {
|
|
return -1;
|
|
}
|
|
|
|
ipv6_register_next_header_handler(IPV6_PROTO_NUM_TCP, tcp_thread_pid);
|
|
|
|
if (thread_create(tcp_timer_stack, TCP_TIMER_STACKSIZE, THREAD_PRIORITY_MAIN + 1,
|
|
CREATE_STACKTEST, tcp_general_timer, NULL, "tcp_general_timer") < 0) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|