2 * Copyright (c) 2018 Cumulus Networks. All rights reserved.
5 * This software is licensed under the GNU General License Version 2,
6 * June 1991 as shown in the file COPYING in the top-level directory of this
9 * THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS"
10 * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
11 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
12 * FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE
13 * OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME
14 * THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
17 #include <linux/bitmap.h>
18 #include <linux/in6.h>
19 #include <linux/kernel.h>
20 #include <linux/list.h>
21 #include <linux/rhashtable.h>
22 #include <linux/spinlock_types.h>
23 #include <linux/types.h>
24 #include <net/fib_notifier.h>
25 #include <net/ip_fib.h>
26 #include <net/ip6_fib.h>
27 #include <net/fib_rules.h>
28 #include <net/net_namespace.h>
29 #include <net/nexthop.h>
30 #include <linux/debugfs.h>
32 #include "netdevsim.h"
34 struct nsim_fib_entry {
39 struct nsim_per_fib_data {
40 struct nsim_fib_entry fib;
41 struct nsim_fib_entry rules;
44 struct nsim_fib_data {
45 struct notifier_block fib_nb;
46 struct nsim_per_fib_data ipv4;
47 struct nsim_per_fib_data ipv6;
48 struct nsim_fib_entry nexthops;
49 struct rhashtable fib_rt_ht;
50 struct list_head fib_rt_list;
51 struct mutex fib_lock; /* Protects FIB HT and list */
52 struct notifier_block nexthop_nb;
53 struct rhashtable nexthop_ht;
54 struct devlink *devlink;
55 struct work_struct fib_event_work;
56 struct list_head fib_event_queue;
57 spinlock_t fib_event_queue_lock; /* Protects fib event queue list */
58 struct mutex nh_lock; /* Protects NH HT */
60 bool fail_route_offload;
61 bool fail_res_nexthop_group_replace;
62 bool fail_nexthop_bucket_replace;
65 struct nsim_fib_rt_key {
66 unsigned char addr[sizeof(struct in6_addr)];
67 unsigned char prefix_len;
73 struct nsim_fib_rt_key key;
74 struct rhash_head ht_node;
75 struct list_head list; /* Member of fib_rt_list */
79 struct nsim_fib_rt common;
86 struct nsim_fib_rt common;
87 struct list_head nh_list;
91 struct nsim_fib6_rt_nh {
92 struct list_head list; /* Member of nh_list */
96 struct nsim_fib6_event {
97 struct fib6_info **rt_arr;
101 struct nsim_fib_event {
102 struct list_head list; /* node in fib queue */
104 struct fib_entry_notifier_info fen_info;
105 struct nsim_fib6_event fib6_event;
107 struct nsim_fib_data *data;
112 static const struct rhashtable_params nsim_fib_rt_ht_params = {
113 .key_offset = offsetof(struct nsim_fib_rt, key),
114 .head_offset = offsetof(struct nsim_fib_rt, ht_node),
115 .key_len = sizeof(struct nsim_fib_rt_key),
116 .automatic_shrinking = true,
119 struct nsim_nexthop {
120 struct rhash_head ht_node;
126 static const struct rhashtable_params nsim_nexthop_ht_params = {
127 .key_offset = offsetof(struct nsim_nexthop, id),
128 .head_offset = offsetof(struct nsim_nexthop, ht_node),
129 .key_len = sizeof(u32),
130 .automatic_shrinking = true,
133 u64 nsim_fib_get_val(struct nsim_fib_data *fib_data,
134 enum nsim_resource_id res_id, bool max)
136 struct nsim_fib_entry *entry;
139 case NSIM_RESOURCE_IPV4_FIB:
140 entry = &fib_data->ipv4.fib;
142 case NSIM_RESOURCE_IPV4_FIB_RULES:
143 entry = &fib_data->ipv4.rules;
145 case NSIM_RESOURCE_IPV6_FIB:
146 entry = &fib_data->ipv6.fib;
148 case NSIM_RESOURCE_IPV6_FIB_RULES:
149 entry = &fib_data->ipv6.rules;
151 case NSIM_RESOURCE_NEXTHOPS:
152 entry = &fib_data->nexthops;
158 return max ? entry->max : atomic64_read(&entry->num);
161 static void nsim_fib_set_max(struct nsim_fib_data *fib_data,
162 enum nsim_resource_id res_id, u64 val)
164 struct nsim_fib_entry *entry;
167 case NSIM_RESOURCE_IPV4_FIB:
168 entry = &fib_data->ipv4.fib;
170 case NSIM_RESOURCE_IPV4_FIB_RULES:
171 entry = &fib_data->ipv4.rules;
173 case NSIM_RESOURCE_IPV6_FIB:
174 entry = &fib_data->ipv6.fib;
176 case NSIM_RESOURCE_IPV6_FIB_RULES:
177 entry = &fib_data->ipv6.rules;
179 case NSIM_RESOURCE_NEXTHOPS:
180 entry = &fib_data->nexthops;
189 static int nsim_fib_rule_account(struct nsim_fib_entry *entry, bool add,
190 struct netlink_ext_ack *extack)
195 if (!atomic64_add_unless(&entry->num, 1, entry->max)) {
197 NL_SET_ERR_MSG_MOD(extack, "Exceeded number of supported fib rule entries");
200 atomic64_dec_if_positive(&entry->num);
206 static int nsim_fib_rule_event(struct nsim_fib_data *data,
207 struct fib_notifier_info *info, bool add)
209 struct netlink_ext_ack *extack = info->extack;
212 switch (info->family) {
214 err = nsim_fib_rule_account(&data->ipv4.rules, add, extack);
217 err = nsim_fib_rule_account(&data->ipv6.rules, add, extack);
224 static int nsim_fib_account(struct nsim_fib_entry *entry, bool add)
229 if (!atomic64_add_unless(&entry->num, 1, entry->max))
232 atomic64_dec_if_positive(&entry->num);
238 static void nsim_fib_rt_init(struct nsim_fib_data *data,
239 struct nsim_fib_rt *fib_rt, const void *addr,
240 size_t addr_len, unsigned int prefix_len,
241 int family, u32 tb_id)
243 memcpy(fib_rt->key.addr, addr, addr_len);
244 fib_rt->key.prefix_len = prefix_len;
245 fib_rt->key.family = family;
246 fib_rt->key.tb_id = tb_id;
247 list_add(&fib_rt->list, &data->fib_rt_list);
250 static void nsim_fib_rt_fini(struct nsim_fib_rt *fib_rt)
252 list_del(&fib_rt->list);
255 static struct nsim_fib_rt *nsim_fib_rt_lookup(struct rhashtable *fib_rt_ht,
256 const void *addr, size_t addr_len,
257 unsigned int prefix_len,
258 int family, u32 tb_id)
260 struct nsim_fib_rt_key key;
262 memset(&key, 0, sizeof(key));
263 memcpy(key.addr, addr, addr_len);
264 key.prefix_len = prefix_len;
268 return rhashtable_lookup_fast(fib_rt_ht, &key, nsim_fib_rt_ht_params);
271 static struct nsim_fib4_rt *
272 nsim_fib4_rt_create(struct nsim_fib_data *data,
273 struct fib_entry_notifier_info *fen_info)
275 struct nsim_fib4_rt *fib4_rt;
277 fib4_rt = kzalloc(sizeof(*fib4_rt), GFP_KERNEL);
281 nsim_fib_rt_init(data, &fib4_rt->common, &fen_info->dst, sizeof(u32),
282 fen_info->dst_len, AF_INET, fen_info->tb_id);
284 fib4_rt->fi = fen_info->fi;
285 fib_info_hold(fib4_rt->fi);
286 fib4_rt->tos = fen_info->tos;
287 fib4_rt->type = fen_info->type;
292 static void nsim_fib4_rt_destroy(struct nsim_fib4_rt *fib4_rt)
294 fib_info_put(fib4_rt->fi);
295 nsim_fib_rt_fini(&fib4_rt->common);
299 static struct nsim_fib4_rt *
300 nsim_fib4_rt_lookup(struct rhashtable *fib_rt_ht,
301 const struct fib_entry_notifier_info *fen_info)
303 struct nsim_fib_rt *fib_rt;
305 fib_rt = nsim_fib_rt_lookup(fib_rt_ht, &fen_info->dst, sizeof(u32),
306 fen_info->dst_len, AF_INET,
311 return container_of(fib_rt, struct nsim_fib4_rt, common);
315 nsim_fib4_rt_offload_failed_flag_set(struct net *net,
316 struct fib_entry_notifier_info *fen_info)
318 u32 *p_dst = (u32 *)&fen_info->dst;
319 struct fib_rt_info fri;
321 fri.fi = fen_info->fi;
322 fri.tb_id = fen_info->tb_id;
323 fri.dst = cpu_to_be32(*p_dst);
324 fri.dst_len = fen_info->dst_len;
325 fri.tos = fen_info->tos;
326 fri.type = fen_info->type;
329 fri.offload_failed = true;
330 fib_alias_hw_flags_set(net, &fri);
333 static void nsim_fib4_rt_hw_flags_set(struct net *net,
334 const struct nsim_fib4_rt *fib4_rt,
337 u32 *p_dst = (u32 *) fib4_rt->common.key.addr;
338 int dst_len = fib4_rt->common.key.prefix_len;
339 struct fib_rt_info fri;
341 fri.fi = fib4_rt->fi;
342 fri.tb_id = fib4_rt->common.key.tb_id;
343 fri.dst = cpu_to_be32(*p_dst);
344 fri.dst_len = dst_len;
345 fri.tos = fib4_rt->tos;
346 fri.type = fib4_rt->type;
349 fri.offload_failed = false;
350 fib_alias_hw_flags_set(net, &fri);
353 static int nsim_fib4_rt_add(struct nsim_fib_data *data,
354 struct nsim_fib4_rt *fib4_rt)
356 struct net *net = devlink_net(data->devlink);
359 err = rhashtable_insert_fast(&data->fib_rt_ht,
360 &fib4_rt->common.ht_node,
361 nsim_fib_rt_ht_params);
363 goto err_fib_dismiss;
365 /* Simulate hardware programming latency. */
367 nsim_fib4_rt_hw_flags_set(net, fib4_rt, true);
372 /* Drop the accounting that was increased from the notification
373 * context when FIB_EVENT_ENTRY_REPLACE was triggered.
375 nsim_fib_account(&data->ipv4.fib, false);
379 static int nsim_fib4_rt_replace(struct nsim_fib_data *data,
380 struct nsim_fib4_rt *fib4_rt,
381 struct nsim_fib4_rt *fib4_rt_old)
383 struct net *net = devlink_net(data->devlink);
386 /* We are replacing a route, so need to remove the accounting which
387 * was increased when FIB_EVENT_ENTRY_REPLACE was triggered.
389 err = nsim_fib_account(&data->ipv4.fib, false);
392 err = rhashtable_replace_fast(&data->fib_rt_ht,
393 &fib4_rt_old->common.ht_node,
394 &fib4_rt->common.ht_node,
395 nsim_fib_rt_ht_params);
400 nsim_fib4_rt_hw_flags_set(net, fib4_rt, true);
402 nsim_fib4_rt_hw_flags_set(net, fib4_rt_old, false);
403 nsim_fib4_rt_destroy(fib4_rt_old);
408 static int nsim_fib4_rt_insert(struct nsim_fib_data *data,
409 struct fib_entry_notifier_info *fen_info)
411 struct nsim_fib4_rt *fib4_rt, *fib4_rt_old;
414 if (data->fail_route_offload) {
415 /* For testing purposes, user set debugfs fail_route_offload
416 * value to true. Simulate hardware programming latency and then
423 fib4_rt = nsim_fib4_rt_create(data, fen_info);
427 fib4_rt_old = nsim_fib4_rt_lookup(&data->fib_rt_ht, fen_info);
429 err = nsim_fib4_rt_add(data, fib4_rt);
431 err = nsim_fib4_rt_replace(data, fib4_rt, fib4_rt_old);
434 nsim_fib4_rt_destroy(fib4_rt);
439 static void nsim_fib4_rt_remove(struct nsim_fib_data *data,
440 const struct fib_entry_notifier_info *fen_info)
442 struct nsim_fib4_rt *fib4_rt;
444 fib4_rt = nsim_fib4_rt_lookup(&data->fib_rt_ht, fen_info);
448 rhashtable_remove_fast(&data->fib_rt_ht, &fib4_rt->common.ht_node,
449 nsim_fib_rt_ht_params);
450 nsim_fib4_rt_destroy(fib4_rt);
453 static int nsim_fib4_event(struct nsim_fib_data *data,
454 struct fib_entry_notifier_info *fen_info,
460 case FIB_EVENT_ENTRY_REPLACE:
461 err = nsim_fib4_rt_insert(data, fen_info);
463 struct net *net = devlink_net(data->devlink);
465 nsim_fib4_rt_offload_failed_flag_set(net, fen_info);
468 case FIB_EVENT_ENTRY_DEL:
469 nsim_fib4_rt_remove(data, fen_info);
478 static struct nsim_fib6_rt_nh *
479 nsim_fib6_rt_nh_find(const struct nsim_fib6_rt *fib6_rt,
480 const struct fib6_info *rt)
482 struct nsim_fib6_rt_nh *fib6_rt_nh;
484 list_for_each_entry(fib6_rt_nh, &fib6_rt->nh_list, list) {
485 if (fib6_rt_nh->rt == rt)
492 static int nsim_fib6_rt_nh_add(struct nsim_fib6_rt *fib6_rt,
493 struct fib6_info *rt)
495 struct nsim_fib6_rt_nh *fib6_rt_nh;
497 fib6_rt_nh = kzalloc(sizeof(*fib6_rt_nh), GFP_KERNEL);
503 list_add_tail(&fib6_rt_nh->list, &fib6_rt->nh_list);
509 #if IS_ENABLED(CONFIG_IPV6)
510 static void nsim_rt6_release(struct fib6_info *rt)
512 fib6_info_release(rt);
515 static void nsim_rt6_release(struct fib6_info *rt)
520 static void nsim_fib6_rt_nh_del(struct nsim_fib6_rt *fib6_rt,
521 const struct fib6_info *rt)
523 struct nsim_fib6_rt_nh *fib6_rt_nh;
525 fib6_rt_nh = nsim_fib6_rt_nh_find(fib6_rt, rt);
530 list_del(&fib6_rt_nh->list);
531 nsim_rt6_release(fib6_rt_nh->rt);
535 static struct nsim_fib6_rt *
536 nsim_fib6_rt_create(struct nsim_fib_data *data,
537 struct fib6_info **rt_arr, unsigned int nrt6)
539 struct fib6_info *rt = rt_arr[0];
540 struct nsim_fib6_rt *fib6_rt;
544 fib6_rt = kzalloc(sizeof(*fib6_rt), GFP_KERNEL);
546 return ERR_PTR(-ENOMEM);
548 nsim_fib_rt_init(data, &fib6_rt->common, &rt->fib6_dst.addr,
549 sizeof(rt->fib6_dst.addr), rt->fib6_dst.plen, AF_INET6,
550 rt->fib6_table->tb6_id);
552 /* We consider a multipath IPv6 route as one entry, but it can be made
553 * up from several fib6_info structs (one for each nexthop), so we
554 * add them all to the same list under the entry.
556 INIT_LIST_HEAD(&fib6_rt->nh_list);
558 for (i = 0; i < nrt6; i++) {
559 err = nsim_fib6_rt_nh_add(fib6_rt, rt_arr[i]);
561 goto err_fib6_rt_nh_del;
567 for (i--; i >= 0; i--) {
568 nsim_fib6_rt_nh_del(fib6_rt, rt_arr[i]);
570 nsim_fib_rt_fini(&fib6_rt->common);
575 static void nsim_fib6_rt_destroy(struct nsim_fib6_rt *fib6_rt)
577 struct nsim_fib6_rt_nh *iter, *tmp;
579 list_for_each_entry_safe(iter, tmp, &fib6_rt->nh_list, list)
580 nsim_fib6_rt_nh_del(fib6_rt, iter->rt);
581 WARN_ON_ONCE(!list_empty(&fib6_rt->nh_list));
582 nsim_fib_rt_fini(&fib6_rt->common);
586 static struct nsim_fib6_rt *
587 nsim_fib6_rt_lookup(struct rhashtable *fib_rt_ht, const struct fib6_info *rt)
589 struct nsim_fib_rt *fib_rt;
591 fib_rt = nsim_fib_rt_lookup(fib_rt_ht, &rt->fib6_dst.addr,
592 sizeof(rt->fib6_dst.addr),
593 rt->fib6_dst.plen, AF_INET6,
594 rt->fib6_table->tb6_id);
598 return container_of(fib_rt, struct nsim_fib6_rt, common);
601 static int nsim_fib6_rt_append(struct nsim_fib_data *data,
602 struct nsim_fib6_event *fib6_event)
604 struct fib6_info *rt = fib6_event->rt_arr[0];
605 struct nsim_fib6_rt *fib6_rt;
608 if (data->fail_route_offload) {
609 /* For testing purposes, user set debugfs fail_route_offload
610 * value to true. Simulate hardware programming latency and then
617 fib6_rt = nsim_fib6_rt_lookup(&data->fib_rt_ht, rt);
621 for (i = 0; i < fib6_event->nrt6; i++) {
622 err = nsim_fib6_rt_nh_add(fib6_rt, fib6_event->rt_arr[i]);
624 goto err_fib6_rt_nh_del;
626 fib6_event->rt_arr[i]->trap = true;
632 for (i--; i >= 0; i--) {
633 fib6_event->rt_arr[i]->trap = false;
634 nsim_fib6_rt_nh_del(fib6_rt, fib6_event->rt_arr[i]);
639 #if IS_ENABLED(CONFIG_IPV6)
640 static void nsim_fib6_rt_offload_failed_flag_set(struct nsim_fib_data *data,
641 struct fib6_info **rt_arr,
645 struct net *net = devlink_net(data->devlink);
648 for (i = 0; i < nrt6; i++)
649 fib6_info_hw_flags_set(net, rt_arr[i], false, false, true);
652 static void nsim_fib6_rt_offload_failed_flag_set(struct nsim_fib_data *data,
653 struct fib6_info **rt_arr,
659 #if IS_ENABLED(CONFIG_IPV6)
660 static void nsim_fib6_rt_hw_flags_set(struct nsim_fib_data *data,
661 const struct nsim_fib6_rt *fib6_rt,
664 struct net *net = devlink_net(data->devlink);
665 struct nsim_fib6_rt_nh *fib6_rt_nh;
667 list_for_each_entry(fib6_rt_nh, &fib6_rt->nh_list, list)
668 fib6_info_hw_flags_set(net, fib6_rt_nh->rt, false, trap, false);
671 static void nsim_fib6_rt_hw_flags_set(struct nsim_fib_data *data,
672 const struct nsim_fib6_rt *fib6_rt,
678 static int nsim_fib6_rt_add(struct nsim_fib_data *data,
679 struct nsim_fib6_rt *fib6_rt)
683 err = rhashtable_insert_fast(&data->fib_rt_ht,
684 &fib6_rt->common.ht_node,
685 nsim_fib_rt_ht_params);
688 goto err_fib_dismiss;
691 nsim_fib6_rt_hw_flags_set(data, fib6_rt, true);
696 /* Drop the accounting that was increased from the notification
697 * context when FIB_EVENT_ENTRY_REPLACE was triggered.
699 nsim_fib_account(&data->ipv6.fib, false);
703 static int nsim_fib6_rt_replace(struct nsim_fib_data *data,
704 struct nsim_fib6_rt *fib6_rt,
705 struct nsim_fib6_rt *fib6_rt_old)
709 /* We are replacing a route, so need to remove the accounting which
710 * was increased when FIB_EVENT_ENTRY_REPLACE was triggered.
712 err = nsim_fib_account(&data->ipv6.fib, false);
716 err = rhashtable_replace_fast(&data->fib_rt_ht,
717 &fib6_rt_old->common.ht_node,
718 &fib6_rt->common.ht_node,
719 nsim_fib_rt_ht_params);
725 nsim_fib6_rt_hw_flags_set(data, fib6_rt, true);
727 nsim_fib6_rt_hw_flags_set(data, fib6_rt_old, false);
728 nsim_fib6_rt_destroy(fib6_rt_old);
733 static int nsim_fib6_rt_insert(struct nsim_fib_data *data,
734 struct nsim_fib6_event *fib6_event)
736 struct fib6_info *rt = fib6_event->rt_arr[0];
737 struct nsim_fib6_rt *fib6_rt, *fib6_rt_old;
740 if (data->fail_route_offload) {
741 /* For testing purposes, user set debugfs fail_route_offload
742 * value to true. Simulate hardware programming latency and then
749 fib6_rt = nsim_fib6_rt_create(data, fib6_event->rt_arr,
752 return PTR_ERR(fib6_rt);
754 fib6_rt_old = nsim_fib6_rt_lookup(&data->fib_rt_ht, rt);
756 err = nsim_fib6_rt_add(data, fib6_rt);
758 err = nsim_fib6_rt_replace(data, fib6_rt, fib6_rt_old);
761 nsim_fib6_rt_destroy(fib6_rt);
766 static void nsim_fib6_rt_remove(struct nsim_fib_data *data,
767 struct nsim_fib6_event *fib6_event)
769 struct fib6_info *rt = fib6_event->rt_arr[0];
770 struct nsim_fib6_rt *fib6_rt;
773 /* Multipath routes are first added to the FIB trie and only then
774 * notified. If we vetoed the addition, we will get a delete
775 * notification for a route we do not have. Therefore, do not warn if
776 * route was not found.
778 fib6_rt = nsim_fib6_rt_lookup(&data->fib_rt_ht, rt);
782 /* If not all the nexthops are deleted, then only reduce the nexthop
785 if (fib6_event->nrt6 != fib6_rt->nhs) {
786 for (i = 0; i < fib6_event->nrt6; i++)
787 nsim_fib6_rt_nh_del(fib6_rt, fib6_event->rt_arr[i]);
791 rhashtable_remove_fast(&data->fib_rt_ht, &fib6_rt->common.ht_node,
792 nsim_fib_rt_ht_params);
793 nsim_fib6_rt_destroy(fib6_rt);
796 static int nsim_fib6_event_init(struct nsim_fib6_event *fib6_event,
797 struct fib6_entry_notifier_info *fen6_info)
799 struct fib6_info *rt = fen6_info->rt;
800 struct fib6_info **rt_arr;
801 struct fib6_info *iter;
805 nrt6 = fen6_info->nsiblings + 1;
807 rt_arr = kcalloc(nrt6, sizeof(struct fib6_info *), GFP_ATOMIC);
811 fib6_event->rt_arr = rt_arr;
812 fib6_event->nrt6 = nrt6;
817 if (!fen6_info->nsiblings)
820 list_for_each_entry(iter, &rt->fib6_siblings, fib6_siblings) {
821 if (i == fen6_info->nsiblings)
824 rt_arr[i + 1] = iter;
825 fib6_info_hold(iter);
828 WARN_ON_ONCE(i != fen6_info->nsiblings);
833 static void nsim_fib6_event_fini(struct nsim_fib6_event *fib6_event)
837 for (i = 0; i < fib6_event->nrt6; i++)
838 nsim_rt6_release(fib6_event->rt_arr[i]);
839 kfree(fib6_event->rt_arr);
842 static int nsim_fib6_event(struct nsim_fib_data *data,
843 struct nsim_fib6_event *fib6_event,
848 if (fib6_event->rt_arr[0]->fib6_src.plen)
852 case FIB_EVENT_ENTRY_REPLACE:
853 err = nsim_fib6_rt_insert(data, fib6_event);
855 goto err_rt_offload_failed_flag_set;
857 case FIB_EVENT_ENTRY_APPEND:
858 err = nsim_fib6_rt_append(data, fib6_event);
860 goto err_rt_offload_failed_flag_set;
862 case FIB_EVENT_ENTRY_DEL:
863 nsim_fib6_rt_remove(data, fib6_event);
871 err_rt_offload_failed_flag_set:
872 nsim_fib6_rt_offload_failed_flag_set(data, fib6_event->rt_arr,
877 static void nsim_fib_event(struct nsim_fib_event *fib_event)
879 switch (fib_event->family) {
881 nsim_fib4_event(fib_event->data, &fib_event->fen_info,
883 fib_info_put(fib_event->fen_info.fi);
886 nsim_fib6_event(fib_event->data, &fib_event->fib6_event,
888 nsim_fib6_event_fini(&fib_event->fib6_event);
893 static int nsim_fib4_prepare_event(struct fib_notifier_info *info,
894 struct nsim_fib_event *fib_event,
897 struct nsim_fib_data *data = fib_event->data;
898 struct fib_entry_notifier_info *fen_info;
899 struct netlink_ext_ack *extack;
902 fen_info = container_of(info, struct fib_entry_notifier_info,
904 fib_event->fen_info = *fen_info;
905 extack = info->extack;
908 case FIB_EVENT_ENTRY_REPLACE:
909 err = nsim_fib_account(&data->ipv4.fib, true);
911 NL_SET_ERR_MSG_MOD(extack, "Exceeded number of supported fib entries");
915 case FIB_EVENT_ENTRY_DEL:
916 nsim_fib_account(&data->ipv4.fib, false);
920 /* Take reference on fib_info to prevent it from being
921 * freed while event is queued. Release it afterwards.
923 fib_info_hold(fib_event->fen_info.fi);
928 static int nsim_fib6_prepare_event(struct fib_notifier_info *info,
929 struct nsim_fib_event *fib_event,
932 struct nsim_fib_data *data = fib_event->data;
933 struct fib6_entry_notifier_info *fen6_info;
934 struct netlink_ext_ack *extack;
937 fen6_info = container_of(info, struct fib6_entry_notifier_info,
940 err = nsim_fib6_event_init(&fib_event->fib6_event, fen6_info);
944 extack = info->extack;
946 case FIB_EVENT_ENTRY_REPLACE:
947 err = nsim_fib_account(&data->ipv6.fib, true);
949 NL_SET_ERR_MSG_MOD(extack, "Exceeded number of supported fib entries");
950 goto err_fib6_event_fini;
953 case FIB_EVENT_ENTRY_DEL:
954 nsim_fib_account(&data->ipv6.fib, false);
961 nsim_fib6_event_fini(&fib_event->fib6_event);
965 static int nsim_fib_event_schedule_work(struct nsim_fib_data *data,
966 struct fib_notifier_info *info,
969 struct nsim_fib_event *fib_event;
972 if (info->family != AF_INET && info->family != AF_INET6)
973 /* netdevsim does not support 'RTNL_FAMILY_IP6MR' and
974 * 'RTNL_FAMILY_IPMR' and should ignore them.
978 fib_event = kzalloc(sizeof(*fib_event), GFP_ATOMIC);
982 fib_event->data = data;
983 fib_event->event = event;
984 fib_event->family = info->family;
986 switch (info->family) {
988 err = nsim_fib4_prepare_event(info, fib_event, event);
991 err = nsim_fib6_prepare_event(info, fib_event, event);
996 goto err_fib_prepare_event;
998 /* Enqueue the event and trigger the work */
999 spin_lock_bh(&data->fib_event_queue_lock);
1000 list_add_tail(&fib_event->list, &data->fib_event_queue);
1001 spin_unlock_bh(&data->fib_event_queue_lock);
1002 schedule_work(&data->fib_event_work);
1006 err_fib_prepare_event:
1011 static int nsim_fib_event_nb(struct notifier_block *nb, unsigned long event,
1014 struct nsim_fib_data *data = container_of(nb, struct nsim_fib_data,
1016 struct fib_notifier_info *info = ptr;
1020 case FIB_EVENT_RULE_ADD:
1021 case FIB_EVENT_RULE_DEL:
1022 err = nsim_fib_rule_event(data, info,
1023 event == FIB_EVENT_RULE_ADD);
1024 return notifier_from_errno(err);
1025 case FIB_EVENT_ENTRY_REPLACE:
1026 case FIB_EVENT_ENTRY_APPEND:
1027 case FIB_EVENT_ENTRY_DEL:
1028 return nsim_fib_event_schedule_work(data, info, event);
1034 static void nsim_fib4_rt_free(struct nsim_fib_rt *fib_rt,
1035 struct nsim_fib_data *data)
1037 struct devlink *devlink = data->devlink;
1038 struct nsim_fib4_rt *fib4_rt;
1040 fib4_rt = container_of(fib_rt, struct nsim_fib4_rt, common);
1041 nsim_fib4_rt_hw_flags_set(devlink_net(devlink), fib4_rt, false);
1042 nsim_fib_account(&data->ipv4.fib, false);
1043 nsim_fib4_rt_destroy(fib4_rt);
1046 static void nsim_fib6_rt_free(struct nsim_fib_rt *fib_rt,
1047 struct nsim_fib_data *data)
1049 struct nsim_fib6_rt *fib6_rt;
1051 fib6_rt = container_of(fib_rt, struct nsim_fib6_rt, common);
1052 nsim_fib6_rt_hw_flags_set(data, fib6_rt, false);
1053 nsim_fib_account(&data->ipv6.fib, false);
1054 nsim_fib6_rt_destroy(fib6_rt);
1057 static void nsim_fib_rt_free(void *ptr, void *arg)
1059 struct nsim_fib_rt *fib_rt = ptr;
1060 struct nsim_fib_data *data = arg;
1062 switch (fib_rt->key.family) {
1064 nsim_fib4_rt_free(fib_rt, data);
1067 nsim_fib6_rt_free(fib_rt, data);
1074 /* inconsistent dump, trying again */
1075 static void nsim_fib_dump_inconsistent(struct notifier_block *nb)
1077 struct nsim_fib_data *data = container_of(nb, struct nsim_fib_data,
1079 struct nsim_fib_rt *fib_rt, *fib_rt_tmp;
1081 /* Flush the work to make sure there is no race with notifications. */
1082 flush_work(&data->fib_event_work);
1084 /* The notifier block is still not registered, so we do not need to
1085 * take any locks here.
1087 list_for_each_entry_safe(fib_rt, fib_rt_tmp, &data->fib_rt_list, list) {
1088 rhashtable_remove_fast(&data->fib_rt_ht, &fib_rt->ht_node,
1089 nsim_fib_rt_ht_params);
1090 nsim_fib_rt_free(fib_rt, data);
1093 atomic64_set(&data->ipv4.rules.num, 0ULL);
1094 atomic64_set(&data->ipv6.rules.num, 0ULL);
1097 static struct nsim_nexthop *nsim_nexthop_create(struct nsim_fib_data *data,
1098 struct nh_notifier_info *info)
1100 struct nsim_nexthop *nexthop;
1104 nexthop = kzalloc(sizeof(*nexthop), GFP_KERNEL);
1106 return ERR_PTR(-ENOMEM);
1108 nexthop->id = info->id;
1110 /* Determine the number of nexthop entries the new nexthop will
1114 switch (info->type) {
1115 case NH_NOTIFIER_INFO_TYPE_SINGLE:
1118 case NH_NOTIFIER_INFO_TYPE_GRP:
1119 for (i = 0; i < info->nh_grp->num_nh; i++)
1120 occ += info->nh_grp->nh_entries[i].weight;
1122 case NH_NOTIFIER_INFO_TYPE_RES_TABLE:
1123 occ = info->nh_res_table->num_nh_buckets;
1124 nexthop->is_resilient = true;
1127 NL_SET_ERR_MSG_MOD(info->extack, "Unsupported nexthop type");
1129 return ERR_PTR(-EOPNOTSUPP);
1136 static void nsim_nexthop_destroy(struct nsim_nexthop *nexthop)
1141 static int nsim_nexthop_account(struct nsim_fib_data *data, u64 occ,
1142 bool add, struct netlink_ext_ack *extack)
1147 for (i = 0; i < occ; i++)
1148 if (!atomic64_add_unless(&data->nexthops.num, 1,
1149 data->nexthops.max)) {
1151 NL_SET_ERR_MSG_MOD(extack, "Exceeded number of supported nexthops");
1152 goto err_num_decrease;
1155 if (WARN_ON(occ > atomic64_read(&data->nexthops.num)))
1157 atomic64_sub(occ, &data->nexthops.num);
1163 atomic64_sub(i, &data->nexthops.num);
1168 static void nsim_nexthop_hw_flags_set(struct net *net,
1169 const struct nsim_nexthop *nexthop,
1174 nexthop_set_hw_flags(net, nexthop->id, false, trap);
1176 if (!nexthop->is_resilient)
1179 for (i = 0; i < nexthop->occ; i++)
1180 nexthop_bucket_set_hw_flags(net, nexthop->id, i, false, trap);
1183 static int nsim_nexthop_add(struct nsim_fib_data *data,
1184 struct nsim_nexthop *nexthop,
1185 struct netlink_ext_ack *extack)
1187 struct net *net = devlink_net(data->devlink);
1190 err = nsim_nexthop_account(data, nexthop->occ, true, extack);
1194 err = rhashtable_insert_fast(&data->nexthop_ht, &nexthop->ht_node,
1195 nsim_nexthop_ht_params);
1197 NL_SET_ERR_MSG_MOD(extack, "Failed to insert nexthop");
1198 goto err_nexthop_dismiss;
1201 nsim_nexthop_hw_flags_set(net, nexthop, true);
1205 err_nexthop_dismiss:
1206 nsim_nexthop_account(data, nexthop->occ, false, extack);
1210 static int nsim_nexthop_replace(struct nsim_fib_data *data,
1211 struct nsim_nexthop *nexthop,
1212 struct nsim_nexthop *nexthop_old,
1213 struct netlink_ext_ack *extack)
1215 struct net *net = devlink_net(data->devlink);
1218 err = nsim_nexthop_account(data, nexthop->occ, true, extack);
1222 err = rhashtable_replace_fast(&data->nexthop_ht,
1223 &nexthop_old->ht_node, &nexthop->ht_node,
1224 nsim_nexthop_ht_params);
1226 NL_SET_ERR_MSG_MOD(extack, "Failed to replace nexthop");
1227 goto err_nexthop_dismiss;
1230 nsim_nexthop_hw_flags_set(net, nexthop, true);
1231 nsim_nexthop_account(data, nexthop_old->occ, false, extack);
1232 nsim_nexthop_destroy(nexthop_old);
1236 err_nexthop_dismiss:
1237 nsim_nexthop_account(data, nexthop->occ, false, extack);
1241 static int nsim_nexthop_insert(struct nsim_fib_data *data,
1242 struct nh_notifier_info *info)
1244 struct nsim_nexthop *nexthop, *nexthop_old;
1247 nexthop = nsim_nexthop_create(data, info);
1248 if (IS_ERR(nexthop))
1249 return PTR_ERR(nexthop);
1251 nexthop_old = rhashtable_lookup_fast(&data->nexthop_ht, &info->id,
1252 nsim_nexthop_ht_params);
1254 err = nsim_nexthop_add(data, nexthop, info->extack);
1256 err = nsim_nexthop_replace(data, nexthop, nexthop_old,
1260 nsim_nexthop_destroy(nexthop);
1265 static void nsim_nexthop_remove(struct nsim_fib_data *data,
1266 struct nh_notifier_info *info)
1268 struct nsim_nexthop *nexthop;
1270 nexthop = rhashtable_lookup_fast(&data->nexthop_ht, &info->id,
1271 nsim_nexthop_ht_params);
1275 rhashtable_remove_fast(&data->nexthop_ht, &nexthop->ht_node,
1276 nsim_nexthop_ht_params);
1277 nsim_nexthop_account(data, nexthop->occ, false, info->extack);
1278 nsim_nexthop_destroy(nexthop);
1281 static int nsim_nexthop_res_table_pre_replace(struct nsim_fib_data *data,
1282 struct nh_notifier_info *info)
1284 if (data->fail_res_nexthop_group_replace) {
1285 NL_SET_ERR_MSG_MOD(info->extack, "Failed to replace a resilient nexthop group");
1292 static int nsim_nexthop_bucket_replace(struct nsim_fib_data *data,
1293 struct nh_notifier_info *info)
1295 if (data->fail_nexthop_bucket_replace) {
1296 NL_SET_ERR_MSG_MOD(info->extack, "Failed to replace nexthop bucket");
1300 nexthop_bucket_set_hw_flags(info->net, info->id,
1301 info->nh_res_bucket->bucket_index,
1307 static int nsim_nexthop_event_nb(struct notifier_block *nb, unsigned long event,
1310 struct nsim_fib_data *data = container_of(nb, struct nsim_fib_data,
1312 struct nh_notifier_info *info = ptr;
1315 mutex_lock(&data->nh_lock);
1317 case NEXTHOP_EVENT_REPLACE:
1318 err = nsim_nexthop_insert(data, info);
1320 case NEXTHOP_EVENT_DEL:
1321 nsim_nexthop_remove(data, info);
1323 case NEXTHOP_EVENT_RES_TABLE_PRE_REPLACE:
1324 err = nsim_nexthop_res_table_pre_replace(data, info);
1326 case NEXTHOP_EVENT_BUCKET_REPLACE:
1327 err = nsim_nexthop_bucket_replace(data, info);
1333 mutex_unlock(&data->nh_lock);
1334 return notifier_from_errno(err);
1337 static void nsim_nexthop_free(void *ptr, void *arg)
1339 struct nsim_nexthop *nexthop = ptr;
1340 struct nsim_fib_data *data = arg;
1343 net = devlink_net(data->devlink);
1344 nsim_nexthop_hw_flags_set(net, nexthop, false);
1345 nsim_nexthop_account(data, nexthop->occ, false, NULL);
1346 nsim_nexthop_destroy(nexthop);
1349 static ssize_t nsim_nexthop_bucket_activity_write(struct file *file,
1350 const char __user *user_buf,
1351 size_t size, loff_t *ppos)
1353 struct nsim_fib_data *data = file->private_data;
1354 struct net *net = devlink_net(data->devlink);
1355 struct nsim_nexthop *nexthop;
1356 unsigned long *activity;
1365 if (size > sizeof(buf))
1367 if (copy_from_user(buf, user_buf, size))
1369 if (sscanf(buf, "%u %hu", &nhid, &bucket_index) != 2)
1374 nexthop = rhashtable_lookup_fast(&data->nexthop_ht, &nhid,
1375 nsim_nexthop_ht_params);
1376 if (!nexthop || !nexthop->is_resilient ||
1377 bucket_index >= nexthop->occ) {
1382 activity = bitmap_zalloc(nexthop->occ, GFP_KERNEL);
1388 bitmap_set(activity, bucket_index, 1);
1389 nexthop_res_grp_activity_update(net, nhid, nexthop->occ, activity);
1390 bitmap_free(activity);
1399 static const struct file_operations nsim_nexthop_bucket_activity_fops = {
1400 .open = simple_open,
1401 .write = nsim_nexthop_bucket_activity_write,
1402 .llseek = no_llseek,
1403 .owner = THIS_MODULE,
1406 static u64 nsim_fib_ipv4_resource_occ_get(void *priv)
1408 struct nsim_fib_data *data = priv;
1410 return nsim_fib_get_val(data, NSIM_RESOURCE_IPV4_FIB, false);
1413 static u64 nsim_fib_ipv4_rules_res_occ_get(void *priv)
1415 struct nsim_fib_data *data = priv;
1417 return nsim_fib_get_val(data, NSIM_RESOURCE_IPV4_FIB_RULES, false);
1420 static u64 nsim_fib_ipv6_resource_occ_get(void *priv)
1422 struct nsim_fib_data *data = priv;
1424 return nsim_fib_get_val(data, NSIM_RESOURCE_IPV6_FIB, false);
1427 static u64 nsim_fib_ipv6_rules_res_occ_get(void *priv)
1429 struct nsim_fib_data *data = priv;
1431 return nsim_fib_get_val(data, NSIM_RESOURCE_IPV6_FIB_RULES, false);
1434 static u64 nsim_fib_nexthops_res_occ_get(void *priv)
1436 struct nsim_fib_data *data = priv;
1438 return nsim_fib_get_val(data, NSIM_RESOURCE_NEXTHOPS, false);
1441 static void nsim_fib_set_max_all(struct nsim_fib_data *data,
1442 struct devlink *devlink)
1444 enum nsim_resource_id res_ids[] = {
1445 NSIM_RESOURCE_IPV4_FIB, NSIM_RESOURCE_IPV4_FIB_RULES,
1446 NSIM_RESOURCE_IPV6_FIB, NSIM_RESOURCE_IPV6_FIB_RULES,
1447 NSIM_RESOURCE_NEXTHOPS,
1451 for (i = 0; i < ARRAY_SIZE(res_ids); i++) {
1455 err = devlink_resource_size_get(devlink, res_ids[i], &val);
1458 nsim_fib_set_max(data, res_ids[i], val);
1462 static void nsim_fib_event_work(struct work_struct *work)
1464 struct nsim_fib_data *data = container_of(work, struct nsim_fib_data,
1466 struct nsim_fib_event *fib_event, *next_fib_event;
1468 LIST_HEAD(fib_event_queue);
1470 spin_lock_bh(&data->fib_event_queue_lock);
1471 list_splice_init(&data->fib_event_queue, &fib_event_queue);
1472 spin_unlock_bh(&data->fib_event_queue_lock);
1474 mutex_lock(&data->fib_lock);
1475 list_for_each_entry_safe(fib_event, next_fib_event, &fib_event_queue,
1477 nsim_fib_event(fib_event);
1478 list_del(&fib_event->list);
1482 mutex_unlock(&data->fib_lock);
1486 nsim_fib_debugfs_init(struct nsim_fib_data *data, struct nsim_dev *nsim_dev)
1488 data->ddir = debugfs_create_dir("fib", nsim_dev->ddir);
1489 if (IS_ERR(data->ddir))
1490 return PTR_ERR(data->ddir);
1492 data->fail_route_offload = false;
1493 debugfs_create_bool("fail_route_offload", 0600, data->ddir,
1494 &data->fail_route_offload);
1496 data->fail_res_nexthop_group_replace = false;
1497 debugfs_create_bool("fail_res_nexthop_group_replace", 0600, data->ddir,
1498 &data->fail_res_nexthop_group_replace);
1500 data->fail_nexthop_bucket_replace = false;
1501 debugfs_create_bool("fail_nexthop_bucket_replace", 0600, data->ddir,
1502 &data->fail_nexthop_bucket_replace);
1504 debugfs_create_file("nexthop_bucket_activity", 0200, data->ddir,
1505 data, &nsim_nexthop_bucket_activity_fops);
1509 static void nsim_fib_debugfs_exit(struct nsim_fib_data *data)
1511 debugfs_remove_recursive(data->ddir);
1514 struct nsim_fib_data *nsim_fib_create(struct devlink *devlink,
1515 struct netlink_ext_ack *extack)
1517 struct nsim_fib_data *data;
1518 struct nsim_dev *nsim_dev;
1521 data = kzalloc(sizeof(*data), GFP_KERNEL);
1523 return ERR_PTR(-ENOMEM);
1524 data->devlink = devlink;
1526 nsim_dev = devlink_priv(devlink);
1527 err = nsim_fib_debugfs_init(data, nsim_dev);
1531 mutex_init(&data->nh_lock);
1532 err = rhashtable_init(&data->nexthop_ht, &nsim_nexthop_ht_params);
1534 goto err_debugfs_exit;
1536 mutex_init(&data->fib_lock);
1537 INIT_LIST_HEAD(&data->fib_rt_list);
1538 err = rhashtable_init(&data->fib_rt_ht, &nsim_fib_rt_ht_params);
1540 goto err_rhashtable_nexthop_destroy;
1542 INIT_WORK(&data->fib_event_work, nsim_fib_event_work);
1543 INIT_LIST_HEAD(&data->fib_event_queue);
1544 spin_lock_init(&data->fib_event_queue_lock);
1546 nsim_fib_set_max_all(data, devlink);
1548 data->nexthop_nb.notifier_call = nsim_nexthop_event_nb;
1549 err = register_nexthop_notifier(devlink_net(devlink), &data->nexthop_nb,
1552 pr_err("Failed to register nexthop notifier\n");
1553 goto err_rhashtable_fib_destroy;
1556 data->fib_nb.notifier_call = nsim_fib_event_nb;
1557 err = register_fib_notifier(devlink_net(devlink), &data->fib_nb,
1558 nsim_fib_dump_inconsistent, extack);
1560 pr_err("Failed to register fib notifier\n");
1561 goto err_nexthop_nb_unregister;
1564 devlink_resource_occ_get_register(devlink,
1565 NSIM_RESOURCE_IPV4_FIB,
1566 nsim_fib_ipv4_resource_occ_get,
1568 devlink_resource_occ_get_register(devlink,
1569 NSIM_RESOURCE_IPV4_FIB_RULES,
1570 nsim_fib_ipv4_rules_res_occ_get,
1572 devlink_resource_occ_get_register(devlink,
1573 NSIM_RESOURCE_IPV6_FIB,
1574 nsim_fib_ipv6_resource_occ_get,
1576 devlink_resource_occ_get_register(devlink,
1577 NSIM_RESOURCE_IPV6_FIB_RULES,
1578 nsim_fib_ipv6_rules_res_occ_get,
1580 devlink_resource_occ_get_register(devlink,
1581 NSIM_RESOURCE_NEXTHOPS,
1582 nsim_fib_nexthops_res_occ_get,
1586 err_nexthop_nb_unregister:
1587 unregister_nexthop_notifier(devlink_net(devlink), &data->nexthop_nb);
1588 err_rhashtable_fib_destroy:
1589 flush_work(&data->fib_event_work);
1590 rhashtable_free_and_destroy(&data->fib_rt_ht, nsim_fib_rt_free,
1592 err_rhashtable_nexthop_destroy:
1593 rhashtable_free_and_destroy(&data->nexthop_ht, nsim_nexthop_free,
1595 mutex_destroy(&data->fib_lock);
1597 mutex_destroy(&data->nh_lock);
1598 nsim_fib_debugfs_exit(data);
1601 return ERR_PTR(err);
1604 void nsim_fib_destroy(struct devlink *devlink, struct nsim_fib_data *data)
1606 devlink_resource_occ_get_unregister(devlink,
1607 NSIM_RESOURCE_NEXTHOPS);
1608 devlink_resource_occ_get_unregister(devlink,
1609 NSIM_RESOURCE_IPV6_FIB_RULES);
1610 devlink_resource_occ_get_unregister(devlink,
1611 NSIM_RESOURCE_IPV6_FIB);
1612 devlink_resource_occ_get_unregister(devlink,
1613 NSIM_RESOURCE_IPV4_FIB_RULES);
1614 devlink_resource_occ_get_unregister(devlink,
1615 NSIM_RESOURCE_IPV4_FIB);
1616 unregister_fib_notifier(devlink_net(devlink), &data->fib_nb);
1617 unregister_nexthop_notifier(devlink_net(devlink), &data->nexthop_nb);
1618 flush_work(&data->fib_event_work);
1619 rhashtable_free_and_destroy(&data->fib_rt_ht, nsim_fib_rt_free,
1621 rhashtable_free_and_destroy(&data->nexthop_ht, nsim_nexthop_free,
1623 WARN_ON_ONCE(!list_empty(&data->fib_event_queue));
1624 WARN_ON_ONCE(!list_empty(&data->fib_rt_list));
1625 mutex_destroy(&data->fib_lock);
1626 mutex_destroy(&data->nh_lock);
1627 nsim_fib_debugfs_exit(data);