/* * lookupiproutemp.{cc,hh} -- element looks up next-hop address in static * routing table, has processor local cache in SMP mode. * Robert Morris * * Copyright (c) 1999-2000 Massachusetts Institute of Technology * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, subject to the conditions * listed in the Click LICENSE file. These conditions include: you must * preserve this copyright notice, and you cannot mention the copyright * holders in advertising related to the Software without their permission. * The Software is provided WITHOUT ANY WARRANTY, EXPRESS OR IMPLIED. This * notice is a summary of the Click LICENSE file; the license in that file is * legally binding. */ #include #include "lookupiproutemp.hh" #include #include #include #include #include CLICK_CXX_PROTECT #ifdef CLICK_LINUXMODULE #include #endif CLICK_CXX_UNPROTECT #include CLICK_DECLS LookupIPRouteMP::LookupIPRouteMP() { } LookupIPRouteMP::~LookupIPRouteMP() { } int LookupIPRouteMP::configure(Vector &conf, ErrorHandler *errh) { int maxout = -1; _t.clear(); int before = errh->nerrors(); for (int i = 0; i < conf.size(); i++) { uint32_t dst, mask, gw = 0; int32_t output_num; bool ok = false; Vector words; cp_spacevec(conf[i], words); if ((words.size() == 2 || words.size() == 3) && cp_ip_prefix(words[0], (unsigned char *)&dst, (unsigned char *)&mask, true, this) // allow base IP addresses && cp_integer(words.back(), &output_num)) { if (words.size() == 3) ok = cp_ip_address(words[1], (unsigned char *)&gw, this); else ok = true; } if (ok && output_num >= 0) { _t.add(dst, mask, gw, output_num); if (output_num > maxout) maxout = output_num; } else errh->error("argument %d should be `DADDR/MASK [GATEWAY] OUTPUT'", i+1); } if (errh->nerrors() != before) return -1; if (maxout < 0) errh->warning("no routes"); if (maxout >= noutputs()) return errh->error("need %d or more output ports", maxout + 1); return 0; } int LookupIPRouteMP::initialize(ErrorHandler *) { click_chatter("LookupIPRouteMP alignment: %p, %p", &(_cache[0]._last_addr_1), &(_cache[1]._last_addr_1)); for (int i=0; i<_cache_buckets; i++) { _cache[i]._last_addr_1 = IPAddress(); _cache[i]._last_addr_2 = IPAddress(); } return 0; } void LookupIPRouteMP::push(int, Packet *p) { IPAddress a = p->dst_ip_anno(); IPAddress gw; int ifi = -1; #ifdef CLICK_LINUXMODULE int bucket = click_current_processor(); struct cache_entry *e = &_cache[bucket]; #else int bucket = 0; #endif if (a) { if (a == e->_last_addr_1) { if (e->_last_gw_1) p->set_dst_ip_anno(e->_last_gw_1); output(e->_last_output_1).push(p); return; } else if (a == e->_last_addr_2) { if (e->_last_gw_2) p->set_dst_ip_anno(e->_last_gw_2); output(e->_last_output_2).push(p); return; } } if (_t.lookup(a, gw, ifi)) { e->_last_addr_2 = e->_last_addr_1; e->_last_gw_2 = e->_last_gw_1; e->_last_output_2 = e->_last_output_1; e->_last_addr_1 = a; e->_last_gw_1 = gw; e->_last_output_1 = ifi; if (gw) p->set_dst_ip_anno(gw); output(ifi).push(p); } else { click_chatter("LookupIPRouteMP: no gw for %x", a.addr()); p->kill(); } } CLICK_ENDDECLS ELEMENT_REQUIRES(linuxmodule) EXPORT_ELEMENT(LookupIPRouteMP)