Brownian controls¤
SDEs are simulated using a Brownian motion as a control. (See the neural SDE example.)
diffrax.AbstractBrownianPath
diffrax.AbstractBrownianPath (AbstractPath)
¤
Abstract base class for all Brownian paths.
evaluate(self, t0: Union[float, int], t1: Union[float, int] = None, left: bool = True, use_levy: bool = False) -> ~_Control
abstractmethod
¤
Samples a Brownian increment \(w(t_1) - w(t_0)\).
Each increment has distribution \(\mathcal{N}(0, t_1 - t_0)\).
Arguments:
t0
: Any point in \([t_0, t_1]\) to evaluate the path at.t1
: If passed, then the increment fromt1
tot0
is evaluated instead.left
: Ignored. (This determines whether to treat the path as left-continuous or right-continuous at any jump points, but Brownian motion has no jump points.)use_levy
: If True, the return type will be aLevyVal
, which contains PyTrees of Brownian increments and their Lévy areas.
Returns:
If t1
is not passed:
The value of the Brownian motion at t0
.
If t1
is passed:
The increment of the Brownian motion between t0
and t1
.
¤
¤
diffrax.UnsafeBrownianPath (AbstractBrownianPath)
¤
Brownian simulation that is only suitable for certain cases.
This is a very quick way to simulate Brownian motion, but can only be used when all of the following are true:
-
You are using a fixed step size controller. (Not an adaptive one.)
-
You do not need to backpropagate through the differential equation.
-
You do not need deterministic solutions with respect to
key
. (This implementation will produce different results based on fluctuations in floating-point arithmetic.)
Internally this operates by just sampling a fresh normal random variable over every interval, ignoring the correlation between samples exhibited in true Brownian motion. Hence the restrictions above. (They describe the general case for which the correlation structure isn't needed.)
Lévy Area
Can be initialised with levy_area
set to diffrax.BrownianIncrement
, or
diffrax.SpaceTimeLevyArea
. If levy_area=diffrax.SpaceTimeLevyArea
, then it
also computes space-time Lévy area H
. This is an additional source of
randomness required for certain stochastic Runge--Kutta solvers; see
diffrax.AbstractSRK
for more information.
An error will be thrown during tracing if Lévy area is required but is not available.
The choice here will impact the Brownian path, so even with the same key, the
trajectory will be different depending on the value of levy_area
.
__init__(self, shape: Union[tuple[int, ...], PyTree[jax._src.api.ShapeDtypeStruct]], key: PRNGKeyArray, levy_area: type[Union[BrownianIncrement, SpaceTimeLevyArea, SpaceTimeTimeLevyArea]] = <class 'diffrax.BrownianIncrement'>)
¤
Arguments:
shape
: Should be a PyTree ofjax.ShapeDtypeStruct
s, representing the shape, dtype, and PyTree structure of the output. For simplicity,shape
can also just be a tuple of integers, describing the shape of a single JAX array. In that case the dtype is chosen to be the default floating-point dtype.key
: A random key.levy_area
: Whether to additionally generate Lévy area. This is required by some SDE solvers.
diffrax.VirtualBrownianTree (AbstractBrownianPath)
¤
Brownian simulation that discretises the interval [t0, t1]
to tolerance tol
.
Lévy Area
The parameter levy_area
can be set to one of:
diffrax.BrownianIncrement
(default, generates the increment of W)diffrax.SpaceTimeLevyArea
(generates W and the space-time Lévy area H)diffrax.SpaceTimeTimeLevyArea
(generates W, H and the space-time-time Lévy area K)
The choice of levy_area
will impact the Brownian path, so even with the same
key, the trajectory will be different depending on the value of levy_area
.
Reference
Virtual Brownian trees were proposed in
@article{li2020scalable,
title={Scalable gradients for stochastic differential equations},
author={Li, Xuechen and Wong, Ting-Kam Leonard and Chen, Ricky T. Q. and
Duvenaud, David},
journal={International Conference on Artificial Intelligence and Statistics},
year={2020}
}
The implementation here is an improvement on the above, in that it additionally simulates space-time and space-time-time Lévy areas, and exactly matches the distribution of the Brownian motion and its Lévy areas at all query times. This is due to the paper
@misc{jelinčič2024singleseed,
title={Single-seed generation of Brownian paths and integrals
for adaptive and high order SDE solvers},
author={Andraž Jelinčič and James Foster and Patrick Kidger},
year={2024},
eprint={2405.06464},
archivePrefix={arXiv},
primaryClass={math.NA}
}
and Theorem 6.1.6 of
@phdthesis{foster2020a,
publisher = {University of Oxford},
school = {University of Oxford},
title = {Numerical approximations for stochastic differential equations},
author = {Foster, James M.},
year = {2020}
}
__init__(self, t0: Union[float, int], t1: Union[float, int], tol: Union[float, int], shape: Union[tuple[int, ...], PyTree[jax._src.api.ShapeDtypeStruct]], key: PRNGKeyArray, levy_area: type[Union[BrownianIncrement, SpaceTimeLevyArea, SpaceTimeTimeLevyArea]] = <class 'diffrax.BrownianIncrement'>)
¤
Initialize self. See help(type(self)) for accurate signature.
Lévy areas¤
Brownian controls can return certain types of Lévy areas. These are iterated integrals
of the Brownian motion, and are used by some SDE solvers. When a solver requires a
Lévy area, it will have a minimal_levy_area
attribute, which will always return an
abstract Lévy area type, and it can accept any subclass of that type.
The inheritance hierarchy is as follows:
AbstractBrownianIncrement
│ └── BrownianIncrement
└── AbstractSpaceTimeLevyArea
│ └── SpaceTimeLevyArea
└── AbstractSpaceTimeTimeLevyArea
└── SpaceTimeTimeLevyArea
solver.minimal_levy_area
returns an AbstractSpaceTimeLevyArea
, then
the Brownian motion (which is either an UnsafeBrownianPath
or
a VirtualBrownianTree
) should be initialized with levy_area=SpaceTimeLevyArea
or
levy_area=SpaceTimeTimeLevyArea
. Note that for the Brownian motion,
a concrete class must be used, not its abstract parent.
diffrax.AbstractBrownianIncrement
¤
Abstract base class for all Brownian increments.
diffrax.BrownianIncrement (AbstractBrownianIncrement)
¤
Pytree containing the dt
time increment and W
the Brownian motion.
diffrax.AbstractSpaceTimeLevyArea (AbstractBrownianIncrement)
¤
Abstract base class for all Space Time Levy Areas.
diffrax.SpaceTimeLevyArea (AbstractSpaceTimeLevyArea)
¤
Pytree containing the dt
time increment, W
the Brownian motion, and H
the Space Time Levy Area.
diffrax.AbstractSpaceTimeTimeLevyArea (AbstractSpaceTimeLevyArea)
¤
Abstract base class for all Space Time Time Levy Areas.
diffrax.SpaceTimeTimeLevyArea (AbstractSpaceTimeTimeLevyArea)
¤
Pytree containing the dt
time increment, W
the Brownian motion, H
the Space Time Levy Area, and K
the Space Time Time Levy Area.