/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * * Copyright (C) 2014 Red Hat, Inc. * */ #include "config.h" #include #include #include #include #include #include #include "NetworkManagerUtils.h" #include "nm-logging.h" #include "nm-multi-index.h" #include "nm-test-utils.h" /*******************************************/ static void test_nm_utils_monotonic_timestamp_as_boottime (void) { gint64 timestamp_ns_per_tick, now, now_boottime, now_boottime_2, now_boottime_3; struct timespec tp; clockid_t clockid; guint i; if (clock_gettime (CLOCK_BOOTTIME, &tp) != 0 && errno == EINVAL) clockid = CLOCK_MONOTONIC; else clockid = CLOCK_BOOTTIME; for (i = 0; i < 10; i++) { if (clock_gettime (clockid, &tp) != 0) g_assert_not_reached (); now_boottime = ( ((gint64) tp.tv_sec) * NM_UTILS_NS_PER_SECOND ) + ((gint64) tp.tv_nsec); now = nm_utils_get_monotonic_timestamp_ns (); now_boottime_2 = nm_utils_monotonic_timestamp_as_boottime (now, 1); g_assert_cmpint (now_boottime_2, >=, 0); g_assert_cmpint (now_boottime_2, >=, now_boottime); g_assert_cmpint (now_boottime_2 - now_boottime, <=, NM_UTILS_NS_PER_SECOND / 1000); for (timestamp_ns_per_tick = 1; timestamp_ns_per_tick <= NM_UTILS_NS_PER_SECOND; timestamp_ns_per_tick *= 10) { now_boottime_3 = nm_utils_monotonic_timestamp_as_boottime (now / timestamp_ns_per_tick, timestamp_ns_per_tick); g_assert_cmpint (now_boottime_2 / timestamp_ns_per_tick, ==, now_boottime_3); } } } /*******************************************/ struct test_nm_utils_kill_child_async_data { GMainLoop *loop; pid_t pid; gboolean called; gboolean expected_success; const int *expected_child_status; }; static void test_nm_utils_kill_child_async_cb (pid_t pid, gboolean success, int child_status, void *user_data) { struct test_nm_utils_kill_child_async_data *data = user_data; g_assert (success == !!data->expected_success); g_assert (pid == data->pid); if (data->expected_child_status) g_assert_cmpint (*data->expected_child_status, ==, child_status); if (!success) g_assert_cmpint (child_status, ==, -1); data->called = TRUE; g_assert (data->loop); g_main_loop_quit (data->loop); } static gboolean test_nm_utils_kill_child_async_fail_cb (void *user_data) { g_assert_not_reached (); } static void test_nm_utils_kill_child_async_do (const char *name, pid_t pid, int sig, guint32 wait_before_kill_msec, gboolean expected_success, const int *expected_child_status) { gboolean success; struct test_nm_utils_kill_child_async_data data = { }; int timeout_id; data.pid = pid; data.expected_success = expected_success; data.expected_child_status = expected_child_status; nm_utils_kill_child_async (pid, sig, LOGD_CORE, name, wait_before_kill_msec, test_nm_utils_kill_child_async_cb, &data); g_assert (!data.called); timeout_id = g_timeout_add_seconds (5, test_nm_utils_kill_child_async_fail_cb, &data); data.loop = g_main_loop_new (NULL, FALSE); g_main_run (data.loop); g_assert (data.called); success = g_source_remove (timeout_id); g_assert (success); g_main_destroy (data.loop); } static void test_nm_utils_kill_child_sync_do (const char *name, pid_t pid, int sig, guint32 wait_before_kill_msec, gboolean expected_success, const int *expected_child_status) { gboolean success; int child_status = -1; success = nm_utils_kill_child_sync (pid, sig, LOGD_CORE, name, &child_status, wait_before_kill_msec, 0); g_assert (success == !!expected_success); if (expected_child_status) g_assert_cmpint (*expected_child_status, ==, child_status); g_test_assert_expected_messages (); } static pid_t test_nm_utils_kill_child_spawn (char **argv, gboolean do_not_reap_child) { GError *error = NULL; int success; GPid child_pid; success = g_spawn_async (NULL, argv, NULL, G_SPAWN_SEARCH_PATH | (do_not_reap_child ? G_SPAWN_DO_NOT_REAP_CHILD : 0), NULL, NULL, &child_pid, &error); g_assert (success && !error); return child_pid; } static pid_t test_nm_utils_kill_child_create_and_join_pgroup (void) { int err, tmp = 0; int pipefd[2]; pid_t pgid; err = pipe (pipefd); g_assert (err == 0); pgid = fork(); if (pgid < 0) { g_assert_not_reached (); return pgid; } if (pgid == 0) { /* child process... */ close (pipefd[0]); err = setpgid (0, 0); g_assert (err == 0); err = write (pipefd[1], &tmp, sizeof (tmp)); g_assert (err == sizeof (tmp)); close (pipefd[1]); exit (0); } close (pipefd[1]); err = read (pipefd[0], &tmp, sizeof (tmp)); g_assert (err == sizeof (tmp)); close (pipefd[0]); err = setpgid (0, pgid); g_assert (err == 0); do { err = waitpid (pgid, &tmp, 0); } while (err == -1 && errno == EINTR); g_assert (err == pgid); g_assert (WIFEXITED (tmp) && WEXITSTATUS(tmp) == 0); return pgid; } #define TEST_TOKEN "nm_test_kill_child_process" static void test_nm_utils_kill_child (void) { int err; GLogLevelFlags fatal_mask; char *argv_watchdog[] = { "sh", "-c", "sleep 4; " "kill -KILL 0; #watchdog for #" TEST_TOKEN, NULL, }; char *argv1[] = { "sh", "-c", "trap \"sleep 0.3; exit 10\" EXIT; " "sleep 100000; exit $? #" TEST_TOKEN, NULL, }; char *argv2[] = { "sh", "-c", "exit 47; #" TEST_TOKEN, NULL, }; char *argv3[] = { "sh", "-c", "trap \"exit 47\" TERM; while true; do :; done; #" TEST_TOKEN, NULL, }; char *argv4[] = { "sh", "-c", "trap \"while true; do :; done\" TERM; while true; do :; done; #" TEST_TOKEN, NULL, }; pid_t gpid; pid_t pid1a_1, pid1a_2, pid1a_3, pid2a, pid3a, pid4a; pid_t pid1s_1, pid1s_2, pid1s_3, pid2s, pid3s, pid4s; const int expected_exit_47 = 12032; /* exit with status 47 */ const int expected_signal_TERM = SIGTERM; const int expected_signal_KILL = SIGKILL; gpid = test_nm_utils_kill_child_create_and_join_pgroup (); test_nm_utils_kill_child_spawn (argv_watchdog, FALSE); pid1s_1 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid1s_2 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid1s_3 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid2s = test_nm_utils_kill_child_spawn (argv2, TRUE); pid3s = test_nm_utils_kill_child_spawn (argv3, TRUE); pid4s = test_nm_utils_kill_child_spawn (argv4, TRUE); pid1a_1 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid1a_2 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid1a_3 = test_nm_utils_kill_child_spawn (argv1, TRUE); pid2a = test_nm_utils_kill_child_spawn (argv2, TRUE); pid3a = test_nm_utils_kill_child_spawn (argv3, TRUE); pid4a = test_nm_utils_kill_child_spawn (argv4, TRUE); /* give processes time to start (and potentially block signals) ... */ g_usleep (G_USEC_PER_SEC / 10); fatal_mask = g_log_set_always_fatal (G_LOG_FATAL_MASK); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-1' (*): waiting up to 500 milliseconds for process to terminate normally after sending SIGTERM (15)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-1' (*): after sending SIGTERM (15), process * exited by signal 15 (* usec elapsed)"); test_nm_utils_kill_child_sync_do ("test-s-1-1", pid1s_1, SIGTERM, 1000 / 2, TRUE, &expected_signal_TERM); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-2' (*): waiting for process to terminate after sending SIGKILL (9)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-2' (*): after sending SIGKILL (9), process * exited by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_sync_do ("test-s-1-2", pid1s_2, SIGKILL, 1000 / 2, TRUE, &expected_signal_KILL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-3' (*): waiting up to 1 milliseconds for process to terminate normally after sending no signal (0)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-3' (*): sending SIGKILL..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-1-3' (*): after sending no signal (0) and SIGKILL, process * exited by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_sync_do ("test-s-1-3", pid1s_3, 0, 1, TRUE, &expected_signal_KILL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-2' (*): process * already terminated normally with status 47"); test_nm_utils_kill_child_sync_do ("test-s-2", pid2s, SIGTERM, 1000 / 2, TRUE, &expected_exit_47); /* send invalid signal. */ g_test_expect_message ("NetworkManager", G_LOG_LEVEL_WARNING, "*kill child process 'test-s-3-0' (*): failed to send Unexpected signal: Invalid argument (22)"); test_nm_utils_kill_child_sync_do ("test-s-3-0", pid3s, -1, 0, FALSE, NULL); /* really kill pid3s */ g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-3-1' (*): waiting up to 500 milliseconds for process to terminate normally after sending SIGTERM (15)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-3-1' (*): after sending SIGTERM (15), process * exited normally with status 47 (* usec elapsed)"); test_nm_utils_kill_child_sync_do ("test-s-3-1", pid3s, SIGTERM, 1000 / 2, TRUE, &expected_exit_47); /* pid3s should not be a valid process, hence the call should fail. Note, that there * is a race here. */ g_test_expect_message ("NetworkManager", G_LOG_LEVEL_WARNING, "*kill child process 'test-s-3-2' (*): failed due to unexpected return value -1 by waitpid (No child processes, 10) after sending no signal (0)"); test_nm_utils_kill_child_sync_do ("test-s-3-2", pid3s, 0, 0, FALSE, NULL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-4' (*): waiting up to 1 milliseconds for process to terminate normally after sending SIGTERM (15)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-4' (*): sending SIGKILL..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-s-4' (*): after sending SIGTERM (15) and SIGKILL, process * exited by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_sync_do ("test-s-4", pid4s, SIGTERM, 1, TRUE, &expected_signal_KILL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-1' (*): wait for process to terminate after sending SIGTERM (15) (send SIGKILL in 500 milliseconds)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-1' (*): terminated by signal 15 (* usec elapsed)"); test_nm_utils_kill_child_async_do ("test-a-1-1", pid1a_1, SIGTERM, 1000 / 2, TRUE, &expected_signal_TERM); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-2' (*): wait for process to terminate after sending SIGKILL (9)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-2' (*): terminated by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_async_do ("test-a-1-2", pid1a_2, SIGKILL, 1000 / 2, TRUE, &expected_signal_KILL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-3' (*): wait for process to terminate after sending no signal (0) (send SIGKILL in 1 milliseconds)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-3' (*): process not terminated after * usec. Sending SIGKILL signal"); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-1-3' (*): terminated by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_async_do ("test-a-1-3", pid1a_3, 0, 1, TRUE, &expected_signal_KILL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-2' (*): process * already terminated normally with status 47"); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-2' (*): invoke callback: terminated normally with status 47"); test_nm_utils_kill_child_async_do ("test-a-2", pid2a, SIGTERM, 1000 / 2, TRUE, &expected_exit_47); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_WARNING, "*kill child process 'test-a-3-0' (*): unexpected error sending Unexpected signal: Invalid argument (22)"); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-3-0' (*): invoke callback: killing child failed"); /* coverity[negative_returns] */ test_nm_utils_kill_child_async_do ("test-a-3-0", pid3a, -1, 1000 / 2, FALSE, NULL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-3-1' (*): wait for process to terminate after sending SIGTERM (15) (send SIGKILL in 500 milliseconds)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-3-1' (*): terminated normally with status 47 (* usec elapsed)"); test_nm_utils_kill_child_async_do ("test-a-3-1", pid3a, SIGTERM, 1000 / 2, TRUE, &expected_exit_47); /* pid3a should not be a valid process, hence the call should fail. Note, that there * is a race here. */ g_test_expect_message ("NetworkManager", G_LOG_LEVEL_WARNING, "*kill child process 'test-a-3-2' (*): failed due to unexpected return value -1 by waitpid (No child processes, 10) after sending no signal (0)"); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-3-2' (*): invoke callback: killing child failed"); test_nm_utils_kill_child_async_do ("test-a-3-2", pid3a, 0, 0, FALSE, NULL); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-4' (*): wait for process to terminate after sending SIGTERM (15) (send SIGKILL in 1 milliseconds)..."); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-4' (*): process not terminated after * usec. Sending SIGKILL signal"); g_test_expect_message ("NetworkManager", G_LOG_LEVEL_DEBUG, "*kill child process 'test-a-4' (*): terminated by signal 9 (* usec elapsed)"); test_nm_utils_kill_child_async_do ("test-a-4", pid4a, SIGTERM, 1, TRUE, &expected_signal_KILL); err = setpgid (0, 0); g_assert (err == 0); kill (-gpid, SIGKILL); g_log_set_always_fatal (fatal_mask); g_test_assert_expected_messages (); } /*******************************************/ static void _remove_at_indexes_init_random_idx (GArray *idx, guint array_len, guint idx_len) { GRand *rand = nmtst_get_rand (); gs_free char *mask = NULL; guint i, max_test_idx; g_assert (idx); g_assert (array_len > 0); g_assert (idx_len >= 1 && idx_len <= array_len); mask = g_new0 (char, array_len); max_test_idx = array_len - 1; for (i = 0; i < idx_len; i++) { guint itest; /* find a index itest that is not yet taken */ if (max_test_idx == 0) itest = 0; else itest = g_rand_int_range (rand, 0, max_test_idx); while (itest < array_len && mask[itest]) itest++; g_assert (itest <= max_test_idx); g_assert (!mask[itest]); mask[itest] = TRUE; if (itest == max_test_idx) { g_assert (max_test_idx > 0 || i == idx_len - 1); if (max_test_idx == 0) g_assert_cmpint (i, ==, idx_len - 1); else { max_test_idx--; while (max_test_idx > 0 && mask[max_test_idx]) max_test_idx--; if (mask[max_test_idx]) g_assert_cmpint (i, ==, idx_len - 1); } } } g_array_set_size (idx, 0); for (i = 0; i < array_len; i++) { if (mask[i]) g_array_append_val (idx, i); } g_assert_cmpint (idx->len, ==, idx_len); } static void test_nm_utils_array_remove_at_indexes (void) { gs_unref_array GArray *idx = NULL, *array = NULL; gs_unref_hashtable GHashTable *unique = NULL; guint i_len, i_idx_len, i_rnd, i; idx = g_array_new (FALSE, FALSE, sizeof (guint)); array = g_array_new (FALSE, FALSE, sizeof (gssize)); unique = g_hash_table_new (NULL, NULL); for (i_len = 1; i_len < 20; i_len++) { for (i_idx_len = 1; i_idx_len <= i_len; i_idx_len++) { for (i_rnd = 0; i_rnd < 20; i_rnd++) { _remove_at_indexes_init_random_idx (idx, i_len, i_idx_len); g_array_set_size (array, i_len); for (i = 0; i < i_len; i++) g_array_index (array, gssize, i) = i; nm_utils_array_remove_at_indexes (array, &g_array_index (idx, guint, 0), i_idx_len); g_hash_table_remove_all (unique); /* ensure that all the indexes are still unique */ for (i = 0; i < array->len; i++) g_hash_table_add (unique, GUINT_TO_POINTER (g_array_index (array, gssize, i))); g_assert_cmpint (g_hash_table_size (unique), ==, array->len); for (i = 0; i < idx->len; i++) g_hash_table_add (unique, GUINT_TO_POINTER (g_array_index (idx, guint, i))); g_assert_cmpint (g_hash_table_size (unique), ==, i_len); /* ensure proper sort order in array */ for (i = 0; i < array->len; i++) { gssize i1 = g_array_index (array, gssize, i); g_assert (i1 >= 0 && i1 < i_len); if (i > 0) { gsize i0 = g_array_index (array, gssize, i - 1); g_assert_cmpint (i0, <, i1); } } } } } } /*******************************************/ static void test_nm_ethernet_address_is_valid (void) { g_assert (!nm_ethernet_address_is_valid (NULL, -1)); g_assert (!nm_ethernet_address_is_valid (NULL, ETH_ALEN)); g_assert (!nm_ethernet_address_is_valid ("FF:FF:FF:FF:FF:FF", -1)); g_assert (!nm_ethernet_address_is_valid ("00:00:00:00:00:00", -1)); g_assert (!nm_ethernet_address_is_valid ("44:44:44:44:44:44", -1)); g_assert (!nm_ethernet_address_is_valid ("00:30:b4:00:00:00", -1)); g_assert (!nm_ethernet_address_is_valid ("", -1)); g_assert (!nm_ethernet_address_is_valid ("1", -1)); g_assert (!nm_ethernet_address_is_valid ("2", -1)); g_assert (!nm_ethernet_address_is_valid (((guint8[8]) { 0x00,0x30,0xb4,0x00,0x00,0x00 }), ETH_ALEN)); g_assert ( nm_ethernet_address_is_valid (((guint8[8]) { 0x00,0x30,0xb4,0x00,0x00,0x01 }), ETH_ALEN)); /* some Broad cast addresses (with MSB of first octet set). */ g_assert (!nm_ethernet_address_is_valid ("57:44:44:44:44:44", -1)); g_assert ( nm_ethernet_address_is_valid ("56:44:44:44:44:44", -1)); g_assert (!nm_ethernet_address_is_valid (((guint8[8]) { 0x03,0x30,0xb4,0x00,0x00,0x00 }), ETH_ALEN)); g_assert ( nm_ethernet_address_is_valid (((guint8[8]) { 0x02,0x30,0xb4,0x00,0x00,0x01 }), ETH_ALEN)); } /*******************************************/ typedef struct { union { NMMultiIndexId id_base; guint bucket; }; } NMMultiIndexIdTest; typedef struct { guint64 buckets; gpointer ptr_value; } NMMultiIndexTestValue; static gboolean _mi_value_bucket_has (const NMMultiIndexTestValue *value, guint bucket) { g_assert (value); g_assert (bucket < 64); return (value->buckets & (((guint64) 0x01) << bucket)) != 0; } static gboolean _mi_value_bucket_set (NMMultiIndexTestValue *value, guint bucket) { g_assert (value); g_assert (bucket < 64); if (_mi_value_bucket_has (value, bucket)) return FALSE; value->buckets |= (((guint64) 0x01) << bucket); return TRUE; } static gboolean _mi_value_bucket_unset (NMMultiIndexTestValue *value, guint bucket) { g_assert (value); g_assert (bucket < 64); if (!_mi_value_bucket_has (value, bucket)) return FALSE; value->buckets &= ~(((guint64) 0x01) << bucket); return TRUE; } static guint _mi_idx_hash (const NMMultiIndexIdTest *id) { g_assert (id && id->bucket < 64); return id->bucket; } static gboolean _mi_idx_equal (const NMMultiIndexIdTest *a, const NMMultiIndexIdTest *b) { g_assert (a && a->bucket < 64); g_assert (b && b->bucket < 64); return a->bucket == b->bucket; } static NMMultiIndexIdTest * _mi_idx_clone (const NMMultiIndexIdTest *id) { NMMultiIndexIdTest *n; g_assert (id && id->bucket < 64); n = g_new0 (NMMultiIndexIdTest, 1); n->bucket = id->bucket; return n; } static void _mi_idx_destroy (NMMultiIndexIdTest *id) { g_assert (id && id->bucket < 64); g_free (id); } static NMMultiIndexTestValue * _mi_create_array (guint num_values) { NMMultiIndexTestValue *array = g_new0 (NMMultiIndexTestValue, num_values); guint i; g_assert (num_values > 0); for (i = 0; i < num_values; i++) { array[i].buckets = 0; array[i].ptr_value = GUINT_TO_POINTER (i + 1); } return array; } typedef struct { guint num_values; guint num_buckets; NMMultiIndexTestValue *array; int test_idx; } NMMultiIndexAssertData; static gboolean _mi_assert_index_equals_array_cb (const NMMultiIndexIdTest *id, void *const* values, guint len, NMMultiIndexAssertData *data) { guint i; gboolean has_test_idx = FALSE; g_assert (id && id->bucket < 64); g_assert (data); g_assert (values); g_assert (len > 0); g_assert (values[len] == NULL); g_assert (data->test_idx >= -1 || data->test_idx < data->num_buckets); g_assert (id->bucket < data->num_buckets); for (i = 0; i < data->num_values; i++) g_assert (!_mi_value_bucket_has (&data->array[i], id->bucket)); for (i = 0; i < len; i++) { guint vi = GPOINTER_TO_UINT (values[i]); g_assert (vi >= 1); g_assert (vi <= data->num_values); vi--; if (data->test_idx == vi) has_test_idx = TRUE; g_assert (data->array[vi].ptr_value == values[i]); if (!_mi_value_bucket_set (&data->array[vi], id->bucket)) g_assert_not_reached (); } g_assert ((data->test_idx == -1 && !has_test_idx) || has_test_idx); return TRUE; } static void _mi_assert_index_equals_array (guint num_values, guint num_buckets, int test_idx, const NMMultiIndexTestValue *array, const NMMultiIndex *index) { NMMultiIndexAssertData data = { .num_values = num_values, .num_buckets = num_buckets, .test_idx = test_idx, }; NMMultiIndexIter iter; const NMMultiIndexIdTest *id; void *const* values; guint len; NMMultiIndexTestValue *v; data.array = _mi_create_array (num_values); v = test_idx >= 0 ? data.array[test_idx].ptr_value : NULL; nm_multi_index_foreach (index, v, (NMMultiIndexFuncForeach) _mi_assert_index_equals_array_cb, &data); if (test_idx >= 0) g_assert (memcmp (&data.array[test_idx], &array[test_idx], sizeof (NMMultiIndexTestValue)) == 0); else g_assert (memcmp (data.array, array, sizeof (NMMultiIndexTestValue) * num_values) == 0); g_free (data.array); data.array = _mi_create_array (num_values); v = test_idx >= 0 ? data.array[test_idx].ptr_value : NULL; nm_multi_index_iter_init (&iter, index, v); while (nm_multi_index_iter_next (&iter, (gpointer) &id, &values, &len)) _mi_assert_index_equals_array_cb (id, values, len, &data); if (test_idx >= 0) g_assert (memcmp (&data.array[test_idx], &array[test_idx], sizeof (NMMultiIndexTestValue)) == 0); else g_assert (memcmp (data.array, array, sizeof (NMMultiIndexTestValue) * num_values) == 0); g_free (data.array); } typedef enum { MI_OP_ADD, MI_OP_REMOVE, MI_OP_MOVE, } NMMultiIndexOperation; static void _mi_rebucket (GRand *rand, guint num_values, guint num_buckets, NMMultiIndexOperation op, guint bucket, guint bucket_old, guint array_idx, NMMultiIndexTestValue *array, NMMultiIndex *index) { NMMultiIndexTestValue *v; NMMultiIndexIdTest id, id_old; const NMMultiIndexIdTest *id_reverse; guint64 buckets_old; guint i; gboolean had_bucket, had_bucket_old; g_assert (array_idx < num_values); g_assert (bucket < (int) num_buckets); v = &array[array_idx]; buckets_old = v->buckets; if (op == MI_OP_MOVE) had_bucket_old = _mi_value_bucket_has (v, bucket_old); else had_bucket_old = FALSE; had_bucket = _mi_value_bucket_has (v, bucket); switch (op) { case MI_OP_ADD: _mi_value_bucket_set (v, bucket); id.bucket = bucket; if (nm_multi_index_add (index, &id.id_base, v->ptr_value)) g_assert (!had_bucket); else g_assert (had_bucket); break; case MI_OP_REMOVE: _mi_value_bucket_unset (v, bucket); id.bucket = bucket; if (nm_multi_index_remove (index, &id.id_base, v->ptr_value)) g_assert (had_bucket); else g_assert (!had_bucket); break; case MI_OP_MOVE: _mi_value_bucket_unset (v, bucket_old); _mi_value_bucket_set (v, bucket); id.bucket = bucket; id_old.bucket = bucket_old; if (nm_multi_index_move (index, &id_old.id_base, &id.id_base, v->ptr_value)) { if (bucket == bucket_old) g_assert (had_bucket_old && had_bucket); else g_assert (had_bucket_old && !had_bucket); } else { if (bucket == bucket_old) g_assert (!had_bucket_old && !had_bucket); else g_assert (!had_bucket_old || had_bucket); } break; default: g_assert_not_reached (); } #if 0 g_print (">>> rebucket: idx=%3u, op=%3s, bucket=%3i%c -> %3i%c, buckets=%08llx -> %08llx %s\n", array_idx, op == MI_OP_ADD ? "ADD" : (op == MI_OP_REMOVE ? "REM" : "MOV"), bucket_old, had_bucket_old ? '*' : ' ', bucket, had_bucket ? '*' : ' ', (long long unsigned) buckets_old, (long long unsigned) v->buckets, buckets_old != v->buckets ? "(changed)" : "(unchanged)"); #endif id_reverse = (const NMMultiIndexIdTest *) nm_multi_index_lookup_first_by_value (index, v->ptr_value); if (id_reverse) g_assert (_mi_value_bucket_has (v, id_reverse->bucket)); else g_assert (v->buckets == 0); for (i = 0; i < 64; i++) { id.bucket = i; if (nm_multi_index_contains (index, &id.id_base, v->ptr_value)) g_assert (_mi_value_bucket_has (v, i)); else g_assert (!_mi_value_bucket_has (v, i)); } _mi_assert_index_equals_array (num_values, num_buckets, -1, array, index); _mi_assert_index_equals_array (num_values, num_buckets, array_idx, array, index); _mi_assert_index_equals_array (num_values, num_buckets, g_rand_int_range (rand, 0, num_values), array, index); } static void _mi_test_run (guint num_values, guint num_buckets) { NMMultiIndex *index = nm_multi_index_new ((NMMultiIndexFuncHash) _mi_idx_hash, (NMMultiIndexFuncEqual) _mi_idx_equal, (NMMultiIndexFuncClone) _mi_idx_clone, (NMMultiIndexFuncDestroy) _mi_idx_destroy); gs_free NMMultiIndexTestValue *array = _mi_create_array (num_values); GRand *rand = nmtst_get_rand (); guint i, i_rd, i_idx, i_bucket; guint num_buckets_all = num_values * num_buckets; g_assert (array[0].ptr_value == GUINT_TO_POINTER (1)); _mi_assert_index_equals_array (num_values, num_buckets, -1, array, index); _mi_rebucket (rand, num_values, num_buckets, MI_OP_ADD, 0, 0, 0, array, index); _mi_rebucket (rand, num_values, num_buckets, MI_OP_REMOVE, 0, 0, 0, array, index); if (num_buckets >= 3) { _mi_rebucket (rand, num_values, num_buckets, MI_OP_ADD, 0, 0, 0, array, index); _mi_rebucket (rand, num_values, num_buckets, MI_OP_MOVE, 2, 0, 0, array, index); _mi_rebucket (rand, num_values, num_buckets, MI_OP_REMOVE, 2, 0, 0, array, index); } g_assert (nm_multi_index_get_num_groups (index) == 0); /* randomly change the bucket of entries. */ for (i = 0; i < 5 * num_values; i++) { guint array_idx = g_rand_int_range (rand, 0, num_values); guint bucket = g_rand_int_range (rand, 0, num_buckets); NMMultiIndexOperation op = g_rand_int_range (rand, 0, MI_OP_MOVE + 1); guint bucket_old = 0; if (op == MI_OP_MOVE) { if ((g_rand_int (rand) % 2) && array[array_idx].buckets != 0) { guint64 b; /* choose the highest (existing) bucket. */ bucket_old = 0; for (b = array[array_idx].buckets; b; b >>= 1) bucket_old++; } else { /* choose a random bucket (even if the item is currently not in that bucket). */ bucket_old = g_rand_int_range (rand, 0, num_buckets); } } _mi_rebucket (rand, num_values, num_buckets, op, bucket, bucket_old, array_idx, array, index); } /* remove all elements from all buckets */ i_rd = g_rand_int (rand); for (i = 0; i < num_buckets_all; i++) { i_rd = (i_rd + 101) % num_buckets_all; i_idx = i_rd / num_buckets; i_bucket = i_rd % num_buckets; if (_mi_value_bucket_has (&array[i_idx], i_bucket)) _mi_rebucket (rand, num_values, num_buckets, MI_OP_REMOVE, i_bucket, 0, i_idx, array, index); } g_assert (nm_multi_index_get_num_groups (index) == 0); nm_multi_index_free (index); } static void test_nm_multi_index (void) { guint i, j; for (i = 1; i < 7; i++) { for (j = 1; j < 6; j++) _mi_test_run (i, j); } _mi_test_run (50, 3); _mi_test_run (50, 18); } /*******************************************/ NMTST_DEFINE (); int main (int argc, char **argv) { nmtst_init_assert_logging (&argc, &argv, "DEBUG", "DEFAULT"); g_test_add_func ("/general/nm_utils_monotonic_timestamp_as_boottime", test_nm_utils_monotonic_timestamp_as_boottime); g_test_add_func ("/general/nm_utils_kill_child", test_nm_utils_kill_child); g_test_add_func ("/general/nm_utils_array_remove_at_indexes", test_nm_utils_array_remove_at_indexes); g_test_add_func ("/general/nm_ethernet_address_is_valid", test_nm_ethernet_address_is_valid); g_test_add_func ("/general/nm_multi_index", test_nm_multi_index); return g_test_run (); }