mirror of https://github.com/xqemu/xqemu.git
char: move front end handlers in CharBackend
Since the hanlders are associated with a CharBackend, rather than the CharDriverState, it is more appropriate to store in CharBackend. This avoids the handler copy dance in qemu_chr_fe_set_handlers() then mux_chr_update_read_handler(), by storing the CharBackend pointer directly. Also a mux CharDriver should go through mux->backends[focused], since chr->be will stay NULL. Before that, it was possible to call chr->handler by mistake with surprising results, for ex through qemu_chr_be_can_write(), which would result in calling the last set handler front end, not the one with focus. Signed-off-by: Marc-André Lureau <marcandre.lureau@redhat.com> Message-Id: <20161022095318.17775-22-marcandre.lureau@redhat.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
ea3af47d75
commit
a4afa548fc
|
@ -78,15 +78,17 @@ enum {
|
||||||
|
|
||||||
static inline void csrhci_fifo_wake(struct csrhci_s *s)
|
static inline void csrhci_fifo_wake(struct csrhci_s *s)
|
||||||
{
|
{
|
||||||
|
CharBackend *be = s->chr.be;
|
||||||
|
|
||||||
if (!s->enable || !s->out_len)
|
if (!s->enable || !s->out_len)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* XXX: Should wait for s->modem_state & CHR_TIOCM_RTS? */
|
/* XXX: Should wait for s->modem_state & CHR_TIOCM_RTS? */
|
||||||
if (s->chr.chr_can_read && s->chr.chr_can_read(s->chr.handler_opaque) &&
|
if (be && be->chr_can_read && be->chr_can_read(be->opaque) &&
|
||||||
s->chr.chr_read) {
|
be->chr_read) {
|
||||||
s->chr.chr_read(s->chr.handler_opaque,
|
be->chr_read(be->opaque,
|
||||||
s->outfifo + s->out_start ++, 1);
|
s->outfifo + s->out_start++, 1);
|
||||||
s->out_len --;
|
s->out_len--;
|
||||||
if (s->out_start >= s->out_size) {
|
if (s->out_start >= s->out_size) {
|
||||||
s->out_start = 0;
|
s->out_start = 0;
|
||||||
s->out_size = FIFO_LEN;
|
s->out_size = FIFO_LEN;
|
||||||
|
|
|
@ -76,6 +76,10 @@ typedef enum {
|
||||||
* CharDriverState */
|
* CharDriverState */
|
||||||
typedef struct CharBackend {
|
typedef struct CharBackend {
|
||||||
CharDriverState *chr;
|
CharDriverState *chr;
|
||||||
|
IOEventHandler *chr_event;
|
||||||
|
IOCanReadHandler *chr_can_read;
|
||||||
|
IOReadHandler *chr_read;
|
||||||
|
void *opaque;
|
||||||
int tag;
|
int tag;
|
||||||
} CharBackend;
|
} CharBackend;
|
||||||
|
|
||||||
|
@ -86,22 +90,19 @@ struct CharDriverState {
|
||||||
const uint8_t *buf, int len);
|
const uint8_t *buf, int len);
|
||||||
GSource *(*chr_add_watch)(struct CharDriverState *s, GIOCondition cond);
|
GSource *(*chr_add_watch)(struct CharDriverState *s, GIOCondition cond);
|
||||||
void (*chr_update_read_handler)(struct CharDriverState *s,
|
void (*chr_update_read_handler)(struct CharDriverState *s,
|
||||||
GMainContext *context, int tag);
|
GMainContext *context);
|
||||||
int (*chr_ioctl)(struct CharDriverState *s, int cmd, void *arg);
|
int (*chr_ioctl)(struct CharDriverState *s, int cmd, void *arg);
|
||||||
int (*get_msgfds)(struct CharDriverState *s, int* fds, int num);
|
int (*get_msgfds)(struct CharDriverState *s, int* fds, int num);
|
||||||
int (*set_msgfds)(struct CharDriverState *s, int *fds, int num);
|
int (*set_msgfds)(struct CharDriverState *s, int *fds, int num);
|
||||||
int (*chr_add_client)(struct CharDriverState *chr, int fd);
|
int (*chr_add_client)(struct CharDriverState *chr, int fd);
|
||||||
int (*chr_wait_connected)(struct CharDriverState *chr, Error **errp);
|
int (*chr_wait_connected)(struct CharDriverState *chr, Error **errp);
|
||||||
IOEventHandler *chr_event;
|
|
||||||
IOCanReadHandler *chr_can_read;
|
|
||||||
IOReadHandler *chr_read;
|
|
||||||
void *handler_opaque;
|
|
||||||
void (*chr_close)(struct CharDriverState *chr);
|
void (*chr_close)(struct CharDriverState *chr);
|
||||||
void (*chr_disconnect)(struct CharDriverState *chr);
|
void (*chr_disconnect)(struct CharDriverState *chr);
|
||||||
void (*chr_accept_input)(struct CharDriverState *chr);
|
void (*chr_accept_input)(struct CharDriverState *chr);
|
||||||
void (*chr_set_echo)(struct CharDriverState *chr, bool echo);
|
void (*chr_set_echo)(struct CharDriverState *chr, bool echo);
|
||||||
void (*chr_set_fe_open)(struct CharDriverState *chr, int fe_open);
|
void (*chr_set_fe_open)(struct CharDriverState *chr, int fe_open);
|
||||||
void (*chr_fe_event)(struct CharDriverState *chr, int event);
|
void (*chr_fe_event)(struct CharDriverState *chr, int event);
|
||||||
|
CharBackend *be;
|
||||||
void *opaque;
|
void *opaque;
|
||||||
char *label;
|
char *label;
|
||||||
char *filename;
|
char *filename;
|
||||||
|
|
141
qemu-char.c
141
qemu-char.c
|
@ -192,6 +192,8 @@ CharDriverState *qemu_chr_alloc(ChardevCommon *backend, Error **errp)
|
||||||
|
|
||||||
void qemu_chr_be_event(CharDriverState *s, int event)
|
void qemu_chr_be_event(CharDriverState *s, int event)
|
||||||
{
|
{
|
||||||
|
CharBackend *be = s->be;
|
||||||
|
|
||||||
/* Keep track if the char device is open */
|
/* Keep track if the char device is open */
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case CHR_EVENT_OPENED:
|
case CHR_EVENT_OPENED:
|
||||||
|
@ -202,9 +204,11 @@ void qemu_chr_be_event(CharDriverState *s, int event)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!s->chr_event)
|
if (!be || !be->chr_event) {
|
||||||
return;
|
return;
|
||||||
s->chr_event(s->handler_opaque, event);
|
}
|
||||||
|
|
||||||
|
be->chr_event(be->opaque, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
void qemu_chr_be_generic_open(CharDriverState *s)
|
void qemu_chr_be_generic_open(CharDriverState *s)
|
||||||
|
@ -398,15 +402,21 @@ int qemu_chr_fe_ioctl(CharBackend *be, int cmd, void *arg)
|
||||||
|
|
||||||
int qemu_chr_be_can_write(CharDriverState *s)
|
int qemu_chr_be_can_write(CharDriverState *s)
|
||||||
{
|
{
|
||||||
if (!s->chr_can_read)
|
CharBackend *be = s->be;
|
||||||
|
|
||||||
|
if (!be || !be->chr_can_read) {
|
||||||
return 0;
|
return 0;
|
||||||
return s->chr_can_read(s->handler_opaque);
|
}
|
||||||
|
|
||||||
|
return be->chr_can_read(be->opaque);
|
||||||
}
|
}
|
||||||
|
|
||||||
void qemu_chr_be_write_impl(CharDriverState *s, uint8_t *buf, int len)
|
void qemu_chr_be_write_impl(CharDriverState *s, uint8_t *buf, int len)
|
||||||
{
|
{
|
||||||
if (s->chr_read) {
|
CharBackend *be = s->be;
|
||||||
s->chr_read(s->handler_opaque, buf, len);
|
|
||||||
|
if (be && be->chr_read) {
|
||||||
|
be->chr_read(be->opaque, buf, len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -488,7 +498,6 @@ void qemu_chr_fe_printf(CharBackend *be, const char *fmt, ...)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void remove_fd_in_watch(CharDriverState *chr);
|
static void remove_fd_in_watch(CharDriverState *chr);
|
||||||
static int mux_chr_new_handler_tag(CharDriverState *chr, Error **errp);
|
|
||||||
static void mux_chr_set_handlers(CharDriverState *chr, GMainContext *context);
|
static void mux_chr_set_handlers(CharDriverState *chr, GMainContext *context);
|
||||||
static void mux_set_focus(MuxDriver *d, int focus);
|
static void mux_set_focus(MuxDriver *d, int focus);
|
||||||
|
|
||||||
|
@ -519,10 +528,7 @@ static CharDriverState *qemu_chr_open_null(const char *id,
|
||||||
#define MUX_BUFFER_SIZE 32 /* Must be a power of 2. */
|
#define MUX_BUFFER_SIZE 32 /* Must be a power of 2. */
|
||||||
#define MUX_BUFFER_MASK (MUX_BUFFER_SIZE - 1)
|
#define MUX_BUFFER_MASK (MUX_BUFFER_SIZE - 1)
|
||||||
struct MuxDriver {
|
struct MuxDriver {
|
||||||
IOCanReadHandler *chr_can_read[MAX_MUX];
|
CharBackend *backends[MAX_MUX];
|
||||||
IOReadHandler *chr_read[MAX_MUX];
|
|
||||||
IOEventHandler *chr_event[MAX_MUX];
|
|
||||||
void *ext_opaque[MAX_MUX];
|
|
||||||
CharBackend chr;
|
CharBackend chr;
|
||||||
int focus;
|
int focus;
|
||||||
int mux_cnt;
|
int mux_cnt;
|
||||||
|
@ -625,8 +631,11 @@ static void mux_print_help(CharDriverState *chr)
|
||||||
|
|
||||||
static void mux_chr_send_event(MuxDriver *d, int mux_nr, int event)
|
static void mux_chr_send_event(MuxDriver *d, int mux_nr, int event)
|
||||||
{
|
{
|
||||||
if (d->chr_event[mux_nr])
|
CharBackend *be = d->backends[mux_nr];
|
||||||
d->chr_event[mux_nr](d->ext_opaque[mux_nr], event);
|
|
||||||
|
if (be && be->chr_event) {
|
||||||
|
be->chr_event(be->opaque, event);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mux_proc_byte(CharDriverState *chr, MuxDriver *d, int ch)
|
static int mux_proc_byte(CharDriverState *chr, MuxDriver *d, int ch)
|
||||||
|
@ -677,12 +686,12 @@ static void mux_chr_accept_input(CharDriverState *chr)
|
||||||
{
|
{
|
||||||
MuxDriver *d = chr->opaque;
|
MuxDriver *d = chr->opaque;
|
||||||
int m = d->focus;
|
int m = d->focus;
|
||||||
|
CharBackend *be = d->backends[m];
|
||||||
|
|
||||||
while (d->prod[m] != d->cons[m] &&
|
while (be && d->prod[m] != d->cons[m] &&
|
||||||
d->chr_can_read[m] &&
|
be->chr_can_read && be->chr_can_read(be->opaque)) {
|
||||||
d->chr_can_read[m](d->ext_opaque[m])) {
|
be->chr_read(be->opaque,
|
||||||
d->chr_read[m](d->ext_opaque[m],
|
&d->buffer[m][d->cons[m]++ & MUX_BUFFER_MASK], 1);
|
||||||
&d->buffer[m][d->cons[m]++ & MUX_BUFFER_MASK], 1);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -691,11 +700,16 @@ static int mux_chr_can_read(void *opaque)
|
||||||
CharDriverState *chr = opaque;
|
CharDriverState *chr = opaque;
|
||||||
MuxDriver *d = chr->opaque;
|
MuxDriver *d = chr->opaque;
|
||||||
int m = d->focus;
|
int m = d->focus;
|
||||||
|
CharBackend *be = d->backends[m];
|
||||||
|
|
||||||
if ((d->prod[m] - d->cons[m]) < MUX_BUFFER_SIZE)
|
if ((d->prod[m] - d->cons[m]) < MUX_BUFFER_SIZE) {
|
||||||
return 1;
|
return 1;
|
||||||
if (d->chr_can_read[m])
|
}
|
||||||
return d->chr_can_read[m](d->ext_opaque[m]);
|
|
||||||
|
if (be && be->chr_can_read) {
|
||||||
|
return be->chr_can_read(be->opaque);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -704,16 +718,17 @@ static void mux_chr_read(void *opaque, const uint8_t *buf, int size)
|
||||||
CharDriverState *chr = opaque;
|
CharDriverState *chr = opaque;
|
||||||
MuxDriver *d = chr->opaque;
|
MuxDriver *d = chr->opaque;
|
||||||
int m = d->focus;
|
int m = d->focus;
|
||||||
|
CharBackend *be = d->backends[m];
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
mux_chr_accept_input (opaque);
|
mux_chr_accept_input(opaque);
|
||||||
|
|
||||||
for(i = 0; i < size; i++)
|
for (i = 0; i < size; i++)
|
||||||
if (mux_proc_byte(chr, d, buf[i])) {
|
if (mux_proc_byte(chr, d, buf[i])) {
|
||||||
if (d->prod[m] == d->cons[m] &&
|
if (d->prod[m] == d->cons[m] &&
|
||||||
d->chr_can_read[m] &&
|
be && be->chr_can_read &&
|
||||||
d->chr_can_read[m](d->ext_opaque[m]))
|
be->chr_can_read(be->opaque))
|
||||||
d->chr_read[m](d->ext_opaque[m], &buf[i], 1);
|
be->chr_read(be->opaque, &buf[i], 1);
|
||||||
else
|
else
|
||||||
d->buffer[m][d->prod[m]++ & MUX_BUFFER_MASK] = buf[i];
|
d->buffer[m][d->prod[m]++ & MUX_BUFFER_MASK] = buf[i];
|
||||||
}
|
}
|
||||||
|
@ -730,21 +745,6 @@ static void mux_chr_event(void *opaque, int event)
|
||||||
mux_chr_send_event(d, i, event);
|
mux_chr_send_event(d, i, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mux_chr_update_read_handler(CharDriverState *chr,
|
|
||||||
GMainContext *context,
|
|
||||||
int tag)
|
|
||||||
{
|
|
||||||
MuxDriver *d = chr->opaque;
|
|
||||||
|
|
||||||
assert(tag >= 0);
|
|
||||||
assert(tag < d->mux_cnt);
|
|
||||||
|
|
||||||
d->ext_opaque[tag] = chr->handler_opaque;
|
|
||||||
d->chr_can_read[tag] = chr->chr_can_read;
|
|
||||||
d->chr_read[tag] = chr->chr_read;
|
|
||||||
d->chr_event[tag] = chr->chr_event;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool muxes_realized;
|
static bool muxes_realized;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -796,12 +796,19 @@ static GSource *mux_chr_add_watch(CharDriverState *s, GIOCondition cond)
|
||||||
static void mux_chr_close(struct CharDriverState *chr)
|
static void mux_chr_close(struct CharDriverState *chr)
|
||||||
{
|
{
|
||||||
MuxDriver *d = chr->opaque;
|
MuxDriver *d = chr->opaque;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < d->mux_cnt; i++) {
|
||||||
|
CharBackend *be = d->backends[i];
|
||||||
|
if (be) {
|
||||||
|
be->chr = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
qemu_chr_fe_deinit(&d->chr);
|
qemu_chr_fe_deinit(&d->chr);
|
||||||
g_free(d);
|
g_free(d);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mux_chr_new_handler_tag(CharDriverState *chr, Error **errp)
|
static int mux_chr_new_fe(CharDriverState *chr, CharBackend *be, Error **errp)
|
||||||
{
|
{
|
||||||
MuxDriver *d = chr->opaque;
|
MuxDriver *d = chr->opaque;
|
||||||
|
|
||||||
|
@ -810,6 +817,8 @@ static int mux_chr_new_handler_tag(CharDriverState *chr, Error **errp)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
d->backends[d->mux_cnt] = be;
|
||||||
|
|
||||||
return d->mux_cnt++;
|
return d->mux_cnt++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -864,7 +873,6 @@ static CharDriverState *qemu_chr_open_mux(const char *id,
|
||||||
d->focus = -1;
|
d->focus = -1;
|
||||||
chr->chr_close = mux_chr_close;
|
chr->chr_close = mux_chr_close;
|
||||||
chr->chr_write = mux_chr_write;
|
chr->chr_write = mux_chr_write;
|
||||||
chr->chr_update_read_handler = mux_chr_update_read_handler;
|
|
||||||
chr->chr_accept_input = mux_chr_accept_input;
|
chr->chr_accept_input = mux_chr_accept_input;
|
||||||
/* Frontend guest-open / -close notification is not support with muxes */
|
/* Frontend guest-open / -close notification is not support with muxes */
|
||||||
chr->chr_set_fe_open = NULL;
|
chr->chr_set_fe_open = NULL;
|
||||||
|
@ -894,10 +902,12 @@ bool qemu_chr_fe_init(CharBackend *b, CharDriverState *s, Error **errp)
|
||||||
int tag = 0;
|
int tag = 0;
|
||||||
|
|
||||||
if (s->is_mux) {
|
if (s->is_mux) {
|
||||||
tag = mux_chr_new_handler_tag(s, errp);
|
tag = mux_chr_new_fe(s, b, errp);
|
||||||
if (tag < 0) {
|
if (tag < 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
s->be = b;
|
||||||
}
|
}
|
||||||
|
|
||||||
b->tag = tag;
|
b->tag = tag;
|
||||||
|
@ -906,6 +916,16 @@ bool qemu_chr_fe_init(CharBackend *b, CharDriverState *s, Error **errp)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool qemu_chr_is_busy(CharDriverState *s)
|
||||||
|
{
|
||||||
|
if (s->is_mux) {
|
||||||
|
MuxDriver *d = s->opaque;
|
||||||
|
return d->mux_cnt >= 0;
|
||||||
|
} else {
|
||||||
|
return s->be != NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void qemu_chr_fe_deinit(CharBackend *b)
|
void qemu_chr_fe_deinit(CharBackend *b)
|
||||||
{
|
{
|
||||||
assert(b);
|
assert(b);
|
||||||
|
@ -913,6 +933,11 @@ void qemu_chr_fe_deinit(CharBackend *b)
|
||||||
if (b->chr) {
|
if (b->chr) {
|
||||||
qemu_chr_fe_set_handlers(b, NULL, NULL, NULL, NULL, NULL);
|
qemu_chr_fe_set_handlers(b, NULL, NULL, NULL, NULL, NULL);
|
||||||
b->chr->avail_connections++;
|
b->chr->avail_connections++;
|
||||||
|
b->chr->be = NULL;
|
||||||
|
if (b->chr->is_mux) {
|
||||||
|
MuxDriver *d = b->chr->opaque;
|
||||||
|
d->backends[b->tag] = NULL;
|
||||||
|
}
|
||||||
b->chr = NULL;
|
b->chr = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -938,12 +963,12 @@ void qemu_chr_fe_set_handlers(CharBackend *b,
|
||||||
} else {
|
} else {
|
||||||
fe_open = 1;
|
fe_open = 1;
|
||||||
}
|
}
|
||||||
s->chr_can_read = fd_can_read;
|
b->chr_can_read = fd_can_read;
|
||||||
s->chr_read = fd_read;
|
b->chr_read = fd_read;
|
||||||
s->chr_event = fd_event;
|
b->chr_event = fd_event;
|
||||||
s->handler_opaque = opaque;
|
b->opaque = opaque;
|
||||||
if (s->chr_update_read_handler) {
|
if (s->chr_update_read_handler) {
|
||||||
s->chr_update_read_handler(s, context, b->tag);
|
s->chr_update_read_handler(s, context);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!s->explicit_fe_open) {
|
if (!s->explicit_fe_open) {
|
||||||
|
@ -1202,8 +1227,7 @@ static GSource *fd_chr_add_watch(CharDriverState *chr, GIOCondition cond)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void fd_chr_update_read_handler(CharDriverState *chr,
|
static void fd_chr_update_read_handler(CharDriverState *chr,
|
||||||
GMainContext *context,
|
GMainContext *context)
|
||||||
int tag)
|
|
||||||
{
|
{
|
||||||
FDCharDriver *s = chr->opaque;
|
FDCharDriver *s = chr->opaque;
|
||||||
|
|
||||||
|
@ -1460,8 +1484,7 @@ static void pty_chr_update_read_handler_locked(CharDriverState *chr)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void pty_chr_update_read_handler(CharDriverState *chr,
|
static void pty_chr_update_read_handler(CharDriverState *chr,
|
||||||
GMainContext *context,
|
GMainContext *context)
|
||||||
int tag)
|
|
||||||
{
|
{
|
||||||
qemu_mutex_lock(&chr->chr_write_lock);
|
qemu_mutex_lock(&chr->chr_write_lock);
|
||||||
pty_chr_update_read_handler_locked(chr);
|
pty_chr_update_read_handler_locked(chr);
|
||||||
|
@ -2707,8 +2730,7 @@ static gboolean udp_chr_read(QIOChannel *chan, GIOCondition cond, void *opaque)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void udp_chr_update_read_handler(CharDriverState *chr,
|
static void udp_chr_update_read_handler(CharDriverState *chr,
|
||||||
GMainContext *context,
|
GMainContext *context)
|
||||||
int tag)
|
|
||||||
{
|
{
|
||||||
NetCharDriver *s = chr->opaque;
|
NetCharDriver *s = chr->opaque;
|
||||||
|
|
||||||
|
@ -3127,8 +3149,7 @@ static void tcp_chr_connect(void *opaque)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcp_chr_update_read_handler(CharDriverState *chr,
|
static void tcp_chr_update_read_handler(CharDriverState *chr,
|
||||||
GMainContext *context,
|
GMainContext *context)
|
||||||
int tag)
|
|
||||||
{
|
{
|
||||||
TCPCharDriver *s = chr->opaque;
|
TCPCharDriver *s = chr->opaque;
|
||||||
|
|
||||||
|
@ -4246,6 +4267,9 @@ void qemu_chr_fe_disconnect(CharBackend *be)
|
||||||
|
|
||||||
static void qemu_chr_free_common(CharDriverState *chr)
|
static void qemu_chr_free_common(CharDriverState *chr)
|
||||||
{
|
{
|
||||||
|
if (chr->be) {
|
||||||
|
chr->be->chr = NULL;
|
||||||
|
}
|
||||||
g_free(chr->filename);
|
g_free(chr->filename);
|
||||||
g_free(chr->label);
|
g_free(chr->label);
|
||||||
if (chr->logfd != -1) {
|
if (chr->logfd != -1) {
|
||||||
|
@ -4790,8 +4814,7 @@ void qmp_chardev_remove(const char *id, Error **errp)
|
||||||
error_setg(errp, "Chardev '%s' not found", id);
|
error_setg(errp, "Chardev '%s' not found", id);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (chr->chr_can_read || chr->chr_read ||
|
if (qemu_chr_is_busy(chr)) {
|
||||||
chr->chr_event || chr->handler_opaque) {
|
|
||||||
error_setg(errp, "Chardev '%s' is busy", id);
|
error_setg(errp, "Chardev '%s' is busy", id);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1083,6 +1083,7 @@ static void kbd_send_chars(void *opaque)
|
||||||
void kbd_put_keysym_console(QemuConsole *s, int keysym)
|
void kbd_put_keysym_console(QemuConsole *s, int keysym)
|
||||||
{
|
{
|
||||||
uint8_t buf[16], *q;
|
uint8_t buf[16], *q;
|
||||||
|
CharBackend *be;
|
||||||
int c;
|
int c;
|
||||||
|
|
||||||
if (!s || (s->console_type == GRAPHIC_CONSOLE))
|
if (!s || (s->console_type == GRAPHIC_CONSOLE))
|
||||||
|
@ -1125,7 +1126,8 @@ void kbd_put_keysym_console(QemuConsole *s, int keysym)
|
||||||
if (s->echo) {
|
if (s->echo) {
|
||||||
console_puts(s->chr, buf, q - buf);
|
console_puts(s->chr, buf, q - buf);
|
||||||
}
|
}
|
||||||
if (s->chr->chr_read) {
|
be = s->chr->be;
|
||||||
|
if (be && be->chr_read) {
|
||||||
qemu_fifo_write(&s->out_fifo, buf, q - buf);
|
qemu_fifo_write(&s->out_fifo, buf, q - buf);
|
||||||
kbd_send_chars(s);
|
kbd_send_chars(s);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue