1.. SPDX-License-Identifier: GPL-2.0 2 3===================================== 4Scaling in the Linux Networking Stack 5===================================== 6 7 8Introduction 9============ 10 11This document describes a set of complementary techniques in the Linux 12networking stack to increase parallelism and improve performance for 13multi-processor systems. 14 15The following technologies are described: 16 17- RSS: Receive Side Scaling 18- RPS: Receive Packet Steering 19- RFS: Receive Flow Steering 20- Accelerated Receive Flow Steering 21- XPS: Transmit Packet Steering 22 23 24RSS: Receive Side Scaling 25========================= 26 27Contemporary NICs support multiple receive and transmit descriptor queues 28(multi-queue). On reception, a NIC can send different packets to different 29queues to distribute processing among CPUs. The NIC distributes packets by 30applying a filter to each packet that assigns it to one of a small number 31of logical flows. Packets for each flow are steered to a separate receive 32queue, which in turn can be processed by separate CPUs. This mechanism is 33generally known as “Receive-side Scaling” (RSS). The goal of RSS and 34the other scaling techniques is to increase performance uniformly. 35Multi-queue distribution can also be used for traffic prioritization, but 36that is not the focus of these techniques. 37 38The filter used in RSS is typically a hash function over the network 39and/or transport layer headers-- for example, a 4-tuple hash over 40IP addresses and TCP ports of a packet. The most common hardware 41implementation of RSS uses an indirection table where each entry 42stores a queue number. The receive queue for a packet is determined 43by indexing the indirection table with the low order bits of the 44computed hash for the packet (usually a Toeplitz hash). 45 46The indirection table helps even out the traffic distribution when queue 47count is not a power of two. NICs should provide an indirection table 48at least 4 times larger than the queue count. 4x table results in ~16% 49imbalance between the queues, which is acceptable for most applications. 50 51Some NICs support symmetric RSS hashing where, if the IP (source address, 52destination address) and TCP/UDP (source port, destination port) tuples 53are swapped, the computed hash is the same. This is beneficial in some 54applications that monitor TCP/IP flows (IDS, firewalls, ...etc) and need 55both directions of the flow to land on the same Rx queue (and CPU). The 56"Symmetric-XOR" and "Symmetric-OR-XOR" are types of RSS algorithms that 57achieve this hash symmetry by XOR/ORing the input source and destination 58fields of the IP and/or L4 protocols. This, however, results in reduced 59input entropy and could potentially be exploited. 60 61Specifically, the "Symmetric-XOR" algorithm XORs the input 62as follows:: 63 64 # (SRC_IP ^ DST_IP, SRC_IP ^ DST_IP, SRC_PORT ^ DST_PORT, SRC_PORT ^ DST_PORT) 65 66The "Symmetric-OR-XOR" algorithm, on the other hand, transforms the input as 67follows:: 68 69 # (SRC_IP | DST_IP, SRC_IP ^ DST_IP, SRC_PORT | DST_PORT, SRC_PORT ^ DST_PORT) 70 71The result is then fed to the underlying RSS algorithm. 72 73Some advanced NICs allow steering packets to queues based on 74programmable filters. For example, webserver bound TCP port 80 packets 75can be directed to their own receive queue. Such “n-tuple” filters can 76be configured from ethtool (--config-ntuple). 77 78 79RSS Configuration 80----------------- 81 82The driver for a multi-queue capable NIC typically provides a kernel 83module parameter for specifying the number of hardware queues to 84configure. In the bnx2x driver, for instance, this parameter is called 85num_queues. A typical RSS configuration would be to have one receive queue 86for each CPU if the device supports enough queues, or otherwise at least 87one for each memory domain, where a memory domain is a set of CPUs that 88share a particular memory level (L1, L2, NUMA node, etc.). 89 90The indirection table of an RSS device, which resolves a queue by masked 91hash, is usually programmed by the driver at initialization. The 92default mapping is to distribute the queues evenly in the table, but the 93indirection table can be retrieved and modified at runtime using ethtool 94commands (--show-rxfh-indir and --set-rxfh-indir). Modifying the 95indirection table could be done to give different queues different 96relative weights. 97 98 99RSS IRQ Configuration 100~~~~~~~~~~~~~~~~~~~~~ 101 102Each receive queue has a separate IRQ associated with it. The NIC triggers 103this to notify a CPU when new packets arrive on the given queue. The 104signaling path for PCIe devices uses message signaled interrupts (MSI-X), 105that can route each interrupt to a particular CPU. The active mapping 106of queues to IRQs can be determined from /proc/interrupts. By default, 107an IRQ may be handled on any CPU. Because a non-negligible part of packet 108processing takes place in receive interrupt handling, it is advantageous 109to spread receive interrupts between CPUs. To manually adjust the IRQ 110affinity of each interrupt see Documentation/core-api/irq/irq-affinity.rst. Some systems 111will be running irqbalance, a daemon that dynamically optimizes IRQ 112assignments and as a result may override any manual settings. 113 114 115Suggested Configuration 116~~~~~~~~~~~~~~~~~~~~~~~ 117 118RSS should be enabled when latency is a concern or whenever receive 119interrupt processing forms a bottleneck. Spreading load between CPUs 120decreases queue length. For low latency networking, the optimal setting 121is to allocate as many queues as there are CPUs in the system (or the 122NIC maximum, if lower). The most efficient high-rate configuration 123is likely the one with the smallest number of receive queues where no 124receive queue overflows due to a saturated CPU, because in default 125mode with interrupt coalescing enabled, the aggregate number of 126interrupts (and thus work) grows with each additional queue. 127 128Per-cpu load can be observed using the mpstat utility, but note that on 129processors with hyperthreading (HT), each hyperthread is represented as 130a separate CPU. For interrupt handling, HT has shown no benefit in 131initial tests, so limit the number of queues to the number of CPU cores 132in the system. 133 134Dedicated RSS contexts 135~~~~~~~~~~~~~~~~~~~~~~ 136 137Modern NICs support creating multiple co-existing RSS configurations 138which are selected based on explicit matching rules. This can be very 139useful when application wants to constrain the set of queues receiving 140traffic for e.g. a particular destination port or IP address. 141The example below shows how to direct all traffic to TCP port 22 142to queues 0 and 1. 143 144To create an additional RSS context use:: 145 146 # ethtool -X eth0 hfunc toeplitz context new 147 New RSS context is 1 148 149Kernel reports back the ID of the allocated context (the default, always 150present RSS context has ID of 0). The new context can be queried and 151modified using the same APIs as the default context:: 152 153 # ethtool -x eth0 context 1 154 RX flow hash indirection table for eth0 with 13 RX ring(s): 155 0: 0 1 2 3 4 5 6 7 156 8: 8 9 10 11 12 0 1 2 157 [...] 158 # ethtool -X eth0 equal 2 context 1 159 # ethtool -x eth0 context 1 160 RX flow hash indirection table for eth0 with 13 RX ring(s): 161 0: 0 1 0 1 0 1 0 1 162 8: 0 1 0 1 0 1 0 1 163 [...] 164 165To make use of the new context direct traffic to it using an n-tuple 166filter:: 167 168 # ethtool -N eth0 flow-type tcp6 dst-port 22 context 1 169 Added rule with ID 1023 170 171When done, remove the context and the rule:: 172 173 # ethtool -N eth0 delete 1023 174 # ethtool -X eth0 context 1 delete 175 176 177RPS: Receive Packet Steering 178============================ 179 180Receive Packet Steering (RPS) is logically a software implementation of 181RSS. Being in software, it is necessarily called later in the datapath. 182Whereas RSS selects the queue and hence CPU that will run the hardware 183interrupt handler, RPS selects the CPU to perform protocol processing 184above the interrupt handler. This is accomplished by placing the packet 185on the desired CPU’s backlog queue and waking up the CPU for processing. 186RPS has some advantages over RSS: 187 1881) it can be used with any NIC 1892) software filters can easily be added to hash over new protocols 1903) it does not increase hardware device interrupt rate (although it does 191 introduce inter-processor interrupts (IPIs)) 192 193RPS is called during bottom half of the receive interrupt handler, when 194a driver sends a packet up the network stack with netif_rx() or 195netif_receive_skb(). These call the get_rps_cpu() function, which 196selects the queue that should process a packet. 197 198The first step in determining the target CPU for RPS is to calculate a 199flow hash over the packet’s addresses or ports (2-tuple or 4-tuple hash 200depending on the protocol). This serves as a consistent hash of the 201associated flow of the packet. The hash is either provided by hardware 202or will be computed in the stack. Capable hardware can pass the hash in 203the receive descriptor for the packet; this would usually be the same 204hash used for RSS (e.g. computed Toeplitz hash). The hash is saved in 205skb->hash and can be used elsewhere in the stack as a hash of the 206packet’s flow. 207 208Each receive hardware queue has an associated list of CPUs to which 209RPS may enqueue packets for processing. For each received packet, 210an index into the list is computed from the flow hash modulo the size 211of the list. The indexed CPU is the target for processing the packet, 212and the packet is queued to the tail of that CPU’s backlog queue. At 213the end of the bottom half routine, IPIs are sent to any CPUs for which 214packets have been queued to their backlog queue. The IPI wakes backlog 215processing on the remote CPU, and any queued packets are then processed 216up the networking stack. 217 218 219RPS Configuration 220----------------- 221 222RPS requires a kernel compiled with the CONFIG_RPS kconfig symbol (on 223by default for SMP). Even when compiled in, RPS remains disabled until 224explicitly configured. The list of CPUs to which RPS may forward traffic 225can be configured for each receive queue using a sysfs file entry:: 226 227 /sys/class/net/<dev>/queues/rx-<n>/rps_cpus 228 229This file implements a bitmap of CPUs. RPS is disabled when it is zero 230(the default), in which case packets are processed on the interrupting 231CPU. Documentation/core-api/irq/irq-affinity.rst explains how CPUs are assigned to 232the bitmap. 233 234 235Suggested Configuration 236~~~~~~~~~~~~~~~~~~~~~~~ 237 238For a single queue device, a typical RPS configuration would be to set 239the rps_cpus to the CPUs in the same memory domain of the interrupting 240CPU. If NUMA locality is not an issue, this could also be all CPUs in 241the system. At high interrupt rate, it might be wise to exclude the 242interrupting CPU from the map since that already performs much work. 243 244For a multi-queue system, if RSS is configured so that a hardware 245receive queue is mapped to each CPU, then RPS is probably redundant 246and unnecessary. If there are fewer hardware queues than CPUs, then 247RPS might be beneficial if the rps_cpus for each queue are the ones that 248share the same memory domain as the interrupting CPU for that queue. 249 250 251RPS Flow Limit 252-------------- 253 254RPS scales kernel receive processing across CPUs without introducing 255reordering. The trade-off to sending all packets from the same flow 256to the same CPU is CPU load imbalance if flows vary in packet rate. 257In the extreme case a single flow dominates traffic. Especially on 258common server workloads with many concurrent connections, such 259behavior indicates a problem such as a misconfiguration or spoofed 260source Denial of Service attack. 261 262Flow Limit is an optional RPS feature that prioritizes small flows 263during CPU contention by dropping packets from large flows slightly 264ahead of those from small flows. It is active only when an RPS or RFS 265destination CPU approaches saturation. Once a CPU's input packet 266queue exceeds half the maximum queue length (as set by sysctl 267net.core.netdev_max_backlog), the kernel starts a per-flow packet 268count over the last 256 packets. If a flow exceeds a set ratio (by 269default, half) of these packets when a new packet arrives, then the 270new packet is dropped. Packets from other flows are still only 271dropped once the input packet queue reaches netdev_max_backlog. 272No packets are dropped when the input packet queue length is below 273the threshold, so flow limit does not sever connections outright: 274even large flows maintain connectivity. 275 276 277Interface 278~~~~~~~~~ 279 280Flow limit is compiled in by default (CONFIG_NET_FLOW_LIMIT), but not 281turned on. It is implemented for each CPU independently (to avoid lock 282and cache contention) and toggled per CPU by setting the relevant bit 283in sysctl net.core.flow_limit_cpu_bitmap. It exposes the same CPU 284bitmap interface as rps_cpus (see above) when called from procfs:: 285 286 /proc/sys/net/core/flow_limit_cpu_bitmap 287 288Per-flow rate is calculated by hashing each packet into a hashtable 289bucket and incrementing a per-bucket counter. The hash function is 290the same that selects a CPU in RPS, but as the number of buckets can 291be much larger than the number of CPUs, flow limit has finer-grained 292identification of large flows and fewer false positives. The default 293table has 4096 buckets. This value can be modified through sysctl:: 294 295 net.core.flow_limit_table_len 296 297The value is only consulted when a new table is allocated. Modifying 298it does not update active tables. 299 300 301Suggested Configuration 302~~~~~~~~~~~~~~~~~~~~~~~ 303 304Flow limit is useful on systems with many concurrent connections, 305where a single connection taking up 50% of a CPU indicates a problem. 306In such environments, enable the feature on all CPUs that handle 307network rx interrupts (as set in /proc/irq/N/smp_affinity). 308 309The feature depends on the input packet queue length to exceed 310the flow limit threshold (50%) + the flow history length (256). 311Setting net.core.netdev_max_backlog to either 1000 or 10000 312performed well in experiments. 313 314 315RFS: Receive Flow Steering 316========================== 317 318While RPS steers packets solely based on hash, and thus generally 319provides good load distribution, it does not take into account 320application locality. This is accomplished by Receive Flow Steering 321(RFS). The goal of RFS is to increase datacache hitrate by steering 322kernel processing of packets to the CPU where the application thread 323consuming the packet is running. RFS relies on the same RPS mechanisms 324to enqueue packets onto the backlog of another CPU and to wake up that 325CPU. 326 327In RFS, packets are not forwarded directly by the value of their hash, 328but the hash is used as index into a flow lookup table. This table maps 329flows to the CPUs where those flows are being processed. The flow hash 330(see RPS section above) is used to calculate the index into this table. 331The CPU recorded in each entry is the one which last processed the flow. 332If an entry does not hold a valid CPU, then packets mapped to that entry 333are steered using plain RPS. Multiple table entries may point to the 334same CPU. Indeed, with many flows and few CPUs, it is very likely that 335a single application thread handles flows with many different flow hashes. 336 337rps_sock_flow_table is a global flow table that contains the *desired* CPU 338for flows: the CPU that is currently processing the flow in userspace. 339Each table value is a CPU index that is updated during calls to recvmsg 340and sendmsg (specifically, inet_recvmsg(), inet_sendmsg() and 341tcp_splice_read()). 342 343When the scheduler moves a thread to a new CPU while it has outstanding 344receive packets on the old CPU, packets may arrive out of order. To 345avoid this, RFS uses a second flow table to track outstanding packets 346for each flow: rps_dev_flow_table is a table specific to each hardware 347receive queue of each device. Each table value stores a CPU index and a 348counter. The CPU index represents the *current* CPU onto which packets 349for this flow are enqueued for further kernel processing. Ideally, kernel 350and userspace processing occur on the same CPU, and hence the CPU index 351in both tables is identical. This is likely false if the scheduler has 352recently migrated a userspace thread while the kernel still has packets 353enqueued for kernel processing on the old CPU. 354 355The counter in rps_dev_flow_table values records the length of the current 356CPU's backlog when a packet in this flow was last enqueued. Each backlog 357queue has a head counter that is incremented on dequeue. A tail counter 358is computed as head counter + queue length. In other words, the counter 359in rps_dev_flow[i] records the last element in flow i that has 360been enqueued onto the currently designated CPU for flow i (of course, 361entry i is actually selected by hash and multiple flows may hash to the 362same entry i). 363 364And now the trick for avoiding out of order packets: when selecting the 365CPU for packet processing (from get_rps_cpu()) the rps_sock_flow table 366and the rps_dev_flow table of the queue that the packet was received on 367are compared. If the desired CPU for the flow (found in the 368rps_sock_flow table) matches the current CPU (found in the rps_dev_flow 369table), the packet is enqueued onto that CPU’s backlog. If they differ, 370the current CPU is updated to match the desired CPU if one of the 371following is true: 372 373 - The current CPU's queue head counter >= the recorded tail counter 374 value in rps_dev_flow[i] 375 - The current CPU is unset (>= nr_cpu_ids) 376 - The current CPU is offline 377 378After this check, the packet is sent to the (possibly updated) current 379CPU. These rules aim to ensure that a flow only moves to a new CPU when 380there are no packets outstanding on the old CPU, as the outstanding 381packets could arrive later than those about to be processed on the new 382CPU. 383 384 385RFS Configuration 386----------------- 387 388RFS is only available if the kconfig symbol CONFIG_RPS is enabled (on 389by default for SMP). The functionality remains disabled until explicitly 390configured. The number of entries in the global flow table is set through:: 391 392 /proc/sys/net/core/rps_sock_flow_entries 393 394The number of entries in the per-queue flow table are set through:: 395 396 /sys/class/net/<dev>/queues/rx-<n>/rps_flow_cnt 397 398 399Suggested Configuration 400~~~~~~~~~~~~~~~~~~~~~~~ 401 402Both of these need to be set before RFS is enabled for a receive queue. 403Values for both are rounded up to the nearest power of two. The 404suggested flow count depends on the expected number of active connections 405at any given time, which may be significantly less than the number of open 406connections. We have found that a value of 65536 for rps_sock_flow_entries 407works fairly well on a moderately loaded server. Big servers might 408need 1048576 or even higher values. 409 410On a NUMA host it is advisable to spread rps_sock_flow_entries on all nodes. 411 412numactl --interleave=all bash -c "echo 1048576 >/proc/sys/net/core/rps_sock_flow_entries" 413 414For a single queue device, the rps_flow_cnt value for the single queue 415would normally be configured to the same value as rps_sock_flow_entries. 416For a multi-queue device, the rps_flow_cnt for each queue might be 417configured as rps_sock_flow_entries / N, where N is the number of 418queues. So for instance, if rps_sock_flow_entries is set to 131072 and there 419are 16 configured receive queues, rps_flow_cnt for each queue might be 420configured as 8192. 421 422 423Accelerated RFS 424=============== 425 426Accelerated RFS is to RFS what RSS is to RPS: a hardware-accelerated load 427balancing mechanism that uses soft state to steer flows based on where 428the application thread consuming the packets of each flow is running. 429Accelerated RFS should perform better than RFS since packets are sent 430directly to a CPU local to the thread consuming the data. The target CPU 431will either be the same CPU where the application runs, or at least a CPU 432which is local to the application thread’s CPU in the cache hierarchy. 433 434To enable accelerated RFS, the networking stack calls the 435ndo_rx_flow_steer driver function to communicate the desired hardware 436queue for packets matching a particular flow. The network stack 437automatically calls this function every time a flow entry in 438rps_dev_flow_table is updated. The driver in turn uses a device specific 439method to program the NIC to steer the packets. 440 441The hardware queue for a flow is derived from the CPU recorded in 442rps_dev_flow_table. The stack consults a CPU to hardware queue map which 443is maintained by the NIC driver. This is an auto-generated reverse map of 444the IRQ affinity table shown by /proc/interrupts. Drivers can use 445functions in the cpu_rmap (“CPU affinity reverse map”) kernel library 446to populate the map. Alternatively, drivers can delegate the cpu_rmap 447management to the Kernel by calling netif_enable_cpu_rmap(). For each CPU, 448the corresponding queue in the map is set to be one whose processing CPU is 449closest in cache locality. 450 451 452Accelerated RFS Configuration 453----------------------------- 454 455Accelerated RFS is only available if the kernel is compiled with 456CONFIG_RFS_ACCEL and support is provided by the NIC device and driver. 457It also requires that ntuple filtering is enabled via ethtool. The map 458of CPU to queues is automatically deduced from the IRQ affinities 459configured for each receive queue by the driver, so no additional 460configuration should be necessary. 461 462 463Suggested Configuration 464~~~~~~~~~~~~~~~~~~~~~~~ 465 466This technique should be enabled whenever one wants to use RFS and the 467NIC supports hardware acceleration. 468 469 470XPS: Transmit Packet Steering 471============================= 472 473Transmit Packet Steering is a mechanism for intelligently selecting 474which transmit queue to use when transmitting a packet on a multi-queue 475device. This can be accomplished by recording two kinds of maps, either 476a mapping of CPU to hardware queue(s) or a mapping of receive queue(s) 477to hardware transmit queue(s). 478 4791. XPS using CPUs map 480 481The goal of this mapping is usually to assign queues 482exclusively to a subset of CPUs, where the transmit completions for 483these queues are processed on a CPU within this set. This choice 484provides two benefits. First, contention on the device queue lock is 485significantly reduced since fewer CPUs contend for the same queue 486(contention can be eliminated completely if each CPU has its own 487transmit queue). Secondly, cache miss rate on transmit completion is 488reduced, in particular for data cache lines that hold the sk_buff 489structures. 490 4912. XPS using receive queues map 492 493This mapping is used to pick transmit queue based on the receive 494queue(s) map configuration set by the administrator. A set of receive 495queues can be mapped to a set of transmit queues (many:many), although 496the common use case is a 1:1 mapping. This will enable sending packets 497on the same queue associations for transmit and receive. This is useful for 498busy polling multi-threaded workloads where there are challenges in 499associating a given CPU to a given application thread. The application 500threads are not pinned to CPUs and each thread handles packets 501received on a single queue. The receive queue number is cached in the 502socket for the connection. In this model, sending the packets on the same 503transmit queue corresponding to the associated receive queue has benefits 504in keeping the CPU overhead low. Transmit completion work is locked into 505the same queue-association that a given application is polling on. This 506avoids the overhead of triggering an interrupt on another CPU. When the 507application cleans up the packets during the busy poll, transmit completion 508may be processed along with it in the same thread context and so result in 509reduced latency. 510 511XPS is configured per transmit queue by setting a bitmap of 512CPUs/receive-queues that may use that queue to transmit. The reverse 513mapping, from CPUs to transmit queues or from receive-queues to transmit 514queues, is computed and maintained for each network device. When 515transmitting the first packet in a flow, the function get_xps_queue() is 516called to select a queue. This function uses the ID of the receive queue 517for the socket connection for a match in the receive queue-to-transmit queue 518lookup table. Alternatively, this function can also use the ID of the 519running CPU as a key into the CPU-to-queue lookup table. If the 520ID matches a single queue, that is used for transmission. If multiple 521queues match, one is selected by using the flow hash to compute an index 522into the set. When selecting the transmit queue based on receive queue(s) 523map, the transmit device is not validated against the receive device as it 524requires expensive lookup operation in the datapath. 525 526The queue chosen for transmitting a particular flow is saved in the 527corresponding socket structure for the flow (e.g. a TCP connection). 528This transmit queue is used for subsequent packets sent on the flow to 529prevent out of order (ooo) packets. The choice also amortizes the cost 530of calling get_xps_queues() over all packets in the flow. To avoid 531ooo packets, the queue for a flow can subsequently only be changed if 532skb->ooo_okay is set for a packet in the flow. This flag indicates that 533there are no outstanding packets in the flow, so the transmit queue can 534change without the risk of generating out of order packets. The 535transport layer is responsible for setting ooo_okay appropriately. TCP, 536for instance, sets the flag when all data for a connection has been 537acknowledged. 538 539XPS Configuration 540----------------- 541 542XPS is only available if the kconfig symbol CONFIG_XPS is enabled (on by 543default for SMP). If compiled in, it is driver dependent whether, and 544how, XPS is configured at device init. The mapping of CPUs/receive-queues 545to transmit queue can be inspected and configured using sysfs: 546 547For selection based on CPUs map:: 548 549 /sys/class/net/<dev>/queues/tx-<n>/xps_cpus 550 551For selection based on receive-queues map:: 552 553 /sys/class/net/<dev>/queues/tx-<n>/xps_rxqs 554 555 556Suggested Configuration 557~~~~~~~~~~~~~~~~~~~~~~~ 558 559For a network device with a single transmission queue, XPS configuration 560has no effect, since there is no choice in this case. In a multi-queue 561system, XPS is preferably configured so that each CPU maps onto one queue. 562If there are as many queues as there are CPUs in the system, then each 563queue can also map onto one CPU, resulting in exclusive pairings that 564experience no contention. If there are fewer queues than CPUs, then the 565best CPUs to share a given queue are probably those that share the cache 566with the CPU that processes transmit completions for that queue 567(transmit interrupts). 568 569For transmit queue selection based on receive queue(s), XPS has to be 570explicitly configured mapping receive-queue(s) to transmit queue(s). If the 571user configuration for receive-queue map does not apply, then the transmit 572queue is selected based on the CPUs map. 573 574 575Per TX Queue rate limitation 576============================ 577 578These are rate-limitation mechanisms implemented by HW, where currently 579a max-rate attribute is supported, by setting a Mbps value to:: 580 581 /sys/class/net/<dev>/queues/tx-<n>/tx_maxrate 582 583A value of zero means disabled, and this is the default. 584 585 586Further Information 587=================== 588RPS and RFS were introduced in kernel 2.6.35. XPS was incorporated into 5892.6.38. Original patches were submitted by Tom Herbert 590(therbert@google.com) 591 592Accelerated RFS was introduced in 2.6.35. Original patches were 593submitted by Ben Hutchings (bwh@kernel.org) 594 595Authors: 596 597- Tom Herbert (therbert@google.com) 598- Willem de Bruijn (willemb@google.com) 599