2019-05-19 15:08:20 +03:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2017-01-09 18:55:13 +03:00
|
|
|
/*
|
|
|
|
* Shared Memory Communications over RDMA (SMC-R) and RoCE
|
|
|
|
*
|
|
|
|
* AF_SMC protocol family socket handler keeping the AF_INET sock address type
|
|
|
|
* applies to SOCK_STREAM sockets only
|
|
|
|
* offers an alternative communication option for TCP-protocol sockets
|
|
|
|
* applicable with RoCE-cards only
|
|
|
|
*
|
2017-01-09 18:55:16 +03:00
|
|
|
* Initial restrictions:
|
|
|
|
* - support for alternate links postponed
|
|
|
|
*
|
2018-03-16 17:06:41 +03:00
|
|
|
* Copyright IBM Corp. 2016, 2018
|
2017-01-09 18:55:13 +03:00
|
|
|
*
|
|
|
|
* Author(s): Ursula Braun <ubraun@linux.vnet.ibm.com>
|
|
|
|
* based on prototype from Frank Blaschka
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define KMSG_COMPONENT "smc"
|
|
|
|
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/socket.h>
|
2017-01-09 18:55:16 +03:00
|
|
|
#include <linux/workqueue.h>
|
2017-01-09 18:55:22 +03:00
|
|
|
#include <linux/in.h>
|
2017-02-02 10:35:14 +03:00
|
|
|
#include <linux/sched/signal.h>
|
2018-06-28 20:05:11 +03:00
|
|
|
#include <linux/if_vlan.h>
|
2019-11-16 19:47:31 +03:00
|
|
|
#include <linux/rcupdate_wait.h>
|
2020-09-26 13:44:31 +03:00
|
|
|
#include <linux/ctype.h>
|
2017-02-02 10:35:14 +03:00
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
#include <net/sock.h>
|
2017-01-09 18:55:16 +03:00
|
|
|
#include <net/tcp.h>
|
2017-01-09 18:55:26 +03:00
|
|
|
#include <net/smc.h>
|
2018-05-02 17:56:46 +03:00
|
|
|
#include <asm/ioctls.h>
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2019-02-21 15:01:02 +03:00
|
|
|
#include <net/net_namespace.h>
|
|
|
|
#include <net/netns/generic.h>
|
|
|
|
#include "smc_netns.h"
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
#include "smc.h"
|
2017-01-09 18:55:16 +03:00
|
|
|
#include "smc_clc.h"
|
2017-01-09 18:55:21 +03:00
|
|
|
#include "smc_llc.h"
|
2017-01-09 18:55:22 +03:00
|
|
|
#include "smc_cdc.h"
|
2017-01-09 18:55:17 +03:00
|
|
|
#include "smc_core.h"
|
2017-01-09 18:55:14 +03:00
|
|
|
#include "smc_ib.h"
|
2018-06-28 20:05:11 +03:00
|
|
|
#include "smc_ism.h"
|
2017-01-09 18:55:15 +03:00
|
|
|
#include "smc_pnet.h"
|
2020-12-01 22:20:43 +03:00
|
|
|
#include "smc_netlink.h"
|
2017-01-09 18:55:23 +03:00
|
|
|
#include "smc_tx.h"
|
2017-01-09 18:55:24 +03:00
|
|
|
#include "smc_rx.h"
|
2017-01-09 18:55:25 +03:00
|
|
|
#include "smc_close.h"
|
2021-06-16 17:52:55 +03:00
|
|
|
#include "smc_stats.h"
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2019-02-07 17:56:18 +03:00
|
|
|
static DEFINE_MUTEX(smc_server_lgr_pending); /* serialize link group
|
|
|
|
* creation on server
|
|
|
|
*/
|
|
|
|
static DEFINE_MUTEX(smc_client_lgr_pending); /* serialize link group
|
|
|
|
* creation on client
|
2017-01-09 18:55:17 +03:00
|
|
|
*/
|
|
|
|
|
2020-09-10 19:48:29 +03:00
|
|
|
struct workqueue_struct *smc_hs_wq; /* wq for handshake work */
|
|
|
|
struct workqueue_struct *smc_close_wq; /* wq for close work */
|
|
|
|
|
2017-01-09 18:55:16 +03:00
|
|
|
static void smc_tcp_listen_work(struct work_struct *);
|
2018-06-27 18:59:50 +03:00
|
|
|
static void smc_connect_work(struct work_struct *);
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
static void smc_set_keepalive(struct sock *sk, int val)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc = smc_sk(sk);
|
|
|
|
|
|
|
|
smc->clcsock->sk->sk_prot->keepalive(smc->clcsock->sk, val);
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:26 +03:00
|
|
|
static struct smc_hashinfo smc_v4_hashinfo = {
|
|
|
|
.lock = __RW_LOCK_UNLOCKED(smc_v4_hashinfo.lock),
|
|
|
|
};
|
|
|
|
|
2018-03-16 17:06:41 +03:00
|
|
|
static struct smc_hashinfo smc_v6_hashinfo = {
|
|
|
|
.lock = __RW_LOCK_UNLOCKED(smc_v6_hashinfo.lock),
|
|
|
|
};
|
|
|
|
|
2017-01-09 18:55:26 +03:00
|
|
|
int smc_hash_sk(struct sock *sk)
|
|
|
|
{
|
|
|
|
struct smc_hashinfo *h = sk->sk_prot->h.smc_hash;
|
|
|
|
struct hlist_head *head;
|
|
|
|
|
|
|
|
head = &h->ht;
|
|
|
|
|
|
|
|
write_lock_bh(&h->lock);
|
|
|
|
sk_add_node(sk, head);
|
|
|
|
sock_prot_inuse_add(sock_net(sk), sk->sk_prot, 1);
|
|
|
|
write_unlock_bh(&h->lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(smc_hash_sk);
|
|
|
|
|
|
|
|
void smc_unhash_sk(struct sock *sk)
|
|
|
|
{
|
|
|
|
struct smc_hashinfo *h = sk->sk_prot->h.smc_hash;
|
|
|
|
|
|
|
|
write_lock_bh(&h->lock);
|
|
|
|
if (sk_del_node_init(sk))
|
|
|
|
sock_prot_inuse_add(sock_net(sk), sk->sk_prot, -1);
|
|
|
|
write_unlock_bh(&h->lock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(smc_unhash_sk);
|
|
|
|
|
|
|
|
struct proto smc_proto = {
|
2017-01-09 18:55:13 +03:00
|
|
|
.name = "SMC",
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.keepalive = smc_set_keepalive,
|
2017-01-09 18:55:26 +03:00
|
|
|
.hash = smc_hash_sk,
|
|
|
|
.unhash = smc_unhash_sk,
|
2017-01-09 18:55:13 +03:00
|
|
|
.obj_size = sizeof(struct smc_sock),
|
2017-01-09 18:55:26 +03:00
|
|
|
.h.smc_hash = &smc_v4_hashinfo,
|
2017-01-18 13:53:44 +03:00
|
|
|
.slab_flags = SLAB_TYPESAFE_BY_RCU,
|
2017-01-09 18:55:13 +03:00
|
|
|
};
|
2017-01-09 18:55:26 +03:00
|
|
|
EXPORT_SYMBOL_GPL(smc_proto);
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2018-03-16 17:06:41 +03:00
|
|
|
struct proto smc_proto6 = {
|
|
|
|
.name = "SMC6",
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.keepalive = smc_set_keepalive,
|
|
|
|
.hash = smc_hash_sk,
|
|
|
|
.unhash = smc_unhash_sk,
|
|
|
|
.obj_size = sizeof(struct smc_sock),
|
|
|
|
.h.smc_hash = &smc_v6_hashinfo,
|
|
|
|
.slab_flags = SLAB_TYPESAFE_BY_RCU,
|
|
|
|
};
|
|
|
|
EXPORT_SYMBOL_GPL(smc_proto6);
|
|
|
|
|
2019-10-23 16:44:05 +03:00
|
|
|
static void smc_restore_fallback_changes(struct smc_sock *smc)
|
|
|
|
{
|
2020-07-18 16:06:18 +03:00
|
|
|
if (smc->clcsock->file) { /* non-accepted sockets have no file yet */
|
|
|
|
smc->clcsock->file->private_data = smc->sk.sk_socket;
|
|
|
|
smc->clcsock->file = NULL;
|
|
|
|
}
|
2019-10-23 16:44:05 +03:00
|
|
|
}
|
|
|
|
|
2019-06-27 16:04:52 +03:00
|
|
|
static int __smc_release(struct smc_sock *smc)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
2019-06-27 16:04:52 +03:00
|
|
|
struct sock *sk = &smc->sk;
|
2017-01-09 18:55:25 +03:00
|
|
|
int rc = 0;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2018-01-26 11:28:48 +03:00
|
|
|
if (!smc->use_fallback) {
|
2017-01-09 18:55:25 +03:00
|
|
|
rc = smc_close_active(smc);
|
|
|
|
sock_set_flag(sk, SOCK_DEAD);
|
|
|
|
sk->sk_shutdown |= SHUTDOWN_MASK;
|
2019-02-07 17:56:15 +03:00
|
|
|
} else {
|
2021-11-10 10:02:34 +03:00
|
|
|
if (sk->sk_state != SMC_CLOSED) {
|
|
|
|
if (sk->sk_state != SMC_LISTEN &&
|
|
|
|
sk->sk_state != SMC_INIT)
|
|
|
|
sock_put(sk); /* passive closing */
|
|
|
|
if (sk->sk_state == SMC_LISTEN) {
|
|
|
|
/* wake up clcsock accept */
|
|
|
|
rc = kernel_sock_shutdown(smc->clcsock,
|
|
|
|
SHUT_RDWR);
|
|
|
|
}
|
|
|
|
sk->sk_state = SMC_CLOSED;
|
|
|
|
sk->sk_state_change(sk);
|
2018-12-18 20:02:25 +03:00
|
|
|
}
|
2019-10-23 16:44:05 +03:00
|
|
|
smc_restore_fallback_changes(smc);
|
2018-01-26 11:28:48 +03:00
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2019-02-07 17:56:15 +03:00
|
|
|
sk->sk_prot->unhash(sk);
|
|
|
|
|
|
|
|
if (sk->sk_state == SMC_CLOSED) {
|
|
|
|
if (smc->clcsock) {
|
2019-04-11 12:17:30 +03:00
|
|
|
release_sock(sk);
|
|
|
|
smc_clcsock_release(smc);
|
|
|
|
lock_sock(sk);
|
2019-02-07 17:56:15 +03:00
|
|
|
}
|
|
|
|
if (!smc->use_fallback)
|
|
|
|
smc_conn_free(&smc->conn);
|
|
|
|
}
|
|
|
|
|
2019-06-27 16:04:52 +03:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_release(struct socket *sock)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
net/smc: fix connection leak
commit 9f1c50cf39167ff71dc5953a3234f3f6eeb8fcb5 upstream.
There's a potential leak issue under following execution sequence :
smc_release smc_connect_work
if (sk->sk_state == SMC_INIT)
send_clc_confirim
tcp_abort();
...
sk.sk_state = SMC_ACTIVE
smc_close_active
switch(sk->sk_state) {
...
case SMC_ACTIVE:
smc_close_final()
// then wait peer closed
Unfortunately, tcp_abort() may discard CLC CONFIRM messages that are
still in the tcp send buffer, in which case our connection token cannot
be delivered to the server side, which means that we cannot get a
passive close message at all. Therefore, it is impossible for the to be
disconnected at all.
This patch tries a very simple way to avoid this issue, once the state
has changed to SMC_ACTIVE after tcp_abort(), we can actively abort the
smc connection, considering that the state is SMC_INIT before
tcp_abort(), abandoning the complete disconnection process should not
cause too much problem.
In fact, this problem may exist as long as the CLC CONFIRM message is
not received by the server. Whether a timer should be added after
smc_close_final() needs to be discussed in the future. But even so, this
patch provides a faster release for connection in above case, it should
also be valuable.
Fixes: 39f41f367b08 ("net/smc: common release code for non-accepted sockets")
Signed-off-by: D. Wythe <alibuda@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-02-24 18:26:19 +03:00
|
|
|
int old_state, rc = 0;
|
2019-06-27 16:04:52 +03:00
|
|
|
|
|
|
|
if (!sk)
|
|
|
|
goto out;
|
|
|
|
|
2019-10-21 17:13:15 +03:00
|
|
|
sock_hold(sk); /* sock_put below */
|
2019-06-27 16:04:52 +03:00
|
|
|
smc = smc_sk(sk);
|
|
|
|
|
net/smc: fix connection leak
commit 9f1c50cf39167ff71dc5953a3234f3f6eeb8fcb5 upstream.
There's a potential leak issue under following execution sequence :
smc_release smc_connect_work
if (sk->sk_state == SMC_INIT)
send_clc_confirim
tcp_abort();
...
sk.sk_state = SMC_ACTIVE
smc_close_active
switch(sk->sk_state) {
...
case SMC_ACTIVE:
smc_close_final()
// then wait peer closed
Unfortunately, tcp_abort() may discard CLC CONFIRM messages that are
still in the tcp send buffer, in which case our connection token cannot
be delivered to the server side, which means that we cannot get a
passive close message at all. Therefore, it is impossible for the to be
disconnected at all.
This patch tries a very simple way to avoid this issue, once the state
has changed to SMC_ACTIVE after tcp_abort(), we can actively abort the
smc connection, considering that the state is SMC_INIT before
tcp_abort(), abandoning the complete disconnection process should not
cause too much problem.
In fact, this problem may exist as long as the CLC CONFIRM message is
not received by the server. Whether a timer should be added after
smc_close_final() needs to be discussed in the future. But even so, this
patch provides a faster release for connection in above case, it should
also be valuable.
Fixes: 39f41f367b08 ("net/smc: common release code for non-accepted sockets")
Signed-off-by: D. Wythe <alibuda@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-02-24 18:26:19 +03:00
|
|
|
old_state = sk->sk_state;
|
|
|
|
|
2019-06-27 16:04:52 +03:00
|
|
|
/* cleanup for a dangling non-blocking connect */
|
net/smc: fix connection leak
commit 9f1c50cf39167ff71dc5953a3234f3f6eeb8fcb5 upstream.
There's a potential leak issue under following execution sequence :
smc_release smc_connect_work
if (sk->sk_state == SMC_INIT)
send_clc_confirim
tcp_abort();
...
sk.sk_state = SMC_ACTIVE
smc_close_active
switch(sk->sk_state) {
...
case SMC_ACTIVE:
smc_close_final()
// then wait peer closed
Unfortunately, tcp_abort() may discard CLC CONFIRM messages that are
still in the tcp send buffer, in which case our connection token cannot
be delivered to the server side, which means that we cannot get a
passive close message at all. Therefore, it is impossible for the to be
disconnected at all.
This patch tries a very simple way to avoid this issue, once the state
has changed to SMC_ACTIVE after tcp_abort(), we can actively abort the
smc connection, considering that the state is SMC_INIT before
tcp_abort(), abandoning the complete disconnection process should not
cause too much problem.
In fact, this problem may exist as long as the CLC CONFIRM message is
not received by the server. Whether a timer should be added after
smc_close_final() needs to be discussed in the future. But even so, this
patch provides a faster release for connection in above case, it should
also be valuable.
Fixes: 39f41f367b08 ("net/smc: common release code for non-accepted sockets")
Signed-off-by: D. Wythe <alibuda@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-02-24 18:26:19 +03:00
|
|
|
if (smc->connect_nonblock && old_state == SMC_INIT)
|
2019-06-27 16:04:52 +03:00
|
|
|
tcp_abort(smc->clcsock->sk, ECONNABORTED);
|
net/smc: Prevent smc_release() from long blocking
[ Upstream commit 5c15b3123f65f8fbb1b445d9a7e8812e0e435df2 ]
In nginx/wrk benchmark, there's a hung problem with high probability
on case likes that: (client will last several minutes to exit)
server: smc_run nginx
client: smc_run wrk -c 10000 -t 1 http://server
Client hangs with the following backtrace:
0 [ffffa7ce8Of3bbf8] __schedule at ffffffff9f9eOd5f
1 [ffffa7ce8Of3bc88] schedule at ffffffff9f9eløe6
2 [ffffa7ce8Of3bcaO] schedule_timeout at ffffffff9f9e3f3c
3 [ffffa7ce8Of3bd2O] wait_for_common at ffffffff9f9el9de
4 [ffffa7ce8Of3bd8O] __flush_work at ffffffff9fOfeOl3
5 [ffffa7ce8øf3bdfO] smc_release at ffffffffcO697d24 [smc]
6 [ffffa7ce8Of3be2O] __sock_release at ffffffff9f8O2e2d
7 [ffffa7ce8Of3be4ø] sock_close at ffffffff9f8ø2ebl
8 [ffffa7ce8øf3be48] __fput at ffffffff9f334f93
9 [ffffa7ce8Of3be78] task_work_run at ffffffff9flOlff5
10 [ffffa7ce8Of3beaO] do_exit at ffffffff9fOe5Ol2
11 [ffffa7ce8Of3bflO] do_group_exit at ffffffff9fOe592a
12 [ffffa7ce8Of3bf38] __x64_sys_exit_group at ffffffff9fOe5994
13 [ffffa7ce8Of3bf4O] do_syscall_64 at ffffffff9f9d4373
14 [ffffa7ce8Of3bfsO] entry_SYSCALL_64_after_hwframe at ffffffff9fa0007c
This issue dues to flush_work(), which is used to wait for
smc_connect_work() to finish in smc_release(). Once lots of
smc_connect_work() was pending or all executing work dangling,
smc_release() has to block until one worker comes to free, which
is equivalent to wait another smc_connnect_work() to finish.
In order to fix this, There are two changes:
1. For those idle smc_connect_work(), cancel it from the workqueue; for
executing smc_connect_work(), waiting for it to finish. For that
purpose, replace flush_work() with cancel_work_sync().
2. Since smc_connect() hold a reference for passive closing, if
smc_connect_work() has been cancelled, release the reference.
Fixes: 24ac3a08e658 ("net/smc: rebuild nonblocking connect")
Reported-by: Tony Lu <tonylu@linux.alibaba.com>
Tested-by: Dust Li <dust.li@linux.alibaba.com>
Reviewed-by: Dust Li <dust.li@linux.alibaba.com>
Reviewed-by: Tony Lu <tonylu@linux.alibaba.com>
Signed-off-by: D. Wythe <alibuda@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Link: https://lore.kernel.org/r/1639571361-101128-1-git-send-email-alibuda@linux.alibaba.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
2021-12-15 15:29:21 +03:00
|
|
|
|
|
|
|
if (cancel_work_sync(&smc->connect_work))
|
|
|
|
sock_put(&smc->sk); /* sock_hold in smc_connect for passive closing */
|
2019-06-27 16:04:52 +03:00
|
|
|
|
|
|
|
if (sk->sk_state == SMC_LISTEN)
|
|
|
|
/* smc_close_non_accepted() is called and acquires
|
|
|
|
* sock lock for child sockets again
|
|
|
|
*/
|
|
|
|
lock_sock_nested(sk, SINGLE_DEPTH_NESTING);
|
|
|
|
else
|
|
|
|
lock_sock(sk);
|
|
|
|
|
net/smc: fix connection leak
commit 9f1c50cf39167ff71dc5953a3234f3f6eeb8fcb5 upstream.
There's a potential leak issue under following execution sequence :
smc_release smc_connect_work
if (sk->sk_state == SMC_INIT)
send_clc_confirim
tcp_abort();
...
sk.sk_state = SMC_ACTIVE
smc_close_active
switch(sk->sk_state) {
...
case SMC_ACTIVE:
smc_close_final()
// then wait peer closed
Unfortunately, tcp_abort() may discard CLC CONFIRM messages that are
still in the tcp send buffer, in which case our connection token cannot
be delivered to the server side, which means that we cannot get a
passive close message at all. Therefore, it is impossible for the to be
disconnected at all.
This patch tries a very simple way to avoid this issue, once the state
has changed to SMC_ACTIVE after tcp_abort(), we can actively abort the
smc connection, considering that the state is SMC_INIT before
tcp_abort(), abandoning the complete disconnection process should not
cause too much problem.
In fact, this problem may exist as long as the CLC CONFIRM message is
not received by the server. Whether a timer should be added after
smc_close_final() needs to be discussed in the future. But even so, this
patch provides a faster release for connection in above case, it should
also be valuable.
Fixes: 39f41f367b08 ("net/smc: common release code for non-accepted sockets")
Signed-off-by: D. Wythe <alibuda@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-02-24 18:26:19 +03:00
|
|
|
if (old_state == SMC_INIT && sk->sk_state == SMC_ACTIVE &&
|
|
|
|
!smc->use_fallback)
|
|
|
|
smc_close_active_abort(smc);
|
|
|
|
|
2019-06-27 16:04:52 +03:00
|
|
|
rc = __smc_release(smc);
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
/* detach socket */
|
|
|
|
sock_orphan(sk);
|
|
|
|
sock->sk = NULL;
|
|
|
|
release_sock(sk);
|
|
|
|
|
2019-10-21 17:13:15 +03:00
|
|
|
sock_put(sk); /* sock_hold above */
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(sk); /* final sock_put */
|
2017-01-09 18:55:13 +03:00
|
|
|
out:
|
2017-01-09 18:55:25 +03:00
|
|
|
return rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_destruct(struct sock *sk)
|
|
|
|
{
|
|
|
|
if (sk->sk_state != SMC_CLOSED)
|
|
|
|
return;
|
|
|
|
if (!sock_flag(sk, SOCK_DEAD))
|
|
|
|
return;
|
|
|
|
|
|
|
|
sk_refcnt_debug_dec(sk);
|
|
|
|
}
|
|
|
|
|
2018-03-16 17:06:41 +03:00
|
|
|
static struct sock *smc_sock_alloc(struct net *net, struct socket *sock,
|
|
|
|
int protocol)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
|
|
|
struct smc_sock *smc;
|
2018-03-16 17:06:41 +03:00
|
|
|
struct proto *prot;
|
2017-01-09 18:55:13 +03:00
|
|
|
struct sock *sk;
|
|
|
|
|
2018-03-16 17:06:41 +03:00
|
|
|
prot = (protocol == SMCPROTO_SMC6) ? &smc_proto6 : &smc_proto;
|
|
|
|
sk = sk_alloc(net, PF_SMC, GFP_KERNEL, prot, 0);
|
2017-01-09 18:55:13 +03:00
|
|
|
if (!sk)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
sock_init_data(sock, sk); /* sets sk_refcnt to 1 */
|
|
|
|
sk->sk_state = SMC_INIT;
|
|
|
|
sk->sk_destruct = smc_destruct;
|
2018-03-16 17:06:41 +03:00
|
|
|
sk->sk_protocol = protocol;
|
2017-01-09 18:55:13 +03:00
|
|
|
smc = smc_sk(sk);
|
2017-01-09 18:55:16 +03:00
|
|
|
INIT_WORK(&smc->tcp_listen_work, smc_tcp_listen_work);
|
2018-06-27 18:59:50 +03:00
|
|
|
INIT_WORK(&smc->connect_work, smc_connect_work);
|
net/smc: init conn.tx_work & conn.send_lock sooner
syzkaller found that following program crashes the host :
{
int fd = socket(AF_SMC, SOCK_STREAM, 0);
int val = 1;
listen(fd, 0);
shutdown(fd, SHUT_RDWR);
setsockopt(fd, 6, TCP_NODELAY, &val, 4);
}
Simply initialize conn.tx_work & conn.send_lock at socket creation,
rather than deeper in the stack.
ODEBUG: assert_init not available (active state 0) object type: timer_list hint: (null)
WARNING: CPU: 1 PID: 13988 at lib/debugobjects.c:329 debug_print_object+0x16a/0x210 lib/debugobjects.c:326
Kernel panic - not syncing: panic_on_warn set ...
CPU: 1 PID: 13988 Comm: syz-executor0 Not tainted 4.17.0-rc4+ #46
Hardware name: Google Google Compute Engine/Google Compute Engine, BIOS Google 01/01/2011
Call Trace:
__dump_stack lib/dump_stack.c:77 [inline]
dump_stack+0x1b9/0x294 lib/dump_stack.c:113
panic+0x22f/0x4de kernel/panic.c:184
__warn.cold.8+0x163/0x1b3 kernel/panic.c:536
report_bug+0x252/0x2d0 lib/bug.c:186
fixup_bug arch/x86/kernel/traps.c:178 [inline]
do_error_trap+0x1de/0x490 arch/x86/kernel/traps.c:296
do_invalid_op+0x1b/0x20 arch/x86/kernel/traps.c:315
invalid_op+0x14/0x20 arch/x86/entry/entry_64.S:992
RIP: 0010:debug_print_object+0x16a/0x210 lib/debugobjects.c:326
RSP: 0018:ffff880197a37880 EFLAGS: 00010086
RAX: 0000000000000061 RBX: 0000000000000005 RCX: ffffc90001ed0000
RDX: 0000000000004aaf RSI: ffffffff8160f6f1 RDI: 0000000000000001
RBP: ffff880197a378c0 R08: ffff8801aa7a0080 R09: ffffed003b5e3eb2
R10: ffffed003b5e3eb2 R11: ffff8801daf1f597 R12: 0000000000000001
R13: ffffffff88d96980 R14: ffffffff87fa19a0 R15: ffffffff81666ec0
debug_object_assert_init+0x309/0x500 lib/debugobjects.c:692
debug_timer_assert_init kernel/time/timer.c:724 [inline]
debug_assert_init kernel/time/timer.c:776 [inline]
del_timer+0x74/0x140 kernel/time/timer.c:1198
try_to_grab_pending+0x439/0x9a0 kernel/workqueue.c:1223
mod_delayed_work_on+0x91/0x250 kernel/workqueue.c:1592
mod_delayed_work include/linux/workqueue.h:541 [inline]
smc_setsockopt+0x387/0x6d0 net/smc/af_smc.c:1367
__sys_setsockopt+0x1bd/0x390 net/socket.c:1903
__do_sys_setsockopt net/socket.c:1914 [inline]
__se_sys_setsockopt net/socket.c:1911 [inline]
__x64_sys_setsockopt+0xbe/0x150 net/socket.c:1911
do_syscall_64+0x1b1/0x800 arch/x86/entry/common.c:287
entry_SYSCALL_64_after_hwframe+0x49/0xbe
Fixes: 01d2f7e2cdd3 ("net/smc: sockopts TCP_NODELAY and TCP_CORK")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Cc: Ursula Braun <ubraun@linux.ibm.com>
Cc: linux-s390@vger.kernel.org
Reported-by: syzbot <syzkaller@googlegroups.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2018-05-17 13:54:21 +03:00
|
|
|
INIT_DELAYED_WORK(&smc->conn.tx_work, smc_tx_work);
|
2017-01-09 18:55:16 +03:00
|
|
|
INIT_LIST_HEAD(&smc->accept_q);
|
|
|
|
spin_lock_init(&smc->accept_q_lock);
|
net/smc: init conn.tx_work & conn.send_lock sooner
syzkaller found that following program crashes the host :
{
int fd = socket(AF_SMC, SOCK_STREAM, 0);
int val = 1;
listen(fd, 0);
shutdown(fd, SHUT_RDWR);
setsockopt(fd, 6, TCP_NODELAY, &val, 4);
}
Simply initialize conn.tx_work & conn.send_lock at socket creation,
rather than deeper in the stack.
ODEBUG: assert_init not available (active state 0) object type: timer_list hint: (null)
WARNING: CPU: 1 PID: 13988 at lib/debugobjects.c:329 debug_print_object+0x16a/0x210 lib/debugobjects.c:326
Kernel panic - not syncing: panic_on_warn set ...
CPU: 1 PID: 13988 Comm: syz-executor0 Not tainted 4.17.0-rc4+ #46
Hardware name: Google Google Compute Engine/Google Compute Engine, BIOS Google 01/01/2011
Call Trace:
__dump_stack lib/dump_stack.c:77 [inline]
dump_stack+0x1b9/0x294 lib/dump_stack.c:113
panic+0x22f/0x4de kernel/panic.c:184
__warn.cold.8+0x163/0x1b3 kernel/panic.c:536
report_bug+0x252/0x2d0 lib/bug.c:186
fixup_bug arch/x86/kernel/traps.c:178 [inline]
do_error_trap+0x1de/0x490 arch/x86/kernel/traps.c:296
do_invalid_op+0x1b/0x20 arch/x86/kernel/traps.c:315
invalid_op+0x14/0x20 arch/x86/entry/entry_64.S:992
RIP: 0010:debug_print_object+0x16a/0x210 lib/debugobjects.c:326
RSP: 0018:ffff880197a37880 EFLAGS: 00010086
RAX: 0000000000000061 RBX: 0000000000000005 RCX: ffffc90001ed0000
RDX: 0000000000004aaf RSI: ffffffff8160f6f1 RDI: 0000000000000001
RBP: ffff880197a378c0 R08: ffff8801aa7a0080 R09: ffffed003b5e3eb2
R10: ffffed003b5e3eb2 R11: ffff8801daf1f597 R12: 0000000000000001
R13: ffffffff88d96980 R14: ffffffff87fa19a0 R15: ffffffff81666ec0
debug_object_assert_init+0x309/0x500 lib/debugobjects.c:692
debug_timer_assert_init kernel/time/timer.c:724 [inline]
debug_assert_init kernel/time/timer.c:776 [inline]
del_timer+0x74/0x140 kernel/time/timer.c:1198
try_to_grab_pending+0x439/0x9a0 kernel/workqueue.c:1223
mod_delayed_work_on+0x91/0x250 kernel/workqueue.c:1592
mod_delayed_work include/linux/workqueue.h:541 [inline]
smc_setsockopt+0x387/0x6d0 net/smc/af_smc.c:1367
__sys_setsockopt+0x1bd/0x390 net/socket.c:1903
__do_sys_setsockopt net/socket.c:1914 [inline]
__se_sys_setsockopt net/socket.c:1911 [inline]
__x64_sys_setsockopt+0xbe/0x150 net/socket.c:1911
do_syscall_64+0x1b1/0x800 arch/x86/entry/common.c:287
entry_SYSCALL_64_after_hwframe+0x49/0xbe
Fixes: 01d2f7e2cdd3 ("net/smc: sockopts TCP_NODELAY and TCP_CORK")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Cc: Ursula Braun <ubraun@linux.ibm.com>
Cc: linux-s390@vger.kernel.org
Reported-by: syzbot <syzkaller@googlegroups.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2018-05-17 13:54:21 +03:00
|
|
|
spin_lock_init(&smc->conn.send_lock);
|
2017-01-09 18:55:26 +03:00
|
|
|
sk->sk_prot->hash(sk);
|
2017-01-09 18:55:16 +03:00
|
|
|
sk_refcnt_debug_inc(sk);
|
2018-12-18 20:02:25 +03:00
|
|
|
mutex_init(&smc->clcsock_release_lock);
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
return sk;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_bind(struct socket *sock, struct sockaddr *uaddr,
|
|
|
|
int addr_len)
|
|
|
|
{
|
|
|
|
struct sockaddr_in *addr = (struct sockaddr_in *)uaddr;
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
|
|
|
|
/* replicate tests from inet_bind(), to be safe wrt. future changes */
|
|
|
|
rc = -EINVAL;
|
|
|
|
if (addr_len < sizeof(struct sockaddr_in))
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
rc = -EAFNOSUPPORT;
|
2018-03-16 17:06:41 +03:00
|
|
|
if (addr->sin_family != AF_INET &&
|
|
|
|
addr->sin_family != AF_INET6 &&
|
|
|
|
addr->sin_family != AF_UNSPEC)
|
|
|
|
goto out;
|
2017-01-09 18:55:13 +03:00
|
|
|
/* accept AF_UNSPEC (mapped to AF_INET) only if s_addr is INADDR_ANY */
|
2018-03-16 17:06:41 +03:00
|
|
|
if (addr->sin_family == AF_UNSPEC &&
|
|
|
|
addr->sin_addr.s_addr != htonl(INADDR_ANY))
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
lock_sock(sk);
|
|
|
|
|
|
|
|
/* Check if socket is already active */
|
|
|
|
rc = -EINVAL;
|
2019-08-02 11:47:50 +03:00
|
|
|
if (sk->sk_state != SMC_INIT || smc->connect_nonblock)
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out_rel;
|
|
|
|
|
|
|
|
smc->clcsock->sk->sk_reuse = sk->sk_reuse;
|
|
|
|
rc = kernel_bind(smc->clcsock, uaddr, addr_len);
|
|
|
|
|
|
|
|
out_rel:
|
|
|
|
release_sock(sk);
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_copy_sock_settings(struct sock *nsk, struct sock *osk,
|
|
|
|
unsigned long mask)
|
|
|
|
{
|
|
|
|
/* options we don't get control via setsockopt for */
|
|
|
|
nsk->sk_type = osk->sk_type;
|
|
|
|
nsk->sk_sndbuf = osk->sk_sndbuf;
|
|
|
|
nsk->sk_rcvbuf = osk->sk_rcvbuf;
|
|
|
|
nsk->sk_sndtimeo = osk->sk_sndtimeo;
|
|
|
|
nsk->sk_rcvtimeo = osk->sk_rcvtimeo;
|
|
|
|
nsk->sk_mark = osk->sk_mark;
|
|
|
|
nsk->sk_priority = osk->sk_priority;
|
|
|
|
nsk->sk_rcvlowat = osk->sk_rcvlowat;
|
|
|
|
nsk->sk_bound_dev_if = osk->sk_bound_dev_if;
|
|
|
|
nsk->sk_err = osk->sk_err;
|
|
|
|
|
|
|
|
nsk->sk_flags &= ~mask;
|
|
|
|
nsk->sk_flags |= osk->sk_flags & mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SK_FLAGS_SMC_TO_CLC ((1UL << SOCK_URGINLINE) | \
|
|
|
|
(1UL << SOCK_KEEPOPEN) | \
|
|
|
|
(1UL << SOCK_LINGER) | \
|
|
|
|
(1UL << SOCK_BROADCAST) | \
|
|
|
|
(1UL << SOCK_TIMESTAMP) | \
|
|
|
|
(1UL << SOCK_DBG) | \
|
|
|
|
(1UL << SOCK_RCVTSTAMP) | \
|
|
|
|
(1UL << SOCK_RCVTSTAMPNS) | \
|
|
|
|
(1UL << SOCK_LOCALROUTE) | \
|
|
|
|
(1UL << SOCK_TIMESTAMPING_RX_SOFTWARE) | \
|
|
|
|
(1UL << SOCK_RXQ_OVFL) | \
|
|
|
|
(1UL << SOCK_WIFI_STATUS) | \
|
|
|
|
(1UL << SOCK_NOFCS) | \
|
2019-02-02 18:34:51 +03:00
|
|
|
(1UL << SOCK_FILTER_LOCKED) | \
|
|
|
|
(1UL << SOCK_TSTAMP_NEW))
|
2017-01-09 18:55:13 +03:00
|
|
|
/* copy only relevant settings and flags of SOL_SOCKET level from smc to
|
|
|
|
* clc socket (since smc is not called for these options from net/core)
|
|
|
|
*/
|
|
|
|
static void smc_copy_sock_settings_to_clc(struct smc_sock *smc)
|
|
|
|
{
|
|
|
|
smc_copy_sock_settings(smc->clcsock->sk, &smc->sk, SK_FLAGS_SMC_TO_CLC);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SK_FLAGS_CLC_TO_SMC ((1UL << SOCK_URGINLINE) | \
|
|
|
|
(1UL << SOCK_KEEPOPEN) | \
|
|
|
|
(1UL << SOCK_LINGER) | \
|
|
|
|
(1UL << SOCK_DBG))
|
|
|
|
/* copy only settings and flags relevant for smc from clc to smc socket */
|
|
|
|
static void smc_copy_sock_settings_to_smc(struct smc_sock *smc)
|
|
|
|
{
|
|
|
|
smc_copy_sock_settings(&smc->sk, smc->clcsock->sk, SK_FLAGS_CLC_TO_SMC);
|
|
|
|
}
|
|
|
|
|
2020-04-29 18:10:41 +03:00
|
|
|
/* register the new rmb on all links */
|
2020-05-01 13:48:01 +03:00
|
|
|
static int smcr_lgr_reg_rmbs(struct smc_link *link,
|
2020-04-29 18:10:41 +03:00
|
|
|
struct smc_buf_desc *rmb_desc)
|
|
|
|
{
|
2020-05-01 13:48:01 +03:00
|
|
|
struct smc_link_group *lgr = link->lgr;
|
|
|
|
int i, rc = 0;
|
2020-04-29 18:10:41 +03:00
|
|
|
|
2020-05-01 13:48:05 +03:00
|
|
|
rc = smc_llc_flow_initiate(lgr, SMC_LLC_FLOW_RKEY);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
/* protect against parallel smc_llc_cli_rkey_exchange() and
|
|
|
|
* parallel smcr_link_reg_rmb()
|
|
|
|
*/
|
|
|
|
mutex_lock(&lgr->llc_conf_mutex);
|
2020-04-29 18:10:41 +03:00
|
|
|
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
2020-07-18 16:06:16 +03:00
|
|
|
if (!smc_link_active(&lgr->lnk[i]))
|
2020-04-29 18:10:41 +03:00
|
|
|
continue;
|
2020-05-01 13:48:01 +03:00
|
|
|
rc = smcr_link_reg_rmb(&lgr->lnk[i], rmb_desc);
|
2020-04-29 18:10:41 +03:00
|
|
|
if (rc)
|
2020-05-01 13:48:01 +03:00
|
|
|
goto out;
|
2018-05-15 18:04:55 +03:00
|
|
|
}
|
2020-05-01 13:48:01 +03:00
|
|
|
|
|
|
|
/* exchange confirm_rkey msg with peer */
|
|
|
|
rc = smc_llc_do_confirm_rkey(link, rmb_desc);
|
|
|
|
if (rc) {
|
|
|
|
rc = -EFAULT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
rmb_desc->is_conf_rkey = true;
|
|
|
|
out:
|
2020-05-01 13:48:05 +03:00
|
|
|
mutex_unlock(&lgr->llc_conf_mutex);
|
|
|
|
smc_llc_flow_stop(lgr, &lgr->llc_flow_lcl);
|
2020-05-01 13:48:01 +03:00
|
|
|
return rc;
|
2018-05-03 18:57:37 +03:00
|
|
|
}
|
|
|
|
|
2020-04-29 18:10:41 +03:00
|
|
|
static int smcr_clnt_conf_first_link(struct smc_sock *smc)
|
2017-01-09 18:55:21 +03:00
|
|
|
{
|
2020-04-29 18:10:40 +03:00
|
|
|
struct smc_link *link = smc->conn.lnk;
|
2020-04-30 16:55:43 +03:00
|
|
|
struct smc_llc_qentry *qentry;
|
2017-01-09 18:55:21 +03:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* receive CONFIRM LINK request from server over RoCE fabric */
|
2020-04-30 16:55:43 +03:00
|
|
|
qentry = smc_llc_wait(link->lgr, NULL, SMC_LLC_WAIT_TIME,
|
|
|
|
SMC_LLC_CONFIRM_LINK);
|
|
|
|
if (!qentry) {
|
2017-01-09 18:55:21 +03:00
|
|
|
struct smc_clc_msg_decline dclc;
|
|
|
|
|
|
|
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
2018-11-22 12:26:39 +03:00
|
|
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
2018-11-22 12:26:37 +03:00
|
|
|
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
2020-05-04 15:18:48 +03:00
|
|
|
smc_llc_save_peer_uid(qentry);
|
2020-04-30 16:55:43 +03:00
|
|
|
rc = smc_llc_eval_conf_link(qentry, SMC_LLC_REQ);
|
|
|
|
smc_llc_flow_qentry_del(&link->lgr->llc_flow_lcl);
|
|
|
|
if (rc)
|
2018-03-01 15:51:31 +03:00
|
|
|
return SMC_CLC_DECL_RMBE_EC;
|
|
|
|
|
2017-01-09 18:55:21 +03:00
|
|
|
rc = smc_ib_modify_qp_rts(link);
|
|
|
|
if (rc)
|
2018-07-25 17:35:32 +03:00
|
|
|
return SMC_CLC_DECL_ERR_RDYLNK;
|
2017-01-09 18:55:21 +03:00
|
|
|
|
|
|
|
smc_wr_remember_qp_attr(link);
|
2017-07-28 14:56:17 +03:00
|
|
|
|
2020-05-01 13:48:01 +03:00
|
|
|
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc))
|
2018-07-25 17:35:32 +03:00
|
|
|
return SMC_CLC_DECL_ERR_REGRMB;
|
2017-07-28 14:56:17 +03:00
|
|
|
|
2020-04-30 16:55:43 +03:00
|
|
|
/* confirm_rkey is implicit on 1st contact */
|
|
|
|
smc->conn.rmb_desc->is_conf_rkey = true;
|
|
|
|
|
2017-01-09 18:55:21 +03:00
|
|
|
/* send CONFIRM LINK response over RoCE fabric */
|
2018-07-25 17:35:30 +03:00
|
|
|
rc = smc_llc_send_confirm_link(link, SMC_LLC_RESP);
|
2017-01-09 18:55:21 +03:00
|
|
|
if (rc < 0)
|
2018-07-25 17:35:32 +03:00
|
|
|
return SMC_CLC_DECL_TIMEOUT_CL;
|
2017-01-09 18:55:21 +03:00
|
|
|
|
2020-04-30 16:55:43 +03:00
|
|
|
smc_llc_link_active(link);
|
2020-05-05 16:01:20 +03:00
|
|
|
smcr_lgr_set_type(link->lgr, SMC_LGR_SINGLE);
|
2020-04-30 16:55:43 +03:00
|
|
|
|
|
|
|
/* optional 2nd link, receive ADD LINK request from server */
|
|
|
|
qentry = smc_llc_wait(link->lgr, NULL, SMC_LLC_WAIT_TIME,
|
|
|
|
SMC_LLC_ADD_LINK);
|
|
|
|
if (!qentry) {
|
2018-03-01 15:51:32 +03:00
|
|
|
struct smc_clc_msg_decline dclc;
|
|
|
|
|
|
|
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
2018-11-22 12:26:39 +03:00
|
|
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
2020-04-30 16:55:43 +03:00
|
|
|
if (rc == -EAGAIN)
|
|
|
|
rc = 0; /* no DECLINE received, go with one link */
|
|
|
|
return rc;
|
2018-03-01 15:51:32 +03:00
|
|
|
}
|
2020-04-30 16:55:43 +03:00
|
|
|
smc_llc_flow_qentry_clr(&link->lgr->llc_flow_lcl);
|
2020-05-03 15:38:42 +03:00
|
|
|
smc_llc_cli_add_link(link, qentry);
|
2018-03-01 15:51:31 +03:00
|
|
|
return 0;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
static void smcr_conn_save_peer_info(struct smc_sock *smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *clc)
|
2017-01-09 18:55:17 +03:00
|
|
|
{
|
2020-09-10 19:48:23 +03:00
|
|
|
int bufsize = smc_uncompress_bufsize(clc->r0.rmbe_size);
|
2018-05-18 10:34:13 +03:00
|
|
|
|
2020-09-10 19:48:23 +03:00
|
|
|
smc->conn.peer_rmbe_idx = clc->r0.rmbe_idx;
|
|
|
|
smc->conn.local_tx_ctrl.token = ntohl(clc->r0.rmbe_alert_token);
|
2018-05-18 10:34:13 +03:00
|
|
|
smc->conn.peer_rmbe_size = bufsize;
|
2017-01-09 18:55:18 +03:00
|
|
|
atomic_set(&smc->conn.peer_rmbe_space, smc->conn.peer_rmbe_size);
|
2018-05-18 10:34:13 +03:00
|
|
|
smc->conn.tx_off = bufsize * (smc->conn.peer_rmbe_idx - 1);
|
2017-01-09 18:55:17 +03:00
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:31 +03:00
|
|
|
static bool smc_isascii(char *hostname)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < SMC_MAX_HOSTNAME_LEN; i++)
|
|
|
|
if (!isascii(hostname[i]))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
static void smcd_conn_save_peer_info(struct smc_sock *smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *clc)
|
|
|
|
{
|
2020-09-10 19:48:23 +03:00
|
|
|
int bufsize = smc_uncompress_bufsize(clc->d0.dmbe_size);
|
2018-06-28 20:05:11 +03:00
|
|
|
|
2020-09-10 19:48:23 +03:00
|
|
|
smc->conn.peer_rmbe_idx = clc->d0.dmbe_idx;
|
|
|
|
smc->conn.peer_token = clc->d0.token;
|
2018-06-28 20:05:11 +03:00
|
|
|
/* msg header takes up space in the buffer */
|
|
|
|
smc->conn.peer_rmbe_size = bufsize - sizeof(struct smcd_cdc_msg);
|
|
|
|
atomic_set(&smc->conn.peer_rmbe_space, smc->conn.peer_rmbe_size);
|
|
|
|
smc->conn.tx_off = bufsize * smc->conn.peer_rmbe_idx;
|
2020-09-26 13:44:31 +03:00
|
|
|
if (clc->hdr.version > SMC_V1 &&
|
|
|
|
(clc->hdr.typev2 & SMC_FIRST_CONTACT_MASK)) {
|
|
|
|
struct smc_clc_msg_accept_confirm_v2 *clc_v2 =
|
|
|
|
(struct smc_clc_msg_accept_confirm_v2 *)clc;
|
|
|
|
struct smc_clc_first_contact_ext *fce =
|
|
|
|
(struct smc_clc_first_contact_ext *)
|
|
|
|
(((u8 *)clc_v2) + sizeof(*clc_v2));
|
|
|
|
|
|
|
|
memcpy(smc->conn.lgr->negotiated_eid, clc_v2->eid,
|
|
|
|
SMC_MAX_EID_LEN);
|
|
|
|
smc->conn.lgr->peer_os = fce->os_type;
|
|
|
|
smc->conn.lgr->peer_smc_release = fce->release;
|
|
|
|
if (smc_isascii(fce->hostname))
|
|
|
|
memcpy(smc->conn.lgr->peer_hostname, fce->hostname,
|
|
|
|
SMC_MAX_HOSTNAME_LEN);
|
|
|
|
}
|
2018-06-28 20:05:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_conn_save_peer_info(struct smc_sock *smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *clc)
|
|
|
|
{
|
|
|
|
if (smc->conn.lgr->is_smcd)
|
|
|
|
smcd_conn_save_peer_info(smc, clc);
|
|
|
|
else
|
|
|
|
smcr_conn_save_peer_info(smc, clc);
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:17 +03:00
|
|
|
static void smc_link_save_peer_info(struct smc_link *link,
|
|
|
|
struct smc_clc_msg_accept_confirm *clc)
|
|
|
|
{
|
2020-09-10 19:48:23 +03:00
|
|
|
link->peer_qpn = ntoh24(clc->r0.qpn);
|
|
|
|
memcpy(link->peer_gid, clc->r0.lcl.gid, SMC_GID_SIZE);
|
|
|
|
memcpy(link->peer_mac, clc->r0.lcl.mac, sizeof(link->peer_mac));
|
|
|
|
link->peer_psn = ntoh24(clc->r0.psn);
|
|
|
|
link->peer_mtu = clc->r0.qp_mtu;
|
2017-01-09 18:55:17 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 17:52:55 +03:00
|
|
|
static void smc_stat_inc_fback_rsn_cnt(struct smc_sock *smc,
|
|
|
|
struct smc_stats_fback *fback_arr)
|
|
|
|
{
|
|
|
|
int cnt;
|
|
|
|
|
|
|
|
for (cnt = 0; cnt < SMC_MAX_FBACK_RSN_CNT; cnt++) {
|
|
|
|
if (fback_arr[cnt].fback_code == smc->fallback_rsn) {
|
|
|
|
fback_arr[cnt].count++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!fback_arr[cnt].fback_code) {
|
|
|
|
fback_arr[cnt].fback_code = smc->fallback_rsn;
|
|
|
|
fback_arr[cnt].count++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_stat_fallback(struct smc_sock *smc)
|
|
|
|
{
|
2021-06-16 17:52:58 +03:00
|
|
|
struct net *net = sock_net(&smc->sk);
|
|
|
|
|
|
|
|
mutex_lock(&net->smc.mutex_fback_rsn);
|
2021-06-16 17:52:55 +03:00
|
|
|
if (smc->listen_smc) {
|
2021-06-16 17:52:58 +03:00
|
|
|
smc_stat_inc_fback_rsn_cnt(smc, net->smc.fback_rsn->srv);
|
|
|
|
net->smc.fback_rsn->srv_fback_cnt++;
|
2021-06-16 17:52:55 +03:00
|
|
|
} else {
|
2021-06-16 17:52:58 +03:00
|
|
|
smc_stat_inc_fback_rsn_cnt(smc, net->smc.fback_rsn->clnt);
|
|
|
|
net->smc.fback_rsn->clnt_fback_cnt++;
|
2021-06-16 17:52:55 +03:00
|
|
|
}
|
2021-06-16 17:52:58 +03:00
|
|
|
mutex_unlock(&net->smc.mutex_fback_rsn);
|
2021-06-16 17:52:55 +03:00
|
|
|
}
|
|
|
|
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
/* must be called under rcu read lock */
|
|
|
|
static void smc_fback_wakeup_waitqueue(struct smc_sock *smc, void *key)
|
|
|
|
{
|
|
|
|
struct socket_wq *wq;
|
|
|
|
__poll_t flags;
|
|
|
|
|
|
|
|
wq = rcu_dereference(smc->sk.sk_wq);
|
|
|
|
if (!skwq_has_sleeper(wq))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* wake up smc sk->sk_wq */
|
|
|
|
if (!key) {
|
|
|
|
/* sk_state_change */
|
|
|
|
wake_up_interruptible_all(&wq->wait);
|
|
|
|
} else {
|
|
|
|
flags = key_to_poll(key);
|
|
|
|
if (flags & (EPOLLIN | EPOLLOUT))
|
|
|
|
/* sk_data_ready or sk_write_space */
|
|
|
|
wake_up_interruptible_sync_poll(&wq->wait, flags);
|
|
|
|
else if (flags & EPOLLERR)
|
|
|
|
/* sk_error_report */
|
|
|
|
wake_up_interruptible_poll(&wq->wait, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_fback_mark_woken(wait_queue_entry_t *wait,
|
|
|
|
unsigned int mode, int sync, void *key)
|
|
|
|
{
|
|
|
|
struct smc_mark_woken *mark =
|
|
|
|
container_of(wait, struct smc_mark_woken, wait_entry);
|
|
|
|
|
|
|
|
mark->woken = true;
|
|
|
|
mark->key = key;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_fback_forward_wakeup(struct smc_sock *smc, struct sock *clcsk,
|
|
|
|
void (*clcsock_callback)(struct sock *sk))
|
|
|
|
{
|
|
|
|
struct smc_mark_woken mark = { .woken = false };
|
|
|
|
struct socket_wq *wq;
|
|
|
|
|
|
|
|
init_waitqueue_func_entry(&mark.wait_entry,
|
|
|
|
smc_fback_mark_woken);
|
|
|
|
rcu_read_lock();
|
|
|
|
wq = rcu_dereference(clcsk->sk_wq);
|
|
|
|
if (!wq)
|
|
|
|
goto out;
|
|
|
|
add_wait_queue(sk_sleep(clcsk), &mark.wait_entry);
|
|
|
|
clcsock_callback(clcsk);
|
|
|
|
remove_wait_queue(sk_sleep(clcsk), &mark.wait_entry);
|
|
|
|
|
|
|
|
if (mark.woken)
|
|
|
|
smc_fback_wakeup_waitqueue(smc, mark.key);
|
|
|
|
out:
|
|
|
|
rcu_read_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_fback_state_change(struct sock *clcsk)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc =
|
|
|
|
smc_clcsock_user_data(clcsk);
|
|
|
|
|
|
|
|
if (!smc)
|
|
|
|
return;
|
|
|
|
smc_fback_forward_wakeup(smc, clcsk, smc->clcsk_state_change);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_fback_data_ready(struct sock *clcsk)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc =
|
|
|
|
smc_clcsock_user_data(clcsk);
|
|
|
|
|
|
|
|
if (!smc)
|
|
|
|
return;
|
|
|
|
smc_fback_forward_wakeup(smc, clcsk, smc->clcsk_data_ready);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_fback_write_space(struct sock *clcsk)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc =
|
|
|
|
smc_clcsock_user_data(clcsk);
|
|
|
|
|
|
|
|
if (!smc)
|
|
|
|
return;
|
|
|
|
smc_fback_forward_wakeup(smc, clcsk, smc->clcsk_write_space);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_fback_error_report(struct sock *clcsk)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc =
|
|
|
|
smc_clcsock_user_data(clcsk);
|
|
|
|
|
|
|
|
if (!smc)
|
|
|
|
return;
|
|
|
|
smc_fback_forward_wakeup(smc, clcsk, smc->clcsk_error_report);
|
|
|
|
}
|
|
|
|
|
2022-01-22 12:43:09 +03:00
|
|
|
static int smc_switch_to_fallback(struct smc_sock *smc, int reason_code)
|
2019-04-11 12:17:32 +03:00
|
|
|
{
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
struct sock *clcsk;
|
2022-02-09 17:10:53 +03:00
|
|
|
int rc = 0;
|
2021-11-13 10:33:35 +03:00
|
|
|
|
2022-01-22 12:43:09 +03:00
|
|
|
mutex_lock(&smc->clcsock_release_lock);
|
|
|
|
if (!smc->clcsock) {
|
2022-02-09 17:10:53 +03:00
|
|
|
rc = -EBADF;
|
|
|
|
goto out;
|
2022-01-22 12:43:09 +03:00
|
|
|
}
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
clcsk = smc->clcsock->sk;
|
|
|
|
|
2022-02-09 17:10:53 +03:00
|
|
|
if (smc->use_fallback)
|
|
|
|
goto out;
|
2019-04-11 12:17:32 +03:00
|
|
|
smc->use_fallback = true;
|
2021-06-16 17:52:55 +03:00
|
|
|
smc->fallback_rsn = reason_code;
|
|
|
|
smc_stat_fallback(smc);
|
2019-04-11 12:17:32 +03:00
|
|
|
if (smc->sk.sk_socket && smc->sk.sk_socket->file) {
|
|
|
|
smc->clcsock->file = smc->sk.sk_socket->file;
|
|
|
|
smc->clcsock->file->private_data = smc->clcsock;
|
2020-02-14 10:58:59 +03:00
|
|
|
smc->clcsock->wq.fasync_list =
|
|
|
|
smc->sk.sk_socket->wq.fasync_list;
|
2021-11-13 10:33:35 +03:00
|
|
|
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
/* There might be some wait entries remaining
|
|
|
|
* in smc sk->sk_wq and they should be woken up
|
|
|
|
* as clcsock's wait queue is woken up.
|
2021-11-13 10:33:35 +03:00
|
|
|
*/
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
smc->clcsk_state_change = clcsk->sk_state_change;
|
|
|
|
smc->clcsk_data_ready = clcsk->sk_data_ready;
|
|
|
|
smc->clcsk_write_space = clcsk->sk_write_space;
|
|
|
|
smc->clcsk_error_report = clcsk->sk_error_report;
|
|
|
|
|
|
|
|
clcsk->sk_state_change = smc_fback_state_change;
|
|
|
|
clcsk->sk_data_ready = smc_fback_data_ready;
|
|
|
|
clcsk->sk_write_space = smc_fback_write_space;
|
|
|
|
clcsk->sk_error_report = smc_fback_error_report;
|
|
|
|
|
|
|
|
smc->clcsock->sk->sk_user_data =
|
|
|
|
(void *)((uintptr_t)smc | SK_USER_DATA_NOCOPY);
|
2019-04-11 12:17:32 +03:00
|
|
|
}
|
2022-02-09 17:10:53 +03:00
|
|
|
out:
|
2022-01-22 12:43:09 +03:00
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
2022-02-09 17:10:53 +03:00
|
|
|
return rc;
|
2019-04-11 12:17:32 +03:00
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* fall back during connect */
|
2018-07-25 17:35:32 +03:00
|
|
|
static int smc_connect_fallback(struct smc_sock *smc, int reason_code)
|
2017-01-09 18:55:16 +03:00
|
|
|
{
|
2022-01-22 12:43:09 +03:00
|
|
|
struct net *net = sock_net(&smc->sk);
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
rc = smc_switch_to_fallback(smc, reason_code);
|
|
|
|
if (rc) { /* fallback fails */
|
|
|
|
this_cpu_inc(net->smc.smc_stats->clnt_hshake_err_cnt);
|
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
sock_put(&smc->sk); /* passive closing */
|
|
|
|
return rc;
|
|
|
|
}
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_copy_sock_settings_to_clc(smc);
|
2019-04-12 13:57:23 +03:00
|
|
|
smc->connect_nonblock = 0;
|
2018-05-18 10:34:18 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
smc->sk.sk_state = SMC_ACTIVE;
|
|
|
|
return 0;
|
|
|
|
}
|
2018-01-26 11:28:48 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* decline and fall back during connect */
|
2020-09-26 13:44:32 +03:00
|
|
|
static int smc_connect_decline_fallback(struct smc_sock *smc, int reason_code,
|
|
|
|
u8 version)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2021-06-16 17:52:58 +03:00
|
|
|
struct net *net = sock_net(&smc->sk);
|
2018-05-18 10:34:18 +03:00
|
|
|
int rc;
|
2018-04-26 18:18:21 +03:00
|
|
|
|
2018-07-05 17:15:30 +03:00
|
|
|
if (reason_code < 0) { /* error, fallback is not possible */
|
2021-06-16 17:52:58 +03:00
|
|
|
this_cpu_inc(net->smc.smc_stats->clnt_hshake_err_cnt);
|
2018-07-05 17:15:30 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
sock_put(&smc->sk); /* passive closing */
|
2018-05-18 10:34:18 +03:00
|
|
|
return reason_code;
|
2018-07-05 17:15:30 +03:00
|
|
|
}
|
2018-07-25 17:35:32 +03:00
|
|
|
if (reason_code != SMC_CLC_DECL_PEERDECL) {
|
2020-09-26 13:44:32 +03:00
|
|
|
rc = smc_clc_send_decline(smc, reason_code, version);
|
2018-07-05 17:15:30 +03:00
|
|
|
if (rc < 0) {
|
2021-06-16 17:52:58 +03:00
|
|
|
this_cpu_inc(net->smc.smc_stats->clnt_hshake_err_cnt);
|
2018-07-05 17:15:30 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
sock_put(&smc->sk); /* passive closing */
|
2018-05-18 10:34:18 +03:00
|
|
|
return rc;
|
2018-07-05 17:15:30 +03:00
|
|
|
}
|
2017-10-25 12:01:46 +03:00
|
|
|
}
|
2018-07-25 17:35:32 +03:00
|
|
|
return smc_connect_fallback(smc, reason_code);
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
2017-10-25 12:01:46 +03:00
|
|
|
|
2020-12-01 22:20:36 +03:00
|
|
|
static void smc_conn_abort(struct smc_sock *smc, int local_first)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2022-01-06 15:42:08 +03:00
|
|
|
struct smc_connection *conn = &smc->conn;
|
|
|
|
struct smc_link_group *lgr = conn->lgr;
|
|
|
|
|
|
|
|
smc_conn_free(conn);
|
2020-09-10 19:48:21 +03:00
|
|
|
if (local_first)
|
2022-01-06 15:42:08 +03:00
|
|
|
smc_lgr_cleanup_early(lgr);
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check if there is a rdma device available for this connection. */
|
|
|
|
/* called for connect and listen */
|
2019-04-12 13:57:28 +03:00
|
|
|
static int smc_find_rdma_device(struct smc_sock *smc, struct smc_init_info *ini)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2017-01-09 18:55:16 +03:00
|
|
|
/* PNET table look up: search active ib_device and port
|
|
|
|
* within same PNETID that also contains the ethernet device
|
|
|
|
* used for the internal TCP socket
|
|
|
|
*/
|
2019-04-12 13:57:26 +03:00
|
|
|
smc_pnet_find_roce_resource(smc->clcsock->sk, ini);
|
2019-04-12 13:57:29 +03:00
|
|
|
if (!ini->ib_dev)
|
|
|
|
return SMC_CLC_DECL_NOSMCRDEV;
|
2019-04-12 13:57:26 +03:00
|
|
|
return 0;
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* check if there is an ISM device available for this connection. */
|
|
|
|
/* called for connect and listen */
|
2019-04-12 13:57:28 +03:00
|
|
|
static int smc_find_ism_device(struct smc_sock *smc, struct smc_init_info *ini)
|
2018-06-28 20:05:11 +03:00
|
|
|
{
|
|
|
|
/* Find ISM device with same PNETID as connecting interface */
|
2019-04-12 13:57:26 +03:00
|
|
|
smc_pnet_find_ism_resource(smc->clcsock->sk, ini);
|
2020-09-26 13:44:23 +03:00
|
|
|
if (!ini->ism_dev[0])
|
2019-04-12 13:57:29 +03:00
|
|
|
return SMC_CLC_DECL_NOSMCDDEV;
|
2020-09-26 13:44:25 +03:00
|
|
|
else
|
|
|
|
ini->ism_chid[0] = smc_ism_get_chid(ini->ism_dev[0]);
|
2018-06-28 20:05:11 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-10-02 18:09:26 +03:00
|
|
|
/* is chid unique for the ism devices that are already determined? */
|
|
|
|
static bool smc_find_ism_v2_is_unique_chid(u16 chid, struct smc_init_info *ini,
|
|
|
|
int cnt)
|
|
|
|
{
|
|
|
|
int i = (!ini->ism_dev[0]) ? 1 : 0;
|
|
|
|
|
|
|
|
for (; i < cnt; i++)
|
|
|
|
if (ini->ism_chid[i] == chid)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:27 +03:00
|
|
|
/* determine possible V2 ISM devices (either without PNETID or with PNETID plus
|
|
|
|
* PNETID matching net_device)
|
|
|
|
*/
|
|
|
|
static int smc_find_ism_v2_device_clnt(struct smc_sock *smc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
int rc = SMC_CLC_DECL_NOSMCDDEV;
|
|
|
|
struct smcd_dev *smcd;
|
|
|
|
int i = 1;
|
2020-10-02 18:09:26 +03:00
|
|
|
u16 chid;
|
2020-09-26 13:44:27 +03:00
|
|
|
|
|
|
|
if (smcd_indicated(ini->smc_type_v1))
|
|
|
|
rc = 0; /* already initialized for V1 */
|
|
|
|
mutex_lock(&smcd_dev_list.mutex);
|
|
|
|
list_for_each_entry(smcd, &smcd_dev_list.list, list) {
|
|
|
|
if (smcd->going_away || smcd == ini->ism_dev[0])
|
|
|
|
continue;
|
2020-10-02 18:09:26 +03:00
|
|
|
chid = smc_ism_get_chid(smcd);
|
|
|
|
if (!smc_find_ism_v2_is_unique_chid(chid, ini, i))
|
|
|
|
continue;
|
2020-09-26 13:44:27 +03:00
|
|
|
if (!smc_pnet_is_pnetid_set(smcd->pnetid) ||
|
|
|
|
smc_pnet_is_ndev_pnetid(sock_net(&smc->sk), smcd->pnetid)) {
|
|
|
|
ini->ism_dev[i] = smcd;
|
2020-10-02 18:09:26 +03:00
|
|
|
ini->ism_chid[i] = chid;
|
2020-09-26 13:44:27 +03:00
|
|
|
ini->is_smcd = true;
|
|
|
|
rc = 0;
|
|
|
|
i++;
|
|
|
|
if (i > SMC_MAX_ISM_DEVS)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&smcd_dev_list.mutex);
|
|
|
|
ini->ism_offered_cnt = i - 1;
|
|
|
|
if (!ini->ism_dev[0] && !ini->ism_dev[1])
|
|
|
|
ini->smcd_version = 0;
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* Check for VLAN ID and register it on ISM device just for CLC handshake */
|
|
|
|
static int smc_connect_ism_vlan_setup(struct smc_sock *smc,
|
2019-04-12 13:57:26 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-06-28 20:05:11 +03:00
|
|
|
{
|
2020-09-26 13:44:23 +03:00
|
|
|
if (ini->vlan_id && smc_ism_get_vlan(ini->ism_dev[0], ini->vlan_id))
|
2019-04-12 13:57:30 +03:00
|
|
|
return SMC_CLC_DECL_ISMVLANERR;
|
2018-06-28 20:05:11 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:27 +03:00
|
|
|
static int smc_find_proposal_devices(struct smc_sock *smc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
/* check if there is an ism device available */
|
|
|
|
if (ini->smcd_version & SMC_V1) {
|
|
|
|
if (smc_find_ism_device(smc, ini) ||
|
|
|
|
smc_connect_ism_vlan_setup(smc, ini)) {
|
|
|
|
if (ini->smc_type_v1 == SMC_TYPE_B)
|
|
|
|
ini->smc_type_v1 = SMC_TYPE_R;
|
|
|
|
else
|
|
|
|
ini->smc_type_v1 = SMC_TYPE_N;
|
|
|
|
} /* else ISM V1 is supported for this connection */
|
|
|
|
if (smc_find_rdma_device(smc, ini)) {
|
|
|
|
if (ini->smc_type_v1 == SMC_TYPE_B)
|
|
|
|
ini->smc_type_v1 = SMC_TYPE_D;
|
|
|
|
else
|
|
|
|
ini->smc_type_v1 = SMC_TYPE_N;
|
|
|
|
} /* else RDMA is supported for this connection */
|
|
|
|
}
|
2020-12-01 22:20:42 +03:00
|
|
|
if (smc_ism_is_v2_capable() && smc_find_ism_v2_device_clnt(smc, ini))
|
2020-09-26 13:44:27 +03:00
|
|
|
ini->smc_type_v2 = SMC_TYPE_N;
|
|
|
|
|
|
|
|
/* if neither ISM nor RDMA are supported, fallback */
|
|
|
|
if (!smcr_indicated(ini->smc_type_v1) &&
|
|
|
|
ini->smc_type_v1 == SMC_TYPE_N && ini->smc_type_v2 == SMC_TYPE_N)
|
|
|
|
rc = SMC_CLC_DECL_NOSMCDEV;
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* cleanup temporary VLAN ID registration used for CLC handshake. If ISM is
|
|
|
|
* used, the VLAN ID will be registered again during the connection setup.
|
|
|
|
*/
|
2020-09-26 13:44:27 +03:00
|
|
|
static int smc_connect_ism_vlan_cleanup(struct smc_sock *smc,
|
2019-04-12 13:57:26 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-06-28 20:05:11 +03:00
|
|
|
{
|
2020-09-26 13:44:27 +03:00
|
|
|
if (!smcd_indicated(ini->smc_type_v1))
|
2018-06-28 20:05:11 +03:00
|
|
|
return 0;
|
2020-09-26 13:44:23 +03:00
|
|
|
if (ini->vlan_id && smc_ism_put_vlan(ini->ism_dev[0], ini->vlan_id))
|
2018-06-28 20:05:11 +03:00
|
|
|
return SMC_CLC_DECL_CNFERR;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
#define SMC_CLC_MAX_ACCEPT_LEN \
|
|
|
|
(sizeof(struct smc_clc_msg_accept_confirm_v2) + \
|
2020-09-26 13:44:31 +03:00
|
|
|
sizeof(struct smc_clc_first_contact_ext) + \
|
2020-09-26 13:44:30 +03:00
|
|
|
sizeof(struct smc_clc_msg_trail))
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* CLC handshake during connect */
|
2020-09-26 13:44:27 +03:00
|
|
|
static int smc_connect_clc(struct smc_sock *smc,
|
2020-09-26 13:44:30 +03:00
|
|
|
struct smc_clc_msg_accept_confirm_v2 *aclc2,
|
2019-04-12 13:57:26 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
|
|
|
int rc = 0;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
|
|
|
/* do inband token exchange */
|
2020-09-26 13:44:27 +03:00
|
|
|
rc = smc_clc_send_proposal(smc, ini);
|
2018-05-18 10:34:18 +03:00
|
|
|
if (rc)
|
|
|
|
return rc;
|
2017-01-09 18:55:16 +03:00
|
|
|
/* receive SMC Accept CLC message */
|
2020-09-26 13:44:30 +03:00
|
|
|
return smc_clc_wait_msg(smc, aclc2, SMC_CLC_MAX_ACCEPT_LEN,
|
|
|
|
SMC_CLC_ACCEPT, CLC_WAIT_TIME);
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* setup for RDMA connection of client */
|
|
|
|
static int smc_connect_rdma(struct smc_sock *smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *aclc,
|
2019-04-12 13:57:26 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2020-04-30 16:55:43 +03:00
|
|
|
int i, reason_code = 0;
|
2018-05-18 10:34:18 +03:00
|
|
|
struct smc_link *link;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2019-04-12 13:57:26 +03:00
|
|
|
ini->is_smcd = false;
|
2020-09-10 19:48:23 +03:00
|
|
|
ini->ib_lcl = &aclc->r0.lcl;
|
|
|
|
ini->ib_clcqpn = ntoh24(aclc->r0.qpn);
|
2020-09-26 13:44:20 +03:00
|
|
|
ini->first_contact_peer = aclc->hdr.typev2 & SMC_FIRST_CONTACT_MASK;
|
2019-04-12 13:57:26 +03:00
|
|
|
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_lock(&smc_client_lgr_pending);
|
2019-04-12 13:57:30 +03:00
|
|
|
reason_code = smc_conn_create(smc, ini);
|
|
|
|
if (reason_code) {
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_unlock(&smc_client_lgr_pending);
|
|
|
|
return reason_code;
|
2017-01-09 18:55:17 +03:00
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_conn_save_peer_info(smc, aclc);
|
2017-01-09 18:55:18 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (ini->first_contact_local) {
|
2020-04-30 16:55:43 +03:00
|
|
|
link = smc->conn.lnk;
|
|
|
|
} else {
|
|
|
|
/* set link that was assigned by server */
|
|
|
|
link = NULL;
|
|
|
|
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
|
|
|
struct smc_link *l = &smc->conn.lgr->lnk[i];
|
|
|
|
|
2020-09-10 19:48:23 +03:00
|
|
|
if (l->peer_qpn == ntoh24(aclc->r0.qpn) &&
|
|
|
|
!memcmp(l->peer_gid, &aclc->r0.lcl.gid,
|
|
|
|
SMC_GID_SIZE) &&
|
|
|
|
!memcmp(l->peer_mac, &aclc->r0.lcl.mac,
|
|
|
|
sizeof(l->peer_mac))) {
|
2020-04-30 16:55:43 +03:00
|
|
|
link = l;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-10-07 23:57:41 +03:00
|
|
|
if (!link) {
|
|
|
|
reason_code = SMC_CLC_DECL_NOSRVLINK;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2021-08-09 12:05:57 +03:00
|
|
|
smc_switch_link_and_count(&smc->conn, link);
|
2020-04-30 16:55:43 +03:00
|
|
|
}
|
|
|
|
|
2017-07-28 14:56:20 +03:00
|
|
|
/* create send buffer and rmb */
|
2020-10-07 23:57:41 +03:00
|
|
|
if (smc_buf_create(smc, false)) {
|
|
|
|
reason_code = SMC_CLC_DECL_MEM;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2017-01-09 18:55:18 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (ini->first_contact_local)
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_link_save_peer_info(link, aclc);
|
2017-01-09 18:55:20 +03:00
|
|
|
|
2020-10-07 23:57:41 +03:00
|
|
|
if (smc_rmb_rtoken_handling(&smc->conn, link, aclc)) {
|
|
|
|
reason_code = SMC_CLC_DECL_ERR_RTOK;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2017-01-09 18:55:20 +03:00
|
|
|
|
2017-04-10 15:58:01 +03:00
|
|
|
smc_close_init(smc);
|
|
|
|
smc_rx_init(smc);
|
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (ini->first_contact_local) {
|
2020-10-07 23:57:41 +03:00
|
|
|
if (smc_ib_ready_link(link)) {
|
|
|
|
reason_code = SMC_CLC_DECL_ERR_RDYLNK;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2017-07-28 14:56:17 +03:00
|
|
|
} else {
|
2020-10-07 23:57:41 +03:00
|
|
|
if (smcr_lgr_reg_rmbs(link, smc->conn.rmb_desc)) {
|
|
|
|
reason_code = SMC_CLC_DECL_ERR_REGRMB;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2017-01-09 18:55:20 +03:00
|
|
|
}
|
2017-07-28 14:56:22 +03:00
|
|
|
smc_rmb_sync_sg_for_device(&smc->conn);
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
reason_code = smc_clc_send_confirm(smc, ini->first_contact_local,
|
|
|
|
SMC_V1);
|
2018-05-18 10:34:18 +03:00
|
|
|
if (reason_code)
|
2020-10-07 23:57:41 +03:00
|
|
|
goto connect_abort;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
|
|
|
smc_tx_init(smc);
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (ini->first_contact_local) {
|
2017-01-09 18:55:21 +03:00
|
|
|
/* QP confirmation over RoCE fabric */
|
2020-04-30 16:55:43 +03:00
|
|
|
smc_llc_flow_initiate(link->lgr, SMC_LLC_FLOW_ADD_LINK);
|
2020-04-29 18:10:41 +03:00
|
|
|
reason_code = smcr_clnt_conf_first_link(smc);
|
2020-04-30 16:55:43 +03:00
|
|
|
smc_llc_flow_stop(link->lgr, &link->lgr->llc_flow_lcl);
|
2018-05-18 10:34:18 +03:00
|
|
|
if (reason_code)
|
2020-10-07 23:57:41 +03:00
|
|
|
goto connect_abort;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_unlock(&smc_client_lgr_pending);
|
2017-01-09 18:55:23 +03:00
|
|
|
|
2017-01-09 18:55:16 +03:00
|
|
|
smc_copy_sock_settings_to_clc(smc);
|
2019-04-12 13:57:23 +03:00
|
|
|
smc->connect_nonblock = 0;
|
2017-01-09 18:55:25 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
smc->sk.sk_state = SMC_ACTIVE;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
return 0;
|
2020-10-07 23:57:41 +03:00
|
|
|
connect_abort:
|
2020-12-01 22:20:36 +03:00
|
|
|
smc_conn_abort(smc, ini->first_contact_local);
|
2020-10-07 23:57:41 +03:00
|
|
|
mutex_unlock(&smc_client_lgr_pending);
|
|
|
|
smc->connect_nonblock = 0;
|
|
|
|
|
|
|
|
return reason_code;
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
/* The server has chosen one of the proposed ISM devices for the communication.
|
|
|
|
* Determine from the CHID of the received CLC ACCEPT the ISM device chosen.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
smc_v2_determine_accepted_chid(struct smc_clc_msg_accept_confirm_v2 *aclc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ini->ism_offered_cnt + 1; i++) {
|
|
|
|
if (ini->ism_chid[i] == ntohs(aclc->chid)) {
|
|
|
|
ini->ism_selected = i;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EPROTO;
|
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* setup for ISM connection of client */
|
|
|
|
static int smc_connect_ism(struct smc_sock *smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *aclc,
|
2019-04-12 13:57:26 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-06-28 20:05:11 +03:00
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
|
2019-04-12 13:57:26 +03:00
|
|
|
ini->is_smcd = true;
|
2020-09-26 13:44:20 +03:00
|
|
|
ini->first_contact_peer = aclc->hdr.typev2 & SMC_FIRST_CONTACT_MASK;
|
2019-04-12 13:57:26 +03:00
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
if (aclc->hdr.version == SMC_V2) {
|
|
|
|
struct smc_clc_msg_accept_confirm_v2 *aclc_v2 =
|
|
|
|
(struct smc_clc_msg_accept_confirm_v2 *)aclc;
|
|
|
|
|
|
|
|
rc = smc_v2_determine_accepted_chid(aclc_v2, ini);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
ini->ism_peer_gid[ini->ism_selected] = aclc->d0.gid;
|
|
|
|
|
2019-02-07 17:56:18 +03:00
|
|
|
/* there is only one lgr role for SMC-D; use server lock */
|
|
|
|
mutex_lock(&smc_server_lgr_pending);
|
2019-04-12 13:57:30 +03:00
|
|
|
rc = smc_conn_create(smc, ini);
|
|
|
|
if (rc) {
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
2019-04-12 13:57:30 +03:00
|
|
|
return rc;
|
2019-02-07 17:56:18 +03:00
|
|
|
}
|
2018-06-28 20:05:11 +03:00
|
|
|
|
|
|
|
/* Create send and receive buffers */
|
2020-07-26 21:34:28 +03:00
|
|
|
rc = smc_buf_create(smc, true);
|
2020-10-07 23:57:41 +03:00
|
|
|
if (rc) {
|
|
|
|
rc = (rc == -ENOSPC) ? SMC_CLC_DECL_MAX_DMB : SMC_CLC_DECL_MEM;
|
|
|
|
goto connect_abort;
|
|
|
|
}
|
2018-06-28 20:05:11 +03:00
|
|
|
|
|
|
|
smc_conn_save_peer_info(smc, aclc);
|
|
|
|
smc_close_init(smc);
|
|
|
|
smc_rx_init(smc);
|
|
|
|
smc_tx_init(smc);
|
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
rc = smc_clc_send_confirm(smc, ini->first_contact_local,
|
|
|
|
aclc->hdr.version);
|
2018-06-28 20:05:11 +03:00
|
|
|
if (rc)
|
2020-10-07 23:57:41 +03:00
|
|
|
goto connect_abort;
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
2018-06-28 20:05:11 +03:00
|
|
|
|
|
|
|
smc_copy_sock_settings_to_clc(smc);
|
2019-04-12 13:57:23 +03:00
|
|
|
smc->connect_nonblock = 0;
|
2018-06-28 20:05:11 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT)
|
|
|
|
smc->sk.sk_state = SMC_ACTIVE;
|
|
|
|
|
|
|
|
return 0;
|
2020-10-07 23:57:41 +03:00
|
|
|
connect_abort:
|
2020-12-01 22:20:36 +03:00
|
|
|
smc_conn_abort(smc, ini->first_contact_local);
|
2020-10-07 23:57:41 +03:00
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
|
|
|
smc->connect_nonblock = 0;
|
|
|
|
|
|
|
|
return rc;
|
2018-06-28 20:05:11 +03:00
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:27 +03:00
|
|
|
/* check if received accept type and version matches a proposed one */
|
|
|
|
static int smc_connect_check_aclc(struct smc_init_info *ini,
|
|
|
|
struct smc_clc_msg_accept_confirm *aclc)
|
|
|
|
{
|
|
|
|
if ((aclc->hdr.typev1 == SMC_TYPE_R &&
|
|
|
|
!smcr_indicated(ini->smc_type_v1)) ||
|
|
|
|
(aclc->hdr.typev1 == SMC_TYPE_D &&
|
2020-09-26 13:44:30 +03:00
|
|
|
((!smcd_indicated(ini->smc_type_v1) &&
|
|
|
|
!smcd_indicated(ini->smc_type_v2)) ||
|
|
|
|
(aclc->hdr.version == SMC_V1 &&
|
|
|
|
!smcd_indicated(ini->smc_type_v1)) ||
|
|
|
|
(aclc->hdr.version == SMC_V2 &&
|
|
|
|
!smcd_indicated(ini->smc_type_v2)))))
|
2020-09-26 13:44:27 +03:00
|
|
|
return SMC_CLC_DECL_MODEUNSUPP;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* perform steps before actually connecting */
|
|
|
|
static int __smc_connect(struct smc_sock *smc)
|
|
|
|
{
|
2020-12-01 22:20:42 +03:00
|
|
|
u8 version = smc_ism_is_v2_capable() ? SMC_V2 : SMC_V1;
|
2020-09-26 13:44:30 +03:00
|
|
|
struct smc_clc_msg_accept_confirm_v2 *aclc2;
|
|
|
|
struct smc_clc_msg_accept_confirm *aclc;
|
2020-09-26 13:44:23 +03:00
|
|
|
struct smc_init_info *ini = NULL;
|
2020-09-26 13:44:30 +03:00
|
|
|
u8 *buf = NULL;
|
2018-05-18 10:34:18 +03:00
|
|
|
int rc = 0;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
if (smc->use_fallback)
|
2018-07-25 17:35:32 +03:00
|
|
|
return smc_connect_fallback(smc, smc->fallback_rsn);
|
2018-05-18 10:34:18 +03:00
|
|
|
|
|
|
|
/* if peer has not signalled SMC-capability, fall back */
|
|
|
|
if (!tcp_sk(smc->clcsock->sk)->syn_smc)
|
2018-07-25 17:35:32 +03:00
|
|
|
return smc_connect_fallback(smc, SMC_CLC_DECL_PEERNOSMC);
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2020-09-26 13:44:32 +03:00
|
|
|
/* IPSec connections opt out of SMC optimizations */
|
2018-05-18 10:34:18 +03:00
|
|
|
if (using_ipsec(smc))
|
2020-09-26 13:44:32 +03:00
|
|
|
return smc_connect_decline_fallback(smc, SMC_CLC_DECL_IPSEC,
|
|
|
|
version);
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2020-09-26 13:44:23 +03:00
|
|
|
ini = kzalloc(sizeof(*ini), GFP_KERNEL);
|
|
|
|
if (!ini)
|
2020-09-26 13:44:32 +03:00
|
|
|
return smc_connect_decline_fallback(smc, SMC_CLC_DECL_MEM,
|
|
|
|
version);
|
2020-09-26 13:44:23 +03:00
|
|
|
|
2020-09-26 13:44:27 +03:00
|
|
|
ini->smcd_version = SMC_V1;
|
2020-12-01 22:20:42 +03:00
|
|
|
ini->smcd_version |= smc_ism_is_v2_capable() ? SMC_V2 : 0;
|
2020-09-26 13:44:27 +03:00
|
|
|
ini->smc_type_v1 = SMC_TYPE_B;
|
2020-12-01 22:20:42 +03:00
|
|
|
ini->smc_type_v2 = smc_ism_is_v2_capable() ? SMC_TYPE_D : SMC_TYPE_N;
|
2020-09-26 13:44:27 +03:00
|
|
|
|
2019-04-12 13:57:27 +03:00
|
|
|
/* get vlan id from IP device */
|
2020-09-26 13:44:23 +03:00
|
|
|
if (smc_vlan_by_tcpsk(smc->clcsock, ini)) {
|
2020-09-26 13:44:27 +03:00
|
|
|
ini->smcd_version &= ~SMC_V1;
|
|
|
|
ini->smc_type_v1 = SMC_TYPE_N;
|
|
|
|
if (!ini->smcd_version) {
|
|
|
|
rc = SMC_CLC_DECL_GETVLANERR;
|
|
|
|
goto fallback;
|
|
|
|
}
|
2018-06-28 20:05:11 +03:00
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:27 +03:00
|
|
|
rc = smc_find_proposal_devices(smc, ini);
|
|
|
|
if (rc)
|
|
|
|
goto fallback;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2020-09-26 13:44:30 +03:00
|
|
|
buf = kzalloc(SMC_CLC_MAX_ACCEPT_LEN, GFP_KERNEL);
|
|
|
|
if (!buf) {
|
|
|
|
rc = SMC_CLC_DECL_MEM;
|
|
|
|
goto fallback;
|
|
|
|
}
|
|
|
|
aclc2 = (struct smc_clc_msg_accept_confirm_v2 *)buf;
|
|
|
|
aclc = (struct smc_clc_msg_accept_confirm *)aclc2;
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* perform CLC handshake */
|
2020-09-26 13:44:30 +03:00
|
|
|
rc = smc_connect_clc(smc, aclc2, ini);
|
2020-09-26 13:44:27 +03:00
|
|
|
if (rc)
|
|
|
|
goto vlan_cleanup;
|
|
|
|
|
|
|
|
/* check if smc modes and versions of CLC proposal and accept match */
|
2020-09-26 13:44:30 +03:00
|
|
|
rc = smc_connect_check_aclc(ini, aclc);
|
2020-11-19 00:40:37 +03:00
|
|
|
version = aclc->hdr.version == SMC_V1 ? SMC_V1 : SMC_V2;
|
|
|
|
ini->smcd_version = version;
|
2020-09-26 13:44:27 +03:00
|
|
|
if (rc)
|
|
|
|
goto vlan_cleanup;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* depending on previous steps, connect using rdma or ism */
|
2020-09-26 13:44:30 +03:00
|
|
|
if (aclc->hdr.typev1 == SMC_TYPE_R)
|
|
|
|
rc = smc_connect_rdma(smc, aclc, ini);
|
|
|
|
else if (aclc->hdr.typev1 == SMC_TYPE_D)
|
|
|
|
rc = smc_connect_ism(smc, aclc, ini);
|
2020-09-26 13:44:27 +03:00
|
|
|
if (rc)
|
|
|
|
goto vlan_cleanup;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_CLNT_SUCC_INC(sock_net(smc->clcsock->sk), aclc);
|
2020-09-26 13:44:27 +03:00
|
|
|
smc_connect_ism_vlan_cleanup(smc, ini);
|
2020-09-26 13:44:30 +03:00
|
|
|
kfree(buf);
|
2020-09-26 13:44:23 +03:00
|
|
|
kfree(ini);
|
2018-05-18 10:34:18 +03:00
|
|
|
return 0;
|
2020-09-26 13:44:27 +03:00
|
|
|
|
|
|
|
vlan_cleanup:
|
|
|
|
smc_connect_ism_vlan_cleanup(smc, ini);
|
2020-09-26 13:44:30 +03:00
|
|
|
kfree(buf);
|
2020-09-26 13:44:27 +03:00
|
|
|
fallback:
|
|
|
|
kfree(ini);
|
2020-09-26 13:44:32 +03:00
|
|
|
return smc_connect_decline_fallback(smc, rc, version);
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
2018-06-27 18:59:50 +03:00
|
|
|
static void smc_connect_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc = container_of(work, struct smc_sock,
|
|
|
|
connect_work);
|
2019-04-12 13:57:23 +03:00
|
|
|
long timeo = smc->sk.sk_sndtimeo;
|
|
|
|
int rc = 0;
|
2018-06-27 18:59:50 +03:00
|
|
|
|
2019-04-12 13:57:23 +03:00
|
|
|
if (!timeo)
|
|
|
|
timeo = MAX_SCHEDULE_TIMEOUT;
|
|
|
|
lock_sock(smc->clcsock->sk);
|
2018-06-27 18:59:50 +03:00
|
|
|
if (smc->clcsock->sk->sk_err) {
|
|
|
|
smc->sk.sk_err = smc->clcsock->sk->sk_err;
|
2019-04-12 13:57:23 +03:00
|
|
|
} else if ((1 << smc->clcsock->sk->sk_state) &
|
2021-10-28 10:13:47 +03:00
|
|
|
(TCPF_SYN_SENT | TCPF_SYN_RECV)) {
|
2019-04-12 13:57:23 +03:00
|
|
|
rc = sk_stream_wait_connect(smc->clcsock->sk, &timeo);
|
|
|
|
if ((rc == -EPIPE) &&
|
|
|
|
((1 << smc->clcsock->sk->sk_state) &
|
|
|
|
(TCPF_ESTABLISHED | TCPF_CLOSE_WAIT)))
|
|
|
|
rc = 0;
|
2018-06-27 18:59:50 +03:00
|
|
|
}
|
2019-04-12 13:57:23 +03:00
|
|
|
release_sock(smc->clcsock->sk);
|
|
|
|
lock_sock(&smc->sk);
|
|
|
|
if (rc != 0 || smc->sk.sk_err) {
|
|
|
|
smc->sk.sk_state = SMC_CLOSED;
|
|
|
|
if (rc == -EPIPE || rc == -EAGAIN)
|
|
|
|
smc->sk.sk_err = EPIPE;
|
net/smc: sync err code when tcp connection was refused
[ Upstream commit 4e2e65e2e56c6ceb4ea1719360080c0af083229e ]
In the current implementation, when TCP initiates a connection
to an unavailable [ip,port], ECONNREFUSED will be stored in the
TCP socket, but SMC will not. However, some apps (like curl) use
getsockopt(,,SO_ERROR,,) to get the error information, which makes
them miss the error message and behave strangely.
Fixes: 50717a37db03 ("net/smc: nonblocking connect rework")
Signed-off-by: liuyacan <liuyacan@corp.netease.com>
Reviewed-by: Tony Lu <tonylu@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Sasha Levin <sashal@kernel.org>
2022-04-21 12:40:27 +03:00
|
|
|
else if (rc == -ECONNREFUSED)
|
|
|
|
smc->sk.sk_err = ECONNREFUSED;
|
2019-04-12 13:57:23 +03:00
|
|
|
else if (signal_pending(current))
|
|
|
|
smc->sk.sk_err = -sock_intr_errno(timeo);
|
2019-11-12 18:03:41 +03:00
|
|
|
sock_put(&smc->sk); /* passive closing */
|
2018-06-27 18:59:50 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = __smc_connect(smc);
|
|
|
|
if (rc < 0)
|
|
|
|
smc->sk.sk_err = -rc;
|
|
|
|
|
|
|
|
out:
|
2019-04-11 12:17:32 +03:00
|
|
|
if (!sock_flag(&smc->sk, SOCK_DEAD)) {
|
|
|
|
if (smc->sk.sk_err) {
|
|
|
|
smc->sk.sk_state_change(&smc->sk);
|
|
|
|
} else { /* allow polling before and after fallback decision */
|
|
|
|
smc->clcsock->sk->sk_write_space(smc->clcsock->sk);
|
|
|
|
smc->sk.sk_write_space(&smc->sk);
|
|
|
|
}
|
|
|
|
}
|
2018-06-27 18:59:50 +03:00
|
|
|
release_sock(&smc->sk);
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
static int smc_connect(struct socket *sock, struct sockaddr *addr,
|
|
|
|
int alen, int flags)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -EINVAL;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
|
|
|
|
/* separate smc parameter checking to be safe */
|
|
|
|
if (alen < sizeof(addr->sa_family))
|
|
|
|
goto out_err;
|
2018-03-16 17:06:41 +03:00
|
|
|
if (addr->sa_family != AF_INET && addr->sa_family != AF_INET6)
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out_err;
|
|
|
|
|
|
|
|
lock_sock(sk);
|
|
|
|
switch (sk->sk_state) {
|
|
|
|
default:
|
|
|
|
goto out;
|
|
|
|
case SMC_ACTIVE:
|
|
|
|
rc = -EISCONN;
|
|
|
|
goto out;
|
|
|
|
case SMC_INIT:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
smc_copy_sock_settings_to_clc(smc);
|
2017-10-25 12:01:46 +03:00
|
|
|
tcp_sk(smc->clcsock->sk)->syn_smc = 1;
|
2019-04-12 13:57:23 +03:00
|
|
|
if (smc->connect_nonblock) {
|
|
|
|
rc = -EALREADY;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
rc = kernel_connect(smc->clcsock, addr, alen, flags);
|
|
|
|
if (rc && rc != -EINPROGRESS)
|
|
|
|
goto out;
|
2019-10-29 14:41:26 +03:00
|
|
|
|
2019-12-13 00:35:58 +03:00
|
|
|
if (smc->use_fallback)
|
|
|
|
goto out;
|
2022-05-23 07:57:07 +03:00
|
|
|
sock_hold(&smc->sk); /* sock put in passive closing */
|
2018-06-27 18:59:50 +03:00
|
|
|
if (flags & O_NONBLOCK) {
|
2020-09-10 19:48:29 +03:00
|
|
|
if (queue_work(smc_hs_wq, &smc->connect_work))
|
2019-04-12 13:57:23 +03:00
|
|
|
smc->connect_nonblock = 1;
|
2018-06-27 18:59:50 +03:00
|
|
|
rc = -EINPROGRESS;
|
|
|
|
} else {
|
|
|
|
rc = __smc_connect(smc);
|
|
|
|
if (rc < 0)
|
|
|
|
goto out;
|
|
|
|
else
|
|
|
|
rc = 0; /* success cases including fallback */
|
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
release_sock(sk);
|
|
|
|
out_err:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_clcsock_accept(struct smc_sock *lsmc, struct smc_sock **new_smc)
|
|
|
|
{
|
2018-01-24 12:28:12 +03:00
|
|
|
struct socket *new_clcsock = NULL;
|
|
|
|
struct sock *lsk = &lsmc->sk;
|
2017-01-09 18:55:13 +03:00
|
|
|
struct sock *new_sk;
|
2018-12-18 20:02:25 +03:00
|
|
|
int rc = -EINVAL;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2018-01-24 12:28:12 +03:00
|
|
|
release_sock(lsk);
|
2018-03-16 17:06:41 +03:00
|
|
|
new_sk = smc_sock_alloc(sock_net(lsk), NULL, lsk->sk_protocol);
|
2017-01-09 18:55:13 +03:00
|
|
|
if (!new_sk) {
|
|
|
|
rc = -ENOMEM;
|
2018-01-24 12:28:12 +03:00
|
|
|
lsk->sk_err = ENOMEM;
|
2017-01-09 18:55:13 +03:00
|
|
|
*new_smc = NULL;
|
2018-01-24 12:28:12 +03:00
|
|
|
lock_sock(lsk);
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
*new_smc = smc_sk(new_sk);
|
|
|
|
|
2018-12-18 20:02:25 +03:00
|
|
|
mutex_lock(&lsmc->clcsock_release_lock);
|
|
|
|
if (lsmc->clcsock)
|
2020-09-10 19:48:20 +03:00
|
|
|
rc = kernel_accept(lsmc->clcsock, &new_clcsock, SOCK_NONBLOCK);
|
2018-12-18 20:02:25 +03:00
|
|
|
mutex_unlock(&lsmc->clcsock_release_lock);
|
2018-01-24 12:28:12 +03:00
|
|
|
lock_sock(lsk);
|
2020-09-10 19:48:20 +03:00
|
|
|
if (rc < 0 && rc != -EAGAIN)
|
2018-01-24 12:28:12 +03:00
|
|
|
lsk->sk_err = -rc;
|
2018-01-24 12:28:13 +03:00
|
|
|
if (rc < 0 || lsk->sk_state == SMC_CLOSED) {
|
2019-04-11 12:17:34 +03:00
|
|
|
new_sk->sk_prot->unhash(new_sk);
|
2017-01-09 18:55:16 +03:00
|
|
|
if (new_clcsock)
|
|
|
|
sock_release(new_clcsock);
|
|
|
|
new_sk->sk_state = SMC_CLOSED;
|
|
|
|
sock_set_flag(new_sk, SOCK_DEAD);
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(new_sk); /* final */
|
2017-01-09 18:55:13 +03:00
|
|
|
*new_smc = NULL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2020-09-10 19:48:20 +03:00
|
|
|
/* new clcsock has inherited the smc listen-specific sk_data_ready
|
|
|
|
* function; switch it back to the original sk_data_ready function
|
|
|
|
*/
|
|
|
|
new_clcsock->sk->sk_data_ready = lsmc->clcsk_data_ready;
|
2017-01-09 18:55:13 +03:00
|
|
|
(*new_smc)->clcsock = new_clcsock;
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:16 +03:00
|
|
|
/* add a just created sock to the accept queue of the listen sock as
|
|
|
|
* candidate for a following socket accept call from user space
|
|
|
|
*/
|
|
|
|
static void smc_accept_enqueue(struct sock *parent, struct sock *sk)
|
|
|
|
{
|
|
|
|
struct smc_sock *par = smc_sk(parent);
|
|
|
|
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_hold(sk); /* sock_put in smc_accept_unlink () */
|
2017-01-09 18:55:16 +03:00
|
|
|
spin_lock(&par->accept_q_lock);
|
|
|
|
list_add_tail(&smc_sk(sk)->accept_q, &par->accept_q);
|
|
|
|
spin_unlock(&par->accept_q_lock);
|
|
|
|
sk_acceptq_added(parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove a socket from the accept queue of its parental listening socket */
|
|
|
|
static void smc_accept_unlink(struct sock *sk)
|
|
|
|
{
|
|
|
|
struct smc_sock *par = smc_sk(sk)->listen_smc;
|
|
|
|
|
|
|
|
spin_lock(&par->accept_q_lock);
|
|
|
|
list_del_init(&smc_sk(sk)->accept_q);
|
|
|
|
spin_unlock(&par->accept_q_lock);
|
|
|
|
sk_acceptq_removed(&smc_sk(sk)->listen_smc->sk);
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(sk); /* sock_hold in smc_accept_enqueue */
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* remove a sock from the accept queue to bind it to a new socket created
|
|
|
|
* for a socket accept call from user space
|
|
|
|
*/
|
2017-01-09 18:55:25 +03:00
|
|
|
struct sock *smc_accept_dequeue(struct sock *parent,
|
|
|
|
struct socket *new_sock)
|
2017-01-09 18:55:16 +03:00
|
|
|
{
|
|
|
|
struct smc_sock *isk, *n;
|
|
|
|
struct sock *new_sk;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(isk, n, &smc_sk(parent)->accept_q, accept_q) {
|
|
|
|
new_sk = (struct sock *)isk;
|
|
|
|
|
|
|
|
smc_accept_unlink(new_sk);
|
|
|
|
if (new_sk->sk_state == SMC_CLOSED) {
|
2019-04-11 12:17:34 +03:00
|
|
|
new_sk->sk_prot->unhash(new_sk);
|
2018-01-26 11:28:49 +03:00
|
|
|
if (isk->clcsock) {
|
|
|
|
sock_release(isk->clcsock);
|
|
|
|
isk->clcsock = NULL;
|
|
|
|
}
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(new_sk); /* final */
|
2017-01-09 18:55:16 +03:00
|
|
|
continue;
|
|
|
|
}
|
2019-04-11 12:17:32 +03:00
|
|
|
if (new_sock) {
|
2017-01-09 18:55:16 +03:00
|
|
|
sock_graft(new_sk, new_sock);
|
2019-04-11 12:17:32 +03:00
|
|
|
if (isk->use_fallback) {
|
|
|
|
smc_sk(new_sk)->clcsock->file = new_sock->file;
|
|
|
|
isk->clcsock->file->private_data = isk->clcsock;
|
|
|
|
}
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
return new_sk;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* clean up for a created but never accepted sock */
|
2017-01-09 18:55:25 +03:00
|
|
|
void smc_close_non_accepted(struct sock *sk)
|
2017-01-09 18:55:16 +03:00
|
|
|
{
|
|
|
|
struct smc_sock *smc = smc_sk(sk);
|
|
|
|
|
2019-10-21 17:13:15 +03:00
|
|
|
sock_hold(sk); /* sock_put below */
|
2017-01-09 18:55:25 +03:00
|
|
|
lock_sock(sk);
|
|
|
|
if (!sk->sk_lingertime)
|
|
|
|
/* wait for peer closing */
|
|
|
|
sk->sk_lingertime = SMC_MAX_STREAM_WAIT_TIMEOUT;
|
2019-06-27 16:04:52 +03:00
|
|
|
__smc_release(smc);
|
2017-01-09 18:55:25 +03:00
|
|
|
release_sock(sk);
|
2019-10-21 17:13:15 +03:00
|
|
|
sock_put(sk); /* sock_hold above */
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(sk); /* final sock_put */
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
2020-04-29 18:10:41 +03:00
|
|
|
static int smcr_serv_conf_first_link(struct smc_sock *smc)
|
2017-01-09 18:55:21 +03:00
|
|
|
{
|
2020-04-29 18:10:40 +03:00
|
|
|
struct smc_link *link = smc->conn.lnk;
|
2020-04-30 16:55:42 +03:00
|
|
|
struct smc_llc_qentry *qentry;
|
2017-01-09 18:55:21 +03:00
|
|
|
int rc;
|
|
|
|
|
2020-05-01 13:48:01 +03:00
|
|
|
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc))
|
2018-07-25 17:35:32 +03:00
|
|
|
return SMC_CLC_DECL_ERR_REGRMB;
|
2017-07-28 14:56:17 +03:00
|
|
|
|
2017-01-09 18:55:21 +03:00
|
|
|
/* send CONFIRM LINK request to client over the RoCE fabric */
|
2018-07-25 17:35:30 +03:00
|
|
|
rc = smc_llc_send_confirm_link(link, SMC_LLC_REQ);
|
2017-01-09 18:55:21 +03:00
|
|
|
if (rc < 0)
|
2018-07-25 17:35:32 +03:00
|
|
|
return SMC_CLC_DECL_TIMEOUT_CL;
|
2017-01-09 18:55:21 +03:00
|
|
|
|
|
|
|
/* receive CONFIRM LINK response from client over the RoCE fabric */
|
2020-04-30 16:55:42 +03:00
|
|
|
qentry = smc_llc_wait(link->lgr, link, SMC_LLC_WAIT_TIME,
|
|
|
|
SMC_LLC_CONFIRM_LINK);
|
|
|
|
if (!qentry) {
|
2017-01-09 18:55:21 +03:00
|
|
|
struct smc_clc_msg_decline dclc;
|
|
|
|
|
|
|
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
2018-11-22 12:26:39 +03:00
|
|
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
2018-11-22 12:26:37 +03:00
|
|
|
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
2020-05-04 15:18:48 +03:00
|
|
|
smc_llc_save_peer_uid(qentry);
|
2020-04-30 16:55:42 +03:00
|
|
|
rc = smc_llc_eval_conf_link(qentry, SMC_LLC_RESP);
|
|
|
|
smc_llc_flow_qentry_del(&link->lgr->llc_flow_lcl);
|
|
|
|
if (rc)
|
2018-03-01 15:51:31 +03:00
|
|
|
return SMC_CLC_DECL_RMBE_EC;
|
|
|
|
|
2020-04-30 16:55:42 +03:00
|
|
|
/* confirm_rkey is implicit on 1st contact */
|
|
|
|
smc->conn.rmb_desc->is_conf_rkey = true;
|
2018-03-01 15:51:32 +03:00
|
|
|
|
2020-04-29 18:10:49 +03:00
|
|
|
smc_llc_link_active(link);
|
2020-05-05 16:01:20 +03:00
|
|
|
smcr_lgr_set_type(link->lgr, SMC_LGR_SINGLE);
|
2018-03-01 15:51:32 +03:00
|
|
|
|
2020-04-30 16:55:42 +03:00
|
|
|
/* initial contact - try to establish second link */
|
2020-05-03 15:38:43 +03:00
|
|
|
smc_llc_srv_add_link(link);
|
2018-03-01 15:51:31 +03:00
|
|
|
return 0;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: finish */
|
|
|
|
static void smc_listen_out(struct smc_sock *new_smc)
|
2017-01-09 18:55:16 +03:00
|
|
|
{
|
|
|
|
struct smc_sock *lsmc = new_smc->listen_smc;
|
|
|
|
struct sock *newsmcsk = &new_smc->sk;
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
if (lsmc->sk.sk_state == SMC_LISTEN) {
|
2019-04-11 12:17:30 +03:00
|
|
|
lock_sock_nested(&lsmc->sk, SINGLE_DEPTH_NESTING);
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_accept_enqueue(&lsmc->sk, newsmcsk);
|
2019-04-11 12:17:30 +03:00
|
|
|
release_sock(&lsmc->sk);
|
2018-05-18 10:34:18 +03:00
|
|
|
} else { /* no longer listening */
|
|
|
|
smc_close_non_accepted(newsmcsk);
|
2017-10-25 12:01:46 +03:00
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* Wake up accept */
|
|
|
|
lsmc->sk.sk_data_ready(&lsmc->sk);
|
|
|
|
sock_put(&lsmc->sk); /* sock_hold in smc_tcp_listen_work */
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: finish in state connected */
|
|
|
|
static void smc_listen_out_connected(struct smc_sock *new_smc)
|
|
|
|
{
|
|
|
|
struct sock *newsmcsk = &new_smc->sk;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
if (newsmcsk->sk_state == SMC_INIT)
|
|
|
|
newsmcsk->sk_state = SMC_ACTIVE;
|
|
|
|
|
|
|
|
smc_listen_out(new_smc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* listen worker: finish in error state */
|
|
|
|
static void smc_listen_out_err(struct smc_sock *new_smc)
|
|
|
|
{
|
|
|
|
struct sock *newsmcsk = &new_smc->sk;
|
2021-06-16 17:52:58 +03:00
|
|
|
struct net *net = sock_net(newsmcsk);
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2021-06-16 17:52:58 +03:00
|
|
|
this_cpu_inc(net->smc.smc_stats->srv_hshake_err_cnt);
|
2018-05-18 10:34:18 +03:00
|
|
|
if (newsmcsk->sk_state == SMC_INIT)
|
|
|
|
sock_put(&new_smc->sk); /* passive closing */
|
|
|
|
newsmcsk->sk_state = SMC_CLOSED;
|
|
|
|
|
|
|
|
smc_listen_out(new_smc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* listen worker: decline and fall back if possible */
|
|
|
|
static void smc_listen_decline(struct smc_sock *new_smc, int reason_code,
|
2020-10-23 21:48:28 +03:00
|
|
|
int local_first, u8 version)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
|
|
|
/* RDMA setup failed, switch back to TCP */
|
2020-12-01 22:20:36 +03:00
|
|
|
smc_conn_abort(new_smc, local_first);
|
2022-01-22 12:43:09 +03:00
|
|
|
if (reason_code < 0 ||
|
|
|
|
smc_switch_to_fallback(new_smc, reason_code)) {
|
|
|
|
/* error, no fallback possible */
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_listen_out_err(new_smc);
|
|
|
|
return;
|
|
|
|
}
|
2018-07-25 17:35:32 +03:00
|
|
|
if (reason_code && reason_code != SMC_CLC_DECL_PEERDECL) {
|
2020-09-26 13:44:32 +03:00
|
|
|
if (smc_clc_send_decline(new_smc, reason_code, version) < 0) {
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_listen_out_err(new_smc);
|
|
|
|
return;
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_listen_out_connected(new_smc);
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
/* listen worker: version checking */
|
|
|
|
static int smc_listen_v2_check(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_proposal *pclc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
struct smc_clc_smcd_v2_extension *pclc_smcd_v2_ext;
|
|
|
|
struct smc_clc_v2_extension *pclc_v2_ext;
|
2020-10-31 21:19:38 +03:00
|
|
|
int rc = SMC_CLC_DECL_PEERNOSMC;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
ini->smc_type_v1 = pclc->hdr.typev1;
|
|
|
|
ini->smc_type_v2 = pclc->hdr.typev2;
|
|
|
|
ini->smcd_version = ini->smc_type_v1 != SMC_TYPE_N ? SMC_V1 : 0;
|
|
|
|
if (pclc->hdr.version > SMC_V1)
|
|
|
|
ini->smcd_version |=
|
|
|
|
ini->smc_type_v2 != SMC_TYPE_N ? SMC_V2 : 0;
|
2020-10-31 21:19:38 +03:00
|
|
|
if (!(ini->smcd_version & SMC_V2)) {
|
|
|
|
rc = SMC_CLC_DECL_PEERNOSMC;
|
|
|
|
goto out;
|
|
|
|
}
|
2020-12-01 22:20:42 +03:00
|
|
|
if (!smc_ism_is_v2_capable()) {
|
2020-09-26 13:44:29 +03:00
|
|
|
ini->smcd_version &= ~SMC_V2;
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = SMC_CLC_DECL_NOISM2SUPP;
|
2020-09-26 13:44:29 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
pclc_v2_ext = smc_get_clc_v2_ext(pclc);
|
|
|
|
if (!pclc_v2_ext) {
|
|
|
|
ini->smcd_version &= ~SMC_V2;
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = SMC_CLC_DECL_NOV2EXT;
|
2020-09-26 13:44:29 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
pclc_smcd_v2_ext = smc_get_clc_smcd_v2_ext(pclc_v2_ext);
|
2020-10-31 21:19:38 +03:00
|
|
|
if (!pclc_smcd_v2_ext) {
|
2020-09-26 13:44:29 +03:00
|
|
|
ini->smcd_version &= ~SMC_V2;
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = SMC_CLC_DECL_NOV2DEXT;
|
|
|
|
}
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
out:
|
2020-10-31 21:19:38 +03:00
|
|
|
if (!ini->smcd_version)
|
|
|
|
return rc;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: check prefixes */
|
2019-04-12 13:57:25 +03:00
|
|
|
static int smc_listen_prfx_check(struct smc_sock *new_smc,
|
2018-05-18 10:34:18 +03:00
|
|
|
struct smc_clc_msg_proposal *pclc)
|
|
|
|
{
|
|
|
|
struct smc_clc_msg_proposal_prefix *pclc_prfx;
|
|
|
|
struct socket *newclcsock = new_smc->clcsock;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
if (pclc->hdr.typev1 == SMC_TYPE_N)
|
|
|
|
return 0;
|
2017-12-07 15:38:49 +03:00
|
|
|
pclc_prfx = smc_clc_proposal_get_prefix(pclc);
|
2018-05-18 10:34:18 +03:00
|
|
|
if (smc_clc_prfx_match(newclcsock, pclc_prfx))
|
2019-04-12 13:57:25 +03:00
|
|
|
return SMC_CLC_DECL_DIFFPREFIX;
|
2018-03-16 17:06:39 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: initialize connection and buffers */
|
|
|
|
static int smc_listen_rdma_init(struct smc_sock *new_smc,
|
2019-04-12 13:57:30 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2019-04-12 13:57:30 +03:00
|
|
|
int rc;
|
|
|
|
|
2017-01-09 18:55:17 +03:00
|
|
|
/* allocate connection / link group */
|
2019-04-12 13:57:30 +03:00
|
|
|
rc = smc_conn_create(new_smc, ini);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2017-07-28 14:56:20 +03:00
|
|
|
/* create send buffer and rmb */
|
net/smc: add base infrastructure for SMC-D and ISM
SMC supports two variants: SMC-R and SMC-D. For data transport, SMC-R
uses RDMA devices, SMC-D uses so-called Internal Shared Memory (ISM)
devices. An ISM device only allows shared memory communication between
SMC instances on the same machine. For example, this allows virtual
machines on the same host to communicate via SMC without RDMA devices.
This patch adds the base infrastructure for SMC-D and ISM devices to
the existing SMC code. It contains the following:
* ISM driver interface:
This interface allows an ISM driver to register ISM devices in SMC. In
the process, the driver provides a set of device ops for each device.
SMC uses these ops to execute SMC specific operations on or transfer
data over the device.
* Core SMC-D link group, connection, and buffer support:
Link groups, SMC connections and SMC buffers (in smc_core) are
extended to support SMC-D.
* SMC type checks:
Some type checks are added to prevent using SMC-R specific code for
SMC-D and vice versa.
To actually use SMC-D, additional changes to pnetid, CLC, CDC, etc. are
required. These are added in follow-up patches.
Signed-off-by: Hans Wippel <hwippel@linux.ibm.com>
Signed-off-by: Ursula Braun <ubraun@linux.ibm.com>
Suggested-by: Thomas Richter <tmricht@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2018-06-28 20:05:07 +03:00
|
|
|
if (smc_buf_create(new_smc, false))
|
2018-05-18 10:34:18 +03:00
|
|
|
return SMC_CLC_DECL_MEM;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-28 20:05:11 +03:00
|
|
|
/* listen worker: initialize connection and buffers for SMC-D */
|
|
|
|
static int smc_listen_ism_init(struct smc_sock *new_smc,
|
2019-04-12 13:57:30 +03:00
|
|
|
struct smc_init_info *ini)
|
2018-06-28 20:05:11 +03:00
|
|
|
{
|
2019-04-12 13:57:30 +03:00
|
|
|
int rc;
|
2018-06-28 20:05:11 +03:00
|
|
|
|
2019-04-12 13:57:30 +03:00
|
|
|
rc = smc_conn_create(new_smc, ini);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
2018-06-28 20:05:11 +03:00
|
|
|
|
|
|
|
/* Create send and receive buffers */
|
2020-07-26 21:34:28 +03:00
|
|
|
rc = smc_buf_create(new_smc, true);
|
|
|
|
if (rc) {
|
2020-12-01 22:20:36 +03:00
|
|
|
smc_conn_abort(new_smc, ini->first_contact_local);
|
2020-07-26 21:34:28 +03:00
|
|
|
return (rc == -ENOSPC) ? SMC_CLC_DECL_MAX_DMB :
|
|
|
|
SMC_CLC_DECL_MEM;
|
2018-06-28 20:05:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
static bool smc_is_already_selected(struct smcd_dev *smcd,
|
|
|
|
struct smc_init_info *ini,
|
|
|
|
int matches)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < matches; i++)
|
|
|
|
if (smcd == ini->ism_dev[i])
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check for ISM devices matching proposed ISM devices */
|
|
|
|
static void smc_check_ism_v2_match(struct smc_init_info *ini,
|
|
|
|
u16 proposed_chid, u64 proposed_gid,
|
|
|
|
unsigned int *matches)
|
|
|
|
{
|
|
|
|
struct smcd_dev *smcd;
|
|
|
|
|
|
|
|
list_for_each_entry(smcd, &smcd_dev_list.list, list) {
|
|
|
|
if (smcd->going_away)
|
|
|
|
continue;
|
|
|
|
if (smc_is_already_selected(smcd, ini, *matches))
|
|
|
|
continue;
|
|
|
|
if (smc_ism_get_chid(smcd) == proposed_chid &&
|
|
|
|
!smc_ism_cantalk(proposed_gid, ISM_RESERVED_VLANID, smcd)) {
|
|
|
|
ini->ism_peer_gid[*matches] = proposed_gid;
|
|
|
|
ini->ism_dev[*matches] = smcd;
|
|
|
|
(*matches)++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-31 21:19:38 +03:00
|
|
|
static void smc_find_ism_store_rc(u32 rc, struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
if (!ini->rc)
|
|
|
|
ini->rc = rc;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
static void smc_find_ism_v2_device_serv(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_proposal *pclc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
|
|
|
struct smc_clc_smcd_v2_extension *smcd_v2_ext;
|
|
|
|
struct smc_clc_v2_extension *smc_v2_ext;
|
|
|
|
struct smc_clc_msg_smcd *pclc_smcd;
|
|
|
|
unsigned int matches = 0;
|
2020-10-07 23:57:43 +03:00
|
|
|
u8 smcd_version;
|
2020-09-26 13:44:29 +03:00
|
|
|
u8 *eid = NULL;
|
2020-10-31 21:19:38 +03:00
|
|
|
int i, rc;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
if (!(ini->smcd_version & SMC_V2) || !smcd_indicated(ini->smc_type_v2))
|
2020-10-07 23:57:43 +03:00
|
|
|
goto not_found;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
pclc_smcd = smc_get_clc_msg_smcd(pclc);
|
|
|
|
smc_v2_ext = smc_get_clc_v2_ext(pclc);
|
|
|
|
smcd_v2_ext = smc_get_clc_smcd_v2_ext(smc_v2_ext);
|
|
|
|
if (!smcd_v2_ext ||
|
2020-10-31 21:19:38 +03:00
|
|
|
!smc_v2_ext->hdr.flag.seid) { /* no system EID support for SMCD */
|
|
|
|
smc_find_ism_store_rc(SMC_CLC_DECL_NOSEID, ini);
|
2020-09-26 13:44:29 +03:00
|
|
|
goto not_found;
|
2020-10-31 21:19:38 +03:00
|
|
|
}
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
mutex_lock(&smcd_dev_list.mutex);
|
|
|
|
if (pclc_smcd->ism.chid)
|
|
|
|
/* check for ISM device matching proposed native ISM device */
|
|
|
|
smc_check_ism_v2_match(ini, ntohs(pclc_smcd->ism.chid),
|
|
|
|
ntohll(pclc_smcd->ism.gid), &matches);
|
|
|
|
for (i = 1; i <= smc_v2_ext->hdr.ism_gid_cnt; i++) {
|
|
|
|
/* check for ISM devices matching proposed non-native ISM
|
|
|
|
* devices
|
|
|
|
*/
|
|
|
|
smc_check_ism_v2_match(ini,
|
|
|
|
ntohs(smcd_v2_ext->gidchid[i - 1].chid),
|
|
|
|
ntohll(smcd_v2_ext->gidchid[i - 1].gid),
|
|
|
|
&matches);
|
|
|
|
}
|
|
|
|
mutex_unlock(&smcd_dev_list.mutex);
|
|
|
|
|
|
|
|
if (ini->ism_dev[0]) {
|
|
|
|
smc_ism_get_system_eid(ini->ism_dev[0], &eid);
|
|
|
|
if (memcmp(eid, smcd_v2_ext->system_eid, SMC_MAX_EID_LEN))
|
|
|
|
goto not_found;
|
|
|
|
} else {
|
|
|
|
goto not_found;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* separate - outside the smcd_dev_list.lock */
|
2020-10-07 23:57:43 +03:00
|
|
|
smcd_version = ini->smcd_version;
|
2020-09-26 13:44:29 +03:00
|
|
|
for (i = 0; i < matches; i++) {
|
|
|
|
ini->smcd_version = SMC_V2;
|
|
|
|
ini->is_smcd = true;
|
|
|
|
ini->ism_selected = i;
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = smc_listen_ism_init(new_smc, ini);
|
|
|
|
if (rc) {
|
|
|
|
smc_find_ism_store_rc(rc, ini);
|
2020-09-26 13:44:29 +03:00
|
|
|
/* try next active ISM device */
|
|
|
|
continue;
|
2020-10-31 21:19:38 +03:00
|
|
|
}
|
2020-09-26 13:44:29 +03:00
|
|
|
return; /* matching and usable V2 ISM device found */
|
|
|
|
}
|
2020-10-07 23:57:43 +03:00
|
|
|
/* no V2 ISM device could be initialized */
|
|
|
|
ini->smcd_version = smcd_version; /* restore original value */
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
not_found:
|
|
|
|
ini->smcd_version &= ~SMC_V2;
|
|
|
|
ini->ism_dev[0] = NULL;
|
|
|
|
ini->is_smcd = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_find_ism_v1_device_serv(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_proposal *pclc,
|
|
|
|
struct smc_init_info *ini)
|
2020-09-26 13:44:21 +03:00
|
|
|
{
|
|
|
|
struct smc_clc_msg_smcd *pclc_smcd = smc_get_clc_msg_smcd(pclc);
|
2020-10-31 21:19:38 +03:00
|
|
|
int rc = 0;
|
2020-09-26 13:44:21 +03:00
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
/* check if ISM V1 is available */
|
|
|
|
if (!(ini->smcd_version & SMC_V1) || !smcd_indicated(ini->smc_type_v1))
|
2020-09-26 13:44:21 +03:00
|
|
|
goto not_found;
|
|
|
|
ini->is_smcd = true; /* prepare ISM check */
|
2020-09-26 13:44:28 +03:00
|
|
|
ini->ism_peer_gid[0] = ntohll(pclc_smcd->ism.gid);
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = smc_find_ism_device(new_smc, ini);
|
|
|
|
if (rc)
|
2020-09-26 13:44:21 +03:00
|
|
|
goto not_found;
|
2020-09-26 13:44:29 +03:00
|
|
|
ini->ism_selected = 0;
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = smc_listen_ism_init(new_smc, ini);
|
|
|
|
if (!rc)
|
2020-09-26 13:44:29 +03:00
|
|
|
return; /* V1 ISM device found */
|
2020-09-26 13:44:21 +03:00
|
|
|
|
|
|
|
not_found:
|
2020-10-31 21:19:38 +03:00
|
|
|
smc_find_ism_store_rc(rc, ini);
|
2020-09-26 13:44:23 +03:00
|
|
|
ini->ism_dev[0] = NULL;
|
2020-09-26 13:44:21 +03:00
|
|
|
ini->is_smcd = false;
|
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: register buffers */
|
2020-09-10 19:48:21 +03:00
|
|
|
static int smc_listen_rdma_reg(struct smc_sock *new_smc, bool local_first)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2020-04-29 18:10:41 +03:00
|
|
|
struct smc_connection *conn = &new_smc->conn;
|
2017-04-10 15:58:01 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (!local_first) {
|
2020-05-01 13:48:01 +03:00
|
|
|
if (smcr_lgr_reg_rmbs(conn->lnk, conn->rmb_desc))
|
2018-11-22 12:26:43 +03:00
|
|
|
return SMC_CLC_DECL_ERR_REGRMB;
|
2017-07-28 14:56:17 +03:00
|
|
|
}
|
2017-07-28 14:56:22 +03:00
|
|
|
smc_rmb_sync_sg_for_device(&new_smc->conn);
|
2017-07-28 14:56:17 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
static int smc_find_rdma_v1_device_serv(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_proposal *pclc,
|
|
|
|
struct smc_init_info *ini)
|
2020-09-26 13:44:21 +03:00
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
if (!smcr_indicated(ini->smc_type_v1))
|
2020-09-26 13:44:21 +03:00
|
|
|
return SMC_CLC_DECL_NOSMCDEV;
|
|
|
|
|
|
|
|
/* prepare RDMA check */
|
|
|
|
ini->ib_lcl = &pclc->lcl;
|
|
|
|
rc = smc_find_rdma_device(new_smc, ini);
|
|
|
|
if (rc) {
|
|
|
|
/* no RDMA device found */
|
2020-09-26 13:44:29 +03:00
|
|
|
if (ini->smc_type_v1 == SMC_TYPE_B)
|
2020-09-26 13:44:21 +03:00
|
|
|
/* neither ISM nor RDMA device found */
|
|
|
|
rc = SMC_CLC_DECL_NOSMCDEV;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
rc = smc_listen_rdma_init(new_smc, ini);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
return smc_listen_rdma_reg(new_smc, ini->first_contact_local);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* determine the local device matching to proposal */
|
|
|
|
static int smc_listen_find_device(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_proposal *pclc,
|
|
|
|
struct smc_init_info *ini)
|
|
|
|
{
|
2020-09-26 13:44:29 +03:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* check for ISM device matching V2 proposed device */
|
|
|
|
smc_find_ism_v2_device_serv(new_smc, pclc, ini);
|
|
|
|
if (ini->ism_dev[0])
|
2020-09-26 13:44:21 +03:00
|
|
|
return 0;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
if (!(ini->smcd_version & SMC_V1))
|
2020-10-31 21:19:38 +03:00
|
|
|
return ini->rc ?: SMC_CLC_DECL_NOSMCD2DEV;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
/* check for matching IP prefix and subnet length */
|
|
|
|
rc = smc_listen_prfx_check(new_smc, pclc);
|
|
|
|
if (rc)
|
2020-10-31 21:19:38 +03:00
|
|
|
return ini->rc ?: rc;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
/* get vlan id from IP device */
|
|
|
|
if (smc_vlan_by_tcpsk(new_smc->clcsock, ini))
|
2020-10-31 21:19:38 +03:00
|
|
|
return ini->rc ?: SMC_CLC_DECL_GETVLANERR;
|
2020-09-26 13:44:29 +03:00
|
|
|
|
|
|
|
/* check for ISM device matching V1 proposed device */
|
|
|
|
smc_find_ism_v1_device_serv(new_smc, pclc, ini);
|
|
|
|
if (ini->ism_dev[0])
|
|
|
|
return 0;
|
|
|
|
|
2020-09-26 13:44:21 +03:00
|
|
|
if (pclc->hdr.typev1 == SMC_TYPE_D)
|
2020-10-31 21:19:38 +03:00
|
|
|
/* skip RDMA and decline */
|
|
|
|
return ini->rc ?: SMC_CLC_DECL_NOSMCDDEV;
|
2020-09-26 13:44:21 +03:00
|
|
|
|
|
|
|
/* check if RDMA is available */
|
2020-10-31 21:19:38 +03:00
|
|
|
rc = smc_find_rdma_v1_device_serv(new_smc, pclc, ini);
|
|
|
|
smc_find_ism_store_rc(rc, ini);
|
|
|
|
|
|
|
|
return (!rc) ? 0 : ini->rc;
|
2020-09-26 13:44:21 +03:00
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* listen worker: finish RDMA setup */
|
2018-09-18 16:46:35 +03:00
|
|
|
static int smc_listen_rdma_finish(struct smc_sock *new_smc,
|
|
|
|
struct smc_clc_msg_accept_confirm *cclc,
|
2020-09-10 19:48:21 +03:00
|
|
|
bool local_first)
|
2018-05-18 10:34:18 +03:00
|
|
|
{
|
2020-04-29 18:10:40 +03:00
|
|
|
struct smc_link *link = new_smc->conn.lnk;
|
2018-05-18 10:34:18 +03:00
|
|
|
int reason_code = 0;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (local_first)
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_link_save_peer_info(link, cclc);
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-10 19:48:25 +03:00
|
|
|
if (smc_rmb_rtoken_handling(&new_smc->conn, link, cclc))
|
|
|
|
return SMC_CLC_DECL_ERR_RTOK;
|
2017-01-09 18:55:20 +03:00
|
|
|
|
2020-09-10 19:48:21 +03:00
|
|
|
if (local_first) {
|
2020-09-10 19:48:25 +03:00
|
|
|
if (smc_ib_ready_link(link))
|
|
|
|
return SMC_CLC_DECL_ERR_RDYLNK;
|
2017-01-09 18:55:21 +03:00
|
|
|
/* QP confirmation over RoCE fabric */
|
2020-04-30 16:55:42 +03:00
|
|
|
smc_llc_flow_initiate(link->lgr, SMC_LLC_FLOW_ADD_LINK);
|
2020-04-29 18:10:41 +03:00
|
|
|
reason_code = smcr_serv_conf_first_link(new_smc);
|
2020-04-30 16:55:42 +03:00
|
|
|
smc_llc_flow_stop(link->lgr, &link->lgr->llc_flow_lcl);
|
2017-01-09 18:55:20 +03:00
|
|
|
}
|
2018-09-18 16:46:35 +03:00
|
|
|
return reason_code;
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
2017-01-09 18:55:23 +03:00
|
|
|
|
2020-09-26 13:44:21 +03:00
|
|
|
/* setup for connection of server */
|
2018-05-18 10:34:18 +03:00
|
|
|
static void smc_listen_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct smc_sock *new_smc = container_of(work, struct smc_sock,
|
|
|
|
smc_listen_work);
|
2020-12-01 22:20:42 +03:00
|
|
|
u8 version = smc_ism_is_v2_capable() ? SMC_V2 : SMC_V1;
|
2018-05-18 10:34:18 +03:00
|
|
|
struct socket *newclcsock = new_smc->clcsock;
|
2020-09-26 13:44:30 +03:00
|
|
|
struct smc_clc_msg_accept_confirm *cclc;
|
2020-09-10 19:48:22 +03:00
|
|
|
struct smc_clc_msg_proposal_area *buf;
|
2018-05-18 10:34:18 +03:00
|
|
|
struct smc_clc_msg_proposal *pclc;
|
2020-09-26 13:44:23 +03:00
|
|
|
struct smc_init_info *ini = NULL;
|
2018-05-18 10:34:18 +03:00
|
|
|
int rc = 0;
|
|
|
|
|
2019-04-11 12:17:30 +03:00
|
|
|
if (new_smc->listen_smc->sk.sk_state != SMC_LISTEN)
|
|
|
|
return smc_listen_out_err(new_smc);
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
if (new_smc->use_fallback) {
|
|
|
|
smc_listen_out_connected(new_smc);
|
|
|
|
return;
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* check if peer is smc capable */
|
|
|
|
if (!tcp_sk(newclcsock->sk)->syn_smc) {
|
2022-01-22 12:43:09 +03:00
|
|
|
rc = smc_switch_to_fallback(new_smc, SMC_CLC_DECL_PEERNOSMC);
|
|
|
|
if (rc)
|
|
|
|
smc_listen_out_err(new_smc);
|
|
|
|
else
|
|
|
|
smc_listen_out_connected(new_smc);
|
2018-05-18 10:34:18 +03:00
|
|
|
return;
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* do inband token exchange -
|
|
|
|
* wait for and receive SMC Proposal CLC message
|
|
|
|
*/
|
2020-09-10 19:48:22 +03:00
|
|
|
buf = kzalloc(sizeof(*buf), GFP_KERNEL);
|
|
|
|
if (!buf) {
|
|
|
|
rc = SMC_CLC_DECL_MEM;
|
|
|
|
goto out_decl;
|
|
|
|
}
|
|
|
|
pclc = (struct smc_clc_msg_proposal *)buf;
|
|
|
|
rc = smc_clc_wait_msg(new_smc, pclc, sizeof(*buf),
|
2019-04-12 13:57:28 +03:00
|
|
|
SMC_CLC_PROPOSAL, CLC_WAIT_TIME);
|
2019-04-12 13:57:29 +03:00
|
|
|
if (rc)
|
|
|
|
goto out_decl;
|
2020-09-26 13:44:32 +03:00
|
|
|
version = pclc->hdr.version == SMC_V1 ? SMC_V1 : version;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2020-09-26 13:44:32 +03:00
|
|
|
/* IPSec connections opt out of SMC optimizations */
|
2018-05-18 10:34:18 +03:00
|
|
|
if (using_ipsec(new_smc)) {
|
2019-04-12 13:57:29 +03:00
|
|
|
rc = SMC_CLC_DECL_IPSEC;
|
|
|
|
goto out_decl;
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:23 +03:00
|
|
|
ini = kzalloc(sizeof(*ini), GFP_KERNEL);
|
|
|
|
if (!ini) {
|
|
|
|
rc = SMC_CLC_DECL_MEM;
|
|
|
|
goto out_decl;
|
|
|
|
}
|
|
|
|
|
2020-09-26 13:44:29 +03:00
|
|
|
/* initial version checking */
|
|
|
|
rc = smc_listen_v2_check(new_smc, pclc, ini);
|
|
|
|
if (rc)
|
2019-04-12 13:57:29 +03:00
|
|
|
goto out_decl;
|
2019-04-12 13:57:27 +03:00
|
|
|
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_lock(&smc_server_lgr_pending);
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_close_init(new_smc);
|
|
|
|
smc_rx_init(new_smc);
|
|
|
|
smc_tx_init(new_smc);
|
|
|
|
|
2020-09-26 13:44:21 +03:00
|
|
|
/* determine ISM or RoCE device used for connection */
|
2020-09-26 13:44:23 +03:00
|
|
|
rc = smc_listen_find_device(new_smc, pclc, ini);
|
2020-09-26 13:44:21 +03:00
|
|
|
if (rc)
|
|
|
|
goto out_unlock;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
|
|
|
/* send SMC Accept CLC message */
|
2020-09-26 13:44:30 +03:00
|
|
|
rc = smc_clc_send_accept(new_smc, ini->first_contact_local,
|
|
|
|
ini->smcd_version == SMC_V2 ? SMC_V2 : SMC_V1);
|
2019-04-12 13:57:29 +03:00
|
|
|
if (rc)
|
|
|
|
goto out_unlock;
|
2018-05-18 10:34:18 +03:00
|
|
|
|
2019-02-07 17:56:17 +03:00
|
|
|
/* SMC-D does not need this lock any more */
|
2020-09-26 13:44:23 +03:00
|
|
|
if (ini->is_smcd)
|
2019-02-07 17:56:18 +03:00
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
2019-02-07 17:56:17 +03:00
|
|
|
|
2018-05-18 10:34:18 +03:00
|
|
|
/* receive SMC Confirm CLC message */
|
2020-10-07 23:57:42 +03:00
|
|
|
memset(buf, 0, sizeof(*buf));
|
|
|
|
cclc = (struct smc_clc_msg_accept_confirm *)buf;
|
|
|
|
rc = smc_clc_wait_msg(new_smc, cclc, sizeof(*buf),
|
2019-04-12 13:57:28 +03:00
|
|
|
SMC_CLC_CONFIRM, CLC_WAIT_TIME);
|
|
|
|
if (rc) {
|
2020-09-26 13:44:23 +03:00
|
|
|
if (!ini->is_smcd)
|
2019-04-12 13:57:29 +03:00
|
|
|
goto out_unlock;
|
|
|
|
goto out_decl;
|
2018-05-18 10:34:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* finish worker */
|
2020-09-26 13:44:23 +03:00
|
|
|
if (!ini->is_smcd) {
|
2020-09-26 13:44:30 +03:00
|
|
|
rc = smc_listen_rdma_finish(new_smc, cclc,
|
2020-09-26 13:44:23 +03:00
|
|
|
ini->first_contact_local);
|
2019-02-07 17:56:17 +03:00
|
|
|
if (rc)
|
2020-09-10 19:48:25 +03:00
|
|
|
goto out_unlock;
|
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
2018-09-18 16:46:35 +03:00
|
|
|
}
|
2020-09-26 13:44:30 +03:00
|
|
|
smc_conn_save_peer_info(new_smc, cclc);
|
2018-05-18 10:34:18 +03:00
|
|
|
smc_listen_out_connected(new_smc);
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_SERV_SUCC_INC(sock_net(newclcsock->sk), ini);
|
2020-09-17 23:46:02 +03:00
|
|
|
goto out_free;
|
2019-04-12 13:57:29 +03:00
|
|
|
|
|
|
|
out_unlock:
|
|
|
|
mutex_unlock(&smc_server_lgr_pending);
|
|
|
|
out_decl:
|
2020-10-23 21:48:28 +03:00
|
|
|
smc_listen_decline(new_smc, rc, ini ? ini->first_contact_local : 0,
|
|
|
|
version);
|
2020-09-17 23:46:02 +03:00
|
|
|
out_free:
|
2020-09-26 13:44:23 +03:00
|
|
|
kfree(ini);
|
2020-09-10 19:48:22 +03:00
|
|
|
kfree(buf);
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_tcp_listen_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct smc_sock *lsmc = container_of(work, struct smc_sock,
|
|
|
|
tcp_listen_work);
|
2018-01-24 12:28:12 +03:00
|
|
|
struct sock *lsk = &lsmc->sk;
|
2017-01-09 18:55:16 +03:00
|
|
|
struct smc_sock *new_smc;
|
|
|
|
int rc = 0;
|
|
|
|
|
2018-01-24 12:28:12 +03:00
|
|
|
lock_sock(lsk);
|
|
|
|
while (lsk->sk_state == SMC_LISTEN) {
|
2017-01-09 18:55:16 +03:00
|
|
|
rc = smc_clcsock_accept(lsmc, &new_smc);
|
2020-09-10 19:48:20 +03:00
|
|
|
if (rc) /* clcsock accept queue empty or error */
|
2017-01-09 18:55:16 +03:00
|
|
|
goto out;
|
|
|
|
if (!new_smc)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
new_smc->listen_smc = lsmc;
|
2018-04-26 18:18:21 +03:00
|
|
|
new_smc->use_fallback = lsmc->use_fallback;
|
2018-07-25 17:35:32 +03:00
|
|
|
new_smc->fallback_rsn = lsmc->fallback_rsn;
|
2018-01-24 12:28:12 +03:00
|
|
|
sock_hold(lsk); /* sock_put in smc_listen_work */
|
2017-01-09 18:55:16 +03:00
|
|
|
INIT_WORK(&new_smc->smc_listen_work, smc_listen_work);
|
|
|
|
smc_copy_sock_settings_to_smc(new_smc);
|
2018-08-08 15:13:20 +03:00
|
|
|
new_smc->sk.sk_sndbuf = lsmc->sk.sk_sndbuf;
|
|
|
|
new_smc->sk.sk_rcvbuf = lsmc->sk.sk_rcvbuf;
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_hold(&new_smc->sk); /* sock_put in passive closing */
|
2020-09-10 19:48:29 +03:00
|
|
|
if (!queue_work(smc_hs_wq, &new_smc->smc_listen_work))
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(&new_smc->sk);
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
2018-01-24 12:28:12 +03:00
|
|
|
release_sock(lsk);
|
2020-09-10 19:48:20 +03:00
|
|
|
sock_put(&lsmc->sk); /* sock_hold in smc_clcsock_data_ready() */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void smc_clcsock_data_ready(struct sock *listen_clcsock)
|
|
|
|
{
|
net/smc: Forward wakeup to smc socket waitqueue after fallback
commit 341adeec9adad0874f29a0a1af35638207352a39 upstream.
When we replace TCP with SMC and a fallback occurs, there may be
some socket waitqueue entries remaining in smc socket->wq, such
as eppoll_entries inserted by userspace applications.
After the fallback, data flows over TCP/IP and only clcsocket->wq
will be woken up. Applications can't be notified by the entries
which were inserted in smc socket->wq before fallback. So we need
a mechanism to wake up smc socket->wq at the same time if some
entries remaining in it.
The current workaround is to transfer the entries from smc socket->wq
to clcsock->wq during the fallback. But this may cause a crash
like this:
general protection fault, probably for non-canonical address 0xdead000000000100: 0000 [#1] PREEMPT SMP PTI
CPU: 3 PID: 0 Comm: swapper/3 Kdump: loaded Tainted: G E 5.16.0+ #107
RIP: 0010:__wake_up_common+0x65/0x170
Call Trace:
<IRQ>
__wake_up_common_lock+0x7a/0xc0
sock_def_readable+0x3c/0x70
tcp_data_queue+0x4a7/0xc40
tcp_rcv_established+0x32f/0x660
? sk_filter_trim_cap+0xcb/0x2e0
tcp_v4_do_rcv+0x10b/0x260
tcp_v4_rcv+0xd2a/0xde0
ip_protocol_deliver_rcu+0x3b/0x1d0
ip_local_deliver_finish+0x54/0x60
ip_local_deliver+0x6a/0x110
? tcp_v4_early_demux+0xa2/0x140
? tcp_v4_early_demux+0x10d/0x140
ip_sublist_rcv_finish+0x49/0x60
ip_sublist_rcv+0x19d/0x230
ip_list_rcv+0x13e/0x170
__netif_receive_skb_list_core+0x1c2/0x240
netif_receive_skb_list_internal+0x1e6/0x320
napi_complete_done+0x11d/0x190
mlx5e_napi_poll+0x163/0x6b0 [mlx5_core]
__napi_poll+0x3c/0x1b0
net_rx_action+0x27c/0x300
__do_softirq+0x114/0x2d2
irq_exit_rcu+0xb4/0xe0
common_interrupt+0xba/0xe0
</IRQ>
<TASK>
The crash is caused by privately transferring waitqueue entries from
smc socket->wq to clcsock->wq. The owners of these entries, such as
epoll, have no idea that the entries have been transferred to a
different socket wait queue and still use original waitqueue spinlock
(smc socket->wq.wait.lock) to make the entries operation exclusive,
but it doesn't work. The operations to the entries, such as removing
from the waitqueue (now is clcsock->wq after fallback), may cause a
crash when clcsock waitqueue is being iterated over at the moment.
This patch tries to fix this by no longer transferring wait queue
entries privately, but introducing own implementations of clcsock's
callback functions in fallback situation. The callback functions will
forward the wakeup to smc socket->wq if clcsock->wq is actually woken
up and smc socket->wq has remaining entries.
Fixes: 2153bd1e3d3d ("net/smc: Transfer remaining wait queue entries during fallback")
Suggested-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: Wen Gu <guwen@linux.alibaba.com>
Acked-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2022-01-26 18:33:04 +03:00
|
|
|
struct smc_sock *lsmc =
|
|
|
|
smc_clcsock_user_data(listen_clcsock);
|
2020-09-10 19:48:20 +03:00
|
|
|
|
|
|
|
if (!lsmc)
|
|
|
|
return;
|
|
|
|
lsmc->clcsk_data_ready(listen_clcsock);
|
|
|
|
if (lsmc->sk.sk_state == SMC_LISTEN) {
|
|
|
|
sock_hold(&lsmc->sk); /* sock_put in smc_tcp_listen_work() */
|
2020-09-10 19:48:29 +03:00
|
|
|
if (!queue_work(smc_hs_wq, &lsmc->tcp_listen_work))
|
2020-09-10 19:48:20 +03:00
|
|
|
sock_put(&lsmc->sk);
|
|
|
|
}
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
static int smc_listen(struct socket *sock, int backlog)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
lock_sock(sk);
|
|
|
|
|
|
|
|
rc = -EINVAL;
|
2019-08-02 11:47:50 +03:00
|
|
|
if ((sk->sk_state != SMC_INIT && sk->sk_state != SMC_LISTEN) ||
|
|
|
|
smc->connect_nonblock)
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
rc = 0;
|
|
|
|
if (sk->sk_state == SMC_LISTEN) {
|
|
|
|
sk->sk_max_ack_backlog = backlog;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
/* some socket options are handled in core, so we could not apply
|
|
|
|
* them to the clc socket -- copy smc socket options to clc socket
|
|
|
|
*/
|
|
|
|
smc_copy_sock_settings_to_clc(smc);
|
2018-04-26 18:18:21 +03:00
|
|
|
if (!smc->use_fallback)
|
|
|
|
tcp_sk(smc->clcsock->sk)->syn_smc = 1;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2020-09-10 19:48:20 +03:00
|
|
|
/* save original sk_data_ready function and establish
|
|
|
|
* smc-specific sk_data_ready function
|
|
|
|
*/
|
|
|
|
smc->clcsk_data_ready = smc->clcsock->sk->sk_data_ready;
|
|
|
|
smc->clcsock->sk->sk_data_ready = smc_clcsock_data_ready;
|
|
|
|
smc->clcsock->sk->sk_user_data =
|
|
|
|
(void *)((uintptr_t)smc | SK_USER_DATA_NOCOPY);
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = kernel_listen(smc->clcsock, backlog);
|
2021-11-24 15:32:38 +03:00
|
|
|
if (rc) {
|
|
|
|
smc->clcsock->sk->sk_data_ready = smc->clcsk_data_ready;
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
2021-11-24 15:32:38 +03:00
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
sk->sk_max_ack_backlog = backlog;
|
|
|
|
sk->sk_ack_backlog = 0;
|
|
|
|
sk->sk_state = SMC_LISTEN;
|
|
|
|
|
|
|
|
out:
|
|
|
|
release_sock(sk);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_accept(struct socket *sock, struct socket *new_sock,
|
2017-03-09 11:09:05 +03:00
|
|
|
int flags, bool kern)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
2017-01-09 18:55:16 +03:00
|
|
|
struct sock *sk = sock->sk, *nsk;
|
|
|
|
DECLARE_WAITQUEUE(wait, current);
|
2017-01-09 18:55:13 +03:00
|
|
|
struct smc_sock *lsmc;
|
2017-01-09 18:55:16 +03:00
|
|
|
long timeo;
|
|
|
|
int rc = 0;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
lsmc = smc_sk(sk);
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_hold(sk); /* sock_put below */
|
2017-01-09 18:55:13 +03:00
|
|
|
lock_sock(sk);
|
|
|
|
|
|
|
|
if (lsmc->sk.sk_state != SMC_LISTEN) {
|
|
|
|
rc = -EINVAL;
|
2018-04-26 18:18:23 +03:00
|
|
|
release_sock(sk);
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:16 +03:00
|
|
|
/* Wait for an incoming connection */
|
|
|
|
timeo = sock_rcvtimeo(sk, flags & O_NONBLOCK);
|
|
|
|
add_wait_queue_exclusive(sk_sleep(sk), &wait);
|
|
|
|
while (!(nsk = smc_accept_dequeue(sk, new_sock))) {
|
|
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
|
|
if (!timeo) {
|
|
|
|
rc = -EAGAIN;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
release_sock(sk);
|
|
|
|
timeo = schedule_timeout(timeo);
|
|
|
|
/* wakeup by sk_data_ready in smc_listen_work() */
|
|
|
|
sched_annotate_sleep();
|
|
|
|
lock_sock(sk);
|
|
|
|
if (signal_pending(current)) {
|
|
|
|
rc = sock_intr_errno(timeo);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
set_current_state(TASK_RUNNING);
|
|
|
|
remove_wait_queue(sk_sleep(sk), &wait);
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2017-01-09 18:55:16 +03:00
|
|
|
if (!rc)
|
|
|
|
rc = sock_error(nsk);
|
2018-04-26 18:18:23 +03:00
|
|
|
release_sock(sk);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (lsmc->sockopt_defer_accept && !(flags & O_NONBLOCK)) {
|
|
|
|
/* wait till data arrives on the socket */
|
|
|
|
timeo = msecs_to_jiffies(lsmc->sockopt_defer_accept *
|
|
|
|
MSEC_PER_SEC);
|
|
|
|
if (smc_sk(nsk)->use_fallback) {
|
|
|
|
struct sock *clcsk = smc_sk(nsk)->clcsock->sk;
|
|
|
|
|
|
|
|
lock_sock(clcsk);
|
|
|
|
if (skb_queue_empty(&clcsk->sk_receive_queue))
|
|
|
|
sk_wait_data(clcsk, &timeo, NULL);
|
|
|
|
release_sock(clcsk);
|
|
|
|
} else if (!atomic_read(&smc_sk(nsk)->conn.bytes_to_rcv)) {
|
|
|
|
lock_sock(nsk);
|
2018-05-03 19:12:37 +03:00
|
|
|
smc_rx_wait(smc_sk(nsk), &timeo, smc_rx_data_available);
|
2018-04-26 18:18:23 +03:00
|
|
|
release_sock(nsk);
|
|
|
|
}
|
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
out:
|
2018-01-26 11:28:48 +03:00
|
|
|
sock_put(sk); /* sock_hold above */
|
2017-01-09 18:55:13 +03:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_getname(struct socket *sock, struct sockaddr *addr,
|
2018-02-12 22:00:20 +03:00
|
|
|
int peer)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
|
|
|
struct smc_sock *smc;
|
|
|
|
|
2017-01-09 18:55:25 +03:00
|
|
|
if (peer && (sock->sk->sk_state != SMC_ACTIVE) &&
|
|
|
|
(sock->sk->sk_state != SMC_APPCLOSEWAIT1))
|
2017-01-09 18:55:13 +03:00
|
|
|
return -ENOTCONN;
|
|
|
|
|
|
|
|
smc = smc_sk(sock->sk);
|
|
|
|
|
2018-02-12 22:00:20 +03:00
|
|
|
return smc->clcsock->ops->getname(smc->clcsock, addr, peer);
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_sendmsg(struct socket *sock, struct msghdr *msg, size_t len)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -EPIPE;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
lock_sock(sk);
|
2017-01-09 18:55:25 +03:00
|
|
|
if ((sk->sk_state != SMC_ACTIVE) &&
|
|
|
|
(sk->sk_state != SMC_APPCLOSEWAIT1) &&
|
|
|
|
(sk->sk_state != SMC_INIT))
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
2018-04-26 18:18:21 +03:00
|
|
|
|
|
|
|
if (msg->msg_flags & MSG_FASTOPEN) {
|
2019-08-02 11:47:50 +03:00
|
|
|
if (sk->sk_state == SMC_INIT && !smc->connect_nonblock) {
|
2022-01-22 12:43:09 +03:00
|
|
|
rc = smc_switch_to_fallback(smc, SMC_CLC_DECL_OPTUNSUPP);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
2018-04-26 18:18:21 +03:00
|
|
|
} else {
|
|
|
|
rc = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 17:52:55 +03:00
|
|
|
if (smc->use_fallback) {
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = smc->clcsock->ops->sendmsg(smc->clcsock, msg, len);
|
2021-06-16 17:52:55 +03:00
|
|
|
} else {
|
2017-01-09 18:55:23 +03:00
|
|
|
rc = smc_tx_sendmsg(smc, msg, len);
|
2021-06-16 17:52:55 +03:00
|
|
|
SMC_STAT_TX_PAYLOAD(smc, len, rc);
|
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
out:
|
|
|
|
release_sock(sk);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_recvmsg(struct socket *sock, struct msghdr *msg, size_t len,
|
|
|
|
int flags)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -ENOTCONN;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
lock_sock(sk);
|
2019-01-30 20:51:04 +03:00
|
|
|
if (sk->sk_state == SMC_CLOSED && (sk->sk_shutdown & RCV_SHUTDOWN)) {
|
|
|
|
/* socket was connected before, no more data to read */
|
|
|
|
rc = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
2017-01-09 18:55:25 +03:00
|
|
|
if ((sk->sk_state == SMC_INIT) ||
|
|
|
|
(sk->sk_state == SMC_LISTEN) ||
|
|
|
|
(sk->sk_state == SMC_CLOSED))
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
|
2017-01-09 18:55:25 +03:00
|
|
|
if (sk->sk_state == SMC_PEERFINCLOSEWAIT) {
|
|
|
|
rc = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2018-05-03 19:12:39 +03:00
|
|
|
if (smc->use_fallback) {
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = smc->clcsock->ops->recvmsg(smc->clcsock, msg, len, flags);
|
2018-05-03 19:12:39 +03:00
|
|
|
} else {
|
|
|
|
msg->msg_namelen = 0;
|
|
|
|
rc = smc_rx_recvmsg(smc, msg, NULL, len, flags);
|
2021-06-16 17:52:55 +03:00
|
|
|
SMC_STAT_RX_PAYLOAD(smc, rc, rc);
|
2018-05-03 19:12:39 +03:00
|
|
|
}
|
2017-01-09 18:55:25 +03:00
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
out:
|
|
|
|
release_sock(sk);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-07-03 07:01:49 +03:00
|
|
|
static __poll_t smc_accept_poll(struct sock *parent)
|
2017-01-09 18:55:16 +03:00
|
|
|
{
|
2018-01-26 11:28:47 +03:00
|
|
|
struct smc_sock *isk = smc_sk(parent);
|
2018-02-01 18:02:53 +03:00
|
|
|
__poll_t mask = 0;
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-01-26 11:28:47 +03:00
|
|
|
spin_lock(&isk->accept_q_lock);
|
|
|
|
if (!list_empty(&isk->accept_q))
|
2018-02-12 01:34:03 +03:00
|
|
|
mask = EPOLLIN | EPOLLRDNORM;
|
2018-01-26 11:28:47 +03:00
|
|
|
spin_unlock(&isk->accept_q_lock);
|
2017-01-09 18:55:16 +03:00
|
|
|
|
2018-01-26 11:28:47 +03:00
|
|
|
return mask;
|
2017-01-09 18:55:16 +03:00
|
|
|
}
|
|
|
|
|
2018-06-28 19:43:44 +03:00
|
|
|
static __poll_t smc_poll(struct file *file, struct socket *sock,
|
|
|
|
poll_table *wait)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
2019-04-12 13:57:23 +03:00
|
|
|
__poll_t mask = 0;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2018-01-26 11:28:47 +03:00
|
|
|
if (!sk)
|
2018-02-12 01:34:03 +03:00
|
|
|
return EPOLLNVAL;
|
2018-01-26 11:28:47 +03:00
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
smc = smc_sk(sock->sk);
|
2018-09-18 16:46:34 +03:00
|
|
|
if (smc->use_fallback) {
|
2017-01-09 18:55:16 +03:00
|
|
|
/* delegate to CLC child sock */
|
2018-06-28 19:43:44 +03:00
|
|
|
mask = smc->clcsock->ops->poll(file, smc->clcsock, wait);
|
2018-05-02 17:53:56 +03:00
|
|
|
sk->sk_err = smc->clcsock->sk->sk_err;
|
2017-01-09 18:55:13 +03:00
|
|
|
} else {
|
2018-07-03 19:53:43 +03:00
|
|
|
if (sk->sk_state != SMC_CLOSED)
|
2018-10-23 14:40:39 +03:00
|
|
|
sock_poll_wait(file, sock, wait);
|
2017-01-09 18:55:16 +03:00
|
|
|
if (sk->sk_err)
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLERR;
|
2017-01-09 18:55:25 +03:00
|
|
|
if ((sk->sk_shutdown == SHUTDOWN_MASK) ||
|
|
|
|
(sk->sk_state == SMC_CLOSED))
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLHUP;
|
2018-01-26 11:28:47 +03:00
|
|
|
if (sk->sk_state == SMC_LISTEN) {
|
|
|
|
/* woken up by sk_data_ready in smc_listen_work() */
|
2019-04-12 13:57:23 +03:00
|
|
|
mask |= smc_accept_poll(sk);
|
|
|
|
} else if (smc->use_fallback) { /* as result of connect_work()*/
|
|
|
|
mask |= smc->clcsock->ops->poll(file, smc->clcsock,
|
|
|
|
wait);
|
|
|
|
sk->sk_err = smc->clcsock->sk->sk_err;
|
2018-01-26 11:28:47 +03:00
|
|
|
} else {
|
2019-04-12 13:57:23 +03:00
|
|
|
if ((sk->sk_state != SMC_INIT &&
|
|
|
|
atomic_read(&smc->conn.sndbuf_space)) ||
|
2018-01-26 11:28:47 +03:00
|
|
|
sk->sk_shutdown & SEND_SHUTDOWN) {
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLOUT | EPOLLWRNORM;
|
2018-01-26 11:28:47 +03:00
|
|
|
} else {
|
|
|
|
sk_set_bit(SOCKWQ_ASYNC_NOSPACE, sk);
|
|
|
|
set_bit(SOCK_NOSPACE, &sk->sk_socket->flags);
|
|
|
|
}
|
|
|
|
if (atomic_read(&smc->conn.bytes_to_rcv))
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLIN | EPOLLRDNORM;
|
2018-01-26 11:28:47 +03:00
|
|
|
if (sk->sk_shutdown & RCV_SHUTDOWN)
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLIN | EPOLLRDNORM | EPOLLRDHUP;
|
2018-01-26 11:28:47 +03:00
|
|
|
if (sk->sk_state == SMC_APPCLOSEWAIT1)
|
2018-02-12 01:34:03 +03:00
|
|
|
mask |= EPOLLIN;
|
2018-09-18 16:46:37 +03:00
|
|
|
if (smc->conn.urg_state == SMC_URG_VALID)
|
|
|
|
mask |= EPOLLPRI;
|
2018-01-26 11:28:47 +03:00
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_shutdown(struct socket *sock, int how)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
2021-11-26 05:41:35 +03:00
|
|
|
bool do_shutdown = true;
|
2017-01-09 18:55:13 +03:00
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -EINVAL;
|
2021-11-26 05:41:35 +03:00
|
|
|
int old_state;
|
2017-01-09 18:55:25 +03:00
|
|
|
int rc1 = 0;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
|
|
|
|
if ((how < SHUT_RD) || (how > SHUT_RDWR))
|
2017-01-09 18:55:25 +03:00
|
|
|
return rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
lock_sock(sk);
|
|
|
|
|
|
|
|
rc = -ENOTCONN;
|
2018-08-08 15:13:19 +03:00
|
|
|
if ((sk->sk_state != SMC_ACTIVE) &&
|
2017-01-09 18:55:25 +03:00
|
|
|
(sk->sk_state != SMC_PEERCLOSEWAIT1) &&
|
|
|
|
(sk->sk_state != SMC_PEERCLOSEWAIT2) &&
|
|
|
|
(sk->sk_state != SMC_APPCLOSEWAIT1) &&
|
|
|
|
(sk->sk_state != SMC_APPCLOSEWAIT2) &&
|
|
|
|
(sk->sk_state != SMC_APPFINCLOSEWAIT))
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
if (smc->use_fallback) {
|
|
|
|
rc = kernel_sock_shutdown(smc->clcsock, how);
|
|
|
|
sk->sk_shutdown = smc->clcsock->sk->sk_shutdown;
|
2022-04-14 10:51:03 +03:00
|
|
|
if (sk->sk_shutdown == SHUTDOWN_MASK) {
|
2017-01-09 18:55:13 +03:00
|
|
|
sk->sk_state = SMC_CLOSED;
|
2022-04-14 10:51:03 +03:00
|
|
|
sock_put(sk);
|
|
|
|
}
|
2017-01-09 18:55:25 +03:00
|
|
|
goto out;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
2017-01-09 18:55:25 +03:00
|
|
|
switch (how) {
|
|
|
|
case SHUT_RDWR: /* shutdown in both directions */
|
2021-11-26 05:41:35 +03:00
|
|
|
old_state = sk->sk_state;
|
2017-01-09 18:55:25 +03:00
|
|
|
rc = smc_close_active(smc);
|
2021-11-26 05:41:35 +03:00
|
|
|
if (old_state == SMC_ACTIVE &&
|
|
|
|
sk->sk_state == SMC_PEERCLOSEWAIT1)
|
|
|
|
do_shutdown = false;
|
2017-01-09 18:55:25 +03:00
|
|
|
break;
|
|
|
|
case SHUT_WR:
|
|
|
|
rc = smc_close_shutdown_write(smc);
|
|
|
|
break;
|
|
|
|
case SHUT_RD:
|
2018-04-19 16:56:40 +03:00
|
|
|
rc = 0;
|
|
|
|
/* nothing more to do because peer is not involved */
|
2017-01-09 18:55:25 +03:00
|
|
|
break;
|
|
|
|
}
|
2021-11-26 05:41:35 +03:00
|
|
|
if (do_shutdown && smc->clcsock)
|
2018-04-19 16:56:40 +03:00
|
|
|
rc1 = kernel_sock_shutdown(smc->clcsock, how);
|
2017-01-09 18:55:25 +03:00
|
|
|
/* map sock_shutdown_cmd constants to sk_shutdown value range */
|
|
|
|
sk->sk_shutdown |= how + 1;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
release_sock(sk);
|
2017-01-09 18:55:25 +03:00
|
|
|
return rc ? rc : rc1;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_setsockopt(struct socket *sock, int level, int optname,
|
2020-07-23 09:09:07 +03:00
|
|
|
sockptr_t optval, unsigned int optlen)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
2018-04-26 18:18:22 +03:00
|
|
|
int val, rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2021-05-05 22:40:48 +03:00
|
|
|
if (level == SOL_TCP && optname == TCP_ULP)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
smc = smc_sk(sk);
|
|
|
|
|
|
|
|
/* generic setsockopts reaching us here always apply to the
|
|
|
|
* CLC socket
|
|
|
|
*/
|
2022-01-22 12:43:09 +03:00
|
|
|
mutex_lock(&smc->clcsock_release_lock);
|
|
|
|
if (!smc->clcsock) {
|
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
|
|
|
return -EBADF;
|
|
|
|
}
|
2020-07-17 09:23:31 +03:00
|
|
|
if (unlikely(!smc->clcsock->ops->setsockopt))
|
|
|
|
rc = -EOPNOTSUPP;
|
|
|
|
else
|
|
|
|
rc = smc->clcsock->ops->setsockopt(smc->clcsock, level, optname,
|
|
|
|
optval, optlen);
|
2018-04-26 18:18:21 +03:00
|
|
|
if (smc->clcsock->sk->sk_err) {
|
|
|
|
sk->sk_err = smc->clcsock->sk->sk_err;
|
2021-06-28 01:48:21 +03:00
|
|
|
sk_error_report(sk);
|
2018-04-26 18:18:21 +03:00
|
|
|
}
|
2022-01-22 12:43:09 +03:00
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
2018-04-26 18:18:21 +03:00
|
|
|
|
2018-04-26 18:18:22 +03:00
|
|
|
if (optlen < sizeof(int))
|
2018-05-31 05:31:22 +03:00
|
|
|
return -EINVAL;
|
2020-07-23 09:09:07 +03:00
|
|
|
if (copy_from_sockptr(&val, optval, sizeof(int)))
|
2018-07-18 16:22:50 +03:00
|
|
|
return -EFAULT;
|
2018-04-26 18:18:22 +03:00
|
|
|
|
2018-04-26 18:18:21 +03:00
|
|
|
lock_sock(sk);
|
2019-12-13 00:35:58 +03:00
|
|
|
if (rc || smc->use_fallback)
|
|
|
|
goto out;
|
2018-04-26 18:18:21 +03:00
|
|
|
switch (optname) {
|
|
|
|
case TCP_FASTOPEN:
|
|
|
|
case TCP_FASTOPEN_CONNECT:
|
|
|
|
case TCP_FASTOPEN_KEY:
|
|
|
|
case TCP_FASTOPEN_NO_COOKIE:
|
|
|
|
/* option not supported by SMC */
|
2019-11-15 14:39:30 +03:00
|
|
|
if (sk->sk_state == SMC_INIT && !smc->connect_nonblock) {
|
2022-01-22 12:43:09 +03:00
|
|
|
rc = smc_switch_to_fallback(smc, SMC_CLC_DECL_OPTUNSUPP);
|
2018-04-26 18:18:21 +03:00
|
|
|
} else {
|
2019-12-13 00:35:58 +03:00
|
|
|
rc = -EINVAL;
|
2018-04-26 18:18:21 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-04-26 18:18:22 +03:00
|
|
|
case TCP_NODELAY:
|
2019-08-02 11:16:38 +03:00
|
|
|
if (sk->sk_state != SMC_INIT &&
|
|
|
|
sk->sk_state != SMC_LISTEN &&
|
|
|
|
sk->sk_state != SMC_CLOSED) {
|
2021-06-16 17:52:55 +03:00
|
|
|
if (val) {
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_INC(smc, ndly_cnt);
|
2022-03-01 12:43:59 +03:00
|
|
|
smc_tx_pending(&smc->conn);
|
|
|
|
cancel_delayed_work(&smc->conn.tx_work);
|
2021-06-16 17:52:55 +03:00
|
|
|
}
|
2018-04-26 18:18:22 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TCP_CORK:
|
2019-08-02 11:16:38 +03:00
|
|
|
if (sk->sk_state != SMC_INIT &&
|
|
|
|
sk->sk_state != SMC_LISTEN &&
|
|
|
|
sk->sk_state != SMC_CLOSED) {
|
2021-06-16 17:52:55 +03:00
|
|
|
if (!val) {
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_INC(smc, cork_cnt);
|
net/smc: Send directly when TCP_CORK is cleared
[ Upstream commit ea785a1a573b390a150010b3c5b81e1ccd8c98a8 ]
According to the man page of TCP_CORK [1], if set, don't send out
partial frames. All queued partial frames are sent when option is
cleared again.
When applications call setsockopt to disable TCP_CORK, this call is
protected by lock_sock(), and tries to mod_delayed_work() to 0, in order
to send pending data right now. However, the delayed work smc_tx_work is
also protected by lock_sock(). There introduces lock contention for
sending data.
To fix it, send pending data directly which acts like TCP, without
lock_sock() protected in the context of setsockopt (already lock_sock()ed),
and cancel unnecessary dealyed work, which is protected by lock.
[1] https://linux.die.net/man/7/tcp
Signed-off-by: Tony Lu <tonylu@linux.alibaba.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: Sasha Levin <sashal@kernel.org>
2022-01-30 21:02:55 +03:00
|
|
|
smc_tx_pending(&smc->conn);
|
|
|
|
cancel_delayed_work(&smc->conn.tx_work);
|
2021-06-16 17:52:55 +03:00
|
|
|
}
|
2018-04-26 18:18:22 +03:00
|
|
|
}
|
|
|
|
break;
|
2018-04-26 18:18:23 +03:00
|
|
|
case TCP_DEFER_ACCEPT:
|
|
|
|
smc->sockopt_defer_accept = val;
|
|
|
|
break;
|
2018-04-26 18:18:21 +03:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2019-12-13 00:35:58 +03:00
|
|
|
out:
|
2018-04-26 18:18:21 +03:00
|
|
|
release_sock(sk);
|
|
|
|
|
|
|
|
return rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_getsockopt(struct socket *sock, int level, int optname,
|
|
|
|
char __user *optval, int __user *optlen)
|
|
|
|
{
|
|
|
|
struct smc_sock *smc;
|
2022-01-22 12:43:09 +03:00
|
|
|
int rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
smc = smc_sk(sock->sk);
|
2022-01-22 12:43:09 +03:00
|
|
|
mutex_lock(&smc->clcsock_release_lock);
|
|
|
|
if (!smc->clcsock) {
|
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
|
|
|
return -EBADF;
|
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
/* socket options apply to the CLC socket */
|
2022-01-22 12:43:09 +03:00
|
|
|
if (unlikely(!smc->clcsock->ops->getsockopt)) {
|
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
2020-07-17 09:23:31 +03:00
|
|
|
return -EOPNOTSUPP;
|
2022-01-22 12:43:09 +03:00
|
|
|
}
|
|
|
|
rc = smc->clcsock->ops->getsockopt(smc->clcsock, level, optname,
|
|
|
|
optval, optlen);
|
|
|
|
mutex_unlock(&smc->clcsock_release_lock);
|
|
|
|
return rc;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int smc_ioctl(struct socket *sock, unsigned int cmd,
|
|
|
|
unsigned long arg)
|
|
|
|
{
|
2018-05-23 17:38:11 +03:00
|
|
|
union smc_host_cursor cons, urg;
|
|
|
|
struct smc_connection *conn;
|
2017-01-09 18:55:13 +03:00
|
|
|
struct smc_sock *smc;
|
2018-05-02 17:56:46 +03:00
|
|
|
int answ;
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
smc = smc_sk(sock->sk);
|
2018-05-23 17:38:11 +03:00
|
|
|
conn = &smc->conn;
|
2018-08-08 15:13:21 +03:00
|
|
|
lock_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
if (smc->use_fallback) {
|
2018-08-08 15:13:21 +03:00
|
|
|
if (!smc->clcsock) {
|
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
return -EBADF;
|
2018-08-08 15:13:21 +03:00
|
|
|
}
|
|
|
|
answ = smc->clcsock->ops->ioctl(smc->clcsock, cmd, arg);
|
|
|
|
release_sock(&smc->sk);
|
|
|
|
return answ;
|
2018-05-02 17:56:46 +03:00
|
|
|
}
|
|
|
|
switch (cmd) {
|
|
|
|
case SIOCINQ: /* same as FIONREAD */
|
2018-07-16 14:56:52 +03:00
|
|
|
if (smc->sk.sk_state == SMC_LISTEN) {
|
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
return -EINVAL;
|
2018-07-16 14:56:52 +03:00
|
|
|
}
|
2018-05-23 17:38:09 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT ||
|
|
|
|
smc->sk.sk_state == SMC_CLOSED)
|
|
|
|
answ = 0;
|
|
|
|
else
|
|
|
|
answ = atomic_read(&smc->conn.bytes_to_rcv);
|
2018-05-02 17:56:46 +03:00
|
|
|
break;
|
|
|
|
case SIOCOUTQ:
|
|
|
|
/* output queue size (not send + not acked) */
|
2018-07-16 14:56:52 +03:00
|
|
|
if (smc->sk.sk_state == SMC_LISTEN) {
|
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
return -EINVAL;
|
2018-07-16 14:56:52 +03:00
|
|
|
}
|
2018-05-23 17:38:09 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT ||
|
|
|
|
smc->sk.sk_state == SMC_CLOSED)
|
|
|
|
answ = 0;
|
|
|
|
else
|
|
|
|
answ = smc->conn.sndbuf_desc->len -
|
2018-05-02 17:56:46 +03:00
|
|
|
atomic_read(&smc->conn.sndbuf_space);
|
|
|
|
break;
|
|
|
|
case SIOCOUTQNSD:
|
|
|
|
/* output queue size (not send only) */
|
2018-07-16 14:56:52 +03:00
|
|
|
if (smc->sk.sk_state == SMC_LISTEN) {
|
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
return -EINVAL;
|
2018-07-16 14:56:52 +03:00
|
|
|
}
|
2018-05-23 17:38:09 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT ||
|
|
|
|
smc->sk.sk_state == SMC_CLOSED)
|
|
|
|
answ = 0;
|
|
|
|
else
|
|
|
|
answ = smc_tx_prepared_sends(&smc->conn);
|
2018-05-02 17:56:46 +03:00
|
|
|
break;
|
2018-05-23 17:38:11 +03:00
|
|
|
case SIOCATMARK:
|
2018-07-16 14:56:52 +03:00
|
|
|
if (smc->sk.sk_state == SMC_LISTEN) {
|
|
|
|
release_sock(&smc->sk);
|
2018-05-23 17:38:11 +03:00
|
|
|
return -EINVAL;
|
2018-07-16 14:56:52 +03:00
|
|
|
}
|
2018-05-23 17:38:11 +03:00
|
|
|
if (smc->sk.sk_state == SMC_INIT ||
|
|
|
|
smc->sk.sk_state == SMC_CLOSED) {
|
|
|
|
answ = 0;
|
|
|
|
} else {
|
2018-07-23 14:53:09 +03:00
|
|
|
smc_curs_copy(&cons, &conn->local_tx_ctrl.cons, conn);
|
|
|
|
smc_curs_copy(&urg, &conn->urg_curs, conn);
|
2018-05-23 17:38:11 +03:00
|
|
|
answ = smc_curs_diff(conn->rmb_desc->len,
|
|
|
|
&cons, &urg) == 1;
|
|
|
|
}
|
|
|
|
break;
|
2018-05-02 17:56:46 +03:00
|
|
|
default:
|
2018-07-16 14:56:52 +03:00
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
return -ENOIOCTLCMD;
|
|
|
|
}
|
2018-07-16 14:56:52 +03:00
|
|
|
release_sock(&smc->sk);
|
2018-05-02 17:56:46 +03:00
|
|
|
|
|
|
|
return put_user(answ, (int __user *)arg);
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t smc_sendpage(struct socket *sock, struct page *page,
|
|
|
|
int offset, size_t size, int flags)
|
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -EPIPE;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
lock_sock(sk);
|
2018-05-03 18:57:39 +03:00
|
|
|
if (sk->sk_state != SMC_ACTIVE) {
|
|
|
|
release_sock(sk);
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
2018-05-03 18:57:39 +03:00
|
|
|
}
|
|
|
|
release_sock(sk);
|
2021-06-16 17:52:55 +03:00
|
|
|
if (smc->use_fallback) {
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = kernel_sendpage(smc->clcsock, page, offset,
|
|
|
|
size, flags);
|
2021-06-16 17:52:55 +03:00
|
|
|
} else {
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_INC(smc, sendpage_cnt);
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = sock_no_sendpage(sock, page, offset, size, flags);
|
2021-06-16 17:52:55 +03:00
|
|
|
}
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2018-05-03 19:12:39 +03:00
|
|
|
/* Map the affected portions of the rmbe into an spd, note the number of bytes
|
|
|
|
* to splice in conn->splice_pending, and press 'go'. Delays consumer cursor
|
|
|
|
* updates till whenever a respective page has been fully processed.
|
|
|
|
* Note that subsequent recv() calls have to wait till all splice() processing
|
|
|
|
* completed.
|
|
|
|
*/
|
2017-01-09 18:55:13 +03:00
|
|
|
static ssize_t smc_splice_read(struct socket *sock, loff_t *ppos,
|
|
|
|
struct pipe_inode_info *pipe, size_t len,
|
2018-05-03 19:12:39 +03:00
|
|
|
unsigned int flags)
|
2017-01-09 18:55:13 +03:00
|
|
|
{
|
|
|
|
struct sock *sk = sock->sk;
|
|
|
|
struct smc_sock *smc;
|
|
|
|
int rc = -ENOTCONN;
|
|
|
|
|
|
|
|
smc = smc_sk(sk);
|
|
|
|
lock_sock(sk);
|
2019-01-30 20:51:04 +03:00
|
|
|
if (sk->sk_state == SMC_CLOSED && (sk->sk_shutdown & RCV_SHUTDOWN)) {
|
|
|
|
/* socket was connected before, no more data to read */
|
|
|
|
rc = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
2018-05-03 19:12:39 +03:00
|
|
|
if (sk->sk_state == SMC_INIT ||
|
|
|
|
sk->sk_state == SMC_LISTEN ||
|
|
|
|
sk->sk_state == SMC_CLOSED)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (sk->sk_state == SMC_PEERFINCLOSEWAIT) {
|
|
|
|
rc = 0;
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
2018-05-03 19:12:39 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
if (smc->use_fallback) {
|
|
|
|
rc = smc->clcsock->ops->splice_read(smc->clcsock, ppos,
|
|
|
|
pipe, len, flags);
|
|
|
|
} else {
|
2018-05-03 19:12:39 +03:00
|
|
|
if (*ppos) {
|
|
|
|
rc = -ESPIPE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (flags & SPLICE_F_NONBLOCK)
|
|
|
|
flags = MSG_DONTWAIT;
|
|
|
|
else
|
|
|
|
flags = 0;
|
2021-06-16 17:52:58 +03:00
|
|
|
SMC_STAT_INC(smc, splice_cnt);
|
2018-05-03 19:12:39 +03:00
|
|
|
rc = smc_rx_recvmsg(smc, NULL, pipe, len, flags);
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
out:
|
|
|
|
release_sock(sk);
|
2018-05-03 19:12:39 +03:00
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* must look like tcp */
|
|
|
|
static const struct proto_ops smc_sock_ops = {
|
|
|
|
.family = PF_SMC,
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.release = smc_release,
|
|
|
|
.bind = smc_bind,
|
|
|
|
.connect = smc_connect,
|
|
|
|
.socketpair = sock_no_socketpair,
|
|
|
|
.accept = smc_accept,
|
|
|
|
.getname = smc_getname,
|
2018-06-28 19:43:44 +03:00
|
|
|
.poll = smc_poll,
|
2017-01-09 18:55:13 +03:00
|
|
|
.ioctl = smc_ioctl,
|
|
|
|
.listen = smc_listen,
|
|
|
|
.shutdown = smc_shutdown,
|
|
|
|
.setsockopt = smc_setsockopt,
|
|
|
|
.getsockopt = smc_getsockopt,
|
|
|
|
.sendmsg = smc_sendmsg,
|
|
|
|
.recvmsg = smc_recvmsg,
|
|
|
|
.mmap = sock_no_mmap,
|
|
|
|
.sendpage = smc_sendpage,
|
|
|
|
.splice_read = smc_splice_read,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int smc_create(struct net *net, struct socket *sock, int protocol,
|
|
|
|
int kern)
|
|
|
|
{
|
2018-03-16 17:06:41 +03:00
|
|
|
int family = (protocol == SMCPROTO_SMC6) ? PF_INET6 : PF_INET;
|
2017-01-09 18:55:13 +03:00
|
|
|
struct smc_sock *smc;
|
|
|
|
struct sock *sk;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
rc = -ESOCKTNOSUPPORT;
|
|
|
|
if (sock->type != SOCK_STREAM)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
rc = -EPROTONOSUPPORT;
|
2018-03-16 17:06:41 +03:00
|
|
|
if (protocol != SMCPROTO_SMC && protocol != SMCPROTO_SMC6)
|
2017-01-09 18:55:13 +03:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
rc = -ENOBUFS;
|
|
|
|
sock->ops = &smc_sock_ops;
|
2018-03-16 17:06:41 +03:00
|
|
|
sk = smc_sock_alloc(net, sock, protocol);
|
2017-01-09 18:55:13 +03:00
|
|
|
if (!sk)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
/* create internal TCP socket for CLC handshake and fallback */
|
|
|
|
smc = smc_sk(sk);
|
2017-01-09 18:55:16 +03:00
|
|
|
smc->use_fallback = false; /* assume rdma capability first */
|
2018-07-25 17:35:32 +03:00
|
|
|
smc->fallback_rsn = 0;
|
2018-03-16 17:06:41 +03:00
|
|
|
rc = sock_create_kern(net, family, SOCK_STREAM, IPPROTO_TCP,
|
|
|
|
&smc->clcsock);
|
2018-02-28 14:44:09 +03:00
|
|
|
if (rc) {
|
2017-01-09 18:55:13 +03:00
|
|
|
sk_common_release(sk);
|
2018-02-28 14:44:09 +03:00
|
|
|
goto out;
|
|
|
|
}
|
2017-01-09 18:55:18 +03:00
|
|
|
smc->sk.sk_sndbuf = max(smc->clcsock->sk->sk_sndbuf, SMC_BUF_MIN_SIZE);
|
|
|
|
smc->sk.sk_rcvbuf = max(smc->clcsock->sk->sk_rcvbuf, SMC_BUF_MIN_SIZE);
|
2017-01-09 18:55:13 +03:00
|
|
|
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct net_proto_family smc_sock_family_ops = {
|
|
|
|
.family = PF_SMC,
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.create = smc_create,
|
|
|
|
};
|
|
|
|
|
2019-02-21 15:01:02 +03:00
|
|
|
unsigned int smc_net_id;
|
|
|
|
|
|
|
|
static __net_init int smc_net_init(struct net *net)
|
|
|
|
{
|
|
|
|
return smc_pnet_net_init(net);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __net_exit smc_net_exit(struct net *net)
|
|
|
|
{
|
|
|
|
smc_pnet_net_exit(net);
|
|
|
|
}
|
|
|
|
|
2021-06-16 17:52:58 +03:00
|
|
|
static __net_init int smc_net_stat_init(struct net *net)
|
|
|
|
{
|
|
|
|
return smc_stats_init(net);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __net_exit smc_net_stat_exit(struct net *net)
|
|
|
|
{
|
|
|
|
smc_stats_exit(net);
|
|
|
|
}
|
|
|
|
|
2019-02-21 15:01:02 +03:00
|
|
|
static struct pernet_operations smc_net_ops = {
|
|
|
|
.init = smc_net_init,
|
|
|
|
.exit = smc_net_exit,
|
|
|
|
.id = &smc_net_id,
|
|
|
|
.size = sizeof(struct smc_net),
|
|
|
|
};
|
|
|
|
|
2021-06-16 17:52:58 +03:00
|
|
|
static struct pernet_operations smc_net_stat_ops = {
|
|
|
|
.init = smc_net_stat_init,
|
|
|
|
.exit = smc_net_stat_exit,
|
|
|
|
};
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
static int __init smc_init(void)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
|
2019-02-21 15:01:02 +03:00
|
|
|
rc = register_pernet_subsys(&smc_net_ops);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
2021-06-16 17:52:58 +03:00
|
|
|
rc = register_pernet_subsys(&smc_net_stat_ops);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
2020-09-26 13:44:24 +03:00
|
|
|
smc_ism_init();
|
2020-09-26 13:44:31 +03:00
|
|
|
smc_clc_init();
|
2020-09-26 13:44:24 +03:00
|
|
|
|
2020-12-01 22:20:43 +03:00
|
|
|
rc = smc_nl_init();
|
2017-01-09 18:55:15 +03:00
|
|
|
if (rc)
|
2019-06-26 18:47:50 +03:00
|
|
|
goto out_pernet_subsys;
|
2017-01-09 18:55:15 +03:00
|
|
|
|
2020-12-01 22:20:43 +03:00
|
|
|
rc = smc_pnet_init();
|
|
|
|
if (rc)
|
|
|
|
goto out_nl;
|
|
|
|
|
2020-09-10 19:48:29 +03:00
|
|
|
rc = -ENOMEM;
|
|
|
|
smc_hs_wq = alloc_workqueue("smc_hs_wq", 0, 0);
|
|
|
|
if (!smc_hs_wq)
|
|
|
|
goto out_pnet;
|
|
|
|
|
|
|
|
smc_close_wq = alloc_workqueue("smc_close_wq", 0, 0);
|
|
|
|
if (!smc_close_wq)
|
|
|
|
goto out_alloc_hs_wq;
|
|
|
|
|
2019-11-16 19:47:29 +03:00
|
|
|
rc = smc_core_init();
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: smc_core_init fails with %d\n", __func__, rc);
|
2021-06-16 17:52:58 +03:00
|
|
|
goto out_alloc_wqs;
|
2019-11-16 19:47:29 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:21 +03:00
|
|
|
rc = smc_llc_init();
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: smc_llc_init fails with %d\n", __func__, rc);
|
2019-11-16 19:47:29 +03:00
|
|
|
goto out_core;
|
2017-01-09 18:55:21 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:22 +03:00
|
|
|
rc = smc_cdc_init();
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: smc_cdc_init fails with %d\n", __func__, rc);
|
2019-11-16 19:47:29 +03:00
|
|
|
goto out_core;
|
2017-01-09 18:55:22 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = proto_register(&smc_proto, 1);
|
|
|
|
if (rc) {
|
2018-03-16 17:06:41 +03:00
|
|
|
pr_err("%s: proto_register(v4) fails with %d\n", __func__, rc);
|
2019-11-16 19:47:29 +03:00
|
|
|
goto out_core;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
2018-03-16 17:06:41 +03:00
|
|
|
rc = proto_register(&smc_proto6, 1);
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: proto_register(v6) fails with %d\n", __func__, rc);
|
|
|
|
goto out_proto;
|
|
|
|
}
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
rc = sock_register(&smc_sock_family_ops);
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: sock_register fails with %d\n", __func__, rc);
|
2018-03-16 17:06:41 +03:00
|
|
|
goto out_proto6;
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
2017-01-09 18:55:26 +03:00
|
|
|
INIT_HLIST_HEAD(&smc_v4_hashinfo.ht);
|
2018-03-16 17:06:41 +03:00
|
|
|
INIT_HLIST_HEAD(&smc_v6_hashinfo.ht);
|
2017-01-09 18:55:13 +03:00
|
|
|
|
2017-01-09 18:55:14 +03:00
|
|
|
rc = smc_ib_register_client();
|
|
|
|
if (rc) {
|
|
|
|
pr_err("%s: ib_register fails with %d\n", __func__, rc);
|
|
|
|
goto out_sock;
|
|
|
|
}
|
|
|
|
|
2017-10-25 12:01:46 +03:00
|
|
|
static_branch_enable(&tcp_have_smc);
|
2017-01-09 18:55:13 +03:00
|
|
|
return 0;
|
|
|
|
|
2017-01-09 18:55:14 +03:00
|
|
|
out_sock:
|
|
|
|
sock_unregister(PF_SMC);
|
2018-03-16 17:06:41 +03:00
|
|
|
out_proto6:
|
|
|
|
proto_unregister(&smc_proto6);
|
2017-01-09 18:55:13 +03:00
|
|
|
out_proto:
|
|
|
|
proto_unregister(&smc_proto);
|
2019-11-16 19:47:29 +03:00
|
|
|
out_core:
|
|
|
|
smc_core_exit();
|
2020-09-10 19:48:29 +03:00
|
|
|
out_alloc_wqs:
|
|
|
|
destroy_workqueue(smc_close_wq);
|
|
|
|
out_alloc_hs_wq:
|
|
|
|
destroy_workqueue(smc_hs_wq);
|
2017-01-09 18:55:15 +03:00
|
|
|
out_pnet:
|
|
|
|
smc_pnet_exit();
|
2020-12-01 22:20:43 +03:00
|
|
|
out_nl:
|
|
|
|
smc_nl_exit();
|
2019-06-26 18:47:50 +03:00
|
|
|
out_pernet_subsys:
|
|
|
|
unregister_pernet_subsys(&smc_net_ops);
|
|
|
|
|
2017-01-09 18:55:13 +03:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit smc_exit(void)
|
|
|
|
{
|
2017-10-25 12:01:46 +03:00
|
|
|
static_branch_disable(&tcp_have_smc);
|
2017-01-09 18:55:13 +03:00
|
|
|
sock_unregister(PF_SMC);
|
2019-11-16 19:47:29 +03:00
|
|
|
smc_core_exit();
|
|
|
|
smc_ib_unregister_client();
|
2020-09-10 19:48:29 +03:00
|
|
|
destroy_workqueue(smc_close_wq);
|
|
|
|
destroy_workqueue(smc_hs_wq);
|
2018-03-16 17:06:41 +03:00
|
|
|
proto_unregister(&smc_proto6);
|
2017-01-09 18:55:13 +03:00
|
|
|
proto_unregister(&smc_proto);
|
2017-01-09 18:55:15 +03:00
|
|
|
smc_pnet_exit();
|
2020-12-01 22:20:43 +03:00
|
|
|
smc_nl_exit();
|
2021-06-16 17:52:58 +03:00
|
|
|
unregister_pernet_subsys(&smc_net_stat_ops);
|
2019-02-21 15:01:02 +03:00
|
|
|
unregister_pernet_subsys(&smc_net_ops);
|
2019-11-16 19:47:31 +03:00
|
|
|
rcu_barrier();
|
2017-01-09 18:55:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
module_init(smc_init);
|
|
|
|
module_exit(smc_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Ursula Braun <ubraun@linux.vnet.ibm.com>");
|
|
|
|
MODULE_DESCRIPTION("smc socket address family");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_ALIAS_NETPROTO(PF_SMC);
|