Using libibverbs
, to do a send
operation, these actions happen in this order:
- Receiver posts a
receive
request usingibv_post_receive()
- Sender posts a
send
request usingibv_post_send()
- A completion queue element
IBV_WC_RECV
get pushed on the queue of the receiver - Receiver pops this element using
ibv_poll_cq()
- A completion queue element
IBV_WC_SEND
get pushed on the queue of the sender - Sender pops this element using
ibv_poll_cq()
My question are the followings:
- Is the order correct and guaranteed? I asking myself about 3 and 5, wheter first the sender's completion queue is pushed or the receiver's.
- When is the memory buffer of the receiver updated? Is it between 2 and 3 or between 4 and 5?
You can't really put an exact order between things that happen on the sender and on the receiver, since the only way they causally interact is by sending network messages, with have a non-zero latency.
Another way to think about what happens is:
Now at this point there is no strict order required by the spec, since the ACK packet travels over the network independently of how the receiver adapter proceeds to the next step, but things that happen next:
If the network latency is extremely low and the receiver adapter runs slowly for some reason, it is possible that by some external timebase the sender completion appears before the receive completion.
The one guarantee that we can make is that the contents of the receive buffer are updated strictly before both the receive completion and send completion are pushed. But there are corner cases, for example if the network fails before delivering the ACK packet back to the sender, it's possible that no successful send completion is generated even though the receiver will see the buffer updated and the receive completion generated.