diff --git a/TrigScint/exampleConfigs/firmwareEx2.py b/TrigScint/exampleConfigs/firmwareEx2.py new file mode 100644 index 000000000..4ab19371a --- /dev/null +++ b/TrigScint/exampleConfigs/firmwareEx2.py @@ -0,0 +1,178 @@ +#!/bin/python + +import sys +import os +import json + +# we need the ldmx configuration package to construct the object + +from LDMX.Framework import ldmxcfg + +# set a 'pass name' +passName="sim" +p=ldmxcfg.Process(passName) + +#import all processors +from LDMX.SimCore import generators +from LDMX.SimCore import simulator +from LDMX.Biasing import filters + +from LDMX.Detectors.makePath import * +from LDMX.SimCore import simcfg + +#pull in command line options +nEle=4 # simulated beam electrons +runNum=10 +version="ldmx-det-v14" +outputNameString= "ldmxdetv14gap10mm_firmware.root" #sample identifier +outDir= "" #sample identifier + +# +# Instantiate the simulator. +# +sim = simulator.simulator("test") + +# +# Set the path to the detector to use (pulled from job config) +# +sim.setDetector( version, True ) +sim.scoringPlanes = makeScoringPlanesPath(version) + +outname=outputNameString #+".root" +print("NAME = " + outname) + +# +# Set run parameters. These are all pulled from the job config +# +p.run = runNum +p.maxEvents = 100 +nElectrons = nEle +beamEnergy = 4.0; #in GeV + +sim.description = "Inclusive "+str(beamEnergy)+" GeV electron events, "+str(nElectrons)+"e" +#sim.randomSeeds = [ SEED1 , SEED2 ] +sim.beamSpotSmear = [20., 80., 0] + + +mpgGen = generators.multi( "mgpGen" ) # this is the line that actually creates the generator +mpgGen.vertex = [ -44., 0., -880. ] # mm +mpgGen.nParticles = nElectrons +mpgGen.pdgID = 11 +mpgGen.enablePoisson = False #True + +import math +theta = math.radians(5.45) +beamEnergyMeV=1000*beamEnergy +px = beamEnergyMeV*math.sin(theta) +py = 0.; +pz= beamEnergyMeV*math.cos(theta) +mpgGen.momentum = [ px, py, pz ] + +# +# Set the multiparticle gun as generator +# +sim.generators = [ mpgGen ] + +#reconstruction and vetoes + +#Ecal and Hcal hardwired/geometry stuff +#import LDMX.Ecal.EcalGeometry +import LDMX.Ecal.ecal_hardcoded_conditions +from LDMX.Ecal import EcalGeometry +#egeom = EcalGeometry.EcalGeometryProvider.getInstance() +#Hcal hardwired/geometry stuff +from LDMX.Hcal import HcalGeometry +import LDMX.Hcal.hcal_hardcoded_conditions +#hgeom = HcalGeometry.HcalGeometryProvider.getInstance() + + +from LDMX.Ecal import digi as eDigi +from LDMX.Ecal import vetos +from LDMX.Hcal import digi as hDigi +from LDMX.Hcal import hcal + +from LDMX.Recon.simpleTrigger import TriggerProcessor + +from LDMX.TrigScint.trigScint import TrigScintDigiProducer +from LDMX.TrigScint.trigScint import TrigScintClusterProducer +from LDMX.TrigScint.trigScint import trigScintTrack +from LDMX.TrigScint.trigScint import TrigScintFirmwareTracker + +tsSimColls=[ "TriggerPad2SimHits", "TriggerPad3SimHits", "TriggerPad1SimHits" ] + +# ecal digi chain +# ecalDigi =eDigi.EcalDigiProducer('EcalDigis') +# ecalReco =eDigi.EcalRecProducer('ecalRecon') +# ecalVeto =vetos.EcalVetoProcessor('ecalVetoBDT') + +# #hcal digi chain +# hcalDigi =hDigi.HcalDigiProducer('hcalDigis') +# hcalReco =hDigi.HcalRecProducer('hcalRecon') +# hcalVeto =hcal.HcalVetoProcessor('hcalVeto') +# #hcalDigi.inputCollName="HcalSimHits" +#hcalDigi.inputPassName=passName + +# TS digi + clustering + track chain +tsDigisTag =TrigScintDigiProducer.pad2() +tsDigisTag.input_collection = tsSimColls[0]# +"_"+passName +tsDigisTag.input_pass_name = "sim" +tsDigisUp =TrigScintDigiProducer.pad3() +tsDigisUp.input_collection = tsSimColls[1]# +"_"+passName +tsDigisUp.input_pass_name = "sim" +tsDigisDown=TrigScintDigiProducer.pad1() +tsDigisDown.input_collection = tsSimColls[2]# +"_"+passName +tsDigisDown.input_pass_name = "sim" + +tsClustersTag =TrigScintClusterProducer.pad2() +tsClustersUp =TrigScintClusterProducer.pad1() +tsClustersDown =TrigScintClusterProducer.pad3() + + +tsDigisUp.verbosity=0 +tsClustersUp.verbosity=1 +trigScintTrack.verbosity=1 + +trigScintTrack.delta_max = 0.75 + +trigFirm = TrigScintFirmwareTracker( "trigFirm" ) +trigFirm.input_pass_name = "sim" +trigFirm.digis1_collection = "trigScintDigisPad1" +trigFirm.digis2_collection = "trigScintDigisPad2" +trigFirm.digis3_collection = "trigScintDigisPad3" +trigFirm.output_collection = "TriggerPadTracksFirmware" + +from LDMX.Recon.electronCounter import ElectronCounter +eCount = ElectronCounter( nElectrons, "ElectronCounter") # first argument is number of electrons in simulation +eCount.use_simulated_electron_number = False +eCount.input_collection="TriggerPadTracks" +eCount.input_pass_name=passName + +from LDMX.TrigScint.trigScint import TrigScintFirmwareHitProducer +from LDMX.TrigScint.trigScint import TrigScintQIEDigiProducer +from LDMX.TrigScint.trigScint import TrigScintRecHitProducer + +qieDigi = TrigScintQIEDigiProducer.pad3() +rechit = TrigScintRecHitProducer.pad3() +hitFirm = TrigScintFirmwareHitProducer( "hitFirm" ) + + + +# # p.sequence=[ sim, ecalDigi, ecalReco, ecalVeto, hcalDigi, hcalReco, hcalVeto, tsDigisTag, tsDigisUp, tsDigisDown, tsClustersTag, tsClustersUp, tsClustersDown, trigScintTrack, eCount ] +# #hcal digi keeps crashing in config step +p.sequence=[ sim, tsDigisTag, tsDigisUp, tsDigisDown, tsClustersTag, tsClustersUp, tsClustersDown, trigScintTrack, trigFirm, eCount, qieDigi, rechit, hitFirm] +# p.sequence=[sim] + +p.outputFiles=[outname] + +p.termLogLevel = 0 # default is 2 (WARNING); but then logFrequency is ignored. level 1 = INFO. + +#print this many events to stdout (independent on number of events, edge case: round-off effects when not divisible. so can go up by a factor 2 or so) +logEvents=20 +if p.maxEvents < logEvents : + logEvents = p.maxEvents +p.logFrequency = int( p.maxEvents/logEvents ) + +json.dumps(p.parameterDump(), indent=2) + +with open('parameterDump.json', 'w') as outfile: + json.dump(p.parameterDump(), outfile, indent=4) diff --git a/TrigScint/include/TrigScint/Firmware/hitproducer.h b/TrigScint/include/TrigScint/Firmware/hitproducer.h new file mode 100644 index 000000000..629c30a28 --- /dev/null +++ b/TrigScint/include/TrigScint/Firmware/hitproducer.h @@ -0,0 +1,11 @@ +#ifndef HITPRODUCER_H +#define HITPRODUCER_H + +#include "objdef.h" + +void copyHit1(Hit One, Hit Two); +void copyHit2(Hit One, Hit Two); +void hitproducer_ref(ap_uint<14> FIFO[NHITS][5],Hit outHit[NHITS],ap_uint<8> Peds[NHITS]); +void hitproducer_hw(ap_uint<14> FIFO[NHITS][5],Hit outHit[NHITS],ap_uint<8> Peds[NHITS]); + +#endif diff --git a/TrigScint/include/TrigScint/Firmware/objdef.h b/TrigScint/include/TrigScint/Firmware/objdef.h index 39c80cda4..f264308dc 100755 --- a/TrigScint/include/TrigScint/Firmware/objdef.h +++ b/TrigScint/include/TrigScint/Firmware/objdef.h @@ -3,7 +3,7 @@ #include "ap_int.h" #define NTIMES 6 -#define NHITS 25 +#define NHITS 50 #define NCLUS 25 #define NCHAN 50 #define NTRK 10 diff --git a/TrigScint/include/TrigScint/NumericalRecHitProducer.h b/TrigScint/include/TrigScint/NumericalRecHitProducer.h new file mode 100644 index 000000000..fd535efa2 --- /dev/null +++ b/TrigScint/include/TrigScint/NumericalRecHitProducer.h @@ -0,0 +1,130 @@ +/** + * @file NumericalRecHitProducer.h + * @brief Class that builds recHits + * @author Andrew Whitbeck, TTU + */ + +#ifndef TRIGSCINT_TRIGSCINTDIGIPRODUCER_H +#define TRIGSCINT_TRIGSCINTDIGIPRODUCER_H + +/*~~~~~~~~~~*/ +/* ROOT */ +/*~~~~~~~~~~*/ +#include "TRandom3.h" +#include "TVectorD.h" + +// LDMX +#include "DetDescr/TrigScintID.h" +#include "Recon/Event/EventConstants.h" +#include "Tools/NoiseGenerator.h" +#include "TrigScint/Event/TrigScintHit.h" +#include "TrigScint/Event/TrigScintQIEDigis.h" + +/*~~~~~~~~~~~~~~~*/ +/* Framework */ +/*~~~~~~~~~~~~~~~*/ +#include "Framework/Configure/Parameters.h" +#include "Framework/EventProcessor.h" + +/*~~~~~~~~~~~*/ +/* TrigScint */ +/*~~~~~~~~~~~*/ +#include "TrigScint/SimQIE.h" + +namespace trigscint { + +/** + * @class NumericalRecHitProducer + * @brief Organizes digis into TrigScintHits, linearizes TDC + * and ADC info, and converts amplitudes to PEs + + */ +class NumericalRecHitProducer : public framework::Producer { + public: + NumericalRecHitProducer(const std::string& name, framework::Process& process); + + ~NumericalRecHitProducer(); + + /** + * Callback for the processor to configure itself from the given set + * of parameters. + * + * @param parameters ParameterSet for configuration. + */ + void configure(framework::config::Parameters& parameters) final override; + + void produce(framework::Event& event); + + /** + * Const function for pulse fitting + * @param params an array of 2 elements specifying + * pulse arrival time and pulse amplitude (Total integral) + */ + double CostFunction(const double* params); + + /// QIE Sampling frequency (in MHz) + float qie_sf_{40.}; + + private: + /** + * Reconstruct true charge deposited in each time sample + * @param adc array of adcs for give event, cell + * @param tdc array of tdcs for give event, cell + * @param sample sample of interest + */ + Double_t ChargeReconstruction(std::vectoradc + ,std::vectortdc + ,int sample=2); + + /// Linearized charge. (Will be updated every time sample) + double Qm{0}; + + /// Time of crossing tdc threshold (Will be updated every time sample) + double tm{0}; + + /// QIE TDC Current threshold + float tdc_thr_; + + /// Class to set the verbosity level. + // TODO: Make use of the global verbose parameter. + bool verbose_{false}; + + /// Name of the input collection containing the sim hits + std::string inputCollection_; + + /// Name of the pass that the input collection is on (empty string means take + /// any pass) + std::string inputPassName_; + + /// Name of the output collection that will be used to stored the + /// digitized trigger scintillator hits + std::string outputCollection_; + + /// SiPM gain + double gain_{1e6}; + + /// QIE pedestal + double pedestal_{6.0}; + + /// QIE pedestal + double noise_{1.5}; + + /// Total MeV per MIP + double mevPerMip_{1.40}; + + /// Total number of photoelectrons per MIP + double pePerMip_{13.5}; + + /// Sample of interest + int sample_of_interest_{2}; + + /// Input pulse shape for fitting + std::string input_pulse_shape_; + + /// Input pulse parameters for fitting + std::vector pulse_params_; +}; + +} // namespace trigscint + +#endif diff --git a/TrigScint/include/TrigScint/TrigScintFirmwareHitProducer.h b/TrigScint/include/TrigScint/TrigScintFirmwareHitProducer.h new file mode 100644 index 000000000..46a003086 --- /dev/null +++ b/TrigScint/include/TrigScint/TrigScintFirmwareHitProducer.h @@ -0,0 +1,97 @@ +/** + * @file TrigScintFirmwareHitProducer.h + * @brief Staging of Real Hits + * @author Lene Kristian Bryngemark, Stanford University + */ + +#ifndef TRIGSCINT_TRIGSCINTFIRMWAREHITPRODUCER_H +#define TRIGSCINT_TRIGSCINTFIRMWAREHITPRODUCER_H + +/*~~~~~~~~~~*/ +/* ROOT */ +/*~~~~~~~~~~*/ +#include "TRandom3.h" + +// LDMX +#include "DetDescr/TrigScintID.h" +#include "Recon/Event/EventConstants.h" +#include "Tools/NoiseGenerator.h" +#include "TrigScint/Event/TrigScintHit.h" +#include "TrigScint/Event/TrigScintQIEDigis.h" + +/*~~~~~~~~~~~~~~~*/ +/* Framework */ +/*~~~~~~~~~~~~~~~*/ +#include "Framework/Configure/Parameters.h" +#include "Framework/EventProcessor.h" + +/*~~~~~~~~~~~*/ +/* TrigScint */ +/*~~~~~~~~~~~*/ +#include "TrigScint/SimQIE.h" +#include "TrigScint/Firmware/objdef.h" + + + +namespace trigscint { + +/** + * @class TrigScintFirmwareHitProducer + * @brief + */ +class TrigScintFirmwareHitProducer : public framework::Producer { + public: + TrigScintFirmwareHitProducer(const std::string& name, framework::Process& process) + : Producer(name, process) {} + + void configure(framework::config::Parameters& ps) override; + + void produce(framework::Event& event) override; + + + /** + * add a hit at index idx to a cluster + */ + + + private: + /// Class to set the verbosity level. + // TODO: Make use of the global verbose parameter. + bool verbose_{false}; + + /// Name of the input collection containing the sim hits + std::string inputCollection_; + + /// Name of the pass that the input collection is on (empty string means take + /// any pass) + std::string inputPassName_; + + /// Name of the output collection that will be used to stored the + /// digitized trigger scintillator hits + std::string outputCollection_; + + /// SiPM gain + double gain_{1e6}; + + /// QIE pedestal + double pedestal_{6.0}; + + /// Total MeV per MIP + double mevPerMip_{1.40}; + + /// Total number of photoelectrons per MIP + double pePerMip_{13.5}; + + /// Total number of photoelectrons per MIP + int sample_of_interest_{2}; + + std::string testCollection_; + + bool doTest_{true}; + + +}; + +} // namespace trigscint + +#endif /* TRIGSCINT_TRIGSCINTFIRMWAREHITPRODUCER_H */ diff --git a/TrigScint/python/trigScint.py b/TrigScint/python/trigScint.py index f8fcbec77..86b4e7ada 100644 --- a/TrigScint/python/trigScint.py +++ b/TrigScint/python/trigScint.py @@ -219,6 +219,97 @@ def pad3() : rechit.output_collection = 'trigScintRecHitsPad3' return rechit +class TrigScintFirmwareHitProducer(ldmxcfg.Producer) : + """Configuration for rechit producer for Trigger Scintillators incorporating validated Firmware, regular and pileUp""" + + def __init__(self,name) : + super().__init__(name,'trigscint::TrigScintFirmwareHitProducer','TrigScint') + + self .mev_per_mip = 0.4 #\ + # >>>both are for converting edep to PEs + self.pe_per_mip = 100. #/ + self.pedestal= 6.0 # QIE pedestal value (in fC) + self.gain = 1.e6 # SiPM Gain + self.input_collection="trigScintQIEDigisPad3" + self.test_collection="trigScintRecHitsPad3" + self.do_test=True + self.input_pass_name="" #take any pass + self.output_collection="trigScintFirmHitsPad3" + self.verbose = False + self.sample_of_interest=2 # Sample of interest. Range 0 to 3 + + def pad1() : + """Get the firmware hit producer for first pad""" + rechit = TrigScintRecHitProducer( 'trigScintFirmHitsPad1' ) + rechit.input_collection = 'trigScintQIEDigisPad1' + rechit.output_collection = 'trigScintFirmHitsPad1' + rechit.test_collection = 'trigScintRecHitsPad1' + rechit.do_test=True + return rechit + + def pad2() : + """Get the firmware hit producer for second pad""" + rechit = TrigScintRecHitProducer( 'trigScintFirmHitsPad2' ) + rechit.input_collection = 'trigScintQIEDigisPad2' + rechit.output_collection = 'trigScintFirmHitsPad2' + rechit.test_collection = 'trigScintRecHitsPad2' + rechit.do_test=True + return rechit + + def pad3() : + """Get the firmware hit for third pad""" + rechit = TrigScintRecHitProducer( 'trigScintFirmHitsPad3' ) + rechit.input_collection = 'trigScintQIEDigisPad3' + rechit.output_collection = 'trigScintFirmHitsPad3' + rechit.test_collection= 'trigScintRecHitsPad3' + rechit.do_test=True + return rechit + +class NumericalRecHitProducer(ldmxcfg.Producer) : + """Configuration for rechit producer for Trigger Scintillators""" + + def __init__(self,name) : + super().__init__(name,'trigscint::NumericalRecHitProducer','TrigScint') + + self .mev_per_mip = 0.4 #\ + # >>>both are for converting edep to PEs + self.pe_per_mip = 100. #/ + self.pedestal= 6.0 # QIE pedestal value (in fC) + self.elec_noise = 1.5 # QIE Electronic noise (in fC) + self.gain = 1.e6 # SiPM Gain + self.input_collection="trigScintQIEDigisUp" + self.input_pass_name="" #take any pass + self.output_collection="trigScintRecHitsUp" + self.verbose = False + self.sample_of_interest=2 # Sample of interest. Range 0 to 3 + + self.input_pulse_shape="Expo" # Name of the input pulse class + self.expo_k=0.1 # Inverse of decay time of piece-wise exponential + self.expo_tmax=5.0 # Time at which piece-wise exponential peaks + self.tdc_thr = 3.4 # Threshold current in uA for TDC latch (as used by QIE) + self.qie_sf = 40. # QIE sampling frequency in MHz + + def up() : + """Get the rechit producer for upstream pad""" + rechit = NumericalRecHitProducer( 'trigScintRecHitsUp' ) + rechit.input_collection = 'trigScintQIEDigisUp' + rechit.output_collection = 'trigScintRecHitsUp' + return rechit + + def down() : + """Get the rechit producer for downstream pad""" + rechit = NumericalRecHitProducer( 'trigScintRecHitsDown' ) + rechit.input_collection = 'trigScintQIEDigisDn' + rechit.output_collection = 'trigScintRecHitsDn' + return rechit + + def tagger() : + """Get the rechit producer for tagger pad""" + rechit = NumericalRecHitProducer( 'trigScintRecHitsTag' ) + rechit.input_collection = 'trigScintQIEDigisTag' + rechit.output_collection = 'trigScintRecHitsTag' + return rechit + class TrigScintClusterProducer(ldmxcfg.Producer) : """Configuration for cluster producer for Trigger Scintillators""" diff --git a/TrigScint/src/TrigScint/Firmware/hitproducer_hw.cxx b/TrigScint/src/TrigScint/Firmware/hitproducer_hw.cxx new file mode 100644 index 000000000..a612e3b73 --- /dev/null +++ b/TrigScint/src/TrigScint/Firmware/hitproducer_hw.cxx @@ -0,0 +1,126 @@ +#include +#include +#include "TrigScint/Firmware/objdef.h" +#include "TrigScint/Firmware/hitproducer.h" + +void hitproducer_hw(ap_uint<14> FIFO[NHITS][5],Hit outHit[NHITS],ap_uint<8> Peds[NHITS]){ + #pragma HLS ARRAY_PARTITION variable=FIFO complete + #pragma HLS ARRAY_PARTITION variable=amplitude complete + #pragma HLS ARRAY_PARTITION variable=Peds complete + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[0] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[1] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[2] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[3] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[4] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[5] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[6] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[7] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[8] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[9] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[10] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[11] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[12] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[13] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[14] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[15] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[16] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[17] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[18] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[19] + + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[20] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[21] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[22] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[23] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[24] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[25] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[26] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[27] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[28] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[29] + + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[30] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[31] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[32] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[33] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[34] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[35] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[36] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[37] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[38] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[39] + + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[40] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[41] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[42] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[43] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[44] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[45] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[46] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[47] + + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[48] + #pragma HLS INTERFACE ap_fifo depth=16 port=FIFO[49] + + + #pragma HLS PIPELINE + + /// Indices of first bin of each subrange + ap_uint<14> nbins_[5] = {0, 16, 36, 57, 64}; + + /// Charge lower limit of all the 16 subranges + ap_uint<14> edges_[17] = {0, 34, 158, 419, 517, 915, + 1910, 3990, 4780, 7960, 15900, 32600, + 38900, 64300, 128000, 261000, 350000}; + /// sensitivity of the subranges (Total charge/no. of bins) + ap_uint<14> sense_[16] = {3, 6, 12, 25, 25, 50, 99, 198, + 198, 397, 794, 1587, 1587, 3174, 6349, 12700}; + + for(int i = 0; i word1=FIFO[i][0];ap_uint<14> word2=FIFO[i][1];ap_uint<14> word3=FIFO[i][2]; + ap_uint<14> word4=FIFO[i][3];ap_uint<14> word5=FIFO[i][4];ap_uint<14> word6=FIFO[i][5]; + ap_uint<16> charge1;ap_uint<16> charge2;ap_uint<16> charge3; + ap_uint<16> charge4;ap_uint<16> charge5;ap_uint<16> charge6; + + ap_uint<14> rr = (word1>>6)/64; + ap_uint<14> v1 = (word1>>6)%64; + ap_uint<14> ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge1 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + rr = (word2>>6)/64; + v1 = (word2>>6)%64; + ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge2 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + rr = (word3>>6)/64; + v1 = (word3>>6)%64; + ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge3 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + rr = (word4>>6)/64; + v1 = (word4>>6)%64; + ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge4 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + rr = (word5>>6)/64; + v1 = (word5>>6)%64; + ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge5 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + rr = (word6>>6)/64; + v1 = (word6>>6)%64; + ss = 1*(v1>nbins_[1])+1*(v1>nbins_[2])+1*(v1>nbins_[3]); + charge6 = edges_[4*rr+ss]+(v1-nbins_[ss])*sense_[4*rr+ss]+sense_[4*rr+ss]/2-1; + + outHit[i].bID=i; + outHit[i].Time=(word1 & 63); + outHit[i].Amp=((charge1+charge2+charge3+charge4+charge5+charge6-36)*.00625); + } + + return; +} + diff --git a/TrigScint/src/TrigScint/NumericalRecHitProducer.cxx b/TrigScint/src/TrigScint/NumericalRecHitProducer.cxx new file mode 100644 index 000000000..08da89015 --- /dev/null +++ b/TrigScint/src/TrigScint/NumericalRecHitProducer.cxx @@ -0,0 +1,279 @@ +#include "TrigScint/NumericalRecHitProducer.h" +#include "Framework/Exception/Exception.h" +#include "Framework/RandomNumberSeedService.h" +#include "TMath.h" + +#include "Math/Minimizer.h" +#include "Math/Factory.h" +#include "Math/Functor.h" + +#include +#include "TLinearFitter.h" +#include + +namespace trigscint { + + float tm_new=0; + std::vector Qm_new; + float TimePeriod=0; // time periodd of 1 TS + float k_,tmax_; + int npulses=0; + double NewCostFunction(const double* params); + +NumericalRecHitProducer::NumericalRecHitProducer(const std::string &name, + framework::Process &process) + : Producer(name, process) {} + +NumericalRecHitProducer::~NumericalRecHitProducer() {} + +void NumericalRecHitProducer::configure( + framework::config::Parameters ¶meters) { + // Configure this instance of the producer + pedestal_ = parameters.getParameter("pedestal"); + noise_ = parameters.getParameter("elec_noise"); + gain_ = parameters.getParameter("gain"); + mevPerMip_ = parameters.getParameter("mev_per_mip"); + pePerMip_ = parameters.getParameter("pe_per_mip"); + inputCollection_ = parameters.getParameter("input_collection"); + inputPassName_ = parameters.getParameter("input_pass_name"); + outputCollection_ = parameters.getParameter("output_collection"); + verbose_ = parameters.getParameter("verbose"); + sample_of_interest_ = parameters.getParameter("sample_of_interest"); + tdc_thr_ = parameters.getParameter("tdc_thr"); + qie_sf_ = parameters.getParameter("qie_sf"); + + input_pulse_shape_ = + parameters.getParameter("input_pulse_shape"); + if (input_pulse_shape_ == "Expo") { + pulse_params_.clear(); + pulse_params_.push_back(parameters.getParameter("expo_k")); + pulse_params_.push_back(parameters.getParameter("expo_tmax")); + + k_ = pulse_params_[0]; + tmax_ = pulse_params_[1]; + + ldmx_log(debug) << "expo_k =" << pulse_params_[0]; + ldmx_log(debug) << "expo_tmax =" << pulse_params_[1]; + } + +} + + void NumericalRecHitProducer::produce(framework::Event &event) { + // initialize QIE object for linearizing ADCs + SimQIE qie; + + // Ensure the sample of interest <4 + if(sample_of_interest_>3) { + ldmx_log(error)<<"sample_of_interest_ should be one of 0,1,2,3\n" + <<"Currently, sample_of_interest = "<( + inputCollection_, inputPassName_)}; + + std::vector trigScintHits; + for (const auto &digi : digis) { + ldmx::TrigScintHit hit; + auto adc{digi.getADC()}; + auto tdc{digi.getTDC()}; + + hit.setModuleID(0); + hit.setBarID(digi.getChanID()); + hit.setBeamEfrac(-1.); + + + if (tdc[sample_of_interest_] > 49) + hit.setTime(-999.); + else + hit.setTime(tdc[sample_of_interest_] * 0.5); + + auto Charge = + ChargeReconstruction(adc,tdc,sample_of_interest_); + + hit.setAmplitude(Charge); + hit.setEnergy(Charge * 6250. / + gain_ * mevPerMip_ / pePerMip_); // MeV + hit.setPE(Charge * 6250. /gain_); + trigScintHits.push_back(hit); + } + // Create the container to hold the + // digitized trigger scintillator hits. + + event.add(outputCollection_, trigScintHits); +} + Double_t NumericalRecHitProducer::ChargeReconstruction + (std::vectoradc,std::vectortdc,int sample) { + + npulses = 0; // No. of true pulses + int poi=0; // The pulse of interest + std::vector Charge_; // stores pulse amplitudes + std::vector InitVal; // Initial values of fitting parameters + auto Qdata = new Double_t[5]; // Linearized charge + float tend = 1000/qie_sf_; // 1 time sample (in ns) + TimePeriod = tend; + SimQIE qie; + auto pulse = new Expo(k_,tmax_); + + // Initialize the minimizer. + ROOT::Math::Minimizer* min = + ROOT::Math::Factory::CreateMinimizer("Minuit","Migrad"); + + + // if(tdc[sample]==63) return(0); // no signal pulse + + Qm_new.clear(); + for(int i=0;iSetFunction(f); + + for(int n = 0;nSetVariable(2*n,id,InitVal[2*n],0.1); + //sprintf(id,"Q%d",n+1); + min->SetVariable(2*n+1,id,InitVal[2*n+1],1); + } + + min->SetMaxFunctionCalls(1000000); // for Minuit/Minuit2 + min->SetMaxIterations(10000); // for GSL + min->SetTolerance(0.001); + // if(verbose_) + // min->SetPrintLevel(1); + + min->Minimize(); + const double *pred = min->X(); + + // pulse->AddPulse(tend*i+pred[0],pred[1]); + // } + + /////////////// For Debigging purposes + // if(verbose_ || pred[2*poi+1]<0) { + // if(verbose_ || pred[2*poi+1]*6250./gain_ * mevPerMip_ / pePerMip_> 1.6) { + if(verbose_){ + std::cout<<"TS \t|\t0\t|\t1\t|\t2\t|\t3\t|\t4\t|\n" + <<"---------------------------------------------" + <<"--------------------------------------------\n" + <<"tdc \t|"; + for(int i=0;i<5;i++) + std::cout<AddPulse(pred[2*n],pred[2*n+1]); + for(int i=0;i<5;i++) + std::cout<Integrate(i*tend,(i+1)*tend)<<"\t|"; + std::cout<<" Q= "<AddPulse(params[2*n],params[2*n+1]); + + // std::cout<<"\nQpred\t|"; + for(int i=0;i<5;i++){ + double Qpred = pulse->Integrate(i*TimePeriod,(i+1)*TimePeriod); + cost += pow(Qm_new[i]-Qpred,2); + // std::cout<AddPulse(params[0],params[1]); + pulse->AddPulse(params[2],params[3]); + + double Qpred = pulse->Integrate(0,1000/qie_sf_); + double Tpred = qie.TDC(pulse,0)/2; + + double cost1 = pow(tm-Tpred,2); + double cost2 = pow(Qm-Qpred,2); + return(cost1+cost2); + } +} // namespace trigscint + +DECLARE_PRODUCER_NS(trigscint, NumericalRecHitProducer); diff --git a/TrigScint/src/TrigScint/TrigScintFirmwareHitProducer.cxx b/TrigScint/src/TrigScint/TrigScintFirmwareHitProducer.cxx new file mode 100644 index 000000000..44b18a566 --- /dev/null +++ b/TrigScint/src/TrigScint/TrigScintFirmwareHitProducer.cxx @@ -0,0 +1,93 @@ + +#include "TrigScint/TrigScintFirmwareHitProducer.h" +#include "TrigScint/Firmware/hitproducer.h" +#include "TrigScint/Firmware/objdef.h" +#include +#include + +namespace trigscint { + +void TrigScintFirmwareHitProducer::configure(framework::config::Parameters &ps) { + pedestal_ = ps.getParameter("pedestal"); + gain_ = ps.getParameter("gain"); + mevPerMip_ = ps.getParameter("mev_per_mip"); + pePerMip_ = ps.getParameter("pe_per_mip"); + inputCollection_ = ps.getParameter("input_collection"); + testCollection_ = ps.getParameter("test_collection"); + doTest_ = ps.getParameter("do_test"); + inputPassName_ = ps.getParameter("input_pass_name"); + outputCollection_ = ps.getParameter("output_collection"); + verbose_ = ps.getParameter("verbose"); + sample_of_interest_ = ps.getParameter("sample_of_interest"); + if (verbose_) { + ldmx_log(info) << "In TrigScintFirmwareHitProducer: configure done!"; + ldmx_log(info) << "\nPedestal: " << pedestal_ + << "\nGain: " << gain_ + << "\nMEV per MIP: " << mevPerMip_ + << "\nPE per MIP: " << pePerMip_ + << "\ninput collection: " << inputCollection_ + << "\ntest collection: " << testCollection_ + << "\nAre we testing: " << doTest_ + << "\nInput pass name: " << inputPassName_ + << "\nOutput collection: " << outputCollection_ + << "\nVerbosity: " << verbose_; + } + + return; +} + +void TrigScintFirmwareHitProducer::produce(framework::Event &event) { + if(doTest_){ + const auto rechits{event.getCollection( + testCollection_, inputPassName_)}; + for(const auto &hit : rechits){ + std::cout<<"Analysis barID: "<( + inputCollection_, inputPassName_)}; + Hit outHit[NHITS]; + ap_uint<14> FIFO[NCHAN][5]; + ap_uint<8> Peds[NCHAN]; + for(int i = 0; i < NCHAN; i++){ + Peds[i]=0; + FIFO[i][0]=(Peds[i]<<6)+63; + FIFO[i][1]=(Peds[i]<<6)+63; + FIFO[i][2]=(Peds[i]<<6)+63; + FIFO[i][3]=(Peds[i]<<6)+63; + FIFO[i][4]=(Peds[i]<<6)+63; + } + std::cout<<"DID I GET HERE 3"< adcs=digi.getADC(); + std::vector tdcs=digi.getTDC(); + for(int i = 0;i<5;i++){ + FIFO[(int)(digi.getChanID())][i]=(ap_uint<14>)((adcs[i]<<6)+(tdcs[i])); + } + } + std::cout<<"DID I GET HERE 4"< trigScintHits; + for(int i=0;i=3){ + if(doTest_){ + std::cout<<"Firmware barID: "<