140 if (
nullptr == fFirstParticle) {
141 fFirstParticle = ∂
143 auto emConfigurator = fEmManager->EmConfigurator();
144 emConfigurator->PrepareModels(&part, ptr,
this);
147 if (fFirstParticle == &part) {
148 G4bool master = fEmManager->IsMaster();
160 const G4int numberOfModels = fModelManager->NumberOfModels();
162 for (
G4int i = 0; i < numberOfModels; ++i) {
163 auto msc =
static_cast<G4VMscModel*
>(fModelManager->GetModel(i));
165 G4double emax = std::min(msc->HighEnergyLimit(), theParameters->MaxKinEnergy());
166 msc->SetHighEnergyLimit(emax);
167 msc->SetUseBaseMaterials(baseMat);
172 if (fEmData ==
nullptr) {
176 fLambdaTable = fEmData->MakeTable(0);
189 if (fFirstParticle == &part) {
190 fEmManager->BuildPhysicsTable(fFirstParticle);
192 if (fEmManager->IsMaster()) {
198 std::size_t numOfCouples = theCoupleTable->
GetTableSize();
200 G4double emin = theParameters->MinKinEnergy();
201 G4double emax = theParameters->MaxKinEnergy();
204 G4int nbin = theParameters->NumberOfBinsPerDecade() *
G4lrint(std::log10(scale));
205 scale = nbin /
G4Log(scale);
208 bin = std::max(bin, 5);
210 for (std::size_t i = 0; i < numOfCouples; ++i) {
211 if (!bld->
GetFlag(i))
continue;
217 fModelManager->FillLambdaVector(aVector, couple,
false);
218 aVector->FillSecondDerivatives();
227 const G4int numberOfModels = fModelManager->NumberOfModels();
229 for (
G4int i = 0; i < numberOfModels; ++i) {
230 auto msc =
static_cast<G4VMscModel*
>(fModelManager->GetModel(i));
231 auto msc0 =
static_cast<G4VMscModel*
>(masterProcess->fModelManager->GetModel(i));
233 msc->InitialiseLocal(fFirstParticle, msc0);
237 this->fLambdaTable = masterProcess->fLambdaTable;
245 if (fMultipleSteps) {
246 G4cout <<
" (multipleSteps: 1)";
285 const G4double physStepLimit = currentMinimumStep;
294 if (physStepLimit > kGeomMin) {
296 G4double pdgMass = particleDefinition->GetPDGMass();
297 if (pdgMass > CLHEP::GeV) {
298 ekinForSelection *= proton_mass_c2 / pdgMass;
301 if (ekinForSelection >= kLowestKinEnergy) {
303 fModelManager->SelectModel(ekinForSelection, couple->GetIndex()));
304 if (mscModel ==
nullptr) {
306 "no MSC model found");
308 if (!mscModel->
IsActive(ekinForSelection)) {
316 if (mscModel !=
nullptr) {
320 const G4Track* currentTrackPtr = &track;
322 G4double currentSafety = proposedSafety;
325 G4double stepLimitLeft = physStepLimit;
326 G4double totalGeometryStepLength = 0, totalTruePathLength = 0;
327 G4bool firstStep =
true, continueStepping = fMultipleSteps;
330 G4double gPathLength = stepLimitLeft;
333 G4bool mscLimitsStep = (tPathLength < stepLimitLeft);
334 if (!fMultipleSteps && mscLimitsStep) {
346 *currentTrackPtr, previousStepSize, gPathLength, currentSafety, &transportSelection);
347 if (geometryStepLength < gPathLength) {
350 continueStepping =
false;
355 continueStepping =
false;
359 proposedSafety = currentSafety;
361 totalGeometryStepLength += geometryStepLength;
364 const G4double range = mscModel->
GetRange(particleDefinition, currentEnergy, couple);
369 tPathLength = std::min(tPathLength, stepLimitLeft);
371 totalTruePathLength += tPathLength;
375 continueStepping =
false;
377 else if (tPathLength >= range) {
379 continueStepping =
false;
382 stepLimitLeft -= tPathLength;
386 if (tPathLength < range && tPathLength > kGeomMin) {
387 static constexpr G4double minSafety = 1.20 * CLHEP::nm;
388 static constexpr G4double sFact = 0.99;
401 if (r2 > kMinDisplacement2) {
402 G4bool positionChanged =
true;
408 if (postSafety > 0.0 && dispR <= postSafety) {
415 if (dispR < postSafety) {
420 else if (postSafety > kGeomMin) {
426 positionChanged =
false;
429 if (positionChanged) {
435 if (continueStepping) {
445 currentEnergy = mscModel->
GetEnergy(particleDefinition, range - tPathLength, couple);
448 currentTrackPtr = fSubStepTrack;
454 G4StepPoint& subPreStepPoint = *fSubStep->GetPreStepPoint();
457 subPreStepPoint.
SetSafety(currentSafety);
461 }
while (continueStepping);
467 if (currentEnergy != ekin) {
475 (void)mscModel->
GetRange(particleDefinition, ekin, couple);
486 return totalGeometryStepLength;
498 G4double pdgMass = particleDefinition->GetPDGMass();
499 if (pdgMass > CLHEP::GeV) {
500 ekinForSelection *= proton_mass_c2 / pdgMass;
503 G4VEmModel* currentModel = fModelManager->SelectModel(ekinForSelection, couple->GetIndex());
504 if (currentModel ==
nullptr) {
506 "no scattering model found");
508 if (!currentModel->
IsActive(ekinForSelection)) {
509 currentModel =
nullptr;
512 if (currentModel !=
nullptr) {
514 G4int coupleIndex = couple->GetIndex();
518 G4double lambda = ((*fLambdaTable)[coupleIndex])->LogVectorValue(ekin, logEkin);
522 G4double meanFreePath = 1.0 / lambda;
523 G4double dedx = fIonisation->GetDEDX(ekin, couple);
525 G4double currentSafety = proposedSafety;
529 const G4Track* currentTrackPtr = &track;
531 G4double stepLimitLeft = physStepLimit;
533 G4bool firstStep =
true, continueStepping = fMultipleSteps;
538 G4bool ssLimitsStep = (interactionLength < stepLimitLeft);
539 G4double gPathLength = stepLimitLeft;
541 if (!fMultipleSteps) {
545 gPathLength = interactionLength;
555 *currentTrackPtr, previousStepSize, gPathLength, currentSafety, &transportSelection);
556 if (geometryStepLength < gPathLength) {
559 ssLimitsStep =
false;
560 continueStepping =
false;
565 continueStepping =
false;
569 proposedSafety = currentSafety;
571 totalStepLength += geometryStepLength;
576 continueStepping =
false;
579 stepLimitLeft -= geometryStepLength;
587 fSubStepDynamicParticle->SetKineticEnergy(energyAfterLinearLoss);
595 (*fCuts)[coupleIndex]);
602 if (fSecondariesSS->size() > 0) {
604 "scattering model created secondaries");
606 if (fParticleChangeForSS->GetLocalEnergyDeposit() > 0) {
608 "scattering model proposed energy deposit");
612 if (continueStepping) {
622 currentEnergy = energyAfterLinearLoss;
625 currentTrackPtr = fSubStepTrack;
631 G4StepPoint& subPreStepPoint = *fSubStep->GetPreStepPoint();
634 subPreStepPoint.
SetSafety(currentSafety);
638 }
while (continueStepping);
644 if (currentEnergy != ekin) {
657 return totalStepLength;
666 track, previousStepSize, currentMinimumStep, proposedSafety, selection);