Back to home page

EIC code displayed by LXR

 
 

    


Warning, /geant4/examples/extended/hadronic/Hadr09/Hadr09.cc-ION_PROJECTILE is written in an unsupported language. File is not indexed.

0001 //
0002 // ********************************************************************
0003 // * License and Disclaimer                                           *
0004 // *                                                                  *
0005 // * The  Geant4 software  is  copyright of the Copyright Holders  of *
0006 // * the Geant4 Collaboration.  It is provided  under  the terms  and *
0007 // * conditions of the Geant4 Software License,  included in the file *
0008 // * LICENSE and available at  http://cern.ch/geant4/license .  These *
0009 // * include a list of copyright holders.                             *
0010 // *                                                                  *
0011 // * Neither the authors of this software system, nor their employing *
0012 // * institutes,nor the agencies providing financial support for this *
0013 // * work  make  any representation or  warranty, express or implied, *
0014 // * regarding  this  software system or assume any liability for its *
0015 // * use.  Please see the license in the file  LICENSE  and URL above *
0016 // * for the full disclaimer and the limitation of liability.         *
0017 // *                                                                  *
0018 // * This  code  implementation is the result of  the  scientific and *
0019 // * technical work of the GEANT4 collaboration.                      *
0020 // * By using,  copying,  modifying or  distributing the software (or *
0021 // * any work based  on the software)  you  agree  to acknowledge its *
0022 // * use  in  resulting  scientific  publications,  and indicate your *
0023 // * acceptance of all terms of the Geant4 Software license.          *
0024 // ********************************************************************
0025 //
0026 //
0027 /// \file Hadr09.cc
0028 /// \brief Main program of the hadronic/Hadr09 example
0029 //
0030 //------------------------------------------------------------------------
0031 // This program shows how to use the class Hadronic Generator.
0032 // The class HadronicGenerator is a kind of "hadronic generator", i.e.
0033 // provides Geant4 final states (i.e. secondary particles) produced by
0034 // hadron-nuclear inelastic collisions.
0035 // Please see the class itself for more information.
0036 //
0037 // The use of the class Hadronic Generator is very simple:
0038 // the constructor needs to be invoked only once - specifying the name
0039 // of the Geant4 "physics case" to consider ("FTFP_BERT" will be
0040 // considered as default is the name is not specified) - and then one
0041 // method needs to be called at each collision, specifying the type of
0042 // collision (hadron, energy, direction, material) to be simulated.
0043 // The class HadronicGenerator is expected to work also in a
0044 // multi-threaded environment with "external" threads (i.e. threads
0045 // that are not necessarily managed by Geant4 run-manager):
0046 // each thread should have its own instance of the class.
0047 //
0048 // See the string "***LOOKHERE***" below for the setting of parameters
0049 // of this example: the "physics case", the set of possibilities from
0050 // which to sample the projectile (i.e. whether the projectile is a
0051 // hadron or an ion - in the case of hadron projectile, a list of hadrons
0052 // is possible from which to sample at each collision; in the case of
0053 // ion projectile, only one type of ion needs to be specified),
0054 // the kinetic energy of the projectile (which can be sampled within
0055 // an interval), whether the direction of the projectile is fixed or
0056 // sampled at each collision, the target material (a list of materials
0057 // is possible, from which the target material can be sampled at each
0058 // collision, and then from this target material, the target nucleus
0059 // will be chosen randomly by Geant4 itself), and whether to print out
0060 // some information or not and how frequently.
0061 // Once a well-defined type of hadron-nucleus or nucleus-nucleus
0062 // inelastic collision has been chosen, the method
0063 //   HadronicGenerator::GenerateInteraction
0064 // returns the secondaries produced by that interaction (in the form
0065 // of a G4VParticleChange object).
0066 // Some information about this final-state is printed out as an example.
0067 //
0068 // Usage:  Hadr09
0069 //------------------------------------------------------------------------
0070 
0071 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0072 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0073 
0074 #include <iomanip>
0075 #include "globals.hh"
0076 #include "G4ios.hh"
0077 #include "G4PhysicalConstants.hh"
0078 #include "G4SystemOfUnits.hh"
0079 #include "G4Material.hh"
0080 #include "G4NistManager.hh"
0081 #include "G4VParticleChange.hh"
0082 #include "G4UnitsTable.hh"
0083 #include "G4SystemOfUnits.hh"
0084 #include "HadronicGenerator.hh"
0085 #include "G4GenericIon.hh"
0086 #include "G4ProcessManager.hh"
0087 #include "G4ParticleTable.hh"
0088 #include "G4IonTable.hh"
0089 #include "CLHEP/Random/Randomize.h" 
0090 #include "CLHEP/Random/Ranlux64Engine.h" 
0091 
0092 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0093 
0094 int main( int , char** ) {
0095   
0096   G4cout << "=== Test of the HadronicGenerator ===" << G4endl;
0097 
0098   // See the HadronicGenerator class for the possibilities and meaning of the "physics cases".
0099   // ( In short, it is the name of the Geant4 hadronic model used for the simulation of
0100   //   the collision, with the possibility of having a transition between two models in
0101   //   a given energy interval, as in physics lists. )
0102   //const G4String namePhysics = "FTFP_BERT";        //***LOOKHERE***  PHYSICS CASE
0103   //const G4String namePhysics = "FTFP_BERT_ATL";
0104   //const G4String namePhysics = "QGSP_BERT";
0105   //const G4String namePhysics = "QGSP_BIC";
0106   //const G4String namePhysics = "FTFP_INCLXX";
0107   const G4String namePhysics = "FTFP";
0108   //const G4String namePhysics = "QGSP";
0109   //const G4String namePhysics = "BERT";
0110   //const G4String namePhysics = "BIC";
0111   //const G4String namePhysics = "IonBIC";
0112   //const G4String namePhysics = "INCL";
0113   
0114   // The kinetic energy of the projectile will be sampled randomly, with flat probability
0115   // in the interval [minEnergy, maxEnergy].
0116   G4double minEnergy =  1.0*CLHEP::GeV;  //***LOOKHERE***  HADRON PROJECTILE MIN Ekin
0117   G4double maxEnergy = 30.0*CLHEP::GeV;  //***LOOKHERE***  HADRON PROJECTILE MAX Ekin
0118   
0119   const G4int numCollisions = 1000;  //***LOOKHERE***  NUMBER OF COLLISIONS
0120 
0121   // Enable or disable the print out of this program: if enabled, the number of secondaries
0122   // produced in each collisions is printed out; moreover, once every "printingGap"
0123   // collisions, the list of secondaries is printed out.
0124   const G4bool isPrintingEnabled = true;           //***LOOKHERE***  PRINT OUT ON/OFF
0125   const G4int  printingGap = 100;                  //***LOOKHERE***  GAP IN PRINTING
0126   
0127   // Vector of Geant4 names of hadron projectiles: one of this will be sampled randomly
0128   // (with uniform probability) for each collision, when the projectile is not an ion.
0129   // Note: comment out the corresponding line in order to exclude a particle.
0130   std::vector< G4String > vecProjectiles;  //***LOOKHERE***  POSSIBLE HADRON PROJECTILES
0131   vecProjectiles.push_back( "pi-" );
0132   //Note: vecProjectiles.push_back( "pi0" );               // Excluded because too short-lived
0133   vecProjectiles.push_back( "pi+" );
0134   vecProjectiles.push_back( "kaon-" );
0135   vecProjectiles.push_back( "kaon+" );
0136   vecProjectiles.push_back( "kaon0L" );
0137   vecProjectiles.push_back( "kaon0S" );
0138   //Note: vecProjectiles.push_back( "eta" );               // Excluded because too short-lived
0139   //Note: vecProjectiles.push_back( "eta_prime" );         // Excluded because too short-lived
0140   vecProjectiles.push_back( "proton" );
0141   vecProjectiles.push_back( "neutron" );
0142   vecProjectiles.push_back( "deuteron" );
0143   vecProjectiles.push_back( "triton" );
0144   vecProjectiles.push_back( "He3" );
0145   vecProjectiles.push_back( "alpha" );
0146   vecProjectiles.push_back( "lambda" );
0147   vecProjectiles.push_back( "sigma-" );
0148   //Note: vecProjectiles.push_back( "sigma0" );            // Excluded because too short-lived
0149   vecProjectiles.push_back( "sigma+" );
0150   vecProjectiles.push_back( "xi-" );
0151   vecProjectiles.push_back( "xi0" );
0152   vecProjectiles.push_back( "omega-" );
0153   vecProjectiles.push_back( "anti_proton" );
0154   vecProjectiles.push_back( "anti_neutron" );
0155   vecProjectiles.push_back( "anti_lambda" );
0156   vecProjectiles.push_back( "anti_sigma-" );
0157   //Note: vecProjectiles.push_back( "anti_sigma0" );       // Excluded because too short-lived
0158   vecProjectiles.push_back( "anti_sigma+" );
0159   vecProjectiles.push_back( "anti_xi-" );
0160   vecProjectiles.push_back( "anti_xi0" );
0161   vecProjectiles.push_back( "anti_omega-" );
0162   vecProjectiles.push_back( "anti_deuteron" );
0163   vecProjectiles.push_back( "anti_triton" );
0164   vecProjectiles.push_back( "anti_He3" );
0165   vecProjectiles.push_back( "anti_alpha" );
0166   // Charm and bottom hadrons
0167   vecProjectiles.push_back( "D+" );
0168   vecProjectiles.push_back( "D-" );
0169   vecProjectiles.push_back( "D0" );
0170   vecProjectiles.push_back( "anti_D0" );
0171   vecProjectiles.push_back( "Ds+" );
0172   vecProjectiles.push_back( "Ds-" );
0173   //Note: vecProjectiles.push_back( "etac" );              // Excluded because too short-lived
0174   //Note: vecProjectiles.push_back( "J/psi" );             // Excluded because too short-lived
0175   vecProjectiles.push_back( "B+" );
0176   vecProjectiles.push_back( "B-" );
0177   vecProjectiles.push_back( "B0" );
0178   vecProjectiles.push_back( "anti_B0" );
0179   vecProjectiles.push_back( "Bs0" );
0180   vecProjectiles.push_back( "anti_Bs0" );
0181   vecProjectiles.push_back( "Bc+" );
0182   vecProjectiles.push_back( "Bc-" );
0183   //Note: vecProjectiles.push_back( "Upsilon" );           // Excluded because too short-lived
0184   vecProjectiles.push_back( "lambda_c+" );
0185   vecProjectiles.push_back( "anti_lambda_c+" );
0186   //Note: vecProjectiles.push_back( "sigma_c+" );          // Excluded because too short-lived
0187   //Note: vecProjectiles.push_back( "anti_sigma_c+" );     // Excluded because too short-lived
0188   //Note: vecProjectiles.push_back( "sigma_c0" );          // Excluded because too short-lived
0189   //Note: vecProjectiles.push_back( "anti_sigma_c0" );     // Excluded because too short-lived
0190   //Note: vecProjectiles.push_back( "sigma_c++" );         // Excluded because too short-lived
0191   //Note: vecProjectiles.push_back( "anti_sigma_c++" );    // Excluded because too short-lived
0192   vecProjectiles.push_back( "xi_c+" );
0193   vecProjectiles.push_back( "anti_xi_c+" );
0194   vecProjectiles.push_back( "xi_c0" );
0195   vecProjectiles.push_back( "anti_xi_c0" );
0196   vecProjectiles.push_back( "omega_c0" );
0197   vecProjectiles.push_back( "anti_omega_c0" );
0198   vecProjectiles.push_back( "lambda_b" );
0199   vecProjectiles.push_back( "anti_lambda_b" );
0200   //Note: vecProjectiles.push_back( "sigma_b+" );          // Excluded because too short-lived
0201   //Note: vecProjectiles.push_back( "anti_sigma_b+" );     // Excluded because too short-lived  
0202   //Note: vecProjectiles.push_back( "sigma_b0" );          // Excluded because too short-lived
0203   //Note: vecProjectiles.push_back( "sigma_b0" );          // Excluded because too short-lived
0204   //Note: vecProjectiles.push_back( "sigma_b-" );          // Excluded because too short-lived
0205   //Note: vecProjectiles.push_back( "anti_sigma_b-" );     // Excluded because too short-lived  
0206   vecProjectiles.push_back( "xi_b0" );
0207   vecProjectiles.push_back( "anti_xi_b0" );
0208   vecProjectiles.push_back( "xi_b-" );
0209   vecProjectiles.push_back( "anti_xi_b-" );
0210   vecProjectiles.push_back( "omega_b-" );
0211   vecProjectiles.push_back( "anti_omega_b-" );
0212 
0213   G4ParticleDefinition* projectileNucleus = nullptr;
0214   G4GenericIon* gion = G4GenericIon::GenericIon();
0215   gion->SetProcessManager( new G4ProcessManager( gion ) );
0216   G4ParticleTable* partTable = G4ParticleTable::GetParticleTable();
0217   G4IonTable* ions = partTable->GetIonTable();
0218   partTable->SetReadiness();
0219   ions->CreateAllIon();
0220   ions->CreateAllIsomer();
0221   
0222   const G4bool isProjectileIon = true;   //***LOOKHERE***  HADRON (false) OR ION (true) PROJECTILE ?
0223   if ( isProjectileIon ) {
0224     minEnergy = 40.0*13.0*CLHEP::GeV;    //***LOOKHERE***  ION PROJECTILE MIN Ekin
0225     maxEnergy = 40.0*13.0*CLHEP::GeV;    //***LOOKHERE***  ION PROJECTILE MAX Ekin
0226     G4int ionZ = 18, ionA = 40;          //***LOOKHERE***  ION PROJECTILE (Z, A)
0227     projectileNucleus = partTable->GetIonTable()->GetIon( ionZ, ionA, 0.0 );
0228   }
0229 
0230   // Vector of Geant4 NIST names of materials: one of this will be sampled randomly
0231   // (with uniform probability) for each collision and used as target material.
0232   // Note: comment out the corresponding line in order to exclude a material;
0233   //       or, vice versa, add a new line to extend the list with another material.
0234   std::vector< G4String > vecMaterials;  //***LOOKHERE*** : NIST TARGET MATERIALS
0235   //vecMaterials.push_back( "G4_H" );
0236   //vecMaterials.push_back( "G4_He" );
0237   //vecMaterials.push_back( "G4_Be" );
0238   //vecMaterials.push_back( "G4_C" );
0239   //vecMaterials.push_back( "G4_Al" );
0240   //vecMaterials.push_back( "G4_Si" );
0241   vecMaterials.push_back( "G4_Sc" );
0242   //vecMaterials.push_back( "G4_Ar" );
0243   //vecMaterials.push_back( "G4_Fe" );
0244   //vecMaterials.push_back( "G4_Cu" );
0245   //vecMaterials.push_back( "G4_W" );
0246   //vecMaterials.push_back( "G4_Pb" );
0247   
0248   const G4int numProjectiles = vecProjectiles.size();
0249   const G4int numMaterials = vecMaterials.size();
0250 
0251   G4cout << G4endl
0252          << "=================  Configuration ==================" << G4endl
0253          << "Model: " << namePhysics << G4endl
0254          << "Ekin: [ " << minEnergy/CLHEP::GeV << " , " << maxEnergy/CLHEP::GeV
0255          << " ] GeV" << G4endl
0256          << "Number of collisions:  " << numCollisions << G4endl
0257          << "Number of hadron projectiles: " << numProjectiles << G4endl
0258          << "Number of materials:   " << numMaterials   << G4endl
0259          << "IsIonProjectile: " << ( projectileNucleus != nullptr ? "true \t" : "false" )
0260          << ( projectileNucleus != nullptr ? projectileNucleus->GetParticleName() : "") << G4endl
0261          << "===================================================" << G4endl
0262          << G4endl;
0263   
0264   CLHEP::Ranlux64Engine defaultEngine( 1234567, 4 ); 
0265   CLHEP::HepRandom::setTheEngine( &defaultEngine ); 
0266   G4int seed = time( NULL ); 
0267   CLHEP::HepRandom::setTheSeed( seed ); 
0268   G4cout << G4endl << " Initial seed = " << seed << G4endl << G4endl; 
0269   
0270   // Instanciate the HadronicGenerator providing the name of the "physics case"
0271   HadronicGenerator* theHadronicGenerator = new HadronicGenerator( namePhysics );
0272   //****************************************************************************
0273   
0274   if ( theHadronicGenerator == nullptr ) {
0275     G4cerr << "ERROR: theHadronicGenerator is NULL !" << G4endl;
0276     return 1;
0277   } else if ( ! theHadronicGenerator->IsPhysicsCaseSupported() ) {
0278     G4cerr << "ERROR: this physics case is NOT supported !" << G4endl;
0279     return 2;
0280   }
0281   
0282   // Loop over the collisions
0283   G4double rnd1, rnd2, rnd3, rnd4, rnd5, rnd6, normalization, projectileEnergy;
0284   G4VParticleChange* aChange = nullptr;
0285   for ( G4int i = 0; i < numCollisions; ++i ) {
0286     // Draw some random numbers to select the hadron-nucleus interaction:
0287     // projectile hadron, projectile kinetic energy, projectile direction, and target material.
0288     rnd1 = CLHEP::HepRandom::getTheEngine()->flat(); 
0289     rnd2 = CLHEP::HepRandom::getTheEngine()->flat();
0290     rnd3 = CLHEP::HepRandom::getTheEngine()->flat();
0291     rnd4 = CLHEP::HepRandom::getTheEngine()->flat();
0292     rnd5 = CLHEP::HepRandom::getTheEngine()->flat();
0293     rnd6 = CLHEP::HepRandom::getTheEngine()->flat();
0294     // Sample the projectile kinetic energy
0295     projectileEnergy = minEnergy + rnd1*( maxEnergy - minEnergy );
0296     if ( projectileEnergy <= 0.0 ) projectileEnergy = minEnergy; 
0297     // Sample the projectile direction
0298     normalization = 1.0 / std::sqrt( rnd2*rnd2 + rnd3*rnd3 + rnd4*rnd4 );
0299     const G4bool isOnSmearingDirection = false;                 //***LOOKHERE***  IF true THEN SMEAR DIRECTION 
0300     G4ThreeVector aDirection = G4ThreeVector( 0.0, 0.0, 1.0 );  //***LOOKHERE***  ELSE USE THIS FIXED DIRECTION
0301     if ( isOnSmearingDirection ) {
0302       aDirection = G4ThreeVector( normalization*rnd2, normalization*rnd3, normalization*rnd4 );
0303     } 
0304     // Sample the projectile hadron from the vector vecProjectiles
0305     G4int index_projectile = std::trunc( rnd5*numProjectiles );
0306     G4String nameProjectile = vecProjectiles[ index_projectile ];
0307     G4ParticleDefinition* projectile = partTable->FindParticle( nameProjectile );
0308     if ( projectileNucleus ) {
0309       nameProjectile = projectileNucleus->GetParticleName();
0310       projectile = projectileNucleus;
0311     }
0312     // Sample the target material from the vector vecMaterials
0313     // (Note: the target nucleus will be sampled by Geant4)
0314     G4int index_material = std::trunc( rnd6*numMaterials );
0315     G4String nameMaterial = vecMaterials[ index_material ];
0316     G4Material* material = G4NistManager::Instance()->FindOrBuildMaterial( nameMaterial );
0317     if ( material == nullptr ) {
0318       G4cerr << "ERROR: Material " << nameMaterial << " is not found !" << G4endl;
0319       return 3;
0320     }
0321     if ( isPrintingEnabled ) {
0322       G4cout << "\t Collision " << i << " ; projectile=" << nameProjectile;
0323       if ( projectileNucleus ) {
0324         G4cout << " ; Ekin[MeV]/nucleon=" << projectileEnergy / 
0325           static_cast< G4double >( std::abs( projectileNucleus->GetBaryonNumber() ) );
0326       } else {
0327         G4cout << " ; Ekin[MeV]=" << projectileEnergy; 
0328       }
0329       G4cout << " ; direction=" << aDirection << " ; material=" << nameMaterial;
0330     }
0331     
0332     // Call here the "hadronic generator" to get the secondaries produced by the hadronic collision
0333     aChange = theHadronicGenerator->GenerateInteraction( projectile, projectileEnergy,
0334     /* ********************************************** */ aDirection, material );
0335     
0336     G4int nsec = aChange ? aChange->GetNumberOfSecondaries() : 0;
0337     G4bool isPrintingOfSecondariesEnabled = false;
0338     if ( isPrintingEnabled ) {
0339       G4cout << G4endl << "\t --> #secondaries=" << nsec 
0340              << " ; impactParameter[fm]=" << theHadronicGenerator->GetImpactParameter() / fermi
0341              << " ; #projectileSpectatorNucleons=" << theHadronicGenerator->GetNumberOfProjectileSpectatorNucleons()
0342              << " ; #targetSpectatorNucleons=" << theHadronicGenerator->GetNumberOfTargetSpectatorNucleons()
0343              << " ; #NNcollisions=" << theHadronicGenerator->GetNumberOfNNcollisions() << G4endl;
0344       if ( i % printingGap == 0 ) {
0345         isPrintingOfSecondariesEnabled = true;
0346         G4cout << "\t \t List of produced secondaries: " << G4endl;
0347       }
0348     }
0349     // Loop over produced secondaries and eventually print out some information.
0350     for ( G4int j = 0; j < nsec; ++j ) {
0351       const G4DynamicParticle* sec = aChange->GetSecondary(j)->GetDynamicParticle();
0352       if ( isPrintingOfSecondariesEnabled ) {
0353         G4cout << "\t \t \t j=" << j << "\t" << sec->GetDefinition()->GetParticleName()
0354                << "\t p=" << sec->Get4Momentum() << " MeV" << G4endl;
0355       }
0356       delete aChange->GetSecondary(j);
0357     }
0358     if ( aChange ) aChange->Clear();
0359   }
0360 
0361   G4cout << G4endl << " Final random number = " << CLHEP::HepRandom::getTheEngine()->flat()
0362          << G4endl << "=== End of test ===" << G4endl;
0363 }
0364 
0365 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......