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.
 
 
 
 
 
 

495 lines
13 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, a = 0.0;
  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. {
  119. h = atof(arg + 1);
  120. arg = strchr(arg, 'r');
  121. if(arg)
  122. a = atof(arg + 1);
  123. }
  124. src = ctx->images[ctx->nimages - 1];
  125. dst = pipi_gaussian_blur_ext(src, w, h, a, 0.0, 0.0);
  126. if(dst == NULL)
  127. return -1;
  128. pipi_free(src);
  129. ctx->images[ctx->nimages - 1] = dst;
  130. }
  131. else if(!strcmp(cmd, "boxblur"))
  132. {
  133. pipi_image_t *src, *dst;
  134. char const *arg;
  135. va_list ap;
  136. double w, h;
  137. if(ctx->nimages < 1)
  138. return -1;
  139. va_start(ap, cmd);
  140. arg = va_arg(ap, char const *);
  141. va_end(ap);
  142. w = h = atof(arg);
  143. arg = strchr(arg, 'x');
  144. if(arg)
  145. h = atof(arg + 1);
  146. src = ctx->images[ctx->nimages - 1];
  147. dst = pipi_box_blur_ext(src, w, h);
  148. if(dst == NULL)
  149. return -1;
  150. pipi_free(src);
  151. ctx->images[ctx->nimages - 1] = dst;
  152. }
  153. else if(!strcmp(cmd, "geometry"))
  154. {
  155. pipi_image_t *src, *dst;
  156. char const *arg;
  157. va_list ap;
  158. int w, h;
  159. if(ctx->nimages < 1)
  160. return -1;
  161. va_start(ap, cmd);
  162. arg = va_arg(ap, char const *);
  163. va_end(ap);
  164. w = atoi(arg);
  165. arg = strchr(arg, 'x');
  166. if(!arg)
  167. return -1;
  168. h = atoi(arg + 1);
  169. if(w <= 0 || h <= 0)
  170. return -1;
  171. src = ctx->images[ctx->nimages - 1];
  172. dst = pipi_resize(src, w, h);
  173. if(dst == NULL)
  174. return -1;
  175. pipi_free(src);
  176. ctx->images[ctx->nimages - 1] = dst;
  177. }
  178. else if(!strcmp(cmd, "tile"))
  179. {
  180. pipi_image_t *src, *dst;
  181. char const *arg;
  182. va_list ap;
  183. int w, h;
  184. if(ctx->nimages < 1)
  185. return -1;
  186. va_start(ap, cmd);
  187. arg = va_arg(ap, char const *);
  188. va_end(ap);
  189. w = atoi(arg);
  190. arg = strchr(arg, 'x');
  191. if(!arg)
  192. return -1;
  193. h = atoi(arg + 1);
  194. if(w <= 0 || h <= 0)
  195. return -1;
  196. src = ctx->images[ctx->nimages - 1];
  197. dst = pipi_tile(src, w, h);
  198. if(dst == NULL)
  199. return -1;
  200. pipi_free(src);
  201. ctx->images[ctx->nimages - 1] = dst;
  202. }
  203. else if(!strcmp(cmd, "scale"))
  204. {
  205. pipi_image_t *src, *dst;
  206. char const *arg;
  207. va_list ap;
  208. double scale;
  209. int w, h;
  210. if(ctx->nimages < 1)
  211. return -1;
  212. src = ctx->images[ctx->nimages - 1];
  213. va_start(ap, cmd);
  214. arg = va_arg(ap, char const *);
  215. va_end(ap);
  216. scale = atof(arg);
  217. w = (int)(scale * src->w + 0.5);
  218. h = (int)(scale * src->h + 0.5);
  219. if(w <= 0 || h <= 0)
  220. return -1;
  221. dst = pipi_resize(src, w, h);
  222. if(dst == NULL)
  223. return -1;
  224. pipi_free(src);
  225. ctx->images[ctx->nimages - 1] = dst;
  226. }
  227. else if(!strcmp(cmd, "brightness"))
  228. {
  229. pipi_image_t *src, *dst;
  230. char const *arg;
  231. va_list ap;
  232. double val;
  233. if(ctx->nimages < 1)
  234. return -1;
  235. va_start(ap, cmd);
  236. arg = va_arg(ap, char const *);
  237. va_end(ap);
  238. val = atof(arg);
  239. src = ctx->images[ctx->nimages - 1];
  240. dst = pipi_brightness(src, val);
  241. if(dst == NULL)
  242. return -1;
  243. pipi_free(src);
  244. ctx->images[ctx->nimages - 1] = dst;
  245. }
  246. else if(!strcmp(cmd, "contrast"))
  247. {
  248. pipi_image_t *src, *dst;
  249. char const *arg;
  250. va_list ap;
  251. double val;
  252. if(ctx->nimages < 1)
  253. return -1;
  254. va_start(ap, cmd);
  255. arg = va_arg(ap, char const *);
  256. va_end(ap);
  257. val = atof(arg);
  258. src = ctx->images[ctx->nimages - 1];
  259. dst = pipi_contrast(src, val);
  260. if(dst == NULL)
  261. return -1;
  262. pipi_free(src);
  263. ctx->images[ctx->nimages - 1] = dst;
  264. }
  265. else if(!strcmp(cmd, "mean"))
  266. {
  267. pipi_image_t *dst;
  268. if(ctx->nimages < 2)
  269. return -1;
  270. dst = pipi_mean(ctx->images[ctx->nimages - 2],
  271. ctx->images[ctx->nimages - 1]);
  272. if(dst == NULL)
  273. return -1;
  274. pipi_free(ctx->images[ctx->nimages - 2]);
  275. pipi_free(ctx->images[ctx->nimages - 1]);
  276. ctx->images[ctx->nimages - 2] = dst;
  277. ctx->nimages--;
  278. }
  279. else if(!strcmp(cmd, "min"))
  280. {
  281. pipi_image_t *dst;
  282. if(ctx->nimages < 2)
  283. return -1;
  284. dst = pipi_min(ctx->images[ctx->nimages - 2],
  285. ctx->images[ctx->nimages - 1]);
  286. if(dst == NULL)
  287. return -1;
  288. pipi_free(ctx->images[ctx->nimages - 2]);
  289. pipi_free(ctx->images[ctx->nimages - 1]);
  290. ctx->images[ctx->nimages - 2] = dst;
  291. ctx->nimages--;
  292. }
  293. else if(!strcmp(cmd, "max"))
  294. {
  295. pipi_image_t *dst;
  296. if(ctx->nimages < 2)
  297. return -1;
  298. dst = pipi_max(ctx->images[ctx->nimages - 2],
  299. ctx->images[ctx->nimages - 1]);
  300. if(dst == NULL)
  301. return -1;
  302. pipi_free(ctx->images[ctx->nimages - 2]);
  303. pipi_free(ctx->images[ctx->nimages - 1]);
  304. ctx->images[ctx->nimages - 2] = dst;
  305. ctx->nimages--;
  306. }
  307. else if(!strcmp(cmd, "add"))
  308. {
  309. pipi_image_t *dst;
  310. if(ctx->nimages < 2)
  311. return -1;
  312. dst = pipi_add(ctx->images[ctx->nimages - 2],
  313. ctx->images[ctx->nimages - 1]);
  314. if(dst == NULL)
  315. return -1;
  316. pipi_free(ctx->images[ctx->nimages - 2]);
  317. pipi_free(ctx->images[ctx->nimages - 1]);
  318. ctx->images[ctx->nimages - 2] = dst;
  319. ctx->nimages--;
  320. }
  321. else if(!strcmp(cmd, "sub"))
  322. {
  323. pipi_image_t *dst;
  324. if(ctx->nimages < 2)
  325. return -1;
  326. dst = pipi_sub(ctx->images[ctx->nimages - 2],
  327. ctx->images[ctx->nimages - 1]);
  328. if(dst == NULL)
  329. return -1;
  330. pipi_free(ctx->images[ctx->nimages - 2]);
  331. pipi_free(ctx->images[ctx->nimages - 1]);
  332. ctx->images[ctx->nimages - 2] = dst;
  333. ctx->nimages--;
  334. }
  335. else if(!strcmp(cmd, "difference"))
  336. {
  337. pipi_image_t *dst;
  338. if(ctx->nimages < 2)
  339. return -1;
  340. dst = pipi_difference(ctx->images[ctx->nimages - 2],
  341. ctx->images[ctx->nimages - 1]);
  342. if(dst == NULL)
  343. return -1;
  344. pipi_free(ctx->images[ctx->nimages - 2]);
  345. pipi_free(ctx->images[ctx->nimages - 1]);
  346. ctx->images[ctx->nimages - 2] = dst;
  347. ctx->nimages--;
  348. }
  349. else if(!strcmp(cmd, "multiply"))
  350. {
  351. pipi_image_t *dst;
  352. if(ctx->nimages < 2)
  353. return -1;
  354. dst = pipi_multiply(ctx->images[ctx->nimages - 2],
  355. ctx->images[ctx->nimages - 1]);
  356. if(dst == NULL)
  357. return -1;
  358. pipi_free(ctx->images[ctx->nimages - 2]);
  359. pipi_free(ctx->images[ctx->nimages - 1]);
  360. ctx->images[ctx->nimages - 2] = dst;
  361. ctx->nimages--;
  362. }
  363. else if(!strcmp(cmd, "divide"))
  364. {
  365. pipi_image_t *dst;
  366. if(ctx->nimages < 2)
  367. return -1;
  368. dst = pipi_divide(ctx->images[ctx->nimages - 2],
  369. ctx->images[ctx->nimages - 1]);
  370. if(dst == NULL)
  371. return -1;
  372. pipi_free(ctx->images[ctx->nimages - 2]);
  373. pipi_free(ctx->images[ctx->nimages - 1]);
  374. ctx->images[ctx->nimages - 2] = dst;
  375. ctx->nimages--;
  376. }
  377. else if(!strcmp(cmd, "screen"))
  378. {
  379. pipi_image_t *dst;
  380. if(ctx->nimages < 2)
  381. return -1;
  382. dst = pipi_screen(ctx->images[ctx->nimages - 2],
  383. ctx->images[ctx->nimages - 1]);
  384. if(dst == NULL)
  385. return -1;
  386. pipi_free(ctx->images[ctx->nimages - 2]);
  387. pipi_free(ctx->images[ctx->nimages - 1]);
  388. ctx->images[ctx->nimages - 2] = dst;
  389. ctx->nimages--;
  390. }
  391. else if(!strcmp(cmd, "overlay"))
  392. {
  393. pipi_image_t *dst;
  394. if(ctx->nimages < 2)
  395. return -1;
  396. dst = pipi_overlay(ctx->images[ctx->nimages - 2],
  397. ctx->images[ctx->nimages - 1]);
  398. if(dst == NULL)
  399. return -1;
  400. pipi_free(ctx->images[ctx->nimages - 2]);
  401. pipi_free(ctx->images[ctx->nimages - 1]);
  402. ctx->images[ctx->nimages - 2] = dst;
  403. ctx->nimages--;
  404. }
  405. else if(!strcmp(cmd, "wrap"))
  406. {
  407. if(ctx->nimages < 1)
  408. return -1;
  409. ctx->images[ctx->nimages - 1]->wrap = 1;
  410. }
  411. else if(!strcmp(cmd, "autocontrast"))
  412. {
  413. pipi_image_t *tmp;
  414. if(ctx->nimages < 1)
  415. return -1;
  416. tmp = ctx->images[ctx->nimages - 1];
  417. ctx->images[ctx->nimages - 1] = pipi_autocontrast(tmp);
  418. pipi_free(tmp);
  419. }
  420. else if(!strcmp(cmd, "invert"))
  421. {
  422. pipi_image_t *tmp;
  423. if(ctx->nimages < 1)
  424. return -1;
  425. tmp = ctx->images[ctx->nimages - 1];
  426. ctx->images[ctx->nimages - 1] = pipi_invert(tmp);
  427. pipi_free(tmp);
  428. }
  429. else if(!strcmp(cmd, "gray"))
  430. {
  431. if(ctx->nimages < 1)
  432. return -1;
  433. pipi_getpixels(ctx->images[ctx->nimages - 1], PIPI_PIXELS_Y_F);
  434. }
  435. else if(!strcmp(cmd, "free"))
  436. {
  437. if(ctx->nimages < 1)
  438. return -1;
  439. ctx->nimages--;
  440. pipi_free(ctx->images[ctx->nimages]);
  441. }
  442. else if(!strcmp(cmd, "dup"))
  443. {
  444. if(ctx->nimages < 1)
  445. return -1;
  446. ctx->images[ctx->nimages] = pipi_copy(ctx->images[ctx->nimages - 1]);
  447. ctx->nimages++;
  448. }
  449. else if(!strcmp(cmd, "swap"))
  450. {
  451. pipi_image_t *tmp;
  452. if(ctx->nimages < 2)
  453. return -1;
  454. tmp = ctx->images[ctx->nimages - 1];
  455. ctx->images[ctx->nimages - 1] = ctx->images[ctx->nimages - 2];
  456. ctx->images[ctx->nimages - 2] = tmp;
  457. }
  458. else
  459. {
  460. return -1;
  461. }
  462. return 0;
  463. }