package org.freertr.rtr; import java.util.List; import org.freertr.addr.addrIP; import org.freertr.addr.addrIPv4; import org.freertr.addr.addrPrefix; import org.freertr.auth.authLocal; import org.freertr.cfg.cfgAll; import org.freertr.cfg.cfgPrfxlst; import org.freertr.cfg.cfgRoump; import org.freertr.cfg.cfgRouplc; import org.freertr.ip.ipFwdIface; import org.freertr.pack.packHolder; import org.freertr.prt.prtGenConn; import org.freertr.tab.tabListing; import org.freertr.tab.tabPrfxlstN; import org.freertr.tab.tabRoute; import org.freertr.tab.tabRouteAttr; import org.freertr.tab.tabRouteEntry; import org.freertr.tab.tabRtrmapN; import org.freertr.tab.tabRtrplcN; import org.freertr.user.userHelping; import org.freertr.util.bits; import org.freertr.util.cmds; import org.freertr.util.debugger; import org.freertr.util.logger; /** * ripv2 interface * * @author matecsaba */ public class rtrRip4iface implements Comparable { /** * advertisement interval */ public int updateTimer = 30000; /** * hold time */ public int holdTimer = 180000; /** * invalid time */ public int flushTimer = 240000; /** * time to wait between packets */ public int interPackTime = 20; /** * default distance */ public int distance = 120; /** * default input metric */ public int metricIn = 0; /** * default output metric */ public int metricOut = 1; /** * bfd enabled */ public boolean bfdTrigger; /** * not advertise routes learned from interface back */ public boolean splitHorizon = true; /** * advertise back best routes with infinity metric */ public boolean poisonReverse = true; /** * check neighbor address is connected */ public boolean connectedCheck = true; /** * authentication string */ public String authentication = null; /** * allow receive of routes */ public boolean allowRx = true; /** * allow transmit of routes */ public boolean allowTx = true; /** * advertise default route */ public boolean defOrigin = false; /** * suppress interface address */ public boolean suppressAddr = false; /** * unsuppress interface address */ public boolean unsuppressAddr = false; /** * ingress prefix list */ public tabListing prflstIn; /** * egress prefix list */ public tabListing prflstOut; /** * ingress route map */ public tabListing roumapIn; /** * egress route map */ public tabListing roumapOut; /** * ingress route policy */ public tabListing roupolIn; /** * egress route policy */ public tabListing roupolOut; /** * the interface this works on */ protected final ipFwdIface iface; /** * the udp connection it uses to multicast */ protected prtGenConn conn; /** * the lower layer */ protected final rtrRip4 lower; /** * create one instance * * @param parent the rip protocol * @param ifc the ip interface to work on */ public rtrRip4iface(rtrRip4 parent, ipFwdIface ifc) { lower = parent; iface = ifc; } /** * unregister from udp */ public void unregister2udp() { lower.udpCore.listenStop(iface, rtrRip4.port, null, 0); conn.setClosing(); } /** * register to udp */ public void register2udp() { addrIP adr = new addrIP(); adr.fromString("224.0.0.9"); lower.udpCore.packetListen(lower, iface, rtrRip4.port, null, 0, "rip", -1, null, -1, -1); conn = lower.udpCore.packetConnect(lower, iface, rtrRip4.port, adr, rtrRip4.port, "rip", -1, null, -1, -1); if (conn == null) { return; } conn.timeout = 0; } public String toString() { return "rip on " + iface; } public int compareTo(rtrRip4iface o) { if (iface.ifwNum < o.iface.ifwNum) { return -1; } if (iface.ifwNum > o.iface.ifwNum) { return +1; } return 0; } /** * get configuration * * @param l list to add * @param beg beginning * @param filter filter defaults */ public void routerGetConfig(List l, String beg, int filter) { l.add(cmds.tabulator + beg + "enable"); cmds.cfgLine(l, !allowRx, cmds.tabulator, beg + "allow-rx", ""); cmds.cfgLine(l, !allowTx, cmds.tabulator, beg + "allow-tx", ""); cmds.cfgLine(l, !bfdTrigger, cmds.tabulator, beg + "bfd", ""); cmds.cfgLine(l, !connectedCheck, cmds.tabulator, beg + "verify-source", ""); cmds.cfgLine(l, !poisonReverse, cmds.tabulator, beg + "poison-reverse", ""); cmds.cfgLine(l, !splitHorizon, cmds.tabulator, beg + "split-horizon", ""); cmds.cfgLine(l, !defOrigin, cmds.tabulator, beg + "default-originate", ""); cmds.cfgLine(l, !suppressAddr, cmds.tabulator, beg + "suppress-prefix", ""); cmds.cfgLine(l, !unsuppressAddr, cmds.tabulator, beg + "unsuppress-prefix", ""); l.add(cmds.tabulator + beg + "distance " + distance); l.add(cmds.tabulator + beg + "metric-in " + metricIn); l.add(cmds.tabulator + beg + "metric-out " + metricOut); l.add(cmds.tabulator + beg + "packet-timer " + interPackTime); l.add(cmds.tabulator + beg + "update-timer " + updateTimer); l.add(cmds.tabulator + beg + "hold-time " + holdTimer); l.add(cmds.tabulator + beg + "flush-time " + flushTimer); cmds.cfgLine(l, authentication == null, cmds.tabulator, beg + "password", authLocal.passwdEncode(authentication, (filter & 2) != 0)); cmds.cfgLine(l, prflstIn == null, cmds.tabulator, beg + "prefix-list-in", "" + prflstIn); cmds.cfgLine(l, prflstOut == null, cmds.tabulator, beg + "prefix-list-out", "" + prflstOut); cmds.cfgLine(l, roumapIn == null, cmds.tabulator, beg + "route-map-in", "" + roumapIn); cmds.cfgLine(l, roumapOut == null, cmds.tabulator, beg + "route-map-out", "" + roumapOut); cmds.cfgLine(l, roupolIn == null, cmds.tabulator, beg + "route-policy-in", "" + roupolIn); cmds.cfgLine(l, roupolOut == null, cmds.tabulator, beg + "route-policy-out", "" + roupolOut); } /** * do configuration * * @param a command * @param cmd parameters */ public void routerDoConfig(String a, cmds cmd) { if (a.equals("allow-rx")) { allowRx = true; return; } if (a.equals("allow-tx")) { allowTx = true; return; } if (a.equals("bfd")) { bfdTrigger = true; return; } if (a.equals("default-originate")) { defOrigin = true; return; } if (a.equals("suppress-prefix")) { suppressAddr = true; return; } if (a.equals("unsuppress-prefix")) { unsuppressAddr = true; return; } if (a.equals("verify-source")) { connectedCheck = true; return; } if (a.equals("poison-reverse")) { poisonReverse = true; return; } if (a.equals("split-horizon")) { splitHorizon = true; return; } if (a.equals("password")) { authentication = authLocal.passwdDecode(cmd.word()); return; } if (a.equals("distance")) { distance = bits.str2num(cmd.word()); return; } if (a.equals("metric-in")) { metricIn = bits.str2num(cmd.word()); return; } if (a.equals("metric-out")) { metricOut = bits.str2num(cmd.word()); return; } if (a.equals("packet-timer")) { interPackTime = bits.str2num(cmd.word()); return; } if (a.equals("update-timer")) { updateTimer = bits.str2num(cmd.word()); return; } if (a.equals("hold-time")) { holdTimer = bits.str2num(cmd.word()); return; } if (a.equals("flush-time")) { flushTimer = bits.str2num(cmd.word()); return; } if (a.equals("prefix-list-in")) { cfgPrfxlst ntry = cfgAll.prfxFind(cmd.word(), false); if (ntry == null) { cmd.error("no such prefix list"); return; } prflstIn = ntry.prflst; return; } if (a.equals("prefix-list-out")) { cfgPrfxlst ntry = cfgAll.prfxFind(cmd.word(), false); if (ntry == null) { cmd.error("no such prefix list"); return; } prflstOut = ntry.prflst; return; } if (a.equals("route-map-in")) { cfgRoump ntry = cfgAll.rtmpFind(cmd.word(), false); if (ntry == null) { cmd.error("no such route map"); return; } roumapIn = ntry.roumap; return; } if (a.equals("route-map-out")) { cfgRoump ntry = cfgAll.rtmpFind(cmd.word(), false); if (ntry == null) { cmd.error("no such route map"); return; } roumapOut = ntry.roumap; return; } if (a.equals("route-policy-in")) { cfgRouplc ntry = cfgAll.rtplFind(cmd.word(), false); if (ntry == null) { cmd.error("no such route policy"); return; } roupolIn = ntry.rouplc; return; } if (a.equals("route-policy-out")) { cfgRouplc ntry = cfgAll.rtplFind(cmd.word(), false); if (ntry == null) { cmd.error("no such route policy"); return; } roupolOut = ntry.rouplc; return; } cmd.badCmd(); } /** * undo configuration * * @param a command * @param cmd parameters */ public void routerUnConfig(String a, cmds cmd) { if (a.equals("allow-rx")) { allowRx = false; return; } if (a.equals("allow-tx")) { allowTx = false; return; } if (a.equals("bfd")) { bfdTrigger = false; return; } if (a.equals("default-originate")) { defOrigin = false; return; } if (a.equals("suppress-prefix")) { suppressAddr = false; return; } if (a.equals("unsuppress-prefix")) { unsuppressAddr = false; return; } if (a.equals("verify-source")) { connectedCheck = false; return; } if (a.equals("poison-reverse")) { poisonReverse = false; return; } if (a.equals("split-horizon")) { splitHorizon = false; return; } if (a.equals("password")) { authentication = null; return; } if (a.equals("prefix-list-in")) { prflstIn = null; return; } if (a.equals("prefix-list-out")) { prflstOut = null; return; } if (a.equals("route-map-in")) { roumapIn = null; return; } if (a.equals("route-map-out")) { roumapOut = null; return; } if (a.equals("route-policy-in")) { roupolIn = null; return; } if (a.equals("route-policy-out")) { roupolOut = null; return; } cmd.badCmd(); } /** * get help text * * @param l list to update */ public static void routerGetHelp(userHelping l) { l.add(null, "4 . enable enable/disable rip updates"); l.add(null, "4 . allow-rx suppress processing routing updates"); l.add(null, "4 . allow-tx suppress sending routing updates"); l.add(null, "4 . bfd enable bfd triggered down"); l.add(null, "4 . default-originate send default route to peer"); l.add(null, "4 . suppress-prefix do not advertise interface"); l.add(null, "4 . unsuppress-prefix do advertise interface"); l.add(null, "4 . verify-source check source address of updates"); l.add(null, "4 . poison-reverse advertise back best routes"); l.add(null, "4 . split-horizon dont advertise back on rx interface"); l.add(null, "4 5 password password for authentication"); l.add(null, "5 . set password"); l.add(null, "4 5 distance administrative distance of routes"); l.add(null, "5 . set administrative distance"); l.add(null, "4 5 metric-out metric of outgoing routes"); l.add(null, "5 . set metric"); l.add(null, "4 5 metric-in metric of incoming routes"); l.add(null, "5 . set metric"); l.add(null, "4 5 packet-timer inter packet gap time"); l.add(null, "5 . time in ms"); l.add(null, "4 5 update-timer time between updates"); l.add(null, "5 . time in ms"); l.add(null, "4 5 hold-time hold prefixes in routing table"); l.add(null, "5 . time in ms"); l.add(null, "4 5 flush-time flush prefix after inactivity"); l.add(null, "5 . time in ms"); l.add(null, "4 5 route-map-in process prefixes in ingress updates"); l.add(null, "5 . name of route map"); l.add(null, "4 5 route-map-out process prefixes in egress updates"); l.add(null, "5 . name of route map"); l.add(null, "4 5 route-policy-in process prefixes in ingress updates"); l.add(null, "5 . name of route policy"); l.add(null, "4 5 route-policy-out process prefixes in egress updates"); l.add(null, "5 . name of route policy"); l.add(null, "4 5 prefix-list-in filter prefixes in ingress updates"); l.add(null, "5 . name of prefix list"); l.add(null, "4 5 prefix-list-out filter prefixes in egress updates"); l.add(null, "5 . name of prefix list"); } /** * create authentication data * * @return byte buffer of authentication data */ protected byte[] getAuthData() { if (authentication == null) { return new byte[0]; } byte[] buf = new byte[rtrRip4.sizeNtry]; byte[] ab = authentication.getBytes(); bits.byteFill(buf, 0, rtrRip4.sizeNtry, 0); bits.msbPutW(buf, 0, 0xffff); // authentication header bits.msbPutW(buf, 2, 2); // authentication type for (int i = 0;; i++) { int o = i + 4; if (i >= ab.length) { break; } if (o >= buf.length) { break; } buf[o] = ab[i]; } return buf; } private void createRIPheader(int cmd, packHolder pck) { pck.clear(); pck.putStart(); pck.putByte(0, cmd); // command pck.putByte(1, rtrRip4.version); // version pck.msbPutW(2, 0); // reserved pck.putSkip(rtrRip4.sizeHead); // size of header byte[] buf = getAuthData(); pck.putCopy(buf, 0, 0, buf.length); pck.putSkip(buf.length); } private void createRIPupdate(tabRouteEntry ntry, packHolder pck) { pck.msbPutW(0, rtrRip4.afiID); // address family id pck.msbPutW(2, ntry.best.tag); // route tag pck.putAddr(4, ntry.prefix.network.toIPv4()); // network pck.putAddr(8, ntry.prefix.mask.toIPv4()); // subnet mask addrIPv4 hop; if (ntry.best.nextHop == null) { hop = new addrIPv4(); } else { hop = ntry.best.nextHop.toIPv4(); } pck.putAddr(12, hop); // next hop int i = ntry.best.metric + metricOut; if (i > rtrRip4.metricMax) { i = rtrRip4.metricMax; } if (i < rtrRip4.metricMin) { i = rtrRip4.metricMin; } ntry.best.metric = i; pck.msbPutD(16, i); // metric pck.putSkip(rtrRip4.sizeNtry); if (debugger.rtrRip4traf) { logger.debug("txnet " + ntry); } } /** * send one update over a connection * * @param conn connection to use */ protected void sendOutUpdates(prtGenConn conn) { packHolder pck = new packHolder(true, true); tabRoute tab1 = new tabRoute("copy"); if (defOrigin) { tab1.add(tabRoute.addType.better, addrPrefix.ip4toIP(addrPrefix.defaultRoute4()), new addrIP()); } tab1.mergeFrom(tabRoute.addType.better, lower.routerComputedU, tabRouteAttr.distanLim); if (splitHorizon) { if (poisonReverse) { for (int i = 0; i < tab1.size(); i++) { tabRouteEntry ntry = tab1.get(i); if (ntry.best.iface != conn.iface) { continue; } ntry.best.metric = rtrRip4.metricMax; } } else { tab1.delIface(conn.iface); } } tab1.mergeFrom(tabRoute.addType.better, lower.routerRedistedU, tabRouteAttr.distanLim); tabRoute tab2 = new tabRoute("copy"); tabRoute.addUpdatedTable(tabRoute.addType.better, rtrBgpUtil.sfiUnicast, 0, tab2, tab1, true, roumapOut, roupolOut, prflstOut); if (debugger.rtrRip4traf) { logger.debug("tx " + conn); } createRIPheader(2, pck); int entries = 0; for (int i = 0; i < tab2.size(); i++) { tabRouteEntry ntry = tab2.get(i); if (ntry == null) { continue; } if (ntry.best.iface != conn.iface) { ntry.best.nextHop = null; } createRIPupdate(ntry, pck); entries++; if (entries < rtrRip4.entryPerUpdate) { continue; } conn.send2net(pck); createRIPheader(2, pck); entries = 0; bits.sleep(interPackTime); } if (entries > 0) { conn.send2net(pck); } } /** * do some work round */ public synchronized void doWork() { if (conn == null) { return; } conn.workInterval = updateTimer; conn.timeout = 0; if (!allowTx) { return; } sendOutUpdates(conn); } }