mitsuba/src/integrators/path/volpath.cpp

304 lines
11 KiB
C++

#include <mitsuba/render/scene.h>
MTS_NAMESPACE_BEGIN
/**
* Volumetric path tracer, which solves the full radiative transfer
* equation in the presence of participating media. Estimates single
* scattering using both phase function and luminaire sampling and
* combines the two with multiple importance sampling and the power
* heuristic. Afterwards, the phase function sample is reused to
* recursively estimate the multiple scattering component, which
* saves an intersection computation.
* On surfaces, this integrator behaves exactly like the standard
* MI path tracer.
*/
class VolumetricPathTracer : public MonteCarloIntegrator {
public:
VolumetricPathTracer(const Properties &props) : MonteCarloIntegrator(props) {
/* Beta factor for the power heuristic */
m_beta = props.getFloat("beta", 2.0f);
}
/// Unserialize from a binary data stream
VolumetricPathTracer(Stream *stream, InstanceManager *manager)
: MonteCarloIntegrator(stream, manager) {
m_beta = stream->readFloat();
}
Spectrum Li(const RayDifferential &r, RadianceQueryRecord &rRec) const {
/* Some aliases and local variables */
const Scene *scene = rRec.scene;
Intersection &its = rRec.its;
LuminaireSamplingRecord lRec;
MediumSamplingRecord mRec;
RayDifferential ray(r);
Intersection prevIts;
Spectrum Li(0.0f);
/* Perform the first ray intersection (or ignore if the
intersection has already been provided). */
rRec.rayIntersect(ray);
Spectrum pathThroughput(1.0f);
while (rRec.depth <= m_maxDepth || m_maxDepth < 0) {
/* ==================================================================== */
/* Radiative Transfer Equation sampling */
/* ==================================================================== */
if (scene->sampleDistance(ray, its.t, mRec, rRec.sampler)) {
const PhaseFunction *phase = mRec.medium->getPhaseFunction();
Vector wo, wi = -ray.d;
if (rRec.depth == m_maxDepth && m_maxDepth > 0) // No more scattering events allowed
break;
/* Sample the integral
\int_x^y tau(x, x') [ \sigma_s \int_{S^2} \rho(\omega,\omega') L(x,\omega') d\omega' ] dx'
*/
pathThroughput *= mRec.sigmaS * mRec.attenuation * mRec.miWeight / mRec.pdf;
/* ==================================================================== */
/* Luminaire sampling */
/* ==================================================================== */
/* Estimate the single scattering component if this is requested */
if (rRec.type & RadianceQueryRecord::EInscatteredDirectRadiance &&
scene->sampleLuminaireAttenuated(mRec.p, lRec, rRec.nextSample2D())) {
/* Evaluate the phase function */
Spectrum phaseVal = phase->f(mRec, -ray.d, -lRec.d);
if (phaseVal.max() > 0) {
/* Calculate prob. of having sampled that direction using
phase function sampling */
Float phasePdf = (lRec.luminaire->isIntersectable()
|| lRec.luminaire->isBackgroundLuminaire()) ?
phase->pdf(mRec, -ray.d, -lRec.d) : 0;
/* Weight using the power heuristic */
const Float weight = miWeight(lRec.pdf, phasePdf);
Li += pathThroughput * lRec.Le * phaseVal * weight;
}
}
/* ==================================================================== */
/* Phase function sampling */
/* ==================================================================== */
PhaseFunction::ESampledType sampledType;
Float phasePdf;
Spectrum phaseVal = phase->sample(mRec, wi, wo, sampledType, phasePdf, rRec.nextSample2D());
if (phaseVal.max() == 0)
break;
phaseVal /= phasePdf;
prevIts = its;
/* Trace a ray in this direction */
ray = Ray(mRec.p, wo);
bool hitLuminaire = false;
if (scene->rayIntersect(ray, its)) {
/* Intersected something - check if it was a luminaire */
if (its.isLuminaire()) {
lRec = LuminaireSamplingRecord(its, -ray.d);
hitLuminaire = true;
}
} else {
/* No intersection found. Possibly, there is a background
luminaire such as an environment map? */
if (scene->hasBackgroundLuminaire()) {
lRec.luminaire = scene->getBackgroundLuminaire();
lRec.d = -ray.d;
hitLuminaire = true;
}
}
ray.mint = 0; ray.maxt = its.t;
rRec.attenuation = scene->getAttenuation(ray);
/* If a luminaire was hit, estimate the local illumination and
weight using the power heuristic */
if (hitLuminaire && (rRec.type & RadianceQueryRecord::EInscatteredDirectRadiance)) {
lRec.Le = lRec.luminaire->Le(lRec);
/* Prob. of having generated this sample using luminaire sampling */
const Float lumPdf = (sampledType != PhaseFunction::EDelta)
? scene->pdfLuminaire(mRec.p, lRec) : 0;
Float weight = miWeight(phasePdf, lumPdf);
Li += pathThroughput * rRec.attenuation * lRec.Le * phaseVal * weight;
}
/* ==================================================================== */
/* Multiple scattering */
/* ==================================================================== */
/* Stop if multiple scattering was not requested */
if (!(rRec.type & RadianceQueryRecord::EInscatteredIndirectRadiance))
break;
rRec.type = RadianceQueryRecord::ERadianceNoEmission;
/* Russian roulette - Possibly stop the recursion */
if (rRec.depth >= m_rrDepth) {
if (rRec.nextSample1D() > mRec.albedo)
break;
else
pathThroughput /= mRec.albedo;
}
pathThroughput *= phaseVal;
rRec.depth++;
} else {
/* Sample
tau(x, y) (Surface integral). This happens with probability mRec.pdf
Divide this out and multiply with the proper per-color-channel attenuation.
*/
pathThroughput *= mRec.attenuation * mRec.miWeight / mRec.pdf;
if (!its.isValid()) {
/* If no intersection could be found, possibly return
attenuated radiance from a background luminaire */
if (rRec.type & RadianceQueryRecord::EEmittedRadiance)
Li += pathThroughput * scene->LeBackground(ray);
break;
}
const BSDF *bsdf = its.getBSDF(ray);
/* Possibly include emitted radiance if requested */
if (its.isLuminaire() && (rRec.type & RadianceQueryRecord::EEmittedRadiance))
Li += pathThroughput * its.Le(-ray.d);
/* Include radiance from a subsurface integrator if requested */
if (its.hasSubsurface() && (rRec.type & RadianceQueryRecord::ESubsurfaceRadiance))
Li += pathThroughput * its.LoSub(rRec.scene, -ray.d);
if (rRec.depth == m_maxDepth && m_maxDepth > 0)
break;
/* ==================================================================== */
/* Luminaire sampling */
/* ==================================================================== */
/* Estimate the direct illumination if this is requested */
if (rRec.type & RadianceQueryRecord::EDirectRadiance &&
scene->sampleLuminaireAttenuated(its, lRec, rRec.nextSample2D())) {
/* Allocate a record for querying the BSDF */
const BSDFQueryRecord bRec(rRec, its, its.toLocal(-lRec.d));
/* Evaluate BSDF * cos(theta) */
const Spectrum bsdfVal = bsdf->fCos(bRec);
if (!bsdfVal.isBlack()) {
/* Calculate prob. of having sampled that direction
using BSDF sampling */
Float bsdfPdf = (lRec.luminaire->isIntersectable()
|| lRec.luminaire->isBackgroundLuminaire()) ?
bsdf->pdf(bRec) : 0;
/* Weight using the power heuristic */
const Float weight = miWeight(lRec.pdf, bsdfPdf);
Li += pathThroughput * lRec.Le * bsdfVal * weight;
}
}
/* ==================================================================== */
/* BSDF sampling */
/* ==================================================================== */
/* Sample BSDF * cos(theta) */
BSDFQueryRecord bRec(rRec, its, rRec.nextSample2D());
Float bsdfPdf;
Spectrum bsdfVal = bsdf->sampleCos(bRec, bsdfPdf);
if (bsdfVal.isBlack())
break;
bsdfVal /= bsdfPdf;
prevIts = its;
/* Trace a ray in this direction */
ray = Ray(its.p, its.toWorld(bRec.wo));
bool hitLuminaire = false;
if (scene->rayIntersect(ray, its)) {
/* Intersected something - check if it was a luminaire */
if (its.isLuminaire()) {
lRec = LuminaireSamplingRecord(its, -ray.d);
hitLuminaire = true;
}
} else {
/* No intersection found. Possibly, there is a background
luminaire such as an environment map? */
if (scene->hasBackgroundLuminaire()) {
lRec.luminaire = scene->getBackgroundLuminaire();
lRec.d = -ray.d;
hitLuminaire = true;
}
}
ray.mint = 0; ray.maxt = its.t;
rRec.attenuation = scene->getAttenuation(ray);
/* If a luminaire was hit, estimate the local illumination and
weight using the power heuristic */
if (hitLuminaire && (rRec.type & RadianceQueryRecord::EDirectRadiance)
&& !(bRec.sampledType & BSDF::EDelta)) {
lRec.Le = lRec.luminaire->Le(lRec);
/* Prob. of having generated this sample using luminaire sampling */
const Float lumPdf = (!(bRec.sampledType & BSDF::EDelta)) ?
scene->pdfLuminaire(prevIts, lRec) : 0;
const Float weight = miWeight(bsdfPdf, lumPdf);
Li += pathThroughput * rRec.attenuation * lRec.Le * bsdfVal * weight;
}
/* ==================================================================== */
/* Indirect illumination */
/* ==================================================================== */
/* Stop if indirect illumination was not requested */
if (!(rRec.type & RadianceQueryRecord::EIndirectRadiance))
break;
rRec.type = RadianceQueryRecord::ERadianceNoEmission;
/* Russian roulette - Possibly stop the recursion */
if (rRec.depth >= m_rrDepth) {
/* Assuming that BSDF importance sampling is perfect,
the following should equal the maximum albedo
over all spectral samples */
Float approxAlbedo = std::min((Float) 0.9f, bsdfVal.max());
if (rRec.nextSample1D() > approxAlbedo)
break;
else
pathThroughput /= approxAlbedo;
}
pathThroughput *= bsdfVal;
rRec.depth++;
}
}
return Li;
}
inline Float miWeight(Float pdfA, Float pdfB) const {
pdfA = std::pow(pdfA, m_beta);
pdfB = std::pow(pdfB, m_beta);
return pdfA / (pdfA + pdfB);
}
void serialize(Stream *stream, InstanceManager *manager) const {
MonteCarloIntegrator::serialize(stream, manager);
stream->writeFloat(m_beta);
}
std::string toString() const {
std::ostringstream oss;
oss << "VolumetricPathTracer[" << std::endl
<< " beta = " << m_beta << "," << std::endl
<< " maxDepth = " << m_maxDepth << "," << std::endl
<< " rrDepth = " << m_rrDepth << std::endl
<< "]";
return oss.str();
}
MTS_DECLARE_CLASS()
private:
Float m_beta;
};
MTS_IMPLEMENT_CLASS_S(VolumetricPathTracer, false, MonteCarloIntegrator)
MTS_EXPORT_PLUGIN(VolumetricPathTracer, "Volumetric path tracer");
MTS_NAMESPACE_END