57 halfkCarTolerance = kCarTolerance*0.5;
58 halfkRadTolerance = kRadTolerance*0.5;
59 halfkAngTolerance = kAngTolerance*0.5;
60 fMinStep = 0.05*kCarTolerance;
69 const G4int replicaNo,
81 G4double coord, rad2, rmin, tolRMax2, rmax, tolRMin2;
90 coord = std::fabs(localPoint(axis))-width*0.5;
91 if ( coord<=-halfkCarTolerance )
95 else if ( coord<=halfkCarTolerance )
101 if ( (localPoint.
y() != 0.0)||(localPoint.
x() != 0.0) )
103 coord = std::fabs(std::atan2(localPoint.
y(),localPoint.
x()))-width*0.5;
104 if ( coord<=-halfkAngTolerance )
108 else if ( coord<=halfkAngTolerance )
119 rad2 = localPoint.
perp2();
120 rmax = (replicaNo+1)*width+
offset;
121 tolRMax2 = rmax-halfkRadTolerance;
122 tolRMax2 *= tolRMax2;
125 tolRMax2 = rmax+halfkRadTolerance;
126 tolRMax2 *= tolRMax2;
127 if ( rad2<=tolRMax2 )
136 if ( (replicaNo != 0)||(
offset != 0.0) )
139 tolRMin2 = rmin-halfkRadTolerance;
140 tolRMin2 *= tolRMin2;
143 tolRMin2 = rmin+halfkRadTolerance;
144 tolRMin2 *= tolRMin2;
145 if ( rad2>=tolRMin2 )
162 G4Exception(
"G4ReplicaNavigation::Inside()",
"GeomNav0002",
175 const G4int replicaNo,
195 coord = localPoint(axis);
196 safe1 = width*0.5-coord;
197 safe2 = width*0.5+coord;
198 safety = (safe1<=safe2) ? safe1 : safe2;
201 if ( localPoint.
y()<=0 )
203 safety = localPoint.
x()*std::sin(width*0.5)
204 + localPoint.
y()*std::cos(width*0.5);
208 safety = localPoint.
x()*std::sin(width*0.5)
209 - localPoint.
y()*std::cos(width*0.5);
213 rho = localPoint.
perp();
214 rmax = width*(replicaNo+1)+
offset;
215 if ( (replicaNo != 0)||(
offset != 0.0) )
220 safety = (safe1<=safe2) ? safe1 : safe2;
228 G4Exception(
"G4ReplicaNavigation::DistanceToOut()",
"GeomNav0002",
232 return (safety >= halfkCarTolerance) ? safety : 0;
241 const G4int replicaNo,
264 coord = localPoint(axis);
265 Comp = localDirection(axis);
268 lindist = width*0.5-coord;
269 Dist = (lindist>0) ? lindist/Comp : 0;
274 lindist = width*0.5+coord;
275 Dist = (lindist>0) ? -lindist/Comp : 0;
283 candidateNormal.
exitNormal = ( signC * VecCartAxes[axis]);
287 (Comp>0) ? SideCartAxesPlus[axis] : SideCartAxesMinus[axis];
290 Dist = DistanceToOutPhi(localPoint,localDirection,width,candidateNormal);
294 Dist = DistanceToOutRad(localPoint,localDirection,width,
offset,
295 replicaNo,candidateNormal);
299 G4Exception(
"G4ReplicaNavigation::DistanceToOut()",
"GeomNav0002",
304 arExitNormal= candidateNormal;
314G4ReplicaNavigation::DistanceToOutPhi(
const G4ThreeVector& localPoint,
322 G4double sinSPhi = -2.0, cosSPhi = -2.0;
323 G4double pDistS, pDistE, compS, compE, Dist, dist2, yi;
327 if ( (localPoint.
x()!=0.0) || (localPoint.
y()!=0.0) )
329 sinSPhi = std::sin(-width*0.5);
330 cosSPhi = std::cos(width*0.5);
334 pDistS = localPoint.
x()*sinSPhi-localPoint.
y()*cosSPhi;
336 pDistE = localPoint.
x()*sinSPhi+localPoint.
y()*cosSPhi;
341 compS = -sinSPhi*localDirection.
x()+cosSPhi*localDirection.
y();
342 compE = -sinSPhi*localDirection.
x()-cosSPhi*localDirection.
y();
344 if ( (pDistS<=halfkCarTolerance)&&(pDistE<=halfkCarTolerance) )
350 dist2 = pDistS/compS;
351 yi = localPoint.
y()+dist2*localDirection.
y();
357 Dist = (pDistS<=-halfkCarTolerance) ? dist2 : 0;
371 dist2 = pDistE/compE;
377 yi = localPoint.
y()+dist2*localDirection.
y();
385 Dist = (pDistE<=-halfkCarTolerance) ? dist2 : 0;
391 else if ( (pDistS>halfkCarTolerance)&&(pDistE>halfkCarTolerance) )
396 Dist = ((compS>=0)&&(compE>=0)) ? kInfinity : 0;
398 else if ( (pDistS>halfkCarTolerance)&&(pDistE<=halfkCarTolerance) )
404 dist2 = pDistE/compE;
405 yi = localPoint.
y()+dist2*localDirection.
y();
410 Dist = (yi>0) ? dist2 : kInfinity;
427 dist2 = pDistS/compS;
428 yi = localPoint.
y()+dist2*localDirection.
y();
433 Dist = (yi<0) ? dist2 : kInfinity;
454 if( (std::fabs(localDirection.
phi())<=width*0.5) )
488G4ReplicaNavigation::DistanceToOutRad(
const G4ThreeVector& localPoint,
492 const G4int replicaNo,
495 G4double rmin, rmax, t1, t2, t3, deltaR;
515 rmin = replicaNo*width+
offset;
516 rmax = (replicaNo+1)*width+
offset;
518 t1 = 1.0-localDirection.
z()*localDirection.
z();
519 t2 = localPoint.
x()*localDirection.
x()+localPoint.
y()*localDirection.
y();
520 t3 = localPoint.
x()*localPoint.
x()+localPoint.
y()*localPoint.
y();
530 deltaR = t3-rmax*rmax;
535 if ( deltaR<-halfkRadTolerance )
539 srd = -b+std::sqrt(b*b-c);
557 deltaR = t3-rmin*rmin;
567 srd = (deltaR>halfkRadTolerance) ? -b-std::sqrt(d2) : 0.0;
576 deltaR = t3-rmax*rmax;
579 srd = (d2 < 0.) ? 0.0 : -b+std::sqrt(d2);
587 deltaR = t3-rmax*rmax;
591 srd = (d2 < 0.) ? 0.0 : -b+std::sqrt(d2);
608 xi = localPoint.
x() + srd*localDirection.
x();
609 yi = localPoint.
y() + srd*localDirection.
y();
618 normalR *= (-1.0)/rmin;
658 val = -width*0.5*(nReplicas-1)+width*replicaNo;
660 point.
setX(point.
x()-val);
663 val = -width*0.5*(nReplicas-1)+width*replicaNo;
665 point.
setY(point.
y()-val);
668 val = -width*0.5*(nReplicas-1)+width*replicaNo;
670 point.
setZ(point.
z()-val);
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;
713 val = -width*0.5*(nReplicas-1)+width*replicaNo;
717 val = -width*0.5*(nReplicas-1)+width*replicaNo;
721 val = -width*0.5*(nReplicas-1)+width*replicaNo;
725 val = -(
offset+width*(replicaNo+0.5));
726 SetPhiTransformation(val,pVol);
744 const G4double currentProposedStepLength,
749 G4bool& calculatedExitNormal,
754 G4int& blockedReplicaNo )
761 G4double ourStep=currentProposedStepLength;
763 G4double sampleStep, sampleSafety, motherStep, motherSafety;
764 G4long localNoDaughters, sampleNo;
769 calculatedExitNormal=
false;
773 if ( exiting&&validExitNormal )
775 if ( localDirection.
dot(exitNormalVector)>=kMinExitingNormalCosine )
779 blockedExitedVol = *pBlockedPhysical;
799 ourSafety = std::min( ourSafety, sampleSafety);
801 if ( sampleSafety<ourStep )
809 if ( sampleStep<ourStep )
811 ourStep = sampleStep;
815 exitNormalStc = normalOutStc;
818 calculatedExitNormal =
true;
821 const G4int secondDepth = topDepth;
834 if ( sampleSafety < ourSafety )
836 ourSafety = sampleSafety;
838 if ( sampleSafety < ourStep )
847 if ( sampleStep < ourStep )
849 ourStep = sampleStep;
858 exitNormalStc = normalOutStc;
860 calculatedExitNormal =
true;
869 G4bool exitConvex =
false;
873 motherPhysical = history.
GetVolume(depth);
878 motherStep = motherSolid->
DistanceToOut(repPoint,repDirection,
true,
879 &exitConvex,&exitVectorMother);
882 motherNormalStc =
G4ExitNormal( exitVectorMother,
true,
false,
884 calculatedExitNormal =
true;
888 G4bool motherDeterminedStep = (motherStep<ourStep);
890 if( (!exitConvex) && motherDeterminedStep )
893 motherNormalStc =
G4ExitNormal( exitVectorMother,
true,
false,
899 calculatedExitNormal =
true;
901 if( motherDeterminedStep )
906 exitNormalStc = motherNormalStc;
907 exitNormalStc.
exitNormal = globalExitNormalTop;
917 if( ourStep<fMinStep )
919 ourStep = 2*kCarTolerance;
922 if ( motherSafety<ourSafety )
924 ourSafety = motherSafety;
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 )
945 "Point is far outside Current Volume !" );
951 "Point is a little outside Current Volume.");
960 if( motherDeterminedStep )
962 ourStep = motherStep;
968 if ( calculatedExitNormal )
970 if ( motherDeterminedStep )
972 exitNormalVector = motherNormalStc.
exitNormal;
977 exitNormalVector = globalToLocalTop.
TransformAxis(exitNormalGlobal);
983 if ( rot !=
nullptr )
985 exitNormalVector *= rot->
inverse();
991 validExitNormal =
false;
995 if ( motherSafety<=ourStep )
997 if ( motherStep<=ourStep )
999 ourStep = motherStep;
1001 if ( validExitNormal )
1012 validExitNormal =
false;
1019 G4bool daughterDeterminedStep =
false;
1028 for ( sampleNo=localNoDaughters-1; sampleNo>=0; sampleNo-- )
1031 if ( samplePhysical!=blockedExitedVol )
1043 const G4double sampleSafetyDistance =
1045 if ( sampleSafetyDistance<ourSafety )
1047 ourSafety = sampleSafetyDistance;
1049 if ( sampleSafetyDistance<=ourStep )
1052 const G4double sampleStepDistance =
1053 sampleSolid->
DistanceToIn(samplePoint,sampleDirection);
1054 if ( sampleStepDistance<=ourStep )
1056 daughterDeterminedStep =
true;
1058 ourStep = sampleStepDistance;
1061 *pBlockedPhysical = samplePhysical;
1062 blockedReplicaNo = (
G4int)sampleNo;
1064#ifdef DAUGHTER_NORMAL_ALSO
1074 if ( ( fCheck ) && ( sampleStepDistance < kInfinity ) )
1077 intersectionPoint = samplePoint
1078 + sampleStepDistance * sampleDirection;
1079 EInside insideIntPt = sampleSolid->
Inside(intersectionPoint);
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-";
1095 message <<
"-kSurface-";
1097 message <<
" for this point !" <<
G4endl
1098 <<
" Point = " << intersectionPoint <<
G4endl;
1099 if ( insideIntPt !=
kInside ) {
1100 message <<
" DistanceToIn(p) = "
1105 message <<
" DistanceToOut(p) = "
1110 G4cout.precision(oldcoutPrec);
1119 calculatedExitNormal &= (!daughterDeterminedStep);
1121#ifdef DAUGHTER_NORMAL_ALSO
1122 if( daughterDeterminedStep )
1129 validExitNormal =
false;
1131 calculatedExitNormal =
true;
1136 newSafety = ourSafety;
1160 G4long localNoDaughters, sampleNo;
1173 if ( sampleSafety<ourSafety )
1175 ourSafety = sampleSafety;
1187 if ( sampleSafety<ourSafety )
1189 ourSafety = sampleSafety;
1197 motherPhysical = history.
GetVolume(depth);
1201 if ( sampleSafety<ourSafety )
1203 ourSafety = sampleSafety;
1209 for ( sampleNo=localNoDaughters-1; sampleNo>=0; sampleNo-- )
1212 if ( samplePhysical!=blockedExitedVol )
1221 const G4double sampleSafetyDistance =
1223 if ( sampleSafetyDistance<ourSafety )
1225 ourSafety = sampleSafetyDistance;
1241 G4bool& notKnownInside )
const
1246 G4int mdepth, depth, cdepth;
1253 for ( mdepth=cdepth-1; mdepth>=0; mdepth-- )
1262 if( pNRMother ==
nullptr )
1267 G4Exception(
"G4ReplicaNavigation::BackLocate()",
"GeomNav0002",
1274 insideCode = motherSolid->
Inside(goodPoint);
1286 notKnownInside =
false;
1291 for ( depth=mdepth+1; depth<cdepth; ++depth)
1299 localPoint = goodPoint;
1303 goodPoint = repPoint;
1315 localPoint = goodPoint;
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
G4ThreadLocal T * G4GeomSplitter< T >::offset
CLHEP::HepRotation G4RotationMatrix
CLHEP::Hep3Vector G4ThreeVector
G4GLOB_DLL std::ostream G4cout
double dot(const Hep3Vector &) const
HepRotation inverse() const
G4double GetSurfaceTolerance() const
G4double GetRadialTolerance() const
static G4GeometryTolerance * GetInstance()
G4double GetAngularTolerance() const
G4VSolid * GetSolid() const
std::size_t GetNoDaughters() const
G4VPhysicalVolume * GetDaughter(const std::size_t i) const
G4int GetReplicaNo(G4int n) const
const G4AffineTransform & GetTopTransform() const
G4int GetTopReplicaNo() const
G4VPhysicalVolume * GetVolume(G4int n) const
std::size_t GetDepth() const
G4VPhysicalVolume * GetTopVolume() const
EVolume GetVolumeType(G4int n) const
const G4AffineTransform & GetTransform(G4int n) const
G4double DistanceToOut(const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
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
void ComputeTransformation(const G4int replicaNo, G4VPhysicalVolume *pVol, G4ThreeVector &point) const
EInside Inside(const G4VPhysicalVolume *pVol, const G4int replicaNo, const G4ThreeVector &localPoint) const
EInside BackLocate(G4NavigationHistory &history, const G4ThreeVector &globalPoint, G4ThreeVector &localPoint, const G4bool &exiting, G4bool ¬KnownInside) const
const G4RotationMatrix * GetRotation() const
const G4ThreeVector GetTranslation() const
G4LogicalVolume * GetLogicalVolume() const
virtual void GetReplicationData(EAxis &axis, G4int &nReplicas, G4double &width, G4double &offset, G4bool &consuming) const =0
const G4String & GetName() const
void SetTranslation(const G4ThreeVector &v)
virtual EInside Inside(const G4ThreeVector &p) const =0
virtual G4double DistanceToOut(const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *n=nullptr) const =0
virtual G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const =0
virtual G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const =0