blob: 6247f9a9388fe0ec1edcd5c32da77ef0ecb9caa7 [file] [log] [blame]
/*
*------------------------------------------------------------------
* Copyright (c) 2017 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 <stdint.h>
#include <sys/ioctl.h>
#include <inttypes.h>
#include <vlib/vlib.h>
#include <vlib/unix/unix.h>
#include <vnet/ip/ip.h>
#include <vnet/fib/fib_entry.h>
#include <vnet/fib/fib_table.h>
#include <vnet/mfib/mfib_table.h>
#include <igmp/igmp.h>
static clib_error_t *
igmp_clear_interface_command_fn (vlib_main_t * vm, unformat_input_t * input,
vlib_cli_command_t * cmd)
{
unformat_input_t _line_input, *line_input = &_line_input;
clib_error_t *error = NULL;
vnet_main_t *vnm = vnet_get_main ();
u32 sw_if_index;
igmp_config_t *config;
if (!unformat_user (input, unformat_line_input, line_input))
{
error =
clib_error_return (0, "'help clear igmp' or 'clear igmp ?' for help");
return error;
}
while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
{
if (unformat
(line_input, "int %U", unformat_vnet_sw_interface, vnm,
&sw_if_index));
else
{
error =
clib_error_return (0, "unknown input '%U'", format_unformat_error,
line_input);
goto done;
}
}
config = igmp_config_lookup (sw_if_index);
if (config)
igmp_clear_config (config);
done:
unformat_free (line_input);
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_clear_interface_command, static) = {
.path = "clear igmp",
.short_help = "clear igmp int <interface>",
.function = igmp_clear_interface_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_listen_command_fn (vlib_main_t * vm, unformat_input_t * input,
vlib_cli_command_t * cmd)
{
unformat_input_t _line_input, *line_input = &_line_input;
clib_error_t *error = NULL;
u8 enable = 1;
ip46_address_t saddr, *saddrs = NULL, gaddr;
vnet_main_t *vnm = vnet_get_main ();
u32 sw_if_index;
int rv;
if (!unformat_user (input, unformat_line_input, line_input))
{
error =
clib_error_return (0,
"'help igmp listen' or 'igmp listen ?' for help");
return error;
}
while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
{
if (unformat (line_input, "enable"))
enable = 1;
else if (unformat (line_input, "disable"))
enable = 0;
else
if (unformat
(line_input, "int %U", unformat_vnet_sw_interface, vnm,
&sw_if_index));
else
if (unformat (line_input, "saddr %U", unformat_ip46_address, &saddr))
vec_add1 (saddrs, saddr);
else
if (unformat (line_input, "gaddr %U", unformat_ip46_address, &gaddr));
else
{
error =
clib_error_return (0, "unknown input '%U'", format_unformat_error,
line_input);
goto done;
}
}
if ((vnet_sw_interface_get_flags (vnm, sw_if_index)
&& VNET_SW_INTERFACE_FLAG_ADMIN_UP) == 0)
{
error = clib_error_return (0, "Interface is down");
goto done;
}
rv = igmp_listen (vm, enable, sw_if_index, saddrs, &gaddr);
if (rv == -1)
{
if (enable)
error =
clib_error_return (0, "This igmp configuration already exists");
else
error =
clib_error_return (0, "This igmp configuration does not exist");
}
else if (rv == -2)
error =
clib_error_return (0,
"Failed to add configuration, interface is in router mode");
done:
unformat_free (line_input);
vec_free (saddrs);
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_listen_command, static) = {
.path = "igmp listen",
.short_help = "igmp listen [<enable|disable>] "
"int <interface> saddr <ip4-address> gaddr <ip4-address>",
.function = igmp_listen_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_enable_cli (vlib_main_t * vm,
unformat_input_t * input, vlib_cli_command_t * cmd)
{
unformat_input_t _line_input, *line_input = &_line_input;
igmp_mode_t mode = IGMP_MODE_ROUTER;
vnet_main_t *vnm = vnet_get_main ();
clib_error_t *error = NULL;
u32 sw_if_index = ~0;
u8 enable = 1;
int rv;
if (!unformat_user (input, unformat_line_input, line_input))
return error;
while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
{
if (unformat (line_input, "enable"))
enable = 1;
else if (unformat (line_input, "disable"))
enable = 0;
if (unformat (line_input, "host"))
mode = IGMP_MODE_HOST;
else if (unformat (line_input, "router"))
mode = IGMP_MODE_ROUTER;
else if (unformat (line_input, "%U",
unformat_vnet_sw_interface, vnm, &sw_if_index));
else
{
error =
clib_error_return (0, "unknown input '%U'", format_unformat_error,
line_input);
goto done;
}
}
if (~0 == sw_if_index)
{
error = clib_error_return (0, "interface must be specified");
goto done;
}
rv = igmp_enable_disable (sw_if_index, enable, mode);
if (0 != rv)
error = clib_error_return (0, "result: %d", rv);
done:
unformat_free (line_input);
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_enable_command, static) = {
.path = "igmp",
.short_help = "igmp <enable|disable> <host|router> <interface>",
.function = igmp_enable_cli,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_proxy_device_add_del_command_fn (vlib_main_t * vm,
unformat_input_t * input,
vlib_cli_command_t * cmd)
{
unformat_input_t _line_input, *line_input = &_line_input;
vnet_main_t *vnm = vnet_get_main ();
clib_error_t *error = NULL;
u32 sw_if_index = ~0;
u32 vrf_id = ~0;
u8 add = 1;
int rv;
if (!unformat_user (input, unformat_line_input, line_input))
return error;
while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
{
if (unformat (line_input, "add"))
add = 1;
else if (unformat (line_input, "del"))
add = 0;
else if (unformat (line_input, "vrf-id %u", &vrf_id))
;
else if (unformat (line_input, "%U",
unformat_vnet_sw_interface, vnm, &sw_if_index));
else
{
error =
clib_error_return (0, "unknown input '%U'", format_unformat_error,
line_input);
goto done;
}
}
if (~0 == sw_if_index)
{
error = clib_error_return (0, "interface must be specified");
goto done;
}
if (~0 == vrf_id)
{
error = clib_error_return (0, "VRF must be specified");
goto done;
}
rv = igmp_proxy_device_add_del (vrf_id, sw_if_index, add);
if (0 != rv)
error = clib_error_return (0, "result: %d", rv);
done:
unformat_free (line_input);
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_proxy_device_add_del_command, static) = {
.path = "igmp proxy-dev",
.short_help = "igmp proxy-dev <add|del> vrf-id <table-id> <interface>",
.function = igmp_proxy_device_add_del_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_proxy_device_add_del_interface_command_fn (vlib_main_t * vm,
unformat_input_t * input,
vlib_cli_command_t * cmd)
{
unformat_input_t _line_input, *line_input = &_line_input;
vnet_main_t *vnm = vnet_get_main ();
clib_error_t *error = NULL;
u32 sw_if_index = ~0;
u32 vrf_id = ~0;
u8 add = 1;
int rv;
if (!unformat_user (input, unformat_line_input, line_input))
return error;
while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
{
if (unformat (line_input, "add"))
add = 1;
else if (unformat (line_input, "del"))
add = 0;
else if (unformat (line_input, "vrf-id %u", &vrf_id))
;
else if (unformat (line_input, "%U",
unformat_vnet_sw_interface, vnm, &sw_if_index));
else
{
error =
clib_error_return (0, "unknown input '%U'", format_unformat_error,
line_input);
goto done;
}
}
if (~0 == sw_if_index)
{
error = clib_error_return (0, "interface must be specified");
goto done;
}
if (~0 == vrf_id)
{
error = clib_error_return (0, "VRF must be specified");
goto done;
}
rv = igmp_proxy_device_add_del_interface (vrf_id, sw_if_index, add);
if (0 != rv)
error = clib_error_return (0, "result: %d", rv);
done:
unformat_free (line_input);
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_proxy_device_add_del_interface_command, static) = {
.path = "igmp proxy-dev itf",
.short_help = "igmp proxy-dev itf <add|del> vrf-id <table-id> <interface>",
.function = igmp_proxy_device_add_del_interface_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_show_command_fn (vlib_main_t * vm, unformat_input_t * input,
vlib_cli_command_t * cmd)
{
clib_error_t *error = NULL;
igmp_main_t *im = &igmp_main;
vnet_main_t *vnm = vnet_get_main ();
igmp_config_t *config;
igmp_group_t *group;
igmp_src_t *src;
/* *INDENT-OFF* */
pool_foreach (config, im->configs,
({
vlib_cli_output (vm, "interface: %U mode: %U %U",
format_vnet_sw_if_index_name, vnm, config->sw_if_index,
format_igmp_mode, config->mode, format_igmp_proxy_device_id, config->proxy_device_id);
FOR_EACH_GROUP (group, config,
({
vlib_cli_output (vm, "\t%U", format_igmp_key, group->key);
FOR_EACH_SRC (src, group, IGMP_FILTER_MODE_INCLUDE,
({
vlib_cli_output (vm, "\t\t%U", format_igmp_key, src->key);
}));
}));
}));
/* *INDENT-ON* */
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_show_command, static) = {
.path = "show igmp config",
.short_help = "show igmp config",
.function = igmp_show_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
igmp_show_timers_command_fn (vlib_main_t * vm,
unformat_input_t * input,
vlib_cli_command_t * cmd)
{
#define _(n,f) vlib_cli_output (vm, "%s: %d", #f, igmp_timer_type_get(n));
foreach_igmp_timer_type
#undef _
return (NULL);
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (igmp_show_timers_command, static) = {
.path = "show igmp timers",
.short_help = "show igmp timers",
.function = igmp_show_timers_command_fn,
};
/* *INDENT-ON* */
static clib_error_t *
test_igmp_command_fn (vlib_main_t * vm,
unformat_input_t * input, vlib_cli_command_t * cmd)
{
clib_error_t *error = NULL;
u32 value;
while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
{
if (unformat (input, "query %d", &value))
igmp_timer_type_set (IGMP_TIMER_QUERY, value);
else if (unformat (input, "src %d", &value))
igmp_timer_type_set (IGMP_TIMER_SRC, value);
else if (unformat (input, "leave %d", &value))
igmp_timer_type_set (IGMP_TIMER_LEAVE, value);
else
error = clib_error_return (0, "query or src timers only");
}
return error;
}
/* *INDENT-OFF* */
VLIB_CLI_COMMAND (test_igmp_command, static) = {
.path = "test igmp timers",
.short_help = "Change the default values for IGMP timers - only sensible during unit tests",
.function = test_igmp_command_fn,
};
/* *INDENT-ON* */
clib_error_t *
igmp_cli_init (vlib_main_t * vm)
{
return 0;
}
VLIB_INIT_FUNCTION (igmp_cli_init);
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/