/* * Copyright (c) 2016, The OpenThread Authors. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the copyright holder nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. *//* ... *//** * @file * This file implements a multiplexed timer service on top of the alarm abstraction. *//* ... */#include"timer.hpp"#include"instance/instance.hpp"namespaceot{//---------------------------------------------------------------------------------------------------------------------// `NextFireTime`NextFireTime::NextFireTime(void):NextFireTime(TimerMilli::GetNow()){}{ ... }NextFireTime::NextFireTime(TimeaNow):mNow(aNow),mNextTime(aNow.GetDistantFuture()){}{ ... }voidNextFireTime::UpdateIfEarlier(TimeaTime){mNextTime=Min(mNextTime,Max(mNow,aTime));}//---------------------------------------------------------------------------------------------------------------------// `Timer`constTimer::Scheduler::AlarmApiTimerMilli::Scheduler::sAlarmMilliApi={&otPlatAlarmMilliStartAt,&otPlatAlarmMilliStop,&otPlatAlarmMilliGetNow,}{...};boolTimer::DoesFireBefore(constTimer&aSecondTimer,TimeaNow)const{// Indicates whether the fire time of this timer is strictly// before the fire time of a second given timer.boolretval;boolisBeforeNow=(GetFireTime()<aNow);// Check if one timer is before `now` and the other one is not.if((aSecondTimer.GetFireTime()<aNow)!=isBeforeNow){// One timer is before `now` and the other one is not, so if// this timer's fire time is before `now` then the second fire// time would be after `now` and this timer would fire before// the second timer.retval=isBeforeNow;}{...}else{// Both timers are before `now` or both are after `now`. Either// way the difference is guaranteed to be less than `kMaxDt` so// we can safely compare the fire times directly.retval=GetFireTime()<aSecondTimer.GetFireTime();}{...}returnretval;}{...}//---------------------------------------------------------------------------------------------------------------------// `TimerMilli`voidTimerMilli::Start(uint32_taDelay){StartAt(GetNow(),aDelay);}voidTimerMilli::StartAt(TimeMilliaStartTime,uint32_taDelay){OT_ASSERT(aDelay<=kMaxDelay);FireAt(aStartTime+aDelay);}{ ... }voidTimerMilli::FireAt(TimeMilliaFireTime){mFireTime=aFireTime;Get<Scheduler>().Add(*this);}{ ... }voidTimerMilli::FireAt(constNextFireTime&aNextFireTime){if(aNextFireTime.IsSet()){FireAt(aNextFireTime.GetNextTime());}{...}else{Stop();}{...}}{ ... }voidTimerMilli::FireAtIfEarlier(TimeMilliaFireTime){if(!IsRunning()||(mFireTime>aFireTime)){FireAt(aFireTime);}{...}}{ ... }voidTimerMilli::FireAtIfEarlier(constNextFireTime&aNextFireTime){if(aNextFireTime.IsSet()){FireAtIfEarlier(aNextFireTime.GetNextTime());}{...}}{ ... }voidTimerMilli::Stop(void){Get<Scheduler>().Remove(*this);}voidTimerMilli::RemoveAll(Instance&aInstance){aInstance.Get<Scheduler>().RemoveAll();}//---------------------------------------------------------------------------------------------------------------------// `Timer::Scheduler`voidTimer::Scheduler::Add(Timer&aTimer,constAlarmApi&aAlarmApi){Timer*prev=nullptr;Timenow(aAlarmApi.AlarmGetNow());Remove(aTimer,aAlarmApi);for(Timer&cur:mTimerList){if(aTimer.DoesFireBefore(cur,now)){break;}{...}prev=&cur;}{...}if(prev==nullptr){mTimerList.Push(aTimer);SetAlarm(aAlarmApi);}{...}else{mTimerList.PushAfter(aTimer,*prev);}{...}}{ ... }voidTimer::Scheduler::Remove(Timer&aTimer,constAlarmApi&aAlarmApi){VerifyOrExit(aTimer.IsRunning());if(mTimerList.GetHead()==&aTimer){mTimerList.Pop();SetAlarm(aAlarmApi);}{...}else{IgnoreError(mTimerList.Remove(aTimer));}{...}aTimer.SetNext(&aTimer);exit:return;}{ ... }voidTimer::Scheduler::SetAlarm(constAlarmApi&aAlarmApi){if(mTimerList.IsEmpty()){aAlarmApi.AlarmStop(&GetInstance());}{...}else{Timer*timer=mTimerList.GetHead();Timenow(aAlarmApi.AlarmGetNow());uint32_tremaining;remaining=(now<timer->mFireTime)?(timer->mFireTime-now):0;aAlarmApi.AlarmStartAt(&GetInstance(),now.GetValue(),remaining);}{...}}{ ... }voidTimer::Scheduler::ProcessTimers(constAlarmApi&aAlarmApi){Timer*timer=mTimerList.GetHead();if(timer){Timenow(aAlarmApi.AlarmGetNow());if(now>=timer->mFireTime){Remove(*timer,aAlarmApi);// `Remove()` will `SetAlarm` for next timer if there is any.timer->Fired();ExitNow();}{...}}{...}SetAlarm(aAlarmApi);exit:return;}{ ... }voidTimer::Scheduler::RemoveAll(constAlarmApi&aAlarmApi){Timer*timer;while((timer=mTimerList.Pop())!=nullptr){timer->SetNext(timer);}{...}SetAlarm(aAlarmApi);}{ ... }extern"C"voidotPlatAlarmMilliFired(otInstance*aInstance){VerifyOrExit(otInstanceIsInitialized(aInstance));AsCoreType(aInstance).Get<TimerMilli::Scheduler>().ProcessTimers();exit:return;}{ ... }//---------------------------------------------------------------------------------------------------------------------// `TimerMicro`#ifOPENTHREAD_CONFIG_PLATFORM_USEC_TIMER_ENABLEconstTimer::Scheduler::AlarmApiTimerMicro::Scheduler::sAlarmMicroApi={&otPlatAlarmMicroStartAt,&otPlatAlarmMicroStop,&otPlatAlarmMicroGetNow,}{...};voidTimerMicro::Start(uint32_taDelay){StartAt(GetNow(),aDelay);}voidTimerMicro::StartAt(TimeMicroaStartTime,uint32_taDelay){OT_ASSERT(aDelay<=kMaxDelay);FireAt(aStartTime+aDelay);}{ ... }voidTimerMicro::FireAt(TimeMicroaFireTime){mFireTime=aFireTime;Get<Scheduler>().Add(*this);}{ ... }voidTimerMicro::Stop(void){Get<Scheduler>().Remove(*this);}voidTimerMicro::RemoveAll(Instance&aInstance){aInstance.Get<Scheduler>().RemoveAll();}extern"C"voidotPlatAlarmMicroFired(otInstance*aInstance){VerifyOrExit(otInstanceIsInitialized(aInstance));AsCoreType(aInstance).Get<TimerMicro::Scheduler>().ProcessTimers();exit:return;}{ ... }#endif/* ... */// OPENTHREAD_CONFIG_PLATFORM_USEC_TIMER_ENABLE}{...}// namespace ot
Details
Show: from
Types: Columns:
All items filtered out
All items filtered out
This file uses the notable symbols shown below. Click anywhere in the file to view more details.