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

📄 transformalarmcounterc.nc

📁 tinyos-2.0源代码!转载而已!要的尽管拿!
💻 NC
字号:
//$Id: TransformAlarmCounterC.nc,v 1.1.2.4 2006/01/30 21:31:27 idgay Exp $/* "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) 2005 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. *//** * TransformAlarmCounterC decreases precision and/or widens an Alarm/Counter * pair. This component has a reduced interrupt overhead compared to using * TransformAlarmC and TransformCounterC separately. However, it is not as  * useful for hardware timers with multiple compare registers. * * @param to_precision_tag A type indicating the precision of the transformed *   interfaces. * @param to_size_type The type for the width of the transformed interfaces. * @param from_precision_tag A type indicating the precision of the original *   interfaces. * @param from_size_type The type for the width of the original interfaces. * @param bit_shift_right Original time units will be 2 to the power  *   <code>bit_shift_right</code> larger than transformed time units. * @param upper_count_type A type large enough to store the upper bits -- *   those needed above from_size_type after its shift right to fill *   to_size_type. * * @author Cory Sharp <cssharp@eecs.berkeley.edu> * @author David Gay */generic module TransformAlarmCounterC(  typedef to_precision_tag,  typedef to_size_type @integer(),  typedef from_precision_tag,  typedef from_size_type @integer(),  uint8_t bit_shift_right,  typedef upper_count_type @integer() ){  provides interface Alarm<to_precision_tag,to_size_type> as Alarm;  provides interface Counter<to_precision_tag,to_size_type> as Counter;  uses interface Counter<from_precision_tag,from_size_type> as CounterFrom;  uses interface Alarm<from_precision_tag,from_size_type> as AlarmFrom;}implementation{  upper_count_type m_upper;  to_size_type m_t0;  to_size_type m_dt;  uint8_t m_skip_overflows;  enum  {    MAX_DELAY_LOG2 = 8 * sizeof(from_size_type) - 1 - bit_shift_right,    MAX_DELAY = ((to_size_type)1) << MAX_DELAY_LOG2,  };  enum  {    LOW_SHIFT_RIGHT = bit_shift_right,    HIGH_SHIFT_LEFT = 8*sizeof(from_size_type) - LOW_SHIFT_RIGHT,    NUM_UPPER_BITS = 8*sizeof(to_size_type) - 8*sizeof(from_size_type) + bit_shift_right,    // 1. hack to remove warning when NUM_UPPER_BITS == 8*sizeof(upper_count_type)    // 2. still provide warning if NUM_UPPER_BITS > 8*sizeof(upper_count_type)    // 3. and allow for the strange case of NUM_UPPER_BITS == 0    OVERFLOW_MASK = NUM_UPPER_BITS ? ((((upper_count_type)2) << (NUM_UPPER_BITS-1)) - 1) : 0,  };  void set_alarm();  async command to_size_type Counter.get()  {    to_size_type rv = 0;    atomic    {      upper_count_type high = m_upper;      from_size_type low = call CounterFrom.get();      if (call CounterFrom.isOverflowPending())      {	// If we signalled CounterFrom.overflow, that might trigger a	// Counter.overflow, which breaks atomicity.  The right thing to do	// increment a cached version of high without overflow signals.	// m_upper will be handled normally as soon as the out-most atomic	// block is left unless Clear.clearOverflow is called in the interim.	// This is all together the expected behavior.	high++;	low = call CounterFrom.get();      }      {	to_size_type high_to = high;	to_size_type low_to = low >> LOW_SHIFT_RIGHT;	rv = (high_to << HIGH_SHIFT_LEFT) | low_to;      }    }    return rv;  }  // isOverflowPending only makes sense when it's already part of a larger  // async block, so there's no async inside the command itself, where it  // wouldn't do anything useful.  async command bool Counter.isOverflowPending()  {    return ((m_upper & OVERFLOW_MASK) == OVERFLOW_MASK)	   && call CounterFrom.isOverflowPending();  }  // clearOverflow also only makes sense inside a larger atomic block, but we  // include the inner atomic block to ensure consistent internal state just in  // case someone calls it non-atomically.  async command void Counter.clearOverflow()  {    atomic    {      if (call Counter.isOverflowPending())      {	m_upper++;	call CounterFrom.clearOverflow();      }    }  }  async event void CounterFrom.overflow()  {    atomic    {      m_upper++;      if ((m_upper & OVERFLOW_MASK) == 0)	signal Counter.overflow();      if (m_skip_overflows && !--m_skip_overflows)	set_alarm();    }  }  async command to_size_type Alarm.getNow()  {    return call Counter.get();  }  async command to_size_type Alarm.getAlarm()  {    atomic return m_t0 + m_dt;  }  async command bool Alarm.isRunning()  {    return call AlarmFrom.isRunning();  }  async command void Alarm.stop()  {    call AlarmFrom.stop();  }  void set_alarm()  {    to_size_type now = call Counter.get(), expires, remaining;    /* m_t0 is assumed to be in the past. If it's > now, we assume       that time has wrapped around */    expires = m_t0 + m_dt;    /* The cast is necessary to get correct wrap-around arithmetic */    remaining = (to_size_type)(expires - now);    /* if (expires <= now) remaining = 0; in wrap-around arithmetic */    if (m_t0 <= now)      {	if (expires >= m_t0 && // if it wraps, it's > now	    expires <= now)	  remaining = 0;      }    else      {	if (expires >= m_t0 || // didn't wrap so < now	    expires <= now)	  remaining = 0;      }    if (remaining > MAX_DELAY * 2)      {	if (remaining >= MAX_DELAY * 2 * (to_size_type)256)	  m_skip_overflows = 255;	else	  m_skip_overflows = remaining / (MAX_DELAY * 2);	return;      }    else      m_skip_overflows = 0;    if (remaining > MAX_DELAY)      {	m_t0 = now + MAX_DELAY;	m_dt = remaining - MAX_DELAY;	remaining = MAX_DELAY;      }    else      {	m_t0 += m_dt;	m_dt = 0;      }    call AlarmFrom.startAt((from_size_type)now << bit_shift_right,			   (from_size_type)remaining << bit_shift_right);  }  async command void Alarm.startAt(to_size_type t0, to_size_type dt)  {    atomic    {      m_t0 = t0;      m_dt = dt;      set_alarm();    }  }  async command void Alarm.start(to_size_type dt)  {    call Alarm.startAt(call Alarm.getNow(), dt);  }  async event void AlarmFrom.fired()  {    atomic    {      if (m_dt == 0)      {	signal Alarm.fired();      }      else      {	set_alarm();      }    }  }  default async event void Alarm.fired()  {  }  default async event void Counter.overflow()  {  }}

⌨️ 快捷键说明

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