00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038 #include "TimerThread.hpp"
00039 #include "PeriodicActivity.hpp"
00040
00041 #include "rtt-config.h"
00042
00043 #include "../Time.hpp"
00044 #include "../Logger.hpp"
00045 #include <algorithm>
00046 #include "../os/MutexLock.hpp"
00047
00048 namespace RTT {
00049 using namespace extras;
00050 using namespace base;
00051 using os::MutexLock;
00052 using namespace std;
00053
00054 TimerThread::TimerThreadList TimerThread::TimerThreads;
00055
00056 TimerThreadPtr TimerThread::Instance(int pri, double per)
00057 {
00058 return Instance(ORO_SCHED_RT, pri, per);
00059 }
00060
00061 TimerThreadPtr TimerThread::Instance(int scheduler, int pri, double per)
00062 {
00063 return Instance(scheduler, pri, per, ~0);
00064 }
00065
00066 TimerThreadPtr TimerThread::Instance(int scheduler, int pri, double per, unsigned cpu_affinity)
00067 {
00068
00069
00070 os::CheckPriority(scheduler, pri);
00071 TimerThreadList::iterator it = TimerThreads.begin();
00072 while ( it != TimerThreads.end() ) {
00073 TimerThreadPtr tptr = it->lock();
00074
00075 if ( !tptr ) {
00076 TimerThreads.erase(it);
00077 it = TimerThreads.begin();
00078 continue;
00079 }
00080 if ( tptr->getScheduler() == scheduler && tptr->getPriority() == pri && tptr->getPeriodNS() == Seconds_to_nsecs(per) ) {
00081 return tptr;
00082 }
00083 ++it;
00084 }
00085 TimerThreadPtr ret( new TimerThread(scheduler, pri, "TimerThreadInstance", per, cpu_affinity) );
00086 TimerThreads.push_back( ret );
00087 return ret;
00088 }
00089
00090 TimerThread::TimerThread(int priority, const std::string& name, double periodicity, unsigned cpu_affinity)
00091 : Thread( ORO_SCHED_RT, priority, periodicity, cpu_affinity, name), cleanup(false)
00092 {
00093 tasks.reserve(MAX_ACTIVITIES);
00094 }
00095
00096 TimerThread::TimerThread(int scheduler, int priority, const std::string& name, double periodicity, unsigned cpu_affinity)
00097 : Thread(scheduler, priority, periodicity, cpu_affinity, name), cleanup(false)
00098 {
00099 tasks.reserve(MAX_ACTIVITIES);
00100 }
00101
00102 TimerThread::~TimerThread()
00103 {
00104
00105 this->stop();
00106 }
00107
00108 bool TimerThread::addActivity( PeriodicActivity* t ) {
00109 MutexLock lock(mutex);
00110 if ( tasks.size() == MAX_ACTIVITIES ) {
00111
00112 return false;
00113 }
00114 tasks.push_back( t );
00115
00116 return true;
00117 }
00118
00119 bool TimerThread::removeActivity( PeriodicActivity* t ) {
00120 MutexLock lock(mutex);
00121 ActivityList::iterator it = find(tasks.begin(), tasks.end(), t);
00122 if ( it != tasks.end() ) {
00123 *it = 0;
00124 cleanup = true;
00125 return true;
00126 }
00127
00128 return false;
00129 }
00130
00131 bool TimerThread::initialize() {
00132 return true;
00133 }
00134
00135 void TimerThread::finalize() {
00136 MutexLock lock(mutex);
00137
00138 for( ActivityList::iterator t_iter = tasks.begin(); t_iter != tasks.end(); ++t_iter)
00139 if ( *t_iter )
00140 (*t_iter)->stop();
00141 if ( cleanup )
00142 this->reorderList();
00143 }
00144
00145 void TimerThread::step() {
00146 MutexLock lock(mutex);
00147
00148
00149
00150 for( ActivityList::iterator t_iter = tasks.begin(); t_iter != tasks.end(); ++t_iter)
00151 if ( *t_iter )
00152 (*t_iter)->step();
00153
00154 if ( cleanup )
00155 this->reorderList();
00156 }
00157
00158 void TimerThread::reorderList() {
00159
00160 ActivityList::iterator begin = tasks.begin();
00161
00162 PeriodicActivity* nullActivity = 0;
00163 ActivityList::iterator it = tasks.begin();
00164
00165 it = find( tasks.begin(), tasks.end(), nullActivity);
00166 begin = it+1;
00167 while ( it != tasks.end() ) {
00168
00169 while ( begin != tasks.end() && *begin == 0 )
00170 ++begin;
00171 if ( begin == tasks.end() ) {
00172
00173 tasks.resize( it - tasks.begin() );
00174
00175 break;
00176 }
00177
00178 ActivityList::iterator end = find ( begin, tasks.end(), nullActivity);
00179
00180
00181 while ( begin != end ) {
00182 *it = *begin;
00183 ++begin;
00184 ++it;
00185 }
00186 }
00187
00188 cleanup = false;
00189 }
00190
00191
00192 }