[3683] | 1 | /*!\file Numericalflux.c
|
---|
| 2 | * \brief: implementation of the Numericalflux object
|
---|
| 3 | */
|
---|
| 4 |
|
---|
[5738] | 5 | /*Headers:*/
|
---|
[12365] | 6 | /*{{{*/
|
---|
[3683] | 7 | #ifdef HAVE_CONFIG_H
|
---|
[9320] | 8 | #include <config.h>
|
---|
[3683] | 9 | #else
|
---|
| 10 | #error "Cannot compile with HAVE_CONFIG_H symbol! run configure first!"
|
---|
| 11 | #endif
|
---|
| 12 |
|
---|
[15012] | 13 | #include "shared/shared.h"
|
---|
| 14 | #include "../classes.h"
|
---|
[5738] | 15 | /*}}}*/
|
---|
[3683] | 16 |
|
---|
[5738] | 17 | /*Load macros*/
|
---|
[14763] | 18 | #define NUMVERTICES 2
|
---|
| 19 | #define NUMNODES_INTERNAL 4
|
---|
| 20 | #define NUMNODES_BOUNDARY 2
|
---|
[3683] | 21 |
|
---|
[4248] | 22 | /*Numericalflux constructors and destructor*/
|
---|
[18301] | 23 | Numericalflux::Numericalflux(){/*{{{*/
|
---|
[14761] | 24 | this->parameters = NULL;
|
---|
| 25 | this->helement = NULL;
|
---|
| 26 | this->element = NULL;
|
---|
| 27 | this->hnodes = NULL;
|
---|
| 28 | this->hvertices = NULL;
|
---|
| 29 | this->nodes = NULL;
|
---|
[3683] | 30 | }
|
---|
| 31 | /*}}}*/
|
---|
[18301] | 32 | Numericalflux::Numericalflux(int numericalflux_id,int i,int index,IoModel* iomodel, int in_analysis_type){/*{{{*/
|
---|
[3683] | 33 |
|
---|
| 34 | /* Intermediary */
|
---|
| 35 | int j;
|
---|
[5452] | 36 | int pos1,pos2,pos3,pos4;
|
---|
[3683] | 37 | int num_nodes;
|
---|
| 38 |
|
---|
| 39 | /*numericalflux constructor data: */
|
---|
| 40 | int numericalflux_elem_ids[2];
|
---|
[14761] | 41 | int numericalflux_vertex_ids[2];
|
---|
[3683] | 42 | int numericalflux_node_ids[4];
|
---|
| 43 | int numericalflux_type;
|
---|
| 44 |
|
---|
[16137] | 45 | /*Get edge*/
|
---|
| 46 | int i1 = iomodel->faces[4*index+0];
|
---|
| 47 | int i2 = iomodel->faces[4*index+1];
|
---|
| 48 | int e1 = iomodel->faces[4*index+2];
|
---|
| 49 | int e2 = iomodel->faces[4*index+3];
|
---|
[9356] | 50 |
|
---|
[12574] | 51 | /*First, see wether this is an internal or boundary edge (if e2=-1)*/
|
---|
[16137] | 52 | if(e2==-1){
|
---|
[3683] | 53 | /* Boundary edge, only one element */
|
---|
[16560] | 54 | num_nodes=2;
|
---|
[3683] | 55 | numericalflux_type=BoundaryEnum;
|
---|
[12574] | 56 | numericalflux_elem_ids[0]=e1;
|
---|
[3683] | 57 | }
|
---|
| 58 | else{
|
---|
| 59 | /* internal edge: connected to 2 elements */
|
---|
[16560] | 60 | num_nodes=4;
|
---|
[3683] | 61 | numericalflux_type=InternalEnum;
|
---|
[12574] | 62 | numericalflux_elem_ids[0]=e1;
|
---|
| 63 | numericalflux_elem_ids[1]=e2;
|
---|
[3683] | 64 | }
|
---|
| 65 |
|
---|
| 66 | /*1: Get vertices ids*/
|
---|
[14761] | 67 | numericalflux_vertex_ids[0]=i1;
|
---|
| 68 | numericalflux_vertex_ids[1]=i2;
|
---|
[3683] | 69 |
|
---|
[14761] | 70 | /*2: Get node ids*/
|
---|
[3683] | 71 | if (numericalflux_type==InternalEnum){
|
---|
| 72 |
|
---|
| 73 | /*Now, we must get the nodes of the 4 nodes located on the edge*/
|
---|
| 74 |
|
---|
| 75 | /*2: Get the column where these ids are located in the index*/
|
---|
[5452] | 76 | pos1=pos2=pos3=pos4=UNDEF;
|
---|
[3683] | 77 | for(j=0;j<3;j++){
|
---|
[16137] | 78 | if(iomodel->elements[3*(e1-1)+j]==i1) pos1=j+1;
|
---|
| 79 | if(iomodel->elements[3*(e1-1)+j]==i2) pos2=j+1;
|
---|
| 80 | if(iomodel->elements[3*(e2-1)+j]==i1) pos3=j+1;
|
---|
| 81 | if(iomodel->elements[3*(e2-1)+j]==i2) pos4=j+1;
|
---|
[3683] | 82 | }
|
---|
[6412] | 83 | _assert_(pos1!=UNDEF && pos2!=UNDEF && pos3!=UNDEF && pos4!=UNDEF);
|
---|
[3683] | 84 |
|
---|
| 85 | /*3: We have the id of the elements and the position of the vertices in the index
|
---|
| 86 | * we can compute their dofs!*/
|
---|
[5452] | 87 | numericalflux_node_ids[0]=iomodel->nodecounter+3*(e1-1)+pos1;
|
---|
| 88 | numericalflux_node_ids[1]=iomodel->nodecounter+3*(e1-1)+pos2;
|
---|
| 89 | numericalflux_node_ids[2]=iomodel->nodecounter+3*(e2-1)+pos3;
|
---|
| 90 | numericalflux_node_ids[3]=iomodel->nodecounter+3*(e2-1)+pos4;
|
---|
[3683] | 91 | }
|
---|
| 92 | else{
|
---|
| 93 |
|
---|
| 94 | /*2: Get the column where these ids are located in the index*/
|
---|
[5452] | 95 | pos1=pos2=UNDEF;
|
---|
[3683] | 96 | for(j=0;j<3;j++){
|
---|
[16137] | 97 | if(iomodel->elements[3*(e1-1)+j]==i1) pos1=j+1;
|
---|
| 98 | if(iomodel->elements[3*(e1-1)+j]==i2) pos2=j+1;
|
---|
[3683] | 99 | }
|
---|
[6412] | 100 | _assert_(pos1!=UNDEF && pos2!=UNDEF);
|
---|
[3683] | 101 |
|
---|
| 102 | /*3: We have the id of the elements and the position of the vertices in the index
|
---|
| 103 | * we can compute their dofs!*/
|
---|
[4456] | 104 | numericalflux_node_ids[0]=iomodel->nodecounter+3*(e1-1)+pos1;
|
---|
[5452] | 105 | numericalflux_node_ids[1]=iomodel->nodecounter+3*(e1-1)+pos2;
|
---|
[3683] | 106 | }
|
---|
| 107 |
|
---|
| 108 | /*Ok, we have everything to build the object: */
|
---|
| 109 | this->id=numericalflux_id;
|
---|
[4007] | 110 | this->analysis_type=in_analysis_type;
|
---|
[17806] | 111 | this->flux_type = numericalflux_type;
|
---|
[3683] | 112 |
|
---|
| 113 | /*Hooks: */
|
---|
[14761] | 114 | this->hnodes =new Hook(numericalflux_node_ids,num_nodes);
|
---|
[14763] | 115 | this->hvertices =new Hook(&numericalflux_vertex_ids[0],2);
|
---|
[14761] | 116 | this->helement =new Hook(numericalflux_elem_ids,1); // take only the first element for now
|
---|
[3683] | 117 |
|
---|
| 118 | //this->parameters: we still can't point to it, it may not even exist. Configure will handle this.
|
---|
| 119 | this->parameters=NULL;
|
---|
[5737] | 120 | this->element=NULL;
|
---|
| 121 | this->nodes=NULL;
|
---|
[3683] | 122 | }
|
---|
| 123 | /*}}}*/
|
---|
[18301] | 124 | Numericalflux::~Numericalflux(){/*{{{*/
|
---|
[3683] | 125 | this->parameters=NULL;
|
---|
[4433] | 126 | delete helement;
|
---|
[4396] | 127 | delete hnodes;
|
---|
[14761] | 128 | delete hvertices;
|
---|
[3683] | 129 | }
|
---|
| 130 | /*}}}*/
|
---|
| 131 |
|
---|
[4248] | 132 | /*Object virtual functions definitions:*/
|
---|
[18301] | 133 | Object* Numericalflux::copy() {/*{{{*/
|
---|
[13622] | 134 |
|
---|
[4248] | 135 | Numericalflux* numericalflux=NULL;
|
---|
| 136 |
|
---|
| 137 | numericalflux=new Numericalflux();
|
---|
| 138 |
|
---|
| 139 | /*copy fields: */
|
---|
| 140 | numericalflux->id=this->id;
|
---|
| 141 | numericalflux->analysis_type=this->analysis_type;
|
---|
[17806] | 142 | numericalflux->flux_type=this->flux_type;
|
---|
| 143 |
|
---|
[4248] | 144 | /*point parameters: */
|
---|
| 145 | numericalflux->parameters=this->parameters;
|
---|
| 146 |
|
---|
| 147 | /*now deal with hooks and objects: */
|
---|
[14761] | 148 | numericalflux->hnodes = (Hook*)this->hnodes->copy();
|
---|
| 149 | numericalflux->hvertices = (Hook*)this->hvertices->copy();
|
---|
| 150 | numericalflux->helement = (Hook*)this->helement->copy();
|
---|
[4248] | 151 |
|
---|
[5737] | 152 | /*corresponding fields*/
|
---|
[14761] | 153 | numericalflux->nodes = (Node**)numericalflux->hnodes->deliverp();
|
---|
| 154 | numericalflux->vertices = (Vertex**)numericalflux->hvertices->deliverp();
|
---|
| 155 | numericalflux->element = (Element*)numericalflux->helement->delivers();
|
---|
[5737] | 156 |
|
---|
[4248] | 157 | return numericalflux;
|
---|
| 158 | }
|
---|
| 159 | /*}}}*/
|
---|
[20500] | 160 | void Numericalflux::Marshall(char** pmarshalled_data,int* pmarshalled_data_size, int marshall_direction){ /*{{{*/
|
---|
| 161 |
|
---|
| 162 | _assert_(this);
|
---|
| 163 |
|
---|
| 164 | /*ok, marshall operations: */
|
---|
| 165 | MARSHALLING_ENUM(NumericalfluxEnum);
|
---|
| 166 | MARSHALLING(id);
|
---|
| 167 | MARSHALLING(analysis_type);
|
---|
| 168 | MARSHALLING(flux_type);
|
---|
| 169 |
|
---|
| 170 | if(marshall_direction==MARSHALLING_BACKWARD){
|
---|
| 171 | this->hnodes = new Hook();
|
---|
| 172 | this->hvertices = new Hook();
|
---|
| 173 | this->helement = new Hook();
|
---|
| 174 | }
|
---|
| 175 |
|
---|
| 176 | this->hnodes->Marshall(pmarshalled_data,pmarshalled_data_size,marshall_direction);
|
---|
| 177 | this->helement->Marshall(pmarshalled_data,pmarshalled_data_size,marshall_direction);
|
---|
| 178 | this->hvertices->Marshall(pmarshalled_data,pmarshalled_data_size,marshall_direction);
|
---|
| 179 |
|
---|
| 180 | /*corresponding fields*/
|
---|
| 181 | nodes =(Node**)this->hnodes->deliverp();
|
---|
| 182 | vertices =(Vertex**)this->hvertices->deliverp();
|
---|
| 183 | element =(Element*)this->helement->delivers();
|
---|
| 184 |
|
---|
| 185 | }
|
---|
| 186 | /*}}}*/
|
---|
[19105] | 187 | void Numericalflux::DeepEcho(void){/*{{{*/
|
---|
[4248] | 188 |
|
---|
[19105] | 189 | _printf_("Numericalflux:\n");
|
---|
| 190 | _printf_(" id: " << id << "\n");
|
---|
| 191 | _printf_(" analysis_type: " << EnumToStringx(analysis_type) << "\n");
|
---|
| 192 | _printf_(" flux_type: " << this->flux_type<< "\n");
|
---|
| 193 | hnodes->DeepEcho();
|
---|
| 194 | hvertices->DeepEcho();
|
---|
| 195 | helement->DeepEcho();
|
---|
| 196 | _printf_(" parameters\n");
|
---|
| 197 | if(parameters)
|
---|
| 198 | parameters->DeepEcho();
|
---|
| 199 | else
|
---|
| 200 | _printf_(" NULL\n");
|
---|
| 201 | }
|
---|
| 202 | /*}}}*/
|
---|
| 203 | void Numericalflux::Echo(void){/*{{{*/
|
---|
| 204 | _printf_("Numericalflux:\n");
|
---|
| 205 | _printf_(" id: " << id << "\n");
|
---|
| 206 | _printf_(" analysis_type: " << EnumToStringx(analysis_type) << "\n");
|
---|
| 207 | _printf_(" flux_type: " << this->flux_type<< "\n");
|
---|
| 208 | hnodes->Echo();
|
---|
| 209 | hvertices->Echo();
|
---|
| 210 | helement->Echo();
|
---|
| 211 | _printf_(" parameters: " << parameters << "\n");
|
---|
| 212 | }
|
---|
| 213 | /*}}}*/
|
---|
| 214 | int Numericalflux::Id(void){/*{{{*/
|
---|
| 215 | return id;
|
---|
| 216 | }
|
---|
| 217 | /*}}}*/
|
---|
| 218 | int Numericalflux::ObjectEnum(void){/*{{{*/
|
---|
| 219 |
|
---|
| 220 | return NumericalfluxEnum;
|
---|
| 221 |
|
---|
| 222 | }
|
---|
| 223 | /*}}}*/
|
---|
| 224 |
|
---|
[4248] | 225 | /*Load virtual functions definitions:*/
|
---|
[18301] | 226 | void Numericalflux::Configure(Elements* elementsin,Loads* loadsin,Nodes* nodesin,Vertices* verticesin,Materials* materialsin,Parameters* parametersin){/*{{{*/
|
---|
[4248] | 227 |
|
---|
| 228 | /*Take care of hooking up all objects for this element, ie links the objects in the hooks to their respective
|
---|
| 229 | * datasets, using internal ids and offsets hidden in hooks: */
|
---|
[14975] | 230 | hnodes->configure((DataSet*)nodesin);
|
---|
| 231 | hvertices->configure((DataSet*)verticesin);
|
---|
| 232 | helement->configure((DataSet*)elementsin);
|
---|
[4248] | 233 |
|
---|
[5737] | 234 | /*Initialize hooked fields*/
|
---|
[14761] | 235 | this->nodes = (Node**)hnodes->deliverp();
|
---|
| 236 | this->vertices = (Vertex**)hvertices->deliverp();
|
---|
| 237 | this->element = (Element*)helement->delivers();
|
---|
[5737] | 238 |
|
---|
[4248] | 239 | /*point parameters to real dataset: */
|
---|
| 240 | this->parameters=parametersin;
|
---|
| 241 | }
|
---|
| 242 | /*}}}*/
|
---|
[18301] | 243 | void Numericalflux::CreateKMatrix(Matrix<IssmDouble>* Kff, Matrix<IssmDouble>* Kfs){/*{{{*/
|
---|
| 244 |
|
---|
[6026] | 245 | /*recover some parameters*/
|
---|
| 246 | ElementMatrix* Ke=NULL;
|
---|
[5455] | 247 | int analysis_type;
|
---|
| 248 | this->parameters->FindParam(&analysis_type,AnalysisTypeEnum);
|
---|
[3683] | 249 |
|
---|
[5911] | 250 | /*Just branch to the correct element stiffness matrix generator, according to the type of analysis we are carrying out: */
|
---|
| 251 | switch(analysis_type){
|
---|
[16137] | 252 | case MasstransportAnalysisEnum:
|
---|
| 253 | Ke=CreateKMatrixMasstransport();
|
---|
[5911] | 254 | break;
|
---|
[8287] | 255 | case BalancethicknessAnalysisEnum:
|
---|
| 256 | Ke=CreateKMatrixBalancethickness();
|
---|
[6026] | 257 | break;
|
---|
[8287] | 258 | case AdjointBalancethicknessAnalysisEnum:
|
---|
| 259 | Ke=CreateKMatrixAdjointBalancethickness();
|
---|
[6026] | 260 | break;
|
---|
[5911] | 261 | default:
|
---|
[13036] | 262 | _error_("analysis " << analysis_type << " (" << EnumToStringx(analysis_type) << ") not supported yet");
|
---|
[3683] | 263 | }
|
---|
[5911] | 264 |
|
---|
| 265 | /*Add to global matrix*/
|
---|
| 266 | if(Ke){
|
---|
[8800] | 267 | Ke->AddToGlobal(Kff,Kfs);
|
---|
[5911] | 268 | delete Ke;
|
---|
[3683] | 269 | }
|
---|
| 270 |
|
---|
| 271 | }
|
---|
| 272 | /*}}}*/
|
---|
[18301] | 273 | void Numericalflux::CreatePVector(Vector<IssmDouble>* pf){/*{{{*/
|
---|
[4248] | 274 |
|
---|
[6026] | 275 | /*recover some parameters*/
|
---|
| 276 | ElementVector* pe=NULL;
|
---|
[5455] | 277 | int analysis_type;
|
---|
| 278 | this->parameters->FindParam(&analysis_type,AnalysisTypeEnum);
|
---|
[4248] | 279 |
|
---|
[5911] | 280 | switch(analysis_type){
|
---|
[16137] | 281 | case MasstransportAnalysisEnum:
|
---|
| 282 | pe=CreatePVectorMasstransport();
|
---|
[5911] | 283 | break;
|
---|
[8287] | 284 | case BalancethicknessAnalysisEnum:
|
---|
| 285 | pe=CreatePVectorBalancethickness();
|
---|
[6026] | 286 | break;
|
---|
[8287] | 287 | case AdjointBalancethicknessAnalysisEnum:
|
---|
| 288 | pe=CreatePVectorAdjointBalancethickness();
|
---|
[6026] | 289 | break;
|
---|
[5911] | 290 | default:
|
---|
[13036] | 291 | _error_("analysis " << analysis_type << " (" << EnumToStringx(analysis_type) << ") not supported yet");
|
---|
[4248] | 292 | }
|
---|
[5911] | 293 |
|
---|
| 294 | /*Add to global matrix*/
|
---|
| 295 | if(pe){
|
---|
[8800] | 296 | pe->AddToGlobal(pf);
|
---|
[5911] | 297 | delete pe;
|
---|
[4248] | 298 | }
|
---|
| 299 |
|
---|
| 300 | }
|
---|
| 301 | /*}}}*/
|
---|
[19105] | 302 | void Numericalflux::GetNodesLidList(int* lidlist){/*{{{*/
|
---|
[13915] | 303 |
|
---|
[19105] | 304 | _assert_(lidlist);
|
---|
[13915] | 305 | _assert_(nodes);
|
---|
| 306 |
|
---|
[17806] | 307 | switch(this->flux_type){
|
---|
[13915] | 308 | case InternalEnum:
|
---|
[19105] | 309 | for(int i=0;i<NUMNODES_INTERNAL;i++) lidlist[i]=nodes[i]->Lid();
|
---|
[13915] | 310 | return;
|
---|
| 311 | case BoundaryEnum:
|
---|
[19105] | 312 | for(int i=0;i<NUMNODES_BOUNDARY;i++) lidlist[i]=nodes[i]->Lid();
|
---|
[13915] | 313 | return;
|
---|
| 314 | default:
|
---|
[17806] | 315 | _error_("Numericalflux type " << EnumToStringx(this->flux_type) << " not supported yet");
|
---|
[13915] | 316 | }
|
---|
| 317 | }
|
---|
| 318 | /*}}}*/
|
---|
[19105] | 319 | void Numericalflux::GetNodesSidList(int* sidlist){/*{{{*/
|
---|
[16137] | 320 |
|
---|
[19105] | 321 | _assert_(sidlist);
|
---|
[16137] | 322 | _assert_(nodes);
|
---|
| 323 |
|
---|
[17806] | 324 | switch(this->flux_type){
|
---|
[16137] | 325 | case InternalEnum:
|
---|
[19105] | 326 | for(int i=0;i<NUMNODES_INTERNAL;i++) sidlist[i]=nodes[i]->Sid();
|
---|
[16137] | 327 | return;
|
---|
| 328 | case BoundaryEnum:
|
---|
[19105] | 329 | for(int i=0;i<NUMNODES_BOUNDARY;i++) sidlist[i]=nodes[i]->Sid();
|
---|
[16137] | 330 | return;
|
---|
| 331 | default:
|
---|
[17806] | 332 | _error_("Numericalflux type " << EnumToStringx(this->flux_type) << " not supported yet");
|
---|
[16137] | 333 | }
|
---|
| 334 | }
|
---|
| 335 | /*}}}*/
|
---|
[18301] | 336 | int Numericalflux::GetNumberOfNodes(void){/*{{{*/
|
---|
[13915] | 337 |
|
---|
[17806] | 338 | switch(this->flux_type){
|
---|
[13915] | 339 | case InternalEnum:
|
---|
[14763] | 340 | return NUMNODES_INTERNAL;
|
---|
[13915] | 341 | case BoundaryEnum:
|
---|
[14763] | 342 | return NUMNODES_BOUNDARY;
|
---|
[13915] | 343 | default:
|
---|
[17806] | 344 | _error_("Numericalflux type " << EnumToStringx(this->flux_type) << " not supported yet");
|
---|
[13915] | 345 | }
|
---|
| 346 |
|
---|
| 347 | }
|
---|
| 348 | /*}}}*/
|
---|
[19105] | 349 | bool Numericalflux::InAnalysis(int in_analysis_type){/*{{{*/
|
---|
| 350 | if (in_analysis_type==this->analysis_type) return true;
|
---|
| 351 | else return false;
|
---|
| 352 | }
|
---|
| 353 | /*}}}*/
|
---|
[18301] | 354 | bool Numericalflux::IsPenalty(void){/*{{{*/
|
---|
[13925] | 355 | return false;
|
---|
| 356 | }
|
---|
| 357 | /*}}}*/
|
---|
[18301] | 358 | void Numericalflux::PenaltyCreateKMatrix(Matrix<IssmDouble>* Kff, Matrix<IssmDouble>* Kfs,IssmDouble kmax){/*{{{*/
|
---|
[4248] | 359 |
|
---|
| 360 | /*No stiffness loads applied, do nothing: */
|
---|
| 361 | return;
|
---|
| 362 |
|
---|
| 363 | }
|
---|
| 364 | /*}}}*/
|
---|
[18301] | 365 | void Numericalflux::PenaltyCreatePVector(Vector<IssmDouble>* pf,IssmDouble kmax){/*{{{*/
|
---|
[4248] | 366 |
|
---|
| 367 | /*No penalty loads applied, do nothing: */
|
---|
| 368 | return;
|
---|
| 369 |
|
---|
| 370 | }
|
---|
| 371 | /*}}}*/
|
---|
[19105] | 372 | void Numericalflux::ResetHooks(){/*{{{*/
|
---|
| 373 |
|
---|
| 374 | this->nodes=NULL;
|
---|
| 375 | this->vertices=NULL;
|
---|
| 376 | this->element=NULL;
|
---|
| 377 | this->parameters=NULL;
|
---|
| 378 |
|
---|
| 379 | /*Get Element type*/
|
---|
| 380 | this->hnodes->reset();
|
---|
| 381 | this->hvertices->reset();
|
---|
| 382 | this->helement->reset();
|
---|
| 383 |
|
---|
[4248] | 384 | }
|
---|
| 385 | /*}}}*/
|
---|
[19105] | 386 | void Numericalflux::SetCurrentConfiguration(Elements* elementsin,Loads* loadsin,Nodes* nodesin,Vertices* verticesin,Materials* materialsin,Parameters* parametersin){/*{{{*/
|
---|
| 387 |
|
---|
| 388 | }
|
---|
| 389 | /*}}}*/
|
---|
[18301] | 390 | void Numericalflux::SetwiseNodeConnectivity(int* pd_nz,int* po_nz,Node* node,bool* flags,int* flagsindices,int set1_enum,int set2_enum){/*{{{*/
|
---|
[4248] | 391 |
|
---|
[13915] | 392 | /*Output */
|
---|
| 393 | int d_nz = 0;
|
---|
| 394 | int o_nz = 0;
|
---|
| 395 |
|
---|
| 396 | /*Loop over all nodes*/
|
---|
| 397 | for(int i=0;i<this->GetNumberOfNodes();i++){
|
---|
| 398 |
|
---|
[16137] | 399 | if(!flags[this->nodes[i]->Lid()]){
|
---|
[13915] | 400 |
|
---|
| 401 | /*flag current node so that no other element processes it*/
|
---|
[16137] | 402 | flags[this->nodes[i]->Lid()]=true;
|
---|
[13915] | 403 |
|
---|
[16137] | 404 | int counter=0;
|
---|
| 405 | while(flagsindices[counter]>=0) counter++;
|
---|
| 406 | flagsindices[counter]=this->nodes[i]->Lid();
|
---|
| 407 |
|
---|
[13915] | 408 | /*if node is clone, we have an off-diagonal non-zero, else it is a diagonal non-zero*/
|
---|
| 409 | switch(set2_enum){
|
---|
| 410 | case FsetEnum:
|
---|
| 411 | if(nodes[i]->indexing.fsize){
|
---|
| 412 | if(this->nodes[i]->IsClone())
|
---|
| 413 | o_nz += 1;
|
---|
| 414 | else
|
---|
| 415 | d_nz += 1;
|
---|
| 416 | }
|
---|
| 417 | break;
|
---|
| 418 | case GsetEnum:
|
---|
| 419 | if(nodes[i]->indexing.gsize){
|
---|
| 420 | if(this->nodes[i]->IsClone())
|
---|
| 421 | o_nz += 1;
|
---|
| 422 | else
|
---|
| 423 | d_nz += 1;
|
---|
| 424 | }
|
---|
| 425 | break;
|
---|
| 426 | case SsetEnum:
|
---|
| 427 | if(nodes[i]->indexing.ssize){
|
---|
| 428 | if(this->nodes[i]->IsClone())
|
---|
| 429 | o_nz += 1;
|
---|
| 430 | else
|
---|
| 431 | d_nz += 1;
|
---|
| 432 | }
|
---|
| 433 | break;
|
---|
| 434 | default: _error_("not supported");
|
---|
| 435 | }
|
---|
| 436 | }
|
---|
| 437 | }
|
---|
| 438 |
|
---|
| 439 | /*Assign output pointers: */
|
---|
| 440 | *pd_nz=d_nz;
|
---|
| 441 | *po_nz=o_nz;
|
---|
| 442 | }
|
---|
| 443 | /*}}}*/
|
---|
| 444 |
|
---|
[4248] | 445 | /*Numericalflux management*/
|
---|
[19105] | 446 | ElementMatrix* Numericalflux::CreateKMatrixAdjointBalancethickness(void){/*{{{*/
|
---|
[3683] | 447 |
|
---|
[17806] | 448 | switch(this->flux_type){
|
---|
[6026] | 449 | case InternalEnum:
|
---|
[19105] | 450 | return CreateKMatrixAdjointBalancethicknessInternal();
|
---|
[6026] | 451 | case BoundaryEnum:
|
---|
[19105] | 452 | return CreateKMatrixAdjointBalancethicknessBoundary();
|
---|
[6026] | 453 | default:
|
---|
[13036] | 454 | _error_("type not supported yet");
|
---|
[6026] | 455 | }
|
---|
| 456 | }
|
---|
| 457 | /*}}}*/
|
---|
[19105] | 458 | ElementMatrix* Numericalflux::CreateKMatrixAdjointBalancethicknessBoundary(void){/*{{{*/
|
---|
[6026] | 459 |
|
---|
[19105] | 460 | ElementMatrix* Ke=CreateKMatrixBalancethicknessBoundary();
|
---|
| 461 | if(Ke) Ke->Transpose();
|
---|
| 462 | return Ke;
|
---|
| 463 | }
|
---|
| 464 | /*}}}*/
|
---|
| 465 | ElementMatrix* Numericalflux::CreateKMatrixAdjointBalancethicknessInternal(void){/*{{{*/
|
---|
[3683] | 466 |
|
---|
[19105] | 467 | ElementMatrix* Ke=CreateKMatrixBalancethicknessInternal();
|
---|
| 468 | if (Ke) Ke->Transpose();
|
---|
| 469 | return Ke;
|
---|
| 470 | }
|
---|
| 471 | /*}}}*/
|
---|
| 472 | ElementMatrix* Numericalflux::CreateKMatrixBalancethickness(void){/*{{{*/
|
---|
[3683] | 473 |
|
---|
[19105] | 474 | switch(this->flux_type){
|
---|
| 475 | case InternalEnum:
|
---|
| 476 | return CreateKMatrixBalancethicknessInternal();
|
---|
| 477 | case BoundaryEnum:
|
---|
| 478 | return CreateKMatrixBalancethicknessBoundary();
|
---|
| 479 | default:
|
---|
| 480 | _error_("type not supported yet");
|
---|
[5738] | 481 | }
|
---|
[3683] | 482 | }
|
---|
| 483 | /*}}}*/
|
---|
[19105] | 484 | ElementMatrix* Numericalflux::CreateKMatrixBalancethicknessBoundary(void){/*{{{*/
|
---|
[3683] | 485 |
|
---|
[5739] | 486 | /* constants*/
|
---|
[14763] | 487 | const int numdof=NDOF1*NUMNODES_BOUNDARY;
|
---|
[3683] | 488 |
|
---|
[5739] | 489 | /* Intermediaries*/
|
---|
[6026] | 490 | int i,j,ig,index1,index2;
|
---|
[19105] | 491 | IssmDouble DL,Jdet,vx,vy,mean_vx,mean_vy,UdotN;
|
---|
[14763] | 492 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
[12472] | 493 | IssmDouble normal[2];
|
---|
| 494 | IssmDouble L[numdof];
|
---|
| 495 | IssmDouble Ke_g[numdof][numdof];
|
---|
[5739] | 496 | GaussTria *gauss;
|
---|
[3683] | 497 |
|
---|
[5911] | 498 | /*Initialize Element matrix and return if necessary*/
|
---|
[6029] | 499 | ElementMatrix* Ke = NULL;
|
---|
[5911] | 500 | Tria* tria=(Tria*)element;
|
---|
[17806] | 501 | if(!tria->IsIceInElement()) return NULL;
|
---|
[5911] | 502 |
|
---|
| 503 | /*Retrieve all inputs and parameters*/
|
---|
[14763] | 504 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
[19105] | 505 | Input* vxaverage_input=tria->inputs->GetInput(VxEnum);
|
---|
| 506 | Input* vyaverage_input=tria->inputs->GetInput(VyEnum);
|
---|
[3683] | 507 | GetNormal(&normal[0],xyz_list);
|
---|
[6026] | 508 |
|
---|
| 509 | /*Check wether it is an inflow or outflow BC (0 is the middle of the segment)*/
|
---|
| 510 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 511 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 512 |
|
---|
| 513 | gauss=new GaussTria();
|
---|
| 514 | gauss->GaussEdgeCenter(index1,index2);
|
---|
[10135] | 515 | vxaverage_input->GetInputValue(&mean_vx,gauss);
|
---|
| 516 | vyaverage_input->GetInputValue(&mean_vy,gauss);
|
---|
[6026] | 517 | delete gauss;
|
---|
| 518 |
|
---|
| 519 | UdotN=mean_vx*normal[0]+mean_vy*normal[1];
|
---|
| 520 | if (UdotN<=0){
|
---|
[6029] | 521 | return NULL; /*(u,n)<0 -> inflow, PenaltyCreatePVector will take care of it*/
|
---|
[6026] | 522 | }
|
---|
[6029] | 523 | else{
|
---|
[14763] | 524 | Ke=new ElementMatrix(nodes,NUMNODES_BOUNDARY,this->parameters);
|
---|
[6029] | 525 | }
|
---|
[6026] | 526 |
|
---|
| 527 | /* Start looping on the number of gaussian points: */
|
---|
| 528 | gauss=new GaussTria(index1,index2,2);
|
---|
| 529 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
| 530 |
|
---|
| 531 | gauss->GaussPoint(ig);
|
---|
| 532 |
|
---|
[18301] | 533 | tria->GetSegmentNodalFunctions(&L[0],gauss,index1,index2,tria->FiniteElement());
|
---|
[6026] | 534 |
|
---|
[10135] | 535 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 536 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
[6026] | 537 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
| 538 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
[19105] | 539 | DL=gauss->weight*Jdet*UdotN;
|
---|
[6026] | 540 |
|
---|
| 541 | TripleMultiply(&L[0],1,numdof,1,
|
---|
| 542 | &DL,1,1,0,
|
---|
| 543 | &L[0],1,numdof,0,
|
---|
| 544 | &Ke_g[0][0],0);
|
---|
| 545 |
|
---|
| 546 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g[i][j];
|
---|
| 547 | }
|
---|
| 548 |
|
---|
| 549 | /*Clean up and return*/
|
---|
| 550 | delete gauss;
|
---|
| 551 | return Ke;
|
---|
| 552 | }
|
---|
| 553 | /*}}}*/
|
---|
[18301] | 554 | ElementMatrix* Numericalflux::CreateKMatrixBalancethicknessInternal(void){/*{{{*/
|
---|
[4904] | 555 |
|
---|
[6026] | 556 | /* constants*/
|
---|
[14763] | 557 | const int numdof=NDOF1*NUMNODES_INTERNAL;
|
---|
[6026] | 558 |
|
---|
| 559 | /* Intermediaries*/
|
---|
| 560 | int i,j,ig,index1,index2;
|
---|
[12472] | 561 | IssmDouble DL1,DL2,Jdet,vx,vy,UdotN;
|
---|
[14763] | 562 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
[12472] | 563 | IssmDouble normal[2];
|
---|
| 564 | IssmDouble B[numdof];
|
---|
| 565 | IssmDouble Bprime[numdof];
|
---|
| 566 | IssmDouble Ke_g1[numdof][numdof];
|
---|
| 567 | IssmDouble Ke_g2[numdof][numdof];
|
---|
[6026] | 568 | GaussTria *gauss;
|
---|
| 569 |
|
---|
| 570 | /*Initialize Element matrix and return if necessary*/
|
---|
| 571 | Tria* tria=(Tria*)element;
|
---|
[17806] | 572 | if(!tria->IsIceInElement()) return NULL;
|
---|
[14763] | 573 | ElementMatrix* Ke=new ElementMatrix(nodes,NUMNODES_INTERNAL,this->parameters);
|
---|
[6026] | 574 |
|
---|
| 575 | /*Retrieve all inputs and parameters*/
|
---|
[14763] | 576 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
[6026] | 577 | Input* vxaverage_input=tria->inputs->GetInput(VxEnum);
|
---|
| 578 | Input* vyaverage_input=tria->inputs->GetInput(VyEnum);
|
---|
| 579 | GetNormal(&normal[0],xyz_list);
|
---|
| 580 |
|
---|
| 581 | /* Start looping on the number of gaussian points: */
|
---|
| 582 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 583 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 584 | gauss=new GaussTria(index1,index2,2);
|
---|
| 585 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
| 586 |
|
---|
| 587 | gauss->GaussPoint(ig);
|
---|
| 588 |
|
---|
[18301] | 589 | tria->GetSegmentBFlux(&B[0],gauss,index1,index2,tria->FiniteElement());
|
---|
| 590 | tria->GetSegmentBprimeFlux(&Bprime[0],gauss,index1,index2,tria->FiniteElement());
|
---|
[6026] | 591 |
|
---|
[10135] | 592 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 593 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
[6026] | 594 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
| 595 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
| 596 | DL1=gauss->weight*Jdet*UdotN/2;
|
---|
| 597 | DL2=gauss->weight*Jdet*fabs(UdotN)/2;
|
---|
| 598 |
|
---|
| 599 | TripleMultiply(&B[0],1,numdof,1,
|
---|
| 600 | &DL1,1,1,0,
|
---|
| 601 | &Bprime[0],1,numdof,0,
|
---|
| 602 | &Ke_g1[0][0],0);
|
---|
| 603 | TripleMultiply(&B[0],1,numdof,1,
|
---|
| 604 | &DL2,1,1,0,
|
---|
| 605 | &B[0],1,numdof,0,
|
---|
| 606 | &Ke_g2[0][0],0);
|
---|
| 607 |
|
---|
| 608 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g1[i][j];
|
---|
| 609 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g2[i][j];
|
---|
| 610 | }
|
---|
| 611 |
|
---|
| 612 | /*Clean up and return*/
|
---|
| 613 | delete gauss;
|
---|
| 614 | return Ke;
|
---|
| 615 | }
|
---|
| 616 | /*}}}*/
|
---|
[19105] | 617 | ElementMatrix* Numericalflux::CreateKMatrixMasstransport(void){/*{{{*/
|
---|
[6026] | 618 |
|
---|
[19105] | 619 | switch(this->flux_type){
|
---|
| 620 | case InternalEnum:
|
---|
| 621 | return CreateKMatrixMasstransportInternal();
|
---|
| 622 | case BoundaryEnum:
|
---|
| 623 | return CreateKMatrixMasstransportBoundary();
|
---|
| 624 | default:
|
---|
| 625 | _error_("type not supported yet");
|
---|
| 626 | }
|
---|
| 627 | }
|
---|
| 628 | /*}}}*/
|
---|
| 629 | ElementMatrix* Numericalflux::CreateKMatrixMasstransportBoundary(void){/*{{{*/
|
---|
| 630 |
|
---|
[6026] | 631 | /* constants*/
|
---|
[14763] | 632 | const int numdof=NDOF1*NUMNODES_BOUNDARY;
|
---|
[6026] | 633 |
|
---|
| 634 | /* Intermediaries*/
|
---|
| 635 | int i,j,ig,index1,index2;
|
---|
[19105] | 636 | IssmDouble DL,Jdet,dt,vx,vy,mean_vx,mean_vy,UdotN;
|
---|
[14763] | 637 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
[12472] | 638 | IssmDouble normal[2];
|
---|
| 639 | IssmDouble L[numdof];
|
---|
| 640 | IssmDouble Ke_g[numdof][numdof];
|
---|
[6026] | 641 | GaussTria *gauss;
|
---|
| 642 |
|
---|
| 643 | /*Initialize Element matrix and return if necessary*/
|
---|
[6029] | 644 | ElementMatrix* Ke = NULL;
|
---|
[6026] | 645 | Tria* tria=(Tria*)element;
|
---|
[17806] | 646 | if(!tria->IsIceInElement()) return NULL;
|
---|
[6026] | 647 |
|
---|
| 648 | /*Retrieve all inputs and parameters*/
|
---|
[14763] | 649 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
[19105] | 650 | parameters->FindParam(&dt,TimesteppingTimeStepEnum);
|
---|
| 651 | Input* vxaverage_input=tria->inputs->GetInput(VxEnum); _assert_(vxaverage_input);
|
---|
| 652 | Input* vyaverage_input=tria->inputs->GetInput(VyEnum); _assert_(vyaverage_input);
|
---|
[6026] | 653 | GetNormal(&normal[0],xyz_list);
|
---|
| 654 |
|
---|
[4422] | 655 | /*Check wether it is an inflow or outflow BC (0 is the middle of the segment)*/
|
---|
[5739] | 656 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 657 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 658 |
|
---|
| 659 | gauss=new GaussTria();
|
---|
| 660 | gauss->GaussEdgeCenter(index1,index2);
|
---|
[10135] | 661 | vxaverage_input->GetInputValue(&mean_vx,gauss);
|
---|
| 662 | vyaverage_input->GetInputValue(&mean_vy,gauss);
|
---|
[5739] | 663 | delete gauss;
|
---|
| 664 |
|
---|
[3683] | 665 | UdotN=mean_vx*normal[0]+mean_vy*normal[1];
|
---|
| 666 | if (UdotN<=0){
|
---|
[6029] | 667 | return NULL; /*(u,n)<0 -> inflow, PenaltyCreatePVector will take care of it*/
|
---|
[3683] | 668 | }
|
---|
[6029] | 669 | else{
|
---|
[14763] | 670 | Ke=new ElementMatrix(nodes,NUMNODES_BOUNDARY,this->parameters);
|
---|
[6029] | 671 | }
|
---|
[3683] | 672 |
|
---|
| 673 | /* Start looping on the number of gaussian points: */
|
---|
[5739] | 674 | gauss=new GaussTria(index1,index2,2);
|
---|
| 675 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
[3683] | 676 |
|
---|
[5739] | 677 | gauss->GaussPoint(ig);
|
---|
[3683] | 678 |
|
---|
[18301] | 679 | tria->GetSegmentNodalFunctions(&L[0],gauss,index1,index2,tria->FiniteElement());
|
---|
[3683] | 680 |
|
---|
[10135] | 681 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 682 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
[3683] | 683 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
[5739] | 684 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
[19105] | 685 | DL=gauss->weight*Jdet*dt*UdotN;
|
---|
[3683] | 686 |
|
---|
| 687 | TripleMultiply(&L[0],1,numdof,1,
|
---|
| 688 | &DL,1,1,0,
|
---|
| 689 | &L[0],1,numdof,0,
|
---|
[5739] | 690 | &Ke_g[0][0],0);
|
---|
[3683] | 691 |
|
---|
[5911] | 692 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g[i][j];
|
---|
[5739] | 693 | }
|
---|
[3683] | 694 |
|
---|
[5911] | 695 | /*Clean up and return*/
|
---|
[5739] | 696 | delete gauss;
|
---|
[5911] | 697 | return Ke;
|
---|
[3683] | 698 | }
|
---|
| 699 | /*}}}*/
|
---|
[19105] | 700 | ElementMatrix* Numericalflux::CreateKMatrixMasstransportInternal(void){/*{{{*/
|
---|
[3683] | 701 |
|
---|
[19105] | 702 | /* constants*/
|
---|
| 703 | const int numdof=NDOF1*NUMNODES_INTERNAL;
|
---|
| 704 |
|
---|
| 705 | /* Intermediaries*/
|
---|
| 706 | int i,j,ig,index1,index2;
|
---|
| 707 | IssmDouble DL1,DL2,Jdet,dt,vx,vy,UdotN;
|
---|
| 708 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
| 709 | IssmDouble normal[2];
|
---|
| 710 | IssmDouble B[numdof];
|
---|
| 711 | IssmDouble Bprime[numdof];
|
---|
| 712 | IssmDouble Ke_g1[numdof][numdof];
|
---|
| 713 | IssmDouble Ke_g2[numdof][numdof];
|
---|
| 714 | GaussTria *gauss;
|
---|
| 715 |
|
---|
| 716 | /*Initialize Element matrix and return if necessary*/
|
---|
| 717 | Tria* tria=(Tria*)element;
|
---|
| 718 | if(!tria->IsIceInElement()) return NULL;
|
---|
| 719 | ElementMatrix* Ke=new ElementMatrix(nodes,NUMNODES_INTERNAL,this->parameters);
|
---|
| 720 |
|
---|
| 721 | /*Retrieve all inputs and parameters*/
|
---|
| 722 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
| 723 | parameters->FindParam(&dt,TimesteppingTimeStepEnum);
|
---|
| 724 | Input* vxaverage_input=tria->inputs->GetInput(VxEnum);
|
---|
| 725 | Input* vyaverage_input=tria->inputs->GetInput(VyEnum);
|
---|
| 726 | GetNormal(&normal[0],xyz_list);
|
---|
| 727 |
|
---|
| 728 | /* Start looping on the number of gaussian points: */
|
---|
| 729 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 730 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 731 | gauss=new GaussTria(index1,index2,2);
|
---|
| 732 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
| 733 |
|
---|
| 734 | gauss->GaussPoint(ig);
|
---|
| 735 |
|
---|
| 736 | tria->GetSegmentBFlux(&B[0],gauss,index1,index2,tria->FiniteElement());
|
---|
| 737 | tria->GetSegmentBprimeFlux(&Bprime[0],gauss,index1,index2,tria->FiniteElement());
|
---|
| 738 |
|
---|
| 739 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 740 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
| 741 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
| 742 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
| 743 | DL1=gauss->weight*Jdet*dt*UdotN/2;
|
---|
| 744 | DL2=gauss->weight*Jdet*dt*fabs(UdotN)/2;
|
---|
| 745 |
|
---|
| 746 | TripleMultiply(&B[0],1,numdof,1,
|
---|
| 747 | &DL1,1,1,0,
|
---|
| 748 | &Bprime[0],1,numdof,0,
|
---|
| 749 | &Ke_g1[0][0],0);
|
---|
| 750 | TripleMultiply(&B[0],1,numdof,1,
|
---|
| 751 | &DL2,1,1,0,
|
---|
| 752 | &B[0],1,numdof,0,
|
---|
| 753 | &Ke_g2[0][0],0);
|
---|
| 754 |
|
---|
| 755 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g1[i][j];
|
---|
| 756 | for(i=0;i<numdof;i++) for(j=0;j<numdof;j++) Ke->values[i*numdof+j]+=Ke_g2[i][j];
|
---|
[6026] | 757 | }
|
---|
| 758 |
|
---|
[19105] | 759 | /*Clean up and return*/
|
---|
| 760 | delete gauss;
|
---|
[6029] | 761 | return Ke;
|
---|
[6026] | 762 | }
|
---|
| 763 | /*}}}*/
|
---|
[19105] | 764 | ElementVector* Numericalflux::CreatePVectorAdjointBalancethickness(void){/*{{{*/
|
---|
[6026] | 765 |
|
---|
[19105] | 766 | /*No PVector for the Adjoint*/
|
---|
| 767 | return NULL;
|
---|
[6026] | 768 | }
|
---|
| 769 | /*}}}*/
|
---|
[19105] | 770 | ElementVector* Numericalflux::CreatePVectorBalancethickness(void){/*{{{*/
|
---|
[6026] | 771 |
|
---|
[17806] | 772 | switch(this->flux_type){
|
---|
[6026] | 773 | case InternalEnum:
|
---|
[19105] | 774 | return CreatePVectorBalancethicknessInternal();
|
---|
[6026] | 775 | case BoundaryEnum:
|
---|
[19105] | 776 | return CreatePVectorBalancethicknessBoundary();
|
---|
[6026] | 777 | default:
|
---|
[13036] | 778 | _error_("type not supported yet");
|
---|
[6026] | 779 | }
|
---|
| 780 | }
|
---|
| 781 | /*}}}*/
|
---|
[19105] | 782 | ElementVector* Numericalflux::CreatePVectorBalancethicknessBoundary(void){/*{{{*/
|
---|
[6026] | 783 |
|
---|
[5739] | 784 | /* constants*/
|
---|
[14763] | 785 | const int numdof=NDOF1*NUMNODES_BOUNDARY;
|
---|
[3683] | 786 |
|
---|
[5739] | 787 | /* Intermediaries*/
|
---|
[13761] | 788 | int i,ig,index1,index2;
|
---|
[19105] | 789 | IssmDouble DL,Jdet,vx,vy,mean_vx,mean_vy,UdotN,thickness;
|
---|
| 790 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
| 791 | IssmDouble normal[2];
|
---|
| 792 | IssmDouble L[numdof];
|
---|
[5739] | 793 | GaussTria *gauss;
|
---|
[3683] | 794 |
|
---|
[6026] | 795 | /*Initialize Load Vector and return if necessary*/
|
---|
[6029] | 796 | ElementVector* pe = NULL;
|
---|
[5911] | 797 | Tria* tria=(Tria*)element;
|
---|
[17806] | 798 | if(!tria->IsIceInElement()) return NULL;
|
---|
[5911] | 799 |
|
---|
| 800 | /*Retrieve all inputs and parameters*/
|
---|
[14763] | 801 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
[19105] | 802 | Input* vxaverage_input=tria->inputs->GetInput(VxEnum); _assert_(vxaverage_input);
|
---|
| 803 | Input* vyaverage_input=tria->inputs->GetInput(VyEnum); _assert_(vyaverage_input);
|
---|
| 804 | Input* thickness_input=tria->inputs->GetInput(ThicknessEnum); _assert_(thickness_input);
|
---|
[6026] | 805 | GetNormal(&normal[0],xyz_list);
|
---|
| 806 |
|
---|
| 807 | /*Check wether it is an inflow or outflow BC (0 is the middle of the segment)*/
|
---|
| 808 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 809 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 810 |
|
---|
| 811 | gauss=new GaussTria();
|
---|
| 812 | gauss->GaussEdgeCenter(index1,index2);
|
---|
[10135] | 813 | vxaverage_input->GetInputValue(&mean_vx,gauss);
|
---|
| 814 | vyaverage_input->GetInputValue(&mean_vy,gauss);
|
---|
[6026] | 815 | delete gauss;
|
---|
| 816 | UdotN=mean_vx*normal[0]+mean_vy*normal[1];
|
---|
| 817 | if (UdotN>0){
|
---|
[6029] | 818 | return NULL; /*(u,n)>0 -> outflow, PenaltyCreateKMatrix will take care of it*/
|
---|
[6026] | 819 | }
|
---|
[6029] | 820 | else{
|
---|
[14763] | 821 | pe=new ElementVector(nodes,NUMNODES_BOUNDARY,this->parameters);
|
---|
[6029] | 822 | }
|
---|
[6026] | 823 |
|
---|
| 824 | /* Start looping on the number of gaussian points: */
|
---|
| 825 | gauss=new GaussTria(index1,index2,2);
|
---|
| 826 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
| 827 |
|
---|
| 828 | gauss->GaussPoint(ig);
|
---|
| 829 |
|
---|
[18301] | 830 | tria->GetSegmentNodalFunctions(&L[0],gauss,index1,index2,tria->FiniteElement());
|
---|
[6026] | 831 |
|
---|
[10135] | 832 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 833 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
[19105] | 834 | thickness_input->GetInputValue(&thickness,gauss);
|
---|
[9232] | 835 |
|
---|
[6026] | 836 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
| 837 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
[19105] | 838 | DL= - gauss->weight*Jdet*UdotN*thickness;
|
---|
[6026] | 839 |
|
---|
| 840 | for(i=0;i<numdof;i++) pe->values[i] += DL*L[i];
|
---|
| 841 | }
|
---|
| 842 |
|
---|
| 843 | /*Clean up and return*/
|
---|
| 844 | delete gauss;
|
---|
| 845 | return pe;
|
---|
| 846 | }
|
---|
| 847 | /*}}}*/
|
---|
[19105] | 848 | ElementVector* Numericalflux::CreatePVectorBalancethicknessInternal(void){/*{{{*/
|
---|
[6026] | 849 |
|
---|
[19105] | 850 | /*Nothing added to PVector*/
|
---|
| 851 | return NULL;
|
---|
| 852 |
|
---|
| 853 | }
|
---|
| 854 | /*}}}*/
|
---|
| 855 | ElementVector* Numericalflux::CreatePVectorMasstransport(void){/*{{{*/
|
---|
| 856 |
|
---|
[17806] | 857 | switch(this->flux_type){
|
---|
[6026] | 858 | case InternalEnum:
|
---|
[19105] | 859 | return CreatePVectorMasstransportInternal();
|
---|
[6026] | 860 | case BoundaryEnum:
|
---|
[19105] | 861 | return CreatePVectorMasstransportBoundary();
|
---|
[6026] | 862 | default:
|
---|
[13036] | 863 | _error_("type not supported yet");
|
---|
[6026] | 864 | }
|
---|
| 865 | }
|
---|
| 866 | /*}}}*/
|
---|
[19105] | 867 | ElementVector* Numericalflux::CreatePVectorMasstransportBoundary(void){/*{{{*/
|
---|
[6026] | 868 |
|
---|
| 869 | /* constants*/
|
---|
[14763] | 870 | const int numdof=NDOF1*NUMNODES_BOUNDARY;
|
---|
[6026] | 871 |
|
---|
| 872 | /* Intermediaries*/
|
---|
[13761] | 873 | int i,ig,index1,index2;
|
---|
[19105] | 874 | IssmDouble DL,Jdet,dt,vx,vy,mean_vx,mean_vy,UdotN,thickness;
|
---|
| 875 | IssmDouble xyz_list[NUMVERTICES][3];
|
---|
| 876 | IssmDouble normal[2];
|
---|
| 877 | IssmDouble L[numdof];
|
---|
[6026] | 878 | GaussTria *gauss;
|
---|
| 879 |
|
---|
| 880 | /*Initialize Load Vector and return if necessary*/
|
---|
[6029] | 881 | ElementVector* pe = NULL;
|
---|
[6026] | 882 | Tria* tria=(Tria*)element;
|
---|
[17806] | 883 | if(!tria->IsIceInElement()) return NULL;
|
---|
[6026] | 884 |
|
---|
| 885 | /*Retrieve all inputs and parameters*/
|
---|
[14763] | 886 | GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES);
|
---|
[19105] | 887 | parameters->FindParam(&dt,TimesteppingTimeStepEnum);
|
---|
| 888 | Input* vxaverage_input =tria->inputs->GetInput(VxEnum); _assert_(vxaverage_input);
|
---|
| 889 | Input* vyaverage_input =tria->inputs->GetInput(VyEnum); _assert_(vyaverage_input);
|
---|
| 890 | Input* spcthickness_input=tria->inputs->GetInput(MasstransportSpcthicknessEnum); _assert_(spcthickness_input);
|
---|
[6026] | 891 | GetNormal(&normal[0],xyz_list);
|
---|
[3683] | 892 |
|
---|
[6026] | 893 | /*Check wether it is an inflow or outflow BC (0 is the middle of the segment)*/
|
---|
| 894 | index1=tria->GetNodeIndex(nodes[0]);
|
---|
| 895 | index2=tria->GetNodeIndex(nodes[1]);
|
---|
| 896 |
|
---|
| 897 | gauss=new GaussTria();
|
---|
| 898 | gauss->GaussEdgeCenter(index1,index2);
|
---|
[10135] | 899 | vxaverage_input->GetInputValue(&mean_vx,gauss);
|
---|
| 900 | vyaverage_input->GetInputValue(&mean_vy,gauss);
|
---|
[6026] | 901 | delete gauss;
|
---|
[19105] | 902 |
|
---|
[6026] | 903 | UdotN=mean_vx*normal[0]+mean_vy*normal[1];
|
---|
| 904 | if (UdotN>0){
|
---|
[6029] | 905 | return NULL; /*(u,n)>0 -> outflow, PenaltyCreateKMatrix will take care of it*/
|
---|
[6026] | 906 | }
|
---|
[6029] | 907 | else{
|
---|
[14763] | 908 | pe=new ElementVector(nodes,NUMNODES_BOUNDARY,this->parameters);
|
---|
[6029] | 909 | }
|
---|
[6026] | 910 |
|
---|
| 911 | /* Start looping on the number of gaussian points: */
|
---|
| 912 | gauss=new GaussTria(index1,index2,2);
|
---|
| 913 | for(ig=gauss->begin();ig<gauss->end();ig++){
|
---|
| 914 |
|
---|
| 915 | gauss->GaussPoint(ig);
|
---|
| 916 |
|
---|
[18301] | 917 | tria->GetSegmentNodalFunctions(&L[0],gauss,index1,index2,tria->FiniteElement());
|
---|
[6026] | 918 |
|
---|
[10135] | 919 | vxaverage_input->GetInputValue(&vx,gauss);
|
---|
| 920 | vyaverage_input->GetInputValue(&vy,gauss);
|
---|
[19105] | 921 | spcthickness_input->GetInputValue(&thickness,gauss);
|
---|
| 922 | if(xIsNan<IssmDouble>(thickness)) _error_("Cannot weakly apply constraint because NaN was provided");
|
---|
[9232] | 923 |
|
---|
[6026] | 924 | UdotN=vx*normal[0]+vy*normal[1];
|
---|
| 925 | tria->GetSegmentJacobianDeterminant(&Jdet,&xyz_list[0][0],gauss);
|
---|
[19105] | 926 | DL= - gauss->weight*Jdet*dt*UdotN*thickness;
|
---|
[6026] | 927 |
|
---|
| 928 | for(i=0;i<numdof;i++) pe->values[i] += DL*L[i];
|
---|
| 929 | }
|
---|
| 930 |
|
---|
| 931 | /*Clean up and return*/
|
---|
| 932 | delete gauss;
|
---|
| 933 | return pe;
|
---|
| 934 | }
|
---|
| 935 | /*}}}*/
|
---|
[19105] | 936 | ElementVector* Numericalflux::CreatePVectorMasstransportInternal(void){/*{{{*/
|
---|
[6026] | 937 |
|
---|
[19105] | 938 | /*Nothing added to PVector*/
|
---|
[6026] | 939 | return NULL;
|
---|
[19105] | 940 |
|
---|
[6026] | 941 | }
|
---|
| 942 | /*}}}*/
|
---|
[19105] | 943 | void Numericalflux::GetNormal(IssmDouble* normal,IssmDouble xyz_list[4][3]){/*{{{*/
|
---|
[3683] | 944 |
|
---|
| 945 | /*Build unit outward pointing vector*/
|
---|
[12472] | 946 | IssmDouble vector[2];
|
---|
| 947 | IssmDouble norm;
|
---|
[3683] | 948 |
|
---|
| 949 | vector[0]=xyz_list[1][0] - xyz_list[0][0];
|
---|
| 950 | vector[1]=xyz_list[1][1] - xyz_list[0][1];
|
---|
| 951 |
|
---|
| 952 | norm=sqrt(pow(vector[0],2.0)+pow(vector[1],2.0));
|
---|
| 953 |
|
---|
| 954 | normal[0]= + vector[1]/norm;
|
---|
| 955 | normal[1]= - vector[0]/norm;
|
---|
| 956 | }
|
---|
| 957 | /*}}}*/
|
---|