blob: e422a8f48c52c28513bc681ed2ada106dfacdfd1 [file]
// Benchmarks for Eigen Spline module.
// Tests fitting, evaluation, and derivative computation.
#include <benchmark/benchmark.h>
#include <Eigen/Core>
#include <unsupported/Eigen/Splines>
using namespace Eigen;
typedef double Scalar;
// --- Spline fitting (interpolation) ---
template <int Dim, int Degree>
static void BM_SplineFit(benchmark::State& state) {
const int n = state.range(0);
typedef Spline<Scalar, Dim> SplineType;
typedef typename SplineType::PointType PointType;
// Generate random points.
Matrix<Scalar, Dim, Dynamic> pts(Dim, n);
pts.setRandom();
for (auto _ : state) {
SplineType spline = SplineFitting<SplineType>::Interpolate(pts, Degree);
benchmark::DoNotOptimize(spline.knots().data());
benchmark::ClobberMemory();
}
}
// --- Spline evaluation ---
template <int Dim, int Degree>
static void BM_SplineEval(benchmark::State& state) {
const int n = state.range(0); // number of control points for fitting
const int neval = 1000; // number of evaluation points
typedef Spline<Scalar, Dim> SplineType;
Matrix<Scalar, Dim, Dynamic> pts(Dim, n);
pts.setRandom();
SplineType spline = SplineFitting<SplineType>::Interpolate(pts, Degree);
// Generate evaluation parameters in [0, 1].
VectorXd u = VectorXd::LinSpaced(neval, 0, 1);
for (auto _ : state) {
for (int i = 0; i < neval; ++i) {
auto pt = spline(u(i));
benchmark::DoNotOptimize(pt.data());
}
benchmark::ClobberMemory();
}
state.counters["Evals/s"] = benchmark::Counter(neval, benchmark::Counter::kIsIterationInvariantRate);
}
// --- Spline derivative evaluation ---
template <int Dim, int Degree>
static void BM_SplineDerivatives(benchmark::State& state) {
const int n = state.range(0);
const int neval = 1000;
typedef Spline<Scalar, Dim> SplineType;
Matrix<Scalar, Dim, Dynamic> pts(Dim, n);
pts.setRandom();
SplineType spline = SplineFitting<SplineType>::Interpolate(pts, Degree);
VectorXd u = VectorXd::LinSpaced(neval, 0, 1);
for (auto _ : state) {
for (int i = 0; i < neval; ++i) {
auto derivs = spline.derivatives(u(i), 1);
benchmark::DoNotOptimize(derivs.data());
}
benchmark::ClobberMemory();
}
state.counters["Evals/s"] = benchmark::Counter(neval, benchmark::Counter::kIsIterationInvariantRate);
}
static void SplineSizes(::benchmark::Benchmark* b) {
for (int n : {10, 50, 200, 1000}) {
b->Arg(n);
}
}
// 2D cubic splines
BENCHMARK(BM_SplineFit<2, 3>)->Apply(SplineSizes)->Name("SplineFit_2D_Cubic");
BENCHMARK(BM_SplineEval<2, 3>)->Apply(SplineSizes)->Name("SplineEval_2D_Cubic");
BENCHMARK(BM_SplineDerivatives<2, 3>)->Apply(SplineSizes)->Name("SplineDerivatives_2D_Cubic");
// 3D cubic splines
BENCHMARK(BM_SplineFit<3, 3>)->Apply(SplineSizes)->Name("SplineFit_3D_Cubic");
BENCHMARK(BM_SplineEval<3, 3>)->Apply(SplineSizes)->Name("SplineEval_3D_Cubic");
BENCHMARK(BM_SplineDerivatives<3, 3>)->Apply(SplineSizes)->Name("SplineDerivatives_3D_Cubic");
// 2D quintic splines
BENCHMARK(BM_SplineFit<2, 5>)->Apply(SplineSizes)->Name("SplineFit_2D_Quintic");
BENCHMARK(BM_SplineEval<2, 5>)->Apply(SplineSizes)->Name("SplineEval_2D_Quintic");