From: Jon Maloy <jma...@redhat.com>

We introduce a new macro TIPC_ANY_SCOPE to make the handling of the
lookup scope value more comprehensible during multicast reception.

The (unchanged) rules go as follows:

1) Multicast messages sent from own node are delivered to all matching
   sockets on the own node, irrespective of their binding scope.

2) Multicast messages sent from other nodes arrive here because they
   have found TIPC_CLUSTER_SCOPE bindings coming from this node. Those
   messages should be delivered to exactly those sockets, but not to
   local sockets bound with TIPC_NODE_SCOPE, since the latter obviously
   were not meant to be visible for those senders.

3) Group multicast/broadcast messages are delivered to the sockets with
   a binding scope matching exactly the lookup scope indicated in the
   message header, and nobodey else.

Signed-off-by: Jon Maloy <jma...@redhat.com>
---
 net/tipc/name_table.c |  6 +++---
 net/tipc/name_table.h |  4 +++-
 net/tipc/socket.c     | 19 +++++++------------
 3 files changed, 13 insertions(+), 16 deletions(-)

diff --git a/net/tipc/name_table.c b/net/tipc/name_table.c
index 6db9f9e7c0ac..86007bcaf47c 100644
--- a/net/tipc/name_table.c
+++ b/net/tipc/name_table.c
@@ -671,12 +671,12 @@ bool tipc_nametbl_lookup_group(struct net *net, struct 
tipc_uaddr *ua,
  * Returns a list of local sockets
  */
 void tipc_nametbl_lookup_mcast_sockets(struct net *net, struct tipc_uaddr *ua,
-                                      bool exact, struct list_head *dports)
+                                      struct list_head *dports)
 {
        struct service_range *sr;
        struct tipc_service *sc;
        struct publication *p;
-       u32 scope = ua->scope;
+       u8 scope = ua->scope;
 
        rcu_read_lock();
        sc = tipc_service_find(net, ua);
@@ -686,7 +686,7 @@ void tipc_nametbl_lookup_mcast_sockets(struct net *net, 
struct tipc_uaddr *ua,
        spin_lock_bh(&sc->lock);
        service_range_foreach_match(sr, sc, ua->sr.lower, ua->sr.upper) {
                list_for_each_entry(p, &sr->local_publ, local_publ) {
-                       if (p->scope == scope || (!exact && p->scope < scope))
+                       if (scope == p->scope || scope == TIPC_ANY_SCOPE)
                                tipc_dest_push(dports, 0, p->sk.ref);
                }
        }
diff --git a/net/tipc/name_table.h b/net/tipc/name_table.h
index c7c9a3ddd420..148b0f640959 100644
--- a/net/tipc/name_table.h
+++ b/net/tipc/name_table.h
@@ -51,6 +51,8 @@ struct tipc_uaddr;
 #define TIPC_PUBL_SCOPE_NUM    (TIPC_NODE_SCOPE + 1)
 #define TIPC_NAMETBL_SIZE      1024    /* must be a power of 2 */
 
+#define TIPC_ANY_SCOPE 255
+
 /**
  * struct publication - info about a published service address or range
  * @sr: service range represented by this publication
@@ -113,7 +115,7 @@ int tipc_nl_name_table_dump(struct sk_buff *skb, struct 
netlink_callback *cb);
 bool tipc_nametbl_lookup_anycast(struct net *net, struct tipc_uaddr *ua,
                                 struct tipc_socket_addr *sk);
 void tipc_nametbl_lookup_mcast_sockets(struct net *net, struct tipc_uaddr *ua,
-                                      bool exact, struct list_head *dports);
+                                      struct list_head *dports);
 void tipc_nametbl_lookup_mcast_nodes(struct net *net, struct tipc_uaddr *ua,
                                     struct tipc_nlist *nodes);
 bool tipc_nametbl_lookup_group(struct net *net, struct tipc_uaddr *ua,
diff --git a/net/tipc/socket.c b/net/tipc/socket.c
index 61aefb9f6fd8..c85859b73adf 100644
--- a/net/tipc/socket.c
+++ b/net/tipc/socket.c
@@ -1200,7 +1200,6 @@ void tipc_sk_mcast_rcv(struct net *net, struct 
sk_buff_head *arrvq,
        struct tipc_msg *hdr;
        struct tipc_uaddr ua;
        int user, mtyp, hlen;
-       bool exact;
 
        __skb_queue_head_init(&tmpq);
        INIT_LIST_HEAD(&dports);
@@ -1214,6 +1213,12 @@ void tipc_sk_mcast_rcv(struct net *net, struct 
sk_buff_head *arrvq,
                hlen = skb_headroom(skb) + msg_hdr_sz(hdr);
                onode = msg_orignode(hdr);
                ua.sr.type = msg_nametype(hdr);
+               ua.sr.lower = msg_namelower(hdr);
+               ua.sr.upper = msg_nameupper(hdr);
+               if (onode == self)
+                       ua.scope = TIPC_ANY_SCOPE;
+               else
+                       ua.scope = TIPC_CLUSTER_SCOPE;
 
                if (mtyp == TIPC_GRP_UCAST_MSG || user == GROUP_PROTOCOL) {
                        spin_lock_bh(&inputq->lock);
@@ -1231,20 +1236,10 @@ void tipc_sk_mcast_rcv(struct net *net, struct 
sk_buff_head *arrvq,
                        ua.sr.lower = 0;
                        ua.sr.upper = ~0;
                        ua.scope = msg_lookup_scope(hdr);
-                       exact = true;
-               } else {
-                       /* TIPC_NODE_SCOPE means "any scope" in this context */
-                       if (onode == self)
-                               ua.scope = TIPC_NODE_SCOPE;
-                       else
-                               ua.scope = TIPC_CLUSTER_SCOPE;
-                       exact = false;
-                       ua.sr.lower = msg_namelower(hdr);
-                       ua.sr.upper = msg_nameupper(hdr);
                }
 
                /* Create destination port list: */
-               tipc_nametbl_lookup_mcast_sockets(net, &ua, exact, &dports);
+               tipc_nametbl_lookup_mcast_sockets(net, &ua, &dports);
 
                /* Clone message per destination */
                while (tipc_dest_pop(&dports, NULL, &portid)) {
-- 
2.29.2



_______________________________________________
tipc-discussion mailing list
tipc-discussion@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/tipc-discussion

Reply via email to