gd_ctx.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Stanislav Malyshev <stas@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #include "php_gd.h"
  19. #define CTX_PUTC(c,ctx) ctx->putC(ctx, c)
  20. static void _php_image_output_putc(struct gdIOCtx *ctx, int c) /* {{{ */
  21. {
  22. /* without the following downcast, the write will fail
  23. * (i.e., will write a zero byte) for all
  24. * big endian architectures:
  25. */
  26. unsigned char ch = (unsigned char) c;
  27. php_write(&ch, 1);
  28. } /* }}} */
  29. static int _php_image_output_putbuf(struct gdIOCtx *ctx, const void* buf, int l) /* {{{ */
  30. {
  31. return php_write((void *)buf, l);
  32. } /* }}} */
  33. static void _php_image_output_ctxfree(struct gdIOCtx *ctx) /* {{{ */
  34. {
  35. if(ctx) {
  36. efree(ctx);
  37. }
  38. } /* }}} */
  39. static void _php_image_stream_putc(struct gdIOCtx *ctx, int c) /* {{{ */ {
  40. char ch = (char) c;
  41. php_stream * stream = (php_stream *)ctx->data;
  42. php_stream_write(stream, &ch, 1);
  43. } /* }}} */
  44. static int _php_image_stream_putbuf(struct gdIOCtx *ctx, const void* buf, int l) /* {{{ */
  45. {
  46. php_stream * stream = (php_stream *)ctx->data;
  47. return php_stream_write(stream, (void *)buf, l);
  48. } /* }}} */
  49. static void _php_image_stream_ctxfree(struct gdIOCtx *ctx) /* {{{ */
  50. {
  51. if(ctx->data) {
  52. ctx->data = NULL;
  53. }
  54. if(ctx) {
  55. efree(ctx);
  56. }
  57. } /* }}} */
  58. static void _php_image_stream_ctxfreeandclose(struct gdIOCtx *ctx) /* {{{ */
  59. {
  60. if(ctx->data) {
  61. php_stream_close((php_stream *) ctx->data);
  62. ctx->data = NULL;
  63. }
  64. if(ctx) {
  65. efree(ctx);
  66. }
  67. } /* }}} */
  68. /* {{{ _php_image_output_ctx */
  69. static void _php_image_output_ctx(INTERNAL_FUNCTION_PARAMETERS, int image_type, char *tn, void (*func_p)())
  70. {
  71. zval *imgind;
  72. char *file = NULL;
  73. size_t file_len = 0;
  74. zend_long quality, basefilter;
  75. zend_bool compressed = 1;
  76. gdImagePtr im;
  77. int argc = ZEND_NUM_ARGS();
  78. int q = -1, i;
  79. int f = -1;
  80. gdIOCtx *ctx = NULL;
  81. zval *to_zval = NULL;
  82. php_stream *stream;
  83. int close_stream = 1;
  84. /* The third (quality) parameter for Wbmp stands for the foreground when called from image2wbmp().
  85. * The third (quality) parameter for Wbmp and Xbm stands for the foreground color index when called
  86. * from imagey<type>().
  87. */
  88. switch (image_type) {
  89. case PHP_GDIMG_TYPE_XBM:
  90. if (zend_parse_parameters(argc, "rp!|ll", &imgind, &file, &file_len, &quality, &basefilter) == FAILURE) {
  91. return;
  92. }
  93. break;
  94. case PHP_GDIMG_TYPE_BMP:
  95. if (zend_parse_parameters(argc, "r|z!b", &imgind, &to_zval, &compressed) == FAILURE) {
  96. return;
  97. }
  98. break;
  99. default:
  100. /* PHP_GDIMG_TYPE_GIF
  101. * PHP_GDIMG_TYPE_PNG
  102. * PHP_GDIMG_TYPE_JPG
  103. * PHP_GDIMG_TYPE_WBM
  104. * PHP_GDIMG_TYPE_WEBP
  105. * */
  106. if (zend_parse_parameters(argc, "r|z!ll", &imgind, &to_zval, &quality, &basefilter) == FAILURE) {
  107. return;
  108. }
  109. }
  110. if ((im = (gdImagePtr)zend_fetch_resource(Z_RES_P(imgind), "Image", phpi_get_le_gd())) == NULL) {
  111. RETURN_FALSE;
  112. }
  113. if (image_type != PHP_GDIMG_TYPE_BMP && argc >= 3) {
  114. q = quality; /* or colorindex for foreground of BW images (defaults to black) */
  115. if (argc == 4) {
  116. f = basefilter;
  117. }
  118. }
  119. if (argc > 1 && to_zval != NULL) {
  120. if (Z_TYPE_P(to_zval) == IS_RESOURCE) {
  121. php_stream_from_zval_no_verify(stream, to_zval);
  122. if (stream == NULL) {
  123. RETURN_FALSE;
  124. }
  125. close_stream = 0;
  126. } else if (Z_TYPE_P(to_zval) == IS_STRING) {
  127. if (CHECK_ZVAL_NULL_PATH(to_zval)) {
  128. php_error_docref(NULL, E_WARNING, "Invalid 2nd parameter, filename must not contain null bytes");
  129. RETURN_FALSE;
  130. }
  131. stream = php_stream_open_wrapper(Z_STRVAL_P(to_zval), "wb", REPORT_ERRORS|IGNORE_PATH|IGNORE_URL_WIN, NULL);
  132. if (stream == NULL) {
  133. RETURN_FALSE;
  134. }
  135. } else {
  136. php_error_docref(NULL, E_WARNING, "Invalid 2nd parameter, it must a filename or a stream");
  137. RETURN_FALSE;
  138. }
  139. } else if (argc > 1 && file != NULL) {
  140. stream = php_stream_open_wrapper(file, "wb", REPORT_ERRORS|IGNORE_PATH|IGNORE_URL_WIN, NULL);
  141. if (stream == NULL) {
  142. RETURN_FALSE;
  143. }
  144. } else {
  145. ctx = ecalloc(1, sizeof(gdIOCtx));
  146. ctx->putC = _php_image_output_putc;
  147. ctx->putBuf = _php_image_output_putbuf;
  148. ctx->gd_free = _php_image_output_ctxfree;
  149. }
  150. if (!ctx) {
  151. ctx = ecalloc(1, sizeof(gdIOCtx));
  152. ctx->putC = _php_image_stream_putc;
  153. ctx->putBuf = _php_image_stream_putbuf;
  154. if (close_stream) {
  155. ctx->gd_free = _php_image_stream_ctxfreeandclose;
  156. } else {
  157. ctx->gd_free = _php_image_stream_ctxfree;
  158. }
  159. ctx->data = (void *)stream;
  160. }
  161. switch(image_type) {
  162. case PHP_GDIMG_CONVERT_WBM:
  163. if(q<0||q>255) {
  164. php_error_docref(NULL, E_WARNING, "Invalid threshold value '%d'. It must be between 0 and 255", q);
  165. }
  166. case PHP_GDIMG_TYPE_JPG:
  167. (*func_p)(im, ctx, q);
  168. break;
  169. case PHP_GDIMG_TYPE_WEBP:
  170. if (q == -1) {
  171. q = 80;
  172. }
  173. (*func_p)(im, ctx, q);
  174. break;
  175. case PHP_GDIMG_TYPE_PNG:
  176. (*func_p)(im, ctx, q, f);
  177. break;
  178. case PHP_GDIMG_TYPE_XBM:
  179. case PHP_GDIMG_TYPE_WBM:
  180. if (argc < 3) {
  181. for(i=0; i < gdImageColorsTotal(im); i++) {
  182. if(!gdImageRed(im, i) && !gdImageGreen(im, i) && !gdImageBlue(im, i)) break;
  183. }
  184. q = i;
  185. }
  186. if (image_type == PHP_GDIMG_TYPE_XBM) {
  187. (*func_p)(im, file ? file : "", q, ctx);
  188. } else {
  189. (*func_p)(im, q, ctx);
  190. }
  191. break;
  192. case PHP_GDIMG_TYPE_BMP:
  193. (*func_p)(im, ctx, (int) compressed);
  194. break;
  195. default:
  196. (*func_p)(im, ctx);
  197. break;
  198. }
  199. ctx->gd_free(ctx);
  200. RETURN_TRUE;
  201. }
  202. /* }}} */
  203. /*
  204. * Local variables:
  205. * tab-width: 4
  206. * c-basic-offset: 4
  207. * End:
  208. * vim600: sw=4 ts=4 fdm=marker
  209. * vim<600: sw=4 ts=4
  210. */