Blame src/core/devices/tests/test-acd.c

Packit Service 5ffa24
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit Service 5ffa24
/*
Packit Service 5ffa24
 * Copyright (C) 2015 Red Hat, Inc.
Packit Service 5ffa24
 */
Packit Service 5ffa24
Packit Service 2bceb2
#include "src/core/nm-default-daemon.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "n-acd/src/n-acd.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include <linux/if_ether.h>
Packit Service 5ffa24
Packit Service 5ffa24
#include "devices/nm-acd-manager.h"
Packit Service 5ffa24
#include "platform/tests/test-common.h"
Packit Service 5ffa24
Packit Service 5ffa24
#define IFACE_VETH0 "nm-test-veth0"
Packit Service 5ffa24
#define IFACE_VETH1 "nm-test-veth1"
Packit Service 5ffa24
Packit Service 5ffa24
#define ADDR1 0x01010101
Packit Service 5ffa24
#define ADDR2 0x02020202
Packit Service 5ffa24
#define ADDR3 0x03030303
Packit Service 5ffa24
#define ADDR4 0x04040404
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
_skip_acd_test_check(void)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NAcd *       acd;
Packit Service 5ffa24
    NAcdConfig * config;
Packit Service 5ffa24
    const guint8 hwaddr[ETH_ALEN] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06};
Packit Service 5ffa24
    int          r;
Packit Service 5ffa24
    static int   skip = -1;
Packit Service 5ffa24
Packit Service 5ffa24
    if (skip == -1) {
Packit Service 5ffa24
        r = n_acd_config_new(&config);
Packit Service 5ffa24
        g_assert(r == 0);
Packit Service 5ffa24
Packit Service 5ffa24
        n_acd_config_set_ifindex(config, 1);
Packit Service 5ffa24
        n_acd_config_set_transport(config, N_ACD_TRANSPORT_ETHERNET);
Packit Service 5ffa24
        n_acd_config_set_mac(config, hwaddr, sizeof(hwaddr));
Packit Service 5ffa24
Packit Service 5ffa24
        r = n_acd_new(&acd, config);
Packit Service 5ffa24
        n_acd_config_free(config);
Packit Service 5ffa24
        if (r == 0)
Packit Service 5ffa24
            n_acd_unref(acd);
Packit Service 5ffa24
Packit Service 5ffa24
        skip = (r != 0);
Packit Service 5ffa24
    }
Packit Service 5ffa24
    return skip;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
#define _skip_acd_test()                                               \
Packit Service 5ffa24
    ({                                                                 \
Packit Service 5ffa24
        gboolean _skip = _skip_acd_test_check();                       \
Packit Service 5ffa24
                                                                       \
Packit Service 5ffa24
        if (_skip)                                                     \
Packit Service 5ffa24
            g_test_skip("Cannot create NAcd. Running under valgind?"); \
Packit Service 5ffa24
        _skip;                                                         \
Packit Service 5ffa24
    })
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    int           ifindex0;
Packit Service 5ffa24
    int           ifindex1;
Packit Service 5ffa24
    const guint8 *hwaddr0;
Packit Service 5ffa24
    const guint8 *hwaddr1;
Packit Service 5ffa24
    size_t        hwaddr0_len;
Packit Service 5ffa24
    size_t        hwaddr1_len;
Packit Service 5ffa24
} test_fixture;
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
fixture_setup(test_fixture *fixture, gconstpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    /* create veth pair. */
Packit Service 5ffa24
    fixture->ifindex0 =
Packit Service 5ffa24
        nmtstp_link_veth_add(NM_PLATFORM_GET, -1, IFACE_VETH0, IFACE_VETH1)->ifindex;
Packit Service 5ffa24
    fixture->ifindex1 =
Packit Service 5ffa24
        nmtstp_link_get_typed(NM_PLATFORM_GET, -1, IFACE_VETH1, NM_LINK_TYPE_VETH)->ifindex;
Packit Service 5ffa24
Packit Service 5ffa24
    g_assert(nm_platform_link_set_up(NM_PLATFORM_GET, fixture->ifindex0, NULL));
Packit Service 5ffa24
    g_assert(nm_platform_link_set_up(NM_PLATFORM_GET, fixture->ifindex1, NULL));
Packit Service 5ffa24
Packit Service 5ffa24
    fixture->hwaddr0 =
Packit Service 5ffa24
        nm_platform_link_get_address(NM_PLATFORM_GET, fixture->ifindex0, &fixture->hwaddr0_len);
Packit Service 5ffa24
    fixture->hwaddr1 =
Packit Service 5ffa24
        nm_platform_link_get_address(NM_PLATFORM_GET, fixture->ifindex1, &fixture->hwaddr1_len);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    in_addr_t addresses[8];
Packit Service 5ffa24
    in_addr_t peer_addresses[8];
Packit Service 5ffa24
    gboolean  expected_result[8];
Packit Service 5ffa24
} TestInfo;
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
acd_manager_probe_terminated(NMAcdManager *acd_manager, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    g_main_loop_quit(user_data);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
test_acd_common(test_fixture *fixture, TestInfo *info)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_auto_free_acdmgr NMAcdManager *manager = NULL;
Packit Service 5ffa24
    nm_auto_unref_gmainloop GMainLoop *loop   = NULL;
Packit Service 5ffa24
    int                                i;
Packit Service 5ffa24
    const guint                        WAIT_TIME_OPTIMISTIC = 50;
Packit Service 5ffa24
    guint                              wait_time;
Packit Service 5ffa24
    static const NMAcdCallbacks        callbacks = {
Packit Service 5ffa24
        .probe_terminated_callback = acd_manager_probe_terminated,
Packit Service 5ffa24
        .user_data_destroy         = (GDestroyNotify) g_main_loop_unref,
Packit Service 5ffa24
    };
Packit Service 5ffa24
    int r;
Packit Service 5ffa24
Packit Service 5ffa24
    if (_skip_acd_test())
Packit Service 5ffa24
        return;
Packit Service 5ffa24
Packit Service 5ffa24
    /* first, try with a short waittime. We hope that this is long enough
Packit Service 5ffa24
     * to successfully complete the test. Only if that's not the case, we
Packit Service 5ffa24
     * assume the computer is currently busy (high load) and we retry with
Packit Service 5ffa24
     * a longer timeout. */
Packit Service 5ffa24
    wait_time = WAIT_TIME_OPTIMISTIC;
Packit Service 5ffa24
again:
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_pointer(&loop, g_main_loop_unref);
Packit Service 5ffa24
    loop = g_main_loop_new(NULL, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_pointer(&manager, nm_acd_manager_free);
Packit Service 5ffa24
    manager = nm_acd_manager_new(fixture->ifindex0,
Packit Service 5ffa24
                                 fixture->hwaddr0,
Packit Service 5ffa24
                                 fixture->hwaddr0_len,
Packit Service 5ffa24
                                 &callbacks,
Packit Service 5ffa24
                                 g_main_loop_ref(loop));
Packit Service 5ffa24
    g_assert(manager != NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    for (i = 0; info->addresses[i]; i++)
Packit Service 5ffa24
        g_assert(nm_acd_manager_add_address(manager, info->addresses[i]));
Packit Service 5ffa24
Packit Service 5ffa24
    for (i = 0; info->peer_addresses[i]; i++) {
Packit Service 5ffa24
        nmtstp_ip4_address_add(NULL,
Packit Service 5ffa24
                               FALSE,
Packit Service 5ffa24
                               fixture->ifindex1,
Packit Service 5ffa24
                               info->peer_addresses[i],
Packit Service 5ffa24
                               24,
Packit Service 5ffa24
                               0,
Packit Service 5ffa24
                               3600,
Packit Service 5ffa24
                               1800,
Packit Service 5ffa24
                               0,
Packit Service 5ffa24
                               NULL);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = nm_acd_manager_start_probe(manager, wait_time);
Packit Service 5ffa24
    g_assert_cmpint(r, ==, 0);
Packit Service 5ffa24
Packit Service 5ffa24
    g_assert(nmtst_main_loop_run(loop, 2000));
Packit Service 5ffa24
Packit Service 5ffa24
    for (i = 0; info->addresses[i]; i++) {
Packit Service 5ffa24
        gboolean val;
Packit Service 5ffa24
        char     sbuf[NM_UTILS_INET_ADDRSTRLEN];
Packit Service 5ffa24
Packit Service 5ffa24
        val = nm_acd_manager_check_address(manager, info->addresses[i]);
Packit Service 5ffa24
        if (val == info->expected_result[i])
Packit Service 5ffa24
            continue;
Packit Service 5ffa24
Packit Service 5ffa24
        if (wait_time == WAIT_TIME_OPTIMISTIC) {
Packit Service 5ffa24
            /* probably we just had a glitch and the system took longer than
Packit Service 5ffa24
             * expected. Re-verify with a large timeout this time. */
Packit Service 5ffa24
            wait_time = 1000;
Packit Service 5ffa24
            goto again;
Packit Service 5ffa24
        }
Packit Service 5ffa24
Packit Service 5ffa24
        g_error("expected check for address #%d (%s) to %s, but it didn't",
Packit Service 5ffa24
                i,
Packit Service 5ffa24
                _nm_utils_inet4_ntop(info->addresses[i], sbuf),
Packit Service 5ffa24
                info->expected_result[i] ? "detect no duplicated" : "detect a duplicate");
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
test_acd_probe_1(test_fixture *fixture, gconstpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    TestInfo info = {.addresses       = {ADDR1, ADDR2, ADDR3},
Packit Service 5ffa24
                     .peer_addresses  = {ADDR4},
Packit Service 5ffa24
                     .expected_result = {TRUE, TRUE, TRUE}};
Packit Service 5ffa24
Packit Service 5ffa24
    test_acd_common(fixture, &info;;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
test_acd_probe_2(test_fixture *fixture, gconstpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    TestInfo info = {.addresses       = {ADDR1, ADDR2, ADDR3, ADDR4},
Packit Service 5ffa24
                     .peer_addresses  = {ADDR3, ADDR2},
Packit Service 5ffa24
                     .expected_result = {TRUE, FALSE, FALSE, TRUE}};
Packit Service 5ffa24
Packit Service 5ffa24
    test_acd_common(fixture, &info;;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
test_acd_announce(test_fixture *fixture, gconstpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_auto_free_acdmgr NMAcdManager *manager = NULL;
Packit Service 5ffa24
    nm_auto_unref_gmainloop GMainLoop *loop   = NULL;
Packit Service 5ffa24
    int                                r;
Packit Service 5ffa24
Packit Service 5ffa24
    if (_skip_acd_test())
Packit Service 5ffa24
        return;
Packit Service 5ffa24
Packit Service 5ffa24
    manager =
Packit Service 5ffa24
        nm_acd_manager_new(fixture->ifindex0, fixture->hwaddr0, fixture->hwaddr0_len, NULL, NULL);
Packit Service 5ffa24
    g_assert(manager != NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    g_assert(nm_acd_manager_add_address(manager, ADDR1));
Packit Service 5ffa24
    g_assert(nm_acd_manager_add_address(manager, ADDR2));
Packit Service 5ffa24
Packit Service 5ffa24
    loop = g_main_loop_new(NULL, FALSE);
Packit Service 5ffa24
    r    = nm_acd_manager_announce_addresses(manager);
Packit Service 5ffa24
    g_assert_cmpint(r, ==, 0);
Packit Service 5ffa24
    g_assert(!nmtst_main_loop_run(loop, 200));
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
fixture_teardown(test_fixture *fixture, gconstpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_platform_link_delete(NM_PLATFORM_GET, fixture->ifindex0);
Packit Service 5ffa24
    nm_platform_link_delete(NM_PLATFORM_GET, fixture->ifindex1);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
NMTstpSetupFunc const _nmtstp_setup_platform_func = nm_linux_platform_setup;
Packit Service 5ffa24
Packit Service 5ffa24
void
Packit Service 5ffa24
_nmtstp_init_tests(int *argc, char ***argv)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nmtst_init_with_logging(argc, argv, NULL, "ALL");
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
void
Packit Service 5ffa24
_nmtstp_setup_tests(void)
Packit Service 5ffa24
{
Packit Service 5ffa24
    g_test_add("/acd/probe/1",
Packit Service 5ffa24
               test_fixture,
Packit Service 5ffa24
               NULL,
Packit Service 5ffa24
               fixture_setup,
Packit Service 5ffa24
               test_acd_probe_1,
Packit Service 5ffa24
               fixture_teardown);
Packit Service 5ffa24
    g_test_add("/acd/probe/2",
Packit Service 5ffa24
               test_fixture,
Packit Service 5ffa24
               NULL,
Packit Service 5ffa24
               fixture_setup,
Packit Service 5ffa24
               test_acd_probe_2,
Packit Service 5ffa24
               fixture_teardown);
Packit Service 5ffa24
    g_test_add("/acd/announce",
Packit Service 5ffa24
               test_fixture,
Packit Service 5ffa24
               NULL,
Packit Service 5ffa24
               fixture_setup,
Packit Service 5ffa24
               test_acd_announce,
Packit Service 5ffa24
               fixture_teardown);
Packit Service 5ffa24
}