blob: 504333a247445903cf846e7f5e59f10ff6347048 [file] [log] [blame]
Neale Ranns32e1c012016-11-22 17:07:28 +00001/*
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 <vlib/vlib.h>
17#include <vnet/dpo/drop_dpo.h>
18
19#include <vnet/mfib/mfib_table.h>
20#include <vnet/mfib/ip4_mfib.h>
21#include <vnet/mfib/ip6_mfib.h>
22#include <vnet/mfib/mfib_entry.h>
Neale Ranns9e829a82018-12-17 05:50:32 -080023#include <vnet/mfib/mfib_entry_src.h>
24#include <vnet/mfib/mfib_entry_cover.h>
Neale Ranns32e1c012016-11-22 17:07:28 +000025#include <vnet/mfib/mfib_signal.h>
26
27mfib_table_t *
28mfib_table_get (fib_node_index_t index,
29 fib_protocol_t proto)
30{
31 switch (proto)
32 {
33 case FIB_PROTOCOL_IP4:
34 return (pool_elt_at_index(ip4_main.mfibs, index));
35 case FIB_PROTOCOL_IP6:
36 return (pool_elt_at_index(ip6_main.mfibs, index));
37 case FIB_PROTOCOL_MPLS:
38 break;
39 }
40 ASSERT(0);
41 return (NULL);
42}
43
44static inline fib_node_index_t
45mfib_table_lookup_i (const mfib_table_t *mfib_table,
46 const mfib_prefix_t *prefix)
47{
48 switch (prefix->fp_proto)
49 {
50 case FIB_PROTOCOL_IP4:
51 return (ip4_mfib_table_lookup(&mfib_table->v4,
52 &prefix->fp_src_addr.ip4,
53 &prefix->fp_grp_addr.ip4,
54 prefix->fp_len));
55 case FIB_PROTOCOL_IP6:
56 return (ip6_mfib_table_lookup(&mfib_table->v6,
57 &prefix->fp_src_addr.ip6,
58 &prefix->fp_grp_addr.ip6,
59 prefix->fp_len));
60 case FIB_PROTOCOL_MPLS:
61 break;
62 }
63 return (FIB_NODE_INDEX_INVALID);
64}
65
66fib_node_index_t
67mfib_table_lookup (u32 fib_index,
68 const mfib_prefix_t *prefix)
69{
70 return (mfib_table_lookup_i(mfib_table_get(fib_index, prefix->fp_proto), prefix));
71}
72
73static inline fib_node_index_t
74mfib_table_lookup_exact_match_i (const mfib_table_t *mfib_table,
75 const mfib_prefix_t *prefix)
76{
77 switch (prefix->fp_proto)
78 {
79 case FIB_PROTOCOL_IP4:
80 return (ip4_mfib_table_lookup_exact_match(&mfib_table->v4,
81 &prefix->fp_grp_addr.ip4,
82 &prefix->fp_src_addr.ip4,
83 prefix->fp_len));
84 case FIB_PROTOCOL_IP6:
85 return (ip6_mfib_table_lookup_exact_match(&mfib_table->v6,
86 &prefix->fp_grp_addr.ip6,
87 &prefix->fp_src_addr.ip6,
88 prefix->fp_len));
89 case FIB_PROTOCOL_MPLS:
90 break;
91 }
92 return (FIB_NODE_INDEX_INVALID);
93}
94
95fib_node_index_t
96mfib_table_lookup_exact_match (u32 fib_index,
97 const mfib_prefix_t *prefix)
98{
99 return (mfib_table_lookup_exact_match_i(mfib_table_get(fib_index,
100 prefix->fp_proto),
101 prefix));
102}
103
Neale Ranns9e829a82018-12-17 05:50:32 -0800104static fib_node_index_t
105mfib_table_get_less_specific_i (const mfib_table_t *mfib_table,
106 const mfib_prefix_t *prefix)
107{
108 switch (prefix->fp_proto)
109 {
110 case FIB_PROTOCOL_IP4:
111 return (ip4_mfib_table_get_less_specific(&mfib_table->v4,
112 &prefix->fp_src_addr.ip4,
113 &prefix->fp_grp_addr.ip4,
114 prefix->fp_len));
115 case FIB_PROTOCOL_IP6:
116 return (ip6_mfib_table_get_less_specific(&mfib_table->v6,
117 &prefix->fp_src_addr.ip6,
118 &prefix->fp_grp_addr.ip6,
119 prefix->fp_len));
120 case FIB_PROTOCOL_MPLS:
121 break;
122 }
123 return (FIB_NODE_INDEX_INVALID);
124}
125
126fib_node_index_t
127mfib_table_get_less_specific (u32 fib_index,
128 const mfib_prefix_t *prefix)
129{
130 return (mfib_table_get_less_specific_i(mfib_table_get(fib_index,
131 prefix->fp_proto),
132 prefix));
133}
134
Neale Ranns32e1c012016-11-22 17:07:28 +0000135static void
136mfib_table_entry_remove (mfib_table_t *mfib_table,
137 const mfib_prefix_t *prefix,
Neale Ranns9e829a82018-12-17 05:50:32 -0800138 fib_node_index_t mfib_entry_index)
Neale Ranns32e1c012016-11-22 17:07:28 +0000139{
140 vlib_smp_unsafe_warning();
141
142 mfib_table->mft_total_route_counts--;
143
144 switch (prefix->fp_proto)
145 {
146 case FIB_PROTOCOL_IP4:
147 ip4_mfib_table_entry_remove(&mfib_table->v4,
148 &prefix->fp_grp_addr.ip4,
149 &prefix->fp_src_addr.ip4,
150 prefix->fp_len);
151 break;
152 case FIB_PROTOCOL_IP6:
153 ip6_mfib_table_entry_remove(&mfib_table->v6,
154 &prefix->fp_grp_addr.ip6,
155 &prefix->fp_src_addr.ip6,
156 prefix->fp_len);
157 break;
158 case FIB_PROTOCOL_MPLS:
159 ASSERT(0);
160 break;
161 }
162
Neale Ranns9e829a82018-12-17 05:50:32 -0800163 mfib_entry_cover_change_notify(mfib_entry_index,
164 FIB_NODE_INDEX_INVALID);
165 mfib_entry_unlock(mfib_entry_index);
Neale Ranns32e1c012016-11-22 17:07:28 +0000166}
167
168static void
Neale Ranns9e829a82018-12-17 05:50:32 -0800169mfib_table_post_insert_actions (mfib_table_t *mfib_table,
170 const mfib_prefix_t *prefix,
171 fib_node_index_t mfib_entry_index)
172{
173 fib_node_index_t mfib_entry_cover_index;
174
175 /*
176 * find the covering entry
177 */
178 mfib_entry_cover_index = mfib_table_get_less_specific_i(mfib_table,
179 prefix);
180 /*
181 * the indicies are the same when the default route is first added
182 */
183 if (mfib_entry_cover_index != mfib_entry_index)
184 {
185 /*
186 * inform the covering entry that a new more specific
187 * has been inserted beneath it.
188 * If the prefix that has been inserted is a host route
189 * then it is not possible that it will be the cover for any
190 * other entry, so we can elide the walk.
191 */
192 if (!mfib_entry_is_host(mfib_entry_index))
193 {
194 mfib_entry_cover_change_notify(mfib_entry_cover_index,
195 mfib_entry_index);
196 }
197 }
198}
199
200
201static void
Neale Ranns32e1c012016-11-22 17:07:28 +0000202mfib_table_entry_insert (mfib_table_t *mfib_table,
203 const mfib_prefix_t *prefix,
204 fib_node_index_t mfib_entry_index)
205{
206 vlib_smp_unsafe_warning();
207
208 mfib_entry_lock(mfib_entry_index);
209 mfib_table->mft_total_route_counts++;
210
211 switch (prefix->fp_proto)
212 {
213 case FIB_PROTOCOL_IP4:
214 ip4_mfib_table_entry_insert(&mfib_table->v4,
215 &prefix->fp_grp_addr.ip4,
216 &prefix->fp_src_addr.ip4,
217 prefix->fp_len,
218 mfib_entry_index);
219 break;
220 case FIB_PROTOCOL_IP6:
221 ip6_mfib_table_entry_insert(&mfib_table->v6,
222 &prefix->fp_grp_addr.ip6,
223 &prefix->fp_src_addr.ip6,
224 prefix->fp_len,
225 mfib_entry_index);
226 break;
227 case FIB_PROTOCOL_MPLS:
228 break;
229 }
Neale Ranns9e829a82018-12-17 05:50:32 -0800230
231 mfib_table_post_insert_actions(mfib_table, prefix, mfib_entry_index);
Neale Ranns32e1c012016-11-22 17:07:28 +0000232}
233
234fib_node_index_t
235mfib_table_entry_update (u32 fib_index,
236 const mfib_prefix_t *prefix,
237 mfib_source_t source,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800238 fib_rpf_id_t rpf_id,
Neale Ranns32e1c012016-11-22 17:07:28 +0000239 mfib_entry_flags_t entry_flags)
240{
241 fib_node_index_t mfib_entry_index;
242 mfib_table_t *mfib_table;
243
244 mfib_table = mfib_table_get(fib_index, prefix->fp_proto);
245 mfib_entry_index = mfib_table_lookup_exact_match_i(mfib_table, prefix);
246
247 if (FIB_NODE_INDEX_INVALID == mfib_entry_index)
248 {
249 if (MFIB_ENTRY_FLAG_NONE != entry_flags)
250 {
251 /*
252 * update to a non-existing entry with non-zero flags
253 */
254 mfib_entry_index = mfib_entry_create(fib_index, source,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800255 prefix, rpf_id,
Neale Ranns9e829a82018-12-17 05:50:32 -0800256 entry_flags,
257 INDEX_INVALID);
Neale Ranns32e1c012016-11-22 17:07:28 +0000258
259 mfib_table_entry_insert(mfib_table, prefix, mfib_entry_index);
260 }
261 /*
262 * else
263 * the entry doesn't exist and the request is to set no flags
264 * the result would be an entry that doesn't exist - so do nothing
265 */
266 }
267 else
268 {
269 mfib_entry_lock(mfib_entry_index);
270
Neale Rannsa9374df2017-02-02 02:18:18 -0800271 if (mfib_entry_update(mfib_entry_index,
272 source,
273 entry_flags,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800274 rpf_id,
Neale Rannsa9374df2017-02-02 02:18:18 -0800275 INDEX_INVALID))
Neale Ranns32e1c012016-11-22 17:07:28 +0000276 {
277 /*
278 * this update means we can now remove the entry.
279 */
280 mfib_table_entry_remove(mfib_table, prefix, mfib_entry_index);
281 }
282
283 mfib_entry_unlock(mfib_entry_index);
284 }
285
286 return (mfib_entry_index);
287}
288
Neale Ranns097fa662018-05-01 05:17:55 -0700289static fib_node_index_t
290mfib_table_entry_paths_update_i (u32 fib_index,
291 const mfib_prefix_t *prefix,
292 mfib_source_t source,
293 const fib_route_path_t *rpaths)
Neale Ranns32e1c012016-11-22 17:07:28 +0000294{
295 fib_node_index_t mfib_entry_index;
296 mfib_table_t *mfib_table;
297
298 mfib_table = mfib_table_get(fib_index, prefix->fp_proto);
299 mfib_entry_index = mfib_table_lookup_exact_match_i(mfib_table, prefix);
300
301 if (FIB_NODE_INDEX_INVALID == mfib_entry_index)
302 {
303 mfib_entry_index = mfib_entry_create(fib_index,
304 source,
305 prefix,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800306 MFIB_RPF_ID_NONE,
Neale Ranns9e829a82018-12-17 05:50:32 -0800307 MFIB_ENTRY_FLAG_NONE,
308 INDEX_INVALID);
309
Neale Ranns097fa662018-05-01 05:17:55 -0700310 mfib_entry_path_update(mfib_entry_index, source, rpaths);
Neale Ranns32e1c012016-11-22 17:07:28 +0000311
312 mfib_table_entry_insert(mfib_table, prefix, mfib_entry_index);
313 }
Neale Ranns9e829a82018-12-17 05:50:32 -0800314 else
315 {
Neale Ranns097fa662018-05-01 05:17:55 -0700316 mfib_entry_path_update(mfib_entry_index, source, rpaths);
Neale Ranns9e829a82018-12-17 05:50:32 -0800317 }
Neale Ranns32e1c012016-11-22 17:07:28 +0000318 return (mfib_entry_index);
319}
320
Neale Ranns097fa662018-05-01 05:17:55 -0700321
322fib_node_index_t
323mfib_table_entry_path_update (u32 fib_index,
Neale Ranns32e1c012016-11-22 17:07:28 +0000324 const mfib_prefix_t *prefix,
325 mfib_source_t source,
326 const fib_route_path_t *rpath)
327{
328 fib_node_index_t mfib_entry_index;
Neale Ranns097fa662018-05-01 05:17:55 -0700329 fib_route_path_t *rpaths = NULL;
330
331 vec_add1(rpaths, *rpath);
332
333 mfib_entry_index = mfib_table_entry_paths_update_i(fib_index, prefix,
334 source, rpaths);
335
336 vec_free(rpaths);
337 return (mfib_entry_index);
338}
339
340fib_node_index_t
341mfib_table_entry_paths_update (u32 fib_index,
342 const mfib_prefix_t *prefix,
343 mfib_source_t source,
344 const fib_route_path_t *rpaths)
345{
346 return (mfib_table_entry_paths_update_i(fib_index, prefix,
347 source, rpaths));
348}
349
350static void
351mfib_table_entry_paths_remove_i (u32 fib_index,
352 const mfib_prefix_t *prefix,
353 mfib_source_t source,
354 const fib_route_path_t *rpaths)
355{
356 fib_node_index_t mfib_entry_index;
Neale Ranns32e1c012016-11-22 17:07:28 +0000357 mfib_table_t *mfib_table;
358
359 mfib_table = mfib_table_get(fib_index, prefix->fp_proto);
360 mfib_entry_index = mfib_table_lookup_exact_match_i(mfib_table, prefix);
361
362 if (FIB_NODE_INDEX_INVALID == mfib_entry_index)
363 {
364 /*
Neale Ranns097fa662018-05-01 05:17:55 -0700365 * removing an entry that does not exist. i'll allow it.
Neale Ranns32e1c012016-11-22 17:07:28 +0000366 */
367 }
368 else
369 {
370 int no_more_sources;
371
372 /*
373 * don't nobody go nowhere
374 */
375 mfib_entry_lock(mfib_entry_index);
376
377 no_more_sources = mfib_entry_path_remove(mfib_entry_index,
378 source,
Neale Ranns097fa662018-05-01 05:17:55 -0700379 rpaths);
Neale Ranns32e1c012016-11-22 17:07:28 +0000380
381 if (no_more_sources)
382 {
383 /*
384 * last source gone. remove from the table
385 */
386 mfib_table_entry_remove(mfib_table, prefix, mfib_entry_index);
387 }
388
389 mfib_entry_unlock(mfib_entry_index);
390 }
391}
Neale Ranns097fa662018-05-01 05:17:55 -0700392void
393mfib_table_entry_paths_remove (u32 fib_index,
394 const mfib_prefix_t *prefix,
395 mfib_source_t source,
396 const fib_route_path_t *rpaths)
397{
398 mfib_table_entry_paths_remove_i(fib_index,
399 prefix,
400 source,
401 rpaths);
402}
403
404void
405mfib_table_entry_path_remove (u32 fib_index,
406 const mfib_prefix_t *prefix,
407 mfib_source_t source,
408 const fib_route_path_t *rpath)
409{
410 fib_route_path_t *rpaths = NULL;
411
412 vec_add1(rpaths, *rpath);
413
414 mfib_table_entry_paths_remove_i(fib_index,
415 prefix,
416 source,
417 rpaths);
418
419 vec_free(rpaths);
420}
Neale Ranns32e1c012016-11-22 17:07:28 +0000421
Neale Rannsa9374df2017-02-02 02:18:18 -0800422fib_node_index_t
423mfib_table_entry_special_add (u32 fib_index,
424 const mfib_prefix_t *prefix,
425 mfib_source_t source,
426 mfib_entry_flags_t entry_flags,
Neale Ranns9e829a82018-12-17 05:50:32 -0800427 index_t repi)
Neale Rannsa9374df2017-02-02 02:18:18 -0800428{
429 fib_node_index_t mfib_entry_index;
430 mfib_table_t *mfib_table;
431
432 mfib_table = mfib_table_get(fib_index, prefix->fp_proto);
433 mfib_entry_index = mfib_table_lookup_exact_match_i(mfib_table, prefix);
434
Neale Ranns9e829a82018-12-17 05:50:32 -0800435 if (INDEX_INVALID != repi)
436 {
437 entry_flags |= MFIB_ENTRY_FLAG_EXCLUSIVE;
438 }
439
Neale Rannsa9374df2017-02-02 02:18:18 -0800440 if (FIB_NODE_INDEX_INVALID == mfib_entry_index)
441 {
442 mfib_entry_index = mfib_entry_create(fib_index,
443 source,
444 prefix,
Neale Ranns0f26c5a2017-03-01 15:12:11 -0800445 MFIB_RPF_ID_NONE,
Neale Ranns9e829a82018-12-17 05:50:32 -0800446 entry_flags,
447 repi);
Neale Rannsa9374df2017-02-02 02:18:18 -0800448
449 mfib_table_entry_insert(mfib_table, prefix, mfib_entry_index);
450 }
Neale Ranns9e829a82018-12-17 05:50:32 -0800451 else
452 {
453 mfib_entry_special_add(mfib_entry_index, source, entry_flags,
454 MFIB_RPF_ID_NONE, repi);
455 }
Neale Rannsa9374df2017-02-02 02:18:18 -0800456
457 return (mfib_entry_index);
458}
459
Neale Ranns32e1c012016-11-22 17:07:28 +0000460static void
461mfib_table_entry_delete_i (u32 fib_index,
462 fib_node_index_t mfib_entry_index,
463 const mfib_prefix_t *prefix,
464 mfib_source_t source)
465{
466 mfib_table_t *mfib_table;
467
468 mfib_table = mfib_table_get(fib_index, prefix->fp_proto);
469
470 /*
471 * don't nobody go nowhere
472 */
473 mfib_entry_lock(mfib_entry_index);
474
475 if (mfib_entry_delete(mfib_entry_index, source))
476 {
477 /*
478 * last source gone. remove from the table
479 */
480 mfib_table_entry_remove(mfib_table, prefix, mfib_entry_index);
481 }
482 /*
483 * else
484 * still has sources, leave it be.
485 */
486
487 mfib_entry_unlock(mfib_entry_index);
488}
489
490void
491mfib_table_entry_delete (u32 fib_index,
492 const mfib_prefix_t *prefix,
493 mfib_source_t source)
494{
495 fib_node_index_t mfib_entry_index;
496
497 mfib_entry_index = mfib_table_lookup_exact_match(fib_index, prefix);
498
499 if (FIB_NODE_INDEX_INVALID == mfib_entry_index)
500 {
501 /*
502 * removing an etry that does not exist.
503 * i'll allow it, but i won't like it.
504 */
505 clib_warning("%U not in FIB", format_mfib_prefix, prefix);
506 }
507 else
508 {
509 mfib_table_entry_delete_i(fib_index, mfib_entry_index,
510 prefix, source);
511 }
512}
513
514void
515mfib_table_entry_delete_index (fib_node_index_t mfib_entry_index,
516 mfib_source_t source)
517{
Neale Ranns32e1c012016-11-22 17:07:28 +0000518 mfib_table_entry_delete_i(mfib_entry_get_fib_index(mfib_entry_index),
Neale Ranns097fa662018-05-01 05:17:55 -0700519 mfib_entry_index,
520 mfib_entry_get_prefix(mfib_entry_index),
521 source);
Neale Ranns32e1c012016-11-22 17:07:28 +0000522}
523
524u32
525mfib_table_get_index_for_sw_if_index (fib_protocol_t proto,
526 u32 sw_if_index)
527{
528 switch (proto)
529 {
530 case FIB_PROTOCOL_IP4:
531 return (ip4_mfib_table_get_index_for_sw_if_index(sw_if_index));
532 case FIB_PROTOCOL_IP6:
533 return (ip6_mfib_table_get_index_for_sw_if_index(sw_if_index));
534 case FIB_PROTOCOL_MPLS:
535 ASSERT(0);
536 break;
537 }
538 return (~0);
539}
540
541u32
Neale Ranns775f73c2018-12-20 03:01:49 -0800542mfib_table_get_table_id (u32 fib_index,
543 fib_protocol_t proto)
544{
545 mfib_table_t *mfib_table;
546
547 mfib_table = mfib_table_get(fib_index, proto);
548
549 return ((NULL != mfib_table ? mfib_table->mft_table_id : ~0));
550}
551
552u32
Neale Ranns32e1c012016-11-22 17:07:28 +0000553mfib_table_find (fib_protocol_t proto,
554 u32 table_id)
555{
556 switch (proto)
557 {
558 case FIB_PROTOCOL_IP4:
559 return (ip4_mfib_index_from_table_id(table_id));
560 case FIB_PROTOCOL_IP6:
561 return (ip6_mfib_index_from_table_id(table_id));
562 case FIB_PROTOCOL_MPLS:
563 ASSERT(0);
564 break;
565 }
566 return (~0);
567}
568
Neale Ranns2297af02017-09-12 09:45:04 -0700569static u32
570mfib_table_find_or_create_and_lock_i (fib_protocol_t proto,
571 u32 table_id,
572 mfib_source_t src,
573 const u8 *name)
Neale Ranns32e1c012016-11-22 17:07:28 +0000574{
575 mfib_table_t *mfib_table;
576 fib_node_index_t fi;
577
578 switch (proto)
579 {
580 case FIB_PROTOCOL_IP4:
Neale Ranns15002542017-09-10 04:39:11 -0700581 fi = ip4_mfib_table_find_or_create_and_lock(table_id, src);
Neale Ranns32e1c012016-11-22 17:07:28 +0000582 break;
583 case FIB_PROTOCOL_IP6:
Neale Ranns15002542017-09-10 04:39:11 -0700584 fi = ip6_mfib_table_find_or_create_and_lock(table_id, src);
Neale Ranns32e1c012016-11-22 17:07:28 +0000585 break;
586 case FIB_PROTOCOL_MPLS:
587 default:
588 return (~0);
589 }
590
591 mfib_table = mfib_table_get(fi, proto);
592
Neale Ranns2297af02017-09-12 09:45:04 -0700593 if (NULL == mfib_table->mft_desc)
594 {
595 if (name && name[0])
596 {
597 mfib_table->mft_desc = format(NULL, "%s", name);
598 }
599 else
600 {
601 mfib_table->mft_desc = format(NULL, "%U-VRF:%d",
602 format_fib_protocol, proto,
603 table_id);
604 }
605 }
Neale Ranns32e1c012016-11-22 17:07:28 +0000606
607 return (fi);
608}
609
Neale Ranns2297af02017-09-12 09:45:04 -0700610u32
611mfib_table_find_or_create_and_lock (fib_protocol_t proto,
612 u32 table_id,
613 mfib_source_t src)
614{
615 return (mfib_table_find_or_create_and_lock_i(proto, table_id,
616 src, NULL));
617}
618
619u32
620mfib_table_find_or_create_and_lock_w_name (fib_protocol_t proto,
621 u32 table_id,
622 mfib_source_t src,
623 const u8 *name)
624{
625 return (mfib_table_find_or_create_and_lock_i(proto, table_id,
626 src, name));
627}
628
Neale Ranns15002542017-09-10 04:39:11 -0700629/**
630 * @brief Table flush context. Store the indicies of matching FIB entries
631 * that need to be removed.
632 */
633typedef struct mfib_table_flush_ctx_t_
634{
635 /**
636 * The list of entries to flush
637 */
638 fib_node_index_t *mftf_entries;
639
640 /**
641 * The source we are flushing
642 */
643 mfib_source_t mftf_source;
644} mfib_table_flush_ctx_t;
645
646static int
647mfib_table_flush_cb (fib_node_index_t mfib_entry_index,
648 void *arg)
649{
650 mfib_table_flush_ctx_t *ctx = arg;
651
652 if (mfib_entry_is_sourced(mfib_entry_index, ctx->mftf_source))
653 {
654 vec_add1(ctx->mftf_entries, mfib_entry_index);
655 }
656 return (1);
657}
658
659void
660mfib_table_flush (u32 mfib_index,
661 fib_protocol_t proto,
662 mfib_source_t source)
663{
664 fib_node_index_t *mfib_entry_index;
665 mfib_table_flush_ctx_t ctx = {
666 .mftf_entries = NULL,
667 .mftf_source = source,
668 };
669
670 mfib_table_walk(mfib_index, proto,
671 mfib_table_flush_cb,
672 &ctx);
673
674 vec_foreach(mfib_entry_index, ctx.mftf_entries)
675 {
676 mfib_table_entry_delete_index(*mfib_entry_index, source);
677 }
678
679 vec_free(ctx.mftf_entries);
680}
681
Neale Ranns32e1c012016-11-22 17:07:28 +0000682static void
683mfib_table_destroy (mfib_table_t *mfib_table)
684{
685 vec_free(mfib_table->mft_desc);
686
687 switch (mfib_table->mft_proto)
688 {
689 case FIB_PROTOCOL_IP4:
690 ip4_mfib_table_destroy(&mfib_table->v4);
691 break;
692 case FIB_PROTOCOL_IP6:
693 ip6_mfib_table_destroy(&mfib_table->v6);
694 break;
695 case FIB_PROTOCOL_MPLS:
696 ASSERT(0);
697 break;
698 }
699}
700
701void
702mfib_table_unlock (u32 fib_index,
Neale Ranns15002542017-09-10 04:39:11 -0700703 fib_protocol_t proto,
704 mfib_source_t source)
Neale Ranns32e1c012016-11-22 17:07:28 +0000705{
706 mfib_table_t *mfib_table;
707
708 mfib_table = mfib_table_get(fib_index, proto);
Neale Ranns15002542017-09-10 04:39:11 -0700709 mfib_table->mft_locks[source]--;
710 mfib_table->mft_locks[MFIB_TABLE_TOTAL_LOCKS]--;
Neale Ranns32e1c012016-11-22 17:07:28 +0000711
Neale Ranns15002542017-09-10 04:39:11 -0700712 if (0 == mfib_table->mft_locks[source])
Neale Ranns32e1c012016-11-22 17:07:28 +0000713 {
Neale Ranns15002542017-09-10 04:39:11 -0700714 /*
715 * The source no longer needs the table. flush any routes
716 * from it just in case
717 */
718 mfib_table_flush(fib_index, proto, source);
719 }
720
721 if (0 == mfib_table->mft_locks[MFIB_TABLE_TOTAL_LOCKS])
722 {
723 /*
724 * no more locak from any source - kill it
725 */
726 mfib_table_destroy(mfib_table);
Neale Ranns32e1c012016-11-22 17:07:28 +0000727 }
728}
729
730void
731mfib_table_lock (u32 fib_index,
Neale Ranns15002542017-09-10 04:39:11 -0700732 fib_protocol_t proto,
733 mfib_source_t source)
Neale Ranns32e1c012016-11-22 17:07:28 +0000734{
735 mfib_table_t *mfib_table;
736
737 mfib_table = mfib_table_get(fib_index, proto);
Neale Ranns15002542017-09-10 04:39:11 -0700738 mfib_table->mft_locks[source]++;
739 mfib_table->mft_locks[MFIB_TABLE_TOTAL_LOCKS]++;
Neale Ranns32e1c012016-11-22 17:07:28 +0000740}
741
Neale Ranns9e829a82018-12-17 05:50:32 -0800742u32
743mfib_table_get_n_routes (fib_node_index_t fib_index,
744 fib_protocol_t proto)
745{
746 mfib_table_t *mfib_table;
747
748 mfib_table = mfib_table_get(fib_index, proto);
749
750 return (mfib_table->mft_total_route_counts);
751}
752
Neale Ranns5a8123b2017-01-26 01:18:23 -0800753void
754mfib_table_walk (u32 fib_index,
755 fib_protocol_t proto,
756 mfib_table_walk_fn_t fn,
757 void *ctx)
758{
759 switch (proto)
760 {
761 case FIB_PROTOCOL_IP4:
762 ip4_mfib_table_walk(ip4_mfib_get(fib_index), fn, ctx);
763 break;
764 case FIB_PROTOCOL_IP6:
765 ip6_mfib_table_walk(ip6_mfib_get(fib_index), fn, ctx);
766 break;
767 case FIB_PROTOCOL_MPLS:
768 break;
769 }
770}
771
Neale Ranns32e1c012016-11-22 17:07:28 +0000772u8*
Christophe Fontained3c008d2017-10-02 18:10:54 +0200773format_mfib_table_name (u8* s, va_list *ap)
Neale Ranns32e1c012016-11-22 17:07:28 +0000774{
Christophe Fontained3c008d2017-10-02 18:10:54 +0200775 fib_node_index_t fib_index = va_arg(*ap, fib_node_index_t);
776 fib_protocol_t proto = va_arg(*ap, int); // int promotion
Neale Ranns32e1c012016-11-22 17:07:28 +0000777 mfib_table_t *mfib_table;
778
779 mfib_table = mfib_table_get(fib_index, proto);
780
781 s = format(s, "%v", mfib_table->mft_desc);
782
783 return (s);
784}
785
Neale Rannsc87aafa2017-11-29 00:59:31 -0800786u8 *
787format_mfib_table_memory (u8 *s, va_list *args)
788{
789 s = format(s, "%U", format_ip4_mfib_table_memory);
790 s = format(s, "%U", format_ip6_mfib_table_memory);
791
792 return (s);
793}
794
Neale Ranns32e1c012016-11-22 17:07:28 +0000795static clib_error_t *
796mfib_module_init (vlib_main_t * vm)
797{
798 clib_error_t * error;
799
Neale Ranns9e829a82018-12-17 05:50:32 -0800800 mfib_entry_src_module_init();
Neale Ranns1ec36522017-11-29 05:20:37 -0800801 mfib_entry_module_init();
802 mfib_signal_module_init();
803
Neale Ranns32e1c012016-11-22 17:07:28 +0000804 if ((error = vlib_call_init_function (vm, fib_module_init)))
805 return (error);
806 if ((error = vlib_call_init_function (vm, rn_module_init)))
807 return (error);
808
Neale Ranns32e1c012016-11-22 17:07:28 +0000809 return (error);
810}
811
812VLIB_INIT_FUNCTION(mfib_module_init);