57{
58 theEnergies.clear();
59
60 if (aStep == nullptr) return false;
61
63
64 if( edep == 0. ) {
65 return 0;
66 }
67
68#ifdef VERBOSE_ENERSPLIT
70 if (verbose)
74#endif
77 {
78 return (
G4int)theEnergies.size();
79 }
81 theEnergies.push_back(edep);
82 return (
G4int)theEnergies.size();
83 }
84
85 if (thePhantomParam == nullptr) GetPhantomParam(true);
86
87
88 std::vector<std::pair<G4int, G4double>> rnsl =
90
93 G4double kinEnergyPre = kinEnergyPreOrig;
94
97 unsigned int ii;
98 for (ii = 0; ii < rnsl.size(); ++ii) {
100 slSum += sl;
101#ifdef VERBOSE_ENERSPLIT
102 if (verbose)
103 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter1 step length geom "
105#endif
106 }
107
108#ifdef VERBOSE_ENERSPLIT
109 if (verbose)
110 G4cout <<
"G4EnergySplitter RN: step length geom TOTAL " << slSum <<
" true TOTAL "
111 << stepLength << " ratio " << stepLength / slSum << " Energy "
115#endif
116
117
118 if (theNIterations == 0) {
119 for (ii = 0; ii < rnsl.size(); ++ii) {
121 G4double edepStep = edep * sl / slSum;
122
123#ifdef VERBOSE_ENERSPLIT
124 if (verbose)
125 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" edep " << edepStep <<
G4endl;
126#endif
127
128 theEnergies.push_back(edepStep);
129 }
130 }
131 else {
132
133#ifdef VERBOSE_ENERSPLIT
134
135 if (verbose) {
137 for (ii = 0; ii < rnsl.size(); ++ii) {
139 slSum += sl;
140 }
141 for (ii = 0; ii < rnsl.size(); ii++) {
142 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes " << ii
143 <<
" RN: iter0 corrected energy lost " << edep * rnsl[ii].second / slSum <<
G4endl;
144 }
145 }
146#endif
147
148 G4double slRatio = stepLength / slSum;
149#ifdef VERBOSE_ENERSPLIT
150 if (verbose)
151 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes RN: iter 0, step ratio " << slRatio
153#endif
154
155
156 G4EmCalculator emcalc;
158 std::vector<G4double> stepLengths;
159 for (
G4int iiter = 1; iiter <= theNIterations; ++iiter) {
160
161
162 if (iiter == 1) {
163 for (ii = 0; ii < rnsl.size(); ++ii) {
165 stepLengths.push_back(sl * slRatio);
166#ifdef VERBOSE_ENERSPLIT
167 if (verbose)
168 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
169 <<
" corrected step length " << sl * slRatio <<
G4endl;
170#endif
171 }
172
173 for (ii = 0; ii < rnsl.size(); ++ii) {
174 const G4Material* mate = thePhantomParam->GetMaterial(rnsl[ii].first);
176 if (kinEnergyPre > 0.) {
177 dEdx = emcalc.
GetDEDX(kinEnergyPre, part, mate);
178 }
179 G4double elost = stepLengths[ii] * dEdx;
180
181#ifdef VERBOSE_ENERSPLIT
182 if (verbose)
183 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter1 energy lost "
184 << elost << " energy at interaction " << kinEnergyPre << " = stepLength "
185 << stepLengths[ii] <<
" * dEdx " << dEdx <<
G4endl;
186#endif
187 kinEnergyPre -= elost;
188 theEnergies.push_back(elost);
189 totalELost += elost;
190 }
191 }
192 else {
193
194
195
196 slSum = 0.;
197 kinEnergyPre = kinEnergyPreOrig;
198 for (ii = 0; ii < rnsl.size(); ++ii) {
199 const G4Material* mate = thePhantomParam->
GetMaterial(rnsl[ii].first);
200 stepLengths[ii] = theElossExt->TrueStepLength(kinEnergyPre, rnsl[ii].second, mate, part);
201 kinEnergyPre -= theEnergies[ii];
202
203#ifdef VERBOSE_ENERSPLIT
204 if (verbose)
205 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
206 << " step length geom " << stepLengths[ii] << " geom2true "
207 << rnsl[ii].second / stepLengths[ii] <<
G4endl;
208#endif
209
210 slSum += stepLengths[ii];
211 }
212
213
215#ifdef VERBOSE_ENERSPLIT
216 if (verbose)
217 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
218 <<
" step ratio " << slRatio <<
G4endl;
219#endif
220 for (ii = 0; ii < rnsl.size(); ++ii) {
221 stepLengths[ii] *= slratio;
222#ifdef VERBOSE_ENERSPLIT
223 if (verbose)
224 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
225 <<
" corrected step length " << stepLengths[ii] <<
G4endl;
226#endif
227 }
228
229
231 totalELost = 0.;
232 for (ii = 0; ii < rnsl.size(); ++ii) {
233 const G4Material* mate = thePhantomParam->
GetMaterial(rnsl[ii].first);
235 if (kinEnergyPre > 0.) {
236 dEdx = emcalc.
GetDEDX(kinEnergyPre, part, mate);
237 }
238 G4double elost = stepLengths[ii] * dEdx;
239#ifdef VERBOSE_ENERSPLIT
240 if (verbose)
241 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
242 << " energy lost " << elost << " energy at interaction " << kinEnergyPre
243 <<
" = stepLength " << stepLengths[ii] <<
" * dEdx " << dEdx <<
G4endl;
244#endif
245 kinEnergyPre -= elost;
246 theEnergies[ii] = elost;
247 totalELost += elost;
248 }
249 }
250
251
252 G4double enerRatio = (edep / totalELost);
253
254#ifdef VERBOSE_ENERSPLIT
255 if (verbose)
256 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes" << ii <<
" RN: iter" << iiter
257 <<
" energy ratio " << enerRatio <<
G4endl;
258#endif
259
260#ifdef VERBOSE_ENERSPLIT
262#endif
263 for (ii = 0; ii < theEnergies.size(); ++ii) {
264 theEnergies[ii] *= enerRatio;
265#ifdef VERBOSE_ENERSPLIT
266 elostTot += theEnergies[ii];
267 if (verbose)
268 G4cout <<
"G4EnergySplitter::SplitEnergyInVolumes " << ii <<
" RN: iter" << iiter
269 << " corrected energy lost " << theEnergies[ii] << " orig elost "
270 << theEnergies[ii] / enerRatio << " energy before interaction "
271 << kinEnergyPreOrig - elostTot + theEnergies[ii] << " energy after interaction "
272 << kinEnergyPreOrig - elostTot <<
G4endl;
273#endif
274 }
275 }
276 }
277
278 return (
G4int)theEnergies.size();
279}
G4GLOB_DLL std::ostream G4cout
G4double GetDEDX(G4double kinEnergy, const G4ParticleDefinition *, const G4Material *, const G4Region *r=nullptr)
const G4String & GetName() const
static G4Material * GetMaterial(const G4String &name, G4bool warning=true)
G4double GetPDGCharge() const
G4Material * GetMaterial() const
G4double GetKineticEnergy() const
G4Track * GetTrack() const
G4StepPoint * GetPreStepPoint() const
G4double GetStepLength() const
G4double GetTotalEnergyDeposit() const
G4ParticleDefinition * GetDefinition() const