OTB  10.0.0
Orfeo Toolbox
otbStreamingMatrixTransposeMatrixImageFilter.hxx
Go to the documentation of this file.
1 /*
2  * Copyright (C) 1999-2011 Insight Software Consortium
3  * Copyright (C) 2005-2024 Centre National d'Etudes Spatiales (CNES)
4  *
5  * This file is part of Orfeo Toolbox
6  *
7  * https://www.orfeo-toolbox.org/
8  *
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  *
13  * http://www.apache.org/licenses/LICENSE-2.0
14  *
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  */
21 
22 #ifndef otbStreamingMatrixTransposeMatrixImageFilter_hxx
23 #define otbStreamingMatrixTransposeMatrixImageFilter_hxx
24 
26 
27 #include "itkImageRegionIterator.h"
28 #include "itkNumericTraits.h"
29 #include "itkProgressReporter.h"
30 
31 namespace otb
32 {
33 
34 template <class TInputImage, class TInputImage2>
36 {
37  this->DynamicMultiThreadingOff();
38  this->SetNumberOfRequiredInputs(2);
39 
40  // first output is a copy of the image, DataObject created by
41  // superclass
42  //
43  // allocate the data objects for the outputs which are
44  // just decorators around pixel types
45 
46  typename ImageType::Pointer output1 = static_cast<ImageType*>(this->MakeOutput(0).GetPointer());
47  this->itk::ProcessObject::SetNthOutput(0, output1.GetPointer());
48  typename MatrixObjectType::Pointer output2 = static_cast<MatrixObjectType*>(this->MakeOutput(1).GetPointer());
49  this->itk::ProcessObject::SetNthOutput(1, output2.GetPointer());
50 
51  // false means no pad added
52  m_UsePadFirstInput = false;
53  m_UsePadSecondInput = false;
54 
55  // Number of component initialization
56  m_NumberOfComponents1 = 0;
57  m_NumberOfComponents2 = 0;
58 }
59 
60 template <class TInputImage, class TInputImage2>
62 {
63  switch (output)
64  {
65  case 0:
66  return static_cast<itk::DataObject*>(TInputImage::New().GetPointer());
67  break;
68  case 1:
69  return static_cast<itk::DataObject*>(MatrixObjectType::New().GetPointer());
70  break;
71  default:
72  // might as well make an image
73  return static_cast<itk::DataObject*>(TInputImage::New().GetPointer());
74  break;
75  }
76 }
77 template <class TInputImage, class TInputImage2>
80 {
81  return static_cast<MatrixObjectType*>(this->itk::ProcessObject::GetOutput(1));
82 }
83 
84 template <class TInputImage, class TInputImage2>
87 {
88  return static_cast<const MatrixObjectType*>(this->itk::ProcessObject::GetOutput(1));
89 }
90 
91 template <class TInputImage, class TInputImage2>
93 {
94  Superclass::GenerateInputRequestedRegion();
95 
96  if (this->GetFirstInput() && this->GetSecondInput())
97  {
98  InputImagePointer image = const_cast<typename Superclass::InputImageType*>(this->GetFirstInput());
99  InputImagePointer image2 = const_cast<typename Superclass::InputImageType*>(this->GetSecondInput());
100  image->SetRequestedRegion(this->GetOutput()->GetRequestedRegion());
101  image2->SetRequestedRegion(this->GetOutput()->GetRequestedRegion());
102  }
103 }
104 template <class TInputImage, class TInputImage2>
106 {
107  Superclass::GenerateOutputInformation();
108  if (this->GetFirstInput())
109  {
110  this->GetOutput()->CopyInformation(this->GetFirstInput());
111  this->GetOutput()->SetLargestPossibleRegion(this->GetFirstInput()->GetLargestPossibleRegion());
112  }
113 
114  if (this->GetOutput()->GetRequestedRegion().GetNumberOfPixels() == 0)
115  {
116  this->GetOutput()->SetRequestedRegion(this->GetOutput()->GetLargestPossibleRegion());
117  }
118 }
119 
120 template <class TInputImage, class TInputImage2>
122 {
123  // This is commented to prevent the streaming of the whole image for the first stream strip
124  // It shall not cause any problem because the output image of this filter is not intended to be used.
125  // InputImagePointer image = const_cast< TInputImage * >( this->GetInput() );
126  // this->GraftOutput( image );
127  // Nothing that needs to be allocated for the remaining outputs
128 }
129 
130 template <class TInputImage, class TInputImage2>
132 {
133 
134  TInputImage* inputPtr1 = const_cast<TInputImage*>(this->GetFirstInput());
135  inputPtr1->UpdateOutputInformation();
136  TInputImage2* inputPtr2 = const_cast<TInputImage2*>(this->GetSecondInput());
137  inputPtr2->UpdateOutputInformation();
138 
139  if (this->GetOutput()->GetRequestedRegion().GetNumberOfPixels() == 0)
140  {
141  this->GetOutput()->SetRequestedRegion(this->GetOutput()->GetLargestPossibleRegion());
142  }
143 
144  if (inputPtr1->GetLargestPossibleRegion().GetSize() != inputPtr2->GetLargestPossibleRegion().GetSize())
145  {
146  itkExceptionMacro(<< " Can't multiply the transposed matrix of a " << inputPtr1->GetLargestPossibleRegion().GetSize() << " and a "
147  << inputPtr2->GetLargestPossibleRegion().GetSize() << " matrix ");
148  }
149 
150  m_NumberOfComponents1 = inputPtr1->GetNumberOfComponentsPerPixel();
151  m_NumberOfComponents2 = inputPtr2->GetNumberOfComponentsPerPixel();
152  unsigned int numberOfThreads = this->GetNumberOfWorkUnits();
153 
154  if (m_UsePadFirstInput == true)
155  {
156  m_NumberOfComponents1++;
157  }
158  if (m_UsePadSecondInput == true)
159  {
160  m_NumberOfComponents2++;
161  }
162 
163  MatrixType tempMatrix, initMatrix;
164  tempMatrix.SetSize(m_NumberOfComponents1, m_NumberOfComponents2);
165  tempMatrix.Fill(itk::NumericTraits<RealType>::Zero);
166  m_ThreadSum = ArrayMatrixType(numberOfThreads, tempMatrix);
167 
168  initMatrix.SetSize(m_NumberOfComponents2, m_NumberOfComponents2);
169  initMatrix.Fill(itk::NumericTraits<RealType>::Zero);
170  this->GetResultOutput()->Set(initMatrix);
171 }
172 
173 template <class TInputImage, class TInputImage2>
175 {
176  unsigned int numberOfThreads = this->GetNumberOfWorkUnits();
177  MatrixType resultMatrix;
178  resultMatrix.SetSize(m_NumberOfComponents1, m_NumberOfComponents2);
179  resultMatrix.Fill(itk::NumericTraits<RealType>::Zero);
180 
181  for (unsigned int thread = 0; thread < numberOfThreads; thread++)
182  {
187  for (unsigned int i = 0; i < resultMatrix.Rows(); ++i)
188  {
189  for (unsigned int j = 0; j < resultMatrix.Cols(); ++j)
190  {
191  resultMatrix(i, j) += m_ThreadSum[thread](i, j);
192  }
193  }
194 
195 /********END TODO ******/
196  }
197  this->GetResultOutput()->Set(resultMatrix);
198 }
199 
200 template <class TInputImage, class TInputImage2>
202  itk::ThreadIdType threadId)
203 {
207  InputImagePointer input1Ptr = const_cast<TInputImage*>(this->GetFirstInput());
208  InputImagePointer input2Ptr = const_cast<TInputImage2*>(this->GetSecondInput());
210 
211  // support progress methods/callbacks
212  itk::ProgressReporter progress(this, threadId, outputRegionForThread.GetNumberOfPixels());
213 
214  itk::ImageRegionConstIterator<TInputImage> it1(input1Ptr, outputRegionForThread);
215  itk::ImageRegionConstIterator<TInputImage2> it2(input2Ptr, outputRegionForThread);
216  it1.GoToBegin();
217  it2.GoToBegin();
218 
219  // loop the second image and get one pixel a time
220  while (!it1.IsAtEnd())
221  {
222  PixelType vectorValue1 = it1.Get();
223  PixelType2 vectorValue2 = it2.Get();
224 
225  // Add a first component to vectorValue2 and vectorValue1 filled with ones.
226  if (m_UsePadFirstInput == true)
227  {
228  PixelType vectortemp1(vectorValue1.Size() + 1);
229  vectortemp1[0] = 1;
230  for (unsigned int n = 0; n < vectorValue1.Size(); ++n)
231  {
232  vectortemp1[n + 1] = vectorValue1[n];
233  }
234  vectorValue1.SetSize(vectortemp1.Size());
235  vectorValue1 = vectortemp1;
236  }
237 
238  if (m_UsePadSecondInput == true)
239  {
240  PixelType2 vectortemp2(vectorValue2.Size() + 1);
241  vectortemp2[0] = 1;
242  for (unsigned int m = 0; m < vectorValue2.Size(); m++)
243  {
244  vectortemp2[m + 1] = vectorValue2[m];
245  }
246  vectorValue2.SetSize(vectortemp2.Size());
247  vectorValue2 = vectortemp2;
248  }
249 
250  for (unsigned int i = 0; i < vectorValue1.Size(); ++i)
251  {
252  for (unsigned int j = 0; j < vectorValue2.Size(); ++j)
253  {
254  m_ThreadSum[threadId](i, j) += static_cast<RealType>(vectorValue1[i]) * static_cast<RealType>(vectorValue2[j]);
255  }
256  }
257  ++it1;
258  ++it2;
259  progress.CompletedPixel();
260  }
261 }
262 
263 template <class TInputImage, class TInputImage2>
265 {
266  Superclass::PrintSelf(os, indent);
267 
268  os << indent << "Result: " << this->GetResultOutput()->Get() << std::endl;
269 }
270 
271 } // end namespace otb
272 #endif
DataObjectPointer MakeOutput(DataObjectPointerArraySizeType idx) override
void ThreadedGenerateData(const RegionType &outputRegionForThread, itk::ThreadIdType threadId) override
void PrintSelf(std::ostream &os, itk::Indent indent) const override
itk::ProcessObject::DataObjectPointerArraySizeType DataObjectPointerArraySizeType
The "otb" namespace contains all Orfeo Toolbox (OTB) classes.