blob: f5611f922714da2d2f1ce16c54bd404c9328caa7 [file] [log] [blame]
Neale Ranns0bfe5d82016-08-25 15:29:12 +01001/*
2 * Copyright (c) 2016 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#include <vnet/mpls/mpls.h>
17#include <vnet/dpo/mpls_label_dpo.h>
18#include <vnet/dpo/load_balance.h>
19#include <vnet/dpo/drop_dpo.h>
20
Neale Rannsad422ed2016-11-02 14:20:04 +000021#include <vnet/fib/fib_path_ext.h>
22#include <vnet/fib/fib_entry_src.h>
23#include <vnet/fib/fib_path.h>
24#include <vnet/fib/fib_path_list.h>
25#include <vnet/fib/fib_internal.h>
Neale Ranns0bfe5d82016-08-25 15:29:12 +010026
Neale Ranns81424992017-05-18 03:03:22 -070027const char *fib_path_ext_adj_flags_names[] = FIB_PATH_EXT_ADJ_ATTR_NAMES;
Neale Ranns31ed7442018-02-23 05:29:09 -080028const char *fib_path_ext_mpls_flags_names[] = FIB_PATH_EXT_MPLS_ATTR_NAMES;
Neale Ranns81424992017-05-18 03:03:22 -070029
Neale Ranns0bfe5d82016-08-25 15:29:12 +010030u8 *
31format_fib_path_ext (u8 * s, va_list * args)
32{
33 fib_path_ext_t *path_ext;
Neale Rannsad422ed2016-11-02 14:20:04 +000034 u32 ii;
Neale Ranns0bfe5d82016-08-25 15:29:12 +010035
36 path_ext = va_arg (*args, fib_path_ext_t *);
37
Neale Ranns81424992017-05-18 03:03:22 -070038 s = format(s, "path:%d ", path_ext->fpe_path_index);
39
40 switch (path_ext->fpe_type)
Neale Rannsad422ed2016-11-02 14:20:04 +000041 {
Neale Ranns31ed7442018-02-23 05:29:09 -080042 case FIB_PATH_EXT_MPLS: {
43 fib_path_ext_mpls_attr_t attr;
44
45 if (path_ext->fpe_mpls_flags)
46 {
47 s = format(s, "mpls-flags:[");
48
49 FOR_EACH_PATH_EXT_MPLS_ATTR(attr)
50 {
51 if ((1<<attr) & path_ext->fpe_mpls_flags) {
52 s = format(s, "%s", fib_path_ext_mpls_flags_names[attr]);
53 }
54 }
55 s = format(s, "]");
56 }
Neale Rannsfe0a2c52018-08-27 05:45:07 -070057 s = format(s, " labels:[");
Neale Ranns81424992017-05-18 03:03:22 -070058 for (ii = 0; ii < vec_len(path_ext->fpe_path.frp_label_stack); ii++)
59 {
Neale Ranns31ed7442018-02-23 05:29:09 -080060 s = format(s, "[%U]",
61 format_fib_mpls_label,
62 &path_ext->fpe_path.frp_label_stack[ii]);
Neale Ranns81424992017-05-18 03:03:22 -070063 }
Neale Ranns31ed7442018-02-23 05:29:09 -080064 s = format(s, "]");
Neale Ranns81424992017-05-18 03:03:22 -070065 break;
Neale Ranns31ed7442018-02-23 05:29:09 -080066 }
Neale Ranns81424992017-05-18 03:03:22 -070067 case FIB_PATH_EXT_ADJ: {
68 fib_path_ext_adj_attr_t attr;
69
Neale Ranns81424992017-05-18 03:03:22 -070070 if (path_ext->fpe_adj_flags)
71 {
Neale Ranns31ed7442018-02-23 05:29:09 -080072 s = format(s, "adj-flags:[");
Neale Ranns81424992017-05-18 03:03:22 -070073 FOR_EACH_PATH_EXT_ADJ_ATTR(attr)
74 {
Neale Ranns31ed7442018-02-23 05:29:09 -080075 if ((1<<attr) & path_ext->fpe_adj_flags)
76 {
77 s = format(s, "%s", fib_path_ext_adj_flags_names[attr]);
78 }
Neale Ranns81424992017-05-18 03:03:22 -070079 }
Neale Ranns31ed7442018-02-23 05:29:09 -080080 s = format(s, "]");
Neale Ranns81424992017-05-18 03:03:22 -070081 }
82 break;
83 }
Neale Rannsad422ed2016-11-02 14:20:04 +000084 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +010085 return (s);
86}
87
88int
89fib_path_ext_cmp (fib_path_ext_t *path_ext,
90 const fib_route_path_t *rpath)
91{
92 return (fib_route_path_cmp(&path_ext->fpe_path, rpath));
93}
94
Neale Ranns81424992017-05-18 03:03:22 -070095static fib_path_list_walk_rc_t
Neale Ranns0bfe5d82016-08-25 15:29:12 +010096fib_path_ext_match (fib_node_index_t pl_index,
97 fib_node_index_t path_index,
98 void *ctx)
99{
100 fib_path_ext_t *path_ext = ctx;
101
102 if (!fib_path_cmp_w_route_path(path_index,
103 &path_ext->fpe_path))
104 {
105 path_ext->fpe_path_index = path_index;
Neale Ranns81424992017-05-18 03:03:22 -0700106 return (FIB_PATH_LIST_WALK_STOP);
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100107 }
Neale Ranns81424992017-05-18 03:03:22 -0700108 return (FIB_PATH_LIST_WALK_CONTINUE);
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100109}
110
111void
112fib_path_ext_resolve (fib_path_ext_t *path_ext,
113 fib_node_index_t path_list_index)
114{
115 /*
116 * Find the path on the path list that this is an extension for
117 */
118 path_ext->fpe_path_index = FIB_NODE_INDEX_INVALID;
119 fib_path_list_walk(path_list_index,
120 fib_path_ext_match,
121 path_ext);
122}
123
Neale Rannsdcd6d622017-05-26 02:59:16 -0700124static void
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100125fib_path_ext_init (fib_path_ext_t *path_ext,
126 fib_node_index_t path_list_index,
Neale Ranns81424992017-05-18 03:03:22 -0700127 fib_path_ext_type_t ext_type,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100128 const fib_route_path_t *rpath)
129{
130 path_ext->fpe_path = *rpath;
131 path_ext->fpe_path_index = FIB_NODE_INDEX_INVALID;
Neale Ranns81424992017-05-18 03:03:22 -0700132 path_ext->fpe_adj_flags = FIB_PATH_EXT_ADJ_FLAG_NONE;
133 path_ext->fpe_type = ext_type;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100134
135 fib_path_ext_resolve(path_ext, path_list_index);
136}
137
Neale Rannsad422ed2016-11-02 14:20:04 +0000138/**
139 * @brief Return true if the label stack is implicit null
Neale Ranns31ed7442018-02-23 05:29:09 -0800140 * imp-null and pop equate to the same this as this level -
141 * the label is coming off.
Neale Rannsad422ed2016-11-02 14:20:04 +0000142 */
143static int
144fib_path_ext_is_imp_null (fib_path_ext_t *path_ext)
145{
146 return ((1 == vec_len(path_ext->fpe_label_stack)) &&
Neale Ranns31ed7442018-02-23 05:29:09 -0800147 ((MPLS_IETF_IMPLICIT_NULL_LABEL == path_ext->fpe_label_stack[0].fml_value) ||
148 (MPLS_LABEL_POP == path_ext->fpe_label_stack[0].fml_value)));
149}
150
151mpls_label_dpo_flags_t
152fib_path_ext_mpls_flags_to_mpls_label (fib_path_ext_mpls_flags_t fpe_flags)
153{
154 mpls_label_dpo_flags_t ml_flags = MPLS_LABEL_DPO_FLAG_NONE;
155
156 if (fpe_flags &FIB_PATH_EXT_MPLS_FLAG_NO_IP_TTL_DECR)
157 {
158 ml_flags |= MPLS_LABEL_DPO_FLAG_NO_IP_TTL_DECR;
159 }
160
161 return (ml_flags);
Neale Rannsad422ed2016-11-02 14:20:04 +0000162}
163
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100164load_balance_path_t *
165fib_path_ext_stack (fib_path_ext_t *path_ext,
Neale Ranns53962fb2021-12-20 18:18:42 +0000166 dpo_proto_t payload_proto,
Neale Rannsad422ed2016-11-02 14:20:04 +0000167 fib_forward_chain_type_t child_fct,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100168 load_balance_path_t *nhs)
169{
Neale Rannsad422ed2016-11-02 14:20:04 +0000170 fib_forward_chain_type_t parent_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100171 load_balance_path_t *nh;
172
173 if (!fib_path_is_resolved(path_ext->fpe_path_index))
174 return (nhs);
175
176 /*
177 * Since we are stacking this path-extension, it must have a valid out
178 * label. From the chain type request by the child, determine what
179 * chain type we will request from the parent.
180 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000181 switch (child_fct)
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100182 {
183 case FIB_FORW_CHAIN_TYPE_MPLS_EOS:
Neale Rannsad422ed2016-11-02 14:20:04 +0000184 {
185 /*
186 * The EOS chain is a tricky since, when the path has an imp NULL one cannot know
187 * the adjacency to link to without knowing what the packets payload protocol
188 * will be once the label is popped.
189 */
190 if (fib_path_ext_is_imp_null(path_ext))
191 {
Neale Ranns53962fb2021-12-20 18:18:42 +0000192 parent_fct = fib_forw_chain_type_from_dpo_proto(payload_proto);
Neale Rannsad422ed2016-11-02 14:20:04 +0000193 }
194 else
195 {
196 /*
197 * we have a label to stack. packets will thus be labelled when
198 * they encounter the child, ergo, non-eos.
199 */
200 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
201 }
Neale Ranns5899fde2016-10-12 13:51:05 +0100202 break;
Neale Rannsad422ed2016-11-02 14:20:04 +0000203 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100204 case FIB_FORW_CHAIN_TYPE_UNICAST_IP4:
205 case FIB_FORW_CHAIN_TYPE_UNICAST_IP6:
Neale Rannsad422ed2016-11-02 14:20:04 +0000206 if (fib_path_ext_is_imp_null(path_ext))
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100207 {
208 /*
209 * implicit-null label for the eos or IP chain, need to pick up
210 * the IP adj
211 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000212 parent_fct = child_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100213 }
214 else
215 {
216 /*
217 * we have a label to stack. packets will thus be labelled when
Neale Rannsad422ed2016-11-02 14:20:04 +0000218 * they encounter the child, ergo, non-eos.
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100219 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000220 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100221 }
222 break;
223 case FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS:
Neale Rannsad422ed2016-11-02 14:20:04 +0000224 parent_fct = child_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100225 break;
Neale Rannsda78f952017-05-24 09:15:43 -0700226 case FIB_FORW_CHAIN_TYPE_ETHERNET:
227 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
228 break;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100229 default:
230 return (nhs);
231 break;
232 }
233
Neale Ranns948e00f2016-10-20 13:39:34 +0100234 dpo_id_t via_dpo = DPO_INVALID;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100235
236 /*
237 * The next object in the graph after the imposition of the label
238 * will be the DPO contributed by the path through which the packets
239 * are to be sent. We stack the MPLS Label DPO on this path DPO
240 */
241 fib_path_contribute_forwarding(path_ext->fpe_path_index,
Neale Rannsad422ed2016-11-02 14:20:04 +0000242 parent_fct,
Neale Ranns53962fb2021-12-20 18:18:42 +0000243 payload_proto,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100244 &via_dpo);
245
246 if (dpo_is_drop(&via_dpo) ||
247 load_balance_is_drop(&via_dpo))
248 {
249 /*
250 * don't stack a path extension on a drop. doing so will create
251 * a LB bucket entry on drop, and we will lose a percentage of traffic.
252 */
253 }
254 else
255 {
256 vec_add2(nhs, nh, 1);
257 nh->path_weight = fib_path_get_weight(path_ext->fpe_path_index);
258 nh->path_index = path_ext->fpe_path_index;
259 dpo_copy(&nh->path_dpo, &via_dpo);
260
261 /*
262 * The label is stackable for this chain type
263 * construct the mpls header that will be imposed in the data-path
264 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000265 if (!fib_path_ext_is_imp_null(path_ext))
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100266 {
Neale Rannsad422ed2016-11-02 14:20:04 +0000267 /*
268 * we use the parent protocol for the label so that
269 * we pickup the correct MPLS imposition nodes to do
270 * ip[46] processing.
271 */
Neale Ranns31ed7442018-02-23 05:29:09 -0800272 dpo_id_t parent = DPO_INVALID;
Neale Rannsad422ed2016-11-02 14:20:04 +0000273 dpo_proto_t chain_proto;
274 mpls_eos_bit_t eos;
Neale Rannsad422ed2016-11-02 14:20:04 +0000275
276 eos = (child_fct == FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS ?
277 MPLS_NON_EOS :
278 MPLS_EOS);
279 chain_proto = fib_forw_chain_type_to_dpo_proto(child_fct);
280
Neale Ranns31ed7442018-02-23 05:29:09 -0800281 dpo_copy(&parent, &nh->path_dpo);
282 mpls_label_dpo_create(path_ext->fpe_label_stack,
283 eos,
284 chain_proto,
285 fib_path_ext_mpls_flags_to_mpls_label(
286 path_ext->fpe_mpls_flags),
287 &parent,
288 &nh->path_dpo);
Neale Rannsad422ed2016-11-02 14:20:04 +0000289
Neale Ranns31ed7442018-02-23 05:29:09 -0800290 dpo_reset(&parent);
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100291 }
Neale Ranns62fe07c2017-10-31 12:28:22 -0700292 else if (child_fct == FIB_FORW_CHAIN_TYPE_MPLS_EOS)
293 {
294 /*
295 * MPLS EOS packets using an imp-null. Insert the disposition.
296 */
297 fib_path_stack_mpls_disp(nh->path_index,
298 fib_forw_chain_type_to_dpo_proto(parent_fct),
Neale Ranns31ed7442018-02-23 05:29:09 -0800299 path_ext->fpe_label_stack[0].fml_mode,
Neale Ranns62fe07c2017-10-31 12:28:22 -0700300 &nh->path_dpo);
301 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100302 }
303 dpo_reset(&via_dpo);
304
305 return (nhs);
306}
Neale Ranns81424992017-05-18 03:03:22 -0700307
308fib_path_ext_t *
309fib_path_ext_list_find (const fib_path_ext_list_t *list,
310 fib_path_ext_type_t ext_type,
311 const fib_route_path_t *rpath)
312{
313 fib_path_ext_t *path_ext;
314
315 vec_foreach(path_ext, list->fpel_exts)
316 {
317 if ((path_ext->fpe_type == ext_type) &&
318 !fib_path_ext_cmp(path_ext, rpath) )
319 {
320 return (path_ext);
321 }
322 }
323 return (NULL);
324}
325
326fib_path_ext_t *
327fib_path_ext_list_find_by_path_index (const fib_path_ext_list_t *list,
328 fib_node_index_t path_index)
329{
330 fib_path_ext_t *path_ext;
331
Neale Ranns775f73c2018-12-20 03:01:49 -0800332 if (NULL != list)
Neale Ranns81424992017-05-18 03:03:22 -0700333 {
Neale Ranns775f73c2018-12-20 03:01:49 -0800334 vec_foreach(path_ext, list->fpel_exts)
Neale Ranns81424992017-05-18 03:03:22 -0700335 {
Neale Ranns775f73c2018-12-20 03:01:49 -0800336 if (path_ext->fpe_path_index == path_index)
337 {
338 return (path_ext);
339 }
Neale Ranns81424992017-05-18 03:03:22 -0700340 }
341 }
342 return (NULL);
343}
344
345
346fib_path_ext_t *
347fib_path_ext_list_push_back (fib_path_ext_list_t *list,
348 fib_node_index_t path_list_index,
349 fib_path_ext_type_t ext_type,
350 const fib_route_path_t *rpath)
351{
352 fib_path_ext_t *path_ext;
353
354 path_ext = fib_path_ext_list_find(list, ext_type, rpath);
355
356 if (NULL == path_ext)
357 {
358 vec_add2(list->fpel_exts, path_ext, 1);
359 fib_path_ext_init(path_ext, path_list_index, ext_type, rpath);
360 }
361
362 return (path_ext);
363}
364
365/*
366 * insert, sorted, a path extension to the entry's list.
367 * It's not strictly necessary to sort the path extensions, since each
368 * extension has the path index to which it resolves. However, by being
369 * sorted the load-balance produced has a deterministic order, not an order
370 * based on the sequence of extension additions. this is a considerable benefit.
371 */
372fib_path_ext_t *
373fib_path_ext_list_insert (fib_path_ext_list_t *list,
374 fib_node_index_t path_list_index,
375 fib_path_ext_type_t ext_type,
376 const fib_route_path_t *rpath)
377{
378 fib_path_ext_t new_path_ext, *path_ext;
379 int i = 0;
380
381 if (0 == fib_path_ext_list_length(list))
382 {
383 return (fib_path_ext_list_push_back(list, path_list_index,
384 ext_type, rpath));
385 }
386
387 fib_path_ext_init(&new_path_ext, path_list_index, ext_type, rpath);
388
389 vec_foreach(path_ext, list->fpel_exts)
390 {
Neale Rannsdcd6d622017-05-26 02:59:16 -0700391 int res = fib_path_ext_cmp(path_ext, rpath);
392
393 if (0 == res)
394 {
395 /*
396 * don't add duplicate extensions. modify instead
397 */
398 vec_free(path_ext->fpe_label_stack);
399 *path_ext = new_path_ext;
400 goto done;
401 }
402 else if (res < 0)
Neale Ranns81424992017-05-18 03:03:22 -0700403 {
404 i++;
405 }
406 else
407 {
408 break;
409 }
410 }
411 vec_insert_elts(list->fpel_exts, &new_path_ext, 1, i);
Neale Rannsdcd6d622017-05-26 02:59:16 -0700412done:
Neale Ranns81424992017-05-18 03:03:22 -0700413 return (&(list->fpel_exts[i]));
414}
415
416void
417fib_path_ext_list_resolve (fib_path_ext_list_t *list,
418 fib_node_index_t path_list_index)
419{
420 fib_path_ext_t *path_ext;
421
422 vec_foreach(path_ext, list->fpel_exts)
423 {
424 fib_path_ext_resolve(path_ext, path_list_index);
425 };
426}
427
428void
429fib_path_ext_list_remove (fib_path_ext_list_t *list,
430 fib_path_ext_type_t ext_type,
431 const fib_route_path_t *rpath)
432{
433 fib_path_ext_t *path_ext;
434
435 path_ext = fib_path_ext_list_find(list, ext_type, rpath);
436
437 if (NULL != path_ext)
438 {
439 /*
440 * delete the element moving the remaining elements down 1 position.
441 * this preserves the sorted order.
442 */
443 vec_free(path_ext->fpe_label_stack);
444 vec_delete(list->fpel_exts, 1, (path_ext - list->fpel_exts));
445 }
446}
447
448void
449fib_path_ext_list_flush (fib_path_ext_list_t *list)
450{
451 fib_path_ext_t *path_ext;
452
453 vec_foreach(path_ext, list->fpel_exts)
454 {
455 vec_free(path_ext->fpe_label_stack);
456 };
457 vec_free(list->fpel_exts);
458 list->fpel_exts = NULL;
459}
460
461u8*
462format_fib_path_ext_list (u8 * s, va_list * args)
463{
464 fib_path_ext_list_t *list;
465 fib_path_ext_t *path_ext;
466
467 list = va_arg (*args, fib_path_ext_list_t *);
468
469 if (fib_path_ext_list_length(list))
470 {
471 s = format(s, " Extensions:");
472 vec_foreach(path_ext, list->fpel_exts)
473 {
474 s = format(s, "\n %U", format_fib_path_ext, path_ext);
475 };
476 }
477
478 return (s);
479}
480
481int
482fib_path_ext_list_length (const fib_path_ext_list_t *list)
483{
484 return (vec_len(list->fpel_exts));
485}