/////////////////////////////////////////////////////////////////////////////
// queue.cc
//
// SIMLIB version: 2.18
// Date: 2004-01-25
//
// Copyright (c) 1991-2004 Petr Peringer 
//
// This library is licensed under GNU Library GPL. See the file COPYING.
//

//
//  implementation of queue
//

#include "simlib.h"
#include "internal.h"


////////////////////////////////////////////////////////////////////////////
//  implementation
//

SIMLIB_IMPLEMENTATION


////////////////////////////////////////////////////////////////////////////
//  constructors
//
Queue::Queue()
{
  dprintf(("Queue{%p}::Queue()", this));
}

Queue::Queue(const char *_name) 
{
  dprintf(("Queue{%p}::Queue(\"%s\")", this, _name));
  SetName(_name);
}

////////////////////////////////////////////////////////////////////////////
//  destructor
//
Queue::~Queue() {
  dprintf(("Queue{%p}::~Queue() // \"%s\" ", this, Name()));
}

////////////////////////////////////////////////////////////////////////////
// Insert --- priority insert into queue
//
void Queue::Insert(Entity *ent)
{
  dprintf(("%s::Insert(%s)", Name(), ent->Name() ));
  Entity::Priority_t prio = ent->Priority;
  // find (higher priority is first)
#if 0 // _INS_FROM_BEGIN
  Queue::iterator p = begin();
  for( ; p!=end() && ((Entity*)(*p))->Priority >= prio; ++p);
#else
  // this is faster (items are inserted at end usually)
  Queue::iterator p = end(); 
  while(p!=begin()) {
        Queue::iterator q = p;
        --p;
        if( ((Entity*)(*p))->Priority >= prio ) { p = q; break; }
  }
#endif
  PredIns(ent,p); // works for end()
}

////////////////////////////////////////////////////////////////////////////
// InsFirst --- insert at first position (special case)
//
void Queue::InsFirst(Entity *ent)
{
  dprintf(("%s::InsFirst(%s)", Name(), ent->Name() ));
  PredIns(ent,begin());
}

////////////////////////////////////////////////////////////////////////////
// InsLast --- insert at last position (for FIFO)
//
void Queue::InsLast(Entity *ent)
{
  dprintf(("%s::InsLast(%s)", Name(), ent->Name() ));
  PredIns(ent,end());
}

////////////////////////////////////////////////////////////////////////////
// PostIns --- insert after entity
//
void Queue::PostIns(Entity *ent, iterator pos)
{
  dprintf(("%s::PostIns(%s,pos)", Name(), ent->Name(), *pos ));
  if(pos==end())
     SIMLIB_internal_error(); // add error message 
  PredIns(ent, ++pos);
}

////////////////////////////////////////////////////////////////////////////
// PredIns --- insert before entity
//
void Queue::PredIns(Entity *ent, iterator pos)
{
  dprintf(("%s::PredIns(%s,pos:%p)", Name(), ent->Name(), *pos ));
  List::PredIns(ent, *pos); // insert before pos, can be end()
  ent->_MarkTime = Time;    // marks input time
  StatN(size());            // length statistic
}

////////////////////////////////////////////////////////////////////////////
// GetFirst --- remove first entity from queue
//
Entity *Queue::GetFirst()
{
  dprintf(("%s::GetFirst()", Name()));
  Entity *ent = Get(begin());
  return ent;
}

////////////////////////////////////////////////////////////////////////////
// GetLast --- remove last entity from queue
//
Entity *Queue::GetLast()
{
  dprintf(("%s::GetLast()", Name()));
  Entity *ent = Get(--end());
  return ent;
}

////////////////////////////////////////////////////////////////////////////
// Get --- remove item
//
Entity *Queue::Get(iterator pos)
{
  dprintf(("%s::Get(pos:%p)", Name(), *pos));
  Entity *ent = (Entity*) List::Get(*pos);
  StatDT(double(Time) - ent->_MarkTime);
  StatN(size());  StatN.n--; // correction !!!
  return ent;
}

////////////////////////////////////////////////////////////////////////////
//  clear - initialization of list
//
void Queue::clear()
{
  dprintf(("%s::Clear()", Name()));
  StatN.Clear();
  StatDT.Clear();
  List::clear(); // problem with WARNING
  StatN.Clear();
  StatDT.Clear();
}

////////////////////////////////////////////////////////////////////////////
//  Process::Name --- name of the process
//
const char *Queue::Name()
{
    const char *name = SimObject::Name();
    if(*name) return name; // has explicit name
    else      return SIMLIB_create_name("Queue{%p}", this);
}

// end



syntax highlighted by Code2HTML, v. 0.9.1