fib_notifier.c 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/rtnetlink.h>
  3. #include <linux/notifier.h>
  4. #include <linux/socket.h>
  5. #include <linux/kernel.h>
  6. #include <linux/export.h>
  7. #include <net/net_namespace.h>
  8. #include <net/fib_notifier.h>
  9. #include <net/netns/ipv4.h>
  10. #include <net/ip_fib.h>
  11. int call_fib4_notifier(struct notifier_block *nb, struct net *net,
  12. enum fib_event_type event_type,
  13. struct fib_notifier_info *info)
  14. {
  15. info->family = AF_INET;
  16. return call_fib_notifier(nb, net, event_type, info);
  17. }
  18. int call_fib4_notifiers(struct net *net, enum fib_event_type event_type,
  19. struct fib_notifier_info *info)
  20. {
  21. ASSERT_RTNL();
  22. info->family = AF_INET;
  23. net->ipv4.fib_seq++;
  24. return call_fib_notifiers(net, event_type, info);
  25. }
  26. static unsigned int fib4_seq_read(struct net *net)
  27. {
  28. ASSERT_RTNL();
  29. return net->ipv4.fib_seq + fib4_rules_seq_read(net);
  30. }
  31. static int fib4_dump(struct net *net, struct notifier_block *nb)
  32. {
  33. int err;
  34. err = fib4_rules_dump(net, nb);
  35. if (err)
  36. return err;
  37. fib_notify(net, nb);
  38. return 0;
  39. }
  40. static const struct fib_notifier_ops fib4_notifier_ops_template = {
  41. .family = AF_INET,
  42. .fib_seq_read = fib4_seq_read,
  43. .fib_dump = fib4_dump,
  44. .owner = THIS_MODULE,
  45. };
  46. int __net_init fib4_notifier_init(struct net *net)
  47. {
  48. struct fib_notifier_ops *ops;
  49. net->ipv4.fib_seq = 0;
  50. ops = fib_notifier_ops_register(&fib4_notifier_ops_template, net);
  51. if (IS_ERR(ops))
  52. return PTR_ERR(ops);
  53. net->ipv4.notifier_ops = ops;
  54. return 0;
  55. }
  56. void __net_exit fib4_notifier_exit(struct net *net)
  57. {
  58. fib_notifier_ops_unregister(net->ipv4.notifier_ops);
  59. }