/*
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