🛠️🐜 Antkeeper superbuild with dependencies included https://antkeeper.com
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

199 lines
5.2 KiB

#include <cstdio>
#include <cstdlib>
#include <vector>
#include "tinyexr.h"
#define TINY_DNG_WRITER_IMPLEMENTATION
#include "tiny_dng_writer.h"
static bool Create32bitFpTiff(
const float *data, // [width x height x in_channels]
const size_t width,
const size_t height,
const size_t in_channels,
const size_t channels,
tinydngwriter::DNGImage *dng_image) {
if (in_channels < 1) return false;
unsigned int image_width = uint32_t(width);
unsigned int image_height = uint32_t(height);
//dng_image->SetSubfileType(false, false, false);
dng_image->SetImageWidth(image_width);
dng_image->SetImageLength(image_height);
dng_image->SetRowsPerStrip(image_height);
dng_image->SetSamplesPerPixel(uint16_t(channels));
std::vector<uint16_t> bps(channels);
for (size_t i = 0; i < bps.size(); i++) {
bps[i] = 32;
}
dng_image->SetBitsPerSample(static_cast<unsigned int>(channels), bps.data());
dng_image->SetPlanarConfig(tinydngwriter::PLANARCONFIG_CONTIG);
dng_image->SetCompression(tinydngwriter::COMPRESSION_NONE);
if (channels == 1) {
dng_image->SetPhotometric(
tinydngwriter::PHOTOMETRIC_BLACK_IS_ZERO); // grayscale
} else {
dng_image->SetPhotometric(
tinydngwriter::PHOTOMETRIC_RGB);
}
dng_image->SetXResolution(1.0);
dng_image->SetYResolution(1.0);
dng_image->SetResolutionUnit(tinydngwriter::RESUNIT_NONE);
std::vector<uint16_t> formats(channels);
for (size_t i = 0; i < formats.size(); i++) {
formats[i] = tinydngwriter::SAMPLEFORMAT_IEEEFP;
}
dng_image->SetSampleFormat(static_cast<unsigned int>(channels), formats.data());
std::vector<float> buf;
buf.resize(size_t(channels) * image_width * image_height);
for (size_t i = 0; i < image_width * image_height; i++) {
size_t in_c = 0;
for (size_t c = 0; c < channels; c++) {
buf[channels * i + c] = data[in_channels * i + in_c];
in_c++;
in_c = std::min(in_c, in_channels - 1);
}
}
//size_t max_dump_pixels = 4096;
//for (size_t i = 0; i < std::min(max_dump_pixels, buf.size()); i++) {
// std::cout << "val[" << i << "] = " << buf[i] << "\n";
//}
//std::cout << "last = " << buf.at(image_width * image_height * channels - 1) << "\n";
// We must retain pointer address of `buf` until calling DNGWriter::WriteToFile
dng_image->SetImageData(reinterpret_cast<unsigned char *>(buf.data()),
buf.size() * sizeof(float));
if (!dng_image->Error().empty()) {
std::cout << "Err: " << dng_image->Error() << "\n";
return false;
}
return true;
}
int main(int argc, char** argv)
{
if (argc < 3) {
printf("Usage: exr2fptiff input.exr output.tiff\n");
exit(-1);
}
std::string input_filename = argv[1];
std::string output_filename = argv[2];
// Get # of layers
size_t num_layers{0};
{
EXRVersion exr_version;
{
int ret = ParseEXRVersionFromFile(&exr_version, input_filename.c_str());
if (ret != 0) {
std::cerr << "Invalid EXR file: " << input_filename << "\n";
return EXIT_FAILURE;
}
if (exr_version.multipart) {
std::cerr << "Multipart EXR file is not supported in this example.\n";
return EXIT_FAILURE;
}
}
EXRHeader exr_header;
InitEXRHeader(&exr_header);
const char* err = nullptr;
int ret = ParseEXRHeaderFromFile(&exr_header, &exr_version, argv[1], &err);
if (ret != TINYEXR_SUCCESS) {
if (err) {
std::cerr << "Parse EXR error: " << err << "\n";
FreeEXRErrorMessage(err); // free's buffer for an error message
} else {
std::cerr << "Parse EXR error.\n";
}
return EXIT_FAILURE;
}
num_layers = size_t(exr_header.num_channels);
if (num_layers == 0) {
std::cerr << "no layers found\n";
return EXIT_FAILURE;
}
if (num_layers > 4) {
std::cerr << "This program supports up to 4(e.g. RGBA) layers.\n";
return EXIT_FAILURE;
}
FreeEXRHeader(&exr_header);
}
std::cout << "# of channels = " << num_layers << "\n";
// Use legacy but easy-to-use API to read image.
float *rgba{nullptr};
int width;
int height;
{
const char *err;
int ret = LoadEXR(&rgba, &width, &height, input_filename.c_str(), &err);
if (ret != TINYEXR_SUCCESS) {
if (err) {
std::cerr << "Load EXR error: " << err << "\n";
FreeEXRErrorMessage(err); // free's buffer for an error message
} else {
std::cerr << "Load EXR error.\n";
}
return EXIT_FAILURE;
}
}
bool big_endian = false;
tinydngwriter::DNGImage tiff;
tiff.SetBigEndian(big_endian);
bool ret = Create32bitFpTiff(rgba, size_t(width), size_t(height), /* in_channels */4, size_t(num_layers), &tiff);
if (!ret) {
std::cerr << "Failed to create floating point tiff data\n";
return EXIT_FAILURE;
}
// 4. Free image data
free(rgba);
tinydngwriter::DNGWriter dng_writer(big_endian);
ret = dng_writer.AddImage(&tiff);
if (!ret) {
std::cerr << "Failed to add TIFF image to TIFF writer.\n";
return EXIT_FAILURE;
}
// 5. write tiff
std::string err;
ret = dng_writer.WriteToFile(output_filename.c_str(), &err);
if (!err.empty()) {
std::cerr << err;
}
if (!ret) {
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}