362 lines
9.9 KiB

  1. /*
  2. * libpipi Proper image processing implementation library
  3. * Copyright (c) 2004-2008 Sam Hocevar <sam@zoy.org>
  4. * All Rights Reserved
  5. *
  6. * $Id$
  7. *
  8. * This library is free software. It comes without any warranty, to
  9. * the extent permitted by applicable law. You can redistribute it
  10. * and/or modify it under the terms of the Do What The Fuck You Want
  11. * To Public License, Version 2, as published by Sam Hocevar. See
  12. * http://sam.zoy.org/wtfpl/COPYING for more details.
  13. */
  14. /*
  15. * context.c: processing stack handling routines
  16. */
  17. #include "config.h"
  18. #include "common.h"
  19. #include <stdio.h>
  20. #include <stdlib.h>
  21. #include <stdarg.h>
  22. #include <string.h>
  23. #include "pipi.h"
  24. #include "pipi_internals.h"
  25. pipi_context_t *pipi_create_context()
  26. {
  27. pipi_context_t *ret;
  28. ret = malloc(sizeof(pipi_context_t));
  29. memset(ret, 0, sizeof(pipi_context_t));
  30. return ret;
  31. }
  32. void pipi_destroy_context(pipi_context_t *ctx)
  33. {
  34. free(ctx);
  35. }
  36. int pipi_command(pipi_context_t *ctx, char const *cmd, ...)
  37. {
  38. if(!strcmp(cmd, "load"))
  39. {
  40. char const *file;
  41. va_list ap;
  42. va_start(ap, cmd);
  43. file = va_arg(ap, char const *);
  44. va_end(ap);
  45. ctx->images[ctx->nimages] = pipi_load(file);
  46. if(ctx->images[ctx->nimages] == NULL)
  47. return -1;
  48. ctx->nimages++;
  49. }
  50. else if(!strcmp(cmd, "save"))
  51. {
  52. char const *file;
  53. va_list ap;
  54. if(ctx->nimages < 1)
  55. return -1;
  56. ctx->nimages--;
  57. va_start(ap, cmd);
  58. file = va_arg(ap, char const *);
  59. va_end(ap);
  60. pipi_save(ctx->images[ctx->nimages], file);
  61. pipi_free(ctx->images[ctx->nimages]);
  62. }
  63. else if(!strcmp(cmd, "dither"))
  64. {
  65. pipi_image_t *src, *dst;
  66. char const *method;
  67. va_list ap;
  68. if(ctx->nimages < 1)
  69. return -1;
  70. va_start(ap, cmd);
  71. method = va_arg(ap, char const *);
  72. va_end(ap);
  73. src = ctx->images[ctx->nimages - 1];
  74. dst = NULL;
  75. if(!strcmp(method, "fs"))
  76. dst = pipi_dither_floydsteinberg(src, 0);
  77. else if(!strcmp(method, "sfs"))
  78. dst = pipi_dither_floydsteinberg(src, 1);
  79. else if(!strcmp(method, "jajuni"))
  80. dst = pipi_dither_jajuni(src, 0);
  81. else if(!strcmp(method, "sjajuni"))
  82. dst = pipi_dither_jajuni(src, 1);
  83. else if(!strcmp(method, "ost"))
  84. dst = pipi_dither_ostromoukhov(src, 0);
  85. else if(!strcmp(method, "sost"))
  86. dst = pipi_dither_ostromoukhov(src, 1);
  87. else if(!strcmp(method, "ordered"))
  88. {
  89. if(ctx->nimages < 2)
  90. return -1;
  91. dst = pipi_dither_ordered(ctx->images[ctx->nimages - 2], src);
  92. pipi_free(ctx->images[ctx->nimages - 2]);
  93. ctx->nimages--;
  94. }
  95. else if(!strcmp(method, "random"))
  96. dst = pipi_dither_random(src);
  97. else if(!strcmp(method, "dbs"))
  98. dst = pipi_dither_dbs(src);
  99. if(dst == NULL)
  100. return -1;
  101. pipi_free(src);
  102. ctx->images[ctx->nimages - 1] = dst;
  103. }
  104. else if(!strcmp(cmd, "blur"))
  105. {
  106. pipi_image_t *src, *dst;
  107. char const *arg;
  108. va_list ap;
  109. if(ctx->nimages < 1)
  110. return -1;
  111. va_start(ap, cmd);
  112. arg = va_arg(ap, char const *);
  113. va_end(ap);
  114. src = ctx->images[ctx->nimages - 1];
  115. dst = pipi_gaussian_blur(src, atof(arg));
  116. if(dst == NULL)
  117. return -1;
  118. pipi_free(src);
  119. ctx->images[ctx->nimages - 1] = dst;
  120. }
  121. else if(!strcmp(cmd, "geometry"))
  122. {
  123. pipi_image_t *src, *dst;
  124. char const *arg;
  125. va_list ap;
  126. int w, h;
  127. if(ctx->nimages < 1)
  128. return -1;
  129. va_start(ap, cmd);
  130. arg = va_arg(ap, char const *);
  131. va_end(ap);
  132. w = atoi(arg);
  133. arg = strchr(arg, 'x');
  134. if(!arg)
  135. return -1;
  136. h = atoi(arg + 1);
  137. if(w <= 0 || h <= 0)
  138. return -1;
  139. src = ctx->images[ctx->nimages - 1];
  140. dst = pipi_resize(src, w, h);
  141. if(dst == NULL)
  142. return -1;
  143. pipi_free(src);
  144. ctx->images[ctx->nimages - 1] = dst;
  145. }
  146. else if(!strcmp(cmd, "mean"))
  147. {
  148. pipi_image_t *dst;
  149. if(ctx->nimages < 2)
  150. return -1;
  151. dst = pipi_mean(ctx->images[ctx->nimages - 2],
  152. ctx->images[ctx->nimages - 1]);
  153. if(dst == NULL)
  154. return -1;
  155. pipi_free(ctx->images[ctx->nimages - 2]);
  156. pipi_free(ctx->images[ctx->nimages - 1]);
  157. ctx->images[ctx->nimages - 2] = dst;
  158. ctx->nimages--;
  159. }
  160. else if(!strcmp(cmd, "min"))
  161. {
  162. pipi_image_t *dst;
  163. if(ctx->nimages < 2)
  164. return -1;
  165. dst = pipi_min(ctx->images[ctx->nimages - 2],
  166. ctx->images[ctx->nimages - 1]);
  167. if(dst == NULL)
  168. return -1;
  169. pipi_free(ctx->images[ctx->nimages - 2]);
  170. pipi_free(ctx->images[ctx->nimages - 1]);
  171. ctx->images[ctx->nimages - 2] = dst;
  172. ctx->nimages--;
  173. }
  174. else if(!strcmp(cmd, "max"))
  175. {
  176. pipi_image_t *dst;
  177. if(ctx->nimages < 2)
  178. return -1;
  179. dst = pipi_max(ctx->images[ctx->nimages - 2],
  180. ctx->images[ctx->nimages - 1]);
  181. if(dst == NULL)
  182. return -1;
  183. pipi_free(ctx->images[ctx->nimages - 2]);
  184. pipi_free(ctx->images[ctx->nimages - 1]);
  185. ctx->images[ctx->nimages - 2] = dst;
  186. ctx->nimages--;
  187. }
  188. else if(!strcmp(cmd, "add"))
  189. {
  190. pipi_image_t *dst;
  191. if(ctx->nimages < 2)
  192. return -1;
  193. dst = pipi_add(ctx->images[ctx->nimages - 2],
  194. ctx->images[ctx->nimages - 1]);
  195. if(dst == NULL)
  196. return -1;
  197. pipi_free(ctx->images[ctx->nimages - 2]);
  198. pipi_free(ctx->images[ctx->nimages - 1]);
  199. ctx->images[ctx->nimages - 2] = dst;
  200. ctx->nimages--;
  201. }
  202. else if(!strcmp(cmd, "sub"))
  203. {
  204. pipi_image_t *dst;
  205. if(ctx->nimages < 2)
  206. return -1;
  207. dst = pipi_sub(ctx->images[ctx->nimages - 2],
  208. ctx->images[ctx->nimages - 1]);
  209. if(dst == NULL)
  210. return -1;
  211. pipi_free(ctx->images[ctx->nimages - 2]);
  212. pipi_free(ctx->images[ctx->nimages - 1]);
  213. ctx->images[ctx->nimages - 2] = dst;
  214. ctx->nimages--;
  215. }
  216. else if(!strcmp(cmd, "difference"))
  217. {
  218. pipi_image_t *dst;
  219. if(ctx->nimages < 2)
  220. return -1;
  221. dst = pipi_difference(ctx->images[ctx->nimages - 2],
  222. ctx->images[ctx->nimages - 1]);
  223. if(dst == NULL)
  224. return -1;
  225. pipi_free(ctx->images[ctx->nimages - 2]);
  226. pipi_free(ctx->images[ctx->nimages - 1]);
  227. ctx->images[ctx->nimages - 2] = dst;
  228. ctx->nimages--;
  229. }
  230. else if(!strcmp(cmd, "multiply"))
  231. {
  232. pipi_image_t *dst;
  233. if(ctx->nimages < 2)
  234. return -1;
  235. dst = pipi_multiply(ctx->images[ctx->nimages - 2],
  236. ctx->images[ctx->nimages - 1]);
  237. if(dst == NULL)
  238. return -1;
  239. pipi_free(ctx->images[ctx->nimages - 2]);
  240. pipi_free(ctx->images[ctx->nimages - 1]);
  241. ctx->images[ctx->nimages - 2] = dst;
  242. ctx->nimages--;
  243. }
  244. else if(!strcmp(cmd, "divide"))
  245. {
  246. pipi_image_t *dst;
  247. if(ctx->nimages < 2)
  248. return -1;
  249. dst = pipi_divide(ctx->images[ctx->nimages - 2],
  250. ctx->images[ctx->nimages - 1]);
  251. if(dst == NULL)
  252. return -1;
  253. pipi_free(ctx->images[ctx->nimages - 2]);
  254. pipi_free(ctx->images[ctx->nimages - 1]);
  255. ctx->images[ctx->nimages - 2] = dst;
  256. ctx->nimages--;
  257. }
  258. else if(!strcmp(cmd, "screen"))
  259. {
  260. pipi_image_t *dst;
  261. if(ctx->nimages < 2)
  262. return -1;
  263. dst = pipi_screen(ctx->images[ctx->nimages - 2],
  264. ctx->images[ctx->nimages - 1]);
  265. if(dst == NULL)
  266. return -1;
  267. pipi_free(ctx->images[ctx->nimages - 2]);
  268. pipi_free(ctx->images[ctx->nimages - 1]);
  269. ctx->images[ctx->nimages - 2] = dst;
  270. ctx->nimages--;
  271. }
  272. else if(!strcmp(cmd, "overlay"))
  273. {
  274. pipi_image_t *dst;
  275. if(ctx->nimages < 2)
  276. return -1;
  277. dst = pipi_overlay(ctx->images[ctx->nimages - 2],
  278. ctx->images[ctx->nimages - 1]);
  279. if(dst == NULL)
  280. return -1;
  281. pipi_free(ctx->images[ctx->nimages - 2]);
  282. pipi_free(ctx->images[ctx->nimages - 1]);
  283. ctx->images[ctx->nimages - 2] = dst;
  284. ctx->nimages--;
  285. }
  286. else if(!strcmp(cmd, "wrap"))
  287. {
  288. if(ctx->nimages < 1)
  289. return -1;
  290. ctx->images[ctx->nimages - 1]->wrap = 1;
  291. }
  292. else if(!strcmp(cmd, "autocontrast"))
  293. {
  294. pipi_image_t *tmp;
  295. if(ctx->nimages < 1)
  296. return -1;
  297. tmp = ctx->images[ctx->nimages - 1];
  298. ctx->images[ctx->nimages - 1] = pipi_autocontrast(tmp);
  299. pipi_free(tmp);
  300. }
  301. else if(!strcmp(cmd, "gray"))
  302. {
  303. if(ctx->nimages < 1)
  304. return -1;
  305. pipi_getpixels(ctx->images[ctx->nimages - 1], PIPI_PIXELS_Y_F);
  306. }
  307. else if(!strcmp(cmd, "free"))
  308. {
  309. if(ctx->nimages < 1)
  310. return -1;
  311. ctx->nimages--;
  312. pipi_free(ctx->images[ctx->nimages]);
  313. }
  314. else if(!strcmp(cmd, "dup"))
  315. {
  316. if(ctx->nimages < 1)
  317. return -1;
  318. ctx->images[ctx->nimages] = pipi_copy(ctx->images[ctx->nimages - 1]);
  319. ctx->nimages++;
  320. }
  321. else if(!strcmp(cmd, "swap"))
  322. {
  323. pipi_image_t *tmp;
  324. if(ctx->nimages < 2)
  325. return -1;
  326. tmp = ctx->images[ctx->nimages - 1];
  327. ctx->images[ctx->nimages - 1] = ctx->images[ctx->nimages - 2];
  328. ctx->images[ctx->nimages - 2] = tmp;
  329. }
  330. else
  331. {
  332. return -1;
  333. }
  334. return 0;
  335. }