blob: b323d3e4a9615966912257e676c1139e31f51c10 [file] [log] [blame]
/*
* Copyright (c) 2016 Cisco and/or its affiliates.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <vnet/vnet.h>
#include <vnet/mfib/mfib_itf.h>
#include <vnet/mfib/mfib_signal.h>
#include <vnet/fib/fib_path.h>
#include <vnet/ethernet/mac_address.h>
mfib_itf_t *mfib_itf_pool;
index_t
mfib_itf_create (fib_node_index_t path_index,
mfib_itf_flags_t mfi_flags)
{
mfib_itf_t *mfib_itf;
pool_get_aligned(mfib_itf_pool, mfib_itf,
CLIB_CACHE_LINE_BYTES);
mfib_itf->mfi_sw_if_index = fib_path_get_resolving_interface(path_index);
mfib_itf->mfi_si = INDEX_INVALID;
/*
* add the path index to the per-path hash
*/
mfib_itf->mfi_hash = hash_set(mfib_itf->mfi_hash, path_index, mfi_flags);
/*
* the combined flags from all the paths is from just the one contributor
*/
mfib_itf->mfi_flags = mfi_flags;
return (mfib_itf - mfib_itf_pool);
}
static mfib_itf_flags_t
mfib_itf_mk_flags (const mfib_itf_t *mfib_itf)
{
mfib_itf_flags_t combined_flags, flags;
fib_node_index_t *path_index;
combined_flags = MFIB_ITF_FLAG_NONE;
hash_foreach(path_index, flags, mfib_itf->mfi_hash,
{
combined_flags |= flags;
});
return (combined_flags);
}
int
mfib_itf_update (mfib_itf_t *mfib_itf,
fib_node_index_t path_index,
mfib_itf_flags_t mfi_flags)
{
/*
* add or remove the path index to the per-path hash
*/
if (MFIB_ITF_FLAG_NONE == mfi_flags)
{
hash_unset(mfib_itf->mfi_hash, path_index);
}
else
{
mfib_itf->mfi_hash = hash_set(mfib_itf->mfi_hash,
path_index,
mfi_flags);
}
/*
* re-generate the combined flags from all the paths.
*/
mfib_itf->mfi_flags = mfib_itf_mk_flags(mfib_itf);
/*
* The interface can be removed if there are no more flags
*/
return (MFIB_ITF_FLAG_NONE == mfib_itf->mfi_flags);
}
static void
mfib_itf_hash_flush (mfib_itf_t *mfi)
{
fib_node_index_t path_index, *path_indexp, *all = NULL;
mfib_itf_flags_t flags;
hash_foreach(path_index, flags, mfi->mfi_hash,
{
vec_add1(all, path_index);
});
vec_foreach(path_indexp, all)
{
hash_unset(mfi->mfi_hash, *path_indexp);
};
}
static void
mfib_itf_prefix4_to_mac (const mfib_prefix_t *pfx,
mac_address_t *mac)
{
mac->bytes[0] = 0x01;
mac->bytes[1] = 0x0;
mac->bytes[2] = 0x5e;
mac->bytes[3] = pfx->fp_grp_addr.ip4.as_u8[1] & 0x7f;
mac->bytes[4] = pfx->fp_grp_addr.ip4.as_u8[2];
mac->bytes[5] = pfx->fp_grp_addr.ip4.as_u8[3];
}
static void
mfib_itf_prefix6_to_mac (const mfib_prefix_t *pfx,
mac_address_t *mac)
{
mac->bytes[0] = 0x33;
mac->bytes[1] = 0x33;
mac->bytes[2] = pfx->fp_grp_addr.ip6.as_u8[12];
mac->bytes[3] = pfx->fp_grp_addr.ip6.as_u8[13];
mac->bytes[4] = pfx->fp_grp_addr.ip6.as_u8[14];
mac->bytes[5] = pfx->fp_grp_addr.ip6.as_u8[15];
}
static void
mfib_itf_prefix_to_mac (const mfib_prefix_t *pfx,
mac_address_t *mac)
{
switch (pfx->fp_proto)
{
case FIB_PROTOCOL_IP4:
mfib_itf_prefix4_to_mac(pfx, mac);
break;
case FIB_PROTOCOL_IP6:
mfib_itf_prefix6_to_mac(pfx, mac);
break;
case FIB_PROTOCOL_MPLS:
break;
}
}
static void
mfib_itf_mac_add_del (mfib_itf_t *itf,
const mfib_prefix_t *pfx,
int add)
{
vnet_sw_interface_t *si;
vnet_main_t *vnm;
mac_address_t mac;
vnm = vnet_get_main();
mfib_itf_prefix_to_mac(pfx, &mac);
si = vnet_get_sw_interface(vnm, itf->mfi_sw_if_index);
vnet_hw_interface_add_del_mac_address (vnet_get_main(),
si->hw_if_index,
mac.bytes, add);
}
void
mfib_itf_mac_add (mfib_itf_t *itf,
const mfib_prefix_t *pfx)
{
mfib_itf_mac_add_del(itf, pfx, 1);
}
void
mfib_itf_mac_del (mfib_itf_t *itf,
const mfib_prefix_t *pfx)
{
mfib_itf_mac_add_del(itf, pfx, 0);
}
void
mfib_itf_delete (mfib_itf_t *mfi)
{
mfib_itf_hash_flush(mfi);
mfib_signal_remove_itf(mfi);
pool_put(mfib_itf_pool, mfi);
}
u8 *
format_mfib_itf (u8 * s, va_list * args)
{
mfib_itf_t *mfib_itf;
vnet_main_t *vnm;
index_t mfi;
mfi = va_arg (*args, index_t);
vnm = vnet_get_main();
mfib_itf = mfib_itf_get(mfi);
if (~0 != mfib_itf->mfi_sw_if_index)
{
return (format(s, " %U: %U",
format_vnet_sw_interface_name,
vnm,
vnet_get_sw_interface(vnm,
mfib_itf->mfi_sw_if_index),
format_mfib_itf_flags, mfib_itf->mfi_flags));
}
else
{
return (format(s, " local: %U",
format_mfib_itf_flags, mfib_itf->mfi_flags));
}
return (s);
}
static clib_error_t *
show_mfib_itf_command (vlib_main_t * vm,
unformat_input_t * input,
vlib_cli_command_t * cmd)
{
index_t mfii;
if (unformat (input, "%d", &mfii))
{
/*
* show one in detail
*/
if (!pool_is_free_index(mfib_itf_pool, mfii))
{
vlib_cli_output (vm, "%d@%U",
mfii,
format_mfib_itf, mfii);
}
else
{
vlib_cli_output (vm, "itf %d invalid", mfii);
}
}
else
{
/*
* show all
*/
vlib_cli_output (vm, "mFIB interfaces::");
pool_foreach_index (mfii, mfib_itf_pool)
{
vlib_cli_output (vm, "%d@%U",
mfii,
format_mfib_itf, mfii);
}
}
return (NULL);
}
/*?
* This command displays an MFIB interface, or all interfaces, indexed by their
* unique numerical identifier.
?*/
VLIB_CLI_COMMAND (show_mfib_itf, static) = {
.path = "show mfib interface",
.function = show_mfib_itf_command,
.short_help = "show mfib interface",
};