// $Id: smear.cc 7650 2011-03-29 22:52:30Z shepherd $ // // Created June 22, 2005 David Lawrence #include #include #include #include using namespace std; #include #include #include "units.h" #include "HDDM/hddm_s.h" #include #include #include #include #include #include "DBCALReadoutChannel.h" #ifndef _DBG_ #define _DBG_ cout<<__FILE__<<":"<<__LINE__<<" " #define _DBG__ cout<<__FILE__<<":"<<__LINE__< bcal_fADCs; // key is DBCALGeometry::fADCId() // Flag used specifically for BCAL extern bool SMEAR_BCAL; // The following are all false by default, but can be // set to true via command line parameters. Setting // one of these to true will turn OFF the feature. extern bool NO_E_SMEAR; extern bool NO_T_SMEAR; extern bool NO_DARK_PULSES; extern bool NO_THRESHOLD_CUT; // setup response parameters extern double BCAL_DARKRATE_GHZ; // 0.0176 (from calibDB BCAL/bcal_parms) for 4x4 array extern double BCAL_SIGMA_SIG_RELATIVE; // 0.105 (from calibDB BCAL/bcal_parms) extern double BCAL_SIGMA_PED_RELATIVE; // 0.139 (from calibDB BCAL/bcal_parms) extern double BCAL_SIPM_GAIN_VARIATION; // 0.04 (from calibDB BCAL/bcal_parms) extern double BCAL_XTALK_FRACT; // 0.157 (from calibDB BCAL/bcal_parms) extern double BCAL_INTWINDOW_NS; // 100 (from calibDB BCAL/bcal_parms) extern double BCAL_DEVICEPDE; // 0.21 (from calibDB BCAL/bcal_parms) extern double BCAL_SAMPLING_FRACT; // 0.095 (from calibDB BCAL/bcal_parms) extern double BCAL_PHOTONSPERSIDEPERMEV_INFIBER;// 75 (from calibDB BCAL/bcal_parms) extern double BCAL_AVG_DARK_DIGI_VALS_PER_EVENT;// 240 used to set thresholds extern double BCAL_SAMPLINGCOEFA; // 0.042 (from calibDB BCAL/bcal_parms) extern double BCAL_SAMPLINGCOEFB; // 0.013 (from calibDB BCAL/bcal_parms) extern double BCAL_TIMEDIFFCOEFA; // 0.07 * sqrt( 2 ) (from calibDB BCAL/bcal_parms) extern double BCAL_TIMEDIFFCOEFB; // 0.00 * sqrt( 2 ) (from calibDB BCAL/bcal_parms) //----------- // SmearBCAL //----------- void SmearBCAL(s_HDDM_t *hddm_s) { DBCALGeometry bcalGeom; // Initialize BCAL globals on first call if(!BCAL_INITIALIZED)bcalInit(); // Container to keep track of the elements in bcal_fADCs that // have hits so we can sparsely clear them at the end of this set fADC_ids_with_hits; // Loop over PhysicsEvents s_PhysicsEvents_t* PE = hddm_s->physicsEvents; if(!PE) return; for(unsigned int iphysics_event=0; iphysics_eventmult; iphysics_event++){ // Get pointer to HDDM hits structure s_HitView_t *hits = PE->in[iphysics_event].hitView; // Make sure HDDM stuctures exist. // In the case of no real BCAL hits, we may still want to emit // dark hit only events. In this case, we must create the BCAL // tree here. if(hits==HDDM_NULL)hits = PE->in[iphysics_event].hitView = make_s_HitView(); if(hits->barrelEMcal == HDDM_NULL)hits->barrelEMcal = make_s_BarrelEMcal(); if(hits->barrelEMcal->bcalCells == HDDM_NULL)hits->barrelEMcal->bcalCells = make_s_BcalCells(0); // Loop over GEANT hits in BCAL s_BcalCells_t *cells = hits->barrelEMcal->bcalCells; for(unsigned int j=0; jmult; j++){ s_BcalCell_t *cell = &cells->in[j]; // If data exists in HDDM tree for SiPM hits, then delete it // so we can write in new hits below. if(cell->bcalSiPMUpHits!=HDDM_NULL)free(cell->bcalSiPMUpHits); if(cell->bcalSiPMDownHits!=HDDM_NULL)free(cell->bcalSiPMDownHits); // We make one SiPM hit for every GEANT hit in both the up and // down stream ends unsigned int Nbcalhits = cell->bcalHits->mult; cell->bcalSiPMUpHits = make_s_BcalSiPMUpHits(Nbcalhits); cell->bcalSiPMDownHits = make_s_BcalSiPMDownHits(Nbcalhits); cell->bcalSiPMUpHits->mult = Nbcalhits; cell->bcalSiPMDownHits->mult = Nbcalhits; // Loop over GEANT hits in BCAL for this cell for(unsigned int k=0; kbcalHits->in[k]; s_BcalSiPMUpHit_t *bcaluphit = &cell->bcalSiPMUpHits->in[k]; s_BcalSiPMDownHit_t *bcaldownhit = &cell->bcalSiPMDownHits->in[k]; // Distance from location of energy deposition to each end of BCAL double upDist = ( bcalGeom.BCALFIBERLENGTH / 2 ) + bcalhit->zLocal; double downDist = ( bcalGeom.BCALFIBERLENGTH / 2 ) - bcalhit->zLocal; // sampling fluctuations are correlated between ends double smearedE = bcalSamplingSmear( bcalhit->E ); double upEnergy = smearedE * exp( -upDist / bcalGeom.ATTEN_LENGTH ); double downEnergy = smearedE * exp( -downDist / bcalGeom.ATTEN_LENGTH ); // independently smear time for both ends -- time smearing // parameters come from data taken with beam at the center of // the module so there is an implicit exp( ( -L / 2 ) / lambda ) // that needs to be canceled out since we are working // at this stage with attenuated energies double smearedtUp = bcalTimeSmear( bcalhit->t, upEnergy*BCAL_UNATTENUATE_TO_CENTER ); double smearedtDown = bcalTimeSmear( bcalhit->t, downEnergy*BCAL_UNATTENUATE_TO_CENTER ); // To get a realistic distribution we have to convert the smeared energy into // an integer number of photoelectrons. We'll add the dark hits to that and // then calculate a sigma in energy based on the total number of pixels that // fired. The number of photoelectrons will be converted back into energy // and then smeared using this sigma. double upPE = floor(0.5 + upEnergy/(BCAL_mevPerPE * k_MeV)) + getDarkHits(); double downPE = floor(0.5 + downEnergy/(BCAL_mevPerPE * k_MeV)) + getDarkHits(); double sigma_up = sqrt(BCAL_sigma_ped_sq + upPE*BCAL_sigma_singlePE_sq); double sigma_down = sqrt(BCAL_sigma_ped_sq + downPE*BCAL_sigma_singlePE_sq); // Convert integer # of PE back into energy upEnergy = upPE * BCAL_mevPerPE * k_MeV; downEnergy = downPE * BCAL_mevPerPE * k_MeV; // Add in pedestal widths due to SiPMs upEnergy += SampleGaussian(sigma_up); downEnergy += SampleGaussian(sigma_down); // now offset times for propagation distance double upTime = smearedtUp + upDist / bcalGeom.C_EFFECTIVE; double downTime = smearedtDown + downDist / bcalGeom.C_EFFECTIVE; // If energy is smeared to negative or time is nan, set to 0. if(upEnergy <= 0 || !finite(upTime)){ upEnergy = 0; upTime = 0; } if(downEnergy <= 0 || !finite(downTime)){ downEnergy = 0; downTime = 0; } // Record upstream SiPM values in HDDM bcaluphit->E = upEnergy; bcaluphit->t = upTime; // Record downstream SiPM values in HDDM bcaldownhit->E = downEnergy; bcaldownhit->t = downTime; // Remember these hits in appropriate readout channel int fADC_Id = bcalGeom.fADCId( cell->module, cell->layer, cell->sector); bcal_fADCs[fADC_Id].uphits.push_back(bcaluphit); bcal_fADCs[fADC_Id].downhits.push_back(bcaldownhit); fADC_ids_with_hits.insert(fADC_Id); } //k (bcal hits) } //j (cells) // Above, we looped over GEANT hits to create SiPM hits. Below we loop // over fADC (readout) channels which may contain multiple SiPMs. // For each channel, we determine how many hits occurred in // BCAL_TIME_WINDOW, including cumulative dark hits. The number of // cells over threshold are counted and the values recorded so that // we can create the correct number of bcalfADCCell stuctures in HDDM // later. // Loop over readout channels set fADC_ids_over_thresh; map::iterator iter; for(iter = bcal_fADCs.begin(); iter!=bcal_fADCs.end(); iter++){ int fADC_id = iter->first; DBCALReadoutChannel &bcalfADC = iter->second; vector &uphits = bcalfADC.uphits; vector &downhits = bcalfADC.downhits; //--------- Upstream ------------- // Initialize bcalfADC.Eup = 0.0; bcalfADC.tup = 0.0; // Sum upstream hits for(unsigned int j=0; jE; bcalfADC.tup += uphits[j]->t * uphits[j]->E; // energy weighted time for sum } // Add in dark pulses for upstream SiPMs not hit, but that are included in sum unsigned int Ndark_channels_up = bcalfADC.NSiPM - uphits.size(); if(uphits.size() > bcalfADC.NSiPM)Ndark_channels_up = 0; for(unsigned int j=0; jE; bcalfADC.tdown += downhits[j]->t * downhits[j]->E; // energy weighted time for sum } // Add in dark pulses for downstream SiPMs not hit, but that are included in sum unsigned int Ndark_channels_down = bcalfADC.NSiPM - downhits.size(); if(downhits.size() > bcalfADC.NSiPM)Ndark_channels_down = 0; for(unsigned int j=0; j bcalfADC.threshold) || (bcalfADC.Edown > bcalfADC.threshold)){ fADC_ids_over_thresh.insert(fADC_id); } } // readout channel // At this point we have summed all SiPMs with dark hits included. Any cell with // either the upstream or downstream fADC over threshold has its id stored in // the fADC_ids_over_thresh container. We now need to create bcalfADCCell, // bcalfADCUpHit, and bcalfADCDownHit structures within the HDDM tree to hold // the fADC values. // Delete any existing bcalfADCCell structures (along with their bcalfADCUpHit // and bcalfADCDownHit structures). if(hits->barrelEMcal->bcalfADCCells!=HDDM_NULL){ for(unsigned int i=0; ibarrelEMcal->bcalfADCCells->mult; i++){ s_BcalfADCUpHits_t *uphits = hits->barrelEMcal->bcalfADCCells->in[i].bcalfADCUpHits; s_BcalfADCDownHits_t *downhits = hits->barrelEMcal->bcalfADCCells->in[i].bcalfADCDownHits; if(uphits!=NULL && uphits!=HDDM_NULL){ free(uphits); } if(downhits!=NULL && downhits!=HDDM_NULL){ free(downhits); } } free(hits->barrelEMcal->bcalfADCCells); } // Make sure bcalfADCCells pointer is empty in case we return early below hits->barrelEMcal->bcalfADCCells = (s_BcalfADCCells_t*)HDDM_NULL; // If we have no cells over threshold, then bail now. if(fADC_ids_over_thresh.size()==0) continue; // next iphysics_event // Create bcalfADCCell structures to hold all of our hits hits->barrelEMcal->bcalfADCCells = make_s_BcalfADCCells(fADC_ids_over_thresh.size()); unsigned int &mult = hits->barrelEMcal->bcalfADCCells->mult; set::iterator it = fADC_ids_over_thresh.begin(); for(mult=0; multbarrelEMcal->bcalfADCCells->in[mult]; // Get pointer to our fADC cell information that needs to be copied to HDDM DBCALReadoutChannel &bcalfADC = bcal_fADCs[*it]; fADCcell->module = bcalfADC.module; fADCcell->layer = bcalfADC.layer; fADCcell->sector = bcalfADC.sector; // Upstream hit if(bcalfADC.Eup > bcalfADC.threshold){ fADCcell->bcalfADCUpHits = make_s_BcalfADCUpHits(1); fADCcell->bcalfADCUpHits->mult = 1; s_BcalfADCUpHit_t *fadcuphit = &fADCcell->bcalfADCUpHits->in[0]; fadcuphit->E = bcalfADC.Eup; fadcuphit->t = bcalfADC.tup; }else{ fADCcell->bcalfADCUpHits = (s_BcalfADCUpHits_t*)HDDM_NULL; } // Downstream hit if(bcalfADC.Edown > bcalfADC.threshold){ fADCcell->bcalfADCDownHits = make_s_BcalfADCDownHits(1); fADCcell->bcalfADCDownHits->mult = 1; s_BcalfADCDownHit_t *fadcdownhit = &fADCcell->bcalfADCDownHits->in[0]; fadcdownhit->E = bcalfADC.Edown; fadcdownhit->t = bcalfADC.tdown; }else{ fADCcell->bcalfADCDownHits = (s_BcalfADCDownHits_t*)HDDM_NULL; } } // fADC_ids_over_thresh } // iphysics_event // Clear the bcal_fADCs elements that have hits. // We keep track above so we can do a sparse clear here to // save time clearing every fADC array for every event. set::iterator iter=fADC_ids_with_hits.begin(); for(; iter!=fADC_ids_with_hits.end(); iter++){ bcal_fADCs[*iter].Clear(); } } //----------- // bcalSamplingSmear //----------- double bcalSamplingSmear( double E ) { if(NO_E_SMEAR)return E; double sigmaSamp = BCAL_SAMPLINGCOEFA / sqrt( E ) + BCAL_SAMPLINGCOEFB; return( E * (1.0 + SampleGaussian(sigmaSamp)) ); } //----------- // bcalTimeSmear //----------- double bcalTimeSmear( double t, double E ) { if(NO_T_SMEAR)return t; double sigmaT = BCAL_TIMEDIFFCOEFA / sqrt( E ) + BCAL_TIMEDIFFCOEFB; return( t + SampleGaussian(sigmaT) ); } //----------- // getDarkHits //----------- int getDarkHits() { if(NO_DARK_PULSES)return 0; int darkPulse = (int)floor(0.5+SamplePoisson( BCAL_DARKRATE_GHZ* BCAL_INTWINDOW_NS )); int xTalk = (int)floor(0.5+SamplePoisson( (double)darkPulse * BCAL_XTALK_FRACT )); return( xTalk + darkPulse ); } //----------- // bcalInit //----------- void bcalInit(void) { // Calculate the inner and outer thresholds based on // the parameters from the calibDB. The threshold values // are left in the global variables BCAL_inner_thresh // and BCAL_outer_thresh. CalculateThresholds(); for (int i=0;i<><><><><><><><><><><><><><><><><><><><><><><><><><><> // now we need to find n such that // sum_i=1^n P(n) > ( 1 - maxOccupancy ) // P(n) is the probability to have n pulses // // P(n) is really a convolution since we have: // P(n) = P( n_d + n_x ) = // P( n_d; nAvg ) * P( n_x; n_d * x ) // // n_d number of dark pulses // x is the cross talk rate // numerically build int_0^n P(n) // in practice the cutoff is going to be < 100 PE // we can build an accurate pdf up to that double pdf[100]; for( int i = 0; i < 100; ++i ){ pdf[i] = 0; } // probability for zero is easy: double darkTerm = exp( -mean_dark_pulses_array ); pdf[0] = darkTerm; for( int n_d = 1; n_d < 100; ++n_d ){ darkTerm *= ( mean_dark_pulses_array / n_d ); double xTalkAvg = n_d * BCAL_XTALK_FRACT; // probability for zero x-talk pulses double xTerm = exp( -xTalkAvg ); pdf[n_d] += ( xTerm * darkTerm ); // now include probability for additional // cross talk pulses for( int n_x = 1; n_x + n_d < 100; ++n_x ){ xTerm *= ( xTalkAvg / n_x ); pdf[n_d+n_x] += ( xTerm * darkTerm ); } } //<><><><><><><><><><><><><><><><><><><><><><><><><><><><> // At this point we add in electronic noise using the sigma_singlePE // and sigma_ped calculated above. This is done by creating a histogram // and filling it with contributions from Npe=0 to Npe=99 // we don't need to delete this because because ROOT owns it, I think? TH1D *bcal_dark_signal = new TH1D("bcal_dark_signal", "fADC signal", 1001, -0.05, 100.05); bcal_dark_signal->SetXTitle("fADC (MeV)"); for(int bin=1; bin<=bcal_dark_signal->GetNbinsX(); bin++){ double x = bcal_dark_signal->GetBinCenter(bin); // Loop over Npe double prob = 0.0; for(int Npe=0; Npe<100; Npe++){ double sigma_tot = sqrt(BCAL_sigma_ped_sq + (double)Npe*BCAL_sigma_singlePE_sq)*1000.0; // in MeV double E = (double)Npe * BCAL_mevPerPE; // in MeV prob += TMath::Gaus(x, E , sigma_tot)*pdf[Npe]; } bcal_dark_signal->SetBinContent(bin, prob); } // The histogram bcal_dark_signal now contains the probability // distribution for the electronic response of a single SiPM array // in units of MeV. We wish to convolute that with itself // multiple times to get the PDF for multiple SiPMs added together // (in the case of summing). for (int i=0;iClone("ibcal_dark_signal_sum_inner"); for(int bin=2; bin<=ibcal_dark_signal_sum_inner->GetNbinsX(); bin++){ double sum = ibcal_dark_signal_sum_inner->GetBinContent(bin-1) + ibcal_dark_signal_sum_inner->GetBinContent(bin); ibcal_dark_signal_sum_inner->SetBinContent(bin, sum); } // Normalize int last_bin = ibcal_dark_signal_sum_inner->GetNbinsX(); ibcal_dark_signal_sum_inner->Scale(1.0/ibcal_dark_signal_sum_inner->GetBinContent(last_bin)); // Finally, search the integrated, normalized histograms for the bin // where the integral fraction above it is less than fraction_above_threshold for(int bin=1 ; bin<=last_bin; bin++){ if(ibcal_dark_signal_sum_inner->GetBinContent(bin) < (1.0 - fraction_above_threshold)){ BCAL_inner_thresh[i] = ibcal_dark_signal_sum_inner->GetBinCenter(bin) * k_MeV; } } } for (int i=0;iClone("ibcal_dark_signal_sum_outer"); for(int bin=2; bin<=ibcal_dark_signal_sum_outer->GetNbinsX(); bin++){ double sum = ibcal_dark_signal_sum_outer->GetBinContent(bin-1) + ibcal_dark_signal_sum_outer->GetBinContent(bin); ibcal_dark_signal_sum_outer->SetBinContent(bin, sum); } // Normalize int last_bin = ibcal_dark_signal_sum_outer->GetNbinsX(); ibcal_dark_signal_sum_outer->Scale(1.0/ibcal_dark_signal_sum_outer->GetBinContent(last_bin)); // Finally, search the integrated, normalized histograms for the bin // where the integral fraction above it is less than fraction_above_threshold for(int bin=1 ; bin<=last_bin; bin++){ if(ibcal_dark_signal_sum_outer->GetBinContent(bin) < (1.0 - fraction_above_threshold)){ BCAL_outer_thresh[i] = ibcal_dark_signal_sum_outer->GetBinCenter(bin) * k_MeV; } } } } //----------- // CombineSiPM_PDFs //----------- TH1D* CombineSiPM_PDFs(const char *hname, TH1D *single, int NSiPM) { // It is quicker (and easier) to do this in stages by calculating the // probability of signals adding to a given energy and then // subsequently calculating the probability of adding one more. TH1D *summed = (TH1D*)single->Clone(hname); for(int i=2; i<=NSiPM; i++){ // Make copy of summed histogram since we'll be overwritting it TH1D *tmp = (TH1D*)summed->Clone("tmp"); // Reset summed histo (for safety) summed->Reset(); // Loop over all combinations of bins in the previous iteration's // summed probability histo and the single SiPM probability // histo. Fill in the new summed probability histo. for(int ibin=1; ibin<=tmp->GetNbinsX(); ibin++){ double NPE1 = tmp->GetBinCenter(ibin); double prob1 = tmp->GetBinContent(ibin); for(int jbin=1; jbin<=single->GetNbinsX(); jbin++){ double NPE2 = single->GetBinCenter(jbin); double prob2 = single->GetBinContent(jbin); // Find the bin in the summed histogram corresponding to // the sum of the energies from the bin in the previous // iteration's summed prob. histo and the single SiPM // prob. histo. If the energy sum is in range, then set // the bin content of the current iteration's summed prob. // histo. int kbin = summed->FindBin(NPE1 + NPE2); if(kbin>=1 && kbin<=summed->GetNbinsX()){ summed->Fill(NPE1 + NPE2, prob1*prob2); } } } delete tmp; } return summed; }