Simbody
|
An Integrator is an object that can simulate the behavior of a System through time. More...
#include <Integrator.h>
Public Types | |
enum | SuccessfulStepStatus { ReachedReportTime = 1, ReachedEventTrigger = 2, ReachedScheduledEvent = 3, TimeHasAdvanced = 4, ReachedStepLimit = 5, EndOfSimulation = 6, StartOfContinuousInterval = 7, InvalidSuccessfulStepStatus = -1 } |
When a step is successful, it will return an indication of what caused it to stop where it did. More... | |
enum | TerminationReason { ReachedFinalTime = 1, AnUnrecoverableErrorOccurred = 2, EventHandlerRequestedTermination = 3, InvalidTerminationReason = -1 } |
Once the simulation has ended, getTerminationReason() may be called to find out what caused it to end. More... | |
Public Member Functions | |
Integrator () | |
const char * | getMethodName () const |
Get the name of this integration method. | |
int | getMethodMinOrder () const |
Get the minimum order this Integrator may use. | |
int | getMethodMaxOrder () const |
Get the maximum order this Integrator may use. | |
bool | methodHasErrorControl () const |
Get whether this Integrator provides error control. | |
void | initialize (const State &state) |
Supply the integrator with a starting state. | |
void | reinitialize (Stage stage, bool shouldTerminate) |
After an event handler has made a discontinuous change to the Integrator's "advanced state", this method must be called to reinitialize the Integrator. | |
const State & | getState () const |
Return a State corresponding to the "current" time at the end of the last call to stepTo() or stepBy(). | |
Real | getTime () const |
Get the time of the current State. This is equivalent to calling getState().getTime(). | |
bool | isStateInterpolated () const |
Get whether getState() will return an interpolated state or just the same thing as getAdvancedState() does. | |
const State & | getAdvancedState () const |
Return the state representing the trajectory point to which the integrator has irreversibly advanced. | |
Real | getAdvancedTime () const |
Get the time of the advanced State. This is equivalent to calling getAdvancedState().getTime(). | |
State & | updAdvancedState () |
Get a non-const reference to the advanced state. | |
Real | getAccuracyInUse () const |
Get the accuracy which is being used for error control. | |
Real | getConstraintToleranceInUse () const |
Get the constraint tolerance which is being used for error control. | |
Real | getTimeScaleInUse () const |
Get the characteristic timescale of the System being integrated. | |
const Vector & | getStateWeightsInUse () const |
Get the vector of weights being used to normalize errors in the state variables. | |
const Vector & | getConstraintWeightsInUse () const |
Get the vector of weights being used to normalize constraint errors. | |
SuccessfulStepStatus | stepTo (Real reportTime, Real scheduledEventTime=Infinity) |
Integrate the System until something happens which requires outside processing, and return a status code describing what happened. | |
SuccessfulStepStatus | stepBy (Real interval, Real scheduledEventTime=Infinity) |
Integrate the System until something happens which requires outside processing, and return a status code describing what happened. | |
Vec2 | getEventWindow () const |
Get the window (tLow, tHigh] within which one or more events have been localized. | |
const Array_< EventId > & | getTriggeredEvents () const |
Get the IDs of all events which have been localized within the event window. | |
const Array_< Real > & | getEstimatedEventTimes () const |
Get the estimated times of all events which have been localized within the event window. | |
const Array_< Event::Trigger > & | getEventTransitionsSeen () const |
Get EventTriggers describing the events which have been localized within the event window. | |
bool | isSimulationOver () const |
Get whether the simulation has terminated. | |
TerminationReason | getTerminationReason () const |
Get the reason the simulation terminated. | |
void | resetAllStatistics () |
Reset all statistics to zero. | |
Real | getActualInitialStepSizeTaken () const |
Get the size of the first successful step after the last initialize() call. | |
Real | getPreviousStepSizeTaken () const |
Get the size of the most recent successful step. | |
Real | getPredictedNextStepSize () const |
Get the step size that will be attempted first on the next call to stepTo() or stepBy(). | |
int | getNumStepsAttempted () const |
Get the total number of steps that have been attempted (successfully or unsuccessfully) since the last call to resetAllStatistics(). | |
int | getNumStepsTaken () const |
Get the total number of steps that have been successfully taken since the last call to resetAllStatistics(). | |
int | getNumRealizations () const |
Get the total number of state realizations that have been performed since the last call to resetAllStatistics(). | |
int | getNumProjections () const |
Get the total number of times a state has been projected since the last call to resetAllStatistics(). | |
int | getNumErrorTestFailures () const |
Get the number of attempted steps that have failed due to the error being unacceptably high since the last call to resetAllStatistics(). | |
int | getNumConvergenceTestFailures () const |
Get the number of attempted steps that failed due to non-convergence of internal step iterations. | |
int | getNumRealizationFailures () const |
Get the number of attempted steps that have failed due to an error when realizing the state since the last call to resetAllStatistics(). | |
int | getNumProjectionFailures () const |
Get the number of attempted steps that have failed due to an error when projecting the state since the last call to resetAllStatistics(). | |
int | getNumConvergentIterations () const |
For iterative methods, get the number of internal step iterations in steps that led to convergence (not necessarily successful steps). | |
int | getNumDivergentIterations () const |
For iterative methods, get the number of internal step iterations in steps that did not lead to convergence. | |
int | getNumIterations () const |
For iterative methods, this is the total number of internal step iterations taken regardless of whether those iterations led to convergence or to successful steps. | |
void | setFinalTime (Real tFinal) |
Set the time at which the simulation should end. | |
void | setInitialStepSize (Real hinit) |
Set the initial step size that should be attempted. | |
void | setMinimumStepSize (Real hmin) |
Set the minimum step size that should ever be used. | |
void | setMaximumStepSize (Real hmax) |
Set the maximum step size that should ever be used. | |
void | setFixedStepSize (Real stepSize) |
Set the integrator to use a single fixed step size for all steps. | |
void | setAccuracy (Real accuracy) |
Set the overall accuracy that should be used for integration. | |
void | setRelativeTolerance (Real relTol) |
Set the relative error tolerance to be used for integration. | |
void | setAbsoluteTolerance (Real absTol) |
Set the absolute error tolerance to be used for integration. | |
void | setConstraintTolerance (Real consTol) |
Set the tolerance within which constraints must be satisfied. | |
void | setInternalStepLimit (int nSteps) |
Set the maximum number of steps that may be taken within a single call to stepTo() or stepBy(). | |
void | setReturnEveryInternalStep (bool shouldReturn) |
Set whether the Integrator should return from stepTo() or stepBy() after every internal step, even if no event has occurred and the report time has not been reached. | |
void | setProjectEveryStep (bool forceProject) |
Set whether the system should be projected back to the constraint manifold after every step. | |
void | setAllowInterpolation (bool shouldInterpolate) |
Set whether the Integrator is permitted to return interpolated states for reporting purposes which may be less accurate than the "real" states that form the trajectory. | |
void | setProjectInterpolatedStates (bool shouldProject) |
Set whether interpolated states should be projected back to the constraint manifold after interpolation is performed. | |
Static Public Member Functions | |
static String | successfulStepStatusString (SuccessfulStepStatus) |
Get a human readable description of the reason a step returned. | |
Protected Member Functions | |
const IntegratorRep & | getRep () const |
IntegratorRep & | updRep () |
Protected Attributes | |
IntegratorRep * | rep |
Friends | |
class | IntegratorRep |
An Integrator is an object that can simulate the behavior of a System through time.
This is an abstract class. Subclasses implement a variety of different integration methods, which vary in their speed, accuracy, stability, and various other properties.
An Integrator is most often used in combination with a TimeStepper. The TimeStepper automates much of the work of using an Integrator: invoking it repeatedly to advance time, calling event handlers, reintializing the Integrator as necessary, and so on. A typical use of an Integrator generally resembles the following:
// Instantiate an Integrator subclass which is appropriate for your problem. VerletIntegrator integ(system); // Set configuration options on the Integrator. integ.setAccuracy(1e-3); // Create a TimeStepper and use it to run the simulation. TimeStepper stepper(system, integ); stepper.initialize(initialState); stepper.stepTo(finalTime);
Given a continuous system of differential equations for state variables y, and optionally a manifold (set of algebraic equations) on which the solution must lie, an Integrator object will advance that system through time. If the full system is continuous, this is sufficient. However, most interesting systems consist of both continuous and discrete equations, in which case the overall time advancement is handled by a TimeStepper object which will use an Integrator as an "inner loop" for advancing the system across continuous intervals between discrete updates. In that case, in addition to continuous state variables y there will be a set of discrete variables d which are held constant during an integration interval.
The continuous part of the system is an ODE-on-a-manifold system suitable for solution via coordinate projection[1], structured like this: (1) y' = f[d](t,y) differential equations (2) 0 = c[d](t,y) algebraic equations (manifold is c=0) (3) e = e[d](t,y) event triggers (watch for zero crossings) with initial conditions t0,y0 such that c=0. The [d] is a reminder that the overall system is dependent on discrete variables d as well as y, but d cannot change during a continuous interval.
By "ODE on a manifold" we mean that the ODE (1) automatically satisfies the condition that IF c==0, THEN c'=0, where c'=partial(c)/partial(t) + [partial(c)/partial(y)]*y'. This is a less stringent condition than an ODE with "first integral" invariant, in which c'=0 regardless of whether c=0.
[1] Hairer, Lubich, Wanner, "Geometric Numerical Integration: Structure-Preserving Algorithms for Ordinary Differential Equations", 2nd ed., section IV.4, pg 109ff, Springer, 2006.
The discrete variables d are updated by the time stepper upon occurence of specific events, which terminate a continuous interval. The Integrator detects these using a set of scalar-valued event trigger functions shown as equation (3) above. An event trigger function for a particular event must be designed so that it has a zero crossing when the event occurs. The integrator can thus watch for sign changes in event triggers and terminate the current step when a zero crossing occurs, notifying the system and giving it a chance to handle the event; that is, update its state variables discontinuously.
The zero crossings of continuous event trigger functions will be isolated quickly; discontinuous ones have to be "binary chopped" which is more expensive but they will still work.
We are given a set of weights W for the y's, and a set of tolerances T for the constraint errors. Given an accuracy specification (like 0.1%), the integrators here are expected to solve for y(t) such that the local error |W*yerr|_RMS <= accuracy, and |T*c(t,y)|_RMS <= accuracy at all times.
TODO: isolation tolerances for witnesses; dealing with simultaneity.
When a step is successful, it will return an indication of what caused it to stop where it did.
When unsuccessful it will throw an exception so you won't see any return value.
When return of control is due ONLY to reaching a report time, (status is ReachedReportTime) the integrator's getState() method may return an interpolated value at an earlier time than its getAdvancedState() method would return. For the other returns, and whenever the report time coincides with the end of an internal step, getState() and getAdvancedState() will be identical.
Note: we ensure algorithmically that no report time, scheduled time, or final time t can occur *within* an event window, that is, we will never have t_low < t < t_high for any interesting t. Further, t_report, t_scheduled and t_final can coincide with t_high but only t_report can be at t_low. The interior of t_low:t_high is a "no man's land" where we don't understand the solution, so must be avoided.
ReachedReportTime |
stopped only to report; might be interpolated |
ReachedEventTrigger |
localized an event; this is the *before* state (interpolated) |
ReachedScheduledEvent |
reached the limit provided in stepTo() (scheduled event) |
TimeHasAdvanced |
user requested control whenever an internal step is successful |
ReachedStepLimit |
took a lot of internal steps but didn't return control yet |
EndOfSimulation |
termination; don't call again |
StartOfContinuousInterval |
the beginning of a continuous interval: either the start of the simulation, or t_high after an event handler has modified the state. |
InvalidSuccessfulStepStatus |
Once the simulation has ended, getTerminationReason() may be called to find out what caused it to end.
ReachedFinalTime |
The simulation reached the time specified by setFinalTime(). |
AnUnrecoverableErrorOccurred |
An error occurred which the Integrator was unable to handle. |
EventHandlerRequestedTermination |
An event handler function requested that the simulation terminate immediately. |
InvalidTerminationReason |
This will be returned if getTerminationReason() is called before the simulation has ended. |
SimTK::Integrator::Integrator | ( | ) | [inline] |
const char* SimTK::Integrator::getMethodName | ( | ) | const |
Get the name of this integration method.
int SimTK::Integrator::getMethodMinOrder | ( | ) | const |
Get the minimum order this Integrator may use.
int SimTK::Integrator::getMethodMaxOrder | ( | ) | const |
Get the maximum order this Integrator may use.
bool SimTK::Integrator::methodHasErrorControl | ( | ) | const |
Get whether this Integrator provides error control.
An error controlled Integrator will dynamically adjust its step size to maintain the level of accuracy specified with setAccuracy(). An Integrator which does not provide error control cannot do this, and will usually ignore the value specified with setAccuracy().
void SimTK::Integrator::initialize | ( | const State & | state | ) |
void SimTK::Integrator::reinitialize | ( | Stage | stage, |
bool | shouldTerminate | ||
) |
After an event handler has made a discontinuous change to the Integrator's "advanced state", this method must be called to reinitialize the Integrator.
Event handlers can do varying amounts of damage and some events will require no reinitialization, or minimal reinitialization, depending on details of the particular integration method. So after the handler has mangled our State, we tell the Integrator the lowest Stage which was changed and allow the Integrator to figure out how much reinitialization to do.
If "shouldTerminate" is passed in true, the Integrator will wrap things up and report that the end of the simulation has been reached.
const State& SimTK::Integrator::getState | ( | ) | const |
Return a State corresponding to the "current" time at the end of the last call to stepTo() or stepBy().
This may be an interpolated value earlier than getAdvancedState().
Real SimTK::Integrator::getTime | ( | ) | const [inline] |
Get the time of the current State. This is equivalent to calling getState().getTime().
bool SimTK::Integrator::isStateInterpolated | ( | ) | const |
Get whether getState() will return an interpolated state or just the same thing as getAdvancedState() does.
In most cases, you should not have reason to care whether the state is interpolated or not.
const State& SimTK::Integrator::getAdvancedState | ( | ) | const |
Return the state representing the trajectory point to which the integrator has irreversibly advanced.
This may be later than the state return by getState().
Real SimTK::Integrator::getAdvancedTime | ( | ) | const [inline] |
Get the time of the advanced State. This is equivalent to calling getAdvancedState().getTime().
State& SimTK::Integrator::updAdvancedState | ( | ) |
Get a non-const reference to the advanced state.
Real SimTK::Integrator::getAccuracyInUse | ( | ) | const |
Get the accuracy which is being used for error control.
Usually this is the same value that was specified to setAccuracy().
Real SimTK::Integrator::getConstraintToleranceInUse | ( | ) | const |
Get the constraint tolerance which is being used for error control.
Usually this is the same value that was specified to setConstraintTolerance().
Real SimTK::Integrator::getTimeScaleInUse | ( | ) | const |
const Vector& SimTK::Integrator::getStateWeightsInUse | ( | ) | const |
Get the vector of weights being used to normalize errors in the state variables.
This is calculated by calling calcYUnitWeights() on the System.
const Vector& SimTK::Integrator::getConstraintWeightsInUse | ( | ) | const |
Get the vector of weights being used to normalize constraint errors.
This is calculated by calling calcYErrUnitTolerances() on the System.
static String SimTK::Integrator::successfulStepStatusString | ( | SuccessfulStepStatus | ) | [static] |
Get a human readable description of the reason a step returned.
SuccessfulStepStatus SimTK::Integrator::stepTo | ( | Real | reportTime, |
Real | scheduledEventTime = Infinity |
||
) |
Integrate the System until something happens which requires outside processing, and return a status code describing what happened.
reportTime | the time of the next scheduled report |
scheduledEventTime | the time of the next scheduled event |
SuccessfulStepStatus SimTK::Integrator::stepBy | ( | Real | interval, |
Real | scheduledEventTime = Infinity |
||
) |
Vec2 SimTK::Integrator::getEventWindow | ( | ) | const |
const Array_<Real>& SimTK::Integrator::getEstimatedEventTimes | ( | ) | const |
const Array_<Event::Trigger>& SimTK::Integrator::getEventTransitionsSeen | ( | ) | const |
bool SimTK::Integrator::isSimulationOver | ( | ) | const |
TerminationReason SimTK::Integrator::getTerminationReason | ( | ) | const |
Get the reason the simulation terminated.
This should only be called if isSimulationOver() returns true.
void SimTK::Integrator::resetAllStatistics | ( | ) |
Reset all statistics to zero.
Real SimTK::Integrator::getActualInitialStepSizeTaken | ( | ) | const |
Get the size of the first successful step after the last initialize() call.
Real SimTK::Integrator::getPreviousStepSizeTaken | ( | ) | const |
Get the size of the most recent successful step.
Real SimTK::Integrator::getPredictedNextStepSize | ( | ) | const |
int SimTK::Integrator::getNumStepsAttempted | ( | ) | const |
Get the total number of steps that have been attempted (successfully or unsuccessfully) since the last call to resetAllStatistics().
int SimTK::Integrator::getNumStepsTaken | ( | ) | const |
Get the total number of steps that have been successfully taken since the last call to resetAllStatistics().
int SimTK::Integrator::getNumRealizations | ( | ) | const |
Get the total number of state realizations that have been performed since the last call to resetAllStatistics().
int SimTK::Integrator::getNumProjections | ( | ) | const |
Get the total number of times a state has been projected since the last call to resetAllStatistics().
int SimTK::Integrator::getNumErrorTestFailures | ( | ) | const |
Get the number of attempted steps that have failed due to the error being unacceptably high since the last call to resetAllStatistics().
int SimTK::Integrator::getNumConvergenceTestFailures | ( | ) | const |
Get the number of attempted steps that failed due to non-convergence of internal step iterations.
This is most common with iterative methods but can occur if for some reason a step can't be completed. It is reset to zero by resetAllStatistics.
int SimTK::Integrator::getNumRealizationFailures | ( | ) | const |
Get the number of attempted steps that have failed due to an error when realizing the state since the last call to resetAllStatistics().
int SimTK::Integrator::getNumProjectionFailures | ( | ) | const |
Get the number of attempted steps that have failed due to an error when projecting the state since the last call to resetAllStatistics().
int SimTK::Integrator::getNumConvergentIterations | ( | ) | const |
For iterative methods, get the number of internal step iterations in steps that led to convergence (not necessarily successful steps).
Reset to zero by resetAllStatistics().
int SimTK::Integrator::getNumDivergentIterations | ( | ) | const |
For iterative methods, get the number of internal step iterations in steps that did not lead to convergence.
Reset to zero by resetAllStatistics().
int SimTK::Integrator::getNumIterations | ( | ) | const |
For iterative methods, this is the total number of internal step iterations taken regardless of whether those iterations led to convergence or to successful steps.
This is the sum of the number of convergent and divergent iterations which are available separately.
void SimTK::Integrator::setFinalTime | ( | Real | tFinal | ) |
Set the time at which the simulation should end.
The default is infinity. Some integrators may not support this option.
void SimTK::Integrator::setInitialStepSize | ( | Real | hinit | ) |
Set the initial step size that should be attempted.
The default depends on the integration method. Some integrators may not support this option.
void SimTK::Integrator::setMinimumStepSize | ( | Real | hmin | ) |
Set the minimum step size that should ever be used.
The default depends on the integration method. Some integrators may not support this option.
void SimTK::Integrator::setMaximumStepSize | ( | Real | hmax | ) |
Set the maximum step size that should ever be used.
The default depends on the integration method. Some integrators may not support this option.
void SimTK::Integrator::setFixedStepSize | ( | Real | stepSize | ) |
Set the integrator to use a single fixed step size for all steps.
This is exactly equivalent to calling setInitialStepSize(), setMinimumStepSize(), and setMaximumStepSize(), passing the same value to each one. This will therefore not work correctly if the integrator does not support minimum and/or maximum step sizes.
void SimTK::Integrator::setAccuracy | ( | Real | accuracy | ) |
Set the overall accuracy that should be used for integration.
If the Integrator does not support error control (methodHasErrorControl() returns false), this may have no effect.
void SimTK::Integrator::setRelativeTolerance | ( | Real | relTol | ) |
Set the relative error tolerance to be used for integration.
Some Integrators do not allow relative and absolute tolerances to be specified independently, in which case this will have no effect. Even for Integrators which do support it, it is usually sufficient to call only setAccuracy(), and allow the Integrator to select relative and absolute tolerances accordingly.
void SimTK::Integrator::setAbsoluteTolerance | ( | Real | absTol | ) |
Set the absolute error tolerance to be used for integration.
Some Integrators do not allow relative and absolute tolerances to be specified independently, in which case this will have no effect. Even for Integrators which do support it, it is usually sufficient to call only setAccuracy(), and allow the Integrator to select relative and absolute tolerances accordingly.
void SimTK::Integrator::setConstraintTolerance | ( | Real | consTol | ) |
Set the tolerance within which constraints must be satisfied.
void SimTK::Integrator::setInternalStepLimit | ( | int | nSteps | ) |
void SimTK::Integrator::setReturnEveryInternalStep | ( | bool | shouldReturn | ) |
Set whether the Integrator should return from stepTo() or stepBy() after every internal step, even if no event has occurred and the report time has not been reached.
The default is false.
void SimTK::Integrator::setProjectEveryStep | ( | bool | forceProject | ) |
Set whether the system should be projected back to the constraint manifold after every step.
If this is false, projection will only be performed when the constraint error exceeds the allowed tolerance.
void SimTK::Integrator::setAllowInterpolation | ( | bool | shouldInterpolate | ) |
Set whether the Integrator is permitted to return interpolated states for reporting purposes which may be less accurate than the "real" states that form the trajectory.
Setting this to false may significantly affect performance, since the Integrator will be forced to decrease its step size at every scheduled reporting time.
This option is generally only meaningful if interpolated states are less accurate than other states on the trajectory. If an Integrator can produce interpolated states that have the same accuracy as the rest of the trajectory, it may ignore this option.
void SimTK::Integrator::setProjectInterpolatedStates | ( | bool | shouldProject | ) |
Set whether interpolated states should be projected back to the constraint manifold after interpolation is performed.
const IntegratorRep& SimTK::Integrator::getRep | ( | ) | const [inline, protected] |
IntegratorRep& SimTK::Integrator::updRep | ( | ) | [inline, protected] |
friend class IntegratorRep [friend] |
IntegratorRep* SimTK::Integrator::rep [protected] |