Open Source Tomb Raider Engine
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.

png.cpp 7.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. /*!
  2. * \file src/utils/png.cpp
  3. * \brief PNG image reader
  4. *
  5. * https://github.com/DavidEGrayson/ahrs-visualizer/blob/master/png_texture.cpp
  6. * http://zarb.org/~gc/html/libpng.html
  7. *
  8. * \author xythobuz
  9. */
  10. #include <png.h>
  11. #include <cstdio>
  12. #include <iostream>
  13. #include "global.h"
  14. #include "utils/pixel.h"
  15. #include "utils/png.h"
  16. int pngCheck(const char* filename) {
  17. png_byte header[8];
  18. assert(filename != nullptr);
  19. assert(filename[0] != '\0');
  20. FILE* fp = fopen(filename, "rb");
  21. if (fp == nullptr) {
  22. std::cout << "Could not open " << filename << std::endl;
  23. return -1;
  24. }
  25. fread(header, 1, 8, fp);
  26. fclose(fp);
  27. if (png_sig_cmp(header, 0, 8)) {
  28. std::cout << "File " << filename << " is not a PNG." << std::endl;
  29. return -2;
  30. }
  31. return 0;
  32. }
  33. int pngLoad(const char* filename, unsigned char** image,
  34. unsigned int* width, unsigned int* height,
  35. TextureManager::ColorMode* mode, unsigned int* bpp) {
  36. png_byte header[8];
  37. assert(filename != nullptr);
  38. assert(filename[0] != '\0');
  39. assert(image != nullptr);
  40. assert(width != nullptr);
  41. assert(height != nullptr);
  42. assert(mode != nullptr);
  43. assert(bpp != nullptr);
  44. FILE* fp = fopen(filename, "rb");
  45. if (fp == nullptr) {
  46. std::cout << "Could not open " << filename << std::endl;
  47. return -1;
  48. }
  49. fread(header, 1, 8, fp);
  50. if (png_sig_cmp(header, 0, 8)) {
  51. std::cout << "File " << filename << " is not a PNG." << std::endl;
  52. fclose(fp);
  53. return -2;
  54. }
  55. png_structp png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
  56. if (!png_ptr) {
  57. std::cout << "png_create_read_struct returned 0." << std::endl;
  58. fclose(fp);
  59. return -3;
  60. }
  61. png_infop info_ptr = png_create_info_struct(png_ptr);
  62. if (!info_ptr) {
  63. std::cout << "png_create_info_struct returned 0." << std::endl;
  64. png_destroy_read_struct(&png_ptr, (png_infopp)nullptr, (png_infopp)nullptr);
  65. fclose(fp);
  66. return -4;
  67. }
  68. png_infop end_info = png_create_info_struct(png_ptr);
  69. if (!end_info) {
  70. std::cout << "png_create_info_struct returned 0." << std::endl;
  71. png_destroy_read_struct(&png_ptr, &info_ptr, (png_infopp) nullptr);
  72. fclose(fp);
  73. return -5;
  74. }
  75. if (setjmp(png_jmpbuf(png_ptr))) {
  76. std::cout << "Error from libpng" << std::endl;
  77. png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
  78. fclose(fp);
  79. return -6;
  80. }
  81. png_init_io(png_ptr, fp);
  82. png_set_sig_bytes(png_ptr, 8);
  83. png_read_info(png_ptr, info_ptr);
  84. int bit_depth, color_type;
  85. png_uint_32 tmpWidth, tmpHeight;
  86. png_get_IHDR(png_ptr, info_ptr, &tmpWidth, &tmpHeight, &bit_depth, &color_type,
  87. nullptr, nullptr, nullptr);
  88. *width = tmpWidth;
  89. *height = tmpHeight;
  90. if (bit_depth != 8) {
  91. std::cout << filename << ": Unsupported bit depth " << bit_depth << ". Must be 8." << std::endl;
  92. return -7;
  93. }
  94. png_read_update_info(png_ptr, info_ptr);
  95. png_size_t rowbytes = png_get_rowbytes(png_ptr, info_ptr);
  96. png_byte* image_data = new png_byte[(rowbytes** height) + 15];
  97. png_bytep* row_pointers = new png_bytep[*height];
  98. for (unsigned int i = 0; i < *height; i++)
  99. row_pointers[*height - 1 - i] = image_data + i * rowbytes;
  100. png_read_image(png_ptr, row_pointers);
  101. png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
  102. delete [] row_pointers;
  103. fclose(fp);
  104. if (color_type == PNG_COLOR_TYPE_RGB) {
  105. *mode = TextureManager::ColorMode::RGB;
  106. *bpp = 24;
  107. } else if (color_type == PNG_COLOR_TYPE_RGB_ALPHA) {
  108. *mode = TextureManager::ColorMode::RGBA;
  109. *bpp = 32;
  110. } else {
  111. std::cout << filename << ": Unknown libpng color type " << color_type << std::endl;
  112. delete [] image_data;
  113. return -8;
  114. }
  115. // Flip
  116. *image = new unsigned char[*width** height * (*bpp / 8)];
  117. for (unsigned int y = 0; y < (*height); y++) {
  118. for (unsigned int x = 0; x < (*width); x++) {
  119. (*image)[((y** width) + x) * (*bpp / 8)]
  120. = image_data[(((*height - y - 1)** width) + x) * (*bpp / 8)];
  121. (*image)[(((y** width) + x) * (*bpp / 8)) + 1]
  122. = image_data[((((*height - y - 1)** width) + x) * (*bpp / 8)) + 1];
  123. (*image)[(((y** width) + x) * (*bpp / 8)) + 2]
  124. = image_data[((((*height - y - 1)** width) + x) * (*bpp / 8)) + 2];
  125. (*image)[(((y** width) + x) * (*bpp / 8)) + 3]
  126. = image_data[((((*height - y - 1)** width) + x) * (*bpp / 8)) + 3];
  127. }
  128. }
  129. delete [] image_data;
  130. return 0;
  131. }
  132. int pngSave(const char* filename, unsigned char* image,
  133. unsigned int width, unsigned int height,
  134. TextureManager::ColorMode mode, unsigned int bpp) {
  135. assert(filename != nullptr);
  136. assert(filename[0] != '\0');
  137. assert(image != nullptr);
  138. assert(width > 0);
  139. assert(height > 0);
  140. FILE* fp = fopen(filename, "wb");
  141. if (!fp) {
  142. std::cout << "File " << filename << " could not be opened for writing" << std::endl;
  143. return -1;
  144. }
  145. png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, nullptr, nullptr);
  146. if (!png_ptr) {
  147. std::cout << "png_create_write_struct failed" << std::endl;
  148. fclose(fp);
  149. return -2;
  150. }
  151. png_infop info_ptr = png_create_info_struct(png_ptr);
  152. if (!info_ptr) {
  153. std::cout << "png_create_info_struct failed" << std::endl;
  154. fclose(fp);
  155. return -3;
  156. }
  157. if (setjmp(png_jmpbuf(png_ptr))) {
  158. std::cout << "Error during init_io" << std::endl;
  159. fclose(fp);
  160. return -4;
  161. }
  162. png_init_io(png_ptr, fp);
  163. if (setjmp(png_jmpbuf(png_ptr))) {
  164. std::cout << "Error during writing header" << std::endl;
  165. fclose(fp);
  166. return -5;
  167. }
  168. int color_type;
  169. if (((mode == TextureManager::ColorMode::RGB)
  170. || (mode == TextureManager::ColorMode::BGR)) && (bpp == 24)) {
  171. if (mode == TextureManager::ColorMode::BGR) {
  172. bgr2rgb24(image, width, height);
  173. }
  174. color_type = PNG_COLOR_TYPE_RGB;
  175. } else if (((mode == TextureManager::ColorMode::RGBA)
  176. || (mode == TextureManager::ColorMode::BGRA)) && (bpp == 32)) {
  177. if (mode == TextureManager::ColorMode::BGRA) {
  178. bgra2rgba32(image, width, height);
  179. }
  180. color_type = PNG_COLOR_TYPE_RGB_ALPHA;
  181. } else {
  182. std::cout << "Error invalid color mode" << std::endl;
  183. fclose(fp);
  184. return -6;
  185. }
  186. png_set_IHDR(png_ptr, info_ptr, width, height,
  187. 8, color_type, PNG_INTERLACE_NONE,
  188. PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
  189. png_write_info(png_ptr, info_ptr);
  190. png_bytep* row_pointers = new png_bytep[height];
  191. for (unsigned int i = 0; i < height; i++)
  192. row_pointers[height - 1 - i] = image + (i * width * 4);
  193. if (setjmp(png_jmpbuf(png_ptr))) {
  194. std::cout << "Error during writing bytes" << std::endl;
  195. delete [] row_pointers;
  196. fclose(fp);
  197. return -7;
  198. }
  199. png_write_image(png_ptr, row_pointers);
  200. if (setjmp(png_jmpbuf(png_ptr))) {
  201. std::cout << "Error during end of write" << std::endl;
  202. delete [] row_pointers;
  203. fclose(fp);
  204. return -8;
  205. }
  206. png_write_end(png_ptr, nullptr);
  207. delete [] row_pointers;
  208. fclose(fp);
  209. return 0;
  210. }