00001
00002
00003
00004
00005
00006
00007
00008
00009 #include <iostream>
00010 #include <cmath>
00011
00012 #include "boost/lexical_cast.hpp"
00013
00014 #include "FWCore/Framework/interface/Event.h"
00015 #include "FWCore/Framework/interface/Run.h"
00016 #include "FWCore/MessageLogger/interface/MessageLogger.h"
00017 #include "FWCore/ParameterSet/interface/ParameterSet.h"
00018 #include "FWCore/ServiceRegistry/interface/Service.h"
00019 #include "FWCore/Utilities/interface/RandomNumberGenerator.h"
00020 #include "FWCore/Utilities/interface/EDMException.h"
00021 #include "GeneratorInterface/Core/interface/RNDMEngineAccess.h"
00022
00023 #include "GeneratorInterface/HydjetInterface/interface/HydjetHadronizer.h"
00024 #include "GeneratorInterface/HydjetInterface/interface/HydjetWrapper.h"
00025 #include "GeneratorInterface/Pythia6Interface/interface/Pythia6Declarations.h"
00026 #include "GeneratorInterface/Pythia6Interface/interface/Pythia6Service.h"
00027
00028 #include "HepMC/PythiaWrapper6_4.h"
00029 #include "HepMC/GenEvent.h"
00030 #include "HepMC/HeavyIon.h"
00031 #include "HepMC/SimpleVector.h"
00032
00033 #include "SimDataFormats/GeneratorProducts/interface/HepMCProduct.h"
00034 #include "SimDataFormats/GeneratorProducts/interface/GenEventInfoProduct.h"
00035 #include "SimDataFormats/GeneratorProducts/interface/GenRunInfoProduct.h"
00036 #include "SimDataFormats/HiGenData/interface/GenHIEvent.h"
00037
00038 using namespace edm;
00039 using namespace std;
00040 using namespace gen;
00041
00042 namespace {
00043 int convertStatus(int st){
00044 if(st<= 0) return 0;
00045 if(st<=10) return 1;
00046 if(st<=20) return 2;
00047 if(st<=30) return 3;
00048 else return st;
00049 }
00050 }
00051
00052
00053
00054 HydjetHadronizer::HydjetHadronizer(const ParameterSet &pset) :
00055 BaseHadronizer(pset),
00056 evt(0),
00057 pset_(pset),
00058 abeamtarget_(pset.getParameter<double>("aBeamTarget")),
00059 bfixed_(pset.getParameter<double>("bFixed")),
00060 bmax_(pset.getParameter<double>("bMax")),
00061 bmin_(pset.getParameter<double>("bMin")),
00062 cflag_(pset.getParameter<int>("cFlag")),
00063 embedding_(pset.getParameter<bool>("embeddingMode")),
00064 comenergy(pset.getParameter<double>("comEnergy")),
00065 doradiativeenloss_(pset.getParameter<bool>("doRadiativeEnLoss")),
00066 docollisionalenloss_(pset.getParameter<bool>("doCollisionalEnLoss")),
00067 fracsoftmult_(pset.getParameter<double>("fracSoftMultiplicity")),
00068 hadfreeztemp_(pset.getParameter<double>("hadronFreezoutTemperature")),
00069 hymode_(pset.getParameter<string>("hydjetMode")),
00070 maxEventsToPrint_(pset.getUntrackedParameter<int>("maxEventsToPrint", 1)),
00071 maxlongy_(pset.getParameter<double>("maxLongitudinalRapidity")),
00072 maxtrany_(pset.getParameter<double>("maxTransverseRapidity")),
00073 nsub_(0),
00074 nhard_(0),
00075 nmultiplicity_(pset.getParameter<int>("nMultiplicity")),
00076 nsoft_(0),
00077 nquarkflavor_(pset.getParameter<int>("qgpNumQuarkFlavor")),
00078 pythiaPylistVerbosity_(pset.getUntrackedParameter<int>("pythiaPylistVerbosity", 0)),
00079 qgpt0_(pset.getParameter<double>("qgpInitialTemperature")),
00080 qgptau0_(pset.getParameter<double>("qgpProperTimeFormation")),
00081 phi0_(0.),
00082 sinphi0_(0.),
00083 cosphi0_(1.),
00084 rotate_(pset.getParameter<bool>("rotateEventPlane")),
00085 shadowingswitch_(pset.getParameter<int>("shadowingSwitch")),
00086 signn_(pset.getParameter<double>("sigmaInelNN")),
00087 pythia6Service_(new Pythia6Service(pset))
00088 {
00089
00090
00091
00092
00093 pythiaPylistVerbosity_ = pset.getUntrackedParameter<int>("pythiaPylistVerbosity",0);
00094 LogDebug("PYLISTverbosity") << "Pythia PYLIST verbosity level = " << pythiaPylistVerbosity_;
00095
00096
00097 maxEventsToPrint_ = pset.getUntrackedParameter<int>("maxEventsToPrint",0);
00098 LogDebug("Events2Print") << "Number of events to be printed = " << maxEventsToPrint_;
00099
00100 if(embedding_) src_ = pset.getParameter<edm::InputTag>("backgroundLabel");
00101
00102 }
00103
00104
00105
00106 HydjetHadronizer::~HydjetHadronizer()
00107 {
00108
00109 call_pystat(1);
00110 delete pythia6Service_;
00111 }
00112
00113
00114
00115 void HydjetHadronizer::add_heavy_ion_rec(HepMC::GenEvent *evt)
00116 {
00117
00118 double npart = hyfpar.npart;
00119 int nproj = static_cast<int>(npart / 2);
00120 int ntarg = static_cast<int>(npart - nproj);
00121
00122 HepMC::HeavyIon* hi = new HepMC::HeavyIon(
00123 nsub_,
00124 nproj,
00125 ntarg,
00126 static_cast<int>(hyfpar.nbcol),
00127 0,
00128 0,
00129 0,
00130 0,
00131 0,
00132 hyfpar.bgen * nuclear_radius(),
00133 phi0_,
00134 0,
00135 hyjpar.sigin
00136 );
00137
00138 evt->set_heavy_ion(*hi);
00139 delete hi;
00140 }
00141
00142
00143 HepMC::GenParticle* HydjetHadronizer::build_hyjet(int index, int barcode)
00144 {
00145
00146
00147 double x0 = hyjets.phj[0][index];
00148 double y0 = hyjets.phj[1][index];
00149
00150 double x = x0*cosphi0_-y0*sinphi0_;
00151 double y = y0*cosphi0_+x0*sinphi0_;
00152
00153 HepMC::GenParticle* p = new HepMC::GenParticle(
00154 HepMC::FourVector(x,
00155 y,
00156 hyjets.phj[2][index],
00157 hyjets.phj[3][index]),
00158 hyjets.khj[1][index],
00159 convertStatus(hyjets.khj[0][index]
00160 )
00161 );
00162
00163 p->suggest_barcode(barcode);
00164 return p;
00165 }
00166
00167
00168 HepMC::GenVertex* HydjetHadronizer::build_hyjet_vertex(int i,int id)
00169 {
00170
00171
00172 double x0=hyjets.vhj[0][i];
00173 double y0=hyjets.vhj[1][i];
00174 double x = x0*cosphi0_-y0*sinphi0_;
00175 double y = y0*cosphi0_+x0*sinphi0_;
00176 double z=hyjets.vhj[2][i];
00177 double t=hyjets.vhj[4][i];
00178
00179 HepMC::GenVertex* vertex = new HepMC::GenVertex(HepMC::FourVector(x,y,z,t),id);
00180 return vertex;
00181 }
00182
00183
00184
00185 bool HydjetHadronizer::generatePartonsAndHadronize()
00186 {
00187 Pythia6Service::InstanceWrapper guard(pythia6Service_);
00188
00189
00190 if(embedding_){
00191 cflag_ = 0;
00192 const edm::Event& e = getEDMEvent();
00193 Handle<HepMCProduct> input;
00194 e.getByLabel(src_,input);
00195 const HepMC::GenEvent * inev = input->GetEvent();
00196 const HepMC::HeavyIon* hi = inev->heavy_ion();
00197 if(hi){
00198 bfixed_ = hi->impact_parameter();
00199 phi0_ = hi->event_plane_angle();
00200 sinphi0_ = sin(phi0_);
00201 cosphi0_ = cos(phi0_);
00202 }else{
00203 LogWarning("EventEmbedding")<<"Background event does not have heavy ion record!";
00204 }
00205 }else if(rotate_) rotateEvtPlane();
00206
00207 nsoft_ = 0;
00208 nhard_ = 0;
00209
00210 edm::LogInfo("HYDJETmode") << "##### HYDJET nhsel = " << hyjpar.nhsel;
00211 edm::LogInfo("HYDJETfpart") << "##### HYDJET fpart = " << hyflow.fpart;
00212 edm::LogInfo("HYDJETtf") << "##### HYDJET hadron freez-out temp, Tf = " << hyflow.Tf;
00213 edm::LogInfo("HYDJETinTemp") << "##### HYDJET: QGP init temperature, T0 ="<<pyqpar.T0u;
00214 edm::LogInfo("HYDJETinTau") << "##### HYDJET: QGP formation time,tau0 ="<<pyqpar.tau0u;
00215
00216
00217 int ntry = 0;
00218 while(nsoft_ == 0 && nhard_ == 0){
00219 if(ntry > 100){
00220 edm::LogError("HydjetEmptyEvent") << "##### HYDJET: No Particles generated, Number of tries ="<<ntry;
00221
00222
00223
00224
00225 std::ostringstream sstr;
00226 sstr << "HydjetHadronizerProducer: No particles generated after " << ntry << " tries.\n";
00227 edm::Exception except(edm::errors::EventCorruption, sstr.str());
00228 throw except;
00229 } else {
00230 HYEVNT();
00231 nsoft_ = hyfpar.nhyd;
00232 nsub_ = hyjpar.njet;
00233 nhard_ = hyfpar.npyt;
00234 ++ntry;
00235 }
00236 }
00237
00238 if(hyjpar.nhsel < 3) nsub_++;
00239
00240
00241 HepMC::GenEvent *evt = new HepMC::GenEvent();
00242
00243 if(nhard_>0 || nsoft_>0) get_particles(evt);
00244
00245 evt->set_signal_process_id(pypars.msti[0]);
00246 evt->set_event_scale(pypars.pari[16]);
00247 add_heavy_ion_rec(evt);
00248
00249 event().reset(evt);
00250 return true;
00251 }
00252
00253
00254
00255 bool HydjetHadronizer::get_particles(HepMC::GenEvent *evt )
00256 {
00257
00258
00259
00260
00261
00262
00263
00264
00265 LogDebug("SubEvent")<< "Number of sub events "<<nsub_;
00266 LogDebug("Hydjet")<<"Number of hard events "<<hyjpar.njet;
00267 LogDebug("Hydjet")<<"Number of hard particles "<<nhard_;
00268 LogDebug("Hydjet")<<"Number of soft particles "<<nsoft_;
00269
00270 vector<HepMC::GenVertex*> sub_vertices(nsub_);
00271
00272 int ihy = 0;
00273 for(int isub=0;isub<nsub_;isub++){
00274 LogDebug("SubEvent") <<"Sub Event ID : "<<isub;
00275
00276 int sub_up = (isub+1)*50000;
00277 vector<HepMC::GenParticle*> particles;
00278 vector<int> mother_ids;
00279 vector<HepMC::GenVertex*> prods;
00280
00281 sub_vertices[isub] = new HepMC::GenVertex(HepMC::FourVector(0,0,0,0),isub);
00282 evt->add_vertex(sub_vertices[isub]);
00283 if(!evt->signal_process_vertex()) evt->set_signal_process_vertex(sub_vertices[isub]);
00284
00285 while(ihy<nhard_+nsoft_ && (hyjets.khj[2][ihy] < sub_up || ihy > nhard_ )){
00286 particles.push_back(build_hyjet(ihy,ihy+1));
00287 prods.push_back(build_hyjet_vertex(ihy,isub));
00288 mother_ids.push_back(hyjets.khj[2][ihy]);
00289 LogDebug("DecayChain")<<"Mother index : "<<hyjets.khj[2][ihy];
00290
00291 ihy++;
00292 }
00293
00294
00295
00296
00297 LogDebug("Hydjet")<<"Number of particles in vector "<<particles.size();
00298
00299 for (unsigned int i = 0; i<particles.size(); i++) {
00300 HepMC::GenParticle* part = particles[i];
00301
00302
00303
00304 int mid = mother_ids[i]-isub*50000-1;
00305 LogDebug("DecayChain")<<"Particle "<<i;
00306 LogDebug("DecayChain")<<"Mother's ID "<<mid;
00307 LogDebug("DecayChain")<<"Particle's PDG ID "<<part->pdg_id();
00308
00309 if(mid <= 0){
00310 sub_vertices[isub]->add_particle_out(part);
00311 continue;
00312 }
00313
00314 if(mid > 0){
00315 HepMC::GenParticle* mother = particles[mid];
00316 LogDebug("DecayChain")<<"Mother's PDG ID "<<mother->pdg_id();
00317
00318 HepMC::GenVertex* prod_vertex = mother->end_vertex();
00319 if(!prod_vertex){
00320 prod_vertex = prods[i];
00321 prod_vertex->add_particle_in(mother);
00322 evt->add_vertex(prod_vertex);
00323 prods[i]=0;
00324 }
00325 prod_vertex->add_particle_out(part);
00326 }
00327 }
00328
00329 for (unsigned int i = 0; i<prods.size(); i++) {
00330 if(prods[i]) delete prods[i];
00331 }
00332 }
00333 return true;
00334 }
00335
00336
00337
00338 bool HydjetHadronizer::call_hyinit(double energy,double a, int ifb, double bmin,
00339 double bmax,double bfix,int nh)
00340 {
00341
00342
00343 pydatr.mrpy[2]=1;
00344 HYINIT(energy,a,ifb,bmin,bmax,bfix,nh);
00345 return true;
00346 }
00347
00348
00349
00350 bool HydjetHadronizer::hydjet_init(const ParameterSet &pset)
00351 {
00352
00353
00354
00355
00356
00357
00358
00359
00360
00361 if(hymode_ == "kHydroOnly") hyjpar.nhsel=0;
00362 else if ( hymode_ == "kHydroJets") hyjpar.nhsel=1;
00363 else if ( hymode_ == "kHydroQJets") hyjpar.nhsel=2;
00364 else if ( hymode_ == "kJetsOnly") hyjpar.nhsel=3;
00365 else if ( hymode_ == "kQJetsOnly") hyjpar.nhsel=4;
00366 else hyjpar.nhsel=2;
00367
00368
00369 hyflow.fpart = fracsoftmult_;
00370
00371
00372 hyflow.Tf = hadfreeztemp_;
00373
00374
00375 hyflow.ylfl = maxlongy_;
00376
00377
00378 hyflow.ytfl = maxtrany_;
00379
00380
00381 hyjpar.ishad = shadowingswitch_;
00382
00383
00384 hyjpar.sigin = signn_;
00385
00386
00387 pyqpar.nfu = nquarkflavor_;
00388
00389
00390 pyqpar.T0u = qgpt0_;
00391
00392
00393 pyqpar.tau0u = qgptau0_;
00394
00395
00396 if( doradiativeenloss_ && docollisionalenloss_ ){
00397 edm::LogInfo("HydjetEnLoss") << "##### Radiative AND Collisional partonic energy loss ON ####";
00398 pyqpar.ienglu = 0;
00399 } else if ( doradiativeenloss_ ) {
00400 edm::LogInfo("HydjetenLoss") << "##### Only RADIATIVE partonic energy loss ON ####";
00401 pyqpar.ienglu = 1;
00402 } else if ( docollisionalenloss_ ) {
00403 edm::LogInfo("HydjetEnLoss") << "##### Only COLLISIONAL partonic energy loss ON ####";
00404 pyqpar.ienglu = 2;
00405 } else {
00406 edm::LogInfo("HydjetEnLoss") << "##### Radiative AND Collisional partonic energy loss ON ####";
00407 pyqpar.ienglu = 0;
00408 }
00409 return true;
00410 }
00411
00412
00413
00414 bool HydjetHadronizer::readSettings( int ) {
00415
00416 Pythia6Service::InstanceWrapper guard(pythia6Service_);
00417 pythia6Service_->setGeneralParams();
00418
00419 return true;
00420
00421 }
00422
00423
00424
00425 bool HydjetHadronizer::initializeForInternalPartons(){
00426
00427 Pythia6Service::InstanceWrapper guard(pythia6Service_);
00428
00429
00430
00431 const float ra = nuclear_radius();
00432 LogInfo("RAScaling")<<"Nuclear radius(RA) = "<<ra;
00433 bmin_ /= ra;
00434 bmax_ /= ra;
00435 bfixed_ /= ra;
00436
00437
00438 hydjet_init(pset_);
00439
00440 LogInfo("HYDJETinAction") << "##### Calling HYINIT("<<comenergy<<","<<abeamtarget_<<","
00441 <<cflag_<<","<<bmin_<<","<<bmax_<<","<<bfixed_<<","<<nmultiplicity_<<") ####";
00442 call_hyinit(comenergy,abeamtarget_,cflag_,bmin_,bmax_,bfixed_,nmultiplicity_);
00443 return true;
00444 }
00445
00446 bool HydjetHadronizer::declareStableParticles(const std::vector<int>& _pdg )
00447 {
00448 std::vector<int> pdg = _pdg;
00449 for ( size_t i=0; i < pdg.size(); i++ ) {
00450 int pyCode = pycomp_( pdg[i] );
00451 std::ostringstream pyCard ;
00452 pyCard << "MDCY(" << pyCode << ",1)=0";
00453 std::cout << pyCard.str() << std::endl;
00454 call_pygive( pyCard.str() );
00455 }
00456 return true;
00457 }
00458
00459
00460 void HydjetHadronizer::rotateEvtPlane()
00461 {
00462 const double pi = 3.14159265358979;
00463 phi0_ = 2.*pi*gen::pyr_(0) - pi;
00464 sinphi0_ = sin(phi0_);
00465 cosphi0_ = cos(phi0_);
00466 }
00467
00468
00469
00470 bool HydjetHadronizer::hadronize()
00471 {
00472 return false;
00473 }
00474
00475 bool HydjetHadronizer::decay()
00476 {
00477 return true;
00478 }
00479
00480 bool HydjetHadronizer::residualDecay()
00481 {
00482 return true;
00483 }
00484
00485 void HydjetHadronizer::finalizeEvent()
00486 {
00487 }
00488
00489 void HydjetHadronizer::statistics()
00490 {
00491 }
00492
00493 const char* HydjetHadronizer::classname() const
00494 {
00495 return "gen::HydjetHadronizer";
00496 }