OSDN Git Service

mptcp: remote addresses fullmesh
authorGeliang Tang <geliangtang@xiaomi.com>
Tue, 17 Aug 2021 22:07:23 +0000 (15:07 -0700)
committerDavid S. Miller <davem@davemloft.net>
Wed, 18 Aug 2021 09:10:01 +0000 (10:10 +0100)
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>
include/uapi/linux/mptcp.h
net/mptcp/pm_netlink.c

index 7b05f71..f66038b 100644 (file)
@@ -73,6 +73,7 @@ enum {
 #define MPTCP_PM_ADDR_FLAG_SIGNAL                      (1 << 0)
 #define MPTCP_PM_ADDR_FLAG_SUBFLOW                     (1 << 1)
 #define MPTCP_PM_ADDR_FLAG_BACKUP                      (1 << 2)
+#define MPTCP_PM_ADDR_FLAG_FULLMESH                    (1 << 3)
 
 enum {
        MPTCP_PM_CMD_UNSPEC,
index 64a39f3..bf5b9b4 100644 (file)
@@ -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)
 {
        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)) {
                local = select_local_address(pernet, msk);
                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.subflows++;
                        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);
-                       __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);
                        return;
                }