📄 output.c
字号:
/* output.c - routines for generating outgoing RIP messages *//* Copyright 1984 - 1998 Wind River Systems, Inc. */#include "copyright_wrs.h"/* * Copyright (c) 1983, 1988, 1993 * The Regents of the University of California. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the University of * California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * @(#)output.c 8.2 (Berkeley) 4/28/95"; *//*modification history--------------------01i,11sep98,spm added option to disable gateway filtering and removed all references to bloated trace commands (SPR #22350)01h,01sep98,spm extended gateway filtering tests to handle classless netmasks and host-specific routes01g,26jul98,spm removed duplicate condition from test in supply routine; corrected stray pointer in ripBuildPacket call and added parameter needed for RIPv2 updates; removed compiler warnings01f,06oct97,gnn added sendHook functionality to sending01e,08may97,gnn fixed an authentication bug.01d,07apr97,gnn cleared up some of the more egregious warnings. added MIB-II interface and option support.01c,13mar97,gnn fixed a minor bug in the output routine01b,24feb97,gnn added rip version 2 functionality01a,26nov96,gnn created from BSD4.4 routed*//*DESCRIPTION*//* * Routing Table Management Daemon */#include "vxWorks.h"#include "logLib.h"#include "rip/defs.h"#include "rip/m2RipLib.h"IMPORT int routedDebug;IMPORT BOOL ripFilterFlag;/* * Apply the function "f" to all non-passive * interfaces. If the interface supports the * use of broadcasting use it, otherwise address * the output to the known router. */void toall(f, rtstate, skipif) int (*f)(); int rtstate; struct interface *skipif;{ register struct interface *ifp; register struct sockaddr *dst; register int flags; extern struct interface *ripIfNet; for (ifp = ripIfNet; ifp; ifp = ifp->int_next) { if (ifp->int_flags & IFF_PASSIVE || ifp == skipif) continue; dst = ifp->int_flags & IFF_BROADCAST ? &ifp->int_broadaddr : ifp->int_flags & IFF_POINTOPOINT ? &ifp->int_dstaddr : &ifp->int_addr; flags = ifp->int_flags & IFF_INTERFACE ? MSG_DONTROUTE : 0; (*f)(dst, flags, ifp, rtstate); }}/* * Output a preformed packet. *//*ARGSUSED*/STATUS sndmsg(dst, flags, ifp, rtstate) struct sockaddr *dst; int flags; struct interface *ifp; int rtstate;{ IMPORT RIP ripState; (*afswitch[dst->sa_family].af_output)(ripState.s, flags, dst, sizeof (RIP_PKT)); if (routedDebug > 2) logMsg ("Transmitting RIP message.\n", 0, 0, 0, 0, 0, 0); return (OK);}/* * Supply dst with the contents of the routing tables. * If this won't fit in one packet, chop it up into several. */STATUS supply ( struct sockaddr *dst, int flags, register struct interface *ifp, int rtstate ) { IMPORT RIP ripState; register struct rt_entry *rt; register struct netinfo *pNetinfo = ripState.msg->rip_nets; register struct rthash *rh; struct rthash *base = hosthash; int doinghost = 1, size; int (*output)() = afswitch[dst->sa_family].af_output; int (*sendroute)() = afswitch[dst->sa_family].af_sendroute; int npackets = 0; struct interface* pIfp; ripState.msg->rip_cmd = RIPCMD_RESPONSE; if (ifp == NULL) pIfp = ripIfLookup(dst); else pIfp = ifp; if (pIfp == NULL) return (ERROR); /* * If this interface has been turned off then silently drop * packets on it. */ if (pIfp->ifConf.rip2IfConfStatus != M2_rip2IfConfStatus_valid) return (ERROR); if (pIfp->ifConf.rip2IfConfSend == M2_rip2IfConfSend_doNotSend) return (ERROR); else if (pIfp->ifConf.rip2IfConfSend == M2_rip2IfConfSend_ripVersion1) ripState.msg->rip_vers = 1; else if ((pIfp->ifConf.rip2IfConfSend == M2_rip2IfConfSend_rip1Compatible) || (pIfp->ifConf.rip2IfConfSend == M2_rip2IfConfSend_ripVersion2)) ripState.msg->rip_vers = 2; else return (ERROR); memset(ripState.msg->rip_domain, 0, sizeof(ripState.msg->rip_domain)); /* * If we are doing authentication then properly fill in the first field. * */ if (pIfp->ifConf.rip2IfConfAuthType == M2_rip2IfConfAuthType_simplePassword) { bzero((char *)pNetinfo, sizeof(RIP2PKT)); ((RIP2PKT *)pNetinfo)->family = RIP2_AUTH; ((RIP2PKT *)pNetinfo)->tag = M2_rip2IfConfAuthType_simplePassword; bcopy((char *)pIfp->ifConf.rip2IfConfAuthKey, (char *)pNetinfo + 4, RIP2_AUTH_LEN); pNetinfo++; } again: for (rh = base; rh < &base[ROUTEHASHSIZ]; rh++) for (rt = rh->rt_forw; rt != (struct rt_entry *)rh; rt = rt->rt_forw) { if (routedDebug) logMsg ("Looking at IF %s, route %d %d %d %d\n", (int )rt->rt_ifp->int_name, rt->rt_router.sa_data[2], rt->rt_router.sa_data[3], rt->rt_router.sa_data[4], rt->rt_router.sa_data[5], 6); /* * This calls the per interface send hook. The user may * add their own hook that allows them to decide on a route * by route basis what routes to add to the update. */ if (pIfp->sendHook != NULL) if (!pIfp->sendHook(rt)) continue; /* * When sending in response to a query, ifp is zero, disabling * this test. Otherwise, information is not sent on the network * from which it was received (simple split horizon). */ if (ifp && (rt->rt_ifp == ifp)) continue; /* * "External" routes are only created for a loopback interface and * are never sent. */ if (rt->rt_state & RTS_EXTERNAL) continue; /* * This test detects routes with the specified state. It limits * the contents of dynamic updates to route entries with * the RTS_CHANGED flag set. */ if (rtstate && (rt->rt_state & rtstate) == 0) continue; /* * This test implements the network-related border gateway * filtering specified by RFC 1058 as well as the restrictions * in section 3.3 of RFC 1723 needed for compatibility between * RIPv1 and RIPv2. It selects between the internally generated * routes sent to "distant" hosts (which are not directly * connected to the destination) and the (possibly classless) * route entries which are only sent to neighbors on the same * logical network. */ if (doinghost == 0 && rt->rt_state & RTS_SUBNET) { if (rt->rt_dst.sa_family != dst->sa_family) continue; if (ripFilterFlag) { /* * Perform border gateway filtering if enabled. The * restrictions are only needed if RIP-1 routers are * in use on the network. */ if ( (*sendroute)(rt, dst, ifp) == 0) continue; } else { /* * Border gateway filtering is disabled. Internally * generated routes (which represent the network as * a whole) are never sent. All other classless routes * are included unconditionally. */ if (rt->rt_state & RTS_INTERNAL) continue; } } /* * Limit any host route to neighbors within the same logical * network. This test also handles some network routes that * appear to be host routes to a router because they use a * longer prefix than the receiving interface. These * restrictions are only necessary if border gateway filtering * is enabled to support an environment with mixed RIP-1 and * RIP-2 routers. */ if (doinghost == 1 && ripFilterFlag) { if (rt->rt_dst.sa_family != dst->sa_family) continue; if ((*sendroute)(rt, dst, ifp) == 0) continue; } size = (char *)pNetinfo - ripState.packet; if (size > MAXPACKETSIZE - sizeof (struct netinfo)) { if (routedDebug > 2) logMsg ("Transmitting RIP message.\n", 0, 0, 0, 0, 0, 0); (*output)(ripState.s, flags, dst, size); /* * If only sending to ourselves, * one packet is enough to monitor interface. */ if (ifp && (ifp->int_flags & (IFF_BROADCAST | IFF_POINTOPOINT | IFF_REMOTE)) == 0) return (ERROR); pNetinfo = ripState.msg->rip_nets; npackets++; }#define osa(x) ((struct osockaddr *)(&(x))) osa(pNetinfo->rip_dst)->sa_family = htons(rt->rt_dst.sa_family); ripBuildPacket((RIP2PKT *)pNetinfo, rt, pIfp, pIfp->ifConf.rip2IfConfSend); pNetinfo++; } if (doinghost) { doinghost = 0; base = nethash; goto again; } if (pNetinfo != ripState.msg->rip_nets || rtstate == 0) { size = (char *)pNetinfo - ripState.packet; if (routedDebug > 2) logMsg ("Transmitting RIP message.\n", 0, 0, 0, 0, 0, 0); (*output)(ripState.s, flags, dst, size); } return (OK); }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -