Ice Sheet System Model  4.18
Code documentation
SegRef.cpp
Go to the documentation of this file.
1 
5 /*Headers:*/
6 /*{{{*/
7 #ifdef HAVE_CONFIG_H
8 #include <config.h>
9 #else
10 #error "Cannot compile with HAVE_CONFIG_H symbol! run configure first!"
11 #endif
12 
13 #include "../classes.h"
14 #include "../../shared/shared.h"
15 /*}}}*/
16 
17 /*Element macros*/
18 #define NUMNODESP0 1
19 #define NUMNODESP1 2
20 #define NUMNODESMAX 2
21 
22 /*Object constructors and destructor*/
23 SegRef::SegRef(){/*{{{*/
24 }
25 /*}}}*/
26 SegRef::~SegRef(){/*{{{*/
27 }
28 /*}}}*/
29 
30 /*Reference Element numerics*/
31 void SegRef::GetInputDerivativeValue(IssmDouble* p, IssmDouble* plist,IssmDouble* xyz_list, GaussSeg* gauss,int finiteelement){/*{{{*/
32 
33  /*From node values of parameter p (plist[0],plist[1]), return parameter derivative value at gaussian
34  * point specified by gauss_basis:
35  * dp/dx=plist[0]*dh1/dx+plist[1]*dh2/dx
36  *
37  * p is a vector already allocated.
38  *
39  * WARNING: For a significant gain in performance, it is better to use
40  * static memory allocation instead of dynamic.
41  */
42 
43  /*Allocate derivatives of basis functions*/
44  IssmDouble dbasis[1*NUMNODESMAX];
45 
46  /*Fetch number of nodes for this finite element*/
47  int numnodes = this->NumberofNodes(finiteelement);
48  _assert_(numnodes<=NUMNODESMAX);
49 
50  /*Get basis functions derivatives at this point*/
51  GetNodalFunctionsDerivatives(&dbasis[0],xyz_list,gauss,finiteelement);
52 
53  /*Calculate parameter for this Gauss point*/
54  IssmDouble dpx=0.;
55  for(int i=0;i<numnodes;i++) dpx += dbasis[0*numnodes+i]*plist[i];
56 
57  /*Assign values*/
58  p[0]=dpx;
59 }
60 /*}}}*/
61 void SegRef::GetInputValue(IssmDouble* p, IssmDouble* plist, GaussSeg* gauss,int finiteelement){/*{{{*/
62  /* WARNING: For a significant gain in performance, it is better to use
63  * static memory allocation instead of dynamic.*/
64 
65  /*Allocate basis functions*/
66  IssmDouble basis[NUMNODESMAX];
67 
68  /*Fetch number of nodes for this finite element*/
69  int numnodes = this->NumberofNodes(finiteelement);
70  _assert_(numnodes<=NUMNODESMAX);
71 
72  /*Get basis functions at this point*/
73  GetNodalFunctions(&basis[0],gauss,finiteelement);
74 
75  /*Calculate parameter for this Gauss point*/
76  IssmDouble value =0.;
77  for(int i=0;i<numnodes;i++) value += basis[i]*plist[i];
78 
79  /*Assign output pointer*/
80  *p = value;
81 }
82 /*}}}*/
83 void SegRef::GetJacobian(IssmDouble* J, IssmDouble* xyz_list,GaussSeg* gauss){/*{{{*/
84  /*The Jacobian is constant over the element, discard the gaussian points.
85  * J is assumed to have been allocated of size 1*/
86 
87  IssmDouble x1=xyz_list[3*0+0];
88  IssmDouble x2=xyz_list[3*1+0];
89 
90  *J=.5*fabs(x2-x1);
91 }
92 /*}}}*/
93 void SegRef::GetJacobianDeterminant(IssmDouble* Jdet, IssmDouble* xyz_list,GaussSeg* gauss){/*{{{*/
94  /*The Jacobian determinant is constant over the element, discard the gaussian points.
95  * J is assumed to have been allocated of size 1.*/
96 
97  /*Call Jacobian routine to get the jacobian:*/
98  GetJacobian(Jdet, xyz_list, gauss);
99  if(*Jdet<0) _error_("negative jacobian determinant!");
100 
101 }
102 /*}}}*/
103 void SegRef::GetJacobianInvert(IssmDouble* Jinv, IssmDouble* xyz_list,GaussSeg* gauss){/*{{{*/
104 
105  /*Jacobian*/
106  IssmDouble J;
107 
108  /*Call Jacobian routine to get the jacobian:*/
109  GetJacobian(&J, xyz_list, gauss);
110 
111  /*Invert Jacobian matrix: */
112  *Jinv = 1./J;
113 }
114 /*}}}*/
115 void SegRef::GetNodalFunctions(IssmDouble* basis,GaussSeg* gauss,int finiteelement){/*{{{*/
116  /*This routine returns the values of the nodal functions at the gaussian point.*/
117 
118  _assert_(basis);
119 
120  switch(finiteelement){
121  case P0Enum:
122  basis[0]=1.;
123  return;
124  case P1Enum: case P1DGEnum:
125  basis[0]=(1.-gauss->coord1)/2.;
126  basis[1]=(1.+gauss->coord1)/2.;
127  return;
128  case P2Enum:
129  basis[0]=(gauss->coord1-1.)*gauss->coord1/2.;
130  basis[1]=gauss->coord1*(1.+gauss->coord1)/2.;
131  basis[2]=(1.-gauss->coord1)*(1.+gauss->coord1);
132  return;
133  default:
134  _error_("Element type "<<EnumToStringx(finiteelement)<<" not supported yet");
135  }
136 }
137 /*}}}*/
138 void SegRef::GetNodalFunctionsDerivatives(IssmDouble* dbasis,IssmDouble* xyz_list, GaussSeg* gauss,int finiteelement){/*{{{*/
139 
140  /*This routine returns the values of the nodal functions derivatives (with respect to the
141  * actual coordinate system): */
142  IssmDouble Jinv;
143 
144  /*Fetch number of nodes for this finite element*/
145  int numnodes = this->NumberofNodes(finiteelement);
146 
147  /*Get nodal functions derivatives in reference triangle*/
148  IssmDouble dbasis_ref[1*NUMNODESMAX];
149  GetNodalFunctionsDerivativesReference(dbasis_ref,gauss,finiteelement);
150 
151  /*Get Jacobian invert: */
152  GetJacobianInvert(&Jinv, xyz_list, gauss);
153 
154  /*Build dbasis:
155  * [dhi/dx]= Jinv*[dhi/dr]
156  */
157  for(int i=0;i<numnodes;i++){
158  dbasis[i] = Jinv*dbasis_ref[i];
159  }
160 }
161 /*}}}*/
162 void SegRef::GetNodalFunctionsDerivativesReference(IssmDouble* dbasis,GaussSeg* gauss,int finiteelement){/*{{{*/
163  /*This routine returns the values of the nodal functions derivatives (with respect to the
164  * natural coordinate system) at the gaussian point. */
165 
166  _assert_(dbasis && gauss);
167 
168  switch(finiteelement){
169  case P0Enum:
170  /*Nodal function 1*/
171  dbasis[0] = 0.;
172  break;
173  case P1Enum: case P1DGEnum:
174  /*Nodal function 1*/
175  dbasis[0] = -0.5;
176  /*Nodal function 2*/
177  dbasis[1] = 0.5;
178  return;
179  case P2Enum:
180  /*Nodal function 1*/
181  dbasis[0] = (gauss->coord1-1.)/2. + gauss->coord1/2.;
182  dbasis[1] = (1.+gauss->coord1)/2. + gauss->coord1/2.;
183  dbasis[2] = -2.*gauss->coord1;
184  return;
185  default:
186  _error_("Element type "<<EnumToStringx(finiteelement)<<" not supported yet");
187  }
188 
189 }
190 /*}}}*/
191 int SegRef::NumberofNodes(int finiteelement){/*{{{*/
192 
193  switch(finiteelement){
194  case P0Enum: return NUMNODESP0;
195  case P1Enum: return NUMNODESP1;
196  case P1DGEnum: return NUMNODESP1;
197  default: _error_("Element type "<<EnumToStringx(finiteelement)<<" not supported yet");
198  }
199 
200  return -1;
201 }
202 /*}}}*/
_assert_
#define _assert_(ignore)
Definition: exceptions.h:37
IssmDouble
double IssmDouble
Definition: types.h:37
GaussSeg::coord1
IssmDouble coord1
Definition: GaussSeg.h:20
SegRef::GetInputValue
void GetInputValue(IssmDouble *p, IssmDouble *plist, GaussSeg *gauss, int finiteelement)
Definition: SegRef.cpp:61
P0Enum
@ P0Enum
Definition: EnumDefinitions.h:661
NUMNODESP0
#define NUMNODESP0
Definition: SegRef.cpp:18
P1DGEnum
@ P1DGEnum
Definition: EnumDefinitions.h:1215
SegRef::GetNodalFunctionsDerivatives
void GetNodalFunctionsDerivatives(IssmDouble *dbasis, IssmDouble *xyz_list, GaussSeg *gauss, int finiteelement)
Definition: SegRef.cpp:138
SegRef::SegRef
SegRef()
Definition: SegRef.cpp:23
SegRef::~SegRef
~SegRef()
Definition: SegRef.cpp:26
NUMNODESMAX
#define NUMNODESMAX
Definition: SegRef.cpp:20
P1Enum
@ P1Enum
Definition: EnumDefinitions.h:662
SegRef::GetNodalFunctionsDerivativesReference
void GetNodalFunctionsDerivativesReference(IssmDouble *dbasis, GaussSeg *gauss, int finiteelement)
Definition: SegRef.cpp:162
EnumToStringx
const char * EnumToStringx(int enum_in)
Definition: EnumToStringx.cpp:15
SegRef::GetJacobianDeterminant
void GetJacobianDeterminant(IssmDouble *Jdet, IssmDouble *xyz_list, GaussSeg *gauss)
Definition: SegRef.cpp:93
SegRef::GetInputDerivativeValue
void GetInputDerivativeValue(IssmDouble *p, IssmDouble *plist, IssmDouble *xyz_list, GaussSeg *gauss, int finiteelement)
Definition: SegRef.cpp:31
SegRef::NumberofNodes
int NumberofNodes(int finiteelement)
Definition: SegRef.cpp:191
SegRef::GetJacobian
void GetJacobian(IssmDouble *J, IssmDouble *xyz_list, GaussSeg *gauss)
Definition: SegRef.cpp:83
_error_
#define _error_(StreamArgs)
Definition: exceptions.h:49
SegRef::GetJacobianInvert
void GetJacobianInvert(IssmDouble *Jinv, IssmDouble *xyz_list, GaussSeg *gauss)
Definition: SegRef.cpp:103
P2Enum
@ P2Enum
Definition: EnumDefinitions.h:1223
SegRef::GetNodalFunctions
void GetNodalFunctions(IssmDouble *basis, GaussSeg *gauss, int finiteelement)
Definition: SegRef.cpp:115
GaussSeg
Definition: GaussSeg.h:12
NUMNODESP1
#define NUMNODESP1
Definition: SegRef.cpp:19