diff options
author | J. Bruce Fields <bfields@redhat.com> | 2013-02-20 17:52:19 -0500 |
---|---|---|
committer | J. Bruce Fields <bfields@redhat.com> | 2013-02-28 12:47:17 -0500 |
commit | dc107402ae06286a9ed33c32daf3f35514a7cb8d (patch) | |
tree | 771882a1a62e2c56ae2c262764690417cecd776a /net | |
parent | 56edc86b5a72bdbc86358e57fb09165136baf0b8 (diff) |
SUNRPC: make AF_LOCAL connect synchronous
It doesn't appear that anyone actually needs to connect asynchronously.
Also, using a workqueue for the connect means we lose the namespace
information from the original process. This is a problem since there's
no way to explicitly pass in a filesystem namespace for resolution of an
AF_LOCAL address.
Acked-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
Diffstat (limited to 'net')
-rw-r--r-- | net/sunrpc/xprtsock.c | 35 |
1 files changed, 27 insertions, 8 deletions
diff --git a/net/sunrpc/xprtsock.c b/net/sunrpc/xprtsock.c index bbc09154df8d..4dc8eb24ac2c 100644 --- a/net/sunrpc/xprtsock.c +++ b/net/sunrpc/xprtsock.c | |||
@@ -1866,13 +1866,9 @@ static int xs_local_finish_connecting(struct rpc_xprt *xprt, | |||
1866 | * @xprt: RPC transport to connect | 1866 | * @xprt: RPC transport to connect |
1867 | * @transport: socket transport to connect | 1867 | * @transport: socket transport to connect |
1868 | * @create_sock: function to create a socket of the correct type | 1868 | * @create_sock: function to create a socket of the correct type |
1869 | * | ||
1870 | * Invoked by a work queue tasklet. | ||
1871 | */ | 1869 | */ |
1872 | static void xs_local_setup_socket(struct work_struct *work) | 1870 | static int xs_local_setup_socket(struct sock_xprt *transport) |
1873 | { | 1871 | { |
1874 | struct sock_xprt *transport = | ||
1875 | container_of(work, struct sock_xprt, connect_worker.work); | ||
1876 | struct rpc_xprt *xprt = &transport->xprt; | 1872 | struct rpc_xprt *xprt = &transport->xprt; |
1877 | struct socket *sock; | 1873 | struct socket *sock; |
1878 | int status = -EIO; | 1874 | int status = -EIO; |
@@ -1917,6 +1913,31 @@ out: | |||
1917 | xprt_clear_connecting(xprt); | 1913 | xprt_clear_connecting(xprt); |
1918 | xprt_wake_pending_tasks(xprt, status); | 1914 | xprt_wake_pending_tasks(xprt, status); |
1919 | current->flags &= ~PF_FSTRANS; | 1915 | current->flags &= ~PF_FSTRANS; |
1916 | return status; | ||
1917 | } | ||
1918 | |||
1919 | static void xs_local_connect(struct rpc_task *task) | ||
1920 | { | ||
1921 | struct rpc_xprt *xprt = task->tk_xprt; | ||
1922 | struct sock_xprt *transport = container_of(xprt, struct sock_xprt, xprt); | ||
1923 | int ret; | ||
1924 | |||
1925 | if (RPC_IS_ASYNC(task)) { | ||
1926 | /* | ||
1927 | * We want the AF_LOCAL connect to be resolved in the | ||
1928 | * filesystem namespace of the process making the rpc | ||
1929 | * call. Thus we connect synchronously. | ||
1930 | * | ||
1931 | * If we want to support asynchronous AF_LOCAL calls, | ||
1932 | * we'll need to figure out how to pass a namespace to | ||
1933 | * connect. | ||
1934 | */ | ||
1935 | rpc_exit(task, -ENOTCONN); | ||
1936 | return; | ||
1937 | } | ||
1938 | ret = xs_local_setup_socket(transport); | ||
1939 | if (ret && !RPC_IS_SOFTCONN(task)) | ||
1940 | msleep_interruptible(15000); | ||
1920 | } | 1941 | } |
1921 | 1942 | ||
1922 | #ifdef CONFIG_SUNRPC_SWAP | 1943 | #ifdef CONFIG_SUNRPC_SWAP |
@@ -2454,7 +2475,7 @@ static struct rpc_xprt_ops xs_local_ops = { | |||
2454 | .alloc_slot = xprt_alloc_slot, | 2475 | .alloc_slot = xprt_alloc_slot, |
2455 | .rpcbind = xs_local_rpcbind, | 2476 | .rpcbind = xs_local_rpcbind, |
2456 | .set_port = xs_local_set_port, | 2477 | .set_port = xs_local_set_port, |
2457 | .connect = xs_connect, | 2478 | .connect = xs_local_connect, |
2458 | .buf_alloc = rpc_malloc, | 2479 | .buf_alloc = rpc_malloc, |
2459 | .buf_free = rpc_free, | 2480 | .buf_free = rpc_free, |
2460 | .send_request = xs_local_send_request, | 2481 | .send_request = xs_local_send_request, |
@@ -2627,8 +2648,6 @@ static struct rpc_xprt *xs_setup_local(struct xprt_create *args) | |||
2627 | goto out_err; | 2648 | goto out_err; |
2628 | } | 2649 | } |
2629 | xprt_set_bound(xprt); | 2650 | xprt_set_bound(xprt); |
2630 | INIT_DELAYED_WORK(&transport->connect_worker, | ||
2631 | xs_local_setup_socket); | ||
2632 | xs_format_peer_addresses(xprt, "local", RPCBIND_NETID_LOCAL); | 2651 | xs_format_peer_addresses(xprt, "local", RPCBIND_NETID_LOCAL); |
2633 | break; | 2652 | break; |
2634 | default: | 2653 | default: |