gd_io_dp.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. /*
  2. * io_dp.c
  3. *
  4. * Implements the dynamic pointer interface.
  5. *
  6. * Based on GD.pm code by Lincoln Stein for interfacing to libgd.
  7. * Added support for reading as well as support for 'tell' and 'seek'.
  8. *
  9. * As will all I/O modules, most functions are for local use only (called
  10. * via function pointers in the I/O context).
  11. *
  12. * gdDPExtractData is the exception to this: it will return the pointer to
  13. * the internal data, and reset the internal storage.
  14. *
  15. * Written/Modified 1999, Philip Warner.
  16. *
  17. */
  18. #include <math.h>
  19. #include <string.h>
  20. #include <stdlib.h>
  21. #include "gd.h"
  22. #include "gdhelpers.h"
  23. #define TRUE 1
  24. #define FALSE 0
  25. /* this is used for creating images in main memory */
  26. typedef struct dpStruct
  27. {
  28. void *data;
  29. int logicalSize;
  30. int realSize;
  31. int dataGood;
  32. int pos;
  33. int freeOK;
  34. } dynamicPtr;
  35. typedef struct dpIOCtx
  36. {
  37. gdIOCtx ctx;
  38. dynamicPtr *dp;
  39. } dpIOCtx;
  40. typedef struct dpIOCtx *dpIOCtxPtr;
  41. /* these functions operate on in-memory dynamic pointers */
  42. static int allocDynamic (dynamicPtr * dp, int initialSize, void *data);
  43. static int appendDynamic (dynamicPtr * dp, const void *src, int size);
  44. static int gdReallocDynamic (dynamicPtr * dp, int required);
  45. static int trimDynamic (dynamicPtr * dp);
  46. static void gdFreeDynamicCtx (struct gdIOCtx *ctx);
  47. static dynamicPtr *newDynamic (int initialSize, void *data, int freeOKFlag);
  48. static int dynamicPutbuf (struct gdIOCtx *, const void *, int);
  49. static void dynamicPutchar (struct gdIOCtx *, int a);
  50. static int dynamicGetbuf (gdIOCtxPtr ctx, void *buf, int len);
  51. static int dynamicGetchar (gdIOCtxPtr ctx);
  52. static int dynamicSeek (struct gdIOCtx *, const int);
  53. static long dynamicTell (struct gdIOCtx *);
  54. /* return data as a dynamic pointer */
  55. gdIOCtx * gdNewDynamicCtx (int initialSize, void *data)
  56. {
  57. return gdNewDynamicCtxEx(initialSize, data, 1);
  58. }
  59. gdIOCtx * gdNewDynamicCtxEx (int initialSize, void *data, int freeOKFlag)
  60. {
  61. dpIOCtx *ctx;
  62. dynamicPtr *dp;
  63. ctx = (dpIOCtx *) gdMalloc (sizeof (dpIOCtx));
  64. dp = newDynamic(initialSize, data, freeOKFlag);
  65. ctx->dp = dp;
  66. ctx->ctx.getC = dynamicGetchar;
  67. ctx->ctx.putC = dynamicPutchar;
  68. ctx->ctx.getBuf = dynamicGetbuf;
  69. ctx->ctx.putBuf = dynamicPutbuf;
  70. ctx->ctx.seek = dynamicSeek;
  71. ctx->ctx.tell = dynamicTell;
  72. ctx->ctx.gd_free = gdFreeDynamicCtx;
  73. return (gdIOCtx *) ctx;
  74. }
  75. void * gdDPExtractData (struct gdIOCtx *ctx, int *size)
  76. {
  77. dynamicPtr *dp;
  78. dpIOCtx *dctx;
  79. void *data;
  80. dctx = (dpIOCtx *) ctx;
  81. dp = dctx->dp;
  82. /* clean up the data block and return it */
  83. if (dp->dataGood) {
  84. trimDynamic (dp);
  85. *size = dp->logicalSize;
  86. data = dp->data;
  87. } else {
  88. *size = 0;
  89. data = NULL;
  90. if (dp->data != NULL && dp->freeOK) {
  91. gdFree(dp->data);
  92. }
  93. }
  94. dp->data = NULL;
  95. dp->realSize = 0;
  96. dp->logicalSize = 0;
  97. return data;
  98. }
  99. static void gdFreeDynamicCtx (struct gdIOCtx *ctx)
  100. {
  101. dynamicPtr *dp;
  102. dpIOCtx *dctx;
  103. dctx = (dpIOCtx *) ctx;
  104. dp = dctx->dp;
  105. gdFree(ctx);
  106. dp->realSize = 0;
  107. dp->logicalSize = 0;
  108. gdFree(dp);
  109. }
  110. static long dynamicTell (struct gdIOCtx *ctx)
  111. {
  112. dpIOCtx *dctx;
  113. dctx = (dpIOCtx *) ctx;
  114. return (dctx->dp->pos);
  115. }
  116. static int dynamicSeek (struct gdIOCtx *ctx, const int pos)
  117. {
  118. int bytesNeeded;
  119. dynamicPtr *dp;
  120. dpIOCtx *dctx;
  121. dctx = (dpIOCtx *) ctx;
  122. dp = dctx->dp;
  123. if (!dp->dataGood) {
  124. return FALSE;
  125. }
  126. bytesNeeded = pos;
  127. if (bytesNeeded > dp->realSize) {
  128. /* 2.0.21 */
  129. if (!dp->freeOK) {
  130. return FALSE;
  131. }
  132. gdReallocDynamic (dp, dp->realSize * 2);
  133. }
  134. /* if we get here, we can be sure that we have enough bytes to copy safely */
  135. /* Extend the logical size if we seek beyond EOF. */
  136. if (pos > dp->logicalSize) {
  137. dp->logicalSize = pos;
  138. }
  139. dp->pos = pos;
  140. return TRUE;
  141. }
  142. /* return data as a dynamic pointer */
  143. static dynamicPtr * newDynamic (int initialSize, void *data, int freeOKFlag)
  144. {
  145. dynamicPtr *dp;
  146. dp = (dynamicPtr *) gdMalloc (sizeof (dynamicPtr));
  147. allocDynamic (dp, initialSize, data);
  148. dp->pos = 0;
  149. dp->freeOK = freeOKFlag;
  150. return dp;
  151. }
  152. static int
  153. dynamicPutbuf (struct gdIOCtx *ctx, const void *buf, int size)
  154. {
  155. dpIOCtx *dctx;
  156. dctx = (dpIOCtx *) ctx;
  157. appendDynamic (dctx->dp, buf, size);
  158. if (dctx->dp->dataGood)
  159. {
  160. return size;
  161. }
  162. else
  163. {
  164. return -1;
  165. };
  166. }
  167. static void dynamicPutchar (struct gdIOCtx *ctx, int a)
  168. {
  169. unsigned char b;
  170. dpIOCtxPtr dctx;
  171. b = a;
  172. dctx = (dpIOCtxPtr) ctx;
  173. appendDynamic(dctx->dp, &b, 1);
  174. }
  175. static int dynamicGetbuf (gdIOCtxPtr ctx, void *buf, int len)
  176. {
  177. int rlen, remain;
  178. dpIOCtxPtr dctx;
  179. dynamicPtr *dp;
  180. dctx = (dpIOCtxPtr) ctx;
  181. dp = dctx->dp;
  182. remain = dp->logicalSize - dp->pos;
  183. if (remain >= len) {
  184. rlen = len;
  185. } else {
  186. if (remain <= 0) {
  187. return EOF;
  188. }
  189. rlen = remain;
  190. }
  191. memcpy(buf, (void *) ((char *) dp->data + dp->pos), rlen);
  192. dp->pos += rlen;
  193. return rlen;
  194. }
  195. static int dynamicGetchar (gdIOCtxPtr ctx)
  196. {
  197. unsigned char b;
  198. int rv;
  199. rv = dynamicGetbuf (ctx, &b, 1);
  200. if (rv != 1) {
  201. return EOF;
  202. } else {
  203. return b; /* (b & 0xff); */
  204. }
  205. }
  206. /* *********************************************************************
  207. * InitDynamic - Return a dynamically resizable void*
  208. *
  209. * *********************************************************************
  210. */
  211. static int
  212. allocDynamic (dynamicPtr * dp, int initialSize, void *data)
  213. {
  214. if (data == NULL) {
  215. dp->logicalSize = 0;
  216. dp->dataGood = FALSE;
  217. dp->data = gdMalloc(initialSize);
  218. } else {
  219. dp->logicalSize = initialSize;
  220. dp->dataGood = TRUE;
  221. dp->data = data;
  222. }
  223. dp->realSize = initialSize;
  224. dp->dataGood = TRUE;
  225. dp->pos = 0;
  226. return TRUE;
  227. }
  228. /* append bytes to the end of a dynamic pointer */
  229. static int appendDynamic (dynamicPtr * dp, const void *src, int size)
  230. {
  231. int bytesNeeded;
  232. char *tmp;
  233. if (!dp->dataGood) {
  234. return FALSE;
  235. }
  236. /* bytesNeeded = dp->logicalSize + size; */
  237. bytesNeeded = dp->pos + size;
  238. if (bytesNeeded > dp->realSize) {
  239. /* 2.0.21 */
  240. if (!dp->freeOK) {
  241. return FALSE;
  242. }
  243. gdReallocDynamic(dp, bytesNeeded * 2);
  244. }
  245. /* if we get here, we can be sure that we have enough bytes to copy safely */
  246. /*printf("Mem OK Size: %d, Pos: %d\n", dp->realSize, dp->pos); */
  247. tmp = (char *) dp->data;
  248. memcpy((void *) (tmp + (dp->pos)), src, size);
  249. dp->pos += size;
  250. if (dp->pos > dp->logicalSize) {
  251. dp->logicalSize = dp->pos;
  252. }
  253. return TRUE;
  254. }
  255. /* grow (or shrink) dynamic pointer */
  256. static int gdReallocDynamic (dynamicPtr * dp, int required)
  257. {
  258. void *newPtr;
  259. /* First try gdRealloc(). If that doesn't work, make a new memory block and copy. */
  260. if ((newPtr = gdRealloc(dp->data, required))) {
  261. dp->realSize = required;
  262. dp->data = newPtr;
  263. return TRUE;
  264. }
  265. /* create a new pointer */
  266. newPtr = gdMalloc(required);
  267. /* copy the old data into it */
  268. memcpy(newPtr, dp->data, dp->logicalSize);
  269. gdFree(dp->data);
  270. dp->data = newPtr;
  271. dp->realSize = required;
  272. return TRUE;
  273. }
  274. /* trim pointer so that its real and logical sizes match */
  275. static int trimDynamic (dynamicPtr * dp)
  276. {
  277. /* 2.0.21: we don't reallocate memory we don't own */
  278. if (!dp->freeOK) {
  279. return FALSE;
  280. }
  281. return gdReallocDynamic(dp, dp->logicalSize);
  282. }