hair primitive compiles again
parent
aede861fb3
commit
589398f61a
|
@ -523,7 +523,7 @@ plugins += env.SharedLibrary('plugins/ply', ['src/shapes/ply/ply.cpp', 'src/shap
|
||||||
plugins += env.SharedLibrary('plugins/serialized', ['src/shapes/serialized.cpp'])
|
plugins += env.SharedLibrary('plugins/serialized', ['src/shapes/serialized.cpp'])
|
||||||
plugins += env.SharedLibrary('plugins/sphere', ['src/shapes/sphere.cpp'])
|
plugins += env.SharedLibrary('plugins/sphere', ['src/shapes/sphere.cpp'])
|
||||||
plugins += env.SharedLibrary('plugins/cylinder', ['src/shapes/cylinder.cpp'])
|
plugins += env.SharedLibrary('plugins/cylinder', ['src/shapes/cylinder.cpp'])
|
||||||
#plugins += env.SharedLibrary('plugins/hair', ['src/shapes/hair.cpp'])
|
plugins += env.SharedLibrary('plugins/hair', ['src/shapes/hair.cpp'])
|
||||||
#plugins += env.SharedLibrary('plugins/group', ['src/shapes/group.cpp'])
|
#plugins += env.SharedLibrary('plugins/group', ['src/shapes/group.cpp'])
|
||||||
|
|
||||||
# Samplers
|
# Samplers
|
||||||
|
|
|
@ -758,8 +758,8 @@ protected:
|
||||||
Log(EError, "The intersection cost must be > 0");
|
Log(EError, "The intersection cost must be > 0");
|
||||||
if (m_emptySpaceBonus <= 0 || m_emptySpaceBonus > 1)
|
if (m_emptySpaceBonus <= 0 || m_emptySpaceBonus > 1)
|
||||||
Log(EError, "The empty space bonus must be in [0, 1]");
|
Log(EError, "The empty space bonus must be in [0, 1]");
|
||||||
if (m_stopPrims <= 0)
|
if (m_stopPrims < 0)
|
||||||
Log(EError, "The stopping primitive count must be > 1");
|
Log(EError, "The stopping primitive count must be >= 0");
|
||||||
if (m_exactPrimThreshold < 0)
|
if (m_exactPrimThreshold < 0)
|
||||||
Log(EError, "The exact primitive threshold must be >= 0");
|
Log(EError, "The exact primitive threshold must be >= 0");
|
||||||
if (m_minMaxBins <= 1)
|
if (m_minMaxBins <= 1)
|
||||||
|
|
|
@ -20,75 +20,88 @@
|
||||||
#include <mitsuba/render/bsdf.h>
|
#include <mitsuba/render/bsdf.h>
|
||||||
#include <mitsuba/render/subsurface.h>
|
#include <mitsuba/render/subsurface.h>
|
||||||
#include <mitsuba/render/luminaire.h>
|
#include <mitsuba/render/luminaire.h>
|
||||||
|
#include <mitsuba/render/gkdtree.h>
|
||||||
#include <mitsuba/core/properties.h>
|
#include <mitsuba/core/properties.h>
|
||||||
#include <mitsuba/core/fstream.h>
|
#include <mitsuba/core/fstream.h>
|
||||||
#include <mitsuba/core/fresolver.h>
|
#include <mitsuba/core/fresolver.h>
|
||||||
|
|
||||||
MTS_NAMESPACE_BEGIN
|
MTS_NAMESPACE_BEGIN
|
||||||
|
|
||||||
class Hair : public Shape {
|
|
||||||
|
/**
|
||||||
|
* \brief Acceleration structure for cylindrical hair
|
||||||
|
* segments with miter joins.
|
||||||
|
*/
|
||||||
|
class HairKDTree : public GenericKDTree<HairKDTree> {
|
||||||
|
friend class GenericKDTree<HairKDTree>;
|
||||||
public:
|
public:
|
||||||
Hair(const Properties &props) : Shape(props) {
|
HairKDTree(std::vector<Point> &vertices,
|
||||||
fs::path path = Thread::getThread()->getFileResolver()->resolve(
|
std::vector<bool> &startFiber, Float radius)
|
||||||
props.getString("filename"));
|
: m_radius(radius) {
|
||||||
m_radius = (Float) props.getFloat("radius", 0.05f);
|
// Take the vertex & start fiber arrays (without copying)
|
||||||
|
m_vertices.swap(vertices);
|
||||||
|
m_startFiber.swap(startFiber);
|
||||||
|
|
||||||
Log(EInfo, "Loading hair geometry from \"%s\" ..", path.leaf().c_str());
|
// Compute the index of the first vertex in each segment.
|
||||||
|
m_segIndex.reserve(m_vertices.size());
|
||||||
|
for (size_t i=0; i<m_vertices.size()-1; i++)
|
||||||
|
if (!m_startFiber[i+1])
|
||||||
|
m_segIndex.push_back(i);
|
||||||
|
|
||||||
fs::ifstream is(path);
|
Log(EDebug, "Building a kd-tree for " SIZE_T_FMT " hair vertices, "
|
||||||
if (is.fail())
|
SIZE_T_FMT " segments,", m_vertices.size(), m_segIndex.size());
|
||||||
Log(EError, "Could not open \"%s\"!", path.file_string().c_str());
|
|
||||||
|
|
||||||
std::string line;
|
setStopPrims(0);
|
||||||
bool newFiber = true;
|
buildInternal();
|
||||||
Point p;
|
|
||||||
size_t segmentCount = 0;
|
|
||||||
|
|
||||||
while (is.good()) {
|
|
||||||
std::getline(is, line);
|
|
||||||
if (line.length() > 0 && line[0] == '#')
|
|
||||||
continue;
|
|
||||||
if (line.length() == 0) {
|
|
||||||
newFiber = true;
|
|
||||||
} else {
|
|
||||||
std::istringstream iss(line);
|
|
||||||
iss >> p.x >> p.y >> p.z;
|
|
||||||
if (!iss.fail()) {
|
|
||||||
if (newFiber)
|
|
||||||
segmentCount++;
|
|
||||||
m_vertices.push_back(p);
|
|
||||||
m_startFiber.push_back(newFiber);
|
|
||||||
newFiber = false;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
m_startFiber.push_back(true);
|
|
||||||
|
|
||||||
for (size_t i=0; i<m_vertices.size()-1; ++i) {
|
inline const AABB &getAABB() const {
|
||||||
const Point a = firstVertex();
|
return m_aabb;
|
||||||
const Point b = secondVertex();
|
}
|
||||||
|
|
||||||
|
inline const std::vector<Point> &getVertices() const {
|
||||||
|
return m_vertices;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline const std::vector<bool> &getStartFiber() const {
|
||||||
|
return m_startFiber;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline Float getRadius() const {
|
||||||
|
return m_radius;
|
||||||
|
}
|
||||||
|
protected:
|
||||||
|
AABB getAABB(int index) const {
|
||||||
|
uint32_t iv = m_segIndex[index];
|
||||||
|
|
||||||
// cosine of steepest miter angle
|
// cosine of steepest miter angle
|
||||||
const Float cos0 = dot(firstMiterNormal(), tangent());
|
const Float cos0 = dot(firstMiterNormal(iv), tangent(iv));
|
||||||
const Float cos1 = dot(secondMiterNormal(), tangent());
|
const Float cos1 = dot(secondMiterNormal(iv), tangent(iv));
|
||||||
const Float maxInvCos = 1.0 / std::min(cos0, cos1);
|
const Float maxInvCos = 1.0 / std::min(cos0, cos1);
|
||||||
const Vector expandVec(m_radius * maxIvCos);
|
const Vector expandVec(m_radius * maxInvCos);
|
||||||
|
|
||||||
result.expandBy(a - expandVec);
|
|
||||||
result.expandBy(a + expandVec);
|
|
||||||
result.expandBy(b - expandVec);
|
|
||||||
result.expandBy(b + expandVec);
|
|
||||||
|
|
||||||
|
const Point a = m_vertices[iv];
|
||||||
|
const Point b = m_vertices[iv+1];
|
||||||
|
AABB aabb;
|
||||||
|
aabb.expandBy(a - expandVec);
|
||||||
|
aabb.expandBy(a + expandVec);
|
||||||
|
aabb.expandBy(b - expandVec);
|
||||||
|
aabb.expandBy(b + expandVec);
|
||||||
|
return aabb;
|
||||||
}
|
}
|
||||||
|
|
||||||
Log(EDebug, "Read " SIZE_T_FMT " hair vertices, " SIZE_T_FMT " segments,",
|
AABB getClippedAABB(int index, const AABB &box) const {
|
||||||
m_vertices.size(), segmentCount);
|
AABB aabb(getAABB(index));
|
||||||
|
aabb.clip(box);
|
||||||
|
return aabb;
|
||||||
}
|
}
|
||||||
|
|
||||||
Hair(Stream *stream, InstanceManager *manager)
|
inline int getPrimitiveCount() const {
|
||||||
: Shape(stream, manager) {
|
return m_segIndex.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/* Some utility functions */
|
||||||
inline Vector tangent(int iv) const {
|
inline Vector tangent(int iv) const {
|
||||||
return normalize(m_vertices[iv+1] - m_vertices[iv]);
|
return normalize(m_vertices[iv+1] - m_vertices[iv]);
|
||||||
}
|
}
|
||||||
|
@ -104,12 +117,90 @@ public:
|
||||||
if (!m_startFiber[iv+2])
|
if (!m_startFiber[iv+2])
|
||||||
return normalize(tangent(iv) + tangent(iv+1));
|
return normalize(tangent(iv) + tangent(iv+1));
|
||||||
else
|
else
|
||||||
return tangent();
|
return tangent(iv);
|
||||||
|
}
|
||||||
|
|
||||||
|
EIntersectionResult intersect(const Ray &ray, index_type idx,
|
||||||
|
Float mint, Float maxt, Float &t, void *tmp) {
|
||||||
|
return ENo;
|
||||||
|
}
|
||||||
|
|
||||||
|
MTS_DECLARE_CLASS()
|
||||||
|
protected:
|
||||||
|
std::vector<Point> m_vertices;
|
||||||
|
std::vector<bool> m_startFiber;
|
||||||
|
std::vector<uint32_t> m_segIndex;
|
||||||
|
Float m_radius;
|
||||||
|
};
|
||||||
|
|
||||||
|
class Hair : public Shape {
|
||||||
|
public:
|
||||||
|
Hair(const Properties &props) : Shape(props) {
|
||||||
|
fs::path path = Thread::getThread()->getFileResolver()->resolve(
|
||||||
|
props.getString("filename"));
|
||||||
|
Float radius = (Float) props.getFloat("radius", 0.05f);
|
||||||
|
|
||||||
|
Log(EInfo, "Loading hair geometry from \"%s\" ..", path.leaf().c_str());
|
||||||
|
|
||||||
|
fs::ifstream is(path);
|
||||||
|
if (is.fail())
|
||||||
|
Log(EError, "Could not open \"%s\"!", path.file_string().c_str());
|
||||||
|
|
||||||
|
std::string line;
|
||||||
|
bool newFiber = true;
|
||||||
|
Point p;
|
||||||
|
std::vector<Point> vertices;
|
||||||
|
std::vector<bool> startFiber;
|
||||||
|
|
||||||
|
while (is.good()) {
|
||||||
|
std::getline(is, line);
|
||||||
|
if (line.length() > 0 && line[0] == '#')
|
||||||
|
continue;
|
||||||
|
if (line.length() == 0) {
|
||||||
|
newFiber = true;
|
||||||
|
} else {
|
||||||
|
std::istringstream iss(line);
|
||||||
|
iss >> p.x >> p.y >> p.z;
|
||||||
|
if (!iss.fail()) {
|
||||||
|
vertices.push_back(p);
|
||||||
|
startFiber.push_back(newFiber);
|
||||||
|
newFiber = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
startFiber.push_back(true);
|
||||||
|
m_kdtree = new HairKDTree(vertices, startFiber, radius);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Hair(Stream *stream, InstanceManager *manager)
|
||||||
|
: Shape(stream, manager) {
|
||||||
|
Float radius = stream->readFloat();
|
||||||
|
size_t vertexCount = (size_t) stream->readUInt();
|
||||||
|
|
||||||
|
std::vector<Point> vertices(vertexCount);
|
||||||
|
std::vector<bool> startFiber(vertexCount+1);
|
||||||
|
stream->readFloatArray((Float *) &vertices[0], vertexCount * 3);
|
||||||
|
|
||||||
|
for (size_t i=0; i<vertexCount; ++i)
|
||||||
|
startFiber[i] = stream->readBool();
|
||||||
|
startFiber[vertexCount] = true;
|
||||||
|
|
||||||
|
m_kdtree = new HairKDTree(vertices, startFiber, radius);
|
||||||
|
}
|
||||||
|
|
||||||
void serialize(Stream *stream, InstanceManager *manager) const {
|
void serialize(Stream *stream, InstanceManager *manager) const {
|
||||||
Shape::serialize(stream, manager);
|
Shape::serialize(stream, manager);
|
||||||
|
|
||||||
|
const std::vector<Point> &vertices = m_kdtree->getVertices();
|
||||||
|
const std::vector<bool> &startFiber = m_kdtree->getStartFiber();
|
||||||
|
|
||||||
|
stream->writeFloat(m_kdtree->getRadius());
|
||||||
|
stream->writeUInt((uint32_t) vertices.size());
|
||||||
|
stream->writeFloatArray((Float *) &vertices[0], vertices.size() * 3);
|
||||||
|
for (size_t i=0; i<vertices.size(); ++i)
|
||||||
|
stream->writeBool(startFiber[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool rayIntersect(const Ray &_ray, Float mint, Float maxt, Float &t, void *temp) const {
|
bool rayIntersect(const Ray &_ray, Float mint, Float maxt, Float &t, void *temp) const {
|
||||||
|
@ -121,8 +212,8 @@ public:
|
||||||
const void *temp, Intersection &its) const {
|
const void *temp, Intersection &its) const {
|
||||||
}
|
}
|
||||||
|
|
||||||
inline AABB getAABB() const {
|
AABB getAABB() const {
|
||||||
return m_aabb;
|
return m_kdtree->getAABB();
|
||||||
}
|
}
|
||||||
|
|
||||||
Float getSurfaceArea() const {
|
Float getSurfaceArea() const {
|
||||||
|
@ -133,20 +224,16 @@ public:
|
||||||
std::string toString() const {
|
std::string toString() const {
|
||||||
std::ostringstream oss;
|
std::ostringstream oss;
|
||||||
oss << "Hair[" << endl
|
oss << "Hair[" << endl
|
||||||
<< " radius = " << m_radius << endl
|
|
||||||
<< "]";
|
<< "]";
|
||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
MTS_DECLARE_CLASS()
|
MTS_DECLARE_CLASS()
|
||||||
private:
|
private:
|
||||||
std::vector<bool> m_startFiber;
|
ref<HairKDTree> m_kdtree;
|
||||||
std::vector<int> m_segIndex;
|
|
||||||
std::vector<Point> m_vertices;
|
|
||||||
AABB m_aabb;
|
|
||||||
Float m_radius;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
MTS_IMPLEMENT_CLASS(HairKDTree, false, GenericKDTree)
|
||||||
MTS_IMPLEMENT_CLASS_S(Hair, false, Shape)
|
MTS_IMPLEMENT_CLASS_S(Hair, false, Shape)
|
||||||
MTS_EXPORT_PLUGIN(Hair, "Hair intersection primitive");
|
MTS_EXPORT_PLUGIN(Hair, "Hair intersection primitive");
|
||||||
MTS_NAMESPACE_END
|
MTS_NAMESPACE_END
|
||||||
|
|
Loading…
Reference in New Issue