2014-03-14 17:49:49 +01:00
|
|
|
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This library 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
|
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
* License along with this library; if not, write to the
|
|
|
|
|
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
|
* Boston, MA 02110-1301 USA.
|
|
|
|
|
*
|
|
|
|
|
* (C) Copyright 2014 Red Hat, Inc.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifndef __NM_TEST_UTILS_H__
|
|
|
|
|
#define __NM_TEST_UTILS_H__
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
#include <glib.h>
|
2014-04-22 13:00:34 +02:00
|
|
|
#include <glib-object.h>
|
2014-04-17 10:49:01 +02:00
|
|
|
#include <string.h>
|
2014-03-14 17:49:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
struct __nmtst_internal
|
|
|
|
|
{
|
|
|
|
|
GRand *rand0;
|
|
|
|
|
guint32 rand_seed;
|
|
|
|
|
GRand *rand;
|
2014-04-17 10:49:01 +02:00
|
|
|
gboolean is_debug;
|
2014-03-14 17:49:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
extern struct __nmtst_internal __nmtst_internal;
|
|
|
|
|
|
|
|
|
|
#define NMTST_DEFINE() \
|
|
|
|
|
struct __nmtst_internal __nmtst_internal = { 0 };
|
|
|
|
|
|
2014-04-17 10:47:48 +02:00
|
|
|
|
|
|
|
|
inline static gboolean
|
|
|
|
|
nmtst_initialized (void)
|
|
|
|
|
{
|
|
|
|
|
return !!__nmtst_internal.rand0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 10:49:01 +02:00
|
|
|
/* split the string inplace at specific delimiters, allowing escaping with '\\'.
|
|
|
|
|
* Returns a zero terminated array of pointers into @str.
|
|
|
|
|
*
|
|
|
|
|
* The caller must g_free() the returned argv array.
|
|
|
|
|
**/
|
|
|
|
|
inline static char **
|
|
|
|
|
nmtst_str_split (char *str, const char *delimiters)
|
|
|
|
|
{
|
|
|
|
|
const char *d;
|
|
|
|
|
GArray *result = g_array_sized_new (TRUE, FALSE, sizeof (char *), 3);
|
|
|
|
|
|
|
|
|
|
g_assert (str);
|
|
|
|
|
g_assert (delimiters && !strchr (delimiters, '\\'));
|
|
|
|
|
|
|
|
|
|
while (*str) {
|
|
|
|
|
gsize i = 0, j = 0;
|
|
|
|
|
|
|
|
|
|
while (TRUE) {
|
|
|
|
|
char c = str[i];
|
|
|
|
|
|
|
|
|
|
if (c == '\0') {
|
|
|
|
|
str[j++] = 0;
|
|
|
|
|
break;
|
|
|
|
|
} else if (c == '\\') {
|
|
|
|
|
str[j++] = str[++i];
|
|
|
|
|
if (!str[i])
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
for (d = delimiters; *d; d++) {
|
|
|
|
|
if (c == *d) {
|
|
|
|
|
str[j++] = 0;
|
|
|
|
|
i++;
|
|
|
|
|
goto BREAK_INNER_LOOPS;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
str[j++] = c;
|
|
|
|
|
}
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BREAK_INNER_LOOPS:
|
|
|
|
|
g_array_append_val (result, str);
|
|
|
|
|
str = &str[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (char **) g_array_free (result, FALSE);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 10:47:48 +02:00
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static void
|
2014-04-14 12:03:05 +02:00
|
|
|
nmtst_init (int *argc, char ***argv, const char *log_level, const char *log_domains)
|
2014-03-14 17:49:49 +01:00
|
|
|
{
|
2014-04-17 10:49:01 +02:00
|
|
|
const char *nmtst_debug;
|
|
|
|
|
gboolean is_debug = FALSE;
|
|
|
|
|
char *c_log_level = NULL, *c_log_domains = NULL;
|
|
|
|
|
GArray *debug_messages = g_array_new (TRUE, FALSE, sizeof (char *));
|
|
|
|
|
int i;
|
|
|
|
|
|
2014-04-17 10:47:48 +02:00
|
|
|
g_assert (!nmtst_initialized ());
|
2014-03-14 17:49:49 +01:00
|
|
|
|
|
|
|
|
g_assert (!((!!argc) ^ (!!argv)));
|
2014-04-17 10:49:01 +02:00
|
|
|
if (argc && !g_test_initialized ()) {
|
2014-03-14 17:49:49 +01:00
|
|
|
/* g_test_init() is a variadic function, so we cannot pass it
|
|
|
|
|
* (variadic) arguments. If you need to pass additional parameters,
|
|
|
|
|
* call nmtst_init() with argc==NULL and call g_test_init() yourself. */
|
|
|
|
|
g_test_init (argc, argv, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-30 18:14:48 +02:00
|
|
|
#if !GLIB_CHECK_VERSION (2, 35, 0)
|
2014-03-14 17:49:49 +01:00
|
|
|
g_type_init ();
|
2014-05-30 18:14:48 +02:00
|
|
|
#endif
|
2014-03-14 17:49:49 +01:00
|
|
|
|
2014-04-17 10:49:01 +02:00
|
|
|
is_debug = g_test_verbose ();
|
|
|
|
|
|
|
|
|
|
nmtst_debug = g_getenv ("NMTST_DEBUG");
|
|
|
|
|
if (nmtst_debug) {
|
|
|
|
|
char **d_argv, **i_argv, *nmtst_debug_copy;
|
|
|
|
|
|
|
|
|
|
/* By setting then NMTST_DEBUG variable, @is_debug is set automatically.
|
|
|
|
|
* This can be reverted with no-debug (on command line or environment variable). */
|
|
|
|
|
is_debug = TRUE;
|
|
|
|
|
|
|
|
|
|
nmtst_debug_copy = g_strdup (nmtst_debug);
|
|
|
|
|
d_argv = nmtst_str_split (nmtst_debug_copy, ",; \t\r\n");
|
|
|
|
|
|
|
|
|
|
for (i_argv = d_argv; *i_argv; i_argv++) {
|
|
|
|
|
const char *debug = *i_argv;
|
|
|
|
|
|
|
|
|
|
if (!g_ascii_strcasecmp (debug, "debug"))
|
|
|
|
|
is_debug = TRUE;
|
|
|
|
|
else if (!g_ascii_strcasecmp (debug, "no-debug")) {
|
|
|
|
|
/* when specifying the NMTST_DEBUG variable, we set is_debug to true. Use this flag to disable this
|
|
|
|
|
* (e.g. for only setting the log-level, but not is_debug). */
|
|
|
|
|
is_debug = FALSE;
|
|
|
|
|
} else if (!g_ascii_strncasecmp (debug, "log-level=", strlen ("log-level="))) {
|
|
|
|
|
g_free (c_log_level);
|
|
|
|
|
log_level = c_log_level = g_strdup (&debug[strlen ("log-level=")]);
|
|
|
|
|
} else if (!g_ascii_strncasecmp (debug, "log-domains=", strlen ("log-domains="))) {
|
|
|
|
|
g_free (c_log_domains);
|
|
|
|
|
log_domains = c_log_domains = g_strdup (&debug[strlen ("log-domains=")]);
|
|
|
|
|
} else {
|
|
|
|
|
char *msg = g_strdup_printf (">>> nmtst: ignore unrecognized NMTST_DEBUG option \"%s\"", debug);
|
|
|
|
|
|
|
|
|
|
g_array_append_val (debug_messages, msg);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (d_argv);
|
|
|
|
|
g_free (nmtst_debug_copy);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (argv && *argv) {
|
|
|
|
|
char **a = *argv;
|
|
|
|
|
|
|
|
|
|
for (; *a; a++) {
|
|
|
|
|
if (!g_ascii_strcasecmp (*a, "--debug"))
|
|
|
|
|
is_debug = TRUE;
|
|
|
|
|
else if (!g_ascii_strcasecmp (*a, "--no-debug"))
|
|
|
|
|
is_debug = FALSE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__nmtst_internal.is_debug = is_debug;
|
2014-03-14 17:49:49 +01:00
|
|
|
__nmtst_internal.rand0 = g_rand_new_with_seed (0);
|
2014-04-14 12:03:05 +02:00
|
|
|
|
2014-04-17 10:49:01 +02:00
|
|
|
if (!log_level && log_domains) {
|
|
|
|
|
/* if the log level is not specified (but the domain is), we assume
|
|
|
|
|
* the caller wants to set it depending on is_debug */
|
|
|
|
|
log_level = is_debug ? "DEBUG" : "WARN";
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-14 12:03:05 +02:00
|
|
|
if (log_level || log_domains) {
|
|
|
|
|
gboolean success = FALSE;
|
|
|
|
|
#ifdef NM_LOGGING_H
|
|
|
|
|
success = nm_logging_setup (log_level, log_domains, NULL, NULL);
|
|
|
|
|
#endif
|
|
|
|
|
g_assert (success);
|
|
|
|
|
}
|
2014-04-17 10:49:01 +02:00
|
|
|
|
|
|
|
|
/* Delay messages until we setup logging. */
|
|
|
|
|
for (i = 0; i < debug_messages->len; i++)
|
|
|
|
|
g_message ("%s", g_array_index (debug_messages, const char *, i));
|
|
|
|
|
|
|
|
|
|
g_strfreev ((char **) g_array_free (debug_messages, FALSE));
|
|
|
|
|
g_free (c_log_level);
|
|
|
|
|
g_free (c_log_domains);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline static gboolean
|
|
|
|
|
nmtst_is_debug (void)
|
|
|
|
|
{
|
|
|
|
|
g_assert (nmtst_initialized ());
|
|
|
|
|
return __nmtst_internal.is_debug;
|
2014-03-14 17:49:49 +01:00
|
|
|
}
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static GRand *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_get_rand0 ()
|
|
|
|
|
{
|
2014-04-17 10:47:48 +02:00
|
|
|
g_assert (nmtst_initialized ());
|
2014-03-14 17:49:49 +01:00
|
|
|
return __nmtst_internal.rand0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static GRand *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_get_rand ()
|
|
|
|
|
{
|
2014-04-17 10:47:48 +02:00
|
|
|
g_assert (nmtst_initialized ());
|
|
|
|
|
|
2014-03-14 17:49:49 +01:00
|
|
|
if (G_UNLIKELY (!__nmtst_internal.rand)) {
|
|
|
|
|
guint32 seed;
|
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
|
|
if ((str = g_getenv ("NMTST_SEED_RAND"))) {
|
|
|
|
|
gchar *s;
|
|
|
|
|
gint64 i;
|
|
|
|
|
|
|
|
|
|
i = g_ascii_strtoll (str, &s, 0);
|
|
|
|
|
g_assert (s[0] == '\0' && i >= 0 && i < G_MAXINT32);
|
|
|
|
|
|
|
|
|
|
seed = i;
|
|
|
|
|
__nmtst_internal.rand = g_rand_new_with_seed (seed);
|
|
|
|
|
} else {
|
|
|
|
|
__nmtst_internal.rand = g_rand_new ();
|
|
|
|
|
|
|
|
|
|
seed = g_rand_int (__nmtst_internal.rand);
|
|
|
|
|
g_rand_set_seed (__nmtst_internal.rand, seed);
|
|
|
|
|
}
|
|
|
|
|
__nmtst_internal.rand_seed = seed;
|
|
|
|
|
|
|
|
|
|
g_message (">> initialize nmtst_get_rand() with seed=%u", seed);
|
|
|
|
|
}
|
|
|
|
|
return __nmtst_internal.rand;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define NMTST_SWAP(x,y) \
|
|
|
|
|
G_STMT_START { \
|
|
|
|
|
char __nmtst_swap_temp[sizeof(x) == sizeof(y) ? (signed) sizeof(x) : -1]; \
|
|
|
|
|
memcpy(__nmtst_swap_temp, &y, sizeof(x)); \
|
|
|
|
|
memcpy(&y, &x, sizeof(x)); \
|
|
|
|
|
memcpy(&x, __nmtst_swap_temp, sizeof(x)); \
|
|
|
|
|
} G_STMT_END
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static guint32
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_inet4_from_string (const char *str)
|
|
|
|
|
{
|
|
|
|
|
guint32 addr;
|
|
|
|
|
int success;
|
|
|
|
|
|
|
|
|
|
if (!str)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
success = inet_pton (AF_INET, str, &addr);
|
|
|
|
|
|
|
|
|
|
g_assert (success == 1);
|
|
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-05 10:11:00 +02:00
|
|
|
inline static const struct in6_addr *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_inet6_from_string (const char *str)
|
|
|
|
|
{
|
|
|
|
|
static struct in6_addr addr;
|
|
|
|
|
int success;
|
|
|
|
|
|
|
|
|
|
if (!str)
|
|
|
|
|
addr = in6addr_any;
|
|
|
|
|
else {
|
|
|
|
|
success = inet_pton (AF_INET6, str, &addr);
|
|
|
|
|
g_assert (success == 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &addr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef NM_PLATFORM_H
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMPlatformIP6Address *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_platform_ip6_address (const char *address, const char *peer_address, guint plen)
|
|
|
|
|
{
|
|
|
|
|
static NMPlatformIP6Address addr;
|
|
|
|
|
|
|
|
|
|
memset (&addr, 0, sizeof (addr));
|
|
|
|
|
addr.address = *nmtst_inet6_from_string (address);
|
|
|
|
|
addr.peer_address = *nmtst_inet6_from_string (peer_address);
|
|
|
|
|
addr.plen = plen;
|
|
|
|
|
|
|
|
|
|
return &addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMPlatformIP6Address *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_platform_ip6_address_full (const char *address, const char *peer_address, guint plen,
|
|
|
|
|
int ifindex, NMPlatformSource source, guint32 timestamp,
|
|
|
|
|
guint32 lifetime, guint32 preferred, guint flags)
|
|
|
|
|
{
|
|
|
|
|
NMPlatformIP6Address *addr = nmtst_platform_ip6_address (address, peer_address, plen);
|
|
|
|
|
|
|
|
|
|
addr->ifindex = ifindex;
|
|
|
|
|
addr->source = source;
|
|
|
|
|
addr->timestamp = timestamp;
|
|
|
|
|
addr->lifetime = lifetime;
|
|
|
|
|
addr->preferred = preferred;
|
|
|
|
|
addr->flags = flags;
|
|
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMPlatformIP6Route *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_platform_ip6_route (const char *network, guint plen, const char *gateway)
|
|
|
|
|
{
|
|
|
|
|
static NMPlatformIP6Route route;
|
|
|
|
|
|
|
|
|
|
memset (&route, 0, sizeof (route));
|
|
|
|
|
route.network = *nmtst_inet6_from_string (network);
|
|
|
|
|
route.plen = plen;
|
|
|
|
|
route.gateway = *nmtst_inet6_from_string (gateway);
|
|
|
|
|
|
|
|
|
|
return &route;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMPlatformIP6Route *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_platform_ip6_route_full (const char *network, guint plen, const char *gateway,
|
|
|
|
|
int ifindex, NMPlatformSource source,
|
|
|
|
|
guint metric, guint mss)
|
|
|
|
|
{
|
|
|
|
|
NMPlatformIP6Route *route = nmtst_platform_ip6_route (network, plen, gateway);
|
|
|
|
|
|
|
|
|
|
route->ifindex = ifindex;
|
|
|
|
|
route->source = source;
|
|
|
|
|
route->metric = metric;
|
|
|
|
|
route->mss = mss;
|
|
|
|
|
|
|
|
|
|
return route;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef NM_IP4_CONFIG_H
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMIP4Config *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_ip4_config_clone (NMIP4Config *config)
|
|
|
|
|
{
|
|
|
|
|
NMIP4Config *copy = nm_ip4_config_new ();
|
|
|
|
|
|
|
|
|
|
g_assert (copy);
|
|
|
|
|
g_assert (config);
|
|
|
|
|
nm_ip4_config_replace (copy, config, NULL);
|
|
|
|
|
return copy;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef NM_IP6_CONFIG_H
|
|
|
|
|
|
2014-04-17 14:55:57 +02:00
|
|
|
inline static NMIP6Config *
|
2014-03-14 17:49:49 +01:00
|
|
|
nmtst_ip6_config_clone (NMIP6Config *config)
|
|
|
|
|
{
|
|
|
|
|
NMIP6Config *copy = nm_ip6_config_new ();
|
|
|
|
|
|
|
|
|
|
g_assert (copy);
|
|
|
|
|
g_assert (config);
|
|
|
|
|
nm_ip6_config_replace (copy, config, NULL);
|
|
|
|
|
return copy;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endif /* __NM_TEST_UTILS_H__ */
|
|
|
|
|
|