diff --git a/include/mitsuba/core/bitmap.h b/include/mitsuba/core/bitmap.h
index a0380fe1..a2516846 100644
--- a/include/mitsuba/core/bitmap.h
+++ b/include/mitsuba/core/bitmap.h
@@ -771,9 +771,58 @@ public:
*/
void applyMatrix(Float matrix[3][3]);
+ /**
+ * \brief Copy the contents of another bitmap into the
+ * region with the specified offset
+ *
+ * Out-of-bounds regions are ignored. It is assumed that
+ * bitmap != this.
+ *
+ * \remark This function throws an exception when the bitmaps
+ * use different component formats or channels, or when the
+ * component format is \ref EBitmask.
+ */
+ void copyFrom(const Bitmap *bitmap, Point2i sourceOffset,
+ Point2i targetOffset, Vector2i size);
+
+ /**
+ * \brief Copy the contents of another bitmap into the
+ * region with the specified offset
+ *
+ * This convenience function calls the main copyFrom()
+ * implementation with size set to bitmap->getSize()
+ * and sourceOffset set to zero. Out-of-bounds regions are
+ * ignored. It is assumed that bitmap != this.
+ *
+ * \remark This function throws an exception when the bitmaps
+ * use different component formats or channels, or when the
+ * component format is \ref EBitmask.
+ */
+ inline void copyFrom(const Bitmap *bitmap, Point2i targetOffset) {
+ copyFrom(bitmap, Point2i(0), targetOffset, bitmap->getSize());
+ }
+
+ /**
+ * \brief Copy the contents of another bitmap into the
+ * region with the specified offset
+ *
+ * This convenience function calls the main copyFrom()
+ * implementation with size set to bitmap->getSize()
+ * and sourceOffset and targetOffsettt> set to zero.
+ * Out-of-bounds regions are ignored. It is assumed
+ * that bitmap != this.
+ *
+ * \remark This function throws an exception when the bitmaps
+ * use different component formats or channels, or when the
+ * component format is \ref EBitmask.
+ */
+ inline void copyFrom(const Bitmap *bitmap) {
+ copyFrom(bitmap, Point2i(0), Point2i(0), bitmap->getSize());
+ }
+
/**
* \brief Accumulate the contents of another bitmap into the
- * region of the specified offset
+ * region with the specified offset
*
* Out-of-bounds regions are ignored. It is assumed that
* bitmap != this.
@@ -784,9 +833,10 @@ public:
*/
void accumulate(const Bitmap *bitmap, Point2i sourceOffset,
Point2i targetOffset, Vector2i size);
+
/**
* \brief Accumulate the contents of another bitmap into the
- * region of the specified offset
+ * region with the specified offset
*
* This convenience function calls the main accumulate()
* implementation with size set to bitmap->getSize()
@@ -803,7 +853,7 @@ public:
/**
* \brief Accumulate the contents of another bitmap into the
- * region of the specified offset
+ * region with the specified offset
*
* This convenience function calls the main accumulate()
* implementation with size set to bitmap->getSize()
diff --git a/src/libcore/bitmap.cpp b/src/libcore/bitmap.cpp
index bb88d531..5b69013b 100644
--- a/src/libcore/bitmap.cpp
+++ b/src/libcore/bitmap.cpp
@@ -514,6 +514,52 @@ ref Bitmap::rotateFlip(ERotateFlipType type) const {
return result;
}
+void Bitmap::copyFrom(const Bitmap *bitmap, Point2i sourceOffset,
+ Point2i targetOffset, Vector2i size) {
+
+ if (m_componentFormat == EBitmask)
+ Log(EError, "Bitmap::copy(): bitmasks are not supported!");
+
+ Assert(getPixelFormat() == bitmap->getPixelFormat() &&
+ getComponentFormat() == bitmap->getComponentFormat() &&
+ getChannelCount() == bitmap->getChannelCount());
+
+ Vector2i offsetIncrease(
+ std::max(0, std::max(-sourceOffset.x, -targetOffset.x)),
+ std::max(0, std::max(-sourceOffset.y, -targetOffset.y))
+ );
+
+ sourceOffset += offsetIncrease;
+ targetOffset += offsetIncrease;
+ size -= offsetIncrease;
+
+ Vector2i sizeDecrease(
+ std::max(0, std::max(sourceOffset.x + size.x - bitmap->getWidth(), targetOffset.x + size.x - getWidth())),
+ std::max(0, std::max(sourceOffset.y + size.y - bitmap->getHeight(), targetOffset.y + size.y - getHeight())));
+
+ size -= sizeDecrease;
+
+ if (size.x <= 0 || size.y <= 0)
+ return;
+
+ const size_t
+ pixelStride = getBytesPerPixel(),
+ sourceStride = bitmap->getWidth() * pixelStride,
+ targetStride = getWidth() * pixelStride;
+
+ const uint8_t *source = bitmap->getUInt8Data() +
+ (sourceOffset.x + sourceOffset.y * (size_t) bitmap->getWidth()) * pixelStride;
+
+ uint8_t *target = m_data +
+ (targetOffset.x + targetOffset.y * (size_t) m_size.x) * pixelStride;
+
+ for (int y = 0; y < size.y; ++y) {
+ memcpy(target, source, size.x * getBytesPerPixel());
+ source += sourceStride;
+ target += targetStride;
+ }
+}
+
void Bitmap::accumulate(const Bitmap *bitmap, Point2i sourceOffset,
Point2i targetOffset, Vector2i size) {
Assert(getPixelFormat() == bitmap->getPixelFormat() &&
diff --git a/src/libpython/core.cpp b/src/libpython/core.cpp
index 5fbb6321..2459629d 100644
--- a/src/libpython/core.cpp
+++ b/src/libpython/core.cpp
@@ -1235,6 +1235,9 @@ void export_core() {
void (Bitmap::*accumulate_1)(const Bitmap *bitmap, Point2i sourceOffset, Point2i targetOffset, Vector2i size) = &Bitmap::accumulate;
void (Bitmap::*accumulate_2)(const Bitmap *bitmap, Point2i targetOffset) = &Bitmap::accumulate;
void (Bitmap::*accumulate_3)(const Bitmap *bitmap) = &Bitmap::accumulate;
+ void (Bitmap::*copyFrom_1)(const Bitmap *bitmap, Point2i sourceOffset, Point2i targetOffset, Vector2i size) = &Bitmap::copyFrom;
+ void (Bitmap::*copyFrom_2)(const Bitmap *bitmap, Point2i targetOffset) = &Bitmap::copyFrom;
+ void (Bitmap::*copyFrom_3)(const Bitmap *bitmap) = &Bitmap::copyFrom;
const Properties &(Bitmap::*get_metadata)() const = &Bitmap::getMetadata;
void (Bitmap::*resample_1)(const ReconstructionFilter *,
@@ -1283,6 +1286,9 @@ void export_core() {
.def("accumulate", accumulate_1)
.def("accumulate", accumulate_2)
.def("accumulate", accumulate_3)
+ .def("copyFrom", copyFrom_1)
+ .def("copyFrom", copyFrom_2)
+ .def("copyFrom", copyFrom_3)
.def("convolve", &Bitmap::convolve)
.def("arithmeticOperation", &Bitmap::arithmeticOperation, BP_RETURN_VALUE)
.def("resample", resample_1)