156 ATH_MSG_WARNING(
"Didn't receive " << FPGAHits.
key() <<
" on first event; assuming no input events.");
158 SmartIF<IEventProcessor> appMgr{service(
"ApplicationMgr")};
160 ATH_MSG_ERROR(
"Failed to retrieve ApplicationMgr as IEventProcessor");
161 return StatusCode::FAILURE;
163 return appMgr->stopRun();
174 auto* FPGAHits_1st_cdv = FPGAHits_1st.
ptr();
175 auto* FPGAHits_2nd_cdv = FPGAHits_2nd.
ptr();
177 ATH_CHECK( FPGARoads_1st.
record (std::make_unique<FPGATrackSimRoadCollection>()));
180 ATH_CHECK(FPGATracks_1stHandle.
record (std::make_unique<FPGATrackSimTrackCollection>()));
183 ATH_CHECK( FPGAHitsFiltered_1st.
record (std::make_unique<FPGATrackSimHitCollection>()));
186 ATH_CHECK( FPGASpacePoints.
record (std::make_unique<FPGATrackSimClusterCollection>()));
189 if (!
m_evtSel->getSelectedEvent()) {
193 std::vector<FPGATrackSimRoad> roads_1st;
194 std::vector<FPGATrackSimTrack> tracks_1st;
195 auto dataFlowInfo = std::make_unique<FPGATrackSimDataFlowInfo>();
199 return StatusCode::SUCCESS;
210 if (!FPGAEventInfo.
isValid()) {
211 ATH_MSG_ERROR(
"Could not find FPGA Event Info with key " << FPGAEventInfo.
key());
212 return StatusCode::FAILURE;
220 std::vector<std::shared_ptr<const FPGATrackSimHit>> phits_output, phits_all, phits_1st, phits_2nd;
221 std::vector<const FPGATrackSimHit*> phits_strips;
224 std::optional<Athena::Chrono> chronoSplitHits;
225 if constexpr (
enableBenchmark) chronoSplitHits.emplace(
"1st Stage: Split hits to 1st and 2nd stage",
m_chrono.get());
227 phits_1st.reserve(FPGAHits->size());
228 phits_2nd.reserve(FPGAHits->size());
234 auto sharedHit = std::shared_ptr<const FPGATrackSimHit>{hit, noDelete};
235 phits_all.push_back(sharedHit);
237 phits_1st.push_back(sharedHit);
238 phits_2nd.push_back(std::move(sharedHit));
239 if(hit->isStrip()) phits_strips.push_back(hit);
248 for (
auto& hit : phits_1st) {
249 FPGAHits_1st_cdv->push_back(hit.get());
256 std::vector<FPGATrackSimCluster> spacepoints;
257 std::optional<Athena::Chrono> chronoSPFormation;
262 FPGASpacePoints->push_back(std::move(cluster));
266 for (
const auto& cluster : *FPGASpacePoints) {
268 for (
const auto& hit : cluster.getHitList()) {
281 for (
auto& hit : phits_2nd) {
282 FPGAHits_2nd_cdv->push_back(hit.get());
289 ATH_MSG_DEBUG(
"1st stage hits: " << phits_1st.size() <<
" 2nd stage hits: " << phits_2nd.size() );
290 if (phits_1st.empty()) {
293 std::vector<FPGATrackSimRoad> roads_1st;
294 std::vector<FPGATrackSimTrack> tracks_1st;
295 auto dataFlowInfo = std::make_unique<FPGATrackSimDataFlowInfo>();
298 return StatusCode::SUCCESS;
303 if (!FPGATruthTracks.
isValid()) {
304 ATH_MSG_ERROR(
"Could not find FPGA Truth Track Collection with key " << FPGATruthTracks.
key());
305 return StatusCode::FAILURE;
309 if (!FPGAOfflineTracks.
isValid()) {
310 ATH_MSG_ERROR(
"Could not find FPGA Offline Track Collection with key " << FPGAOfflineTracks.
key());
311 return StatusCode::FAILURE;
320 const std::vector<FPGATrackSimTruthTrack>& truthtracks = *FPGATruthTracks;
325 auto monitorRoads = [&](
auto& monitor,
const auto& roads) {
326 if (!monitor.empty()) {
327 monitor->fillRoad(roads, truthtracks, nLogicalLayers);
331 std::vector<FPGATrackSimRoad> roads_1st;
333 std::optional<Athena::Chrono> chronoGetRoads;
343 std::optional<Athena::Chrono> chronoRoadFiltering;
345 std::vector<FPGATrackSimRoad> postfilter_roads;
348 roads_1st = std::move(postfilter_roads);
357 std::optional<Athena::Chrono> chronoOverlapRemoval;
367 std::optional<Athena::Chrono> chronoRoadFiltering2;
369 std::vector<FPGATrackSimRoad> postfilter2_roads;
372 roads_1st = std::move(postfilter2_roads);
383 auto monitorTracks = [&](
auto& monitor,
const auto& tracks) {
384 if (monitor.empty())
return;
386 std::vector<const FPGATrackSimTrack*> track_ptrs;
387 track_ptrs.reserve(tracks.size());
389 if constexpr (std::is_pointer_v<
typename std::decay_t<
decltype(tracks)>::value_type>) {
391 track_ptrs.insert(track_ptrs.end(), tracks.begin(), tracks.end());
394 std::transform(tracks.begin(), tracks.end(), std::back_inserter(track_ptrs), [](
const auto& t) { return &t; });
397 monitor->fillTrack(track_ptrs, truthtracks, 1.e15);
400 std::vector<FPGATrackSimTrack> tracks_1st;
403 std::optional<Athena::Chrono> chronoGettingTracks;
417 for (
const auto& road : roads_1st) {
418 std::vector<FPGATrackSimTrack> tracksForCurrentRoad;
421 std::vector<FPGATrackSimRoad> roadVec = {road};
425 if (!tracksForCurrentRoad.empty()) {
426 auto bestTrackIter = std::min_element(
427 tracksForCurrentRoad.begin(), tracksForCurrentRoad.end(),
429 return a.getChi2ndof() < b.getChi2ndof();
432 if (bestTrackIter != tracksForCurrentRoad.end() && bestTrackIter->getChi2ndof() < 1.e15) {
433 tracks_1st.push_back(*bestTrackIter);
442 ATH_MSG_DEBUG(
"No tracking. Just running dummy road2track algorith");
444 for (
const auto& road : roads_1st) {
445 std::vector<std::shared_ptr<const FPGATrackSimHit>> track_hits;
446 for (
unsigned layer = 0; layer < road.getNLayers(); ++layer) {
447 track_hits.insert(track_hits.end(), road.getHitPtrs(layer).begin(), road.getHitPtrs(layer).end());
452 for (
size_t ihit = 0; ihit < track_hits.size(); ++ihit) {
455 tracks_1st.push_back(std::move(track_cand));
465 for (
auto &track : tracks_1st)
466 track.calculateTruth();
476 std::optional<Athena::Chrono> chronoSetTruthParams;
477 if constexpr (
enableBenchmark) chronoSetTruthParams.emplace(
"1st Stage: Set Track Parameters to Truth",
m_chrono.get());
483 track.setQOverPt(truthtracks.front().getQOverPt());
485 track.setD0(truthtracks.front().getD0());
487 track.setPhi(truthtracks.front().getPhi());
489 track.setZ0(truthtracks.front().getZ0());
491 track.setEta(truthtracks.front().getEta());
500 for (
auto const& road : roads_1st) {
501 FPGARoads_1st->push_back(road);
505 if (truthtracks.size() > 0) {
509 if (tracks_1st.size() > 0) {
517 for (
auto itrack = tracks_1st.begin(); itrack != tracks_1st.end();) {
518 if (!
passesChi2Cut(*itrack)) itrack = tracks_1st.erase(itrack);
528 std::optional<Athena::Chrono> chronoOverlapRemoval2;
532 std::vector<const FPGATrackSimTrack*> tracks_1st_after_chi2;
533 std::vector<const FPGATrackSimTrack*> tracks_1st_after_overlap;
535 if (track.passedOR()) {
536 tracks_1st_after_overlap.push_back(&track);
547 if (truthtracks.size() > 0) {
549 unsigned npasschi2(0);
550 unsigned npasschi2OLR(0);
551 if (tracks_1st.size() > 0) {
552 for (
const auto& track : tracks_1st) {
554 if (track.passedOR()) {
569 FPGATracks_1stHandle->push_back(track);
574 std::vector<FPGATrackSimRoad> roadsLRT;
575 std::vector<FPGATrackSimTrack> tracksLRT;
578 std::vector<std::shared_ptr<const FPGATrackSimHit>> remainingHits;
581 ATH_MSG_DEBUG(
"Doing hit filtering based on prompt tracks.");
587 ATH_MSG_DEBUG(
"No hit filtering requested; using all hits for LRT.");
588 remainingHits = std::move(phits_1st);
596 auto dataFlowInfo = std::make_unique<FPGATrackSimDataFlowInfo>();
607 SmartIF<IEventProcessor> appMgr{service(
"ApplicationMgr")};
609 ATH_MSG_ERROR(
"Failed to retrieve ApplicationMgr as IEventProcessor");
610 return StatusCode::FAILURE;
621 return StatusCode::SUCCESS;