tipc: ensure sequential message delivery across dual bearers
When we run broadcast packets over dual bearers/interfaces, the current transmission code is flipping bearers between each sent packet, with the purpose of leveraging the double bandwidth available. The receiving bclink is resequencing the packets if needed, so all messages are delivered upwards from the broadcast link in the correct order, even if they may arrive in concurrent interrupts. However, at the moment of delivery upwards to the socket, we release all spinlocks (bclink_lock, node_lock), so it is still possible that arriving messages bypass each other before they reach the socket queue. We fix this by applying the same technique we are using for unicast traffic. We use a link selector (i.e., the last bit of sending port number) to ensure that messages from the same sender socket always are sent over the same bearer. This guarantees sequential delivery between socket pairs, which is sufficient to satisfy the protocol spec, as well as all known user requirements. Signed-off-by: Jon Maloy <jon.maloy@ericsson.com> Reviewed-by: Erik Hugne <erik.hugne@ericsson.com> Reviewed-by: Ying Xue <ying.xue@windriver.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
9fbfb8b120
commit
6f92ee54b3
1 changed files with 5 additions and 12 deletions
|
@ -637,6 +637,7 @@ static int tipc_bcbearer_send(struct sk_buff *buf, struct tipc_bearer *unused1,
|
|||
struct tipc_media_addr *unused2)
|
||||
{
|
||||
int bp_index;
|
||||
struct tipc_msg *msg = buf_msg(buf);
|
||||
|
||||
/* Prepare broadcast link message for reliable transmission,
|
||||
* if first time trying to send it;
|
||||
|
@ -644,10 +645,7 @@ static int tipc_bcbearer_send(struct sk_buff *buf, struct tipc_bearer *unused1,
|
|||
* since they are sent in an unreliable manner and don't need it
|
||||
*/
|
||||
if (likely(!msg_non_seq(buf_msg(buf)))) {
|
||||
struct tipc_msg *msg;
|
||||
|
||||
bcbuf_set_acks(buf, bclink->bcast_nodes.count);
|
||||
msg = buf_msg(buf);
|
||||
msg_set_non_seq(msg, 1);
|
||||
msg_set_mc_netid(msg, tipc_net_id);
|
||||
bcl->stats.sent_info++;
|
||||
|
@ -664,12 +662,14 @@ static int tipc_bcbearer_send(struct sk_buff *buf, struct tipc_bearer *unused1,
|
|||
for (bp_index = 0; bp_index < MAX_BEARERS; bp_index++) {
|
||||
struct tipc_bearer *p = bcbearer->bpairs[bp_index].primary;
|
||||
struct tipc_bearer *s = bcbearer->bpairs[bp_index].secondary;
|
||||
struct tipc_bearer *b = p;
|
||||
struct tipc_bearer *bp[2] = {p, s};
|
||||
struct tipc_bearer *b = bp[msg_link_selector(msg)];
|
||||
struct sk_buff *tbuf;
|
||||
|
||||
if (!p)
|
||||
break; /* No more bearers to try */
|
||||
|
||||
if (!b)
|
||||
b = p;
|
||||
tipc_nmap_diff(&bcbearer->remains, &b->nodes,
|
||||
&bcbearer->remains_new);
|
||||
if (bcbearer->remains_new.count == bcbearer->remains.count)
|
||||
|
@ -686,13 +686,6 @@ static int tipc_bcbearer_send(struct sk_buff *buf, struct tipc_bearer *unused1,
|
|||
tipc_bearer_send(b->identity, tbuf, &b->bcast_addr);
|
||||
kfree_skb(tbuf); /* Bearer keeps a clone */
|
||||
}
|
||||
|
||||
/* Swap bearers for next packet */
|
||||
if (s) {
|
||||
bcbearer->bpairs[bp_index].primary = s;
|
||||
bcbearer->bpairs[bp_index].secondary = p;
|
||||
}
|
||||
|
||||
if (bcbearer->remains_new.count == 0)
|
||||
break; /* All targets reached */
|
||||
|
||||
|
|
Loading…
Reference in a new issue