diff options
author | José Fonseca <jfonseca@vmware.com> | 2010-12-04 13:13:47 +0000 |
---|---|---|
committer | José Fonseca <jfonseca@vmware.com> | 2010-12-04 13:13:47 +0000 |
commit | 0a91a3e83760539edc00b80cfb4338c95dfdedf2 (patch) | |
tree | db4f9e97a25a89f24dd82affc8766754651e8752 | |
parent | bdecd71b96b839c9a5a8e9a639eef497e00e1f7b (diff) |
Write PNG images instead.
-rw-r--r-- | .gitignore | 1 | ||||
-rw-r--r-- | glretrace.py | 4 | ||||
-rw-r--r-- | image.cpp | 76 | ||||
-rw-r--r-- | image.hpp | 1 |
4 files changed, 80 insertions, 2 deletions
@@ -12,6 +12,7 @@ *.o *.obj *.pdb +*.png *.pyc *.pyo *.so diff --git a/glretrace.py b/glretrace.py index bee6728b..28b47c15 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); } } @@ -24,6 +24,8 @@ **************************************************************************/ +#include <png.h> + #include <stdint.h> #include <fstream> @@ -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 */ @@ -60,6 +60,7 @@ public: } bool writeBMP(const char *filename) const; + bool writePNG(const char *filename) const; }; |