fixed thread safety issue in FFT-based Bitmap::convolve

metadata
Wenzel Jakob 2013-11-13 11:12:06 +01:00
parent dac8d57857
commit 9bc91123c6
1 changed files with 12 additions and 1 deletions

View File

@ -22,6 +22,7 @@
#include <mitsuba/core/fstream.h> #include <mitsuba/core/fstream.h>
#include <boost/algorithm/string.hpp> #include <boost/algorithm/string.hpp>
#include <boost/scoped_array.hpp> #include <boost/scoped_array.hpp>
#include <boost/thread/mutex.hpp>
#if defined(__WINDOWS__) #if defined(__WINDOWS__)
#undef _CRT_SECURE_NO_WARNINGS #undef _CRT_SECURE_NO_WARNINGS
@ -591,6 +592,10 @@ void Bitmap::accumulate(const Bitmap *bitmap, Point2i sourceOffset,
} }
} }
#if defined(MTS_HAS_FFTW)
static boost::mutex __fftw_lock;
#endif
void Bitmap::convolve(const Bitmap *_kernel) { void Bitmap::convolve(const Bitmap *_kernel) {
if (_kernel->getWidth() != _kernel->getHeight()) if (_kernel->getWidth() != _kernel->getHeight())
Log(EError, "Bitmap::convolve(): convolution kernel must be square!"); Log(EError, "Bitmap::convolve(): convolution kernel must be square!");
@ -617,17 +622,21 @@ void Bitmap::convolve(const Bitmap *_kernel) {
paddedHeight = height + hKernelSize, paddedHeight = height + hKernelSize,
paddedSize = paddedWidth*paddedHeight; paddedSize = paddedWidth*paddedHeight;
__fftw_lock.lock();
complex *kernel = (complex *) fftw_malloc(sizeof(complex) * paddedSize), complex *kernel = (complex *) fftw_malloc(sizeof(complex) * paddedSize),
*kernelS = (complex *) fftw_malloc(sizeof(complex) * paddedSize), *kernelS = (complex *) fftw_malloc(sizeof(complex) * paddedSize),
*data = (complex *) fftw_malloc(sizeof(complex) * paddedSize), *data = (complex *) fftw_malloc(sizeof(complex) * paddedSize),
*dataS = (complex *) fftw_malloc(sizeof(complex) * paddedSize); *dataS = (complex *) fftw_malloc(sizeof(complex) * paddedSize);
if (!kernel || !kernelS || !data || !dataS) if (!kernel || !kernelS || !data || !dataS) {
__fftw_lock.unlock();
SLog(EError, "Bitmap::convolve(): Unable to allocate temporary memory!"); SLog(EError, "Bitmap::convolve(): Unable to allocate temporary memory!");
}
/* Create a FFTW plan for a 2D DFT of this size */ /* Create a FFTW plan for a 2D DFT of this size */
fftw_plan p = fftw_plan_dft_2d((int) paddedHeight, (int) paddedWidth, fftw_plan p = fftw_plan_dft_2d((int) paddedHeight, (int) paddedWidth,
(fftw_complex *) kernel, (fftw_complex *) kernelS, FFTW_FORWARD, FFTW_ESTIMATE); (fftw_complex *) kernel, (fftw_complex *) kernelS, FFTW_FORWARD, FFTW_ESTIMATE);
__fftw_lock.unlock();
memset(kernel, 0, sizeof(complex)*paddedSize); memset(kernel, 0, sizeof(complex)*paddedSize);
@ -730,11 +739,13 @@ void Bitmap::convolve(const Bitmap *_kernel) {
Log(EError, "Unsupported component format!"); Log(EError, "Unsupported component format!");
} }
} }
__fftw_lock.lock();
fftw_destroy_plan(p); fftw_destroy_plan(p);
fftw_free(kernel); fftw_free(kernel);
fftw_free(kernelS); fftw_free(kernelS);
fftw_free(data); fftw_free(data);
fftw_free(dataS); fftw_free(dataS);
__fftw_lock.unlock();
#else #else
/* Brute force fallback version */ /* Brute force fallback version */
uint8_t *output_ = static_cast<uint8_t *>(allocAligned(getBufferSize())); uint8_t *output_ = static_cast<uint8_t *>(allocAligned(getBufferSize()));