1483 lines
32 KiB
C
1483 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= %u, SND_NXT= %u\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;
|
|
int fin_recv, urg, push;
|
|
|
|
*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 $
|
|
*/
|