Derivative access patterns for curves#16503
Conversation
|
Re: VectorSpace: HasTangent |
Yeah, that's a good point. In that case, I guess I don't really care if the tangents are meaningful (or it's just a downstream problem). |
|
I guess I should finish my review! |
| } | ||
|
|
||
| #[test] | ||
| fn curve_reparam_curve() { |
There was a problem hiding this comment.
Perhaps a good testcase could be constructing a cubic hermite spline which equal the identity, and verifying that the derivatives are unchanged under that parametrisation.
|
Typos CI is complaining at you @mweatherley :) Fix it and I'll merge? |
|
But |
|
It's not a typo and it's also explicitly in our |
|
Ah, I bet the branch was behind. Updating and trying again. |
|
@BenjaminBrienen @homersimpsons do either of you have advice / ideas on the typos CI failure here? It's a false positive and should be ignored, but that doesn't seem to work correctly. |
Typos want "reparameterization", to me we should just add "reparameterization" as an exception in typos.toml next to "reparametrize". |
Head branch was pushed to by a user without write access
|
I had forgotten to update the local branch when I ran |
Head branch was pushed to by a user without write access
# Objective
- For curves that also include derivatives, make accessing derivative
information via the `Curve` API ergonomic: that is, provide access to a
curve that also samples derivative information.
- Implement this functionality for cubic spline curves provided by
`bevy_math`.
Ultimately, this is to serve the purpose of doing more geometric
operations on curves, like reparametrization by arclength and the
construction of moving frames.
## Solution
This has several parts, some of which may seem redundant. However, care
has been put into this to satisfy the following constraints:
- Accessing a `Curve` that samples derivative information should be not
just possible but easy and non-error-prone. For example, given a
differentiable `Curve<Vec2>`, one should be able to access something
like a `Curve<(Vec2, Vec2)>` ergonomically, and not just sample the
derivatives piecemeal from point to point.
- Derivative access should not step on the toes of ordinary curve usage.
In particular, in the above scenario, we want to avoid simply making the
same curve both a `Curve<Vec2>` and a `Curve<(Vec2, Vec2)>` because this
requires manual disambiguation when the API is used.
- Derivative access must work gracefully in both owned and borrowed
contexts.
### `HasTangent`
We introduce a trait `HasTangent` that provides an associated `Tangent`
type for types that have tangent spaces:
```rust
pub trait HasTangent {
/// The tangent type.
type Tangent: VectorSpace;
}
```
(Mathematically speaking, it would be more precise to say that these are
types that represent spaces which are canonically
[parallelized](https://en.wikipedia.org/wiki/Parallelizable_manifold). )
The idea here is that a point moving through a `HasTangent` type may
have a derivative valued in the associated `Tangent` type at each time
in its journey. We reify this with a `WithDerivative<T>` type that uses
`HasTangent` to include derivative information:
```rust
pub struct WithDerivative<T>
where
T: HasTangent,
{
/// The underlying value.
pub value: T,
/// The derivative at `value`.
pub derivative: T::Tangent,
}
```
And we can play the same game with second derivatives as well, since
every `VectorSpace` type is `HasTangent` where `Tangent` is itself (we
may want to be more restrictive with this in practice, but this holds
mathematically).
```rust
pub struct WithTwoDerivatives<T>
where
T: HasTangent,
{
/// The underlying value.
pub value: T,
/// The derivative at `value`.
pub derivative: T::Tangent,
/// The second derivative at `value`.
pub second_derivative: <T::Tangent as HasTangent>::Tangent,
}
```
In this PR, `HasTangent` is only implemented for `VectorSpace` types,
but it would be valuable to have this implementation for types like
`Rot2` and `Quat` as well. We could also do it for the isometry types
and, potentially, transforms as well. (This is in decreasing order of
value in my opinion.)
### `CurveWithDerivative`
This is a trait for a `Curve<T>` which allows the construction of a
`Curve<WithDerivative<T>>` when derivative information is known
intrinsically. It looks like this:
```rust
/// Trait for curves that have a well-defined notion of derivative, allowing for
/// derivatives to be extracted along with values.
pub trait CurveWithDerivative<T>
where
T: HasTangent,
{
/// This curve, but with its first derivative included in sampling.
fn with_derivative(self) -> impl Curve<WithDerivative<T>>;
}
```
The idea here is to provide patterns like this:
```rust
let value_and_derivative = my_curve.with_derivative().sample_clamped(t);
```
One of the main points here is that `Curve<WithDerivative<T>>` is useful
as an output because it can be used durably. For example, in a dynamic
context, something that needs curves with derivatives can store
something like a `Box<dyn Curve<WithDerivative<T>>>`. Note that
`CurveWithDerivative` is not dyn-compatible.
### `SampleDerivative`
Many curves "know" how to sample their derivatives instrinsically, but
implementing `CurveWithDerivative` as given would be onerous or require
an annoying amount of boilerplate. There are also hurdles to overcome
that involve references to curves: for the `Curve` API, the expectation
is that curve transformations like `with_derivative` take things by
value, with the contract that they can still be used by reference
through deref-magic by including `by_ref` in a method chain.
These problems are solved simultaneously by a trait `SampleDerivative`
which, when implemented, automatically derives `CurveWithDerivative` for
a type and all types that dereference to it. It just looks like this:
```rust
pub trait SampleDerivative<T>: Curve<T>
where
T: HasTangent,
{
fn sample_with_derivative_unchecked(&self, t: f32) -> WithDerivative<T>;
// ... other sampling variants as default methods
}
```
The point is that the output of `with_derivative` is a
`Curve<WithDerivative<T>>` that uses the `SampleDerivative`
implementation. On a `SampleDerivative` type, you can also just call
`my_curve.sample_with_derivative(t)` instead of something like
`my_curve.by_ref().with_derivative().sample(t)`, which is more verbose
and less accessible.
In practice, `CurveWithDerivative<T>` is actually a "sealed" extension
trait of `SampleDerivative<T>`.
## Adaptors
`SampleDerivative` has automatic implementations on all curve adaptors
except for `FunctionCurve`, `MapCurve`, and `ReparamCurve` (because we
do not have a notion of differentiable Rust functions).
For example, `CurveReparamCurve` (the reparametrization of a curve by
another curve) can compute derivatives using the chain rule in the case
both its constituents have them.
## Testing
Tests for derivatives on the curve adaptors are included.
---
## Showcase
This development allows derivative information to be included with and
extracted from curves using the `Curve` API.
```rust
let points = [
vec2(-1.0, -20.0),
vec2(3.0, 2.0),
vec2(5.0, 3.0),
vec2(9.0, 8.0),
];
// A cubic spline curve that goes through `points`.
let curve = CubicCardinalSpline::new(0.3, points).to_curve().unwrap();
// Calling `with_derivative` causes derivative output to be included in the output of the curve API.
let curve_with_derivative = curve.with_derivative();
// A `Curve<f32>` that outputs the speed of the original.
let speed_curve = curve_with_derivative.map(|x| x.derivative.norm());
```
---
## Questions
- ~~Maybe we should seal `WithDerivative` or make it require
`SampleDerivative` (i.e. make it unimplementable except through
`SampleDerivative`).~~ I decided this is a good idea.
- ~~Unclear whether `VectorSpace: HasTangent` blanket implementation is
really appropriate. For colors, for example, I'm not sure that the
derivative values can really be interpreted as a color. In any case, it
should still remain the case that `VectorSpace` types are `HasTangent`
and that `HasTangent::Tangent: HasTangent`.~~ I think this is fine.
- Infinity bikeshed on names of traits and things.
## Future
- Faster implementations of `SampleDerivative` for cubic spline curves.
- Improve ergonomics for accessing only derivatives (and other kinds of
transformations on derivative curves).
- Implement `HasTangent` for:
- `Rot2`/`Quat`
- `Isometry` types
- `Transform`, maybe
- Implement derivatives for easing curves.
- Marker traits for continuous/differentiable curves. (It's actually
unclear to me how much value this has in practice, but we have discussed
it in the past.)
---------
Co-authored-by: Alice Cecile <alice.i.cecile@gmail.com>
|
Thank you to everyone involved with the authoring or reviewing of this PR! This work is relatively important and needs release notes! Head over to bevyengine/bevy-website#1970 if you'd like to help out. |
Objective
CurveAPI ergonomic: that is, provide access to a curve that also samples derivative information.bevy_math.Ultimately, this is to serve the purpose of doing more geometric operations on curves, like reparametrization by arclength and the construction of moving frames.
Solution
This has several parts, some of which may seem redundant. However, care has been put into this to satisfy the following constraints:
Curvethat samples derivative information should be not just possible but easy and non-error-prone. For example, given a differentiableCurve<Vec2>, one should be able to access something like aCurve<(Vec2, Vec2)>ergonomically, and not just sample the derivatives piecemeal from point to point.Curve<Vec2>and aCurve<(Vec2, Vec2)>because this requires manual disambiguation when the API is used.HasTangentWe introduce a trait
HasTangentthat provides an associatedTangenttype for types that have tangent spaces:(Mathematically speaking, it would be more precise to say that these are types that represent spaces which are canonically parallelized. )
The idea here is that a point moving through a
HasTangenttype may have a derivative valued in the associatedTangenttype at each time in its journey. We reify this with aWithDerivative<T>type that usesHasTangentto include derivative information:And we can play the same game with second derivatives as well, since every
VectorSpacetype isHasTangentwhereTangentis itself (we may want to be more restrictive with this in practice, but this holds mathematically).In this PR,
HasTangentis only implemented forVectorSpacetypes, but it would be valuable to have this implementation for types likeRot2andQuatas well. We could also do it for the isometry types and, potentially, transforms as well. (This is in decreasing order of value in my opinion.)CurveWithDerivativeThis is a trait for a
Curve<T>which allows the construction of aCurve<WithDerivative<T>>when derivative information is known intrinsically. It looks like this:The idea here is to provide patterns like this:
One of the main points here is that
Curve<WithDerivative<T>>is useful as an output because it can be used durably. For example, in a dynamic context, something that needs curves with derivatives can store something like aBox<dyn Curve<WithDerivative<T>>>. Note thatCurveWithDerivativeis not dyn-compatible.SampleDerivativeMany curves "know" how to sample their derivatives instrinsically, but implementing
CurveWithDerivativeas given would be onerous or require an annoying amount of boilerplate. There are also hurdles to overcome that involve references to curves: for theCurveAPI, the expectation is that curve transformations likewith_derivativetake things by value, with the contract that they can still be used by reference through deref-magic by includingby_refin a method chain.These problems are solved simultaneously by a trait
SampleDerivativewhich, when implemented, automatically derivesCurveWithDerivativefor a type and all types that dereference to it. It just looks like this:The point is that the output of
with_derivativeis aCurve<WithDerivative<T>>that uses theSampleDerivativeimplementation. On aSampleDerivativetype, you can also just callmy_curve.sample_with_derivative(t)instead of something likemy_curve.by_ref().with_derivative().sample(t), which is more verbose and less accessible.In practice,
CurveWithDerivative<T>is actually a "sealed" extension trait ofSampleDerivative<T>.Adaptors
SampleDerivativehas automatic implementations on all curve adaptors except forFunctionCurve,MapCurve, andReparamCurve(because we do not have a notion of differentiable Rust functions).For example,
CurveReparamCurve(the reparametrization of a curve by another curve) can compute derivatives using the chain rule in the case both its constituents have them.Testing
Tests for derivatives on the curve adaptors are included.
Showcase
This development allows derivative information to be included with and extracted from curves using the
CurveAPI.Questions
Maybe we should sealI decided this is a good idea.WithDerivativeor make it requireSampleDerivative(i.e. make it unimplementable except throughSampleDerivative).Unclear whetherI think this is fine.VectorSpace: HasTangentblanket implementation is really appropriate. For colors, for example, I'm not sure that the derivative values can really be interpreted as a color. In any case, it should still remain the case thatVectorSpacetypes areHasTangentand thatHasTangent::Tangent: HasTangent.Future
SampleDerivativefor cubic spline curves.HasTangentfor:Rot2/QuatIsometrytypesTransform, maybe