1148{
1149 const G4double toleratedRelativeError = 1.0e-10;
1150 G4double minStep= kInfinity, newSafety = 0.0;
1152 G4FieldTrack fieldTrack = initialState;
1154
1155
1156 G4EquationOfMotion* equationOfMotion =
1157 fpFieldPropagator->GetCurrentEquationOfMotion();
1158
1160 initialState.GetMomentum().mag(),
1161 initialState.GetRestMass() );
1162
1163#ifdef G4DEBUG_PATHFINDER
1165 if( fVerboseLevel > 2 )
1166 {
1167 G4cout <<
" G4PathFinder::DoNextCurvedStep ****** " <<
G4endl;
1168 G4cout <<
" Initial value of field track is " << fieldTrack
1169 <<
" and proposed step= " << proposedStepLength <<
G4endl;
1170 }
1171#endif
1172
1173 fPreStepCenterRenewed = true;
1174
1175 if( fNoActiveNavigators > 1 )
1176 {
1177
1178
1179 G4double minSafety= kInfinity, safety;
1180 for( numNav=0; numNav < fNoActiveNavigators; ++numNav )
1181 {
1182 safety= fpNavigator[numNav]->ComputeSafety( startPoint,
DBL_MAX,
false );
1183 fPreSafetyValues[numNav] = safety;
1184 fCurrentPreStepSafety[numNav] = safety;
1185 minSafety = std::min( safety, minSafety );
1186 }
1187
1188
1189
1190 fPreSafetyLocation = startPoint;
1191 fPreSafetyMinValue = minSafety;
1192 fPreStepLocation = startPoint;
1193 fMinSafety_PreStepPt = minSafety;
1194 }
1195
1196
1197
1198 minStep = fpFieldPropagator->ComputeStep( fieldTrack,
1199 proposedStepLength,
1200 newSafety,
1201 pCurrentPhysicalVolume,
1202 false);
1203
1204
1205
1206 fEndState = fieldTrack;
1207 fMinStep = minStep;
1208 fTrueMinStep = std::min( minStep, proposedStepLength );
1209
1210 if( fNoActiveNavigators == 1 )
1211 {
1212
1213
1214
1215 fPreSafetyValues[0] = newSafety;
1216 fPreSafetyLocation = startPoint;
1217 fPreSafetyMinValue = newSafety;
1218
1219
1220
1221 fCurrentPreStepSafety[0] = newSafety;
1222 fPreStepLocation = startPoint;
1223 fMinSafety_PreStepPt= newSafety;
1224 }
1225
1226#ifdef G4DEBUG_PATHFINDER
1227 if( fVerboseLevel > 2 )
1228 {
1229 G4cout <<
"G4PathFinder::DoNextCurvedStep : " <<
G4endl
1230 <<
" initialState = " << initialState <<
G4endl
1231 <<
" and endState = " << fEndState <<
G4endl;
1232 G4cout <<
"G4PathFinder::DoNextCurvedStep : "
1233 << " minStep = " << minStep
1234 << " proposedStepLength " << proposedStepLength
1235 <<
" safety = " << newSafety <<
G4endl;
1236 }
1237#endif
1239 if( minStep < proposedStepLength )
1240 {
1241
1242
1243
1244 G4int noLimited = 0;
1245 for( numNav=0; numNav < fNoActiveNavigators; ++numNav )
1246 {
1247 G4double finalStep, lastPreSafety = 0.0, minStepLast;
1250
1251 finalStep= fpMultiNavigator->ObtainFinalStep( numNav, lastPreSafety,
1252 minStepLast, didLimit );
1253
1254
1255
1256
1257 currentStepSize = fTrueMinStep;
1259 if( (minStepLast != kInfinity) )
1260 {
1261 diffStep = (finalStep-minStepLast);
1262 if ( std::abs(diffStep) <= toleratedRelativeError * finalStep )
1263 {
1264 diffStep = 0.0;
1265 }
1266 currentStepSize += diffStep;
1267 }
1268 fCurrentStepSize[numNav] = currentStepSize;
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281 fLimitedStep[numNav] = didLimit;
1282 fLimitTruth[numNav] = limited = (didLimit !=
kDoNot );
1283 if( limited ) { ++noLimited; }
1284
1285#ifdef G4DEBUG_PATHFINDER
1286 G4bool StepError = (currentStepSize < 0)
1287 || ( (minStepLast != kInfinity) && (diffStep < 0) ) ;
1288 if( StepError || (fVerboseLevel > 2) )
1289 {
1290 G4String limitedString =
LimitedString( fLimitedStep[numNav] );
1291
1292 G4cout <<
" G4PathFinder::ComputeStep. Geometry " << numNav
1293 << " step= " << fCurrentStepSize[numNav]
1294 << " from final-step= " << finalStep
1295 << " fTrueMinStep= " << fTrueMinStep
1296 << " minStepLast= " << minStepLast
1297 << " limited = " << (fLimitTruth[numNav] ? "YES" : " NO")
1298 << " ";
1299 G4cout <<
" status = " << limitedString <<
" #= " << didLimit
1301
1302 if( StepError )
1303 {
1304 std::ostringstream message;
1305 message << "Incorrect calculation of step size for one navigator"
1307 << " currentStepSize = " << currentStepSize
1308 <<
", diffStep= " << diffStep <<
G4endl
1309 << "ERROR in computing step size for this navigator.";
1312 }
1313 }
1314#endif
1315 }
1316
1317 fNoGeometriesLimiting = noLimited;
1318 }
1319 else if ( (minStep == proposedStepLength)
1320 || (minStep == kInfinity)
1321 || ( std::abs(minStep-proposedStepLength)
1322 < toleratedRelativeError * proposedStepLength ) )
1323 {
1324
1325
1326
1327
1328
1329
1330
1331 currentStepSize = minStep;
1332 for( numNav=0; numNav < fNoActiveNavigators; ++numNav )
1333 {
1334 fCurrentStepSize[numNav] = minStep;
1335
1336 fLimitedStep[numNav] =
kDoNot;
1337 fLimitTruth[numNav] = false;
1338 }
1339 fNoGeometriesLimiting = 0;
1340 }
1341 else
1342 {
1343 std::ostringstream message;
1344 message <<
"Incorrect calculation of step size for one navigator." <<
G4endl
1345 << " currentStepSize = " << minStep << " is larger than "
1346 << " proposed StepSize = " << proposedStepLength << ".";
1349 }
1350
1351#ifdef G4DEBUG_PATHFINDER
1352 if( fVerboseLevel > 2 )
1353 {
1354 G4cout <<
" Exiting G4PathFinder::DoNextCurvedStep " <<
G4endl;
1356 }
1358#endif
1359
1360 return minStep;
1361}
virtual void SetChargeMomentumMass(G4ChargeState particleCharge, G4double MomentumXc, G4double MassXc2)=0
G4String & LimitedString(ELimited lim)