2 * QOS-assisted fuzzing helpers
4 * Copyright (c) 2018 Emanuele Giuseppe Esposito <e.emanuelegiuseppe@gmail.com>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License version 2.1 as published by the Free Software Foundation.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, see <http://www.gnu.org/licenses/>
19 #include "qemu/osdep.h"
20 #include "qemu/units.h"
21 #include "qapi/error.h"
22 #include "qemu-common.h"
23 #include "exec/memory.h"
24 #include "qemu/main-loop.h"
26 #include "tests/qtest/libqos/libqtest.h"
27 #include "tests/qtest/libqos/malloc.h"
28 #include "tests/qtest/libqos/qgraph.h"
29 #include "tests/qtest/libqos/qgraph_internal.h"
30 #include "tests/qtest/libqos/qos_external.h"
35 #include "qapi/qapi-commands-machine.h"
36 #include "qapi/qapi-commands-qom.h"
40 QGuestAllocator
*fuzz_qos_alloc
;
42 static const char *fuzz_target_name
;
43 static char **fuzz_path_vec
;
45 static void qos_set_machines_devices_available(void)
47 MachineInfoList
*mach_info
;
48 ObjectTypeInfoList
*type_info
;
50 mach_info
= qmp_query_machines(&error_abort
);
51 machines_apply_to_node(mach_info
);
52 qapi_free_MachineInfoList(mach_info
);
54 type_info
= qmp_qom_list_types(true, "device", true, true,
56 types_apply_to_node(type_info
);
57 qapi_free_ObjectTypeInfoList(type_info
);
60 static char **current_path
;
62 void *qos_allocate_objects(QTestState
*qts
, QGuestAllocator
**p_alloc
)
64 return allocate_objects(qts
, current_path
+ 1, p_alloc
);
67 static GString
*qos_build_main_args(void)
69 char **path
= fuzz_path_vec
;
70 QOSGraphNode
*test_node
;
75 fprintf(stderr
, "QOS Path not found\n");
80 cmd_line
= g_string_new(path
[0]);
82 test_node
= qos_graph_get_node(path
[(g_strv_length(path
) - 1)]);
83 test_arg
= test_node
->u
.test
.arg
;
84 if (test_node
->u
.test
.before
) {
85 test_arg
= test_node
->u
.test
.before(cmd_line
, test_arg
);
87 /* Prepend the arguments that we need */
88 g_string_prepend(cmd_line
,
89 TARGET_NAME
" -display none -machine accel=qtest -m 64 ");
94 * This function is largely a copy of qos-test.c:walk_path. Since walk_path
95 * is itself a callback, its a little annoying to add another argument/layer of
98 static void walk_path(QOSGraphNode
*orig_path
, int len
)
104 * etype set to QEDGE_CONSUMED_BY so that machine can add to the command
107 QOSEdgeType etype
= QEDGE_CONSUMED_BY
;
109 /* twice QOS_PATH_MAX_ELEMENT_SIZE since each edge can have its arg */
110 char **path_vec
= g_new0(char *, (QOS_PATH_MAX_ELEMENT_SIZE
* 2));
111 int path_vec_size
= 0;
113 char *after_cmd
, *before_cmd
, *after_device
;
114 GString
*after_device_str
= g_string_new("");
115 char *node_name
= orig_path
->name
, *path_str
;
117 GString
*cmd_line
= g_string_new("");
118 GString
*cmd_line2
= g_string_new("");
120 path
= qos_graph_get_node(node_name
); /* root */
121 node_name
= qos_graph_edge_get_dest(path
->path_edge
); /* machine name */
123 path_vec
[path_vec_size
++] = node_name
;
124 path_vec
[path_vec_size
++] = qos_get_machine_type(node_name
);
127 path
= qos_graph_get_node(node_name
);
128 if (!path
->path_edge
) {
132 node_name
= qos_graph_edge_get_dest(path
->path_edge
);
134 /* append node command line + previous edge command line */
135 if (path
->command_line
&& etype
== QEDGE_CONSUMED_BY
) {
136 g_string_append(cmd_line
, path
->command_line
);
137 g_string_append(cmd_line
, after_device_str
->str
);
138 g_string_truncate(after_device_str
, 0);
141 path_vec
[path_vec_size
++] = qos_graph_edge_get_name(path
->path_edge
);
142 /* detect if edge has command line args */
143 after_cmd
= qos_graph_edge_get_after_cmd_line(path
->path_edge
);
144 after_device
= qos_graph_edge_get_extra_device_opts(path
->path_edge
);
145 before_cmd
= qos_graph_edge_get_before_cmd_line(path
->path_edge
);
146 edge
= qos_graph_get_edge(path
->name
, node_name
);
147 etype
= qos_graph_edge_get_type(edge
);
150 g_string_append(cmd_line
, before_cmd
);
153 g_string_append(cmd_line2
, after_cmd
);
156 g_string_append(after_device_str
, after_device
);
160 path_vec
[path_vec_size
++] = NULL
;
161 g_string_append(cmd_line
, after_device_str
->str
);
162 g_string_free(after_device_str
, true);
164 g_string_append(cmd_line
, cmd_line2
->str
);
165 g_string_free(cmd_line2
, true);
168 * here position 0 has <arch>/<machine>, position 1 has <machine>.
169 * The path must not have the <arch>, qtest_add_data_func adds it.
171 path_str
= g_strjoinv("/", path_vec
+ 1);
173 /* Check that this is the test we care about: */
174 char *test_name
= strrchr(path_str
, '/') + 1;
175 if (strcmp(test_name
, fuzz_target_name
) == 0) {
177 * put arch/machine in position 1 so run_one_test can do its work
178 * and add the command line at position 0.
180 path_vec
[1] = path_vec
[0];
181 path_vec
[0] = g_string_free(cmd_line
, false);
183 fuzz_path_vec
= path_vec
;
191 static GString
*qos_get_cmdline(FuzzTarget
*t
)
194 * Set a global variable that we use to identify the qos_path for our
197 fuzz_target_name
= t
->name
;
198 qos_set_machines_devices_available();
199 qos_graph_foreach_test_path(walk_path
);
200 return qos_build_main_args();
203 void fuzz_add_qos_target(
204 FuzzTarget
*fuzz_opts
,
205 const char *interface
,
206 QOSGraphTestOptions
*opts
209 qos_add_test(fuzz_opts
->name
, interface
, NULL
, opts
);
210 fuzz_opts
->get_init_cmdline
= qos_get_cmdline
;
211 fuzz_add_target(fuzz_opts
);
214 void qos_init_path(QTestState
*s
)
216 fuzz_qos_obj
= qos_allocate_objects(s
, &fuzz_qos_alloc
);