📄 lineariplookup.cc
字号:
// -*- c-basic-offset: 4 -*-/* * lineariplookup.{cc,hh} -- element looks up next-hop address in linear * routing table * Robert Morris, Eddie Kohler * * Copyright (c) 1999-2000 Massachusetts Institute of Technology * Copyright (c) 2002 International Computer Science Institute * * 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 <click/config.h>#include "lineariplookup.hh"#include <click/ipaddress.hh>#include <click/straccum.hh>#include <click/error.hh>CLICK_DECLSLinearIPLookup::LinearIPLookup() : _zero_route(-1){ add_input();}LinearIPLookup::~LinearIPLookup(){}voidLinearIPLookup::notify_noutputs(int n){ set_noutputs(n);}intLinearIPLookup::initialize(ErrorHandler *){ _last_addr = IPAddress();#ifdef IP_RT_CACHE2 _last_addr2 = _last_addr;#endif return 0;}boolLinearIPLookup::check() const{ bool ok = true; //click_chatter("%s\n", ((LinearIPLookup*)this)->dump_routes().c_str()); // 'next' pointers are correct for (int i = 0; i < _t.size(); i++) { if (!_t[i].real()) continue; for (int j = i + 1; j < _t[i].extra && j < _t.size(); j++) if (_t[i].contains(_t[j]) && _t[j].real()) { click_chatter("%s: bad next pointers: routes %s, %s", declaration().cc(), _t[i].unparse_addr().cc(), _t[j].unparse_addr().cc()); ok = false; }#if 0 // This invariant actually does not hold. int j = _t[i].extra; if (j < _t.size()) if (!_t[i].contains(_t[j]) && _t[j].real()) { click_chatter("%s: bad next pointers': routes %s, %s", declaration().cc(), _t[i].unparse_addr().cc(), _t[j].unparse_addr().cc()); ok = false; }#endif } // no duplicate routes for (int i = 0; i < _t.size(); i++) for (int j = i + 1; j < _t.size(); j++) if (_t[i].addr == _t[j].addr && _t[i].mask == _t[j].mask && _t[i].real() && _t[j].real()) { click_chatter("%s: duplicate routes for %s", declaration().cc(), _t[i].unparse_addr().cc()); ok = false; } // caches point to the right place if (_last_addr && lookup_entry(_last_addr) != _last_entry) { click_chatter("%s: bad cache entry for %s", declaration().cc(), _last_addr.unparse().cc()); ok = false; }#ifdef IP_RT_CACHE2 if (_last_addr2 && lookup_entry(_last_addr2) != _last_entry2) { click_chatter("%s: bad cache entry for %s", declaration().cc(), _last_addr2.unparse().cc()); ok = false; }#endif return ok;}intLinearIPLookup::add_route(const IPRoute &r, bool allow_replace, IPRoute* replaced, ErrorHandler *){ // overwrite any existing route int found = _t.size(); for (int i = 0; i < _t.size(); i++) if (!_t[i].real()) { if (found == _t.size()) found = i; } else if (_t[i].addr == r.addr && _t[i].mask == r.mask) { if (replaced) *replaced = _t[i]; if (!allow_replace) return -EEXIST; _t[i].gw = r.gw; _t[i].port = r.port; check(); return 0; } // maybe make a new slot if (found == _t.size()) _t.push_back(r); else _t[found] = r; // patch up next pointers _t[found].extra = 0x7FFFFFFF; for (int i = found - 1; i >= 0; i--) if (_t[i].contains(r) && _t[i].extra > found) _t[i].extra = found; for (int i = found + 1; i < _t.size(); i++) if (r.contains(_t[i]) && _t[i].real()) { _t[found].extra = i; break; } // remember zero route if (!r.addr && r.mask.addr() == 0xFFFFFFFFU) _zero_route = found; // get rid of caches _last_addr = IPAddress();#ifdef IP_RT_CACHE2 _last_addr2 = IPAddress();#endif check(); return 0;}intLinearIPLookup::remove_route(const IPRoute& route, IPRoute* old_route, ErrorHandler *errh){ for (int i = 0; i < _t.size(); i++) if (route.match(_t[i])) { if (old_route) *old_route = _t[i]; _t[i].kill(); // need to handle zero routes, bummer if (i == _zero_route) _zero_route = -1; else if (i < _zero_route) { IPRoute zero(_t[_zero_route]); _t[_zero_route].kill(); int r = add_route(zero, false, 0, errh); assert(r >= 0); } // get rid of caches _last_addr = IPAddress();#ifdef IP_RT_CACHE2 _last_addr2 = IPAddress();#endif check(); return 0; } return -ENOENT;}intLinearIPLookup::lookup_entry(IPAddress a) const{ for (int i = 0; i < _t.size(); i++) if (_t[i].contains(a)) { int found = i; for (int j = _t[i].extra; j < _t.size(); j++) if (_t[j].contains(a) && _t[j].mask_as_specific(_t[found].mask)) found = j; return found; } return -1;}intLinearIPLookup::lookup_route(IPAddress a, IPAddress &gw) const{ int ei = lookup_entry(a); if (ei >= 0) { gw = _t[ei].gw; return _t[ei].port; } else return -1;}StringLinearIPLookup::dump_routes(){ StringAccum sa; for (int i = 0; i < _t.size(); i++) if (_t[i].real()) _t[i].unparse(sa, true) << '\n'; return sa.take_string();}voidLinearIPLookup::push(int, Packet *p){#define EXCHANGE(a,b,t) { t = a; a = b; b = t; } IPAddress a = p->dst_ip_anno(); int ei = -1; if (a && a == _last_addr) ei = _last_entry;#ifdef IP_RT_CACHE2 else if (a && a == _last_addr2) ei = _last_entry2;#endif else if ((ei = lookup_entry(a)) >= 0) {#ifdef IP_RT_CACHE2 _last_addr2 = _last_addr; _last_entry2 = _last_entry;#endif _last_addr = a; _last_entry = ei; } else { click_chatter("LinearIPLookup: no gw for %x", a.addr()); p->kill(); return; } const IPRoute &e = _t[ei]; if (e.gw) p->set_dst_ip_anno(e.gw); output(e.port).push(p);}#include <click/vector.cc>CLICK_ENDDECLSELEMENT_REQUIRES(IPRouteTable)EXPORT_ELEMENT(LinearIPLookup)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -