Skip site navigation (1)Skip section navigation (2)
Date:      Fri, 28 Dec 2007 15:36:39 +1100 (EST)
From:      Bruce Evans <brde@optusnet.com.au>
To:        Mark Fullmer <maf@eng.oar.net>
Cc:        Kostik Belousov <kostikbel@gmail.com>, freebsd-net@FreeBSD.org, freebsd-stable@FreeBSD.org, Bruce Evans <brde@optusnet.com.au>
Subject:   Re: Packet loss every 30.999 seconds
Message-ID:  <20071228143411.C3587@besplex.bde.org>
In-Reply-To: <985A3F99-B3F4-451E-BD77-E2EB4351E323@eng.oar.net>
References:  <20071221234347.GS25053@tnn.dglawrence.com> <MDEHLPKNGKAHNMBLJOLKMEKLJAAC.davids@webmaster.com> <20071222050743.GP57756@deviant.kiev.zoral.com.ua> <20071223032944.G48303@delplex.bde.org> <985A3F99-B3F4-451E-BD77-E2EB4351E323@eng.oar.net>

next in thread | previous in thread | raw e-mail | index | archive | help
On Sat, 22 Dec 2007, Mark Fullmer wrote:

> On Dec 22, 2007, at 12:08 PM, Bruce Evans wrote:
>> 
>> I still don't understand the original problem, that the kernel is not
>> even preemptible enough for network interrupts to work (except in 5.2
>> where Giant breaks things).  Perhaps I misread the problem, and it is
>> actually that networking works but userland is unable to run in time
>> to avoid packet loss.
>
> The test is done with UDP packets between two servers.  The em
> driver is incrementing the received packet count correctly but
> the packet is not making it up the network stack.  If
> the application was not servicing the socket fast enough I would
> expect to see the "dropped due to full socket buffers" (udps_fullsock)
> counter incrementing, as shown by netstat -s.

I couldn't see any sign of PREEMPTION not working in 6.3-PREREALEASE.
em seemed to keep up with the maximum rate that I can easily generate
(640 kpps with tiny udp packets), though it cannot transmit at more than
400 kpps on the same hardware.  This is without aby syncer activity to
cause glitches.  The rest of the system couldn't keep up, and with my
normal configuration of net.isr.direct=1, systat -ip (udps_fullsock)
showed too many packets being dropped, but all the numbers seemed to
add up right.  (I didn't do end-to-end packet counts.  I'm using ttcp
to send and receive packets; the receiver loses so many packets that
it rarely terminates properly, and when it does terminate it always
shows many dropped.)  However, with net.isr.direct=0, packets are dropped
with no sign of the problem except a reduced count of good packets in
systat -ip.

Packet rate counter     net.isr.direct=1      net.isr.direct=0
-------------------     ----------------      ----------------
netstat -I              639042                643522 (faster later)
systat -ip (total rx)   639042                382567 (dropped many b4 here)
           (UDP total)   639042                382567
       (udps_fullsock)   298911                70340
      (diff of prev 2)   340031                312227 (300+k always dropped)
net.isr.count           small                 large (seems to be correct 643k)
net.isr.directed        large (correct?)      no change
net.isr.queued          0                     0
net.isr.drop            0                     0

net.isr.direct=0 is apparently causing dropped packets without even counting
them.  However, the drop seems to be below the netisr level.

More worryingly, with full 1500-byte packets (1472 data + 28 UDP
header), packets can be sent at a rate of 76 kpps (nearly 950 Mbps)
with a load of only 80% on the receiver, yet the ttcp receiver still
drops about 1000 pps due top "socket buffer full".  With net.usr.direct=0
it drops an additinal 700 pps due to this.  Glitches from sync(2)
taking 25 ms increase the loss by about 1000 packets, and using rtprio
for the ttcp receiver doesn't seem to help at all.

In previous mail, you (Mark) wrote:

# With FreeBSD 4 I was able to run a UDP data collector with rtprio set,
# kern.ipc.maxsockbuf=20480000, then use setsockopt() with SO_RCVBUF
# in the application.  If packets were dropped they would show up
# with netstat -s as "dropped due to full socket buffers".
# 
# Since the packet never makes it to ip_input() I no longer have
# any way to count drops.  There will always be corner cases where
# interrupts are lost and drops not accounted for if the adapter
# hardware can't report them, but right now I've got no way to
# estimate any loss.

I tried using SO_RCVBUF in ttcp (it's an old version of ttcp that doesn't
have an option for this).  With the default kern.ipc.maxsockbuf of 256K,
this didn't seem to help.  20MB should work better :-) but I didn't try that.
I don't understand how fast the socket buffer fills up and would have
thought that 256K was enough for tiny packets but not for 1500-byte packets.
Their seems to be a general problem that 1Gbps NICs have or should have
rings of size >= 256 or 512 so that they aren't forced to drop packets
when their interrupt handler has a reasonable but larger latency, yet if
we actually use this feature then we flood the upper layers with hundreds
of packets and fill up socket buffers etc. there.

Bruce



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20071228143411.C3587>