mirror of
				https://git.proxmox.com/git/mirror_frr
				synced 2025-10-26 05:24:56 +00:00 
			
		
		
		
	 efba0985fc
			
		
	
	
		efba0985fc
		
	
	
	
	
		
			
			This new dynamic module makes pathd behave as a PCC for dynamic candidate path using the external library pcpelib https://github.com/volta-networks/pceplib . The candidate paths defined as dynamic will trigger computation requests to the configured PCE, and the PCE response will be used to update the policy. It supports multiple PCE. The one with smaller precedence will be elected as the master PCE, and only if the connection repeatedly fails, the PCC will switch to another PCE. Example of configuration: segment-routing traffic-eng pcep pce-config CONF source-address ip 10.10.10.10 sr-draft07 ! pce PCE1 config CONF address ip 1.1.1.1 ! pce PCE2 config CONF address ip 2.2.2.2 ! pcc peer PCE1 precedence 10 peer PCE2 precedence 20 ! ! ! ! Co-authored-by: Brady Johnson <brady@voltanet.io> Co-authored-by: Emanuele Di Pascale <emanuele@voltanet.io> Co-authored-by: GalaxyGorilla <sascha@netdef.org> Co-authored-by: Javier Garcia <javier.garcia@voltanet.io> Co-authored-by: Renato Westphal <renato@opensourcerouting.org> Co-authored-by: Sebastien Merle <sebastien@netdef.org> Signed-off-by: Sebastien Merle <sebastien@netdef.org>
		
			
				
	
	
		
			327 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			327 lines
		
	
	
		
			11 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
 | |
|  */
 | |
| 
 | |
| #ifndef _PATH_PCEP_H_
 | |
| #define _PATH_PCEP_H_
 | |
| 
 | |
| #include <stdbool.h>
 | |
| #include <debug.h>
 | |
| #include <netinet/tcp.h>
 | |
| #include <pcep_utils_logging.h>
 | |
| #include <pcep_pcc_api.h>
 | |
| #include "mpls.h"
 | |
| #include "pathd/pathd.h"
 | |
| #include "pathd/path_pcep_memory.h"
 | |
| 
 | |
| #define PCEP_DEFAULT_PORT 4189
 | |
| #define MAX_PCC 32
 | |
| #define MAX_PCE 32
 | |
| #define MAX_TAG_SIZE 50
 | |
| #define PCEP_DEBUG_MODE_BASIC 0x01
 | |
| #define PCEP_DEBUG_MODE_PATH 0x02
 | |
| #define PCEP_DEBUG_MODE_PCEP 0x04
 | |
| #define PCEP_DEBUG_MODE_PCEPLIB 0x08
 | |
| #define PCEP_DEBUG(fmt, ...)                                                   \
 | |
| 	do {                                                                   \
 | |
| 		if (DEBUG_FLAGS_CHECK(&pcep_g->dbg, PCEP_DEBUG_MODE_BASIC))    \
 | |
| 			DEBUGD(&pcep_g->dbg, "pcep: " fmt, ##__VA_ARGS__);     \
 | |
| 	} while (0)
 | |
| #define PCEP_DEBUG_PATH(fmt, ...)                                              \
 | |
| 	do {                                                                   \
 | |
| 		if (DEBUG_FLAGS_CHECK(&pcep_g->dbg, PCEP_DEBUG_MODE_PATH))     \
 | |
| 			DEBUGD(&pcep_g->dbg, "pcep: " fmt, ##__VA_ARGS__);     \
 | |
| 	} while (0)
 | |
| #define PCEP_DEBUG_PCEP(fmt, ...)                                              \
 | |
| 	do {                                                                   \
 | |
| 		if (DEBUG_FLAGS_CHECK(&pcep_g->dbg, PCEP_DEBUG_MODE_PCEP))     \
 | |
| 			DEBUGD(&pcep_g->dbg, "pcep: " fmt, ##__VA_ARGS__);     \
 | |
| 	} while (0)
 | |
| #define PCEP_DEBUG_PCEPLIB(priority, fmt, ...)                                 \
 | |
| 	do {                                                                   \
 | |
| 		switch (priority) {                                            \
 | |
| 		case LOG_DEBUG:                                                \
 | |
| 			if (DEBUG_FLAGS_CHECK(&pcep_g->dbg,                    \
 | |
| 					      PCEP_DEBUG_MODE_PCEPLIB))        \
 | |
| 				DEBUGD(&pcep_g->dbg, "pcep: " fmt,             \
 | |
| 				       ##__VA_ARGS__);                         \
 | |
| 			break;                                                 \
 | |
| 		case LOG_INFO:                                                 \
 | |
| 			if (DEBUG_FLAGS_CHECK(&pcep_g->dbg,                    \
 | |
| 					      PCEP_DEBUG_MODE_PCEPLIB))        \
 | |
| 				DEBUGI(&pcep_g->dbg, "pcep: " fmt,             \
 | |
| 				       ##__VA_ARGS__);                         \
 | |
| 			break;                                                 \
 | |
| 		case LOG_NOTICE:                                               \
 | |
| 			if (DEBUG_FLAGS_CHECK(&pcep_g->dbg,                    \
 | |
| 					      PCEP_DEBUG_MODE_PCEPLIB))        \
 | |
| 				DEBUGN(&pcep_g->dbg, "pcep: " fmt,             \
 | |
| 				       ##__VA_ARGS__);                         \
 | |
| 			break;                                                 \
 | |
| 		case LOG_WARNING:                                              \
 | |
| 		case LOG_ERR:                                                  \
 | |
| 		default:                                                       \
 | |
| 			zlog(priority, "pcep: " fmt, ##__VA_ARGS__);           \
 | |
| 			break;                                                 \
 | |
| 		}                                                              \
 | |
| 	} while (0)
 | |
| 
 | |
| struct pcep_config_group_opts {
 | |
| 	char name[64];
 | |
| 	char tcp_md5_auth[TCP_MD5SIG_MAXKEYLEN];
 | |
| 	struct ipaddr source_ip;
 | |
| 	short source_port;
 | |
| 	bool draft07;
 | |
| 	bool pce_initiated;
 | |
| 	int keep_alive_seconds;
 | |
| 	int min_keep_alive_seconds;
 | |
| 	int max_keep_alive_seconds;
 | |
| 	int dead_timer_seconds;
 | |
| 	int min_dead_timer_seconds;
 | |
| 	int max_dead_timer_seconds;
 | |
| 	int pcep_request_time_seconds;
 | |
| 	int session_timeout_inteval_seconds;
 | |
| 	int delegation_timeout_seconds;
 | |
| };
 | |
| 
 | |
| struct pce_opts {
 | |
| 	struct ipaddr addr;
 | |
| 	short port;
 | |
| 	char pce_name[64];
 | |
| 	struct pcep_config_group_opts config_opts;
 | |
| 	uint8_t precedence; /* Multi-PCE precedence */
 | |
| };
 | |
| 
 | |
| struct pcc_opts {
 | |
| 	struct ipaddr addr;
 | |
| 	short port;
 | |
| 	short msd;
 | |
| };
 | |
| 
 | |
| /* Encapsulate the pce_opts with needed CLI information */
 | |
| struct pce_opts_cli {
 | |
| 	struct pce_opts pce_opts;
 | |
| 	char config_group_name[64];
 | |
| 	/* These are the values configured in the pcc-peer sub-commands.
 | |
| 	 * These need to be stored for later merging. Notice, it could
 | |
| 	 * be that not all of them are set. */
 | |
| 	struct pcep_config_group_opts pce_config_group_opts;
 | |
| 	/* The pce_opts->config_opts will be a merge of the default values,
 | |
| 	 * optional config_group values (which overwrite default values),
 | |
| 	 * and any values configured in the pce sub-commands (which overwrite
 | |
| 	 * both default and config_group values). This flag indicates of the
 | |
| 	 * values need to be merged or not. */
 | |
| 	bool merged;
 | |
| };
 | |
| 
 | |
| struct lsp_nb_key {
 | |
| 	uint32_t color;
 | |
| 	struct ipaddr endpoint;
 | |
| 	uint32_t preference;
 | |
| };
 | |
| 
 | |
| struct sid_mpls {
 | |
| 	mpls_label_t label;
 | |
| 	uint8_t traffic_class;
 | |
| 	bool is_bottom;
 | |
| 	uint8_t ttl;
 | |
| };
 | |
| 
 | |
| struct pcep_caps {
 | |
| 	bool is_stateful;
 | |
| 	/* If we know the objective functions supported by the PCE.
 | |
| 	 * If we don't know, it doesn't mean the PCE doesn't support any */
 | |
| 	bool supported_ofs_are_known;
 | |
| 	/* Defined if we know which objective funtions are supported by the PCE.
 | |
| 	 * One bit per objective function, the bit index being equal to
 | |
| 	 * enum pcep_objfun_type values: bit 0 is not used, bit 1 is
 | |
| 	 * PCEP_OBJFUN_MCP, up to bit 17 that is PCEP_OBJFUN_MSN */
 | |
| 	uint32_t supported_ofs;
 | |
| };
 | |
| 
 | |
| union sid {
 | |
| 	uint32_t value;
 | |
| 	struct sid_mpls mpls;
 | |
| };
 | |
| 
 | |
| struct nai {
 | |
| 	/* NAI type */
 | |
| 	enum pcep_sr_subobj_nai type;
 | |
| 	/* Local IP address*/
 | |
| 	struct ipaddr local_addr;
 | |
| 	/* Local interface identifier if the NAI is an unnumbered adjacency */
 | |
| 	uint32_t local_iface;
 | |
| 	/* Remote address if the NAI is an adjacency */
 | |
| 	struct ipaddr remote_addr;
 | |
| 	/* Remote interface identifier if the NAI is an unnumbered adjacency */
 | |
| 	uint32_t remote_iface;
 | |
| };
 | |
| 
 | |
| struct path_hop {
 | |
| 	/* Pointer to the next hop in the path */
 | |
| 	struct path_hop *next;
 | |
| 	/* Indicateif this ia a loose or strict hop */
 | |
| 	bool is_loose;
 | |
| 	/* Indicate if there is an SID for the hop */
 | |
| 	bool has_sid;
 | |
| 	/* Indicate if the hop as a MPLS label */
 | |
| 	bool is_mpls;
 | |
| 	/* Indicate if the MPLS label has extra attributes (TTL, class..)*/
 | |
| 	bool has_attribs;
 | |
| 	/* Hop's SID if available */
 | |
| 	union sid sid;
 | |
| 	/* Indicate if there is a NAI for this hop */
 | |
| 	bool has_nai;
 | |
| 	/* NAI if available */
 | |
| 	struct nai nai;
 | |
| };
 | |
| 
 | |
| struct path_metric {
 | |
| 	/* Pointer to the next metric */
 | |
| 	struct path_metric *next;
 | |
| 	/* The metric type */
 | |
| 	enum pcep_metric_types type;
 | |
| 	/* If the metric should be enforced */
 | |
| 	bool enforce;
 | |
| 	/* If the metric value is bound (a maximum) */
 | |
| 	bool is_bound;
 | |
| 	/* If the metric value is computed */
 | |
| 	bool is_computed;
 | |
| 	/* The metric value */
 | |
| 	float value;
 | |
| };
 | |
| 
 | |
| struct path {
 | |
| 	/* Both the nbkey and the plspid are keys comming from the PCC,
 | |
| 	but the PCE is only using the plspid. The missing key is looked up by
 | |
| 	the PCC so we always have both */
 | |
| 
 | |
| 	/* The northbound key identifying this path */
 | |
| 	struct lsp_nb_key nbkey;
 | |
| 	/* The generated unique PLSP identifier for this path.
 | |
| 	   See draft-ietf-pce-stateful-pce */
 | |
| 	uint32_t plsp_id;
 | |
| 
 | |
| 	/* The transport address the path is comming from, PCE or PCC*/
 | |
| 	struct ipaddr sender;
 | |
| 	/* The pcc protocol address, must be the same family as the endpoint */
 | |
| 	struct ipaddr pcc_addr;
 | |
| 
 | |
| 	/* The identifier of the PCC the path is for/from. If 0 it is undefined,
 | |
| 	meaning it hasn't be set yet or is for all the PCC */
 | |
| 	int pcc_id;
 | |
| 
 | |
| 	/* The origin of the path creation */
 | |
| 	enum srte_protocol_origin create_origin;
 | |
| 	/* The origin of the path modification */
 | |
| 	enum srte_protocol_origin update_origin;
 | |
| 	/* The identifier of the entity that originated the path */
 | |
| 	const char *originator;
 | |
| 	/* The type of the path, for PCE initiated or updated path it is always
 | |
| 	SRTE_CANDIDATE_TYPE_DYNAMIC */
 | |
| 	enum srte_candidate_type type;
 | |
| 
 | |
| 	/* The following data comes from either the PCC or the PCE if available
 | |
| 	 */
 | |
| 
 | |
| 	/* Path's binding SID */
 | |
| 	mpls_label_t binding_sid;
 | |
| 	/* The name of the path */
 | |
| 	const char *name;
 | |
| 	/* The request identifier from the PCE, when getting a path from the
 | |
| 	   PCE. See draft-ietf-pce-stateful-pce */
 | |
| 	uint32_t srp_id;
 | |
| 	/* The request identifier from the PCC , when getting a path from the
 | |
| 	   PCE after a computation request. See rfc5440, section-7.4 */
 | |
| 	uint32_t req_id;
 | |
| 	/* The operational status of the path */
 | |
| 	enum pcep_lsp_operational_status status;
 | |
| 	/* If true, the receiver (PCC) must remove the path.
 | |
| 	   See draft-ietf-pce-pce-initiated-lsp */
 | |
| 	bool do_remove;
 | |
| 	/* Indicate the given path was removed by the PCC.
 | |
| 	   See draft-ietf-pce-stateful-pce, section-7.3, flag R */
 | |
| 	bool was_removed;
 | |
| 	/* Indicate the path is part of the synchronization process.
 | |
| 	   See draft-ietf-pce-stateful-pce, section-7.3, flag S */
 | |
| 	bool is_synching;
 | |
| 	/* Indicate if the path bandwidth requirment is defined */
 | |
| 	bool has_bandwidth;
 | |
| 	/* Indicate if the bandwidth requirment should be enforced */
 | |
| 	bool enforce_bandwidth;
 | |
| 	/* Path required bandwidth if defined */
 | |
| 	float bandwidth;
 | |
| 	/* Specify the list of hop defining the path */
 | |
| 	struct path_hop *first_hop;
 | |
| 	/* Specify the list of metrics */
 | |
| 	struct path_metric *first_metric;
 | |
| 	/* Indicate if the path has a PCC-defined objective function */
 | |
| 	bool has_pcc_objfun;
 | |
| 	/* Indicate the PCC-defined objective function is required */
 | |
| 	bool enforce_pcc_objfun;
 | |
| 	/* PCC-defined Objective Function */
 | |
| 	enum objfun_type pcc_objfun;
 | |
| 	/* Indicate if the path has a PCE-defined objective function */
 | |
| 	bool has_pce_objfun;
 | |
| 	/* PCE-defined Objective Function */
 | |
| 	enum objfun_type pce_objfun;
 | |
| 	/* Indicate if some affinity filters are defined */
 | |
| 	bool has_affinity_filters;
 | |
| 	/* Affinity attribute filters indexed by enum affinity_filter_type - 1
 | |
| 	 */
 | |
| 	uint32_t affinity_filters[MAX_AFFINITY_FILTER_TYPE];
 | |
| 
 | |
| 	/* The following data need to be specialized for a given PCE */
 | |
| 
 | |
| 	/* Indicate the path is delegated to the PCE.
 | |
| 	   See draft-ietf-pce-stateful-pce, section-7.3, flag D */
 | |
| 	bool is_delegated;
 | |
| 	/* Indicate if the PCE wants the path to get active.
 | |
| 	   See draft-ietf-pce-stateful-pce, section-7.3, flag A */
 | |
| 	bool go_active;
 | |
| 	/* Indicate the given path was created by the PCE,
 | |
| 	   See draft-ietf-pce-pce-initiated-lsp, section-5.3.1, flag C */
 | |
| 	bool was_created;
 | |
| 
 | |
| 	/* The following data is defined for comnputation replies */
 | |
| 
 | |
| 	/* Indicate that no path could be computed */
 | |
| 	bool no_path;
 | |
| };
 | |
| 
 | |
| struct pcep_glob {
 | |
| 	struct debug dbg;
 | |
| 	struct thread_master *master;
 | |
| 	struct frr_pthread *fpt;
 | |
| 	uint8_t num_pce_opts_cli;
 | |
| 	struct pce_opts_cli *pce_opts_cli[MAX_PCE];
 | |
| 	uint8_t num_config_group_opts;
 | |
| 	struct pcep_config_group_opts *config_group_opts[MAX_PCE];
 | |
| };
 | |
| 
 | |
| extern struct pcep_glob *pcep_g;
 | |
| 
 | |
| /* Path Helper Functions */
 | |
| struct path *pcep_new_path(void);
 | |
| struct path_hop *pcep_new_hop(void);
 | |
| struct path_metric *pcep_new_metric(void);
 | |
| struct path *pcep_copy_path(struct path *path);
 | |
| void pcep_free_path(struct path *path);
 | |
| 
 | |
| 
 | |
| #endif // _PATH_PCEP_H_
 |