Skip site navigation (1)Skip section navigation (2)
Date:      Sun, 5 Apr 2009 23:17:58 +0100 (BST)
From:      Robert Watson <rwatson@FreeBSD.org>
To:        Ivan Voras <ivoras@freebsd.org>
Cc:        freebsd-net@freebsd.org
Subject:   Re: Advice on a multithreaded netisr  patch?
Message-ID:  <alpine.BSF.2.00.0904052243250.34905@fledge.watson.org>
In-Reply-To: <grappq$tsg$1@ger.gmane.org>
References:  <gra7mq$ei8$1@ger.gmane.org> <alpine.BSF.2.00.0904051422280.12639@fledge.watson.org> <grac1s$p56$1@ger.gmane.org> <alpine.BSF.2.00.0904051440460.12639@fledge.watson.org> <grappq$tsg$1@ger.gmane.org>

next in thread | previous in thread | raw e-mail | index | archive | help

On Sun, 5 Apr 2009, Ivan Voras wrote:

>> The argument is not that they are slower (although they probably are a bit 
>> slower), rather that they introduce serialization bottlenecks by requiring 
>> synchronization between CPUs in order to distribute the work. Certainly 
>> some of the scalability issues in the stack are not a result of that, but a 
>> good number are.
>
> I'd like to understand more. If (in netisr) I have a mbuf with headers, is 
> this data already transfered from the card or is it magically "not here 
> yet"?

A lot depends on the details of the card and driver.  The driver will take 
cache misses on the descriptor ring entry, if it's not already in cache, and 
the link layer will take a cache miss on the front of the ethernet frame in 
the cluster pointed to by the mbuf header as part of its demux.  What happens 
next depends on your dispatch model and cache line size.  Let's make a few 
simplifying assumptions that are mostly true:

- The driver associats a single cluster with each receive ring entry for each
   packet to be stored in, and the cluster is cacheline-aligned.  No header
   splitting is enabled.

- Standard ethernet encapsulation of IP is used, without additional VLAN
   headers or other encapsulation, etc.  There are no IP options.

- We don't need to validate any checksums because the hardware has done it for
   us, so no need to take cache misses on data that doesn't matter until we
   reach higher layers.

In the device driver/ithread code, we'll now proceed to take some cache 
misses assuming we're not pretty lucky:

(1) The descriptor ring entry
(2) The mbuf packet header
(3) The first cache line in the cluster

This is sufficient to figure out what protocol we're going to dispatch to, and 
depending on dispatch model, we now either enqueue the packet for delivery to 
a netisr, or we directly dispatch the handler for IP.

If the packet is processed on the current CPU and we're direct dispatching, or 
if we've dispatched to a netisr on the same CPU and we're quite lucky, the 
mbuf packet header and front of the cluster will be in the cache.

However, what happens next depends on the cache fetch and line size.  If 
things happen in 32-byte cache lines or smaller, we cache miss on the end of 
the IP header, because the last two bytes of the destination IP address start 
at offset 32 into the cluster.  If we have 64-byte fetching and line size, 
things go better because both the full IP and TCP headers should be in that 
first cache line.

One big advantage to direct dispatch is that it maximizes the chances that we 
don't blow out the low-level CPU caches between link-layer and IP-layer 
processing, meaning that we might actually get through all the IP and TCP 
headers without a cache miss on a 64-byte line size.  If we netisr dispatch to 
another CPU without a shared cache, or we netisr dispatch to the current CPU 
but there's a scheduling delay, other packets queued first, etc, we'll take a 
number of the same cache misses over again as things get pulled into the right 
cache.

This presents a strong cache motivation to keep a packet "on" a CPU and even 
in the same thread once you've started processing it.  If you have to enqueue, 
you take locks, take a context switch, deal with the fact that LRU on cache 
lines isn't going to like your queue depth, and potentially pay a number of 
additional cache misses on the same data.  There are also some other good 
reasons to use direct dispatch, such as avoiding doing work on packets that 
will later be dropped if the netisr queue overflows.

This is why we direct dispatch by default, and why this is quite a good 
strategy for multiple input queue network cards, where it also buys us 
parallelism.

Note that if the flow RSS hash is in the same cache line as the rest of the 
receive descriptor ring entry, you may be able to avoid the cache miss on the 
cluster and simply redirect it to another CPU's netisr without ever reading 
packet data, which avoids at least one and possibly two cache misses, but also 
means that you have to run the link layer in the remote netisr, rather than 
locally in the ithread.

> In the first case, the package reception code path is not changed until it's 
> queued on a thread, on which it's handled in the future (or is the influence 
> of "other" data like timers and internal TCP reassembly buffers so large?). 
> In the second case, why?

The good news about TCP reassembly is that we don't have to look at the data, 
only mbuf headers and reassembly buffer entries, so with any luck we've 
avoided actually taking a cache miss on the data.  If things go well, we can 
avoid looking at anything but mbuf and packet headers until the socket copies 
out, but I'm not sure how well we do that in practice.

> As the card and the OS can already process many packets per second for 
> something fairly complex as routing (http://www.tancsa.com/blast.html), and 
> TCP chokes swi:net at 100% of a core, isn't this indication there's 
> certainly more space for improvement even with a single-queue old-fashioned 
> NICs?

Maybe.  It depends on the relative costs of local processing vs redistributing 
the work, which involves schedulers, IPIs, additional cache misses, lock 
contention, and so on.  This means there's a period where it can't possibly be 
a win, and then at some point it's a win as long as the stack scales.  This is 
essentially the usual trade-off in using threads and parallelism: does the 
benefit of multiple parallel execution units make up for the overheads of 
synchronization and data migration?

There are some previous e-mail threads where people have observed that for 
some workloads, switching to netisr wins over direct dispatch.  For example, 
if you have a number of cores and are doing firewall processing, offloading 
work to the netisr from the input ithread may improve performance.  However, 
this appears not to be the common case for end-host workloads on the hardware 
we mostly target, and this is increasingly true as multiple input queues come 
into play, as the card itself will allow us to use multiple CPUs without any 
interactions between the CPUs.

This isn't to say that work redistribution using a netisr-like scheme isn't a 
good idea: in a world where CPU threads are weak compared to the wire 
workflow, and there's cache locality across threads on the same core, or NUMA 
is present, there may be a potential for a big win when available work 
significantly exceeds what a single CPU thread/core can handle.  In that case, 
we want to place the work as close as possible to take advantage of shared 
caches or the memory being local to the CPU thread/core doing the deferred 
work.

FYI, the localhost case is a bit weird -- I think we have some scheduling 
issues that are causing loopback netisr stuff to be pessimally scheduled. 
Here are some suggestions for things to try and see if they help, though:

- Comment out all ifnet, IP, and TCP global statistics in your local stack --
   especially look for things tcpstat.whatever++;.

- Use cpuset to pin ithreads, the netisr, and whatever else, to specific cores
   so that they don't migrate, and if your system uses HTT, experiment with
   pinning the ithread and the netisr on different threads on the same core, or
   at least, different cores on the same die.

- Experiment with using just the source IP, the source + destination IP, and
   both IPs plus TCP ports in your hash.

- If your card supports RSS, pass the flowid up the stack in the mbuf packet
   header flowid field, and use that instead of the hash for work placement.

- If you're doing pure PPS tests with UDP (or the like), and your test can
   tolerate disordering, try hashing based on the mbuf header address or
   something else that will distribute the work but not take a cache miss.

- If you have a flowid or the above disordered condition applies, try shifting
   the link layer dispatch to the netisr, rather than doing the demux in the
   ithread, as that will avoid cache misses in the ithread and do all the demux
   in the netisr.

Robert N M Watson
Computer Laboratory
University of Cambridge



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