123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470 |
- #if defined(_WIN32)
- #ifndef NOMINMAX
- #define NOMINMAX
- #endif
- #include <tchar.h>
- #include <windows.h>
- #endif
- #include <cstdio>
- #include <cstdlib>
- #include <iostream>
- #include <vector>
- // Uncomment if you want to use system provided zlib.
- // #define TINYEXR_USE_MINIZ (0)
- // #include <zlib.h>
- #define TINYEXR_IMPLEMENTATION
- #include "tinyexr.h"
- #ifdef __clang__
- #if __has_warning("-Wzero-as-null-pointer-constant")
- #pragma clang diagnostic ignored "-Wzero-as-null-pointer-constant"
- #endif
- #endif
- #define SIMPLE_API_EXAMPLE
- //#define TEST_ZFP_COMPRESSION
- #ifdef SIMPLE_API_EXAMPLE
- #if 0
- static void
- SaveAsPFM(const char* filename, int width, int height, float* data)
- {
- #ifdef _WIN32
- FILE* fp = NULL;
- fopen_s(&fp, filename, "wb");
- #else
- FILE* fp = fopen(filename, "wb");
- #endif
- if (!fp) {
- fprintf(stderr, "failed to write a PFM file.\n");
- return;
- }
- fprintf(fp, "PF\n");
- fprintf(fp, "%d %d\n", width, height);
- fprintf(fp, "-1\n"); // -1: little endian, 1: big endian
- // RGBA -> RGB
- std::vector<float> rgb(static_cast<size_t>(width*height*3));
- for (size_t i = 0; i < static_cast<size_t>(width * height); i++) {
- rgb[3*i+0] = data[4*i+0];
- rgb[3*i+1] = data[4*i+1];
- rgb[3*i+2] = data[4*i+2];
- }
- fwrite(&rgb.at(0), sizeof(float), static_cast<size_t>(width * height * 3), fp);
- fclose(fp);
- }
- #endif
- #else
- static const char* GetPixelType(int id) {
- if (id == TINYEXR_PIXELTYPE_HALF) {
- return "HALF";
- } else if (id == TINYEXR_PIXELTYPE_FLOAT) {
- return "FLOAT";
- } else if (id == TINYEXR_PIXELTYPE_UINT) {
- return "UINT";
- }
- return "???";
- }
- // Simple tile -> scanline converter. Assumes FLOAT pixel type for all channels.
- static void TiledImageToScanlineImage(EXRImage* src, const EXRHeader* header) {
- assert(header->data_window.max_x - header->data_window.min_x + 1 >= 0);
- assert(header->data_window.max_y - header->data_window.min_y + 1 >= 0);
- size_t data_width =
- static_cast<size_t>(header->data_window.max_x - header->data_window.min_x + 1);
- size_t data_height =
- static_cast<size_t>(header->data_window.max_y - header->data_window.min_y + 1);
- src->images = static_cast<unsigned char**>(
- malloc(sizeof(float*) * static_cast<size_t>(header->num_channels)));
- for (size_t c = 0; c < static_cast<size_t>(header->num_channels); c++) {
- assert(header->pixel_types[c] == TINYEXR_PIXELTYPE_FLOAT);
- src->images[c] = static_cast<unsigned char*>(
- malloc(sizeof(float) * data_width * data_height));
- memset(src->images[c], 0, sizeof(float) * data_width * data_height);
- }
- for (size_t tile_idx = 0; tile_idx < static_cast<size_t>(src->num_tiles);
- tile_idx++) {
- size_t sx = static_cast<size_t>(src->tiles[tile_idx].offset_x *
- header->tile_size_x);
- size_t sy = static_cast<size_t>(src->tiles[tile_idx].offset_y *
- header->tile_size_y);
- size_t ex = static_cast<size_t>(src->tiles[tile_idx].offset_x *
- header->tile_size_x +
- src->tiles[tile_idx].width);
- size_t ey = static_cast<size_t>(src->tiles[tile_idx].offset_y *
- header->tile_size_y +
- src->tiles[tile_idx].height);
- for (size_t c = 0; c < static_cast<size_t>(header->num_channels); c++) {
- float* dst_image = reinterpret_cast<float*>(src->images[c]);
- const float* src_image =
- reinterpret_cast<const float*>(src->tiles[tile_idx].images[c]);
- for (size_t y = 0; y < static_cast<size_t>(ey - sy); y++) {
- for (size_t x = 0; x < static_cast<size_t>(ex - sx); x++) {
- dst_image[(y + sy) * data_width + (x + sx)] =
- src_image[y * static_cast<size_t>(header->tile_size_x) + x];
- }
- }
- }
- }
- }
- #endif
- #if defined(_WIN32)
- #if defined(__MINGW32__)
- // __wgetmainargs is not defined in windows.h
- extern "C" int __wgetmainargs(int*, wchar_t***, wchar_t***, int, int*);
- #endif
- // https://gist.github.com/trueroad/fb4d0c3f67285bf66804
- namespace {
- std::vector<char> utf16_to_utf8(const wchar_t* wc) {
- int size = WideCharToMultiByte(CP_UTF8, 0, wc, -1, NULL, 0, NULL, NULL);
- std::vector<char> retval(size);
- if (size) {
- WideCharToMultiByte(CP_UTF8, 0, wc, -1, retval.data(), retval.size(), NULL,
- NULL);
- } else
- retval.push_back('\0');
- return retval;
- }
- } // namespace
- #endif
- static int test_main(int argc, char** argv);
- #if defined(_WIN32)
- #if defined(__MINGW32__)
- int main() {
- wchar_t** wargv;
- wchar_t** wenpv;
- int argc = 0, si = 0;
- __wgetmainargs(&argc, &wargv, &wenpv, 1, &si);
- std::vector<std::vector<char> > argv_vvc(argc);
- std::vector<char*> argv_vc(argc);
- for (int i = 0; i < argc; i++) {
- argv_vvc.at(i) = utf16_to_utf8(wargv[i]);
- argv_vc.at(i) = argv_vvc.at(i).data();
- }
- // TODO(syoyo): envp
- return test_main(argc, argv_vc.data());
- }
- #else // Assume MSVC
- int _tmain(int argc, _TCHAR** wargv) {
- std::vector<std::vector<char> > argv_vvc(argc);
- std::vector<char*> argv_vc(argc);
- for (int i = 0; i < argc; i++) {
- #if defined(UNICODE) || defined(_UNICODE)
- argv_vvc.at(i) = utf16_to_utf8(wargv[i]);
- #else
- size_t slen = _tcslen(wargv[i]);
- std::vector<char> buf(slen + 1);
- memcpy(buf.data(), wargv[i], slen);
- buf[slen] = '\0';
- argv_vvc.at(i) = buf;
- #endif
- argv_vc.at(i) = argv_vvc.at(i).data();
- }
- return test_main(argc, argv_vc.data());
- }
- #endif
- #else
- int main(int argc, char** argv) { return test_main(argc, argv); }
- #endif
- int test_main(int argc, char** argv) {
- const char* outfilename = "output_test.exr";
- const char* err = NULL;
- if (argc < 2) {
- fprintf(stderr, "Needs input.exr.\n");
- exit(-1);
- }
- if (argc > 2) {
- outfilename = argv[2];
- }
- const char* input_filename = argv[1];
- #ifdef SIMPLE_API_EXAMPLE
- (void)outfilename;
- int width, height;
- float* image;
- int ret = IsEXR(input_filename);
- if (ret != TINYEXR_SUCCESS) {
- fprintf(stderr, "Header err. code %d\n", ret);
- exit(-1);
- }
- ret = LoadEXR(&image, &width, &height, input_filename, &err);
- if (ret != TINYEXR_SUCCESS) {
- if (err) {
- fprintf(stderr, "Load EXR err: %s(code %d)\n", err, ret);
- } else {
- fprintf(stderr, "Load EXR err: code = %d\n", ret);
- }
- FreeEXRErrorMessage(err);
- return ret;
- }
- // SaveAsPFM("output.pfm", width, height, image);
- ret = SaveEXR(image, width, height, 4 /* =RGBA*/,
- 1 /* = save as fp16 format */, "output.exr", &err);
- if (ret != TINYEXR_SUCCESS) {
- if (err) {
- fprintf(stderr, "Save EXR err: %s(code %d)\n", err, ret);
- } else {
- fprintf(stderr, "Failed to save EXR image. code = %d\n", ret);
- }
- }
- free(image);
- std::cout << "Wrote output.exr." << std::endl;
- #else
- EXRVersion exr_version;
- int ret = ParseEXRVersionFromFile(&exr_version, input_filename);
- if (ret != 0) {
- fprintf(stderr, "Invalid EXR file: %s\n", input_filename);
- return -1;
- }
- printf(
- "version: tiled = %d, long_name = %d, non_image = %d, multipart = %d\n",
- exr_version.tiled, exr_version.long_name, exr_version.non_image,
- exr_version.multipart);
- if (exr_version.multipart) {
- EXRHeader** exr_headers; // list of EXRHeader pointers.
- int num_exr_headers;
- ret = ParseEXRMultipartHeaderFromFile(&exr_headers, &num_exr_headers,
- &exr_version, argv[1], &err);
- if (ret != 0) {
- fprintf(stderr, "Parse EXR err: %s\n", err);
- return ret;
- }
- printf("num parts = %d\n", num_exr_headers);
- for (size_t i = 0; i < static_cast<size_t>(num_exr_headers); i++) {
- const EXRHeader& exr_header = *(exr_headers[i]);
- printf("Part: %lu\n", static_cast<unsigned long>(i));
- printf("dataWindow = %d, %d, %d, %d\n", exr_header.data_window.min_x,
- exr_header.data_window.min_y, exr_header.data_window.max_x,
- exr_header.data_window.max_y);
- printf("displayWindow = %d, %d, %d, %d\n", exr_header.display_window.min_x,
- exr_header.display_window.min_y, exr_header.display_window.max_x,
- exr_header.display_window.max_y);
- printf("screenWindowCenter = %f, %f\n",
- static_cast<double>(exr_header.screen_window_center[0]),
- static_cast<double>(exr_header.screen_window_center[1]));
- printf("screenWindowWidth = %f\n",
- static_cast<double>(exr_header.screen_window_width));
- printf("pixelAspectRatio = %f\n",
- static_cast<double>(exr_header.pixel_aspect_ratio));
- printf("lineOrder = %d\n", exr_header.line_order);
- if (exr_header.num_custom_attributes > 0) {
- printf("# of custom attributes = %d\n",
- exr_header.num_custom_attributes);
- for (int a = 0; a < exr_header.num_custom_attributes; a++) {
- printf(" [%d] name = %s, type = %s, size = %d\n", a,
- exr_header.custom_attributes[a].name,
- exr_header.custom_attributes[a].type,
- exr_header.custom_attributes[a].size);
- // if (strcmp(exr_header.custom_attributes[i].type, "float") == 0) {
- // printf(" value = %f\n", *reinterpret_cast<float
- // *>(exr_header.custom_attributes[i].value));
- //}
- }
- }
- }
- std::vector<EXRImage> images(static_cast<size_t>(num_exr_headers));
- for (size_t i = 0; i < static_cast<size_t>(num_exr_headers); i++) {
- InitEXRImage(&images[i]);
- }
- ret = LoadEXRMultipartImageFromFile(
- &images.at(0), const_cast<const EXRHeader**>(exr_headers),
- static_cast<unsigned int>(num_exr_headers), input_filename, &err);
- if (ret != 0) {
- fprintf(stderr, "Load EXR err: %s\n", err);
- FreeEXRErrorMessage(err);
- return ret;
- }
- printf("Loaded %d part images\n", num_exr_headers);
- printf(
- "There is no saving feature for multi-part images, thus just exit an "
- "application...\n");
- for (size_t i = 0; i < static_cast<size_t>(num_exr_headers); i++) {
- FreeEXRImage(&images.at(i));
- }
- for (size_t i = 0; i < static_cast<size_t>(num_exr_headers); i++) {
- FreeEXRHeader(exr_headers[i]);
- free(exr_headers[i]);
- }
- free(exr_headers);
- } else { // single-part EXR
- EXRHeader exr_header;
- InitEXRHeader(&exr_header);
- ret =
- ParseEXRHeaderFromFile(&exr_header, &exr_version, input_filename, &err);
- if (ret != 0) {
- fprintf(stderr, "Parse single-part EXR err: %s\n", err);
- FreeEXRErrorMessage(err);
- return ret;
- }
- printf("dataWindow = %d, %d, %d, %d\n", exr_header.data_window.min_x,
- exr_header.data_window.min_y, exr_header.data_window.max_x,
- exr_header.data_window.max_y);
- printf("displayWindow = %d, %d, %d, %d\n", exr_header.display_window.min_x,
- exr_header.display_window.min_y, exr_header.display_window.max_x,
- exr_header.display_window.max_y);
- printf("screenWindowCenter = %f, %f\n",
- static_cast<double>(exr_header.screen_window_center[0]),
- static_cast<double>(exr_header.screen_window_center[1]));
- printf("screenWindowWidth = %f\n",
- static_cast<double>(exr_header.screen_window_width));
- printf("pixelAspectRatio = %f\n",
- static_cast<double>(exr_header.pixel_aspect_ratio));
- printf("lineOrder = %d\n", exr_header.line_order);
- if (exr_header.num_custom_attributes > 0) {
- printf("# of custom attributes = %d\n", exr_header.num_custom_attributes);
- for (int i = 0; i < exr_header.num_custom_attributes; i++) {
- printf(" [%d] name = %s, type = %s, size = %d\n", i,
- exr_header.custom_attributes[i].name,
- exr_header.custom_attributes[i].type,
- exr_header.custom_attributes[i].size);
- // if (strcmp(exr_header.custom_attributes[i].type, "float") == 0) {
- // printf(" value = %f\n", *reinterpret_cast<float
- // *>(exr_header.custom_attributes[i].value));
- //}
- }
- }
- // Read HALF channel as FLOAT.
- for (int i = 0; i < exr_header.num_channels; i++) {
- if (exr_header.pixel_types[i] == TINYEXR_PIXELTYPE_HALF) {
- exr_header.requested_pixel_types[i] = TINYEXR_PIXELTYPE_FLOAT;
- }
- }
- EXRImage exr_image;
- InitEXRImage(&exr_image);
- ret = LoadEXRImageFromFile(&exr_image, &exr_header, input_filename, &err);
- if (ret != 0) {
- fprintf(stderr, "Load EXR err: %s\n", err);
- FreeEXRHeader(&exr_header);
- FreeEXRErrorMessage(err);
- return ret;
- }
- printf("EXR: %d x %d\n", exr_image.width, exr_image.height);
- for (int i = 0; i < exr_header.num_channels; i++) {
- printf("pixelType[%d]: %s\n", i, GetPixelType(exr_header.pixel_types[i]));
- printf("chan[%d] = %s\n", i, exr_header.channels[i].name);
- printf("requestedPixelType[%d]: %s\n", i,
- GetPixelType(exr_header.requested_pixel_types[i]));
- }
- #if 0 // example to write custom attribute
- int version_minor = 3;
- exr_header.num_custom_attributes = 1;
- exr_header.custom_attributes = reinterpret_cast<EXRAttribute *>(malloc(sizeof(EXRAttribute) * exr_header.custom_attributes));
- strcpy(exr_header.custom_attributes[0].name, "tinyexr_version_minor");
- exr_header.custom_attributes[0].name[strlen("tinyexr_version_minor")] = '\0';
- strcpy(exr_header.custom_attributes[0].type, "int");
- exr_header.custom_attributes[0].type[strlen("int")] = '\0';
- exr_header.custom_attributes[0].size = sizeof(int);
- exr_header.custom_attributes[0].value = (unsigned char*)malloc(sizeof(int));
- memcpy(exr_header.custom_attributes[0].value, &version_minor, sizeof(int));
- #endif
- if (exr_header.tiled) {
- TiledImageToScanlineImage(&exr_image, &exr_header);
- }
- exr_header.compression_type = TINYEXR_COMPRESSIONTYPE_NONE;
- #ifdef TEST_ZFP_COMPRESSION
- // Assume input image is FLOAT pixel type.
- for (int i = 0; i < exr_header.num_channels; i++) {
- exr_header.channels[i].pixel_type = TINYEXR_PIXELTYPE_FLOAT;
- exr_header.requested_pixel_types[i] = TINYEXR_PIXELTYPE_FLOAT;
- }
- unsigned char zfp_compression_type = TINYEXR_ZFP_COMPRESSIONTYPE_RATE;
- double zfp_compression_rate = 4;
- exr_header.num_custom_attributes = 2;
- strcpy(exr_header.custom_attributes[0].name, "zfpCompressionType");
- exr_header.custom_attributes[0].name[strlen("zfpCompressionType")] = '\0';
- exr_header.custom_attributes[0].size = 1;
- exr_header.custom_attributes[0].value =
- (unsigned char*)malloc(sizeof(unsigned char));
- exr_header.custom_attributes[0].value[0] = zfp_compression_type;
- strcpy(exr_header.custom_attributes[1].name, "zfpCompressionRate");
- exr_header.custom_attributes[1].name[strlen("zfpCompressionRate")] = '\0';
- exr_header.custom_attributes[1].size = sizeof(double);
- exr_header.custom_attributes[1].value =
- (unsigned char*)malloc(sizeof(double));
- memcpy(exr_header.custom_attributes[1].value, &zfp_compression_rate,
- sizeof(double));
- exr_header.compression_type = TINYEXR_COMPRESSIONTYPE_ZFP;
- #endif
- ret = SaveEXRImageToFile(&exr_image, &exr_header, outfilename, &err);
- if (ret != 0) {
- fprintf(stderr, "Save EXR err: %s\n", err);
- FreeEXRHeader(&exr_header);
- FreeEXRErrorMessage(err);
- return ret;
- }
- printf("Saved exr file. [ %s ] \n", outfilename);
- FreeEXRHeader(&exr_header);
- FreeEXRImage(&exr_image);
- }
- #endif
- return ret;
- }
|