188{
189 verboseLevel = verb;
190 if(1 < verboseLevel) {
191 G4cout <<
"G4EmModelManager::Initialise() for "
192 << p->GetParticleName() <<
" Nmodels= " << nEmModels <<
G4endl;
193 }
194
195 if(nEmModels < 1) {
197 ed << "No models found out for " << p->GetParticleName()
198 << " !";
199 G4Exception(
"G4EmModelManager::Initialise",
"em0002",
201 }
202
203 particle = p;
205
207 const G4Region* world =
208 regionStore->
GetRegion(
"DefaultRegionForTheWorld",
false);
209
210
211 nRegions = 1;
212 std::vector<const G4Region*> setr;
213 setr.push_back(world);
215
216 for (
G4int ii=0; ii<nEmModels; ++ii) {
217 const G4Region* r = regions[ii];
218 if ( r == nullptr || r == world) {
219 isWorld = true;
220 regions[ii] = world;
221 } else {
223 if (nRegions>1) {
224 for (
G4int j=1; j<nRegions; ++j) {
225 if ( r == setr[j] ) { newRegion = false; }
226 }
227 }
228 if (newRegion) {
229 setr.push_back(r);
230 ++nRegions;
231 }
232 }
233 }
234
235 if(!isWorld) {
237 ed << "No models defined for the World volume for "
238 << p->GetParticleName() << " !";
239 G4Exception(
"G4EmModelManager::Initialise",
"em0002",
241 }
242
243 G4ProductionCutsTable* theCoupleTable=
245 std::size_t numOfCouples = theCoupleTable->
GetTableSize();
246
247
248
249 if(nRegions > 1 && nEmModels > 1) {
250 idxOfRegionModels.resize(numOfCouples,0);
251 setOfRegionModels.resize((std::size_t)nRegions,nullptr);
252 } else {
253 idxOfRegionModels.resize(1,0);
254 setOfRegionModels.resize(1,nullptr);
255 }
256
257 std::vector<G4int> modelAtRegion(nEmModels);
258 std::vector<G4int> modelOrd(nEmModels);
259 G4DataVector eLow(nEmModels+1);
260 G4DataVector eHigh(nEmModels);
261
262 if(1 < verboseLevel) {
263 G4cout <<
" Nregions= " << nRegions
264 <<
" Nmodels= " << nEmModels <<
G4endl;
265 }
266
267
268 for (
G4int reg=0; reg<nRegions; ++reg) {
269 const G4Region* region = setr[reg];
271
272 for (
G4int ii=0; ii<nEmModels; ++ii) {
273
274 G4VEmModel* model = models[ii];
275 if ( region == regions[ii] ) {
276
279 G4int ord = orderOfModels[ii];
283
284 if(1 < verboseLevel) {
286 <<
" <" << model->
GetName() <<
"> for region <";
289 << " tmin(MeV)= " << tmin/MeV
290 << "; tmax(MeV)= " << tmax/MeV
291 << "; order= " << ord
295 }
296
297 static const G4double limitdelta = 0.01*eV;
298 if(n > 0) {
299
300
301 tmin = std::min(tmin, eHigh[n-1]);
302 tmax = std::max(tmax, eLow[0]);
303
304
305
306 if( tmax - tmin <= limitdelta) { push = false; }
307
308 else if (tmax == eLow[0]) {
309 push = false;
310 insert = true;
311 idx = 0;
312
313 } else if(tmin < eHigh[n-1]) {
314
315 for(
G4int k=0; k<
n; ++k) {
316
317
318
319 if(ord >= modelOrd[k]) {
320 if(tmin < eHigh[k] && tmin >= eLow[k]) { tmin = eHigh[k]; }
321 if(tmax <= eHigh[k] && tmax > eLow[k]) { tmax = eLow[k]; }
322 if(tmax > eHigh[k] && tmin < eLow[k]) {
323 if(tmax - eHigh[k] > eLow[k] - tmin) { tmin = eHigh[k]; }
324 else { tmax = eLow[k]; }
325 }
326 if( tmax - tmin <= limitdelta) {
327 push = false;
328 break;
329 }
330 }
331 }
332
333
334
335
336
337 if (tmax <= eLow[0]) {
338 push = false;
339 insert = true;
340 idx = 0;
341
342 } else if(tmin < eHigh[n-1]) {
343
344 if(tmin > eLow[n-1] && tmax >= eHigh[n-1]) {
346
347 } else if(tmin <= eLow[0] && tmax < eHigh[0]) {
348 eLow[0] = tmax;
349 push = false;
350 insert = true;
351 idx = 0;
352
353 } else {
354 for(
G4int k=n-1; k>=0; --k) {
355 if(tmin <= eLow[k] && tmax >= eHigh[k]) {
356
357 isUsed[modelAtRegion[k]] = 0;
358 idx = k;
359 if(k < n-1) {
360
361 for(
G4int kk=k; kk<
n-1; ++kk) {
362 modelAtRegion[kk] = modelAtRegion[kk+1];
363 modelOrd[kk] = modelOrd[kk+1];
364 eLow[kk] = eLow[kk+1];
365 eHigh[kk] = eHigh[kk+1];
366 }
367 ++k;
368 }
370 } else {
371
372 if(tmin <= eLow[k] && tmax > eLow[k]) {
373 eLow[k] = tmax;
374 idx = k;
375 insert = true;
376 push = false;
377 } else if(tmin < eHigh[k] && tmax >= eHigh[k]) {
378 eHigh[k] = tmin;
379 idx = k + 1;
380 if(idx < n) {
381 insert = true;
382 push = false;
383 }
384 } else if(tmin > eLow[k] && tmax < eHigh[k]) {
385 if(eHigh[k] - tmax > tmin - eLow[k]) {
386 eLow[k] = tmax;
387 idx = k;
388 insert = true;
389 push = false;
390 } else {
391 eHigh[k] = tmin;
392 idx = k + 1;
393 if(idx < n) {
394 insert = true;
395 push = false;
396 }
397 }
398 }
399 }
400 }
401 }
402 }
403 }
404 }
405
406 if(insert) {
407 for(
G4int k=n-1; k>=idx; --k) {
408 modelAtRegion[k+1] = modelAtRegion[k];
409 modelOrd[k+1] = modelOrd[k];
410 eLow[k+1] = eLow[k];
411 eHigh[k+1] = eHigh[k];
412 }
413 }
414
415
416
417 if (push || insert) {
419 modelAtRegion[idx] = ii;
420 modelOrd[idx] = ord;
421 eLow[idx] = tmin;
422 eHigh[idx] = tmax;
423 isUsed[ii] = 1;
424 }
425
426 for(
G4int k=n-1; k>=0; --k) {
427 if(eHigh[k] - eLow[k] <= limitdelta) {
428 isUsed[modelAtRegion[k]] = 0;
429 if(k < n-1) {
430 for(
G4int kk=k; kk<
n-1; ++kk) {
431 modelAtRegion[kk] = modelAtRegion[kk+1];
432 modelOrd[kk] = modelOrd[kk+1];
433 eLow[kk] = eLow[kk+1];
434 eHigh[kk] = eHigh[kk+1];
435 }
436 }
438 }
439 }
440 }
441 }
442 eLow[0] = 0.0;
443 eLow[
n] = eHigh[
n-1];
444
445 if(1 < verboseLevel) {
446 G4cout <<
"### New G4RegionModels set with " <<
n
447 << " models for region <";
449 G4cout <<
"> Elow(MeV)= ";
450 for(
G4int iii=0; iii<=
n; ++iii) {
G4cout << eLow[iii]/MeV <<
" ";}
452 }
453 auto rm = new G4RegionModels(n, modelAtRegion, eLow, region);
454 setOfRegionModels[reg] = rm;
455
456 if(1 == nEmModels) { break; }
457 }
458
459 currRegionModel = setOfRegionModels[0];
460 currModel = models[0];
461
462
463 std::size_t idx = 1;
464 if(nullptr != secondaryParticle) {
468 else { idx = 3; }
469 }
470
471 theCuts =
473
474
475 if(nullptr != theCutsNew) { *theCutsNew = *theCuts; }
476
477
478
479 for(std::size_t i=0; i<numOfCouples; ++i) {
480
481 const G4MaterialCutsCouple* couple =
483 const G4Material* material = couple->
GetMaterial();
485
487 if(nRegions > 1 && nEmModels > 1) {
488 reg = nRegions;
489
490 do {--reg;} while (reg>0 && pcuts != (setr[reg]->GetProductionCuts()));
491 idxOfRegionModels[i] = reg;
492 }
493 if(1 < verboseLevel) {
494 G4cout <<
"G4EmModelManager::Initialise() for "
496 << " indexOfCouple= " << i
497 << " indexOfRegion= " << reg
499 }
500
502 if(nullptr != secondaryParticle) {
503
504
507 if(nRegions > 1 && nEmModels > 1) {
508 inn = idxOfRegionModels[i];
509 }
510
511
512 currRegionModel = setOfRegionModels[inn];
513 nnm = currRegionModel->NumberOfModels();
514
515
516
517 for(
G4int jj=0; jj<nnm; ++jj) {
518
519
520 currModel = models[currRegionModel->ModelIndex(jj)];
521 G4double cutlim = currModel->MinEnergyCut(particle,couple);
522 if(cutlim > cut) {
523 if(nullptr == theCutsNew) { theCutsNew = new G4DataVector(*theCuts); }
524 (*theCutsNew)[i] = cutlim;
525
526
527
528
529
530
531
532 }
533 }
534 }
535 }
536 if(nullptr != theCutsNew) { theCuts = theCutsNew; }
537
538
540 severalModels = true;
541 for(
G4int jj=0; jj<nEmModels; ++jj) {
542 if(1 == isUsed[jj]) {
544 currModel = models[jj];
545 currModel->Initialise(particle, *theCuts);
546 if(nullptr != flucModels[jj]) { flucModels[jj]->InitialiseMe(particle); }
547 }
548 }
549 if(1 == nn) { severalModels = false; }
550
551 if(1 < verboseLevel) {
552 G4cout <<
"G4EmModelManager for " << particle->GetParticleName()
553 << " is initialised; nRegions= " << nRegions
554 << " severalModels: " << severalModels
556 }
557 return theCuts;
558}
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
static G4Electron * Electron()
G4ProductionCuts * GetProductionCuts() const
static G4Positron * Positron()
const G4MaterialCutsCouple * GetMaterialCutsCouple(G4int i) const
std::size_t GetTableSize() const
const std::vector< G4double > * GetEnergyCutsVector(std::size_t pcIdx) const
static G4ProductionCutsTable * GetProductionCutsTable()
static G4RegionStore * GetInstance()
G4Region * GetRegion(const G4String &name, G4bool verbose=true) const
G4double LowEnergyLimit() const
G4double HighEnergyLimit() const