-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathSBSGEMModule.h
755 lines (591 loc) · 40.9 KB
/
SBSGEMModule.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
#ifndef SBSGEMModule_H
#define SBSGEMModule_H
#include "THaSubDetector.h"
#include <vector>
#include <set>
#include <map>
#include <array>
#include <deque>
//using namespace std;
class THaDetectorBase;
class THaEvData;
class THaRunBase;
class TH1D;
class TH2D;
class TF1;
class TClonesArray;
namespace SBSGEM {
enum GEMaxis_t { kUaxis=0, kVaxis };
enum APVmap_t { kINFN=0, kUVA_XY, kUVA_UV, kMC };
}
struct mpdmap_t {
UInt_t crate;
UInt_t slot;
UInt_t mpd_id;
UInt_t gem_id;
UInt_t adc_id;
UInt_t i2c;
UInt_t pos;
UInt_t invert;
UInt_t axis; //needed to add axis to the decode map
UInt_t index;
};
//Clustering results can be held in a simple C struct, as a cluster is just a collection of basic data types (not even arrays):
//Each module will have an array of clusters as a data member:
struct sbsgemhit_t { //2D reconstructed hits
//Module and layer info: I think we don't need here because it's already associated with the module containing the cluster!
//Track info:
Bool_t keep; //Should this cluster be considered for tracking? We use this variable to implement "cluster quality" cuts (thresholds, XY ADC and time correlation, etc.)
Bool_t ontrack; //Is this cluster on any track?
Bool_t highquality; //is this a "high quality" hit?
Int_t trackidx; //Index of track containing this cluster (within the array of tracks found by the parent SBSGEMTracker
UInt_t iuclust; //Index in (1D) U cluster array of the "U" cluster used to define this 2D hit.
UInt_t ivclust; //Index in (1D) V cluster array of the "V" cluster used to define this 2D hit.
//Strip info: should we store a list of strips? We shouldn't need to if the clustering algorithm requires all strips in a cluster to be contiguous:
//Some of this information is probably redundant with 1D clustering results stored in sbsgemcluster_t, but it may or may not be more efficient to store a copy here:
/* UShort_t nstripu; //Number of strips in cluster along "U" axis */
/* UShort_t nstripv; //Number of strips in cluster along "V" axis */
/* UShort_t ustriplo; //lowest u strip index in cluster */
/* UShort_t ustriphi; //highest u strip index in cluster */
/* UShort_t ustripmaxADC; //index of U strip with largest ADC in cluster */
/* UShort_t vstriplo; //lowest v strip index in cluster */
/* UShort_t vstriphi; //highest v strip index in cluster; */
/* UShort_t vstripmaxADC; //index of V strip with largest ADC in cluster */
//Coordinate info:
Double_t umom; //umean - u of center of strip with max ADC (could be used to refine hit coordinate reconstruction, probably unnecessary), in units of strip pitch
Double_t vmom; //vmean - v of center of strip with max ADC (could be used to refine hit coordinate reconstruction, probably unnecessary), in units of strip pitch
Double_t uhit; //Reconstructed U position of cluster in local module/strip coordinates
Double_t vhit; //Reconstructed V position of cluster in local module/strip coordinates
Double_t xhit; //Reconstructed "X" position of cluster in local module/strip coordinates
Double_t yhit; //Reconstructed "Y" position of cluster in local module/strip coordinates
Double_t xghit; //"Global" X coordinate of hit (in coordinate system of parent SBSGEMTracker: usually spectrometer TRANSPORT coordinate)
Double_t yghit; //"Global" Y coordinate of hit (in coordinates of parent SBSGEMTracker)
Double_t zghit; //"Global" Z coordinate of hit (in coordinates of parent SBSGEMTracker)
//
Double_t Ehit; //Sum of all ADC values on all strips in the cluster: actually 1/2*( ADCX + ADCY ); i.e., average of cluster sums in X and Y
/* Double_t Euhit; //Sum of all ADC values on U strips in the cluster; */
/* Double_t Evhit; //Sum of all ADC values on V strips in the cluster; */
Double_t thit; //Average of ADC-weighted mean U strip time and V strip time
/* Double_t tuhit; //Average time of U strips in cluster; */
/* Double_t tvhit; //Average time of V strips in cluster; */
Double_t thitcorr; //"Corrected" hit time (with any trigger or other corrections we might want to apply)
/* Double_t tuhitcorr; //"Corrected" U hit time (with any trigger time or other corrections) */
/* Double_t tvhitcorr; //"Corrected" V hit time */
Double_t ADCasym; //(ADCU-ADCV)/(ADCU+ADCV)
Double_t tdiff; //tu - tv
Double_t corrcoeff_clust; //"Cluster" level XY correlation coefficient
Double_t corrcoeff_strip; //Correlation coefficient of best XY strip pair, used to "seed" the cluster
Double_t corrcoeff_clust_deconv; //"Cluster" level XY correlation coefficient, deconvoluted samples
Double_t corrcoeff_strip_deconv; //Correlation coefficient of maximum X and Y strips
Double_t ADCasymDeconv; //Same as ADC asym but for deconvoluted ADC "max combo"
Double_t EhitDeconv; //Same as "Ehit" but for deconvoluted ADC "max combo"
Double_t thitDeconv; // same as "thit" but using deconvoluted ADC samples
Double_t tdiffDeconv; // same as "tdiff" but using deconvoluted ADC Samples.
Double_t thitFit; //same as "thit" but using "fit times";
Double_t tdiffFit; //same as "tdiff" but using "fit times";
};
struct sbsgemcluster_t { //1D clusters;
UInt_t nstrips;
UInt_t istriplo;
UInt_t istriphi;
UInt_t istripmax;
UInt_t isampmax; //time sample in which the cluster-summed ADC samples peaks:
UInt_t isampmaxDeconv; //time sample in which the deconvoluted cluster-summed ADC samples peaks.
UInt_t icombomaxDeconv; //2nd time sample of max two-sample combo for cluster-summed deconvoluted ADC samples
std::vector<Double_t> ADCsamples; //cluster-summed ADC samples (accounting for split fraction)
//New variables:
std::vector<Double_t> DeconvADCsamples; //cluster-summed deconvoluted ADC ssamples (accounting for split fraction)
Double_t hitpos_mean; //ADC-weighted mean coordinate along the direction measured by the strip
Double_t hitpos_sigma; //ADC-weighted RMS coordinate deviation from the mean along the direction measured by the strip
Double_t clusterADCsum; //Sum of ADCs over all samples on all strips, accounting for split fraction and first/last sample suppression
Double_t clusterADCsumDeconv; //sum of all deconvoluted ADC samples over all strips in the cluster
Double_t clusterADCsumDeconvMaxCombo; //sum over all strips in the cluster of max two-sample combo
std::vector<Double_t> stripADCsum; //Sum of individual strip ADCs over all samples on all strips; accounting for split fraction
std::vector<Double_t> DeconvADCsum; //Sum of individual deconvoluted ADC samples over all samples on all strips; accounting for split fraction
Double_t t_mean; //reconstructed hit time
Double_t t_sigma; //unclear what we might use this for
Double_t t_mean_deconv; //cluster-summed mean deconvoluted hit time.
Double_t t_mean_fit; //cluster-summed "fit" time.
//Do we want to store the individual strip ADC Samples with the 1D clustering results? I don't think so; as these can be accessed via the decoded strip info.
std::vector<UInt_t> hitindex; //position in decoded hit array of each strip in the cluster:
UInt_t rawstrip; //Raw APV strip number before decoding
UInt_t rawMPD; //Raw MPD number before decoding
UInt_t rawAPV; //Raw APV number before decoding
bool isneg; //Cluster is from negative strips
bool isnegontrack; //Cluster is from negative strips
bool keep;
bool ontrack;
};
//Should these be hardcoded? Probably not!
/* #define N_APV25_CHAN 128 */
/* #define N_MPD_TIME_SAMP 6 */
/* #define MPDMAP_ROW_SIZE 8 */
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// GEM module class: a GEM module consists of one part of a tracking unit;
// Basic assumptions:
// 1. Rectangular active area
// 2. Strip readout based on two non-parallel strip orientations, denoted "U" and "V", with some constant number of time samples
// 3. A logical GEM layer or GEM plane will consist of one or more GEM modules, and a GEM "Tracker"
// will consist of one or more GEM "layers" or "planes". We need a separate "module" class, but we probably don't need a separate "GEM layer" class,
// since the layer will merely be an integer index that logically groups one or more modules together as a single plane for purposes of tracking.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//Is the use of THaSubDetector appropriate here? Or should we just use THaDetector or similar?
class SBSGEMModule : public THaSubDetector {
public:
explicit SBSGEMModule( const char *name, const char *description = "",
THaDetectorBase* parent = nullptr );
virtual ~SBSGEMModule();
virtual void Clear( Option_t* opt="" ); //should be called once per event
virtual Int_t Decode( const THaEvData& );
virtual void Print( Option_t* opt="" ) const;
virtual Int_t ReadDatabase(const TDatime& );
virtual Int_t DefineVariables( EMode mode );
//We are overriding THaDetectorBase's ReadGeometry method for SBSGEMModule, because we use a different definition of the angles:
virtual Int_t ReadGeometry( FILE* file, const TDatime& date,
Bool_t required = true );
virtual Int_t Begin( THaRunBase* r=0 );
virtual Int_t End( THaRunBase* r=0 );
void SetMakeCommonModePlots( int cmplots=0 ){ fMakeCommonModePlots = cmplots != 0; fCommonModePlots_DBoverride = true; }
//Don't call this method directly, it is called by find_2Dhits. Call that instead:
void find_clusters_1D(SBSGEM::GEMaxis_t axis, Double_t constraint_center=0.0, Double_t constraint_width=1000.0); //Assuming decode has already been called; this method is fast so we probably don't need to implement constraint points and widths here, or do we?
//new version to do clustering and spatial splitting in each time sample, and then combine the different time samples
void find_clusters_1D_experimental(SBSGEM::GEMaxis_t axis, Double_t constraint_center=0.0, Double_t constraint_width=1000.0);
void add_constraint( TVector2 constraint_center, TVector2 constraint_width );
void find_2Dhits(); // Version with no arguments assumes no constraint points
//void find_2Dhits(TVector2 constraint_center, TVector2 constraint_width); // Version with TVector2 arguments
// fill the 2D hit arrays from the 1D cluster arrays:
void fill_2D_hit_arrays();
//Filter 1D hits by criteria possibly to include ADC threshold, cluster size
void filter_1Dhits(SBSGEM::GEMaxis_t axis);
//Filter 2D hits by criteria possibly to include ADC X/Y asymmetry, cluster size, time correlation, (lack of) overlap, possibly others:
void filter_2Dhits();
//Utility function to calculate correlation coefficient between U and V time samples:
Double_t CorrCoeff( int nsamples, const std::vector<double> &Usamples, const std::vector<double> &Vsamples, int firstsample=0 );
Double_t StripTSchi2( int hitindex );
//Utility functions to compute "module local" X and Y coordinates from U and V (strip coordinates) to "transport" coordinates (x,y) and vice-versa:
TVector2 UVtoXY( TVector2 UV );
TVector2 XYtoUV( TVector2 XY );
//function to convert from APV channel number to strip number ordered by position:
Int_t GetStripNumber( UInt_t rawstrip, UInt_t pos, UInt_t invert );
void PrintPedestals( std::ofstream &dbfile_CM, std::ofstream &daqfile_ped, std::ofstream &daqfile_CM );
double GetCommonMode( UInt_t isamp, Int_t flag, const mpdmap_t &apvinfo, UInt_t nhits=128 ); //default to "sorting" method:
//Simulation of common-mode correction algorithm as applied to full readout events:
double GetCommonModeCorrection( UInt_t isamp, const mpdmap_t &apvinfo, UInt_t &ngood, const UInt_t &nhits=128, bool fullreadout=false, Int_t flag=0 );
void fill_ADCfrac_vs_time_sample_goodstrip( Int_t hitindex, bool max=false );
double FitStripTime( int striphitindex, double RMS=20.0 ); // "dumb" fit method
void FitClusterTime( sbsgemcluster_t &clus ); //calculate "fit time" for cluster-summed ADC samples
double CalcFitTime( const std::vector<Double_t> &samples, double RMS=20.0 );
//Since we have two different kinds of rolling averages to evaluate, we consolidate both codes into one method.
//Since we can only declare references with initialization, we have to pass the underlying arrays as arguments:
void UpdateRollingAverage( int iapv, double val, std::vector<std::deque<Double_t> > &RC, std::vector<Double_t> &AVG, std::vector<Double_t> &RMS, std::vector<UInt_t> &Nevt );
void UpdateRollingCommonModeAverage(int iapv, double CM_sample);
//void UpdateRollingCommonModeBias( int iapv, double CM_bias );
void CalcDeconvolutedSamples( const std::vector<Double_t> &ADCs, std::vector<Double_t> &DeconvADCs );
void SetTriggerTime( Double_t ttrig );
TF1 *fStripTimeFunc;
bool fIsDecoded;
//UShort_t GetLayer() const { return fLayer; }
//std::vector<sbsgemhit_t> GetHitList() { return fHits; }
//If we are going to declare all these data members private, we will need to write public getters and setters for at least the information required by the tracker classes:
//private:
//Let's just make all this stuff public because we're lazy. A lot of this is used by the track-finding routines
//anyway
//Decode map information:
std::vector<mpdmap_t> fMPDmap; //this may need to be modified
std::vector<Int_t> fChanMapData;
// vectors to keep track of rolling common-mode mean and RMS, using sorting method.
// These will follow the same ordering as fMPDmap:
//std::vector<Double_t> fCommonModeRollingFirstEvent_by_APV; //need to keep track of first event in the rolling average for updating the calculation:
std::vector<std::deque<Double_t> > fCommonModeResultContainer_by_APV;
std::vector<Double_t> fCommonModeRollingAverage_by_APV;
std::vector<Double_t> fCommonModeRollingRMS_by_APV;
std::vector<UInt_t> fNeventsRollingAverage_by_APV;
std::vector<std::deque<Double_t> > fCMbiasResultContainer_by_APV;
std::vector<Double_t> fCommonModeOnlineBiasRollingAverage_by_APV;
std::vector<Double_t> fCommonModeOnlineBiasRollingRMS_by_APV;
std::vector<UInt_t> fNeventsOnlineBias_by_APV;
SBSGEM::APVmap_t fAPVmapping; //choose APV channel --> strip mapping; there are only three possible values supported for now (see SBSGEM::APVmap_t)
std::array<std::vector<UInt_t>, 4 > APVMAP;
void InitAPVMAP();
//some convenience maps: are these actually used yet?
std::map<Int_t, Int_t> fAPVch_by_Ustrip;
std::map<Int_t, Int_t> fAPVch_by_Vstrip;
std::map<Int_t, Int_t> fMPDID_by_Ustrip;
std::map<Int_t, Int_t> fMPDID_by_Vstrip;
std::map<Int_t, Int_t> fADCch_by_Ustrip;
std::map<Int_t, Int_t> fADCch_by_Vstrip;
//Bool_t firstevent=
//////////// MPD Time Stamp Info: //////////////////////////////////
//// An MPD is uniquely defined by crate, slot, and fiber
// For time stamp info, we basically want to store one per APV card, although many of
// these will be redundant with each other:
//
std::vector<double> fT0_by_APV; //T0 for each APV card (really for each MPD)
double fTref_coarse; //T0 of the "reference" channel (time stamp coarse - T0 of the "reference" APV)
std::vector<double> fTcoarse_by_APV; //Time stamp coarse - T0 of this channel - Tref for each channel
std::vector<UInt_t> fTfine_by_APV; //"Fine time stamp" by APV:
std::vector<UInt_t> fEventCount_by_APV; //MPD event counter by APV:
std::vector<double> fTimeStamp_ns_by_APV; //Coarse time stamp - T0 + fine time stamp % 6
//What should we use to hold the rolling average of common-mode means?
Bool_t fPedestalMode;
//Bool_t fPedestalsInitialized;
Bool_t fSubtractPedBeforeCommonMode;
Double_t fZeroSuppressRMS;
Bool_t fZeroSuppress;
Bool_t fNegSignalStudy;
//Moved to the MPD module class:
Bool_t fOnlineZeroSuppression; //this MIGHT be redundant with fZeroSuppress (or not)
Int_t fCODA_BUILD_ALL_SAMPLES;
Int_t fCODA_CM_ENABLED;
Int_t fCommonModeFlag; //default = 0 = sorting method, 1 = Danning method, 2 = histogramming method, 3 = "online" Danning-method
Int_t fCommonModeOnlFlag; //default = 3 = Danning method during GMn, 4 = Danning method during GEn
Int_t fPedSubFlag; //default = 0 (pedestal subtraction NOT done for full readout events).
// 1 = pedestal subtraction WAS done online, even for full readout events, only subtract the common-mode
Double_t fCommonModeDanningMethod_NsigmaCut;
Int_t fSuppressFirstLast; // Suppress strips peaking in first or last time sample:
Int_t fUseStripTimingCuts; // Apply strip timing cuts:
Double_t fStripTau; //time constant for strip timing fit
Double_t fDeconv_weights[3]; //
//Make these fixed-size arrays defined per strip axis direction, this will require some painful code changes but will improve efficiency and S/N
Double_t fStripMaxTcut_central[2], fStripMaxTcut_width[2], fStripMaxTcut_sigma[2]; // Strip timing cuts for local maximum used to seed cluster
Double_t fStripMaxTcut_central_deconv[2], fStripMaxTcut_width_deconv[2], fStripMaxTcut_sigma_deconv[2]; //Strip timing cuts based on deconvoluted strip time
Double_t fStripMaxTcut_central_fit[2], fStripMaxTcut_width_fit[2], fStripMaxTcut_sigma_fit[2]; //Strip timing cuts based on strip "fit" time.
Double_t fStripAddTcut_width; //Time cut for adding strips to a cluster
//Double_t fStripAddTcut_sigma;
Double_t fStripAddCorrCoeffCut; //cut on correlation coefficient for adding neighboring strips to a cluster.
//These are database parameters used to reject out-of-time background strips:
bool fUseTSchi2cut;
std::vector<double> fGoodStrip_TSfrac_mean; //should have same dimension as number of APV25 time samples:
std::vector<double> fGoodStrip_TSfrac_sigma; //
Double_t fStripTSchi2Cut;
//In principle we will eventually also require some time walk corrections
Bool_t fMeasureCommonMode; //Default = false; use full-readout events to measure the common-mode mean and rms in real time
UInt_t fNeventsCommonModeLookBack; // number of events to use for rolling average; default = 100
Bool_t fCorrectCommonMode; //Default = false; use rolling common-mode mean and calculated common-mode values to detect a condition where negative pulses bias the common-mode down and attempt to correct it.
UInt_t fCorrectCommonModeMinStrips;
Double_t fCorrectCommonMode_Nsigma;
//Strip cuts:
//Number of strips on low and high side to reject for common-mode calculation:
Int_t fCommonModeNstripRejectHigh; //default = 28;
Int_t fCommonModeNstripRejectLow; //default = 28;
Int_t fCommonModeNumIterations; //number of iterations for Danning Method: default = 3
Int_t fCommonModeMinStripsInRange; //Minimum strips in range for Danning Method: default = 10;
Double_t fCommonModeBinWidth_Nsigma; //Bin width for "histogramming-method" common-mode calculation, in units of the RMS in a particular APV, default = 2
Double_t fCommonModeScanRange_Nsigma; //Scan range for common-mode histogramming method calculation, in units of RMS, +/- Nsigma about the mean, default = 4
Double_t fCommonModeStepSize_Nsigma; //Step size in units of RMS, default = 1/5:
//same as in MPDModule: unavoidable to duplicate this definition unless someone smarter than I cam
//can come up with a way to do so:
UInt_t fChan_CM_flags;
UInt_t fChan_TimeStamp_low;
UInt_t fChan_TimeStamp_high;
UInt_t fChan_MPD_EventCount;
UInt_t fChan_MPD_Debug;
Double_t fTrigTime; //trigger time; to be decoded once by parent class
Double_t fMaxTrigTimeCorrection; //Maximum (absolute) correction to be applied to strip times based on trigger time (Default = 25 ns)
Double_t fTrigTimeSlope; //Slope of GEM time versus trig time correlation (default = 1)
//move these to trackerbase:
//Double_t fSigma_hitpos; //sigma parameter controlling resolution entering track chi^2 calculation
//Double_t fSigma_hitshape; //Sigma parameter controlling hit shape for cluster-splitting algorithm.
//Note on ROOT data types:
// Char_t/UChar_t is one byte signed unsigned = 8 bits = 0..255 for unsigned
// Short_t/UShort_t is two bytes signed/unsigned = 16 bits = 0..65535 for unsigned
// Int_t/UInt_t is four bytes signed/unsigned = 32 bits
// Long_t/ULong_t is eight bytes = 64 bit
// Double_t/Double_t is four bytes/eight bytes
UChar_t fN_APV25_CHAN; //number of APV25 channels, default 128
UChar_t fN_MPD_TIME_SAMP; //number of MPD time samples, default = 6
UShort_t fMPDMAP_ROW_SIZE; //MPDMAP_ROW_SIZE: default = 9, let's not hardcode
UShort_t fNumberOfChannelInFrame; //default 128, not clear if this is used: This might not be constant, in fact
Double_t fSamplePeriod; //for timing calculations: default = 24 ns for Hall A .
//variables defining rectangular track search region constraint (NOTE: these will change event-to-event, they are NOT constant!)
std::vector<Double_t> fxcmin, fxcmax;
std::vector<Double_t> fycmin, fycmax;
//Arrays to temporarily hold raw data from ONE APV card:
std::vector<UInt_t> fStripAPV;
std::vector<UInt_t> fRawStripAPV;
std::vector<Int_t> fRawADC_APV;
std::vector<Double_t> fPedSubADC_APV;
std::vector<Double_t> fCommonModeSubtractedADC_APV;
//Let's store the online calculated common-mode in its own dedicated array as well:
std::vector<Double_t> fCM_online; //size equal to fN_MPD_TIME_SAMP
//BASIC DECODED STRIP HIT INFO:
//By the time the information is populated here, the ADC values are already assumed to be pedestal/common-mode subtracted and/or zero-suppressed as appropriate:
Int_t fNstrips_hit; //total Number of strips fired (after common-mode subtraction and zero suppression)
Int_t fNstrips_hit_pos; //total Number of strips fired after positive zero suppression
Int_t fNstrips_hit_neg; //total Number of strips fired after negative zero suppression
Int_t fNdecoded_ADCsamples; //= fNstrips_hit * fN_MPD_TIME_SAMP
UInt_t fNstrips_hitU; //total number of U strips fired
UInt_t fNstrips_hitV; //total number of V strips fired
UInt_t fNstrips_hitU_neg; //total number of U strips fired negative
UInt_t fNstrips_hitV_neg; //total number of V strips fired negative
// Number of strips passing basic zero suppression thresholds:
UInt_t fNstrips_keep;
UInt_t fNstrips_keepU;
UInt_t fNstrips_keepV;
//Number of strips passing "local max" thresholds:
UInt_t fNstrips_keep_lmax;
UInt_t fNstrips_keep_lmaxU;
UInt_t fNstrips_keep_lmaxV;
UInt_t fTrackPassedThrough; //flag to indicate track passed through module:
//Map strip indices in this array:
//key = U or V strip number, mapped value is position of that strip's information in the "decoded strip" arrays below:
/* std::map<UInt_t, UInt_t> fUstripIndex; */
/* std::map<UInt_t, UInt_t> fVstripIndex; */
//std::vector<sbsgemstrip_t> fDecodedStrips;
std::vector<UInt_t> fStrip; //Strip index of hit (these could be "U" or "V" generalized X and Y), assumed to run from 0..N-1
std::vector<SBSGEM::GEMaxis_t> fAxis; //We just made our enumerated type that has two possible values, makes the code more readable (maybe)
std::vector<std::vector<Double_t> > fADCsamples; //2D array of ADC samples by hit: Outer index runs over hits; inner index runs over ADC samples
std::vector<std::vector<Int_t> > fRawADCsamples; //2D array of raw (non-baseline-subtracted) ADC values.
std::vector<std::vector<Double_t> > fADCsamples_deconv; //"Deconvoluted" ADC samples
std::vector<Double_t> fADCsums;
std::vector<Double_t> fADCsumsDeconv; //deconvoluted strip ADC sums
std::vector<Double_t> fStripADCavg;
std::vector<UInt_t> fStripIsU; // is this a U strip? 0/1
std::vector<UInt_t> fStripIsV; // is this a V strip? 0/1
std::vector<UInt_t> fStripOnTrack; //Is this strip on any track?
std::vector<UInt_t> fStripIsNeg; //Is this strip negative?
std::vector<UInt_t> fStripIsNegU; //Is this strip negative?
std::vector<UInt_t> fStripIsNegV; //Is this strip negative?
std::vector<UInt_t> fStripIsNegOnTrack; //Is this strip negative and on a track?
std::vector<UInt_t> fStripIsNegOnTrackU; //Is this strip negative and on a track?
std::vector<UInt_t> fStripIsNegOnTrackV; //Is this strip negative and on a track?
std::vector<UInt_t> fStripRaw; //Raw strip numbers on track?
std::vector<UInt_t> fStripEvent; //strip raw info
std::vector<UInt_t> fStripCrate; //strip raw info
std::vector<UInt_t> fStripMPD; //strip raw info
std::vector<UInt_t> fStripADC_ID; //strip raw info
std::vector<Int_t> fStripTrackIndex; // If this strip is included in a cluster that ends up on a good track, we want to record the index in the track array of the track that contains this strip.
std::vector<bool> fKeepStrip; //keep this strip?
//std::vector<bool> fNegStrip; //Does this strip pass negative zero suppression? - neg pulse study
//std::vector<Int_t> fStripKeep; //Strip passes timing cuts (and part of a cluster)?
std::vector<UInt_t> fMaxSamp; //APV25 time sample with maximum ADC;
std::vector<UInt_t> fMaxSampDeconv; //Sample with largest deconvoluted ADC value
std::vector<UInt_t> fMaxSampDeconvCombo; //last sample of two-sample combination with largest deconvoluted ADC
std::vector<Double_t> fADCmax; //largest ADC sample on the strip:
std::vector<Double_t> fADCmaxDeconv; //Largest deconvoluted ADC sample
std::vector<Double_t> fADCmaxDeconvCombo; //Largest combination of two deconvoluted samples
std::vector<Double_t> fTmeanDeconv; //mean deconvoluted strip time
std::vector<Double_t> fTmean; //ADC-weighted mean strip time:
std::vector<Double_t> fTsigma; //ADC-weighted RMS deviation from the mean
std::vector<Double_t> fStripTfit; //Dumb strip fit:
std::vector<Double_t> fStripTdiff; //strip time diff wrt max strip in cluster
std::vector<Double_t> fStripTSchi2; //strip time-sample chi2 wrt "good" pulse shape
std::vector<Double_t> fStripTSprob; //p-value associated with strip TS chi2
std::vector<Double_t> fStripCorrCoeff; //strip correlation coeff wrt max strip in cluster
std::vector<Double_t> fTcorr; //Strip time with all applicable corrections; e.g., trigger time, etc.
std::vector<UInt_t> fStrip_ENABLE_CM; //Flag to indicate whether CM was done online or offline for this strip
std::vector<UInt_t> fStrip_CM_GOOD; //Flag to indicate whether online CM succeeded
std::vector<UInt_t> fStrip_BUILD_ALL_SAMPLES; //Flag to indicate whether online zero suppression was enabled
//because the cut definition machinery sucks, let's define some more booleans:
std::vector<UInt_t> fStripUonTrack;
std::vector<UInt_t> fStripVonTrack;
////// (1D and 2D) Clustering results (see above for definition of struct sbsgemcluster_t and struct sbsgemhit_t):
Bool_t fClustering1DIsDone;
Bool_t fStoreAll1Dclusters; //default false; this is set based on the "fMultiTrackSearch" of the parent trackerbase
UInt_t fNclustU; // number of U clusters found
UInt_t fNclustV; // number of V clusters found
UInt_t fNclustU_pos; // number of positive U clusters found
UInt_t fNclustV_pos; // number of positive V clusters found
UInt_t fNclustU_neg; // number of negative U clusters found
UInt_t fNclustV_neg; // number of negative V clusters found
UInt_t fNclustU_total; // Number of U clusters found in entire active area, without enforcing search region constraint
UInt_t fNclustV_total; // Number of U clusters found in entire active area, without enforcing search region constraint
std::vector<sbsgemcluster_t> fUclusters; //1D clusters along "U" direction
std::vector<sbsgemcluster_t> fVclusters; //1D clusters along "V" direction
UInt_t fMAX2DHITS; // Max. 2d hits per module, to limit memory usage:
UInt_t fN2Dhits; // number of 2D hits found in region of interest:
std::vector<sbsgemhit_t> fHits; //2D hit reconstruction results
/////////////////////// Global variables that are more convenient for ROOT Tree/Histogram Output (to the extent needed): ///////////////////////
//Raw strip info:
//std::vector<UInt_t> fStripAxis; //redundant with fAxis, but more convenient for Podd global variable definition
std::vector<Double_t> fADCsamples1D; //1D array to hold ADC samples; should end up with dimension fNstrips_hit*fN_MPD_TIME_SAMP
std::vector<Int_t> fRawADCsamples1D;
std::vector<Double_t> fADCsamplesDeconv1D; //1D array of deconvoluted ADC samples
/////////////////////// End of global variables needed for ROOT Tree/Histogram output ///////////////////////
//////////////////////////////////////////////////////////////////////////////////////////////
// Define some global variables for tests/cuts //
//bool fDidHit; //a good track passed within the active area of this module AND a hit occurred within some small distance from the projected track
//bool fShouldHit; //a good track passed within the active area of this module
//moved the above to TrackerBase
//////////////////////////////////////////////////////////////////////////////////////////////
//Constant, module-specific parameters:
UShort_t fModule; // Module index within a tracker. Should be unique! Since this is a GEM module class, this parameter should be unchanging
UShort_t fLayer; // Layer index of this module within a tracker. Since this is a GEM module class, this parameter should be unchanging
UInt_t fNstripsU; // Total number of strips in this module along the generic "U" axis
UInt_t fNstripsV; // Total number of strips in this module along the generic "V" axis
//Pedestal means and RMS values for all channels:
std::vector<Double_t> fPedestalU, fPedRMSU;
std::vector<Double_t> fPedestalV, fPedRMSV;
Double_t fRMS_ConversionFactor; // = sqrt(fN_MPD_TIME_SAMP);
//To be determined from channel map/strip count information:
UInt_t fNAPVs_U; //Number of APV cards per module along "U" strip direction; this is typically 8, 10, or 12, but could be larger for U/V GEMs
UInt_t fNAPVs_V; //Number of APV cards per module along "V" strip direction;
//UInt_t fNTimeSamplesADC; //Number of ADC time samples (this could be variable in principle, but should be the same for all strips within a module within a run) redundant with fN_MPD_TIME_SAMP
// Should we define the parameters controlling cluster-finding in the Module class? Why not:
std::vector<Double_t> fUgain; // Internal "gain match" coefficients for U strips by APV card;
std::vector<Double_t> fVgain; // Internal "gain match" coefficients for V strips by APV card;
Double_t fModuleGain; // Module gain relative to some "Target" ADC value:
//Optional database parameters for monitoring common-mode fluctuations in pedestal runs and/or full readout events:
std::vector<Double_t> fCommonModeMeanU;
std::vector<Double_t> fCommonModeMeanV;
std::vector<Double_t> fCommonModeRMSU;
std::vector<Double_t> fCommonModeRMSV;
std::vector<Double_t> fCMbiasU;
std::vector<Double_t> fCMbiasV;
double fCommonModeRange_nsigma; //default = 5
///////////////////////////////////////////////////////////////////////////////////////////////////
// CLUSTERING PARAMETERS (to be read from database and/or given sensible default values) //
///////////////////////////////////////////////////////////////////////////////////////////////////
//Making our clustering algorithm a little more "fancy":
Int_t fClusteringFlag; //Which quantities to use for clustering?
Int_t fDeconvolutionFlag; //reject strips failing basic deconvolution criteria?
Double_t fThresholdSample; //Threshold on the (gain-matched and pedestal-subtracted) max. sample on a strip to keep that strip for clustering
Double_t fThresholdStripSum; //Threshold on the sum of (pedestal-subtracted) ADC samples on a strip
Double_t fThresholdClusterSum; //Threshold on the sum of (pedestal-subtracted) ADC samples
Double_t fThresholdSampleDeconv; //Threshold on the max. deconbo
Double_t fThresholdDeconvADCMaxCombo; //Threshold on the maximal sum of two consecutive deconvoluted ADC samples;
Double_t fThresholdClusterSumDeconv;
Double_t fADCratioSigma; // sigma of ADCV/ADCU-1
Double_t fADCasymCut,fADCasymSigma; // Filtering criterion for ADC X/Y (or U/V) asymmetry
Double_t fTimeCutUVdiff,fTimeCutUVsigma; // Filtering criterion for ADC X/Y (or U/V) time difference (this is distinct from any timing cuts relative to
//trigger or reference timing at the individual strip level
Double_t fTimeCutUVdiffDeconv,fTimeCutUVsigmaDeconv;
Double_t fTimeCutUVdiffFit,fTimeCutUVsigmaFit;
Double_t fHitTimeMean[2], fHitTimeSigma[2];
Double_t fHitTimeMeanDeconv[2], fHitTimeSigmaDeconv[2];
Double_t fHitTimeMeanFit[2], fHitTimeSigmaFit[2];
Double_t fSigmaHitTimeAverageCorrected;
Double_t fCorrCoeffCut; //Filtering of 2D clusters based on ADC correlation
Double_t fCorrCoeffCutDeconv; //Filtering of 2D clusters based on
Int_t fFiltering_flag1D; // controls whether 1D cluster filtering criteria are strict requirements or "soft" requirements
Int_t fFiltering_flag2D; // controls whether 2D hit association criteria are strict requirements of "soft" requirements
//Parameters controlling cluster splitting and insignificant peak elimination based on "peak prominence" calculation
Double_t fThresh_2ndMax_nsigma; //Number of sigmas above noise level for minimum peak prominence in splitting overlapping clusters
Double_t fThresh_2ndMax_fraction; //Peak prominence threshold as a fraction of peak height for splitting overlapping clusters
UShort_t fMaxNeighborsU_totalcharge; //Only strips within +/- fMaxNeighborsU of the peak can be added to a cluster for total charge calculation
UShort_t fMaxNeighborsV_totalcharge; //Only strips within +/- fMaxNeighborsV of the peak can be added to a cluster for total charge calculation
//Only strips within these limits around the peak can be used for hit position reconstruction
UShort_t fMaxNeighborsU_hitpos;
UShort_t fMaxNeighborsV_hitpos;
Double_t fSigma_hitshape; //Sigma parameter controlling hit shape for cluster-splitting algorithm.
//GEOMETRICAL PARAMETERS:
Double_t fUStripPitch; //strip pitch along U, will virtually always be 0.4 mm
Double_t fVStripPitch; //strip pitch along V, will virtually always be 0.4 mm
Double_t fUStripOffset; //position of first U strip along the direction it measures:
Double_t fVStripOffset; //position of first V sttrip alogn the direction it measures:
Double_t fUAngle; //Angle between U strips and "X" axis of TRANSPORT coordinates;
Double_t fVAngle; //Angle between V strips and "X" axis of TRANSPORT coordinates;
Double_t fPxU; //U Strip X projection = cos( UAngle );
Double_t fPyU; //U Strip Y projection = sin( UAngle );
Double_t fPxV; //V Strip X projection = cos( VAngle );
Double_t fPyV; //V Strip Y projection = sin( VAngle );
//Note: These size variables are redundant with THaDetectorBase
//Double_t fLx; //Full width of module active area along "X" (usually + dispersive direction)
//Double_t fLy; //Full width of module active area along "Y" (usually + dispersive direction);
//Alignment: These alignment variables are redundant with THaDetectorBase, so re-use those:
/* TVector3 fModPos; //Module position in BigBite TRANSPORT coordinates (or other global coordinate system); position within the overall tracking unit: */
/* Double_t fModAlphaX; //Module rotational offset about X axis of BigBite TRANSPORT coordinates (or other global coordinate system) */
/* Double_t fModAlphaY; //Module rotational offset about Y axis of BigBite TRANSPORT coordinates (or other global coordinate system) */
/* Double_t fModAlphaZ; //Module rotational offset about Z axis of BigBite TRANSPORT coordinates (or other global coordinate system) */
/* TRotation fModRot; //Module rotation matrix to go from local --> global */
/* TRotation fModRotInv; //Module rotation matrix to go from global --> local */
//For geometry parameters, we will re-use the THaDetectorBase functionalities as much as possible
Bool_t fIsMC;//we kinda want this guy no matter what don't we...
//Efficiency histograms:
TH1D *fhdidhitx;
TH1D *fhdidhity;
TH2D *fhdidhitxy;
TH1D *fhshouldhitx;
TH1D *fhshouldhity;
TH2D *fhshouldhitxy;
//These will be copied down from GEMtrackerbase
double fBinSize_efficiency2D;
double fBinSize_efficiency1D;
//we should let the user configure this: this is set at the "tracker level" which then propagates down to all the modules:
bool fMakeEfficiencyPlots;
bool fEfficiencyInitialized;
bool fMakeCommonModePlots; //diagnostic plots for offline common-mode stuff: default = false;
bool fCommonModePlotsInitialized;
bool fCommonModePlots_DBoverride;
bool fMakeEventInfoPlots;
bool fEventInfoPlotsInitialized;
//Let's revamp the common-mode diagnostic plots altogether:
//If applicable, make common mode. In principle these should all be broken down by APV, but let's leave as 1D for now.
TH2D *fCommonModeDistU; //user
TH2D *fCommonModeDistU_Histo; //Distribution of calculated common-mode (minus expected common-mode mean) using chosen method:
TH2D *fCommonModeDistU_Sorting;
TH2D *fCommonModeDistU_Danning;
TH2D *fCommonModeDiffU; //difference between "user" and "online" danning-mode calculations:
TH2D *fCommonModeDiffU_Uncorrected;
TH2D *fCommonModeCorrectionU; //common-mode correction applied
//TH2D *fCommonModeNotCorrectionU;
//For full readout events that WEREN'T corrected, plot the difference between "online Danning method" and "true" common-mode:
TH2D *fCommonModeResidualBiasU; //for full-readout events only: "online corrected" - "true" common-mode vs APV
TH2D *fCommonModeResidualBias_vs_OccupancyU; //for full-readout events only: "online corrected" - "true" common-mode vs. occupancy
TH2D *fCommonModeResidualBiasU_corrected;
//If applicable, make common mode. In principle these should all be broken down by APV, but let's leave as 1D for now.
TH2D *fCommonModeDistV;
TH2D *fCommonModeDistV_Histo; //Distribution of calculated common-mode (minus expected common-mode mean) using chosen method:
TH2D *fCommonModeDistV_Sorting;
TH2D *fCommonModeDistV_Danning;
TH2D *fCommonModeDiffV; //difference between "user" and "online" danning mode calculations
TH2D *fCommonModeDiffV_Uncorrected;
TH2D *fCommonModeCorrectionV;
//TH2D *fCommonModeNotCorrectionV;
TH2D *fCommonModeResidualBiasV; //for full-readout events only: "online corrected" - "true" common-mode vs APV
TH2D *fCommonModeResidualBias_vs_OccupancyV; //for full-readout events only: "online corrected" - "true" common-mode vs.
TH2D *fCommonModeResidualBiasV_corrected;
//Pedestal plots: only generate if pedestal mode = true:
bool fPedHistosInitialized;
TH2D *hrawADCs_by_stripU; //raw adcs by strip, no corrections, filled for each SAMPLE:
TH2D *hrawADCs_by_stripV; //raw adcs by strip, no corrections, filled for each SAMPLE:
TH2D *hcommonmode_subtracted_ADCs_by_stripU; //common-mode subtracted ADCS without ped subtraction
TH2D *hcommonmode_subtracted_ADCs_by_stripV;
TH2D *hpedestal_subtracted_ADCs_by_stripU; //common-mode AND pedestal subtracted ADCs
TH2D *hpedestal_subtracted_ADCs_by_stripV;
TH2D *hpedestal_subtracted_rawADCs_by_stripU; //pedestal-subtracted ADCs w/o common-mode correction
TH2D *hpedestal_subtracted_rawADCs_by_stripV;
//Summed over all strips, pedestal-subtracted (but not common-mode subtracted) ADCs:
TH1D *hpedestal_subtracted_rawADCsU;
TH1D *hpedestal_subtracted_rawADCsV;
//Summed over all strips, pedestal-subtracted and common-mode subtracted ADCs:
TH1D *hpedestal_subtracted_ADCsU;
TH1D *hpedestal_subtracted_ADCsV;
TH1D *hdeconv_ADCsU; //full readout events only
TH1D *hdeconv_ADCsV; //full readout events only
//in pedestal-mode analysis, we
TH2D *hcommonmode_mean_by_APV_U;
TH2D *hcommonmode_mean_by_APV_V;
TH1D *hpedrmsU_distribution;
TH1D *hpedrmsU_by_strip;
TH1D *hpedmeanU_distribution;
TH1D *hpedmeanU_by_strip;
TH1D *hpedrmsV_distribution;
TH1D *hpedrmsV_by_strip;
TH1D *hpedmeanV_distribution;
TH1D *hpedmeanV_by_strip;
TClonesArray *hpedrms_by_APV;
TH1D *hMPD_EventCount_Alignment;
TH2D *hMPD_EventCount_Alignment_vs_Fiber;
TH2D *hMPD_FineTimeStamp_vs_Fiber;
//Pulse-shape diagnostics:
bool fPulseShapeInitialized;
TH2D *hADCfrac_vs_timesample_allstrips; //All fired strips
TH2D *hADCfrac_vs_timesample_goodstrips; //strips on good tracks
TH2D *hADCfrac_vs_timesample_maxstrip; //max strip in cluster on good track
//Let's make hard-coded, automated cluster-based "occupancy" histos for tracking, with or without external constraint:
//Basic idea: count number of found clusters within constraint region, normalize to area of search region and some kind of "effective time window":
TH1D *hClusterBasedOccupancyUstrips;
TH1D *hClusterBasedOccupancyVstrips;
TH1D *hClusterMultiplicityUstrips;
TH1D *hClusterMultiplicityVstrips;
//Comment out for now, uncomment later if we deem these interesting:
// TClonesArray *hrawADCs_by_strip_sampleU;
// TClonesArray *hrawADCs_by_strip_sampleV;
// TClonesArray *hcommonmode_subtracted_ADCs_by_strip_sampleU;
// TClonesArray *hcommonmode_subtracted_ADCs_by_strip_sampleV;
// TClonesArray *hpedestal_subtracted_ADCs_by_strip_sampleU;
// TClonesArray *hpedestal_subtracted_ADCs_by_strip_sampleV;
ClassDef(SBSGEMModule,0);
};
#endif