LORENE
mg3d.C
1 /*
2  * Methods of class Mg3d
3  *
4  */
5 
6 /*
7  * Copyright (c) 1999-2000 Jean-Alain Marck
8  * Copyright (c) 1999-2001 Eric Gourgoulhon
9  *
10  * This file is part of LORENE.
11  *
12  * LORENE is free software; you can redistribute it and/or modify
13  * it under the terms of the GNU General Public License as published by
14  * the Free Software Foundation; either version 2 of the License, or
15  * (at your option) any later version.
16  *
17  * LORENE is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20  * GNU General Public License for more details.
21  *
22  * You should have received a copy of the GNU General Public License
23  * along with LORENE; if not, write to the Free Software
24  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25  *
26  */
27 
28 
29 
30 
31 /*
32  * $Id: mg3d.C,v 1.23 2025/03/26 10:12:20 j_novak Exp $
33  * $Log: mg3d.C,v $
34  * Revision 1.23 2025/03/26 10:12:20 j_novak
35  * New derived grid containing 50% more points in every direction, additional tool for de-aliasing.
36  *
37  * Revision 1.22 2023/05/24 09:48:50 g_servignat
38  * Added plus_half grid in angular direction for dealiasing
39  *
40  * Revision 1.21 2020/01/27 11:00:19 j_novak
41  * New include <stdexcept> to be compatible with older versions of g++
42  *
43  * Revision 1.20 2018/12/05 15:03:20 j_novak
44  * New Mg3d constructor from a formatted file.
45  *
46  * Revision 1.19 2016/12/05 16:17:59 j_novak
47  * Suppression of some global variables (file names, loch, ...) to prevent redefinitions
48  *
49  * Revision 1.18 2014/10/13 08:53:07 j_novak
50  * Lorene classes and functions now belong to the namespace Lorene.
51  *
52  * Revision 1.17 2014/10/06 15:13:14 j_novak
53  * Modified #include directives to use c++ syntax.
54  *
55  * Revision 1.16 2013/06/05 15:00:26 j_novak
56  * Suppression of all classes derived from Grille3d. Now Grille3d is no
57  * longer an abstract class. r-samplings are only one of RARE, FIN or
58  * UNSURR (FINJAC has been removed). Instead, Mg3d possesses a new member
59  * colloc_r[nzone] defining the type of collocation points (spectral
60  * bases) in each domain.
61  *
62  * Revision 1.15 2012/01/17 10:37:42 j_penner
63  * added a constructor that treats all domains as type FIN
64  *
65  * Revision 1.14 2008/08/19 06:42:00 j_novak
66  * Minor modifications to avoid warnings with gcc 4.3. Most of them concern
67  * cast-type operations, and constant strings that must be defined as const char*
68  *
69  * Revision 1.13 2007/12/11 15:28:15 jl_cornou
70  * Jacobi(0,2) polynomials partially implemented
71  *
72  * Revision 1.12 2006/05/17 13:17:03 j_novak
73  * New member g_angu_1dom, the one-domain angular grid associated with the
74  * current grid.
75  *
76  * Revision 1.11 2005/10/07 08:47:21 j_novak
77  * Addition of the pointer g_non_axi on a grid, with at least 5 points in the
78  * theta direction and 4 in the phi one (for tensor rotations).
79  *
80  * Revision 1.10 2004/07/06 13:36:29 j_novak
81  * Added methods for desaliased product (operator |) only in r direction.
82  *
83  * Revision 1.9 2003/10/27 16:21:54 e_gourgoulhon
84  * Treated the special case nz=1 in the simplified constructor.
85  *
86  * Revision 1.8 2003/06/20 14:50:15 f_limousin
87  * Add the operator==
88  *
89  * Revision 1.7 2003/06/18 08:45:27 j_novak
90  * In class Mg3d: added the member get_radial, returning only a radial grid
91  * For dAlembert solver: the way the coefficients of the operator are defined has been changed.
92  *
93  * Revision 1.6 2002/10/16 14:36:42 j_novak
94  * Reorganization of #include instructions of standard C++, in order to
95  * use experimental version 3 of gcc.
96  *
97  * Revision 1.5 2002/05/07 07:36:03 e_gourgoulhon
98  * Compatibilty with xlC compiler on IBM SP2:
99  * suppressed the parentheses around argument of instruction new:
100  * e.g. t = new (Tbl *[nzone]) --> t = new Tbl*[nzone]
101  *
102  * Revision 1.4 2001/12/12 09:23:46 e_gourgoulhon
103  * Parameter compact added to the simplified constructor of class Mg3d
104  *
105  * Revision 1.3 2001/12/11 06:48:30 e_gourgoulhon
106  * Addition of the simplified constructor
107  *
108  * Revision 1.2 2001/12/04 21:27:54 e_gourgoulhon
109  *
110  * All writing/reading to a binary file are now performed according to
111  * the big endian convention, whatever the system is big endian or
112  * small endian, thanks to the functions fwrite_be and fread_be
113  *
114  * Revision 1.1.1.1 2001/11/20 15:19:27 e_gourgoulhon
115  * LORENE
116  *
117  * Revision 2.10 2001/05/26 14:50:46 eric
118  * *** empty log message ***
119  *
120  * Revision 2.9 2001/05/26 13:25:59 eric
121  * Ajout du membre g_twice (grille double pour le desaliasing)
122  * Modif de la declaration de g_angu (pointeur mutable)
123  * g_twice et g_angu ne sont calcules que si necessaire (cad si
124  * on appelle la fonction get_twice() ou get_angu()).
125  *
126  * Revision 2.8 2000/03/22 13:38:51 eric
127  * Remplacement des iendl par endl dans <<
128  *
129  * Revision 2.7 1999/10/12 15:04:29 eric
130  * *** empty log message ***
131  *
132  * Revision 2.6 1999/10/12 15:03:30 eric
133  * *** empty log message ***
134  *
135  * Revision 2.5 1999/09/30 14:58:16 eric
136  * Operator!= declare const
137  *
138  * Revision 2.4 1999/09/30 14:12:04 eric
139  * sauve declaree const.
140  *
141  * Revision 2.3 1999/09/30 12:52:52 eric
142  * Depoussierage.
143  * Documentation.
144  *
145  * Revision 2.2 1999/03/01 14:35:21 eric
146  * Modif affichage (operator<<)
147  *
148  *
149  * $Header: /cvsroot/Lorene/C++/Source/Mg3d/mg3d.C,v 1.23 2025/03/26 10:12:20 j_novak Exp $
150  *
151  */
152 // C++ Headers
153 #include <stdexcept>
154 
155 // C Headers
156 #include <cstdlib>
157 #include <cassert>
158 
159 // Lorene headers
160 #include "grilles.h"
161 #include "type_parite.h"
162 #include "utilitaires.h"
163 
164  //--------------//
165  // Multi-grille //
166  //--------------//
167 
168 
169 //=============================================================================
170 // General constructor
171 //=============================================================================
172 
173 
174 namespace Lorene {
175 Mg3d::Mg3d(int nz, int nbr[], int typr[], int nbt[], int typt, int nbp[],
176  int typp, int* base_r)
177  : nzone(nz), type_t(typt), type_p(typp)
178 {
179 
180  // Type d'echantillonnage dans chaque zone
181  type_r = new int[nz];
182  colloc_r = new int[nz] ;
183  bool cheb = (base_r == 0x0) ;
184  for (int i=0 ; i<nz ; i++) {
185  type_r[i] = typr[i];
186  colloc_r[i] = cheb ? BASE_CHEB : base_r[i] ;
187  }
188 
189  // Nombre de points
190  nr = new int[nz];
191  nt = new int[nz];
192  np = new int[nz];
193  for (int i=0 ; i<nz ; i++) {
194  nr[i] = nbr[i] ;
195  nt[i] = nbt[i] ;
196  np[i] = nbp[i] ;
197  }
198 
199  // Les grilles
200  // -----------
201  g = new Grille3d*[nz] ;
202 
203  for (int i=0; i<nz; i++) {
204 
205  g[i] = new Grille3d(nr[i], nt[i], np[i], type_r[i], type_t, type_p,
206  colloc_r[i]) ;
207  } // fin de la boucle sur les zones
208 
209  // Pointers on derived grids initiated to 0x0:
210  // -------------------------------------------
211 
212  set_deriv_0x0() ;
213 
214 
215 }
216 
217 //=============================================================================
218 // Simplified constructor
219 //=============================================================================
220 
221 Mg3d::Mg3d(int nz, int nbr, int nbt, int nbp, int typt, int typp,
222  bool compact, bool leg)
223  : nzone(nz),
224  type_t(typt),
225  type_p(typp) {
226 
227  // Type of r sampling in each domain:
228  type_r = new int[nz];
229  colloc_r = new int[nz];
230  type_r[0] = RARE ;
231  colloc_r[0] = leg ? BASE_LEG : BASE_CHEB ;
232  for (int l=1; l<nz-1; l++) {
233  type_r[l] = FIN ;
234  colloc_r[l] = leg ? BASE_LEG : BASE_CHEB ;
235  }
236  if (nz > 1) {
237  if (compact) {
238  type_r[nz-1] = UNSURR ;
239  colloc_r[nz-1] = BASE_CHEB ;
240  }
241  else {
242  type_r[nz-1] = FIN ;
243  colloc_r[nz-1] = leg ? BASE_LEG : BASE_CHEB ;
244  }
245  }
246 
247  // Same number of points in all domains:
248  nr = new int[nz];
249  nt = new int[nz];
250  np = new int[nz];
251  for (int l=0 ; l<nz ; l++) {
252  nr[l] = nbr ;
253  nt[l] = nbt ;
254  np[l] = nbp ;
255  }
256 
257  // Les grilles
258  // -----------
259  g = new Grille3d*[nz] ;
260 
261  for (int i=0; i<nz; i++) {
262 
263  g[i] = new Grille3d(nr[i], nt[i], np[i], type_r[i], type_t, type_p,
264  colloc_r[i]) ;
265  } // fin de la boucle sur les zones
266 
267  // Pointers on derived grids initiated to 0x0:
268  // -------------------------------------------
269 
270  set_deriv_0x0() ;
271 
272 }
273 
274 //=============================================================================
275 // Simplified shell constructor
276 // Note: This does not handle the nucleus or the CED!
277 //=============================================================================
278 
279 Mg3d::Mg3d(int nz, int nbr, int nbt, int nbp, int typt, int typp)
280  : nzone(nz),
281  type_t(typt),
282  type_p(typp) {
283 
284  // Type of r sampling in each domain:
285  type_r = new int[nz];
286  colloc_r = new int[nz] ;
287  for (int l=0; l<nz; l++) {
288  type_r[l] = FIN ;
289  colloc_r[l] = BASE_CHEB ;
290  }
291 
292  // Same number of points in all domains:
293  nr = new int[nz];
294  nt = new int[nz];
295  np = new int[nz];
296  for (int l=0 ; l<nz ; l++) {
297  nr[l] = nbr ;
298  nt[l] = nbt ;
299  np[l] = nbp ;
300  }
301 
302  // Les grilles
303  // -----------
304  g = new Grille3d*[nz] ;
305 
306  for (int i=0; i<nz; i++) {
307 
308  g[i] = new Grille3d(nr[i], nt[i], np[i], type_r[i], type_t, type_p,
309  colloc_r[i]) ;
310 
311  } // fin de la boucle sur les zones
312 
313  // Pointers on derived grids initiated to 0x0:
314  // -------------------------------------------
315 
316  set_deriv_0x0() ;
317 
318 }
319 
320 //=============================================================================
321 // Constructors from a file
322 //=============================================================================
323 
324 
325  // From a formatted file...
326  //=========================
327  Mg3d::Mg3d(const string& filename) {
328 
329  // Opening of the file
330  //-----------------------------------------------------------------------
331  ifstream parfile(filename.c_str()) ;
332  if (!parfile) {
333  string message = "Unable to open file " ;
334  message += filename ;
335  throw ios_base::failure(message);
336  };
337 
338  string tmp_str = "Definition of the Mg3d" ;
339  if (!search_file(parfile, tmp_str)) {
340  string mesg = "No data found to contruct an object Mg3d in " ;
341  mesg += filename ;
342  throw invalid_argument(mesg) ;
343  }
344  // parfile >> tmp_str ;
345  //cout << tmp_str ;
346  parfile.ignore(1000, '\n') ;
347 
348  // Number of domains
349  //---------------------
350  parfile >> nzone ; parfile.ignore(1000, '\n') ; // Number of domains
351  parfile >> tmp_str ;
352 
353  // Theta & phi symmetries + number of grid points
354  //-----------------------------------------------
355  if ( tmp_str.compare("SYM") == 0) {
356  type_t = SYM ;
357  }
358  else if ( tmp_str.compare("NONSYM") == 0)
359  {
360  type_t = NONSYM ;
361  }
362  else
363  throw
364  invalid_argument("Mg3d::Mg3d(string): incorrect value for theta symmetry") ;
365  parfile.ignore(1000, '\n') ;
366 
367  parfile >> tmp_str ;
368  if ( tmp_str.compare("SYM") == 0)
369  {
370  type_p = SYM ;
371  }
372  else if ( tmp_str.compare("NONSYM") == 0)
373  {
374  type_p = NONSYM ;
375  }
376  else
377  throw
378  invalid_argument("Mg3d::Mg3d(string): incorrect value for phi symmetry") ;
379  parfile.ignore(1000, '\n') ;
380 
381  int nbp ; parfile >> nbp ; parfile.ignore(1000, '\n') ;
382  int nbt ; parfile >> nbt ; parfile.ignore(1000, '\n') ;
383 
384  // Radial number of points and sampling type
385  //---------------------------------------------
386  nr = new int[nzone] ;
387  nt = new int[nzone] ;
388  np = new int[nzone] ;
389  type_r = new int[nzone] ;
390 
391  for (int i=0; i<nzone; i++) {
392  parfile >> nr[i] >> tmp_str ;
393  assert (nr[i] > 0) ;
394  if ( tmp_str.compare("nucleus") == 0)
395  {
396  type_r[i] = RARE ;
397  }
398  else if ( tmp_str.compare("shell") == 0)
399  {
400  type_r[i] = FIN ;
401  }
402  else if ( tmp_str.compare("ced") == 0)
403  {
404  type_r[i] = UNSURR ;
405  }
406  else
407  throw
408  invalid_argument("Mg3d::Mg3d(string): incorrect value for the type of domain") ;
409  parfile.ignore(1000, '\n') ;
410  nt[i] = nbt ;
411  np[i] = nbp ;
412  } // end of loop on domains
413 
414  colloc_r = new int[nzone] ;
415  bool legendre ;
416  parfile >> legendre ;
417  for (int i=0; i<nzone; i++)
418  colloc_r[i] = (legendre ? BASE_LEG : BASE_CHEB) ;
419 
420 
421  // Grids
422  // -----------
423  g = new Grille3d*[nzone] ;
424 
425  for (int i=0; i<nzone; i++) {
426 
427  g[i] = new Grille3d(nr[i], nt[i], np[i], type_r[i], type_t, type_p,
428  colloc_r[i]) ;
429 
430  } // end of loop on domains
431 
432  // Pointers on derived grids initiated to 0x0:
433  // -------------------------------------------
434 
435  set_deriv_0x0() ;
436 
437  }
438 
439 
440 /*
441  * Construction a partir d'un fichier.
442  * Cette facon de faire est abominable. Cependant je ne vois pas comment
443  * faire autrement... j.a.
444  */
445 Mg3d::Mg3d(FILE* fd, bool read_base)
446 {
447  // Lecture sur le fichier
448  fread_be(&nzone, sizeof(int), 1, fd) ; // nzone
449  nr = new int[nzone] ;
450  fread_be(nr, sizeof(int), nzone, fd) ; // nr
451  nt = new int[nzone] ;
452  fread_be(nt, sizeof(int), nzone, fd) ; // nt
453  np = new int[nzone] ;
454  fread_be(np, sizeof(int), nzone, fd) ; // np
455  type_r = new int[nzone] ;
456  fread_be(type_r, sizeof(int), nzone, fd) ; // type_r
457  fread_be(&type_t, sizeof(int), 1, fd) ; // type_t
458  fread_be(&type_p, sizeof(int), 1, fd) ; // type_p
459  colloc_r = new int[nzone] ;
460  if (read_base)
461  fread_be(colloc_r, sizeof(int), nzone, fd) ; // colloc_r
462 
463  // Les grilles
464  // -----------
465 
466  g = new Grille3d*[nzone] ;
467  for (int i=0; i<nzone; i++) {
468  if (!read_base) colloc_r[i] = BASE_CHEB ;
469  g[i] = new Grille3d(nr[i], nt[i], np[i], type_r[i], type_t, type_p,
470  colloc_r[i]) ;
471 
472  } // fin de la boucle sur les zones
473 
474  // Pointers on derived grids initiated to 0x0:
475  // -------------------------------------------
476 
477  set_deriv_0x0() ;
478 
479 }
480 
481 // Destructeur
482 // -----------
484 
485  del_deriv() ; // Deletes the derived quantities
486 
487  delete [] nr ;
488  delete [] nt ;
489  delete [] np ;
490  delete [] type_r ;
491  delete [] colloc_r ;
492  for (int i=0 ; i<nzone ; i++) {
493  delete g[i] ;
494  }
495  delete [] g ;
496 
497 }
498 
499 //==================================================================
500 // Write in a file
501 //==================================================================
502 
503 void Mg3d::sauve(FILE* fd, bool save_base) const {
504  fwrite_be(&nzone, sizeof(int), 1, fd) ; // nzone
505  fwrite_be(nr, sizeof(int), nzone, fd) ; // nr
506  fwrite_be(nt, sizeof(int), nzone, fd) ; // nt
507  fwrite_be(np, sizeof(int), nzone, fd) ; // np
508  fwrite_be(type_r, sizeof(int), nzone, fd) ; // type_r
509  fwrite_be(&type_t, sizeof(int), 1, fd) ; // type_t
510  fwrite_be(&type_p, sizeof(int), 1, fd) ; // type_p
511  if (save_base) {
512  fwrite_be(colloc_r, sizeof(int), nzone, fd) ; // colloc_r
513  }
514  else
515  for (int l=0; l<nzone; l++)
516  if (colloc_r[l] != BASE_CHEB) {
517  cout << "Mg3d::sauve(FILE*, bool) : " << endl ;
518  cout << "The multi-grid is not with Chebyshev basis!!" << endl ;
519  cout << "Consider setting the 'save_base' flaf to 'true'!!"
520  << endl ;
521  arrete() ;
522 
523  }
524 }
525 
526  //--------------------------//
527  // Surcharge des operateurs //
528  //--------------------------//
529 
530 // Operateur <<
531 ostream& operator<<(ostream& o, const Mg3d& g) {
532  const char* tr[3] ;
533  tr[FIN] = "FIN" ; tr[RARE] = "RARE" ; tr[UNSURR] = "UNSURR" ;
534  const char* tang[2] ;
535  tang[NONSYM] = "NONSYM" ; tang[SYM] = "SYM" ;
536  const char* tbase[3] ;
537  tbase[BASE_CHEB] = "Chebyshev" ; tbase[BASE_LEG] = "Legendre" ;
538  tbase[BASE_JAC02] = "Jacobi(0,2)" ;
539  o << "Number of domains: " << g.nzone << endl ;
540  for (int i=0 ; i< g.nzone ; i++) {
541  o << " Domain #" << i << ": "
542  << "nr = " << g.nr[i] << ", " << tr[g.type_r[i]] << "; "
543  << "nt = " << g.nt[i] << ", " << tang[g.type_t] << "; "
544  << "np = " << g.np[i] << ", " << tang[g.type_p] << "; "
545  << "Collocation points type : " << tbase[g.colloc_r[i]] << endl ;
546  }
547  o << endl ;
548  return o ;
549 }
550 
551 // Operateur !=
552 bool Mg3d::operator!=(const Mg3d & titi) const {
553 
554  if (nzone != titi.nzone) return true ; // C'est vrai que c'est faux...
555 
556  for (int i=0 ; i<nzone ; i++) {
557  if (nr[i] != titi.nr[i]) return true ;
558  if (nt[i] != titi.nt[i]) return true ;
559  if (np[i] != titi.np[i]) return true ;
560 
561  if (type_r[i] != titi.type_r[i]) return true ;
562  if (colloc_r[i] != titi.colloc_r[i]) return true ;
563  }
564 
565  if (type_t != titi.type_t) return true ;
566  if (type_p != titi.type_p) return true ;
567 
568  // C'est faux que c'est vrai...
569  return false ;
570 }
571 
572 
573  //----------------------------------//
574  // Management of derived quantities //
575  //----------------------------------//
576 
577 void Mg3d::del_deriv() const {
578 
579  if (g_angu != 0x0) delete g_angu ;
580  if (g_angu_1dom != 0x0) delete g_angu_1dom ;
581  if (g_radial != 0x0) delete g_radial ;
582  if (g_three_halves != 0x0) delete g_three_halves ;
583  if (g_twice != 0x0) delete g_twice ;
584  if (g_plus_half != 0x0) delete g_plus_half ;
585  if (g_plus_half_angu != 0x0) delete g_plus_half_angu ;
586  if (g_non_axi != 0x0) delete g_non_axi ;
587 
588  set_deriv_0x0() ;
589 
590 }
591 
592 void Mg3d::set_deriv_0x0() const {
593 
594  g_angu = 0x0 ;
595  g_angu_1dom = 0x0 ;
596  g_radial = 0x0 ;
597  g_three_halves = 0x0 ;
598  g_twice = 0x0 ;
599  g_plus_half = 0x0 ;
600  g_plus_half_angu = 0x0 ;
601  g_non_axi = 0x0 ;
602 }
603 
604 
605  //--------------//
606  // Angular grid //
607  //--------------//
608 
609 const Mg3d* Mg3d::get_angu() const {
610 
611  if (g_angu == 0x0) { // The construction is required
612 
613  int* nbr_angu = new int[nzone] ;
614  for (int i=0 ; i<nzone ; i++) {
615  nbr_angu[i] = 1 ;
616  }
617  g_angu = new Mg3d(nzone, nbr_angu, type_r, nt, type_t, np, type_p,
618  colloc_r) ;
619  delete [] nbr_angu ;
620  }
621 
622  return g_angu ;
623 
624 }
625 
626  //-----------------------------//
627  // Angular grid for one domain //
628  //-----------------------------//
629 
630 const Mg3d* Mg3d::get_angu_1dom() const {
631 
632  if (g_angu_1dom == 0x0) { // The construction is required
633  int* nbr_angu = new int(1) ;
634  int* nbt_angu = new int(nt[0]) ;
635  int* nbp_angu = new int(np[0]) ;
636  int* type_r_angu = new int(FIN) ;
637 
638  g_angu_1dom = new Mg3d(1, nbr_angu, type_r_angu, nbt_angu, type_t,
639  nbp_angu, type_p) ;
640  delete nbr_angu ;
641  delete nbt_angu ;
642  delete nbp_angu ;
643  delete type_r_angu ;
644  }
645 
646  return g_angu_1dom ;
647 
648 }
649 
650  //--------------//
651  // Radial grid //
652  //--------------//
653 
654 const Mg3d* Mg3d::get_radial() const {
655 
656  if (g_radial == 0x0) { // The construction is required
657 
658  int* nbr_radial = new int[nzone] ;
659  for (int i=0 ; i<nzone ; i++) {
660  nbr_radial[i] = 1 ;
661  }
662  g_radial = new Mg3d(nzone, nr, type_r, nbr_radial, SYM, nbr_radial,
663  SYM, colloc_r) ;
664  delete [] nbr_radial ;
665  }
666 
667  return g_radial ;
668 
669 }
670 
671 const Mg3d* Mg3d::get_three_halves() const {
672 
673  if (g_three_halves == 0x0) { // There are things to do...
674 
675  int* nbr = new int[nzone] ;
676  int* nbt = new int[nzone] ;
677  int* nbp = new int[nzone] ;
678 
679  for (int l=0; l<nzone; l++) {
680  if (nr[l] == 1) {
681  nbr[l] = 1 ;
682  }
683  else {
684  nbr[l] = (3*(nr[l]-1)) / 2 + 1 ;
685  if (nbr[l] % 2 == 0) nbr[l]++ ;
686  }
687 
688  if (nt[l] == 1) {
689  nbt[l] = 1 ;
690  }
691  else {
692  nbt[l] = (3*(nt[l]-1)) / 2 + 1 ;
693  if (nbt[l] % 2 == 0) nbt[l]++ ;
694  }
695 
696  if (np[l] == 1) {
697  nbp[l] = 1 ;
698  }
699  else {
700  nbp[l] = 3*np[l]/2 ;
701  if (nbp[l] % 2 == 1) nbp[l]++ ;
702  }
703  }
704 
706  = new Mg3d(nzone, nbr, type_r, nbt, type_t, nbp, type_p, colloc_r) ;
707 
708  delete [] nbr ;
709  delete [] nbt ;
710  delete [] nbp ;
711  }
712 
713  return g_three_halves ;
714 }
715 
716  //--------------------------------------//
717  // Grid with twice the number of points //
718  //--------------------------------------//
719 
720 const Mg3d* Mg3d::get_twice() const {
721 
722  if (g_twice == 0x0) { // The construction is required
723 
724  int* nbr = new int[nzone] ;
725  int* nbt = new int[nzone] ;
726  int* nbp = new int[nzone] ;
727 
728  for (int l=0; l<nzone; l++) {
729  if (nr[l] == 1) {
730  nbr[l] = 1 ;
731  }
732  else {
733  nbr[l] = 2*nr[l] - 1 ;
734  }
735 
736  if (nt[l] == 1) {
737  nbt[l] = 1 ;
738  }
739  else {
740  nbt[l] = 2*nt[l] - 1 ;
741  }
742 
743  if (np[l] == 1) {
744  nbp[l] = 1 ;
745  }
746  else {
747  nbp[l] = 2*np[l] ;
748  }
749  }
750 
751  g_twice = new Mg3d(nzone, nbr, type_r, nbt, type_t, nbp, type_p, colloc_r) ;
752 
753  delete [] nbr ;
754  delete [] nbt ;
755  delete [] nbp ;
756 
757  }
758 
759  return g_twice ;
760 
761 }
762 
763 
764  //--------------------------------------//
765  // Grid with 50% additional points in r //
766  //--------------------------------------//
767 
768 const Mg3d* Mg3d::plus_half() const {
769 
770  if (g_plus_half == 0x0) { // The construction is required
771 
772  int* nbr = new int[nzone] ;
773 
774  for (int l=0; l<nzone; l++) {
775  if (nr[l] == 1)
776  nbr[l] = 1 ;
777  else
778  nbr[l] = (3*nr[l])/2 ;
779  }
780 
781  g_plus_half = new Mg3d(nzone, nbr, type_r, nt, type_t, np, type_p, colloc_r) ;
782 
783  delete [] nbr ;
784 
785 
786  }
787 
788  return g_plus_half ;
789 
790 }
791 
792  //-------------------------------------------//
793  // Grid with 50% additional points in angles //
794  //-------------------------------------------//
795 
796 const Mg3d* Mg3d::plus_half_angu() const {
797 
798  if (g_plus_half_angu == 0x0) { // The construction is required
799 
800  int* nbt = new int[nzone] ;
801  int* nbp = new int[nzone] ;
802 
803  for (int l=0; l<nzone; l++) {
804  if (nt[l] == 1)
805  nbt[l] = 1 ;
806  else {
807  int nt_bis = (3*(nt[l]))/2;
808  if (nt_bis % 2 == 0) nt_bis += 1 ;
809  nbt[l] = nt_bis ;
810  }
811 
812  if (np[l] == 1)
813  nbp[l] = 1 ;
814  else
815  nbp[l] = 3*np[l]/2 ;
816  }
817 
818  g_plus_half_angu = new Mg3d(nzone, nr, type_r, nbt, type_t, nbp, type_p, colloc_r) ;
819 
820  delete [] nbt ;
821  delete [] nbp ;
822 
823 
824  }
825 
826  return g_plus_half_angu ;
827 
828 }
829 
830  //----------------------------------------------//
831  // Grid for rotations (5/4 points in theta/phi) //
832  //----------------------------------------------//
833 
834 const Mg3d* Mg3d::get_non_axi() const {
835 
836  if (g_non_axi == 0x0) { // The construction is required
837 
838  int* nbt = new int[nzone] ;
839  int* nbp = new int[nzone] ;
840 
841  for (int l=0; l<nzone; l++) {
842  if (nt[l] < 5)
843  nbt[l] = 5 ;
844  else
845  nbt[l] = nt[l] ;
846  if (np[l] < 4)
847  nbp[l] = 4 ;
848  else
849  nbp[l] = np[l] ;
850  }
851 
852  g_non_axi = new Mg3d(nzone, nr, type_r, nbt, type_t, nbp, type_p, colloc_r) ;
853 
854  delete [] nbt ;
855  delete [] nbp ;
856 
857 
858  }
859 
860  return g_non_axi ;
861 
862 }
863 
864 
865 bool Mg3d::operator==(const Mg3d& mgi) const {
866 
867  bool resu = true ;
868 
869  if (mgi.get_nzone() != nzone) {
870  resu = false ;
871  }
872  else {
873  for (int i=0; i<nzone; i++) {
874  if (mgi.get_nr(i) != nr[i]) resu = false ;
875  if (mgi.get_np(i) != np[i]) resu = false ;
876  if (mgi.get_nt(i) != nt[i]) resu = false ;
877  if (mgi.get_type_r(i) != type_r[i]) resu =false ;
878  if (mgi.get_colloc_r(i) != colloc_r[i]) resu = false ;
879  }
880  }
881 
882  if (mgi.get_type_t() != type_t) resu = false ;
883  if (mgi.get_type_p() != type_p) resu = false ;
884 
885  return resu ;
886 
887 }
888 }
void set_deriv_0x0() const
Sets to 0x0 all the pointers on derived quantities (g_radial , g_angu, g_twice, ...
Definition: mg3d.C:592
int get_type_p() const
Returns the type of sampling in the direction: SYM : : symmetry with respect to the transformatio...
Definition: grilles.h:514
const Mg3d * get_non_axi() const
Returns the pointer on the grid which has at least 4 points in the direction and at least 5 in the ...
Definition: mg3d.C:834
Mg3d * g_non_axi
Pointer on the grid which has at least 4 points in the direction and at least 5 in the direction (f...
Definition: grilles.h:336
int get_np(int l) const
Returns the number of points in the azimuthal direction ( ) in domain no. l.
Definition: grilles.h:481
int type_p
Type of sampling in (SYM, NONSYM)
Definition: grilles.h:296
Lorene prototypes.
Definition: app_hor.h:67
Mg3d * g_plus_half
Pointer on the grid which has 50% more points in r dimension (for desaliasing).
Definition: grilles.h:325
int * np
Array (size: nzone) of nb. of points in .
Definition: grilles.h:285
Mg3d * g_angu_1dom
Pointer on the associated angular grid with only one domain.
Definition: grilles.h:309
const Mg3d * get_twice() const
Returns the pointer on the grid which has twice the number of points in each dimension (for desaliasi...
Definition: mg3d.C:720
void del_deriv() const
Deletes all the derived quantities (g_radial , g_angu, g_twice, ...)
Definition: mg3d.C:577
int get_type_t() const
Returns the type of sampling in the direction: SYM : : symmetry with respect to the equatorial pl...
Definition: grilles.h:504
const Mg3d * get_radial() const
Returns the pointer on the associated radial grid.
Definition: mg3d.C:654
bool operator==(const Mg3d &) const
Comparison operator (egality)
Definition: mg3d.C:865
int type_t
Type of sampling in (SYM, NONSYM)
Definition: grilles.h:293
int get_colloc_r(int l) const
Returns the type of collocation points used in domain no.
Definition: grilles.h:530
~Mg3d()
Destructor.
Definition: mg3d.C:483
Mg3d * g_plus_half_angu
Pointer on the grid which has 50% more points in and dimension (for desaliasing).
Definition: grilles.h:330
void sauve(FILE *fd, bool save_base=false) const
Saves into a file.
Definition: mg3d.C:503
const Mg3d * get_angu() const
Returns the pointer on the associated angular grid.
Definition: mg3d.C:609
bool search_file(ifstream &infile, const string &pattern)
A function that searches for a pattern in a file and places the file stream after the found pattern...
Definition: misc.C:53
int nzone
Number of domains (zones)
Definition: grilles.h:281
Mg3d * g_angu
Pointer on the associated angular grid.
Definition: grilles.h:307
int get_nzone() const
Returns the number of domains.
Definition: grilles.h:467
int fwrite_be(const int *aa, int size, int nb, FILE *fich)
Writes integer(s) into a binary file according to the big endian convention.
Definition: fwrite_be.C:73
Mg3d * g_twice
Pointer on the grid which has twice the number of points in each dimension (for desaliasing).
Definition: grilles.h:320
Mg3d * g_radial
Pointer on the associated radial grid.
Definition: grilles.h:310
const Mg3d * plus_half() const
Returns the pointer on the grid which has 50% more points in r dimension (for desaliasing).
Definition: mg3d.C:768
Grille3d ** g
Array (size: nzone) of pointers on the Grille3d&#39;s.
Definition: grilles.h:305
int * type_r
Array (size: nzone) of type of sampling in r ( ) (RARE,FIN, UNSURR)
Definition: grilles.h:290
int fread_be(int *aa, int size, int nb, FILE *fich)
Reads integer(s) from a binary file according to the big endian convention.
Definition: fread_be.C:72
int get_nr(int l) const
Returns the number of points in the radial direction ( ) in domain no. l.
Definition: grilles.h:471
int * colloc_r
Array (size: nzone) of type of collocation points in r ( ) and related decompoisition bases (BASE_CHE...
Definition: grilles.h:302
Multi-domain grid.
Definition: grilles.h:276
const Mg3d * plus_half_angu() const
Returns the pointer on the grid which has 50% more points in and dimension (for desaliasing)...
Definition: mg3d.C:796
void arrete(int a=0)
Setting a stop point in a code.
Definition: arrete.C:64
int get_nt(int l) const
Returns the number of points in the co-latitude direction ( ) in domain no. l.
Definition: grilles.h:476
bool operator!=(const Mg3d &) const
Operator !=.
Definition: mg3d.C:552
int get_type_r(int l) const
Returns the type of sampling in the radial direction in domain no.
Definition: grilles.h:493
Mg3d(int nz, int nbr[], int typr[], int nbt[], int typt, int nbp[], int typp, int *base_r=0x0)
General constructor.
Definition: mg3d.C:175
int * nt
Array (size: nzone) of nb. of points in .
Definition: grilles.h:284
const Mg3d * get_angu_1dom() const
Returns the pointer on the associated mono-domain angular grid.
Definition: mg3d.C:630
const Mg3d * get_three_halves() const
Returns the pointer on the associated grid with 50% points in every direction for dealiasing...
Definition: mg3d.C:671
Mg3d * g_three_halves
Pointer on the grid having 50% more points in every direction, for dealiasing.
Definition: grilles.h:315
3D grid class in one domain.
Definition: grilles.h:197
int * nr
Array (size: nzone) of nb. of points in r ( )
Definition: grilles.h:283