mbox series

[00/26] rxrpc: Increasing SACK size and moving away from softirq, part 1

Message ID 166794587113.2389296.16484814996876530222.stgit@warthog.procyon.org.uk (mailing list archive)
Headers show
Series rxrpc: Increasing SACK size and moving away from softirq, part 1 | expand

Message

David Howells Nov. 8, 2022, 10:17 p.m. UTC
AF_RXRPC has some issues that need addressing:

 (1) The SACK table has a maximum capacity of 255, but for modern networks
     that isn't sufficient.  This is hard to increase in the upstream code
     because of the way the application thread is coupled to the softirq
     and retransmission side through a ring buffer.  Adjustments to the rx
     protocol allows a capacity of up to 8192, and having a ring
     sufficiently large to accommodate that would use an excessive amount
     of memory as this is per-call.

 (2) Processing ACKs in softirq mode causes the ACKs get conflated, with
     only the most recent being considered.  Whilst this has the upside
     that the retransmission algorithm only needs to deal with the most
     recent ACK, it causes DATA transmission for a call to be very bursty
     because DATA packets cannot be transmitted in softirq mode.  Rather
     transmission must be delegated to either the application thread or a
     workqueue, so there tend to be sudden bursts of traffic for any
     particular call due to scheduling delays.

 (3) All crypto in a single call is done in series; however, each DATA
     packet is individually encrypted so encryption and decryption of large
     calls could be parallelised if spare CPU resources are available.

This is the first of a number of sets of patches that try and address them.
The overall aims of these changes include:

 (1) To get rid of the TxRx ring and instead pass the packets round in
     queues (eg. sk_buff_head).  On the Tx side, each ACK packet comes with
     a SACK table that can be parsed as-is, so there's no particular need
     to maintain our own; we just have to refer to the ACK.

     On the Rx side, we do need to maintain a SACK table with one bit per
     entry - but only if packets go missing - and we don't want to have to
     perform a complex transformation to get the information into an ACK
     packet.

 (2) To try and move almost all processing of received packets out of the
     softirq handler and into a high-priority kernel I/O thread.  Only the
     transferral of packets would be left there.  I would still use the
     encap_rcv hook to receive packets as there's a noticeable performance
     drop from letting the UDP socket put the packets into its own queue
     and then getting them out of there.

 (3) To make the I/O thread also do all the transmission.  The app thread
     would be responsible for packaging the data into packets and then
     buffering them for the I/O thread to transmit.  This would make it
     easier for the app thread to run ahead of the I/O thread, and would
     mean the I/O thread is less likely to have to wait around for a new
     packet to come available for transmission.

 (4) To logically partition the socket/UAPI/KAPI side of things from the
     I/O side of things.  The local endpoint, connection, peer and call
     objects would belong to the I/O side.  The socket side would not then
     touch the private internals of calls and suchlike and would not change
     their states.  It would only look at the send queue, receive queue and
     a way to pass a message to cause an abort.

 (5) To remove as much locking, synchronisation, barriering and atomic ops
     as possible from the I/O side.  Exclusion would be achieved by
     limiting modification of state to the I/O thread only.  Locks would
     still need to be used in communication with the UDP socket and the
     AF_RXRPC socket API.

 (6) To provide crypto offload kernel threads that, when there's slack in
     the system, can see packets that need crypting and provide
     parallelisation in dealing with them.

 (7) To remove the use of system timers.  Since each timer would then send
     a poke to the I/O thread, which would then deal with it when it had
     the opportunity, there seems no point in using system timers if,
     instead, a list of timeouts can be sensibly consulted.  An I/O thread
     only then needs to schedule with a timeout when it is idle.

 (8) To use zero-copy sendmsg to send packets.  This would make use of the
     I/O thread being the sole transmitter on the socket to manage the
     dead-reckoning sequencing of the completion notifications.  There is a
     problem with zero-copy, though: the UDP socket doesn't handle running
     out of option memory very gracefully.

With regard to this first patchset, the changes made include:

 (1) Some fixes, including a fallback for proc_create_net_single_write(),
     setting ack.bufferSize to 0 in ACK packets and a fix for rxrpc
     congestion management, which shouldn't be saving the cwnd value
     between calls.

 (2) Improvements in rxrpc tracepoints, including splitting the timer
     tracepoint into a set-timer and a timer-expired trace.

 (3) Addition of a new proc file to display some stats.

 (4) Some code cleanups, including removing some unused bits and
     unnecessary header inclusions.

 (5) A change to the recently added UDP encap_err_rcv hook so that it has
     the same signature as {ip,ipv6}_icmp_error(), and then just have rxrpc
     point its UDP socket's hook directly at those.

 (6) Definition of a new struct, rxrpc_txbuf, that is used to hold
     transmissible packets of DATA and ACK type in a single 2KiB block
     rather than using an sk_buff.  This allows the buffer to be on a
     number of queues simultaneously more easily, and also guarantees that
     the entire block is in a single unit for zerocopy purposes and that
     the data payload is aligned for in-place crypto purposes.

 (7) ACK txbufs are allocated at proposal and queued for later transmission
     rather than being stored in a single place in the rxrpc_call struct,
     which means only a single ACK can be pending transmission at a time.
     The queue is then drained at various points.  This allows the ACK
     generation code to be simplified.

 (8) The Rx ring buffer is removed.  When a jumbo packet is received (which
     comprises a number of ordinary DATA packets glued together), it used
     to be pointed to by the ring multiple times, with an annotation in a
     side ring indicating which subpacket was in that slot - but this is no
     longer possible.  Instead, the packet is cloned once for each
     subpacket, barring the last, and the range of data is set in the skb
     private area.  This makes it easier for the subpackets in a jumbo
     packet to be decrypted in parallel.

 (9) The Tx ring buffer is removed.  The side annotation ring that held the
     SACK information is also removed.  Instead, in the event of packet
     loss, the SACK data attached an ACK packet is parsed.

(10) Allocate an skcipher request when needed in the rxkad security class
     rather than caching one in the rxrpc_call struct.  This deals with a
     race between externally-driven call disconnection getting rid of the
     skcipher request and sendmsg/recvmsg trying to use it because they
     haven't seen the completion yet.  This is also needed to support
     parallelisation as the skcipher request cannot be used by two or more
     threads simultaneously.

(11) Call udp_sendmsg() and udpv6_sendmsg() directly rather than going
     through kernel_sendmsg() so that we can provide our own iterator
     (zerocopy explicitly doesn't work with a KVEC iterator).  This also
     lets us avoid the overhead of the security hook.

---
The patches are tagged here:

	git://git.kernel.org/pub/scm/linux/kernel/git/dhowells/linux-fs.git tags/rxrpc-next-20221108

And can be found on this branch:

	http://git.kernel.org/cgit/linux/kernel/git/dhowells/linux-fs.git/log/?h=rxrpc-next

David
---
David Howells (26):
      net, proc: Provide PROC_FS=n fallback for proc_create_net_single_write()
      rxrpc: Trace setting of the request-ack flag
      rxrpc: Split call timer-expiration from call timer-set tracepoint
      rxrpc: Track highest acked serial
      rxrpc: Add stats procfile and DATA packet stats
      rxrpc: Record statistics about ACK types
      rxrpc: Record stats for why the REQUEST-ACK flag is being set
      rxrpc: Fix ack.bufferSize to be 0 when generating an ack
      net: Change the udp encap_err_rcv to allow use of {ip,ipv6}_icmp_error()
      rxrpc: Use the core ICMP/ICMP6 parsers
      rxrpc: Call udp_sendmsg() directly
      rxrpc: Remove unnecessary header inclusions
      rxrpc: Remove the flags from the rxrpc_skb tracepoint
      rxrpc: Remove call->tx_phase
      rxrpc: Define rxrpc_txbuf struct to carry data to be transmitted
      rxrpc: Allocate ACK records at proposal and queue for transmission
      rxrpc: Clean up ACK handling
      rxrpc: Split the rxrpc_recvmsg tracepoint
      rxrpc: Clone received jumbo subpackets and queue separately
      rxrpc: Get rid of the Rx ring
      rxrpc: Don't use a ring buffer for call Tx queue
      rxrpc: Remove call->lock
      rxrpc: Save last ACK's SACK table rather than marking txbufs
      rxrpc: Remove the rxtx ring
      rxrpc: Fix congestion management
      rxrpc: Allocate an skcipher each time needed rather than reusing


 include/linux/proc_fs.h      |   2 +
 include/linux/udp.h          |   3 +-
 include/net/udp_tunnel.h     |   4 +-
 include/trace/events/rxrpc.h | 361 ++++++++++++----
 net/core/skbuff.c            |   1 +
 net/ipv4/ip_sockglue.c       |   1 +
 net/ipv4/udp.c               |   3 +-
 net/ipv6/datagram.c          |   1 +
 net/ipv6/udp.c               |   4 +-
 net/rxrpc/Makefile           |   1 +
 net/rxrpc/af_rxrpc.c         |   5 +-
 net/rxrpc/ar-internal.h      | 224 ++++++----
 net/rxrpc/call_accept.c      |   8 +-
 net/rxrpc/call_event.c       | 427 +++++++++----------
 net/rxrpc/call_object.c      |  63 ++-
 net/rxrpc/conn_client.c      |   3 +-
 net/rxrpc/conn_object.c      |   4 +-
 net/rxrpc/input.c            | 770 +++++++++++++++++------------------
 net/rxrpc/insecure.c         |  16 +-
 net/rxrpc/local_object.c     |  20 +
 net/rxrpc/misc.c             |  23 +-
 net/rxrpc/net_ns.c           |   2 +
 net/rxrpc/output.c           | 390 ++++++++++--------
 net/rxrpc/peer_event.c       | 282 ++-----------
 net/rxrpc/peer_object.c      |   7 +-
 net/rxrpc/proc.c             | 110 ++++-
 net/rxrpc/protocol.h         |   9 +-
 net/rxrpc/recvmsg.c          | 268 ++++--------
 net/rxrpc/rxkad.c            | 245 ++++-------
 net/rxrpc/sendmsg.c          | 218 +++++-----
 net/rxrpc/skbuff.c           |  20 +-
 net/rxrpc/sysctl.c           |  11 +-
 net/rxrpc/txbuf.c            | 135 ++++++
 33 files changed, 1844 insertions(+), 1797 deletions(-)
 create mode 100644 net/rxrpc/txbuf.c

Comments

patchwork-bot+netdevbpf@kernel.org Nov. 9, 2022, 2:10 p.m. UTC | #1
Hello:

This series was applied to netdev/net-next.git (master)
by David Howells <dhowells@redhat.com>:

On Tue, 08 Nov 2022 22:17:51 +0000 you wrote:
> AF_RXRPC has some issues that need addressing:
> 
>  (1) The SACK table has a maximum capacity of 255, but for modern networks
>      that isn't sufficient.  This is hard to increase in the upstream code
>      because of the way the application thread is coupled to the softirq
>      and retransmission side through a ring buffer.  Adjustments to the rx
>      protocol allows a capacity of up to 8192, and having a ring
>      sufficiently large to accommodate that would use an excessive amount
>      of memory as this is per-call.
> 
> [...]

Here is the summary with links:
  - [net-next,01/26] net, proc: Provide PROC_FS=n fallback for proc_create_net_single_write()
    https://git.kernel.org/netdev/net-next/c/c3d96f690a79
  - [net-next,02/26] rxrpc: Trace setting of the request-ack flag
    https://git.kernel.org/netdev/net-next/c/4d843be56ba6
  - [net-next,03/26] rxrpc: Split call timer-expiration from call timer-set tracepoint
    https://git.kernel.org/netdev/net-next/c/334dfbfc5a71
  - [net-next,04/26] rxrpc: Track highest acked serial
    https://git.kernel.org/netdev/net-next/c/589a0c1e0ac3
  - [net-next,05/26] rxrpc: Add stats procfile and DATA packet stats
    https://git.kernel.org/netdev/net-next/c/b015424695f0
  - [net-next,06/26] rxrpc: Record statistics about ACK types
    https://git.kernel.org/netdev/net-next/c/f2a676d10038
  - [net-next,07/26] rxrpc: Record stats for why the REQUEST-ACK flag is being set
    https://git.kernel.org/netdev/net-next/c/f7fa52421f76
  - [net-next,08/26] rxrpc: Fix ack.bufferSize to be 0 when generating an ack
    https://git.kernel.org/netdev/net-next/c/8889a711f9b4
  - [net-next,09/26] net: Change the udp encap_err_rcv to allow use of {ip,ipv6}_icmp_error()
    https://git.kernel.org/netdev/net-next/c/42fb06b391ac
  - [net-next,10/26] rxrpc: Use the core ICMP/ICMP6 parsers
    https://git.kernel.org/netdev/net-next/c/b6c66c4324e7
  - [net-next,11/26] rxrpc: Call udp_sendmsg() directly
    https://git.kernel.org/netdev/net-next/c/ed472b0c8783
  - [net-next,12/26] rxrpc: Remove unnecessary header inclusions
    https://git.kernel.org/netdev/net-next/c/23b237f32592
  - [net-next,13/26] rxrpc: Remove the flags from the rxrpc_skb tracepoint
    https://git.kernel.org/netdev/net-next/c/27f699ccb89d
  - [net-next,14/26] rxrpc: Remove call->tx_phase
    https://git.kernel.org/netdev/net-next/c/a11e6ff961a0
  - [net-next,15/26] rxrpc: Define rxrpc_txbuf struct to carry data to be transmitted
    https://git.kernel.org/netdev/net-next/c/02a1935640f8
  - [net-next,16/26] rxrpc: Allocate ACK records at proposal and queue for transmission
    https://git.kernel.org/netdev/net-next/c/72f0c6fb0579
  - [net-next,17/26] rxrpc: Clean up ACK handling
    https://git.kernel.org/netdev/net-next/c/530403d9ba1c
  - [net-next,18/26] rxrpc: Split the rxrpc_recvmsg tracepoint
    https://git.kernel.org/netdev/net-next/c/faf92e8d53f5
  - [net-next,19/26] rxrpc: Clone received jumbo subpackets and queue separately
    https://git.kernel.org/netdev/net-next/c/d4d02d8bb5c4
  - [net-next,20/26] rxrpc: Get rid of the Rx ring
    https://git.kernel.org/netdev/net-next/c/5d7edbc9231e
  - [net-next,21/26] rxrpc: Don't use a ring buffer for call Tx queue
    https://git.kernel.org/netdev/net-next/c/a4ea4c477619
  - [net-next,22/26] rxrpc: Remove call->lock
    https://git.kernel.org/netdev/net-next/c/4e76bd406d6e
  - [net-next,23/26] rxrpc: Save last ACK's SACK table rather than marking txbufs
    https://git.kernel.org/netdev/net-next/c/d57a3a151660
  - [net-next,24/26] rxrpc: Remove the rxtx ring
    https://git.kernel.org/netdev/net-next/c/6869ddb87d47
  - [net-next,25/26] rxrpc: Fix congestion management
    https://git.kernel.org/netdev/net-next/c/1fc4fa2ac93d
  - [net-next,26/26] rxrpc: Allocate an skcipher each time needed rather than reusing
    https://git.kernel.org/netdev/net-next/c/30d95efe06e1

You are awesome, thank you!