304 lines
11 KiB
C++
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
|