aboutsummaryrefslogtreecommitdiffstats
path: root/Documentation/networking
diff options
context:
space:
mode:
authorWillem de Bruijn <willemb@google.com>2011-08-09 00:20:48 -0400
committerDavid S. Miller <davem@davemloft.net>2011-08-10 10:56:58 -0400
commit56c07271307b4a20802005692b2b70dfe13d72e8 (patch)
tree018bcae56c42e8adc3f77a0de58e8374a9953a08 /Documentation/networking
parent3557619f0f6f7496ed453d4825e24958ab1884e0 (diff)
net: add Documentation/networking/scaling.txt
Describes RSS, RPS, RFS, accelerated RFS, and XPS. This version incorporates comments by Randy Dunlap and Rick Jones. Besides text cleanup, it adds an explicit "Suggested Configuration" heading to each section. Signed-off-by: Willem de Bruijn <willemb@google.com> Acked-By: Rick Jones <rick.jones2@hp.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'Documentation/networking')
-rw-r--r--Documentation/networking/scaling.txt371
1 files changed, 371 insertions, 0 deletions
diff --git a/Documentation/networking/scaling.txt b/Documentation/networking/scaling.txt
new file mode 100644
index 000000000000..7254b4b5910e
--- /dev/null
+++ b/Documentation/networking/scaling.txt
@@ -0,0 +1,371 @@
1Scaling in the Linux Networking Stack
2
3
4Introduction
5============
6
7This document describes a set of complementary techniques in the Linux
8networking stack to increase parallelism and improve performance for
9multi-processor systems.
10
11The following technologies are described:
12
13 RSS: Receive Side Scaling
14 RPS: Receive Packet Steering
15 RFS: Receive Flow Steering
16 Accelerated Receive Flow Steering
17 XPS: Transmit Packet Steering
18
19
20RSS: Receive Side Scaling
21=========================
22
23Contemporary NICs support multiple receive and transmit descriptor queues
24(multi-queue). On reception, a NIC can send different packets to different
25queues to distribute processing among CPUs. The NIC distributes packets by
26applying a filter to each packet that assigns it to one of a small number
27of logical flows. Packets for each flow are steered to a separate receive
28queue, which in turn can be processed by separate CPUs. This mechanism is
29generally known as “Receive-side Scaling” (RSS). The goal of RSS and
30the other scaling techniques to increase performance uniformly.
31Multi-queue distribution can also be used for traffic prioritization, but
32that is not the focus of these techniques.
33
34The filter used in RSS is typically a hash function over the network
35and/or transport layer headers-- for example, a 4-tuple hash over
36IP addresses and TCP ports of a packet. The most common hardware
37implementation of RSS uses a 128-entry indirection table where each entry
38stores a queue number. The receive queue for a packet is determined
39by masking out the low order seven bits of the computed hash for the
40packet (usually a Toeplitz hash), taking this number as a key into the
41indirection table and reading the corresponding value.
42
43Some advanced NICs allow steering packets to queues based on
44programmable filters. For example, webserver bound TCP port 80 packets
45can be directed to their own receive queue. Such “n-tuple” filters can
46be configured from ethtool (--config-ntuple).
47
48==== RSS Configuration
49
50The driver for a multi-queue capable NIC typically provides a kernel
51module parameter for specifying the number of hardware queues to
52configure. In the bnx2x driver, for instance, this parameter is called
53num_queues. A typical RSS configuration would be to have one receive queue
54for each CPU if the device supports enough queues, or otherwise at least
55one for each cache domain at a particular cache level (L1, L2, etc.).
56
57The indirection table of an RSS device, which resolves a queue by masked
58hash, is usually programmed by the driver at initialization. The
59default mapping is to distribute the queues evenly in the table, but the
60indirection table can be retrieved and modified at runtime using ethtool
61commands (--show-rxfh-indir and --set-rxfh-indir). Modifying the
62indirection table could be done to give different queues different
63relative weights.
64
65== RSS IRQ Configuration
66
67Each receive queue has a separate IRQ associated with it. The NIC triggers
68this to notify a CPU when new packets arrive on the given queue. The
69signaling path for PCIe devices uses message signaled interrupts (MSI-X),
70that can route each interrupt to a particular CPU. The active mapping
71of queues to IRQs can be determined from /proc/interrupts. By default,
72an IRQ may be handled on any CPU. Because a non-negligible part of packet
73processing takes place in receive interrupt handling, it is advantageous
74to spread receive interrupts between CPUs. To manually adjust the IRQ
75affinity of each interrupt see Documentation/IRQ-affinity. Some systems
76will be running irqbalance, a daemon that dynamically optimizes IRQ
77assignments and as a result may override any manual settings.
78
79== Suggested Configuration
80
81RSS should be enabled when latency is a concern or whenever receive
82interrupt processing forms a bottleneck. Spreading load between CPUs
83decreases queue length. For low latency networking, the optimal setting
84is to allocate as many queues as there are CPUs in the system (or the
85NIC maximum, if lower). Because the aggregate number of interrupts grows
86with each additional queue, the most efficient high-rate configuration
87is likely the one with the smallest number of receive queues where no
88CPU that processes receive interrupts reaches 100% utilization. Per-cpu
89load can be observed using the mpstat utility.
90
91
92RPS: Receive Packet Steering
93============================
94
95Receive Packet Steering (RPS) is logically a software implementation of
96RSS. Being in software, it is necessarily called later in the datapath.
97Whereas RSS selects the queue and hence CPU that will run the hardware
98interrupt handler, RPS selects the CPU to perform protocol processing
99above the interrupt handler. This is accomplished by placing the packet
100on the desired CPU’s backlog queue and waking up the CPU for processing.
101RPS has some advantages over RSS: 1) it can be used with any NIC,
1022) software filters can easily be added to hash over new protocols,
1033) it does not increase hardware device interrupt rate (although it does
104introduce inter-processor interrupts (IPIs)).
105
106RPS is called during bottom half of the receive interrupt handler, when
107a driver sends a packet up the network stack with netif_rx() or
108netif_receive_skb(). These call the get_rps_cpu() function, which
109selects the queue that should process a packet.
110
111The first step in determining the target CPU for RPS is to calculate a
112flow hash over the packet’s addresses or ports (2-tuple or 4-tuple hash
113depending on the protocol). This serves as a consistent hash of the
114associated flow of the packet. The hash is either provided by hardware
115or will be computed in the stack. Capable hardware can pass the hash in
116the receive descriptor for the packet; this would usually be the same
117hash used for RSS (e.g. computed Toeplitz hash). The hash is saved in
118skb->rx_hash and can be used elsewhere in the stack as a hash of the
119packet’s flow.
120
121Each receive hardware queue has an associated list of CPUs to which
122RPS may enqueue packets for processing. For each received packet,
123an index into the list is computed from the flow hash modulo the size
124of the list. The indexed CPU is the target for processing the packet,
125and the packet is queued to the tail of that CPU’s backlog queue. At
126the end of the bottom half routine, IPIs are sent to any CPUs for which
127packets have been queued to their backlog queue. The IPI wakes backlog
128processing on the remote CPU, and any queued packets are then processed
129up the networking stack.
130
131==== RPS Configuration
132
133RPS requires a kernel compiled with the CONFIG_RPS kconfig symbol (on
134by default for SMP). Even when compiled in, RPS remains disabled until
135explicitly configured. The list of CPUs to which RPS may forward traffic
136can be configured for each receive queue using a sysfs file entry:
137
138 /sys/class/net/<dev>/queues/rx-<n>/rps_cpus
139
140This file implements a bitmap of CPUs. RPS is disabled when it is zero
141(the default), in which case packets are processed on the interrupting
142CPU. Documentation/IRQ-affinity.txt explains how CPUs are assigned to
143the bitmap.
144
145== Suggested Configuration
146
147For a single queue device, a typical RPS configuration would be to set
148the rps_cpus to the CPUs in the same cache domain of the interrupting
149CPU. If NUMA locality is not an issue, this could also be all CPUs in
150the system. At high interrupt rate, it might be wise to exclude the
151interrupting CPU from the map since that already performs much work.
152
153For a multi-queue system, if RSS is configured so that a hardware
154receive queue is mapped to each CPU, then RPS is probably redundant
155and unnecessary. If there are fewer hardware queues than CPUs, then
156RPS might be beneficial if the rps_cpus for each queue are the ones that
157share the same cache domain as the interrupting CPU for that queue.
158
159
160RFS: Receive Flow Steering
161==========================
162
163While RPS steers packets solely based on hash, and thus generally
164provides good load distribution, it does not take into account
165application locality. This is accomplished by Receive Flow Steering
166(RFS). The goal of RFS is to increase datacache hitrate by steering
167kernel processing of packets to the CPU where the application thread
168consuming the packet is running. RFS relies on the same RPS mechanisms
169to enqueue packets onto the backlog of another CPU and to wake up that
170CPU.
171
172In RFS, packets are not forwarded directly by the value of their hash,
173but the hash is used as index into a flow lookup table. This table maps
174flows to the CPUs where those flows are being processed. The flow hash
175(see RPS section above) is used to calculate the index into this table.
176The CPU recorded in each entry is the one which last processed the flow.
177If an entry does not hold a valid CPU, then packets mapped to that entry
178are steered using plain RPS. Multiple table entries may point to the
179same CPU. Indeed, with many flows and few CPUs, it is very likely that
180a single application thread handles flows with many different flow hashes.
181
182rps_sock_table is a global flow table that contains the *desired* CPU for
183flows: the CPU that is currently processing the flow in userspace. Each
184table value is a CPU index that is updated during calls to recvmsg and
185sendmsg (specifically, inet_recvmsg(), inet_sendmsg(), inet_sendpage()
186and tcp_splice_read()).
187
188When the scheduler moves a thread to a new CPU while it has outstanding
189receive packets on the old CPU, packets may arrive out of order. To
190avoid this, RFS uses a second flow table to track outstanding packets
191for each flow: rps_dev_flow_table is a table specific to each hardware
192receive queue of each device. Each table value stores a CPU index and a
193counter. The CPU index represents the *current* CPU onto which packets
194for this flow are enqueued for further kernel processing. Ideally, kernel
195and userspace processing occur on the same CPU, and hence the CPU index
196in both tables is identical. This is likely false if the scheduler has
197recently migrated a userspace thread while the kernel still has packets
198enqueued for kernel processing on the old CPU.
199
200The counter in rps_dev_flow_table values records the length of the current
201CPU's backlog when a packet in this flow was last enqueued. Each backlog
202queue has a head counter that is incremented on dequeue. A tail counter
203is computed as head counter + queue length. In other words, the counter
204in rps_dev_flow_table[i] records the last element in flow i that has
205been enqueued onto the currently designated CPU for flow i (of course,
206entry i is actually selected by hash and multiple flows may hash to the
207same entry i).
208
209And now the trick for avoiding out of order packets: when selecting the
210CPU for packet processing (from get_rps_cpu()) the rps_sock_flow table
211and the rps_dev_flow table of the queue that the packet was received on
212are compared. If the desired CPU for the flow (found in the
213rps_sock_flow table) matches the current CPU (found in the rps_dev_flow
214table), the packet is enqueued onto that CPU’s backlog. If they differ,
215the current CPU is updated to match the desired CPU if one of the
216following is true:
217
218- The current CPU's queue head counter >= the recorded tail counter
219 value in rps_dev_flow[i]
220- The current CPU is unset (equal to NR_CPUS)
221- The current CPU is offline
222
223After this check, the packet is sent to the (possibly updated) current
224CPU. These rules aim to ensure that a flow only moves to a new CPU when
225there are no packets outstanding on the old CPU, as the outstanding
226packets could arrive later than those about to be processed on the new
227CPU.
228
229==== RFS Configuration
230
231RFS is only available if the kconfig symbol CONFIG_RFS is enabled (on
232by default for SMP). The functionality remains disabled until explicitly
233configured. The number of entries in the global flow table is set through:
234
235 /proc/sys/net/core/rps_sock_flow_entries
236
237The number of entries in the per-queue flow table are set through:
238
239 /sys/class/net/<dev>/queues/tx-<n>/rps_flow_cnt
240
241== Suggested Configuration
242
243Both of these need to be set before RFS is enabled for a receive queue.
244Values for both are rounded up to the nearest power of two. The
245suggested flow count depends on the expected number of active connections
246at any given time, which may be significantly less than the number of open
247connections. We have found that a value of 32768 for rps_sock_flow_entries
248works fairly well on a moderately loaded server.
249
250For a single queue device, the rps_flow_cnt value for the single queue
251would normally be configured to the same value as rps_sock_flow_entries.
252For a multi-queue device, the rps_flow_cnt for each queue might be
253configured as rps_sock_flow_entries / N, where N is the number of
254queues. So for instance, if rps_flow_entries is set to 32768 and there
255are 16 configured receive queues, rps_flow_cnt for each queue might be
256configured as 2048.
257
258
259Accelerated RFS
260===============
261
262Accelerated RFS is to RFS what RSS is to RPS: a hardware-accelerated load
263balancing mechanism that uses soft state to steer flows based on where
264the application thread consuming the packets of each flow is running.
265Accelerated RFS should perform better than RFS since packets are sent
266directly to a CPU local to the thread consuming the data. The target CPU
267will either be the same CPU where the application runs, or at least a CPU
268which is local to the application thread’s CPU in the cache hierarchy.
269
270To enable accelerated RFS, the networking stack calls the
271ndo_rx_flow_steer driver function to communicate the desired hardware
272queue for packets matching a particular flow. The network stack
273automatically calls this function every time a flow entry in
274rps_dev_flow_table is updated. The driver in turn uses a device specific
275method to program the NIC to steer the packets.
276
277The hardware queue for a flow is derived from the CPU recorded in
278rps_dev_flow_table. The stack consults a CPU to hardware queue map which
279is maintained by the NIC driver. This is an auto-generated reverse map of
280the IRQ affinity table shown by /proc/interrupts. Drivers can use
281functions in the cpu_rmap (“CPU affinity reverse map”) kernel library
282to populate the map. For each CPU, the corresponding queue in the map is
283set to be one whose processing CPU is closest in cache locality.
284
285==== Accelerated RFS Configuration
286
287Accelerated RFS is only available if the kernel is compiled with
288CONFIG_RFS_ACCEL and support is provided by the NIC device and driver.
289It also requires that ntuple filtering is enabled via ethtool. The map
290of CPU to queues is automatically deduced from the IRQ affinities
291configured for each receive queue by the driver, so no additional
292configuration should be necessary.
293
294== Suggested Configuration
295
296This technique should be enabled whenever one wants to use RFS and the
297NIC supports hardware acceleration.
298
299XPS: Transmit Packet Steering
300=============================
301
302Transmit Packet Steering is a mechanism for intelligently selecting
303which transmit queue to use when transmitting a packet on a multi-queue
304device. To accomplish this, a mapping from CPU to hardware queue(s) is
305recorded. The goal of this mapping is usually to assign queues
306exclusively to a subset of CPUs, where the transmit completions for
307these queues are processed on a CPU within this set. This choice
308provides two benefits. First, contention on the device queue lock is
309significantly reduced since fewer CPUs contend for the same queue
310(contention can be eliminated completely if each CPU has its own
311transmit queue). Secondly, cache miss rate on transmit completion is
312reduced, in particular for data cache lines that hold the sk_buff
313structures.
314
315XPS is configured per transmit queue by setting a bitmap of CPUs that
316may use that queue to transmit. The reverse mapping, from CPUs to
317transmit queues, is computed and maintained for each network device.
318When transmitting the first packet in a flow, the function
319get_xps_queue() is called to select a queue. This function uses the ID
320of the running CPU as a key into the CPU-to-queue lookup table. If the
321ID matches a single queue, that is used for transmission. If multiple
322queues match, one is selected by using the flow hash to compute an index
323into the set.
324
325The queue chosen for transmitting a particular flow is saved in the
326corresponding socket structure for the flow (e.g. a TCP connection).
327This transmit queue is used for subsequent packets sent on the flow to
328prevent out of order (ooo) packets. The choice also amortizes the cost
329of calling get_xps_queues() over all packets in the connection. To avoid
330ooo packets, the queue for a flow can subsequently only be changed if
331skb->ooo_okay is set for a packet in the flow. This flag indicates that
332there are no outstanding packets in the flow, so the transmit queue can
333change without the risk of generating out of order packets. The
334transport layer is responsible for setting ooo_okay appropriately. TCP,
335for instance, sets the flag when all data for a connection has been
336acknowledged.
337
338==== XPS Configuration
339
340XPS is only available if the kconfig symbol CONFIG_XPS is enabled (on by
341default for SMP). The functionality remains disabled until explicitly
342configured. To enable XPS, the bitmap of CPUs that may use a transmit
343queue is configured using the sysfs file entry:
344
345/sys/class/net/<dev>/queues/tx-<n>/xps_cpus
346
347== Suggested Configuration
348
349For a network device with a single transmission queue, XPS configuration
350has no effect, since there is no choice in this case. In a multi-queue
351system, XPS is preferably configured so that each CPU maps onto one queue.
352If there are as many queues as there are CPUs in the system, then each
353queue can also map onto one CPU, resulting in exclusive pairings that
354experience no contention. If there are fewer queues than CPUs, then the
355best CPUs to share a given queue are probably those that share the cache
356with the CPU that processes transmit completions for that queue
357(transmit interrupts).
358
359
360Further Information
361===================
362RPS and RFS were introduced in kernel 2.6.35. XPS was incorporated into
3632.6.38. Original patches were submitted by Tom Herbert
364(therbert@google.com)
365
366Accelerated RFS was introduced in 2.6.35. Original patches were
367submitted by Ben Hutchings (bhutchings@solarflare.com)
368
369Authors:
370Tom Herbert (therbert@google.com)
371Willem de Bruijn (willemb@google.com)