xref: /openbmc/qemu/tests/qtest/libqos/qgraph.c (revision 460056dbe6b2df363de6d8c2a9c9ba862653d6bb)
1 /*
2  * libqos driver framework
3  *
4  * Copyright (c) 2018 Emanuele Giuseppe Esposito <e.emanuelegiuseppe@gmail.com>
5  *
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.
9  *
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.
14  *
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/>
17  */
18 
19 #include "qemu/osdep.h"
20 #include "libqtest.h"
21 #include "qemu/queue.h"
22 #include "qgraph_internal.h"
23 #include "qgraph.h"
24 
25 #define QGRAPH_PRINT_DEBUG 0
26 #define QOS_ROOT ""
27 typedef struct QOSStackElement QOSStackElement;
28 
29 /* Graph Edge.*/
30 struct QOSGraphEdge {
31     QOSEdgeType type;
32     char *dest;
33     void *arg;                /* just for QEDGE_CONTAINS
34                                * and QEDGE_CONSUMED_BY */
35     char *extra_device_opts;  /* added to -device option, "," is
36                                * automatically added
37                                */
38     char *before_cmd_line;    /* added before node cmd_line */
39     char *after_cmd_line;     /* added after -device options */
40     char *edge_name;          /* used by QEDGE_CONTAINS */
41     QSLIST_ENTRY(QOSGraphEdge) edge_list;
42 };
43 
44 typedef QSLIST_HEAD(, QOSGraphEdge) QOSGraphEdgeList;
45 
46 /**
47  * Stack used to keep track of the discovered path when using
48  * the DFS algorithm
49  */
50 struct QOSStackElement {
51     QOSGraphNode *node;
52     QOSStackElement *parent;
53     QOSGraphEdge *parent_edge;
54     int length;
55 };
56 
57 /* Each enty in these hash table will consist of <string, node/edge> pair. */
58 static GHashTable *edge_table;
59 static GHashTable *node_table;
60 
61 /* stack used by the DFS algorithm to store the path from machine to test */
62 static QOSStackElement qos_node_stack[QOS_PATH_MAX_ELEMENT_SIZE];
63 static int qos_node_tos;
64 
65 /**
66  * add_edge(): creates an edge of type @type
67  * from @source to @dest node, and inserts it in the
68  * edges hash table
69  *
70  * Nodes @source and @dest do not necessarily need to exist.
71  * Possibility to add also options (see #QOSGraphEdgeOptions)
72  * edge->edge_name is used as identifier for get_device relationships,
73  * so by default is equal to @dest.
74  */
75 static void add_edge(const char *source, const char *dest,
76                      QOSEdgeType type, QOSGraphEdgeOptions *opts)
77 {
78     char *key;
79     QOSGraphEdgeList *list = g_hash_table_lookup(edge_table, source);
80     QOSGraphEdgeOptions def_opts = { };
81 
82     if (!list) {
83         list = g_new0(QOSGraphEdgeList, 1);
84         key = g_strdup(source);
85         g_hash_table_insert(edge_table, key, list);
86     }
87 
88     if (!opts) {
89         opts = &def_opts;
90     }
91 
92     QOSGraphEdge *edge = g_new0(QOSGraphEdge, 1);
93     edge->type = type;
94     edge->dest = g_strdup(dest);
95     edge->edge_name = g_strdup(opts->edge_name ?: dest);
96     edge->arg = g_memdup2(opts->arg, opts->size_arg);
97 
98     edge->before_cmd_line =
99         opts->before_cmd_line ? g_strconcat(" ", opts->before_cmd_line, NULL) : NULL;
100     edge->extra_device_opts =
101         opts->extra_device_opts ? g_strconcat(",", opts->extra_device_opts, NULL) : NULL;
102     edge->after_cmd_line =
103         opts->after_cmd_line ? g_strconcat(" ", opts->after_cmd_line, NULL) : NULL;
104 
105     QSLIST_INSERT_HEAD(list, edge, edge_list);
106 }
107 
108 /* destroy_edges(): frees all edges inside a given @list */
109 static void destroy_edges(void *list)
110 {
111     QOSGraphEdge *temp;
112     QOSGraphEdgeList *elist = list;
113 
114     while (!QSLIST_EMPTY(elist)) {
115         temp = QSLIST_FIRST(elist);
116         QSLIST_REMOVE_HEAD(elist, edge_list);
117         g_free(temp->dest);
118         g_free(temp->before_cmd_line);
119         g_free(temp->after_cmd_line);
120         g_free(temp->extra_device_opts);
121         g_free(temp->edge_name);
122         g_free(temp->arg);
123         g_free(temp);
124     }
125     g_free(elist);
126 }
127 
128 /**
129  * create_node(): creates a node @name of type @type
130  * and inserts it to the nodes hash table.
131  * By default, node is not available.
132  */
133 static QOSGraphNode *create_node(const char *name, QOSNodeType type)
134 {
135     if (g_hash_table_lookup(node_table, name)) {
136         g_printerr("Node %s already created\n", name);
137         abort();
138     }
139 
140     QOSGraphNode *node = g_new0(QOSGraphNode, 1);
141     node->type = type;
142     node->available = false;
143     node->name = g_strdup(name);
144     g_hash_table_insert(node_table, node->name, node);
145     return node;
146 }
147 
148 /**
149  * destroy_node(): frees a node @val from the nodes hash table.
150  * Note that node->name is not free'd since it will represent the
151  * hash table key
152  */
153 static void destroy_node(void *val)
154 {
155     QOSGraphNode *node = val;
156     g_free(node->qemu_name);
157     g_free(node->command_line);
158     g_free(node);
159 }
160 
161 /**
162  * destroy_string(): frees @key from the nodes hash table.
163  * Actually frees the node->name
164  */
165 static void destroy_string(void *key)
166 {
167     g_free(key);
168 }
169 
170 /**
171  * search_node(): search for a node @key in the nodes hash table
172  * Returns the QOSGraphNode if found, #NULL otherwise
173  */
174 static QOSGraphNode *search_node(const char *key)
175 {
176     return g_hash_table_lookup(node_table, key);
177 }
178 
179 /**
180  * get_edgelist(): returns the edge list (value) assigned to
181  * the @key in the edge hash table.
182  * This list will contain all edges with source equal to @key
183  *
184  * Returns: on success: the %QOSGraphEdgeList
185  *          otherwise: abort()
186  */
187 static QOSGraphEdgeList *get_edgelist(const char *key)
188 {
189     return g_hash_table_lookup(edge_table, key);
190 }
191 
192 /**
193  * search_list_edges(): search for an edge with destination @dest
194  * in the given @edgelist.
195  *
196  * Returns: on success: the %QOSGraphEdge
197  *          otherwise: #NULL
198  */
199 static QOSGraphEdge *search_list_edges(QOSGraphEdgeList *edgelist,
200                                        const char *dest)
201 {
202     QOSGraphEdge *tmp, *next;
203     if (!edgelist) {
204         return NULL;
205     }
206     QSLIST_FOREACH_SAFE(tmp, edgelist, edge_list, next) {
207         if (g_strcmp0(tmp->dest, dest) == 0) {
208             break;
209         }
210     }
211     return tmp;
212 }
213 
214 /**
215  * search_machine(): search for a machine @name in the node hash
216  * table. A machine is the child of the root node.
217  * This function forces the research in the childs of the root,
218  * to check the node is a proper machine
219  *
220  * Returns: on success: the %QOSGraphNode
221  *          otherwise: #NULL
222  */
223 static QOSGraphNode *search_machine(const char *name)
224 {
225     QOSGraphNode *n;
226     QOSGraphEdgeList *root_list = get_edgelist(QOS_ROOT);
227     QOSGraphEdge *e = search_list_edges(root_list, name);
228     if (!e) {
229         return NULL;
230     }
231     n = search_node(e->dest);
232     if (n->type == QNODE_MACHINE) {
233         return n;
234     }
235     return NULL;
236 }
237 
238 /**
239  * create_interface(): checks if there is already
240  * a node @node in the node hash table, if not
241  * creates a node @node of type #QNODE_INTERFACE
242  * and inserts it. If there is one, check it's
243  * a #QNODE_INTERFACE and abort() if it's not.
244  */
245 static void create_interface(const char *node)
246 {
247     QOSGraphNode *interface;
248     interface = search_node(node);
249     if (!interface) {
250         create_node(node, QNODE_INTERFACE);
251     } else if (interface->type != QNODE_INTERFACE) {
252         fprintf(stderr, "Error: Node %s is not an interface\n", node);
253         abort();
254     }
255 }
256 
257 /**
258  * build_machine_cmd_line(): builds the command line for the machine
259  * @node. The node name must be a valid qemu identifier, since it
260  * will be used to build the command line.
261  *
262  * It is also possible to pass an optional @args that will be
263  * concatenated to the command line.
264  *
265  * For machines, prepend -M to the machine name. ", @rgs" is added
266  * after the -M <machine> command.
267  */
268 static void build_machine_cmd_line(QOSGraphNode *node, const char *args)
269 {
270     char *machine = qos_get_machine_type(node->name);
271     if (args) {
272         node->command_line = g_strconcat("-M ", machine, ",", args, NULL);
273     } else {
274         node->command_line = g_strconcat("-M ", machine, " ", NULL);
275     }
276 }
277 
278 /**
279  * build_driver_cmd_line(): builds the command line for the driver
280  * @node. The node name must be a valid qemu identifier, since it
281  * will be used to build the command line.
282  *
283  * Driver do not need additional command line, since it will be
284  * provided by the edge options.
285  *
286  * For drivers, prepend -device to the node name.
287  */
288 static void build_driver_cmd_line(QOSGraphNode *node)
289 {
290     const char *name = node->qemu_name ?: node->name;
291     node->command_line = g_strconcat(" -device ", name, NULL);
292 }
293 
294 /* qos_print_cb(): callback prints all path found by the DFS algorithm. */
295 static void qos_print_cb(QOSGraphNode *path, int length)
296 {
297     #if QGRAPH_PRINT_DEBUG
298         printf("%d elements\n", length);
299 
300         if (!path) {
301             return;
302         }
303 
304         while (path->path_edge) {
305             printf("%s ", path->name);
306             switch (path->path_edge->type) {
307             case QEDGE_PRODUCES:
308                 printf("--PRODUCES--> ");
309                 break;
310             case QEDGE_CONSUMED_BY:
311                 printf("--CONSUMED_BY--> ");
312                 break;
313             case QEDGE_CONTAINS:
314                 printf("--CONTAINS--> ");
315                 break;
316             }
317             path = search_node(path->path_edge->dest);
318         }
319 
320         printf("%s\n\n", path->name);
321     #endif
322 }
323 
324 /* qos_push(): push a node @el and edge @e in the qos_node_stack */
325 static void qos_push(QOSGraphNode *el, QOSStackElement *parent,
326                      QOSGraphEdge *e)
327 {
328     int len = 0; /* root is not counted */
329     if (qos_node_tos == QOS_PATH_MAX_ELEMENT_SIZE) {
330         g_printerr("QOSStack: full stack, cannot push");
331         abort();
332     }
333 
334     if (parent) {
335         len = parent->length + 1;
336     }
337     qos_node_stack[qos_node_tos++] = (QOSStackElement) {
338         .node = el,
339         .parent = parent,
340         .parent_edge = e,
341         .length = len,
342     };
343 }
344 
345 /* qos_tos(): returns the top of stack, without popping */
346 static QOSStackElement *qos_tos(void)
347 {
348     return &qos_node_stack[qos_node_tos - 1];
349 }
350 
351 /* qos_pop(): pops an element from the tos, setting it unvisited*/
352 static QOSStackElement *qos_pop(void)
353 {
354     if (qos_node_tos == 0) {
355         g_printerr("QOSStack: empty stack, cannot pop");
356         abort();
357     }
358     QOSStackElement *e = qos_tos();
359     e->node->visited = false;
360     qos_node_tos--;
361     return e;
362 }
363 
364 /**
365  * qos_reverse_path(): reverses the found path, going from
366  * test-to-machine to machine-to-test
367  */
368 static QOSGraphNode *qos_reverse_path(QOSStackElement *el)
369 {
370     if (!el) {
371         return NULL;
372     }
373 
374     el->node->path_edge = NULL;
375 
376     while (el->parent) {
377         el->parent->node->path_edge = el->parent_edge;
378         el = el->parent;
379     }
380 
381     return el->node;
382 }
383 
384 /**
385  * qos_traverse_graph(): graph-walking algorithm, using Depth First Search it
386  * starts from the root @machine and walks all possible path until it
387  * reaches a test node.
388  * At that point, it reverses the path found and invokes the @callback.
389  *
390  * Being Depth First Search, time complexity is O(|V| + |E|), while
391  * space is O(|V|). In this case, the maximum stack size is set by
392  * QOS_PATH_MAX_ELEMENT_SIZE.
393  */
394 static void qos_traverse_graph(QOSGraphNode *root, QOSTestCallback callback)
395 {
396     QOSGraphNode *v, *dest_node, *path;
397     QOSStackElement *s_el;
398     QOSGraphEdge *e, *next;
399     QOSGraphEdgeList *list;
400 
401     qos_push(root, NULL, NULL);
402 
403     while (qos_node_tos > 0) {
404         s_el = qos_tos();
405         v = s_el->node;
406         if (v->visited) {
407             qos_pop();
408             continue;
409         }
410         v->visited = true;
411         list = get_edgelist(v->name);
412         if (!list) {
413             qos_pop();
414             if (v->type == QNODE_TEST) {
415                 v->visited = false;
416                 path = qos_reverse_path(s_el);
417                 callback(path, s_el->length);
418             }
419         } else {
420             QSLIST_FOREACH_SAFE(e, list, edge_list, next) {
421                 dest_node = search_node(e->dest);
422 
423                 if (!dest_node) {
424                     fprintf(stderr, "node %s in %s -> %s does not exist\n",
425                             e->dest, v->name, e->dest);
426                     abort();
427                 }
428 
429                 if (!dest_node->visited && dest_node->available) {
430                     qos_push(dest_node, s_el, e);
431                 }
432             }
433         }
434     }
435 }
436 
437 /* QGRAPH API*/
438 
439 QOSGraphNode *qos_graph_get_node(const char *key)
440 {
441     return search_node(key);
442 }
443 
444 bool qos_graph_has_node(const char *node)
445 {
446     QOSGraphNode *n = search_node(node);
447     return n != NULL;
448 }
449 
450 QOSNodeType qos_graph_get_node_type(const char *node)
451 {
452     QOSGraphNode *n = search_node(node);
453     if (n) {
454         return n->type;
455     }
456     return -1;
457 }
458 
459 bool qos_graph_get_node_availability(const char *node)
460 {
461     QOSGraphNode *n = search_node(node);
462     if (n) {
463         return n->available;
464     }
465     return false;
466 }
467 
468 QOSGraphEdge *qos_graph_get_edge(const char *node, const char *dest)
469 {
470     QOSGraphEdgeList *list = get_edgelist(node);
471     return search_list_edges(list, dest);
472 }
473 
474 QOSEdgeType qos_graph_edge_get_type(QOSGraphEdge *edge)
475 {
476     if (!edge) {
477         return -1;
478     }
479     return edge->type;
480 }
481 
482 char *qos_graph_edge_get_dest(QOSGraphEdge *edge)
483 {
484     if (!edge) {
485         return NULL;
486     }
487     return edge->dest;
488 }
489 
490 void *qos_graph_edge_get_arg(QOSGraphEdge *edge)
491 {
492     if (!edge) {
493         return NULL;
494     }
495     return edge->arg;
496 }
497 
498 char *qos_graph_edge_get_after_cmd_line(QOSGraphEdge *edge)
499 {
500     if (!edge) {
501         return NULL;
502     }
503     return edge->after_cmd_line;
504 }
505 
506 char *qos_graph_edge_get_before_cmd_line(QOSGraphEdge *edge)
507 {
508     if (!edge) {
509         return NULL;
510     }
511     return edge->before_cmd_line;
512 }
513 
514 char *qos_graph_edge_get_extra_device_opts(QOSGraphEdge *edge)
515 {
516     if (!edge) {
517         return NULL;
518     }
519     return edge->extra_device_opts;
520 }
521 
522 char *qos_graph_edge_get_name(QOSGraphEdge *edge)
523 {
524     if (!edge) {
525         return NULL;
526     }
527     return edge->edge_name;
528 }
529 
530 bool qos_graph_has_edge(const char *start, const char *dest)
531 {
532     QOSGraphEdgeList *list = get_edgelist(start);
533     QOSGraphEdge *e = search_list_edges(list, dest);
534     return e != NULL;
535 }
536 
537 QOSGraphNode *qos_graph_get_machine(const char *node)
538 {
539     return search_machine(node);
540 }
541 
542 bool qos_graph_has_machine(const char *node)
543 {
544     QOSGraphNode *m = search_machine(node);
545     return m != NULL;
546 }
547 
548 void qos_print_graph(void)
549 {
550     qos_graph_foreach_test_path(qos_print_cb);
551 }
552 
553 void qos_graph_init(void)
554 {
555     if (!node_table) {
556         node_table = g_hash_table_new_full(g_str_hash, g_str_equal,
557                                            destroy_string, destroy_node);
558         create_node(QOS_ROOT, QNODE_DRIVER);
559     }
560 
561     if (!edge_table) {
562         edge_table = g_hash_table_new_full(g_str_hash, g_str_equal,
563                                            destroy_string, destroy_edges);
564     }
565 }
566 
567 void qos_graph_destroy(void)
568 {
569     if (node_table) {
570         g_hash_table_destroy(node_table);
571     }
572 
573     if (edge_table) {
574         g_hash_table_destroy(edge_table);
575     }
576 
577     node_table = NULL;
578     edge_table = NULL;
579 }
580 
581 void qos_node_destroy(void *key)
582 {
583     g_hash_table_remove(node_table, key);
584 }
585 
586 void qos_edge_destroy(void *key)
587 {
588     g_hash_table_remove(edge_table, key);
589 }
590 
591 void qos_add_test(const char *name, const char *interface,
592                   QOSTestFunc test_func, QOSGraphTestOptions *opts)
593 {
594     QOSGraphNode *node;
595     char *test_name = g_strdup_printf("%s-tests/%s", interface, name);
596     QOSGraphTestOptions def_opts = { };
597 
598     if (!opts) {
599         opts = &def_opts;
600     }
601     node = create_node(test_name, QNODE_TEST);
602     node->u.test.function = test_func;
603     node->u.test.arg = opts->arg;
604     assert(!opts->edge.arg);
605     assert(!opts->edge.size_arg);
606 
607     node->u.test.before = opts->before;
608     node->u.test.subprocess = opts->subprocess;
609     node->available = true;
610     add_edge(interface, test_name, QEDGE_CONSUMED_BY, &opts->edge);
611     g_free(test_name);
612 }
613 
614 void qos_node_create_machine(const char *name, QOSCreateMachineFunc function)
615 {
616     qos_node_create_machine_args(name, function, NULL);
617 }
618 
619 void qos_node_create_machine_args(const char *name,
620                                   QOSCreateMachineFunc function,
621                                   const char *opts)
622 {
623     QOSGraphNode *node = create_node(name, QNODE_MACHINE);
624     build_machine_cmd_line(node, opts);
625     node->u.machine.constructor = function;
626     add_edge(QOS_ROOT, name, QEDGE_CONTAINS, NULL);
627 }
628 
629 void qos_node_create_driver(const char *name, QOSCreateDriverFunc function)
630 {
631     QOSGraphNode *node = create_node(name, QNODE_DRIVER);
632     build_driver_cmd_line(node);
633     node->u.driver.constructor = function;
634 }
635 
636 void qos_node_create_driver_named(const char *name, const char *qemu_name,
637                                   QOSCreateDriverFunc function)
638 {
639     QOSGraphNode *node = create_node(name, QNODE_DRIVER);
640     node->qemu_name = g_strdup(qemu_name);
641     build_driver_cmd_line(node);
642     node->u.driver.constructor = function;
643 }
644 
645 void qos_node_contains(const char *container, const char *contained,
646                        QOSGraphEdgeOptions *opts, ...)
647 {
648     va_list va;
649 
650     if (opts == NULL) {
651         add_edge(container, contained, QEDGE_CONTAINS, NULL);
652         return;
653     }
654 
655     va_start(va, opts);
656     do {
657         add_edge(container, contained, QEDGE_CONTAINS, opts);
658         opts = va_arg(va, QOSGraphEdgeOptions *);
659     } while (opts != NULL);
660 
661     va_end(va);
662 }
663 
664 void qos_node_produces(const char *producer, const char *interface)
665 {
666     create_interface(interface);
667     add_edge(producer, interface, QEDGE_PRODUCES, NULL);
668 }
669 
670 void qos_node_consumes(const char *consumer, const char *interface,
671                        QOSGraphEdgeOptions *opts)
672 {
673     create_interface(interface);
674     add_edge(interface, consumer, QEDGE_CONSUMED_BY, opts);
675 }
676 
677 static void qos_graph_node_set_availability_explicit(const char *node, bool av)
678 {
679     QOSGraphEdgeList *elist;
680     QOSGraphNode *n = search_node(node);
681     QOSGraphEdge *e, *next;
682     if (!n) {
683         return;
684     }
685     n->available = av;
686     elist = get_edgelist(node);
687     if (!elist) {
688         return;
689     }
690     QSLIST_FOREACH_SAFE(e, elist, edge_list, next) {
691         if (e->type == QEDGE_CONTAINS || e->type == QEDGE_PRODUCES) {
692             qos_graph_node_set_availability_explicit(e->dest, av);
693         }
694     }
695 }
696 
697 /*
698  * Behaves as qos_graph_node_set_availability_explicit(), except that the
699  * former always matches by node name only, whereas this function matches both
700  * by node name and node's optional 'qemu_name' field.
701  */
702 void qos_graph_node_set_availability(const char *node, bool av)
703 {
704     GList *l;
705     QOSGraphEdgeList *elist;
706     QOSGraphEdge *e, *next;
707     QOSGraphNode *n;
708     GList *keys = g_hash_table_get_keys(node_table);
709 
710     for (l = keys; l != NULL; l = l->next) {
711         const gchar *key = l->data;
712         n = g_hash_table_lookup(node_table, key);
713         /*
714          * node's 'qemu_name' is set if there is more than one device with
715          * the same QEMU (QMP) device name
716          */
717         const char *node_name = n->qemu_name ?: n->name;
718         if (g_strcmp0(node_name, node) == 0) {
719             n->available = av;
720             elist = get_edgelist(n->name);
721             if (elist) {
722                 QSLIST_FOREACH_SAFE(e, elist, edge_list, next) {
723                     if (e->type == QEDGE_CONTAINS || e->type == QEDGE_PRODUCES)
724                     {
725                         qos_graph_node_set_availability_explicit(e->dest, av);
726                     }
727                 }
728             }
729         }
730     }
731     g_list_free(keys);
732 }
733 
734 void qos_graph_foreach_test_path(QOSTestCallback fn)
735 {
736     QOSGraphNode *root = qos_graph_get_node(QOS_ROOT);
737     qos_traverse_graph(root, fn);
738 }
739 
740 QOSGraphObject *qos_machine_new(QOSGraphNode *node, QTestState *qts)
741 {
742     QOSGraphObject *obj;
743 
744     g_assert(node->type == QNODE_MACHINE);
745     obj = node->u.machine.constructor(qts);
746     obj->free = g_free;
747     return obj;
748 }
749 
750 QOSGraphObject *qos_driver_new(QOSGraphNode *node, QOSGraphObject *parent,
751                                QGuestAllocator *alloc, void *arg)
752 {
753     QOSGraphObject *obj;
754 
755     g_assert(node->type == QNODE_DRIVER);
756     obj = node->u.driver.constructor(parent, alloc, arg);
757     obj->free = g_free;
758     return obj;
759 }
760 
761 void qos_object_destroy(QOSGraphObject *obj)
762 {
763     if (!obj) {
764         return;
765     }
766     if (obj->destructor) {
767         obj->destructor(obj);
768     }
769     if (obj->free) {
770         obj->free(obj);
771     }
772 }
773 
774 void qos_object_queue_destroy(QOSGraphObject *obj)
775 {
776     g_test_queue_destroy((GDestroyNotify) qos_object_destroy, obj);
777 }
778 
779 void qos_object_start_hw(QOSGraphObject *obj)
780 {
781     if (obj->start_hw) {
782         obj->start_hw(obj);
783     }
784 }
785 
786 char *qos_get_machine_type(char *name)
787 {
788     while (*name != '\0' && *name != '/') {
789         name++;
790     }
791 
792     if (!*name || !name[1]) {
793         fprintf(stderr, "Machine name has to be of the form <arch>/<machine>\n");
794         abort();
795     }
796 
797     return name + 1;
798 }
799 
800 void qos_delete_cmd_line(const char *name)
801 {
802     QOSGraphNode *node = search_node(name);
803     if (node) {
804         g_free(node->command_line);
805         node->command_line = NULL;
806     }
807 }
808 
809 void qos_dump_graph(void)
810 {
811     GList *keys;
812     GList *l;
813     QOSGraphEdgeList *list;
814     QOSGraphEdge *e, *next;
815     QOSGraphNode *dest_node, *node;
816 
817     qos_printf("ALL QGRAPH EDGES: {\n");
818     keys = g_hash_table_get_keys(edge_table);
819     for (l = keys; l != NULL; l = l->next) {
820         const gchar *key = l->data;
821         qos_printf("\t src='%s'\n", key);
822         list = get_edgelist(key);
823         QSLIST_FOREACH_SAFE(e, list, edge_list, next) {
824             dest_node = g_hash_table_lookup(node_table, e->dest);
825             qos_printf("\t\t|-> dest='%s' type=%d (node=%p)",
826                        e->dest, e->type, dest_node);
827             if (!dest_node) {
828                 qos_printf_literal(" <------- ERROR !");
829             }
830             qos_printf_literal("\n");
831         }
832     }
833     g_list_free(keys);
834     qos_printf("}\n");
835 
836     qos_printf("ALL QGRAPH NODES: {\n");
837     keys = g_hash_table_get_keys(node_table);
838     for (l = keys; l != NULL; l = l->next) {
839         const gchar *key = l->data;
840         node = g_hash_table_lookup(node_table, key);
841         qos_printf("\t name='%s' ", key);
842         if (node->qemu_name) {
843             qos_printf_literal("qemu_name='%s' ", node->qemu_name);
844         }
845         qos_printf_literal("type=%d cmd_line='%s' [%s]\n",
846                            node->type, node->command_line,
847                            node->available ? "available" : "UNAVAILABLE"
848         );
849     }
850     g_list_free(keys);
851     qos_printf("}\n");
852 }
853