Blame src/ndisc/nm-fake-ndisc.c

Packit Service b23acc
// SPDX-License-Identifier: GPL-2.0+
Packit Service b23acc
/*
Packit Service b23acc
 * Copyright (C) 2013 Red Hat, Inc.
Packit Service b23acc
 */
Packit Service b23acc
Packit Service b23acc
#include "nm-default.h"
Packit Service b23acc
Packit Service b23acc
#include "nm-fake-ndisc.h"
Packit Service b23acc
Packit Service b23acc
#include <arpa/inet.h>
Packit Service b23acc
Packit Service b23acc
#include "nm-ndisc-private.h"
Packit Service b23acc
Packit Service b23acc
#define _NMLOG_PREFIX_NAME                "ndisc-fake"
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	guint id;
Packit Service b23acc
	guint when;
Packit Service b23acc
Packit Service b23acc
	NMNDiscDHCPLevel dhcp_level;
Packit Service b23acc
	GArray *gateways;
Packit Service b23acc
	GArray *prefixes;
Packit Service b23acc
	GArray *dns_servers;
Packit Service b23acc
	GArray *dns_domains;
Packit Service b23acc
	int hop_limit;
Packit Service b23acc
	guint32 mtu;
Packit Service b23acc
} FakeRa;
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
        struct in6_addr network;
Packit Service b23acc
        int plen;
Packit Service b23acc
        struct in6_addr gateway;
Packit Service b23acc
        guint32 timestamp;
Packit Service b23acc
        guint32 lifetime;
Packit Service b23acc
        guint32 preferred;
Packit Service b23acc
        NMIcmpv6RouterPref preference;
Packit Service b23acc
} FakePrefix;
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
enum {
Packit Service b23acc
	RS_SENT,
Packit Service b23acc
	LAST_SIGNAL,
Packit Service b23acc
};
Packit Service b23acc
static guint signals[LAST_SIGNAL] = { 0 };
Packit Service b23acc
Packit Service b23acc
typedef struct {
Packit Service b23acc
	guint receive_ra_id;
Packit Service b23acc
	GSList *ras;
Packit Service b23acc
} NMFakeNDiscPrivate;
Packit Service b23acc
Packit Service b23acc
struct _NMFakeRNDisc {
Packit Service b23acc
	NMNDisc parent;
Packit Service b23acc
	NMFakeNDiscPrivate _priv;
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
struct _NMFakeRNDiscClass {
Packit Service b23acc
	NMNDiscClass parent;
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
G_DEFINE_TYPE (NMFakeNDisc, nm_fake_ndisc, NM_TYPE_NDISC)
Packit Service b23acc
Packit Service b23acc
#define NM_FAKE_NDISC_GET_PRIVATE(self) _NM_GET_PRIVATE (self, NMFakeNDisc, NM_IS_FAKE_NDISC, NMNDisc)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
fake_ra_free (gpointer data)
Packit Service b23acc
{
Packit Service b23acc
	FakeRa *ra = data;
Packit Service b23acc
Packit Service b23acc
	g_array_free (ra->gateways, TRUE);
Packit Service b23acc
	g_array_free (ra->prefixes, TRUE);
Packit Service b23acc
	g_array_free (ra->dns_servers, TRUE);
Packit Service b23acc
	g_array_free (ra->dns_domains, TRUE);
Packit Service b23acc
	g_free (ra);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
ra_dns_domain_free (gpointer data)
Packit Service b23acc
{
Packit Service b23acc
	g_free (((NMNDiscDNSDomain *)(data))->domain);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static FakeRa *
Packit Service b23acc
find_ra (GSList *ras, guint id)
Packit Service b23acc
{
Packit Service b23acc
	GSList *iter;
Packit Service b23acc
Packit Service b23acc
	for (iter = ras; iter; iter = iter->next) {
Packit Service b23acc
		if (((FakeRa *) iter->data)->id == id)
Packit Service b23acc
			return iter->data;
Packit Service b23acc
	}
Packit Service b23acc
	return NULL;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
guint
Packit Service b23acc
nm_fake_ndisc_add_ra (NMFakeNDisc *self,
Packit Service b23acc
                      guint seconds_after_previous,
Packit Service b23acc
                      NMNDiscDHCPLevel dhcp_level,
Packit Service b23acc
                      int hop_limit,
Packit Service b23acc
                      guint32 mtu)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	static guint counter = 1;
Packit Service b23acc
	FakeRa *ra;
Packit Service b23acc
Packit Service b23acc
	ra = g_malloc0 (sizeof (*ra));
Packit Service b23acc
	ra->id = counter++;
Packit Service b23acc
	ra->when = seconds_after_previous;
Packit Service b23acc
	ra->dhcp_level = dhcp_level;
Packit Service b23acc
	ra->hop_limit = hop_limit;
Packit Service b23acc
	ra->mtu = mtu;
Packit Service b23acc
	ra->gateways = g_array_new (FALSE, FALSE, sizeof (NMNDiscGateway));
Packit Service b23acc
	ra->prefixes = g_array_new (FALSE, FALSE, sizeof (FakePrefix));
Packit Service b23acc
	ra->dns_servers = g_array_new (FALSE, FALSE, sizeof (NMNDiscDNSServer));
Packit Service b23acc
	ra->dns_domains = g_array_new (FALSE, FALSE, sizeof (NMNDiscDNSDomain));
Packit Service b23acc
	g_array_set_clear_func (ra->dns_domains, ra_dns_domain_free);
Packit Service b23acc
Packit Service b23acc
	priv->ras = g_slist_append (priv->ras, ra);
Packit Service b23acc
	return ra->id;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nm_fake_ndisc_add_gateway (NMFakeNDisc *self,
Packit Service b23acc
                           guint ra_id,
Packit Service b23acc
                           const char *addr,
Packit Service b23acc
                           guint32 timestamp,
Packit Service b23acc
                           guint32 lifetime,
Packit Service b23acc
                           NMIcmpv6RouterPref preference)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	FakeRa *ra = find_ra (priv->ras, ra_id);
Packit Service b23acc
	NMNDiscGateway *gw;
Packit Service b23acc
Packit Service b23acc
	g_assert (ra);
Packit Service b23acc
	g_array_set_size (ra->gateways, ra->gateways->len + 1);
Packit Service b23acc
	gw = &g_array_index (ra->gateways, NMNDiscGateway, ra->gateways->len - 1);
Packit Service b23acc
	g_assert (inet_pton (AF_INET6, addr, &gw->address) == 1);
Packit Service b23acc
	gw->timestamp = timestamp;
Packit Service b23acc
	gw->lifetime = lifetime;
Packit Service b23acc
	gw->preference = preference;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nm_fake_ndisc_add_prefix (NMFakeNDisc *self,
Packit Service b23acc
                          guint ra_id,
Packit Service b23acc
                          const char *network,
Packit Service b23acc
                          guint plen,
Packit Service b23acc
                          const char *gateway,
Packit Service b23acc
                          guint32 timestamp,
Packit Service b23acc
                          guint32 lifetime,
Packit Service b23acc
                          guint32 preferred,
Packit Service b23acc
                          NMIcmpv6RouterPref preference)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	FakeRa *ra = find_ra (priv->ras, ra_id);
Packit Service b23acc
	FakePrefix *prefix;
Packit Service b23acc
Packit Service b23acc
	g_assert (ra);
Packit Service b23acc
	g_array_set_size (ra->prefixes, ra->prefixes->len + 1);
Packit Service b23acc
	prefix = &g_array_index (ra->prefixes, FakePrefix, ra->prefixes->len - 1);
Packit Service b23acc
	memset (prefix, 0, sizeof (*prefix));
Packit Service b23acc
	g_assert (inet_pton (AF_INET6, network, &prefix->network) == 1);
Packit Service b23acc
	g_assert (inet_pton (AF_INET6, gateway, &prefix->gateway) == 1);
Packit Service b23acc
	prefix->plen = plen;
Packit Service b23acc
	prefix->timestamp = timestamp;
Packit Service b23acc
	prefix->lifetime = lifetime;
Packit Service b23acc
	prefix->preferred = preferred;
Packit Service b23acc
	prefix->preference = preference;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nm_fake_ndisc_add_dns_server (NMFakeNDisc *self,
Packit Service b23acc
                              guint ra_id,
Packit Service b23acc
                              const char *address,
Packit Service b23acc
                              guint32 timestamp,
Packit Service b23acc
                              guint32 lifetime)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	FakeRa *ra = find_ra (priv->ras, ra_id);
Packit Service b23acc
	NMNDiscDNSServer *dns;
Packit Service b23acc
Packit Service b23acc
	g_assert (ra);
Packit Service b23acc
	g_array_set_size (ra->dns_servers, ra->dns_servers->len + 1);
Packit Service b23acc
	dns = &g_array_index (ra->dns_servers, NMNDiscDNSServer, ra->dns_servers->len - 1);
Packit Service b23acc
	g_assert (inet_pton (AF_INET6, address, &dns->address) == 1);
Packit Service b23acc
	dns->timestamp = timestamp;
Packit Service b23acc
	dns->lifetime = lifetime;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nm_fake_ndisc_add_dns_domain (NMFakeNDisc *self,
Packit Service b23acc
                              guint ra_id,
Packit Service b23acc
                              const char *domain,
Packit Service b23acc
                              guint32 timestamp,
Packit Service b23acc
                              guint32 lifetime)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	FakeRa *ra = find_ra (priv->ras, ra_id);
Packit Service b23acc
	NMNDiscDNSDomain *dns;
Packit Service b23acc
Packit Service b23acc
	g_assert (ra);
Packit Service b23acc
	g_array_set_size (ra->dns_domains, ra->dns_domains->len + 1);
Packit Service b23acc
	dns = &g_array_index (ra->dns_domains, NMNDiscDNSDomain, ra->dns_domains->len - 1);
Packit Service b23acc
	dns->domain = g_strdup (domain);
Packit Service b23acc
	dns->timestamp = timestamp;
Packit Service b23acc
	dns->lifetime = lifetime;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
gboolean
Packit Service b23acc
nm_fake_ndisc_done (NMFakeNDisc *self)
Packit Service b23acc
{
Packit Service b23acc
	return !NM_FAKE_NDISC_GET_PRIVATE (self)->ras;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
send_rs (NMNDisc *ndisc, GError **error)
Packit Service b23acc
{
Packit Service b23acc
	g_signal_emit (ndisc, signals[RS_SENT], 0);
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
receive_ra (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDisc *self = user_data;
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (self);
Packit Service b23acc
	NMNDisc *ndisc = NM_NDISC (self);
Packit Service b23acc
	NMNDiscDataInternal *rdata = ndisc->rdata;
Packit Service b23acc
	FakeRa *ra = priv->ras->data;
Packit Service b23acc
	NMNDiscConfigMap changed = 0;
Packit Service b23acc
	gint32 now = nm_utils_get_monotonic_timestamp_sec ();
Packit Service b23acc
	guint i;
Packit Service b23acc
	NMNDiscDHCPLevel dhcp_level;
Packit Service b23acc
Packit Service b23acc
	priv->receive_ra_id = 0;
Packit Service b23acc
Packit Service b23acc
	/* preserve the "most managed" level  on updates. */
Packit Service b23acc
	dhcp_level = MAX (rdata->public.dhcp_level, ra->dhcp_level);
Packit Service b23acc
Packit Service b23acc
	if (rdata->public.dhcp_level != dhcp_level) {
Packit Service b23acc
		rdata->public.dhcp_level = dhcp_level;
Packit Service b23acc
		changed |= NM_NDISC_CONFIG_DHCP_LEVEL;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < ra->gateways->len; i++) {
Packit Service b23acc
		NMNDiscGateway *item = &g_array_index (ra->gateways, NMNDiscGateway, i);
Packit Service b23acc
Packit Service b23acc
		if (nm_ndisc_add_gateway (ndisc, item))
Packit Service b23acc
			changed |= NM_NDISC_CONFIG_GATEWAYS;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < ra->prefixes->len; i++) {
Packit Service b23acc
		FakePrefix *item = &g_array_index (ra->prefixes, FakePrefix, i);
Packit Service b23acc
		NMNDiscRoute route = {
Packit Service b23acc
			.network = item->network,
Packit Service b23acc
			.plen = item->plen,
Packit Service b23acc
			.gateway = item->gateway,
Packit Service b23acc
			.timestamp = item->timestamp,
Packit Service b23acc
			.lifetime = item->lifetime,
Packit Service b23acc
			.preference = item->preference,
Packit Service b23acc
		};
Packit Service b23acc
Packit Service b23acc
		g_assert (route.plen > 0 && route.plen <= 128);
Packit Service b23acc
Packit Service b23acc
		if (nm_ndisc_add_route (ndisc, &route))
Packit Service b23acc
			changed |= NM_NDISC_CONFIG_ROUTES;
Packit Service b23acc
Packit Service b23acc
		if (item->plen == 64) {
Packit Service b23acc
			NMNDiscAddress address = {
Packit Service b23acc
				.address = item->network,
Packit Service b23acc
				.timestamp = item->timestamp,
Packit Service b23acc
				.lifetime = item->lifetime,
Packit Service b23acc
				.preferred = item->preferred,
Packit Service b23acc
				.dad_counter = 0,
Packit Service b23acc
			};
Packit Service b23acc
Packit Service b23acc
			if (nm_ndisc_complete_and_add_address (ndisc, &address, now))
Packit Service b23acc
				changed |= NM_NDISC_CONFIG_ADDRESSES;
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < ra->dns_servers->len; i++) {
Packit Service b23acc
		NMNDiscDNSServer *item = &g_array_index (ra->dns_servers, NMNDiscDNSServer, i);
Packit Service b23acc
Packit Service b23acc
		if (nm_ndisc_add_dns_server (ndisc, item))
Packit Service b23acc
			changed |= NM_NDISC_CONFIG_DNS_SERVERS;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	for (i = 0; i < ra->dns_domains->len; i++) {
Packit Service b23acc
		NMNDiscDNSDomain *item = &g_array_index (ra->dns_domains, NMNDiscDNSDomain, i);
Packit Service b23acc
Packit Service b23acc
		if (nm_ndisc_add_dns_domain (ndisc, item))
Packit Service b23acc
			changed |= NM_NDISC_CONFIG_DNS_DOMAINS;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (rdata->public.mtu != ra->mtu) {
Packit Service b23acc
		rdata->public.mtu = ra->mtu;
Packit Service b23acc
		changed |= NM_NDISC_CONFIG_MTU;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (rdata->public.hop_limit != ra->hop_limit) {
Packit Service b23acc
		rdata->public.hop_limit = ra->hop_limit;
Packit Service b23acc
		changed |= NM_NDISC_CONFIG_HOP_LIMIT;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	priv->ras = g_slist_remove (priv->ras, priv->ras->data);
Packit Service b23acc
	fake_ra_free (ra);
Packit Service b23acc
Packit Service b23acc
	nm_ndisc_ra_received (NM_NDISC (self), now, changed);
Packit Service b23acc
Packit Service b23acc
	/* Schedule next RA */
Packit Service b23acc
	if (priv->ras) {
Packit Service b23acc
		ra = priv->ras->data;
Packit Service b23acc
		priv->receive_ra_id = g_timeout_add_seconds (ra->when, receive_ra, self);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
start (NMNDisc *ndisc)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (ndisc);
Packit Service b23acc
	FakeRa *ra;
Packit Service b23acc
Packit Service b23acc
	/* Queue up the first fake RA */
Packit Service b23acc
	g_assert (priv->ras);
Packit Service b23acc
	ra = priv->ras->data;
Packit Service b23acc
Packit Service b23acc
	g_assert (!priv->receive_ra_id);
Packit Service b23acc
	priv->receive_ra_id = g_timeout_add_seconds (ra->when, receive_ra, ndisc);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
void
Packit Service b23acc
nm_fake_ndisc_emit_new_ras (NMFakeNDisc *self)
Packit Service b23acc
{
Packit Service b23acc
	if (!NM_FAKE_NDISC_GET_PRIVATE (self)->receive_ra_id)
Packit Service b23acc
		start (NM_NDISC (self));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nm_fake_ndisc_init (NMFakeNDisc *fake_ndisc)
Packit Service b23acc
{
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
NMNDisc *
Packit Service b23acc
nm_fake_ndisc_new (int ifindex, const char *ifname)
Packit Service b23acc
{
Packit Service b23acc
	return g_object_new (NM_TYPE_FAKE_NDISC,
Packit Service b23acc
	                     NM_NDISC_IFINDEX, ifindex,
Packit Service b23acc
	                     NM_NDISC_IFNAME, ifname,
Packit Service b23acc
	                     NM_NDISC_NODE_TYPE, (int) NM_NDISC_NODE_TYPE_HOST,
Packit Service b23acc
	                     NM_NDISC_STABLE_TYPE, (int) NM_UTILS_STABLE_TYPE_UUID,
Packit Service b23acc
	                     NM_NDISC_NETWORK_ID, "fake",
Packit Service b23acc
	                     NULL);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
dispose (GObject *object)
Packit Service b23acc
{
Packit Service b23acc
	NMFakeNDiscPrivate *priv = NM_FAKE_NDISC_GET_PRIVATE (object);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->receive_ra_id);
Packit Service b23acc
Packit Service b23acc
	g_slist_free_full (priv->ras, fake_ra_free);
Packit Service b23acc
	priv->ras = NULL;
Packit Service b23acc
Packit Service b23acc
	G_OBJECT_CLASS (nm_fake_ndisc_parent_class)->dispose (object);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nm_fake_ndisc_class_init (NMFakeNDiscClass *klass)
Packit Service b23acc
{
Packit Service b23acc
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit Service b23acc
	NMNDiscClass *ndisc_class = NM_NDISC_CLASS (klass);
Packit Service b23acc
Packit Service b23acc
	object_class->dispose = dispose;
Packit Service b23acc
Packit Service b23acc
	ndisc_class->start = start;
Packit Service b23acc
	ndisc_class->send_rs = send_rs;
Packit Service b23acc
Packit Service b23acc
	signals[RS_SENT] =
Packit Service b23acc
	    g_signal_new (NM_FAKE_NDISC_RS_SENT,
Packit Service b23acc
	                  G_OBJECT_CLASS_TYPE (klass),
Packit Service b23acc
	                  G_SIGNAL_RUN_FIRST,
Packit Service b23acc
	                  0,  NULL, NULL, NULL,
Packit Service b23acc
	                  G_TYPE_NONE, 0);
Packit Service b23acc
}