Commit f662c070 authored by Allan Stephens's avatar Allan Stephens Committed by David S. Miller

tipc: correct problems with misleading flags returned using poll()

Prevent TIPC from incorrectly setting returned flags to poll()
in the following cases:

- an unconnected socket no longer indicates that it is always readable

- an unconnected, connecting, or listening socket no longer indicates
  that it is always writable
Signed-off-by: default avatarAllan Stephens <allan.stephens@windriver.com>
Signed-off-by: default avatarPaul Gortmaker <paul.gortmaker@windriver.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 35997e31
...@@ -429,36 +429,55 @@ static int get_name(struct socket *sock, struct sockaddr *uaddr, ...@@ -429,36 +429,55 @@ static int get_name(struct socket *sock, struct sockaddr *uaddr,
* to handle any preventable race conditions, so TIPC will do the same ... * to handle any preventable race conditions, so TIPC will do the same ...
* *
* TIPC sets the returned events as follows: * TIPC sets the returned events as follows:
* a) POLLRDNORM and POLLIN are set if the socket's receive queue is non-empty *
* or if a connection-oriented socket is does not have an active connection * socket state flags set
* (i.e. a read operation will not block). * ------------ ---------
* b) POLLOUT is set except when a socket's connection has been terminated * unconnected no read flags
* (i.e. a write operation will not block). * no write flags
* c) POLLHUP is set when a socket's connection has been terminated. *
* * connecting POLLIN/POLLRDNORM if ACK/NACK in rx queue
* IMPORTANT: The fact that a read or write operation will not block does NOT * no write flags
* imply that the operation will succeed! *
* connected POLLIN/POLLRDNORM if data in rx queue
* POLLOUT if port is not congested
*
* disconnecting POLLIN/POLLRDNORM/POLLHUP
* no write flags
*
* listening POLLIN if SYN in rx queue
* no write flags
*
* ready POLLIN/POLLRDNORM if data in rx queue
* [connectionless] POLLOUT (since port cannot be congested)
*
* IMPORTANT: The fact that a read or write operation is indicated does NOT
* imply that the operation will succeed, merely that it should be performed
* and will not block.
*/ */
static unsigned int poll(struct file *file, struct socket *sock, static unsigned int poll(struct file *file, struct socket *sock,
poll_table *wait) poll_table *wait)
{ {
struct sock *sk = sock->sk; struct sock *sk = sock->sk;
u32 mask; u32 mask = 0;
poll_wait(file, sk_sleep(sk), wait); poll_wait(file, sk_sleep(sk), wait);
if (!skb_queue_empty(&sk->sk_receive_queue) || switch ((int)sock->state) {
(sock->state == SS_UNCONNECTED) || case SS_READY:
(sock->state == SS_DISCONNECTING)) case SS_CONNECTED:
mask = (POLLRDNORM | POLLIN); if (!tipc_sk_port(sk)->congested)
else mask |= POLLOUT;
mask = 0; /* fall thru' */
case SS_CONNECTING:
if (sock->state == SS_DISCONNECTING) case SS_LISTENING:
mask |= POLLHUP; if (!skb_queue_empty(&sk->sk_receive_queue))
else mask |= (POLLIN | POLLRDNORM);
mask |= POLLOUT; break;
case SS_DISCONNECTING:
mask = (POLLIN | POLLRDNORM | POLLHUP);
break;
}
return mask; return mask;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment