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

📄 timerm.nc

📁 无线传感器网络中的节点定位算法。详见ReadMe文件。在TinyOS上实现的节点定位算法。
💻 NC
字号:
// $Id: TimerM.nc,v 1.1 2004/09/24 21:44:32 dcm Exp $/*									tab:4 * "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. *//* * * Authors:             Su Ping <sping@intel-research.net> * Revision:            $Id: TimerM.nc,v 1.1 2004/09/24 21:44:32 dcm Exp $ * This implementation assumes that DEFAULT_SCALE is 3. *//** * @author Su Ping <sping@intel-research.net> */module TimerM {    provides interface Timer[uint8_t id];    provides interface StdControl;    uses {	interface Leds;	interface Clock;	interface PowerManagement;    }}implementation {    uint32_t mState;		// each bit represent a timer state     uint8_t  setIntervalFlag;     uint8_t mScale, mInterval;    int8_t queue_head;    int8_t queue_tail;    uint8_t queue_size;    uint8_t queue[NUM_TIMERS];    volatile uint16_t interval_outstanding;    task void HandleFire();    struct timer_s {        uint8_t type;		// one-short or repeat timer        int32_t ticks;		// clock ticks for a repeat timer         int32_t ticksLeft;	// ticks left before the timer expires    } mTimerList[NUM_TIMERS];      enum {	maxTimerInterval = 230    };    command result_t StdControl.init() {        mState=0;        setIntervalFlag = 0;        queue_head = queue_tail = -1;        queue_size = 0;        mScale = 3;        mInterval = maxTimerInterval;        return call Clock.setRate(mInterval, mScale) ;    }    command result_t StdControl.start() {        return SUCCESS;    }    command result_t StdControl.stop() {        mState=0;        mInterval = maxTimerInterval;        setIntervalFlag = 0;        return SUCCESS;    }    command result_t Timer.start[uint8_t id](char type, 				   uint32_t interval) {	uint8_t diff;        if (id >= NUM_TIMERS) return FAIL;        if (type>1) return FAIL;        mTimerList[id].ticks = interval ;        mTimerList[id].type = type;	        atomic {            diff = call Clock.readCounter();            interval += diff;            mTimerList[id].ticksLeft = interval;            mState|=(0x1L<<id);            if (interval < mInterval) {                mInterval=interval;                call Clock.setInterval(mInterval);                setIntervalFlag = 0;                call PowerManagement.adjustPower();            }	}         return SUCCESS;    }    static void adjustInterval() {        uint8_t i, val = maxTimerInterval;        /* The following experimental section is to address the         * problem that some Timer0 interrupts are occasionally         * missed by the Atmega128.  My guess is that changing the         * Timer0 interval shortly before a rollover may sometimes         * cause this behavior.  This check may prevent such behavior,         * although it is just a guess. */        /* Begin experimental code by DCM */        if (mInterval >= 2 && call Clock.readCounter() >= (mInterval-2)) {            /* Adjust the interval later */            post HandleFire();            return;        }        /* End experimental code by DCM */        if ( mState) {            for (i=0;i<NUM_TIMERS;i++) {                if ((mState&(0x1L<<i)) && (mTimerList[i].ticksLeft <val )) {                    val = mTimerList[i].ticksLeft;		}            }            /* DCM: If the interval is set to be less than the current             * counter value, the timer will count an extra 256 ticks before             * hitting the interrupt.  Thus, we check for this condition             * and avoid it. */            i = call Clock.readCounter() + 5;            if (val < i)                val = i;            atomic {                mInterval =  val;                call Clock.setInterval(mInterval);                setIntervalFlag = 0;            }        } else {            atomic {              mInterval=maxTimerInterval;              call Clock.setInterval(mInterval);              setIntervalFlag = 0;            }        }	call PowerManagement.adjustPower();    }    command result_t Timer.stop[uint8_t id]() {        if (id>=NUM_TIMERS) return FAIL;        if (mState&(0x1L<<id)) { // if the timer is running 	    atomic mState &= ~(0x1L<<id);	    if (!mState) {	        setIntervalFlag = 1;	    } 	            return SUCCESS;        }        return FAIL; //timer not running    }    default event result_t Timer.fired[uint8_t id]() {        return SUCCESS;    }    void enqueue(uint8_t value) {      if (queue_tail == NUM_TIMERS - 1)	queue_tail = -1;      queue_tail++;      queue_size++;      queue[(uint8_t)queue_tail] = value;    }    uint8_t dequeue() {      if (queue_size == 0)        return NUM_TIMERS;      if (queue_head == NUM_TIMERS - 1)        queue_head = -1;      queue_head++;      queue_size--;      return queue[(uint8_t)queue_head];    }    task void signalOneTimer() {      uint8_t itimer = dequeue();      if (itimer < NUM_TIMERS)        signal Timer.fired[itimer]();    }    task void HandleFire() {        uint8_t i;         uint16_t int_out;	setIntervalFlag = 1;        /* DCM: read the number of ticks elapsed since the last firing         * was handled. */        atomic {            int_out = interval_outstanding;            interval_outstanding = 0;        }        if (mState) {            for (i=0;i<NUM_TIMERS;i++)  {                if (mState&(0x1L<<i)) {                    mTimerList[i].ticksLeft -= int_out;                     if (mTimerList[i].ticksLeft<=2) {                        /* DCM: only update the timer structure if the                         * signalOneTimer() task was able to be posted. */                        if (post signalOneTimer()) {                            if (mTimerList[i].type==TIMER_REPEAT) {                                mTimerList[i].ticksLeft += mTimerList[i].ticks;                            } else {// one shot timer                                 mState &=~(0x1L<<i);                             }                            enqueue(i);                        }                        else {                            /* DCM: wait another interval in hopes that                             * the task queue will clear out. */                            mTimerList[i].ticksLeft = mInterval;                        }                    }                }            }        }        /* DCM: don't bother adjusting the interval if another interrupt         * is hot on our tail. */        atomic int_out = interval_outstanding;        if (int_out == 0)            adjustInterval();    }    async event result_t Clock.fire() {        atomic {            /* DCM: Once we've posted HandleFire(), don't post it again until             * the original one is handled.  This prevents the task queue             * from getting flooded when mInterval is small. */            if (interval_outstanding == 0)                post HandleFire();            /* DCM: Keep track of the interval since the last interrupt */            interval_outstanding += call Clock.getInterval() + 1;        }        return SUCCESS;    }}

⌨️ 快捷键说明

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