// $Id$ // // File: DTranslationTable.h // Created: Thu Jun 27 15:33:38 EDT 2013 // Creator: davidl (on Darwin harriet.jlab.org 11.4.2 i386) // #ifndef _DTranslationTable_ #define _DTranslationTable_ #include #include using namespace std; #include #include #include #include #include #include using namespace jana; #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "GlueX.h" class DTranslationTable:public jana::JObject{ public: JOBJECT_PUBLIC(DTranslationTable); DTranslationTable(JEventLoop *loop); ~DTranslationTable(); // Each detector system has its own native indexing scheme. // Here, we define a class for each of them that has those // indexes. These are then used below in the DChannelInfo // class to relate them to the DAQ indexing scheme of // crate, slot, channel. struct csc_t{ uint32_t rocid; uint32_t slot; uint32_t channel; inline bool operator==(const struct csc_t &rhs) const { return (rocid==rhs.rocid) && (slot==rhs.slot) && (channel==rhs.channel); } }; enum Detector_t{ UNKNOWN_DETECTOR, BCAL, CDC, FCAL, FDC_CATHODES, FDC_WIRES, PS, PSC, RF, SC, TAGH, TAGM, TOF, NUM_DETECTOR_TYPES }; string DetectorName(Detector_t type) const { switch(type){ case BCAL: return "BCAL"; case CDC: return "CDC"; case FCAL: return "FCAL"; case FDC_CATHODES: return "FDC_CATHODES"; case FDC_WIRES: return "FDC_WIRES"; case PS: return "PS"; case PSC: return "PSC"; case RF: return "RF"; case SC: return "SC"; case TAGH: return "TAGH"; case TAGM: return "TAGM"; case TOF: return "TOF"; case UNKNOWN_DETECTOR: default: return "UNKNOWN"; } } class BCALIndex_t{ public: uint32_t module; uint32_t layer; uint32_t sector; uint32_t end; inline bool operator==(const BCALIndex_t &rhs) const { return (module==rhs.module) && (layer==rhs.layer) && (sector==rhs.sector) && (end==rhs.end); } }; class CDCIndex_t{ public: uint32_t ring; uint32_t straw; inline bool operator==(const CDCIndex_t &rhs) const { return (ring==rhs.ring) && (straw==rhs.straw); } }; class FCALIndex_t{ public: uint32_t row; uint32_t col; inline bool operator==(const FCALIndex_t &rhs) const { return (row==rhs.row) && (col==rhs.col); } }; class FDC_CathodesIndex_t{ public: uint32_t package; uint32_t chamber; uint32_t view; uint32_t strip; uint32_t strip_type; inline bool operator==(const FDC_CathodesIndex_t &rhs) const { return (package==rhs.package) && (chamber==rhs.chamber) && (view==rhs.view) && (strip==rhs.strip) && (strip_type==rhs.strip_type); } }; class FDC_WiresIndex_t{ public: uint32_t package; uint32_t chamber; uint32_t wire; inline bool operator==(const FDC_WiresIndex_t &rhs) const { return (package==rhs.package) && (chamber==rhs.chamber) && (wire==rhs.wire); } }; class PSIndex_t{ public: uint32_t side; uint32_t id; inline bool operator==(const PSIndex_t &rhs) const { return (side==rhs.side) && (id==rhs.id); } }; class PSCIndex_t{ public: uint32_t id; inline bool operator==(const PSCIndex_t &rhs) const { return (id==rhs.id); } }; class RFIndex_t{ public: DetectorSystem_t dSystem; inline bool operator==(const RFIndex_t &rhs) const { return (dSystem == rhs.dSystem); } }; class SCIndex_t{ public: uint32_t sector; inline bool operator==(const SCIndex_t &rhs) const { return (sector==rhs.sector); } }; class TAGHIndex_t{ public: uint32_t id; inline bool operator==(const TAGHIndex_t &rhs) const { return (id==rhs.id); } }; class TAGMIndex_t{ public: uint32_t col; uint32_t row; inline bool operator==(const TAGMIndex_t &rhs) const { return (col==rhs.col) && (row==rhs.row); } }; class TOFIndex_t{ public: uint32_t plane; uint32_t bar; uint32_t end; inline bool operator==(const TOFIndex_t &rhs) const { return (plane==rhs.plane) && (bar==rhs.bar) && (end==rhs.end); } }; // DChannelInfo holds translation between indexing schemes // for one channel. class DChannelInfo{ public: csc_t CSC; DModuleType::type_id_t module_type; Detector_t det_sys; union{ BCALIndex_t bcal; CDCIndex_t cdc; FCALIndex_t fcal; FDC_CathodesIndex_t fdc_cathodes; FDC_WiresIndex_t fdc_wires; PSIndex_t ps; PSCIndex_t psc; RFIndex_t rf; SCIndex_t sc; TAGHIndex_t tagh; TAGMIndex_t tagm; TOFIndex_t tof; }; }; // Full translation table is collection of DChannelInfo objects //map TT; // Methods bool IsSuppliedType(string dataClassName) const; void ApplyTranslationTable(jana::JEventLoop *loop) const; // fADC250 DBCALDigiHit* MakeBCALDigiHit(const BCALIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DFCALDigiHit* MakeFCALDigiHit(const FCALIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DSCDigiHit* MakeSCDigiHit( const SCIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DTOFDigiHit* MakeTOFDigiHit( const TOFIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DTAGMDigiHit* MakeTAGMDigiHit(const TAGMIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DTAGHDigiHit* MakeTAGHDigiHit(const TAGHIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DPSDigiHit* MakePSDigiHit( const PSIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DPSCDigiHit* MakePSCDigiHit( const PSCIndex_t &idx, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; DRFDigiTime* MakeRFDigiTime( const RFIndex_t &idx, const Df250PulseTime *hit) const; // fADC125 DCDCDigiHit* MakeCDCDigiHit(const CDCIndex_t &idx, const Df125PulseIntegral *pi, const Df125PulseTime *pt, const Df125PulsePedestal *pp) const; DFDCCathodeDigiHit* MakeFDCCathodeDigiHit(const FDC_CathodesIndex_t &idx, const Df125PulseIntegral *pi, const Df125PulseTime *pt, const Df125PulsePedestal *pp) const; // F1TDC DBCALTDCDigiHit* MakeBCALTDCDigiHit(const BCALIndex_t &idx, const DF1TDCHit *hit) const; DFDCWireDigiHit* MakeFDCWireDigiHit(const FDC_WiresIndex_t &idx, const DF1TDCHit *hit) const; DRFTDCDigiTime* MakeRFTDCDigiTime( const RFIndex_t &idx, const DF1TDCHit *hit) const; DSCTDCDigiHit* MakeSCTDCDigiHit( const SCIndex_t &idx, const DF1TDCHit *hit) const; DTAGMTDCDigiHit* MakeTAGMTDCDigiHit(const TAGMIndex_t &idx, const DF1TDCHit *hit) const; DTAGHTDCDigiHit* MakeTAGHTDCDigiHit(const TAGHIndex_t &idx, const DF1TDCHit *hit) const; DPSCTDCDigiHit* MakePSCTDCDigiHit( const PSCIndex_t &idx, const DF1TDCHit *hit) const; // CAEN1290TDC DTOFTDCDigiHit* MakeTOFTDCDigiHit(const TOFIndex_t &idx, const DCAEN1290TDCHit *hit) const; DRFTDCDigiTime* MakeRFTDCDigiTime(const RFIndex_t &idx, const DCAEN1290TDCHit *hit) const; void Addf250ObjectsToCallStack(JEventLoop *loop, string caller) const; void Addf125ObjectsToCallStack(JEventLoop *loop, string caller) const; void AddF1TDCObjectsToCallStack(JEventLoop *loop, string caller) const; void AddCAEN1290TDCObjectsToCallStack(JEventLoop *loop, string caller) const; void AddToCallStack(JEventLoop *loop, string caller, string callee) const; void ReadOptionalROCidTranslation(void); void SetSystemsToParse(string systems, JEventSource *eventsource); void SetSystemsToParse(JEventSource *eventsource){SetSystemsToParse(SYSTEMS_TO_PARSE, eventsource);} void ReadTranslationTable(JCalibration *jcalib=NULL); template void CopyToFactory(JEventLoop *loop, vector &v) const; template void CopyDf250Info(T *h, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const; template void CopyDf125Info(T *h, const Df125PulseIntegral *pi, const Df125PulseTime *pt, const Df125PulsePedestal *pp) const; template void CopyDF1TDCInfo(T *h, const DF1TDCHit *hit) const; template void CopyDCAEN1290TDCInfo(T *h, const DCAEN1290TDCHit *hit) const; // methods for others to search the Translation Table const DChannelInfo &GetDetectorIndex(const csc_t &in_daq_index) const; const csc_t &GetDAQIndex(const DChannelInfo &in_channel) const; //public so that StartElement can access it static map >& Get_ROCID_By_System(void); //this is static so that StartElement can access it protected: string XML_FILENAME; bool NO_CCDB; set supplied_data_types; int VERBOSE; string SYSTEMS_TO_PARSE; string ROCID_MAP_FILENAME; mutable JStreamLog ttout; string Channel2Str(const DChannelInfo &in_channel) const; private: /****************************************** STATIC-VARIABLE-ACCESSING PRIVATE MEMBER FUNCTIONS ******************************************/ //Some variables needs to be shared amongst threads (e.g. the memory used for the branch variables) //However, you cannot make them global/extern/static/static-member variables in the header file: //The header file is included in the TTAB library AND in each plugin that uses it //When a header file is included in a src file, it's contents are essentially copied directly into it //Thus there are two instances of each static variable: one in each translation unit (library) //Supposedly(?) they are linked together during runtime when loading, so there is (supposedly) no undefined behavior. //However, this causes a double free (double-deletion) when these libraries are closed at the end of the program, crashing it. //Thus the variables must be in a single source file that is compiled into a single library //However, you (somehow?) cannot make them global/extern variables in the cpp function //This also (somehow?) causes the double-free problem above for (at least) stl containers //It works for pointers-to-stl-containers and fundamental types, but I dunno why. //It's not good encapsulation anyway though. //THE SOLUTION: //Define the variables as static, in the source file, WITHIN A PRIVATE MEMBER FUNCTION. //Thus the static variables themselves only have function scope. //Access is only available via the private member functions, thus access is fully controlled. //They are shared amongst threads, so locks are necessary, but since they are private this class can handle it internally pthread_mutex_t& Get_TT_Mutex(void) const; bool& Get_TT_Initialized(void) const; map& Get_TT(void) const; map& Get_ROCID_Map(void) const; map& Get_ROCID_Inv_Map(void) const; }; //--------------------------------- // CopyToFactory //--------------------------------- template void DTranslationTable::CopyToFactory(JEventLoop *loop, vector &v) const { /// Template method for copying values from local containers into /// factories. This is done in a template since the type appears /// in at least 3 places below. It makes the code calling this /// more succinct and therefore easier to add new types. // It would be a little safer to use a dynamic_cast here, but // all documentation seems to discourage using that as it is // inefficient. JFactory *fac = (JFactory *)loop->GetFactory(T::static_className(), "", false); // false=don't allow deftags if(VERBOSE>8) ttout << " Copying " << T::static_className() << " objects to factory: " << hex << fac << dec << endl; if(fac) fac->CopyTo(v); } //--------------------------------- // CopyDf250Info //--------------------------------- template void DTranslationTable::CopyDf250Info(T *h, const Df250PulseIntegral *pi, const Df250PulseTime *pt, const Df250PulsePedestal *pp) const { /// Copy info from the fADC250 into a hit object. h->pulse_integral = pi->integral; h->pulse_time = pt==NULL ? 0:pt->time; h->pedestal = pi->pedestal; h->QF = pi->quality_factor; h->nsamples_integral = pi->nsamples_integral; h->nsamples_pedestal = pi->nsamples_pedestal; h->AddAssociatedObject(pi); if(pt) h->AddAssociatedObject(pt); if(pp) h->AddAssociatedObject(pp); } //--------------------------------- // CopyDf125Info //--------------------------------- template void DTranslationTable::CopyDf125Info(T *h, const Df125PulseIntegral *pi, const Df125PulseTime *pt, const Df125PulsePedestal *pp) const { /// Copy info from the fADC125 into a hit object. h->pulse_integral = pi->integral; h->pulse_time = pt==NULL ? 0:pt->time; h->pedestal = pi->pedestal; h->QF = pi->quality_factor; h->nsamples_integral = pi->nsamples_integral; h->nsamples_pedestal = pi->nsamples_pedestal; h->AddAssociatedObject(pi); if(pt) h->AddAssociatedObject(pt); if(pp) h->AddAssociatedObject(pp); } //--------------------------------- // CopyDF1TDCInfo //--------------------------------- template void DTranslationTable::CopyDF1TDCInfo(T *h, const DF1TDCHit *hit) const { /// Copy info from the f1tdc into a hit object. h->time = hit->time; h->AddAssociatedObject(hit); } //--------------------------------- // CopyDCAEN1290TDCInfo //--------------------------------- template void DTranslationTable::CopyDCAEN1290TDCInfo(T *h, const DCAEN1290TDCHit *hit) const { /// Copy info from the CAEN1290 into a hit object. h->time = hit->time; h->AddAssociatedObject(hit); } #endif // _DTranslationTable_