282{
284
285 const G4double physStepLimit = currentMinimumStep;
286
287 switch (fType) {
289
290 G4VMscModel* mscModel = nullptr;
294 if (physStepLimit > kGeomMin) {
296 G4double pdgMass = particleDefinition->GetPDGMass();
297 if (pdgMass > CLHEP::GeV) {
298 ekinForSelection *= proton_mass_c2 / pdgMass;
299 }
300
301 if (ekinForSelection >= kLowestKinEnergy) {
302 mscModel = static_cast<G4VMscModel*>(
303 fModelManager->SelectModel(ekinForSelection, couple->GetIndex()));
304 if (mscModel == nullptr) {
306 "no MSC model found");
307 }
308 if (!mscModel->
IsActive(ekinForSelection)) {
309 mscModel = nullptr;
310 }
311 }
312 }
313
314
315
316 if (mscModel != nullptr) {
318
319
320 const G4Track* currentTrackPtr = &track;
321
322 G4double currentSafety = proposedSafety;
324
325 G4double stepLimitLeft = physStepLimit;
326 G4double totalGeometryStepLength = 0, totalTruePathLength = 0;
327 G4bool firstStep =
true, continueStepping = fMultipleSteps;
328
329 do {
330 G4double gPathLength = stepLimitLeft;
333 G4bool mscLimitsStep = (tPathLength < stepLimitLeft);
334 if (!fMultipleSteps && mscLimitsStep) {
335
337 }
338
339 if (!firstStep) {
340
342 }
343
346 *currentTrackPtr, previousStepSize, gPathLength, currentSafety, &transportSelection);
347 if (geometryStepLength < gPathLength) {
348
350 continueStepping = false;
351 }
353
354
355 continueStepping = false;
356 }
357
358 if (firstStep) {
359 proposedSafety = currentSafety;
360 }
361 totalGeometryStepLength += geometryStepLength;
362
363
364 const G4double range = mscModel->
GetRange(particleDefinition, currentEnergy, couple);
365
367
368
369 tPathLength = std::min(tPathLength, stepLimitLeft);
370
371 totalTruePathLength += tPathLength;
373
374
375 continueStepping = false;
376 }
377 else if (tPathLength >= range) {
378
379 continueStepping = false;
380 }
381 else {
382 stepLimitLeft -= tPathLength;
383 }
384
385
386 if (tPathLength < range && tPathLength > kGeomMin) {
387 static constexpr G4double minSafety = 1.20 * CLHEP::nm;
388 static constexpr G4double sFact = 0.99;
389
390
391
392
394
397
399
401 if (r2 > kMinDisplacement2) {
402 G4bool positionChanged =
true;
406
407
408 if (postSafety > 0.0 && dispR <= postSafety) {
410
411
412 }
413 else {
414
415 if (dispR < postSafety) {
417
418
419 }
420 else if (postSafety > kGeomMin) {
422
423
424 }
425 else {
426 positionChanged = false;
427 }
428 }
429 if (positionChanged) {
431 }
432 }
433 }
434
435 if (continueStepping) {
436
438 currentSafety = 0;
439 }
440 else {
442 }
443
444
445 currentEnergy = mscModel->
GetEnergy(particleDefinition, range - tPathLength, couple);
446
447
448 currentTrackPtr = fSubStepTrack;
449
453
454 G4StepPoint& subPreStepPoint = *fSubStep->GetPreStepPoint();
457 subPreStepPoint.
SetSafety(currentSafety);
459 }
460 firstStep = false;
461 } while (continueStepping);
462
463
464
465
466
467 if (currentEnergy != ekin) {
468
469
470
472
473
474
475 (void)mscModel->
GetRange(particleDefinition, ekin, couple);
476 }
477
479
480
481
482
483
485
486 return totalGeometryStepLength;
487 }
488 break;
489 }
490
492
496
499 if (pdgMass > CLHEP::GeV) {
500 ekinForSelection *= proton_mass_c2 / pdgMass;
501 }
502
503 G4VEmModel* currentModel = fModelManager->SelectModel(ekinForSelection, couple->GetIndex());
504 if (currentModel == nullptr) {
506 "no scattering model found");
507 }
508 if (!currentModel->
IsActive(ekinForSelection)) {
509 currentModel = nullptr;
510 }
511
512 if (currentModel != nullptr) {
514 G4int coupleIndex = couple->GetIndex();
515
516
518 G4double lambda = ((*fLambdaTable)[coupleIndex])->LogVectorValue(ekin, logEkin);
519 if (lambda > 0.0) {
520
521
523 G4double dedx = fIonisation->GetDEDX(ekin, couple);
524
525 G4double currentSafety = proposedSafety;
527
528
529 const G4Track* currentTrackPtr = &track;
530
531 G4double stepLimitLeft = physStepLimit;
533 G4bool firstStep =
true, continueStepping = fMultipleSteps;
534
535 do {
537
538 G4bool ssLimitsStep = (interactionLength < stepLimitLeft);
539 G4double gPathLength = stepLimitLeft;
540 if (ssLimitsStep) {
541 if (!fMultipleSteps) {
542
544 }
545 gPathLength = interactionLength;
546 }
547
548 if (!firstStep) {
549
551 }
552
555 *currentTrackPtr, previousStepSize, gPathLength, currentSafety, &transportSelection);
556 if (geometryStepLength < gPathLength) {
557
559 ssLimitsStep = false;
560 continueStepping = false;
561 }
563
564
565 continueStepping = false;
566 }
567
568 if (firstStep) {
569 proposedSafety = currentSafety;
570 }
571 totalStepLength += geometryStepLength;
572
574
575
576 continueStepping = false;
577 }
578 else {
579 stepLimitLeft -= geometryStepLength;
580 }
581
582
585
586 if (ssLimitsStep) {
587 fSubStepDynamicParticle->SetKineticEnergy(energyAfterLinearLoss);
588
589
590
591
593
595 (*fCuts)[coupleIndex]);
596
598
599
600
601
602 if (fSecondariesSS->size() > 0) {
604 "scattering model created secondaries");
605 }
606 if (fParticleChangeForSS->GetLocalEnergyDeposit() > 0) {
608 "scattering model proposed energy deposit");
609 }
610 }
611
612 if (continueStepping) {
613
615 currentSafety = 0;
616 }
617 else {
619 }
620
621
622 currentEnergy = energyAfterLinearLoss;
623
624
625 currentTrackPtr = fSubStepTrack;
626
630
631 G4StepPoint& subPreStepPoint = *fSubStep->GetPreStepPoint();
634 subPreStepPoint.
SetSafety(currentSafety);
636 }
637 firstStep = false;
638 } while (continueStepping);
639
640
641
642
643
644 if (currentEnergy != ekin) {
645
646
647
649 }
650
652
653
654
656
657 return totalStepLength;
658 }
659 }
660 break;
661 }
662 }
663
664
666 track, previousStepSize, currentMinimumStep, proposedSafety, selection);
667}
@ NotCandidateForSelection
G4double G4Log(G4double x)
G4double GetLogKineticEnergy() const
G4double GetPDGMass() const
void SetSafety(const G4double aValue)
void SetStepStatus(const G4StepStatus aValue)
void SetMaterialCutsCouple(const G4MaterialCutsCouple *)
void SetPosition(const G4ThreeVector &aValue)
const G4ParticleDefinition * GetParticleDefinition() const
const G4DynamicParticle * GetDynamicParticle() const
G4double GetKineticEnergy() const
void SetKineticEnergy(const G4double aValue)
const G4MaterialCutsCouple * GetMaterialCutsCouple() const
G4double AlongStepGetPhysicalInteractionLength(const G4Track &track, G4double previousStepSize, G4double currentMinimumStep, G4double ¤tSafety, G4GPILSelection *selection)
G4double fTransportEndKineticEnergy
G4ThreeVector fTransportEndPosition
G4ParticleChangeForTransport fParticleChange
G4double fEndPointDistance
G4Navigator * fLinearNavigator
G4ThreeVector fTransportEndMomentumDir
G4SafetyHelper * fpSafetyHelper
void SetCurrentCouple(const G4MaterialCutsCouple *)
G4bool IsActive(G4double kinEnergy) const
virtual void SampleSecondaries(std::vector< G4DynamicParticle * > *, const G4MaterialCutsCouple *, const G4DynamicParticle *, G4double tmin=0.0, G4double tmax=DBL_MAX)=0
virtual G4double ComputeTruePathLengthLimit(const G4Track &track, G4double &stepLimit)=0
virtual G4double ComputeTrueStepLength(G4double geomPathLength)=0
G4double GetEnergy(const G4ParticleDefinition *part, G4double range, const G4MaterialCutsCouple *couple)
G4double GetRange(const G4ParticleDefinition *part, G4double kineticEnergy, const G4MaterialCutsCouple *couple)
virtual G4ThreeVector & SampleScattering(const G4ThreeVector &, G4double safety)=0