#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; #include "StandardLabels.C" #include "E_calibrate.cc" #include "Angle.h" #include "Geom.h" void GetPoints(TH2D *h, TGraphErrors* &gres, TGraphErrors* &gmean); //------------------------------------ // 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; //------------------------------------ //----------------- // Eres_vs_E //----------------- void Eres_vs_E(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.200, 2.0, 200, -0.5, 0.5); TCanvas *c1 = new TCanvas("c1"); c1->SetTicks(); c1->SetGrid(); // 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){ E_tot = BCAL_fADC2E(E_tot); // convert to GeV h->Fill(E_thrown, E_tot - E_thrown); 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 = 3.0/LSB_MeV_fADC; // threshold for 3.0 attenuated MeV in fADC counts fADC_threshold = floor(0.5 + fADC_threshold); // integerize if(hit.fADC_upSetParameter(0, 0.05); fun->SetParameter(1, 0.03); fun->SetParameter(2, 0.0); gres->Fit(fun, "0E"); // Now, convert function and data to DeltaE/E fo plotting // I'm not sure if this is necessary, but it seems most // resolution plots are of sigmaE/E. The fitting to find // sigmaE though seems like it should be done to DeltaE for(int i=0; iGetN(); i++){ double E, DeltaE; gres->GetPoint(i, E, DeltaE); DeltaE /= E; gres->SetPoint(i, E, DeltaE); } TF1 *fun2 = new TF1("fun2", "sqrt(pow([0]/sqrt(x),2.0) + pow([1],2.0) + pow([2]/x,2.0))", 0.0, 2.0); fun2->SetParameter(0, fun->GetParameter(0)); fun2->SetParameter(1, fun->GetParameter(1)); fun2->SetParameter(2, fun->GetParameter(2)); //gres->Fit(fun2); TH2D *axes = new TH2D("axes","", 100, 0.0, 2.0, 100, 0.0, 1.0); axes->SetStats(0); axes->SetYTitle("#sigma_{E}/E"); axes->SetXTitle("Generated Energy (GeV)"); axes->Draw(); gres->Draw("Psame"); fun2->Draw("same"); char str[256]; sprintf(str, "#frac{#sigma_{E}}{E} = #frac{(%3.1f#pm%3.1f)%%}{#sqrt{E}} #oplus (%3.1f#pm%3.1f)%% #oplus #frac{(%3.1f#pm%3.1f)%%}{E}" , fun->GetParameter(0)*100.0, fun->GetParError(0)*100.0 , fun->GetParameter(1)*100.0, fun->GetParError(1)*100.0 , fun->GetParameter(2)*100.0, fun->GetParError(2)*100.0); TLatex *lab = new TLatex(1.2, 0.7, str); lab->SetTextSize(0.05); lab->SetTextAlign(22); lab->Draw(); StandardLabels(axes, Scheme(), "", AngleStr("#theta_{#gamma}=")); c1->SaveAs("Eres_vs_E.pdf"); c1->SaveAs("Eres_vs_E.png"); // Write parameter to a file so they can be read in and used // to automatically generate the LaTeX table later ofstream ofs("Eres_parms.dat"); ofs<GetParameter(0)*100.0<<" "<GetParError(0)*100.0<GetParameter(1)*100.0<<" "<GetParError(1)*100.0<GetParameter(2)*100.0<<" "<GetParError(2)*100.0<ProjectionX("_px", 1); // Loop over (uneven) bins int Npoints = 0; vector x; vector y; vector y_mean; vector xerr; 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){ // 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); // 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); // Error on reconstructed energy is difficult to estimate here. // It is definitely worse than the final algorithm will yield. // The fact that the "reconstructed" energy extends out to 2.5GeV // when the generated data set only went to 2GeV says it is // considerably worse than 5.5%/sqrt(E). For lack of a better // idea, I use the RMS of the fADC values about the mean // in the current range. double fADC_rms = 0.0; double norm_rms = 0.0; for(int bin=start_bin; bin<=end_bin; bin++){ double weight = h_px->GetBinContent(bin); fADC_rms += pow(h_px->GetBinCenter(bin)-fADC, 2.0)*weight; norm_rms += weight; } fADC_rms /= norm_rms; fADC_rms = sqrt(fADC_rms); x.push_back(fADC); y.push_back(sigma); y_mean.push_back(mean); xerr.push_back(fADC_rms); 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], &xerr[0], &yerr[0]); gres->SetMarkerColor(kMagenta); gres->SetLineColor(kMagenta); gres->SetLineWidth(2); gmean = new TGraphErrors(Npoints, &x[0], &y_mean[0], &xerr[0], &y_mean_err[0]); gmean->SetMarkerColor(kRed); gmean->SetMarkerStyle(22); gmean->SetLineColor(kRed); }