Blame ImathTest/testFrustum.cpp

Packit 8dc392
///////////////////////////////////////////////////////////////////////////
Packit 8dc392
//
Packit 8dc392
// Copyright (c) 2002, Industrial Light & Magic, a division of Lucas
Packit 8dc392
// Digital Ltd. LLC
Packit 8dc392
// 
Packit 8dc392
// All rights reserved.
Packit 8dc392
// 
Packit 8dc392
// Redistribution and use in source and binary forms, with or without
Packit 8dc392
// modification, are permitted provided that the following conditions are
Packit 8dc392
// met:
Packit 8dc392
// *       Redistributions of source code must retain the above copyright
Packit 8dc392
// notice, this list of conditions and the following disclaimer.
Packit 8dc392
// *       Redistributions in binary form must reproduce the above
Packit 8dc392
// copyright notice, this list of conditions and the following disclaimer
Packit 8dc392
// in the documentation and/or other materials provided with the
Packit 8dc392
// distribution.
Packit 8dc392
// *       Neither the name of Industrial Light & Magic nor the names of
Packit 8dc392
// its contributors may be used to endorse or promote products derived
Packit 8dc392
// from this software without specific prior written permission. 
Packit 8dc392
// 
Packit 8dc392
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 8dc392
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 8dc392
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit 8dc392
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit 8dc392
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit 8dc392
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit 8dc392
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit 8dc392
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit 8dc392
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit 8dc392
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit 8dc392
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 8dc392
//
Packit 8dc392
///////////////////////////////////////////////////////////////////////////
Packit 8dc392
Packit 8dc392
Packit 8dc392
Packit 8dc392
#include <testFrustum.h>
Packit 8dc392
#include "ImathFrustum.h"
Packit 8dc392
#include "ImathEuler.h"
Packit 8dc392
#include "ImathFun.h"
Packit 8dc392
#include <iostream>
Packit 8dc392
#include <assert.h>
Packit 8dc392
Packit 8dc392
Packit 8dc392
using namespace std;
Packit 8dc392
Packit 8dc392
Packit 8dc392
namespace 
Packit 8dc392
{
Packit 8dc392
Packit 8dc392
void
Packit 8dc392
testFrustumPlanes  (IMATH_INTERNAL_NAMESPACE::Frustumf &frustum)
Packit 8dc392
{
Packit 8dc392
    bool ortho = frustum.orthographic();
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::V3f o (0.0f, 0.0f, 0.0f);
Packit 8dc392
    float eps = 5.0e-4;
Packit 8dc392
Packit 8dc392
    for (float xRo = 0.0f; xRo < 360.0f; xRo += 100.0f)
Packit 8dc392
    {
Packit 8dc392
        for (float yRo = 0.0f; yRo < 360.0f; yRo += 105.0f)
Packit 8dc392
        {
Packit 8dc392
            for (float zRo = 0.0f; zRo < 360.0f; zRo += 110.0f)
Packit 8dc392
            {
Packit 8dc392
                for (float xTr = -10.0f; xTr < 10.0f; xTr += 2)
Packit 8dc392
                {
Packit 8dc392
                    for (float yTr = -10.0f; yTr < 10.0f; yTr += 3)
Packit 8dc392
                    {
Packit 8dc392
                        for (float zTr = -10.0f; zTr < 10.0f; zTr += 4)
Packit 8dc392
                        {
Packit 8dc392
                            float xRoRad = xRo * (2.0f * float(M_PI) / 360.0f);
Packit 8dc392
                            float yRoRad = yRo * (2.0f * float(M_PI) / 360.0f);
Packit 8dc392
                            float zRoRad = zRo * (2.0f * float(M_PI) / 360.0f);
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::Eulerf e(xRoRad, yRoRad, zRoRad);
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::M44f mView = e.toMatrix44();
Packit 8dc392
                            mView.translate (IMATH_INTERNAL_NAMESPACE::V3f(xTr, yTr, zTr));
Packit 8dc392
                            
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::Plane3f planes0[6];
Packit 8dc392
                            frustum.planes (planes0);
Packit 8dc392
                            
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::Plane3f planes[6];
Packit 8dc392
                            frustum.planes (planes, mView);
Packit 8dc392
                            
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f up = IMATH_INTERNAL_NAMESPACE::V3f(0, 1, 0);
Packit 8dc392
                            assert ((up ^ planes0[0].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (up, up);
Packit 8dc392
                            assert ((up ^ planes[0].normal) > 0.0);
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f pt = (! ortho) ? o :
Packit 8dc392
                                IMATH_INTERNAL_NAMESPACE::V3f (0.0f, frustum.top(), 0.0f);
Packit 8dc392
                            float d = planes0[0].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[0].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f right = IMATH_INTERNAL_NAMESPACE::V3f(1, 0, 0);
Packit 8dc392
                            assert ((right ^ planes0[1].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (right, right);
Packit 8dc392
                            assert ((right ^ planes[1].normal) > 0.0);
Packit 8dc392
                            
Packit 8dc392
                            pt = (! ortho) ? o :
Packit 8dc392
                                IMATH_INTERNAL_NAMESPACE::V3f (frustum.right(), 0.0f, 0.0f);
Packit 8dc392
                            d = planes0[1].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[1].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f down = IMATH_INTERNAL_NAMESPACE::V3f(0, -1, 0);
Packit 8dc392
                            assert ((down ^ planes0[2].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (down, down);
Packit 8dc392
                            assert ((down ^ planes[2].normal) > 0.0);
Packit 8dc392
                            
Packit 8dc392
                            pt = (! ortho) ? o :
Packit 8dc392
                                IMATH_INTERNAL_NAMESPACE::V3f (0.0f, frustum.bottom(), 0.0f);
Packit 8dc392
                            d = planes0[2].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[2].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f left = IMATH_INTERNAL_NAMESPACE::V3f(-1, 0, 0);
Packit 8dc392
                            assert ((left ^ planes0[3].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (left, left);
Packit 8dc392
                            assert ((left ^ planes[3].normal) > 0.0);
Packit 8dc392
                            
Packit 8dc392
                            pt = (! ortho) ? o :
Packit 8dc392
                                IMATH_INTERNAL_NAMESPACE::V3f (frustum.left(), 0.0f, 0.0f);
Packit 8dc392
                            d = planes0[3].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[3].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f front = IMATH_INTERNAL_NAMESPACE::V3f(0, 0, 1);
Packit 8dc392
                            assert ((front ^ planes0[4].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (front, front);
Packit 8dc392
                            assert ((front ^ planes[4].normal) > 0.0);
Packit 8dc392
                            
Packit 8dc392
                            pt = IMATH_INTERNAL_NAMESPACE::V3f (0.0f, 0.0f, -frustum.nearPlane());
Packit 8dc392
                            d = planes0[4].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[4].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
Packit 8dc392
                            IMATH_INTERNAL_NAMESPACE::V3f back = IMATH_INTERNAL_NAMESPACE::V3f(0, 0, -1);
Packit 8dc392
                            assert ((back ^ planes0[5].normal) > 0.0);
Packit 8dc392
                            mView.multDirMatrix (back, back);
Packit 8dc392
                            assert ((back ^ planes[5].normal) > 0.0);
Packit 8dc392
                            
Packit 8dc392
                            pt = IMATH_INTERNAL_NAMESPACE::V3f (0.0f, 0.0f, -frustum.farPlane());
Packit 8dc392
                            d = planes0[5].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                            pt = pt * mView;
Packit 8dc392
                            d = planes[5].distanceTo (pt);
Packit 8dc392
                            assert (IMATH_INTERNAL_NAMESPACE::iszero (d, eps));
Packit 8dc392
                        }
Packit 8dc392
                    }
Packit 8dc392
                }
Packit 8dc392
            }
Packit 8dc392
        }
Packit 8dc392
    }
Packit 8dc392
Packit 8dc392
}
Packit 8dc392
Packit 8dc392
}
Packit 8dc392
Packit 8dc392
Packit 8dc392
void
Packit 8dc392
testFrustum ()
Packit 8dc392
{
Packit 8dc392
    cout << "Testing functions in ImathFrustum.h";
Packit 8dc392
Packit 8dc392
    cout << "\nperspective ";
Packit 8dc392
Packit 8dc392
    float n = 1.7;
Packit 8dc392
    float f = 567.0;
Packit 8dc392
    float l = -3.5;
Packit 8dc392
    float r = 2.0;
Packit 8dc392
    float b = -1.3;
Packit 8dc392
    float t = 0.9;
Packit 8dc392
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::Frustum<float> frustum (n, f, l, r, t, b, false);
Packit 8dc392
Packit 8dc392
    assert (IMATH_INTERNAL_NAMESPACE::abs<float> (frustum.fovx() -
Packit 8dc392
			       (atan2(r,n) - atan2(l,n))) < 1e-6);
Packit 8dc392
    assert (IMATH_INTERNAL_NAMESPACE::abs<float> (frustum.fovy() -
Packit 8dc392
			       (atan2(t,n) - atan2(b,n))) < 1e-6);
Packit 8dc392
    cout << "1";
Packit 8dc392
    assert (IMATH_INTERNAL_NAMESPACE::abs<float> (frustum.aspect() - ((r-l)/(t-b))) < 1e-6);
Packit 8dc392
    cout << "2";
Packit 8dc392
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::M44f m = frustum.projectionMatrix();
Packit 8dc392
    assert (IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][0] - ((2*n)/(r-l)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][1])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][2])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][3])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][0])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][1] - ((2*n)/(t-b)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][2])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][3])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][0] - ((r+l)/(r-l)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][1] - ((t+b)/(t-b)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][2] - (-(f+n)/(f-n)))    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][3] - -1.0)		    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][0])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][1])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][2] - ((-2*f*n)/(f-n)))  < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][3])			    < 1e-6);
Packit 8dc392
    cout << "3";
Packit 8dc392
Packit 8dc392
    cout << "\nplanes ";
Packit 8dc392
    testFrustumPlanes (frustum);
Packit 8dc392
Packit 8dc392
    cout << "\nexceptions ";
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::Frustum<float> badFrustum;
Packit 8dc392
Packit 8dc392
    badFrustum.set (n, n, l, r, t, b, false);
Packit 8dc392
    try
Packit 8dc392
    {
Packit 8dc392
	(void)badFrustum.projectionMatrix();
Packit 8dc392
	assert (!"near == far didn't throw an exception");
Packit 8dc392
    }
Packit 8dc392
    catch (IEX_NAMESPACE::DivzeroExc) {}
Packit 8dc392
    cout << "1";
Packit 8dc392
Packit 8dc392
    badFrustum.set (n, f, l, l, t, b, false);
Packit 8dc392
    try
Packit 8dc392
    {
Packit 8dc392
	(void)badFrustum.projectionMatrix();
Packit 8dc392
	assert (!"left == right didn't throw an exception");
Packit 8dc392
    }
Packit 8dc392
    catch (IEX_NAMESPACE::DivzeroExc) {}
Packit 8dc392
    cout << "2";
Packit 8dc392
Packit 8dc392
    badFrustum.set (n, f, l, r, t, t, false);
Packit 8dc392
    try
Packit 8dc392
    {
Packit 8dc392
	(void)badFrustum.projectionMatrix();
Packit 8dc392
	assert (!"top == bottom didn't throw an exception");
Packit 8dc392
    }
Packit 8dc392
    catch (IEX_NAMESPACE::DivzeroExc) {}
Packit 8dc392
    cout << "3";
Packit 8dc392
Packit 8dc392
    cout << "\northographic ";
Packit 8dc392
Packit 8dc392
    frustum.setOrthographic (true);
Packit 8dc392
Packit 8dc392
    m = frustum.projectionMatrix();
Packit 8dc392
    assert (IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][0] - (2/(r-l)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][1])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][2])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[0][3])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][0])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][1] - (2/(t-b)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][2])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[1][3])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][0])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][1])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][2] - (-2/(f-n)))	    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[2][3])			    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][0] - (-(r+l)/(r-l)))    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][1] - (-(t+b)/(t-b)))    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][2] - (-(f+n)/(f-n)))    < 1e-6 &&
Packit 8dc392
	    IMATH_INTERNAL_NAMESPACE::abs<float> (m[3][3] - 1.0)		    < 1e-6);
Packit 8dc392
    cout << "1";
Packit 8dc392
Packit 8dc392
    cout << "\nplanes ";
Packit 8dc392
    testFrustumPlanes (frustum);
Packit 8dc392
Packit 8dc392
    // TODO - There are many little functions in IMATH_INTERNAL_NAMESPACE::Frustum which
Packit 8dc392
    // aren't tested here.  Those test should be added.  But this is
Packit 8dc392
    // a start.
Packit 8dc392
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::Frustum<float> f1 (n, f, l, r, t, b, false);
Packit 8dc392
    IMATH_INTERNAL_NAMESPACE::Frustum<float> f2 (n, f, l, r, t, b, true);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n + 0.1, f, l, r, t, b, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n, f + 0.1, l, r, t, b, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n, f, l + 0.1, r, t, b, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n, f, l, r + 0.1, t, b, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n, f, l, r, t + 0.1, b, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    f2.set(n, f, l, r, t, b + 0.1, false);
Packit 8dc392
    assert (f1 != f2);
Packit 8dc392
    cout << "\npassed inequality test";
Packit 8dc392
Packit 8dc392
    f1 = f2;
Packit 8dc392
    assert (f1 == f2);
Packit 8dc392
    cout << "\npassed equality test";
Packit 8dc392
Packit 8dc392
    cout << "\nok\n\n";
Packit 8dc392
}