47G4DNAIndependentReactionTimeStepper::Utils::Utils(
const G4Track& trackA,
const G4Track& trackB)
48 : fTrackA(trackA), fTrackB(trackB)
56 fReactionSet->SortByTime();
62 fSampledPositions.clear();
66void G4DNAIndependentReactionTimeStepper::InitializeForNewTrack()
72 fHasAlreadyReachedNullTime =
false;
79 InitializeForNewTrack();
85 G4cout <<
"________________________________________________________________"
88 G4cout <<
"G4DNAIndependentReactionTimeStepper::CalculateStep" <<
G4endl;
89 G4cout <<
"Check done for molecule : " << pMoleculeA->GetName() <<
" (" << trackA.
GetTrackID()
94 auto pMolConfA = pMoleculeA->GetMolecularConfiguration();
96 const auto pReactantList = fMolecularReactionTable->CanReactWith(pMolConfA);
98 if (pReactantList ==
nullptr) {
101 msg <<
"G4DNAIndependentReactionTimeStepper::CalculateStep will return infinity "
102 "for the reaction because the molecule "
103 << pMoleculeA->GetName() <<
" does not have any reactants given in the reaction table."
105 G4Exception(
"G4DNAIndependentReactionTimeStepper::CalculateStep",
106 "G4DNAIndependentReactionTimeStepper03",
JustWarning, msg);
111 auto nbReactives = (
G4int)pReactantList->size();
113 if (nbReactives == 0) {
116 msg <<
"G4DNAIndependentReactionTimeStepper::CalculateStep will "
118 "for the reaction because the molecule "
119 << pMoleculeA->GetName() <<
" does not have any reactants given in the reaction table."
120 <<
"This message can also result from a wrong implementation of "
121 "the reaction table."
123 G4Exception(
"G4DNAIndependentReactionTimeStepper::CalculateStep",
124 "G4DNAIndependentReactionTimeStepper04",
JustWarning, msg);
128 fReactants = std::make_shared<vector<G4Track*>>();
129 fReactionModel->Initialise(pMolConfA, trackA);
130 for (
G4int i = 0; i < nbReactives; ++i) {
131 auto pMoleculeB = (*pReactantList)[i];
132 G4int key = pMoleculeB->GetMoleculeID();
138 const G4double Reff = fReactionModel->GetReactionRadius(i);
139 std::vector<std::pair<G4TrackList::iterator, G4double>> resultIndices;
140 resultIndices.clear();
143 if (resultIndices.empty()) {
146 for (
auto& it : resultIndices) {
147 G4Track* pTrackB = *(std::get<0>(it));
149 if (pTrackB == &trackA) {
152 if (pTrackB ==
nullptr) {
154 exceptionDescription <<
"No trackB no valid";
156 "G4DNAIndependentReactionTimeStepper"
158 "G4DNAIndependentReactionTimeStepper007",
FatalException, exceptionDescription);
161 if (fCheckedTracks.find(pTrackB->
GetTrackID()) != fCheckedTracks.end()) {
165 Utils utils(trackA, *pTrackB);
168 auto pMolConfB = pMolB->GetMolecularConfiguration();
170 if (distance * distance < Reff * Reff) {
171 auto reactionData = fMolecularReactionTable->GetReactionData(pMolConfA, pMolConfB);
174 if (!fHasAlreadyReachedNullTime) {
176 fHasAlreadyReachedNullTime =
true;
179 CheckAndRecordResults(utils);
184 G4double tempMinET = GetTimeToEncounter(trackA, *pTrackB);
193 CheckAndRecordResults(utils);
201 G4cout <<
"G4DNAIndependentReactionTimeStepper::CalculateStep will finally "
206 G4cout <<
"Selected reactants for trackA: " << pMoleculeA->GetName() <<
" ("
209 vector<G4Track*>::iterator it;
221void G4DNAIndependentReactionTimeStepper::CheckAndRecordResults(
const Utils& utils)
223 if (utils.fTrackB.GetTrackStatus() !=
fAlive) {
227 if (&utils.fTrackB == &utils.fTrackA) {
229 msg <<
"A track is reacting with itself"
230 " (which is impossible) ie fpTrackA == trackB"
232 msg <<
"Molecule A is of type : " << utils.fpMoleculeA->GetName()
233 <<
" with trackID : " << utils.fTrackA.GetTrackID()
234 <<
" and B : " << utils.fpMoleculeB->GetName()
235 <<
" with trackID : " << utils.fTrackB.GetTrackID() <<
G4endl;
236 G4Exception(
"G4DNAIndependentReactionTimeStepper::RetrieveResults",
241 if (fabs(utils.fTrackB.GetGlobalTime() - utils.fTrackA.GetGlobalTime())
242 > utils.fTrackA.GetGlobalTime() * (1. - 1. / 100))
246 msg <<
"The interacting tracks are not synchronized in time" <<
G4endl;
247 msg <<
"trackB->GetGlobalTime() != fpTrackA.GetGlobalTime()" <<
G4endl;
249 msg <<
"fpTrackA : trackID : " << utils.fTrackA.GetTrackID()
250 <<
"\t Name :" << utils.fpMoleculeA->GetName()
251 <<
"\t fpTrackA->GetGlobalTime() = "
254 msg <<
"trackB : trackID : " << utils.fTrackB.GetTrackID()
255 <<
"\t Name :" << utils.fpMoleculeB->GetName()
256 <<
"\t trackB->GetGlobalTime() = "
259 G4Exception(
"G4DNAIndependentReactionTimeStepper::RetrieveResults",
263 fReactants->push_back(
const_cast<G4Track*
>(&utils.fTrackB));
270 if (pReactionSet ==
nullptr) {
275 if (reactionPerTime.empty()) {
278 for (
auto reaction_i = reactionPerTime.begin(); reaction_i != reactionPerTime.end();
279 reaction_i = reactionPerTime.begin())
281 if ((*reaction_i)->GetTime() > currentStepTime) {
282 fReactionSet->CleanAllReaction();
286 G4Track* pTrackA = (*reaction_i)->GetReactants().first;
290 G4Track* pTrackB = (*reaction_i)->GetReactant(pTrackA);
295 if (pTrackB == pTrackA) {
297 msg <<
"The IT reaction process sent back a reaction "
298 "between trackA and trackB. ";
299 msg <<
"The problem is trackA == trackB";
300 G4Exception(
"G4DNAIndependentReactionTimeStepper::FindReaction",
305 if (fpReactionProcess !=
nullptr
306 && fpReactionProcess->TestReactibility(*pTrackA, *pTrackB, currentStepTime,
false))
308 if ((fSampledPositions.find(pTrackA->
GetTrackID()) == fSampledPositions.end()
309 && (fSampledPositions.find(pTrackB->
GetTrackID()) == fSampledPositions.end())))
312 msg <<
"The positions of trackA and trackB have no counted ";
313 G4Exception(
"G4DNAIndependentReactionTimeStepper::FindReaction",
320 auto pReactionChange = fpReactionProcess->MakeReaction(*pTrackA, *pTrackB);
321 if (pReactionChange ==
nullptr) {
324 return pReactionChange;
332 fReactionModel = pReactionModel;
337 return fReactionModel;
345G4double G4DNAIndependentReactionTimeStepper::GetTimeToEncounter(
const G4Track& trackA,
349 ->GetTimeToEncounter(trackA, trackB);
350 return timeToReaction;
355 fpReactionProcess = pReactionProcess;
361 fCheckedTracks.clear();
363 for (
auto pTrack : *fpTrackContainer->GetMainList()) {
364 if (pTrack ==
nullptr) {
366 msg <<
"No track found.";
367 G4Exception(
"G4DNAIndependentReactionTimeStepper::CalculateMinTimeStep",
381 if (sampledMinTimeStep < fTSTimeStep) {
382 fTSTimeStep = sampledMinTimeStep;
384 fReactionSet->AddReactions(fTSTimeStep,
const_cast<G4Track*
>(pTrack), std::move(reactants));
386 fSampledPositions[pTrack->GetTrackID()] = pTrack->GetPosition();
389 fSampledPositions[pTrackB->GetTrackID()] = pTrackB->GetPosition();
394 else if (fTSTimeStep == sampledMinTimeStep &&
G4bool(reactants)) {
395 fReactionSet->AddReactions(fTSTimeStep,
const_cast<G4Track*
>(pTrack), std::move(reactants));
397 fSampledPositions[pTrack->GetTrackID()] = pTrack->GetPosition();
400 fSampledPositions[pTrackB->GetTrackID()] = pTrackB->GetPosition();
404 else if (reactants) {
#define BuildChemicalMoleculeFinder()
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
std::multiset< G4ITReactionPtr, compReactionPerTime > G4ITReactionPerTime
G4shared_ptr< std::vector< G4Track * > > G4TrackVectorHandle
G4Molecule * GetMolecule(const G4Track &track)
G4GLOB_DLL std::ostream G4cout
G4double CalculateStep(const G4Track &, const G4double &) override
G4DNAIndependentReactionTimeStepper()
G4VDNAReactionModel * GetReactionModel()
void SetReactionModel(G4VDNAReactionModel *)
G4double CalculateMinTimeStep(G4double, G4double) override
std::unique_ptr< G4ITReactionChange > FindReaction(G4ITReactionSet *pReactionSet, const G4double ¤tStepTime=0, const G4double &previousStepTime=0, const G4bool &reachedUserStepTimeLimit=false)
void SetReactionProcess(G4VITReactionProcess *pReactionProcess)
static G4double GetRCutOff()
void SelectThisReaction(G4ITReactionPtr reaction)
G4ITReactionPerTime & GetReactionsPerTime()
const G4String & GetName() const override
static G4OctreeFinder * Instance()
void FindNearestInRange(const G4Track &track, const int &key, G4double R, std::vector< std::pair< typename CONTAINER::iterator, G4double > > &result, G4bool isSort=false) const
static G4Scheduler * Instance()
G4TrackStatus GetTrackStatus() const
void SetPosition(const G4ThreeVector &aValue)
const G4ThreeVector & GetPosition() const
G4TrackVectorHandle fReactants
G4TrackVectorHandle GetReactants()
virtual void ResetReactants()
static G4ThreadLocal G4double fUserMinTimeStep
G4double fSampledMinTimeStep