Newer
Older

Sylvester Joosten
committed
} // condition for good scin time
ihhit++;
} // loop over hits of a plane
} // loop over planes

Sylvester Joosten
committed
fBetaNoTrk = fBetaNoTrk / 29.979; // velocity / c

Sylvester Joosten
committed
} // condition for fTmpDenom

Sylvester Joosten
committed
fBetaNoTrk = 0.;
fBetaNoTrkChiSq = -2.;
} // else condition for fTmpDenom

Sylvester Joosten
committed
fGoodEventTOFCalib = kFALSE;
if ((fNumPlanesBetaCalc == 4) && goodplanetime[0] && goodplanetime[1] && goodplanetime[2] &&
goodplanetime[3] && fPlanes[0]->GetNGoodHits() == 1 && fPlanes[1]->GetNGoodHits() == 1 &&
fPlanes[2]->GetNGoodHits() == 1 && fPlanes[3]->GetNGoodHits() == 1)
fGoodEventTOFCalib = kTRUE;
if ((fNumPlanesBetaCalc == 3) && goodplanetime[0] && goodplanetime[1] && goodplanetime[2] &&
fPlanes[0]->GetNGoodHits() == 1 && fPlanes[1]->GetNGoodHits() == 1 &&
fPlanes[2]->GetNGoodHits() == 1)
fGoodEventTOFCalib = kTRUE;
//_____________________________________________________________________________

Sylvester Joosten
committed
Int_t THcHodoscope::ApplyCorrections(void) { return (0); }
//_____________________________________________________________________________

Sylvester Joosten
committed
Double_t THcHodoscope::TimeWalkCorrection(const Int_t& paddle, const ESide side) { return (0.0); }
//_____________________________________________________________________________

Sylvester Joosten
committed
Int_t THcHodoscope::CoarseProcess(TClonesArray& tracks) {

Sylvester Joosten
committed
Int_t ntracks = tracks.GetLast() + 1; // Number of reconstructed tracks
// -------------------------------------------------
// fDumpOut << " ntrack = " << ntracks << endl;

Sylvester Joosten
committed
if (ntracks > 0) {
// **MAIN LOOP: Loop over all tracks and get corrected time, tof, beta...
vector<Double_t> nPmtHit(ntracks);
vector<Double_t> timeAtFP(ntracks);
fdEdX.reserve(ntracks);
fGoodFlags.reserve(ntracks);

Sylvester Joosten
committed
for (Int_t itrack = 0; itrack < ntracks; itrack++) { // Line 133
nPmtHit[itrack] = 0;
timeAtFP[itrack] = 0;
THaTrack* theTrack = dynamic_cast<THaTrack*>(tracks.At(itrack));
if (!theTrack)
return -1;
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
fGoodPlaneTime[ip] = kFALSE;
fNScinHits[ip] = 0;
fNPlaneTime[ip] = 0;
fSumPlaneTime[ip] = 0.;

Sylvester Joosten
committed
std::vector<Double_t> dedx_temp;
std::vector<std::vector<GoodFlags>> goodflagstmp1;
goodflagstmp1.reserve(fNumPlanesBetaCalc);
#if __cplusplus >= 201103L
fdEdX.push_back(std::move(dedx_temp)); // Create array of dedx per hit
fGoodFlags.push_back(std::move(goodflagstmp1));
#else
fdEdX.push_back(dedx_temp); // Create array of dedx per hit
fGoodFlags.push_back(goodflagstmp1);

Sylvester Joosten
committed
Int_t nFPTime = 0;

Sylvester Joosten
committed
Double_t beta = 0;
// timeAtFP[itrack] = 0.;
Double_t sumFPTime = 0.; // Line 138
fNScinHit.push_back(0);
//! Calculate all corrected hit times and histogram
//! This uses a copy of code below. Results are save in time_pos,neg
//! including the z-pos. correction assuming nominal value of betap
//! Code is currently hard-wired to look for a peak in the
//! range of 0 to 100 nsec, with a group of times that all
//! agree withing a time_tolerance of time_tolerance nsec. The normal
//! peak position appears to be around 35 nsec.
//! NOTE: if want to find particles with beta different than
//! reference particle, need to make sure this is big enough
//! to accomodate difference in TOF for other particles
//! Default value in case user hasnt defined something reasonable
// Loop over scintillator planes.
// In ENGINE, its loop over good scintillator hits.

Sylvester Joosten
committed
fTOFCalc.clear(); // SAW - Can we
fTOFPInfo.clear(); // SAW - combine these two?
Int_t ihhit = 0; // Hit # overall

Sylvester Joosten
committed
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {

Sylvester Joosten
committed
std::vector<GoodFlags> goodflagstmp2;
goodflagstmp2.reserve(fNScinHits[ip]);
#if __cplusplus >= 201103L

Sylvester Joosten
committed
fGoodFlags[itrack].push_back(std::move(goodflagstmp2));

Sylvester Joosten
committed
fGoodFlags[itrack].push_back(goodflagstmp2);

Sylvester Joosten
committed
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
fNScinHits[ip] = fPlanes[ip]->GetNScinHits();
TClonesArray* hodoHits = fPlanes[ip]->GetHits();
Double_t zPos = fPlanes[ip]->GetZpos();
Double_t dzPos = fPlanes[ip]->GetDzpos();
// first loop over hits with in a single plane
for (Int_t iphit = 0; iphit < fNScinHits[ip]; iphit++) {
// iphit is hit # within a plane
THcHodoHit* hit = (THcHodoHit*)hodoHits->At(iphit);
fTOFPInfo.push_back(TOFPInfo());
// Can remove these as we will initialize in the constructor
// fTOFPInfo[ihhit].time_pos = -99.0;
// fTOFPInfo[ihhit].time_neg = -99.0;
// fTOFPInfo[ihhit].keep_pos = kFALSE;
// fTOFPInfo[ihhit].keep_neg = kFALSE;
fTOFPInfo[ihhit].scin_pos_time = 0.0;
fTOFPInfo[ihhit].scin_neg_time = 0.0;
fTOFPInfo[ihhit].hit = hit;
fTOFPInfo[ihhit].planeIndex = ip;
fTOFPInfo[ihhit].hitNumInPlane = iphit;
fTOFPInfo[ihhit].onTrack = kFALSE;
Int_t paddle = hit->GetPaddleNumber() - 1;
Double_t zposition = zPos + (paddle % 2) * dzPos;
Double_t xHitCoord = theTrack->GetX() + theTrack->GetTheta() * (zposition); // Line 183
Double_t yHitCoord = theTrack->GetY() + theTrack->GetPhi() * (zposition); // Line 184
Double_t scinTrnsCoord, scinLongCoord;
if ((ip == 0) || (ip == 2)) { // !x plane. Line 185
scinTrnsCoord = xHitCoord;
scinLongCoord = yHitCoord;
} else if ((ip == 1) || (ip == 3)) { // !y plane. Line 188
scinTrnsCoord = yHitCoord;
scinLongCoord = xHitCoord;
} else {
return -1;
} // Line 195
fTOFPInfo[ihhit].scinTrnsCoord = scinTrnsCoord;
fTOFPInfo[ihhit].scinLongCoord = scinLongCoord;
Double_t scinCenter = fPlanes[ip]->GetPosCenter(paddle) + fPlanes[ip]->GetPosOffset();
// Index to access the 2d arrays of paddle/scintillator properties
Int_t fPIndex = GetScinIndex(ip, paddle);
Double_t betatrack = theTrack->GetP() / TMath::Sqrt(theTrack->GetP() * theTrack->GetP() +
fPartMass * fPartMass);
if (TMath::Abs(scinCenter - scinTrnsCoord) <
(fPlanes[ip]->GetSize() * 0.5 + fPlanes[ip]->GetHodoSlop())) { // Line 293
fTOFPInfo[ihhit].onTrack = kTRUE;
Double_t zcor = zposition / (29.979 * betatrack) *
TMath::Sqrt(1. + theTrack->GetTheta() * theTrack->GetTheta() +
theTrack->GetPhi() * theTrack->GetPhi());
fTOFPInfo[ihhit].zcor = zcor;
if (fCosmicFlag) {
Double_t zcor = -zposition / (29.979 * 1.0) *
TMath::Sqrt(1. + theTrack->GetTheta() * theTrack->GetTheta() +
theTrack->GetPhi() * theTrack->GetPhi());
fTOFPInfo[ihhit].zcor = zcor;
}
Double_t tdc_pos = hit->GetPosTDC();
if (tdc_pos >= fScinTdcMin && tdc_pos <= fScinTdcMax) {
Double_t adc_pos = hit->GetPosADC();
Double_t adcamp_pos = hit->GetPosADCpeak();
Double_t pathp = fPlanes[ip]->GetPosLeft() - scinLongCoord;
fTOFPInfo[ihhit].pathp = pathp;
Double_t timep = tdc_pos * fScinTdcToTime;
if (fTofUsingInvAdc) {
timep -= fHodoPosInvAdcOffset[fPIndex] + pathp / fHodoPosInvAdcLinear[fPIndex] +
fHodoPosInvAdcAdc[fPIndex] / TMath::Sqrt(TMath::Max(20.0 * .020, adc_pos));
} else {
// Double_t tw_corr_pos =
// fHodoPos_c1[fPIndex]/pow(adcamp_pos/fTdc_Thrs,fHodoPos_c2[fPIndex]) -
// fHodoPos_c1[fPIndex]/pow(200./fTdc_Thrs, fHodoPos_c2[fPIndex]);
Double_t tw_corr_pos = 0.;
pathp = scinLongCoord;
if (adcamp_pos > 0)
tw_corr_pos = 1. / pow(adcamp_pos / fTdc_Thrs, fHodoPos_c2[fPIndex]) -
1. / pow(200. / fTdc_Thrs, fHodoPos_c2[fPIndex]);
timep += -tw_corr_pos + fHodo_LCoeff[fPIndex] + pathp / fHodoVelFit[fPIndex];
}
fTOFPInfo[ihhit].scin_pos_time = timep;
timep -= zcor;
fTOFPInfo[ihhit].time_pos = timep;

Sylvester Joosten
committed
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
}
Double_t tdc_neg = hit->GetNegTDC();
if (tdc_neg >= fScinTdcMin && tdc_neg <= fScinTdcMax) {
Double_t adc_neg = hit->GetNegADC();
Double_t adcamp_neg = hit->GetNegADCpeak();
Double_t pathn = scinLongCoord - fPlanes[ip]->GetPosRight();
fTOFPInfo[ihhit].pathn = pathn;
Double_t timen = tdc_neg * fScinTdcToTime;
if (fTofUsingInvAdc) {
timen -= fHodoNegInvAdcOffset[fPIndex] + pathn / fHodoNegInvAdcLinear[fPIndex] +
fHodoNegInvAdcAdc[fPIndex] / TMath::Sqrt(TMath::Max(20.0 * .020, adc_neg));
} else {
pathn = scinLongCoord;
Double_t tw_corr_neg = 0;
if (adcamp_neg > 0)
tw_corr_neg = 1. / pow(adcamp_neg / fTdc_Thrs, fHodoNeg_c2[fPIndex]) -
1. / pow(200. / fTdc_Thrs, fHodoNeg_c2[fPIndex]);
timen += -tw_corr_neg - 2 * fHodoCableFit[fPIndex] + fHodo_LCoeff[fPIndex] -
pathn / fHodoVelFit[fPIndex];
}
fTOFPInfo[ihhit].scin_neg_time = timen;
timen -= zcor;
fTOFPInfo[ihhit].time_neg = timen;

Sylvester Joosten
committed
}
} // condition for cenetr on a paddle
ihhit++;
} // First loop over hits in a plane <---------
//-----------------------------------------------------------------------------------------------
//------------- First large loop over scintillator hits ends here --------------------
//-----------------------------------------------------------------------------------------------

Sylvester Joosten
committed
Int_t nhits = ihhit;
if (0.5 * hTime->GetMaximumBin() > 0) {
Double_t tmin = 0.5 * hTime->GetMaximumBin();
for (Int_t ih = 0; ih < nhits; ih++) { // loop over all scintillator hits
if ((fTOFPInfo[ih].time_pos > (tmin - fTofTolerance)) &&
(fTOFPInfo[ih].time_pos < (tmin + fTofTolerance))) {
fTOFPInfo[ih].keep_pos = kTRUE;
}
if ((fTOFPInfo[ih].time_neg > (tmin - fTofTolerance)) &&
(fTOFPInfo[ih].time_neg < (tmin + fTofTolerance))) {
fTOFPInfo[ih].keep_neg = kTRUE;
}
}
//---------------------------------------------------------------------------------------------

Sylvester Joosten
committed
// ---------------------- Second loop over scint. hits in a plane
// -----------------------------
//---------------------------------------------------------------------------------------------

Sylvester Joosten
committed
fdEdX[itrack].reserve(nhits);
fTOFCalc.reserve(nhits);

Sylvester Joosten
committed
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
for (Int_t ih = 0; ih < nhits; ih++) {
THcHodoHit* hit = fTOFPInfo[ih].hit;
Int_t iphit = fTOFPInfo[ih].hitNumInPlane;
Int_t ip = fTOFPInfo[ih].planeIndex;
// fDumpOut << " looping over hits = " << ih << " plane = " << ip+1 << endl;
// Flags are used by THcHodoEff
fGoodFlags[itrack][ip].reserve(nhits);
fGoodFlags[itrack][ip].push_back(GoodFlags());
assert(iphit >= 0 && (size_t)iphit < fGoodFlags[itrack][ip].size());
fGoodFlags[itrack][ip][iphit].onTrack = kFALSE;
fGoodFlags[itrack][ip][iphit].goodScinTime = kFALSE;
fGoodFlags[itrack][ip][iphit].goodTdcNeg = kFALSE;
fGoodFlags[itrack][ip][iphit].goodTdcPos = kFALSE;
fTOFCalc.push_back(TOFCalc());
// Do we set back to false for each track, or just once per event?
assert(ih >= 0 && (size_t)ih < fTOFCalc.size());
fTOFCalc[ih].good_scin_time = kFALSE;
// These need a track index too to calculate efficiencies
fTOFCalc[ih].good_tdc_pos = kFALSE;
fTOFCalc[ih].good_tdc_neg = kFALSE;
fTOFCalc[ih].pindex = ip;
Int_t paddle = hit->GetPaddleNumber() - 1;
fTOFCalc[ih].hit_paddle = paddle;
fTOFCalc[ih].good_raw_pad = paddle;
// Double_t scinCenter = fPlanes[ip]->GetPosCenter(paddle) +
// fPlanes[ip]->GetPosOffset(); Double_t scinTrnsCoord =
// fTOFPInfo[ih].scinTrnsCoord; Double_t scinLongCoord =
// fTOFPInfo[ih].scinLongCoord;
Int_t fPIndex = GetScinIndex(ip, paddle);
if (fTOFPInfo[ih].onTrack) {
fGoodFlags[itrack][ip][iphit].onTrack = kTRUE;
if (fTOFPInfo[ih].keep_pos) { // 301
fTOFCalc[ih].good_tdc_pos = kTRUE;
fGoodFlags[itrack][ip][iphit].goodTdcPos = kTRUE;
}
if (fTOFPInfo[ih].keep_neg) { //
fTOFCalc[ih].good_tdc_neg = kTRUE;
fGoodFlags[itrack][ip][iphit].goodTdcNeg = kTRUE;
}
// ** Calculate ave time for scin and error.
if (fTOFCalc[ih].good_tdc_pos) {
if (fTOFCalc[ih].good_tdc_neg) {
fTOFCalc[ih].scin_time =
(fTOFPInfo[ih].scin_pos_time + fTOFPInfo[ih].scin_neg_time) / 2.;
fTOFCalc[ih].scin_time_fp = (fTOFPInfo[ih].time_pos + fTOFPInfo[ih].time_neg) / 2.;
if (fTofUsingInvAdc) {
fTOFCalc[ih].scin_sigma =
TMath::Sqrt(fHodoPosSigma[fPIndex] * fHodoPosSigma[fPIndex] +
fHodoNegSigma[fPIndex] * fHodoNegSigma[fPIndex]) /
2.;
} else {
fTOFCalc[ih].scin_sigma =
TMath::Sqrt(fHodoSigmaPos[fPIndex] * fHodoSigmaPos[fPIndex] +
fHodoSigmaNeg[fPIndex] * fHodoSigmaNeg[fPIndex]) /
2.;
}
fTOFCalc[ih].good_scin_time = kTRUE;
fGoodFlags[itrack][ip][iphit].goodScinTime = kTRUE;
} else {
fTOFCalc[ih].scin_time = fTOFPInfo[ih].scin_pos_time;
fTOFCalc[ih].scin_time_fp = fTOFPInfo[ih].time_pos;
if (fTofUsingInvAdc) {
fTOFCalc[ih].scin_sigma = fHodoPosSigma[fPIndex];
} else {
fTOFCalc[ih].scin_sigma = fHodoSigmaPos[fPIndex];
}
fTOFCalc[ih].good_scin_time = kTRUE;
fGoodFlags[itrack][ip][iphit].goodScinTime = kTRUE;
}
} else {
if (fTOFCalc[ih].good_tdc_neg) {
fTOFCalc[ih].scin_time = fTOFPInfo[ih].scin_neg_time;
fTOFCalc[ih].scin_time_fp = fTOFPInfo[ih].time_neg;
if (fTofUsingInvAdc) {
fTOFCalc[ih].scin_sigma = fHodoNegSigma[fPIndex];
} else {
fTOFCalc[ih].scin_sigma = fHodoSigmaNeg[fPIndex];
}
fTOFCalc[ih].good_scin_time = kTRUE;
fGoodFlags[itrack][ip][iphit].goodScinTime = kTRUE;
}
} // In h_tof.f this includes the following if condition for time at focal plane
// // because it is written in FORTRAN code
// c Get time at focal plane
if (fTOFCalc[ih].good_scin_time) {
// scin_time_fp doesn't need to be an array
// Is this any different than the average of time_pos and time_neg?
// Double_t scin_time_fp = ( fTOFPInfo[ih].time_pos +
// fTOFPInfo[ih].time_neg ) / 2.;
Double_t scin_time_fp = fTOFCalc[ih].scin_time_fp;
sumFPTime = sumFPTime + scin_time_fp;
nFPTime++;
fSumPlaneTime[ip] = fSumPlaneTime[ip] + scin_time_fp;
fNPlaneTime[ip]++;
fNScinHit[itrack]++;
if ((fTOFCalc[ih].good_tdc_pos) && (fTOFCalc[ih].good_tdc_neg)) {
nPmtHit[itrack] = nPmtHit[itrack] + 2;
} else {
nPmtHit[itrack] = nPmtHit[itrack] + 1;
}
fdEdX[itrack].push_back(0.0);
assert(fNScinHit[itrack] > 0 && (size_t)fNScinHit[itrack] < fdEdX[itrack].size() + 1);
// --------------------------------------------------------------------------------------------
if (fTOFCalc[ih].good_tdc_pos) {
if (fTOFCalc[ih].good_tdc_neg) {
fdEdX[itrack][fNScinHit[itrack] - 1] =
TMath::Sqrt(TMath::Max(0., hit->GetPosADC() * hit->GetNegADC()));
} else {
fdEdX[itrack][fNScinHit[itrack] - 1] = TMath::Max(0., hit->GetPosADC());
}
} else {
if (fTOFCalc[ih].good_tdc_neg) {
fdEdX[itrack][fNScinHit[itrack] - 1] = TMath::Max(0., hit->GetNegADC());
} else {
fdEdX[itrack][fNScinHit[itrack] - 1] = 0.0;
}
}
// --------------------------------------------------------------------------------------------
} // time at focal plane condition
} // on track condition
// ** See if there are any good time measurements in the plane.
if (fTOFCalc[ih].good_scin_time) {
fGoodPlaneTime[ip] = kTRUE;
fTOFCalc[ih].dedx = fdEdX[itrack][fNScinHit[itrack] - 1];
} else {
fTOFCalc[ih].dedx = 0.0;
}
} // Second loop over hits of a scintillator plane ends here

Sylvester Joosten
committed
theTrack->SetGoodPlane3(fGoodPlaneTime[2] ? 1 : 0);
if (fNumPlanesBetaCalc == 4)
theTrack->SetGoodPlane4(fGoodPlaneTime[3] ? 1 : 0);
//
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
// * * Fit beta if there are enough time measurements (one upper, one lower)
// From h_tof_fit

Sylvester Joosten
committed
if (((fGoodPlaneTime[0]) || (fGoodPlaneTime[1])) &&
((fGoodPlaneTime[2]) || (fGoodPlaneTime[3]))) {

Sylvester Joosten
committed
Double_t sumW = 0.;
Double_t sumT = 0.;
Double_t sumZ = 0.;
Double_t sumZZ = 0.;
Double_t sumTZ = 0.;

Sylvester Joosten
committed
for (Int_t ih = 0; ih < nhits; ih++) {
Int_t ip = fTOFPInfo[ih].planeIndex;

Sylvester Joosten
committed
if (fTOFCalc[ih].good_scin_time) {

Sylvester Joosten
committed
Double_t scinWeight = 1 / (fTOFCalc[ih].scin_sigma * fTOFCalc[ih].scin_sigma);
Double_t zPosition =
(fPlanes[ip]->GetZpos() + (fTOFCalc[ih].hit_paddle % 2) * fPlanes[ip]->GetDzpos());

Sylvester Joosten
committed
sumW += scinWeight;
sumT += scinWeight * fTOFCalc[ih].scin_time;
sumZ += scinWeight * zPosition;
sumZZ += scinWeight * (zPosition * zPosition);
sumTZ += scinWeight * zPosition * fTOFCalc[ih].scin_time;

Sylvester Joosten
committed
} // condition of good scin time
} // loop over hits

Sylvester Joosten
committed
Double_t tmp = sumW * sumZZ - sumZ * sumZ;
Double_t t0 = (sumT * sumZZ - sumZ * sumTZ) / tmp;
Double_t tmpDenom = sumW * sumTZ - sumZ * sumT;

Sylvester Joosten
committed
if (TMath::Abs(tmpDenom) > (1 / 10000000000.0)) {

Sylvester Joosten
committed
beta = tmp / tmpDenom;
betaChiSq = 0.;

Sylvester Joosten
committed
for (Int_t ih = 0; ih < nhits; ih++) {
Int_t ip = fTOFPInfo[ih].planeIndex;

Sylvester Joosten
committed
if (fTOFCalc[ih].good_scin_time) {

Sylvester Joosten
committed
Double_t zPosition = (fPlanes[ip]->GetZpos() +
(fTOFCalc[ih].hit_paddle % 2) * fPlanes[ip]->GetDzpos());
Double_t timeDif = (fTOFCalc[ih].scin_time - t0);
betaChiSq += ((zPosition / beta - timeDif) * (zPosition / beta - timeDif)) /
(fTOFCalc[ih].scin_sigma * fTOFCalc[ih].scin_sigma);

Sylvester Joosten
committed
} // condition for good scin time
} // loop over hits

Sylvester Joosten
committed
Double_t pathNorm = TMath::Sqrt(1. + theTrack->GetTheta() * theTrack->GetTheta() +
theTrack->GetPhi() * theTrack->GetPhi());
// Take angle into account
beta = beta / pathNorm;
beta = beta / 29.979; // velocity / c

Sylvester Joosten
committed
} // condition for fTmpDenom
else {
beta = 0.;
betaChiSq = -2.;
} // else condition for fTmpDenom

Sylvester Joosten
committed
beta = 0.;
betaChiSq = -1;

Sylvester Joosten
committed
if (nFPTime != 0) {
timeAtFP[itrack] = (sumFPTime / nFPTime);
//
// ---------------------------------------------------------------------------

Sylvester Joosten
committed
Double_t FPTimeSum = 0.0;
Int_t nFPTimeSum = 0;
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
if (fNPlaneTime[ip] != 0) {
fFPTime[ip] = (fSumPlaneTime[ip] / fNPlaneTime[ip]);
FPTimeSum += fSumPlaneTime[ip];
nFPTimeSum += fNPlaneTime[ip];
} else {
fFPTime[ip] = 1000. * (ip + 1);
}

Sylvester Joosten
committed
Double_t fptime = -1000;
if (nFPTimeSum > 0)
fptime = FPTimeSum / nFPTimeSum;
fFPTimeAll = fptime;
Double_t dedx = 0.0;
for (UInt_t ih = 0; ih < fTOFCalc.size(); ih++) {
if (fTOFCalc[ih].good_scin_time) {
dedx = fTOFCalc[ih].dedx;
break;
}
}
theTrack->SetDedx(dedx);
theTrack->SetFPTime(fptime);
theTrack->SetBeta(beta);

Sylvester Joosten
committed
theTrack->SetBetaChi2(betaChiSq);

Sylvester Joosten
committed
theTrack->SetFPTime(timeAtFP[itrack]);
} // Main loop over tracks ends here.
} // If condition for at least one track

Sylvester Joosten
committed
// OriginalTrackEffTest();

Sylvester Joosten
committed
void THcHodoscope::TrackEffTest(void) {
// assume X planes are 0,2 and Y planes are 1,3

Sylvester Joosten
committed
std::array<int, 4> PadLow = {fxLoScin[0], fyLoScin[0], fxLoScin[1], fyLoScin[1]};
std::array<int, 4> PadHigh = {fxHiScin[0], fyHiScin[0], fxHiScin[1], fyHiScin[1]};

Sylvester Joosten
committed
std::array<double, 4> PadPosLo;
std::array<double, 4> PadPosHi;
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
Double_t lowtemp = fPlanes[ip]->GetPosCenter(PadLow[ip] - 1) + fPlanes[ip]->GetPosOffset();
Double_t hitemp = fPlanes[ip]->GetPosCenter(PadHigh[ip] - 1) + fPlanes[ip]->GetPosOffset();

Sylvester Joosten
committed
PadPosLo[ip] = lowtemp;
PadPosHi[ip] = hitemp;

Sylvester Joosten
committed
PadPosLo[ip] = hitemp;
PadPosHi[ip] = lowtemp;
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
1576
1577
1578
//{
// std::vector<int> test_vector = {1,2,5,6,7, 9,10, 20};
// auto test_res = hcana::find_discontinuity(test_vector);
// std::cout << " find_discontinuity test: " << *test_res << "\n";
// std::cout << " count_discontinuities test: " << hcana::count_discontinuities(test_vector)
// << "\n";
// auto cont_ranges = hcana::get_discontinuities(test_vector);
// int ir = 0;
// std::cout << "ranges: \n";
// for(auto r : cont_ranges) {
// for(auto v = r.first; v< r.second; v++) {
// std::cout << *v << " ";
// }
// std::cout << "\n";
// }
//}
//{
// std::vector<int> test_vector = {0,2,4,8,10,12,13, 16,18,20,23};
// auto test_res = hcana::find_discontinuity(
// test_vector, [](const int& x1, const int& x2) { return x1 + 2 == x2; });
// std::cout << " find_discontinuity test2: " << *test_res << "\n";
// std::cout << " count_discontinuities test2: "
// << hcana::count_discontinuities(
// test_vector, [](const int& x1, const int& x2) { return x1 + 2 == x2; })
// << "\n";
//}
using HitIterator = std::vector<THcHodoHit*>::iterator;
using HitRangeVector = typename std::vector<std::pair<HitIterator, HitIterator>>;
using ClusterPositions = typename std::vector<double>;

Sylvester Joosten
committed
std::map<int, std::vector<THcHodoHit*>> hit_vecs;
std::vector<HitRangeVector> hit_clusters;
std::vector<ClusterPositions> pos_clusters;
std::vector<std::vector<int>> good_clusters;
std::array<int, 4> n_good_clusters;
// Loop over each plane

Sylvester Joosten
committed
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
// Get the hit array
TClonesArray* hodoHits = fPlanes[ip]->GetHits();
// Create vector for good hits.
hit_vecs[ip] = std::vector<THcHodoHit*>();

Sylvester Joosten
committed
for (Int_t iphit = 0; iphit < fPlanes[ip]->GetNScinHits(); iphit++) {
THcHodoHit* hit = (THcHodoHit*)hodoHits->At(iphit);
// keep only those with "two good times"

Sylvester Joosten
committed
if (hit->GetTwoGoodTimes()) {
hit_vecs[ip].push_back(hit);

Sylvester Joosten
committed
// Cluster (group adjacent) hits
if (hit_vecs[ip].size() > 0) {
auto& hv = hit_vecs[ip];
hcana::sort(hv, [](const THcHodoHit* h1, const THcHodoHit* h2) {

Sylvester Joosten
committed
return h1->GetPaddleNumber() < h2->GetPaddleNumber();
});
// get the clusters for this layer
hit_clusters.push_back(
hcana::get_discontinuities(hv, [](const THcHodoHit* h1, const THcHodoHit* h2) {
return h1->GetPaddleNumber() + 1 == h2->GetPaddleNumber();
}));
} else {

Sylvester Joosten
committed
hit_clusters.push_back(HitRangeVector{});

Sylvester Joosten
committed
fPlanes[ip]->SetNumberClusters(hit_clusters.back().size());
// Calculate each cluster's mean position (in one coordinate)
ClusterPositions clust_positions;
std::vector<int> clust_bound;

Sylvester Joosten
committed
int ic = 0;
for (auto& r : hit_clusters.back()) {
double a_pos = 0.0;

Sylvester Joosten
committed
for (auto chit = r.first; chit < r.second; chit++) {
a_pos +=
fPlanes[ip]->GetPosCenter((*chit)->GetPaddleNumber() - 1) + fPlanes[ip]->GetPosOffset();
n_hit++;
}
// take the average position
double avg_pos = a_pos / double(n_hit);

Sylvester Joosten
committed
fPlanes[ip]->SetCluster(ic, avg_pos);
fPlanes[ip]->SetClusterSize(ic, n_hit);
// Calculate if it is the bound by the upper and lower limits where we expect
// full tracks to be reconstructed.

Sylvester Joosten
committed
int is_bound = int((avg_pos >= PadPosLo[ip]) && (avg_pos <= PadPosHi[ip]));
clust_bound.push_back(is_bound);

Sylvester Joosten
committed
if (is_bound) {
// we are only interested in clusters positioned inthe "sweet spot"
clust_positions.push_back(avg_pos);
n_good_clusters[ip]++;
if (n_hit > 10) {
_det_logger->warn("cluster in hodoscope track efficiency has {} hits", n_hit);

Sylvester Joosten
committed
ic += 1;
pos_clusters.push_back(clust_positions);
good_clusters.push_back(clust_bound);
//_logger->info("{} clusters in Hod plane {}", hit_clusters.back().size(), ip );
bool has_both_x_clusters = (n_good_clusters[0] > 0) && (n_good_clusters[2] > 0);
bool has_both_y_clusters = (n_good_clusters[1] > 0) && (n_good_clusters[3] > 0);
bool at_least_one_good_x_clusters = (n_good_clusters[0] > 0) || (n_good_clusters[2] > 0);
bool at_least_one_good_y_clusters = (n_good_clusters[1] > 0) || (n_good_clusters[3] > 0);

Sylvester Joosten
committed
bool good_x_match = false;
bool good_y_match = false;
// Superficial matching. Find out if clusters in front and back could possibly belong to the same
// track. This done by checking to see if the position differences are less than 10 cm;
// ie, |x1-x2|<10 or |y1-y2| <10
// TODO do actual matching with tracks elsewhere

Sylvester Joosten
committed
if (has_both_x_clusters) {
int ic0 = 0;
// factor to project X1 to X2 z position
const double x1_projection_factor =
(1 + fRatio_xpfp_to_xfp * (fPlanes[2]->GetZpos() - fPlanes[0]->GetZpos()));
for (auto x1_pos : pos_clusters[0]) {
int ic2 = 0;
// project X1 to X2 Z position
Double_t x1_proj = x1_pos * x1_projection_factor;
for (auto x2_pos : pos_clusters[2]) {
if (std::abs(x1_proj - x2_pos) < trackeff_scint_xdiff_max) {
good_x_match = true;

Sylvester Joosten
committed
fPlanes[0]->SetClusterUsedFlag(ic0, 1.);
fPlanes[2]->SetClusterUsedFlag(ic2, 1.);

Sylvester Joosten
committed
ic2 += 1;

Sylvester Joosten
committed
ic0 += 1;

Sylvester Joosten
committed
if (has_both_y_clusters) {
int ic1 = 0;
for (auto y1_pos : pos_clusters[1]) {
int ic3 = 0;
for (auto y2_pos : pos_clusters[3]) {
if (std::abs(y1_pos - y2_pos) < trackeff_scint_ydiff_max) {
good_y_match = true;

Sylvester Joosten
committed
fPlanes[1]->SetClusterUsedFlag(ic1, 1.);
fPlanes[3]->SetClusterUsedFlag(ic3, 1.);

Sylvester Joosten
committed
ic3 += 1;

Sylvester Joosten
committed
ic1 += 1;
fGoodScinHits = 0;

Sylvester Joosten
committed
// (2+2) 4 good hits
if (fTrackEffTestNScinPlanes == 4 || fTrackEffTestNScinPlanes == 3) {
if (good_y_match && good_x_match) {
fGoodScinHits = 1;

Sylvester Joosten
committed
// (2+1) 3 good hits
if (at_least_one_good_y_clusters && good_x_match) {
fGoodScinHits = 1;

Sylvester Joosten
committed
// (1+2) 3 good hits
if (at_least_one_good_x_clusters && good_y_match) {
fGoodScinHits = 1;

Sylvester Joosten
committed
void THcHodoscope::OriginalTrackEffTest(void) {
/**
Translation of h_track_tests.f file for tracking efficiency
*/
//************************now look at some hodoscope tests
// *second, we move the scintillators. here we use scintillator cuts to see
// *if a track should have been found.

Sylvester Joosten
committed
cout << " enter track eff" << fNumPlanesBetaCalc << endl;
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
cout << " loop over planes " << ip + 1 << endl;
TClonesArray* hodoHits = fPlanes[ip]->GetHits();
// TClonesArray* scinPosTDC = fPlanes[ip]->GetPosTDC();
// TClonesArray* scinNegTDC = fPlanes[ip]->GetNegTDC();
fNScinHits[ip] = fPlanes[ip]->GetNScinHits();

Sylvester Joosten
committed
cout << " hits = " << fNScinHits[ip] << endl;
for (Int_t iphit = 0; iphit < fNScinHits[ip]; iphit++) {
Int_t paddle = ((THcHodoHit*)hodoHits->At(iphit))->GetPaddleNumber() - 1;
fScinHitPaddle[ip][paddle] = 1;

Sylvester Joosten
committed
cout << " hit = " << iphit + 1 << " " << paddle + 1 << endl;
}
}
// *next, look for clusters of hits in a scin plane. a cluster is a group of
// *adjacent scintillator hits separated by a non-firing scintillator.
// *Wwe count the number of three adjacent scintillators too. (A signle track
// *shouldn't fire three adjacent scintillators.

Sylvester Joosten
committed
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
// Planes ip = 0 = 1X
// Planes ip = 2 = 2X
fNClust.push_back(0);
fThreeScin.push_back(0);
}
// *look for clusters in x planes... (16 scins) !this assume both x planes have same
// *number of scintillators.
cout << " looking for cluster in x planes" << endl;

Sylvester Joosten
committed
for (Int_t ip = 0; ip < 3; ip += 2) {

Sylvester Joosten
committed
if (fScinHitPaddle[ip][0] == 1)
icount++;
cout << "plane =" << ip << "check if paddle 1 hit " << icount << endl;

Sylvester Joosten
committed
for (Int_t ipaddle = 0; ipaddle < (Int_t)fNPaddle[ip] - 1; ipaddle++) {
// !look for number of clusters of 1 or more hits

Sylvester Joosten
committed
if ((fScinHitPaddle[ip][ipaddle] == 0) && (fScinHitPaddle[ip][ipaddle + 1] == 1))
icount++;
cout << " paddle = " << ipaddle + 1 << " " << icount << endl;

Sylvester Joosten
committed
cout << "Two cluster in plane = " << ip + 1 << " " << icount << endl;

Sylvester Joosten
committed
icount = 0;

Sylvester Joosten
committed
for (Int_t ipaddle = 0; ipaddle < (Int_t)fNPaddle[ip] - 2; ipaddle++) {
// !look for three or more adjacent hits

Sylvester Joosten
committed
if ((fScinHitPaddle[ip][ipaddle] == 1) && (fScinHitPaddle[ip][ipaddle + 1] == 1) &&
(fScinHitPaddle[ip][ipaddle + 2] == 1))
icount++;

Sylvester Joosten
committed
cout << "Three clusters in plane = " << ip + 1 << " " << icount << endl;

Sylvester Joosten
committed
if (icount > 0)
fThreeScin[ip] = 1;
} // Loop over X plane
// *look for clusters in y planes... (10 scins) !this assume both y planes have same
cout << " looking for cluster in y planes" << endl;

Sylvester Joosten
committed
for (Int_t ip = 1; ip < fNumPlanesBetaCalc; ip += 2) {
// Planes ip = 1 = 1Y
// Planes ip = 3 = 2Y
icount = 0;

Sylvester Joosten
committed
if (fScinHitPaddle[ip][0] == 1)
icount++;
cout << "plane =" << ip << "check if paddle 1 hit " << icount << endl;

Sylvester Joosten
committed
for (Int_t ipaddle = 0; ipaddle < (Int_t)fNPaddle[ip] - 1; ipaddle++) {
// !look for number of clusters of 1 or more hits

Sylvester Joosten
committed
if ((fScinHitPaddle[ip][ipaddle] == 0) && (fScinHitPaddle[ip][ipaddle + 1] == 1))
icount++;
cout << " paddle = " << ipaddle + 1 << " " << icount << endl;

Sylvester Joosten
committed
cout << "Two cluster in plane = " << ip + 1 << " " << icount << endl;

Sylvester Joosten
committed
icount = 0;

Sylvester Joosten
committed
for (Int_t ipaddle = 0; ipaddle < (Int_t)fNPaddle[ip] - 2; ipaddle++) {
// !look for three or more adjacent hits

Sylvester Joosten
committed
if ((fScinHitPaddle[ip][ipaddle] == 1) && (fScinHitPaddle[ip][ipaddle + 1] == 1) &&
(fScinHitPaddle[ip][ipaddle + 2] == 1))
icount++;
} // Second loop over Y paddles

Sylvester Joosten
committed
cout << "Three clusters in plane = " << ip + 1 << " " << icount << endl;

Sylvester Joosten
committed
if (icount > 0)

Sylvester Joosten
committed
} // Loop over Y planes
// *next we mask out the edge scintillators, and look at triggers that happened
// *at the center of the acceptance. To change which scins are in the mask
// *change the values of h*loscin and h*hiscin in htracking.param
// fGoodScinHits = 0;

Sylvester Joosten
committed
for (Int_t ifidx = fxLoScin[0]; ifidx < (Int_t)fxHiScin[0]; ifidx++) {
fGoodScinHitsX.push_back(0);
}

Sylvester Joosten
committed
fHitSweet1X = 0;
fHitSweet2X = 0;
fHitSweet1Y = 0;
fHitSweet2Y = 0;
// *first x plane. first see if there are hits inside the scin region

Sylvester Joosten
committed
for (Int_t ifidx = fxLoScin[0] - 1; ifidx < fxHiScin[0]; ifidx++) {
if (fScinHitPaddle[0][ifidx] == 1) {
fHitSweet1X = 1;
fSweet1XScin = ifidx + 1;
}
}
// * next make sure nothing fired outside the good region

Sylvester Joosten
committed
for (Int_t ifidx = 0; ifidx < fxLoScin[0] - 1; ifidx++) {
if (fScinHitPaddle[0][ifidx] == 1) {
fHitSweet1X = -1;
}

Sylvester Joosten
committed
for (Int_t ifidx = fxHiScin[0]; ifidx < (Int_t)fNPaddle[0]; ifidx++) {
if (fScinHitPaddle[0][ifidx] == 1) {
fHitSweet1X = -1;
}
}
// *second x plane. first see if there are hits inside the scin region

Sylvester Joosten
committed
for (Int_t ifidx = fxLoScin[1] - 1; ifidx < fxHiScin[1]; ifidx++) {
if (fScinHitPaddle[2][ifidx] == 1) {
fHitSweet2X = 1;
fSweet2XScin = ifidx + 1;
}
}
// * next make sure nothing fired outside the good region

Sylvester Joosten
committed
for (Int_t ifidx = 0; ifidx < fxLoScin[1] - 1; ifidx++) {
if (fScinHitPaddle[2][ifidx] == 1) {
fHitSweet2X = -1;
}

Sylvester Joosten
committed
for (Int_t ifidx = fxHiScin[1]; ifidx < (Int_t)fNPaddle[2]; ifidx++) {
if (fScinHitPaddle[2][ifidx] == 1) {
fHitSweet2X = -1;
}
}
// *first y plane. first see if there are hits inside the scin region

Sylvester Joosten
committed
for (Int_t ifidx = fyLoScin[0] - 1; ifidx < fyHiScin[0]; ifidx++) {
if (fScinHitPaddle[1][ifidx] == 1) {
fHitSweet1Y = 1;
fSweet1YScin = ifidx + 1;
}
}
// * next make sure nothing fired outside the good region

Sylvester Joosten
committed
for (Int_t ifidx = 0; ifidx < fyLoScin[0] - 1; ifidx++) {
if (fScinHitPaddle[1][ifidx] == 1) {
fHitSweet1Y = -1;
}

Sylvester Joosten
committed
for (Int_t ifidx = fyHiScin[0]; ifidx < (Int_t)fNPaddle[1]; ifidx++) {
if (fScinHitPaddle[1][ifidx] == 1) {
fHitSweet1Y = -1;
}
}
// *second y plane. first see if there are hits inside the scin region

Sylvester Joosten
committed
for (Int_t ifidx = fyLoScin[1] - 1; ifidx < fyHiScin[1]; ifidx++) {
if (fScinHitPaddle[3][ifidx] == 1) {
fHitSweet2Y = 1;
fSweet2YScin = ifidx + 1;
}
}
// * next make sure nothing fired outside the good region

Sylvester Joosten
committed
for (Int_t ifidx = 0; ifidx < fyLoScin[1] - 1; ifidx++) {
if (fScinHitPaddle[3][ifidx] == 1) {
fHitSweet2Y = -1;
}

Sylvester Joosten
committed
for (Int_t ifidx = fyHiScin[1]; ifidx < (Int_t)fNPaddle[3]; ifidx++) {
if (fScinHitPaddle[3][ifidx] == 1) {
fHitSweet2Y = -1;
}
}
fTestSum = fHitSweet1X + fHitSweet2X + fHitSweet1Y + fHitSweet2Y;
// * now define a 3/4 or 4/4 trigger of only good scintillators the value

Sylvester Joosten
committed
if (fTestSum >= fTrackEffTestNScinPlanes) {

Sylvester Joosten
committed
for (Int_t ifidx = fxLoScin[0]; ifidx < fxHiScin[0]; ifidx++) {
if (fSweet1XScin == ifidx)
fGoodScinHitsX[ifidx] = 1;
}
}
// * require front/back hodoscopes be close to each other

Sylvester Joosten
committed
if ((fGoodScinHits == 1) && (fTrackEffTestNScinPlanes == 4)) {
if (TMath::Abs(fSweet1XScin - fSweet2XScin) > 3)

Sylvester Joosten
committed
if (TMath::Abs(fSweet1YScin - fSweet2YScin) > 2)

Sylvester Joosten
committed
//
Gabriel Niculescu
committed
//_____________________________________________________________________________

Sylvester Joosten
committed
Int_t THcHodoscope::FineProcess(TClonesArray& tracks) {
Int_t Ntracks = tracks.GetLast() + 1; // Number of reconstructed tracks
Double_t hitPos;
Double_t hitDistance;

Sylvester Joosten
committed
Int_t ih = 0;
for (Int_t itrk = 0; itrk < Ntracks; itrk++) {
THaTrack* theTrack = static_cast<THaTrack*>(tracks[itrk]);
if (theTrack->GetIndex() == 0) {
fBeta = theTrack->GetBeta();
fFPTimeAll = theTrack->GetFPTime();
_basic_data.fBeta = fBeta;
_basic_data.fFPTimeAll = fFPTimeAll;

Sylvester Joosten
committed
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
Double_t shower_track_enorm = theTrack->GetEnergy() / theTrack->GetP();
for (Int_t ip = 0; ip < fNumPlanesBetaCalc; ip++) {
Double_t pl_xypos = 0;
Double_t pl_zpos = 0;
Int_t num_good_pad = 0;
TClonesArray* hodoHits = fPlanes[ip]->GetHits();
for (Int_t iphit = 0; iphit < fPlanes[ip]->GetNScinHits(); iphit++) {
THcHodoHit* hit = fTOFPInfo[ih].hit;
if (fTOFCalc[ih].good_tdc_pos && fTOFCalc[ih].good_tdc_neg) {
Bool_t sh_pid = (shower_track_enorm > fTOFCalib_shtrk_lo &&
shower_track_enorm < fTOFCalib_shtrk_hi);
Bool_t beta_pid = (fBeta > fTOFCalib_beta_lo && fBeta < fTOFCalib_beta_hi);
// cer_pid is true if there is no Cherenkov detector
Bool_t cer_pid = (fCherenkov ? (fCherenkov->GetCerNPE() > fTOFCalib_cer_lo) : kTRUE);
if (fDumpTOF && Ntracks == 1 && fGoodEventTOFCalib && sh_pid && beta_pid && cer_pid) {
fDumpOut << fixed << setprecision(2);
fDumpOut << showpoint << " 1" << setw(3) << ip + 1 << setw(3)
<< hit->GetPaddleNumber() << setw(10) << hit->GetPosTDC() * fScinTdcToTime
<< setw(10) << fTOFPInfo[ih].pathp << setw(10) << fTOFPInfo[ih].zcor
<< setw(10) << fTOFPInfo[ih].time_pos << setw(10) << hit->GetPosADC()
<< endl;
fDumpOut << showpoint << " 2" << setw(3) << ip + 1 << setw(3)
<< hit->GetPaddleNumber() << setw(10) << hit->GetNegTDC() * fScinTdcToTime
<< setw(10) << fTOFPInfo[ih].pathn << setw(10) << fTOFPInfo[ih].zcor
<< setw(10) << fTOFPInfo[ih].time_neg << setw(10) << hit->GetNegADC()
<< endl;
}
Int_t padind = ((THcHodoHit*)hodoHits->At(iphit))->GetPaddleNumber() - 1;
pl_xypos += fPlanes[ip]->GetPosCenter(padind) + fPlanes[ip]->GetPosOffset();
pl_zpos += fPlanes[ip]->GetZpos() + (padind % 2) * fPlanes[ip]->GetDzpos();
num_good_pad++;
}
ih++;
// cout << ip << " " << iphit << " " << fGoodFlags[itrk][ip][iphit].goodScinTime <<
//" " << fGoodFlags[itrk][ip][iphit].goodTdcPos << " " <<
// fGoodFlags[itrk][ip][iphit].goodTdcNeg << endl;
}