From: José Fonseca Date: Sat, 4 Dec 2010 13:13:47 +0000 (+0000) Subject: Write PNG images instead. X-Git-Url: https://git.cworth.org/git?a=commitdiff_plain;h=0a91a3e83760539edc00b80cfb4338c95dfdedf2;p=apitrace Write PNG images instead. --- diff --git a/.gitignore b/.gitignore index 6e2b74c..3673b52 100644 --- a/.gitignore +++ b/.gitignore @@ -12,6 +12,7 @@ *.o *.obj *.pdb +*.png *.pyc *.pyo *.so diff --git a/glretrace.py b/glretrace.py index bee6728..28b47c1 100644 --- a/glretrace.py +++ b/glretrace.py @@ -223,10 +223,10 @@ static void frame_complete(void) { if (__screenshots && !__reshape_window) { char filename[PATH_MAX]; - snprintf(filename, sizeof filename, "screenshot_%04u.bmp", __frame); + snprintf(filename, sizeof filename, "screenshot_%04u.png", __frame); Image::Image image(__window_width, __window_height, true); glReadPixels(0, 0, __window_width, __window_height, GL_RGBA, GL_UNSIGNED_BYTE, image.pixels); - image.writeBMP(filename); + image.writePNG(filename); } } diff --git a/image.cpp b/image.cpp index 6cd39b2..7cb7c02 100644 --- a/image.cpp +++ b/image.cpp @@ -24,6 +24,8 @@ **************************************************************************/ +#include + #include #include @@ -132,5 +134,79 @@ Image::writeBMP(const char *filename) const { return true; } +bool +Image::writePNG(const char *filename) const { + FILE *fp; + png_structp png_ptr; + png_infop info_ptr; + + /* Open the file */ + fp = fopen(filename, "wb"); + if (!fp) + goto no_fp; + + /* Create and initialize the png_struct with the desired error handler + * functions. If you want to use the default stderr and longjump method, + * you can supply NULL for the last three parameters. We also check that + * the library version is compatible with the one used at compile time, + * in case we are using dynamically linked libraries. REQUIRED. + */ + png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); + if (!png_ptr) + goto no_png; + + /* Allocate/initialize the image information data. REQUIRED */ + info_ptr = png_create_info_struct(png_ptr); + if (!info_ptr) { + png_destroy_write_struct(&png_ptr, NULL); + goto no_png; + } + + /* Set error handling. REQUIRED if you aren't supplying your own + * error handling functions in the png_create_write_struct() call. + */ + if (setjmp(png_jmpbuf(png_ptr))) { + png_destroy_write_struct(&png_ptr, &info_ptr); + goto no_png; + } + +#if 1 + png_init_io(png_ptr, fp); +#else + png_set_write_fn(png_ptr, (void *)user_io_ptr, user_write_fn, + user_IO_flush_function); +#endif + + png_set_IHDR(png_ptr, info_ptr, width, height, 8, PNG_COLOR_TYPE_RGB_ALPHA, + PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE); + + png_set_compression_level(png_ptr, Z_BEST_COMPRESSION); + + png_write_info(png_ptr, info_ptr); + + if (!flipped) { + for (unsigned y = 0; y < height; ++y) { + png_byte *row = (png_byte *)(pixels + y*width*4); + png_write_rows(png_ptr, &row, 1); + } + } else { + unsigned y = height; + while (y--) { + png_byte *row = (png_byte *)(pixels + y*width*4); + png_write_rows(png_ptr, &row, 1); + } + } + + png_write_end(png_ptr, info_ptr); + png_destroy_write_struct(&png_ptr, &info_ptr); + + fclose(fp); + return true; + +no_png: + fclose(fp); +no_fp: + return false; +} } /* namespace Image */ diff --git a/image.hpp b/image.hpp index 4d57de1..190b093 100644 --- a/image.hpp +++ b/image.hpp @@ -60,6 +60,7 @@ public: } bool writeBMP(const char *filename) const; + bool writePNG(const char *filename) const; };