|
|
@ -12,35 +12,35 @@
|
|
|
|
namespace hdrplus
|
|
|
|
namespace hdrplus
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
bayer_image::bayer_image( const std::string& bayer_image_path )
|
|
|
|
bayer_image::bayer_image(const std::string& bayer_image_path)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
|
|
|
|
|
|
|
|
// Open RAW image file
|
|
|
|
// Open RAW image file
|
|
|
|
int return_code;
|
|
|
|
int return_code;
|
|
|
|
if ( ( return_code = libraw_processor->open_file( bayer_image_path.c_str() ) ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->open_file(bayer_image_path.c_str())) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor->recycle();
|
|
|
|
libraw_processor->recycle();
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
#else
|
|
|
|
throw std::runtime_error("Error opening file " + bayer_image_path + " " + libraw_strerror( return_code ));
|
|
|
|
throw std::runtime_error("Error opening file " + bayer_image_path + " " + libraw_strerror(return_code));
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Unpack the raw image
|
|
|
|
// Unpack the raw image
|
|
|
|
if ( ( return_code = libraw_processor->unpack() ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->unpack()) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
#else
|
|
|
|
throw std::runtime_error("Error unpack file " + bayer_image_path + " " + libraw_strerror( return_code ));
|
|
|
|
throw std::runtime_error("Error unpack file " + bayer_image_path + " " + libraw_strerror(return_code));
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Get image basic info
|
|
|
|
// Get image basic info
|
|
|
|
width = int( libraw_processor->imgdata.rawdata.sizes.raw_width );
|
|
|
|
width = int(libraw_processor->imgdata.rawdata.sizes.raw_width);
|
|
|
|
height = int( libraw_processor->imgdata.rawdata.sizes.raw_height );
|
|
|
|
height = int(libraw_processor->imgdata.rawdata.sizes.raw_height);
|
|
|
|
|
|
|
|
|
|
|
|
// Read exif tags
|
|
|
|
// Read exif tags
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(bayer_image_path);
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(bayer_image_path);
|
|
|
@ -54,7 +54,7 @@ bayer_image::bayer_image( const std::string& bayer_image_path )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
black_level_per_channel.resize( 4 );
|
|
|
|
black_level_per_channel.resize(4);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
@ -64,26 +64,26 @@ bayer_image::bayer_image( const std::string& bayer_image_path )
|
|
|
|
// Create CV mat
|
|
|
|
// Create CV mat
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
raw_image = cv::Mat( height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image ).clone(); // changed the order of width and height
|
|
|
|
raw_image = cv::Mat(height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image).clone(); // changed the order of width and height
|
|
|
|
|
|
|
|
|
|
|
|
// 2x2 box filter
|
|
|
|
// 2x2 box filter
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>( raw_image );
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>(raw_image);
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
#ifndef NDEBUG
|
|
|
|
printf("%s::%s read bayer image %s with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
printf("%s::%s read bayer image %s with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
__FILE__, __func__, bayer_image_path.c_str(), width, height, iso, white_level, \
|
|
|
|
__FILE__, __func__, bayer_image_path.c_str(), width, height, iso, white_level, \
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3] );
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3]);
|
|
|
|
fflush( stdout );
|
|
|
|
fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
bayer_image::bayer_image(const std::vector<uint8_t>& bayer_image_content)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
|
|
|
|
|
|
|
|
// Open RAW image file
|
|
|
|
// Open RAW image file
|
|
|
|
int return_code;
|
|
|
|
int return_code;
|
|
|
|
if ( ( return_code = libraw_processor->open_buffer( (void *)(&bayer_image_content[0]), bayer_image_content.size() ) ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->open_buffer((void *)(&bayer_image_content[0]), bayer_image_content.size())) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor->recycle();
|
|
|
|
libraw_processor->recycle();
|
|
|
|
#ifdef __ANDROID__ || _WIN32
|
|
|
|
#ifdef __ANDROID__ || _WIN32
|
|
|
@ -95,7 +95,7 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Unpack the raw image
|
|
|
|
// Unpack the raw image
|
|
|
|
if ( ( return_code = libraw_processor->unpack() ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->unpack()) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
#ifdef __ANDROID__ || _WIN32
|
|
|
|
#ifdef __ANDROID__ || _WIN32
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -106,8 +106,8 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Get image basic info
|
|
|
|
// Get image basic info
|
|
|
|
width = int( libraw_processor->imgdata.rawdata.sizes.raw_width );
|
|
|
|
width = int(libraw_processor->imgdata.rawdata.sizes.raw_width);
|
|
|
|
height = int( libraw_processor->imgdata.rawdata.sizes.raw_height );
|
|
|
|
height = int(libraw_processor->imgdata.rawdata.sizes.raw_height);
|
|
|
|
|
|
|
|
|
|
|
|
// Read exif tags
|
|
|
|
// Read exif tags
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(&bayer_image_content[0], bayer_image_content.size());
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(&bayer_image_content[0], bayer_image_content.size());
|
|
|
@ -120,7 +120,7 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
black_level_per_channel.resize( 4 );
|
|
|
|
black_level_per_channel.resize(4);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
@ -130,20 +130,20 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
// Create CV mat
|
|
|
|
// Create CV mat
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
raw_image = cv::Mat( height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image ).clone(); // changed the order of width and height
|
|
|
|
raw_image = cv::Mat(height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image).clone(); // changed the order of width and height
|
|
|
|
|
|
|
|
|
|
|
|
// 2x2 box filter
|
|
|
|
// 2x2 box filter
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>( raw_image );
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>(raw_image);
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
#ifndef NDEBUG
|
|
|
|
printf("%s::%s read bayer image with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
printf("%s::%s read bayer image with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
__FILE__, __func__, width, height, iso, white_level, \
|
|
|
|
__FILE__, __func__, width, height, iso, white_level, \
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3] );
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3]);
|
|
|
|
fflush( stdout );
|
|
|
|
fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bayer_image::bayer_image( std::shared_ptr<MemFile> bayer_image_file )
|
|
|
|
bayer_image::bayer_image(std::shared_ptr<MemFile> bayer_image_file)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
libraw_processor = std::make_shared<LibRaw>();
|
|
|
|
|
|
|
|
|
|
|
@ -151,7 +151,7 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
int return_code;
|
|
|
|
int return_code;
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::vector<uint8_t>& fileData = bayer_image_file->content;
|
|
|
|
std::vector<uint8_t>& fileData = bayer_image_file->content;
|
|
|
|
if ( ( return_code = libraw_processor->open_buffer( (void *)(&fileData[0]), fileData.size() ) ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->open_buffer((void *)(&fileData[0]), fileData.size())) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
libraw_processor->recycle();
|
|
|
|
libraw_processor->recycle();
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
#ifdef __ANDROID__
|
|
|
@ -165,7 +165,7 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Unpack the raw image
|
|
|
|
// Unpack the raw image
|
|
|
|
if ( ( return_code = libraw_processor->unpack() ) != LIBRAW_SUCCESS )
|
|
|
|
if ((return_code = libraw_processor->unpack()) != LIBRAW_SUCCESS)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
#ifdef __ANDROID__
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -176,8 +176,8 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Get image basic info
|
|
|
|
// Get image basic info
|
|
|
|
width = int( libraw_processor->imgdata.rawdata.sizes.raw_width );
|
|
|
|
width = int(libraw_processor->imgdata.rawdata.sizes.raw_width);
|
|
|
|
height = int( libraw_processor->imgdata.rawdata.sizes.raw_height );
|
|
|
|
height = int(libraw_processor->imgdata.rawdata.sizes.raw_height);
|
|
|
|
|
|
|
|
|
|
|
|
// Read exif tags
|
|
|
|
// Read exif tags
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(&bayer_image_file->content[0], bayer_image_file->content.size());
|
|
|
|
Exiv2::Image::AutoPtr image = Exiv2::ImageFactory::open(&bayer_image_file->content[0], bayer_image_file->content.size());
|
|
|
@ -190,7 +190,7 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
white_level = exifData["Exif.Image.WhiteLevel"].toLong();
|
|
|
|
black_level_per_channel.resize( 4 );
|
|
|
|
black_level_per_channel.resize(4);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(0) = exifData["Exif.Image.BlackLevel"].toLong(0);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(1) = exifData["Exif.Image.BlackLevel"].toLong(1);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
|
black_level_per_channel.at(2) = exifData["Exif.Image.BlackLevel"].toLong(2);
|
|
|
@ -200,21 +200,21 @@ bayer_image::bayer_image( const std::vector<uint8_t>& bayer_image_content )
|
|
|
|
// Create CV mat
|
|
|
|
// Create CV mat
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://answers.opencv.org/question/105972/de-bayering-a-cr2-image/
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
// https://www.libraw.org/node/2141
|
|
|
|
raw_image = cv::Mat( height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image ).clone(); // changed the order of width and height
|
|
|
|
raw_image = cv::Mat(height, width, CV_16U, libraw_processor->imgdata.rawdata.raw_image).clone(); // changed the order of width and height
|
|
|
|
|
|
|
|
|
|
|
|
// 2x2 box filter
|
|
|
|
// 2x2 box filter
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>( raw_image );
|
|
|
|
grayscale_image = box_filter_kxk<uint16_t, 2>(raw_image);
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
#ifndef NDEBUG
|
|
|
|
printf("%s::%s read bayer image with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
printf("%s::%s read bayer image with\n width %zu\n height %zu\n iso %.3f\n white level %d\n black level %d %d %d %d\n", \
|
|
|
|
__FILE__, __func__, width, height, iso, white_level, \
|
|
|
|
__FILE__, __func__, width, height, iso, white_level, \
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3] );
|
|
|
|
black_level_per_channel[0], black_level_per_channel[1], black_level_per_channel[2], black_level_per_channel[3]);
|
|
|
|
fflush( stdout );
|
|
|
|
fflush(stdout);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::pair<double, double> bayer_image::get_noise_params() const
|
|
|
|
std::pair<double, double> bayer_image::get_noise_params() const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// Set ISO to 100 if not positive
|
|
|
|
// Set ISO to 100 if not positive
|
|
|
|
double iso_ = iso <= 0 ? 100 : iso;
|
|
|
|
double iso_ = iso <= 0 ? 100 : iso;
|
|
|
|
|
|
|
|
|
|
|
@ -233,6 +233,6 @@ std::pair<double, double> bayer_image::get_noise_params() const
|
|
|
|
|
|
|
|
|
|
|
|
// return pair
|
|
|
|
// return pair
|
|
|
|
return std::make_pair(lambda_shot, lambda_read);
|
|
|
|
return std::make_pair(lambda_shot, lambda_read);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|