Merge branch 'master' of ssh://git.code.sf.net/p/foam-extend/foam-extend-3.2
[foam-extend-3.2.git] / src / foam / matrices / Matrix / Matrix.C
blob9f0cc88f1932b04afc7c26b65112be7ef173beb8
1 /*---------------------------------------------------------------------------*\
2   =========                 |
3   \\      /  F ield         | foam-extend: Open Source CFD
4    \\    /   O peration     | Version:     3.2
5     \\  /    A nd           | Web:         http://www.foam-extend.org
6      \\/     M anipulation  | For copyright notice see file Copyright
7 -------------------------------------------------------------------------------
8 License
9     This file is part of foam-extend.
11     foam-extend is free software: you can redistribute it and/or modify it
12     under the terms of the GNU General Public License as published by the
13     Free Software Foundation, either version 3 of the License, or (at your
14     option) any later version.
16     foam-extend is distributed in the hope that it will be useful, but
17     WITHOUT ANY WARRANTY; without even the implied warranty of
18     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
19     General Public License for more details.
21     You should have received a copy of the GNU General Public License
22     along with foam-extend.  If not, see <http://www.gnu.org/licenses/>.
24 \*---------------------------------------------------------------------------*/
26 #include "Matrix.H"
28 // * * * * * * * * * * * * * * * Static Members  * * * * * * * * * * * * * * //
30 template<class Form, class Type>
31 const Foam::Matrix<Form, Type> Foam::Matrix<Form, Type>::zero;
34 // * * * * * * * * * * * * Private Member Functions  * * * * * * * * * * * * //
36 template<class Form, class Type>
37 void Foam::Matrix<Form, Type>::allocate()
39     if (n_ && m_)
40     {
41         v_ = new Type*[n_];
42         v_[0] = new Type[n_*m_];
44         for (register label i=1; i<n_; i++)
45         {
46             v_[i] = v_[i-1] + m_;
47         }
48     }
52 // * * * * * * * * * * * * * * * Destructor  * * * * * * * * * * * * * * * * //
54 template<class Form, class Type>
55 Foam::Matrix<Form, Type>::~Matrix()
57     if (v_)
58     {
59         delete[] (v_[0]);
60         delete[] v_;
61     }
65 // * * * * * * * * * * * * * * * Member Functions  * * * * * * * * * * * * * //
67 template<class Form, class Type>
68 Foam::Matrix<Form, Type>::Matrix(const label n, const label m)
70     v_(NULL),
71     n_(n),
72     m_(m)
74     if (n_ < 0 || m_ < 0)
75     {
76         FatalErrorIn
77         (
78             "Matrix<Form, Type>::Matrix(const label n, const label m)"
79         )   << "bad n, m " << n_ << ", " << m_
80             << abort(FatalError);
81     }
83     allocate();
87 template<class Form, class Type>
88 Foam::Matrix<Form, Type>::Matrix(const label n, const label m, const Type& a)
90     v_(NULL),
91     n_(n),
92     m_(m)
94     if (n_ < 0 || m_ < 0)
95     {
96         FatalErrorIn
97         (
98             "Matrix<Form, Type>::Matrix"
99             "(const label n, const label m, const T&)"
100         )   << "bad n, m " << n_ << ", " << m_
101             << abort(FatalError);
102     }
104     allocate();
106     if (v_)
107     {
108         Type* v = v_[0];
110         label nm = n_*m_;
112         for (register label i=0; i<nm; i++)
113         {
114             v[i] = a;
115         }
116     }
120 template<class Form, class Type>
121 Foam::Matrix<Form, Type>::Matrix(const Matrix<Form, Type>& a)
123     v_(NULL),
124     n_(a.n_),
125     m_(a.m_)
127     if (a.v_)
128     {
129         allocate();
130         Type* v = v_[0];
131         const Type* av = a.v_[0];
133         label nm = n_*m_;
134         for (register label i=0; i<nm; i++)
135         {
136             v[i] = av[i];
137         }
138     }
142 template<class Form, class Type>
143 void Foam::Matrix<Form, Type>::clear()
145     if (v_)
146     {
147         delete[] (v_[0]);
148         delete[] v_;
149     }
150     n_ = 0;
151     m_ = 0;
152     v_ = NULL;
156 template<class Form, class Type>
157 void Foam::Matrix<Form, Type>::transfer(Matrix<Form, Type>& a)
159     clear();
161     n_ = a.n_;
162     a.n_ = 0;
164     m_ = a.m_;
165     a.m_ = 0;
167     v_ = a.v_;
168     a.v_ = NULL;
172 template<class Form, class Type>
173 Form Foam::Matrix<Form, Type>::T() const
175     const Matrix<Form, Type>& A = *this;
176     Form At(m(), n());
178     for (register label i=0; i<n(); i++)
179     {
180         for (register label j=0; j<m(); j++)
181         {
182             At[j][i] = A[i][j];
183         }
184     }
186     return At;
190 // * * * * * * * * * * * * * * * Member Operators  * * * * * * * * * * * * * //
192 template<class Form, class Type>
193 void Foam::Matrix<Form, Type>::operator=(const Type& t)
195     if (v_)
196     {
197         Type* v = v_[0];
199         label nm = n_*m_;
200         for (register label i=0; i<nm; i++)
201         {
202             v[i] = t;
203         }
204     }
208 // Assignment operator. Takes linear time.
209 template<class Form, class Type>
210 void Foam::Matrix<Form, Type>::operator=(const Matrix<Form, Type>& a)
212     if (this == &a)
213     {
214         FatalErrorIn
215         (
216             "Matrix<Form, Type>::operator=(const Matrix<Form, Type>&)"
217         )   << "attempted assignment to self"
218             << abort(FatalError);
219     }
221     if (n_ != a.n_ || m_ != a.m_)
222     {
223         clear();
224         n_ = a.n_;
225         m_ = a.m_;
226         allocate();
227     }
229     if (v_)
230     {
231         Type* v = v_[0];
232         const Type* av = a.v_[0];
234         label nm = n_*m_;
235         for (register label i=0; i<nm; i++)
236         {
237             v[i] = av[i];
238         }
239     }
243 // * * * * * * * * * * * * * * * Global Functions  * * * * * * * * * * * * * //
245 template<class Form, class Type>
246 const Type& Foam::max(const Matrix<Form, Type>& a)
248     label nm = a.n()*a.m();
250     if (nm)
251     {
252         label curMaxI = 0;
253         const Type* v = a[0];
255         for (register label i=1; i<nm; i++)
256         {
257             if (v[i] > v[curMaxI])
258             {
259                 curMaxI = i;
260             }
261         }
263         return v[curMaxI];
264     }
265     else
266     {
267         FatalErrorIn("max(const Matrix<Form, Type>&)")
268             << "matrix is empty"
269             << abort(FatalError);
271         // Return in error to keep compiler happy
272         return a[0][0];
273     }
277 template<class Form, class Type>
278 const Type& Foam::min(const Matrix<Form, Type>& a)
280     label nm = a.n()*a.m();
282     if (nm)
283     {
284         label curMinI = 0;
285         const Type* v = a[0];
287         for (register label i=1; i<nm; i++)
288         {
289             if (v[i] < v[curMinI])
290             {
291                 curMinI = i;
292             }
293         }
295         return v[curMinI];
296     }
297     else
298     {
299         FatalErrorIn("min(const Matrix<Form, Type>&)")
300             << "matrix is empty"
301             << abort(FatalError);
303         // Return in error to keep compiler happy
304         return a[0][0];
305     }
309 // * * * * * * * * * * * * * * * Global Operators  * * * * * * * * * * * * * //
311 template<class Form, class Type>
312 Form Foam::operator-(const Matrix<Form, Type>& a)
314     Form na(a.n(), a.m());
316     if (a.n() && a.m())
317     {
318         Type* nav = na[0];
319         const Type* av = a[0];
321         label nm = a.n()*a.m();
322         for (register label i=0; i<nm; i++)
323         {
324             nav[i] = -av[i];
325         }
326     }
328     return na;
332 template<class Form, class Type>
333 Form Foam::operator+(const Matrix<Form, Type>& a, const Matrix<Form, Type>& b)
335     if (a.n() != b.n())
336     {
337         FatalErrorIn
338         (
339             "Matrix<Form, Type>::operator+(const Matrix<Form, Type>&, "
340             "const Matrix<Form, Type>&)"
341         )   << "attempted add matrices with different number of rows: "
342             << a.n() << ", " << b.n()
343             << abort(FatalError);
344     }
346     if (a.m() != b.m())
347     {
348         FatalErrorIn
349         (
350             "Matrix<Form, Type>::operator+(const Matrix<Form, Type>&, "
351             "const Matrix<Form, Type>&)"
352         )   << "attempted add matrices with different number of columns: "
353             << a.m() << ", " << b.m()
354             << abort(FatalError);
355     }
357     Form ab(a.n(), a.m());
359     Type* abv = ab[0];
360     const Type* av = a[0];
361     const Type* bv = b[0];
363     label nm = a.n()*a.m();
364     for (register label i=0; i<nm; i++)
365     {
366         abv[i] = av[i] + bv[i];
367     }
369     return ab;
373 template<class Form, class Type>
374 Form Foam::operator-(const Matrix<Form, Type>& a, const Matrix<Form, Type>& b)
376     if (a.n() != b.n())
377     {
378         FatalErrorIn
379         (
380             "Matrix<Form, Type>::operator-(const Matrix<Form, Type>&, "
381             "const Matrix<Form, Type>&)"
382         )   << "attempted add matrices with different number of rows: "
383             << a.n() << ", " << b.n()
384             << abort(FatalError);
385     }
387     if (a.m() != b.m())
388     {
389         FatalErrorIn
390         (
391             "Matrix<Form, Type>::operator-(const Matrix<Form, Type>&, "
392             "const Matrix<Form, Type>&)"
393         )   << "attempted add matrices with different number of columns: "
394             << a.m() << ", " << b.m()
395             << abort(FatalError);
396     }
398     Form ab(a.n(), a.m());
400     Type* abv = ab[0];
401     const Type* av = a[0];
402     const Type* bv = b[0];
404     label nm = a.n()*a.m();
405     for (register label i=0; i<nm; i++)
406     {
407         abv[i] = av[i] - bv[i];
408     }
410     return ab;
414 template<class Form, class Type>
415 Form Foam::operator*(const scalar s, const Matrix<Form, Type>& a)
417     Form sa(a.n(), a.m());
419     if (a.n() && a.m())
420     {
421         Type* sav = sa[0];
422         const Type* av = a[0];
424         label nm = a.n()*a.m();
425         for (register label i=0; i<nm; i++)
426         {
427             sav[i] = s*av[i];
428         }
429     }
431     return sa;
435 // * * * * * * * * * * * * * * * *  IOStream operators * * * * * * * * * * * //
437 #include "MatrixIO.C"
439 // ************************************************************************* //