mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
710c7e6cf6
Currently, the tcp and udp implementations are bound to each other in a module called *destiny*. Thus, when using only one of them then the other one gets also compiled into the binary and initialized, which results in unnecessary RAM usage and workload for the CPU. The approach in this PR defines a common module named *socket_base*, which contains functions used by the posix layer. Compiled by it's own, those functions return negative error codes, to symbolize upper layers that they are not supported. When also including the modules *udp* or *tcp* respectively, functions from *socket_base* get overwritten with the correct functionality. Defining *udp* or *tcp* in a Makefile also includes *socket_base*. Defining *pnet* in a Makefile also includes *socket_base*.
1447 lines
52 KiB
C
1447 lines
52 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 tcp.c
|
|
* @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 printArrayRange_tcp(uint8_t *udp_header, uint16_t len)
|
|
{
|
|
int i = 0;
|
|
printf("-------------MEMORY-------------\n");
|
|
|
|
for (i = 0; i < len; i++) {
|
|
printf("%#x ", *(udp_header + i));
|
|
}
|
|
|
|
printf("-------------MEMORY-------------\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);
|
|
#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 *current_socket, *listening_socket = socket_base_get_socket(socket);
|
|
|
|
for (i = 1; i < MAX_SOCKETS + 1; i++) {
|
|
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;
|
|
uint8_t target_pid;
|
|
|
|
if (tcp_socket->socket_values.tcp_control.state == TCP_LAST_ACK) {
|
|
target_pid = tcp_socket->recv_pid;
|
|
memset(tcp_socket, 0, sizeof(socket_internal_t));
|
|
msg_send(&m_send_tcp, target_pid, 0);
|
|
return;
|
|
}
|
|
else if (tcp_socket->socket_values.tcp_control.state == TCP_CLOSING) {
|
|
msg_send(&m_send_tcp, tcp_socket->recv_pid, 0);
|
|
msg_send(&m_send_tcp, tcp_socket->send_pid, 0);
|
|
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_send(&m_send, tcp_socket->send_pid, 0);
|
|
msg_send(&m_send, tcp_socket->recv_pid, 0);
|
|
}
|
|
|
|
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)
|
|
{
|
|
uint8_t read_bytes = 0;
|
|
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) {
|
|
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;
|
|
ipv6_hdr_t *ipv6_header;
|
|
tcp_hdr_t *tcp_header;
|
|
uint8_t *payload;
|
|
socket_internal_t *tcp_socket = NULL;
|
|
uint16_t chksum;
|
|
|
|
while (1) {
|
|
msg_receive(&m_recv_ip);
|
|
|
|
ipv6_header = ((ipv6_hdr_t *)m_recv_ip.content.ptr);
|
|
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
|
|
chksum = tcp_csum(ipv6_header, tcp_header);
|
|
|
|
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 */
|
|
uint8_t tcp_flags = tcp_header->reserved_flags;
|
|
|
|
switch (tcp_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);
|
|
printArrayRange(((uint8_t *)ipv6_header), IPV6_HDR_LEN +
|
|
NTOHS(ipv6_header->length), "Incoming");
|
|
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;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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 */
|
|
uint8_t read_bytes;
|
|
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)) {
|
|
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,
|
|
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, PRIORITY_MAIN + 1,
|
|
CREATE_STACKTEST, tcp_general_timer, NULL, "tcp_general_timer") < 0) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|