stb_image_write.h 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056
  1. /* stb_image_write - v1.02 - public domain - http://nothings.org/stb/stb_image_write.h
  2. writes out PNG/BMP/TGA images to C stdio - Sean Barrett 2010-2015
  3. no warranty implied; use at your own risk
  4. Before #including,
  5. #define STB_IMAGE_WRITE_IMPLEMENTATION
  6. in the file that you want to have the implementation.
  7. Will probably not work correctly with strict-aliasing optimizations.
  8. ABOUT:
  9. This header file is a library for writing images to C stdio. It could be
  10. adapted to write to memory or a general streaming interface; let me know.
  11. The PNG output is not optimal; it is 20-50% larger than the file
  12. written by a decent optimizing implementation. This library is designed
  13. for source code compactness and simplicity, not optimal image file size
  14. or run-time performance.
  15. BUILDING:
  16. You can #define STBIW_ASSERT(x) before the #include to avoid using assert.h.
  17. You can #define STBIW_MALLOC(), STBIW_REALLOC(), and STBIW_FREE() to replace
  18. malloc,realloc,free.
  19. You can define STBIW_MEMMOVE() to replace memmove()
  20. USAGE:
  21. There are four functions, one for each image file format:
  22. int stbi_write_png(char const *filename, int w, int h, int comp, const void *data, int stride_in_bytes);
  23. int stbi_write_bmp(char const *filename, int w, int h, int comp, const void *data);
  24. int stbi_write_tga(char const *filename, int w, int h, int comp, const void *data);
  25. int stbi_write_hdr(char const *filename, int w, int h, int comp, const float *data);
  26. There are also four equivalent functions that use an arbitrary write function. You are
  27. expected to open/close your file-equivalent before and after calling these:
  28. int stbi_write_png_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data, int stride_in_bytes);
  29. int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data);
  30. int stbi_write_tga_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data);
  31. int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const float *data);
  32. where the callback is:
  33. void stbi_write_func(void *context, void *data, int size);
  34. You can define STBI_WRITE_NO_STDIO to disable the file variant of these
  35. functions, so the library will not use stdio.h at all. However, this will
  36. also disable HDR writing, because it requires stdio for formatted output.
  37. Each function returns 0 on failure and non-0 on success.
  38. The functions create an image file defined by the parameters. The image
  39. is a rectangle of pixels stored from left-to-right, top-to-bottom.
  40. Each pixel contains 'comp' channels of data stored interleaved with 8-bits
  41. per channel, in the following order: 1=Y, 2=YA, 3=RGB, 4=RGBA. (Y is
  42. monochrome color.) The rectangle is 'w' pixels wide and 'h' pixels tall.
  43. The *data pointer points to the first byte of the top-left-most pixel.
  44. For PNG, "stride_in_bytes" is the distance in bytes from the first byte of
  45. a row of pixels to the first byte of the next row of pixels.
  46. PNG creates output files with the same number of components as the input.
  47. The BMP format expands Y to RGB in the file format and does not
  48. output alpha.
  49. PNG supports writing rectangles of data even when the bytes storing rows of
  50. data are not consecutive in memory (e.g. sub-rectangles of a larger image),
  51. by supplying the stride between the beginning of adjacent rows. The other
  52. formats do not. (Thus you cannot write a native-format BMP through the BMP
  53. writer, both because it is in BGR order and because it may have padding
  54. at the end of the line.)
  55. HDR expects linear float data. Since the format is always 32-bit rgb(e)
  56. data, alpha (if provided) is discarded, and for monochrome data it is
  57. replicated across all three channels.
  58. TGA supports RLE or non-RLE compressed data. To use non-RLE-compressed
  59. data, set the global variable 'stbi_write_tga_with_rle' to 0.
  60. CREDITS:
  61. PNG/BMP/TGA
  62. Sean Barrett
  63. HDR
  64. Baldur Karlsson
  65. TGA monochrome:
  66. Jean-Sebastien Guay
  67. misc enhancements:
  68. Tim Kelsey
  69. TGA RLE
  70. Alan Hickman
  71. initial file IO callback implementation
  72. Emmanuel Julien
  73. bugfixes:
  74. github:Chribba
  75. Guillaume Chereau
  76. github:jry2
  77. github:romigrou
  78. Sergio Gonzalez
  79. Jonas Karlsson
  80. Filip Wasil
  81. Thatcher Ulrich
  82. LICENSE
  83. This software is dual-licensed to the public domain and under the following
  84. license: you are granted a perpetual, irrevocable license to copy, modify,
  85. publish, and distribute this file as you see fit.
  86. */
  87. // Modified by Lasse Oorni for Urho3D
  88. #ifndef INCLUDE_STB_IMAGE_WRITE_H
  89. #define INCLUDE_STB_IMAGE_WRITE_H
  90. #ifdef __cplusplus
  91. extern "C" {
  92. #endif
  93. #ifdef STB_IMAGE_WRITE_STATIC
  94. #define STBIWDEF static
  95. #else
  96. #define STBIWDEF extern
  97. extern int stbi_write_tga_with_rle;
  98. #endif
  99. #ifndef STBI_WRITE_NO_STDIO
  100. STBIWDEF int stbi_write_png(char const *filename, int w, int h, int comp, const void *data, int stride_in_bytes);
  101. STBIWDEF int stbi_write_bmp(char const *filename, int w, int h, int comp, const void *data);
  102. STBIWDEF int stbi_write_tga(char const *filename, int w, int h, int comp, const void *data);
  103. STBIWDEF int stbi_write_hdr(char const *filename, int w, int h, int comp, const float *data);
  104. #endif
  105. typedef void stbi_write_func(void *context, void *data, int size);
  106. STBIWDEF int stbi_write_png_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data, int stride_in_bytes);
  107. STBIWDEF int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data);
  108. STBIWDEF int stbi_write_tga_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data);
  109. STBIWDEF int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const float *data);
  110. #ifdef __cplusplus
  111. }
  112. #endif
  113. #endif//INCLUDE_STB_IMAGE_WRITE_H
  114. #ifdef STB_IMAGE_WRITE_IMPLEMENTATION
  115. #ifdef _WIN32
  116. #ifndef _CRT_SECURE_NO_WARNINGS
  117. #define _CRT_SECURE_NO_WARNINGS
  118. #endif
  119. #ifndef _CRT_NONSTDC_NO_DEPRECATE
  120. #define _CRT_NONSTDC_NO_DEPRECATE
  121. #endif
  122. #endif
  123. #ifndef STBI_WRITE_NO_STDIO
  124. #include <stdio.h>
  125. #endif // STBI_WRITE_NO_STDIO
  126. #include <stdarg.h>
  127. #include <stdlib.h>
  128. #include <string.h>
  129. #include <math.h>
  130. #if defined(STBIW_MALLOC) && defined(STBIW_FREE) && (defined(STBIW_REALLOC) || defined(STBIW_REALLOC_SIZED))
  131. // ok
  132. #elif !defined(STBIW_MALLOC) && !defined(STBIW_FREE) && !defined(STBIW_REALLOC) && !defined(STBIW_REALLOC_SIZED)
  133. // ok
  134. #else
  135. #error "Must define all or none of STBIW_MALLOC, STBIW_FREE, and STBIW_REALLOC (or STBIW_REALLOC_SIZED)."
  136. #endif
  137. #ifndef STBIW_MALLOC
  138. #define STBIW_MALLOC(sz) malloc(sz)
  139. #define STBIW_REALLOC(p,newsz) realloc(p,newsz)
  140. #define STBIW_FREE(p) free(p)
  141. #endif
  142. #ifndef STBIW_REALLOC_SIZED
  143. #define STBIW_REALLOC_SIZED(p,oldsz,newsz) STBIW_REALLOC(p,newsz)
  144. #endif
  145. #ifndef STBIW_MEMMOVE
  146. #define STBIW_MEMMOVE(a,b,sz) memmove(a,b,sz)
  147. #endif
  148. #ifndef STBIW_ASSERT
  149. #include <assert.h>
  150. #define STBIW_ASSERT(x) assert(x)
  151. #endif
  152. #define STBIW_UCHAR(x) (unsigned char) ((x) & 0xff)
  153. typedef struct
  154. {
  155. stbi_write_func *func;
  156. void *context;
  157. } stbi__write_context;
  158. // initialize a callback-based context
  159. static void stbi__start_write_callbacks(stbi__write_context *s, stbi_write_func *c, void *context)
  160. {
  161. s->func = c;
  162. s->context = context;
  163. }
  164. #ifndef STBI_WRITE_NO_STDIO
  165. static void stbi__stdio_write(void *context, void *data, int size)
  166. {
  167. fwrite(data,1,size,(FILE*) context);
  168. }
  169. static int stbi__start_write_file(stbi__write_context *s, const char *filename)
  170. {
  171. // Urho3D: proper UTF8 handling for Windows, requires Urho3D WString class
  172. #ifndef _WIN32
  173. FILE *f = fopen(filename, "wb");
  174. #else
  175. Urho3D::WString wstr(filename);
  176. FILE *f = _wfopen(wstr.CString(), L"wb");
  177. #endif
  178. stbi__start_write_callbacks(s, stbi__stdio_write, (void *) f);
  179. return f != NULL;
  180. }
  181. static void stbi__end_write_file(stbi__write_context *s)
  182. {
  183. fclose((FILE *)s->context);
  184. }
  185. #endif // !STBI_WRITE_NO_STDIO
  186. typedef unsigned int stbiw_uint32;
  187. typedef int stb_image_write_test[sizeof(stbiw_uint32)==4 ? 1 : -1];
  188. #ifdef STB_IMAGE_WRITE_STATIC
  189. static int stbi_write_tga_with_rle = 1;
  190. #else
  191. int stbi_write_tga_with_rle = 1;
  192. #endif
  193. static void stbiw__writefv(stbi__write_context *s, const char *fmt, va_list v)
  194. {
  195. while (*fmt) {
  196. switch (*fmt++) {
  197. case ' ': break;
  198. case '1': { unsigned char x = STBIW_UCHAR(va_arg(v, int));
  199. s->func(s->context,&x,1);
  200. break; }
  201. case '2': { int x = va_arg(v,int);
  202. unsigned char b[2];
  203. b[0] = STBIW_UCHAR(x);
  204. b[1] = STBIW_UCHAR(x>>8);
  205. s->func(s->context,b,2);
  206. break; }
  207. case '4': { stbiw_uint32 x = va_arg(v,int);
  208. unsigned char b[4];
  209. b[0]=STBIW_UCHAR(x);
  210. b[1]=STBIW_UCHAR(x>>8);
  211. b[2]=STBIW_UCHAR(x>>16);
  212. b[3]=STBIW_UCHAR(x>>24);
  213. s->func(s->context,b,4);
  214. break; }
  215. default:
  216. STBIW_ASSERT(0);
  217. return;
  218. }
  219. }
  220. }
  221. static void stbiw__writef(stbi__write_context *s, const char *fmt, ...)
  222. {
  223. va_list v;
  224. va_start(v, fmt);
  225. stbiw__writefv(s, fmt, v);
  226. va_end(v);
  227. }
  228. static void stbiw__write3(stbi__write_context *s, unsigned char a, unsigned char b, unsigned char c)
  229. {
  230. unsigned char arr[3];
  231. arr[0] = a, arr[1] = b, arr[2] = c;
  232. s->func(s->context, arr, 3);
  233. }
  234. static void stbiw__write_pixel(stbi__write_context *s, int rgb_dir, int comp, int write_alpha, int expand_mono, unsigned char *d)
  235. {
  236. unsigned char bg[3] = { 255, 0, 255}, px[3];
  237. int k;
  238. if (write_alpha < 0)
  239. s->func(s->context, &d[comp - 1], 1);
  240. switch (comp) {
  241. case 1:
  242. s->func(s->context,d,1);
  243. break;
  244. case 2:
  245. if (expand_mono)
  246. stbiw__write3(s, d[0], d[0], d[0]); // monochrome bmp
  247. else
  248. s->func(s->context, d, 1); // monochrome TGA
  249. break;
  250. case 4:
  251. if (!write_alpha) {
  252. // composite against pink background
  253. for (k = 0; k < 3; ++k)
  254. px[k] = bg[k] + ((d[k] - bg[k]) * d[3]) / 255;
  255. stbiw__write3(s, px[1 - rgb_dir], px[1], px[1 + rgb_dir]);
  256. break;
  257. }
  258. /* FALLTHROUGH */
  259. case 3:
  260. stbiw__write3(s, d[1 - rgb_dir], d[1], d[1 + rgb_dir]);
  261. break;
  262. }
  263. if (write_alpha > 0)
  264. s->func(s->context, &d[comp - 1], 1);
  265. }
  266. static void stbiw__write_pixels(stbi__write_context *s, int rgb_dir, int vdir, int x, int y, int comp, void *data, int write_alpha, int scanline_pad, int expand_mono)
  267. {
  268. stbiw_uint32 zero = 0;
  269. int i,j, j_end;
  270. if (y <= 0)
  271. return;
  272. if (vdir < 0)
  273. j_end = -1, j = y-1;
  274. else
  275. j_end = y, j = 0;
  276. for (; j != j_end; j += vdir) {
  277. for (i=0; i < x; ++i) {
  278. unsigned char *d = (unsigned char *) data + (j*x+i)*comp;
  279. stbiw__write_pixel(s, rgb_dir, comp, write_alpha, expand_mono, d);
  280. }
  281. s->func(s->context, &zero, scanline_pad);
  282. }
  283. }
  284. static int stbiw__outfile(stbi__write_context *s, int rgb_dir, int vdir, int x, int y, int comp, int expand_mono, void *data, int alpha, int pad, const char *fmt, ...)
  285. {
  286. if (y < 0 || x < 0) {
  287. return 0;
  288. } else {
  289. va_list v;
  290. va_start(v, fmt);
  291. stbiw__writefv(s, fmt, v);
  292. va_end(v);
  293. stbiw__write_pixels(s,rgb_dir,vdir,x,y,comp,data,alpha,pad, expand_mono);
  294. return 1;
  295. }
  296. }
  297. static int stbi_write_bmp_core(stbi__write_context *s, int x, int y, int comp, const void *data)
  298. {
  299. int pad = (-x*3) & 3;
  300. return stbiw__outfile(s,-1,-1,x,y,comp,1,(void *) data,0,pad,
  301. "11 4 22 4" "4 44 22 444444",
  302. 'B', 'M', 14+40+(x*3+pad)*y, 0,0, 14+40, // file header
  303. 40, x,y, 1,24, 0,0,0,0,0,0); // bitmap header
  304. }
  305. STBIWDEF int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data)
  306. {
  307. stbi__write_context s;
  308. stbi__start_write_callbacks(&s, func, context);
  309. return stbi_write_bmp_core(&s, x, y, comp, data);
  310. }
  311. #ifndef STBI_WRITE_NO_STDIO
  312. STBIWDEF int stbi_write_bmp(char const *filename, int x, int y, int comp, const void *data)
  313. {
  314. stbi__write_context s;
  315. if (stbi__start_write_file(&s,filename)) {
  316. int r = stbi_write_bmp_core(&s, x, y, comp, data);
  317. stbi__end_write_file(&s);
  318. return r;
  319. } else
  320. return 0;
  321. }
  322. #endif //!STBI_WRITE_NO_STDIO
  323. static int stbi_write_tga_core(stbi__write_context *s, int x, int y, int comp, void *data)
  324. {
  325. int has_alpha = (comp == 2 || comp == 4);
  326. int colorbytes = has_alpha ? comp-1 : comp;
  327. int format = colorbytes < 2 ? 3 : 2; // 3 color channels (RGB/RGBA) = 2, 1 color channel (Y/YA) = 3
  328. if (y < 0 || x < 0)
  329. return 0;
  330. if (!stbi_write_tga_with_rle) {
  331. return stbiw__outfile(s, -1, -1, x, y, comp, 0, (void *) data, has_alpha, 0,
  332. "111 221 2222 11", 0, 0, format, 0, 0, 0, 0, 0, x, y, (colorbytes + has_alpha) * 8, has_alpha * 8);
  333. } else {
  334. int i,j,k;
  335. stbiw__writef(s, "111 221 2222 11", 0,0,format+8, 0,0,0, 0,0,x,y, (colorbytes + has_alpha) * 8, has_alpha * 8);
  336. for (j = y - 1; j >= 0; --j) {
  337. unsigned char *row = (unsigned char *) data + j * x * comp;
  338. int len;
  339. for (i = 0; i < x; i += len) {
  340. unsigned char *begin = row + i * comp;
  341. int diff = 1;
  342. len = 1;
  343. if (i < x - 1) {
  344. ++len;
  345. diff = memcmp(begin, row + (i + 1) * comp, comp);
  346. if (diff) {
  347. const unsigned char *prev = begin;
  348. for (k = i + 2; k < x && len < 128; ++k) {
  349. if (memcmp(prev, row + k * comp, comp)) {
  350. prev += comp;
  351. ++len;
  352. } else {
  353. --len;
  354. break;
  355. }
  356. }
  357. } else {
  358. for (k = i + 2; k < x && len < 128; ++k) {
  359. if (!memcmp(begin, row + k * comp, comp)) {
  360. ++len;
  361. } else {
  362. break;
  363. }
  364. }
  365. }
  366. }
  367. if (diff) {
  368. unsigned char header = STBIW_UCHAR(len - 1);
  369. s->func(s->context, &header, 1);
  370. for (k = 0; k < len; ++k) {
  371. stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin + k * comp);
  372. }
  373. } else {
  374. unsigned char header = STBIW_UCHAR(len - 129);
  375. s->func(s->context, &header, 1);
  376. stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin);
  377. }
  378. }
  379. }
  380. }
  381. return 1;
  382. }
  383. int stbi_write_tga_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data)
  384. {
  385. stbi__write_context s;
  386. stbi__start_write_callbacks(&s, func, context);
  387. return stbi_write_tga_core(&s, x, y, comp, (void *) data);
  388. }
  389. #ifndef STBI_WRITE_NO_STDIO
  390. int stbi_write_tga(char const *filename, int x, int y, int comp, const void *data)
  391. {
  392. stbi__write_context s;
  393. if (stbi__start_write_file(&s,filename)) {
  394. int r = stbi_write_tga_core(&s, x, y, comp, (void *) data);
  395. stbi__end_write_file(&s);
  396. return r;
  397. } else
  398. return 0;
  399. }
  400. #endif
  401. // *************************************************************************************************
  402. // Radiance RGBE HDR writer
  403. // by Baldur Karlsson
  404. #ifndef STBI_WRITE_NO_STDIO
  405. #define stbiw__max(a, b) ((a) > (b) ? (a) : (b))
  406. void stbiw__linear_to_rgbe(unsigned char *rgbe, float *linear)
  407. {
  408. int exponent;
  409. float maxcomp = stbiw__max(linear[0], stbiw__max(linear[1], linear[2]));
  410. if (maxcomp < 1e-32f) {
  411. rgbe[0] = rgbe[1] = rgbe[2] = rgbe[3] = 0;
  412. } else {
  413. float normalize = (float) frexp(maxcomp, &exponent) * 256.0f/maxcomp;
  414. rgbe[0] = (unsigned char)(linear[0] * normalize);
  415. rgbe[1] = (unsigned char)(linear[1] * normalize);
  416. rgbe[2] = (unsigned char)(linear[2] * normalize);
  417. rgbe[3] = (unsigned char)(exponent + 128);
  418. }
  419. }
  420. void stbiw__write_run_data(stbi__write_context *s, int length, unsigned char databyte)
  421. {
  422. unsigned char lengthbyte = STBIW_UCHAR(length+128);
  423. STBIW_ASSERT(length+128 <= 255);
  424. s->func(s->context, &lengthbyte, 1);
  425. s->func(s->context, &databyte, 1);
  426. }
  427. void stbiw__write_dump_data(stbi__write_context *s, int length, unsigned char *data)
  428. {
  429. unsigned char lengthbyte = STBIW_UCHAR(length);
  430. STBIW_ASSERT(length <= 128); // inconsistent with spec but consistent with official code
  431. s->func(s->context, &lengthbyte, 1);
  432. s->func(s->context, data, length);
  433. }
  434. void stbiw__write_hdr_scanline(stbi__write_context *s, int width, int ncomp, unsigned char *scratch, float *scanline)
  435. {
  436. unsigned char scanlineheader[4] = { 2, 2, 0, 0 };
  437. unsigned char rgbe[4];
  438. float linear[3];
  439. int x;
  440. scanlineheader[2] = (width&0xff00)>>8;
  441. scanlineheader[3] = (width&0x00ff);
  442. /* skip RLE for images too small or large */
  443. if (width < 8 || width >= 32768) {
  444. for (x=0; x < width; x++) {
  445. switch (ncomp) {
  446. case 4: /* fallthrough */
  447. case 3: linear[2] = scanline[x*ncomp + 2];
  448. linear[1] = scanline[x*ncomp + 1];
  449. linear[0] = scanline[x*ncomp + 0];
  450. break;
  451. default:
  452. linear[0] = linear[1] = linear[2] = scanline[x*ncomp + 0];
  453. break;
  454. }
  455. stbiw__linear_to_rgbe(rgbe, linear);
  456. s->func(s->context, rgbe, 4);
  457. }
  458. } else {
  459. int c,r;
  460. /* encode into scratch buffer */
  461. for (x=0; x < width; x++) {
  462. switch(ncomp) {
  463. case 4: /* fallthrough */
  464. case 3: linear[2] = scanline[x*ncomp + 2];
  465. linear[1] = scanline[x*ncomp + 1];
  466. linear[0] = scanline[x*ncomp + 0];
  467. break;
  468. default:
  469. linear[0] = linear[1] = linear[2] = scanline[x*ncomp + 0];
  470. break;
  471. }
  472. stbiw__linear_to_rgbe(rgbe, linear);
  473. scratch[x + width*0] = rgbe[0];
  474. scratch[x + width*1] = rgbe[1];
  475. scratch[x + width*2] = rgbe[2];
  476. scratch[x + width*3] = rgbe[3];
  477. }
  478. s->func(s->context, scanlineheader, 4);
  479. /* RLE each component separately */
  480. for (c=0; c < 4; c++) {
  481. unsigned char *comp = &scratch[width*c];
  482. x = 0;
  483. while (x < width) {
  484. // find first run
  485. r = x;
  486. while (r+2 < width) {
  487. if (comp[r] == comp[r+1] && comp[r] == comp[r+2])
  488. break;
  489. ++r;
  490. }
  491. if (r+2 >= width)
  492. r = width;
  493. // dump up to first run
  494. while (x < r) {
  495. int len = r-x;
  496. if (len > 128) len = 128;
  497. stbiw__write_dump_data(s, len, &comp[x]);
  498. x += len;
  499. }
  500. // if there's a run, output it
  501. if (r+2 < width) { // same test as what we break out of in search loop, so only true if we break'd
  502. // find next byte after run
  503. while (r < width && comp[r] == comp[x])
  504. ++r;
  505. // output run up to r
  506. while (x < r) {
  507. int len = r-x;
  508. if (len > 127) len = 127;
  509. stbiw__write_run_data(s, len, comp[x]);
  510. x += len;
  511. }
  512. }
  513. }
  514. }
  515. }
  516. }
  517. static int stbi_write_hdr_core(stbi__write_context *s, int x, int y, int comp, float *data)
  518. {
  519. if (y <= 0 || x <= 0 || data == NULL)
  520. return 0;
  521. else {
  522. // Each component is stored separately. Allocate scratch space for full output scanline.
  523. unsigned char *scratch = (unsigned char *) STBIW_MALLOC(x*4);
  524. int i, len;
  525. char buffer[128];
  526. char header[] = "#?RADIANCE\n# Written by stb_image_write.h\nFORMAT=32-bit_rle_rgbe\n";
  527. s->func(s->context, header, sizeof(header)-1);
  528. len = sprintf(buffer, "EXPOSURE= 1.0000000000000\n\n-Y %d +X %d\n", y, x);
  529. s->func(s->context, buffer, len);
  530. for(i=0; i < y; i++)
  531. stbiw__write_hdr_scanline(s, x, comp, scratch, data + comp*i*x);
  532. STBIW_FREE(scratch);
  533. return 1;
  534. }
  535. }
  536. int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const float *data)
  537. {
  538. stbi__write_context s;
  539. stbi__start_write_callbacks(&s, func, context);
  540. return stbi_write_hdr_core(&s, x, y, comp, (float *) data);
  541. }
  542. int stbi_write_hdr(char const *filename, int x, int y, int comp, const float *data)
  543. {
  544. stbi__write_context s;
  545. if (stbi__start_write_file(&s,filename)) {
  546. int r = stbi_write_hdr_core(&s, x, y, comp, (float *) data);
  547. stbi__end_write_file(&s);
  548. return r;
  549. } else
  550. return 0;
  551. }
  552. #endif // STBI_WRITE_NO_STDIO
  553. //////////////////////////////////////////////////////////////////////////////
  554. //
  555. // PNG writer
  556. //
  557. // stretchy buffer; stbiw__sbpush() == vector<>::push_back() -- stbiw__sbcount() == vector<>::size()
  558. #define stbiw__sbraw(a) ((int *) (a) - 2)
  559. #define stbiw__sbm(a) stbiw__sbraw(a)[0]
  560. #define stbiw__sbn(a) stbiw__sbraw(a)[1]
  561. #define stbiw__sbneedgrow(a,n) ((a)==0 || stbiw__sbn(a)+n >= stbiw__sbm(a))
  562. #define stbiw__sbmaybegrow(a,n) (stbiw__sbneedgrow(a,(n)) ? stbiw__sbgrow(a,n) : 0)
  563. #define stbiw__sbgrow(a,n) stbiw__sbgrowf((void **) &(a), (n), sizeof(*(a)))
  564. #define stbiw__sbpush(a, v) (stbiw__sbmaybegrow(a,1), (a)[stbiw__sbn(a)++] = (v))
  565. #define stbiw__sbcount(a) ((a) ? stbiw__sbn(a) : 0)
  566. #define stbiw__sbfree(a) ((a) ? STBIW_FREE(stbiw__sbraw(a)),0 : 0)
  567. static void *stbiw__sbgrowf(void **arr, int increment, int itemsize)
  568. {
  569. int m = *arr ? 2*stbiw__sbm(*arr)+increment : increment+1;
  570. void *p = STBIW_REALLOC_SIZED(*arr ? stbiw__sbraw(*arr) : 0, *arr ? (stbiw__sbm(*arr)*itemsize + sizeof(int)*2) : 0, itemsize * m + sizeof(int)*2);
  571. STBIW_ASSERT(p);
  572. if (p) {
  573. if (!*arr) ((int *) p)[1] = 0;
  574. *arr = (void *) ((int *) p + 2);
  575. stbiw__sbm(*arr) = m;
  576. }
  577. return *arr;
  578. }
  579. static unsigned char *stbiw__zlib_flushf(unsigned char *data, unsigned int *bitbuffer, int *bitcount)
  580. {
  581. while (*bitcount >= 8) {
  582. stbiw__sbpush(data, STBIW_UCHAR(*bitbuffer));
  583. *bitbuffer >>= 8;
  584. *bitcount -= 8;
  585. }
  586. return data;
  587. }
  588. static int stbiw__zlib_bitrev(int code, int codebits)
  589. {
  590. int res=0;
  591. while (codebits--) {
  592. res = (res << 1) | (code & 1);
  593. code >>= 1;
  594. }
  595. return res;
  596. }
  597. static unsigned int stbiw__zlib_countm(unsigned char *a, unsigned char *b, int limit)
  598. {
  599. int i;
  600. for (i=0; i < limit && i < 258; ++i)
  601. if (a[i] != b[i]) break;
  602. return i;
  603. }
  604. static unsigned int stbiw__zhash(unsigned char *data)
  605. {
  606. stbiw_uint32 hash = data[0] + (data[1] << 8) + (data[2] << 16);
  607. hash ^= hash << 3;
  608. hash += hash >> 5;
  609. hash ^= hash << 4;
  610. hash += hash >> 17;
  611. hash ^= hash << 25;
  612. hash += hash >> 6;
  613. return hash;
  614. }
  615. #define stbiw__zlib_flush() (out = stbiw__zlib_flushf(out, &bitbuf, &bitcount))
  616. #define stbiw__zlib_add(code,codebits) \
  617. (bitbuf |= (code) << bitcount, bitcount += (codebits), stbiw__zlib_flush())
  618. #define stbiw__zlib_huffa(b,c) stbiw__zlib_add(stbiw__zlib_bitrev(b,c),c)
  619. // default huffman tables
  620. #define stbiw__zlib_huff1(n) stbiw__zlib_huffa(0x30 + (n), 8)
  621. #define stbiw__zlib_huff2(n) stbiw__zlib_huffa(0x190 + (n)-144, 9)
  622. #define stbiw__zlib_huff3(n) stbiw__zlib_huffa(0 + (n)-256,7)
  623. #define stbiw__zlib_huff4(n) stbiw__zlib_huffa(0xc0 + (n)-280,8)
  624. #define stbiw__zlib_huff(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : (n) <= 255 ? stbiw__zlib_huff2(n) : (n) <= 279 ? stbiw__zlib_huff3(n) : stbiw__zlib_huff4(n))
  625. #define stbiw__zlib_huffb(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : stbiw__zlib_huff2(n))
  626. #define stbiw__ZHASH 16384
  627. unsigned char * stbi_zlib_compress(unsigned char *data, int data_len, int *out_len, int quality)
  628. {
  629. static unsigned short lengthc[] = { 3,4,5,6,7,8,9,10,11,13,15,17,19,23,27,31,35,43,51,59,67,83,99,115,131,163,195,227,258, 259 };
  630. static unsigned char lengtheb[]= { 0,0,0,0,0,0,0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0 };
  631. static unsigned short distc[] = { 1,2,3,4,5,7,9,13,17,25,33,49,65,97,129,193,257,385,513,769,1025,1537,2049,3073,4097,6145,8193,12289,16385,24577, 32768 };
  632. static unsigned char disteb[] = { 0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13 };
  633. unsigned int bitbuf=0;
  634. int i,j, bitcount=0;
  635. unsigned char *out = NULL;
  636. unsigned char ***hash_table = (unsigned char***) STBIW_MALLOC(stbiw__ZHASH * sizeof(char**));
  637. if (quality < 5) quality = 5;
  638. stbiw__sbpush(out, 0x78); // DEFLATE 32K window
  639. stbiw__sbpush(out, 0x5e); // FLEVEL = 1
  640. stbiw__zlib_add(1,1); // BFINAL = 1
  641. stbiw__zlib_add(1,2); // BTYPE = 1 -- fixed huffman
  642. for (i=0; i < stbiw__ZHASH; ++i)
  643. hash_table[i] = NULL;
  644. i=0;
  645. while (i < data_len-3) {
  646. // hash next 3 bytes of data to be compressed
  647. int h = stbiw__zhash(data+i)&(stbiw__ZHASH-1), best=3;
  648. unsigned char *bestloc = 0;
  649. unsigned char **hlist = hash_table[h];
  650. int n = stbiw__sbcount(hlist);
  651. for (j=0; j < n; ++j) {
  652. if (hlist[j]-data > i-32768) { // if entry lies within window
  653. int d = stbiw__zlib_countm(hlist[j], data+i, data_len-i);
  654. if (d >= best) best=d,bestloc=hlist[j];
  655. }
  656. }
  657. // when hash table entry is too long, delete half the entries
  658. if (hash_table[h] && stbiw__sbn(hash_table[h]) == 2*quality) {
  659. STBIW_MEMMOVE(hash_table[h], hash_table[h]+quality, sizeof(hash_table[h][0])*quality);
  660. stbiw__sbn(hash_table[h]) = quality;
  661. }
  662. stbiw__sbpush(hash_table[h],data+i);
  663. if (bestloc) {
  664. // "lazy matching" - check match at *next* byte, and if it's better, do cur byte as literal
  665. h = stbiw__zhash(data+i+1)&(stbiw__ZHASH-1);
  666. hlist = hash_table[h];
  667. n = stbiw__sbcount(hlist);
  668. for (j=0; j < n; ++j) {
  669. if (hlist[j]-data > i-32767) {
  670. int e = stbiw__zlib_countm(hlist[j], data+i+1, data_len-i-1);
  671. if (e > best) { // if next match is better, bail on current match
  672. bestloc = NULL;
  673. break;
  674. }
  675. }
  676. }
  677. }
  678. if (bestloc) {
  679. int d = (int) (data+i - bestloc); // distance back
  680. STBIW_ASSERT(d <= 32767 && best <= 258);
  681. for (j=0; best > lengthc[j+1]-1; ++j);
  682. stbiw__zlib_huff(j+257);
  683. if (lengtheb[j]) stbiw__zlib_add(best - lengthc[j], lengtheb[j]);
  684. for (j=0; d > distc[j+1]-1; ++j);
  685. stbiw__zlib_add(stbiw__zlib_bitrev(j,5),5);
  686. if (disteb[j]) stbiw__zlib_add(d - distc[j], disteb[j]);
  687. i += best;
  688. } else {
  689. stbiw__zlib_huffb(data[i]);
  690. ++i;
  691. }
  692. }
  693. // write out final bytes
  694. for (;i < data_len; ++i)
  695. stbiw__zlib_huffb(data[i]);
  696. stbiw__zlib_huff(256); // end of block
  697. // pad with 0 bits to byte boundary
  698. while (bitcount)
  699. stbiw__zlib_add(0,1);
  700. for (i=0; i < stbiw__ZHASH; ++i)
  701. (void) stbiw__sbfree(hash_table[i]);
  702. STBIW_FREE(hash_table);
  703. {
  704. // compute adler32 on input
  705. unsigned int s1=1, s2=0;
  706. int blocklen = (int) (data_len % 5552);
  707. j=0;
  708. while (j < data_len) {
  709. for (i=0; i < blocklen; ++i) s1 += data[j+i], s2 += s1;
  710. s1 %= 65521, s2 %= 65521;
  711. j += blocklen;
  712. blocklen = 5552;
  713. }
  714. stbiw__sbpush(out, STBIW_UCHAR(s2 >> 8));
  715. stbiw__sbpush(out, STBIW_UCHAR(s2));
  716. stbiw__sbpush(out, STBIW_UCHAR(s1 >> 8));
  717. stbiw__sbpush(out, STBIW_UCHAR(s1));
  718. }
  719. *out_len = stbiw__sbn(out);
  720. // make returned pointer freeable
  721. STBIW_MEMMOVE(stbiw__sbraw(out), out, *out_len);
  722. return (unsigned char *) stbiw__sbraw(out);
  723. }
  724. static unsigned int stbiw__crc32(unsigned char *buffer, int len)
  725. {
  726. static unsigned int crc_table[256] =
  727. {
  728. 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3,
  729. 0x0eDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91,
  730. 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE, 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7,
  731. 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC, 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5,
  732. 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172, 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B,
  733. 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940, 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59,
  734. 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116, 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F,
  735. 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924, 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D,
  736. 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A, 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433,
  737. 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818, 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01,
  738. 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E, 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457,
  739. 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C, 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65,
  740. 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2, 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB,
  741. 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0, 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9,
  742. 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086, 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F,
  743. 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4, 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD,
  744. 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A, 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683,
  745. 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8, 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1,
  746. 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE, 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7,
  747. 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC, 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5,
  748. 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252, 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B,
  749. 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60, 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79,
  750. 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236, 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F,
  751. 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04, 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D,
  752. 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A, 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713,
  753. 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38, 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21,
  754. 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E, 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777,
  755. 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C, 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45,
  756. 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2, 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB,
  757. 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0, 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9,
  758. 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6, 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF,
  759. 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94, 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D
  760. };
  761. unsigned int crc = ~0u;
  762. int i;
  763. for (i=0; i < len; ++i)
  764. crc = (crc >> 8) ^ crc_table[buffer[i] ^ (crc & 0xff)];
  765. return ~crc;
  766. }
  767. #define stbiw__wpng4(o,a,b,c,d) ((o)[0]=STBIW_UCHAR(a),(o)[1]=STBIW_UCHAR(b),(o)[2]=STBIW_UCHAR(c),(o)[3]=STBIW_UCHAR(d),(o)+=4)
  768. #define stbiw__wp32(data,v) stbiw__wpng4(data, (v)>>24,(v)>>16,(v)>>8,(v));
  769. #define stbiw__wptag(data,s) stbiw__wpng4(data, s[0],s[1],s[2],s[3])
  770. static void stbiw__wpcrc(unsigned char **data, int len)
  771. {
  772. unsigned int crc = stbiw__crc32(*data - len - 4, len+4);
  773. stbiw__wp32(*data, crc);
  774. }
  775. static unsigned char stbiw__paeth(int a, int b, int c)
  776. {
  777. int p = a + b - c, pa = abs(p-a), pb = abs(p-b), pc = abs(p-c);
  778. if (pa <= pb && pa <= pc) return STBIW_UCHAR(a);
  779. if (pb <= pc) return STBIW_UCHAR(b);
  780. return STBIW_UCHAR(c);
  781. }
  782. unsigned char *stbi_write_png_to_mem(unsigned char *pixels, int stride_bytes, int x, int y, int n, int *out_len)
  783. {
  784. int ctype[5] = { -1, 0, 4, 2, 6 };
  785. unsigned char sig[8] = { 137,80,78,71,13,10,26,10 };
  786. unsigned char *out,*o, *filt, *zlib;
  787. signed char *line_buffer;
  788. int i,j,k,p,zlen;
  789. if (stride_bytes == 0)
  790. stride_bytes = x * n;
  791. filt = (unsigned char *) STBIW_MALLOC((x*n+1) * y); if (!filt) return 0;
  792. line_buffer = (signed char *) STBIW_MALLOC(x * n); if (!line_buffer) { STBIW_FREE(filt); return 0; }
  793. for (j=0; j < y; ++j) {
  794. static int mapping[] = { 0,1,2,3,4 };
  795. static int firstmap[] = { 0,1,0,5,6 };
  796. int *mymap = j ? mapping : firstmap;
  797. int best = 0, bestval = 0x7fffffff;
  798. for (p=0; p < 2; ++p) {
  799. for (k= p?best:0; k < 5; ++k) {
  800. int type = mymap[k],est=0;
  801. unsigned char *z = pixels + stride_bytes*j;
  802. for (i=0; i < n; ++i)
  803. switch (type) {
  804. case 0: line_buffer[i] = z[i]; break;
  805. case 1: line_buffer[i] = z[i]; break;
  806. case 2: line_buffer[i] = z[i] - z[i-stride_bytes]; break;
  807. case 3: line_buffer[i] = z[i] - (z[i-stride_bytes]>>1); break;
  808. case 4: line_buffer[i] = (signed char) (z[i] - stbiw__paeth(0,z[i-stride_bytes],0)); break;
  809. case 5: line_buffer[i] = z[i]; break;
  810. case 6: line_buffer[i] = z[i]; break;
  811. }
  812. for (i=n; i < x*n; ++i) {
  813. switch (type) {
  814. case 0: line_buffer[i] = z[i]; break;
  815. case 1: line_buffer[i] = z[i] - z[i-n]; break;
  816. case 2: line_buffer[i] = z[i] - z[i-stride_bytes]; break;
  817. case 3: line_buffer[i] = z[i] - ((z[i-n] + z[i-stride_bytes])>>1); break;
  818. case 4: line_buffer[i] = z[i] - stbiw__paeth(z[i-n], z[i-stride_bytes], z[i-stride_bytes-n]); break;
  819. case 5: line_buffer[i] = z[i] - (z[i-n]>>1); break;
  820. case 6: line_buffer[i] = z[i] - stbiw__paeth(z[i-n], 0,0); break;
  821. }
  822. }
  823. if (p) break;
  824. for (i=0; i < x*n; ++i)
  825. est += abs((signed char) line_buffer[i]);
  826. if (est < bestval) { bestval = est; best = k; }
  827. }
  828. }
  829. // when we get here, best contains the filter type, and line_buffer contains the data
  830. filt[j*(x*n+1)] = (unsigned char) best;
  831. STBIW_MEMMOVE(filt+j*(x*n+1)+1, line_buffer, x*n);
  832. }
  833. STBIW_FREE(line_buffer);
  834. zlib = stbi_zlib_compress(filt, y*( x*n+1), &zlen, 8); // increase 8 to get smaller but use more memory
  835. STBIW_FREE(filt);
  836. if (!zlib) return 0;
  837. // each tag requires 12 bytes of overhead
  838. out = (unsigned char *) STBIW_MALLOC(8 + 12+13 + 12+zlen + 12);
  839. if (!out) return 0;
  840. *out_len = 8 + 12+13 + 12+zlen + 12;
  841. o=out;
  842. STBIW_MEMMOVE(o,sig,8); o+= 8;
  843. stbiw__wp32(o, 13); // header length
  844. stbiw__wptag(o, "IHDR");
  845. stbiw__wp32(o, x);
  846. stbiw__wp32(o, y);
  847. *o++ = 8;
  848. *o++ = STBIW_UCHAR(ctype[n]);
  849. *o++ = 0;
  850. *o++ = 0;
  851. *o++ = 0;
  852. stbiw__wpcrc(&o,13);
  853. stbiw__wp32(o, zlen);
  854. stbiw__wptag(o, "IDAT");
  855. STBIW_MEMMOVE(o, zlib, zlen);
  856. o += zlen;
  857. STBIW_FREE(zlib);
  858. stbiw__wpcrc(&o, zlen);
  859. stbiw__wp32(o,0);
  860. stbiw__wptag(o, "IEND");
  861. stbiw__wpcrc(&o,0);
  862. STBIW_ASSERT(o == out + *out_len);
  863. return out;
  864. }
  865. #ifndef STBI_WRITE_NO_STDIO
  866. STBIWDEF int stbi_write_png(char const *filename, int x, int y, int comp, const void *data, int stride_bytes)
  867. {
  868. FILE *f;
  869. int len;
  870. unsigned char *png = stbi_write_png_to_mem((unsigned char *) data, stride_bytes, x, y, comp, &len);
  871. if (png == NULL) return 0;
  872. f = fopen(filename, "wb");
  873. if (!f) { STBIW_FREE(png); return 0; }
  874. fwrite(png, 1, len, f);
  875. fclose(f);
  876. STBIW_FREE(png);
  877. return 1;
  878. }
  879. #endif
  880. STBIWDEF int stbi_write_png_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data, int stride_bytes)
  881. {
  882. int len;
  883. unsigned char *png = stbi_write_png_to_mem((unsigned char *) data, stride_bytes, x, y, comp, &len);
  884. if (png == NULL) return 0;
  885. func(context, png, len);
  886. STBIW_FREE(png);
  887. return 1;
  888. }
  889. #endif // STB_IMAGE_WRITE_IMPLEMENTATION
  890. /* Revision history
  891. 1.02 (2016-04-02)
  892. avoid allocating large structures on the stack
  893. 1.01 (2016-01-16)
  894. STBIW_REALLOC_SIZED: support allocators with no realloc support
  895. avoid race-condition in crc initialization
  896. minor compile issues
  897. 1.00 (2015-09-14)
  898. installable file IO function
  899. 0.99 (2015-09-13)
  900. warning fixes; TGA rle support
  901. 0.98 (2015-04-08)
  902. added STBIW_MALLOC, STBIW_ASSERT etc
  903. 0.97 (2015-01-18)
  904. fixed HDR asserts, rewrote HDR rle logic
  905. 0.96 (2015-01-17)
  906. add HDR output
  907. fix monochrome BMP
  908. 0.95 (2014-08-17)
  909. add monochrome TGA output
  910. 0.94 (2014-05-31)
  911. rename private functions to avoid conflicts with stb_image.h
  912. 0.93 (2014-05-27)
  913. warning fixes
  914. 0.92 (2010-08-01)
  915. casts to unsigned char to fix warnings
  916. 0.91 (2010-07-17)
  917. first public release
  918. 0.90 first internal release
  919. */