blob: 8e3fda9e725cba97973ed2ce85ebe6f2e926e7cf [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Generic address resolution entity
3 *
4 * Authors:
5 * Pedro Roque <roque@di.fc.ul.pt>
6 * Alexey Kuznetsov <kuznet@ms2.inr.ac.ru>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version
11 * 2 of the License, or (at your option) any later version.
12 *
13 * Fixes:
14 * Vitaly E. Lavrov releasing NULL neighbor in neigh_add.
15 * Harald Welte Add neighbour cache statistics like rtstat
16 */
17
Joe Perchese005d192012-05-16 19:58:40 +000018#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
19
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090020#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070021#include <linux/types.h>
22#include <linux/kernel.h>
23#include <linux/module.h>
24#include <linux/socket.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070025#include <linux/netdevice.h>
26#include <linux/proc_fs.h>
27#ifdef CONFIG_SYSCTL
28#include <linux/sysctl.h>
29#endif
30#include <linux/times.h>
Eric W. Biederman457c4cb2007-09-12 12:01:34 +020031#include <net/net_namespace.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070032#include <net/neighbour.h>
33#include <net/dst.h>
34#include <net/sock.h>
Tom Tucker8d717402006-07-30 20:43:36 -070035#include <net/netevent.h>
Thomas Grafa14a49d2006-08-07 17:53:08 -070036#include <net/netlink.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070037#include <linux/rtnetlink.h>
38#include <linux/random.h>
Paulo Marques543537b2005-06-23 00:09:02 -070039#include <linux/string.h>
vignesh babuc3609d52007-08-24 22:27:55 -070040#include <linux/log2.h>
Jiri Pirko1d4c8c22013-12-07 19:26:56 +010041#include <linux/inetdevice.h>
Jiri Pirkobba24892013-12-07 19:26:57 +010042#include <net/addrconf.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070043
Joe Perchesd5d427c2013-04-15 15:17:19 +000044#define DEBUG
Linus Torvalds1da177e2005-04-16 15:20:36 -070045#define NEIGH_DEBUG 1
Joe Perchesd5d427c2013-04-15 15:17:19 +000046#define neigh_dbg(level, fmt, ...) \
47do { \
48 if (level <= NEIGH_DEBUG) \
49 pr_debug(fmt, ##__VA_ARGS__); \
50} while (0)
Linus Torvalds1da177e2005-04-16 15:20:36 -070051
52#define PNEIGH_HASHMASK 0xF
53
Kees Cooke99e88a2017-10-16 14:43:17 -070054static void neigh_timer_handler(struct timer_list *t);
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -070055static void __neigh_notify(struct neighbour *n, int type, int flags,
56 u32 pid);
57static void neigh_update_notify(struct neighbour *neigh, u32 nlmsg_pid);
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +020058static int pneigh_ifdown_and_unlock(struct neigh_table *tbl,
59 struct net_device *dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -070060
Amos Waterland45fc3b12005-09-24 16:53:16 -070061#ifdef CONFIG_PROC_FS
Christoph Hellwig71a50532018-04-15 10:16:41 +020062static const struct seq_operations neigh_stat_seq_ops;
Amos Waterland45fc3b12005-09-24 16:53:16 -070063#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -070064
65/*
66 Neighbour hash table buckets are protected with rwlock tbl->lock.
67
68 - All the scans/updates to hash buckets MUST be made under this lock.
69 - NOTHING clever should be made under this lock: no callbacks
70 to protocol backends, no attempts to send something to network.
71 It will result in deadlocks, if backend/driver wants to use neighbour
72 cache.
73 - If the entry requires some non-trivial actions, increase
74 its reference count and release table lock.
75
76 Neighbour entries are protected:
77 - with reference count.
78 - with rwlock neigh->lock
79
80 Reference count prevents destruction.
81
82 neigh->lock mainly serializes ll address data and its validity state.
83 However, the same lock is used to protect another entry fields:
84 - timer
85 - resolution queue
86
87 Again, nothing clever shall be made under neigh->lock,
88 the most complicated procedure, which we allow is dev->hard_header.
89 It is supposed, that dev->hard_header is simplistic and does
90 not make callbacks to neighbour tables.
Linus Torvalds1da177e2005-04-16 15:20:36 -070091 */
92
David S. Miller8f40b162011-07-17 13:34:11 -070093static int neigh_blackhole(struct neighbour *neigh, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -070094{
95 kfree_skb(skb);
96 return -ENETDOWN;
97}
98
Thomas Graf4f494552007-08-08 23:12:36 -070099static void neigh_cleanup_and_release(struct neighbour *neigh)
100{
101 if (neigh->parms->neigh_cleanup)
102 neigh->parms->neigh_cleanup(neigh);
103
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -0700104 __neigh_notify(neigh, RTM_DELNEIGH, 0, 0);
Ido Schimmel53f800e2016-12-23 09:32:48 +0100105 call_netevent_notifiers(NETEVENT_NEIGH_UPDATE, neigh);
Thomas Graf4f494552007-08-08 23:12:36 -0700106 neigh_release(neigh);
107}
108
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109/*
110 * It is random distribution in the interval (1/2)*base...(3/2)*base.
111 * It corresponds to default IPv6 settings and is not overridable,
112 * because it is really reasonable choice.
113 */
114
115unsigned long neigh_rand_reach_time(unsigned long base)
116{
Aruna-Hewapathirane63862b52014-01-11 07:15:59 -0500117 return base ? (prandom_u32() % base) + (base >> 1) : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900119EXPORT_SYMBOL(neigh_rand_reach_time);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700120
121
Roopa Prabhuf6a6f202018-06-12 21:26:10 -0700122static bool neigh_del(struct neighbour *n, __u8 state, __u8 flags,
Sowmini Varadhan50710342017-06-02 09:01:49 -0700123 struct neighbour __rcu **np, struct neigh_table *tbl)
124{
125 bool retval = false;
126
127 write_lock(&n->lock);
Roopa Prabhuf6a6f202018-06-12 21:26:10 -0700128 if (refcount_read(&n->refcnt) == 1 && !(n->nud_state & state) &&
129 !(n->flags & flags)) {
Sowmini Varadhan50710342017-06-02 09:01:49 -0700130 struct neighbour *neigh;
131
132 neigh = rcu_dereference_protected(n->next,
133 lockdep_is_held(&tbl->lock));
134 rcu_assign_pointer(*np, neigh);
135 n->dead = 1;
136 retval = true;
137 }
138 write_unlock(&n->lock);
139 if (retval)
140 neigh_cleanup_and_release(n);
141 return retval;
142}
143
144bool neigh_remove_one(struct neighbour *ndel, struct neigh_table *tbl)
145{
146 struct neigh_hash_table *nht;
147 void *pkey = ndel->primary_key;
148 u32 hash_val;
149 struct neighbour *n;
150 struct neighbour __rcu **np;
151
152 nht = rcu_dereference_protected(tbl->nht,
153 lockdep_is_held(&tbl->lock));
154 hash_val = tbl->hash(pkey, ndel->dev, nht->hash_rnd);
155 hash_val = hash_val >> (32 - nht->hash_shift);
156
157 np = &nht->hash_buckets[hash_val];
158 while ((n = rcu_dereference_protected(*np,
159 lockdep_is_held(&tbl->lock)))) {
160 if (n == ndel)
Roopa Prabhuf6a6f202018-06-12 21:26:10 -0700161 return neigh_del(n, 0, 0, np, tbl);
Sowmini Varadhan50710342017-06-02 09:01:49 -0700162 np = &n->next;
163 }
164 return false;
165}
166
Linus Torvalds1da177e2005-04-16 15:20:36 -0700167static int neigh_forced_gc(struct neigh_table *tbl)
168{
169 int shrunk = 0;
170 int i;
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000171 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172
173 NEIGH_CACHE_STAT_INC(tbl, forced_gc_runs);
174
175 write_lock_bh(&tbl->lock);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000176 nht = rcu_dereference_protected(tbl->nht,
177 lockdep_is_held(&tbl->lock));
David S. Millercd089332011-07-11 01:28:12 -0700178 for (i = 0; i < (1 << nht->hash_shift); i++) {
Eric Dumazet767e97e2010-10-06 17:49:21 -0700179 struct neighbour *n;
180 struct neighbour __rcu **np;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700181
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000182 np = &nht->hash_buckets[i];
Eric Dumazet767e97e2010-10-06 17:49:21 -0700183 while ((n = rcu_dereference_protected(*np,
184 lockdep_is_held(&tbl->lock))) != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700185 /* Neighbour record may be discarded if:
186 * - nobody refers to it.
187 * - it is not permanent
188 */
Roopa Prabhuf6a6f202018-06-12 21:26:10 -0700189 if (neigh_del(n, NUD_PERMANENT, NTF_EXT_LEARNED, np,
190 tbl)) {
Sowmini Varadhan50710342017-06-02 09:01:49 -0700191 shrunk = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700192 continue;
193 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700194 np = &n->next;
195 }
196 }
197
198 tbl->last_flush = jiffies;
199
200 write_unlock_bh(&tbl->lock);
201
202 return shrunk;
203}
204
Pavel Emelyanova43d8992007-12-20 15:49:05 -0800205static void neigh_add_timer(struct neighbour *n, unsigned long when)
206{
207 neigh_hold(n);
208 if (unlikely(mod_timer(&n->timer, when))) {
209 printk("NEIGH: BUG, double timer add, state is %x\n",
210 n->nud_state);
211 dump_stack();
212 }
213}
214
Linus Torvalds1da177e2005-04-16 15:20:36 -0700215static int neigh_del_timer(struct neighbour *n)
216{
217 if ((n->nud_state & NUD_IN_TIMER) &&
218 del_timer(&n->timer)) {
219 neigh_release(n);
220 return 1;
221 }
222 return 0;
223}
224
225static void pneigh_queue_purge(struct sk_buff_head *list)
226{
227 struct sk_buff *skb;
228
229 while ((skb = skb_dequeue(list)) != NULL) {
230 dev_put(skb->dev);
231 kfree_skb(skb);
232 }
233}
234
Herbert Xu49636bb2005-10-23 17:18:00 +1000235static void neigh_flush_dev(struct neigh_table *tbl, struct net_device *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236{
237 int i;
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000238 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700239
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000240 nht = rcu_dereference_protected(tbl->nht,
241 lockdep_is_held(&tbl->lock));
242
David S. Millercd089332011-07-11 01:28:12 -0700243 for (i = 0; i < (1 << nht->hash_shift); i++) {
Eric Dumazet767e97e2010-10-06 17:49:21 -0700244 struct neighbour *n;
245 struct neighbour __rcu **np = &nht->hash_buckets[i];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700246
Eric Dumazet767e97e2010-10-06 17:49:21 -0700247 while ((n = rcu_dereference_protected(*np,
248 lockdep_is_held(&tbl->lock))) != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249 if (dev && n->dev != dev) {
250 np = &n->next;
251 continue;
252 }
Eric Dumazet767e97e2010-10-06 17:49:21 -0700253 rcu_assign_pointer(*np,
254 rcu_dereference_protected(n->next,
255 lockdep_is_held(&tbl->lock)));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700256 write_lock(&n->lock);
257 neigh_del_timer(n);
258 n->dead = 1;
259
Reshetova, Elena9f237432017-06-30 13:07:55 +0300260 if (refcount_read(&n->refcnt) != 1) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700261 /* The most unpleasant situation.
262 We must destroy neighbour entry,
263 but someone still uses it.
264
265 The destroy will be delayed until
266 the last user releases us, but
267 we must kill timers etc. and move
268 it to safe state.
269 */
Eric Dumazetc9ab4d82013-06-28 02:37:42 -0700270 __skb_queue_purge(&n->arp_queue);
Eric Dumazet8b5c1712011-11-09 12:07:14 +0000271 n->arp_queue_len_bytes = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700272 n->output = neigh_blackhole;
273 if (n->nud_state & NUD_VALID)
274 n->nud_state = NUD_NOARP;
275 else
276 n->nud_state = NUD_NONE;
Joe Perchesd5d427c2013-04-15 15:17:19 +0000277 neigh_dbg(2, "neigh %p is stray\n", n);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700278 }
279 write_unlock(&n->lock);
Thomas Graf4f494552007-08-08 23:12:36 -0700280 neigh_cleanup_and_release(n);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700281 }
282 }
Herbert Xu49636bb2005-10-23 17:18:00 +1000283}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700284
Herbert Xu49636bb2005-10-23 17:18:00 +1000285void neigh_changeaddr(struct neigh_table *tbl, struct net_device *dev)
286{
287 write_lock_bh(&tbl->lock);
288 neigh_flush_dev(tbl, dev);
289 write_unlock_bh(&tbl->lock);
290}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900291EXPORT_SYMBOL(neigh_changeaddr);
Herbert Xu49636bb2005-10-23 17:18:00 +1000292
293int neigh_ifdown(struct neigh_table *tbl, struct net_device *dev)
294{
295 write_lock_bh(&tbl->lock);
296 neigh_flush_dev(tbl, dev);
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +0200297 pneigh_ifdown_and_unlock(tbl, dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700298
299 del_timer_sync(&tbl->proxy_timer);
300 pneigh_queue_purge(&tbl->proxy_queue);
301 return 0;
302}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900303EXPORT_SYMBOL(neigh_ifdown);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700304
David Miller596b9b62011-07-25 00:01:25 +0000305static struct neighbour *neigh_alloc(struct neigh_table *tbl, struct net_device *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700306{
307 struct neighbour *n = NULL;
308 unsigned long now = jiffies;
309 int entries;
310
311 entries = atomic_inc_return(&tbl->entries) - 1;
312 if (entries >= tbl->gc_thresh3 ||
313 (entries >= tbl->gc_thresh2 &&
314 time_after(now, tbl->last_flush + 5 * HZ))) {
315 if (!neigh_forced_gc(tbl) &&
Rick Jonesfb811392015-08-07 11:10:37 -0700316 entries >= tbl->gc_thresh3) {
317 net_info_ratelimited("%s: neighbor table overflow!\n",
318 tbl->id);
319 NEIGH_CACHE_STAT_INC(tbl, table_fulls);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320 goto out_entries;
Rick Jonesfb811392015-08-07 11:10:37 -0700321 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700322 }
323
YOSHIFUJI Hideaki / 吉藤英明08433ef2013-01-24 00:44:23 +0000324 n = kzalloc(tbl->entry_size + dev->neigh_priv_len, GFP_ATOMIC);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700325 if (!n)
326 goto out_entries;
327
Eric Dumazetc9ab4d82013-06-28 02:37:42 -0700328 __skb_queue_head_init(&n->arp_queue);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700329 rwlock_init(&n->lock);
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +0000330 seqlock_init(&n->ha_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331 n->updated = n->used = now;
332 n->nud_state = NUD_NONE;
333 n->output = neigh_blackhole;
David S. Millerf6b72b622011-07-14 07:53:20 -0700334 seqlock_init(&n->hh.hh_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335 n->parms = neigh_parms_clone(&tbl->parms);
Kees Cooke99e88a2017-10-16 14:43:17 -0700336 timer_setup(&n->timer, neigh_timer_handler, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700337
338 NEIGH_CACHE_STAT_INC(tbl, allocs);
339 n->tbl = tbl;
Reshetova, Elena9f237432017-06-30 13:07:55 +0300340 refcount_set(&n->refcnt, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341 n->dead = 1;
342out:
343 return n;
344
345out_entries:
346 atomic_dec(&tbl->entries);
347 goto out;
348}
349
David S. Miller2c2aba62011-12-28 15:06:58 -0500350static void neigh_get_hash_rnd(u32 *x)
351{
Jason A. Donenfeldb3d0f782017-06-07 23:00:05 -0400352 *x = get_random_u32() | 1;
David S. Miller2c2aba62011-12-28 15:06:58 -0500353}
354
David S. Millercd089332011-07-11 01:28:12 -0700355static struct neigh_hash_table *neigh_hash_alloc(unsigned int shift)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700356{
David S. Millercd089332011-07-11 01:28:12 -0700357 size_t size = (1 << shift) * sizeof(struct neighbour *);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000358 struct neigh_hash_table *ret;
Eric Dumazet6193d2b2011-01-19 22:02:47 +0000359 struct neighbour __rcu **buckets;
David S. Miller2c2aba62011-12-28 15:06:58 -0500360 int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700361
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000362 ret = kmalloc(sizeof(*ret), GFP_ATOMIC);
363 if (!ret)
364 return NULL;
365 if (size <= PAGE_SIZE)
366 buckets = kzalloc(size, GFP_ATOMIC);
367 else
Eric Dumazet6193d2b2011-01-19 22:02:47 +0000368 buckets = (struct neighbour __rcu **)
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000369 __get_free_pages(GFP_ATOMIC | __GFP_ZERO,
370 get_order(size));
371 if (!buckets) {
372 kfree(ret);
373 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374 }
Eric Dumazet6193d2b2011-01-19 22:02:47 +0000375 ret->hash_buckets = buckets;
David S. Millercd089332011-07-11 01:28:12 -0700376 ret->hash_shift = shift;
David S. Miller2c2aba62011-12-28 15:06:58 -0500377 for (i = 0; i < NEIGH_NUM_HASH_RND; i++)
378 neigh_get_hash_rnd(&ret->hash_rnd[i]);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700379 return ret;
380}
381
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000382static void neigh_hash_free_rcu(struct rcu_head *head)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383{
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000384 struct neigh_hash_table *nht = container_of(head,
385 struct neigh_hash_table,
386 rcu);
David S. Millercd089332011-07-11 01:28:12 -0700387 size_t size = (1 << nht->hash_shift) * sizeof(struct neighbour *);
Eric Dumazet6193d2b2011-01-19 22:02:47 +0000388 struct neighbour __rcu **buckets = nht->hash_buckets;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700389
390 if (size <= PAGE_SIZE)
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000391 kfree(buckets);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 else
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000393 free_pages((unsigned long)buckets, get_order(size));
394 kfree(nht);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700395}
396
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000397static struct neigh_hash_table *neigh_hash_grow(struct neigh_table *tbl,
David S. Millercd089332011-07-11 01:28:12 -0700398 unsigned long new_shift)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700399{
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000400 unsigned int i, hash;
401 struct neigh_hash_table *new_nht, *old_nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402
403 NEIGH_CACHE_STAT_INC(tbl, hash_grows);
404
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000405 old_nht = rcu_dereference_protected(tbl->nht,
406 lockdep_is_held(&tbl->lock));
David S. Millercd089332011-07-11 01:28:12 -0700407 new_nht = neigh_hash_alloc(new_shift);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000408 if (!new_nht)
409 return old_nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700410
David S. Millercd089332011-07-11 01:28:12 -0700411 for (i = 0; i < (1 << old_nht->hash_shift); i++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700412 struct neighbour *n, *next;
413
Eric Dumazet767e97e2010-10-06 17:49:21 -0700414 for (n = rcu_dereference_protected(old_nht->hash_buckets[i],
415 lockdep_is_held(&tbl->lock));
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000416 n != NULL;
417 n = next) {
418 hash = tbl->hash(n->primary_key, n->dev,
419 new_nht->hash_rnd);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700420
David S. Millercd089332011-07-11 01:28:12 -0700421 hash >>= (32 - new_nht->hash_shift);
Eric Dumazet767e97e2010-10-06 17:49:21 -0700422 next = rcu_dereference_protected(n->next,
423 lockdep_is_held(&tbl->lock));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700424
Eric Dumazet767e97e2010-10-06 17:49:21 -0700425 rcu_assign_pointer(n->next,
426 rcu_dereference_protected(
427 new_nht->hash_buckets[hash],
428 lockdep_is_held(&tbl->lock)));
429 rcu_assign_pointer(new_nht->hash_buckets[hash], n);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700430 }
431 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000433 rcu_assign_pointer(tbl->nht, new_nht);
434 call_rcu(&old_nht->rcu, neigh_hash_free_rcu);
435 return new_nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436}
437
438struct neighbour *neigh_lookup(struct neigh_table *tbl, const void *pkey,
439 struct net_device *dev)
440{
441 struct neighbour *n;
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +0900442
Linus Torvalds1da177e2005-04-16 15:20:36 -0700443 NEIGH_CACHE_STAT_INC(tbl, lookups);
444
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000445 rcu_read_lock_bh();
Eric W. Biederman60395a22015-03-03 17:10:44 -0600446 n = __neigh_lookup_noref(tbl, pkey, dev);
447 if (n) {
Reshetova, Elena9f237432017-06-30 13:07:55 +0300448 if (!refcount_inc_not_zero(&n->refcnt))
Eric W. Biederman60395a22015-03-03 17:10:44 -0600449 n = NULL;
450 NEIGH_CACHE_STAT_INC(tbl, hits);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700451 }
Eric Dumazet767e97e2010-10-06 17:49:21 -0700452
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000453 rcu_read_unlock_bh();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700454 return n;
455}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900456EXPORT_SYMBOL(neigh_lookup);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700457
Eric W. Biederman426b5302008-01-24 00:13:18 -0800458struct neighbour *neigh_lookup_nodev(struct neigh_table *tbl, struct net *net,
459 const void *pkey)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700460{
461 struct neighbour *n;
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300462 unsigned int key_len = tbl->key_len;
Pavel Emelyanovbc4bf5f2008-02-23 19:57:02 -0800463 u32 hash_val;
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000464 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700465
466 NEIGH_CACHE_STAT_INC(tbl, lookups);
467
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000468 rcu_read_lock_bh();
469 nht = rcu_dereference_bh(tbl->nht);
David S. Millercd089332011-07-11 01:28:12 -0700470 hash_val = tbl->hash(pkey, NULL, nht->hash_rnd) >> (32 - nht->hash_shift);
Eric Dumazet767e97e2010-10-06 17:49:21 -0700471
472 for (n = rcu_dereference_bh(nht->hash_buckets[hash_val]);
473 n != NULL;
474 n = rcu_dereference_bh(n->next)) {
Eric W. Biederman426b5302008-01-24 00:13:18 -0800475 if (!memcmp(n->primary_key, pkey, key_len) &&
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +0900476 net_eq(dev_net(n->dev), net)) {
Reshetova, Elena9f237432017-06-30 13:07:55 +0300477 if (!refcount_inc_not_zero(&n->refcnt))
Eric Dumazet767e97e2010-10-06 17:49:21 -0700478 n = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700479 NEIGH_CACHE_STAT_INC(tbl, hits);
480 break;
481 }
482 }
Eric Dumazet767e97e2010-10-06 17:49:21 -0700483
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000484 rcu_read_unlock_bh();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700485 return n;
486}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900487EXPORT_SYMBOL(neigh_lookup_nodev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700488
David S. Millera263b302012-07-02 02:02:15 -0700489struct neighbour *__neigh_create(struct neigh_table *tbl, const void *pkey,
490 struct net_device *dev, bool want_ref)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700491{
492 u32 hash_val;
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300493 unsigned int key_len = tbl->key_len;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700494 int error;
David Miller596b9b62011-07-25 00:01:25 +0000495 struct neighbour *n1, *rc, *n = neigh_alloc(tbl, dev);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000496 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497
498 if (!n) {
499 rc = ERR_PTR(-ENOBUFS);
500 goto out;
501 }
502
503 memcpy(n->primary_key, pkey, key_len);
504 n->dev = dev;
505 dev_hold(dev);
506
507 /* Protocol specific setup. */
508 if (tbl->constructor && (error = tbl->constructor(n)) < 0) {
509 rc = ERR_PTR(error);
510 goto out_neigh_release;
511 }
512
David Millerda6a8fa2011-07-25 00:01:38 +0000513 if (dev->netdev_ops->ndo_neigh_construct) {
Jiri Pirko503eebc2016-07-05 11:27:37 +0200514 error = dev->netdev_ops->ndo_neigh_construct(dev, n);
David Millerda6a8fa2011-07-25 00:01:38 +0000515 if (error < 0) {
516 rc = ERR_PTR(error);
517 goto out_neigh_release;
518 }
519 }
520
David S. Miller447f2192011-12-19 15:04:41 -0500521 /* Device specific setup. */
522 if (n->parms->neigh_setup &&
523 (error = n->parms->neigh_setup(n)) < 0) {
524 rc = ERR_PTR(error);
525 goto out_neigh_release;
526 }
527
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100528 n->confirmed = jiffies - (NEIGH_VAR(n->parms, BASE_REACHABLE_TIME) << 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700529
530 write_lock_bh(&tbl->lock);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000531 nht = rcu_dereference_protected(tbl->nht,
532 lockdep_is_held(&tbl->lock));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700533
David S. Millercd089332011-07-11 01:28:12 -0700534 if (atomic_read(&tbl->entries) > (1 << nht->hash_shift))
535 nht = neigh_hash_grow(tbl, nht->hash_shift + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700536
Jim Westfall096b9852018-01-14 04:18:50 -0800537 hash_val = tbl->hash(n->primary_key, dev, nht->hash_rnd) >> (32 - nht->hash_shift);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538
539 if (n->parms->dead) {
540 rc = ERR_PTR(-EINVAL);
541 goto out_tbl_unlock;
542 }
543
Eric Dumazet767e97e2010-10-06 17:49:21 -0700544 for (n1 = rcu_dereference_protected(nht->hash_buckets[hash_val],
545 lockdep_is_held(&tbl->lock));
546 n1 != NULL;
547 n1 = rcu_dereference_protected(n1->next,
548 lockdep_is_held(&tbl->lock))) {
Jim Westfall096b9852018-01-14 04:18:50 -0800549 if (dev == n1->dev && !memcmp(n1->primary_key, n->primary_key, key_len)) {
David S. Millera263b302012-07-02 02:02:15 -0700550 if (want_ref)
551 neigh_hold(n1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700552 rc = n1;
553 goto out_tbl_unlock;
554 }
555 }
556
Linus Torvalds1da177e2005-04-16 15:20:36 -0700557 n->dead = 0;
David S. Millera263b302012-07-02 02:02:15 -0700558 if (want_ref)
559 neigh_hold(n);
Eric Dumazet767e97e2010-10-06 17:49:21 -0700560 rcu_assign_pointer(n->next,
561 rcu_dereference_protected(nht->hash_buckets[hash_val],
562 lockdep_is_held(&tbl->lock)));
563 rcu_assign_pointer(nht->hash_buckets[hash_val], n);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700564 write_unlock_bh(&tbl->lock);
Joe Perchesd5d427c2013-04-15 15:17:19 +0000565 neigh_dbg(2, "neigh %p is created\n", n);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700566 rc = n;
567out:
568 return rc;
569out_tbl_unlock:
570 write_unlock_bh(&tbl->lock);
571out_neigh_release:
572 neigh_release(n);
573 goto out;
574}
David S. Millera263b302012-07-02 02:02:15 -0700575EXPORT_SYMBOL(__neigh_create);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300577static u32 pneigh_hash(const void *pkey, unsigned int key_len)
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700578{
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700579 u32 hash_val = *(u32 *)(pkey + key_len - 4);
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700580 hash_val ^= (hash_val >> 16);
581 hash_val ^= hash_val >> 8;
582 hash_val ^= hash_val >> 4;
583 hash_val &= PNEIGH_HASHMASK;
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900584 return hash_val;
585}
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700586
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900587static struct pneigh_entry *__pneigh_lookup_1(struct pneigh_entry *n,
588 struct net *net,
589 const void *pkey,
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300590 unsigned int key_len,
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900591 struct net_device *dev)
592{
593 while (n) {
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700594 if (!memcmp(n->key, pkey, key_len) &&
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900595 net_eq(pneigh_net(n), net) &&
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700596 (n->dev == dev || !n->dev))
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900597 return n;
598 n = n->next;
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700599 }
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900600 return NULL;
601}
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700602
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900603struct pneigh_entry *__pneigh_lookup(struct neigh_table *tbl,
604 struct net *net, const void *pkey, struct net_device *dev)
605{
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300606 unsigned int key_len = tbl->key_len;
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900607 u32 hash_val = pneigh_hash(pkey, key_len);
608
609 return __pneigh_lookup_1(tbl->phash_buckets[hash_val],
610 net, pkey, key_len, dev);
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700611}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900612EXPORT_SYMBOL_GPL(__pneigh_lookup);
Pavel Emelyanovfa86d322008-03-24 14:48:59 -0700613
Eric W. Biederman426b5302008-01-24 00:13:18 -0800614struct pneigh_entry * pneigh_lookup(struct neigh_table *tbl,
615 struct net *net, const void *pkey,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700616 struct net_device *dev, int creat)
617{
618 struct pneigh_entry *n;
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300619 unsigned int key_len = tbl->key_len;
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900620 u32 hash_val = pneigh_hash(pkey, key_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700621
622 read_lock_bh(&tbl->lock);
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900623 n = __pneigh_lookup_1(tbl->phash_buckets[hash_val],
624 net, pkey, key_len, dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700625 read_unlock_bh(&tbl->lock);
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900626
627 if (n || !creat)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700628 goto out;
629
Pavel Emelyanov4ae28942007-10-15 12:54:15 -0700630 ASSERT_RTNL();
631
Linus Torvalds1da177e2005-04-16 15:20:36 -0700632 n = kmalloc(sizeof(*n) + key_len, GFP_KERNEL);
633 if (!n)
634 goto out;
635
Eric W. Biedermanefd7ef12015-03-11 23:04:08 -0500636 write_pnet(&n->net, net);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700637 memcpy(n->key, pkey, key_len);
638 n->dev = dev;
639 if (dev)
640 dev_hold(dev);
641
642 if (tbl->pconstructor && tbl->pconstructor(n)) {
643 if (dev)
644 dev_put(dev);
645 kfree(n);
646 n = NULL;
647 goto out;
648 }
649
650 write_lock_bh(&tbl->lock);
651 n->next = tbl->phash_buckets[hash_val];
652 tbl->phash_buckets[hash_val] = n;
653 write_unlock_bh(&tbl->lock);
654out:
655 return n;
656}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900657EXPORT_SYMBOL(pneigh_lookup);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700658
659
Eric W. Biederman426b5302008-01-24 00:13:18 -0800660int pneigh_delete(struct neigh_table *tbl, struct net *net, const void *pkey,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700661 struct net_device *dev)
662{
663 struct pneigh_entry *n, **np;
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +0300664 unsigned int key_len = tbl->key_len;
YOSHIFUJI Hideakibe01d652008-03-28 12:46:53 +0900665 u32 hash_val = pneigh_hash(pkey, key_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700666
667 write_lock_bh(&tbl->lock);
668 for (np = &tbl->phash_buckets[hash_val]; (n = *np) != NULL;
669 np = &n->next) {
Eric W. Biederman426b5302008-01-24 00:13:18 -0800670 if (!memcmp(n->key, pkey, key_len) && n->dev == dev &&
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +0900671 net_eq(pneigh_net(n), net)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700672 *np = n->next;
673 write_unlock_bh(&tbl->lock);
674 if (tbl->pdestructor)
675 tbl->pdestructor(n);
676 if (n->dev)
677 dev_put(n->dev);
678 kfree(n);
679 return 0;
680 }
681 }
682 write_unlock_bh(&tbl->lock);
683 return -ENOENT;
684}
685
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +0200686static int pneigh_ifdown_and_unlock(struct neigh_table *tbl,
687 struct net_device *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700688{
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +0200689 struct pneigh_entry *n, **np, *freelist = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700690 u32 h;
691
692 for (h = 0; h <= PNEIGH_HASHMASK; h++) {
693 np = &tbl->phash_buckets[h];
694 while ((n = *np) != NULL) {
695 if (!dev || n->dev == dev) {
696 *np = n->next;
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +0200697 n->next = freelist;
698 freelist = n;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700699 continue;
700 }
701 np = &n->next;
702 }
703 }
Wolfgang Bumiller53b76cd2018-04-12 10:46:55 +0200704 write_unlock_bh(&tbl->lock);
705 while ((n = freelist)) {
706 freelist = n->next;
707 n->next = NULL;
708 if (tbl->pdestructor)
709 tbl->pdestructor(n);
710 if (n->dev)
711 dev_put(n->dev);
712 kfree(n);
713 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700714 return -ENOENT;
715}
716
Denis V. Lunev06f05112008-01-24 00:30:58 -0800717static void neigh_parms_destroy(struct neigh_parms *parms);
718
719static inline void neigh_parms_put(struct neigh_parms *parms)
720{
Reshetova, Elena63439442017-06-30 13:07:56 +0300721 if (refcount_dec_and_test(&parms->refcnt))
Denis V. Lunev06f05112008-01-24 00:30:58 -0800722 neigh_parms_destroy(parms);
723}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700724
725/*
726 * neighbour must already be out of the table;
727 *
728 */
729void neigh_destroy(struct neighbour *neigh)
730{
David Millerda6a8fa2011-07-25 00:01:38 +0000731 struct net_device *dev = neigh->dev;
732
Linus Torvalds1da177e2005-04-16 15:20:36 -0700733 NEIGH_CACHE_STAT_INC(neigh->tbl, destroys);
734
735 if (!neigh->dead) {
Joe Perchese005d192012-05-16 19:58:40 +0000736 pr_warn("Destroying alive neighbour %p\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700737 dump_stack();
738 return;
739 }
740
741 if (neigh_del_timer(neigh))
Joe Perchese005d192012-05-16 19:58:40 +0000742 pr_warn("Impossible event\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700743
Eric Dumazetc9ab4d82013-06-28 02:37:42 -0700744 write_lock_bh(&neigh->lock);
745 __skb_queue_purge(&neigh->arp_queue);
746 write_unlock_bh(&neigh->lock);
Eric Dumazet8b5c1712011-11-09 12:07:14 +0000747 neigh->arp_queue_len_bytes = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700748
David S. Miller447f2192011-12-19 15:04:41 -0500749 if (dev->netdev_ops->ndo_neigh_destroy)
Jiri Pirko503eebc2016-07-05 11:27:37 +0200750 dev->netdev_ops->ndo_neigh_destroy(dev, neigh);
David S. Miller447f2192011-12-19 15:04:41 -0500751
David Millerda6a8fa2011-07-25 00:01:38 +0000752 dev_put(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700753 neigh_parms_put(neigh->parms);
754
Joe Perchesd5d427c2013-04-15 15:17:19 +0000755 neigh_dbg(2, "neigh %p is destroyed\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756
757 atomic_dec(&neigh->tbl->entries);
David Miller5b8b0062011-07-25 00:01:22 +0000758 kfree_rcu(neigh, rcu);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700759}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +0900760EXPORT_SYMBOL(neigh_destroy);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700761
762/* Neighbour state is suspicious;
763 disable fast path.
764
765 Called with write_locked neigh.
766 */
767static void neigh_suspect(struct neighbour *neigh)
768{
Joe Perchesd5d427c2013-04-15 15:17:19 +0000769 neigh_dbg(2, "neigh %p is suspected\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700770
771 neigh->output = neigh->ops->output;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700772}
773
774/* Neighbour state is OK;
775 enable fast path.
776
777 Called with write_locked neigh.
778 */
779static void neigh_connect(struct neighbour *neigh)
780{
Joe Perchesd5d427c2013-04-15 15:17:19 +0000781 neigh_dbg(2, "neigh %p is connected\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700782
783 neigh->output = neigh->ops->connected_output;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700784}
785
Eric Dumazete4c4e442009-07-30 03:15:07 +0000786static void neigh_periodic_work(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700787{
Eric Dumazete4c4e442009-07-30 03:15:07 +0000788 struct neigh_table *tbl = container_of(work, struct neigh_table, gc_work.work);
Eric Dumazet767e97e2010-10-06 17:49:21 -0700789 struct neighbour *n;
790 struct neighbour __rcu **np;
Eric Dumazete4c4e442009-07-30 03:15:07 +0000791 unsigned int i;
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000792 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700793
794 NEIGH_CACHE_STAT_INC(tbl, periodic_gc_runs);
795
Eric Dumazete4c4e442009-07-30 03:15:07 +0000796 write_lock_bh(&tbl->lock);
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000797 nht = rcu_dereference_protected(tbl->nht,
798 lockdep_is_held(&tbl->lock));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700799
800 /*
801 * periodically recompute ReachableTime from random function
802 */
803
Eric Dumazete4c4e442009-07-30 03:15:07 +0000804 if (time_after(jiffies, tbl->last_rand + 300 * HZ)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700805 struct neigh_parms *p;
Eric Dumazete4c4e442009-07-30 03:15:07 +0000806 tbl->last_rand = jiffies;
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +0100807 list_for_each_entry(p, &tbl->parms_list, list)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700808 p->reachable_time =
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100809 neigh_rand_reach_time(NEIGH_VAR(p, BASE_REACHABLE_TIME));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700810 }
811
Duan Jiongfeff9ab2014-02-27 17:14:41 +0800812 if (atomic_read(&tbl->entries) < tbl->gc_thresh1)
813 goto out;
814
David S. Millercd089332011-07-11 01:28:12 -0700815 for (i = 0 ; i < (1 << nht->hash_shift); i++) {
Eric Dumazetd6bf7812010-10-04 06:15:44 +0000816 np = &nht->hash_buckets[i];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700817
Eric Dumazet767e97e2010-10-06 17:49:21 -0700818 while ((n = rcu_dereference_protected(*np,
819 lockdep_is_held(&tbl->lock))) != NULL) {
Eric Dumazete4c4e442009-07-30 03:15:07 +0000820 unsigned int state;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700821
Eric Dumazete4c4e442009-07-30 03:15:07 +0000822 write_lock(&n->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700823
Eric Dumazete4c4e442009-07-30 03:15:07 +0000824 state = n->nud_state;
Roopa Prabhu9ce33e42018-04-24 13:49:34 -0700825 if ((state & (NUD_PERMANENT | NUD_IN_TIMER)) ||
826 (n->flags & NTF_EXT_LEARNED)) {
Eric Dumazete4c4e442009-07-30 03:15:07 +0000827 write_unlock(&n->lock);
828 goto next_elt;
829 }
830
831 if (time_before(n->used, n->confirmed))
832 n->used = n->confirmed;
833
Reshetova, Elena9f237432017-06-30 13:07:55 +0300834 if (refcount_read(&n->refcnt) == 1 &&
Eric Dumazete4c4e442009-07-30 03:15:07 +0000835 (state == NUD_FAILED ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100836 time_after(jiffies, n->used + NEIGH_VAR(n->parms, GC_STALETIME)))) {
Eric Dumazete4c4e442009-07-30 03:15:07 +0000837 *np = n->next;
838 n->dead = 1;
839 write_unlock(&n->lock);
840 neigh_cleanup_and_release(n);
841 continue;
842 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700843 write_unlock(&n->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700844
845next_elt:
Eric Dumazete4c4e442009-07-30 03:15:07 +0000846 np = &n->next;
847 }
848 /*
849 * It's fine to release lock here, even if hash table
850 * grows while we are preempted.
851 */
852 write_unlock_bh(&tbl->lock);
853 cond_resched();
854 write_lock_bh(&tbl->lock);
Michel Machado84338a62012-02-21 16:04:13 -0500855 nht = rcu_dereference_protected(tbl->nht,
856 lockdep_is_held(&tbl->lock));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700857 }
YOSHIFUJI Hideaki / 吉藤英明27246802013-01-22 05:20:05 +0000858out:
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100859 /* Cycle through all hash buckets every BASE_REACHABLE_TIME/2 ticks.
860 * ARP entry timeouts range from 1/2 BASE_REACHABLE_TIME to 3/2
861 * BASE_REACHABLE_TIME.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700862 */
viresh kumarf6180022014-01-22 12:23:33 +0530863 queue_delayed_work(system_power_efficient_wq, &tbl->gc_work,
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100864 NEIGH_VAR(&tbl->parms, BASE_REACHABLE_TIME) >> 1);
Eric Dumazete4c4e442009-07-30 03:15:07 +0000865 write_unlock_bh(&tbl->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700866}
867
868static __inline__ int neigh_max_probes(struct neighbour *n)
869{
870 struct neigh_parms *p = n->parms;
YOSHIFUJI Hideaki/吉藤英明8da86462015-03-19 22:41:46 +0900871 return NEIGH_VAR(p, UCAST_PROBES) + NEIGH_VAR(p, APP_PROBES) +
872 (n->nud_state & NUD_PROBE ? NEIGH_VAR(p, MCAST_REPROBES) :
873 NEIGH_VAR(p, MCAST_PROBES));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700874}
875
Timo Teras5ef12d92009-06-11 04:16:28 -0700876static void neigh_invalidate(struct neighbour *neigh)
Eric Dumazet0a141502010-03-09 19:40:54 +0000877 __releases(neigh->lock)
878 __acquires(neigh->lock)
Timo Teras5ef12d92009-06-11 04:16:28 -0700879{
880 struct sk_buff *skb;
881
882 NEIGH_CACHE_STAT_INC(neigh->tbl, res_failed);
Joe Perchesd5d427c2013-04-15 15:17:19 +0000883 neigh_dbg(2, "neigh %p is failed\n", neigh);
Timo Teras5ef12d92009-06-11 04:16:28 -0700884 neigh->updated = jiffies;
885
886 /* It is very thin place. report_unreachable is very complicated
887 routine. Particularly, it can hit the same neighbour entry!
888
889 So that, we try to be accurate and avoid dead loop. --ANK
890 */
891 while (neigh->nud_state == NUD_FAILED &&
892 (skb = __skb_dequeue(&neigh->arp_queue)) != NULL) {
893 write_unlock(&neigh->lock);
894 neigh->ops->error_report(neigh, skb);
895 write_lock(&neigh->lock);
896 }
Eric Dumazetc9ab4d82013-06-28 02:37:42 -0700897 __skb_queue_purge(&neigh->arp_queue);
Eric Dumazet8b5c1712011-11-09 12:07:14 +0000898 neigh->arp_queue_len_bytes = 0;
Timo Teras5ef12d92009-06-11 04:16:28 -0700899}
900
Eric Dumazetcd28ca02011-08-09 08:15:58 +0000901static void neigh_probe(struct neighbour *neigh)
902 __releases(neigh->lock)
903{
Hannes Frederic Sowa4ed377e2013-09-21 06:32:34 +0200904 struct sk_buff *skb = skb_peek_tail(&neigh->arp_queue);
Eric Dumazetcd28ca02011-08-09 08:15:58 +0000905 /* keep skb alive even if arp_queue overflows */
906 if (skb)
Martin Zhang19125c12015-11-17 20:49:30 +0800907 skb = skb_clone(skb, GFP_ATOMIC);
Eric Dumazetcd28ca02011-08-09 08:15:58 +0000908 write_unlock(&neigh->lock);
Eric Dumazet48481c82017-03-23 12:39:21 -0700909 if (neigh->ops->solicit)
910 neigh->ops->solicit(neigh, skb);
Eric Dumazetcd28ca02011-08-09 08:15:58 +0000911 atomic_inc(&neigh->probes);
912 kfree_skb(skb);
913}
914
Linus Torvalds1da177e2005-04-16 15:20:36 -0700915/* Called when a timer expires for a neighbour entry. */
916
Kees Cooke99e88a2017-10-16 14:43:17 -0700917static void neigh_timer_handler(struct timer_list *t)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700918{
919 unsigned long now, next;
Kees Cooke99e88a2017-10-16 14:43:17 -0700920 struct neighbour *neigh = from_timer(neigh, t, timer);
Eric Dumazet95c96172012-04-15 05:58:06 +0000921 unsigned int state;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700922 int notify = 0;
923
924 write_lock(&neigh->lock);
925
926 state = neigh->nud_state;
927 now = jiffies;
928 next = now + HZ;
929
David S. Miller045f7b32011-11-01 17:45:55 -0400930 if (!(state & NUD_IN_TIMER))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700931 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700932
933 if (state & NUD_REACHABLE) {
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +0900934 if (time_before_eq(now,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700935 neigh->confirmed + neigh->parms->reachable_time)) {
Joe Perchesd5d427c2013-04-15 15:17:19 +0000936 neigh_dbg(2, "neigh %p is still alive\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700937 next = neigh->confirmed + neigh->parms->reachable_time;
938 } else if (time_before_eq(now,
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100939 neigh->used +
940 NEIGH_VAR(neigh->parms, DELAY_PROBE_TIME))) {
Joe Perchesd5d427c2013-04-15 15:17:19 +0000941 neigh_dbg(2, "neigh %p is delayed\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700942 neigh->nud_state = NUD_DELAY;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -0800943 neigh->updated = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700944 neigh_suspect(neigh);
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100945 next = now + NEIGH_VAR(neigh->parms, DELAY_PROBE_TIME);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700946 } else {
Joe Perchesd5d427c2013-04-15 15:17:19 +0000947 neigh_dbg(2, "neigh %p is suspected\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700948 neigh->nud_state = NUD_STALE;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -0800949 neigh->updated = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950 neigh_suspect(neigh);
Tom Tucker8d717402006-07-30 20:43:36 -0700951 notify = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952 }
953 } else if (state & NUD_DELAY) {
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +0900954 if (time_before_eq(now,
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100955 neigh->confirmed +
956 NEIGH_VAR(neigh->parms, DELAY_PROBE_TIME))) {
Joe Perchesd5d427c2013-04-15 15:17:19 +0000957 neigh_dbg(2, "neigh %p is now reachable\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958 neigh->nud_state = NUD_REACHABLE;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -0800959 neigh->updated = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700960 neigh_connect(neigh);
Tom Tucker8d717402006-07-30 20:43:36 -0700961 notify = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962 next = neigh->confirmed + neigh->parms->reachable_time;
963 } else {
Joe Perchesd5d427c2013-04-15 15:17:19 +0000964 neigh_dbg(2, "neigh %p is probed\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700965 neigh->nud_state = NUD_PROBE;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -0800966 neigh->updated = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700967 atomic_set(&neigh->probes, 0);
Erik Kline765c9c62015-05-18 19:44:41 +0900968 notify = 1;
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100969 next = now + NEIGH_VAR(neigh->parms, RETRANS_TIME);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700970 }
971 } else {
972 /* NUD_PROBE|NUD_INCOMPLETE */
Jiri Pirko1f9248e2013-12-07 19:26:53 +0100973 next = now + NEIGH_VAR(neigh->parms, RETRANS_TIME);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700974 }
975
976 if ((neigh->nud_state & (NUD_INCOMPLETE | NUD_PROBE)) &&
977 atomic_read(&neigh->probes) >= neigh_max_probes(neigh)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978 neigh->nud_state = NUD_FAILED;
979 notify = 1;
Timo Teras5ef12d92009-06-11 04:16:28 -0700980 neigh_invalidate(neigh);
Duan Jiong5e2c21d2014-02-27 17:03:03 +0800981 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700982 }
983
984 if (neigh->nud_state & NUD_IN_TIMER) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700985 if (time_before(next, jiffies + HZ/2))
986 next = jiffies + HZ/2;
Herbert Xu6fb99742005-10-23 16:37:48 +1000987 if (!mod_timer(&neigh->timer, next))
988 neigh_hold(neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700989 }
990 if (neigh->nud_state & (NUD_INCOMPLETE | NUD_PROBE)) {
Eric Dumazetcd28ca02011-08-09 08:15:58 +0000991 neigh_probe(neigh);
David S. Miller9ff56602008-02-17 18:39:54 -0800992 } else {
David S. Miller69cc64d2008-02-11 21:45:44 -0800993out:
David S. Miller9ff56602008-02-17 18:39:54 -0800994 write_unlock(&neigh->lock);
995 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700996
Thomas Grafd961db32007-08-08 23:12:56 -0700997 if (notify)
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -0700998 neigh_update_notify(neigh, 0);
Thomas Grafd961db32007-08-08 23:12:56 -0700999
Linus Torvalds1da177e2005-04-16 15:20:36 -07001000 neigh_release(neigh);
1001}
1002
1003int __neigh_event_send(struct neighbour *neigh, struct sk_buff *skb)
1004{
1005 int rc;
Eric Dumazetcd28ca02011-08-09 08:15:58 +00001006 bool immediate_probe = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001007
1008 write_lock_bh(&neigh->lock);
1009
1010 rc = 0;
1011 if (neigh->nud_state & (NUD_CONNECTED | NUD_DELAY | NUD_PROBE))
1012 goto out_unlock_bh;
Julian Anastasov2c51a972015-06-16 22:56:39 +03001013 if (neigh->dead)
1014 goto out_dead;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001015
Linus Torvalds1da177e2005-04-16 15:20:36 -07001016 if (!(neigh->nud_state & (NUD_STALE | NUD_INCOMPLETE))) {
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001017 if (NEIGH_VAR(neigh->parms, MCAST_PROBES) +
1018 NEIGH_VAR(neigh->parms, APP_PROBES)) {
Eric Dumazetcd28ca02011-08-09 08:15:58 +00001019 unsigned long next, now = jiffies;
1020
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001021 atomic_set(&neigh->probes,
1022 NEIGH_VAR(neigh->parms, UCAST_PROBES));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001023 neigh->nud_state = NUD_INCOMPLETE;
Eric Dumazetcd28ca02011-08-09 08:15:58 +00001024 neigh->updated = now;
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001025 next = now + max(NEIGH_VAR(neigh->parms, RETRANS_TIME),
1026 HZ/2);
Eric Dumazetcd28ca02011-08-09 08:15:58 +00001027 neigh_add_timer(neigh, next);
1028 immediate_probe = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001029 } else {
1030 neigh->nud_state = NUD_FAILED;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -08001031 neigh->updated = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001032 write_unlock_bh(&neigh->lock);
1033
Wei Yongjunf3fbbe02009-02-25 00:37:32 +00001034 kfree_skb(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001035 return 1;
1036 }
1037 } else if (neigh->nud_state & NUD_STALE) {
Joe Perchesd5d427c2013-04-15 15:17:19 +00001038 neigh_dbg(2, "neigh %p is delayed\n", neigh);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001039 neigh->nud_state = NUD_DELAY;
YOSHIFUJI Hideaki955aaa22006-03-20 16:52:52 -08001040 neigh->updated = jiffies;
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001041 neigh_add_timer(neigh, jiffies +
1042 NEIGH_VAR(neigh->parms, DELAY_PROBE_TIME));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001043 }
1044
1045 if (neigh->nud_state == NUD_INCOMPLETE) {
1046 if (skb) {
Eric Dumazet8b5c1712011-11-09 12:07:14 +00001047 while (neigh->arp_queue_len_bytes + skb->truesize >
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001048 NEIGH_VAR(neigh->parms, QUEUE_LEN_BYTES)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001049 struct sk_buff *buff;
Eric Dumazet8b5c1712011-11-09 12:07:14 +00001050
David S. Millerf72051b2008-09-23 01:11:18 -07001051 buff = __skb_dequeue(&neigh->arp_queue);
Eric Dumazet8b5c1712011-11-09 12:07:14 +00001052 if (!buff)
1053 break;
1054 neigh->arp_queue_len_bytes -= buff->truesize;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001055 kfree_skb(buff);
Neil Horman9a6d2762008-07-16 20:50:49 -07001056 NEIGH_CACHE_STAT_INC(neigh->tbl, unres_discards);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001057 }
Eric Dumazeta4731132010-05-27 16:09:39 -07001058 skb_dst_force(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001059 __skb_queue_tail(&neigh->arp_queue, skb);
Eric Dumazet8b5c1712011-11-09 12:07:14 +00001060 neigh->arp_queue_len_bytes += skb->truesize;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001061 }
1062 rc = 1;
1063 }
1064out_unlock_bh:
Eric Dumazetcd28ca02011-08-09 08:15:58 +00001065 if (immediate_probe)
1066 neigh_probe(neigh);
1067 else
1068 write_unlock(&neigh->lock);
1069 local_bh_enable();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001070 return rc;
Julian Anastasov2c51a972015-06-16 22:56:39 +03001071
1072out_dead:
1073 if (neigh->nud_state & NUD_STALE)
1074 goto out_unlock_bh;
1075 write_unlock_bh(&neigh->lock);
1076 kfree_skb(skb);
1077 return 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001078}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001079EXPORT_SYMBOL(__neigh_event_send);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001080
David S. Millerf6b72b622011-07-14 07:53:20 -07001081static void neigh_update_hhs(struct neighbour *neigh)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001082{
1083 struct hh_cache *hh;
Stephen Hemminger3b04ddd2007-10-09 01:40:57 -07001084 void (*update)(struct hh_cache*, const struct net_device*, const unsigned char *)
Doug Kehn91a72a72010-07-14 18:02:16 -07001085 = NULL;
1086
1087 if (neigh->dev->header_ops)
1088 update = neigh->dev->header_ops->cache_update;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001089
1090 if (update) {
David S. Millerf6b72b622011-07-14 07:53:20 -07001091 hh = &neigh->hh;
1092 if (hh->hh_len) {
Stephen Hemminger3644f0c2006-12-07 15:08:17 -08001093 write_seqlock_bh(&hh->hh_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001094 update(hh, neigh->dev, neigh->ha);
Stephen Hemminger3644f0c2006-12-07 15:08:17 -08001095 write_sequnlock_bh(&hh->hh_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001096 }
1097 }
1098}
1099
1100
1101
1102/* Generic update routine.
1103 -- lladdr is new lladdr or NULL, if it is not supplied.
1104 -- new is new state.
1105 -- flags
1106 NEIGH_UPDATE_F_OVERRIDE allows to override existing lladdr,
1107 if it is different.
1108 NEIGH_UPDATE_F_WEAK_OVERRIDE will suspect existing "connected"
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001109 lladdr instead of overriding it
Linus Torvalds1da177e2005-04-16 15:20:36 -07001110 if it is different.
Linus Torvalds1da177e2005-04-16 15:20:36 -07001111 NEIGH_UPDATE_F_ADMIN means that the change is administrative.
1112
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001113 NEIGH_UPDATE_F_OVERRIDE_ISROUTER allows to override existing
Linus Torvalds1da177e2005-04-16 15:20:36 -07001114 NTF_ROUTER flag.
1115 NEIGH_UPDATE_F_ISROUTER indicates if the neighbour is known as
1116 a router.
1117
1118 Caller MUST hold reference count on the entry.
1119 */
1120
1121int neigh_update(struct neighbour *neigh, const u8 *lladdr, u8 new,
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07001122 u32 flags, u32 nlmsg_pid)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001123{
1124 u8 old;
1125 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001126 int notify = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001127 struct net_device *dev;
1128 int update_isrouter = 0;
1129
1130 write_lock_bh(&neigh->lock);
1131
1132 dev = neigh->dev;
1133 old = neigh->nud_state;
1134 err = -EPERM;
1135
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001136 if (!(flags & NEIGH_UPDATE_F_ADMIN) &&
Linus Torvalds1da177e2005-04-16 15:20:36 -07001137 (old & (NUD_NOARP | NUD_PERMANENT)))
1138 goto out;
Julian Anastasov2c51a972015-06-16 22:56:39 +03001139 if (neigh->dead)
1140 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001141
Roopa Prabhu9ce33e42018-04-24 13:49:34 -07001142 neigh_update_ext_learned(neigh, flags, &notify);
1143
Linus Torvalds1da177e2005-04-16 15:20:36 -07001144 if (!(new & NUD_VALID)) {
1145 neigh_del_timer(neigh);
1146 if (old & NUD_CONNECTED)
1147 neigh_suspect(neigh);
1148 neigh->nud_state = new;
1149 err = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001150 notify = old & NUD_VALID;
Timo Teras5ef12d92009-06-11 04:16:28 -07001151 if ((old & (NUD_INCOMPLETE | NUD_PROBE)) &&
1152 (new & NUD_FAILED)) {
1153 neigh_invalidate(neigh);
1154 notify = 1;
1155 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001156 goto out;
1157 }
1158
1159 /* Compare new lladdr with cached one */
1160 if (!dev->addr_len) {
1161 /* First case: device needs no address. */
1162 lladdr = neigh->ha;
1163 } else if (lladdr) {
1164 /* The second case: if something is already cached
1165 and a new address is proposed:
1166 - compare new & old
1167 - if they are different, check override flag
1168 */
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001169 if ((old & NUD_VALID) &&
Linus Torvalds1da177e2005-04-16 15:20:36 -07001170 !memcmp(lladdr, neigh->ha, dev->addr_len))
1171 lladdr = neigh->ha;
1172 } else {
1173 /* No address is supplied; if we know something,
1174 use it, otherwise discard the request.
1175 */
1176 err = -EINVAL;
1177 if (!(old & NUD_VALID))
1178 goto out;
1179 lladdr = neigh->ha;
1180 }
1181
Linus Torvalds1da177e2005-04-16 15:20:36 -07001182 /* If entry was valid and address is not changed,
1183 do not change entry state, if new one is STALE.
1184 */
1185 err = 0;
1186 update_isrouter = flags & NEIGH_UPDATE_F_OVERRIDE_ISROUTER;
1187 if (old & NUD_VALID) {
1188 if (lladdr != neigh->ha && !(flags & NEIGH_UPDATE_F_OVERRIDE)) {
1189 update_isrouter = 0;
1190 if ((flags & NEIGH_UPDATE_F_WEAK_OVERRIDE) &&
1191 (old & NUD_CONNECTED)) {
1192 lladdr = neigh->ha;
1193 new = NUD_STALE;
1194 } else
1195 goto out;
1196 } else {
Julian Anastasov0e7bbcc2016-07-27 09:56:50 +03001197 if (lladdr == neigh->ha && new == NUD_STALE &&
1198 !(flags & NEIGH_UPDATE_F_ADMIN))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001199 new = old;
1200 }
1201 }
1202
Ihar Hrachyshka77d71232017-05-16 08:44:24 -07001203 /* Update timestamps only once we know we will make a change to the
1204 * neighbour entry. Otherwise we risk to move the locktime window with
1205 * noop updates and ignore relevant ARP updates.
1206 */
1207 if (new != old || lladdr != neigh->ha) {
1208 if (new & NUD_CONNECTED)
1209 neigh->confirmed = jiffies;
1210 neigh->updated = jiffies;
1211 }
1212
Linus Torvalds1da177e2005-04-16 15:20:36 -07001213 if (new != old) {
1214 neigh_del_timer(neigh);
Erik Kline765c9c62015-05-18 19:44:41 +09001215 if (new & NUD_PROBE)
1216 atomic_set(&neigh->probes, 0);
Pavel Emelyanova43d8992007-12-20 15:49:05 -08001217 if (new & NUD_IN_TIMER)
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001218 neigh_add_timer(neigh, (jiffies +
1219 ((new & NUD_REACHABLE) ?
David S. Miller667347f2005-09-27 12:07:44 -07001220 neigh->parms->reachable_time :
1221 0)));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 neigh->nud_state = new;
Bob Gilligan53385d22013-12-15 13:39:56 -08001223 notify = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001224 }
1225
1226 if (lladdr != neigh->ha) {
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001227 write_seqlock(&neigh->ha_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001228 memcpy(&neigh->ha, lladdr, dev->addr_len);
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001229 write_sequnlock(&neigh->ha_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001230 neigh_update_hhs(neigh);
1231 if (!(new & NUD_CONNECTED))
1232 neigh->confirmed = jiffies -
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001233 (NEIGH_VAR(neigh->parms, BASE_REACHABLE_TIME) << 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001234 notify = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001235 }
1236 if (new == old)
1237 goto out;
1238 if (new & NUD_CONNECTED)
1239 neigh_connect(neigh);
1240 else
1241 neigh_suspect(neigh);
1242 if (!(old & NUD_VALID)) {
1243 struct sk_buff *skb;
1244
1245 /* Again: avoid dead loop if something went wrong */
1246
1247 while (neigh->nud_state & NUD_VALID &&
1248 (skb = __skb_dequeue(&neigh->arp_queue)) != NULL) {
David S. Miller69cce1d2011-07-17 23:09:49 -07001249 struct dst_entry *dst = skb_dst(skb);
1250 struct neighbour *n2, *n1 = neigh;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001251 write_unlock_bh(&neigh->lock);
roy.qing.li@gmail.come049f282011-10-17 22:32:42 +00001252
1253 rcu_read_lock();
David S. Miller13a43d92012-07-02 22:15:37 -07001254
1255 /* Why not just use 'neigh' as-is? The problem is that
1256 * things such as shaper, eql, and sch_teql can end up
1257 * using alternative, different, neigh objects to output
1258 * the packet in the output path. So what we need to do
1259 * here is re-lookup the top-level neigh in the path so
1260 * we can reinject the packet there.
1261 */
1262 n2 = NULL;
1263 if (dst) {
1264 n2 = dst_neigh_lookup_skb(dst, skb);
1265 if (n2)
1266 n1 = n2;
1267 }
David S. Miller8f40b162011-07-17 13:34:11 -07001268 n1->output(n1, skb);
David S. Miller13a43d92012-07-02 22:15:37 -07001269 if (n2)
1270 neigh_release(n2);
roy.qing.li@gmail.come049f282011-10-17 22:32:42 +00001271 rcu_read_unlock();
1272
Linus Torvalds1da177e2005-04-16 15:20:36 -07001273 write_lock_bh(&neigh->lock);
1274 }
Eric Dumazetc9ab4d82013-06-28 02:37:42 -07001275 __skb_queue_purge(&neigh->arp_queue);
Eric Dumazet8b5c1712011-11-09 12:07:14 +00001276 neigh->arp_queue_len_bytes = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277 }
1278out:
1279 if (update_isrouter) {
1280 neigh->flags = (flags & NEIGH_UPDATE_F_ISROUTER) ?
1281 (neigh->flags | NTF_ROUTER) :
1282 (neigh->flags & ~NTF_ROUTER);
1283 }
1284 write_unlock_bh(&neigh->lock);
Tom Tucker8d717402006-07-30 20:43:36 -07001285
1286 if (notify)
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07001287 neigh_update_notify(neigh, nlmsg_pid);
Thomas Grafd961db32007-08-08 23:12:56 -07001288
Linus Torvalds1da177e2005-04-16 15:20:36 -07001289 return err;
1290}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001291EXPORT_SYMBOL(neigh_update);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001292
Jiri Benc7e980562013-12-11 13:48:20 +01001293/* Update the neigh to listen temporarily for probe responses, even if it is
1294 * in a NUD_FAILED state. The caller has to hold neigh->lock for writing.
1295 */
1296void __neigh_set_probe_once(struct neighbour *neigh)
1297{
Julian Anastasov2c51a972015-06-16 22:56:39 +03001298 if (neigh->dead)
1299 return;
Jiri Benc7e980562013-12-11 13:48:20 +01001300 neigh->updated = jiffies;
1301 if (!(neigh->nud_state & NUD_FAILED))
1302 return;
Duan Jiong2176d5d2014-05-09 13:16:48 +08001303 neigh->nud_state = NUD_INCOMPLETE;
1304 atomic_set(&neigh->probes, neigh_max_probes(neigh));
Jiri Benc7e980562013-12-11 13:48:20 +01001305 neigh_add_timer(neigh,
1306 jiffies + NEIGH_VAR(neigh->parms, RETRANS_TIME));
1307}
1308EXPORT_SYMBOL(__neigh_set_probe_once);
1309
Linus Torvalds1da177e2005-04-16 15:20:36 -07001310struct neighbour *neigh_event_ns(struct neigh_table *tbl,
1311 u8 *lladdr, void *saddr,
1312 struct net_device *dev)
1313{
1314 struct neighbour *neigh = __neigh_lookup(tbl, saddr, dev,
1315 lladdr || !dev->addr_len);
1316 if (neigh)
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001317 neigh_update(neigh, lladdr, NUD_STALE,
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07001318 NEIGH_UPDATE_F_OVERRIDE, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319 return neigh;
1320}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001321EXPORT_SYMBOL(neigh_event_ns);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001322
Eric Dumazet34d101d2010-10-11 09:16:57 -07001323/* called with read_lock_bh(&n->lock); */
Eric W. Biedermanbdf53c52015-03-02 00:13:22 -06001324static void neigh_hh_init(struct neighbour *n)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001325{
Eric W. Biedermanbdf53c52015-03-02 00:13:22 -06001326 struct net_device *dev = n->dev;
1327 __be16 prot = n->tbl->protocol;
David S. Millerf6b72b622011-07-14 07:53:20 -07001328 struct hh_cache *hh = &n->hh;
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001329
1330 write_lock_bh(&n->lock);
Eric Dumazet34d101d2010-10-11 09:16:57 -07001331
David S. Millerf6b72b622011-07-14 07:53:20 -07001332 /* Only one thread can come in here and initialize the
1333 * hh_cache entry.
1334 */
David S. Millerb23b5452011-07-16 17:45:02 -07001335 if (!hh->hh_len)
1336 dev->header_ops->cache(n, hh, prot);
David S. Millerf6b72b622011-07-14 07:53:20 -07001337
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001338 write_unlock_bh(&n->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001339}
1340
Linus Torvalds1da177e2005-04-16 15:20:36 -07001341/* Slow and careful. */
1342
David S. Miller8f40b162011-07-17 13:34:11 -07001343int neigh_resolve_output(struct neighbour *neigh, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001344{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001345 int rc = 0;
1346
Linus Torvalds1da177e2005-04-16 15:20:36 -07001347 if (!neigh_event_send(neigh, skb)) {
1348 int err;
1349 struct net_device *dev = neigh->dev;
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001350 unsigned int seq;
Eric Dumazet34d101d2010-10-11 09:16:57 -07001351
David S. Millerf6b72b622011-07-14 07:53:20 -07001352 if (dev->header_ops->cache && !neigh->hh.hh_len)
Eric W. Biedermanbdf53c52015-03-02 00:13:22 -06001353 neigh_hh_init(neigh);
Eric Dumazet34d101d2010-10-11 09:16:57 -07001354
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001355 do {
ramesh.nagappa@gmail.come1f16502012-10-05 19:10:15 +00001356 __skb_pull(skb, skb_network_offset(skb));
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001357 seq = read_seqbegin(&neigh->ha_lock);
1358 err = dev_hard_header(skb, dev, ntohs(skb->protocol),
1359 neigh->ha, NULL, skb->len);
1360 } while (read_seqretry(&neigh->ha_lock, seq));
Eric Dumazet34d101d2010-10-11 09:16:57 -07001361
Linus Torvalds1da177e2005-04-16 15:20:36 -07001362 if (err >= 0)
David S. Miller542d4d62011-07-16 18:06:24 -07001363 rc = dev_queue_xmit(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001364 else
1365 goto out_kfree_skb;
1366 }
1367out:
1368 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001369out_kfree_skb:
1370 rc = -EINVAL;
1371 kfree_skb(skb);
1372 goto out;
1373}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001374EXPORT_SYMBOL(neigh_resolve_output);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001375
1376/* As fast as possible without hh cache */
1377
David S. Miller8f40b162011-07-17 13:34:11 -07001378int neigh_connected_output(struct neighbour *neigh, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001379{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001380 struct net_device *dev = neigh->dev;
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001381 unsigned int seq;
David S. Miller8f40b162011-07-17 13:34:11 -07001382 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001383
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001384 do {
ramesh.nagappa@gmail.come1f16502012-10-05 19:10:15 +00001385 __skb_pull(skb, skb_network_offset(skb));
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00001386 seq = read_seqbegin(&neigh->ha_lock);
1387 err = dev_hard_header(skb, dev, ntohs(skb->protocol),
1388 neigh->ha, NULL, skb->len);
1389 } while (read_seqretry(&neigh->ha_lock, seq));
1390
Linus Torvalds1da177e2005-04-16 15:20:36 -07001391 if (err >= 0)
David S. Miller542d4d62011-07-16 18:06:24 -07001392 err = dev_queue_xmit(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001393 else {
1394 err = -EINVAL;
1395 kfree_skb(skb);
1396 }
1397 return err;
1398}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001399EXPORT_SYMBOL(neigh_connected_output);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001400
David S. Miller8f40b162011-07-17 13:34:11 -07001401int neigh_direct_output(struct neighbour *neigh, struct sk_buff *skb)
1402{
1403 return dev_queue_xmit(skb);
1404}
1405EXPORT_SYMBOL(neigh_direct_output);
1406
Kees Cooke99e88a2017-10-16 14:43:17 -07001407static void neigh_proxy_process(struct timer_list *t)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001408{
Kees Cooke99e88a2017-10-16 14:43:17 -07001409 struct neigh_table *tbl = from_timer(tbl, t, proxy_timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001410 long sched_next = 0;
1411 unsigned long now = jiffies;
David S. Millerf72051b2008-09-23 01:11:18 -07001412 struct sk_buff *skb, *n;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001413
1414 spin_lock(&tbl->proxy_queue.lock);
1415
David S. Millerf72051b2008-09-23 01:11:18 -07001416 skb_queue_walk_safe(&tbl->proxy_queue, skb, n) {
1417 long tdif = NEIGH_CB(skb)->sched_next - now;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001418
Linus Torvalds1da177e2005-04-16 15:20:36 -07001419 if (tdif <= 0) {
David S. Millerf72051b2008-09-23 01:11:18 -07001420 struct net_device *dev = skb->dev;
Eric Dumazet20e60742011-08-22 19:32:42 +00001421
David S. Millerf72051b2008-09-23 01:11:18 -07001422 __skb_unlink(skb, &tbl->proxy_queue);
Eric Dumazet20e60742011-08-22 19:32:42 +00001423 if (tbl->proxy_redo && netif_running(dev)) {
1424 rcu_read_lock();
David S. Millerf72051b2008-09-23 01:11:18 -07001425 tbl->proxy_redo(skb);
Eric Dumazet20e60742011-08-22 19:32:42 +00001426 rcu_read_unlock();
1427 } else {
David S. Millerf72051b2008-09-23 01:11:18 -07001428 kfree_skb(skb);
Eric Dumazet20e60742011-08-22 19:32:42 +00001429 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001430
1431 dev_put(dev);
1432 } else if (!sched_next || tdif < sched_next)
1433 sched_next = tdif;
1434 }
1435 del_timer(&tbl->proxy_timer);
1436 if (sched_next)
1437 mod_timer(&tbl->proxy_timer, jiffies + sched_next);
1438 spin_unlock(&tbl->proxy_queue.lock);
1439}
1440
1441void pneigh_enqueue(struct neigh_table *tbl, struct neigh_parms *p,
1442 struct sk_buff *skb)
1443{
1444 unsigned long now = jiffies;
Aruna-Hewapathirane63862b52014-01-11 07:15:59 -05001445
1446 unsigned long sched_next = now + (prandom_u32() %
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001447 NEIGH_VAR(p, PROXY_DELAY));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001448
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001449 if (tbl->proxy_queue.qlen > NEIGH_VAR(p, PROXY_QLEN)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001450 kfree_skb(skb);
1451 return;
1452 }
Patrick McHardya61bbcf2005-08-14 17:24:31 -07001453
1454 NEIGH_CB(skb)->sched_next = sched_next;
1455 NEIGH_CB(skb)->flags |= LOCALLY_ENQUEUED;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001456
1457 spin_lock(&tbl->proxy_queue.lock);
1458 if (del_timer(&tbl->proxy_timer)) {
1459 if (time_before(tbl->proxy_timer.expires, sched_next))
1460 sched_next = tbl->proxy_timer.expires;
1461 }
Eric Dumazetadf30902009-06-02 05:19:30 +00001462 skb_dst_drop(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001463 dev_hold(skb->dev);
1464 __skb_queue_tail(&tbl->proxy_queue, skb);
1465 mod_timer(&tbl->proxy_timer, sched_next);
1466 spin_unlock(&tbl->proxy_queue.lock);
1467}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001468EXPORT_SYMBOL(pneigh_enqueue);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001469
Tobias Klauser97fd5bc2009-07-13 11:17:49 -07001470static inline struct neigh_parms *lookup_neigh_parms(struct neigh_table *tbl,
Eric W. Biederman426b5302008-01-24 00:13:18 -08001471 struct net *net, int ifindex)
1472{
1473 struct neigh_parms *p;
1474
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01001475 list_for_each_entry(p, &tbl->parms_list, list) {
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09001476 if ((p->dev && p->dev->ifindex == ifindex && net_eq(neigh_parms_net(p), net)) ||
Gao feng170d6f92013-06-20 10:01:33 +08001477 (!p->dev && !ifindex && net_eq(net, &init_net)))
Eric W. Biederman426b5302008-01-24 00:13:18 -08001478 return p;
1479 }
1480
1481 return NULL;
1482}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001483
1484struct neigh_parms *neigh_parms_alloc(struct net_device *dev,
1485 struct neigh_table *tbl)
1486{
Gao fengcf89d6b2013-06-20 10:01:32 +08001487 struct neigh_parms *p;
Stephen Hemminger00829822008-11-20 20:14:53 -08001488 struct net *net = dev_net(dev);
1489 const struct net_device_ops *ops = dev->netdev_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001490
Gao fengcf89d6b2013-06-20 10:01:32 +08001491 p = kmemdup(&tbl->parms, sizeof(*p), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001492 if (p) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493 p->tbl = tbl;
Reshetova, Elena63439442017-06-30 13:07:56 +03001494 refcount_set(&p->refcnt, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001495 p->reachable_time =
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001496 neigh_rand_reach_time(NEIGH_VAR(p, BASE_REACHABLE_TIME));
Denis V. Lunev486b51d2008-01-14 22:59:59 -08001497 dev_hold(dev);
1498 p->dev = dev;
Eric W. Biedermanefd7ef12015-03-11 23:04:08 -05001499 write_pnet(&p->net, net);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001500 p->sysctl_table = NULL;
Veaceslav Falico63134802013-08-02 19:07:38 +02001501
1502 if (ops->ndo_neigh_setup && ops->ndo_neigh_setup(dev, p)) {
Veaceslav Falico63134802013-08-02 19:07:38 +02001503 dev_put(dev);
1504 kfree(p);
1505 return NULL;
1506 }
1507
Linus Torvalds1da177e2005-04-16 15:20:36 -07001508 write_lock_bh(&tbl->lock);
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01001509 list_add(&p->list, &tbl->parms.list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001510 write_unlock_bh(&tbl->lock);
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01001511
1512 neigh_parms_data_state_cleanall(p);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001513 }
1514 return p;
1515}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001516EXPORT_SYMBOL(neigh_parms_alloc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001517
1518static void neigh_rcu_free_parms(struct rcu_head *head)
1519{
1520 struct neigh_parms *parms =
1521 container_of(head, struct neigh_parms, rcu_head);
1522
1523 neigh_parms_put(parms);
1524}
1525
1526void neigh_parms_release(struct neigh_table *tbl, struct neigh_parms *parms)
1527{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001528 if (!parms || parms == &tbl->parms)
1529 return;
1530 write_lock_bh(&tbl->lock);
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01001531 list_del(&parms->list);
1532 parms->dead = 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001533 write_unlock_bh(&tbl->lock);
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01001534 if (parms->dev)
1535 dev_put(parms->dev);
1536 call_rcu(&parms->rcu_head, neigh_rcu_free_parms);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001537}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001538EXPORT_SYMBOL(neigh_parms_release);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001539
Denis V. Lunev06f05112008-01-24 00:30:58 -08001540static void neigh_parms_destroy(struct neigh_parms *parms)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001541{
1542 kfree(parms);
1543}
1544
Pavel Emelianovc2ecba72007-04-17 12:45:31 -07001545static struct lock_class_key neigh_table_proxy_queue_class;
1546
WANG Congd7480fd2014-11-10 15:59:36 -08001547static struct neigh_table *neigh_tables[NEIGH_NR_TABLES] __read_mostly;
1548
1549void neigh_table_init(int index, struct neigh_table *tbl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001550{
1551 unsigned long now = jiffies;
1552 unsigned long phsize;
1553
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01001554 INIT_LIST_HEAD(&tbl->parms_list);
1555 list_add(&tbl->parms.list, &tbl->parms_list);
Eric Dumazete42ea982008-11-12 00:54:54 -08001556 write_pnet(&tbl->parms.net, &init_net);
Reshetova, Elena63439442017-06-30 13:07:56 +03001557 refcount_set(&tbl->parms.refcnt, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001558 tbl->parms.reachable_time =
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001559 neigh_rand_reach_time(NEIGH_VAR(&tbl->parms, BASE_REACHABLE_TIME));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001560
Linus Torvalds1da177e2005-04-16 15:20:36 -07001561 tbl->stats = alloc_percpu(struct neigh_statistics);
1562 if (!tbl->stats)
1563 panic("cannot create neighbour cache statistics");
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001564
Linus Torvalds1da177e2005-04-16 15:20:36 -07001565#ifdef CONFIG_PROC_FS
Christoph Hellwig71a50532018-04-15 10:16:41 +02001566 if (!proc_create_seq_data(tbl->id, 0, init_net.proc_net_stat,
1567 &neigh_stat_seq_ops, tbl))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001568 panic("cannot create neighbour proc dir entry");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001569#endif
1570
David S. Millercd089332011-07-11 01:28:12 -07001571 RCU_INIT_POINTER(tbl->nht, neigh_hash_alloc(3));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001572
1573 phsize = (PNEIGH_HASHMASK + 1) * sizeof(struct pneigh_entry *);
Andrew Morton77d04bd2006-04-07 14:52:59 -07001574 tbl->phash_buckets = kzalloc(phsize, GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001575
Eric Dumazetd6bf7812010-10-04 06:15:44 +00001576 if (!tbl->nht || !tbl->phash_buckets)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001577 panic("cannot allocate neighbour cache hashes");
1578
YOSHIFUJI Hideaki / 吉藤英明08433ef2013-01-24 00:44:23 +00001579 if (!tbl->entry_size)
1580 tbl->entry_size = ALIGN(offsetof(struct neighbour, primary_key) +
1581 tbl->key_len, NEIGH_PRIV_ALIGN);
1582 else
1583 WARN_ON(tbl->entry_size % NEIGH_PRIV_ALIGN);
1584
Linus Torvalds1da177e2005-04-16 15:20:36 -07001585 rwlock_init(&tbl->lock);
Tejun Heo203b42f2012-08-21 13:18:23 -07001586 INIT_DEFERRABLE_WORK(&tbl->gc_work, neigh_periodic_work);
viresh kumarf6180022014-01-22 12:23:33 +05301587 queue_delayed_work(system_power_efficient_wq, &tbl->gc_work,
1588 tbl->parms.reachable_time);
Kees Cooke99e88a2017-10-16 14:43:17 -07001589 timer_setup(&tbl->proxy_timer, neigh_proxy_process, 0);
Pavel Emelianovc2ecba72007-04-17 12:45:31 -07001590 skb_queue_head_init_class(&tbl->proxy_queue,
1591 &neigh_table_proxy_queue_class);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001592
1593 tbl->last_flush = now;
1594 tbl->last_rand = now + tbl->parms.reachable_time * 20;
Simon Kelleybd89efc2006-05-12 14:56:08 -07001595
WANG Congd7480fd2014-11-10 15:59:36 -08001596 neigh_tables[index] = tbl;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001597}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001598EXPORT_SYMBOL(neigh_table_init);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001599
WANG Congd7480fd2014-11-10 15:59:36 -08001600int neigh_table_clear(int index, struct neigh_table *tbl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001601{
WANG Congd7480fd2014-11-10 15:59:36 -08001602 neigh_tables[index] = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001603 /* It is not clean... Fix it to unload IPv6 module safely */
Tejun Heoa5c30b32010-10-19 06:04:42 +00001604 cancel_delayed_work_sync(&tbl->gc_work);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001605 del_timer_sync(&tbl->proxy_timer);
1606 pneigh_queue_purge(&tbl->proxy_queue);
1607 neigh_ifdown(tbl, NULL);
1608 if (atomic_read(&tbl->entries))
Joe Perchese005d192012-05-16 19:58:40 +00001609 pr_crit("neighbour leakage\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001610
Eric Dumazet6193d2b2011-01-19 22:02:47 +00001611 call_rcu(&rcu_dereference_protected(tbl->nht, 1)->rcu,
1612 neigh_hash_free_rcu);
Eric Dumazetd6bf7812010-10-04 06:15:44 +00001613 tbl->nht = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001614
1615 kfree(tbl->phash_buckets);
1616 tbl->phash_buckets = NULL;
1617
Alexey Dobriyan3f192b52007-11-05 21:28:13 -08001618 remove_proc_entry(tbl->id, init_net.proc_net_stat);
1619
Kirill Korotaev3fcde742006-09-01 01:34:10 -07001620 free_percpu(tbl->stats);
1621 tbl->stats = NULL;
1622
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623 return 0;
1624}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09001625EXPORT_SYMBOL(neigh_table_clear);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001626
WANG Congd7480fd2014-11-10 15:59:36 -08001627static struct neigh_table *neigh_find_table(int family)
1628{
1629 struct neigh_table *tbl = NULL;
1630
1631 switch (family) {
1632 case AF_INET:
1633 tbl = neigh_tables[NEIGH_ARP_TABLE];
1634 break;
1635 case AF_INET6:
1636 tbl = neigh_tables[NEIGH_ND_TABLE];
1637 break;
1638 case AF_DECnet:
1639 tbl = neigh_tables[NEIGH_DN_TABLE];
1640 break;
1641 }
1642
1643 return tbl;
1644}
1645
David Ahernc21ef3e2017-04-16 09:48:24 -07001646static int neigh_delete(struct sk_buff *skb, struct nlmsghdr *nlh,
1647 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001648{
YOSHIFUJI Hideaki3b1e0a62008-03-26 02:26:21 +09001649 struct net *net = sock_net(skb->sk);
Thomas Grafa14a49d2006-08-07 17:53:08 -07001650 struct ndmsg *ndm;
1651 struct nlattr *dst_attr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001652 struct neigh_table *tbl;
WANG Congd7480fd2014-11-10 15:59:36 -08001653 struct neighbour *neigh;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001654 struct net_device *dev = NULL;
Thomas Grafa14a49d2006-08-07 17:53:08 -07001655 int err = -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001656
Eric Dumazet110b2492010-10-04 04:27:36 +00001657 ASSERT_RTNL();
Thomas Grafa14a49d2006-08-07 17:53:08 -07001658 if (nlmsg_len(nlh) < sizeof(*ndm))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001659 goto out;
1660
Thomas Grafa14a49d2006-08-07 17:53:08 -07001661 dst_attr = nlmsg_find_attr(nlh, sizeof(*ndm), NDA_DST);
1662 if (dst_attr == NULL)
1663 goto out;
1664
1665 ndm = nlmsg_data(nlh);
1666 if (ndm->ndm_ifindex) {
Eric Dumazet110b2492010-10-04 04:27:36 +00001667 dev = __dev_get_by_index(net, ndm->ndm_ifindex);
Thomas Grafa14a49d2006-08-07 17:53:08 -07001668 if (dev == NULL) {
1669 err = -ENODEV;
1670 goto out;
1671 }
1672 }
1673
WANG Congd7480fd2014-11-10 15:59:36 -08001674 tbl = neigh_find_table(ndm->ndm_family);
1675 if (tbl == NULL)
1676 return -EAFNOSUPPORT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001677
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +03001678 if (nla_len(dst_attr) < (int)tbl->key_len)
WANG Congd7480fd2014-11-10 15:59:36 -08001679 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001680
WANG Congd7480fd2014-11-10 15:59:36 -08001681 if (ndm->ndm_flags & NTF_PROXY) {
1682 err = pneigh_delete(tbl, net, nla_data(dst_attr), dev);
Eric Dumazet110b2492010-10-04 04:27:36 +00001683 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001684 }
WANG Congd7480fd2014-11-10 15:59:36 -08001685
1686 if (dev == NULL)
1687 goto out;
1688
1689 neigh = neigh_lookup(tbl, nla_data(dst_attr), dev);
1690 if (neigh == NULL) {
1691 err = -ENOENT;
1692 goto out;
1693 }
1694
1695 err = neigh_update(neigh, NULL, NUD_FAILED,
1696 NEIGH_UPDATE_F_OVERRIDE |
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07001697 NEIGH_UPDATE_F_ADMIN,
1698 NETLINK_CB(skb).portid);
Sowmini Varadhan50710342017-06-02 09:01:49 -07001699 write_lock_bh(&tbl->lock);
WANG Congd7480fd2014-11-10 15:59:36 -08001700 neigh_release(neigh);
Sowmini Varadhan50710342017-06-02 09:01:49 -07001701 neigh_remove_one(neigh, tbl);
1702 write_unlock_bh(&tbl->lock);
Thomas Grafa14a49d2006-08-07 17:53:08 -07001703
Linus Torvalds1da177e2005-04-16 15:20:36 -07001704out:
1705 return err;
1706}
1707
David Ahernc21ef3e2017-04-16 09:48:24 -07001708static int neigh_add(struct sk_buff *skb, struct nlmsghdr *nlh,
1709 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001710{
WANG Congd7480fd2014-11-10 15:59:36 -08001711 int flags = NEIGH_UPDATE_F_ADMIN | NEIGH_UPDATE_F_OVERRIDE;
YOSHIFUJI Hideaki3b1e0a62008-03-26 02:26:21 +09001712 struct net *net = sock_net(skb->sk);
Thomas Graf5208deb2006-08-07 17:55:40 -07001713 struct ndmsg *ndm;
1714 struct nlattr *tb[NDA_MAX+1];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001715 struct neigh_table *tbl;
1716 struct net_device *dev = NULL;
WANG Congd7480fd2014-11-10 15:59:36 -08001717 struct neighbour *neigh;
1718 void *dst, *lladdr;
Thomas Graf5208deb2006-08-07 17:55:40 -07001719 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001720
Eric Dumazet110b2492010-10-04 04:27:36 +00001721 ASSERT_RTNL();
David Ahernc21ef3e2017-04-16 09:48:24 -07001722 err = nlmsg_parse(nlh, sizeof(*ndm), tb, NDA_MAX, NULL, extack);
Thomas Graf5208deb2006-08-07 17:55:40 -07001723 if (err < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001724 goto out;
1725
Thomas Graf5208deb2006-08-07 17:55:40 -07001726 err = -EINVAL;
1727 if (tb[NDA_DST] == NULL)
1728 goto out;
1729
1730 ndm = nlmsg_data(nlh);
1731 if (ndm->ndm_ifindex) {
Eric Dumazet110b2492010-10-04 04:27:36 +00001732 dev = __dev_get_by_index(net, ndm->ndm_ifindex);
Thomas Graf5208deb2006-08-07 17:55:40 -07001733 if (dev == NULL) {
1734 err = -ENODEV;
1735 goto out;
1736 }
1737
1738 if (tb[NDA_LLADDR] && nla_len(tb[NDA_LLADDR]) < dev->addr_len)
Eric Dumazet110b2492010-10-04 04:27:36 +00001739 goto out;
Thomas Graf5208deb2006-08-07 17:55:40 -07001740 }
1741
WANG Congd7480fd2014-11-10 15:59:36 -08001742 tbl = neigh_find_table(ndm->ndm_family);
1743 if (tbl == NULL)
1744 return -EAFNOSUPPORT;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001745
Alexey Dobriyan01ccdf12017-09-23 23:03:04 +03001746 if (nla_len(tb[NDA_DST]) < (int)tbl->key_len)
WANG Congd7480fd2014-11-10 15:59:36 -08001747 goto out;
1748 dst = nla_data(tb[NDA_DST]);
1749 lladdr = tb[NDA_LLADDR] ? nla_data(tb[NDA_LLADDR]) : NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001750
WANG Congd7480fd2014-11-10 15:59:36 -08001751 if (ndm->ndm_flags & NTF_PROXY) {
1752 struct pneigh_entry *pn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001753
WANG Congd7480fd2014-11-10 15:59:36 -08001754 err = -ENOBUFS;
1755 pn = pneigh_lookup(tbl, net, dst, dev, 1);
1756 if (pn) {
1757 pn->flags = ndm->ndm_flags;
Eric Biederman0c5c2d32009-03-04 00:03:08 -08001758 err = 0;
WANG Congd7480fd2014-11-10 15:59:36 -08001759 }
Eric Dumazet110b2492010-10-04 04:27:36 +00001760 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001761 }
1762
WANG Congd7480fd2014-11-10 15:59:36 -08001763 if (dev == NULL)
1764 goto out;
1765
1766 neigh = neigh_lookup(tbl, dst, dev);
1767 if (neigh == NULL) {
1768 if (!(nlh->nlmsg_flags & NLM_F_CREATE)) {
1769 err = -ENOENT;
1770 goto out;
1771 }
1772
1773 neigh = __neigh_lookup_errno(tbl, dst, dev);
1774 if (IS_ERR(neigh)) {
1775 err = PTR_ERR(neigh);
1776 goto out;
1777 }
1778 } else {
1779 if (nlh->nlmsg_flags & NLM_F_EXCL) {
1780 err = -EEXIST;
1781 neigh_release(neigh);
1782 goto out;
1783 }
1784
1785 if (!(nlh->nlmsg_flags & NLM_F_REPLACE))
1786 flags &= ~NEIGH_UPDATE_F_OVERRIDE;
1787 }
1788
Roopa Prabhu9ce33e42018-04-24 13:49:34 -07001789 if (ndm->ndm_flags & NTF_EXT_LEARNED)
1790 flags |= NEIGH_UPDATE_F_EXT_LEARNED;
1791
WANG Congd7480fd2014-11-10 15:59:36 -08001792 if (ndm->ndm_flags & NTF_USE) {
1793 neigh_event_send(neigh, NULL);
1794 err = 0;
1795 } else
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07001796 err = neigh_update(neigh, lladdr, ndm->ndm_state, flags,
1797 NETLINK_CB(skb).portid);
WANG Congd7480fd2014-11-10 15:59:36 -08001798 neigh_release(neigh);
1799
Linus Torvalds1da177e2005-04-16 15:20:36 -07001800out:
1801 return err;
1802}
1803
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001804static int neightbl_fill_parms(struct sk_buff *skb, struct neigh_parms *parms)
1805{
Thomas Grafca860fb2006-08-07 18:00:18 -07001806 struct nlattr *nest;
1807
1808 nest = nla_nest_start(skb, NDTA_PARMS);
1809 if (nest == NULL)
1810 return -ENOBUFS;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001811
David S. Miller9a6308d2012-04-01 20:06:28 -04001812 if ((parms->dev &&
1813 nla_put_u32(skb, NDTPA_IFINDEX, parms->dev->ifindex)) ||
Reshetova, Elena63439442017-06-30 13:07:56 +03001814 nla_put_u32(skb, NDTPA_REFCNT, refcount_read(&parms->refcnt)) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001815 nla_put_u32(skb, NDTPA_QUEUE_LENBYTES,
1816 NEIGH_VAR(parms, QUEUE_LEN_BYTES)) ||
David S. Miller9a6308d2012-04-01 20:06:28 -04001817 /* approximative value for deprecated QUEUE_LEN (in packets) */
1818 nla_put_u32(skb, NDTPA_QUEUE_LEN,
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001819 NEIGH_VAR(parms, QUEUE_LEN_BYTES) / SKB_TRUESIZE(ETH_FRAME_LEN)) ||
1820 nla_put_u32(skb, NDTPA_PROXY_QLEN, NEIGH_VAR(parms, PROXY_QLEN)) ||
1821 nla_put_u32(skb, NDTPA_APP_PROBES, NEIGH_VAR(parms, APP_PROBES)) ||
1822 nla_put_u32(skb, NDTPA_UCAST_PROBES,
1823 NEIGH_VAR(parms, UCAST_PROBES)) ||
1824 nla_put_u32(skb, NDTPA_MCAST_PROBES,
1825 NEIGH_VAR(parms, MCAST_PROBES)) ||
YOSHIFUJI Hideaki/吉藤英明8da86462015-03-19 22:41:46 +09001826 nla_put_u32(skb, NDTPA_MCAST_REPROBES,
1827 NEIGH_VAR(parms, MCAST_REPROBES)) ||
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001828 nla_put_msecs(skb, NDTPA_REACHABLE_TIME, parms->reachable_time,
1829 NDTPA_PAD) ||
David S. Miller9a6308d2012-04-01 20:06:28 -04001830 nla_put_msecs(skb, NDTPA_BASE_REACHABLE_TIME,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001831 NEIGH_VAR(parms, BASE_REACHABLE_TIME), NDTPA_PAD) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001832 nla_put_msecs(skb, NDTPA_GC_STALETIME,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001833 NEIGH_VAR(parms, GC_STALETIME), NDTPA_PAD) ||
David S. Miller9a6308d2012-04-01 20:06:28 -04001834 nla_put_msecs(skb, NDTPA_DELAY_PROBE_TIME,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001835 NEIGH_VAR(parms, DELAY_PROBE_TIME), NDTPA_PAD) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001836 nla_put_msecs(skb, NDTPA_RETRANS_TIME,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001837 NEIGH_VAR(parms, RETRANS_TIME), NDTPA_PAD) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001838 nla_put_msecs(skb, NDTPA_ANYCAST_DELAY,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001839 NEIGH_VAR(parms, ANYCAST_DELAY), NDTPA_PAD) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001840 nla_put_msecs(skb, NDTPA_PROXY_DELAY,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001841 NEIGH_VAR(parms, PROXY_DELAY), NDTPA_PAD) ||
Jiri Pirko1f9248e2013-12-07 19:26:53 +01001842 nla_put_msecs(skb, NDTPA_LOCKTIME,
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001843 NEIGH_VAR(parms, LOCKTIME), NDTPA_PAD))
David S. Miller9a6308d2012-04-01 20:06:28 -04001844 goto nla_put_failure;
Thomas Grafca860fb2006-08-07 18:00:18 -07001845 return nla_nest_end(skb, nest);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001846
Thomas Grafca860fb2006-08-07 18:00:18 -07001847nla_put_failure:
Thomas Grafbc3ed282008-06-03 16:36:54 -07001848 nla_nest_cancel(skb, nest);
1849 return -EMSGSIZE;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001850}
1851
Thomas Grafca860fb2006-08-07 18:00:18 -07001852static int neightbl_fill_info(struct sk_buff *skb, struct neigh_table *tbl,
1853 u32 pid, u32 seq, int type, int flags)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001854{
1855 struct nlmsghdr *nlh;
1856 struct ndtmsg *ndtmsg;
1857
Thomas Grafca860fb2006-08-07 18:00:18 -07001858 nlh = nlmsg_put(skb, pid, seq, type, sizeof(*ndtmsg), flags);
1859 if (nlh == NULL)
Patrick McHardy26932562007-01-31 23:16:40 -08001860 return -EMSGSIZE;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001861
Thomas Grafca860fb2006-08-07 18:00:18 -07001862 ndtmsg = nlmsg_data(nlh);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001863
1864 read_lock_bh(&tbl->lock);
1865 ndtmsg->ndtm_family = tbl->family;
Patrick McHardy9ef1d4c2005-06-28 12:55:30 -07001866 ndtmsg->ndtm_pad1 = 0;
1867 ndtmsg->ndtm_pad2 = 0;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001868
David S. Miller9a6308d2012-04-01 20:06:28 -04001869 if (nla_put_string(skb, NDTA_NAME, tbl->id) ||
Nicolas Dichtel2175d872016-04-22 17:31:21 +02001870 nla_put_msecs(skb, NDTA_GC_INTERVAL, tbl->gc_interval, NDTA_PAD) ||
David S. Miller9a6308d2012-04-01 20:06:28 -04001871 nla_put_u32(skb, NDTA_THRESH1, tbl->gc_thresh1) ||
1872 nla_put_u32(skb, NDTA_THRESH2, tbl->gc_thresh2) ||
1873 nla_put_u32(skb, NDTA_THRESH3, tbl->gc_thresh3))
1874 goto nla_put_failure;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001875 {
1876 unsigned long now = jiffies;
1877 unsigned int flush_delta = now - tbl->last_flush;
1878 unsigned int rand_delta = now - tbl->last_rand;
Eric Dumazetd6bf7812010-10-04 06:15:44 +00001879 struct neigh_hash_table *nht;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001880 struct ndt_config ndc = {
1881 .ndtc_key_len = tbl->key_len,
1882 .ndtc_entry_size = tbl->entry_size,
1883 .ndtc_entries = atomic_read(&tbl->entries),
1884 .ndtc_last_flush = jiffies_to_msecs(flush_delta),
1885 .ndtc_last_rand = jiffies_to_msecs(rand_delta),
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001886 .ndtc_proxy_qlen = tbl->proxy_queue.qlen,
1887 };
1888
Eric Dumazetd6bf7812010-10-04 06:15:44 +00001889 rcu_read_lock_bh();
1890 nht = rcu_dereference_bh(tbl->nht);
David S. Miller2c2aba62011-12-28 15:06:58 -05001891 ndc.ndtc_hash_rnd = nht->hash_rnd[0];
David S. Millercd089332011-07-11 01:28:12 -07001892 ndc.ndtc_hash_mask = ((1 << nht->hash_shift) - 1);
Eric Dumazetd6bf7812010-10-04 06:15:44 +00001893 rcu_read_unlock_bh();
1894
David S. Miller9a6308d2012-04-01 20:06:28 -04001895 if (nla_put(skb, NDTA_CONFIG, sizeof(ndc), &ndc))
1896 goto nla_put_failure;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001897 }
1898
1899 {
1900 int cpu;
1901 struct ndt_stats ndst;
1902
1903 memset(&ndst, 0, sizeof(ndst));
1904
KAMEZAWA Hiroyuki6f912042006-04-10 22:52:50 -07001905 for_each_possible_cpu(cpu) {
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001906 struct neigh_statistics *st;
1907
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001908 st = per_cpu_ptr(tbl->stats, cpu);
1909 ndst.ndts_allocs += st->allocs;
1910 ndst.ndts_destroys += st->destroys;
1911 ndst.ndts_hash_grows += st->hash_grows;
1912 ndst.ndts_res_failed += st->res_failed;
1913 ndst.ndts_lookups += st->lookups;
1914 ndst.ndts_hits += st->hits;
1915 ndst.ndts_rcv_probes_mcast += st->rcv_probes_mcast;
1916 ndst.ndts_rcv_probes_ucast += st->rcv_probes_ucast;
1917 ndst.ndts_periodic_gc_runs += st->periodic_gc_runs;
1918 ndst.ndts_forced_gc_runs += st->forced_gc_runs;
Rick Jonesfb811392015-08-07 11:10:37 -07001919 ndst.ndts_table_fulls += st->table_fulls;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001920 }
1921
Nicolas Dichtelb6763382016-04-26 10:06:17 +02001922 if (nla_put_64bit(skb, NDTA_STATS, sizeof(ndst), &ndst,
1923 NDTA_PAD))
David S. Miller9a6308d2012-04-01 20:06:28 -04001924 goto nla_put_failure;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001925 }
1926
1927 BUG_ON(tbl->parms.dev);
1928 if (neightbl_fill_parms(skb, &tbl->parms) < 0)
Thomas Grafca860fb2006-08-07 18:00:18 -07001929 goto nla_put_failure;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001930
1931 read_unlock_bh(&tbl->lock);
Johannes Berg053c0952015-01-16 22:09:00 +01001932 nlmsg_end(skb, nlh);
1933 return 0;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001934
Thomas Grafca860fb2006-08-07 18:00:18 -07001935nla_put_failure:
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001936 read_unlock_bh(&tbl->lock);
Patrick McHardy26932562007-01-31 23:16:40 -08001937 nlmsg_cancel(skb, nlh);
1938 return -EMSGSIZE;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001939}
1940
Thomas Grafca860fb2006-08-07 18:00:18 -07001941static int neightbl_fill_param_info(struct sk_buff *skb,
1942 struct neigh_table *tbl,
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001943 struct neigh_parms *parms,
Thomas Grafca860fb2006-08-07 18:00:18 -07001944 u32 pid, u32 seq, int type,
1945 unsigned int flags)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001946{
1947 struct ndtmsg *ndtmsg;
1948 struct nlmsghdr *nlh;
1949
Thomas Grafca860fb2006-08-07 18:00:18 -07001950 nlh = nlmsg_put(skb, pid, seq, type, sizeof(*ndtmsg), flags);
1951 if (nlh == NULL)
Patrick McHardy26932562007-01-31 23:16:40 -08001952 return -EMSGSIZE;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001953
Thomas Grafca860fb2006-08-07 18:00:18 -07001954 ndtmsg = nlmsg_data(nlh);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001955
1956 read_lock_bh(&tbl->lock);
1957 ndtmsg->ndtm_family = tbl->family;
Patrick McHardy9ef1d4c2005-06-28 12:55:30 -07001958 ndtmsg->ndtm_pad1 = 0;
1959 ndtmsg->ndtm_pad2 = 0;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001960
Thomas Grafca860fb2006-08-07 18:00:18 -07001961 if (nla_put_string(skb, NDTA_NAME, tbl->id) < 0 ||
1962 neightbl_fill_parms(skb, parms) < 0)
1963 goto errout;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001964
1965 read_unlock_bh(&tbl->lock);
Johannes Berg053c0952015-01-16 22:09:00 +01001966 nlmsg_end(skb, nlh);
1967 return 0;
Thomas Grafca860fb2006-08-07 18:00:18 -07001968errout:
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001969 read_unlock_bh(&tbl->lock);
Patrick McHardy26932562007-01-31 23:16:40 -08001970 nlmsg_cancel(skb, nlh);
1971 return -EMSGSIZE;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07001972}
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09001973
Patrick McHardyef7c79e2007-06-05 12:38:30 -07001974static const struct nla_policy nl_neightbl_policy[NDTA_MAX+1] = {
Thomas Graf6b3f8672006-08-07 17:58:53 -07001975 [NDTA_NAME] = { .type = NLA_STRING },
1976 [NDTA_THRESH1] = { .type = NLA_U32 },
1977 [NDTA_THRESH2] = { .type = NLA_U32 },
1978 [NDTA_THRESH3] = { .type = NLA_U32 },
1979 [NDTA_GC_INTERVAL] = { .type = NLA_U64 },
1980 [NDTA_PARMS] = { .type = NLA_NESTED },
1981};
1982
Patrick McHardyef7c79e2007-06-05 12:38:30 -07001983static const struct nla_policy nl_ntbl_parm_policy[NDTPA_MAX+1] = {
Thomas Graf6b3f8672006-08-07 17:58:53 -07001984 [NDTPA_IFINDEX] = { .type = NLA_U32 },
1985 [NDTPA_QUEUE_LEN] = { .type = NLA_U32 },
1986 [NDTPA_PROXY_QLEN] = { .type = NLA_U32 },
1987 [NDTPA_APP_PROBES] = { .type = NLA_U32 },
1988 [NDTPA_UCAST_PROBES] = { .type = NLA_U32 },
1989 [NDTPA_MCAST_PROBES] = { .type = NLA_U32 },
YOSHIFUJI Hideaki/吉藤英明8da86462015-03-19 22:41:46 +09001990 [NDTPA_MCAST_REPROBES] = { .type = NLA_U32 },
Thomas Graf6b3f8672006-08-07 17:58:53 -07001991 [NDTPA_BASE_REACHABLE_TIME] = { .type = NLA_U64 },
1992 [NDTPA_GC_STALETIME] = { .type = NLA_U64 },
1993 [NDTPA_DELAY_PROBE_TIME] = { .type = NLA_U64 },
1994 [NDTPA_RETRANS_TIME] = { .type = NLA_U64 },
1995 [NDTPA_ANYCAST_DELAY] = { .type = NLA_U64 },
1996 [NDTPA_PROXY_DELAY] = { .type = NLA_U64 },
1997 [NDTPA_LOCKTIME] = { .type = NLA_U64 },
1998};
1999
David Ahernc21ef3e2017-04-16 09:48:24 -07002000static int neightbl_set(struct sk_buff *skb, struct nlmsghdr *nlh,
2001 struct netlink_ext_ack *extack)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002002{
YOSHIFUJI Hideaki3b1e0a62008-03-26 02:26:21 +09002003 struct net *net = sock_net(skb->sk);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002004 struct neigh_table *tbl;
Thomas Graf6b3f8672006-08-07 17:58:53 -07002005 struct ndtmsg *ndtmsg;
2006 struct nlattr *tb[NDTA_MAX+1];
WANG Congd7480fd2014-11-10 15:59:36 -08002007 bool found = false;
2008 int err, tidx;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002009
Thomas Graf6b3f8672006-08-07 17:58:53 -07002010 err = nlmsg_parse(nlh, sizeof(*ndtmsg), tb, NDTA_MAX,
David Ahernc21ef3e2017-04-16 09:48:24 -07002011 nl_neightbl_policy, extack);
Thomas Graf6b3f8672006-08-07 17:58:53 -07002012 if (err < 0)
2013 goto errout;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002014
Thomas Graf6b3f8672006-08-07 17:58:53 -07002015 if (tb[NDTA_NAME] == NULL) {
2016 err = -EINVAL;
2017 goto errout;
2018 }
2019
2020 ndtmsg = nlmsg_data(nlh);
WANG Congd7480fd2014-11-10 15:59:36 -08002021
2022 for (tidx = 0; tidx < NEIGH_NR_TABLES; tidx++) {
2023 tbl = neigh_tables[tidx];
2024 if (!tbl)
2025 continue;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002026 if (ndtmsg->ndtm_family && tbl->family != ndtmsg->ndtm_family)
2027 continue;
WANG Congd7480fd2014-11-10 15:59:36 -08002028 if (nla_strcmp(tb[NDTA_NAME], tbl->id) == 0) {
2029 found = true;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002030 break;
WANG Congd7480fd2014-11-10 15:59:36 -08002031 }
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002032 }
2033
WANG Congd7480fd2014-11-10 15:59:36 -08002034 if (!found)
2035 return -ENOENT;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002036
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09002037 /*
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002038 * We acquire tbl->lock to be nice to the periodic timers and
2039 * make sure they always see a consistent set of values.
2040 */
2041 write_lock_bh(&tbl->lock);
2042
Thomas Graf6b3f8672006-08-07 17:58:53 -07002043 if (tb[NDTA_PARMS]) {
2044 struct nlattr *tbp[NDTPA_MAX+1];
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002045 struct neigh_parms *p;
Thomas Graf6b3f8672006-08-07 17:58:53 -07002046 int i, ifindex = 0;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002047
Thomas Graf6b3f8672006-08-07 17:58:53 -07002048 err = nla_parse_nested(tbp, NDTPA_MAX, tb[NDTA_PARMS],
David Ahernc21ef3e2017-04-16 09:48:24 -07002049 nl_ntbl_parm_policy, extack);
Thomas Graf6b3f8672006-08-07 17:58:53 -07002050 if (err < 0)
2051 goto errout_tbl_lock;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002052
Thomas Graf6b3f8672006-08-07 17:58:53 -07002053 if (tbp[NDTPA_IFINDEX])
2054 ifindex = nla_get_u32(tbp[NDTPA_IFINDEX]);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002055
Tobias Klauser97fd5bc2009-07-13 11:17:49 -07002056 p = lookup_neigh_parms(tbl, net, ifindex);
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002057 if (p == NULL) {
2058 err = -ENOENT;
Thomas Graf6b3f8672006-08-07 17:58:53 -07002059 goto errout_tbl_lock;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002060 }
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002061
Thomas Graf6b3f8672006-08-07 17:58:53 -07002062 for (i = 1; i <= NDTPA_MAX; i++) {
2063 if (tbp[i] == NULL)
2064 continue;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002065
Thomas Graf6b3f8672006-08-07 17:58:53 -07002066 switch (i) {
2067 case NDTPA_QUEUE_LEN:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002068 NEIGH_VAR_SET(p, QUEUE_LEN_BYTES,
2069 nla_get_u32(tbp[i]) *
2070 SKB_TRUESIZE(ETH_FRAME_LEN));
Eric Dumazet8b5c1712011-11-09 12:07:14 +00002071 break;
2072 case NDTPA_QUEUE_LENBYTES:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002073 NEIGH_VAR_SET(p, QUEUE_LEN_BYTES,
2074 nla_get_u32(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002075 break;
2076 case NDTPA_PROXY_QLEN:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002077 NEIGH_VAR_SET(p, PROXY_QLEN,
2078 nla_get_u32(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002079 break;
2080 case NDTPA_APP_PROBES:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002081 NEIGH_VAR_SET(p, APP_PROBES,
2082 nla_get_u32(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002083 break;
2084 case NDTPA_UCAST_PROBES:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002085 NEIGH_VAR_SET(p, UCAST_PROBES,
2086 nla_get_u32(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002087 break;
2088 case NDTPA_MCAST_PROBES:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002089 NEIGH_VAR_SET(p, MCAST_PROBES,
2090 nla_get_u32(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002091 break;
YOSHIFUJI Hideaki/吉藤英明8da86462015-03-19 22:41:46 +09002092 case NDTPA_MCAST_REPROBES:
2093 NEIGH_VAR_SET(p, MCAST_REPROBES,
2094 nla_get_u32(tbp[i]));
2095 break;
Thomas Graf6b3f8672006-08-07 17:58:53 -07002096 case NDTPA_BASE_REACHABLE_TIME:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002097 NEIGH_VAR_SET(p, BASE_REACHABLE_TIME,
2098 nla_get_msecs(tbp[i]));
Jean-Francois Remy4bf69802015-01-14 04:22:39 +01002099 /* update reachable_time as well, otherwise, the change will
2100 * only be effective after the next time neigh_periodic_work
2101 * decides to recompute it (can be multiple minutes)
2102 */
2103 p->reachable_time =
2104 neigh_rand_reach_time(NEIGH_VAR(p, BASE_REACHABLE_TIME));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002105 break;
2106 case NDTPA_GC_STALETIME:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002107 NEIGH_VAR_SET(p, GC_STALETIME,
2108 nla_get_msecs(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002109 break;
2110 case NDTPA_DELAY_PROBE_TIME:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002111 NEIGH_VAR_SET(p, DELAY_PROBE_TIME,
2112 nla_get_msecs(tbp[i]));
Ido Schimmel2a4501a2016-07-05 11:27:42 +02002113 call_netevent_notifiers(NETEVENT_DELAY_PROBE_TIME_UPDATE, p);
Thomas Graf6b3f8672006-08-07 17:58:53 -07002114 break;
2115 case NDTPA_RETRANS_TIME:
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002116 NEIGH_VAR_SET(p, RETRANS_TIME,
2117 nla_get_msecs(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002118 break;
2119 case NDTPA_ANYCAST_DELAY:
Jiri Pirko39774582014-01-14 15:46:07 +01002120 NEIGH_VAR_SET(p, ANYCAST_DELAY,
2121 nla_get_msecs(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002122 break;
2123 case NDTPA_PROXY_DELAY:
Jiri Pirko39774582014-01-14 15:46:07 +01002124 NEIGH_VAR_SET(p, PROXY_DELAY,
2125 nla_get_msecs(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002126 break;
2127 case NDTPA_LOCKTIME:
Jiri Pirko39774582014-01-14 15:46:07 +01002128 NEIGH_VAR_SET(p, LOCKTIME,
2129 nla_get_msecs(tbp[i]));
Thomas Graf6b3f8672006-08-07 17:58:53 -07002130 break;
2131 }
2132 }
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002133 }
2134
Gao fengdc25c672013-06-20 10:01:34 +08002135 err = -ENOENT;
2136 if ((tb[NDTA_THRESH1] || tb[NDTA_THRESH2] ||
2137 tb[NDTA_THRESH3] || tb[NDTA_GC_INTERVAL]) &&
2138 !net_eq(net, &init_net))
2139 goto errout_tbl_lock;
2140
Thomas Graf6b3f8672006-08-07 17:58:53 -07002141 if (tb[NDTA_THRESH1])
2142 tbl->gc_thresh1 = nla_get_u32(tb[NDTA_THRESH1]);
2143
2144 if (tb[NDTA_THRESH2])
2145 tbl->gc_thresh2 = nla_get_u32(tb[NDTA_THRESH2]);
2146
2147 if (tb[NDTA_THRESH3])
2148 tbl->gc_thresh3 = nla_get_u32(tb[NDTA_THRESH3]);
2149
2150 if (tb[NDTA_GC_INTERVAL])
2151 tbl->gc_interval = nla_get_msecs(tb[NDTA_GC_INTERVAL]);
2152
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002153 err = 0;
2154
Thomas Graf6b3f8672006-08-07 17:58:53 -07002155errout_tbl_lock:
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002156 write_unlock_bh(&tbl->lock);
Thomas Graf6b3f8672006-08-07 17:58:53 -07002157errout:
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002158 return err;
2159}
2160
Thomas Grafc8822a42007-03-22 11:50:06 -07002161static int neightbl_dump_info(struct sk_buff *skb, struct netlink_callback *cb)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002162{
YOSHIFUJI Hideaki3b1e0a62008-03-26 02:26:21 +09002163 struct net *net = sock_net(skb->sk);
Thomas Grafca860fb2006-08-07 18:00:18 -07002164 int family, tidx, nidx = 0;
2165 int tbl_skip = cb->args[0];
2166 int neigh_skip = cb->args[1];
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002167 struct neigh_table *tbl;
2168
Thomas Grafca860fb2006-08-07 18:00:18 -07002169 family = ((struct rtgenmsg *) nlmsg_data(cb->nlh))->rtgen_family;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002170
WANG Congd7480fd2014-11-10 15:59:36 -08002171 for (tidx = 0; tidx < NEIGH_NR_TABLES; tidx++) {
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002172 struct neigh_parms *p;
2173
WANG Congd7480fd2014-11-10 15:59:36 -08002174 tbl = neigh_tables[tidx];
2175 if (!tbl)
2176 continue;
2177
Thomas Grafca860fb2006-08-07 18:00:18 -07002178 if (tidx < tbl_skip || (family && tbl->family != family))
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002179 continue;
2180
Eric W. Biederman15e47302012-09-07 20:12:54 +00002181 if (neightbl_fill_info(skb, tbl, NETLINK_CB(cb->skb).portid,
Thomas Grafca860fb2006-08-07 18:00:18 -07002182 cb->nlh->nlmsg_seq, RTM_NEWNEIGHTBL,
David S. Miller7b46a642015-01-18 23:36:08 -05002183 NLM_F_MULTI) < 0)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002184 break;
2185
Nicolas Dichtel75fbfd32014-10-29 19:29:31 +01002186 nidx = 0;
2187 p = list_next_entry(&tbl->parms, list);
2188 list_for_each_entry_from(p, &tbl->parms_list, list) {
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09002189 if (!net_eq(neigh_parms_net(p), net))
Eric W. Biederman426b5302008-01-24 00:13:18 -08002190 continue;
2191
Gautam Kachrooefc683f2009-02-06 00:52:04 -08002192 if (nidx < neigh_skip)
2193 goto next;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002194
Thomas Grafca860fb2006-08-07 18:00:18 -07002195 if (neightbl_fill_param_info(skb, tbl, p,
Eric W. Biederman15e47302012-09-07 20:12:54 +00002196 NETLINK_CB(cb->skb).portid,
Thomas Grafca860fb2006-08-07 18:00:18 -07002197 cb->nlh->nlmsg_seq,
2198 RTM_NEWNEIGHTBL,
David S. Miller7b46a642015-01-18 23:36:08 -05002199 NLM_F_MULTI) < 0)
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002200 goto out;
Gautam Kachrooefc683f2009-02-06 00:52:04 -08002201 next:
2202 nidx++;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002203 }
2204
Thomas Grafca860fb2006-08-07 18:00:18 -07002205 neigh_skip = 0;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002206 }
2207out:
Thomas Grafca860fb2006-08-07 18:00:18 -07002208 cb->args[0] = tidx;
2209 cb->args[1] = nidx;
Thomas Grafc7fb64d2005-06-18 22:50:55 -07002210
2211 return skb->len;
2212}
Linus Torvalds1da177e2005-04-16 15:20:36 -07002213
Thomas Graf8b8aec52006-08-07 17:56:37 -07002214static int neigh_fill_info(struct sk_buff *skb, struct neighbour *neigh,
2215 u32 pid, u32 seq, int type, unsigned int flags)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002216{
2217 unsigned long now = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002218 struct nda_cacheinfo ci;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002219 struct nlmsghdr *nlh;
2220 struct ndmsg *ndm;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002221
Thomas Graf8b8aec52006-08-07 17:56:37 -07002222 nlh = nlmsg_put(skb, pid, seq, type, sizeof(*ndm), flags);
2223 if (nlh == NULL)
Patrick McHardy26932562007-01-31 23:16:40 -08002224 return -EMSGSIZE;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002225
2226 ndm = nlmsg_data(nlh);
2227 ndm->ndm_family = neigh->ops->family;
Patrick McHardy9ef1d4c2005-06-28 12:55:30 -07002228 ndm->ndm_pad1 = 0;
2229 ndm->ndm_pad2 = 0;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002230 ndm->ndm_flags = neigh->flags;
2231 ndm->ndm_type = neigh->type;
2232 ndm->ndm_ifindex = neigh->dev->ifindex;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002233
David S. Miller9a6308d2012-04-01 20:06:28 -04002234 if (nla_put(skb, NDA_DST, neigh->tbl->key_len, neigh->primary_key))
2235 goto nla_put_failure;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002236
2237 read_lock_bh(&neigh->lock);
2238 ndm->ndm_state = neigh->nud_state;
Eric Dumazet0ed8ddf2010-10-07 10:44:07 +00002239 if (neigh->nud_state & NUD_VALID) {
2240 char haddr[MAX_ADDR_LEN];
2241
2242 neigh_ha_snapshot(haddr, neigh, neigh->dev);
2243 if (nla_put(skb, NDA_LLADDR, neigh->dev->addr_len, haddr) < 0) {
2244 read_unlock_bh(&neigh->lock);
2245 goto nla_put_failure;
2246 }
Thomas Graf8b8aec52006-08-07 17:56:37 -07002247 }
2248
Stephen Hemmingerb9f5f522008-06-03 16:03:15 -07002249 ci.ndm_used = jiffies_to_clock_t(now - neigh->used);
2250 ci.ndm_confirmed = jiffies_to_clock_t(now - neigh->confirmed);
2251 ci.ndm_updated = jiffies_to_clock_t(now - neigh->updated);
Reshetova, Elena9f237432017-06-30 13:07:55 +03002252 ci.ndm_refcnt = refcount_read(&neigh->refcnt) - 1;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002253 read_unlock_bh(&neigh->lock);
2254
David S. Miller9a6308d2012-04-01 20:06:28 -04002255 if (nla_put_u32(skb, NDA_PROBES, atomic_read(&neigh->probes)) ||
2256 nla_put(skb, NDA_CACHEINFO, sizeof(ci), &ci))
2257 goto nla_put_failure;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002258
Johannes Berg053c0952015-01-16 22:09:00 +01002259 nlmsg_end(skb, nlh);
2260 return 0;
Thomas Graf8b8aec52006-08-07 17:56:37 -07002261
2262nla_put_failure:
Patrick McHardy26932562007-01-31 23:16:40 -08002263 nlmsg_cancel(skb, nlh);
2264 return -EMSGSIZE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002265}
2266
Tony Zelenoff84920c12012-01-26 22:28:58 +00002267static int pneigh_fill_info(struct sk_buff *skb, struct pneigh_entry *pn,
2268 u32 pid, u32 seq, int type, unsigned int flags,
2269 struct neigh_table *tbl)
2270{
2271 struct nlmsghdr *nlh;
2272 struct ndmsg *ndm;
2273
2274 nlh = nlmsg_put(skb, pid, seq, type, sizeof(*ndm), flags);
2275 if (nlh == NULL)
2276 return -EMSGSIZE;
2277
2278 ndm = nlmsg_data(nlh);
2279 ndm->ndm_family = tbl->family;
2280 ndm->ndm_pad1 = 0;
2281 ndm->ndm_pad2 = 0;
2282 ndm->ndm_flags = pn->flags | NTF_PROXY;
Jun Zhao545469f2014-07-26 00:38:59 +08002283 ndm->ndm_type = RTN_UNICAST;
Konstantin Khlebnikov6adc5fd2015-12-01 01:14:48 +03002284 ndm->ndm_ifindex = pn->dev ? pn->dev->ifindex : 0;
Tony Zelenoff84920c12012-01-26 22:28:58 +00002285 ndm->ndm_state = NUD_NONE;
2286
David S. Miller9a6308d2012-04-01 20:06:28 -04002287 if (nla_put(skb, NDA_DST, tbl->key_len, pn->key))
2288 goto nla_put_failure;
Tony Zelenoff84920c12012-01-26 22:28:58 +00002289
Johannes Berg053c0952015-01-16 22:09:00 +01002290 nlmsg_end(skb, nlh);
2291 return 0;
Tony Zelenoff84920c12012-01-26 22:28:58 +00002292
2293nla_put_failure:
2294 nlmsg_cancel(skb, nlh);
2295 return -EMSGSIZE;
2296}
2297
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07002298static void neigh_update_notify(struct neighbour *neigh, u32 nlmsg_pid)
Thomas Grafd961db32007-08-08 23:12:56 -07002299{
2300 call_netevent_notifiers(NETEVENT_NEIGH_UPDATE, neigh);
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07002301 __neigh_notify(neigh, RTM_NEWNEIGH, 0, nlmsg_pid);
Thomas Grafd961db32007-08-08 23:12:56 -07002302}
Linus Torvalds1da177e2005-04-16 15:20:36 -07002303
David Ahern21fdd092015-09-29 09:32:03 -07002304static bool neigh_master_filtered(struct net_device *dev, int master_idx)
2305{
2306 struct net_device *master;
2307
2308 if (!master_idx)
2309 return false;
2310
2311 master = netdev_master_upper_dev_get(dev);
2312 if (!master || master->ifindex != master_idx)
2313 return true;
2314
2315 return false;
2316}
2317
David Ahern16660f02015-10-03 11:43:46 -07002318static bool neigh_ifindex_filtered(struct net_device *dev, int filter_idx)
2319{
2320 if (filter_idx && dev->ifindex != filter_idx)
2321 return true;
2322
2323 return false;
2324}
2325
Linus Torvalds1da177e2005-04-16 15:20:36 -07002326static int neigh_dump_table(struct neigh_table *tbl, struct sk_buff *skb,
2327 struct netlink_callback *cb)
2328{
Eric Dumazet767e97e2010-10-06 17:49:21 -07002329 struct net *net = sock_net(skb->sk);
David Ahern21fdd092015-09-29 09:32:03 -07002330 const struct nlmsghdr *nlh = cb->nlh;
2331 struct nlattr *tb[NDA_MAX + 1];
Linus Torvalds1da177e2005-04-16 15:20:36 -07002332 struct neighbour *n;
2333 int rc, h, s_h = cb->args[1];
2334 int idx, s_idx = idx = cb->args[2];
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002335 struct neigh_hash_table *nht;
David Ahern16660f02015-10-03 11:43:46 -07002336 int filter_master_idx = 0, filter_idx = 0;
David Ahern21fdd092015-09-29 09:32:03 -07002337 unsigned int flags = NLM_F_MULTI;
2338 int err;
2339
Johannes Bergfceb6432017-04-12 14:34:07 +02002340 err = nlmsg_parse(nlh, sizeof(struct ndmsg), tb, NDA_MAX, NULL, NULL);
David Ahern21fdd092015-09-29 09:32:03 -07002341 if (!err) {
Eric Dumazet7dd07c12018-04-11 14:46:00 -07002342 if (tb[NDA_IFINDEX]) {
2343 if (nla_len(tb[NDA_IFINDEX]) != sizeof(u32))
2344 return -EINVAL;
David Ahern16660f02015-10-03 11:43:46 -07002345 filter_idx = nla_get_u32(tb[NDA_IFINDEX]);
Eric Dumazet7dd07c12018-04-11 14:46:00 -07002346 }
2347 if (tb[NDA_MASTER]) {
2348 if (nla_len(tb[NDA_MASTER]) != sizeof(u32))
2349 return -EINVAL;
David Ahern21fdd092015-09-29 09:32:03 -07002350 filter_master_idx = nla_get_u32(tb[NDA_MASTER]);
Eric Dumazet7dd07c12018-04-11 14:46:00 -07002351 }
David Ahern16660f02015-10-03 11:43:46 -07002352 if (filter_idx || filter_master_idx)
David Ahern21fdd092015-09-29 09:32:03 -07002353 flags |= NLM_F_DUMP_FILTERED;
2354 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002355
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002356 rcu_read_lock_bh();
2357 nht = rcu_dereference_bh(tbl->nht);
2358
Eric Dumazet4bd6683b2012-06-07 04:58:35 +00002359 for (h = s_h; h < (1 << nht->hash_shift); h++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002360 if (h > s_h)
2361 s_idx = 0;
Eric Dumazet767e97e2010-10-06 17:49:21 -07002362 for (n = rcu_dereference_bh(nht->hash_buckets[h]), idx = 0;
2363 n != NULL;
2364 n = rcu_dereference_bh(n->next)) {
Zhang Shengju18502ac2016-11-30 11:24:42 +08002365 if (idx < s_idx || !net_eq(dev_net(n->dev), net))
2366 goto next;
2367 if (neigh_ifindex_filtered(n->dev, filter_idx) ||
2368 neigh_master_filtered(n->dev, filter_master_idx))
Gautam Kachrooefc683f2009-02-06 00:52:04 -08002369 goto next;
Eric W. Biederman15e47302012-09-07 20:12:54 +00002370 if (neigh_fill_info(skb, n, NETLINK_CB(cb->skb).portid,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002371 cb->nlh->nlmsg_seq,
Jamal Hadi Salimb6544c02005-06-18 22:54:12 -07002372 RTM_NEWNEIGH,
David Ahern21fdd092015-09-29 09:32:03 -07002373 flags) < 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002374 rc = -1;
2375 goto out;
2376 }
Eric Dumazet767e97e2010-10-06 17:49:21 -07002377next:
Gautam Kachrooefc683f2009-02-06 00:52:04 -08002378 idx++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002379 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002380 }
2381 rc = skb->len;
2382out:
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002383 rcu_read_unlock_bh();
Linus Torvalds1da177e2005-04-16 15:20:36 -07002384 cb->args[1] = h;
2385 cb->args[2] = idx;
2386 return rc;
2387}
2388
Tony Zelenoff84920c12012-01-26 22:28:58 +00002389static int pneigh_dump_table(struct neigh_table *tbl, struct sk_buff *skb,
2390 struct netlink_callback *cb)
2391{
2392 struct pneigh_entry *n;
2393 struct net *net = sock_net(skb->sk);
2394 int rc, h, s_h = cb->args[3];
2395 int idx, s_idx = idx = cb->args[4];
2396
2397 read_lock_bh(&tbl->lock);
2398
Eric Dumazet4bd6683b2012-06-07 04:58:35 +00002399 for (h = s_h; h <= PNEIGH_HASHMASK; h++) {
Tony Zelenoff84920c12012-01-26 22:28:58 +00002400 if (h > s_h)
2401 s_idx = 0;
2402 for (n = tbl->phash_buckets[h], idx = 0; n; n = n->next) {
Zhang Shengju18502ac2016-11-30 11:24:42 +08002403 if (idx < s_idx || pneigh_net(n) != net)
Tony Zelenoff84920c12012-01-26 22:28:58 +00002404 goto next;
Eric W. Biederman15e47302012-09-07 20:12:54 +00002405 if (pneigh_fill_info(skb, n, NETLINK_CB(cb->skb).portid,
Tony Zelenoff84920c12012-01-26 22:28:58 +00002406 cb->nlh->nlmsg_seq,
2407 RTM_NEWNEIGH,
David S. Miller7b46a642015-01-18 23:36:08 -05002408 NLM_F_MULTI, tbl) < 0) {
Tony Zelenoff84920c12012-01-26 22:28:58 +00002409 read_unlock_bh(&tbl->lock);
2410 rc = -1;
2411 goto out;
2412 }
2413 next:
2414 idx++;
2415 }
2416 }
2417
2418 read_unlock_bh(&tbl->lock);
2419 rc = skb->len;
2420out:
2421 cb->args[3] = h;
2422 cb->args[4] = idx;
2423 return rc;
2424
2425}
2426
Thomas Grafc8822a42007-03-22 11:50:06 -07002427static int neigh_dump_info(struct sk_buff *skb, struct netlink_callback *cb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002428{
2429 struct neigh_table *tbl;
2430 int t, family, s_t;
Tony Zelenoff84920c12012-01-26 22:28:58 +00002431 int proxy = 0;
Eric Dumazet4bd6683b2012-06-07 04:58:35 +00002432 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002433
Thomas Graf8b8aec52006-08-07 17:56:37 -07002434 family = ((struct rtgenmsg *) nlmsg_data(cb->nlh))->rtgen_family;
Tony Zelenoff84920c12012-01-26 22:28:58 +00002435
2436 /* check for full ndmsg structure presence, family member is
2437 * the same for both structures
2438 */
2439 if (nlmsg_len(cb->nlh) >= sizeof(struct ndmsg) &&
2440 ((struct ndmsg *) nlmsg_data(cb->nlh))->ndm_flags == NTF_PROXY)
2441 proxy = 1;
2442
Linus Torvalds1da177e2005-04-16 15:20:36 -07002443 s_t = cb->args[0];
2444
WANG Congd7480fd2014-11-10 15:59:36 -08002445 for (t = 0; t < NEIGH_NR_TABLES; t++) {
2446 tbl = neigh_tables[t];
2447
2448 if (!tbl)
2449 continue;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002450 if (t < s_t || (family && tbl->family != family))
2451 continue;
2452 if (t > s_t)
2453 memset(&cb->args[1], 0, sizeof(cb->args) -
2454 sizeof(cb->args[0]));
Tony Zelenoff84920c12012-01-26 22:28:58 +00002455 if (proxy)
2456 err = pneigh_dump_table(tbl, skb, cb);
2457 else
2458 err = neigh_dump_table(tbl, skb, cb);
Eric Dumazet4bd6683b2012-06-07 04:58:35 +00002459 if (err < 0)
2460 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002461 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002462
2463 cb->args[0] = t;
2464 return skb->len;
2465}
2466
2467void neigh_for_each(struct neigh_table *tbl, void (*cb)(struct neighbour *, void *), void *cookie)
2468{
2469 int chain;
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002470 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002471
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002472 rcu_read_lock_bh();
2473 nht = rcu_dereference_bh(tbl->nht);
2474
Eric Dumazet767e97e2010-10-06 17:49:21 -07002475 read_lock(&tbl->lock); /* avoid resizes */
David S. Millercd089332011-07-11 01:28:12 -07002476 for (chain = 0; chain < (1 << nht->hash_shift); chain++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002477 struct neighbour *n;
2478
Eric Dumazet767e97e2010-10-06 17:49:21 -07002479 for (n = rcu_dereference_bh(nht->hash_buckets[chain]);
2480 n != NULL;
2481 n = rcu_dereference_bh(n->next))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002482 cb(n, cookie);
2483 }
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002484 read_unlock(&tbl->lock);
2485 rcu_read_unlock_bh();
Linus Torvalds1da177e2005-04-16 15:20:36 -07002486}
2487EXPORT_SYMBOL(neigh_for_each);
2488
2489/* The tbl->lock must be held as a writer and BH disabled. */
2490void __neigh_for_each_release(struct neigh_table *tbl,
2491 int (*cb)(struct neighbour *))
2492{
2493 int chain;
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002494 struct neigh_hash_table *nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002495
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002496 nht = rcu_dereference_protected(tbl->nht,
2497 lockdep_is_held(&tbl->lock));
David S. Millercd089332011-07-11 01:28:12 -07002498 for (chain = 0; chain < (1 << nht->hash_shift); chain++) {
Eric Dumazet767e97e2010-10-06 17:49:21 -07002499 struct neighbour *n;
2500 struct neighbour __rcu **np;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002501
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002502 np = &nht->hash_buckets[chain];
Eric Dumazet767e97e2010-10-06 17:49:21 -07002503 while ((n = rcu_dereference_protected(*np,
2504 lockdep_is_held(&tbl->lock))) != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002505 int release;
2506
2507 write_lock(&n->lock);
2508 release = cb(n);
2509 if (release) {
Eric Dumazet767e97e2010-10-06 17:49:21 -07002510 rcu_assign_pointer(*np,
2511 rcu_dereference_protected(n->next,
2512 lockdep_is_held(&tbl->lock)));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002513 n->dead = 1;
2514 } else
2515 np = &n->next;
2516 write_unlock(&n->lock);
Thomas Graf4f494552007-08-08 23:12:36 -07002517 if (release)
2518 neigh_cleanup_and_release(n);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002519 }
2520 }
2521}
2522EXPORT_SYMBOL(__neigh_for_each_release);
2523
Eric W. Biedermanb79bda32015-03-07 16:25:56 -06002524int neigh_xmit(int index, struct net_device *dev,
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002525 const void *addr, struct sk_buff *skb)
2526{
Eric W. Biedermanb79bda32015-03-07 16:25:56 -06002527 int err = -EAFNOSUPPORT;
2528 if (likely(index < NEIGH_NR_TABLES)) {
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002529 struct neigh_table *tbl;
2530 struct neighbour *neigh;
2531
Eric W. Biedermanb79bda32015-03-07 16:25:56 -06002532 tbl = neigh_tables[index];
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002533 if (!tbl)
2534 goto out;
David Barrosob560f032016-06-28 11:16:43 +03002535 rcu_read_lock_bh();
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002536 neigh = __neigh_lookup_noref(tbl, addr, dev);
2537 if (!neigh)
2538 neigh = __neigh_create(tbl, addr, dev, false);
2539 err = PTR_ERR(neigh);
David Barrosob560f032016-06-28 11:16:43 +03002540 if (IS_ERR(neigh)) {
2541 rcu_read_unlock_bh();
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002542 goto out_kfree_skb;
David Barrosob560f032016-06-28 11:16:43 +03002543 }
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002544 err = neigh->output(neigh, skb);
David Barrosob560f032016-06-28 11:16:43 +03002545 rcu_read_unlock_bh();
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002546 }
Eric W. Biedermanb79bda32015-03-07 16:25:56 -06002547 else if (index == NEIGH_LINK_TABLE) {
2548 err = dev_hard_header(skb, dev, ntohs(skb->protocol),
2549 addr, NULL, skb->len);
2550 if (err < 0)
2551 goto out_kfree_skb;
2552 err = dev_queue_xmit(skb);
2553 }
Eric W. Biederman4fd3d7d2015-03-03 17:11:16 -06002554out:
2555 return err;
2556out_kfree_skb:
2557 kfree_skb(skb);
2558 goto out;
2559}
2560EXPORT_SYMBOL(neigh_xmit);
2561
Linus Torvalds1da177e2005-04-16 15:20:36 -07002562#ifdef CONFIG_PROC_FS
2563
2564static struct neighbour *neigh_get_first(struct seq_file *seq)
2565{
2566 struct neigh_seq_state *state = seq->private;
YOSHIFUJI Hideaki12188542008-03-26 02:36:06 +09002567 struct net *net = seq_file_net(seq);
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002568 struct neigh_hash_table *nht = state->nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002569 struct neighbour *n = NULL;
2570 int bucket = state->bucket;
2571
2572 state->flags &= ~NEIGH_SEQ_IS_PNEIGH;
David S. Millercd089332011-07-11 01:28:12 -07002573 for (bucket = 0; bucket < (1 << nht->hash_shift); bucket++) {
Eric Dumazet767e97e2010-10-06 17:49:21 -07002574 n = rcu_dereference_bh(nht->hash_buckets[bucket]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002575
2576 while (n) {
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09002577 if (!net_eq(dev_net(n->dev), net))
Eric W. Biederman426b5302008-01-24 00:13:18 -08002578 goto next;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002579 if (state->neigh_sub_iter) {
2580 loff_t fakep = 0;
2581 void *v;
2582
2583 v = state->neigh_sub_iter(state, n, &fakep);
2584 if (!v)
2585 goto next;
2586 }
2587 if (!(state->flags & NEIGH_SEQ_SKIP_NOARP))
2588 break;
2589 if (n->nud_state & ~NUD_NOARP)
2590 break;
Eric Dumazet767e97e2010-10-06 17:49:21 -07002591next:
2592 n = rcu_dereference_bh(n->next);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002593 }
2594
2595 if (n)
2596 break;
2597 }
2598 state->bucket = bucket;
2599
2600 return n;
2601}
2602
2603static struct neighbour *neigh_get_next(struct seq_file *seq,
2604 struct neighbour *n,
2605 loff_t *pos)
2606{
2607 struct neigh_seq_state *state = seq->private;
YOSHIFUJI Hideaki12188542008-03-26 02:36:06 +09002608 struct net *net = seq_file_net(seq);
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002609 struct neigh_hash_table *nht = state->nht;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002610
2611 if (state->neigh_sub_iter) {
2612 void *v = state->neigh_sub_iter(state, n, pos);
2613 if (v)
2614 return n;
2615 }
Eric Dumazet767e97e2010-10-06 17:49:21 -07002616 n = rcu_dereference_bh(n->next);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002617
2618 while (1) {
2619 while (n) {
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09002620 if (!net_eq(dev_net(n->dev), net))
Eric W. Biederman426b5302008-01-24 00:13:18 -08002621 goto next;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002622 if (state->neigh_sub_iter) {
2623 void *v = state->neigh_sub_iter(state, n, pos);
2624 if (v)
2625 return n;
2626 goto next;
2627 }
2628 if (!(state->flags & NEIGH_SEQ_SKIP_NOARP))
2629 break;
2630
2631 if (n->nud_state & ~NUD_NOARP)
2632 break;
Eric Dumazet767e97e2010-10-06 17:49:21 -07002633next:
2634 n = rcu_dereference_bh(n->next);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002635 }
2636
2637 if (n)
2638 break;
2639
David S. Millercd089332011-07-11 01:28:12 -07002640 if (++state->bucket >= (1 << nht->hash_shift))
Linus Torvalds1da177e2005-04-16 15:20:36 -07002641 break;
2642
Eric Dumazet767e97e2010-10-06 17:49:21 -07002643 n = rcu_dereference_bh(nht->hash_buckets[state->bucket]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002644 }
2645
2646 if (n && pos)
2647 --(*pos);
2648 return n;
2649}
2650
2651static struct neighbour *neigh_get_idx(struct seq_file *seq, loff_t *pos)
2652{
2653 struct neighbour *n = neigh_get_first(seq);
2654
2655 if (n) {
Chris Larson745e2032008-08-03 01:10:55 -07002656 --(*pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002657 while (*pos) {
2658 n = neigh_get_next(seq, n, pos);
2659 if (!n)
2660 break;
2661 }
2662 }
2663 return *pos ? NULL : n;
2664}
2665
2666static struct pneigh_entry *pneigh_get_first(struct seq_file *seq)
2667{
2668 struct neigh_seq_state *state = seq->private;
YOSHIFUJI Hideaki12188542008-03-26 02:36:06 +09002669 struct net *net = seq_file_net(seq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002670 struct neigh_table *tbl = state->tbl;
2671 struct pneigh_entry *pn = NULL;
2672 int bucket = state->bucket;
2673
2674 state->flags |= NEIGH_SEQ_IS_PNEIGH;
2675 for (bucket = 0; bucket <= PNEIGH_HASHMASK; bucket++) {
2676 pn = tbl->phash_buckets[bucket];
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09002677 while (pn && !net_eq(pneigh_net(pn), net))
Eric W. Biederman426b5302008-01-24 00:13:18 -08002678 pn = pn->next;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002679 if (pn)
2680 break;
2681 }
2682 state->bucket = bucket;
2683
2684 return pn;
2685}
2686
2687static struct pneigh_entry *pneigh_get_next(struct seq_file *seq,
2688 struct pneigh_entry *pn,
2689 loff_t *pos)
2690{
2691 struct neigh_seq_state *state = seq->private;
YOSHIFUJI Hideaki12188542008-03-26 02:36:06 +09002692 struct net *net = seq_file_net(seq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002693 struct neigh_table *tbl = state->tbl;
2694
Jorge Boncompte [DTI2]df07a942011-11-25 13:24:49 -05002695 do {
2696 pn = pn->next;
2697 } while (pn && !net_eq(pneigh_net(pn), net));
2698
Linus Torvalds1da177e2005-04-16 15:20:36 -07002699 while (!pn) {
2700 if (++state->bucket > PNEIGH_HASHMASK)
2701 break;
2702 pn = tbl->phash_buckets[state->bucket];
YOSHIFUJI Hideaki878628f2008-03-26 03:57:35 +09002703 while (pn && !net_eq(pneigh_net(pn), net))
Eric W. Biederman426b5302008-01-24 00:13:18 -08002704 pn = pn->next;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002705 if (pn)
2706 break;
2707 }
2708
2709 if (pn && pos)
2710 --(*pos);
2711
2712 return pn;
2713}
2714
2715static struct pneigh_entry *pneigh_get_idx(struct seq_file *seq, loff_t *pos)
2716{
2717 struct pneigh_entry *pn = pneigh_get_first(seq);
2718
2719 if (pn) {
Chris Larson745e2032008-08-03 01:10:55 -07002720 --(*pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002721 while (*pos) {
2722 pn = pneigh_get_next(seq, pn, pos);
2723 if (!pn)
2724 break;
2725 }
2726 }
2727 return *pos ? NULL : pn;
2728}
2729
2730static void *neigh_get_idx_any(struct seq_file *seq, loff_t *pos)
2731{
2732 struct neigh_seq_state *state = seq->private;
2733 void *rc;
Chris Larson745e2032008-08-03 01:10:55 -07002734 loff_t idxpos = *pos;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002735
Chris Larson745e2032008-08-03 01:10:55 -07002736 rc = neigh_get_idx(seq, &idxpos);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002737 if (!rc && !(state->flags & NEIGH_SEQ_NEIGH_ONLY))
Chris Larson745e2032008-08-03 01:10:55 -07002738 rc = pneigh_get_idx(seq, &idxpos);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002739
2740 return rc;
2741}
2742
2743void *neigh_seq_start(struct seq_file *seq, loff_t *pos, struct neigh_table *tbl, unsigned int neigh_seq_flags)
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002744 __acquires(rcu_bh)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002745{
2746 struct neigh_seq_state *state = seq->private;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002747
2748 state->tbl = tbl;
2749 state->bucket = 0;
2750 state->flags = (neigh_seq_flags & ~NEIGH_SEQ_IS_PNEIGH);
2751
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002752 rcu_read_lock_bh();
2753 state->nht = rcu_dereference_bh(tbl->nht);
Eric Dumazet767e97e2010-10-06 17:49:21 -07002754
Chris Larson745e2032008-08-03 01:10:55 -07002755 return *pos ? neigh_get_idx_any(seq, pos) : SEQ_START_TOKEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002756}
2757EXPORT_SYMBOL(neigh_seq_start);
2758
2759void *neigh_seq_next(struct seq_file *seq, void *v, loff_t *pos)
2760{
2761 struct neigh_seq_state *state;
2762 void *rc;
2763
2764 if (v == SEQ_START_TOKEN) {
Chris Larsonbff69732008-08-03 01:02:41 -07002765 rc = neigh_get_first(seq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002766 goto out;
2767 }
2768
2769 state = seq->private;
2770 if (!(state->flags & NEIGH_SEQ_IS_PNEIGH)) {
2771 rc = neigh_get_next(seq, v, NULL);
2772 if (rc)
2773 goto out;
2774 if (!(state->flags & NEIGH_SEQ_NEIGH_ONLY))
2775 rc = pneigh_get_first(seq);
2776 } else {
2777 BUG_ON(state->flags & NEIGH_SEQ_NEIGH_ONLY);
2778 rc = pneigh_get_next(seq, v, NULL);
2779 }
2780out:
2781 ++(*pos);
2782 return rc;
2783}
2784EXPORT_SYMBOL(neigh_seq_next);
2785
2786void neigh_seq_stop(struct seq_file *seq, void *v)
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002787 __releases(rcu_bh)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002788{
Eric Dumazetd6bf7812010-10-04 06:15:44 +00002789 rcu_read_unlock_bh();
Linus Torvalds1da177e2005-04-16 15:20:36 -07002790}
2791EXPORT_SYMBOL(neigh_seq_stop);
2792
2793/* statistics via seq_file */
2794
2795static void *neigh_stat_seq_start(struct seq_file *seq, loff_t *pos)
2796{
Christoph Hellwig71a50532018-04-15 10:16:41 +02002797 struct neigh_table *tbl = PDE_DATA(file_inode(seq->file));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002798 int cpu;
2799
2800 if (*pos == 0)
2801 return SEQ_START_TOKEN;
YOSHIFUJI Hideaki4ec93ed2007-02-09 23:24:36 +09002802
Rusty Russell0f23174a2008-12-29 12:23:42 +00002803 for (cpu = *pos-1; cpu < nr_cpu_ids; ++cpu) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002804 if (!cpu_possible(cpu))
2805 continue;
2806 *pos = cpu+1;
2807 return per_cpu_ptr(tbl->stats, cpu);
2808 }
2809 return NULL;
2810}
2811
2812static void *neigh_stat_seq_next(struct seq_file *seq, void *v, loff_t *pos)
2813{
Christoph Hellwig71a50532018-04-15 10:16:41 +02002814 struct neigh_table *tbl = PDE_DATA(file_inode(seq->file));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002815 int cpu;
2816
Rusty Russell0f23174a2008-12-29 12:23:42 +00002817 for (cpu = *pos; cpu < nr_cpu_ids; ++cpu) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002818 if (!cpu_possible(cpu))
2819 continue;
2820 *pos = cpu+1;
2821 return per_cpu_ptr(tbl->stats, cpu);
2822 }
2823 return NULL;
2824}
2825
2826static void neigh_stat_seq_stop(struct seq_file *seq, void *v)
2827{
2828
2829}
2830
2831static int neigh_stat_seq_show(struct seq_file *seq, void *v)
2832{
Christoph Hellwig71a50532018-04-15 10:16:41 +02002833 struct neigh_table *tbl = PDE_DATA(file_inode(seq->file));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002834 struct neigh_statistics *st = v;
2835
2836 if (v == SEQ_START_TOKEN) {
Rick Jonesfb811392015-08-07 11:10:37 -07002837 seq_printf(seq, "entries allocs destroys hash_grows lookups hits res_failed rcv_probes_mcast rcv_probes_ucast periodic_gc_runs forced_gc_runs unresolved_discards table_fulls\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002838 return 0;
2839 }
2840
2841 seq_printf(seq, "%08x %08lx %08lx %08lx %08lx %08lx %08lx "
Rick Jonesfb811392015-08-07 11:10:37 -07002842 "%08lx %08lx %08lx %08lx %08lx %08lx\n",
Linus Torvalds1da177e2005-04-16 15:20:36 -07002843 atomic_read(&tbl->entries),
2844
2845 st->allocs,
2846 st->destroys,
2847 st->hash_grows,
2848
2849 st->lookups,
2850 st->hits,
2851
2852 st->res_failed,
2853
2854 st->rcv_probes_mcast,
2855 st->rcv_probes_ucast,
2856
2857 st->periodic_gc_runs,
Neil Horman9a6d2762008-07-16 20:50:49 -07002858 st->forced_gc_runs,
Rick Jonesfb811392015-08-07 11:10:37 -07002859 st->unres_discards,
2860 st->table_fulls
Linus Torvalds1da177e2005-04-16 15:20:36 -07002861 );
2862
2863 return 0;
2864}
2865
Stephen Hemmingerf6908082007-03-12 14:34:29 -07002866static const struct seq_operations neigh_stat_seq_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002867 .start = neigh_stat_seq_start,
2868 .next = neigh_stat_seq_next,
2869 .stop = neigh_stat_seq_stop,
2870 .show = neigh_stat_seq_show,
2871};
Linus Torvalds1da177e2005-04-16 15:20:36 -07002872#endif /* CONFIG_PROC_FS */
2873
Thomas Graf339bf982006-11-10 14:10:15 -08002874static inline size_t neigh_nlmsg_size(void)
2875{
2876 return NLMSG_ALIGN(sizeof(struct ndmsg))
2877 + nla_total_size(MAX_ADDR_LEN) /* NDA_DST */
2878 + nla_total_size(MAX_ADDR_LEN) /* NDA_LLADDR */
2879 + nla_total_size(sizeof(struct nda_cacheinfo))
2880 + nla_total_size(4); /* NDA_PROBES */
2881}
2882
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07002883static void __neigh_notify(struct neighbour *n, int type, int flags,
2884 u32 pid)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002885{
YOSHIFUJI Hideakic346dca2008-03-25 21:47:49 +09002886 struct net *net = dev_net(n->dev);
Thomas Graf8b8aec52006-08-07 17:56:37 -07002887 struct sk_buff *skb;
Thomas Grafb8673312006-08-15 00:33:14 -07002888 int err = -ENOBUFS;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002889
Thomas Graf339bf982006-11-10 14:10:15 -08002890 skb = nlmsg_new(neigh_nlmsg_size(), GFP_ATOMIC);
Thomas Graf8b8aec52006-08-07 17:56:37 -07002891 if (skb == NULL)
Thomas Grafb8673312006-08-15 00:33:14 -07002892 goto errout;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002893
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07002894 err = neigh_fill_info(skb, n, pid, 0, type, flags);
Patrick McHardy26932562007-01-31 23:16:40 -08002895 if (err < 0) {
2896 /* -EMSGSIZE implies BUG in neigh_nlmsg_size() */
2897 WARN_ON(err == -EMSGSIZE);
2898 kfree_skb(skb);
2899 goto errout;
2900 }
Pablo Neira Ayuso1ce85fe2009-02-24 23:18:28 -08002901 rtnl_notify(skb, net, 0, RTNLGRP_NEIGH, NULL, GFP_ATOMIC);
2902 return;
Thomas Grafb8673312006-08-15 00:33:14 -07002903errout:
2904 if (err < 0)
Eric W. Biederman426b5302008-01-24 00:13:18 -08002905 rtnl_set_sk_err(net, RTNLGRP_NEIGH, err);
Thomas Grafb8673312006-08-15 00:33:14 -07002906}
2907
2908void neigh_app_ns(struct neighbour *n)
2909{
Roopa Prabhu7b8f7a42017-03-19 22:01:28 -07002910 __neigh_notify(n, RTM_GETNEIGH, NLM_F_REQUEST, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002911}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09002912EXPORT_SYMBOL(neigh_app_ns);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002913
2914#ifdef CONFIG_SYSCTL
Cong Wangb93196d2012-12-06 10:04:04 +08002915static int zero;
Francesco Fusco555445c2013-07-24 10:39:06 +02002916static int int_max = INT_MAX;
Cong Wangb93196d2012-12-06 10:04:04 +08002917static int unres_qlen_max = INT_MAX / SKB_TRUESIZE(ETH_FRAME_LEN);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002918
Joe Perchesfe2c6332013-06-11 23:04:25 -07002919static int proc_unres_qlen(struct ctl_table *ctl, int write,
2920 void __user *buffer, size_t *lenp, loff_t *ppos)
Eric Dumazet8b5c1712011-11-09 12:07:14 +00002921{
2922 int size, ret;
Joe Perchesfe2c6332013-06-11 23:04:25 -07002923 struct ctl_table tmp = *ctl;
Eric Dumazet8b5c1712011-11-09 12:07:14 +00002924
Shan Weice46cc62012-12-04 18:49:15 +00002925 tmp.extra1 = &zero;
2926 tmp.extra2 = &unres_qlen_max;
Eric Dumazet8b5c1712011-11-09 12:07:14 +00002927 tmp.data = &size;
Shan Weice46cc62012-12-04 18:49:15 +00002928
2929 size = *(int *)ctl->data / SKB_TRUESIZE(ETH_FRAME_LEN);
2930 ret = proc_dointvec_minmax(&tmp, write, buffer, lenp, ppos);
2931
Eric Dumazet8b5c1712011-11-09 12:07:14 +00002932 if (write && !ret)
2933 *(int *)ctl->data = size * SKB_TRUESIZE(ETH_FRAME_LEN);
2934 return ret;
2935}
2936
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002937static struct neigh_parms *neigh_get_dev_parms_rcu(struct net_device *dev,
2938 int family)
2939{
Jiri Pirkobba24892013-12-07 19:26:57 +01002940 switch (family) {
2941 case AF_INET:
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002942 return __in_dev_arp_parms_get_rcu(dev);
Jiri Pirkobba24892013-12-07 19:26:57 +01002943 case AF_INET6:
2944 return __in6_dev_nd_parms_get_rcu(dev);
2945 }
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002946 return NULL;
2947}
2948
2949static void neigh_copy_dflt_parms(struct net *net, struct neigh_parms *p,
2950 int index)
2951{
2952 struct net_device *dev;
2953 int family = neigh_parms_family(p);
2954
2955 rcu_read_lock();
2956 for_each_netdev_rcu(net, dev) {
2957 struct neigh_parms *dst_p =
2958 neigh_get_dev_parms_rcu(dev, family);
2959
2960 if (dst_p && !test_bit(index, dst_p->data_state))
2961 dst_p->data[index] = p->data[index];
2962 }
2963 rcu_read_unlock();
2964}
2965
2966static void neigh_proc_update(struct ctl_table *ctl, int write)
2967{
2968 struct net_device *dev = ctl->extra1;
2969 struct neigh_parms *p = ctl->extra2;
Jiri Pirko77d47af2013-12-10 23:55:07 +01002970 struct net *net = neigh_parms_net(p);
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002971 int index = (int *) ctl->data - p->data;
2972
2973 if (!write)
2974 return;
2975
2976 set_bit(index, p->data_state);
Marcus Huewe7627ae62017-02-15 01:00:36 +01002977 if (index == NEIGH_VAR_DELAY_PROBE_TIME)
2978 call_netevent_notifiers(NETEVENT_DELAY_PROBE_TIME_UPDATE, p);
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002979 if (!dev) /* NULL dev means this is default value */
2980 neigh_copy_dflt_parms(net, p, index);
2981}
2982
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002983static int neigh_proc_dointvec_zero_intmax(struct ctl_table *ctl, int write,
2984 void __user *buffer,
2985 size_t *lenp, loff_t *ppos)
2986{
2987 struct ctl_table tmp = *ctl;
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002988 int ret;
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002989
2990 tmp.extra1 = &zero;
2991 tmp.extra2 = &int_max;
2992
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01002993 ret = proc_dointvec_minmax(&tmp, write, buffer, lenp, ppos);
2994 neigh_proc_update(ctl, write);
2995 return ret;
Jiri Pirko1f9248e2013-12-07 19:26:53 +01002996}
2997
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01002998int neigh_proc_dointvec(struct ctl_table *ctl, int write,
2999 void __user *buffer, size_t *lenp, loff_t *ppos)
3000{
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003001 int ret = proc_dointvec(ctl, write, buffer, lenp, ppos);
3002
3003 neigh_proc_update(ctl, write);
3004 return ret;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003005}
3006EXPORT_SYMBOL(neigh_proc_dointvec);
3007
3008int neigh_proc_dointvec_jiffies(struct ctl_table *ctl, int write,
3009 void __user *buffer,
3010 size_t *lenp, loff_t *ppos)
3011{
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003012 int ret = proc_dointvec_jiffies(ctl, write, buffer, lenp, ppos);
3013
3014 neigh_proc_update(ctl, write);
3015 return ret;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003016}
3017EXPORT_SYMBOL(neigh_proc_dointvec_jiffies);
3018
3019static int neigh_proc_dointvec_userhz_jiffies(struct ctl_table *ctl, int write,
3020 void __user *buffer,
3021 size_t *lenp, loff_t *ppos)
3022{
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003023 int ret = proc_dointvec_userhz_jiffies(ctl, write, buffer, lenp, ppos);
3024
3025 neigh_proc_update(ctl, write);
3026 return ret;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003027}
3028
3029int neigh_proc_dointvec_ms_jiffies(struct ctl_table *ctl, int write,
3030 void __user *buffer,
3031 size_t *lenp, loff_t *ppos)
3032{
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003033 int ret = proc_dointvec_ms_jiffies(ctl, write, buffer, lenp, ppos);
3034
3035 neigh_proc_update(ctl, write);
3036 return ret;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003037}
3038EXPORT_SYMBOL(neigh_proc_dointvec_ms_jiffies);
3039
3040static int neigh_proc_dointvec_unres_qlen(struct ctl_table *ctl, int write,
3041 void __user *buffer,
3042 size_t *lenp, loff_t *ppos)
3043{
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003044 int ret = proc_unres_qlen(ctl, write, buffer, lenp, ppos);
3045
3046 neigh_proc_update(ctl, write);
3047 return ret;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003048}
3049
Jean-Francois Remy4bf69802015-01-14 04:22:39 +01003050static int neigh_proc_base_reachable_time(struct ctl_table *ctl, int write,
3051 void __user *buffer,
3052 size_t *lenp, loff_t *ppos)
3053{
3054 struct neigh_parms *p = ctl->extra2;
3055 int ret;
3056
3057 if (strcmp(ctl->procname, "base_reachable_time") == 0)
3058 ret = neigh_proc_dointvec_jiffies(ctl, write, buffer, lenp, ppos);
3059 else if (strcmp(ctl->procname, "base_reachable_time_ms") == 0)
3060 ret = neigh_proc_dointvec_ms_jiffies(ctl, write, buffer, lenp, ppos);
3061 else
3062 ret = -1;
3063
3064 if (write && ret == 0) {
3065 /* update reachable_time as well, otherwise, the change will
3066 * only be effective after the next time neigh_periodic_work
3067 * decides to recompute it
3068 */
3069 p->reachable_time =
3070 neigh_rand_reach_time(NEIGH_VAR(p, BASE_REACHABLE_TIME));
3071 }
3072 return ret;
3073}
3074
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003075#define NEIGH_PARMS_DATA_OFFSET(index) \
3076 (&((struct neigh_parms *) 0)->data[index])
3077
3078#define NEIGH_SYSCTL_ENTRY(attr, data_attr, name, mval, proc) \
3079 [NEIGH_VAR_ ## attr] = { \
3080 .procname = name, \
3081 .data = NEIGH_PARMS_DATA_OFFSET(NEIGH_VAR_ ## data_attr), \
3082 .maxlen = sizeof(int), \
3083 .mode = mval, \
3084 .proc_handler = proc, \
3085 }
3086
3087#define NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(attr, name) \
3088 NEIGH_SYSCTL_ENTRY(attr, attr, name, 0644, neigh_proc_dointvec_zero_intmax)
3089
3090#define NEIGH_SYSCTL_JIFFIES_ENTRY(attr, name) \
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003091 NEIGH_SYSCTL_ENTRY(attr, attr, name, 0644, neigh_proc_dointvec_jiffies)
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003092
3093#define NEIGH_SYSCTL_USERHZ_JIFFIES_ENTRY(attr, name) \
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003094 NEIGH_SYSCTL_ENTRY(attr, attr, name, 0644, neigh_proc_dointvec_userhz_jiffies)
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003095
3096#define NEIGH_SYSCTL_MS_JIFFIES_ENTRY(attr, name) \
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003097 NEIGH_SYSCTL_ENTRY(attr, attr, name, 0644, neigh_proc_dointvec_ms_jiffies)
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003098
3099#define NEIGH_SYSCTL_MS_JIFFIES_REUSED_ENTRY(attr, data_attr, name) \
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003100 NEIGH_SYSCTL_ENTRY(attr, data_attr, name, 0644, neigh_proc_dointvec_ms_jiffies)
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003101
3102#define NEIGH_SYSCTL_UNRES_QLEN_REUSED_ENTRY(attr, data_attr, name) \
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003103 NEIGH_SYSCTL_ENTRY(attr, data_attr, name, 0644, neigh_proc_dointvec_unres_qlen)
Eric W. Biederman54716e32010-02-14 03:27:03 +00003104
Linus Torvalds1da177e2005-04-16 15:20:36 -07003105static struct neigh_sysctl_table {
3106 struct ctl_table_header *sysctl_header;
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003107 struct ctl_table neigh_vars[NEIGH_VAR_MAX + 1];
Brian Haleyab32ea52006-09-22 14:15:41 -07003108} neigh_sysctl_template __read_mostly = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003109 .neigh_vars = {
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003110 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(MCAST_PROBES, "mcast_solicit"),
3111 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(UCAST_PROBES, "ucast_solicit"),
3112 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(APP_PROBES, "app_solicit"),
YOSHIFUJI Hideaki/吉藤英明8da86462015-03-19 22:41:46 +09003113 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(MCAST_REPROBES, "mcast_resolicit"),
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003114 NEIGH_SYSCTL_USERHZ_JIFFIES_ENTRY(RETRANS_TIME, "retrans_time"),
3115 NEIGH_SYSCTL_JIFFIES_ENTRY(BASE_REACHABLE_TIME, "base_reachable_time"),
3116 NEIGH_SYSCTL_JIFFIES_ENTRY(DELAY_PROBE_TIME, "delay_first_probe_time"),
3117 NEIGH_SYSCTL_JIFFIES_ENTRY(GC_STALETIME, "gc_stale_time"),
3118 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(QUEUE_LEN_BYTES, "unres_qlen_bytes"),
3119 NEIGH_SYSCTL_ZERO_INTMAX_ENTRY(PROXY_QLEN, "proxy_qlen"),
3120 NEIGH_SYSCTL_USERHZ_JIFFIES_ENTRY(ANYCAST_DELAY, "anycast_delay"),
3121 NEIGH_SYSCTL_USERHZ_JIFFIES_ENTRY(PROXY_DELAY, "proxy_delay"),
3122 NEIGH_SYSCTL_USERHZ_JIFFIES_ENTRY(LOCKTIME, "locktime"),
3123 NEIGH_SYSCTL_UNRES_QLEN_REUSED_ENTRY(QUEUE_LEN, QUEUE_LEN_BYTES, "unres_qlen"),
3124 NEIGH_SYSCTL_MS_JIFFIES_REUSED_ENTRY(RETRANS_TIME_MS, RETRANS_TIME, "retrans_time_ms"),
3125 NEIGH_SYSCTL_MS_JIFFIES_REUSED_ENTRY(BASE_REACHABLE_TIME_MS, BASE_REACHABLE_TIME, "base_reachable_time_ms"),
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003126 [NEIGH_VAR_GC_INTERVAL] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003127 .procname = "gc_interval",
3128 .maxlen = sizeof(int),
3129 .mode = 0644,
Alexey Dobriyan6d9f2392008-11-03 18:21:05 -08003130 .proc_handler = proc_dointvec_jiffies,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003131 },
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003132 [NEIGH_VAR_GC_THRESH1] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003133 .procname = "gc_thresh1",
3134 .maxlen = sizeof(int),
3135 .mode = 0644,
Francesco Fusco555445c2013-07-24 10:39:06 +02003136 .extra1 = &zero,
3137 .extra2 = &int_max,
3138 .proc_handler = proc_dointvec_minmax,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003139 },
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003140 [NEIGH_VAR_GC_THRESH2] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003141 .procname = "gc_thresh2",
3142 .maxlen = sizeof(int),
3143 .mode = 0644,
Francesco Fusco555445c2013-07-24 10:39:06 +02003144 .extra1 = &zero,
3145 .extra2 = &int_max,
3146 .proc_handler = proc_dointvec_minmax,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003147 },
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003148 [NEIGH_VAR_GC_THRESH3] = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003149 .procname = "gc_thresh3",
3150 .maxlen = sizeof(int),
3151 .mode = 0644,
Francesco Fusco555445c2013-07-24 10:39:06 +02003152 .extra1 = &zero,
3153 .extra2 = &int_max,
3154 .proc_handler = proc_dointvec_minmax,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003155 },
Pavel Emelyanovc3bac5a2007-12-02 00:08:16 +11003156 {},
Linus Torvalds1da177e2005-04-16 15:20:36 -07003157 },
3158};
3159
3160int neigh_sysctl_register(struct net_device *dev, struct neigh_parms *p,
Jiri Pirko73af6142013-12-07 19:26:55 +01003161 proc_handler *handler)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003162{
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003163 int i;
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003164 struct neigh_sysctl_table *t;
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003165 const char *dev_name_source;
Eric W. Biederman8f40a1f2012-04-19 13:38:03 +00003166 char neigh_path[ sizeof("net//neigh/") + IFNAMSIZ + IFNAMSIZ ];
Jiri Pirko73af6142013-12-07 19:26:55 +01003167 char *p_name;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003168
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003169 t = kmemdup(&neigh_sysctl_template, sizeof(*t), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003170 if (!t)
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003171 goto err;
3172
Jiri Pirkob194c1f2014-02-21 14:52:57 +01003173 for (i = 0; i < NEIGH_VAR_GC_INTERVAL; i++) {
Jiri Pirko1f9248e2013-12-07 19:26:53 +01003174 t->neigh_vars[i].data += (long) p;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003175 t->neigh_vars[i].extra1 = dev;
Jiri Pirko1d4c8c22013-12-07 19:26:56 +01003176 t->neigh_vars[i].extra2 = p;
Jiri Pirkocb5b09c2013-12-07 19:26:54 +01003177 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07003178
3179 if (dev) {
3180 dev_name_source = dev->name;
Eric W. Biedermand12af672007-10-18 03:05:25 -07003181 /* Terminate the table early */
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003182 memset(&t->neigh_vars[NEIGH_VAR_GC_INTERVAL], 0,
3183 sizeof(t->neigh_vars[NEIGH_VAR_GC_INTERVAL]));
Linus Torvalds1da177e2005-04-16 15:20:36 -07003184 } else {
Mathias Krause9ecf07a2014-07-12 22:36:44 +02003185 struct neigh_table *tbl = p->tbl;
Eric W. Biederman8f40a1f2012-04-19 13:38:03 +00003186 dev_name_source = "default";
Mathias Krause9ecf07a2014-07-12 22:36:44 +02003187 t->neigh_vars[NEIGH_VAR_GC_INTERVAL].data = &tbl->gc_interval;
3188 t->neigh_vars[NEIGH_VAR_GC_THRESH1].data = &tbl->gc_thresh1;
3189 t->neigh_vars[NEIGH_VAR_GC_THRESH2].data = &tbl->gc_thresh2;
3190 t->neigh_vars[NEIGH_VAR_GC_THRESH3].data = &tbl->gc_thresh3;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003191 }
3192
Eric W. Biedermanf8572d82009-11-05 13:32:03 -08003193 if (handler) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003194 /* RetransTime */
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003195 t->neigh_vars[NEIGH_VAR_RETRANS_TIME].proc_handler = handler;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003196 /* ReachableTime */
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003197 t->neigh_vars[NEIGH_VAR_BASE_REACHABLE_TIME].proc_handler = handler;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003198 /* RetransTime (in milliseconds)*/
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003199 t->neigh_vars[NEIGH_VAR_RETRANS_TIME_MS].proc_handler = handler;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003200 /* ReachableTime (in milliseconds) */
Eric Dumazet8b5c1712011-11-09 12:07:14 +00003201 t->neigh_vars[NEIGH_VAR_BASE_REACHABLE_TIME_MS].proc_handler = handler;
Jean-Francois Remy4bf69802015-01-14 04:22:39 +01003202 } else {
3203 /* Those handlers will update p->reachable_time after
3204 * base_reachable_time(_ms) is set to ensure the new timer starts being
3205 * applied after the next neighbour update instead of waiting for
3206 * neigh_periodic_work to update its value (can be multiple minutes)
3207 * So any handler that replaces them should do this as well
3208 */
3209 /* ReachableTime */
3210 t->neigh_vars[NEIGH_VAR_BASE_REACHABLE_TIME].proc_handler =
3211 neigh_proc_base_reachable_time;
3212 /* ReachableTime (in milliseconds) */
3213 t->neigh_vars[NEIGH_VAR_BASE_REACHABLE_TIME_MS].proc_handler =
3214 neigh_proc_base_reachable_time;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003215 }
3216
Eric W. Biederman464dc802012-11-16 03:02:59 +00003217 /* Don't export sysctls to unprivileged users */
3218 if (neigh_parms_net(p)->user_ns != &init_user_ns)
3219 t->neigh_vars[0].procname = NULL;
3220
Jiri Pirko73af6142013-12-07 19:26:55 +01003221 switch (neigh_parms_family(p)) {
3222 case AF_INET:
3223 p_name = "ipv4";
3224 break;
3225 case AF_INET6:
3226 p_name = "ipv6";
3227 break;
3228 default:
3229 BUG();
3230 }
3231
Eric W. Biederman8f40a1f2012-04-19 13:38:03 +00003232 snprintf(neigh_path, sizeof(neigh_path), "net/%s/neigh/%s",
3233 p_name, dev_name_source);
Denis V. Lunev4ab438f2008-02-28 20:48:01 -08003234 t->sysctl_header =
Eric W. Biederman8f40a1f2012-04-19 13:38:03 +00003235 register_net_sysctl(neigh_parms_net(p), neigh_path, t->neigh_vars);
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003236 if (!t->sysctl_header)
Eric W. Biederman8f40a1f2012-04-19 13:38:03 +00003237 goto free;
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003238
Linus Torvalds1da177e2005-04-16 15:20:36 -07003239 p->sysctl_table = t;
3240 return 0;
3241
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003242free:
Linus Torvalds1da177e2005-04-16 15:20:36 -07003243 kfree(t);
Pavel Emelyanov3c607bb2007-12-02 00:06:34 +11003244err:
3245 return -ENOBUFS;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003246}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09003247EXPORT_SYMBOL(neigh_sysctl_register);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003248
3249void neigh_sysctl_unregister(struct neigh_parms *p)
3250{
3251 if (p->sysctl_table) {
3252 struct neigh_sysctl_table *t = p->sysctl_table;
3253 p->sysctl_table = NULL;
Eric W. Biederman5dd3df12012-04-19 13:24:33 +00003254 unregister_net_sysctl_table(t->sysctl_header);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003255 kfree(t);
3256 }
3257}
YOSHIFUJI Hideaki0a204502008-03-24 18:39:10 +09003258EXPORT_SYMBOL(neigh_sysctl_unregister);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003259
3260#endif /* CONFIG_SYSCTL */
3261
Thomas Grafc8822a42007-03-22 11:50:06 -07003262static int __init neigh_init(void)
3263{
Florian Westphalb97bac62017-08-09 20:41:48 +02003264 rtnl_register(PF_UNSPEC, RTM_NEWNEIGH, neigh_add, NULL, 0);
3265 rtnl_register(PF_UNSPEC, RTM_DELNEIGH, neigh_delete, NULL, 0);
3266 rtnl_register(PF_UNSPEC, RTM_GETNEIGH, NULL, neigh_dump_info, 0);
Thomas Grafc8822a42007-03-22 11:50:06 -07003267
Greg Rosec7ac8672011-06-10 01:27:09 +00003268 rtnl_register(PF_UNSPEC, RTM_GETNEIGHTBL, NULL, neightbl_dump_info,
Florian Westphalb97bac62017-08-09 20:41:48 +02003269 0);
3270 rtnl_register(PF_UNSPEC, RTM_SETNEIGHTBL, neightbl_set, NULL, 0);
Thomas Grafc8822a42007-03-22 11:50:06 -07003271
3272 return 0;
3273}
3274
3275subsys_initcall(neigh_init);
3276