Blame src/core/ndisc/nm-ndisc.h

Packit Service 5ffa24
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit Service 5ffa24
/*
Packit Service 5ffa24
 * Copyright (C) 2013 Red Hat, Inc.
Packit Service 5ffa24
 */
Packit Service 5ffa24
Packit Service 5ffa24
#ifndef __NETWORKMANAGER_NDISC_H__
Packit Service 5ffa24
#define __NETWORKMANAGER_NDISC_H__
Packit Service 5ffa24
Packit Service 5ffa24
#include <stdlib.h>
Packit Service 5ffa24
#include <netinet/in.h>
Packit Service 5ffa24
#include <linux/if_addr.h>
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-setting-ip6-config.h"
Packit Service 5ffa24
#include "NetworkManagerUtils.h"
Packit Service 5ffa24
Packit Service dff8e4
#include "libnm-platform/nm-platform.h"
Packit Service dff8e4
#include "libnm-platform/nmp-object.h"
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_RA_TIMEOUT_DEFAULT  ((guint32) 0)
Packit Service 5ffa24
#define NM_RA_TIMEOUT_INFINITY ((guint32) G_MAXINT32)
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_TYPE_NDISC            (nm_ndisc_get_type())
Packit Service 5ffa24
#define NM_NDISC(obj)            (G_TYPE_CHECK_INSTANCE_CAST((obj), NM_TYPE_NDISC, NMNDisc))
Packit Service 5ffa24
#define NM_NDISC_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST((klass), NM_TYPE_NDISC, NMNDiscClass))
Packit Service 5ffa24
#define NM_IS_NDISC(obj)         (G_TYPE_CHECK_INSTANCE_TYPE((obj), NM_TYPE_NDISC))
Packit Service 5ffa24
#define NM_IS_NDISC_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), NM_TYPE_NDISC))
Packit Service 5ffa24
#define NM_NDISC_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS((obj), NM_TYPE_NDISC, NMNDiscClass))
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_NDISC_PLATFORM                     "platform"
Packit Service 5ffa24
#define NM_NDISC_IFINDEX                      "ifindex"
Packit Service 5ffa24
#define NM_NDISC_IFNAME                       "ifname"
Packit Service 5ffa24
#define NM_NDISC_NETWORK_ID                   "network-id"
Packit Service 5ffa24
#define NM_NDISC_ADDR_GEN_MODE                "addr-gen-mode"
Packit Service 5ffa24
#define NM_NDISC_STABLE_TYPE                  "stable-type"
Packit Service 5ffa24
#define NM_NDISC_NODE_TYPE                    "node-type"
Packit Service 5ffa24
#define NM_NDISC_MAX_ADDRESSES                "max-addresses"
Packit Service 5ffa24
#define NM_NDISC_RA_TIMEOUT                   "ra-timeout"
Packit Service 5ffa24
#define NM_NDISC_ROUTER_SOLICITATIONS         "router-solicitations"
Packit Service 5ffa24
#define NM_NDISC_ROUTER_SOLICITATION_INTERVAL "router-solicitation-interval"
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_NDISC_CONFIG_RECEIVED   "config-received"
Packit Service 5ffa24
#define NM_NDISC_RA_TIMEOUT_SIGNAL "ra-timeout-signal"
Packit Service 5ffa24
Packit Service 5ffa24
typedef enum {
Packit Service 5ffa24
    NM_NDISC_DHCP_LEVEL_UNKNOWN,
Packit Service 5ffa24
    NM_NDISC_DHCP_LEVEL_NONE,
Packit Service 5ffa24
    NM_NDISC_DHCP_LEVEL_OTHERCONF,
Packit Service 5ffa24
    NM_NDISC_DHCP_LEVEL_MANAGED
Packit Service 5ffa24
} NMNDiscDHCPLevel;
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_NDISC_INFINITY_U32 ((uint32_t) -1)
Packit Service 5ffa24
Packit Service 5ffa24
/* It's important that this is G_MAXINT64, so that we can meaningfully do
Packit Service 5ffa24
 * MIN(e1, e2) to find the minimum expiry time (and properly handle if any
Packit Service 5ffa24
 * of them is infinity).
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * While usually you assign this to "expiry_msec", you might say the
Packit Service 5ffa24
 * unit of it is milliseconds. But of course, infinity has not really a unit. */
Packit Service 5ffa24
#define NM_NDISC_EXPIRY_INFINITY G_MAXINT64
Packit Service 5ffa24
Packit Service 5ffa24
/* in common cases, the expiry_msec tracks the timestamp in nm_utils_get_monotonic_timestamp_mses()
Packit Service 5ffa24
 * timestamp when the item expires.
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * When we configure an NMNDiscAddress to be announced via the router advertisement,
Packit Service 5ffa24
 * then that address does not have a fixed expiry point in time, instead, the expiry
Packit Service 5ffa24
 * really contains the lifetime from the moment when we send the router advertisement.
Packit Service 5ffa24
 * In that case, the expiry_msec is more a "lifetime" that starts counting at timestamp
Packit Service 5ffa24
 * zero.
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * The unit is milliseconds (but of course, the timestamp is zero, so it doesn't really matter). */
Packit Service 5ffa24
#define NM_NDISC_EXPIRY_BASE_TIMESTAMP ((gint64) 0)
Packit Service 5ffa24
Packit Service 5ffa24
static inline gint64
Packit Service 5ffa24
_nm_ndisc_lifetime_to_expiry(gint64 now_msec, guint32 lifetime)
Packit Service 5ffa24
{
Packit Service 5ffa24
    if (lifetime == NM_NDISC_INFINITY_U32)
Packit Service 5ffa24
        return NM_NDISC_EXPIRY_INFINITY;
Packit Service 5ffa24
    return now_msec + (((gint64) lifetime) * 1000);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static inline gint64
Packit Service 5ffa24
_nm_ndisc_lifetime_from_expiry(gint64 now_msec, gint64 expiry_msec, gboolean ceil)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gint64 diff;
Packit Service 5ffa24
Packit Service 5ffa24
    if (expiry_msec == NM_NDISC_EXPIRY_INFINITY)
Packit Service 5ffa24
        return NM_NDISC_INFINITY_U32;
Packit Service 5ffa24
Packit Service 5ffa24
    /* we don't expect nor handle integer overflow. The time stamp and expiry
Packit Service 5ffa24
     * should be reasonably small so that it cannot happen. */
Packit Service 5ffa24
Packit Service 5ffa24
    diff = expiry_msec - now_msec;
Packit Service 5ffa24
Packit Service 5ffa24
    if (diff <= 0)
Packit Service 5ffa24
        return 0;
Packit Service 5ffa24
Packit Service 5ffa24
    if (ceil) {
Packit Service 5ffa24
        /* we ceil() towards the next full second (instead of floor()). */
Packit Service 5ffa24
        diff += 999;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return NM_MIN(diff / 1000, (gint64)(G_MAXUINT32 - 1));
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct _NMNDiscGateway {
Packit Service 5ffa24
    struct in6_addr    address;
Packit Service 5ffa24
    gint64             expiry_msec;
Packit Service 5ffa24
    NMIcmpv6RouterPref preference;
Packit Service 5ffa24
} NMNDiscGateway;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct _NMNDiscAddress {
Packit Service 5ffa24
    struct in6_addr address;
Packit Service 5ffa24
    gint64          expiry_msec;
Packit Service 5ffa24
    gint64          expiry_preferred_msec;
Packit Service 5ffa24
    guint8          dad_counter;
Packit Service 5ffa24
} NMNDiscAddress;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct _NMNDiscRoute {
Packit Service 5ffa24
    struct in6_addr    network;
Packit Service 5ffa24
    struct in6_addr    gateway;
Packit Service 5ffa24
    gint64             expiry_msec;
Packit Service 5ffa24
    NMIcmpv6RouterPref preference;
Packit Service 5ffa24
    guint8             plen;
Packit Service 5ffa24
} NMNDiscRoute;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    struct in6_addr address;
Packit Service 5ffa24
    gint64          expiry_msec;
Packit Service 5ffa24
} NMNDiscDNSServer;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    char * domain;
Packit Service 5ffa24
    gint64 expiry_msec;
Packit Service 5ffa24
} NMNDiscDNSDomain;
Packit Service 5ffa24
Packit Service 5ffa24
typedef enum {
Packit Service 5ffa24
    NM_NDISC_CONFIG_NONE           = 0,
Packit Service 5ffa24
    NM_NDISC_CONFIG_DHCP_LEVEL     = 1 << 0,
Packit Service 5ffa24
    NM_NDISC_CONFIG_GATEWAYS       = 1 << 1,
Packit Service 5ffa24
    NM_NDISC_CONFIG_ADDRESSES      = 1 << 2,
Packit Service 5ffa24
    NM_NDISC_CONFIG_ROUTES         = 1 << 3,
Packit Service 5ffa24
    NM_NDISC_CONFIG_DNS_SERVERS    = 1 << 4,
Packit Service 5ffa24
    NM_NDISC_CONFIG_DNS_DOMAINS    = 1 << 5,
Packit Service 5ffa24
    NM_NDISC_CONFIG_HOP_LIMIT      = 1 << 6,
Packit Service 5ffa24
    NM_NDISC_CONFIG_MTU            = 1 << 7,
Packit Service 5ffa24
    NM_NDISC_CONFIG_REACHABLE_TIME = 1 << 8,
Packit Service 5ffa24
    NM_NDISC_CONFIG_RETRANS_TIMER  = 1 << 9,
Packit Service 5ffa24
} NMNDiscConfigMap;
Packit Service 5ffa24
Packit Service 5ffa24
typedef enum {
Packit Service 5ffa24
    NM_NDISC_NODE_TYPE_INVALID,
Packit Service 5ffa24
    NM_NDISC_NODE_TYPE_HOST,
Packit Service 5ffa24
    NM_NDISC_NODE_TYPE_ROUTER,
Packit Service 5ffa24
} NMNDiscNodeType;
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_NDISC_RFC4861_RTR_SOLICITATION_INTERVAL  4 /* seconds */
Packit Service 5ffa24
#define NM_NDISC_RFC4861_MAX_RTR_SOLICITATION_DELAY 1 /* seconds */
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_NDISC_MAX_ADDRESSES_DEFAULT          16
Packit Service 5ffa24
#define NM_NDISC_ROUTER_SOLICITATIONS_DEFAULT   3 /* RFC4861, MAX_RTR_SOLICITATIONS */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_ADVERTISEMENTS_DEFAULT  3 /* RFC4861, MAX_INITIAL_RTR_ADVERTISEMENTS */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_ADVERT_DELAY            3 /* RFC4861, MIN_DELAY_BETWEEN_RAS */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_ADVERT_INITIAL_INTERVAL 16 /* RFC4861, MAX_INITIAL_RTR_ADVERT_INTERVAL */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_ADVERT_DELAY_MS         500 /* RFC4861, MAX_RA_DELAY_TIME */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_ADVERT_MAX_INTERVAL     600 /* RFC4861, MaxRtrAdvInterval default */
Packit Service 5ffa24
#define NM_NDISC_ROUTER_LIFETIME                900 /* 1.5 * NM_NDISC_ROUTER_ADVERT_MAX_INTERVAL */
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMNDiscPrivate;
Packit Service 5ffa24
struct _NMNDiscDataInternal;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    NMNDiscDHCPLevel dhcp_level;
Packit Service 5ffa24
    guint32          mtu;
Packit Service 5ffa24
    int              hop_limit;
Packit Service 5ffa24
    guint32          reachable_time_ms;
Packit Service 5ffa24
    guint32          retrans_timer_ms;
Packit Service 5ffa24
Packit Service 5ffa24
    guint gateways_n;
Packit Service 5ffa24
    guint addresses_n;
Packit Service 5ffa24
    guint routes_n;
Packit Service 5ffa24
    guint dns_servers_n;
Packit Service 5ffa24
    guint dns_domains_n;
Packit Service 5ffa24
Packit Service 5ffa24
    const NMNDiscGateway *  gateways;
Packit Service 5ffa24
    const NMNDiscAddress *  addresses;
Packit Service 5ffa24
    const NMNDiscRoute *    routes;
Packit Service 5ffa24
    const NMNDiscDNSServer *dns_servers;
Packit Service 5ffa24
    const NMNDiscDNSDomain *dns_domains;
Packit Service 5ffa24
} NMNDiscData;
Packit Service 5ffa24
Packit Service 5ffa24
/**
Packit Service 5ffa24
 * NMNDisc:
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * Interface-specific structure that handles incoming router advertisements,
Packit Service 5ffa24
 * caches advertised items and removes them when they are obsolete.
Packit Service 5ffa24
 */
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    GObject parent;
Packit Service 5ffa24
    union {
Packit Service 5ffa24
        struct _NMNDiscPrivate *     _priv;
Packit Service 5ffa24
        struct _NMNDiscDataInternal *rdata;
Packit Service 5ffa24
    };
Packit Service 5ffa24
} NMNDisc;
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    GObjectClass parent;
Packit Service 5ffa24
Packit Service 5ffa24
    void (*start)(NMNDisc *ndisc);
Packit Service 5ffa24
    void (*stop)(NMNDisc *ndisc);
Packit Service 5ffa24
    gboolean (*send_rs)(NMNDisc *ndisc, GError **error);
Packit Service 5ffa24
    gboolean (*send_ra)(NMNDisc *ndisc, GError **error);
Packit Service 5ffa24
} NMNDiscClass;
Packit Service 5ffa24
Packit Service 5ffa24
GType nm_ndisc_get_type(void);
Packit Service 5ffa24
Packit Service 5ffa24
void nm_ndisc_emit_config_change(NMNDisc *self, NMNDiscConfigMap changed);
Packit Service 5ffa24
Packit Service 5ffa24
int             nm_ndisc_get_ifindex(NMNDisc *self);
Packit Service 5ffa24
const char *    nm_ndisc_get_ifname(NMNDisc *self);
Packit Service 5ffa24
NMNDiscNodeType nm_ndisc_get_node_type(NMNDisc *self);
Packit Service 5ffa24
Packit Service 5ffa24
gboolean nm_ndisc_set_iid(NMNDisc *ndisc, const NMUtilsIPv6IfaceId iid);
Packit Service 5ffa24
void     nm_ndisc_start(NMNDisc *ndisc);
Packit Service 5ffa24
void     nm_ndisc_stop(NMNDisc *ndisc);
Packit Service 5ffa24
NMNDiscConfigMap
Packit Service 5ffa24
nm_ndisc_dad_failed(NMNDisc *ndisc, const struct in6_addr *address, gboolean emit_changed_signal);
Packit Service 5ffa24
void nm_ndisc_set_config(NMNDisc *     ndisc,
Packit Service 5ffa24
                         const GArray *addresses,
Packit Service 5ffa24
                         const GArray *dns_servers,
Packit Service 5ffa24
                         const GArray *dns_domains);
Packit Service 5ffa24
Packit Service 5ffa24
NMPlatform *nm_ndisc_get_platform(NMNDisc *self);
Packit Service 5ffa24
NMPNetns *  nm_ndisc_netns_get(NMNDisc *self);
Packit Service 5ffa24
gboolean    nm_ndisc_netns_push(NMNDisc *self, NMPNetns **netns);
Packit Service 5ffa24
Packit Service 5ffa24
static inline gboolean
Packit Service 5ffa24
nm_ndisc_dad_addr_is_fail_candidate_event(NMPlatformSignalChangeType  change_type,
Packit Service 5ffa24
                                          const NMPlatformIP6Address *addr)
Packit Service 5ffa24
{
Packit Service 5ffa24
    return !NM_FLAGS_HAS(addr->n_ifa_flags, IFA_F_TEMPORARY)
Packit Service 5ffa24
           && ((change_type == NM_PLATFORM_SIGNAL_CHANGED && addr->n_ifa_flags & IFA_F_DADFAILED)
Packit Service 5ffa24
               || (change_type == NM_PLATFORM_SIGNAL_REMOVED
Packit Service 5ffa24
                   && addr->n_ifa_flags & IFA_F_TENTATIVE));
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static inline gboolean
Packit Service 5ffa24
nm_ndisc_dad_addr_is_fail_candidate(NMPlatform *platform, const NMPObject *obj)
Packit Service 5ffa24
{
Packit Service 5ffa24
    const NMPlatformIP6Address *addr;
Packit Service 5ffa24
Packit Service 5ffa24
    addr = NMP_OBJECT_CAST_IP6_ADDRESS(
Packit Service 5ffa24
        nm_platform_lookup_obj(platform, NMP_CACHE_ID_TYPE_OBJECT_TYPE, obj));
Packit Service 5ffa24
    if (addr
Packit Service 5ffa24
        && (NM_FLAGS_HAS(addr->n_ifa_flags, IFA_F_TEMPORARY)
Packit Service 5ffa24
            || !NM_FLAGS_HAS(addr->n_ifa_flags, IFA_F_DADFAILED))) {
Packit Service 5ffa24
        /* the address still/again exists and is not in DADFAILED state. Skip it. */
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
struct _NML3ConfigData;
Packit Service 5ffa24
Packit Service 5ffa24
struct _NML3ConfigData *nm_ndisc_data_to_l3cd(NMDedupMultiIndex *       multi_idx,
Packit Service 5ffa24
                                              int                       ifindex,
Packit Service 5ffa24
                                              const NMNDiscData *       rdata,
Packit Service 5ffa24
                                              NMSettingIP6ConfigPrivacy ip6_privacy,
Packit Service 5ffa24
                                              guint32                   route_table,
Packit Service 5ffa24
                                              guint32                   route_metric,
Packit Service 5ffa24
                                              gboolean                  kernel_support_rta_pref,
Packit Service 5ffa24
                                              gboolean kernel_support_extended_ifa_flags);
Packit Service 5ffa24
Packit Service 5ffa24
#endif /* __NETWORKMANAGER_NDISC_H__ */