|  | /* | 
|  | * SPDX-License-Identifier: LGPL-2.1-or-later | 
|  | * | 
|  | * Tests for QTree. | 
|  | * Original source: glib | 
|  | *   https://gitlab.gnome.org/GNOME/glib/-/blob/main/glib/tests/tree.c | 
|  | *   LGPL license. | 
|  | *   Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "qemu/qtree.h" | 
|  |  | 
|  | static gint my_compare(gconstpointer a, gconstpointer b) | 
|  | { | 
|  | const char *cha = a; | 
|  | const char *chb = b; | 
|  |  | 
|  | return *cha - *chb; | 
|  | } | 
|  |  | 
|  | static gint my_compare_with_data(gconstpointer a, | 
|  | gconstpointer b, | 
|  | gpointer user_data) | 
|  | { | 
|  | const char *cha = a; | 
|  | const char *chb = b; | 
|  |  | 
|  | /* just check that we got the right data */ | 
|  | g_assert(GPOINTER_TO_INT(user_data) == 123); | 
|  |  | 
|  | return *cha - *chb; | 
|  | } | 
|  |  | 
|  | static gint my_search(gconstpointer a, gconstpointer b) | 
|  | { | 
|  | return my_compare(b, a); | 
|  | } | 
|  |  | 
|  | static gpointer destroyed_key; | 
|  | static gpointer destroyed_value; | 
|  | static guint destroyed_key_count; | 
|  | static guint destroyed_value_count; | 
|  |  | 
|  | static void my_key_destroy(gpointer key) | 
|  | { | 
|  | destroyed_key = key; | 
|  | destroyed_key_count++; | 
|  | } | 
|  |  | 
|  | static void my_value_destroy(gpointer value) | 
|  | { | 
|  | destroyed_value = value; | 
|  | destroyed_value_count++; | 
|  | } | 
|  |  | 
|  | static gint my_traverse(gpointer key, gpointer value, gpointer data) | 
|  | { | 
|  | char *ch = key; | 
|  |  | 
|  | g_assert((*ch) > 0); | 
|  |  | 
|  | if (*ch == 'd') { | 
|  | return TRUE; | 
|  | } | 
|  |  | 
|  | return FALSE; | 
|  | } | 
|  |  | 
|  | char chars[] = | 
|  | "0123456789" | 
|  | "ABCDEFGHIJKLMNOPQRSTUVWXYZ" | 
|  | "abcdefghijklmnopqrstuvwxyz"; | 
|  |  | 
|  | char chars2[] = | 
|  | "0123456789" | 
|  | "abcdefghijklmnopqrstuvwxyz"; | 
|  |  | 
|  | static gint check_order(gpointer key, gpointer value, gpointer data) | 
|  | { | 
|  | char **p = data; | 
|  | char *ch = key; | 
|  |  | 
|  | g_assert(**p == *ch); | 
|  |  | 
|  | (*p)++; | 
|  |  | 
|  | return FALSE; | 
|  | } | 
|  |  | 
|  | static void test_tree_search(void) | 
|  | { | 
|  | gint i; | 
|  | QTree *tree; | 
|  | gboolean removed; | 
|  | gchar c; | 
|  | gchar *p, *d; | 
|  |  | 
|  | tree = q_tree_new_with_data(my_compare_with_data, GINT_TO_POINTER(123)); | 
|  |  | 
|  | for (i = 0; chars[i]; i++) { | 
|  | q_tree_insert(tree, &chars[i], &chars[i]); | 
|  | } | 
|  |  | 
|  | q_tree_foreach(tree, my_traverse, NULL); | 
|  |  | 
|  | g_assert(q_tree_nnodes(tree) == strlen(chars)); | 
|  | g_assert(q_tree_height(tree) == 6); | 
|  |  | 
|  | p = chars; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | for (i = 0; i < 26; i++) { | 
|  | removed = q_tree_remove(tree, &chars[i + 10]); | 
|  | g_assert(removed); | 
|  | } | 
|  |  | 
|  | c = '\0'; | 
|  | removed = q_tree_remove(tree, &c); | 
|  | g_assert(!removed); | 
|  |  | 
|  | q_tree_foreach(tree, my_traverse, NULL); | 
|  |  | 
|  | g_assert(q_tree_nnodes(tree) == strlen(chars2)); | 
|  | g_assert(q_tree_height(tree) == 6); | 
|  |  | 
|  | p = chars2; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | for (i = 25; i >= 0; i--) { | 
|  | q_tree_insert(tree, &chars[i + 10], &chars[i + 10]); | 
|  | } | 
|  |  | 
|  | p = chars; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | c = '0'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p && *p == c); | 
|  | g_assert(q_tree_lookup_extended(tree, &c, (gpointer *)&d, (gpointer *)&p)); | 
|  | g_assert(c == *d && c == *p); | 
|  |  | 
|  | c = 'A'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = 'a'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = 'z'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = '!'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | c = '='; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | c = '|'; | 
|  | p = q_tree_lookup(tree, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | c = '0'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = 'A'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = 'a'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = 'z'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p && *p == c); | 
|  |  | 
|  | c = '!'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | c = '='; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | c = '|'; | 
|  | p = q_tree_search(tree, my_search, &c); | 
|  | g_assert(p == NULL); | 
|  |  | 
|  | q_tree_destroy(tree); | 
|  | } | 
|  |  | 
|  | static void test_tree_remove(void) | 
|  | { | 
|  | QTree *tree; | 
|  | char c, d; | 
|  | gint i; | 
|  | gboolean removed; | 
|  |  | 
|  | tree = q_tree_new_full((GCompareDataFunc)my_compare, NULL, | 
|  | my_key_destroy, | 
|  | my_value_destroy); | 
|  |  | 
|  | for (i = 0; chars[i]; i++) { | 
|  | q_tree_insert(tree, &chars[i], &chars[i]); | 
|  | } | 
|  |  | 
|  | c = '0'; | 
|  | q_tree_insert(tree, &c, &c); | 
|  | g_assert(destroyed_key == &c); | 
|  | g_assert(destroyed_value == &chars[0]); | 
|  | destroyed_key = NULL; | 
|  | destroyed_value = NULL; | 
|  |  | 
|  | d = '1'; | 
|  | q_tree_replace(tree, &d, &d); | 
|  | g_assert(destroyed_key == &chars[1]); | 
|  | g_assert(destroyed_value == &chars[1]); | 
|  | destroyed_key = NULL; | 
|  | destroyed_value = NULL; | 
|  |  | 
|  | c = '2'; | 
|  | removed = q_tree_remove(tree, &c); | 
|  | g_assert(removed); | 
|  | g_assert(destroyed_key == &chars[2]); | 
|  | g_assert(destroyed_value == &chars[2]); | 
|  | destroyed_key = NULL; | 
|  | destroyed_value = NULL; | 
|  |  | 
|  | c = '3'; | 
|  | removed = q_tree_steal(tree, &c); | 
|  | g_assert(removed); | 
|  | g_assert(destroyed_key == NULL); | 
|  | g_assert(destroyed_value == NULL); | 
|  |  | 
|  | const gchar *remove = "omkjigfedba"; | 
|  | for (i = 0; remove[i]; i++) { | 
|  | removed = q_tree_remove(tree, &remove[i]); | 
|  | g_assert(removed); | 
|  | } | 
|  |  | 
|  | q_tree_destroy(tree); | 
|  | } | 
|  |  | 
|  | static void test_tree_destroy(void) | 
|  | { | 
|  | QTree *tree; | 
|  | gint i; | 
|  |  | 
|  | tree = q_tree_new(my_compare); | 
|  |  | 
|  | for (i = 0; chars[i]; i++) { | 
|  | q_tree_insert(tree, &chars[i], &chars[i]); | 
|  | } | 
|  |  | 
|  | g_assert(q_tree_nnodes(tree) == strlen(chars)); | 
|  |  | 
|  | g_test_message("nnodes: %d", q_tree_nnodes(tree)); | 
|  | q_tree_ref(tree); | 
|  | q_tree_destroy(tree); | 
|  |  | 
|  | g_test_message("nnodes: %d", q_tree_nnodes(tree)); | 
|  | g_assert(q_tree_nnodes(tree) == 0); | 
|  |  | 
|  | q_tree_unref(tree); | 
|  | } | 
|  |  | 
|  | static void test_tree_insert(void) | 
|  | { | 
|  | QTree *tree; | 
|  | gchar *p; | 
|  | gint i; | 
|  | gchar *scrambled; | 
|  |  | 
|  | tree = q_tree_new(my_compare); | 
|  |  | 
|  | for (i = 0; chars[i]; i++) { | 
|  | q_tree_insert(tree, &chars[i], &chars[i]); | 
|  | } | 
|  | p = chars; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | q_tree_unref(tree); | 
|  | tree = q_tree_new(my_compare); | 
|  |  | 
|  | for (i = strlen(chars) - 1; i >= 0; i--) { | 
|  | q_tree_insert(tree, &chars[i], &chars[i]); | 
|  | } | 
|  | p = chars; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | q_tree_unref(tree); | 
|  | tree = q_tree_new(my_compare); | 
|  |  | 
|  | scrambled = g_strdup(chars); | 
|  |  | 
|  | for (i = 0; i < 30; i++) { | 
|  | gchar tmp; | 
|  | gint a, b; | 
|  |  | 
|  | a = g_random_int_range(0, strlen(scrambled)); | 
|  | b = g_random_int_range(0, strlen(scrambled)); | 
|  | tmp = scrambled[a]; | 
|  | scrambled[a] = scrambled[b]; | 
|  | scrambled[b] = tmp; | 
|  | } | 
|  |  | 
|  | for (i = 0; scrambled[i]; i++) { | 
|  | q_tree_insert(tree, &scrambled[i], &scrambled[i]); | 
|  | } | 
|  | p = chars; | 
|  | q_tree_foreach(tree, check_order, &p); | 
|  |  | 
|  | g_free(scrambled); | 
|  | q_tree_unref(tree); | 
|  | } | 
|  |  | 
|  | int main(int argc, char *argv[]) | 
|  | { | 
|  | g_test_init(&argc, &argv, NULL); | 
|  |  | 
|  | g_test_add_func("/qtree/search", test_tree_search); | 
|  | g_test_add_func("/qtree/remove", test_tree_remove); | 
|  | g_test_add_func("/qtree/destroy", test_tree_destroy); | 
|  | g_test_add_func("/qtree/insert", test_tree_insert); | 
|  |  | 
|  | return g_test_run(); | 
|  | } |