730{
731 G4double motherMinExtent= kInfinity, motherMaxExtent= -kInfinity,
732 targetMinExtent= kInfinity, targetMaxExtent= -kInfinity;
733 G4VPhysicalVolume* pDaughter = nullptr;
734 G4VPVParameterisation* pParam = nullptr;
735 G4VSolid *targetSolid;
736 G4AffineTransform targetTransform;
738 std::size_t nCandidates = pCandidates->size();
739 std::size_t nVol, nNode, targetVolNo;
740 G4VoxelLimits noLimits;
741
742#ifdef G4GEOMETRY_VOXELDEBUG
743 G4cout <<
"**** G4SmartVoxelHeader::BuildNodes" <<
G4endl
744 <<
" Limits = " << pLimits <<
G4endl
745 <<
" Axis = " << pAxis <<
G4endl
746 <<
" Candidates = " << nCandidates <<
G4endl;
747#endif
748
749
750
751
752 G4VSolid* outerSolid = pVolume->
GetSolid();
753 const G4AffineTransform origin;
755 motherMinExtent, motherMaxExtent) )
756 {
758 motherMinExtent, motherMaxExtent);
759 }
762
765 {
766
767
768
771 if (pParam == nullptr)
772 {
773 std::ostringstream message;
774 message <<
"PANIC! - Missing parameterisation." <<
G4endl
775 << " Replicated volume with no parameterisation object !";
776 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
778 return nullptr;
779 }
780
781
782
783 targetTransform = G4AffineTransform(pDaughter->
GetRotation(),
785 replicated = true;
786 }
787 else
788 {
789 replicated = false;
790 }
791
792
793
794 for (nVol=0; nVol<nCandidates; ++nVol)
795 {
796 targetVolNo = (*pCandidates)[nVol];
797 if (!replicated)
798 {
800
801
802
803 targetTransform = G4AffineTransform(pDaughter->
GetRotation(),
805
806
808 }
809 else
810 {
811
812
814
815
816
818
819
820
822 targetTransform = G4AffineTransform(pDaughter->
GetRotation(),
824 }
825
826
828 targetMinExtent, targetMaxExtent))
829 {
831 targetMinExtent,targetMaxExtent);
832 }
833 minExtents[nVol] = targetMinExtent;
834 maxExtents[nVol] = targetMaxExtent;
835
836#ifdef G4GEOMETRY_VOXELDEBUG
837 G4cout <<
"---------------------------------------------------" <<
G4endl
839 <<
" Min Extent = " << targetMinExtent <<
G4endl
840 <<
" Max Extent = " << targetMaxExtent <<
G4endl
841 <<
"---------------------------------------------------" <<
G4endl;
842#endif
843
844
845
846 if ( (!pLimits.
IsLimited()) && ((targetMaxExtent<=motherMinExtent)
847 ||(targetMinExtent>=motherMaxExtent)) )
848 {
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",
857 }
858
859#ifdef G4GEOMETRY_VOXELDEBUG
860
861
862
863
865 {
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) )
871 {
872 G4cout <<
"**** G4SmartVoxelHeader::BuildNodes" <<
G4endl
873 << " WARNING : Daughter # " << nVol
875 << " Crosses mother boundary of logical volume, name = "
877 << " by more than " << kStraddlePercent
879 }
880 }
881#endif
882 }
883
884
885
886
887
890 for (nVol=0; nVol<nCandidates; ++nVol)
891 {
892
893
894
895
896 currentWidth = std::abs(maxExtents[nVol]-minExtents[nVol]);
897 if ( (currentWidth<minWidth)
900 {
901 minWidth = currentWidth;
902 }
903 }
904
905
906
907
908 G4double noNodesExactD = ((motherMaxExtent-motherMinExtent)*2.0/minWidth)+1.0;
909
910
911
912
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; }
922
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;
929#endif
930
932 {
934#ifdef G4GEOMETRY_VOXELDEBUG
936#endif
937 }
938 G4double nodeWidth = (motherMaxExtent-motherMinExtent)/noNodes;
939
940
941
943 if (nodeList == nullptr)
944 {
945 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
947 return nullptr;
948 }
949 nodeList->reserve(noNodes);
950
951 for (nNode=0;
G4long(nNode)<noNodes; ++nNode)
952 {
953 G4SmartVoxelNode *pNode;
954 pNode =
new G4SmartVoxelNode((
G4int)nNode);
955 if (pNode == nullptr)
956 {
957 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
959 return nullptr;
960 }
961 nodeList->push_back(pNode);
962 }
963
964
965
966
967
968 for (nVol=0; nVol<nCandidates; ++nVol)
969 {
970 G4long nodeNo, minContainingNode, maxContainingNode;
971 minContainingNode = (minExtents[nVol]-motherMinExtent)/nodeWidth;
972 maxContainingNode = (maxExtents[nVol]-motherMinExtent)/nodeWidth;
973
974
975
976 if ( (maxContainingNode>=0) && (minContainingNode<noNodes) )
977 {
978
979
980
981 if (maxContainingNode>=noNodes)
982 {
983 maxContainingNode = noNodes-1;
984 }
985
986
987
988 if (minContainingNode<0)
989 {
990 minContainingNode = 0;
991 }
992 for (nodeNo=minContainingNode; nodeNo<=maxContainingNode; ++nodeNo)
993 {
994 (*nodeList)[nodeNo]->Insert((*pCandidates)[nVol]);
995 }
996 }
997 }
998
999
1000
1001
1002
1003
1005 if (proxyList == nullptr)
1006 {
1007 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
1009 return nullptr;
1010 }
1011 proxyList->reserve(noNodes);
1012
1013
1014
1015
1016 for (nNode=0;
G4long(nNode)<noNodes; ++nNode)
1017 {
1018
1019
1020 ((*nodeList)[nNode])->Shrink();
1021 auto* pProxyNode = new G4SmartVoxelProxy((*nodeList)[nNode]);
1022 if (pProxyNode == nullptr)
1023 {
1024 G4Exception(
"G4SmartVoxelHeader::BuildNodes()",
"GeomMgt0003",
1026 return nullptr;
1027 }
1028 proxyList->push_back(pProxyNode);
1029 }
1030 delete nodeList;
1031 return proxyList;
1032}
G4VSolid * GetSolid() const
G4double GetSmartless() const
const G4String & GetName() const
virtual G4VSolid * ComputeSolid(const G4int, G4VPhysicalVolume *)
virtual void ComputeTransformation(const G4int, G4VPhysicalVolume *) const =0
const G4RotationMatrix * GetRotation() const
const G4ThreeVector GetTranslation() const
G4LogicalVolume * GetLogicalVolume() const
const G4String & GetName() const
virtual G4VPVParameterisation * GetParameterisation() const =0
virtual G4bool CalculateExtent(const EAxis pAxis, const G4VoxelLimits &pVoxelLimit, const G4AffineTransform &pTransform, G4double &pMin, G4double &pMax) const =0
virtual void ComputeDimensions(G4VPVParameterisation *p, const G4int n, const G4VPhysicalVolume *pRep)
G4double GetMinExtent(const EAxis pAxis) const
G4double GetMaxExtent(const EAxis pAxis) const
const G4int kMaxVoxelNodes