755{
756 G4VPhysicalVolume *repPhysical, *motherPhysical;
757 G4VPhysicalVolume *samplePhysical, *blockedExitedVol = nullptr;
758 G4LogicalVolume *repLogical;
759 G4VSolid *motherSolid;
761 G4double ourStep=currentProposedStepLength;
763 G4double sampleStep, sampleSafety, motherStep, motherSafety;
764 G4long localNoDaughters, sampleNo;
766 G4ExitNormal exitNormalStc;
767
768
769 calculatedExitNormal= false;
770
771
772
773 if ( exiting&&validExitNormal )
774 {
775 if ( localDirection.
dot(exitNormalVector)>=kMinExitingNormalCosine )
776 {
777
778
779 blockedExitedVol = *pBlockedPhysical;
780 ourSafety = 0;
781 }
782 }
783 exiting = false;
784 entering = false;
785
788
789
790
791
792
795 localPoint);
796 G4ExitNormal normalOutStc;
798
799 ourSafety = std::min( ourSafety, sampleSafety);
800
801 if ( sampleSafety<ourStep )
802 {
803
806 localPoint,
807 localDirection,
808 normalOutStc);
809 if ( sampleStep<ourStep )
810 {
811 ourStep = sampleStep;
812 exiting = true;
814
815 exitNormalStc = normalOutStc;
818 calculatedExitNormal = true;
819 }
820 }
821 const G4int secondDepth = topDepth;
822 depth = secondDepth;
823
824
826 {
827 const G4AffineTransform& GlobalToLocal = history.
GetTransform(depth);
829
830
833 repPoint);
834 if ( sampleSafety < ourSafety )
835 {
836 ourSafety = sampleSafety;
837 }
838 if ( sampleSafety < ourStep )
839 {
844 repPoint,
845 newLocalDirection,
846 normalOutStc);
847 if ( sampleStep < ourStep )
848 {
849 ourStep = sampleStep;
850 exiting = true;
851
852
853
857
858 exitNormalStc = normalOutStc;
860 calculatedExitNormal = true;
861 }
862 }
863 depth--;
864 }
865
866
867
869 G4bool exitConvex =
false;
870 G4ExitNormal motherNormalStc;
871
873 motherPhysical = history.
GetVolume(depth);
877
878 motherStep = motherSolid->
DistanceToOut(repPoint,repDirection,
true,
879 &exitConvex,&exitVectorMother);
880 if( exitConvex )
881 {
882 motherNormalStc = G4ExitNormal( exitVectorMother, true, false,
884 calculatedExitNormal = true;
885 }
887
888 G4bool motherDeterminedStep = (motherStep<ourStep);
889
890 if( (!exitConvex) && motherDeterminedStep )
891 {
893 motherNormalStc = G4ExitNormal( exitVectorMother, true, false,
895
896
897
898
899 calculatedExitNormal = true;
900 }
901 if( motherDeterminedStep )
902 {
905
906 exitNormalStc = motherNormalStc;
907 exitNormalStc.
exitNormal = globalExitNormalTop;
908 }
909
910
911
912
913
914
915
916
917 if( ourStep<fMinStep )
918 {
919 ourStep = 2*kCarTolerance;
920 }
921
922 if ( motherSafety<ourSafety )
923 {
924 ourSafety = motherSafety;
925 }
926
927#ifdef G4VERBOSE
928 if ( fCheck )
929 {
931 {
932 std::ostringstream message;
933 message <<
"Point outside volume !" <<
G4endl
934 << " Point " << localPoint
935 <<
" is outside current volume " << motherPhysical->
GetName()
938 message << " Estimated isotropic distance to solid (distToIn)= "
939 << estDistToSolid <<
G4endl;
940 if( estDistToSolid > 100.0 * kCarTolerance )
941 {
945 "Point is far outside Current Volume !" );
946 }
947 else
948 {
951 "Point is a little outside Current Volume.");
952 }
953 }
954 }
955#endif
956
957
958
959#if 1
960 if( motherDeterminedStep )
961 {
962 ourStep = motherStep;
963 exiting = true;
964 }
965
966
967
968 if ( calculatedExitNormal )
969 {
970 if ( motherDeterminedStep )
971 {
972 exitNormalVector = motherNormalStc.
exitNormal;
973 }
974 else
975 {
977 exitNormalVector = globalToLocalTop.
TransformAxis(exitNormalGlobal);
978
979
980 }
981
983 if ( rot != nullptr )
984 {
985 exitNormalVector *= rot->
inverse();
986 }
987
988 }
989 else
990 {
991 validExitNormal = false;
992 }
993
994#else
995 if ( motherSafety<=ourStep )
996 {
997 if ( motherStep<=ourStep )
998 {
999 ourStep = motherStep;
1000 exiting = true;
1001 if ( validExitNormal )
1002 {
1004 if ( rot )
1005 {
1007 }
1008 }
1009 }
1010 else
1011 {
1012 validExitNormal = false;
1013
1014 }
1015 }
1016#endif
1017
1018
1019 G4bool daughterDeterminedStep =
false;
1021
1022
1023
1024
1025
1026
1028 for ( sampleNo=localNoDaughters-1; sampleNo>=0; sampleNo-- )
1029 {
1031 if ( samplePhysical!=blockedExitedVol )
1032 {
1035
1036 G4AffineTransform sampleTf(samplePhysical->
GetRotation(),
1038 sampleTf.Invert();
1040 sampleTf.TransformPoint(localPoint);
1041 const G4VSolid* sampleSolid =
1043 const G4double sampleSafetyDistance =
1045 if ( sampleSafetyDistance<ourSafety )
1046 {
1047 ourSafety = sampleSafetyDistance;
1048 }
1049 if ( sampleSafetyDistance<=ourStep )
1050 {
1051 sampleDirection = sampleTf.TransformAxis(localDirection);
1052 const G4double sampleStepDistance =
1053 sampleSolid->
DistanceToIn(samplePoint,sampleDirection);
1054 if ( sampleStepDistance<=ourStep )
1055 {
1056 daughterDeterminedStep = true;
1057
1058 ourStep = sampleStepDistance;
1059 entering = true;
1060 exiting = false;
1061 *pBlockedPhysical = samplePhysical;
1062 blockedReplicaNo = (
G4int)sampleNo;
1063
1064#ifdef DAUGHTER_NORMAL_ALSO
1065
1067 daughtNormRepCrd = sampleTf.InverseTransformAxis(localExitNorm);
1068#endif
1069
1070#ifdef G4VERBOSE
1071
1072
1073
1074 if ( ( fCheck ) && ( sampleStepDistance < kInfinity ) )
1075 {
1077 intersectionPoint = samplePoint
1078 + sampleStepDistance * sampleDirection;
1079 EInside insideIntPt = sampleSolid->
Inside(intersectionPoint);
1081 {
1083 std::ostringstream message;
1084 message << "Navigator gets conflicting response from Solid."
1086 << " Inaccurate DistanceToIn for solid "
1088 << " Solid gave DistanceToIn = "
1089 << sampleStepDistance << " yet returns " ;
1090 if ( insideIntPt ==
kInside ) {
1091 message << "-kInside-";
1092 }
else if ( insideIntPt ==
kOutside ) {
1093 message << "-kOutside-";
1094 } else {
1095 message << "-kSurface-";
1096 }
1097 message <<
" for this point !" <<
G4endl
1098 <<
" Point = " << intersectionPoint <<
G4endl;
1099 if ( insideIntPt !=
kInside ) {
1100 message << " DistanceToIn(p) = "
1103}
1105 message << " DistanceToOut(p) = "
1107}
1110 G4cout.precision(oldcoutPrec);
1111 }
1112 }
1113#endif
1114 }
1115 }
1116 }
1117 }
1118
1119 calculatedExitNormal &= (!daughterDeterminedStep);
1120
1121#ifdef DAUGHTER_NORMAL_ALSO
1122 if( daughterDeterminedStep )
1123 {
1124
1125
1126
1127
1129 validExitNormal = false;
1130
1131 calculatedExitNormal = true;
1132 }
1133
1134#endif
1135
1136 newSafety = ourSafety;
1137 return ourStep;
1138}
CLHEP::HepRotation G4RotationMatrix
G4GLOB_DLL std::ostream G4cout
double dot(const Hep3Vector &) const
HepRotation inverse() const
const G4AffineTransform & GetTopTransform() const
const G4String & GetName() const
virtual G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const =0