| // This file is part of Eigen, a lightweight C++ template library |
| // for linear algebra. |
| // |
| // Copyright (C) 2011 Gael Guennebaud <gael.guennebaud@inria.fr> |
| // |
| // This Source Code Form is subject to the terms of the Mozilla |
| // Public License v. 2.0. If a copy of the MPL was not distributed |
| // with this file, You can obtain one at http://mozilla.org/MPL/2.0/. |
| |
| |
| // Various sanity tests with exceptions: |
| // - no memory leak when a custom scalar type trow an exceptions |
| // - todo: complete the list of tests! |
| |
| #define EIGEN_STACK_ALLOCATION_LIMIT 100000000 |
| |
| #include "main.h" |
| |
| struct my_exception |
| { |
| my_exception() {} |
| ~my_exception() {} |
| }; |
| |
| class ScalarWithExceptions |
| { |
| public: |
| ScalarWithExceptions() { init(); } |
| ScalarWithExceptions(const float& _v) { init(); *v = _v; } |
| ScalarWithExceptions(const ScalarWithExceptions& other) { init(); *v = *(other.v); } |
| ~ScalarWithExceptions() { |
| delete v; |
| instances--; |
| } |
| |
| void init() { |
| v = new float; |
| instances++; |
| } |
| |
| ScalarWithExceptions operator+(const ScalarWithExceptions& other) const |
| { |
| countdown--; |
| if(countdown<=0) |
| throw my_exception(); |
| return ScalarWithExceptions(*v+*other.v); |
| } |
| |
| ScalarWithExceptions operator-(const ScalarWithExceptions& other) const |
| { return ScalarWithExceptions(*v-*other.v); } |
| |
| ScalarWithExceptions operator*(const ScalarWithExceptions& other) const |
| { return ScalarWithExceptions((*v)*(*other.v)); } |
| |
| ScalarWithExceptions& operator+=(const ScalarWithExceptions& other) |
| { *v+=*other.v; return *this; } |
| ScalarWithExceptions& operator-=(const ScalarWithExceptions& other) |
| { *v-=*other.v; return *this; } |
| ScalarWithExceptions& operator=(const ScalarWithExceptions& other) |
| { *v = *(other.v); return *this; } |
| |
| bool operator==(const ScalarWithExceptions& other) const |
| { return *v==*other.v; } |
| bool operator!=(const ScalarWithExceptions& other) const |
| { return *v!=*other.v; } |
| |
| float* v; |
| static int instances; |
| static int countdown; |
| }; |
| |
| ScalarWithExceptions real(const ScalarWithExceptions &x) { return x; } |
| ScalarWithExceptions imag(const ScalarWithExceptions & ) { return 0; } |
| ScalarWithExceptions conj(const ScalarWithExceptions &x) { return x; } |
| |
| int ScalarWithExceptions::instances = 0; |
| int ScalarWithExceptions::countdown = 0; |
| |
| |
| #define CHECK_MEMLEAK(OP) { \ |
| ScalarWithExceptions::countdown = 100; \ |
| int before = ScalarWithExceptions::instances; \ |
| bool exception_thrown = false; \ |
| try { OP; } \ |
| catch (my_exception) { \ |
| exception_thrown = true; \ |
| VERIFY(ScalarWithExceptions::instances==before && "memory leak detected in " && EIGEN_MAKESTRING(OP)); \ |
| } \ |
| VERIFY(exception_thrown && " no exception thrown in " && EIGEN_MAKESTRING(OP)); \ |
| } |
| |
| void memoryleak() |
| { |
| typedef Eigen::Matrix<ScalarWithExceptions,Dynamic,1> VectorType; |
| typedef Eigen::Matrix<ScalarWithExceptions,Dynamic,Dynamic> MatrixType; |
| |
| { |
| int n = 50; |
| VectorType v0(n), v1(n); |
| MatrixType m0(n,n), m1(n,n), m2(n,n); |
| v0.setOnes(); v1.setOnes(); |
| m0.setOnes(); m1.setOnes(); m2.setOnes(); |
| CHECK_MEMLEAK(v0 = m0 * m1 * v1); |
| CHECK_MEMLEAK(m2 = m0 * m1 * m2); |
| CHECK_MEMLEAK((v0+v1).dot(v0+v1)); |
| } |
| VERIFY(ScalarWithExceptions::instances==0 && "global memory leak detected in " && EIGEN_MAKESTRING(OP)); \ |
| } |
| |
| void test_exceptions() |
| { |
| CALL_SUBTEST( memoryleak() ); |
| } |