7a03620a3b
- Rename output routines tcp_gen_* -> tcp_output_*. - Rename notification routines that turn in to no-ops in the absence of TOE from tcp_gen_* -> tcp_offload_*. - Fix some minor comment nits. - Add a /* FALLTHROUGH */ Reviewed by: Sam Leffler, Robert Watson, and Mike Silbersack
331 lines
11 KiB
C
331 lines
11 KiB
C
/*-
|
|
* Copyright (c) 2007, Chelsio Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Neither the name of the Chelsio Corporation nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#ifndef _NETINET_TCP_OFFLOAD_H_
|
|
#define _NETINET_TCP_OFFLOAD_H_
|
|
|
|
#ifndef _KERNEL
|
|
#error "no user-serviceable parts inside"
|
|
#endif
|
|
|
|
/*
|
|
* A driver publishes that it provides offload services
|
|
* by setting IFCAP_TOE in the ifnet. The offload connect
|
|
* will bypass any further work if the interface that a
|
|
* connection would use does not support TCP offload.
|
|
*
|
|
* The TOE API assumes that the tcp offload engine can offload the
|
|
* the entire connection from set up to teardown, with some provision
|
|
* being made to allowing the software stack to handle time wait. If
|
|
* the device does not meet these criteria, it is the driver's responsibility
|
|
* to overload the functions that it needs to in tcp_usrreqs and make
|
|
* its own calls to tcp_output if it needs to do so.
|
|
*
|
|
* There is currently no provision for the device advertising the congestion
|
|
* control algorithms it supports as there is currently no API for querying
|
|
* an operating system for the protocols that it has loaded. This is a desirable
|
|
* future extension.
|
|
*
|
|
*
|
|
*
|
|
* It is assumed that individuals deploying TOE will want connections
|
|
* to be offloaded without software changes so all connections on an
|
|
* interface providing TOE are offloaded unless the the SO_NO_OFFLOAD
|
|
* flag is set on the socket.
|
|
*
|
|
*
|
|
* The toe_usrreqs structure constitutes the TOE driver's
|
|
* interface to the TCP stack for functionality that doesn't
|
|
* interact directly with userspace. If one wants to provide
|
|
* (optional) functionality to do zero-copy to/from
|
|
* userspace one still needs to override soreceive/sosend
|
|
* with functions that fault in and pin the user buffers.
|
|
*
|
|
* + tu_send
|
|
* - tells the driver that new data may have been added to the
|
|
* socket's send buffer - the driver should not fail if the
|
|
* buffer is in fact unchanged
|
|
* - the driver is responsible for providing credits (bytes in the send window)
|
|
* back to the socket by calling sbdrop() as segments are acknowledged.
|
|
* - The driver expects the inpcb lock to be held - the driver is expected
|
|
* not to drop the lock. Hence the driver is not allowed to acquire the
|
|
* pcbinfo lock during this call.
|
|
*
|
|
* + tu_rcvd
|
|
* - returns credits to the driver and triggers window updates
|
|
* to the peer (a credit as used here is a byte in the peer's receive window)
|
|
* - the driver is expected to determine how many bytes have been
|
|
* consumed and credit that back to the card so that it can grow
|
|
* the window again by maintaining its own state between invocations.
|
|
* - In principle this could be used to shrink the window as well as
|
|
* grow the window, although it is not used for that now.
|
|
* - this function needs to correctly handle being called any number of
|
|
* times without any bytes being consumed from the receive buffer.
|
|
* - The driver expects the inpcb lock to be held - the driver is expected
|
|
* not to drop the lock. Hence the driver is not allowed to acquire the
|
|
* pcbinfo lock during this call.
|
|
*
|
|
* + tu_disconnect
|
|
* - tells the driver to send FIN to peer
|
|
* - driver is expected to send the remaining data and then do a clean half close
|
|
* - disconnect implies at least half-close so only send, reset, and detach
|
|
* are legal
|
|
* - the driver is expected to handle transition through the shutdown
|
|
* state machine and allow the stack to support SO_LINGER.
|
|
* - The driver expects the inpcb lock to be held - the driver is expected
|
|
* not to drop the lock. Hence the driver is not allowed to acquire the
|
|
* pcbinfo lock during this call.
|
|
*
|
|
* + tu_reset
|
|
* - closes the connection and sends a RST to peer
|
|
* - driver is expectd to trigger an RST and detach the toepcb
|
|
* - no further calls are legal after reset
|
|
* - The driver expects the inpcb lock to be held - the driver is expected
|
|
* not to drop the lock. Hence the driver is not allowed to acquire the
|
|
* pcbinfo lock during this call.
|
|
*
|
|
* The following fields in the tcpcb are expected to be referenced by the driver:
|
|
* + iss
|
|
* + rcv_nxt
|
|
* + rcv_wnd
|
|
* + snd_isn
|
|
* + snd_max
|
|
* + snd_nxt
|
|
* + snd_una
|
|
* + t_flags
|
|
* + t_inpcb
|
|
* + t_maxseg
|
|
* + t_toe
|
|
*
|
|
* The following fields in the inpcb are expected to be referenced by the driver:
|
|
* + inp_lport
|
|
* + inp_fport
|
|
* + inp_laddr
|
|
* + inp_fport
|
|
* + inp_socket
|
|
* + inp_ip_tos
|
|
*
|
|
* The following fields in the socket are expected to be referenced by the
|
|
* driver:
|
|
* + so_comp
|
|
* + so_error
|
|
* + so_linger
|
|
* + so_options
|
|
* + so_rcv
|
|
* + so_snd
|
|
* + so_state
|
|
* + so_timeo
|
|
*
|
|
* These functions all return 0 on success and can return the following errors
|
|
* as appropriate:
|
|
* + EPERM:
|
|
* + ENOBUFS: memory allocation failed
|
|
* + EMSGSIZE: MTU changed during the call
|
|
* + EHOSTDOWN:
|
|
* + EHOSTUNREACH:
|
|
* + ENETDOWN:
|
|
* * ENETUNREACH: the peer is no longer reachable
|
|
*
|
|
* + tu_detach
|
|
* - tells driver that the socket is going away so disconnect
|
|
* the toepcb and free appropriate resources
|
|
* - allows the driver to cleanly handle the case of connection state
|
|
* outliving the socket
|
|
* - no further calls are legal after detach
|
|
* - the driver is expected to provide its own synchronization between
|
|
* detach and receiving new data.
|
|
*
|
|
* + tu_syncache_event
|
|
* - even if it is not actually needed, the driver is expected to
|
|
* call syncache_add for the initial SYN and then syncache_expand
|
|
* for the SYN,ACK
|
|
* - tells driver that a connection either has not been added or has
|
|
* been dropped from the syncache
|
|
* - the driver is expected to maintain state that lives outside the
|
|
* software stack so the syncache needs to be able to notify the
|
|
* toe driver that the software stack is not going to create a connection
|
|
* for a received SYN
|
|
* - The driver is responsible for any synchronization required between
|
|
* the syncache dropping an entry and the driver processing the SYN,ACK.
|
|
*
|
|
*/
|
|
struct toe_usrreqs {
|
|
int (*tu_send)(struct tcpcb *tp);
|
|
int (*tu_rcvd)(struct tcpcb *tp);
|
|
int (*tu_disconnect)(struct tcpcb *tp);
|
|
int (*tu_reset)(struct tcpcb *tp);
|
|
void (*tu_detach)(struct tcpcb *tp);
|
|
void (*tu_syncache_event)(int event, void *toep);
|
|
};
|
|
|
|
#define TOE_SC_ENTRY_PRESENT 1 /* 4-tuple already present */
|
|
#define TOE_SC_DROP 2 /* connection was timed out */
|
|
|
|
/*
|
|
* Because listen is a one-to-many relationship (a socket can be listening
|
|
* on all interfaces on a machine some of which may be using different TCP
|
|
* offload devices), listen uses a publish/subscribe mechanism. The TCP
|
|
* offload driver registers a listen notification function with the stack.
|
|
* When a listen socket is created all TCP offload devices are notified
|
|
* so that they can do the appropriate set up to offload connections on the
|
|
* port to which the socket is bound. When the listen socket is closed,
|
|
* the offload devices are notified so that they will stop listening on that
|
|
* port and free any associated resources as well as sending RSTs on any
|
|
* connections in the SYN_RCVD state.
|
|
*
|
|
*/
|
|
|
|
typedef void (*tcp_offload_listen_start_fn)(void *, struct tcpcb *);
|
|
typedef void (*tcp_offload_listen_stop_fn)(void *, struct tcpcb *);
|
|
|
|
EVENTHANDLER_DECLARE(tcp_offload_listen_start, tcp_offload_listen_start_fn);
|
|
EVENTHANDLER_DECLARE(tcp_offload_listen_stop, tcp_offload_listen_stop_fn);
|
|
|
|
/*
|
|
* Check if the socket can be offloaded by the following steps:
|
|
* - determine the egress interface
|
|
* - check the interface for TOE capability and TOE is enabled
|
|
* - check if the device has resources to offload the connection
|
|
*/
|
|
int tcp_offload_connect(struct socket *so, struct sockaddr *nam);
|
|
|
|
/*
|
|
* The tcp_output_* routines are wrappers around the toe_usrreqs calls
|
|
* which trigger packet transmission. In the non-offloaded case they
|
|
* translate to tcp_output. The tcp_offload_* routines notify TOE
|
|
* of specific events. I the non-offloaded case they are no-ops.
|
|
*
|
|
* Listen is a special case because it is a 1 to many relationship
|
|
* and there can be more than one offload driver in the system.
|
|
*/
|
|
|
|
/*
|
|
* Connection is offloaded
|
|
*/
|
|
#define tp_offload(tp) ((tp)->t_flags & TF_TOE)
|
|
/*
|
|
* The socket has not been marked as "do not offload"
|
|
*/
|
|
#define SO_OFFLOADABLE(so) ((so->so_options & SO_NO_OFFLOAD) == 0)
|
|
|
|
static __inline int
|
|
tcp_output_connect(struct socket *so, struct sockaddr *nam)
|
|
{
|
|
struct tcpcb *tp = sototcpcb(so);
|
|
int error;
|
|
|
|
/*
|
|
* If offload has been disabled for this socket or the
|
|
* connection cannot be offloaded just call tcp_output
|
|
* to start the TCP state machine.
|
|
*/
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (!SO_OFFLOADABLE(so) || (error = tcp_offload_connect(so, nam)) != 0)
|
|
#endif
|
|
error = tcp_output(tp);
|
|
return (error);
|
|
}
|
|
|
|
static __inline int
|
|
tcp_output_send(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (tp_offload(tp))
|
|
return (tp->t_tu->tu_send(tp));
|
|
#endif
|
|
return (tcp_output(tp));
|
|
}
|
|
|
|
static __inline int
|
|
tcp_output_rcvd(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (tp_offload(tp))
|
|
return (tp->t_tu->tu_rcvd(tp));
|
|
#endif
|
|
return (tcp_output(tp));
|
|
}
|
|
|
|
static __inline int
|
|
tcp_output_disconnect(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (tp_offload(tp))
|
|
return (tp->t_tu->tu_disconnect(tp));
|
|
#endif
|
|
return (tcp_output(tp));
|
|
}
|
|
|
|
static __inline int
|
|
tcp_output_reset(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (tp_offload(tp))
|
|
return (tp->t_tu->tu_reset(tp));
|
|
#endif
|
|
return (tcp_output(tp));
|
|
}
|
|
|
|
static __inline void
|
|
tcp_offload_detach(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (tp_offload(tp))
|
|
tp->t_tu->tu_detach(tp);
|
|
#endif
|
|
}
|
|
|
|
static __inline void
|
|
tcp_offload_listen_open(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
if (SO_OFFLOADABLE(tp->t_inpcb->inp_socket))
|
|
EVENTHANDLER_INVOKE(tcp_offload_listen_start, tp);
|
|
#endif
|
|
}
|
|
|
|
static __inline void
|
|
tcp_offload_listen_close(struct tcpcb *tp)
|
|
{
|
|
|
|
#ifndef TCP_OFFLOAD_DISABLE
|
|
EVENTHANDLER_INVOKE(tcp_offload_listen_stop, tp);
|
|
#endif
|
|
}
|
|
|
|
#undef tp_offload
|
|
#undef SO_OFFLOADABLE
|
|
#endif /* _NETINET_TCP_OFFLOAD_H_ */
|