436 lines
13 KiB
C
436 lines
13 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 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/>
|
||
|
*/
|
||
|
|
||
|
#include "qemu/osdep.h"
|
||
|
#include "libqtest.h"
|
||
|
#include "libqos/qgraph.h"
|
||
|
#include "libqos/qgraph_internal.h"
|
||
|
|
||
|
#define MACHINE_PC "x86_64/pc"
|
||
|
#define MACHINE_RASPI2 "arm/raspi2"
|
||
|
#define I440FX "i440FX-pcihost"
|
||
|
#define PCIBUS_PC "pcibus-pc"
|
||
|
#define SDHCI "sdhci"
|
||
|
#define PCIBUS "pci-bus"
|
||
|
#define SDHCI_PCI "sdhci-pci"
|
||
|
#define SDHCI_MM "generic-sdhci"
|
||
|
#define REGISTER_TEST "register-test"
|
||
|
|
||
|
int npath;
|
||
|
|
||
|
static void *machinefunct(QTestState *qts)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
static void *driverfunct(void *obj, QGuestAllocator *machine, void *arg)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
static void testfunct(void *obj, void *arg, QGuestAllocator *alloc)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
static void check_interface(const char *interface)
|
||
|
{
|
||
|
g_assert_cmpint(qos_graph_has_machine(interface), ==, FALSE);
|
||
|
g_assert_nonnull(qos_graph_get_node(interface));
|
||
|
g_assert_cmpint(qos_graph_has_node(interface), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_type(interface), ==, QNODE_INTERFACE);
|
||
|
qos_graph_node_set_availability(interface, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(interface), ==, TRUE);
|
||
|
}
|
||
|
|
||
|
static void check_machine(const char *machine)
|
||
|
{
|
||
|
qos_node_create_machine(machine, machinefunct);
|
||
|
g_assert_nonnull(qos_graph_get_machine(machine));
|
||
|
g_assert_cmpint(qos_graph_has_machine(machine), ==, TRUE);
|
||
|
g_assert_nonnull(qos_graph_get_node(machine));
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(machine), ==, FALSE);
|
||
|
qos_graph_node_set_availability(machine, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(machine), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_has_node(machine), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_type(machine), ==, QNODE_MACHINE);
|
||
|
}
|
||
|
|
||
|
static void check_contains(const char *machine, const char *driver)
|
||
|
{
|
||
|
QOSGraphEdge *edge;
|
||
|
qos_node_contains(machine, driver, NULL);
|
||
|
|
||
|
edge = qos_graph_get_edge(machine, driver);
|
||
|
g_assert_nonnull(edge);
|
||
|
g_assert_cmpint(qos_graph_edge_get_type(edge), ==, QEDGE_CONTAINS);
|
||
|
g_assert_cmpint(qos_graph_has_edge(machine, driver), ==, TRUE);
|
||
|
}
|
||
|
|
||
|
static void check_produces(const char *machine, const char *interface)
|
||
|
{
|
||
|
QOSGraphEdge *edge;
|
||
|
|
||
|
qos_node_produces(machine, interface);
|
||
|
check_interface(interface);
|
||
|
edge = qos_graph_get_edge(machine, interface);
|
||
|
g_assert_nonnull(edge);
|
||
|
g_assert_cmpint(qos_graph_edge_get_type(edge), ==,
|
||
|
QEDGE_PRODUCES);
|
||
|
g_assert_cmpint(qos_graph_has_edge(machine, interface), ==, TRUE);
|
||
|
}
|
||
|
|
||
|
static void check_consumes(const char *driver, const char *interface)
|
||
|
{
|
||
|
QOSGraphEdge *edge;
|
||
|
|
||
|
qos_node_consumes(driver, interface, NULL);
|
||
|
check_interface(interface);
|
||
|
edge = qos_graph_get_edge(interface, driver);
|
||
|
g_assert_nonnull(edge);
|
||
|
g_assert_cmpint(qos_graph_edge_get_type(edge), ==, QEDGE_CONSUMED_BY);
|
||
|
g_assert_cmpint(qos_graph_has_edge(interface, driver), ==, TRUE);
|
||
|
}
|
||
|
|
||
|
static void check_driver(const char *driver)
|
||
|
{
|
||
|
qos_node_create_driver(driver, driverfunct);
|
||
|
g_assert_cmpint(qos_graph_has_machine(driver), ==, FALSE);
|
||
|
g_assert_nonnull(qos_graph_get_node(driver));
|
||
|
g_assert_cmpint(qos_graph_has_node(driver), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_type(driver), ==, QNODE_DRIVER);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(driver), ==, FALSE);
|
||
|
qos_graph_node_set_availability(driver, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(driver), ==, TRUE);
|
||
|
}
|
||
|
|
||
|
static void check_test(const char *test, const char *interface)
|
||
|
{
|
||
|
QOSGraphEdge *edge;
|
||
|
char *full_name = g_strdup_printf("%s-tests/%s", interface, test);
|
||
|
|
||
|
qos_add_test(test, interface, testfunct, NULL);
|
||
|
g_assert_cmpint(qos_graph_has_machine(test), ==, FALSE);
|
||
|
g_assert_cmpint(qos_graph_has_machine(full_name), ==, FALSE);
|
||
|
g_assert_nonnull(qos_graph_get_node(full_name));
|
||
|
g_assert_cmpint(qos_graph_has_node(full_name), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_type(full_name), ==, QNODE_TEST);
|
||
|
edge = qos_graph_get_edge(interface, full_name);
|
||
|
g_assert_nonnull(edge);
|
||
|
g_assert_cmpint(qos_graph_edge_get_type(edge), ==,
|
||
|
QEDGE_CONSUMED_BY);
|
||
|
g_assert_cmpint(qos_graph_has_edge(interface, full_name), ==, TRUE);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(full_name), ==, TRUE);
|
||
|
qos_graph_node_set_availability(full_name, FALSE);
|
||
|
g_assert_cmpint(qos_graph_get_node_availability(full_name), ==, FALSE);
|
||
|
g_free(full_name);
|
||
|
}
|
||
|
|
||
|
static void count_each_test(QOSGraphNode *path, int len)
|
||
|
{
|
||
|
npath++;
|
||
|
}
|
||
|
|
||
|
static void check_leaf_discovered(int n)
|
||
|
{
|
||
|
npath = 0;
|
||
|
qos_graph_foreach_test_path(count_each_test);
|
||
|
g_assert_cmpint(n, ==, npath);
|
||
|
}
|
||
|
|
||
|
/* G_Test functions */
|
||
|
|
||
|
static void init_nop(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_machine(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_contains(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_contains(MACHINE_PC, I440FX);
|
||
|
g_assert_null(qos_graph_get_machine(MACHINE_PC));
|
||
|
g_assert_null(qos_graph_get_machine(I440FX));
|
||
|
g_assert_null(qos_graph_get_node(MACHINE_PC));
|
||
|
g_assert_null(qos_graph_get_node(I440FX));
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_multiple_contains(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_contains(MACHINE_PC, I440FX);
|
||
|
check_contains(MACHINE_PC, PCIBUS_PC);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_produces(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_produces(MACHINE_PC, I440FX);
|
||
|
g_assert_null(qos_graph_get_machine(MACHINE_PC));
|
||
|
g_assert_null(qos_graph_get_machine(I440FX));
|
||
|
g_assert_null(qos_graph_get_node(MACHINE_PC));
|
||
|
g_assert_nonnull(qos_graph_get_node(I440FX));
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_multiple_produces(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_produces(MACHINE_PC, I440FX);
|
||
|
check_produces(MACHINE_PC, PCIBUS_PC);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_consumes(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_consumes(I440FX, SDHCI);
|
||
|
g_assert_null(qos_graph_get_machine(I440FX));
|
||
|
g_assert_null(qos_graph_get_machine(SDHCI));
|
||
|
g_assert_null(qos_graph_get_node(I440FX));
|
||
|
g_assert_nonnull(qos_graph_get_node(SDHCI));
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_multiple_consumes(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_consumes(I440FX, SDHCI);
|
||
|
check_consumes(PCIBUS_PC, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_driver(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_driver(I440FX);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_test(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_test(REGISTER_TEST, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_machine_contains_driver(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
check_driver(I440FX);
|
||
|
check_contains(MACHINE_PC, I440FX);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_driver_contains_driver(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_driver(PCIBUS_PC);
|
||
|
check_driver(I440FX);
|
||
|
check_contains(PCIBUS_PC, I440FX);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_machine_produces_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
check_produces(MACHINE_PC, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_driver_produces_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_driver(I440FX);
|
||
|
check_produces(I440FX, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_machine_consumes_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
check_consumes(MACHINE_PC, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_driver_consumes_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_driver(I440FX);
|
||
|
check_consumes(I440FX, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_test_consumes_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_test(REGISTER_TEST, SDHCI);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_full_sample(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
check_contains(MACHINE_PC, I440FX);
|
||
|
check_driver(I440FX);
|
||
|
check_driver(PCIBUS_PC);
|
||
|
check_contains(I440FX, PCIBUS_PC);
|
||
|
check_produces(PCIBUS_PC, PCIBUS);
|
||
|
check_driver(SDHCI_PCI);
|
||
|
qos_node_consumes(SDHCI_PCI, PCIBUS, NULL);
|
||
|
check_produces(SDHCI_PCI, SDHCI);
|
||
|
check_driver(SDHCI_MM);
|
||
|
check_produces(SDHCI_MM, SDHCI);
|
||
|
qos_add_test(REGISTER_TEST, SDHCI, testfunct, NULL);
|
||
|
check_leaf_discovered(1);
|
||
|
qos_print_graph();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_full_sample_raspi(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_PC);
|
||
|
check_contains(MACHINE_PC, I440FX);
|
||
|
check_driver(I440FX);
|
||
|
check_driver(PCIBUS_PC);
|
||
|
check_contains(I440FX, PCIBUS_PC);
|
||
|
check_produces(PCIBUS_PC, PCIBUS);
|
||
|
check_driver(SDHCI_PCI);
|
||
|
qos_node_consumes(SDHCI_PCI, PCIBUS, NULL);
|
||
|
check_produces(SDHCI_PCI, SDHCI);
|
||
|
check_machine(MACHINE_RASPI2);
|
||
|
check_contains(MACHINE_RASPI2, SDHCI_MM);
|
||
|
check_driver(SDHCI_MM);
|
||
|
check_produces(SDHCI_MM, SDHCI);
|
||
|
qos_add_test(REGISTER_TEST, SDHCI, testfunct, NULL);
|
||
|
qos_print_graph();
|
||
|
check_leaf_discovered(2);
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_cycle(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_RASPI2);
|
||
|
check_driver("B");
|
||
|
check_driver("C");
|
||
|
check_driver("D");
|
||
|
check_contains(MACHINE_RASPI2, "B");
|
||
|
check_contains("B", "C");
|
||
|
check_contains("C", "D");
|
||
|
check_contains("D", MACHINE_RASPI2);
|
||
|
check_leaf_discovered(0);
|
||
|
qos_print_graph();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_two_test_same_interface(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_RASPI2);
|
||
|
check_produces(MACHINE_RASPI2, "B");
|
||
|
qos_add_test("C", "B", testfunct, NULL);
|
||
|
qos_add_test("D", "B", testfunct, NULL);
|
||
|
check_contains(MACHINE_RASPI2, "B");
|
||
|
check_leaf_discovered(4);
|
||
|
qos_print_graph();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_test_in_path(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_RASPI2);
|
||
|
check_produces(MACHINE_RASPI2, "B");
|
||
|
qos_add_test("C", "B", testfunct, NULL);
|
||
|
check_driver("D");
|
||
|
check_consumes("D", "B");
|
||
|
check_produces("D", "E");
|
||
|
qos_add_test("F", "E", testfunct, NULL);
|
||
|
check_leaf_discovered(2);
|
||
|
qos_print_graph();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
static void test_double_edge(void)
|
||
|
{
|
||
|
qos_graph_init();
|
||
|
check_machine(MACHINE_RASPI2);
|
||
|
check_produces("B", "C");
|
||
|
qos_node_consumes("C", "B", NULL);
|
||
|
qos_add_test("D", "C", testfunct, NULL);
|
||
|
check_contains(MACHINE_RASPI2, "B");
|
||
|
qos_print_graph();
|
||
|
qos_graph_destroy();
|
||
|
}
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
g_test_init(&argc, &argv, NULL);
|
||
|
g_test_add_func("/qgraph/init_nop", init_nop);
|
||
|
g_test_add_func("/qgraph/test_machine", test_machine);
|
||
|
g_test_add_func("/qgraph/test_contains", test_contains);
|
||
|
g_test_add_func("/qgraph/test_multiple_contains", test_multiple_contains);
|
||
|
g_test_add_func("/qgraph/test_produces", test_produces);
|
||
|
g_test_add_func("/qgraph/test_multiple_produces", test_multiple_produces);
|
||
|
g_test_add_func("/qgraph/test_consumes", test_consumes);
|
||
|
g_test_add_func("/qgraph/test_multiple_consumes",
|
||
|
test_multiple_consumes);
|
||
|
g_test_add_func("/qgraph/test_driver", test_driver);
|
||
|
g_test_add_func("/qgraph/test_test", test_test);
|
||
|
g_test_add_func("/qgraph/test_machine_contains_driver",
|
||
|
test_machine_contains_driver);
|
||
|
g_test_add_func("/qgraph/test_driver_contains_driver",
|
||
|
test_driver_contains_driver);
|
||
|
g_test_add_func("/qgraph/test_machine_produces_interface",
|
||
|
test_machine_produces_interface);
|
||
|
g_test_add_func("/qgraph/test_driver_produces_interface",
|
||
|
test_driver_produces_interface);
|
||
|
g_test_add_func("/qgraph/test_machine_consumes_interface",
|
||
|
test_machine_consumes_interface);
|
||
|
g_test_add_func("/qgraph/test_driver_consumes_interface",
|
||
|
test_driver_consumes_interface);
|
||
|
g_test_add_func("/qgraph/test_test_consumes_interface",
|
||
|
test_test_consumes_interface);
|
||
|
g_test_add_func("/qgraph/test_full_sample", test_full_sample);
|
||
|
g_test_add_func("/qgraph/test_full_sample_raspi", test_full_sample_raspi);
|
||
|
g_test_add_func("/qgraph/test_cycle", test_cycle);
|
||
|
g_test_add_func("/qgraph/test_two_test_same_interface",
|
||
|
test_two_test_same_interface);
|
||
|
g_test_add_func("/qgraph/test_test_in_path", test_test_in_path);
|
||
|
g_test_add_func("/qgraph/test_double_edge", test_double_edge);
|
||
|
|
||
|
g_test_run();
|
||
|
return 0;
|
||
|
}
|