[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[PATCH v6 22/25] console: make screendump asynchronous
From: |
Marc-André Lureau |
Subject: |
[PATCH v6 22/25] console: make screendump asynchronous |
Date: |
Fri, 8 Nov 2019 19:01:20 +0400 |
Make screendump asynchronous to provide correct screendumps.
For now, HMP doesn't have async support, so it has to remain
synchronous and potentially incorrect to avoid races (following
patches will add HMP asynchronous commands)
Fixes:
https://bugzilla.redhat.com/show_bug.cgi?id=1230527
Signed-off-by: Marc-André Lureau <address@hidden>
---
include/ui/console.h | 3 ++
monitor/hmp-cmds.c | 2 +-
qapi/ui.json | 3 +-
ui/console.c | 117 +++++++++++++++++++++++++++++++++++++++----
4 files changed, 113 insertions(+), 12 deletions(-)
diff --git a/include/ui/console.h b/include/ui/console.h
index 281f9c145b..a1935557cc 100644
--- a/include/ui/console.h
+++ b/include/ui/console.h
@@ -74,6 +74,9 @@ typedef struct MouseTransformInfo {
} MouseTransformInfo;
void hmp_mouse_set(Monitor *mon, const QDict *qdict);
+void hmp_screendump_sync(const char *filename,
+ bool has_device, const char *device,
+ bool has_head, int64_t head, Error **errp);
/* keysym is a unicode code except for special keys (see QEMU_KEY_xxx
constants) */
diff --git a/monitor/hmp-cmds.c b/monitor/hmp-cmds.c
index b2551c16d1..c52e78fedf 100644
--- a/monitor/hmp-cmds.c
+++ b/monitor/hmp-cmds.c
@@ -2308,7 +2308,7 @@ void hmp_screendump(Monitor *mon, const QDict *qdict)
int64_t head = qdict_get_try_int(qdict, "head", 0);
Error *err = NULL;
- qmp_screendump(filename, id != NULL, id, id != NULL, head, &err);
+ hmp_screendump_sync(filename, id != NULL, id, id != NULL, head, &err);
hmp_handle_error(mon, &err);
}
diff --git a/qapi/ui.json b/qapi/ui.json
index e04525d8b4..148ae4c062 100644
--- a/qapi/ui.json
+++ b/qapi/ui.json
@@ -96,7 +96,8 @@
#
##
{ 'command': 'screendump',
- 'data': {'filename': 'str', '*device': 'str', '*head': 'int'} }
+ 'data': {'filename': 'str', '*device': 'str', '*head': 'int'},
+ 'async': true }
##
# == Spice
diff --git a/ui/console.c b/ui/console.c
index e6ac462aa0..fd87605b7c 100644
--- a/ui/console.c
+++ b/ui/console.c
@@ -34,6 +34,7 @@
#include "trace.h"
#include "exec/memory.h"
#include "io/channel-file.h"
+#include "monitor/monitor.h"
#define DEFAULT_BACKSCROLL 512
#define CONSOLE_CURSOR_PERIOD 500
@@ -118,6 +119,13 @@ typedef enum {
TEXT_CONSOLE_FIXED_SIZE
} console_type_t;
+struct qmp_screendump {
+ int fd;
+ gchar *filename;
+ QmpReturn *ret;
+ QLIST_ENTRY(qmp_screendump) link;
+};
+
struct QemuConsole {
Object parent;
@@ -168,6 +176,8 @@ struct QemuConsole {
uint8_t out_fifo_buf[16];
QEMUTimer *kbd_timer;
+ QLIST_HEAD(, qmp_screendump) qmp_screendumps;
+
QTAILQ_ENTRY(QemuConsole) next;
};
@@ -261,8 +271,51 @@ static void gui_setup_refresh(DisplayState *ds)
ds->have_text = have_text;
}
+static void qmp_screendump_finish(QemuConsole *con, struct qmp_screendump
*dump)
+{
+ Error *err = NULL;
+ DisplaySurface *surface;
+
+ if (qmp_return_is_cancelled(dump->ret)) {
+ goto cleanup;
+ }
+
+ surface = qemu_console_surface(con);
+ if (!surface) {
+ error_setg(&err, "no surface");
+ } else {
+ /*
+ * FIXME: async save with coroutine? it would have to copy or
+ * lock the surface.
+ */
+ if (!ppm_save(dump->fd, surface, &err)) {
+ qemu_unlink(dump->filename);
+ }
+ dump->fd = -1;
+ }
+
+ if (err) {
+ qmp_return_error(dump->ret, err);
+ } else {
+ qmp_screendump_return(dump->ret);
+ }
+
+cleanup:
+ if (dump->fd != -1) {
+ qemu_close(dump->fd);
+ }
+ g_free(dump->filename);
+ QLIST_REMOVE(dump, link);
+ g_free(dump);
+}
+
void graphic_hw_update_done(QemuConsole *con)
{
+ struct qmp_screendump *dump, *next;
+
+ QLIST_FOREACH_SAFE(dump, &con->qmp_screendumps, link, next) {
+ qmp_screendump_finish(con, dump);
+ }
}
void graphic_hw_update(QemuConsole *con)
@@ -341,31 +394,42 @@ static bool ppm_save(int fd, DisplaySurface *ds, Error
**errp)
return true;
}
-void qmp_screendump(const char *filename, bool has_device, const char *device,
- bool has_head, int64_t head, Error **errp)
+
+static QemuConsole *get_console(bool has_device, const char *device,
+ bool has_head, int64_t head, Error **errp)
{
- QemuConsole *con;
- DisplaySurface *surface;
- int fd;
+ QemuConsole *con = NULL;
if (has_device) {
con = qemu_console_lookup_by_device_name(device, has_head ? head : 0,
errp);
- if (!con) {
- return;
- }
} else {
if (has_head) {
error_setg(errp, "'head' must be specified together with
'device'");
- return;
+ return NULL;
}
con = qemu_console_lookup_by_index(0);
if (!con) {
error_setg(errp, "There is no console to take a screendump from");
- return;
}
}
+ return con;
+}
+
+void hmp_screendump_sync(const char *filename,
+ bool has_device, const char *device,
+ bool has_head, int64_t head, Error **errp)
+{
+ DisplaySurface *surface;
+ QemuConsole *con = get_console(has_device, device, has_head, head, errp);
+ int fd;
+
+ if (!con) {
+ return;
+ }
+ /* This may not complete the drawing with Spice, you may have
+ * glitches or outdated dumps, use qmp instead! */
graphic_hw_update(con);
surface = qemu_console_surface(con);
if (!surface) {
@@ -385,6 +449,38 @@ void qmp_screendump(const char *filename, bool has_device,
const char *device,
}
}
+void qmp_screendump(const char *filename,
+ bool has_device, const char *device,
+ bool has_head, int64_t head,
+ QmpReturn *qret)
+{
+ Error *err = NULL;
+ struct qmp_screendump *dump = NULL;
+ QemuConsole *con = get_console(has_device, device, has_head, head, &err);
+ int fd;
+
+ if (!con) {
+ qmp_return_error(qret, err);
+ return;
+ }
+
+ fd = qemu_open(filename, O_WRONLY | O_CREAT | O_TRUNC | O_BINARY, 0666);
+ if (fd == -1) {
+ error_setg(&err, "failed to open file '%s': %s",
+ filename, strerror(errno));
+ qmp_return_error(qret, err);
+ return;
+ }
+
+ dump = g_new(struct qmp_screendump, 1);
+ dump->fd = fd;
+ dump->filename = g_strdup(filename);
+ dump->ret = qret;
+ QLIST_INSERT_HEAD(&con->qmp_screendumps, dump, link);
+
+ graphic_hw_update(con);
+}
+
void graphic_hw_text_update(QemuConsole *con, console_ch_t *chardata)
{
if (!con) {
@@ -1295,6 +1391,7 @@ static QemuConsole *new_console(DisplayState *ds,
console_type_t console_type,
obj = object_new(TYPE_QEMU_CONSOLE);
s = QEMU_CONSOLE(obj);
s->head = head;
+ QLIST_INIT(&s->qmp_screendumps);
object_property_add_link(obj, "device", TYPE_DEVICE,
(Object **)&s->device,
object_property_allow_set_link,
--
2.24.0
- [PATCH v6 12/25] qmp: introduce asynchronous command type, (continued)
- [PATCH v6 12/25] qmp: introduce asynchronous command type, Marc-André Lureau, 2019/11/08
- [PATCH v6 14/25] qmp: add qmp_return_is_cancelled(), Marc-André Lureau, 2019/11/08
- [PATCH v6 13/25] scripts: learn 'async' qapi commands, Marc-André Lureau, 2019/11/08
- [PATCH v6 15/25] console: add graphic_hw_update_done(), Marc-André Lureau, 2019/11/08
- [PATCH v6 16/25] ppm-save: pass opened fd, Marc-André Lureau, 2019/11/08
- [PATCH v6 17/25] ui: add pixman image g_autoptr support, Marc-André Lureau, 2019/11/08
- [PATCH v6 18/25] object: add g_autoptr support, Marc-André Lureau, 2019/11/08
- [PATCH v6 19/25] screendump: replace FILE with QIOChannel and fix close()/qemu_close(), Marc-André Lureau, 2019/11/08
- [PATCH v6 20/25] osdep: add qemu_unlink(), Marc-André Lureau, 2019/11/08
- [PATCH v6 21/25] screendump: use qemu_unlink(), Marc-André Lureau, 2019/11/08
- [PATCH v6 22/25] console: make screendump asynchronous,
Marc-André Lureau <=
- [PATCH v6 23/25] monitor: start making qmp_human_monitor_command() asynchronous, Marc-André Lureau, 2019/11/08
- [PATCH v6 24/25] monitor: teach HMP about asynchronous commands, Marc-André Lureau, 2019/11/08
- [PATCH v6 25/25] hmp: call the asynchronous QMP screendump to fix outdated/glitches, Marc-André Lureau, 2019/11/08