Project
Loading...
Searching...
No Matches
genEvents.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
14
15#include <iostream>
16#include <fstream>
17#include <cstring>
18#include <sys/stat.h>
19
20#include "Rtypes.h"
21#include "TRandom.h"
22#include "TH1F.h"
23#include "TFile.h"
24#include "TCanvas.h"
25#include "TPad.h"
26
27#include <iostream>
28#include <iomanip>
29#include <limits>
30
31#include "genEvents.h"
32#include "GPUTPCClusterData.h"
33#include "GPUTPCMCInfo.h"
35#include "GPUParam.inc"
37#include "GPUTPCGMPropagator.h"
38#include "GPUTPCGMMerger.h"
39#include "GPUChainTracking.h"
40
41#include "../utils/qconfig.h"
42
43using namespace o2::gpu;
44using namespace std;
45namespace o2::gpu
46{
47extern GPUSettingsStandalone configStandalone;
48}
49
50int32_t genEvents::GetSector(double GlobalPhi)
51{
52 double phi = GlobalPhi;
53 // std::cout<<" GetSector: phi = "<<phi<<std::endl;
54
55 if (phi >= mTwoPi) {
56 phi -= mTwoPi;
57 }
58 if (phi < 0) {
59 phi += mTwoPi;
60 }
61 return (int32_t)(phi / mSectorDAngle);
62}
63
64int32_t genEvents::GetDSector(double LocalPhi) { return GetSector(LocalPhi + mSectorAngleOffset); }
65
66double genEvents::GetSectorAngle(int32_t iSector) { return mSectorAngleOffset + iSector * mSectorDAngle; }
67
68int32_t genEvents::RecalculateSector(GPUTPCGMPhysicalTrackModel& t, int32_t& iSector)
69{
70 double phi = atan2(t.GetY(), t.GetX());
71 // std::cout<<" recalculate: phi = "<<phi<<std::endl;
72 int32_t dSector = GetDSector(phi);
73
74 if (dSector == 0) {
75 return 0; // nothing to do
76 }
77 // std::cout<<" dSector = "<<dSector<<std::endl;
78 double dAlpha = dSector * mSectorDAngle;
79 // rotate track on angle dAlpha
80
81 t.Rotate(dAlpha);
82
83 iSector += dSector;
84 if (iSector >= 18) {
85 iSector -= 18;
86 }
87 return 1;
88}
89
90double genEvents::GetGaus(double sigma)
91{
92 double x = 0;
93 do {
94 x = gRandom->Gaus(0., sigma);
95 if (fabs(x) <= 3.5 * sigma) {
96 break;
97 }
98 } while (1);
99 return x;
100}
101
103{
104 const char* rows[3] = {"0-63", "128-159", "64-127"};
105 for (int32_t i = 0; i < 3; i++) {
106 for (int32_t j = 0; j < 2; j++) {
107 char name[1024], title[1024];
108
109 snprintf(name, 1024, "clError%s%d", (j == 0 ? "Y" : "Z"), i);
110
111 snprintf(title, 1024, "Cluster %s Error for rows %s", (j == 0 ? "Y" : "Z"), rows[i]);
112
113 mClusterError[i][j] = new TH1F(name, title, 1000, 0., .7);
114 mClusterError[i][j]->GetXaxis()->SetTitle("Cluster Error [cm]");
115 }
116 }
117}
118
120{
121 TFile* tout = new TFile("generator.root", "RECREATE");
122 TCanvas* c = new TCanvas("ClusterErrors", "Cluste rErrors", 0, 0, 700, 700. * 2. / 3.);
123 c->Divide(3, 2);
124 int32_t ipad = 1;
125 for (int32_t j = 0; j < 2; j++) {
126 for (int32_t i = 0; i < 3; i++) {
127 c->cd(ipad++);
128 int32_t k = i;
129 if (i == 1) {
130 k = 2;
131 }
132 if (i == 2) {
133 k = 1;
134 }
135 if (tout) {
136 mClusterError[k][j]->Write();
137 }
138 gPad->SetLogy();
139 mClusterError[k][j]->Draw();
140 // delete mClusterError[i][j];
141 // mClusterError[i][j]=0;
142 }
143 }
144 c->Print("plots/clusterErrors.pdf");
145 delete c;
146 if (tout) {
147 tout->Close();
148 delete tout;
149 }
150}
151
152int32_t genEvents::GenerateEvent(const GPUParam& param, char* filename)
153{
154 mRec->ClearIOPointers();
155 static int32_t iEvent = -1;
156 iEvent++;
157 if (iEvent == 0) {
158 gRandom->SetSeed(configStandalone.seed);
159 }
160
161 int32_t nTracks = configStandalone.EG.numberOfTracks; // Number of MC tracks, must be at least as large as the largest fMCID assigned above
162 cout << "NTracks " << nTracks << endl;
163 std::vector<GPUTPCMCInfo> mcInfo(nTracks);
164 memset(mcInfo.data(), 0, nTracks * sizeof(mcInfo[0]));
165
166 // double Bz = param.ConstBz();
167 // std::cout<<"Bz[kG] = "<<param.BzkG()<<std::endl;
168
170 {
171 prop.SetToyMCEventsFlag(kTRUE);
172 const GPUTPCGMMerger& merger = mRec->GetTPCMerger();
173 prop.SetPolynomialField(&merger.Param().polynomialField);
174 }
175
176 // Bz*=o2::gpu::gpu_common_constants::kCLight;
177
178 std::vector<GenCluster> vClusters;
179 int32_t clusterId = 0; // Here we count up the cluster ids we fill (must be unique).
180 // gRandom->SetSeed(0);
181 // uint32_t seed = gRandom->GetSeed();
182
183 for (int32_t itr = 0; itr < nTracks; itr++) {
184 // std::cout<<"Track "<<itr<<":"<<std::endl;
185 // gRandom->SetSeed(seed);
186
187 mcInfo[itr].pid = -100; //-100: Unknown / other, 0: Electron, 1, Muon, 2: Pion, 3: Kaon, 4: Proton
188 mcInfo[itr].charge = 1;
189 mcInfo[itr].prim = 1; // Primary particle
190 mcInfo[itr].primDaughters = 0; // Primary particle with daughters in the TPC
191 mcInfo[itr].x = 0; // Position of MC track at entry of TPC / first hit in the TPC
192 mcInfo[itr].y = 0;
193 mcInfo[itr].z = 0;
194 mcInfo[itr].pX = 0; // Momentum of MC track at that position
195 mcInfo[itr].pY = 0;
196 mcInfo[itr].pZ = 0;
197
199 double dphi = mTwoPi / nTracks;
200 double phi = mSectorAngleOffset + dphi * itr;
201 double eta = gRandom->Uniform(-1.5, 1.5);
202
203 double theta = 2 * std::atan(1. / exp(eta));
204 double lambda = theta - M_PI / 2;
205 // double theta = gRandom->Uniform(-60,60)*M_PI/180.;
206 double pt = .08 * std::pow(10, gRandom->Uniform(0, 2.2));
207
208 double q = 1.;
209 int32_t iSector = GetSector(phi);
210 phi = phi - GetSectorAngle(iSector);
211
212 // std::cout<<"phi = "<<phi<<std::endl;
213 double x0 = cosf(phi);
214 double y0 = sinf(phi);
215 double z0 = tanf(lambda);
216 t.Set(x0, y0, z0, pt * x0, pt * y0, pt * z0, q);
217
218 if (RecalculateSector(t, iSector) != 0) {
219 std::cout << "Initial sector wrong!!!" << std::endl;
220 // exit(0);
221 }
222
223 for (int32_t iRow = 0; iRow < GPUCA_ROW_COUNT; iRow++) {
224 // if( iRow>=50 ) break; //SG!!!
225 float xRow = GPUTPCGeometry::Row2X(iRow);
226 // transport to row
227 int32_t err = 0;
228 for (int32_t itry = 0; itry < 1; itry++) {
229 float B[3];
230 prop.GetBxByBz(GetSectorAngle(iSector), t.GetX(), t.GetY(), t.GetZ(), B);
231 float dLp = 0;
232 err = t.PropagateToXBxByBz(xRow, B[0], B[1], B[2], dLp);
233 if (err) {
234 std::cout << "Can not propagate to x = " << xRow << std::endl;
235 t.Print();
236 break;
237 }
238 if (fabsf(t.GetZ()) >= GPUTPCGeometry::TPCLength()) {
239 std::cout << "Can not propagate to x = " << xRow << ": Z outside the volume" << std::endl;
240 t.Print();
241 err = -1;
242 break;
243 }
244 // rotate track coordinate system to current sector
245 int32_t isNewSector = RecalculateSector(t, iSector);
246 if (!isNewSector) {
247 break;
248 } else {
249 std::cout << "track " << itr << ": new sector " << iSector << " at row " << iRow << std::endl;
250 }
251 }
252 if (err) {
253 break;
254 }
255 // std::cout<<" track "<<itr<<": Sector "<<iSector<<" row "<<iRow<<" params :"<<std::endl;
256 // t.Print();
257 // track at row iRow, sector iSector
258 if (iRow == 0) { // store MC track at first row
259 // std::cout<<std::setprecision( 20 );
260 // std::cout<<"track "<<itr<<": x "<<t.X()<<" y "<<t.Y()<<" z "<<t.Z()<<std::endl;
261 GPUTPCGMPhysicalTrackModel tg(t); // global coordinates
262 tg.Rotate(-GetSectorAngle(iSector));
263
264 mcInfo[itr].pid = 2; // pion
265 mcInfo[itr].charge = 3 * q;
266 mcInfo[itr].x = tg.GetX(); // Position of MC track at entry of TPC / first hit in the TPC
267 mcInfo[itr].y = tg.GetY();
268 mcInfo[itr].z = tg.GetZ();
269 mcInfo[itr].pX = tg.GetPx(); // Momentum of MC track at that position
270 mcInfo[itr].pY = tg.GetPy();
271 mcInfo[itr].pZ = tg.GetPz();
272 // std::cout<<" mc Z = "<<tg.GetZ()<<std::endl;
273 }
274
275 // create cluster
276 GenCluster c;
277 float sigmaY = 0.3;
278 float sigmaZ = 0.5;
279 const int32_t rowType = iRow < 64 ? 0 : iRow < 128 ? 2 : 1;
280 t.UpdateValues();
281 param.GetClusterErrors2(iSector, rowType, t.GetZ(), t.GetSinPhi(), t.GetDzDs(), -1.f, 0.f, 0.f, sigmaY, sigmaZ);
282 sigmaY = std::sqrt(sigmaY);
283 sigmaZ = std::sqrt(sigmaZ);
284 mClusterError[rowType][0]->Fill(sigmaY);
285 mClusterError[rowType][1]->Fill(sigmaZ);
286 // std::cout<<sigmaY<<" "<<sigmaY<<std::endl;
287 // if( sigmaY > 0.5 ) sigmaY = 0.5;
288 // if( sigmaZ > 0.5 ) sigmaZ = 0.5;
289 c.sector = (t.GetZ() >= 0.) ? iSector : iSector + 18;
290 c.row = iRow;
291 c.mcID = itr;
292 c.x = t.GetX();
293 c.y = t.GetY() + GetGaus(sigmaY);
294 c.z = t.GetZ() + GetGaus(sigmaZ);
295 c.id = clusterId++;
296 vClusters.push_back(c);
297 } // iRow
298 } // itr
299
300 std::vector<AliHLTTPCClusterMCLabel> labels;
301
302 std::unique_ptr<GPUTPCClusterData> clSectors[GPUChainTracking::NSECTORS];
303
304 for (int32_t iSector = 0; iSector < (int32_t)GPUChainTracking::NSECTORS; iSector++) // HLT Sector numbering, sectors go from 0 to 35, all spanning all rows from 0 to 158.
305 {
306 int32_t nNumberOfHits = 0;
307 for (uint32_t i = 0; i < vClusters.size(); i++) {
308 if (vClusters[i].sector == iSector) {
309 nNumberOfHits++;
310 }
311 }
312 // For every sector we first have to fill the number of hits in this sector to the file
313 mRec->mIOPtrs.nClusterData[iSector] = nNumberOfHits;
314
315 GPUTPCClusterData* clusters = new GPUTPCClusterData[nNumberOfHits];
316 clSectors[iSector].reset(clusters);
317 int32_t icl = 0;
318 for (uint32_t i = 0; i < vClusters.size(); i++) {
319 GenCluster& c = vClusters[i];
320 if (c.sector == iSector) {
321 clusters[icl].id = c.id;
322 clusters[icl].row = c.row; // We fill one hit per TPC row
323 clusters[icl].x = c.x;
324 clusters[icl].y = c.y;
325 clusters[icl].z = c.z;
326 clusters[icl].amp = 100; // Arbitrary amplitude
327 icl++;
328 AliHLTTPCClusterMCLabel clusterLabel;
329 for (int32_t j = 0; j < 3; j++) {
330 clusterLabel.fClusterID[j].fMCID = -1;
331 clusterLabel.fClusterID[j].fWeight = 0;
332 }
333 clusterLabel.fClusterID[0].fMCID = c.mcID;
334 clusterLabel.fClusterID[0].fWeight = 1;
335 labels.push_back(clusterLabel);
336 }
337 }
338 mRec->mIOPtrs.clusterData[iSector] = clusters;
339 }
340
341 // Create vector with cluster MC labels, clusters are counter from 0 to clusterId in the order they have been written above. No separation in sectors.
342
343 mRec->mIOPtrs.nMCLabelsTPC = labels.size();
344 mRec->mIOPtrs.mcLabelsTPC = labels.data();
345
346 mRec->mIOPtrs.nMCInfosTPC = mcInfo.size();
347 mRec->mIOPtrs.mcInfosTPC = mcInfo.data();
348 static const GPUTPCMCInfoCol mcColInfo = {0, (uint32_t)mcInfo.size()};
349 mRec->mIOPtrs.mcInfosTPCCol = &mcColInfo;
350 mRec->mIOPtrs.nMCInfosTPCCol = 1;
351
352 mRec->DumpData(filename);
353 labels.clear();
354 mcInfo.clear();
355 return (0);
356}
357
359{
360 std::unique_ptr<genEvents> gen(new genEvents(rec));
361 char dirname[256];
362 snprintf(dirname, 256, "events/%s/", configStandalone.eventsDir);
363 mkdir(dirname, S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);
364 rec->DumpSettings(dirname);
365
366 gen->InitEventGenerator();
367
368 for (int32_t i = 0; i < (configStandalone.nEvents == -1 ? 10 : configStandalone.nEvents); i++) {
369 GPUInfo("Generating event %d/%d", i, configStandalone.nEvents == -1 ? 10 : configStandalone.nEvents);
370 snprintf(dirname, 256, "events/%s/" GPUCA_EVDUMP_FILE ".%d.dump", configStandalone.eventsDir, i);
371 gen->GenerateEvent(rec->GetParam(), dirname);
372 }
373 gen->FinishEventGenerator();
374}
default_random_engine gen(dev())
uint64_t exp(uint64_t base, uint8_t exp) noexcept
int32_t i
#define GPUCA_EVDUMP_FILE
Definition GPUDef.h:40
#define GPUCA_ROW_COUNT
uint32_t j
Definition RawData.h:0
uint32_t c
Definition RawData.h:2
Definition B.h:16
static constexpr int32_t NSECTORS
Definition GPUChain.h:54
const GPUParam & GetParam() const
void DumpSettings(const char *dir="")
float const GPUParam float float float float float int32_t int16_t int8_t float float float charge
static void RunEventGenerator(GPUChainTracking *rec)
Definition genEvents.h:34
void InitEventGenerator()
Definition genEvents.h:30
int32_t GenerateEvent(const GPUParam &sectorParam, char *filename)
Definition genEvents.h:31
void FinishEventGenerator()
Definition genEvents.h:32
GLint GLenum GLint x
Definition glcorearb.h:403
GLuint const GLchar * name
Definition glcorearb.h:781
GLuint GLfloat x0
Definition glcorearb.h:5034
GLenum GLfloat param
Definition glcorearb.h:271
GLuint GLfloat GLfloat y0
Definition glcorearb.h:5034
std::ostream & cout()
GPUSettingsStandalone configStandalone
Definition genEvents.cxx:47
Defining DataPointCompositeObject explicitly as copiable.
std::string filename()
GPUReconstruction * rec
AliHLTTPCClusterMCWeight fClusterID[3]
const int nEvents
Definition test_Fifo.cxx:27
std::vector< Cluster > clusters
std::vector< ReadoutWindowData > rows