2011-10-13 04:31:07 +02:00
|
|
|
/*
|
|
|
|
* tcp.c
|
|
|
|
*
|
|
|
|
* Created on: 29.09.2011
|
|
|
|
* Author: Oliver
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <thread.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "tcp.h"
|
|
|
|
#include "in.h"
|
|
|
|
#include "socket.h"
|
|
|
|
#include "sys/net/net_help/net_help.h"
|
|
|
|
|
2011-11-02 02:55:38 +01:00
|
|
|
void printTCPHeader(tcp_hdr_t *tcp_header)
|
2011-11-01 05:47:28 +01:00
|
|
|
{
|
|
|
|
printf("\nBEGIN: TCP HEADER\n");
|
|
|
|
printf("ack_nr: %lu\n", tcp_header->ack_nr);
|
|
|
|
printf("checksum: %i\n", tcp_header->checksum);
|
|
|
|
printf("dataOffset_reserved: %i\n", tcp_header->dataOffset_reserved);
|
|
|
|
printf("dst_port: %i\n", tcp_header->dst_port);
|
|
|
|
printf("reserved_flags: %i\n", tcp_header->reserved_flags);
|
|
|
|
printf("seq_nr: %lu\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");
|
|
|
|
}
|
2011-10-13 04:31:07 +02:00
|
|
|
|
|
|
|
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");
|
|
|
|
}
|
|
|
|
|
2011-10-28 04:37:12 +02:00
|
|
|
uint16_t tcp_csum(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header)
|
|
|
|
{
|
2011-10-13 04:31:07 +02:00
|
|
|
uint16_t sum;
|
|
|
|
uint16_t len = ipv6_header->length;
|
|
|
|
|
|
|
|
sum = len + IPPROTO_TCP;
|
|
|
|
sum = csum(sum, (uint8_t *)&ipv6_header->srcaddr, 2 * sizeof(ipv6_addr_t));
|
2011-11-01 05:47:28 +01:00
|
|
|
sum = csum(sum, (uint8_t *)tcp_header, len);
|
2011-10-13 04:31:07 +02:00
|
|
|
return (sum == 0) ? 0xffff : HTONS(sum);
|
2011-10-28 04:37:12 +02:00
|
|
|
}
|
2011-10-13 04:31:07 +02:00
|
|
|
|
2011-11-08 02:14:58 +01:00
|
|
|
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 (getWaitingConnectionSocket(tcp_socket->socket_id)->local_tcp_status.state == SYN_RCVD)
|
|
|
|
{
|
|
|
|
msg_send_receive(&m_send_tcp, &m_recv_tcp, tcp_socket->pid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void handle_tcp_rst_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header, socket_internal_t *tcp_socket)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
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->in_socket.local_tcp_status.state == LISTEN)
|
|
|
|
{
|
|
|
|
socket_t *new_socket = new_tcp_queued_socket(ipv6_header, tcp_header, tcp_socket);
|
|
|
|
if (new_socket != NULL)
|
|
|
|
{
|
|
|
|
// notify socket function accept(..) that a new connection request has arrived
|
|
|
|
// No need to wait for an answer because the server accept() function isnt reading from anything other than the queued sockets
|
|
|
|
msg_send(&m_send_tcp, tcp_socket->pid, 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Dropped TCP SYN Message because an error occured while requesting a new queued socket!\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Droppec TCP SYN Message because socket was not in state LISTEN!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void handle_tcp_syn_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->in_socket.local_tcp_status.state == SYN_SENT)
|
|
|
|
{
|
|
|
|
msg_send_receive(&m_send_tcp, &m_recv_tcp, tcp_socket->pid);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Socket not in state 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 handle_tcp_fin_ack_packet(ipv6_hdr_t *ipv6_header, tcp_hdr_t *tcp_header, socket_internal_t *tcp_socket)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-10-13 04:31:07 +02:00
|
|
|
void tcp_packet_handler (void)
|
|
|
|
{
|
2011-11-08 02:14:58 +01:00
|
|
|
msg_t m_recv_ip, m_send_ip;
|
2011-11-01 21:04:28 +01:00
|
|
|
ipv6_hdr_t *ipv6_header;
|
|
|
|
tcp_hdr_t *tcp_header;
|
2011-10-13 04:31:07 +02:00
|
|
|
uint8_t *payload;
|
|
|
|
socket_internal_t *tcp_socket = NULL;
|
|
|
|
uint16_t chksum;
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
msg_receive(&m_recv_ip);
|
2011-11-01 21:04:28 +01:00
|
|
|
ipv6_header = ((ipv6_hdr_t*)&buffer_tcp);
|
|
|
|
tcp_header = ((tcp_hdr_t*)(&buffer_tcp[IPV6_HDR_LEN]));
|
2011-11-01 05:47:28 +01:00
|
|
|
payload = &buffer_tcp[IPV6_HDR_LEN+TCP_HDR_LEN];
|
2011-11-08 02:14:58 +01:00
|
|
|
//printTCPHeader(tcp_header);
|
2011-10-13 04:31:07 +02:00
|
|
|
chksum = tcp_csum(ipv6_header, tcp_header);
|
|
|
|
printf("Checksum is %x!\n", chksum);
|
|
|
|
|
2011-10-28 04:37:12 +02:00
|
|
|
tcp_socket = get_tcp_socket(ipv6_header, tcp_header);
|
2011-10-13 04:31:07 +02:00
|
|
|
|
2011-11-08 02:14:58 +01:00
|
|
|
print_tcp_status(INC_PACKET, ipv6_header, tcp_header);
|
|
|
|
|
2011-10-28 04:37:12 +02:00
|
|
|
if ((chksum == 0xffff) && (tcp_socket != NULL))
|
2011-10-13 04:31:07 +02:00
|
|
|
{
|
2011-10-28 04:37:12 +02:00
|
|
|
// Remove reserved bits from tcp flags field
|
2011-11-01 05:47:28 +01:00
|
|
|
uint8_t tcp_flags = tcp_header->reserved_flags & REMOVE_RESERVED;
|
2011-10-13 04:31:07 +02:00
|
|
|
|
2011-10-28 04:37:12 +02:00
|
|
|
//TODO: URG Flag and PSH flag are currently being ignored
|
|
|
|
switch (tcp_flags)
|
2011-10-13 04:31:07 +02:00
|
|
|
{
|
2011-10-28 04:37:12 +02:00
|
|
|
case TCP_ACK:
|
|
|
|
{
|
|
|
|
// only ACK Bit set
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TCP_RST:
|
|
|
|
{
|
2011-11-01 05:47:28 +01:00
|
|
|
printf("RST Bit set!\n");
|
2011-10-28 04:37:12 +02:00
|
|
|
// only RST Bit set
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_rst_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TCP_SYN:
|
|
|
|
{
|
|
|
|
// only SYN Bit set, look for matching, listening socket and request new queued socket
|
2011-11-01 05:47:28 +01:00
|
|
|
printf("SYN Bit set!\n");
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_syn_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TCP_SYN_ACK:
|
2011-10-13 04:31:07 +02:00
|
|
|
{
|
2011-10-28 04:37:12 +02:00
|
|
|
// only SYN and ACK Bit set, complete three way handshake when socket in state SYN_SENT
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_syn_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
2011-10-13 04:31:07 +02:00
|
|
|
}
|
2011-10-28 04:37:12 +02:00
|
|
|
case TCP_FIN:
|
2011-10-13 04:31:07 +02:00
|
|
|
{
|
2011-11-01 05:47:28 +01:00
|
|
|
printf("FIN Bit set!\n");
|
2011-10-28 04:37:12 +02:00
|
|
|
// only FIN Bit set
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_fin_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TCP_FIN_ACK:
|
|
|
|
{
|
2011-11-01 05:47:28 +01:00
|
|
|
printf("FIN ACK Bit set!\n");
|
2011-10-28 04:37:12 +02:00
|
|
|
// only FIN and ACK Bit set
|
2011-11-08 02:14:58 +01:00
|
|
|
handle_tcp_fin_ack_packet(ipv6_header, tcp_header, tcp_socket);
|
2011-10-28 04:37:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
// TODO: any other case
|
2011-11-01 05:47:28 +01:00
|
|
|
printf("DEFAULT!\n");
|
2011-10-13 04:31:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Wrong checksum (%x) or no corresponding socket found!\n", chksum);
|
|
|
|
}
|
|
|
|
|
|
|
|
tcp_socket = NULL;
|
|
|
|
msg_reply(&m_recv_ip, &m_send_ip);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|