op_repelem_meat.hpp 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. // Copyright 2008-2016 Conrad Sanderson (http://conradsanderson.id.au)
  2. // Copyright 2008-2016 National ICT Australia (NICTA)
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License");
  5. // you may not use this file except in compliance with the License.
  6. // You may obtain a copy of the License at
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. // ------------------------------------------------------------------------
  15. //! \addtogroup op_repelem
  16. //! @{
  17. template<typename obj>
  18. inline
  19. void
  20. op_repelem::apply_noalias(Mat<typename obj::elem_type>& out, const obj& X, const uword copies_per_row, const uword copies_per_col)
  21. {
  22. arma_extra_debug_sigprint();
  23. typedef typename obj::elem_type eT;
  24. const uword X_n_rows = obj::is_row ? uword(1) : X.n_rows;
  25. const uword X_n_cols = obj::is_col ? uword(1) : X.n_cols;
  26. out.set_size(X_n_rows * copies_per_row, X_n_cols * copies_per_col);
  27. if(out.n_elem == 0) { return; }
  28. for(uword col=0; col < X_n_cols; ++col)
  29. {
  30. const uword out_col_offset = col * copies_per_col;
  31. eT* out_colptr_first = out.colptr(out_col_offset);
  32. for(uword row=0; row < X_n_rows; ++row)
  33. {
  34. const uword out_row_offset = row * copies_per_row;
  35. const eT copy_value = X.at(row, col);
  36. for(uword row_copy=0; row_copy < copies_per_row; ++row_copy)
  37. {
  38. out_colptr_first[out_row_offset + row_copy] = copy_value;
  39. }
  40. if(copies_per_col != 1)
  41. {
  42. for(uword col_copy=1; col_copy < copies_per_col; ++col_copy)
  43. {
  44. eT* out_colptr = out.colptr(out_col_offset + col_copy);
  45. arrayops::copy(&out_colptr[out_row_offset], &out_colptr_first[out_row_offset], copies_per_row);
  46. }
  47. }
  48. }
  49. }
  50. }
  51. template<typename T1>
  52. inline
  53. void
  54. op_repelem::apply(Mat<typename T1::elem_type>& out, const Op<T1,op_repelem>& in)
  55. {
  56. arma_extra_debug_sigprint();
  57. typedef typename T1::elem_type eT;
  58. const uword copies_per_row = in.aux_uword_a;
  59. const uword copies_per_col = in.aux_uword_b;
  60. const quasi_unwrap<T1> U(in.m);
  61. if(U.is_alias(out))
  62. {
  63. Mat<eT> tmp;
  64. op_repelem::apply_noalias(tmp, U.M, copies_per_row, copies_per_col);
  65. out.steal_mem(tmp);
  66. }
  67. else
  68. {
  69. op_repelem::apply_noalias(out, U.M, copies_per_row, copies_per_col);
  70. }
  71. }
  72. //! @}