[Twisted-Python] linux kernel 2.6.11-rc broke twisted process pipes

James Y Knight foom at fuhm.net
Mon Feb 28 20:40:00 MST 2005


On Feb 28, 2005, at 7:37 PM, Andrea Arcangeli wrote:
> It's never the inverse since pollin was always forced to be set, both
> for readers and writers, so a write fd was always returned as readable
> unconditionally.

I'm glad we all agree as to what the new behavior should be, but you're 
incorrect about the extent of the old brokenness. Simple tests prove 
that POLLIN was not forced to be true always:

On a debian 2.6.9-1-k7 kernel:

====
import os,select
r,w=os.pipe()
p=select.poll()
p.register(w, select.POLLIN|select.POLLOUT)

print select.select([w],[w],[w],0)
print p.poll(0)

os.write(w, "asdf")
print select.select([w],[w],[w],0)
print p.poll(0)

os.close(r)
print select.select([w],[w],[w],0)
print p.poll(0)
====

outputs:
([], [4], [])
[(4, 4)] (aka POLLOUT)
([4], [], [])
[(4, 1)] (aka POLLIN)
([4], [4], [])
[(4, 9)] (aka POLLIN|POLLERR)

> It's POLLERR not POLLHUP. POLLHUP is set only when the "writer" side
> disconnected and you're listening to a reader fd. POLLERR is instead
> returned when the "reader" disconnected and you're listening to a
> writer fd.

Right. Twisted treats all of POLLERR/HUP/NVAL the same, anyways.

> 671         if (events & (POLLIN | POLLRDNORM)) {
> 672                 if ((rpipe->pipe_buffer.cnt > 0) ||
> 673                     (rpipe->pipe_state & PIPE_EOF))
> 674                         revents |= events & (POLLIN | POLLRDNORM);
> 675         }
> The way I read it, even openbsd will report the fd as readable
> regardless if the channel is disconnected, if the buffer has something
> into it.

Nope -- Notice that rpipe and wpipe are backwards for the read and 
write fds. "rpipe" on the write fd won't ever have any data in it. 
(Except of course that pipes in BSD are bidirectional!). In BSD, 
everything is symmetric, so it'd be very difficult for a stupid bug 
like reporting the same status for both ends to occur.

> Are you reall sure it can't be called with pollreactor?
>
> Sorry for asking again but I really want to be sure select won't screw
> things up with >1024 fds open ;)

Yes, it will get called, because linux returns (returned?) 
POLLERR|POLLIN when the other side is closed. pollreactor doesn't 
assume the connection was lost immediately when POLLERR/etc is set, if 
POLLIN is also set, because POLLIN|POLLHUP means there is more data 
available to be read from the kernel buffer that was sent before the 
connection closed. And, since normal transport implementations of 
doRead actually do try to read, if it was lying about POLLIN, that's 
okay: the read syscall will fail, and CONNECTION_LOST will be returned 
from doRead.

James





More information about the Twisted-Python mailing list