#automatic-differentiation #automatic #differentiation #derivative #gradient #auto

no-std autofloat

Pure Rust library for efficient automatic differentiation

4 stable releases

1.0.3 Dec 11, 2024
1.0.2 Dec 3, 2024
1.0.0 Nov 30, 2024

#352 in Rust patterns

Download history 236/week @ 2024-11-27 142/week @ 2024-12-04 156/week @ 2024-12-11 1/week @ 2024-12-18

426 downloads per month

MIT/Apache

100KB
3K SLoC

autofloat

license workflow codecov

autofloat is a pure Rust library, which implements efficient automatic differentiation in forward mode.

The library currently provides scalar datatypes to efficiently compute gradients.

Optionally you can use the library with the nalgebra feature to compute gradients and jacobians using the nalgebra library.

Usage

autofloatcan compute derivatives for single and multivariate functions. The library provides a float-like type AutoFloat to automatically compute the derivate while the target function is computed.

First, make sure that the function for which you want to compute a derivate can handle the AutoFloat type (either by generics or explicitly). Then simply instantiate the variables for which you want to compute the derivative and pass them into your target function, that's it!

Here's a simple example, which computes the gradient of a function wrt. two input variables. The function is implemented using generics and can be used with different floating point types.

use autofloat::AutoFloat2;
use num_traits::float::FloatCore;

// Define some target function for which we want to compute the derivative.
// This variant is generic in T, but you could also use the `AutoFloat` type directly.
fn quadratic_func<T>(x: T, y: T) -> T
where
    T: FloatCore,
{
    (x - T::one()) * (T::from(2).unwrap() * y - T::one())
}

fn main() {
    // Use AutoFloat2 because we use a 2-dimensional function and
    // we want a 2-dimensional gradient.
    // The first parameter determines the value of the variable.
    // The second prameter determines the index of the derivative
    // for this variable within the gradient vector.
    let x = AutoFloat2::variable(2.25, 0);
    let y = AutoFloat2::variable(-1.75, 1);

    let result = quadratic_func(x, y);

    println!(
        "result={} gradient_x={} gradient_y={}",
        result.x, result.dx[0], result.dx[1]
    );
}

See also the examples/directory for more examples on the different options how to define target functions.

License

This repository is licensed under either of

at your option.

Contributing

The easiest way to contribute is to log an issue for bugs or new features. I will then review and discuss the contents of ticket with you and eventually maybe implement it.

A faster way to get your features or fixes into autofloat is to file a pull request. There are just a few rules you should adhere:

  • Provide a meaningful PR description
  • All CI checks of the PR must succeed, otherwise it will not be merged
  • Have a code coverage of at least 80% of the lines you want to contribute

I will review the PR and we will discuss how and if your PR can be merged. If your PR might get large, feel free to log a ticket first and we can discuss the details before you implement everything.

Acknowledgements

This library started as a fork of autodiff by Egor Larionov, which again was forked from rust-ad by Igor Babuschkin. Some parts of the code base still contain signifcant parts of these upstream repositories.

Dependencies

~93–770KB
~15K SLoC