interp.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. /*!
  2. * \file raster/interp.c
  3. *
  4. * \brief Raster Library - Interpolation
  5. *
  6. * (C) 2001-2009 GRASS Development Team
  7. *
  8. * This program is free software under the GNU General Public License
  9. * (>=v2). Read the file COPYING that comes with GRASS for details.
  10. *
  11. * \author Original author CERL
  12. */
  13. #include <math.h>
  14. #include <grass/gis.h>
  15. #include <grass/raster.h>
  16. #define LANCZOS_FILTER(x) ((2 * sin(x) * sin((x) / 2)) / ((x) * (x)))
  17. DCELL Rast_interp_linear(double u, DCELL c0, DCELL c1)
  18. {
  19. return u * (c1 - c0) + c0;
  20. }
  21. DCELL Rast_interp_bilinear(double u, double v,
  22. DCELL c00, DCELL c01, DCELL c10, DCELL c11)
  23. {
  24. DCELL c0 = Rast_interp_linear(u, c00, c01);
  25. DCELL c1 = Rast_interp_linear(u, c10, c11);
  26. return Rast_interp_linear(v, c0, c1);
  27. }
  28. DCELL Rast_interp_cubic(double u, DCELL c0, DCELL c1, DCELL c2, DCELL c3)
  29. {
  30. return (u * (u * (u * (c3 - 3 * c2 + 3 * c1 - c0) +
  31. (-c3 + 4 * c2 - 5 * c1 + 2 * c0)) + (c2 - c0)) +
  32. 2 * c1) / 2;
  33. }
  34. DCELL Rast_interp_bicubic(double u, double v,
  35. DCELL c00, DCELL c01, DCELL c02, DCELL c03,
  36. DCELL c10, DCELL c11, DCELL c12, DCELL c13,
  37. DCELL c20, DCELL c21, DCELL c22, DCELL c23,
  38. DCELL c30, DCELL c31, DCELL c32, DCELL c33)
  39. {
  40. DCELL c0 = Rast_interp_cubic(u, c00, c01, c02, c03);
  41. DCELL c1 = Rast_interp_cubic(u, c10, c11, c12, c13);
  42. DCELL c2 = Rast_interp_cubic(u, c20, c21, c22, c23);
  43. DCELL c3 = Rast_interp_cubic(u, c30, c31, c32, c33);
  44. return Rast_interp_cubic(v, c0, c1, c2, c3);
  45. }
  46. DCELL Rast_interp_lanczos(double u, double v, DCELL *c)
  47. {
  48. int i;
  49. double uweight[5], vweight[5], d;
  50. for (i = 0; i < 5; i++) {
  51. d = v - i + 2;
  52. if (d == 0)
  53. vweight[i] = 1;
  54. else {
  55. d *= M_PI;
  56. vweight[i] = LANCZOS_FILTER(d);
  57. }
  58. d = u - i + 2;
  59. if (d == 0)
  60. uweight[i] = 1;
  61. else {
  62. d *= M_PI;
  63. uweight[i] = LANCZOS_FILTER(d);
  64. }
  65. }
  66. return ((c[0] * vweight[0] + c[1] * vweight[1] + c[2] * vweight[2]
  67. + c[3] * vweight[3] + c[4] * vweight[4]) * uweight[0] +
  68. (c[5] * vweight[0] + c[6] * vweight[1] + c[7] * vweight[2]
  69. + c[8] * vweight[3] + c[9] * vweight[4]) * uweight[1] +
  70. (c[10] * vweight[0] + c[11] * vweight[1] + c[12] * vweight[2]
  71. + c[13] * vweight[3] + c[14] * vweight[4]) * uweight[2] +
  72. (c[15] * vweight[0] + c[16] * vweight[1] + c[17] * vweight[2]
  73. + c[18] * vweight[3] + c[19] * vweight[4]) * uweight[3] +
  74. (c[20] * vweight[0] + c[21] * vweight[1] + c[22] * vweight[2]
  75. + c[23] * vweight[3] + c[24] * vweight[4]) * uweight[4]);
  76. }
  77. DCELL Rast_interp_cubic_bspline(double u, DCELL c0, DCELL c1, DCELL c2, DCELL c3)
  78. {
  79. return (u * (u * (u * (c3 - 3 * c2 + 3 * c1 - c0) +
  80. (3 * c2 - 6 * c1 + 3 * c0)) +
  81. (3 * c2 - 3 * c0)) +
  82. c2 + 4 * c1 + c0) / 6;
  83. }
  84. DCELL Rast_interp_bicubic_bspline(double u, double v,
  85. DCELL c00, DCELL c01, DCELL c02, DCELL c03,
  86. DCELL c10, DCELL c11, DCELL c12, DCELL c13,
  87. DCELL c20, DCELL c21, DCELL c22, DCELL c23,
  88. DCELL c30, DCELL c31, DCELL c32, DCELL c33)
  89. {
  90. DCELL c0 = Rast_interp_cubic_bspline(u, c00, c01, c02, c03);
  91. DCELL c1 = Rast_interp_cubic_bspline(u, c10, c11, c12, c13);
  92. DCELL c2 = Rast_interp_cubic_bspline(u, c20, c21, c22, c23);
  93. DCELL c3 = Rast_interp_cubic_bspline(u, c30, c31, c32, c33);
  94. return Rast_interp_cubic_bspline(v, c0, c1, c2, c3);
  95. }