/** * 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 * @author Cenk Gündoğan * @} */ #include #include #include #include #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 *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); 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; } } } } 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, 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; }