提交 55db4c64 编写于 作者: L Linus Torvalds

Revert "tty: make receive_buf() return the amout of bytes received"

This reverts commit b1c43f82.

It was broken in so many ways, and results in random odd pty issues.

It re-introduced the buggy schedule_work() in flush_to_ldisc() that can
cause endless work-loops (see commit a5660b41: "tty: fix endless
work loop when the buffer fills up").

It also used an "unsigned int" return value fo the ->receive_buf()
function, but then made multiple functions return a negative error code,
and didn't actually check for the error in the caller.

And it didn't actually work at all.  BenH bisected down odd tty behavior
to it:
  "It looks like the patch is causing some major malfunctions of the X
   server for me, possibly related to PTYs.  For example, cat'ing a
   large file in a gnome terminal hangs the kernel for -minutes- in a
   loop of what looks like flush_to_ldisc/workqueue code, (some ftrace
   data in the quoted bits further down).

   ...

   Some more data: It -looks- like what happens is that the
   flush_to_ldisc work queue entry constantly re-queues itself (because
   the PTY is full ?) and the workqueue thread will basically loop
   forver calling it without ever scheduling, thus starving the consumer
   process that could have emptied the PTY."

which is pretty much exactly the problem we fixed in a5660b41.

Milton Miller pointed out the 'unsigned int' issue.
Reported-by: NBenjamin Herrenschmidt <benh@kernel.crashing.org>
Reported-by: NMilton Miller <miltonm@bga.com>
Cc: Stefan Bigler <stefan.bigler@keymile.com>
Cc: Toby Gray <toby.gray@realvnc.com>
Cc: Felipe Balbi <balbi@ti.com>
Cc: Greg Kroah-Hartman <gregkh@suse.de>
Cc: Alan Cox <alan@lxorguk.ukuu.org.uk>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 1fa7b6a2
...@@ -355,29 +355,24 @@ static void hci_uart_tty_wakeup(struct tty_struct *tty) ...@@ -355,29 +355,24 @@ static void hci_uart_tty_wakeup(struct tty_struct *tty)
* flags pointer to flags for data * flags pointer to flags for data
* count count of received data in bytes * count count of received data in bytes
* *
* Return Value: Number of bytes received * Return Value: None
*/ */
static unsigned int hci_uart_tty_receive(struct tty_struct *tty, static void hci_uart_tty_receive(struct tty_struct *tty, const u8 *data, char *flags, int count)
const u8 *data, char *flags, int count)
{ {
struct hci_uart *hu = (void *)tty->disc_data; struct hci_uart *hu = (void *)tty->disc_data;
int received;
if (!hu || tty != hu->tty) if (!hu || tty != hu->tty)
return -ENODEV; return;
if (!test_bit(HCI_UART_PROTO_SET, &hu->flags)) if (!test_bit(HCI_UART_PROTO_SET, &hu->flags))
return -EINVAL; return;
spin_lock(&hu->rx_lock); spin_lock(&hu->rx_lock);
received = hu->proto->recv(hu, (void *) data, count); hu->proto->recv(hu, (void *) data, count);
if (received > 0) hu->hdev->stat.byte_rx += count;
hu->hdev->stat.byte_rx += received;
spin_unlock(&hu->rx_lock); spin_unlock(&hu->rx_lock);
tty_unthrottle(tty); tty_unthrottle(tty);
return received;
} }
static int hci_uart_register_dev(struct hci_uart *hu) static int hci_uart_register_dev(struct hci_uart *hu)
......
...@@ -120,21 +120,17 @@ static void serport_ldisc_close(struct tty_struct *tty) ...@@ -120,21 +120,17 @@ static void serport_ldisc_close(struct tty_struct *tty)
* 'interrupt' routine. * 'interrupt' routine.
*/ */
static unsigned int serport_ldisc_receive(struct tty_struct *tty, static void serport_ldisc_receive(struct tty_struct *tty, const unsigned char *cp, char *fp, int count)
const unsigned char *cp, char *fp, int count)
{ {
struct serport *serport = (struct serport*) tty->disc_data; struct serport *serport = (struct serport*) tty->disc_data;
unsigned long flags; unsigned long flags;
unsigned int ch_flags; unsigned int ch_flags;
int ret = 0;
int i; int i;
spin_lock_irqsave(&serport->lock, flags); spin_lock_irqsave(&serport->lock, flags);
if (!test_bit(SERPORT_ACTIVE, &serport->flags)) { if (!test_bit(SERPORT_ACTIVE, &serport->flags))
ret = -EINVAL;
goto out; goto out;
}
for (i = 0; i < count; i++) { for (i = 0; i < count; i++) {
switch (fp[i]) { switch (fp[i]) {
...@@ -156,8 +152,6 @@ static unsigned int serport_ldisc_receive(struct tty_struct *tty, ...@@ -156,8 +152,6 @@ static unsigned int serport_ldisc_receive(struct tty_struct *tty,
out: out:
spin_unlock_irqrestore(&serport->lock, flags); spin_unlock_irqrestore(&serport->lock, flags);
return ret == 0 ? count : ret;
} }
/* /*
......
...@@ -674,7 +674,7 @@ gigaset_tty_ioctl(struct tty_struct *tty, struct file *file, ...@@ -674,7 +674,7 @@ gigaset_tty_ioctl(struct tty_struct *tty, struct file *file,
* cflags buffer containing error flags for received characters (ignored) * cflags buffer containing error flags for received characters (ignored)
* count number of received characters * count number of received characters
*/ */
static unsigned int static void
gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf, gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf,
char *cflags, int count) char *cflags, int count)
{ {
...@@ -683,12 +683,12 @@ gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -683,12 +683,12 @@ gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf,
struct inbuf_t *inbuf; struct inbuf_t *inbuf;
if (!cs) if (!cs)
return -ENODEV; return;
inbuf = cs->inbuf; inbuf = cs->inbuf;
if (!inbuf) { if (!inbuf) {
dev_err(cs->dev, "%s: no inbuf\n", __func__); dev_err(cs->dev, "%s: no inbuf\n", __func__);
cs_put(cs); cs_put(cs);
return -EINVAL; return;
} }
tail = inbuf->tail; tail = inbuf->tail;
...@@ -725,8 +725,6 @@ gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -725,8 +725,6 @@ gigaset_tty_receive(struct tty_struct *tty, const unsigned char *buf,
gig_dbg(DEBUG_INTR, "%s-->BH", __func__); gig_dbg(DEBUG_INTR, "%s-->BH", __func__);
gigaset_schedule_event(cs); gigaset_schedule_event(cs);
cs_put(cs); cs_put(cs);
return count;
} }
/* /*
......
...@@ -747,8 +747,8 @@ static void st_tty_close(struct tty_struct *tty) ...@@ -747,8 +747,8 @@ static void st_tty_close(struct tty_struct *tty)
pr_debug("%s: done ", __func__); pr_debug("%s: done ", __func__);
} }
static unsigned int st_tty_receive(struct tty_struct *tty, static void st_tty_receive(struct tty_struct *tty, const unsigned char *data,
const unsigned char *data, char *tty_flags, int count) char *tty_flags, int count)
{ {
#ifdef VERBOSE #ifdef VERBOSE
print_hex_dump(KERN_DEBUG, ">in>", DUMP_PREFIX_NONE, print_hex_dump(KERN_DEBUG, ">in>", DUMP_PREFIX_NONE,
...@@ -761,8 +761,6 @@ static unsigned int st_tty_receive(struct tty_struct *tty, ...@@ -761,8 +761,6 @@ static unsigned int st_tty_receive(struct tty_struct *tty,
*/ */
st_recv(tty->disc_data, data, count); st_recv(tty->disc_data, data, count);
pr_debug("done %s", __func__); pr_debug("done %s", __func__);
return count;
} }
/* wake-up function called in from the TTY layer /* wake-up function called in from the TTY layer
......
...@@ -167,8 +167,8 @@ static inline void debugfs_tx(struct ser_device *ser, const u8 *data, int size) ...@@ -167,8 +167,8 @@ static inline void debugfs_tx(struct ser_device *ser, const u8 *data, int size)
#endif #endif
static unsigned int ldisc_receive(struct tty_struct *tty, static void ldisc_receive(struct tty_struct *tty, const u8 *data,
const u8 *data, char *flags, int count) char *flags, int count)
{ {
struct sk_buff *skb = NULL; struct sk_buff *skb = NULL;
struct ser_device *ser; struct ser_device *ser;
...@@ -215,8 +215,6 @@ static unsigned int ldisc_receive(struct tty_struct *tty, ...@@ -215,8 +215,6 @@ static unsigned int ldisc_receive(struct tty_struct *tty,
} else } else
++ser->dev->stats.rx_dropped; ++ser->dev->stats.rx_dropped;
update_tty_status(ser); update_tty_status(ser);
return count;
} }
static int handle_tx(struct ser_device *ser) static int handle_tx(struct ser_device *ser)
......
...@@ -425,17 +425,16 @@ static void slc_setup(struct net_device *dev) ...@@ -425,17 +425,16 @@ static void slc_setup(struct net_device *dev)
* in parallel * in parallel
*/ */
static unsigned int slcan_receive_buf(struct tty_struct *tty, static void slcan_receive_buf(struct tty_struct *tty,
const unsigned char *cp, char *fp, int count) const unsigned char *cp, char *fp, int count)
{ {
struct slcan *sl = (struct slcan *) tty->disc_data; struct slcan *sl = (struct slcan *) tty->disc_data;
int bytes = count;
if (!sl || sl->magic != SLCAN_MAGIC || !netif_running(sl->dev)) if (!sl || sl->magic != SLCAN_MAGIC || !netif_running(sl->dev))
return -ENODEV; return;
/* Read the characters out of the buffer */ /* Read the characters out of the buffer */
while (bytes--) { while (count--) {
if (fp && *fp++) { if (fp && *fp++) {
if (!test_and_set_bit(SLF_ERROR, &sl->flags)) if (!test_and_set_bit(SLF_ERROR, &sl->flags))
sl->dev->stats.rx_errors++; sl->dev->stats.rx_errors++;
...@@ -444,8 +443,6 @@ static unsigned int slcan_receive_buf(struct tty_struct *tty, ...@@ -444,8 +443,6 @@ static unsigned int slcan_receive_buf(struct tty_struct *tty,
} }
slcan_unesc(sl, *cp++); slcan_unesc(sl, *cp++);
} }
return count;
} }
/************************************ /************************************
......
...@@ -456,7 +456,7 @@ static void sixpack_write_wakeup(struct tty_struct *tty) ...@@ -456,7 +456,7 @@ static void sixpack_write_wakeup(struct tty_struct *tty)
* a block of 6pack data has been received, which can now be decapsulated * a block of 6pack data has been received, which can now be decapsulated
* and sent on to some IP layer for further processing. * and sent on to some IP layer for further processing.
*/ */
static unsigned int sixpack_receive_buf(struct tty_struct *tty, static void sixpack_receive_buf(struct tty_struct *tty,
const unsigned char *cp, char *fp, int count) const unsigned char *cp, char *fp, int count)
{ {
struct sixpack *sp; struct sixpack *sp;
...@@ -464,11 +464,11 @@ static unsigned int sixpack_receive_buf(struct tty_struct *tty, ...@@ -464,11 +464,11 @@ static unsigned int sixpack_receive_buf(struct tty_struct *tty,
int count1; int count1;
if (!count) if (!count)
return 0; return;
sp = sp_get(tty); sp = sp_get(tty);
if (!sp) if (!sp)
return -ENODEV; return;
memcpy(buf, cp, count < sizeof(buf) ? count : sizeof(buf)); memcpy(buf, cp, count < sizeof(buf) ? count : sizeof(buf));
...@@ -487,8 +487,6 @@ static unsigned int sixpack_receive_buf(struct tty_struct *tty, ...@@ -487,8 +487,6 @@ static unsigned int sixpack_receive_buf(struct tty_struct *tty,
sp_put(sp); sp_put(sp);
tty_unthrottle(tty); tty_unthrottle(tty);
return count1;
} }
/* /*
......
...@@ -923,14 +923,13 @@ static long mkiss_compat_ioctl(struct tty_struct *tty, struct file *file, ...@@ -923,14 +923,13 @@ static long mkiss_compat_ioctl(struct tty_struct *tty, struct file *file,
* a block of data has been received, which can now be decapsulated * a block of data has been received, which can now be decapsulated
* and sent on to the AX.25 layer for further processing. * and sent on to the AX.25 layer for further processing.
*/ */
static unsigned int mkiss_receive_buf(struct tty_struct *tty, static void mkiss_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
struct mkiss *ax = mkiss_get(tty); struct mkiss *ax = mkiss_get(tty);
int bytes = count;
if (!ax) if (!ax)
return -ENODEV; return;
/* /*
* Argh! mtu change time! - costs us the packet part received * Argh! mtu change time! - costs us the packet part received
...@@ -940,7 +939,7 @@ static unsigned int mkiss_receive_buf(struct tty_struct *tty, ...@@ -940,7 +939,7 @@ static unsigned int mkiss_receive_buf(struct tty_struct *tty,
ax_changedmtu(ax); ax_changedmtu(ax);
/* Read the characters out of the buffer */ /* Read the characters out of the buffer */
while (bytes--) { while (count--) {
if (fp != NULL && *fp++) { if (fp != NULL && *fp++) {
if (!test_and_set_bit(AXF_ERROR, &ax->flags)) if (!test_and_set_bit(AXF_ERROR, &ax->flags))
ax->dev->stats.rx_errors++; ax->dev->stats.rx_errors++;
...@@ -953,8 +952,6 @@ static unsigned int mkiss_receive_buf(struct tty_struct *tty, ...@@ -953,8 +952,6 @@ static unsigned int mkiss_receive_buf(struct tty_struct *tty,
mkiss_put(ax); mkiss_put(ax);
tty_unthrottle(tty); tty_unthrottle(tty);
return count;
} }
/* /*
......
...@@ -216,23 +216,23 @@ static int irtty_do_write(struct sir_dev *dev, const unsigned char *ptr, size_t ...@@ -216,23 +216,23 @@ static int irtty_do_write(struct sir_dev *dev, const unsigned char *ptr, size_t
* usbserial: urb-complete-interrupt / softint * usbserial: urb-complete-interrupt / softint
*/ */
static unsigned int irtty_receive_buf(struct tty_struct *tty, static void irtty_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
struct sir_dev *dev; struct sir_dev *dev;
struct sirtty_cb *priv = tty->disc_data; struct sirtty_cb *priv = tty->disc_data;
int i; int i;
IRDA_ASSERT(priv != NULL, return -ENODEV;); IRDA_ASSERT(priv != NULL, return;);
IRDA_ASSERT(priv->magic == IRTTY_MAGIC, return -EINVAL;); IRDA_ASSERT(priv->magic == IRTTY_MAGIC, return;);
if (unlikely(count==0)) /* yes, this happens */ if (unlikely(count==0)) /* yes, this happens */
return 0; return;
dev = priv->dev; dev = priv->dev;
if (!dev) { if (!dev) {
IRDA_WARNING("%s(), not ready yet!\n", __func__); IRDA_WARNING("%s(), not ready yet!\n", __func__);
return -ENODEV; return;
} }
for (i = 0; i < count; i++) { for (i = 0; i < count; i++) {
...@@ -242,13 +242,11 @@ static unsigned int irtty_receive_buf(struct tty_struct *tty, ...@@ -242,13 +242,11 @@ static unsigned int irtty_receive_buf(struct tty_struct *tty,
if (fp && *fp++) { if (fp && *fp++) {
IRDA_DEBUG(0, "Framing or parity error!\n"); IRDA_DEBUG(0, "Framing or parity error!\n");
sirdev_receive(dev, NULL, 0); /* notify sir_dev (updating stats) */ sirdev_receive(dev, NULL, 0); /* notify sir_dev (updating stats) */
return -EINVAL; return;
} }
} }
sirdev_receive(dev, cp, count); sirdev_receive(dev, cp, count);
return count;
} }
/* /*
......
...@@ -340,7 +340,7 @@ ppp_asynctty_poll(struct tty_struct *tty, struct file *file, poll_table *wait) ...@@ -340,7 +340,7 @@ ppp_asynctty_poll(struct tty_struct *tty, struct file *file, poll_table *wait)
} }
/* May sleep, don't call from interrupt level or with interrupts disabled */ /* May sleep, don't call from interrupt level or with interrupts disabled */
static unsigned int static void
ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf, ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf,
char *cflags, int count) char *cflags, int count)
{ {
...@@ -348,7 +348,7 @@ ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -348,7 +348,7 @@ ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf,
unsigned long flags; unsigned long flags;
if (!ap) if (!ap)
return -ENODEV; return;
spin_lock_irqsave(&ap->recv_lock, flags); spin_lock_irqsave(&ap->recv_lock, flags);
ppp_async_input(ap, buf, cflags, count); ppp_async_input(ap, buf, cflags, count);
spin_unlock_irqrestore(&ap->recv_lock, flags); spin_unlock_irqrestore(&ap->recv_lock, flags);
...@@ -356,8 +356,6 @@ ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -356,8 +356,6 @@ ppp_asynctty_receive(struct tty_struct *tty, const unsigned char *buf,
tasklet_schedule(&ap->tsk); tasklet_schedule(&ap->tsk);
ap_put(ap); ap_put(ap);
tty_unthrottle(tty); tty_unthrottle(tty);
return count;
} }
static void static void
......
...@@ -381,7 +381,7 @@ ppp_sync_poll(struct tty_struct *tty, struct file *file, poll_table *wait) ...@@ -381,7 +381,7 @@ ppp_sync_poll(struct tty_struct *tty, struct file *file, poll_table *wait)
} }
/* May sleep, don't call from interrupt level or with interrupts disabled */ /* May sleep, don't call from interrupt level or with interrupts disabled */
static unsigned int static void
ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf, ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf,
char *cflags, int count) char *cflags, int count)
{ {
...@@ -389,7 +389,7 @@ ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -389,7 +389,7 @@ ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf,
unsigned long flags; unsigned long flags;
if (!ap) if (!ap)
return -ENODEV; return;
spin_lock_irqsave(&ap->recv_lock, flags); spin_lock_irqsave(&ap->recv_lock, flags);
ppp_sync_input(ap, buf, cflags, count); ppp_sync_input(ap, buf, cflags, count);
spin_unlock_irqrestore(&ap->recv_lock, flags); spin_unlock_irqrestore(&ap->recv_lock, flags);
...@@ -397,8 +397,6 @@ ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf, ...@@ -397,8 +397,6 @@ ppp_sync_receive(struct tty_struct *tty, const unsigned char *buf,
tasklet_schedule(&ap->tsk); tasklet_schedule(&ap->tsk);
sp_put(ap); sp_put(ap);
tty_unthrottle(tty); tty_unthrottle(tty);
return count;
} }
static void static void
......
...@@ -670,17 +670,16 @@ static void sl_setup(struct net_device *dev) ...@@ -670,17 +670,16 @@ static void sl_setup(struct net_device *dev)
* in parallel * in parallel
*/ */
static unsigned int slip_receive_buf(struct tty_struct *tty, static void slip_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
struct slip *sl = tty->disc_data; struct slip *sl = tty->disc_data;
int bytes = count;
if (!sl || sl->magic != SLIP_MAGIC || !netif_running(sl->dev)) if (!sl || sl->magic != SLIP_MAGIC || !netif_running(sl->dev))
return -ENODEV; return;
/* Read the characters out of the buffer */ /* Read the characters out of the buffer */
while (bytes--) { while (count--) {
if (fp && *fp++) { if (fp && *fp++) {
if (!test_and_set_bit(SLF_ERROR, &sl->flags)) if (!test_and_set_bit(SLF_ERROR, &sl->flags))
sl->dev->stats.rx_errors++; sl->dev->stats.rx_errors++;
...@@ -694,8 +693,6 @@ static unsigned int slip_receive_buf(struct tty_struct *tty, ...@@ -694,8 +693,6 @@ static unsigned int slip_receive_buf(struct tty_struct *tty,
#endif #endif
slip_unesc(sl, *cp++); slip_unesc(sl, *cp++);
} }
return count;
} }
/************************************ /************************************
......
...@@ -517,18 +517,17 @@ static int x25_asy_close(struct net_device *dev) ...@@ -517,18 +517,17 @@ static int x25_asy_close(struct net_device *dev)
* and sent on to some IP layer for further processing. * and sent on to some IP layer for further processing.
*/ */
static unsigned int x25_asy_receive_buf(struct tty_struct *tty, static void x25_asy_receive_buf(struct tty_struct *tty,
const unsigned char *cp, char *fp, int count) const unsigned char *cp, char *fp, int count)
{ {
struct x25_asy *sl = tty->disc_data; struct x25_asy *sl = tty->disc_data;
int bytes = count;
if (!sl || sl->magic != X25_ASY_MAGIC || !netif_running(sl->dev)) if (!sl || sl->magic != X25_ASY_MAGIC || !netif_running(sl->dev))
return; return;
/* Read the characters out of the buffer */ /* Read the characters out of the buffer */
while (bytes--) { while (count--) {
if (fp && *fp++) { if (fp && *fp++) {
if (!test_and_set_bit(SLF_ERROR, &sl->flags)) if (!test_and_set_bit(SLF_ERROR, &sl->flags))
sl->dev->stats.rx_errors++; sl->dev->stats.rx_errors++;
...@@ -537,8 +536,6 @@ static unsigned int x25_asy_receive_buf(struct tty_struct *tty, ...@@ -537,8 +536,6 @@ static unsigned int x25_asy_receive_buf(struct tty_struct *tty,
} }
x25_asy_unesc(sl, *cp++); x25_asy_unesc(sl, *cp++);
} }
return count;
} }
/* /*
......
...@@ -2128,8 +2128,8 @@ static void gsmld_detach_gsm(struct tty_struct *tty, struct gsm_mux *gsm) ...@@ -2128,8 +2128,8 @@ static void gsmld_detach_gsm(struct tty_struct *tty, struct gsm_mux *gsm)
gsm->tty = NULL; gsm->tty = NULL;
} }
static unsigned int gsmld_receive_buf(struct tty_struct *tty, static void gsmld_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
struct gsm_mux *gsm = tty->disc_data; struct gsm_mux *gsm = tty->disc_data;
const unsigned char *dp; const unsigned char *dp;
...@@ -2162,8 +2162,6 @@ static unsigned int gsmld_receive_buf(struct tty_struct *tty, ...@@ -2162,8 +2162,6 @@ static unsigned int gsmld_receive_buf(struct tty_struct *tty,
} }
/* FASYNC if needed ? */ /* FASYNC if needed ? */
/* If clogged call tty_throttle(tty); */ /* If clogged call tty_throttle(tty); */
return count;
} }
/** /**
......
...@@ -188,8 +188,8 @@ static unsigned int n_hdlc_tty_poll(struct tty_struct *tty, struct file *filp, ...@@ -188,8 +188,8 @@ static unsigned int n_hdlc_tty_poll(struct tty_struct *tty, struct file *filp,
poll_table *wait); poll_table *wait);
static int n_hdlc_tty_open(struct tty_struct *tty); static int n_hdlc_tty_open(struct tty_struct *tty);
static void n_hdlc_tty_close(struct tty_struct *tty); static void n_hdlc_tty_close(struct tty_struct *tty);
static unsigned int n_hdlc_tty_receive(struct tty_struct *tty, static void n_hdlc_tty_receive(struct tty_struct *tty, const __u8 *cp,
const __u8 *cp, char *fp, int count); char *fp, int count);
static void n_hdlc_tty_wakeup(struct tty_struct *tty); static void n_hdlc_tty_wakeup(struct tty_struct *tty);
#define bset(p,b) ((p)[(b) >> 5] |= (1 << ((b) & 0x1f))) #define bset(p,b) ((p)[(b) >> 5] |= (1 << ((b) & 0x1f)))
...@@ -509,8 +509,8 @@ static void n_hdlc_tty_wakeup(struct tty_struct *tty) ...@@ -509,8 +509,8 @@ static void n_hdlc_tty_wakeup(struct tty_struct *tty)
* Called by tty low level driver when receive data is available. Data is * Called by tty low level driver when receive data is available. Data is
* interpreted as one HDLC frame. * interpreted as one HDLC frame.
*/ */
static unsigned int n_hdlc_tty_receive(struct tty_struct *tty, static void n_hdlc_tty_receive(struct tty_struct *tty, const __u8 *data,
const __u8 *data, char *flags, int count) char *flags, int count)
{ {
register struct n_hdlc *n_hdlc = tty2n_hdlc (tty); register struct n_hdlc *n_hdlc = tty2n_hdlc (tty);
register struct n_hdlc_buf *buf; register struct n_hdlc_buf *buf;
...@@ -521,20 +521,20 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty, ...@@ -521,20 +521,20 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty,
/* This can happen if stuff comes in on the backup tty */ /* This can happen if stuff comes in on the backup tty */
if (!n_hdlc || tty != n_hdlc->tty) if (!n_hdlc || tty != n_hdlc->tty)
return -ENODEV; return;
/* verify line is using HDLC discipline */ /* verify line is using HDLC discipline */
if (n_hdlc->magic != HDLC_MAGIC) { if (n_hdlc->magic != HDLC_MAGIC) {
printk("%s(%d) line not using HDLC discipline\n", printk("%s(%d) line not using HDLC discipline\n",
__FILE__,__LINE__); __FILE__,__LINE__);
return -EINVAL; return;
} }
if ( count>maxframe ) { if ( count>maxframe ) {
if (debuglevel >= DEBUG_LEVEL_INFO) if (debuglevel >= DEBUG_LEVEL_INFO)
printk("%s(%d) rx count>maxframesize, data discarded\n", printk("%s(%d) rx count>maxframesize, data discarded\n",
__FILE__,__LINE__); __FILE__,__LINE__);
return -EINVAL; return;
} }
/* get a free HDLC buffer */ /* get a free HDLC buffer */
...@@ -550,7 +550,7 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty, ...@@ -550,7 +550,7 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty,
if (debuglevel >= DEBUG_LEVEL_INFO) if (debuglevel >= DEBUG_LEVEL_INFO)
printk("%s(%d) no more rx buffers, data discarded\n", printk("%s(%d) no more rx buffers, data discarded\n",
__FILE__,__LINE__); __FILE__,__LINE__);
return -EINVAL; return;
} }
/* copy received data to HDLC buffer */ /* copy received data to HDLC buffer */
...@@ -565,8 +565,6 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty, ...@@ -565,8 +565,6 @@ static unsigned int n_hdlc_tty_receive(struct tty_struct *tty,
if (n_hdlc->tty->fasync != NULL) if (n_hdlc->tty->fasync != NULL)
kill_fasync (&n_hdlc->tty->fasync, SIGIO, POLL_IN); kill_fasync (&n_hdlc->tty->fasync, SIGIO, POLL_IN);
return count;
} /* end of n_hdlc_tty_receive() */ } /* end of n_hdlc_tty_receive() */
/** /**
......
...@@ -139,8 +139,8 @@ static int r3964_ioctl(struct tty_struct *tty, struct file *file, ...@@ -139,8 +139,8 @@ static int r3964_ioctl(struct tty_struct *tty, struct file *file,
static void r3964_set_termios(struct tty_struct *tty, struct ktermios *old); static void r3964_set_termios(struct tty_struct *tty, struct ktermios *old);
static unsigned int r3964_poll(struct tty_struct *tty, struct file *file, static unsigned int r3964_poll(struct tty_struct *tty, struct file *file,
struct poll_table_struct *wait); struct poll_table_struct *wait);
static unsigned int r3964_receive_buf(struct tty_struct *tty, static void r3964_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count); char *fp, int count);
static struct tty_ldisc_ops tty_ldisc_N_R3964 = { static struct tty_ldisc_ops tty_ldisc_N_R3964 = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
...@@ -1239,8 +1239,8 @@ static unsigned int r3964_poll(struct tty_struct *tty, struct file *file, ...@@ -1239,8 +1239,8 @@ static unsigned int r3964_poll(struct tty_struct *tty, struct file *file,
return result; return result;
} }
static unsigned int r3964_receive_buf(struct tty_struct *tty, static void r3964_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
struct r3964_info *pInfo = tty->disc_data; struct r3964_info *pInfo = tty->disc_data;
const unsigned char *p; const unsigned char *p;
...@@ -1257,8 +1257,6 @@ static unsigned int r3964_receive_buf(struct tty_struct *tty, ...@@ -1257,8 +1257,6 @@ static unsigned int r3964_receive_buf(struct tty_struct *tty,
} }
} }
return count;
} }
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
......
...@@ -81,6 +81,38 @@ static inline int tty_put_user(struct tty_struct *tty, unsigned char x, ...@@ -81,6 +81,38 @@ static inline int tty_put_user(struct tty_struct *tty, unsigned char x,
return put_user(x, ptr); return put_user(x, ptr);
} }
/**
* n_tty_set__room - receive space
* @tty: terminal
*
* Called by the driver to find out how much data it is
* permitted to feed to the line discipline without any being lost
* and thus to manage flow control. Not serialized. Answers for the
* "instant".
*/
static void n_tty_set_room(struct tty_struct *tty)
{
/* tty->read_cnt is not read locked ? */
int left = N_TTY_BUF_SIZE - tty->read_cnt - 1;
int old_left;
/*
* If we are doing input canonicalization, and there are no
* pending newlines, let characters through without limit, so
* that erase characters will be handled. Other excess
* characters will be beeped.
*/
if (left <= 0)
left = tty->icanon && !tty->canon_data;
old_left = tty->receive_room;
tty->receive_room = left;
/* Did this open up the receive buffer? We may need to flip */
if (left && !old_left)
schedule_work(&tty->buf.work);
}
static void put_tty_queue_nolock(unsigned char c, struct tty_struct *tty) static void put_tty_queue_nolock(unsigned char c, struct tty_struct *tty)
{ {
if (tty->read_cnt < N_TTY_BUF_SIZE) { if (tty->read_cnt < N_TTY_BUF_SIZE) {
...@@ -152,6 +184,7 @@ static void reset_buffer_flags(struct tty_struct *tty) ...@@ -152,6 +184,7 @@ static void reset_buffer_flags(struct tty_struct *tty)
tty->canon_head = tty->canon_data = tty->erasing = 0; tty->canon_head = tty->canon_data = tty->erasing = 0;
memset(&tty->read_flags, 0, sizeof tty->read_flags); memset(&tty->read_flags, 0, sizeof tty->read_flags);
n_tty_set_room(tty);
check_unthrottle(tty); check_unthrottle(tty);
} }
...@@ -1327,19 +1360,17 @@ static void n_tty_write_wakeup(struct tty_struct *tty) ...@@ -1327,19 +1360,17 @@ static void n_tty_write_wakeup(struct tty_struct *tty)
* calls one at a time and in order (or using flush_to_ldisc) * calls one at a time and in order (or using flush_to_ldisc)
*/ */
static unsigned int n_tty_receive_buf(struct tty_struct *tty, static void n_tty_receive_buf(struct tty_struct *tty, const unsigned char *cp,
const unsigned char *cp, char *fp, int count) char *fp, int count)
{ {
const unsigned char *p; const unsigned char *p;
char *f, flags = TTY_NORMAL; char *f, flags = TTY_NORMAL;
int i; int i;
char buf[64]; char buf[64];
unsigned long cpuflags; unsigned long cpuflags;
int left;
int ret = 0;
if (!tty->read_buf) if (!tty->read_buf)
return 0; return;
if (tty->real_raw) { if (tty->real_raw) {
spin_lock_irqsave(&tty->read_lock, cpuflags); spin_lock_irqsave(&tty->read_lock, cpuflags);
...@@ -1349,7 +1380,6 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty, ...@@ -1349,7 +1380,6 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty,
memcpy(tty->read_buf + tty->read_head, cp, i); memcpy(tty->read_buf + tty->read_head, cp, i);
tty->read_head = (tty->read_head + i) & (N_TTY_BUF_SIZE-1); tty->read_head = (tty->read_head + i) & (N_TTY_BUF_SIZE-1);
tty->read_cnt += i; tty->read_cnt += i;
ret += i;
cp += i; cp += i;
count -= i; count -= i;
...@@ -1359,10 +1389,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty, ...@@ -1359,10 +1389,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty,
memcpy(tty->read_buf + tty->read_head, cp, i); memcpy(tty->read_buf + tty->read_head, cp, i);
tty->read_head = (tty->read_head + i) & (N_TTY_BUF_SIZE-1); tty->read_head = (tty->read_head + i) & (N_TTY_BUF_SIZE-1);
tty->read_cnt += i; tty->read_cnt += i;
ret += i;
spin_unlock_irqrestore(&tty->read_lock, cpuflags); spin_unlock_irqrestore(&tty->read_lock, cpuflags);
} else { } else {
ret = count;
for (i = count, p = cp, f = fp; i; i--, p++) { for (i = count, p = cp, f = fp; i; i--, p++) {
if (f) if (f)
flags = *f++; flags = *f++;
...@@ -1390,6 +1418,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty, ...@@ -1390,6 +1418,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty,
tty->ops->flush_chars(tty); tty->ops->flush_chars(tty);
} }
n_tty_set_room(tty);
if ((!tty->icanon && (tty->read_cnt >= tty->minimum_to_wake)) || if ((!tty->icanon && (tty->read_cnt >= tty->minimum_to_wake)) ||
L_EXTPROC(tty)) { L_EXTPROC(tty)) {
kill_fasync(&tty->fasync, SIGIO, POLL_IN); kill_fasync(&tty->fasync, SIGIO, POLL_IN);
...@@ -1402,12 +1432,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty, ...@@ -1402,12 +1432,8 @@ static unsigned int n_tty_receive_buf(struct tty_struct *tty,
* mode. We don't want to throttle the driver if we're in * mode. We don't want to throttle the driver if we're in
* canonical mode and don't have a newline yet! * canonical mode and don't have a newline yet!
*/ */
left = N_TTY_BUF_SIZE - tty->read_cnt - 1; if (tty->receive_room < TTY_THRESHOLD_THROTTLE)
if (left < TTY_THRESHOLD_THROTTLE)
tty_throttle(tty); tty_throttle(tty);
return ret;
} }
int is_ignored(int sig) int is_ignored(int sig)
...@@ -1451,6 +1477,7 @@ static void n_tty_set_termios(struct tty_struct *tty, struct ktermios *old) ...@@ -1451,6 +1477,7 @@ static void n_tty_set_termios(struct tty_struct *tty, struct ktermios *old)
if (test_bit(TTY_HW_COOK_IN, &tty->flags)) { if (test_bit(TTY_HW_COOK_IN, &tty->flags)) {
tty->raw = 1; tty->raw = 1;
tty->real_raw = 1; tty->real_raw = 1;
n_tty_set_room(tty);
return; return;
} }
if (I_ISTRIP(tty) || I_IUCLC(tty) || I_IGNCR(tty) || if (I_ISTRIP(tty) || I_IUCLC(tty) || I_IGNCR(tty) ||
...@@ -1503,6 +1530,7 @@ static void n_tty_set_termios(struct tty_struct *tty, struct ktermios *old) ...@@ -1503,6 +1530,7 @@ static void n_tty_set_termios(struct tty_struct *tty, struct ktermios *old)
else else
tty->real_raw = 0; tty->real_raw = 0;
} }
n_tty_set_room(tty);
/* The termios change make the tty ready for I/O */ /* The termios change make the tty ready for I/O */
wake_up_interruptible(&tty->write_wait); wake_up_interruptible(&tty->write_wait);
wake_up_interruptible(&tty->read_wait); wake_up_interruptible(&tty->read_wait);
...@@ -1784,6 +1812,8 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file, ...@@ -1784,6 +1812,8 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file,
retval = -ERESTARTSYS; retval = -ERESTARTSYS;
break; break;
} }
/* FIXME: does n_tty_set_room need locking ? */
n_tty_set_room(tty);
timeout = schedule_timeout(timeout); timeout = schedule_timeout(timeout);
continue; continue;
} }
...@@ -1855,8 +1885,10 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file, ...@@ -1855,8 +1885,10 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file,
* longer than TTY_THRESHOLD_UNTHROTTLE in canonical mode, * longer than TTY_THRESHOLD_UNTHROTTLE in canonical mode,
* we won't get any more characters. * we won't get any more characters.
*/ */
if (n_tty_chars_in_buffer(tty) <= TTY_THRESHOLD_UNTHROTTLE) if (n_tty_chars_in_buffer(tty) <= TTY_THRESHOLD_UNTHROTTLE) {
n_tty_set_room(tty);
check_unthrottle(tty); check_unthrottle(tty);
}
if (b - buf >= minimum) if (b - buf >= minimum)
break; break;
...@@ -1878,6 +1910,7 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file, ...@@ -1878,6 +1910,7 @@ static ssize_t n_tty_read(struct tty_struct *tty, struct file *file,
} else if (test_and_clear_bit(TTY_PUSH, &tty->flags)) } else if (test_and_clear_bit(TTY_PUSH, &tty->flags))
goto do_it_again; goto do_it_again;
n_tty_set_room(tty);
return retval; return retval;
} }
......
...@@ -416,7 +416,6 @@ static void flush_to_ldisc(struct work_struct *work) ...@@ -416,7 +416,6 @@ static void flush_to_ldisc(struct work_struct *work)
struct tty_buffer *head, *tail = tty->buf.tail; struct tty_buffer *head, *tail = tty->buf.tail;
int seen_tail = 0; int seen_tail = 0;
while ((head = tty->buf.head) != NULL) { while ((head = tty->buf.head) != NULL) {
int copied;
int count; int count;
char *char_buf; char *char_buf;
unsigned char *flag_buf; unsigned char *flag_buf;
...@@ -443,19 +442,17 @@ static void flush_to_ldisc(struct work_struct *work) ...@@ -443,19 +442,17 @@ static void flush_to_ldisc(struct work_struct *work)
line discipline as we want to empty the queue */ line discipline as we want to empty the queue */
if (test_bit(TTY_FLUSHPENDING, &tty->flags)) if (test_bit(TTY_FLUSHPENDING, &tty->flags))
break; break;
if (!tty->receive_room || seen_tail)
break;
if (count > tty->receive_room)
count = tty->receive_room;
char_buf = head->char_buf_ptr + head->read; char_buf = head->char_buf_ptr + head->read;
flag_buf = head->flag_buf_ptr + head->read; flag_buf = head->flag_buf_ptr + head->read;
head->read += count;
spin_unlock_irqrestore(&tty->buf.lock, flags); spin_unlock_irqrestore(&tty->buf.lock, flags);
copied = disc->ops->receive_buf(tty, char_buf, disc->ops->receive_buf(tty, char_buf,
flag_buf, count); flag_buf, count);
spin_lock_irqsave(&tty->buf.lock, flags); spin_lock_irqsave(&tty->buf.lock, flags);
head->read += copied;
if (copied == 0 || seen_tail) {
schedule_work(&tty->buf.work);
break;
}
} }
clear_bit(TTY_FLUSHING, &tty->flags); clear_bit(TTY_FLUSHING, &tty->flags);
} }
......
...@@ -332,7 +332,8 @@ int paste_selection(struct tty_struct *tty) ...@@ -332,7 +332,8 @@ int paste_selection(struct tty_struct *tty)
continue; continue;
} }
count = sel_buffer_lth - pasted; count = sel_buffer_lth - pasted;
count = tty->ldisc->ops->receive_buf(tty, sel_buffer + pasted, count = min(count, tty->receive_room);
tty->ldisc->ops->receive_buf(tty, sel_buffer + pasted,
NULL, count); NULL, count);
pasted += count; pasted += count;
} }
......
...@@ -76,7 +76,7 @@ ...@@ -76,7 +76,7 @@
* tty device. It is solely the responsibility of the line * tty device. It is solely the responsibility of the line
* discipline to handle poll requests. * discipline to handle poll requests.
* *
* unsigned int (*receive_buf)(struct tty_struct *, const unsigned char *cp, * void (*receive_buf)(struct tty_struct *, const unsigned char *cp,
* char *fp, int count); * char *fp, int count);
* *
* This function is called by the low-level tty driver to send * This function is called by the low-level tty driver to send
...@@ -84,8 +84,7 @@ ...@@ -84,8 +84,7 @@
* processing. <cp> is a pointer to the buffer of input * processing. <cp> is a pointer to the buffer of input
* character received by the device. <fp> is a pointer to a * character received by the device. <fp> is a pointer to a
* pointer of flag bytes which indicate whether a character was * pointer of flag bytes which indicate whether a character was
* received with a parity error, etc. Returns the amount of bytes * received with a parity error, etc.
* received.
* *
* void (*write_wakeup)(struct tty_struct *); * void (*write_wakeup)(struct tty_struct *);
* *
...@@ -141,8 +140,8 @@ struct tty_ldisc_ops { ...@@ -141,8 +140,8 @@ struct tty_ldisc_ops {
/* /*
* The following routines are called from below. * The following routines are called from below.
*/ */
unsigned int (*receive_buf)(struct tty_struct *, void (*receive_buf)(struct tty_struct *, const unsigned char *cp,
const unsigned char *cp, char *fp, int count); char *fp, int count);
void (*write_wakeup)(struct tty_struct *); void (*write_wakeup)(struct tty_struct *);
void (*dcd_change)(struct tty_struct *, unsigned int, void (*dcd_change)(struct tty_struct *, unsigned int,
struct pps_event_time *); struct pps_event_time *);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册