blob: 02045494c24cccaa8b50af839aae975695319922 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001// SPDX-License-Identifier: GPL-2.0
Pravin B Shelar5f5624c2013-04-25 11:08:30 +00002#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
12static ip6_icmp_send_t __rcu *ip6_icmp_send;
13
14int inet6_register_icmp_sender(ip6_icmp_send_t *fn)
15{
16 return (cmpxchg((ip6_icmp_send_t **)&ip6_icmp_send, NULL, fn) == NULL) ?
Ian Morris67ba4152014-08-24 21:53:10 +010017 0 : -EBUSY;
Pravin B Shelar5f5624c2013-04-25 11:08:30 +000018}
19EXPORT_SYMBOL(inet6_register_icmp_sender);
20
21int 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}
32EXPORT_SYMBOL(inet6_unregister_icmp_sender);
33
34void 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;
Eric Dumazetb1cadc12016-06-18 21:52:02 -070043 send(skb, type, code, info, NULL);
Pravin B Shelar5f5624c2013-04-25 11:08:30 +000044out:
45 rcu_read_unlock();
46}
47EXPORT_SYMBOL(icmpv6_send);
48#endif