interp.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /*!
  2. \file lib/raster/interp.c
  3. \brief Raster Library - Interpolation methods
  4. (C) 2001-2009,2013 GRASS Development Team
  5. This program is free software under the GNU General Public License
  6. (>=v2). Read the file COPYING that comes with GRASS for details.
  7. \author Original author CERL
  8. */
  9. #include <math.h>
  10. #include <string.h>
  11. #include <grass/gis.h>
  12. #include <grass/raster.h>
  13. #include <grass/glocale.h>
  14. DCELL Rast_interp_linear(double u, DCELL c0, DCELL c1)
  15. {
  16. return u * (c1 - c0) + c0;
  17. }
  18. DCELL Rast_interp_bilinear(double u, double v,
  19. DCELL c00, DCELL c01, DCELL c10, DCELL c11)
  20. {
  21. DCELL c0 = Rast_interp_linear(u, c00, c01);
  22. DCELL c1 = Rast_interp_linear(u, c10, c11);
  23. return Rast_interp_linear(v, c0, c1);
  24. }
  25. DCELL Rast_interp_cubic(double u, DCELL c0, DCELL c1, DCELL c2, DCELL c3)
  26. {
  27. return (u * (u * (u * (c3 - 3 * c2 + 3 * c1 - c0) +
  28. (-c3 + 4 * c2 - 5 * c1 + 2 * c0)) + (c2 - c0)) +
  29. 2 * c1) / 2;
  30. }
  31. DCELL Rast_interp_bicubic(double u, double v,
  32. DCELL c00, DCELL c01, DCELL c02, DCELL c03,
  33. DCELL c10, DCELL c11, DCELL c12, DCELL c13,
  34. DCELL c20, DCELL c21, DCELL c22, DCELL c23,
  35. DCELL c30, DCELL c31, DCELL c32, DCELL c33)
  36. {
  37. DCELL c0 = Rast_interp_cubic(u, c00, c01, c02, c03);
  38. DCELL c1 = Rast_interp_cubic(u, c10, c11, c12, c13);
  39. DCELL c2 = Rast_interp_cubic(u, c20, c21, c22, c23);
  40. DCELL c3 = Rast_interp_cubic(u, c30, c31, c32, c33);
  41. return Rast_interp_cubic(v, c0, c1, c2, c3);
  42. }
  43. DCELL Rast_interp_lanczos(double u, double v, DCELL *c)
  44. {
  45. double uweight[5], vweight[5], d, d_pi;
  46. double usum, vsum;
  47. DCELL c0, c1, c2, c3, c4;
  48. double sind, sincd1, sincd2;
  49. d_pi = u * M_PI;
  50. sind = 2 * sin(d_pi);
  51. sincd1 = sind * sin(d_pi / 2);
  52. uweight[2] = (u == 0 ? 1 : sincd1 / (d_pi * d_pi));
  53. usum = uweight[2];
  54. d = u + 2;
  55. d_pi = d * M_PI;
  56. if (d > 2)
  57. uweight[0] = 0.;
  58. else
  59. uweight[0] = (d == 0 ? 1 : -sincd1 / (d_pi * d_pi));
  60. usum += uweight[0];
  61. d = u + 1.;
  62. d_pi = d * M_PI;
  63. sincd2 = sind * sin(d_pi / 2);
  64. uweight[1] = (d == 0 ? 1 : -sincd2 / (d_pi * d_pi));
  65. usum += uweight[1];
  66. d = u - 1.;
  67. d_pi = d * M_PI;
  68. uweight[3] = (d == 0 ? 1 : sincd2 / (d_pi * d_pi));
  69. usum += uweight[3];
  70. d = u - 2.;
  71. d_pi = d * M_PI;
  72. if (d < -2)
  73. uweight[4] = 0.;
  74. else
  75. uweight[4] = (d == 0 ? 1 : -sincd1 / (d_pi * d_pi));
  76. usum += uweight[4];
  77. d_pi = v * M_PI;
  78. sind = 2 * sin(d_pi);
  79. sincd1 = sind * sin(d_pi / 2);
  80. vweight[2] = (v == 0 ? 1 : sincd1 / (d_pi * d_pi));
  81. vsum = vweight[2];
  82. d = v + 2;
  83. d_pi = d * M_PI;
  84. if (d > 2)
  85. vweight[0] = 0;
  86. else
  87. vweight[0] = (d == 0 ? 1 : -sincd1 / (d_pi * d_pi));
  88. vsum += vweight[0];
  89. d = v + 1.;
  90. d_pi = d * M_PI;
  91. sincd2 = sind * sin(d_pi / 2);
  92. vweight[1] = (d == 0 ? 1 : -sincd2 / (d_pi * d_pi));
  93. vsum += vweight[1];
  94. d = v - 1.;
  95. d_pi = d * M_PI;
  96. vweight[3] = (d == 0 ? 1 : sincd2 / (d_pi * d_pi));
  97. vsum += vweight[3];
  98. d = v - 2.;
  99. d_pi = d * M_PI;
  100. if (d < -2)
  101. vweight[4] = 0;
  102. else
  103. vweight[4] = (d == 0 ? 1 : -sincd1 / (d_pi * d_pi));
  104. vsum += vweight[4];
  105. c0 = (c[0] * uweight[0] + c[1] * uweight[1] + c[2] * uweight[2]
  106. + c[3] * uweight[3] + c[4] * uweight[4]);
  107. c1 = (c[5] * uweight[0] + c[6] * uweight[1] + c[7] * uweight[2]
  108. + c[8] * uweight[3] + c[9] * uweight[4]);
  109. c2 = (c[10] * uweight[0] + c[11] * uweight[1] + c[12] * uweight[2]
  110. + c[13] * uweight[3] + c[14] * uweight[4]);
  111. c3 = (c[15] * uweight[0] + c[16] * uweight[1] + c[17] * uweight[2]
  112. + c[18] * uweight[3] + c[19] * uweight[4]);
  113. c4 = (c[20] * uweight[0] + c[21] * uweight[1] + c[22] * uweight[2]
  114. + c[23] * uweight[3] + c[24] * uweight[4]);
  115. return ((c0 * vweight[0] + c1 * vweight[1] + c2 * vweight[2] +
  116. c3 * vweight[3] + c4 * vweight[4]) / (usum * vsum));
  117. }
  118. DCELL Rast_interp_cubic_bspline(double u, DCELL c0, DCELL c1, DCELL c2, DCELL c3)
  119. {
  120. return (u * (u * (u * (c3 - 3 * c2 + 3 * c1 - c0) +
  121. (3 * c2 - 6 * c1 + 3 * c0)) +
  122. (3 * c2 - 3 * c0)) +
  123. c2 + 4 * c1 + c0) / 6;
  124. }
  125. DCELL Rast_interp_bicubic_bspline(double u, double v,
  126. DCELL c00, DCELL c01, DCELL c02, DCELL c03,
  127. DCELL c10, DCELL c11, DCELL c12, DCELL c13,
  128. DCELL c20, DCELL c21, DCELL c22, DCELL c23,
  129. DCELL c30, DCELL c31, DCELL c32, DCELL c33)
  130. {
  131. DCELL c0 = Rast_interp_cubic_bspline(u, c00, c01, c02, c03);
  132. DCELL c1 = Rast_interp_cubic_bspline(u, c10, c11, c12, c13);
  133. DCELL c2 = Rast_interp_cubic_bspline(u, c20, c21, c22, c23);
  134. DCELL c3 = Rast_interp_cubic_bspline(u, c30, c31, c32, c33);
  135. return Rast_interp_cubic_bspline(v, c0, c1, c2, c3);
  136. }
  137. /*!
  138. \brief Get interpolation method from the option.
  139. Calls G_fatal_error() on unknown interpolation method.
  140. Supported methods:
  141. - NEAREST
  142. - BILINEAR
  143. - CUBIC
  144. \code
  145. int interp_method
  146. struct Option *opt_method;
  147. opt_method = G_define_standard_option(G_OPT_R_INTERP_TYPE);
  148. if (G_parser(argc, argv))
  149. exit(EXIT_FAILURE);
  150. interp_method = G_option_to_interp_type(opt_method);
  151. \endcode
  152. \param option pointer to interpolation option
  153. \return interpolation method code
  154. */
  155. int Rast_option_to_interp_type(const struct Option *option)
  156. {
  157. int interp_type;
  158. interp_type = INTERP_UNKNOWN;
  159. if (option->answer) {
  160. if (strcmp(option->answer, "nearest") == 0) {
  161. interp_type = INTERP_NEAREST;
  162. }
  163. else if (strcmp(option->answer, "bilinear") == 0) {
  164. interp_type = INTERP_BILINEAR;
  165. }
  166. else if (strcmp(option->answer, "bicubic") == 0) {
  167. interp_type = INTERP_BICUBIC;
  168. }
  169. }
  170. if (interp_type == INTERP_UNKNOWN)
  171. G_fatal_error(_("Unknown interpolation method: %s"), option->answer);
  172. return interp_type;
  173. }