// Module.cpp // $Id$ #if !defined (ACE_MODULE_C) #define ACE_MODULE_C #define ACE_BUILD_DLL #include "ace/Module.h" #include "ace/Stream_Modules.h" #if !defined (__ACE_INLINE__) #include "ace/Module.i" #endif /* __ACE_INLINE__ */ ACE_ALLOC_HOOK_DEFINE(ACE_Module) template void ACE_Module::dump (void) const { ACE_TRACE ("ACE_Module::dump"); } template void ACE_Module::writer (ACE_Task *q, int flags /* = M_DELETE_WRITER */) { ACE_TRACE ("ACE_Module::writer"); // Close and maybe delete old writer this->close_i (1); this->q_pair_[1] = q; if (q != 0) ACE_CLR_BITS (q->flags_, ACE_Task_Flags::ACE_READER); // don't allow the caller to change the reader status ACE_SET_BITS (flags_, (flags & M_DELETE_WRITER)); } template void ACE_Module::reader (ACE_Task *q, int flags /* = M_DELETE_READER */) { ACE_TRACE ("ACE_Module::reader"); // Close and maybe delete old writer this->close_i (0); this->q_pair_[0] = q; if (q != 0) ACE_SET_BITS (q->flags_, ACE_Task_Flags::ACE_READER); // don't allow the caller to change the reader status ACE_SET_BITS (flags_, (flags & M_DELETE_READER)); } // Link this ACE_Module on top of ACE_Module M. template void ACE_Module::link (ACE_Module *m) { ACE_TRACE ("ACE_Module::link"); this->next (m); this->writer ()->next (m->writer ()); m->reader ()->next (this->reader ()); } template int ACE_Module::open (const char *mod_name, ACE_Task *writer_q, ACE_Task *reader_q, void *arg, int flags /* = M_DELETE */) { ACE_TRACE ("ACE_Module::open"); this->name (mod_name); this->arg_ = arg; // we may already have readers and/or writers if (this->reader ()) this->close_i (0); if (this->writer ()) this->close_i (1); if (writer_q == 0) { writer_q = new ACE_Thru_Task; ACE_SET_BITS (flags, M_DELETE_WRITER); } if (reader_q == 0) { reader_q = new ACE_Thru_Task; ACE_SET_BITS (flags, M_DELETE_READER); } this->reader (reader_q); this->writer (writer_q); // Setup back pointers. reader_q->mod_ = this; writer_q->mod_ = this; // Save the flags ACE_SET_BITS (flags_, flags); // Make sure that the memory is allocated before proceding. if (writer_q == 0 || reader_q == 0) { this->close_i (0); this->close_i (1); // Reset back pointers. reader_q->mod_ = NULL; writer_q->mod_ = NULL; delete writer_q; delete reader_q; errno = ENOMEM; return -1; } return 0; } // Set and get pointer to sibling ACE_Task in ACE_Module. template ACE_Task * ACE_Module::sibling (ACE_Task *orig) { ACE_TRACE ("ACE_Module::sibling"); if (this->q_pair_[0] == orig) return this->q_pair_[1]; else if (this->q_pair_[1] == orig) return this->q_pair_[0]; else return 0; } template ACE_INLINE ACE_Module::ACE_Module (void) { ACE_TRACE ("ACE_Module::ACE_Module"); this->name (""); // Do nothing... } template ACE_INLINE ACE_Module::~ACE_Module (void) { ACE_TRACE ("ACE_Module::~ACE_Module"); // Only close down if we haven't already done so. if (this->reader () || this->writer ()) this->close (); } template ACE_INLINE ACE_Module::ACE_Module (const char *mod_name, ACE_Task *writer_q, ACE_Task *reader_q, void *args, int flags /* = M_DELETE */) { ACE_TRACE ("ACE_Module::ACE_Module"); this->q_pair_[0] = 0; this->q_pair_[1] = 0; if (this->open (mod_name, writer_q, reader_q, args, flags) == -1) ACE_ERROR ((LM_ERROR, "%p\n", "ACE_Module")); } template int ACE_Module::close (int flags /* = M_DELETE_NONE */) { ACE_TRACE ("ACE_Module::close"); ACE_Task *reader_q = this->reader (); ACE_Task *writer_q = this->writer (); int result = 0; ACE_SET_BITS (flags_, flags); if (this->close_i (0) == -1) result = -1; if (this->close_i (1) == -1) result = -1; return result; } template int ACE_Module::close_i (int which) { ACE_TRACE ("ACE_Module::close_i"); if (this->q_pair_[which] == NULL) return 0; // Copy task pointer to prevent problems when ACE_Task::close // changes the task pointer ACE_Task *task = this->q_pair_[which]; // Change so that close doesn't get called again from the task base. // Now close the task int result = 0; if (task->module_closed() == -1) result = -1; task->flush (); task->next (0); // Should we also delete it ? if (ACE_BIT_ENABLED (flags_, which+1) ) { // Only delete the Tasks if there aren't any more threads // running in them. if (task->thr_count() == 0) delete task; } // Set the tasks pointer to NULL so that we don't try to close() // this object again if the destructor gets called. this->q_pair_[which] = NULL; // Finally remove the delete bit. ACE_CLR_BITS (flags_, which + 1); return result; } #endif /* ACE_MODULE_C */