⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 linkcache.cc

📁 NS2的dsr-ocean仿真代码,对学习研究NS2的人非常有研究价值
💻 CC
📖 第 1 页 / 共 2 页
字号:
/* * linkcache.cc * Copyright (C) 2000 by the University of Southern California * $Id: linkcache.cc,v 1.2 2005/08/25 18:58:05 johnh Exp $ * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * 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. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. * * * The copyright of this module includes the following * linking-with-specific-other-licenses addition: * * In addition, as a special exception, the copyright holders of * this module give you permission to combine (via static or * dynamic linking) this module with free software programs or * libraries that are released under the GNU LGPL and with code * included in the standard release of ns-2 under the Apache 2.0 * license or under otherwise-compatible licenses with advertising * requirements (or modified versions of such code, with unchanged * license).  You may copy and distribute such a system following the * terms of the GNU GPL for this module and the licenses of the * other code concerned, provided that you include the source code of * that other code when and as the GNU GPL requires distribution of * source code. * * Note that people who make modified versions of this module * are not obligated to grant this special exception for their * modified versions; it is their choice whether to do so.  The GNU * General Public License gives permission to release a modified * version without this exception; this exception also makes it * possible to release a modified version which carries forward this * exception. * *///// Other copyrights might apply to parts of this software and are so// noted when applicable.//// Ported from CMU/Monarch's code, appropriate copyright applies.  #ifdef DSR_LINKCACHE#include <god.h>#include "path.h"#include "routecache.h"#include <list.h>#ifdef DSR_CACHE_STATS#include "cache_stats.h"#endif#define CURRENT_TIME Scheduler::instance().clock()#define MAX_SIMTIME  86400bool cache_ignore_hints = false;     // ignore all hints?bool cache_use_overheard_routes = true; // Specifies whether or not to retain links after they go bad// I don't think this is implemented.static const bool lc_cache_negative_links = false;static const double lc_neg_cache_life = 1.0;// When inserting/deleting link A->B, also insert/delete B->A.static const bool lc_bidirectional_links = true;// Evict "unreachable" links from the cache.static const bool lc_evict_unreachable_links = false;// do we expire in realtime? or do we wait for use?//#define REALTIME_EXPIRE// The maximum number of nodes supported by the cache.#define LC_MAX_NODES 200// General generational stuff...static const double lc_gen_use_bonus = 300;// Static stuffstatic const double lc_static_discover_life = 5.0;static const double lc_static_overhear_life = 5.0;// Table stuffstatic const double lc_table_init_val = 50.0;static const double lc_table_mult_by  = 0.5;static const double lc_linear_alpha   = 1.0/8.0;static const double lc_exp_add_amt    = 2.0;static const double lc_exp_div_amt    = 2.0;static const double lc_minlifetime    = 2.0;// types of linkgens#define EXPPOLICY_INFINITE  0#define EXPPOLICY_STATIC    1#define EXPPOLICY_LINEAR    2#define EXPPOLICY_EXP       3#ifdef GOD_STABILITY#define EXPPOLICY_GOD_OMNI  4#define EXPPOLICY_GOD_TABLE 5#endif#define LONGEST_LIVED_ROUTEstatic const int lc_exppolicy = EXPPOLICY_EXP;#ifdef GOD_STABILITYextern double *godavglife;#endif#ifndef REALTIME_EXPIRE// statistics...#define IS_GOOD     0#define IS_BAD      1#define CALLED_GOOD 0#define CALLED_BAD  2static int expirestats[4];#endifstatic int verbose_debug = 0;// *****************************************************class Link {public:	Link(nsaddr_t dst) {		ln_dst = dst;		ln_flags = ln_cost = 0;		ln_timeout = 0.0;		ln_insert = 0.0;		ln_t = 0.0;		ln_linktype = LT_NONE;		ln_logstat = LS_UNLOGGED;	}	inline bool expired() {		return (ln_timeout <= CURRENT_TIME);	}	LIST_ENTRY(Link) ln_link;	nsaddr_t   ln_dst;	int        ln_flags;       // link status information#define LINK_FLAG_UP 0x01	/*	 * Right now all links have the same cost (1). - josh	 */	int        ln_cost;        // cost of using the link	/*	 * the use of ln_timeout to expire links from the	 * cache has not yet been implemented. - josh	 * Implemented. - yihchun	 */	double     ln_timeout;     // when the link expires	double     ln_insert;      // when the link expires#define LINK_TIMEOUT MAX_SIMTIME	// mirror the values kept in class ID	Time	   ln_t;	Link_Type  ln_linktype;	Log_Status ln_logstat;};LIST_HEAD(dsrLinkHead, Link);class LinkCache : public RouteCache {friend class MobiHandler;public:	LinkCache();	void noticeDeadLink(const ID&from, const ID& to, Time t);	// the link from->to isn't working anymore, purge routes containing	// it from the cache	void noticeRouteUsed(const Path& route, Time t,			     const ID& who_from);	// tell the cache about a route we saw being used	void addRoute(const Path& route, Time t, const ID& who_from);	// add this route to the cache (presumably we did a route request	// to find this route and don't want to lose it)	bool findRoute(ID dest, Path& route, int for_me = 0);	// if there is a cached path from us to dest returns true and fills in	// the route accordingly. returns false otherwise	// if for_use, then we assume that the node really wants to keep 	// the returned route so it will be promoted to primary storage	// if not there already	int command(int argc, const char*const* argv);protected:	dsrLinkHead lcache[LC_MAX_NODES + 1];	// note the zeroth index is not used	int addLink(const ID& from, const ID& to,		    int flags, double timeout = LINK_TIMEOUT, int cost = 1);	int delLink(const ID& from, const ID& to);	Link* findLink(int from, int to);	void purgeLink(void);	void dumpLink(void);	void Link_to_ID(Link* l, ID& id);#ifdef DSR_CACHE_STATS	void checkLink_logall(const ID& from, const ID& to, int action);	void checkLink(const ID& from, const ID& to, int action);	void periodic_checkCache();	struct cache_stats stat;#endifprivate:	////////////////////////////////////////////////////////////	// Dijkstra's Algorithm	double dirty; // the next time it gets dirty#ifdef LONGEST_LIVED_ROUTE	double dl[LC_MAX_NODES + 1];#endif	u_int32_t d[LC_MAX_NODES + 1];	u_int32_t pi[LC_MAX_NODES + 1];	bool S[LC_MAX_NODES + 1];	double exptable[LC_MAX_NODES + 1];#define INFINITY 0x7fffffff	void init_single_source(int s);#ifdef LONGEST_LIVED_ROUTE	void relax(u_int32_t u, u_int32_t v, u_int32_t w, double);#else	void relax(u_int32_t u, u_int32_t v, u_int32_t w);#endif	int  extract_min_q(void);	void dijkstra(void);	void dump_dijkstra(int dst);	double find_timeout(ID a, ID b, bool discovered);};///////////////////////////////////////////////////////////////////////////double LinkCache::find_timeout(ID a, ID b, bool discovered) {	double lifetime = 0.0;	switch (lc_exppolicy) {	case EXPPOLICY_INFINITE:#ifdef GOD_STABILITY        case EXPPOLICY_GOD_OMNI:#endif		return MAX_SIMTIME;	case EXPPOLICY_LINEAR: case EXPPOLICY_EXP:		lifetime = exptable[a.addr] > exptable[b.addr] ? 		  exptable[b.addr] : exptable[a.addr];		lifetime *= lc_table_mult_by;		break;#ifdef GOD_STABILITY	case EXPPOLICY_GOD_TABLE:		lifetime = godavglife[a.addr] > godavglife[b.addr] ? 		  godavglife[b.addr] : godavglife[a.addr];		lifetime *= lc_table_mult_by;		break;#endif	case EXPPOLICY_STATIC:		lifetime = discovered ? lc_static_discover_life :		  lc_static_overhear_life;		break;	default:		assert(0);	}	if (lifetime < lc_minlifetime)		lifetime = lc_minlifetime;	return CURRENT_TIME + lifetime;}RouteCache *makeRouteCache(){  return new LinkCache();}LinkCache::LinkCache() : RouteCache(){	int i = 0;	for(i = 0; i <= LC_MAX_NODES; i++) {		LIST_INIT(&lcache[i]);		exptable[i] = lc_table_init_val;	}#ifdef DSR_CACHE_STATS	stat.reset();#endif	dirty = -1;}intLinkCache::command(int argc, const char*const* argv){  if(argc == 2 && strcasecmp(argv[1], "startdsr") == 0)    {       if (ID(1,::IP) == net_id) 	trace("Sconfig %.5f using LINKCACHE %d", Scheduler::instance().clock(),		lc_exppolicy);      // FALL-THROUGH    }  if(argc == 3)    {         if (strcasecmp(argv[1], "ip-addr") == 0)        {          assert(atoi(argv[2]) <= LC_MAX_NODES);#ifndef REALTIME_EXPIRE	  if (atoi(argv[2]) == 1)	    bzero(expirestats, sizeof(expirestats));#endif          // don't return        }     }  return RouteCache::command(argc, argv);}voidLinkCache::noticeDeadLink(const ID& from, const ID& to, Time){	Link *l = findLink(from.addr, to.addr);	if (l && (l->ln_flags & LINK_FLAG_UP) &&	    (lc_exppolicy == EXPPOLICY_LINEAR || 	     lc_exppolicy == EXPPOLICY_EXP)) {		if (lc_exppolicy == EXPPOLICY_LINEAR) {			exptable[from.addr]=lc_linear_alpha*exptable[from.addr]			  +(CURRENT_TIME-l->ln_insert)*(1-lc_linear_alpha);			exptable[to.addr]=lc_linear_alpha*exptable[to.addr]			  +(CURRENT_TIME-l->ln_insert)*(1-lc_linear_alpha);		} else {			exptable[from.addr] /= lc_exp_div_amt;			exptable[to.addr] /= lc_exp_div_amt;		}	}	if(lc_cache_negative_links == true) {		if(l) {			l->ln_flags &= ~LINK_FLAG_UP;			l->ln_insert = CURRENT_TIME;			l->ln_timeout = CURRENT_TIME + lc_neg_cache_life;			dirty = CURRENT_TIME;		} else {			addLink(from, to, 0, CURRENT_TIME + lc_neg_cache_life);		}	}	else {		if(l) {#ifdef DSR_CACHE_STATS			checkLink(from, to, ACTION_DEAD_LINK);#endif			(void) delLink(from, to);		}	}}voidLinkCache::noticeRouteUsed(const Path& p, Time t, const ID& who_from){  Path stub;  int c;  if(pre_noticeRouteUsed(p, stub, t, who_from) == 0)	  return;  #ifdef DSR_CACHE_STATS  int link_notice_count = stat.link_notice_count;   int link_notice_bad_count = stat.link_notice_bad_count;   #endif  /*   * Add each link in the "path" to the route cache.   */  for(c = 0; c < stub.length() - 1; c++) {	  if(addLink(stub[c], stub[c+1], LINK_FLAG_UP, 		     find_timeout(stub[c], stub[c+1], false))) {#ifdef DSR_CACHE_STATS		checkLink(stub[c], stub[c+1], ACTION_NOTICE_ROUTE);#endif	  }  }#ifdef DSR_CACHE_STATS  if(stat.link_notice_count > link_notice_count)	  stat.route_notice_count++;  if(stat.link_notice_bad_count > link_notice_bad_count)	  stat.route_notice_bad_count++;#endif}voidLinkCache::addRoute(const Path& route, Time t, const ID& who_from){  Path rt;  int c;  if(pre_addRoute(route, rt, t, who_from) == 0)	  return;  #ifdef DSR_CACHE_STATS  int link_add_count = stat.link_add_count;   int link_add_bad_count = stat.link_add_bad_count;   #endif  for(c = 0; c < rt.length() - 1; c++) {	  if(addLink(rt[c], rt[c+1], LINK_FLAG_UP,		     find_timeout(rt[c], rt[c+1], true))) {#ifdef DSR_CACHE_STATS		  checkLink(rt[c], rt[c+1], ACTION_ADD_ROUTE);#endif	  }  }#ifdef DSR_CACHE_STATS  if(stat.link_add_count > link_add_count)	  stat.route_add_count++;  if(stat.link_add_bad_count > link_add_bad_count)	  stat.route_add_bad_count++;#endif}boolLinkCache::findRoute(ID dest, Path& route, int for_me = 0){	u_int32_t v;	int rpath[MAX_SR_LEN];	u_int32_t roff = 0;	if(verbose_debug) dumpLink();	/*	 * Compute all of the shortest paths...	 */	if(dirty <= CURRENT_TIME) {		dijkstra();		// remove all of the links for nodes that are unreachable		if(lc_evict_unreachable_links == true) {			purgeLink();		}	}	if(verbose_debug) dump_dijkstra(dest.addr);	/*	 * Trace backwards to figure out the path to DEST	 */	for(v = dest.addr; d[v] < INFINITY; v = pi[v]) {	  //assert(v >= 1 && v <= LC_MAX_NODES);		if (roff >= MAX_SR_LEN) { 		        // path between us and dest is too long to be useful		        break;		}		rpath[roff] = v;		roff++;		if(v == net_id.addr)			break;	}	if(roff < 2 || d[v] == INFINITY || roff >= MAX_SR_LEN) {#ifdef DSR_CACHE_STATS		stat.route_find_count += 1;		if (for_me) stat.route_find_for_me += 1; 		stat.route_find_miss_count += 1;#endif		return false; // no path	}#ifdef GOD_STABILITY	/* The logic for God's omniscient expiration policy:	 *   if we expire in realtime, kill dead links immediately, and	 *     goto top. XXX this could create unnecessary overhead.	 *   if we expire on the fly, we set the timeout to an appropriate	 *     value, depending on whether or not the link is valid.	 */	if (lc_exppolicy == EXPPOLICY_GOD_OMNI) {		int last = net_id.addr;		for (v=1;v<roff;v++) {			Link *l = findLink(last, rpath[roff-v-1]);#ifdef REALTIME_EXPIRE			if (God::instance()->hops(last, rpath[roff-v-1])!=1) {				LIST_REMOVE(l, ln_link);				delete l;				dirty = CURRENT_TIME;				return findRoute(dest, route, for_me);			}#else			if (God::instance()->hops(last, rpath[roff-v-1])!=1) {				l->ln_timeout = CURRENT_TIME;			} else {				l->ln_timeout = MAX_SIMTIME;			}#endif

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -