#if ISFB31
#include "opt_inet.h"
#include "opt_tcpdebug.h"
#endif
#ifndef INET
#error The option TCPDEBUG requires option INET.
#endif
#if TCPDEBUG
#define PRUREQUESTS
#define TCPSTATES
#define TCPTIMERS
#define TANAMES
#endif
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/protosw.h>
#include <netinet/in.h>
#include <netinet/in_systm.h>
#include <netinet/ip.h>
#include <netinet/ip6.h>
#include <netinet/ip_var.h>
#include <netinet/tcp.h>
#include <netinet/tcp_fsm.h>
#include <netinet/tcp_timer.h>
#include <netinet/tcp_var.h>
#include <netinet/tcpip.h>
#include <netinet/tcp_debug.h>
#if TCPDEBUG
static int tcpconsdebug = 0;
#endif
static struct tcp_debug tcp_debug[TCP_NDEBUG];
static int tcp_debx;
void
tcp_trace(act, ostate, tp, ip, th, req)
short act, ostate;
struct tcpcb *tp;
#if INET6
void *ip;
#else
struct ip *ip;
#endif
struct tcphdr *th;
int req;
{
#if INET6
int isipv6 = (ip != NULL && ((struct ip *)ip)->ip_v == 6) ? 1 : 0;
#endif
tcp_seq seq, ack;
int len, flags;
struct tcp_debug *td = &tcp_debug[tcp_debx++];
if (tcp_debx == TCP_NDEBUG)
tcp_debx = 0;
td->td_time = iptime();
td->td_act = act;
td->td_ostate = ostate;
td->td_tcb = (caddr_t)tp;
if (tp)
td->td_cb = *tp;
else
bzero((caddr_t)&td->td_cb, sizeof (*tp));
if (ip) {
#if INET6
if (isipv6)
td->td_ip6 = *(struct ip6_hdr *)ip;
else
td->td_ip = *(struct ip *)ip;
#else
td->td_ip = *ip;
#endif
} else
#if INET6
bzero((caddr_t)&td->_td_ipx, sizeof (td->_td_ipx));
#else
bzero((caddr_t)&td->td_ip, sizeof (*ip));
#endif
if (th)
td->td_th = *th;
td->td_req = req;
#if TCPDEBUG
if (tcpconsdebug == 0)
return;
if (tp)
printf("%p %s:", tp, tcpstates[ostate]);
else
printf("???????? ");
printf("%s ", tanames[act]);
switch (act) {
case TA_INPUT:
case TA_OUTPUT:
case TA_DROP:
if (ip == 0)
break;
#if INET6
if (isipv6) {
len = ((struct ip6_hdr *)ip)->ip6_plen;
} else {
len = ((struct ip *)ip)->ip_len;
}
#else
len = ip->ip_len;
#endif
seq = th->th_seq;
ack = th->th_ack;
if (act == TA_OUTPUT) {
seq = ntohl(seq);
ack = ntohl(ack);
len = ntohs((u_short)len);
}
if (act == TA_OUTPUT)
len -= sizeof (struct tcphdr);
if (len)
printf("[%x..%x)", seq, seq+len);
else
printf("%x", seq);
printf("@%x, urp=%x", ack, th->th_urp);
flags = th->th_flags;
if (flags) {
char *cp = "<";
#define pf(f) { \
if (th->th_flags & TH_##f) { \
printf("%s%s", cp, #f); \
cp = ","; \
} \
}
pf(SYN); pf(ACK); pf(FIN); pf(RST); pf(PUSH); pf(URG);
printf(">");
}
break;
case TA_USER:
printf("%s", prurequests[req&0xff]);
if ((req & 0xff) == PRU_SLOWTIMO)
printf("<%s>", tcptimers[req>>8]);
break;
}
if (tp)
printf(" -> %s", tcpstates[tp->t_state]);
printf("\n");
if (tp == 0)
return;
printf(
"\trcv_(nxt,wnd,up) (%lx,%lx,%lx) snd_(una,nxt,max) (%lx,%lx,%lx)\n",
(u_long)tp->rcv_nxt, tp->rcv_wnd, (u_long)tp->rcv_up,
(u_long)tp->snd_una, (u_long)tp->snd_nxt, (u_long)tp->snd_max);
printf("\tsnd_(wl1,wl2,wnd) (%lx,%lx,%lx)\n",
(u_long)tp->snd_wl1, (u_long)tp->snd_wl2, tp->snd_wnd);
#endif
}