2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or
5 * (at your option) any later version.
7 * Copyright Jonathan Naylor G4KLX (g4klx@g4klx.demon.co.uk)
8 * Copyright Alan Cox GW4PTS (alan@lxorguk.ukuu.org.uk)
9 * Copyright Tomi Manninen OH2BNS (oh2bns@sral.fi)
11 #include <linux/errno.h>
12 #include <linux/types.h>
13 #include <linux/socket.h>
15 #include <linux/kernel.h>
16 #include <linux/sched.h>
17 #include <linux/timer.h>
18 #include <linux/string.h>
19 #include <linux/sockios.h>
20 #include <linux/net.h>
22 #include <linux/inet.h>
23 #include <linux/netdevice.h>
25 #include <linux/if_arp.h>
26 #include <linux/skbuff.h>
28 #include <asm/uaccess.h>
29 #include <asm/system.h>
30 #include <linux/fcntl.h>
31 #include <linux/termios.h> /* For TIOCINQ/OUTQ */
33 #include <linux/interrupt.h>
34 #include <linux/notifier.h>
35 #include <linux/netfilter.h>
36 #include <linux/init.h>
37 #include <linux/spinlock.h>
38 #include <net/netrom.h>
39 #include <linux/seq_file.h>
41 static unsigned int nr_neigh_no
= 1;
43 static HLIST_HEAD(nr_node_list
);
44 static DEFINE_SPINLOCK(nr_node_list_lock
);
45 static HLIST_HEAD(nr_neigh_list
);
46 static DEFINE_SPINLOCK(nr_neigh_list_lock
);
48 static struct nr_node
*nr_node_get(ax25_address
*callsign
)
50 struct nr_node
*found
= NULL
;
51 struct nr_node
*nr_node
;
52 struct hlist_node
*node
;
54 spin_lock_bh(&nr_node_list_lock
);
55 nr_node_for_each(nr_node
, node
, &nr_node_list
)
56 if (ax25cmp(callsign
, &nr_node
->callsign
) == 0) {
57 nr_node_hold(nr_node
);
61 spin_unlock_bh(&nr_node_list_lock
);
65 static struct nr_neigh
*nr_neigh_get_dev(ax25_address
*callsign
,
66 struct net_device
*dev
)
68 struct nr_neigh
*found
= NULL
;
69 struct nr_neigh
*nr_neigh
;
70 struct hlist_node
*node
;
72 spin_lock_bh(&nr_neigh_list_lock
);
73 nr_neigh_for_each(nr_neigh
, node
, &nr_neigh_list
)
74 if (ax25cmp(callsign
, &nr_neigh
->callsign
) == 0 &&
75 nr_neigh
->dev
== dev
) {
76 nr_neigh_hold(nr_neigh
);
80 spin_unlock_bh(&nr_neigh_list_lock
);
84 static void nr_remove_neigh(struct nr_neigh
*);
87 * Add a new route to a node, and in the process add the node and the
88 * neighbour if it is new.
90 static int nr_add_node(ax25_address
*nr
, const char *mnemonic
, ax25_address
*ax25
,
91 ax25_digi
*ax25_digi
, struct net_device
*dev
, int quality
, int obs_count
)
93 struct nr_node
*nr_node
;
94 struct nr_neigh
*nr_neigh
;
95 struct nr_route nr_route
;
97 struct net_device
*odev
;
99 if ((odev
=nr_dev_get(nr
)) != NULL
) { /* Can't add routes to ourself */
104 nr_node
= nr_node_get(nr
);
106 nr_neigh
= nr_neigh_get_dev(ax25
, dev
);
109 * The L2 link to a neighbour has failed in the past
110 * and now a frame comes from this neighbour. We assume
111 * it was a temporary trouble with the link and reset the
112 * routes now (and not wait for a node broadcast).
114 if (nr_neigh
!= NULL
&& nr_neigh
->failed
!= 0 && quality
== 0) {
115 struct nr_node
*nr_nodet
;
116 struct hlist_node
*node
;
118 spin_lock_bh(&nr_node_list_lock
);
119 nr_node_for_each(nr_nodet
, node
, &nr_node_list
) {
120 nr_node_lock(nr_nodet
);
121 for (i
= 0; i
< nr_nodet
->count
; i
++)
122 if (nr_nodet
->routes
[i
].neighbour
== nr_neigh
)
123 if (i
< nr_nodet
->which
)
125 nr_node_unlock(nr_nodet
);
127 spin_unlock_bh(&nr_node_list_lock
);
130 if (nr_neigh
!= NULL
)
131 nr_neigh
->failed
= 0;
133 if (quality
== 0 && nr_neigh
!= NULL
&& nr_node
!= NULL
) {
134 nr_neigh_put(nr_neigh
);
135 nr_node_put(nr_node
);
139 if (nr_neigh
== NULL
) {
140 if ((nr_neigh
= kmalloc(sizeof(*nr_neigh
), GFP_ATOMIC
)) == NULL
) {
142 nr_node_put(nr_node
);
146 nr_neigh
->callsign
= *ax25
;
147 nr_neigh
->digipeat
= NULL
;
148 nr_neigh
->ax25
= NULL
;
150 nr_neigh
->quality
= sysctl_netrom_default_path_quality
;
151 nr_neigh
->locked
= 0;
153 nr_neigh
->number
= nr_neigh_no
++;
154 nr_neigh
->failed
= 0;
155 atomic_set(&nr_neigh
->refcount
, 1);
157 if (ax25_digi
!= NULL
&& ax25_digi
->ndigi
> 0) {
158 nr_neigh
->digipeat
= kmemdup(ax25_digi
,
161 if (nr_neigh
->digipeat
== NULL
) {
164 nr_node_put(nr_node
);
169 spin_lock_bh(&nr_neigh_list_lock
);
170 hlist_add_head(&nr_neigh
->neigh_node
, &nr_neigh_list
);
171 nr_neigh_hold(nr_neigh
);
172 spin_unlock_bh(&nr_neigh_list_lock
);
175 if (quality
!= 0 && ax25cmp(nr
, ax25
) == 0 && !nr_neigh
->locked
)
176 nr_neigh
->quality
= quality
;
178 if (nr_node
== NULL
) {
179 if ((nr_node
= kmalloc(sizeof(*nr_node
), GFP_ATOMIC
)) == NULL
) {
181 nr_neigh_put(nr_neigh
);
185 nr_node
->callsign
= *nr
;
186 strcpy(nr_node
->mnemonic
, mnemonic
);
190 atomic_set(&nr_node
->refcount
, 1);
191 spin_lock_init(&nr_node
->node_lock
);
193 nr_node
->routes
[0].quality
= quality
;
194 nr_node
->routes
[0].obs_count
= obs_count
;
195 nr_node
->routes
[0].neighbour
= nr_neigh
;
197 nr_neigh_hold(nr_neigh
);
200 spin_lock_bh(&nr_node_list_lock
);
201 hlist_add_head(&nr_node
->node_node
, &nr_node_list
);
202 /* refcount initialized at 1 */
203 spin_unlock_bh(&nr_node_list_lock
);
207 nr_node_lock(nr_node
);
210 strcpy(nr_node
->mnemonic
, mnemonic
);
212 for (found
= 0, i
= 0; i
< nr_node
->count
; i
++) {
213 if (nr_node
->routes
[i
].neighbour
== nr_neigh
) {
214 nr_node
->routes
[i
].quality
= quality
;
215 nr_node
->routes
[i
].obs_count
= obs_count
;
222 /* We have space at the bottom, slot it in */
223 if (nr_node
->count
< 3) {
224 nr_node
->routes
[2] = nr_node
->routes
[1];
225 nr_node
->routes
[1] = nr_node
->routes
[0];
227 nr_node
->routes
[0].quality
= quality
;
228 nr_node
->routes
[0].obs_count
= obs_count
;
229 nr_node
->routes
[0].neighbour
= nr_neigh
;
233 nr_neigh_hold(nr_neigh
);
236 /* It must be better than the worst */
237 if (quality
> nr_node
->routes
[2].quality
) {
238 nr_node
->routes
[2].neighbour
->count
--;
239 nr_neigh_put(nr_node
->routes
[2].neighbour
);
241 if (nr_node
->routes
[2].neighbour
->count
== 0 && !nr_node
->routes
[2].neighbour
->locked
)
242 nr_remove_neigh(nr_node
->routes
[2].neighbour
);
244 nr_node
->routes
[2].quality
= quality
;
245 nr_node
->routes
[2].obs_count
= obs_count
;
246 nr_node
->routes
[2].neighbour
= nr_neigh
;
248 nr_neigh_hold(nr_neigh
);
254 /* Now re-sort the routes in quality order */
255 switch (nr_node
->count
) {
257 if (nr_node
->routes
[1].quality
> nr_node
->routes
[0].quality
) {
258 switch (nr_node
->which
) {
259 case 0: nr_node
->which
= 1; break;
260 case 1: nr_node
->which
= 0; break;
263 nr_route
= nr_node
->routes
[0];
264 nr_node
->routes
[0] = nr_node
->routes
[1];
265 nr_node
->routes
[1] = nr_route
;
267 if (nr_node
->routes
[2].quality
> nr_node
->routes
[1].quality
) {
268 switch (nr_node
->which
) {
269 case 1: nr_node
->which
= 2;
272 case 2: nr_node
->which
= 1;
278 nr_route
= nr_node
->routes
[1];
279 nr_node
->routes
[1] = nr_node
->routes
[2];
280 nr_node
->routes
[2] = nr_route
;
283 if (nr_node
->routes
[1].quality
> nr_node
->routes
[0].quality
) {
284 switch (nr_node
->which
) {
285 case 0: nr_node
->which
= 1;
288 case 1: nr_node
->which
= 0;
293 nr_route
= nr_node
->routes
[0];
294 nr_node
->routes
[0] = nr_node
->routes
[1];
295 nr_node
->routes
[1] = nr_route
;
301 for (i
= 0; i
< nr_node
->count
; i
++) {
302 if (nr_node
->routes
[i
].neighbour
== nr_neigh
) {
303 if (i
< nr_node
->which
)
309 nr_neigh_put(nr_neigh
);
310 nr_node_unlock(nr_node
);
311 nr_node_put(nr_node
);
315 static inline void __nr_remove_node(struct nr_node
*nr_node
)
317 hlist_del_init(&nr_node
->node_node
);
318 nr_node_put(nr_node
);
321 #define nr_remove_node_locked(__node) \
322 __nr_remove_node(__node)
324 static void nr_remove_node(struct nr_node
*nr_node
)
326 spin_lock_bh(&nr_node_list_lock
);
327 __nr_remove_node(nr_node
);
328 spin_unlock_bh(&nr_node_list_lock
);
331 static inline void __nr_remove_neigh(struct nr_neigh
*nr_neigh
)
333 hlist_del_init(&nr_neigh
->neigh_node
);
334 nr_neigh_put(nr_neigh
);
337 #define nr_remove_neigh_locked(__neigh) \
338 __nr_remove_neigh(__neigh)
340 static void nr_remove_neigh(struct nr_neigh
*nr_neigh
)
342 spin_lock_bh(&nr_neigh_list_lock
);
343 __nr_remove_neigh(nr_neigh
);
344 spin_unlock_bh(&nr_neigh_list_lock
);
348 * "Delete" a node. Strictly speaking remove a route to a node. The node
349 * is only deleted if no routes are left to it.
351 static int nr_del_node(ax25_address
*callsign
, ax25_address
*neighbour
, struct net_device
*dev
)
353 struct nr_node
*nr_node
;
354 struct nr_neigh
*nr_neigh
;
357 nr_node
= nr_node_get(callsign
);
362 nr_neigh
= nr_neigh_get_dev(neighbour
, dev
);
364 if (nr_neigh
== NULL
) {
365 nr_node_put(nr_node
);
369 nr_node_lock(nr_node
);
370 for (i
= 0; i
< nr_node
->count
; i
++) {
371 if (nr_node
->routes
[i
].neighbour
== nr_neigh
) {
373 nr_neigh_put(nr_neigh
);
375 if (nr_neigh
->count
== 0 && !nr_neigh
->locked
)
376 nr_remove_neigh(nr_neigh
);
377 nr_neigh_put(nr_neigh
);
381 if (nr_node
->count
== 0) {
382 nr_remove_node(nr_node
);
386 nr_node
->routes
[0] = nr_node
->routes
[1];
388 nr_node
->routes
[1] = nr_node
->routes
[2];
392 nr_node_put(nr_node
);
394 nr_node_unlock(nr_node
);
399 nr_neigh_put(nr_neigh
);
400 nr_node_unlock(nr_node
);
401 nr_node_put(nr_node
);
407 * Lock a neighbour with a quality.
409 static int nr_add_neigh(ax25_address
*callsign
, ax25_digi
*ax25_digi
, struct net_device
*dev
, unsigned int quality
)
411 struct nr_neigh
*nr_neigh
;
413 nr_neigh
= nr_neigh_get_dev(callsign
, dev
);
415 nr_neigh
->quality
= quality
;
416 nr_neigh
->locked
= 1;
417 nr_neigh_put(nr_neigh
);
421 if ((nr_neigh
= kmalloc(sizeof(*nr_neigh
), GFP_ATOMIC
)) == NULL
)
424 nr_neigh
->callsign
= *callsign
;
425 nr_neigh
->digipeat
= NULL
;
426 nr_neigh
->ax25
= NULL
;
428 nr_neigh
->quality
= quality
;
429 nr_neigh
->locked
= 1;
431 nr_neigh
->number
= nr_neigh_no
++;
432 nr_neigh
->failed
= 0;
433 atomic_set(&nr_neigh
->refcount
, 1);
435 if (ax25_digi
!= NULL
&& ax25_digi
->ndigi
> 0) {
436 nr_neigh
->digipeat
= kmemdup(ax25_digi
, sizeof(*ax25_digi
),
438 if (nr_neigh
->digipeat
== NULL
) {
444 spin_lock_bh(&nr_neigh_list_lock
);
445 hlist_add_head(&nr_neigh
->neigh_node
, &nr_neigh_list
);
446 /* refcount is initialized at 1 */
447 spin_unlock_bh(&nr_neigh_list_lock
);
453 * "Delete" a neighbour. The neighbour is only removed if the number
454 * of nodes that may use it is zero.
456 static int nr_del_neigh(ax25_address
*callsign
, struct net_device
*dev
, unsigned int quality
)
458 struct nr_neigh
*nr_neigh
;
460 nr_neigh
= nr_neigh_get_dev(callsign
, dev
);
462 if (nr_neigh
== NULL
) return -EINVAL
;
464 nr_neigh
->quality
= quality
;
465 nr_neigh
->locked
= 0;
467 if (nr_neigh
->count
== 0)
468 nr_remove_neigh(nr_neigh
);
469 nr_neigh_put(nr_neigh
);
475 * Decrement the obsolescence count by one. If a route is reduced to a
476 * count of zero, remove it. Also remove any unlocked neighbours with
477 * zero nodes routing via it.
479 static int nr_dec_obs(void)
481 struct nr_neigh
*nr_neigh
;
483 struct hlist_node
*node
, *nodet
;
486 spin_lock_bh(&nr_node_list_lock
);
487 nr_node_for_each_safe(s
, node
, nodet
, &nr_node_list
) {
489 for (i
= 0; i
< s
->count
; i
++) {
490 switch (s
->routes
[i
].obs_count
) {
491 case 0: /* A locked entry */
494 case 1: /* From 1 -> 0 */
495 nr_neigh
= s
->routes
[i
].neighbour
;
498 nr_neigh_put(nr_neigh
);
500 if (nr_neigh
->count
== 0 && !nr_neigh
->locked
)
501 nr_remove_neigh(nr_neigh
);
507 s
->routes
[0] = s
->routes
[1];
509 s
->routes
[1] = s
->routes
[2];
516 s
->routes
[i
].obs_count
--;
523 nr_remove_node_locked(s
);
526 spin_unlock_bh(&nr_node_list_lock
);
532 * A device has been removed. Remove its routes and neighbours.
534 void nr_rt_device_down(struct net_device
*dev
)
537 struct hlist_node
*node
, *nodet
, *node2
, *node2t
;
541 spin_lock_bh(&nr_neigh_list_lock
);
542 nr_neigh_for_each_safe(s
, node
, nodet
, &nr_neigh_list
) {
544 spin_lock_bh(&nr_node_list_lock
);
545 nr_node_for_each_safe(t
, node2
, node2t
, &nr_node_list
) {
547 for (i
= 0; i
< t
->count
; i
++) {
548 if (t
->routes
[i
].neighbour
== s
) {
553 t
->routes
[0] = t
->routes
[1];
555 t
->routes
[1] = t
->routes
[2];
563 nr_remove_node_locked(t
);
566 spin_unlock_bh(&nr_node_list_lock
);
568 nr_remove_neigh_locked(s
);
571 spin_unlock_bh(&nr_neigh_list_lock
);
575 * Check that the device given is a valid AX.25 interface that is "up".
576 * Or a valid ethernet interface with an AX.25 callsign binding.
578 static struct net_device
*nr_ax25_dev_get(char *devname
)
580 struct net_device
*dev
;
582 if ((dev
= dev_get_by_name(devname
)) == NULL
)
585 if ((dev
->flags
& IFF_UP
) && dev
->type
== ARPHRD_AX25
)
593 * Find the first active NET/ROM device, usually "nr0".
595 struct net_device
*nr_dev_first(void)
597 struct net_device
*dev
, *first
= NULL
;
599 read_lock(&dev_base_lock
);
600 for (dev
= dev_base
; dev
!= NULL
; dev
= dev
->next
) {
601 if ((dev
->flags
& IFF_UP
) && dev
->type
== ARPHRD_NETROM
)
602 if (first
== NULL
|| strncmp(dev
->name
, first
->name
, 3) < 0)
607 read_unlock(&dev_base_lock
);
613 * Find the NET/ROM device for the given callsign.
615 struct net_device
*nr_dev_get(ax25_address
*addr
)
617 struct net_device
*dev
;
619 read_lock(&dev_base_lock
);
620 for (dev
= dev_base
; dev
!= NULL
; dev
= dev
->next
) {
621 if ((dev
->flags
& IFF_UP
) && dev
->type
== ARPHRD_NETROM
&& ax25cmp(addr
, (ax25_address
*)dev
->dev_addr
) == 0) {
627 read_unlock(&dev_base_lock
);
631 static ax25_digi
*nr_call_to_digi(int ndigis
, ax25_address
*digipeaters
)
633 static ax25_digi ax25_digi
;
639 for (i
= 0; i
< ndigis
; i
++) {
640 ax25_digi
.calls
[i
] = digipeaters
[i
];
641 ax25_digi
.repeated
[i
] = 0;
644 ax25_digi
.ndigi
= ndigis
;
645 ax25_digi
.lastrepeat
= -1;
651 * Handle the ioctls that control the routing functions.
653 int nr_rt_ioctl(unsigned int cmd
, void __user
*arg
)
655 struct nr_route_struct nr_route
;
656 struct net_device
*dev
;
661 if (copy_from_user(&nr_route
, arg
, sizeof(struct nr_route_struct
)))
663 if ((dev
= nr_ax25_dev_get(nr_route
.device
)) == NULL
)
665 if (nr_route
.ndigis
< 0 || nr_route
.ndigis
> AX25_MAX_DIGIS
) {
669 switch (nr_route
.type
) {
671 ret
= nr_add_node(&nr_route
.callsign
,
674 nr_call_to_digi(nr_route
.ndigis
, nr_route
.digipeaters
),
675 dev
, nr_route
.quality
,
679 ret
= nr_add_neigh(&nr_route
.callsign
,
680 nr_call_to_digi(nr_route
.ndigis
, nr_route
.digipeaters
),
681 dev
, nr_route
.quality
);
690 if (copy_from_user(&nr_route
, arg
, sizeof(struct nr_route_struct
)))
692 if ((dev
= nr_ax25_dev_get(nr_route
.device
)) == NULL
)
694 switch (nr_route
.type
) {
696 ret
= nr_del_node(&nr_route
.callsign
,
697 &nr_route
.neighbour
, dev
);
700 ret
= nr_del_neigh(&nr_route
.callsign
,
701 dev
, nr_route
.quality
);
720 * A level 2 link has timed out, therefore it appears to be a poor link,
721 * then don't use that neighbour until it is reset.
723 void nr_link_failed(ax25_cb
*ax25
, int reason
)
725 struct nr_neigh
*s
, *nr_neigh
= NULL
;
726 struct hlist_node
*node
;
727 struct nr_node
*nr_node
= NULL
;
729 spin_lock_bh(&nr_neigh_list_lock
);
730 nr_neigh_for_each(s
, node
, &nr_neigh_list
) {
731 if (s
->ax25
== ax25
) {
737 spin_unlock_bh(&nr_neigh_list_lock
);
739 if (nr_neigh
== NULL
)
742 nr_neigh
->ax25
= NULL
;
745 if (++nr_neigh
->failed
< sysctl_netrom_link_fails_count
) {
746 nr_neigh_put(nr_neigh
);
749 spin_lock_bh(&nr_node_list_lock
);
750 nr_node_for_each(nr_node
, node
, &nr_node_list
) {
751 nr_node_lock(nr_node
);
752 if (nr_node
->which
< nr_node
->count
&&
753 nr_node
->routes
[nr_node
->which
].neighbour
== nr_neigh
)
755 nr_node_unlock(nr_node
);
757 spin_unlock_bh(&nr_node_list_lock
);
758 nr_neigh_put(nr_neigh
);
762 * Route a frame to an appropriate AX.25 connection. A NULL ax25_cb
763 * indicates an internally generated frame.
765 int nr_route_frame(struct sk_buff
*skb
, ax25_cb
*ax25
)
767 ax25_address
*nr_src
, *nr_dest
;
768 struct nr_neigh
*nr_neigh
;
769 struct nr_node
*nr_node
;
770 struct net_device
*dev
;
774 struct sk_buff
*skbn
;
777 nr_src
= (ax25_address
*)(skb
->data
+ 0);
778 nr_dest
= (ax25_address
*)(skb
->data
+ 7);
781 nr_add_node(nr_src
, "", &ax25
->dest_addr
, ax25
->digipeat
,
782 ax25
->ax25_dev
->dev
, 0, sysctl_netrom_obsolescence_count_initialiser
);
784 if ((dev
= nr_dev_get(nr_dest
)) != NULL
) { /* Its for me */
785 if (ax25
== NULL
) /* Its from me */
786 ret
= nr_loopback_queue(skb
);
788 ret
= nr_rx_frame(skb
, dev
);
793 if (!sysctl_netrom_routing_control
&& ax25
!= NULL
)
796 /* Its Time-To-Live has expired */
797 if (skb
->data
[14] == 1) {
801 nr_node
= nr_node_get(nr_dest
);
804 nr_node_lock(nr_node
);
806 if (nr_node
->which
>= nr_node
->count
) {
807 nr_node_unlock(nr_node
);
808 nr_node_put(nr_node
);
812 nr_neigh
= nr_node
->routes
[nr_node
->which
].neighbour
;
814 if ((dev
= nr_dev_first()) == NULL
) {
815 nr_node_unlock(nr_node
);
816 nr_node_put(nr_node
);
820 /* We are going to change the netrom headers so we should get our
821 own skb, we also did not know until now how much header space
822 we had to reserve... - RXQ */
823 if ((skbn
=skb_copy_expand(skb
, dev
->hard_header_len
, 0, GFP_ATOMIC
)) == NULL
) {
824 nr_node_unlock(nr_node
);
825 nr_node_put(nr_node
);
833 dptr
= skb_push(skb
, 1);
834 *dptr
= AX25_P_NETROM
;
836 ax25s
= ax25_send_frame(skb
, 256, (ax25_address
*)dev
->dev_addr
, &nr_neigh
->callsign
, nr_neigh
->digipeat
, nr_neigh
->dev
);
837 if (nr_neigh
->ax25
&& ax25s
) {
838 /* We were already holding this ax25_cb */
841 nr_neigh
->ax25
= ax25s
;
844 ret
= (nr_neigh
->ax25
!= NULL
);
845 nr_node_unlock(nr_node
);
846 nr_node_put(nr_node
);
850 #ifdef CONFIG_PROC_FS
852 static void *nr_node_start(struct seq_file
*seq
, loff_t
*pos
)
854 struct nr_node
*nr_node
;
855 struct hlist_node
*node
;
858 spin_lock_bh(&nr_node_list_lock
);
860 return SEQ_START_TOKEN
;
862 nr_node_for_each(nr_node
, node
, &nr_node_list
) {
871 static void *nr_node_next(struct seq_file
*seq
, void *v
, loff_t
*pos
)
873 struct hlist_node
*node
;
876 node
= (v
== SEQ_START_TOKEN
)
878 : ((struct nr_node
*)v
)->node_node
.next
;
880 return hlist_entry(node
, struct nr_node
, node_node
);
883 static void nr_node_stop(struct seq_file
*seq
, void *v
)
885 spin_unlock_bh(&nr_node_list_lock
);
888 static int nr_node_show(struct seq_file
*seq
, void *v
)
893 if (v
== SEQ_START_TOKEN
)
895 "callsign mnemonic w n qual obs neigh qual obs neigh qual obs neigh\n");
897 struct nr_node
*nr_node
= v
;
898 nr_node_lock(nr_node
);
899 seq_printf(seq
, "%-9s %-7s %d %d",
900 ax2asc(buf
, &nr_node
->callsign
),
901 (nr_node
->mnemonic
[0] == '\0') ? "*" : nr_node
->mnemonic
,
905 for (i
= 0; i
< nr_node
->count
; i
++) {
906 seq_printf(seq
, " %3d %d %05d",
907 nr_node
->routes
[i
].quality
,
908 nr_node
->routes
[i
].obs_count
,
909 nr_node
->routes
[i
].neighbour
->number
);
911 nr_node_unlock(nr_node
);
918 static struct seq_operations nr_node_seqops
= {
919 .start
= nr_node_start
,
920 .next
= nr_node_next
,
921 .stop
= nr_node_stop
,
922 .show
= nr_node_show
,
925 static int nr_node_info_open(struct inode
*inode
, struct file
*file
)
927 return seq_open(file
, &nr_node_seqops
);
930 struct file_operations nr_nodes_fops
= {
931 .owner
= THIS_MODULE
,
932 .open
= nr_node_info_open
,
935 .release
= seq_release
,
938 static void *nr_neigh_start(struct seq_file
*seq
, loff_t
*pos
)
940 struct nr_neigh
*nr_neigh
;
941 struct hlist_node
*node
;
944 spin_lock_bh(&nr_neigh_list_lock
);
946 return SEQ_START_TOKEN
;
948 nr_neigh_for_each(nr_neigh
, node
, &nr_neigh_list
) {
955 static void *nr_neigh_next(struct seq_file
*seq
, void *v
, loff_t
*pos
)
957 struct hlist_node
*node
;
960 node
= (v
== SEQ_START_TOKEN
)
961 ? nr_neigh_list
.first
962 : ((struct nr_neigh
*)v
)->neigh_node
.next
;
964 return hlist_entry(node
, struct nr_neigh
, neigh_node
);
967 static void nr_neigh_stop(struct seq_file
*seq
, void *v
)
969 spin_unlock_bh(&nr_neigh_list_lock
);
972 static int nr_neigh_show(struct seq_file
*seq
, void *v
)
977 if (v
== SEQ_START_TOKEN
)
978 seq_puts(seq
, "addr callsign dev qual lock count failed digipeaters\n");
980 struct nr_neigh
*nr_neigh
= v
;
982 seq_printf(seq
, "%05d %-9s %-4s %3d %d %3d %3d",
984 ax2asc(buf
, &nr_neigh
->callsign
),
985 nr_neigh
->dev
? nr_neigh
->dev
->name
: "???",
991 if (nr_neigh
->digipeat
!= NULL
) {
992 for (i
= 0; i
< nr_neigh
->digipeat
->ndigi
; i
++)
993 seq_printf(seq
, " %s",
994 ax2asc(buf
, &nr_neigh
->digipeat
->calls
[i
]));
1002 static struct seq_operations nr_neigh_seqops
= {
1003 .start
= nr_neigh_start
,
1004 .next
= nr_neigh_next
,
1005 .stop
= nr_neigh_stop
,
1006 .show
= nr_neigh_show
,
1009 static int nr_neigh_info_open(struct inode
*inode
, struct file
*file
)
1011 return seq_open(file
, &nr_neigh_seqops
);
1014 struct file_operations nr_neigh_fops
= {
1015 .owner
= THIS_MODULE
,
1016 .open
= nr_neigh_info_open
,
1018 .llseek
= seq_lseek
,
1019 .release
= seq_release
,
1025 * Free all memory associated with the nodes and routes lists.
1027 void __exit
nr_rt_free(void)
1029 struct nr_neigh
*s
= NULL
;
1030 struct nr_node
*t
= NULL
;
1031 struct hlist_node
*node
, *nodet
;
1033 spin_lock_bh(&nr_neigh_list_lock
);
1034 spin_lock_bh(&nr_node_list_lock
);
1035 nr_node_for_each_safe(t
, node
, nodet
, &nr_node_list
) {
1037 nr_remove_node_locked(t
);
1040 nr_neigh_for_each_safe(s
, node
, nodet
, &nr_neigh_list
) {
1045 nr_remove_neigh_locked(s
);
1047 spin_unlock_bh(&nr_node_list_lock
);
1048 spin_unlock_bh(&nr_neigh_list_lock
);