xref: /qemu/tests/qtest/libqmp.c (revision 56b6dab2743b276a50036cd0a115e98cb1134f38)
1c6d3bcb4SMarc-André Lureau /*
2c6d3bcb4SMarc-André Lureau  * QTest
3c6d3bcb4SMarc-André Lureau  *
4c6d3bcb4SMarc-André Lureau  * Copyright IBM, Corp. 2012
5c6d3bcb4SMarc-André Lureau  * Copyright Red Hat, Inc. 2012
6c6d3bcb4SMarc-André Lureau  * Copyright SUSE LINUX Products GmbH 2013
7c6d3bcb4SMarc-André Lureau  *
8c6d3bcb4SMarc-André Lureau  * Authors:
9c6d3bcb4SMarc-André Lureau  *  Anthony Liguori   <aliguori@us.ibm.com>
10c6d3bcb4SMarc-André Lureau  *  Paolo Bonzini     <pbonzini@redhat.com>
11c6d3bcb4SMarc-André Lureau  *  Andreas Färber    <afaerber@suse.de>
12c6d3bcb4SMarc-André Lureau  *
13c6d3bcb4SMarc-André Lureau  * This work is licensed under the terms of the GNU GPL, version 2 or later.
14c6d3bcb4SMarc-André Lureau  * See the COPYING file in the top-level directory.
15c6d3bcb4SMarc-André Lureau  */
16c6d3bcb4SMarc-André Lureau 
17c6d3bcb4SMarc-André Lureau #include "qemu/osdep.h"
18c6d3bcb4SMarc-André Lureau 
19c6d3bcb4SMarc-André Lureau #include "libqmp.h"
20c6d3bcb4SMarc-André Lureau 
21*56b6dab2SMarc-André Lureau #ifndef _WIN32
22*56b6dab2SMarc-André Lureau #include <sys/socket.h>
23*56b6dab2SMarc-André Lureau #endif
24*56b6dab2SMarc-André Lureau 
25*56b6dab2SMarc-André Lureau #include "qemu/cutils.h"
26c6d3bcb4SMarc-André Lureau #include "qapi/error.h"
27c6d3bcb4SMarc-André Lureau #include "qapi/qmp/json-parser.h"
28c6d3bcb4SMarc-André Lureau #include "qapi/qmp/qjson.h"
29c6d3bcb4SMarc-André Lureau 
30c6d3bcb4SMarc-André Lureau #define SOCKET_MAX_FDS 16
31c6d3bcb4SMarc-André Lureau 
32c6d3bcb4SMarc-André Lureau typedef struct {
33c6d3bcb4SMarc-André Lureau     JSONMessageParser parser;
34c6d3bcb4SMarc-André Lureau     QDict *response;
35c6d3bcb4SMarc-André Lureau } QMPResponseParser;
36c6d3bcb4SMarc-André Lureau 
37c6d3bcb4SMarc-André Lureau static void socket_send(int fd, const char *buf, size_t size)
38c6d3bcb4SMarc-André Lureau {
39c6d3bcb4SMarc-André Lureau     size_t res = qemu_write_full(fd, buf, size);
40c6d3bcb4SMarc-André Lureau 
41c6d3bcb4SMarc-André Lureau     assert(res == size);
42c6d3bcb4SMarc-André Lureau }
43c6d3bcb4SMarc-André Lureau 
44c6d3bcb4SMarc-André Lureau static void qmp_response(void *opaque, QObject *obj, Error *err)
45c6d3bcb4SMarc-André Lureau {
46c6d3bcb4SMarc-André Lureau     QMPResponseParser *qmp = opaque;
47c6d3bcb4SMarc-André Lureau 
48c6d3bcb4SMarc-André Lureau     assert(!obj != !err);
49c6d3bcb4SMarc-André Lureau 
50c6d3bcb4SMarc-André Lureau     if (err) {
51c6d3bcb4SMarc-André Lureau         error_prepend(&err, "QMP JSON response parsing failed: ");
52c6d3bcb4SMarc-André Lureau         error_report_err(err);
53c6d3bcb4SMarc-André Lureau         abort();
54c6d3bcb4SMarc-André Lureau     }
55c6d3bcb4SMarc-André Lureau 
56c6d3bcb4SMarc-André Lureau     g_assert(!qmp->response);
57c6d3bcb4SMarc-André Lureau     qmp->response = qobject_to(QDict, obj);
58c6d3bcb4SMarc-André Lureau     g_assert(qmp->response);
59c6d3bcb4SMarc-André Lureau }
60c6d3bcb4SMarc-André Lureau 
61c6d3bcb4SMarc-André Lureau QDict *qmp_fd_receive(int fd)
62c6d3bcb4SMarc-André Lureau {
63c6d3bcb4SMarc-André Lureau     QMPResponseParser qmp;
64c6d3bcb4SMarc-André Lureau     bool log = getenv("QTEST_LOG") != NULL;
65c6d3bcb4SMarc-André Lureau 
66c6d3bcb4SMarc-André Lureau     qmp.response = NULL;
67c6d3bcb4SMarc-André Lureau     json_message_parser_init(&qmp.parser, qmp_response, &qmp, NULL);
68c6d3bcb4SMarc-André Lureau     while (!qmp.response) {
69c6d3bcb4SMarc-André Lureau         ssize_t len;
70c6d3bcb4SMarc-André Lureau         char c;
71c6d3bcb4SMarc-André Lureau 
72c6d3bcb4SMarc-André Lureau         len = read(fd, &c, 1);
73c6d3bcb4SMarc-André Lureau         if (len == -1 && errno == EINTR) {
74c6d3bcb4SMarc-André Lureau             continue;
75c6d3bcb4SMarc-André Lureau         }
76c6d3bcb4SMarc-André Lureau 
77c6d3bcb4SMarc-André Lureau         if (len == -1 || len == 0) {
78c6d3bcb4SMarc-André Lureau             fprintf(stderr, "Broken pipe\n");
79c6d3bcb4SMarc-André Lureau             abort();
80c6d3bcb4SMarc-André Lureau         }
81c6d3bcb4SMarc-André Lureau 
82c6d3bcb4SMarc-André Lureau         if (log) {
83c6d3bcb4SMarc-André Lureau             g_assert(write(2, &c, 1) == 1);
84c6d3bcb4SMarc-André Lureau         }
85c6d3bcb4SMarc-André Lureau         json_message_parser_feed(&qmp.parser, &c, 1);
86c6d3bcb4SMarc-André Lureau     }
87c6d3bcb4SMarc-André Lureau     if (log) {
88c6d3bcb4SMarc-André Lureau         g_assert(write(2, "\n", 1) == 1);
89c6d3bcb4SMarc-André Lureau     }
90c6d3bcb4SMarc-André Lureau     json_message_parser_destroy(&qmp.parser);
91c6d3bcb4SMarc-André Lureau 
92c6d3bcb4SMarc-André Lureau     return qmp.response;
93c6d3bcb4SMarc-André Lureau }
94c6d3bcb4SMarc-André Lureau 
95*56b6dab2SMarc-André Lureau #ifndef _WIN32
96c6d3bcb4SMarc-André Lureau /* Sends a message and file descriptors to the socket.
97c6d3bcb4SMarc-André Lureau  * It's needed for qmp-commands like getfd/add-fd */
98c6d3bcb4SMarc-André Lureau static void socket_send_fds(int socket_fd, int *fds, size_t fds_num,
99c6d3bcb4SMarc-André Lureau                             const char *buf, size_t buf_size)
100c6d3bcb4SMarc-André Lureau {
101c6d3bcb4SMarc-André Lureau     ssize_t ret;
102c6d3bcb4SMarc-André Lureau     struct msghdr msg = { 0 };
103c6d3bcb4SMarc-André Lureau     char control[CMSG_SPACE(sizeof(int) * SOCKET_MAX_FDS)] = { 0 };
104c6d3bcb4SMarc-André Lureau     size_t fdsize = sizeof(int) * fds_num;
105c6d3bcb4SMarc-André Lureau     struct cmsghdr *cmsg;
106c6d3bcb4SMarc-André Lureau     struct iovec iov = { .iov_base = (char *)buf, .iov_len = buf_size };
107c6d3bcb4SMarc-André Lureau 
108c6d3bcb4SMarc-André Lureau     msg.msg_iov = &iov;
109c6d3bcb4SMarc-André Lureau     msg.msg_iovlen = 1;
110c6d3bcb4SMarc-André Lureau 
111c6d3bcb4SMarc-André Lureau     if (fds && fds_num > 0) {
112c6d3bcb4SMarc-André Lureau         g_assert_cmpuint(fds_num, <, SOCKET_MAX_FDS);
113c6d3bcb4SMarc-André Lureau 
114c6d3bcb4SMarc-André Lureau         msg.msg_control = control;
115c6d3bcb4SMarc-André Lureau         msg.msg_controllen = CMSG_SPACE(fdsize);
116c6d3bcb4SMarc-André Lureau 
117c6d3bcb4SMarc-André Lureau         cmsg = CMSG_FIRSTHDR(&msg);
118c6d3bcb4SMarc-André Lureau         cmsg->cmsg_len = CMSG_LEN(fdsize);
119c6d3bcb4SMarc-André Lureau         cmsg->cmsg_level = SOL_SOCKET;
120c6d3bcb4SMarc-André Lureau         cmsg->cmsg_type = SCM_RIGHTS;
121c6d3bcb4SMarc-André Lureau         memcpy(CMSG_DATA(cmsg), fds, fdsize);
122c6d3bcb4SMarc-André Lureau     }
123c6d3bcb4SMarc-André Lureau 
124c6d3bcb4SMarc-André Lureau     do {
125c6d3bcb4SMarc-André Lureau         ret = sendmsg(socket_fd, &msg, 0);
126c6d3bcb4SMarc-André Lureau     } while (ret < 0 && errno == EINTR);
127c6d3bcb4SMarc-André Lureau     g_assert_cmpint(ret, >, 0);
128c6d3bcb4SMarc-André Lureau }
129*56b6dab2SMarc-André Lureau #endif
130c6d3bcb4SMarc-André Lureau 
131c6d3bcb4SMarc-André Lureau /**
132c6d3bcb4SMarc-André Lureau  * Allow users to send a message without waiting for the reply,
133c6d3bcb4SMarc-André Lureau  * in the case that they choose to discard all replies up until
134c6d3bcb4SMarc-André Lureau  * a particular EVENT is received.
135c6d3bcb4SMarc-André Lureau  */
136*56b6dab2SMarc-André Lureau static void
137*56b6dab2SMarc-André Lureau _qmp_fd_vsend_fds(int fd, int *fds, size_t fds_num,
138c6d3bcb4SMarc-André Lureau                   const char *fmt, va_list ap)
139c6d3bcb4SMarc-André Lureau {
140c6d3bcb4SMarc-André Lureau     QObject *qobj;
141c6d3bcb4SMarc-André Lureau 
142*56b6dab2SMarc-André Lureau #ifdef _WIN32
143*56b6dab2SMarc-André Lureau     assert(fds_num == 0);
144*56b6dab2SMarc-André Lureau #endif
145*56b6dab2SMarc-André Lureau 
146c6d3bcb4SMarc-André Lureau     /* Going through qobject ensures we escape strings properly */
147c6d3bcb4SMarc-André Lureau     qobj = qobject_from_vjsonf_nofail(fmt, ap);
148c6d3bcb4SMarc-André Lureau 
149c6d3bcb4SMarc-André Lureau     /* No need to send anything for an empty QObject.  */
150c6d3bcb4SMarc-André Lureau     if (qobj) {
151c6d3bcb4SMarc-André Lureau         int log = getenv("QTEST_LOG") != NULL;
152c6d3bcb4SMarc-André Lureau         GString *str = qobject_to_json(qobj);
153c6d3bcb4SMarc-André Lureau 
154c6d3bcb4SMarc-André Lureau         /*
155c6d3bcb4SMarc-André Lureau          * BUG: QMP doesn't react to input until it sees a newline, an
156c6d3bcb4SMarc-André Lureau          * object, or an array.  Work-around: give it a newline.
157c6d3bcb4SMarc-André Lureau          */
158c6d3bcb4SMarc-André Lureau         g_string_append_c(str, '\n');
159c6d3bcb4SMarc-André Lureau 
160c6d3bcb4SMarc-André Lureau         if (log) {
161c6d3bcb4SMarc-André Lureau             fprintf(stderr, "%s", str->str);
162c6d3bcb4SMarc-André Lureau         }
163*56b6dab2SMarc-André Lureau 
164*56b6dab2SMarc-André Lureau #ifndef _WIN32
165c6d3bcb4SMarc-André Lureau         /* Send QMP request */
166c6d3bcb4SMarc-André Lureau         if (fds && fds_num > 0) {
167c6d3bcb4SMarc-André Lureau             socket_send_fds(fd, fds, fds_num, str->str, str->len);
168*56b6dab2SMarc-André Lureau         } else
169*56b6dab2SMarc-André Lureau #endif
170*56b6dab2SMarc-André Lureau         {
171c6d3bcb4SMarc-André Lureau             socket_send(fd, str->str, str->len);
172c6d3bcb4SMarc-André Lureau         }
173c6d3bcb4SMarc-André Lureau 
174c6d3bcb4SMarc-André Lureau         g_string_free(str, true);
175c6d3bcb4SMarc-André Lureau         qobject_unref(qobj);
176c6d3bcb4SMarc-André Lureau     }
177c6d3bcb4SMarc-André Lureau }
178c6d3bcb4SMarc-André Lureau 
179*56b6dab2SMarc-André Lureau #ifndef _WIN32
180*56b6dab2SMarc-André Lureau void qmp_fd_vsend_fds(int fd, int *fds, size_t fds_num,
181*56b6dab2SMarc-André Lureau                       const char *fmt, va_list ap)
182*56b6dab2SMarc-André Lureau {
183*56b6dab2SMarc-André Lureau     _qmp_fd_vsend_fds(fd, fds, fds_num, fmt, ap);
184*56b6dab2SMarc-André Lureau }
185*56b6dab2SMarc-André Lureau #endif
186*56b6dab2SMarc-André Lureau 
187c6d3bcb4SMarc-André Lureau void qmp_fd_vsend(int fd, const char *fmt, va_list ap)
188c6d3bcb4SMarc-André Lureau {
189*56b6dab2SMarc-André Lureau     _qmp_fd_vsend_fds(fd, NULL, 0, fmt, ap);
190c6d3bcb4SMarc-André Lureau }
191c6d3bcb4SMarc-André Lureau 
192c6d3bcb4SMarc-André Lureau 
193c6d3bcb4SMarc-André Lureau QDict *qmp_fdv(int fd, const char *fmt, va_list ap)
194c6d3bcb4SMarc-André Lureau {
195*56b6dab2SMarc-André Lureau     _qmp_fd_vsend_fds(fd, NULL, 0, fmt, ap);
196c6d3bcb4SMarc-André Lureau 
197c6d3bcb4SMarc-André Lureau     return qmp_fd_receive(fd);
198c6d3bcb4SMarc-André Lureau }
199c6d3bcb4SMarc-André Lureau 
200c6d3bcb4SMarc-André Lureau QDict *qmp_fd(int fd, const char *fmt, ...)
201c6d3bcb4SMarc-André Lureau {
202c6d3bcb4SMarc-André Lureau     va_list ap;
203c6d3bcb4SMarc-André Lureau     QDict *response;
204c6d3bcb4SMarc-André Lureau 
205c6d3bcb4SMarc-André Lureau     va_start(ap, fmt);
206c6d3bcb4SMarc-André Lureau     response = qmp_fdv(fd, fmt, ap);
207c6d3bcb4SMarc-André Lureau     va_end(ap);
208c6d3bcb4SMarc-André Lureau     return response;
209c6d3bcb4SMarc-André Lureau }
210c6d3bcb4SMarc-André Lureau 
211c6d3bcb4SMarc-André Lureau void qmp_fd_send(int fd, const char *fmt, ...)
212c6d3bcb4SMarc-André Lureau {
213c6d3bcb4SMarc-André Lureau     va_list ap;
214c6d3bcb4SMarc-André Lureau 
215c6d3bcb4SMarc-André Lureau     va_start(ap, fmt);
216c6d3bcb4SMarc-André Lureau     qmp_fd_vsend(fd, fmt, ap);
217c6d3bcb4SMarc-André Lureau     va_end(ap);
218c6d3bcb4SMarc-André Lureau }
219c6d3bcb4SMarc-André Lureau 
220c6d3bcb4SMarc-André Lureau void qmp_fd_vsend_raw(int fd, const char *fmt, va_list ap)
221c6d3bcb4SMarc-André Lureau {
222c6d3bcb4SMarc-André Lureau     bool log = getenv("QTEST_LOG") != NULL;
223c6d3bcb4SMarc-André Lureau     char *str = g_strdup_vprintf(fmt, ap);
224c6d3bcb4SMarc-André Lureau 
225c6d3bcb4SMarc-André Lureau     if (log) {
226c6d3bcb4SMarc-André Lureau         fprintf(stderr, "%s", str);
227c6d3bcb4SMarc-André Lureau     }
228c6d3bcb4SMarc-André Lureau     socket_send(fd, str, strlen(str));
229c6d3bcb4SMarc-André Lureau     g_free(str);
230c6d3bcb4SMarc-André Lureau }
231c6d3bcb4SMarc-André Lureau 
232c6d3bcb4SMarc-André Lureau void qmp_fd_send_raw(int fd, const char *fmt, ...)
233c6d3bcb4SMarc-André Lureau {
234c6d3bcb4SMarc-André Lureau     va_list ap;
235c6d3bcb4SMarc-André Lureau 
236c6d3bcb4SMarc-André Lureau     va_start(ap, fmt);
237c6d3bcb4SMarc-André Lureau     qmp_fd_vsend_raw(fd, fmt, ap);
238c6d3bcb4SMarc-André Lureau     va_end(ap);
239c6d3bcb4SMarc-André Lureau }
240c6d3bcb4SMarc-André Lureau 
241c6d3bcb4SMarc-André Lureau bool qmp_rsp_is_err(QDict *rsp)
242c6d3bcb4SMarc-André Lureau {
243c6d3bcb4SMarc-André Lureau     QDict *error = qdict_get_qdict(rsp, "error");
244c6d3bcb4SMarc-André Lureau     qobject_unref(rsp);
245c6d3bcb4SMarc-André Lureau     return !!error;
246c6d3bcb4SMarc-André Lureau }
247c6d3bcb4SMarc-André Lureau 
248c6d3bcb4SMarc-André Lureau void qmp_expect_error_and_unref(QDict *rsp, const char *class)
249c6d3bcb4SMarc-André Lureau {
250c6d3bcb4SMarc-André Lureau     QDict *error = qdict_get_qdict(rsp, "error");
251c6d3bcb4SMarc-André Lureau 
252c6d3bcb4SMarc-André Lureau     g_assert_cmpstr(qdict_get_try_str(error, "class"), ==, class);
253c6d3bcb4SMarc-André Lureau     g_assert_nonnull(qdict_get_try_str(error, "desc"));
254c6d3bcb4SMarc-André Lureau     g_assert(!qdict_haskey(rsp, "return"));
255c6d3bcb4SMarc-André Lureau 
256c6d3bcb4SMarc-André Lureau     qobject_unref(rsp);
257c6d3bcb4SMarc-André Lureau }
258