[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[PATCH 5/6] tests/qtest/qos-test: Plug a couple of leaks
From: |
Fabiano Rosas |
Subject: |
[PATCH 5/6] tests/qtest/qos-test: Plug a couple of leaks |
Date: |
Mon, 9 Dec 2024 17:44:26 -0300 |
The walk_path() function of qos-test.c, which walks the graph and adds
tests to the test suite uses GLib's g_test_add_data_func_full()
function:
g_test_add_data_func_full (const char *testpath,
gpointer test_data,
GTestDataFunc test_func,
GDestroyNotify data_free_func)
Despite GLib's documentation stating that @data_free_func is a
destructor for @test_data, this is not the case. The destructor is
supposed to be paired with a constructor, which GLib only accepts via
g_test_create_case().
Providing externally allocated data plus a destructor function only
works if the test is guaranteed to execute, otherwise the test_data is
never deallocated.
Due to how subprocessess are implemented in qos-test, each test gets
added twice and an extra test gets added per subprocess. In a regular
run, the extra subprocess will not be executed and in a single test
run (-p), none of the other tests will be executed (+1 per
subprocess), leaking 'path_vec' and 'subprocess_path'.
Fix this by storing all the path vectors in a list and freeing them
all at the end of the program (including subprocess invocations) and
moving the allocation of 'subprocess_path' into run_one_subprocess().
While here add some documentation explaining why the graph needs to be
walked twice and tests re-added.
Signed-off-by: Fabiano Rosas <farosas@suse.de>
---
tests/qtest/qos-test.c | 35 +++++++++++++++++++++++++----------
1 file changed, 25 insertions(+), 10 deletions(-)
diff --git a/tests/qtest/qos-test.c b/tests/qtest/qos-test.c
index 114f6bef27..67abc660b5 100644
--- a/tests/qtest/qos-test.c
+++ b/tests/qtest/qos-test.c
@@ -31,6 +31,7 @@
#include "libqos/qos_external.h"
static char *old_path;
+static GSList *path_vecs;
/**
@@ -183,11 +184,16 @@ static void run_one_test(const void *arg)
static void subprocess_run_one_test(const void *arg)
{
- const gchar *path = arg;
- g_test_trap_subprocess(path, 180 * G_USEC_PER_SEC,
+ char **path_vec = (char **) arg;
+ gchar *path = g_strjoinv("/", path_vec + 1);
+ gchar *subprocess_path = g_strdup_printf("/%s/subprocess", path);
+
+ g_test_trap_subprocess(subprocess_path, 180 * G_USEC_PER_SEC,
G_TEST_SUBPROCESS_INHERIT_STDOUT |
G_TEST_SUBPROCESS_INHERIT_STDERR);
g_test_trap_assert_passed();
+ g_free(path);
+ g_free(subprocess_path);
}
static void destroy_pathv(void *arg)
@@ -239,6 +245,7 @@ static void walk_path(QOSGraphNode *orig_path, int len)
GString *cmd_line = g_string_new("");
GString *cmd_line2 = g_string_new("");
+ path_vecs = g_slist_append(path_vecs, path_vec);
path = qos_graph_get_node(node_name); /* root */
node_name = qos_graph_edge_get_dest(path->path_edge); /* machine name */
@@ -298,15 +305,15 @@ static void walk_path(QOSGraphNode *orig_path, int len)
path_vec[0] = g_string_free(cmd_line, false);
if (path->u.test.subprocess) {
- gchar *subprocess_path = g_strdup_printf("/%s/%s/subprocess",
- qtest_get_arch(), path_str);
- qtest_add_data_func_full(path_str, subprocess_path,
- subprocess_run_one_test, g_free);
- g_test_add_data_func_full(subprocess_path, path_vec,
- run_one_test, destroy_pathv);
+ gchar *subprocess_path = g_strdup_printf("%s/%s", path_str,
+ "subprocess");
+
+ qtest_add_data_func(path_str, path_vec, subprocess_run_one_test);
+ qtest_add_data_func(subprocess_path, path_vec, run_one_test);
+
+ g_free(subprocess_path);
} else {
- qtest_add_data_func_full(path_str, path_vec,
- run_one_test, destroy_pathv);
+ qtest_add_data_func(path_str, path_vec, run_one_test);
}
g_free(path_str);
@@ -332,6 +339,13 @@ int main(int argc, char **argv, char** envp)
if (g_test_subprocess()) {
qos_printf("qos_test running single test in subprocess\n");
+
+ /*
+ * Although this invocation was done to run a single test in a
+ * subprocess, gtester doesn't expose the test name, so still
+ * execute the rest of main(), including adding all tests once
+ * more in order for g_test_run() to find the /subprocess.
+ */
}
if (g_test_verbose()) {
@@ -354,5 +368,6 @@ int main(int argc, char **argv, char** envp)
qtest_end();
qos_graph_destroy();
g_free(old_path);
+ g_slist_free_full(path_vecs, (GDestroyNotify)destroy_pathv);
return 0;
}
--
2.35.3
- [PATCH 0/6] qtest: Fix some memory issues, Fabiano Rosas, 2024/12/09
- [PATCH 1/6] tests/qtest/migration: Do proper cleanup in the dirty_limit test, Fabiano Rosas, 2024/12/09
- [PATCH 2/6] tests/qtest/migration: Initialize buffer in probe_o_direct_support, Fabiano Rosas, 2024/12/09
- [PATCH 3/6] tests/qtest/bios-tables-test: Free tables at dump_aml_files, Fabiano Rosas, 2024/12/09
- [PATCH 5/6] tests/qtest/qos-test: Plug a couple of leaks,
Fabiano Rosas <=
- [PATCH 6/6] tests/qtest/test-x86-cpuid-compat: Free allocated memory, Fabiano Rosas, 2024/12/09
- [PATCH 4/6] tests/qtest/virtio-iommu-test: Don't pass uninitialized data into qtest_memwrite, Fabiano Rosas, 2024/12/09
- Re: [PATCH 0/6] qtest: Fix some memory issues, Fabiano Rosas, 2024/12/17