Merge with default

metadata
Edgar Velazquez-Armendariz 2013-06-05 17:00:49 -04:00
commit bf670cca99
14 changed files with 170 additions and 54 deletions

View File

@ -92,7 +92,7 @@ and one or more emitters. Here is a more complex example:
<shape type="serialized">
<string name="filename" value="lightsource.serialized"/>
<transform name="toWorld">
<translate x="5" x="-3" z="1"/>
<translate x="5" y="-3" z="1"/>
</transform>
<!-- This mesh is an area emitter -->

View File

@ -83,7 +83,7 @@ public:
* underlying random number generator has been used
* outside of this class
*/
inline void updateSampleIndex(uint64_t index) { m_sampleIndex = index; }
inline void updateSampleIndex(size_t index) { m_sampleIndex = index; }
MTS_DECLARE_CLASS()
protected:

View File

@ -250,6 +250,26 @@ public:
EAuto
};
/// List of different rotation/flip types that can be passed to \ref rotateFlip()
enum ERotateFlipType {
ERotateNoneFlipNone = 0,
ERotate180FlipXY = ERotateNoneFlipNone,
ERotate90FlipNone = 1,
ERotate270FlipXY = ERotate90FlipNone,
ERotate180FlipNone = 2,
ERotateNoneFlipXY = ERotate180FlipNone,
ERotate270FlipNone = 3,
ERotate90FlipXY = ERotate270FlipNone,
ERotateNoneFlipX = 4,
ERotate180FlipY = ERotateNoneFlipX,
ERotate90FlipX = 5,
ERotate270FlipY = ERotate90FlipX,
ERotate180FlipX = 6,
ERotateNoneFlipY = ERotate180FlipX,
ERotate270FlipX = 7,
ERotate90FlipY = ERotate270FlipX
};
/**
* \brief Create a bitmap of the specified type and allocate
* the necessary amount of memory
@ -320,6 +340,15 @@ public:
/// Return whether this image has matching width and height
inline bool isSquare() const { return m_size.x == m_size.y; }
/// Return whether this image has an alpha channel
inline bool hasAlpha() const {
return
m_pixelFormat == ELuminanceAlpha ||
m_pixelFormat == ERGBA ||
m_pixelFormat == EXYZA ||
m_pixelFormat == ESpectrumAlpha;
}
/**
* \brief Return the number bits per component
*
@ -679,6 +708,9 @@ public:
/// Vertically flip the image contents
void flipVertically();
/// Perform the specified rotatation & flip operation
ref<Bitmap> rotateFlip(ERotateFlipType type) const;
/**
* \brief Accumulate the contents of another bitmap into the
* region of the specified offset

View File

@ -662,7 +662,6 @@ template <typename T> struct TVector4 {
/// Number of dimensions
const static int dim = 4;
/** \brief Construct a new vector without initializing it.
*
* This construtor is useful when the vector will either not

View File

@ -210,8 +210,8 @@ public:
void serialize(Stream *stream, InstanceManager *manager) const {
Film::serialize(stream, manager);
stream->writeBool(m_hasBanner);
stream->writeUInt(m_fileFormat);
stream->writeUInt(m_pixelFormat);
stream->writeUInt(m_fileFormat);
stream->writeFloat(m_gamma);
stream->writeUInt(m_tonemapMethod);
stream->writeFloat(m_exposure);

View File

@ -116,21 +116,20 @@ void BeamRadianceEstimator::serialize(Stream *stream, InstanceManager *manager)
AABB BeamRadianceEstimator::buildHierarchy(IndexType index) {
BRENode &node = m_nodes[index];
if (!node.photon.isLeaf()) {
IndexType left = node.photon.getLeftIndex(index);
IndexType right = node.photon.getRightIndex(index);
node.aabb.reset();
if (left)
node.aabb.expandBy(buildHierarchy(left));
if (right)
node.aabb.expandBy(buildHierarchy(right));
} else {
Point center = node.photon.getPosition();
Float radius = node.radius;
node.aabb = AABB(
center - Vector(radius, radius, radius),
center + Vector(radius, radius, radius)
);
if (!node.photon.isLeaf()) {
IndexType left = node.photon.getLeftIndex(index);
IndexType right = node.photon.getRightIndex(index);
if (left)
node.aabb.expandBy(buildHierarchy(left));
if (right)
node.aabb.expandBy(buildHierarchy(right));
}
return node.aabb;

View File

@ -580,7 +580,7 @@ public:
RadianceQueryRecord rRec2;
Intersection &bsdfIts = rRec2.its;
DirectSamplingRecord dRec;
DirectSamplingRecord dRec(its);
for (int i=0; i<numBSDFSamples; ++i) {
/* Sample BSDF * cos(theta) */
BSDFSamplingRecord bRec(its, rRec.sampler, ERadiance);

View File

@ -422,6 +422,60 @@ void Bitmap::flipVertically() {
}
}
ref<Bitmap> Bitmap::rotateFlip(ERotateFlipType type) const {
/* Based on the GDI+ rotate/flip function in Wine */
if (m_componentFormat == EBitmask)
Log(EError, "Transformations involving bitmasks are currently not supported!");
int width = m_size.x, height = m_size.y;
bool flip_x = (type & 6) == 2 || (type & 6) == 4;
bool flip_y = (type & 3) == 1 || (type & 3) == 2;
bool rotate_90 = type & 1;
if (rotate_90)
std::swap(width, height);
ref<Bitmap> result = new Bitmap(m_pixelFormat, m_componentFormat,
Vector2i(width, height), m_channelCount);
ssize_t bypp = getBytesPerPixel(),
src_stride = m_size.x * bypp,
dst_stride = width * bypp;
uint8_t *dst = result->getUInt8Data();
uint8_t *dst_row = dst, *src_row = m_data;
if (flip_x)
src_row += bypp * (m_size.x - 1);
if (flip_y)
src_row += src_stride * (m_size.y - 1);
ssize_t src_x_step, src_y_step;
if (rotate_90) {
src_x_step = flip_y ? -src_stride : src_stride;
src_y_step = flip_x ? -bypp : bypp;
} else {
src_x_step = flip_x ? -bypp : bypp;
src_y_step = flip_y ? -src_stride : src_stride;
}
for (int y=0; y<height; y++) {
uint8_t *src_pixel = src_row;
uint8_t *dst_pixel = dst_row;
for (int x=0; x<width; x++) {
memcpy(dst_pixel, src_pixel, bypp);
dst_pixel += bypp;
src_pixel += src_x_step;
}
src_row += src_y_step;
dst_row += dst_stride;
}
return result;
}
void Bitmap::accumulate(const Bitmap *bitmap, Point2i sourceOffset,
Point2i targetOffset, Vector2i size) {
@ -906,7 +960,9 @@ ref<Bitmap> Bitmap::separateChannel(int channelIndex) {
if (channelIndex == 0 && channelCount == 1)
return this;
Assert(channelIndex > 0 && channelIndex < channelCount);
if (channelIndex < 0 || channelIndex >= channelCount)
Log(EError, "Bitmap::separateChannel(%i): channel index "
"must be between 0 and %i", channelIndex, channelCount-1);
ref<Bitmap> result = new Bitmap(ELuminance, m_componentFormat, m_size);
result->setMetadata(m_metadata);

View File

@ -32,7 +32,8 @@ FileResolver::FileResolver() {
dladdr((const void *) &dummySymbol, &info);
if (info.dli_fname) {
/* Try to detect a few default setups */
if (boost::starts_with(info.dli_fname, "/usr/lib")) {
if (boost::starts_with(info.dli_fname, "/usr/lib") ||
boost::starts_with(info.dli_fname, "/lib")) {
basePath = fs::path("/usr/share/mitsuba");
} else if (boost::starts_with(info.dli_fname, "/usr/local/lib")) {
basePath = fs::path("/usr/local/share/mitsuba");

View File

@ -477,11 +477,11 @@ Random::Random() : mt(NULL) {
seed();
#else
#if 0
uint64_t buf[MT_N];
memset(buf, 0, MT_N * sizeof(uint64_t)); /* Make GCC happy */
uint64_t buf[N64];
memset(buf, 0, N64 * sizeof(uint64_t)); /* Make GCC happy */
ref<FileStream> urandom = new FileStream("/dev/urandom", FileStream::EReadOnly);
urandom->readULongArray(buf, MT_N);
seed(buf, MT_N);
urandom->readULongArray(buf, N64);
seed(buf, N64);
#else
seed();
#endif

View File

@ -375,10 +375,10 @@ public:
/* Find the componentwise maxima of the ellipse */
for (int i=0; i<2; ++i) {
int j = (i==0) ? axis1 : axis2;
Float alpha = ellipseAxes[0][j];
Float beta = ellipseAxes[1][j];
Float ratio = beta/alpha, tmp = std::sqrt(1+ratio*ratio);
Float cosTheta = 1/tmp, sinTheta = ratio/tmp;
Float alpha = ellipseAxes[0][j], beta = ellipseAxes[1][j];
Float tmp = 1 / std::sqrt(alpha*alpha + beta*beta);
Float cosTheta = alpha * tmp, sinTheta = beta*tmp;
Point p1 = ellipseCenter + cosTheta*ellipseAxes[0] + sinTheta*ellipseAxes[1];
Point p2 = ellipseCenter - cosTheta*ellipseAxes[0] - sinTheta*ellipseAxes[1];

View File

@ -350,8 +350,9 @@ public:
int j = (i==0) ? axis1 : axis2;
Float alpha = ellipseAxes[0][j];
Float beta = ellipseAxes[1][j];
Float ratio = beta/alpha, tmp = std::sqrt(1+ratio*ratio);
Float cosTheta = 1/tmp, sinTheta = ratio/tmp;
Float tmp = 1 / std::sqrt(alpha*alpha + beta*beta);
Float cosTheta = alpha * tmp, sinTheta = beta*tmp;
Point p1 = ellipseCenter + cosTheta*ellipseAxes[0] + sinTheta*ellipseAxes[1];
Point p2 = ellipseCenter - cosTheta*ellipseAxes[0] - sinTheta*ellipseAxes[1];

View File

@ -177,6 +177,8 @@ public:
fs::path cacheFile;
ref<Bitmap> bitmap;
m_separateAlpha = props.getBoolean("separateAlpha", false);
if (props.hasProperty("bitmap")) {
/* Support initialization via raw data passed from another plugin */
bitmap = reinterpret_cast<Bitmap *>(props.getData("bitmap").ptr);
@ -194,7 +196,12 @@ public:
Log(EError, "Could not determine modification time of \"%s\"!", m_filename.string().c_str());
cacheFile = m_filename;
if (!m_separateAlpha)
cacheFile.replace_extension(".mip");
else
cacheFile.replace_extension(".alpha.mip");
tryReuseCache = fs::exists(cacheFile) && props.getBoolean("cache", true);
}
@ -241,6 +248,14 @@ public:
}
Bitmap::EPixelFormat pixelFormat;
if (m_separateAlpha) {
/* Create a texture from the alpha channel of an image */
if (!bitmap->hasAlpha())
Log(EError, "separateAlpha specified, but the image contains no alpha channel!");
pixelFormat = Bitmap::ELuminance;
bitmap = bitmap->separateChannel(bitmap->getChannelCount()-1);
bitmap->setGamma(1.0f);
} else {
switch (bitmap->getPixelFormat()) {
case Bitmap::ELuminance:
case Bitmap::ELuminanceAlpha:
@ -254,6 +269,7 @@ public:
Log(EError, "The input image has an unsupported pixel format!");
return;
}
}
/* (Re)generate the MIP map hierarchy; downsample using a
2-lobed Lanczos reconstruction filter */
@ -305,6 +321,7 @@ public:
m_wrapModeV = (ReconstructionFilter::EBoundaryCondition) stream->readUInt();
m_gamma = stream->readFloat();
m_maxAnisotropy = stream->readFloat();
m_separateAlpha = stream->readBool();
size_t size = stream->readSize();
ref<MemoryStream> mStream = new MemoryStream(size);
@ -323,6 +340,14 @@ public:
rfilter->configure();
Bitmap::EPixelFormat pixelFormat;
if (m_separateAlpha) {
/* Create a texture from the alpha channel of an image */
if (!bitmap->hasAlpha())
Log(EError, "separateAlpha specified, but the image contains no alpha channel!");
pixelFormat = Bitmap::ELuminance;
bitmap = bitmap->separateChannel(bitmap->getChannelCount()-1);
bitmap->setGamma(1.0f);
} else {
switch (bitmap->getPixelFormat()) {
case Bitmap::ELuminance:
case Bitmap::ELuminanceAlpha:
@ -336,6 +361,7 @@ public:
Log(EError, "The input image has an unsupported pixel format!");
return;
}
}
if (pixelFormat == Bitmap::ELuminance)
m_mipmap1 = new MIPMap1(bitmap, pixelFormat, Bitmap::EFloat,
@ -355,6 +381,7 @@ public:
stream->writeUInt(m_wrapModeV);
stream->writeFloat(m_gamma);
stream->writeFloat(m_maxAnisotropy);
stream->writeBool(m_separateAlpha);
if (!m_filename.empty() && fs::exists(m_filename)) {
/* We still have access to the original image -- use that, since
@ -503,6 +530,7 @@ protected:
ReconstructionFilter::EBoundaryCondition m_wrapModeU;
ReconstructionFilter::EBoundaryCondition m_wrapModeV;
Float m_gamma, m_maxAnisotropy;
bool m_separateAlpha;
fs::path m_filename;
};

View File

@ -237,9 +237,9 @@ public:
}
fs::path inputFile = fileResolver->resolve(argv[i]);
Log(EInfo, "Loading EXR image \"%s\" ..", inputFile.string().c_str());
Log(EInfo, "Loading image \"%s\" ..", inputFile.string().c_str());
ref<FileStream> is = new FileStream(inputFile, FileStream::EReadOnly);
ref<Bitmap> input = new Bitmap(Bitmap::EOpenEXR, is);
ref<Bitmap> input = new Bitmap(Bitmap::EAuto, is);
if (crop[2] != -1 && crop[3] != -1)
input = input->crop(Point2i(crop[0], crop[1]), Vector2i(crop[2], crop[3]));
@ -282,9 +282,9 @@ public:
} else {
for (int i=optind; i<argc; ++i) {
fs::path inputFile = fileResolver->resolve(argv[i]);
Log(EInfo, "Loading EXR image \"%s\" ..", inputFile.string().c_str());
Log(EInfo, "Loading image \"%s\" ..", inputFile.string().c_str());
ref<FileStream> is = new FileStream(inputFile, FileStream::EReadOnly);
ref<Bitmap> input = new Bitmap(Bitmap::EOpenEXR, is);
ref<Bitmap> input = new Bitmap(Bitmap::EAuto, is);
if (crop[2] != -1 && crop[3] != -1)
input = input->crop(Point2i(crop[0], crop[1]), Vector2i(crop[2], crop[3]));