before: active events were not properly deleted
(in both cases, ev == 0 and ev != 0)
after: deficiencies repaired; simpler code
side-effect: doesn't return whether the event was found anymore
(can be re-added at a later time)
git-svn-id: https://www4.informatik.uni-erlangen.de/i4svn/danceos/trunk/devel/fail@1024 8c4709b5-6ec9-48aa-a5cd-a96041d1645a
146 lines
3.7 KiB
C++
146 lines
3.7 KiB
C++
#include <set>
|
|
|
|
#include "EventList.hpp"
|
|
#include "../SAL/SALInst.hpp"
|
|
|
|
namespace fi
|
|
{
|
|
|
|
EventId EventList::add(BaseEvent* ev, ExperimentFlow* pExp)
|
|
{
|
|
assert(ev != NULL && "FATAL ERROR: Event (of base type BaseEvent*) cannot be NULL!");
|
|
// a zero counter does not make sense
|
|
assert(ev->getCounter() != 0);
|
|
ev->setParent(pExp); // event is linked to experiment flow
|
|
m_BufferList.push_back(ev);
|
|
return (ev->getId());
|
|
}
|
|
|
|
void EventList::remove(BaseEvent* ev)
|
|
{
|
|
// possible cases:
|
|
// - ev == 0 -> remove all events
|
|
// * clear m_BufferList
|
|
// * copy m_FireList to m_DeleteList
|
|
if (ev == 0) {
|
|
m_BufferList.clear();
|
|
// all remaining active events must not fire anymore
|
|
m_DeleteList.insert(m_DeleteList.end(), m_FireList.begin(), m_FireList.end());
|
|
|
|
// - ev != 0 -> remove single event
|
|
// * find/remove ev in m_BufferList
|
|
// * if ev in m_FireList, copy to m_DeleteList
|
|
} else {
|
|
m_BufferList.remove(ev);
|
|
firelist_t::const_iterator it =
|
|
std::find(m_FireList.begin(), m_FireList.end(), ev);
|
|
if (it != m_FireList.end()) {
|
|
m_DeleteList.push_back(ev);
|
|
}
|
|
}
|
|
}
|
|
|
|
EventList::iterator EventList::remove(iterator it)
|
|
{
|
|
return (m_remove(it, false));
|
|
}
|
|
|
|
EventList::iterator EventList::m_remove(iterator it, bool skip_deletelist)
|
|
{
|
|
if(!skip_deletelist)
|
|
m_DeleteList.push_back(*it);
|
|
return (m_BufferList.erase(it));
|
|
}
|
|
|
|
void EventList::remove(ExperimentFlow* flow)
|
|
{
|
|
// all events?
|
|
if (flow == 0) {
|
|
m_BufferList.clear();
|
|
} else {
|
|
for (bufferlist_t::iterator it = m_BufferList.begin();
|
|
it != m_BufferList.end(); ) {
|
|
if ((*it)->getParent() == flow) {
|
|
it = m_BufferList.erase(it);
|
|
} else {
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
// events that just fired / are about to fire ...
|
|
for (firelist_t::const_iterator it = m_FireList.begin();
|
|
it != m_FireList.end(); it++) {
|
|
if (std::find(m_DeleteList.begin(), m_DeleteList.end(), *it)
|
|
!= m_DeleteList.end()) {
|
|
continue;
|
|
}
|
|
// ... need to be pushed into m_DeleteList, as we're currently
|
|
// iterating over m_FireList in fireActiveEvents() and cannot modify it
|
|
if (flow == 0 || (*it)->getParent() == flow) {
|
|
m_DeleteList.push_back(*it);
|
|
}
|
|
}
|
|
}
|
|
|
|
EventList::~EventList()
|
|
{
|
|
// nothing to do here yet
|
|
}
|
|
|
|
BaseEvent* EventList::getEventFromId(EventId id)
|
|
{
|
|
// Loop through all events:
|
|
for(bufferlist_t::iterator it = m_BufferList.begin();
|
|
it != m_BufferList.end(); it++)
|
|
if((*it)->getId() == id)
|
|
return (*it);
|
|
return (NULL); // Nothing found.
|
|
}
|
|
|
|
EventList::iterator EventList::makeActive(iterator it)
|
|
{
|
|
assert(it != m_BufferList.end() &&
|
|
"FATAL ERROR: Iterator has already reached the end!");
|
|
BaseEvent* ev = *it;
|
|
assert(ev && "FATAL ERROR: Event object pointer cannot be NULL!");
|
|
ev->decreaseCounter();
|
|
if (ev->getCounter() > 0) {
|
|
return ++it;
|
|
}
|
|
ev->resetCounter();
|
|
// Note: This is the one and only situation in which remove() should NOT
|
|
// store the removed item in the delete-list.
|
|
iterator it_next = m_remove(it, true); // remove event from buffer-list
|
|
m_FireList.push_back(ev);
|
|
return (it_next);
|
|
}
|
|
|
|
void EventList::fireActiveEvents()
|
|
{
|
|
for(firelist_t::iterator it = m_FireList.begin();
|
|
it != m_FireList.end(); it++)
|
|
{
|
|
if(std::find(m_DeleteList.begin(), m_DeleteList.end(), *it)
|
|
== m_DeleteList.end()) // not found in delete-list?
|
|
{
|
|
m_pFired = *it;
|
|
ExperimentFlow* pFlow = m_pFired->getParent();
|
|
assert(pFlow && "FATAL ERROR: The event has no parent experiment (owner)!");
|
|
sal::simulator.m_Flows.toggle(pFlow);
|
|
}
|
|
}
|
|
m_FireList.clear();
|
|
m_DeleteList.clear();
|
|
}
|
|
|
|
size_t EventList::getContextCount() const
|
|
{
|
|
set<ExperimentFlow*> uniqueFlows; // count unique ExperimentFlow-ptr
|
|
for(bufferlist_t::const_iterator it = m_BufferList.begin();
|
|
it != m_BufferList.end(); it++)
|
|
uniqueFlows.insert((*it)->getParent());
|
|
return (uniqueFlows.size());
|
|
}
|
|
|
|
} // end-of-namespace: fi
|