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);
|
|
}
|