116ab6d7dSIdo Schimmel #include <linux/notifier.h>
216ab6d7dSIdo Schimmel #include <linux/socket.h>
316ab6d7dSIdo Schimmel #include <linux/kernel.h>
4864150dfSIdo Schimmel #include <linux/export.h>
516ab6d7dSIdo Schimmel #include <net/net_namespace.h>
616ab6d7dSIdo Schimmel #include <net/fib_notifier.h>
716ab6d7dSIdo Schimmel #include <net/netns/ipv6.h>
816ab6d7dSIdo Schimmel #include <net/ip6_fib.h>
916ab6d7dSIdo Schimmel
call_fib6_notifier(struct notifier_block * nb,enum fib_event_type event_type,struct fib_notifier_info * info)107c550dafSJiri Pirko int call_fib6_notifier(struct notifier_block *nb,
1116ab6d7dSIdo Schimmel enum fib_event_type event_type,
1216ab6d7dSIdo Schimmel struct fib_notifier_info *info)
1316ab6d7dSIdo Schimmel {
1416ab6d7dSIdo Schimmel info->family = AF_INET6;
157c550dafSJiri Pirko return call_fib_notifier(nb, event_type, info);
1616ab6d7dSIdo Schimmel }
1716ab6d7dSIdo Schimmel
call_fib6_notifiers(struct net * net,enum fib_event_type event_type,struct fib_notifier_info * info)1816ab6d7dSIdo Schimmel int call_fib6_notifiers(struct net *net, enum fib_event_type event_type,
1916ab6d7dSIdo Schimmel struct fib_notifier_info *info)
2016ab6d7dSIdo Schimmel {
2116ab6d7dSIdo Schimmel info->family = AF_INET6;
2216ab6d7dSIdo Schimmel return call_fib_notifiers(net, event_type, info);
2316ab6d7dSIdo Schimmel }
2416ab6d7dSIdo Schimmel
fib6_seq_read(struct net * net)2516ab6d7dSIdo Schimmel static unsigned int fib6_seq_read(struct net *net)
2616ab6d7dSIdo Schimmel {
27e1ee0a5bSIdo Schimmel return fib6_tables_seq_read(net) + fib6_rules_seq_read(net);
2816ab6d7dSIdo Schimmel }
2916ab6d7dSIdo Schimmel
fib6_dump(struct net * net,struct notifier_block * nb,struct netlink_ext_ack * extack)30*b7a59557SJiri Pirko static int fib6_dump(struct net *net, struct notifier_block *nb,
31*b7a59557SJiri Pirko struct netlink_ext_ack *extack)
3216ab6d7dSIdo Schimmel {
33e1ee0a5bSIdo Schimmel int err;
34e1ee0a5bSIdo Schimmel
35*b7a59557SJiri Pirko err = fib6_rules_dump(net, nb, extack);
36e1ee0a5bSIdo Schimmel if (err)
37e1ee0a5bSIdo Schimmel return err;
38e1ee0a5bSIdo Schimmel
39*b7a59557SJiri Pirko return fib6_tables_dump(net, nb, extack);
4016ab6d7dSIdo Schimmel }
4116ab6d7dSIdo Schimmel
4216ab6d7dSIdo Schimmel static const struct fib_notifier_ops fib6_notifier_ops_template = {
4316ab6d7dSIdo Schimmel .family = AF_INET6,
4416ab6d7dSIdo Schimmel .fib_seq_read = fib6_seq_read,
4516ab6d7dSIdo Schimmel .fib_dump = fib6_dump,
46864150dfSIdo Schimmel .owner = THIS_MODULE,
4716ab6d7dSIdo Schimmel };
4816ab6d7dSIdo Schimmel
fib6_notifier_init(struct net * net)4916ab6d7dSIdo Schimmel int __net_init fib6_notifier_init(struct net *net)
5016ab6d7dSIdo Schimmel {
5116ab6d7dSIdo Schimmel struct fib_notifier_ops *ops;
5216ab6d7dSIdo Schimmel
5316ab6d7dSIdo Schimmel ops = fib_notifier_ops_register(&fib6_notifier_ops_template, net);
5416ab6d7dSIdo Schimmel if (IS_ERR(ops))
5516ab6d7dSIdo Schimmel return PTR_ERR(ops);
5616ab6d7dSIdo Schimmel net->ipv6.notifier_ops = ops;
5716ab6d7dSIdo Schimmel
5816ab6d7dSIdo Schimmel return 0;
5916ab6d7dSIdo Schimmel }
6016ab6d7dSIdo Schimmel
fib6_notifier_exit(struct net * net)6116ab6d7dSIdo Schimmel void __net_exit fib6_notifier_exit(struct net *net)
6216ab6d7dSIdo Schimmel {
6316ab6d7dSIdo Schimmel fib_notifier_ops_unregister(net->ipv6.notifier_ops);
6416ab6d7dSIdo Schimmel }
65