2005-08-12 03:25:23 +07:00
|
|
|
/*
|
|
|
|
* linux/net/sunrpc/xprtsock.c
|
|
|
|
*
|
|
|
|
* Client-side transport implementation for sockets.
|
|
|
|
*
|
2008-10-14 09:01:08 +07:00
|
|
|
* TCP callback races fixes (C) 1998 Red Hat
|
|
|
|
* TCP send fixes (C) 1998 Red Hat
|
2005-08-12 03:25:23 +07:00
|
|
|
* TCP NFS related read + write fixes
|
|
|
|
* (C) 1999 Dave Airlie, University of Limerick, Ireland <airlied@linux.ie>
|
|
|
|
*
|
|
|
|
* Rewrite of larges part of the code in order to stabilize TCP stuff.
|
|
|
|
* Fix behaviour when socket buffer is full.
|
|
|
|
* (C) 1999 Trond Myklebust <trond.myklebust@fys.uio.no>
|
2005-08-12 03:25:47 +07:00
|
|
|
*
|
|
|
|
* IP socket transport implementation, (C) 2005 Chuck Lever <cel@netapp.com>
|
2007-08-06 22:57:53 +07:00
|
|
|
*
|
|
|
|
* IPv6 support contributed by Gilles Quillard, Bull Open Source, 2005.
|
|
|
|
* <gilles.quillard@bull.net>
|
2005-08-12 03:25:23 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/types.h>
|
2011-05-10 02:22:44 +07:00
|
|
|
#include <linux/string.h>
|
2005-08-12 03:25:23 +07:00
|
|
|
#include <linux/slab.h>
|
2007-09-11 00:46:39 +07:00
|
|
|
#include <linux/module.h>
|
2005-08-12 03:25:23 +07:00
|
|
|
#include <linux/capability.h>
|
|
|
|
#include <linux/pagemap.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/socket.h>
|
|
|
|
#include <linux/in.h>
|
|
|
|
#include <linux/net.h>
|
|
|
|
#include <linux/mm.h>
|
2011-05-10 02:22:44 +07:00
|
|
|
#include <linux/un.h>
|
2005-08-12 03:25:23 +07:00
|
|
|
#include <linux/udp.h>
|
|
|
|
#include <linux/tcp.h>
|
|
|
|
#include <linux/sunrpc/clnt.h>
|
2013-02-05 00:50:00 +07:00
|
|
|
#include <linux/sunrpc/addr.h>
|
2006-01-03 15:55:49 +07:00
|
|
|
#include <linux/sunrpc/sched.h>
|
2009-09-10 21:32:28 +07:00
|
|
|
#include <linux/sunrpc/svcsock.h>
|
2007-09-11 00:47:31 +07:00
|
|
|
#include <linux/sunrpc/xprtsock.h>
|
2005-08-12 03:25:23 +07:00
|
|
|
#include <linux/file.h>
|
2011-07-14 06:20:49 +07:00
|
|
|
#ifdef CONFIG_SUNRPC_BACKCHANNEL
|
2009-04-01 20:23:02 +07:00
|
|
|
#include <linux/sunrpc/bc_xprt.h>
|
|
|
|
#endif
|
2005-08-12 03:25:23 +07:00
|
|
|
|
|
|
|
#include <net/sock.h>
|
|
|
|
#include <net/checksum.h>
|
|
|
|
#include <net/udp.h>
|
|
|
|
#include <net/tcp.h>
|
|
|
|
|
2013-09-04 23:16:23 +07:00
|
|
|
#include <trace/events/sunrpc.h>
|
|
|
|
|
2009-09-10 21:32:28 +07:00
|
|
|
#include "sunrpc.h"
|
2011-05-10 02:22:44 +07:00
|
|
|
|
|
|
|
static void xs_close(struct rpc_xprt *xprt);
|
|
|
|
|
2005-11-02 00:24:48 +07:00
|
|
|
/*
|
|
|
|
* xprtsock tunables
|
|
|
|
*/
|
2012-03-12 02:22:54 +07:00
|
|
|
static unsigned int xprt_udp_slot_table_entries = RPC_DEF_SLOT_TABLE;
|
|
|
|
static unsigned int xprt_tcp_slot_table_entries = RPC_MIN_SLOT_TABLE;
|
|
|
|
static unsigned int xprt_max_tcp_slot_table_entries = RPC_MAX_SLOT_TABLE;
|
2005-11-02 00:24:48 +07:00
|
|
|
|
2012-03-12 02:22:54 +07:00
|
|
|
static unsigned int xprt_min_resvport = RPC_DEF_MIN_RESVPORT;
|
|
|
|
static unsigned int xprt_max_resvport = RPC_DEF_MAX_RESVPORT;
|
2005-11-02 00:24:48 +07:00
|
|
|
|
2015-02-09 23:01:02 +07:00
|
|
|
#if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
|
|
|
|
|
2009-03-12 01:38:03 +07:00
|
|
|
#define XS_TCP_LINGER_TO (15U * HZ)
|
2009-03-12 01:38:03 +07:00
|
|
|
static unsigned int xs_tcp_fin_timeout __read_mostly = XS_TCP_LINGER_TO;
|
2009-03-12 01:38:03 +07:00
|
|
|
|
2006-12-06 04:35:54 +07:00
|
|
|
/*
|
|
|
|
* We can register our own files under /proc/sys/sunrpc by
|
|
|
|
* calling register_sysctl_table() again. The files in that
|
|
|
|
* directory become the union of all files registered there.
|
|
|
|
*
|
|
|
|
* We simply need to make sure that we don't collide with
|
|
|
|
* someone else's file names!
|
|
|
|
*/
|
|
|
|
|
|
|
|
static unsigned int min_slot_table_size = RPC_MIN_SLOT_TABLE;
|
|
|
|
static unsigned int max_slot_table_size = RPC_MAX_SLOT_TABLE;
|
2011-07-18 05:11:30 +07:00
|
|
|
static unsigned int max_tcp_slot_table_limit = RPC_MAX_SLOT_TABLE_LIMIT;
|
2006-12-06 04:35:54 +07:00
|
|
|
static unsigned int xprt_min_resvport_limit = RPC_MIN_RESVPORT;
|
|
|
|
static unsigned int xprt_max_resvport_limit = RPC_MAX_RESVPORT;
|
|
|
|
|
|
|
|
static struct ctl_table_header *sunrpc_table_header;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FIXME: changing the UDP slot table size should also resize the UDP
|
|
|
|
* socket buffers for existing UDP transports
|
|
|
|
*/
|
2013-06-12 13:04:25 +07:00
|
|
|
static struct ctl_table xs_tunables_table[] = {
|
2006-12-06 04:35:54 +07:00
|
|
|
{
|
|
|
|
.procname = "udp_slot_table_entries",
|
|
|
|
.data = &xprt_udp_slot_table_entries,
|
|
|
|
.maxlen = sizeof(unsigned int),
|
|
|
|
.mode = 0644,
|
2009-11-16 18:11:48 +07:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
2006-12-06 04:35:54 +07:00
|
|
|
.extra1 = &min_slot_table_size,
|
|
|
|
.extra2 = &max_slot_table_size
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.procname = "tcp_slot_table_entries",
|
|
|
|
.data = &xprt_tcp_slot_table_entries,
|
|
|
|
.maxlen = sizeof(unsigned int),
|
|
|
|
.mode = 0644,
|
2009-11-16 18:11:48 +07:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
2006-12-06 04:35:54 +07:00
|
|
|
.extra1 = &min_slot_table_size,
|
|
|
|
.extra2 = &max_slot_table_size
|
|
|
|
},
|
2011-07-18 05:11:30 +07:00
|
|
|
{
|
|
|
|
.procname = "tcp_max_slot_table_entries",
|
|
|
|
.data = &xprt_max_tcp_slot_table_entries,
|
|
|
|
.maxlen = sizeof(unsigned int),
|
|
|
|
.mode = 0644,
|
|
|
|
.proc_handler = proc_dointvec_minmax,
|
|
|
|
.extra1 = &min_slot_table_size,
|
|
|
|
.extra2 = &max_tcp_slot_table_limit
|
|
|
|
},
|
2006-12-06 04:35:54 +07:00
|
|
|
{
|
|
|
|
.procname = "min_resvport",
|
|
|
|
.data = &xprt_min_resvport,
|
|
|
|
.maxlen = sizeof(unsigned int),
|
|
|
|
.mode = 0644,
|
2009-11-16 18:11:48 +07:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
2006-12-06 04:35:54 +07:00
|
|
|
.extra1 = &xprt_min_resvport_limit,
|
|
|
|
.extra2 = &xprt_max_resvport_limit
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.procname = "max_resvport",
|
|
|
|
.data = &xprt_max_resvport,
|
|
|
|
.maxlen = sizeof(unsigned int),
|
|
|
|
.mode = 0644,
|
2009-11-16 18:11:48 +07:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
2006-12-06 04:35:54 +07:00
|
|
|
.extra1 = &xprt_min_resvport_limit,
|
|
|
|
.extra2 = &xprt_max_resvport_limit
|
|
|
|
},
|
2009-03-12 01:38:03 +07:00
|
|
|
{
|
|
|
|
.procname = "tcp_fin_timeout",
|
|
|
|
.data = &xs_tcp_fin_timeout,
|
|
|
|
.maxlen = sizeof(xs_tcp_fin_timeout),
|
|
|
|
.mode = 0644,
|
2009-11-16 18:11:48 +07:00
|
|
|
.proc_handler = proc_dointvec_jiffies,
|
2006-12-06 04:35:54 +07:00
|
|
|
},
|
2009-11-06 04:32:03 +07:00
|
|
|
{ },
|
2006-12-06 04:35:54 +07:00
|
|
|
};
|
|
|
|
|
2013-06-12 13:04:25 +07:00
|
|
|
static struct ctl_table sunrpc_table[] = {
|
2006-12-06 04:35:54 +07:00
|
|
|
{
|
|
|
|
.procname = "sunrpc",
|
|
|
|
.mode = 0555,
|
|
|
|
.child = xs_tunables_table
|
|
|
|
},
|
2009-11-06 04:32:03 +07:00
|
|
|
{ },
|
2006-12-06 04:35:54 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2005-08-26 06:25:55 +07:00
|
|
|
/*
|
|
|
|
* Wait duration for a reply from the RPC portmapper.
|
|
|
|
*/
|
|
|
|
#define XS_BIND_TO (60U * HZ)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Delay if a UDP socket connect error occurs. This is most likely some
|
|
|
|
* kind of resource problem on the local host.
|
|
|
|
*/
|
|
|
|
#define XS_UDP_REEST_TO (2U * HZ)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The reestablish timeout allows clients to delay for a bit before attempting
|
|
|
|
* to reconnect to a server that just dropped our connection.
|
|
|
|
*
|
|
|
|
* We implement an exponential backoff when trying to reestablish a TCP
|
|
|
|
* transport connection with the server. Some servers like to drop a TCP
|
|
|
|
* connection when they are overworked, so we start with a short timeout and
|
|
|
|
* increase over time if the server is down or not responding.
|
|
|
|
*/
|
|
|
|
#define XS_TCP_INIT_REEST_TO (3U * HZ)
|
|
|
|
#define XS_TCP_MAX_REEST_TO (5U * 60 * HZ)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TCP idle timeout; client drops the transport socket if it is idle
|
|
|
|
* for this long. Note that we also timeout UDP sockets to prevent
|
|
|
|
* holding port numbers when there is no RPC traffic.
|
|
|
|
*/
|
|
|
|
#define XS_IDLE_DISC_TO (5U * 60 * HZ)
|
|
|
|
|
2014-11-18 04:58:04 +07:00
|
|
|
#if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
|
2005-08-12 03:25:23 +07:00
|
|
|
# undef RPC_DEBUG_DATA
|
2005-08-12 03:25:26 +07:00
|
|
|
# define RPCDBG_FACILITY RPCDBG_TRANS
|
2005-08-12 03:25:23 +07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef RPC_DEBUG_DATA
|
2005-08-12 03:25:26 +07:00
|
|
|
static void xs_pktdump(char *msg, u32 *packet, unsigned int count)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2005-08-12 03:25:26 +07:00
|
|
|
u8 *buf = (u8 *) packet;
|
|
|
|
int j;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: %s\n", msg);
|
2005-08-12 03:25:23 +07:00
|
|
|
for (j = 0; j < count && j < 128; j += 4) {
|
|
|
|
if (!(j & 31)) {
|
|
|
|
if (j)
|
|
|
|
dprintk("\n");
|
|
|
|
dprintk("0x%04x ", j);
|
|
|
|
}
|
|
|
|
dprintk("%02x%02x%02x%02x ",
|
|
|
|
buf[j], buf[j+1], buf[j+2], buf[j+3]);
|
|
|
|
}
|
|
|
|
dprintk("\n");
|
|
|
|
}
|
|
|
|
#else
|
2005-08-12 03:25:26 +07:00
|
|
|
static inline void xs_pktdump(char *msg, u32 *packet, unsigned int count)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
/* NOP */
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-01-01 01:22:59 +07:00
|
|
|
static inline struct rpc_xprt *xprt_from_sock(struct sock *sk)
|
|
|
|
{
|
|
|
|
return (struct rpc_xprt *) sk->sk_user_data;
|
|
|
|
}
|
|
|
|
|
2007-08-06 22:57:58 +07:00
|
|
|
static inline struct sockaddr *xs_addr(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
return (struct sockaddr *) &xprt->addr;
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static inline struct sockaddr_un *xs_addr_un(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
return (struct sockaddr_un *) &xprt->addr;
|
|
|
|
}
|
|
|
|
|
2007-08-06 22:57:58 +07:00
|
|
|
static inline struct sockaddr_in *xs_addr_in(struct rpc_xprt *xprt)
|
2006-08-23 07:06:18 +07:00
|
|
|
{
|
2007-08-06 22:57:58 +07:00
|
|
|
return (struct sockaddr_in *) &xprt->addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct sockaddr_in6 *xs_addr_in6(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
return (struct sockaddr_in6 *) &xprt->addr;
|
|
|
|
}
|
|
|
|
|
2009-08-10 02:09:36 +07:00
|
|
|
static void xs_format_common_peer_addresses(struct rpc_xprt *xprt)
|
2006-08-23 07:06:18 +07:00
|
|
|
{
|
2009-08-10 02:09:36 +07:00
|
|
|
struct sockaddr *sap = xs_addr(xprt);
|
2009-08-10 02:09:46 +07:00
|
|
|
struct sockaddr_in6 *sin6;
|
|
|
|
struct sockaddr_in *sin;
|
2011-05-10 02:22:44 +07:00
|
|
|
struct sockaddr_un *sun;
|
2009-08-10 02:09:36 +07:00
|
|
|
char buf[128];
|
2006-08-23 07:06:18 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
switch (sap->sa_family) {
|
2011-05-10 02:22:44 +07:00
|
|
|
case AF_LOCAL:
|
|
|
|
sun = xs_addr_un(xprt);
|
|
|
|
strlcpy(buf, sun->sun_path, sizeof(buf));
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR] =
|
|
|
|
kstrdup(buf, GFP_KERNEL);
|
|
|
|
break;
|
2009-08-10 02:09:46 +07:00
|
|
|
case AF_INET:
|
2011-05-10 02:22:44 +07:00
|
|
|
(void)rpc_ntop(sap, buf, sizeof(buf));
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR] =
|
|
|
|
kstrdup(buf, GFP_KERNEL);
|
2009-08-10 02:09:46 +07:00
|
|
|
sin = xs_addr_in(xprt);
|
2010-03-09 03:15:28 +07:00
|
|
|
snprintf(buf, sizeof(buf), "%08x", ntohl(sin->sin_addr.s_addr));
|
2009-08-10 02:09:46 +07:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2011-05-10 02:22:44 +07:00
|
|
|
(void)rpc_ntop(sap, buf, sizeof(buf));
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR] =
|
|
|
|
kstrdup(buf, GFP_KERNEL);
|
2009-08-10 02:09:46 +07:00
|
|
|
sin6 = xs_addr_in6(xprt);
|
2010-03-09 03:15:28 +07:00
|
|
|
snprintf(buf, sizeof(buf), "%pi6", &sin6->sin6_addr);
|
2009-08-10 02:09:46 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG();
|
2007-08-17 03:03:26 +07:00
|
|
|
}
|
2011-05-10 02:22:44 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
xprt->address_strings[RPC_DISPLAY_HEX_ADDR] = kstrdup(buf, GFP_KERNEL);
|
2006-08-23 07:06:18 +07:00
|
|
|
}
|
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
static void xs_format_common_peer_ports(struct rpc_xprt *xprt)
|
2007-08-06 22:57:12 +07:00
|
|
|
{
|
2009-08-10 02:09:46 +07:00
|
|
|
struct sockaddr *sap = xs_addr(xprt);
|
|
|
|
char buf[128];
|
2007-08-06 22:57:12 +07:00
|
|
|
|
2010-03-09 03:15:59 +07:00
|
|
|
snprintf(buf, sizeof(buf), "%u", rpc_get_port(sap));
|
2009-08-10 02:09:36 +07:00
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT] = kstrdup(buf, GFP_KERNEL);
|
2007-08-06 22:57:12 +07:00
|
|
|
|
2010-03-09 03:15:59 +07:00
|
|
|
snprintf(buf, sizeof(buf), "%4hx", rpc_get_port(sap));
|
2009-08-10 02:09:36 +07:00
|
|
|
xprt->address_strings[RPC_DISPLAY_HEX_PORT] = kstrdup(buf, GFP_KERNEL);
|
|
|
|
}
|
2007-08-06 22:57:12 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
static void xs_format_peer_addresses(struct rpc_xprt *xprt,
|
|
|
|
const char *protocol,
|
|
|
|
const char *netid)
|
2009-08-10 02:09:36 +07:00
|
|
|
{
|
2008-01-08 06:34:48 +07:00
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO] = protocol;
|
|
|
|
xprt->address_strings[RPC_DISPLAY_NETID] = netid;
|
2009-08-10 02:09:36 +07:00
|
|
|
xs_format_common_peer_addresses(xprt);
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_common_peer_ports(xprt);
|
2006-08-23 07:06:18 +07:00
|
|
|
}
|
2007-08-06 22:57:12 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
static void xs_update_peer_port(struct rpc_xprt *xprt)
|
2007-08-06 22:57:12 +07:00
|
|
|
{
|
2009-08-10 02:09:46 +07:00
|
|
|
kfree(xprt->address_strings[RPC_DISPLAY_HEX_PORT]);
|
|
|
|
kfree(xprt->address_strings[RPC_DISPLAY_PORT]);
|
2007-09-11 00:43:05 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_common_peer_ports(xprt);
|
2006-08-23 07:06:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void xs_free_peer_addresses(struct rpc_xprt *xprt)
|
|
|
|
{
|
2008-01-15 00:32:20 +07:00
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < RPC_DISPLAY_MAX; i++)
|
|
|
|
switch (i) {
|
|
|
|
case RPC_DISPLAY_PROTO:
|
|
|
|
case RPC_DISPLAY_NETID:
|
|
|
|
continue;
|
|
|
|
default:
|
|
|
|
kfree(xprt->address_strings[i]);
|
|
|
|
}
|
2006-08-23 07:06:18 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:29 +07:00
|
|
|
#define XS_SENDMSG_FLAGS (MSG_DONTWAIT | MSG_NOSIGNAL)
|
|
|
|
|
2006-10-18 02:06:22 +07:00
|
|
|
static int xs_send_kvec(struct socket *sock, struct sockaddr *addr, int addrlen, struct kvec *vec, unsigned int base, int more)
|
2005-08-12 03:25:29 +07:00
|
|
|
{
|
|
|
|
struct msghdr msg = {
|
|
|
|
.msg_name = addr,
|
|
|
|
.msg_namelen = addrlen,
|
2006-10-18 02:06:22 +07:00
|
|
|
.msg_flags = XS_SENDMSG_FLAGS | (more ? MSG_MORE : 0),
|
|
|
|
};
|
|
|
|
struct kvec iov = {
|
|
|
|
.iov_base = vec->iov_base + base,
|
|
|
|
.iov_len = vec->iov_len - base,
|
2005-08-12 03:25:29 +07:00
|
|
|
};
|
|
|
|
|
2006-10-18 02:06:22 +07:00
|
|
|
if (iov.iov_len != 0)
|
2005-08-12 03:25:29 +07:00
|
|
|
return kernel_sendmsg(sock, &msg, &iov, 1, iov.iov_len);
|
|
|
|
return kernel_sendmsg(sock, &msg, NULL, 0, 0);
|
|
|
|
}
|
|
|
|
|
2014-09-25 01:08:00 +07:00
|
|
|
static int xs_send_pagedata(struct socket *sock, struct xdr_buf *xdr, unsigned int base, int more, bool zerocopy, int *sent_p)
|
2005-08-12 03:25:29 +07:00
|
|
|
{
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
ssize_t (*do_sendpage)(struct socket *sock, struct page *page,
|
|
|
|
int offset, size_t size, int flags);
|
2006-10-18 02:06:22 +07:00
|
|
|
struct page **ppage;
|
|
|
|
unsigned int remainder;
|
2014-09-25 01:08:00 +07:00
|
|
|
int err;
|
2006-10-18 02:06:22 +07:00
|
|
|
|
|
|
|
remainder = xdr->page_len - base;
|
|
|
|
base += xdr->page_base;
|
|
|
|
ppage = xdr->pages + (base >> PAGE_SHIFT);
|
|
|
|
base &= ~PAGE_MASK;
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
do_sendpage = sock->ops->sendpage;
|
|
|
|
if (!zerocopy)
|
|
|
|
do_sendpage = sock_no_sendpage;
|
2006-10-18 02:06:22 +07:00
|
|
|
for(;;) {
|
|
|
|
unsigned int len = min_t(unsigned int, PAGE_SIZE - base, remainder);
|
|
|
|
int flags = XS_SENDMSG_FLAGS;
|
2005-08-12 03:25:29 +07:00
|
|
|
|
2006-10-18 02:06:22 +07:00
|
|
|
remainder -= len;
|
|
|
|
if (remainder != 0 || more)
|
|
|
|
flags |= MSG_MORE;
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
err = do_sendpage(sock, *ppage, base, len, flags);
|
2006-10-18 02:06:22 +07:00
|
|
|
if (remainder == 0 || err != len)
|
|
|
|
break;
|
2014-09-25 01:08:00 +07:00
|
|
|
*sent_p += err;
|
2006-10-18 02:06:22 +07:00
|
|
|
ppage++;
|
|
|
|
base = 0;
|
|
|
|
}
|
2014-09-25 01:08:00 +07:00
|
|
|
if (err > 0) {
|
|
|
|
*sent_p += err;
|
|
|
|
err = 0;
|
|
|
|
}
|
|
|
|
return err;
|
2005-08-12 03:25:29 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_sendpages - write pages directly to a socket
|
|
|
|
* @sock: socket to send on
|
|
|
|
* @addr: UDP only -- address of destination
|
|
|
|
* @addrlen: UDP only -- length of destination address
|
|
|
|
* @xdr: buffer containing this request
|
|
|
|
* @base: starting position in the buffer
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
* @zerocopy: true if it is safe to use sendpage()
|
2014-09-25 01:08:00 +07:00
|
|
|
* @sent_p: return the total number of bytes successfully queued for sending
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
2005-08-12 03:25:23 +07:00
|
|
|
*/
|
2014-09-25 01:08:00 +07:00
|
|
|
static int xs_sendpages(struct socket *sock, struct sockaddr *addr, int addrlen, struct xdr_buf *xdr, unsigned int base, bool zerocopy, int *sent_p)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-10-18 02:06:22 +07:00
|
|
|
unsigned int remainder = xdr->len - base;
|
2014-09-25 01:08:00 +07:00
|
|
|
int err = 0;
|
|
|
|
int sent = 0;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
if (unlikely(!sock))
|
2009-03-12 01:06:41 +07:00
|
|
|
return -ENOTSOCK;
|
2005-08-12 03:25:56 +07:00
|
|
|
|
|
|
|
clear_bit(SOCK_ASYNC_NOSPACE, &sock->flags);
|
2006-10-18 02:06:22 +07:00
|
|
|
if (base != 0) {
|
|
|
|
addr = NULL;
|
|
|
|
addrlen = 0;
|
|
|
|
}
|
2005-08-12 03:25:56 +07:00
|
|
|
|
2006-10-18 02:06:22 +07:00
|
|
|
if (base < xdr->head[0].iov_len || addr != NULL) {
|
|
|
|
unsigned int len = xdr->head[0].iov_len - base;
|
|
|
|
remainder -= len;
|
|
|
|
err = xs_send_kvec(sock, addr, addrlen, &xdr->head[0], base, remainder != 0);
|
|
|
|
if (remainder == 0 || err != len)
|
2005-08-12 03:25:23 +07:00
|
|
|
goto out;
|
2014-09-25 01:08:00 +07:00
|
|
|
*sent_p += err;
|
2005-08-12 03:25:23 +07:00
|
|
|
base = 0;
|
|
|
|
} else
|
2006-10-18 02:06:22 +07:00
|
|
|
base -= xdr->head[0].iov_len;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-10-18 02:06:22 +07:00
|
|
|
if (base < xdr->page_len) {
|
|
|
|
unsigned int len = xdr->page_len - base;
|
|
|
|
remainder -= len;
|
2014-09-25 01:08:00 +07:00
|
|
|
err = xs_send_pagedata(sock, xdr, base, remainder != 0, zerocopy, &sent);
|
|
|
|
*sent_p += sent;
|
|
|
|
if (remainder == 0 || sent != len)
|
2005-08-12 03:25:23 +07:00
|
|
|
goto out;
|
|
|
|
base = 0;
|
2006-10-18 02:06:22 +07:00
|
|
|
} else
|
|
|
|
base -= xdr->page_len;
|
|
|
|
|
|
|
|
if (base >= xdr->tail[0].iov_len)
|
2014-09-25 01:08:00 +07:00
|
|
|
return 0;
|
2006-10-18 02:06:22 +07:00
|
|
|
err = xs_send_kvec(sock, NULL, 0, &xdr->tail[0], base, 0);
|
2005-08-12 03:25:23 +07:00
|
|
|
out:
|
2014-09-25 01:08:00 +07:00
|
|
|
if (err > 0) {
|
|
|
|
*sent_p += err;
|
|
|
|
err = 0;
|
|
|
|
}
|
|
|
|
return err;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2008-04-18 05:52:19 +07:00
|
|
|
static void xs_nospace_callback(struct rpc_task *task)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(task->tk_rqstp->rq_xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
transport->inet->sk_write_pending--;
|
|
|
|
clear_bit(SOCK_ASYNC_NOSPACE, &transport->sock->flags);
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
2005-08-12 03:25:56 +07:00
|
|
|
* xs_nospace - place task on wait queue if transmit was incomplete
|
|
|
|
* @task: task to put to sleep
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
2005-08-12 03:25:23 +07:00
|
|
|
*/
|
2009-03-12 01:38:01 +07:00
|
|
|
static int xs_nospace(struct rpc_task *task)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2005-08-12 03:25:56 +07:00
|
|
|
struct rpc_rqst *req = task->tk_rqstp;
|
|
|
|
struct rpc_xprt *xprt = req->rq_xprt;
|
2006-12-06 04:35:15 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2014-02-11 21:15:54 +07:00
|
|
|
struct sock *sk = transport->inet;
|
2011-11-22 19:44:28 +07:00
|
|
|
int ret = -EAGAIN;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: %5u xmit incomplete (%u left of %u)\n",
|
2005-08-12 03:25:56 +07:00
|
|
|
task->tk_pid, req->rq_slen - req->rq_bytes_sent,
|
|
|
|
req->rq_slen);
|
|
|
|
|
2008-04-18 05:52:19 +07:00
|
|
|
/* Protect against races with write_space */
|
|
|
|
spin_lock_bh(&xprt->transport_lock);
|
|
|
|
|
|
|
|
/* Don't race with disconnect */
|
|
|
|
if (xprt_connected(xprt)) {
|
|
|
|
if (test_bit(SOCK_ASYNC_NOSPACE, &transport->sock->flags)) {
|
|
|
|
/*
|
|
|
|
* Notify TCP that we're limited by the application
|
|
|
|
* window size
|
|
|
|
*/
|
|
|
|
set_bit(SOCK_NOSPACE, &transport->sock->flags);
|
2014-02-11 21:15:54 +07:00
|
|
|
sk->sk_write_pending++;
|
2008-04-18 05:52:19 +07:00
|
|
|
/* ...and wait for more buffer space */
|
|
|
|
xprt_wait_for_buffer_space(task, xs_nospace_callback);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
clear_bit(SOCK_ASYNC_NOSPACE, &transport->sock->flags);
|
2009-03-12 01:38:01 +07:00
|
|
|
ret = -ENOTCONN;
|
2008-04-18 05:52:19 +07:00
|
|
|
}
|
2005-08-12 03:25:56 +07:00
|
|
|
|
2008-04-18 05:52:19 +07:00
|
|
|
spin_unlock_bh(&xprt->transport_lock);
|
2014-02-11 21:15:54 +07:00
|
|
|
|
|
|
|
/* Race breaker in case memory is freed before above code is called */
|
|
|
|
sk->sk_write_space(sk);
|
2009-03-12 01:38:01 +07:00
|
|
|
return ret;
|
2005-08-12 03:25:56 +07:00
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:34 +07:00
|
|
|
/*
|
|
|
|
* Construct a stream transport record marker in @buf.
|
|
|
|
*/
|
|
|
|
static inline void xs_encode_stream_record_marker(struct xdr_buf *buf)
|
|
|
|
{
|
|
|
|
u32 reclen = buf->len - sizeof(rpc_fraghdr);
|
|
|
|
rpc_fraghdr *base = buf->head[0].iov_base;
|
|
|
|
*base = cpu_to_be32(RPC_LAST_STREAM_FRAGMENT | reclen);
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
/**
|
|
|
|
* xs_local_send_request - write an RPC request to an AF_LOCAL socket
|
|
|
|
* @task: RPC task that manages the state of an RPC request
|
|
|
|
*
|
|
|
|
* Return values:
|
|
|
|
* 0: The request has been sent
|
|
|
|
* EAGAIN: The socket was blocked, please call again later to
|
|
|
|
* complete the request
|
|
|
|
* ENOTCONN: Caller needs to invoke connect logic then call again
|
|
|
|
* other: Some other error occured, the request was not sent
|
|
|
|
*/
|
|
|
|
static int xs_local_send_request(struct rpc_task *task)
|
|
|
|
{
|
|
|
|
struct rpc_rqst *req = task->tk_rqstp;
|
|
|
|
struct rpc_xprt *xprt = req->rq_xprt;
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct xdr_buf *xdr = &req->rq_snd_buf;
|
|
|
|
int status;
|
2014-09-25 01:08:00 +07:00
|
|
|
int sent = 0;
|
2011-05-10 02:22:44 +07:00
|
|
|
|
|
|
|
xs_encode_stream_record_marker(&req->rq_snd_buf);
|
|
|
|
|
|
|
|
xs_pktdump("packet data:",
|
|
|
|
req->rq_svec->iov_base, req->rq_svec->iov_len);
|
|
|
|
|
2014-09-25 01:08:00 +07:00
|
|
|
status = xs_sendpages(transport->sock, NULL, 0, xdr, req->rq_bytes_sent,
|
|
|
|
true, &sent);
|
2011-05-10 02:22:44 +07:00
|
|
|
dprintk("RPC: %s(%u) = %d\n",
|
|
|
|
__func__, xdr->len - req->rq_bytes_sent, status);
|
2015-07-27 07:55:35 +07:00
|
|
|
|
|
|
|
if (status == -EAGAIN && sock_writeable(transport->inet))
|
|
|
|
status = -ENOBUFS;
|
|
|
|
|
2014-09-25 01:08:00 +07:00
|
|
|
if (likely(sent > 0) || status == 0) {
|
|
|
|
req->rq_bytes_sent += sent;
|
|
|
|
req->rq_xmit_bytes_sent += sent;
|
2011-05-10 02:22:44 +07:00
|
|
|
if (likely(req->rq_bytes_sent >= req->rq_slen)) {
|
|
|
|
req->rq_bytes_sent = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
status = -EAGAIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (status) {
|
2014-07-01 00:42:19 +07:00
|
|
|
case -ENOBUFS:
|
2015-07-03 20:32:23 +07:00
|
|
|
break;
|
2011-05-10 02:22:44 +07:00
|
|
|
case -EAGAIN:
|
|
|
|
status = xs_nospace(task);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dprintk("RPC: sendmsg returned unrecognized error %d\n",
|
|
|
|
-status);
|
|
|
|
case -EPIPE:
|
|
|
|
xs_close(xprt);
|
|
|
|
status = -ENOTCONN;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
/**
|
|
|
|
* xs_udp_send_request - write an RPC request to a UDP socket
|
|
|
|
* @task: address of RPC task that manages the state of an RPC request
|
|
|
|
*
|
|
|
|
* Return values:
|
|
|
|
* 0: The request has been sent
|
|
|
|
* EAGAIN: The socket was blocked, please call again later to
|
|
|
|
* complete the request
|
|
|
|
* ENOTCONN: Caller needs to invoke connect logic then call again
|
2011-03-31 08:57:33 +07:00
|
|
|
* other: Some other error occurred, the request was not sent
|
2005-08-12 03:25:56 +07:00
|
|
|
*/
|
|
|
|
static int xs_udp_send_request(struct rpc_task *task)
|
|
|
|
{
|
|
|
|
struct rpc_rqst *req = task->tk_rqstp;
|
|
|
|
struct rpc_xprt *xprt = req->rq_xprt;
|
2006-12-06 04:35:15 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:56 +07:00
|
|
|
struct xdr_buf *xdr = &req->rq_snd_buf;
|
2014-09-25 01:08:00 +07:00
|
|
|
int sent = 0;
|
2005-08-12 03:25:56 +07:00
|
|
|
int status;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
xs_pktdump("packet data:",
|
2005-08-12 03:25:23 +07:00
|
|
|
req->rq_svec->iov_base,
|
|
|
|
req->rq_svec->iov_len);
|
|
|
|
|
2009-03-12 01:09:39 +07:00
|
|
|
if (!xprt_bound(xprt))
|
|
|
|
return -ENOTCONN;
|
2014-09-25 01:08:00 +07:00
|
|
|
status = xs_sendpages(transport->sock, xs_addr(xprt), xprt->addrlen,
|
|
|
|
xdr, req->rq_bytes_sent, true, &sent);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_udp_send_request(%u) = %d\n",
|
2005-08-12 03:25:56 +07:00
|
|
|
xdr->len - req->rq_bytes_sent, status);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
rpc: Add -EPERM processing for xs_udp_send_request()
If an iptables drop rule is added for an nfs server, the client can end up in
a softlockup. Because of the way that xs_sendpages() is structured, the -EPERM
is ignored since the prior bits of the packet may have been successfully queued
and thus xs_sendpages() returns a non-zero value. Then, xs_udp_send_request()
thinks that because some bits were queued it should return -EAGAIN. We then try
the request again and again, resulting in cpu spinning. Reproducer:
1) open a file on the nfs server '/nfs/foo' (mounted using udp)
2) iptables -A OUTPUT -d <nfs server ip> -j DROP
3) write to /nfs/foo
4) close /nfs/foo
5) iptables -D OUTPUT -d <nfs server ip> -j DROP
The softlockup occurs in step 4 above.
The previous patch, allows xs_sendpages() to return both a sent count and
any error values that may have occurred. Thus, if we get an -EPERM, return
that to the higher level code.
With this patch in place we can successfully abort the above sequence and
avoid the softlockup.
I also tried the above test case on an nfs mount on tcp and although the system
does not softlockup, I still ended up with the 'hung_task' firing after 120
seconds, due to the i/o being stuck. The tcp case appears a bit harder to fix,
since -EPERM appears to get ignored much lower down in the stack and does not
propogate up to xs_sendpages(). This case is not quite as insidious as the
softlockup and it is not addressed here.
Reported-by: Yigong Lou <ylou@akamai.com>
Signed-off-by: Jason Baron <jbaron@akamai.com>
Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
2014-09-25 01:08:04 +07:00
|
|
|
/* firewall is blocking us, don't return -EAGAIN or we end up looping */
|
|
|
|
if (status == -EPERM)
|
|
|
|
goto process_status;
|
|
|
|
|
2015-07-27 07:55:35 +07:00
|
|
|
if (status == -EAGAIN && sock_writeable(transport->inet))
|
|
|
|
status = -ENOBUFS;
|
|
|
|
|
2014-09-25 01:08:00 +07:00
|
|
|
if (sent > 0 || status == 0) {
|
|
|
|
req->rq_xmit_bytes_sent += sent;
|
|
|
|
if (sent >= req->rq_slen)
|
2007-10-01 22:43:37 +07:00
|
|
|
return 0;
|
|
|
|
/* Still some bytes left; set up for a retry later. */
|
2005-08-12 03:25:56 +07:00
|
|
|
status = -EAGAIN;
|
2007-10-01 22:43:37 +07:00
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
|
rpc: Add -EPERM processing for xs_udp_send_request()
If an iptables drop rule is added for an nfs server, the client can end up in
a softlockup. Because of the way that xs_sendpages() is structured, the -EPERM
is ignored since the prior bits of the packet may have been successfully queued
and thus xs_sendpages() returns a non-zero value. Then, xs_udp_send_request()
thinks that because some bits were queued it should return -EAGAIN. We then try
the request again and again, resulting in cpu spinning. Reproducer:
1) open a file on the nfs server '/nfs/foo' (mounted using udp)
2) iptables -A OUTPUT -d <nfs server ip> -j DROP
3) write to /nfs/foo
4) close /nfs/foo
5) iptables -D OUTPUT -d <nfs server ip> -j DROP
The softlockup occurs in step 4 above.
The previous patch, allows xs_sendpages() to return both a sent count and
any error values that may have occurred. Thus, if we get an -EPERM, return
that to the higher level code.
With this patch in place we can successfully abort the above sequence and
avoid the softlockup.
I also tried the above test case on an nfs mount on tcp and although the system
does not softlockup, I still ended up with the 'hung_task' firing after 120
seconds, due to the i/o being stuck. The tcp case appears a bit harder to fix,
since -EPERM appears to get ignored much lower down in the stack and does not
propogate up to xs_sendpages(). This case is not quite as insidious as the
softlockup and it is not addressed here.
Reported-by: Yigong Lou <ylou@akamai.com>
Signed-off-by: Jason Baron <jbaron@akamai.com>
Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
2014-09-25 01:08:04 +07:00
|
|
|
process_status:
|
2005-08-12 03:25:56 +07:00
|
|
|
switch (status) {
|
2009-03-12 01:06:41 +07:00
|
|
|
case -ENOTSOCK:
|
|
|
|
status = -ENOTCONN;
|
|
|
|
/* Should we call xs_close() here? */
|
|
|
|
break;
|
2008-04-18 05:52:19 +07:00
|
|
|
case -EAGAIN:
|
2009-03-12 01:38:01 +07:00
|
|
|
status = xs_nospace(task);
|
2008-04-18 05:52:19 +07:00
|
|
|
break;
|
2009-03-12 01:37:59 +07:00
|
|
|
default:
|
|
|
|
dprintk("RPC: sendmsg returned unrecognized error %d\n",
|
|
|
|
-status);
|
2005-08-12 03:25:56 +07:00
|
|
|
case -ENETUNREACH:
|
2014-07-01 00:42:19 +07:00
|
|
|
case -ENOBUFS:
|
2005-08-12 03:25:56 +07:00
|
|
|
case -EPIPE:
|
2005-08-12 03:25:23 +07:00
|
|
|
case -ECONNREFUSED:
|
rpc: Add -EPERM processing for xs_udp_send_request()
If an iptables drop rule is added for an nfs server, the client can end up in
a softlockup. Because of the way that xs_sendpages() is structured, the -EPERM
is ignored since the prior bits of the packet may have been successfully queued
and thus xs_sendpages() returns a non-zero value. Then, xs_udp_send_request()
thinks that because some bits were queued it should return -EAGAIN. We then try
the request again and again, resulting in cpu spinning. Reproducer:
1) open a file on the nfs server '/nfs/foo' (mounted using udp)
2) iptables -A OUTPUT -d <nfs server ip> -j DROP
3) write to /nfs/foo
4) close /nfs/foo
5) iptables -D OUTPUT -d <nfs server ip> -j DROP
The softlockup occurs in step 4 above.
The previous patch, allows xs_sendpages() to return both a sent count and
any error values that may have occurred. Thus, if we get an -EPERM, return
that to the higher level code.
With this patch in place we can successfully abort the above sequence and
avoid the softlockup.
I also tried the above test case on an nfs mount on tcp and although the system
does not softlockup, I still ended up with the 'hung_task' firing after 120
seconds, due to the i/o being stuck. The tcp case appears a bit harder to fix,
since -EPERM appears to get ignored much lower down in the stack and does not
propogate up to xs_sendpages(). This case is not quite as insidious as the
softlockup and it is not addressed here.
Reported-by: Yigong Lou <ylou@akamai.com>
Signed-off-by: Jason Baron <jbaron@akamai.com>
Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
2014-09-25 01:08:04 +07:00
|
|
|
case -EPERM:
|
2005-08-12 03:25:23 +07:00
|
|
|
/* When the server has died, an ICMP port unreachable message
|
2005-08-12 03:25:26 +07:00
|
|
|
* prompts ECONNREFUSED. */
|
2008-04-18 05:52:19 +07:00
|
|
|
clear_bit(SOCK_ASYNC_NOSPACE, &transport->sock->flags);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2010-03-08 13:49:01 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
return status;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
2005-08-12 03:25:56 +07:00
|
|
|
* xs_tcp_send_request - write an RPC request to a TCP socket
|
2005-08-12 03:25:26 +07:00
|
|
|
* @task: address of RPC task that manages the state of an RPC request
|
|
|
|
*
|
|
|
|
* Return values:
|
2005-08-12 03:25:56 +07:00
|
|
|
* 0: The request has been sent
|
|
|
|
* EAGAIN: The socket was blocked, please call again later to
|
|
|
|
* complete the request
|
|
|
|
* ENOTCONN: Caller needs to invoke connect logic then call again
|
2011-03-31 08:57:33 +07:00
|
|
|
* other: Some other error occurred, the request was not sent
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
|
|
|
* XXX: In the case of soft timeouts, should we eventually give up
|
2005-08-12 03:25:56 +07:00
|
|
|
* if sendmsg is not able to make progress?
|
2005-08-12 03:25:26 +07:00
|
|
|
*/
|
2005-08-12 03:25:56 +07:00
|
|
|
static int xs_tcp_send_request(struct rpc_task *task)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
struct rpc_rqst *req = task->tk_rqstp;
|
|
|
|
struct rpc_xprt *xprt = req->rq_xprt;
|
2006-12-06 04:35:15 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:56 +07:00
|
|
|
struct xdr_buf *xdr = &req->rq_snd_buf;
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
bool zerocopy = true;
|
2007-08-06 22:56:42 +07:00
|
|
|
int status;
|
2014-09-25 01:08:00 +07:00
|
|
|
int sent;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2011-05-10 02:22:34 +07:00
|
|
|
xs_encode_stream_record_marker(&req->rq_snd_buf);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
xs_pktdump("packet data:",
|
|
|
|
req->rq_svec->iov_base,
|
|
|
|
req->rq_svec->iov_len);
|
SUNRPC: Fix a data corruption issue when retransmitting RPC calls
The following scenario can cause silent data corruption when doing
NFS writes. It has mainly been observed when doing database writes
using O_DIRECT.
1) The RPC client uses sendpage() to do zero-copy of the page data.
2) Due to networking issues, the reply from the server is delayed,
and so the RPC client times out.
3) The client issues a second sendpage of the page data as part of
an RPC call retransmission.
4) The reply to the first transmission arrives from the server
_before_ the client hardware has emptied the TCP socket send
buffer.
5) After processing the reply, the RPC state machine rules that
the call to be done, and triggers the completion callbacks.
6) The application notices the RPC call is done, and reuses the
pages to store something else (e.g. a new write).
7) The client NIC drains the TCP socket send buffer. Since the
page data has now changed, it reads a corrupted version of the
initial RPC call, and puts it on the wire.
This patch fixes the problem in the following manner:
The ordering guarantees of TCP ensure that when the server sends a
reply, then we know that the _first_ transmission has completed. Using
zero-copy in that situation is therefore safe.
If a time out occurs, we then send the retransmission using sendmsg()
(i.e. no zero-copy), We then know that the socket contains a full copy of
the data, and so it will retransmit a faithful reproduction even if the
RPC call completes, and the application reuses the O_DIRECT buffer in
the meantime.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Cc: stable@vger.kernel.org
2013-11-09 04:03:50 +07:00
|
|
|
/* Don't use zero copy if this is a resend. If the RPC call
|
|
|
|
* completes while the socket holds a reference to the pages,
|
|
|
|
* then we may end up resending corrupted data.
|
|
|
|
*/
|
|
|
|
if (task->tk_flags & RPC_TASK_SENT)
|
|
|
|
zerocopy = false;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
|
|
|
/* Continue transmitting the packet/record. We must be careful
|
|
|
|
* to cope with writespace callbacks arriving _after_ we have
|
2005-08-12 03:25:56 +07:00
|
|
|
* called sendmsg(). */
|
2005-08-12 03:25:23 +07:00
|
|
|
while (1) {
|
2014-09-25 01:08:00 +07:00
|
|
|
sent = 0;
|
|
|
|
status = xs_sendpages(transport->sock, NULL, 0, xdr,
|
|
|
|
req->rq_bytes_sent, zerocopy, &sent);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_tcp_send_request(%u) = %d\n",
|
2005-08-12 03:25:56 +07:00
|
|
|
xdr->len - req->rq_bytes_sent, status);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
/* If we've sent the entire packet, immediately
|
|
|
|
* reset the count of bytes sent. */
|
2014-09-25 01:08:00 +07:00
|
|
|
req->rq_bytes_sent += sent;
|
|
|
|
req->rq_xmit_bytes_sent += sent;
|
2005-08-12 03:25:56 +07:00
|
|
|
if (likely(req->rq_bytes_sent >= req->rq_slen)) {
|
|
|
|
req->rq_bytes_sent = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2015-07-11 22:48:52 +07:00
|
|
|
if (status < 0)
|
|
|
|
break;
|
|
|
|
if (sent == 0) {
|
|
|
|
status = -EAGAIN;
|
|
|
|
break;
|
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2015-07-27 07:55:35 +07:00
|
|
|
if (status == -EAGAIN && sk_stream_is_writeable(transport->inet))
|
|
|
|
status = -ENOBUFS;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
switch (status) {
|
2009-03-12 01:06:41 +07:00
|
|
|
case -ENOTSOCK:
|
|
|
|
status = -ENOTCONN;
|
|
|
|
/* Should we call xs_close() here? */
|
|
|
|
break;
|
2005-08-12 03:25:56 +07:00
|
|
|
case -EAGAIN:
|
2009-03-12 01:38:01 +07:00
|
|
|
status = xs_nospace(task);
|
2005-08-12 03:25:56 +07:00
|
|
|
break;
|
2009-03-12 01:37:59 +07:00
|
|
|
default:
|
|
|
|
dprintk("RPC: sendmsg returned unrecognized error %d\n",
|
|
|
|
-status);
|
2005-08-12 03:25:56 +07:00
|
|
|
case -ECONNRESET:
|
2008-10-29 02:21:39 +07:00
|
|
|
case -ECONNREFUSED:
|
2005-08-12 03:25:56 +07:00
|
|
|
case -ENOTCONN:
|
2015-02-09 09:44:04 +07:00
|
|
|
case -EADDRINUSE:
|
2015-07-03 20:32:23 +07:00
|
|
|
case -ENOBUFS:
|
2012-10-23 22:40:02 +07:00
|
|
|
case -EPIPE:
|
2008-04-18 05:52:19 +07:00
|
|
|
clear_bit(SOCK_ASYNC_NOSPACE, &transport->sock->flags);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2010-03-08 13:49:01 +07:00
|
|
|
|
2005-08-12 03:25:23 +07:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2006-07-28 04:22:50 +07:00
|
|
|
/**
|
|
|
|
* xs_tcp_release_xprt - clean up after a tcp transmission
|
|
|
|
* @xprt: transport
|
|
|
|
* @task: rpc task
|
|
|
|
*
|
|
|
|
* This cleans up if an error causes us to abort the transmission of a request.
|
|
|
|
* In this case, the socket may need to be reset in order to avoid confusing
|
|
|
|
* the server.
|
|
|
|
*/
|
|
|
|
static void xs_tcp_release_xprt(struct rpc_xprt *xprt, struct rpc_task *task)
|
|
|
|
{
|
|
|
|
struct rpc_rqst *req;
|
|
|
|
|
|
|
|
if (task != xprt->snd_task)
|
|
|
|
return;
|
|
|
|
if (task == NULL)
|
|
|
|
goto out_release;
|
|
|
|
req = task->tk_rqstp;
|
2011-07-18 03:01:03 +07:00
|
|
|
if (req == NULL)
|
|
|
|
goto out_release;
|
2006-07-28 04:22:50 +07:00
|
|
|
if (req->rq_bytes_sent == 0)
|
|
|
|
goto out_release;
|
|
|
|
if (req->rq_bytes_sent == req->rq_snd_buf.len)
|
|
|
|
goto out_release;
|
2013-01-08 21:10:21 +07:00
|
|
|
set_bit(XPRT_CLOSE_WAIT, &xprt->state);
|
2006-07-28 04:22:50 +07:00
|
|
|
out_release:
|
|
|
|
xprt_release_xprt(xprt, task);
|
|
|
|
}
|
|
|
|
|
2008-10-29 02:21:39 +07:00
|
|
|
static void xs_save_old_callbacks(struct sock_xprt *transport, struct sock *sk)
|
|
|
|
{
|
|
|
|
transport->old_data_ready = sk->sk_data_ready;
|
|
|
|
transport->old_state_change = sk->sk_state_change;
|
|
|
|
transport->old_write_space = sk->sk_write_space;
|
2014-01-01 01:22:59 +07:00
|
|
|
transport->old_error_report = sk->sk_error_report;
|
2008-10-29 02:21:39 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void xs_restore_old_callbacks(struct sock_xprt *transport, struct sock *sk)
|
|
|
|
{
|
|
|
|
sk->sk_data_ready = transport->old_data_ready;
|
|
|
|
sk->sk_state_change = transport->old_state_change;
|
|
|
|
sk->sk_write_space = transport->old_write_space;
|
2014-01-01 01:22:59 +07:00
|
|
|
sk->sk_error_report = transport->old_error_report;
|
|
|
|
}
|
|
|
|
|
2015-02-09 21:41:32 +07:00
|
|
|
static void xs_sock_reset_connection_flags(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
smp_mb__before_atomic();
|
|
|
|
clear_bit(XPRT_CLOSE_WAIT, &xprt->state);
|
|
|
|
clear_bit(XPRT_CLOSING, &xprt->state);
|
|
|
|
smp_mb__after_atomic();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xs_sock_mark_closed(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
xs_sock_reset_connection_flags(xprt);
|
|
|
|
/* Mark transport as closed and wake up all pending tasks */
|
|
|
|
xprt_disconnect_done(xprt);
|
|
|
|
}
|
|
|
|
|
2014-01-01 01:22:59 +07:00
|
|
|
/**
|
|
|
|
* xs_error_report - callback to handle TCP socket state errors
|
|
|
|
* @sk: socket
|
|
|
|
*
|
|
|
|
* Note: we don't call sock_error() since there may be a rpc_task
|
|
|
|
* using the socket, and so we don't want to clear sk->sk_err.
|
|
|
|
*/
|
|
|
|
static void xs_error_report(struct sock *sk)
|
|
|
|
{
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
|
|
|
if (!(xprt = xprt_from_sock(sk)))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
err = -sk->sk_err;
|
|
|
|
if (err == 0)
|
|
|
|
goto out;
|
2015-02-09 21:41:32 +07:00
|
|
|
/* Is this a reset event? */
|
|
|
|
if (sk->sk_state == TCP_CLOSE)
|
|
|
|
xs_sock_mark_closed(xprt);
|
2014-01-01 01:22:59 +07:00
|
|
|
dprintk("RPC: xs_error_report client %p, error=%d...\n",
|
|
|
|
xprt, -err);
|
2014-01-01 01:39:22 +07:00
|
|
|
trace_rpc_socket_error(xprt, sk->sk_socket, err);
|
2014-01-01 01:22:59 +07:00
|
|
|
xprt_wake_pending_tasks(xprt, err);
|
|
|
|
out:
|
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2008-10-29 02:21:39 +07:00
|
|
|
}
|
|
|
|
|
2009-03-12 01:10:21 +07:00
|
|
|
static void xs_reset_transport(struct sock_xprt *transport)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-06 04:35:15 +07:00
|
|
|
struct socket *sock = transport->sock;
|
|
|
|
struct sock *sk = transport->inet;
|
2015-02-09 06:35:25 +07:00
|
|
|
struct rpc_xprt *xprt = &transport->xprt;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2009-03-12 01:10:21 +07:00
|
|
|
if (sk == NULL)
|
|
|
|
return;
|
2005-08-12 03:25:26 +07:00
|
|
|
|
2015-06-04 03:14:27 +07:00
|
|
|
if (atomic_read(&transport->xprt.swapper))
|
|
|
|
sk_clear_memalloc(sk);
|
|
|
|
|
2015-08-30 09:11:21 +07:00
|
|
|
kernel_sock_shutdown(sock, SHUT_RDWR);
|
|
|
|
|
2005-08-12 03:25:23 +07:00
|
|
|
write_lock_bh(&sk->sk_callback_lock);
|
2006-12-06 04:35:15 +07:00
|
|
|
transport->inet = NULL;
|
|
|
|
transport->sock = NULL;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
sk->sk_user_data = NULL;
|
2008-10-29 02:21:39 +07:00
|
|
|
|
|
|
|
xs_restore_old_callbacks(transport, sk);
|
2015-08-30 03:36:30 +07:00
|
|
|
xprt_clear_connected(xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
write_unlock_bh(&sk->sk_callback_lock);
|
2015-02-09 06:35:25 +07:00
|
|
|
xs_sock_reset_connection_flags(xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2015-02-09 06:35:25 +07:00
|
|
|
trace_rpc_socket_close(xprt, sock);
|
2005-08-12 03:25:23 +07:00
|
|
|
sock_release(sock);
|
2009-03-12 01:10:21 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xs_close - close a socket
|
|
|
|
* @xprt: transport
|
|
|
|
*
|
|
|
|
* This is used when all requests are complete; ie, no DRC state remains
|
|
|
|
* on the server we want to save.
|
2009-04-22 04:18:20 +07:00
|
|
|
*
|
|
|
|
* The caller _must_ be holding XPRT_LOCKED in order to avoid issues with
|
|
|
|
* xs_reset_transport() zeroing the socket from underneath a writer.
|
2009-03-12 01:10:21 +07:00
|
|
|
*/
|
|
|
|
static void xs_close(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
dprintk("RPC: xs_close xprt %p\n", xprt);
|
|
|
|
|
|
|
|
xs_reset_transport(transport);
|
NFS/RPC: fix problems with reestablish_timeout and related code.
[[resending with correct cc: - "vfs.kernel.org" just isn't right!]]
xprt->reestablish_timeout is used to cause TCP connection attempts to
back off if the connection fails so as not to hammer the network,
but to still allow immediate connections when there is no reason to
believe there is a problem.
It is not used for the first connection (when transport->sock is NULL)
but only on reconnects.
It is currently set:
a/ to 0 when xs_tcp_state_change finds a state of TCP_FIN_WAIT1
on the assumption that the client has closed the connection
so the reconnect should be immediate when needed.
b/ to at least XS_TCP_INIT_REEST_TO when xs_tcp_state_change
detects TCP_CLOSING or TCP_CLOSE_WAIT on the assumption that the
server closed the connection so a small delay at least is
required.
c/ as above when xs_tcp_state_change detects TCP_SYN_SENT, so that
it is never 0 while a connection has been attempted, else
the doubling will produce 0 and there will be no backoff.
d/ to double is value (up to a limit) when delaying a connection,
thus providing exponential backoff and
e/ to XS_TCP_INIT_REEST_TO in xs_setup_tcp as simple initialisation.
So you can see it is highly dependant on xs_tcp_state_change being
called as expected. However experimental evidence shows that
xs_tcp_state_change does not see all state changes.
("rpcdebug -m rpc trans" can help show what actually happens).
Results show:
TCP_ESTABLISHED is reported when a connection is made. TCP_SYN_SENT
is never reported, so rule 'c' above is never effective.
When the server closes the connection, TCP_CLOSE_WAIT and
TCP_LAST_ACK *might* be reported, and TCP_CLOSE is always
reported. This rule 'b' above will sometimes be effective, but
not reliably.
When the client closes the connection, it used to result in
TCP_FIN_WAIT1, TCP_FIN_WAIT2, TCP_CLOSE. However since commit
f75e674 (SUNRPC: Fix the problem of EADDRNOTAVAIL syslog floods on
reconnect) we don't see *any* events on client-close. I think this
is because xs_restore_old_callbacks is called to disconnect
xs_tcp_state_change before the socket is closed.
In any case, rule 'a' no longer applies.
So all that is left are rule d, which successfully doubles the
timeout which is never rest, and rule e which initialises the timeout.
Even if the rules worked as expected, there would be a problem because
a successful connection does not reset the timeout, so a sequence
of events where the server closes the connection (e.g. during failover
testing) will cause longer and longer timeouts with no good reason.
This patch:
- sets reestablish_timeout to 0 in xs_close thus effecting rule 'a'
- sets it to 0 in xs_tcp_data_ready to ensure that a successful
connection resets the timeout
- sets it to at least XS_TCP_INIT_REEST_TO after it is doubled,
thus effecting rule c
I have not reimplemented rule b and the new version of rule c
seems sufficient.
I suspect other code in xs_tcp_data_ready needs to be revised as well.
For example I don't think connect_cookie is being incremented as often
as it should be.
Signed-off-by: NeilBrown <neilb@suse.de>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
2009-09-24 01:36:37 +07:00
|
|
|
xprt->reestablish_timeout = 0;
|
2009-03-12 01:10:21 +07:00
|
|
|
|
2007-11-07 06:44:20 +07:00
|
|
|
xprt_disconnect_done(xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:25 +07:00
|
|
|
static void xs_inject_disconnect(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
dprintk("RPC: injecting transport disconnect on xprt=%p\n",
|
|
|
|
xprt);
|
|
|
|
xprt_disconnect_done(xprt);
|
|
|
|
}
|
|
|
|
|
2014-03-24 10:07:22 +07:00
|
|
|
static void xs_xprt_free(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
xs_free_peer_addresses(xprt);
|
|
|
|
xprt_free(xprt);
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_destroy - prepare to shutdown a transport
|
|
|
|
* @xprt: doomed transport
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_destroy(struct rpc_xprt *xprt)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2015-09-17 21:42:27 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt,
|
|
|
|
struct sock_xprt, xprt);
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_destroy xprt %p\n", xprt);
|
2005-08-12 03:25:26 +07:00
|
|
|
|
2015-09-17 21:42:27 +07:00
|
|
|
cancel_delayed_work_sync(&transport->connect_worker);
|
2013-10-31 20:18:49 +07:00
|
|
|
xs_close(xprt);
|
2014-03-24 10:07:22 +07:00
|
|
|
xs_xprt_free(xprt);
|
2013-10-31 20:18:49 +07:00
|
|
|
module_put(THIS_MODULE);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static int xs_local_copy_to_xdr(struct xdr_buf *xdr, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct xdr_skb_reader desc = {
|
|
|
|
.skb = skb,
|
|
|
|
.offset = sizeof(rpc_fraghdr),
|
|
|
|
.count = skb->len - sizeof(rpc_fraghdr),
|
|
|
|
};
|
|
|
|
|
|
|
|
if (xdr_partial_copy_from_skb(xdr, 0, &desc, xdr_skb_read_bits) < 0)
|
|
|
|
return -1;
|
|
|
|
if (desc.count)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xs_local_data_ready - "data ready" callback for AF_LOCAL sockets
|
|
|
|
* @sk: socket with data to read
|
|
|
|
*
|
|
|
|
* Currently this assumes we can read the whole reply in a single gulp.
|
|
|
|
*/
|
2014-04-12 03:15:36 +07:00
|
|
|
static void xs_local_data_ready(struct sock *sk)
|
2011-05-10 02:22:44 +07:00
|
|
|
{
|
|
|
|
struct rpc_task *task;
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
struct rpc_rqst *rovr;
|
|
|
|
struct sk_buff *skb;
|
|
|
|
int err, repsize, copied;
|
|
|
|
u32 _xid;
|
|
|
|
__be32 *xp;
|
|
|
|
|
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
|
|
|
dprintk("RPC: %s...\n", __func__);
|
|
|
|
xprt = xprt_from_sock(sk);
|
|
|
|
if (xprt == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
skb = skb_recv_datagram(sk, 0, 1, &err);
|
|
|
|
if (skb == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
repsize = skb->len - sizeof(rpc_fraghdr);
|
|
|
|
if (repsize < 4) {
|
|
|
|
dprintk("RPC: impossible RPC reply size %d\n", repsize);
|
|
|
|
goto dropit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the XID from the skb... */
|
|
|
|
xp = skb_header_pointer(skb, sizeof(rpc_fraghdr), sizeof(_xid), &_xid);
|
|
|
|
if (xp == NULL)
|
|
|
|
goto dropit;
|
|
|
|
|
|
|
|
/* Look up and lock the request corresponding to the given XID */
|
|
|
|
spin_lock(&xprt->transport_lock);
|
|
|
|
rovr = xprt_lookup_rqst(xprt, *xp);
|
|
|
|
if (!rovr)
|
|
|
|
goto out_unlock;
|
|
|
|
task = rovr->rq_task;
|
|
|
|
|
|
|
|
copied = rovr->rq_private_buf.buflen;
|
|
|
|
if (copied > repsize)
|
|
|
|
copied = repsize;
|
|
|
|
|
|
|
|
if (xs_local_copy_to_xdr(&rovr->rq_private_buf, skb)) {
|
|
|
|
dprintk("RPC: sk_buff copy failed\n");
|
|
|
|
goto out_unlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
xprt_complete_rqst(task, copied);
|
|
|
|
|
|
|
|
out_unlock:
|
|
|
|
spin_unlock(&xprt->transport_lock);
|
|
|
|
dropit:
|
|
|
|
skb_free_datagram(sk, skb);
|
|
|
|
out:
|
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_udp_data_ready - "data ready" callback for UDP sockets
|
|
|
|
* @sk: socket with data to read
|
|
|
|
*
|
2005-08-12 03:25:23 +07:00
|
|
|
*/
|
2014-04-12 03:15:36 +07:00
|
|
|
static void xs_udp_data_ready(struct sock *sk)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2005-08-12 03:25:26 +07:00
|
|
|
struct rpc_task *task;
|
|
|
|
struct rpc_xprt *xprt;
|
2005-08-12 03:25:23 +07:00
|
|
|
struct rpc_rqst *rovr;
|
2005-08-12 03:25:26 +07:00
|
|
|
struct sk_buff *skb;
|
2005-08-12 03:25:23 +07:00
|
|
|
int err, repsize, copied;
|
2006-09-27 12:29:38 +07:00
|
|
|
u32 _xid;
|
|
|
|
__be32 *xp;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2010-09-22 19:43:39 +07:00
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_udp_data_ready...\n");
|
2005-08-12 03:25:26 +07:00
|
|
|
if (!(xprt = xprt_from_sock(sk)))
|
2005-08-12 03:25:23 +07:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
if ((skb = skb_recv_datagram(sk, 0, 1, &err)) == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
repsize = skb->len - sizeof(struct udphdr);
|
|
|
|
if (repsize < 4) {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: impossible RPC reply size %d!\n", repsize);
|
2005-08-12 03:25:23 +07:00
|
|
|
goto dropit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the XID from the skb... */
|
|
|
|
xp = skb_header_pointer(skb, sizeof(struct udphdr),
|
|
|
|
sizeof(_xid), &_xid);
|
|
|
|
if (xp == NULL)
|
|
|
|
goto dropit;
|
|
|
|
|
|
|
|
/* Look up and lock the request corresponding to the given XID */
|
2005-08-12 03:25:32 +07:00
|
|
|
spin_lock(&xprt->transport_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
rovr = xprt_lookup_rqst(xprt, *xp);
|
|
|
|
if (!rovr)
|
|
|
|
goto out_unlock;
|
|
|
|
task = rovr->rq_task;
|
|
|
|
|
|
|
|
if ((copied = rovr->rq_private_buf.buflen) > repsize)
|
|
|
|
copied = repsize;
|
|
|
|
|
|
|
|
/* Suck it into the iovec, verify checksum if not done by hw. */
|
2007-12-12 02:30:32 +07:00
|
|
|
if (csum_partial_copy_to_xdr(&rovr->rq_private_buf, skb)) {
|
|
|
|
UDPX_INC_STATS_BH(sk, UDP_MIB_INERRORS);
|
2005-08-12 03:25:23 +07:00
|
|
|
goto out_unlock;
|
2007-12-12 02:30:32 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
UDPX_INC_STATS_BH(sk, UDP_MIB_INDATAGRAMS);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2013-01-08 21:48:15 +07:00
|
|
|
xprt_adjust_cwnd(xprt, task, copied);
|
2005-08-26 06:25:52 +07:00
|
|
|
xprt_complete_rqst(task, copied);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
|
|
|
out_unlock:
|
2005-08-12 03:25:32 +07:00
|
|
|
spin_unlock(&xprt->transport_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
dropit:
|
|
|
|
skb_free_datagram(sk, skb);
|
|
|
|
out:
|
2010-09-22 19:43:39 +07:00
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2012-09-13 03:49:15 +07:00
|
|
|
/*
|
|
|
|
* Helper function to force a TCP close if the server is sending
|
|
|
|
* junk and/or it has put us in CLOSE_WAIT
|
|
|
|
*/
|
|
|
|
static void xs_tcp_force_close(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
xprt_force_disconnect(xprt);
|
|
|
|
}
|
|
|
|
|
2006-12-06 04:35:44 +07:00
|
|
|
static inline void xs_tcp_read_fraghdr(struct rpc_xprt *xprt, struct xdr_skb_reader *desc)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-06 04:35:19 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
size_t len, used;
|
|
|
|
char *p;
|
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
p = ((char *) &transport->tcp_fraghdr) + transport->tcp_offset;
|
|
|
|
len = sizeof(transport->tcp_fraghdr) - transport->tcp_offset;
|
2006-12-06 04:35:41 +07:00
|
|
|
used = xdr_skb_read_bits(desc, p, len);
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset += used;
|
2005-08-12 03:25:23 +07:00
|
|
|
if (used != len)
|
|
|
|
return;
|
2005-08-26 06:25:49 +07:00
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_reclen = ntohl(transport->tcp_fraghdr);
|
|
|
|
if (transport->tcp_reclen & RPC_LAST_STREAM_FRAGMENT)
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags |= TCP_RCV_LAST_FRAG;
|
2005-08-12 03:25:23 +07:00
|
|
|
else
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_LAST_FRAG;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_reclen &= RPC_FRAGMENT_SIZE_MASK;
|
2005-08-26 06:25:49 +07:00
|
|
|
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_FRAGHDR;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset = 0;
|
2005-08-26 06:25:49 +07:00
|
|
|
|
2005-08-12 03:25:23 +07:00
|
|
|
/* Sanity check of the record length */
|
2009-04-01 20:22:53 +07:00
|
|
|
if (unlikely(transport->tcp_reclen < 8)) {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: invalid TCP record fragment length\n");
|
2012-09-13 03:49:15 +07:00
|
|
|
xs_tcp_force_close(xprt);
|
2005-08-12 03:25:26 +07:00
|
|
|
return;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: reading TCP record fragment of length %d\n",
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_reclen);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
static void xs_tcp_check_fraghdr(struct sock_xprt *transport)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-06 04:35:19 +07:00
|
|
|
if (transport->tcp_offset == transport->tcp_reclen) {
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_FRAGHDR;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset = 0;
|
2006-12-06 04:35:23 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_LAST_FRAG) {
|
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_DATA;
|
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_XID;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_copied = 0;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-12-06 04:35:44 +07:00
|
|
|
static inline void xs_tcp_read_xid(struct sock_xprt *transport, struct xdr_skb_reader *desc)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
size_t len, used;
|
|
|
|
char *p;
|
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
len = sizeof(transport->tcp_xid) - transport->tcp_offset;
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: reading XID (%Zu bytes)\n", len);
|
2006-12-06 04:35:19 +07:00
|
|
|
p = ((char *) &transport->tcp_xid) + transport->tcp_offset;
|
2006-12-06 04:35:41 +07:00
|
|
|
used = xdr_skb_read_bits(desc, p, len);
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset += used;
|
2005-08-12 03:25:23 +07:00
|
|
|
if (used != len)
|
|
|
|
return;
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_XID;
|
2009-04-01 20:22:54 +07:00
|
|
|
transport->tcp_flags |= TCP_RCV_READ_CALLDIR;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_copied = 4;
|
2009-04-01 20:22:53 +07:00
|
|
|
dprintk("RPC: reading %s XID %08x\n",
|
|
|
|
(transport->tcp_flags & TCP_RPC_REPLY) ? "reply for"
|
|
|
|
: "request with",
|
2006-12-06 04:35:19 +07:00
|
|
|
ntohl(transport->tcp_xid));
|
|
|
|
xs_tcp_check_fraghdr(transport);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2009-04-01 20:22:53 +07:00
|
|
|
static inline void xs_tcp_read_calldir(struct sock_xprt *transport,
|
|
|
|
struct xdr_skb_reader *desc)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2009-04-01 20:22:53 +07:00
|
|
|
size_t len, used;
|
|
|
|
u32 offset;
|
2010-06-17 00:57:32 +07:00
|
|
|
char *p;
|
2009-04-01 20:22:53 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We want transport->tcp_offset to be 8 at the end of this routine
|
|
|
|
* (4 bytes for the xid and 4 bytes for the call/reply flag).
|
|
|
|
* When this function is called for the first time,
|
|
|
|
* transport->tcp_offset is 4 (after having already read the xid).
|
|
|
|
*/
|
|
|
|
offset = transport->tcp_offset - sizeof(transport->tcp_xid);
|
2010-06-17 00:57:32 +07:00
|
|
|
len = sizeof(transport->tcp_calldir) - offset;
|
2009-04-01 20:22:53 +07:00
|
|
|
dprintk("RPC: reading CALL/REPLY flag (%Zu bytes)\n", len);
|
2010-06-17 00:57:32 +07:00
|
|
|
p = ((char *) &transport->tcp_calldir) + offset;
|
|
|
|
used = xdr_skb_read_bits(desc, p, len);
|
2009-04-01 20:22:53 +07:00
|
|
|
transport->tcp_offset += used;
|
|
|
|
if (used != len)
|
|
|
|
return;
|
2009-04-01 20:22:54 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_READ_CALLDIR;
|
|
|
|
/*
|
|
|
|
* We don't yet have the XDR buffer, so we will write the calldir
|
|
|
|
* out after we get the buffer from the 'struct rpc_rqst'
|
|
|
|
*/
|
2010-06-17 00:57:32 +07:00
|
|
|
switch (ntohl(transport->tcp_calldir)) {
|
|
|
|
case RPC_REPLY:
|
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_CALLDIR;
|
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_DATA;
|
2009-04-01 20:22:53 +07:00
|
|
|
transport->tcp_flags |= TCP_RPC_REPLY;
|
2010-06-17 00:57:32 +07:00
|
|
|
break;
|
|
|
|
case RPC_CALL:
|
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_CALLDIR;
|
|
|
|
transport->tcp_flags |= TCP_RCV_COPY_DATA;
|
2009-04-01 20:22:53 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RPC_REPLY;
|
2010-06-17 00:57:32 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dprintk("RPC: invalid request message type\n");
|
2012-09-13 03:49:15 +07:00
|
|
|
xs_tcp_force_close(&transport->xprt);
|
2010-06-17 00:57:32 +07:00
|
|
|
}
|
2009-04-01 20:22:53 +07:00
|
|
|
xs_tcp_check_fraghdr(transport);
|
|
|
|
}
|
|
|
|
|
2009-04-01 20:23:02 +07:00
|
|
|
static inline void xs_tcp_read_common(struct rpc_xprt *xprt,
|
|
|
|
struct xdr_skb_reader *desc,
|
|
|
|
struct rpc_rqst *req)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2009-04-01 20:23:02 +07:00
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
struct xdr_buf *rcvbuf;
|
|
|
|
size_t len;
|
|
|
|
ssize_t r;
|
|
|
|
|
|
|
|
rcvbuf = &req->rq_private_buf;
|
2009-04-01 20:22:54 +07:00
|
|
|
|
|
|
|
if (transport->tcp_flags & TCP_RCV_COPY_CALLDIR) {
|
|
|
|
/*
|
|
|
|
* Save the RPC direction in the XDR buffer
|
|
|
|
*/
|
|
|
|
memcpy(rcvbuf->head[0].iov_base + transport->tcp_copied,
|
2010-06-17 00:57:32 +07:00
|
|
|
&transport->tcp_calldir,
|
|
|
|
sizeof(transport->tcp_calldir));
|
|
|
|
transport->tcp_copied += sizeof(transport->tcp_calldir);
|
2009-04-01 20:22:54 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_CALLDIR;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
len = desc->count;
|
2006-12-06 04:35:19 +07:00
|
|
|
if (len > transport->tcp_reclen - transport->tcp_offset) {
|
2006-12-06 04:35:44 +07:00
|
|
|
struct xdr_skb_reader my_desc;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
len = transport->tcp_reclen - transport->tcp_offset;
|
2005-08-12 03:25:23 +07:00
|
|
|
memcpy(&my_desc, desc, sizeof(my_desc));
|
|
|
|
my_desc.count = len;
|
2006-12-06 04:35:19 +07:00
|
|
|
r = xdr_partial_copy_from_skb(rcvbuf, transport->tcp_copied,
|
2006-12-06 04:35:41 +07:00
|
|
|
&my_desc, xdr_skb_read_bits);
|
2005-08-12 03:25:23 +07:00
|
|
|
desc->count -= r;
|
|
|
|
desc->offset += r;
|
|
|
|
} else
|
2006-12-06 04:35:19 +07:00
|
|
|
r = xdr_partial_copy_from_skb(rcvbuf, transport->tcp_copied,
|
2006-12-06 04:35:41 +07:00
|
|
|
desc, xdr_skb_read_bits);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
|
|
|
if (r > 0) {
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_copied += r;
|
|
|
|
transport->tcp_offset += r;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
if (r != len) {
|
|
|
|
/* Error when copying to the receive buffer,
|
|
|
|
* usually because we weren't able to allocate
|
|
|
|
* additional buffer pages. All we can do now
|
2006-12-06 04:35:23 +07:00
|
|
|
* is turn off TCP_RCV_COPY_DATA, so the request
|
2005-08-12 03:25:23 +07:00
|
|
|
* will not receive any additional updates,
|
|
|
|
* and time out.
|
|
|
|
* Any remaining data from this record will
|
|
|
|
* be discarded.
|
|
|
|
*/
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_DATA;
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: XID %08x truncated request\n",
|
2006-12-06 04:35:19 +07:00
|
|
|
ntohl(transport->tcp_xid));
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xprt = %p, tcp_copied = %lu, "
|
|
|
|
"tcp_offset = %u, tcp_reclen = %u\n",
|
|
|
|
xprt, transport->tcp_copied,
|
|
|
|
transport->tcp_offset, transport->tcp_reclen);
|
2009-04-01 20:23:02 +07:00
|
|
|
return;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: XID %08x read %Zd bytes\n",
|
2006-12-06 04:35:19 +07:00
|
|
|
ntohl(transport->tcp_xid), r);
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xprt = %p, tcp_copied = %lu, tcp_offset = %u, "
|
|
|
|
"tcp_reclen = %u\n", xprt, transport->tcp_copied,
|
|
|
|
transport->tcp_offset, transport->tcp_reclen);
|
2006-12-06 04:35:19 +07:00
|
|
|
|
|
|
|
if (transport->tcp_copied == req->rq_private_buf.buflen)
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_DATA;
|
2006-12-06 04:35:19 +07:00
|
|
|
else if (transport->tcp_offset == transport->tcp_reclen) {
|
2006-12-06 04:35:23 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_LAST_FRAG)
|
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_DATA;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2009-04-01 20:23:02 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Finds the request corresponding to the RPC xid and invokes the common
|
|
|
|
* tcp read code to read the data.
|
|
|
|
*/
|
|
|
|
static inline int xs_tcp_read_reply(struct rpc_xprt *xprt,
|
|
|
|
struct xdr_skb_reader *desc)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct rpc_rqst *req;
|
|
|
|
|
|
|
|
dprintk("RPC: read reply XID %08x\n", ntohl(transport->tcp_xid));
|
|
|
|
|
|
|
|
/* Find and lock the request corresponding to this xid */
|
|
|
|
spin_lock(&xprt->transport_lock);
|
|
|
|
req = xprt_lookup_rqst(xprt, transport->tcp_xid);
|
|
|
|
if (!req) {
|
|
|
|
dprintk("RPC: XID %08x request not found!\n",
|
|
|
|
ntohl(transport->tcp_xid));
|
|
|
|
spin_unlock(&xprt->transport_lock);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
xs_tcp_read_common(xprt, desc, req);
|
|
|
|
|
2006-12-06 04:35:23 +07:00
|
|
|
if (!(transport->tcp_flags & TCP_RCV_COPY_DATA))
|
2006-12-06 04:35:19 +07:00
|
|
|
xprt_complete_rqst(req->rq_task, transport->tcp_copied);
|
2009-04-01 20:23:02 +07:00
|
|
|
|
2005-08-12 03:25:32 +07:00
|
|
|
spin_unlock(&xprt->transport_lock);
|
2009-04-01 20:23:02 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-07-14 06:20:49 +07:00
|
|
|
#if defined(CONFIG_SUNRPC_BACKCHANNEL)
|
2009-04-01 20:23:02 +07:00
|
|
|
/*
|
|
|
|
* Obtains an rpc_rqst previously allocated and invokes the common
|
|
|
|
* tcp read code to read the data. The result is placed in the callback
|
|
|
|
* queue.
|
|
|
|
* If we're unable to obtain the rpc_rqst we schedule the closing of the
|
|
|
|
* connection and return -1.
|
|
|
|
*/
|
2014-02-10 23:18:39 +07:00
|
|
|
static int xs_tcp_read_callback(struct rpc_xprt *xprt,
|
2009-04-01 20:23:02 +07:00
|
|
|
struct xdr_skb_reader *desc)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct rpc_rqst *req;
|
|
|
|
|
2014-02-10 23:18:39 +07:00
|
|
|
/* Look up and lock the request corresponding to the given XID */
|
|
|
|
spin_lock(&xprt->transport_lock);
|
|
|
|
req = xprt_lookup_bc_request(xprt, transport->tcp_xid);
|
2009-04-01 20:23:02 +07:00
|
|
|
if (req == NULL) {
|
2014-02-10 23:18:39 +07:00
|
|
|
spin_unlock(&xprt->transport_lock);
|
2009-04-01 20:23:02 +07:00
|
|
|
printk(KERN_WARNING "Callback slot table overflowed\n");
|
|
|
|
xprt_force_disconnect(xprt);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
dprintk("RPC: read callback XID %08x\n", ntohl(req->rq_xid));
|
|
|
|
xs_tcp_read_common(xprt, desc, req);
|
|
|
|
|
2014-02-10 23:18:39 +07:00
|
|
|
if (!(transport->tcp_flags & TCP_RCV_COPY_DATA))
|
|
|
|
xprt_complete_bc_request(req, transport->tcp_copied);
|
|
|
|
spin_unlock(&xprt->transport_lock);
|
2009-04-01 20:23:02 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int _xs_tcp_read_data(struct rpc_xprt *xprt,
|
|
|
|
struct xdr_skb_reader *desc)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
return (transport->tcp_flags & TCP_RPC_REPLY) ?
|
|
|
|
xs_tcp_read_reply(xprt, desc) :
|
|
|
|
xs_tcp_read_callback(xprt, desc);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static inline int _xs_tcp_read_data(struct rpc_xprt *xprt,
|
|
|
|
struct xdr_skb_reader *desc)
|
|
|
|
{
|
|
|
|
return xs_tcp_read_reply(xprt, desc);
|
|
|
|
}
|
2011-07-14 06:20:49 +07:00
|
|
|
#endif /* CONFIG_SUNRPC_BACKCHANNEL */
|
2009-04-01 20:23:02 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Read data off the transport. This can be either an RPC_CALL or an
|
|
|
|
* RPC_REPLY. Relay the processing to helper functions.
|
|
|
|
*/
|
|
|
|
static void xs_tcp_read_data(struct rpc_xprt *xprt,
|
|
|
|
struct xdr_skb_reader *desc)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
if (_xs_tcp_read_data(xprt, desc) == 0)
|
|
|
|
xs_tcp_check_fraghdr(transport);
|
|
|
|
else {
|
|
|
|
/*
|
|
|
|
* The transport_lock protects the request handling.
|
|
|
|
* There's no need to hold it to update the tcp_flags.
|
|
|
|
*/
|
|
|
|
transport->tcp_flags &= ~TCP_RCV_COPY_DATA;
|
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2006-12-06 04:35:44 +07:00
|
|
|
static inline void xs_tcp_read_discard(struct sock_xprt *transport, struct xdr_skb_reader *desc)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
|
2006-12-06 04:35:19 +07:00
|
|
|
len = transport->tcp_reclen - transport->tcp_offset;
|
2005-08-12 03:25:23 +07:00
|
|
|
if (len > desc->count)
|
|
|
|
len = desc->count;
|
|
|
|
desc->count -= len;
|
|
|
|
desc->offset += len;
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset += len;
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: discarded %Zu bytes\n", len);
|
2006-12-06 04:35:19 +07:00
|
|
|
xs_tcp_check_fraghdr(transport);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
static int xs_tcp_data_recv(read_descriptor_t *rd_desc, struct sk_buff *skb, unsigned int offset, size_t len)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
struct rpc_xprt *xprt = rd_desc->arg.data;
|
2006-12-06 04:35:19 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2006-12-06 04:35:44 +07:00
|
|
|
struct xdr_skb_reader desc = {
|
2005-08-12 03:25:23 +07:00
|
|
|
.skb = skb,
|
|
|
|
.offset = offset,
|
|
|
|
.count = len,
|
2005-08-12 03:25:26 +07:00
|
|
|
};
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_tcp_data_recv started\n");
|
2005-08-12 03:25:23 +07:00
|
|
|
do {
|
2014-10-29 01:24:14 +07:00
|
|
|
trace_xs_tcp_data_recv(transport);
|
2005-08-12 03:25:23 +07:00
|
|
|
/* Read in a new fragment marker if necessary */
|
|
|
|
/* Can we ever really expect to get completely empty fragments? */
|
2006-12-06 04:35:23 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_COPY_FRAGHDR) {
|
2005-08-12 03:25:26 +07:00
|
|
|
xs_tcp_read_fraghdr(xprt, &desc);
|
2005-08-12 03:25:23 +07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Read in the xid if necessary */
|
2006-12-06 04:35:23 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_COPY_XID) {
|
2006-12-06 04:35:19 +07:00
|
|
|
xs_tcp_read_xid(transport, &desc);
|
2005-08-12 03:25:23 +07:00
|
|
|
continue;
|
|
|
|
}
|
2009-04-01 20:22:53 +07:00
|
|
|
/* Read in the call/reply flag */
|
2009-04-01 20:22:54 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_READ_CALLDIR) {
|
2009-04-01 20:22:53 +07:00
|
|
|
xs_tcp_read_calldir(transport, &desc);
|
|
|
|
continue;
|
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
/* Read in the request data */
|
2006-12-06 04:35:23 +07:00
|
|
|
if (transport->tcp_flags & TCP_RCV_COPY_DATA) {
|
2009-04-01 20:23:02 +07:00
|
|
|
xs_tcp_read_data(xprt, &desc);
|
2005-08-12 03:25:23 +07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Skip over any trailing bytes on short reads */
|
2006-12-06 04:35:19 +07:00
|
|
|
xs_tcp_read_discard(transport, &desc);
|
2005-08-12 03:25:23 +07:00
|
|
|
} while (desc.count);
|
2014-10-29 01:24:14 +07:00
|
|
|
trace_xs_tcp_data_recv(transport);
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_tcp_data_recv done\n");
|
2005-08-12 03:25:23 +07:00
|
|
|
return len - desc.count;
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_tcp_data_ready - "data ready" callback for TCP sockets
|
|
|
|
* @sk: socket with data to read
|
|
|
|
*
|
|
|
|
*/
|
2014-04-12 03:15:36 +07:00
|
|
|
static void xs_tcp_data_ready(struct sock *sk)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
read_descriptor_t rd_desc;
|
2008-02-26 12:40:51 +07:00
|
|
|
int read;
|
2014-10-29 01:24:13 +07:00
|
|
|
unsigned long total = 0;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_tcp_data_ready...\n");
|
|
|
|
|
2010-09-22 19:43:39 +07:00
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
2014-10-29 01:24:13 +07:00
|
|
|
if (!(xprt = xprt_from_sock(sk))) {
|
|
|
|
read = 0;
|
2005-08-12 03:25:23 +07:00
|
|
|
goto out;
|
2014-10-29 01:24:13 +07:00
|
|
|
}
|
NFS/RPC: fix problems with reestablish_timeout and related code.
[[resending with correct cc: - "vfs.kernel.org" just isn't right!]]
xprt->reestablish_timeout is used to cause TCP connection attempts to
back off if the connection fails so as not to hammer the network,
but to still allow immediate connections when there is no reason to
believe there is a problem.
It is not used for the first connection (when transport->sock is NULL)
but only on reconnects.
It is currently set:
a/ to 0 when xs_tcp_state_change finds a state of TCP_FIN_WAIT1
on the assumption that the client has closed the connection
so the reconnect should be immediate when needed.
b/ to at least XS_TCP_INIT_REEST_TO when xs_tcp_state_change
detects TCP_CLOSING or TCP_CLOSE_WAIT on the assumption that the
server closed the connection so a small delay at least is
required.
c/ as above when xs_tcp_state_change detects TCP_SYN_SENT, so that
it is never 0 while a connection has been attempted, else
the doubling will produce 0 and there will be no backoff.
d/ to double is value (up to a limit) when delaying a connection,
thus providing exponential backoff and
e/ to XS_TCP_INIT_REEST_TO in xs_setup_tcp as simple initialisation.
So you can see it is highly dependant on xs_tcp_state_change being
called as expected. However experimental evidence shows that
xs_tcp_state_change does not see all state changes.
("rpcdebug -m rpc trans" can help show what actually happens).
Results show:
TCP_ESTABLISHED is reported when a connection is made. TCP_SYN_SENT
is never reported, so rule 'c' above is never effective.
When the server closes the connection, TCP_CLOSE_WAIT and
TCP_LAST_ACK *might* be reported, and TCP_CLOSE is always
reported. This rule 'b' above will sometimes be effective, but
not reliably.
When the client closes the connection, it used to result in
TCP_FIN_WAIT1, TCP_FIN_WAIT2, TCP_CLOSE. However since commit
f75e674 (SUNRPC: Fix the problem of EADDRNOTAVAIL syslog floods on
reconnect) we don't see *any* events on client-close. I think this
is because xs_restore_old_callbacks is called to disconnect
xs_tcp_state_change before the socket is closed.
In any case, rule 'a' no longer applies.
So all that is left are rule d, which successfully doubles the
timeout which is never rest, and rule e which initialises the timeout.
Even if the rules worked as expected, there would be a problem because
a successful connection does not reset the timeout, so a sequence
of events where the server closes the connection (e.g. during failover
testing) will cause longer and longer timeouts with no good reason.
This patch:
- sets reestablish_timeout to 0 in xs_close thus effecting rule 'a'
- sets it to 0 in xs_tcp_data_ready to ensure that a successful
connection resets the timeout
- sets it to at least XS_TCP_INIT_REEST_TO after it is doubled,
thus effecting rule c
I have not reimplemented rule b and the new version of rule c
seems sufficient.
I suspect other code in xs_tcp_data_ready needs to be revised as well.
For example I don't think connect_cookie is being incremented as often
as it should be.
Signed-off-by: NeilBrown <neilb@suse.de>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
2009-09-24 01:36:37 +07:00
|
|
|
/* Any data means we had a useful conversation, so
|
|
|
|
* the we don't need to delay the next reconnect
|
|
|
|
*/
|
|
|
|
if (xprt->reestablish_timeout)
|
|
|
|
xprt->reestablish_timeout = 0;
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/* We use rd_desc to pass struct xprt to xs_tcp_data_recv */
|
2005-08-12 03:25:23 +07:00
|
|
|
rd_desc.arg.data = xprt;
|
2008-02-26 12:40:51 +07:00
|
|
|
do {
|
|
|
|
rd_desc.count = 65536;
|
|
|
|
read = tcp_read_sock(sk, &rd_desc, xs_tcp_data_recv);
|
2014-10-29 01:24:13 +07:00
|
|
|
if (read > 0)
|
|
|
|
total += read;
|
2008-02-26 12:40:51 +07:00
|
|
|
} while (read > 0);
|
2005-08-12 03:25:23 +07:00
|
|
|
out:
|
2014-10-29 01:24:13 +07:00
|
|
|
trace_xs_tcp_data_ready(xprt, read, total);
|
2010-09-22 19:43:39 +07:00
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_tcp_state_change - callback to handle TCP socket state changes
|
|
|
|
* @sk: socket whose state has changed
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_tcp_state_change(struct sock *sk)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2005-08-12 03:25:26 +07:00
|
|
|
struct rpc_xprt *xprt;
|
2015-09-17 10:43:17 +07:00
|
|
|
struct sock_xprt *transport;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2010-09-22 19:43:39 +07:00
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
if (!(xprt = xprt_from_sock(sk)))
|
|
|
|
goto out;
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_tcp_state_change client %p...\n", xprt);
|
2010-08-10 21:19:53 +07:00
|
|
|
dprintk("RPC: state %x conn %d dead %d zapped %d sk_shutdown %d\n",
|
2007-02-01 00:14:08 +07:00
|
|
|
sk->sk_state, xprt_connected(xprt),
|
|
|
|
sock_flag(sk, SOCK_DEAD),
|
2010-08-10 21:19:53 +07:00
|
|
|
sock_flag(sk, SOCK_ZAPPED),
|
|
|
|
sk->sk_shutdown);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2015-09-17 10:43:17 +07:00
|
|
|
transport = container_of(xprt, struct sock_xprt, xprt);
|
2013-09-04 23:16:23 +07:00
|
|
|
trace_rpc_socket_state_change(xprt, sk->sk_socket);
|
2005-08-12 03:25:23 +07:00
|
|
|
switch (sk->sk_state) {
|
|
|
|
case TCP_ESTABLISHED:
|
2010-09-22 19:43:39 +07:00
|
|
|
spin_lock(&xprt->transport_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
if (!xprt_test_and_set_connected(xprt)) {
|
2006-12-06 04:35:19 +07:00
|
|
|
|
2005-08-12 03:25:23 +07:00
|
|
|
/* Reset TCP record info */
|
2006-12-06 04:35:19 +07:00
|
|
|
transport->tcp_offset = 0;
|
|
|
|
transport->tcp_reclen = 0;
|
|
|
|
transport->tcp_copied = 0;
|
2006-12-06 04:35:23 +07:00
|
|
|
transport->tcp_flags =
|
|
|
|
TCP_RCV_COPY_FRAGHDR | TCP_RCV_COPY_XID;
|
2013-09-26 21:18:04 +07:00
|
|
|
xprt->connect_cookie++;
|
2015-09-17 10:43:17 +07:00
|
|
|
clear_bit(XPRT_SOCK_CONNECTING, &transport->sock_state);
|
|
|
|
xprt_clear_connecting(xprt);
|
2006-12-06 04:35:19 +07:00
|
|
|
|
2009-03-12 01:38:00 +07:00
|
|
|
xprt_wake_pending_tasks(xprt, -EAGAIN);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2010-09-22 19:43:39 +07:00
|
|
|
spin_unlock(&xprt->transport_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
break;
|
2007-11-06 05:42:39 +07:00
|
|
|
case TCP_FIN_WAIT1:
|
|
|
|
/* The client initiated a shutdown of the socket */
|
2008-04-18 03:52:57 +07:00
|
|
|
xprt->connect_cookie++;
|
2008-01-02 06:42:12 +07:00
|
|
|
xprt->reestablish_timeout = 0;
|
2007-11-06 05:42:39 +07:00
|
|
|
set_bit(XPRT_CLOSING, &xprt->state);
|
2014-03-18 00:06:10 +07:00
|
|
|
smp_mb__before_atomic();
|
2007-11-06 05:42:39 +07:00
|
|
|
clear_bit(XPRT_CONNECTED, &xprt->state);
|
2008-01-01 04:19:17 +07:00
|
|
|
clear_bit(XPRT_CLOSE_WAIT, &xprt->state);
|
2014-03-18 00:06:10 +07:00
|
|
|
smp_mb__after_atomic();
|
2005-08-12 03:25:23 +07:00
|
|
|
break;
|
2006-01-03 15:55:55 +07:00
|
|
|
case TCP_CLOSE_WAIT:
|
2007-11-06 05:42:39 +07:00
|
|
|
/* The server initiated a shutdown of the socket */
|
2008-04-18 03:52:57 +07:00
|
|
|
xprt->connect_cookie++;
|
2012-10-23 22:35:47 +07:00
|
|
|
clear_bit(XPRT_CONNECTED, &xprt->state);
|
2012-09-13 03:49:15 +07:00
|
|
|
xs_tcp_force_close(xprt);
|
2008-01-02 06:42:12 +07:00
|
|
|
case TCP_CLOSING:
|
|
|
|
/*
|
|
|
|
* If the server closed down the connection, make sure that
|
|
|
|
* we back off before reconnecting
|
|
|
|
*/
|
|
|
|
if (xprt->reestablish_timeout < XS_TCP_INIT_REEST_TO)
|
|
|
|
xprt->reestablish_timeout = XS_TCP_INIT_REEST_TO;
|
2007-11-06 05:42:39 +07:00
|
|
|
break;
|
|
|
|
case TCP_LAST_ACK:
|
2009-03-12 01:37:58 +07:00
|
|
|
set_bit(XPRT_CLOSING, &xprt->state);
|
2014-03-18 00:06:10 +07:00
|
|
|
smp_mb__before_atomic();
|
2007-11-06 05:42:39 +07:00
|
|
|
clear_bit(XPRT_CONNECTED, &xprt->state);
|
2014-03-18 00:06:10 +07:00
|
|
|
smp_mb__after_atomic();
|
2007-11-06 05:42:39 +07:00
|
|
|
break;
|
|
|
|
case TCP_CLOSE:
|
2015-09-17 10:43:17 +07:00
|
|
|
if (test_and_clear_bit(XPRT_SOCK_CONNECTING,
|
|
|
|
&transport->sock_state))
|
|
|
|
xprt_clear_connecting(xprt);
|
2009-03-12 01:38:03 +07:00
|
|
|
xs_sock_mark_closed(xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
out:
|
2010-09-22 19:43:39 +07:00
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
net/sunrpc/xprtsock.c: some common code found
$ diff-funcs xs_udp_write_space net/sunrpc/xprtsock.c
net/sunrpc/xprtsock.c xs_tcp_write_space
--- net/sunrpc/xprtsock.c:xs_udp_write_space()
+++ net/sunrpc/xprtsock.c:xs_tcp_write_space()
@@ -1,4 +1,4 @@
- * xs_udp_write_space - callback invoked when socket buffer space
+ * xs_tcp_write_space - callback invoked when socket buffer space
* becomes available
* @sk: socket whose state has changed
*
@@ -7,12 +7,12 @@
* progress, otherwise we'll waste resources thrashing kernel_sendmsg
* with a bunch of small requests.
*/
-static void xs_udp_write_space(struct sock *sk)
+static void xs_tcp_write_space(struct sock *sk)
{
read_lock(&sk->sk_callback_lock);
- /* from net/core/sock.c:sock_def_write_space */
- if (sock_writeable(sk)) {
+ /* from net/core/stream.c:sk_stream_write_space */
+ if (sk_stream_wspace(sk) >= sk_stream_min_wspace(sk)) {
struct socket *sock;
struct rpc_xprt *xprt;
$ codiff net/sunrpc/xprtsock.o net/sunrpc/xprtsock.o.new
net/sunrpc/xprtsock.c:
xs_tcp_write_space | -163
xs_udp_write_space | -163
2 functions changed, 326 bytes removed
net/sunrpc/xprtsock.c:
xs_write_space | +179
1 function changed, 179 bytes added
net/sunrpc/xprtsock.o.new:
3 functions changed, 179 bytes added, 326 bytes removed, diff: -147
Signed-off-by: Ilpo Järvinen <ilpo.jarvinen@helsinki.fi>
Signed-off-by: David S. Miller <davem@davemloft.net>
2009-02-07 14:48:33 +07:00
|
|
|
static void xs_write_space(struct sock *sk)
|
|
|
|
{
|
|
|
|
struct socket *sock;
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
|
|
|
|
if (unlikely(!(sock = sk->sk_socket)))
|
|
|
|
return;
|
|
|
|
clear_bit(SOCK_NOSPACE, &sock->flags);
|
|
|
|
|
|
|
|
if (unlikely(!(xprt = xprt_from_sock(sk))))
|
|
|
|
return;
|
|
|
|
if (test_and_clear_bit(SOCK_ASYNC_NOSPACE, &sock->flags) == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
xprt_write_space(xprt);
|
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
2005-08-12 03:25:50 +07:00
|
|
|
* xs_udp_write_space - callback invoked when socket buffer space
|
|
|
|
* becomes available
|
2005-08-12 03:25:26 +07:00
|
|
|
* @sk: socket whose state has changed
|
|
|
|
*
|
2005-08-12 03:25:23 +07:00
|
|
|
* Called when more output buffer space is available for this socket.
|
|
|
|
* We try not to wake our writers until they can make "significant"
|
2005-08-12 03:25:50 +07:00
|
|
|
* progress, otherwise we'll waste resources thrashing kernel_sendmsg
|
2005-08-12 03:25:23 +07:00
|
|
|
* with a bunch of small requests.
|
|
|
|
*/
|
2005-08-12 03:25:50 +07:00
|
|
|
static void xs_udp_write_space(struct sock *sk)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2010-09-22 19:43:39 +07:00
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:50 +07:00
|
|
|
/* from net/core/sock.c:sock_def_write_space */
|
net/sunrpc/xprtsock.c: some common code found
$ diff-funcs xs_udp_write_space net/sunrpc/xprtsock.c
net/sunrpc/xprtsock.c xs_tcp_write_space
--- net/sunrpc/xprtsock.c:xs_udp_write_space()
+++ net/sunrpc/xprtsock.c:xs_tcp_write_space()
@@ -1,4 +1,4 @@
- * xs_udp_write_space - callback invoked when socket buffer space
+ * xs_tcp_write_space - callback invoked when socket buffer space
* becomes available
* @sk: socket whose state has changed
*
@@ -7,12 +7,12 @@
* progress, otherwise we'll waste resources thrashing kernel_sendmsg
* with a bunch of small requests.
*/
-static void xs_udp_write_space(struct sock *sk)
+static void xs_tcp_write_space(struct sock *sk)
{
read_lock(&sk->sk_callback_lock);
- /* from net/core/sock.c:sock_def_write_space */
- if (sock_writeable(sk)) {
+ /* from net/core/stream.c:sk_stream_write_space */
+ if (sk_stream_wspace(sk) >= sk_stream_min_wspace(sk)) {
struct socket *sock;
struct rpc_xprt *xprt;
$ codiff net/sunrpc/xprtsock.o net/sunrpc/xprtsock.o.new
net/sunrpc/xprtsock.c:
xs_tcp_write_space | -163
xs_udp_write_space | -163
2 functions changed, 326 bytes removed
net/sunrpc/xprtsock.c:
xs_write_space | +179
1 function changed, 179 bytes added
net/sunrpc/xprtsock.o.new:
3 functions changed, 179 bytes added, 326 bytes removed, diff: -147
Signed-off-by: Ilpo Järvinen <ilpo.jarvinen@helsinki.fi>
Signed-off-by: David S. Miller <davem@davemloft.net>
2009-02-07 14:48:33 +07:00
|
|
|
if (sock_writeable(sk))
|
|
|
|
xs_write_space(sk);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2010-09-22 19:43:39 +07:00
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:50 +07:00
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:50 +07:00
|
|
|
/**
|
|
|
|
* xs_tcp_write_space - callback invoked when socket buffer space
|
|
|
|
* becomes available
|
|
|
|
* @sk: socket whose state has changed
|
|
|
|
*
|
|
|
|
* Called when more output buffer space is available for this socket.
|
|
|
|
* We try not to wake our writers until they can make "significant"
|
|
|
|
* progress, otherwise we'll waste resources thrashing kernel_sendmsg
|
|
|
|
* with a bunch of small requests.
|
|
|
|
*/
|
|
|
|
static void xs_tcp_write_space(struct sock *sk)
|
|
|
|
{
|
2010-09-22 19:43:39 +07:00
|
|
|
read_lock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:50 +07:00
|
|
|
|
|
|
|
/* from net/core/stream.c:sk_stream_write_space */
|
2013-07-23 10:26:31 +07:00
|
|
|
if (sk_stream_is_writeable(sk))
|
net/sunrpc/xprtsock.c: some common code found
$ diff-funcs xs_udp_write_space net/sunrpc/xprtsock.c
net/sunrpc/xprtsock.c xs_tcp_write_space
--- net/sunrpc/xprtsock.c:xs_udp_write_space()
+++ net/sunrpc/xprtsock.c:xs_tcp_write_space()
@@ -1,4 +1,4 @@
- * xs_udp_write_space - callback invoked when socket buffer space
+ * xs_tcp_write_space - callback invoked when socket buffer space
* becomes available
* @sk: socket whose state has changed
*
@@ -7,12 +7,12 @@
* progress, otherwise we'll waste resources thrashing kernel_sendmsg
* with a bunch of small requests.
*/
-static void xs_udp_write_space(struct sock *sk)
+static void xs_tcp_write_space(struct sock *sk)
{
read_lock(&sk->sk_callback_lock);
- /* from net/core/sock.c:sock_def_write_space */
- if (sock_writeable(sk)) {
+ /* from net/core/stream.c:sk_stream_write_space */
+ if (sk_stream_wspace(sk) >= sk_stream_min_wspace(sk)) {
struct socket *sock;
struct rpc_xprt *xprt;
$ codiff net/sunrpc/xprtsock.o net/sunrpc/xprtsock.o.new
net/sunrpc/xprtsock.c:
xs_tcp_write_space | -163
xs_udp_write_space | -163
2 functions changed, 326 bytes removed
net/sunrpc/xprtsock.c:
xs_write_space | +179
1 function changed, 179 bytes added
net/sunrpc/xprtsock.o.new:
3 functions changed, 179 bytes added, 326 bytes removed, diff: -147
Signed-off-by: Ilpo Järvinen <ilpo.jarvinen@helsinki.fi>
Signed-off-by: David S. Miller <davem@davemloft.net>
2009-02-07 14:48:33 +07:00
|
|
|
xs_write_space(sk);
|
2008-04-18 05:52:19 +07:00
|
|
|
|
2010-09-22 19:43:39 +07:00
|
|
|
read_unlock_bh(&sk->sk_callback_lock);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2005-08-26 06:25:56 +07:00
|
|
|
static void xs_udp_do_set_buffer_size(struct rpc_xprt *xprt)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-06 04:35:15 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct sock *sk = transport->inet;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-12-06 04:35:30 +07:00
|
|
|
if (transport->rcvsize) {
|
2005-08-12 03:25:23 +07:00
|
|
|
sk->sk_userlocks |= SOCK_RCVBUF_LOCK;
|
2006-12-06 04:35:30 +07:00
|
|
|
sk->sk_rcvbuf = transport->rcvsize * xprt->max_reqs * 2;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2006-12-06 04:35:30 +07:00
|
|
|
if (transport->sndsize) {
|
2005-08-12 03:25:23 +07:00
|
|
|
sk->sk_userlocks |= SOCK_SNDBUF_LOCK;
|
2006-12-06 04:35:30 +07:00
|
|
|
sk->sk_sndbuf = transport->sndsize * xprt->max_reqs * 2;
|
2005-08-12 03:25:23 +07:00
|
|
|
sk->sk_write_space(sk);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-08-26 06:25:49 +07:00
|
|
|
/**
|
2005-08-26 06:25:56 +07:00
|
|
|
* xs_udp_set_buffer_size - set send and receive limits
|
2005-08-26 06:25:49 +07:00
|
|
|
* @xprt: generic transport
|
2005-08-26 06:25:56 +07:00
|
|
|
* @sndsize: requested size of send buffer, in bytes
|
|
|
|
* @rcvsize: requested size of receive buffer, in bytes
|
2005-08-26 06:25:49 +07:00
|
|
|
*
|
2005-08-26 06:25:56 +07:00
|
|
|
* Set socket send and receive buffer size limits.
|
2005-08-26 06:25:49 +07:00
|
|
|
*/
|
2005-08-26 06:25:56 +07:00
|
|
|
static void xs_udp_set_buffer_size(struct rpc_xprt *xprt, size_t sndsize, size_t rcvsize)
|
2005-08-26 06:25:49 +07:00
|
|
|
{
|
2006-12-06 04:35:30 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
transport->sndsize = 0;
|
2005-08-26 06:25:56 +07:00
|
|
|
if (sndsize)
|
2006-12-06 04:35:30 +07:00
|
|
|
transport->sndsize = sndsize + 1024;
|
|
|
|
transport->rcvsize = 0;
|
2005-08-26 06:25:56 +07:00
|
|
|
if (rcvsize)
|
2006-12-06 04:35:30 +07:00
|
|
|
transport->rcvsize = rcvsize + 1024;
|
2005-08-26 06:25:56 +07:00
|
|
|
|
|
|
|
xs_udp_do_set_buffer_size(xprt);
|
2005-08-26 06:25:49 +07:00
|
|
|
}
|
|
|
|
|
2005-08-26 06:25:52 +07:00
|
|
|
/**
|
|
|
|
* xs_udp_timer - called when a retransmit timeout occurs on a UDP transport
|
|
|
|
* @task: task that timed out
|
|
|
|
*
|
|
|
|
* Adjust the congestion window after a retransmit timeout has occurred.
|
|
|
|
*/
|
2013-01-08 21:48:15 +07:00
|
|
|
static void xs_udp_timer(struct rpc_xprt *xprt, struct rpc_task *task)
|
2005-08-26 06:25:52 +07:00
|
|
|
{
|
2013-01-08 21:48:15 +07:00
|
|
|
xprt_adjust_cwnd(xprt, task, -ETIMEDOUT);
|
2005-08-26 06:25:52 +07:00
|
|
|
}
|
|
|
|
|
2006-05-25 12:40:49 +07:00
|
|
|
static unsigned short xs_get_random_port(void)
|
|
|
|
{
|
|
|
|
unsigned short range = xprt_max_resvport - xprt_min_resvport;
|
2014-01-11 19:15:59 +07:00
|
|
|
unsigned short rand = (unsigned short) prandom_u32() % range;
|
2006-05-25 12:40:49 +07:00
|
|
|
return rand + xprt_min_resvport;
|
|
|
|
}
|
|
|
|
|
2015-02-09 03:00:06 +07:00
|
|
|
/**
|
|
|
|
* xs_set_reuseaddr_port - set the socket's port and address reuse options
|
|
|
|
* @sock: socket
|
|
|
|
*
|
|
|
|
* Note that this function has to be called on all sockets that share the
|
|
|
|
* same port, and it must be called before binding.
|
|
|
|
*/
|
|
|
|
static void xs_sock_set_reuseport(struct socket *sock)
|
|
|
|
{
|
2015-02-10 05:20:14 +07:00
|
|
|
int opt = 1;
|
2015-02-09 03:00:06 +07:00
|
|
|
|
2015-02-10 05:20:14 +07:00
|
|
|
kernel_setsockopt(sock, SOL_SOCKET, SO_REUSEPORT,
|
|
|
|
(char *)&opt, sizeof(opt));
|
2015-02-09 03:00:06 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned short xs_sock_getport(struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sockaddr_storage buf;
|
|
|
|
int buflen;
|
|
|
|
unsigned short port = 0;
|
|
|
|
|
|
|
|
if (kernel_getsockname(sock, (struct sockaddr *)&buf, &buflen) < 0)
|
|
|
|
goto out;
|
|
|
|
switch (buf.ss_family) {
|
|
|
|
case AF_INET6:
|
|
|
|
port = ntohs(((struct sockaddr_in6 *)&buf)->sin6_port);
|
|
|
|
break;
|
|
|
|
case AF_INET:
|
|
|
|
port = ntohs(((struct sockaddr_in *)&buf)->sin_port);
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2006-01-03 15:55:51 +07:00
|
|
|
/**
|
|
|
|
* xs_set_port - reset the port number in the remote endpoint address
|
|
|
|
* @xprt: generic transport
|
|
|
|
* @port: new port number
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_set_port(struct rpc_xprt *xprt, unsigned short port)
|
|
|
|
{
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: setting port for xprt %p to %u\n", xprt, port);
|
2006-08-23 07:06:19 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
rpc_set_port(xs_addr(xprt), port);
|
|
|
|
xs_update_peer_port(xprt);
|
2006-01-03 15:55:51 +07:00
|
|
|
}
|
|
|
|
|
2015-02-09 03:00:06 +07:00
|
|
|
static void xs_set_srcport(struct sock_xprt *transport, struct socket *sock)
|
|
|
|
{
|
|
|
|
if (transport->srcport == 0)
|
|
|
|
transport->srcport = xs_sock_getport(sock);
|
|
|
|
}
|
|
|
|
|
2010-10-04 19:51:23 +07:00
|
|
|
static unsigned short xs_get_srcport(struct sock_xprt *transport)
|
2007-11-06 05:40:58 +07:00
|
|
|
{
|
2009-08-10 02:09:46 +07:00
|
|
|
unsigned short port = transport->srcport;
|
2007-11-06 05:40:58 +07:00
|
|
|
|
|
|
|
if (port == 0 && transport->xprt.resvport)
|
|
|
|
port = xs_get_random_port();
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2010-10-04 19:51:56 +07:00
|
|
|
static unsigned short xs_next_srcport(struct sock_xprt *transport, unsigned short port)
|
2007-11-06 05:40:58 +07:00
|
|
|
{
|
2009-08-10 02:09:46 +07:00
|
|
|
if (transport->srcport != 0)
|
|
|
|
transport->srcport = 0;
|
2007-11-06 05:40:58 +07:00
|
|
|
if (!transport->xprt.resvport)
|
|
|
|
return 0;
|
|
|
|
if (port <= xprt_min_resvport || port > xprt_max_resvport)
|
|
|
|
return xprt_max_resvport;
|
|
|
|
return --port;
|
|
|
|
}
|
2010-10-05 18:53:08 +07:00
|
|
|
static int xs_bind(struct sock_xprt *transport, struct socket *sock)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2010-10-05 18:53:08 +07:00
|
|
|
struct sockaddr_storage myaddr;
|
2007-11-06 05:40:58 +07:00
|
|
|
int err, nloop = 0;
|
2010-10-04 19:51:23 +07:00
|
|
|
unsigned short port = xs_get_srcport(transport);
|
2007-11-06 05:40:58 +07:00
|
|
|
unsigned short last;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
rpc: xs_bind - do not bind when requesting a random ephemeral port
When attempting to establish a local ephemeral endpoint for a TCP or UDP
socket, do not explicitly call bind, instead let it happen implicilty when the
socket is first used.
The main motivating factor for this change is when TCP runs out of unique
ephemeral ports (i.e. cannot find any ephemeral ports which are not a part of
*any* TCP connection). In this situation if you explicitly call bind, then the
call will fail with EADDRINUSE. However, if you allow the allocation of an
ephemeral port to happen implicitly as part of connect (or other functions),
then ephemeral ports can be reused, so long as the combination of (local_ip,
local_port, remote_ip, remote_port) is unique for TCP sockets on the system.
This doesn't matter for UDP sockets, but it seemed easiest to treat TCP and UDP
sockets the same.
This can allow mount.nfs(8) to continue to function successfully, even in the
face of misbehaving applications which are creating a large number of TCP
connections.
Signed-off-by: Chris Perl <chris.perl@gmail.com>
Signed-off-by: Trond Myklebust <trond.myklebust@primarydata.com>
2014-09-06 02:40:21 +07:00
|
|
|
/*
|
|
|
|
* If we are asking for any ephemeral port (i.e. port == 0 &&
|
|
|
|
* transport->xprt.resvport == 0), don't bind. Let the local
|
|
|
|
* port selection happen implicitly when the socket is used
|
|
|
|
* (for example at connect time).
|
|
|
|
*
|
|
|
|
* This ensures that we can continue to establish TCP
|
|
|
|
* connections even when all local ephemeral ports are already
|
|
|
|
* a part of some TCP connection. This makes no difference
|
|
|
|
* for UDP sockets, but also doens't harm them.
|
|
|
|
*
|
|
|
|
* If we're asking for any reserved port (i.e. port == 0 &&
|
|
|
|
* transport->xprt.resvport == 1) xs_get_srcport above will
|
|
|
|
* ensure that port is non-zero and we will bind as needed.
|
|
|
|
*/
|
|
|
|
if (port == 0)
|
|
|
|
return 0;
|
|
|
|
|
2010-10-05 18:53:08 +07:00
|
|
|
memcpy(&myaddr, &transport->srcaddr, transport->xprt.addrlen);
|
2005-08-12 03:25:23 +07:00
|
|
|
do {
|
2010-10-05 18:53:08 +07:00
|
|
|
rpc_set_port((struct sockaddr *)&myaddr, port);
|
|
|
|
err = kernel_bind(sock, (struct sockaddr *)&myaddr,
|
|
|
|
transport->xprt.addrlen);
|
2005-08-12 03:25:23 +07:00
|
|
|
if (err == 0) {
|
2009-08-10 02:09:46 +07:00
|
|
|
transport->srcport = port;
|
2007-07-10 03:23:35 +07:00
|
|
|
break;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2007-11-06 05:40:58 +07:00
|
|
|
last = port;
|
2010-10-04 19:51:56 +07:00
|
|
|
port = xs_next_srcport(transport, port);
|
2007-11-06 05:40:58 +07:00
|
|
|
if (port > last)
|
|
|
|
nloop++;
|
|
|
|
} while (err == -EADDRINUSE && nloop != 2);
|
2007-08-06 22:57:33 +07:00
|
|
|
|
2010-10-20 22:52:51 +07:00
|
|
|
if (myaddr.ss_family == AF_INET)
|
2010-10-05 18:53:08 +07:00
|
|
|
dprintk("RPC: %s %pI4:%u: %s (%d)\n", __func__,
|
|
|
|
&((struct sockaddr_in *)&myaddr)->sin_addr,
|
|
|
|
port, err ? "failed" : "ok", err);
|
|
|
|
else
|
|
|
|
dprintk("RPC: %s %pI6:%u: %s (%d)\n", __func__,
|
|
|
|
&((struct sockaddr_in6 *)&myaddr)->sin6_addr,
|
|
|
|
port, err ? "failed" : "ok", err);
|
2005-08-12 03:25:23 +07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
/*
|
|
|
|
* We don't support autobind on AF_LOCAL sockets
|
|
|
|
*/
|
|
|
|
static void xs_local_rpcbind(struct rpc_task *task)
|
|
|
|
{
|
2013-01-08 21:31:13 +07:00
|
|
|
rcu_read_lock();
|
|
|
|
xprt_set_bound(rcu_dereference(task->tk_client->cl_xprt));
|
|
|
|
rcu_read_unlock();
|
2011-05-10 02:22:44 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void xs_local_set_port(struct rpc_xprt *xprt, unsigned short port)
|
|
|
|
{
|
|
|
|
}
|
2010-10-05 18:53:08 +07:00
|
|
|
|
2006-12-07 11:35:24 +07:00
|
|
|
#ifdef CONFIG_DEBUG_LOCK_ALLOC
|
|
|
|
static struct lock_class_key xs_key[2];
|
|
|
|
static struct lock_class_key xs_slock_key[2];
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static inline void xs_reclassify_socketu(struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
|
|
|
|
sock_lock_init_class_and_name(sk, "slock-AF_LOCAL-RPC",
|
|
|
|
&xs_slock_key[1], "sk_lock-AF_LOCAL-RPC", &xs_key[1]);
|
|
|
|
}
|
|
|
|
|
2007-08-06 22:58:04 +07:00
|
|
|
static inline void xs_reclassify_socket4(struct socket *sock)
|
2006-12-07 11:35:24 +07:00
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
2007-08-06 22:58:04 +07:00
|
|
|
|
|
|
|
sock_lock_init_class_and_name(sk, "slock-AF_INET-RPC",
|
|
|
|
&xs_slock_key[0], "sk_lock-AF_INET-RPC", &xs_key[0]);
|
|
|
|
}
|
2006-12-07 11:35:24 +07:00
|
|
|
|
2007-08-06 22:58:04 +07:00
|
|
|
static inline void xs_reclassify_socket6(struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
2006-12-07 11:35:24 +07:00
|
|
|
|
2007-08-06 22:58:04 +07:00
|
|
|
sock_lock_init_class_and_name(sk, "slock-AF_INET6-RPC",
|
|
|
|
&xs_slock_key[1], "sk_lock-AF_INET6-RPC", &xs_key[1]);
|
2006-12-07 11:35:24 +07:00
|
|
|
}
|
2010-10-04 19:56:38 +07:00
|
|
|
|
|
|
|
static inline void xs_reclassify_socket(int family, struct socket *sock)
|
|
|
|
{
|
2012-10-23 21:43:39 +07:00
|
|
|
WARN_ON_ONCE(sock_owned_by_user(sock->sk));
|
|
|
|
if (sock_owned_by_user(sock->sk))
|
|
|
|
return;
|
|
|
|
|
2010-10-20 22:52:51 +07:00
|
|
|
switch (family) {
|
2011-05-10 02:22:44 +07:00
|
|
|
case AF_LOCAL:
|
|
|
|
xs_reclassify_socketu(sock);
|
|
|
|
break;
|
2010-10-20 22:52:51 +07:00
|
|
|
case AF_INET:
|
2010-10-04 19:56:38 +07:00
|
|
|
xs_reclassify_socket4(sock);
|
2010-10-20 22:52:51 +07:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
2010-10-04 19:56:38 +07:00
|
|
|
xs_reclassify_socket6(sock);
|
2010-10-20 22:52:51 +07:00
|
|
|
break;
|
|
|
|
}
|
2010-10-04 19:56:38 +07:00
|
|
|
}
|
2006-12-07 11:35:24 +07:00
|
|
|
#else
|
2011-05-10 02:22:44 +07:00
|
|
|
static inline void xs_reclassify_socketu(struct socket *sock)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-08-06 22:58:04 +07:00
|
|
|
static inline void xs_reclassify_socket4(struct socket *sock)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xs_reclassify_socket6(struct socket *sock)
|
2006-12-07 11:35:24 +07:00
|
|
|
{
|
|
|
|
}
|
2010-10-04 19:56:38 +07:00
|
|
|
|
|
|
|
static inline void xs_reclassify_socket(int family, struct socket *sock)
|
|
|
|
{
|
|
|
|
}
|
2006-12-07 11:35:24 +07:00
|
|
|
#endif
|
|
|
|
|
2013-10-31 12:14:36 +07:00
|
|
|
static void xs_dummy_setup_socket(struct work_struct *work)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-10-04 19:56:38 +07:00
|
|
|
static struct socket *xs_create_sock(struct rpc_xprt *xprt,
|
2015-02-09 03:00:06 +07:00
|
|
|
struct sock_xprt *transport, int family, int type,
|
|
|
|
int protocol, bool reuseport)
|
2010-10-04 19:54:26 +07:00
|
|
|
{
|
|
|
|
struct socket *sock;
|
|
|
|
int err;
|
|
|
|
|
2010-10-04 19:56:38 +07:00
|
|
|
err = __sock_create(xprt->xprt_net, family, type, protocol, &sock, 1);
|
2010-10-04 19:54:26 +07:00
|
|
|
if (err < 0) {
|
|
|
|
dprintk("RPC: can't create %d transport socket (%d).\n",
|
|
|
|
protocol, -err);
|
|
|
|
goto out;
|
|
|
|
}
|
2010-10-04 19:56:38 +07:00
|
|
|
xs_reclassify_socket(family, sock);
|
2010-10-04 19:54:26 +07:00
|
|
|
|
2015-02-09 03:00:06 +07:00
|
|
|
if (reuseport)
|
|
|
|
xs_sock_set_reuseport(sock);
|
|
|
|
|
2011-02-23 04:54:34 +07:00
|
|
|
err = xs_bind(transport, sock);
|
|
|
|
if (err) {
|
2010-10-04 19:54:26 +07:00
|
|
|
sock_release(sock);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sock;
|
|
|
|
out:
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static int xs_local_finish_connecting(struct rpc_xprt *xprt,
|
|
|
|
struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt,
|
|
|
|
xprt);
|
|
|
|
|
|
|
|
if (!transport->inet) {
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
|
|
|
|
write_lock_bh(&sk->sk_callback_lock);
|
|
|
|
|
|
|
|
xs_save_old_callbacks(transport, sk);
|
|
|
|
|
|
|
|
sk->sk_user_data = xprt;
|
|
|
|
sk->sk_data_ready = xs_local_data_ready;
|
|
|
|
sk->sk_write_space = xs_udp_write_space;
|
2014-01-01 01:22:59 +07:00
|
|
|
sk->sk_error_report = xs_error_report;
|
2015-08-20 09:46:15 +07:00
|
|
|
sk->sk_allocation = GFP_NOIO;
|
2011-05-10 02:22:44 +07:00
|
|
|
|
|
|
|
xprt_clear_connected(xprt);
|
|
|
|
|
|
|
|
/* Reset to new socket */
|
|
|
|
transport->sock = sock;
|
|
|
|
transport->inet = sk;
|
|
|
|
|
|
|
|
write_unlock_bh(&sk->sk_callback_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Tell the socket layer to start connecting... */
|
|
|
|
xprt->stat.connect_count++;
|
|
|
|
xprt->stat.connect_start = jiffies;
|
|
|
|
return kernel_connect(sock, xs_addr(xprt), xprt->addrlen, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xs_local_setup_socket - create AF_LOCAL socket, connect to a local endpoint
|
|
|
|
* @transport: socket transport to connect
|
|
|
|
*/
|
2013-02-21 05:52:19 +07:00
|
|
|
static int xs_local_setup_socket(struct sock_xprt *transport)
|
2011-05-10 02:22:44 +07:00
|
|
|
{
|
|
|
|
struct rpc_xprt *xprt = &transport->xprt;
|
|
|
|
struct socket *sock;
|
|
|
|
int status = -EIO;
|
|
|
|
|
|
|
|
status = __sock_create(xprt->xprt_net, AF_LOCAL,
|
|
|
|
SOCK_STREAM, 0, &sock, 1);
|
|
|
|
if (status < 0) {
|
|
|
|
dprintk("RPC: can't create AF_LOCAL "
|
|
|
|
"transport socket (%d).\n", -status);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
xs_reclassify_socketu(sock);
|
|
|
|
|
|
|
|
dprintk("RPC: worker connecting xprt %p via AF_LOCAL to %s\n",
|
|
|
|
xprt, xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
|
|
|
|
status = xs_local_finish_connecting(xprt, sock);
|
2013-09-04 23:16:23 +07:00
|
|
|
trace_rpc_socket_connect(xprt, sock, status);
|
2011-05-10 02:22:44 +07:00
|
|
|
switch (status) {
|
|
|
|
case 0:
|
|
|
|
dprintk("RPC: xprt %p connected to %s\n",
|
|
|
|
xprt, xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
xprt_set_connected(xprt);
|
2014-07-01 00:42:19 +07:00
|
|
|
case -ENOBUFS:
|
2011-05-10 02:22:44 +07:00
|
|
|
break;
|
|
|
|
case -ENOENT:
|
|
|
|
dprintk("RPC: xprt %p: socket %s does not exist\n",
|
|
|
|
xprt, xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
break;
|
2012-12-16 05:02:29 +07:00
|
|
|
case -ECONNREFUSED:
|
|
|
|
dprintk("RPC: xprt %p: connection refused for %s\n",
|
|
|
|
xprt, xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
break;
|
2011-05-10 02:22:44 +07:00
|
|
|
default:
|
|
|
|
printk(KERN_ERR "%s: unhandled error (%d) connecting to %s\n",
|
|
|
|
__func__, -status,
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
xprt_clear_connecting(xprt);
|
|
|
|
xprt_wake_pending_tasks(xprt, status);
|
2013-02-21 05:52:19 +07:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2013-03-01 09:02:55 +07:00
|
|
|
static void xs_local_connect(struct rpc_xprt *xprt, struct rpc_task *task)
|
2013-02-21 05:52:19 +07:00
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (RPC_IS_ASYNC(task)) {
|
|
|
|
/*
|
|
|
|
* We want the AF_LOCAL connect to be resolved in the
|
|
|
|
* filesystem namespace of the process making the rpc
|
|
|
|
* call. Thus we connect synchronously.
|
|
|
|
*
|
|
|
|
* If we want to support asynchronous AF_LOCAL calls,
|
|
|
|
* we'll need to figure out how to pass a namespace to
|
|
|
|
* connect.
|
|
|
|
*/
|
|
|
|
rpc_exit(task, -ENOTCONN);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ret = xs_local_setup_socket(transport);
|
|
|
|
if (ret && !RPC_IS_SOFTCONN(task))
|
|
|
|
msleep_interruptible(15000);
|
2011-05-10 02:22:44 +07:00
|
|
|
}
|
|
|
|
|
2015-06-04 03:14:25 +07:00
|
|
|
#if IS_ENABLED(CONFIG_SUNRPC_SWAP)
|
2015-06-04 03:14:28 +07:00
|
|
|
/*
|
|
|
|
* Note that this should be called with XPRT_LOCKED held (or when we otherwise
|
|
|
|
* know that we have exclusive access to the socket), to guard against
|
|
|
|
* races with xs_reset_transport.
|
|
|
|
*/
|
2012-08-01 06:45:12 +07:00
|
|
|
static void xs_set_memalloc(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt,
|
|
|
|
xprt);
|
|
|
|
|
2015-06-04 03:14:28 +07:00
|
|
|
/*
|
|
|
|
* If there's no sock, then we have nothing to set. The
|
|
|
|
* reconnecting process will get it for us.
|
|
|
|
*/
|
|
|
|
if (!transport->inet)
|
|
|
|
return;
|
2015-06-04 03:14:26 +07:00
|
|
|
if (atomic_read(&xprt->swapper))
|
2012-08-01 06:45:12 +07:00
|
|
|
sk_set_memalloc(transport->inet);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-06-04 03:14:29 +07:00
|
|
|
* xs_enable_swap - Tag this transport as being used for swap.
|
2012-08-01 06:45:12 +07:00
|
|
|
* @xprt: transport to tag
|
|
|
|
*
|
2015-06-04 03:14:26 +07:00
|
|
|
* Take a reference to this transport on behalf of the rpc_clnt, and
|
|
|
|
* optionally mark it for swapping if it wasn't already.
|
2012-08-01 06:45:12 +07:00
|
|
|
*/
|
2015-06-04 03:14:29 +07:00
|
|
|
static int
|
|
|
|
xs_enable_swap(struct rpc_xprt *xprt)
|
2012-08-01 06:45:12 +07:00
|
|
|
{
|
2015-06-04 03:14:28 +07:00
|
|
|
struct sock_xprt *xs = container_of(xprt, struct sock_xprt, xprt);
|
2012-08-01 06:45:12 +07:00
|
|
|
|
2015-06-04 03:14:28 +07:00
|
|
|
if (atomic_inc_return(&xprt->swapper) != 1)
|
|
|
|
return 0;
|
|
|
|
if (wait_on_bit_lock(&xprt->state, XPRT_LOCKED, TASK_KILLABLE))
|
|
|
|
return -ERESTARTSYS;
|
|
|
|
if (xs->inet)
|
|
|
|
sk_set_memalloc(xs->inet);
|
|
|
|
xprt_release_xprt(xprt, NULL);
|
2015-06-04 03:14:26 +07:00
|
|
|
return 0;
|
|
|
|
}
|
2012-08-01 06:45:12 +07:00
|
|
|
|
2015-06-04 03:14:26 +07:00
|
|
|
/**
|
2015-06-04 03:14:29 +07:00
|
|
|
* xs_disable_swap - Untag this transport as being used for swap.
|
2015-06-04 03:14:26 +07:00
|
|
|
* @xprt: transport to tag
|
|
|
|
*
|
|
|
|
* Drop a "swapper" reference to this xprt on behalf of the rpc_clnt. If the
|
|
|
|
* swapper refcount goes to 0, untag the socket as a memalloc socket.
|
|
|
|
*/
|
2015-06-04 03:14:29 +07:00
|
|
|
static void
|
|
|
|
xs_disable_swap(struct rpc_xprt *xprt)
|
2015-06-04 03:14:26 +07:00
|
|
|
{
|
2015-06-04 03:14:28 +07:00
|
|
|
struct sock_xprt *xs = container_of(xprt, struct sock_xprt, xprt);
|
2015-06-04 03:14:26 +07:00
|
|
|
|
2015-06-04 03:14:28 +07:00
|
|
|
if (!atomic_dec_and_test(&xprt->swapper))
|
|
|
|
return;
|
|
|
|
if (wait_on_bit_lock(&xprt->state, XPRT_LOCKED, TASK_KILLABLE))
|
|
|
|
return;
|
|
|
|
if (xs->inet)
|
|
|
|
sk_clear_memalloc(xs->inet);
|
|
|
|
xprt_release_xprt(xprt, NULL);
|
2012-08-01 06:45:12 +07:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void xs_set_memalloc(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
}
|
2015-06-04 03:14:29 +07:00
|
|
|
|
|
|
|
static int
|
|
|
|
xs_enable_swap(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
xs_disable_swap(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
}
|
2012-08-01 06:45:12 +07:00
|
|
|
#endif
|
|
|
|
|
2007-08-06 22:57:38 +07:00
|
|
|
static void xs_udp_finish_connecting(struct rpc_xprt *xprt, struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
if (!transport->inet) {
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
|
|
|
|
write_lock_bh(&sk->sk_callback_lock);
|
|
|
|
|
2008-10-29 02:21:39 +07:00
|
|
|
xs_save_old_callbacks(transport, sk);
|
|
|
|
|
2007-08-06 22:57:38 +07:00
|
|
|
sk->sk_user_data = xprt;
|
|
|
|
sk->sk_data_ready = xs_udp_data_ready;
|
|
|
|
sk->sk_write_space = xs_udp_write_space;
|
2015-08-20 09:46:15 +07:00
|
|
|
sk->sk_allocation = GFP_NOIO;
|
2007-08-06 22:57:38 +07:00
|
|
|
|
|
|
|
xprt_set_connected(xprt);
|
|
|
|
|
|
|
|
/* Reset to new socket */
|
|
|
|
transport->sock = sock;
|
|
|
|
transport->inet = sk;
|
|
|
|
|
2012-08-01 06:45:12 +07:00
|
|
|
xs_set_memalloc(xprt);
|
|
|
|
|
2007-08-06 22:57:38 +07:00
|
|
|
write_unlock_bh(&sk->sk_callback_lock);
|
|
|
|
}
|
|
|
|
xs_udp_do_set_buffer_size(xprt);
|
|
|
|
}
|
|
|
|
|
2010-10-04 19:58:02 +07:00
|
|
|
static void xs_udp_setup_socket(struct work_struct *work)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-08 03:48:15 +07:00
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(work, struct sock_xprt, connect_worker.work);
|
2006-12-06 04:35:26 +07:00
|
|
|
struct rpc_xprt *xprt = &transport->xprt;
|
2006-12-06 04:35:15 +07:00
|
|
|
struct socket *sock = transport->sock;
|
2010-10-04 19:53:46 +07:00
|
|
|
int status = -EIO;
|
2005-08-12 03:25:26 +07:00
|
|
|
|
2010-10-04 19:58:02 +07:00
|
|
|
sock = xs_create_sock(xprt, transport,
|
2015-02-09 03:00:06 +07:00
|
|
|
xs_addr(xprt)->sa_family, SOCK_DGRAM,
|
|
|
|
IPPROTO_UDP, false);
|
2010-10-04 19:53:46 +07:00
|
|
|
if (IS_ERR(sock))
|
2005-08-12 03:25:53 +07:00
|
|
|
goto out;
|
2007-08-06 22:57:48 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
dprintk("RPC: worker connecting xprt %p via %s to "
|
|
|
|
"%s (port %s)\n", xprt,
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT]);
|
2007-08-06 22:57:48 +07:00
|
|
|
|
|
|
|
xs_udp_finish_connecting(xprt, sock);
|
2013-09-04 23:16:23 +07:00
|
|
|
trace_rpc_socket_connect(xprt, sock, 0);
|
2005-08-12 03:25:53 +07:00
|
|
|
status = 0;
|
|
|
|
out:
|
2015-02-09 06:19:25 +07:00
|
|
|
xprt_unlock_connect(xprt, transport);
|
2005-08-12 03:25:53 +07:00
|
|
|
xprt_clear_connecting(xprt);
|
2009-03-12 01:38:03 +07:00
|
|
|
xprt_wake_pending_tasks(xprt, status);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2015-06-20 03:17:57 +07:00
|
|
|
/**
|
|
|
|
* xs_tcp_shutdown - gracefully shut down a TCP socket
|
|
|
|
* @xprt: transport
|
|
|
|
*
|
|
|
|
* Initiates a graceful shutdown of the TCP socket by calling the
|
|
|
|
* equivalent of shutdown(SHUT_RDWR);
|
|
|
|
*/
|
|
|
|
static void xs_tcp_shutdown(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct socket *sock = transport->sock;
|
|
|
|
|
|
|
|
if (sock == NULL)
|
|
|
|
return;
|
|
|
|
if (xprt_connected(xprt)) {
|
|
|
|
kernel_sock_shutdown(sock, SHUT_RDWR);
|
|
|
|
trace_rpc_socket_shutdown(xprt, sock);
|
|
|
|
} else
|
|
|
|
xs_reset_transport(transport);
|
|
|
|
}
|
|
|
|
|
2007-08-06 22:57:38 +07:00
|
|
|
static int xs_tcp_finish_connecting(struct rpc_xprt *xprt, struct socket *sock)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2007-08-06 22:57:38 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2011-03-19 07:21:23 +07:00
|
|
|
int ret = -ENOTCONN;
|
2006-08-23 07:06:18 +07:00
|
|
|
|
2006-12-06 04:35:15 +07:00
|
|
|
if (!transport->inet) {
|
2005-08-12 03:25:53 +07:00
|
|
|
struct sock *sk = sock->sk;
|
2013-09-24 22:25:22 +07:00
|
|
|
unsigned int keepidle = xprt->timeout->to_initval / HZ;
|
|
|
|
unsigned int keepcnt = xprt->timeout->to_retries + 1;
|
|
|
|
unsigned int opt_on = 1;
|
2015-06-21 02:31:54 +07:00
|
|
|
unsigned int timeo;
|
2013-09-24 22:25:22 +07:00
|
|
|
|
|
|
|
/* TCP Keepalive options */
|
|
|
|
kernel_setsockopt(sock, SOL_SOCKET, SO_KEEPALIVE,
|
|
|
|
(char *)&opt_on, sizeof(opt_on));
|
|
|
|
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPIDLE,
|
|
|
|
(char *)&keepidle, sizeof(keepidle));
|
|
|
|
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPINTVL,
|
|
|
|
(char *)&keepidle, sizeof(keepidle));
|
|
|
|
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPCNT,
|
|
|
|
(char *)&keepcnt, sizeof(keepcnt));
|
2005-08-12 03:25:53 +07:00
|
|
|
|
2015-06-21 02:31:54 +07:00
|
|
|
/* TCP user timeout (see RFC5482) */
|
|
|
|
timeo = jiffies_to_msecs(xprt->timeout->to_initval) *
|
|
|
|
(xprt->timeout->to_retries + 1);
|
|
|
|
kernel_setsockopt(sock, SOL_TCP, TCP_USER_TIMEOUT,
|
|
|
|
(char *)&timeo, sizeof(timeo));
|
|
|
|
|
2005-08-12 03:25:53 +07:00
|
|
|
write_lock_bh(&sk->sk_callback_lock);
|
|
|
|
|
2008-10-29 02:21:39 +07:00
|
|
|
xs_save_old_callbacks(transport, sk);
|
|
|
|
|
2005-08-12 03:25:53 +07:00
|
|
|
sk->sk_user_data = xprt;
|
|
|
|
sk->sk_data_ready = xs_tcp_data_ready;
|
|
|
|
sk->sk_state_change = xs_tcp_state_change;
|
|
|
|
sk->sk_write_space = xs_tcp_write_space;
|
2014-01-01 01:22:59 +07:00
|
|
|
sk->sk_error_report = xs_error_report;
|
2015-08-20 09:46:15 +07:00
|
|
|
sk->sk_allocation = GFP_NOIO;
|
2005-08-26 06:25:55 +07:00
|
|
|
|
|
|
|
/* socket options */
|
|
|
|
sock_reset_flag(sk, SOCK_LINGER);
|
|
|
|
tcp_sk(sk)->nonagle |= TCP_NAGLE_OFF;
|
2005-08-12 03:25:53 +07:00
|
|
|
|
|
|
|
xprt_clear_connected(xprt);
|
|
|
|
|
|
|
|
/* Reset to new socket */
|
2006-12-06 04:35:15 +07:00
|
|
|
transport->sock = sock;
|
|
|
|
transport->inet = sk;
|
2005-08-12 03:25:53 +07:00
|
|
|
|
|
|
|
write_unlock_bh(&sk->sk_callback_lock);
|
|
|
|
}
|
|
|
|
|
2009-03-12 01:09:39 +07:00
|
|
|
if (!xprt_bound(xprt))
|
2011-03-19 07:21:23 +07:00
|
|
|
goto out;
|
2009-03-12 01:09:39 +07:00
|
|
|
|
2012-08-01 06:45:12 +07:00
|
|
|
xs_set_memalloc(xprt);
|
|
|
|
|
2005-08-12 03:25:53 +07:00
|
|
|
/* Tell the socket layer to start connecting... */
|
2006-03-21 01:44:16 +07:00
|
|
|
xprt->stat.connect_count++;
|
|
|
|
xprt->stat.connect_start = jiffies;
|
2015-09-17 10:43:17 +07:00
|
|
|
set_bit(XPRT_SOCK_CONNECTING, &transport->sock_state);
|
2011-03-19 07:21:23 +07:00
|
|
|
ret = kernel_connect(sock, xs_addr(xprt), xprt->addrlen, O_NONBLOCK);
|
|
|
|
switch (ret) {
|
|
|
|
case 0:
|
2015-02-09 03:00:06 +07:00
|
|
|
xs_set_srcport(transport, sock);
|
2011-03-19 07:21:23 +07:00
|
|
|
case -EINPROGRESS:
|
|
|
|
/* SYN_SENT! */
|
|
|
|
if (xprt->reestablish_timeout < XS_TCP_INIT_REEST_TO)
|
|
|
|
xprt->reestablish_timeout = XS_TCP_INIT_REEST_TO;
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return ret;
|
2007-08-06 22:57:38 +07:00
|
|
|
}
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
2009-03-12 01:38:04 +07:00
|
|
|
* xs_tcp_setup_socket - create a TCP socket and connect to a remote endpoint
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
|
|
|
* Invoked by a work queue tasklet.
|
2005-08-12 03:25:23 +07:00
|
|
|
*/
|
2010-10-04 19:57:40 +07:00
|
|
|
static void xs_tcp_setup_socket(struct work_struct *work)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2010-10-04 19:57:40 +07:00
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(work, struct sock_xprt, connect_worker.work);
|
2006-12-06 04:35:15 +07:00
|
|
|
struct socket *sock = transport->sock;
|
2010-10-04 19:52:25 +07:00
|
|
|
struct rpc_xprt *xprt = &transport->xprt;
|
2009-03-12 01:38:04 +07:00
|
|
|
int status = -EIO;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-12-06 04:35:15 +07:00
|
|
|
if (!sock) {
|
2010-10-04 19:57:40 +07:00
|
|
|
sock = xs_create_sock(xprt, transport,
|
2015-02-09 03:00:06 +07:00
|
|
|
xs_addr(xprt)->sa_family, SOCK_STREAM,
|
|
|
|
IPPROTO_TCP, true);
|
2009-03-12 01:38:04 +07:00
|
|
|
if (IS_ERR(sock)) {
|
|
|
|
status = PTR_ERR(sock);
|
2005-08-26 06:25:55 +07:00
|
|
|
goto out;
|
|
|
|
}
|
2009-03-12 01:38:03 +07:00
|
|
|
}
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
dprintk("RPC: worker connecting xprt %p via %s to "
|
|
|
|
"%s (port %s)\n", xprt,
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT]);
|
2006-08-23 07:06:18 +07:00
|
|
|
|
2007-08-06 22:57:38 +07:00
|
|
|
status = xs_tcp_finish_connecting(xprt, sock);
|
2013-09-04 23:16:23 +07:00
|
|
|
trace_rpc_socket_connect(xprt, sock, status);
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: %p connect status %d connected %d sock state %d\n",
|
|
|
|
xprt, -status, xprt_connected(xprt),
|
|
|
|
sock->sk->sk_state);
|
2009-03-12 01:38:00 +07:00
|
|
|
switch (status) {
|
2009-04-22 04:18:20 +07:00
|
|
|
default:
|
|
|
|
printk("%s: connect returned unhandled error %d\n",
|
|
|
|
__func__, status);
|
|
|
|
case -EADDRNOTAVAIL:
|
|
|
|
/* We're probably in TIME_WAIT. Get rid of existing socket,
|
|
|
|
* and retry
|
|
|
|
*/
|
2012-09-13 03:49:15 +07:00
|
|
|
xs_tcp_force_close(xprt);
|
2009-06-18 03:22:57 +07:00
|
|
|
break;
|
2009-03-12 01:38:00 +07:00
|
|
|
case 0:
|
|
|
|
case -EINPROGRESS:
|
|
|
|
case -EALREADY:
|
2015-02-09 06:19:25 +07:00
|
|
|
xprt_unlock_connect(xprt, transport);
|
2009-03-12 01:38:03 +07:00
|
|
|
return;
|
2010-03-03 01:06:21 +07:00
|
|
|
case -EINVAL:
|
|
|
|
/* Happens, for instance, if the user specified a link
|
|
|
|
* local IPv6 address without a scope-id.
|
|
|
|
*/
|
2013-03-05 05:29:33 +07:00
|
|
|
case -ECONNREFUSED:
|
|
|
|
case -ECONNRESET:
|
|
|
|
case -ENETUNREACH:
|
2015-02-09 09:44:04 +07:00
|
|
|
case -EADDRINUSE:
|
2014-07-01 00:42:19 +07:00
|
|
|
case -ENOBUFS:
|
2013-03-05 05:29:33 +07:00
|
|
|
/* retry with existing socket, after a delay */
|
2015-02-09 03:34:28 +07:00
|
|
|
xs_tcp_force_close(xprt);
|
2010-03-03 01:06:21 +07:00
|
|
|
goto out;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2009-03-12 01:38:00 +07:00
|
|
|
status = -EAGAIN;
|
2005-08-12 03:25:23 +07:00
|
|
|
out:
|
2015-02-09 06:19:25 +07:00
|
|
|
xprt_unlock_connect(xprt, transport);
|
2007-08-06 22:57:48 +07:00
|
|
|
xprt_clear_connecting(xprt);
|
2009-03-12 01:38:03 +07:00
|
|
|
xprt_wake_pending_tasks(xprt, status);
|
2007-08-06 22:57:48 +07:00
|
|
|
}
|
2005-08-12 03:25:53 +07:00
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_connect - connect a socket to a remote endpoint
|
2013-01-08 21:26:49 +07:00
|
|
|
* @xprt: pointer to transport structure
|
2005-08-12 03:25:26 +07:00
|
|
|
* @task: address of RPC task that manages state of connect request
|
|
|
|
*
|
|
|
|
* TCP: If the remote end dropped the connection, delay reconnecting.
|
2005-08-26 06:25:55 +07:00
|
|
|
*
|
|
|
|
* UDP socket connects are synchronous, but we use a work queue anyway
|
|
|
|
* to guarantee that even unprivileged user processes can set up a
|
|
|
|
* socket on a privileged port.
|
|
|
|
*
|
|
|
|
* If a UDP socket connect fails, the delay behavior here prevents
|
|
|
|
* retry floods (hard mounts).
|
2005-08-12 03:25:26 +07:00
|
|
|
*/
|
2013-01-08 21:26:49 +07:00
|
|
|
static void xs_connect(struct rpc_xprt *xprt, struct rpc_task *task)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2006-12-06 04:35:15 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2015-02-09 06:19:25 +07:00
|
|
|
WARN_ON_ONCE(!xprt_lock_connect(xprt, task, transport));
|
|
|
|
|
2015-08-14 02:33:51 +07:00
|
|
|
if (transport->sock != NULL) {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_connect delayed xprt %p for %lu "
|
|
|
|
"seconds\n",
|
2005-08-26 06:25:55 +07:00
|
|
|
xprt, xprt->reestablish_timeout / HZ);
|
2015-08-14 02:33:51 +07:00
|
|
|
|
|
|
|
/* Start by resetting any existing state */
|
|
|
|
xs_reset_transport(transport);
|
|
|
|
|
2007-06-15 05:00:42 +07:00
|
|
|
queue_delayed_work(rpciod_workqueue,
|
|
|
|
&transport->connect_worker,
|
|
|
|
xprt->reestablish_timeout);
|
2005-08-26 06:25:55 +07:00
|
|
|
xprt->reestablish_timeout <<= 1;
|
NFS/RPC: fix problems with reestablish_timeout and related code.
[[resending with correct cc: - "vfs.kernel.org" just isn't right!]]
xprt->reestablish_timeout is used to cause TCP connection attempts to
back off if the connection fails so as not to hammer the network,
but to still allow immediate connections when there is no reason to
believe there is a problem.
It is not used for the first connection (when transport->sock is NULL)
but only on reconnects.
It is currently set:
a/ to 0 when xs_tcp_state_change finds a state of TCP_FIN_WAIT1
on the assumption that the client has closed the connection
so the reconnect should be immediate when needed.
b/ to at least XS_TCP_INIT_REEST_TO when xs_tcp_state_change
detects TCP_CLOSING or TCP_CLOSE_WAIT on the assumption that the
server closed the connection so a small delay at least is
required.
c/ as above when xs_tcp_state_change detects TCP_SYN_SENT, so that
it is never 0 while a connection has been attempted, else
the doubling will produce 0 and there will be no backoff.
d/ to double is value (up to a limit) when delaying a connection,
thus providing exponential backoff and
e/ to XS_TCP_INIT_REEST_TO in xs_setup_tcp as simple initialisation.
So you can see it is highly dependant on xs_tcp_state_change being
called as expected. However experimental evidence shows that
xs_tcp_state_change does not see all state changes.
("rpcdebug -m rpc trans" can help show what actually happens).
Results show:
TCP_ESTABLISHED is reported when a connection is made. TCP_SYN_SENT
is never reported, so rule 'c' above is never effective.
When the server closes the connection, TCP_CLOSE_WAIT and
TCP_LAST_ACK *might* be reported, and TCP_CLOSE is always
reported. This rule 'b' above will sometimes be effective, but
not reliably.
When the client closes the connection, it used to result in
TCP_FIN_WAIT1, TCP_FIN_WAIT2, TCP_CLOSE. However since commit
f75e674 (SUNRPC: Fix the problem of EADDRNOTAVAIL syslog floods on
reconnect) we don't see *any* events on client-close. I think this
is because xs_restore_old_callbacks is called to disconnect
xs_tcp_state_change before the socket is closed.
In any case, rule 'a' no longer applies.
So all that is left are rule d, which successfully doubles the
timeout which is never rest, and rule e which initialises the timeout.
Even if the rules worked as expected, there would be a problem because
a successful connection does not reset the timeout, so a sequence
of events where the server closes the connection (e.g. during failover
testing) will cause longer and longer timeouts with no good reason.
This patch:
- sets reestablish_timeout to 0 in xs_close thus effecting rule 'a'
- sets it to 0 in xs_tcp_data_ready to ensure that a successful
connection resets the timeout
- sets it to at least XS_TCP_INIT_REEST_TO after it is doubled,
thus effecting rule c
I have not reimplemented rule b and the new version of rule c
seems sufficient.
I suspect other code in xs_tcp_data_ready needs to be revised as well.
For example I don't think connect_cookie is being incremented as often
as it should be.
Signed-off-by: NeilBrown <neilb@suse.de>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
2009-09-24 01:36:37 +07:00
|
|
|
if (xprt->reestablish_timeout < XS_TCP_INIT_REEST_TO)
|
|
|
|
xprt->reestablish_timeout = XS_TCP_INIT_REEST_TO;
|
2005-08-26 06:25:55 +07:00
|
|
|
if (xprt->reestablish_timeout > XS_TCP_MAX_REEST_TO)
|
|
|
|
xprt->reestablish_timeout = XS_TCP_MAX_REEST_TO;
|
2005-08-12 03:25:53 +07:00
|
|
|
} else {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_connect scheduled xprt %p\n", xprt);
|
2007-06-15 05:00:42 +07:00
|
|
|
queue_delayed_work(rpciod_workqueue,
|
|
|
|
&transport->connect_worker, 0);
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
/**
|
|
|
|
* xs_local_print_stats - display AF_LOCAL socket-specifc stats
|
|
|
|
* @xprt: rpc_xprt struct containing statistics
|
|
|
|
* @seq: output file
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_local_print_stats(struct rpc_xprt *xprt, struct seq_file *seq)
|
|
|
|
{
|
|
|
|
long idle_time = 0;
|
|
|
|
|
|
|
|
if (xprt_connected(xprt))
|
|
|
|
idle_time = (long)(jiffies - xprt->last_used) / HZ;
|
|
|
|
|
|
|
|
seq_printf(seq, "\txprt:\tlocal %lu %lu %lu %ld %lu %lu %lu "
|
2012-02-15 04:19:18 +07:00
|
|
|
"%llu %llu %lu %llu %llu\n",
|
2011-05-10 02:22:44 +07:00
|
|
|
xprt->stat.bind_count,
|
|
|
|
xprt->stat.connect_count,
|
|
|
|
xprt->stat.connect_time,
|
|
|
|
idle_time,
|
|
|
|
xprt->stat.sends,
|
|
|
|
xprt->stat.recvs,
|
|
|
|
xprt->stat.bad_xids,
|
|
|
|
xprt->stat.req_u,
|
2012-02-15 04:19:18 +07:00
|
|
|
xprt->stat.bklog_u,
|
|
|
|
xprt->stat.max_slots,
|
|
|
|
xprt->stat.sending_u,
|
|
|
|
xprt->stat.pending_u);
|
2011-05-10 02:22:44 +07:00
|
|
|
}
|
|
|
|
|
2006-03-21 01:44:16 +07:00
|
|
|
/**
|
|
|
|
* xs_udp_print_stats - display UDP socket-specifc stats
|
|
|
|
* @xprt: rpc_xprt struct containing statistics
|
|
|
|
* @seq: output file
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_udp_print_stats(struct rpc_xprt *xprt, struct seq_file *seq)
|
|
|
|
{
|
2006-12-06 04:35:26 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
2012-02-15 04:19:18 +07:00
|
|
|
seq_printf(seq, "\txprt:\tudp %u %lu %lu %lu %lu %llu %llu "
|
|
|
|
"%lu %llu %llu\n",
|
2009-08-10 02:09:46 +07:00
|
|
|
transport->srcport,
|
2006-03-21 01:44:16 +07:00
|
|
|
xprt->stat.bind_count,
|
|
|
|
xprt->stat.sends,
|
|
|
|
xprt->stat.recvs,
|
|
|
|
xprt->stat.bad_xids,
|
|
|
|
xprt->stat.req_u,
|
2012-02-15 04:19:18 +07:00
|
|
|
xprt->stat.bklog_u,
|
|
|
|
xprt->stat.max_slots,
|
|
|
|
xprt->stat.sending_u,
|
|
|
|
xprt->stat.pending_u);
|
2006-03-21 01:44:16 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xs_tcp_print_stats - display TCP socket-specifc stats
|
|
|
|
* @xprt: rpc_xprt struct containing statistics
|
|
|
|
* @seq: output file
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void xs_tcp_print_stats(struct rpc_xprt *xprt, struct seq_file *seq)
|
|
|
|
{
|
2006-12-06 04:35:26 +07:00
|
|
|
struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt);
|
2006-03-21 01:44:16 +07:00
|
|
|
long idle_time = 0;
|
|
|
|
|
|
|
|
if (xprt_connected(xprt))
|
|
|
|
idle_time = (long)(jiffies - xprt->last_used) / HZ;
|
|
|
|
|
2012-02-15 04:19:18 +07:00
|
|
|
seq_printf(seq, "\txprt:\ttcp %u %lu %lu %lu %ld %lu %lu %lu "
|
|
|
|
"%llu %llu %lu %llu %llu\n",
|
2009-08-10 02:09:46 +07:00
|
|
|
transport->srcport,
|
2006-03-21 01:44:16 +07:00
|
|
|
xprt->stat.bind_count,
|
|
|
|
xprt->stat.connect_count,
|
|
|
|
xprt->stat.connect_time,
|
|
|
|
idle_time,
|
|
|
|
xprt->stat.sends,
|
|
|
|
xprt->stat.recvs,
|
|
|
|
xprt->stat.bad_xids,
|
|
|
|
xprt->stat.req_u,
|
2012-02-15 04:19:18 +07:00
|
|
|
xprt->stat.bklog_u,
|
|
|
|
xprt->stat.max_slots,
|
|
|
|
xprt->stat.sending_u,
|
|
|
|
xprt->stat.pending_u);
|
2006-03-21 01:44:16 +07:00
|
|
|
}
|
|
|
|
|
2009-09-10 21:32:28 +07:00
|
|
|
/*
|
|
|
|
* Allocate a bunch of pages for a scratch buffer for the rpc code. The reason
|
|
|
|
* we allocate pages instead doing a kmalloc like rpc_malloc is because we want
|
|
|
|
* to use the server side send routines.
|
|
|
|
*/
|
2010-01-15 05:38:31 +07:00
|
|
|
static void *bc_malloc(struct rpc_task *task, size_t size)
|
2009-09-10 21:32:28 +07:00
|
|
|
{
|
|
|
|
struct page *page;
|
|
|
|
struct rpc_buffer *buf;
|
|
|
|
|
2012-10-23 21:43:43 +07:00
|
|
|
WARN_ON_ONCE(size > PAGE_SIZE - sizeof(struct rpc_buffer));
|
|
|
|
if (size > PAGE_SIZE - sizeof(struct rpc_buffer))
|
|
|
|
return NULL;
|
2009-09-10 21:32:28 +07:00
|
|
|
|
2012-10-23 21:43:43 +07:00
|
|
|
page = alloc_page(GFP_KERNEL);
|
2009-09-10 21:32:28 +07:00
|
|
|
if (!page)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
buf = page_address(page);
|
|
|
|
buf->len = PAGE_SIZE;
|
|
|
|
|
|
|
|
return buf->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Free the space allocated in the bc_alloc routine
|
|
|
|
*/
|
2010-01-15 05:38:31 +07:00
|
|
|
static void bc_free(void *buffer)
|
2009-09-10 21:32:28 +07:00
|
|
|
{
|
|
|
|
struct rpc_buffer *buf;
|
|
|
|
|
|
|
|
if (!buffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
buf = container_of(buffer, struct rpc_buffer, data);
|
|
|
|
free_page((unsigned long)buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Use the svc_sock to send the callback. Must be called with svsk->sk_mutex
|
|
|
|
* held. Borrows heavily from svc_tcp_sendto and xs_tcp_send_request.
|
|
|
|
*/
|
|
|
|
static int bc_sendto(struct rpc_rqst *req)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
struct xdr_buf *xbufp = &req->rq_snd_buf;
|
|
|
|
struct rpc_xprt *xprt = req->rq_xprt;
|
|
|
|
struct sock_xprt *transport =
|
|
|
|
container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
struct socket *sock = transport->sock;
|
|
|
|
unsigned long headoff;
|
|
|
|
unsigned long tailoff;
|
|
|
|
|
2011-05-10 02:22:34 +07:00
|
|
|
xs_encode_stream_record_marker(xbufp);
|
2009-09-10 21:32:28 +07:00
|
|
|
|
|
|
|
tailoff = (unsigned long)xbufp->tail[0].iov_base & ~PAGE_MASK;
|
|
|
|
headoff = (unsigned long)xbufp->head[0].iov_base & ~PAGE_MASK;
|
|
|
|
len = svc_send_common(sock, xbufp,
|
|
|
|
virt_to_page(xbufp->head[0].iov_base), headoff,
|
|
|
|
xbufp->tail[0].iov_base, tailoff);
|
|
|
|
|
|
|
|
if (len != xbufp->len) {
|
|
|
|
printk(KERN_NOTICE "Error sending entire callback!\n");
|
|
|
|
len = -EAGAIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The send routine. Borrows from svc_send
|
|
|
|
*/
|
|
|
|
static int bc_send_request(struct rpc_task *task)
|
|
|
|
{
|
|
|
|
struct rpc_rqst *req = task->tk_rqstp;
|
|
|
|
struct svc_xprt *xprt;
|
|
|
|
u32 len;
|
|
|
|
|
|
|
|
dprintk("sending request with xid: %08x\n", ntohl(req->rq_xid));
|
|
|
|
/*
|
|
|
|
* Get the server socket associated with this callback xprt
|
|
|
|
*/
|
|
|
|
xprt = req->rq_xprt->bc_xprt;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Grab the mutex to serialize data as the connection is shared
|
|
|
|
* with the fore channel
|
|
|
|
*/
|
|
|
|
if (!mutex_trylock(&xprt->xpt_mutex)) {
|
|
|
|
rpc_sleep_on(&xprt->xpt_bc_pending, task, NULL);
|
|
|
|
if (!mutex_trylock(&xprt->xpt_mutex))
|
|
|
|
return -EAGAIN;
|
|
|
|
rpc_wake_up_queued_task(&xprt->xpt_bc_pending, task);
|
|
|
|
}
|
|
|
|
if (test_bit(XPT_DEAD, &xprt->xpt_flags))
|
|
|
|
len = -ENOTCONN;
|
|
|
|
else
|
|
|
|
len = bc_sendto(req);
|
|
|
|
mutex_unlock(&xprt->xpt_mutex);
|
|
|
|
|
|
|
|
if (len > 0)
|
|
|
|
len = 0;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The close routine. Since this is client initiated, we do nothing
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void bc_close(struct rpc_xprt *xprt)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The xprt destroy routine. Again, because this connection is client
|
|
|
|
* initiated, we do nothing
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void bc_destroy(struct rpc_xprt *xprt)
|
|
|
|
{
|
2014-03-24 10:58:16 +07:00
|
|
|
dprintk("RPC: bc_destroy xprt %p\n", xprt);
|
|
|
|
|
|
|
|
xs_xprt_free(xprt);
|
|
|
|
module_put(THIS_MODULE);
|
2009-09-10 21:32:28 +07:00
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static struct rpc_xprt_ops xs_local_ops = {
|
|
|
|
.reserve_xprt = xprt_reserve_xprt,
|
|
|
|
.release_xprt = xs_tcp_release_xprt,
|
2012-09-07 22:08:50 +07:00
|
|
|
.alloc_slot = xprt_alloc_slot,
|
2011-05-10 02:22:44 +07:00
|
|
|
.rpcbind = xs_local_rpcbind,
|
|
|
|
.set_port = xs_local_set_port,
|
2013-02-21 05:52:19 +07:00
|
|
|
.connect = xs_local_connect,
|
2011-05-10 02:22:44 +07:00
|
|
|
.buf_alloc = rpc_malloc,
|
|
|
|
.buf_free = rpc_free,
|
|
|
|
.send_request = xs_local_send_request,
|
|
|
|
.set_retrans_timeout = xprt_set_retrans_timeout_def,
|
|
|
|
.close = xs_close,
|
2013-10-31 20:18:49 +07:00
|
|
|
.destroy = xs_destroy,
|
2011-05-10 02:22:44 +07:00
|
|
|
.print_stats = xs_local_print_stats,
|
2015-06-04 03:14:29 +07:00
|
|
|
.enable_swap = xs_enable_swap,
|
|
|
|
.disable_swap = xs_disable_swap,
|
2011-05-10 02:22:44 +07:00
|
|
|
};
|
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
static struct rpc_xprt_ops xs_udp_ops = {
|
2005-08-26 06:25:49 +07:00
|
|
|
.set_buffer_size = xs_udp_set_buffer_size,
|
2005-08-26 06:25:51 +07:00
|
|
|
.reserve_xprt = xprt_reserve_xprt_cong,
|
2005-08-26 06:25:51 +07:00
|
|
|
.release_xprt = xprt_release_xprt_cong,
|
2012-09-07 22:08:50 +07:00
|
|
|
.alloc_slot = xprt_alloc_slot,
|
2007-07-01 23:13:17 +07:00
|
|
|
.rpcbind = rpcb_getport_async,
|
2006-01-03 15:55:51 +07:00
|
|
|
.set_port = xs_set_port,
|
2005-08-12 03:25:56 +07:00
|
|
|
.connect = xs_connect,
|
2006-01-03 15:55:49 +07:00
|
|
|
.buf_alloc = rpc_malloc,
|
|
|
|
.buf_free = rpc_free,
|
2005-08-12 03:25:56 +07:00
|
|
|
.send_request = xs_udp_send_request,
|
2005-08-26 06:25:50 +07:00
|
|
|
.set_retrans_timeout = xprt_set_retrans_timeout_rtt,
|
2005-08-26 06:25:52 +07:00
|
|
|
.timer = xs_udp_timer,
|
2005-08-26 06:25:53 +07:00
|
|
|
.release_request = xprt_release_rqst_cong,
|
2005-08-12 03:25:56 +07:00
|
|
|
.close = xs_close,
|
|
|
|
.destroy = xs_destroy,
|
2006-03-21 01:44:16 +07:00
|
|
|
.print_stats = xs_udp_print_stats,
|
2015-06-04 03:14:29 +07:00
|
|
|
.enable_swap = xs_enable_swap,
|
|
|
|
.disable_swap = xs_disable_swap,
|
2015-05-12 01:02:25 +07:00
|
|
|
.inject_disconnect = xs_inject_disconnect,
|
2005-08-12 03:25:56 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct rpc_xprt_ops xs_tcp_ops = {
|
2005-08-26 06:25:51 +07:00
|
|
|
.reserve_xprt = xprt_reserve_xprt,
|
2006-07-28 04:22:50 +07:00
|
|
|
.release_xprt = xs_tcp_release_xprt,
|
2012-09-07 22:08:50 +07:00
|
|
|
.alloc_slot = xprt_lock_and_alloc_slot,
|
2007-07-01 23:13:17 +07:00
|
|
|
.rpcbind = rpcb_getport_async,
|
2006-01-03 15:55:51 +07:00
|
|
|
.set_port = xs_set_port,
|
2010-04-17 03:41:57 +07:00
|
|
|
.connect = xs_connect,
|
2006-01-03 15:55:49 +07:00
|
|
|
.buf_alloc = rpc_malloc,
|
|
|
|
.buf_free = rpc_free,
|
2005-08-12 03:25:56 +07:00
|
|
|
.send_request = xs_tcp_send_request,
|
2005-08-26 06:25:50 +07:00
|
|
|
.set_retrans_timeout = xprt_set_retrans_timeout_def,
|
2015-02-10 23:06:04 +07:00
|
|
|
.close = xs_tcp_shutdown,
|
2005-08-12 03:25:26 +07:00
|
|
|
.destroy = xs_destroy,
|
2006-03-21 01:44:16 +07:00
|
|
|
.print_stats = xs_tcp_print_stats,
|
2015-06-04 03:14:29 +07:00
|
|
|
.enable_swap = xs_enable_swap,
|
|
|
|
.disable_swap = xs_disable_swap,
|
2015-05-12 01:02:25 +07:00
|
|
|
.inject_disconnect = xs_inject_disconnect,
|
2005-08-12 03:25:23 +07:00
|
|
|
};
|
|
|
|
|
2009-09-10 21:32:28 +07:00
|
|
|
/*
|
|
|
|
* The rpc_xprt_ops for the server backchannel
|
|
|
|
*/
|
|
|
|
|
|
|
|
static struct rpc_xprt_ops bc_tcp_ops = {
|
|
|
|
.reserve_xprt = xprt_reserve_xprt,
|
|
|
|
.release_xprt = xprt_release_xprt,
|
2012-09-25 00:39:01 +07:00
|
|
|
.alloc_slot = xprt_alloc_slot,
|
2009-09-10 21:32:28 +07:00
|
|
|
.buf_alloc = bc_malloc,
|
|
|
|
.buf_free = bc_free,
|
|
|
|
.send_request = bc_send_request,
|
|
|
|
.set_retrans_timeout = xprt_set_retrans_timeout_def,
|
|
|
|
.close = bc_close,
|
|
|
|
.destroy = bc_destroy,
|
|
|
|
.print_stats = xs_tcp_print_stats,
|
2015-06-04 03:14:29 +07:00
|
|
|
.enable_swap = xs_enable_swap,
|
|
|
|
.disable_swap = xs_disable_swap,
|
2015-05-12 01:02:25 +07:00
|
|
|
.inject_disconnect = xs_inject_disconnect,
|
2009-09-10 21:32:28 +07:00
|
|
|
};
|
|
|
|
|
SUNRPC: Properly initialize sock_xprt.srcaddr in all cases
The source address field in the transport's sock_xprt is initialized
ONLY IF the RPC application passed a pointer to a source address
during the call to rpc_create(). However, xs_bind() subsequently uses
the value of this field without regard to whether the source address
was initialized during transport creation or not.
So far we've been lucky: the uninitialized value of this field is
zeroes. xs_bind(), until recently, used only the sin[6]_addr field in
this sockaddr, and all zeroes is a valid value for this: it means
ANYADDR. This is a happy coincidence.
However, xs_bind() now wants to use the sa_family field as well, and
expects it to be initialized to something other than zero.
Therefore, the source address sockaddr field should be fully
initialized at transport create time in _every_ case, not just when
the RPC application wants to use a specific bind address.
Bruce added a workaround for this missing initialization by adjusting
commit 6bc9638a, but the "right" way to do this is to ensure that the
source address sockaddr is always correctly initialized from the
get-go.
This patch doesn't introduce a behavior change. It's simply a
clean-up of Bruce's fix, to prevent future problems of this kind. It
may look like overkill, but
a) it clearly documents the default initial value of this field,
b) it doesn't assume that the sockaddr_storage memory is first
initialized to any particular value, and
c) it will fail verbosely if some unknown address family is passed
in
Originally introduced by commit d3bc9a1d.
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
2010-10-20 22:53:01 +07:00
|
|
|
static int xs_init_anyaddr(const int family, struct sockaddr *sap)
|
|
|
|
{
|
|
|
|
static const struct sockaddr_in sin = {
|
|
|
|
.sin_family = AF_INET,
|
|
|
|
.sin_addr.s_addr = htonl(INADDR_ANY),
|
|
|
|
};
|
|
|
|
static const struct sockaddr_in6 sin6 = {
|
|
|
|
.sin6_family = AF_INET6,
|
|
|
|
.sin6_addr = IN6ADDR_ANY_INIT,
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (family) {
|
2011-05-10 02:22:44 +07:00
|
|
|
case AF_LOCAL:
|
|
|
|
break;
|
SUNRPC: Properly initialize sock_xprt.srcaddr in all cases
The source address field in the transport's sock_xprt is initialized
ONLY IF the RPC application passed a pointer to a source address
during the call to rpc_create(). However, xs_bind() subsequently uses
the value of this field without regard to whether the source address
was initialized during transport creation or not.
So far we've been lucky: the uninitialized value of this field is
zeroes. xs_bind(), until recently, used only the sin[6]_addr field in
this sockaddr, and all zeroes is a valid value for this: it means
ANYADDR. This is a happy coincidence.
However, xs_bind() now wants to use the sa_family field as well, and
expects it to be initialized to something other than zero.
Therefore, the source address sockaddr field should be fully
initialized at transport create time in _every_ case, not just when
the RPC application wants to use a specific bind address.
Bruce added a workaround for this missing initialization by adjusting
commit 6bc9638a, but the "right" way to do this is to ensure that the
source address sockaddr is always correctly initialized from the
get-go.
This patch doesn't introduce a behavior change. It's simply a
clean-up of Bruce's fix, to prevent future problems of this kind. It
may look like overkill, but
a) it clearly documents the default initial value of this field,
b) it doesn't assume that the sockaddr_storage memory is first
initialized to any particular value, and
c) it will fail verbosely if some unknown address family is passed
in
Originally introduced by commit d3bc9a1d.
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
2010-10-20 22:53:01 +07:00
|
|
|
case AF_INET:
|
|
|
|
memcpy(sap, &sin, sizeof(sin));
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
memcpy(sap, &sin6, sizeof(sin6));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dprintk("RPC: %s: Bad address family\n", __func__);
|
|
|
|
return -EAFNOSUPPORT;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-11 00:47:07 +07:00
|
|
|
static struct rpc_xprt *xs_setup_xprt(struct xprt_create *args,
|
2011-07-18 05:11:30 +07:00
|
|
|
unsigned int slot_table_size,
|
|
|
|
unsigned int max_slot_table_size)
|
2006-10-18 01:44:27 +07:00
|
|
|
{
|
|
|
|
struct rpc_xprt *xprt;
|
2006-12-06 04:35:11 +07:00
|
|
|
struct sock_xprt *new;
|
2006-10-18 01:44:27 +07:00
|
|
|
|
2007-07-08 18:08:54 +07:00
|
|
|
if (args->addrlen > sizeof(xprt->addr)) {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_setup_xprt: address too large\n");
|
2006-10-18 01:44:27 +07:00
|
|
|
return ERR_PTR(-EBADF);
|
|
|
|
}
|
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
xprt = xprt_alloc(args->net, sizeof(*new), slot_table_size,
|
|
|
|
max_slot_table_size);
|
2010-09-29 19:02:43 +07:00
|
|
|
if (xprt == NULL) {
|
2007-02-01 00:14:08 +07:00
|
|
|
dprintk("RPC: xs_setup_xprt: couldn't allocate "
|
|
|
|
"rpc_xprt\n");
|
2006-10-18 01:44:27 +07:00
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
}
|
|
|
|
|
2010-09-29 19:02:43 +07:00
|
|
|
new = container_of(xprt, struct sock_xprt, xprt);
|
2007-07-08 18:08:54 +07:00
|
|
|
memcpy(&xprt->addr, args->dstaddr, args->addrlen);
|
|
|
|
xprt->addrlen = args->addrlen;
|
2007-07-10 03:23:35 +07:00
|
|
|
if (args->srcaddr)
|
2009-08-10 02:09:46 +07:00
|
|
|
memcpy(&new->srcaddr, args->srcaddr, args->addrlen);
|
SUNRPC: Properly initialize sock_xprt.srcaddr in all cases
The source address field in the transport's sock_xprt is initialized
ONLY IF the RPC application passed a pointer to a source address
during the call to rpc_create(). However, xs_bind() subsequently uses
the value of this field without regard to whether the source address
was initialized during transport creation or not.
So far we've been lucky: the uninitialized value of this field is
zeroes. xs_bind(), until recently, used only the sin[6]_addr field in
this sockaddr, and all zeroes is a valid value for this: it means
ANYADDR. This is a happy coincidence.
However, xs_bind() now wants to use the sa_family field as well, and
expects it to be initialized to something other than zero.
Therefore, the source address sockaddr field should be fully
initialized at transport create time in _every_ case, not just when
the RPC application wants to use a specific bind address.
Bruce added a workaround for this missing initialization by adjusting
commit 6bc9638a, but the "right" way to do this is to ensure that the
source address sockaddr is always correctly initialized from the
get-go.
This patch doesn't introduce a behavior change. It's simply a
clean-up of Bruce's fix, to prevent future problems of this kind. It
may look like overkill, but
a) it clearly documents the default initial value of this field,
b) it doesn't assume that the sockaddr_storage memory is first
initialized to any particular value, and
c) it will fail verbosely if some unknown address family is passed
in
Originally introduced by commit d3bc9a1d.
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
2010-10-20 22:53:01 +07:00
|
|
|
else {
|
|
|
|
int err;
|
|
|
|
err = xs_init_anyaddr(args->dstaddr->sa_family,
|
|
|
|
(struct sockaddr *)&new->srcaddr);
|
2011-11-10 18:33:23 +07:00
|
|
|
if (err != 0) {
|
|
|
|
xprt_free(xprt);
|
SUNRPC: Properly initialize sock_xprt.srcaddr in all cases
The source address field in the transport's sock_xprt is initialized
ONLY IF the RPC application passed a pointer to a source address
during the call to rpc_create(). However, xs_bind() subsequently uses
the value of this field without regard to whether the source address
was initialized during transport creation or not.
So far we've been lucky: the uninitialized value of this field is
zeroes. xs_bind(), until recently, used only the sin[6]_addr field in
this sockaddr, and all zeroes is a valid value for this: it means
ANYADDR. This is a happy coincidence.
However, xs_bind() now wants to use the sa_family field as well, and
expects it to be initialized to something other than zero.
Therefore, the source address sockaddr field should be fully
initialized at transport create time in _every_ case, not just when
the RPC application wants to use a specific bind address.
Bruce added a workaround for this missing initialization by adjusting
commit 6bc9638a, but the "right" way to do this is to ensure that the
source address sockaddr is always correctly initialized from the
get-go.
This patch doesn't introduce a behavior change. It's simply a
clean-up of Bruce's fix, to prevent future problems of this kind. It
may look like overkill, but
a) it clearly documents the default initial value of this field,
b) it doesn't assume that the sockaddr_storage memory is first
initialized to any particular value, and
c) it will fail verbosely if some unknown address family is passed
in
Originally introduced by commit d3bc9a1d.
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
2010-10-20 22:53:01 +07:00
|
|
|
return ERR_PTR(err);
|
2011-11-10 18:33:23 +07:00
|
|
|
}
|
SUNRPC: Properly initialize sock_xprt.srcaddr in all cases
The source address field in the transport's sock_xprt is initialized
ONLY IF the RPC application passed a pointer to a source address
during the call to rpc_create(). However, xs_bind() subsequently uses
the value of this field without regard to whether the source address
was initialized during transport creation or not.
So far we've been lucky: the uninitialized value of this field is
zeroes. xs_bind(), until recently, used only the sin[6]_addr field in
this sockaddr, and all zeroes is a valid value for this: it means
ANYADDR. This is a happy coincidence.
However, xs_bind() now wants to use the sa_family field as well, and
expects it to be initialized to something other than zero.
Therefore, the source address sockaddr field should be fully
initialized at transport create time in _every_ case, not just when
the RPC application wants to use a specific bind address.
Bruce added a workaround for this missing initialization by adjusting
commit 6bc9638a, but the "right" way to do this is to ensure that the
source address sockaddr is always correctly initialized from the
get-go.
This patch doesn't introduce a behavior change. It's simply a
clean-up of Bruce's fix, to prevent future problems of this kind. It
may look like overkill, but
a) it clearly documents the default initial value of this field,
b) it doesn't assume that the sockaddr_storage memory is first
initialized to any particular value, and
c) it will fail verbosely if some unknown address family is passed
in
Originally introduced by commit d3bc9a1d.
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
2010-10-20 22:53:01 +07:00
|
|
|
}
|
2006-10-18 01:44:27 +07:00
|
|
|
|
|
|
|
return xprt;
|
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static const struct rpc_timeout xs_local_default_timeout = {
|
|
|
|
.to_initval = 10 * HZ,
|
|
|
|
.to_maxval = 10 * HZ,
|
|
|
|
.to_retries = 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xs_setup_local - Set up transport to use an AF_LOCAL socket
|
|
|
|
* @args: rpc transport creation arguments
|
|
|
|
*
|
|
|
|
* AF_LOCAL is a "tpi_cots_ord" transport, just like TCP
|
|
|
|
*/
|
|
|
|
static struct rpc_xprt *xs_setup_local(struct xprt_create *args)
|
|
|
|
{
|
|
|
|
struct sockaddr_un *sun = (struct sockaddr_un *)args->dstaddr;
|
|
|
|
struct sock_xprt *transport;
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
struct rpc_xprt *ret;
|
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
xprt = xs_setup_xprt(args, xprt_tcp_slot_table_entries,
|
|
|
|
xprt_max_tcp_slot_table_entries);
|
2011-05-10 02:22:44 +07:00
|
|
|
if (IS_ERR(xprt))
|
|
|
|
return xprt;
|
|
|
|
transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
xprt->prot = 0;
|
|
|
|
xprt->tsh_size = sizeof(rpc_fraghdr) / sizeof(u32);
|
|
|
|
xprt->max_payload = RPC_MAX_FRAGMENT_SIZE;
|
|
|
|
|
|
|
|
xprt->bind_timeout = XS_BIND_TO;
|
|
|
|
xprt->reestablish_timeout = XS_TCP_INIT_REEST_TO;
|
|
|
|
xprt->idle_timeout = XS_IDLE_DISC_TO;
|
|
|
|
|
|
|
|
xprt->ops = &xs_local_ops;
|
|
|
|
xprt->timeout = &xs_local_default_timeout;
|
|
|
|
|
2013-10-31 12:14:36 +07:00
|
|
|
INIT_DELAYED_WORK(&transport->connect_worker,
|
|
|
|
xs_dummy_setup_socket);
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
switch (sun->sun_family) {
|
|
|
|
case AF_LOCAL:
|
|
|
|
if (sun->sun_path[0] != '/') {
|
|
|
|
dprintk("RPC: bad AF_LOCAL address: %s\n",
|
|
|
|
sun->sun_path);
|
|
|
|
ret = ERR_PTR(-EINVAL);
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
xs_format_peer_addresses(xprt, "local", RPCBIND_NETID_LOCAL);
|
2013-02-21 22:14:22 +07:00
|
|
|
ret = ERR_PTR(xs_local_setup_socket(transport));
|
|
|
|
if (ret)
|
|
|
|
goto out_err;
|
2011-05-10 02:22:44 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = ERR_PTR(-EAFNOSUPPORT);
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
dprintk("RPC: set up xprt to %s via AF_LOCAL\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR]);
|
|
|
|
|
|
|
|
if (try_module_get(THIS_MODULE))
|
|
|
|
return xprt;
|
|
|
|
ret = ERR_PTR(-EINVAL);
|
|
|
|
out_err:
|
2014-03-24 10:07:22 +07:00
|
|
|
xs_xprt_free(xprt);
|
2011-05-10 02:22:44 +07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-12-21 04:03:54 +07:00
|
|
|
static const struct rpc_timeout xs_udp_default_timeout = {
|
|
|
|
.to_initval = 5 * HZ,
|
|
|
|
.to_maxval = 30 * HZ,
|
|
|
|
.to_increment = 5 * HZ,
|
|
|
|
.to_retries = 5,
|
|
|
|
};
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_setup_udp - Set up transport to use a UDP socket
|
2007-07-08 18:08:54 +07:00
|
|
|
* @args: rpc transport creation arguments
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
|
|
|
*/
|
2007-10-24 23:24:02 +07:00
|
|
|
static struct rpc_xprt *xs_setup_udp(struct xprt_create *args)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2007-08-06 22:57:53 +07:00
|
|
|
struct sockaddr *addr = args->dstaddr;
|
2006-10-18 01:44:27 +07:00
|
|
|
struct rpc_xprt *xprt;
|
2006-12-06 04:35:26 +07:00
|
|
|
struct sock_xprt *transport;
|
2010-05-26 19:42:24 +07:00
|
|
|
struct rpc_xprt *ret;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
xprt = xs_setup_xprt(args, xprt_udp_slot_table_entries,
|
|
|
|
xprt_udp_slot_table_entries);
|
2006-10-18 01:44:27 +07:00
|
|
|
if (IS_ERR(xprt))
|
|
|
|
return xprt;
|
2006-12-06 04:35:26 +07:00
|
|
|
transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-08-23 07:06:15 +07:00
|
|
|
xprt->prot = IPPROTO_UDP;
|
2005-08-26 06:25:49 +07:00
|
|
|
xprt->tsh_size = 0;
|
2005-08-12 03:25:23 +07:00
|
|
|
/* XXX: header size can vary due to auth type, IPv6, etc. */
|
|
|
|
xprt->max_payload = (1U << 16) - (MAX_HEADER << 3);
|
|
|
|
|
2005-08-26 06:25:55 +07:00
|
|
|
xprt->bind_timeout = XS_BIND_TO;
|
|
|
|
xprt->reestablish_timeout = XS_UDP_REEST_TO;
|
|
|
|
xprt->idle_timeout = XS_IDLE_DISC_TO;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
xprt->ops = &xs_udp_ops;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-12-21 04:03:55 +07:00
|
|
|
xprt->timeout = &xs_udp_default_timeout;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-08-06 22:57:53 +07:00
|
|
|
switch (addr->sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
if (((struct sockaddr_in *)addr)->sin_port != htons(0))
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
|
|
|
|
INIT_DELAYED_WORK(&transport->connect_worker,
|
2010-10-04 19:58:02 +07:00
|
|
|
xs_udp_setup_socket);
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_peer_addresses(xprt, "udp", RPCBIND_NETID_UDP);
|
2007-08-06 22:57:53 +07:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
if (((struct sockaddr_in6 *)addr)->sin6_port != htons(0))
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
|
|
|
|
INIT_DELAYED_WORK(&transport->connect_worker,
|
2010-10-04 19:58:02 +07:00
|
|
|
xs_udp_setup_socket);
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_peer_addresses(xprt, "udp", RPCBIND_NETID_UDP6);
|
2007-08-06 22:57:53 +07:00
|
|
|
break;
|
|
|
|
default:
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EAFNOSUPPORT);
|
|
|
|
goto out_err;
|
2007-08-06 22:57:53 +07:00
|
|
|
}
|
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
if (xprt_bound(xprt))
|
|
|
|
dprintk("RPC: set up xprt to %s (port %s) via %s\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO]);
|
|
|
|
else
|
|
|
|
dprintk("RPC: set up xprt to %s (autobind) via %s\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO]);
|
2006-08-23 07:06:18 +07:00
|
|
|
|
2007-09-11 00:46:39 +07:00
|
|
|
if (try_module_get(THIS_MODULE))
|
|
|
|
return xprt;
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EINVAL);
|
|
|
|
out_err:
|
2014-03-24 10:07:22 +07:00
|
|
|
xs_xprt_free(xprt);
|
2010-05-26 19:42:24 +07:00
|
|
|
return ret;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
|
|
|
|
2007-12-21 04:03:54 +07:00
|
|
|
static const struct rpc_timeout xs_tcp_default_timeout = {
|
|
|
|
.to_initval = 60 * HZ,
|
|
|
|
.to_maxval = 60 * HZ,
|
|
|
|
.to_retries = 2,
|
|
|
|
};
|
|
|
|
|
2005-08-12 03:25:26 +07:00
|
|
|
/**
|
|
|
|
* xs_setup_tcp - Set up transport to use a TCP socket
|
2007-07-08 18:08:54 +07:00
|
|
|
* @args: rpc transport creation arguments
|
2005-08-12 03:25:26 +07:00
|
|
|
*
|
|
|
|
*/
|
2007-10-24 23:24:02 +07:00
|
|
|
static struct rpc_xprt *xs_setup_tcp(struct xprt_create *args)
|
2005-08-12 03:25:23 +07:00
|
|
|
{
|
2007-08-06 22:57:53 +07:00
|
|
|
struct sockaddr *addr = args->dstaddr;
|
2006-10-18 01:44:27 +07:00
|
|
|
struct rpc_xprt *xprt;
|
2006-12-06 04:35:26 +07:00
|
|
|
struct sock_xprt *transport;
|
2010-05-26 19:42:24 +07:00
|
|
|
struct rpc_xprt *ret;
|
2013-04-14 22:42:00 +07:00
|
|
|
unsigned int max_slot_table_size = xprt_max_tcp_slot_table_entries;
|
|
|
|
|
|
|
|
if (args->flags & XPRT_CREATE_INFINITE_SLOTS)
|
|
|
|
max_slot_table_size = RPC_MAX_SLOT_TABLE_LIMIT;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
xprt = xs_setup_xprt(args, xprt_tcp_slot_table_entries,
|
2013-04-14 22:42:00 +07:00
|
|
|
max_slot_table_size);
|
2006-10-18 01:44:27 +07:00
|
|
|
if (IS_ERR(xprt))
|
|
|
|
return xprt;
|
2006-12-06 04:35:26 +07:00
|
|
|
transport = container_of(xprt, struct sock_xprt, xprt);
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2006-08-23 07:06:15 +07:00
|
|
|
xprt->prot = IPPROTO_TCP;
|
2005-08-26 06:25:49 +07:00
|
|
|
xprt->tsh_size = sizeof(rpc_fraghdr) / sizeof(u32);
|
|
|
|
xprt->max_payload = RPC_MAX_FRAGMENT_SIZE;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-26 06:25:55 +07:00
|
|
|
xprt->bind_timeout = XS_BIND_TO;
|
|
|
|
xprt->reestablish_timeout = XS_TCP_INIT_REEST_TO;
|
|
|
|
xprt->idle_timeout = XS_IDLE_DISC_TO;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2005-08-12 03:25:56 +07:00
|
|
|
xprt->ops = &xs_tcp_ops;
|
2007-12-21 04:03:55 +07:00
|
|
|
xprt->timeout = &xs_tcp_default_timeout;
|
2005-08-12 03:25:23 +07:00
|
|
|
|
2007-08-06 22:57:53 +07:00
|
|
|
switch (addr->sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
if (((struct sockaddr_in *)addr)->sin_port != htons(0))
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
INIT_DELAYED_WORK(&transport->connect_worker,
|
2010-10-04 19:57:40 +07:00
|
|
|
xs_tcp_setup_socket);
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_peer_addresses(xprt, "tcp", RPCBIND_NETID_TCP);
|
2007-08-06 22:57:53 +07:00
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
if (((struct sockaddr_in6 *)addr)->sin6_port != htons(0))
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
INIT_DELAYED_WORK(&transport->connect_worker,
|
2010-10-04 19:57:40 +07:00
|
|
|
xs_tcp_setup_socket);
|
2009-08-10 02:09:46 +07:00
|
|
|
xs_format_peer_addresses(xprt, "tcp", RPCBIND_NETID_TCP6);
|
2007-08-06 22:57:53 +07:00
|
|
|
break;
|
|
|
|
default:
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EAFNOSUPPORT);
|
|
|
|
goto out_err;
|
2007-08-06 22:57:53 +07:00
|
|
|
}
|
|
|
|
|
2009-08-10 02:09:46 +07:00
|
|
|
if (xprt_bound(xprt))
|
|
|
|
dprintk("RPC: set up xprt to %s (port %s) via %s\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO]);
|
|
|
|
else
|
|
|
|
dprintk("RPC: set up xprt to %s (autobind) via %s\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO]);
|
|
|
|
|
2007-09-11 00:46:39 +07:00
|
|
|
if (try_module_get(THIS_MODULE))
|
|
|
|
return xprt;
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EINVAL);
|
|
|
|
out_err:
|
2014-03-24 10:07:22 +07:00
|
|
|
xs_xprt_free(xprt);
|
2010-05-26 19:42:24 +07:00
|
|
|
return ret;
|
2005-08-12 03:25:23 +07:00
|
|
|
}
|
2006-12-06 04:35:51 +07:00
|
|
|
|
2009-09-10 21:33:30 +07:00
|
|
|
/**
|
|
|
|
* xs_setup_bc_tcp - Set up transport to use a TCP backchannel socket
|
|
|
|
* @args: rpc transport creation arguments
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static struct rpc_xprt *xs_setup_bc_tcp(struct xprt_create *args)
|
|
|
|
{
|
|
|
|
struct sockaddr *addr = args->dstaddr;
|
|
|
|
struct rpc_xprt *xprt;
|
|
|
|
struct sock_xprt *transport;
|
|
|
|
struct svc_sock *bc_sock;
|
2010-05-26 19:42:24 +07:00
|
|
|
struct rpc_xprt *ret;
|
2009-09-10 21:33:30 +07:00
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
xprt = xs_setup_xprt(args, xprt_tcp_slot_table_entries,
|
|
|
|
xprt_tcp_slot_table_entries);
|
2009-09-10 21:33:30 +07:00
|
|
|
if (IS_ERR(xprt))
|
|
|
|
return xprt;
|
|
|
|
transport = container_of(xprt, struct sock_xprt, xprt);
|
|
|
|
|
|
|
|
xprt->prot = IPPROTO_TCP;
|
|
|
|
xprt->tsh_size = sizeof(rpc_fraghdr) / sizeof(u32);
|
|
|
|
xprt->max_payload = RPC_MAX_FRAGMENT_SIZE;
|
|
|
|
xprt->timeout = &xs_tcp_default_timeout;
|
|
|
|
|
|
|
|
/* backchannel */
|
|
|
|
xprt_set_bound(xprt);
|
|
|
|
xprt->bind_timeout = 0;
|
|
|
|
xprt->reestablish_timeout = 0;
|
|
|
|
xprt->idle_timeout = 0;
|
|
|
|
|
|
|
|
xprt->ops = &bc_tcp_ops;
|
|
|
|
|
|
|
|
switch (addr->sa_family) {
|
|
|
|
case AF_INET:
|
|
|
|
xs_format_peer_addresses(xprt, "tcp",
|
|
|
|
RPCBIND_NETID_TCP);
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
xs_format_peer_addresses(xprt, "tcp",
|
|
|
|
RPCBIND_NETID_TCP6);
|
|
|
|
break;
|
|
|
|
default:
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EAFNOSUPPORT);
|
|
|
|
goto out_err;
|
2009-09-10 21:33:30 +07:00
|
|
|
}
|
|
|
|
|
2010-10-05 23:49:35 +07:00
|
|
|
dprintk("RPC: set up xprt to %s (port %s) via %s\n",
|
|
|
|
xprt->address_strings[RPC_DISPLAY_ADDR],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PORT],
|
|
|
|
xprt->address_strings[RPC_DISPLAY_PROTO]);
|
2009-09-10 21:33:30 +07:00
|
|
|
|
2010-12-09 00:45:44 +07:00
|
|
|
/*
|
|
|
|
* Once we've associated a backchannel xprt with a connection,
|
2013-11-30 16:56:44 +07:00
|
|
|
* we want to keep it around as long as the connection lasts,
|
|
|
|
* in case we need to start using it for a backchannel again;
|
|
|
|
* this reference won't be dropped until bc_xprt is destroyed.
|
2010-12-09 00:45:44 +07:00
|
|
|
*/
|
|
|
|
xprt_get(xprt);
|
|
|
|
args->bc_xprt->xpt_bc_xprt = xprt;
|
|
|
|
xprt->bc_xprt = args->bc_xprt;
|
|
|
|
bc_sock = container_of(args->bc_xprt, struct svc_sock, sk_xprt);
|
|
|
|
transport->sock = bc_sock->sk_sock;
|
|
|
|
transport->inet = bc_sock->sk_sk;
|
|
|
|
|
2009-09-10 21:33:30 +07:00
|
|
|
/*
|
|
|
|
* Since we don't want connections for the backchannel, we set
|
|
|
|
* the xprt status to connected
|
|
|
|
*/
|
|
|
|
xprt_set_connected(xprt);
|
|
|
|
|
|
|
|
if (try_module_get(THIS_MODULE))
|
|
|
|
return xprt;
|
2014-03-24 11:00:28 +07:00
|
|
|
|
|
|
|
args->bc_xprt->xpt_bc_xprt = NULL;
|
2010-12-09 00:45:44 +07:00
|
|
|
xprt_put(xprt);
|
2010-05-26 19:42:24 +07:00
|
|
|
ret = ERR_PTR(-EINVAL);
|
|
|
|
out_err:
|
2014-03-24 10:07:22 +07:00
|
|
|
xs_xprt_free(xprt);
|
2010-05-26 19:42:24 +07:00
|
|
|
return ret;
|
2009-09-10 21:33:30 +07:00
|
|
|
}
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
static struct xprt_class xs_local_transport = {
|
|
|
|
.list = LIST_HEAD_INIT(xs_local_transport.list),
|
|
|
|
.name = "named UNIX socket",
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.ident = XPRT_TRANSPORT_LOCAL,
|
|
|
|
.setup = xs_setup_local,
|
|
|
|
};
|
|
|
|
|
2007-09-11 00:46:39 +07:00
|
|
|
static struct xprt_class xs_udp_transport = {
|
|
|
|
.list = LIST_HEAD_INIT(xs_udp_transport.list),
|
|
|
|
.name = "udp",
|
|
|
|
.owner = THIS_MODULE,
|
2009-09-10 21:33:30 +07:00
|
|
|
.ident = XPRT_TRANSPORT_UDP,
|
2007-09-11 00:46:39 +07:00
|
|
|
.setup = xs_setup_udp,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct xprt_class xs_tcp_transport = {
|
|
|
|
.list = LIST_HEAD_INIT(xs_tcp_transport.list),
|
|
|
|
.name = "tcp",
|
|
|
|
.owner = THIS_MODULE,
|
2009-09-10 21:33:30 +07:00
|
|
|
.ident = XPRT_TRANSPORT_TCP,
|
2007-09-11 00:46:39 +07:00
|
|
|
.setup = xs_setup_tcp,
|
|
|
|
};
|
|
|
|
|
2009-09-10 21:33:30 +07:00
|
|
|
static struct xprt_class xs_bc_tcp_transport = {
|
|
|
|
.list = LIST_HEAD_INIT(xs_bc_tcp_transport.list),
|
|
|
|
.name = "tcp NFSv4.1 backchannel",
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.ident = XPRT_TRANSPORT_BC_TCP,
|
|
|
|
.setup = xs_setup_bc_tcp,
|
|
|
|
};
|
|
|
|
|
2006-12-06 04:35:51 +07:00
|
|
|
/**
|
2007-09-11 00:46:39 +07:00
|
|
|
* init_socket_xprt - set up xprtsock's sysctls, register with RPC client
|
2006-12-06 04:35:51 +07:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
int init_socket_xprt(void)
|
|
|
|
{
|
2014-11-18 04:58:04 +07:00
|
|
|
#if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
|
2007-02-14 15:33:24 +07:00
|
|
|
if (!sunrpc_table_header)
|
2007-02-14 15:34:09 +07:00
|
|
|
sunrpc_table_header = register_sysctl_table(sunrpc_table);
|
2006-12-06 04:35:54 +07:00
|
|
|
#endif
|
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
xprt_register_transport(&xs_local_transport);
|
2007-09-11 00:46:39 +07:00
|
|
|
xprt_register_transport(&xs_udp_transport);
|
|
|
|
xprt_register_transport(&xs_tcp_transport);
|
2009-09-10 21:33:30 +07:00
|
|
|
xprt_register_transport(&xs_bc_tcp_transport);
|
2007-09-11 00:46:39 +07:00
|
|
|
|
2006-12-06 04:35:51 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2007-09-11 00:46:39 +07:00
|
|
|
* cleanup_socket_xprt - remove xprtsock's sysctls, unregister
|
2006-12-06 04:35:51 +07:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
void cleanup_socket_xprt(void)
|
|
|
|
{
|
2014-11-18 04:58:04 +07:00
|
|
|
#if IS_ENABLED(CONFIG_SUNRPC_DEBUG)
|
2006-12-06 04:35:54 +07:00
|
|
|
if (sunrpc_table_header) {
|
|
|
|
unregister_sysctl_table(sunrpc_table_header);
|
|
|
|
sunrpc_table_header = NULL;
|
|
|
|
}
|
|
|
|
#endif
|
2007-09-11 00:46:39 +07:00
|
|
|
|
2011-05-10 02:22:44 +07:00
|
|
|
xprt_unregister_transport(&xs_local_transport);
|
2007-09-11 00:46:39 +07:00
|
|
|
xprt_unregister_transport(&xs_udp_transport);
|
|
|
|
xprt_unregister_transport(&xs_tcp_transport);
|
2009-09-10 21:33:30 +07:00
|
|
|
xprt_unregister_transport(&xs_bc_tcp_transport);
|
2006-12-06 04:35:51 +07:00
|
|
|
}
|
2009-08-10 02:06:19 +07:00
|
|
|
|
2010-08-12 12:04:12 +07:00
|
|
|
static int param_set_uint_minmax(const char *val,
|
|
|
|
const struct kernel_param *kp,
|
2009-08-10 02:06:19 +07:00
|
|
|
unsigned int min, unsigned int max)
|
|
|
|
{
|
2014-06-21 19:06:38 +07:00
|
|
|
unsigned int num;
|
2009-08-10 02:06:19 +07:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!val)
|
|
|
|
return -EINVAL;
|
2014-06-21 19:06:38 +07:00
|
|
|
ret = kstrtouint(val, 0, &num);
|
2009-08-10 02:06:19 +07:00
|
|
|
if (ret == -EINVAL || num < min || num > max)
|
|
|
|
return -EINVAL;
|
|
|
|
*((unsigned int *)kp->arg) = num;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-08-12 12:04:12 +07:00
|
|
|
static int param_set_portnr(const char *val, const struct kernel_param *kp)
|
2009-08-10 02:06:19 +07:00
|
|
|
{
|
|
|
|
return param_set_uint_minmax(val, kp,
|
|
|
|
RPC_MIN_RESVPORT,
|
|
|
|
RPC_MAX_RESVPORT);
|
|
|
|
}
|
|
|
|
|
2015-05-27 08:39:38 +07:00
|
|
|
static const struct kernel_param_ops param_ops_portnr = {
|
2010-08-12 12:04:12 +07:00
|
|
|
.set = param_set_portnr,
|
|
|
|
.get = param_get_uint,
|
|
|
|
};
|
|
|
|
|
2009-08-10 02:06:19 +07:00
|
|
|
#define param_check_portnr(name, p) \
|
|
|
|
__param_check(name, p, unsigned int);
|
|
|
|
|
|
|
|
module_param_named(min_resvport, xprt_min_resvport, portnr, 0644);
|
|
|
|
module_param_named(max_resvport, xprt_max_resvport, portnr, 0644);
|
|
|
|
|
2010-08-12 12:04:12 +07:00
|
|
|
static int param_set_slot_table_size(const char *val,
|
|
|
|
const struct kernel_param *kp)
|
2009-08-10 02:06:19 +07:00
|
|
|
{
|
|
|
|
return param_set_uint_minmax(val, kp,
|
|
|
|
RPC_MIN_SLOT_TABLE,
|
|
|
|
RPC_MAX_SLOT_TABLE);
|
|
|
|
}
|
|
|
|
|
2015-05-27 08:39:38 +07:00
|
|
|
static const struct kernel_param_ops param_ops_slot_table_size = {
|
2010-08-12 12:04:12 +07:00
|
|
|
.set = param_set_slot_table_size,
|
|
|
|
.get = param_get_uint,
|
|
|
|
};
|
|
|
|
|
2009-08-10 02:06:19 +07:00
|
|
|
#define param_check_slot_table_size(name, p) \
|
|
|
|
__param_check(name, p, unsigned int);
|
|
|
|
|
2011-07-18 05:11:30 +07:00
|
|
|
static int param_set_max_slot_table_size(const char *val,
|
|
|
|
const struct kernel_param *kp)
|
|
|
|
{
|
|
|
|
return param_set_uint_minmax(val, kp,
|
|
|
|
RPC_MIN_SLOT_TABLE,
|
|
|
|
RPC_MAX_SLOT_TABLE_LIMIT);
|
|
|
|
}
|
|
|
|
|
2015-05-27 08:39:38 +07:00
|
|
|
static const struct kernel_param_ops param_ops_max_slot_table_size = {
|
2011-07-18 05:11:30 +07:00
|
|
|
.set = param_set_max_slot_table_size,
|
|
|
|
.get = param_get_uint,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define param_check_max_slot_table_size(name, p) \
|
|
|
|
__param_check(name, p, unsigned int);
|
|
|
|
|
2009-08-10 02:06:19 +07:00
|
|
|
module_param_named(tcp_slot_table_entries, xprt_tcp_slot_table_entries,
|
|
|
|
slot_table_size, 0644);
|
2011-07-18 05:11:30 +07:00
|
|
|
module_param_named(tcp_max_slot_table_entries, xprt_max_tcp_slot_table_entries,
|
|
|
|
max_slot_table_size, 0644);
|
2009-08-10 02:06:19 +07:00
|
|
|
module_param_named(udp_slot_table_entries, xprt_udp_slot_table_entries,
|
|
|
|
slot_table_size, 0644);
|
|
|
|
|