Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4ReplicaNavigation Class Reference

#include <G4ReplicaNavigation.hh>

Public Member Functions

 G4ReplicaNavigation ()
 
 ~G4ReplicaNavigation ()=default
 
G4bool LevelLocate (G4NavigationHistory &history, const G4VPhysicalVolume *blockedVol, const G4int blockedNum, const G4ThreeVector &globalPoint, const G4ThreeVector *globalDirection, const G4bool pLocatedOnEdge, G4ThreeVector &localPoint)
 
G4double ComputeStep (const G4ThreeVector &globalPoint, const G4ThreeVector &globalDirection, const G4ThreeVector &localPoint, const G4ThreeVector &localDirection, const G4double currentProposedStepLength, G4double &newSafety, G4NavigationHistory &history, G4bool &validExitNormal, G4bool &calculatedExitNormal, G4ThreeVector &exitNormal, G4bool &exiting, G4bool &entering, G4VPhysicalVolume *(*pBlockedPhysical), G4int &blockedReplicaNo)
 
G4double ComputeSafety (const G4ThreeVector &globalPoint, const G4ThreeVector &localPoint, const G4NavigationHistory &history, const G4double pProposedMaxLength=DBL_MAX) const
 
EInside BackLocate (G4NavigationHistory &history, const G4ThreeVector &globalPoint, G4ThreeVector &localPoint, const G4bool &exiting, G4bool &notKnownInside) const
 
void ComputeTransformation (const G4int replicaNo, G4VPhysicalVolume *pVol, G4ThreeVector &point) const
 
void ComputeTransformation (const G4int replicaNo, G4VPhysicalVolume *pVol) const
 
EInside Inside (const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
 
G4double DistanceToOut (const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
 
G4double DistanceToOut (const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint, const G4ThreeVector &localDirection, G4ExitNormal &candidateNormal) const
 
G4int GetVerboseLevel () const
 
void SetVerboseLevel (G4int level)
 
void CheckMode (G4bool mode)
 

Detailed Description

Definition at line 78 of file G4ReplicaNavigation.hh.

Constructor & Destructor Documentation

◆ G4ReplicaNavigation()

G4ReplicaNavigation::G4ReplicaNavigation ( )

Definition at line 52 of file G4ReplicaNavigation.cc.

53{
57 halfkCarTolerance = kCarTolerance*0.5;
58 halfkRadTolerance = kRadTolerance*0.5;
59 halfkAngTolerance = kAngTolerance*0.5;
60 fMinStep = 0.05*kCarTolerance;
61}
G4double GetSurfaceTolerance() const
G4double GetRadialTolerance() const
static G4GeometryTolerance * GetInstance()
G4double GetAngularTolerance() const

◆ ~G4ReplicaNavigation()

G4ReplicaNavigation::~G4ReplicaNavigation ( )
default

Member Function Documentation

◆ BackLocate()

EInside G4ReplicaNavigation::BackLocate ( G4NavigationHistory & history,
const G4ThreeVector & globalPoint,
G4ThreeVector & localPoint,
const G4bool & exiting,
G4bool & notKnownInside ) const

Definition at line 1237 of file G4ReplicaNavigation.cc.

1242{
1243 G4VPhysicalVolume *pNRMother = nullptr;
1244 G4VSolid *motherSolid;
1245 G4ThreeVector repPoint, goodPoint;
1246 G4int mdepth, depth, cdepth;
1247 EInside insideCode;
1248
1249 cdepth = (G4int)history.GetDepth();
1250
1251 // Find non replicated mother
1252 //
1253 for ( mdepth=cdepth-1; mdepth>=0; mdepth-- )
1254 {
1255 if ( history.GetVolumeType(mdepth)!=kReplica )
1256 {
1257 pNRMother = history.GetVolume(mdepth);
1258 break;
1259 }
1260 }
1261
1262 if( pNRMother == nullptr )
1263 {
1264 // All the tree of mother volumes were Replicas.
1265 // This is an error, as the World volume must be a Placement
1266 //
1267 G4Exception("G4ReplicaNavigation::BackLocate()", "GeomNav0002",
1268 FatalException, "The World volume must be a Placement!");
1269 return kInside;
1270 }
1271
1272 motherSolid = pNRMother->GetLogicalVolume()->GetSolid();
1273 goodPoint = history.GetTransform(mdepth).TransformPoint(globalPoint);
1274 insideCode = motherSolid->Inside(goodPoint);
1275 if ( (insideCode==kOutside)||((insideCode==kSurface)&&exiting) )
1276 {
1277 // Outside mother -> back up to mother level
1278 // Locate.. in Navigator will back up one more level
1279 // localPoint not required
1280 //
1281 history.BackLevel(cdepth-mdepth);
1282 // localPoint = goodPoint;
1283 }
1284 else
1285 {
1286 notKnownInside = false;
1287
1288 // Still within replications
1289 // Check down: if on outside stop at this level
1290 //
1291 for ( depth=mdepth+1; depth<cdepth; ++depth)
1292 {
1293 repPoint = history.GetTransform(depth).TransformPoint(globalPoint);
1294 insideCode = Inside(history.GetVolume(depth),
1295 history.GetReplicaNo(depth),
1296 repPoint);
1297 if ( (insideCode==kOutside)||((insideCode==kSurface)&&exiting) )
1298 {
1299 localPoint = goodPoint;
1300 history.BackLevel(cdepth-depth);
1301 return insideCode;
1302 }
1303 goodPoint = repPoint;
1304 }
1305 localPoint = history.GetTransform(depth).TransformPoint(globalPoint);
1306 insideCode = Inside(history.GetVolume(depth),
1307 history.GetReplicaNo(depth),
1308 localPoint);
1309 // If outside level, set localPoint = coordinates in reference system
1310 // of *previous* level - location code in navigator will back up one
1311 // level [And also manage blocking]
1312 //
1313 if ( (insideCode==kOutside)||((insideCode==kSurface)&&exiting) )
1314 {
1315 localPoint = goodPoint;
1316 }
1317 }
1318 return insideCode;
1319}
@ FatalException
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
CLHEP::Hep3Vector G4ThreeVector
int G4int
Definition G4Types.hh:85
G4ThreeVector TransformPoint(const G4ThreeVector &vec) const
G4VSolid * GetSolid() const
G4int GetReplicaNo(G4int n) const
G4VPhysicalVolume * GetVolume(G4int n) const
std::size_t GetDepth() const
EVolume GetVolumeType(G4int n) const
const G4AffineTransform & GetTransform(G4int n) const
EInside Inside(const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
G4LogicalVolume * GetLogicalVolume() const
virtual EInside Inside(const G4ThreeVector &p) const =0
EInside
Definition geomdefs.hh:67
@ kInside
Definition geomdefs.hh:70
@ kOutside
Definition geomdefs.hh:68
@ kSurface
Definition geomdefs.hh:69
@ kReplica
Definition geomdefs.hh:85

◆ CheckMode()

void G4ReplicaNavigation::CheckMode ( G4bool mode)
inline

◆ ComputeSafety()

G4double G4ReplicaNavigation::ComputeSafety ( const G4ThreeVector & globalPoint,
const G4ThreeVector & localPoint,
const G4NavigationHistory & history,
const G4double pProposedMaxLength = DBL_MAX ) const

Definition at line 1148 of file G4ReplicaNavigation.cc.

1152{
1153 G4VPhysicalVolume *repPhysical, *motherPhysical;
1154 G4VPhysicalVolume *samplePhysical, *blockedExitedVol = nullptr;
1155 G4LogicalVolume *repLogical;
1156 G4VSolid *motherSolid;
1157 G4ThreeVector repPoint;
1158 G4double ourSafety = kInfinity;
1159 G4double sampleSafety;
1160 G4long localNoDaughters, sampleNo;
1161 G4int depth;
1162
1163 repPhysical = history.GetTopVolume();
1164 repLogical = repPhysical->GetLogicalVolume();
1165
1166 //
1167 // Compute intersection with replica boundaries & replica safety
1168 //
1169
1170 sampleSafety = DistanceToOut(history.GetTopVolume(),
1171 history.GetTopReplicaNo(),
1172 localPoint);
1173 if ( sampleSafety<ourSafety )
1174 {
1175 ourSafety = sampleSafety;
1176 }
1177
1178 depth = (G4int)history.GetDepth()-1;
1179
1180 // Loop checking, 07.10.2016, JA -- need to add: assert(depth>0)
1181 while ( history.GetVolumeType(depth)==kReplica )
1182 {
1183 repPoint = history.GetTransform(depth).TransformPoint(globalPoint);
1184 sampleSafety = DistanceToOut(history.GetVolume(depth),
1185 history.GetReplicaNo(depth),
1186 repPoint);
1187 if ( sampleSafety<ourSafety )
1188 {
1189 ourSafety = sampleSafety;
1190 }
1191 depth--;
1192 }
1193
1194 // Compute mother safety & intersection
1195 //
1196 repPoint = history.GetTransform(depth).TransformPoint(globalPoint);
1197 motherPhysical = history.GetVolume(depth);
1198 motherSolid = motherPhysical->GetLogicalVolume()->GetSolid();
1199 sampleSafety = motherSolid->DistanceToOut(repPoint);
1200
1201 if ( sampleSafety<ourSafety )
1202 {
1203 ourSafety = sampleSafety;
1204 }
1205
1206 // Compute daughter safeties & intersections
1207 //
1208 localNoDaughters = repLogical->GetNoDaughters();
1209 for ( sampleNo=localNoDaughters-1; sampleNo>=0; sampleNo-- )
1210 {
1211 samplePhysical = repLogical->GetDaughter((G4int)sampleNo);
1212 if ( samplePhysical!=blockedExitedVol )
1213 {
1214 G4AffineTransform sampleTf(samplePhysical->GetRotation(),
1215 samplePhysical->GetTranslation());
1216 sampleTf.Invert();
1217 const G4ThreeVector samplePoint =
1218 sampleTf.TransformPoint(localPoint);
1219 const G4VSolid *sampleSolid =
1220 samplePhysical->GetLogicalVolume()->GetSolid();
1221 const G4double sampleSafetyDistance =
1222 sampleSolid->DistanceToIn(samplePoint);
1223 if ( sampleSafetyDistance<ourSafety )
1224 {
1225 ourSafety = sampleSafetyDistance;
1226 }
1227 }
1228 }
1229 return ourSafety;
1230}
double G4double
Definition G4Types.hh:83
long G4long
Definition G4Types.hh:87
std::size_t GetNoDaughters() const
G4VPhysicalVolume * GetDaughter(const std::size_t i) const
G4int GetTopReplicaNo() const
G4VPhysicalVolume * GetTopVolume() const
G4double DistanceToOut(const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
const G4RotationMatrix * GetRotation() const
const G4ThreeVector GetTranslation() const
virtual G4double DistanceToOut(const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *n=nullptr) const =0
virtual G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const =0

◆ ComputeStep()

G4double G4ReplicaNavigation::ComputeStep ( const G4ThreeVector & globalPoint,
const G4ThreeVector & globalDirection,
const G4ThreeVector & localPoint,
const G4ThreeVector & localDirection,
const G4double currentProposedStepLength,
G4double & newSafety,
G4NavigationHistory & history,
G4bool & validExitNormal,
G4bool & calculatedExitNormal,
G4ThreeVector & exitNormal,
G4bool & exiting,
G4bool & entering,
G4VPhysicalVolume ** pBlockedPhysical,
G4int & blockedReplicaNo )

Definition at line 740 of file G4ReplicaNavigation.cc.

755{
756 G4VPhysicalVolume *repPhysical, *motherPhysical;
757 G4VPhysicalVolume *samplePhysical, *blockedExitedVol = nullptr;
758 G4LogicalVolume *repLogical;
759 G4VSolid *motherSolid;
760 G4ThreeVector repPoint, repDirection, sampleDirection;
761 G4double ourStep=currentProposedStepLength;
762 G4double ourSafety=kInfinity;
763 G4double sampleStep, sampleSafety, motherStep, motherSafety;
764 G4long localNoDaughters, sampleNo;
765 G4int depth;
766 G4ExitNormal exitNormalStc;
767 // G4int depthDeterminingStep= -1; // Useful only for debugging - for now
768
769 calculatedExitNormal= false;
770
771 // Exiting normal optimisation
772 //
773 if ( exiting&&validExitNormal )
774 {
775 if ( localDirection.dot(exitNormalVector)>=kMinExitingNormalCosine )
776 {
777 // Block exited daughter volume
778 //
779 blockedExitedVol = *pBlockedPhysical;
780 ourSafety = 0;
781 }
782 }
783 exiting = false;
784 entering = false;
785
786 repPhysical = history.GetTopVolume();
787 repLogical = repPhysical->GetLogicalVolume();
788
789 //
790 // Compute intersection with replica boundaries & replica safety
791 //
792
793 sampleSafety = DistanceToOut(repPhysical,
794 history.GetTopReplicaNo(),
795 localPoint);
796 G4ExitNormal normalOutStc;
797 const auto topDepth= (G4int)history.GetDepth();
798
799 ourSafety = std::min( ourSafety, sampleSafety);
800
801 if ( sampleSafety<ourStep )
802 {
803
804 sampleStep = DistanceToOut(repPhysical,
805 history.GetTopReplicaNo(),
806 localPoint,
807 localDirection,
808 normalOutStc);
809 if ( sampleStep<ourStep )
810 {
811 ourStep = sampleStep;
812 exiting = true;
813 validExitNormal = normalOutStc.validConvex; // false; -> Old,Conservative
814
815 exitNormalStc = normalOutStc;
816 exitNormalStc.exitNormal =
817 history.GetTopTransform().InverseTransformAxis(normalOutStc.exitNormal);
818 calculatedExitNormal = true;
819 }
820 }
821 const G4int secondDepth = topDepth;
822 depth = secondDepth;
823
824 // Loop checking, 07.10.2016, JA -- Need to add: assert(depth>0)
825 while ( history.GetVolumeType(depth)==kReplica )
826 {
827 const G4AffineTransform& GlobalToLocal = history.GetTransform(depth);
828 repPoint = GlobalToLocal.TransformPoint(globalPoint);
829 // repPoint = history.GetTransform(depth).TransformPoint(globalPoint);
830
831 sampleSafety = DistanceToOut(history.GetVolume(depth),
832 history.GetReplicaNo(depth),
833 repPoint);
834 if ( sampleSafety < ourSafety )
835 {
836 ourSafety = sampleSafety;
837 }
838 if ( sampleSafety < ourStep )
839 {
840 G4ThreeVector newLocalDirection =
841 GlobalToLocal.TransformAxis(globalDirection);
842 sampleStep = DistanceToOut(history.GetVolume(depth),
843 history.GetReplicaNo(depth),
844 repPoint,
845 newLocalDirection,
846 normalOutStc);
847 if ( sampleStep < ourStep )
848 {
849 ourStep = sampleStep;
850 exiting = true;
851
852 // As step is limited by this level, must set Exit Normal
853 //
854 G4ThreeVector localExitNorm = normalOutStc.exitNormal;
855 G4ThreeVector globalExitNorm =
856 GlobalToLocal.InverseTransformAxis(localExitNorm);
857
858 exitNormalStc = normalOutStc; // Normal, convex, calculated, side
859 exitNormalStc.exitNormal = globalExitNorm;
860 calculatedExitNormal = true;
861 }
862 }
863 depth--;
864 }
865
866 // Compute mother safety & intersection
867 //
868 G4ThreeVector exitVectorMother;
869 G4bool exitConvex = false; // Value obtained in DistanceToOut(p,v) call
870 G4ExitNormal motherNormalStc;
871
872 repPoint = history.GetTransform(depth).TransformPoint(globalPoint);
873 motherPhysical = history.GetVolume(depth);
874 motherSolid = motherPhysical->GetLogicalVolume()->GetSolid();
875 motherSafety = motherSolid->DistanceToOut(repPoint);
876 repDirection = history.GetTransform(depth).TransformAxis(globalDirection);
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 }
886 const G4AffineTransform& globalToLocalTop = history.GetTopTransform();
887
888 G4bool motherDeterminedStep = (motherStep<ourStep);
889
890 if( (!exitConvex) && motherDeterminedStep )
891 {
892 exitVectorMother = motherSolid->SurfaceNormal( repPoint );
893 motherNormalStc = G4ExitNormal( exitVectorMother, true, false,
895 // CalculatedExitNormal -> true;
896 // Convex -> false: do not know value
897 // ExitSide -> kMother (or kNull)
898
899 calculatedExitNormal = true;
900 }
901 if( motherDeterminedStep )
902 {
903 G4ThreeVector globalExitNormalTop =
904 globalToLocalTop.InverseTransformAxis(exitVectorMother);
905
906 exitNormalStc = motherNormalStc;
907 exitNormalStc.exitNormal = globalExitNormalTop;
908 }
909
910 // Push in principle no longer necessary. G4Navigator now takes care of ...
911 // Removing this however may cause additional almost-zero steps and generate
912 // warnings for pushed particles from G4Navigator, particularly for the case
913 // of 3D replicas (Cartesian or combined Radial/Phi cases).
914 // Requires further investigation and eventually reimplementation of
915 // LevelLocate() to take into account point and direction ...
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 {
930 if( motherSolid->Inside(localPoint)==kOutside )
931 {
932 std::ostringstream message;
933 message << "Point outside volume !" << G4endl
934 << " Point " << localPoint
935 << " is outside current volume " << motherPhysical->GetName()
936 << G4endl;
937 G4double estDistToSolid= motherSolid->DistanceToIn(localPoint);
938 message << " Estimated isotropic distance to solid (distToIn)= "
939 << estDistToSolid << G4endl;
940 if( estDistToSolid > 100.0 * kCarTolerance )
941 {
942 motherSolid->DumpInfo();
943 G4Exception("G4ReplicaNavigation::ComputeStep()",
944 "GeomNav0003", FatalException, message,
945 "Point is far outside Current Volume !" );
946 }
947 else
948 {
949 G4Exception("G4ReplicaNavigation::ComputeStep()",
950 "GeomNav1002", JustWarning, message,
951 "Point is a little outside Current Volume.");
952 }
953 }
954 }
955#endif
956
957 // Comparison of steps may need precision protection
958 //
959#if 1
960 if( motherDeterminedStep )
961 {
962 ourStep = motherStep;
963 exiting = true;
964 }
965
966 // Transform it to the Grand-Mother Reference Frame (current convention)
967 //
968 if ( calculatedExitNormal )
969 {
970 if ( motherDeterminedStep )
971 {
972 exitNormalVector = motherNormalStc.exitNormal;
973 }
974 else
975 {
976 G4ThreeVector exitNormalGlobal = exitNormalStc.exitNormal;
977 exitNormalVector = globalToLocalTop.TransformAxis(exitNormalGlobal);
978 // exitNormalVector= globalToLocal2nd.TransformAxis(exitNormalGlobal);
979 // Alt Make it in one go to Grand-Mother, avoiding transform below
980 }
981 // Transform to Grand-mother reference frame
982 const G4RotationMatrix* rot = motherPhysical->GetRotation();
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 {
1003 const G4RotationMatrix* rot = motherPhysical->GetRotation();
1004 if ( rot )
1005 {
1006 exitNormal *= rot->inverse();
1007 }
1008 }
1009 }
1010 else
1011 {
1012 validExitNormal = false;
1013 // calculatedExitNormal= false;
1014 }
1015 }
1016#endif
1017
1018
1019 G4bool daughterDeterminedStep = false;
1020 G4ThreeVector daughtNormRepCrd;
1021 // Exit normal of daughter transformed to
1022 // the coordinate system of Replica (i.e. last depth)
1023
1024 //
1025 // Compute daughter safeties & intersections
1026 //
1027 localNoDaughters = repLogical->GetNoDaughters();
1028 for ( sampleNo=localNoDaughters-1; sampleNo>=0; sampleNo-- )
1029 {
1030 samplePhysical = repLogical->GetDaughter((G4int)sampleNo);
1031 if ( samplePhysical!=blockedExitedVol )
1032 {
1033 G4ThreeVector localExitNorm;
1034 G4ThreeVector normReplicaCoord;
1035
1036 G4AffineTransform sampleTf(samplePhysical->GetRotation(),
1037 samplePhysical->GetTranslation());
1038 sampleTf.Invert();
1039 const G4ThreeVector samplePoint =
1040 sampleTf.TransformPoint(localPoint);
1041 const G4VSolid* sampleSolid =
1042 samplePhysical->GetLogicalVolume()->GetSolid();
1043 const G4double sampleSafetyDistance =
1044 sampleSolid->DistanceToIn(samplePoint);
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 // This norm can be calculated later, if needed daughter is available
1066 localExitNorm = sampleSolid->SurfaceNormal(samplePoint);
1067 daughtNormRepCrd = sampleTf.InverseTransformAxis(localExitNorm);
1068#endif
1069
1070#ifdef G4VERBOSE
1071 // Check to see that the resulting point is indeed in/on volume.
1072 // This check could eventually be made only for successful candidate.
1073
1074 if ( ( fCheck ) && ( sampleStepDistance < kInfinity ) )
1075 {
1076 G4ThreeVector intersectionPoint;
1077 intersectionPoint = samplePoint
1078 + sampleStepDistance * sampleDirection;
1079 EInside insideIntPt = sampleSolid->Inside(intersectionPoint);
1080 if ( insideIntPt != kSurface )
1081 {
1082 G4long oldcoutPrec = G4cout.precision(16);
1083 std::ostringstream message;
1084 message << "Navigator gets conflicting response from Solid."
1085 << G4endl
1086 << " Inaccurate DistanceToIn for solid "
1087 << sampleSolid->GetName() << G4endl
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) = "
1101 << sampleSolid->DistanceToIn(intersectionPoint)
1102 << G4endl;
1103}
1104 if ( insideIntPt != kOutside ) {
1105 message << " DistanceToOut(p) = "
1106 << sampleSolid->DistanceToOut(intersectionPoint);
1107}
1108 G4Exception("G4ReplicaNavigation::ComputeStep()",
1109 "GeomNav1002", JustWarning, message);
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 // G4ThreeVector daughtNormGlobal =
1125 // GlobalToLastDepth.Inverse().TransformAxis(daughtNormRepCrd);
1126 // ==> Can calculate it, but have no way to transmit it to caller (for now)
1127
1128 exitNormalVector = globalToLocalTop.InverseTransformAxis(daughtNormGlobal);
1129 validExitNormal = false; // Entering daughter - never convex for parent
1130
1131 calculatedExitNormal = true;
1132 }
1133 // calculatedExitNormal= true; // Force it to true -- dubious
1134#endif
1135
1136 newSafety = ourSafety;
1137 return ourStep;
1138}
@ JustWarning
CLHEP::HepRotation G4RotationMatrix
bool G4bool
Definition G4Types.hh:86
#define G4endl
Definition G4ios.hh:67
G4GLOB_DLL std::ostream G4cout
double dot(const Hep3Vector &) const
HepRotation inverse() const
G4ThreeVector InverseTransformAxis(const G4ThreeVector &axis) const
G4ThreeVector TransformAxis(const G4ThreeVector &axis) const
const G4AffineTransform & GetTopTransform() const
const G4String & GetName() const
G4String GetName() const
void DumpInfo() const
virtual G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const =0
G4ThreeVector exitNormal

◆ ComputeTransformation() [1/2]

void G4ReplicaNavigation::ComputeTransformation ( const G4int replicaNo,
G4VPhysicalVolume * pVol ) const

Definition at line 696 of file G4ReplicaNavigation.cc.

698{
699 G4double val;
700
701 // Replication data
702 //
703 EAxis axis;
704 G4int nReplicas;
705 G4double width, offset;
706 G4bool consuming;
707
708 pVol->GetReplicationData(axis, nReplicas, width, offset, consuming);
709
710 switch (axis)
711 {
712 case kXAxis:
713 val = -width*0.5*(nReplicas-1)+width*replicaNo;
714 pVol->SetTranslation(G4ThreeVector(val,0,0));
715 break;
716 case kYAxis:
717 val = -width*0.5*(nReplicas-1)+width*replicaNo;
718 pVol->SetTranslation(G4ThreeVector(0,val,0));
719 break;
720 case kZAxis:
721 val = -width*0.5*(nReplicas-1)+width*replicaNo;
722 pVol->SetTranslation(G4ThreeVector(0,0,val));
723 break;
724 case kPhi:
725 val = -(offset+width*(replicaNo+0.5));
726 SetPhiTransformation(val,pVol);
727 break;
728 case kRho:
729 // No setup required for radial case
730 default:
731 break;
732 }
733}
G4ThreadLocal T * G4GeomSplitter< T >::offset
virtual void GetReplicationData(EAxis &axis, G4int &nReplicas, G4double &width, G4double &offset, G4bool &consuming) const =0
void SetTranslation(const G4ThreeVector &v)
EAxis
Definition geomdefs.hh:54
@ kPhi
Definition geomdefs.hh:60
@ kYAxis
Definition geomdefs.hh:56
@ kXAxis
Definition geomdefs.hh:55
@ kZAxis
Definition geomdefs.hh:57
@ kRho
Definition geomdefs.hh:58

◆ ComputeTransformation() [2/2]

void G4ReplicaNavigation::ComputeTransformation ( const G4int replicaNo,
G4VPhysicalVolume * pVol,
G4ThreeVector & point ) const

Definition at line 640 of file G4ReplicaNavigation.cc.

643{
644 G4double val,cosv,sinv,tmpx,tmpy;
645
646 // Replication data
647 //
648 EAxis axis;
649 G4int nReplicas;
650 G4double width,offset;
651 G4bool consuming;
652
653 pVol->GetReplicationData(axis, nReplicas, width, offset, consuming);
654
655 switch (axis)
656 {
657 case kXAxis:
658 val = -width*0.5*(nReplicas-1)+width*replicaNo;
659 pVol->SetTranslation(G4ThreeVector(val,0,0));
660 point.setX(point.x()-val);
661 break;
662 case kYAxis:
663 val = -width*0.5*(nReplicas-1)+width*replicaNo;
664 pVol->SetTranslation(G4ThreeVector(0,val,0));
665 point.setY(point.y()-val);
666 break;
667 case kZAxis:
668 val = -width*0.5*(nReplicas-1)+width*replicaNo;
669 pVol->SetTranslation(G4ThreeVector(0,0,val));
670 point.setZ(point.z()-val);
671 break;
672 case kPhi:
673 val = -(offset+width*(replicaNo+0.5));
674 SetPhiTransformation(val,pVol);
675 cosv = std::cos(val);
676 sinv = std::sin(val);
677 tmpx = point.x()*cosv-point.y()*sinv;
678 tmpy = point.x()*sinv+point.y()*cosv;
679 point.setY(tmpy);
680 point.setX(tmpx);
681 break;
682 case kRho:
683 // No setup required for radial case
684 default:
685 break;
686 }
687}
double z() const
double x() const
void setY(double)
double y() const
void setZ(double)
void setX(double)

◆ DistanceToOut() [1/2]

G4double G4ReplicaNavigation::DistanceToOut ( const G4VPhysicalVolume * pVol,
const G4int replicaNo,
const G4ThreeVector & localPoint ) const

Definition at line 174 of file G4ReplicaNavigation.cc.

177{
178 // Replication data
179 //
180 EAxis axis;
181 G4int nReplicas;
182 G4double width,offset;
183 G4bool consuming;
184
185 G4double safety = 0.;
186 G4double safe1,safe2;
187 G4double coord, rho, rmin, rmax;
188
189 pVol->GetReplicationData(axis, nReplicas, width, offset, consuming);
190 switch(axis)
191 {
192 case kXAxis:
193 case kYAxis:
194 case kZAxis:
195 coord = localPoint(axis);
196 safe1 = width*0.5-coord;
197 safe2 = width*0.5+coord;
198 safety = (safe1<=safe2) ? safe1 : safe2;
199 break;
200 case kPhi:
201 if ( localPoint.y()<=0 )
202 {
203 safety = localPoint.x()*std::sin(width*0.5)
204 + localPoint.y()*std::cos(width*0.5);
205 }
206 else
207 {
208 safety = localPoint.x()*std::sin(width*0.5)
209 - localPoint.y()*std::cos(width*0.5);
210 }
211 break;
212 case kRho:
213 rho = localPoint.perp();
214 rmax = width*(replicaNo+1)+offset;
215 if ( (replicaNo != 0)||(offset != 0.0) )
216 {
217 rmin = rmax-width;
218 safe1 = rho-rmin;
219 safe2 = rmax-rho;
220 safety = (safe1<=safe2) ? safe1 : safe2;
221 }
222 else
223 {
224 safety = rmax-rho;
225 }
226 break;
227 default:
228 G4Exception("G4ReplicaNavigation::DistanceToOut()", "GeomNav0002",
229 FatalException, "Unknown axis!");
230 break;
231 }
232 return (safety >= halfkCarTolerance) ? safety : 0;
233}
double perp() const

Referenced by ComputeSafety(), and ComputeStep().

◆ DistanceToOut() [2/2]

G4double G4ReplicaNavigation::DistanceToOut ( const G4VPhysicalVolume * pVol,
const G4int replicaNo,
const G4ThreeVector & localPoint,
const G4ThreeVector & localDirection,
G4ExitNormal & candidateNormal ) const

Definition at line 240 of file G4ReplicaNavigation.cc.

245{
246 // Replication data
247 //
248 EAxis axis;
249 G4int nReplicas;
250 G4double width, offset;
251 G4bool consuming;
252
253 G4double Dist=kInfinity;
254 G4double coord, Comp, lindist;
255 G4double signC = 0.0;
256 G4ExitNormal candidateNormal;
257
258 pVol->GetReplicationData(axis, nReplicas, width, offset, consuming);
259 switch(axis)
260 {
261 case kXAxis:
262 case kYAxis:
263 case kZAxis:
264 coord = localPoint(axis);
265 Comp = localDirection(axis);
266 if ( Comp>0 )
267 {
268 lindist = width*0.5-coord;
269 Dist = (lindist>0) ? lindist/Comp : 0;
270 signC= 1.0;
271 }
272 else if ( Comp<0 )
273 {
274 lindist = width*0.5+coord;
275 Dist = (lindist>0) ? -lindist/Comp : 0;
276 signC= -1.0;
277 }
278 else
279 {
280 Dist = kInfinity;
281 }
282 // signC = sign<G4double>(Comp)
283 candidateNormal.exitNormal = ( signC * VecCartAxes[axis]);
284 candidateNormal.calculated = true;
285 candidateNormal.validConvex = true;
286 candidateNormal.exitSide =
287 (Comp>0) ? SideCartAxesPlus[axis] : SideCartAxesMinus[axis];
288 break;
289 case kPhi:
290 Dist = DistanceToOutPhi(localPoint,localDirection,width,candidateNormal);
291 // candidateNormal set in call
292 break;
293 case kRho:
294 Dist = DistanceToOutRad(localPoint,localDirection,width,offset,
295 replicaNo,candidateNormal);
296 // candidateNormal set in call
297 break;
298 default:
299 G4Exception("G4ReplicaNavigation::DistanceToOut()", "GeomNav0002",
300 FatalException, "Unknown axis!");
301 break;
302 }
303
304 arExitNormal= candidateNormal; // .exitNormal;
305
306 return Dist;
307}

◆ GetVerboseLevel()

G4int G4ReplicaNavigation::GetVerboseLevel ( ) const
inline

◆ Inside()

EInside G4ReplicaNavigation::Inside ( const G4VPhysicalVolume * pVol,
const G4int replicaNo,
const G4ThreeVector & localPoint ) const

Definition at line 68 of file G4ReplicaNavigation.cc.

71{
72 EInside in = kOutside;
73
74 // Replication data
75 //
76 EAxis axis;
77 G4int nReplicas;
78 G4double width, offset;
79 G4bool consuming;
80
81 G4double coord, rad2, rmin, tolRMax2, rmax, tolRMin2;
82
83 pVol->GetReplicationData(axis, nReplicas, width, offset, consuming);
84
85 switch (axis)
86 {
87 case kXAxis:
88 case kYAxis:
89 case kZAxis:
90 coord = std::fabs(localPoint(axis))-width*0.5;
91 if ( coord<=-halfkCarTolerance )
92 {
93 in = kInside;
94 }
95 else if ( coord<=halfkCarTolerance )
96 {
97 in = kSurface;
98 }
99 break;
100 case kPhi:
101 if ( (localPoint.y() != 0.0)||(localPoint.x() != 0.0) )
102 {
103 coord = std::fabs(std::atan2(localPoint.y(),localPoint.x()))-width*0.5;
104 if ( coord<=-halfkAngTolerance )
105 {
106 in = kInside;
107 }
108 else if ( coord<=halfkAngTolerance )
109 {
110 in = kSurface;
111 }
112 }
113 else
114 {
115 in = kSurface;
116 }
117 break;
118 case kRho:
119 rad2 = localPoint.perp2();
120 rmax = (replicaNo+1)*width+offset;
121 tolRMax2 = rmax-halfkRadTolerance;
122 tolRMax2 *= tolRMax2;
123 if ( rad2>tolRMax2 )
124 {
125 tolRMax2 = rmax+halfkRadTolerance;
126 tolRMax2 *= tolRMax2;
127 if ( rad2<=tolRMax2 )
128 {
129 in = kSurface;
130 }
131 }
132 else
133 {
134 // Known to be inside outer radius
135 //
136 if ( (replicaNo != 0)||(offset != 0.0) )
137 {
138 rmin = rmax-width;
139 tolRMin2 = rmin-halfkRadTolerance;
140 tolRMin2 *= tolRMin2;
141 if ( rad2>tolRMin2 )
142 {
143 tolRMin2 = rmin+halfkRadTolerance;
144 tolRMin2 *= tolRMin2;
145 if ( rad2>=tolRMin2 )
146 {
147 in = kInside;
148 }
149 else
150 {
151 in = kSurface;
152 }
153 }
154 }
155 else
156 {
157 in = kInside;
158 }
159 }
160 break;
161 default:
162 G4Exception("G4ReplicaNavigation::Inside()", "GeomNav0002",
163 FatalException, "Unknown axis!");
164 break;
165 }
166 return in;
167}
double perp2() const

Referenced by BackLocate().

◆ LevelLocate()

G4bool G4ReplicaNavigation::LevelLocate ( G4NavigationHistory & history,
const G4VPhysicalVolume * blockedVol,
const G4int blockedNum,
const G4ThreeVector & globalPoint,
const G4ThreeVector * globalDirection,
const G4bool pLocatedOnEdge,
G4ThreeVector & localPoint )
inline

◆ SetVerboseLevel()

void G4ReplicaNavigation::SetVerboseLevel ( G4int level)
inline

The documentation for this class was generated from the following files: