// $Id$ // // File: DFDCHit_factory.cc // Created: Wed Aug 7 11:55:02 EDT 2013 // Creator: davidl (on Darwin harriet.jlab.org 11.4.2 i386) // #include #include using namespace std; #include #include #include #include "DFDCHit_factory.h" #include #include #include using namespace jana; //------------------ // init //------------------ jerror_t DFDCHit_factory::init(void) { /// set the base conversion scales a_scale = 2.4E4/1.3E5; // cathodes t_scale = 8.0/10.0; // 8 ns/count and integer time is in 1/10th of sample t_base = 0.; // ns fadc_t_base = 0.; // ns tdc_scale = 0.115; // 115 ps/count return NOERROR; } //------------------ // brun //------------------ jerror_t DFDCHit_factory::brun(jana::JEventLoop *eventLoop, int runnumber) { // Only print messages for one thread whenever run number change static pthread_mutex_t print_mutex = PTHREAD_MUTEX_INITIALIZER; static set runs_announced; pthread_mutex_lock(&print_mutex); bool print_messages = false; if(runs_announced.find(runnumber) == runs_announced.end()){ print_messages = true; runs_announced.insert(runnumber); } pthread_mutex_unlock(&print_mutex); // reset constants tables a_gains.clear(); a_pedestals.clear(); timing_offsets.clear(); // now load them all if(print_messages) jout << "In DFDCHit_factory, loading constants..." << endl; // load scale factors map scale_factors; if(eventLoop->GetCalib("/FDC/digi_scales", scale_factors)) jout << "Error loading /FDC/digi_scales !" << endl; if( scale_factors.find("FDC_ADC_ASCALE") != scale_factors.end() ) { a_scale = scale_factors["FDC_ADC_ASCALE"]; } else { jerr << "Unable to get FDC_ADC_ASCALE from /FDC/digi_scales !" << endl; } if( scale_factors.find("FDC_ADC_TSCALE") != scale_factors.end() ) { t_scale = scale_factors["FDC_ADC_TSCALE"]; } else { jerr << "Unable to get FDC_ADC_TSCALE from /FDC/digi_scales !" << endl; } if( scale_factors.find("FDC_TDC_SCALE") != scale_factors.end() ) { tdc_scale = scale_factors["FDC_TDC_SCALE"]; } else { jerr << "Unable to get FDC_TDC_SCALE from /FDC/digi_scales !" << endl; } // load base time offset map base_time_offset; if (eventLoop->GetCalib("/FDC/base_time_offset",base_time_offset)) jout << "Error loading /FDC/base_time_offset !" << endl; if (base_time_offset.find("FDC_BASE_TIME_OFFSET") != base_time_offset.end()) fadc_t_base = base_time_offset["FDC_BASE_TIME_OFFSET"]; else jerr << "Unable to get FDC_BASE_TIME_OFFSET from /FDC/base_time_offset !" << endl; if (base_time_offset.find("FDC_TDC_BASE_TIME_OFFSET") != base_time_offset.end()) t_base = base_time_offset["FDC_TDC_BASE_TIME_OFFSET"]; else jerr << "Unable to get FDC_TDC_BASE_TIME_OFFSET from /FDC/base_time_offset !" << endl; // each FDC package has the same set of constants LoadPackageCalibTables(eventLoop,"/FDC/package1"); LoadPackageCalibTables(eventLoop,"/FDC/package2"); LoadPackageCalibTables(eventLoop,"/FDC/package3"); LoadPackageCalibTables(eventLoop,"/FDC/package4"); // Verify that the right number of layers were loaded char str[256]; if(a_gains.size() != FDC_NUM_PLANES) { sprintf(str, "Bad # of planes for FDC gains from CCDB! CCDB=%zu , should be %d", a_gains.size(), FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } if(a_pedestals.size() != FDC_NUM_PLANES) { sprintf(str, "Bad # of planes for FDC pedestals from CCDB! CCDB=%zu , should be %d", a_pedestals.size(), FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } if(timing_offsets.size() != FDC_NUM_PLANES) { sprintf(str, "Bad # of planes for FDC timing offsets from CCDB! CCDB=%zu , should be %d", timing_offsets.size(), FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } return NOERROR; } //------------------ // evnt //------------------ jerror_t DFDCHit_factory::evnt(JEventLoop *loop, int eventnumber) { /// Generate DFDCHit object for each DFDCCathodeDigiHit and /// each DFDCWireDigiHit object. /// This is where the first set of calibration constants /// is applied to convert from digitzed units into natural /// units. /// /// Note that this code does NOT get called for simulated /// data in HDDM format. The HDDM event source will copy /// the precalibrated values directly into the _data vector. char str[256]; // Make hits out of all DFDCCathodeDigiHit hits vector cathodedigihits; loop->Get(cathodedigihits); for(unsigned int i=0; iview; int gLayer=digihit->chamber + 6*(digihit->package - 1); int gPlane=layer + 3*(gLayer - 1); // Make sure gPlane and stripare in valid range if((gPlane < 1) || (gPlane > FDC_NUM_PLANES)) { sprintf(str, "DFDCDigiHit plane out of range! gPlane=%d (should be 1-%d)", gPlane, FDC_NUM_PLANES); throw JException(str); } if( (digihit->strip < 1) || (digihit->strip > STRIPS_PER_PLANE)) { sprintf(str, "DFDCDigiHit straw out of range! strip=%d for plane=%d (should be 1-%d)", digihit->strip, gPlane, STRIPS_PER_PLANE); throw JException(str); } int plane_index=gPlane-1; int strip_index=digihit->strip-1; // Apply calibration constants here double T = (double)digihit->pulse_time; //if (T<=0.) continue; // There is a slight difference between Mode 7 and 8 data // The following condition signals an error state in the flash algorithm // Do not make hits out of these const Df125PulsePedestal* PPobj = NULL; digihit->GetSingle(PPobj); if (PPobj != NULL){ if (PPobj->pedestal == 0 || PPobj->pulse_peak == 0) continue; if (PPobj->pulse_number == 1) continue; // Unintentionally had 2 pulses found in fall data (0-1 counting issue) } double pedestal = a_pedestals[plane_index][strip_index]; const Df125PulseIntegral* PIobj = NULL; digihit->GetSingle(PIobj); if (PIobj != NULL) { // the measured pedestal must be scaled by the ratio of the number // of samples used to calculate the pedestal and the actual pulse double single_sample_ped = (double)PIobj->pedestal; double nsamples_integral = (double)PIobj->nsamples_integral; double nsamples_pedestal = (double)PIobj->nsamples_pedestal; pedestal = single_sample_ped * nsamples_integral/nsamples_pedestal; } if ( PPobj == NULL || PIobj == NULL) continue; // We don't want hits where ANY of the associated information is missing /* const Df125PulsePedestal *PP=NULL; digihit->GetSingle(PP); double A=0.; if (PP!=NULL){ A=PP->pulse_peak; pedestal=PP->pedestal; //_DBG_ << A << " " << pedestal << endl; } if (A-pedestal<0.) continue; */ double A = (double)digihit->pulse_integral; double q = a_scale * a_gains[plane_index][strip_index] * (A-pedestal); double t = t_scale * T - timing_offsets[plane_index][strip_index]+fadc_t_base; DFDCHit *hit = new DFDCHit; hit->layer = digihit->view; hit->gLayer = gLayer; hit->gPlane = gPlane; hit->module = 1 + (gLayer-1)/3; hit->element = digihit->strip; hit->plane = digihit->view; // "plane" is apparently the same as "layer" hit->r = DFDCGeometry::getWireR(hit); hit->d = 0.0; // MC data only hit->type = digihit->strip_type; // n.b. DEventSourceHDDM hardwires this as "1" for cathodes! hit->itrack = -1; // MC data only hit->ptype = 0;// MC data only hit->q = q; hit->t = t; //cerr << "FDC hitL plane = " << hit->gPlane << " element = " << hit->element << endl; hit->AddAssociatedObject(digihit); _data.push_back(hit); } // Make hits out of all DFDCWireDigiHit hits vector wiredigihits; loop->Get(wiredigihits); for(unsigned int i=0; igLayer = digihit->chamber + 6*(digihit->package - 1); hit->module = 1 + (hit->gLayer-1)/3; hit->layer = hit->gLayer - (hit->module-1)*3; hit->plane = 2; // wire is always in "X" layer hit->gPlane = hit->plane + 3*(hit->gLayer - 1); hit->element = digihit->wire; hit->r = DFDCGeometry::getWireR(hit); hit->d = 0.0; // MC data only hit->type = DFDCHit::AnodeWire; // n.b. DEventSourceHDDM hardwires this as "0" for anodes! hit->itrack = -1; // MC data only hit->ptype = 0; // MC data only // Make sure gPlane and wire are in valid range if( (hit->gPlane < 1) || (hit->gPlane > FDC_NUM_PLANES)) { sprintf(str, "DFDCDigiHit plane out of range! gPlane=%d (should be 1-%d)", hit->gPlane, FDC_NUM_PLANES); throw JException(str); } if( (digihit->wire < 1) || (digihit->wire > WIRES_PER_PLANE)) { sprintf(str, "DFDCDigiHit straw out of range! wire=%d for plane=%d (should be 1-%d)", digihit->wire, hit->gPlane, WIRES_PER_PLANE); throw JException(str); } // Apply calibration constants here double T = (double)digihit->time; T = tdc_scale * (T - timing_offsets[hit->gPlane-1][hit->element-1]) + t_base; hit->q = 0.0; // no charge measured for wires in FDC hit->t = T; hit->AddAssociatedObject(digihit); _data.push_back(hit); } return NOERROR; } //------------------ // erun //------------------ jerror_t DFDCHit_factory::erun(void) { return NOERROR; } //------------------ // fini //------------------ jerror_t DFDCHit_factory::fini(void) { return NOERROR; } //------------------ // LoadPackageCalibTables //------------------ void DFDCHit_factory::LoadPackageCalibTables(jana::JEventLoop *eventLoop, string ccdb_prefix) { vector< vector > new_gains, new_pedestals, new_strip_t0s, new_wire_t0s; char str[256]; if(eventLoop->GetCalib(ccdb_prefix+"/strip_gains", new_gains)) cout << "Error loading "+ccdb_prefix+"/strip_gains !" << endl; if(eventLoop->GetCalib(ccdb_prefix+"/strip_pedestals", new_pedestals)) cout << "Error loading "+ccdb_prefix+"/strip_pedestals !" << endl; if(eventLoop->GetCalib(ccdb_prefix+"/strip_timing_offsets", new_strip_t0s)) cout << "Error loading "+ccdb_prefix+"/strip_timing_offsets!" << endl; if(eventLoop->GetCalib(ccdb_prefix+"/wire_timing_offsets", new_wire_t0s)) cout << "Error loading "+ccdb_prefix+"/wire_timing_offsets!" << endl; for(int nchamber=0; nchamber<6; nchamber++) { // check the size of table rows if(new_gains[2*nchamber].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC gain from CCDB! CCDB=%zu , should be %d", new_gains[2*nchamber].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_gains[2*nchamber+1].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC gain from CCDB! CCDB=%zu , should be %d", new_gains[2*nchamber+1].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_pedestals[2*nchamber].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC pedestals from CCDB! CCDB=%zu , should be %d", new_pedestals[2*nchamber].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_pedestals[2*nchamber+1].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC pedestals from CCDB! CCDB=%zu , should be %d", new_pedestals[2*nchamber+1].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_strip_t0s[2*nchamber].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC timing offsets from CCDB! CCDB=%zu , should be %d", new_strip_t0s[2*nchamber].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_strip_t0s[2*nchamber+1].size() != STRIPS_PER_PLANE) { sprintf(str, "Bad # of strips for FDC timing offsets from CCDB! CCDB=%zu , should be %d", new_strip_t0s[2*nchamber+1].size(), STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } if(new_wire_t0s[nchamber].size() != WIRES_PER_PLANE) { sprintf(str, "Bad # of wires for FDC timing offsets from CCDB! CCDB=%zu , should be %d", new_wire_t0s[2*nchamber].size(), WIRES_PER_PLANE); cerr << str << endl; throw JException(str); } // load ADC gains (only for cathode strips) a_gains.push_back( new_gains[2*nchamber] ); a_gains.push_back( vector() ); a_gains.push_back( new_gains[2*nchamber+1] ); // load ADC pedestals (only for cathode strips) a_pedestals.push_back( new_pedestals[2*nchamber] ); a_pedestals.push_back( vector() ); a_pedestals.push_back( new_pedestals[2*nchamber+1] ); // load t0's for strips and wires timing_offsets.push_back( new_strip_t0s[2*nchamber] ); timing_offsets.push_back( new_wire_t0s[nchamber] ); timing_offsets.push_back( new_strip_t0s[2*nchamber+1] ); } } //------------------------------------ // GetConstant // Allow a few different interfaces //------------------------------------ const double DFDCHit_factory::GetConstant(const fdc_digi_constants_t &the_table, const int in_gPlane, const int in_element) const { char str[256]; if( (in_gPlane <= 0) || (static_cast(in_gPlane) > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", in_gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (in_element <= 0) || (static_cast(in_element) > the_table[in_gPlane].size())) { sprintf(str, "Bad element # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %zu", in_element, the_table[in_gPlane].size()); cerr << str << endl; throw JException(str); } return the_table[in_gPlane-1][in_element-1]; } const double DFDCHit_factory::GetConstant(const fdc_digi_constants_t &the_table, const DFDCCathodeDigiHit *in_digihit) const { char str[256]; int gLayer = in_digihit->chamber + 6*(in_digihit->package - 1); int gPlane = in_digihit->view + 3*(gLayer - 1); if( (gPlane <= 0) || (static_cast(gPlane) > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (in_digihit->strip <= 0) || (static_cast(in_digihit->strip) > STRIPS_PER_PLANE)) { sprintf(str, "Bad strip # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", in_digihit->strip, STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } return the_table[gPlane-1][in_digihit->strip-1]; } const double DFDCHit_factory::GetConstant(const fdc_digi_constants_t &the_table, const DFDCWireDigiHit *in_digihit) const { char str[256]; int gLayer = in_digihit->chamber + 6*(in_digihit->package - 1); int gPlane = 2 + 3*(gLayer - 1); if( (gPlane <= 0) || (static_cast(gPlane) > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (in_digihit->wire <= 0) || (static_cast(in_digihit->wire) > WIRES_PER_PLANE)) { sprintf(str, "Bad wire # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", in_digihit->wire, WIRES_PER_PLANE); cerr << str << endl; throw JException(str); } return the_table[gPlane-1][in_digihit->wire-1]; } const double DFDCHit_factory::GetConstant(const fdc_digi_constants_t &the_table, const DFDCHit *in_hit) const { char str[256]; if( (in_hit->gPlane <= 0) || (static_cast(in_hit->gPlane) > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", in_hit->gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (in_hit->element <= 0) || (static_cast(in_hit->element) > the_table[in_hit->gPlane].size())) { sprintf(str, "Bad element # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %zu", in_hit->element, the_table[in_hit->gPlane].size()); cerr << str << endl; throw JException(str); } return the_table[in_hit->gPlane-1][in_hit->element-1]; } /* const double DFDCHit_factory::GetConstant(const fdc_digi_constants_t &the_table, const DTranslationTable *ttab, const int in_rocid, const int in_slot, const int in_channel) const { char str[256]; DTranslationTable::csc_t daq_index = { in_rocid, in_slot, in_channel }; DTranslationTable::DChannelInfo channel_info = ttab->GetDetectorIndex(daq_index); if( channel_info.det_sys == DTranslationTable::FDC_CATHODES ) { // FDC Cathodes int gLayer = channel_info.fdc_cathodes.chamber + 6*(channel_info.fdc_cathodes.package - 1); int gPlane = channel_info.fdc_cathodes.view + 3*(gLayer - 1); if( (gPlane <= 0) || (gPlane > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (channel_info.fdc_cathodes.strip <= 0) || (channel_info.fdc_cathodes.strip > STRIPS_PER_PLANE)) { sprintf(str, "Bad strip # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", channel_info.fdc_cathodes.strip, STRIPS_PER_PLANE); cerr << str << endl; throw JException(str); } return the_table[gPlane-1][channel_info.fdc_cathodes.strip-1]; } else if( channel_info.det_sys == DTranslationTable::FDC_WIRES ) { // FDC Wirees int gLayer = channel_info.fdc_wires.chamber + 6*(channel_info.fdc_wires.package - 1); int gPlane = 2 + 3*(gLayer - 1); // wire planes are always layer 2 if( (gPlane <= 0) || (gPlane > FDC_NUM_PLANES)) { sprintf(str, "Bad gPlane # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", gPlane, FDC_NUM_PLANES); cerr << str << endl; throw JException(str); } // strip and wire planes have different numbers of elements if( (channel_info.fdc_wires.wire <= 0) || (channel_info.fdc_wires.wire > WIRES_PER_PLANE)) { sprintf(str, "Bad strip # requested in DFDCHit_factory::GetConstant()! requested=%d , should be %ud", channel_info.fdc_wires.wire, WIRES_PER_PLANE); cerr << str << endl; throw JException(str); } return the_table[gPlane-1][channel_info.fdc_wires.wire-1]; } else { sprintf(str, "Got bad detector type in DFDCHit_factory::GetConstant()! requested=%d", channel_info.module_type); cerr << str << endl; throw JException(str); return -1.; // should never reach here! } } */