Files
epics-base/src/cas/generic/casEventSys.cc
1999-09-15 00:05:17 +00:00

271 lines
5.2 KiB
C++

/*
* $Id$
*
* Author Jeffrey O. Hill
* johill@lanl.gov
* 505 665 1831
*
* Experimental Physics and Industrial Control System (EPICS)
*
* Copyright 1991, the Regents of the University of California,
* and the University of Chicago Board of Governors.
*
* This software was produced under U.S. Government contracts:
* (W-7405-ENG-36) at the Los Alamos National Laboratory,
* and (W-31-109-ENG-38) at Argonne National Laboratory.
*
* Initial development by:
* The Controls and Automation Group (AT-8)
* Ground Test Accelerator
* Accelerator Technology Division
* Los Alamos National Laboratory
*
* Co-developed with
* The Controls and Computing Group
* Accelerator Systems Division
* Advanced Photon Source
* Argonne National Laboratory
*
*/
/*
* ANSI C
*/
#include <string.h>
/*
* EPICS
*/
#include "server.h"
#include "casEventSysIL.h" // casMonitor inline func
//
// casEventSys::show()
//
void casEventSys::show(unsigned level) const
{
printf ("casEventSys at %p\n", this);
if (level>=1u) {
printf ("\tnumEventBlocks = %u, maxLogEntries = %u\n",
this->numEventBlocks, this->maxLogEntries);
printf ("\tthere are %d events in the queue\n",
this->eventLogQue.count());
printf ("Replace events flag = %d, dontProcess flag = %d\n",
this->replaceEvents, this->dontProcess);
}
}
//
// casEventSys::~casEventSys()
//
casEventSys::~casEventSys()
{
casEvent *pE;
this->mutex.lock();
if (this->pPurgeEvent != NULL) {
this->eventLogQue.remove(*this->pPurgeEvent);
delete this->pPurgeEvent;
}
/*
* all active event blocks must be canceled first
*/
casVerify (this->numEventBlocks==0);
while ( (pE = this->eventLogQue.get()) ) {
delete pE;
}
this->mutex.unlock();
}
//
// casEventSys::installMonitor()
//
void casEventSys::installMonitor()
{
this->mutex.lock();
this->numEventBlocks++;
this->maxLogEntries += averageEventEntries;
this->mutex.unlock();
}
//
// casEventSys::removeMonitor()
//
void casEventSys::removeMonitor()
{
this->mutex.lock();
assert (this->numEventBlocks>=1u);
this->numEventBlocks--;
this->maxLogEntries -= averageEventEntries;
this->mutex.unlock();
}
//
// casEventSys::process()
//
casProcCond casEventSys::process()
{
casEvent *pEvent;
caStatus status;
casProcCond cond = casProcOk;
unsigned long nAccepted = 0u;
this->mutex.lock();
while (!this->dontProcess) {
pEvent = this->eventLogQue.get();
if (pEvent==NULL) {
break;
}
//
// lock must remain on until the event queue
// event is called
//
status = pEvent->cbFunc(*this);
if (status==S_cas_success) {
/*
* only remove it after it was accepted by the
* client
*/
nAccepted++;
}
else if (status==S_cas_sendBlocked) {
/*
* not accepted so return to the head of the list
* (we will try again later)
*/
this->pushOnToEventQueue(*pEvent);
cond = casProcOk;
break;
}
else if (status==S_cas_disconnect) {
cond = casProcDisconnect;
break;
}
else {
errMessage(status, "- unexpected error processing event");
cond = casProcDisconnect;
break;
}
}
/*
* call flush function if they provided one
*/
if (nAccepted > 0u) {
this->eventFlush ();
}
this->mutex.unlock();
//
// allows the derived class to be informed that it
// needs to delete itself via the event system
//
// this gets the server out of nasty situations
// where the client needs to be deleted but
// the caller may be using the client's "this"
// pointer.
//
if (this->destroyPending) {
cond = casProcDisconnect;
}
return cond;
}
//
// casEventSys::eventsOn()
//
void casEventSys::eventsOn()
{
this->mutex.lock();
//
// allow multiple events for each monitor
//
this->replaceEvents = FALSE;
//
// allow the event queue to be processed
//
this->dontProcess = FALSE;
//
// remove purge event if it is still pending
//
if (this->pPurgeEvent != NULL) {
this->eventLogQue.remove (*this->pPurgeEvent);
delete this->pPurgeEvent;
this->pPurgeEvent = NULL;
}
this->mutex.unlock();
//
// wakes up the event queue consumer
//
this->eventSignal ();
}
//
// casEventSys::eventsOff()
//
caStatus casEventSys::eventsOff()
{
this->mutex.lock();
//
// new events will replace the last event on
// the queue for a particular monitor
//
this->replaceEvents = TRUE;
//
// suppress the processing and sending of events
// only after we have purged the event queue
// for this particular client
//
if (this->pPurgeEvent==NULL) {
this->pPurgeEvent = new casEventPurgeEv;
if (this->pPurgeEvent==NULL) {
//
// if there is no room for the event then immediately
// stop processing and sending events to the client
// until we exit flow control
//
this->dontProcess = TRUE;
}
else {
this->casEventSys::addToEventQueue(*this->pPurgeEvent);
}
}
this->mutex.unlock();
return S_cas_success;
}
//
// casEventPurgeEv::cbFunc()
//
caStatus casEventPurgeEv::cbFunc (casEventSys &evSys)
{
evSys.mutex.lock();
evSys.dontProcess = TRUE;
evSys.pPurgeEvent = NULL;
evSys.mutex.unlock();
delete this;
return S_cas_success;
}