2019-08-25 09:49:16 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
/*
|
2007-10-16 08:26:41 +00:00
|
|
|
* Copyright (C) 2000 - 2007 Jeff Dike (jdike@{linux.intel,addtoit}.com)
|
2005-04-16 22:20:36 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/tty.h>
|
|
|
|
#include <linux/tty_flip.h>
|
2011-08-18 19:08:29 +00:00
|
|
|
#include "chan.h"
|
2012-10-08 02:27:32 +00:00
|
|
|
#include <os.h>
|
|
|
|
#include <irq_kern.h>
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-09-23 04:44:22 +00:00
|
|
|
#ifdef CONFIG_NOCONFIG_CHAN
|
2007-02-10 09:43:53 +00:00
|
|
|
static void *not_configged_init(char *str, int device,
|
|
|
|
const struct chan_opts *opts)
|
2005-09-23 04:44:22 +00:00
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return NULL;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int not_configged_open(int input, int output, int primary, void *data,
|
|
|
|
char **dev_out)
|
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return -ENODEV;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void not_configged_close(int fd, void *data)
|
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
|
|
|
}
|
|
|
|
|
2023-12-06 07:37:11 +00:00
|
|
|
static int not_configged_read(int fd, u8 *c_out, void *data)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return -EIO;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:37:11 +00:00
|
|
|
static int not_configged_write(int fd, const u8 *buf, size_t len, void *data)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return -EIO;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-14 00:07:11 +00:00
|
|
|
static int not_configged_console_write(int fd, const char *buf, int len)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return -EIO;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int not_configged_window_size(int fd, void *data, unsigned short *rows,
|
|
|
|
unsigned short *cols)
|
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
2006-01-06 08:18:50 +00:00
|
|
|
return -ENODEV;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void not_configged_free(void *data)
|
|
|
|
{
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_ERR "Using a channel type which is configured out of "
|
2005-04-16 22:20:36 +00:00
|
|
|
"UML\n");
|
|
|
|
}
|
|
|
|
|
2006-09-27 08:50:33 +00:00
|
|
|
static const struct chan_ops not_configged_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.init = not_configged_init,
|
|
|
|
.open = not_configged_open,
|
|
|
|
.close = not_configged_close,
|
|
|
|
.read = not_configged_read,
|
|
|
|
.write = not_configged_write,
|
|
|
|
.console_write = not_configged_console_write,
|
|
|
|
.window_size = not_configged_window_size,
|
|
|
|
.free = not_configged_free,
|
|
|
|
.winch = 0,
|
|
|
|
};
|
|
|
|
#endif /* CONFIG_NOCONFIG_CHAN */
|
|
|
|
|
2023-10-18 12:36:43 +00:00
|
|
|
static inline bool need_output_blocking(void)
|
|
|
|
{
|
|
|
|
return time_travel_mode == TT_MODE_INFCPU ||
|
|
|
|
time_travel_mode == TT_MODE_EXTERNAL;
|
|
|
|
}
|
|
|
|
|
2006-01-06 08:18:50 +00:00
|
|
|
static int open_one_chan(struct chan *chan)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-07-31 07:37:44 +00:00
|
|
|
int fd, err;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->opened)
|
2006-01-06 08:18:50 +00:00
|
|
|
return 0;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->ops->open == NULL)
|
2006-01-06 08:18:50 +00:00
|
|
|
fd = 0;
|
|
|
|
else fd = (*chan->ops->open)(chan->input, chan->output, chan->primary,
|
|
|
|
chan->data, &chan->dev);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (fd < 0)
|
2006-01-06 08:18:50 +00:00
|
|
|
return fd;
|
2007-07-31 07:37:44 +00:00
|
|
|
|
|
|
|
err = os_set_fd_block(fd, 0);
|
2023-10-18 12:36:43 +00:00
|
|
|
if (err)
|
|
|
|
goto out_close;
|
|
|
|
|
|
|
|
chan->fd_in = fd;
|
|
|
|
chan->fd_out = fd;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* In time-travel modes infinite-CPU and external we need to guarantee
|
|
|
|
* that any writes to the output succeed immdiately from the point of
|
|
|
|
* the VM. The best way to do this is to put the FD in blocking mode
|
|
|
|
* and simply wait/retry until everything is written.
|
|
|
|
* As every write is guaranteed to complete, we also do not need to
|
|
|
|
* request an IRQ for the output.
|
|
|
|
*
|
|
|
|
* Note that input cannot happen in a time synchronized way. We permit
|
|
|
|
* it, but time passes very quickly if anything waits for a read.
|
|
|
|
*/
|
|
|
|
if (chan->output && need_output_blocking()) {
|
|
|
|
err = os_dup_file(chan->fd_out);
|
|
|
|
if (err < 0)
|
|
|
|
goto out_close;
|
2007-07-31 07:37:44 +00:00
|
|
|
|
2023-10-18 12:36:43 +00:00
|
|
|
chan->fd_out = err;
|
|
|
|
|
|
|
|
err = os_set_fd_block(chan->fd_out, 1);
|
|
|
|
if (err) {
|
|
|
|
os_close_file(chan->fd_out);
|
|
|
|
goto out_close;
|
|
|
|
}
|
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
chan->opened = 1;
|
2006-01-06 08:18:50 +00:00
|
|
|
return 0;
|
2023-10-18 12:36:43 +00:00
|
|
|
|
|
|
|
out_close:
|
|
|
|
(*chan->ops->close)(fd, chan->data);
|
|
|
|
return err;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2008-04-28 09:13:55 +00:00
|
|
|
static int open_chan(struct list_head *chans)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
struct list_head *ele;
|
|
|
|
struct chan *chan;
|
|
|
|
int ret, err = 0;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
list_for_each(ele, chans) {
|
2005-04-16 22:20:36 +00:00
|
|
|
chan = list_entry(ele, struct chan, list);
|
2006-01-06 08:18:50 +00:00
|
|
|
ret = open_one_chan(chan);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->primary)
|
2006-01-06 08:18:50 +00:00
|
|
|
err = ret;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2006-01-06 08:18:50 +00:00
|
|
|
return err;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2013-03-11 09:05:45 +00:00
|
|
|
void chan_enable_winch(struct chan *chan, struct tty_port *port)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2011-09-08 14:49:34 +00:00
|
|
|
if (chan && chan->primary && chan->ops->winch)
|
2023-10-18 12:36:43 +00:00
|
|
|
register_winch(chan->fd_in, port);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2011-09-10 12:17:04 +00:00
|
|
|
static void line_timer_cb(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct line *line = container_of(work, struct line, task.work);
|
|
|
|
|
|
|
|
if (!line->throttled)
|
um: line: Use separate IRQs per line
Today, all possible serial lines (ssl*=) as well as all
possible consoles (con*=) each share a single interrupt
(with a fixed number) with others of the same type.
Now, if you have two lines, say ssl0 and ssl1, and one
of them is connected to an fd you cannot read (e.g. a
file), but the other gets a read interrupt, then both
of them get the interrupt since it's shared. Then, the
read() call will return EOF, since it's a file being
written and there's nothing to read (at least not at
the current offset, at the end).
Unfortunately, this is treated as a read error, and we
close this line, losing all the possible output.
It might be possible to work around this and make the
IRQ sharing work, however, now that we have dynamically
allocated IRQs that are easy to use, simply use that to
achieve separating between the events; then there's no
interrupt for that line and we never attempt the read
in the first place, thus not closing the line.
This manifested itself in the wifi hostap/hwsim tests
where the parallel script communicates via one serial
console and the kernel messages go to another (a file)
and sending data on the communication console caused
the kernel messages to stop flowing into the file.
Reported-by: Jouni Malinen <j@w1.fi>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Acked-By: anton ivanov <anton.ivanov@cambridgegreys.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
2022-05-06 13:46:12 +00:00
|
|
|
chan_interrupt(line, line->read_irq);
|
2011-09-10 12:17:04 +00:00
|
|
|
}
|
|
|
|
|
2007-07-16 06:38:54 +00:00
|
|
|
int enable_chan(struct line *line)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
struct list_head *ele;
|
|
|
|
struct chan *chan;
|
2007-07-16 06:38:54 +00:00
|
|
|
int err;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2011-09-10 12:17:04 +00:00
|
|
|
INIT_DELAYED_WORK(&line->task, line_timer_cb);
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
list_for_each(ele, &line->chan_list) {
|
2005-04-16 22:20:36 +00:00
|
|
|
chan = list_entry(ele, struct chan, list);
|
2007-07-16 06:38:54 +00:00
|
|
|
err = open_one_chan(chan);
|
|
|
|
if (err) {
|
|
|
|
if (chan->primary)
|
|
|
|
goto out_close;
|
|
|
|
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
continue;
|
2007-07-16 06:38:54 +00:00
|
|
|
}
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->enabled)
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
continue;
|
2023-10-18 12:36:43 +00:00
|
|
|
err = line_setup_irq(chan->fd_in, chan->input,
|
|
|
|
chan->output && !need_output_blocking(),
|
|
|
|
line, chan);
|
2007-07-16 06:38:54 +00:00
|
|
|
if (err)
|
|
|
|
goto out_close;
|
|
|
|
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
chan->enabled = 1;
|
|
|
|
}
|
2007-07-16 06:38:54 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_close:
|
2011-09-10 12:39:18 +00:00
|
|
|
close_chan(line);
|
2007-07-16 06:38:54 +00:00
|
|
|
return err;
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
}
|
|
|
|
|
2019-05-06 12:39:35 +00:00
|
|
|
/* Items are added in IRQ context, when free_irq can't be called, and
|
|
|
|
* removed in process context, when it can.
|
|
|
|
* This handles interrupt sources which disappear, and which need to
|
|
|
|
* be permanently disabled. This is discovered in IRQ context, but
|
|
|
|
* the freeing of the IRQ must be done later.
|
|
|
|
*/
|
|
|
|
static DEFINE_SPINLOCK(irqs_to_free_lock);
|
|
|
|
static LIST_HEAD(irqs_to_free);
|
|
|
|
|
|
|
|
void free_irqs(void)
|
|
|
|
{
|
|
|
|
struct chan *chan;
|
|
|
|
LIST_HEAD(list);
|
|
|
|
struct list_head *ele;
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&irqs_to_free_lock, flags);
|
|
|
|
list_splice_init(&irqs_to_free, &list);
|
|
|
|
spin_unlock_irqrestore(&irqs_to_free_lock, flags);
|
|
|
|
|
|
|
|
list_for_each(ele, &list) {
|
|
|
|
chan = list_entry(ele, struct chan, free_list);
|
|
|
|
|
|
|
|
if (chan->input && chan->enabled)
|
um: line: Use separate IRQs per line
Today, all possible serial lines (ssl*=) as well as all
possible consoles (con*=) each share a single interrupt
(with a fixed number) with others of the same type.
Now, if you have two lines, say ssl0 and ssl1, and one
of them is connected to an fd you cannot read (e.g. a
file), but the other gets a read interrupt, then both
of them get the interrupt since it's shared. Then, the
read() call will return EOF, since it's a file being
written and there's nothing to read (at least not at
the current offset, at the end).
Unfortunately, this is treated as a read error, and we
close this line, losing all the possible output.
It might be possible to work around this and make the
IRQ sharing work, however, now that we have dynamically
allocated IRQs that are easy to use, simply use that to
achieve separating between the events; then there's no
interrupt for that line and we never attempt the read
in the first place, thus not closing the line.
This manifested itself in the wifi hostap/hwsim tests
where the parallel script communicates via one serial
console and the kernel messages go to another (a file)
and sending data on the communication console caused
the kernel messages to stop flowing into the file.
Reported-by: Jouni Malinen <j@w1.fi>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Acked-By: anton ivanov <anton.ivanov@cambridgegreys.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
2022-05-06 13:46:12 +00:00
|
|
|
um_free_irq(chan->line->read_irq, chan);
|
2023-10-18 12:36:43 +00:00
|
|
|
if (chan->output && chan->enabled &&
|
|
|
|
!need_output_blocking())
|
um: line: Use separate IRQs per line
Today, all possible serial lines (ssl*=) as well as all
possible consoles (con*=) each share a single interrupt
(with a fixed number) with others of the same type.
Now, if you have two lines, say ssl0 and ssl1, and one
of them is connected to an fd you cannot read (e.g. a
file), but the other gets a read interrupt, then both
of them get the interrupt since it's shared. Then, the
read() call will return EOF, since it's a file being
written and there's nothing to read (at least not at
the current offset, at the end).
Unfortunately, this is treated as a read error, and we
close this line, losing all the possible output.
It might be possible to work around this and make the
IRQ sharing work, however, now that we have dynamically
allocated IRQs that are easy to use, simply use that to
achieve separating between the events; then there's no
interrupt for that line and we never attempt the read
in the first place, thus not closing the line.
This manifested itself in the wifi hostap/hwsim tests
where the parallel script communicates via one serial
console and the kernel messages go to another (a file)
and sending data on the communication console caused
the kernel messages to stop flowing into the file.
Reported-by: Jouni Malinen <j@w1.fi>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Acked-By: anton ivanov <anton.ivanov@cambridgegreys.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
2022-05-06 13:46:12 +00:00
|
|
|
um_free_irq(chan->line->write_irq, chan);
|
2019-05-06 12:39:35 +00:00
|
|
|
chan->enabled = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
static void close_one_chan(struct chan *chan, int delay_free_irq)
|
|
|
|
{
|
2019-05-06 12:39:35 +00:00
|
|
|
unsigned long flags;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (!chan->opened)
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
return;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2019-05-06 12:39:35 +00:00
|
|
|
if (delay_free_irq) {
|
|
|
|
spin_lock_irqsave(&irqs_to_free_lock, flags);
|
|
|
|
list_add(&chan->free_list, &irqs_to_free);
|
|
|
|
spin_unlock_irqrestore(&irqs_to_free_lock, flags);
|
|
|
|
} else {
|
|
|
|
if (chan->input && chan->enabled)
|
um: line: Use separate IRQs per line
Today, all possible serial lines (ssl*=) as well as all
possible consoles (con*=) each share a single interrupt
(with a fixed number) with others of the same type.
Now, if you have two lines, say ssl0 and ssl1, and one
of them is connected to an fd you cannot read (e.g. a
file), but the other gets a read interrupt, then both
of them get the interrupt since it's shared. Then, the
read() call will return EOF, since it's a file being
written and there's nothing to read (at least not at
the current offset, at the end).
Unfortunately, this is treated as a read error, and we
close this line, losing all the possible output.
It might be possible to work around this and make the
IRQ sharing work, however, now that we have dynamically
allocated IRQs that are easy to use, simply use that to
achieve separating between the events; then there's no
interrupt for that line and we never attempt the read
in the first place, thus not closing the line.
This manifested itself in the wifi hostap/hwsim tests
where the parallel script communicates via one serial
console and the kernel messages go to another (a file)
and sending data on the communication console caused
the kernel messages to stop flowing into the file.
Reported-by: Jouni Malinen <j@w1.fi>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Acked-By: anton ivanov <anton.ivanov@cambridgegreys.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
2022-05-06 13:46:12 +00:00
|
|
|
um_free_irq(chan->line->read_irq, chan);
|
2023-10-18 12:36:43 +00:00
|
|
|
if (chan->output && chan->enabled &&
|
|
|
|
!need_output_blocking())
|
um: line: Use separate IRQs per line
Today, all possible serial lines (ssl*=) as well as all
possible consoles (con*=) each share a single interrupt
(with a fixed number) with others of the same type.
Now, if you have two lines, say ssl0 and ssl1, and one
of them is connected to an fd you cannot read (e.g. a
file), but the other gets a read interrupt, then both
of them get the interrupt since it's shared. Then, the
read() call will return EOF, since it's a file being
written and there's nothing to read (at least not at
the current offset, at the end).
Unfortunately, this is treated as a read error, and we
close this line, losing all the possible output.
It might be possible to work around this and make the
IRQ sharing work, however, now that we have dynamically
allocated IRQs that are easy to use, simply use that to
achieve separating between the events; then there's no
interrupt for that line and we never attempt the read
in the first place, thus not closing the line.
This manifested itself in the wifi hostap/hwsim tests
where the parallel script communicates via one serial
console and the kernel messages go to another (a file)
and sending data on the communication console caused
the kernel messages to stop flowing into the file.
Reported-by: Jouni Malinen <j@w1.fi>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Acked-By: anton ivanov <anton.ivanov@cambridgegreys.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
2022-05-06 13:46:12 +00:00
|
|
|
um_free_irq(chan->line->write_irq, chan);
|
2019-05-06 12:39:35 +00:00
|
|
|
chan->enabled = 0;
|
|
|
|
}
|
2023-10-18 12:36:43 +00:00
|
|
|
if (chan->fd_out != chan->fd_in)
|
|
|
|
os_close_file(chan->fd_out);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->ops->close != NULL)
|
2023-10-18 12:36:43 +00:00
|
|
|
(*chan->ops->close)(chan->fd_in, chan->data);
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
|
|
|
|
chan->opened = 0;
|
2023-10-18 12:36:43 +00:00
|
|
|
chan->fd_in = -1;
|
|
|
|
chan->fd_out = -1;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2011-09-10 12:39:18 +00:00
|
|
|
void close_chan(struct line *line)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
struct chan *chan;
|
|
|
|
|
|
|
|
/* Close in reverse order as open in case more than one of them
|
|
|
|
* refers to the same device and they save and restore that device's
|
|
|
|
* state. Then, the first one opened will have the original state,
|
|
|
|
* so it must be the last closed.
|
|
|
|
*/
|
2011-09-10 12:39:18 +00:00
|
|
|
list_for_each_entry_reverse(chan, &line->chan_list, list) {
|
|
|
|
close_one_chan(chan, 0);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
void deactivate_chan(struct chan *chan, int irq)
|
2006-01-06 08:18:58 +00:00
|
|
|
{
|
2011-09-08 14:49:34 +00:00
|
|
|
if (chan && chan->enabled)
|
2023-10-18 12:36:43 +00:00
|
|
|
deactivate_fd(chan->fd_in, irq);
|
2006-01-06 08:18:58 +00:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:37:11 +00:00
|
|
|
int write_chan(struct chan *chan, const u8 *buf, size_t len, int write_irq)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int n, ret = 0;
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
if (len == 0 || !chan || !chan->ops->write)
|
2007-10-16 08:26:42 +00:00
|
|
|
return 0;
|
|
|
|
|
2023-10-18 12:36:43 +00:00
|
|
|
n = chan->ops->write(chan->fd_out, buf, len, chan->data);
|
2011-09-08 14:49:34 +00:00
|
|
|
if (chan->primary) {
|
|
|
|
ret = n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2006-01-06 08:18:50 +00:00
|
|
|
return ret;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
int console_write_chan(struct chan *chan, const char *buf, int len)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int n, ret = 0;
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
if (!chan || !chan->ops->console_write)
|
|
|
|
return 0;
|
2007-10-16 08:26:41 +00:00
|
|
|
|
2023-10-18 12:36:43 +00:00
|
|
|
n = chan->ops->console_write(chan->fd_out, buf, len);
|
2011-09-08 14:49:34 +00:00
|
|
|
if (chan->primary)
|
|
|
|
ret = n;
|
2006-01-06 08:18:50 +00:00
|
|
|
return ret;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2007-02-10 09:44:06 +00:00
|
|
|
int console_open_chan(struct line *line, struct console *co)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2006-01-06 08:18:55 +00:00
|
|
|
int err;
|
|
|
|
|
|
|
|
err = open_chan(&line->chan_list);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (err)
|
2006-01-06 08:18:55 +00:00
|
|
|
return err;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
printk(KERN_INFO "Console initialized on /dev/%s%d\n", co->name,
|
|
|
|
co->index);
|
2005-04-16 22:20:36 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
int chan_window_size(struct line *line, unsigned short *rows_out,
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned short *cols_out)
|
|
|
|
{
|
|
|
|
struct chan *chan;
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
chan = line->chan_in;
|
|
|
|
if (chan && chan->primary) {
|
|
|
|
if (chan->ops->window_size == NULL)
|
|
|
|
return 0;
|
2023-10-18 12:36:43 +00:00
|
|
|
return chan->ops->window_size(chan->fd_in, chan->data,
|
2011-09-08 14:49:34 +00:00
|
|
|
rows_out, cols_out);
|
|
|
|
}
|
|
|
|
chan = line->chan_out;
|
|
|
|
if (chan && chan->primary) {
|
|
|
|
if (chan->ops->window_size == NULL)
|
|
|
|
return 0;
|
2023-10-18 12:36:43 +00:00
|
|
|
return chan->ops->window_size(chan->fd_in, chan->data,
|
2011-09-08 14:49:34 +00:00
|
|
|
rows_out, cols_out);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2006-01-06 08:18:50 +00:00
|
|
|
return 0;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2011-08-18 19:12:39 +00:00
|
|
|
static void free_one_chan(struct chan *chan)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
list_del(&chan->list);
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
|
2011-08-18 19:12:39 +00:00
|
|
|
close_one_chan(chan, 0);
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->ops->free != NULL)
|
2005-04-16 22:20:36 +00:00
|
|
|
(*chan->ops->free)(chan->data);
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->primary && chan->output)
|
2023-10-18 12:36:43 +00:00
|
|
|
ignore_sigio_fd(chan->fd_in);
|
2005-04-16 22:20:36 +00:00
|
|
|
kfree(chan);
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:12:39 +00:00
|
|
|
static void free_chan(struct list_head *chans)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
struct list_head *ele, *next;
|
|
|
|
struct chan *chan;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
list_for_each_safe(ele, next, chans) {
|
2005-04-16 22:20:36 +00:00
|
|
|
chan = list_entry(ele, struct chan, list);
|
2011-08-18 19:12:39 +00:00
|
|
|
free_one_chan(chan);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int one_chan_config_string(struct chan *chan, char *str, int size,
|
|
|
|
char **error_out)
|
|
|
|
{
|
|
|
|
int n = 0;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan == NULL) {
|
2005-04-16 22:20:36 +00:00
|
|
|
CONFIG_CHUNK(str, size, n, "none", 1);
|
2006-01-06 08:18:50 +00:00
|
|
|
return n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CONFIG_CHUNK(str, size, n, chan->ops->type, 0);
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan->dev == NULL) {
|
2005-04-16 22:20:36 +00:00
|
|
|
CONFIG_CHUNK(str, size, n, "", 1);
|
2006-01-06 08:18:50 +00:00
|
|
|
return n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CONFIG_CHUNK(str, size, n, ":", 0);
|
|
|
|
CONFIG_CHUNK(str, size, n, chan->dev, 0);
|
|
|
|
|
2006-01-06 08:18:50 +00:00
|
|
|
return n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2006-01-06 08:18:50 +00:00
|
|
|
static int chan_pair_config_string(struct chan *in, struct chan *out,
|
2005-04-16 22:20:36 +00:00
|
|
|
char *str, int size, char **error_out)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
n = one_chan_config_string(in, str, size, error_out);
|
|
|
|
str += n;
|
|
|
|
size -= n;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (in == out) {
|
2005-04-16 22:20:36 +00:00
|
|
|
CONFIG_CHUNK(str, size, n, "", 1);
|
2006-01-06 08:18:50 +00:00
|
|
|
return n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CONFIG_CHUNK(str, size, n, ",", 1);
|
|
|
|
n = one_chan_config_string(out, str, size, error_out);
|
|
|
|
str += n;
|
|
|
|
size -= n;
|
|
|
|
CONFIG_CHUNK(str, size, n, "", 1);
|
|
|
|
|
2006-01-06 08:18:50 +00:00
|
|
|
return n;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
int chan_config_string(struct line *line, char *str, int size,
|
2005-04-16 22:20:36 +00:00
|
|
|
char **error_out)
|
|
|
|
{
|
2011-09-08 14:49:34 +00:00
|
|
|
struct chan *in = line->chan_in, *out = line->chan_out;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
if (in && !in->primary)
|
|
|
|
in = NULL;
|
|
|
|
if (out && !out->primary)
|
|
|
|
out = NULL;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2006-01-06 08:18:50 +00:00
|
|
|
return chan_pair_config_string(in, out, str, size, error_out);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct chan_type {
|
|
|
|
char *key;
|
2006-09-27 08:50:33 +00:00
|
|
|
const struct chan_ops *ops;
|
2005-04-16 22:20:36 +00:00
|
|
|
};
|
|
|
|
|
2006-09-27 08:50:33 +00:00
|
|
|
static const struct chan_type chan_table[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
{ "fd", &fd_ops },
|
|
|
|
|
|
|
|
#ifdef CONFIG_NULL_CHAN
|
|
|
|
{ "null", &null_ops },
|
|
|
|
#else
|
|
|
|
{ "null", ¬_configged_ops },
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_PORT_CHAN
|
|
|
|
{ "port", &port_ops },
|
|
|
|
#else
|
|
|
|
{ "port", ¬_configged_ops },
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_PTY_CHAN
|
|
|
|
{ "pty", &pty_ops },
|
|
|
|
{ "pts", &pts_ops },
|
|
|
|
#else
|
|
|
|
{ "pty", ¬_configged_ops },
|
|
|
|
{ "pts", ¬_configged_ops },
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_TTY_CHAN
|
|
|
|
{ "tty", &tty_ops },
|
|
|
|
#else
|
|
|
|
{ "tty", ¬_configged_ops },
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_XTERM_CHAN
|
|
|
|
{ "xterm", &xterm_ops },
|
|
|
|
#else
|
|
|
|
{ "xterm", ¬_configged_ops },
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
static struct chan *parse_chan(struct line *line, char *str, int device,
|
2007-02-10 09:43:53 +00:00
|
|
|
const struct chan_opts *opts, char **error_out)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2006-09-27 08:50:33 +00:00
|
|
|
const struct chan_type *entry;
|
|
|
|
const struct chan_ops *ops;
|
2005-04-16 22:20:36 +00:00
|
|
|
struct chan *chan;
|
|
|
|
void *data;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
ops = NULL;
|
|
|
|
data = NULL;
|
2007-10-16 08:26:41 +00:00
|
|
|
for(i = 0; i < ARRAY_SIZE(chan_table); i++) {
|
2005-04-16 22:20:36 +00:00
|
|
|
entry = &chan_table[i];
|
2007-10-16 08:26:41 +00:00
|
|
|
if (!strncmp(str, entry->key, strlen(entry->key))) {
|
2005-04-16 22:20:36 +00:00
|
|
|
ops = entry->ops;
|
|
|
|
str += strlen(entry->key);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-10-16 08:26:41 +00:00
|
|
|
if (ops == NULL) {
|
2007-02-10 09:43:53 +00:00
|
|
|
*error_out = "No match for configured backends";
|
2006-01-06 08:18:50 +00:00
|
|
|
return NULL;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2007-02-10 09:43:53 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
data = (*ops->init)(str, device, opts);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (data == NULL) {
|
2007-02-10 09:43:53 +00:00
|
|
|
*error_out = "Configuration failed";
|
2006-01-06 08:18:50 +00:00
|
|
|
return NULL;
|
2007-02-10 09:43:53 +00:00
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-09-23 04:44:21 +00:00
|
|
|
chan = kmalloc(sizeof(*chan), GFP_ATOMIC);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (chan == NULL) {
|
2007-02-10 09:43:53 +00:00
|
|
|
*error_out = "Memory allocation failed";
|
2006-01-06 08:18:50 +00:00
|
|
|
return NULL;
|
2007-02-10 09:43:53 +00:00
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
*chan = ((struct chan) { .list = LIST_HEAD_INIT(chan->list),
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
.free_list =
|
|
|
|
LIST_HEAD_INIT(chan->free_list),
|
|
|
|
.line = line,
|
2005-04-16 22:20:36 +00:00
|
|
|
.primary = 1,
|
|
|
|
.input = 0,
|
|
|
|
.output = 0,
|
|
|
|
.opened = 0,
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
.enabled = 0,
|
2023-10-18 12:36:43 +00:00
|
|
|
.fd_in = -1,
|
|
|
|
.fd_out = -1,
|
2005-04-16 22:20:36 +00:00
|
|
|
.ops = ops,
|
|
|
|
.data = data });
|
2006-01-06 08:18:50 +00:00
|
|
|
return chan;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
int parse_chan_pair(char *str, struct line *line, int device,
|
2007-02-10 09:43:53 +00:00
|
|
|
const struct chan_opts *opts, char **error_out)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
[PATCH] uml: Simplify console opening/closing and irq registration
This patch simplifies the opening and closing of host console devices and the
registration and deregistration of IRQs. The intent is to make it obvious
that an IRQ can't exist without an open file descriptor.
chan_enable will now open the channel, and when both opening and IRQ
registration are desired, this should be used. Opening only is done for the
initial console, so that interface still needs to exist.
The free_irqs_later interface is now gone. It was intended to avoid freeing
an IRQ while it was being processed. It did this, but it didn't eliminate the
possiblity of free_irq being called from an interrupt, which is bad. In its
place is a list of irqs to be freed, which is processed by the signal handler
just before exiting. close_one_chan now disables irqs.
When a host device disappears, it is just closed, and that disables IRQs.
The device id registered with the IRQ is now the chan structure, not the tty.
This is because the interrupt arrives on a descriptor associated with the
channel. This caused equivalent changes in the arguments to line_timer_cb.
line_disable is gone since it is not used any more.
The count field in the line structure is gone. tty->count is used instead.
The complicated logic in sigio_handler with freeing IRQs when necessary and
making sure its idea of the next irq is correct is now much simpler. The irq
list can't be rearranged underneath it, so it is now a simple list walk.
Signed-off-by: Jeff Dike <jdike@addtoit.com>
Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2006-01-06 08:18:57 +00:00
|
|
|
struct list_head *chans = &line->chan_list;
|
2011-07-26 00:12:55 +00:00
|
|
|
struct chan *new;
|
2005-04-16 22:20:36 +00:00
|
|
|
char *in, *out;
|
|
|
|
|
2007-10-16 08:26:41 +00:00
|
|
|
if (!list_empty(chans)) {
|
2011-09-08 11:07:26 +00:00
|
|
|
line->chan_in = line->chan_out = NULL;
|
2011-08-18 19:12:39 +00:00
|
|
|
free_chan(chans);
|
2005-04-16 22:20:36 +00:00
|
|
|
INIT_LIST_HEAD(chans);
|
|
|
|
}
|
|
|
|
|
2011-09-09 23:14:02 +00:00
|
|
|
if (!str)
|
|
|
|
return 0;
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
out = strchr(str, ',');
|
2007-10-16 08:26:41 +00:00
|
|
|
if (out != NULL) {
|
2005-04-16 22:20:36 +00:00
|
|
|
in = str;
|
|
|
|
*out = '\0';
|
|
|
|
out++;
|
2007-02-10 09:43:53 +00:00
|
|
|
new = parse_chan(line, in, device, opts, error_out);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (new == NULL)
|
2006-01-06 08:18:50 +00:00
|
|
|
return -1;
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
new->input = 1;
|
|
|
|
list_add(&new->list, chans);
|
2011-09-08 11:07:26 +00:00
|
|
|
line->chan_in = new;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2007-02-10 09:43:53 +00:00
|
|
|
new = parse_chan(line, out, device, opts, error_out);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (new == NULL)
|
2006-01-06 08:18:50 +00:00
|
|
|
return -1;
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
list_add(&new->list, chans);
|
|
|
|
new->output = 1;
|
2011-09-08 11:07:26 +00:00
|
|
|
line->chan_out = new;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
else {
|
2007-02-10 09:43:53 +00:00
|
|
|
new = parse_chan(line, str, device, opts, error_out);
|
2007-10-16 08:26:41 +00:00
|
|
|
if (new == NULL)
|
2006-01-06 08:18:50 +00:00
|
|
|
return -1;
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
list_add(&new->list, chans);
|
|
|
|
new->input = 1;
|
|
|
|
new->output = 1;
|
2011-09-08 11:07:26 +00:00
|
|
|
line->chan_in = line->chan_out = new;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2006-01-06 08:18:50 +00:00
|
|
|
return 0;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2013-01-03 14:53:06 +00:00
|
|
|
void chan_interrupt(struct line *line, int irq)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2013-01-03 14:53:01 +00:00
|
|
|
struct tty_port *port = &line->port;
|
2011-09-08 14:49:34 +00:00
|
|
|
struct chan *chan = line->chan_in;
|
2005-04-16 22:20:36 +00:00
|
|
|
int err;
|
2023-12-06 07:37:11 +00:00
|
|
|
u8 c;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2011-09-08 14:49:34 +00:00
|
|
|
if (!chan || !chan->ops->read)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
do {
|
2013-01-03 14:53:01 +00:00
|
|
|
if (!tty_buffer_request_room(port, 1)) {
|
2011-09-10 12:17:04 +00:00
|
|
|
schedule_delayed_work(&line->task, 1);
|
2011-09-08 14:49:34 +00:00
|
|
|
goto out;
|
|
|
|
}
|
2023-10-18 12:36:43 +00:00
|
|
|
err = chan->ops->read(chan->fd_in, &c, chan->data);
|
2011-09-08 14:49:34 +00:00
|
|
|
if (err > 0)
|
2013-01-03 14:53:03 +00:00
|
|
|
tty_insert_flip_char(port, c, TTY_NORMAL);
|
2011-09-08 14:49:34 +00:00
|
|
|
} while (err > 0);
|
|
|
|
|
|
|
|
if (err == -EIO) {
|
|
|
|
if (chan->primary) {
|
2013-03-07 12:12:30 +00:00
|
|
|
tty_port_tty_hangup(&line->port, false);
|
2011-09-10 12:39:18 +00:00
|
|
|
if (line->chan_out != chan)
|
|
|
|
close_one_chan(line->chan_out, 1);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
2011-09-10 12:39:18 +00:00
|
|
|
close_one_chan(chan, 1);
|
|
|
|
if (chan->primary)
|
|
|
|
return;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
out:
|
2013-01-03 14:53:06 +00:00
|
|
|
tty_flip_buffer_push(port);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|