187 if((isIonTrack && ((trackZ<=0 && trackL==0) || trackA<=trackZ)) ||
188 (nucleusA>1 && (nucleusZ<=0 || nucleusA<=nucleusZ))) {
190 theResult.SetStatusChange(
isAlive);
198 if(trackA<=1 && nucleusA<=1 && (trackZ>=0 || trackA==0)) {
199 return theBackupModelNucleon->ApplyYourself(aTrack, theNucleus);
204 const G4int theMaxProjMassINCL = theInterfaceStore->GetMaxProjMassINCL();
205 if(trackA > theMaxProjMassINCL && nucleusA > theMaxProjMassINCL) {
206 if(!complainedAboutBackupModel) {
207 complainedAboutBackupModel =
true;
208 std::stringstream ss;
209 ss <<
"INCL++ refuses to handle reactions between nuclei with A>"
210 << theMaxProjMassINCL
211 <<
". A backup model ("
212 << theBackupModel->GetModelName()
213 <<
") will be used instead.";
214 theInterfaceStore->EmitBigWarning(ss.str());
216 return theBackupModel->ApplyYourself(aTrack, theNucleus);
220 const G4double cascadeMinEnergyPerNucleon = theInterfaceStore->GetCascadeMinEnergyPerNucleon();
223 && trackKinE < cascadeMinEnergyPerNucleon) {
224 if(!complainedAboutPreCompound) {
225 complainedAboutPreCompound =
true;
226 std::stringstream ss;
227 ss <<
"INCL++ refuses to handle nucleon-induced reactions below "
228 << cascadeMinEnergyPerNucleon / MeV
229 <<
" MeV. A PreCoumpound model ("
230 << thePreCompoundModel->GetModelName()
231 <<
") will be used instead.";
232 theInterfaceStore->EmitBigWarning(ss.str());
234 return thePreCompoundModel->ApplyYourself(aTrack, theNucleus);
238 const G4double theNucleusMass = theIonTable->GetIonMass(nucleusZ, nucleusA);
240 const G4double theTrackEnergy = trackKinE + theTrackMass;
241 const G4double theTrackMomentumAbs2 = theTrackEnergy*theTrackEnergy - theTrackMass*theTrackMass;
242 const G4double theTrackMomentumAbs = ((theTrackMomentumAbs2>0.0) ? std::sqrt(theTrackMomentumAbs2) : 0.0);
247 fourMomentumIn.
setE(theTrackEnergy + theNucleusMass);
248 fourMomentumIn.
setVect(theTrackMomentum);
251 const G4bool inverseKinematics = AccurateProjectile(aTrack, theNucleus);
257 G4Nucleus *theTargetNucleus = &theNucleus;
258 if(inverseKinematics) {
262 if(oldProjectileDef != 0 && oldTargetDef != 0) {
266 if(newTargetA > 0 && newTargetZ > 0) {
268 theTargetNucleus =
new G4Nucleus(newTargetA, newTargetZ, newTargetL);
271 G4DynamicParticle swappedProjectileParticle(oldTargetDef, (*toInverseKinematics) * theProjectile4Momentum);
274 G4String message =
"badly defined target after swapping. Falling back to normal (non-swapped) mode.";
275 theInterfaceStore->EmitWarning(message);
279 G4String message =
"oldProjectileDef or oldTargetDef was null";
280 theInterfaceStore->EmitWarning(message);
315 std::list<G4Fragment> remnants;
317 const G4int maxTries = 200;
325 const G4double kineticEnergy = toINCLKineticEnergy(*aProjectileTrack);
330 const G4INCL::EventInfo eventInfo = theINCLModel->processEvent(theSpecies, kineticEnergy,
333 -theTargetNucleus->
GetL());
340 if(inverseKinematics) {
360 momentum *= toLabFrame;
362 if(inverseKinematics) {
363 momentum *= *toDirectKinematics;
369 fourMomentumOut += momentum;
380 theResult.AddSecondary(secondary);
383 G4String message =
"the model produced a particle that couldn't be converted to Geant4 particle.";
384 theInterfaceStore->EmitWarning(message);
393 if(( Z == 0 &&
S == 0 &&
A > 1 ) ||
394 ( Z == 0 &&
S != 0 &&
A < 4 ) ||
395 ( Z != 0 &&
S != 0 &&
A == Z + std::abs(
S) )) {
396 std::stringstream ss;
397 ss <<
"unphysical residual fragment : Z=" << Z <<
" S=" <<
S <<
" A=" <<
A
398 <<
" skipping it and resampling the collision";
399 theInterfaceStore->EmitWarning(ss.str());
408 eventInfo.
jxRem[i]*hbar_Planck,
409 eventInfo.
jyRem[i]*hbar_Planck,
410 eventInfo.
jzRem[i]*hbar_Planck
420 const G4double scaling = remnant4MomentumScaling(nuclearMass, kinE, px, py, pz);
423 if(std::abs(scaling - 1.0) > 0.01) {
424 std::stringstream ss;
425 ss <<
"momentum scaling = " << scaling
426 <<
"\n Lorentz vector = " << fourMomentum
427 <<
")\n A = " <<
A <<
", Z = " << Z <<
", S = " <<
S
428 <<
"\n E* = " << excitationE <<
", nuclearMass = " << nuclearMass
429 <<
"\n remnant i=" << i <<
", nRemnants=" << eventInfo.
nRemnants
433 <<
", in " << (inverseKinematics ?
"inverse" :
"direct") <<
" kinematics.";
434 theInterfaceStore->EmitWarning(ss.str());
438 fourMomentum *= toLabFrame;
441 if(inverseKinematics) {
442 fourMomentum *= *toDirectKinematics;
446 fourMomentumOut += fourMomentum;
450 if(dumpRemnantInfo) {
451 G4cerr <<
"G4INCLXX_DUMP_REMNANT: " << remnant <<
" spin: " << spin <<
G4endl;
453 remnants.push_back(remnant);
458 const G4int nSecondaries = (
G4int)theResult.GetNumberOfSecondaries();
459 for(
G4int j=0; j<nSecondaries; ++j)
delete theResult.GetSecondary(j)->GetParticle();
465 const G4LorentzVector violation4Momentum = fourMomentumOut - fourMomentumIn;
466 const G4double energyViolation = std::abs(violation4Momentum.
e());
467 const G4double momentumViolation = violation4Momentum.
rho();
469 std::stringstream ss;
470 ss <<
"energy conservation violated by " << energyViolation/MeV <<
" MeV in "
473 <<
" inelastic reaction, in " << (inverseKinematics ?
"inverse" :
"direct") <<
" kinematics. Will resample.";
474 theInterfaceStore->EmitWarning(ss.str());
476 const G4int nSecondaries = (
G4int)theResult.GetNumberOfSecondaries();
477 for(
G4int j=0; j<nSecondaries; ++j)
delete theResult.GetSecondary(j)->GetParticle();
482 std::stringstream ss;
483 ss <<
"momentum conservation violated by " << momentumViolation/MeV <<
" MeV in "
486 <<
" inelastic reaction, in " << (inverseKinematics ?
"inverse" :
"direct") <<
" kinematics. Will resample.";
487 theInterfaceStore->EmitWarning(ss.str());
489 const G4int nSecondaries = (
G4int)theResult.GetNumberOfSecondaries();
490 for(
G4int j=0; j<nSecondaries; ++j)
delete theResult.GetSecondary(j)->GetParticle();
498 }
while(!eventIsOK && nTries < maxTries);
501 if(inverseKinematics) {
502 delete aProjectileTrack;
503 delete theTargetNucleus;
504 delete toInverseKinematics;
505 delete toDirectKinematics;
509 std::stringstream ss;
510 ss <<
"maximum number of tries exceeded for the proposed "
512 <<
" + " << theIonTable->GetIonName(nucleusZ, nucleusA, 0)
513 <<
" inelastic reaction, in " << (inverseKinematics ?
"inverse" :
"direct") <<
" kinematics.";
514 theInterfaceStore->EmitWarning(ss.str());
515 theResult.SetStatusChange(
isAlive);
524 for(std::list<G4Fragment>::iterator i = remnants.begin();
525 i != remnants.end(); ++i) {
528 for(G4ReactionProductVector::iterator fragment = deExcitationResult->begin();
529 fragment != deExcitationResult->end(); ++fragment) {
533 theResult.AddSecondary(theFragment, (*fragment)->GetCreatorModelID());
537 for(G4ReactionProductVector::iterator fragment = deExcitationResult->begin();
538 fragment != deExcitationResult->end(); ++fragment) {
541 deExcitationResult->clear();
542 delete deExcitationResult;
548 if((theTally = theInterfaceStore->GetTally()))
549 theTally->Tally(aTrack, theNucleus, theResult);