444 G4double goodSliceScore=kInfinity, testSliceScore;
446 std::size_t node, maxNode;
455 pTestSlices =
BuildNodes(pVolume,pLimits,pCandidates,testAxis);
457 if ( (pGoodSlices ==
nullptr) || (testSliceScore<goodSliceScore) )
459 goodSliceAxis = testAxis;
460 goodSliceScore = testSliceScore;
461 std::swap( pGoodSlices, pTestSlices);
463 if (pTestSlices !=
nullptr)
467 maxNode=pTestSlices->size();
468 for (node=0; node<maxNode; ++node)
470 delete (*pTestSlices)[node]->GetNode();
473 while (!pTestSlices->empty())
475 tmpProx = pTestSlices->back();
476 pTestSlices->pop_back();
477 for (
auto i=pTestSlices->cbegin(); i!=pTestSlices->cend(); )
481 i = pTestSlices->erase(i);
497 if (pGoodSlices ==
nullptr)
499 G4Exception(
"G4SmartVoxelHeader::BuildVoxelsWithinLimits()",
501 "Cannot select more than 3 axis for optimisation.");
514 faxis = goodSliceAxis;
516#ifdef G4GEOMETRY_VOXELDEBUG
519 for (
auto islice=0; islice<
fslices.size(); ++islice)
521 G4cout <<
" Node #" << islice <<
" = {";
522 for (
auto j=0; j<
fslices[islice]->GetNode()->GetNoContained(); ++j)
731 G4double motherMinExtent= kInfinity, motherMaxExtent= -kInfinity,
732 targetMinExtent= kInfinity, targetMaxExtent= -kInfinity;
738 std::size_t nCandidates = pCandidates->size();
739 std::size_t nVol, nNode, targetVolNo;
742#ifdef G4GEOMETRY_VOXELDEBUG
743 G4cout <<
"**** G4SmartVoxelHeader::BuildNodes" <<
G4endl
744 <<
" Limits = " << pLimits <<
G4endl
745 <<
" Axis = " << pAxis <<
G4endl
746 <<
" Candidates = " << nCandidates <<
G4endl;
755 motherMinExtent, motherMaxExtent) )
758 motherMinExtent, motherMaxExtent);
771 if (pParam ==
nullptr)
773 std::ostringstream message;
774 message <<
"PANIC! - Missing parameterisation." <<
G4endl
775 <<
" Replicated volume with no parameterisation object !";
776 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
794 for (nVol=0; nVol<nCandidates; ++nVol)
796 targetVolNo = (*pCandidates)[nVol];
828 targetMinExtent, targetMaxExtent))
831 targetMinExtent,targetMaxExtent);
833 minExtents[nVol] = targetMinExtent;
834 maxExtents[nVol] = targetMaxExtent;
836#ifdef G4GEOMETRY_VOXELDEBUG
837 G4cout <<
"---------------------------------------------------" <<
G4endl
839 <<
" Min Extent = " << targetMinExtent <<
G4endl
840 <<
" Max Extent = " << targetMaxExtent <<
G4endl
841 <<
"---------------------------------------------------" <<
G4endl;
846 if ( (!pLimits.
IsLimited()) && ((targetMaxExtent<=motherMinExtent)
847 ||(targetMinExtent>=motherMaxExtent)) )
849 std::ostringstream message;
850 message <<
"PANIC! - Overlapping daughter with mother volume." <<
G4endl
851 <<
" Daughter physical volume "
853 <<
" is entirely outside mother logical volume "
854 << pVolume->
GetName() <<
" !!";
855 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0002",
859#ifdef G4GEOMETRY_VOXELDEBUG
867 G4int kStraddlePercent = 5;
868 width = maxExtents[nVol]-minExtents[nVol];
869 if ( (((motherMinExtent-minExtents[nVol])*100/width) > kStraddlePercent)
870 ||(((maxExtents[nVol]-motherMaxExtent)*100/width) > kStraddlePercent) )
872 G4cout <<
"**** G4SmartVoxelHeader::BuildNodes" <<
G4endl
873 <<
" WARNING : Daughter # " << nVol
875 <<
" Crosses mother boundary of logical volume, name = "
877 <<
" by more than " << kStraddlePercent
890 for (nVol=0; nVol<nCandidates; ++nVol)
896 currentWidth = std::abs(maxExtents[nVol]-minExtents[nVol]);
897 if ( (currentWidth<minWidth)
901 minWidth = currentWidth;
908 G4double noNodesExactD = ((motherMaxExtent-motherMinExtent)*2.0/minWidth)+1.0;
913 G4double smartlessComputed = noNodesExactD / nCandidates;
915 G4double smartless = (smartlessComputed <= smartlessUser)
916 ? smartlessComputed : smartlessUser;
917 G4double noNodesSmart = smartless*nCandidates;
918 auto noNodesExactI =
G4int(noNodesSmart);
919 G4long noNodes = ((noNodesSmart-noNodesExactI)>=0.5)
920 ? noNodesExactI+1 : noNodesExactI;
921 if( noNodes == 0 ) { noNodes=1; }
923#ifdef G4GEOMETRY_VOXELDEBUG
924 G4cout <<
" Smartless computed = " << smartlessComputed <<
G4endl
925 <<
" Smartless volume = " << smartlessUser
926 <<
" => # Smartless = " << smartless <<
G4endl;
927 G4cout <<
" Min width = " << minWidth
928 <<
" => # Nodes = " << noNodes <<
G4endl;
934#ifdef G4GEOMETRY_VOXELDEBUG
938 G4double nodeWidth = (motherMaxExtent-motherMinExtent)/noNodes;
943 if (nodeList ==
nullptr)
945 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
949 nodeList->reserve(noNodes);
951 for (nNode=0;
G4long(nNode)<noNodes; ++nNode)
955 if (pNode ==
nullptr)
957 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
961 nodeList->push_back(pNode);
968 for (nVol=0; nVol<nCandidates; ++nVol)
970 G4long nodeNo, minContainingNode, maxContainingNode;
971 minContainingNode = (minExtents[nVol]-motherMinExtent)/nodeWidth;
972 maxContainingNode = (maxExtents[nVol]-motherMinExtent)/nodeWidth;
976 if ( (maxContainingNode>=0) && (minContainingNode<noNodes) )
981 if (maxContainingNode>=noNodes)
983 maxContainingNode = noNodes-1;
988 if (minContainingNode<0)
990 minContainingNode = 0;
992 for (nodeNo=minContainingNode; nodeNo<=maxContainingNode; ++nodeNo)
994 (*nodeList)[nodeNo]->Insert((*pCandidates)[nVol]);
1005 if (proxyList ==
nullptr)
1007 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
1011 proxyList->reserve(noNodes);
1016 for (nNode=0;
G4long(nNode)<noNodes; ++nNode)
1020 ((*nodeList)[nNode])->Shrink();
1022 if (pProxyNode ==
nullptr)
1024 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
1028 proxyList->push_back(pProxyNode);
1052 std::size_t nNodes = pSlice->size();
1053 std::size_t noContained, maxContained=0, sumContained=0, sumNonEmptyNodes=0;
1056 for (std::size_t i=0; i<nNodes; ++i)
1058 if ((*pSlice)[i]->IsNode())
1062 node = (*pSlice)[i]->GetNode();
1064 if (noContained != 0)
1067 sumContained += noContained;
1071 if (noContained>maxContained)
1073 maxContained = noContained;
1079 G4Exception(
"G4SmartVoxelHeader::CalculateQuality()",
"GeomMgt0001",
1086 if (sumNonEmptyNodes != 0)
1088 quality = sumContained/sumNonEmptyNodes;
1092 quality = kInfinity;
1095#ifdef G4GEOMETRY_VOXELDEBUG
1096 G4cout <<
"**** G4SmartVoxelHeader::CalculateQuality" <<
G4endl
1097 <<
" Quality = " << quality <<
G4endl
1098 <<
" Nodes = " << nNodes
1099 <<
" of which " << sumNonEmptyNodes <<
" non empty" <<
G4endl
1100 <<
" Max Contained = " << maxContained <<
G4endl;
1118 std::size_t refinedDepth=0, minVolumes;
1119 std::size_t maxNode =
fslices.size();
1136 switch (refinedDepth)
1151 std::size_t targetNo, noContainedDaughters, minNo, maxNo, replaceNo, i;
1161 for (targetNo=0; targetNo<maxNode; ++targetNo)
1165 targetNodeProxy =
fslices[targetNo];
1166 targetNode = targetNodeProxy->
GetNode();
1172 if (targetList ==
nullptr)
1176 "Target volume node list allocation error.");
1179 targetList->reserve(noContainedDaughters);
1180 for (i=0; i<noContainedDaughters; ++i)
1187#ifdef G4GEOMETRY_VOXELDEBUG
1188 G4cout <<
"**** G4SmartVoxelHeader::RefineNodes" <<
G4endl
1189 <<
" Refining nodes " << minNo
1190 <<
" - " << maxNo <<
" inclusive" <<
G4endl;
1202 for (replaceNo=minNo; replaceNo<=maxNo; ++replaceNo)
1204 if (lastProxy !=
fslices[replaceNo])
1216 newLimits = pLimits;
1220 targetList,(
G4int)replaceNo);
1221 if (replaceHeader ==
nullptr)
1223 G4Exception(
"G4SmartVoxelHeader::RefineNodes()",
"GeomMgt0003",
1230 if (replaceHeaderProxy ==
nullptr)
1232 G4Exception(
"G4SmartVoxelHeader::RefineNodes()",
"GeomMgt0003",
1236 for (replaceNo=minNo; replaceNo<=maxNo; ++replaceNo)
1238 fslices[replaceNo] = replaceHeaderProxy;