3 * This module is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU General Public License
5 * as published by the Free Software Foundation; either version
6 * 2 of the License, or (at your option) any later version.
9 * 03-01-2007 Added forwarding for x.25 Andrew Hendry
11 #include <linux/if_arp.h>
12 #include <linux/init.h>
15 struct list_head x25_forward_list
= LIST_HEAD_INIT(x25_forward_list
);
16 DEFINE_RWLOCK(x25_forward_list_lock
);
18 int x25_forward_call(struct x25_address
*dest_addr
, struct x25_neigh
*from
,
19 struct sk_buff
*skb
, int lci
)
22 struct x25_neigh
*neigh_new
= NULL
;
23 struct list_head
*entry
;
24 struct x25_forward
*x25_frwd
, *new_frwd
;
29 if ((rt
= x25_get_route(dest_addr
)) != NULL
) {
31 if ((neigh_new
= x25_get_neigh(rt
->dev
)) == NULL
) {
32 /* This shouldnt happen, if it occurs somehow
33 * do something sensible
38 /* Avoid a loop. This is the normal exit path for a
39 * system with only one x.25 iface and default route
41 if (rt
->dev
== from
->dev
) {
45 /* Remote end sending a call request on an already
46 * established LCI? It shouldnt happen, just in case..
48 read_lock_bh(&x25_forward_list_lock
);
49 list_for_each(entry
, &x25_forward_list
) {
50 x25_frwd
= list_entry(entry
, struct x25_forward
, node
);
51 if (x25_frwd
->lci
== lci
) {
52 printk(KERN_WARNING
"X.25: call request for lci which is already registered!, transmitting but not registering new pair\n");
56 read_unlock_bh(&x25_forward_list_lock
);
58 /* Save the forwarding details for future traffic */
60 if ((new_frwd
= kmalloc(sizeof(struct x25_forward
),
61 GFP_ATOMIC
)) == NULL
){
66 new_frwd
->dev1
= rt
->dev
;
67 new_frwd
->dev2
= from
->dev
;
68 write_lock_bh(&x25_forward_list_lock
);
69 list_add(&new_frwd
->node
, &x25_forward_list
);
70 write_unlock_bh(&x25_forward_list_lock
);
73 /* Forward the call request */
74 if ( (skbn
= skb_clone(skb
, GFP_ATOMIC
)) == NULL
){
77 x25_transmit_link(skbn
, neigh_new
);
83 x25_neigh_put(neigh_new
);
91 int x25_forward_data(int lci
, struct x25_neigh
*from
, struct sk_buff
*skb
) {
93 struct x25_forward
*frwd
;
94 struct list_head
*entry
;
95 struct net_device
*peer
= NULL
;
100 read_lock_bh(&x25_forward_list_lock
);
101 list_for_each(entry
, &x25_forward_list
) {
102 frwd
= list_entry(entry
, struct x25_forward
, node
);
103 if (frwd
->lci
== lci
) {
104 /* The call is established, either side can send */
105 if (from
->dev
== frwd
->dev1
) {
113 read_unlock_bh(&x25_forward_list_lock
);
115 if ( (nb
= x25_get_neigh(peer
)) == NULL
)
118 if ( (skbn
= pskb_copy(skb
, GFP_ATOMIC
)) == NULL
){
122 x25_transmit_link(skbn
, nb
);
130 void x25_clear_forward_by_lci(unsigned int lci
)
132 struct x25_forward
*fwd
;
133 struct list_head
*entry
, *tmp
;
135 write_lock_bh(&x25_forward_list_lock
);
137 list_for_each_safe(entry
, tmp
, &x25_forward_list
) {
138 fwd
= list_entry(entry
, struct x25_forward
, node
);
139 if (fwd
->lci
== lci
) {
140 list_del(&fwd
->node
);
144 write_unlock_bh(&x25_forward_list_lock
);
148 void x25_clear_forward_by_dev(struct net_device
*dev
)
150 struct x25_forward
*fwd
;
151 struct list_head
*entry
, *tmp
;
153 write_lock_bh(&x25_forward_list_lock
);
155 list_for_each_safe(entry
, tmp
, &x25_forward_list
) {
156 fwd
= list_entry(entry
, struct x25_forward
, node
);
157 if ((fwd
->dev1
== dev
) || (fwd
->dev2
== dev
)){
158 list_del(&fwd
->node
);
162 write_unlock_bh(&x25_forward_list_lock
);