gd_webp.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. #include <stdio.h>
  2. #include <math.h>
  3. #include <string.h>
  4. #include <stdlib.h>
  5. #include "gd.h"
  6. #ifdef HAVE_LIBVPX
  7. #include "webpimg.h"
  8. #include "gdhelpers.h"
  9. extern void gd_YUV420toRGBA(uint8* Y,
  10. uint8* U,
  11. uint8* V,
  12. gdImagePtr im);
  13. extern void gd_RGBAToYUV420(gdImagePtr im2,
  14. uint8* Y,
  15. uint8* U,
  16. uint8* V);
  17. const char * gdWebpGetVersionString()
  18. {
  19. return "not defined";
  20. }
  21. gdImagePtr gdImageCreateFromWebp (FILE * inFile)
  22. {
  23. gdImagePtr im;
  24. gdIOCtx *in = gdNewFileCtx(inFile);
  25. im = gdImageCreateFromWebpCtx(in);
  26. in->gd_free(in);
  27. return im;
  28. }
  29. gdImagePtr gdImageCreateFromWebpPtr (int size, void *data)
  30. {
  31. int width, height, ret;
  32. unsigned char *Y = NULL;
  33. unsigned char *U = NULL;
  34. unsigned char *V = NULL;
  35. gdImagePtr im;
  36. ret = WebPDecode(data, size, &Y, &U, &V, &width, &height);
  37. if (ret != webp_success) {
  38. if (Y) free(Y);
  39. if (U) free(U);
  40. if (V) free(V);
  41. php_gd_error("WebP decode: fail to decode input data");
  42. return NULL;
  43. }
  44. im = gdImageCreateTrueColor(width, height);
  45. if (!im) {
  46. return NULL;
  47. }
  48. gd_YUV420toRGBA(Y, U, V, im);
  49. return im;
  50. }
  51. #define GD_WEBP_ALLOC_STEP (4*1024)
  52. gdImagePtr gdImageCreateFromWebpCtx (gdIOCtx * infile)
  53. {
  54. int width, height, ret;
  55. unsigned char *filedata = NULL;
  56. unsigned char *read, *temp;
  57. unsigned char *Y = NULL;
  58. unsigned char *U = NULL;
  59. unsigned char *V = NULL;
  60. size_t size = 0, n;
  61. gdImagePtr im;
  62. do {
  63. temp = gdRealloc(filedata, size+GD_WEBP_ALLOC_STEP);
  64. if (temp) {
  65. filedata = temp;
  66. read = temp + size;
  67. } else {
  68. if (filedata) {
  69. gdFree(filedata);
  70. }
  71. php_gd_error("WebP decode: realloc failed");
  72. return NULL;
  73. }
  74. n = gdGetBuf(read, GD_WEBP_ALLOC_STEP, infile);
  75. /* differs from upstream where gdGetBuf return 0 instead of EOF */
  76. if (n>0 && n!=EOF) {
  77. size += n;
  78. }
  79. } while (n>0 && n!=EOF);
  80. ret = WebPDecode(filedata, size, &Y, &U, &V, &width, &height);
  81. gdFree(filedata);
  82. if (ret != webp_success) {
  83. if (Y) free(Y);
  84. if (U) free(U);
  85. if (V) free(V);
  86. php_gd_error("WebP decode: fail to decode input data");
  87. return NULL;
  88. }
  89. im = gdImageCreateTrueColor(width, height);
  90. gd_YUV420toRGBA(Y, U, V, im);
  91. return im;
  92. }
  93. void gdImageWebpEx (gdImagePtr im, FILE * outFile, int quantization)
  94. {
  95. gdIOCtx *out = gdNewFileCtx(outFile);
  96. gdImageWebpCtx(im, out, quantization);
  97. out->gd_free(out);
  98. }
  99. void gdImageWebp (gdImagePtr im, FILE * outFile)
  100. {
  101. gdIOCtx *out = gdNewFileCtx(outFile);
  102. gdImageWebpCtx(im, out, -1);
  103. out->gd_free(out);
  104. }
  105. void * gdImageWebpPtr (gdImagePtr im, int *size)
  106. {
  107. void *rv;
  108. gdIOCtx *out = gdNewDynamicCtx(2048, NULL);
  109. gdImageWebpCtx(im, out, -1);
  110. rv = gdDPExtractData(out, size);
  111. out->gd_free(out);
  112. return rv;
  113. }
  114. void * gdImageWebpPtrEx (gdImagePtr im, int *size, int quantization)
  115. {
  116. void *rv;
  117. gdIOCtx *out = gdNewDynamicCtx(2048, NULL);
  118. gdImageWebpCtx(im, out, quantization);
  119. rv = gdDPExtractData(out, size);
  120. out->gd_free(out);
  121. return rv;
  122. }
  123. /*
  124. * Maps normalized QP (quality) to VP8 QP
  125. */
  126. int mapQualityToVP8QP(int quality) {
  127. #define MIN_QUALITY 0
  128. #define MAX_QUALITY 100
  129. #define MIN_VP8QP 1
  130. #define MAX_VP8QP 63
  131. const float scale = MAX_VP8QP - MIN_VP8QP;
  132. const float vp8qp =
  133. scale * (MAX_QUALITY - quality) / (MAX_QUALITY - MIN_QUALITY) + MIN_VP8QP;
  134. if (quality < MIN_QUALITY || quality > MAX_QUALITY) {
  135. php_gd_error("Wrong quality value %d.", quality);
  136. return -1;
  137. }
  138. return (int)(vp8qp + 0.5);
  139. }
  140. /* This routine is based in part on code from Dale Lutz (Safe Software Inc.)
  141. * and in part on demo code from Chapter 15 of "PNG: The Definitive Guide"
  142. * (http://www.cdrom.com/pub/png/pngbook.html).
  143. */
  144. void gdImageWebpCtx (gdImagePtr im, gdIOCtx * outfile, int quantization)
  145. {
  146. int width = im->sx;
  147. int height = im->sy;
  148. int colors = im->colorsTotal;
  149. int *open = im->open;
  150. int yuv_width, yuv_height, yuv_nbytes, ret;
  151. int vp8_quality;
  152. unsigned char *Y = NULL,
  153. *U = NULL,
  154. *V = NULL;
  155. unsigned char *filedata = NULL;
  156. /* Conversion to Y,U,V buffer */
  157. yuv_width = (width + 1) >> 1;
  158. yuv_height = (height + 1) >> 1;
  159. if (overflow2(width, height)) {
  160. return;
  161. }
  162. /* simplification possible, because WebP must not be larger than 16384**2 */
  163. if (overflow2(width * height, 2 * sizeof(unsigned char))) {
  164. return;
  165. }
  166. yuv_nbytes = width * height + 2 * yuv_width * yuv_height;
  167. if ((Y = (unsigned char *)gdCalloc(yuv_nbytes, sizeof(unsigned char))) == NULL) {
  168. php_gd_error("gd-webp error: cannot allocate Y buffer");
  169. return;
  170. }
  171. vp8_quality = mapQualityToVP8QP(quantization);
  172. U = Y + width * height;
  173. V = U + yuv_width * yuv_height;
  174. gd_RGBAToYUV420(im, Y, U, V);
  175. /* Encode Y,U,V and write data to file */
  176. ret = WebPEncode(Y, U, V, width, height, width, yuv_width, yuv_height, yuv_width,
  177. vp8_quality, &filedata, &yuv_nbytes, NULL);
  178. gdFree(Y);
  179. if (ret != webp_success) {
  180. if (filedata) {
  181. free(filedata);
  182. }
  183. php_gd_error("gd-webp error: WebP Encoder failed");
  184. return;
  185. }
  186. gdPutBuf (filedata, yuv_nbytes, outfile);
  187. free(filedata);
  188. }
  189. #endif /* HAVE_LIBVPX */