 0472b2e541
			
		
	
	
		0472b2e541
		
	
	
	
	
		
			
			Signed-off-by: Daniel P. Berrangé <berrange@redhat.com> Message-Id: <20221219130205.687815-6-berrange@redhat.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
		
			
				
	
	
		
			259 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			259 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * QTest
 | |
|  *
 | |
|  * Copyright IBM, Corp. 2012
 | |
|  * Copyright Red Hat, Inc. 2012
 | |
|  * Copyright SUSE LINUX Products GmbH 2013
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Anthony Liguori   <aliguori@us.ibm.com>
 | |
|  *  Paolo Bonzini     <pbonzini@redhat.com>
 | |
|  *  Andreas Färber    <afaerber@suse.de>
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| 
 | |
| #include "libqmp.h"
 | |
| 
 | |
| #ifndef _WIN32
 | |
| #include <sys/socket.h>
 | |
| #endif
 | |
| 
 | |
| #include "qemu/cutils.h"
 | |
| #include "qemu/sockets.h"
 | |
| #include "qapi/error.h"
 | |
| #include "qapi/qmp/json-parser.h"
 | |
| #include "qapi/qmp/qjson.h"
 | |
| 
 | |
| #define SOCKET_MAX_FDS 16
 | |
| 
 | |
| typedef struct {
 | |
|     JSONMessageParser parser;
 | |
|     QDict *response;
 | |
| } QMPResponseParser;
 | |
| 
 | |
| static void socket_send(int fd, const char *buf, size_t size)
 | |
| {
 | |
|     ssize_t res = qemu_send_full(fd, buf, size);
 | |
| 
 | |
|     assert(res == size);
 | |
| }
 | |
| 
 | |
| static void qmp_response(void *opaque, QObject *obj, Error *err)
 | |
| {
 | |
|     QMPResponseParser *qmp = opaque;
 | |
| 
 | |
|     assert(!obj != !err);
 | |
| 
 | |
|     if (err) {
 | |
|         error_prepend(&err, "QMP JSON response parsing failed: ");
 | |
|         error_report_err(err);
 | |
|         abort();
 | |
|     }
 | |
| 
 | |
|     g_assert(!qmp->response);
 | |
|     qmp->response = qobject_to(QDict, obj);
 | |
|     g_assert(qmp->response);
 | |
| }
 | |
| 
 | |
| QDict *qmp_fd_receive(int fd)
 | |
| {
 | |
|     QMPResponseParser qmp;
 | |
|     bool log = getenv("QTEST_LOG") != NULL;
 | |
| 
 | |
|     qmp.response = NULL;
 | |
|     json_message_parser_init(&qmp.parser, qmp_response, &qmp, NULL);
 | |
|     while (!qmp.response) {
 | |
|         ssize_t len;
 | |
|         char c;
 | |
| 
 | |
|         len = recv(fd, &c, 1, 0);
 | |
|         if (len == -1 && errno == EINTR) {
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         if (len == -1 || len == 0) {
 | |
|             fprintf(stderr, "Broken pipe\n");
 | |
|             abort();
 | |
|         }
 | |
| 
 | |
|         if (log) {
 | |
|             g_assert(write(2, &c, 1) == 1);
 | |
|         }
 | |
|         json_message_parser_feed(&qmp.parser, &c, 1);
 | |
|     }
 | |
|     if (log) {
 | |
|         g_assert(write(2, "\n", 1) == 1);
 | |
|     }
 | |
|     json_message_parser_destroy(&qmp.parser);
 | |
| 
 | |
|     return qmp.response;
 | |
| }
 | |
| 
 | |
| #ifndef _WIN32
 | |
| /* Sends a message and file descriptors to the socket.
 | |
|  * It's needed for qmp-commands like getfd/add-fd */
 | |
| static void socket_send_fds(int socket_fd, int *fds, size_t fds_num,
 | |
|                             const char *buf, size_t buf_size)
 | |
| {
 | |
|     ssize_t ret;
 | |
|     struct msghdr msg = { 0 };
 | |
|     char control[CMSG_SPACE(sizeof(int) * SOCKET_MAX_FDS)] = { 0 };
 | |
|     size_t fdsize = sizeof(int) * fds_num;
 | |
|     struct cmsghdr *cmsg;
 | |
|     struct iovec iov = { .iov_base = (char *)buf, .iov_len = buf_size };
 | |
| 
 | |
|     msg.msg_iov = &iov;
 | |
|     msg.msg_iovlen = 1;
 | |
| 
 | |
|     if (fds && fds_num > 0) {
 | |
|         g_assert_cmpuint(fds_num, <, SOCKET_MAX_FDS);
 | |
| 
 | |
|         msg.msg_control = control;
 | |
|         msg.msg_controllen = CMSG_SPACE(fdsize);
 | |
| 
 | |
|         cmsg = CMSG_FIRSTHDR(&msg);
 | |
|         cmsg->cmsg_len = CMSG_LEN(fdsize);
 | |
|         cmsg->cmsg_level = SOL_SOCKET;
 | |
|         cmsg->cmsg_type = SCM_RIGHTS;
 | |
|         memcpy(CMSG_DATA(cmsg), fds, fdsize);
 | |
|     }
 | |
| 
 | |
|     do {
 | |
|         ret = sendmsg(socket_fd, &msg, 0);
 | |
|     } while (ret < 0 && errno == EINTR);
 | |
|     g_assert_cmpint(ret, >, 0);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /**
 | |
|  * Allow users to send a message without waiting for the reply,
 | |
|  * in the case that they choose to discard all replies up until
 | |
|  * a particular EVENT is received.
 | |
|  */
 | |
| static G_GNUC_PRINTF(4, 0) void
 | |
| _qmp_fd_vsend_fds(int fd, int *fds, size_t fds_num,
 | |
|                   const char *fmt, va_list ap)
 | |
| {
 | |
|     QObject *qobj;
 | |
| 
 | |
| #ifdef _WIN32
 | |
|     assert(fds_num == 0);
 | |
| #endif
 | |
| 
 | |
|     /* Going through qobject ensures we escape strings properly */
 | |
|     qobj = qobject_from_vjsonf_nofail(fmt, ap);
 | |
| 
 | |
|     /* No need to send anything for an empty QObject.  */
 | |
|     if (qobj) {
 | |
|         int log = getenv("QTEST_LOG") != NULL;
 | |
|         GString *str = qobject_to_json(qobj);
 | |
| 
 | |
|         /*
 | |
|          * BUG: QMP doesn't react to input until it sees a newline, an
 | |
|          * object, or an array.  Work-around: give it a newline.
 | |
|          */
 | |
|         g_string_append_c(str, '\n');
 | |
| 
 | |
|         if (log) {
 | |
|             fprintf(stderr, "%s", str->str);
 | |
|         }
 | |
| 
 | |
| #ifndef _WIN32
 | |
|         /* Send QMP request */
 | |
|         if (fds && fds_num > 0) {
 | |
|             socket_send_fds(fd, fds, fds_num, str->str, str->len);
 | |
|         } else
 | |
| #endif
 | |
|         {
 | |
|             socket_send(fd, str->str, str->len);
 | |
|         }
 | |
| 
 | |
|         g_string_free(str, true);
 | |
|         qobject_unref(qobj);
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifndef _WIN32
 | |
| void qmp_fd_vsend_fds(int fd, int *fds, size_t fds_num,
 | |
|                       const char *fmt, va_list ap)
 | |
| {
 | |
|     _qmp_fd_vsend_fds(fd, fds, fds_num, fmt, ap);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void qmp_fd_vsend(int fd, const char *fmt, va_list ap)
 | |
| {
 | |
|     _qmp_fd_vsend_fds(fd, NULL, 0, fmt, ap);
 | |
| }
 | |
| 
 | |
| 
 | |
| QDict *qmp_fdv(int fd, const char *fmt, va_list ap)
 | |
| {
 | |
|     _qmp_fd_vsend_fds(fd, NULL, 0, fmt, ap);
 | |
| 
 | |
|     return qmp_fd_receive(fd);
 | |
| }
 | |
| 
 | |
| QDict *qmp_fd(int fd, const char *fmt, ...)
 | |
| {
 | |
|     va_list ap;
 | |
|     QDict *response;
 | |
| 
 | |
|     va_start(ap, fmt);
 | |
|     response = qmp_fdv(fd, fmt, ap);
 | |
|     va_end(ap);
 | |
|     return response;
 | |
| }
 | |
| 
 | |
| void qmp_fd_send(int fd, const char *fmt, ...)
 | |
| {
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, fmt);
 | |
|     qmp_fd_vsend(fd, fmt, ap);
 | |
|     va_end(ap);
 | |
| }
 | |
| 
 | |
| void qmp_fd_vsend_raw(int fd, const char *fmt, va_list ap)
 | |
| {
 | |
|     bool log = getenv("QTEST_LOG") != NULL;
 | |
|     char *str = g_strdup_vprintf(fmt, ap);
 | |
| 
 | |
|     if (log) {
 | |
|         fprintf(stderr, "%s", str);
 | |
|     }
 | |
|     socket_send(fd, str, strlen(str));
 | |
|     g_free(str);
 | |
| }
 | |
| 
 | |
| void qmp_fd_send_raw(int fd, const char *fmt, ...)
 | |
| {
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, fmt);
 | |
|     qmp_fd_vsend_raw(fd, fmt, ap);
 | |
|     va_end(ap);
 | |
| }
 | |
| 
 | |
| bool qmp_rsp_is_err(QDict *rsp)
 | |
| {
 | |
|     QDict *error = qdict_get_qdict(rsp, "error");
 | |
|     qobject_unref(rsp);
 | |
|     return !!error;
 | |
| }
 | |
| 
 | |
| void qmp_expect_error_and_unref(QDict *rsp, const char *class)
 | |
| {
 | |
|     QDict *error = qdict_get_qdict(rsp, "error");
 | |
| 
 | |
|     g_assert_cmpstr(qdict_get_try_str(error, "class"), ==, class);
 | |
|     g_assert_nonnull(qdict_get_try_str(error, "desc"));
 | |
|     g_assert(!qdict_haskey(rsp, "return"));
 | |
| 
 | |
|     qobject_unref(rsp);
 | |
| }
 |