Skip to content
Snippets Groups Projects
NPReaction.cxx 10.42 KiB
#include <iostream>
#include <fstream>
#include <string>
#include <cstdlib>
#include <sstream>
//#include <stdlib.h>
#include <cmath>
#include <vector>

#include "NPReaction.h"

// Use CLHEP System of unit and Physical Constant
#include "CLHEP/Units/GlobalSystemOfUnits.h"
#include "CLHEP/Units/PhysicalConstants.h"

using namespace NPL;


//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
Reaction::Reaction()
{
   //------------- Default Constructor -------------

   fNoy1       = new Nucleus();
   fNoy2       = new Nucleus();
   fNoy3       = new Nucleus();
   fNoy4       = new Nucleus();
   fBeamEnergy = 0;
   fThetaCM    = 0;
   fExcitation = 0;
   fQValue     = 0;
}

//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
Reaction::Reaction(string name1, string name2, string name3, string name4, double BeamEnergy, double ExcitationEnergy,string Path)
{
	SetEveryThing( name1, name2, name3, name4, BeamEnergy, ExcitationEnergy, Path) ;
}

//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
void Reaction::SetEveryThing(string name1, string name2, string name3, string name4, double BeamEnergy, double ExcitationEnergy,string Path)
{
   //------------- Constructor with nuclei names and beam energy ------------
     
   fNoy1       = new Nucleus(name1);
   fNoy2       = new Nucleus(name2);
   fNoy3       = new Nucleus(name3);
   fNoy4       = new Nucleus(name4);
   fBeamEnergy = BeamEnergy;
   fThetaCM    = 0;
   fExcitation = ExcitationEnergy;
   fQValue     = (  fNoy1->GetMassExcess() + fNoy2->GetMassExcess()
		  - fNoy3->GetMassExcess() - fNoy4->GetMassExcess()) / 1000;

   int masse  = fNoy1->GetA() + fNoy2->GetA() - fNoy3->GetA() - fNoy4->GetA();
   int charge = fNoy1->GetZ() + fNoy2->GetZ() - fNoy3->GetZ() - fNoy4->GetZ();
   if (masse || charge) cout << "Problem with charge or mass conservation" << endl;
   
   ///Read the differential cross section
   string GlobalPath = getenv("NPTOOL");
   Path = GlobalPath + "/Inputs/CrossSection/" + Path;
   ifstream CSFile;
   CSFile.open(Path.c_str());
   double CSBuffer,AngleBuffer;
   string echo ;
   if(CSFile.is_open()) { cout << "Reading Cross Section File " << Path << endl;}
   else {cout << "Cross Section File " << Path << " not found" << endl;return;}

	vector<double> CrossSectionBuffer ;
	
   while(!CSFile.eof())
   {
     CSFile >> AngleBuffer;
     CSFile >> CSBuffer;   
     double CSFinal = CSBuffer*sin(AngleBuffer*deg);
     CrossSectionBuffer.push_back(CSFinal);
   }
  
   CSFile.close();
   CrossSectionSize = CrossSectionBuffer.size();
   CrossSection = new double[CrossSectionSize] ;
   for(int i = 0 ; i <CrossSectionSize ; i++ )	CrossSection[i] = CrossSectionBuffer[i];
}

//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
Reaction::~Reaction()
{
   //------------- Default Destructor ------------

   delete fNoy1;
   delete fNoy2;
   delete fNoy3;
   delete fNoy4;
}


//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 

//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
void Reaction::KineRelativistic(double &ThetaLab3, double &EnergieLab3,
			        double &ThetaLab4, double &EnergieLab4) const
{
// 2-body relativistic kinematics: direct + inverse
// EnergieLab3,4 : lab energy in MeV of the 2 ejectiles
// ThetaLab3,4   : angles in rad

   double m1 = fNoy1->Mass();
   double m2 = fNoy2->Mass();
   double m3 = fNoy3->Mass();
   double m4 = fNoy4->Mass() + fExcitation;

   // center-of-mass velocity
   double WtotLab = (fBeamEnergy + m1) + m2;
   double P1 = sqrt(pow(fBeamEnergy,2) + 2*m1*fBeamEnergy);
   double B = P1 / WtotLab;
   double G = 1 / sqrt(1 - pow(B,2));

   // total energy of the ejectiles in the center-of-mass
   double W3cm = (pow(WtotLab,2) + pow(G,2)*(pow(m3,2) - pow(m4,2)))
   		   / (2 * G * WtotLab);
   double W4cm = (pow(WtotLab,2) + pow(G,2)*(pow(m4,2) - pow(m3,2)))
   		   / (2 * G * WtotLab);

   // velocity of the ejectiles in the center-of-mass
   double beta3cm  = sqrt(1 - pow(m3,2)/pow(W3cm,2));
   double beta4cm  = sqrt(1 - pow(m4,2)/pow(W4cm,2));
//   double gamma3cm = 1 / sqrt(1 - pow(beta3cm,2));
//   double gamma4cm = 1 / sqrt(1 - pow(beta4cm,2));

   // Constants of the kinematics
   double K3 = B / beta3cm;
   double K4 = B / beta4cm;
   
   // case of inverse kinematics
   double theta = fThetaCM;
   if (m1 > m2) theta = M_PI - fThetaCM;

   // lab angles
   ThetaLab3 = atan(sin(theta) / (cos(theta) + K3) / G);
   if (ThetaLab3 < 0) ThetaLab3 += M_PI;
   ThetaLab4 = atan(sin(M_PI + theta) / (cos(M_PI + theta) + K4) / G);
   if (fabs(ThetaLab3) < 1e-6) ThetaLab3 = 0;
   ThetaLab4 = fabs(ThetaLab4);
   if (fabs(ThetaLab4) < 1e-6) ThetaLab4 = 0;
   
   // total and kinetic energies in the lab
   double W3lab = W3cm * G * (1 + B*beta3cm*cos(theta));
   double W4lab = W4cm * G * (1 + B*beta4cm*cos(theta + M_PI));
   // test for total energy conversion
   if (fabs(WtotLab - (W3lab+W4lab)) > 1e-6) 
      cout << "Problem for energy conservation" << endl;
   EnergieLab3 = W3lab - m3;
   EnergieLab4 = W4lab - m4;
}


//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
double Reaction::ReconstructRelativistic(double EnergyLab, double ThetaLab) const
	{
		// EnergyLab in MeV
		// ThetaLab in rad

		double m1 = fNoy1->Mass()	;
		double m2 = fNoy2->Mass()	;
		double m3 = fNoy3->Mass()	;
		double m4 = fNoy4->Mass()	;	

		double P1 = sqrt(2*m1*fBeamEnergy+(fBeamEnergy*fBeamEnergy))	;
		double P3 = sqrt(2*m3*EnergyLab+(EnergyLab*EnergyLab))			;
		double P4 = sqrt(P1*P1+P3*P3-(2*P1*P3*cos(ThetaLab)))			;
		double E4 = fBeamEnergy+m1+m2-(EnergyLab+m3)					;
		double m4e = sqrt((E4*E4)-(P4*P4))								;
		double Eex= m4e-m4												;
		
		return Eex;
	}
	
/*double Reaction::ReconstructThetaHeavy(double EnergyLab, double ThetaLab)
	{
		// EnergyLab in MeV
		// ThetaLab in rad

		double m1 = fNoy1->Mass()	;
		double m2 = fNoy2->Mass()	;
		double m3 = fNoy3->Mass()	;
		double m4 = fNoy4->Mass()	;	

		double P1 = sqrt(2*m1*fBeamEnergy+(fBeamEnergy*fBeamEnergy))	;
		double P3 = sqrt(2*m3*EnergyLab+(EnergyLab*EnergyLab))			;
		double P4 = sqrt(P1*P1+P3*P3-(2*P1*P3*cos(ThetaLab)))			;
		
		
			double cosThetaHeavy = (P1 - P4cos(hetaLab))/P3	;

	}*/


//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
//Return ThetaCM									
double  Reaction::EnergyLabToThetaCM( double EnergyLab , double ExcitationEnergy = -500) const
	{
		if(ExcitationEnergy == -500) ExcitationEnergy = fExcitation; 
	
		double m1 = fNoy1->Mass()				;
		double m2 = fNoy2->Mass()				;
		double m3 = fNoy3->Mass()				;
		double m4 = (fNoy4->Mass()+ExcitationEnergy)	;
		double E1 = (fBeamEnergy+m1)			;
		double E3 = (EnergyLab+m3)				;
		  
		// Compute Mandelstan variable
		double s =  2*m2*E1 + m1*m1 + m2*m2	;
		double u = -2*m2*E3 + m2*m2 + m3*m3	;
		// Compute CM impulsion:
		//before reaction
		double P2CM =  ( sqrt(  ( s-(m1-m2)*(m1-m2) )*( s-(m1+m2)*(m1+m2) )  ) ) / (2*sqrt(s))	;
		// after reaction
		double P3CM =  ( sqrt(  ( s-(m3-m4)*(m3-m4) )*( s-(m3+m4)*(m3+m4) )  ) ) / (2*sqrt(s))	;

		// Compute CM Energy
		double E2CM = (s + m2*m2 -m1*m1)/(2*sqrt(s))			;
		double E3CM = (s + m3*m3 -m4*m4)/(2*sqrt(s))			;

		double u0 = m2*m2 + m3*m3 - 2*(E2CM*E3CM + P2CM*P3CM)	;

		double Pi = 3.141592654 	;
		double ThetaCM = Pi - acos (  1-(u-u0)/(2*P2CM*P3CM)  )	;

		return(ThetaCM);
	}

//....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo...... 
void Reaction::Print() const
	{
		// Print informations concerning the reaction

		cout << "Reaction : " << fNoy2->GetName() << "(" << fNoy1->GetName() 
			 << "," << fNoy3->GetName() << ")" << fNoy4->GetName() << "  @  " 
			 << fBeamEnergy << " MeV" 
		<< endl	;
		
		cout << "Exc = " << fExcitation << " MeV" << endl;
		cout << "Qgg = " << fQValue << " MeV" << endl;
	}
	
	
void Reaction::ReadConfigurationFile(string Path)
	{
	
	////////General Reading needs////////
	string LineBuffer;
	string DataBuffer;
	
////////Reaction Setting needs///////
string Beam,Target,Heavy,Light,CrossSectionPath;
double BeamEnergy,ExcitationEnergy;
Reaction* myReaction;

//////////////////////////////////////////////////////////////////////////////////////////
	ifstream ReactionFile;
	string GlobalPath = getenv("NPTOOL");
   	Path = GlobalPath + "/Inputs/Reaction/" + Path;
	
	ReactionFile.open(Path.c_str());
	
	if( ReactionFile.is_open() ) 
	cout << " Reaction file " << Path << " loading " << endl;
	else{
	cout << " Error, no Reaction file" << Path << " found" << endl; }
	
	
	int i=0;
	while( !ReactionFile.eof() )	
	{
		//Pick-up next line
		getline(ReactionFile, LineBuffer); i++;

		//Search for comment Symbol %
		if(LineBuffer.compare(0,1,"%")==0) {;}
		
		else if(LineBuffer.compare(0,9,"Transfert")==0)  
		{
		
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,5,"Beam=")==0) 
				{
				ReactionFile >> DataBuffer;
				Beam = DataBuffer;
				cout << "Beam " << Beam << endl;
				}
			
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,7,"Target=")==0) 
				{
				ReactionFile >> DataBuffer;
				Target = DataBuffer;
				cout << "Target " << Target << endl;
				}
				
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,6,"Light=")==0) 
				{
				ReactionFile >> DataBuffer;
				Light = DataBuffer;
				cout << "Light " << Light << endl;
				}
			
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,6,"Heavy=")==0) 
				{
				ReactionFile >> DataBuffer;
				Heavy = DataBuffer;
				cout << "Heavy " << Heavy << endl;
				}
			
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,17,"ExcitationEnergy=")==0) 
				{
				ReactionFile >> DataBuffer;
				ExcitationEnergy = atof(DataBuffer.c_str());
				cout << "ExcitationEnergy " << ExcitationEnergy << " MeV" << endl;
				}
				
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,11,"BeamEnergy=")==0) 
				{
				ReactionFile >> DataBuffer;
				BeamEnergy = atof(DataBuffer.c_str());
				cout << "EnergyBeam " << BeamEnergy << " MeV" << endl;
				}
				
			ReactionFile >> DataBuffer;ReactionFile >> DataBuffer;
			ReactionFile >> DataBuffer;ReactionFile >> DataBuffer;
			ReactionFile >> DataBuffer;ReactionFile >> DataBuffer;
			ReactionFile >> DataBuffer;ReactionFile >> DataBuffer;
		
				
			ReactionFile >> DataBuffer;
			if(DataBuffer.compare(0,17,"CrossSectionPath=")==0) 
				{
				ReactionFile >> CrossSectionPath;
				cout << "CrossSectionPath " << CrossSectionPath << endl;
				}
		}
	}
	
SetEveryThing(Beam, Target, Light, Heavy,BeamEnergy,ExcitationEnergy,CrossSectionPath);

	}