/octave-3.6.2/liboctave/oct-binmap.h

# · C Header · 467 lines · 321 code · 80 blank · 66 comment · 50 complexity · a1bfdcec4d2aace363ed7a0ef5d52ef8 MD5 · raw file

  1. /*
  2. Copyright (C) 2010-2012 VZLU Prague
  3. This file is part of Octave.
  4. Octave is free software; you can redistribute it and/or modify it
  5. under the terms of the GNU General Public License as published by the
  6. Free Software Foundation; either version 3 of the License, or (at your
  7. option) any later version.
  8. Octave is distributed in the hope that it will be useful, but WITHOUT
  9. ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
  11. for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with Octave; see the file COPYING. If not, see
  14. <http://www.gnu.org/licenses/>.
  15. */
  16. #if !defined (octave_binmap_h)
  17. #define octave_binmap_h 1
  18. #include "Array.h"
  19. #include "Sparse.h"
  20. #include "Array-util.h"
  21. #include "bsxfun.h"
  22. // This source file implements a general binary maping function for
  23. // arrays. The syntax is binmap<type> (a, b, f, [name]). type denotes
  24. // the expected return type of the operation. a, b, should be one of
  25. // the 6 combinations:
  26. //
  27. // Array-Array
  28. // Array-scalar
  29. // scalar-Array
  30. // Sparse-Sparse
  31. // Sparse-scalar
  32. // scalar-Sparse
  33. //
  34. // If both operands are nonscalar, name must be supplied. It is used
  35. // as the base for error message when operands are nonconforming.
  36. //
  37. // The operation needs not be homogeneous, i.e. a, b and the result
  38. // may be of distinct types. f can have any of the four signatures:
  39. //
  40. // U f (T, R)
  41. // U f (const T&, R)
  42. // U f (T, const R&)
  43. // U f (const T&, const R&)
  44. //
  45. // Additionally, f can be an arbitrary functor object.
  46. //
  47. // octave_quit() is called at appropriate places, hence the operation
  48. // is breakable.
  49. // The following template wrappers are provided for automatic bsxfun
  50. // calls (see the function signature for do_bsxfun_op).
  51. template<typename R, typename X, typename Y, typename F>
  52. class bsxfun_wrapper
  53. {
  54. private:
  55. static F f;
  56. public:
  57. static void
  58. set_f (const F& f_in)
  59. {
  60. f = f_in;
  61. }
  62. static void
  63. op_mm (size_t n, R* r, const X* x , const Y* y)
  64. {
  65. for (size_t i = 0; i < n; i++)
  66. r[i] = f (x[i], y[i]);
  67. }
  68. static void
  69. op_sm (size_t n, R* r, X x, const Y* y)
  70. {
  71. for (size_t i = 0; i < n; i++)
  72. r[i] = f (x, y[i]);
  73. }
  74. static void
  75. op_ms (size_t n , R* r, const X* x, Y y)
  76. {
  77. for (size_t i = 0; i < n; i++)
  78. r[i] = f (x[i], y);
  79. }
  80. };
  81. // Static init
  82. template<typename R, typename X, typename Y, typename F>
  83. F bsxfun_wrapper<R, X, Y, F>::f;
  84. // scalar-Array
  85. template <class U, class T, class R, class F>
  86. Array<U>
  87. binmap (const T& x, const Array<R>& ya, F fcn)
  88. {
  89. octave_idx_type len = ya.numel ();
  90. const R *y = ya.data ();
  91. Array<U> result (ya.dims ());
  92. U *p = result.fortran_vec ();
  93. octave_idx_type i;
  94. for (i = 0; i < len - 3; i += 4)
  95. {
  96. octave_quit ();
  97. p[i] = fcn (x, y[i]);
  98. p[i+1] = fcn (x, y[i+1]);
  99. p[i+2] = fcn (x, y[i+2]);
  100. p[i+3] = fcn (x, y[i+3]);
  101. }
  102. octave_quit ();
  103. for (; i < len; i++)
  104. p[i] = fcn (x, y[i]);
  105. return result;
  106. }
  107. // Array-scalar
  108. template <class U, class T, class R, class F>
  109. Array<U>
  110. binmap (const Array<T>& xa, const R& y, F fcn)
  111. {
  112. octave_idx_type len = xa.numel ();
  113. const R *x = xa.data ();
  114. Array<U> result (xa.dims ());
  115. U *p = result.fortran_vec ();
  116. octave_idx_type i;
  117. for (i = 0; i < len - 3; i += 4)
  118. {
  119. octave_quit ();
  120. p[i] = fcn (x[i], y);
  121. p[i+1] = fcn (x[i+1], y);
  122. p[i+2] = fcn (x[i+2], y);
  123. p[i+3] = fcn (x[i+3], y);
  124. }
  125. octave_quit ();
  126. for (; i < len; i++)
  127. p[i] = fcn (x[i], y);
  128. return result;
  129. }
  130. // Array-Array (treats singletons as scalars)
  131. template <class U, class T, class R, class F>
  132. Array<U>
  133. binmap (const Array<T>& xa, const Array<R>& ya, F fcn, const char *name)
  134. {
  135. dim_vector xad = xa.dims (), yad = ya.dims ();
  136. if (xa.numel () == 1)
  137. return binmap<U, T, R, F> (xa(0), ya, fcn);
  138. else if (ya.numel () == 1)
  139. return binmap<U, T, R, F> (xa, ya(0), fcn);
  140. else if (xad != yad)
  141. {
  142. if (is_valid_bsxfun (name, xad, yad))
  143. {
  144. bsxfun_wrapper<U, T, R, F>::set_f(fcn);
  145. return do_bsxfun_op (xa, ya,
  146. bsxfun_wrapper<U, T, R, F>::op_mm,
  147. bsxfun_wrapper<U, T, R, F>::op_sm,
  148. bsxfun_wrapper<U, T, R, F>::op_ms);
  149. }
  150. else
  151. gripe_nonconformant (name, xad, yad);
  152. }
  153. octave_idx_type len = xa.numel ();
  154. const T *x = xa.data ();
  155. const T *y = ya.data ();
  156. Array<U> result (xa.dims ());
  157. U *p = result.fortran_vec ();
  158. octave_idx_type i;
  159. for (i = 0; i < len - 3; i += 4)
  160. {
  161. octave_quit ();
  162. p[i] = fcn (x[i], y[i]);
  163. p[i+1] = fcn (x[i+1], y[i+1]);
  164. p[i+2] = fcn (x[i+2], y[i+2]);
  165. p[i+3] = fcn (x[i+3], y[i+3]);
  166. }
  167. octave_quit ();
  168. for (; i < len; i++)
  169. p[i] = fcn (x[i], y[i]);
  170. return result;
  171. }
  172. // scalar-Sparse
  173. template <class U, class T, class R, class F>
  174. Sparse<U>
  175. binmap (const T& x, const Sparse<R>& ys, F fcn)
  176. {
  177. octave_idx_type nz = ys.nnz ();
  178. Sparse<U> retval (ys.rows (), ys.cols (), nz);
  179. for (octave_idx_type i = 0; i < nz; i++)
  180. {
  181. octave_quit ();
  182. retval.xdata(i) = fcn (x, ys.data(i));
  183. }
  184. octave_quit ();
  185. retval.maybe_compress ();
  186. return retval;
  187. }
  188. // Sparse-scalar
  189. template <class U, class T, class R, class F>
  190. Sparse<U>
  191. binmap (const Sparse<T>& xs, const R& y, F fcn)
  192. {
  193. octave_idx_type nz = xs.nnz ();
  194. Sparse<U> retval (xs.rows (), xs.cols (), nz);
  195. for (octave_idx_type i = 0; i < nz; i++)
  196. {
  197. octave_quit ();
  198. retval.xdata(i) = fcn (xs.data(i), y);
  199. }
  200. octave_quit ();
  201. retval.maybe_compress ();
  202. return retval;
  203. }
  204. // Sparse-Sparse (treats singletons as scalars)
  205. template <class U, class T, class R, class F>
  206. Sparse<U>
  207. binmap (const Sparse<T>& xs, const Sparse<R>& ys, F fcn, const char *name)
  208. {
  209. if (xs.rows () == 1 && xs.cols () == 1)
  210. return binmap<U, T, R, F> (xs(0,0), ys, fcn);
  211. else if (ys.rows () == 1 && ys.cols () == 1)
  212. return binmap<U, T, R, F> (xs, ys(0,0), fcn);
  213. else if (xs.dims () != ys.dims ())
  214. gripe_nonconformant (name, xs.dims (), ys.dims ());
  215. T xzero = T ();
  216. R yzero = R ();
  217. U fz = fcn (xzero, yzero);
  218. if (fz == U())
  219. {
  220. // Sparsity-preserving function. Do it efficiently.
  221. octave_idx_type nr = xs.rows (), nc = xs.cols ();
  222. Sparse<T> retval (nr, nc);
  223. octave_idx_type nz = 0;
  224. // Count nonzeros.
  225. for (octave_idx_type j = 0; j < nc; j++)
  226. {
  227. octave_quit ();
  228. octave_idx_type ix = xs.cidx(j), iy = ys.cidx(j);
  229. octave_idx_type ux = xs.cidx(j+1), uy = ys.cidx(j+1);
  230. while (ix != ux || iy != uy)
  231. {
  232. octave_idx_type rx = xs.ridx(ix), ry = ys.ridx(ix);
  233. ix += rx <= ry;
  234. iy += ry <= rx;
  235. nz++;
  236. }
  237. retval.xcidx(j+1) = nz;
  238. }
  239. // Allocate space.
  240. retval.change_capacity (retval.xcidx(nc));
  241. // Fill.
  242. nz = 0;
  243. for (octave_idx_type j = 0; j < nc; j++)
  244. {
  245. octave_quit ();
  246. octave_idx_type ix = xs.cidx(j), iy = ys.cidx(j);
  247. octave_idx_type ux = xs.cidx(j+1), uy = ys.cidx(j+1);
  248. while (ix != ux || iy != uy)
  249. {
  250. octave_idx_type rx = xs.ridx(ix), ry = ys.ridx(ix);
  251. if (rx == ry)
  252. {
  253. retval.xridx(nz) = rx;
  254. retval.xdata(nz) = fcn (xs.data(ix), ys.data(iy));
  255. ix++;
  256. iy++;
  257. }
  258. else if (rx < ry)
  259. {
  260. retval.xridx(nz) = rx;
  261. retval.xdata(nz) = fcn (xs.data(ix), yzero);
  262. ix++;
  263. }
  264. else if (ry < rx)
  265. {
  266. retval.xridx(nz) = ry;
  267. retval.xdata(nz) = fcn (xzero, ys.data(iy));
  268. iy++;
  269. }
  270. nz++;
  271. }
  272. }
  273. retval.maybe_compress ();
  274. return retval;
  275. }
  276. else
  277. return Sparse<U> (binmap<U, T, R, F> (xs.array_value (), ys.array_value (),
  278. fcn, name));
  279. }
  280. // Overloads for function pointers.
  281. // Signature (T, R)
  282. template <class U, class T, class R>
  283. inline Array<U>
  284. binmap (const Array<T>& xa, const Array<R>& ya, U (*fcn) (T, R), const char *name)
  285. { return binmap<U, T, R, U (*) (T, R)> (xa, ya, fcn, name); }
  286. template <class U, class T, class R>
  287. inline Array<U>
  288. binmap (const T& x, const Array<R>& ya, U (*fcn) (T, R))
  289. { return binmap<U, T, R, U (*) (T, R)> (x, ya, fcn); }
  290. template <class U, class T, class R>
  291. inline Array<U>
  292. binmap (const Array<T>& xa, const R& y, U (*fcn) (T, R))
  293. { return binmap<U, T, R, U (*) (T, R)> (xa, y, fcn); }
  294. template <class U, class T, class R>
  295. inline Sparse<U>
  296. binmap (const Sparse<T>& xa, const Sparse<R>& ya, U (*fcn) (T, R), const char *name)
  297. { return binmap<U, T, R, U (*) (T, R)> (xa, ya, fcn, name); }
  298. template <class U, class T, class R>
  299. inline Sparse<U>
  300. binmap (const T& x, const Sparse<R>& ya, U (*fcn) (T, R))
  301. { return binmap<U, T, R, U (*) (T, R)> (x, ya, fcn); }
  302. template <class U, class T, class R>
  303. inline Sparse<U>
  304. binmap (const Sparse<T>& xa, const R& y, U (*fcn) (T, R))
  305. { return binmap<U, T, R, U (*) (T, R)> (xa, y, fcn); }
  306. // Signature (const T&, const R&)
  307. template <class U, class T, class R>
  308. inline Array<U>
  309. binmap (const Array<T>& xa, const Array<R>& ya, U (*fcn) (const T&, const R&), const char *name)
  310. { return binmap<U, T, R, U (*) (const T&, const R&)> (xa, ya, fcn, name); }
  311. template <class U, class T, class R>
  312. inline Array<U>
  313. binmap (const T& x, const Array<R>& ya, U (*fcn) (const T&, const R&))
  314. { return binmap<U, T, R, U (*) (const T&, const R&)> (x, ya, fcn); }
  315. template <class U, class T, class R>
  316. inline Array<U>
  317. binmap (const Array<T>& xa, const R& y, U (*fcn) (const T&, const R&))
  318. { return binmap<U, T, R, U (*) (const T&, const R&)> (xa, y, fcn); }
  319. template <class U, class T, class R>
  320. inline Sparse<U>
  321. binmap (const Sparse<T>& xa, const Sparse<R>& ya, U (*fcn) (const T&, const R&), const char *name)
  322. { return binmap<U, T, R, U (*) (const T&, const R&)> (xa, ya, fcn, name); }
  323. template <class U, class T, class R>
  324. inline Sparse<U>
  325. binmap (const T& x, const Sparse<R>& ya, U (*fcn) (const T&, const R&))
  326. { return binmap<U, T, R, U (*) (const T&, const R&)> (x, ya, fcn); }
  327. template <class U, class T, class R>
  328. inline Sparse<U>
  329. binmap (const Sparse<T>& xa, const R& y, U (*fcn) (const T&, const R&))
  330. { return binmap<U, T, R, U (*) (const T&, const R&)> (xa, y, fcn); }
  331. // Signature (const T&, R)
  332. template <class U, class T, class R>
  333. inline Array<U>
  334. binmap (const Array<T>& xa, const Array<R>& ya, U (*fcn) (const T&, R), const char *name)
  335. { return binmap<U, T, R, U (*) (const T&, R)> (xa, ya, fcn, name); }
  336. template <class U, class T, class R>
  337. inline Array<U>
  338. binmap (const T& x, const Array<R>& ya, U (*fcn) (const T&, R))
  339. { return binmap<U, T, R, U (*) (const T&, R)> (x, ya, fcn); }
  340. template <class U, class T, class R>
  341. inline Array<U>
  342. binmap (const Array<T>& xa, const R& y, U (*fcn) (const T&, R))
  343. { return binmap<U, T, R, U (*) (const T&, R)> (xa, y, fcn); }
  344. template <class U, class T, class R>
  345. inline Sparse<U>
  346. binmap (const Sparse<T>& xa, const Sparse<R>& ya, U (*fcn) (const T&, R), const char *name)
  347. { return binmap<U, T, R, U (*) (const T&, R)> (xa, ya, fcn, name); }
  348. template <class U, class T, class R>
  349. inline Sparse<U>
  350. binmap (const T& x, const Sparse<R>& ya, U (*fcn) (const T&, R))
  351. { return binmap<U, T, R, U (*) (const T&, R)> (x, ya, fcn); }
  352. template <class U, class T, class R>
  353. inline Sparse<U>
  354. binmap (const Sparse<T>& xa, const R& y, U (*fcn) (const T&, R))
  355. { return binmap<U, T, R, U (*) (const T&, R)> (xa, y, fcn); }
  356. // Signature (T, const R&)
  357. template <class U, class T, class R>
  358. inline Array<U>
  359. binmap (const Array<T>& xa, const Array<R>& ya, U (*fcn) (T, const R&), const char *name)
  360. { return binmap<U, T, R, U (*) (T, const R&)> (xa, ya, fcn, name); }
  361. template <class U, class T, class R>
  362. inline Array<U>
  363. binmap (const T& x, const Array<R>& ya, U (*fcn) (T, const R&))
  364. { return binmap<U, T, R, U (*) (T, const R&)> (x, ya, fcn); }
  365. template <class U, class T, class R>
  366. inline Array<U>
  367. binmap (const Array<T>& xa, const R& y, U (*fcn) (T, const R&))
  368. { return binmap<U, T, R, U (*) (T, const R&)> (xa, y, fcn); }
  369. template <class U, class T, class R>
  370. inline Sparse<U>
  371. binmap (const Sparse<T>& xa, const Sparse<R>& ya, U (*fcn) (T, const R&), const char *name)
  372. { return binmap<U, T, R, U (*) (T, const R&)> (xa, ya, fcn, name); }
  373. template <class U, class T, class R>
  374. inline Sparse<U>
  375. binmap (const T& x, const Sparse<R>& ya, U (*fcn) (T, const R&))
  376. { return binmap<U, T, R, U (*) (T, const R&)> (x, ya, fcn); }
  377. template <class U, class T, class R>
  378. inline Sparse<U>
  379. binmap (const Sparse<T>& xa, const R& y, U (*fcn) (T, const R&))
  380. { return binmap<U, T, R, U (*) (T, const R&)> (xa, y, fcn); }
  381. #endif