From owner-svn-src-head@FreeBSD.ORG Tue Jan 6 20:05:13 2015 Return-Path: Delivered-To: svn-src-head@freebsd.org Received: from mx1.freebsd.org (mx1.freebsd.org [IPv6:2001:1900:2254:206a::19:1]) (using TLSv1.2 with cipher AECDH-AES256-SHA (256/256 bits)) (No client certificate requested) by hub.freebsd.org (Postfix) with ESMTPS id A60E2D4F; Tue, 6 Jan 2015 20:05:13 +0000 (UTC) Received: from mail-lb0-x232.google.com (mail-lb0-x232.google.com [IPv6:2a00:1450:4010:c04::232]) (using TLSv1 with cipher ECDHE-RSA-RC4-SHA (128/128 bits)) (Client CN "smtp.gmail.com", Issuer "Google Internet Authority G2" (verified OK)) by mx1.freebsd.org (Postfix) with ESMTPS id C4873646FF; Tue, 6 Jan 2015 18:58:36 +0000 (UTC) Received: by mail-lb0-f178.google.com with SMTP id u14so1453732lbd.23; Tue, 06 Jan 2015 10:58:34 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20120113; h=mime-version:sender:in-reply-to:references:date:message-id:subject :from:to:cc:content-type; bh=R9cvpLIlukPrhs1rm4VZAAOJR/zNj2OpIGJdYatrrhU=; b=ipJb8gp86BzzObONzbHDkLBzecu7Ywj8UdjE56t4xoBdGhB4L4G3PDR94+zwtB7Xd8 A5u/QA2eC7XMUOCoehS/f+20wZGEWm5cyqe/+hRXZC3S1A4V0xyOP8BsorYoqyhuJFQZ GJRjMyCeqpYaUU5wYAxgXBm4EjC6WPh/R0S08PbgMp2DTQevp22dK60IyJwbz7KUzUnY BstgItxnr2E+BBnEKoxuWAQDhAdGQNxZEdk7jt9SM9okQFrzWsCzSbPcbTFW8OT9R/Wx OnUlPKjgug4SbijpRG+pE8/n+MDLgX0Va8VDGCsB7QWw1hBtuYMjGo/qa+INkKt5WiXG hEUQ== MIME-Version: 1.0 X-Received: by 10.112.125.41 with SMTP id mn9mr87774788lbb.80.1420570714674; Tue, 06 Jan 2015 10:58:34 -0800 (PST) Sender: rizzo.unipi@gmail.com Received: by 10.114.10.168 with HTTP; Tue, 6 Jan 2015 10:58:34 -0800 (PST) In-Reply-To: <201412312319.sBVNJHca031041@svn.freebsd.org> References: <201412312319.sBVNJHca031041@svn.freebsd.org> Date: Tue, 6 Jan 2015 19:58:34 +0100 X-Google-Sender-Auth: NeG45TrP-2S4RhHveommZSUTeE8 Message-ID: Subject: Re: svn commit: r276485 - in head/sys: conf dev/cxgbe modules/cxgbe/if_cxgbe From: Luigi Rizzo To: Navdeep Parhar Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: quoted-printable X-Content-Filtered-By: Mailman/MimeDel 2.1.18-1 Cc: "svn-src-head@freebsd.org" , "svn-src-all@freebsd.org" , "src-committers@freebsd.org" X-BeenThere: svn-src-head@freebsd.org X-Mailman-Version: 2.1.18-1 Precedence: list List-Id: SVN commit messages for the src tree for head/-current List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Tue, 06 Jan 2015 20:05:13 -0000 On Thu, Jan 1, 2015 at 12:19 AM, Navdeep Parhar wrote: > Author: np > Date: Wed Dec 31 23:19:16 2014 > New Revision: 276485 > URL: https://svnweb.freebsd.org/changeset/base/276485 > > Log: > cxgbe(4): major tx rework. > =E2=80=8BFYI, this commit has some unnamed unions (eg. in t4_mp_ring.c) which prevent the kernel from compiling with our stock gcc and its standard kernel build flags (specifically -std=3D...). Adding the following in the kernel config makeoptions COPTFLAGS=3D"-fms-extensions" seems to do the job I know it is unavoidable that we'll end up with gcc not working, =E2=80=8Bbut maybe we can still avoid unnamed unions. cheers luigi =E2=80=8B =E2=80=8B > > a) Front load as much work as possible in if_transmit, before any drive= r > lock or software queue has to get involved. > > b) Replace buf_ring with a brand new mp_ring (multiproducer ring). Thi= s > is specifically for the tx multiqueue model where one of the if_transmi= t > producer threads becomes the consumer and other producers carry on as > usual. mp_ring is implemented as standalone code and it should be > possible to use it in any driver with tx multiqueue. It also has: > - the ability to enqueue/dequeue multiple items. This might become > significant if packet batching is ever implemented. > - an abdication mechanism to allow a thread to give up writing tx > descriptors and have another if_transmit thread take over. A thread > that's writing tx descriptors can end up doing so for an unbounded > time period if a) there are other if_transmit threads continuously > feeding the sofware queue, and b) the chip keeps up with whatever the > thread is throwing at it. > - accurate statistics about interesting events even when the stats come > at the expense of additional branches/conditional code. > > The NIC txq lock is uncontested on the fast path at this point. I've > left it there for synchronization with the control events (interface > up/down, modload/unload). > > c) Add support for "type 1" coalescing work request in the normal NIC t= x > path. This work request is optimized for frames with a single item in > the DMA gather list. These are very common when forwarding packets. > Note that netmap tx in cxgbe already uses these "type 1" work requests. > > d) Do not request automatic cidx updates every 32 descriptors. Instead= , > request updates via bits in individual work requests (still every 32 > descriptors approximately). Also, request an automatic final update > when the queue idles after activity. This means NIC tx reclaim is stil= l > performed lazily but it will catch up quickly as soon as the queue > idles. This seems to be the best middle ground and I'll probably do > something similar for netmap tx as well. > > e) Implement a faster tx path for WRQs (used by TOE tx and control > queues, _not_ by the normal NIC tx). Allow work requests to be written > directly to the hardware descriptor ring if room is available. I will > convert t4_tom and iw_cxgbe modules to this faster style gradually. > > MFC after: 2 months > > Added: > head/sys/dev/cxgbe/t4_mp_ring.c (contents, props changed) > head/sys/dev/cxgbe/t4_mp_ring.h (contents, props changed) > Modified: > head/sys/conf/files > head/sys/dev/cxgbe/adapter.h > head/sys/dev/cxgbe/t4_l2t.c > head/sys/dev/cxgbe/t4_main.c > head/sys/dev/cxgbe/t4_sge.c > head/sys/modules/cxgbe/if_cxgbe/Makefile > > Modified: head/sys/conf/files > > =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D > --- head/sys/conf/files Wed Dec 31 22:52:43 2014 (r276484) > +++ head/sys/conf/files Wed Dec 31 23:19:16 2014 (r276485) > @@ -1142,6 +1142,8 @@ dev/cxgb/sys/uipc_mvec.c optional cxgb p > compile-with "${NORMAL_C} -I$S/dev/cxgb" > dev/cxgb/cxgb_t3fw.c optional cxgb cxgb_t3fw \ > compile-with "${NORMAL_C} -I$S/dev/cxgb" > +dev/cxgbe/t4_mp_ring.c optional cxgbe pci \ > + compile-with "${NORMAL_C} -I$S/dev/cxgbe" > dev/cxgbe/t4_main.c optional cxgbe pci \ > compile-with "${NORMAL_C} -I$S/dev/cxgbe" > dev/cxgbe/t4_netmap.c optional cxgbe pci \ > > Modified: head/sys/dev/cxgbe/adapter.h > > =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D > --- head/sys/dev/cxgbe/adapter.h Wed Dec 31 22:52:43 2014 > (r276484) > +++ head/sys/dev/cxgbe/adapter.h Wed Dec 31 23:19:16 2014 > (r276485) > @@ -152,7 +152,8 @@ enum { > CL_METADATA_SIZE =3D CACHE_LINE_SIZE, > > SGE_MAX_WR_NDESC =3D SGE_MAX_WR_LEN / EQ_ESIZE, /* max WR size in > desc */ > - TX_SGL_SEGS =3D 36, > + TX_SGL_SEGS =3D 39, > + TX_SGL_SEGS_TSO =3D 38, > TX_WR_FLITS =3D SGE_MAX_WR_LEN / 8 > }; > > @@ -273,6 +274,7 @@ struct port_info { > struct timeval last_refreshed; > struct port_stats stats; > u_int tnl_cong_drops; > + u_int tx_parse_error; > > eventhandler_tag vlan_c; > > @@ -308,23 +310,9 @@ struct tx_desc { > __be64 flit[8]; > }; > > -struct tx_map { > - struct mbuf *m; > - bus_dmamap_t map; > -}; > - > -/* DMA maps used for tx */ > -struct tx_maps { > - struct tx_map *maps; > - uint32_t map_total; /* # of DMA maps */ > - uint32_t map_pidx; /* next map to be used */ > - uint32_t map_cidx; /* reclaimed up to this index */ > - uint32_t map_avail; /* # of available maps */ > -}; > - > struct tx_sdesc { > + struct mbuf *m; /* m_nextpkt linked chain of frames */ > uint8_t desc_used; /* # of hardware descriptors used by the > WR */ > - uint8_t credits; /* NIC txq: # of frames sent out in the W= R > */ > }; > > > @@ -378,16 +366,12 @@ struct sge_iq { > enum { > EQ_CTRL =3D 1, > EQ_ETH =3D 2, > -#ifdef TCP_OFFLOAD > EQ_OFLD =3D 3, > -#endif > > /* eq flags */ > - EQ_TYPEMASK =3D 7, /* 3 lsbits hold the type */ > - EQ_ALLOCATED =3D (1 << 3), /* firmware resources allocated= */ > - EQ_DOOMED =3D (1 << 4), /* about to be destroyed */ > - EQ_CRFLUSHED =3D (1 << 5), /* expecting an update from SGE= */ > - EQ_STALLED =3D (1 << 6), /* out of hw descriptors or > dmamaps */ > + EQ_TYPEMASK =3D 0x3, /* 2 lsbits hold the type (see > above) */ > + EQ_ALLOCATED =3D (1 << 2), /* firmware resources allocated= */ > + EQ_ENABLED =3D (1 << 3), /* open for business */ > }; > > /* Listed in order of preference. Update t4_sysctls too if you change > these */ > @@ -402,32 +386,25 @@ enum {DOORBELL_UDB, DOORBELL_WCWR, DOORB > struct sge_eq { > unsigned int flags; /* MUST be first */ > unsigned int cntxt_id; /* SGE context id for the eq */ > - bus_dma_tag_t desc_tag; > - bus_dmamap_t desc_map; > - char lockname[16]; > struct mtx eq_lock; > > struct tx_desc *desc; /* KVA of descriptor ring */ > - bus_addr_t ba; /* bus address of descriptor ring */ > - struct sge_qstat *spg; /* status page, for convenience */ > uint16_t doorbells; > volatile uint32_t *udb; /* KVA of doorbell (lies within BAR2) */ > u_int udb_qid; /* relative qid within the doorbell page = */ > - uint16_t cap; /* max # of desc, for convenience */ > - uint16_t avail; /* available descriptors, for convenience > */ > - uint16_t qsize; /* size (# of entries) of the queue */ > + uint16_t sidx; /* index of the entry with the status pag= e > */ > uint16_t cidx; /* consumer idx (desc idx) */ > uint16_t pidx; /* producer idx (desc idx) */ > - uint16_t pending; /* # of descriptors used since last > doorbell */ > + uint16_t equeqidx; /* EQUEQ last requested at this pidx */ > + uint16_t dbidx; /* pidx of the most recent doorbell */ > uint16_t iqid; /* iq that gets egr_update for the eq */ > uint8_t tx_chan; /* tx channel used by the eq */ > - struct task tx_task; > - struct callout tx_callout; > + volatile u_int equiq; /* EQUIQ outstanding */ > > - /* stats */ > - > - uint32_t egr_update; /* # of SGE_EGR_UPDATE notifications for > eq */ > - uint32_t unstalled; /* recovered from stall */ > + bus_dma_tag_t desc_tag; > + bus_dmamap_t desc_map; > + bus_addr_t ba; /* bus address of descriptor ring */ > + char lockname[16]; > }; > > struct sw_zone_info { > @@ -499,18 +476,19 @@ struct sge_fl { > struct cluster_layout cll_alt; /* alternate refill zone, layout = */ > }; > > +struct mp_ring; > + > /* txq: SGE egress queue + what's needed for Ethernet NIC */ > struct sge_txq { > struct sge_eq eq; /* MUST be first */ > > struct ifnet *ifp; /* the interface this txq belongs to */ > - bus_dma_tag_t tx_tag; /* tag for transmit buffers */ > - struct buf_ring *br; /* tx buffer ring */ > + struct mp_ring *r; /* tx software ring */ > struct tx_sdesc *sdesc; /* KVA of software descriptor ring */ > - struct mbuf *m; /* held up due to temporary resource > shortage */ > - > - struct tx_maps txmaps; > + struct sglist *gl; > + __be32 cpl_ctrl0; /* for convenience */ > > + struct task tx_reclaim_task; > /* stats for common events first */ > > uint64_t txcsum; /* # of times hardware assisted with > checksum */ > @@ -519,13 +497,12 @@ struct sge_txq { > uint64_t imm_wrs; /* # of work requests with immediate data > */ > uint64_t sgl_wrs; /* # of work requests with direct SGL */ > uint64_t txpkt_wrs; /* # of txpkt work requests (not > coalesced) */ > - uint64_t txpkts_wrs; /* # of coalesced tx work requests */ > - uint64_t txpkts_pkts; /* # of frames in coalesced tx work > requests */ > + uint64_t txpkts0_wrs; /* # of type0 coalesced tx work requests = */ > + uint64_t txpkts1_wrs; /* # of type1 coalesced tx work requests = */ > + uint64_t txpkts0_pkts; /* # of frames in type0 coalesced tx WRs = */ > + uint64_t txpkts1_pkts; /* # of frames in type1 coalesced tx WRs = */ > > /* stats for not-that-common events */ > - > - uint32_t no_dmamap; /* no DMA map to load the mbuf */ > - uint32_t no_desc; /* out of hardware descriptors */ > } __aligned(CACHE_LINE_SIZE); > > /* rxq: SGE ingress queue + SGE free list + miscellaneous items */ > @@ -574,7 +551,13 @@ struct wrqe { > STAILQ_ENTRY(wrqe) link; > struct sge_wrq *wrq; > int wr_len; > - uint64_t wr[] __aligned(16); > + char wr[] __aligned(16); > +}; > + > +struct wrq_cookie { > + TAILQ_ENTRY(wrq_cookie) link; > + int ndesc; > + int pidx; > }; > > /* > @@ -585,17 +568,32 @@ struct sge_wrq { > struct sge_eq eq; /* MUST be first */ > > struct adapter *adapter; > + struct task wrq_tx_task; > + > + /* Tx desc reserved but WR not "committed" yet. */ > + TAILQ_HEAD(wrq_incomplete_wrs , wrq_cookie) incomplete_wrs; > > - /* List of WRs held up due to lack of tx descriptors */ > + /* List of WRs ready to go out as soon as descriptors are > available. */ > STAILQ_HEAD(, wrqe) wr_list; > + u_int nwr_pending; > + u_int ndesc_needed; > > /* stats for common events first */ > > - uint64_t tx_wrs; /* # of tx work requests */ > + uint64_t tx_wrs_direct; /* # of WRs written directly to desc ring= . > */ > + uint64_t tx_wrs_ss; /* # of WRs copied from scratch space. */ > + uint64_t tx_wrs_copied; /* # of WRs queued and copied to desc > ring. */ > > /* stats for not-that-common events */ > > - uint32_t no_desc; /* out of hardware descriptors */ > + /* > + * Scratch space for work requests that wrap around after reachin= g > the > + * status page, and some infomation about the last WR that used i= t. > + */ > + uint16_t ss_pidx; > + uint16_t ss_len; > + uint8_t ss[SGE_MAX_WR_LEN]; > + > } __aligned(CACHE_LINE_SIZE); > > > @@ -744,7 +742,7 @@ struct adapter { > struct sge sge; > int lro_timeout; > > - struct taskqueue *tq[NCHAN]; /* taskqueues that flush data out > */ > + struct taskqueue *tq[NCHAN]; /* General purpose taskqueues */ > struct port_info *port[MAX_NPORTS]; > uint8_t chan_map[NCHAN]; > > @@ -978,12 +976,11 @@ static inline int > tx_resume_threshold(struct sge_eq *eq) > { > > - return (eq->qsize / 4); > + /* not quite the same as qsize / 4, but this will do. */ > + return (eq->sidx / 4); > } > > /* t4_main.c */ > -void t4_tx_task(void *, int); > -void t4_tx_callout(void *); > int t4_os_find_pci_capability(struct adapter *, int); > int t4_os_pci_save_state(struct adapter *); > int t4_os_pci_restore_state(struct adapter *); > @@ -1024,16 +1021,15 @@ int t4_setup_adapter_queues(struct adapt > int t4_teardown_adapter_queues(struct adapter *); > int t4_setup_port_queues(struct port_info *); > int t4_teardown_port_queues(struct port_info *); > -int t4_alloc_tx_maps(struct tx_maps *, bus_dma_tag_t, int, int); > -void t4_free_tx_maps(struct tx_maps *, bus_dma_tag_t); > void t4_intr_all(void *); > void t4_intr(void *); > void t4_intr_err(void *); > void t4_intr_evt(void *); > void t4_wrq_tx_locked(struct adapter *, struct sge_wrq *, struct wrqe *)= ; > -int t4_eth_tx(struct ifnet *, struct sge_txq *, struct mbuf *); > void t4_update_fl_bufsize(struct ifnet *); > -int can_resume_tx(struct sge_eq *); > +int parse_pkt(struct mbuf **); > +void *start_wrq_wr(struct sge_wrq *, int, struct wrq_cookie *); > +void commit_wrq_wr(struct sge_wrq *, void *, struct wrq_cookie *); > > /* t4_tracer.c */ > struct t4_tracer; > > Modified: head/sys/dev/cxgbe/t4_l2t.c > > =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D > --- head/sys/dev/cxgbe/t4_l2t.c Wed Dec 31 22:52:43 2014 (r276484) > +++ head/sys/dev/cxgbe/t4_l2t.c Wed Dec 31 23:19:16 2014 (r276485) > @@ -113,16 +113,15 @@ found: > int > t4_write_l2e(struct adapter *sc, struct l2t_entry *e, int sync) > { > - struct wrqe *wr; > + struct wrq_cookie cookie; > struct cpl_l2t_write_req *req; > int idx =3D e->idx + sc->vres.l2t.start; > > mtx_assert(&e->lock, MA_OWNED); > > - wr =3D alloc_wrqe(sizeof(*req), &sc->sge.mgmtq); > - if (wr =3D=3D NULL) > + req =3D start_wrq_wr(&sc->sge.mgmtq, howmany(sizeof(*req), 16), > &cookie); > + if (req =3D=3D NULL) > return (ENOMEM); > - req =3D wrtod(wr); > > INIT_TP_WR(req, 0); > OPCODE_TID(req) =3D htonl(MK_OPCODE_TID(CPL_L2T_WRITE_REQ, idx | > @@ -132,7 +131,7 @@ t4_write_l2e(struct adapter *sc, struct > req->vlan =3D htons(e->vlan); > memcpy(req->dst_mac, e->dmac, sizeof(req->dst_mac)); > > - t4_wrq_tx(sc, wr); > + commit_wrq_wr(&sc->sge.mgmtq, req, &cookie); > > if (sync && e->state !=3D L2T_STATE_SWITCHING) > e->state =3D L2T_STATE_SYNC_WRITE; > > Modified: head/sys/dev/cxgbe/t4_main.c > > =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D > --- head/sys/dev/cxgbe/t4_main.c Wed Dec 31 22:52:43 2014 > (r276484) > +++ head/sys/dev/cxgbe/t4_main.c Wed Dec 31 23:19:16 2014 > (r276485) > @@ -66,6 +66,7 @@ __FBSDID("$FreeBSD$"); > #include "common/t4_regs_values.h" > #include "t4_ioctl.h" > #include "t4_l2t.h" > +#include "t4_mp_ring.h" > > /* T4 bus driver interface */ > static int t4_probe(device_t); > @@ -378,7 +379,8 @@ static void build_medialist(struct port_ > static int cxgbe_init_synchronized(struct port_info *); > static int cxgbe_uninit_synchronized(struct port_info *); > static int setup_intr_handlers(struct adapter *); > -static void quiesce_eq(struct adapter *, struct sge_eq *); > +static void quiesce_txq(struct adapter *, struct sge_txq *); > +static void quiesce_wrq(struct adapter *, struct sge_wrq *); > static void quiesce_iq(struct adapter *, struct sge_iq *); > static void quiesce_fl(struct adapter *, struct sge_fl *); > static int t4_alloc_irq(struct adapter *, struct irq *, int rid, > @@ -434,7 +436,6 @@ static int sysctl_tx_rate(SYSCTL_HANDLER > static int sysctl_ulprx_la(SYSCTL_HANDLER_ARGS); > static int sysctl_wcwr_stats(SYSCTL_HANDLER_ARGS); > #endif > -static inline void txq_start(struct ifnet *, struct sge_txq *); > static uint32_t fconf_to_mode(uint32_t); > static uint32_t mode_to_fconf(uint32_t); > static uint32_t fspec_to_fconf(struct t4_filter_specification *); > @@ -1429,67 +1430,36 @@ cxgbe_transmit(struct ifnet *ifp, struct > { > struct port_info *pi =3D ifp->if_softc; > struct adapter *sc =3D pi->adapter; > - struct sge_txq *txq =3D &sc->sge.txq[pi->first_txq]; > - struct buf_ring *br; > + struct sge_txq *txq; > + void *items[1]; > int rc; > > M_ASSERTPKTHDR(m); > + MPASS(m->m_nextpkt =3D=3D NULL); /* not quite ready for this y= et */ > > if (__predict_false(pi->link_cfg.link_ok =3D=3D 0)) { > m_freem(m); > return (ENETDOWN); > } > > - /* check if flowid is set */ > - if (M_HASHTYPE_GET(m) !=3D M_HASHTYPE_NONE) > - txq +=3D ((m->m_pkthdr.flowid % (pi->ntxq - > pi->rsrv_noflowq)) > - + pi->rsrv_noflowq); > - br =3D txq->br; > - > - if (TXQ_TRYLOCK(txq) =3D=3D 0) { > - struct sge_eq *eq =3D &txq->eq; > - > - /* > - * It is possible that t4_eth_tx finishes up and releases > the > - * lock between the TRYLOCK above and the drbr_enqueue > here. We > - * need to make sure that this mbuf doesn't just sit ther= e > in > - * the drbr. > - */ > - > - rc =3D drbr_enqueue(ifp, br, m); > - if (rc =3D=3D 0 && callout_pending(&eq->tx_callout) =3D= =3D 0 && > - !(eq->flags & EQ_DOOMED)) > - callout_reset(&eq->tx_callout, 1, t4_tx_callout, > eq); > + rc =3D parse_pkt(&m); > + if (__predict_false(rc !=3D 0)) { > + MPASS(m =3D=3D NULL); /* was freed > already */ > + atomic_add_int(&pi->tx_parse_error, 1); /* rare, atomic i= s > ok */ > return (rc); > } > > - /* > - * txq->m is the mbuf that is held up due to a temporary shortage > of > - * resources and it should be put on the wire first. Then what's > in > - * drbr and finally the mbuf that was just passed in to us. > - * > - * Return code should indicate the fate of the mbuf that was > passed in > - * this time. > - */ > - > - TXQ_LOCK_ASSERT_OWNED(txq); > - if (drbr_needs_enqueue(ifp, br) || txq->m) { > - > - /* Queued for transmission. */ > - > - rc =3D drbr_enqueue(ifp, br, m); > - m =3D txq->m ? txq->m : drbr_dequeue(ifp, br); > - (void) t4_eth_tx(ifp, txq, m); > - TXQ_UNLOCK(txq); > - return (rc); > - } > + /* Select a txq. */ > + txq =3D &sc->sge.txq[pi->first_txq]; > + if (M_HASHTYPE_GET(m) !=3D M_HASHTYPE_NONE) > + txq +=3D ((m->m_pkthdr.flowid % (pi->ntxq - > pi->rsrv_noflowq)) + > + pi->rsrv_noflowq); > > - /* Direct transmission. */ > - rc =3D t4_eth_tx(ifp, txq, m); > - if (rc !=3D 0 && txq->m) > - rc =3D 0; /* held, will be transmitted soon (hopefully) *= / > + items[0] =3D m; > + rc =3D mp_ring_enqueue(txq->r, items, 1, 4096); > + if (__predict_false(rc !=3D 0)) > + m_freem(m); > > - TXQ_UNLOCK(txq); > return (rc); > } > > @@ -1499,17 +1469,17 @@ cxgbe_qflush(struct ifnet *ifp) > struct port_info *pi =3D ifp->if_softc; > struct sge_txq *txq; > int i; > - struct mbuf *m; > > /* queues do not exist if !PORT_INIT_DONE. */ > if (pi->flags & PORT_INIT_DONE) { > for_each_txq(pi, i, txq) { > TXQ_LOCK(txq); > - m_freem(txq->m); > - txq->m =3D NULL; > - while ((m =3D buf_ring_dequeue_sc(txq->br)) !=3D = NULL) > - m_freem(m); > + txq->eq.flags &=3D ~EQ_ENABLED; > TXQ_UNLOCK(txq); > + while (!mp_ring_is_idle(txq->r)) { > + mp_ring_check_drainage(txq->r, 0); > + pause("qflush", 1); > + } > } > } > if_qflush(ifp); > @@ -1564,7 +1534,7 @@ cxgbe_get_counter(struct ifnet *ifp, ift > struct sge_txq *txq; > > for_each_txq(pi, i, txq) > - drops +=3D txq->br->br_drops; > + drops +=3D counter_u64_fetch(txq->r->drop= s); > } > > return (drops); > @@ -3236,7 +3206,8 @@ cxgbe_init_synchronized(struct port_info > { > struct adapter *sc =3D pi->adapter; > struct ifnet *ifp =3D pi->ifp; > - int rc =3D 0; > + int rc =3D 0, i; > + struct sge_txq *txq; > > ASSERT_SYNCHRONIZED_OP(sc); > > @@ -3265,6 +3236,17 @@ cxgbe_init_synchronized(struct port_info > } > > /* > + * Can't fail from this point onwards. Review > cxgbe_uninit_synchronized > + * if this changes. > + */ > + > + for_each_txq(pi, i, txq) { > + TXQ_LOCK(txq); > + txq->eq.flags |=3D EQ_ENABLED; > + TXQ_UNLOCK(txq); > + } > + > + /* > * The first iq of the first port to come up is used for tracing. > */ > if (sc->traceq < 0) { > @@ -3297,7 +3279,8 @@ cxgbe_uninit_synchronized(struct port_in > { > struct adapter *sc =3D pi->adapter; > struct ifnet *ifp =3D pi->ifp; > - int rc; > + int rc, i; > + struct sge_txq *txq; > > ASSERT_SYNCHRONIZED_OP(sc); > > @@ -3314,6 +3297,12 @@ cxgbe_uninit_synchronized(struct port_in > return (rc); > } > > + for_each_txq(pi, i, txq) { > + TXQ_LOCK(txq); > + txq->eq.flags &=3D ~EQ_ENABLED; > + TXQ_UNLOCK(txq); > + } > + > clrbit(&sc->open_device_map, pi->port_id); > PORT_LOCK(pi); > ifp->if_drv_flags &=3D ~IFF_DRV_RUNNING; > @@ -3543,15 +3532,17 @@ port_full_uninit(struct port_info *pi) > > if (pi->flags & PORT_INIT_DONE) { > > - /* Need to quiesce queues. XXX: ctrl queues? */ > + /* Need to quiesce queues. */ > + > + quiesce_wrq(sc, &sc->sge.ctrlq[pi->port_id]); > > for_each_txq(pi, i, txq) { > - quiesce_eq(sc, &txq->eq); > + quiesce_txq(sc, txq); > } > > #ifdef TCP_OFFLOAD > for_each_ofld_txq(pi, i, ofld_txq) { > - quiesce_eq(sc, &ofld_txq->eq); > + quiesce_wrq(sc, ofld_txq); > } > #endif > > @@ -3576,23 +3567,39 @@ port_full_uninit(struct port_info *pi) > } > > static void > -quiesce_eq(struct adapter *sc, struct sge_eq *eq) > +quiesce_txq(struct adapter *sc, struct sge_txq *txq) > { > - EQ_LOCK(eq); > - eq->flags |=3D EQ_DOOMED; > + struct sge_eq *eq =3D &txq->eq; > + struct sge_qstat *spg =3D (void *)&eq->desc[eq->sidx]; > > - /* > - * Wait for the response to a credit flush if one's > - * pending. > - */ > - while (eq->flags & EQ_CRFLUSHED) > - mtx_sleep(eq, &eq->eq_lock, 0, "crflush", 0); > - EQ_UNLOCK(eq); > + (void) sc; /* unused */ > > - callout_drain(&eq->tx_callout); /* XXX: iffy */ > - pause("callout", 10); /* Still iffy */ > +#ifdef INVARIANTS > + TXQ_LOCK(txq); > + MPASS((eq->flags & EQ_ENABLED) =3D=3D 0); > + TXQ_UNLOCK(txq); > +#endif > > - taskqueue_drain(sc->tq[eq->tx_chan], &eq->tx_task); > + /* Wait for the mp_ring to empty. */ > + while (!mp_ring_is_idle(txq->r)) { > + mp_ring_check_drainage(txq->r, 0); > + pause("rquiesce", 1); > + } > + > + /* Then wait for the hardware to finish. */ > + while (spg->cidx !=3D htobe16(eq->pidx)) > + pause("equiesce", 1); > + > + /* Finally, wait for the driver to reclaim all descriptors. */ > + while (eq->cidx !=3D eq->pidx) > + pause("dquiesce", 1); > +} > + > +static void > +quiesce_wrq(struct adapter *sc, struct sge_wrq *wrq) > +{ > + > + /* XXXTX */ > } > > static void > @@ -4892,6 +4899,9 @@ cxgbe_sysctls(struct port_info *pi) > oid =3D SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats", CTLFLAG= _RD, > NULL, "port statistics"); > children =3D SYSCTL_CHILDREN(oid); > + SYSCTL_ADD_UINT(ctx, children, OID_AUTO, "tx_parse_error", > CTLFLAG_RD, > + &pi->tx_parse_error, 0, > + "# of tx packets with invalid length or # of segments"); > > #define SYSCTL_ADD_T4_REG64(pi, name, desc, reg) \ > SYSCTL_ADD_OID(ctx, children, OID_AUTO, name, \ > @@ -6947,74 +6957,6 @@ sysctl_wcwr_stats(SYSCTL_HANDLER_ARGS) > } > #endif > > -static inline void > -txq_start(struct ifnet *ifp, struct sge_txq *txq) > -{ > - struct buf_ring *br; > - struct mbuf *m; > - > - TXQ_LOCK_ASSERT_OWNED(txq); > - > - br =3D txq->br; > - m =3D txq->m ? txq->m : drbr_dequeue(ifp, br); > - if (m) > - t4_eth_tx(ifp, txq, m); > -} > - > -void > -t4_tx_callout(void *arg) > -{ > - struct sge_eq *eq =3D arg; > - struct adapter *sc; > - > - if (EQ_TRYLOCK(eq) =3D=3D 0) > - goto reschedule; > - > - if (eq->flags & EQ_STALLED && !can_resume_tx(eq)) { > - EQ_UNLOCK(eq); > -reschedule: > - if (__predict_true(!(eq->flags && EQ_DOOMED))) > - callout_schedule(&eq->tx_callout, 1); > - return; > - } > - > - EQ_LOCK_ASSERT_OWNED(eq); > - > - if (__predict_true((eq->flags & EQ_DOOMED) =3D=3D 0)) { > - > - if ((eq->flags & EQ_TYPEMASK) =3D=3D EQ_ETH) { > - struct sge_txq *txq =3D arg; > - struct port_info *pi =3D txq->ifp->if_softc; > - > - sc =3D pi->adapter; > - } else { > - struct sge_wrq *wrq =3D arg; > - > - sc =3D wrq->adapter; > - } > - > - taskqueue_enqueue(sc->tq[eq->tx_chan], &eq->tx_task); > - } > - > - EQ_UNLOCK(eq); > -} > - > -void > -t4_tx_task(void *arg, int count) > -{ > - struct sge_eq *eq =3D arg; > - > - EQ_LOCK(eq); > - if ((eq->flags & EQ_TYPEMASK) =3D=3D EQ_ETH) { > - struct sge_txq *txq =3D arg; > - txq_start(txq->ifp, txq); > - } else { > - struct sge_wrq *wrq =3D arg; > - t4_wrq_tx_locked(wrq->adapter, wrq, NULL); > - } > - EQ_UNLOCK(eq); > -} > - > static uint32_t > fconf_to_mode(uint32_t fconf) > { > @@ -7452,9 +7394,9 @@ static int > set_filter_wr(struct adapter *sc, int fidx) > { > struct filter_entry *f =3D &sc->tids.ftid_tab[fidx]; > - struct wrqe *wr; > struct fw_filter_wr *fwr; > unsigned int ftid; > + struct wrq_cookie cookie; > > ASSERT_SYNCHRONIZED_OP(sc); > > @@ -7473,12 +7415,10 @@ set_filter_wr(struct adapter *sc, int fi > > ftid =3D sc->tids.ftid_base + fidx; > > - wr =3D alloc_wrqe(sizeof(*fwr), &sc->sge.mgmtq); > - if (wr =3D=3D NULL) > + fwr =3D start_wrq_wr(&sc->sge.mgmtq, howmany(sizeof(*fwr), 16), > &cookie); > + if (fwr =3D=3D NULL) > return (ENOMEM); > - > - fwr =3D wrtod(wr); > - bzero(fwr, sizeof (*fwr)); > + bzero(fwr, sizeof(*fwr)); > > fwr->op_pkd =3D htobe32(V_FW_WR_OP(FW_FILTER_WR)); > fwr->len16_pkd =3D htobe32(FW_LEN16(*fwr)); > @@ -7547,7 +7487,7 @@ set_filter_wr(struct adapter *sc, int fi > f->pending =3D 1; > sc->tids.ftids_in_use++; > > - t4_wrq_tx(sc, wr); > + commit_wrq_wr(&sc->sge.mgmtq, fwr, &cookie); > return (0); > } > > @@ -7555,22 +7495,21 @@ static int > del_filter_wr(struct adapter *sc, int fidx) > { > struct filter_entry *f =3D &sc->tids.ftid_tab[fidx]; > - struct wrqe *wr; > struct fw_filter_wr *fwr; > unsigned int ftid; > + struct wrq_cookie cookie; > > ftid =3D sc->tids.ftid_base + fidx; > > - wr =3D alloc_wrqe(sizeof(*fwr), &sc->sge.mgmtq); > - if (wr =3D=3D NULL) > + fwr =3D start_wrq_wr(&sc->sge.mgmtq, howmany(sizeof(*fwr), 16), > &cookie); > + if (fwr =3D=3D NULL) > return (ENOMEM); > - fwr =3D wrtod(wr); > bzero(fwr, sizeof (*fwr)); > > t4_mk_filtdelwr(ftid, fwr, sc->sge.fwq.abs_id); > > f->pending =3D 1; > - t4_wrq_tx(sc, wr); > + commit_wrq_wr(&sc->sge.mgmtq, fwr, &cookie); > return (0); > } > > @@ -8170,6 +8109,7 @@ t4_ioctl(struct cdev *dev, unsigned long > > /* MAC stats */ > t4_clr_port_stats(sc, pi->tx_chan); > + pi->tx_parse_error =3D 0; > > if (pi->flags & PORT_INIT_DONE) { > struct sge_rxq *rxq; > @@ -8192,24 +8132,24 @@ t4_ioctl(struct cdev *dev, unsigned long > txq->imm_wrs =3D 0; > txq->sgl_wrs =3D 0; > txq->txpkt_wrs =3D 0; > - txq->txpkts_wrs =3D 0; > - txq->txpkts_pkts =3D 0; > - txq->br->br_drops =3D 0; > - txq->no_dmamap =3D 0; > - txq->no_desc =3D 0; > + txq->txpkts0_wrs =3D 0; > + txq->txpkts1_wrs =3D 0; > + txq->txpkts0_pkts =3D 0; > + txq->txpkts1_pkts =3D 0; > + mp_ring_reset_stats(txq->r); > } > > #ifdef TCP_OFFLOAD > /* nothing to clear for each ofld_rxq */ > > for_each_ofld_txq(pi, i, wrq) { > - wrq->tx_wrs =3D 0; > - wrq->no_desc =3D 0; > + wrq->tx_wrs_direct =3D 0; > + wrq->tx_wrs_copied =3D 0; > } > #endif > wrq =3D &sc->sge.ctrlq[pi->port_id]; > - wrq->tx_wrs =3D 0; > - wrq->no_desc =3D 0; > + wrq->tx_wrs_direct =3D 0; > + wrq->tx_wrs_copied =3D 0; > } > break; > } > > Added: head/sys/dev/cxgbe/t4_mp_ring.c > > =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D=3D= =3D=3D=3D=3D > --- /dev/null 00:00:00 1970 (empty, because file is newly added) > +++ head/sys/dev/cxgbe/t4_mp_ring.c Wed Dec 31 23:19:16 2014 > (r276485) > @@ -0,0 +1,364 @@ > +/*- > + * Copyright (c) 2014 Chelsio Communications, Inc. > + * All rights reserved. > + * Written by: Navdeep Parhar > + * > + * Redistribution and use in source and binary forms, with or without > + * modification, are permitted provided that the following conditions > + * are met: > + * 1. Redistributions of source code must retain the above copyright > + * notice, this list of conditions and the following disclaimer. > + * 2. Redistributions in binary form must reproduce the above copyright > + * notice, this list of conditions and the following disclaimer in th= e > + * documentation and/or other materials provided with the distributio= n. > + * > + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AN= D > + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE > + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR > PURPOSE > + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIAB= LE > + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR > CONSEQUENTIAL > + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOO= DS > + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) > + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, > STRICT > + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY > WAY > + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY O= F > + * SUCH DAMAGE. > + */ > + > +#include > +__FBSDID("$FreeBSD$"); > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#include "t4_mp_ring.h" > + > +union ring_state { > + struct { > + uint16_t pidx_head; > + uint16_t pidx_tail; > + uint16_t cidx; > + uint16_t flags; > + }; > + uint64_t state; > +}; > + > +enum { > + IDLE =3D 0, /* consumer ran to completion, nothing more to = do. > */ > + BUSY, /* consumer is running already, or will be > shortly. */ > + STALLED, /* consumer stopped due to lack of resources. */ > + ABDICATED, /* consumer stopped even though there was work to > be > + done because it wants another thread to take > over. */ > +}; > + > +static inline uint16_t > +space_available(struct mp_ring *r, union ring_state s) > +{ > + uint16_t x =3D r->size - 1; > + > + if (s.cidx =3D=3D s.pidx_head) > + return (x); > + else if (s.cidx > s.pidx_head) > + return (s.cidx - s.pidx_head - 1); > + else > + return (x - s.pidx_head + s.cidx); > +} > + > +static inline uint16_t > +increment_idx(struct mp_ring *r, uint16_t idx, uint16_t n) > +{ > + int x =3D r->size - idx; > + > + MPASS(x > 0); > + return (x > n ? idx + n : n - x); > +} > + > +/* Consumer is about to update the ring's state to s */ > +static inline uint16_t > +state_to_flags(union ring_state s, int abdicate) > +{ > + > + if (s.cidx =3D=3D s.pidx_tail) > + return (IDLE); > + else if (abdicate && s.pidx_tail !=3D s.pidx_head) > + return (ABDICATED); > + > + return (BUSY); > +} > + > +/* > + * Caller passes in a state, with a guarantee that there is work to do > and that > + * all items up to the pidx_tail in the state are visible. > + */ > +static void > +drain_ring(struct mp_ring *r, union ring_state os, uint16_t prev, int > budget) > +{ > + union ring_state ns; > + int n, pending, total; > + uint16_t cidx =3D os.cidx; > + uint16_t pidx =3D os.pidx_tail; > + > + MPASS(os.flags =3D=3D BUSY); > + MPASS(cidx !=3D pidx); > + > + if (prev =3D=3D IDLE) > + counter_u64_add(r->starts, 1); > + pending =3D 0; > + total =3D 0; > + > + while (cidx !=3D pidx) { > + > + /* Items from cidx to pidx are available for consumption. > */ > + n =3D r->drain(r, cidx, pidx); > + if (n =3D=3D 0) { > + critical_enter(); > + do { > + os.state =3D ns.state =3D r->state; > + ns.cidx =3D cidx; > + ns.flags =3D STALLED; > + } while (atomic_cmpset_64(&r->state, os.state, > + ns.state) =3D=3D 0); > + critical_exit(); > + if (prev !=3D STALLED) > + counter_u64_add(r->stalls, 1); > + else if (total > 0) { > + counter_u64_add(r->restarts, 1); > + counter_u64_add(r->stalls, 1); > + } > + break; > + } > + cidx =3D increment_idx(r, cidx, n); > + pending +=3D n; > + total +=3D n; > + > + /* > + * We update the cidx only if we've caught up with the > pidx, the > + * real cidx is getting too far ahead of the one visible = to > + * everyone else, or we have exceeded our budget. > + */ > + if (cidx !=3D pidx && pending < 64 && total < budget) > + continue; > + critical_enter(); > + do { > + os.state =3D ns.state =3D r->state; > + ns.cidx =3D cidx; > + ns.flags =3D state_to_flags(ns, total >=3D budget= ); > + } while (atomic_cmpset_acq_64(&r->state, os.state, > ns.state) =3D=3D 0); > + critical_exit(); > + > + if (ns.flags =3D=3D ABDICATED) > + counter_u64_add(r->abdications, 1); > + if (ns.flags !=3D BUSY) { > + /* Wrong loop exit if we're going to stall. */ > + MPASS(ns.flags !=3D STALLED); > + if (prev =3D=3D STALLED) { > + MPASS(total > 0); > + counter_u64_add(r->restarts, 1); > + } > + break; > + } > + > + /* > + * The acquire style atomic above guarantees visibility o= f > items > + * associated with any pidx change that we notice here. > + */ > + pidx =3D ns.pidx_tail; > + pending =3D 0; > + } > +} > + > +int > +mp_ring_alloc(struct mp_ring **pr, int size, void *cookie, ring_drain_t > drain, > + ring_can_drain_t can_drain, struct malloc_type *mt, int flags) > +{ > + struct mp_ring *r; > + > + /* All idx are 16b so size can be 65536 at most */ > + if (pr =3D=3D NULL || size < 2 || size > 65536 || drain =3D=3D NU= LL || > + can_drain =3D=3D NULL) > + return (EINVAL); > + *pr =3D NULL; > + flags &=3D M_NOWAIT | M_WAITOK; > + MPASS(flags !=3D 0); > + > + r =3D malloc(__offsetof(struct mp_ring, items[size]), mt, flags | > M_ZERO); > + if (r =3D=3D NULL) > + return (ENOMEM); > + r->size =3D size; > + r->cookie =3D cookie; > + r->mt =3D mt; > + r->drain =3D drain; > + r->can_drain =3D can_drain; > + r->enqueues =3D counter_u64_alloc(flags); > + r->drops =3D counter_u64_alloc(flags); > + r->starts =3D counter_u64_alloc(flags); > + r->stalls =3D counter_u64_alloc(flags); > + r->restarts =3D counter_u64_alloc(flags); > + r->abdications =3D counter_u64_alloc(flags); > + if (r->enqueues =3D=3D NULL || r->drops =3D=3D NULL || r->starts = =3D=3D NULL || > + r->stalls =3D=3D NULL || r->restarts =3D=3D NULL || > + r->abdications =3D=3D NULL) { > + mp_ring_free(r); > + return (ENOMEM); > + } > + > + *pr =3D r; > + return (0); > +} > + > +void > + > +mp_ring_free(struct mp_ring *r) > +{ > + > + if (r =3D=3D NULL) > + return; > + > + if (r->enqueues !=3D NULL) > + counter_u64_free(r->enqueues); > + if (r->drops !=3D NULL) > + counter_u64_free(r->drops); > + if (r->starts !=3D NULL) > + counter_u64_free(r->starts); > + if (r->stalls !=3D NULL) > + counter_u64_free(r->stalls); > + if (r->restarts !=3D NULL) > + counter_u64_free(r->restarts); > + if (r->abdications !=3D NULL) > + counter_u64_free(r->abdications); > + > + free(r, r->mt); > +} > + > +/* > + * Enqueue n items and maybe drain the ring for some time. > + * > + * Returns an errno. > + */ > +int > +mp_ring_enqueue(struct mp_ring *r, void **items, int n, int budget) > +{ > + union ring_state os, ns; > + uint16_t pidx_start, pidx_stop; > + int i; > + > + MPASS(items !=3D NULL); > + MPASS(n > 0); > + > > *** DIFF OUTPUT TRUNCATED AT 1000 LINES *** > > --=20 -----------------------------------------+------------------------------- Prof. Luigi RIZZO, rizzo@iet.unipi.it . Dip. di Ing. dell'Informazione http://www.iet.unipi.it/~luigi/ . Universita` di Pisa TEL +39-050-2211611 . via Diotisalvi 2 Mobile +39-338-6809875 . 56122 PISA (Italy) -----------------------------------------+-------------------------------