U0 TCPGlobalsInit()
{
    tcp_globals.bound_socket_tree = NULL;
    tcp_globals.next_source_port  = 0;
}

Bool IsTCPStateSync(CTCPSocket *tcp_socket)
{ 
    switch (tcp_socket->state)
    {
        case TCP_STATE_ESTABLISHED:
        case TCP_STATE_FIN_WAIT1:
        case TCP_STATE_FIN_WAIT2:
        case TCP_STATE_CLOSE_WAIT:
        case TCP_STATE_CLOSING:
        case TCP_STATE_LAST_ACK:
        case TCP_STATE_TIME_WAIT:
            return TRUE;
        default:
            return FALSE;
    }
}

// TODO: Merge IPV4Checksum, TCPChecksumPartial, and TCPChecksumFinal into 1.
//       All use same alg.
U16 TCPChecksumPartial(U32 sum, U8 *header, I64 length)
{ // same algorithm as IPV4Checksum, except sum is arg. First half.
  //todo. make names clearer, and better comments.
    I64  nleft = length;
    U16 *w = header;

    while (nleft > 1)
    {
        sum += *w++;
        nleft -= 2;
    }

    return sum;
}

U16 TCPChecksumFinal(U32 sum, U8 *header, I64 length)
{ // same algorithm as IPV4Checksum, except sum is arg. Both halves.
  //todo. make names clearer, and better comments.
    I64  nleft = length;
    U16 *w = header;

    while (nleft > 1)
    {
        sum += *w++;
        nleft -= 2;
    }

    // "mop up an odd byte, if necessary"
    if (nleft == 1)
    {
        sum += *w & 0x00FF;
    }

    // "add back carry outs from top 16 bits to low 16 bits"
    sum = sum >> 16 + sum & 0xFFFF; // "add hi 16 to low 16"
    sum += sum >> 16; // add carry
    return ~sum & 0xFFFF;

}

I64 TCPPacketAllocate(U8 **frame_out,
                      U32  source_ip,
                      U16  source_port,
                      U32  destination_ip,
                      U16  destination_port,
                      U32  seq_num,
                      U32  ack_num,
                      U8   flags,
                      I64  length)
{
    U8          *tcp_frame;
    I64          de_index;
    CTCPHeader  *header;

    de_index = IPV4PacketAllocate(&tcp_frame,
                                  IP_PROTOCOL_TCP,
                                  source_ip,
                                  destination_ip,
                                  sizeof(CTCPHeader) + length);
    if (de_index < 0)
    {
        NetLog("TCP PACKET ALLOCATE: Ethernet Frame Allocate failed.");
        return de_index;
    }

    header = tcp_frame;

    header->source_port         = EndianU16(source_port);
    header->destination_port    = EndianU16(destination_port);

    NetDebug("TCP PACKET ALLOCATE: Outputting header port src/dest 0x%0X/0x%0X:",
                header->source_port, header->destination_port);

    header->seq_num             = EndianU32(seq_num);
    header->ack_num             = EndianU32(ack_num);
    header->data_offset         = (sizeof(CTCPHeader) / 4) << 4; // ???
    header->flags               = flags;
    header->window_size         = EndianU16(TCP_WINDOW_SIZE / 2);// TODO: What is window size supposed to be ?
    header->checksum            = 0;
    header->urgent_pointer      = 0;

    *frame_out = tcp_frame + sizeof(CTCPHeader);

    return de_index;
}

U0 TCPPacketFinish(I64            de_index,
                   U32            source_ip,
                   U32            destination_ip,
                   U8            *frame,
                   I64            length,
                   CTCPAckQueue **send_buffer_out=NULL)
{
    CTCPHeader       *header        = frame - sizeof(CTCPHeader);
    CTCPPseudoHeader *pseudoheader  = CAlloc(sizeof(CTCPPseudoHeader));
    U32               checksum;
    CTCPAckQueue     *send_buffer;

    pseudoheader->source_address        = EndianU32(source_ip);
    pseudoheader->destination_address   = EndianU32(destination_ip);
    pseudoheader->zeros                 = 0;
    pseudoheader->protocol              = IP_PROTOCOL_TCP;
    pseudoheader->tcp_length            = EndianU16(sizeof(CTCPHeader) + length);

    checksum = TCPChecksumPartial(0, pseudoheader, sizeof(CTCPPseudoHeader));
    header->checksum = TCPChecksumFinal(checksum, header, sizeof(CTCPHeader) + length);

    if (send_buffer_out)
    {
        send_buffer = CAlloc(sizeof(CTCPAckQueue));

        send_buffer->time_sent          = tS;
        send_buffer->retries            = 0;
        send_buffer->start_seq_num      = EndianU32(header->seq_num);
        send_buffer->end_seq_num        = 0; // must set this upstream
        send_buffer->tcp_packet_size    = length;
        send_buffer->tcp_packet         = CAlloc(sizeof(CTCPHeader) + length);

        MemCopy(send_buffer->tcp_packet, frame, sizeof(CTCPHeader) + length);

        *send_buffer_out = send_buffer; // set output param as new send buffer
    }

    IPV4PacketFinish(de_index);
}

I64 TCPSend(U32 source_ip,
            U16 source_port,
            U32 destination_ip,
            U16 destination_port,
            U32 seq_num,
            U32 ack_num,
            U8  flags)
{ // send raw unsocketed TCP packet, flags and/or data. Does not store into ACK Queue (Send Buffer).

    U8 *payload_frame;
    I64 de_index = TCPPacketAllocate(&payload_frame,
                                     source_ip,
                                     source_port,
                                     destination_ip,
                                     destination_port,
                                     seq_num,
                                     ack_num,
                                     flags,
                                     0);
    if (de_index < 0)
    {
        NetErr("TCP SEND: TCP Packet Allocate failed.");
        return de_index;
    }

    TCPPacketFinish(de_index, source_ip, destination_ip, payload_frame, 0, NULL);

    return 0;
}

I64 TCPSendFlags(CTCPSocket *tcp_socket, U8 flags)
{ // send TCP packet, flags only. Stores into ACK Queue (Send Buffer) if flags SYN or FIN.

    U8                      *payload_frame;
    CSocketAddressStorage   *dest   = &tcp_socket->destination_address;
    CSocketAddressStorage   *source = &tcp_socket->source_address;
    CSocketAddressIPV4      *dest_ipv4;
    CSocketAddressIPV6      *dest_ipv6;
    CSocketAddressIPV4      *source_ipv4;
    CSocketAddressIPV6      *source_ipv6;
    I64                      de_index;
    CTCPAckQueue            *send_buffer     = NULL;
    CTCPAckQueue           **send_buffer_ptr = NULL;

    NetDebug("TCP SEND FLAGS: Attempting Send using param TCP socket");

    switch (dest->family)
    {
        case AF_INET:
            dest_ipv4 = dest;

            switch (source->family)
            {
                case AF_INET:
                    source_ipv4 = source;

                    NetDebug("TCP SEND FLAGS: Trying TCP Packet Allocate... port src/dest 0x%0X/0x%0X",
                                source_ipv4->port, dest_ipv4->port);

                    de_index = TCPPacketAllocate(&payload_frame,
                                                 source_ipv4->address.address,
                                                 EndianU16(source_ipv4->port),
                                                 dest_ipv4->address.address,
                                                 EndianU16(dest_ipv4->port),
                                                 tcp_socket->next_send_seq_num,
                                                 tcp_socket->next_recv_seq_num,
                                                 flags,
                                                 0);
                    if (de_index < 0)
                    {
                        NetErr("TCP SEND FLAGS: TCP Packet Allocate failed.");
                        return de_index;
                    }

                    break;
                case AF_INET6:
                    source_ipv6 = source;
                    NetErr("TCP SEND FLAGS: FIXME, IPV6 not implemented yet");
                    return -1;
                case AF_UNSPEC:
                    NetErr("TCP SEND FLAGS: Error, TCP Send from AF_UNSPEC\n");
                    return -1;
            }

            break;
        case AF_INET6:
            dest_ipv6 = dest;
            NetErr("TCP SEND FLAGS: FIXME, IPV6 not implemented yet");
            return -1;
        case AF_UNSPEC:
            NetErr("TCP SEND FLAGS: Error, TCP Send to AF_UNSPEC\n");
            return -1;
    }

    // at this point, a TCP packet has been allocated.
    // TODO: Is there a better way to manage links across different address types?

    if (Bt(&flags, TCPf_SYN))
        tcp_socket->next_send_seq_num++;

    if (Bt(&flags, TCPf_FIN))
        tcp_socket->next_send_seq_num++;

    if (Bt(&flags, TCPf_FIN) || Bt(&flags, TCPf_SYN))
    {
        send_buffer_ptr = &send_buffer;
    }


    switch (dest->family)
    {
        case AF_INET:
            dest_ipv4 = dest;

            switch (source->family)
            {
                case AF_INET:
                    source_ipv4 = source;

                    TCPPacketFinish(de_index,
                                    source_ipv4->address.address,
                                    dest_ipv4->address.address,
                                    payload_frame,
                                    0,
                                    send_buffer_ptr);

                    if (send_buffer)
                    { // modifying ptr should affect our send_buffer variable
                        send_buffer->end_seq_num = tcp_socket->next_send_seq_num;

                        // append it to the socket, ACK Queue needs to be ready (head)...
                        QueueInsertRev(send_buffer, tcp_socket->ack_queue);

                    }

                    break;
                case AF_INET6:
                    source_ipv6 = source;
                    NetErr("TCP SEND FLAGS: FIXME, IPV6 not implemented yet");
                    return -1;
                case AF_UNSPEC:
                    NetErr("TCP SEND FLAGS: Error, TCP Send from AF_UNSPEC\n");
                    return -1;
            }

            break;
        case AF_INET6:
            dest_ipv6 = dest;
            NetErr("TCP SEND FLAGS: FIXME, IPV6 not implemented yet");
            return -1;
        case AF_UNSPEC:
            NetErr("TCP SEND FLAGS: Error, TCP Send to AF_UNSPEC\n");
            return -1;
    }
}

I64 TCPSendData(CTCPSocket *tcp_socket, U8 flags, U8 *data, I64 length)
{ // send TCP packet, flags and data. Stores into ACK Queue (Send Buffer).

    U8                      *payload_frame;
    CSocketAddressStorage   *dest   = &tcp_socket->destination_address;
    CSocketAddressStorage   *source = &tcp_socket->source_address;
    CSocketAddressIPV4      *dest_ipv4;
    CSocketAddressIPV6      *dest_ipv6;
    CSocketAddressIPV4      *source_ipv4;
    CSocketAddressIPV6      *source_ipv6;
    I64                      de_index;
    CTCPAckQueue            *send_buffer;

    NetDebug("TCP SEND DATA: Attempting Send using param TCP socket");

    switch (dest->family)
    {
        case AF_INET:
            dest_ipv4 = dest;

            switch (source->family)
            {
                case AF_INET:
                    source_ipv4 = source;

                    NetDebug("TCP SEND DATA: Trying TCP Packet Allocate... port src/dest 0x%0X/0x%0X",
                                source_ipv4->port, dest_ipv4->port);

                    de_index = TCPPacketAllocate(&payload_frame,
                                                 source_ipv4->address.address,
                                                 EndianU16(source_ipv4->port),
                                                 dest_ipv4->address.address,
                                                 EndianU16(dest_ipv4->port),
                                                 tcp_socket->next_send_seq_num,
                                                 tcp_socket->next_recv_seq_num,
                                                 flags,
                                                 length);
                    if (de_index < 0)
                    {
                        NetLog("TCP SEND DATA: TCP Packet Allocate failed.");
                        return de_index;
                    }

                    break;
                case AF_INET6:
                    source_ipv6 = source;
                    NetErr("TCP SEND DATA: FIXME, IPV6 not implemented yet");
                    return -1;
                case AF_UNSPEC:
                    NetErr("TCP SEND DATA: Error, TCP Send from AF_UNSPEC\n");
                    return -1;
            }

            break;
        case AF_INET6:
            dest_ipv6 = dest;
            NetErr("TCP SEND DATA: FIXME, IPV6 not implemented yet");
            return -1;
        case AF_UNSPEC:
            NetErr("TCP SEND DATA: Error, TCP Send to AF_UNSPEC\n");
            return -1;
    }

    // at this point, a TCP packet has been allocated.
    // TODO: Is there a better way to manage links across different address types?

    if (length)
        MemCopy(payload_frame, data, length);

    if (Bt(&flags, TCPf_SYN))
        tcp_socket->next_send_seq_num++;

    tcp_socket->next_send_seq_num += length;

    if (Bt(&flags, TCPf_FIN))
        tcp_socket->next_send_seq_num++;

    switch (dest->family)
    {
        case AF_INET:
            dest_ipv4 = dest;

            switch (source->family)
            {
                case AF_INET:
                    source_ipv4 = source;

                    TCPPacketFinish(de_index,
                                    source_ipv4->address.address,
                                    dest_ipv4->address.address,
                                    payload_frame,
                                    length,
                                    &send_buffer);

                    send_buffer->end_seq_num = tcp_socket->next_send_seq_num;

                    // append it to the socket, ACK Queue needs to be ready (head)...
                    QueueInsertRev(send_buffer, tcp_socket->ack_queue);

                    break;
                case AF_INET6:
                    source_ipv6 = source;
                    NetErr("TCP SEND DATA: FIXME, IPV6 not implemented yet");
                    return -1;
                case AF_UNSPEC:
                    NetErr("TCP SEND DATA: Error, TCP Send from AF_UNSPEC\n");
                    return -1;
            }

            break;
        case AF_INET6:
            dest_ipv6 = dest;
            NetErr("TCP SEND DATA: FIXME, IPV6 not implemented yet");
            return -1;
        case AF_UNSPEC:
            NetErr("TCP SEND DATA: Error, TCP Send to AF_UNSPEC\n");
            return -1;
    }
}

I64 TCPPacketParse(CTCPHeader **header_out,
                   U8         **data_out,
                   I64         *length_out,
                   CIPV4Packet *packet)
{
    CTCPHeader       *header        = packet->data;
    I64               header_length = (header->data_offset >> 4) * 4; // ?? why
//  U32               checksum, sum;
//  CTCPPseudoHeader *pseudoheader  = CAlloc(sizeof(CTCPPseudoHeader));

    // TODO: Validate packet->length !

    // Verify Checksum:
/*  NetWarn("DEBUG: TCPPACKETPARSE:     0x%X = checksum from packet, verifying...", EndianU16(header->checksum));
    pseudoheader->source_address        = EndianU32(packet->source_ip_address);
    pseudoheader->destination_address   = EndianU32(packet->destination_ip_address);
    pseudoheader->zeros                 = 0;
    pseudoheader->protocol              = IP_PROTOCOL_TCP;
    pseudoheader->tcp_length            = EndianU16(packet->length);

    checksum = TCPChecksumPartial(0, pseudoheader, sizeof(CTCPPseudoHeader));
    NetWarn("DEBUG: TCPPACKETPARSE:     0x%X = partial checksum", checksum);

    Free(pseudoheader); // No longer needed, discard

    sum = TCPChecksumFinal(checksum, header, packet->length);

    if (sum != EndianU16(header->checksum))
    {
        NetErr("DEBUG: TCPPACKETPARSE:      0x%X = NOT MATCH", sum);
        //return -1; // is checksum not being 0 in received header going to mess this up?..
    }
    // Checksum verified.
*/

    *header_out = header;
    *data_out   = packet->data   + header_length;
    *length_out = packet->length - header_length;

    return 0;
}

U0 TCPAcknowledgePacket(CTCPSocket *tcp_socket, U32 segment_ack)
{
    F64              time = tS;
    F64              rtt;
    CTCPAckQueue    *send_buffer = tcp_socket->ack_queue->next;
    CTCPAckQueue    *send_buffer_temp;
    I64              segment_ack_relative;
    I64              send_next_relative;

    NetDebug("TCP ACKNOWLEDGE PACKET: Looking to see if there are any send buffers");

    while (send_buffer != tcp_socket->ack_queue)
    {
        segment_ack_relative    = (segment_ack - send_buffer->end_seq_num) & 0xFFFFFFFF;
        send_next_relative      = (tcp_socket->next_send_seq_num - send_buffer->end_seq_num) & 0xFFFFFFFF;
        // ???

        if (segment_ack_relative <= send_next_relative)
        {
            NetDebug("TCP ACKNOWLEDGE PACKET: Saw send buffer entry with ACK <= next relative send ACK, removing.");
            rtt = time - send_buffer->time_sent; // Round-Trip Time

            tcp_socket->srtt = (tcp_socket->srtt * TCP_SRTT_ALPHA) + ((1.0 - TCP_SRTT_ALPHA) * rtt);
            // Smoothed Round-Trip Time

            QueueRemove(send_buffer);
            Free(send_buffer->tcp_packet);

            send_buffer_temp = send_buffer->next; // QueueRemove doesn't change removed queue's links
            Free(send_buffer);

            send_buffer = send_buffer_temp;
            
        }
        else
            break;
    }
}

U0 TCPCheckACKQueue(CTCPSocket *tcp_socket)
{
    F64                      time           = tS;
    F64                      rto            = TCP_RTO_BETA * tcp_socket->srtt; // Retransmission Timeout, Smoothed Round-Trip Time
    CTCPAckQueue            *send_buffer    = tcp_socket->ack_queue->next;
    CTCPAckQueue            *first_buffer   = send_buffer;
    U8                      *tcp_frame;
    I64                      de_index;
    CSocketAddressStorage   *dest           = &tcp_socket->destination_address;
    CSocketAddressStorage   *source         = &tcp_socket->source_address;
    CSocketAddressIPV4      *dest_ipv4;
    CSocketAddressIPV6      *dest_ipv6;
    CSocketAddressIPV4      *source_ipv4;
    CSocketAddressIPV6      *source_ipv6;


    if (send_buffer == tcp_socket->ack_queue)
    {
//      NetWarn("TCP CHECK ACK QUEUE: ACK Queue empty, bailing early.");
        return;
    }

    if (rto < TCP_RTO_MIN)
        rto = TCP_RTO_MIN;
    if (rto < TCP_RTO_MAX)
        rto = TCP_RTO_MAX;

    do
    {
        if (time > send_buffer->time_sent + rto)
        {
            NetWarn("TCP CHECK ACK QUEUE: Retransmitting TCP packet in ACK Queue.");

            switch (dest->family)
            {
                case AF_INET:
                    dest_ipv4 = dest;

                    switch (source->family)
                    {
                        case AF_INET:
                            source_ipv4 = source;

                            de_index = IPV4PacketAllocate(&tcp_frame,
                                                          IP_PROTOCOL_TCP,
                                                          source_ipv4->address.address,
                                                          dest_ipv4->address.address,
                                                          send_buffer->tcp_packet_size);

                            if (de_index < 0)
                            {
                                NetErr("TCP CHECK ACK QUEUE: IPV4 Packet Allocate failed.");
                                return;// de_index;
                            }

                            MemCopy(&tcp_frame, send_buffer->tcp_packet, send_buffer->tcp_packet_size);

                            IPV4PacketFinish(de_index);

                            break;

                        case AF_INET6:
                            source_ipv6 = source;
                            NetErr("TCP CHECK ACK QUEUE: FIXME, IPV6 not implemented yet");
                            return;// -1;
                        case AF_UNSPEC:
                            NetErr("TCP CHECK ACK QUEUE: Error, TCP Send from AF_UNSPEC\n");
                            return;// -1;
                    }

                    break;
                case AF_INET6:
                    dest_ipv6 = dest;
                    NetErr("TCP CHECK ACK QUEUE: FIXME, IPV6 not implemented yet");
                    return;// -1;
                case AF_UNSPEC:
                    NetErr("TCP CHECK ACK QUEUE: Error, TCP Send to AF_UNSPEC\n");
                    return;// -1;
            }

            send_buffer->time_sent = tS;

            // If it gets here, retransmit was successful.

            // Move to the END OF THE CHAIN (BACK OF THE QUEUE. SO, REQUEUE.)
            QueueRemove(send_buffer);
            QueueInsertRev(send_buffer, tcp_socket->ack_queue);

            send_buffer = tcp_socket->ack_queue->next;

        }
        else
            break;
    }
    while (send_buffer != first_buffer);
}

CTCPSocket TCPSocket(U16 domain=AF_UNSPEC)
{
    U16                  type           = SOCKET_STREAM;
    CTCPSocket          *tcp_socket     = CAlloc(sizeof(CTCPSocket));
    CTCPAckQueue        *ack_queue      = CAlloc(sizeof(CTCPAckQueue));
    CTCPAcceptQueue     *accept_queue   = CAlloc(sizeof(CTCPAcceptQueue));
    CSocketAddressIPV4  *ipv4_address;
    CSocketAddressIPV6  *ipv6_address;

    tcp_socket->socket  = Socket(domain, type);
    tcp_socket->state   = TCP_STATE_CLOSED;

    tcp_socket->source_address.family = domain; // INET, INET6, or unspecified

    switch (domain)
    {
        case AF_INET:
            ipv4_address = &tcp_socket->source_address;

            ipv4_address->address.address = IPV4AddressGet;
            // do we need to set local port? does that only make sense to do later?
            // CAlloc will leave it at 0

            break;

        case AF_INET6:
            ipv6_address = &tcp_socket->source_address;
            NetErr("TCP SOCKET: FIXME, IPV6 not implemented yet");
            return NULL;

        case AF_UNSPEC:
            NetErr("TCP SOCKET: Error TCP Socket to AF_UNSPEC");
            return NULL;
    }

    QueueInit(ack_queue);  // init send buffer queue head
    tcp_socket->ack_queue = ack_queue;

    QueueInit(accept_queue); // init pending connection queue
    tcp_socket->accept_queue = accept_queue;

    tcp_socket->receive_buffer_size = TCP_WINDOW_SIZE;
    tcp_socket->receive_buffer = CAlloc(TCP_WINDOW_SIZE);

    tcp_socket->max_segment_size = TCP_MSS;

    return tcp_socket;
}

I64 TCPSocketBind(CTCPSocket *tcp_socket, CSocketAddressStorage *address)
{ // Binds a TCP socket to address, which contains the local port and remote address to use.
    CTCPTreeNode       *head = tcp_globals.bound_socket_tree;
    CTCPTreeNode       *temp_node;
    CSocketAddressIPV4 *ipv4_address;
    CSocketAddressIPV4 *ipv4_destination;
    CSocketAddressIPV4 *ipv4_source;
    CSocketAddressIPV6 *ipv6_address;
    CSocketAddressIPV6 *ipv6_destination;
    U16                 port;

    if (!SocketBind(tcp_socket->socket))
    {
        NetErr("TCP SOCKET BIND: Failed, Socket state-machine must be in READY state.");
        return -1;
    }

    if (tcp_socket->state != TCP_STATE_CLOSED)
    {
        NetErr("TCP SOCKET BIND: Failed, TCP Socket must be in CLOSED state.");
        return -1;
    }

    switch (address->family)
    {
        case AF_INET:

            if (tcp_socket->source_address.family == AF_INET6)
            {
                NetErr("TCP SOCKET BIND: FIXME, IPV6->IPV4 TCP BIND");
                return -1;
            }

            ipv4_address = address;
            ipv4_destination = &tcp_socket->destination_address;

            ipv4_destination->family            = AF_INET;
            ipv4_destination->address.address   = ipv4_address->address.address; // bind socket to address in parameter.
//          ipv4_destination->port              = ipv4_address->port; // ... consistency would say keep in Big Endian ...

            ipv4_source = &tcp_socket->source_address;

            ipv4_source->port = ipv4_address->port;

            port = EndianU16(ipv4_address->port); // port member should be Big Endian,  so now we're going L.E (?)

            break;

        case AF_INET6:

            if (tcp_socket->source_address.family == AF_INET)
            {
                NetErr("TCP SOCKET BIND: Incompatible Address type.");
                return -1;
            }

            ipv6_address = address;
            ipv6_destination = &tcp_socket->destination_address;
            // ...
            // ...

            port = EndianU16(ipv6_address->port); // port member should be Big Endian,  so now we're going L.E (?)

            NetErr("TCP SOCKET BIND: FIXME, IPV6 TCP BIND");

            break;

        case AF_UNSPEC:
            NetErr("TCP SOCKET BIND: Error, AF_UNSPEC TCP BIND -- param family");
            break;
    }

    if (head)
    {
        // look for our port.
        temp_node = TCPTreeNodeFind(port, head);

        if (temp_node)
        { // if we find we have bound sockets at port, check address before adding to queue
            switch (address->family)
            {
                case AF_INET:
                    if (TCPTreeNodeQueueIPV4Find(ipv4_destination->address.address, temp_node, TRUE))
                    {
                        NetErr("TCP SOCKET BIND: Address already in Bound Socket Tree !");
                        return -1;
                    }
                    else
                    { // if no address match, free to add socket to the node queue
                        NetDebug("TCP SOCKET BIND: No address match, adding socket to node queue");
                        TCPTreeNodeQueueAdd(tcp_socket, temp_node);
                    }

                    break;

                case AF_INET6:
                    NetErr("TCP SOCKET BIND: FIXME, IPV6 TCP BIND");
                    break;

                case AF_UNSPEC:
                    NetErr("TCP SOCKET BIND: Error, AF_UNSPEC TCP BIND -- found in bound tree");
                    break;
            }
        }
        else
        { // if we get no node back from port search, we didn't find it and are free to add a new node.
            temp_node = TCPTreeNodeParamAdd(port, head); // add new node with port, return its *.
            TCPTreeNodeQueueAdd(tcp_socket, temp_node);
        }
    }
    else // if no bound sockets, we init the tree as a new node
    {
        tcp_globals.bound_socket_tree = head = TCPTreeNodeParamInit(port); //... shouuuld be in L.E .. ?
        TCPTreeNodeQueueAdd(tcp_socket, head); // add the tcp socket to the port queue
        // maybe more checks to do before this, dunno rn.
    }


    // do we need to do anything else after bind add to tree? ...


    switch (tcp_socket->socket->state)
    {
        case SOCKET_STATE_BIND_REQ: //  if BIND request success, set BOUND.
            tcp_socket->socket->state = SOCKET_STATE_BOUND;
            break;

        default:
            NetErr("TCP SOCKET BIND: Failed, Misconfigured Socket state-machine.");
            return -1;
    }

    return 0;
}

I64 TCPSocketClose(CTCPSocket *tcp_socket)
{
    CTCPTreeNode        *head = tcp_globals.bound_socket_tree;
    CTCPTreeNode        *node;
    CTCPTreeQueue       *queue;
    CTCPAckQueue        *send_buffer;
    CTCPAcceptQueue     *pending;
    CSocketAddressIPV4  *address = &tcp_socket->source_address;
    I64                  timeout;

    SocketClose(tcp_socket->socket); // TODO: testing on closing a socket while another task is using it

    switch (tcp_socket->state)
    {
        case TCP_STATE_ESTABLISHED:
            while (TCPSendFlags(tcp_socket, TCPF_FIN | TCPF_ACK) < 0)
            {
                TCPCheckACKQueue(tcp_socket);
                Sleep(1);
            }

            tcp_socket->state = TCP_STATE_FIN_WAIT1;

            // Shrine has TODOs:    - What states are allowed here?
            //                      - This can block forever if our recv buffer fills up
            //                        but remote insists on sending more.
            // Implementing timeout here to force close.
            timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;

            while (tcp_socket->state == TCP_STATE_FIN_WAIT1 &&
                   tcp_socket->first_unacked_seq != tcp_socket->next_send_seq_num &&
                   counts.jiffies < timeout)
            {
                TCPCheckACKQueue(tcp_socket);
                Sleep(1);
            }

            if (tcp_socket->state == TCP_STATE_FIN_WAIT1)
                tcp_socket->state = TCP_STATE_FIN_WAIT2;

            timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;
            while (tcp_socket->state == TCP_STATE_FIN_WAIT2 && counts.jiffies < timeout)
                Sleep(1);

            break;

        case TCP_STATE_CLOSE_WAIT:
            while (TCPSendFlags(tcp_socket, TCPF_FIN | TCPF_ACK) < 0)
            {
                TCPCheckACKQueue(tcp_socket);
                Sleep(1);
            }

            if (tcp_socket->state == TCP_STATE_CLOSE_WAIT)
                tcp_socket->state = TCP_STATE_LAST_ACK;

            timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;
            while (tcp_socket->state == TCP_STATE_LAST_ACK &&
                   tcp_socket->first_unacked_seq != tcp_socket->next_send_seq_num &&
                   counts.jiffies < timeout)
            {
                TCPCheckACKQueue(tcp_socket);
                Sleep(1);
            }

            break;
    }

    if (IsTCPStateSync(tcp_socket))
        TCPSendFlags(tcp_socket, TCPF_RST);

    // remove from bound list, free associated allocs

    node = TCPTreeNodeFind(EndianU16(address->port), head);

    if (node)
        queue = TCPTreeNodeQueueSocketFind(tcp_socket, node);
    else
    {
        Debug("TODO: Didn't find node at socket during TCPSocketClose!\n");
        return -1;
    }

    if (queue)
    {
        TCPTreeNodeQueueSocketSinglePop(tcp_socket, node);

        if (node->queue == node->queue->next)
        { // if we popped the only queue on the node, remove the node.
            if (node == head)
            { // head is the global. if node is the global, change it and add branches.
                if (node->left)
                {
                    tcp_globals.bound_socket_tree = head = node->left;
                    if (node->right)
                        TCPTreeNodeAdd(node->right, head);
                }
                else if (node->right)
                    tcp_globals.bound_socket_tree = node->right;
                else
                    tcp_globals.bound_socket_tree = NULL;
            }
            else // if node is not the global, just pop it from the tree
                TCPTreeNodeSinglePop(node->value, head);

            Free(node);
        }

        Free(tcp_socket->socket);

        NetWarn("TCP SOCKET CLOSE: Freeing socket, ACK queue, accept() queue, & receive buffer.");
        Free(tcp_socket->receive_buffer);

        send_buffer = tcp_socket->ack_queue->next;
        while (send_buffer != tcp_socket->ack_queue)
        {
            NetWarn("TCP SOCKET CLOSE: Freeing send buffer @ 0x%0X", send_buffer);
            QueueRemove(send_buffer);
            Free(send_buffer->tcp_packet);
            Free(send_buffer);
            send_buffer = tcp_socket->ack_queue->next;
        }

        pending = tcp_socket->accept_queue->next;
        while (pending != tcp_socket->accept_queue)
        {
            NetWarn("TCP SOCKET CLOSE: Freeing pending connection @ 0x%0X", pending);
            QueueRemove(pending);
            Free(pending);
            pending = tcp_socket->accept_queue->next;
        }


        Free(tcp_socket);
        Free(queue);
    }
    else
    {
        Debug("TODO: Didn't find queue at socket during TCPSocketClose!\n");
        return -1;
    }
    
    return 0;

}

I64 TCPSocketConnect(CTCPSocket *tcp_socket, CSocketAddressStorage *address)
{ // Bind and connect a socket to the destination address and port in address param, local port automatic.
    U16                 source_port = EndianU16(0x8000 + tcp_globals.next_source_port++ & 0x7FFF);
    CTCPTreeNode       *head = tcp_globals.bound_socket_tree;
    CTCPTreeNode       *temp_node;
    CSocketAddressIPV4 *ipv4_address;
    CSocketAddressIPV4 *ipv4_destination;
    CSocketAddressIPV4 *ipv4_source;
    CSocketAddressIPV6 *ipv6_address;
    CSocketAddressIPV6 *ipv6_destination;
    I64                 timeout;
    U16                 port;


    if (!SocketConnect(tcp_socket->socket))
    {
        NetErr("TCP SOCKET CONNECT: Failed, Socket state-machine must be in READY state.");
        return -1;
    }

    if (tcp_socket->state != TCP_STATE_CLOSED)
    {
        NetErr("TCP SOCKET CONNECT: TCP Socket must be in CLOSED state.");
        return -1;
    }

    switch (address->family)
    {
        case AF_INET:

            if (tcp_socket->source_address.family == AF_INET6)
            {
                NetErr("TCP SOCKET CONNECT: FIXME, IPV6->IPV4 TCP CONNECT");
                return -1;
            }

            ipv4_address = address;
            ipv4_destination = &tcp_socket->destination_address;

            ipv4_destination->family            = AF_INET;
            ipv4_destination->address.address   = ipv4_address->address.address; // bind socket to address in parameter.
            ipv4_destination->port              = ipv4_address->port; // ... consistency would say keep in Big Endian ...

            ipv4_source = &tcp_socket->source_address;

            ipv4_source->port = source_port;

            port = EndianU16(source_port); // port member should be Big Endian,  so now we're going L.E (?)

            break;

        case AF_INET6:

            if (tcp_socket->source_address.family == AF_INET)
            {
                NetErr("TCP SOCKET CONNECT: Incompatible Address type.");
                return -1;
            }

            ipv6_address = address;
            ipv6_destination = &tcp_socket->destination_address;
            // ...
            // ...

            port = EndianU16(source_port); // port member should be Big Endian,  so now we're going L.E (?)

            NetErr("TCP SOCKET CONNECT: FIXME, IPV6 TCP CONNECT");

            break;

        case AF_UNSPEC:
            NetErr("TCP SOCKET CONNECT: Error, AF_UNSPEC TCP CONNECT -- param family");
            break;
    }

    if (head)
    {
        // look for our port.
        temp_node = TCPTreeNodeFind(port, head);

        if (temp_node)
        { // if we find we have bound sockets at port, check address before adding to queue
            switch (address->family)
            {
                case AF_INET:
                    if (TCPTreeNodeQueueIPV4Find(ipv4_destination->address.address, temp_node, TRUE))
                    {
                        NetErr("TCP SOCKET CONNECT: Address already in Bound Socket Tree !");
                        return -1;
                    }
                    else
                    { // if no address match, free to add socket to the node queue
                        NetDebug("TCP SOCKET CONNECT: No address match, adding socket to node queue");
                        TCPTreeNodeQueueAdd(tcp_socket, temp_node);
                    }

                    break;

                case AF_INET6:
                    NetErr("TCP SOCKET CONNECT: FIXME, IPV6 TCP CONNECT");
                    break;

                case AF_UNSPEC:
                    NetErr("TCP SOCKET CONNECT: Error, AF_UNSPEC TCP CONNECT -- found in bound tree");
                    break;
            }
        }
        else
        { // if we get no node back from port search, we didn't find it and are free to add a new node.
            temp_node = TCPTreeNodeParamAdd(port, head); // add new node with port, return its *.
            TCPTreeNodeQueueAdd(tcp_socket, temp_node);
        }
    }
    else // if no bound sockets, we init the tree as a new node
    {
        tcp_globals.bound_socket_tree = head = TCPTreeNodeParamInit(port); //... shouuuld be in L.E .. ?
        TCPTreeNodeQueueAdd(tcp_socket, head); // add the tcp socket to the port queue
        // maybe more checks to do before this, dunno rn.
    }

    tcp_socket->connection_time = tS;
    tcp_socket->receive_window  = TCP_WINDOW_SIZE;

    tcp_socket->state = TCP_STATE_SYN_SENT;
    TCPSendFlags(tcp_socket, TCPF_SYN);

    timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;
    while (counts.jiffies < timeout)
    {
        switch (tcp_socket->state)
        {
            case TCP_STATE_ESTABLISHED:
            case TCP_STATE_CLOSED:
                timeout = 0; // break out of while loop
                break;

            default:
                Sleep(1);
        }
    }

    if (tcp_socket->state != TCP_STATE_ESTABLISHED)
        return -1;

    switch (tcp_socket->socket->state)
    {
        case SOCKET_STATE_CONNECT_REQ: //   if CONNECT request success, set OPEN.
            tcp_socket->socket->state = SOCKET_STATE_OPEN;
            break;

        default:
            NetErr("TCP SOCKET CONNECT: Failed, Misconfigured Socket state-machine.");
            return -1;
    }

    return 0;
}

I64 TCPSocketListen(CTCPSocket *tcp_socket, I64 backlog_size)
{ // Set a bound socket to Listen for incoming connections. Backlog size is max amount of waiting connections allowed
    if (!SocketListen(tcp_socket->socket))
    {
        NetErr("TCP SOCKET LISTEN: Socket state-machine must be in BOUND state.");
        return -1;
    }

    if (tcp_socket->state != TCP_STATE_CLOSED)
    {
        NetErr("TCP SOCKET LISTEN: TCP Socket must be in CLOSED state.");
        return -1;
    }

    tcp_socket->state = TCP_STATE_LISTEN;
    tcp_socket->accept_queue_limit = backlog_size;

    switch (tcp_socket->socket->state)
    {
        case SOCKET_STATE_LISTEN_REQ: //    if LISTEN request success, set BOUND.
            tcp_socket->socket->state = SOCKET_STATE_LISTENING;
            break;

        default:
            NetErr("TCP SOCKET BIND: Failed, Misconfigured Socket state-machine.");
            return -1;
    }

    return 0;
}

CTCPSocket *TCPSocketAccept(CTCPSocket *tcp_socket)
{ // Accepts & creates a new socket, uses timeout inherited from Listening socket.
    CTCPAcceptQueue     *pending;
    CTCPSocket          *new_socket;
    CSocketAddressIPV4  *temp_addr;
    CSocketAddressIPV4   ipv4_address;
    I64                  timeout;

    if (!SocketAccept(tcp_socket->socket))
    {
        NetErr("TCP SOCKET ACCEPT: Failed, Socket state-machine must be in LISTENING state.");
        return NULL;
    }

    if (tcp_socket->state != TCP_STATE_LISTEN)
    {
        NetErr("TCP SOCKET LISTEN: TCP Socket must be in LISTEN state.");
        return NULL;
    }

    timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;
    while ((pending = tcp_socket->accept_queue->next) == tcp_socket->accept_queue)
    {
        if (counts.jiffies > timeout)
            return NULL;
        else
            Sleep(1);
            // Yield;
    }

    QueueRemove(pending); // whether successful accept() or not, remove pending connection.

// TODO: rework accept logic to handle IPV6 addresses
    new_socket = TCPSocket(AF_INET);

    new_socket->next_recv_seq_num   = ++pending->segment_seq_num;
    new_socket->connection_time     = tS;
    new_socket->receive_window      = TCP_WINDOW_SIZE;
    new_socket->timeout             = tcp_socket->timeout;

    temp_addr = &tcp_socket->source_address;

    ipv4_address.family             = AF_INET;
//  ipv4_address.port               = pending->port;
    ipv4_address.port               = temp_addr->port;
    ipv4_address.address.address    = pending->ipv4_address;

    NetDebug("TCP SOCKET ACCEPT: Attempting to Bind to pending connection %0X @ src/dst ports %d,%d.",
                pending->ipv4_address, ipv4_address.port, pending->port);

    if (TCPSocketBind(new_socket, &ipv4_address) == -1)
    {
        Free(pending);
        return NULL;
    }

    temp_addr = &new_socket->destination_address;

    temp_addr->port = pending->port;

    new_socket->state = TCP_STATE_SYN_RECEIVED;

    NetDebug("TCP SOCKET ACCEPT: Attempting Send Flags SYN ACK back to requester.");
    TCPSendFlags(new_socket, TCPF_SYN | TCPF_ACK);

    timeout = counts.jiffies + new_socket->timeout * JIFFY_FREQ / 1000;
    while (counts.jiffies < timeout)
    {
        switch (new_socket->state)
        {
            case TCP_STATE_ESTABLISHED:
            case TCP_STATE_CLOSED:
                timeout = 0; // break out of while loop
                break;

            default:
                Sleep(1);
        }
    }

    if (new_socket->state != TCP_STATE_ESTABLISHED)
    {
        TCPSocketClose(new_socket);
        Free(pending);
        return NULL;
    }

    Free(pending);

    new_socket->socket->state = SOCKET_STATE_OPEN;

    return new_socket;
}

I64 TCPSocketReceive(CTCPSocket *tcp_socket, U8 *buffer, I64 length)
{
    I64 read_position;
    I64 write_position;
    I64 read_total = 0;
    I64 step;
    I64 timeout;

    if (!SocketReceive(tcp_socket->socket))
    {
        NetErr("TCP SOCKET RECEIVE: Failed, Socket state-machine must be in OPEN or BOUND state.");
        return NULL;
    }

    timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;

    while ((tcp_socket->state == TCP_STATE_ESTABLISHED || tcp_socket->state == TCP_STATE_FIN_WAIT1) &&
           tcp_socket->read_position == tcp_socket->write_position)
    {
        TCPCheckACKQueue(tcp_socket);
        Sleep(1);

        if (counts.jiffies > timeout)
        {
//          if (tcp_socket->timeout != 0) // Don't flood NetLog on non-blocking receives.
//              NetErr("TCP SOCKET RECEIVE: Timed out.");

//          return -1;
            break;
        }


    }

    // Shrine has TODO: Should still be able to receive in closing states ...
    if ((tcp_socket->state != TCP_STATE_ESTABLISHED || tcp_socket->state == TCP_STATE_FIN_WAIT1) &&
        tcp_socket->read_position == tcp_socket->write_position || length == 0)
        return 0;


    if (counts.jiffies > timeout)
    {
        if (tcp_socket->timeout != 0) // Don't flood NetLog on non-blocking receives.
            NetErr("TCP SOCKET RECEIVE: Timed out.");

        return -1;
    }


    read_position   = tcp_socket->read_position;
    write_position  = tcp_socket->write_position;

    if (write_position < read_position)
    {
        step = tcp_socket->receive_buffer_size - read_position;

        if (step > length)
            step = length;

        MemCopy(buffer, tcp_socket->receive_buffer + read_position, step);
        buffer += step;
        length -= step;
//      read_position = (read_position + step) & (tcp_socket->receive_buffer_size - 1);
        read_position = (read_position + step) % tcp_socket->receive_buffer_size;
        read_total += step;
    }

    if (length > 0)
    {
        step = write_position - read_position;

        if (step > length)
            step = length;

        MemCopy(buffer, tcp_socket->receive_buffer + read_position, step);
        buffer += step;
        length -= step;
        read_position += step;
        read_total += step;
    }

    tcp_socket->read_position = read_position;

    return read_total;
}

I64 TCPSocketSend(CTCPSocket *tcp_socket, U8 *buffer, I64 length)
{
    I64 sent_total = 0;
    I64 timeout = counts.jiffies + tcp_socket->timeout * JIFFY_FREQ / 1000;
    I64 send_length;

    if (!SocketSend(tcp_socket->socket))
    {
        NetErr("TCP SOCKET SEND: Failed, Socket state-machine must be in OPEN state.");
        return -1;
    }

    while ((tcp_socket->state == TCP_STATE_ESTABLISHED || tcp_socket->state == TCP_STATE_CLOSE_WAIT) &&
           length)
    {
        send_length = (tcp_socket->first_unacked_seq + tcp_socket->send_window - tcp_socket->next_send_seq_num) & 0xFFFFFFFF;

        // Shrine has TODO: Keep trying, tie to timeout: RFC 793 "Managing The Window"

        if (send_length == 0)
        {
            if (sent_total > 0)
                break;
            else
            {
                TCPCheckACKQueue(tcp_socket);
                Sleep(1);
            }
        }
        else
        {
            if (send_length > length)
                send_length = length;

            if (send_length > tcp_socket->max_segment_size)
                send_length = tcp_socket->max_segment_size;

            NetDebug("TCP SOCKET SEND: Trying TCPSendData() of %d bytes.", send_length);
            if (TCPSendData(tcp_socket, TCPF_ACK, buffer, send_length) < 0)
            { // Stall until outgoing data acknowledged.
                if (sent_total > 0)
                    break;
                else
                {
                    TCPCheckACKQueue(tcp_socket);
                    Sleep(1);
                }

            }
            else
            {
                buffer      += send_length;
                length      -= send_length;
                sent_total  += send_length;
            }
        }

        if (counts.jiffies > timeout)
        {
            if (tcp_socket->timeout != 0) // Don't flood NetLog on non-blocking sends.
                NetErr("TCP SOCKET SEND: Timed out.");

            break;
        }
    }

    return sent_total;
}

I64 TCPSocketSendAll(CTCPSocket *tcp_socket, U8 *buffer, I64 length)
{
    I64 total   = 0;
    I64 sent    = 0;

    while (length)
    {
        sent = TCPSocketSend(tcp_socket, buffer, length);

        if (sent > 0)
        {
            buffer  += sent;
            total   += sent;
            length  -= sent;
        }
        else
            break;
    }

    return total;
}

I64 TCPSocketSendString(CTCPSocket *tcp_socket, U8 *string)
{
    return TCPSocketSendAll(tcp_socket, string, StrLen(string));
}

U0 TCPTreeNodeRep(CTCPTreeNode *node)
{
    CTCPTreeQueue       *queue = node->queue->next;
    CTCPSocket          *socket;
    CSocketAddressIPV4  *ipv4_addr;
    CSocketAddressIPV6  *ipv6_addr;
    U8                  *string;
    CTCPAcceptQueue     *pending;
    

    "Local Port $YELLOW$%d$FG$ (TCP Node @ $CYAN$0x%X$FG$):\n", node->value, node;

    while (queue != node->queue)
    {

        socket = queue->socket;

        switch (socket->destination_address.family)
        {

            case AF_UNSPEC:
                break;

            case AF_INET:
                ipv4_addr = &socket->destination_address;
                string = MStrPrint("%d.%d.%d.%d",
                            ipv4_addr->address.address.u8[3],
                            ipv4_addr->address.address.u8[2],
                            ipv4_addr->address.address.u8[1],
                            ipv4_addr->address.address.u8[0]); // todo: kludge, endianness...

                "   $BROWN$%s$FG$ Destination Port $YELLOW$%d$FG$ (TCP Tree Queue @ $CYAN$0x%X$FG$):\n",
                    string, EndianU16(ipv4_addr->port), queue;
                Free(string);

                break;

            case AF_INET6:
                ipv6_addr = &socket->destination_address;

                break;

            default:
                break;

        }

        "       Timeout: %dms\n", socket->timeout;

        pending = socket->accept_queue->next;

        while (pending != socket->accept_queue->next)
        {
            "       Pending connection from $BROWN$%d.%d.%d.%d$FG$:",
                pending->ipv4_address.u8[3],
                pending->ipv4_address.u8[2],
                pending->ipv4_address.u8[1],
                pending->ipv4_address.u8[0]; //todo: kludge, endianness?

            pending = pending->next;
        }


        queue = queue->next;
    }

    "\n";
}

U0 TCPRep()
{
    CTCPTreeNode    *node = tcp_globals.bound_socket_tree;
    CTCPRepEntry    *head;
    CTCPRepEntry    *entry;
    CTCPRepEntry    *temp_entry;

    "$LTBLUE$TCP Report:$FG$\n\n";

    if (node)
    {
        head = CAlloc(sizeof(CTCPRepEntry));
        QueueInit(head); // no QueueRemove the head

        entry = CAlloc(sizeof(CTCPRepEntry));
        entry->node = node;
        QueueInsert(entry, head);

        // perform depth-first-search while Entry Queue has nodes not fully visited.
        while (entry != head)
        {
            if (entry->node->left)
            { // if node has one, add an Entry for the left branch, continue loop.
                temp_entry = CAlloc(sizeof(CTCPRepEntry));
                temp_entry->node = entry->node->left;
                QueueInsertRev(temp_entry, head);

                // if left branch, but no right: toss early, now fully traveled.
                if (!entry->node->right)
                {
                    QueueRemove(entry);
                    TCPTreeNodeRep(entry->node);
                    Free(entry);
                }

                entry = temp_entry;
            }
            else if (entry->node->right)
            { // if no left, but right: add right to queue, pop Entry, Rep, set entry to right.
                temp_entry = CAlloc(sizeof(CTCPRepEntry));
                temp_entry->node = entry->node->right;
                QueueInsertRev(temp_entry, head);

                QueueRemove(entry);
                TCPTreeNodeRep(entry->node);
                Free(entry);

                entry = temp_entry;
            }
            else
            { // pop Entry, Rep, if last Entry in Queue has right add it, pop & Rep travelled Entry, entry = right.
                QueueRemove(entry);
                TCPTreeNodeRep(entry->node);
                Free(entry);

                if (head->last != head)
                {
                    temp_entry = head->last;

                    if (temp_entry->node->right)
                    {
                        entry = temp_entry;

                        temp_entry = CAlloc(sizeof(CTCPRepEntry));
                        temp_entry->node = entry->node->right;
                        QueueInsertRev(temp_entry, head);

                        QueueRemove(entry);
                        TCPTreeNodeRep(entry->node);
                        Free(entry);

                        entry = temp_entry;
                    }
                    else
                    {
                        QueueRemove(temp_entry);
                        TCPTreeNodeRep(temp_entry->node);
                        Free(temp_entry);

                        entry = head->last;
                    }
                }
                else
                    break;
            }
        }

        Free(head);
    }
    else
        "No TCP Sockets currently bound.\n\n";
}


TCPGlobalsInit;