37#include <TStopwatch.h>
50 std::vector<OutputSpec> outputs;
52 outputs.emplace_back(detOrig,
"DIGITS", iLayer, Lifetime::Timeframe);
53 outputs.emplace_back(detOrig,
"DIGITSROF", iLayer, Lifetime::Timeframe);
55 outputs.emplace_back(detOrig,
"DIGITSMC2ROF", iLayer, Lifetime::Timeframe);
56 outputs.emplace_back(detOrig,
"DIGITSMCTR", iLayer, Lifetime::Timeframe);
59 outputs.emplace_back(detOrig,
"ROMode", 0, Lifetime::Timeframe);
76 mDisableQED = ic.
options().
get<
bool>(
"disable-qed");
77 mLocalRespFile = ic.
options().
get<std::string>(
"local-response-file");
92 const bool withQED = context->isQEDProvided() && !mDisableQED;
93 auto& timesview = context->getEventRecords(withQED);
94 LOG(info) <<
"GOT " << timesview.size() <<
" COLLISION TIMES";
95 LOG(info) <<
"SIMCHAINS " << mSimChains.size();
98 if (timesview.empty()) {
103 LOG(info) <<
" CALLING TRK DIGITIZATION ";
105 auto& eventParts = context->getEventParts(withQED);
107 for (uint32_t iLayer = 0; iLayer < static_cast<uint32_t>(mLayers); ++iLayer) {
108 mDigits[iLayer].clear();
109 mROFRecords[iLayer].clear();
110 mROFRecordsAccum[iLayer].clear();
112 mLabels[iLayer].clear();
113 mLabelsAccum[iLayer].clear();
114 mMC2ROFRecordsAccum[iLayer].clear();
123 auto& digitsAccum = pc.
outputs().
make<std::vector<itsmft::Digit>>(
Output{mOrigin,
"DIGITS", iLayer});
128 mROFRecordsAccum[iLayer].reserve(nROFsTF);
130 auto accumulate = [
this, &digitsAccum, &iLayer]() {
133 if (mDigits[iLayer].
empty()) {
136 auto ndigAcc = digitsAccum.size();
137 std::copy(mDigits[iLayer].begin(), mDigits[iLayer].
end(), std::back_inserter(digitsAccum));
139 for (
auto& rof : mROFRecords[iLayer]) {
140 rof.setFirstEntry(ndigAcc + rof.getFirstEntry());
143 std::copy(mROFRecords[iLayer].begin(), mROFRecords[iLayer].
end(), std::back_inserter(mROFRecordsAccum[iLayer]));
145 mLabelsAccum[iLayer].mergeAtBack(mLabels[iLayer]);
147 LOG(info) <<
"Added " << mDigits[iLayer].size() <<
" digits on layer " << iLayer;
148 mLabels[iLayer].clear();
149 mDigits[iLayer].clear();
150 mROFRecords[iLayer].clear();
154 for (
size_t collID = 0; collID < timesview.size(); ++collID) {
155 auto irt = timesview[collID];
156 if (irt.toLong() < bcShift) {
163 for (
auto& part : eventParts[collID]) {
167 if (!mHits.empty()) {
168 LOG(
debug) <<
"For collision " << collID <<
" eventID " << part.entryID
169 <<
" found " << mHits.size() <<
" hits on layer " << iLayer;
170 mDigitizer.
process(&mHits, part.entryID, part.sourceID, iLayer);
180 nDigits += digitsAccum.size();
182 std::vector<o2::itsmft::ROFRecord> expDigitRofVec(nROFsTF);
183 for (
int iROF = 0; iROF < nROFsTF; ++iROF) {
184 auto& rof = expDigitRofVec[iROF];
188 rof.setROFrame(iROF);
190 rof.setFirstEntry(-1);
193 for (
const auto& rof : mROFRecordsAccum[iLayer]) {
194 const auto&
ir = rof.getBCData();
195 const auto irToFirst =
ir - firstIR;
196 const auto irROF = irToFirst.
toLong() / roFrameLengthInBC;
197 if (irROF < 0 || irROF >= nROFsTF) {
200 auto& expROF = expDigitRofVec[irROF];
201 expROF.setFirstEntry(rof.getFirstEntry());
202 expROF.setNEntries(rof.getNEntries());
203 if (expROF.getBCData() != rof.getBCData()) {
204 LOGP(fatal,
"detected mismatch between expected {} and received {}", expROF.asString(), rof.asString());
209 for (
auto& rof : expDigitRofVec) {
210 if (rof.getFirstEntry() < 0) {
211 rof.setFirstEntry(prevFirst);
213 prevFirst = rof.getFirstEntry();
218 std::vector<o2::itsmft::MC2ROFRecord> clippedMC2ROFRecords;
219 clippedMC2ROFRecords.reserve(mMC2ROFRecordsAccum[iLayer].
size());
220 for (
auto mc2rof : mMC2ROFRecordsAccum[iLayer]) {
221 if (mc2rof.minROF >=
static_cast<uint32_t
>(nROFsTF) || mc2rof.minROF > mc2rof.maxROF) {
222 mc2rof.rofRecordID = -1;
226 mc2rof.maxROF = std::min<uint32_t>(mc2rof.maxROF, nROFsTF - 1);
227 if (mc2rof.minROF > mc2rof.maxROF) {
228 mc2rof.rofRecordID = -1;
232 mc2rof.rofRecordID = mc2rof.minROF;
235 clippedMC2ROFRecords.push_back(mc2rof);
239 mLabelsAccum[iLayer].flatten_to(sharedlabels);
240 mLabels[iLayer].clear_andfreememory();
241 mLabelsAccum[iLayer].clear_andfreememory();
244 LOG(info) << mID.
getName() <<
": Sending ROMode= " << mROMode <<
" to GRPUpdater";
248 LOG(info) <<
"Digitization took " << timer.CpuTime() <<
"s";
249 LOG(info) <<
"Produced " << nDigits <<
" digits";
259 std::unique_ptr<TFile> file(TFile::Open(mLocalRespFile.data(),
"READ"));
261 LOG(fatal) <<
"Cannot open response file " << mLocalRespFile;
268 static bool initOnce{
false};
283 mDigits.resize(mLayers);
284 mROFRecords.resize(mLayers);
285 mROFRecordsAccum.resize(mLayers);
286 mLabels.resize(mLayers);
287 mLabelsAccum.resize(mLayers);
288 mMC2ROFRecordsAccum.resize(mLayers);
290 for (
int iLayer = 0; iLayer < mLayers; ++iLayer) {
291 const auto roFrameLengthInBC = aopt.getROFLengthInBC(iLayer);
293 digipar.setROFrameLengthInBC(roFrameLengthInBC, iLayer);
295 digipar.setROFrameBiasInBC(aopt.getROFBiasInBC(iLayer) + aopt.getROFDelayInBC(iLayer), iLayer);
296 digipar.setStrobeDelay(aopt.getStrobeDelay(iLayer), iLayer);
297 const auto strobeLengthCont = aopt.getStrobeLengthCont(iLayer);
298 digipar.setStrobeLength(strobeLengthCont > 0 ? strobeLengthCont : frameNS - aopt.getStrobeDelay(iLayer), iLayer);
299 digipar.setROFrameLength(frameNS, iLayer);
302 digipar.getSignalShape().setParameters(dopt.strobeFlatTop, dopt.strobeMaxRiseTime, dopt.strobeQRiseTime0);
303 digipar.setChargeThreshold(dopt.chargeThreshold);
304 digipar.setNoisePerPixel(dopt.noisePerPixel);
305 digipar.setTimeOffset(dopt.timeOffset);
306 digipar.setNSimSteps(dopt.nSimSteps);
309 LOG(info) << mID.
getName() <<
" simulated in CONTINUOUS RO mode";
325 LOG(info) << mID.
getName() <<
" Almira param updated";
327 par.printKeyValues();
336 LOG(info) << mID.
getName() <<
" loaded APTSResponseData";
337 if (mLocalRespFile.empty()) {
338 LOG(info) <<
"Using CCDB/APTS response file";
342 LOG(info) <<
"Response function will be loaded from local file: " << mLocalRespFile;
350 bool mWithMCTruth{
true};
351 bool mFinished{
false};
352 bool mDisableQED{
false};
353 unsigned long mFirstOrbitTF = 0x0;
354 std::string mLocalRespFile{
""};
359 std::vector<std::vector<o2::itsmft::Digit>> mDigits{};
360 std::vector<std::vector<o2::itsmft::ROFRecord>> mROFRecords{};
361 std::vector<std::vector<o2::itsmft::ROFRecord>> mROFRecordsAccum{};
362 std::vector<o2::trk::Hit> mHits{};
363 std::vector<o2::trk::Hit>* mHitsP{&mHits};
364 std::vector<o2::dataformats::MCTruthContainer<o2::MCCompLabel>> mLabels{};
365 std::vector<o2::dataformats::MCTruthContainer<o2::MCCompLabel>> mLabelsAccum{};
366 std::vector<std::vector<o2::itsmft::MC2ROFRecord>> mMC2ROFRecordsAccum{};
367 std::vector<TChain*> mSimChains{};
375 std::vector<InputSpec> inputs;
376 inputs.emplace_back(
"collisioncontext",
"SIM",
"COLLISIONCONTEXT",
static_cast<SubSpecificationType>(channel), Lifetime::Timeframe);
381 inputs.emplace_back(
"TRK_aptsresp",
"TRK",
"APTSRESP", 0, Lifetime::Condition,
ccdbParamSpec(
"IT3/Calib/APTSResponse"));
384 inputs, makeOutChannels(detOrig, mctruth),
Definition of the base digitizer task class.
A const (ready only) version of MCTruthContainer.
o2::framework::DataAllocator::SubSpecificationType SubSpecificationType
Header of the General Run Parameters object.
Definition of the TRK digitizer.
virtual void init(o2::framework::InitContext &) final
static const HBFUtils & Instance()
Static class with identifiers, bitmasks and names for ALICE detectors.
static constexpr const char * getName(ID id)
names of defined detectors
static const std::array< std::vector< std::string >, DetID::nDetectors > DETECTORBRANCHNAMES
T get(const char *key) const
void snapshot(const Output &spec, T const &object)
o2::header::DataHeader::SubSpecificationType SubSpecificationType
decltype(auto) make(const Output &spec, Args... args)
ConfigParamRegistry const & options()
DataAllocator & outputs()
The data allocator is used to allocate memory for the output data.
InputRecord & inputs()
The inputs associated with this processing context.
ServiceRegistryRef services()
The services registry associated with this processing context.
bool initSimChains(o2::detectors::DetID detid, std::vector< TChain * > &simchains) const
int getROFrameBiasInBC(int layer) const
void setResponse(const o2::itsmft::AlpideSimResponse *)
int getROFrameLengthInBC(int layer) const
o2::trk::DigiParams & getParams()
void setResponseName(const std::string &name)
void setEventTime(const o2::InteractionTimeRecord &irt, int layer)
uint32_t getEventROFrameMin() const
void process(const std::vector< o2::trk::Hit > *hits, int evID, int srcID, int layer)
Steer conversion of hits to digits.
void setDigits(std::vector< o2::itsmft::Digit > *dig)
void setROFRecords(std::vector< o2::itsmft::ROFRecord > *rec)
void setMCLabels(o2::dataformats::MCTruthContainer< o2::MCCompLabel > *mclb)
void fillOutputContainer(uint32_t maxFrame, int layer)
void resetROFrameBounds()
void setGeometry(const o2::trk::GeometryTGeo *gm)
uint32_t getEventROFrameMax() const
void resetEventROFrames()
void Print(Option_t *opt="") const
void fillMatrixCache(int mask)
int getNumberOfLayersMLOT() const
static GeometryTGeo * Instance()
TRKDPLDigitizerTask(bool mctruth=true)
void run(framework::ProcessingContext &pc)
void updateTimeDependentParams(ProcessingContext &pc)
void finaliseCCDB(ConcreteDataMatcher &matcher, void *obj)
void setLocalResponseFunction()
void initDigitizerTask(framework::InitContext &ic) override
constexpr o2::header::DataOrigin gDataOriginTRK
constexpr int LHCMaxBunches
constexpr double LHCBunchSpacingNS
Defining ITS Vertex explicitly as messageable.
std::vector< ConfigParamSpec > ccdbParamSpec(std::string const &path, int runDependent, std::vector< CCDBMetadata > metadata={}, int qrate=0)
std::vector< ConfigParamSpec > Options
header::DataHeader::SubSpecificationType SubSpecificationType
DataProcessorSpec getTRKDigitizerSpec(int channel, bool mctruth)
class listing possible services
int getNOrbitsPerTF() const
get IR corresponding to start of the HBF
static constexpr size_t getNLayers()
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
o2::InteractionRecord ir(0, 0)