Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add mpls auto mode to update mpls control upon routing events #13965

Open
wants to merge 8 commits into
base: master
Choose a base branch
from
6 changes: 4 additions & 2 deletions doc/user/zebra.rst
Original file line number Diff line number Diff line change
Expand Up @@ -170,13 +170,15 @@ Standard Commands
Set description for the interface.


.. clicmd:: mpls <enable|disable>
.. clicmd:: mpls <enable|disable|auto>

Choose mpls kernel processing value on the interface, for linux. Interfaces
configured with mpls will not automatically turn on if mpls kernel modules do not
happen to be loaded. This command will fail on 3.X linux kernels and does not
work on non-linux systems at all. 'enable' and 'disable' will respectively turn
on and off mpls on the given interface.
on and off mpls on the given interface. The 'auto' mode will enable mpls on the
given interface, if an mpls entry or a labeled route happens to be installed and
needs mpls processing to be turned on.

.. clicmd:: multicast <enable|disable>

Expand Down
18 changes: 18 additions & 0 deletions ldpd/interface.c
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
*/

#include <zebra.h>
#include "privs.h"

#include "ldpd.h"
#include "ldpe.h"
Expand All @@ -16,6 +17,8 @@

#include "sockopt.h"

extern struct zebra_privs_t ldpe_privs;

static __inline int iface_compare(const struct iface *, const struct iface *);
static struct if_addr *if_addr_new(struct kaddr *);
static struct if_addr *if_addr_lookup(struct if_addr_head *, struct kaddr *);
Expand Down Expand Up @@ -93,6 +96,20 @@ ldpe_if_init(struct iface *iface)
ldp_sync_fsm_init(iface, LDP_SYNC_STA_NOT_ACH);
}

/* proc sys net mpls conf <name> input set to 1 or 0
* should be used from ldpe only
*/
void ldpe_interface_set_mpls(struct iface *iface, bool val)
{
struct ldp_igp_configure_if_mpls config;

config.ifindex = iface->ifindex;
config.enabled = val;

ldpe_imsg_compose_parent(IMSG_LDP_CONFIGURE_IF_MPLS, getpid(), &config,
sizeof(config));
}

void
ldpe_if_exit(struct iface *iface)
{
Expand Down Expand Up @@ -277,6 +294,7 @@ if_start(struct iface *iface, int af)

log_debug("%s: %s address-family %s", __func__, iface->name, af_name(af));

ldpe_interface_set_mpls(iface, true);
ia = iface_af_get(iface, af);

gettimeofday(&now, NULL);
Expand Down
9 changes: 9 additions & 0 deletions ldpd/ldp_zebra.c
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,15 @@ ldp_sync_zebra_send_state_update(struct ldp_igp_sync_if_state *state)
return 0;
}

int ldp_sync_zebra_send_configure_if_mpls(struct ldp_igp_configure_if_mpls *config)
{
if (zebra_send_interface_mpls_set(zclient, VRF_DEFAULT, config->ifindex,
config->enabled) <= 0)
return -1;

return zclient_send_message(zclient);
}

static int
ldp_sync_zebra_send_announce(void)
{
Expand Down
10 changes: 10 additions & 0 deletions ldpd/ldpd.c
Original file line number Diff line number Diff line change
Expand Up @@ -610,6 +610,15 @@ static void main_dispatch_ldpe(struct event *thread)

ldp_sync_zebra_send_state_update((struct ldp_igp_sync_if_state *)imsg.data);
break;
case IMSG_LDP_CONFIGURE_IF_MPLS:
if (imsg.hdr.len !=
IMSG_HEADER_SIZE +
sizeof(struct ldp_igp_configure_if_mpls))
fatalx("IMSG_LDP_CONFIGURE_IF_MPLS imsg with wrong len");

ldp_sync_zebra_send_configure_if_mpls(
(struct ldp_igp_configure_if_mpls *)imsg.data);
break;
default:
log_debug("%s: error handling imsg %d", __func__,
imsg.hdr.type);
Expand Down Expand Up @@ -1524,6 +1533,7 @@ merge_ifaces(struct ldpd_conf *conf, struct ldpd_conf *xconf)
if (if_lookup_name(xconf, iface->name) == NULL) {
switch (ldpd_process) {
case PROC_LDP_ENGINE:
ldpe_interface_set_mpls(iface, false);
ldpe_if_exit(iface);
break;
case PROC_LDE_ENGINE:
Expand Down
2 changes: 2 additions & 0 deletions ldpd/ldpd.h
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,7 @@ enum imsg_type {
IMSG_RLFA_UNREG_ALL,
IMSG_RLFA_LABELS,
IMSG_AGENTX_ENABLED,
IMSG_LDP_CONFIGURE_IF_MPLS,
};

struct ldpd_init {
Expand Down Expand Up @@ -890,6 +891,7 @@ extern char ctl_sock_path[MAXPATHLEN];
void ldp_zebra_init(struct event_loop *m);
void ldp_zebra_destroy(void);
int ldp_sync_zebra_send_state_update(struct ldp_igp_sync_if_state *);
int ldp_sync_zebra_send_configure_if_mpls(struct ldp_igp_configure_if_mpls *);
int ldp_zebra_send_rlfa_labels(struct zapi_rlfa_response *
rlfa_labels);

Expand Down
1 change: 1 addition & 0 deletions ldpd/ldpe.h
Original file line number Diff line number Diff line change
Expand Up @@ -306,6 +306,7 @@ void ldpe_l2vpn_init(struct l2vpn *);
void ldpe_l2vpn_exit(struct l2vpn *);
void ldpe_l2vpn_pw_init(struct l2vpn_pw *);
void ldpe_l2vpn_pw_exit(struct l2vpn_pw *);
extern void ldpe_interface_set_mpls(struct iface *iface, bool val);

DECLARE_HOOK(ldp_nbr_state_change, (struct nbr * nbr, int old_state),
(nbr, old_state));
Expand Down
4 changes: 4 additions & 0 deletions lib/defaults.c
Original file line number Diff line number Diff line change
Expand Up @@ -166,6 +166,8 @@ static void frr_default_apply_one(struct frr_default *dflt, bool check)
*dflt->dflt_ulong = dfltentry->val_ulong;
if (dflt->dflt_float)
*dflt->dflt_float = dfltentry->val_float;
if (dflt->dflt_uint8_t)
*dflt->dflt_uint8_t = dfltentry->val_uint8_t;
if (dflt->save_bool)
*dflt->save_bool = saveentry->val_bool;
if (dflt->save_str)
Expand All @@ -176,6 +178,8 @@ static void frr_default_apply_one(struct frr_default *dflt, bool check)
*dflt->save_ulong = saveentry->val_ulong;
if (dflt->save_float)
*dflt->save_float = saveentry->val_float;
if (dflt->save_uint8_t)
*dflt->save_uint8_t = saveentry->val_uint8_t;
}

void frr_defaults_apply(void)
Expand Down
5 changes: 5 additions & 0 deletions lib/defaults.h
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ struct frr_default_entry {
long val_long;
unsigned long val_ulong;
float val_float;
uint8_t val_uint8_t;
};

/* one struct frr_default exists for each malleable default value */
Expand All @@ -62,13 +63,15 @@ struct frr_default {
long *dflt_long;
unsigned long *dflt_ulong;
float *dflt_float;
uint8_t *dflt_uint8_t;

/* variable to use when comparing for config save */
bool *save_bool;
const char **save_str;
long *save_long;
unsigned long *save_ulong;
float *save_float;
uint8_t *save_uint8_t;

struct frr_default_entry entries[];
};
Expand Down Expand Up @@ -113,6 +116,8 @@ struct frr_default {
_FRR_CFG_DEFAULT(float, float, varname, ## __VA_ARGS__)
#define FRR_CFG_DEFAULT_STR(varname, ...) \
_FRR_CFG_DEFAULT(const char *, str, varname, ## __VA_ARGS__)
#define FRR_CFG_DEFAULT_UINT8_T(varname, ...) \
_FRR_CFG_DEFAULT(uint8_t, uint8_t, varname, ##__VA_ARGS__)


/* daemons don't need to call any of these, libfrr handles that */
Expand Down
5 changes: 5 additions & 0 deletions lib/ldp_sync.h
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,11 @@ struct ldp_igp_sync_if_state {
bool sync_start;
};

struct ldp_igp_configure_if_mpls {
ifindex_t ifindex;
bool enabled;
};

struct ldp_igp_sync_if_state_req {
int proto;
ifindex_t ifindex;
Expand Down
45 changes: 45 additions & 0 deletions lib/zclient.c
Original file line number Diff line number Diff line change
Expand Up @@ -2867,6 +2867,51 @@ size_t zebra_interface_link_params_write(struct stream *s,
return w;
}

int zebra_send_interface_mpls_set(struct zclient *zclient, vrf_id_t vrf_id,
ifindex_t ifindex, bool set)
{
struct stream *s;

if (!zclient->obuf)
return 0;

s = zclient->obuf;
stream_reset(s);

zclient_create_header(s, ZEBRA_INTERFACE_MPLS_SET, VRF_DEFAULT);

stream_putl(s, ifindex);
stream_putc(s, set);

stream_putw_at(s, 0, stream_get_endp(s));

return zclient_send_message(zclient);
}

struct interface *zebra_interface_mpls_set_read(struct stream *s,
vrf_id_t vrf_id,
bool *value_to_set)
{
ifindex_t ifindex;
struct interface *ifp = NULL;

STREAM_GETL(s, ifindex);

ifp = if_lookup_by_index(ifindex, vrf_id);

if (ifp == NULL) {
flog_err(EC_LIB_ZAPI_ENCODE,
"%s: unknown ifindex %u, shouldn't happen", __func__,
ifindex);
return NULL;
}
if (value_to_set)
STREAM_GETC(s, *value_to_set);

stream_failure:
return ifp;
}

/*
* format of message for address addition is:
* 0
Expand Down
7 changes: 7 additions & 0 deletions lib/zclient.h
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,7 @@ typedef enum {
ZEBRA_INTERFACE_DISABLE_RADV,
ZEBRA_NEXTHOP_LOOKUP_MRIB,
ZEBRA_INTERFACE_LINK_PARAMS,
ZEBRA_INTERFACE_MPLS_SET,
ZEBRA_MPLS_LABELS_ADD,
ZEBRA_MPLS_LABELS_DELETE,
ZEBRA_MPLS_LABELS_REPLACE,
Expand Down Expand Up @@ -1055,6 +1056,12 @@ extern struct interface *zebra_interface_link_params_read(struct stream *s,
bool *changed);
extern size_t zebra_interface_link_params_write(struct stream *,
struct interface *);
extern int zebra_send_interface_mpls_set(struct zclient *zclient,
vrf_id_t vrf_id, ifindex_t ifindex,
bool set);
extern struct interface *zebra_interface_mpls_set_read(struct stream *s,
vrf_id_t vrf_id,
bool *value_to_set);
extern enum zclient_send_status
zclient_send_get_label_chunk(struct zclient *zclient, uint8_t keep,
uint32_t chunk_size, uint32_t base);
Expand Down
1 change: 1 addition & 0 deletions tests/topotests/ldp_topo1/r1/zebra.conf
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ interface r1-eth0
description to sw0
ip address 10.0.1.1/24
no link-detect
mpls auto
!
ip forwarding
!
Expand Down
3 changes: 3 additions & 0 deletions tests/topotests/ldp_topo1/r2/zebra.conf
Original file line number Diff line number Diff line change
Expand Up @@ -9,16 +9,19 @@ interface r2-eth0
description to sw0
ip address 10.0.1.2/24
no link-detect
mpls auto
!
interface r2-eth1
description to sw1
ip address 10.0.2.2/24
no link-detect
mpls auto
!
interface r2-eth2
description to sw2
ip address 10.0.3.2/24
no link-detect
mpls auto
!
ip forwarding
!
Expand Down
2 changes: 2 additions & 0 deletions tests/topotests/ldp_topo1/r3/zebra.conf
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,13 @@ interface r3-eth0
description to sw1
ip address 10.0.2.3/24
no link-detect
mpls auto
!
interface r3-eth1
description to sw2
ip address 10.0.3.3/24
no link-detect
mpls auto
!
ip forwarding
!
Expand Down
1 change: 1 addition & 0 deletions tests/topotests/ldp_topo1/r4/zebra.conf
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ interface r4-eth0
description to sw1
ip address 10.0.2.4/24
no link-detect
mpls auto
!
ip forwarding
!
Expand Down
34 changes: 32 additions & 2 deletions tests/topotests/ldp_topo1/test_ldp_topo1.py
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,6 @@


def build_topo(tgen):

# Setup Routers
for i in range(1, 5):
tgen.add_router("r%s" % i)
Expand Down Expand Up @@ -134,6 +133,7 @@ def setup_module(module):
tgen.start_topology()

net = tgen.net
os.environ["TOPOTESTS_MPLS_AUTO"] = "1"

# Starting Routers
for i in range(1, 5):
Expand Down Expand Up @@ -781,6 +781,37 @@ def test_linux_mpls_routes():
assert fatal_error == "", fatal_error


def _check_mpls_interface_set(rname, ifname, value):
tgen = get_topogen()
val = (
tgen.net[rname]
.cmd_raises("sysctl -n " + "net.mpls.conf." + ifname + ".input")
.strip()
)
assert (
val == value
), f"{rname}, interface {ifname} has MPLS flag set to {val}, unexpected"


def test_linux_mpls_interface_settings():
global fatal_error

# Skip if previous fatal error condition is raised
if fatal_error != "":
pytest.skip(fatal_error)

# Verify Linux Kernel MPLS settings
print("\n\n** Verifying Linux MPLS interface settings")
print("******************************************\n")
_check_mpls_interface_set("r1", "r1-eth0", "1")
_check_mpls_interface_set("r2", "r2-eth0", "1")
_check_mpls_interface_set("r2", "r2-eth1", "1")
_check_mpls_interface_set("r2", "r2-eth2", "1")
_check_mpls_interface_set("r3", "r3-eth0", "1")
_check_mpls_interface_set("r3", "r3-eth1", "1")
_check_mpls_interface_set("r4", "r4-eth0", "1")


def test_shutdown_check_stderr():
global fatal_error
net = get_topogen().net
Expand Down Expand Up @@ -837,7 +868,6 @@ def test_shutdown_check_memleak():


if __name__ == "__main__":

# To suppress tracebacks, either use the following pytest call or add "--tb=no" to cli
# retval = pytest.main(["-s", "--tb=no"])
retval = pytest.main(["-s"])
Expand Down
6 changes: 5 additions & 1 deletion tests/topotests/lib/topogen.py
Original file line number Diff line number Diff line change
Expand Up @@ -907,9 +907,13 @@ def start(self):
daemon=daemon,
)

if nrouter.daemons["ldpd"] == 1 and not os.environ.get("TOPOTESTS_MPLS_AUTO"):
need_mpls_for_ldp = True
else:
need_mpls_for_ldp = False
if result != "":
self.tgen.set_error(result)
elif nrouter.daemons["ldpd"] == 1 or nrouter.daemons["pathd"] == 1:
elif need_mpls_for_ldp or nrouter.daemons["pathd"] == 1:
# Enable MPLS processing on all interfaces.
for interface in self.links:
topotest.sysctl_assure(
Expand Down
Loading
Loading