Date: Sat, 04 Oct 2008 03:57:50 +0200 From: Ivan Voras <ivoras@freebsd.org> To: freebsd-net@freebsd.org Cc: freebsd-hackers@freebsd.org Subject: Network IO & scheduling problem? (was: Optimizing for high PPS, Intel NICs) Message-ID: <gc6iip$n2c$1@ger.gmane.org> In-Reply-To: <gbu0im$t4r$1@ger.gmane.org> References: <gbk0j9$cpj$1@ger.gmane.org> <gbu0im$t4r$1@ger.gmane.org>
next in thread | previous in thread | raw e-mail | index | archive | help
I experimented some more with my problem and it would be pretty incredible if it weren't for the fact that I can reliably reproduce it. Excuse me if this description is overly verbose, I can't decide which information might be important. First, here's some more background on the problem: The application (not created for the purpose of being a benchmark) accepts TCP connections and assigns them into one of predefined connection groups, configured at serve startup. Each connection group is polled for network IO events by its own thread. There are no overlaps between these groups. This polling can be done by either kqueue() or poll(). The client application is a stress test application that creates 40+ parallel, long lived TCP connections and tries to saturate the server with queries (so, for example with 40 client connections and 4 connection groups on the server, each kqueue or poll list has 10 entries). For testing purposes, the server doesn't actually do any useful work so the emphasis is on network IO. The server hardware is 2x quad Xeon 5405, 8 cores total, running FreeBSD 8-CURRENT amd64 (debugging options turned off). The client system actually doesn't matter, I've tested with many systems, including desktops and laptops with different NICs. The problem is: a) When IO polling on the server is done with kqueues, one kqueue per thread / connection group, I can create up to 3 threads / connection groups without any problems. When I create 4 threads, suddenly the em1 taskq thread starts eating 100% CPU. With 3 or less threads, em taskq spends less than 1% CPU time. At this point I can push 150,000 packets in each direction. b) When polling with poll(), I can create up to 4 server threads without saturating the em taskq, but at 4 threads it starts to spend high random amounts of CPU time, from 30% to 80%. At 5 or more threads it's pinned to 100%. With 4 threads I can push 170,000 packets per direction. With 3 or less threads the em taskq seems to spike in CPU usage right at the start when clients connect and then goes to < 1% CPU time. c) It looks like the effect is much less pronounced on a 4-core machine. I don't have it now but previous tests showed em taskq at 10% with 5 threads and kqueue polling. Some things I tried: disabling TSO doesn't help, disabling PREEMPTION doesn't help, it's not an interrupt storm, the taskq thread doesn't seem to jump all over cpu cores, BUT the amount of context switches rises sharply from ~~12,000 with 3 threads to ~~65,000 with 4 threads to ~~220,000 with 5 threads. Interrupt rate varies between 1000 and 3000 (interrupt moderation by the NIC?). I'm looking for ideas that can explain all this, and also for guidance on how to instrument the kernel to find out what is happening here. Fixes would also be welcome :)
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?gc6iip$n2c$1>