mirror of
				https://git.proxmox.com/git/mirror_frr
				synced 2025-10-25 11:00:27 +00:00 
			
		
		
		
	 4d7b695d3a
			
		
	
	
		4d7b695d3a
		
	
	
	
	
		
			
			This new daemon manages Segment-Routing Traffic-Engineering (SR-TE) Policies and installs them into zebra. It provides the usual yang support and vtysh commands to define or change SR-TE Policies. In a nutshell SR-TE Policies provide the possibility to steer traffic through a (possibly dynamic) list of Segment Routing segments to the endpoint of the policy. This list of segments is part of a Candidate Path which again belongs to the SR-TE Policy. SR-TE Policies are uniquely identified by their color and endpoint. The color can be used to e.g. match BGP communities on incoming traffic. There can be multiple Candidate Paths for a single policy, the active Candidate Path is chosen according to certain conditions of which the most important is its preference. Candidate Paths can be explicit (fixed list of segments) or dynamic (list of segment comes from e.g. PCEP, see below). Configuration example: segment-routing traffic-eng segment-list SL index 10 mpls label 1111 index 20 mpls label 2222 ! policy color 4 endpoint 10.10.10.4 name POL4 binding-sid 104 candidate-path preference 100 name exp explicit segment-list SL candidate-path preference 200 name dyn dynamic ! ! ! There is an important connection between dynamic Candidate Paths and the overall topic of Path Computation. Later on for pathd a dynamic module will be introduced that is capable of communicating via the PCEP protocol with a PCE (Path Computation Element) which again is capable of calculating paths according to its local TED (Traffic Engineering Database). This dynamic module will be able to inject the mentioned dynamic Candidate Paths into pathd based on calculated paths from a PCE. https://tools.ietf.org/html/draft-ietf-spring-segment-routing-policy-06 Co-authored-by: Sebastien Merle <sebastien@netdef.org> Co-authored-by: Renato Westphal <renato@opensourcerouting.org> Co-authored-by: GalaxyGorilla <sascha@netdef.org> Co-authored-by: Emanuele Di Pascale <emanuele@voltanet.io> Signed-off-by: Sebastien Merle <sebastien@netdef.org>
		
			
				
	
	
		
			190 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			190 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2020  NetDEF, Inc.
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of the GNU General Public License as published by the Free
 | |
|  * Software Foundation; either version 2 of the License, or (at your option)
 | |
|  * any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program; see the file COPYING; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 | |
|  */
 | |
| 
 | |
| #include <zebra.h>
 | |
| 
 | |
| #include "log.h"
 | |
| #include "prefix.h"
 | |
| #include "table.h"
 | |
| #include "command.h"
 | |
| #include "northbound.h"
 | |
| #include "libfrr.h"
 | |
| 
 | |
| #include "pathd/pathd.h"
 | |
| #include "pathd/path_nb.h"
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/segment-list
 | |
|  */
 | |
| const void *pathd_srte_segment_list_get_next(struct nb_cb_get_next_args *args)
 | |
| {
 | |
| 	struct srte_segment_list *segment_list =
 | |
| 		(struct srte_segment_list *)args->list_entry;
 | |
| 
 | |
| 	if (args->list_entry == NULL)
 | |
| 		segment_list =
 | |
| 			RB_MIN(srte_segment_list_head, &srte_segment_lists);
 | |
| 	else
 | |
| 		segment_list = RB_NEXT(srte_segment_list_head, segment_list);
 | |
| 
 | |
| 	return segment_list;
 | |
| }
 | |
| 
 | |
| int pathd_srte_segment_list_get_keys(struct nb_cb_get_keys_args *args)
 | |
| {
 | |
| 	const struct srte_segment_list *segment_list =
 | |
| 		(struct srte_segment_list *)args->list_entry;
 | |
| 
 | |
| 	args->keys->num = 1;
 | |
| 	snprintf(args->keys->key[0], sizeof(args->keys->key[0]), "%s",
 | |
| 		 segment_list->name);
 | |
| 
 | |
| 	return NB_OK;
 | |
| }
 | |
| 
 | |
| const void *
 | |
| pathd_srte_segment_list_lookup_entry(struct nb_cb_lookup_entry_args *args)
 | |
| {
 | |
| 	return srte_segment_list_find(args->keys->key[0]);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/policy
 | |
|  */
 | |
| const void *pathd_srte_policy_get_next(struct nb_cb_get_next_args *args)
 | |
| {
 | |
| 	struct srte_policy *policy = (struct srte_policy *)args->list_entry;
 | |
| 
 | |
| 	if (args->list_entry == NULL)
 | |
| 		policy = RB_MIN(srte_policy_head, &srte_policies);
 | |
| 	else
 | |
| 		policy = RB_NEXT(srte_policy_head, policy);
 | |
| 
 | |
| 	return policy;
 | |
| }
 | |
| 
 | |
| int pathd_srte_policy_get_keys(struct nb_cb_get_keys_args *args)
 | |
| {
 | |
| 	const struct srte_policy *policy =
 | |
| 		(struct srte_policy *)args->list_entry;
 | |
| 
 | |
| 	args->keys->num = 2;
 | |
| 	snprintf(args->keys->key[0], sizeof(args->keys->key[0]), "%u",
 | |
| 		 policy->color);
 | |
| 	ipaddr2str(&policy->endpoint, args->keys->key[1],
 | |
| 		   sizeof(args->keys->key[1]));
 | |
| 
 | |
| 	return NB_OK;
 | |
| }
 | |
| 
 | |
| const void *pathd_srte_policy_lookup_entry(struct nb_cb_lookup_entry_args *args)
 | |
| {
 | |
| 	uint32_t color;
 | |
| 	struct ipaddr endpoint;
 | |
| 
 | |
| 	color = yang_str2uint32(args->keys->key[0]);
 | |
| 	yang_str2ip(args->keys->key[1], &endpoint);
 | |
| 
 | |
| 	return srte_policy_find(color, &endpoint);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/policy/is-operational
 | |
|  */
 | |
| struct yang_data *
 | |
| pathd_srte_policy_is_operational_get_elem(struct nb_cb_get_elem_args *args)
 | |
| {
 | |
| 	struct srte_policy *policy = (struct srte_policy *)args->list_entry;
 | |
| 	bool is_operational = false;
 | |
| 
 | |
| 	if (policy->status == SRTE_POLICY_STATUS_UP)
 | |
| 		is_operational = true;
 | |
| 
 | |
| 	return yang_data_new_bool(args->xpath, is_operational);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/policy/candidate-path
 | |
|  */
 | |
| const void *
 | |
| pathd_srte_policy_candidate_path_get_next(struct nb_cb_get_next_args *args)
 | |
| {
 | |
| 	struct srte_policy *policy =
 | |
| 		(struct srte_policy *)args->parent_list_entry;
 | |
| 	struct srte_candidate *candidate =
 | |
| 		(struct srte_candidate *)args->list_entry;
 | |
| 
 | |
| 	if (args->list_entry == NULL)
 | |
| 		candidate =
 | |
| 			RB_MIN(srte_candidate_head, &policy->candidate_paths);
 | |
| 	else
 | |
| 		candidate = RB_NEXT(srte_candidate_head, candidate);
 | |
| 
 | |
| 	return candidate;
 | |
| }
 | |
| 
 | |
| int pathd_srte_policy_candidate_path_get_keys(struct nb_cb_get_keys_args *args)
 | |
| {
 | |
| 	const struct srte_candidate *candidate =
 | |
| 		(struct srte_candidate *)args->list_entry;
 | |
| 
 | |
| 	args->keys->num = 1;
 | |
| 	snprintf(args->keys->key[0], sizeof(args->keys->key[0]), "%u",
 | |
| 		 candidate->preference);
 | |
| 
 | |
| 	return NB_OK;
 | |
| }
 | |
| 
 | |
| const void *pathd_srte_policy_candidate_path_lookup_entry(
 | |
| 	struct nb_cb_lookup_entry_args *args)
 | |
| {
 | |
| 	struct srte_policy *policy =
 | |
| 		(struct srte_policy *)args->parent_list_entry;
 | |
| 	uint32_t preference;
 | |
| 
 | |
| 	preference = yang_str2uint32(args->keys->key[0]);
 | |
| 
 | |
| 	return srte_candidate_find(policy, preference);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/policy/candidate_path/is-best-candidate-path
 | |
|  */
 | |
| struct yang_data *
 | |
| pathd_srte_policy_candidate_path_is_best_candidate_path_get_elem(
 | |
| 	struct nb_cb_get_elem_args *args)
 | |
| {
 | |
| 	struct srte_candidate *candidate =
 | |
| 		(struct srte_candidate *)args->list_entry;
 | |
| 
 | |
| 	return yang_data_new_bool(
 | |
| 		args->xpath, CHECK_FLAG(candidate->flags, F_CANDIDATE_BEST));
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * XPath: /frr-pathd:pathd/srte/policy/candidate-path/discriminator
 | |
|  */
 | |
| struct yang_data *pathd_srte_policy_candidate_path_discriminator_get_elem(
 | |
| 	struct nb_cb_get_elem_args *args)
 | |
| {
 | |
| 	struct srte_candidate *candidate =
 | |
| 		(struct srte_candidate *)args->list_entry;
 | |
| 
 | |
| 	return yang_data_new_uint32(args->xpath, candidate->discriminator);
 | |
| }
 |