83ff78e567
If qtests were run in verbose mode (i.e. if --verbose CL argument was provided) then dump the generated qos graph (all nodes and edges, along with their current individual availability status) to stdout, which allows to identify problems in the created qos graph e.g. when writing new qos tests. See API doc comment on function qos_dump_graph() for details. Signed-off-by: Christian Schoenebeck <qemu_oss@crudebyte.com> Message-Id: <6bffb6e38589fb2c06a2c1b5deed33f3e710fed1.1611704181.git.qemu_oss@crudebyte.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
611 lines
24 KiB
C
611 lines
24 KiB
C
/*
|
|
* libqos driver framework
|
|
*
|
|
* Copyright (c) 2018 Emanuele Giuseppe Esposito <e.emanuelegiuseppe@gmail.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License version 2.1 as published by the Free Software Foundation.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>
|
|
*/
|
|
|
|
#ifndef QGRAPH_H
|
|
#define QGRAPH_H
|
|
|
|
#include <gmodule.h>
|
|
#include "qemu/module.h"
|
|
#include "malloc.h"
|
|
|
|
/* maximum path length */
|
|
#define QOS_PATH_MAX_ELEMENT_SIZE 50
|
|
|
|
typedef struct QOSGraphObject QOSGraphObject;
|
|
typedef struct QOSGraphNode QOSGraphNode;
|
|
typedef struct QOSGraphEdge QOSGraphEdge;
|
|
typedef struct QOSGraphNodeOptions QOSGraphNodeOptions;
|
|
typedef struct QOSGraphEdgeOptions QOSGraphEdgeOptions;
|
|
typedef struct QOSGraphTestOptions QOSGraphTestOptions;
|
|
|
|
/* Constructor for drivers, machines and test */
|
|
typedef void *(*QOSCreateDriverFunc) (void *parent, QGuestAllocator *alloc,
|
|
void *addr);
|
|
typedef void *(*QOSCreateMachineFunc) (QTestState *qts);
|
|
typedef void (*QOSTestFunc) (void *parent, void *arg, QGuestAllocator *alloc);
|
|
|
|
/* QOSGraphObject functions */
|
|
typedef void *(*QOSGetDriver) (void *object, const char *interface);
|
|
typedef QOSGraphObject *(*QOSGetDevice) (void *object, const char *name);
|
|
typedef void (*QOSDestructorFunc) (QOSGraphObject *object);
|
|
typedef void (*QOSStartFunct) (QOSGraphObject *object);
|
|
|
|
/* Test options functions */
|
|
typedef void *(*QOSBeforeTest) (GString *cmd_line, void *arg);
|
|
|
|
/**
|
|
* SECTION: qgraph.h
|
|
* @title: Qtest Driver Framework
|
|
* @short_description: interfaces to organize drivers and tests
|
|
* as nodes in a graph
|
|
*
|
|
* This Qgraph API provides all basic functions to create a graph
|
|
* and instantiate nodes representing machines, drivers and tests
|
|
* representing their relations with CONSUMES, PRODUCES, and CONTAINS
|
|
* edges.
|
|
*
|
|
* The idea is to have a framework where each test asks for a specific
|
|
* driver, and the framework takes care of allocating the proper devices
|
|
* required and passing the correct command line arguments to QEMU.
|
|
*
|
|
* A node can be of four types:
|
|
* - QNODE_MACHINE: for example "arm/raspi2"
|
|
* - QNODE_DRIVER: for example "generic-sdhci"
|
|
* - QNODE_INTERFACE: for example "sdhci" (interface for all "-sdhci" drivers)
|
|
* an interface is not explicitly created, it will be auto-
|
|
* matically instantiated when a node consumes or produces
|
|
* it.
|
|
* - QNODE_TEST: for example "sdhci-test", consumes an interface and tests
|
|
* the functions provided
|
|
*
|
|
* Notes for the nodes:
|
|
* - QNODE_MACHINE: each machine struct must have a QGuestAllocator and
|
|
* implement get_driver to return the allocator passing
|
|
* "memory". The function can also return NULL if the
|
|
* allocator is not set.
|
|
* - QNODE_DRIVER: driver names must be unique, and machines and nodes
|
|
* planned to be "consumed" by other nodes must match QEMU
|
|
* drivers name, otherwise they won't be discovered
|
|
*
|
|
* An edge relation between two nodes (drivers or machines) X and Y can be:
|
|
* - X CONSUMES Y: Y can be plugged into X
|
|
* - X PRODUCES Y: X provides the interface Y
|
|
* - X CONTAINS Y: Y is part of X component
|
|
*
|
|
* Basic framework steps are the following:
|
|
* - All nodes and edges are created in their respective
|
|
* machine/driver/test files
|
|
* - The framework starts QEMU and asks for a list of available devices
|
|
* and machines (note that only machines and "consumed" nodes are mapped
|
|
* 1:1 with QEMU devices)
|
|
* - The framework walks the graph starting from the available machines and
|
|
* performs a Depth First Search for tests
|
|
* - Once a test is found, the path is walked again and all drivers are
|
|
* allocated accordingly and the final interface is passed to the test
|
|
* - The test is executed
|
|
* - Unused objects are cleaned and the path discovery is continued
|
|
*
|
|
* Depending on the QEMU binary used, only some drivers/machines will be
|
|
* available and only test that are reached by them will be executed.
|
|
*
|
|
* <example>
|
|
* <title>Creating new driver an its interface</title>
|
|
* <programlisting>
|
|
#include "qgraph.h"
|
|
|
|
struct My_driver {
|
|
QOSGraphObject obj;
|
|
Node_produced prod;
|
|
Node_contained cont;
|
|
}
|
|
|
|
static void my_destructor(QOSGraphObject *obj)
|
|
{
|
|
g_free(obj);
|
|
}
|
|
|
|
static void my_get_driver(void *object, const char *interface) {
|
|
My_driver *dev = object;
|
|
if (!g_strcmp0(interface, "my_interface")) {
|
|
return &dev->prod;
|
|
}
|
|
abort();
|
|
}
|
|
|
|
static void my_get_device(void *object, const char *device) {
|
|
My_driver *dev = object;
|
|
if (!g_strcmp0(device, "my_driver_contained")) {
|
|
return &dev->cont;
|
|
}
|
|
abort();
|
|
}
|
|
|
|
static void *my_driver_constructor(void *node_consumed,
|
|
QOSGraphObject *alloc)
|
|
{
|
|
My_driver dev = g_new(My_driver, 1);
|
|
// get the node pointed by the produce edge
|
|
dev->obj.get_driver = my_get_driver;
|
|
// get the node pointed by the contains
|
|
dev->obj.get_device = my_get_device;
|
|
// free the object
|
|
dev->obj.destructor = my_destructor;
|
|
do_something_with_node_consumed(node_consumed);
|
|
// set all fields of contained device
|
|
init_contained_device(&dev->cont);
|
|
return &dev->obj;
|
|
}
|
|
|
|
static void register_my_driver(void)
|
|
{
|
|
qos_node_create_driver("my_driver", my_driver_constructor);
|
|
// contained drivers don't need a constructor,
|
|
// they will be init by the parent.
|
|
qos_node_create_driver("my_driver_contained", NULL);
|
|
|
|
// For the sake of this example, assume machine x86_64/pc contains
|
|
// "other_node".
|
|
// This relation, along with the machine and "other_node" creation,
|
|
// should be defined in the x86_64_pc-machine.c file.
|
|
// "my_driver" will then consume "other_node"
|
|
qos_node_contains("my_driver", "my_driver_contained");
|
|
qos_node_produces("my_driver", "my_interface");
|
|
qos_node_consumes("my_driver", "other_node");
|
|
}
|
|
* </programlisting>
|
|
* </example>
|
|
*
|
|
* In the above example, all possible types of relations are created:
|
|
* node "my_driver" consumes, contains and produces other nodes.
|
|
* more specifically:
|
|
* x86_64/pc -->contains--> other_node <--consumes-- my_driver
|
|
* |
|
|
* my_driver_contained <--contains--+
|
|
* |
|
|
* my_interface <--produces--+
|
|
*
|
|
* or inverting the consumes edge in consumed_by:
|
|
*
|
|
* x86_64/pc -->contains--> other_node --consumed_by--> my_driver
|
|
* |
|
|
* my_driver_contained <--contains--+
|
|
* |
|
|
* my_interface <--produces--+
|
|
*
|
|
* <example>
|
|
* <title>Creating new test</title>
|
|
* <programlisting>
|
|
* #include "qgraph.h"
|
|
*
|
|
* static void my_test_function(void *obj, void *data)
|
|
* {
|
|
* Node_produced *interface_to_test = obj;
|
|
* // test interface_to_test
|
|
* }
|
|
*
|
|
* static void register_my_test(void)
|
|
* {
|
|
* qos_add_test("my_interface", "my_test", my_test_function);
|
|
* }
|
|
*
|
|
* libqos_init(register_my_test);
|
|
*
|
|
* </programlisting>
|
|
* </example>
|
|
*
|
|
* Here a new test is created, consuming "my_interface" node
|
|
* and creating a valid path from a machine to a test.
|
|
* Final graph will be like this:
|
|
* x86_64/pc -->contains--> other_node <--consumes-- my_driver
|
|
* |
|
|
* my_driver_contained <--contains--+
|
|
* |
|
|
* my_test --consumes--> my_interface <--produces--+
|
|
*
|
|
* or inverting the consumes edge in consumed_by:
|
|
*
|
|
* x86_64/pc -->contains--> other_node --consumed_by--> my_driver
|
|
* |
|
|
* my_driver_contained <--contains--+
|
|
* |
|
|
* my_test <--consumed_by-- my_interface <--produces--+
|
|
*
|
|
* Assuming there the binary is
|
|
* QTEST_QEMU_BINARY=./qemu-system-x86_64
|
|
* a valid test path will be:
|
|
* "/x86_64/pc/other_node/my_driver/my_interface/my_test".
|
|
*
|
|
* Additional examples are also in test-qgraph.c
|
|
*
|
|
* Command line:
|
|
* Command line is built by using node names and optional arguments
|
|
* passed by the user when building the edges.
|
|
*
|
|
* There are three types of command line arguments:
|
|
* - in node : created from the node name. For example, machines will
|
|
* have "-M <machine>" to its command line, while devices
|
|
* "-device <device>". It is automatically done by the
|
|
* framework.
|
|
* - after node : added as additional argument to the node name.
|
|
* This argument is added optionally when creating edges,
|
|
* by setting the parameter @after_cmd_line and
|
|
* @extra_edge_opts in #QOSGraphEdgeOptions.
|
|
* The framework automatically adds
|
|
* a comma before @extra_edge_opts,
|
|
* because it is going to add attributes
|
|
* after the destination node pointed by
|
|
* the edge containing these options, and automatically
|
|
* adds a space before @after_cmd_line, because it
|
|
* adds an additional device, not an attribute.
|
|
* - before node : added as additional argument to the node name.
|
|
* This argument is added optionally when creating edges,
|
|
* by setting the parameter @before_cmd_line in
|
|
* #QOSGraphEdgeOptions. This attribute
|
|
* is going to add attributes before the destination node
|
|
* pointed by the edge containing these options. It is
|
|
* helpful to commands that are not node-representable,
|
|
* such as "-fdsev" or "-netdev".
|
|
*
|
|
* While adding command line in edges is always used, not all nodes names are
|
|
* used in every path walk: this is because the contained or produced ones
|
|
* are already added by QEMU, so only nodes that "consumes" will be used to
|
|
* build the command line. Also, nodes that will have { "abstract" : true }
|
|
* as QMP attribute will loose their command line, since they are not proper
|
|
* devices to be added in QEMU.
|
|
*
|
|
* Example:
|
|
*
|
|
QOSGraphEdgeOptions opts = {
|
|
.arg = NULL,
|
|
.size_arg = 0,
|
|
.after_cmd_line = "-device other",
|
|
.before_cmd_line = "-netdev something",
|
|
.extra_edge_opts = "addr=04.0",
|
|
};
|
|
QOSGraphNode * node = qos_node_create_driver("my_node", constructor);
|
|
qos_node_consumes_args("my_node", "interface", &opts);
|
|
*
|
|
* Will produce the following command line:
|
|
* "-netdev something -device my_node,addr=04.0 -device other"
|
|
*/
|
|
|
|
/**
|
|
* Edge options to be passed to the contains/consumes *_args function.
|
|
*/
|
|
struct QOSGraphEdgeOptions {
|
|
void *arg; /*
|
|
* optional arg that will be used by
|
|
* dest edge
|
|
*/
|
|
uint32_t size_arg; /*
|
|
* optional arg size that will be used by
|
|
* dest edge
|
|
*/
|
|
const char *extra_device_opts;/*
|
|
*optional additional command line for dest
|
|
* edge, used to add additional attributes
|
|
* *after* the node command line, the
|
|
* framework automatically prepends ","
|
|
* to this argument.
|
|
*/
|
|
const char *before_cmd_line; /*
|
|
* optional additional command line for dest
|
|
* edge, used to add additional attributes
|
|
* *before* the node command line, usually
|
|
* other non-node represented commands,
|
|
* like "-fdsev synt"
|
|
*/
|
|
const char *after_cmd_line; /*
|
|
* optional extra command line to be added
|
|
* after the device command. This option
|
|
* is used to add other devices
|
|
* command line that depend on current node.
|
|
* Automatically prepends " " to this
|
|
* argument
|
|
*/
|
|
const char *edge_name; /*
|
|
* optional edge to differentiate multiple
|
|
* devices with same node name
|
|
*/
|
|
};
|
|
|
|
/**
|
|
* Test options to be passed to the test functions.
|
|
*/
|
|
struct QOSGraphTestOptions {
|
|
QOSGraphEdgeOptions edge; /* edge arguments that will be used by test.
|
|
* Note that test *does not* use edge_name,
|
|
* and uses instead arg and size_arg as
|
|
* data arg for its test function.
|
|
*/
|
|
void *arg; /* passed to the .before function, or to the
|
|
* test function if there is no .before
|
|
* function
|
|
*/
|
|
QOSBeforeTest before; /* executed before the test. Can add
|
|
* additional parameters to the command line
|
|
* and modify the argument to the test function.
|
|
*/
|
|
bool subprocess; /* run the test in a subprocess */
|
|
};
|
|
|
|
/**
|
|
* Each driver, test or machine of this framework will have a
|
|
* QOSGraphObject as first field.
|
|
*
|
|
* This set of functions offered by QOSGraphObject are executed
|
|
* in different stages of the framework:
|
|
* - get_driver / get_device : Once a machine-to-test path has been
|
|
* found, the framework traverses it again and allocates all the
|
|
* nodes, using the provided constructor. To satisfy their relations,
|
|
* i.e. for produces or contains, where a struct constructor needs
|
|
* an external parameter represented by the previous node,
|
|
* the framework will call get_device (for contains) or
|
|
* get_driver (for produces), depending on the edge type, passing
|
|
* them the name of the next node to be taken and getting from them
|
|
* the corresponding pointer to the actual structure of the next node to
|
|
* be used in the path.
|
|
*
|
|
* - start_hw: This function is executed after all the path objects
|
|
* have been allocated, but before the test is run. It starts the hw, setting
|
|
* the initial configurations (*_device_enable) and making it ready for the
|
|
* test.
|
|
*
|
|
* - destructor: Opposite to the node constructor, destroys the object.
|
|
* This function is called after the test has been executed, and performs
|
|
* a complete cleanup of each node allocated field. In case no constructor
|
|
* is provided, no destructor will be called.
|
|
*
|
|
*/
|
|
struct QOSGraphObject {
|
|
/* for produces edges, returns void * */
|
|
QOSGetDriver get_driver;
|
|
/* for contains edges, returns a QOSGraphObject * */
|
|
QOSGetDevice get_device;
|
|
/* start the hw, get ready for the test */
|
|
QOSStartFunct start_hw;
|
|
/* destroy this QOSGraphObject */
|
|
QOSDestructorFunc destructor;
|
|
/* free the memory associated to the QOSGraphObject and its contained
|
|
* children */
|
|
GDestroyNotify free;
|
|
};
|
|
|
|
/**
|
|
* qos_graph_init(): initialize the framework, creates two hash
|
|
* tables: one for the nodes and another for the edges.
|
|
*/
|
|
void qos_graph_init(void);
|
|
|
|
/**
|
|
* qos_graph_destroy(): deallocates all the hash tables,
|
|
* freeing all nodes and edges.
|
|
*/
|
|
void qos_graph_destroy(void);
|
|
|
|
/**
|
|
* qos_node_destroy(): removes and frees a node from the,
|
|
* nodes hash table.
|
|
*/
|
|
void qos_node_destroy(void *key);
|
|
|
|
/**
|
|
* qos_edge_destroy(): removes and frees an edge from the,
|
|
* edges hash table.
|
|
*/
|
|
void qos_edge_destroy(void *key);
|
|
|
|
/**
|
|
* qos_add_test(): adds a test node @name to the nodes hash table.
|
|
*
|
|
* The test will consume a @interface node, and once the
|
|
* graph walking algorithm has found it, the @test_func will be
|
|
* executed. It also has the possibility to
|
|
* add an optional @opts (see %QOSGraphNodeOptions).
|
|
*
|
|
* For tests, opts->edge.arg and size_arg represent the arg to pass
|
|
* to @test_func
|
|
*/
|
|
void qos_add_test(const char *name, const char *interface,
|
|
QOSTestFunc test_func,
|
|
QOSGraphTestOptions *opts);
|
|
|
|
/**
|
|
* qos_node_create_machine(): creates the machine @name and
|
|
* adds it to the node hash table.
|
|
*
|
|
* This node will be of type QNODE_MACHINE and have @function
|
|
* as constructor
|
|
*/
|
|
void qos_node_create_machine(const char *name, QOSCreateMachineFunc function);
|
|
|
|
/**
|
|
* qos_node_create_machine_args(): same as qos_node_create_machine,
|
|
* but with the possibility to add an optional ", @opts" after -M machine
|
|
* command line.
|
|
*/
|
|
void qos_node_create_machine_args(const char *name,
|
|
QOSCreateMachineFunc function,
|
|
const char *opts);
|
|
|
|
/**
|
|
* qos_node_create_driver(): creates the driver @name and
|
|
* adds it to the node hash table.
|
|
*
|
|
* This node will be of type QNODE_DRIVER and have @function
|
|
* as constructor
|
|
*/
|
|
void qos_node_create_driver(const char *name, QOSCreateDriverFunc function);
|
|
|
|
/**
|
|
* Behaves as qos_node_create_driver() with the extension of allowing to
|
|
* specify a different node name vs. associated QEMU device name.
|
|
*
|
|
* Use this function instead of qos_node_create_driver() if you need to create
|
|
* several instances of the same QEMU device. You are free to choose a custom
|
|
* node name, however the chosen node name must always be unique.
|
|
*
|
|
* @param name: custom, unique name of the node to be created
|
|
* @param qemu_name: actual (official) QEMU driver name the node shall be
|
|
* associated with
|
|
* @param function: driver constructor
|
|
*/
|
|
void qos_node_create_driver_named(const char *name, const char *qemu_name,
|
|
QOSCreateDriverFunc function);
|
|
|
|
/**
|
|
* qos_node_contains(): creates one or more edges of type QEDGE_CONTAINS
|
|
* and adds them to the edge list mapped to @container in the
|
|
* edge hash table.
|
|
*
|
|
* The edges will have @container as source and @contained as destination.
|
|
*
|
|
* If @opts is NULL, a single edge will be added with no options.
|
|
* If @opts is non-NULL, the arguments after @contained represent a
|
|
* NULL-terminated list of %QOSGraphEdgeOptions structs, and an
|
|
* edge will be added for each of them.
|
|
*
|
|
* This function can be useful when there are multiple devices
|
|
* with the same node name contained in a machine/other node
|
|
*
|
|
* For example, if "arm/raspi2" contains 2 "generic-sdhci"
|
|
* devices, the right commands will be:
|
|
* qos_node_create_machine("arm/raspi2");
|
|
* qos_node_create_driver("generic-sdhci", constructor);
|
|
* //assume rest of the fields are set NULL
|
|
* QOSGraphEdgeOptions op1 = { .edge_name = "emmc" };
|
|
* QOSGraphEdgeOptions op2 = { .edge_name = "sdcard" };
|
|
* qos_node_contains("arm/raspi2", "generic-sdhci", &op1, &op2, NULL);
|
|
*
|
|
* Of course this also requires that the @container's get_device function
|
|
* should implement a case for "emmc" and "sdcard".
|
|
*
|
|
* For contains, op1.arg and op1.size_arg represent the arg to pass
|
|
* to @contained constructor to properly initialize it.
|
|
*/
|
|
void qos_node_contains(const char *container, const char *contained,
|
|
QOSGraphEdgeOptions *opts, ...);
|
|
|
|
/**
|
|
* qos_node_produces(): creates an edge of type QEDGE_PRODUCES and
|
|
* adds it to the edge list mapped to @producer in the
|
|
* edge hash table.
|
|
*
|
|
* This edge will have @producer as source and @interface as destination.
|
|
*/
|
|
void qos_node_produces(const char *producer, const char *interface);
|
|
|
|
/**
|
|
* qos_node_consumes(): creates an edge of type QEDGE_CONSUMED_BY and
|
|
* adds it to the edge list mapped to @interface in the
|
|
* edge hash table.
|
|
*
|
|
* This edge will have @interface as source and @consumer as destination.
|
|
* It also has the possibility to add an optional @opts
|
|
* (see %QOSGraphEdgeOptions)
|
|
*/
|
|
void qos_node_consumes(const char *consumer, const char *interface,
|
|
QOSGraphEdgeOptions *opts);
|
|
|
|
/**
|
|
* qos_invalidate_command_line(): invalidates current command line, so that
|
|
* qgraph framework cannot try to cache the current command line and
|
|
* forces QEMU to restart.
|
|
*/
|
|
void qos_invalidate_command_line(void);
|
|
|
|
/**
|
|
* qos_get_current_command_line(): return the command line required by the
|
|
* machine and driver objects. This is the same string that was passed to
|
|
* the test's "before" callback, if any.
|
|
*/
|
|
const char *qos_get_current_command_line(void);
|
|
|
|
/**
|
|
* qos_allocate_objects():
|
|
* @qts: The #QTestState that will be referred to by the machine object.
|
|
* @alloc: Where to store the allocator for the machine object, or %NULL.
|
|
*
|
|
* Allocate driver objects for the current test
|
|
* path, but relative to the QTestState @qts.
|
|
*
|
|
* Returns a test object just like the one that was passed to
|
|
* the test function, but relative to @qts.
|
|
*/
|
|
void *qos_allocate_objects(QTestState *qts, QGuestAllocator **p_alloc);
|
|
|
|
/**
|
|
* qos_object_destroy(): calls the destructor for @obj
|
|
*/
|
|
void qos_object_destroy(QOSGraphObject *obj);
|
|
|
|
/**
|
|
* qos_object_queue_destroy(): queue the destructor for @obj so that it is
|
|
* called at the end of the test
|
|
*/
|
|
void qos_object_queue_destroy(QOSGraphObject *obj);
|
|
|
|
/**
|
|
* qos_object_start_hw(): calls the start_hw function for @obj
|
|
*/
|
|
void qos_object_start_hw(QOSGraphObject *obj);
|
|
|
|
/**
|
|
* qos_machine_new(): instantiate a new machine node
|
|
* @node: A machine node to be instantiated
|
|
* @qts: The #QTestState that will be referred to by the machine object.
|
|
*
|
|
* Returns a machine object.
|
|
*/
|
|
QOSGraphObject *qos_machine_new(QOSGraphNode *node, QTestState *qts);
|
|
|
|
/**
|
|
* qos_machine_new(): instantiate a new driver node
|
|
* @node: A driver node to be instantiated
|
|
* @parent: A #QOSGraphObject to be consumed by the new driver node
|
|
* @alloc: An allocator to be used by the new driver node.
|
|
* @arg: The argument for the consumed-by edge to @node.
|
|
*
|
|
* Calls the constructor for the driver object.
|
|
*/
|
|
QOSGraphObject *qos_driver_new(QOSGraphNode *node, QOSGraphObject *parent,
|
|
QGuestAllocator *alloc, void *arg);
|
|
|
|
/**
|
|
* Just for debugging purpose: prints all currently existing nodes and
|
|
* edges to stdout.
|
|
*
|
|
* All qtests add themselves to the overall qos graph by calling qgraph
|
|
* functions that add device nodes and edges between the individual graph
|
|
* nodes for tests. As the actual graph is assmbled at runtime by the qos
|
|
* subsystem, it is sometimes not obvious how the overall graph looks like.
|
|
* E.g. when writing new tests it may happen that those new tests are simply
|
|
* ignored by the qtest framework.
|
|
*
|
|
* This function allows to identify problems in the created qgraph. Keep in
|
|
* mind: only tests with a path down from the actual test case node (leaf) up
|
|
* to the graph's root node are actually executed by the qtest framework. And
|
|
* the qtest framework uses QMP to automatically check which QEMU drivers are
|
|
* actually currently available, and accordingly qos marks certain pathes as
|
|
* 'unavailable' in such cases (e.g. when QEMU was compiled without support for
|
|
* a certain feature).
|
|
*/
|
|
void qos_dump_graph(void);
|
|
|
|
#endif
|