/* This file is part of Mitsuba, a physically based rendering system. Copyright (c) 2007-2012 by Wenzel Jakob and others. Mitsuba is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License Version 3 as published by the Free Software Foundation. Mitsuba is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #include MTS_NAMESPACE_BEGIN Vector Warp::squareToUniformSphere(const Point2 &sample) { Float z = 1.0f - 2.0f * sample.y; Float r = math::safe_sqrt(1.0f - z*z); Float sinPhi, cosPhi; math::sincos(2.0f * M_PI * sample.x, &sinPhi, &cosPhi); return Vector(r * cosPhi, r * sinPhi, z); } Vector Warp::squareToUniformHemisphere(const Point2 &sample) { Float z = sample.x; Float tmp = math::safe_sqrt(1.0f - z*z); Float sinPhi, cosPhi; math::sincos(2.0f * M_PI * sample.y, &sinPhi, &cosPhi); return Vector(cosPhi * tmp, sinPhi * tmp, z); } Vector Warp::squareToCosineHemisphere(const Point2 &sample) { Point2 p = Warp::squareToUniformDiskConcentric(sample); Float z = math::safe_sqrt(1.0f - p.x*p.x - p.y*p.y); /* Guard against numerical imprecisions */ if (EXPECT_NOT_TAKEN(z == 0)) z = 1e-10f; return Vector(p.x, p.y, z); } Vector Warp::squareToUniformCone(Float cosCutoff, const Point2 &sample) { Float cosTheta = (1-sample.x) + sample.x * cosCutoff; Float sinTheta = math::safe_sqrt(1.0f - cosTheta * cosTheta); Float sinPhi, cosPhi; math::sincos(2.0f * M_PI * sample.y, &sinPhi, &cosPhi); return Vector(cosPhi * sinTheta, sinPhi * sinTheta, cosTheta); } Point2 Warp::squareToUniformDisk(const Point2 &sample) { Float r = std::sqrt(sample.x); Float sinPhi, cosPhi; math::sincos(2.0f * M_PI * sample.y, &sinPhi, &cosPhi); return Point2( cosPhi * r, sinPhi * r ); } Point2 Warp::squareToUniformTriangle(const Point2 &sample) { Float a = math::safe_sqrt(1.0f - sample.x); return Point2(1 - a, a * sample.y); } Point2 Warp::squareToUniformDiskConcentric(const Point2 &sample) { Float r1 = 2.0f*sample.x - 1.0f; Float r2 = 2.0f*sample.y - 1.0f; Point2 coords; if (r1 == 0 && r2 == 0) { coords = Point2(0, 0); } else if (r1 > -r2) { /* Regions 1/2 */ if (r1 > r2) coords = Point2(r1, (M_PI/4.0f) * r2/r1); else coords = Point2(r2, (M_PI/4.0f) * (2.0f - r1/r2)); } else { /* Regions 3/4 */ if (r1