224 行
5.9 KiB

  1. /*
  2. * libpipi Pathetic image processing interface 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. * blur.c: blur functions
  16. */
  17. #include "config.h"
  18. #include <stdlib.h>
  19. #include <stdio.h>
  20. #include <string.h>
  21. #include <math.h>
  22. #ifndef M_PI
  23. # define M_PI 3.14159265358979323846
  24. #endif
  25. #include "pipi.h"
  26. #include "pipi-internals.h"
  27. #if !defined TEMPLATE_FILE /* This file uses the template system */
  28. #define TEMPLATE_FLAGS SET_FLAG_GRAY | SET_FLAG_WRAP
  29. #define TEMPLATE_FILE "filter/blur.c"
  30. #include "pipi-template.h"
  31. pipi_image_t *pipi_box_blur(pipi_image_t *src, int size)
  32. {
  33. return pipi_box_blur_ext(src, size, size);
  34. }
  35. pipi_image_t *pipi_box_blur_ext(pipi_image_t *src, int m, int n)
  36. {
  37. if(src->wrap)
  38. {
  39. if(src->last_modified == PIPI_PIXELS_Y_F32)
  40. return boxblur_gray_wrap(src, m, n);
  41. return boxblur_wrap(src, m, n);
  42. }
  43. else
  44. {
  45. if(src->last_modified == PIPI_PIXELS_Y_F32)
  46. return boxblur_gray(src, m, n);
  47. return boxblur(src, m, n);
  48. }
  49. }
  50. #else /* XXX: the following functions use the template system */
  51. static pipi_image_t *T(boxblur)(pipi_image_t *src, int m, int n)
  52. {
  53. pipi_image_t *dst;
  54. pipi_pixels_t *srcp, *dstp;
  55. float *srcdata, *dstdata;
  56. double *acc;
  57. int x, y, w, h, i, j, i2, j2, size;
  58. w = src->w;
  59. h = src->h;
  60. size = (2 * m + 1) * (2 * n + 1);
  61. srcp = FLAG_GRAY ? pipi_get_pixels(src, PIPI_PIXELS_Y_F32)
  62. : pipi_get_pixels(src, PIPI_PIXELS_RGBA_F32);
  63. srcdata = (float *)srcp->pixels;
  64. dst = pipi_new(w, h);
  65. dstp = FLAG_GRAY ? pipi_get_pixels(dst, PIPI_PIXELS_Y_F32)
  66. : pipi_get_pixels(dst, PIPI_PIXELS_RGBA_F32);
  67. dstdata = (float *)dstp->pixels;
  68. acc = malloc(w * (FLAG_GRAY ? 1 : 4) * sizeof(double));
  69. /* Step 1: fill the accumulator */
  70. for(x = 0; x < w; x++)
  71. {
  72. double r = 0., g = 0., b = 0., a = 0.;
  73. double t = 0.;
  74. for(j = -n; j <= n; j++)
  75. {
  76. if(FLAG_WRAP)
  77. j2 = (j < 0) ? h - 1 - ((-j - 1) % h) : j % h;
  78. else
  79. j2 = (j < 0) ? 0 : (j >= h) ? h - 1 : j;
  80. if(FLAG_GRAY)
  81. t += srcdata[j2 * w + x];
  82. else
  83. {
  84. r += srcdata[4 * (j2 * w + x)];
  85. g += srcdata[4 * (j2 * w + x) + 1];
  86. b += srcdata[4 * (j2 * w + x) + 2];
  87. a += srcdata[4 * (j2 * w + x) + 3];
  88. }
  89. }
  90. if(FLAG_GRAY)
  91. acc[x] = t;
  92. else
  93. {
  94. acc[4 * x] = r;
  95. acc[4 * x + 1] = g;
  96. acc[4 * x + 2] = b;
  97. acc[4 * x + 3] = a;
  98. }
  99. }
  100. /* Step 2: blur the image, line by line */
  101. for(y = 0; y < h; y++)
  102. {
  103. double r = 0., g = 0., b = 0., a = 0.;
  104. double t = 0.;
  105. /* 2.1: compute the first pixel */
  106. for(i = -m; i <= m; i++)
  107. {
  108. if(FLAG_WRAP)
  109. i2 = (i < 0) ? w - 1 - ((-i - 1) % w) : i % w;
  110. else
  111. i2 = (i < 0) ? 0 : (i >= w) ? w - 1 : i;
  112. if(FLAG_GRAY)
  113. t += acc[i2];
  114. else
  115. {
  116. r += acc[4 * i2];
  117. g += acc[4 * i2 + 1];
  118. b += acc[4 * i2 + 2];
  119. a += acc[4 * i2 + 3];
  120. }
  121. }
  122. /* 2.2: iterate on the whole line */
  123. for(x = 0; x < w; x++)
  124. {
  125. int u, u2, v, v2;
  126. if(FLAG_GRAY)
  127. {
  128. dstdata[y * w + x] = t / size;
  129. }
  130. else
  131. {
  132. dstdata[4 * (y * w + x)] = r / size;
  133. dstdata[4 * (y * w + x) + 1] = g / size;
  134. dstdata[4 * (y * w + x) + 2] = b / size;
  135. dstdata[4 * (y * w + x) + 3] = a / size;
  136. }
  137. u = x - m;
  138. if(FLAG_WRAP)
  139. u2 = (u < 0) ? w - 1 - ((-u - 1) % w) : u % w;
  140. else
  141. u2 = (u < 0) ? 0 : (u >= w) ? w - 1 : u;
  142. v = x + m + 1;
  143. if(FLAG_WRAP)
  144. v2 = (v < 0) ? w - 1 - ((-v - 1) % w) : v % w;
  145. else
  146. v2 = (v < 0) ? 0 : (v >= w) ? w - 1 : v;
  147. if(FLAG_GRAY)
  148. {
  149. t = t - acc[u2] + acc[v2];
  150. }
  151. else
  152. {
  153. r = r - acc[4 * u2] + acc[4 * v2];
  154. g = g - acc[4 * u2 + 1] + acc[4 * v2 + 1];
  155. b = b - acc[4 * u2 + 2] + acc[4 * v2 + 2];
  156. a = a - acc[4 * u2 + 3] + acc[4 * v2 + 3];
  157. }
  158. }
  159. /* 2.3: update the accumulator */
  160. for(x = 0; x < w; x++)
  161. {
  162. int u, u2, v, v2;
  163. u = y - n;
  164. if(FLAG_WRAP)
  165. u2 = (u < 0) ? h - 1 - ((-u - 1) % h) : u % h;
  166. else
  167. u2 = (u < 0) ? 0 : (u >= h) ? h - 1 : u;
  168. v = y + n + 1;
  169. if(FLAG_WRAP)
  170. v2 = (v < 0) ? h - 1 - ((-v - 1) % h) : v % h;
  171. else
  172. v2 = (v < 0) ? 0 : (v >= h) ? h - 1 : v;
  173. if(FLAG_GRAY)
  174. {
  175. acc[x] += srcdata[v2 * w + x] - srcdata[u2 * w + x];
  176. }
  177. else
  178. {
  179. int uoff = 4 * (u2 * w + x);
  180. int voff = 4 * (v2 * w + x);
  181. acc[4 * x] += srcdata[voff] - srcdata[uoff];
  182. acc[4 * x + 1] += srcdata[voff + 1] - srcdata[uoff + 1];
  183. acc[4 * x + 2] += srcdata[voff + 2] - srcdata[uoff + 2];
  184. acc[4 * x + 3] += srcdata[voff + 3] - srcdata[uoff + 3];
  185. }
  186. }
  187. }
  188. free(acc);
  189. return dst;
  190. }
  191. #endif