// // File: DFCALShower_factory.cc // Created: Tue May 17 11:57:50 EST 2005 // Creator: remitche (on Linux mantrid00 2.4.20-18.8smp i686) #include #include using namespace std; #include "FCAL/DFCALShower_factory.h" #include "FCAL/DFCALGeometry.h" #include "FCAL/DFCALCluster.h" #include "FCAL/DFCALHit.h" #include #include using namespace jana; //---------------- // Constructor //---------------- DFCALShower_factory::DFCALShower_factory() { // should we use CCDB constants? LOAD_CCDB_CONSTANTS = 1.; gPARMS->SetDefaultParameter("FCAL:LOAD_NONLIN_CCDB", LOAD_CCDB_CONSTANTS); SHOWER_ENERGY_THRESHOLD = 50*k_MeV; gPARMS->SetDefaultParameter("FCAL:SHOWER_ENERGY_THRESHOLD", SHOWER_ENERGY_THRESHOLD); // these need to come from database to ensure accuracy // remove default value which might be close to the right solution, // but not quite correct -- allow command line tuning NON_LIN_COEF_A = 0; NON_LIN_COEF_B = 0; NON_LIN_COEF_C = 0; NON_LIN_COEF_alfa = 0; gPARMS->SetDefaultParameter("FCAL:NON_LIN_COEF_A", NON_LIN_COEF_A); gPARMS->SetDefaultParameter("FCAL:NON_LIN_COEF_B", NON_LIN_COEF_B); gPARMS->SetDefaultParameter("FCAL:NON_LIN_COEF_C", NON_LIN_COEF_C); gPARMS->SetDefaultParameter("FCAL:NON_LIN_COEF_alfa", NON_LIN_COEF_alfa); // Parameters to make shower-depth correction taken from Radphi, // slightly modifed to match photon-polar angle FCAL_RADIATION_LENGTH = 3.1; FCAL_CRITICAL_ENERGY = 0.035; FCAL_SHOWER_OFFSET = 1.0; gPARMS->SetDefaultParameter("FCAL:FCAL_RADIATION_LENGTH", FCAL_RADIATION_LENGTH); gPARMS->SetDefaultParameter("FCAL:FCAL_CRITICAL_ENERGY", FCAL_CRITICAL_ENERGY); gPARMS->SetDefaultParameter("FCAL:FCAL_SHOWER_OFFSET", FCAL_SHOWER_OFFSET); } //------------------ // brun //------------------ jerror_t DFCALShower_factory::brun(JEventLoop *loop, int runnumber) { // Get calibration constants map fcal_parms; loop->GetCalib("FCAL/fcal_parms", fcal_parms); if (fcal_parms.find("FCAL_C_EFFECTIVE")!=fcal_parms.end()){ FCAL_C_EFFECTIVE = fcal_parms["FCAL_C_EFFECTIVE"]; if(debug_level>0)jout<<"FCAL_C_EFFECTIVE = "<(loop->GetJApplication()); const DGeometry *geom = dapp->GetDGeometry(runnumber); if (geom) { geom->GetTargetZ(m_zTarget); geom->GetFCALZ(m_FCALfront); } else{ cerr << "No geometry accessbile." << endl; return RESOURCE_UNAVAILABLE; } // by default, load non-linear shower corrections from the CCDB // but allow these to be overridden by command line parameters if(LOAD_CCDB_CONSTANTS > 0.1) { map shower_calib; loop->GetCalib("FCAL/shower_calib", shower_calib); NON_LIN_COEF_A = shower_calib["FCAL_SHOWER_CALIB_A"]; NON_LIN_COEF_B = shower_calib["FCAL_SHOWER_CALIB_B"]; NON_LIN_COEF_C = shower_calib["FCAL_SHOWER_CALIB_C"]; NON_LIN_COEF_alfa = shower_calib["FCAL_SHOWER_CALIB_D"]; if(debug_level>0) { jout << "NON_LIN_COEF_A = " << NON_LIN_COEF_A << endl; jout << "NON_LIN_COEF_B = " << NON_LIN_COEF_B << endl; jout << "NON_LIN_COEF_C = " << NON_LIN_COEF_C << endl; jout << "NON_LIN_COEF_alfa = " << NON_LIN_COEF_alfa << endl; } } return NOERROR; } //------------------ // evnt //------------------ jerror_t DFCALShower_factory::evnt(JEventLoop *eventLoop, int eventnumber) { vector fcalClusters; eventLoop->Get(fcalClusters); if(fcalClusters.size()<1)return NOERROR; // Use the center of the target as an approximation for the vertex position DVector3 vertex(0.0, 0.0, m_zTarget); // Loop over list of DFCALCluster objects and calculate the "Non-linear" corrected // energy and position for each. We'll use a logarithmic energy-weighting to // find the final position and error. for( vector< const DFCALCluster* >::const_iterator clItr = fcalClusters.begin(); clItr != fcalClusters.end(); ++clItr ){ const DFCALCluster* cluster=*clItr; double cTime = cluster->getTime(); double errX = cluster->getRMS_x(); double errY = cluster->getRMS_y(); double errZ; // will be filled by call to GetCorrectedEnergyAndPosition() // Get corrected energy, position, and errZ double Ecorrected; DVector3 pos_corrected; GetCorrectedEnergyAndPosition( cluster , Ecorrected, pos_corrected, errZ, &vertex); if (Ecorrected>0.){ //up to this point, all times have been times at which light reaches //the back of the detector. Here we correct for the time that it //takes the Cherenkov light to reach the back of the detector //so that the t reported is roughly the time of the shower at the //position pos_corrected cTime -= ( m_FCALfront + DFCALGeometry::blockLength() - pos_corrected.Z() )/FCAL_C_EFFECTIVE; // Make the DFCALShower object DFCALShower* shower = new DFCALShower; shower->setEnergy( Ecorrected ); shower->setPosition( pos_corrected ); shower->setPosError( errX, errY, errZ ); shower->setTime ( cTime ); shower->AddAssociatedObject(cluster); _data.push_back(shower); } } return NOERROR; } //-------------------------------- // GetCorrectedEnergyAndPosition // // Non-linear and depth corrections should be fixed within DFCALShower member functions //-------------------------------- void DFCALShower_factory::GetCorrectedEnergyAndPosition(const DFCALCluster* cluster, double &Ecorrected, DVector3 &pos_corrected, double &errZ, const DVector3 *vertex) { // Non-linear energy correction are done here int MAXITER = 1000; DVector3 posInCal = cluster->getCentroid(); float x0 = posInCal.Px(); float y0 = posInCal.Py(); double Eclust = cluster->getEnergy(); double A = NON_LIN_COEF_A; double B = NON_LIN_COEF_B; double C = NON_LIN_COEF_C; double alfa = NON_LIN_COEF_alfa; double Egamma = 0.; if ( A > 0 ) { Egamma = Eclust/A; for ( int niter=0; 1; niter++) { double energy = Egamma; double non_lin_part = pow(Egamma,1+alfa)/(B+C*Egamma); Egamma = Eclust/A - non_lin_part; if ( fabs( (Egamma-energy)/energy ) < 0.001 ) { break; } else if ( niter > MAXITER ) { cout << " Iteration failed for cluster energy " << Eclust << endl; Egamma = 0; break; } } } else { cout << "Warning: DFCALShower : parameter A=" << NON_LIN_COEF_A << " is not valid!" << endl; } // then depth corrections if ( Egamma > 0 ) { float dxV = x0-vertex->X(); float dyV = y0-vertex->Y(); float zV = vertex->Z(); double z0 = m_FCALfront - zV; double zMax = FCAL_RADIATION_LENGTH*(FCAL_SHOWER_OFFSET + log(Egamma/FCAL_CRITICAL_ENERGY)); double zed = z0; double zed1 = z0 + zMax; double r0 = sqrt(dxV*dxV + dyV*dyV ); int niter; for ( niter=0; niter<100; niter++) { double tt = r0/zed1; zed = z0 + zMax/sqrt( 1 + tt*tt ); if ( fabs( (zed-zed1) ) < 0.001) { break; } zed1 = zed; } posInCal.SetZ( zed + zV ); errZ = zed - zed1; } Ecorrected = Egamma; pos_corrected = posInCal; }