diff --git a/.github/workflows/smash_test_completes.yaml b/.github/workflows/smash_test_completes.yaml
new file mode 100644
index 00000000..f9ac42a8
--- /dev/null
+++ b/.github/workflows/smash_test_completes.yaml
@@ -0,0 +1,67 @@
+name: smash test completes
+
+on:
+ pull_request:
+ branches:
+ - main
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
+
+ push:
+ branches:
+ - main
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
+
+env:
+ REPO_NAME: ${{ github.event.repository.name }}
+
+jobs:
+ build:
+ name: build for smash test
+ runs-on: ubuntu-latest
+
+ container:
+ image: jetscape/base:stable
+ options: --user root
+
+ steps:
+
+ - name: Checkout Repository
+ uses: actions/checkout@v4
+ with:
+ path: ${{ github.event.repository.name }}
+
+ - name: Download MUSIC
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
+ ./get_music.sh
+
+ - name: Download ISS
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
+ ./get_iSS.sh
+
+ - name: Download FREESTREAM
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
+ ./get_freestream-milne.sh
+
+ - name: Download SMASH
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
+ ./get_smash.sh
+
+ - name: Build Application
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}
+ mkdir build
+ cd build
+ export SMASH_DIR="${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages/smash/smash_code"
+ cmake .. -DUSE_MUSIC=ON -DUSE_ISS=ON -DUSE_FREESTREAM=ON -DUSE_SMASH=ON
+ make -j6
+
+ - name: Run Application
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/build
+ ./runJetscape ../config/jetscape_user_test_smash_RC1_1.xml
diff --git a/.github/workflows/test-build-external.yaml b/.github/workflows/test-build-external.yaml
index 663b3c9b..fab2fe1a 100644
--- a/.github/workflows/test-build-external.yaml
+++ b/.github/workflows/test-build-external.yaml
@@ -4,14 +4,14 @@ on:
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +22,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -52,11 +52,16 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
./get_smash.sh
+ - name: Download 3DGlauber
+ run: |
+ cd ${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages
+ ./get_3dglauber.sh
+
- name: Build Application
run: |
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
export SMASH_DIR="${GITHUB_WORKSPACE}/${REPO_NAME}/external_packages/smash/smash_code"
- cmake .. -DCMAKE_CXX_STANDARD=14 -DUSE_MUSIC=ON -DUSE_ISS=ON -DUSE_FREESTREAM=ON -DUSE_SMASH=ON
+ cmake .. -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON -DUSE_FREESTREAM=ON -DUSE_SMASH=ON
make -j2
diff --git a/.github/workflows/test-events-PbPb.yaml b/.github/workflows/test-events-PbPb.yaml
index 942e9a5b..92e83520 100644
--- a/.github/workflows/test-events-PbPb.yaml
+++ b/.github/workflows/test-events-PbPb.yaml
@@ -4,14 +4,14 @@ on:
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +22,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -37,11 +37,11 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
- cmake .. -DCMAKE_CXX_STANDARD=14
+ cmake ..
make -j2
- name: Checkout TEST-EXAMPLES
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
repository: JETSCAPE/TEST-EXAMPLES
path: TEST-EXAMPLES
diff --git a/.github/workflows/test-events-Pythia-Isr-Dat.yaml b/.github/workflows/test-events-Pythia-Isr-Dat.yaml
index e2ecd4a5..7b946eaa 100644
--- a/.github/workflows/test-events-Pythia-Isr-Dat.yaml
+++ b/.github/workflows/test-events-Pythia-Isr-Dat.yaml
@@ -1,17 +1,19 @@
name: build+regression test Pythia-ISR Dat Output
on:
+ workflow_dispatch:
+
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +24,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -44,11 +46,11 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
- cmake .. -DCMAKE_CXX_STANDARD=14 -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
+ cmake .. -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
make -j2
- name: Checkout TEST-EXAMPLES
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
repository: JETSCAPE/TEST-EXAMPLES
path: TEST-EXAMPLES
diff --git a/.github/workflows/test-events-Pythia-Isr-Hadron.yaml b/.github/workflows/test-events-Pythia-Isr-Hadron.yaml
index 07f7696f..7340dc1b 100644
--- a/.github/workflows/test-events-Pythia-Isr-Hadron.yaml
+++ b/.github/workflows/test-events-Pythia-Isr-Hadron.yaml
@@ -4,14 +4,14 @@ on:
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +22,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -44,11 +44,11 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
- cmake .. -DCMAKE_CXX_STANDARD=14 -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
+ cmake .. -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
make -j2
- name: Checkout TEST-EXAMPLES
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
repository: JETSCAPE/TEST-EXAMPLES
path: TEST-EXAMPLES
diff --git a/.github/workflows/test-events-Pythia-Isr-Parton.yaml b/.github/workflows/test-events-Pythia-Isr-Parton.yaml
index 7888f559..4c698d8a 100644
--- a/.github/workflows/test-events-Pythia-Isr-Parton.yaml
+++ b/.github/workflows/test-events-Pythia-Isr-Parton.yaml
@@ -4,14 +4,14 @@ on:
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +22,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -44,11 +44,11 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
- cmake .. -DCMAKE_CXX_STANDARD=14 -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
+ cmake .. -DUSE_3DGlauber=ON -DUSE_MUSIC=ON -DUSE_ISS=ON
make -j2
- name: Checkout TEST-EXAMPLES
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
repository: JETSCAPE/TEST-EXAMPLES
path: TEST-EXAMPLES
diff --git a/.github/workflows/test-events-pp.yaml b/.github/workflows/test-events-pp.yaml
index 1903ef5f..06844456 100644
--- a/.github/workflows/test-events-pp.yaml
+++ b/.github/workflows/test-events-pp.yaml
@@ -4,14 +4,14 @@ on:
pull_request:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
push:
branches:
- main
- - XSCAPE-RC2
- - XSCAPE-1.0-for-public-repo
+ - XSCAPE-1.1-RC
+ - XSCAPE-1.1-RC_mergeMUSIC
env:
REPO_NAME: ${{ github.event.repository.name }}
@@ -22,13 +22,13 @@ jobs:
runs-on: ubuntu-latest
container:
- image: jetscape/base:v1.8
+ image: jetscape/base:stable
options: --user root
steps:
- name: Checkout Repository
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
path: ${{ github.event.repository.name }}
@@ -37,11 +37,11 @@ jobs:
cd ${GITHUB_WORKSPACE}/${REPO_NAME}
mkdir build
cd build
- cmake .. -DCMAKE_CXX_STANDARD=14
+ cmake ..
make -j2
- name: Checkout TEST-EXAMPLES
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
with:
repository: JETSCAPE/TEST-EXAMPLES
path: TEST-EXAMPLES
diff --git a/CMakeLists.txt b/CMakeLists.txt
index df52f9ed..b5912a18 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -102,7 +102,7 @@ endif (USE_SMASH)
### Compiler & Linker Flags ###
###############################
message("Compiler and Linker flags ...")
-set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fPIC -pipe -Wall -std=c++11")
+set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fPIC -pipe -Wall -std=c++17")
## can turn off some warnings
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wno-reorder -Wno-unused-variable -Wno-inconsistent-missing-override -Wno-unused-private-field")
## Then set the build type specific options. These will be automatically appended.
@@ -480,12 +480,19 @@ target_link_libraries(FinalStatePartons JetScape )
add_executable(PythiaBrickTest ./examples/custom_examples/PythiaBrickTest.cc)
target_link_libraries(PythiaBrickTest JetScape )
-if(USE_3DGlauber AND USE_MUSIC AND USE_ISS)
- add_executable(PythiaIsrTest ./examples/PythiaIsrTest.cc)
- target_link_libraries(PythiaIsrTest JetScape )
- add_executable(PythiaIsrMUSIC ./examples/custom_examples/PythiaIsrMUSIC.cc)
- target_link_libraries(PythiaIsrMUSIC JetScape )
-endif (USE_3DGlauber AND USE_MUSIC AND USE_ISS )
+
+if(USE_MUSIC AND USE_ISS)
+ add_executable(MUSICMainClockTest ./examples/custom_examples/MUSICMainClockTest.cc)
+ target_link_libraries(MUSICMainClockTest JetScape )
+ add_executable(MUSICTest ./examples/custom_examples/MUSICTest.cc)
+ target_link_libraries(MUSICTest JetScape )
+ if(USE_3DGlauber)
+ add_executable(PythiaIsrTest ./examples/PythiaIsrTest.cc)
+ target_link_libraries(PythiaIsrTest JetScape )
+ add_executable(PythiaIsrMUSIC ./examples/custom_examples/PythiaIsrMUSIC.cc)
+ target_link_libraries(PythiaIsrMUSIC JetScape )
+ endif (USE_3DGlauber)
+endif (USE_MUSIC AND USE_ISS )
#add_executable(PythiaBDMTest ./examples/custom_examples/PythiaBDMTest.cc)
#target_link_libraries(PythiaBDMTest JetScape )
diff --git a/README.md b/README.md
index 7d29694b..02ff6790 100644
--- a/README.md
+++ b/README.md
@@ -1,4 +1,4 @@
-# X-SCAPE 1.0
+# X-SCAPE 1.1
The X-ion collisions with a Statistically and Computationally Advanced Program Envelope (X-SCAPE) is the enhanced (and 2nd) project of the JETSCAPE
collaboration which extends the framework to include small systems created in p-A and p-p collisions, lower energy heavy-ion collisions and electron-Ion collisions.
@@ -16,22 +16,7 @@ Please cite [The JETSCAPE framework](https://arxiv.org/abs/1903.07706) if you us
## Installation
-Note that X-SCAPE dependencies require cmake to be called specifying the language standard C++14 or greater.
-
-For example, to compile X-SCAPE for the ISR run with [3DGlauber support](#3dglauber-support), run the get_3dglauber.sh script from the external_packages folder:
-```bash
-./get_3dglauber.sh
-```
-Then from the build folder, call cmake with the C++14 and 3DGlauber flags:
-```bash
-mkdir build
-cd build
-cmake .. -DCMAKE_CXX_STANDARD=14 -DUSE_3DGlauber=ON
-make -j4 # Builds using 4 cores; adapt as appropriate
-```
-If installing X-SCAPE with SMASH, the [get_smash.sh](https://github.com/JETSCAPE/X-SCAPE-COMP/blob/main/external_packages/get_smash.sh) script will clone an [alternate branch](https://github.com/smash-transport/smash/tree/staudenmaier/mod_plist_ts) of SMASH currently required for use with X-SCAPE. A future release of SMASH will bring X-SCAPE support to SMASH's main branch.
-
-Please see the complete [Installation Instructions](https://github.com/JETSCAPE/X-SCAPE/wiki/Doc.Installation) here.
+Please see the [Installation Instructions](https://github.com/JETSCAPE/X-SCAPE/wiki/Doc.Installation) here.
## Running X-SCAPE/JETSCAPE
@@ -116,7 +101,7 @@ To compile the 3DGlauber code together with the JETSCAPE framework, please turn
```bash
mkdir build
cd build
- cmake -DCMAKE_CXX_STANDARD=14 -DUSE_3DGlauber=ON ..
+ cmake -DUSE_3DGlauber=ON ..
make -j4
```
diff --git a/config/HH_LightExcitedMesons.xml b/config/HH_LightExcitedMesons.xml
new file mode 100644
index 00000000..8f5c7395
--- /dev/null
+++ b/config/HH_LightExcitedMesons.xml
@@ -0,0 +1,2014 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/config/jetscape_main.xml b/config/jetscape_main.xml
index adee6052..1acdda88 100644
--- a/config/jetscape_main.xml
+++ b/config/jetscape_main.xml
@@ -105,7 +105,6 @@
PGun
100
21
- 0
@@ -116,13 +115,24 @@
100
120
5020
- 0
+
+ 2.0
HardQCD:all = on
+
+
+ epemgun
+ 5020
+
+
+
+ HadronLevel::all = off
+
+
@@ -138,7 +148,6 @@
Matter
- 0
1
2.0
0.16
@@ -222,6 +231,22 @@
+
+
+
+
+ CausalLiquefier
+ 0.02
+ 0.3
+ 0.3
+ 0.2
+ 2.0
+ 0.1
+ 0.08
+ 0.1
+
+
+
@@ -324,17 +349,33 @@
colored/colorless/hybrid
1
- 2510
- 3
+
+
+ 5020
+ 1
+ 1
+ 0
5.0
- 0.3333333
- 0.3333333
- on
+ 1.0
+ 1.0
+ 0
+ 0.16
+ 2.0
+ 0.0
+ 1.0
+ on
+ 10.0
+ depracted
+ 1
+
+ 0
+
+ ../config/HH_LightExcitedMesons.xml
@@ -365,6 +406,7 @@
+
0
diff --git a/config/jetscape_user.xml b/config/jetscape_user.xml
index 5e853f0d..3ab1b191 100644
--- a/config/jetscape_user.xml
+++ b/config/jetscape_user.xml
@@ -26,9 +26,12 @@
-
+
+
+ MUSIC
+ 42
+
+
diff --git a/config/jetscape_user_FSFileTest.xml b/config/jetscape_user_FSFileTest.xml
index 42bc62ce..6beff8c0 100644
--- a/config/jetscape_user_FSFileTest.xml
+++ b/config/jetscape_user_FSFileTest.xml
@@ -26,7 +26,6 @@
PGun
100
- 0
diff --git a/config/jetscape_user_HydroFile.xml b/config/jetscape_user_HydroFile.xml
index 39600f12..1d326f6b 100644
--- a/config/jetscape_user_HydroFile.xml
+++ b/config/jetscape_user_HydroFile.xml
@@ -26,7 +26,6 @@
PGun
100
- 0
diff --git a/config/jetscape_user_MUSIC.xml b/config/jetscape_user_MUSIC.xml
new file mode 100644
index 00000000..512f361c
--- /dev/null
+++ b/config/jetscape_user_MUSIC.xml
@@ -0,0 +1,84 @@
+
+
+
+
+ 0
+ 1
+
+ on
+
+
+
+ 15
+ 15
+ 0
+ 0.2
+ 0.2
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 20
+ 30
+ 200
+
+
+
+
+
+
+
+
+
+
+
+ 1
+ 0.6
+ 0.15
+ 0.10
+
+
+
+
+
+
+ 0
+
+
+
+
+
+ 1
+ 1
+
+
+
+
+
+ colorless
+
+
+
diff --git a/config/jetscape_user_MUSICMainClockTest.xml b/config/jetscape_user_MUSICMainClockTest.xml
new file mode 100644
index 00000000..92537789
--- /dev/null
+++ b/config/jetscape_user_MUSICMainClockTest.xml
@@ -0,0 +1,103 @@
+
+
+
+
+ 1
+ false
+ 1
+ 5
+ false
+
+
+ on
+
+
+
+ 15
+ 15
+ 3
+ 0.3
+ 0.3
+ 0.2
+
+
+
+
+
+
+
+
+
+ PGun
+ 100
+ 0
+
+
+
+
+
+
+
+
+
+
+
+ MUSIC
+ 42
+
+
+
+
+
+ 0.1
+ -0.1
+ 250
+ ON
+
+
+ Matter
+ 1
+ 2.0
+ 0.16
+ 0.25
+ 0
+ 1
+ 0
+ 0
+ 0.16
+
+
+
+
+
+
+ 1
+ -2.0
+ 0.25
+
+
+
+ Lbt
+ 2.0
+ 0
+ 0
+ 0.16
+ 0.25
+ 1
+
+
+
+
+
+ colorless
+ 1
+ 2510
+
+
+
+
+
+
+
+
diff --git a/config/jetscape_user_brick_hybrid_hadronization.xml b/config/jetscape_user_brick_hybrid_hadronization.xml
new file mode 100644
index 00000000..8c0c5205
--- /dev/null
+++ b/config/jetscape_user_brick_hybrid_hadronization.xml
@@ -0,0 +1,79 @@
+
+
+ 10
+ 0
+ true
+ 10
+ 1
+
+ test_out
+ off
+ off
+ on
+
+
+ 42
+
+
+
+
+
+
+ PGun
+ 100
+ 1
+
+
+
+
+
+ 20
+ 0.6
+
+
+
+ Brick
+ 0.3
+
+
+
+
+
+
+ 100
+
+
+ 0
+ 1
+ 0
+ 4.0
+ 0.16
+ 2.0
+ 0.25
+
+
+
+ Lbt
+ 2.0
+ 0
+ 0
+ 0.16
+ 0.2
+ 1
+
+
+
+
+
+ hybrid
+ 0.1
+ 0.0
+ on
+ 10.0
+
+
+ 0
+ 1
+ 0
+
+
diff --git a/config/jetscape_user_iMATTERMCGlauber.xml b/config/jetscape_user_iMATTERMCGlauber.xml
index e4aa6b44..97fb401d 100644
--- a/config/jetscape_user_iMATTERMCGlauber.xml
+++ b/config/jetscape_user_iMATTERMCGlauber.xml
@@ -42,7 +42,6 @@
SigmaProcess:alphaSvalue = 0.13
MultipartonInteractions:alphaSvalue = 0.13
- 1
@@ -62,7 +61,6 @@
2.0
1
- 1
0.25
0
0
diff --git a/config/jetscape_user_iMATTERMCGlauberMUSIC.xml b/config/jetscape_user_iMATTERMCGlauberMUSIC.xml
index a127bf7a..721f412b 100644
--- a/config/jetscape_user_iMATTERMCGlauberMUSIC.xml
+++ b/config/jetscape_user_iMATTERMCGlauberMUSIC.xml
@@ -31,7 +31,6 @@
MultipartonInteractions:alphaSvalue = 0.16
MultipartonInteractions:pTmin = 7.0
- 1
diff --git a/config/jetscape_user_pbpb_grid.xml b/config/jetscape_user_pbpb_grid.xml
index 10ed923e..7a09f97d 100644
--- a/config/jetscape_user_pbpb_grid.xml
+++ b/config/jetscape_user_pbpb_grid.xml
@@ -1,11 +1,11 @@
-
+
5
true
10
-
+
on
test_out
0
@@ -19,7 +19,7 @@
../examples/test_hydro_files
-
+
@@ -30,24 +30,23 @@
-
+
HardQCD:all = on
PDF:useHardNPDFA=on
PDF:useHardNPDFB=on
- PDF:nPDFSetA=1
+ PDF:nPDFSetA=1
PDF:nPDFSetB=1
PDF:nPDFBeamA = 100822080
PDF:nPDFBeamB = 100822080
- 0
-
+
-
+
@@ -57,7 +56,7 @@
../examples/test_hydro_files
-
+
@@ -108,7 +107,7 @@
1
-
+
colorless
@@ -118,5 +117,5 @@
111:mayDecay = off
-
+
diff --git a/config/jetscape_user_test_smash_RC1_1.xml b/config/jetscape_user_test_smash_RC1_1.xml
new file mode 100644
index 00000000..b1e53f56
--- /dev/null
+++ b/config/jetscape_user_test_smash_RC1_1.xml
@@ -0,0 +1,81 @@
+
+
+
+
+ 0
+ 2
+
+ on
+
+
+
+ 15
+ 15
+ 0
+ 0.2
+ 0.2
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 1
+ 42
+ 0.6
+ 0.15
+ 0.10
+
+
+
+
+
+ 1
+ 0
+
+
+
+
+
+
+ 0
+
+
+ 1
+
+
+ SMASH
+
+ 300.0
+
+ 0
+
+
+
+
diff --git a/config/jetscape_user_twostagehydro.xml b/config/jetscape_user_twostagehydro.xml
index 9b1e7394..b295bd40 100644
--- a/config/jetscape_user_twostagehydro.xml
+++ b/config/jetscape_user_twostagehydro.xml
@@ -6,7 +6,7 @@
true
5
0
-
+
on
@@ -20,7 +20,6 @@
PGun
100
- 0
@@ -35,7 +34,7 @@
MUSIC_1
-
+
@@ -51,7 +50,7 @@
0.1
-
+
0.1
@@ -92,7 +91,7 @@
true
-
+
@@ -100,7 +99,7 @@
true
-
+
colorless
@@ -112,5 +111,5 @@
-
+
diff --git a/docker/Dockerfile.base b/docker/Dockerfile.base
index ae54b384..cc56c679 100644
--- a/docker/Dockerfile.base
+++ b/docker/Dockerfile.base
@@ -1,5 +1,6 @@
# Build from the official docker python base image, based on Debian
-FROM debian:stable
+FROM ubuntu:focal
+ARG DEBIAN_FRONTEND=noninteractive
# Install pre-reqs (commented ones are already in base image)
RUN apt update && apt install -y \
@@ -28,6 +29,7 @@ tclsh \
tcl-dev \
libxft-dev \
libxext-dev \
+libfftw3-3 \
#zlib1g-dev \
&& rm -rf /var/lib/apt/lists/*
@@ -72,22 +74,22 @@ RUN mkdir -p ${ROOTSYS} && mkdir -p ${HOME}/root && cd ${HOME}/root \
&& make -j8 install \
&& rm -r ${HOME}/root
-# Install HepMC 3.2.5
-RUN curl -SL http://hepmc.web.cern.ch/hepmc/releases/HepMC3-3.2.5.tar.gz | tar -xvzC /usr/local \
+# Install HepMC 3.2.6
+RUN curl -SL http://hepmc.web.cern.ch/hepmc/releases/HepMC3-3.2.6.tar.gz | tar -xvzC /usr/local \
&& cd /usr/local \
&& mkdir hepmc3-build \
&& cd hepmc3-build \
-&& cmake ../HepMC3-3.2.5 -DCMAKE_INSTALL_PREFIX=/usr/local -DHEPMC3_ENABLE_ROOTIO=ON -DROOT_DIR=${ROOTSYS} -DHEPMC3_BUILD_EXAMPLES=ON \
+&& cmake ../HepMC3-3.2.6 -DCMAKE_INSTALL_PREFIX=/usr/local -DHEPMC3_ENABLE_ROOTIO=ON -DROOT_DIR=${ROOTSYS} -DHEPMC3_BUILD_EXAMPLES=ON \
&& make -j8 install \
&& rm -r /usr/local/hepmc3-build
ENV LD_LIBRARY_PATH="/usr/local/lib:${LD_LIBRARY_PATH}"
-# Install Pythia8 with HepMC3-3.2.5 (that is needed by SMASH)
-ARG pythiaV="8307"
+# Install Pythia8 with HepMC3-3.2.6 (that is needed by SMASH)
+ARG pythiaV="8309"
RUN curl -SLk http://pythia.org/download/pythia83/pythia${pythiaV}.tgz \
| tar -xvzC /usr/local \
&& cd /usr/local/pythia${pythiaV} \
-&& ./configure --enable-shared --prefix=/usr/local/ --with-hepmc3=/usr/local/HepMC3-3.2.5 \
+&& ./configure --enable-shared --prefix=/usr/local/ --with-hepmc3=/usr/local/HepMC3-3.2.6 \
&& make -j8 \
&& make -j8 install
@@ -103,7 +105,7 @@ ENV PATH $PATH:$PYTHIA8DIR/bin
# Build heppy (various HEP tools via python)
RUN cd / \
-&& git clone https://github.com/matplo/heppy.git \
+&& git clone --depth 1 --branch v1.3.8.8 https://github.com/matplo/heppy.git \
&& cd heppy \
&& ./external/fastjet/build.sh \
&& ./external/hepmc2/build.sh \
diff --git a/examples/custom_examples/MUSICMainClockTest.cc b/examples/custom_examples/MUSICMainClockTest.cc
new file mode 100644
index 00000000..c2d5a5e7
--- /dev/null
+++ b/examples/custom_examples/MUSICMainClockTest.cc
@@ -0,0 +1,356 @@
+/*******************************************************************************
+ * Copyright (c) The JETSCAPE Collaboration, 2018
+ *
+ * Modular, task-based framework for simulating all aspects of heavy-ion collisions
+ *
+ * For the list of contributors see AUTHORS.
+ *
+ * Report issues at https://github.com/JETSCAPE/JETSCAPE/issues
+ *
+ * or via email to bugs.jetscape@gmail.com
+ *
+ * Distributed under the GNU General Public License 3.0 (GPLv3 or later).
+ * See COPYING for details.
+ ******************************************************************************/
+// -------------------------------------------------
+// XSCAPE Framework Clock Pythia Brick Test Program
+// -------------------------------------------------
+
+#include
+#include
+
+// JetScape Framework includes ...
+#include "JetScape.h"
+#include "JetEnergyLoss.h"
+#include "JetEnergyLossManager.h"
+#include "JetScapeWriterStream.h"
+#ifdef USE_HEPMC
+#include "JetScapeWriterHepMC.h"
+#endif
+
+
+// User modules derived from jetscape framework clasess
+#include "TrentoInitial.h"
+#include "AdSCFT.h"
+#include "Matter.h"
+#include "LBT.h"
+#include "Martini.h"
+#include "NullPreDynamics.h"
+#include "MusicWrapper.h"
+#include "iSpectraSamplerWrapper.h"
+#include "PythiaGun.h"
+#include "InitialStateRadiationTest.h"
+#include "HadronizationManager.h"
+#include "Hadronization.h"
+#include "ColoredHadronization.h"
+#include "ColorlessHadronization.h"
+#include "CascadeTest.h"
+
+#include "MainClock.h"
+#include "ModuleClock.h"
+#include "MilneClock.h"
+
+#include "QueryHistory.h"
+
+#include
+#include
+
+using namespace std;
+
+using namespace Jetscape;
+
+// ------------------------------------------
+//JP: Quick test module for access to history ...
+//To be done currently hardcoded for a quick test in JetScape.cc ...
+class HistTest : public JetScapeModuleBase
+{
+ public:
+
+ HistTest() : JetScapeModuleBase() {SetId("HistTest");}
+
+ //virtual void InitPerEvent() {QueryHistory::Instance()->PrintTaskMap();}
+
+ virtual void ExecTime()
+ {
+
+ if (GetMainClock()->GetCurrentTime()GetDeltaT()) QueryHistory::Instance()->PrintTaskMap();
+
+ vector eLossHistories = QueryHistory::Instance()->GetHistoryFromModules("JetEnergyLoss");
+
+ if (GetMainClock()->GetCurrentTime()<2) {
+
+ cout<< "HistTest::ExecTime(): Current Main Clock Time = "<GetCurrentTime() << endl;
+ cout<< "HistTest::ExecTime(): Print Histories via vector eLossHistories = QueryHistory::Instance()->GetHistoryFromModules(\"JetEnergyLoss\")" <>(mHist)->PrintEdges(false);
+ }
+ }
+ }
+
+ private:
+};
+// ------------------------------------------
+class ClockTest : public JetScapeModuleBase
+{
+public:
+
+ ClockTest() : JetScapeModuleBase() {SetId("ClockTest");}
+
+ virtual void ExecTime(){
+ cout<< "ClockTest::ExecTime():============================ " << endl;
+ cout<< "ClockTest::ExecTime(): Current Main Clock Time = "<GetCurrentTime() << endl;
+ cout<< "ClockTest::ExecTime(): Current Main Clock Detla T = "<GetDeltaT() << endl;
+ if(UseModuleClock())cout<< "ClockTest::ExecTime(): Current Module Clock Time = "<GetCurrentTime() << endl;
+ if(UseModuleClock())cout<< "ClockTest::ExecTime(): Current Module Clock Detla T = "<GetDeltaT() << endl;
+ cout<< "ClockTest::ExecTime(): Current Module Start Time = "<< GetTStart() << endl;
+ cout<< "ClockTest::ExecTime(): Current Module End Time = "<< GetTEnd() << endl;
+ cout<< "ClockTest::ExecTime(): IsValidModuleTime? "<< IsValidModuleTime() << endl;
+ cout<< "ClockTest::ExecTime():============================ " << endl;
+ }
+
+private:
+};
+// Forward declaration
+void Show();
+
+// -------------------------------------
+
+int main(int argc, char** argv)
+{
+ clock_t t; t = clock();
+ time_t start, end; time(&start);
+
+ cout<SetInfo(true);
+ JetScapeLogger::Instance()->SetDebug(false);
+ JetScapeLogger::Instance()->SetRemark(false);
+ //SetVerboseLevel (9 a lot of additional debug output ...)
+ //If you want to suppress it: use SetVerboseLevle(0) or max SetVerboseLevle(9) or 10
+ JetScapeLogger::Instance()->SetVerboseLevel(0);
+
+
+ Show();
+
+ // -------------
+ //Test clock ...
+
+ //auto mClock = make_shared();
+ //mClock->SetTimeRefFrameId("SpaceTime");
+
+ // clocks here are defaulted for testing, clocks can costumized via inhererting from the MainClock/ModuleClock base classes ...
+
+ //auto mClock = make_shared("SpaceTime",-1,5,0.1); // JP: make consistent with reading from XML in init phase ...
+ //auto mClock = make_shared("SpaceTime",-0.1,0.1,0.1);
+ auto mClock = make_shared("SpaceTime", 0.5,5.0,0.1);
+
+ auto mModuleClock = make_shared();
+ mModuleClock->SetTimeRefFrameId("SpaceTime * 2");
+ auto mMilneClock = make_shared();
+ mMilneClock->setEtaMax(5.0);
+
+ mClock->Info();
+ mModuleClock->Info();
+ mMilneClock->Info();
+
+ /*
+ mClock->Info();
+
+ //while (mClock->Next()) {
+
+ mClock->Tick();
+ mClock->Info();
+
+ mModuleClock->Transform(mClock);
+ mModuleClock->Info();
+
+ //};
+ */
+ // -------------
+
+ auto jetscape = make_shared();
+ jetscape->SetXMLMainFileName("../config/jetscape_main.xml");
+ jetscape->SetXMLUserFileName("../config/jetscape_user_MUSICMainClockTest.xml");
+ jetscape->SetId("primary");
+ jetscape->AddMainClock(mClock);
+ jetscape->ClockInfo();
+
+ auto clockTest1 = make_shared();
+ clockTest1->SetTimeStepped(true);
+ clockTest1->SetTimeRange(-1,0);
+
+ auto clockTest2 = make_shared();
+ clockTest2->SetTimeStepped(true);
+ //clockTest2->SetTimeRange(0,3.5);
+
+ auto clockTest3 = make_shared();
+ clockTest3->SetTimeStepped(true);
+ //clockTest3->SetTimeRange(0,4.);
+ clockTest3->AddModuleClock(mModuleClock);
+
+ jetscape->Add(clockTest1);
+ jetscape->Add(clockTest2);
+ jetscape->Add(clockTest3);
+
+ // Initial conditions and hydro
+ auto trento = make_shared();
+ auto pythiaGun= make_shared ();
+ auto isr = make_shared ();
+ auto null_predynamics = make_shared ();
+ auto hydro = make_shared ();
+ //hydro->AddModuleClock(mMilneClock);
+ hydro->SetTimeStepped(true);
+
+ // surface sampler
+ auto iSS = make_shared ();
+ iSS->SetTimeStepped(true);
+
+ //auto hydroTest = make_shared ();
+ //hydroTest->SetMultiThread(true);
+ //hydroTest->SetTimeStepped(true);
+
+ jetscape->Add(trento);
+ jetscape->Add(pythiaGun);
+ //jetscape->Add(isr);
+ jetscape->Add(null_predynamics);
+ jetscape->Add(hydro);
+ //jetscape->Add(hydroTest);
+ jetscape->Add(iSS);
+
+ // Energy loss
+ auto jlossmanager = make_shared ();
+ auto jloss = make_shared ();
+
+ //Do per time step for these modules with main clock attached ...
+ //Needed to overwrite functions: CalculateTime() and ExecTime(), in these functions get
+ //time, either main clock time or if module clock attached the tranformed time via: GetModuleCurrentTime();
+
+ jlossmanager->SetTimeStepped(true);
+ jloss->SetTimeStepped(true);
+
+ // To test for time step consistency execution settings uncomment next line ...
+ //jloss->SetTimeStepped(false);
+
+ //quick and dirty to check if module clock transformation is working conceptually ...
+ //jloss->AddModuleClock(mModuleClock);
+
+ //Matter is added but not executed, need to implement the per time step execution in JetEnergyLoss::DoShower()...
+ auto matter = make_shared ();
+ // auto lbt = make_shared ();
+ //auto martini = make_shared ();
+ //auto adscft = make_shared ();
+
+ // Note: if you use Matter, it MUST come first (to set virtuality)
+ //jloss->Add(matter);
+ // jloss->Add(lbt); // go to 3rd party and ./get_lbtTab before adding this module
+ // jloss->Add(martini);
+ //jloss->Add(adscft);
+ //jlossmanager->Add(jloss);
+ //jetscape->Add(jlossmanager);
+
+ auto cascadeTest = make_shared ();
+ cascadeTest->SetMultiThread(true);
+ cascadeTest->SetTimeStepped(true);
+ jetscape->Add(cascadeTest);
+
+ //Test task for access of History via QueryHistory instance and use any data-type for generic access via JetScapeModuleBase::GetHistory()
+ auto histTest = make_shared();
+ histTest->SetTimeStepped(true);// to be executed per time step
+ //jetscape->Add(histTest);
+
+ // JP: Leave out for now for testing clock(s) ... has to be updated accordingly ...
+ // (Hadronization not yet modified for per-timestep evolution see JetEnergyLossManager as an example ...)
+ // Hadronization
+ /*
+ auto hadroMgr = make_shared ();
+ auto hadro = make_shared ();
+ //auto hadroModule = make_shared ();
+ //hadro->Add(hadroModule);
+ auto colorless = make_shared ();
+ hadro->Add(colorless);
+ hadroMgr->Add(hadro);
+ jetscape->Add(hadroMgr);
+ */
+
+ // Output
+ auto writer= make_shared ("test_out.dat");
+ writer->SetId("AsciiWriter"); //for task search test ...
+ jetscape->Add(writer);
+
+ /*
+#ifdef USE_GZIP
+ // same as JetScapeWriterAscii but gzipped
+ auto writergz= make_shared ("test_out.dat.gz");
+ jetscape->Add(writergz);
+#endif
+ // HEPMC3
+#ifdef USE_HEPMC
+ auto hepmcwriter= make_shared ("test_out.hepmc");
+ jetscape->Add(hepmcwriter);
+#endif
+ */
+
+ //test ...
+ //QueryHistory::Instance()->AddMainTask(jetscape);
+ //QueryHistory::Instance()->PrintTasks();
+ //QueryHistory::Instance()->PrintTaskMap();
+
+ //check with quick and dirty ... make recursive ...
+ /*
+ cout<GetNumberOfTasks()<GetTaskList();
+ for (auto it : taskList) {
+ cout<GetId()<GetTaskList()) {
+ cout<<" "<GetId()<GetTaskList())
+ cout<<" "<GetId()<Init();
+
+ // Run JetScape with all task/modules as specified
+ jetscape->Exec();
+
+ // For the future, cleanup is mostly already done in write and clear
+ jetscape->Finish();
+
+ INFO_NICE<<"Finished!";
+ cout<stat();
+
+ // // Demonstrate how to work with pythia statistics
+ // //Pythia8::Info& info = pythiaGun->info;
+ // cout << " nTried = " << info.nTried() << endl;
+ // cout << " nSelected = " << info.nSelected() << endl;
+ // cout << " nAccepted = " << info.nAccepted() << endl;
+ // cout << " sigmaGen = " << info.sigmaGen() << endl;
+ // cout << " sigmaErr = " << info.sigmaErr() << endl;
+
+ return 0;
+}
+
+// -------------------------------------
+
+void Show()
+{
+ INFO_NICE<<"-------------------------------------------";
+ INFO_NICE<<"| Clock Brick Test XSCAPE Framework ... |";
+ INFO_NICE<<"-------------------------------------------";
+ INFO_NICE;
+}
diff --git a/external_packages/get_3dglauber.sh b/external_packages/get_3dglauber.sh
index 3456170e..88458854 100755
--- a/external_packages/get_3dglauber.sh
+++ b/external_packages/get_3dglauber.sh
@@ -14,8 +14,10 @@
##############################################################################
folderName="3dMCGlauber"
+commitHash="056a6d03bf6fb097b14d82874cebf6cce2fa0e5a" # for xscape 1.1
# download the code package
rm -fr $folderName
-git clone --depth=1 https://github.com/chunshen1987/3dMCGlauber.git --branch JETSCAPE $folderName
+git clone https://github.com/chunshen1987/3dMCGlauber.git --branch JETSCAPE $folderName
cd $folderName
+git checkout $commitHash
./get_LHAPDF.sh
diff --git a/external_packages/get_iSS.sh b/external_packages/get_iSS.sh
index 544c5deb..ee169a6d 100755
--- a/external_packages/get_iSS.sh
+++ b/external_packages/get_iSS.sh
@@ -14,8 +14,10 @@
##############################################################################
# download the code package
+commitHash="4bc2badcd31401bcdea8be5a2efc778bdf99fc57" # for xscape 1.1
rm -fr iSS
-git clone https://github.com/chunshen1987/iSS iSS
+git clone https://github.com/chunshen1987/iSS -b JETSCAPE iSS
cd iSS
+git checkout $commitHash
#git checkout tags/v1.1.1 -b v1.1.1
rm -fr iSS/.git
diff --git a/external_packages/get_music.sh b/external_packages/get_music.sh
index 5d4bb810..8b3fe8b8 100755
--- a/external_packages/get_music.sh
+++ b/external_packages/get_music.sh
@@ -14,9 +14,12 @@
##############################################################################
folderName="music"
+commitHash="52b0454b0a07a76f26c8138245c0b3c54d00af65" # for xscape 1.1
# download the code package
rm -fr $folderName
-git clone --depth=1 https://github.com/MUSIC-fluid/MUSIC.git --branch JETSCAPE $folderName
+git clone https://github.com/MUSIC-fluid/MUSIC.git --branch JETSCAPE $folderName
+cd $folderName
+git checkout $commitHash
### ALTERNATIVE VERSION
### Download a zipped snapshot
diff --git a/external_packages/get_smash.sh b/external_packages/get_smash.sh
index e2ab461e..3866456e 100755
--- a/external_packages/get_smash.sh
+++ b/external_packages/get_smash.sh
@@ -14,14 +14,13 @@
##############################################################################
# 1) Download the SMASH code
-# TODO: Change branch back to main SMASH version before public XSCAPE version
-git clone --depth=1 https://github.com/smash-transport/smash.git --branch staudenmaier/mod_plist_ts smash/smash_code
+git clone --depth=1 https://github.com/smash-transport/smash.git --branch SMASH-3.0 smash/smash_code
# 2) Compile SMASH
cd smash/smash_code
mkdir build
cd build
-cmake .. -DCMAKE_CXX_STANDARD=14 -DPythia_CONFIG_EXECUTABLE=${PYTHIA8DIR}/bin/pythia8-config
+cmake .. -DPythia_CONFIG_EXECUTABLE=${PYTHIA8DIR}/bin/pythia8-config
number_of_cores=`nproc --all`
number_of_cores_to_compile=$(( ${number_of_cores} > 20 ? 20 : ${number_of_cores} ))
echo "Compiling SMASH using ${number_of_cores_to_compile} cores."
diff --git a/external_packages/smash/smash_config.yaml b/external_packages/smash/smash_config.yaml
index 81465f9a..47be0b70 100644
--- a/external_packages/smash/smash_config.yaml
+++ b/external_packages/smash/smash_config.yaml
@@ -2,22 +2,29 @@ Logging:
default: INFO
General:
- Modus: Afterburner # Dummy, but some value has to be present
+ Modus: List
Time_Step_Mode: Fixed
- Delta_Time: 0.1
- End_Time: 100.0 # Dummy
+ Delta_Time: 0.5
+ End_Time: 100.0 # Reset from the JetScape xml config
Randomseed: -1 # Reset from the JetScape xml config
Nevents: 1 # Dummy
-#Collision_Term:
-# Strings: False
+Modi:
+ List:
+ File_Directory: .
+ File_Prefix: "dummy"
+ Shift_Id: 0
-Output:
- Output_Interval: 5.0
-# This section is dummy
-List:
- File_Directory: .
- File_Prefix: "dummy"
- Shift_Id: 0
- Start_Time: 0.0
+# if some SMASH output in OSCAR,... format is wanted, then uncomment
+# the following lines and add a 'smash_output' folder to your build folder
+Output:
+ Output_Interval: 100.0
+# Particles:
+# Format: ["Oscar2013"]
+# Extended: True
+# Only_Final: Yes
+# Collisions:
+# Format: ["Oscar2013"]
+# Extended: True
+# Print_Start_End: True
\ No newline at end of file
diff --git a/external_packages/trento/CMakeLists.txt b/external_packages/trento/CMakeLists.txt
index c47a7c91..53889833 100644
--- a/external_packages/trento/CMakeLists.txt
+++ b/external_packages/trento/CMakeLists.txt
@@ -4,8 +4,10 @@ project(trento VERSION 1.5.1 LANGUAGES CXX)
-# require C++11
-set(CMAKE_CXX_STANDARD 11)
+# require at least C++11
+if(NOT "${CMAKE_CXX_STANDARD}")
+ set(CMAKE_CXX_STANDARD 11)
+endif()
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)
diff --git a/src/afterburner/SmashWrapper.cc b/src/afterburner/SmashWrapper.cc
index edee5101..907ffde5 100644
--- a/src/afterburner/SmashWrapper.cc
+++ b/src/afterburner/SmashWrapper.cc
@@ -24,8 +24,7 @@
#include
#include
-#include
-#include
+#include
using namespace Jetscape;
@@ -38,31 +37,32 @@ SmashWrapper::SmashWrapper() {
void SmashWrapper::InitTask() {
JSINFO << "SMASH: picking SMASH-specific configuration from xml file";
- std::string smash_config =
+ std::string smash_config_file =
GetXMLElementText({"Afterburner", "SMASH", "SMASH_config_file"});
std::string smash_hadron_list =
GetXMLElementText({"Afterburner", "SMASH", "SMASH_particles_file"});
std::string smash_decays_list =
GetXMLElementText({"Afterburner", "SMASH", "SMASH_decaymodes_file"});
// output path is just dummy here, because no output from SMASH is foreseen
- boost::filesystem::path output_path("./smash_output");
+ std::filesystem::path output_path("./smash_output");
// do not store tabulation, which is achieved by an empty tabulations path
std::string tabulations_path("");
const std::string smash_version(SMASH_VERSION);
- auto config = smash::setup_config_and_logging(smash_config, smash_hadron_list,
+ auto config = smash::setup_config_and_logging(smash_config_file,
+ smash_hadron_list,
smash_decays_list);
// Take care of the random seed. This will make SMASH results reproducible.
auto random_seed = (*GetMt19937Generator())();
- config["General"]["Randomseed"] = random_seed;
+ config.set_value({"General","Randomseed"}, random_seed);
// Read in the rest of configuration
if (IsTimeStepped()) {
end_time_ = GetMainClock()->GetEndTime();
} else {
end_time_ = GetXMLElementDouble({"Afterburner", "SMASH", "end_time"});
}
- config["General"]["End_Time"] = end_time_;
+ config.set_value({"General","End_Time"}, end_time_);
JSINFO << "End time until which SMASH propagates is " << end_time_ << " fm/c";
only_final_decays_ =
GetXMLElementInt({"Afterburner", "SMASH", "only_decays"});
@@ -183,7 +183,7 @@ void SmashWrapper::FinishPerEvent() {
modus->jetscape_hadrons_[ev_no - 1]);
JSINFO << modus->jetscape_hadrons_[ev_no - 1].size()
<< " hadrons from SMASH.";
- smash_experiment_->increase_event_no(); // internal SMASH event counter
+ smash_experiment_->increase_event_number(); // internal SMASH event counter
}
void SmashWrapper::WriteTask(weak_ptr w) {
diff --git a/src/afterburner/SmashWrapper.h b/src/afterburner/SmashWrapper.h
index 5648a252..26359a72 100644
--- a/src/afterburner/SmashWrapper.h
+++ b/src/afterburner/SmashWrapper.h
@@ -36,8 +36,9 @@ using namespace Jetscape;
class AfterburnerModus : public smash::ListModus {
public:
// Unlike for ListModus there is no need to get any data from the config
- AfterburnerModus(smash::Configuration, const smash::ExperimentParameters &) {
+ AfterburnerModus(smash::Configuration config, const smash::ExperimentParameters &) {
JSINFO << "Constructing AfterburnerModus";
+ config.clear();
}
void reset_event_numbering() { event_number_ = 0; }
int current_event_number() {return event_number_;}
diff --git a/src/framework/Afterburner.cc b/src/framework/Afterburner.cc
index db29231e..91b858c7 100644
--- a/src/framework/Afterburner.cc
+++ b/src/framework/Afterburner.cc
@@ -24,6 +24,8 @@ void Afterburner::Init() {
// Makes sure that XML file with options and parameters is loaded
JetScapeModuleBase::InitTask();
JSINFO << "Initializing Afterburner : " << GetId() << " ...";
+ // Initialize random number distribution
+ ZeroOneDistribution = uniform_real_distribution{0.0, 1.0};
InitTask();
InitTasks();
}
@@ -40,11 +42,14 @@ void Afterburner::CalculateTime() {
std::vector>> Afterburner::GetSoftParticlizationHadrons() {
auto soft_particlization = JetScapeSignalManager::Instance()->GetSoftParticlizationPointer().lock();
if (!soft_particlization) {
- JSWARN << "No soft particlization module found. It is necessary to provide"
- << " hadrons to afterburner.";
- exit(1);
+ JSWARN << "No soft particlization module found. Check if fragmentation"
+ << " hadrons are handed to afterburner.";
+ std::vector> hadrons;
+ dummy.push_back(hadrons);
+ return dummy;
+ } else {
+ return soft_particlization->Hadron_list_;
}
- return soft_particlization->Hadron_list_;
}
std::vector> Afterburner::GetFragmentationHadrons() {
@@ -58,26 +63,49 @@ std::vector> Afterburner::GetFragmentationHadrons() {
std::vector> h_list;
hadronization_mgr->GetHadrons(h_list);
JSINFO << "Got " << h_list.size() << " fragmentation hadrons from HadronizationManager.";
-
+
+ std::vector> h_list_new;
rand_int_ptr_ = (std::make_shared>(0,1));
for (auto h : h_list) {
if (h->has_no_position()) {
- // No position info set in hadronization module
- JSWARN << "Found fragmentation hadron without properly set position in "
+ JSDEBUG << "Found fragmentation hadron without properly set position in "
"Afterburner.\nInclusion of fragmentation hadrons only "
- "possible for HybridHadronization.\nExiting.";
- exit(1);
+ "possible for HybridHadronization.";
}
- // convert Kaon-L or Kaon-S into K0 or Anti-K0
- // Kaon-L or Kaon-S are unknown particles for SMASH
- if (h->pid() == 310 || h->pid() == 130) {
- const int rand_int = (*rand_int_ptr_)(*GetMt19937Generator());
- const int id = (rand_int == 0) ? 311 : -311;
- h->set_id(id);
+
+ //move all the fragmentation hadrons a little bit around to avoid having
+ //multiple hadrons at the same position if they are at the same position
+ const FourVector r = h->x_in();
+ const double rand_x = ZeroOneDistribution(*GetMt19937Generator()) * 2e-4 - 1e-4;
+ const double rand_y = ZeroOneDistribution(*GetMt19937Generator()) * 2e-4 - 1e-4;
+ const double rand_z = ZeroOneDistribution(*GetMt19937Generator()) * 2e-4 - 1e-4;
+ double position_smeared[4] = {r.t(), r.x()+rand_x, r.y()+rand_y, r.z()+rand_z};
+ h->set_x(position_smeared);
+
+ if ((std::abs(h->pid())>10) && (h->pid() != 21)) {
+ if (h->pstat() > 0) {
+ // convert Kaon-L or Kaon-S into K0 or Anti-K0
+ if (h->pid() == 310 || h->pid() == 130) {
+ const int rand_int = (*rand_int_ptr_)(*GetMt19937Generator());
+ const int id = (rand_int == 0) ? 311 : -311;
+ h->set_id(id);
+ }
+ h_list_new.push_back(h);
+ } else if(h->pstat() < 0) {
+ // convert Kaon-L or Kaon-S into K0 or Anti-K0
+ // change id of negative Kaons to make them consistent with the SMASH output
+ if (h->pid() == 310 || h->pid() == 130) {
+ const int rand_int = (*rand_int_ptr_)(*GetMt19937Generator());
+ const int id = (rand_int == 0) ? 311 : -311;
+ h->set_id(id);
+ }
+ }
+ } else if((std::abs(h->pid())<10) || (h->pid() == 21)){
+ JSWARN << "Found a free quark or gluon! This can not be handed over to SMASH.\n"
+ "Check confinement in hadronization module!";
}
-
}
- return h_list;
+ return h_list_new;
}
std::vector>> Afterburner::GatherAfterburnerHadrons() {
@@ -85,15 +113,17 @@ std::vector>> Afterburner::GatherAfterburner
afterburner_had_events = GetSoftParticlizationHadrons();
if (GetXMLElementInt({"Afterburner", "output_only_final_state_hadrons"})) {
- // clear Hadron_list_ in soft_particlization, otherwise the final hadron
- // output of the writer contains also the soft hadrons which were used as
+ // clear Hadron_list_ in soft_particlization, otherwise the final hadron
+ // output of the writer contains also the soft hadrons which were used as
// input for SMASH
auto soft_particlization = JetScapeSignalManager::Instance()->GetSoftParticlizationPointer().lock();
- soft_particlization->Hadron_list_.clear();
+ if (soft_particlization) {
+ soft_particlization->Hadron_list_.clear();
+ }
}
if (GetXMLElementInt({"Afterburner", "include_fragmentation_hadrons"})) {
- if (afterburner_had_events.size() != 1) {
+ if (afterburner_had_events.size() > 1) {
JSWARN << "Fragmentation hadrons in Afterburner are only possible without "
"repeated sampling from SoftParticlization. Exiting.";
exit(1);
@@ -104,11 +134,12 @@ std::vector>> Afterburner::GatherAfterburner
// empty the hadron vector in the hadronization manager to circumvent the
// output of these hadrons if they are implemented in the SMASH afterburner
auto hadronization_mgr = JetScapeSignalManager::Instance()->GetHadronizationManagerPointer().lock();
- hadronization_mgr->DeleteHadrons();
+ hadronization_mgr->DeleteRealHadrons();
}
afterburner_had_events[0].insert(afterburner_had_events[0].end(),
frag_hadrons.begin(), frag_hadrons.end());
+ dummy.clear();
}
return afterburner_had_events;
}
diff --git a/src/framework/Afterburner.h b/src/framework/Afterburner.h
index a09e0e73..37aed7a8 100644
--- a/src/framework/Afterburner.h
+++ b/src/framework/Afterburner.h
@@ -64,9 +64,11 @@ class Afterburner : public JetScapeModuleBase {
/// Get the list of hadrons for the upcoming timestep from BulkDynamicsManager (will clear the list)
std::vector> GetTimestepParticlizationHadrons();
+ std::vector>> dummy;
+ std::uniform_real_distribution ZeroOneDistribution;
+
// rng for the Kaon-L / Kaon-S switch to K0 / Anti-K0
std::shared_ptr> rand_int_ptr_;
-
};
} // end namespace Jetscape
diff --git a/src/framework/FluidDynamics.h b/src/framework/FluidDynamics.h
index 190c5112..5a06a857 100644
--- a/src/framework/FluidDynamics.h
+++ b/src/framework/FluidDynamics.h
@@ -2,7 +2,7 @@
* Copyright (c) The JETSCAPE Collaboration, 2018
*
* Modular, task-based framework for simulating all aspects of heavy-ion collisions
- *
+ *
* For the list of contributors see AUTHORS.
*
* Report issues at https://github.com/JETSCAPE/JETSCAPE/issues
@@ -71,11 +71,12 @@ class FluidDynamics : public JetScapeModuleBase {
// for large dataset, std::deque is better than std::vector.
/** Stores the evolution history. */
EvolutionHistory bulk_info;
+ std::vector surfaceCellVector_;
std::weak_ptr liquefier_ptr;
public:
- /** Default constructor. task ID as "FluidDynamics",
+ /** Default constructor. task ID as "FluidDynamics",
eta is initialized to -99.99.
*/
FluidDynamics();
@@ -112,7 +113,7 @@ class FluidDynamics : public JetScapeModuleBase {
/** It stores the temperature of the fluid cell at location (t or tau,x,y,z or eta) into an input variable "mT". It can be overridden by modules attached to the FluidDynamics class.
@param t tau or t coordinate.
- @param x space x coordinate.
+ @param x space x coordinate.
@param y space y coordinate.
@param z rapidity eta or space z coordinate.
@param mT temperature.
@@ -122,12 +123,12 @@ class FluidDynamics : public JetScapeModuleBase {
mT = GetTemperature(t, x, y, z);
}
- /** It calls GetHydroInfo(t,x,y,z,fCell) to retrieve the properties of the fluid cell at location (t or tau,x,y,z or eta). It can be overridden by modules attached to the FluidDynamics class.
+ /** It calls GetHydroInfo(t,x,y,z,fCell) to retrieve the properties of the fluid cell at location (t or tau,x,y,z or eta). It can be overridden by modules attached to the FluidDynamics class.
@param t tau or t coordinate.
- @param x space x coordinate.
- @param y space y coordinate.
+ @param x space x coordinate.
+ @param y space y coordinate.
@param z rapidity eta or space z coordinate.
- @param fCell A pointer of type FluidCellInfo class.
+ @param fCell A pointer of type FluidCellInfo class.
*/
virtual void GetHydroCell(double t, double x, double y, double z,
std::unique_ptr &fCell) {
@@ -137,13 +138,15 @@ class FluidDynamics : public JetScapeModuleBase {
// currently we have no standard for passing configurations
// pure virtual function; to be implemented by users
// should make it easy to save evolution history to bulk_info
- /** Default function to initialize the hydrodynamics. It can be overridden by different modules.
+ /** Default function to initialize the hydrodynamics. It can be overridden by different modules.
@param parameter_list An object of the class Parameter. */
virtual void InitializeHydro(Parameter parameter_list){};
/** Default function to evolve the hydrodynamics. It can be overridden by different modules. */
virtual void EvolveHydro(){};
+ virtual void EvolveHydroUpto(const double tauEnd){};
+
/** @return Status of the hydrodynamics (NOT_START, INITIALIZED, EVOLVING, FINISHED, ERROR). */
int GetHydroStatus() const { return (hydro_status); }
@@ -152,6 +155,19 @@ class FluidDynamics : public JetScapeModuleBase {
bulk_info.data.push_back(*fluid_cell_info_ptr);
}
+ void StoreSurfaceCell(SurfaceCellInfo &surface_cell_info) {
+ surfaceCellVector_.push_back(surface_cell_info);
+ }
+
+ void getSurfaceCellVector(std::vector & surfCellVec) {
+ surfCellVec = surfaceCellVector_;
+ JSINFO << "Fluid out: surface vector size = " << surfCellVec.size();
+ }
+
+ void clearSurfaceCellVector() {
+ surfaceCellVector_.clear();
+ }
+
void clear_up_evolution_data() { bulk_info.clear_up_evolution_data(); }
/** @return Start time (or tau) for hydrodynamic evolution.
@@ -169,8 +185,8 @@ class FluidDynamics : public JetScapeModuleBase {
/** Retrieves the hydro information at a given space-time point.
* It throws a InvalidSpaceTimeRange message when
- * (t or tau, x, y, z or eta) is out of the evolution history range.
- @param time Time or tau coordinate.
+ * (t or tau, x, y, z or eta) is out of the evolution history range.
+ @param time Time or tau coordinate.
@param x Space coordinate.
@param y Space coordinate.
@param z Space or eta coordinate.
@@ -218,7 +234,7 @@ class FluidDynamics : public JetScapeModuleBase {
/** @return Energy density at point (t or tau, x, y, z or eta)
@param time Time or tau coordinate.
@param x Space coordinate.
- @param y Space coordinate.
+ @param y Space coordinate.
@param z Space or eta coordinate.
*/
virtual Jetscape::real GetEnergyDensity(Jetscape::real time, Jetscape::real x,
@@ -227,7 +243,7 @@ class FluidDynamics : public JetScapeModuleBase {
/** @return Entropy density at point (t or tau, x, y, z or eta)
@param time Time or tau coordinate.
@param x Space coordinate.
- @param y Space coordinate.
+ @param y Space coordinate.
@param z Space or eta coordinate.
*/
virtual Jetscape::real GetEntropyDensity(Jetscape::real time,
@@ -298,6 +314,9 @@ class FluidDynamics : public JetScapeModuleBase {
/// slots for "jet" signals (future)
virtual void GetEnergyDensity(int t, double &edensity) { edensity = 0.0; }
+ // get a reference to the bulk_info object
+ const EvolutionHistory& get_bulk_info() const { return bulk_info; }
+
}; // end class FluidDynamics
} // end namespace Jetscape
diff --git a/src/framework/FluidEvolutionHistory.cc b/src/framework/FluidEvolutionHistory.cc
index 6b217f4e..3158d9f8 100644
--- a/src/framework/FluidEvolutionHistory.cc
+++ b/src/framework/FluidEvolutionHistory.cc
@@ -292,8 +292,8 @@ FluidCellInfo EvolutionHistory::get_tz(Jetscape::real t, Jetscape::real x,
tau = sqrt(t * t - z * z);
eta = 0.5 * log((t + z) / (t - z));
} else {
- JSWARN << "the quest point is outside the light cone! "
- << "t = " << t << ", z = " << z;
+ VERBOSE(4) << "the quest point is outside the light cone! "
+ << "t = " << t << ", z = " << z;
}
return (get(tau, x, y, eta));
}
diff --git a/src/framework/FluidEvolutionHistory.h b/src/framework/FluidEvolutionHistory.h
index 92cb9338..80df829d 100644
--- a/src/framework/FluidEvolutionHistory.h
+++ b/src/framework/FluidEvolutionHistory.h
@@ -2,7 +2,7 @@
* Copyright (c) The JETSCAPE Collaboration, 2018
*
* Modular, task-based framework for simulating all aspects of heavy-ion collisions
- *
+ *
* For the list of contributors see AUTHORS.
*
* Report issues at https://github.com/JETSCAPE/JETSCAPE/issues
@@ -92,7 +92,7 @@ class EvolutionHistory {
bool boost_invariant;
- /** The bulk information of hydro dynamics, in the form of
+ /** The bulk information of hydro dynamics, in the form of
* vector of FluidCellInfo objects. */
std::vector data;
@@ -133,6 +133,7 @@ class EvolutionHistory {
int get_data_size() const { return (data.size()); }
bool is_boost_invariant() const { return (boost_invariant); }
+ bool is_Cartesian() const { return (tau_eta_is_tz); }
Jetscape::real Tau0() const { return (tau_min); }
Jetscape::real XMin() const { return (x_min); }
@@ -153,7 +154,7 @@ class EvolutionHistory {
/** It checks whether a space-time point (tau, x, y, eta) is inside evolution history or outside.
@param tau Light-cone coordinate.
- @param x Space coordinate.
+ @param x Space coordinate.
@param y Space coordinate.
@param eta Light-cone coordinate.
*/
@@ -169,16 +170,16 @@ class EvolutionHistory {
}
// get the lower bound of the fluid cell along x
- /** @return Fluid cell number along the x-grid.
- @param x Space coordinate.
+ /** @return Fluid cell number along the x-grid.
+ @param x Space coordinate.
*/
inline int GetIdX(Jetscape::real x) const {
return (static_cast((x - x_min) / dx));
}
// get the lower bound of the fluid cell along y
- /** @return Fluid cell number along the y-grid.
- @param y Space coordinate.
+ /** @return Fluid cell number along the y-grid.
+ @param y Space coordinate.
*/
inline int GetIdY(Jetscape::real y) const {
return (static_cast((y - y_min) / dy));
@@ -250,7 +251,7 @@ class EvolutionHistory {
@param tau Light-cone coordinate.
@param x Space coordinate.
@param y Space coordinate.
- @param eta Light-cone coordinate.
+ @param eta Light-cone coordinate.
*/
FluidCellInfo get(Jetscape::real tau, Jetscape::real x, Jetscape::real y,
Jetscape::real etas) const;
diff --git a/src/framework/Hadronization.cc b/src/framework/Hadronization.cc
index 6908097a..1fea2045 100644
--- a/src/framework/Hadronization.cc
+++ b/src/framework/Hadronization.cc
@@ -101,4 +101,17 @@ void Hadronization::DeleteHadrons() {
outHadrons.clear();
}
+void Hadronization::DeleteRealHadrons() {
+ outHadrons.erase(
+ std::remove_if(
+ outHadrons.begin(),
+ outHadrons.end(),
+ [](const std::shared_ptr& hadron) {
+ return hadron->pstat() > 0;
+ }
+ ),
+ outHadrons.end()
+ );
+}
+
} // namespace Jetscape
diff --git a/src/framework/Hadronization.h b/src/framework/Hadronization.h
index 9af2126c..85c907f2 100644
--- a/src/framework/Hadronization.h
+++ b/src/framework/Hadronization.h
@@ -83,10 +83,13 @@ class Hadronization : public JetScapeModuleBase,
void AddInHadrons(vector> ih) { outHadrons = ih; }
- // empties the outHadrons vector, explanation why this is necessary is given
+ // empties the outHadrons vector, explanation why this is necessary is given
// in HadronizationManager.h
void DeleteHadrons();
+ // erases the outHadrons with positive status flag
+ void DeleteRealHadrons();
+
private:
vector>> inPartons;
vector> outHadrons;
diff --git a/src/framework/HadronizationManager.cc b/src/framework/HadronizationManager.cc
index 149ccace..586d587d 100644
--- a/src/framework/HadronizationManager.cc
+++ b/src/framework/HadronizationManager.cc
@@ -157,4 +157,13 @@ void HadronizationManager::DeleteHadrons() {
}
}
+void HadronizationManager::DeleteRealHadrons() {
+ // foreach hadronizon object tasks
+ for(shared_ptr it : GetTaskList()){
+ JetScapeTask *jet = it.get();
+ Hadronization *hit = (Hadronization *) jet;
+ hit->DeleteRealHadrons();
+ }
+}
+
} // namespace Jetscape
diff --git a/src/framework/HadronizationManager.h b/src/framework/HadronizationManager.h
index ec196730..41e1f9f6 100644
--- a/src/framework/HadronizationManager.h
+++ b/src/framework/HadronizationManager.h
@@ -54,10 +54,13 @@ class HadronizationManager
// deletes the hadrons from the different hadronization modules
// this is used in the case of hadronization hadrons in the afterburner
- // otherwise these hadrons are printed to file and the same hadrons will be
+ // otherwise these hadrons are printed to file and the same hadrons will be
// modified in the transport and printed again
void DeleteHadrons();
-
+ // this function removes all positive hadrons, the negative ones are not deleted
+ // needed, when positive hadrons are given to the afterburner
+ void DeleteRealHadrons();
+
sigslot::signal1> &> GetHadronList; //get Hadrons from HardProcess NOT Hadronization submodules
sigslot::signal1>> &> GetFinalPartonList;
diff --git a/src/framework/JetScape.cc b/src/framework/JetScape.cc
index d34c963a..4970fb09 100644
--- a/src/framework/JetScape.cc
+++ b/src/framework/JetScape.cc
@@ -323,7 +323,18 @@ void JetScape::DetermineTaskListFromXML() {
JSINFO << "JetScape::DetermineTaskList() -- Hard Process: Added "
"PythiaGun to task list.";
}
- } else if (((int)childElementName.find("CustomModule") >= 0)) {
+ }
+ // - epemGun
+ else if (childElementName == "epemGun") {
+ auto EpemGun =
+ JetScapeModuleFactory::createInstance(childElementName);
+ if (EpemGun) {
+ Add(EpemGun);
+ JSINFO << "JetScape::DetermineTaskList() -- Hard Process: Added "
+ "epemGun to task list.";
+ }
+ }
+ else if (((int)childElementName.find("CustomModule") >= 0)) {
auto customModule =
JetScapeModuleFactory::createInstance(childElementName);
if (customModule) {
@@ -916,6 +927,10 @@ void JetScape::SetPointers() {
JetScapeSignalManager::Instance()->SetSoftParticlizationPointer(
dynamic_pointer_cast(it));
iss_pointer_is_set = true;
+ JetScapeSignalManager::Instance()->ConnectGetHydroHyperSurfaceSignal(
+ dynamic_pointer_cast(it));
+ JetScapeSignalManager::Instance()->ConnectClearHydroHyperSurfaceSignal(
+ dynamic_pointer_cast(it));
} else if (dynamic_pointer_cast(it)) {
JetScapeSignalManager::Instance()->SetHadronizationManagerPointer(
dynamic_pointer_cast(it));
diff --git a/src/framework/JetScapeConstants.h b/src/framework/JetScapeConstants.h
index 5f440d53..33306282 100644
--- a/src/framework/JetScapeConstants.h
+++ b/src/framework/JetScapeConstants.h
@@ -2,7 +2,7 @@
* Copyright (c) The JETSCAPE Collaboration, 2018
*
* Modular, task-based framework for simulating all aspects of heavy-ion collisions
- *
+ *
* For the list of contributors see AUTHORS.
*
* Report issues at https://github.com/JETSCAPE/JETSCAPE/issues
@@ -34,11 +34,10 @@ static double Nc = 3.0;
static double Lambda_QCD = 0.2;
// 0.4 is the value chosen in JETSET
-static double fmToGeVinv = 5.0;
-/// < should be 1/0.197, but 5 helps in debugging.
-
static const double hbarC = 0.197327053;
+static const double fmToGeVinv = 1.0 / hbarC;
+
static double zeta3 = 1.20206;
static double mu = 0.722;
diff --git a/src/framework/JetScapeSignalManager.cc b/src/framework/JetScapeSignalManager.cc
index d82af920..86c802c9 100644
--- a/src/framework/JetScapeSignalManager.cc
+++ b/src/framework/JetScapeSignalManager.cc
@@ -232,6 +232,32 @@ void JetScapeSignalManager::ConnectGetHydroHyperSurfaceSignal(
}
}
+
+void JetScapeSignalManager::ConnectGetHydroHyperSurfaceSignal(
+ shared_ptr hSoft) {
+ if (!hSoft->GetGetHydroHyperSurfaceConnected()) {
+ auto hp = GetHydroPointer().lock();
+ if (hp) {
+ hSoft->GetHydroHyperSurface.connect(
+ hp.get(), &FluidDynamics::getSurfaceCellVector);
+ hSoft->SetGetHydroHyperSurfaceConnected(true);
+ }
+ }
+}
+
+
+void JetScapeSignalManager::ConnectClearHydroHyperSurfaceSignal(
+ shared_ptr hSoft) {
+ if (!hSoft->GetClearHydroHyperSurfaceConnected()) {
+ auto hp = GetHydroPointer().lock();
+ if (hp) {
+ hSoft->ClearHydroHyperSurface.connect(
+ hp.get(), &FluidDynamics::clearSurfaceCellVector);
+ hSoft->SetClearHydroHyperSurfaceConnected(true);
+ }
+ }
+}
+
void JetScapeSignalManager::CleanUp() {
VERBOSE(8);
diff --git a/src/framework/JetScapeSignalManager.h b/src/framework/JetScapeSignalManager.h
index b032b4cf..0c32f5a7 100644
--- a/src/framework/JetScapeSignalManager.h
+++ b/src/framework/JetScapeSignalManager.h
@@ -130,9 +130,12 @@ class
void ConnectGetFinalPartonListSignal(shared_ptr hm);
void ConnectTransformPartonsSignal(shared_ptr h,
shared_ptr h2);
- void ConnectGetFinalHadronListSignal(shared_ptr h);
+ void ConnectGetFinalHadronListSignal(shared_ptr h);
void ConnectGetHydroHyperSurfaceSignal(shared_ptr h);
+ void ConnectGetHydroHyperSurfaceSignal(shared_ptr hSoft);
+ void ConnectClearHydroHyperSurfaceSignal(
+ shared_ptr hSoft);
void
DisconnectSignal(){}; // to be implememted if needed maybe for Eloss ...!???
diff --git a/src/framework/JetScapeWriterFinalStateStream.cc b/src/framework/JetScapeWriterFinalStateStream.cc
index e7992e5b..cbf0083a 100644
--- a/src/framework/JetScapeWriterFinalStateStream.cc
+++ b/src/framework/JetScapeWriterFinalStateStream.cc
@@ -75,9 +75,6 @@ template void JetScapeWriterFinalStateStream::WriteEvent() {
auto particle = p.get();
- // Skip ISR partons
- if(particle->pstat() < 0) continue;
-
output_file << ipart
// << " " << particle->plabel()
<< " " << particle->pid()
diff --git a/src/framework/SoftParticlization.cc b/src/framework/SoftParticlization.cc
index 58b5d9a5..af900363 100644
--- a/src/framework/SoftParticlization.cc
+++ b/src/framework/SoftParticlization.cc
@@ -18,10 +18,14 @@
// -----------------------------------------
#include "SoftParticlization.h"
+#include "JetScapeSignalManager.h"
namespace Jetscape {
-SoftParticlization::SoftParticlization() { boost_invariance = false; }
+SoftParticlization::SoftParticlization() {
+ boost_invariance = false;
+ HydroHyperSurfaceConnected_ = false;
+}
SoftParticlization::~SoftParticlization() {
for (unsigned i = 0; i < Hadron_list_.size(); i++) {
@@ -40,6 +44,7 @@ void SoftParticlization::Init() {
InitTask();
InitTasks();
+ //CreateSignalSlots();
}
void SoftParticlization::ExecuteTask() {}
diff --git a/src/framework/SoftParticlization.h b/src/framework/SoftParticlization.h
index d12057bb..83c86961 100644
--- a/src/framework/SoftParticlization.h
+++ b/src/framework/SoftParticlization.h
@@ -25,11 +25,14 @@
#include "JetScapeModuleBase.h"
#include "JetClass.h"
#include "JetScapeWriter.h"
+#include "SurfaceCellInfo.h"
namespace Jetscape {
class SoftParticlization : public JetScapeModuleBase {
private:
+ bool HydroHyperSurfaceConnected_;
+ bool ClearHydroHyperSurfaceConnected_;
public:
SoftParticlization();
~SoftParticlization();
@@ -40,6 +43,26 @@ class SoftParticlization : public JetScapeModuleBase {
virtual void ExecuteTask();
virtual void ClearTask();
+ sigslot::signal1 &,
+ multi_threaded_local> GetHydroHyperSurface;
+ sigslot::signal0 ClearHydroHyperSurface;
+
+ void SetGetHydroHyperSurfaceConnected(bool m_GetHydroHyperSurfaceConnected) {
+ HydroHyperSurfaceConnected_ = m_GetHydroHyperSurfaceConnected;
+ }
+
+ void SetClearHydroHyperSurfaceConnected(bool m_ClearHydroHyperSurfaceConnected) {
+ ClearHydroHyperSurfaceConnected_ = m_ClearHydroHyperSurfaceConnected;
+ }
+
+ bool GetGetHydroHyperSurfaceConnected() const {
+ return HydroHyperSurfaceConnected_;
+ }
+
+ bool GetClearHydroHyperSurfaceConnected() const {
+ return ClearHydroHyperSurfaceConnected_;
+ }
+
std::vector>> Hadron_list_;
bool boost_invariance;
diff --git a/src/framework/SurfaceCellInfo.h b/src/framework/SurfaceCellInfo.h
index e3cfed51..99dd53a2 100644
--- a/src/framework/SurfaceCellInfo.h
+++ b/src/framework/SurfaceCellInfo.h
@@ -33,13 +33,11 @@ class SurfaceCellInfo {
Jetscape::real entropy_density; //!< Local entropy density [1/fm^3].
Jetscape::real temperature; //!< Local temperature [GeV].
Jetscape::real pressure; //!< Thermal pressure [GeV/fm^3].
- Jetscape::real
- qgp_fraction; //!< Fraction of quark gluon plasma assuming medium is in QGP+HRG phase.
Jetscape::real mu_B; //!< Net baryon chemical potential [GeV].
- Jetscape::real mu_C; //!< Net charge chemical potential [GeV].
+ Jetscape::real mu_Q; //!< Net charge chemical potential [GeV].
Jetscape::real mu_S; //!< Net strangeness chemical potential [GeV].
- Jetscape::real vx, vy, vz; //!< Flow velocity.
- Jetscape::real pi[4][4]; //!< Shear stress tensor [GeV/fm^3].
+ Jetscape::real umu[4];
+ Jetscape::real pi[10]; //!< Shear stress tensor [GeV/fm^3].
Jetscape::real bulk_Pi; //!< Bulk viscous pressure [GeV/fm^3].
/** Default constructor. */
diff --git a/src/framework/SurfaceFinder.cc b/src/framework/SurfaceFinder.cc
index e328d512..21ca2952 100644
--- a/src/framework/SurfaceFinder.cc
+++ b/src/framework/SurfaceFinder.cc
@@ -347,23 +347,32 @@ SurfaceCellInfo SurfaceFinder::PrepareASurfaceCell(
temp_cell.d3sigma_mu[3] = da3;
temp_cell.energy_density = fluid_cell.energy_density;
- temp_cell.entropy_density = fluid_cell.entropy_density;
temp_cell.temperature = fluid_cell.temperature;
temp_cell.pressure = fluid_cell.pressure;
- temp_cell.qgp_fraction = fluid_cell.qgp_fraction;
temp_cell.mu_B = fluid_cell.mu_B;
- temp_cell.mu_C = fluid_cell.mu_C;
+ temp_cell.mu_Q = fluid_cell.mu_C;
temp_cell.mu_S = fluid_cell.mu_S;
- temp_cell.vx = fluid_cell.vx;
- temp_cell.vy = fluid_cell.vy;
- temp_cell.vz = fluid_cell.vz;
+ double u0 = sqrt(1. + fluid_cell.vx*fluid_cell.vx
+ + fluid_cell.vy*fluid_cell.vy
+ + fluid_cell.vz*fluid_cell.vz);
+ double uz = u0*fluid_cell.vz;
+ temp_cell.umu[0] = u0*cosh(eta) - uz*sinh(eta);
+ temp_cell.umu[1] = u0*fluid_cell.vx;
+ temp_cell.umu[2] = u0*fluid_cell.vy;
+ temp_cell.umu[3] = - u0*sinh(eta) + uz*cosh(eta);
+
+ temp_cell.pi[0] = fluid_cell.pi[0][0];
+ temp_cell.pi[1] = fluid_cell.pi[0][1];
+ temp_cell.pi[2] = fluid_cell.pi[0][2];
+ temp_cell.pi[3] = fluid_cell.pi[0][3];
+ temp_cell.pi[4] = fluid_cell.pi[1][1];
+ temp_cell.pi[5] = fluid_cell.pi[1][2];
+ temp_cell.pi[6] = fluid_cell.pi[1][3];
+ temp_cell.pi[7] = fluid_cell.pi[2][2];
+ temp_cell.pi[8] = fluid_cell.pi[2][3];
+ temp_cell.pi[9] = fluid_cell.pi[3][3];
- for (int i = 0; i < 4; i++) {
- for (int j = 0; j < 4; j++) {
- temp_cell.pi[i][j] = fluid_cell.pi[i][j];
- }
- }
temp_cell.bulk_Pi = fluid_cell.bulk_Pi;
return (temp_cell);
diff --git a/src/framework/Version.h b/src/framework/Version.h
index 59638adf..3953c19c 100644
--- a/src/framework/Version.h
+++ b/src/framework/Version.h
@@ -20,8 +20,8 @@
namespace Jetscape {
- const std::string JetScapeVersion = "3.5.4";
- const std::string XscapeVersion = "1.0";
+ const std::string JetScapeVersion = "3.6";
+ const std::string XscapeVersion = "1.1";
} // end namespace Jetscape
diff --git a/src/hadronization/ColoredHadronization.cc b/src/hadronization/ColoredHadronization.cc
index b3aee2fa..14a11ff4 100644
--- a/src/hadronization/ColoredHadronization.cc
+++ b/src/hadronization/ColoredHadronization.cc
@@ -43,8 +43,8 @@ void ColoredHadronization::InitTask() {
GetXMLElementDouble({"JetHadronization", "eCMforHadronization"});
p_fake = p_read_xml;
- std::string weak_decays =
- GetXMLElementText({"JetHadronization", "weak_decays"});
+ /*std::string weak_decays =
+ GetXMLElementText({"JetHadronization", "weak_decays"});*/
VERBOSE(2) << "Start Hadronizing using the PYTHIA module...";
@@ -74,11 +74,34 @@ void ColoredHadronization::InitTask() {
pythia.readString("ProcessLevel:all = off");
pythia.readString("PartonLevel:FSR=off");
+
+ // General settings for hadron decays
+ std::string pythia_decays = GetXMLElementText({"JetHadronization", "pythia_decays"});
+ double tau0Max = 10.0;
+ double tau0Max_xml = GetXMLElementDouble({"JetHadronization", "tau0Max"});
+ if(tau0Max_xml >= 0){tau0Max = tau0Max_xml;}
+ else{JSWARN << "tau0Max should be larger than 0. Set it to 10.";}
+ if(pythia_decays == "on"){
+ JSINFO << "Pythia decays are turned on for tau0Max < " << tau0Max;
+ pythia.readString("HadronLevel:Decay = on");
+ pythia.readString("ParticleDecays:limitTau0 = on");
+ pythia.readString("ParticleDecays:tau0Max = " + std::to_string(tau0Max));
+ } else {
+ JSINFO << "Pythia decays are turned off";
+ pythia.readString("HadronLevel:Decay = off");
+ }
+
+ // Settings for decays (old flag, will be depracted at some point)
+ // This overwrites the previous settings if the user xml file contains the flag
+ std::string weak_decays =
+ GetXMLElementText({"JetHadronization", "weak_decays"});
if (weak_decays == "off") {
- JSINFO << "Weak decays are turned off";
+ JSINFO << "Hadron decays are turned off.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' instead.\nOverwriting 'pythia_decays'.";
pythia.readString("HadronLevel:Decay = off");
- } else {
- JSINFO << "Weak decays are turned on";
+ } else if(weak_decays == "on") {
+ JSINFO << "Hadron decays inside a range of 10 mm/c are turned on.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' and 'tau0Max' for more control on decays.\nOverwriting 'pythia_decays' and fix 'tau0Max' to 10.";
pythia.readString("HadronLevel:Decay = on");
pythia.readString("ParticleDecays:limitTau0 = on");
pythia.readString("ParticleDecays:tau0Max = 10.0");
@@ -86,7 +109,6 @@ void ColoredHadronization::InitTask() {
std::stringstream lines;
lines << GetXMLElementText({"JetHadronization", "LinesToRead"}, false);
- int i = 0;
while (std::getline(lines, s, '\n')) {
if (s.find_first_not_of(" \t\v\f\r") == s.npos)
continue; // skip empty lines
diff --git a/src/hadronization/ColorlessHadronization.cc b/src/hadronization/ColorlessHadronization.cc
index bad39392..dfa3bb88 100644
--- a/src/hadronization/ColorlessHadronization.cc
+++ b/src/hadronization/ColorlessHadronization.cc
@@ -56,8 +56,6 @@ void ColorlessHadronization::InitTask() {
GetXMLElementDouble({"JetHadronization", "eCMforHadronization"});
p_fake = p_read_xml;
- std::string weak_decays =
- GetXMLElementText({"JetHadronization", "weak_decays"});
take_recoil = GetXMLElementInt({"JetHadronization", "take_recoil"});
JSDEBUG << "Initialize ColorlessHadronization";
@@ -70,20 +68,35 @@ void ColorlessHadronization::InitTask() {
// Standard settings
pythia.readString("ProcessLevel:all = off");
-
- // Don't let pi0 decay
- //pythia.readString("111:mayDecay = off");
-
- // Don't let any hadron decay
- //pythia.readString("HadronLevel:Decay = off");
-
pythia.readString("PartonLevel:FSR=off");
+ // General settings for hadron decays
+ std::string pythia_decays = GetXMLElementText({"JetHadronization", "pythia_decays"});
+ double tau0Max = 10.0;
+ double tau0Max_xml = GetXMLElementDouble({"JetHadronization", "tau0Max"});
+ if(tau0Max_xml >= 0){tau0Max = tau0Max_xml;}
+ else{JSWARN << "tau0Max should be larger than 0. Set it to 10.";}
+ if(pythia_decays == "on"){
+ JSINFO << "Pythia decays are turned on for tau0Max < " << tau0Max;
+ pythia.readString("HadronLevel:Decay = on");
+ pythia.readString("ParticleDecays:limitTau0 = on");
+ pythia.readString("ParticleDecays:tau0Max = " + std::to_string(tau0Max));
+ } else {
+ JSINFO << "Pythia decays are turned off";
+ pythia.readString("HadronLevel:Decay = off");
+ }
+
+ // Settings for decays (old flag, will be depracted at some point)
+ // This overwrites the previous settings if the user xml file contains the flag
+ std::string weak_decays =
+ GetXMLElementText({"JetHadronization", "weak_decays"});
if (weak_decays == "off") {
- JSINFO << "Weak decays are turned off";
+ JSINFO << "Hadron decays are turned off.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' instead.\nOverwriting 'pythia_decays'.";
pythia.readString("HadronLevel:Decay = off");
- } else {
- JSINFO << "Weak decays are turned on";
+ } else if(weak_decays == "on") {
+ JSINFO << "Hadron decays inside a range of 10 mm/c are turned on.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' and 'tau0Max' for more control on decays.\nOverwriting 'pythia_decays' and fix 'tau0Max' to 10.";
pythia.readString("HadronLevel:Decay = on");
pythia.readString("ParticleDecays:limitTau0 = on");
pythia.readString("ParticleDecays:tau0Max = 10.0");
@@ -91,7 +104,6 @@ void ColorlessHadronization::InitTask() {
std::stringstream lines;
lines << GetXMLElementText({"JetHadronization", "LinesToRead"}, false);
- int i = 0;
while (std::getline(lines, s, '\n')) {
if (s.find_first_not_of(" \t\v\f\r") == s.npos)
continue; // skip empty lines
diff --git a/src/hadronization/HybridHadronization.cc b/src/hadronization/HybridHadronization.cc
index 5da49050..8463fac8 100644
--- a/src/hadronization/HybridHadronization.cc
+++ b/src/hadronization/HybridHadronization.cc
@@ -2,7 +2,7 @@
* Copyright (c) The JETSCAPE Collaboration, 2019
*
* Modular, task-based framework for simulating all aspects of heavy-ion collisions
- *
+ *
* For the list of contributors see AUTHORS.
*
* Report issues at https://github.com/JETSCAPE/JETSCAPE/issues
@@ -14,702 +14,990 @@
******************************************************************************/
#include "HybridHadronization.h"
+#include "ThermPtnSampler.h"
#include "JetScapeXML.h"
#include "JetScapeLogger.h"
#include "tinyxml2.h"
#include "JetScapeConstants.h"
+
+#include "FluidDynamics.h"
+#include "FluidCellInfo.h"
+#include "SurfaceCellInfo.h"
+#include "FluidEvolutionHistory.h"
+#include "JetScapeSignalManager.h"
+
#include
#include
#include
#include
#include
#include
+#include
+//#include
using namespace Jetscape;
using namespace Pythia8;
// Register the module with the base class
-RegisterJetScapeModule
- HybridHadronization::reg("HybridHadronization");
+RegisterJetScapeModule HybridHadronization::reg("HybridHadronization");
// Initialize static helper here
-Pythia8::Pythia HybridHadronization::pythia("IntentionallyEmpty", false);
+Pythia8::Pythia HybridHadronization::pythia ("IntentionallyEmpty",false);
//RNG - Mersenne Twist - 64 bit
//std::mt19937_64 eng(std::random_device{}());
//std::mt19937_64 eng(1);
//returns a random number between 0 and 1, based on above engine
double HybridHadronization::ran() {
- std::uniform_real_distribution uniran(0.0, 1.0);
+ std::uniform_real_distribution uniran(0.0,1.0);
return uniran(eng);
}
-HybridHadronization::HybridHadronization() {
+HybridHadronization::HybridHadronization(){
SetId("HybridHadronization");
VERBOSE(8);
}
-HybridHadronization::~HybridHadronization() { VERBOSE(8); }
+HybridHadronization::~HybridHadronization(){
+ VERBOSE(8);
+}
-//meson&baryon width functions
-double HybridHadronization::SigM2_calc(double R2chg, double qm1, double qm2,
- double qq1, double qq2) {
- return R2chg * (2. / 3.) * (qm1 + qm2) * (qm1 + qm2) /
- (std::abs(qq1 * qm2 * qm2 + -1. * qq2 * qm1 * qm1));
+//meson width function
+double HybridHadronization::SigM2_calc(double R2chg, double qm1, double qm2, double qq1, double qq2){
+ return R2chg*(2./3.)*(qm1+qm2)*(qm1+qm2)/(std::abs(qq1)*qm2*qm2 + std::abs(qq2)*qm1*qm1) * (std::abs(qq1) + std::abs(qq2));
}
-//CAREFUL, qmi's and qqi's need to be consistent for BOTH of these (mass1(2)(3) is the same for both!)...
-double HybridHadronization::SigBR2_calc(double R2chg, double qm1, double qm2,
- double qm3, double qq1, double qq2,
- double qq3) {
- return R2chg * 2. * (2. / 3.) * (2. / 3.) * (qm1 + qm2 + qm3) *
- (qm1 + qm2 + qm3) /
- ((qm1 + qm2) *
- std::abs(qq1 * (qm2 + qm3) * (qm3 / qm1) +
- qq2 * (qm1 + qm3) * (qm3 / qm2) + qq3 * (qm1 + qm2)));
+
+//baryon width function
+double HybridHadronization::SigBR2_calc(double R2chg, double qm1, double qm2, double qm3, double qq1, double qq2, double qq3){
+ return R2chg*(2./3.)*(qm1+qm2+qm3)/(std::abs(qq1)*qm2*(qm2+qm3)/(qm1+qm2)+std::abs(qq2)*qm1*(qm1+qm3)/(qm1+qm2)+std::abs(qq3)*(qm1*qm2)/qm3) * (std::abs(qq1) + std::abs(qq2) + std::abs(qq3));
}
-double HybridHadronization::SigBL2_calc(double SigBR2, double qm1, double qm2,
- double qm3) {
- return SigBR2 * ((qm1 * qm2) / (qm1 + qm2)) /
- (qm3 * (qm1 + qm2) / (qm1 + qm2 + qm3));
+
+double HybridHadronization::SigBL2_calc(double SigBR2, double qm1, double qm2, double qm3){
+ return SigBR2*((qm1*qm2)/(qm1+qm2))/(qm3*(qm1+qm2)/(qm1+qm2+qm3));
}
-void HybridHadronization::InitTask() {
+void HybridHadronization::InitTask(){
tinyxml2::XMLElement *hadronization = GetXMLElement({"JetHadronization"});
- if (!hadronization) {
+ if ( !hadronization ) {
JSWARN << "Couldn't find tag Jet Hadronization";
- throw std::runtime_error("Couldn't find tag Jet Hadronization");
+ throw std::runtime_error ("Couldn't find tag Jet Hadronization");
}
if (hadronization) {
- std::string s = GetXMLElementText({"JetHadronization", "name"});
- JSDEBUG << s << " to be initialized ...";
- JSINFO << "Initialize Hybrid Hadronization ...";
+ string s = hadronization->FirstChildElement( "name" )->GetText();
+ //std::string s = GetXMLElementText({"JetHadronization", "name"});
+ JSDEBUG << s << " to be initialized ...";
+ JSINFO<<"Initialize Hybrid Hadronization ...";
- JSDEBUG << "Initialize HybridHadronization";
+ JSDEBUG<<"Initialize HybridHadronization";
VERBOSE(8);
- maxE_level =
- 3; //maximum energy level considered for the recombination (was 3 in recent fortran code, prev. set to 8)
- gmax = 1.25; //maximum allowed mass of the gluon (for q-qbar split), in GeV
- xmq = 0.33; //light quark mass, in GeV
- xms = 0.5; //strange quark mass, in GeV
- hbarc = 0.197327; // GeV*fm - maybe just set this as a constant in common?
- dist2cut =
- 25.; //maximum distance [fm] squared for recombination (involving thermal partons) - in lab frame
- sh_recofactor =
- 1. /
- 3.; //suppression/enhancement factor for shower-shower recombination
- th_recofactor =
- 1. /
- 3.; //suppression/enhancement factor for shower-thermal recombination
- attempts_max =
- 10; //maximum number of failed attempts to hadronize a single event before we give up.
- p_fake = 0.; //momentum used for fake parton, if needed
- rand_seed =
- 0; //seed for RNGs used - 0 means use a randomly determined random seed (from system time or std::random_device{}())
-
- //xml read in to alter settings...
- double xml_doublein = -1.;
- int xml_intin = -1;
- unsigned int xml_uintin = std::numeric_limits::max();
-
- xml_doublein =
- GetXMLElementDouble({"JetHadronization", "eCMforHadronization"});
- if (xml_doublein >= 0.) {
- p_fake = xml_doublein;
- }
- xml_doublein = -1.;
-
- xml_doublein =
- GetXMLElementDouble({"JetHadronization", "thermreco_distmax"});
- if (xml_doublein >= 0.) {
- dist2cut = xml_doublein * xml_doublein;
- }
- xml_doublein = -1.;
-
- xml_doublein =
- GetXMLElementDouble({"JetHadronization", "shower_recofactor"});
- if (xml_doublein >= 0.) {
- sh_recofactor = xml_doublein;
- }
- xml_doublein = -1.;
-
- xml_doublein =
- GetXMLElementDouble({"JetHadronization", "thermal_recofactor"});
- if (xml_doublein >= 0.) {
- th_recofactor = xml_doublein;
- }
- xml_doublein = -1.;
-
- xml_intin = GetXMLElementInt({"JetHadronization", "reco_Elevelmax"});
- if (xml_intin >= 0) {
- maxE_level = xml_intin;
- }
- xml_intin = -1;
-
- // random seed
- // xml limits us to unsigned int :-/ -- but so does 32 bits Mersenne Twist
- tinyxml2::XMLElement *RandomXmlDescription = GetXMLElement({"Random"});
- if (RandomXmlDescription) {
- tinyxml2::XMLElement *xmle;
- xmle = RandomXmlDescription->FirstChildElement("seed");
- if (xmle) {
- xmle->QueryUnsignedText(&xml_uintin);
- } else {
- JSWARN << "No element found for element in xml, "
- "seeding to 0";
- }
- if (xml_uintin < std::numeric_limits::max()) {
- rand_seed = xml_uintin;
- }
- xml_intin = std::numeric_limits::max();
- } else {
- JSWARN << "No element found in xml, seeding to 0";
- }
- VERBOSE(7) << "Seeding PYTHIA(hadronization) to " << rand_seed;
-
- //not sure if we can seed with a negative integer...
- if (rand_seed != 0) {
- eng.seed(rand_seed);
- }
- //else{eng.seed(std::random_device{}());}
- else { //seeding the mt19937_64 object 'eng' PROPERLY!
- std::random_device rd;
- std::array seedarray;
- std::generate_n(seedarray.data(), seedarray.size(), std::ref(rd));
- std::seed_seq seeds(std::begin(seedarray), std::end(seedarray));
- eng.seed(seeds);
- }
-
- //Since PYTHIA has no spacetime information, we shouldn't use recombination as it is necessary to calculate recombination probabilities
- //later, this will instead update to set a flag to attempt to artificially generate this information
- //for now, we just print a warning - but still try to run recombination. It will just be unphysically enhanced, esp. for certain configurations
- //tinyxml2::XMLElement *XmlPythiaGun=JetScapeXML::Instance()->GetXMLRoot()->FirstChildElement("Hard" )->FirstChildElement("PythiaGun");
- tinyxml2::XMLElement *XmlPythiaGun = GetXMLElement({"Hard", "PythiaGun"});
- bool PYgun_FSRon = false;
- if (XmlPythiaGun) {
- tinyxml2::XMLElement *xmle;
- xmle = XmlPythiaGun->FirstChildElement("FSR_on");
- if (xmle) {
- xmle->QueryIntText(&xml_intin);
- }
- if (xml_intin == 1) {
- PYgun_FSRon = true;
- }
- xml_intin = -1;
- }
- if (PYgun_FSRon && (sh_recofactor > 0.0000000001)) {
- JSWARN
- << "Recombination with a PYTHIA FSR shower is not fully implemented.";
- }
-
- //quark masses/charges used to get widths from charged radii
- double Qm_ud = 0.338;
- double Qm_s = 0.486;
- double Qm_c = 1.55;
- double Qm_b = 4.73;
- double chg_u = 2. / 3.;
- double chg_d = -1. / 3.;
-
- //rms charge radii
- //mesons
- double R2chg_Pi = 0.42;
- double R2chg_Phi = 0.21;
- double R2chg_K = 0.34;
- double R2chg_Jpi = 0.04;
- double R2chg_Ds = 0.09;
- double R2chg_D = 0.165;
- double R2chg_Ups =
- 0.032; //????? setting to a linear extrapolation from Jpsi -> Bc -> Ups
- double R2chg_Bc = 0.036;
- double R2chg_B = 0.273;
- //baryons
- double R2chg_Nuc = 0.69;
- double R2chg_Omg = 0.355;
- double R2chg_Xi = 0.52;
- double R2chg_Sig = 0.61;
- double R2chg_Occc = 0.179; //?????
- double R2chg_Occ = 0.043; //?
- double R2chg_Xicc = 0.049; //?
- double R2chg_Oc = 0.1; //?????
- double R2chg_Xic = 0.24; //?
- double R2chg_Sigc = 0.27; //?
- double R2chg_Obbb = 0.001; //?????
- double R2chg_Obbc = 0.001; //?????
- double R2chg_Obb = 0.001; //?????
- double R2chg_Xibb = 0.001; //?????
- double R2chg_Obcc = 0.001; //?????
- double R2chg_Obc = 0.001; //?????
- double R2chg_Xibc = 0.001; //?????
- double R2chg_Ob = 0.6; //?????
- double R2chg_Xib = 0.63; //?????
- double R2chg_Sigb = 0.66; //?????
-
- //meson width calculations (r2) - recalc if r2chg is changed on command line...
- SigPi2 = SigM2_calc(R2chg_Pi, Qm_ud, Qm_ud, chg_d, chg_u);
- SigPhi2 = SigM2_calc(R2chg_Phi, Qm_s, Qm_s, chg_d, -chg_d) *
- (2. / 3.); //normalizing
- SigK2 = SigM2_calc(R2chg_K, Qm_s, Qm_ud, chg_d, chg_u);
- SigJpi2 = SigM2_calc(R2chg_Jpi, Qm_c, Qm_c, chg_u, -chg_u) *
- (4. / 3.); //normalizing
- SigDs2 = SigM2_calc(R2chg_Ds, Qm_c, Qm_s, chg_u, chg_d);
- SigD2 = SigM2_calc(R2chg_D, Qm_c, Qm_ud, chg_u, chg_d);
- SigUps2 = SigM2_calc(R2chg_Ups, Qm_b, Qm_b, chg_d, -chg_d) *
- (2. / 3.); //normalizing
- SigBc2 = SigM2_calc(R2chg_Bc, Qm_b, Qm_c, chg_d, chg_u);
- SigB2 =
- SigM2_calc(R2chg_B, Qm_b, Qm_ud, chg_d, chg_u); // (treating B_s as B)
-
- //baryon width calculations (r2) - recalc if r2chg is changed on command line...
- //light/strange baryons
- SigNucR2 = SigBR2_calc(R2chg_Nuc, Qm_ud, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
- SigNucL2 = SigBL2_calc(SigNucR2, Qm_ud, Qm_ud, Qm_ud);
- SigOmgR2 = SigBR2_calc(R2chg_Omg, Qm_s, Qm_s, Qm_s, chg_d, chg_d, chg_d);
- SigOmgL2 = SigBL2_calc(SigOmgR2, Qm_s, Qm_s, Qm_s);
- SigXiR2 = SigBR2_calc(R2chg_Xi, Qm_s, Qm_s, Qm_ud, chg_d, chg_d, chg_d);
- SigXiL2 = SigBL2_calc(SigXiR2, Qm_s, Qm_s, Qm_ud);
- SigSigR2 = SigBR2_calc(R2chg_Sig, Qm_s, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
- SigSigL2 = SigBL2_calc(SigSigR2, Qm_s, Qm_ud, Qm_ud);
-
- //charm baryons
- SigOcccR2 = SigBR2_calc(
- R2chg_Occc, Qm_c, Qm_c, Qm_c, chg_d, chg_d,
- chg_d); // ! maybe need to normalize? (just setting all to -1/3 for now)
- SigOcccL2 = SigBL2_calc(SigOcccR2, Qm_c, Qm_c, Qm_c);
- SigOccR2 = SigBR2_calc(R2chg_Occ, Qm_c, Qm_c, Qm_s, chg_u, chg_u, chg_d);
- SigOccL2 = SigBL2_calc(SigOccR2, Qm_c, Qm_c, Qm_s);
- SigXiccR2 = SigBR2_calc(R2chg_Xicc, Qm_c, Qm_c, Qm_ud, chg_u, chg_u, chg_d);
- SigXiccL2 = SigBL2_calc(SigXiccR2, Qm_c, Qm_c, Qm_ud);
- SigOcR2 = SigBR2_calc(R2chg_Oc, Qm_c, Qm_s, Qm_s, chg_d, chg_d,
- chg_d); // ! setting all quark charges to -1/3
- SigOcL2 = SigBL2_calc(SigOcR2, Qm_c, Qm_s, Qm_s);
- SigXicR2 = SigBR2_calc(R2chg_Xic, Qm_c, Qm_s, Qm_ud, chg_u, chg_d, chg_u);
- SigXicL2 = SigBL2_calc(SigXicR2, Qm_c, Qm_s, Qm_ud);
- SigSigcR2 =
- SigBR2_calc(R2chg_Sigc, Qm_c, Qm_ud, Qm_ud, chg_u, chg_d, chg_u);
- SigSigcL2 = SigBL2_calc(SigSigcR2, Qm_c, Qm_ud, Qm_ud);
-
- //bottom baryons
- SigObbbR2 = SigBR2_calc(R2chg_Obbb, Qm_b, Qm_b, Qm_b, chg_d, chg_d, chg_d);
- SigObbbL2 = SigBL2_calc(SigObbbR2, Qm_b, Qm_b, Qm_b);
- SigObbcR2 = SigBR2_calc(R2chg_Obbc, Qm_b, Qm_b, Qm_c, chg_d, chg_d,
- chg_d); // ! setting all quark charges to -1/3
- SigObbcL2 = SigBL2_calc(SigObbcR2, Qm_b, Qm_b, Qm_c);
- SigObbR2 = SigBR2_calc(R2chg_Obb, Qm_b, Qm_b, Qm_s, chg_d, chg_d, chg_d);
- SigObbL2 = SigBL2_calc(SigObbR2, Qm_b, Qm_b, Qm_s);
- SigXibbR2 = SigBR2_calc(R2chg_Xibb, Qm_b, Qm_b, Qm_ud, chg_d, chg_d, chg_d);
- SigXibbL2 = SigBL2_calc(SigXibbR2, Qm_b, Qm_b, Qm_ud);
- SigObccR2 = SigBR2_calc(R2chg_Obcc, Qm_b, Qm_c, Qm_c, chg_d, chg_u, chg_u);
- SigObccL2 = SigBL2_calc(SigObccR2, Qm_b, Qm_c, Qm_c);
- SigObcR2 = SigBR2_calc(R2chg_Obc, Qm_b, Qm_c, Qm_s, chg_d, chg_d,
- chg_d); // ! flipping c quark charge (all to -1/3)
- SigObcL2 = SigBL2_calc(SigObcR2, Qm_b, Qm_c, Qm_s);
- SigXibcR2 = SigBR2_calc(R2chg_Xibc, Qm_b, Qm_c, Qm_ud, chg_d, chg_u, chg_u);
- SigXibcL2 = SigBL2_calc(SigXibcR2, Qm_b, Qm_c, Qm_ud);
- SigObR2 = SigBR2_calc(R2chg_Ob, Qm_b, Qm_s, Qm_s, chg_d, chg_d, chg_d);
- SigObL2 = SigBL2_calc(SigObR2, Qm_b, Qm_s, Qm_s);
- SigXibR2 = SigBR2_calc(R2chg_Xib, Qm_b, Qm_s, Qm_ud, chg_d, chg_d, chg_d);
- SigXibL2 = SigBL2_calc(SigXibR2, Qm_b, Qm_s, Qm_ud);
- SigSigbR2 =
- SigBR2_calc(R2chg_Sigb, Qm_b, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
- SigSigbL2 = SigBL2_calc(SigSigbR2, Qm_b, Qm_ud, Qm_ud);
+ maxM_level = 1; //maximum energy level considered for meson recombination; maximum: 4
+ maxB_level = 1; //maximum energy level considered for baryon recombination; no maximum but no physical baryons beyond 0
+ goldstonereco = false; // don't allow recombination of Goldstone bosons
+ gmax = 1.25; //maximum allowed mass of the gluon (for q-qbar split), in GeV
+ xmq = 0.33; //light quark mass, in GeV
+ xms = 0.5; //strange quark mass, in GeV
+ xmc = 1.5; //charm quark mass in GeV
+ xmb = 4.8; //bottom quark mass in GeV
+ hbarc = 0.197327; // GeV*fm - maybe just set this as a constant in common?
+ dist2cut = 25.; //maximum distance [fm] squared for recombination (involving thermal partons) - in lab frame
+ sh_recofactor = 1.; //suppression/enhancement factor for shower-shower recombination
+ th_recofactor = 1.; //suppression/enhancement factor for shower-thermal recombination
+ attempts_max = 15; //maximum number of failed attempts to hadronize a single event before we give up.
+ p_fake = 0.; //momentum used for fake parton, if needed
+ rand_seed = 0; //seed for RNGs used - 0 means use a randomly determined random seed (from system time or std::random_device{}())
+ had_prop = 0.; //propagation of hadrons after formation by this time in lab frame
+ part_prop = 0.; //minimum propagation time of partons after last split
+ reco_hadrons_pythia = 0; //flag to put recombination hadrons into pythia for decays (position would be lost)
+
+ //xml read in to alter settings...
+ double xml_doublein = -1.; int xml_intin = -1; unsigned int xml_uintin = std::numeric_limits::max();
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "eCMforHadronization"});
+ if(xml_doublein >= 0.){p_fake = xml_doublein / 6.;} xml_doublein = -1.; // for colliders set fake momentum to valence quark energy
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "thermreco_distmax"});
+ if(xml_doublein >= 0.){dist2cut = xml_doublein*xml_doublein;} xml_doublein = -1.;
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "shower_recofactor"});
+ if(xml_doublein >= 0.){sh_recofactor = xml_doublein;} xml_doublein = -1.;
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "thermal_recofactor"});
+ if(xml_doublein >= 0.){th_recofactor = xml_doublein;} xml_doublein = -1.;
+
+ xml_intin = GetXMLElementInt({"JetHadronization", "reco_Mlevelmax"});
+ if(xml_intin >= 0){maxM_level = xml_intin;} xml_intin = -1;
+
+ xml_intin = GetXMLElementInt({"JetHadronization", "reco_Blevelmax"});
+ if(xml_intin >= -1){maxB_level = xml_intin;} xml_intin = -1;
+
+ xml_intin = GetXMLElementInt({"JetHadronization", "reco_goldstone"});
+ if(xml_intin >= 0){goldstonereco = xml_intin;} xml_intin = -1;
+
+ torder_reco = false;
+ xml_intin = GetXMLElementInt({"JetHadronization", "recobias_t"});
+ if(xml_intin == 1){torder_reco = true;} xml_intin = -1;
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "hydro_Tc"});
+ if(xml_doublein >= 0){hydro_Tc = xml_doublein;} xml_doublein = -1;
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "had_postprop"});
+ if(xml_doublein >= 0){had_prop = xml_doublein;} xml_doublein = -1;
+
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "part_prop"});
+ if(xml_doublein >= 0){part_prop = xml_doublein;} xml_doublein = -1;
+
+ xml_intin = GetXMLElementInt({"JetHadronization", "reco_hadrons_in_pythia"});
+ if(xml_intin == 0 || xml_intin == 1){reco_hadrons_pythia = xml_intin;} xml_intin = -1;
+
+ xml_intin = GetXMLElementInt({"Afterburner", "include_fragmentation_hadrons"});
+ if(xml_intin == 1){afterburner_frag_hadrons = true;} xml_intin = -1;
+
+ if(maxM_level > 4) {
+ maxM_level=4;
+ JSWARN << "Requested maximum energy level for mesons too large. Set it to 4.";
+ }
+ // No, really, the maximum is four
+
+ // random seed
+ // xml limits us to unsigned int :-/ -- but so does 32 bits Mersenne Twist
+ tinyxml2::XMLElement *RandomXmlDescription = GetXMLElement({"Random"});
+ if ( RandomXmlDescription ){
+ tinyxml2::XMLElement *xmle; xmle = RandomXmlDescription->FirstChildElement( "seed" );
+ xmle->QueryUnsignedText(&xml_uintin);
+ if(xml_uintin < std::numeric_limits::max()){rand_seed = xml_uintin;} xml_uintin = std::numeric_limits::max();
+ }else{
+ JSWARN << "No element found in xml, seeding to 0";
+ }
+ VERBOSE(7) <<"Seeding PYTHIA(hadronization) to "<< rand_seed;
+
+ //not sure if we can seed with a negative integer...
+ if(rand_seed != 0){eng.seed(rand_seed);}
+ //else{eng.seed(std::random_device{}());}
+ else{ //seeding the mt19937_64 object 'eng' PROPERLY!
+ std::random_device rd;
+ std::array seedarray; std::generate_n(seedarray.data(), seedarray.size(), std::ref(rd));
+ std::seed_seq seeds(std::begin(seedarray), std::end(seedarray)); eng.seed(seeds);
+ }
+
+
+ //Since PYTHIA has no spacetime information, we shouldn't use recombination as it is necessary to calculate recombination probabilities
+ //later, this will instead update to set a flag to attempt to artificially generate this information
+ //for now, we just print a warning - but still try to run recombination. It will just be unphysically enhanced, esp. for certain configurations
+ //tinyxml2::XMLElement *XmlPythiaGun=JetScapeXML::Instance()->GetXMLRoot()->FirstChildElement("Hard" )->FirstChildElement("PythiaGun");
+ /* tinyxml2::XMLElement *XmlPythiaGun = GetXMLElement({"Hard", "PythiaGun"});
+ bool PYgun_FSRon = false;
+ if ( XmlPythiaGun ){
+ tinyxml2::XMLElement *xmle; xmle = XmlPythiaGun->FirstChildElement( "FSR_on" );
+ xmle->QueryIntText(&xml_intin);
+ if(xml_intin == 1){PYgun_FSRon = true;} xml_intin = -1;
+ }
+ if(PYgun_FSRon && (sh_recofactor > 0.0000000001)){JSWARN << "Recombination with a PYTHIA FSR shower is not fully implemented.";}
+ */
+ xml_intin = GetXMLElementInt({"Hard", "PythiaGun", "FSR_on"});
+ if(xml_intin && (sh_recofactor > 0.0000000001)){JSWARN << "Recombination with a PYTHIA FSR shower is not fully implemented.";}
+ xml_intin = -1;
+
+ //quark masses/charges used to get widths from charged radii (use Pythia masses)
+ double Qm_ud = xmq; double Qm_s = xms; double Qm_c = xmc; double Qm_b = xmb;
+ double chg_u = 2./3.; double chg_d = -1./3.;
+
+ //rms charge radii
+ //mesons
+ double R2chg_Pi = 0.42 ;
+ double R2chg_Phi = 0.21 ;
+ double R2chg_K = 0.34 ;
+ double R2chg_Jpi = 0.04 ;
+ double R2chg_Ds = 0.09 ;
+ double R2chg_D = 0.165;
+ double R2chg_Ups = 0.032; //????? setting to a linear extrapolation from Jpsi -> Bc -> Ups
+ double R2chg_Bc = 0.036;
+ double R2chg_B = 0.273;
+ //baryons
+ double R2chg_Nuc = 0.69 ;
+ double R2chg_Omg = 0.355;
+ double R2chg_Xi = 0.52 ;
+ double R2chg_Sig = 0.61 ;
+ double R2chg_Occc = 0.179; //?????
+ double R2chg_Occ = 0.043; //?
+ double R2chg_Xicc = 0.049; //?
+ double R2chg_Oc = 0.1 ; //?????
+ double R2chg_Xic = 0.24 ; //?
+ double R2chg_Sigc = 0.27 ; //?
+ double R2chg_Obbb = 0.001; //?????
+ double R2chg_Obbc = 0.001; //?????
+ double R2chg_Obb = 0.001; //?????
+ double R2chg_Xibb = 0.001; //?????
+ double R2chg_Obcc = 0.001; //?????
+ double R2chg_Obc = 0.001; //?????
+ double R2chg_Xibc = 0.001; //?????
+ double R2chg_Ob = 0.6 ; //?????
+ double R2chg_Xib = 0.63 ; //?????
+ double R2chg_Sigb = 0.66 ; //?????
+
+ //meson width calculations (r2) - recalc if r2chg is changed on command line...
+ SigPi2 = SigM2_calc(R2chg_Pi, Qm_ud, Qm_ud, chg_d, chg_u);
+ SigPhi2 = SigM2_calc(R2chg_Phi, Qm_s, Qm_s, chg_d, -chg_d); //normalizing
+ SigK2 = SigM2_calc(R2chg_K, Qm_s, Qm_ud, chg_d, chg_u);
+ SigJpi2 = SigM2_calc(R2chg_Jpi, Qm_c, Qm_c, chg_u, -chg_u); //normalizing
+ SigDs2 = SigM2_calc(R2chg_Ds, Qm_c, Qm_s, chg_u, chg_d);
+ SigD2 = SigM2_calc(R2chg_D, Qm_c, Qm_ud, chg_u, chg_d);
+ SigUps2 = SigM2_calc(R2chg_Ups, Qm_b, Qm_b, chg_d, -chg_d); //normalizing
+ SigBc2 = SigM2_calc(R2chg_Bc, Qm_b, Qm_c, chg_d, chg_u);
+ SigB2 = SigM2_calc(R2chg_B, Qm_b, Qm_ud, chg_d, chg_u); // (treating B_s as B)
+
+ //baryon width calculations (r2) - recalc if r2chg is changed on command line...
+ //light/strange baryons
+ SigNucR2 = SigBR2_calc(R2chg_Nuc, Qm_ud, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
+ SigNucL2 = SigBL2_calc(SigNucR2, Qm_ud, Qm_ud, Qm_ud);
+ SigOmgR2 = SigBR2_calc(R2chg_Omg, Qm_s, Qm_s, Qm_s, chg_d, chg_d, chg_d);
+ SigOmgL2 = SigBL2_calc(SigOmgR2, Qm_s, Qm_s, Qm_s );
+ SigXiR2 = SigBR2_calc(R2chg_Xi, Qm_s, Qm_s, Qm_ud, chg_d, chg_d, chg_d);
+ SigXiL2 = SigBL2_calc(SigXiR2, Qm_s, Qm_s, Qm_ud);
+ SigSigR2 = SigBR2_calc(R2chg_Sig, Qm_s, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
+ SigSigL2 = SigBL2_calc(SigSigR2, Qm_s, Qm_ud, Qm_ud);
+
+ //charm baryons
+ SigOcccR2 = SigBR2_calc(R2chg_Occc, Qm_c, Qm_c, Qm_c, chg_d, chg_d, chg_d); // ! maybe need to normalize? (just setting all to -1/3 for now)
+ SigOcccL2 = SigBL2_calc(SigOcccR2, Qm_c, Qm_c, Qm_c );
+ SigOccR2 = SigBR2_calc(R2chg_Occ, Qm_c, Qm_c, Qm_s, chg_u, chg_u, chg_d);
+ SigOccL2 = SigBL2_calc(SigOccR2, Qm_c, Qm_c, Qm_s );
+ SigXiccR2 = SigBR2_calc(R2chg_Xicc, Qm_c, Qm_c, Qm_ud, chg_u, chg_u, chg_d);
+ SigXiccL2 = SigBL2_calc(SigXiccR2, Qm_c, Qm_c, Qm_ud);
+ SigOcR2 = SigBR2_calc(R2chg_Oc, Qm_c, Qm_s, Qm_s, chg_d, chg_d, chg_d); // ! setting all quark charges to -1/3
+ SigOcL2 = SigBL2_calc(SigOcR2, Qm_c, Qm_s, Qm_s );
+ SigXicR2 = SigBR2_calc(R2chg_Xic, Qm_c, Qm_s, Qm_ud, chg_u, chg_d, chg_u);
+ SigXicL2 = SigBL2_calc(SigXicR2, Qm_c, Qm_s, Qm_ud);
+ SigSigcR2 = SigBR2_calc(R2chg_Sigc, Qm_c, Qm_ud, Qm_ud, chg_u, chg_d, chg_u);
+ SigSigcL2 = SigBL2_calc(SigSigcR2, Qm_c, Qm_ud, Qm_ud);
+
+ //bottom baryons
+ SigObbbR2 = SigBR2_calc(R2chg_Obbb, Qm_b, Qm_b, Qm_b, chg_d, chg_d, chg_d);
+ SigObbbL2 = SigBL2_calc(SigObbbR2, Qm_b, Qm_b, Qm_b );
+ SigObbcR2 = SigBR2_calc(R2chg_Obbc, Qm_b, Qm_b, Qm_c, chg_d, chg_d, chg_d); // ! setting all quark charges to -1/3
+ SigObbcL2 = SigBL2_calc(SigObbcR2, Qm_b, Qm_b, Qm_c );
+ SigObbR2 = SigBR2_calc(R2chg_Obb, Qm_b, Qm_b, Qm_s, chg_d, chg_d, chg_d);
+ SigObbL2 = SigBL2_calc(SigObbR2, Qm_b, Qm_b, Qm_s );
+ SigXibbR2 = SigBR2_calc(R2chg_Xibb, Qm_b, Qm_b, Qm_ud, chg_d, chg_d, chg_d);
+ SigXibbL2 = SigBL2_calc(SigXibbR2, Qm_b, Qm_b, Qm_ud);
+ SigObccR2 = SigBR2_calc(R2chg_Obcc, Qm_b, Qm_c, Qm_c, chg_d, chg_u, chg_u);
+ SigObccL2 = SigBL2_calc(SigObccR2, Qm_b, Qm_c, Qm_c );
+ SigObcR2 = SigBR2_calc(R2chg_Obc, Qm_b, Qm_c, Qm_s, chg_d, chg_d, chg_d); // ! flipping c quark charge (all to -1/3)
+ SigObcL2 = SigBL2_calc(SigObcR2, Qm_b, Qm_c, Qm_s );
+ SigXibcR2 = SigBR2_calc(R2chg_Xibc, Qm_b, Qm_c, Qm_ud, chg_d, chg_u, chg_u);
+ SigXibcL2 = SigBL2_calc(SigXibcR2, Qm_b, Qm_c, Qm_ud);
+ SigObR2 = SigBR2_calc(R2chg_Ob, Qm_b, Qm_s, Qm_s, chg_d, chg_d, chg_d);
+ SigObL2 = SigBL2_calc(SigObR2, Qm_b, Qm_s, Qm_s );
+ SigXibR2 = SigBR2_calc(R2chg_Xib, Qm_b, Qm_s, Qm_ud, chg_d, chg_d, chg_d);
+ SigXibL2 = SigBL2_calc(SigXibR2, Qm_b, Qm_s, Qm_ud);
+ SigSigbR2 = SigBR2_calc(R2chg_Sigb, Qm_b, Qm_ud, Qm_ud, chg_d, chg_u, chg_u);
+ SigSigbL2 = SigBL2_calc(SigSigbR2, Qm_b, Qm_ud, Qm_ud);
// No event record printout.
pythia.readString("Next:numberShowInfo = 0");
pythia.readString("Next:numberShowProcess = 0");
pythia.readString("Next:numberShowEvent = 0");
-
- // Show initialization at DEBUG or high verbose level
pythia.readString("Init:showProcesses = off");
pythia.readString("Init:showChangedSettings = off");
pythia.readString("Init:showMultipartonInteractions = off");
pythia.readString("Init:showChangedParticleData = off");
- if (JetScapeLogger::Instance()->GetDebug() ||
- JetScapeLogger::Instance()->GetVerboseLevel() > 2) {
- pythia.readString("Init:showProcesses = on");
- pythia.readString("Init:showChangedSettings = on");
- pythia.readString("Init:showMultipartonInteractions = on");
- pythia.readString("Init:showChangedParticleData = on");
+
+ // Standard settings
+ pythia.readString("ProcessLevel:all = off");
+ //pythia.readString("PartonLevel:FSR=off"); //is this necessary?
+
+ // General settings for hadron decays
+ pythia_decays = GetXMLElementText({"JetHadronization", "pythia_decays"});
+ double tau0Max = 10.0;
+ double tau0Max_xml = GetXMLElementDouble({"JetHadronization", "tau0Max"});
+ if(tau0Max_xml >= 0){tau0Max = tau0Max_xml;}
+ else{JSWARN << "tau0Max should be larger than 0. Set it to 10.";}
+ if(pythia_decays == "on"){
+ JSINFO << "Pythia decays are turned on for tau0Max < " << tau0Max;
+ pythia.readString("HadronLevel:Decay = on");
+ pythia.readString("ParticleDecays:limitTau0 = on");
+ pythia.readString("ParticleDecays:tau0Max = " + std::to_string(tau0Max));
+ } else {
+ JSINFO << "Pythia decays are turned off";
+ pythia.readString("HadronLevel:Decay = off");
}
- // No event record printout.
- pythia.readString("Next:numberShowInfo = 0");
- pythia.readString("Next:numberShowProcess = 0");
- pythia.readString("Next:numberShowEvent = 0");
- if (JetScapeLogger::Instance()->GetDebug() ||
- JetScapeLogger::Instance()->GetVerboseLevel() > 2) {
- pythia.readString("Next:numberShowInfo = 1");
- pythia.readString("Next:numberShowProcess = 1");
- pythia.readString("Next:numberShowEvent = 1");
+ // Settings for decays (old flag, will be depracted at some point)
+ // This overwrites the previous settings if the user xml file contains the flag
+ std::string weak_decays =
+ GetXMLElementText({"JetHadronization", "weak_decays"});
+ if (weak_decays == "off") {
+ JSINFO << "Hadron decays are turned off.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' instead.\nOverwriting 'pythia_decays'.";
+ pythia.readString("HadronLevel:Decay = off");
+ } else if(weak_decays == "on") {
+ JSINFO << "Hadron decays inside a range of 10 mm/c are turned on.";
+ JSWARN << "This parameter will be depracted at some point. Use 'pythia_decays' and 'tau0Max' for more control on decays.\nOverwriting 'pythia_decays' and fix 'tau0Max' to 10.";
+ pythia.readString("HadronLevel:Decay = on");
+ pythia.readString("ParticleDecays:limitTau0 = on");
+ pythia.readString("ParticleDecays:tau0Max = 10.0");
}
- // Standard settings
- pythia.readString("ProcessLevel:all = off");
- //pythia.readString("PartonLevel:FSR=off"); //is this necessary?
-
- // Don't let pi0 decay
- //pythia.readString("111:mayDecay = off");
-
- // Don't let any hadron decay
- //pythia.readString("HadronLevel:Decay = off");
-
- //setting seed, or using random seed
- pythia.readString("Random:setSeed = on");
- pythia.readString("Random:seed = " + std::to_string(rand_seed));
-
- //additional settings
- //turning off pythia checks for runtime decrease (can be turned back on if necessary, but it shouldn't make much of a difference)
- pythia.readString(
- "Check:event = off"); // is probably a bad idea, but shouldn't really be necessary... will use a bit of runtime on event checks...
- pythia.readString(
- "Check:history = off"); // might be a good idea to set 'off' as it saves runtime - provided we know that we've set up mother/daughter relations correctly...
-
- //making the pythia event checks a little less stringent (PYTHIA documentation already states that LHC events will occasionally violate default constraint, without concern)
- //pythia.readString("Check:epTolWarn = 1e-4"); // setting E/P conservation violation constraint somewhat weaker, just for ease
- //pythia.readString("Check:epTolErr = 1e-2"); // setting E/P conservation violation constraint somewhat weaker, just for ease
- //pythia.readString("Check:mTolWarn = 1e-2"); // setting EP/M conservation violation constraint somewhat weaker, just for ease
- //pythia.readString("Check:mTolErr = 1e-1"); // setting EP/M conservation violation constraint somewhat weaker, just for ease
-
- //setting a decay threshold for subsequent hadron production
- pythia.readString(
- "ParticleDecays:limitTau0 = on"); //When on, only particles with tau0 < tau0Max are decayed
- //pythia.readString("ParticleDecays:tau0Max = 0.000003"); //The above tau0Max, expressed in mm/c :: default = 10. :: default, mayDecay()=true for tau0 below 1000 mm
- //set to 1E-17sec (in mm/c) to be smaller than pi0 lifetime
- pythia.readString("ParticleDecays:tau0Max = 10.0");
-
- //allowing for partonic space-time information to be used by PYTHIA
- pythia.readString(
- "PartonVertex:setVertex = on"); //this might allow PYTHIA to keep track of partonic space-time information (default was for 'rope hadronization')
-
- //using QCD based color reconnection (original PYTHIA MPI based CR can't be used at hadron level)
- pythia.readString(
- "ColourReconnection:reconnect = on"); //allowing color reconnections (should have been default on, but doing it here for clarity)
- pythia.readString(
- "ColourReconnection:mode = 1"); //sets the color reconnection scheme to 'new' QCD based scheme (TODO: make sure this is better than (2)gluon move)
- pythia.readString(
- "ColourReconnection:forceHadronLevelCR = on"); //allowing color reconnections for these constructed strings!
- //a few params for the QCD based color reconnection scheme are set below.
- pythia.readString(
- "MultipartonInteractions:pT0Ref = 2.15"); //not sure if this is needed for this setup, but is part of the recommended 'default'
- pythia.readString(
- "ColourReconnection:allowDoubleJunRem = off"); //default on - allows directly connected double junction systems to split into two strings
- pythia.readString("ColourReconnection:junctionCorrection = 1.15");
- pythia.readString(
- "ColourReconnection:timeDilationMode = 3"); //allow reconnection if single pair of dipoles are in causal contact (maybe try 5 as well?)
- pythia.readString(
- "ColourReconnection:timeDilationPar = 0.18"); //parameter used in causal interaction between strings (mode set above)(maybe try 0.073?)
+ //setting seed, or using random seed
+ pythia.readString("Random:setSeed = on");
+ pythia.readString("Random:seed = " + std::to_string(rand_seed));
+
+ //additional settings
+ //turning off pythia checks for runtime decrease (can be turned back on if necessary, but it shouldn't make much of a difference)
+ pythia.readString("Check:event = off"); // is probably a bad idea, but shouldn't really be necessary... will use a bit of runtime on event checks...
+ pythia.readString("Check:history = off"); // might be a good idea to set 'off' as it saves runtime - provided we know that we've set up mother/daughter relations correctly...
+
+ //making the pythia event checks a little less stringent (PYTHIA documentation already states that LHC events will occasionally violate default constraint, without concern)
+ //pythia.readString("Check:epTolWarn = 1e-4"); // setting E/P conservation violation constraint somewhat weaker, just for ease
+ //pythia.readString("Check:epTolErr = 1e-2"); // setting E/P conservation violation constraint somewhat weaker, just for ease
+ //pythia.readString("Check:mTolWarn = 1e-2"); // setting EP/M conservation violation constraint somewhat weaker, just for ease
+ //pythia.readString("Check:mTolErr = 1e-1"); // setting EP/M conservation violation constraint somewhat weaker, just for ease
+
+ //allowing for partonic space-time information to be used by PYTHIA
+ //pythia.readString("PartonVertex:setVertex = on"); //this might allow PYTHIA to keep track of partonic space-time information (default was for 'rope hadronization')
+
+ //setting hadron color tags so that spacetime information can be reconstructed
+ pythia.readString("StringFragmentation:TraceColours = on");
+
+ //using QCD based color reconnection (original PYTHIA MPI based CR can't be used at hadron level)
+ pythia.readString("ColourReconnection:reconnect = off"); //allowing color reconnections (should have been default on, but doing it here for clarity)
+ /*pythia.readString("ColourReconnection:mode = 1"); //sets the color reconnection scheme to 'new' QCD based scheme (TODO: make sure this is better than (2)gluon move)
+ pythia.readString("ColourReconnection:forceHadronLevelCR = on"); //allowing color reconnections for these constructed strings!
+ //a few params for the QCD based color reconnection scheme are set below.
+ pythia.readString("MultipartonInteractions:pT0Ref = 2.15"); //not sure if this is needed for this setup, but is part of the recommended 'default'
+ pythia.readString("ColourReconnection:allowDoubleJunRem = off"); //default on - allows directly connected double junction systems to split into two strings
+ pythia.readString("ColourReconnection:junctionCorrection = 1.15");
+ pythia.readString("ColourReconnection:timeDilationMode = 3"); //allow reconnection if single pair of dipoles are in causal contact (maybe try 5 as well?)
+ pythia.readString("ColourReconnection:timeDilationPar = 0.18"); //parameter used in causal interaction between strings (mode set above)(maybe try 0.073?)
+ */
+
+ std::stringstream lines;
+ lines << GetXMLElementText({"JetHadronization", "LinesToRead"}, false);
+ while (std::getline(lines, s, '\n')) {
+ if (s.find_first_not_of(" \t\v\f\r") == s.npos)
+ continue; // skip empty lines
+ JSINFO << "Also reading in: " << s;
+ pythia.readString(s);
+ }
+
+ // optional input of another pythia particle data xml file (higher excited states,...)
+ xml_intin = GetXMLElementInt({"JetHadronization", "additional_pythia_particles"});
+ if(xml_intin == 0 || xml_intin == 1){additional_pythia_particles = xml_intin;} xml_intin = -1;
+
+ if(additional_pythia_particles == 1) {
+ std::string additional_pythia_particle_file =
+ GetXMLElementText({"JetHadronization", "additional_pythia_particles_path"});
+ pythia.particleData.readXML(additional_pythia_particle_file,false);
+ }
// And initialize
pythia.init();
- //setting up thermal partons...
- //read in thermal partons, THEN do sibling setup...
- for (int i = 0; i < HH_thermal.num(); ++i) {
- HH_thermal[i].sibling(i);
- HH_thermal[i].string_id(-i);
- HH_thermal[i].is_used(true);
- HH_thermal[i].sibling(findthermalsibling(i, HH_thermal));
- HH_thermal[i].is_used(false);
+ //reading in info for thermal partons
+ inbrick = false; brickL = -1.;
+ inhydro = false; nreusehydro = 1;
+
+ xml_intin = GetXMLElementInt({"Eloss", "Matter", "brick_med"});
+ if(xml_intin == 1){inbrick = true;} xml_intin = -1;
+ xml_doublein = GetXMLElementDouble({"Eloss", "Matter", "brick_length"});
+ if(inbrick && (xml_doublein >= 0.)){brickL = xml_doublein;} xml_doublein = -1.;
+ xml_intin = GetXMLElementInt({"nReuseHydro"});
+ if(xml_intin > 0){nreusehydro = xml_intin;} xml_intin = -1;
+ xml_doublein = GetXMLElementDouble({"Eloss", "deltaT"});
+ if(xml_doublein >= 0.){delta_t = xml_doublein;} xml_doublein = -1.;
+
+ // Check if hydro is used in user xml file
+ tinyxml2::XMLElement *elementXML = (tinyxml2::XMLElement *)JetScapeXML::Instance()->GetXMLRootUser()->FirstChildElement();
+ while (elementXML) {
+ std::string elementName = elementXML->Name();
+ if (elementName == "Hydro") {
+ inhydro = true;
+ }
+ elementXML = elementXML->NextSiblingElement();
+ } // at this point the hydro could still be a brick -> check this in the following part
+
+ if(inhydro){
+ tinyxml2::XMLElement *element = (tinyxml2::XMLElement *)JetScapeXML::Instance()->GetXMLRootUser()->FirstChildElement();
+ while (element) {
+ std::string elementName = element->Name();
+ if (elementName == "Hydro") {
+ inhydro = true;
+ tinyxml2::XMLElement *childElement = (tinyxml2::XMLElement *)element->FirstChildElement();
+ while (childElement) {
+ std::string childElementName = childElement->Name();
+ if (childElementName == "Brick") {
+ inbrick = true;
+ inhydro = false;
+ }
+ childElement = childElement->NextSiblingElement();
+ }
+ }
+ element = element->NextSiblingElement();
+ }
}
+
+ // this is only important if a boost invariant 2+1d hydro is used
+ xml_doublein = GetXMLElementDouble({"JetHadronization", "eta_max_boost_inv"});
+ if(inhydro && (xml_doublein >= 0.)){eta_max_boost_inv = xml_doublein;} xml_doublein = -1.;
}
}
-void HybridHadronization::WriteTask(weak_ptr w) {
+void HybridHadronization::WriteTask(weak_ptr w){
VERBOSE(8);
auto f = w.lock();
- if (!f)
- return;
- f->WriteComment("Hadronization Module : " + GetId());
+ if ( !f ) return;
+ f->WriteComment("Hadronization Module : "+GetId());
}
//TODO: Junction Strings, Thermal Partons
-void HybridHadronization::DoHadronization(
- vector>> &shower,
- vector> &hOut, vector> &pOut) {
-
- VERBOSE(2) << "Start Hybrid Hadronization using both Recombination and "
- "PYTHIA Lund string model.";
- pythia.event.reset();
- HH_shower.clear();
-
- for (unsigned int ishower = 0; ishower < shower.size(); ++ishower) {
- for (unsigned int ipart = 0; ipart < shower.at(ishower).size(); ++ipart) {
- HHparton sh_parton;
- sh_parton.is_shower(true);
- sh_parton.id(shower.at(ishower).at(ipart)->pid());
- sh_parton.orig(0); //sh_parton.string_id(str)
- sh_parton.px(shower.at(ishower).at(ipart)->px());
- sh_parton.py(shower.at(ishower).at(ipart)->py());
- sh_parton.pz(shower.at(ishower).at(ipart)->pz());
- sh_parton.e(shower.at(ishower).at(ipart)->e());
- sh_parton.x(shower.at(ishower).at(ipart)->x_in().x());
- sh_parton.y(shower.at(ishower).at(ipart)->x_in().y());
- sh_parton.z(shower.at(ishower).at(ipart)->x_in().z());
- sh_parton.x_t(shower.at(ishower).at(ipart)->x_in().t());
- sh_parton.mass(
- sh_parton.e() * sh_parton.e() - sh_parton.px() * sh_parton.px() -
- sh_parton.py() * sh_parton.py() - sh_parton.pz() * sh_parton.pz());
- sh_parton.mass((sh_parton.mass() >= 0.) ? sqrt(sh_parton.mass())
- : sqrt(-sh_parton.mass()));
- sh_parton.col(shower.at(ishower).at(ipart)->color());
- sh_parton.acol(shower.at(ishower).at(ipart)->anti_color());
-
- HH_shower.add(sh_parton);
- }
- JSDEBUG << "Shower#" << ishower + 1
- << ". Number of partons to hadronize so far: " << HH_shower.num();
+void HybridHadronization::DoHadronization(vector>>& shower, vector>& hOut, vector>& pOut){
+ number_p_fake = 0; //reset counter for the number of fake partons
+ double energy_hadrons = 0.; //needed for the kinetic scaling of the negative hadrons
+
+ //JSINFO<<"Start Hybrid Hadronization using both Recombination and PYTHIA Lund string model.";
+ pythia.event.reset(); HH_shower.clear();
+ parton_collection neg_ptns;
+
+ //pointer to get hydro temperature info
+ std::unique_ptr check_fluid_info_ptr;
+
+ bool boost_invariant = true;
+ bool Cartesian_hydro = false; // not properly initialized by all hydro modules at the moment, don't use it
+ if(inhydro){
+ std::shared_ptr hydro_ptr = JetScapeSignalManager::Instance()->GetHydroPointer().lock();
+ const EvolutionHistory& bulk_info = hydro_ptr->get_bulk_info();
+
+ boost_invariant = bulk_info.is_boost_invariant();
+ Cartesian_hydro = bulk_info.is_Cartesian();
+ Cartesian_hydro = false;
}
- VERBOSE(2) << "# Partons to hadronize: " << HH_shower.num();
-
- int num_strings = 0;
-
- int attempt_num = 0;
- bool run_successfully = false;
- while ((attempt_num < attempts_max) && (!run_successfully)) {
- HH_showerptns = HH_shower;
- //clearing hadrons and remnants collections
- HH_hadrons.clear();
- HH_remnants.clear();
- HH_pyremn.clear();
-
- //since we 'might' need to reset the thermal partons (if present!)... because we alter the thermal parton collection in the string repair routine
- for (int i = 0; i < HH_thermal.num(); ++i) {
- HH_thermal[i].is_used(false);
- HH_thermal[i].is_decayedglu(false);
- HH_thermal[i].is_remnant(false);
- HH_thermal[i].used_reco(false);
- HH_thermal[i].used_str(false);
- HH_thermal[i].orig(1);
- HH_thermal[i].par(-1);
- HH_thermal[i].status(0);
- HH_thermal[i].col(0);
- HH_thermal[i].acol(0);
- HH_thermal[i].PY_par1(-1);
- HH_thermal[i].PY_par2(-1);
- HH_thermal[i].PY_dau1(-1);
- HH_thermal[i].PY_dau2(-1);
- HH_thermal[i].PY_stat(23);
- HH_thermal[i].string_id(-i);
- HH_thermal[i].is_strendpt(false);
- HH_thermal[i].pos_str(1);
- HH_thermal[i].endpt_id(0);
- }
-
- //checking the shower for any color singlet particles to just dump into PYTHIA
- //also handling colored non-partonic particles
- int i_show = 0;
- while (i_show < HH_showerptns.num()) {
- if (HH_showerptns[i_show].id() == 21) {
- ++i_show;
- continue;
- } //is gluon
- else if (std::abs(HH_showerptns[i_show].id()) <= 6) {
- ++i_show;
- continue;
- } //is quark
- else if ((std::abs(HH_showerptns[i_show].id()) >= 1103) &&
- (std::abs(HH_showerptns[i_show].id()) <= 5503) &&
- ((HH_showerptns[i_show].id() / 10) % 10 == 0)) {
- ++i_show;
- continue;
- } //is diquark
- else if (pythia.particleData.colType(HH_showerptns[i_show].id()) ==
- 2) { //this is a non-gluon color octet...
- HH_showerptns[i_show].PY_origid(HH_showerptns[i_show].id());
- HH_showerptns[i_show].id(21);
- ++i_show;
- continue;
- } //this is a non-(simple)quark color triplet... (pid 42(scalar leptoquark), 4000001(1-6, excited quarks), and SUSY particles)
- else if (std::abs(pythia.particleData.colType(
- HH_showerptns[i_show].id())) == 1) {
- HH_showerptns[i_show].PY_origid(HH_showerptns[i_show].id());
- HH_showerptns[i_show].id(1 *
- (2 * std::signbit(-pythia.particleData.colType(
- HH_showerptns[i_show].id())) -
- 1));
- ++i_show;
- continue;
- } //and the two above should catch 'colored technihadrons' too!
- else if (HH_showerptns[i_show].id() == 90) {
- HH_showerptns.remove(i_show);
- continue;
- } //PYTHIA specific pid for the 'system' as a whole, shouldn't actually be here. Dumping it.
- //is none of the above (a colorless object of some sort (a hadron, photon, lepton...))
- HHhadron had;
- had.id(HH_showerptns[i_show].id());
- had.orig(HH_showerptns[i_show].orig());
- had.mass(HH_showerptns[i_show].mass());
- had.pos(HH_showerptns[i_show].pos());
- had.P(HH_showerptns[i_show].P());
- HH_hadrons.add(had);
- HH_showerptns.remove(i_show);
- }
- /*
- std::cout << "\n\n Start(initial shower):\n";
- for(int i=0;ipid()); sh_parton.orig(0); //sh_parton.string_id(str)
+ sh_parton.px(shower.at(ishower).at(ipart)->px()); sh_parton.py(shower.at(ishower).at(ipart)->py());
+ sh_parton.pz(shower.at(ishower).at(ipart)->pz()); sh_parton.e(shower.at(ishower).at(ipart)->e());
+ sh_parton.x( shower.at(ishower).at(ipart)->x_in().x() ); sh_parton.y( shower.at(ishower).at(ipart)->x_in().y() );
+ sh_parton.z( shower.at(ishower).at(ipart)->x_in().z() ); sh_parton.x_t(shower.at(ishower).at(ipart)->x_in().t() );
+ sh_parton.mass( sh_parton.e()*sh_parton.e() - sh_parton.px()*sh_parton.px() - sh_parton.py()*sh_parton.py() - sh_parton.pz()*sh_parton.pz() );
+ sh_parton.mass( (sh_parton.mass() >= 0.) ? sqrt(sh_parton.mass()) : sqrt(-sh_parton.mass()) );
+ sh_parton.col( shower.at(ishower).at(ipart)->color() ); sh_parton.acol( shower.at(ishower).at(ipart)->anti_color() );
+
+ if(shower.at(ishower).at(ipart)->pstat()>-1){
+ HH_shower.add(sh_parton);
+ total_shower_energy += HH_shower[HH_shower.num()-1].e();
+ }else{
+ neg_ptns.add(sh_parton);
+ total_shower_energy_neg += neg_ptns[neg_ptns.num()-1].e();
+ }
+ }
+ JSDEBUG<<"Shower#"<= 0.) ? sqrt(thparton.mass()) : sqrt(-thparton.mass()) );
+ thparton.pos_str(1);
+ HH_thermal.add(thparton); //adding this parton to thermal collection
+ }
+ //read in thermal partons, THEN do sibling setup...
+ for(int i=0;i surface_cells;
+ GetHydroHyperSurface(hydro_Tc, surface_cells);
+ /*std::cout << "\n\n Surface Cells\n";
+ for(int icel=0; icel> surface;
+ for(int icel=0; icel cell;
+ cell.push_back(surface_cells[icel].tau); cell.push_back(surface_cells[icel].x); cell.push_back(surface_cells[icel].y); cell.push_back(surface_cells[icel].eta);
+ cell.push_back(surface_cells[icel].d3sigma_mu[0]); cell.push_back(surface_cells[icel].d3sigma_mu[1]);
+ cell.push_back(surface_cells[icel].d3sigma_mu[2]); cell.push_back(surface_cells[icel].d3sigma_mu[3]);
+ cell.push_back(surface_cells[icel].temperature);
+ double vx = surface_cells[icel].umu[1]/surface_cells[icel].umu[0];
+ double vy = surface_cells[icel].umu[2]/surface_cells[icel].umu[0];
+ double vz = surface_cells[icel].umu[3]/surface_cells[icel].umu[0];
+ cell.push_back(vx); cell.push_back(vy); cell.push_back(vz);
+ surface.push_back(cell);
+ }
+
+ ThermalPartonSampler part_samp(rand_seed); //initializing sampler with random seed
+ part_samp.set_hypersurface(surface);
+ if(boost_invariant){
+ part_samp.sample_2p1d(eta_max_boost_inv);
+ }else{
+ part_samp.sample_3p1d(Cartesian_hydro);
+ }
+
+ JSINFO << "Hydro was sampled, generating " << part_samp.nTot() << " partons (" << part_samp.th_nL() << " light, " << part_samp.th_nS() << " strange).";
+
+ for(int ith=0; ith= 0.) ? sqrt(thparton.mass()) : sqrt(-thparton.mass()) );
+ thparton.pos_str(1);
+ HH_thermal.add(thparton); //adding this parton to thermal collection
+ }
+ //read in thermal partons, THEN do sibling setup...
+ for(int i=0;i hydro_Tc){
+ double tnow = HH_shower[i_sh].x_t() + delta_t*double(i);
+ double xnow = HH_shower[i_sh].x() + vel[0]*delta_t*double(i);
+ double ynow = HH_shower[i_sh].y() + vel[1]*delta_t*double(i);
+ double znow = HH_shower[i_sh].z() + vel[2]*delta_t*double(i);
+ GetHydroCellSignal(tnow, xnow, ynow, znow, check_fluid_info_ptr);
+ temp = check_fluid_info_ptr->temperature;
+ if(temp <= hydro_Tc){
+ t_dif = delta_t*double(i);
+ }
+ ++i;
+ }
+ }
+ double max_t_dif = std::max(t_dif,part_prop);
+ HH_shower[i_sh].x(HH_shower[i_sh].x() + vel[0]*max_t_dif); HH_shower[i_sh].y(HH_shower[i_sh].y() + vel[1]*max_t_dif); HH_shower[i_sh].z(HH_shower[i_sh].z() + vel[2]*max_t_dif);
+ HH_shower[i_sh].x_t(HH_shower[i_sh].x_t() + max_t_dif);
+ //JSINFO<<"Parton propagated to: " << HH_shower[i_sh].x() << ", " << HH_shower[i_sh].y() << ", " << HH_shower[i_sh].z() << ", " << HH_shower[i_sh].x_t();
}
- std::cout << "End start(initial shower)\n\n"; std::flush(std::cout);
-*/
- //setting up the strings appropriately for showers - assumes that color tags are set.
- //if there are colored particles without set color tags, it will dump those partons in particular into a single string
- stringform();
- if (HH_showerptns.num() > 0)
- num_strings = HH_showerptns[HH_showerptns.num() - 1].string_id();
-
- //running recombination
- recomb();
-
- //function to prepare strings for input into PYTHIA8
- //will need a final reindexing for py_remn (all PY_par#, PY_dau# will need to be += 1) AFTER this function (either here, or in invoke_py)
- //when recursive 'workaround' for large strings is properly handled/removed, then can put this reindexing inside the function
- if (HH_remnants.num()) {
- bool cut = (attempt_num > -1) ? true : false;
- stringprep(HH_remnants, HH_pyremn, cut);
- }
-
- //running remaining partons through PYTHIA8 string fragmentation
- run_successfully = invoke_py();
-
- //for a successful run, go though final hadrons here and set parton parents up
- if (run_successfully) {
- //partons that have parh set need to have the reco parents of that hadron
- //partons that do not have parh set need to have the parents vector reset to the original shower/thermal partons
- //until this is done, the par(i) for each parton is *wrong*
- for (int iHad = 0; iHad < HH_hadrons.num(); ++iHad) {
- //final state string_frag hadron
- if ((HH_hadrons[iHad].is_final()) && (HH_hadrons[iHad].parh() == -1) &&
- (HH_hadrons[iHad].parents.size() > 0)) {
- std::vector py_remn_parents = HH_hadrons[iHad].parents;
- HH_hadrons[iHad].parents.clear();
- //just making sure that we got all the partons from the parent string as parents (if there were junctions, this might not have been complete)
- for (int i = 0; i < HH_pyremn.num(); ++i) {
- if ((HH_pyremn[i].string_id() ==
- HH_pyremn[py_remn_parents[0]].string_id()) &&
- (HH_pyremn[i].orig() != -1)) {
- py_remn_parents.push_back(i);
- }
- }
- std::sort(py_remn_parents.begin(), py_remn_parents.end());
- py_remn_parents.erase(
- std::unique(py_remn_parents.begin(), py_remn_parents.end()),
- py_remn_parents.end());
- std::vector temp;
- for (int i = 0; i < py_remn_parents.size(); ++i) {
- temp.push_back(HH_pyremn[py_remn_parents[i]].par());
- }
- std::sort(temp.begin(), temp.end());
- temp.erase(std::unique(temp.begin(), temp.end()), temp.end());
- for (int i = 0; i < temp.size(); ++i) {
- if (temp[i] < -1) {
- ++temp[i];
- } else if (temp[i] == -1) {
- temp[i] = -99999;
- }
- }
- HH_hadrons[iHad].parents = temp;
-
- //a last check to make sure that any strings with junctions and thermal partons are accounted for
- for (int ipar = 0; ipar < py_remn_parents.size(); ++ipar) {
- if (HH_pyremn[py_remn_parents[ipar]].is_thermal()) {
- HH_hadrons[iHad].is_shth(true);
- HH_hadrons[iHad].is_shsh(false);
- break;
- }
- }
+ //do the same for the negative partons
+ for(int i_sh=0; i_sh hydro_Tc){
+ double tnow = neg_ptns[i_sh].x_t() + delta_t*double(i);
+ double xnow = neg_ptns[i_sh].x() + vel[0]*delta_t*double(i);
+ double ynow = neg_ptns[i_sh].y() + vel[1]*delta_t*double(i);
+ double znow = neg_ptns[i_sh].z() + vel[2]*delta_t*double(i);
+ GetHydroCellSignal(tnow, xnow, ynow, znow, check_fluid_info_ptr);
+ temp = check_fluid_info_ptr->temperature;
+ if(temp <= hydro_Tc){
+ t_dif = delta_t*double(i);
}
- //final state reco hadron
- else if ((HH_hadrons[iHad].is_final()) &&
- (HH_hadrons[iHad].parh() >= 0) &&
- (HH_hadrons[iHad].parh() < HH_hadrons.num()) &&
- (HH_hadrons[iHad].parh() != iHad)) {
- HH_hadrons[iHad].parents =
- HH_hadrons[HH_hadrons[iHad].parh()].parents;
+ ++i;
+ }
+ }
+ double max_t_dif = std::max(t_dif,part_prop);
+ neg_ptns[i_sh].x(neg_ptns[i_sh].x() + vel[0]*max_t_dif); neg_ptns[i_sh].y(neg_ptns[i_sh].y() + vel[1]*max_t_dif); neg_ptns[i_sh].z(neg_ptns[i_sh].z() + vel[2]*max_t_dif);
+ neg_ptns[i_sh].x_t(neg_ptns[i_sh].x_t() + max_t_dif);
+ //JSINFO<<"Parton propagated to: " << neg_ptns[i_sh].x() << ", " << neg_ptns[i_sh].y() << ", " << neg_ptns[i_sh].z() << ", " << neg_ptns[i_sh].x_t();
+ }
+
+ /*for(int i=0; i=0; --pos_ptn){
+ double tmp_threco = th_recofactor; //need to keep track of th_recofactor (strength) to disable it for negative partons
+ double tmp_maxB_level = maxB_level; //need to keep track of maxB_level to disable baryon recombination for negative partons
+ int tmp_reco_hadrons_pythia = reco_hadrons_pythia;
+ if(pos_ptn == 0){//handling negative partons by wiping shower partons & hadrons, then rerunning with negative partons
+ pythia.event.reset(); HH_shower.clear(); HH_shower = neg_ptns; //hadrons, remnants, pyremn cleared below
+
+ // if the fragmentation hadrons are given to the afterburner module
+ // let the negative partons decay here, as they are not propagated in the
+ // afterburner
+ if(afterburner_frag_hadrons){
+ reco_hadrons_pythia = 1;
+ pythia.readString("HadronLevel:Decay = on");
+ pythia.init();
+ }
+
+ //add holes left by used thermal partons to HH_shower
+ for(int i=0; i= 1103) && (std::abs(HH_showerptns[i_show].id()) <= 5503) && ((HH_showerptns[i_show].id()/10)%10 == 0)){++i_show; continue;} //is diquark
+ else if(pythia.particleData.colType(HH_showerptns[i_show].id()) == 2){//this is a non-gluon color octet...
+ HH_showerptns[i_show].PY_origid(HH_showerptns[i_show].id()); HH_showerptns[i_show].id(21); ++i_show; continue;
+ }//this is a non-(simple)quark color triplet... (pid 42(scalar leptoquark), 4000001(1-6, excited quarks), and SUSY particles)
+ else if(std::abs(pythia.particleData.colType(HH_showerptns[i_show].id())) == 1){
+ HH_showerptns[i_show].PY_origid( HH_showerptns[i_show].id() );
+ HH_showerptns[i_show].id( 1*(2*std::signbit(-pythia.particleData.colType(HH_showerptns[i_show].id()))-1) ); ++i_show; continue;
+ } //and the two above should catch 'colored technihadrons' too!
+ else if(HH_showerptns[i_show].id() == 90){HH_showerptns.remove(i_show); continue;} //PYTHIA specific pid for the 'system' as a whole, shouldn't actually be here. Dumping it.
+ //is none of the above (a colorless object of some sort (a hadron, photon, lepton...))
+ HHhadron had; had.id( HH_showerptns[i_show].id() ); had.orig( HH_showerptns[i_show].orig() ); had.mass( HH_showerptns[i_show].mass() );
+ had.pos(HH_showerptns[i_show].pos()); had.P(HH_showerptns[i_show].P());
+ HH_hadrons.add(had); HH_showerptns.remove(i_show);
+ }
+
+ //setting up the strings appropriately for showers - assumes that color tags are set.
+ //if there are colored particles without set color tags, it will dump those partons in particular into a single string
+ int num_strings = 0;
+ stringform();
+ num_strings = HH_showerptns[HH_showerptns.num()-1].string_id();
+
+ // last attempt in hadronization is done without recombination
+ double sh_recofactor_store = sh_recofactor;
+ if(attempt_num == attempts_max -1) {
+ sh_recofactor = 0.;
+ if(pos_ptn == 1) {
+ JSWARN << "Hadronization failed, try without recombination one more time";
+ } else {
+ JSWARN << "Hadronization of negative partons failed, try without recombination one more time";
+ }
+ }
- ++attempt_num;
- }
+ //running recombination
+ recomb();
+ sh_recofactor = sh_recofactor_store;
- if (!run_successfully) {
- HH_hadrons.clear();
- JSWARN << "This event could not be hadronized.";
- }
+ //function to prepare strings for input into PYTHIA8
+ //will need a final reindexing for py_remn (all PY_par#, PY_dau# will need to be += 1) AFTER this function (either here, or in invoke_py)
+ //when recursive 'workaround' for large strings is properly handled/removed, then can put this reindexing inside the function
+ if(HH_remnants.num()){
+ bool cut = (attempt_num > -1) ? true : false;
+ stringprep(HH_remnants, HH_pyremn, cut);
+ }
+
+ //temporary workaround to force all formed hadrons to be final state - so pythia won't overwrite spacetime info
+ for(int i=0; i (2.*pythia.particleData.m0(211) + 0.01)) {
+ HH_pyremn[ipart].mass(2.*pythia.particleData.m0(211) + 0.01);
+ }
+ HH_pyremn[ipart].e(HH_pyremn[ipart].mass());
+ HH_pyremn[ipart].px(0.);
+ HH_pyremn[ipart].py(0.);
+ HH_pyremn[ipart].pz(0.);
+ }
+ }
+ }
- for (unsigned int iHad = 0; iHad < HH_hadrons.num(); ++iHad) {
- if (HH_hadrons[iHad].is_final()) {
- int idH = HH_hadrons[iHad].id();
- double mH = HH_hadrons[iHad].mass();
- FourVector p(HH_hadrons[iHad].P());
- FourVector x(HH_hadrons[iHad].pos());
- hOut.push_back(std::make_shared(Hadron(0, idH, 1, p, x, mH)));
+ //running remaining partons through PYTHIA8 string fragmentation
+ run_successfully = invoke_py();
+
+ //for a successful run, go though final hadrons here and set parton parents up
+ //bring the hadrons to their corresponding pythia mass shell
+ if(run_successfully){
+
+ //remove all hadrons from HH_hadrons if reco hadrons are put into pythia
+ //in this case they are already contained in HH_pythia_hadrons
+ if(reco_hadrons_pythia) {
+ HH_hadrons.clear();
+ }
+
+ //add all fragmentation hadrons from pythia to HH_hadrons, now that pythia hadronization was successfull
+ for(int iHad = 0; iHad < HH_pythia_hadrons.num(); iHad++) {
+ HH_hadrons.add(HH_pythia_hadrons[iHad]);
+ }
+
+ bring_hadrons_to_mass_shell(HH_hadrons);
+
+ //hadron propagation (neglect the photons)
+ for(int iHad=0; iHad 0.0001 && HH_hadrons[iHad].id() != 22){ //assumes that hadron will be propagated by more than 0.0001 fm/c in lab frame - can even propagate backwards, if that's something wanted...
+ double vel[3]; vel[0]=HH_hadrons[iHad].px()/HH_hadrons[iHad].e(); vel[1]=HH_hadrons[iHad].py()/HH_hadrons[iHad].e(); vel[2]=HH_hadrons[iHad].pz()/HH_hadrons[iHad].e();
+ double t_prop = had_prop/sqrt(1. - vel[0]*vel[0] - vel[1]*vel[1] - vel[2]*vel[2]);
+ HH_hadrons[iHad].x(HH_hadrons[iHad].x() + vel[0]*t_prop); HH_hadrons[iHad].y(HH_hadrons[iHad].y() + vel[1]*t_prop); HH_hadrons[iHad].z(HH_hadrons[iHad].z() + vel[2]*t_prop);
+ HH_hadrons[iHad].x_t(HH_hadrons[iHad].x_t() + had_prop);
+ //JSINFO<<"Hadron propagated to: " << HH_hadrons[iHad].x() << ", " << HH_hadrons[iHad].y() << ", " << HH_hadrons[iHad].z() << ", " << HH_hadrons[iHad].x_t();
+ }
+ }
+
+ double energy_check = 0.;
+ for(int iHad = 0; iHad < HH_hadrons.num(); iHad++){
+ energy_check += HH_hadrons[iHad].e();
+ }
+ if(pos_ptn == 1 && (energy_check < 0.99*total_shower_energy)){
+ run_successfully = false;
+ }
+ }
+ ++attempt_num;
}
- }
- VERBOSE(2) << "#Showers hadronized together: " << shower.size() << " ( "
- << num_strings << " initial strings ). There are " << hOut.size()
- << " hadrons and " << pOut.size()
- << " partons after Hybrid Hadronization";
+
+ if(!run_successfully){
+ HH_hadrons.clear();
+ if(pos_ptn == 1) {
+ JSWARN << "This event could not be hadronized.";
+ } else {
+ JSWARN << "The negative partons of this event could not be hadronized.";
+ }
+ }
+
+ // scale the negative hadron energies/momenta such that the energy is
+ // conserved when subtracting negative hadrons from the positive ones
+ if(pos_ptn == 0){
+ if(HH_hadrons.num() > 0){
+ scale_kinematics_negative_hadrons(HH_hadrons, total_shower_energy-total_shower_energy_neg, energy_hadrons);
+ }
+ }
+
+ for(unsigned int iHad=0; iHad xy -> x=1 reco, x=2 frag; y=1 sh-sh, y=2 sh-th; neg sign means negative hadron
+ //all colorless particles get fragmentation status labels
+ int stat = (HH_hadrons[iHad].is_recohad()) ? 810 : 820;
+ stat += (HH_hadrons[iHad].is_shth()) ? 2 : 1;
+ stat *= (pos_ptn == 0) ? -1 : 1 ;
+ int lab = (pos_ptn == 0) ? -1 : 1 ;
+ int idH = HH_hadrons[iHad].id(); double mH = HH_hadrons[iHad].mass();
+ FourVector p(HH_hadrons[iHad].P()); FourVector x(HH_hadrons[iHad].pos());
+ hOut.push_back(std::make_shared (Hadron (lab,idH,stat,p,x,mH)));
+
+ if(pos_ptn == 1){ // used for scaling of negative hadrons
+ energy_hadrons += HH_hadrons[iHad].e();
+ }
+ }
+ }
+ th_recofactor = tmp_threco;
+ maxB_level = tmp_maxB_level;
+ reco_hadrons_pythia = tmp_reco_hadrons_pythia;
+ if(pythia_decays == "off"){pythia.readString("HadronLevel:Decay = off"); pythia.init();}
+ }
}
-//was used in the pygen code to create ordered strings (and assigned string ids) from color tags
-//currently has junction functionality commented out until I know how to incorporate it within JETSCAPE
-//junction functionality WILL be needed in the event that any given string configurations contain them!
-void HybridHadronization::stringform() {
+// scale the negative hadron energies/momenta such that the energy is
+// conserved when subtracting negative hadrons from the positive ones
+void HybridHadronization::scale_kinematics_negative_hadrons(hadron_collection& HH_hadrons, double shower_energy, double positive_hadrons_energy){
+ double negative_hadrons_energy_initial = 0;
+ for(int i = 0; i < HH_hadrons.num(); i++){
+ negative_hadrons_energy_initial += HH_hadrons[i].e();
+ }
- int nstr = 1;
+ double e_scaling_factor = (positive_hadrons_energy - shower_energy) / negative_hadrons_energy_initial;
+ if(e_scaling_factor < 0.){
+ JSDEBUG << "e_scaling_factor negative, hadron energy below parton energy: " << positive_hadrons_energy - shower_energy << "GeV";
+ return;
+ }
- for (int i = 0; i < HH_showerptns.num(); ++i) {
- if (HH_showerptns[i].string_id() != 0) {
- continue;
+ for(int i = 0; i < HH_hadrons.num(); i++){
+ double new_energy = HH_hadrons[i].e() * e_scaling_factor;
+ if(new_energy > HH_hadrons[i].mass()){
+ HH_hadrons[i].e(new_energy);
+ double new_abs_momentum = std::sqrt(new_energy*new_energy - HH_hadrons[i].mass()*HH_hadrons[i].mass());
+ double p_scaling_factor = new_abs_momentum / std::sqrt(HH_hadrons[i].px()*HH_hadrons[i].px()+HH_hadrons[i].py()*HH_hadrons[i].py()+HH_hadrons[i].pz()*HH_hadrons[i].pz());
+ HH_hadrons[i].px(HH_hadrons[i].px() * p_scaling_factor);
+ HH_hadrons[i].py(HH_hadrons[i].py() * p_scaling_factor);
+ HH_hadrons[i].pz(HH_hadrons[i].pz() * p_scaling_factor);
+ }else{
+ HH_hadrons[i].e(HH_hadrons[i].mass());
+ HH_hadrons[i].px(0.0);
+ HH_hadrons[i].py(0.0);
+ HH_hadrons[i].pz(0.0);
}
+ }
+}
- std::vector colors;
- if (HH_showerptns[i].col() > 0) {
- colors.push_back(HH_showerptns[i].col());
+void HybridHadronization::convert_color_tags_to_int_type(vector>>& shower){
+ //check which partons have anti-/color tags larger than the int limit
+ std::vector used_tags;
+ //reduce the upper limit -> the maxtag could fail otherwise, if it becomes larger than maximum int
+ int max_allowed_tag = std::numeric_limits::max() - 1e6;
+ for(unsigned int ishower = 0; ishower < shower.size(); ishower++) {
+ for(unsigned int ipart = 0; ipart < shower.at(ishower).size(); ipart++) {
+ used_tags.push_back(shower.at(ishower).at(ipart)->color());
+ used_tags.push_back(shower.at(ishower).at(ipart)->anti_color());
}
- if (HH_showerptns[i].acol() > 0) {
- colors.push_back(HH_showerptns[i].acol());
+ }
+ //remove consecutive duplicates in the sorted vector
+ std::sort(used_tags.begin(), used_tags.end());
+ used_tags.resize(std::distance(used_tags.begin(), std::unique(used_tags.begin(), used_tags.end())));
+
+ //count the number of needed tags
+ std::vector needed_tags_values;
+ int number_needed_tags = 0;
+ for (unsigned int tag : used_tags) {
+ if (tag > max_allowed_tag) {
+ needed_tags_values.push_back(tag);
+ number_needed_tags++;
}
- HH_showerptns[i].string_id(nstr);
- ++nstr;
- bool newcolor = true;
- while (newcolor) {
- newcolor = false;
- //checking all final particles not in a string for new color id matches...
- int j = 0;
- while (j < HH_showerptns.num()) {
- if (HH_showerptns[j].string_id() != 0) {
- ++j;
- continue;
+ }
+
+ //find the first free tag that is not yet used
+ int lower_limit_tag = 100;
+ std::vector new_tags;
+ while (new_tags.size() < number_needed_tags) {
+ if (std::find(used_tags.begin(), used_tags.end(), lower_limit_tag) == used_tags.end()) {
+ new_tags.push_back(lower_limit_tag);
+ }
+ lower_limit_tag++;
+ }
+
+ //exchange the tags which are too large by the new ones
+ for(int tag = 0; tag < number_needed_tags; tag++) {
+ for(unsigned int ishower = 0; ishower < shower.size(); ishower++) {
+ for(unsigned int ipart = 0; ipart < shower.at(ishower).size(); ipart++) {
+ unsigned int current_tag = needed_tags_values.at(tag);
+ if (shower.at(ishower).at(ipart)->color() == current_tag) {
+ shower.at(ishower).at(ipart)->set_color(new_tags.at(tag));
}
- for (int k = 0; k < colors.size(); ++k) {
- if (colors[k] == HH_showerptns[j].col() &&
- (HH_showerptns[j].acol() > 0)) {
- colors.push_back(HH_showerptns[j].acol());
- newcolor = true;
- HH_showerptns[j].string_id(HH_showerptns[i].string_id());
- j = -1;
- break;
- } else if (colors[k] == HH_showerptns[j].acol() &&
- (HH_showerptns[j].col() > 0)) {
- colors.push_back(HH_showerptns[j].col());
- newcolor = true;
- HH_showerptns[j].string_id(HH_showerptns[i].string_id());
- j = -1;
- break;
- } else if (colors[k] == HH_showerptns[j].col() ||
- colors[k] == HH_showerptns[j].acol()) {
- HH_showerptns[j].string_id(HH_showerptns[i].string_id());
- break;
- }
+ if (shower.at(ishower).at(ipart)->anti_color() == current_tag) {
+ shower.at(ishower).at(ipart)->set_anti_color(new_tags.at(tag));
}
- ++j;
}
- /* //this code WILL be needed in the event that any input string configurations contain junctions!
+ }
+ }
+}
+
+//was used in the pygen code to create ordered strings (and assigned string ids) from color tags
+//currently has junction functionality commented out until I know how to incorporate it within JETSCAPE
+//junction functionality WILL be needed in the event that any given string configurations contain them!
+void HybridHadronization::stringform(){
+
+ int nstr=1;
+
+ for(int i=0; i colors;
+ if(HH_showerptns[i].col() > 0){colors.push_back(HH_showerptns[i].col()) ;}
+ if(HH_showerptns[i].acol() > 0){colors.push_back(HH_showerptns[i].acol());}
+ HH_showerptns[i].string_id(nstr); ++nstr;
+ bool newcolor = true;
+ while(newcolor){
+ newcolor = false;
+ //checking all final particles not in a string for new color id matches...
+ int j=0; while(j0)){
+ colors.push_back(HH_showerptns[j].acol()); newcolor=true; HH_showerptns[j].string_id(HH_showerptns[i].string_id()); j=-1; break;
+ }
+ else if(colors[k]==HH_showerptns[j].acol() && (HH_showerptns[j].col() >0)){
+ colors.push_back(HH_showerptns[j].col()) ; newcolor=true; HH_showerptns[j].string_id(HH_showerptns[i].string_id()); j=-1; break;
+ }
+ else if(colors[k]==HH_showerptns[j].col() || colors[k]==HH_showerptns[j].acol()){HH_showerptns[j].string_id(HH_showerptns[i].string_id()); break;}
+ }
+ ++j;
+ }
+ /* //this code WILL be needed in the event that any input string configurations contain junctions!
//checking all the junctions in the event to see if there are any new color id matches...
for(int iJun=0;iJun is_used;
- std::vector ptns_new;
- std::vector ptns_strnow;
- for (int j = 0; j < HH_showerptns.num(); ++j) {
- if (HH_showerptns[j].string_id() == i) {
- ptns_strnow.push_back(j);
- is_used.push_back(false);
- }
- }
- //std::vector usedJuns; for(int j=0; j stack;
- bool readcolor = true;
- int firstcol = 0;
- for (int j = 0; j < ptns_strnow.size(); ++j) {
- if (!(HH_showerptns[ptns_strnow[j]].id() == 21)) {
- if ((HH_showerptns[ptns_strnow[j]].id() > 0 &&
- HH_showerptns[ptns_strnow[j]].id() <= 6) ||
- (HH_showerptns[ptns_strnow[j]].id() < -6)) {
- stack.push_back(ptns_strnow[j]);
- is_used[j] = true;
- firstcol = HH_showerptns[stack[0]].acol();
- break;
- } else {
- stack.push_back(ptns_strnow[j]);
- is_used[j] = true;
- firstcol = HH_showerptns[stack[0]].col();
- readcolor = false;
- break;
- }
- }
- //gluon loop catch
- else if (j == ptns_strnow.size() - 1) {
- stack.push_back(ptns_strnow[0]);
- is_used[0] = true;
- firstcol = HH_showerptns[stack[0]].acol();
- }
- }
- ptns_new.push_back(stack[0]);
+ //now, finding all partons in string 'i'(out of nstr-1 total strings) and reordering them
+ //for "string" i=0 final partons - these are noncolored particles that should just be written out as-is (shouldn't actually be present...)
+ for(int i=1;i is_used; std::vector ptns_new;
+ std::vector ptns_strnow;
+ for(int j=0; j usedJuns; for(int j=0; j stack;
+ bool readcolor=true; int firstcol = 0;
+ for(int j=0;j 0 && HH_showerptns[ptns_strnow[j]].id() <= 6) || (HH_showerptns[ptns_strnow[j]].id() < -6)){
+ stack.push_back(ptns_strnow[j]); is_used[j]=true; firstcol=HH_showerptns[stack[0]].acol(); break;
+ }
+ else{stack.push_back(ptns_strnow[j]); is_used[j]=true; firstcol=HH_showerptns[stack[0]].col(); readcolor=false; break;}
+ }
+ //gluon loop catch
+ else if(j==ptns_strnow.size()-1){stack.push_back(ptns_strnow[0]); is_used[0]=true; firstcol=HH_showerptns[stack[0]].acol();}
+ }
+ ptns_new.push_back(stack[0]);
- while (stack.size() > 0) {
+ while(stack.size()>0){
- //catching the end of either a string/junction leg or we've circled back on a gluon loop.
- if (readcolor && (HH_showerptns[stack.back()].col() == firstcol)) {
- stack.pop_back();
- continue;
- } else if (!readcolor &&
- (HH_showerptns[stack.back()].acol() == firstcol)) {
- stack.pop_back();
- continue;
- }
+ //catching the end of either a string/junction leg or we've circled back on a gluon loop.
+ if( readcolor && (HH_showerptns[stack.back()].col() == firstcol)){stack.pop_back(); continue;}
+ else if(!readcolor && (HH_showerptns[stack.back()].acol() == firstcol)){stack.pop_back(); continue;}
- //keeping track if we've added (a) new parton(s) to the stack
- bool found = false;
-
- for (int j = 0; j < ptns_strnow.size(); ++j) {
- if (readcolor && !is_used[j] &&
- (HH_showerptns[stack.back()].col() ==
- HH_showerptns[ptns_strnow[j]].acol())) {
- stack[stack.size() - 1] = ptns_strnow[j];
- is_used[j] = true;
- ptns_new.push_back(stack.back()); //--stack.size(); ++stack.size();
- found = true;
- break;
- } else if (!readcolor && !is_used[j] &&
- (HH_showerptns[stack.back()].acol() ==
- HH_showerptns[ptns_strnow[j]].col())) {
- stack[stack.size() - 1] = ptns_strnow[j];
- is_used[j] = true;
- ptns_new.push_back(stack.back()); //--stack.size(); ++stack.size();
- found = true;
- break;
- }
- }
+ //keeping track if we've added (a) new parton(s) to the stack
+ bool found=false;
+
+ for(int j=0;j 0)){
+ //this code WILL be needed in the event that any input string configurations contain junctions!
+ /*if(!found && ((pythia.event.sizeJunction() - numJunsused) > 0)){
int iJun(0), iCol(0);
if(readcolor){
for(int iJ=0;iJcols[1]){int temp=cols[0]; cols[0]=cols[1]; cols[1]=temp;}
-
+
if(readcolor){
for(int j=0;j list_strs;
+ //adding the first string to the list
+ list_strs.push_back(HH_showerptns[0].string_id());
+ //looping over all the partons in the event, and writing each 'unique' string to the list
+ for(int i=0;i ColInfo3;
+ for(int i=0; i Info;// location of i, location of k, temp color tag will be saved in this vector.
+ Info.push_back(i);
+ Info.push_back(k);
+ Info.push_back(temptag);
+
+ HH_showerptns[i].col(temptag);
+ HH_showerptns[k].acol(temptag); //apply the change for the col, acol pair to preserve the color flow with least impact to the final result.
+ temptag++;
+ Info.clear();
+ }
+ }
+ }
}
}
-}
-void HybridHadronization::recomb() {
-
- //declaring a few needed values (based on values in init)
- double hbarc2 = hbarc * hbarc;
-
- //should create a new parton collection here, one with only shower quarks (inc. from gluon splitting)
- //to consider for recombination - then afterwards can reform gluons and output remnants
- parton_collection showerquarks;
-
- //clearing remnants, in case it hasn't been done before
- HH_remnants.clear();
-
- //constructing a list of all the strings in the event
- std::vector list_strs;
- //adding the first string to the list
- if (HH_showerptns.num() > 0)
- list_strs.push_back(HH_showerptns[0].string_id());
- //looping over all the partons in the event, and writing each 'unique' string to the list
- for (int i = 0; i < HH_showerptns.num(); ++i) {
- bool str_match = false;
- for (int j = 0; j < list_strs.size(); ++j) {
- if (HH_showerptns[i].string_id() == list_strs[j]) {
- str_match = true;
+ for(int i = 0; i < HH_showerptns.num(); i++){
+ for(int j = i+1; j < HH_showerptns.num(); j++){
+ if(HH_showerptns[i].acol() != 0 && HH_showerptns[i].acol() == HH_showerptns[j].acol()){ //same acol tag for different particles detected.
+ for(int k = 0; k < HH_showerptns.num(); k++){
+ if(HH_showerptns[i].acol() == HH_showerptns[k].col()){
+ vector Info;// location of i, location of k, temp color tag will be saved in this vector.
+ Info.push_back(i);
+ Info.push_back(k);
+ Info.push_back(temptag);
+
+ HH_showerptns[i].acol(temptag);
+ HH_showerptns[k].col(temptag); //apply the change for the col, acol pair to preserve the color flow with least impact to the final result.
+ temptag++;
+ Info.clear();
+ }
+ }
}
}
- if (!str_match) {
- list_strs.push_back(HH_showerptns[i].string_id());
- }
}
- //*********************************************************************************************************************
- // Splitting gluons into q-qbar pairs for recombination
- //*********************************************************************************************************************
+ //std::cout < ColInfo2;
+ for(int i=0; i 0, thermal are < 0
- //perm2 will always access element [std::abs(perm2[i]) - 1]
- int perm1[showerquarks.num()], perm2[showerquarks.num() + HH_thermal.num()];
- for (int i = 0; i < showerquarks.num(); ++i) {
- perm1[i] = i;
- }
- for (int i = 0; i < HH_thermal.num(); ++i) {
- perm2[i] = (i - HH_thermal.num());
+ //randomly permute an integer array with 'n' entries from 1 to n (actually from 0 to n-1)
+ //the array will be used to access the i'th element of the showerquarks collection
+ //done as showerquarks[perm0_sharray[i]]
+ //since we have separate shower and thermal arrays, we will force the first quark to always be from the shower
+ //
+ //construct perm1 array of size showerquarks.num() and perm2 array of size showerquarks.num() + HH_thermal.num()
+ //to access an element from either shower or thermal partons
+ //there will be showerquarks.num() positive entries and HH_thermal.num() negative entries (going to exclude 0 in these, just -1 i's)
+ //thus, perm2 = { 3, -42, -33,...} will access i=2 from showerquarks, then 41 from thermal, then 32 from thermal...
+ //
+ //lastly, need to bypass scenarios where the quark has been used, or we're trying to use the same quark twice
+ //to determine if we're trying to use the same quark twice, check if perm[i]=perm[j]
+ //if used=true OR p[i]=p[j]?, then skip this attempt (continue works well for these cases?)
+
+ //constructing permutation arrays; shower quarks are > 0, thermal are < 0
+ //perm2 will always access element [std::abs(perm2[i]) - 1]
+ int perm1[showerquarks.num()], perm2[showerquarks.num()+HH_thermal.num()];
+
+ //option to either bias recombination, earlier particles attempt to recombine first
+ if(torder_reco){
+ //placeholders to sort by time
+ std::vector> tosort1, tosort2;
+ for (int i=0;i ColInfo;
+ for(int i=0; i> IndiceForCol1; // this vector will form vector of vectors(location in the string, anti or not, color tag)
+ std::vector IndiceForCol2; // this vector is one component of vector above
+ std::vector IndiceForColFin; // this vector is final vector that'll be used for process
+
+ for(int iIndice=0; iIndice < HH_showerptns.num(); iIndice++) {
+ IndiceForCol2.push_back(iIndice);
+ IndiceForCol2.push_back(1);
+ IndiceForCol2.push_back(HH_showerptns[iIndice].col());
+ IndiceForCol1.push_back(IndiceForCol2);
+ IndiceForCol2.clear();
+
+ IndiceForCol2.push_back(iIndice);
+ IndiceForCol2.push_back(-1);
+ IndiceForCol2.push_back(HH_showerptns[iIndice].acol());
+ IndiceForCol1.push_back(IndiceForCol2);
+ IndiceForCol2.clear();
+ } // As a result, two vectors with 3 component keep being added to bigger vector, now exclude same tag and zero
+ //dignostic measure
+ //std::cout < tempcol;
+ // now put all the values in ColInfo into IndiceForColFin
+ for(int icol = 0; icol < IndiceForCol1.size(); icol++){
+ tempcol.push_back(IndiceForCol1.at(icol).at(2));
}
-
- //permuting these arrays using Fisher-Yates algorithm
- //-- To shuffle an array a of n elements (indices 0..n-1):
- //for i from 0 to n−2 do
- // j ← random integer such that i ≤ j < n
- // exchange a[i] and a[j]
- for (int i = 0; i < showerquarks.num() - 1; ++i) {
- int ranelement = i + floor((showerquarks.num() - i) * ran());
- int temp = perm1[i];
- perm1[i] = perm1[ranelement];
- perm1[ranelement] = temp;
+ if(tempcol.at(0) != 0){
+ IndiceForColFin.push_back(tempcol.at(0));
}
- for (int i = 0; i < showerquarks.num() + HH_thermal.num() - 1; ++i) {
- int ranelement =
- i + floor((showerquarks.num() + HH_thermal.num() - i) * ran());
- int temp = perm2[i];
- perm2[i] = perm2[ranelement];
- perm2[ranelement] = temp;
+ if(tempcol.at(1) != 0){
+ IndiceForColFin.push_back(tempcol.at(1));
}
- //looping over all the quarks that we have...
- //'q1' loops over all quarks in the shower
- //'q2' loops over all quarks in the event
- //'q3' loops over all quarks in the event, starting from 'q2' and ending at the last quark
- //when 'q2' is at the last quark, we will not consider quark 'q3' - can only make a meson at that point...
+ for(int iclear1 = 0; iclear1 < tempcol.size(); iclear1++){
+ bool foundsame = false;
+ for(int iclear2 = 0; iclear2 < IndiceForColFin.size(); iclear2++){
+ if(tempcol.at(iclear1) == IndiceForColFin.at(iclear2) || tempcol.at(iclear1) == 0){
+ foundsame = true;
+ }
+ }
+ if(!foundsame){
+ IndiceForColFin.push_back(tempcol.at(iclear1));
+ }
+ foundsame = false;
+ }
- parton_collection considering;
- int element[3];
+ //last checking whether there is zero in the color tag list
+ for(int i = 0; i < IndiceForColFin.size(); i++){
+ if(IndiceForColFin[i] == 0){
+ std::vector::iterator i1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), 0 );
+ int distance = std::distance(IndiceForColFin.begin() , i1);
+ //std::cout <> MesonrecoMatrix1; // vector of double (prob) , ,,,// in a big scheme, these two vector form a Matrix
+ std::vector MesonrecoMatrix2; // this is 1d vector of (probabilty)
+
+ //first, Form Meson recombination Matrix
+ for(int irow=0; irow < IndiceForColFin.size(); irow++){
+ for(int icol=0; icol < IndiceForColFin.size(); icol++){
+ double Mesonfactor;
+ int distance; // distance between the color tags in string(not c++ function)
+ int tag1 = IndiceForColFin.at(irow);
+ int tag2 = IndiceForColFin.at(icol);
+ std::vector::iterator it1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag1);
+ std::vector::iterator it2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag2);// set up for finding the location of co tag in vector that will be used to find distance
+ int pos1 = std::distance(IndiceForColFin.begin(), it1); //location of col tag of irow
+ int pos2 = std::distance(IndiceForColFin.begin(), it2); //location of col tag of icol
+ //std::cout < 0) {
+ Mesonfactor = 0.111;
+ //std::cout <::iterator it1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag1);
+ std::vector::iterator it2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag2);// set up for finding the location of co tag in vector that will be used to find distance
+ int pos1 = std::distance(IndiceForColFin.begin(), it1); //location of col tag of irow
+ int pos2 = std::distance(IndiceForColFin.begin(), it2); //location of col tag of icol
+ MesonrecoMatrix1.at(pos1).at(pos2) = 0;
+ MesonrecoMatrix1.at(pos2).at(pos1) = 0;
+ }
+ }
+ /*
+ //dignostic measure
+ std::cout <>> BaryonrecoMatrix1; // vector of double (prob) , ,,,// in a big scheme, these three items form a Matrix of vector
+ std::vector> BaryonrecoMatrix2; // this is vector of 2d vector below (probabilty, required color charge for color neutrality), ( , ) , ( , ) ...
+ std::vector BaryonrecoMatrix3; // this is 2d vector of (probabilty, required color charge for color neutrality)
- //taking the id from the permutation array, and turning it into quark array element
- element[0] = perm1[q1];
+ for(int irow=0; irow < IndiceForColFin.size(); irow++){
+ for(int icol=0; icol < IndiceForColFin.size(); icol++){
+ double Baryonfactor;
+ BaryonrecoMatrix3.push_back(1./27.);//1/27: probability to get singlet state
+ BaryonrecoMatrix3.push_back(0.); // now the probability factor and color(not specified yet, it'll be done with baryon formation) for neutrality are saved.
- //skipping if current quark has been used or isn't a u,d,s quark or antiquark
- if (showerquarks[element[0]].status() != 0 ||
- showerquarks[element[0]].is_used()) {
- continue;
- } else if (std::abs(showerquarks[element[0]].id()) > 5) {
- JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for "
- "recombination, THIS SHOULD NOT HAPPEN!";
- continue;
+ BaryonrecoMatrix2.push_back(BaryonrecoMatrix3);
+ BaryonrecoMatrix3.clear();
}
+ BaryonrecoMatrix1.push_back(BaryonrecoMatrix2);
+ BaryonrecoMatrix2.clear();
+ }
+ /*
+ //dignostic measure
+ std::cout < 0){JSWARN << "SOMETHING HAS GONE VERY WRONG WITH COLLECTION; HAS " << considering.num() << " ENTRIES (SHOULD BE 0)"; considering.clear();}
-
- considering.add(showerquarks[element[0]]);
- showerquarks[element[0]].status(-991);
-
- for (int q2 = 0; q2 < showerquarks.num() + HH_thermal.num(); ++q2) {
- //set q2 variables here - if we can form a meson, then skip q3 loop
- //also skip q3 loop if q2 is at last quark
+ //looping over all the quarks that we have...
+ //'q1' loops over all quarks in the shower
+ //'q2' loops over all quarks in the event
+ //'q3' loops over all quarks in the event, starting from 'q2' and ending at the last quark
+ //when 'q2' is at the last quark, we will not consider quark 'q3' - can only make a meson at that point...
- double recofactor2 = 0.;
+ parton_collection considering;
+ int element[3];
- //accessing the second considered quark
- //this will skip over non-quark entries in HH_thermal
- if (perm2[q2] > 0) { /*is shower quark*/
- element[1] = perm2[q2] - 1;
- } else { /*is thermal quark*/
- element[1] = perm2[q2] + 1;
- }
+ for(int q1=0;q1 5){JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for recombination, THIS SHOULD NOT HAPPEN!"; continue;}
+
+ //assigning quark values to considering - and setting the status to -991
+ //if at the end of the check, we make a hadron, we will set all -99* entries to 1, else we'll set back to 0
+
+ considering.add(showerquarks[element[0]]);
+ showerquarks[element[0]].status(-991);
+
+ for(int q2=0;q20){/*is shower quark*/ element[1] = perm2[q2] - 1;}
+ else{/*is thermal quark*/ element[1] = perm2[q2] + 1;}
+
+ //checking to see if quark is in shower or thermal
+ //only need to check if quark is the same as the previous quark IFF it is in the shower
+ //only need to check if quark is from the same gluon if it is from a gluon decay in the shower...
+ //want to check if we've messed up and are accessing a gluon if it is in original shower/thermal
+ //need to check if used for all cases...
+ if(perm2[q2]>0){
+ //skipping if current quark has been used
+ if(showerquarks[element[1]].status() != 0 || showerquarks[element[1]].is_used()){continue;}
+ //skipping if current quark is the same as q1
+ else if(element[0]==element[1]){continue;}
+ //skipping if the current quark is from the same gluon as q1
+ else if((showerquarks[element[0]].par() != -1) && (showerquarks[element[0]].par() == showerquarks[element[1]].par())){continue;}
+ //skipping if the current quark is not in the same string as q1 (both are shower partons)
+ //else if(showerquarks[element[1]].string_id() != showerquarks[element[0]].string_id()){continue;}
+ //skipping if current quark is not a u,d,s,c,b quark
+ else if(std::abs(showerquarks[element[1]].id()) > 5){JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for recombination, THIS SHOULD NOT HAPPEN!"; continue;}
+
+ considering.add(showerquarks[element[1]]);
+ showerquarks[element[1]].status(-992);
+ }
+ else if(perm2[q2]<0){
+ //skipping if current quark has been used
+ if(HH_thermal[-element[1]].status() != 0 || HH_thermal[-element[1]].is_used()){continue;}
+ //skipping if current quark is not a u,d,s,c,b quark
+ else if(std::abs(HH_thermal[-element[1]].id()) > 5){JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for recombination, THIS SHOULD NOT HAPPEN!"; continue;}
+ //turning off recombination for thermal partons in a computationally friendly way...
+ else if(th_recofactor < 0.001){continue;}
+
+ //checking distance cut ONLY if this is a thermal parton - skip if dist2 > dist2cut
+ FourVector pos_ptn1 = considering[0].pos(); FourVector pos_ptn2 = HH_thermal[-element[1]].pos();
+ double dt = pos_ptn1.t() - pos_ptn2.t();
+ if(dt > 0.){
+ double dt_E = dt/HH_thermal[-element[1]].e(); // P/E * dT = dist = P*(dT/E)
+ pos_ptn2.Set(pos_ptn2.x()+HH_thermal[-element[1]].px()*dt_E,pos_ptn2.y()+HH_thermal[-element[1]].py()*dt_E,pos_ptn2.z()+HH_thermal[-element[1]].pz()*dt_E,0.);
+ }
+ else{
+ double dt_E = -dt/considering[0].e();
+ pos_ptn1.Set(pos_ptn1.x()+considering[0].px()*dt_E,pos_ptn1.y()+considering[0].py()*dt_E,pos_ptn1.z()+considering[0].pz()*dt_E, 0.);
+ }
+ if(dif2(pos_ptn1,pos_ptn2) > dist2cut){continue;}
- //checking to see if quark is in shower or thermal
- //only need to check if quark is the same as the previous quark IFF it is in the shower
- //only need to check if quark is from the same gluon if it is from a gluon decay in the shower...
- //want to check if we've messed up and are accessing a gluon if it is in original shower/thermal
- //need to check if used for all cases...
- if (perm2[q2] > 0) {
- //skipping if current quark has been used
- if (showerquarks[element[1]].status() != 0 ||
- showerquarks[element[1]].is_used()) {
- continue;
- }
- //skipping if current quark is the same as q1
- else if (element[0] == element[1]) {
- continue;
- }
- //skipping if the current quark is from the same gluon as q1
- else if ((showerquarks[element[0]].par() != -1) &&
- (showerquarks[element[0]].par() ==
- showerquarks[element[1]].par())) {
- continue;
- }
- //skipping if the current quark is not in the same string as q1 (both are shower partons)
- else if (showerquarks[element[1]].string_id() !=
- showerquarks[element[0]].string_id()) {
- continue;
- }
- //skipping if current quark is not a u,d,s quark
- else if (std::abs(showerquarks[element[1]].id()) > 5) {
- JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for "
- "recombination, THIS SHOULD NOT HAPPEN!";
- continue;
- }
-
- considering.add(showerquarks[element[1]]);
- showerquarks[element[1]].status(-992);
- recofactor2 = sh_recofactor * sh_recofactor;
- } else if (perm2[q2] < 0) {
- //skipping if current quark has been used
- if (HH_thermal[-element[1]].status() != 0 ||
- HH_thermal[-element[1]].is_used()) {
- continue;
- }
- //skipping if current quark is not a u,d,s quark
- else if (std::abs(HH_thermal[-element[1]].id()) > 5) {
- JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for "
- "recombination, THIS SHOULD NOT HAPPEN!";
- continue;
- }
- //turning off recombination for thermal partons in a computationally friendly way...
- else if (th_recofactor < 0.0000000001) {
- continue;
- }
-
- //checking distance cut ONLY if this is a thermal parton - skip if dist2 > dist2cut
- FourVector pos_ptn1 = considering[0].pos();
- FourVector pos_ptn2 = HH_thermal[-element[1]].pos();
- double dt = pos_ptn1.t() - pos_ptn2.t();
- if (dt > 0.) {
- double dt_E =
- dt / HH_thermal[-element[1]].e(); // P/E * dT = dist = P*(dT/E)
- pos_ptn2.Set(pos_ptn2.x() + HH_thermal[-element[1]].px() * dt_E,
- pos_ptn2.y() + HH_thermal[-element[1]].py() * dt_E,
- pos_ptn2.z() + HH_thermal[-element[1]].pz() * dt_E, 0.);
- } else {
- double dt_E = -dt / considering[0].e();
- pos_ptn1.Set(pos_ptn1.x() + considering[0].px() * dt_E,
- pos_ptn1.y() + considering[0].py() * dt_E,
- pos_ptn1.z() + considering[0].pz() * dt_E, 0.);
- }
- if (dif2(pos_ptn1, pos_ptn2) > dist2cut) {
- continue;
- }
-
- considering.add(HH_thermal[-element[1]]);
- HH_thermal[-element[1]].status(-992);
- recofactor2 = th_recofactor * sh_recofactor;
- } else {
- JSWARN << "SOMETHING WENT HORRIBLY WRONG - DO NOT KNOW WHERE CURRENT "
- "QUARK CAME FROM?!";
- }
-
- //now that we have two 'acceptable' quarks to check hadron formation against,
- //there is no reason to bother checking if we can make a baryon if we have a q-qbar at this point...
- //will skip third loop in this case - otherwise we will check if we can make a baryon...
- if ((considering[0].id() * considering[1].id() > 0) &&
- (q2 < showerquarks.num() + HH_thermal.num() - 1)) {
- for (int q3 = q2 + 1; q3 < showerquarks.num() + HH_thermal.num();
- ++q3) {
-
- double recofactor3 = recofactor2;
-
- //removing all but the first two entries in the considering collection...
- //this should have been done before, but have this here just in case - remove if this doesn't ever trigger
- //while(considering.num() > 2){considering--;}
- while (considering.num() > 2) {
- considering.partons.pop_back();
- }
-
- //accessing the third considered quark
- //this will skip over non-quark entries in HH_thermal
- if (perm2[q3] > 0) { /*is shower quark*/
- element[2] = perm2[q3] - 1;
- } else { /*is thermal quark*/
- element[2] = perm2[q3] + 1;
- }
-
- //now that we have q3, we need to check if it is valid:
- //q3 needs to be checked if used (all cases)
- //q3 needs to be checked if it is a erroneously accessed parton (not u,d,s quark)
- //q3 needs to be checked against q1 to see if it is the same, or from the same gluon
- //q3 does not need to be checked against q2 to see if it is the same
- //q3 does need to be checked to see if it is from the same gluon as q2
- //q3 needs to be checked to make sure that it can form a baryon with q1 and q2 (checking against either is ok)
- //check: used, sameq1, sameg_q1, sameg_q2, isglu
- if (perm2[q3] > 0) {
- //skipping if the current quark cannot make a baryon with other two quarks
- if (showerquarks[element[2]].id() * considering[0].id() < 0) {
- continue;
- }
- //skipping if current quark has been used
- if (showerquarks[element[2]].status() != 0 ||
- showerquarks[element[2]].is_used()) {
- continue;
- }
- //skipping if current quark is the same as q1
- else if (element[0] == element[2]) {
- continue;
- }
- //skipping if the current quark is from the same gluon as q1
- else if ((showerquarks[element[0]].par() != -1) &&
- (showerquarks[element[0]].par() ==
- showerquarks[element[2]].par())) {
- continue;
- }
- //skipping if the current quark is from the same gluon as q2
- else if ((perm2[q2] > 0) &&
- (showerquarks[element[1]].par() != -1) &&
- (showerquarks[element[1]].par() ==
- showerquarks[element[2]].par())) {
- continue;
- }
- //skipping if the current quark is not in the same string as q1
- //(q2 MUST be in the same string as q1 if it's in the shower, and doesn't need to be checked if thermal)
- else if (showerquarks[element[2]].string_id() !=
- showerquarks[element[0]].string_id()) {
- continue;
- }
- //skipping if current quark is not a u,d,s quark
- else if (std::abs(showerquarks[element[2]].id()) > 5) {
- JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for "
- "recombination, THIS SHOULD NOT HAPPEN!";
- continue;
- }
-
- considering.add(showerquarks[element[2]]);
- showerquarks[element[2]].status(-993);
- recofactor3 = sh_recofactor * recofactor2;
- } else if (perm2[q3] < 0) {
- //skipping if the current quark cannot make a baryon with other two quarks
- if (HH_thermal[-element[2]].id() * considering[0].id() < 0) {
- continue;
- }
- //skipping if current quark has been used
- if (HH_thermal[-element[2]].status() != 0 ||
- HH_thermal[-element[2]].is_used()) {
- continue;
- }
- //skipping if the current quark is from the same gluon as q2
- else if ((perm2[q2] < 0) && (HH_thermal[-element[1]].par() != -1) &&
- (HH_thermal[-element[1]].par() ==
- HH_thermal[-element[2]].par())) {
- continue;
- }
- //skipping if current quark is not a u,d,s quark
- else if (std::abs(HH_thermal[-element[2]].id()) > 5) {
- JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for "
- "recombination, THIS SHOULD NOT HAPPEN!";
- continue;
- }
- //turning off recombination for thermal partons in a computationally friendly way...
- else if (th_recofactor < 0.0000000001) {
- continue;
- }
-
- //checking distance cut ONLY if this is a thermal parton - skip if dist2 > dist2cut
- FourVector pos_ptn1 = considering[0].pos();
- FourVector pos_ptn2 = considering[1].pos();
- FourVector pos_ptn3 = HH_thermal[-element[2]].pos();
- if ((pos_ptn1.t() > pos_ptn2.t()) &&
- (pos_ptn1.t() > pos_ptn3.t())) {
- double dt_E2 = (pos_ptn1.t() - pos_ptn2.t()) / considering[1].e();
- double dt_E3 =
- (pos_ptn1.t() - pos_ptn3.t()) / HH_thermal[-element[2]].e();
- pos_ptn2.Set(pos_ptn2.x() + considering[1].px() * dt_E2,
- pos_ptn2.y() + considering[1].py() * dt_E2,
- pos_ptn2.z() + considering[1].pz() * dt_E2, 0.);
- pos_ptn3.Set(pos_ptn3.x() + HH_thermal[-element[2]].px() * dt_E3,
- pos_ptn3.y() + HH_thermal[-element[2]].py() * dt_E3,
- pos_ptn3.z() + HH_thermal[-element[2]].pz() * dt_E3,
- 0.);
- } else if ((pos_ptn2.t() > pos_ptn1.t()) &&
- (pos_ptn2.t() > pos_ptn3.t())) {
- double dt_E1 = (pos_ptn2.t() - pos_ptn1.t()) / considering[0].e();
- double dt_E3 =
- (pos_ptn2.t() - pos_ptn3.t()) / HH_thermal[-element[2]].e();
- pos_ptn1.Set(pos_ptn1.x() + considering[0].px() * dt_E1,
- pos_ptn1.y() + considering[0].py() * dt_E1,
- pos_ptn1.z() + considering[0].pz() * dt_E1, 0.);
- pos_ptn3.Set(pos_ptn3.x() + HH_thermal[-element[2]].px() * dt_E3,
- pos_ptn3.y() + HH_thermal[-element[2]].py() * dt_E3,
- pos_ptn3.z() + HH_thermal[-element[2]].pz() * dt_E3,
- 0.);
- } else {
- double dt_E1 = (pos_ptn3.t() - pos_ptn1.t()) / considering[0].e();
- double dt_E2 = (pos_ptn3.t() - pos_ptn2.t()) / considering[1].e();
- pos_ptn1.Set(pos_ptn1.x() + considering[0].px() * dt_E1,
- pos_ptn1.y() + considering[0].py() * dt_E1,
- pos_ptn1.z() + considering[0].pz() * dt_E1, 0.);
- pos_ptn2.Set(pos_ptn2.x() + considering[1].px() * dt_E2,
- pos_ptn2.y() + considering[1].py() * dt_E2,
- pos_ptn2.z() + considering[1].pz() * dt_E2, 0.);
- }
- if ((dif2(pos_ptn3, pos_ptn1) > dist2cut) ||
- (dif2(pos_ptn3, pos_ptn2) > dist2cut) ||
- (dif2(pos_ptn1, pos_ptn2) > dist2cut)) {
- continue;
- }
-
- considering.add(HH_thermal[-element[2]]);
- HH_thermal[-element[2]].status(-993);
- recofactor3 = th_recofactor * recofactor2;
- } else {
- JSWARN << "SOMETHING WENT HORRIBLY WRONG - DO NOT KNOW WHERE "
- "CURRENT QUARK CAME FROM?!";
- }
-
- //now that we *could* form a baryon, now we check if we actually do form one
- //baryon momentum
- FourVector Pbaryon;
- Pbaryon.Set(
- considering[0].px() + considering[1].px() + considering[2].px(),
- considering[0].py() + considering[1].py() + considering[2].py(),
- considering[0].pz() + considering[1].pz() + considering[2].pz(),
- 0.);
-
- //baryon(CM) velocity
- FourVector betaB; //really p[i]/e below
- betaB.Set(Pbaryon.x() / (considering[0].e() + considering[1].e() +
- considering[2].e()),
- Pbaryon.y() / (considering[0].e() + considering[1].e() +
- considering[2].e()),
- Pbaryon.z() / (considering[0].e() + considering[1].e() +
- considering[2].e()),
- 0.);
- betaB.Set(
- betaB.x(), betaB.y(), betaB.z(),
- 1. / (sqrt(1. - (betaB.x() * betaB.x() + betaB.y() * betaB.y() +
- betaB.z() * betaB.z()))));
-
- //boosting into CM frame
- FourVector pos_BCM[3], p_BCM[3];
- pos_BCM[0] = considering[0].boost_pos(betaB);
- pos_BCM[1] = considering[1].boost_pos(betaB);
- pos_BCM[2] = considering[2].boost_pos(betaB);
- p_BCM[0] = considering[0].boost_P(betaB);
- p_BCM[1] = considering[1].boost_P(betaB);
- p_BCM[2] = considering[2].boost_P(betaB);
-
- //velocities in CM frame
- FourVector v_BCM[3];
- v_BCM[0].Set(p_BCM[0].x() / p_BCM[0].t(), p_BCM[0].y() / p_BCM[0].t(),
- p_BCM[0].z() / p_BCM[0].t(),
- 0.); //these are really p[i]/e
- v_BCM[1].Set(p_BCM[1].x() / p_BCM[1].t(), p_BCM[1].y() / p_BCM[1].t(),
- p_BCM[1].z() / p_BCM[1].t(), 0.);
- v_BCM[2].Set(p_BCM[2].x() / p_BCM[2].t(), p_BCM[2].y() / p_BCM[2].t(),
- p_BCM[2].z() / p_BCM[2].t(), 0.);
-
- //propagating quarks until time of youngest quark
- double curtime = std::max(std::max(pos_BCM[0].t(), pos_BCM[1].t()),
- pos_BCM[2].t());
- FourVector cur_pos[3];
- cur_pos[0].Set(
- pos_BCM[0].x() + v_BCM[0].x() * (curtime - pos_BCM[0].t()),
- pos_BCM[0].y() + v_BCM[0].y() * (curtime - pos_BCM[0].t()),
- pos_BCM[0].z() + v_BCM[0].z() * (curtime - pos_BCM[0].t()),
- curtime);
- cur_pos[1].Set(
- pos_BCM[1].x() + v_BCM[1].x() * (curtime - pos_BCM[1].t()),
- pos_BCM[1].y() + v_BCM[1].y() * (curtime - pos_BCM[1].t()),
- pos_BCM[1].z() + v_BCM[1].z() * (curtime - pos_BCM[1].t()),
- curtime);
- cur_pos[2].Set(
- pos_BCM[2].x() + v_BCM[2].x() * (curtime - pos_BCM[2].t()),
- pos_BCM[2].y() + v_BCM[2].y() * (curtime - pos_BCM[2].t()),
- pos_BCM[2].z() + v_BCM[2].z() * (curtime - pos_BCM[2].t()),
- curtime);
-
- //finding position of CM at curtime
- FourVector pos_CM;
- pos_CM.Set((cur_pos[0].x() * considering[0].mass() +
- cur_pos[1].x() * considering[1].mass() +
- cur_pos[2].x() * considering[2].mass()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- (cur_pos[0].y() * considering[0].mass() +
- cur_pos[1].y() * considering[1].mass() +
- cur_pos[2].y() * considering[2].mass()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- (cur_pos[0].z() * considering[0].mass() +
- cur_pos[1].z() * considering[1].mass() +
- cur_pos[2].z() * considering[2].mass()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- curtime);
-
- //finding position of baryon in lab frame
- betaB.Set(-betaB.x(), -betaB.y(), -betaB.z(), betaB.t());
- FourVector pos_lab = HHboost(betaB, pos_CM);
-
- //finding relative momenta of partons in CM frame
- FourVector k_rel[2];
- k_rel[0].Set((considering[1].mass() * p_BCM[0].x() -
- considering[0].mass() * p_BCM[1].x()) /
- (considering[0].mass() + considering[1].mass()),
- (considering[1].mass() * p_BCM[0].y() -
- considering[0].mass() * p_BCM[1].y()) /
- (considering[0].mass() + considering[1].mass()),
- (considering[1].mass() * p_BCM[0].z() -
- considering[0].mass() * p_BCM[1].z()) /
- (considering[0].mass() + considering[1].mass()),
- 0.);
- k_rel[1].Set(
- (considering[2].mass() * (p_BCM[0].x() + p_BCM[1].x()) -
- (considering[0].mass() + considering[1].mass()) * p_BCM[2].x()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- (considering[2].mass() * (p_BCM[0].y() + p_BCM[1].y()) -
- (considering[0].mass() + considering[1].mass()) * p_BCM[2].y()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- (considering[2].mass() * (p_BCM[0].z() + p_BCM[1].z()) -
- (considering[0].mass() + considering[1].mass()) * p_BCM[2].z()) /
- (considering[0].mass() + considering[1].mass() +
- considering[2].mass()),
- 0.);
-
- //finding relative positions of partons in CM frame
- FourVector pos_rel[2];
- pos_rel[0].Set((cur_pos[0].x() - cur_pos[1].x()) / sqrt(2.),
- (cur_pos[0].y() - cur_pos[1].y()) / sqrt(2.),
- (cur_pos[0].z() - cur_pos[1].z()) / sqrt(2.), 0.);
- pos_rel[1].Set(((cur_pos[0].x() * considering[0].mass() +
- cur_pos[1].x() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()) -
- cur_pos[2].x()) *
- sqrt(2. / 3.),
- ((cur_pos[0].y() * considering[0].mass() +
- cur_pos[1].y() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()) -
- cur_pos[2].y()) *
- sqrt(2. / 3.),
- ((cur_pos[0].z() * considering[0].mass() +
- cur_pos[1].z() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()) -
- cur_pos[2].z()) *
- sqrt(2. / 3.),
- 0.);
-
- double SigRB2 = SigNucR2;
- double SigLB2 = SigNucL2;
- int sortid[3] = {std::abs(considering[0].id()),
- std::abs(considering[1].id()),
- std::abs(considering[2].id())};
- std::stable_sort(std::begin(sortid), std::end(sortid),
- std::greater());
-
- //for particles we don't want to form, setting recofactor3 to 0
- if (sortid[0] == 3) {
- if (sortid[1] == 3) {
- if (sortid[2] == 3) {
- SigRB2 = SigOmgR2;
- SigLB2 = SigOmgL2;
- } else {
- SigRB2 = SigXiR2;
- SigLB2 = SigXiL2;
- }
- } else {
- SigRB2 = SigSigR2;
- SigLB2 = SigSigL2;
- }
- } else if (sortid[0] == 4) {
- if (sortid[1] == 4) {
- if (sortid[2] == 4) {
- SigRB2 = SigOcccR2;
- SigLB2 = SigOcccL2;
- } else if (sortid[2] == 3) {
- SigRB2 = SigOccR2;
- SigLB2 = SigOccL2;
- } else {
- SigRB2 = SigXiccR2;
- SigLB2 = SigXiccL2;
+ considering.add(HH_thermal[-element[1]]);
+ HH_thermal[-element[1]].status(-992);
+ }
+ else{JSWARN << "SOMETHING WENT HORRIBLY WRONG - DO NOT KNOW WHERE CURRENT QUARK CAME FROM?!";}
+
+ //now that we have two 'acceptable' quarks to check hadron formation against,
+ //there is no reason to bother checking if we can make a baryon if we have a q-qbar at this point...
+ //will skip third loop in this case - otherwise we will check if we can make a baryon...
+ if((considering[0].id()*considering[1].id() > 0) && (q2 < showerquarks.num()+HH_thermal.num()-1) && (maxB_level > -1)){
+ for(int q3=q2+1;q3 2){considering.partons.pop_back();}
+
+ //accessing the third considered quark
+ //this will skip over non-quark entries in HH_thermal
+ if(perm2[q3]>0){/*is shower quark*/ element[2] = perm2[q3] - 1;}
+ else{/*is thermal quark*/ element[2] = perm2[q3] + 1;}
+
+ //now that we have q3, we need to check if it is valid:
+ //q3 needs to be checked if used (all cases)
+ //q3 needs to be checked if it is a erroneously accessed parton (not u,d,s,c,b quark)
+ //q3 needs to be checked against q1 to see if it is the same, or from the same gluon
+ //q3 does not need to be checked against q2 to see if it is the same
+ //q3 does need to be checked to see if it is from the same gluon as q2
+ //q3 needs to be checked to make sure that it can form a baryon with q1 and q2 (checking against either is ok)
+ //check: used, sameq1, sameg_q1, sameg_q2, isglu
+ if(perm2[q3]>0){
+ //skipping if the current quark cannot make a baryon with other two quarks
+ if(showerquarks[element[2]].id()*considering[0].id() < 0){continue;}
+ //skipping if current quark has been used
+ if(showerquarks[element[2]].status() != 0 || showerquarks[element[2]].is_used()){continue;}
+ //skipping if current quark is the same as q1
+ else if(element[0]==element[2] || element[1] == element[2]){continue;}
+ //skipping if the current quark is from the same gluon as q1
+ else if((showerquarks[element[0]].par() != -1) && (showerquarks[element[0]].par() == showerquarks[element[2]].par())){continue;}
+ //skipping if the current quark is from the same gluon as q2
+ else if((perm2[q2] > 0) && (showerquarks[element[1]].par() != -1) && (showerquarks[element[1]].par() == showerquarks[element[2]].par())){continue;}
+ //skipping if the current quark is not in the same string as q1
+ //(q2 MUST be in the same string as q1 if it's in the shower, and doesn't need to be checked if thermal)
+ //else if(showerquarks[element[2]].string_id() != showerquarks[element[0]].string_id()){continue;}
+ //skipping if current quark is not a u,d,s,c,b quark
+ else if(std::abs(showerquarks[element[2]].id()) > 5){JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for recombination, THIS SHOULD NOT HAPPEN!"; continue;}
+
+ considering.add(showerquarks[element[2]]);
+ showerquarks[element[2]].status(-993);
+
+ //TODO: Here we need to establish the probability about baryon formation. first, need to find 3sets of col tag pair are included junction info at the same time. if all 3 does, prob is 1/27, if 1 does, find other component's tag with element[2](3rd one)'s in Meson Matrix!
+ //second, scan all junction lists
+ //before that, we need to evaluate temporary junction, because some junctions could be eliminated, So need to make tempjunction list. then we need to proocedure to erase the factor in the vectors!
+ //Now evaluating the Tempjunctions element first, before this, declare bool variables for the next proocedure
+ bool element1 = false;
+ bool element2 = false;
+ bool element3 = false;
+ int juncnum1 = 999999999;
+ int juncnum2 = 999999999;
+ int juncnum3 = 999999999;
+
+ int standard = IndiceForColFin.size(); // to apply the "if" statement, set the condition for later syntax
+ int tagformatrix = 999999999; // if all three tags are in same junction, we need to set large number not to confuse, this will be filtered by later syntax
+ int loc1 = standard; //location in the MesonrecoMatrix
+ std::vector::iterator I1;
+ std::vector::iterator I2;
+ int loc2 = standard;
+
+ if(considering[0].id()*considering[1].id()*considering[2].id() > 0){
+ // baryon case, and this will check whether two particles are in same junction and the other is not.
+ // and evaluate(by mesonrecomatrix) the other particle's color tag with the the color tag of 3rd particle collected to be baryon.
+ for(int ijunc=0; ijunc < Tempjunctions.size(); ijunc++){
+ if((considering[0].col() == Tempjunctions.at(ijunc).at(1).at(1)) ||
+ (considering[0].col() == Tempjunctions.at(ijunc).at(2).at(1)) ||
+ (considering[0].col() == Tempjunctions.at(ijunc).at(3).at(1)) ) {
+ element1 = true;
+ juncnum1 = ijunc;
+ }
+ if((considering[1].col() == Tempjunctions.at(ijunc).at(1).at(1)) ||
+ (considering[1].col() == Tempjunctions.at(ijunc).at(2).at(1)) ||
+ (considering[1].col() == Tempjunctions.at(ijunc).at(3).at(1)) ) {
+ element2 = true;
+ juncnum2 = ijunc;
+ }
+ if((considering[2].col() == Tempjunctions.at(ijunc).at(1).at(1)) ||
+ (considering[2].col() == Tempjunctions.at(ijunc).at(2).at(1)) ||
+ (considering[2].col() == Tempjunctions.at(ijunc).at(3).at(1)) ) {
+ element3 = true;
+ juncnum3 = ijunc;
+ }
}
- } else if (sortid[1] == 3) {
- if (sortid[2] == 3) {
- SigRB2 = SigOcR2;
- SigLB2 = SigOcL2;
- } else {
- SigRB2 = SigXicR2;
- SigLB2 = SigXicL2;
+ if((juncnum1 == juncnum2) && (juncnum1 != juncnum3) && juncnum1 != 999999999 && considering[2].col() != 0){
+ tagformatrix = Tempjunctions.at(juncnum1).at(3).at(1);
+ I2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), considering[2].col());
+ loc2 = std::distance(IndiceForColFin.begin(), I2);
+ //std::cout < 5){JSWARN << "SOMETHING OTHER THAN u,d,s,c,b WAS considered for recombination, THIS SHOULD NOT HAPPEN!"; continue;}
+ //turning off recombination for thermal partons in a computationally friendly way...
+ else if(th_recofactor < 0.001){continue;}
+
+ //checking distance cut ONLY if this is a thermal parton - skip if dist2 > dist2cut
+ FourVector pos_ptn1 = considering[0].pos(); FourVector pos_ptn2 = considering[1].pos(); FourVector pos_ptn3 = HH_thermal[-element[2]].pos();
+ if((pos_ptn1.t() > pos_ptn2.t()) && (pos_ptn1.t() > pos_ptn3.t())){
+ double dt_E2 = (pos_ptn1.t() - pos_ptn2.t())/considering[1].e(); double dt_E3 = (pos_ptn1.t() - pos_ptn3.t())/HH_thermal[-element[2]].e();
+ pos_ptn2.Set(pos_ptn2.x()+considering[1].px()*dt_E2,pos_ptn2.y()+considering[1].py()*dt_E2,pos_ptn2.z()+considering[1].pz()*dt_E2,0.);
+ pos_ptn3.Set(pos_ptn3.x()+HH_thermal[-element[2]].px()*dt_E3,pos_ptn3.y()+HH_thermal[-element[2]].py()*dt_E3,pos_ptn3.z()+HH_thermal[-element[2]].pz()*dt_E3,0.);
+ }
+ else if((pos_ptn2.t() > pos_ptn1.t()) && (pos_ptn2.t() > pos_ptn3.t())){
+ double dt_E1 = (pos_ptn2.t() - pos_ptn1.t())/considering[0].e(); double dt_E3 = (pos_ptn2.t() - pos_ptn3.t())/HH_thermal[-element[2]].e();
+ pos_ptn1.Set(pos_ptn1.x()+considering[0].px()*dt_E1,pos_ptn1.y()+considering[0].py()*dt_E1,pos_ptn1.z()+considering[0].pz()*dt_E1,0.);
+ pos_ptn3.Set(pos_ptn3.x()+HH_thermal[-element[2]].px()*dt_E3,pos_ptn3.y()+HH_thermal[-element[2]].py()*dt_E3,pos_ptn3.z()+HH_thermal[-element[2]].pz()*dt_E3,0.);
+ }
+ else{
+ double dt_E1 = (pos_ptn3.t() - pos_ptn1.t())/considering[0].e(); double dt_E2 = (pos_ptn3.t() - pos_ptn2.t())/considering[1].e();
+ pos_ptn1.Set(pos_ptn1.x()+considering[0].px()*dt_E1,pos_ptn1.y()+considering[0].py()*dt_E1,pos_ptn1.z()+considering[0].pz()*dt_E1,0.);
+ pos_ptn2.Set(pos_ptn2.x()+considering[1].px()*dt_E2,pos_ptn2.y()+considering[1].py()*dt_E2,pos_ptn2.z()+considering[1].pz()*dt_E2,0.);
+ }
+ if((dif2(pos_ptn3,pos_ptn1) > dist2cut) || (dif2(pos_ptn3,pos_ptn2) > dist2cut) || (dif2(pos_ptn1,pos_ptn2) > dist2cut)){continue;}
+
+ considering.add(HH_thermal[-element[2]]);
+ HH_thermal[-element[2]].status(-993);
+ }else{JSWARN << "SOMETHING WENT HORRIBLY WRONG - DO NOT KNOW WHERE CURRENT QUARK CAME FROM?!";}
+
+ //now that we *could* form a baryon, now we check if we actually do form one
+ //baryon momentum
+ FourVector Pbaryon;
+ Pbaryon.Set(considering[0].px()+considering[1].px()+considering[2].px(),considering[0].py()+considering[1].py()+considering[2].py(),considering[0].pz()+considering[1].pz()+considering[2].pz(),0.);
+
+ //baryon(CM) velocity
+ FourVector betaB; //really p[i]/e below
+ betaB.Set(Pbaryon.x()/(considering[0].e()+considering[1].e()+considering[2].e()),Pbaryon.y()/(considering[0].e()+considering[1].e()+considering[2].e()),Pbaryon.z()/(considering[0].e()+considering[1].e()+considering[2].e()),0.);
+ betaB.Set(betaB.x(),betaB.y(),betaB.z(),1./(sqrt(1. - (betaB.x()*betaB.x() + betaB.y()*betaB.y() + betaB.z()*betaB.z()))));
+
+ //boosting into CM frame
+ FourVector pos_BCM[3], p_BCM[3];
+ pos_BCM[0] = considering[0].boost_pos(betaB); pos_BCM[1] = considering[1].boost_pos(betaB); pos_BCM[2] = considering[2].boost_pos(betaB);
+ p_BCM[0] = considering[0].boost_P(betaB); p_BCM[1] = considering[1].boost_P(betaB); p_BCM[2] = considering[2].boost_P(betaB);
+
+ //velocities in CM frame
+ FourVector v_BCM[3];
+ v_BCM[0].Set(p_BCM[0].x()/p_BCM[0].t(),p_BCM[0].y()/p_BCM[0].t(),p_BCM[0].z()/p_BCM[0].t(),0.); //these are really p[i]/e
+ v_BCM[1].Set(p_BCM[1].x()/p_BCM[1].t(),p_BCM[1].y()/p_BCM[1].t(),p_BCM[1].z()/p_BCM[1].t(),0.);
+ v_BCM[2].Set(p_BCM[2].x()/p_BCM[2].t(),p_BCM[2].y()/p_BCM[2].t(),p_BCM[2].z()/p_BCM[2].t(),0.);
+
+ //propagating quarks until time of youngest quark
+ double curtime = std::max(std::max(pos_BCM[0].t(), pos_BCM[1].t()), pos_BCM[2].t());
+ FourVector cur_pos[3];
+ cur_pos[0].Set(pos_BCM[0].x()+v_BCM[0].x()*(curtime-pos_BCM[0].t()),pos_BCM[0].y()+v_BCM[0].y()*(curtime-pos_BCM[0].t()),pos_BCM[0].z()+v_BCM[0].z()*(curtime-pos_BCM[0].t()),curtime);
+ cur_pos[1].Set(pos_BCM[1].x()+v_BCM[1].x()*(curtime-pos_BCM[1].t()),pos_BCM[1].y()+v_BCM[1].y()*(curtime-pos_BCM[1].t()),pos_BCM[1].z()+v_BCM[1].z()*(curtime-pos_BCM[1].t()),curtime);
+ cur_pos[2].Set(pos_BCM[2].x()+v_BCM[2].x()*(curtime-pos_BCM[2].t()),pos_BCM[2].y()+v_BCM[2].y()*(curtime-pos_BCM[2].t()),pos_BCM[2].z()+v_BCM[2].z()*(curtime-pos_BCM[2].t()),curtime);
+
+ //finding position of CM at curtime
+ FourVector pos_CM;
+ pos_CM.Set(
+ (cur_pos[0].x()*considering[0].mass()+cur_pos[1].x()*considering[1].mass()+cur_pos[2].x()*considering[2].mass())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ (cur_pos[0].y()*considering[0].mass()+cur_pos[1].y()*considering[1].mass()+cur_pos[2].y()*considering[2].mass())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ (cur_pos[0].z()*considering[0].mass()+cur_pos[1].z()*considering[1].mass()+cur_pos[2].z()*considering[2].mass())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ curtime
+ );
+
+ //finding position of baryon in lab frame
+ betaB.Set(-betaB.x(),-betaB.y(),-betaB.z(),betaB.t());
+ FourVector pos_lab = HHboost(betaB, pos_CM);
+
+ //finding relative momenta of partons in CM frame
+ FourVector k_rel_square[2];
+ k_rel_square[0].Set(
+ (considering[1].mass()*p_BCM[0].x()-considering[0].mass()*p_BCM[1].x())/(considering[0].mass()+considering[1].mass()),
+ (considering[1].mass()*p_BCM[0].y()-considering[0].mass()*p_BCM[1].y())/(considering[0].mass()+considering[1].mass()),
+ (considering[1].mass()*p_BCM[0].z()-considering[0].mass()*p_BCM[1].z())/(considering[0].mass()+considering[1].mass()),
+ 0.);
+ k_rel_square[1].Set(
+ (considering[2].mass()*(p_BCM[0].x()+p_BCM[1].x())-(considering[0].mass()+considering[1].mass())*p_BCM[2].x())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ (considering[2].mass()*(p_BCM[0].y()+p_BCM[1].y())-(considering[0].mass()+considering[1].mass())*p_BCM[2].y())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ (considering[2].mass()*(p_BCM[0].z()+p_BCM[1].z())-(considering[0].mass()+considering[1].mass())*p_BCM[2].z())/(considering[0].mass()+considering[1].mass()+considering[2].mass()),
+ 0.);
+
+ //finding relative positions of partons in CM frame
+ FourVector pos_rel_square[2];
+ pos_rel_square[0].Set((cur_pos[0].x()-cur_pos[1].x()),(cur_pos[0].y()-cur_pos[1].y()),(cur_pos[0].z()-cur_pos[1].z()),0.);
+ pos_rel_square[1].Set(
+ ((cur_pos[0].x()*considering[0].mass()+cur_pos[1].x()*considering[1].mass())/(considering[0].mass()+considering[1].mass())-cur_pos[2].x()),
+ ((cur_pos[0].y()*considering[0].mass()+cur_pos[1].y()*considering[1].mass())/(considering[0].mass()+considering[1].mass())-cur_pos[2].y()),
+ ((cur_pos[0].z()*considering[0].mass()+cur_pos[1].z()*considering[1].mass())/(considering[0].mass()+considering[1].mass())-cur_pos[2].z()),
+ 0.);
+
+
+ double SigRB2 = SigNucR2; double SigLB2 = SigNucL2;
+ int sortid[3] = {std::abs(considering[0].id()),std::abs(considering[1].id()),std::abs(considering[2].id())};
+ std::stable_sort(std::begin(sortid), std::end(sortid), std::greater());
+
+ //for particles we don't want to form, setting recofactor3 to 0
+ if( sortid[0] == 3){
+ if( sortid[1] == 3){
+ if( sortid[2] == 3){SigRB2 = SigOmgR2; SigLB2 = SigOmgL2;}
+ else{ SigRB2 = SigXiR2; SigLB2 = SigXiL2;}
+ }
+ else{ SigRB2 = SigSigR2; SigLB2 = SigSigL2;}
+ }
+ else if(sortid[0] == 4){
+ if( sortid[1] == 4){
+ if( sortid[2] == 4){SigRB2 = SigOcccR2; SigLB2 = SigOcccL2;}
+ else if(sortid[2] == 3){SigRB2 = SigOccR2; SigLB2 = SigOccL2;}
+ else{ SigRB2 = SigXiccR2; SigLB2 = SigXiccL2;}
+ }
+ else if(sortid[1] == 3){
+ if( sortid[2] == 3){SigRB2 = SigOcR2; SigLB2 = SigOcL2;}
+ else{ SigRB2 = SigXicR2; SigLB2 = SigXicL2;}
+ }
+ else{ SigRB2 = SigSigcR2; SigLB2 = SigSigcL2;}
+ }
+ else if(sortid[0] == 5){
+ if( sortid[1] == 5){
+ if( sortid[2] == 5){SigRB2 = SigObbbR2; SigLB2 = SigObbbL2; recofactor3=0.;}
+ else if(sortid[2] == 4){SigRB2 = SigObbcR2; SigLB2 = SigObbcL2; recofactor3=0.;}
+ else if(sortid[2] == 3){SigRB2 = SigObbR2; SigLB2 = SigObbL2; recofactor3=0.;}
+ else{ SigRB2 = SigXibbR2; SigLB2 = SigXibbL2; recofactor3=0.;}
+ }
+ else if(sortid[1] == 4){
+ if( sortid[2] == 4){SigRB2 = SigObccR2; SigLB2 = SigObccL2; recofactor3=0.;}
+ else if(sortid[2] == 3){SigRB2 = SigObcR2; SigLB2 = SigObcL2; recofactor3=0.;}
+ else{ SigRB2 = SigXibcR2; SigLB2 = SigXibcL2; recofactor3=0.;}
+ }
+ else if(sortid[1] == 3){
+ if( sortid[2] == 3){SigRB2 = SigObR2; SigLB2 = SigObL2;}
+ else{ SigRB2 = SigXibR2; SigLB2 = SigXibL2;}
+ }
+ else{ SigRB2 = SigSigbR2; SigLB2 = SigSigbL2;}
+ }
+
+ //precalc's for Wigner Wavefunction
+ //0:x, 1:y, 2:z ::: urho:(rel. between partons 1,2), ulamb:(rel. between partons (1,2),3)
+ double urho[3], ulamb[3];
+ urho[0] = 0.5*(pos_rel_square[0].x()*pos_rel_square[0].x()/SigRB2 + k_rel_square[0].x()*k_rel_square[0].x()*SigRB2/hbarc2);
+ urho[1] = 0.5*(pos_rel_square[0].y()*pos_rel_square[0].y()/SigRB2 + k_rel_square[0].y()*k_rel_square[0].y()*SigRB2/hbarc2);
+ urho[2] = 0.5*(pos_rel_square[0].z()*pos_rel_square[0].z()/SigRB2 + k_rel_square[0].z()*k_rel_square[0].z()*SigRB2/hbarc2);
+ ulamb[0] = 0.5*(pos_rel_square[1].x()*pos_rel_square[1].x()/SigLB2 + k_rel_square[1].x()*k_rel_square[1].x()*SigLB2/hbarc2);
+ ulamb[1] = 0.5*(pos_rel_square[1].y()*pos_rel_square[1].y()/SigLB2 + k_rel_square[1].y()*k_rel_square[1].y()*SigLB2/hbarc2);
+ ulamb[2] = 0.5*(pos_rel_square[1].z()*pos_rel_square[1].z()/SigLB2 + k_rel_square[1].z()*k_rel_square[1].z()*SigLB2/hbarc2);
+
+ //1D GS Wig. wavefunction
+ double wig0[2][3];
+ wig0[0][0] = std::exp(-urho[0]); wig0[0][1] = std::exp(-urho[1]); wig0[0][2] = std::exp(-urho[2]);
+ wig0[1][0] = std::exp(-ulamb[0]); wig0[1][1] = std::exp(-ulamb[1]); wig0[1][2] = std::exp(-ulamb[2]);
+ //3D GS Wig. wavefunction
+ double WigB[2];
+ WigB[0] = wig0[0][0]*wig0[0][1]*wig0[0][2]*wig0[1][0]*wig0[1][1]*wig0[1][2];
+
+ //summing up 3D Wig. wavefunctions over nlev excited states
+ WigB[1] = 0.; //sumWigB;
+
+ double wigE[2][3];
+ for(int i=0;i<2;++i){for(int j=0;j<2;++j){wigE[i][j]=0.;}}
+
+ wigE[0][0] = wig0[0][0];
+ for(int iRx=0; iRx<=maxB_level; ++iRx){
+ wigE[0][1] = wig0[0][1];
+ for(int iRy=0; iRy<=maxB_level-iRx; ++iRy){
+ wigE[0][2] = wig0[0][2];
+ for(int iRz=0; iRz<=maxB_level-iRx-iRy; ++iRz){
+ wigE[1][0] = wig0[1][0];
+ for(int iLx=0; iLx<=maxB_level-iRx-iRy-iRz; ++iLx){
+ wigE[1][1] = wig0[1][1];
+ for(int iLy=0; iLy<=maxB_level-iRx-iRy-iRz-iLx; ++iLy){
+ wigE[1][2] = wig0[1][2];
+ for(int iLz=0; iLz<=maxB_level-iRx-iRy-iRz-iLx-iLy; ++iLz){
+ WigB[1] += wigE[0][0]*wigE[0][1]*wigE[0][2]*wigE[1][0]*wigE[1][1]*wigE[1][2];
+ wigE[1][2] *= ulamb[2]/((double(iLz))+1.);
+ }
+ wigE[1][1] *= ulamb[1]/((double(iLy))+1.);
+ }
+ wigE[1][0] *= ulamb[0]/((double(iLx))+1.);
+ }
+ wigE[0][2] *= urho[2]/((double(iRz))+1.);
+ }
+ wigE[0][1] *= urho[1]/((double(iRy))+1.);
+ }
+ wigE[0][0] *= urho[0]/((double(iRx))+1.);
+ }
+
+ if(maxB_level == -1) {
+ WigB[0] = 0.;
+ WigB[1] = 0.;
}
- wigE[0][0] = wig0[0][0];
- for (int iRx = 0; iRx <= maxE_level; ++iRx) {
- wigE[0][1] = wig0[0][1];
- for (int iRy = 0; iRy <= maxE_level - iRx; ++iRy) {
- wigE[0][2] = wig0[0][2];
- for (int iRz = 0; iRz <= maxE_level - iRx - iRy; ++iRz) {
- wigE[1][0] = wig0[1][0];
- for (int iLx = 0; iLx <= maxE_level - iRx - iRy - iRz; ++iLx) {
- wigE[1][1] = wig0[1][1];
- for (int iLy = 0; iLy <= maxE_level - iRx - iRy - iRz - iLx;
- ++iLy) {
- wigE[1][2] = wig0[1][2];
- for (int iLz = 0;
- iLz <= maxE_level - iRx - iRy - iRz - iLx - iLy;
- ++iLz) {
- WigB[1] += wigE[0][0] * wigE[0][1] * wigE[0][2] *
- wigE[1][0] * wigE[1][1] * wigE[1][2];
- wigE[1][2] *= ulamb[2] / ((double(iLz)) + 1.);
- }
- wigE[1][1] *= ulamb[1] / ((double(iLy)) + 1.);
+ //Checking if baryon is formed (either ground or excited state)
+ double rndbaryon = ran();
+ double mult = (considering[1].is_thermal() || considering[2].is_thermal()) ? th_recofactor : sh_recofactor;
+ if(WigB[1]*recofactor3*mult >= rndbaryon){
+ int junction_with_thermal_parton = 0;
+
+ /*std::cout << "Baryons" << std::endl;
+ std::cout << considering[0].id() << "," << considering[0].col() << "," << considering[0].acol() << std::endl;
+ std::cout << considering[1].id() << "," << considering[1].col() << "," << considering[1].acol() << std::endl;
+ std::cout << considering[2].id() << "," << considering[2].col() << "," << considering[2].acol() << std::endl;*/
+ if(considering[0].id() * considering[1].id() * considering[2].id() > 0){
+ // baryon is to be formed, so temporary anti junction is defined
+ if(considering[0].col() > 0 && considering[1].col() > 0 && considering[2].col() > 0) {
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+ IdColInfo1.push_back(junction_with_thermal_parton);
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(considering[0].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(considering[1].col());
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(considering[2].col());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+
+ //Since Baryon is formed, color tag for neutrality should be added, casting int into double is needed!!(ex (double) intvalue
+ int coltag1 = considering[0].col();
+ int coltag2 = considering[1].col();
+ int coltag3 = considering[2].col();
+ if(coltag1 > 0 && coltag2 > 0 && coltag3 > 0 && coltag1 <= limit && coltag2 <= limit && coltag3 <= limit ){
+ double tag1 = (double)coltag1; // they are casted to be inserted into the matrix(since it's vector of double
+ double tag2 = (double)coltag2;
+ double tag3 = (double)coltag3;
+ std::vector::iterator I1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag1);
+ std::vector::iterator I2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag2);
+ std::vector::iterator I3 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag3);
+ int loc1 = std::distance(IndiceForColFin.begin(), I1);
+ int loc2 = std::distance(IndiceForColFin.begin(), I2);
+ int loc3 = std::distance(IndiceForColFin.begin(), I3); //set up for find matrix indices corresponding to the color tags (we just found the corresponding indice in BaryonrecoMatrix1 with col tags )
+
+ BaryonrecoMatrix1.at(loc1).at(loc2).at(1) = tag3;
+ BaryonrecoMatrix1.at(loc2).at(loc1).at(1) = tag3;
+ BaryonrecoMatrix1.at(loc2).at(loc3).at(1) = tag1;
+ BaryonrecoMatrix1.at(loc3).at(loc2).at(1) = tag1;
+ BaryonrecoMatrix1.at(loc1).at(loc3).at(1) = tag2;
+ BaryonrecoMatrix1.at(loc3).at(loc1).at(1) = tag2; // now the color tag info for color neutrality is saved in Matrix, also we need to consider the impact from this to Meson Formation
+
+ //MesonrecoMatrix1 is modified by below
+ MesonrecoMatrix1.at(loc1).at(loc2) = 0;
+ MesonrecoMatrix1.at(loc2).at(loc1) = 0;
+ MesonrecoMatrix1.at(loc2).at(loc3) = 0;
+ MesonrecoMatrix1.at(loc3).at(loc2) = 0;
+ MesonrecoMatrix1.at(loc1).at(loc3) = 0;
+ MesonrecoMatrix1.at(loc3).at(loc1) = 0; // since three color tags of baryon are different from each other, so that these tags can't form meson with each other.
+ }
+ /*
+ //dignostic measure
+ std::cout <= rndbaryon) {
- //*******************************************string repair functionality below*******************************************
- //determining which partons, of the 3 being considered, are endpoints
- int numendpoints = 0;
- bool is_endpoint[3];
- is_endpoint[0] = false;
- is_endpoint[1] = false;
- is_endpoint[2] = false;
- if (showerquarks[element[0]].is_strendpt()) {
- ++numendpoints;
- is_endpoint[0] = true;
- }
- if (perm2[q2] > 0) {
- if (showerquarks[element[1]].is_strendpt()) {
- ++numendpoints;
- is_endpoint[1] = true;
- }
- } else {
- if (HH_thermal[-element[1]].is_strendpt()) {
- ++numendpoints;
- is_endpoint[1] = true;
+ //dignostic measure
+ std::cout < 0) {
- if (showerquarks[element[2]].is_strendpt()) {
- ++numendpoints;
- is_endpoint[2] = true;
+ if(considering[0].col() > 0 && considering[1].col() == 0 && considering[2].col() > 0){ // MAT + Therm/LBT + MAT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0){showerquarks[loc - 1].acol(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ //TODO: give thermal partons "ANTI COLOR TAGS" to form anti junction and conserve baryon number.
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(considering[0].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(maxtag);
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(considering[2].col());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
}
- } else {
- if (HH_thermal[-element[2]].is_strendpt()) {
- ++numendpoints;
- is_endpoint[2] = true;
+ if(considering[0].col() == 0 && considering[1].col() > 0 && considering[2].col() > 0){ // LBT + MAT + MAT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0){showerquarks[loc - 1].acol(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(maxtag);// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(considering[1].col());
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(considering[2].col());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
}
- }
+ if(considering[0].col() > 0 && considering[1].col() > 0 && considering[2].col() == 0){ // MAT + MAT + Therm/LBT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0){showerquarks[loc - 1].acol(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(considering[0].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(considering[1].col());
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(maxtag);
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with one zero component.
+ if(considering[0].col() > 0 && considering[1].col() == 0 && considering[2].col() == 0){ // MAT + Therm/LBT + Therm/LBT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(considering[0].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0){showerquarks[loc1 - 1].acol(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].acol(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
- //setting the current string value to the value of the first parton (since it can't be a thermal parton)
- //this might have to change if partons are allowed to recombine from multiple strings...
- int cur_str = showerquarks[element[0]].string_id();
+ IdColInfo1.push_back(junction_with_thermal_parton);
- if (numendpoints == 0) { //no parton is an endpoint
- //finding the string id for this new string...
- //start by choosing an appropriate string 'index'
- int new_str = 100 * cur_str + 1;
- //looping over all the current string indices to determine if the first guess is unique; if not, then increment it until it is unique
- //this might loop multiple times if the chosen index is not unique
- {
- int i = 0;
- while (i < list_strs.size()) {
- if (new_str == list_strs[i]) {
- ++new_str;
- i = -1;
- }
- ++i;
- }
- }
- //need to add the new string to the list of strings, unless for some reason we want to keep adding the 'new' strings together?
- list_strs.push_back(new_str);
-
- showerquarks[showerquarks[element[0]].sibling()].string_id(
- new_str);
- showerquarks[showerquarks[element[0]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[0]].sibling()].pos_str(0);
- showerquarks[showerquarks[element[0]].sibling()].endpt_id(1);
- if (perm2[q2] > 0) {
- showerquarks[showerquarks[element[1]].sibling()].string_id(
- new_str);
- showerquarks[showerquarks[element[1]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[1]].sibling()].pos_str(0);
- showerquarks[showerquarks[element[1]].sibling()].endpt_id(1);
- } else {
- int thermsib = findthermalsibling(-element[1], HH_thermal);
- HH_thermal[thermsib].string_id(new_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(0);
- HH_thermal[thermsib].endpt_id(1);
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(maxtag);
+ //TODO: give thermal partons "ANTI COLOR TAGS" to form anti junction and conserve baryon number.
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
}
- if (perm2[q3] > 0) {
- showerquarks[showerquarks[element[2]].sibling()].string_id(
- new_str);
- showerquarks[showerquarks[element[2]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[2]].sibling()].pos_str(0);
- showerquarks[showerquarks[element[2]].sibling()].endpt_id(1);
- } else {
- int thermsib = findthermalsibling(-element[2], HH_thermal);
- HH_thermal[thermsib].string_id(new_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(0);
- HH_thermal[thermsib].endpt_id(1);
+ if(considering[0].col() == 0 && considering[1].col() == 0 && considering[2].col() > 0){ // Therm/LBT + Therm/LBT + MAT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0){showerquarks[loc1 - 1].acol(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].acol(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(maxtag);
+ //TODO: give thermal partons "ANTI COLOR TAGS" to form anti junction and conserve baryon number.
+
+ IdColInfo2.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo2.push_back(considering[2].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
}
- } else if (numendpoints == 1) { //only one parton is an endpoint
- if (is_endpoint[0]) {
- if (perm2[q2] > 0) {
- showerquarks[showerquarks[element[1]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[1]].sibling()].pos_str(
- showerquarks[element[0]].pos_str());
- showerquarks[showerquarks[element[1]].sibling()].endpt_id(
- showerquarks[element[0]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[1], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- showerquarks[element[0]].pos_str());
- HH_thermal[thermsib].endpt_id(
- showerquarks[element[0]].endpt_id());
+ if(considering[0].col() == 0 && considering[1].col() > 0 && considering[2].col() == 0){ // Therm/LBT + Therm/LBT + MAT case
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0) {
- showerquarks[showerquarks[element[2]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[2]].sibling()].pos_str(
- showerquarks[element[0]].pos_str());
- showerquarks[showerquarks[element[2]].sibling()].endpt_id(
- showerquarks[element[0]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[2], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- showerquarks[element[0]].pos_str());
- HH_thermal[thermsib].endpt_id(
- showerquarks[element[0]].endpt_id());
+ else if (loc1 > 0){showerquarks[loc1 - 1].acol(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo2.push_back(-1);
+ IdColInfo2.push_back(maxtag);
+
+ IdColInfo3.push_back(-1); // means anticolor tag(negative color charge)
+ IdColInfo3.push_back(considering[1].col());// {-1, anticolor tag} will be at 2nd, 3rd, 4th
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0) {
- showerquarks[showerquarks[element[0]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[0]].sibling()].pos_str(
- showerquarks[element[1]].pos_str());
- showerquarks[showerquarks[element[0]].sibling()].endpt_id(
- showerquarks[element[1]].endpt_id());
- if (perm2[q3] > 0) {
- showerquarks[showerquarks[element[2]].sibling()]
- .is_strendpt(true);
- showerquarks[showerquarks[element[2]].sibling()].pos_str(
- showerquarks[element[1]].pos_str());
- showerquarks[showerquarks[element[2]].sibling()].endpt_id(
- showerquarks[element[1]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[2], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- showerquarks[element[1]].pos_str());
- HH_thermal[thermsib].endpt_id(
- showerquarks[element[1]].endpt_id());
- }
- } else { //these shouldn't actually ever trigger, as thermal partons should never 'be' an endpoint...
- showerquarks[showerquarks[element[0]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[0]].sibling()].pos_str(
- HH_thermal[-element[1]].pos_str());
- showerquarks[showerquarks[element[0]].sibling()].endpt_id(
- HH_thermal[-element[1]].endpt_id());
- if (perm2[q3] > 0) {
- showerquarks[showerquarks[element[2]].sibling()]
- .is_strendpt(true);
- showerquarks[showerquarks[element[2]].sibling()].pos_str(
- HH_thermal[-element[1]].pos_str());
- showerquarks[showerquarks[element[2]].sibling()].endpt_id(
- HH_thermal[-element[1]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[2], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- HH_thermal[-element[1]].pos_str());
- HH_thermal[thermsib].endpt_id(
- HH_thermal[-element[1]].endpt_id());
- }
+ else if (loc2 > 0){showerquarks[loc2 - 1].acol(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(maxtag);
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with two zeros
+ if(considering[0].col() == 0 && considering[1].col() == 0 && considering[2].col() == 0){ // Therm/LBT + Therm/LBT + Therm/LBT
+ IdColInfo1.push_back(-1); // antijunction tag(-1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0) {
- showerquarks[showerquarks[element[0]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[0]].sibling()].pos_str(
- showerquarks[element[2]].pos_str());
- showerquarks[showerquarks[element[0]].sibling()].endpt_id(
- showerquarks[element[2]].endpt_id());
- if (perm2[q2] > 0) {
- showerquarks[showerquarks[element[1]].sibling()]
- .is_strendpt(true);
- showerquarks[showerquarks[element[1]].sibling()].pos_str(
- showerquarks[element[2]].pos_str());
- showerquarks[showerquarks[element[1]].sibling()].endpt_id(
- showerquarks[element[2]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[1], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- showerquarks[element[2]].pos_str());
- HH_thermal[thermsib].endpt_id(
- showerquarks[element[2]].endpt_id());
- }
- } else { //these shouldn't actually ever trigger, as thermal partons should never 'be' an endpoint...
- showerquarks[showerquarks[element[0]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[0]].sibling()].pos_str(
- HH_thermal[-element[2]].pos_str());
- showerquarks[showerquarks[element[0]].sibling()].endpt_id(
- HH_thermal[-element[2]].endpt_id());
- if (perm2[q2] > 0) {
- showerquarks[showerquarks[element[1]].sibling()]
- .is_strendpt(true);
- showerquarks[showerquarks[element[1]].sibling()].pos_str(
- HH_thermal[-element[2]].pos_str());
- showerquarks[showerquarks[element[1]].sibling()].endpt_id(
- HH_thermal[-element[2]].endpt_id());
- } else {
- int thermsib = findthermalsibling(-element[1], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(
- HH_thermal[-element[2]].pos_str());
- HH_thermal[thermsib].endpt_id(
- HH_thermal[-element[2]].endpt_id());
- }
+ else if (loc1 > 0){showerquarks[loc1 - 1].acol(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo2.push_back(-1);
+ IdColInfo2.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].acol(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo3.push_back(-1);
+ IdColInfo3.push_back(maxtag);
+
+ maxtag++;
+ int loc3 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc3 == 999999999){
+ //std::cout < 0){showerquarks[loc3 - 1].acol(maxtag); }
+ else if(loc3 < 0){HH_thermal[-loc3 - 1].acol(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo4.push_back(-1);
+ IdColInfo4.push_back(maxtag);
+
+ //TODO: give thermal partons "ANTI COLOR TAGS" to form anti junction and conserve baryon number.
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with three zeros
+ }
+ else if(considering[0].id() * considering[1].id() * considering[2].id() < 0){ //anti baryon would be made so temporary junction is defined
+ //std::cout < 0 && considering[1].acol() > 0 && considering[2].acol() > 0){
+ IdColInfo1.push_back(1); // junction tag(1)
+ IdColInfo1.push_back(junction_with_thermal_parton); // zero(just room for the other usage) : {1, 0} at 1st
+ IdColInfo2.push_back(1); // means color tag(positive color charge) : { 1, color tag } at 2nd, 3rd, 4th in the vector
+ IdColInfo2.push_back(considering[0].acol());
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(considering[1].acol());
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(considering[2].acol());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+
+ //Since Baryon is formed, color tag for neutrality should be added, casting int into double is needed!!(ex (double) intvalue
+ int coltag1 = considering[0].acol();
+ int coltag2 = considering[1].acol();
+ int coltag3 = considering[2].acol();
+ if(coltag1 > 0 && coltag2 > 0 && coltag3 > 0 && coltag1 <= limit && coltag2 <= limit && coltag3 <= limit ){
+ double tag1 = (double)coltag1; // they are casted to be inserted into the matrix(since it's vector of double
+ double tag2 = (double)coltag2;
+ double tag3 = (double)coltag3;
+ std::vector::iterator I1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag1);
+ std::vector::iterator I2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag2);
+ std::vector::iterator I3 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), coltag3);
+ int loc1 = std::distance(IndiceForColFin.begin(), I1);
+ int loc2 = std::distance(IndiceForColFin.begin(), I2);
+ int loc3 = std::distance(IndiceForColFin.begin(), I3); //set up for find matrix indices corresponding to the color tags (we just found the corresponding indice in BaryonrecoMatrix1 with col tags )
+
+ BaryonrecoMatrix1.at(loc1).at(loc2).at(1) = tag3;
+ BaryonrecoMatrix1.at(loc2).at(loc1).at(1) = tag3;
+ BaryonrecoMatrix1.at(loc2).at(loc3).at(1) = tag1;
+ BaryonrecoMatrix1.at(loc3).at(loc2).at(1) = tag1;
+ BaryonrecoMatrix1.at(loc1).at(loc3).at(1) = tag2;
+ BaryonrecoMatrix1.at(loc3).at(loc1).at(1) = tag2; // now the color tag info for color neutrality is saved in Matrix, also we need to consider the impact from this to Meson Formation
+
+ //MesonrecoMatrix1 is modified by below
+ MesonrecoMatrix1.at(loc1).at(loc2) = 0;
+ MesonrecoMatrix1.at(loc2).at(loc1) = 0;
+ MesonrecoMatrix1.at(loc2).at(loc3) = 0;
+ MesonrecoMatrix1.at(loc3).at(loc2) = 0;
+ MesonrecoMatrix1.at(loc1).at(loc3) = 0;
+ MesonrecoMatrix1.at(loc3).at(loc1) = 0; // since three color tags of baryon are different from each other, so that these tags can't form meson with each other.
}
}
- } else if (numendpoints ==
- 2) { //there are two partons that are endpoints
- if (!is_endpoint[0]) {
- //replacing the closest endpoint with the sibling of the non-endpoint parton
- //since neither of the endpoints can be thermal partons, don't need to check
- int set_pos, set_endptid;
- if (std::abs(showerquarks[element[0]].pos_str() -
- showerquarks[element[1]].pos_str()) <=
- std::abs(showerquarks[element[0]].pos_str() -
- showerquarks[element[2]].pos_str())) {
- set_pos = showerquarks[element[1]].pos_str();
- set_endptid = showerquarks[element[1]].endpt_id();
- } else {
- set_pos = showerquarks[element[2]].pos_str();
- set_endptid = showerquarks[element[2]].endpt_id();
+ if(considering[0].acol() > 0 && considering[1].acol() == 0 && considering[2].acol() > 0){ // MAT + Therm/LBT + MAT case
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0){showerquarks[loc - 1].col(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].col(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo2.push_back(1); // color tag
+ IdColInfo2.push_back(considering[0].acol());// {1, color tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(maxtag);
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(considering[2].acol());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ }
+ if(considering[0].acol() == 0 && considering[1].acol() > 0 && considering[2].acol() > 0){ // LBT + MAT + MAT case
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0) {
- showerquarks[showerquarks[element[1]].sibling()].string_id(
- cur_str);
- showerquarks[showerquarks[element[1]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[1]].sibling()].pos_str(
- set_pos);
- showerquarks[showerquarks[element[1]].sibling()].endpt_id(
- set_endptid);
- } else {
- int thermsib = findthermalsibling(-element[1], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(set_pos);
- HH_thermal[thermsib].endpt_id(set_endptid);
+ else if (loc > 0){showerquarks[loc - 1].col(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].col(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo2.push_back(1); // means color tag
+ IdColInfo2.push_back(maxtag);// {1, color tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(considering[1].acol());
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(considering[2].acol());
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ }
+ if(considering[0].acol() > 0 && considering[1].acol() > 0 && considering[2].acol() == 0){ // MAT + MAT + Therm/LBT case
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc == 999999999){
+ //std::cout < 0){showerquarks[loc - 1].col(maxtag); }
+ else if(loc < 0){HH_thermal[-loc - 1].col(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo2.push_back(1); // means color tag
+ IdColInfo2.push_back(considering[0].acol());// {1, color tag} will be at 2nd, 3rd, 4th
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(considering[1].acol());
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(maxtag);
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with one zero component.
+ if(considering[0].acol() > 0 && considering[1].acol() == 0 && considering[2].acol() == 0){ // MAT + Therm/LBT + Therm/LBT case
+ IdColInfo1.push_back(1); // junction tag(1)
+ IdColInfo2.push_back(1); // means color tag
+ IdColInfo2.push_back(considering[0].acol());// {1, color tag} will be at 2nd, 3rd, 4th
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0) {
- showerquarks[showerquarks[element[2]].sibling()].string_id(
- cur_str);
- showerquarks[showerquarks[element[2]].sibling()].is_strendpt(
- true);
- showerquarks[showerquarks[element[2]].sibling()].pos_str(
- set_pos);
- showerquarks[showerquarks[element[2]].sibling()].endpt_id(
- set_endptid);
- } else {
- int thermsib = findthermalsibling(-element[2], HH_thermal);
- HH_thermal[thermsib].string_id(cur_str);
- HH_thermal[thermsib].is_strendpt(true);
- HH_thermal[thermsib].pos_str(set_pos);
- HH_thermal[thermsib].endpt_id(set_endptid);
+ else if (loc1 > 0){showerquarks[loc1 - 1].col(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0) {
- formedhadron.add_par(showerquarks[element[1]].par());
- } else {
- formedhadron.add_par(element[1]);
- }
- if (perm2[q3] > 0) {
- formedhadron.add_par(showerquarks[element[2]].par());
- } else {
- formedhadron.add_par(element[2]);
- }
+ else if (loc2 > 0){showerquarks[loc2 - 1].col(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].col(maxtag); junction_with_thermal_parton = 1;}
- //now setting if baryon is in excited state
- if (WigB[0] * recofactor3 < rndbaryon) {
- formedhadron.is_excited(true);
- }
+ IdColInfo1.push_back(junction_with_thermal_parton);
- //setting if there are any thermal partons used to make the hadron (with the '0' thermal parent as -99999 so that it doesn't conflict with '0' shower parton)
- if (perm2[q2] > 0 && perm2[q3] > 0) { /*is sh-sh-sh*/
- formedhadron.is_shsh(true);
- } else if (perm2[q2] > 0 && perm2[q3] < 0) { /*is sh-sh-th*/
- formedhadron.is_shth(true);
- if (element[2] == 0) {
- formedhadron.parents[2] = -99999;
- }
- } else if (perm2[q2] < 0 && perm2[q3] > 0) { /*is sh-th-sh*/
- formedhadron.is_shth(true);
- if (element[1] == 0) {
- formedhadron.parents[1] = -99999;
- }
- } else if (perm2[q2] < 0 && perm2[q3] < 0) { /*is sh-th-th*/
- formedhadron.is_shth(true);
- if (element[1] == 0) {
- formedhadron.parents[1] = -99999;
- }
- if (element[2] == 0) {
- formedhadron.parents[2] = -99999;
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(maxtag);
+ //TODO: give thermal partons "ANTI COLOR TAGS" to form anti junction and conserve baryon number.
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
}
- }
+ if(considering[0].acol() == 0 && considering[1].acol() == 0 && considering[2].acol() > 0){ // Therm/LBT + Therm/LBT + MAT case
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0){showerquarks[loc1 - 1].col(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo2.push_back(1);
+ IdColInfo2.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].col(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].col(maxtag); junction_with_thermal_parton = 1;}
- //setting hadron position and momentum vectors
- Pbaryon.Set(Pbaryon.x(), Pbaryon.y(), Pbaryon.z(),
- sqrt(Pbaryon.x() * Pbaryon.x() +
- Pbaryon.y() * Pbaryon.y() +
- Pbaryon.z() * Pbaryon.z() +
- formedhadron.mass() * formedhadron.mass()));
- formedhadron.pos(pos_lab);
- formedhadron.P(Pbaryon);
-
- //need to choose *what* hadron we've formed... base this on the parton id's, mass, & if excited
- //might want to do this differently? void f'n(partoncollection, formedhadron)?
- set_baryon_id(considering, formedhadron);
-
- //need to add the hadron to the collection
- HH_hadrons.add(formedhadron);
-
- //now that we've formed the hadron, need to set ALL (3) the 'considering' flags to used
- showerquarks[element[0]].status(1);
- showerquarks[element[0]].is_used(true);
- if (perm2[q2] > 0) {
- showerquarks[element[1]].status(1);
- showerquarks[element[1]].is_used(true);
- } else {
- HH_thermal[-element[1]].status(1);
- HH_thermal[-element[1]].is_used(true);
- }
- if (perm2[q3] > 0) {
- showerquarks[element[2]].status(1);
- showerquarks[element[2]].is_used(true);
- } else {
- HH_thermal[-element[2]].status(1);
- HH_thermal[-element[2]].is_used(true);
- }
+ IdColInfo1.push_back(junction_with_thermal_parton);
- madehadron = true;
- considering.clear();
- break;
- } else {
- //since we've not formed a baryon on this try, need to revert the third quark 'considering' used flag
- if (perm2[q3] > 0) {
- showerquarks[element[2]].status(0);
- } else {
- HH_thermal[-element[2]].status(0);
- }
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(maxtag);
- //and remove the third entry in considering
- considering.partons.pop_back(); //considering--;
- }
- }
- } else if (considering[0].id() * considering[1].id() < 0) {
- //now that we *could* form a meson, now we check if we actually do form one
- //meson momentum
- FourVector Pmeson;
- Pmeson.Set(considering[0].px() + considering[1].px(),
- considering[0].py() + considering[1].py(),
- considering[0].pz() + considering[1].pz(), 0.);
-
- //meson(CM) velocity
- FourVector betaM;
- betaM.Set(Pmeson.x() / (considering[0].e() + considering[1].e()),
- Pmeson.y() / (considering[0].e() + considering[1].e()),
- Pmeson.z() / (considering[0].e() + considering[1].e()), 0.);
- betaM.Set(
- betaM.x(), betaM.y(), betaM.z(),
- 1. / (sqrt(1. - (betaM.x() * betaM.x() + betaM.y() * betaM.y() +
- betaM.z() * betaM.z()))));
-
- //boosting into CM frame
- FourVector pos_MCM[2], p_MCM[2];
- pos_MCM[0] = considering[0].boost_pos(betaM);
- pos_MCM[1] = considering[1].boost_pos(betaM);
- p_MCM[0] = considering[0].boost_P(betaM);
- p_MCM[1] = considering[1].boost_P(betaM);
-
- //velocities in CM frame
- FourVector v_MCM[2];
- v_MCM[0].Set(p_MCM[0].x() / p_MCM[0].t(), p_MCM[0].y() / p_MCM[0].t(),
- p_MCM[0].z() / p_MCM[0].t(), 0.);
- v_MCM[1].Set(p_MCM[1].x() / p_MCM[1].t(), p_MCM[1].y() / p_MCM[1].t(),
- p_MCM[1].z() / p_MCM[1].t(), 0.);
-
- //propagating quarks until time of youngest quark
- //is just max(pos_MCM[0].t(), pos_MCM[1].t());
- double curtime =
- (pos_MCM[0].t() > pos_MCM[1].t()) ? pos_MCM[0].t() : pos_MCM[1].t();
- FourVector cur_pos[2];
- cur_pos[0].Set(
- pos_MCM[0].x() + v_MCM[0].x() * (curtime - pos_MCM[0].t()),
- pos_MCM[0].y() + v_MCM[0].y() * (curtime - pos_MCM[0].t()),
- pos_MCM[0].z() + v_MCM[0].z() * (curtime - pos_MCM[0].t()),
- curtime);
- cur_pos[0].Set(
- pos_MCM[1].x() + v_MCM[1].x() * (curtime - pos_MCM[1].t()),
- pos_MCM[1].y() + v_MCM[1].y() * (curtime - pos_MCM[1].t()),
- pos_MCM[1].z() + v_MCM[1].z() * (curtime - pos_MCM[1].t()),
- curtime);
-
- //finding position of CM at curtime
- FourVector pos_CM;
- pos_CM.Set((cur_pos[0].x() * considering[0].mass() +
- cur_pos[1].x() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()),
- (cur_pos[0].y() * considering[0].mass() +
- cur_pos[1].y() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()),
- (cur_pos[0].z() * considering[0].mass() +
- cur_pos[1].z() * considering[1].mass()) /
- (considering[0].mass() + considering[1].mass()),
- curtime);
-
- //finding position of meson in lab frame
- betaM.Set(-betaM.x(), -betaM.y(), -betaM.z(), betaM.t());
- FourVector pos_lab = HHboost(betaM, pos_CM);
-
- //finding relative momenta of partons in CM frame
- FourVector k_rel;
- k_rel.Set(
- (p_MCM[1].x() - p_MCM[0].x()) * (p_MCM[1].x() - p_MCM[0].x()) / 4.,
- (p_MCM[1].y() - p_MCM[0].y()) * (p_MCM[1].y() - p_MCM[0].y()) / 4.,
- (p_MCM[1].z() - p_MCM[0].z()) * (p_MCM[1].z() - p_MCM[0].z()) / 4.,
- 0.);
- k_rel.Set(k_rel.x(), k_rel.y(), k_rel.z(),
- k_rel.x() + k_rel.y() + k_rel.z());
-
- //finding relative positions of partons in CM frame
- FourVector pos_rel;
- pos_rel.Set((cur_pos[0].x() - cur_pos[1].x()) *
- (cur_pos[0].x() - cur_pos[1].x()),
- (cur_pos[0].y() - cur_pos[1].y()) *
- (cur_pos[0].y() - cur_pos[1].y()),
- (cur_pos[0].z() - cur_pos[1].z()) *
- (cur_pos[0].z() - cur_pos[1].z()),
- 0.);
- pos_rel.Set(pos_rel.x(), pos_rel.y(), pos_rel.z(),
- pos_rel.x() + pos_rel.y() + pos_rel.z());
-
- //setting appropriate sigma...
- double SigM2 = SigPi2;
- int sortid[2] = {0, 0};
- if (std::abs(considering[0].id()) >= std::abs(considering[1].id())) {
- sortid[0] = std::abs(considering[0].id());
- sortid[1] = std::abs(considering[1].id());
- } else {
- sortid[0] = std::abs(considering[1].id());
- sortid[1] = std::abs(considering[0].id());
- }
+ IdColInfo4.push_back(1); // means color tag
+ IdColInfo4.push_back(considering[2].acol());// {1, color tag} will be at 2nd, 3rd, 4th
- if (sortid[0] == 3) {
- if (sortid[1] == 3) {
- SigM2 = SigPhi2;
- } else {
- SigM2 = SigK2;
- }
- } else if (sortid[0] == 4) {
- if (sortid[1] == 4) {
- SigM2 = SigJpi2;
- } else if (sortid[1] == 3) {
- SigM2 = SigDs2;
- } else {
- SigM2 = SigD2;
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ }
+ if(considering[0].acol() == 0 && considering[1].acol() > 0 && considering[2].acol() == 0){ // Therm/LBT + Therm/LBT + MAT case
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0){showerquarks[loc1 - 1].col(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo2.push_back(1);
+ IdColInfo2.push_back(maxtag);
+
+ IdColInfo3.push_back(1); // means color tag
+ IdColInfo3.push_back(considering[1].acol());// {1, color tag} will be at 2nd, 3rd, 4th
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].col(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(maxtag);
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with two zeros
+ if(considering[0].acol() == 0 && considering[1].acol() == 0 && considering[2].acol() == 0){ // Therm/LBT + Therm/LBT + Therm/LBT
+ IdColInfo1.push_back(1); // junction tag(1)
+
+ maxtag++;
+ int loc1 = findcloserepl(considering[0], element[0] + 1, true, true, showerquarks, HH_thermal);
+ if(loc1 == 999999999){
+ //std::cout < 0){showerquarks[loc1 - 1].col(maxtag); }
+ else if(loc1 < 0){HH_thermal[-loc1 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo2.push_back(1);
+ IdColInfo2.push_back(maxtag);
+
+ maxtag++;
+ int loc2 = findcloserepl(considering[1], perm2[q2], true, true, showerquarks, HH_thermal);
+ if(loc2 == 999999999){
+ //std::cout < 0){showerquarks[loc2 - 1].col(maxtag); }
+ else if(loc2 < 0){HH_thermal[-loc2 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+ IdColInfo3.push_back(1);
+ IdColInfo3.push_back(maxtag);
+
+ maxtag++;
+ int loc3 = findcloserepl(considering[2], perm2[q3], true, true, showerquarks, HH_thermal);
+ if(loc3 == 999999999){
+ //std::cout < 0){showerquarks[loc3 - 1].col(maxtag); }
+ else if(loc3 < 0){HH_thermal[-loc3 - 1].col(maxtag); junction_with_thermal_parton = 1;}
+
+ IdColInfo1.push_back(junction_with_thermal_parton);
+
+ IdColInfo4.push_back(1);
+ IdColInfo4.push_back(maxtag);
+
+ JunctionInfo.push_back(IdColInfo1);
+ JunctionInfo.push_back(IdColInfo2);
+ JunctionInfo.push_back(IdColInfo3);
+ JunctionInfo.push_back(IdColInfo4);
+ Tempjunctions.push_back(JunctionInfo); // information of tempjunction is saved so et clear subordinate vector for next entry
+ IdColInfo1.clear();
+ IdColInfo2.clear();
+ IdColInfo3.clear();
+ IdColInfo4.clear();
+ JunctionInfo.clear();
+ } // so far, we are finished with the list with three zeros
+ //dignostic measure
+ /*
+ std::cout <0){formedhadron.add_par(showerquarks[element[1]].par());}else{formedhadron.add_par(element[1]);}
+ if(perm2[q3]>0){formedhadron.add_par(showerquarks[element[2]].par());}else{formedhadron.add_par(element[2]);}
+
+ //now setting if baryon is in excited state
+ if(WigB[0]*recofactor3*mult < rndbaryon){formedhadron.is_excited(true);}
+
+ //setting if there are any thermal partons used to make the hadron (with the '0' thermal parent as -99999 so that it doesn't conflict with '0' shower parton)
+ if( perm2[q2]>0 && perm2[q3]>0){/*is sh-sh-sh*/ formedhadron.is_shsh(true);}
+ else if(perm2[q2]>0 && perm2[q3]<0){/*is sh-sh-th*/ formedhadron.is_shth(true); if(element[2] == 0){formedhadron.parents[2] = -99999;}}
+ else if(perm2[q2]<0 && perm2[q3]>0){/*is sh-th-sh*/ formedhadron.is_shth(true); if(element[1] == 0){formedhadron.parents[1] = -99999;}}
+ else if(perm2[q2]<0 && perm2[q3]<0){/*is sh-th-th*/ formedhadron.is_shth(true); if(element[1] == 0){formedhadron.parents[1] = -99999;}
+ if(element[2] == 0){formedhadron.parents[2] = -99999;}
+ }
+
+ //setting hadron position and momentum vectors
+ Pbaryon.Set(Pbaryon.x(),Pbaryon.y(),Pbaryon.z(),sqrt(Pbaryon.x()*Pbaryon.x() + Pbaryon.y()*Pbaryon.y() + Pbaryon.z()*Pbaryon.z() + formedhadron.mass()*formedhadron.mass()));
+ formedhadron.pos(pos_lab); formedhadron.P(Pbaryon);
+
+ //setting hadron color tags (for tracing colors of the constituent partons) [**]
+ //will need to update to reflect color tags given to random (thermal/lbt) partons
+ formedhadron.add_col((considering[0].col()>0)?considering[0].col():considering[0].acol());
+ formedhadron.add_col((considering[1].col()>0)?considering[1].col():considering[1].acol());
+ formedhadron.add_col((considering[2].col()>0)?considering[2].col():considering[2].acol());
+
+ //need to choose *what* hadron we've formed... base this on the parton id's, mass, & if excited
+ //might want to do this differently? void f'n(partoncollection, formedhadron)?
+ set_baryon_id(considering, formedhadron);
+
+ //need to add the hadron to the collection
+ HH_hadrons.add(formedhadron);
+
+ //now that we've formed the hadron, need to set ALL (3) the 'considering' flags to used
+ showerquarks[element[0]].status(1); showerquarks[element[0]].is_used(true);
+ if(perm2[q2]>0){showerquarks[element[1]].status(1); showerquarks[element[1]].is_used(true);}
+ else{ HH_thermal[-element[1]].status(1); HH_thermal[-element[1]].is_used(true);}
+ if(perm2[q3]>0){showerquarks[element[2]].status(1); showerquarks[element[2]].is_used(true);}
+ else{ HH_thermal[-element[2]].status(1); HH_thermal[-element[2]].is_used(true);}
+
+ madehadron = true; considering.clear();
+ break;
+ }
+ else{
+ //since we've not formed a baryon on this try, need to revert the third quark 'considering' used flag
+ if(perm2[q3]>0){showerquarks[element[2]].status(0);}
+ else{ HH_thermal[-element[2]].status(0);}
+
+ //and remove the third entry in considering
+ considering.partons.pop_back();
+ }
+ }
+ }else if(considering[0].id()*considering[1].id() < 0){
+ //Key point is determing recofactor2
+ if(considering[0].id() > 0 && considering[1].id() < 0){
+ int tag0 = considering[0].col();
+ int tag1 = considering[1].acol();
+ if(tag0 > 0 && tag1 > 0 && tag0 <= limit && tag1 <= limit ){
+ std::vector::iterator L1 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag0);
+ std::vector::iterator L2 = std::find(IndiceForColFin.begin(), IndiceForColFin.end(), tag1);
+ int indexMatrix1 = std::distance(IndiceForColFin.begin(), L1);
+ int indexMatrix2 = std::distance(IndiceForColFin.begin(), L2);
+
+ recofactor2 = (MesonrecoMatrix1.at(indexMatrix1).at(indexMatrix2));
+ //std::cout < 0 && considering[0].id() < 0){
+ int tag0 = considering[1].col();
+ int tag1 = considering[0].acol();
+ if(tag0 > 0 && tag1 > 0 && tag0 <= limit && tag1 <= limit){
+ std::vector