48{
49
51
52
53 if ( wrappedProcessParticleChange->GetNumberOfSecondaries() == 0 ) return wrappedProcessParticleChange;
54 if ( wrappedProcessParticleChange->GetTrackStatus() ==
fKillTrackAndSecondaries )
return wrappedProcessParticleChange;
55
56 std::vector < G4Track* > secondariesAndPrimary;
57 for ( auto i = 0 ; i < wrappedProcessParticleChange->GetNumberOfSecondaries() ; i++ ) secondariesAndPrimary.push_back( wrappedProcessParticleChange->GetSecondary( i ) );
58
59
60
61
62
63
64
65
66
67 G4ParticleChange* castParticleChange ( nullptr );
68 G4Track* finalStatePrimary ( nullptr );
69 if ( ( wrappedProcessParticleChange->GetTrackStatus() !=
fStopAndKill ) )
70 {
71
72
73
74
75 castParticleChange = dynamic_cast< G4ParticleChange* >( wrappedProcessParticleChange );
76 if ( castParticleChange == nullptr )
77 {
78 G4cout <<
" **** G4BOptnLeadingParticle::ApplyFinalStateBiasing(...) : can not bias for " << callingProcess->
GetProcessName() <<
", this is just a warning." <<
G4endl;
79 return wrappedProcessParticleChange;
80 }
81 finalStatePrimary = new G4Track( *track );
82 finalStatePrimary->SetKineticEnergy ( castParticleChange->GetEnergy() );
83 finalStatePrimary->SetWeight ( castParticleChange->GetWeight() );
84 finalStatePrimary->SetMomentumDirection( *(castParticleChange->GetMomentumDirection()) );
85
86 secondariesAndPrimary.push_back( finalStatePrimary );
87 }
88
89
90
92 if ( finalStatePrimary ) primaryWeight = finalStatePrimary->GetWeight();
94
95 for (auto i = 0; i < wrappedProcessParticleChange->GetNumberOfSecondaries(); ++i)
96 secondariesAndPrimary[ i ]->SetWeight( primaryWeight );
97
98
99 std::size_t leadingIDX = 0;
101 std::map< G4Track*, G4bool > survivingMap;
102 for ( std::size_t idx = 0; idx < secondariesAndPrimary.size(); ++idx )
103 {
104 survivingMap[ secondariesAndPrimary[idx] ] = false;
105 if ( secondariesAndPrimary[idx]->GetKineticEnergy() > leadingEnergy )
106 {
107 leadingEnergy = secondariesAndPrimary[idx]->GetKineticEnergy();
108 leadingIDX = idx;
109 }
110 }
111 survivingMap[ secondariesAndPrimary[leadingIDX] ] = true;
112
113
114 std::map < G4int, std::vector< G4Track* > > typesAndTracks;
115 for ( std::size_t idx = 0; idx < secondariesAndPrimary.size(); ++idx )
116 {
117 if ( idx == leadingIDX ) continue;
118 auto currentTrack = secondariesAndPrimary[idx];
119 auto GROUP = std::abs( currentTrack->GetDefinition()->GetPDGEncoding() );
120 if ( currentTrack->GetDefinition()->GetBaryonNumber() >= 2 )
121 GROUP = -1000;
122
123 if ( typesAndTracks.find( GROUP ) == typesAndTracks.cend() )
124 {
125 std::vector< G4Track* > v;
126 v.push_back( currentTrack );
127 typesAndTracks[ GROUP ] = std::move(v);
128 }
129 else
130 {
131 typesAndTracks[ GROUP ].push_back( currentTrack );
132 }
133 }
134
135
136
137 G4int nSecondaries = 0;
138 for ( auto& typeAndTrack : typesAndTracks )
139 {
140 std::size_t nTracks = (typeAndTrack.second).size();
141 G4Track* keptTrack;
142
143 if ( nTracks > 1 )
144 {
145 auto keptTrackIDX = G4RandFlat::shootInt( nTracks );
146 keptTrack = (typeAndTrack.second)[keptTrackIDX];
148 }
149 else
150 {
151 keptTrack = (typeAndTrack.second)[0];
152 }
153
155 if ( fRussianRouletteKillingProbability > 0.0 )
156 {
158 {
159 keptTrack->
SetWeight( keptTrack->
GetWeight() / (1. - fRussianRouletteKillingProbability) );
160 keepTrack = true;
161 }
162 }
163 else keepTrack = true;
164 if ( keepTrack )
165 {
166 survivingMap[ keptTrack ] = true;
167 if ( keptTrack != finalStatePrimary ) ++nSecondaries;
168 }
169 }
170
171 if ( secondariesAndPrimary[leadingIDX] != finalStatePrimary ) ++nSecondaries;
172
173
174 G4bool primarySurvived =
false;
175 if ( finalStatePrimary )
176 primarySurvived = survivingMap[ finalStatePrimary ];
177
178
179
180
181 fParticleChange.Initialize(*track);
182 if ( primarySurvived )
183 {
184 fParticleChange.ProposeTrackStatus ( wrappedProcessParticleChange->GetTrackStatus() );
185 fParticleChange.ProposeParentWeight ( finalStatePrimary->GetWeight() );
186
187
188 fParticleChange.ProposeEnergy ( finalStatePrimary->GetKineticEnergy() );
189 fParticleChange.ProposeMomentumDirection ( finalStatePrimary->GetMomentumDirection() );
190 }
191 else
192 {
194 fParticleChange.ProposeParentWeight( 0.0 );
195 fParticleChange.ProposeEnergy ( 0.0 );
196 }
197
198 fParticleChange.SetSecondaryWeightByProcess(true);
199 fParticleChange.SetNumberOfSecondaries(nSecondaries);
200
201
202 for ( auto idx = 0 ; idx < wrappedProcessParticleChange->GetNumberOfSecondaries() ; ++idx )
203 {
204 G4Track* secondary = secondariesAndPrimary[idx];
205
206
207
208
209
210
211
212 if ( survivingMap[ secondary ] )
213 fParticleChange.AddSecondary( secondary );
214 else
215 delete secondary;
216 }
217
218
219
220 wrappedProcessParticleChange->Clear();
221
222 if ( finalStatePrimary ) delete finalStatePrimary;
223
224
225 return &fParticleChange;
226}
@ fKillTrackAndSecondaries
G4GLOB_DLL std::ostream G4cout
G4VProcess * GetWrappedProcess() const
G4double GetWeight() const
void SetWeight(G4double aValue)
virtual G4VParticleChange * PostStepDoIt(const G4Track &track, const G4Step &stepData)=0
const G4String & GetProcessName() const