mirror of
https://gitee.com/bianbu-linux/linux-6.6
synced 2025-04-24 14:07:52 -04:00
mptcp: remote addresses fullmesh
This patch added and managed a new per endpoint flag, named MPTCP_PM_ADDR_FLAG_FULLMESH. In mptcp_pm_create_subflow_or_signal_addr(), if such flag is set, instead of: remote_address((struct sock_common *)sk, &remote); fill a temporary allocated array of all known remote address. After releaseing the pm lock loop on such array and create a subflow for each remote address from the given local. Note that the we could still use an array even for non 'fullmesh' endpoint: with a single entry corresponding to the primary MPC subflow remote address. Suggested-by: Paolo Abeni <pabeni@redhat.com> Signed-off-by: Geliang Tang <geliangtang@xiaomi.com> Signed-off-by: Mat Martineau <mathew.j.martineau@linux.intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
ee285257a9
commit
2843ff6f36
2 changed files with 56 additions and 4 deletions
|
@ -73,6 +73,7 @@ enum {
|
||||||
#define MPTCP_PM_ADDR_FLAG_SIGNAL (1 << 0)
|
#define MPTCP_PM_ADDR_FLAG_SIGNAL (1 << 0)
|
||||||
#define MPTCP_PM_ADDR_FLAG_SUBFLOW (1 << 1)
|
#define MPTCP_PM_ADDR_FLAG_SUBFLOW (1 << 1)
|
||||||
#define MPTCP_PM_ADDR_FLAG_BACKUP (1 << 2)
|
#define MPTCP_PM_ADDR_FLAG_BACKUP (1 << 2)
|
||||||
|
#define MPTCP_PM_ADDR_FLAG_FULLMESH (1 << 3)
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
MPTCP_PM_CMD_UNSPEC,
|
MPTCP_PM_CMD_UNSPEC,
|
||||||
|
|
|
@ -410,6 +410,55 @@ void mptcp_pm_free_anno_list(struct mptcp_sock *msk)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool lookup_address_in_vec(struct mptcp_addr_info *addrs, unsigned int nr,
|
||||||
|
struct mptcp_addr_info *addr)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < nr; i++) {
|
||||||
|
if (addresses_equal(&addrs[i], addr, addr->port))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fill all the remote addresses into the array addrs[],
|
||||||
|
* and return the array size.
|
||||||
|
*/
|
||||||
|
static unsigned int fill_remote_addresses_vec(struct mptcp_sock *msk, bool fullmesh,
|
||||||
|
struct mptcp_addr_info *addrs)
|
||||||
|
{
|
||||||
|
struct sock *sk = (struct sock *)msk, *ssk;
|
||||||
|
struct mptcp_subflow_context *subflow;
|
||||||
|
struct mptcp_addr_info remote = { 0 };
|
||||||
|
unsigned int subflows_max;
|
||||||
|
int i = 0;
|
||||||
|
|
||||||
|
subflows_max = mptcp_pm_get_subflows_max(msk);
|
||||||
|
|
||||||
|
/* Non-fullmesh endpoint, fill in the single entry
|
||||||
|
* corresponding to the primary MPC subflow remote address
|
||||||
|
*/
|
||||||
|
if (!fullmesh) {
|
||||||
|
remote_address((struct sock_common *)sk, &remote);
|
||||||
|
msk->pm.subflows++;
|
||||||
|
addrs[i++] = remote;
|
||||||
|
} else {
|
||||||
|
mptcp_for_each_subflow(msk, subflow) {
|
||||||
|
ssk = mptcp_subflow_tcp_sock(subflow);
|
||||||
|
remote_address((struct sock_common *)ssk, &remote);
|
||||||
|
if (!lookup_address_in_vec(addrs, i, &remote) &&
|
||||||
|
msk->pm.subflows < subflows_max) {
|
||||||
|
msk->pm.subflows++;
|
||||||
|
addrs[i++] = remote;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
static void mptcp_pm_create_subflow_or_signal_addr(struct mptcp_sock *msk)
|
static void mptcp_pm_create_subflow_or_signal_addr(struct mptcp_sock *msk)
|
||||||
{
|
{
|
||||||
struct sock *sk = (struct sock *)msk;
|
struct sock *sk = (struct sock *)msk;
|
||||||
|
@ -455,14 +504,16 @@ static void mptcp_pm_create_subflow_or_signal_addr(struct mptcp_sock *msk)
|
||||||
!READ_ONCE(msk->pm.remote_deny_join_id0)) {
|
!READ_ONCE(msk->pm.remote_deny_join_id0)) {
|
||||||
local = select_local_address(pernet, msk);
|
local = select_local_address(pernet, msk);
|
||||||
if (local) {
|
if (local) {
|
||||||
struct mptcp_addr_info remote = { 0 };
|
bool fullmesh = !!(local->flags & MPTCP_PM_ADDR_FLAG_FULLMESH);
|
||||||
|
struct mptcp_addr_info addrs[MPTCP_PM_ADDR_MAX];
|
||||||
|
int i, nr;
|
||||||
|
|
||||||
msk->pm.local_addr_used++;
|
msk->pm.local_addr_used++;
|
||||||
msk->pm.subflows++;
|
|
||||||
check_work_pending(msk);
|
check_work_pending(msk);
|
||||||
remote_address((struct sock_common *)sk, &remote);
|
nr = fill_remote_addresses_vec(msk, fullmesh, addrs);
|
||||||
spin_unlock_bh(&msk->pm.lock);
|
spin_unlock_bh(&msk->pm.lock);
|
||||||
__mptcp_subflow_connect(sk, &local->addr, &remote);
|
for (i = 0; i < nr; i++)
|
||||||
|
__mptcp_subflow_connect(sk, &local->addr, &addrs[i]);
|
||||||
spin_lock_bh(&msk->pm.lock);
|
spin_lock_bh(&msk->pm.lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue