epfl-archive/cs440-acg/ext/openexr/IlmBase/ImathTest/testJacobiEigenSolver.cpp
2022-04-07 18:46:57 +02:00

314 lines
9.7 KiB
C++

///////////////////////////////////////////////////////////////////////////
//
// Copyright (c) 2010, Industrial Light & Magic, a division of Lucas
// Digital Ltd. LLC
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Industrial Light & Magic nor the names of
// its contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
//
///////////////////////////////////////////////////////////////////////////
#include "ImathMatrix.h"
#include "ImathMatrixAlgo.h"
#include <iostream>
#include <math.h>
#include <cmath>
#include <ctime>
#include <cassert>
#include <limits>
#include <algorithm>
using namespace std;
using namespace IMATH_INTERNAL_NAMESPACE;
const Matrix33<double> A33_1 ( 1, 0, 0, 0, 1, 0, 0, 0, 1 );
const Matrix33<double> A33_2 ( 1, 0, 0, 0,-1, 0, 0, 0, 1 );
const Matrix33<double> A33_3 ( 1, 0, 0, 0, 1, 0, 0, 0, 0 );
const Matrix33<double> A33_4 ( 1, 0, 0, 0, 0, 0, 0, 0, 0 );
const Matrix33<double> A33_5 ( 0, 0, 0, 0, 0, 0, 0, 0, 0 );
const Matrix33<double> A33_6 ( 1, 0, 0, 0, 1e-10, 0, 0, 0, 0 );
const Matrix33<double> A33_7 ( 1, 0, 0, 0, 1e-10, 0, 0, 0, 1e+10 );
const Matrix33<double> A33_8 (
0.25058694044821, 0.49427229444416, 0.81415724537748,
0.49427229444416, 0.80192384710853, -0.61674948224910,
0.81415724537748, -0.61674948224910, -1.28486154645285);
const Matrix33<double> A33_9 (
4, -30, 60,
-30, 300, -675,
60, -675, 1620);
const Matrix44<double> A44_1 ( 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 );
const Matrix44<double> A44_2 ( 1, 0, 0, 0, 0,-1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 );
const Matrix44<double> A44_3 ( 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 );
const Matrix44<double> A44_4 ( 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 );
const Matrix44<double> A44_5 ( 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 );
const Matrix44<double> A44_6 ( 1, 0, 0, 0, 0, 1e-20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
const Matrix44<double> A44_7 ( 1, 0, 0, 0, 0, 1e-20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1e+20 );
const Matrix44<double> A44_8 (
4.05747631538951, 0.16358123075600, 0.11541756047409, -1.65369223465270,
0.16358123075600, 0.57629829390780, 3.88542912704029, 0.92016316185369,
0.11541756047409, 3.88542912704029, 0.65367032943707, -0.21971103270410,
-1.65369223465270, 0.92016316185369, -0.21971103270410, -0.28108876552761);
const Matrix44<double> A44_9 (
4, -30, 60, -35,
-30, 300, -675, 420,
60, -675, 1620, -1050,
-35, 420, -1050, 700);
template <typename TM>
void
verifyOrthonormal (const TM& A, const typename TM::BaseType threshold)
{
const TM prod = A * A.transposed();
for (int i = 0; i < TM::dimensions(); ++i)
for (int j = 0; j < TM::dimensions(); ++j)
if (i == j)
assert (std::abs (prod[i][j] - 1) < threshold);
else
assert (std::abs (prod[i][j]) < threshold);
}
template <typename TM>
typename TM::BaseType
computeThreshold(const TM& A)
{
typedef typename TM::BaseType T;
T maxAbsEntry(0);
for (int i = 0; i < TM::dimensions(); ++i)
for (int j = 0; j < TM::dimensions(); ++j)
maxAbsEntry = std::max (maxAbsEntry, std::abs(A[i][j]));
const T eps = std::numeric_limits<T>::epsilon();
maxAbsEntry = std::max(maxAbsEntry, eps);
return maxAbsEntry * T(100) * eps;
}
template<class TM>
void
testJacobiEigenSolver(const TM& A)
{
using std::abs;
typedef typename TM::BaseType T;
typedef typename TM::BaseVecType TV;
const T threshold = computeThreshold(A);
TM AA(A);
TV S;
TM V;
jacobiEigenSolver(AA, S, V);
// Orthogonality of V
verifyOrthonormal(V, threshold);
// Determinant of V
assert(abs(V.determinant()) - 1 < threshold);
// Determinant of A and S
TM MS;
for (int i = 0; i < TM::dimensions(); ++i)
for (int j = 0; j < TM::dimensions(); ++j)
if(i == j)
MS[i][j] = S[i];
else
MS[i][j] = 0;
assert(abs(A.determinant()) - abs(MS.determinant()) <
threshold);
// A = V * S * V^T
TM MA = V * MS * V.transposed();
for (int i = 0; i < TM::dimensions(); ++i)
for (int j =0; j < TM::dimensions(); ++j)
assert(abs(A[i][j]-MA[i][j]) < threshold);
}
template<class TM>
void
testMinMaxEigenValue(const TM& A)
{
typedef typename TM::BaseVecType TV;
typedef typename TM::BaseType T;
TV minV, maxV, S;
TM U, V;
const T threshold = computeThreshold(A);
{
TM A1(A);
minEigenVector(A1, minV);
TM A2(A);
maxEigenVector(A2, maxV);
}
{
TM A3(A);
jacobiSVD(A3, U, S, V);
}
const int dim = TM::dimensions();
for(int i = 0; i < dim; ++i) {
assert(abs(minV[i]-V[i][dim - 1]) < threshold);
assert(abs(maxV[i]-V[i][0]) < threshold);
}
}
template <class T>
void
testJacobiTiming()
{
int rounds(100000);
clock_t tJacobi,tSVD, t;
{
Matrix33<T> A,V,U;
Vec3<T> S;
t = clock();
for(int i = 0; i < rounds; ++i) {
A = Matrix33<T>(A33_7);
jacobiEigenSolver(A, S, V);
A = Matrix33<T>(A33_8);
jacobiEigenSolver(A, S, V);
}
tJacobi = clock() - t;
cout << "Jacobi EigenSolver of 3x3 matrices took " << tJacobi << " clocks." << endl;
t = clock();
for(int i = 0; i < rounds; ++i) {
A = Matrix33<T>(A33_7);
jacobiSVD(A, U, S, V);
A = Matrix33<T>(A33_8);
jacobiSVD(A, U, S, V);
}
tSVD = clock() - t;
cout << "TinySVD of 3x3 matrices took " << tSVD << " clocks." << endl;
cout << (float)(tSVD-tJacobi)*100.0f/(float)(tSVD) << "% speed up." << endl;
}
{
Matrix44<T> A,V,U;
Vec4<T> S;
t = clock();
for(int i = 0; i < rounds; ++i) {
A = Matrix44<T>(A44_7);
jacobiEigenSolver(A, S, V);
A = Matrix44<T>(A44_8);
jacobiEigenSolver(A, S, V);
}
tJacobi = clock() - t;
cout << "Jacobi EigenSolver of 4x4 matrices took " << tJacobi << " clocks" << endl;
t = clock();
for(int i = 0; i < rounds; ++i) {
A = Matrix44<T>(A44_7);
jacobiSVD(A, U, S, V);
A = Matrix44<T>(A44_8);
jacobiSVD(A, U, S, V);
}
tSVD = clock() - t;
cout << "TinySVD of 4x4 matrices took " << tSVD << " clocks" << endl;
cout << (float)(tSVD-tJacobi)*100.0f/(float)(tSVD) << "% speed up." << endl;
}
}
template <class T>
void
testJacobiEigenSolverImp()
{
testJacobiEigenSolver(Matrix33<T>(A33_1));
testJacobiEigenSolver(Matrix33<T>(A33_2));
testJacobiEigenSolver(Matrix33<T>(A33_3));
testJacobiEigenSolver(Matrix33<T>(A33_4));
testJacobiEigenSolver(Matrix33<T>(A33_5));
testJacobiEigenSolver(Matrix33<T>(A33_6));
testJacobiEigenSolver(Matrix33<T>(A33_7));
testJacobiEigenSolver(Matrix33<T>(A33_8));
testJacobiEigenSolver(Matrix44<T>(A44_1));
testJacobiEigenSolver(Matrix44<T>(A44_2));
testJacobiEigenSolver(Matrix44<T>(A44_3));
testJacobiEigenSolver(Matrix44<T>(A44_4));
testJacobiEigenSolver(Matrix44<T>(A44_5));
testJacobiEigenSolver(Matrix44<T>(A44_6));
testJacobiEigenSolver(Matrix44<T>(A44_7));
testJacobiEigenSolver(Matrix44<T>(A44_8));
}
template <class T>
void
testMinMaxEigenValueImp()
{
testMinMaxEigenValue(Matrix33<T>(A33_7));
testMinMaxEigenValue(Matrix33<T>(A33_8));
testMinMaxEigenValue(Matrix44<T>(A44_7));
testMinMaxEigenValue(Matrix44<T>(A44_8));
}
void
testJacobiEigenSolver()
{
cout << endl;
cout << "************ Testing IMATH_INTERNAL_NAMESPACE::ImathJacobiEigenSolver ************" << endl;
cout << "Jacobi EigenSolver in single precision...";
testJacobiEigenSolverImp<float>();
cout << "PASS" << endl;
cout << "Jacobi EigenSolver in double precision...";
testJacobiEigenSolverImp<double>();
cout << "PASS" << endl;
cout << "Min/Max EigenValue in single precision...";
testMinMaxEigenValueImp<float>();
cout << "PASS" << endl;
cout << "Min/Max EigenValue in double precision...";
testMinMaxEigenValueImp<double>();
cout << "PASS" << endl;
cout << "Timing Jacobi EigenSolver in single precision...\n";
testJacobiTiming<float>();
cout << "Timing Jacobi EigenSolver in double precision...\n";
testJacobiTiming<double>();
cout << "************ ALL PASS ************" << endl;
}