566 lines
15 KiB
C
566 lines
15 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Zebra SRv6 VTY functions
|
|
* Copyright (C) 2020 Hiroki Shirokura, LINE Corporation
|
|
*/
|
|
|
|
#include <zebra.h>
|
|
|
|
#include "memory.h"
|
|
#include "if.h"
|
|
#include "prefix.h"
|
|
#include "command.h"
|
|
#include "table.h"
|
|
#include "rib.h"
|
|
#include "nexthop.h"
|
|
#include "vrf.h"
|
|
#include "srv6.h"
|
|
#include "lib/json.h"
|
|
|
|
#include "zebra/zserv.h"
|
|
#include "zebra/zebra_router.h"
|
|
#include "zebra/zebra_vrf.h"
|
|
#include "zebra/zebra_srv6.h"
|
|
#include "zebra/zebra_srv6_vty.h"
|
|
#include "zebra/zebra_rnh.h"
|
|
#include "zebra/redistribute.h"
|
|
#include "zebra/zebra_routemap.h"
|
|
#include "zebra/zebra_dplane.h"
|
|
|
|
#include "zebra/zebra_srv6_vty_clippy.c"
|
|
|
|
static int zebra_sr_config(struct vty *vty);
|
|
|
|
static struct cmd_node sr_node = {
|
|
.name = "sr",
|
|
.node = SEGMENT_ROUTING_NODE,
|
|
.parent_node = CONFIG_NODE,
|
|
.prompt = "%s(config-sr)# ",
|
|
.config_write = zebra_sr_config,
|
|
};
|
|
|
|
static struct cmd_node srv6_node = {
|
|
.name = "srv6",
|
|
.node = SRV6_NODE,
|
|
.parent_node = SEGMENT_ROUTING_NODE,
|
|
.prompt = "%s(config-srv6)# ",
|
|
|
|
};
|
|
|
|
static struct cmd_node srv6_locs_node = {
|
|
.name = "srv6-locators",
|
|
.node = SRV6_LOCS_NODE,
|
|
.parent_node = SRV6_NODE,
|
|
.prompt = "%s(config-srv6-locators)# ",
|
|
};
|
|
|
|
static struct cmd_node srv6_loc_node = {
|
|
.name = "srv6-locator",
|
|
.node = SRV6_LOC_NODE,
|
|
.parent_node = SRV6_LOCS_NODE,
|
|
.prompt = "%s(config-srv6-locator)# "
|
|
};
|
|
|
|
static struct cmd_node srv6_encap_node = {
|
|
.name = "srv6-encap",
|
|
.node = SRV6_ENCAP_NODE,
|
|
.parent_node = SRV6_NODE,
|
|
.prompt = "%s(config-srv6-encap)# "
|
|
};
|
|
|
|
DEFPY (show_srv6_manager,
|
|
show_srv6_manager_cmd,
|
|
"show segment-routing srv6 manager [json]",
|
|
SHOW_STR
|
|
"Segment Routing\n"
|
|
"Segment Routing SRv6\n"
|
|
"Verify SRv6 Manager\n"
|
|
JSON_STR)
|
|
{
|
|
const bool uj = use_json(argc, argv);
|
|
struct zebra_srv6 *srv6 = zebra_srv6_get_default();
|
|
json_object *json = NULL;
|
|
json_object *json_parameters = NULL;
|
|
json_object *json_encapsulation = NULL;
|
|
json_object *json_source_address = NULL;
|
|
|
|
if (uj) {
|
|
json = json_object_new_object();
|
|
json_parameters = json_object_new_object();
|
|
json_object_object_add(json, "parameters", json_parameters);
|
|
json_encapsulation = json_object_new_object();
|
|
json_object_object_add(json_parameters, "encapsulation",
|
|
json_encapsulation);
|
|
json_source_address = json_object_new_object();
|
|
json_object_object_add(json_encapsulation, "sourceAddress",
|
|
json_source_address);
|
|
json_object_string_addf(json_source_address, "configured",
|
|
"%pI6", &srv6->encap_src_addr);
|
|
vty_json(vty, json);
|
|
} else {
|
|
vty_out(vty, "Parameters:\n");
|
|
vty_out(vty, " Encapsulation:\n");
|
|
vty_out(vty, " Source Address:\n");
|
|
vty_out(vty, " Configured: %pI6\n", &srv6->encap_src_addr);
|
|
}
|
|
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN (show_srv6_locator,
|
|
show_srv6_locator_cmd,
|
|
"show segment-routing srv6 locator [json]",
|
|
SHOW_STR
|
|
"Segment Routing\n"
|
|
"Segment Routing SRv6\n"
|
|
"Locator Information\n"
|
|
JSON_STR)
|
|
{
|
|
const bool uj = use_json(argc, argv);
|
|
struct zebra_srv6 *srv6 = zebra_srv6_get_default();
|
|
struct srv6_locator *locator;
|
|
struct listnode *node;
|
|
char str[256];
|
|
int id;
|
|
json_object *json = NULL;
|
|
json_object *json_locators = NULL;
|
|
json_object *json_locator = NULL;
|
|
|
|
if (uj) {
|
|
json = json_object_new_object();
|
|
json_locators = json_object_new_array();
|
|
json_object_object_add(json, "locators", json_locators);
|
|
|
|
for (ALL_LIST_ELEMENTS_RO(srv6->locators, node, locator)) {
|
|
json_locator = srv6_locator_json(locator);
|
|
if (!json_locator)
|
|
continue;
|
|
json_object_array_add(json_locators, json_locator);
|
|
|
|
}
|
|
|
|
vty_json(vty, json);
|
|
} else {
|
|
vty_out(vty, "Locator:\n");
|
|
vty_out(vty, "Name ID Prefix Status\n");
|
|
vty_out(vty, "-------------------- ------- ------------------------ -------\n");
|
|
|
|
id = 1;
|
|
for (ALL_LIST_ELEMENTS_RO(srv6->locators, node, locator)) {
|
|
prefix2str(&locator->prefix, str, sizeof(str));
|
|
vty_out(vty, "%-20s %7d %-24s %s\n",
|
|
locator->name, id, str,
|
|
locator->status_up ? "Up" : "Down");
|
|
++id;
|
|
}
|
|
vty_out(vty, "\n");
|
|
}
|
|
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN (show_srv6_locator_detail,
|
|
show_srv6_locator_detail_cmd,
|
|
"show segment-routing srv6 locator NAME detail [json]",
|
|
SHOW_STR
|
|
"Segment Routing\n"
|
|
"Segment Routing SRv6\n"
|
|
"Locator Information\n"
|
|
"Locator Name\n"
|
|
"Detailed information\n"
|
|
JSON_STR)
|
|
{
|
|
const bool uj = use_json(argc, argv);
|
|
struct zebra_srv6 *srv6 = zebra_srv6_get_default();
|
|
struct srv6_locator *locator;
|
|
struct listnode *node;
|
|
char str[256];
|
|
const char *locator_name = argv[4]->arg;
|
|
json_object *json_locator = NULL;
|
|
|
|
if (uj) {
|
|
locator = zebra_srv6_locator_lookup(locator_name);
|
|
if (!locator)
|
|
return CMD_WARNING;
|
|
|
|
json_locator = srv6_locator_detailed_json(locator);
|
|
vty_json(vty, json_locator);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
for (ALL_LIST_ELEMENTS_RO(srv6->locators, node, locator)) {
|
|
struct listnode *node;
|
|
struct srv6_locator_chunk *chunk;
|
|
|
|
if (strcmp(locator->name, locator_name) != 0)
|
|
continue;
|
|
|
|
prefix2str(&locator->prefix, str, sizeof(str));
|
|
vty_out(vty, "Name: %s\n", locator->name);
|
|
vty_out(vty, "Prefix: %s\n", str);
|
|
vty_out(vty, "Block-Bit-Len: %u\n", locator->block_bits_length);
|
|
vty_out(vty, "Node-Bit-Len: %u\n", locator->node_bits_length);
|
|
vty_out(vty, "Function-Bit-Len: %u\n",
|
|
locator->function_bits_length);
|
|
vty_out(vty, "Argument-Bit-Len: %u\n",
|
|
locator->argument_bits_length);
|
|
|
|
if (CHECK_FLAG(locator->flags, SRV6_LOCATOR_USID))
|
|
vty_out(vty, "Behavior: uSID\n");
|
|
|
|
vty_out(vty, "Chunks:\n");
|
|
for (ALL_LIST_ELEMENTS_RO((struct list *)locator->chunks, node,
|
|
chunk)) {
|
|
prefix2str(&chunk->prefix, str, sizeof(str));
|
|
vty_out(vty, "- prefix: %s, owner: %s\n", str,
|
|
zebra_route_string(chunk->proto));
|
|
}
|
|
}
|
|
|
|
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN_NOSH (segment_routing,
|
|
segment_routing_cmd,
|
|
"segment-routing",
|
|
"Segment Routing\n")
|
|
{
|
|
vty->node = SEGMENT_ROUTING_NODE;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN_NOSH (srv6,
|
|
srv6_cmd,
|
|
"srv6",
|
|
"Segment Routing SRv6\n")
|
|
{
|
|
vty->node = SRV6_NODE;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN (no_srv6,
|
|
no_srv6_cmd,
|
|
"no srv6",
|
|
NO_STR
|
|
"Segment Routing SRv6\n")
|
|
{
|
|
struct zebra_srv6 *srv6 = zebra_srv6_get_default();
|
|
struct srv6_locator *locator;
|
|
struct listnode *node, *nnode;
|
|
|
|
for (ALL_LIST_ELEMENTS(srv6->locators, node, nnode, locator))
|
|
zebra_srv6_locator_delete(locator);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN_NOSH (srv6_locators,
|
|
srv6_locators_cmd,
|
|
"locators",
|
|
"Segment Routing SRv6 locators\n")
|
|
{
|
|
vty->node = SRV6_LOCS_NODE;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN_NOSH (srv6_locator,
|
|
srv6_locator_cmd,
|
|
"locator WORD",
|
|
"Segment Routing SRv6 locator\n"
|
|
"Specify locator-name\n")
|
|
{
|
|
struct srv6_locator *locator = NULL;
|
|
|
|
locator = zebra_srv6_locator_lookup(argv[1]->arg);
|
|
if (locator) {
|
|
VTY_PUSH_CONTEXT(SRV6_LOC_NODE, locator);
|
|
locator->status_up = true;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
locator = srv6_locator_alloc(argv[1]->arg);
|
|
if (!locator) {
|
|
vty_out(vty, "%% Alloc failed\n");
|
|
return CMD_WARNING_CONFIG_FAILED;
|
|
}
|
|
locator->status_up = true;
|
|
|
|
VTY_PUSH_CONTEXT(SRV6_LOC_NODE, locator);
|
|
vty->node = SRV6_LOC_NODE;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN (no_srv6_locator,
|
|
no_srv6_locator_cmd,
|
|
"no locator WORD",
|
|
NO_STR
|
|
"Segment Routing SRv6 locator\n"
|
|
"Specify locator-name\n")
|
|
{
|
|
struct srv6_locator *locator = zebra_srv6_locator_lookup(argv[2]->arg);
|
|
if (!locator) {
|
|
vty_out(vty, "%% Can't find SRv6 locator\n");
|
|
return CMD_WARNING_CONFIG_FAILED;
|
|
}
|
|
|
|
zebra_srv6_locator_delete(locator);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFPY (locator_prefix,
|
|
locator_prefix_cmd,
|
|
"prefix X:X::X:X/M$prefix [block-len (16-64)$block_bit_len] \
|
|
[node-len (16-64)$node_bit_len] [func-bits (0-64)$func_bit_len]",
|
|
"Configure SRv6 locator prefix\n"
|
|
"Specify SRv6 locator prefix\n"
|
|
"Configure SRv6 locator block length in bits\n"
|
|
"Specify SRv6 locator block length in bits\n"
|
|
"Configure SRv6 locator node length in bits\n"
|
|
"Specify SRv6 locator node length in bits\n"
|
|
"Configure SRv6 locator function length in bits\n"
|
|
"Specify SRv6 locator function length in bits\n")
|
|
{
|
|
VTY_DECLVAR_CONTEXT(srv6_locator, locator);
|
|
struct srv6_locator_chunk *chunk = NULL;
|
|
struct listnode *node = NULL;
|
|
|
|
locator->prefix = *prefix;
|
|
func_bit_len = func_bit_len ?: ZEBRA_SRV6_FUNCTION_LENGTH;
|
|
|
|
/* Resolve optional arguments */
|
|
if (block_bit_len == 0 && node_bit_len == 0) {
|
|
block_bit_len =
|
|
prefix->prefixlen - ZEBRA_SRV6_LOCATOR_NODE_LENGTH;
|
|
node_bit_len = ZEBRA_SRV6_LOCATOR_NODE_LENGTH;
|
|
} else if (block_bit_len == 0) {
|
|
block_bit_len = prefix->prefixlen - node_bit_len;
|
|
} else if (node_bit_len == 0) {
|
|
node_bit_len = prefix->prefixlen - block_bit_len;
|
|
} else {
|
|
if (block_bit_len + node_bit_len != prefix->prefixlen) {
|
|
vty_out(vty,
|
|
"%% block-len + node-len must be equal to the selected prefix length %d\n",
|
|
prefix->prefixlen);
|
|
return CMD_WARNING_CONFIG_FAILED;
|
|
}
|
|
}
|
|
|
|
if (prefix->prefixlen + func_bit_len + 0 > 128) {
|
|
vty_out(vty,
|
|
"%% prefix-len + function-len + arg-len (%ld) cannot be greater than 128\n",
|
|
prefix->prefixlen + func_bit_len + 0);
|
|
return CMD_WARNING_CONFIG_FAILED;
|
|
}
|
|
|
|
/*
|
|
* Currently, the SID transposition algorithm implemented in bgpd
|
|
* handles incorrectly the SRv6 locators with function length greater
|
|
* than 20 bits. To prevent issues, we currently limit the function
|
|
* length to 20 bits.
|
|
* This limit will be removed when the bgpd SID transposition is fixed.
|
|
*/
|
|
if (func_bit_len > 20) {
|
|
vty_out(vty,
|
|
"%% currently func_bit_len > 20 is not supported\n");
|
|
return CMD_WARNING_CONFIG_FAILED;
|
|
}
|
|
|
|
locator->block_bits_length = block_bit_len;
|
|
locator->node_bits_length = node_bit_len;
|
|
locator->function_bits_length = func_bit_len;
|
|
locator->argument_bits_length = 0;
|
|
|
|
if (list_isempty(locator->chunks)) {
|
|
chunk = srv6_locator_chunk_alloc();
|
|
chunk->prefix = *prefix;
|
|
chunk->proto = 0;
|
|
listnode_add(locator->chunks, chunk);
|
|
} else {
|
|
for (ALL_LIST_ELEMENTS_RO(locator->chunks, node, chunk)) {
|
|
uint8_t zero[16] = {0};
|
|
|
|
if (memcmp(&chunk->prefix.prefix, zero, 16) == 0) {
|
|
struct zserv *client;
|
|
struct listnode *client_node;
|
|
|
|
chunk->prefix = *prefix;
|
|
for (ALL_LIST_ELEMENTS_RO(zrouter.client_list,
|
|
client_node,
|
|
client)) {
|
|
struct srv6_locator *tmp;
|
|
|
|
if (client->proto != chunk->proto)
|
|
continue;
|
|
|
|
srv6_manager_get_locator_chunk_call(
|
|
&tmp, client,
|
|
locator->name,
|
|
VRF_DEFAULT);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
zebra_srv6_locator_add(locator);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFPY (locator_behavior,
|
|
locator_behavior_cmd,
|
|
"[no] behavior usid",
|
|
NO_STR
|
|
"Configure SRv6 behavior\n"
|
|
"Specify SRv6 behavior uSID\n")
|
|
{
|
|
VTY_DECLVAR_CONTEXT(srv6_locator, locator);
|
|
|
|
if (no && !CHECK_FLAG(locator->flags, SRV6_LOCATOR_USID))
|
|
/* SRv6 locator uSID flag already unset, nothing to do */
|
|
return CMD_SUCCESS;
|
|
|
|
if (!no && CHECK_FLAG(locator->flags, SRV6_LOCATOR_USID))
|
|
/* SRv6 locator uSID flag already set, nothing to do */
|
|
return CMD_SUCCESS;
|
|
|
|
/* Remove old locator from zclients */
|
|
zebra_notify_srv6_locator_delete(locator);
|
|
|
|
/* Set/Unset the SRV6_LOCATOR_USID */
|
|
if (no)
|
|
UNSET_FLAG(locator->flags, SRV6_LOCATOR_USID);
|
|
else
|
|
SET_FLAG(locator->flags, SRV6_LOCATOR_USID);
|
|
|
|
/* Notify the new locator to zclients */
|
|
zebra_notify_srv6_locator_add(locator);
|
|
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFUN_NOSH (srv6_encap,
|
|
srv6_encap_cmd,
|
|
"encapsulation",
|
|
"Segment Routing SRv6 encapsulation\n")
|
|
{
|
|
vty->node = SRV6_ENCAP_NODE;
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFPY (srv6_src_addr,
|
|
srv6_src_addr_cmd,
|
|
"source-address X:X::X:X$encap_src_addr",
|
|
"Segment Routing SRv6 source address\n"
|
|
"Specify source address for SRv6 encapsulation\n")
|
|
{
|
|
zebra_srv6_encap_src_addr_set(&encap_src_addr);
|
|
dplane_srv6_encap_srcaddr_set(&encap_src_addr, NS_DEFAULT);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
DEFPY (no_srv6_src_addr,
|
|
no_srv6_src_addr_cmd,
|
|
"no source-address [X:X::X:X$encap_src_addr]",
|
|
NO_STR
|
|
"Segment Routing SRv6 source address\n"
|
|
"Specify source address for SRv6 encapsulation\n")
|
|
{
|
|
zebra_srv6_encap_src_addr_unset();
|
|
dplane_srv6_encap_srcaddr_set(&in6addr_any, NS_DEFAULT);
|
|
return CMD_SUCCESS;
|
|
}
|
|
|
|
static int zebra_sr_config(struct vty *vty)
|
|
{
|
|
struct zebra_srv6 *srv6 = zebra_srv6_get_default();
|
|
struct listnode *node;
|
|
struct srv6_locator *locator;
|
|
char str[256];
|
|
bool display_source_srv6 = false;
|
|
|
|
if (srv6 && !IPV6_ADDR_SAME(&srv6->encap_src_addr, &in6addr_any))
|
|
display_source_srv6 = true;
|
|
|
|
vty_out(vty, "!\n");
|
|
if (display_source_srv6 || zebra_srv6_is_enable()) {
|
|
vty_out(vty, "segment-routing\n");
|
|
vty_out(vty, " srv6\n");
|
|
}
|
|
if (display_source_srv6) {
|
|
if (!IPV6_ADDR_SAME(&srv6->encap_src_addr, &in6addr_any)) {
|
|
vty_out(vty, " encapsulation\n");
|
|
vty_out(vty, " source-address %pI6\n",
|
|
&srv6->encap_src_addr);
|
|
}
|
|
}
|
|
if (zebra_srv6_is_enable()) {
|
|
vty_out(vty, " locators\n");
|
|
for (ALL_LIST_ELEMENTS_RO(srv6->locators, node, locator)) {
|
|
inet_ntop(AF_INET6, &locator->prefix.prefix,
|
|
str, sizeof(str));
|
|
vty_out(vty, " locator %s\n", locator->name);
|
|
vty_out(vty, " prefix %s/%u", str,
|
|
locator->prefix.prefixlen);
|
|
if (locator->block_bits_length)
|
|
vty_out(vty, " block-len %u",
|
|
locator->block_bits_length);
|
|
if (locator->node_bits_length)
|
|
vty_out(vty, " node-len %u",
|
|
locator->node_bits_length);
|
|
if (locator->function_bits_length)
|
|
vty_out(vty, " func-bits %u",
|
|
locator->function_bits_length);
|
|
if (locator->argument_bits_length)
|
|
vty_out(vty, " arg-len %u",
|
|
locator->argument_bits_length);
|
|
vty_out(vty, "\n");
|
|
if (CHECK_FLAG(locator->flags, SRV6_LOCATOR_USID))
|
|
vty_out(vty, " behavior usid\n");
|
|
vty_out(vty, " exit\n");
|
|
vty_out(vty, " !\n");
|
|
}
|
|
vty_out(vty, " exit\n");
|
|
vty_out(vty, " !\n");
|
|
vty_out(vty, " exit\n");
|
|
vty_out(vty, " !\n");
|
|
}
|
|
if (display_source_srv6 || zebra_srv6_is_enable()) {
|
|
vty_out(vty, "exit\n");
|
|
vty_out(vty, "!\n");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void zebra_srv6_vty_init(void)
|
|
{
|
|
/* Install nodes and its default commands */
|
|
install_node(&sr_node);
|
|
install_node(&srv6_node);
|
|
install_node(&srv6_locs_node);
|
|
install_node(&srv6_loc_node);
|
|
install_node(&srv6_encap_node);
|
|
install_default(SEGMENT_ROUTING_NODE);
|
|
install_default(SRV6_NODE);
|
|
install_default(SRV6_LOCS_NODE);
|
|
install_default(SRV6_LOC_NODE);
|
|
install_default(SRV6_ENCAP_NODE);
|
|
|
|
/* Command for change node */
|
|
install_element(CONFIG_NODE, &segment_routing_cmd);
|
|
install_element(SEGMENT_ROUTING_NODE, &srv6_cmd);
|
|
install_element(SEGMENT_ROUTING_NODE, &no_srv6_cmd);
|
|
install_element(SRV6_NODE, &srv6_locators_cmd);
|
|
install_element(SRV6_NODE, &srv6_encap_cmd);
|
|
install_element(SRV6_LOCS_NODE, &srv6_locator_cmd);
|
|
install_element(SRV6_LOCS_NODE, &no_srv6_locator_cmd);
|
|
|
|
/* Command for configuration */
|
|
install_element(SRV6_LOC_NODE, &locator_prefix_cmd);
|
|
install_element(SRV6_LOC_NODE, &locator_behavior_cmd);
|
|
install_element(SRV6_ENCAP_NODE, &srv6_src_addr_cmd);
|
|
install_element(SRV6_ENCAP_NODE, &no_srv6_src_addr_cmd);
|
|
|
|
/* Command for operation */
|
|
install_element(VIEW_NODE, &show_srv6_locator_cmd);
|
|
install_element(VIEW_NODE, &show_srv6_locator_detail_cmd);
|
|
install_element(VIEW_NODE, &show_srv6_manager_cmd);
|
|
}
|