Update nfsrv_getsocksndseq() for changes in TCP internals since FreeBSD 6.x:

- so_pcb is now guaranteed to be non-NULL and valid if a valid socket
  reference is held.

- Need to check INP_TIMEWAIT and INP_DROPPED before assuming inp_ppcb is a
  tcpcb, as it might be a tcptw or NULL otherwise.

- tp can never be NULL by the end of the function, so only check
  TCPS_ESTABLISHED before extracting tcpcb fields.

The NFS server arguably incorporates too many assumptions about TCP
internals, but fixing that is left for nother day.

MFC after:		1 week
Reviewed by:		bz
Reviewed and tested by:	rmacklem
Sponsored by:		Juniper Networks
This commit is contained in:
Robert Watson 2010-03-11 11:33:04 +00:00
parent e8b9127d7e
commit 2684bef615

View File

@ -2671,24 +2671,23 @@ nfsrv_getsocksndseq(struct socket *so, tcp_seq *maxp, tcp_seq *unap)
{ {
struct inpcb *inp; struct inpcb *inp;
struct tcpcb *tp; struct tcpcb *tp;
int error = EPIPE;
INP_INFO_RLOCK(&V_tcbinfo);
inp = sotoinpcb(so); inp = sotoinpcb(so);
if (inp == NULL) { KASSERT(inp != NULL, ("nfsrv_getsocksndseq: inp == NULL"));
INP_INFO_RUNLOCK(&V_tcbinfo);
return (error);
}
INP_RLOCK(inp); INP_RLOCK(inp);
INP_INFO_RUNLOCK(&V_tcbinfo); if (inp->inp_flags & (INP_TIMEWAIT | INP_DROPPED)) {
tp = intotcpcb(inp); INP_RUNLOCK(inp);
if (tp != NULL && tp->t_state == TCPS_ESTABLISHED) { return (EPIPE);
*maxp = tp->snd_max;
*unap = tp->snd_una;
error = 0;
} }
tp = intotcpcb(inp);
if (tp->t_state != TCPS_ESTABLISHED) {
INP_RUNLOCK(inp);
return (EPIPE);
}
*maxp = tp->snd_max;
*unap = tp->snd_una;
INP_RUNLOCK(inp); INP_RUNLOCK(inp);
return (error); return (0);
} }
/* /*