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

📄 lqiroutingenginep.nc

📁 tinyos2.0版本驱动
💻 NC
字号:
/* * "Copyright (c) 2000-2003 The Regents of the University  of California.   * All rights reserved. * * Permission to use, copy, modify, and distribute this software and its * documentation for any purpose, without fee, and without written agreement is * hereby granted, provided that the above copyright notice, the following * two paragraphs and the author appear in all copies of this software. *  * IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY FOR * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES ARISING OUT * OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF * CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *  * THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY * AND FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS * ON AN "AS IS" BASIS, AND THE UNIVERSITY OF CALIFORNIA HAS NO OBLIGATION TO * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS." * * Copyright (c) 2002-2003 Intel Corporation * All rights reserved. * * This file is distributed under the terms in the attached INTEL-LICENSE      * file. If you do not find these files, copies can be found by writing to * Intel Research Berkeley, 2150 Shattuck Avenue, Suite 1300, Berkeley, CA,  * 94704.  Attention:  Intel License Inquiry. *//** * * @author Gilman Tolle * @author Philip Levis (port to TinyOS 2.x) */#include "MultiHopLqi.h"#include "CollectionDebugMsg.h"module LqiRoutingEngineP {  provides {    interface Init;    interface StdControl;    interface RouteSelect;    interface RouteControl;    interface RootControl;  }  uses {    interface Timer<TMilli>;    interface AMSend;    interface Receive;    interface Random;    interface Packet;    interface AMPacket;    interface LqiRouteStats;    interface CC2420Packet;    interface Leds;    interface CollectionDebug;  }}implementation {  enum {    BASE_STATION_ADDRESS = 0,    BEACON_PERIOD        = 32,    BEACON_TIMEOUT       = 8,  };  enum {    ROUTE_INVALID    = 0xff  };  bool isRoot = FALSE;    message_t msgBuf;  bool msgBufBusy;  uint16_t gbCurrentParent;  uint16_t gbCurrentParentCost;  uint16_t gbCurrentLinkEst;  uint8_t  gbCurrentHopCount;  uint16_t gbCurrentCost;  uint8_t gLastHeard;  int16_t gCurrentSeqNo;  int16_t gOriginSeqNo;    uint16_t gUpdateInterval;  uint8_t gRecentIndex;  uint16_t gRecentPacketSender[MHOP_HISTORY_SIZE];  int16_t gRecentPacketSeqNo[MHOP_HISTORY_SIZE];  uint8_t gRecentOriginIndex;  uint16_t gRecentOriginPacketSender[MHOP_HISTORY_SIZE];  int16_t gRecentOriginPacketSeqNo[MHOP_HISTORY_SIZE];  uint16_t adjustLQI(uint8_t val) {    uint16_t result = (80 - (val - 50));    result = (((result * result) >> 3) * result) >> 3;    return result;  }  lqi_header_t* getHeader(message_t* msg) {    return (lqi_header_t*)call Packet.getPayload(msg, sizeof(lqi_header_t));  }    lqi_beacon_msg_t* getBeacon(message_t* msg) {    return (lqi_beacon_msg_t*)call Packet.getPayload(msg, sizeof(lqi_beacon_msg_t));  }  task void SendRouteTask() {    lqi_beacon_msg_t* bMsg = getBeacon(&msgBuf);    uint8_t length = sizeof(lqi_beacon_msg_t);        dbg("LQI","MultiHopRSSI Sending route update msg.\n");    if (gbCurrentParent != TOS_BCAST_ADDR) {      dbg("LQI","MultiHopRSSI: Parent = %d\n", gbCurrentParent);    }        if (msgBufBusy) {      post SendRouteTask();      return;    }    dbg("LQI","MultiHopRSSI: Current cost: %d.\n", 	gbCurrentParentCost + gbCurrentLinkEst);    if (isRoot) {      bMsg->parent = TOS_NODE_ID;      bMsg->cost = 0;      bMsg->originaddr = TOS_NODE_ID;      bMsg->hopcount = 0;      bMsg->seqno = gCurrentSeqNo++;    }    else {      bMsg->parent = gbCurrentParent;      bMsg->cost = gbCurrentParentCost + gbCurrentLinkEst;      bMsg->originaddr = TOS_NODE_ID;      bMsg->hopcount = gbCurrentHopCount;      bMsg->seqno = gCurrentSeqNo++;    }        if (call AMSend.send(TOS_BCAST_ADDR, &msgBuf, length) == SUCCESS) {      msgBufBusy = TRUE;      call CollectionDebug.logEventRoute(NET_C_TREE_SENT_BEACON, bMsg->parent, 0, bMsg->cost);    }  }  task void TimerTask() {    uint8_t val;    val = ++gLastHeard;    if (!isRoot && (val > BEACON_TIMEOUT)) {      gbCurrentParent = TOS_BCAST_ADDR;      gbCurrentParentCost = 0x7fff;      gbCurrentLinkEst = 0x7fff;      gbCurrentHopCount = ROUTE_INVALID;      gbCurrentCost = 0xfffe;    }    post SendRouteTask();  }  command error_t Init.init() {    int n;    gRecentIndex = 0;    for (n = 0; n < MHOP_HISTORY_SIZE; n++) {      gRecentPacketSender[n] = TOS_BCAST_ADDR;      gRecentPacketSeqNo[n] = 0;    }    gRecentOriginIndex = 0;    for (n = 0; n < MHOP_HISTORY_SIZE; n++) {      gRecentOriginPacketSender[n] = TOS_BCAST_ADDR;      gRecentOriginPacketSeqNo[n] = 0;    }    gbCurrentParent = TOS_BCAST_ADDR;    gbCurrentParentCost = 0x7fff;    gbCurrentLinkEst = 0x7fff;    gbCurrentHopCount = ROUTE_INVALID;    gbCurrentCost = 0xfffe;    gOriginSeqNo = 0;    gCurrentSeqNo = 0;    gUpdateInterval = BEACON_PERIOD;    msgBufBusy = FALSE;    return SUCCESS;  }  command error_t RootControl.setRoot() {    call Leds.led2On();    call CollectionDebug.logEventRoute(NET_C_TREE_NEW_PARENT, TOS_NODE_ID, 0, 0);    isRoot = TRUE;    return SUCCESS;  }  command error_t RootControl.unsetRoot() {    isRoot = FALSE;    return SUCCESS;  }  command bool RootControl.isRoot() {    return isRoot;  }    command error_t StdControl.start() {    gLastHeard = 0;    call Timer.startOneShot(call Random.rand32() % (1024 * gUpdateInterval));    return SUCCESS;  }    command error_t StdControl.stop() {    call Timer.stop();    return SUCCESS;  }  command bool RouteSelect.isActive() {    return TRUE;  }  command error_t RouteSelect.selectRoute(message_t* msg, uint8_t resend) {    int i;    lqi_header_t* hdr = getHeader(msg);    if (isRoot) {      return FAIL;    }    if (hdr->originaddr != TOS_NODE_ID && resend == 0) {      // supress duplicate packets      for (i = 0; i < MHOP_HISTORY_SIZE; i++) {        if ((gRecentPacketSender[i] == call AMPacket.source(msg)) &&            (gRecentPacketSeqNo[i] == hdr->seqno)) {	  call CollectionDebug.logEvent(NET_C_FE_DUPLICATE_CACHE_AT_SEND);	  dbg("LQI", "%s no route as this is a duplicate!\n", __FUNCTION__);          return FAIL;        }      }          gRecentPacketSender[gRecentIndex] = call AMPacket.source(msg);      gRecentPacketSeqNo[gRecentIndex] = hdr->seqno;      gRecentIndex = (gRecentIndex + 1) % MHOP_HISTORY_SIZE;      // supress multihop cycles and try to break out of it      for (i = 0; i < MHOP_HISTORY_SIZE; i++) {        if ((gRecentOriginPacketSender[i] == hdr->originaddr) &&            (gRecentOriginPacketSeqNo[i] == hdr->originseqno)) {          gbCurrentParentCost = 0x7fff;          gbCurrentLinkEst = 0x7fff;          gbCurrentParent = TOS_BCAST_ADDR;          gbCurrentHopCount = ROUTE_INVALID;	  dbg("LQI", "%s no route as we are in a cycle!\n", __FUNCTION__);          return FAIL;        }      }      gRecentOriginPacketSender[gRecentOriginIndex] = hdr->originaddr;      gRecentOriginPacketSeqNo[gRecentOriginIndex] = hdr->originseqno;      gRecentOriginIndex = (gRecentOriginIndex + 1) % MHOP_HISTORY_SIZE;    }    if (resend == 0) {      hdr->seqno = gCurrentSeqNo++;    }        dbg("LQI", "LQI setting destination to %hu and link quality ?\n", gbCurrentParent);    call AMPacket.setDestination(msg, gbCurrentParent);    return SUCCESS;  }  command error_t RouteSelect.initializeFields(message_t* msg) {    lqi_header_t* header = getHeader(msg);    header->originaddr = TOS_NODE_ID;    header->originseqno = gOriginSeqNo++;    header->seqno = gCurrentSeqNo;        if (isRoot) {      header->hopcount = 0;    }    else {      header->hopcount = gbCurrentHopCount;    }    dbg("LQI", "LQI setting hopcount to %hhu\n", gbCurrentHopCount);    return SUCCESS;  }  command uint8_t* RouteSelect.getBuffer(message_t* Msg, uint16_t* Len) {  }  command uint16_t RouteControl.getParent() {    return gbCurrentParent;  }  command uint8_t RouteControl.getQuality() {    return gbCurrentLinkEst;  }  command uint8_t RouteControl.getDepth() {    return gbCurrentHopCount;  }  command uint8_t RouteControl.getOccupancy() {    return 0;  }  command error_t RouteControl.setUpdateInterval(uint16_t Interval) {    gUpdateInterval = Interval;    return SUCCESS;  }  command error_t RouteControl.manualUpdate() {    post SendRouteTask();    return SUCCESS;  }  event void Timer.fired() {    call Leds.led0Toggle();    post TimerTask();    call Timer.startOneShot((uint32_t)1024 * gUpdateInterval + 1);  }  event message_t* Receive.receive(message_t* msg, void* payload, uint8_t len) {    lqi_beacon_msg_t* bMsg = (lqi_beacon_msg_t*)payload;    am_addr_t source = call AMPacket.source(msg);    uint8_t lqi = call CC2420Packet.getLqi(msg);        call CollectionDebug.logEventRoute(NET_C_TREE_RCV_BEACON, source, 0, bMsg->cost);        if (isRoot) {      return msg;    }    else {      dbg("LQI,LQIRoute", "LQI receiving routing beacon from %hu with LQI %hhu that advertises %hu.\n", source, lqi, bMsg->cost);      if (source == gbCurrentParent) {	// try to prevent cycles	if (bMsg->parent != TOS_NODE_ID) {	  gLastHeard = 0;	  gbCurrentParentCost = bMsg->cost;	  gbCurrentLinkEst = adjustLQI(lqi);	  gbCurrentHopCount = bMsg->hopcount + 1;	  dbg("LQI,LQIRoute", "  -- Not a loop\n");	}	else {	  gLastHeard = 0;	  gbCurrentParentCost = 0x7fff;	  gbCurrentLinkEst = 0x7fff;	  gbCurrentParent = TOS_BCAST_ADDR;	  gbCurrentHopCount = ROUTE_INVALID;	  dbg("LQI,LQIRoute", "  -- Detected a loop\n");	}	      } else {		/* if the message is not from my parent, 	   compare the message's cost + link estimate to my current cost,	   switch if necessary */		// make sure you don't pick a parent that creates a cycle	if (((uint32_t) bMsg->cost + (uint32_t) adjustLQI(lqi) 	     <	     ((uint32_t) gbCurrentParentCost + (uint32_t) gbCurrentLinkEst) -	     (((uint32_t) gbCurrentParentCost + (uint32_t) gbCurrentLinkEst) >> 2)	     ) &&	    (bMsg->parent != TOS_NODE_ID)) {	  gLastHeard = 0;	  gbCurrentParent = call AMPacket.source(msg);	  gbCurrentParentCost = bMsg->cost;	  gbCurrentLinkEst = adjustLQI(lqi);		  gbCurrentHopCount = bMsg->hopcount + 1;	  call CollectionDebug.logEventRoute(NET_C_TREE_NEW_PARENT, gbCurrentParent, 0, gbCurrentParentCost + gbCurrentLinkEst);	  dbg("LQI,LQIRoute", "  -- Not a cycle.\n");	}	else {	  dbg("LQI,LQIRoute", "  -- CYCLE.\n");	}      }    }    dbg("LQI,LQIRoute", "Set my count to %hhu, my link to %hu and my cost to %hu.\n", gbCurrentHopCount, gbCurrentLinkEst, gbCurrentParentCost);    return msg;  }  event void AMSend.sendDone(message_t* msg, error_t success) {    msgBufBusy = FALSE;  }    /* Default implementations for CollectionDebug calls.     * These allow CollectionDebug not to be wired to anything if debugging     * is not desired. */  default command error_t CollectionDebug.logEvent(uint8_t type) {    return SUCCESS;  }  default command error_t CollectionDebug.logEventSimple(uint8_t type, uint16_t arg) {    return SUCCESS;  }  default command error_t CollectionDebug.logEventDbg(uint8_t type, uint16_t arg1, uint16_t arg2, uint16_t arg3) {    return SUCCESS;  }  default command error_t CollectionDebug.logEventMsg(uint8_t type, uint16_t msg, am_addr_t origin, am_addr_t node) {    return SUCCESS;  }  default command error_t CollectionDebug.logEventRoute(uint8_t type, am_addr_t parent, uint8_t hopcount, uint16_t etx) {    return SUCCESS;  }  }  

⌨️ 快捷键说明

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