Table of contents:

Event selection service wagon

o2-analysis-event-selection-service wagon (see eventSelectionService.cxx) integrates several service modules:

  • Timestamp module, see details here
  • Event selection modules for bc-based and collision-based analyses, see details here
  • Luminosity module, see details here

NOTE: Previously (until July 2025), the timestamp and event selection tables were produced in dedicated timestamp.cxx and eventSelection.cxx tasks. These tasks are now deprecated and obsolete (and will be removed). All ongoing developments are being carried out in eventSelectionService.cxx task and corresponding timestampModule.h and EventSelectionModule.h modules. A new core wagon called eventSelectionService has been created in the Hyperloop to replace the existing timestamp and event selection wagons and provide users with exactly the same tables but with a much reduced memory overhead.

Timestamp module

The timestamp module (timestampModule.h) is needed to fill the Timestamps table with timestamps corresponding to bunch crossings in the BCs table. These timestamps (counted in ms from Jan 1, 1970, UTC) are often needed to retrieve CCDB objects (see Tutorial CCDB).

Since the Timestamps table has an entry per bunch crossing it can be joined with BCs table. The join is defined by o2::aod::BCsWithTimestamps (see list of defined joins and iterators).

Event selection module

The main purpose of the event selection framework in O2 is to provide tools to select triggered events and reject pileup, beam-gas and poor quality collisions. Event selection in O2 is based on the concept of derived tables created in dedicated tasks from available AOD contents. The event selection module (EventSelectionModule.h) produces two in-memory tables described in EventSelection.h:

  • EvSels table joinable with Collisions table. To be used in analyses based on loops over Collisions, i.e. majority of ALICE analyses.
  • BcSels table joinable with BCs table. To be used in analyses based on loops over BCs table such as muon arm UPCs, luminosity monitoring etc.

The structure of BcSels and EvSels tables is kept the same for Run 2 and Run 3. However, there are conceptual differences between Run 2 and Run 3 workflows:

  • Run 3 setup is significantly different from Run 2 setup, e. g. V0C detector is not available in Run 3 etc. Therefore Run-2 minimum bias trigger based on V0A & V0C is no longer available and is replaced with FT0A & FT0C requirement in Run 3. Many other selection criteria used in Run 2 are not applicable in Run 3 (e. g. tracklet-vs-cluser correlation cut).
  • While in Run 2 there is a unique matching between Collisions and BCs, it is not the case in Run 3. Time resolution for collisions (=primary vertices) is not precise enough to identify corresponding bunch crossing (=25 ns) without ambiguities. The collision time resolution depends on the number of contributed ITS-TPC tracks, availability of TOF-matched tracks and other factors. One of the main goals of event selection task in Run 3 is to find the original bunch crossing for each collision and check for relevant info in forward detectors (FIT, ZDC). Unambiguous association of collisions to bunch crossings might become very nontrivial in high rate environment.

BcSels and EvSels tables contain the following information:

  • alias: fired trigger alias bits, see Trigger aliases section
  • selection: offline event selection bits such as beam-beam and beam-gas decisions from forward detectors (FV0, FT0, FDD, ZDC) and various in-bunch and out-of-bunch pileup checks, see Event selection criteria section
  • rct: RCT flags, see Usage of RCT flags section

In addition EvSels table contains additional info:

  • sel7 and sel8 event selection decisions, i.e. logical combinations of various offline event selection criteria, see Event selection decisions section. For example, sel7 (main Run 2 selection) is based on beam-beam decisions in V0A and V0C with additional background, pileup and quality checks, while sel8 (main Run 3 selection) is based on TVX decision from FT0 detector and additional time-border cuts.
  • foundBC, foundFT0, foundFV0, foundFDD , foundZDC: indices to found bunch crossings and corresponding FT0, FV0, FDD and ZDC entries, see Found bunch crossings section.
  • trackOccupancyInTimeRange, ft0cOccupancyInTimeRange: raw occupancy estimators, see Occupancy estimation section for details.

BcSels and EvSels tables are produced by BcSelectionModule and EventSelectionModule, respectively, see O2Physics/Common/Tools/EventSelectionModule.h, the process functions are called from the task O2Physics/Common/TableProducer/eventSelectionService.cxx. There are separate process functions for Run 2 and Run 3 in both modules, and the eventSelectionService autodetects if the dataset being analysed is Run 2 or 3.

Basic usage in user tasks

In general, one has to follow the following steps:

  • add EventSelection.h header:

      #include "Common/DataModel/EventSelection.h""
    
  • join Collisions and EvSels tables and use corresponding iterator as an argument of the process function:

      void process(soa::Join<aod::Collisions, aod::EvSels>::iterator const& col, ...)
    
  • check if your trigger alias is fired if you run over Run1-2 data or triggered Run3 data:

      if (!col.alias_bit(kINT7)) {
        return;
      }
    

    Bypass this check if you analyse MC or continuous Run3 data.

  • apply further offline selection criteria:
    • for Run 2 data and MC:

        if (!col.sel7()) {
          return;
        }
      
    • for Run 3 data and MC:

        if (!col.sel8()) {
          return;
        }
      

    The sel8 event selection bit is a logical and between the kIsTriggerTVX bit (based on FT0A & FT0C requirement, with time-based constraints on the vertex position), and (since April 2024) kNoTimeFrameBorder and kNoITSROFrameBorder bits - see below.

  • run your tasks in stack with event-selection-service task:

      o2-analysis-event-selection-service -b | o2-analysis-user-task -b --aod-file AO2D.root -b
    

    A number of special configurables are available, see Configurables section.

    o2-analysis-zdc-converter and o2-analysis-collision-converter might be also necessary for old datasets to account for changes in the data model.

Trigger aliases

Direct selection on trigger class names in O2 is rather complicated. In contrast to Run 2 AODs, there is no way to get the list of fired classes in a string-like format. Instead one has to check bits corresponding to trigger class ids either in triggerMask column in BCs table or triggerMaskNext50 in Run2BCInfos table (for Run 2 if the trigger class id is larger than 50). This approach is complicated since trigger class ids for the same class vary from run to run.

To simplify trigger checks, we use trigger alias approach. Fired trigger classes are mapped to trigger alias bits in the alias array of BcSels and EvSels tables. Aliases have at least two advantages:

  • several classes based on similar logic can be grouped together into one alias (see kINT7 for example)
  • alias bits do not change from run to run in contrast to trigger class ids

The list of available trigger alises can be found in Common/CCDB/TriggerAliases.h. The mapping between trigger classes (and their indices) and trigger aliases is stored in CCDB run-by-run in dedicated TriggerAliases objects. Current mapping can be checked in upload_trigger_aliases.C macro for Run2 (as of November 2025):

  mAliases[kINT7] = "CINT7-B-NOPF-CENT,CINT7-B-NOPF-FAST,CV0L7-B-NOPF-CENT,CINT7-B-NOPF-CENTNOTRD,CINT7ZAC-B-NOPF-CENTNOPMD,CINT7-B-NOPF-ALLNOTRD,CINT7-I-NOPF-ALLNOTRD,CINT7-S-NOPF-ALLNOTRD,CMBAC-B-NOPF-ALL,CMBACS2-B-NOPF-ALLNOTRD,CMBACS2-B-NOPF-ALL";
  mAliases[kEMC7] = "CEMC7-B-NOPF-CENTNOPMD,CEMC7-B-NOPF-CENT,CEMC7-B-NOPF-CENTNOTRD,CEMC7-B-NOPF-ALLNOTRD,CEMC7-S-NOPF-ALLNOTRD";
  mAliases[kINT7inMUON] = "CINT7-B-NOPF-MUFAST";
  mAliases[kMuonSingleLowPt7] = "CMSL7-B-NOPF-MUFAST,CMSL7-B-NOPF-MUON,CMSL7-S-NOPF-MUON,CMSL7-SC-NOPF-MUON,CPBI1MSL-B-NOPF-MUON,CMUS7-B-NOPF-MUON";
  mAliases[kMuonUnlikeLowPt7] = "CMUL7-B-NOPF-MUFAST,CMUL7-B-NOPF-MUON,CMUL7-S-NOPF-MUON,CMUL7-S-NOPF-ALLNOTRD,CPBI1MUL-B-NOPF-MUON,CMUU7-B-NOPF-MUON,CMUU7-B-NOPF-ALLNOTRD";
  mAliases[kMuonLikeLowPt7] = "CMLL7-B-NOPF-MUFAST,CMLL7-B-NOPF-MUON,CMLL7-S-NOPF-MUON,CPBI1MLL-B-NOPF-MUON";
  mAliases[kMuonSingleHighPt7] = "CMSH7-B-NOPF-MUFAST,CMSH7-B-NOPF-MUON,CMSH7-S-NOPF-MUON,CMSH7-S-NOPF-ALLNOTRD,CPBI1MSH-B-NOPF-MUON,CMUSH7-B-NOPF-MUON";
  mAliases[kCUP8] = "CCUP8-B-NOPF-CENTNOTRD";
  mAliases[kCUP9] = "CCUP9-B-NOPF-CENTNOTRD";
  mAliases[kMUP10] = "CMUP10-B-NOPF-MUFAST";
  mAliases[kMUP11] = "CMUP11-B-NOPF-MUFAST";
  mAliases[kINT1] = "CINT1B-ABCE-NOPF-ALL,CINT1-B-NOPF-ALLNOTRD,CINT1-B-NOPF-ALLNOTRD";
  mAliases[kUnbiased] = "CBEAMB-ABCE-NOPF-ALL,CBEAMB-B-NOPF-ALLNOTRD,CTRUE-B-NOPF-ALLNOTRD,CTRUE-S-NOPF-ALLNOTRD,CTRUE-B-NOPF-CENTNOTRD";
  mAliases[kDMC7] = "CDMC7-B-NOPF-CENTNOPMD,CDMC7-B-NOPF-CENT,CDMC7-B-NOPF-CENTNOTRD,CDMC7-B-NOPF-ALLNOTRD";
  mAliases[kEG1] = "CINT7EG1-B-NOPF-CENTNOPMD,CEMC7EG1-B-NOPF-CENT,CEMC7EG1-B-NOPF-CENTNOTRD,CEMC7EG1-B-NOPF-CENTNOPMD,CEMC7EG1-B-NOPF-ALLNOTRD,CEMC7EGA-B-NOPF-CENTNOTRD,CEMC7EGA-S-NOPF-CENTNOTRD,CEMC7EGA-S-NOPF-ALLNOTRD,CPBI2EGA-B-NOPF-CENTNOTRD";
  mAliases[kEJ1] = "CINT7EJ1-B-NOPF-CENTNOTRD,CEMC7EJ1-B-NOPF-CENT,CEMC7EJ1-B-NOPF-CENTNOTRD,CEMC7EJ1-B-NOPF-CENTNOPMD,CEMC7EJ1-B-NOPF-ALLNOTRD,CEMC7EJE-B-NOPF-CENTNOTRD,CEMC7EJE-S-NOPF-CENTNOTRD,CEMC7EJE-S-NOPF-ALLNOTRD,CPBI2EJE-B-NOPF-CENTNOTRD";
  mAliases[kEG2] = "CINT7EG2-B-NOPF-CENTNOPMD,CEMC7EG2-B-NOPF-CENT,CEMC7EG2-B-NOPF-CENTNOTRD,CEMC7EG2-B-NOPF-CENTNOPMD,CEMC7EG2-B-NOPF-ALLNOTRD,CEMC7EG2PER-B-NOPF-CENTNOPMD";
  mAliases[kEJ2] = "CINT7EJ2-B-NOPF-CENTNOPMD,CEMC7EJ2-B-NOPF-CENT,CEMC7EJ2-B-NOPF-CENTNOTRD,CEMC7EJ2-B-NOPF-CENTNOPMD,CEMC7EJ2-B-NOPF-ALLNOTRD";
  mAliases[kDG1] = "CINT7DG1-B-NOPF-CENTNOPMD,CDMC7DG1-B-NOPF-CENT,CDMC7DG1-B-NOPF-CENTNOTRD,CDMC7DG1-B-NOPF-CENTNOPMD";
  mAliases[kDJ1] = "CINT7DJ1-B-NOPF-CENTNOPMD,CDMC7DJ1-B-NOPF-CENT,CDMC7DJ1-B-NOPF-CENTNOTRD,CDMC7DJ1-B-NOPF-CENTNOPMD";
  mAliases[kDG2] = "CINT7DG2-B-NOPF-CENTNOPMD,CDMC7DG2-B-NOPF-CENT,CDMC7DG2-B-NOPF-CENTNOTRD,CDMC7DG2-B-NOPF-CENTNOPMD,CDMC7DG2PER-B-NOPF-CENTNOPMD";
  mAliases[kDJ2] = "CINT7DJ2-B-NOPF-CENTNOPMD,CDMC7DJ2-B-NOPF-CENT,CDMC7DJ2-B-NOPF-CENTNOTRD,CDMC7DJ2-B-NOPF-CENTNOPMD";

and in upload_trigger_aliases_run3.C for Run 3 (as of November 2025):

  mAliases[kEMC7] = "CTVXEMC-B-NOPF-EMC";
  mAliases[kDMC7] = "CTVXDMC-B-NOPF-EMC";
  mAliases[kTVXinTRD] = "CMTVX-B-NOPF-TRD,minbias_TVX";
  mAliases[kTVXinEMC] = "C0TVX-B-NOPF-EMC,minbias_TVX_L0,CMTVXTSC-B-NOPF-EMC,CMTVXTCE-B-NOPF-EMC";
  mAliases[kTVXinPHOS] = "C0TVX-B-NOPF-PHSCPV,minbias_TVX_L0,CMTVXTSC-B-NOPF-PHSCPV,CMTVXTSC-B-NOPF-PHSCPV";
  mAliases[kTVXinHMP] = "C0TVX-B-NOPF-HMP,minbias_TVX_L0,CMTVXTSC-B-NOPF-HMP";
  mAliases[kPHOS] = "CTVXPH0-B-NOPF-PHSCPV,mb_PH0_TVX,CPH0SC-B-NOPF-PHSCPV,CPH0CE-B-NOPF-PHSCPV";

This list of trigger aliases and classes is not complete but it should be enough for tests in various PWGs. New trigger classes and aliases can be added upon request (contact Evgeny Kryshen).

Event selection criteria

Full list of event selection criteria can be found in Common/CCDB/EventSelectionParams.h

enum EventSelectionFlags {
  kIsBBV0A = 0,               // cell-averaged time in V0A in beam-beam window
  kIsBBV0C,                   // cell-averaged time in V0C in beam-beam window (for Run 2 only)
  kIsBBFDA,                   // cell-averaged time in FDA (or AD in Run2) in beam-beam window
  kIsBBFDC,                   // cell-averaged time in FDC (or AD in Run2) in beam-beam window
  kIsBBT0A,                   // cell-averaged time in T0A in beam-beam window
  kIsBBT0C,                   // cell-averaged time in T0C in beam-beam window
  kNoBGV0A,                   // cell-averaged time in V0A in beam-gas window
  kNoBGV0C,                   // cell-averaged time in V0C in beam-gas window (for Run 2 only)
  kNoBGFDA,                   // cell-averaged time in FDA (AD in Run2) in beam-gas window
  kNoBGFDC,                   // cell-averaged time in FDC (AD in Run2) in beam-gas window
  kNoBGT0A,                   // cell-averaged time in T0A in beam-gas window
  kNoBGT0C,                   // cell-averaged time in T0C in beam-gas window
  kIsBBZNA,                   // time in common ZNA channel in beam-beam window
  kIsBBZNC,                   // time in common ZNC channel in beam-beam window
  kIsBBZAC,                   // time in ZNA and ZNC in beam-beam window - circular cut in ZNA-ZNC plane
  kNoBGZNA,                   // time in common ZNA channel is outside of beam-gas window
  kNoBGZNC,                   // time in common ZNC channel is outside of beam-gas window
  kNoV0MOnVsOfPileup,         // no out-of-bunch pileup according to online-vs-offline VOM correlation
  kNoSPDOnVsOfPileup,         // no out-of-bunch pileup according to online-vs-offline SPD correlation
  kNoV0Casymmetry,            // no beam-gas according to correlation of V0C multiplicities in V0C3 and V0C012
  kIsGoodTimeRange,           // good time range
  kNoIncompleteDAQ,           // complete event according to DAQ flags
  kNoTPCLaserWarmUp,          // no TPC laser warm-up event (used in Run 1)
  kNoTPCHVdip,                // no TPC HV dip
  kNoPileupFromSPD,           // no pileup according to SPD vertexer
  kNoV0PFPileup,              // no out-of-bunch pileup according to V0 past-future info
  kNoSPDClsVsTklBG,           // no beam-gas according to cluster-vs-tracklet correlation
  kNoV0C012vsTklBG,           // no beam-gas according to V0C012-vs-tracklet correlation
  kNoInconsistentVtx,         // no inconsistency in SPD and Track vertices
  kNoPileupInMultBins,        // no pileup according to multiplicity-differential pileup checks
  kNoPileupMV,                // no pileup according to multi-vertexer
  kNoPileupTPC,               // no pileup in TPC
  kIsTriggerTVX,              // FT0 vertex (acceptable FT0C-FT0A time difference) at trigger level
  kIsINT1,                    // SPDGFO >= 1 || V0A || V0C
  kNoITSROFrameBorder,        // bunch crossing is far from ITS RO Frame border
  kNoTimeFrameBorder,         // bunch crossing is far from Time Frame borders
  kNoSameBunchPileup,         // reject collisions in case of pileup with another collision in the same foundBC
  kIsGoodZvtxFT0vsPV,         // small difference between z-vertex from PV and from FT0
  kIsVertexITSTPC,            // at least one ITS-TPC track (reject vertices built from ITS-only tracks)
  kIsVertexTOFmatched,        // at least one of vertex contributors is matched to TOF
  kIsVertexTRDmatched,        // at least one of vertex contributors is matched to TRD
  kNoCollInTimeRangeNarrow,   // no other collisions in specified time range (narrower than Strict)
  kNoCollInTimeRangeStrict,   // no other collisions in specified time range
  kNoCollInTimeRangeStandard, // no other collisions in specified time range with per-collision multiplicity above threshold
  kNoCollInRofStrict,         // no other collisions in this Readout Frame
  kNoCollInRofStandard,       // no other collisions in this Readout Frame with per-collision multiplicity above threshold
  kNoHighMultCollInPrevRof,   // veto an event if FT0C amplitude in previous ITS ROF is above threshold
  kIsGoodITSLayer3,           // number of inactive chips on ITS layer 3 is below maximum allowed value
  kIsGoodITSLayer0123,        // numbers of inactive chips on ITS layers 0-3 are below maximum allowed values
  kIsGoodITSLayersAll,        // numbers of inactive chips on all ITS layers are below maximum allowed values
  kNsel                       // counter
}; // (as of October 2025)

Technically there are three types of criteria:

  • based on flags from bc-joinable aod::Run2BCInfos table (kIsGoodTimeRange, kNoIncompleteDAQ, kNoTPCLaserWarmUp, kNoTPCHVdip, kNoPileupFromSPD, kNoV0PFPileup)
  • based on information from FIT and ZDC detectors (kIsBB…, kIsBG…) and/or additional information stored in aod::Run2BCInfos table (kNoV0MOnVsOfPileup,kNoSPDOnVsOfPileup)
  • based on additional information from aod::Collisions table

Decisions on inidividual selection criteria are stored in selection bitmask BcSels and EvSels tables. E.g. one can check if a given collision passed kNoSameBunchPileup selection:

  bool noSameBunchPileup = col.selection_bit(evsel::kNoSameBunchPileup);

kNoTimeFrameBorder and kNoITSROFrameBorder selections are described in detail in this section.

Occupancy-related selections (kNoCollInTimeRangeNarrow, kNoCollInTimeRangeStandard, kNoCollInTimeRangeStrict, kNoCollInRofStrict, kNoCollInRofStandard, kNoHighMultCollInPrevRof) are described in this section.

Time Frame and ITS ROF border cuts

Time Frame borders

In Run 3 and 4, ALICE operates in continuous readout mode, where data are stored in Time Frames (TFs) that correspond to 32 LHC orbits, ≈ 2.9 ms (note: in 2022 pp TFs were longer - 128 LHC orbits), and each TF is reconstructed independently.

Because the drift time of electrons in the TPC is ≈ 100 μs, collisions near the end of a TF lack full information, resulting in a depletion of vertex contributors and a drop in ITS+TPC tracking efficiency during the last ≈ 1.1 LHC orbits of the TF (LHC orbit is ≈89 μs). Additional effect takes place at the beginning of the next TF, where the reconstruction starts when the electrons from pre-TF collisions are still drifting.

Mitigation in event selection:

  • A dedicated event-selection bit kNoTimeFrameBorder was introduced (February 2024) to reject events close to TF edges:
    • Cuts ≈ 300 bunch crossings (BCs) at the start and ≈ 4000 BCs at the end of each TF.
    • Corresponds to ≈ 3.7% event loss for 2023–25 data.
  • This cut ensures full TPC drift information for all accepted events, removing TF-edge artefacts in vertex and track distributions.
  • Usage in analysis:
    if (col.selection_bit(o2::aod::evsel::kNoTimeFrameBorder)) { /* do analysis */ }
    

ITS Readout Frame borders

  • Although the global readout is continuous, the ITS2 detector is read out in discrete Readout Frames (ROFs):
    • in pp: 18 ROFs per LHC orbit, each ≈ 5 μs (198 BCs; full orbit contains 3564 BCs)
    • in Pb–Pb: 6 ROFs per orbit, ≈ 15 μs each (594 BCs)
  • Cluster losses occur at ROF boundaries due to the ALPIDE chip's time-walk effect:
    • ITS clusters from an interaction in ROF i may appear only in ROF i + 1
    • this causes a sharp drop in ITS cluster and track counts at ROF edges.
  • The effect is particle-dependent, e.g. protons (larger dE/dx) are recorded earlier than pions—leading to small PID-dependent distortions near borders.

Mitigation in event selection

  • The selection bit kNoITSROFrameBorder rejects events near ITS ROF edges:
    • removes collisions within 10 BCs at the beginning and 20 BCs at the end of each ROF
    • it correponds to ≈ 15 % of (nominal) BCs in pp and ≈5 % of BCs in Pb–Pb (corresponding event losses depend on the LHC filling scheme)
  • Usage in analysis:
    if (col.selection_bit(o2::aod::evsel::kNoITSROFrameBorder)) { /* do analysis */ }
    
  • Note that this cut also helps to remove collisions at the ROF borders in the MFT detector (where the same chips as in the ITS are used, and the MFT ROFs are time-aligned with the ITS ROFs).

More details on the TF and ROF border effects and cuts can be found e.g. in slides 2-17 of presentation (November 2024).

Integration into event selection

  • Since April 2024, both the Time Frame border and ITS Readout Frame border cuts are combined with the kIsTriggerTVX condition via logical and in the sel8 event-selection bit, ensuring events used for analysis are free from TF and ROF boundary artifacts.

Occupancy estimation

In ALICE Run 3 Pb–Pb collisions, occupancy effects in the TPC refer to the deterioration of the quality of an event's TPC clusters by signals from other nearby collisions within the TPC drift time window.

  • The TPC has a long drift time (~100 µs), so clusters from particles originated from multiple collisions can overlap in the detector.
  • Higher occupancy worsens:
    • Tracking efficiency
    • PID performance (dE/dx shifts, peak broadening)

Occupancy estimators

A single-value "integrated" occupancy estimator for a given collision can be calculated by summing

  • the number of ITS tracks from other collisions within a defined time window around the given event. In the analysis, it can be accessed as:
    int occupancyByTracks = col.trackOccupancyInTimeRange(); // range: from 0 up to ~15k
    
  • alternatively, we can sum up FT0C amplitudes from other collisions:
     float occupancyByFT0C = col.ft0cOccupancyInTimeRange();  // range: from 0 up to ~150k
    

Notes:

  • Both occupancy estimators are pre-calculated per each collision in the event selection routine, EventSelectionModule.h.
  • In the occupancy calculation, multiplicities of nearby collisions are "weighted" according to their time separation from a collision-of-interest, to reflect the "severity" of their influence on that collision
  • Estimators return -1 if a given collision is close to Time Frame borders (so, not enough information for the occupancy calculation, while we need information within -40 µs…+100 µs time range wrt a given collision), event loss ~1.2%.

Occupancy selection bits

In addition to the occupancy estimators described above, several special event selection bits are implemented to better clean up various nearby effects from other collisions (related not only the TPC but also the ITS, e.g. due to high occupancies in the ITS Readout Frames).

The following table summarizes the event selection bits that can help to mitigate the occupancy effects:

Bit Definition Strictness Typical Effect / Event Loss
kNoCollInTimeRangeNarrow Rejects events if another collision within ±0.25 µs Narrow veto Useful to suppress residual BC mis-associations; minimal event loss, ~1-1.5%
kNoCollInTimeRangeStandard Rejects if: (1) another coll. within ±0.25 µs, or (2) multiplicity of a coll. in delta time −4…+2 µs > threshold Medium Further suppression of effects from nearby collisions; ~3-7% event loss depending on IR
kNoCollInTimeRangeStrict Rejects events if another collision is within ±10 µs Very strict Strongly reduces effects from nearby events; large loss of statistics at high IR (can exceed 30–40%)
kNoCollInRofStrict Rejects events if >1 collision in the same ITS Readout Frame (~15 µs in Pb-Pb) Very strict Removes in-ROF pileup; at 38 kHz Pb–Pb cuts ~35% of events
kNoCollInRofStandard Allows >1 collision per ROF but rejects if another has multiplicity > threshold (default: FT0C amplitude >5000 a.u. ≈ 500 tracks) Medium Retains more stats, but protects against large in-ROF pileup
kNoHighMultCollInPrevRof Vetoes event if previous ROF has high multiplicity (FT0C >5000 a.u.); only for cross-ROF ITS reco Medium Removes cases where previous ROF "steals" clusters; few % loss, but improves ITS tracking quality

These bits can be used as follows:

if (col.selection_bit(o2::aod::evsel::kNoCollInTimeRangeNarrow)) { /* do analysis */ }

Discussion

More details on occupancy in Pb-Pb can be found in the report at the APW 2024, where the concepts and observations are explained (while some figures and indicated values might be outdated).

Tight cuts on occupancy improve quality (better S/B, cleaner PID, less bias in kinematics), but reduce event statistics.

However, sensitivity to the occupancy effects depends on analysis. Therefore, the suggested approach is to study how results of a given analysis change as a function of occupancy: one may try several occupancy "bins", e.g. [0,500), [500, 1000), [1000-2000), [2000-4000), etc., and, in addition, apply occupancy selection bits, e.g. kNoCollInTimeRangeNarrow to eliminate the bc-collision mismatches, or kNoCollInTimeRangeStandard to make a further cleaunup.

Note that TPC-related occupancy effects are most pronounced in Pb–Pb runs, however, the tools described above can also be used for occupancy studies in pp and light-ion runs.

Rejection of events with dead zones in ITS

Problem

  • The ITS occasionally develops "holes" in acceptance lasting about 6–8 seconds, visible as gaps in φ distributions on time-dependent plots.
  • These are caused by reboots of ITS staves, typically triggered by recovery of failed lanes (groups of 7 chips sharing one data link). When a lane fails, the full stave becomes temporarily blind while the DCS recovers it.
  • The issue affects, in particular, ITS Layer 3, which is critical for achieving four consecutive ITS hits in tracking.
  • These dead periods correlate also with drops in ITS–TPC matching efficiency.
  • The effect appears both in A-A and pp data.

For more details, see the report at DPG AOT meeting, Jan 2025.

Using special event selection bits

To reject events recorded during the affected time intervals, several event selection bits were introduced in O2Physics (PR #9038, Dec 2024):

kIsGoodITSLayer3,           // number of inactive chips on ITS layer 3 is below maximum allowed value
kIsGoodITSLayer0123,        // numbers of inactive chips on ITS layers 0-3 are below maximum allowed values
kIsGoodITSLayersAll,        // numbers of inactive chips on all ITS layers are below maximum allowed values

Example usage:

if (col.selection_bit(o2::aod::evsel::kIsGoodITSLayersAll)) {
    // do analysis
}

This particular bit kIsGoodITSLayersAll ensures that all ITS layers are in a good state (i.e. no rebooting staves; note that at the same time some chips can be inactive, so, smaller holes in the acceptance can remain).

The logic behind these cuts uses CCDB maps of dead chips and defines per-layer thresholds for allowed inactive chips:

maxInactiveChipsPerLayer = {8, 8, 8, 111, 111, 195, 195};

If any layer exceeds its threshold, the event is flagged as bad (likely during a stave reboot).

Applying this cut removes time intervals with dead ITS staves and, correspondingly, the large acceptance holes, significantly flattening time dependence of the observables, like the 2- and 4-particle correlators in Pb-Pb. Note that in pp the kIsGoodITSLayersAll bit can reject a huge fraction of events (the holes in pp are more frequent), instead, the kIsGoodITSLayer0123 bit can be tried (e.g. to study effects from the rebooting staves on track DCA).

Usage of RCT flags

See slides for details.

Found bunch crossings

One of the main goals of the event selection task in Run 3 is to find the original bunch crossing for each collision. The basic approach is to start from estimated collision bc and search for closest BC containing FT0 entries in a +/-4 sigma window where sigma corresponds to the estimated collision time resolution from col.collisionTimeRes(). Implementation details can be found in eventSelection.cxx#L348.

Users can access found bunch crossings and FT0 entries using foundBC or foundFT0 indices stored in the EvSels table:

if (collision.has_foundBC()) {
  auto bc = collision.foundBC();
  uint64_t globalBC = bc.globalBC();
}

or

if (collision.has_foundFT0()) {
  auto ft0 = collision.foundFT0();
  int triggersignals = ft0.triggerMask();
}

If bunch crossing with FT0 entries is not found, foundBC and foundFT0 indices are set to -1 therefore one has to check collision.has_foundBC() or collision.has_foundFT0() before accessing corresponding info.

Configurables

Event selection module supports a number of configurables, see bcselConfigurables and evselConfigurables, e.g.

  • muonSelection allows to activate reduced set of checks for muon analyses in Run 2:

    Configurable<int> muonSelection{"muonSelection", 0, "0 - barrel, 1 - muon selection with pileup cuts, 2 - muon selection without pileup cuts"};
    
  • TODO: add other examples

One can set various configurables in the json file. This json file has to be provided using --configuration option:

  o2-analysis-event-selection-service --configuration json://config.json -b

Event selection decisions for Run 2

Offline event selection decisions for Run 2 (e.g. sel7) are constructed based on a subsample of individual checks stored in selection bits. The default list of checks may depend on colliding system, specific run conditions and specific analysis requirements. Default set of checks for Run 2 can be found in Common/CCDB/EventSelectionParams.cxx. The default selectionBarrel masks for pp, pA, Ap and AA are summarized below:

  • default sel7 selection in pp is based on the requirement of beam-beam timing in V0A and V0C and a number of pileup, beam-gas and othe quality checks
    selectionBarrel[kIsBBV0A] = 1;
    selectionBarrel[kIsBBV0C] = 1;
    selectionBarrel[kNoV0MOnVsOfPileup] = 1;
    selectionBarrel[kNoSPDOnVsOfPileup] = 1;
    selectionBarrel[kNoV0Casymmetry] = 1;
    selectionBarrel[kIsGoodTimeRange] = 1;
    selectionBarrel[kNoIncompleteDAQ] = 1;
    selectionBarrel[kNoTPCHVdip] = 1;
    selectionBarrel[kNoPileupFromSPD] = 1;
    selectionBarrel[kNoV0PFPileup] = 1;
    selectionBarrel[kNoSPDClsVsTklBG] = 1;
    selectionBarrel[kNoV0C012vsTklBG] = 1;
  • checks for pA system are similar to pp but in addition they include no beam-gas in ZNA:
    selectionBarrel[kNoBGZNA] = 1;
  • checks for Ap system are similar to pp but in addition they include no beam-gas in ZNC:
    selectionBarrel[kNoBGZNC] = 1;
  • default checks for AA are much simpler compared to pp since hadronic pileup is at per-mile level and can be ignored in the first approximation. Default checks include beam-beam timing in V0A, V0C, ZNA and ZNC detectors and a couple of quality checks.
    selectionBarrel[kIsBBV0A] = 1;
    selectionBarrel[kIsBBV0C] = 1;
    selectionBarrel[kIsBBZAC] = 1;
    selectionBarrel[kIsGoodTimeRange] = 1;
    selectionBarrel[kNoTPCHVdip] = 1;

In addition we define selectionMuonWithPileupCuts and selectionMuonWithoutPileupCuts with reduced set of checks, see Common/CCDB/EventSelectionParams.cxx for more details.

Besides, there are special settings for some run ranges, e.g. we remove checks on out-of-bunch pileup for runs with isolated bunches:

  selectionBarrel[kNoV0MOnVsOfPileup] = 0;
  selectionBarrel[kNoSPDOnVsOfPileup] = 0;
  selectionBarrel[kNoV0Casymmetry] = 0;
  selectionBarrel[kNoV0PFPileup] = 0;

These special settings are stored in CCDB. One can find relevant details in upload_event_selection_params.C macro.

Finally, it is worth mentioning that out-of-bunch pileup cuts as well as ZDC timing checks are disabled in MC eventSelection.cxx#L265:

    if (isMC) {
      applySelection[kIsBBZAC] = 0;
      applySelection[kNoV0MOnVsOfPileup] = 0;
      applySelection[kNoSPDOnVsOfPileup] = 0;
      applySelection[kNoV0Casymmetry] = 0;
      applySelection[kNoV0PFPileup] = 0;
    }

Selection mask applySelection is obtained from CCDB in eventSelection.cxx:

  EventSelectionParams* par = ccdb->getForTimeStamp<EventSelectionParams>("EventSelection/EventSelectionParams", bc.timestamp());

Then sel7 decision is constructed from active checks: Common/TableProducer/eventSelection.cxx

    bool sel7 = 1;
    for (int i = 0; i < kNsel; i++) {
      sel7 &= applySelection[i] ? selection[i] : 1;
    }

Remarks

  • One has to apply offline selections in O2 explicitly in contrast to AliPhysics where these selections were applied together with trigger alias selection.
  • EvSel table might be also useful in user tasks relying on beam-beam and beam-gas decisions in forward detectors, e.g. in UPC tasks.

Luminosity module

(to be added)