1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/export.h>
3 #include <linux/icmpv6.h>
4 #include <linux/mutex.h>
5 #include <linux/netdevice.h>
6 #include <linux/spinlock.h>
7
8 #include <net/ipv6.h>
9
10 #if IS_ENABLED(CONFIG_IPV6)
11
12 static ip6_icmp_send_t __rcu *ip6_icmp_send;
13
inet6_register_icmp_sender(ip6_icmp_send_t * fn)14 int inet6_register_icmp_sender(ip6_icmp_send_t *fn)
15 {
16 return (cmpxchg((ip6_icmp_send_t **)&ip6_icmp_send, NULL, fn) == NULL) ?
17 0 : -EBUSY;
18 }
19 EXPORT_SYMBOL(inet6_register_icmp_sender);
20
inet6_unregister_icmp_sender(ip6_icmp_send_t * fn)21 int inet6_unregister_icmp_sender(ip6_icmp_send_t *fn)
22 {
23 int ret;
24
25 ret = (cmpxchg((ip6_icmp_send_t **)&ip6_icmp_send, fn, NULL) == fn) ?
26 0 : -EINVAL;
27
28 synchronize_net();
29
30 return ret;
31 }
32 EXPORT_SYMBOL(inet6_unregister_icmp_sender);
33
icmpv6_send(struct sk_buff * skb,u8 type,u8 code,__u32 info)34 void icmpv6_send(struct sk_buff *skb, u8 type, u8 code, __u32 info)
35 {
36 ip6_icmp_send_t *send;
37
38 rcu_read_lock();
39 send = rcu_dereference(ip6_icmp_send);
40
41 if (!send)
42 goto out;
43 send(skb, type, code, info, NULL);
44 out:
45 rcu_read_unlock();
46 }
47 EXPORT_SYMBOL(icmpv6_send);
48 #endif
49