// // ******************************************************************** // * License and Disclaimer * // * * // * The Geant4 software is copyright of the Copyright Holders of * // * the Geant4 Collaboration. It is provided under the terms and * // * conditions of the Geant4 Software License, included in the file * // * LICENSE and available at http://cern.ch/geant4/license . These * // * include a list of copyright holders. * // * * // * Neither the authors of this software system, nor their employing * // * institutes,nor the agencies providing financial support for this * // * work make any representation or warranty, express or implied, * // * regarding this software system or assume any liability for its * // * use. Please see the license in the file LICENSE and URL above * // * for the full disclaimer and the limitation of liability. * // * * // * This code implementation is the result of the scientific and * // * technical work of the GEANT4 collaboration. * // * By using, copying, modifying or distributing the software (or * // * any work based on the software) you agree to acknowledge its * // * use in resulting scientific publications, and indicate your * // * acceptance of all terms of the Geant4 Software license. * // ******************************************************************** // // $Id: testG4Sphere.cc,v 1.30 2010/03/24 09:50:03 grichine Exp $ // GEANT4 tag $Name: geant4-09-04-ref-00 $ // // G4Sphere Test File // // o Basic asserts on each function + // awkward cases for tracking / geom algorithms // // o Add tests on dicovering bugs in G4Sphere.cc... // // History: // 28.03.95 P.Kent Initial version // 20.10.96 V.Grichine Final modifications to commit #include "G4ios.hh" #include #include #include "globals.hh" #include "geomdefs.hh" #include "ApproxEqual.hh" #include "G4GeometryTolerance.hh" #include "G4ThreeVector.hh" #include "G4RotationMatrix.hh" #include "G4AffineTransform.hh" #include "G4VoxelLimits.hh" #include "G4Sphere.hh" #include "Randomize.hh" //const G4double kApproxEqualTolerance = kCarTolerance; // Return true if the double check is approximately equal to target // // Process: // // Return true is difference < kApproxEqualTolerance //G4bool ApproxEqual(const G4double check,const G4double target) //{ // return (std::fabs(check-target)GetAngularTolerance(); EInside inside; G4int i, iMax; G4double Dist, vol, volCheck; G4double phi, cosTheta, sinTheta, rMax, rRand, zP; G4ThreeVector pzero(0,0,0),px(30,0,0),py(0,30,0),pz(0,0,30); G4ThreeVector pmx(-30,0,0),pmy(0,-30,0),pmz(0,0,-30); G4ThreeVector pbigx(100,0,0),pbigy(0,100,0),pbigz(0,0,100); G4ThreeVector pbigmx(-100,0,0),pbigmy(0,-100,0),pbigmz(0,0,-100); G4ThreeVector ponrmin1(45,0,0),ponrmax1(50,0,0),ponzmax(0,0,50), ponrmin2(45/std::sqrt(2.),45/std::sqrt(2.),0), ponrmin3(0,0,-45),ponrminJ(0,0,-300),ponrmaxJ(0,0,-500), ponrmax2(50/std::sqrt(2.),50/std::sqrt(2.),0); G4ThreeVector ponphi1(48/std::sqrt(2.),-48/std::sqrt(2.),0), ponphi2(48/std::sqrt(2.),48/std::sqrt(2.),0), pInPhi(48*0.866,-24,0), pOverPhi(-48/std::sqrt(2.),48/std::sqrt(2.),0); G4ThreeVector pontheta1(0,48*std::sin(pi/4),48*std::cos(pi/4)), pontheta2(0,48*std::sin(pi/4),-48*std::cos(pi/4)); G4ThreeVector ptestphi1(-100,-45/std::sqrt(2.),0), ptestphi2(-100,45/std::sqrt(2.),0); G4ThreeVector ptesttheta1(0,48/std::sqrt(2.),100), ptesttheta2(0,48/std::sqrt(2.),-100); // Directions G4ThreeVector vx(1,0,0),vy(0,1,0),vz(0,0,1); G4ThreeVector vmx(-1,0,0),vmy(0,-1,0),vmz(0,0,-1); G4ThreeVector vxy(1/std::sqrt(2.),1/std::sqrt(2.),0),vmxmy(-1/std::sqrt(2.),-1/std::sqrt(2.),0); G4ThreeVector vxmy(1/std::sqrt(2.),-1/std::sqrt(2.),0),vmxy(-1/std::sqrt(2.),1/std::sqrt(2.),0); G4ThreeVector vxmz(1/std::sqrt(2.),0.,-1/std::sqrt(2.)),vymz(0.,1/std::sqrt(2.),-1/std::sqrt(2.)); G4ThreeVector vmxmz(-1/std::sqrt(2.),0.,-1/std::sqrt(2.)),vmxz(-1/std::sqrt(2.),0.,1/std::sqrt(2.)); G4ThreeVector v345exit1(-0.8,0.6,0),v345exit2(0.8,0.6,0), v345exit3(0.6,0.8,0); G4ThreeVector pRand, vRand, norm, *pNorm; G4bool *pgoodNorm,goodNorm,calcNorm=true; pNorm=&norm; pgoodNorm=&goodNorm; G4Sphere s1("Solid G4Sphere",0,50,0,twopi,0,pi); G4Sphere sn1("sn1",0,50,halfpi,3.*halfpi,0,pi); // Theta sections G4Sphere sn11("sn11",0,50,0,twopi,0.,halfpi); G4Sphere sn12("sn12",0,50,0,twopi,0.,0.25*pi); G4Sphere sn13("sn13",0,50,0,twopi,0.75*pi,0.25*pi); G4Sphere sn14("sn14",0,50,0,twopi,0.25*pi,0.75*pi); G4Sphere sn15("sn15",0,50,0,twopi,89.*deg,91.*deg); G4Sphere sn16("sn16",0,50,0,twopi,0.*deg,89.*deg); G4Sphere sn17("sn17",0,50,0,twopi,91.*deg,89.*deg); G4Sphere s2("Spherical Shell",45,50,0,twopi,0,pi); G4Sphere sn2("sn2",45,50,halfpi,halfpi,0,pi); G4Sphere sn22("sn22",0,50,halfpi,halfpi,0,pi); G4Sphere s3("Band (theta segment)",45,50,0,twopi,pi/4,halfpi); G4Sphere s32("Band (theta segment2)",45,50,0,twopi,0,pi/4); G4Sphere s33("Band (theta segment1)",45,50,0,twopi,pi*3/4,pi/4); G4Sphere s34("Band (theta segment)",4,50,0,twopi,pi/4,halfpi); G4Sphere s4("Band (phi segment)",45,50,-pi/4,halfpi,0,twopi); // G4cout<<"s4.fSPhi = "<unit(); assert(ApproxEqual(Dist,0)&&*pgoodNorm&&ApproxEqual(*pNorm,vx)); Dist=s2.DistanceToOut(ponrmin1,vx,calcNorm,pgoodNorm,pNorm); *pNorm=pNorm->unit(); assert(ApproxEqual(Dist,5)&&*pgoodNorm&&ApproxEqual(*pNorm,vx)); Dist=s2.DistanceToOut(ponrmax2,vx,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,0)&&*pgoodNorm&&ApproxEqual(*pNorm,vxy)); Dist=s1.DistanceToOut(pzero,vx,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vx)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vmx,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vmx)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vy,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vy)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vmy,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vmy)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vz,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vz)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vmz,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vmz)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vxy,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vxy)&&*pgoodNorm); Dist=s4.DistanceToOut(ponphi1,vx,calcNorm,pgoodNorm,pNorm); //assert(ApproxEqual(Dist,0)&&ApproxEqual(pNorm->unit(),vmxmy)&&*pgoodNorm); Dist=s4.DistanceToOut(ponphi2,vx,calcNorm,pgoodNorm,pNorm); // assert(ApproxEqual(Dist,0)&&ApproxEqual(pNorm->unit(),vmxy)&&*pgoodNorm); Dist=s3.DistanceToOut(pontheta1,vz,calcNorm,pgoodNorm,pNorm); // assert(ApproxEqual(Dist,0)&&ApproxEqual(pNorm->unit(),vy)&&*pgoodNorm); Dist=s32.DistanceToOut(pontheta1,vmz,calcNorm,pgoodNorm,pNorm); //assert(ApproxEqual(Dist,0)&&ApproxEqual(pNorm->unit(),vmy)&&*pgoodNorm); Dist=s32.DistanceToOut(pontheta1,vz,calcNorm,pgoodNorm,pNorm); //assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vz)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vmz,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vmz)&&*pgoodNorm); Dist=s1.DistanceToOut(pzero,vxy,calcNorm,pgoodNorm,pNorm); assert(ApproxEqual(Dist,50)&&ApproxEqual(pNorm->unit(),vxy)&&*pgoodNorm); Dist=s2.DistanceToOut(ponrmin1,vxy,calcNorm,pgoodNorm,pNorm); // G4cout<<"Dist=s2.DistanceToOut(pormin1,vxy) = "< Vlad. Grichine test case, 30 Oct 2003 // G4cout << G4endl << G4endl << "" << G4endl; G4cout << "========================================================= " << G4endl; G4Sphere SphDeepNeg("DeepNegPhiSphere", 10.*mm, 1000.*mm, -270.0*degree, 280.0*degree, // start Phi, delta Phi 0.*degree, 180.0*degree ); // start Theta, delta Theta G4double phiPoint = 160.0 * degree; G4ThreeVector StartPt( radOne * std::cos(phiPoint), radOne * std::sin(phiPoint), 0.0); G4cout << "For sphere " << SphDeepNeg.GetName() << G4endl; G4cout << " Starting from point " << ptPhiSurfExct << G4endl; checkPoint( SphDeepNeg, StartPt, 0.0, vy, kInside); // Try the edges G4ThreeVector NegEdgePt( radOne * std::cos(-270.0*degree), radOne * std::sin(-270.0*degree), 0.0); G4ThreeVector PosEdgePt( radOne * std::cos(10.0*degree), radOne * std::sin(10.0*degree), 0.0); G4cout << "--------------------------------------------------------" << G4endl; G4cout << " New point " << NegEdgePt << " should be at Neg edge of -270.0 degrees " << G4endl; checkPoint( SphDeepNeg, NegEdgePt, 0.0, -vx, kSurface); checkPoint( SphDeepNeg, NegEdgePt, radOne*kAngTolerance * 0.25, -vx, kSurface); checkPoint( SphDeepNeg, NegEdgePt, -radOne*kAngTolerance * 0.25, -vx, kSurface); checkPoint( SphDeepNeg, NegEdgePt, radOne*kAngTolerance * 1.25, -vx, kInside); checkPoint( SphDeepNeg, NegEdgePt, -radOne*kAngTolerance * 1.25, -vx, kOutside); G4cout << "--------------------------------------------------------" << G4endl; G4cout << " New point " << PosEdgePt << " should be at Pos edge of +10.0 degrees " << G4endl; checkPoint( SphDeepNeg, PosEdgePt, 0.0, -vy, kSurface); checkPoint( SphDeepNeg, PosEdgePt, radOne*kAngTolerance * 0.25, -vy, kSurface); checkPoint( SphDeepNeg, PosEdgePt, -radOne*kAngTolerance * 0.25, -vy, kSurface); checkPoint( SphDeepNeg, PosEdgePt, -radOne*kAngTolerance * 1.25, -vy, kOutside); checkPoint( SphDeepNeg, PosEdgePt, radOne*kAngTolerance * 1.25, -vy, kInside); // G4double sTheta = 135*degree, pxt = -10., pyt = 10.; // G4cout<<"sTheta = "< --> --> // the function below checks that the point pnew=( origin + dist * direction ) // - the point (p+dist*dir) is located in the part of the solid given by 'expectedInResult' // - and that from there, the DistanceToIn along 'dir' is not negative or Infinite // // Use cases expected: // - 'origin' is on/near a surface and 'direction' is pointing towards the inside of the solid G4bool checkPoint( const G4Sphere &rSphere, G4ThreeVector origin, G4double dist, G4ThreeVector direction, EInside expectedInResult) { G4int verbose = 0; G4ThreeVector newPoint; G4double distIn=-1.0, distOut=-1.0; newPoint = origin + dist * direction; G4int oldPrecision= G4cout.precision(10); // G4cout << " --- Sphere " << rSphere.GetName() << "" << G4endl; if( verbose > 0 ) { G4cout << G4endl; if (verbose > 2 ) G4cout << " Sphere " << rSphere.GetName(); G4cout.precision(10); if (verbose > 1 ) G4cout << " dir= " << direction; G4cout << " dist= " << dist; } EInside inSphere= rSphere.Inside( newPoint ) ; /*======*/ G4cout.precision(15); // G4cout << " NewPoint " << newPoint << " is " G4bool goodIn= (inSphere == expectedInResult) ; if ( !goodIn ) { G4cout << " ************ Unexpected Result for Inside *************** " << G4endl; } if ( verbose || !goodIn ) { G4cout << " New point " << " is " << OutputInside( inSphere ) << " vs " << OutputInside( expectedInResult ) << " expected." << G4endl ; } G4bool goodDistIn = true; distIn = rSphere.DistanceToIn( newPoint, direction ); /*===========*/ if ( verbose ) G4cout << " DistToIn (p, dir) = " << distIn << G4endl; if( (inSphere == kOutside) && (distIn < 0.0 ) // Cannot use 0.5*kCarTolerance for Angular tolerance!! ){ G4cout << " ********** Unexpected Result for DistanceToIn from outside ********* " << G4endl; // G4cout << " It should be " << G4endl; goodDistIn = false; } if( (inSphere == kSurface ) && ( (distIn < 0.0) || (distIn >= kInfinity )) ){ G4cout << " ********** Unexpected Result for DistanceToIn on surface ********* " << G4endl; // if ( (distIn != 0.0) ) // - Can check that the return value must be 0.0 // But in general case the direction can be away from the solid, // and then a finite or kInfinity answer is correct // --> must check the direction against the normal // in order to perform this check in general case. goodDistIn = false; } if ( verbose || !goodDistIn ) { G4cout << " DistToIn (p, dir) = " << distIn << G4endl; } G4bool good= (goodIn && goodDistIn); if ( !good ){ // There was an error -- document the use case! G4cout << " --- Sphere " << rSphere.GetName() << "" << G4endl; G4cout << " Origin= " << origin << G4endl; G4cout << " Direction= " << direction << G4endl; G4cout << " dist= " << dist; G4cout << " Actual-point= " << newPoint << G4endl; } distOut = rSphere.DistanceToOut( newPoint, direction ); /*=============*/ if ( verbose ) G4cout << " DistToOut (p, dir) = " << distOut << G4endl; G4cout.precision(oldPrecision); return good; }