// SPDX-License-Identifier: GPL-2.0+
/*
* Copyright (C) 2015 - 2017 Red Hat, Inc.
*/
#include "nm-default.h"
#include <libudev.h>
#include <linux/pkt_sched.h>
#include "platform/nmp-object.h"
#include "nm-udev-aux/nm-udev-utils.h"
#include "nm-test-utils-core.h"
struct {
GList *udev_devices;
} global;
/*****************************************************************************/
static void
test_obj_base (void)
{
static const union {
GObject g;
NMPObject k;
} x = { };
static const union {
GTypeClass k;
NMPClass c;
} l = { };
static const GObject *g = &x.g;
static const GTypeClass *k = &l.k;
static const NMPObject *o = &x.k;
static const NMPClass *c = &l.c;
NMObjBaseInst *obj;
gs_unref_object GCancellable *obj_cancellable = g_cancellable_new ();
nm_auto_nmpobj NMPObject *obj_link = nmp_object_new_link (10);
g_assert (&g->g_type_instance == (void *) &o->_class);
g_assert (&g->g_type_instance.g_class == (void *) &o->_class);
g_assert (sizeof (o->parent.parent) == sizeof (GTypeInstance));
g_assert (&c->parent == (void *) c);
g_assert (&c->parent.parent.g_type_class == (void *) c);
g_assert (&c->parent.parent.g_type == (void *) c);
g_assert (&c->parent.parent.g_type == &k->g_type);
g_assert (sizeof (c->parent.parent) == sizeof (GTypeClass));
g_assert (&o->parent == (void *) o);
g_assert (&o->parent.klass == (void *) &o->_class);
obj = (NMObjBaseInst *) obj_cancellable;
g_assert (!NMP_CLASS_IS_VALID ((NMPClass *) obj->klass));
g_assert (G_TYPE_CHECK_INSTANCE_TYPE (obj, G_TYPE_CANCELLABLE));
obj = (NMObjBaseInst *) obj_link;
g_assert (NMP_CLASS_IS_VALID ((NMPClass *) obj->klass));
g_assert (!G_TYPE_CHECK_INSTANCE_TYPE (obj, G_TYPE_CANCELLABLE));
}
/*****************************************************************************/
static gboolean
_nmp_object_id_equal (const NMPObject *a, const NMPObject *b)
{
gboolean a_b = nmp_object_id_equal (a, b);
g_assert (NM_IN_SET (a_b, FALSE, TRUE) && a_b == nmp_object_id_equal (b, a));
return a_b;
}
#define nmp_object_id_equal _nmp_object_id_equal
static gboolean
_nmp_object_equal (const NMPObject *a, const NMPObject *b)
{
gboolean a_b = nmp_object_equal (a, b);
g_assert (NM_IN_SET (a_b, FALSE, TRUE) && a_b == nmp_object_equal (b, a));
return a_b;
}
#define nmp_object_equal _nmp_object_equal
/*****************************************************************************/
static void
_assert_cache_multi_lookup_contains (const NMPCache *cache, const NMDedupMultiHeadEntry *head_entry, const NMPObject *obj, gboolean visible_only, gboolean contains)
{
NMDedupMultiIter iter;
gboolean found;
guint i, len;
const NMPObject *o;
g_assert (NMP_OBJECT_IS_VALID (obj));
g_assert (nmp_cache_lookup_obj (cache, obj) == obj);
g_assert (!head_entry || (head_entry->len > 0 && c_list_length (&head_entry->lst_entries_head) == head_entry->len));
len = head_entry ? head_entry->len : 0;
found = FALSE;
i = 0;
nmp_cache_iter_for_each (&iter,
head_entry,
&o) {
g_assert (NMP_OBJECT_IS_VALID (o));
if (obj == o) {
if ( !visible_only
|| nmp_object_is_visible (o)) {
g_assert (!found);
found = TRUE;
}
}
i++;
}
g_assert (len == i);
g_assert (!!contains == found);
}
static void
_assert_cache_multi_lookup_contains_link (const NMPCache *cache,
gboolean visible_only,
const NMPObject *obj,
gboolean contains)
{
const NMDedupMultiHeadEntry *head_entry;
NMPLookup lookup;
g_assert (cache);
nmp_lookup_init_obj_type (&lookup, NMP_OBJECT_TYPE_LINK);
head_entry = nmp_cache_lookup (cache, &lookup);
_assert_cache_multi_lookup_contains (cache, head_entry, obj, visible_only, contains);
}
/*****************************************************************************/
static void
ops_post_check (NMPCache *cache,
NMPCacheOpsType ops_type,
const NMPObject *obj_old,
const NMPObject *obj_new,
const NMPObject *obj_new_expected,
NMPCacheOpsType expected_ops_type)
{
g_assert (cache);
g_assert_cmpint (expected_ops_type, ==, ops_type);
switch (ops_type) {
case NMP_CACHE_OPS_ADDED:
g_assert (!obj_old);
g_assert (NMP_OBJECT_IS_VALID (obj_new));
g_assert (nmp_object_is_alive (obj_new));
g_assert (nmp_object_id_equal (obj_new_expected, obj_new));
g_assert (nmp_object_equal (obj_new_expected, obj_new));
break;
case NMP_CACHE_OPS_UPDATED:
g_assert (obj_old != obj_new);
g_assert (NMP_OBJECT_IS_VALID (obj_old));
g_assert (NMP_OBJECT_IS_VALID (obj_new));
g_assert (nmp_object_is_alive (obj_old));
g_assert (nmp_object_is_alive (obj_new));
g_assert (nmp_object_id_equal (obj_new_expected, obj_new));
g_assert (nmp_object_id_equal (obj_new_expected, obj_old));
g_assert (nmp_object_id_equal (obj_old, obj_new));
g_assert (nmp_object_equal (obj_new_expected, obj_new));
g_assert (!nmp_object_equal (obj_new_expected, obj_old));
g_assert (!nmp_object_equal (obj_old, obj_new));
break;
case NMP_CACHE_OPS_REMOVED:
g_assert (!obj_new);
g_assert (NMP_OBJECT_IS_VALID (obj_old));
g_assert (nmp_object_is_alive (obj_old));
if (obj_new_expected)
g_assert (nmp_object_id_equal (obj_new_expected, obj_old));
break;
case NMP_CACHE_OPS_UNCHANGED:
g_assert (obj_old == obj_new);
if (obj_old) {
g_assert (NMP_OBJECT_IS_VALID (obj_old));
g_assert (nmp_object_is_alive (obj_old));
g_assert (nmp_object_equal (obj_old, obj_new));
g_assert (nmp_object_id_equal (obj_new_expected, obj_new));
} else
g_assert (!obj_new_expected);
break;
default:
g_assert_not_reached ();
}
}
static void
_nmp_cache_update_netlink (NMPCache *cache, NMPObject *obj, const NMPObject **out_obj_old, const NMPObject **out_obj_new, NMPCacheOpsType expected_ops_type)
{
NMPCacheOpsType ops_type;
const NMPObject *obj_prev;
const NMPObject *obj_old;
const NMPObject *obj_new;
nm_auto_nmpobj NMPObject *obj_new_expected = NULL;
g_assert (cache);
g_assert (NMP_OBJECT_IS_VALID (obj));
obj_prev = nmp_cache_lookup_link (cache, NMP_OBJECT_CAST_LINK (obj)->ifindex);
obj_new_expected = nmp_object_clone (obj, FALSE);
if (obj_prev && obj_prev->_link.udev.device)
obj_new_expected->_link.udev.device = udev_device_ref (obj_prev->_link.udev.device);
_nmp_object_fixup_link_udev_fields (&obj_new_expected, NULL, nmp_cache_use_udev_get (cache));
ops_type = nmp_cache_update_netlink (cache, obj, FALSE, &obj_old, &obj_new);
ops_post_check (cache, ops_type, obj_old, obj_new,
nmp_object_is_alive (obj_new_expected) ? obj_new_expected : NULL,
expected_ops_type);
if (out_obj_new)
*out_obj_new = obj_new;
else
nmp_object_unref (obj_new);
if (out_obj_old)
*out_obj_old = obj_old;
else
nmp_object_unref (obj_old);
}
static const NMPlatformLink pl_link_2 = {
.ifindex = 2,
.name = "eth0",
.type = NM_LINK_TYPE_ETHERNET,
};
static const NMPlatformLink pl_link_3 = {
.ifindex = 3,
.name = "wlan0",
.type = NM_LINK_TYPE_WIFI,
};
static void
test_cache_link (void)
{
NMPCache *cache;
NMPObject *objm1;
const NMPObject *obj_old, *obj_new;
NMPObject objs1;
struct udev_device *udev_device_2 = g_list_nth_data (global.udev_devices, 0);
struct udev_device *udev_device_3 = g_list_nth_data (global.udev_devices, 0);
NMPCacheOpsType ops_type;
nm_auto_unref_dedup_multi_index NMDedupMultiIndex *multi_idx = NULL;
gboolean use_udev = nmtst_get_rand_uint32 () % 2;
multi_idx = nm_dedup_multi_index_new ();
cache = nmp_cache_new (multi_idx, use_udev);
/* if we have a link, and don't set is_in_netlink, adding it has no effect. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
g_assert (NMP_OBJECT_UP_CAST (&objm1->object) == objm1);
g_assert (!nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, NMP_CACHE_OPS_UNCHANGED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (!obj_old);
g_assert (!obj_new);
g_assert (!nmp_cache_lookup_obj (cache, objm1));
g_assert (!nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)));
nmp_object_unref (objm1);
/* Only when setting @is_in_netlink the link is added. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
objm1->_link.netlink.is_in_netlink = TRUE;
g_assert (nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, NMP_CACHE_OPS_ADDED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (!obj_old);
g_assert (obj_new);
g_assert (objm1 == obj_new);
g_assert (nmp_object_equal (objm1, obj_new));
g_assert (nmp_cache_lookup_obj (cache, objm1) == obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == obj_new);
g_assert (nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, TRUE);
nmp_object_unref (objm1);
nmp_object_unref (obj_new);
/* updating the same link with identical value, has no effect. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
objm1->_link.netlink.is_in_netlink = TRUE;
g_assert (nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, NMP_CACHE_OPS_UNCHANGED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (obj_old);
g_assert (obj_new);
g_assert (obj_new != objm1);
g_assert (nmp_object_equal (objm1, obj_new));
g_assert (nmp_cache_lookup_obj (cache, objm1) == obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == obj_new);
nmp_object_unref (objm1);
nmp_object_unref (obj_new);
nmp_object_unref (obj_new);
/* remove the link from netlink */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
g_assert (!nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, NMP_CACHE_OPS_REMOVED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (obj_old);
g_assert (!obj_new);
g_assert (!nmp_cache_lookup_obj (cache, objm1));
g_assert (!nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)));
nmp_object_unref (objm1);
nmp_object_unref (obj_old);
nmp_object_unref (obj_new);
if (udev_device_2) {
/* now add the link only with aspect UDEV. */
ops_type = nmp_cache_update_link_udev (cache, pl_link_2.ifindex, udev_device_2, &obj_old, &obj_new);
ASSERT_nmp_cache_is_consistent (cache);
g_assert_cmpint (ops_type, ==, NMP_CACHE_OPS_ADDED);
g_assert (!obj_old);
g_assert (obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == obj_new);
g_assert (!nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, FALSE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
nmp_object_unref (obj_new);
}
/* add it in netlink too. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
objm1->_link.netlink.is_in_netlink = TRUE;
g_assert (nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, udev_device_2 ? NMP_CACHE_OPS_UPDATED : NMP_CACHE_OPS_ADDED);
ASSERT_nmp_cache_is_consistent (cache);
if (udev_device_2) {
g_assert (obj_old);
g_assert (!nmp_object_is_visible (obj_old));
} else
g_assert (!obj_old);
g_assert (nmp_object_equal (objm1, obj_new));
g_assert (nmp_cache_lookup_obj (cache, objm1) == obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == obj_new);
g_assert (nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, TRUE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
nmp_object_unref (objm1);
nmp_object_unref (obj_old);
nmp_object_unref (obj_new);
/* remove again from netlink. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_2);
objm1->_link.netlink.is_in_netlink = FALSE;
g_assert (!nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, udev_device_2 ? NMP_CACHE_OPS_UPDATED : NMP_CACHE_OPS_REMOVED);
ASSERT_nmp_cache_is_consistent (cache);
if (udev_device_2)
g_assert (obj_new == objm1);
else
g_assert (!obj_new);
g_assert (obj_old);
g_assert (nmp_object_is_alive (obj_old));
if (udev_device_2) {
g_assert (nmp_cache_lookup_obj (cache, objm1) == obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == obj_new);
g_assert (!nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, FALSE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
} else {
g_assert (nmp_cache_lookup_obj (cache, objm1) == NULL);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_2.ifindex)) == NULL);
g_assert (!nmp_object_is_alive (obj_new));
g_assert (!nmp_object_is_visible (obj_new));
}
nmp_object_unref (objm1);
nmp_object_unref (obj_old);
nmp_object_unref (obj_new);
/* now another link only with aspect UDEV. */
if (udev_device_3) {
/* now add the link only with aspect UDEV. */
ops_type = nmp_cache_update_link_udev (cache, pl_link_3.ifindex, udev_device_3, &obj_old, &obj_new);
g_assert_cmpint (ops_type, ==, NMP_CACHE_OPS_ADDED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (NMP_OBJECT_IS_VALID (obj_new));
g_assert (!obj_old);
g_assert (!nmp_object_is_visible (obj_new));
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_3.ifindex)) == obj_new);
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, FALSE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
g_assert_cmpint (obj_new->_link.netlink.is_in_netlink, ==, FALSE);
g_assert_cmpint (obj_new->link.initialized, ==, FALSE);
nmp_object_unref (obj_new);
/* add it in netlink too. */
objm1 = nmp_object_new (NMP_OBJECT_TYPE_LINK, (NMPlatformObject *) &pl_link_3);
objm1->_link.netlink.is_in_netlink = TRUE;
g_assert (nmp_object_is_alive (objm1));
_nmp_cache_update_netlink (cache, objm1, &obj_old, &obj_new, NMP_CACHE_OPS_UPDATED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (obj_old);
g_assert (obj_new == objm1);
g_assert (nmp_object_equal (objm1, obj_new));
g_assert (!obj_old || !nmp_object_is_visible (obj_old));
g_assert (nmp_cache_lookup_obj (cache, objm1) == obj_new);
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_3.ifindex)) == obj_new);
g_assert (nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, TRUE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
g_assert_cmpint (obj_new->_link.netlink.is_in_netlink, ==, TRUE);
g_assert_cmpint (obj_new->link.initialized, ==, TRUE);
nmp_object_unref (objm1);
nmp_object_unref (obj_old);
nmp_object_unref (obj_new);
/* remove UDEV. */
ops_type = nmp_cache_update_link_udev (cache, pl_link_3.ifindex, NULL, &obj_old, &obj_new);
g_assert_cmpint (ops_type, ==, NMP_CACHE_OPS_UPDATED);
ASSERT_nmp_cache_is_consistent (cache);
g_assert (obj_old && nmp_object_is_visible (obj_old));
g_assert (nmp_cache_lookup_obj (cache, nmp_object_stackinit_id_link (&objs1, pl_link_3.ifindex)) == obj_new);
g_assert (nmp_object_is_visible (obj_new));
_assert_cache_multi_lookup_contains_link (cache, TRUE, obj_new, TRUE);
_assert_cache_multi_lookup_contains_link (cache, FALSE, obj_new, TRUE);
g_assert_cmpint (obj_new->_link.netlink.is_in_netlink, ==, TRUE);
g_assert_cmpint (obj_new->link.initialized, ==, !nmp_cache_use_udev_get (cache));
nmp_object_unref (obj_new);
nmp_object_unref (obj_old);
}
nmp_cache_free (cache);
}
const char noqueue[] = "noqueue";
const char fq_codel[] = "fq_codel";
const char ingress[] = "ingress";
static const NMPlatformQdisc pl_qdisc_1a = {
.ifindex = 1,
.kind = noqueue,
.addr_family = AF_UNSPEC,
.handle = 0,
.parent = TC_H_ROOT,
.info = 0,
};
static const NMPlatformQdisc pl_qdisc_1b = {
.ifindex = 1,
.kind = fq_codel,
.addr_family = AF_UNSPEC,
.handle = 0,
.parent = TC_H_ROOT,
.info = 0,
};
static const NMPlatformQdisc pl_qdisc_1c = {
.ifindex = 1,
.kind = ingress,
.addr_family = AF_UNSPEC,
.handle = TC_H_MAKE(TC_H_INGRESS, 0),
.parent = TC_H_INGRESS,
.info = 0,
};
static const NMPlatformQdisc pl_qdisc_2 = {
.ifindex = 2,
.kind = fq_codel,
.addr_family = AF_UNSPEC,
.handle = 0,
.parent = TC_H_ROOT,
.info = 0,
};
static void
test_cache_qdisc (void)
{
NMPCache *cache;
nm_auto_unref_dedup_multi_index NMDedupMultiIndex *multi_idx = NULL;
NMPLookup lookup;
const NMDedupMultiHeadEntry *head_entry;
nm_auto_nmpobj NMPObject *obj1a = nmp_object_new (NMP_OBJECT_TYPE_QDISC, (NMPlatformObject *) &pl_qdisc_1a);
nm_auto_nmpobj NMPObject *obj1b = nmp_object_new (NMP_OBJECT_TYPE_QDISC, (NMPlatformObject *) &pl_qdisc_1b);
nm_auto_nmpobj NMPObject *obj1c = nmp_object_new (NMP_OBJECT_TYPE_QDISC, (NMPlatformObject *) &pl_qdisc_1c);
nm_auto_nmpobj NMPObject *obj2 = nmp_object_new (NMP_OBJECT_TYPE_QDISC, (NMPlatformObject *) &pl_qdisc_2);
multi_idx = nm_dedup_multi_index_new ();
cache = nmp_cache_new (multi_idx, nmtst_get_rand_uint32 () % 2);
g_assert (nmp_cache_lookup_obj (cache, obj1a) == NULL);
g_assert (nmp_cache_update_netlink (cache, obj1a, FALSE, NULL, NULL) == NMP_CACHE_OPS_ADDED);
g_assert (nmp_cache_lookup_obj (cache, obj1a) == obj1a);
g_assert (nmp_cache_lookup_obj (cache, obj1b) == obj1a);
g_assert (nmp_cache_lookup_obj (cache, obj2) == NULL);
g_assert (nmp_cache_update_netlink (cache, obj1b, FALSE, NULL, NULL) == NMP_CACHE_OPS_UPDATED);
g_assert (nmp_cache_lookup_obj (cache, obj1a) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj1b) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj2) == NULL);
g_assert (nmp_cache_update_netlink (cache, obj1c, FALSE, NULL, NULL) == NMP_CACHE_OPS_ADDED);
g_assert (nmp_cache_lookup_obj (cache, obj1a) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj1b) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj1c) == obj1c);
g_assert (nmp_cache_lookup_obj (cache, obj2) == NULL);
g_assert (nmp_cache_update_netlink (cache, obj2, FALSE, NULL, NULL) == NMP_CACHE_OPS_ADDED);
g_assert (nmp_cache_lookup_obj (cache, obj1a) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj1b) == obj1b);
g_assert (nmp_cache_lookup_obj (cache, obj2) == obj2);
head_entry = nmp_cache_lookup (cache,
nmp_lookup_init_object (&lookup,
NMP_OBJECT_TYPE_QDISC,
1));
g_assert (head_entry->len == 2);
nmp_cache_free (cache);
}
/*****************************************************************************/
NMTST_DEFINE ();
int
main (int argc, char **argv)
{
int result;
NMUdevClient *udev_client;
nmtst_init_assert_logging (&argc, &argv, "INFO", "DEFAULT");
udev_client = nm_udev_client_new ((const char *[]) { "net", NULL },
NULL, NULL);
{
struct udev_enumerate *enumerator;
struct udev_list_entry *devices, *l;
enumerator = nm_udev_client_enumerate_new (udev_client);
/* Demand that the device is initialized (udev rules ran,
* device has a stable name now) in case udev is running
* (not in a container). */
if (access ("/sys", W_OK) == 0)
udev_enumerate_add_match_is_initialized (enumerator);
udev_enumerate_scan_devices (enumerator);
devices = udev_enumerate_get_list_entry (enumerator);
for (l = devices; l != NULL; l = udev_list_entry_get_next (l)) {
struct udev_device *udevice;
udevice = udev_device_new_from_syspath (udev_enumerate_get_udev (enumerator),
udev_list_entry_get_name (l));
if (udevice == NULL)
continue;
global.udev_devices = g_list_prepend (global.udev_devices, udevice);
}
global.udev_devices = g_list_reverse (global.udev_devices);
udev_enumerate_unref (enumerator);
}
g_test_add_func ("/nmp-object/obj-base", test_obj_base);
g_test_add_func ("/nmp-object/cache_link", test_cache_link);
g_test_add_func ("/nmp-object/cache_qdisc", test_cache_qdisc);
result = g_test_run ();
while (global.udev_devices) {
udev_device_unref (global.udev_devices->data);
global.udev_devices = g_list_delete_link (global.udev_devices, global.udev_devices);
}
nm_udev_client_unref (udev_client);
return result;
}