| #include "qemu/osdep.h" | 
 | #include <locale.h> | 
 | #include <glib/gstdio.h> | 
 | #include <sys/socket.h> | 
 | #include <sys/un.h> | 
 |  | 
 | #include "../qtest/libqtest.h" | 
 | #include "qapi/qmp/qdict.h" | 
 | #include "qapi/qmp/qlist.h" | 
 |  | 
 | typedef struct { | 
 |     char *test_dir; | 
 |     GMainLoop *loop; | 
 |     int fd; | 
 |     GPid pid; | 
 | } TestFixture; | 
 |  | 
 | static int connect_qga(char *path) | 
 | { | 
 |     int s, ret, len, i = 0; | 
 |     struct sockaddr_un remote; | 
 |  | 
 |     s = socket(AF_UNIX, SOCK_STREAM, 0); | 
 |     g_assert(s != -1); | 
 |  | 
 |     remote.sun_family = AF_UNIX; | 
 |     do { | 
 |         strcpy(remote.sun_path, path); | 
 |         len = strlen(remote.sun_path) + sizeof(remote.sun_family); | 
 |         ret = connect(s, (struct sockaddr *)&remote, len); | 
 |         if (ret == -1) { | 
 |             g_usleep(G_USEC_PER_SEC); | 
 |         } | 
 |         if (i++ == 10) { | 
 |             close(s); | 
 |             return -1; | 
 |         } | 
 |     } while (ret == -1); | 
 |  | 
 |     return s; | 
 | } | 
 |  | 
 | static void qga_watch(GPid pid, gint status, gpointer user_data) | 
 | { | 
 |     TestFixture *fixture = user_data; | 
 |  | 
 |     g_assert_cmpint(status, ==, 0); | 
 |     g_main_loop_quit(fixture->loop); | 
 | } | 
 |  | 
 | static void | 
 | fixture_setup(TestFixture *fixture, gconstpointer data, gchar **envp) | 
 | { | 
 |     const gchar *extra_arg = data; | 
 |     GError *error = NULL; | 
 |     g_autofree char *cwd = NULL; | 
 |     g_autofree char *path = NULL; | 
 |     g_autofree char *cmd = NULL; | 
 |     g_auto(GStrv) argv = NULL; | 
 |  | 
 |     fixture->loop = g_main_loop_new(NULL, FALSE); | 
 |  | 
 |     fixture->test_dir = g_strdup_printf("%s/qgatest.XXXXXX", g_get_tmp_dir()); | 
 |     g_assert_nonnull(g_mkdtemp(fixture->test_dir)); | 
 |  | 
 |     path = g_build_filename(fixture->test_dir, "sock", NULL); | 
 |     cwd = g_get_current_dir(); | 
 |     cmd = g_strdup_printf("%s%cqga%cqemu-ga -m unix-listen -t %s -p %s %s %s", | 
 |                           cwd, G_DIR_SEPARATOR, G_DIR_SEPARATOR, | 
 |                           fixture->test_dir, path, | 
 |                           getenv("QTEST_LOG") ? "-v" : "", | 
 |                           extra_arg ?: ""); | 
 |     g_shell_parse_argv(cmd, NULL, &argv, &error); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     g_spawn_async(fixture->test_dir, argv, envp, | 
 |                   G_SPAWN_SEARCH_PATH|G_SPAWN_DO_NOT_REAP_CHILD, | 
 |                   NULL, NULL, &fixture->pid, &error); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     g_child_watch_add(fixture->pid, qga_watch, fixture); | 
 |  | 
 |     fixture->fd = connect_qga(path); | 
 |     g_assert_cmpint(fixture->fd, !=, -1); | 
 | } | 
 |  | 
 | static void | 
 | fixture_tear_down(TestFixture *fixture, gconstpointer data) | 
 | { | 
 |     g_autofree char *tmp = NULL; | 
 |  | 
 |     kill(fixture->pid, SIGTERM); | 
 |  | 
 |     g_main_loop_run(fixture->loop); | 
 |     g_main_loop_unref(fixture->loop); | 
 |  | 
 |     g_spawn_close_pid(fixture->pid); | 
 |  | 
 |     tmp = g_build_filename(fixture->test_dir, "foo", NULL); | 
 |     g_unlink(tmp); | 
 |     g_free(tmp); | 
 |  | 
 |     tmp = g_build_filename(fixture->test_dir, "qga.state", NULL); | 
 |     g_unlink(tmp); | 
 |     g_free(tmp); | 
 |  | 
 |     tmp = g_build_filename(fixture->test_dir, "sock", NULL); | 
 |     g_unlink(tmp); | 
 |  | 
 |     g_rmdir(fixture->test_dir); | 
 |     g_free(fixture->test_dir); | 
 |     close(fixture->fd); | 
 | } | 
 |  | 
 | static void qmp_assertion_message_error(const char     *domain, | 
 |                                         const char     *file, | 
 |                                         int             line, | 
 |                                         const char     *func, | 
 |                                         const char     *expr, | 
 |                                         QDict          *dict) | 
 | { | 
 |     const char *class, *desc; | 
 |     g_autofree char *s = NULL; | 
 |     QDict *error; | 
 |  | 
 |     error = qdict_get_qdict(dict, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |  | 
 |     s = g_strdup_printf("assertion failed %s: %s %s", expr, class, desc); | 
 |     g_assertion_message(domain, file, line, func, s); | 
 | } | 
 |  | 
 | #define qmp_assert_no_error(err) do {                                   \ | 
 |     if (qdict_haskey(err, "error")) {                                   \ | 
 |         qmp_assertion_message_error(G_LOG_DOMAIN, __FILE__, __LINE__,   \ | 
 |                                     G_STRFUNC, #err, err);              \ | 
 |     }                                                                   \ | 
 | } while (0) | 
 |  | 
 | static void test_qga_sync_delimited(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     guint32 v, r = g_test_rand_int(); | 
 |     unsigned char c; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |  | 
 |     qmp_fd_send_raw(fixture->fd, "\xff"); | 
 |     qmp_fd_send(fixture->fd, | 
 |                 "{'execute': 'guest-sync-delimited'," | 
 |                 " 'arguments': {'id': %u } }", | 
 |                 r); | 
 |  | 
 |     /* | 
 |      * Read and ignore garbage until resynchronized. | 
 |      * | 
 |      * Note that the full reset sequence would involve checking the | 
 |      * response of guest-sync-delimited and repeating the loop if | 
 |      * 'id' field of the response does not match the 'id' field of | 
 |      * the request. Testing this fully would require inserting | 
 |      * garbage in the response stream and is left as a future test | 
 |      * to implement. | 
 |      * | 
 |      * TODO: The server shouldn't emit so much garbage (among other | 
 |      * things, it loudly complains about the client's \xff being | 
 |      * invalid JSON, even though it is a documented part of the | 
 |      * handshake. | 
 |      */ | 
 |     do { | 
 |         v = read(fixture->fd, &c, 1); | 
 |         g_assert_cmpint(v, ==, 1); | 
 |     } while (c != 0xff); | 
 |  | 
 |     ret = qmp_fd_receive(fixture->fd); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     v = qdict_get_int(ret, "return"); | 
 |     g_assert_cmpint(r, ==, v); | 
 | } | 
 |  | 
 | static void test_qga_sync(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     guint32 v, r = g_test_rand_int(); | 
 |     g_autoptr(QDict) ret = NULL; | 
 |  | 
 |     /* | 
 |      * TODO guest-sync is inherently limited: we cannot distinguish | 
 |      * failure caused by reacting to garbage on the wire prior to this | 
 |      * command, from failure of this actual command. Clients are | 
 |      * supposed to be able to send a raw '\xff' byte to at least | 
 |      * re-synchronize the server's parser prior to this command, but | 
 |      * we are not in a position to test that here because (at least | 
 |      * for now) it causes the server to issue an error message about | 
 |      * invalid JSON. Testing of '\xff' handling is done in | 
 |      * guest-sync-delimited instead. | 
 |      */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-sync', 'arguments': {'id': %u } }", | 
 |                  r); | 
 |  | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     v = qdict_get_int(ret, "return"); | 
 |     g_assert_cmpint(r, ==, v); | 
 | } | 
 |  | 
 | static void test_qga_ping(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 | } | 
 |  | 
 | static void test_qga_id(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping', 'id': 1}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     g_assert_cmpint(qdict_get_int(ret, "id"), ==, 1); | 
 | } | 
 |  | 
 | static void test_qga_invalid_oob(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     QDict *ret; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'exec-oob': 'guest-ping'}"); | 
 |     g_assert_nonnull(ret); | 
 |  | 
 |     qmp_expect_error_and_unref(ret, "GenericError"); | 
 | } | 
 |  | 
 | static void test_qga_invalid_args(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *error; | 
 |     const gchar *class, *desc; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping', " | 
 |                  "'arguments': {'foo': 42 }}"); | 
 |     g_assert_nonnull(ret); | 
 |  | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |  | 
 |     g_assert_cmpstr(class, ==, "GenericError"); | 
 |     g_assert_cmpstr(desc, ==, "Parameter 'foo' is unexpected"); | 
 | } | 
 |  | 
 | static void test_qga_invalid_cmd(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *error; | 
 |     const gchar *class, *desc; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-invalid-cmd'}"); | 
 |     g_assert_nonnull(ret); | 
 |  | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |  | 
 |     g_assert_cmpstr(class, ==, "CommandNotFound"); | 
 |     g_assert_cmpint(strlen(desc), >, 0); | 
 | } | 
 |  | 
 | static void test_qga_info(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     const gchar *version; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-info'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     version = qdict_get_try_str(val, "version"); | 
 |     g_assert_cmpstr(version, ==, QEMU_VERSION); | 
 | } | 
 |  | 
 | static void test_qga_get_vcpus(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QList *list; | 
 |     const QListEntry *entry; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-vcpus'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     /* check there is at least a cpu */ | 
 |     list = qdict_get_qlist(ret, "return"); | 
 |     entry = qlist_first(list); | 
 |     g_assert(qdict_haskey(qobject_to(QDict, entry->value), "online")); | 
 |     g_assert(qdict_haskey(qobject_to(QDict, entry->value), "logical-id")); | 
 | } | 
 |  | 
 | static void test_qga_get_fsinfo(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QList *list; | 
 |     const QListEntry *entry; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-fsinfo'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     /* sanity-check the response if there are any filesystems */ | 
 |     list = qdict_get_qlist(ret, "return"); | 
 |     entry = qlist_first(list); | 
 |     if (entry) { | 
 |         g_assert(qdict_haskey(qobject_to(QDict, entry->value), "name")); | 
 |         g_assert(qdict_haskey(qobject_to(QDict, entry->value), "mountpoint")); | 
 |         g_assert(qdict_haskey(qobject_to(QDict, entry->value), "type")); | 
 |         g_assert(qdict_haskey(qobject_to(QDict, entry->value), "disk")); | 
 |     } | 
 | } | 
 |  | 
 | static void test_qga_get_memory_block_info(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     int64_t size; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-memory-block-info'}"); | 
 |     g_assert_nonnull(ret); | 
 |  | 
 |     /* some systems might not expose memory block info in sysfs */ | 
 |     if (!qdict_haskey(ret, "error")) { | 
 |         /* check there is at least some memory */ | 
 |         val = qdict_get_qdict(ret, "return"); | 
 |         size = qdict_get_int(val, "size"); | 
 |         g_assert_cmpint(size, >, 0); | 
 |     } | 
 | } | 
 |  | 
 | static void test_qga_get_memory_blocks(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QList *list; | 
 |     const QListEntry *entry; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-memory-blocks'}"); | 
 |     g_assert_nonnull(ret); | 
 |  | 
 |     /* some systems might not expose memory block info in sysfs */ | 
 |     if (!qdict_haskey(ret, "error")) { | 
 |         list = qdict_get_qlist(ret, "return"); | 
 |         entry = qlist_first(list); | 
 |         /* newer versions of qga may return empty list without error */ | 
 |         if (entry) { | 
 |             g_assert(qdict_haskey(qobject_to(QDict, entry->value), | 
 |                                   "phys-index")); | 
 |             g_assert(qdict_haskey(qobject_to(QDict, entry->value), "online")); | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | static void test_qga_network_get_interfaces(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QList *list; | 
 |     const QListEntry *entry; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-network-get-interfaces'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     /* check there is at least an interface */ | 
 |     list = qdict_get_qlist(ret, "return"); | 
 |     entry = qlist_first(list); | 
 |     g_assert(qdict_haskey(qobject_to(QDict, entry->value), "name")); | 
 | } | 
 |  | 
 | static void test_qga_file_ops(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     const unsigned char helloworld[] = "Hello World!\n"; | 
 |     const char *b64; | 
 |     gchar *path, *enc; | 
 |     unsigned char *dec; | 
 |     QDict *ret, *val; | 
 |     int64_t id, eof; | 
 |     gsize count; | 
 |     FILE *f; | 
 |     char tmp[100]; | 
 |  | 
 |     /* open */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," | 
 |                  " 'arguments': { 'path': 'foo', 'mode': 'w+' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     id = qdict_get_int(ret, "return"); | 
 |     qobject_unref(ret); | 
 |  | 
 |     enc = g_base64_encode(helloworld, sizeof(helloworld)); | 
 |     /* write */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-write'," | 
 |                  " 'arguments': { 'handle': %" PRId64 ", 'buf-b64': %s } }", | 
 |                  id, enc); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld)); | 
 |     g_assert_cmpint(eof, ==, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* flush */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-flush'," | 
 |                  " 'arguments': {'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* close */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-close'," | 
 |                  " 'arguments': {'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* check content */ | 
 |     path = g_build_filename(fixture->test_dir, "foo", NULL); | 
 |     f = fopen(path, "r"); | 
 |     g_free(path); | 
 |     g_assert_nonnull(f); | 
 |     count = fread(tmp, 1, sizeof(tmp), f); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld)); | 
 |     tmp[count] = 0; | 
 |     g_assert_cmpstr(tmp, ==, (char *)helloworld); | 
 |     fclose(f); | 
 |  | 
 |     /* open */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," | 
 |                  " 'arguments': { 'path': 'foo', 'mode': 'r' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     id = qdict_get_int(ret, "return"); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* read */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-read'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     b64 = qdict_get_str(val, "buf-b64"); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld)); | 
 |     g_assert(eof); | 
 |     g_assert_cmpstr(b64, ==, enc); | 
 |  | 
 |     qobject_unref(ret); | 
 |     g_free(enc); | 
 |  | 
 |     /* read eof */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-read'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     b64 = qdict_get_str(val, "buf-b64"); | 
 |     g_assert_cmpint(count, ==, 0); | 
 |     g_assert(eof); | 
 |     g_assert_cmpstr(b64, ==, ""); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* seek */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-seek'," | 
 |                  " 'arguments': { 'handle': %" PRId64 ", " | 
 |                  " 'offset': %d, 'whence': %s } }", | 
 |                  id, 6, "set"); | 
 |     qmp_assert_no_error(ret); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "position"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     g_assert_cmpint(count, ==, 6); | 
 |     g_assert(!eof); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* partial read */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-read'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     b64 = qdict_get_str(val, "buf-b64"); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld) - 6); | 
 |     g_assert(eof); | 
 |     dec = g_base64_decode(b64, &count); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld) - 6); | 
 |     g_assert_cmpmem(dec, count, helloworld + 6, sizeof(helloworld) - 6); | 
 |     g_free(dec); | 
 |  | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* close */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-close'," | 
 |                  " 'arguments': {'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     qobject_unref(ret); | 
 | } | 
 |  | 
 | static void test_qga_file_write_read(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     const unsigned char helloworld[] = "Hello World!\n"; | 
 |     const char *b64; | 
 |     gchar *enc; | 
 |     QDict *ret, *val; | 
 |     int64_t id, eof; | 
 |     gsize count; | 
 |  | 
 |     /* open */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," | 
 |                  " 'arguments': { 'path': 'foo', 'mode': 'w+' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     id = qdict_get_int(ret, "return"); | 
 |     qobject_unref(ret); | 
 |  | 
 |     enc = g_base64_encode(helloworld, sizeof(helloworld)); | 
 |     /* write */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-write'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "," | 
 |                  " 'buf-b64': %s } }", id, enc); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld)); | 
 |     g_assert_cmpint(eof, ==, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* read (check implicit flush) */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-read'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     b64 = qdict_get_str(val, "buf-b64"); | 
 |     g_assert_cmpint(count, ==, 0); | 
 |     g_assert(eof); | 
 |     g_assert_cmpstr(b64, ==, ""); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* seek to 0 */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-seek'," | 
 |                  " 'arguments': { 'handle': %" PRId64 ", " | 
 |                  " 'offset': %d, 'whence': %s } }", | 
 |                  id, 0, "set"); | 
 |     qmp_assert_no_error(ret); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "position"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     g_assert_cmpint(count, ==, 0); | 
 |     g_assert(!eof); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* read */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-read'," | 
 |                  " 'arguments': { 'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     count = qdict_get_int(val, "count"); | 
 |     eof = qdict_get_bool(val, "eof"); | 
 |     b64 = qdict_get_str(val, "buf-b64"); | 
 |     g_assert_cmpint(count, ==, sizeof(helloworld)); | 
 |     g_assert(eof); | 
 |     g_assert_cmpstr(b64, ==, enc); | 
 |     qobject_unref(ret); | 
 |     g_free(enc); | 
 |  | 
 |     /* close */ | 
 |     ret = qmp_fd(fixture->fd, | 
 |                  "{'execute': 'guest-file-close'," | 
 |                  " 'arguments': {'handle': %" PRId64 "} }", | 
 |                  id); | 
 |     qobject_unref(ret); | 
 | } | 
 |  | 
 | static void test_qga_get_time(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     int64_t time; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-time'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     time = qdict_get_int(ret, "return"); | 
 |     g_assert_cmpint(time, >, 0); | 
 | } | 
 |  | 
 | static void test_qga_blockedrpcs(gconstpointer data) | 
 | { | 
 |     TestFixture fix; | 
 |     QDict *ret, *error; | 
 |     const gchar *class, *desc; | 
 |  | 
 |     fixture_setup(&fix, "-b guest-ping,guest-get-time", NULL); | 
 |  | 
 |     /* check blocked RPCs */ | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-ping'}"); | 
 |     g_assert_nonnull(ret); | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |     g_assert_cmpstr(class, ==, "CommandNotFound"); | 
 |     g_assert_nonnull(g_strstr_len(desc, -1, "has been disabled")); | 
 |     qobject_unref(ret); | 
 |  | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-get-time'}"); | 
 |     g_assert_nonnull(ret); | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |     g_assert_cmpstr(class, ==, "CommandNotFound"); | 
 |     g_assert_nonnull(g_strstr_len(desc, -1, "has been disabled")); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* check something work */ | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-get-fsinfo'}"); | 
 |     qmp_assert_no_error(ret); | 
 |     qobject_unref(ret); | 
 |  | 
 |     fixture_tear_down(&fix, NULL); | 
 | } | 
 |  | 
 | static void test_qga_allowedrpcs(gconstpointer data) | 
 | { | 
 |     TestFixture fix; | 
 |     QDict *ret, *error; | 
 |     const gchar *class, *desc; | 
 |  | 
 |     fixture_setup(&fix, "-a guest-ping,guest-get-time", NULL); | 
 |  | 
 |     /* check allowed RPCs */ | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-ping'}"); | 
 |     qmp_assert_no_error(ret); | 
 |     qobject_unref(ret); | 
 |  | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-get-time'}"); | 
 |     qmp_assert_no_error(ret); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* check something else */ | 
 |     ret = qmp_fd(fix.fd, "{'execute': 'guest-get-fsinfo'}"); | 
 |     g_assert_nonnull(ret); | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     class = qdict_get_try_str(error, "class"); | 
 |     desc = qdict_get_try_str(error, "desc"); | 
 |     g_assert_cmpstr(class, ==, "CommandNotFound"); | 
 |     g_assert_nonnull(g_strstr_len(desc, -1, "has been disabled")); | 
 |     qobject_unref(ret); | 
 |  | 
 |     fixture_tear_down(&fix, NULL); | 
 | } | 
 |  | 
 | static void test_qga_config(gconstpointer data) | 
 | { | 
 |     GError *error = NULL; | 
 |     g_autofree char *out = NULL; | 
 |     g_autofree char *err = NULL; | 
 |     g_autofree char *cwd = NULL; | 
 |     g_autofree char *cmd = NULL; | 
 |     g_auto(GStrv) argv = NULL; | 
 |     g_auto(GStrv) strv = NULL; | 
 |     g_autoptr(GKeyFile) kf = NULL; | 
 |     char *str; | 
 |     char *env[2]; | 
 |     int status; | 
 |     gsize n; | 
 |  | 
 |     cwd = g_get_current_dir(); | 
 |     cmd = g_strdup_printf("%s%cqga%cqemu-ga -D", | 
 |                           cwd, G_DIR_SEPARATOR, G_DIR_SEPARATOR); | 
 |     g_shell_parse_argv(cmd, NULL, &argv, &error); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     env[0] = g_strdup_printf("QGA_CONF=tests%cdata%ctest-qga-config", | 
 |                              G_DIR_SEPARATOR, G_DIR_SEPARATOR); | 
 |     env[1] = NULL; | 
 |     g_spawn_sync(NULL, argv, env, 0, | 
 |                  NULL, NULL, &out, &err, &status, &error); | 
 |  | 
 |     g_assert_no_error(error); | 
 |     g_assert_cmpstr(err, ==, ""); | 
 |     g_assert_cmpint(status, ==, 0); | 
 |  | 
 |     kf = g_key_file_new(); | 
 |     g_key_file_load_from_data(kf, out, -1, G_KEY_FILE_NONE, &error); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     str = g_key_file_get_start_group(kf); | 
 |     g_assert_cmpstr(str, ==, "general"); | 
 |     g_free(str); | 
 |  | 
 |     g_assert_false(g_key_file_get_boolean(kf, "general", "daemon", &error)); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     str = g_key_file_get_string(kf, "general", "method", &error); | 
 |     g_assert_no_error(error); | 
 |     g_assert_cmpstr(str, ==, "virtio-serial"); | 
 |     g_free(str); | 
 |  | 
 |     str = g_key_file_get_string(kf, "general", "path", &error); | 
 |     g_assert_no_error(error); | 
 |     g_assert_cmpstr(str, ==, "/path/to/org.qemu.guest_agent.0"); | 
 |     g_free(str); | 
 |  | 
 |     str = g_key_file_get_string(kf, "general", "pidfile", &error); | 
 |     g_assert_no_error(error); | 
 |     g_assert_cmpstr(str, ==, "/var/foo/qemu-ga.pid"); | 
 |     g_free(str); | 
 |  | 
 |     str = g_key_file_get_string(kf, "general", "statedir", &error); | 
 |     g_assert_no_error(error); | 
 |     g_assert_cmpstr(str, ==, "/var/state"); | 
 |     g_free(str); | 
 |  | 
 |     g_assert_true(g_key_file_get_boolean(kf, "general", "verbose", &error)); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     strv = g_key_file_get_string_list(kf, "general", "block-rpcs", &n, &error); | 
 |     g_assert_cmpint(n, ==, 2); | 
 |     g_assert_true(g_strv_contains((const char * const *)strv, | 
 |                                   "guest-ping")); | 
 |     g_assert_true(g_strv_contains((const char * const *)strv, | 
 |                                   "guest-get-time")); | 
 |     g_assert_no_error(error); | 
 |  | 
 |     g_free(env[0]); | 
 | } | 
 |  | 
 | static void test_qga_fsfreeze_status(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     const gchar *status; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-fsfreeze-status'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     status = qdict_get_try_str(ret, "return"); | 
 |     g_assert_cmpstr(status, ==, "thawed"); | 
 | } | 
 |  | 
 | static QDict *wait_for_guest_exec_completion(int fd, int64_t pid) | 
 | { | 
 |     QDict *ret = NULL; | 
 |     int64_t now; | 
 |     bool exited; | 
 |     QDict *val; | 
 |  | 
 |     now = g_get_monotonic_time(); | 
 |     do { | 
 |         ret = qmp_fd(fd, | 
 |                      "{'execute': 'guest-exec-status'," | 
 |                      " 'arguments': { 'pid': %" PRId64 " } }", pid); | 
 |         g_assert_nonnull(ret); | 
 |         val = qdict_get_qdict(ret, "return"); | 
 |         exited = qdict_get_bool(val, "exited"); | 
 |         if (!exited) { | 
 |             qobject_unref(ret); | 
 |         } | 
 |     } while (!exited && | 
 |              g_get_monotonic_time() < now + 5 * G_TIME_SPAN_SECOND); | 
 |     g_assert(exited); | 
 |  | 
 |     return ret; | 
 | } | 
 |  | 
 | static void test_qga_guest_exec(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     const gchar *out; | 
 |     g_autofree guchar *decoded = NULL; | 
 |     int64_t pid, exitcode; | 
 |     gsize len; | 
 |  | 
 |     /* exec 'echo foo bar' */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" | 
 |                  " 'path': 'echo', 'arg': [ '-n', '\" test_str \"' ]," | 
 |                  " 'capture-output': true } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     pid = qdict_get_int(val, "pid"); | 
 |     g_assert_cmpint(pid, >, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     ret = wait_for_guest_exec_completion(fixture->fd, pid); | 
 |  | 
 |     /* check stdout */ | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     exitcode = qdict_get_int(val, "exitcode"); | 
 |     g_assert_cmpint(exitcode, ==, 0); | 
 |     out = qdict_get_str(val, "out-data"); | 
 |     decoded = g_base64_decode(out, &len); | 
 |     g_assert_cmpint(len, ==, 12); | 
 |     g_assert_cmpstr((char *)decoded, ==, "\" test_str \""); | 
 | } | 
 |  | 
 | #if defined(G_OS_WIN32) | 
 | static void test_qga_guest_exec_separated(gconstpointer fix) | 
 | { | 
 | } | 
 | static void test_qga_guest_exec_merged(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     const gchar *class, *desc; | 
 |     g_autofree guchar *decoded = NULL; | 
 |  | 
 |     /* exec 'echo foo bar' */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" | 
 |                  " 'path': 'echo'," | 
 |                  " 'arg': [ 'execution never reaches here' ]," | 
 |                  " 'capture-output': 'merged' } }"); | 
 |  | 
 |     g_assert_nonnull(ret); | 
 |     val = qdict_get_qdict(ret, "error"); | 
 |     g_assert_nonnull(val); | 
 |     class = qdict_get_str(val, "class"); | 
 |     desc = qdict_get_str(val, "desc"); | 
 |     g_assert_cmpstr(class, ==, "GenericError"); | 
 |     g_assert_cmpint(strlen(desc), >, 0); | 
 | } | 
 | #else | 
 | static void test_qga_guest_exec_separated(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     const gchar *out, *err; | 
 |     g_autofree guchar *out_decoded = NULL; | 
 |     g_autofree guchar *err_decoded = NULL; | 
 |     int64_t pid, exitcode; | 
 |     gsize len; | 
 |  | 
 |     /* exec 'echo foo bar' */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" | 
 |                  " 'path': 'bash'," | 
 |                  " 'arg': [ '-c', 'for i in $(seq 4); do if (( $i %% 2 )); then echo stdout; else echo stderr 1>&2; fi; done;' ]," | 
 |                  " 'capture-output': 'separated' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     pid = qdict_get_int(val, "pid"); | 
 |     g_assert_cmpint(pid, >, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     ret = wait_for_guest_exec_completion(fixture->fd, pid); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     exitcode = qdict_get_int(val, "exitcode"); | 
 |     g_assert_cmpint(exitcode, ==, 0); | 
 |  | 
 |     /* check stdout */ | 
 |     out = qdict_get_str(val, "out-data"); | 
 |     out_decoded = g_base64_decode(out, &len); | 
 |     g_assert_cmpint(len, ==, 14); | 
 |     g_assert_cmpstr((char *)out_decoded, ==, "stdout\nstdout\n"); | 
 |  | 
 |     /* check stderr */ | 
 |     err = qdict_get_try_str(val, "err-data"); | 
 |     err_decoded = g_base64_decode(err, &len); | 
 |     g_assert_cmpint(len, ==, 14); | 
 |     g_assert_cmpstr((char *)err_decoded, ==, "stderr\nstderr\n"); | 
 | } | 
 |  | 
 | static void test_qga_guest_exec_merged(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |     const gchar *out, *err; | 
 |     g_autofree guchar *decoded = NULL; | 
 |     int64_t pid, exitcode; | 
 |     gsize len; | 
 |  | 
 |     /* exec 'echo foo bar' */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" | 
 |                  " 'path': 'bash'," | 
 |                  " 'arg': [ '-c', 'for i in $(seq 4); do if (( $i %% 2 )); then echo stdout; else echo stderr 1>&2; fi; done;' ]," | 
 |                  " 'capture-output': 'merged' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     pid = qdict_get_int(val, "pid"); | 
 |     g_assert_cmpint(pid, >, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     ret = wait_for_guest_exec_completion(fixture->fd, pid); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     exitcode = qdict_get_int(val, "exitcode"); | 
 |     g_assert_cmpint(exitcode, ==, 0); | 
 |  | 
 |     /* check stdout */ | 
 |     out = qdict_get_str(val, "out-data"); | 
 |     decoded = g_base64_decode(out, &len); | 
 |     g_assert_cmpint(len, ==, 28); | 
 |     g_assert_cmpstr((char *)decoded, ==, "stdout\nstderr\nstdout\nstderr\n"); | 
 |  | 
 |     /* check stderr */ | 
 |     err = qdict_get_try_str(val, "err-data"); | 
 |     g_assert_null(err); | 
 | } | 
 | #endif | 
 |  | 
 | static void test_qga_guest_exec_invalid(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *error; | 
 |     const gchar *class, *desc; | 
 |  | 
 |     /* invalid command */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" | 
 |                  " 'path': '/bin/invalid-cmd42' } }"); | 
 |     g_assert_nonnull(ret); | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     g_assert_nonnull(error); | 
 |     class = qdict_get_str(error, "class"); | 
 |     desc = qdict_get_str(error, "desc"); | 
 |     g_assert_cmpstr(class, ==, "GenericError"); | 
 |     g_assert_cmpint(strlen(desc), >, 0); | 
 |     qobject_unref(ret); | 
 |  | 
 |     /* invalid pid */ | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec-status'," | 
 |                  " 'arguments': { 'pid': 0 } }"); | 
 |     g_assert_nonnull(ret); | 
 |     error = qdict_get_qdict(ret, "error"); | 
 |     g_assert_nonnull(error); | 
 |     class = qdict_get_str(error, "class"); | 
 |     desc = qdict_get_str(error, "desc"); | 
 |     g_assert_cmpstr(class, ==, "GenericError"); | 
 |     g_assert_cmpint(strlen(desc), >, 0); | 
 | } | 
 |  | 
 | static void test_qga_guest_get_host_name(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-host-name'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     g_assert(qdict_haskey(val, "host-name")); | 
 | } | 
 |  | 
 | static void test_qga_guest_get_timezone(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QDict *val; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-timezone'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     /* Make sure there's at least offset */ | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |     g_assert(qdict_haskey(val, "offset")); | 
 | } | 
 |  | 
 | static void test_qga_guest_get_users(gconstpointer fix) | 
 | { | 
 |     const TestFixture *fixture = fix; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     QList *val; | 
 |  | 
 |     ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-users'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     /* There is not much to test here */ | 
 |     val = qdict_get_qlist(ret, "return"); | 
 |     g_assert_nonnull(val); | 
 | } | 
 |  | 
 | static void test_qga_guest_get_osinfo(gconstpointer data) | 
 | { | 
 |     TestFixture fixture; | 
 |     const gchar *str; | 
 |     g_autoptr(QDict) ret = NULL; | 
 |     char *env[2]; | 
 |     QDict *val; | 
 |  | 
 |     env[0] = g_strdup_printf( | 
 |         "QGA_OS_RELEASE=%s%c..%cdata%ctest-qga-os-release", | 
 |         g_test_get_dir(G_TEST_DIST), G_DIR_SEPARATOR, G_DIR_SEPARATOR, G_DIR_SEPARATOR); | 
 |     env[1] = NULL; | 
 |     fixture_setup(&fixture, NULL, env); | 
 |  | 
 |     ret = qmp_fd(fixture.fd, "{'execute': 'guest-get-osinfo'}"); | 
 |     g_assert_nonnull(ret); | 
 |     qmp_assert_no_error(ret); | 
 |  | 
 |     val = qdict_get_qdict(ret, "return"); | 
 |  | 
 |     str = qdict_get_try_str(val, "id"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "qemu-ga-test"); | 
 |  | 
 |     str = qdict_get_try_str(val, "name"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "QEMU-GA"); | 
 |  | 
 |     str = qdict_get_try_str(val, "pretty-name"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "QEMU Guest Agent test"); | 
 |  | 
 |     str = qdict_get_try_str(val, "version"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "Test 1"); | 
 |  | 
 |     str = qdict_get_try_str(val, "version-id"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "1"); | 
 |  | 
 |     str = qdict_get_try_str(val, "variant"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "Unit test \"'$`\\ and \\\\ etc."); | 
 |  | 
 |     str = qdict_get_try_str(val, "variant-id"); | 
 |     g_assert_nonnull(str); | 
 |     g_assert_cmpstr(str, ==, "unit-test"); | 
 |  | 
 |     g_free(env[0]); | 
 |     fixture_tear_down(&fixture, NULL); | 
 | } | 
 |  | 
 | int main(int argc, char **argv) | 
 | { | 
 |     TestFixture fix; | 
 |     int ret; | 
 |  | 
 | #ifdef QEMU_SANITIZE_THREAD | 
 |     { | 
 |         g_test_skip("tsan enabled, https://github.com/google/sanitizers/issues/1116"); | 
 |         return 0; | 
 |     } | 
 | #endif | 
 |  | 
 |     setlocale (LC_ALL, ""); | 
 |     g_test_init(&argc, &argv, NULL); | 
 |     fixture_setup(&fix, NULL, NULL); | 
 |  | 
 |     g_test_add_data_func("/qga/sync-delimited", &fix, test_qga_sync_delimited); | 
 |     g_test_add_data_func("/qga/sync", &fix, test_qga_sync); | 
 |     g_test_add_data_func("/qga/ping", &fix, test_qga_ping); | 
 |     g_test_add_data_func("/qga/info", &fix, test_qga_info); | 
 |     g_test_add_data_func("/qga/network-get-interfaces", &fix, | 
 |                          test_qga_network_get_interfaces); | 
 |     if (!access("/sys/devices/system/cpu/cpu0", F_OK)) { | 
 |         g_test_add_data_func("/qga/get-vcpus", &fix, test_qga_get_vcpus); | 
 |     } | 
 |     g_test_add_data_func("/qga/get-fsinfo", &fix, test_qga_get_fsinfo); | 
 |     g_test_add_data_func("/qga/get-memory-block-info", &fix, | 
 |                          test_qga_get_memory_block_info); | 
 |     g_test_add_data_func("/qga/get-memory-blocks", &fix, | 
 |                          test_qga_get_memory_blocks); | 
 |     g_test_add_data_func("/qga/file-ops", &fix, test_qga_file_ops); | 
 |     g_test_add_data_func("/qga/file-write-read", &fix, test_qga_file_write_read); | 
 |     g_test_add_data_func("/qga/get-time", &fix, test_qga_get_time); | 
 |     g_test_add_data_func("/qga/id", &fix, test_qga_id); | 
 |     g_test_add_data_func("/qga/invalid-oob", &fix, test_qga_invalid_oob); | 
 |     g_test_add_data_func("/qga/invalid-cmd", &fix, test_qga_invalid_cmd); | 
 |     g_test_add_data_func("/qga/invalid-args", &fix, test_qga_invalid_args); | 
 |     g_test_add_data_func("/qga/fsfreeze-status", &fix, | 
 |                          test_qga_fsfreeze_status); | 
 |  | 
 |     g_test_add_data_func("/qga/blockedrpcs", NULL, test_qga_blockedrpcs); | 
 |     g_test_add_data_func("/qga/allowedrpcs", NULL, test_qga_allowedrpcs); | 
 |     g_test_add_data_func("/qga/config", NULL, test_qga_config); | 
 |     g_test_add_data_func("/qga/guest-exec", &fix, test_qga_guest_exec); | 
 |     g_test_add_data_func("/qga/guest-exec-separated", &fix, | 
 |                          test_qga_guest_exec_separated); | 
 |     g_test_add_data_func("/qga/guest-exec-merged", &fix, | 
 |                          test_qga_guest_exec_merged); | 
 |     g_test_add_data_func("/qga/guest-exec-invalid", &fix, | 
 |                          test_qga_guest_exec_invalid); | 
 |     g_test_add_data_func("/qga/guest-get-osinfo", &fix, | 
 |                          test_qga_guest_get_osinfo); | 
 |     g_test_add_data_func("/qga/guest-get-host-name", &fix, | 
 |                          test_qga_guest_get_host_name); | 
 |     g_test_add_data_func("/qga/guest-get-timezone", &fix, | 
 |                          test_qga_guest_get_timezone); | 
 |     g_test_add_data_func("/qga/guest-get-users", &fix, | 
 |                          test_qga_guest_get_users); | 
 |  | 
 |     ret = g_test_run(); | 
 |  | 
 |     fixture_tear_down(&fix, NULL); | 
 |  | 
 |     return ret; | 
 | } |