Linux kernel & device driver programming

Cross-Referenced Linux and Device Driver Code

[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ]
Version: [ 2.6.11.8 ] [ 2.6.25 ] [ 2.6.25.8 ] [ 2.6.31.13 ] Architecture: [ i386 ]
  1 /*
  2  *      Implements the IPX routing routines.
  3  *      Code moved from af_ipx.c.
  4  *
  5  *      Arnaldo Carvalho de Melo <acme@conectiva.com.br>, 2003
  6  *
  7  *      See net/ipx/ChangeLog.
  8  */
  9 
 10 #include <linux/list.h>
 11 #include <linux/route.h>
 12 #include <linux/spinlock.h>
 13 
 14 #include <net/ipx.h>
 15 #include <net/sock.h>
 16 
 17 LIST_HEAD(ipx_routes);
 18 DEFINE_RWLOCK(ipx_routes_lock);
 19 
 20 extern struct ipx_interface *ipx_internal_net;
 21 
 22 extern __be16 ipx_cksum(struct ipxhdr *packet, int length);
 23 extern struct ipx_interface *ipxitf_find_using_net(__be32 net);
 24 extern int ipxitf_demux_socket(struct ipx_interface *intrfc,
 25                                struct sk_buff *skb, int copy);
 26 extern int ipxitf_demux_socket(struct ipx_interface *intrfc,
 27                                struct sk_buff *skb, int copy);
 28 extern int ipxitf_send(struct ipx_interface *intrfc, struct sk_buff *skb,
 29                        char *node);
 30 extern struct ipx_interface *ipxitf_find_using_net(__be32 net);
 31 
 32 struct ipx_route *ipxrtr_lookup(__be32 net)
 33 {
 34         struct ipx_route *r;
 35 
 36         read_lock_bh(&ipx_routes_lock);
 37         list_for_each_entry(r, &ipx_routes, node)
 38                 if (r->ir_net == net) {
 39                         ipxrtr_hold(r);
 40                         goto unlock;
 41                 }
 42         r = NULL;
 43 unlock:
 44         read_unlock_bh(&ipx_routes_lock);
 45         return r;
 46 }
 47 
 48 /*
 49  * Caller must hold a reference to intrfc
 50  */
 51 int ipxrtr_add_route(__be32 network, struct ipx_interface *intrfc,
 52                      unsigned char *node)
 53 {
 54         struct ipx_route *rt;
 55         int rc;
 56 
 57         /* Get a route structure; either existing or create */
 58         rt = ipxrtr_lookup(network);
 59         if (!rt) {
 60                 rt = kmalloc(sizeof(*rt), GFP_ATOMIC);
 61                 rc = -EAGAIN;
 62                 if (!rt)
 63                         goto out;
 64 
 65                 atomic_set(&rt->refcnt, 1);
 66                 ipxrtr_hold(rt);
 67                 write_lock_bh(&ipx_routes_lock);
 68                 list_add(&rt->node, &ipx_routes);
 69                 write_unlock_bh(&ipx_routes_lock);
 70         } else {
 71                 rc = -EEXIST;
 72                 if (intrfc == ipx_internal_net)
 73                         goto out_put;
 74         }
 75 
 76         rt->ir_net      = network;
 77         rt->ir_intrfc   = intrfc;
 78         if (!node) {
 79                 memset(rt->ir_router_node, '\0', IPX_NODE_LEN);
 80                 rt->ir_routed = 0;
 81         } else {
 82                 memcpy(rt->ir_router_node, node, IPX_NODE_LEN);
 83                 rt->ir_routed = 1;
 84         }
 85 
 86         rc = 0;
 87 out_put:
 88         ipxrtr_put(rt);
 89 out:
 90         return rc;
 91 }
 92 
 93 void ipxrtr_del_routes(struct ipx_interface *intrfc)
 94 {
 95         struct ipx_route *r, *tmp;
 96 
 97         write_lock_bh(&ipx_routes_lock);
 98         list_for_each_entry_safe(r, tmp, &ipx_routes, node)
 99                 if (r->ir_intrfc == intrfc) {
100                         list_del(&r->node);
101                         ipxrtr_put(r);
102                 }
103         write_unlock_bh(&ipx_routes_lock);
104 }
105 
106 static int ipxrtr_create(struct ipx_route_definition *rd)
107 {
108         struct ipx_interface *intrfc;
109         int rc = -ENETUNREACH;
110 
111         /* Find the appropriate interface */
112         intrfc = ipxitf_find_using_net(rd->ipx_router_network);
113         if (!intrfc)
114                 goto out;
115         rc = ipxrtr_add_route(rd->ipx_network, intrfc, rd->ipx_router_node);
116         ipxitf_put(intrfc);
117 out:
118         return rc;
119 }
120 
121 static int ipxrtr_delete(__be32 net)
122 {
123         struct ipx_route *r, *tmp;
124         int rc;
125 
126         write_lock_bh(&ipx_routes_lock);
127         list_for_each_entry_safe(r, tmp, &ipx_routes, node)
128                 if (r->ir_net == net) {
129                         /* Directly connected; can't lose route */
130                         rc = -EPERM;
131                         if (!r->ir_routed)
132                                 goto out;
133                         list_del(&r->node);
134                         ipxrtr_put(r);
135                         rc = 0;
136                         goto out;
137                 }
138         rc = -ENOENT;
139 out:
140         write_unlock_bh(&ipx_routes_lock);
141         return rc;
142 }
143 
144 /*
145  * The skb has to be unshared, we'll end up calling ipxitf_send, that'll
146  * modify the packet
147  */
148 int ipxrtr_route_skb(struct sk_buff *skb)
149 {
150         struct ipxhdr *ipx = ipx_hdr(skb);
151         struct ipx_route *r = ipxrtr_lookup(IPX_SKB_CB(skb)->ipx_dest_net);
152 
153         if (!r) {       /* no known route */
154                 kfree_skb(skb);
155                 return 0;
156         }
157 
158         ipxitf_hold(r->ir_intrfc);
159         ipxitf_send(r->ir_intrfc, skb, r->ir_routed ?
160                         r->ir_router_node : ipx->ipx_dest.node);
161         ipxitf_put(r->ir_intrfc);
162         ipxrtr_put(r);
163 
164         return 0;
165 }
166 
167 /*
168  * Route an outgoing frame from a socket.
169  */
170 int ipxrtr_route_packet(struct sock *sk, struct sockaddr_ipx *usipx,
171                         struct iovec *iov, size_t len, int noblock)
172 {
173         struct sk_buff *skb;
174         struct ipx_sock *ipxs = ipx_sk(sk);
175         struct ipx_interface *intrfc;
176         struct ipxhdr *ipx;
177         size_t size;
178         int ipx_offset;
179         struct ipx_route *rt = NULL;
180         int rc;
181 
182         /* Find the appropriate interface on which to send packet */
183         if (!usipx->sipx_network && ipx_primary_net) {
184                 usipx->sipx_network = ipx_primary_net->if_netnum;
185                 intrfc = ipx_primary_net;
186         } else {
187                 rt = ipxrtr_lookup(usipx->sipx_network);
188                 rc = -ENETUNREACH;
189                 if (!rt)
190                         goto out;
191                 intrfc = rt->ir_intrfc;
192         }
193 
194         ipxitf_hold(intrfc);
195         ipx_offset = intrfc->if_ipx_offset;
196         size = sizeof(struct ipxhdr) + len + ipx_offset;
197 
198         skb = sock_alloc_send_skb(sk, size, noblock, &rc);
199         if (!skb)
200                 goto out_put;
201 
202         skb_reserve(skb, ipx_offset);
203         skb->sk = sk;
204 
205         /* Fill in IPX header */
206         skb_reset_network_header(skb);
207         skb_reset_transport_header(skb);
208         skb_put(skb, sizeof(struct ipxhdr));
209         ipx = ipx_hdr(skb);
210         ipx->ipx_pktsize = htons(len + sizeof(struct ipxhdr));
211         IPX_SKB_CB(skb)->ipx_tctrl = 0;
212         ipx->ipx_type    = usipx->sipx_type;
213 
214         IPX_SKB_CB(skb)->last_hop.index = -1;
215 #ifdef CONFIG_IPX_INTERN
216         IPX_SKB_CB(skb)->ipx_source_net = ipxs->intrfc->if_netnum;
217         memcpy(ipx->ipx_source.node, ipxs->node, IPX_NODE_LEN);
218 #else
219         rc = ntohs(ipxs->port);
220         if (rc == 0x453 || rc == 0x452) {
221                 /* RIP/SAP special handling for mars_nwe */
222                 IPX_SKB_CB(skb)->ipx_source_net = intrfc->if_netnum;
223                 memcpy(ipx->ipx_source.node, intrfc->if_node, IPX_NODE_LEN);
224         } else {
225                 IPX_SKB_CB(skb)->ipx_source_net = ipxs->intrfc->if_netnum;
226                 memcpy(ipx->ipx_source.node, ipxs->intrfc->if_node,
227                         IPX_NODE_LEN);
228         }
229 #endif  /* CONFIG_IPX_INTERN */
230         ipx->ipx_source.sock            = ipxs->port;
231         IPX_SKB_CB(skb)->ipx_dest_net   = usipx->sipx_network;
232         memcpy(ipx->ipx_dest.node, usipx->sipx_node, IPX_NODE_LEN);
233         ipx->ipx_dest.sock              = usipx->sipx_port;
234 
235         rc = memcpy_fromiovec(skb_put(skb, len), iov, len);
236         if (rc) {
237                 kfree_skb(skb);
238                 goto out_put;
239         }
240 
241         /* Apply checksum. Not allowed on 802.3 links. */
242         if (sk->sk_no_check || intrfc->if_dlink_type == htons(IPX_FRAME_8023))
243                 ipx->ipx_checksum = htons(0xFFFF);
244         else
245                 ipx->ipx_checksum = ipx_cksum(ipx, len + sizeof(struct ipxhdr));
246 
247         rc = ipxitf_send(intrfc, skb, (rt && rt->ir_routed) ?
248                          rt->ir_router_node : ipx->ipx_dest.node);
249 out_put:
250         ipxitf_put(intrfc);
251         if (rt)
252                 ipxrtr_put(rt);
253 out:
254         return rc;
255 }
256 
257 /*
258  * We use a normal struct rtentry for route handling
259  */
260 int ipxrtr_ioctl(unsigned int cmd, void __user *arg)
261 {
262         struct rtentry rt;      /* Use these to behave like 'other' stacks */
263         struct sockaddr_ipx *sg, *st;
264         int rc = -EFAULT;
265 
266         if (copy_from_user(&rt, arg, sizeof(rt)))
267                 goto out;
268 
269         sg = (struct sockaddr_ipx *)&rt.rt_gateway;
270         st = (struct sockaddr_ipx *)&rt.rt_dst;
271 
272         rc = -EINVAL;
273         if (!(rt.rt_flags & RTF_GATEWAY) || /* Direct routes are fixed */
274             sg->sipx_family != AF_IPX ||
275             st->sipx_family != AF_IPX)
276                 goto out;
277 
278         switch (cmd) {
279         case SIOCDELRT:
280                 rc = ipxrtr_delete(st->sipx_network);
281                 break;
282         case SIOCADDRT: {
283                 struct ipx_route_definition f;
284                 f.ipx_network           = st->sipx_network;
285                 f.ipx_router_network    = sg->sipx_network;
286                 memcpy(f.ipx_router_node, sg->sipx_node, IPX_NODE_LEN);
287                 rc = ipxrtr_create(&f);
288                 break;
289         }
290         }
291 
292 out:
293         return rc;
294 }
295 
  This page was automatically generated by the LXR engine.