Changeset 16560 for issm/trunk/src/c/classes/Elements/Penta.cpp
- Timestamp:
- 10/28/13 14:43:03 (11 years ago)
- Location:
- issm/trunk
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
issm/trunk
- Property svn:mergeinfo changed
/issm/trunk-jpl merged: 16138-16453,16455-16554
- Property svn:mergeinfo changed
-
issm/trunk/src
- Property svn:mergeinfo changed
-
issm/trunk/src/c/classes/Elements/Penta.cpp
r16137 r16560 30 30 this->inputs = NULL; 31 31 this->parameters = NULL; 32 this->results = NULL;33 32 } 34 33 /*}}}*/ … … 36 35 Penta::~Penta(){ 37 36 delete inputs; 38 delete results;39 37 this->parameters=NULL; 40 38 } … … 43 41 Penta::Penta(int penta_id, int penta_sid, int index, IoModel* iomodel,int nummodels) 44 42 :PentaRef(nummodels) 45 ,ElementHook(nummodels,index+1,6,iomodel) //index+1: material id, iomodel->numberofelements+1: matpar id 46 { //i is the element index 47 48 int i; 43 ,ElementHook(nummodels,index+1,6,iomodel){ 44 49 45 int penta_elements_ids[2]; 50 46 51 47 /*Checks in debugging mode*/ 52 /*{{{*/53 48 _assert_(iomodel->Data(MeshUpperelementsEnum)); 54 49 _assert_(iomodel->Data(MeshLowerelementsEnum)); 55 /*}}}*/56 50 57 51 /*id: */ … … 69 63 this->parameters=NULL; 70 64 71 /*intialize inputs and results: */65 /*intialize inputs: */ 72 66 this->inputs=new Inputs(); 73 this->results=new Results();74 67 75 68 /*initialize pointers:*/ … … 112 105 penta->inputs=new Inputs(); 113 106 } 114 if(this->results){115 penta->results=(Results*)this->results->Copy();116 }117 else{118 penta->results=new Results();119 }120 107 /*point parameters: */ 121 108 penta->parameters=this->parameters; … … 161 148 void Penta::BasalFrictionCreateInput(void){ 162 149 163 /*Constants*/164 const int numdof=NUMVERTICES*NDOF1;165 166 150 /*Intermediaries */ 167 int count;168 IssmDouble basalfriction[NUMVERTICES]={0,0,0,0,0,0};169 IssmDouble alpha2,vx,vy;170 Friction * friction=NULL;171 GaussPenta * gauss=NULL;151 int count; 152 IssmDouble basalfriction[NUMVERTICES]; 153 IssmDouble alpha2 ,vx,vy; 154 Friction *friction = NULL; 155 GaussPenta *gauss = NULL; 172 156 173 157 /* Basal friction can only be found at the base of an ice sheet: */ … … 437 421 438 422 /*retrieve parameters: */ 439 ElementMatrix* Ke=NULL;440 423 int analysis_type; 441 424 parameters->FindParam(&analysis_type,AnalysisTypeEnum); … … 464 447 break; 465 448 #endif 466 case BedSlopeXAnalysisEnum: case SurfaceSlopeXAnalysisEnum: case BedSlopeYAnalysisEnum: case SurfaceSlopeYAnalysisEnum:449 case L2ProjectionBaseAnalysisEnum: 467 450 return CreateBasalMassMatrix(); 468 451 break; … … 501 484 #endif 502 485 default: 503 _error_("analysis " << analysis_type << " (" << EnumToStringx(analysis_type) << ")not supported yet");486 _error_("analysis " << EnumToStringx(analysis_type) << " not supported yet"); 504 487 } 505 488 … … 572 555 573 556 /*retrieve parameters: */ 574 ElementMatrix* Ke=NULL;575 557 ElementVector* De=NULL; 576 558 int analysis_type; 577 559 parameters->FindParam(&analysis_type,AnalysisTypeEnum); 578 560 579 /*Checks in debugging {{{*/561 /*Checks in debugging*/ 580 562 _assert_(this->nodes && this->material && this->matpar && this->verticalneighbors && this->parameters && this->inputs); 581 /*}}}*/582 563 583 564 switch(analysis_type){ … … 692 673 break; 693 674 #endif 694 case BedSlopeXAnalysisEnum: case SurfaceSlopeXAnalysisEnum: case BedSlopeYAnalysisEnum: case SurfaceSlopeYAnalysisEnum:695 return CreatePVector Slope();675 case L2ProjectionBaseAnalysisEnum: 676 return CreatePVectorL2ProjectionBase(); 696 677 break; 697 678 case MasstransportAnalysisEnum: … … 775 756 } 776 757 /*}}}*/ 777 /*FUNCTION Penta::CreatePVector Slope {{{*/778 ElementVector* Penta::CreatePVector Slope(void){758 /*FUNCTION Penta::CreatePVectorL2ProjectionBase {{{*/ 759 ElementVector* Penta::CreatePVectorL2ProjectionBase(void){ 779 760 780 761 if (!IsOnBed()) return NULL; … … 782 763 /*Call Tria function*/ 783 764 Tria* tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 784 ElementVector* pe=tria->CreatePVector Slope();765 ElementVector* pe=tria->CreatePVectorL2Projection(); 785 766 delete tria->material; delete tria; 786 767 … … 855 836 _printf_(" inputs\n"); 856 837 inputs->DeepEcho(); 857 _printf_(" results\n");858 results->DeepEcho();859 }860 /*}}}*/861 /*FUNCTION Penta::DeleteResults {{{*/862 void Penta::DeleteResults(void){863 864 /*Delete and reinitialize results*/865 delete this->results;866 this->results=new Results();867 868 838 } 869 839 /*}}}*/ … … 929 899 NewPrecipitationInput->AddTimeInput(newmonthinput2,time+imonth/12.*yts); 930 900 } 901 NewTemperatureInput->Configure(this->parameters); 902 NewPrecipitationInput->Configure(this->parameters); 931 903 932 904 this->inputs->AddInput(NewTemperatureInput); … … 1417 1389 /*Recover input*/ 1418 1390 Input* input=inputs->GetInput(enumtype); 1419 if (!input) _error_("Input " << EnumToStringx(enumtype) << " not found in element"); 1420 int numnodes = this->NumberofNodes(); 1391 if(!input) _error_("Input " << EnumToStringx(enumtype) << " not found in element"); 1421 1392 1422 1393 /* Start looping on the number of vertices: */ 1423 1394 GaussPenta* gauss=new GaussPenta(); 1424 for 1395 for(int iv=0;iv<this->NumberofNodes();iv++){ 1425 1396 gauss->GaussNode(this->element_type,iv); 1426 1397 input->GetInputValue(&pvalue[iv],gauss); … … 1433 1404 1434 1405 Input* input=inputs->GetInput(enumtype); 1406 if(!input) _error_("No input of type " << EnumToStringx(enumtype) << " found in tria"); 1407 1408 GaussPenta* gauss=new GaussPenta(); 1409 gauss->GaussVertex(this->GetNodeIndex(node)); 1410 1411 input->GetInputValue(pvalue,gauss); 1412 delete gauss; 1413 } 1414 /*}}}*/ 1415 /*FUNCTION Penta::GetMaterialInputValue(IssmDouble* pvalue,Node* node,int enumtype) {{{*/ 1416 void Penta::GetMaterialInputValue(IssmDouble* pvalue,Node* node,int enumtype){ 1417 1418 Input* input=this->material->inputs->GetInput(enumtype); 1435 1419 if(!input) _error_("No input of type " << EnumToStringx(enumtype) << " found in tria"); 1436 1420 … … 1529 1513 break; 1530 1514 case StressbalanceVerticalAnalysisEnum: 1531 //GetSolutionFromInputsStressbalanceVert(solution);1532 1515 GetSolutionFromInputsOneDof(solution, VzEnum); 1533 1516 break; … … 1535 1518 #ifdef _HAVE_THERMAL_ 1536 1519 case ThermalAnalysisEnum: 1537 //GetSolutionFromInputsThermal(solution);1538 1520 GetSolutionFromInputsOneDof(solution, TemperatureEnum); 1539 1521 break; 1540 1522 case EnthalpyAnalysisEnum: 1541 //GetSolutionFromInputsEnthalpy(solution);1542 1523 GetSolutionFromInputsOneDof(solution, EnthalpyEnum); 1543 1524 break; … … 1657 1638 } 1658 1639 /*}}}*/ 1659 /*FUNCTION Penta::GetVectorFromResults{{{*/1660 void Penta::GetVectorFromResults(Vector<IssmDouble>* vector,int offset,int enum_in,int interp){1661 1662 /*Get result*/1663 ElementResult* elementresult=(ElementResult*)this->results->GetObjectByOffset(offset);1664 if(elementresult->InstanceEnum()!=enum_in){1665 _error_("Results of offset "<<offset<<" is "<<EnumToStringx(elementresult->InstanceEnum())<<" when "<<EnumToStringx(enum_in)<<" was expected");1666 }1667 if(interp==P1Enum){1668 int vertexpidlist[NUMVERTICES];1669 int connectivity[NUMVERTICES];1670 this->GetVertexSidList(&vertexpidlist[0]);1671 this->GetConnectivityList(&connectivity[0]);1672 elementresult->GetVectorFromResults(vector,&vertexpidlist[0],&connectivity[0],NUMVERTICES);1673 }1674 else if(interp==P0Enum){1675 elementresult->GetElementVectorFromResults(vector,sid);1676 }1677 else{1678 _printf_("Interpolation " << EnumToStringx(interp) << " not supported\n");1679 }1680 }1681 /*}}}*/1682 1640 /*FUNCTION Penta::GetZcoord {{{*/ 1683 1641 IssmDouble Penta::GetZcoord(GaussPenta* gauss){ … … 1699 1657 /*Computeportion of the element that is grounded*/ 1700 1658 1701 int normal_orientation ;1659 int normal_orientation=0; 1702 1660 IssmDouble s1,s2; 1703 1661 IssmDouble levelset[NUMVERTICES]; … … 1706 1664 GetInputListOnVertices(&levelset[0],levelsetenum); 1707 1665 1708 if(levelset[0]*levelset[1]>0 ){ //Nodes 0 and 1 are similar, so points must be found on segment 0-2 and 1-21666 if(levelset[0]*levelset[1]>0.){ //Nodes 0 and 1 are similar, so points must be found on segment 0-2 and 1-2 1709 1667 /*Portion of the segments*/ 1710 1668 s1=levelset[2]/(levelset[2]-levelset[1]); 1711 1669 s2=levelset[2]/(levelset[2]-levelset[0]); 1712 1670 1713 if(levelset[2]>0 ) normal_orientation=1;1671 if(levelset[2]>0.) normal_orientation=1; 1714 1672 /*New point 1*/ 1715 1673 xyz_zero[3*normal_orientation+0]=xyz_list[2][0]+s1*(xyz_list[1][0]-xyz_list[2][0]); … … 1732 1690 xyz_zero[3*(2+normal_orientation)+2]=xyz_list[5][2]+s2*(xyz_list[3][2]-xyz_list[5][2]); 1733 1691 } 1734 else if(levelset[1]*levelset[2]>0 ){ //Nodes 1 and 2 are similar, so points must be found on segment 0-1 and 0-21692 else if(levelset[1]*levelset[2]>0.){ //Nodes 1 and 2 are similar, so points must be found on segment 0-1 and 0-2 1735 1693 /*Portion of the segments*/ 1736 1694 s1=levelset[0]/(levelset[0]-levelset[2]); 1737 1695 s2=levelset[0]/(levelset[0]-levelset[1]); 1738 1696 1739 if(levelset[0]>0 ) normal_orientation=1;1697 if(levelset[0]>0.) normal_orientation=1; 1740 1698 /*New point 1*/ 1741 1699 xyz_zero[3*normal_orientation+0]=xyz_list[0][0]+s1*(xyz_list[2][0]-xyz_list[0][0]); … … 1758 1716 xyz_zero[3*(2+normal_orientation)+2]=xyz_list[3][2]+s2*(xyz_list[4][2]-xyz_list[3][2]); 1759 1717 } 1760 else if(levelset[0]*levelset[2]>0 ){ //Nodes 0 and 2 are similar, so points must be found on segment 1-0 and 1-21718 else if(levelset[0]*levelset[2]>0.){ //Nodes 0 and 2 are similar, so points must be found on segment 1-0 and 1-2 1761 1719 /*Portion of the segments*/ 1762 1720 s1=levelset[1]/(levelset[1]-levelset[0]); 1763 1721 s2=levelset[1]/(levelset[1]-levelset[2]); 1764 1722 1765 if(levelset[1]>0 ) normal_orientation=1;1723 if(levelset[1]>0.) normal_orientation=1; 1766 1724 /*New point 0*/ 1767 1725 xyz_zero[3*normal_orientation+0]=xyz_list[1][0]+s1*(xyz_list[0][0]-xyz_list[1][0]); … … 1784 1742 xyz_zero[3*(2+normal_orientation)+2]=xyz_list[4][2]+s2*(xyz_list[5][2]-xyz_list[4][2]); 1785 1743 } 1786 else if(levelset[0]==0 && levelset[1]==0){ //front is on point 0 and 11744 else if(levelset[0]==0. && levelset[1]==0.){ //front is on point 0 and 1 1787 1745 xyz_zero[3*0+0]=xyz_list[0][0]; 1788 1746 xyz_zero[3*0+1]=xyz_list[0][1]; … … 1804 1762 xyz_zero[3*3+2]=xyz_list[3][2]; 1805 1763 } 1806 else if(levelset[0]==0 && levelset[2]==0){ //front is on point 0 and 11764 else if(levelset[0]==0. && levelset[2]==0.){ //front is on point 0 and 1 1807 1765 xyz_zero[3*0+0]=xyz_list[2][0]; 1808 1766 xyz_zero[3*0+1]=xyz_list[2][1]; … … 1824 1782 xyz_zero[3*3+2]=xyz_list[5][2]; 1825 1783 } 1826 else if(levelset[1]==0 && levelset[2]==0){ //front is on point 0 and 11784 else if(levelset[1]==0. && levelset[2]==0.){ //front is on point 0 and 1 1827 1785 xyz_zero[3*0+0]=xyz_list[1][0]; 1828 1786 xyz_zero[3*0+1]=xyz_list[1][1]; … … 1859 1817 } 1860 1818 /*}}}*/ 1861 /*FUNCTION Penta::InputArtificialNoise{{{*/ 1862 void Penta::InputArtificialNoise(int enum_type,IssmDouble min,IssmDouble max){ 1863 1864 Input* input=NULL; 1865 1866 /*Make a copy of the original input: */ 1867 input=(Input*)this->inputs->GetInput(enum_type); 1868 if(!input)_error_("could not find old input with enum: " << EnumToStringx(enum_type)); 1869 1870 /*ArtificialNoise: */ 1871 input->ArtificialNoise(min,max); 1872 } 1873 /*}}}*/ 1874 /*FUNCTION Penta::InputCreate(IssmDouble scalar,int enum,int code);{{{*/ 1875 void Penta::InputCreate(IssmDouble scalar,int name,int code){ 1876 1877 /*Check that name is an element input*/ 1878 if (!IsInput(name)) return; 1879 1880 if ((code==5) || (code==1)){ //boolean 1881 this->inputs->AddInput(new BoolInput(name,reCast<bool,IssmDouble>(scalar))); 1882 } 1883 else if ((code==6) || (code==2)){ //integer 1884 this->inputs->AddInput(new IntInput(name,reCast<int,IssmDouble>(scalar))); 1885 } 1886 else if ((code==7) || (code==3)){ //IssmDouble 1887 this->inputs->AddInput(new DoubleInput(name,scalar)); 1888 } 1889 else _error_("could not recognize nature of vector from code " << code); 1890 1891 } 1892 /*}}}*/ 1893 /*FUNCTION Penta::InputCreate(IssmDouble* vector,int index,IoModel* iomodel,int M,int N,int vector_type,int vector_enum,int code){{{*/ 1894 void Penta::InputCreate(IssmDouble* vector, int index,IoModel* iomodel,int M,int N,int vector_type,int vector_enum,int code){//index into elements 1819 /*FUNCTION Penta::InputCreate(IssmDouble* vector,IoModel* iomodel,int M,int N,int vector_type,int vector_enum,int code){{{*/ 1820 void Penta::InputCreate(IssmDouble* vector,IoModel* iomodel,int M,int N,int vector_type,int vector_enum,int code){ 1895 1821 1896 1822 /*Intermediaries*/ … … 1908 1834 for(i=0;i<6;i++){ 1909 1835 _assert_(iomodel->elements); 1910 penta_vertex_ids[i]=iomodel->elements[6* index+i]; //ids for vertices are in the elements array from Matlab1836 penta_vertex_ids[i]=iomodel->elements[6*this->sid+i]; //ids for vertices are in the elements array from Matlab 1911 1837 } 1912 1838 … … 1947 1873 1948 1874 if (code==5){ //boolean 1949 this->inputs->AddInput(new BoolInput(vector_enum,reCast<bool,IssmDouble>(vector[ index])));1875 this->inputs->AddInput(new BoolInput(vector_enum,reCast<bool,IssmDouble>(vector[this->sid]))); 1950 1876 } 1951 1877 else if (code==6){ //integer 1952 this->inputs->AddInput(new IntInput(vector_enum,reCast<int,IssmDouble>(vector[ index])));1878 this->inputs->AddInput(new IntInput(vector_enum,reCast<int,IssmDouble>(vector[this->sid]))); 1953 1879 } 1954 1880 else if (code==7){ //IssmDouble 1955 this->inputs->AddInput(new DoubleInput(vector_enum,vector[ index]));1881 this->inputs->AddInput(new DoubleInput(vector_enum,vector[this->sid])); 1956 1882 } 1957 1883 else _error_("could not recognize nature of vector from code " << code); … … 2147 2073 } 2148 2074 /*}}}*/ 2149 /*FUNCTION Penta::InputToResult{{{*/2150 void Penta::InputToResult(int enum_type,int step,IssmDouble time){2151 2152 bool found = false;2153 Input *input = NULL;2154 2155 /*Go through all the input objects, and find the one corresponding to enum_type, if it exists: */2156 if (enum_type==MaterialsRheologyBbarEnum) input=this->material->inputs->GetInput(MaterialsRheologyBEnum);2157 else if (enum_type==MaterialsRheologyZbarEnum) input=this->material->inputs->GetInput(MaterialsRheologyZEnum);2158 else input=this->inputs->GetInput(enum_type);2159 //if (!input) _error_("Input " << EnumToStringx(enum_type) << " not found in penta->inputs"); why error out? if the requested input does not exist, we should still2160 //try and output whatever we can instead of just failing.2161 if(!input)return;2162 2163 /*If we don't find it, no big deal, just don't do the transfer. Otherwise, build a new Result2164 * object out of the input, with the additional step and time information: */2165 this->results->AddResult(input->SpawnResult(step,time));2166 #ifdef _HAVE_CONTROL_2167 if(input->ObjectEnum()==ControlInputEnum){2168 if(((ControlInput*)input)->gradient!=NULL) this->results->AddResult(((ControlInput*)input)->SpawnGradient(step,time));2169 }2170 #endif2171 }2172 /*}}}*/2173 2075 /*FUNCTION Penta::InputUpdateFromConstant(bool value, int name);{{{*/ 2174 2076 void Penta::InputUpdateFromConstant(bool constant, int name){ … … 2211 2113 IssmDouble yts; 2212 2114 bool control_analysis; 2213 int num_control_type; 2214 int num_cm_responses; 2115 int num_control_type,num_responses; 2215 2116 2216 2117 /*Fetch parameters: */ … … 2218 2119 iomodel->Constant(&control_analysis,InversionIscontrolEnum); 2219 2120 if(control_analysis) iomodel->Constant(&num_control_type,InversionNumControlParametersEnum); 2220 if(control_analysis) iomodel->Constant(&num_ cm_responses,InversionNumCostFunctionsEnum);2121 if(control_analysis) iomodel->Constant(&num_responses,InversionNumCostFunctionsEnum); 2221 2122 2222 2123 /*Checks if debuging*/ … … 2267 2168 } 2268 2169 break; 2269 case MaterialsRheologyBbarEnum:2270 case MaterialsRheology ZbarEnum:2271 /*Material will take care of it*/break;2170 /*Material will take care of it*/ 2171 case MaterialsRheologyBbarEnum: break; 2172 case DamageDbarEnum:break; 2272 2173 default: 2273 2174 _error_("Control " << EnumToStringx(reCast<int,IssmDouble>(iomodel->Data(InversionControlParametersEnum)[i])) << " not implemented yet"); … … 2277 2178 #endif 2278 2179 2279 / /Need to know the type of approximation for this element2180 /*Need to know the type of approximation for this element*/ 2280 2181 if(iomodel->Data(FlowequationElementEquationEnum)){ 2281 if (iomodel->Data(FlowequationElementEquationEnum)[index]==SSAApproximationEnum){ 2282 this->inputs->AddInput(new IntInput(ApproximationEnum,SSAApproximationEnum)); 2283 } 2284 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==HOApproximationEnum){ 2285 this->inputs->AddInput(new IntInput(ApproximationEnum,HOApproximationEnum)); 2286 } 2287 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==SSAHOApproximationEnum){ 2288 this->inputs->AddInput(new IntInput(ApproximationEnum,SSAHOApproximationEnum)); 2289 } 2290 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==SIAApproximationEnum){ 2291 this->inputs->AddInput(new IntInput(ApproximationEnum,SIAApproximationEnum)); 2292 } 2293 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==L1L2ApproximationEnum){ 2294 this->inputs->AddInput(new IntInput(ApproximationEnum,L1L2ApproximationEnum)); 2295 } 2296 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==FSApproximationEnum){ 2297 this->inputs->AddInput(new IntInput(ApproximationEnum,FSApproximationEnum)); 2298 } 2299 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==SSAFSApproximationEnum){ 2300 this->inputs->AddInput(new IntInput(ApproximationEnum,SSAFSApproximationEnum)); 2301 } 2302 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==HOFSApproximationEnum){ 2303 this->inputs->AddInput(new IntInput(ApproximationEnum,HOFSApproximationEnum)); 2304 } 2305 else if (iomodel->Data(FlowequationElementEquationEnum)[index]==NoneApproximationEnum){ 2306 this->inputs->AddInput(new IntInput(ApproximationEnum,NoneApproximationEnum)); 2307 } 2308 else{ 2309 _error_("Approximation type " << EnumToStringx(reCast<int,IssmDouble>(iomodel->Data(FlowequationElementEquationEnum)[index])) << " not supported yet"); 2310 } 2182 this->inputs->AddInput(new IntInput(ApproximationEnum,reCast<int>(iomodel->Data(FlowequationElementEquationEnum)[index]))); 2311 2183 } 2312 2184 … … 2316 2188 /*Create inputs and add to DataSetInput*/ 2317 2189 DatasetInput* datasetinput=new DatasetInput(InversionCostFunctionsCoefficientsEnum); 2318 for(i=0;i<num_ cm_responses;i++){2319 for(j=0;j<6;j++)nodeinputs[j]=iomodel->Data(InversionCostFunctionsCoefficientsEnum)[(penta_vertex_ids[j]-1)*num_ cm_responses+i];2320 datasetinput-> inputs->AddObject(new PentaInput(InversionCostFunctionsCoefficientsEnum,nodeinputs,P1Enum));2190 for(i=0;i<num_responses;i++){ 2191 for(j=0;j<6;j++)nodeinputs[j]=iomodel->Data(InversionCostFunctionsCoefficientsEnum)[(penta_vertex_ids[j]-1)*num_responses+i]; 2192 datasetinput->AddInput(new PentaInput(InversionCostFunctionsCoefficientsEnum,nodeinputs,P1Enum),reCast<int>(iomodel->Data(InversionCostFunctionsEnum)[i])); 2321 2193 } 2322 2194 … … 2329 2201 void Penta::InputUpdateFromSolution(IssmDouble* solution){ 2330 2202 2331 int analysis_type ;2203 int analysis_type,inputenum; 2332 2204 2333 2205 /*retreive parameters: */ … … 2370 2242 break; 2371 2243 #endif 2372 case BedSlopeXAnalysisEnum: 2373 InputUpdateFromSolutionOneDofCollapsed(solution,BedSlopeXEnum); 2374 break; 2375 case BedSlopeYAnalysisEnum: 2376 InputUpdateFromSolutionOneDofCollapsed(solution,BedSlopeYEnum); 2377 break; 2378 case SurfaceSlopeXAnalysisEnum: 2379 InputUpdateFromSolutionOneDofCollapsed(solution,SurfaceSlopeXEnum); 2380 break; 2381 case SurfaceSlopeYAnalysisEnum: 2382 InputUpdateFromSolutionOneDofCollapsed(solution,SurfaceSlopeYEnum); 2244 case L2ProjectionBaseAnalysisEnum: 2245 this->parameters->FindParam(&inputenum,InputToL2ProjectEnum); 2246 InputUpdateFromSolutionOneDofCollapsed(solution,inputenum); 2383 2247 break; 2384 2248 case MasstransportAnalysisEnum: … … 2676 2540 } 2677 2541 /*update input*/ 2678 if (name==MaterialsRheologyZEnum || name==MaterialsRheologyZbarEnum){ 2679 material->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2680 } 2681 else{ 2682 this->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2683 } 2542 this->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2684 2543 return; 2685 2544 … … 2689 2548 } 2690 2549 /*update input*/ 2691 if (name==MaterialsRheologyZEnum || name==MaterialsRheologyZbarEnum){ 2692 material->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2693 } 2694 else{ 2695 this->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2696 } 2550 this->inputs->AddInput(new PentaInput(name,values,P1Enum)); 2697 2551 return; 2698 2552 … … 2744 2598 name==SurfaceEnum || 2745 2599 name==BedEnum || 2600 name==BathymetryEnum || 2746 2601 name==SurfaceSlopeXEnum || 2747 2602 name==SurfaceSlopeYEnum || … … 2782 2637 name==QmuTemperatureEnum || 2783 2638 name==QmuMeltingEnum || 2639 name==QmuMaskGroundediceLevelsetEnum || 2640 name==QmuMaskIceLevelsetEnum || 2784 2641 name==GiaWEnum || 2785 2642 name==GiadWdtEnum || … … 2854 2711 } 2855 2712 /*}}}*/ 2856 /*FUNCTION Penta::ListResultsInfo{{{*/2857 void Penta::ListResultsInfo(int** in_resultsenums,int** in_resultssizes,IssmDouble** in_resultstimes,int** in_resultssteps,int* in_num_results){2858 2859 /*Intermediaries*/2860 int i;2861 int numberofresults = 0;2862 int *resultsenums = NULL;2863 int *resultssizes = NULL;2864 IssmDouble *resultstimes = NULL;2865 int *resultssteps = NULL;2866 2867 /*Checks*/2868 _assert_(in_num_results);2869 2870 /*Count number of results*/2871 for(i=0;i<this->results->Size();i++){2872 ElementResult* elementresult=(ElementResult*)this->results->GetObjectByOffset(i);2873 numberofresults++;2874 }2875 2876 if(numberofresults){2877 2878 /*Allocate output*/2879 resultsenums=xNew<int>(numberofresults);2880 resultssizes=xNew<int>(numberofresults);2881 resultstimes=xNew<IssmDouble>(numberofresults);2882 resultssteps=xNew<int>(numberofresults);2883 2884 /*populate enums*/2885 for(i=0;i<this->results->Size();i++){2886 ElementResult* elementresult=(ElementResult*)this->results->GetObjectByOffset(i);2887 resultsenums[i]=elementresult->InstanceEnum();2888 resultstimes[i]=elementresult->GetTime();2889 resultssteps[i]=elementresult->GetStep();2890 if(elementresult->ObjectEnum()==PentaP1ElementResultEnum){2891 resultssizes[i]=P1Enum;2892 }2893 else{2894 resultssizes[i]=P0Enum;2895 }2896 }2897 }2898 2899 /*Assign output pointers:*/2900 *in_num_results=numberofresults;2901 *in_resultsenums=resultsenums;2902 *in_resultssizes=resultssizes;2903 *in_resultstimes=resultstimes;2904 *in_resultssteps=resultssteps;2905 2906 }/*}}}*/2907 2713 /*FUNCTION Penta::MinEdgeLength{{{*/ 2908 2714 IssmDouble Penta::MinEdgeLength(IssmDouble xyz_list[6][3]){ … … 2957 2763 if(found)*pvalue=value; 2958 2764 return found; 2959 }2960 /*}}}*/2961 /*FUNCTION Penta::PatchFill{{{*/2962 void Penta::PatchFill(int* pcount, Patch* patch){2963 2964 int i,count;2965 int vertices_ids[6];2966 2967 /*recover pointer: */2968 count=*pcount;2969 2970 /*will be needed later: */2971 for(i=0;i<6;i++) vertices_ids[i]=vertices[i]->Id(); //vertices id start at column 3 of the patch.2972 2973 for(i=0;i<this->results->Size();i++){2974 ElementResult* elementresult=(ElementResult*)this->results->GetObjectByOffset(i);2975 2976 /*For this result,fill the information in the Patch object (element id + vertices ids), and then hand2977 *it to the result object, to fill the rest: */2978 patch->fillelementinfo(count,this->sid+1,vertices_ids,6);2979 elementresult->PatchFill(count,patch);2980 2981 /*increment counter: */2982 count++;2983 }2984 2985 /*Assign output pointers:*/2986 *pcount=count;2987 }/*}}}*/2988 /*FUNCTION Penta::PatchSize{{{*/2989 void Penta::PatchSize(int* pnumrows, int* pnumvertices,int* pnumnodes){2990 2991 int i;2992 int numrows = 0;2993 int numnodes = 0;2994 int temp_numnodes = 0;2995 2996 /*Go through all the results objects, and update the counters: */2997 for (i=0;i<this->results->Size();i++){2998 ElementResult* elementresult=(ElementResult*)this->results->GetObjectByOffset(i);2999 /*first, we have one more result: */3000 numrows++;3001 /*now, how many vertices and how many nodal values for this result? :*/3002 temp_numnodes=elementresult->NumberOfNodalValues(); //ask result object.3003 if(temp_numnodes>numnodes)numnodes=temp_numnodes;3004 }3005 3006 /*Assign output pointers:*/3007 *pnumrows=numrows;3008 *pnumvertices=NUMVERTICES;3009 *pnumnodes=numnodes;3010 2765 } 3011 2766 /*}}}*/ … … 3062 2817 } 3063 2818 /*}}}*/ 3064 /*FUNCTION Penta::ReduceMatrixFS {{{*/ 3065 void Penta::ReduceMatrixFS(IssmDouble* Ke_reduced, IssmDouble* Ke_temp){ 3066 3067 int i,j; 3068 IssmDouble Kii[24][24]; 3069 IssmDouble Kib[24][3]; 3070 IssmDouble Kbb[3][3]; 3071 IssmDouble Kbi[3][24]; 3072 IssmDouble Kbbinv[3][3]; 3073 IssmDouble Kright[24][24]; 3074 3075 /*Create the four matrices used for reduction */ 3076 for(i=0;i<24;i++){ 3077 for(j=0;j<24;j++){ 3078 Kii[i][j]=*(Ke_temp+27*i+j); 3079 } 3080 for(j=0;j<3;j++){ 3081 Kib[i][j]=*(Ke_temp+27*i+j+24); 3082 } 3083 } 3084 for(i=0;i<3;i++){ 3085 for(j=0;j<24;j++){ 3086 Kbi[i][j]=*(Ke_temp+27*(i+24)+j); 3087 } 3088 for(j=0;j<3;j++){ 3089 Kbb[i][j]=*(Ke_temp+27*(i+24)+j+24); 3090 } 3091 } 3092 3093 /*Inverse the matrix corresponding to bubble part Kbb */ 3094 Matrix3x3Invert(&Kbbinv[0][0], &Kbb[0][0]); 3095 3096 /*Multiply matrices to create the reduce matrix Ke_reduced */ 3097 TripleMultiply(&Kib[0][0],24,3,0, 3098 &Kbbinv[0][0],3,3,0, 3099 &Kbi[0][0],3,24,0, 3100 &Kright[0][0],0); 3101 3102 /*Affect value to the reduced matrix */ 3103 for(i=0;i<24;i++) for(j=0;j<24;j++) *(Ke_reduced+24*i+j)=Kii[i][j]-Kright[i][j]; 3104 } 3105 /*}}}*/ 3106 /*FUNCTION Penta::ReduceVectorFS {{{*/ 3107 void Penta::ReduceVectorFS(IssmDouble* Pe_reduced, IssmDouble* Ke_temp, IssmDouble* Pe_temp){ 3108 3109 int i,j; 3110 IssmDouble Pi[24]; 3111 IssmDouble Pb[3]; 3112 IssmDouble Kbb[3][3]; 3113 IssmDouble Kib[24][3]; 3114 IssmDouble Kbbinv[3][3]; 3115 IssmDouble Pright[24]; 3116 3117 /*Create the four matrices used for reduction */ 3118 for(i=0;i<24;i++) Pi[i]=*(Pe_temp+i); 3119 for(i=0;i<3;i++) Pb[i]=*(Pe_temp+i+24); 3120 for(j=0;j<3;j++){ 3121 for(i=0;i<24;i++){ 3122 Kib[i][j]=*(Ke_temp+3*i+j); 3123 } 3124 for(i=0;i<3;i++){ 3125 Kbb[i][j]=*(Ke_temp+3*(i+24)+j); 3126 } 3127 } 3128 3129 /*Inverse the matrix corresponding to bubble part Kbb */ 3130 Matrix3x3Invert(&Kbbinv[0][0], &Kbb[0][0]); 3131 3132 /*Multiply matrices to create the reduce matrix Ke_reduced */ 3133 TripleMultiply(&Kib[0][0],24,3,0, 3134 &Kbbinv[0][0],3,3,0, 3135 &Pb[0],3,1,0,&Pright[0],0); 3136 3137 /*Affect value to the reduced matrix */ 3138 for(i=0;i<24;i++) *(Pe_reduced+i)=Pi[i]-Pright[i]; 3139 } 3140 /*}}}*/ 3141 /*FUNCTION Penta::RequestedOutput{{{*/ 3142 void Penta::RequestedOutput(int output_enum,int step,IssmDouble time){ 3143 if(IsInput(output_enum)){ 3144 /*just transfer this input to results, and we are done: */ 3145 InputToResult(output_enum,step,time); 3146 } 3147 else{ 3148 /*this input does not exist, compute it, and then transfer to results: */ 2819 /*FUNCTION Penta::ResultInterpolation{{{*/ 2820 void Penta::ResultInterpolation(int* pinterpolation,int output_enum){ 2821 2822 Input* input=this->inputs->GetInput(output_enum); 2823 2824 /*If this input is not already in Inputs, maybe it needs to be computed?*/ 2825 if(!input){ 3149 2826 switch(output_enum){ 3150 case BasalFrictionEnum: 3151 3152 /*create input: */ 3153 BasalFrictionCreateInput(); 3154 3155 /*transfer to results :*/ 3156 InputToResult(output_enum,step,time); 3157 3158 /*erase input: */ 3159 inputs->DeleteInput(output_enum); 2827 case ViscousHeatingEnum: 2828 this->ViscousHeatingCreateInput(); 2829 input=this->inputs->GetInput(output_enum); 3160 2830 break; 3161 case ViscousHeatingEnum: 3162 3163 /*create input: */ 3164 ViscousHeatingCreateInput(); 3165 3166 /*transfer to results :*/ 3167 InputToResult(output_enum,step,time); 3168 3169 /*erase input: */ 3170 inputs->DeleteInput(output_enum); 2831 case StressTensorxxEnum: 2832 this->ComputeStressTensor(); 2833 input=this->inputs->GetInput(output_enum); 3171 2834 break; 3172 3173 case StressTensorEnum: 2835 case StressTensorxyEnum: 3174 2836 this->ComputeStressTensor(); 3175 InputToResult(StressTensorxxEnum,step,time); 3176 InputToResult(StressTensorxyEnum,step,time); 3177 InputToResult(StressTensorxzEnum,step,time); 3178 InputToResult(StressTensoryyEnum,step,time); 3179 InputToResult(StressTensoryzEnum,step,time); 3180 InputToResult(StressTensorzzEnum,step,time); 2837 input=this->inputs->GetInput(output_enum); 3181 2838 break; 3182 2839 case StressTensorxzEnum: 2840 this->ComputeStressTensor(); 2841 input=this->inputs->GetInput(output_enum); 2842 break; 2843 case StressTensoryyEnum: 2844 this->ComputeStressTensor(); 2845 input=this->inputs->GetInput(output_enum); 2846 break; 2847 case StressTensoryzEnum: 2848 this->ComputeStressTensor(); 2849 input=this->inputs->GetInput(output_enum); 2850 break; 2851 case StressTensorzzEnum: 2852 this->ComputeStressTensor(); 2853 input=this->inputs->GetInput(output_enum); 2854 break; 3183 2855 default: 3184 /*do nothing, no need to derail the computation because one of the outputs requested cannot be found: */ 3185 break; 3186 } 3187 } 2856 _error_("input "<<EnumToStringx(output_enum)<<" not found in element"); 2857 } 2858 } 2859 2860 /*Assign output pointer*/ 2861 *pinterpolation = input->GetResultInterpolation(); 2862 2863 } 2864 /*}}}*/ 2865 /*FUNCTION Penta::ResultToVector{{{*/ 2866 void Penta::ResultToVector(Vector<IssmPDouble>* vector,int output_enum){ 2867 2868 Input* input=this->inputs->GetInput(output_enum); 2869 if(!input) _error_("input "<<EnumToStringx(output_enum)<<" not found in element"); 2870 2871 switch(input->GetResultInterpolation()){ 2872 case P0Enum: 2873 _error_("not implemented..."); 2874 break; 2875 case P1Enum:{ 2876 IssmDouble values[NUMVERTICES]; 2877 IssmPDouble pvalues[NUMVERTICES]; 2878 int connectivity[NUMVERTICES]; 2879 int sidlist[NUMVERTICES]; 2880 2881 this->GetVertexSidList(&sidlist[0]); 2882 this->GetConnectivityList(&connectivity[0]); 2883 this->GetInputListOnVertices(&values[0],output_enum); 2884 for(int i=0;i<NUMVERTICES;i++) pvalues[i] = reCast<IssmPDouble>(values[i])/reCast<IssmPDouble>(connectivity[i]); 2885 2886 vector->SetValues(NUMVERTICES,&sidlist[0],&pvalues[0],ADD_VAL); 2887 break; 2888 } 2889 default: 2890 _error_("interpolation "<<EnumToStringx(input->GetResultInterpolation())<<" not supported yet"); 2891 } 2892 3188 2893 3189 2894 } … … 3340 3045 3341 3046 /*Spawn material*/ 3342 tria->material=NULL;3343 3047 tria->material=(Material*)this->material->copy(); 3344 3048 delete tria->material->inputs; … … 3515 3219 bool dakota_analysis; 3516 3220 bool isFS; 3517 IssmDouble beta,heatcapacity,referencetemperature,meltingpoint,latentheat;3518 3221 int numnodes; 3519 3222 int* penta_node_ids = NULL; … … 3523 3226 iomodel->Constant(&dakota_analysis,QmuIsdakotaEnum); 3524 3227 iomodel->Constant(&isFS,FlowequationIsFSEnum); 3525 iomodel->Constant(&beta,MaterialsBetaEnum);3526 iomodel->Constant(&heatcapacity,MaterialsHeatcapacityEnum);3527 iomodel->Constant(&referencetemperature,ConstantsReferencetemperatureEnum);3528 iomodel->Constant(&meltingpoint,MaterialsMeltingpointEnum);3529 iomodel->Constant(&latentheat,MaterialsLatentheatEnum);3530 3228 3531 3229 /*Checks if debuging*/ 3532 /*{{{*/3533 3230 _assert_(iomodel->elements); 3534 /*}}}*/3535 3231 3536 3232 /*Recover element type*/ … … 3688 3384 case StressbalanceAnalysisEnum: 3689 3385 3690 /*default vx,vy and vz: either observation or 0 */3691 if(!iomodel->Data(VxEnum)){3692 for(i=0;i<6;i++)nodeinputs[i]=0;3693 this->inputs->AddInput(new PentaInput(VxEnum,nodeinputs,P1Enum));3694 if(dakota_analysis) this->inputs->AddInput(new PentaInput(QmuVxEnum,nodeinputs,P1Enum));3695 }3696 if(!iomodel->Data(VyEnum)){3697 for(i=0;i<6;i++)nodeinputs[i]=0;3698 this->inputs->AddInput(new PentaInput(VyEnum,nodeinputs,P1Enum));3699 if(dakota_analysis) this->inputs->AddInput(new PentaInput(QmuVyEnum,nodeinputs,P1Enum));3700 }3701 if(!iomodel->Data(VzEnum)){3702 for(i=0;i<6;i++)nodeinputs[i]=0;3703 this->inputs->AddInput(new PentaInput(VzEnum,nodeinputs,P1Enum));3704 if(dakota_analysis) this->inputs->AddInput(new PentaInput(QmuVzEnum,nodeinputs,P1Enum));3705 }3706 if(!iomodel->Data(PressureEnum)){3707 for(i=0;i<6;i++)nodeinputs[i]=0;3708 if(dakota_analysis){3709 this->inputs->AddInput(new PentaInput(PressureEnum,nodeinputs,P1Enum));3710 this->inputs->AddInput(new PentaInput(QmuPressureEnum,nodeinputs,P1Enum));3711 }3712 if(isFS){3713 this->inputs->AddInput(new PentaInput(PressureEnum,nodeinputs,P1Enum));3714 this->inputs->AddInput(new PentaInput(PressurePicardEnum,nodeinputs,P1Enum));3715 }3716 }3717 3386 if(*(iomodel->Data(FlowequationElementEquationEnum)+index)==HOFSApproximationEnum){ 3718 3387 /*Create VzHO and VzFS Enums*/ … … 3744 3413 } 3745 3414 break; 3746 3747 case ThermalAnalysisEnum:3748 /*Initialize mesh velocity*/3749 for(i=0;i<6;i++)nodeinputs[i]=0;3750 this->inputs->AddInput(new PentaInput(VxMeshEnum,nodeinputs,P1Enum));3751 this->inputs->AddInput(new PentaInput(VyMeshEnum,nodeinputs,P1Enum));3752 this->inputs->AddInput(new PentaInput(VzMeshEnum,nodeinputs,P1Enum));3753 if(dakota_analysis){3754 this->inputs->AddInput(new PentaInput(QmuVxMeshEnum,nodeinputs,P1Enum));3755 this->inputs->AddInput(new PentaInput(QmuVyMeshEnum,nodeinputs,P1Enum));3756 this->inputs->AddInput(new PentaInput(QmuVzMeshEnum,nodeinputs,P1Enum));3757 }3758 break;3759 3760 case EnthalpyAnalysisEnum:3761 /*Initialize mesh velocity*/3762 for(i=0;i<6;i++)nodeinputs[i]=0;3763 this->inputs->AddInput(new PentaInput(VxMeshEnum,nodeinputs,P1Enum));3764 this->inputs->AddInput(new PentaInput(VyMeshEnum,nodeinputs,P1Enum));3765 this->inputs->AddInput(new PentaInput(VzMeshEnum,nodeinputs,P1Enum));3766 if (iomodel->Data(TemperatureEnum) && iomodel->Data(WaterfractionEnum) && iomodel->Data(PressureEnum)) {3767 for(i=0;i<6;i++){3768 if(iomodel->Data(TemperatureEnum)[penta_vertex_ids[i]-1] < meltingpoint-beta*iomodel->Data(PressureEnum)[penta_vertex_ids[i]-1]){3769 nodeinputs[i]=heatcapacity*(iomodel->Data(TemperatureEnum)[penta_vertex_ids[i]-1]-referencetemperature);3770 }3771 else nodeinputs[i]=heatcapacity*3772 (meltingpoint-beta*iomodel->Data(PressureEnum)[penta_vertex_ids[i]-1]-referencetemperature)3773 +latentheat*iomodel->Data(WaterfractionEnum)[penta_vertex_ids[i]-1];3774 }3775 this->inputs->AddInput(new PentaInput(EnthalpyEnum,nodeinputs,P1Enum));3776 }3777 else _error_("temperature and waterfraction required for the enthalpy solution");3778 break;3779 3780 3415 default: 3781 3416 /*No update for other solution types*/ … … 3786 3421 /*FUNCTION Penta::ViscousHeatingCreateInput {{{*/ 3787 3422 void Penta::ViscousHeatingCreateInput(void){ 3788 3789 /*Constants*/3790 const int numdof=NUMVERTICES*NDOF1;3791 3423 3792 3424 /*Intermediaries*/ … … 3795 3427 IssmDouble xyz_list[NUMVERTICES][3]; 3796 3428 IssmDouble epsilon[6]; 3797 IssmDouble viscousheating[NUMVERTICES] ={0,0,0,0,0,0};3429 IssmDouble viscousheating[NUMVERTICES]; 3798 3430 IssmDouble thickness; 3799 GaussPenta *gauss=NULL;3800 3801 /*Initialize Element vector*/3802 ElementVector* pe=new ElementVector(nodes,NUMVERTICES,this->parameters);3803 3431 3804 3432 /*Retrieve all inputs and parameters*/ … … 3810 3438 3811 3439 /*loop over vertices: */ 3812 gauss=new GaussPenta();3440 GaussPenta* gauss=new GaussPenta(); 3813 3441 for (int iv=0;iv<NUMVERTICES;iv++){ 3814 3442 gauss->GaussVertex(iv); … … 3860 3488 } 3861 3489 /*}}}*/ 3490 /*FUNCTION Penta::IceVolumeAboveFloatation {{{*/ 3491 IssmDouble Penta::IceVolumeAboveFloatation(void){ 3492 3493 /*Volume above floatation: H + rho_water/rho_ice*bathymetry for nodes on the bed*/ 3494 IssmDouble rho_ice,rho_water; 3495 IssmDouble base,bed,surface,bathymetry; 3496 IssmDouble xyz_list[NUMVERTICES][3]; 3497 3498 if(NoIceInElement() || IsFloating() || !IsOnBed())return 0; 3499 3500 rho_ice=matpar->GetRhoIce(); 3501 rho_water=matpar->GetRhoWater(); 3502 GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES); 3503 3504 /*First calculate the area of the base (cross section triangle) 3505 * http://en.wikipedia.org/wiki/Pentangle 3506 * base = 1/2 abs((xA-xC)(yB-yA)-(xA-xB)(yC-yA))*/ 3507 base = 1./2.*fabs((xyz_list[0][0]-xyz_list[2][0])*(xyz_list[1][1]-xyz_list[0][1]) - (xyz_list[0][0]-xyz_list[1][0])*(xyz_list[2][1]-xyz_list[0][1])); 3508 3509 /*Now get the average height above floatation*/ 3510 Input* surface_input = inputs->GetInput(SurfaceEnum); _assert_(surface_input); 3511 Input* bed_input = inputs->GetInput(BedEnum); _assert_(bed_input); 3512 Input* bathymetry_input = inputs->GetInput(BathymetryEnum); _assert_(bathymetry_input); 3513 surface_input->GetInputAverage(&surface); 3514 bed_input->GetInputAverage(&bed); 3515 bathymetry_input->GetInputAverage(&bathymetry); 3516 3517 /*Return: */ 3518 return base*(surface - bed + min( rho_water/rho_ice * bathymetry, 0.) ); 3519 } 3520 /*}}}*/ 3862 3521 /*FUNCTION Penta::MinVel{{{*/ 3863 3522 void Penta::MinVel(IssmDouble* pminvel){ … … 3924 3583 } 3925 3584 /*}}}*/ 3585 /*FUNCTION Penta::MassFlux {{{*/ 3586 IssmDouble Penta::MassFlux( IssmDouble x1, IssmDouble y1, IssmDouble x2, IssmDouble y2,int segment_id){ 3587 3588 IssmDouble mass_flux=0; 3589 3590 if(!IsOnBed()) return mass_flux; 3591 3592 /*Depth Averaging Vx and Vy*/ 3593 this->InputDepthAverageAtBase(VxEnum,VxAverageEnum); 3594 this->InputDepthAverageAtBase(VyEnum,VyAverageEnum); 3595 3596 /*Spawn Tria element from the base of the Penta: */ 3597 Tria* tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 3598 mass_flux=tria->MassFlux(x1,y1,x2,y2,segment_id); 3599 delete tria->material; delete tria; 3600 3601 /*Delete Vx and Vy averaged*/ 3602 this->inputs->DeleteInput(VxAverageEnum); 3603 this->inputs->DeleteInput(VyAverageEnum); 3604 3605 /*clean up and return*/ 3606 return mass_flux; 3607 } 3608 /*}}}*/ 3926 3609 /*FUNCTION Penta::MaxAbsVx{{{*/ 3927 3610 void Penta::MaxAbsVx(IssmDouble* pmaxabsvx){ … … 4002 3685 *presponse=this->material->GetBbar(); 4003 3686 break; 4004 case MaterialsRheologyZbarEnum:4005 *presponse=this->material->Get Zbar();3687 case DamageDbarEnum: 3688 *presponse=this->material->GetDbar(); 4006 3689 break; 4007 3690 case VelEnum: … … 4088 3771 /*Intermediaries */ 4089 3772 int stabilization; 4090 int i,j ,found=0;3773 int i,j; 4091 3774 IssmDouble Jdet,u,v,w,um,vm,wm; 4092 3775 IssmDouble h,hx,hy,hz,vx,vy,vz,vel; … … 4107 3790 IssmDouble D[3][3]; 4108 3791 IssmDouble K[3][3]={0.0}; 4109 Tria* tria=NULL;4110 3792 GaussPenta *gauss=NULL; 4111 3793 … … 4142 3824 4143 3825 /*Conduction: */ 4144 /*Need to change that depending on enthalpy value -> cold or temperate ice: */4145 3826 GetBConduct(&B_conduct[0][0],&xyz_list[0][0],gauss); 4146 3827 4147 3828 enthalpy_input->GetInputValue(&enthalpy, gauss); 4148 3829 pressure_input->GetInputValue(&pressure, gauss); 4149 kappa=matpar->GetEnthalpyDiffusionParameter(enthalpy,pressure); 4150 D_scalar_conduct=gauss->weight*Jdet*kappa ;3830 kappa=matpar->GetEnthalpyDiffusionParameter(enthalpy,pressure); _assert_(kappa>0.); 3831 D_scalar_conduct=gauss->weight*Jdet*kappa/rho_ice; 4151 3832 if(reCast<bool,IssmDouble>(dt)) D_scalar_conduct=D_scalar_conduct*dt; 4152 3833 … … 4191 3872 } 4192 3873 4193 /*Artif ficial diffusivity*/3874 /*Artificial diffusivity*/ 4194 3875 if(stabilization==1){ 4195 3876 /*Build K: */ … … 4197 3878 vel=sqrt(vx*vx + vy*vy + vz*vz)+1.e-14; 4198 3879 h=sqrt( pow(hx*vx/vel,2) + pow(hy*vy/vel,2) + pow(hz*vz/vel,2)); 3880 4199 3881 K[0][0]=h/(2*vel)*vx*vx; K[0][1]=h/(2*vel)*vx*vy; K[0][2]=h/(2*vel)*vx*vz; 4200 3882 K[1][0]=h/(2*vel)*vy*vx; K[1][1]=h/(2*vel)*vy*vy; K[1][2]=h/(2*vel)*vy*vz; 4201 3883 K[2][0]=h/(2*vel)*vz*vx; K[2][1]=h/(2*vel)*vz*vy; K[2][2]=h/(2*vel)*vz*vz; 3884 4202 3885 D_scalar_stab=gauss->weight*Jdet; 4203 3886 if(reCast<bool,IssmDouble>(dt)) D_scalar_stab=D_scalar_stab*dt; … … 4324 4007 /*Intermediaries */ 4325 4008 int stabilization; 4326 int i,j ,found=0;4009 int i,j; 4327 4010 IssmDouble Jdet,u,v,w,um,vm,wm,vel; 4328 4011 IssmDouble h,hx,hy,hz,vx,vy,vz; … … 4340 4023 IssmDouble D[3][3]; 4341 4024 IssmDouble K[3][3]={0.0}; 4342 Tria* tria=NULL;4343 4025 GaussPenta *gauss=NULL; 4344 4026 … … 4389 4071 4390 4072 /*Advection: */ 4391 4392 4073 GetBAdvec(&B_advec[0][0],&xyz_list[0][0],gauss); 4393 4074 GetBprimeAdvec(&Bprime_advec[0][0],&xyz_list[0][0],gauss); … … 4544 4225 4545 4226 /*Intermediaries*/ 4546 int i,j,found=0;4547 int friction_type,stabilization;4227 int i; 4228 int stabilization; 4548 4229 IssmDouble Jdet,phi,dt; 4549 4230 IssmDouble rho_ice,heatcapacity; … … 4554 4235 IssmDouble u,v,w; 4555 4236 IssmDouble scalar_def,scalar_transient; 4556 IssmDouble temperature_list[NUMVERTICES];4557 4237 IssmDouble xyz_list[NUMVERTICES][3]; 4558 4238 IssmDouble L[numdof]; … … 4600 4280 scalar_def=phi/rho_ice*Jdet*gauss->weight; 4601 4281 if(reCast<bool,IssmDouble>(dt)) scalar_def=scalar_def*dt; 4602 4282 4283 /*TODO: add -beta*laplace T_m(p)*/ 4603 4284 for(i=0;i<NUMVERTICES;i++) pe->values[i]+=scalar_def*L[i]; 4604 4285 … … 4619 4300 enthalpypicard_input->GetInputValue(&enthalpypicard, gauss); 4620 4301 kappa=matpar->GetEnthalpyDiffusionParameter(enthalpy,pressure); 4621 tau_parameter=GetStabilizationParameter(u,v,w,diameter,kappa); 4302 kappa/=rho_ice; 4303 tau_parameter=GetStabilizationParameter(u,v,w,diameter,kappa); 4622 4304 4623 4305 for(i=0;i<NUMVERTICES;i++) pe->values[i]+=tau_parameter*scalar_def*(u*dbasis[0][i]+v*dbasis[1][i]+w*dbasis[2][i]); … … 4702 4384 IssmDouble xyz_list_tria[NUMVERTICES2D][3]={0.0}; 4703 4385 IssmDouble Jdet2d,dt; 4704 IssmDouble rho_ice,heatcapacity,geothermalflux_value; 4705 IssmDouble basalfriction,alpha2,vx,vy; 4386 IssmDouble rho_ice,heatcapacity; 4387 IssmDouble geothermalflux_value, heatflux_value; 4388 IssmDouble basalfriction,alpha2,vx,vy,vz; 4706 4389 IssmDouble scalar,enthalpy,enthalpyup; 4707 4390 IssmDouble pressure,pressureup; … … 4739 4422 gauss=new GaussPenta(0,1,2,2); 4740 4423 gaussup=new GaussPenta(3,4,5,2); 4424 4741 4425 for(int ig=gauss->begin();ig<gauss->end();ig++){ 4742 4426 … … 4749 4433 enthalpy_input->GetInputValue(&enthalpy,gauss); 4750 4434 pressure_input->GetInputValue(&pressure,gauss); 4751 if(enthalpy>=matpar->PureIceEnthalpy(pressure)){ 4435 watercolumn_input->GetInputValue(&watercolumn,gauss); 4436 4437 if((watercolumn<=0.) && (enthalpy<matpar->PureIceEnthalpy(pressure))){ 4438 /* the above check is equivalent to 4439 NOT ((watercolumn>0.) AND (enthalpy<PIE)) AND (enthalpy<PIE)*/ 4440 geothermalflux_input->GetInputValue(&geothermalflux_value,gauss); 4441 4442 friction->GetAlpha2(&alpha2,gauss,VxEnum,VyEnum,VzEnum); 4443 vx_input->GetInputValue(&vx,gauss); 4444 vy_input->GetInputValue(&vy,gauss); 4445 vz_input->GetInputValue(&vz,gauss); 4446 basalfriction=alpha2*(pow(vx,2.0)+pow(vy,2.0)+pow(vz,2.0)); 4447 4448 heatflux_value=(basalfriction+geothermalflux_value)/(rho_ice); 4449 scalar=gauss->weight*Jdet2d*heatflux_value; 4450 if(reCast<bool,IssmDouble>(dt)) scalar=dt*scalar; 4451 for(i=0;i<numdof;i++) pe->values[i]+=scalar*basis[i]; 4452 } 4453 else if(enthalpy>=matpar->PureIceEnthalpy(pressure)){ 4454 /* check positive thickness of temperate basal ice layer */ 4752 4455 enthalpy_input->GetInputValue(&enthalpyup,gaussup); 4753 4456 pressure_input->GetInputValue(&pressureup,gaussup); 4754 4457 if(enthalpyup>=matpar->PureIceEnthalpy(pressureup)){ 4755 // temperate ice has positive thickness: grad enthalpy*n=0.4458 // TODO: temperate ice has positive thickness: grad enthalpy*n=0. 4756 4459 } 4757 4460 else{ 4758 // only base temperate, set Dirichlet BCs in Penta::Update ThermalBasalConstraints()4461 // only base temperate, set Dirichlet BCs in Penta::UpdateBasalConstraintsEnthalpy() 4759 4462 } 4760 4463 } 4761 else{ 4762 watercolumn_input->GetInputValue(&watercolumn,gauss); 4763 if(watercolumn==0.){ 4764 /*add geothermal heat flux and basal friction*/ 4765 geothermalflux_input->GetInputValue(&geothermalflux_value,gauss); 4766 friction->GetAlpha2(&alpha2,gauss,VxEnum,VyEnum,VzEnum); 4767 vx_input->GetInputValue(&vx,gauss); 4768 vy_input->GetInputValue(&vy,gauss); 4769 basalfriction=alpha2*(pow(vx,2.0)+pow(vy,2.0)); 4770 4771 scalar=gauss->weight*Jdet2d*(basalfriction+geothermalflux_value)/(rho_ice); 4772 if(reCast<bool,IssmDouble>(dt)) scalar=dt*scalar; 4773 4774 for(i=0;i<numdof;i++) pe->values[i]+=scalar*basis[i]; 4775 } 4776 else{ /*do nothing (water layer acts as insulation)*/ } 4464 else { 4465 // base cold, but watercolumn positive. Set base to h_pmp. 4777 4466 } 4778 4467 } … … 4813 4502 4814 4503 /*Intermediaries*/ 4815 int i,j,found=0;4816 int friction_type,stabilization;4504 int i; 4505 int stabilization; 4817 4506 IssmDouble Jdet,phi,dt; 4818 4507 IssmDouble rho_ice,heatcapacity; … … 4822 4511 IssmDouble u,v,w; 4823 4512 IssmDouble scalar_def,scalar_transient; 4824 IssmDouble temperature_list[NUMVERTICES];4825 4513 IssmDouble xyz_list[NUMVERTICES][3]; 4826 4514 IssmDouble L[numdof]; … … 4844 4532 Input* vz_input=inputs->GetInput(VzEnum); _assert_(vz_input); 4845 4533 Input* temperature_input=NULL; 4846 if 4847 if 4534 if(reCast<bool,IssmDouble>(dt)) temperature_input=inputs->GetInput(TemperatureEnum); _assert_(inputs); 4535 if(stabilization==2) diameter=MinEdgeLength(xyz_list); 4848 4536 4849 4537 /* Start looping on the number of gaussian points: */ … … 4957 4645 4958 4646 /*Intermediaries */ 4959 int i,j;4960 int analysis_type;4961 IssmDouble xyz_list[NUMVERTICES][3];4962 IssmDouble xyz_list_tria[NUMVERTICES2D][3]={0.0};4963 IssmDouble Jdet2d,dt;4964 IssmDouble rho_ice,heatcapacity,geothermalflux_value;4965 IssmDouble basalfriction,alpha2,vx,vy;4966 IssmDouble basis[NUMVERTICES];4967 IssmDouble scalar;4968 Friction* friction=NULL;4647 int i,j; 4648 int analysis_type; 4649 IssmDouble xyz_list[NUMVERTICES][3]; 4650 IssmDouble xyz_list_tria[NUMVERTICES2D][3]={0.0}; 4651 IssmDouble Jdet2d,dt; 4652 IssmDouble rho_ice,heatcapacity,geothermalflux_value; 4653 IssmDouble basalfriction,alpha2,vx,vy; 4654 IssmDouble basis[NUMVERTICES]; 4655 IssmDouble scalar; 4656 Friction* friction=NULL; 4969 4657 GaussPenta* gauss=NULL; 4970 4658 … … 5017 4705 } 5018 4706 /*}}}*/ 5019 /*FUNCTION Penta::GetSolutionFromInputsThermal{{{*/5020 void Penta::GetSolutionFromInputsThermal(Vector<IssmDouble>* solution){5021 5022 const int numdof=NDOF1*NUMVERTICES;5023 5024 int i;5025 int *doflist = NULL;5026 IssmDouble values[numdof];5027 IssmDouble temp;5028 GaussPenta *gauss = NULL;5029 5030 /*Get dof list: */5031 GetDofList(&doflist,NoneApproximationEnum,GsetEnum);5032 Input* t_input=inputs->GetInput(TemperatureEnum); _assert_(t_input);5033 5034 gauss=new GaussPenta();5035 for(i=0;i<NUMVERTICES;i++){5036 /*Recover temperature*/5037 gauss->GaussVertex(i);5038 t_input->GetInputValue(&temp,gauss);5039 values[i]=temp;5040 }5041 5042 /*Add value to global vector*/5043 solution->SetValues(numdof,doflist,values,INS_VAL);5044 5045 /*Free ressources:*/5046 delete gauss;5047 xDelete<int>(doflist);5048 }5049 /*}}}*/5050 /*FUNCTION Penta::GetSolutionFromInputsEnthalpy{{{*/5051 void Penta::GetSolutionFromInputsEnthalpy(Vector<IssmDouble>* solution){5052 5053 const int numdof=NDOF1*NUMVERTICES;5054 5055 int* doflist=NULL;5056 IssmDouble values[numdof];5057 IssmDouble enthalpy;5058 GaussPenta *gauss=NULL;5059 5060 /*Get dof list: */5061 GetDofList(&doflist,NoneApproximationEnum,GsetEnum);5062 Input* h_input=inputs->GetInput(EnthalpyEnum); _assert_(h_input);5063 5064 gauss=new GaussPenta();5065 for(int i=0;i<NUMVERTICES;i++){5066 /*Recover temperature*/5067 gauss->GaussVertex(i);5068 h_input->GetInputValue(&enthalpy,gauss);5069 values[i]=enthalpy;5070 }5071 5072 /*Add value to global vector*/5073 solution->SetValues(numdof,doflist,values,INS_VAL);5074 5075 /*Free ressources:*/5076 delete gauss;5077 xDelete<int>(doflist);5078 }5079 /*}}}*/5080 4707 /*FUNCTION Penta::InputUpdateFromSolutionThermal {{{*/ 5081 4708 void Penta::InputUpdateFromSolutionThermal(IssmDouble* solution){ … … 5090 4717 IssmDouble B_average,s_average; 5091 4718 int *doflist = NULL; 5092 IssmDouble pressure[numdof];4719 bool hack = false; 5093 4720 5094 4721 /*Get dof list: */ … … 5098 4725 for(i=0;i<numdof;i++){ 5099 4726 values[i]=solution[doflist[i]]; 5100 //GetInputListOnVertices(&pressure[0],PressureEnum);5101 //if(values[i]>matpar->TMeltingPoint(pressure[i])) values[i]=matpar->TMeltingPoint(pressure[i]);5102 //if(values[i]<matpar->TMeltingPoint(pressure[i])-50) values[i]=matpar->TMeltingPoint(pressure[i])-50;5103 4727 5104 4728 /*Check solution*/ … … 5107 4731 //if(values[i]>275) _printf_("temperature > 275°K found in solution vector (Paterson's rheology associated is negative)\n"); 5108 4732 } 4733 4734 if(hack){ 4735 /*Force temperature between [Tpmp-50 Tpmp] to disable penalties*/ 4736 IssmDouble pressure[numdof]; 4737 GetInputListOnVertices(&pressure[0],PressureEnum); 4738 for(i=0;i<numdof;i++){ 4739 if(values[i]>matpar->TMeltingPoint(pressure[i])) values[i]=matpar->TMeltingPoint(pressure[i]); 4740 if(values[i]<matpar->TMeltingPoint(pressure[i])-50.) values[i]=matpar->TMeltingPoint(pressure[i])-50.; 4741 } 4742 } 4743 5109 4744 5110 4745 /*Get all inputs and parameters*/ … … 5216 4851 case LliboutryDuvalEnum: 5217 4852 B_average=LliboutryDuval((values[0]+values[1]+values[2]+values[3]+values[4]+values[5])/6.0, 5218 (pressure[0]+pressure[1]+pressure[2]+pressure[3]+pressure[4]+pressure[5])/6.0, 5219 material->GetN()); 4853 (pressure[0]+pressure[1]+pressure[2]+pressure[3]+pressure[4]+pressure[5])/6.0, 4854 material->GetN(), 4855 matpar->GetBeta(), matpar->GetReferenceTemperature(), matpar->GetHeatCapacity(), matpar->GetLatentHeat()); 5220 4856 for(i=0;i<numdof;i++) B[i]=B_average; 5221 4857 this->material->inputs->AddInput(new PentaInput(MaterialsRheologyBEnum,B,P1Enum)); … … 5233 4869 } 5234 4870 /*}}}*/ 5235 /*FUNCTION Penta::Update ThermalBasalConstraints{{{*/5236 void Penta::Update ThermalBasalConstraints(void){4871 /*FUNCTION Penta::UpdateBasalConstraintsEnthalpy{{{*/ 4872 void Penta::UpdateBasalConstraintsEnthalpy(void){ 5237 4873 5238 4874 /*Intermediary*/ 5239 bool isenthalpy,isdynamicbasalspc, istemperatelayer;4875 bool isenthalpy,isdynamicbasalspc,setspc; 5240 4876 int numindices, numindicesup; 5241 IssmDouble h_pmp,pressure, pressureup; 5242 IssmDouble enthalpy, enthalpyup; 4877 IssmDouble pressure, pressureup; 4878 IssmDouble h_pmp, enthalpy, enthalpyup; 4879 IssmDouble watercolumn; 5243 4880 int *indices = NULL, *indicesup = NULL; 5244 4881 … … 5249 4886 parameters->FindParam(&isenthalpy,ThermalIsenthalpyEnum); 5250 4887 if(!isenthalpy) return; 5251 //parameters->FindParam(&isdynamicbasalspc,ThermalIsdynamicbasalspcEnum); 5252 isdynamicbasalspc = true; 4888 parameters->FindParam(&isdynamicbasalspc,ThermalIsdynamicbasalspcEnum); 5253 4889 if(!isdynamicbasalspc) return; 5254 4890 5255 4891 /*Fetch indices of basal & surface nodes for this finite element*/ 5256 4892 BasalNodeIndices(&numindices,&indices,this->element_type); 5257 5258 4893 SurfaceNodeIndices(&numindicesup,&indicesup,this->element_type); 4894 _assert_(numindices==numindicesup); 5259 4895 5260 4896 /*Get parameters and inputs: */ 5261 4897 Input* pressure_input=inputs->GetInput(PressureEnum); _assert_(pressure_input); 5262 Input* enthalpy_input=inputs->GetInput(EnthalpyEnum); _assert_(enthalpy_input); 5263 5264 /*if there is a temperate layer of positive thickness, set enthalpy=h_pmp at that node*/ 4898 Input* enthalpy_input=inputs->GetInput(EnthalpyEnum); _assert_(enthalpy_input); 4899 Input* watercolumn_input=inputs->GetInput(WatercolumnEnum); //_assert_(watercolumn_input); 4900 4901 /*if there is a temperate layer of zero thickness, set spc enthalpy=h_pmp at that node*/ 5265 4902 GaussPenta* gauss=new GaussPenta(); 5266 4903 GaussPenta* gaussup=new GaussPenta(); 5267 4904 for(int i=0;i<numindices;i++){ 5268 4905 gauss->GaussNode(this->element_type,indices[i]); 5269 gaussup->GaussNode(this->element_type,indicesup[i]); // TODO: check: are the nodes corresponding? 5270 5271 /*Check wether there is a temperate layer at the base or not -> TODO: Johannes:)*/5272 /*check if node is temperate, if not, return*/5273 4906 gaussup->GaussNode(this->element_type,indicesup[i]); 4907 4908 /*Check wether there is a temperate layer at the base or not */ 4909 /*check if node is temperate, if not, continue*/ 4910 enthalpy_input->GetInputValue(&enthalpy, gauss); 5274 4911 pressure_input->GetInputValue(&pressure, gauss); 5275 if (enthalpy<matpar->PureIceEnthalpy(pressure)){ 5276 // TODO: reset, if necessary, all spcs to non-valid 5277 continue; 5278 } 5279 /*check if upper node is temperate. if yes, then we have a temperate layer of positive thickness. if not, continue.*/ 5280 enthalpy_input->GetInputValue(&enthalpyup, gaussup); 5281 pressure_input->GetInputValue(&pressureup, gaussup); 5282 istemperatelayer = false; 5283 if (enthalpyup>=matpar->PureIceEnthalpy(pressureup)) 5284 istemperatelayer=true; 5285 5286 /*Add Dirichlet constraint to this node if there is no layer of temperate ice with positive thickness*/ 5287 if(!istemperatelayer){ 4912 watercolumn_input->GetInputValue(&watercolumn,gauss); 4913 setspc = false; 4914 if (enthalpy>=matpar->PureIceEnthalpy(pressure)){ 4915 /*check if upper node is temperate, too. 4916 if yes, then we have a temperate layer of positive thickness and reset the spc. 4917 if not, apply dirichlet BC.*/ 4918 enthalpy_input->GetInputValue(&enthalpyup, gaussup); 4919 pressure_input->GetInputValue(&pressureup, gaussup); 4920 setspc=((enthalpyup<matpar->PureIceEnthalpy(pressureup)) && (watercolumn>=0.))?true:false; 4921 } 4922 4923 if (setspc) { 5288 4924 /*Calculate enthalpy at pressure melting point */ 5289 4925 h_pmp=matpar->PureIceEnthalpy(pressure); 5290 5291 4926 /*Apply Dirichlet condition (dof = 0 here, since there is only one degree of freedom per node)*/ 5292 4927 nodes[indices[i]]->ApplyConstraint(1,h_pmp); 5293 4928 } 5294 5295 5296 5297 4929 else { 4930 /*remove spc*/ 4931 nodes[indices[i]]->DofInFSet(0); 4932 } 5298 4933 } 5299 4934 5300 4935 /*Free ressources:*/ 5301 4936 xDelete<int>(indices); 5302 xDelete<int>(indicesup); 4937 xDelete<int>(indicesup); 4938 delete gauss; 4939 delete gaussup; 5303 4940 } 5304 4941 /*}}}*/ … … 5306 4943 void Penta::ComputeBasalMeltingrate(void){ 5307 4944 /*Calculate the basal melt rates of the enthalpy model after Aschwanden 2012*/ 4945 /* melting rate is positive when melting, negative when refreezing*/ 5308 4946 5309 4947 /* Intermediaries */ … … 5322 4960 IssmDouble vx[NUMVERTICES],vy[NUMVERTICES],vz[NUMVERTICES]; 5323 4961 IssmDouble geothermalflux[NUMVERTICES]; 5324 IssmDouble dt,meltingrate_enthalpy; 4962 IssmDouble dt, yts; 4963 IssmDouble melting_overshoot,meltingrate_enthalpy; 4964 IssmDouble lambda,heating; 5325 4965 Friction *friction = NULL; 5326 4966 … … 5334 4974 /*Fetch parameters and inputs */ 5335 4975 latentheat=matpar->GetLatentHeat(); 5336 rho_ice= this->matpar->GetRhoIce();4976 rho_ice=matpar->GetRhoIce(); 5337 4977 GetInputListOnVertices(&vx[0],VxEnum); 5338 4978 GetInputListOnVertices(&vy[0],VyEnum); … … 5359 4999 checkpositivethickness=true; 5360 5000 5001 _assert_(watercolumn[iv]>=0.); 5002 5361 5003 /*Calculate basal meltingrate after Fig.5 of A.Aschwanden 2012*/ 5362 5004 meltingrate_enthalpy=0.; 5005 heating=0.; 5363 5006 if((watercolumn[iv]>0.) && (enthalpy[iv]<matpar->PureIceEnthalpy(pressure[iv]))){ 5364 5007 /*ensure that no ice is at T<Tm(p), if water layer present*/ 5365 5008 enthalpy[iv]=matpar->PureIceEnthalpy(pressure[iv]); 5366 //meltingrate_enthalpy=0.;5367 5009 } 5368 5010 else if(enthalpy[iv]<matpar->PureIceEnthalpy(pressure[iv])){ 5369 5011 /*cold base: set q*n=q_geo*n+frictionheating as Neumann BC in Penta::CreatePVectorEnthalpySheet*/ 5370 meltingrate_enthalpy=0.; 5371 checkpositivethickness=false; 5372 } 5373 else {/*do nothing, go to next check*/} 5012 checkpositivethickness=false; // cold base, skip next test 5013 } 5014 else {/*we have a temperate base, go to next test*/} 5374 5015 5375 5016 if(checkpositivethickness){ … … 5377 5018 istemperatelayer=false; 5378 5019 if(enthalpy[iv+3]>=matpar->PureIceEnthalpy(pressure[iv+3])) istemperatelayer=true; 5379 if(istemperatelayer) for(i=0;i<3;i++) vec_heatflux[i]=0.; 5020 if(istemperatelayer) for(i=0;i<3;i++) vec_heatflux[i]=0.; // TODO: add -k*nabla T_pmp 5380 5021 else{ 5381 5022 enthalpy_input->GetInputDerivativeValue(&d1enthalpy[0],&xyz_list[0][0],gauss); 5382 kappa=matpar->GetEnthalpyDiffusionParameter (enthalpy[iv],pressure[iv]);5023 kappa=matpar->GetEnthalpyDiffusionParameterVolume(enthalpy[iv],enthalpy[iv+NUMVERTICES2D], pressure[iv],pressure[iv+NUMVERTICES2D]); _assert_(kappa>0.); 5383 5024 for(i=0;i<3;i++) vec_heatflux[i]=-kappa*d1enthalpy[i]; 5384 5025 } … … 5394 5035 5395 5036 matpar->EnthalpyToThermal(&temperature, &waterfraction, enthalpy[iv],pressure[iv]); 5396 meltingrate_enthalpy=(basalfriction-(heatflux-geothermalflux[iv]))/((1-waterfraction)*latentheat*rho_ice); // m/yr water equivalent 5037 // -Mb= Fb-(q-q_geo)/((1-w)*L), cf Aschwanden 2012, eq.66 5038 heating=(heatflux+basalfriction+geothermalflux[iv]); 5039 meltingrate_enthalpy=heating/((1-waterfraction)*latentheat*rho_ice); // m/s water equivalent 5397 5040 } 5398 5041 5399 5042 /*Update water column, basal meltingrate*/ 5400 basalmeltingrate[iv]+=meltingrate_enthalpy;5401 5043 this->parameters->FindParam(&dt,TimesteppingTimeStepEnum); 5402 if(reCast<bool,IssmDouble>(dt)) 5403 watercolumn[iv]+=dt*meltingrate_enthalpy; 5404 else 5405 watercolumn[iv]+=meltingrate_enthalpy; 5044 if(reCast<bool,IssmDouble>(dt)){ 5045 if(watercolumn[iv]+meltingrate_enthalpy*dt<0.){ 5046 melting_overshoot=watercolumn[iv]+meltingrate_enthalpy*dt; 5047 lambda=melting_overshoot/(meltingrate_enthalpy*dt); _assert_(lambda>0); _assert_(lambda<1); 5048 basalmeltingrate[iv]=(1.-lambda)*meltingrate_enthalpy; 5049 watercolumn[iv]=0.; 5050 yts=365*24*60*60; 5051 enthalpy[iv]+=dt/yts*lambda*heating; 5052 } 5053 else{ 5054 basalmeltingrate[iv]=meltingrate_enthalpy; 5055 watercolumn[iv]+=dt*meltingrate_enthalpy; 5056 } 5057 } 5058 else{ 5059 basalmeltingrate[iv]=meltingrate_enthalpy; 5060 watercolumn[iv]+=meltingrate_enthalpy; 5061 } 5406 5062 } 5407 5063 /*feed updated variables back into model*/ … … 5417 5073 /*FUNCTION Penta::DrainWaterfraction{{{*/ 5418 5074 void Penta::DrainWaterfraction(void){ 5419 5075 5420 5076 /*Intermediaries*/ 5421 5077 bool isenthalpy; 5422 5078 IssmDouble waterfraction[NUMVERTICES], temperature[NUMVERTICES]; 5079 IssmDouble watercolumnbase[NUMVERTICES]; 5423 5080 IssmDouble enthalpy[NUMVERTICES], pressure[NUMVERTICES]; 5424 5081 IssmDouble latentheat, dt; 5082 IssmDouble dw, dwc; 5083 Penta *pentabase = NULL; 5425 5084 5426 5085 /*Check wether enthalpy is activated*/ 5427 5086 parameters->FindParam(&isenthalpy,ThermalIsenthalpyEnum); 5428 5087 if(!isenthalpy) return; 5429 5430 GetInputListOnVertices(&enthalpy[0],EnthalpyEnum); 5431 GetInputListOnVertices(&pressure[0],PressureEnum); 5088 5089 /*get basal element, needed for basal watercolumn*/ 5090 pentabase=this->GetBasalElement(); 5091 5092 this->GetInputListOnVertices(&enthalpy[0],EnthalpyEnum); 5093 this->GetInputListOnVertices(&pressure[0],PressureEnum); 5094 pentabase->GetInputListOnVertices(&watercolumnbase[0], WatercolumnEnum); 5095 5432 5096 this->parameters->FindParam(&dt,TimesteppingTimeStepEnum); 5433 5097 latentheat=matpar->GetLatentHeat(); … … 5435 5099 for(int iv=0;iv<NUMVERTICES;iv++){ 5436 5100 matpar->EnthalpyToThermal(&temperature[iv],&waterfraction[iv], enthalpy[iv],pressure[iv]); 5437 5101 dw=DrainageFunctionWaterfraction(waterfraction[iv], dt); 5438 5102 /*drain water fraction & update enthalpy*/ 5439 waterfraction[iv]-= DrainageFunctionWaterfraction(waterfraction[iv], dt);5103 waterfraction[iv]-=dw; 5440 5104 matpar->ThermalToEnthalpy(&enthalpy[iv], temperature[iv], waterfraction[iv], pressure[iv]); 5441 } 5105 /*add drained water to watercolumn at base*/ 5106 dwc=dw*this->IceVolume(); 5107 watercolumnbase[iv%NUMVERTICES2D]+=dwc; 5108 } 5109 5442 5110 /*feed updated results back into model*/ 5443 5111 this->inputs->AddInput(new PentaInput(EnthalpyEnum,enthalpy,P1Enum)); 5444 5112 this->inputs->AddInput(new PentaInput(WaterfractionEnum,waterfraction,P1Enum)); 5445 // this->inputs->AddInput(new PentaInput(TemperatureEnum,temperature,P1Enum)); // temperature should not change during drainage... 5113 pentabase->inputs->AddInput(new PentaInput(WatercolumnEnum, watercolumnbase,P1Enum)); 5114 5446 5115 } 5447 5116 /*}}}*/ … … 5459 5128 input=(Input*)material->inputs->GetInput(MaterialsRheologyBEnum); 5460 5129 } 5461 else if(enum_type== MaterialsRheologyZbarEnum){5130 else if(enum_type==DamageDbarEnum){ 5462 5131 if(!IsOnBed()) return; 5463 input=(Input*)material->inputs->GetInput(MaterialsRheologyZEnum); 5464 } 5465 5132 input=(Input*)material->inputs->GetInput(DamageDEnum); 5133 } 5466 5134 else{ 5467 5135 input=inputs->GetInput(enum_type); … … 5482 5150 input=(Input*)material->inputs->GetInput(MaterialsRheologyBEnum); 5483 5151 } 5484 else if(enum_type== MaterialsRheologyZbarEnum){5485 input=(Input*)material->inputs->GetInput( MaterialsRheologyZEnum);5152 else if(enum_type==DamageDbarEnum){ 5153 input=(Input*)material->inputs->GetInput(DamageDEnum); 5486 5154 } 5487 5155 else{ … … 5504 5172 input=(Input*)material->inputs->GetInput(MaterialsRheologyBEnum); 5505 5173 } 5506 else if(enum_type== MaterialsRheologyZbarEnum){5507 input=(Input*)material->inputs->GetInput( MaterialsRheologyZEnum);5174 else if(enum_type==DamageDbarEnum){ 5175 input=(Input*)material->inputs->GetInput(DamageDEnum); 5508 5176 } 5509 5177 else{ … … 5517 5185 grad_input=new PentaInput(GradientEnum,grad_list,P1Enum); 5518 5186 ((ControlInput*)input)->SetGradient(grad_input); 5187 5188 }/*}}}*/ 5189 /*FUNCTION Penta::ControlToVectors{{{*/ 5190 void Penta::ControlToVectors(Vector<IssmPDouble>* vector_control, Vector<IssmPDouble>* vector_gradient,int control_enum){ 5191 5192 Input* input=NULL; 5193 5194 if(control_enum==MaterialsRheologyBbarEnum){ 5195 input=(Input*)material->inputs->GetInput(MaterialsRheologyBEnum); 5196 } 5197 else if(control_enum==DamageDbarEnum){ 5198 input=(Input*)material->inputs->GetInput(DamageDEnum); 5199 } 5200 else{ 5201 input=inputs->GetInput(control_enum); 5202 } 5203 if (!input) _error_("Input " << EnumToStringx(control_enum) << " not found"); 5204 if (input->ObjectEnum()!=ControlInputEnum) _error_("Input " << EnumToStringx(control_enum) << " is not a ControlInput"); 5205 5206 int sidlist[NUMVERTICES]; 5207 int connectivity[NUMVERTICES]; 5208 IssmPDouble values[NUMVERTICES]; 5209 IssmPDouble gradients[NUMVERTICES]; 5210 IssmDouble value,gradient; 5211 5212 this->GetConnectivityList(&connectivity[0]); 5213 this->GetVertexSidList(&sidlist[0]); 5214 5215 GaussPenta* gauss=new GaussPenta(); 5216 for (int iv=0;iv<NUMVERTICES;iv++){ 5217 gauss->GaussVertex(iv); 5218 5219 ((ControlInput*)input)->GetInputValue(&value,gauss); 5220 ((ControlInput*)input)->GetGradientValue(&gradient,gauss); 5221 5222 values[iv] = reCast<IssmPDouble>(value)/reCast<IssmPDouble>(connectivity[iv]); 5223 gradients[iv] = reCast<IssmPDouble>(gradient)/reCast<IssmPDouble>(connectivity[iv]); 5224 } 5225 delete gauss; 5226 5227 vector_control->SetValues(NUMVERTICES,&sidlist[0],&values[0],ADD_VAL); 5228 vector_gradient->SetValues(NUMVERTICES,&sidlist[0],&gradients[0],ADD_VAL); 5519 5229 5520 5230 }/*}}}*/ … … 5551 5261 case MaticeEnum: 5552 5262 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 5553 break; 5554 case MatdamageiceEnum: 5555 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 5556 this->InputDepthAverageAtBase(MaterialsRheologyZEnum,MaterialsRheologyZbarEnum,MaterialsEnum); 5263 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 5557 5264 break; 5558 5265 default: … … 5567 5274 /*Delete averaged fields*/ 5568 5275 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 5569 this->material->inputs->DeleteInput( MaterialsRheologyZbarEnum);5276 this->material->inputs->DeleteInput(DamageDbarEnum); 5570 5277 5571 5278 /*clean up and return*/ … … 5586 5293 IssmDouble epsilon[5]; /* epsilon=[exx,eyy,exy,exz,eyz];*/ 5587 5294 IssmDouble eps1[3],eps2[3]; 5588 IssmDouble phi[NUMVERTICES];5589 5295 IssmDouble dphi[3][NUMVERTICES]; 5590 5296 GaussPenta *gauss=NULL; … … 5755 5461 ElementVector* Penta::CreatePVectorAdjointHO(void){ 5756 5462 5757 5758 5463 /*Nothing to be done if not on surface*/ 5759 5464 if (!IsOnSurface()) return NULL; … … 5783 5488 GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES); 5784 5489 this->parameters->FindParam(&num_responses,InversionNumCostFunctionsEnum); 5785 this->parameters->FindParam(&responses,NULL, NULL,StepResponsesEnum);5490 this->parameters->FindParam(&responses,NULL,InversionCostFunctionsEnum); 5786 5491 Input* weights_input=inputs->GetInput(InversionCostFunctionsCoefficientsEnum); _assert_(weights_input); 5787 5492 Input* vx_input =inputs->GetInput(VxEnum); _assert_(vx_input); … … 5818 5523 for(resp=0;resp<num_responses;resp++){ 5819 5524 5820 weights_input->GetInputValue(&weight,gauss,resp );5525 weights_input->GetInputValue(&weight,gauss,responses[resp]); 5821 5526 5822 5527 switch(responses[resp]){ … … 5979 5684 GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES); 5980 5685 this->parameters->FindParam(&num_responses,InversionNumCostFunctionsEnum); 5981 this->parameters->FindParam(&responses,NULL, NULL,StepResponsesEnum);5686 this->parameters->FindParam(&responses,NULL,InversionCostFunctionsEnum); 5982 5687 Input* weights_input = inputs->GetInput(InversionCostFunctionsCoefficientsEnum); _assert_(weights_input); 5983 5688 Input* vx_input = inputs->GetInput(VxEnum); _assert_(vx_input); … … 6014 5719 for(resp=0;resp<num_responses;resp++){ 6015 5720 6016 weights_input->GetInputValue(&weight,gauss,resp );5721 weights_input->GetInputValue(&weight,gauss,responses[resp]); 6017 5722 6018 5723 switch(responses[resp]){ … … 6160 5865 /*dJ/dalpha = ∂L/∂alpha = ∂J/∂alpha + ∂/∂alpha(KU-F)*/ 6161 5866 6162 int i,approximation;6163 Tria* 5867 int approximation; 5868 Tria* tria=NULL; 6164 5869 6165 5870 /*If on water, skip grad (=0): */ 6166 5871 if(NoIceInElement())return; 6167 5872 6168 5873 /*First deal with ∂/∂alpha(KU-F)*/ 6169 5874 switch(control_type){ … … 6214 5919 6215 5920 /*Now deal with ∂J/∂alpha*/ 6216 int 6217 int 5921 int *responses = NULL; 5922 int num_responses,resp; 6218 5923 this->parameters->FindParam(&num_responses,InversionNumCostFunctionsEnum); 6219 this->parameters->FindParam(&responses,NULL, NULL,StepResponsesEnum);5924 this->parameters->FindParam(&responses,NULL,InversionCostFunctionsEnum); 6220 5925 6221 5926 for(resp=0;resp<num_responses;resp++) switch(responses[resp]){ … … 6232 5937 if(IsOnBed()){ 6233 5938 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6234 tria->GradjDragGradient(gradient, resp,control_index);5939 tria->GradjDragGradient(gradient,control_index); 6235 5940 delete tria->material; delete tria; 6236 5941 } … … 6239 5944 if(IsOnBed()){ 6240 5945 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6241 tria->GradjBGradient(gradient, resp,control_index);5946 tria->GradjBGradient(gradient,control_index); 6242 5947 delete tria->material; delete tria; 6243 5948 } … … 6432 6137 /*Depth Average B*/ 6433 6138 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 6139 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 6434 6140 6435 6141 /*Collapse element to the base*/ … … 6440 6146 /*delete Average B*/ 6441 6147 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 6148 this->material->inputs->DeleteInput(DamageDbarEnum); 6442 6149 6443 6150 } /*}}}*/ … … 6448 6155 if (!IsOnBed()) return; 6449 6156 6450 /*Depth Average B */6157 /*Depth Average B and D*/ 6451 6158 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 6159 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 6452 6160 6453 6161 /*Collapse element to the base*/ … … 6458 6166 /*delete Average B*/ 6459 6167 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 6168 this->material->inputs->DeleteInput(DamageDbarEnum); 6460 6169 } /*}}}*/ 6461 6170 /*FUNCTION Penta::GradjBbarFS {{{*/ … … 6465 6174 if (!IsOnBed()) return; 6466 6175 6467 /*Depth Average B */6176 /*Depth Average B and D*/ 6468 6177 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 6178 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 6469 6179 6470 6180 /*Collapse element to the base*/ … … 6475 6185 /*delete Average B*/ 6476 6186 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 6187 this->material->inputs->DeleteInput(DamageDbarEnum); 6477 6188 } /*}}}*/ 6478 6189 /*FUNCTION Penta::InputControlUpdate{{{*/ … … 6494 6205 input=(Input*)material->inputs->GetInput(MaterialsRheologyBEnum); _assert_(input); 6495 6206 } 6496 else if(control_type[i]== MaterialsRheologyZbarEnum){6207 else if(control_type[i]==DamageDbarEnum){ 6497 6208 if (!IsOnBed()) goto cleanup_and_return; 6498 input=(Input*)material->inputs->GetInput( MaterialsRheologyZEnum); _assert_(input);6209 input=(Input*)material->inputs->GetInput(DamageDEnum); _assert_(input); 6499 6210 } 6500 6211 else{ … … 6511 6222 this->InputExtrude(MaterialsRheologyBEnum,MaterialsEnum); 6512 6223 } 6513 else if(control_type[i]== MaterialsRheologyZbarEnum){6514 this->InputExtrude( MaterialsRheologyZEnum,MaterialsEnum);6224 else if(control_type[i]==DamageDbarEnum){ 6225 this->InputExtrude(DamageDEnum,MaterialsEnum); 6515 6226 } 6516 6227 } … … 6528 6239 int* pdoflist=NULL; 6529 6240 IssmDouble FSreconditioning; 6530 GaussPenta *gauss;6531 6241 6532 6242 /*Fetch number of nodes and dof for this finite element*/ … … 6622 6332 /*}}}*/ 6623 6333 /*FUNCTION Penta::SurfaceAverageVelMisfit {{{*/ 6624 IssmDouble Penta::SurfaceAverageVelMisfit( int weight_index){6334 IssmDouble Penta::SurfaceAverageVelMisfit(void){ 6625 6335 6626 6336 int approximation; … … 6645 6355 * and compute SurfaceAverageVelMisfit*/ 6646 6356 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6647 J=tria->SurfaceAverageVelMisfit( weight_index);6357 J=tria->SurfaceAverageVelMisfit(); 6648 6358 delete tria->material; delete tria; 6649 6359 return J; … … 6652 6362 6653 6363 tria=(Tria*)SpawnTria(1); //lower face is 0, upper face is 1. 6654 J=tria->SurfaceAverageVelMisfit( weight_index);6364 J=tria->SurfaceAverageVelMisfit(); 6655 6365 delete tria->material; delete tria; 6656 6366 return J; … … 6659 6369 /*}}}*/ 6660 6370 /*FUNCTION Penta::SurfaceAbsVelMisfit {{{*/ 6661 IssmDouble Penta::SurfaceAbsVelMisfit( int weight_index){6371 IssmDouble Penta::SurfaceAbsVelMisfit(void){ 6662 6372 6663 6373 int approximation; … … 6682 6392 * and compute SurfaceAbsVelMisfit*/ 6683 6393 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6684 J=tria->SurfaceAbsVelMisfit( weight_index);6394 J=tria->SurfaceAbsVelMisfit(); 6685 6395 delete tria->material; delete tria; 6686 6396 return J; … … 6689 6399 6690 6400 tria=(Tria*)SpawnTria(1); //lower face is 0, upper face is 1. 6691 J=tria->SurfaceAbsVelMisfit( weight_index);6401 J=tria->SurfaceAbsVelMisfit(); 6692 6402 delete tria->material; delete tria; 6693 6403 return J; … … 6696 6406 /*}}}*/ 6697 6407 /*FUNCTION Penta::SurfaceLogVelMisfit {{{*/ 6698 IssmDouble Penta::SurfaceLogVelMisfit( int weight_index){6408 IssmDouble Penta::SurfaceLogVelMisfit(void){ 6699 6409 6700 6410 int approximation; … … 6719 6429 * and compute SurfaceLogVelMisfit*/ 6720 6430 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6721 J=tria->SurfaceLogVelMisfit( weight_index);6431 J=tria->SurfaceLogVelMisfit(); 6722 6432 delete tria->material; delete tria; 6723 6433 return J; … … 6726 6436 6727 6437 tria=(Tria*)SpawnTria(1); //lower face is 0, upper face is 1. 6728 J=tria->SurfaceLogVelMisfit( weight_index);6438 J=tria->SurfaceLogVelMisfit(); 6729 6439 delete tria->material; delete tria; 6730 6440 return J; … … 6733 6443 /*}}}*/ 6734 6444 /*FUNCTION Penta::SurfaceLogVxVyMisfit {{{*/ 6735 IssmDouble Penta::SurfaceLogVxVyMisfit( int weight_index){6445 IssmDouble Penta::SurfaceLogVxVyMisfit(void){ 6736 6446 6737 6447 IssmDouble J; … … 6758 6468 * and compute SurfaceLogVxVyMisfit*/ 6759 6469 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6760 J=tria->SurfaceLogVxVyMisfit( weight_index);6470 J=tria->SurfaceLogVxVyMisfit(); 6761 6471 delete tria->material; delete tria; 6762 6472 return J; … … 6765 6475 6766 6476 tria=(Tria*)SpawnTria(1); //lower face is 0, upper face is 1. 6767 J=tria->SurfaceLogVxVyMisfit( weight_index);6477 J=tria->SurfaceLogVxVyMisfit(); 6768 6478 delete tria->material; delete tria; 6769 6479 return J; … … 6772 6482 /*}}}*/ 6773 6483 /*FUNCTION Penta::SurfaceRelVelMisfit {{{*/ 6774 IssmDouble Penta::SurfaceRelVelMisfit( int weight_index){6484 IssmDouble Penta::SurfaceRelVelMisfit(void){ 6775 6485 6776 6486 int approximation; … … 6795 6505 * and compute SurfaceRelVelMisfit*/ 6796 6506 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1. 6797 J=tria->SurfaceRelVelMisfit( weight_index);6507 J=tria->SurfaceRelVelMisfit(); 6798 6508 delete tria->material; delete tria; 6799 6509 return J; … … 6802 6512 6803 6513 tria=(Tria*)SpawnTria(1); //lower face is 0, upper face is 1. 6804 J=tria->SurfaceRelVelMisfit( weight_index);6514 J=tria->SurfaceRelVelMisfit(); 6805 6515 delete tria->material; delete tria; 6806 6516 return J; … … 6809 6519 /*}}}*/ 6810 6520 /*FUNCTION Penta::ThicknessAbsGradient{{{*/ 6811 IssmDouble Penta::ThicknessAbsGradient( int weight_index){6521 IssmDouble Penta::ThicknessAbsGradient(void){ 6812 6522 6813 6523 _error_("Not implemented yet"); … … 6815 6525 /*}}}*/ 6816 6526 /*FUNCTION Penta::ThicknessAbsMisfit {{{*/ 6817 IssmDouble Penta::ThicknessAbsMisfit( int weight_index){6527 IssmDouble Penta::ThicknessAbsMisfit(void){ 6818 6528 6819 6529 int approximation; … … 6829 6539 6830 6540 tria=(Tria*)SpawnTria(0); 6831 J=tria->ThicknessAbsMisfit( weight_index);6541 J=tria->ThicknessAbsMisfit(); 6832 6542 delete tria->material; delete tria; 6833 6543 return J; … … 6835 6545 /*}}}*/ 6836 6546 /*FUNCTION Penta::DragCoefficientAbsGradient{{{*/ 6837 IssmDouble Penta::DragCoefficientAbsGradient( int weight_index){6547 IssmDouble Penta::DragCoefficientAbsGradient(void){ 6838 6548 6839 6549 IssmDouble J; … … 6844 6554 6845 6555 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1 6846 J=tria->DragCoefficientAbsGradient( weight_index);6556 J=tria->DragCoefficientAbsGradient(); 6847 6557 delete tria->material; delete tria; 6848 6558 return J; … … 6850 6560 /*}}}*/ 6851 6561 /*FUNCTION Penta::RheologyBbarAbsGradient{{{*/ 6852 IssmDouble Penta::RheologyBbarAbsGradient( int weight_index){6562 IssmDouble Penta::RheologyBbarAbsGradient(void){ 6853 6563 6854 6564 IssmDouble J; … … 6859 6569 6860 6570 tria=(Tria*)SpawnTria(0); //lower face is 0, upper face is 1 6861 J=tria->RheologyBbarAbsGradient( weight_index);6571 J=tria->RheologyBbarAbsGradient(); 6862 6572 delete tria->material; delete tria; 6863 6573 return J; … … 7592 7302 ElementMatrix* Ke=new ElementMatrix(Ke1,Ke2); 7593 7303 delete Ke1; delete Ke2; 7594 7304 7595 7305 /*Compute HO Matrix with P1 element type\n");*/ 7596 7306 this->element_type=P1Enum; … … 7724 7434 case MaticeEnum: 7725 7435 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 7726 break; 7727 case MatdamageiceEnum: 7728 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 7729 this->InputDepthAverageAtBase(MaterialsRheologyZEnum,MaterialsRheologyZbarEnum,MaterialsEnum); 7436 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 7730 7437 break; 7731 7438 default: … … 7740 7447 /*Delete averaged fields*/ 7741 7448 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 7742 this->material->inputs->DeleteInput( MaterialsRheologyZbarEnum);7449 this->material->inputs->DeleteInput(DamageDbarEnum); 7743 7450 7744 7451 /*clean up and return*/ … … 7925 7632 int i,j; 7926 7633 IssmDouble Jdet,viscosity; 7927 IssmDouble epsilon[5]; /* epsilon=[exx,eyy,exy,exz,eyz];*/7928 7634 IssmDouble xyz_list[NUMVERTICES][3]; 7929 7635 IssmDouble B[3][numdof2d]; … … 8020 7726 8021 7727 /*Intermediaries */ 8022 int i,j;8023 7728 int approximation; 8024 7729 IssmDouble xyz_list[NUMVERTICES][3]; … … 8065 7770 8066 7771 D_scalar=2*newviscosity*gauss->weight*Jdet; 8067 for (i=0;i<5;i++) D[i*5+i]=D_scalar;7772 for(int i=0;i<5;i++) D[i*5+i]=D_scalar; 8068 7773 8069 7774 TripleMultiply(B,5,numdof,1, … … 8221 7926 8222 7927 /*Intermediaries */ 8223 int i,j ,approximation;7928 int i,j; 8224 7929 IssmDouble Jdet,viscosity,FSreconditioning,diameter,rigidity; 8225 7930 IssmDouble xyz_list[NUMVERTICES][3]; 8226 7931 IssmDouble epsilon[6]; /* epsilon=[exx,eyy,ezz,exy,exz,eyz];*/ 8227 IssmDouble B[8][24];8228 IssmDouble B_prime[8][24];8229 IssmDouble B_stab[3][NUMVERTICES];8230 IssmDouble D_scalar,D_scalar_stab;8231 IssmDouble D[8][8]={0.0};8232 IssmDouble D_stab[3][3]={0.0};8233 IssmDouble Ke_temp[24][24]={0.0}; //for the six nodes8234 IssmDouble Ke_temp_stab[6][6]={0.0}; //for the six nodes8235 7932 GaussPenta *gauss=NULL; 8236 7933 8237 7934 /*Stabilization*/ 8238 bool stabilization = true;7935 IssmDouble D_scalar; 8239 7936 IssmDouble dbasis[3][6]; 8240 7937 IssmDouble dmu[3]; … … 8281 7978 8282 7979 D_scalar=gauss->weight*Jdet; 8283 8284 7980 8285 7981 /*Add stabilization*/ … … 8333 8029 8334 8030 /*Intermediaries */ 8335 int i, j,approximation;8031 int i,approximation; 8336 8032 IssmDouble Jdet,viscosity,FSreconditioning,D_scalar; 8337 8033 IssmDouble xyz_list[NUMVERTICES][3]; … … 8408 8104 int analysis_type,approximation; 8409 8105 IssmDouble alpha2,Jdet2d; 8410 IssmDouble FSreconditioning,viscosity;8411 IssmDouble epsilon[6]; /* epsilon=[exx,eyy,ezz,exy,exz,eyz];*/8412 8106 IssmDouble xyz_list[NUMVERTICES][3]; 8413 8107 IssmDouble xyz_list_tria[NUMVERTICES2D][3]; … … 8435 8129 GetVerticesCoordinates(&xyz_list[0][0],vertices,NUMVERTICES); 8436 8130 parameters->FindParam(&analysis_type,AnalysisTypeEnum); 8437 parameters->FindParam(&FSreconditioning,StressbalanceFSreconditioningEnum);8438 8131 Input* vx_input=inputs->GetInput(VxEnum); _assert_(vx_input); 8439 8132 Input* vy_input=inputs->GetInput(VyEnum); _assert_(vy_input); … … 8452 8145 GetTriaJacobianDeterminant(&Jdet2d, &xyz_list_tria[0][0],gauss); 8453 8146 GetLFS(BFriction,gauss); 8454 8455 this->GetStrainRate3d(&epsilon[0],&xyz_list[0][0],gauss,vx_input,vy_input,vz_input);8456 material->GetViscosity3dFS(&viscosity,&epsilon[0]);8457 8147 8458 8148 friction->GetAlpha2(&alpha2,gauss,VxEnum,VyEnum,VzEnum); … … 8598 8288 8599 8289 /*Intermediaries */ 8600 int i,j; 8601 int approximation; 8290 int i,approximation; 8602 8291 IssmDouble viscosity,Jdet; 8603 8292 IssmDouble FSreconditioning; … … 8676 8365 IssmDouble xyz_list[NUMVERTICES][3]; 8677 8366 IssmDouble basis[6]; //for the six nodes of the penta 8678 Tria* tria=NULL;8679 8367 Friction* friction=NULL; 8680 8368 GaussPenta *gauss=NULL; … … 8839 8527 IssmDouble xyz_list[NUMVERTICES][3]; 8840 8528 IssmDouble basis[6]; //for the six nodes of the penta 8841 Tria* tria=NULL;8842 8529 Friction* friction=NULL; 8843 8530 GaussPenta *gauss=NULL; … … 9137 8824 9138 8825 /*Intermediaries*/ 9139 int i,j;9140 8826 IssmDouble Jdet; 9141 8827 IssmDouble slope[3]; //do not put 2! this goes into GetInputDerivativeValue, which addresses slope[3] also! … … 9170 8856 driving_stress_baseline=matpar->GetRhoIce()*matpar->GetG(); 9171 8857 9172 for(i =0;i<numnodes;i++){8858 for(int i=0;i<numnodes;i++){ 9173 8859 pe->values[i*NDOF2+0]+= -driving_stress_baseline*slope[0]*Jdet*gauss->weight*basis[i]; 9174 8860 pe->values[i*NDOF2+1]+= -driving_stress_baseline*slope[1]*Jdet*gauss->weight*basis[i]; … … 9382 9068 /*Intermediaries*/ 9383 9069 int i,j; 9384 int approximation;9385 9070 IssmDouble Jdet,gravity,rho_ice,B,D_scalar_stab,viscosity; 9386 9071 IssmDouble forcex,forcey,forcez,diameter,FSreconditioning; 9387 9072 IssmDouble xyz_list[NUMVERTICES][3]; 9388 9073 IssmDouble epsilon[6]; /* epsilon=[exx,eyy,ezz,exy,exz,eyz];*/ 9389 IssmDouble l1l6[6]; //for the six nodes and the bubble9390 IssmDouble dh1dh6[3][NUMVERTICES];9391 9074 GaussPenta *gauss=NULL; 9392 9075 9393 9076 /*Stabilization*/ 9394 bool stabilization = true;9395 9077 IssmDouble dbasis[3][6]; 9396 9078 IssmDouble dmu[3]; … … 9398 9080 IssmDouble dnodalbasis[6][6][3]; 9399 9081 IssmDouble SW[6][4][4]; 9400 int p, q,ii;9082 int p,ii; 9401 9083 int c=3; //index of pressure 9402 9084 … … 9432 9114 9433 9115 GetJacobianDeterminant(&Jdet, &xyz_list[0][0],gauss); 9434 GetNodalFunctionsP1(&l1l6[0], gauss);9435 9116 this->GetStrainRate3d(&epsilon[0],&xyz_list[0][0],gauss,vx_input,vy_input,vz_input); 9436 9117 material->GetViscosity3dFS(&viscosity,&epsilon[0]); … … 9473 9154 9474 9155 /*Intermediaries*/ 9475 int i ,j;9156 int i; 9476 9157 int approximation; 9477 9158 IssmDouble Jdet,gravity,rho_ice; … … 9786 9467 case MaticeEnum: 9787 9468 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 9788 break; 9789 case MatdamageiceEnum: 9790 this->InputDepthAverageAtBase(MaterialsRheologyBEnum,MaterialsRheologyBbarEnum,MaterialsEnum); 9791 this->InputDepthAverageAtBase(MaterialsRheologyZEnum,MaterialsRheologyZbarEnum,MaterialsEnum); 9469 this->InputDepthAverageAtBase(DamageDEnum,DamageDbarEnum,MaterialsEnum); 9792 9470 break; 9793 9471 default: … … 9802 9480 /*Delete averaged inputs*/ 9803 9481 this->material->inputs->DeleteInput(MaterialsRheologyBbarEnum); 9804 this->material->inputs->DeleteInput( MaterialsRheologyZbarEnum);9482 this->material->inputs->DeleteInput(DamageDbarEnum); 9805 9483 9806 9484 /*clean up and return*/ … … 9810 9488 /*FUNCTION Penta::CreateJacobianStressbalanceHO{{{*/ 9811 9489 ElementMatrix* Penta::CreateJacobianStressbalanceHO(void){ 9812 9813 /*Constants*/9814 const int numdof=NDOF2*NUMVERTICES;9815 9490 9816 9491 /*Intermediaries */ … … 9876 9551 9877 9552 /*Intermediaries */ 9878 int i,j ,approximation;9553 int i,j; 9879 9554 IssmDouble xyz_list[NUMVERTICES][3]; 9880 9555 IssmDouble Jdet; … … 10032 9707 values[i*NDOF2+0]=vx; 10033 9708 values[i*NDOF2+1]=vy; 10034 }10035 10036 /*Add value to global vector*/10037 solution->SetValues(numdof,doflist,values,INS_VAL);10038 10039 /*Free ressources:*/10040 delete gauss;10041 xDelete<int>(doflist);10042 }10043 /*}}}*/10044 /*FUNCTION Penta::GetSolutionFromInputsStressbalanceVert{{{*/10045 void Penta::GetSolutionFromInputsStressbalanceVert(Vector<IssmDouble>* solution){10046 10047 const int numdof=NDOF1*NUMVERTICES;10048 10049 int i;10050 int* doflist=NULL;10051 IssmDouble vz;10052 IssmDouble values[numdof];10053 GaussPenta* gauss=NULL;10054 10055 /*Get dof list: */10056 GetDofList(&doflist,NoneApproximationEnum,GsetEnum);10057 Input* vz_input=inputs->GetInput(VzEnum); _assert_(vz_input);10058 10059 /*Ok, we have vx and vy in values, fill in vx and vy arrays: */10060 /*P1 element only for now*/10061 gauss=new GaussPenta();10062 for(i=0;i<NUMVERTICES;i++){10063 /*Recover vz */10064 gauss->GaussVertex(i);10065 vz_input->GetInputValue(&vz,gauss);10066 values[i]=vz;10067 9709 } 10068 9710 … … 10412 10054 void Penta::InputUpdateFromSolutionStressbalanceSSAFS(IssmDouble* solution){ 10413 10055 10414 const int numdof m=NDOF2*NUMVERTICES;10415 const int numdof s=NDOF3*NUMVERTICES;10416 const int numdof 2d=NDOF2*NUMVERTICES2D;10417 const int numdof pressure=NDOF1*NUMVERTICES;10056 const int numdofSSA = NDOF2*NUMVERTICES; 10057 const int numdof2d = NDOF2*NUMVERTICES2D; 10058 const int numdofFSv = NDOF3*NUMVERTICES; 10059 const int numdofFSp = NDOF1*NUMVERTICES; 10418 10060 10419 10061 int i; 10420 10062 IssmDouble FSreconditioning; 10421 IssmDouble SSA_values[numdofm]; 10422 IssmDouble FS_values[numdofs]; 10423 IssmDouble Pressure_values[numdofs]; 10063 IssmDouble SSA_values[numdofSSA]; 10064 IssmDouble FS_values[numdofFSv+numdofFSp]; 10424 10065 IssmDouble vx[NUMVERTICES]; 10425 10066 IssmDouble vy[NUMVERTICES]; … … 10430 10071 IssmDouble pressure[NUMVERTICES]; 10431 10072 IssmDouble xyz_list[NUMVERTICES][3]; 10432 int* doflistm = NULL; 10433 int* doflists = NULL; 10434 int* doflistpressure = NULL; 10435 Penta *penta = NULL; 10073 int *doflistSSA = NULL; 10074 int *doflistFSv = NULL; 10075 int *doflistFSp = NULL; 10076 Penta *penta = NULL; 10077 10078 /*Prepare coordinate system list*/ 10079 int* cs_list = xNew<int>(2*NUMVERTICES); 10080 for(i=0;i<NUMVERTICES;i++) cs_list[i] = XYZEnum; 10081 for(i=0;i<NUMVERTICES;i++) cs_list[NUMVERTICES+i] = PressureEnum; 10436 10082 10437 10083 /*OK, we have to add results of this element for SSA … … 10440 10086 10441 10087 /*Get dof listof this element (SSA dofs) and of the penta at base (SSA dofs): */ 10442 penta->GetDofList(&doflist m,SSAApproximationEnum,GsetEnum);10443 GetDofList(&doflist s,FSvelocityEnum,GsetEnum);10444 GetDofListPressure(&doflist pressure,GsetEnum);10088 penta->GetDofList(&doflistSSA,SSAApproximationEnum,GsetEnum); 10089 GetDofList(&doflistFSv,FSvelocityEnum,GsetEnum); 10090 GetDofListPressure(&doflistFSp,GsetEnum); 10445 10091 this->parameters->FindParam(&FSreconditioning,StressbalanceFSreconditioningEnum); 10446 10092 … … 10450 10096 /*Use the dof list to index into the solution vector: */ 10451 10097 for(i=0;i<numdof2d;i++){ 10452 SSA_values[i]=solution[doflist m[i]];10453 SSA_values[i+numdof2d]=solution[doflist m[i]];10454 } 10455 for(i=0;i<numdof s;i++)FS_values[i]=solution[doflists[i]];10456 for(i=0;i<numdof pressure;i++) Pressure_values[i]=solution[doflistpressure[i]];10098 SSA_values[i]=solution[doflistSSA[i]]; 10099 SSA_values[i+numdof2d]=solution[doflistSSA[i]]; 10100 } 10101 for(i=0;i<numdofFSv;i++) FS_values[i]=solution[doflistFSv[i]]; 10102 for(i=0;i<numdofFSp;i++) FS_values[numdofFSv+i]=solution[doflistFSp[i]]; 10457 10103 10458 10104 /*Transform solution in Cartesian Space*/ 10459 10105 TransformSolutionCoord(&SSA_values[0],this->nodes,NUMVERTICES,XYEnum); 10460 TransformSolutionCoord(&FS_values[0],this->nodes, NUMVERTICES,XYZEnum);10106 TransformSolutionCoord(&FS_values[0],this->nodes,2*NUMVERTICES,cs_list); 10461 10107 10462 10108 /*Ok, we have vx and vy in values, fill in vx and vy arrays: */ 10463 10109 for(i=0;i<NUMVERTICES;i++){ 10464 vx[i] =FS_values[i*NDOF3+0]+SSA_values[i*NDOF2+0];10465 vy[i] =FS_values[i*NDOF3+1]+SSA_values[i*NDOF2+1];10466 vzFS[i] =FS_values[i*NDOF3+2];10467 pressure[i] =Pressure_values[i*NDOF1]*FSreconditioning;10110 vx[i] = FS_values[i*NDOF3+0]+SSA_values[i*NDOF2+0]; 10111 vy[i] = FS_values[i*NDOF3+1]+SSA_values[i*NDOF2+1]; 10112 vzFS[i] = FS_values[i*NDOF3+2]; 10113 pressure[i] = FS_values[NUMVERTICES*NDOF3+i]*FSreconditioning; 10468 10114 10469 10115 /*Check solution*/ 10470 10116 if(xIsNan<IssmDouble>(vx[i])) _error_("NaN found in solution vector"); 10471 10117 if(xIsNan<IssmDouble>(vy[i])) _error_("NaN found in solution vector"); 10472 if(xIsNan<IssmDouble>(vzFS[i])) _error_("NaN found in solution vector");10118 if(xIsNan<IssmDouble>(vzFS[i])) _error_("NaN found in solution vector"); 10473 10119 if(xIsNan<IssmDouble>(pressure[i])) _error_("NaN found in solution vector"); 10474 10120 } … … 10488 10134 /*Now Compute vel*/ 10489 10135 for(i=0;i<NUMVERTICES;i++) { 10490 vz[i] =vzSSA[i]+vzFS[i];10491 vel[i] =pow( pow(vx[i],2.0) + pow(vy[i],2.0) + pow(vz[i],2.0) , 0.5);10136 vz[i] = vzSSA[i]+vzFS[i]; 10137 vel[i] = sqrt(vx[i]*vx[i] + vy[i]*vy[i] + vz[i]*vz[i]); 10492 10138 } 10493 10139 … … 10508 10154 10509 10155 /*Free ressources:*/ 10510 xDelete<int>(doflistm); 10511 xDelete<int>(doflists); 10512 xDelete<int>(doflistpressure); 10156 xDelete<int>(doflistSSA); 10157 xDelete<int>(doflistFSv); 10158 xDelete<int>(doflistFSp); 10159 xDelete<int>(cs_list); 10513 10160 } 10514 10161 /*}}}*/ … … 10670 10317 void Penta::InputUpdateFromSolutionStressbalanceHOFS(IssmDouble* solution){ 10671 10318 10672 const int numdof p=NDOF2*NUMVERTICES;10673 const int numdof s=NDOF3*NUMVERTICES;10674 const int numdof pressure=NDOF1*NUMVERTICES;10319 const int numdofHO = NDOF2*NUMVERTICES; 10320 const int numdofFSv = NDOF3*NUMVERTICES; 10321 const int numdofFSp = NDOF1*NUMVERTICES; 10675 10322 10676 10323 int i; 10677 IssmDouble HO_values[numdofp]; 10678 IssmDouble FS_values[numdofs]; 10679 IssmDouble Pressure_values[numdofpressure]; 10324 IssmDouble HO_values[numdofHO]; 10325 IssmDouble FS_values[numdofFSv+numdofFSp]; 10680 10326 IssmDouble vx[NUMVERTICES]; 10681 10327 IssmDouble vy[NUMVERTICES]; … … 10687 10333 IssmDouble xyz_list[NUMVERTICES][3]; 10688 10334 IssmDouble FSreconditioning; 10689 int* doflist p= NULL;10690 int* doflist s= NULL;10691 int* doflist pressure= NULL;10692 Penta *penta = NULL; 10693 10694 /*OK, we have to add results of this element for HO10695 * and results from the penta at base for SSA. Now recover results*/10696 penta=GetBasalElement();10335 int* doflistHO = NULL; 10336 int* doflistFSv = NULL; 10337 int* doflistFSp = NULL; 10338 10339 /*Prepare coordinate system list*/ 10340 int* cs_list = xNew<int>(2*NUMVERTICES); 10341 for(i=0;i<NUMVERTICES;i++) cs_list[i] = XYZEnum; 10342 for(i=0;i<NUMVERTICES;i++) cs_list[NUMVERTICES+i] = PressureEnum; 10697 10343 10698 10344 /*Get dof listof this element (HO dofs) and of the penta at base (SSA dofs): */ 10699 GetDofList(&doflist p,HOApproximationEnum,GsetEnum);10700 GetDofList(&doflist s,FSvelocityEnum,GsetEnum);10701 GetDofListPressure(&doflist pressure,GsetEnum);10345 GetDofList(&doflistHO,HOApproximationEnum,GsetEnum); 10346 GetDofList(&doflistFSv,FSvelocityEnum,GsetEnum); 10347 GetDofListPressure(&doflistFSp,GsetEnum); 10702 10348 this->parameters->FindParam(&FSreconditioning,StressbalanceFSreconditioningEnum); 10703 10349 … … 10706 10352 10707 10353 /*Use the dof list to index into the solution vector: */ 10708 for(i=0;i<numdof p;i++) HO_values[i]=solution[doflistp[i]];10709 for(i=0;i<numdof s;i++) FS_values[i]=solution[doflists[i]];10710 for(i=0;i<numdof pressure;i++) Pressure_values[i]=solution[doflistpressure[i]];10354 for(i=0;i<numdofHO;i++) HO_values[i]=solution[doflistHO[i]]; 10355 for(i=0;i<numdofFSv;i++) FS_values[i]=solution[doflistFSv[i]]; 10356 for(i=0;i<numdofFSp;i++) FS_values[numdofFSv+i]=solution[doflistFSp[i]]; 10711 10357 10712 10358 /*Transform solution in Cartesian Space*/ 10713 10359 TransformSolutionCoord(&HO_values[0],this->nodes,NUMVERTICES,XYEnum); 10714 TransformSolutionCoord(&FS_values[0],this->nodes, NUMVERTICES,XYZEnum);10360 TransformSolutionCoord(&FS_values[0],this->nodes,2*NUMVERTICES,cs_list); 10715 10361 10716 10362 /*Ok, we have vx and vy in values, fill in vx and vy arrays: */ 10717 10363 for(i=0;i<NUMVERTICES;i++){ 10718 vx[i] =FS_values[i*NDOF3+0]+HO_values[i*NDOF2+0];10719 vy[i] =FS_values[i*NDOF3+1]+HO_values[i*NDOF2+1];10720 vzFS[i] =FS_values[i*NDOF3+2];10721 pressure[i] =Pressure_values[i*NDOF1]*FSreconditioning;10364 vx[i] = FS_values[i*NDOF3+0]+HO_values[i*NDOF2+0]; 10365 vy[i] = FS_values[i*NDOF3+1]+HO_values[i*NDOF2+1]; 10366 vzFS[i] = FS_values[i*NDOF3+2]; 10367 pressure[i] = FS_values[NUMVERTICES*NDOF3+i]*FSreconditioning; 10722 10368 10723 10369 /*Check solution*/ 10724 10370 if(xIsNan<IssmDouble>(vx[i])) _error_("NaN found in solution vector"); 10725 10371 if(xIsNan<IssmDouble>(vy[i])) _error_("NaN found in solution vector"); 10726 if(xIsNan<IssmDouble>(vzFS[i])) _error_("NaN found in solution vector");10372 if(xIsNan<IssmDouble>(vzFS[i])) _error_("NaN found in solution vector"); 10727 10373 if(xIsNan<IssmDouble>(pressure[i])) _error_("NaN found in solution vector"); 10728 10374 } … … 10742 10388 /*Now Compute vel*/ 10743 10389 for(i=0;i<NUMVERTICES;i++) { 10744 vz[i] =vzHO[i]+vzFS[i];10745 vel[i] =pow( pow(vx[i],2.0) + pow(vy[i],2.0) + pow(vz[i],2.0) , 0.5);10390 vz[i] = vzHO[i]+vzFS[i]; 10391 vel[i] = sqrt(vx[i]*vx[i] + vy[i]*vy[i] + vz[i]*vz[i]); 10746 10392 } 10747 10393 … … 10762 10408 10763 10409 /*Free ressources:*/ 10764 xDelete<int>(doflistp); 10765 xDelete<int>(doflists); 10766 xDelete<int>(doflistpressure); 10410 xDelete<int>(doflistHO); 10411 xDelete<int>(doflistFSv); 10412 xDelete<int>(doflistFSp); 10413 xDelete<int>(cs_list); 10767 10414 } 10768 10415 /*}}}*/ … … 10942 10589 int* pdoflist=NULL; 10943 10590 IssmDouble FSreconditioning; 10944 GaussPenta *gauss;10945 10591 10946 10592 /*Fetch number of nodes and dof for this finite element*/ … … 10951 10597 10952 10598 /*Initialize values*/ 10953 IssmDouble* vvalues = xNew<IssmDouble>(vnumdof); 10954 IssmDouble* pvalues = xNew<IssmDouble>(pnumdof); 10599 IssmDouble* values = xNew<IssmDouble>(vnumdof+pnumdof); 10955 10600 IssmDouble* vx = xNew<IssmDouble>(vnumnodes); 10956 10601 IssmDouble* vy = xNew<IssmDouble>(vnumnodes); 10957 10602 IssmDouble* vz = xNew<IssmDouble>(vnumnodes); 10958 10603 IssmDouble* vel = xNew<IssmDouble>(vnumnodes); 10959 IssmDouble* pressure = xNew<IssmDouble>(vnumnodes); 10604 IssmDouble* pressure = xNew<IssmDouble>(pnumnodes); 10605 10606 /*Prepare coordinate system list*/ 10607 int* cs_list = xNew<int>(vnumnodes+pnumnodes); 10608 for(i=0;i<vnumnodes;i++) cs_list[i] = XYZEnum; 10609 for(i=0;i<pnumnodes;i++) cs_list[vnumnodes+i] = PressureEnum; 10960 10610 10961 10611 /*Get dof list: */ … … 10964 10614 10965 10615 /*Use the dof list to index into the solution vector: */ 10966 for(i=0;i<vnumdof;i++) v values[i]=solution[vdoflist[i]];10967 for(i=0;i<pnumdof;i++) pvalues[i]=solution[pdoflist[i]];10616 for(i=0;i<vnumdof;i++) values[i] =solution[vdoflist[i]]; 10617 for(i=0;i<pnumdof;i++) values[vnumdof+i]=solution[pdoflist[i]]; 10968 10618 10969 10619 /*Transform solution in Cartesian Space*/ 10970 TransformSolutionCoord(&v values[0],nodes,vnumnodes,XYZEnum);10620 TransformSolutionCoord(&values[0],nodes,vnumnodes+pnumdof,cs_list); 10971 10621 10972 10622 /*Ok, we have vx and vy in values, fill in all arrays: */ 10973 10623 for(i=0;i<vnumnodes;i++){ 10974 vx[i] = v values[i*NDOF3+0];10975 vy[i] = v values[i*NDOF3+1];10976 vz[i] = v values[i*NDOF3+2];10624 vx[i] = values[i*NDOF3+0]; 10625 vy[i] = values[i*NDOF3+1]; 10626 vz[i] = values[i*NDOF3+2]; 10977 10627 if(xIsNan<IssmDouble>(vx[i])) _error_("NaN found in solution vector"); 10978 10628 if(xIsNan<IssmDouble>(vy[i])) _error_("NaN found in solution vector"); … … 10980 10630 } 10981 10631 for(i=0;i<pnumnodes;i++){ 10982 pressure[i] = pvalues[i];10632 pressure[i] = values[vnumdof+i]; 10983 10633 if(xIsNan<IssmDouble>(pressure[i])) _error_("NaN found in solution vector"); 10984 10634 } … … 10986 10636 /*Recondition pressure and compute vel: */ 10987 10637 this->parameters->FindParam(&FSreconditioning,StressbalanceFSreconditioningEnum); 10988 for(i =0;i<pnumnodes;i++) pressure[i]=pressure[i]*FSreconditioning;10989 for(i =0;i<vnumnodes;i++) vel[i]=pow( pow(vx[i],2.0) + pow(vy[i],2.0) + pow(vz[i],2.0) , 0.5);10638 for(i = 0;i<pnumnodes;i++) pressure[i] = pressure[i]*FSreconditioning; 10639 for(i = 0;i<vnumnodes;i++) vel[i] = sqrt(vx[i]*vx[i] + vy[i]*vy[i] + vz[i]*vz[i]); 10990 10640 10991 10641 /*Now, we have to move the previous inputs to old … … 11009 10659 xDelete<IssmDouble>(vy); 11010 10660 xDelete<IssmDouble>(vx); 11011 xDelete<IssmDouble>(vvalues); 11012 xDelete<IssmDouble>(pvalues); 10661 xDelete<IssmDouble>(values); 11013 10662 xDelete<int>(vdoflist); 11014 10663 xDelete<int>(pdoflist); 10664 xDelete<int>(cs_list); 11015 10665 } 11016 10666 /*}}}*/ … … 11278 10928 IssmDouble h[NUMVERTICES],s[NUMVERTICES],b[NUMVERTICES],r[NUMVERTICES]; 11279 10929 IssmDouble melting[NUMVERTICES],phi[NUMVERTICES]; 11280 bool grounded[NUMVERTICES],floating[NUMVERTICES];11281 10930 11282 10931 if(!IsOnBed()) return; … … 11303 10952 b[i] = r[i]; 11304 10953 s[i] = b[i]+h[i]; 11305 floating[i] = false;11306 grounded[i] = true;11307 10954 } 11308 10955 } … … 11316 10963 s[i] = (1-density)*h[i]; 11317 10964 b[i] = -density*h[i]; 11318 floating[i] = true;11319 grounded[i] = false;11320 10965 } 11321 10966 else if(migration_style==SoftMigrationEnum && phi_ungrounding[vertices[i]->Pid()]<0.){ 11322 10967 s[i] = (1-density)*h[i]; 11323 10968 b[i] = -density*h[i]; 11324 floating[i] = true;11325 grounded[i] = false;11326 10969 } 11327 10970 else{
Note:
See TracChangeset
for help on using the changeset viewer.