minix/servers/inet/generic/tcp_recv.c
2006-03-24 14:06:03 +00:00

1485 lines
32 KiB
C

/*
tcp_recv.c
Copyright 1995 Philip Homburg
*/
#include "inet.h"
#include "buf.h"
#include "clock.h"
#include "event.h"
#include "type.h"
#include "sr.h"
#include "io.h"
#include "tcp_int.h"
#include "tcp.h"
#include "assert.h"
THIS_FILE
FORWARD void create_RST ARGS(( tcp_conn_t *tcp_conn,
ip_hdr_t *ip_hdr, tcp_hdr_t *tcp_hdr, int data_len ));
FORWARD void process_data ARGS(( tcp_conn_t *tcp_conn,
tcp_hdr_t *tcp_hdr, acc_t *tcp_data, int data_len ));
FORWARD void process_advanced_data ARGS(( tcp_conn_t *tcp_conn,
tcp_hdr_t *tcp_hdr, acc_t *tcp_data, int data_len ));
PUBLIC void tcp_frag2conn(tcp_conn, ip_hdr, tcp_hdr, tcp_data, data_len)
tcp_conn_t *tcp_conn;
ip_hdr_t *ip_hdr;
tcp_hdr_t *tcp_hdr;
acc_t *tcp_data;
size_t data_len;
{
tcp_fd_t *connuser;
int tcp_hdr_flags;
int ip_hdr_len, tcp_hdr_len;
u32_t seg_ack, seg_seq, rcv_hi, snd_una, snd_nxt;
u16_t seg_wnd, mtu;
size_t mss;
int acceptable_ACK, segm_acceptable, send_rst, close_connection;
ip_hdr_len= (ip_hdr->ih_vers_ihl & IH_IHL_MASK) << 2;
tcp_hdr_len= (tcp_hdr->th_data_off & TH_DO_MASK) >> 2;
tcp_hdr_flags= tcp_hdr->th_flags & TH_FLAGS_MASK;
seg_ack= ntohl(tcp_hdr->th_ack_nr);
seg_seq= ntohl(tcp_hdr->th_seq_nr);
seg_wnd= ntohs(tcp_hdr->th_window);
#if 0
{ where(); tcp_print_conn(tcp_conn); printf("\n");
tcp_print_pack(ip_hdr, tcp_hdr); printf("\n"); }
#endif
switch (tcp_conn->tc_state)
{
case TCS_CLOSED:
/*
CLOSED:
discard all data.
!RST ?
ACK ?
<SEQ=SEG.ACK><CTL=RST>
exit
:
<SEQ=0><ACK=SEG.SEQ+SEG.LEN><CTL=RST,ACK>
exit
:
discard packet
exit
*/
if (!(tcp_hdr_flags & THF_RST))
{
create_RST(tcp_conn, ip_hdr, tcp_hdr, data_len);
tcp_conn_write(tcp_conn, 1);
}
break;
case TCS_LISTEN:
/*
LISTEN:
RST ?
discard packet
exit
ACK ?
<SEQ=SEG.ACK><CTL=RST>
exit
SYN ?
BUG: no security check
RCV.NXT= SEG.SEQ+1
IRS= SEG.SEQ
ISS should already be selected
<SEQ=ISS><ACK=RCV.NXT><CTL=SYN,ACK>
SND.NXT=ISS+1
SND.UNA=ISS
state= SYN-RECEIVED
exit
:
shouldnot occur
discard packet
exit
*/
if (tcp_hdr_flags & THF_RST)
break;
if (tcp_hdr_flags & THF_ACK)
{
create_RST (tcp_conn, ip_hdr, tcp_hdr, data_len);
tcp_conn_write(tcp_conn, 1);
break;
}
if (tcp_hdr_flags & THF_SYN)
{
tcp_extract_ipopt(tcp_conn, ip_hdr);
tcp_extract_tcpopt(tcp_conn, tcp_hdr, &mss);
mtu= mss+IP_TCP_MIN_HDR_SIZE;
if (mtu < IP_MIN_MTU)
{
/* No or unrealistic mss, use default MTU */
mtu= IP_DEF_MTU;
}
if (mtu < tcp_conn->tc_max_mtu)
{
tcp_conn->tc_max_mtu= mtu;
tcp_conn->tc_mtu= mtu;
DBLOCK(1, printf(
"tcp[%d]: conn[%d]: mtu = %d\n",
tcp_conn->tc_port-tcp_port_table,
tcp_conn-tcp_conn_table,
mtu););
}
tcp_conn->tc_RCV_LO= seg_seq+1;
tcp_conn->tc_RCV_NXT= seg_seq+1;
tcp_conn->tc_RCV_HI= tcp_conn->tc_RCV_LO+
tcp_conn->tc_rcv_wnd;
tcp_conn->tc_RCV_UP= seg_seq;
tcp_conn->tc_IRS= seg_seq;
tcp_conn->tc_SND_UNA= tcp_conn->tc_ISS;
tcp_conn->tc_SND_TRM= tcp_conn->tc_ISS;
tcp_conn->tc_SND_NXT= tcp_conn->tc_ISS+1;
tcp_conn->tc_SND_UP= tcp_conn->tc_ISS-1;
tcp_conn->tc_SND_PSH= tcp_conn->tc_ISS-1;
tcp_conn->tc_state= TCS_SYN_RECEIVED;
tcp_conn->tc_stt= 0;
assert (tcp_check_conn(tcp_conn));
tcp_conn->tc_locaddr= ip_hdr->ih_dst;
tcp_conn->tc_locport= tcp_hdr->th_dstport;
tcp_conn->tc_remaddr= ip_hdr->ih_src;
tcp_conn->tc_remport= tcp_hdr->th_srcport;
tcp_conn_write(tcp_conn, 1);
DIFBLOCK(0x10, seg_seq == 0,
printf("warning got 0 IRS from ");
writeIpAddr(tcp_conn->tc_remaddr);
printf("\n"));
/* Start the timer (if necessary) */
tcp_set_send_timer(tcp_conn);
break;
}
/* do nothing */
break;
case TCS_SYN_SENT:
/*
SYN-SENT:
ACK ?
SEG.ACK <= ISS || SEG.ACK > SND.NXT ?
RST ?
discard packet
exit
:
<SEQ=SEG.ACK><CTL=RST>
exit
SND.UNA <= SEG.ACK && SEG.ACK <= SND.NXT ?
ACK is acceptable
:
ACK is !acceptable
:
ACK is !acceptable
RST ?
ACK acceptable ?
discard segment
state= CLOSED
error "connection refused"
exit
:
discard packet
exit
BUG: no security check
SYN ?
IRS= SEG.SEQ
RCV.NXT= IRS+1
ACK ?
SND.UNA= SEG.ACK
SND.UNA > ISS ?
state= ESTABLISHED
<SEQ=SND.NXT><ACK= RCV.NXT><CTL=ACK>
process ev. URG and text
exit
:
state= SYN-RECEIVED
SND.WND= SEG.WND
SND.WL1= SEG.SEQ
SND.WL2= SEG.ACK
<SEQ=ISS><ACK=RCV.NXT><CTL=SYN,ACK>
exit
:
discard segment
exit
*/
if (tcp_hdr_flags & THF_ACK)
{
if (tcp_LEmod4G(seg_ack, tcp_conn->tc_ISS) ||
tcp_Gmod4G(seg_ack, tcp_conn->tc_SND_NXT))
if (tcp_hdr_flags & THF_RST)
break;
else
{
/* HACK: force sending a RST,
* normally, RSTs are not send
* if the segment is an ACK.
*/
create_RST (tcp_conn, ip_hdr,
tcp_hdr, data_len+1);
tcp_conn_write(tcp_conn, 1);
break;
}
acceptable_ACK= (tcp_LEmod4G(tcp_conn->tc_SND_UNA,
seg_ack) && tcp_LEmod4G(seg_ack,
tcp_conn->tc_SND_NXT));
}
else
acceptable_ACK= FALSE;
if (tcp_hdr_flags & THF_RST)
{
if (acceptable_ACK)
{
DBLOCK(1, printf(
"calling tcp_close_connection\n"));
tcp_close_connection(tcp_conn,
ECONNREFUSED);
}
break;
}
if (tcp_hdr_flags & THF_SYN)
{
tcp_extract_ipopt(tcp_conn, ip_hdr);
tcp_extract_tcpopt(tcp_conn, tcp_hdr, &mss);
mtu= mss+IP_TCP_MIN_HDR_SIZE;
if (mtu < IP_MIN_MTU)
{
/* No or unrealistic mss, use default MTU */
mtu= IP_DEF_MTU;
}
if (mtu < tcp_conn->tc_max_mtu)
{
tcp_conn->tc_max_mtu= mtu;
tcp_conn->tc_mtu= mtu;
DBLOCK(1, printf(
"tcp[%d]: conn[%d]: mtu = %d\n",
tcp_conn->tc_port-tcp_port_table,
tcp_conn-tcp_conn_table,
mtu););
}
tcp_conn->tc_RCV_LO= seg_seq+1;
tcp_conn->tc_RCV_NXT= seg_seq+1;
tcp_conn->tc_RCV_HI= tcp_conn->tc_RCV_LO +
tcp_conn->tc_rcv_wnd;
tcp_conn->tc_RCV_UP= seg_seq;
tcp_conn->tc_IRS= seg_seq;
if (tcp_hdr_flags & THF_ACK)
tcp_conn->tc_SND_UNA= seg_ack;
if (tcp_Gmod4G(tcp_conn->tc_SND_UNA,
tcp_conn->tc_ISS))
{
tcp_conn->tc_state= TCS_ESTABLISHED;
tcp_conn->tc_rt_dead= TCP_DEF_RT_DEAD;
assert (tcp_check_conn(tcp_conn));
assert(tcp_conn->tc_connInprogress);
tcp_restart_connect(tcp_conn);
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
if (data_len != 0)
{
tcp_frag2conn(tcp_conn, ip_hdr,
tcp_hdr, tcp_data, data_len);
/* tcp_data is already freed */
return;
}
break;
}
tcp_conn->tc_state= TCS_SYN_RECEIVED;
assert (tcp_check_conn(tcp_conn));
tcp_conn->tc_SND_TRM= tcp_conn->tc_ISS;
tcp_conn_write(tcp_conn, 1);
}
break;
case TCS_SYN_RECEIVED:
/*
SYN-RECEIVED:
test if segment is acceptable:
Segment Receive Test
Length Window
0 0 SEG.SEQ == RCV.NXT
0 >0 RCV.NXT <= SEG.SEQ && SEG.SEQ < RCV.NXT+RCV.WND
>0 0 not acceptable
>0 >0 (RCV.NXT <= SEG.SEQ && SEG.SEQ < RCV.NXT+RCV.WND)
|| (RCV.NXT <= SEG.SEQ+SEG.LEN-1 &&
SEG.SEQ+SEG.LEN-1 < RCV.NXT+RCV.WND)
for urgent data: use RCV.WND+1 for RCV.WND
Special: Send RST if SEG.SEQ < IRS or SEG.SEQ > RCV.NXT+64K (and
the packet is not a RST packet itself).
*/
rcv_hi= tcp_conn->tc_RCV_HI;
if (tcp_hdr_flags & THF_URG)
rcv_hi++;
send_rst= tcp_Lmod4G(seg_seq, tcp_conn->tc_IRS) ||
tcp_Gmod4G(seg_seq, tcp_conn->tc_RCV_NXT+0x10000);
close_connection= 0;
if (!data_len)
{
if (rcv_hi == tcp_conn->tc_RCV_NXT)
segm_acceptable= (seg_seq == rcv_hi);
else
{
assert (tcp_Gmod4G(rcv_hi,
tcp_conn->tc_RCV_NXT));
segm_acceptable= (tcp_LEmod4G(tcp_conn->
tc_RCV_NXT, seg_seq) &&
tcp_Lmod4G(seg_seq, rcv_hi));
}
}
else
{
if (tcp_Gmod4G(rcv_hi, tcp_conn->tc_RCV_NXT))
{
segm_acceptable= (tcp_LEmod4G(tcp_conn->
tc_RCV_NXT, seg_seq) &&
tcp_Lmod4G(seg_seq, rcv_hi)) ||
(tcp_LEmod4G(tcp_conn->tc_RCV_NXT,
seg_seq+data_len-1) &&
tcp_Lmod4G(seg_seq+data_len-1,
rcv_hi));
}
else
{
segm_acceptable= FALSE;
}
}
/*
!segment acceptable ?
RST ?
discard packet
exit
:
<SEG=SND.NXT><ACK=RCV.NXT><CTL=ACK>
exit
*/
if (!segm_acceptable)
{
if (tcp_hdr_flags & THF_RST)
; /* do nothing */
else if (send_rst)
{
create_RST(tcp_conn, ip_hdr, tcp_hdr,
data_len);
tcp_conn_write(tcp_conn, 1);
}
else
{
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
}
break;
}
/*
RST ?
initiated by a LISTEN ?
state= LISTEN
exit
:
state= CLOSED
error "connection refused"
exit
*/
if (tcp_hdr_flags & THF_RST)
close_connection= 1;
/*
SYN in window ?
initiated by a LISTEN ?
state= LISTEN
exit
:
state= CLOSED
error "connection reset"
exit
*/
if ((tcp_hdr_flags & THF_SYN) && tcp_GEmod4G(seg_seq,
tcp_conn->tc_RCV_NXT))
{
close_connection= 1;
}
if (close_connection)
{
if (!tcp_conn->tc_orglisten)
{
tcp_close_connection(tcp_conn, ECONNREFUSED);
break;
}
connuser= tcp_conn->tc_fd;
assert(connuser);
if (connuser->tf_flags & TFF_LISTENQ)
{
tcp_close_connection (tcp_conn,
ECONNREFUSED);
}
else
{
tcp_conn->tc_connInprogress= 0;
tcp_conn->tc_fd= NULL;
tcp_close_connection (tcp_conn,
ECONNREFUSED);
/* Pick a new ISS next time */
tcp_conn->tc_ISS= 0;
(void)tcp_su4listen(connuser, tcp_conn,
0 /* !do_listenq */);
}
break;
}
/*
!ACK ?
discard packet
exit
*/
if (!(tcp_hdr_flags & THF_ACK))
break;
/*
SND.UNA < SEG.ACK <= SND.NXT ?
state= ESTABLISHED
:
<SEG=SEG.ACK><CTL=RST>
exit
*/
if (tcp_Lmod4G(tcp_conn->tc_SND_UNA, seg_ack) &&
tcp_LEmod4G(seg_ack, tcp_conn->tc_SND_NXT))
{
tcp_conn->tc_state= TCS_ESTABLISHED;
tcp_conn->tc_rt_dead= TCP_DEF_RT_DEAD;
tcp_release_retrans(tcp_conn, seg_ack, seg_wnd);
assert (tcp_check_conn(tcp_conn));
assert(tcp_conn->tc_connInprogress);
tcp_restart_connect(tcp_conn);
tcp_frag2conn(tcp_conn, ip_hdr, tcp_hdr, tcp_data,
data_len);
/* tcp_data is already freed */
return;
}
else
{
create_RST (tcp_conn, ip_hdr, tcp_hdr, data_len);
tcp_conn_write(tcp_conn, 1);
break;
}
break;
case TCS_ESTABLISHED:
case TCS_CLOSING:
/*
ESTABLISHED:
FIN-WAIT-1:
FIN-WAIT-2:
CLOSE-WAIT:
CLOSING:
LAST-ACK:
TIME-WAIT:
test if segment is acceptable:
Segment Receive Test
Length Window
0 0 SEG.SEQ == RCV.NXT
0 >0 RCV.NXT <= SEG.SEQ && SEG.SEQ < RCV.NXT+RCV.WND
>0 0 not acceptable
>0 >0 (RCV.NXT <= SEG.SEQ && SEG.SEQ < RCV.NXT+RCV.WND)
|| (RCV.NXT <= SEG.SEQ+SEG.LEN-1 &&
SEG.SEQ+SEG.LEN-1 < RCV.NXT+RCV.WND)
for urgent data: use RCV.WND+1 for RCV.WND
*/
rcv_hi= tcp_conn->tc_RCV_HI;
if (tcp_hdr_flags & THF_URG)
rcv_hi++;
if (!data_len)
{
if (rcv_hi == tcp_conn->tc_RCV_NXT)
segm_acceptable= (seg_seq == rcv_hi);
else
{
assert (tcp_Gmod4G(rcv_hi,
tcp_conn->tc_RCV_NXT));
segm_acceptable= (tcp_LEmod4G(tcp_conn->
tc_RCV_NXT, seg_seq) &&
tcp_Lmod4G(seg_seq, rcv_hi));
}
}
else
{
if (tcp_Gmod4G(rcv_hi, tcp_conn->tc_RCV_NXT))
{
segm_acceptable= (tcp_LEmod4G(tcp_conn->
tc_RCV_NXT, seg_seq) &&
tcp_Lmod4G(seg_seq, rcv_hi)) ||
(tcp_LEmod4G(tcp_conn->tc_RCV_NXT,
seg_seq+data_len-1) &&
tcp_Lmod4G(seg_seq+data_len-1,
rcv_hi));
}
else
{
segm_acceptable= FALSE;
}
}
/*
!segment acceptable ?
RST ?
discard packet
exit
:
<SEG=SND.NXT><ACK=RCV.NXT><CTL=ACK>
exit
*/
if (!segm_acceptable)
{
if (!(tcp_hdr_flags & THF_RST))
{
DBLOCK(0x20,
printf("segment is not acceptable\n");
printf("\t");
tcp_print_pack(ip_hdr, tcp_hdr);
printf("\n\t");
tcp_print_conn(tcp_conn);
printf("\n"));
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
/* Sometimes, a retransmission sets the PSH
* flag (Solaris 2.4)
*/
if (tcp_conn->tc_rcvd_data != NULL &&
(tcp_hdr_flags & THF_PSH))
{
tcp_conn->tc_flags |= TCF_RCV_PUSH;
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags &
TFF_READ_IP))
{
tcp_fd_read(tcp_conn, 1);
}
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags &
TFF_SEL_READ))
{
tcp_rsel_read(tcp_conn);
}
}
}
break;
}
/*
RST ?
state == CLOSING || state == LAST-ACK ||
state == TIME-WAIT ?
state= CLOSED
exit
:
state= CLOSED
error "connection reset"
exit
*/
if (tcp_hdr_flags & THF_RST)
{
if ((tcp_conn->tc_flags &
(TCF_FIN_SENT|TCF_FIN_RECV)) ==
(TCF_FIN_SENT|TCF_FIN_RECV) &&
tcp_conn->tc_send_data == NULL)
{
/* Clean shutdown, but the other side
* doesn't want to ACK our FIN.
*/
tcp_close_connection (tcp_conn, 0);
}
else
tcp_close_connection(tcp_conn, ECONNRESET);
break;
}
/*
SYN in window ?
state= CLOSED
error "connection reset"
exit
*/
if ((tcp_hdr_flags & THF_SYN) && tcp_GEmod4G(seg_seq,
tcp_conn->tc_RCV_NXT))
{
tcp_close_connection(tcp_conn, ECONNRESET);
break;
}
/*
!ACK ?
discard packet
exit
*/
if (!(tcp_hdr_flags & THF_ACK))
break;
/*
SND.UNA < SEG.ACK <= SND.NXT ?
SND.UNA= SEG.ACK
reply "send ok"
SND.WL1 < SEG.SEQ || (SND.WL1 == SEG.SEQ &&
SND.WL2 <= SEG.ACK ?
SND.WND= SEG.WND
SND.Wl1= SEG.SEQ
SND.WL2= SEG.ACK
SEG.ACK <= SND.UNA ?
ignore ACK
SEG.ACK > SND.NXT ?
<SEQ=SND.NXT><ACK=RCV.NXT><CTL=ACK>
discard packet
exit
*/
/* Always reset the send timer after a valid ack is
* received. The assumption is that either the ack really
* acknowledges some data (normal case), contains a zero
* window, or the remote host has another reason not
* to accept any data. In all cases, the remote host is
* alive, so the connection should stay alive too.
* Do not reset stt if the state is CLOSING, i.e. if
* the user closed the connection and we still have
* some data to deliver. We don't want a zero window
* to keep us from closing the connection.
*/
if (tcp_conn->tc_state != TCS_CLOSING)
tcp_conn->tc_stt= 0;
snd_una= tcp_conn->tc_SND_UNA;
snd_nxt= tcp_conn->tc_SND_NXT;
if (seg_ack == snd_una)
{
if (tcp_Gmod4G(snd_nxt, snd_una))
{
/* Duplicate ACK */
if (++tcp_conn->tc_snd_dack ==
TCP_DACK_RETRANS)
{
tcp_fast_retrans(tcp_conn);
}
}
/* This ACK doesn't acknowledge any new data, this
* is a likely situation if we are only receiving
* data. We only update the window if we are
* actually sending or if we currently have a
* zero window.
*/
if (tcp_conn->tc_snd_cwnd == snd_una &&
seg_wnd != 0)
{
DBLOCK(2, printf("zero window opened\n"));
/* The other side opened up its receive
* window. */
mss= tcp_conn->tc_mtu-IP_TCP_MIN_HDR_SIZE;
if (seg_wnd > 2*mss)
seg_wnd= 2*mss;
tcp_conn->tc_snd_cwnd= snd_una+seg_wnd;
tcp_conn_write(tcp_conn, 1);
}
if (seg_wnd == 0)
{
tcp_conn->tc_snd_cwnd= tcp_conn->tc_SND_TRM=
snd_una;
}
}
else if (tcp_Lmod4G(snd_una, seg_ack) &&
tcp_LEmod4G(seg_ack, snd_nxt))
{
tcp_release_retrans(tcp_conn, seg_ack, seg_wnd);
if (tcp_conn->tc_state == TCS_CLOSED)
break;
}
else if (tcp_Gmod4G(seg_ack,
snd_nxt))
{
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
DBLOCK(1, printf(
"got an ack of something I haven't send\n");
printf( "seg_ack= %lu, SND_NXT= %lu\n",
seg_ack, snd_nxt));
break;
}
/*
process data...
*/
tcp_extract_ipopt(tcp_conn, ip_hdr);
tcp_extract_tcpopt(tcp_conn, tcp_hdr, &mss);
if (data_len)
{
if (tcp_LEmod4G(seg_seq, tcp_conn->tc_RCV_NXT))
{
process_data (tcp_conn, tcp_hdr,
tcp_data, data_len);
}
else
{
process_advanced_data (tcp_conn,
tcp_hdr, tcp_data, data_len);
}
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
/* Don't process a FIN if we got new data */
break;
}
/*
FIN ?
reply pending receives
advace RCV.NXT over the FIN
<SEQ=SND.NXT><ACK=RCV.NXT><CTL=ACK>
state == ESTABLISHED ?
state= CLOSE-WAIT
state == FIN-WAIT-1 ?
state= CLOSING
state == FIN-WAIT-2 ?
state= TIME-WAIT
state == TIME-WAIT ?
restart the TIME-WAIT timer
exit
*/
if ((tcp_hdr_flags & THF_FIN) && tcp_LEmod4G(seg_seq,
tcp_conn->tc_RCV_NXT))
{
if (!(tcp_conn->tc_flags & TCF_FIN_RECV) &&
tcp_Lmod4G(tcp_conn->tc_RCV_NXT,
tcp_conn->tc_RCV_HI))
{
tcp_conn->tc_RCV_NXT++;
tcp_conn->tc_flags |= TCF_FIN_RECV;
}
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags & TFF_READ_IP))
{
tcp_fd_read(tcp_conn, 1);
}
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags & TFF_SEL_READ))
{
tcp_rsel_read(tcp_conn);
}
}
break;
default:
printf("tcp_frag2conn: unknown state ");
tcp_print_state(tcp_conn);
break;
}
if (tcp_data != NULL)
bf_afree(tcp_data);
}
PRIVATE void
process_data(tcp_conn, tcp_hdr, tcp_data, data_len)
tcp_conn_t *tcp_conn;
tcp_hdr_t *tcp_hdr;
acc_t *tcp_data;
int data_len;
{
u32_t lo_seq, hi_seq, urg_seq, seq_nr, adv_seq, nxt;
u32_t urgptr;
int tcp_hdr_flags;
unsigned int offset;
acc_t *tmp_data, *rcvd_data, *adv_data;
int len_diff;
assert(tcp_conn->tc_busy);
/* Note, tcp_data will be freed by the caller. */
assert (!(tcp_hdr->th_flags & THF_SYN));
seq_nr= ntohl(tcp_hdr->th_seq_nr);
urgptr= ntohs(tcp_hdr->th_urgptr);
tcp_data->acc_linkC++;
lo_seq= seq_nr;
tcp_hdr_flags= tcp_hdr->th_flags & TH_FLAGS_MASK;
if (tcp_Lmod4G(lo_seq, tcp_conn->tc_RCV_NXT))
{
DBLOCK(0x10,
printf("segment is a retransmission\n"));
offset= tcp_conn->tc_RCV_NXT-lo_seq;
tcp_data= bf_delhead(tcp_data, offset);
lo_seq += offset;
data_len -= offset;
if (tcp_hdr_flags & THF_URG)
{
printf("process_data: updating urgent pointer\n");
if (urgptr >= offset)
urgptr -= offset;
else
tcp_hdr_flags &= ~THF_URG;
}
}
assert (lo_seq == tcp_conn->tc_RCV_NXT);
if (tcp_hdr_flags & THF_URG)
{
if (!(tcp_conn->tc_flags & TCF_BSD_URG))
{
/* Update urgent pointer to point past the urgent
* data
*/
urgptr++;
}
if (urgptr == 0)
tcp_hdr_flags &= ~THF_URG;
}
if (tcp_hdr_flags & THF_URG)
{
if (urgptr > data_len)
urgptr= data_len;
urg_seq= lo_seq+urgptr;
if (tcp_GEmod4G(urg_seq, tcp_conn->tc_RCV_HI))
urg_seq= tcp_conn->tc_RCV_HI;
if (tcp_conn->tc_flags & TCF_BSD_URG)
{
if (tcp_Gmod4G(tcp_conn->tc_RCV_NXT,
tcp_conn->tc_RCV_LO))
{
DBLOCK(1, printf(
"ignoring urgent data\n"));
bf_afree(tcp_data);
/* Should set advertised window to
* zero */
/* Flush */
tcp_conn->tc_flags |= TCF_RCV_PUSH;
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags &
TFF_READ_IP))
{
tcp_fd_read(tcp_conn, 1);
}
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags &
TFF_SEL_READ))
{
tcp_rsel_read(tcp_conn);
}
return;
}
}
if (tcp_Gmod4G(urg_seq, tcp_conn->tc_RCV_UP))
tcp_conn->tc_RCV_UP= urg_seq;
#if 0
if (urgptr < data_len)
{
data_len= urgptr;
tmp_data= bf_cut(tcp_data, 0, data_len);
bf_afree(tcp_data);
tcp_data= tmp_data;
tcp_hdr_flags &= ~THF_FIN;
}
#endif
tcp_conn->tc_flags |= TCF_RCV_PUSH;
}
else
{
/* Normal data. */
}
if (tcp_hdr_flags & THF_PSH)
{
tcp_conn->tc_flags |= TCF_RCV_PUSH;
}
hi_seq= lo_seq+data_len;
if (tcp_Gmod4G(hi_seq, tcp_conn->tc_RCV_HI))
{
data_len= tcp_conn->tc_RCV_HI-lo_seq;
tmp_data= bf_cut(tcp_data, 0, data_len);
bf_afree(tcp_data);
tcp_data= tmp_data;
hi_seq= lo_seq+data_len;
tcp_hdr_flags &= ~THF_FIN;
}
assert (tcp_LEmod4G (hi_seq, tcp_conn->tc_RCV_HI));
rcvd_data= tcp_conn->tc_rcvd_data;
tcp_conn->tc_rcvd_data= 0;
tmp_data= bf_append(rcvd_data, tcp_data);
tcp_conn->tc_rcvd_data= tmp_data;
tcp_conn->tc_RCV_NXT= hi_seq;
if ((tcp_hdr_flags & THF_FIN) &&
tcp_Lmod4G(tcp_conn->tc_RCV_NXT, tcp_conn->tc_RCV_HI) &&
!(tcp_conn->tc_flags & TCF_FIN_RECV))
{
tcp_conn->tc_RCV_NXT++;
tcp_conn->tc_flags |= TCF_FIN_RECV;
}
if (tcp_conn->tc_fd && (tcp_conn->tc_fd->tf_flags & TFF_READ_IP))
tcp_fd_read(tcp_conn, 1);
if (tcp_conn->tc_fd && (tcp_conn->tc_fd->tf_flags & TFF_SEL_READ))
tcp_rsel_read(tcp_conn);
DIFBLOCK(2, (tcp_conn->tc_RCV_NXT == tcp_conn->tc_RCV_HI),
printf("conn[[%d] full receive buffer\n",
tcp_conn-tcp_conn_table));
if (tcp_conn->tc_adv_data == NULL)
return;
if (tcp_hdr_flags & THF_FIN)
{
printf("conn[%d]: advanced data after FIN\n",
tcp_conn-tcp_conn_table);
tcp_data= tcp_conn->tc_adv_data;
tcp_conn->tc_adv_data= NULL;
bf_afree(tcp_data);
return;
}
lo_seq= tcp_conn->tc_adv_seq;
if (tcp_Gmod4G(lo_seq, tcp_conn->tc_RCV_NXT))
return; /* Not yet */
tcp_data= tcp_conn->tc_adv_data;
tcp_conn->tc_adv_data= NULL;
data_len= bf_bufsize(tcp_data);
if (tcp_Lmod4G(lo_seq, tcp_conn->tc_RCV_NXT))
{
offset= tcp_conn->tc_RCV_NXT-lo_seq;
if (offset >= data_len)
{
bf_afree(tcp_data);
return;
}
tcp_data= bf_delhead(tcp_data, offset);
lo_seq += offset;
data_len -= offset;
}
assert (lo_seq == tcp_conn->tc_RCV_NXT);
hi_seq= lo_seq+data_len;
assert (tcp_LEmod4G (hi_seq, tcp_conn->tc_RCV_HI));
rcvd_data= tcp_conn->tc_rcvd_data;
tcp_conn->tc_rcvd_data= 0;
tmp_data= bf_append(rcvd_data, tcp_data);
tcp_conn->tc_rcvd_data= tmp_data;
tcp_conn->tc_RCV_NXT= hi_seq;
assert (tcp_conn->tc_RCV_LO + bf_bufsize(tcp_conn->tc_rcvd_data) ==
tcp_conn->tc_RCV_NXT ||
(tcp_print_conn(tcp_conn), printf("\n"), 0));
if (tcp_conn->tc_fd && (tcp_conn->tc_fd->tf_flags & TFF_READ_IP))
tcp_fd_read(tcp_conn, 1);
if (tcp_conn->tc_fd && (tcp_conn->tc_fd->tf_flags & TFF_SEL_READ))
tcp_rsel_read(tcp_conn);
adv_data= tcp_conn->tc_adv_data;
if (adv_data != NULL)
{
/* Try to use advanced data. */
adv_seq= tcp_conn->tc_adv_seq;
nxt= tcp_conn->tc_RCV_NXT;
if (tcp_Gmod4G(adv_seq, nxt))
return; /* not yet */
tcp_conn->tc_adv_data= NULL;
data_len= bf_bufsize(adv_data);
if (tcp_Lmod4G(adv_seq, nxt))
{
if (tcp_LEmod4G(adv_seq+data_len, nxt))
{
/* Data is not needed anymore. */
bf_afree(adv_data);
return;
}
len_diff= nxt-adv_seq;
adv_data= bf_delhead(adv_data, len_diff);
data_len -= len_diff;
}
DBLOCK(1, printf("using advanced data\n"));
/* Append data to the input buffer */
if (tcp_conn->tc_rcvd_data == NULL)
{
tcp_conn->tc_rcvd_data= adv_data;
}
else
{
tcp_conn->tc_rcvd_data=
bf_append(tcp_conn->tc_rcvd_data, adv_data);
}
tcp_conn->tc_SND_NXT += data_len;
assert(tcp_check_conn(tcp_conn));
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags & TFF_READ_IP))
{
tcp_fd_read(tcp_conn, 1);
}
if (tcp_conn->tc_fd &&
(tcp_conn->tc_fd->tf_flags & TFF_SEL_READ))
{
tcp_rsel_read(tcp_conn);
}
}
}
PRIVATE void process_advanced_data(tcp_conn, tcp_hdr, tcp_data, data_len)
tcp_conn_t *tcp_conn;
tcp_hdr_t *tcp_hdr;
acc_t *tcp_data;
int data_len;
{
u32_t seq, adv_seq;
acc_t *adv_data;
assert(tcp_conn->tc_busy);
/* Note, tcp_data will be freed by the caller. */
/* Always send an ACK, this allows the sender to do a fast
* retransmit.
*/
tcp_conn->tc_flags |= TCF_SEND_ACK;
tcp_conn_write(tcp_conn, 1);
if (tcp_hdr->th_flags & THF_URG)
return; /* Urgent data is to complicated */
if (tcp_hdr->th_flags & THF_PSH)
tcp_conn->tc_flags |= TCF_RCV_PUSH;
seq= ntohl(tcp_hdr->th_seq_nr);
/* Make sure that the packet doesn't fall outside of the window
* we offered.
*/
if (tcp_Gmod4G(seq+data_len, tcp_conn->tc_RCV_HI))
return;
adv_data= tcp_conn->tc_adv_data;
adv_seq= tcp_conn->tc_adv_seq;
tcp_conn->tc_adv_data= NULL;
tcp_data->acc_linkC++;
if (adv_data == NULL)
{
adv_seq= seq;
adv_data= tcp_data;
}
else if (seq + data_len == adv_seq)
{
/* New data fits right before exiting data. */
adv_data= bf_append(tcp_data, adv_data);
adv_seq= seq;
}
else if (adv_seq + bf_bufsize(adv_data) == seq)
{
/* New data fits right after exiting data. */
adv_data= bf_append(adv_data, tcp_data);
}
else
{
/* New data doesn't fit. */
bf_afree(tcp_data);
}
tcp_conn->tc_adv_data= adv_data;
tcp_conn->tc_adv_seq= adv_seq;
}
PRIVATE void create_RST(tcp_conn, ip_hdr, tcp_hdr, data_len)
tcp_conn_t *tcp_conn;
ip_hdr_t *ip_hdr;
tcp_hdr_t *tcp_hdr;
int data_len;
{
acc_t *tmp_ipopt, *tmp_tcpopt, *tcp_pack;
acc_t *RST_acc;
ip_hdr_t *RST_ip_hdr;
tcp_hdr_t *RST_tcp_hdr;
size_t pack_size, ip_hdr_len, mss;
DBLOCK(0x10, printf("in create_RST, bad pack is:\n");
tcp_print_pack(ip_hdr, tcp_hdr); tcp_print_state(tcp_conn);
printf("\n"));
assert(tcp_conn->tc_busy);
/* Only send RST packets in reponse to actual data (or SYN, FIN)
* this solves a problem during connection shutdown. The problem
* is the follow senario: a senders closes the connection instead
* of doing a shutdown and waiting for the receiver to shutdown.
* The receiver is slow in processing the last data. After the
* sender has completely closed the connection, the receiver
* sends a window update which triggers the sender to send a
* RST. The receiver closes the connection in reponse to the RST.
*/
if ((tcp_hdr->th_flags & (THF_FIN|THF_SYN)) == 0 &&
data_len == 0)
{
#if DEBUG
{ printf("tcp_recv`create_RST: no data, no RST\n"); }
#endif
return;
}
tmp_ipopt= tcp_conn->tc_remipopt;
if (tmp_ipopt)
tmp_ipopt->acc_linkC++;
tmp_tcpopt= tcp_conn->tc_tcpopt;
if (tmp_tcpopt)
tmp_tcpopt->acc_linkC++;
tcp_extract_ipopt (tcp_conn, ip_hdr);
tcp_extract_tcpopt (tcp_conn, tcp_hdr, &mss);
RST_acc= tcp_make_header (tcp_conn, &RST_ip_hdr, &RST_tcp_hdr,
(acc_t *)0);
if (tcp_conn->tc_remipopt)
bf_afree(tcp_conn->tc_remipopt);
tcp_conn->tc_remipopt= tmp_ipopt;
if (tcp_conn->tc_tcpopt)
bf_afree(tcp_conn->tc_tcpopt);
tcp_conn->tc_tcpopt= tmp_tcpopt;
RST_ip_hdr->ih_src= ip_hdr->ih_dst;
RST_ip_hdr->ih_dst= ip_hdr->ih_src;
RST_tcp_hdr->th_srcport= tcp_hdr->th_dstport;
RST_tcp_hdr->th_dstport= tcp_hdr->th_srcport;
if (tcp_hdr->th_flags & THF_ACK)
{
RST_tcp_hdr->th_seq_nr= tcp_hdr->th_ack_nr;
RST_tcp_hdr->th_flags= THF_RST;
}
else
{
RST_tcp_hdr->th_seq_nr= 0;
RST_tcp_hdr->th_ack_nr=
htonl(
ntohl(tcp_hdr->th_seq_nr)+
data_len +
(tcp_hdr->th_flags & THF_SYN ? 1 : 0) +
(tcp_hdr->th_flags & THF_FIN ? 1 : 0));
RST_tcp_hdr->th_flags= THF_RST|THF_ACK;
}
pack_size= bf_bufsize(RST_acc);
RST_ip_hdr->ih_length= htons(pack_size);
RST_tcp_hdr->th_window= htons(tcp_conn->tc_rcv_wnd);
RST_tcp_hdr->th_chksum= 0;
RST_acc->acc_linkC++;
ip_hdr_len= (ip_hdr->ih_vers_ihl & IH_IHL_MASK) << 2;
tcp_pack= bf_delhead(RST_acc, ip_hdr_len);
RST_tcp_hdr->th_chksum= ~tcp_pack_oneCsum (RST_ip_hdr, tcp_pack);
bf_afree(tcp_pack);
DBLOCK(2, tcp_print_pack(ip_hdr, tcp_hdr); printf("\n");
tcp_print_pack(RST_ip_hdr, RST_tcp_hdr); printf("\n"));
if (tcp_conn->tc_frag2send)
bf_afree(tcp_conn->tc_frag2send);
tcp_conn->tc_frag2send= RST_acc;
tcp_conn_write(tcp_conn, 1);
}
PUBLIC void
tcp_fd_read(tcp_conn, enq)
tcp_conn_t *tcp_conn;
int enq; /* Enqueue writes. */
{
tcp_fd_t *tcp_fd;
size_t data_size, read_size;
acc_t *data;
int fin_recv, urg, push, result;
i32_t old_window, new_window;
u16_t mss;
assert(tcp_conn->tc_busy);
tcp_fd= tcp_conn->tc_fd;
assert (tcp_fd->tf_flags & TFF_READ_IP);
if (tcp_conn->tc_state == TCS_CLOSED)
{
if (tcp_fd->tf_read_offset)
tcp_reply_read (tcp_fd, tcp_fd->tf_read_offset);
else
tcp_reply_read (tcp_fd, tcp_conn->tc_error);
return;
}
urg= tcp_Gmod4G(tcp_conn->tc_RCV_UP, tcp_conn->tc_RCV_LO);
push= (tcp_conn->tc_flags & TCF_RCV_PUSH);
fin_recv= (tcp_conn->tc_flags & TCF_FIN_RECV);
data_size= tcp_conn->tc_RCV_NXT-tcp_conn->tc_RCV_LO;
if (fin_recv)
data_size--;
if (urg)
{
#if DEBUG
printf("tcp_fd_read: RCV_UP = 0x%x, RCV_LO = 0x%x\n",
tcp_conn->tc_RCV_UP, tcp_conn->tc_RCV_LO);
#endif
read_size= tcp_conn->tc_RCV_UP-tcp_conn->tc_RCV_LO;
}
else
read_size= data_size;
if (read_size >= tcp_fd->tf_read_count)
read_size= tcp_fd->tf_read_count;
else if (!push && !fin_recv && !urg &&
data_size < TCP_MIN_RCV_WND_SIZE)
{
/* Defer the copy out until later. */
return;
}
else if (data_size == 0 && !fin_recv)
{
/* No data, and no end of file. */
return;
}
if (read_size)
{
if (urg && !(tcp_fd->tf_flags & TFF_RECV_URG))
{
if (tcp_fd->tf_read_offset)
{
tcp_reply_read (tcp_fd,
tcp_fd->tf_read_offset);
}
else
{
tcp_reply_read (tcp_fd, EURG);
}
return;
}
else if (!urg && (tcp_fd->tf_flags & TFF_RECV_URG))
{
if (tcp_fd->tf_read_offset)
{
tcp_reply_read (tcp_fd,
tcp_fd->tf_read_offset);
}
else
{
tcp_reply_read(tcp_fd, ENOURG);
}
return;
}
if (read_size == data_size)
{
data= tcp_conn->tc_rcvd_data;
data->acc_linkC++;
}
else
{
data= bf_cut(tcp_conn->tc_rcvd_data, 0, read_size);
}
result= (*tcp_fd->tf_put_userdata) (tcp_fd->tf_srfd,
tcp_fd->tf_read_offset, data, FALSE);
if (result<0)
{
if (tcp_fd->tf_read_offset)
tcp_reply_read(tcp_fd, tcp_fd->
tf_read_offset);
else
tcp_reply_read(tcp_fd, result);
return;
}
tcp_fd->tf_read_offset += read_size;
tcp_fd->tf_read_count -= read_size;
if (data_size == read_size)
{
bf_afree(tcp_conn->tc_rcvd_data);
tcp_conn->tc_rcvd_data= 0;
}
else
{
tcp_conn->tc_rcvd_data=
bf_delhead(tcp_conn->tc_rcvd_data,
read_size);
}
tcp_conn->tc_RCV_LO += read_size;
data_size -= read_size;
}
/* Update IRS and often RCV_UP every 0.5GB */
if (tcp_conn->tc_RCV_LO - tcp_conn->tc_IRS > 0x40000000)
{
tcp_conn->tc_IRS += 0x20000000;
DBLOCK(1, printf("tcp_fd_read: updating IRS to 0x%lx\n",
(unsigned long)tcp_conn->tc_IRS););
if (tcp_Lmod4G(tcp_conn->tc_RCV_UP, tcp_conn->tc_IRS))
{
tcp_conn->tc_RCV_UP= tcp_conn->tc_IRS;
DBLOCK(1, printf(
"tcp_fd_read: updating RCV_UP to 0x%lx\n",
(unsigned long)tcp_conn->tc_RCV_UP););
}
DBLOCK(1, printf("tcp_fd_read: RCP_LO = 0x%lx\n",
(unsigned long)tcp_conn->tc_RCV_LO););
}
mss= tcp_conn->tc_mtu-IP_TCP_MIN_HDR_SIZE;
if (tcp_conn->tc_RCV_HI-tcp_conn->tc_RCV_LO <=
tcp_conn->tc_rcv_wnd-mss)
{
old_window= tcp_conn->tc_RCV_HI-tcp_conn->tc_RCV_NXT;
tcp_conn->tc_RCV_HI= tcp_conn->tc_RCV_LO +
tcp_conn->tc_rcv_wnd;
new_window= tcp_conn->tc_RCV_HI-tcp_conn->tc_RCV_NXT;
assert(old_window >=0 && new_window >= old_window);
if (old_window < mss && new_window >= mss)
{
tcp_conn->tc_flags |= TCF_SEND_ACK;
DBLOCK(2, printf("opening window\n"));
tcp_conn_write(tcp_conn, 1);
}
}
if (tcp_conn->tc_rcvd_data == NULL &&
tcp_conn->tc_adv_data == NULL)
{
/* Out of data, clear PUSH flag and reply to a read. */
tcp_conn->tc_flags &= ~TCF_RCV_PUSH;
}
if (fin_recv || urg || tcp_fd->tf_read_offset ||
!tcp_fd->tf_read_count)
{
tcp_reply_read (tcp_fd, tcp_fd->tf_read_offset);
return;
}
}
PUBLIC unsigned
tcp_sel_read(tcp_conn)
tcp_conn_t *tcp_conn;
{
tcp_fd_t *tcp_fd;
size_t data_size;
int fin_recv, urg, push;
tcp_fd= tcp_conn->tc_fd;
if (tcp_conn->tc_state == TCS_CLOSED)
return 1;
fin_recv= (tcp_conn->tc_flags & TCF_FIN_RECV);
if (fin_recv)
return 1;
data_size= tcp_conn->tc_RCV_NXT-tcp_conn->tc_RCV_LO;
if (data_size == 0)
{
/* No data, and no end of file. */
return 0;
}
urg= tcp_Gmod4G(tcp_conn->tc_RCV_UP, tcp_conn->tc_RCV_LO);
push= (tcp_conn->tc_flags & TCF_RCV_PUSH);
if (!push && !urg && data_size < TCP_MIN_RCV_WND_SIZE)
{
/* Defer until later. */
return 0;
}
return 1;
}
PUBLIC void
tcp_rsel_read(tcp_conn)
tcp_conn_t *tcp_conn;
{
tcp_fd_t *tcp_fd;
if (tcp_sel_read(tcp_conn) == 0)
return;
tcp_fd= tcp_conn->tc_fd;
tcp_fd->tf_flags &= ~TFF_SEL_READ;
if (tcp_fd->tf_select_res)
tcp_fd->tf_select_res(tcp_fd->tf_srfd, SR_SELECT_READ);
else
printf("tcp_rsel_read: no select_res\n");
}
PUBLIC void tcp_bytesavailable(tcp_fd, bytesp)
tcp_fd_t *tcp_fd;
int *bytesp;
{
tcp_conn_t *tcp_conn;
size_t data_size, read_size;
acc_t *data;
int fin_recv, urg, push, result;
i32_t old_window, new_window;
u16_t mss;
*bytesp= 0; /* The default is that nothing is available */
if (!(tcp_fd->tf_flags & TFF_CONNECTED))
return;
tcp_conn= tcp_fd->tf_conn;
if (tcp_conn->tc_state == TCS_CLOSED)
return;
urg= tcp_Gmod4G(tcp_conn->tc_RCV_UP, tcp_conn->tc_RCV_LO);
push= (tcp_conn->tc_flags & TCF_RCV_PUSH);
fin_recv= (tcp_conn->tc_flags & TCF_FIN_RECV);
data_size= tcp_conn->tc_RCV_NXT-tcp_conn->tc_RCV_LO;
if (fin_recv)
data_size--;
if (urg)
data_size= tcp_conn->tc_RCV_UP-tcp_conn->tc_RCV_LO;
if (urg && !(tcp_fd->tf_flags & TFF_RECV_URG))
return;
else if (!urg && (tcp_fd->tf_flags & TFF_RECV_URG))
return;
*bytesp= data_size;
}
/*
* $PchId: tcp_recv.c,v 1.30 2005/06/28 14:21:35 philip Exp $
*/