blob: a285ba07f7c3064fb329a11ab6133cae1f707d14 [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;
28
Neale Ranns0bfe5d82016-08-25 15:29:12 +010029u8 *
30format_fib_path_ext (u8 * s, va_list * args)
31{
32 fib_path_ext_t *path_ext;
Neale Rannsad422ed2016-11-02 14:20:04 +000033 u32 ii;
Neale Ranns0bfe5d82016-08-25 15:29:12 +010034
35 path_ext = va_arg (*args, fib_path_ext_t *);
36
Neale Ranns81424992017-05-18 03:03:22 -070037 s = format(s, "path:%d ", path_ext->fpe_path_index);
38
39 switch (path_ext->fpe_type)
Neale Rannsad422ed2016-11-02 14:20:04 +000040 {
Neale Ranns81424992017-05-18 03:03:22 -070041 case FIB_PATH_EXT_MPLS:
42 s = format(s, "labels:",
43 path_ext->fpe_path_index);
44 for (ii = 0; ii < vec_len(path_ext->fpe_path.frp_label_stack); ii++)
45 {
46 s = format(s, "%U ",
47 format_mpls_unicast_label,
48 path_ext->fpe_path.frp_label_stack[ii]);
49 }
50 break;
51 case FIB_PATH_EXT_ADJ: {
52 fib_path_ext_adj_attr_t attr;
53
54 s = format(s, "adj-flags:");
55 if (path_ext->fpe_adj_flags)
56 {
57 FOR_EACH_PATH_EXT_ADJ_ATTR(attr)
58 {
59 s = format(s, "%s", fib_path_ext_adj_flags_names[attr]);
60 }
61 }
62 else
63 {
64 s = format(s, "None");
65 }
66 break;
67 }
Neale Rannsad422ed2016-11-02 14:20:04 +000068 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +010069 return (s);
70}
71
72int
73fib_path_ext_cmp (fib_path_ext_t *path_ext,
74 const fib_route_path_t *rpath)
75{
76 return (fib_route_path_cmp(&path_ext->fpe_path, rpath));
77}
78
Neale Ranns81424992017-05-18 03:03:22 -070079static fib_path_list_walk_rc_t
Neale Ranns0bfe5d82016-08-25 15:29:12 +010080fib_path_ext_match (fib_node_index_t pl_index,
81 fib_node_index_t path_index,
82 void *ctx)
83{
84 fib_path_ext_t *path_ext = ctx;
85
86 if (!fib_path_cmp_w_route_path(path_index,
87 &path_ext->fpe_path))
88 {
89 path_ext->fpe_path_index = path_index;
Neale Ranns81424992017-05-18 03:03:22 -070090 return (FIB_PATH_LIST_WALK_STOP);
Neale Ranns0bfe5d82016-08-25 15:29:12 +010091 }
Neale Ranns81424992017-05-18 03:03:22 -070092 return (FIB_PATH_LIST_WALK_CONTINUE);
Neale Ranns0bfe5d82016-08-25 15:29:12 +010093}
94
95void
96fib_path_ext_resolve (fib_path_ext_t *path_ext,
97 fib_node_index_t path_list_index)
98{
99 /*
100 * Find the path on the path list that this is an extension for
101 */
102 path_ext->fpe_path_index = FIB_NODE_INDEX_INVALID;
103 fib_path_list_walk(path_list_index,
104 fib_path_ext_match,
105 path_ext);
106}
107
Neale Rannsdcd6d622017-05-26 02:59:16 -0700108static void
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100109fib_path_ext_init (fib_path_ext_t *path_ext,
110 fib_node_index_t path_list_index,
Neale Ranns81424992017-05-18 03:03:22 -0700111 fib_path_ext_type_t ext_type,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100112 const fib_route_path_t *rpath)
113{
114 path_ext->fpe_path = *rpath;
115 path_ext->fpe_path_index = FIB_NODE_INDEX_INVALID;
Neale Ranns81424992017-05-18 03:03:22 -0700116 path_ext->fpe_adj_flags = FIB_PATH_EXT_ADJ_FLAG_NONE;
117 path_ext->fpe_type = ext_type;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100118
119 fib_path_ext_resolve(path_ext, path_list_index);
120}
121
Neale Rannsad422ed2016-11-02 14:20:04 +0000122/**
123 * @brief Return true if the label stack is implicit null
124 */
125static int
126fib_path_ext_is_imp_null (fib_path_ext_t *path_ext)
127{
128 return ((1 == vec_len(path_ext->fpe_label_stack)) &&
129 (MPLS_IETF_IMPLICIT_NULL_LABEL == path_ext->fpe_label_stack[0]));
130}
131
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100132load_balance_path_t *
133fib_path_ext_stack (fib_path_ext_t *path_ext,
Neale Rannsad422ed2016-11-02 14:20:04 +0000134 fib_forward_chain_type_t child_fct,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800135 fib_forward_chain_type_t imp_null_fct,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100136 load_balance_path_t *nhs)
137{
Neale Rannsad422ed2016-11-02 14:20:04 +0000138 fib_forward_chain_type_t parent_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100139 load_balance_path_t *nh;
140
141 if (!fib_path_is_resolved(path_ext->fpe_path_index))
142 return (nhs);
143
144 /*
145 * Since we are stacking this path-extension, it must have a valid out
146 * label. From the chain type request by the child, determine what
147 * chain type we will request from the parent.
148 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000149 switch (child_fct)
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100150 {
151 case FIB_FORW_CHAIN_TYPE_MPLS_EOS:
Neale Rannsad422ed2016-11-02 14:20:04 +0000152 {
153 /*
154 * The EOS chain is a tricky since, when the path has an imp NULL one cannot know
155 * the adjacency to link to without knowing what the packets payload protocol
156 * will be once the label is popped.
157 */
158 if (fib_path_ext_is_imp_null(path_ext))
159 {
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800160 parent_fct = imp_null_fct;
Neale Rannsad422ed2016-11-02 14:20:04 +0000161 }
162 else
163 {
164 /*
165 * we have a label to stack. packets will thus be labelled when
166 * they encounter the child, ergo, non-eos.
167 */
168 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
169 }
Neale Ranns5899fde2016-10-12 13:51:05 +0100170 break;
Neale Rannsad422ed2016-11-02 14:20:04 +0000171 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100172 case FIB_FORW_CHAIN_TYPE_UNICAST_IP4:
173 case FIB_FORW_CHAIN_TYPE_UNICAST_IP6:
Neale Rannsad422ed2016-11-02 14:20:04 +0000174 if (fib_path_ext_is_imp_null(path_ext))
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100175 {
176 /*
177 * implicit-null label for the eos or IP chain, need to pick up
178 * the IP adj
179 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000180 parent_fct = child_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100181 }
182 else
183 {
184 /*
185 * we have a label to stack. packets will thus be labelled when
Neale Rannsad422ed2016-11-02 14:20:04 +0000186 * they encounter the child, ergo, non-eos.
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100187 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000188 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100189 }
190 break;
191 case FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS:
Neale Rannsad422ed2016-11-02 14:20:04 +0000192 parent_fct = child_fct;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100193 break;
Neale Rannsda78f952017-05-24 09:15:43 -0700194 case FIB_FORW_CHAIN_TYPE_ETHERNET:
195 parent_fct = FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS;
196 break;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100197 default:
198 return (nhs);
199 break;
200 }
201
Neale Ranns948e00f2016-10-20 13:39:34 +0100202 dpo_id_t via_dpo = DPO_INVALID;
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100203
204 /*
205 * The next object in the graph after the imposition of the label
206 * will be the DPO contributed by the path through which the packets
207 * are to be sent. We stack the MPLS Label DPO on this path DPO
208 */
209 fib_path_contribute_forwarding(path_ext->fpe_path_index,
Neale Rannsad422ed2016-11-02 14:20:04 +0000210 parent_fct,
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100211 &via_dpo);
212
213 if (dpo_is_drop(&via_dpo) ||
214 load_balance_is_drop(&via_dpo))
215 {
216 /*
217 * don't stack a path extension on a drop. doing so will create
218 * a LB bucket entry on drop, and we will lose a percentage of traffic.
219 */
220 }
221 else
222 {
223 vec_add2(nhs, nh, 1);
224 nh->path_weight = fib_path_get_weight(path_ext->fpe_path_index);
225 nh->path_index = path_ext->fpe_path_index;
226 dpo_copy(&nh->path_dpo, &via_dpo);
227
228 /*
229 * The label is stackable for this chain type
230 * construct the mpls header that will be imposed in the data-path
231 */
Neale Rannsad422ed2016-11-02 14:20:04 +0000232 if (!fib_path_ext_is_imp_null(path_ext))
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100233 {
Neale Rannsad422ed2016-11-02 14:20:04 +0000234 /*
235 * we use the parent protocol for the label so that
236 * we pickup the correct MPLS imposition nodes to do
237 * ip[46] processing.
238 */
239 dpo_proto_t chain_proto;
240 mpls_eos_bit_t eos;
241 index_t mldi;
242
243 eos = (child_fct == FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS ?
244 MPLS_NON_EOS :
245 MPLS_EOS);
246 chain_proto = fib_forw_chain_type_to_dpo_proto(child_fct);
247
248 mldi = mpls_label_dpo_create(path_ext->fpe_label_stack,
249 eos, 255, 0,
250 chain_proto,
251 &nh->path_dpo);
252
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100253 dpo_set(&nh->path_dpo,
254 DPO_MPLS_LABEL,
Neale Rannsad422ed2016-11-02 14:20:04 +0000255 chain_proto,
256 mldi);
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100257 }
Neale Ranns62fe07c2017-10-31 12:28:22 -0700258 else if (child_fct == FIB_FORW_CHAIN_TYPE_MPLS_EOS)
259 {
260 /*
261 * MPLS EOS packets using an imp-null. Insert the disposition.
262 */
263 fib_path_stack_mpls_disp(nh->path_index,
264 fib_forw_chain_type_to_dpo_proto(parent_fct),
265 &nh->path_dpo);
266 }
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100267 }
268 dpo_reset(&via_dpo);
269
270 return (nhs);
271}
Neale Ranns81424992017-05-18 03:03:22 -0700272
273fib_path_ext_t *
274fib_path_ext_list_find (const fib_path_ext_list_t *list,
275 fib_path_ext_type_t ext_type,
276 const fib_route_path_t *rpath)
277{
278 fib_path_ext_t *path_ext;
279
280 vec_foreach(path_ext, list->fpel_exts)
281 {
282 if ((path_ext->fpe_type == ext_type) &&
283 !fib_path_ext_cmp(path_ext, rpath) )
284 {
285 return (path_ext);
286 }
287 }
288 return (NULL);
289}
290
291fib_path_ext_t *
292fib_path_ext_list_find_by_path_index (const fib_path_ext_list_t *list,
293 fib_node_index_t path_index)
294{
295 fib_path_ext_t *path_ext;
296
297 vec_foreach(path_ext, list->fpel_exts)
298 {
299 if (path_ext->fpe_path_index == path_index)
300 {
301 return (path_ext);
302 }
303 }
304 return (NULL);
305}
306
307
308fib_path_ext_t *
309fib_path_ext_list_push_back (fib_path_ext_list_t *list,
310 fib_node_index_t path_list_index,
311 fib_path_ext_type_t ext_type,
312 const fib_route_path_t *rpath)
313{
314 fib_path_ext_t *path_ext;
315
316 path_ext = fib_path_ext_list_find(list, ext_type, rpath);
317
318 if (NULL == path_ext)
319 {
320 vec_add2(list->fpel_exts, path_ext, 1);
321 fib_path_ext_init(path_ext, path_list_index, ext_type, rpath);
322 }
323
324 return (path_ext);
325}
326
327/*
328 * insert, sorted, a path extension to the entry's list.
329 * It's not strictly necessary to sort the path extensions, since each
330 * extension has the path index to which it resolves. However, by being
331 * sorted the load-balance produced has a deterministic order, not an order
332 * based on the sequence of extension additions. this is a considerable benefit.
333 */
334fib_path_ext_t *
335fib_path_ext_list_insert (fib_path_ext_list_t *list,
336 fib_node_index_t path_list_index,
337 fib_path_ext_type_t ext_type,
338 const fib_route_path_t *rpath)
339{
340 fib_path_ext_t new_path_ext, *path_ext;
341 int i = 0;
342
343 if (0 == fib_path_ext_list_length(list))
344 {
345 return (fib_path_ext_list_push_back(list, path_list_index,
346 ext_type, rpath));
347 }
348
349 fib_path_ext_init(&new_path_ext, path_list_index, ext_type, rpath);
350
351 vec_foreach(path_ext, list->fpel_exts)
352 {
Neale Rannsdcd6d622017-05-26 02:59:16 -0700353 int res = fib_path_ext_cmp(path_ext, rpath);
354
355 if (0 == res)
356 {
357 /*
358 * don't add duplicate extensions. modify instead
359 */
360 vec_free(path_ext->fpe_label_stack);
361 *path_ext = new_path_ext;
362 goto done;
363 }
364 else if (res < 0)
Neale Ranns81424992017-05-18 03:03:22 -0700365 {
366 i++;
367 }
368 else
369 {
370 break;
371 }
372 }
373 vec_insert_elts(list->fpel_exts, &new_path_ext, 1, i);
Neale Rannsdcd6d622017-05-26 02:59:16 -0700374done:
Neale Ranns81424992017-05-18 03:03:22 -0700375 return (&(list->fpel_exts[i]));
376}
377
378void
379fib_path_ext_list_resolve (fib_path_ext_list_t *list,
380 fib_node_index_t path_list_index)
381{
382 fib_path_ext_t *path_ext;
383
384 vec_foreach(path_ext, list->fpel_exts)
385 {
386 fib_path_ext_resolve(path_ext, path_list_index);
387 };
388}
389
390void
391fib_path_ext_list_remove (fib_path_ext_list_t *list,
392 fib_path_ext_type_t ext_type,
393 const fib_route_path_t *rpath)
394{
395 fib_path_ext_t *path_ext;
396
397 path_ext = fib_path_ext_list_find(list, ext_type, rpath);
398
399 if (NULL != path_ext)
400 {
401 /*
402 * delete the element moving the remaining elements down 1 position.
403 * this preserves the sorted order.
404 */
405 vec_free(path_ext->fpe_label_stack);
406 vec_delete(list->fpel_exts, 1, (path_ext - list->fpel_exts));
407 }
408}
409
410void
411fib_path_ext_list_flush (fib_path_ext_list_t *list)
412{
413 fib_path_ext_t *path_ext;
414
415 vec_foreach(path_ext, list->fpel_exts)
416 {
417 vec_free(path_ext->fpe_label_stack);
418 };
419 vec_free(list->fpel_exts);
420 list->fpel_exts = NULL;
421}
422
423u8*
424format_fib_path_ext_list (u8 * s, va_list * args)
425{
426 fib_path_ext_list_t *list;
427 fib_path_ext_t *path_ext;
428
429 list = va_arg (*args, fib_path_ext_list_t *);
430
431 if (fib_path_ext_list_length(list))
432 {
433 s = format(s, " Extensions:");
434 vec_foreach(path_ext, list->fpel_exts)
435 {
436 s = format(s, "\n %U", format_fib_path_ext, path_ext);
437 };
438 }
439
440 return (s);
441}
442
443int
444fib_path_ext_list_length (const fib_path_ext_list_t *list)
445{
446 return (vec_len(list->fpel_exts));
447}