blob: 33e7b91fc805a7fd3537df3484b256cb1bdd8ee7 [file] [log] [blame]
Thomas Gleixner2874c5f2019-05-27 08:55:01 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Linus Torvalds1da177e2005-04-16 15:20:36 -07002/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 *
4 * Copyright Jonathan Naylor G4KLX (g4klx@g4klx.demon.co.uk)
5 * Copyright Alan Cox GW4PTS (alan@lxorguk.ukuu.org.uk)
6 * Copyright Tomi Manninen OH2BNS (oh2bns@sral.fi)
7 */
8#include <linux/errno.h>
9#include <linux/types.h>
10#include <linux/socket.h>
11#include <linux/in.h>
12#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070013#include <linux/timer.h>
14#include <linux/string.h>
15#include <linux/sockios.h>
16#include <linux/net.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090017#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070018#include <net/ax25.h>
19#include <linux/inet.h>
20#include <linux/netdevice.h>
21#include <net/arp.h>
22#include <linux/if_arp.h>
23#include <linux/skbuff.h>
24#include <net/sock.h>
Fabian Frederickdc8e5412014-10-17 22:00:22 +020025#include <linux/uaccess.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include <linux/fcntl.h>
27#include <linux/termios.h> /* For TIOCINQ/OUTQ */
28#include <linux/mm.h>
29#include <linux/interrupt.h>
30#include <linux/notifier.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070031#include <linux/init.h>
32#include <linux/spinlock.h>
33#include <net/netrom.h>
34#include <linux/seq_file.h>
Paul Gortmakerbc3b2d72011-07-15 11:47:34 -040035#include <linux/export.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070036
37static unsigned int nr_neigh_no = 1;
38
39static HLIST_HEAD(nr_node_list);
40static DEFINE_SPINLOCK(nr_node_list_lock);
41static HLIST_HEAD(nr_neigh_list);
42static DEFINE_SPINLOCK(nr_neigh_list_lock);
43
44static struct nr_node *nr_node_get(ax25_address *callsign)
45{
46 struct nr_node *found = NULL;
47 struct nr_node *nr_node;
Linus Torvalds1da177e2005-04-16 15:20:36 -070048
49 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -080050 nr_node_for_each(nr_node, &nr_node_list)
Linus Torvalds1da177e2005-04-16 15:20:36 -070051 if (ax25cmp(callsign, &nr_node->callsign) == 0) {
52 nr_node_hold(nr_node);
53 found = nr_node;
54 break;
55 }
56 spin_unlock_bh(&nr_node_list_lock);
57 return found;
58}
59
60static struct nr_neigh *nr_neigh_get_dev(ax25_address *callsign,
61 struct net_device *dev)
62{
63 struct nr_neigh *found = NULL;
64 struct nr_neigh *nr_neigh;
Linus Torvalds1da177e2005-04-16 15:20:36 -070065
66 spin_lock_bh(&nr_neigh_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -080067 nr_neigh_for_each(nr_neigh, &nr_neigh_list)
Linus Torvalds1da177e2005-04-16 15:20:36 -070068 if (ax25cmp(callsign, &nr_neigh->callsign) == 0 &&
69 nr_neigh->dev == dev) {
70 nr_neigh_hold(nr_neigh);
71 found = nr_neigh;
72 break;
73 }
74 spin_unlock_bh(&nr_neigh_list_lock);
75 return found;
76}
77
78static void nr_remove_neigh(struct nr_neigh *);
79
Gustavo A. R. Silva4c316062017-10-27 00:51:04 -050080/* re-sort the routes in quality order. */
81static void re_sort_routes(struct nr_node *nr_node, int x, int y)
82{
83 if (nr_node->routes[y].quality > nr_node->routes[x].quality) {
84 if (nr_node->which == x)
85 nr_node->which = y;
86 else if (nr_node->which == y)
87 nr_node->which = x;
88
89 swap(nr_node->routes[x], nr_node->routes[y]);
90 }
91}
92
Linus Torvalds1da177e2005-04-16 15:20:36 -070093/*
94 * Add a new route to a node, and in the process add the node and the
95 * neighbour if it is new.
96 */
Ralf Baechlec9266b92006-12-14 15:49:28 -080097static int __must_check nr_add_node(ax25_address *nr, const char *mnemonic,
98 ax25_address *ax25, ax25_digi *ax25_digi, struct net_device *dev,
99 int quality, int obs_count)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100{
101 struct nr_node *nr_node;
102 struct nr_neigh *nr_neigh;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103 int i, found;
104 struct net_device *odev;
105
106 if ((odev=nr_dev_get(nr)) != NULL) { /* Can't add routes to ourself */
107 dev_put(odev);
108 return -EINVAL;
109 }
110
111 nr_node = nr_node_get(nr);
112
113 nr_neigh = nr_neigh_get_dev(ax25, dev);
114
115 /*
116 * The L2 link to a neighbour has failed in the past
117 * and now a frame comes from this neighbour. We assume
118 * it was a temporary trouble with the link and reset the
119 * routes now (and not wait for a node broadcast).
120 */
121 if (nr_neigh != NULL && nr_neigh->failed != 0 && quality == 0) {
122 struct nr_node *nr_nodet;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700123
124 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800125 nr_node_for_each(nr_nodet, &nr_node_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126 nr_node_lock(nr_nodet);
127 for (i = 0; i < nr_nodet->count; i++)
128 if (nr_nodet->routes[i].neighbour == nr_neigh)
129 if (i < nr_nodet->which)
130 nr_nodet->which = i;
131 nr_node_unlock(nr_nodet);
132 }
133 spin_unlock_bh(&nr_node_list_lock);
134 }
135
136 if (nr_neigh != NULL)
137 nr_neigh->failed = 0;
138
139 if (quality == 0 && nr_neigh != NULL && nr_node != NULL) {
140 nr_neigh_put(nr_neigh);
141 nr_node_put(nr_node);
142 return 0;
143 }
144
145 if (nr_neigh == NULL) {
146 if ((nr_neigh = kmalloc(sizeof(*nr_neigh), GFP_ATOMIC)) == NULL) {
147 if (nr_node)
148 nr_node_put(nr_node);
149 return -ENOMEM;
150 }
151
152 nr_neigh->callsign = *ax25;
153 nr_neigh->digipeat = NULL;
154 nr_neigh->ax25 = NULL;
155 nr_neigh->dev = dev;
156 nr_neigh->quality = sysctl_netrom_default_path_quality;
157 nr_neigh->locked = 0;
158 nr_neigh->count = 0;
159 nr_neigh->number = nr_neigh_no++;
160 nr_neigh->failed = 0;
Reshetova, Elenaaf420742017-07-04 15:53:11 +0300161 refcount_set(&nr_neigh->refcount, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700162
163 if (ax25_digi != NULL && ax25_digi->ndigi > 0) {
Arnaldo Carvalho de Meloeafff862006-11-17 13:05:04 -0200164 nr_neigh->digipeat = kmemdup(ax25_digi,
165 sizeof(*ax25_digi),
166 GFP_KERNEL);
167 if (nr_neigh->digipeat == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700168 kfree(nr_neigh);
169 if (nr_node)
170 nr_node_put(nr_node);
171 return -ENOMEM;
172 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700173 }
174
175 spin_lock_bh(&nr_neigh_list_lock);
176 hlist_add_head(&nr_neigh->neigh_node, &nr_neigh_list);
177 nr_neigh_hold(nr_neigh);
178 spin_unlock_bh(&nr_neigh_list_lock);
179 }
180
181 if (quality != 0 && ax25cmp(nr, ax25) == 0 && !nr_neigh->locked)
182 nr_neigh->quality = quality;
183
184 if (nr_node == NULL) {
185 if ((nr_node = kmalloc(sizeof(*nr_node), GFP_ATOMIC)) == NULL) {
186 if (nr_neigh)
187 nr_neigh_put(nr_neigh);
188 return -ENOMEM;
189 }
190
191 nr_node->callsign = *nr;
192 strcpy(nr_node->mnemonic, mnemonic);
193
194 nr_node->which = 0;
195 nr_node->count = 1;
Reshetova, Elena156be7e2017-07-04 15:53:12 +0300196 refcount_set(&nr_node->refcount, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700197 spin_lock_init(&nr_node->node_lock);
198
199 nr_node->routes[0].quality = quality;
200 nr_node->routes[0].obs_count = obs_count;
201 nr_node->routes[0].neighbour = nr_neigh;
202
203 nr_neigh_hold(nr_neigh);
204 nr_neigh->count++;
205
206 spin_lock_bh(&nr_node_list_lock);
207 hlist_add_head(&nr_node->node_node, &nr_node_list);
208 /* refcount initialized at 1 */
209 spin_unlock_bh(&nr_node_list_lock);
210
211 return 0;
212 }
213 nr_node_lock(nr_node);
214
215 if (quality != 0)
216 strcpy(nr_node->mnemonic, mnemonic);
217
218 for (found = 0, i = 0; i < nr_node->count; i++) {
219 if (nr_node->routes[i].neighbour == nr_neigh) {
220 nr_node->routes[i].quality = quality;
221 nr_node->routes[i].obs_count = obs_count;
222 found = 1;
223 break;
224 }
225 }
226
227 if (!found) {
228 /* We have space at the bottom, slot it in */
229 if (nr_node->count < 3) {
230 nr_node->routes[2] = nr_node->routes[1];
231 nr_node->routes[1] = nr_node->routes[0];
232
233 nr_node->routes[0].quality = quality;
234 nr_node->routes[0].obs_count = obs_count;
235 nr_node->routes[0].neighbour = nr_neigh;
236
237 nr_node->which++;
238 nr_node->count++;
239 nr_neigh_hold(nr_neigh);
240 nr_neigh->count++;
241 } else {
242 /* It must be better than the worst */
243 if (quality > nr_node->routes[2].quality) {
244 nr_node->routes[2].neighbour->count--;
245 nr_neigh_put(nr_node->routes[2].neighbour);
246
247 if (nr_node->routes[2].neighbour->count == 0 && !nr_node->routes[2].neighbour->locked)
248 nr_remove_neigh(nr_node->routes[2].neighbour);
249
250 nr_node->routes[2].quality = quality;
251 nr_node->routes[2].obs_count = obs_count;
252 nr_node->routes[2].neighbour = nr_neigh;
253
254 nr_neigh_hold(nr_neigh);
255 nr_neigh->count++;
256 }
257 }
258 }
259
260 /* Now re-sort the routes in quality order */
261 switch (nr_node->count) {
262 case 3:
Gustavo A. R. Silva4c316062017-10-27 00:51:04 -0500263 re_sort_routes(nr_node, 0, 1);
264 re_sort_routes(nr_node, 1, 2);
Gustavo A. R. Silva31f74f02017-10-27 00:51:08 -0500265 /* fall through */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266 case 2:
Gustavo A. R. Silva4c316062017-10-27 00:51:04 -0500267 re_sort_routes(nr_node, 0, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700268 case 1:
269 break;
270 }
271
272 for (i = 0; i < nr_node->count; i++) {
273 if (nr_node->routes[i].neighbour == nr_neigh) {
274 if (i < nr_node->which)
275 nr_node->which = i;
276 break;
277 }
278 }
279
280 nr_neigh_put(nr_neigh);
281 nr_node_unlock(nr_node);
282 nr_node_put(nr_node);
283 return 0;
284}
285
286static inline void __nr_remove_node(struct nr_node *nr_node)
287{
288 hlist_del_init(&nr_node->node_node);
289 nr_node_put(nr_node);
290}
291
292#define nr_remove_node_locked(__node) \
293 __nr_remove_node(__node)
294
295static void nr_remove_node(struct nr_node *nr_node)
296{
297 spin_lock_bh(&nr_node_list_lock);
298 __nr_remove_node(nr_node);
299 spin_unlock_bh(&nr_node_list_lock);
300}
301
302static inline void __nr_remove_neigh(struct nr_neigh *nr_neigh)
303{
304 hlist_del_init(&nr_neigh->neigh_node);
305 nr_neigh_put(nr_neigh);
306}
307
308#define nr_remove_neigh_locked(__neigh) \
309 __nr_remove_neigh(__neigh)
310
311static void nr_remove_neigh(struct nr_neigh *nr_neigh)
312{
313 spin_lock_bh(&nr_neigh_list_lock);
314 __nr_remove_neigh(nr_neigh);
315 spin_unlock_bh(&nr_neigh_list_lock);
316}
317
318/*
319 * "Delete" a node. Strictly speaking remove a route to a node. The node
320 * is only deleted if no routes are left to it.
321 */
322static int nr_del_node(ax25_address *callsign, ax25_address *neighbour, struct net_device *dev)
323{
324 struct nr_node *nr_node;
325 struct nr_neigh *nr_neigh;
326 int i;
327
328 nr_node = nr_node_get(callsign);
329
330 if (nr_node == NULL)
331 return -EINVAL;
332
333 nr_neigh = nr_neigh_get_dev(neighbour, dev);
334
335 if (nr_neigh == NULL) {
336 nr_node_put(nr_node);
337 return -EINVAL;
338 }
339
340 nr_node_lock(nr_node);
341 for (i = 0; i < nr_node->count; i++) {
342 if (nr_node->routes[i].neighbour == nr_neigh) {
343 nr_neigh->count--;
344 nr_neigh_put(nr_neigh);
345
346 if (nr_neigh->count == 0 && !nr_neigh->locked)
347 nr_remove_neigh(nr_neigh);
348 nr_neigh_put(nr_neigh);
349
350 nr_node->count--;
351
352 if (nr_node->count == 0) {
353 nr_remove_node(nr_node);
354 } else {
355 switch (i) {
356 case 0:
357 nr_node->routes[0] = nr_node->routes[1];
Gustavo A. R. Silva31f74f02017-10-27 00:51:08 -0500358 /* fall through */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700359 case 1:
360 nr_node->routes[1] = nr_node->routes[2];
361 case 2:
362 break;
363 }
364 nr_node_put(nr_node);
365 }
366 nr_node_unlock(nr_node);
367
368 return 0;
369 }
370 }
371 nr_neigh_put(nr_neigh);
372 nr_node_unlock(nr_node);
373 nr_node_put(nr_node);
374
375 return -EINVAL;
376}
377
378/*
379 * Lock a neighbour with a quality.
380 */
Ralf Baechlec9266b92006-12-14 15:49:28 -0800381static int __must_check nr_add_neigh(ax25_address *callsign,
382 ax25_digi *ax25_digi, struct net_device *dev, unsigned int quality)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383{
384 struct nr_neigh *nr_neigh;
385
386 nr_neigh = nr_neigh_get_dev(callsign, dev);
387 if (nr_neigh) {
388 nr_neigh->quality = quality;
389 nr_neigh->locked = 1;
390 nr_neigh_put(nr_neigh);
391 return 0;
392 }
393
394 if ((nr_neigh = kmalloc(sizeof(*nr_neigh), GFP_ATOMIC)) == NULL)
395 return -ENOMEM;
396
397 nr_neigh->callsign = *callsign;
398 nr_neigh->digipeat = NULL;
399 nr_neigh->ax25 = NULL;
400 nr_neigh->dev = dev;
401 nr_neigh->quality = quality;
402 nr_neigh->locked = 1;
403 nr_neigh->count = 0;
404 nr_neigh->number = nr_neigh_no++;
405 nr_neigh->failed = 0;
Reshetova, Elenaaf420742017-07-04 15:53:11 +0300406 refcount_set(&nr_neigh->refcount, 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407
408 if (ax25_digi != NULL && ax25_digi->ndigi > 0) {
Arnaldo Carvalho de Meloeafff862006-11-17 13:05:04 -0200409 nr_neigh->digipeat = kmemdup(ax25_digi, sizeof(*ax25_digi),
410 GFP_KERNEL);
411 if (nr_neigh->digipeat == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700412 kfree(nr_neigh);
413 return -ENOMEM;
414 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700415 }
416
417 spin_lock_bh(&nr_neigh_list_lock);
418 hlist_add_head(&nr_neigh->neigh_node, &nr_neigh_list);
419 /* refcount is initialized at 1 */
420 spin_unlock_bh(&nr_neigh_list_lock);
421
422 return 0;
423}
424
425/*
426 * "Delete" a neighbour. The neighbour is only removed if the number
427 * of nodes that may use it is zero.
428 */
429static int nr_del_neigh(ax25_address *callsign, struct net_device *dev, unsigned int quality)
430{
431 struct nr_neigh *nr_neigh;
432
433 nr_neigh = nr_neigh_get_dev(callsign, dev);
434
435 if (nr_neigh == NULL) return -EINVAL;
436
437 nr_neigh->quality = quality;
438 nr_neigh->locked = 0;
439
440 if (nr_neigh->count == 0)
441 nr_remove_neigh(nr_neigh);
442 nr_neigh_put(nr_neigh);
443
444 return 0;
445}
446
447/*
448 * Decrement the obsolescence count by one. If a route is reduced to a
449 * count of zero, remove it. Also remove any unlocked neighbours with
450 * zero nodes routing via it.
451 */
452static int nr_dec_obs(void)
453{
454 struct nr_neigh *nr_neigh;
455 struct nr_node *s;
Sasha Levinb67bfe02013-02-27 17:06:00 -0800456 struct hlist_node *nodet;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700457 int i;
458
459 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800460 nr_node_for_each_safe(s, nodet, &nr_node_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700461 nr_node_lock(s);
462 for (i = 0; i < s->count; i++) {
463 switch (s->routes[i].obs_count) {
464 case 0: /* A locked entry */
465 break;
466
467 case 1: /* From 1 -> 0 */
468 nr_neigh = s->routes[i].neighbour;
469
470 nr_neigh->count--;
471 nr_neigh_put(nr_neigh);
472
473 if (nr_neigh->count == 0 && !nr_neigh->locked)
474 nr_remove_neigh(nr_neigh);
475
476 s->count--;
477
478 switch (i) {
Joe Perchesc4855382011-07-01 09:43:10 +0000479 case 0:
480 s->routes[0] = s->routes[1];
481 /* Fallthrough */
482 case 1:
483 s->routes[1] = s->routes[2];
484 case 2:
485 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700486 }
487 break;
488
489 default:
490 s->routes[i].obs_count--;
491 break;
492
493 }
494 }
495
496 if (s->count <= 0)
497 nr_remove_node_locked(s);
498 nr_node_unlock(s);
499 }
500 spin_unlock_bh(&nr_node_list_lock);
501
502 return 0;
503}
504
505/*
506 * A device has been removed. Remove its routes and neighbours.
507 */
508void nr_rt_device_down(struct net_device *dev)
509{
510 struct nr_neigh *s;
Sasha Levinb67bfe02013-02-27 17:06:00 -0800511 struct hlist_node *nodet, *node2t;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512 struct nr_node *t;
513 int i;
514
515 spin_lock_bh(&nr_neigh_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800516 nr_neigh_for_each_safe(s, nodet, &nr_neigh_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700517 if (s->dev == dev) {
518 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800519 nr_node_for_each_safe(t, node2t, &nr_node_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700520 nr_node_lock(t);
521 for (i = 0; i < t->count; i++) {
522 if (t->routes[i].neighbour == s) {
523 t->count--;
524
525 switch (i) {
526 case 0:
527 t->routes[0] = t->routes[1];
Gustavo A. R. Silva31f74f02017-10-27 00:51:08 -0500528 /* fall through */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700529 case 1:
530 t->routes[1] = t->routes[2];
531 case 2:
532 break;
533 }
534 }
535 }
536
537 if (t->count <= 0)
538 nr_remove_node_locked(t);
539 nr_node_unlock(t);
540 }
541 spin_unlock_bh(&nr_node_list_lock);
542
543 nr_remove_neigh_locked(s);
544 }
545 }
546 spin_unlock_bh(&nr_neigh_list_lock);
547}
548
549/*
550 * Check that the device given is a valid AX.25 interface that is "up".
551 * Or a valid ethernet interface with an AX.25 callsign binding.
552 */
553static struct net_device *nr_ax25_dev_get(char *devname)
554{
555 struct net_device *dev;
556
Eric W. Biederman881d9662007-09-17 11:56:21 -0700557 if ((dev = dev_get_by_name(&init_net, devname)) == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558 return NULL;
559
560 if ((dev->flags & IFF_UP) && dev->type == ARPHRD_AX25)
561 return dev;
562
563 dev_put(dev);
564 return NULL;
565}
566
567/*
568 * Find the first active NET/ROM device, usually "nr0".
569 */
570struct net_device *nr_dev_first(void)
571{
572 struct net_device *dev, *first = NULL;
573
Eric Dumazetc6d14c82009-11-04 05:43:23 -0800574 rcu_read_lock();
575 for_each_netdev_rcu(&init_net, dev) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 if ((dev->flags & IFF_UP) && dev->type == ARPHRD_NETROM)
577 if (first == NULL || strncmp(dev->name, first->name, 3) < 0)
578 first = dev;
579 }
580 if (first)
581 dev_hold(first);
Eric Dumazetc6d14c82009-11-04 05:43:23 -0800582 rcu_read_unlock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700583
584 return first;
585}
586
587/*
588 * Find the NET/ROM device for the given callsign.
589 */
590struct net_device *nr_dev_get(ax25_address *addr)
591{
592 struct net_device *dev;
593
Eric Dumazetc6d14c82009-11-04 05:43:23 -0800594 rcu_read_lock();
595 for_each_netdev_rcu(&init_net, dev) {
596 if ((dev->flags & IFF_UP) && dev->type == ARPHRD_NETROM &&
597 ax25cmp(addr, (ax25_address *)dev->dev_addr) == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700598 dev_hold(dev);
599 goto out;
600 }
601 }
Pavel Emelianov7562f872007-05-03 15:13:45 -0700602 dev = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700603out:
Eric Dumazetc6d14c82009-11-04 05:43:23 -0800604 rcu_read_unlock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700605 return dev;
606}
607
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700608static ax25_digi *nr_call_to_digi(ax25_digi *digi, int ndigis,
609 ax25_address *digipeaters)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700610{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700611 int i;
612
613 if (ndigis == 0)
614 return NULL;
615
616 for (i = 0; i < ndigis; i++) {
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700617 digi->calls[i] = digipeaters[i];
618 digi->repeated[i] = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700619 }
620
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700621 digi->ndigi = ndigis;
622 digi->lastrepeat = -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700623
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700624 return digi;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700625}
626
627/*
628 * Handle the ioctls that control the routing functions.
629 */
630int nr_rt_ioctl(unsigned int cmd, void __user *arg)
631{
632 struct nr_route_struct nr_route;
633 struct net_device *dev;
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700634 ax25_digi digi;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700635 int ret;
636
637 switch (cmd) {
638 case SIOCADDRT:
639 if (copy_from_user(&nr_route, arg, sizeof(struct nr_route_struct)))
640 return -EFAULT;
Ralf Baechle10cae1c2011-11-24 23:09:00 +0000641 if (nr_route.ndigis > AX25_MAX_DIGIS)
642 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700643 if ((dev = nr_ax25_dev_get(nr_route.device)) == NULL)
644 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700645 switch (nr_route.type) {
646 case NETROM_NODE:
Ralf Baechleac1a1de2011-11-24 23:08:49 +0000647 if (strnlen(nr_route.mnemonic, 7) == 7) {
648 ret = -EINVAL;
649 break;
650 }
651
Linus Torvalds1da177e2005-04-16 15:20:36 -0700652 ret = nr_add_node(&nr_route.callsign,
653 nr_route.mnemonic,
654 &nr_route.neighbour,
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700655 nr_call_to_digi(&digi, nr_route.ndigis,
656 nr_route.digipeaters),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700657 dev, nr_route.quality,
658 nr_route.obs_count);
659 break;
660 case NETROM_NEIGH:
661 ret = nr_add_neigh(&nr_route.callsign,
Ralf Baechlec6ba9732009-08-17 18:05:32 -0700662 nr_call_to_digi(&digi, nr_route.ndigis,
663 nr_route.digipeaters),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700664 dev, nr_route.quality);
665 break;
666 default:
667 ret = -EINVAL;
668 }
669 dev_put(dev);
670 return ret;
671
672 case SIOCDELRT:
673 if (copy_from_user(&nr_route, arg, sizeof(struct nr_route_struct)))
674 return -EFAULT;
675 if ((dev = nr_ax25_dev_get(nr_route.device)) == NULL)
676 return -EINVAL;
677 switch (nr_route.type) {
678 case NETROM_NODE:
679 ret = nr_del_node(&nr_route.callsign,
680 &nr_route.neighbour, dev);
681 break;
682 case NETROM_NEIGH:
683 ret = nr_del_neigh(&nr_route.callsign,
684 dev, nr_route.quality);
685 break;
686 default:
687 ret = -EINVAL;
688 }
689 dev_put(dev);
690 return ret;
691
692 case SIOCNRDECOBS:
693 return nr_dec_obs();
694
695 default:
696 return -EINVAL;
697 }
698
699 return 0;
700}
701
702/*
703 * A level 2 link has timed out, therefore it appears to be a poor link,
704 * then don't use that neighbour until it is reset.
705 */
706void nr_link_failed(ax25_cb *ax25, int reason)
707{
708 struct nr_neigh *s, *nr_neigh = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700709 struct nr_node *nr_node = NULL;
710
711 spin_lock_bh(&nr_neigh_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800712 nr_neigh_for_each(s, &nr_neigh_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700713 if (s->ax25 == ax25) {
714 nr_neigh_hold(s);
715 nr_neigh = s;
716 break;
717 }
Ralf Baechle52383672006-06-26 00:05:23 -0700718 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700719 spin_unlock_bh(&nr_neigh_list_lock);
720
Ralf Baechle52383672006-06-26 00:05:23 -0700721 if (nr_neigh == NULL)
722 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700723
724 nr_neigh->ax25 = NULL;
725 ax25_cb_put(ax25);
726
727 if (++nr_neigh->failed < sysctl_netrom_link_fails_count) {
728 nr_neigh_put(nr_neigh);
729 return;
730 }
731 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800732 nr_node_for_each(nr_node, &nr_node_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700733 nr_node_lock(nr_node);
Ralf Baechle52383672006-06-26 00:05:23 -0700734 if (nr_node->which < nr_node->count &&
735 nr_node->routes[nr_node->which].neighbour == nr_neigh)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736 nr_node->which++;
737 nr_node_unlock(nr_node);
Ralf Baechle52383672006-06-26 00:05:23 -0700738 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700739 spin_unlock_bh(&nr_node_list_lock);
740 nr_neigh_put(nr_neigh);
741}
742
743/*
744 * Route a frame to an appropriate AX.25 connection. A NULL ax25_cb
745 * indicates an internally generated frame.
746 */
747int nr_route_frame(struct sk_buff *skb, ax25_cb *ax25)
748{
749 ax25_address *nr_src, *nr_dest;
750 struct nr_neigh *nr_neigh;
751 struct nr_node *nr_node;
752 struct net_device *dev;
753 unsigned char *dptr;
754 ax25_cb *ax25s;
755 int ret;
756 struct sk_buff *skbn;
757
758
759 nr_src = (ax25_address *)(skb->data + 0);
760 nr_dest = (ax25_address *)(skb->data + 7);
761
Ralf Baechle58bc5742006-12-14 15:50:58 -0800762 if (ax25 != NULL) {
763 ret = nr_add_node(nr_src, "", &ax25->dest_addr, ax25->digipeat,
YOSHIFUJI Hideaki5f8f59d2007-02-09 23:25:09 +0900764 ax25->ax25_dev->dev, 0,
765 sysctl_netrom_obsolescence_count_initialiser);
Ralf Baechle58bc5742006-12-14 15:50:58 -0800766 if (ret)
767 return ret;
768 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700769
770 if ((dev = nr_dev_get(nr_dest)) != NULL) { /* Its for me */
771 if (ax25 == NULL) /* Its from me */
772 ret = nr_loopback_queue(skb);
773 else
774 ret = nr_rx_frame(skb, dev);
775 dev_put(dev);
776 return ret;
777 }
778
779 if (!sysctl_netrom_routing_control && ax25 != NULL)
780 return 0;
781
782 /* Its Time-To-Live has expired */
783 if (skb->data[14] == 1) {
784 return 0;
785 }
786
787 nr_node = nr_node_get(nr_dest);
788 if (nr_node == NULL)
789 return 0;
790 nr_node_lock(nr_node);
791
792 if (nr_node->which >= nr_node->count) {
793 nr_node_unlock(nr_node);
794 nr_node_put(nr_node);
795 return 0;
796 }
797
798 nr_neigh = nr_node->routes[nr_node->which].neighbour;
799
800 if ((dev = nr_dev_first()) == NULL) {
801 nr_node_unlock(nr_node);
802 nr_node_put(nr_node);
803 return 0;
804 }
805
806 /* We are going to change the netrom headers so we should get our
807 own skb, we also did not know until now how much header space
808 we had to reserve... - RXQ */
809 if ((skbn=skb_copy_expand(skb, dev->hard_header_len, 0, GFP_ATOMIC)) == NULL) {
810 nr_node_unlock(nr_node);
811 nr_node_put(nr_node);
812 dev_put(dev);
813 return 0;
814 }
815 kfree_skb(skb);
816 skb=skbn;
817 skb->data[14]--;
818
819 dptr = skb_push(skb, 1);
820 *dptr = AX25_P_NETROM;
821
Jarek Poplawskid00c3622010-01-16 01:04:04 -0800822 ax25s = nr_neigh->ax25;
823 nr_neigh->ax25 = ax25_send_frame(skb, 256,
824 (ax25_address *)dev->dev_addr,
825 &nr_neigh->callsign,
826 nr_neigh->digipeat, nr_neigh->dev);
827 if (ax25s)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700828 ax25_cb_put(ax25s);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700829
830 dev_put(dev);
831 ret = (nr_neigh->ax25 != NULL);
832 nr_node_unlock(nr_node);
833 nr_node_put(nr_node);
Ralf Baechle58bc5742006-12-14 15:50:58 -0800834
Linus Torvalds1da177e2005-04-16 15:20:36 -0700835 return ret;
836}
837
838#ifdef CONFIG_PROC_FS
839
840static void *nr_node_start(struct seq_file *seq, loff_t *pos)
Jules Irenge5018adf2020-02-23 23:16:51 +0000841 __acquires(&nr_node_list_lock)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700842{
YOSHIFUJI Hideaki5f8f59d2007-02-09 23:25:09 +0900843 spin_lock_bh(&nr_node_list_lock);
Li Zefan90dd7f52010-02-08 23:19:42 +0000844 return seq_hlist_start_head(&nr_node_list, *pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700845}
846
847static void *nr_node_next(struct seq_file *seq, void *v, loff_t *pos)
848{
Li Zefan90dd7f52010-02-08 23:19:42 +0000849 return seq_hlist_next(v, &nr_node_list, pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850}
851
852static void nr_node_stop(struct seq_file *seq, void *v)
Jules Irenge0eb713f2020-02-23 23:16:52 +0000853 __releases(&nr_node_list_lock)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700854{
855 spin_unlock_bh(&nr_node_list_lock);
856}
857
858static int nr_node_show(struct seq_file *seq, void *v)
859{
Ralf Baechlef75268c2005-09-06 15:49:39 -0700860 char buf[11];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700861 int i;
862
863 if (v == SEQ_START_TOKEN)
864 seq_puts(seq,
865 "callsign mnemonic w n qual obs neigh qual obs neigh qual obs neigh\n");
866 else {
Li Zefan90dd7f52010-02-08 23:19:42 +0000867 struct nr_node *nr_node = hlist_entry(v, struct nr_node,
868 node_node);
869
Linus Torvalds1da177e2005-04-16 15:20:36 -0700870 nr_node_lock(nr_node);
871 seq_printf(seq, "%-9s %-7s %d %d",
Ralf Baechlef75268c2005-09-06 15:49:39 -0700872 ax2asc(buf, &nr_node->callsign),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700873 (nr_node->mnemonic[0] == '\0') ? "*" : nr_node->mnemonic,
874 nr_node->which + 1,
875 nr_node->count);
876
877 for (i = 0; i < nr_node->count; i++) {
878 seq_printf(seq, " %3d %d %05d",
879 nr_node->routes[i].quality,
880 nr_node->routes[i].obs_count,
881 nr_node->routes[i].neighbour->number);
882 }
883 nr_node_unlock(nr_node);
884
885 seq_puts(seq, "\n");
886 }
887 return 0;
888}
889
Christoph Hellwigfddda2b2018-04-13 19:44:18 +0200890const struct seq_operations nr_node_seqops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700891 .start = nr_node_start,
892 .next = nr_node_next,
893 .stop = nr_node_stop,
894 .show = nr_node_show,
895};
896
Linus Torvalds1da177e2005-04-16 15:20:36 -0700897static void *nr_neigh_start(struct seq_file *seq, loff_t *pos)
Jules Irenge2d6b6ac2020-02-23 23:16:53 +0000898 __acquires(&nr_neigh_list_lock)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700899{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700900 spin_lock_bh(&nr_neigh_list_lock);
Li Zefan90dd7f52010-02-08 23:19:42 +0000901 return seq_hlist_start_head(&nr_neigh_list, *pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700902}
903
904static void *nr_neigh_next(struct seq_file *seq, void *v, loff_t *pos)
905{
Li Zefan90dd7f52010-02-08 23:19:42 +0000906 return seq_hlist_next(v, &nr_neigh_list, pos);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700907}
908
909static void nr_neigh_stop(struct seq_file *seq, void *v)
910{
911 spin_unlock_bh(&nr_neigh_list_lock);
912}
913
914static int nr_neigh_show(struct seq_file *seq, void *v)
915{
Ralf Baechlef75268c2005-09-06 15:49:39 -0700916 char buf[11];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700917 int i;
918
919 if (v == SEQ_START_TOKEN)
920 seq_puts(seq, "addr callsign dev qual lock count failed digipeaters\n");
921 else {
Li Zefan90dd7f52010-02-08 23:19:42 +0000922 struct nr_neigh *nr_neigh;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700923
Li Zefan90dd7f52010-02-08 23:19:42 +0000924 nr_neigh = hlist_entry(v, struct nr_neigh, neigh_node);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700925 seq_printf(seq, "%05d %-9s %-4s %3d %d %3d %3d",
926 nr_neigh->number,
Ralf Baechlef75268c2005-09-06 15:49:39 -0700927 ax2asc(buf, &nr_neigh->callsign),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700928 nr_neigh->dev ? nr_neigh->dev->name : "???",
929 nr_neigh->quality,
930 nr_neigh->locked,
931 nr_neigh->count,
932 nr_neigh->failed);
933
934 if (nr_neigh->digipeat != NULL) {
935 for (i = 0; i < nr_neigh->digipeat->ndigi; i++)
YOSHIFUJI Hideaki5f8f59d2007-02-09 23:25:09 +0900936 seq_printf(seq, " %s",
Ralf Baechlef75268c2005-09-06 15:49:39 -0700937 ax2asc(buf, &nr_neigh->digipeat->calls[i]));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700938 }
939
940 seq_puts(seq, "\n");
941 }
942 return 0;
943}
944
Christoph Hellwigfddda2b2018-04-13 19:44:18 +0200945const struct seq_operations nr_neigh_seqops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700946 .start = nr_neigh_start,
947 .next = nr_neigh_next,
948 .stop = nr_neigh_stop,
949 .show = nr_neigh_show,
950};
Linus Torvalds1da177e2005-04-16 15:20:36 -0700951#endif
952
953/*
954 * Free all memory associated with the nodes and routes lists.
955 */
YueHaibingd3706562019-04-09 19:53:55 +0800956void nr_rt_free(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700957{
958 struct nr_neigh *s = NULL;
959 struct nr_node *t = NULL;
Sasha Levinb67bfe02013-02-27 17:06:00 -0800960 struct hlist_node *nodet;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700961
962 spin_lock_bh(&nr_neigh_list_lock);
963 spin_lock_bh(&nr_node_list_lock);
Sasha Levinb67bfe02013-02-27 17:06:00 -0800964 nr_node_for_each_safe(t, nodet, &nr_node_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700965 nr_node_lock(t);
966 nr_remove_node_locked(t);
967 nr_node_unlock(t);
968 }
Sasha Levinb67bfe02013-02-27 17:06:00 -0800969 nr_neigh_for_each_safe(s, nodet, &nr_neigh_list) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700970 while(s->count) {
971 s->count--;
972 nr_neigh_put(s);
973 }
974 nr_remove_neigh_locked(s);
975 }
976 spin_unlock_bh(&nr_node_list_lock);
977 spin_unlock_bh(&nr_neigh_list_lock);
978}