]> err.no Git - linux-2.6/commit
[SCTP]: A better solution to fix the race between sctp_peeloff() and
authorVladislav Yasevich <vladislav.yasevich@hp.com>
Fri, 19 May 2006 18:01:18 +0000 (11:01 -0700)
committerSridhar Samudrala <sri@us.ibm.com>
Fri, 19 May 2006 18:01:18 +0000 (11:01 -0700)
commit61c9fed41638249f8b6ca5345064eb1beb50179f
tree8855a0e6cbee58a9d94e30396b7d0a1baa526900
parent8de8c8738086501bbe3057ed6f4b70dded657488
[SCTP]: A better solution to fix the race between sctp_peeloff() and
sctp_rcv().

The goal is to hold the ref on the association/endpoint throughout the
state-machine process.  We accomplish like this:

  /* ref on the assoc/ep is taken during lookup */

  if owned_by_user(sk)
  sctp_add_backlog(skb, sk);
  else
  inqueue_push(skb, sk);

  /* drop the ref on the assoc/ep */

However, in sctp_add_backlog() we take the ref on assoc/ep and hold it
while the skb is on the backlog queue.  This allows us to get rid of the
sock_hold/sock_put in the lookup routines.

Now sctp_backlog_rcv() needs to account for potential association move.
In the unlikely event that association moved, we need to retest if the
new socket is locked by user.  If we don't this, we may have two packets
racing up the stack toward the same socket and we can't deal with it.
If the new socket is still locked, we'll just add the skb to its backlog
continuing to hold the ref on the association.  This get's rid of the
need to move packets from one backlog to another and it also safe in
case new packets arrive on the same backlog queue.

The last step, is to lock the new socket when we are moving the
association to it.  This is needed in case any new packets arrive on
the association when it moved.  We want these to go to the backlog since
we would like to avoid the race between this new packet and a packet
that may be sitting on the backlog queue of the old socket toward the
same association.

Signed-off-by: Vladislav Yasevich <vladislav.yasevich@hp.com>
Signed-off-by: Sridhar Samudrala <sri@us.ibm.com>
net/sctp/input.c
net/sctp/socket.c