#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; #include "StandardLabels.C" #include "Angle.h" #include "Geom.h" void GetPoints(TH2D *h, TGraphErrors* &gres, TGraphErrors* &gmean); Double_t asymmetric_gaus(Double_t *xptr, Double_t *par); //------------------------------------ // Copied from JEventProcessor_bcal_timing.cc typedef struct{ int event; int layer; int sector; int fADC_up; int fADC_dn; float Etot; float geometric_mean; float tup; float tdn; float tup_corrected; float tdn_corrected; float theta_thrown; float E_thrown; }HIT_t; //------------------------------------ //----------------- // E_calibrate //----------------- void E_calibrate(void) { TFile *f = new TFile("hd_root.root"); f->cd(); TTree *tree = (TTree*)gROOT->FindObject("tree"); HIT_t hit; HIT_t *hitptr = &hit; TBranch *branch = tree->GetBranch("T"); branch->SetAddress(hitptr); TH2D *h = new TH2D("h", "", 200, 0.0, 11500.0, 200, 0.0, 2.0); TH2D *y = new TH2D("y", "", 200, 0.0, 2.0, 200, 0.0, 11500.0); // Loop over all events and fill histogram. We do this // rather than TTree::Project since we need to call the // BCAL_tdiff_res for each entry int Nentries = branch->GetEntries(); double E_thrown = 0.0; double E_tot = 0.0; int last_event=1; for(int i=1; i<=Nentries; i++){ tree->GetEntry(i); // Fill histogram on event boundaries and reset for next event if(hit.event!=last_event){ h->Fill(E_tot, E_thrown); y->Fill(E_thrown, E_tot); E_thrown = 0.0; E_tot = 0.0; last_event = hit.event; } if(hit.layer<1 || hit.layer>MaxSummedLayers())continue; // // Apply fADC threshold double mV_per_MeV = 5.0833; // see bcal_timing/DBCALTimeSpectrum.cc::Convolute mV_per_MeV /= 5.0; // This scales down the second stage TDC amplification so we are in units of the fADC leg double LSB_pC_fADC = 2.0/50.0*4.0E3/4096.0; // see top of bcal_timing/DBCALTimeSpectrum.cc double SiPM_pulse_integral_pC = 1200.0; double SiPM_pulse_peak_mV = 2293.0; double pC_per_MeV = mV_per_MeV * (SiPM_pulse_integral_pC/SiPM_pulse_peak_mV); double LSB_MeV_fADC = LSB_pC_fADC/pC_per_MeV; double fADC_threshold = 1.0/LSB_MeV_fADC; // threshold for 3.0 attenuated MeV in fADC counts fADC_threshold = floor(0.5 + fADC_threshold); // integerize //cout<<"fADC threshold in fADC counts: "<SetTicks(); c1->SetGrid(); //c1->SetLogx(); TF1 *fun = new TF1("fun", "[0]+x*([1]+x*([2]+x*([3])))", 100, 1100); fun->SetLineColor(kYellow); fun->SetLineWidth(1); fun->SetParameter(0, 0.0); fun->SetParameter(1, 0.57/1000.0); fun->FixParameter(2, 0.0); fun->FixParameter(3, 0.0); gmean->Fit(fun,"E", ""); TH2D *axes = new TH2D("axes","", 100, 0.0, 11500.0, 100, 0.0, 2.0); axes->SetStats(0); axes->SetXTitle("#Sigma Geometric mean (fADC counts)"); axes->SetYTitle("Energy generated (GeV)"); axes->Draw(); h->Draw("same"); gmean->Draw("Psame"); gres->Draw("Psame"); StandardLabels(axes, Scheme(), "", AngleStr("#theta_{#gamma}=")); c1->SaveAs("E_calibrate.pdf"); c1->SaveAs("E_calibrate.png"); // For the "calibration" file we want two functions. One // that converts fADC values to GeV and the one that does // the inverse. For the inverse function we just do a fit // to the "y" histogram (x-y axes exchanged relative to "h") TGraphErrors *ygres; TGraphErrors *ygmean; GetPoints(y, ygres, ygmean); TF1 *fun2 = new TF1("fun2", "[0]+x*([1]+x*([2]+x*([3])))"); fun2->SetLineColor(kGreen); fun2->SetLineWidth(1); fun2->SetParameter(0, 0.0); fun2->SetParameter(1, 1000.0/0.57); fun2->FixParameter(2, 0.0); fun2->FixParameter(3, 0.0); ygmean->Fit(fun2, "E"); axes = new TH2D("axes","", 100, 0.0, 2.0, 100, 0.0, 8000.0); axes->SetStats(0); axes->SetYTitle("#Sigma Geometric mean (fADC counts)"); axes->SetXTitle("Energy generated (GeV)"); axes->Draw(); y->Draw("same"); ygmean->Draw("Psame"); ygres->Draw("Psame"); // Write parameters out in form of C++ file const char *cppfname = "E_calibrate.cc"; ofstream pout(cppfname); pout<"<GetParameter(j); } pout<<" };"<GetParameter(j); } pout<<" };"<ProjectionX("_px", 1); // Define an asymmetric Gaussian function TF1 *f = new TF1("agaus", asymmetric_gaus, 0.0, 2.0, 4); // Loop over (uneven) bins int Npoints = 0; vector x; vector y; vector y_mean; vector yerr; vector y_mean_err; int Nbins = h->GetNbinsX(); for(int start_bin=1; start_bin<=Nbins; ){ // Find limits to give us min_entries entries int end_bin = start_bin; while(h_px->Integral(start_bin, end_bin)Integral(end_bin, Nbins)Integral()>=20){ #if 0 // Fit to Gaussian int status = h_py->Fit("gaus",""); if(status!=0){ start_bin++; continue; } TCanvas *c1 = (TCanvas*)gROOT->FindObject("c1"); c1->Update(); //c1->SaveAs("symmetric_gauss_fit.png"); //if(start_bin>30)return; // Get fit results double mean = h_py->GetFunction("gaus")->GetParameter(1); double sigma = h_py->GetFunction("gaus")->GetParameter(2); double mean_err = h_py->GetFunction("gaus")->GetParError(1); double sigma_err = h_py->GetFunction("gaus")->GetParError(2); #else // Fit to Asymmetric Gaussian f->SetParameter(0, h_py->GetMaximum()); f->SetParameter(1, h_py->GetBinCenter(h_py->GetMaximumBin())); f->SetParameter(2, 0.006); f->SetParameter(3, 0.006); Int_t status = h_py->Fit(f,"QE"); double max_sigma = (h_py->GetXaxis()->GetXmax()-h_py->GetXaxis()->GetXmin())/10.0; double sigma1 = fabs(h_py->GetFunction("agaus")->GetParameter(2)); double sigma2 = fabs(h_py->GetFunction("agaus")->GetParameter(3)); if(status==0 && (sigma1>max_sigma || sigma2>max_sigma))status = -1; if(status != 0){ // Fit failed cout<<"BAD FIT! Skipping start_bin and trying again ....(sigma1="<Clone(hname); TCanvas *c1 = (TCanvas*)gROOT->FindObject("c1"); if(c1)c1->Update(); //c1->SaveAs("asymmetric_gauss_fit.png"); //sleep(1); //if(start_bin>30)return; // Get fit results double mean = h_py->GetFunction("agaus")->GetParameter(1); double mean_err = h_py->GetFunction("agaus")->GetParError(1); // Variance of asymmetric Gaussian function about center // of individual Gaussians is given by // // // (s1^3 + s2^3) // sigma_tot^2 = ---------------------------- // (s1 + s2) // // // where s1 = sigma_1 and s2 = sigma_2 // // // The uncertainty on the variance is derived by taking // derivatives of the above. It ends up being: // // // 3[(s1^2)(s1_err) + (s2^2)(s2_err)] - (sigma_tot^2)[s1_err+s2_err] // sigma_tot_err = -------------------------------------------------------------------- // 2sigma_tot(s1 + s2) // // // // The variance about the mean of the asymmetric distribution // is given by // // sigma_mu^2 = sigma_tot^2 - mu^2 // // // The mean "mu" is given by // // (s2^2 - s1^2) // mu = sqrt(2/pi)* ----------------- // (s1 + s2) // // cout<<"sigma1="<GetParError(2); double sigma2_err = h_py->GetFunction("agaus")->GetParError(2); double sigma_err = 3.0*(pow(sigma1,2.0)*sigma1_err + pow(sigma2,2.0)*sigma2_err) - pow(sigma,2.0)*(sigma1_err+sigma2_err); sigma_err /= 2.0*sigma*(sigma1 + sigma2); mean += mu; #endif // Find average fADC value for hits in this bin range double fADC = 0.0; double norm = 0.0; for(int bin=start_bin; bin<=end_bin; bin++){ double weight = h_px->GetBinContent(bin); fADC += h_px->GetBinCenter(bin)*weight; norm += weight; } fADC /= norm; // Add floor term to error obtained from Gaussian fit double epsilon = 0.010; // GeV mean_err = sqrt(mean_err*mean_err + epsilon*epsilon); sigma_err = sqrt(sigma_err*sigma_err + epsilon*epsilon); x.push_back(fADC); y.push_back(sigma); y_mean.push_back(mean); yerr.push_back(sigma_err); y_mean_err.push_back(sigma_err); Npoints++; } // Increment for next iteration start_bin=end_bin+1; } gres = new TGraphErrors(Npoints, &x[0], &y[0], 0, &yerr[0]); gres->SetMarkerColor(kMagenta); gres->SetLineColor(kMagenta); gres->SetLineWidth(2); gmean = new TGraphErrors(Npoints, &x[0], &y_mean[0], 0, &y_mean_err[0]); gmean->SetMarkerColor(kRed); gmean->SetMarkerStyle(22); gmean->SetLineColor(kRed); } //----------------- // asymmetric_gaus //----------------- Double_t asymmetric_gaus(Double_t *xptr, Double_t *par) { double x = xptr[0]; double amp = par[0]; double mean = par[1]; double sigma = x