You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

367 lines
10 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. double w, h;
  110. if(ctx->nimages < 1)
  111. return -1;
  112. va_start(ap, cmd);
  113. arg = va_arg(ap, char const *);
  114. va_end(ap);
  115. w = h = atof(arg);
  116. arg = strchr(arg, 'x');
  117. if(arg)
  118. h = atof(arg + 1);
  119. src = ctx->images[ctx->nimages - 1];
  120. dst = pipi_gaussian_blur_ext(src, w, h, 0.0, 0.0);
  121. if(dst == NULL)
  122. return -1;
  123. pipi_free(src);
  124. ctx->images[ctx->nimages - 1] = dst;
  125. }
  126. else if(!strcmp(cmd, "geometry"))
  127. {
  128. pipi_image_t *src, *dst;
  129. char const *arg;
  130. va_list ap;
  131. int w, h;
  132. if(ctx->nimages < 1)
  133. return -1;
  134. va_start(ap, cmd);
  135. arg = va_arg(ap, char const *);
  136. va_end(ap);
  137. w = atoi(arg);
  138. arg = strchr(arg, 'x');
  139. if(!arg)
  140. return -1;
  141. h = atoi(arg + 1);
  142. if(w <= 0 || h <= 0)
  143. return -1;
  144. src = ctx->images[ctx->nimages - 1];
  145. dst = pipi_resize(src, w, h);
  146. if(dst == NULL)
  147. return -1;
  148. pipi_free(src);
  149. ctx->images[ctx->nimages - 1] = dst;
  150. }
  151. else if(!strcmp(cmd, "mean"))
  152. {
  153. pipi_image_t *dst;
  154. if(ctx->nimages < 2)
  155. return -1;
  156. dst = pipi_mean(ctx->images[ctx->nimages - 2],
  157. ctx->images[ctx->nimages - 1]);
  158. if(dst == NULL)
  159. return -1;
  160. pipi_free(ctx->images[ctx->nimages - 2]);
  161. pipi_free(ctx->images[ctx->nimages - 1]);
  162. ctx->images[ctx->nimages - 2] = dst;
  163. ctx->nimages--;
  164. }
  165. else if(!strcmp(cmd, "min"))
  166. {
  167. pipi_image_t *dst;
  168. if(ctx->nimages < 2)
  169. return -1;
  170. dst = pipi_min(ctx->images[ctx->nimages - 2],
  171. ctx->images[ctx->nimages - 1]);
  172. if(dst == NULL)
  173. return -1;
  174. pipi_free(ctx->images[ctx->nimages - 2]);
  175. pipi_free(ctx->images[ctx->nimages - 1]);
  176. ctx->images[ctx->nimages - 2] = dst;
  177. ctx->nimages--;
  178. }
  179. else if(!strcmp(cmd, "max"))
  180. {
  181. pipi_image_t *dst;
  182. if(ctx->nimages < 2)
  183. return -1;
  184. dst = pipi_max(ctx->images[ctx->nimages - 2],
  185. ctx->images[ctx->nimages - 1]);
  186. if(dst == NULL)
  187. return -1;
  188. pipi_free(ctx->images[ctx->nimages - 2]);
  189. pipi_free(ctx->images[ctx->nimages - 1]);
  190. ctx->images[ctx->nimages - 2] = dst;
  191. ctx->nimages--;
  192. }
  193. else if(!strcmp(cmd, "add"))
  194. {
  195. pipi_image_t *dst;
  196. if(ctx->nimages < 2)
  197. return -1;
  198. dst = pipi_add(ctx->images[ctx->nimages - 2],
  199. ctx->images[ctx->nimages - 1]);
  200. if(dst == NULL)
  201. return -1;
  202. pipi_free(ctx->images[ctx->nimages - 2]);
  203. pipi_free(ctx->images[ctx->nimages - 1]);
  204. ctx->images[ctx->nimages - 2] = dst;
  205. ctx->nimages--;
  206. }
  207. else if(!strcmp(cmd, "sub"))
  208. {
  209. pipi_image_t *dst;
  210. if(ctx->nimages < 2)
  211. return -1;
  212. dst = pipi_sub(ctx->images[ctx->nimages - 2],
  213. ctx->images[ctx->nimages - 1]);
  214. if(dst == NULL)
  215. return -1;
  216. pipi_free(ctx->images[ctx->nimages - 2]);
  217. pipi_free(ctx->images[ctx->nimages - 1]);
  218. ctx->images[ctx->nimages - 2] = dst;
  219. ctx->nimages--;
  220. }
  221. else if(!strcmp(cmd, "difference"))
  222. {
  223. pipi_image_t *dst;
  224. if(ctx->nimages < 2)
  225. return -1;
  226. dst = pipi_difference(ctx->images[ctx->nimages - 2],
  227. ctx->images[ctx->nimages - 1]);
  228. if(dst == NULL)
  229. return -1;
  230. pipi_free(ctx->images[ctx->nimages - 2]);
  231. pipi_free(ctx->images[ctx->nimages - 1]);
  232. ctx->images[ctx->nimages - 2] = dst;
  233. ctx->nimages--;
  234. }
  235. else if(!strcmp(cmd, "multiply"))
  236. {
  237. pipi_image_t *dst;
  238. if(ctx->nimages < 2)
  239. return -1;
  240. dst = pipi_multiply(ctx->images[ctx->nimages - 2],
  241. ctx->images[ctx->nimages - 1]);
  242. if(dst == NULL)
  243. return -1;
  244. pipi_free(ctx->images[ctx->nimages - 2]);
  245. pipi_free(ctx->images[ctx->nimages - 1]);
  246. ctx->images[ctx->nimages - 2] = dst;
  247. ctx->nimages--;
  248. }
  249. else if(!strcmp(cmd, "divide"))
  250. {
  251. pipi_image_t *dst;
  252. if(ctx->nimages < 2)
  253. return -1;
  254. dst = pipi_divide(ctx->images[ctx->nimages - 2],
  255. ctx->images[ctx->nimages - 1]);
  256. if(dst == NULL)
  257. return -1;
  258. pipi_free(ctx->images[ctx->nimages - 2]);
  259. pipi_free(ctx->images[ctx->nimages - 1]);
  260. ctx->images[ctx->nimages - 2] = dst;
  261. ctx->nimages--;
  262. }
  263. else if(!strcmp(cmd, "screen"))
  264. {
  265. pipi_image_t *dst;
  266. if(ctx->nimages < 2)
  267. return -1;
  268. dst = pipi_screen(ctx->images[ctx->nimages - 2],
  269. ctx->images[ctx->nimages - 1]);
  270. if(dst == NULL)
  271. return -1;
  272. pipi_free(ctx->images[ctx->nimages - 2]);
  273. pipi_free(ctx->images[ctx->nimages - 1]);
  274. ctx->images[ctx->nimages - 2] = dst;
  275. ctx->nimages--;
  276. }
  277. else if(!strcmp(cmd, "overlay"))
  278. {
  279. pipi_image_t *dst;
  280. if(ctx->nimages < 2)
  281. return -1;
  282. dst = pipi_overlay(ctx->images[ctx->nimages - 2],
  283. ctx->images[ctx->nimages - 1]);
  284. if(dst == NULL)
  285. return -1;
  286. pipi_free(ctx->images[ctx->nimages - 2]);
  287. pipi_free(ctx->images[ctx->nimages - 1]);
  288. ctx->images[ctx->nimages - 2] = dst;
  289. ctx->nimages--;
  290. }
  291. else if(!strcmp(cmd, "wrap"))
  292. {
  293. if(ctx->nimages < 1)
  294. return -1;
  295. ctx->images[ctx->nimages - 1]->wrap = 1;
  296. }
  297. else if(!strcmp(cmd, "autocontrast"))
  298. {
  299. pipi_image_t *tmp;
  300. if(ctx->nimages < 1)
  301. return -1;
  302. tmp = ctx->images[ctx->nimages - 1];
  303. ctx->images[ctx->nimages - 1] = pipi_autocontrast(tmp);
  304. pipi_free(tmp);
  305. }
  306. else if(!strcmp(cmd, "gray"))
  307. {
  308. if(ctx->nimages < 1)
  309. return -1;
  310. pipi_getpixels(ctx->images[ctx->nimages - 1], PIPI_PIXELS_Y_F);
  311. }
  312. else if(!strcmp(cmd, "free"))
  313. {
  314. if(ctx->nimages < 1)
  315. return -1;
  316. ctx->nimages--;
  317. pipi_free(ctx->images[ctx->nimages]);
  318. }
  319. else if(!strcmp(cmd, "dup"))
  320. {
  321. if(ctx->nimages < 1)
  322. return -1;
  323. ctx->images[ctx->nimages] = pipi_copy(ctx->images[ctx->nimages - 1]);
  324. ctx->nimages++;
  325. }
  326. else if(!strcmp(cmd, "swap"))
  327. {
  328. pipi_image_t *tmp;
  329. if(ctx->nimages < 2)
  330. return -1;
  331. tmp = ctx->images[ctx->nimages - 1];
  332. ctx->images[ctx->nimages - 1] = ctx->images[ctx->nimages - 2];
  333. ctx->images[ctx->nimages - 2] = tmp;
  334. }
  335. else
  336. {
  337. return -1;
  338. }
  339. return 0;
  340. }