On Sat, Oct 5, 2013 at 10:34 AM, Greg KH <gregkh@xxxxxxxxxxxxxxxxxxx> wrote: > > One fixes the reported regression in the n_tty code that a number of > people found recently That one looks broken. Well, it looks like it might "work", but do so by hiding the issue for one case, while leaving it in the more general case. Why does it do that up_read(&tty->termios_rwsem); tty_flush_to_ldisc(tty); down_read(&tty->termios_rwsem); only if TTY_OTHER_CLOSED is set? If flushing the ldisc can generate more data, then we should do it *unconditionally* when we see that we currently have no data to read. As it is, it looks like the patch fixes the TTY_OTHER_CLOSED case ("read all pending data before returning -EIO"), but it leaves the same broken case for the O_NONBLOCK case and for a hung up tty. The O_NONBLOCK case is presumably just a performance problem (the data will come at _some_ point later), but it just looks bad in general. And the tty_hung_up_p() looks actiely buggy, with the same bug as the TTY_OTHER_CLOSED case that the patch tried to fix. Hmm? Am I missing something? The code is a bit confusing in *other* ways too: if you look later, it does this: n_tty_set_room(tty); which is documented to have to happen inside the termios_rwsem. HOWEVER, what does that do? It actually does an _asynchronous_ queue_work() of &tty->port->buf.work, in case there is now more room, and the previous one was blocked. And guess what that workqueue is all about? Right: it's flush_to_ldisc() - which is the work that tty_flush_to_ldisc() is trying to flush. So we're actually basically making sure we've flush the previous pending work. So even the tty_flush_to_ldisc(tty) that gets done in that patch is not necessarily sufficient, because the work might not have been scheduled because the flip buffer used to be full. Then flushing the work won't do anything, even though there is actually more data. Now, that is a very unlikely situation (I think it requires two concurrent readers), but it looks like it might be real. So I suspect we should *unconditionally* do n_tty_set_room(tty); up_read(&tty->termios_rwsem); tty_flush_to_ldisc(tty); down_read(&tty->termios_rwsem); if we don't have any pending input. And then test input_available_p() again. And only if we don't have any input after that flushing do we start doing the whole TTY_OTHER_CLOSED and tty_hung_up_p() tests. Hmm? Linus -- To unsubscribe from this list: send the line "unsubscribe linux-serial" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html