From: Anthony PERARD <anthony.perard@citrix.com>
To: xen-devel@lists.xenproject.org
Cc: Anthony PERARD <anthony.perard@citrix.com>,
Wei Liu <wei.liu2@citrix.com>,
Ian Jackson <ian.jackson@eu.citrix.com>
Subject: [PATCH v3 15/31] libxl_qmp_ev: Implement fd callback and read data
Date: Fri, 1 Jun 2018 15:37:04 +0100 [thread overview]
Message-ID: <20180601143720.24637-16-anthony.perard@citrix.com> (raw)
In-Reply-To: <20180601143720.24637-1-anthony.perard@citrix.com>
First step into taking care of the input from QEMU's QMP socket. For
now, we read data and store them in buffers.
Parsing of the data will be done in the following patches.
Signed-off-by: Anthony PERARD <anthony.perard@citrix.com>
---
tools/libxl/libxl_qmp.c | 113 ++++++++++++++++++++++++++++++++++++++++
1 file changed, 113 insertions(+)
diff --git a/tools/libxl/libxl_qmp.c b/tools/libxl/libxl_qmp.c
index 077cac9c8a..48dc376307 100644
--- a/tools/libxl/libxl_qmp.c
+++ b/tools/libxl/libxl_qmp.c
@@ -75,6 +75,12 @@
# define DEBUG_REPORT_RECEIVED(dom, buf, len) ((void)0)
#endif
+#ifdef DEBUG_QMP_CLIENT
+# define LOG_QMP(f, ...) LOGD(DEBUG, qmp->domid, f, ##__VA_ARGS__)
+#else
+# define LOG_QMP(f, ...)
+#endif
+
/*
* QMP types & constant
*/
@@ -1353,15 +1359,115 @@ int libxl__qmp_initializations(libxl__gc *gc, uint32_t domid,
/* ------------ Implementation of libxl__ev_qmp ---------------- */
+typedef struct libxl__qmp_rx_buf libxl__qmp_rx_buf;
+struct libxl__qmp_rx_buf {
+ LIBXL_TAILQ_ENTRY(libxl__qmp_rx_buf) entry;
+ /* How much data there is in buf */
+ int used;
+ /* How much have been parsed */
+ size_t consumed;
+ char buf[QMP_RECEIVE_BUFFER_SIZE];
+};
+
struct libxl__ev_qmp_state {
libxl__carefd *cfd;
libxl__ev_fd efd;
uint32_t domid;
+
+ LIBXL_TAILQ_HEAD(libxl__qmp_bufs, libxl__qmp_rx_buf) bufs;
};
+
+static int ev_qmp_callback_readable(libxl__egc *egc, libxl__ev_qmp_state *qmp,
+ int fd)
+{
+ EGC_GC;
+ ssize_t r;
+ libxl__qmp_rx_buf *buf;
+
+ /* Check if last buffer still have space, or alloc a new one */
+ buf = LIBXL_TAILQ_LAST(&qmp->bufs, libxl__qmp_bufs);
+ if (buf) {
+ /* The -1 is because there is always space for a NUL character */
+ if (buf->used == sizeof(buf->buf) - 1) {
+ buf = NULL;
+ }
+ }
+ if (!buf) {
+ buf = libxl__malloc(NOGC, sizeof(*buf));
+ buf->used = 0;
+ buf->consumed = 0;
+ LIBXL_TAILQ_INSERT_TAIL(&qmp->bufs, buf, entry);
+ }
+
+ for (;;) {
+ /* The -1 is because there is always space for a NUL character */
+ r = read(fd, buf->buf + buf->used, sizeof(buf->buf) - buf->used - 1);
+ if (r < 0) {
+ if (errno == EINTR) continue;
+ assert(errno);
+ if (errno == EWOULDBLOCK) {
+ return 0;
+ }
+ LOGED(ERROR, qmp->domid, "error reading QMP socket");
+ return ERROR_FAIL;
+ }
+ break;
+ }
+
+ if (r == 0) {
+ LOGD(ERROR, qmp->domid, "No data read on QMP socket");
+ return 0;
+ }
+
+ LOG_QMP("received %ldB: '%.*s'", r, (int)r, buf->buf + buf->used);
+
+ buf->used += r;
+ assert(buf->used < sizeof(buf->buf));
+
+ return 0;
+}
+
+/* When the QMP client reach the conclusion that the QMP connection doesn't
+ * work anymore, this function can be called to propagate the error to every
+ * callback registered. And stop the client. */
+static void ev_qmp_callback_error(libxl__egc *egc, libxl__ev_qmp_state *qmp)
+{
+ EGC_GC;
+
+ LOGD(ERROR, qmp->domid, "Error happend with the QMP connection to QEMU");
+ libxl__ev_qmp_stop(gc, qmp);
+}
+
static void ev_qmp_fd_callback(libxl__egc *egc, libxl__ev_fd *ev_fd,
int fd, short events, short revents)
{
+ EGC_GC;
+ int rc;
+
+ libxl__ev_qmp_state *qmp = CONTAINER_OF(ev_fd, *qmp, efd);
+
+ if (revents & (POLLHUP)) {
+ LOGD(DEBUG, qmp->domid, "received POLLHUP from QMP socket");
+ ev_qmp_callback_error(egc, qmp);
+ return;
+ }
+ if (revents & ~(POLLIN|POLLOUT)) {
+ LOGD(ERROR, qmp->domid,
+ "unexpected poll event 0x%x on QMP socket (expected POLLIN "
+ "and/or POLLOUT)",
+ revents);
+ ev_qmp_callback_error(egc, qmp);
+ return;
+ }
+
+ if (revents & POLLIN) {
+ rc = ev_qmp_callback_readable(egc, qmp, fd);
+ if (rc) {
+ ev_qmp_callback_error(egc, qmp);
+ return;
+ }
+ }
}
static void libxl__ev_qmp_state_init(libxl__ev_qmp_state *qmp)
@@ -1369,6 +1475,7 @@ static void libxl__ev_qmp_state_init(libxl__ev_qmp_state *qmp)
qmp->domid = INVALID_DOMID;
qmp->cfd = NULL;
libxl__ev_fd_init(&qmp->efd);
+ LIBXL_TAILQ_INIT(&qmp->bufs);
}
libxl__ev_qmp_state *libxl__ev_qmp_start(libxl__gc *gc, uint32_t domid)
@@ -1444,6 +1551,8 @@ out:
void libxl__ev_qmp_stop(libxl__gc *gc, libxl__ev_qmp_state *qmp)
{
+ libxl__qmp_rx_buf *buf, *tbuf;
+
if (!qmp)
return;
@@ -1451,6 +1560,10 @@ void libxl__ev_qmp_stop(libxl__gc *gc, libxl__ev_qmp_state *qmp)
LOGD(DEBUG, qmp->domid, "Stopping QMP handler");
+ LIBXL_TAILQ_FOREACH_SAFE(buf, &qmp->bufs, entry, tbuf)
+ free(buf);
+ LIBXL_TAILQ_INIT(&qmp->bufs);
+
libxl__ev_fd_deregister(gc, &qmp->efd);
libxl__carefd_close(qmp->cfd);
qmp->cfd = NULL;
--
Anthony PERARD
_______________________________________________
Xen-devel mailing list
Xen-devel@lists.xenproject.org
https://lists.xenproject.org/mailman/listinfo/xen-devel
next prev parent reply other threads:[~2018-06-01 14:58 UTC|newest]
Thread overview: 78+ messages / expand[flat|nested] mbox.gz Atom feed top
2018-06-01 14:36 [PATCH v3 00/31] libxl: Enable save/restore/migration of a restricted QEMU + libxl__ev_qmp_* Anthony PERARD
2018-06-01 14:36 ` [PATCH v3 01/31] libxl_event: Fix DEBUG prints Anthony PERARD
2018-06-27 14:19 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 02/31] libxl_qmp: Documentation of the logic of the QMP client Anthony PERARD
2018-06-27 14:20 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 03/31] libxl_qmp: Fix use of DEBUG_RECEIVED Anthony PERARD
2018-06-27 14:20 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 04/31] libxl_json: fix build with DEBUG_ANSWER Anthony PERARD
2018-06-27 14:22 ` Ian Jackson
2018-07-17 14:35 ` Anthony PERARD
2018-06-01 14:36 ` [PATCH v3 05/31] libxl_qmp: Move the buffer realloc to the same scope level as read Anthony PERARD
2018-06-27 14:25 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 06/31] libxl_qmp: Add a warning to not trust QEMU Anthony PERARD
2018-06-27 14:25 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 07/31] libxl_qmp: Learned to send FD through QMP to QEMU Anthony PERARD
2018-06-27 14:26 ` Ian Jackson
2018-06-27 16:50 ` Anthony PERARD
2018-06-28 9:56 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 08/31] libxl_qmp: Have QEMU save its state to a file descriptor Anthony PERARD
2018-06-27 14:29 ` Ian Jackson
2018-06-01 14:36 ` [PATCH v3 09/31] libxl_qmp: Move struct sockaddr_un variable to qmp_open() Anthony PERARD
2018-06-27 14:31 ` Ian Jackson
2018-06-27 16:54 ` Anthony PERARD
2018-06-01 14:36 ` [PATCH v3 10/31] libxl_qmp: Move buffers to the stack of qmp_next Anthony PERARD
2018-06-27 14:32 ` Ian Jackson
2018-06-27 16:58 ` Anthony PERARD
2018-06-28 9:57 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 11/31] libxl_qmp: Remove unused yajl_ctx form handler Anthony PERARD
2018-06-27 14:32 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 12/31] libxl_json: constify libxl__json_object_to_yajl_gen arguments Anthony PERARD
2018-06-27 14:32 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 13/31] libxl_qmp: Separate QMP message generation from qmp_send_prepare Anthony PERARD
2018-06-27 14:45 ` Ian Jackson
2018-06-27 17:12 ` Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 14/31] libxl_qmp_ev: Introduce libxl__ev_qmp_start() to connect to QMP Anthony PERARD
2018-06-27 15:07 ` Ian Jackson
2018-06-28 11:28 ` Anthony PERARD
2018-06-28 11:44 ` Ian Jackson
2018-06-28 13:01 ` Anthony PERARD
2018-06-01 14:37 ` Anthony PERARD [this message]
2018-06-27 15:10 ` [PATCH v3 15/31] libxl_qmp_ev: Implement fd callback and read data Ian Jackson
2018-06-28 11:33 ` Anthony PERARD
2018-06-28 11:37 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 16/31] libxl_json: Allow partial parsing Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 17/31] libxl_json: Enable yajl_allow_trailing_garbage Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 18/31] libxl_json: libxl__json_object_to_json Anthony PERARD
2018-06-27 15:51 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 19/31] libxl_qmp_ev: Parse JSON input from QMP Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 20/31] libxl_qmp: Introduce libxl__ev_qmp functions Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 21/31] libxl_qmp_ev: Handle write to socket Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 22/31] libxl_qmp: Simplify qmp_response_type() prototype Anthony PERARD
2018-06-27 16:03 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 23/31] libxl_qmp_ev: Handle messages from QEMU Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 24/31] libxl_qmp_ev: Respond to QMP greeting Anthony PERARD
2018-06-27 16:07 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 25/31] libxl_qmp_ev: Disconnect QMP when no more events Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 26/31] libxl_qmp: Disable beautify for QMP generated cmd Anthony PERARD
2018-06-27 16:07 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 27/31] libxl_qmp: Implement libxl__qmp_insert_cdrom_ev Anthony PERARD
2018-06-27 16:10 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 28/31] libxl_disk: Cut libxl_cdrom_insert into step Anthony PERARD
2018-06-01 14:37 ` [PATCH v3 29/31] libxl_disk: Have libxl_cdrom_insert use libxl__ev_qmp Anthony PERARD
2018-06-27 16:12 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 30/31] libxl_dm: Pre-open QMP socket for QEMU Anthony PERARD
2018-06-27 16:14 ` Ian Jackson
2018-06-01 14:37 ` [PATCH v3 31/31] libxl: QEMU startup sync based on QMP Anthony PERARD
2018-06-27 16:16 ` Ian Jackson
2018-07-26 14:59 ` Anthony PERARD
2018-06-01 14:47 ` [PATCH v3 00/31] libxl: Enable save/restore/migration of a restricted QEMU + libxl__ev_qmp_* Anthony PERARD
2018-07-03 9:47 ` [PATCH v3.1] libxl: Design of an async API to issue QMP commands to QEMU Anthony PERARD
2018-07-03 14:48 ` Ian Jackson
2018-07-04 11:11 ` Anthony PERARD
2018-07-12 16:36 ` Ian Jackson
2018-07-16 15:27 ` Anthony PERARD
2018-07-16 15:28 ` [PATCH v3.2] " Anthony PERARD
2018-07-16 16:33 ` Anthony PERARD
2018-07-16 17:04 ` [PATCH v3.2] libxl: Design of an async API to issue QMP commands to QEMU [and 1 more messages] Ian Jackson
2018-07-18 10:30 ` Anthony PERARD
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20180601143720.24637-16-anthony.perard@citrix.com \
--to=anthony.perard@citrix.com \
--cc=ian.jackson@eu.citrix.com \
--cc=wei.liu2@citrix.com \
--cc=xen-devel@lists.xenproject.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).