]> jspc29.x-matter.uni-frankfurt.de Git - daqtools.git/commitdiff
current state mt
authorYour Name <you@example.com>
Fri, 8 Oct 2021 09:40:14 +0000 (11:40 +0200)
committerYour Name <you@example.com>
Fri, 8 Oct 2021 09:40:14 +0000 (11:40 +0200)
users/ufsd_trb3_sdalinac/TdcEventBuilder.xml [new file with mode: 0644]
users/ufsd_trb3_sdalinac/db/register_configgbe.db [new file with mode: 0644]
users/ufsd_trb3_sdalinac/db/register_configgbe_ip.db [new file with mode: 0644]
users/ufsd_trb3_sdalinac/db/register_configtdc.db [new file with mode: 0644]
users/ufsd_trb3_sdalinac/first.C [new file with mode: 0644]
users/ufsd_trb3_sdalinac/second.C [new file with mode: 0644]
users/ufsd_trb3_sdalinac/startup.sh [new file with mode: 0755]

diff --git a/users/ufsd_trb3_sdalinac/TdcEventBuilder.xml b/users/ufsd_trb3_sdalinac/TdcEventBuilder.xml
new file mode 100644 (file)
index 0000000..8e28c5d
--- /dev/null
@@ -0,0 +1,154 @@
+<?xml version="1.0"?>
+
+<!-- 
+This is example file how HADAQ event building should be configured in DABC.
+Event building process implemented in Combiner module of hadaq::CombinerModule class.
+Module can have several inputs, each with separate port number for receiving data 
+from TRB boards. For every input one could configure only UDP port number - host name is ignored.
+First output of combiner module reserved for the connection to the MBS transmitter module. 
+Second output can be use to store data in hld files. 
+To enable HLD file storage one should specify NumOutputs for Combiner module
+and provide hld file name for Output1 port   
+
+     <NumOutputs value="2"/>
+     <OutputPort name="Output1" url="hld://dabc.hld?maxsize=2000"/>
+     
+File name fill be extended according to HADAQ naming convetion.       
+
+Optionally one can enable MBS transmitter module, which converts HLD to LMD format.
+To enable transmitter, one should specify auto="true" in configuration which says DABC
+to automatically create module when starting application. 
+First output of the module reserved for stream server, second output can be used to
+store data in lmd files. 
+
+By default, HTTP server is enabled. Do disable it, remove <HttpServer> section or
+put <HttpServer name="http" auto="false">. One could change http port number. 
+When dabc runs, in any browser address like
+http://your_dabc_host_name:8090 can be opened. At the moment http server provides: 
+  - ratemeters from EventBuilder and Transmitter
+  - log fields 
+  - commands to start/stop hld and lmd files from browser
+
+It is also possible to attach go4 analysis to that server, that also histograms
+from online analysis will appear. For that one should specify 
+"-dabc your_dabc_host_name" argument when starting analysis. Like:
+   [shell] go4analysis -stream dabc_node -dabc dabc_node
+When launching analysis from the gui, extra arguments "-dabc your_dabc_host_name" 
+should be specified.
+
+There is well-known problem with using VNC viewer and mbs stream server. 
+Both by default are using port 6002. One could change port number for stream server.
+Just set other number in configuration of output port of transmitter module, for instance
+       <OutputPort name="Output0" url="mbs://Stream:6789"/>
+In this case one should specify that port number when starting go4 analysis like:
+   [shell] go4analysis -stream dabc_node:6789 -dabc dabc_node:4444 
+When starting analysis from the go4 gui, one should specify stream server with port number too.
+
+-->
+
+<dabc version="2">
+  <Context host="localhost" name="EventBuilder">
+    <Run>
+      <lib value="libDabcMbs.so"/>
+      <lib value="libDabcHadaq.so"/>  
+      <lib value="libDabcHttp.so"/>  
+      <lib value="libDabcStream.so"/>  
+      <logfile value="hadaqevtbuild.log"/>
+      <loglevel value="-1"/>
+      <loglimit value="1000"/>
+      <control value="true"/>
+      <threads_layout value="balanced"/>
+    </Run>
+    
+    <HttpServer name="http" port="8091"/>
+    
+    <!-- If uncommented, all internal manager structures will be published in the web server -->
+    <!-- Publisher name="publ" manager="true"/ -->
+    
+    <!-- If uncommented, profiling will be enabled for all threads -->
+    <!-- Thread name="*" publ="true" prof="true"/ -->
+    
+    <MemoryPool name="Pool">
+       <BufferSize value="400000"/>
+       <NumBuffers value="400"/>
+    </MemoryPool>
+    
+    <!-- these are default parameters for TDC calibration modules -->
+    <Module name="Input*TdcCal">    
+       <InputPort name="Input*" queue="30"/>
+
+       <FineMin value="31"/>
+       <FineMax value="515"/>
+       <NumChannels value="33"/>
+       <EdgeMask value="2"/> <!-- 1: only leading edge 2:leading/trailing individual 3:statistics from leading used 4:statistics merged -->
+       <HistFilling value="4"/>
+       <CalibrFile value="local"/>
+       <!--DisableCalibrationFor value="0"/-->  
+       <Auto value="100000"/>
+       
+       <!-- TRB value="0x8010"/-->
+       <!-- HUB value="0x9000"/-->
+       <!-- TDC value="[0x8a00,0x8a01,0x8a02,0x8a03]"/-->
+       <Dummy value="true"/>
+    </Module>
+    
+
+    <Module name="Combiner" class="hadaq::CombinerModule">    
+        <!-- these parameters will force to create inputs/outputs of module -->
+       <NumInputs value="1"/>
+       <NumOutputs value="1"/>
+
+       <InputPort name="Input0" url="hadaq://host:50094" urlopt1="" thread="inp2thrd"/>
+       <InputPort name="Input1" url="hadaq://host:50061" urlopt1="" thread="inp1thrd"/>
+<!--       <InputPort name="Input1" url="hadaq://host:50113" urlopt1=""/> -->
+       <InputPort name="Input2" url="hadaq://host:50113" urlopt1="" thread="inp3thrd"/>
+       <InputPort name="Input3" url="hadaq://host:50158" urlopt1="" thread="inp0thrd"/>       
+       <InputPort name="Input4" url="hadaq://host:10104" urlopt1="trb=0x8030&tdc=[0x3030,0x3031,0x3032,0x3033]&dummy"/>
+       <InputPort name="Input5" url="hadaq://host:10105" urlopt1="trb=0x8040&tdc=[0x3040,0x3041,0x3042,0x3043]&dummy"/>
+
+       <!-- <InputPort name="Input0" url="hadaq://host:50021" urlopt1="trb=0x8000&hub=0x8100&tdc=[0xC000,0xC001,0xC003]&dummy"/> -->
+       <!-- <InputPort name="Input1" url="hadaq://host:50094" urlopt1="trb=0x8003&tdc=[0xC00C,0xC00D,0xC00E,0xC00F]&dummy"/> -->
+       <!-- <InputPort name="Input2" url="hadaq://host:50096" urlopt1="trb=0x8002&tdc=[0xC008,0xC009,0xC00A,0xC00B]&dummy"/> -->
+       <!-- <InputPort name="Input3" url="hadaq://host:10104" urlopt1="trb=0x8030&tdc=[0x3030,0x3031,0x3032,0x3033]&dummy"/> -->
+       <!-- <InputPort name="Input4" url="hadaq://host:10105" urlopt1="trb=0x8040&tdc=[0x3040,0x3041,0x3042,0x3043]&dummy"/> -->
+
+       <InputPort name="Input*" queue="30" urlopt="udpbuf=200000&mtu=64512&flush=2.0&observer=false&debug"/>
+
+       <ExtraDebug value="false"/>
+
+       <!--  this is stream server for online monitoring, normally always on -->
+       <OutputPort name="Output0" url="mbs://Stream:6789?iter=hadaq_iter&subid=0x1f"/>
+
+       <!--  this is example of HLD file storage - local and RFIO -->
+       <OutputPort name="Output1" url="hld:///home/data/trashcan/pulser.hld?maxsize=2000"/>
+       <!--OutputPort name="Output1" url="hld:///linev/path/dabc.hld?maxsize=1900&rfio"/-->
+
+       <DoShmControl value="false"/>  
+       <FlushTimeout value="2.0"/>   
+
+       <!-- take event sequence number from vulom/roc sync message at cts -->
+       <UseSyncSequenceNumber value="false"/>
+       <SyncSubeventId value="0x8000"/>
+       <SyncTriggerMask value="0x01"/>
+       <PrintSync value="false"/>
+       <FlushBySync value="false"/>
+       
+       <!--TriggerNumRange: defines when trigger sequence number wraps. only 16 bit for HADES EBs, 24 bit for trb3!  -->
+       <TriggerNumRange value="0x10000"/>
+       
+       <!--AccountLostEventDiff: if true, missing trigger sequence number are added as lost events to stats. Disabled for multiple event builder mode!  -->
+       <AccountLostEventDiff value="true"/>
+       
+       <!-- rate meters configuration -->
+       <HadaqData width="4" prec="2" low="0" up="10" debug="1"/>
+       <HadaqEvents width="5" prec="1" low="0" up="1000" debug="1"/>
+       <HadaqDroppedData width="5" prec="3" low="0" up="1" debug="1"/>
+       <HadaqLostEvents width="4" prec="2" low="0" up="100" debug="1"/>
+     </Module>
+     
+     <!--  Uncomment to see terminal output like old event builder -->
+     <Module name="Term" class="hadaq::TerminalModule" period="0.3" clear="false"/>
+
+  </Context>
+
+</dabc>
diff --git a/users/ufsd_trb3_sdalinac/db/register_configgbe.db b/users/ufsd_trb3_sdalinac/db/register_configgbe.db
new file mode 100644 (file)
index 0000000..eff7b47
--- /dev/null
@@ -0,0 +1,25 @@
+!Register table
+#  Type  #   C0   #   C1   #   C2   #   C3   #   C4   #   C5   #   C6   #   C7   #   C8   #  C9   #  C10
+###########################################################################################################
+     0     0x8300   0x8305   0x8307   0x8308   0x830b   
+     1     0x8301   0x8302   0x8304   0x8309   0x830c   0x830e   0x830f   0x8310
+
+!Value table
+#                                                                                                                                                      Enable
+#                    SubEvtId      UseGbE      MultiQueue   Trig. Num.  InclTrgType  
+# Hub    #  Type  #     C0     #     C1     #     C2     #    C3     #      C4      
+##################################################################################
+   0xc001      0       0xc001          1            1        0xffffff         1      
+#  0x8610      0       0x8000          1            0        0xffffff         1      
+#  0x8113      0       0x8001          1            0        0xffffff         1      
+#  0x8158      0       0x8002          1            0        0xffffff         1      
+
+
+
+
+
+#These values to not need to be written - for completeness only  
+#                    SubEvtDec     QueDec     FrameSize   RX enable    SubEvtSize   Evt/Queue    QueueClose  MaxQueueSize
+# Hub    #  Type  #     C0     #     C1     #     C2    #     C3     #     C4     #     C5     #     C6     #     C7     #
+##########################################################################################################################
+# 0xff7f      1      0x00020001   0x00030062     0x578         1         59800         200        32000         60000
diff --git a/users/ufsd_trb3_sdalinac/db/register_configgbe_ip.db b/users/ufsd_trb3_sdalinac/db/register_configgbe_ip.db
new file mode 100644 (file)
index 0000000..64e7d9d
--- /dev/null
@@ -0,0 +1,38 @@
+###########################################################################################
+#Eventbuilders:
+# EB 0:   kp1pc105  eth1  00:1B:21:43:97:EA 192.168.0.2 ports 50000 - 50099
+
+!Register table
+#  Type  #   C0   #   C1   #   C2   #   C3   #   C4   #
+#######################################################
+#new memory locations
+     0     0x8100   0x8101   0x8102   0x8103   0x8107  
+     1     0x8110   0x8111   0x8112   0x8113   0x8117  
+     2     0x8120   0x8121   0x8122   0x8123   0x8127  
+     3     0x8130   0x8131   0x8132   0x8133   0x8137  
+     4     0x8140   0x8141   0x8142   0x8143   0x8147  
+     5     0x8150   0x8151   0x8152   0x8153   0x8157  
+     6     0x8160   0x8161   0x8162   0x8163   0x8167  
+     7     0x8170   0x8171   0x8172   0x8173   0x8177  
+     8     0x8180   0x8181   0x8182   0x8183   0x8187  
+     9     0x8190   0x8191   0x8192   0x8193   0x8197  
+    10     0x81A0   0x81A1   0x81A2   0x81A3   0x81A7  
+    11     0x81B0   0x81B1   0x81B2   0x81B3   0x81B7  
+    12     0x81C0   0x81C1   0x81C2   0x81C3   0x81C7  
+    13     0x81D0   0x81D1   0x81D2   0x81D3   0x81D7  
+    14     0x81E0   0x81E1   0x81E2   0x81E3   0x81E7  
+    15     0x81F0   0x81F1   0x81F2   0x81F3   0x81F7  
+
+!Value table
+#                Dest MAC   Dest MAC    Dest IP     Dest Port  Src Port  
+# Hub  # Type #     C0    #     C1   #     C2     #    C3    #    C7   #
+########################################################################
+
+#14:dd:a9:d3:f4:b8 # hadesp39
+#00:24:9b:2c:84:c9  # ufsd1
+#00:24:9b:2c:84:cc  # ufsd2
+#00:24:9b:2c:84:c9  # ufsd2, SDALINAC 
+  0xc001    0    0x9b2c84c9    0x0024    0xc0a8c801    50094   0xc350
+# 0x8610    0    0xa9d3f4b8    0x14dd    0xc0a800b1    50061     0xc350
+# 0x8113          0    0xa9d3f4b8    0x14dd    0xc0a800b1    50113     0xc350
+# 0x8158          0    0xa9d3f4b8    0x14dd    0xc0a800b1    50158     0xc350
diff --git a/users/ufsd_trb3_sdalinac/db/register_configtdc.db b/users/ufsd_trb3_sdalinac/db/register_configtdc.db
new file mode 100644 (file)
index 0000000..d0a6c3f
--- /dev/null
@@ -0,0 +1,29 @@
+# TDC config registers
+
+!Register table
+#  Type  #   C0   #   C1   #   C2   #   C3   #   C4   #   C5   #
+################################################################
+     0     0xc800   0xc801   0xc802   0xc803   0xc804   0xc805
+
+!Value table
+#                    Gnl Conf    Trg Window   Ch En 1-32   Ch En 33-64 RingBufSize    Invert
+# TDC    #  Type  #     C0     #     C1     #     C2     #     C3     #     C4    #     C5     #
+#################################################################################################
+
+0xfe4c         0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000
+#0xfe4a                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # gpin addon
+#0xfe4c                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # padiwa addon
+#0xfe4e                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # ADA addon
+#0xfe50                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # cbmtof
+#0xfe62                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # trb3sc
+#0xfe51                0   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000   0x00000000 # dirich
+#0xfe45                0   0x00000000   0x00000000   0x00000000   0x00000000  #0x00000000   0x00000000 # peripheral
+
+0xfe4c         0   0x50003000   0x80640064   0x00000000   0x00000000   0x00000018   0x00000000
+#0xfe4a                0   0x50003000   0x80640064   0x00000000   0x00000000   0x0000007b   0x00000000 # gpin addon
+#0xfe4c                0   0x50003000   0x80640064   0x00000000   0x00000000   0x0000007b   0x00000000 # padiwa addon
+#0xfe4e                0   0x50003000   0x80640064   0x00000000   0x00000000   0x0000007b   0x00000000 # ADA addon
+#0xfe50                0   0x50003000   0x80640064   0x00000000   0x00000000   0x0000007b   0x00000000 # cbmtof
+#0xfe62                0   0x50003000   0x80640064   0x00000000   0x00000000   0x0000007b   0x00000000 # trb3sc
+#0xfe51                0   0x50003000   0x00640064   0x00000000   0x00000000   0x00000078   0x00000000 # dirich
+#0xfe45                0   0x50003000   0x00640064   0x00000000   0x00000000   0x00000078   0x00000000 # peripheral
diff --git a/users/ufsd_trb3_sdalinac/first.C b/users/ufsd_trb3_sdalinac/first.C
new file mode 100644 (file)
index 0000000..fcac420
--- /dev/null
@@ -0,0 +1,145 @@
+// this is example for
+#include "hadaq/HldProcessor.h"
+#include "hadaq/TrbProcessor.h"
+#include "hadaq/TdcProcessor.h"
+#include "base/ProcMgr.h"
+
+void first()
+{
+   //base::ProcMgr::instance()->SetRawAnalysis(true);
+   base::ProcMgr::instance()->SetTriggeredAnalysis(true);
+
+   // all new instances get this value
+   base::ProcMgr::instance()->SetHistFilling(4);
+
+   // configure bubbles
+   //hadaq::TdcProcessor::SetBubbleMode(3, 18);
+
+   // command from Jan :) for calibration procedure
+   hadaq::TdcProcessor::SetTriggerDWindow(-10,70);
+
+   // this limits used for linear calibrations when nothing else is available
+   hadaq::TdcMessage::SetFineLimits(0, 464);
+
+   // default channel numbers and edges mask
+   hadaq::TrbProcessor::SetDefaults(33, 2);
+
+   // [min..max] range for TDC ids
+   //hadaq::TrbProcessor::SetTDCRange(0x1000, 0x1FFF);
+   hadaq::TrbProcessor::SetTDCRange(0x2600, 0x2FFF); // 0x1fff
+
+   // [min..max] range for HUB ids
+   hadaq::TrbProcessor::SetHUBRange(0xc001, 0xcfff);
+
+   // when first argument true - TRB/TDC will be created on-the-fly
+   // second parameter is function name, called after elements are created
+   hadaq::HldProcessor* hld = new hadaq::HldProcessor(true, "after_create");
+
+   const char* calname = getenv("CALNAME");
+   if ((calname==0) || (*calname==0)) calname = "calib_";
+   const char* calmode = getenv("CALMODE");
+   // doing calibration by hand.... each restart needs calibration 
+   // int cnt = (calmode && *calmode) ? atoi(calmode) : 30000;
+   // use existing calibration files test_xyx.cal
+    int cnt = (calmode && *calmode) ? atoi(calmode) : 1E5;
+   //cnt=100000;
+   const char* caltrig = getenv("CALTRIG");
+   unsigned trig = (caltrig && *caltrig) ? atoi(caltrig) : 0xd;
+   const char* uset = getenv("USETEMP");
+   unsigned use_temp = 0; // 0x80000000;
+   if ((uset!=0) && (*uset!=0) && (strcmp(uset,"1")==0)) use_temp = 0x80000000;
+
+   // printf("TDC CALIBRATION MODE %d\n", cnt);
+
+   printf("HLD configure calibration calfile:%s  cnt:%d trig:%X temp:%X\n", calname, cnt, trig, use_temp);
+
+   // first parameter if filename  prefix for calibration files
+   //     and calibration mode (empty string - no file I/O)
+   // second parameter is hits count for autocalibration
+   //     0 - only load calibration
+   //    -1 - accumulate data and store calibrations only at the end
+   //    >0 - automatic calibration after N hits in each active channel
+   // third parameter is trigger type mask used for calibration
+   //   (1 << 0xD) - special 0XD trigger with internal pulser, used also for TOT calibration
+   //    0x3FFF - all kinds of trigger types will be used for calibration (excluding 0xE and 0xF)
+   //   0x80000000 in mask enables usage of temperature correction
+   hld->ConfigureCalibration(calname, cnt, (1 << trig) | use_temp);
+   //hld->ConfigureCalibration(calname, 100000, 1);
+
+   // only accept trigger type 0x1 when storing file
+   // new hadaq::HldFilter(0x1);
+
+   // create ROOT file store
+   //base::ProcMgr::instance()->CreateStore("td.root");
+
+   // 0 - disable store
+   // 1 - std::vector<hadaq::TdcMessageExt> - includes original TDC message
+   // 2 - std::vector<hadaq::MessageFloat>  - compact form, without channel 0, stamp as float (relative to ch0)
+   // 3 - std::vector<hadaq::MessageDouble> - compact form, with channel 0, absolute time stamp as double
+   base::ProcMgr::instance()->SetStoreKind(2);
+
+
+   // when configured as output in DABC, one specifies:
+   // <OutputPort name="Output2" url="stream://file.root?maxsize=5000&kind=3"/>
+
+
+}
+
+// extern "C" required by DABC to find function from compiled code
+
+extern "C" void after_create(hadaq::HldProcessor* hld)
+{
+   printf("Called after all sub-components are created\n");
+
+   if (hld==0) return;
+
+   for (unsigned k=0;k<hld->NumberOfTRB();k++) {
+      hadaq::TrbProcessor* trb = hld->GetTRB(k);
+      if (trb==0) continue;
+      printf("Configure %s!\n", trb->GetName());
+      trb->SetPrintErrors(100);
+   }
+
+
+
+
+
+   for (unsigned k=0;k<hld->NumberOfTDC();k++) {
+      hadaq::TdcProcessor* tdc = hld->GetTDC(k);
+      if (tdc==0) continue;
+
+      printf("Configure %s!\n", tdc->GetName());
+
+      tdc->SetUseLastHit(false);
+
+      //tdc->SetStoreEnabled();
+      //for (unsigned nch=1; nch<tdc->NumChannels(); nch++) {
+      //  tdc->SetRefChannel(nch, 1, 0x1061, 10000,  -90., 90.);
+      //}
+
+      //      for (unsigned j=10; j<33; j++) {
+      //  tdc->SetRefChannel(j, 1, 0xffff, 10000,  -40., 40.);
+      //}
+
+      tdc->SetRefChannel(10, 1, 0xffff, 10000,  -100.,100.);
+      tdc->SetRefChannel(0, 0, 0x1601, 10000,  -100.,100.);
+      //tdc->SetRefChannel(0, 0, 0x1202, 10000,  -20., 20.);
+      //
+      //tdc->SetRefChannel(4, 2, 0xffff, 10000,  -20., 20.);
+      //tdc->SetRefChannel(1, tdc->NumChannels() -1 , 0xffff, 20000,  -10., 10.);
+
+      //      tdc->SetRefChannel(6, 2 , 0xffff, 20000,  -10., 10.);
+      //tdc->SetRefChannel(4, 2 , 0xffff, 20000,  -10., 10.);
+      //tdc->SetRefChannel(7, 6 , 0xffff, 20000,  -10., 10.);
+      
+      
+   }
+   
+ //  hadaq::TdcProcessor* tdc_0 = hld->GetTDC(0x1060,true);
+ //  hadaq::TdcProcessor* tdc_1 = hld->GetTDC(0x1061,true);
+ //  tdc_0->SetRefChannel(1, 10, 0x1061, 10000,  -20., 20.);
+
+
+
+}
+
diff --git a/users/ufsd_trb3_sdalinac/second.C b/users/ufsd_trb3_sdalinac/second.C
new file mode 100644 (file)
index 0000000..7073dd0
--- /dev/null
@@ -0,0 +1,1142 @@
+#include <stdio.h>
+#include <math.h>
+#include <string>
+#include <sstream>
+#include <fstream>
+#include <vector>
+
+#include "base/EventProc.h"
+#include "base/Event.h"
+#include "hadaq/TdcSubEvent.h"
+#include "hadaq/HldProcessor.h"
+#include <typeinfo>
+
+#include <iostream>
+
+#include "TFile.h"
+#include "TH2.h"
+#include "TH1.h"
+
+using namespace std;
+
+class SecondProc : public base::EventProc 
+{
+    protected:
+    /*Analysis flags*/
+    bool makeTdiffCorr = false;
+    bool makeTdiffCorrFunction = false;
+    bool makeTWCorrParameters = false;
+    bool makeTWCorrFunction = false;
+    bool makeRescale = false;
+    bool makeClusterRemovalCorrCh = false;
+    bool makeClusterRemovalBothCh = false;
+    bool makeCutsCorrChannel = false;
+    bool makeCutsBothChannels = false; 
+    bool makeCrosstalkAnalysis = false;
+    bool makeClusterAnalysis = false;
+
+    
+    double CorrCutMax=10000;
+    double CorrCutMin=0;
+    double RefCutMax=10000;
+    double RefCutMin=0;
+    
+    double TDiffCut = 10000000.;
+    
+    static const int numOfTDCs = 5; // number of used TDCs +1
+    static const int numOfChannels = 17; // number of channels +1 
+    static const int numOfDetectors = 3; // number of detectors + 1
+    
+    int eventnumber=0;
+    
+    // Correlation scheme for setup: [TDC][channel][{channel_number, correlated_channel_number_in_other_TDC, detector_number, position_in_detector}]
+    int correlationScheme[numOfTDCs][numOfChannels][4] = {
+        
+        {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+        
+        {{0,0,0,0}, {1,1,1,2},  {2,2,1,4},  {3,3,1,6},  {4,4,1,8}, {5,5,1,10},  {6,6,1,12},  {7,5,1,14},  {8,8,1,16}, {9,9,1,15},  {10,10,1,13},  {11,12,1,11},  {12,12,1,9}, {13,13,1,7},  {14,14,1,5},  {15,15,1,3}, {16,16,1,1}}, 
+        
+        {{0,0,0,0}, {1,2,1,2},  {2,1,1,4},  {3,3,1,6},  {4,4,1,8}, {5,5,1,10},  {6,6,1,12},  {7,5,1,14},  {8,8,1,16}, {9,9,1,15},  {10,10,1,13},  {11,12,1,11},  {12,12,1,9}, {13,13,1,7},  {14,14,1,5},  {15,15,1,3}, {16,16,1,1}},
+        
+        {{0,0,0,0}, {1,1,1,2},  {2,2,1,4},  {3,3,1,6},  {4,4,1,8}, {5,5,1,10},  {6,6,1,12},  {7,5,1,14},  {8,8,1,16}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0}},
+        
+        {{0,0,0,0}, {1,1,2,2},  {2,2,2,4},  {3,3,2,6},  {4,4,2,8}, {5,5,2,10},  {6,6,2,12},  {7,5,2,14},  {8,8,2,16}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0}}
+
+    };
+    
+    
+    
+//         int correlationScheme[numOfTDCs][numOfChannels][4] = { // Correlation scheme which correlates only with channel 1 of the other detector
+//         
+//         {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+//         
+//         {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {9,9,1,15},  {10,10,1,13},  {11,12,1,11},  {12,12,1,9}, {13,13,1,7},  {14,14,1,5},  {15,15,1,3}, {16,16,1,1}}, 
+//         
+//         {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {9,9,2,15},  {10,10,2,13},  {11,12,2,11},  {12,12,2,9}, {13,13,2,7},  {14,14,2,5},  {15,15,2,3}, {16,16,2,1}},
+//         
+//         {{0,0,0,0}, {1,1,1,2},  {2,1,1,4},  {3,1,1,6},  {4,1,1,8}, {5,1,1,10},  {6,1,1,12},  {7,1,1,14},  {8,1,1,16}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0}},
+//         
+//         {{0,0,0,0}, {1,1,2,2},  {2,1,2,4},  {3,1,2,6},  {4,1,2,8}, {5,1,2,10},  {6,1,2,12},  {7,1,2,14},  {8,1,2,16}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0}}
+// 
+//     };
+    
+   
+      double TWalkCorrFunction [numOfTDCs][numOfChannels][4]={
+                
+//             {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+//             
+//            {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+//             
+//             {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+//             
+//             {{0,0,0,0}, {-183.685, 2.17905, 0.00191316, 0.160909},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}}, 
+//             
+//             {{0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}, {0,0,0,0},  {0,0,0,0},  {0,0,0,0},  {0,0,0,0}},
+          
+          
+          
+        { {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}},
+          
+        { {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}},
+
+        { {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}},
+
+        { {0,0,0,0}, {-202.368,2.2181,0.000303494,0.0765854}, {-176.324,2.10958,-0.00205473,0.178147}, {-237.962,2.34708,0.00103295,0.0445063}, {-291.213,2.42026,0.000945137,0.0717607}, {-196.381,2.23578,8.7334e-05,0.0848016}, {-171.153,2.14604,-8.42002e-05,0.0879119}, {-375.62,2.25024,-0.42893,6.41755}, {-163.742,1.82885,-0.00667092,0.472399}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}},
+
+        { {0,0,0,0}, {-214.793,2.35896,0.00166868,0.121898}, {-224.044,2.33503,0.00109374,0.158345}, {-233.527,2.40481,0.00249939,0.030645}, {-263.802,2.42822,0.00152629,0.0869203}, {-268.726,2.44789,0.0015527,0.116454}, {-112.684,2.0367,-0.00324999,0.490074}, {-263.643,2.51534,0.00296977,0.0282881}, {-305.755,2.25468,0.000236872,0.202798}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}}
+
+          
+          
+            
+        };
+    
+        
+    
+    
+//     double cutValuesLower [numOfTDCs] [numOfChannels]= { 
+//         
+//         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,},
+//         
+//         {0, 0, 0, 0, 0, 0, 0, 0, 0, 4.25668, 5.28533, 4.64967, 5.47919, 4.89474, 4.1154, 5.33308, 6.47866, 0, 0, 0, 0, 0, 0, 0, 0, 5.38717, 5.64863, 0, 5.17477, 5.95315, 5.34243, 5.13881, 6.10115},
+//         
+//         {0, 0, 0, 0, 0, 0, 0, 0, 0, 6.59853, 6.93174, 6.53145, 6.88821, 6.51382, 6.3149, 0, 6.68963, 0, 0, 0, 0, 0, 0, 0, 0, 16.5065, 6.78918, 0, 15.8425, 0, 16.0851, 0, 7.1868}
+//     } ;
+    
+     double cutValuesLower [numOfTDCs] [numOfChannels]= { // test to check if ToT cuts after cluster removal are necessary, or really improve resolution
+        
+         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
+        
+         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
+        
+         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
+         
+         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
+         
+         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
+    } ;
+    
+//     double cutValuesUpper [numOfTDCs] [numOfChannels]= { 
+//         
+//         {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,},
+//         
+//         {0, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 9.70702, 10.2622, 10.1603, 10.3137, 9.33094, 9.10002, 9.26649, 10.2667, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 10.2682, 11.3712, 0, 8.70153, 10.0298, 9.49266, 9.26477, 10.3372,},
+//         
+//         {0,1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 11.8185, 11.7319, 12.0786, 11.3878, 10.693, 10.2768, 0, 10.856, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 24.4851, 10.942, 0, 25.1402, 0, 25.0127, 0, 10.259 }
+//     } ;
+    
+     double cutValuesUpper [numOfTDCs] [numOfChannels]= { // cut values set by hand ca. to bins, where TWalk correction is not done due to missing statistics  to check if ToT cuts after cluster removal are necessary, or really improve resolution
+        
+        {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
+        
+        {0, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000},
+        
+        {0, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000},
+        
+        {0, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000},
+        
+        {0, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000, 1000}
+    } ;
+    
+
+    
+    
+    int detectorScheme[numOfDetectors][17][2];
+    
+    /* Variables */
+    std::string fTdcId1;      //!< tdc id where channels will be selected
+    std::string fTdcId2;      //!< tdc id where channels will be selected
+    std::string fTdcId3;      //!< tdc id where channels will be selected
+    std::string fTdcId4;      //!< tdc id where channels will be selected
+    
+    double fHits[numOfTDCs][numOfChannels][3]; // 4 TDCs, 17 channels, 2 edges + ToT
+    double fHitsCleared[numOfTDCs][numOfChannels][3]; // 4 TDCs, 17 channels, 2 edges + ToT
+    double fHitsDetector[numOfDetectors][18][3]; //3 detectors, 17 channels, 2 edges + ToT rescaled
+
+    std::vector<std::vector<int>> detectorClusters[numOfDetectors];
+    
+    int fSubEvents [numOfTDCs][numOfChannels][2];
+    std::vector<double> fSubTdiff[numOfTDCs][numOfChannels];
+    
+    std::vector<double> Timecorr;
+    std::vector<double> rescale;
+    
+    
+    /* Histograms */
+    base::H1handle hNumHits;   
+    base::H1handle hNumEntries;
+    
+    base::H1handle clusterSizeDetector[numOfDetectors];
+    base::H1handle multiplicityDetector[numOfDetectors];
+    
+    base::H2handle  ToTAllChannels;
+    
+    base:: H2handle hChMap[numOfTDCs];
+       
+    //Number of events in subevent per channel for each TDC
+    base::H2handle hChEvents[numOfTDCs];
+    
+    //Subevents Tdiff per event per chanel for each TDC
+    base::H2handle hChTdiff1[numOfTDCs];
+    base::H2handle hChTdiff2[numOfTDCs];
+    
+    // Create ToTvsTDiff Histograms
+    base::H2handle ToTvsTDiff[numOfTDCs][numOfChannels];
+    base::H2handle TDiffAllChannels[numOfTDCs][numOfChannels];
+    
+    
+    base::H2handle NeighbourCrosstalk[numOfTDCs][numOfChannels];
+    base::H2handle LocalMaxNeighbourCrosstalk[numOfTDCs][numOfChannels];
+    
+    base::H2handle Right_TDiff_Crosstalk[numOfTDCs][numOfChannels];
+    base::H2handle Left_TDiff_Crosstalk[numOfTDCs][numOfChannels];
+    
+    base::H2handle LocalMax_Right_TDiff_Crosstalk[numOfTDCs][numOfChannels];
+    base::H2handle LocalMax_Left_TDiff_Crosstalk[numOfTDCs][numOfChannels];
+    
+    
+    base::H2handle Right_ToT_Crosstalk[numOfTDCs][numOfChannels];
+    base::H2handle Left_ToT_Crosstalk[numOfTDCs][numOfChannels];
+    
+    base::H2handle LocalMax_Right_ToT_Crosstalk[numOfTDCs][numOfChannels];
+    base::H2handle LocalMax_Left_ToT_Crosstalk[numOfTDCs][numOfChannels];
+    
+    base::H1handle sparkMultiplicity[numOfTDCs];
+    
+    
+    //Timediff multiplicity
+    base::H2handle TDTF[numOfTDCs];
+    
+    base::H2handle hEdges[numOfTDCs][numOfChannels];
+    
+    //TW histograms
+    TH2F* TW_histo[5];
+        
+    public:
+    SecondProc(const char* procname, const char* _tdcid1, const char* _tdcid2, const char* _tdcid3, const char* _tdcid4) :
+    base::EventProc(procname),
+    fTdcId1(_tdcid1),
+    fTdcId2(_tdcid2),
+    fTdcId3(_tdcid3),
+    fTdcId4(_tdcid4),
+    ToTAllChannels(0)
+    {   
+        //Time Walk correction source file reading
+        if(makeTWCorrParameters) {
+            TFile *inFile = new TFile("2M_TW.root");
+            if(!inFile) cerr<<"TW file not found \n"<<endl;
+
+//             Select Histogram - pointer to the histogram
+            for(int i = 1; i < numOfTDCs; i++)
+                TW_histo[i] = (TH2F*)inFile->Get(Form("histo%i", i));
+                     
+        }
+        
+       
+        
+        
+        if(makeTdiffCorr || makeTdiffCorrFunction)
+            Timecorr = {
+                
+               // TDiff calibration before everything else
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+// 
+//              0, 0, 0, 0, 0, 0, 0, 0, 0, -0.379719, 
+//             -0.807165, 0, 4.29306, 1.92666, 0.98097, 1.08864, 1.77656, 0, 0, 0,
+// 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+// 
+//             0, 1.07526, 0.138509, 0.779046, 1.56691, 1.46053, 1.59004, 1.00212, 0.464563, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//     
+//             0 
+                // after TWalk correction
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0.259997, 
+//             0.234026, 3.03417, 5.12116, 2.59994, 1.75335, 1.74567, 2.51493, 0, 0, 0,  
+// 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, -0.389521, 
+//             -0.810575, 0, 4.29027, 1.92367, 0.984565, 1.08809, 1.7731, 0, 0, 0, 
+// 
+//             0, 0.973676, -1.5956e-05, 0.4992, 1.17251, 1.31455, 1.55055, -1.33833, 0.288798, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+// 
+//             0, 1.0702, 0.140716, 0.7769, 1.56193, 1.45633, 1.59049, 0.995151, 0.458112, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//     
+//             0 
+                // after TWalk correction and double bin (5 ps bin Width)
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0.259981, 
+//             0.234088, 3.03421, 5.12126, 2.6, 1.75323, 1.74567, 2.51498, 0, 0, 0,  
+// 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, -0.389559, 
+//             -0.810571, 0, 4.29025, 1.92365, 0.984593, 1.08807, 1.7953, 0, 0, 0, 
+// 
+//             0, 0.973764, -7.69487e-06, 0.499233, 1.17253, 1.3146, 1.55058, -1.3353, 0.288827, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+// 
+//             0, 1.07014, 0.140753, 0.776919, 1.56193, 1.45629, 1.59047, 0.995192, 0.458113, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//     
+//             0
+                
+                
+                // TDiff correction parameter after cuts, TWalk correction and cluster removal, 10 ps bin width
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, -0.399899, 
+//                 -0.812385, 0, 4.28664, 1.92076, 0.986521, 1.09002, 1.77038, 0, 0, 0, 
+//                 
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 
+//                 0, 1.06717, 0.140334, 0.776773, 1.55831, 1.45336, 1.58836, 0.992174, 0.456232, 0, 
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 0
+                
+                
+                // TDiff correction parameter after cuts, TWalk correction and cluster removal, 5 ps bin width
+               
+                
+              /*  
+                0, 0, 0, 0, 0, 0, 0, 0, 0, 0.256519, 
+                0.229751, 3.03087, 5.122, 2.6181, 1.75288, 1.75239, 2.51801, 0, 0, 0, 
+            
+                0, 0, 0, 0, 0, 0, 0, 0, 0, -0.399977, 
+                -0.812364, 0, 4.28658, 1.92077, 0.986573, 1.09, 1.7705, 0, 0, 0, 
+                
+                0, 0.972426, -1.15853e-05, 0.497899, 1.17262, 1.31565, 1.54837, 0, 0.289097, 0, 
+                0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+
+                0, 1.06703, 0.140375, 0.776698, 1.55825, 1.45338, 1.58829, 0.992229, 0.456329, 0, 
+                0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+                0
+                */
+
+                
+                
+              // Mean TDiff parameters. Acquired by first making normal TDiff calibration, then finding the mean correction parameters for the other TDC and then running analysis to get mean correction 
+              //parameters of the first TDC
+
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0.256519, 
+//                 0.229751, 3.03087, 5.122, 2.6181, 1.75288, 1.75239, 2.51801, 0, 0, 0, 
+//             
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, -0.399977, 
+//                 -0.812364, 0, 4.28658, 1.92077, 0.986573, 1.09, 1.7705, 0, 0, 0, 
+//             
+//                 
+//                0, 0.978305 , 0, 0.50338, 1.17631, 1.31946, 1.55397, 0, 0.293874, 0,
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+// 
+//                 0, 1.07614, 0.14949, 0.789326, 1.56768, 1.46074, 1.59607, 1.00274, 0.467685, 0,
+//                 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//                 0
+
+
+            // New TDiff parameters generated after TWalk with function in one step
+/*
+
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0.241436, 
+            0.216216, 3.00324, 5.11242, 2.53487, 1.71291, 1.69343, 2.44475, 0, 0, 0, 
+            
+            0, 0, 0, 0, 0, 0, 0, 0, 0, -0.432846, 
+            -0.865883, 0, 4.22372, 1.89486, 0.95267, 1.1048, 1.78417, 0, 0, 0, 
+            
+            0, 0.956509, -0.00782424 , 0.483118, 1.15808, 1.30818, 1.58146, 0, 0.286464, 0, 
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+            
+            0, 1.06409, 0.134968, 0.762091, 1.55373, 1.43657, 1.55457, 0.96967, 0.467006, 0, 
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+            0, */
+            
+            
+            // TDiff Parameters after TWalk correction with a function and Cluster removal and ToT cuts generated in Two steps/two files
+            
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0.249248, 
+//             0.224022, 3.01106, 5.1202, 2.54274, 1.7208, 1.70127, 2.45253, 0, 0, 0, 
+//             
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, -0.425687, 
+//             -0.861085, 0, 4.22932, 1.89926, 0.959421, 1.11553, 1.79165, 0, 0, 0, 
+//             
+//             0, 0.964389, 3.0101e-05, 0.490929, 1.16599, 1.31605, 1.58926, 0, 0.294278, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+// 
+//             0, 1.07204, 0.142791, 0.771455, 1.55964, 1.44587, 1.56574, 0.978453, 0.469983, 0, 
+//             0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+//             0
+
+             // TDiff Parameters after TWalk correction with a function and Cluster removal and ToT cuts generated in one step/one file
+
+            
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0.106466, 
+            0.0812532, 2.86827, 4.97746, 2.39991, 1.57795, 1.55847, 2.30977, 0, 0, 0, 
+            
+            
+            0, 0, 0, 0, 0, 0, 0, 0, 0, -0.425687, 
+            -0.861085, 0, 4.22932, 1.89926, 0.959421, 1.11553, 1.79165, 0, 0, 0, 
+            0, 0.821543, -0.142791, 0.348155, 1.02312, 1.1732, 1.4465, 0, 0.151474, 0, 
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+            0, 1.07204, 0.142791, 0.771455, 1.55964, 1.44587, 1.56574, 0.978453, 0.469983, 0, 
+            0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 
+            0
+
+            
+
+            
+       };else Timecorr = std::vector<double> (81, 0);
+        
+        if(/*makeClusterRemovalCorrCh || makeClusterRemovalBothCh ||*/ makeRescale)
+            rescale = {
+            1, 
+        
+            1, 1, 1, 1, 1, 1, 1, 1, 4.02951, 3.94215, 4.44876, 4.36756, 4.26959, 4.07083, 4.39062, 3.93766, 1, 1, 1, 1, 
+            1, 1, 1, 1, 1, 1, 1, 1, 4.32465, 4.53967, 1, 4.78249, 4.63644, 4.60738, 4.75951, 4.45675, 1, 1, 1, 1, 
+
+            // normalize nino to 35
+            1.09407, 1.12791, 1.09457, 1.20249, 1.10512, 1.09579, 1 , 0.998456, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 
+            1.55471, 1.58341, 1.2849, 1.49527, 1.59509, 2.31983, 1.4235, 1.39305, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 
+
+                
+                
+            };
+        else
+            rescale = std::vector<double> (81, 1);
+        
+        hNumHits = MakeH1("NumHits","Num hits", 100, 0, 100, "number");
+        hNumEntries = MakeH1("NumEntries", "Num Entries", 60, 0, 60, "channel; entries");
+        
+        // Time over Threshold for all channels histogram
+        ToTAllChannels = MakeH2("ToTAllChannels","ToTAllChannels" , 100, 0, 100, 1000, -10, 200, "TDC_ch;ToT[ns]");
+               
+        char histoName[100]; 
+        for (int i = 1; i < numOfTDCs; i++) {
+            sprintf(histoName,"Spark_Multiplicity_TDC%i", i);
+            sparkMultiplicity[i] = MakeH1(histoName, histoName, 35, 0, 35, "Multiplicity; Entries");
+            
+            for (int j=1; j<numOfChannels; j++){
+                if(correlationScheme[i][j][0]){
+                    
+                    if (i<3){
+                    sprintf(histoName,"ToT%i_vs_TDiff_ch%i_%i", i,correlationScheme[i][j][0] ,correlationScheme[i][j][1]);
+                    ToTvsTDiff[i][j] = MakeH2(histoName, histoName,600,0,60,500,-10,10,"ToT [ns]; TDiff [ns]");
+                    }
+                    
+                    else if (i>2){
+                    sprintf(histoName,"ToT%i_vs_TDiff_ch%i_%i", i,correlationScheme[i][j][0] ,correlationScheme[i][j][1]);
+                    ToTvsTDiff[i][j] = MakeH2(histoName, histoName,180,0,60,500,-10,10,"ToT [ns]; TDiff [ns]");
+                    }
+                    
+                    if(j < 9 || (j > 8 && correlationScheme[i][j][0] != correlationScheme[i][j - 8][0] )){
+                        sprintf(histoName,"RefTDC%i_ch%i", i,correlationScheme[i][j][0]);
+                        TDiffAllChannels[i][j] = MakeH2(histoName, histoName, 80,0,80,4000,-10.0025,9.9975,"Second_TDC_channels; TDiff [ns]");
+                    } 
+                    sprintf(histoName,"NumberOfEdgesTDC_%i_ch_%i", i,correlationScheme[i][j][0]);
+                    hEdges[i][j] = MakeH2(histoName, histoName, 20,0,20,20,0,20,"Leading edges; Trailing edges");
+                    
+                     if (makeCrosstalkAnalysis){
+                         if (i>2){
+                        sprintf(histoName,"NeighbouringCrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]);
+                        NeighbourCrosstalk[i][j] = MakeH2(histoName, histoName, 180,0,60,5 ,0,5,"ToT [ns]; ClusterSize");
+                        
+                        sprintf(histoName,"LocalMaxNeighbouringCrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]);
+                        LocalMaxNeighbourCrosstalk[i][j] = MakeH2(histoName, histoName, 180,0,60,5 ,0,5,"ToT [ns]; ClusterSize");
+                                                               
+                        Left_TDiff_Crosstalk[i][j] = MakeH2(Form("Left_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Left_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,250 ,-5,5,"ToT [ns]; TDiff_Left");
+                        Right_TDiff_Crosstalk[i][j] = MakeH2(Form("Right_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Right_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,250 ,-5,5,"ToT [ns]; TDiff_Right");
+                        
+                        LocalMax_Left_TDiff_Crosstalk[i][j] = MakeH2(Form("LocalMaxLeft_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxLeft_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,250 ,-5,5,"ToT [ns]; TDiff_Left");
+                        LocalMax_Right_TDiff_Crosstalk[i][j] = MakeH2(Form("LocalMaxRight_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxRight_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,250 ,-5,5,"ToT [ns]; TDiff_Right");
+                        
+                        
+                        Left_ToT_Crosstalk[i][j] = MakeH2(Form("Left_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Left_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,180 ,0,18,"ToT [ns]; ToT_Left");
+                        Right_ToT_Crosstalk[i][j] = MakeH2(Form("Right_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Right_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,60,180 ,0,15,"ToT [ns]; ToT_Right");
+                        
+                        LocalMax_Left_ToT_Crosstalk[i][j] = MakeH2(Form("LocalMaxLeft_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxLeft_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 90,0,60,90 ,0,60,"ToT [ns]; ToT_Left");
+                        LocalMax_Right_ToT_Crosstalk[i][j] = MakeH2(Form("LocalMaxRight_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxRight_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 90,0,60,90 ,0,60,"ToT [ns]; ToT_Right");
+                         }
+                         
+                         
+                         else if (i<3){
+                             sprintf(histoName,"NeighbouringCrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]);
+                        NeighbourCrosstalk[i][j] = MakeH2(histoName, histoName, 180,0,18,5 ,0,5,"ToT [ns]; ClusterSize");
+                        
+                        sprintf(histoName,"LocalMaxNeighbouringCrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]);
+                        LocalMaxNeighbourCrosstalk[i][j] = MakeH2(histoName, histoName, 180,0,18,5 ,0,5,"ToT [ns]; ClusterSize");
+                                                               
+                        Left_TDiff_Crosstalk[i][j] = MakeH2(Form("Left_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Left_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,250 ,-5,5,"ToT [ns]; TDiff_Left");
+                        Right_TDiff_Crosstalk[i][j] = MakeH2(Form("Right_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Right_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,250 ,-5,5,"ToT [ns]; TDiff_Right");
+                        
+                        LocalMax_Left_TDiff_Crosstalk[i][j] = MakeH2(Form("LocalMaxLeft_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxLeft_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,250 ,-5,5,"ToT [ns]; TDiff_Left");
+                        LocalMax_Right_TDiff_Crosstalk[i][j] = MakeH2(Form("LocalMaxRight_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxRight_TDiff_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,250 ,-5,5,"ToT [ns]; TDiff_Right");
+                        
+                        
+                        Left_ToT_Crosstalk[i][j] = MakeH2(Form("Left_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Left_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,180 ,0,60,"ToT [ns]; ToT_Left");
+                        Right_ToT_Crosstalk[i][j] = MakeH2(Form("Right_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("Right_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 180,0,18,180 ,0,60,"ToT [ns]; ToT_Right");
+                        
+                        LocalMax_Left_ToT_Crosstalk[i][j] = MakeH2(Form("LocalMaxLeft_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxLeft_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 90,0,60,90 ,0,60,"ToT [ns]; ToT_Left");
+                        LocalMax_Right_ToT_Crosstalk[i][j] = MakeH2(Form("LocalMaxRight_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), Form("LocalMaxRight_ToT_CrosstalkTDC_%i_ch%i", i,correlationScheme[i][j][0]), 90,0,60,90 ,0,60,"ToT [ns]; ToT_Right");
+                             
+                             
+                             
+                         }
+                     }
+                     
+                }
+            }
+             if (i<numOfDetectors){
+                 sprintf(histoName,"MulitplicityDetector_%i", i);
+                 multiplicityDetector[i] =  MakeH1(histoName, histoName, 17, 0,17,"Multiplicity");
+                 sprintf(histoName,"clusterSizeDetector%i", i);
+                 clusterSizeDetector[i] =  MakeH1(histoName, histoName, 17, 0,17,"ClusterSize");
+             }
+           
+        //Number of events in subevent per channel for each TDC
+           sprintf(histoName,"SignalsPerEventTDC%i", i);
+        hChEvents[i] = MakeH2(histoName, histoName, 16, 1,17, 30, 0, 30,"channel;# of edges");
+            
+        // Multiplicity for time difference histograms
+           sprintf(histoName, "TDTF%i", i);
+        TDTF[i] = MakeH2(histoName, histoName,4000,0,1000,17,0,17,"Tdiff[ns];TDC1_ch");
+            
+        // Channel correlation mapping histograms
+        sprintf(histoName, "ChMapTDC%i_TDCs", i);
+           char axisLabel[50];
+           sprintf(axisLabel, "ch_TDCs;ch_refTDC%i", i);
+        hChMap[i] = MakeH2(histoName, histoName, 80, 0, 80, 80, 0, 80, axisLabel);
+        }
+        
+        CreateDetectorScheme();
+    }
+    
+    double CalculateTWCorrection(double ToT) {
+        if(ToT >= 9.45)
+            return 0.0106702 * ToT - 0.112834;
+        else
+            return -0.0313777 * ToT * ToT + 0.527936 * ToT - 2.2427;
+        
+    }
+
+    long Unpack(hadaq::TdcSubEventFloat* sub, int shift, double hits[][3], int eventsMap[][2], std::vector<double> channelTdiff[numOfChannels]) 
+    {
+        long msgcnt = 0;
+
+        double t0=500000; //variable to store the time of the first particle
+        int ch_ref=0; //the chid for the first hit
+        
+        // Set all passed variables to 0
+        for (unsigned n=0; n<numOfChannels; n++) {
+            hits[n][0] = hits[n][1] = hits[n][2] =  0.;
+            eventsMap[n][0] = 0; // number of leading edges for each channel
+            eventsMap[n][1] = 0; // number of trailing edges 
+            channelTdiff[n].clear();
+        }
+        
+        if (!sub) return 0;
+        
+        for (unsigned cnt=0;cnt<sub->Size();cnt++) 
+        {   
+            const hadaq::MessageFloat& msg = sub->msg(cnt);
+            unsigned chid = msg.getCh();
+            unsigned edge = msg.getEdge(); // 0 -leading, 1 - trailing
+
+       
+            FillH1(hNumEntries, chid+shift);
+            
+            double tm = msg.stamp;
+            if (makeTdiffCorr)
+            tm = tm + Timecorr[chid+shift];
+            
+//             if(shift == 60)
+//                 tm += Timecorr[42];
+
+//             if ((chid > numOfChannels - 1) || (edge > 1)) printf("ERROR !!!! %u %u\n", chid, edge);
+            if ((chid > 26) || (edge > 1)) printf("ERROR !!!! %u %u\n", chid, edge);
+            if ((chid < numOfChannels) && (edge < 2)) 
+            {
+                hits[chid][edge] = tm;   // this is for the last hit
+                msgcnt++;
+                
+                if(edge == 0){
+                    channelTdiff[chid].push_back(tm);
+                    eventsMap[chid][0] += 1;
+                } else if (edge == 1)
+                    eventsMap[chid][1] += 1;
+                    
+            }
+          if ( (chid < numOfChannels) && (edge==0) && (t0>hits[chid][edge]) ) 
+            { 
+                ch_ref=chid;
+                t0=hits[chid][edge]; //save the new reference
+            }
+        }
+       
+        // fill ToT
+        int Tot1=0;
+//         int Tot2=0;
+        
+        double TW_corr = 0;
+        
+        //Time Walk correction mulitpliers
+        double TDC12 = 3.0;
+        double TDC34 = 3.0;
+        // ToT values have to be mulitplied by those values to aquire proper bin number in TW correction histogram
+        // TDC 1 and TDC2 have range 0-15 and 150 bins -> mulitplier 10.0
+        // TDC3 and TDC4 have range 0-50 and 150 bins -> mulitplier 3.0
+        
+        //TW correction and ToT
+        for (int ch=1; ch < numOfChannels; ch++) 
+        {
+            //Calculate ToT
+            hits[ch][2]=hits[ch][1]-hits[ch][0];
+//             if(hits[ch][2] < 0)
+//                 return -999;
+//             
+            // Accessing bins with TW correction
+            Tot1 = int(floor(TDC12*(hits[ch][2]))); 
+//             Tot2 = int(floor(TDC34*(hits[ch][2]))); 
+            
+            int index = shift / 20 + 1; // CHANGE IF SHIFT CHANGED
+            if (makeTWCorrParameters && hits[ch][0]!=0 && hits[ch][1]!=0 && correlationScheme[index][ch][1] != 0) // TWalk correction with the generated Parameterfiles if both t1 and t0 are not 0 (recording went fine) and any channel is correlated with it (Not a broken channel)
+            {   
+                if(int(floor(hits[ch][1] - hits[ch][0])) < 0) {
+                    hits[ch][1] = 0;
+                    hits[ch][0] = 0;
+                } else {
+                    TW_corr = ((TH2F*)TW_histo[index])->GetBinContent(ch,Tot1+2);
+                    hits[ch][1] = hits[ch][1] - TW_corr;
+                    hits[ch][0] = hits[ch][0] - TW_corr;
+                }
+            }
+            
+            
+            
+            
+            
+            if (makeTWCorrFunction && hits[ch][0]!=0 && hits[ch][1]!=0 && correlationScheme[index][ch][1] != 0) //TWalk correction with the generated parameters via fit functions if both t1 and t0 are not 0 (recording went fine) and any channel is correlated with it (Not a broken channel)
+            {   
+                if(int(floor(hits[ch][1] - hits[ch][0])) < 0) {
+                    hits[ch][1] = 0;
+                    hits[ch][0] = 0;
+                } else {
+                    double param0=TWalkCorrFunction[index][ch][0];
+                    double param1=TWalkCorrFunction[index][ch][1];
+                    double param2=TWalkCorrFunction[index][ch][2];
+                    double param3=TWalkCorrFunction[index][ch][3];
+                    if (makeTdiffCorrFunction){
+                        param3=param3 + Timecorr[ ch+shift];
+                         if(shift == 60)
+                            param3 += Timecorr[42];
+                        
+                    }
+                    
+                    
+                    if (param0!=0 || param1!=0 || param2!=0 || param3!=0)
+                    TW_corr = param0/(pow(hits[ch][2],param1))+param2*hits[ch][2]+param3 ;
+                    else TW_corr =0;
+                    hits[ch][1] = hits[ch][1] + TW_corr;
+                    hits[ch][0] = hits[ch][0] + TW_corr;
+                }
+            }
+            
+            
+            
+            
+            
+            
+            
+            if ( (msgcnt>=4) && (ch!=ch_ref) ) //at least two particles
+            {
+                FillH2(TDTF[index], hits[ch][0]-t0, ch_ref);
+            }
+            
+            if(hits[ch][0]!=0)
+                FillH2(ToTAllChannels, shift+ch, hits[ch][2]);
+            
+        }
+        
+        return msgcnt;
+    }
+    
+    void FillTDiff(int referenceTDC, int correlatedTDC, int shift)
+    {   
+        int refChannel, corrChannel;
+        double *refData, *corrData;
+        double refChCutLower = 0, refChCutUpper = 0,corrChCutLower = 0, corrChCutUpper = 0;
+        
+        for (int n1=1; n1<numOfChannels; n1++) {
+            refChannel = correlationScheme[referenceTDC][n1][0];
+            
+            if(makeClusterRemovalBothCh || makeRescale)
+                refData = fHitsCleared[referenceTDC][refChannel];
+            else 
+                refData = fHits[referenceTDC][refChannel];
+            
+            refChCutLower = makeCutsBothChannels ? cutValuesLower[referenceTDC][refChannel] : RefCutMin;
+            refChCutUpper = makeCutsBothChannels ? cutValuesUpper[referenceTDC][refChannel] : RefCutMax;
+            
+            if (refChannel != 0 && refData[2] > refChCutLower && refData[2] < refChCutUpper ) {
+                for (int n2=1; n2<numOfChannels; n2++) {
+                    corrChannel= correlationScheme[correlatedTDC][n2][0];
+                    
+                    if(!makeClusterRemovalCorrCh && !makeClusterRemovalBothCh && !makeRescale)
+                        corrData = fHits[correlatedTDC][corrChannel];
+                    else 
+                        corrData = fHitsCleared[correlatedTDC][corrChannel];
+                    
+                    corrChCutLower = (makeCutsCorrChannel || makeCutsBothChannels) ? cutValuesLower[correlatedTDC][corrChannel] : CorrCutMin,
+                    corrChCutUpper = (makeCutsCorrChannel || makeCutsBothChannels) ? cutValuesUpper[correlatedTDC][corrChannel] : CorrCutMax;
+                    
+                    if (corrChannel != 0 && corrData[2] > corrChCutLower && corrData[2] < corrChCutUpper && abs(corrData[0] - refData[0]) <TDiffCut) { 
+                        if(corrChannel!=refChannel)
+                            FillH2(hChMap[referenceTDC], corrChannel+shift, refChannel);
+                        if (referenceTDC!= correlatedTDC)
+                        FillH2(TDiffAllChannels[referenceTDC][n1], corrChannel+shift, corrData[0] - refData[0]); // n1 because of correlationScheme of TDC3
+                    }
+                }
+            }
+        }
+    }
+    
+    void FillToT(int referenceTDC, int correlatedTDC) 
+    {        
+        int refChannel, corrChannel;
+        double *refData, *corrData;
+        double refChCutLower = 0, refChCutUpper = 0,corrChCutLower = 0, corrChCutUpper = 0;
+        
+        for (int i=1; i<numOfChannels; i++){
+            refChannel= correlationScheme[referenceTDC][i][0];
+            
+            if(makeClusterRemovalBothCh || makeRescale)
+                refData = fHitsCleared[referenceTDC][refChannel];
+            else 
+                refData = fHits[referenceTDC][refChannel];
+            
+            refChCutLower = makeCutsBothChannels ? cutValuesLower[referenceTDC][refChannel] : RefCutMin;
+            refChCutUpper = makeCutsBothChannels ? cutValuesUpper[referenceTDC][refChannel] : RefCutMax;
+            
+            if(refChannel != 0 && refData[2]>refChCutLower && refData[2]<refChCutUpper){
+                corrChannel= correlationScheme[referenceTDC][i][1];
+                
+                if(!makeClusterRemovalCorrCh && !makeClusterRemovalBothCh && !makeRescale)
+                    corrData = fHits[correlatedTDC][corrChannel];
+                else 
+                    corrData = fHitsCleared[correlatedTDC][corrChannel];
+                 
+                corrChCutLower = (makeCutsCorrChannel || makeCutsBothChannels) ? cutValuesLower[correlatedTDC][corrChannel] : CorrCutMin,
+                corrChCutUpper = (makeCutsCorrChannel || makeCutsBothChannels) ? cutValuesUpper[correlatedTDC][corrChannel] : CorrCutMax;
+                
+                if(corrData[2]>corrChCutLower && corrData[2]<corrChCutUpper && abs(corrData[0] - refData[0]) < TDiffCut)
+                    FillH2(ToTvsTDiff[referenceTDC][i], refData[2], corrData[0] - refData[0]);
+                
+//                 if (refData[2]<28)
+//                     cout << "TDC"<< referenceTDC<< "    channel    " << refChannel<< " T0:    "<< refData[0] << " T1:  " << refData[1]<< " ToT:   "<< refData[2]<<endl;
+//                     FillH2(ToTvsTDiff[referenceTDC][i], fHits[referenceTDC][refChannel][2], corrData[0] - refData[0]);
+            }
+        }
+    }
+    
+//     void FillSubevents()
+ bool FillSubevents() 
+    {   
+        int channel, firedChannelsCounter = 0;
+        bool spark = false; 
+        bool edgeMissed = false; 
+        for(int i = 1; i < numOfTDCs; i++){ // TDC number
+            spark = false;
+            
+            for(int j = 1; j < numOfChannels; j++) // channel number
+                if(correlationScheme[i][j][0] != 0){
+                    channel = correlationScheme[i][j][0];
+                    
+                    // Number of subevents per channel
+                   // FillH2(hChEvents[i], channel, fSubEvents[i][channel]);
+                    if (fSubEvents[i][channel][0]!=0 || fSubEvents[i][channel][1]!=0)
+                        FillH2(hEdges[i][channel], fSubEvents[i][channel][0], fSubEvents[i][channel][1]);
+                
+                    //Multiplicity
+                    if(fSubEvents[i][channel][0] > 3 || fSubEvents[i][channel][1] > 3 || abs(fSubEvents[i][channel][0] - fSubEvents[i][channel][1]) > 0) 
+                       spark = true;
+                    if  (abs(fSubEvents[i][channel][0] - fSubEvents[i][channel][1]) > 0)
+                        edgeMissed=true;
+//                     
+                }
+        
+            if (spark){
+                firedChannelsCounter = 0;
+                
+                for(int j = 1; j < numOfChannels; j++){
+                    channel = correlationScheme[i][j][0];
+                    
+                    if(fSubEvents[i][channel][0] > 0 || fSubEvents[i][channel][1] > 0)
+                        firedChannelsCounter += 1;
+                }
+                
+                FillH1(sparkMultiplicity[i], firedChannelsCounter);
+//                 if (i==2 && firedChannelsCounter){
+//                     for (int j=1; j<33; j++)
+//                         if (correlationScheme[i][j][0] !=0)
+//                             cout << fHits[i][j][2] << " ";
+//                     cout  << eventnumber <<endl;  
+//                 }
+            }   
+        }
+        return edgeMissed;
+    }
+    
+    void FillDetectorHits(int detector)
+    {   
+        //Set all to 0
+        for(int j = 0; j < 18; j++)
+            for(int k = 0; k < 3; k++)
+                fHitsDetector[detector][j][k] = 0;
+        
+        for(int i = 1; i < numOfTDCs; i++){
+            for(int j = 0; j < numOfChannels; j++){
+                if(correlationScheme[i][j][2] == detector){
+                    for(int k = 0; k < 3; k++)
+                        if(k<2)
+                            fHitsDetector[detector][correlationScheme[i][j][3]][k] = fHits[i][correlationScheme[i][j][0]][k];
+                        else
+                            fHitsDetector[detector][correlationScheme[i][j][3]][k] = fHits[i][correlationScheme[i][j][0]][k] * rescale[(i-1)*20 + correlationScheme[i][j][0]];
+                }
+            }
+        }
+    }
+    
+    std::vector<std::vector<int>> findClusters(int detector)
+    {
+        std::vector<std::vector<int>> result;
+        result.clear();
+        
+        int firstPosition = 0;
+        int lastPosition = 0;
+        int maxPosition = 0;
+        int currentPosition = 1;
+        
+        std::vector<int> clusterData;
+        while(currentPosition<numOfChannels) {
+            clusterData.clear();
+            
+            for (int k=currentPosition; k<17; k++) {   
+                if(fHitsDetector[detector][k][2] != 0 ) {   
+                    if (fHitsDetector[detector][k-1][2] == 0) {
+                        firstPosition = k ;
+                    } 
+                           
+                    if (fHitsDetector[detector][k][2] > fHitsDetector[detector][maxPosition][2]) {
+                        maxPosition = k;
+                    }
+                            
+                    if ((k < 16) && (fHitsDetector[detector][k+1][2] == 0) ) {
+                        lastPosition = k + 1;
+                        currentPosition = k;
+                    }
+                    else if( k == 16) {   
+                        lastPosition = k+1; // cluster size = last position - first position, index 17 is OK!
+                        currentPosition = k;
+                    }  
+                }
+                
+                if( lastPosition!=0) break;
+            }
+            
+            if(firstPosition != 0 && lastPosition !=0){
+                clusterData.push_back(firstPosition);
+                clusterData.push_back(maxPosition);
+                clusterData.push_back(lastPosition);
+                result.push_back(clusterData);
+            }
+            
+            currentPosition++;
+            maxPosition = 0;
+            lastPosition = 0;
+            firstPosition = 0;
+        }
+        
+        return result;
+    }
+
+    void RemoveClusters(int detector) 
+    {
+        auto detectorClustersLocal = detectorClusters[detector];
+        for(auto subCluster : detectorClustersLocal )
+            for(int i = subCluster[0]; i < subCluster[2]; i++){
+                if ( i != subCluster[1] ) {      
+                    fHitsDetector[detector][i][0] = 0;
+                    fHitsDetector[detector][i][1] = 0;
+                    fHitsDetector[detector][i][2] = 0;
+                }
+            }
+    }
+    
+    void createClearedfHits() 
+    {
+        int* coordinates;
+        for(int i = 0; i < numOfTDCs; i++)
+            for(int j = 0; j < numOfChannels; j++)
+                for(int k = 0; k < 3; k++)
+                      fHitsCleared[i][j][k] = 0.;
+        
+        for(int i = 1; i < numOfDetectors; i++) {
+            for(int j = 1; j < 17; j++){
+                if(fHitsDetector[i][j][2] != 0){
+                    coordinates = DetectorHitsToHits(i, j);
+                    
+                    for(int k = 0; k < 3; k++){
+                        fHitsCleared[coordinates[0]][coordinates[1]][k] = fHits[coordinates[0]][coordinates[1]][k];
+                        
+                        if(makeRescale)
+                            fHitsCleared[coordinates[0]][coordinates[1]][2] = fHitsDetector[i][j][2];  
+                    }
+                }
+            }
+        }
+    }
+
+    int* DetectorHitsToHits(int detector, int position)
+    {
+        return detectorScheme[detector][position];
+    }
+    
+    void FillClusterDetector (int detector) {
+        int multiplicity=0;
+        int clusterSize = 0;
+        
+        for (unsigned int i=0; i< detectorClusters[detector].size(); i++ ){
+            clusterSize = detectorClusters[detector][i][2]-detectorClusters[detector][i][0];
+            multiplicity += clusterSize ;
+            if(clusterSize>0)
+                FillH1 (clusterSizeDetector[detector], clusterSize);
+        }
+            
+        if(multiplicity>0)
+            FillH1 (multiplicityDetector[detector], multiplicity);
+    }
+    
+    void FillNeighbouringCrosstalk (int detector){
+        int clusterSize=0;
+        int otherDetector =0;
+        if (detector == 1)
+            otherDetector =2;
+        else otherDetector=1;
+        
+        for (int i=1; i< 17; i++){ // 17 is num of channels connected to detector
+            
+            auto position = DetectorHitsToHits(detector, i); // [TDC, channel]
+            auto otherPosition = DetectorHitsToHits(otherDetector, i); // output->[(other)TDC, channel]
+            
+            int channel= position[1];
+            int TDC=position[0];
+            int otherTDC= otherPosition[0];
+            int otherChannel= correlationScheme[TDC][channel][1];
+            
+            if (fHitsDetector[detector][i][2]>0 && fHits[otherTDC][otherChannel][2] >0 && TMath::Abs(fHitsDetector[detector][i][0]-fHits[otherTDC][otherChannel][0]) <15 ){// this checks if signals used for filling are really coming from a particle traversing both detectors, i.e. inducing a signal in both detectors, which a TDiff of smaller than 15 ns 
+                clusterSize +=1;
+                if (fHitsDetector[detector][i+1][2]>0 && TMath::Abs(fHitsDetector[detector][i][0] - fHitsDetector[detector][i+1][0]) <10)
+                    clusterSize +=1;
+                if (fHitsDetector[detector][i-1][2]>0 && TMath::Abs(fHitsDetector[detector][i][0] - fHitsDetector[detector][i-1][0]) <10)
+                    clusterSize +=1;
+                
+                FillH2 (NeighbourCrosstalk[position[0]][position[1]], fHitsDetector[detector][i][2] ,clusterSize); // TODO correlation between channels for hit-> TDiff around 10 ns, not more
+                
+                if( fHitsDetector[detector][i][2]>fHitsDetector[detector][i+1][2] && fHitsDetector[detector][i][2]>fHitsDetector[detector][i-1][2])
+                    FillH2 (LocalMaxNeighbourCrosstalk[position[0]][position[1]], fHitsDetector[detector][i][2] ,clusterSize);
+                    
+                clusterSize = 0;
+            }
+        }
+    }
+    
+    void FillTDiff_Left_Right (int detector){
+        int otherDetector =0;
+        if (detector == 1)
+            otherDetector =2;
+        else otherDetector=1;
+        
+        for (int i=1; i< 17; i++){
+            
+            auto position = DetectorHitsToHits(detector, i); // [TDC, channel]
+            auto otherPosition = DetectorHitsToHits(otherDetector, i); // output->[(other)TDC, channel]
+            
+            int channel= position[1];
+            int TDC=position[0];
+            int otherTDC= otherPosition[0];
+            int otherChannel= correlationScheme[TDC][channel][1];
+            
+            if (fHitsDetector[detector][i][2]>0 && fHitsDetector[detector][i+1][2]>0 && fHits[otherTDC][otherChannel][2] >0 && TMath::Abs(fHitsDetector[detector][i][0]-fHits[otherTDC][otherChannel][0]) <15  ) {
+                FillH2(Right_TDiff_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i][0]-fHitsDetector[detector][i+1][0]);
+                
+              if( fHitsDetector[detector][i][2]>fHitsDetector[detector][i+1][2] && fHitsDetector[detector][i][2]>fHitsDetector[detector][i-1][2] )
+                    FillH2(LocalMax_Right_TDiff_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i][0]-fHitsDetector[detector][i+1][0]);
+            }
+            if (fHitsDetector[detector][i][2]>0 && fHitsDetector[detector][i-1][2]>0 && fHits[otherTDC][otherChannel][2] >0 && TMath::Abs(fHitsDetector[detector][i][0]-fHits[otherTDC][otherChannel][0]) <15 ){
+                FillH2(Left_TDiff_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i][0]-fHitsDetector[detector][i-1][0]);
+              //  FillH2(Left_TDiff_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i-1][2]);
+                if( fHitsDetector[detector][i][2]>fHitsDetector[detector][i-1][2] && fHitsDetector[detector][i][2]>fHitsDetector[detector][i+1][2])
+                    FillH2(LocalMax_Left_TDiff_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i][0]-fHitsDetector[detector][i-1][0]);
+            }
+            
+            channel =0;
+            TDC =0;
+            otherTDC=0;
+            otherChannel=0;
+        }
+    }
+   
+    void FillToT_Left_Right (int detector){
+        int otherDetector =0;
+        if (detector == 1)
+            otherDetector =2;
+        else otherDetector=1;
+        
+        for (int i=1; i< 17; i++){
+            auto position = DetectorHitsToHits(detector, i); // [TDC, channel]
+            auto otherPosition = DetectorHitsToHits(otherDetector, i); // output->[(other)TDC, channel]
+            
+            int channel= position[1];
+            int TDC=position[0];
+            int otherTDC= otherPosition[0];
+            int otherChannel= correlationScheme[TDC][channel][1];
+            
+            if (fHitsDetector[detector][i][2]>0 && fHitsDetector[detector][i+1][2]>0 && fHits[otherTDC][otherChannel][2] >0 && TMath::Abs(fHitsDetector[detector][i][0]-fHits[otherTDC][otherChannel][0]) <15  ) {
+                FillH2(Right_ToT_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2],fHitsDetector[detector][i+1][2]);
+                
+                if( fHitsDetector[detector][i][2]>fHitsDetector[detector][i+1][2] && fHitsDetector[detector][i][2]>fHitsDetector[detector][i-1][2] )
+                    FillH2(LocalMax_Right_ToT_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i+1][2]);
+            }
+            if (fHitsDetector[detector][i][2]>0 && fHitsDetector[detector][i-1][2]>0 && fHits[otherTDC][otherChannel][2] >0 && TMath::Abs(fHitsDetector[detector][i][0]-fHits[otherTDC][otherChannel][0]) <15 ){
+                FillH2(Left_ToT_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i-1][2]);
+                
+                if( fHitsDetector[detector][i][2]>fHitsDetector[detector][i-1][2] && fHitsDetector[detector][i][2]>fHitsDetector[detector][i+1][2])
+                    FillH2(LocalMax_Left_ToT_Crosstalk[position[0]][position[1]], fHitsDetector[detector][i][2], fHitsDetector[detector][i-1][2]);
+            }
+            
+            channel =0;
+            TDC =0;
+            otherTDC=0;
+            otherChannel=0;
+        }
+    }
+    
+    void CreateDetectorScheme(){
+        // Iterate over detectorScheme and fill it
+        for(int i = 1; i < numOfDetectors; i++)
+            for(int j = 1; j < 17; j++) {
+                
+                // Iterate over correlationScheme
+                for(int k = 1; k < numOfTDCs; k++) // TDC number
+                    for(int z = 1; z < numOfChannels; z++) // channel
+                        if(correlationScheme[k][z][2] == i && correlationScheme[k][z][3] == j){
+                            detectorScheme[i][j][0] = k;
+                            detectorScheme[i][j][1] = z;
+                        }
+            }
+    }
+    
+    //Main function of this analysis
+    virtual bool Process(base::Event* ev)
+    {
+        eventnumber++;
+//         if(eventnumber > 2000000){
+//             cout << "2 000 000 events passed, analysis stopped" << endl;
+//             return false;
+//         }
+        
+        hadaq::TdcSubEventFloat* sub1 =
+            dynamic_cast<hadaq::TdcSubEventFloat*> (ev->GetSubEvent(fTdcId1));
+
+        hadaq::TdcSubEventFloat* sub2 =
+            dynamic_cast<hadaq::TdcSubEventFloat*> (ev->GetSubEvent(fTdcId2));
+
+        hadaq::TdcSubEventFloat* sub3 =
+           dynamic_cast<hadaq::TdcSubEventFloat*> (ev->GetSubEvent(fTdcId3));
+
+        hadaq::TdcSubEventFloat* sub4 =
+           dynamic_cast<hadaq::TdcSubEventFloat*> (ev->GetSubEvent(fTdcId4));
+        
+        long num1 = 0;
+        long num2 = 0;
+        long num3 = 0;
+        long num4 = 0;
+        long num = 0;
+        
+        num1 = Unpack(sub1, 0, fHits[1], fSubEvents[1], fSubTdiff[1]);
+        num2 = Unpack(sub2, 20, fHits[2], fSubEvents[2], fSubTdiff[2]);
+        num3 = Unpack(sub3, 40, fHits[3], fSubEvents[3], fSubTdiff[3]);
+        num4 = Unpack(sub4, 60, fHits[4], fSubEvents[4], fSubTdiff[4]);
+       
+        num=num1+num2+num3+num4;
+        
+        // when return false, event processing is cancelled
+        if (num == 0 ) return false;
+        
+        FillH1(hNumHits, num);
+         
+        for(int i = 1; i < numOfDetectors; i++) {
+            FillDetectorHits(i);
+            
+//            
+//             if(makeCrosstalkAnalysis){
+//                 FillNeighbouringCrosstalk(i);
+//                 FillTDiff_Left_Right(i);
+//                 FillToT_Left_Right(i);
+//             }
+//             
+            if (makeClusterAnalysis || makeClusterRemovalCorrCh || makeClusterRemovalBothCh){
+                detectorClusters[i] = findClusters(i);
+//                 
+                if (makeClusterAnalysis )
+                    FillClusterDetector(i);
+                if (makeClusterRemovalCorrCh || makeClusterRemovalBothCh) 
+                    RemoveClusters(i);
+            }
+        }
+//         
+        createClearedfHits();
+
+        //Fill ToT    
+        FillToT(1,1);
+//         FillToT(1,2);
+//         FillToT(2,1);
+//         FillToT(3,4);
+//         FillToT(4,3);
+        
+        //Fill TDiff histograms
+        for(int i=1; i < numOfTDCs; i++) // Reference TDC
+            for(int j = 1; j <numOfTDCs; j++) // Correlated TDC
+               // if(i != j)
+                    FillTDiff(i, j, (j-1) * 20);
+        
+        
+        
+        return true;
+    };
+};
+
+void second()
+{
+    new SecondProc("TEST", "TDC_2600", "TDC_2601", "TDC_2602", "TDC_2603");
+}
diff --git a/users/ufsd_trb3_sdalinac/startup.sh b/users/ufsd_trb3_sdalinac/startup.sh
new file mode 100755 (executable)
index 0000000..953a768
--- /dev/null
@@ -0,0 +1,138 @@
+#!/bin/bash
+
+DAQ_TOOLS_PATH=~/trbsoft/daqtools
+USER_DIR=~/trbsoft/daqtools/users/ufsd_trb3_sdalinac
+TRB_WEB_DIR=$DAQ_TOOLS_PATH/web
+
+export PATH=$PATH:$DAQ_TOOLS_PATH
+export PATH=$PATH:$DAQ_TOOLS_PATH/tools
+export PATH=$PATH:$USER_DIR
+
+export TRB3_SERVER=trb260:26000
+export TRBNETDPID=$(pgrep -f "trbnetd -i 96")
+#export DAQOPSERVER=hadeb05:84
+export DAQOPSERVER=localhost:96
+
+echo "- trbnetd pid: $TRBNETDPID"
+
+if [[ -z "$TRBNETDPID" ]] 
+then
+    ~/trbsoft/trbnettools/bin/trbnetd -i 96
+fi
+
+#./check_ping.pl --reboot
+
+
+echo "reset"
+trbcmd reset
+#./trbreset_loop.pl
+#sleep 1;
+
+##################################################
+## Set addresses
+##################################################
+merge_serial_address.pl $DAQ_TOOLS_PATH/base/serials_trb3.db $USER_DIR/db/addresses_trb3.db
+#merge_serial_address.pl $DAQ_TOOLS_PATH/base/serials_dirich.db $USER_DIR/db/addresses_dirich.db
+#merge_serial_address.pl $DAQ_TOOLS_PATH/base/serials_dirich_concentrator.db $USER_DIR/db/addresses_dirich_concentrator.db
+
+
+#echo "disable port 6 on hub 0x8841"
+#trbcmd clearbit 0x8841 0xc0 0x40
+#trbcmd clearbit 0x8841 0xc1 0x40
+#trbcmd clearbit 0x8841 0xc3 0x40
+
+
+echo "GbE settings"
+loadregisterdb.pl db/register_configgbe.db
+loadregisterdb.pl db/register_configgbe_ip.db
+
+echo "TDC settings"
+loadregisterdb.pl db/register_configtdc.db
+echo "TDC settings end"
+
+# setup central FPGA - enable peripherial signals
+#switchport.pl 0x8841 6 off
+
+
+
+
+
+
+# pulser to 100kHz and 50kHz
+#trbcmd w 0xc840 0xa156 0x0000270f #10khz pulser 0
+
+#trbcmd w 0xc840 0xa150 0x000003e7 #100khz
+#trbcmd w 0xc840 0xa150 0x0001869f #1khz
+#trbcmd w 0xc840 0xa150 0x00001387 #20khz
+#trbcmd w 0xc840 0xa150 0x00000d04 #30khz
+#trbcmd w 0xc840 0xa150 0x000007cf #50khz
+#trbcmd w 0xc840 0xa157 0x0000270f #10khz
+
+#trbcmd setbit 0xc840 0xa101 0x2 #enable pulser 0
+#trbcmd setbit 0xc840 0xa101 0x2 #enable pulser 1
+#trbcmd setbit 0xc840 0xa101 0x20 #enable Addon Multiplexer 1
+#trbcmd setbit 0xc840 0xa101 0x8 #enable CTS Addon 0
+#trbcmd setbit 0xc840 0xa101 0x200 #enable periph fpga input as trigger
+
+
+# trigger on TDC channel 1
+#trbcmd setbit 0x0810 0xcf00 0x1     #direct TDC input to CTS
+#trbcmd setbit 0xc001 0xa14d 0x2     #select F5_COMM input
+#trbcmd setbit 0xc840 0xa101 0x200   #enable input at CTS
+
+# set correct timeout: off for channel 0, 1, 2sec for 2
+trbcmd w 0xfffe 0xc5 0x50ff
+
+#Dirich-Concentrator: enable reference time from RJ45
+######trbcmd loadbit 0x8300 0xd580 0x6 0x6
+
+echo "pulser"
+# pulser #0 to 10 kHz
+trbcmd w 0xc001 0xa158 0x0000270f   
+#trbcmd w 0xc001 0xa150 0x0022270f   
+
+echo "trigger type"
+# set trigger type to 0x1
+#trbcmd setbit 0xc001 0xa15e 0x10
+
+
+#trbcmd clearbit 0x1130 0xc801 0x80000000 # disable window
+#trbcmd w 0x1130 0xc802 0xffff0000 # enable upper 16 channels for padiwa
+#trbcmd w 0x1580 0xc802 0xffffffff # enable upper 16 channels for padiwa
+
+
+cd ~/trbsoft/daqtools/xml-db
+./put.pl Readout 0xfe51 SetMaxEventSize 500
+cd $USER_DIR
+
+#trbcmd w 0xfe51 0xdf80 0xffffffff # enable monitor counters
+
+#trbcmd w 0x1133 0xc804 0x7c # max number of words
+#trbcmd clearbit 0x1133 0xc801 0x80000000 # disable window
+
+#trbcmd w 0x1133 0xc802 0x00000c03 # enable pulser
+
+#trbcmd setbit 0xc001 0xa101 0x8 # enable external trigger in of CTS
+
+#echo "enable trigger inputs on peripheral FPGA"
+#trbcmd w 0x2600 0xdf00 0xffff
+
+~/trbsoft/daqtools/tools//padiwa.pl 0x2600 0 invert 0xffff 
+
+#echo "pulser enable"
+# pulser enable
+ trbcmd setbit 0xc001 0xa101 0x1
+
+#echo "enable external trigger from peripheral FPGA"
+#trbcmd setbit 0xc001 0xa101 0x1000
+
+
+# enable trigger from peripheral FPGA
+trbcmd w 0xc001 0xa153 0x2
+
+#echo "Loading NINO Thresholds"
+#../../tools/dac_program.pl DAC_nino.db
+
+
+
+