Date: Tue, 27 Nov 2007 06:36:37 GMT From: Kip Macy <kmacy@FreeBSD.org> To: Perforce Change Reviews <perforce@freebsd.org> Subject: PERFORCE change 129611 for review Message-ID: <200711270636.lAR6abVJ003056@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=129611 Change 129611 by kmacy@kmacy:storage:toestack on 2007/11/27 06:36:32 make locking a bit more consistent fix socket buffer accounting in t3_push_frames by only skipping the last buffer if sndptroff is not zero Affected files ... .. //depot/projects/toestack/sys/dev/cxgb/ulp/tom/cxgb_cpl_io.c#27 edit Differences ... ==== //depot/projects/toestack/sys/dev/cxgb/ulp/tom/cxgb_cpl_io.c#27 (text+ko) ==== @@ -170,10 +170,13 @@ send_or_defer(struct socket *so, struct tcpcb *tp, struct mbuf *m, int through_l2t) { struct toepcb *toep = tp->t_toe; + - if (__predict_false(tp->t_state == TCPS_SYN_SENT)) + if (__predict_false(tp->t_state == TCPS_SYN_SENT)) { + INP_LOCK(tp->t_inpcb); mbufq_tail(&toep->out_of_order_queue, m); // defer - else if (through_l2t) + INP_UNLOCK(tp->t_inpcb); + } else if (through_l2t) l2t_send(T3C_DEV(so), m, toep->tp_l2t); // send through L2T else cxgb_ofld_send(T3C_DEV(so), m); // send directly @@ -207,11 +210,12 @@ struct toepcb *toep = tp->t_toe; struct tx_data_wr *req; + INP_LOCK_ASSERT(tp->t_inpcb); + req = mtod(m, struct tx_data_wr *); m->m_len = sizeof(*req); req->wr_hi = htonl(V_WR_OP(FW_WROPCODE_OFLD_TX_DATA)); req->wr_lo = htonl(V_WR_TID(toep->tp_tid)); - req->sndseq = htonl(tp->snd_nxt); /* len includes the length of any HW ULP additions */ req->len = htonl(len); req->param = htonl(V_TX_PORT(toep->tp_l2t->smt_idx)); @@ -220,7 +224,7 @@ V_TX_URG(/* skb_urgent(skb) */ 0 ) | V_TX_SHOVE((!(tp->t_flags & TF_MORETOCOME) && (tail ? 0 : 1)))); - + req->sndseq = htonl(tp->snd_nxt); if (__predict_false((toep->tp_flags & TP_DATASENT) == 0)) { req->flags |= htonl(V_TX_ACK_PAGES(2) | F_TX_INIT | V_TX_CPU_IDX(toep->tp_qset)); @@ -259,16 +263,18 @@ return (0); } + INP_LOCK_ASSERT(tp->t_inpcb); + SOCKBUF_LOCK(&so->so_snd); d = TOM_DATA(TOE_DEV(so)); cdev = d->cdev; last = tail = so->so_snd.sb_sndptr ? so->so_snd.sb_sndptr : so->so_snd.sb_mb; total_bytes = 0; - printf("tail=%p snd.cc=%d tp_last=%p\n", tail, so->so_snd.sb_cc, - toep->tp_m_last); + DPRINTF("wr_avail=%d tail=%p snd.cc=%d tp_last=%p\n", + toep->tp_wr_avail, tail, so->so_snd.sb_cc, toep->tp_m_last); - if (last && toep->tp_m_last == last) { + if (last && toep->tp_m_last == last && so->so_snd.sb_sndptroff != 0) { KASSERT(tail, ("sbdrop error")); last = tail = tail->m_next; } @@ -279,16 +285,15 @@ return (0); } + toep->tp_m_last = NULL; while (toep->tp_wr_avail && (tail != NULL)) { - - count = bytes = 0; if ((m0 = m_gethdr(M_NOWAIT, MT_DATA)) == NULL) { SOCKBUF_UNLOCK(&so->so_snd); return (0); } - - while ((mbuf_wrs[count + 1] <= toep->tp_wr_avail) && (tail != NULL) && (count < TX_MAX_SEGS)) { + while ((mbuf_wrs[count + 1] <= toep->tp_wr_avail) + && (tail != NULL) && (count < TX_MAX_SEGS)) { bytes += tail->m_len; count++; last = tail; @@ -360,8 +365,6 @@ return (total_bytes); } - - /* * Close a connection by sending a CPL_CLOSE_CON_REQ message. Cannot fail * under any circumstances. We take the easy way out and always queue the @@ -475,6 +478,8 @@ if (!((tp->t_state == TCPS_ESTABLISHED) || (tp->t_state == TCPS_FIN_WAIT_1) || (tp->t_state == TCPS_FIN_WAIT_2))) return; + INP_LOCK_ASSERT(tp->t_inpcb); + so = tp->t_inpcb->inp_socket; SOCKBUF_LOCK(&so->so_rcv); read = toep->tp_enqueued_bytes - so->so_rcv.sb_cc; @@ -764,7 +769,7 @@ { struct toepcb *toep = sototoep(so); toepcb_hold(toep); - + cxgb_insert_tid(d->cdev, d->client, toep, tid); } @@ -843,6 +848,7 @@ if (!cdev) return; + INP_LOCK_ASSERT(tp->t_inpcb); toep->tp_qset = 0; t3_release_ddp_resources(so); @@ -945,8 +951,9 @@ { struct tcpcb *tp = sototcpcb(so); struct t3c_data *td = T3C_DATA(TOM_DATA(dev)->cdev); + + SOCK_LOCK_ASSERT(so); - printf("initializing offload socket\n"); so->so_rcv.sb_flags |= SB_TOE; so->so_snd.sb_flags |= SB_TOE; @@ -1070,7 +1077,8 @@ fail_act_open(struct socket *so, int errno) { struct tcpcb *tp = sototcpcb(so); - + + INP_LOCK_ASSERT(tp->t_inpcb); t3_release_offload_resources(so); tcp_drop(tp, errno); @@ -1193,18 +1201,23 @@ e = t3_l2t_get(d->cdev, dst, egress_ifp); if (!e) goto free_tid; - + + INP_LOCK_ASSERT(inp); m = m_gethdr(MT_DATA, M_WAITOK); #if 0 m->m_toe.mt_toepcb = tp->t_toe; set_arp_failure_handler((struct mbuf *)m, act_open_req_arp_failure); -#endif +#endif + SOCK_LOCK(so); + init_offload_socket(so, tdev, atid, e, dst, toep); install_offload_ops(so); mk_act_open_req(so, m, atid, e); + SOCK_UNLOCK(so); + soisconnecting(so); toep = tp->t_toe; m_set_toep(m, tp->t_toe); @@ -1239,9 +1252,9 @@ int mode = CPL_ABORT_SEND_RST; struct mbuf *m; + INP_LOCK_ASSERT(tp->t_inpcb); if (__predict_false((toep->tp_flags & TP_ABORT_SHUTDOWN) || !TOE_DEV(so))) return; - toep->tp_flags |= (TP_ABORT_RPL_PENDING|TP_ABORT_SHUTDOWN); /* Purge the send queue so we don't send anything after an abort. */ @@ -1260,6 +1273,7 @@ req->rsvd0 = htonl(tp->snd_nxt); req->rsvd1 = !(toep->tp_flags & TP_DATASENT); req->cmd = mode; + INP_LOCK_ASSERT(tp->t_inpcb); if (tp->t_state == TCPS_SYN_SENT) mbufq_tail(&toep->out_of_order_queue, m); // defer else @@ -1440,21 +1454,7 @@ #endif SOCKBUF_LOCK(&so->so_rcv); if (sb_notify(&so->so_rcv)) - printf("rx_data so=%p flags=0x%x len=%d\n", so, so->so_rcv.sb_flags, m->m_pkthdr.len); - if (m->m_pkthdr.len == 80) { - int i; - uint64_t *data; - - data = mtod(m, uint64_t *); - printf("80 byte packet:\n"); - for (i = 0; i < 10; i++, data++) { - uint8_t *dp = (uint8_t *)data; - - printf("%02x%02x%02x%02x%02x%02x%02x%02x\n", - dp[0],dp[1],dp[2],dp[3],dp[4],dp[5],dp[6],dp[7]); - } - } - + DPRINTF("rx_data so=%p flags=0x%x len=%d\n", so, so->so_rcv.sb_flags, m->m_pkthdr.len); sbappendstream_locked(&so->so_rcv, m); INP_UNLOCK(tp->t_inpcb); @@ -1695,6 +1695,7 @@ { struct tcpcb *tp = sototcpcb(so); + INP_LOCK_ASSERT(tp->t_inpcb); /* * Bump rcv_nxt for the peer FIN. We don't do this at the time we * process peer_close because we don't want to carry the peer FIN in @@ -1709,7 +1710,6 @@ tcp_twstart(tp); } - /* * Handle a peer FIN. */ @@ -1741,6 +1741,8 @@ sk->sk_shutdown |= RCV_SHUTDOWN; sock_set_flag(so, SOCK_DONE); #endif + INP_INFO_WLOCK(&tcbinfo); + INP_LOCK(tp->t_inpcb); if (TCPS_HAVERCVDFIN(tp->t_state) == 0) socantrcvmore(so); switch (tp->t_state) { @@ -1765,11 +1767,7 @@ */ t3_release_offload_resources(so); if (toep->tp_flags & TP_ABORT_RPL_PENDING) { - INP_INFO_WLOCK(&tcbinfo); - INP_LOCK(tp->t_inpcb); tcp_close(tp); - INP_INFO_WUNLOCK(&tcbinfo); - INP_UNLOCK(tp->t_inpcb); } else enter_timewait(so); break; @@ -1778,6 +1776,8 @@ "%s: TID %u received PEER_CLOSE in bad state %d\n", TOE_DEV(so)->name, toep->tp_tid, tp->t_state); } + INP_INFO_WUNLOCK(&tcbinfo); + INP_UNLOCK(tp->t_inpcb); if (!dead) { printf("waking up waiters on %p rcv_notify=%d flags=0x%x\n", so, sb_notify(&so->so_rcv), so->so_rcv.sb_flags); @@ -1831,16 +1831,14 @@ if (!is_t3a(TOE_DEV(so)) && (toep->tp_flags & TP_ABORT_RPL_PENDING)) goto out; + INP_INFO_WLOCK(&tcbinfo); + INP_LOCK(inp); switch (tp->t_state) { case TCPS_CLOSING: /* see FIN_WAIT2 case in do_peer_fin */ t3_release_offload_resources(so); if (toep->tp_flags & TP_ABORT_RPL_PENDING) { - INP_INFO_WLOCK(&tcbinfo); - INP_LOCK(inp); tp = tcp_close(tp); - INP_INFO_WUNLOCK(&tcbinfo); - if (tp) - INP_UNLOCK(inp); + } else enter_timewait(so); break; @@ -1851,13 +1849,7 @@ * late, this close_con_rpl is the actual last message. */ t3_release_offload_resources(so); - INP_INFO_WLOCK(&tcbinfo); - INP_LOCK(inp); tp = tcp_close(tp); - INP_INFO_WUNLOCK(&tcbinfo); - if (tp) - INP_UNLOCK(inp); - break; case TCPS_FIN_WAIT_1: #ifdef notyet @@ -1885,6 +1877,9 @@ TOE_DEV(so)->name, toep->tp_tid, tp->t_state); } + INP_INFO_WUNLOCK(&tcbinfo); + if (tp) + INP_UNLOCK(inp); out: m_free(m); } @@ -1922,7 +1917,8 @@ "process_abort_rpl: GTS rpl pending %d", sock_flag(sk, ABORT_RPL_PENDING)); #endif - + INP_LOCK_ASSERT(tp->t_inpcb); + if (toep->tp_flags & TP_ABORT_RPL_PENDING) { if (!(toep->tp_flags & TP_ABORT_RPL_RCVD) && !is_t3a(TOE_DEV(so))) toep->tp_flags |= TP_ABORT_RPL_RCVD; @@ -1968,7 +1964,6 @@ } toep = (struct toepcb *)ctx; - so = toeptoso(toep); /* * Sometimes we've already closed the socket, e.g., a post-close @@ -1977,9 +1972,10 @@ * but FW turns the ABORT_REQ into a regular one and so we get * ABORT_RPL_RSS with status 0 and no socket. Only on T3A. */ - if (!so) + if (!toep) goto discard; + so = toeptoso(toep); toepcb_hold(toep); process_abort_rpl(so, m); toepcb_release(toep); @@ -2161,7 +2157,8 @@ const struct cpl_abort_req_rss *req = cplhdr(m); struct tcpcb *tp = sototcpcb(so); struct toepcb *toep = tp->t_toe; - + + INP_LOCK_ASSERT(tp->t_inpcb); if ((toep->tp_flags & TP_ABORT_REQ_RCVD) == 0) { toep->tp_flags |= (TP_ABORT_REQ_RCVD|TP_ABORT_SHUTDOWN); m_free(m); @@ -2745,6 +2742,8 @@ const struct t3c_data *td = T3C_DATA(T3C_DEV(so)); struct tcpcb *tp = sototcpcb(so); struct toepcb *toep = tp->t_toe; + + INP_LOCK_ASSERT(tp->t_inpcb); toep->tp_mss_clamp = td->mtus[G_TCPOPT_MSS(opt)] - 40; tp->t_flags |= G_TCPOPT_TSTAMP(opt) ? TF_RCVD_TSTMP : 0; @@ -2884,6 +2883,7 @@ } tp = sototcpcb(so); + INP_LOCK(tp->t_inpcb); so->so_snd.sb_flags |= SB_TOE; so->so_rcv.sb_flags |= SB_TOE; toep->tp_tp = tp; @@ -2906,7 +2906,7 @@ make_established(so, ntohl(req->snd_isn), ntohs(req->tcp_opt)); INP_INFO_WUNLOCK(&tcbinfo); - + INP_UNLOCK(tp->t_inpcb); soisconnected(so); #ifdef notyet @@ -2954,6 +2954,7 @@ printf("fixup_and_send_ofo\n"); + INP_LOCK_ASSERT(tp->t_inpcb); while ((m = mbufq_dequeue(&toep->out_of_order_queue)) != NULL) { /* * A variety of messages can be waiting but the fields we'll @@ -3192,9 +3193,7 @@ static void t3_reset_listen_child(struct socket *child) { - SOCK_LOCK(child); t3_send_reset(child); - SOCK_UNLOCK(child); } /*
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200711270636.lAR6abVJ003056>