|
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__ */
|