patch-2.3.4 linux/net/decnet/dn_route.c
Next file: linux/net/decnet/dn_timer.c
Previous file: linux/net/decnet/dn_raw.c
Back to the patch index
Back to the overall index
- Lines: 1029
- Date:
Wed May 26 09:36:36 1999
- Orig file:
v2.3.3/linux/net/decnet/dn_route.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.3.3/linux/net/decnet/dn_route.c linux/net/decnet/dn_route.c
@@ -0,0 +1,1028 @@
+
+/*
+ * DECnet An implementation of the DECnet protocol suite for the LINUX
+ * operating system. DECnet is implemented using the BSD Socket
+ * interface as the means of communication with the user level.
+ *
+ * DECnet Routing Functions (Endnode and Router)
+ *
+ * Authors: Steve Whitehouse <SteveW@ACM.org>
+ * Eduardo Marcelo Serrat <emserrat@geocities.com>
+ *
+ * Changes:
+ * Steve Whitehouse : Fixes to allow "intra-ethernet" and
+ * "return-to-sender" bits on outgoing
+ * packets.
+ * Steve Whitehouse : Timeouts for cached routes.
+ * Steve Whitehouse : Use dst cache for input routes too.
+ * Steve Whitehouse : Fixed error values in dn_send_skb.
+ */
+
+/******************************************************************************
+ (c) 1995-1998 E.M. Serrat emserrat@geocities.com
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+*******************************************************************************/
+
+#include <linux/config.h>
+#include <linux/errno.h>
+#include <linux/types.h>
+#include <linux/socket.h>
+#include <linux/in.h>
+#include <linux/kernel.h>
+#include <linux/timer.h>
+#include <linux/sockios.h>
+#include <linux/net.h>
+#include <linux/netdevice.h>
+#include <linux/inet.h>
+#include <linux/route.h>
+#include <net/sock.h>
+#include <linux/fcntl.h>
+#include <linux/mm.h>
+#include <linux/proc_fs.h>
+#include <linux/init.h>
+#include <linux/firewall.h>
+#include <linux/rtnetlink.h>
+#include <linux/string.h>
+#include <asm/errno.h>
+#include <net/neighbour.h>
+#include <net/dst.h>
+#include <net/dn.h>
+#include <net/dn_dev.h>
+#include <net/dn_nsp.h>
+#include <net/dn_route.h>
+#include <net/dn_neigh.h>
+#include <net/dn_fib.h>
+#include <net/dn_raw.h>
+
+extern struct neigh_table dn_neigh_table;
+
+#define DN_HASHBUCKETS 16
+
+static unsigned char dn_hiord_addr[6] = {0xAA,0x00,0x04,0x00,0x00,0x00};
+
+static int dn_dst_gc(void);
+static struct dst_entry *dn_dst_check(struct dst_entry *, __u32);
+static struct dst_entry *dn_dst_reroute(struct dst_entry *, struct sk_buff *skb);
+static struct dst_entry *dn_dst_negative_advice(struct dst_entry *);
+static void dn_dst_link_failure(struct sk_buff *);
+static int dn_route_input(struct sk_buff *);
+
+static struct dn_route *dn_route_cache[DN_HASHBUCKETS];
+static struct timer_list dn_route_timer = { NULL, NULL, 0, 0L, NULL };
+int decnet_dst_gc_interval = 2;
+
+static struct dst_ops dn_dst_ops = {
+ PF_DECnet,
+ __constant_htons(ETH_P_DNA_RT),
+ 128,
+ dn_dst_gc,
+ dn_dst_check,
+ dn_dst_reroute,
+ NULL,
+ dn_dst_negative_advice,
+ dn_dst_link_failure,
+ ATOMIC_INIT(0)
+};
+
+static __inline__ unsigned dn_hash(unsigned short dest)
+{
+ unsigned short tmp = (dest&0xff) ^ (dest>>8);
+ return (tmp&0x0f) ^ (tmp>>4);
+}
+
+static void dn_dst_check_expire(unsigned long dummy)
+{
+ int i;
+ struct dn_route *rt, **rtp;
+ unsigned long now = jiffies;
+ unsigned long expire = 120 * HZ;
+
+ for(i = 0; i < DN_HASHBUCKETS; i++) {
+ rtp = &dn_route_cache[i];
+ for(;(rt=*rtp); rtp = &rt->u.rt_next) {
+ if (atomic_read(&rt->u.dst.use) ||
+ (now - rt->u.dst.lastuse) < expire)
+ continue;
+ *rtp = rt->u.rt_next;
+ rt->u.rt_next = NULL;
+ dst_free(&rt->u.dst);
+ }
+
+ if ((jiffies - now) > 0)
+ break;
+ }
+
+ dn_route_timer.expires = now + decnet_dst_gc_interval * HZ;
+ add_timer(&dn_route_timer);
+}
+
+static int dn_dst_gc(void)
+{
+ struct dn_route *rt, **rtp;
+ int i;
+ unsigned long now = jiffies;
+ unsigned long expire = 10 * HZ;
+
+ start_bh_atomic();
+ for(i = 0; i < DN_HASHBUCKETS; i++) {
+ rtp = &dn_route_cache[i];
+ for(; (rt=*rtp); rtp = &rt->u.rt_next) {
+ if (atomic_read(&rt->u.dst.use) ||
+ (now - rt->u.dst.lastuse) < expire)
+ continue;
+ *rtp = rt->u.rt_next;
+ rt->u.rt_next = NULL;
+ dst_free(&rt->u.dst);
+ break;
+ }
+ }
+ end_bh_atomic();
+
+ return 0;
+}
+
+static struct dst_entry *dn_dst_check(struct dst_entry *dst, __u32 cookie)
+{
+ dst_release(dst);
+ return NULL;
+}
+
+static struct dst_entry *dn_dst_reroute(struct dst_entry *dst,
+ struct sk_buff *skb)
+{
+ return NULL;
+}
+
+/*
+ * This is called through sendmsg() when you specify MSG_TRYHARD
+ * and there is already a route in cache.
+ */
+static struct dst_entry *dn_dst_negative_advice(struct dst_entry *dst)
+{
+ dst_release(dst);
+ return NULL;
+}
+
+static void dn_dst_link_failure(struct sk_buff *skb)
+{
+ return;
+}
+
+static void dn_insert_route(struct dn_route *rt)
+{
+ unsigned hash = dn_hash(rt->rt_daddr);
+ unsigned long now = jiffies;
+
+ start_bh_atomic();
+
+ rt->u.rt_next = dn_route_cache[hash];
+ dn_route_cache[hash] = rt;
+
+ atomic_inc(&rt->u.dst.refcnt);
+ atomic_inc(&rt->u.dst.use);
+ rt->u.dst.lastuse = now;
+
+ end_bh_atomic();
+}
+
+#if defined(CONFIG_DECNET_MODULE)
+static void dn_run_flush(unsigned long dummy)
+{
+ int i;
+ struct dn_route *rt, *next;
+
+ for(i = 0; i < DN_HASHBUCKETS; i++) {
+ if ((rt = xchg(&dn_route_cache[i], NULL)) == NULL)
+ continue;
+
+ for(; rt; rt=next) {
+ next = rt->u.rt_next;
+ rt->u.rt_next = NULL;
+ dst_free((struct dst_entry *)rt);
+ }
+ }
+}
+#endif /* CONFIG_DECNET_MODULE */
+
+static int dn_route_rx_long(struct sk_buff *skb)
+{
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ unsigned char *ptr = skb->data;
+
+ if (skb->len < 21) /* 20 for long header, 1 for shortest nsp */
+ goto drop_it;
+
+ skb_pull(skb, 20);
+ skb->h.raw = skb->data;
+
+ /* Destination info */
+ ptr += 2;
+ cb->dst = dn_htons(dn_eth2dn(ptr));
+ if (memcmp(ptr, dn_hiord_addr, 4) != 0)
+ goto drop_it;
+ ptr += 6;
+
+
+ /* Source info */
+ ptr += 2;
+ cb->src = dn_htons(dn_eth2dn(ptr));
+ if (memcmp(ptr, dn_hiord_addr, 4) != 0)
+ goto drop_it;
+ ptr += 6;
+ /* Other junk */
+ ptr++;
+ cb->hops = *ptr++; /* Visit Count */
+
+ if (dn_route_input(skb) == 0) {
+
+#ifdef CONFIG_DECNET_FW
+ struct neighbour *neigh = skb->dst->neighbour;
+
+ switch(call_in_firewall(PF_DECnet, skb->dev, NULL, NULL, &skb)) {
+ case FW_REJECT:
+ neigh->ops->error_report(neigh, skb);
+ return 0;
+ case FW_BLOCK:
+ default:
+ goto drop_it;
+ case FW_ACCEPT:
+ }
+#endif /* CONFIG_DECNET_FW */
+
+ return skb->dst->input(skb);
+ }
+
+drop_it:
+ kfree_skb(skb);
+ return 0;
+}
+
+
+
+static int dn_route_rx_short(struct sk_buff *skb)
+{
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ unsigned char *ptr = skb->data;
+
+ if (skb->len < 6) /* 5 for short header + 1 for shortest nsp */
+ goto drop_it;
+
+ skb_pull(skb, 5);
+ skb->h.raw = skb->data;
+
+ cb->dst = *(dn_address *)ptr;
+ ptr += 2;
+ cb->src = *(dn_address *)ptr;
+ ptr += 2;
+ cb->hops = *ptr & 0x3f;
+
+ if (dn_route_input(skb) == 0) {
+
+#ifdef CONFIG_DECNET_FW
+ struct neighbour *neigh = skb->dst->neighbour;
+
+ switch(call_in_firewall(PF_DECnet, skb->dev, NULL, NULL, &skb)) {
+ case FW_REJECT:
+ neigh->ops->error_report(neigh, skb);
+ return 0;
+ case FW_BLOCK:
+ default:
+ goto drop_it;
+
+ case FW_ACCEPT:
+ }
+#endif /* CONFIG_DECNET_FW */
+
+ return skb->dst->input(skb);
+ }
+
+drop_it:
+ kfree_skb(skb);
+ return 0;
+}
+
+int dn_route_rcv(struct sk_buff *skb, struct device *dev, struct packet_type *pt)
+{
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ unsigned char flags = 0;
+ int padlen = 0;
+ __u16 len = dn_ntohs(*(__u16 *)skb->data);
+ struct dn_dev *dn = (struct dn_dev *)dev->dn_ptr;
+
+ if (dn == NULL)
+ goto dump_it;
+
+ cb->stamp = jiffies;
+ cb->iif = dev->ifindex;
+
+ skb_pull(skb, 2);
+
+ if (len > skb->len)
+ goto dump_it;
+
+ skb_trim(skb, len);
+
+ flags = *skb->data;
+
+ /*
+ * If we have padding, remove it.
+ */
+ if (flags & DN_RT_F_PF) {
+ padlen = flags & ~DN_RT_F_PF;
+ skb_pull(skb, padlen);
+ flags = *skb->data;
+ }
+
+ skb->nh.raw = skb->data;
+
+ /*
+ * Weed out future version DECnet
+ */
+ if (flags & DN_RT_F_VER)
+ goto dump_it;
+
+ cb->rt_flags = flags;
+
+ if (dn->parms.setsrc)
+ dn->parms.setsrc(skb);
+
+ /* printk(KERN_DEBUG "dn_route_rcv: got 0x%02x from %s [%d %d %d]\n", (int)flags,
+ (dev) ? dev->name : "???", len, skb->len, padlen); */
+
+#ifdef CONFIG_DECNET_RAW
+ dn_raw_rx_routing(skb);
+#endif /* CONFIG_DECNET_RAW */
+
+ if (flags & DN_RT_PKT_CNTL) {
+ switch(flags & DN_RT_CNTL_MSK) {
+ case DN_RT_PKT_INIT:
+ dn_dev_init_pkt(skb);
+ break;
+ case DN_RT_PKT_VERI:
+ dn_dev_veri_pkt(skb);
+ break;
+ }
+
+ if (dn->parms.state != DN_DEV_S_RU)
+ goto dump_it;
+
+ switch(flags & DN_RT_CNTL_MSK) {
+ case DN_RT_PKT_HELO:
+ dn_dev_hello(skb);
+ dn_neigh_pointopoint_hello(skb);
+ return 0;
+
+ case DN_RT_PKT_L1RT:
+ case DN_RT_PKT_L2RT:
+#ifdef CONFIG_DECNET_ROUTER
+ return dn_fib_rt_message(skb);
+#else
+ break;
+#endif /* CONFIG_DECNET_ROUTER */
+ case DN_RT_PKT_ERTH:
+ dn_neigh_router_hello(skb);
+ return 0;
+
+ case DN_RT_PKT_EEDH:
+ dn_neigh_endnode_hello(skb);
+ return 0;
+ }
+ } else {
+ if (dn->parms.state != DN_DEV_S_RU)
+ goto dump_it;
+
+ skb_pull(skb, 1); /* Pull flags */
+
+ switch(flags & DN_RT_PKT_MSK) {
+ case DN_RT_PKT_LONG:
+ return dn_route_rx_long(skb);
+ case DN_RT_PKT_SHORT:
+ return dn_route_rx_short(skb);
+ }
+ }
+
+dump_it:
+ if (net_ratelimit())
+ printk(KERN_DEBUG "dn_route_rcv: Dumping packet\n");
+ kfree_skb(skb);
+ return 0;
+}
+
+
+void dn_send_skb(struct sk_buff *skb)
+{
+ struct sock *sk = skb->sk;
+ struct dn_scp *scp = &sk->protinfo.dn;
+
+ if (sk == NULL) {
+ dev_queue_xmit(skb);
+ return ;
+ }
+
+ skb->h.raw = skb->data;
+
+ scp->stamp = jiffies; /* Record time packet was sent */
+
+ /* printk(KERN_DEBUG "dn_send_skb\n"); */
+
+ if (sk->dst_cache && sk->dst_cache->obsolete) {
+ dst_release(sk->dst_cache);
+ sk->dst_cache = NULL;
+ }
+
+ if (sk->dst_cache == NULL) {
+ if (dn_route_output(sk) != 0) {
+ kfree_skb(skb);
+ sk->err = EHOSTUNREACH;
+ if (!sk->dead)
+ sk->state_change(sk);
+ return;
+ }
+ }
+
+ skb->dst = dst_clone(sk->dst_cache);
+
+ sk->dst_cache->output(skb);
+}
+
+
+static int dn_output(struct sk_buff *skb)
+{
+ struct dst_entry *dst = skb->dst;
+ struct dn_route *rt = (struct dn_route *)dst;
+ struct device *dev = dst->dev;
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ int err = -EINVAL;
+
+ if (!dst->neighbour)
+ goto error;
+
+ skb->dev = dev;
+
+ cb->src = rt->rt_saddr;
+ cb->dst = rt->rt_daddr;
+
+ /*
+ * Always set the Intra-Ethernet bit on all outgoing packets
+ * originated on this node. Only valid flag from upper layers
+ * is return-to-sender-requested. Set hop count to 0 too.
+ */
+ cb->rt_flags &= ~DN_RT_F_RQR;
+ cb->rt_flags |= DN_RT_F_IE;
+ cb->hops = 0;
+
+ /*
+ * Filter through the outgoing firewall
+ */
+#ifdef CONFIG_DECNET_FW
+ switch(call_out_firewall(PF_DECnet, dst->dev, NULL, NULL, &skb)) {
+ case FW_REJECT:
+ err = -EPERM;
+ goto drop;
+ case FW_BLOCK:
+ default:
+ err = 0;
+ goto drop;
+ case FW_ACCEPT:
+ }
+#endif /* CONFIG_DECNET_FW */
+
+ return dst->neighbour->output(skb);
+
+error:
+ if (net_ratelimit())
+ printk(KERN_DEBUG "dn_output: This should not happen\n");
+
+#ifdef CONFIG_DECNET_FW
+drop:
+#endif
+ kfree_skb(skb);
+
+ return err;
+}
+
+#ifdef CONFIG_DECNET_ROUTER
+static int dn_l2_forward(struct sk_buff *skb)
+{
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ struct dst_entry *dst = skb->dst;
+ int err = -EINVAL;
+
+ if (!dst->neighbour)
+ goto error;
+
+ /*
+ * Hop count exceeded.
+ */
+ err = 0;
+ if (++cb->hops > 30)
+ goto drop;
+
+ /*
+ * Forwarding firewall
+ */
+#ifdef CONFIG_DECNET_FW
+ switch(call_fw_firewall(PF_DECnet, dst->dev, NULL, NULL, &skb)) {
+ case FW_REJECT:
+ dst->neighbour->ops->error_report(dst->neighbour, skb);
+ return -EPERM;
+ case FW_BLOCK:
+ default:
+ goto drop;
+ case FW_ACCEPT:
+ }
+#endif /* CONFIG_DECNET_FW */
+
+ skb->dev = dst->dev;
+
+ /*
+ * If packet goes out same interface it came in on, then set
+ * the Intra-Ethernet bit. This has no effect for short
+ * packets, so we don't need to test for them here.
+ */
+ if (cb->iif == dst->dev->ifindex)
+ cb->rt_flags |= DN_RT_F_IE;
+ else
+ cb->rt_flags &= ~DN_RT_F_IE;
+
+#ifdef CONFIG_DECNET_FW
+ switch(call_out_firewall(PF_DECnet, dst->dev, NULL, NULL, &skb)) {
+ case FW_REJECT:
+ dst->neighbour->ops->error_report(dst->neighbour, skb);
+ return -EPERM;
+ case FW_BLOCK:
+ default:
+ goto drop;
+ case FW_ACCEPT:
+ }
+#endif /* CONFIG_DECNET_FW */
+
+ return dst->neighbour->output(skb);
+
+
+error:
+ if (net_ratelimit())
+ printk(KERN_DEBUG "dn_forward: This should not happen\n");
+drop:
+ kfree_skb(skb);
+
+ return err;
+}
+
+/*
+ * Simple frontend to the l2 routing function which filters
+ * traffic not in our area when we should only do l1
+ * routing.
+ */
+static int dn_l1_forward(struct sk_buff *skb)
+{
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+
+ if ((dn_ntohs(cb->dst ^ decnet_address) & 0xfc00) == 0)
+ return dn_l2_forward(skb);
+
+ kfree_skb(skb);
+ return 0;
+}
+#endif
+
+/*
+ * Drop packet. This is used for endnodes and for
+ * when we should not be forwarding packets from
+ * this dest.
+ */
+static int dn_blackhole(struct sk_buff *skb)
+{
+ kfree_skb(skb);
+ return 0;
+}
+
+/*
+ * Used to catch bugs. This should never normally get
+ * called.
+ */
+static int dn_rt_bug(struct sk_buff *skb)
+{
+ if (net_ratelimit()) {
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+
+ printk(KERN_DEBUG "dn_rt_bug: skb from:%04x to:%04x\n",
+ cb->src, cb->dst);
+ }
+
+ kfree_skb(skb);
+
+ return -EINVAL;
+}
+
+static int dn_route_output_slow(struct sock *sk)
+{
+ struct dn_scp *scp = &sk->protinfo.dn;
+ dn_address dest = dn_saddr2dn(&scp->peer);
+ struct dn_route *rt = NULL;
+ struct device *dev = decnet_default_device;
+ struct neighbour *neigh = NULL;
+ struct dn_dev *dn_db;
+ unsigned char addr[6];
+
+#ifdef CONFIG_DECNET_ROUTER
+ if ((decnet_node_type == DN_RT_INFO_L1RT) || (decnet_node_type == DN_RT_INFO_L2RT)) {
+#if 0
+ struct dn_fib_ent *fe = dn_fib_lookup(dest, decnet_address);
+
+ if (fe != NULL) {
+ neigh = neigh_clone(fe->neigh);
+ dn_fib_release(fe);
+ goto got_route;
+ }
+#endif
+ }
+#endif
+
+ dn_dn2eth(addr, dest);
+
+ /* Look in On-Ethernet cache first */
+ if ((neigh = dn_neigh_lookup(&dn_neigh_table, &addr)) != NULL)
+ goto got_route;
+
+ if (dev == NULL)
+ return -EINVAL;
+
+ /* FIXME: We need to change this for routing nodes */
+ /* Send to default router if that doesn't work */
+ if ((neigh = neigh_lookup(&dn_neigh_table, &addr, dev)) != NULL)
+ goto got_route;
+
+ /* Send to default device (and hope for the best) if above fail */
+ if ((neigh = __neigh_lookup(&dn_neigh_table, &addr, dev, 1)) != NULL)
+ goto got_route;
+
+
+ return -EINVAL;
+
+got_route:
+
+ if ((rt = dst_alloc(sizeof(struct dn_route), &dn_dst_ops)) == NULL) {
+ neigh_release(neigh);
+ return -EINVAL;
+ }
+
+ dn_db = (struct dn_dev *)neigh->dev->dn_ptr;
+
+ rt->rt_saddr = decnet_address;
+ rt->rt_daddr = dest;
+ rt->rt_oif = neigh->dev->ifindex;
+ rt->rt_iif = 0;
+
+ rt->u.dst.neighbour = neigh;
+ rt->u.dst.dev = neigh->dev;
+ rt->u.dst.lastuse = jiffies;
+ rt->u.dst.output = dn_output;
+ rt->u.dst.input = dn_rt_bug;
+
+ if (dn_dev_islocal(neigh->dev, rt->rt_daddr))
+ rt->u.dst.input = dn_nsp_rx;
+
+ dn_insert_route(rt);
+ sk->dst_cache = &rt->u.dst;
+
+ return 0;
+}
+
+int dn_route_output(struct sock *sk)
+{
+ struct dn_scp *scp = &sk->protinfo.dn;
+ dn_address dest = dn_saddr2dn(&scp->peer);
+ unsigned hash = dn_hash(dest);
+ struct dn_route *rt = NULL;
+ unsigned short src = dn_saddr2dn(&scp->addr);
+
+ start_bh_atomic();
+ for(rt = dn_route_cache[hash]; rt; rt = rt->u.rt_next) {
+ if ((dest == rt->rt_daddr) &&
+ (src == rt->rt_saddr) &&
+ (rt->rt_iif == 0) &&
+ (rt->rt_oif != 0)) {
+ rt->u.dst.lastuse = jiffies;
+ atomic_inc(&rt->u.dst.use);
+ atomic_inc(&rt->u.dst.refcnt);
+ end_bh_atomic();
+ sk->dst_cache = &rt->u.dst;
+ return 0;
+ }
+ }
+ end_bh_atomic();
+
+ return dn_route_output_slow(sk);
+}
+
+static int dn_route_input_slow(struct sk_buff *skb)
+{
+ struct dn_route *rt = NULL;
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ struct device *dev = skb->dev;
+ struct neighbour *neigh = NULL;
+ unsigned char addr[6];
+
+ /*
+ * In this case we've just received a packet from a source
+ * outside ourselves pretending to come from us. We don't
+ * allow it any further to prevent routing loops, spoofing and
+ * other nasties. Loopback packets already have the dst attached
+ * so this only affects packets which have originated elsewhere.
+ */
+ if (dn_dev_islocal(dev, cb->src))
+ return 1;
+
+#ifdef CONFIG_DECNET_ROUTER
+ if ((decnet_node_type == DN_RT_INFO_L1RT) || (decnet_node_type == DN_RT_INFO_L2RT)) {
+#if 0
+ struct dn_fib_ent *fe = NULL;
+
+ fe = dn_fib_lookup(cb->src, cb->dst);
+
+ /* Try routing table first */
+ if (fe != NULL) {
+ neigh = neigh_clone(fe->neigh);
+ dn_fib_release(fe);
+ goto got_route;
+ }
+#endif
+ }
+#endif
+
+ dn_dn2eth(addr, cb->src);
+
+ /* Now see if we are directly connected */
+ if ((neigh = dn_neigh_lookup(&dn_neigh_table, &addr)) != NULL)
+ goto got_route;
+
+ if (dev == NULL)
+ return -EINVAL;
+
+ /* FIXME: Try the default router here .... */
+
+ if ((neigh = __neigh_lookup(&dn_neigh_table, &addr, dev, 1)) != NULL)
+ goto got_route;
+
+ return -EINVAL;
+
+got_route:
+
+ if ((rt = dst_alloc(sizeof(struct dn_route), &dn_dst_ops)) == NULL) {
+ neigh_release(neigh);
+ return -EINVAL;
+ }
+
+ rt->rt_saddr = cb->dst;
+ rt->rt_daddr = cb->src;
+ rt->rt_oif = 0;
+ rt->rt_iif = neigh->dev->ifindex;
+
+ rt->u.dst.neighbour = neigh;
+ rt->u.dst.dev = neigh->dev;
+ rt->u.dst.lastuse = jiffies;
+ rt->u.dst.output = dn_output;
+
+ switch(decnet_node_type) {
+ case DN_RT_INFO_ENDN:
+ rt->u.dst.input = dn_blackhole;
+ break;
+#ifdef CONFIG_DECNET_ROUTER
+ case DN_RT_INFO_L1RT:
+ rt->u.dst.input = dn_l1_forward;
+ break;
+ case DN_RT_INFO_L2RT:
+ rt->u.dst.input = dn_l2_forward;
+ break;
+#endif /* CONFIG_DECNET_ROUTER */
+ default:
+ rt->u.dst.input = dn_blackhole;
+ if (net_ratelimit())
+ printk(KERN_DEBUG "dn_route_input_slow: What kind of node are we?\n");
+ }
+
+ if (dn_dev_islocal(dev, cb->dst))
+ rt->u.dst.input = dn_nsp_rx;
+
+ dn_insert_route(rt);
+ skb->dst = (struct dst_entry *)rt;
+
+ return 0;
+}
+
+int dn_route_input(struct sk_buff *skb)
+{
+ struct dn_route *rt;
+ struct dn_skb_cb *cb = (struct dn_skb_cb *)skb->cb;
+ unsigned hash = dn_hash(cb->src);
+
+ if (skb->dst)
+ return 0;
+
+ for(rt = dn_route_cache[hash]; rt != NULL; rt = rt->u.rt_next) {
+ if ((rt->rt_saddr == cb->dst) &&
+ (rt->rt_daddr == cb->src) &&
+ (rt->rt_oif == 0) &&
+ (rt->rt_iif == cb->iif)) {
+ rt->u.dst.lastuse = jiffies;
+ atomic_inc(&rt->u.dst.use);
+ atomic_inc(&rt->u.dst.refcnt);
+ skb->dst = (struct dst_entry *)rt;
+ return 0;
+ }
+ }
+
+ return dn_route_input_slow(skb);
+}
+
+#ifdef CONFIG_DECNET_ROUTER
+#ifdef CONFIG_RTNETLINK
+static int dn_rt_fill_info(struct sk_buff *skb, u32 pid, u32 seq, int event, int nowait)
+{
+ struct dn_route *rt = (struct dn_route *)skb->dst;
+ struct rtmsg *r;
+ struct nlmsghdr *nlh;
+ unsigned char *b = skb->tail;
+
+ nlh = NLMSG_PUT(skb, pid, seq, event, sizeof(*r));
+ r = NLMSG_DATA(nlh);
+ nlh->nlmsg_flags = nowait ? NLM_F_MULTI : 0;
+ r->rtm_family = AF_DECnet;
+ r->rtm_dst_len = 16;
+ r->rtm_src_len = 16;
+ r->rtm_tos = 0;
+ r->rtm_table = 0;
+ r->rtm_type = 0;
+ r->rtm_scope = RT_SCOPE_UNIVERSE;
+ r->rtm_protocol = RTPROT_UNSPEC;
+ RTA_PUT(skb, RTA_DST, 2, &rt->rt_daddr);
+ RTA_PUT(skb, RTA_SRC, 2, &rt->rt_saddr);
+ if (rt->u.dst.dev)
+ RTA_PUT(skb, RTA_OIF, sizeof(int), &rt->u.dst.dev->ifindex);
+ if (rt->u.dst.window)
+ RTA_PUT(skb, RTAX_WINDOW, sizeof(unsigned), &rt->u.dst.window);
+ if (rt->u.dst.rtt)
+ RTA_PUT(skb, RTAX_RTT, sizeof(unsigned), &rt->u.dst.rtt);
+
+ nlh->nlmsg_len = skb->tail - b;
+ return skb->len;
+
+nlmsg_failure:
+rtattr_failure:
+ skb_trim(skb, b - skb->data);
+ return -1;
+}
+
+int dn_fib_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr *nlh, void *arg)
+{
+ struct rtattr **rta = arg;
+ /* struct rtmsg *rtm = NLMSG_DATA(nlh); */
+ struct dn_route *rt = NULL;
+ dn_address dst = 0;
+ dn_address src = 0;
+ int iif = 0;
+ int err;
+ struct sk_buff *skb;
+
+ skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL);
+ if (skb == NULL)
+ return -ENOBUFS;
+ skb->mac.raw = skb->data;
+
+ if (rta[RTA_SRC-1])
+ memcpy(&src, RTA_DATA(rta[RTA_SRC-1]), 2);
+ if (rta[RTA_DST-1])
+ memcpy(&dst, RTA_DATA(rta[RTA_DST-1]), 2);
+ if (rta[RTA_IIF-1])
+ memcpy(&iif, RTA_DATA(rta[RTA_IIF-1]), sizeof(int));
+
+ if (iif) {
+ struct device *dev;
+ if ((dev = dev_get_by_index(iif)) == NULL)
+ return -ENODEV;
+ if (!dev->dn_ptr)
+ return -ENODEV;
+ skb->protocol = __constant_htons(ETH_P_DNA_RT);
+ skb->dev = dev;
+ start_bh_atomic();
+ err = dn_route_input(skb);
+ end_bh_atomic();
+ rt = (struct dn_route *)skb->dst;
+ if (!err && rt->u.dst.error)
+ err = rt->u.dst.error;
+ } else {
+ int oif = 0;
+ if (rta[RTA_OIF-1])
+ memcpy(&oif, RTA_DATA(rta[RTA_OIF-1]), sizeof(int));
+ err = -EOPNOTSUPP;
+ }
+ if (err) {
+ kfree_skb(skb);
+ return err;
+ }
+ skb->dst = &rt->u.dst;
+
+ NETLINK_CB(skb).dst_pid = NETLINK_CB(in_skb).pid;
+
+ err = dn_rt_fill_info(skb, NETLINK_CB(in_skb).pid, nlh->nlmsg_seq, RTM_NEWROUTE, 0);
+
+ if (err == 0)
+ return 0;
+ if (err < 0)
+ return -EMSGSIZE;
+
+ err = netlink_unicast(rtnl, skb, NETLINK_CB(in_skb).pid, MSG_DONTWAIT);
+
+ return err;
+}
+#endif /* CONFIG_RTNETLINK */
+#endif /* CONFIG_DECNET_ROUTER */
+
+#ifdef CONFIG_PROC_FS
+
+static int decnet_cache_get_info(char *buffer, char **start, off_t offset, int length, int dummy)
+{
+ int len = 0;
+ off_t pos = 0;
+ off_t begin = 0;
+ struct dn_route *rt;
+ int i;
+ char buf1[DN_ASCBUF_LEN], buf2[DN_ASCBUF_LEN];
+
+ start_bh_atomic();
+ for(i = 0; i < DN_HASHBUCKETS; i++) {
+ rt = dn_route_cache[i];
+ for(; rt != NULL; rt = rt->u.rt_next) {
+ len += sprintf(buffer + len, "%-8s %-7s %-7s %04d %04d %04d\n",
+ rt->u.dst.dev ? rt->u.dst.dev->name : "*",
+ dn_addr2asc(dn_ntohs(rt->rt_daddr), buf1),
+ dn_addr2asc(dn_ntohs(rt->rt_saddr), buf2),
+ atomic_read(&rt->u.dst.use),
+ atomic_read(&rt->u.dst.refcnt),
+ (int)rt->u.dst.rtt
+ );
+
+ pos = begin + len;
+
+ if (pos < offset) {
+ len = 0;
+ begin = pos;
+ }
+ if (pos > offset + length)
+ break;
+ }
+ if (pos > offset + length)
+ break;
+ }
+ end_bh_atomic();
+
+ *start = buffer + (offset - begin);
+ len -= (offset - begin);
+
+ if (len > length) len = length;
+
+ return(len);
+}
+
+static struct proc_dir_entry proc_net_decnet_cache = {
+ PROC_NET_DN_CACHE, 12, "decnet_cache",
+ S_IFREG | S_IRUGO, 1, 0, 0,
+ 0, &proc_net_inode_operations,
+ decnet_cache_get_info
+};
+
+#endif /* CONFIG_PROC_FS */
+
+void __init dn_route_init(void)
+{
+ memset(dn_route_cache, 0, sizeof(struct dn_route *) * DN_HASHBUCKETS);
+
+ dn_route_timer.function = dn_dst_check_expire;
+ dn_route_timer.expires = jiffies + decnet_dst_gc_interval * HZ;
+ add_timer(&dn_route_timer);
+
+#ifdef CONFIG_PROC_FS
+ proc_net_register(&proc_net_decnet_cache);
+#endif /* CONFIG_PROC_FS */
+}
+
+#ifdef CONFIG_DECNET_MODULE
+void dn_route_cleanup(void)
+{
+ del_timer(&dn_route_timer);
+ dn_run_flush(0);
+#ifdef CONFIG_PROC_FS
+ proc_net_unregister(PROC_NET_DN_CACHE);
+#endif /* CONFIG_PROC_FS */
+}
+#endif /* CONFIG_DECNET_MODULE */
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)