fsi_osc_ring_compare_jacs.cc
Go to the documentation of this file.
1 //LIC// ====================================================================
2 //LIC// This file forms part of oomph-lib, the object-oriented,
3 //LIC// multi-physics finite-element library, available
4 //LIC// at http://www.oomph-lib.org.
5 //LIC//
6 //LIC// Copyright (C) 2006-2024 Matthias Heil and Andrew Hazel
7 //LIC//
8 //LIC// This library is free software; you can redistribute it and/or
9 //LIC// modify it under the terms of the GNU Lesser General Public
10 //LIC// License as published by the Free Software Foundation; either
11 //LIC// version 2.1 of the License, or (at your option) any later version.
12 //LIC//
13 //LIC// This library is distributed in the hope that it will be useful,
14 //LIC// but WITHOUT ANY WARRANTY; without even the implied warranty of
15 //LIC// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 //LIC// Lesser General Public License for more details.
17 //LIC//
18 //LIC// You should have received a copy of the GNU Lesser General Public
19 //LIC// License along with this library; if not, write to the Free Software
20 //LIC// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
21 //LIC// 02110-1301 USA.
22 //LIC//
23 //LIC// The authors may be contacted at oomph-lib@maths.man.ac.uk.
24 //LIC//
25 //LIC//====================================================================
26 // Driver for 2D Navier Stokes flow interacting with an elastic ring
27 
28 // Oomph-lib include files
29 #include "generic.h"
30 #include "navier_stokes.h"
31 #include "beam.h"
32 
33 //Need to include templated meshes, so that all functions
34 //are instantiated for our particular element types.
35 #include "meshes/quarter_circle_sector_mesh.h"
36 #include "meshes/one_d_lagrangian_mesh.h"
37 
38 using namespace std;
39 
40 using namespace oomph;
41 
42 //===========================================================================
43 /// Namespace for physical parameters
44 //===========================================================================
46 {
47 
48  // Independent parameters:
49  //------------------------
50 
51  /// Square of Womersly number (a frequency parameter)
52  double Alpha_sq=50.0;
53 
54  /// Density ratio of the solid and the fluid
55  double Density_ratio=1.0;
56 
57  /// External Pressure
58  double Pext=0.0;
59 
60  /// Poisson ratio
61  double Nu=0.49;
62 
63  /// Nondimensional thickness of the beam
64  double H=0.05;
65 
66  /// Perturbation pressure
67  double Pcos=0.0;
68 
69 
70  // Dependent parameters:
71  //----------------------
72 
73  /// Reynolds number
74  double Re;
75 
76  /// Reynolds x Strouhal number
77  double ReSt;
78 
79  /// Timescale ratio (non-dimensation density)
80  double Lambda_sq;
81 
82  /// Stress ratio
83  double Q;
84 
85  /// Set the parameters that are used in the code as a function
86  /// of the Womersley number, the density ratio and H
87  void set_params()
88  {
89  cout << "\n\n======================================================" <<std::endl;
90  cout << "\nSetting parameters. \n\n";
91  cout << "Prescribed: Square of Womersley number: Alpha_sq = "
92  << Alpha_sq << std::endl;
93  cout << " Density ratio: Density_ratio = "
94  << Density_ratio << std::endl;
95  cout << " Wall thickness: H = "
96  << H << std::endl;
97  cout << " Poisson ratio: Nu = "
98  << Nu << std::endl;
99  cout << " Pressure perturbation: Pcos = "
100  << Pcos << std::endl;
101 
102 
103  Q=1.0/12.0*pow(H,3)/Alpha_sq;
104  cout << "\nDependent: Stress ratio: Q = "
105  << Q << std::endl;
106 
107  Lambda_sq=1.0/12.0*pow(H,3)*Density_ratio;
108  cout << " Timescale ratio: Lambda_sq = "
109  << Lambda_sq << std::endl;
110 
111  Re=Alpha_sq;
112  cout << " Reynolds number: Re = "
113  << Re << std::endl;
114 
115  ReSt=Re;
116  cout << " Womersley number: ReSt = "
117  << ReSt << std::endl;
118  cout << "\n======================================================\n\n"
119  <<std::endl;
120 
121  }
122 
123  /// Non-FSI load function, a constant external pressure plus
124  /// a (small) sinusoidal perturbation of wavenumber two.
125  void pcos_load(const Vector<double>& xi, const Vector<double> &x,
126  const Vector<double>& N, Vector<double>& load)
127  {
128  for(unsigned i=0;i<2;i++)
129  {load[i] = (Pext - Pcos*cos(2.0*xi[0]))*N[i];}
130  }
131 
132 }
133 
134 
135 //======================================================================
136 /// FSI Ring problem: a fluid-structure interaction problem in which
137 /// a viscous fluid bounded by an initially circular beam is set into motion
138 /// by a small sinusoidal perturbation of the beam (the domain boundary).
139 //======================================================================
140 class FSIRingProblem : public Problem
141 {
142  /// There are very few element types that will work for this problem.
143  /// Rather than passing the element type as a template parameter to the
144  /// problem, we choose instead to use a typedef to specify the
145  /// particular element fluid used.
146  typedef AlgebraicElement<RefineableQCrouzeixRaviartElement<2> > FLUID_ELEMENT;
147 
148  /// Typedef to specify the solid element used
149  typedef FSIHermiteBeamElement SOLID_ELEMENT;
150 
151 public:
152 
153  /// Constructor: Number of elements in wall mesh, amplitude of the
154  /// initial wall deformation, amplitude of pcos perturbation and its duration.
155  FSIRingProblem(const unsigned &nelement_wall,
156  const double& eps_ampl, const double& pcos_initial,
157  const double& pcos_duration, const unsigned& i_case);
158 
159  /// Update after solve (empty)
161 
162  /// Update before solve (empty)
164 
165  /// Update the problem specs before checking Newton
166  /// convergence
168  {
169  // Update the fluid mesh -- auxiliary update function for algebraic
170  // nodes automatically updates no slip condition.
171  Fluid_mesh_pt->node_update();
172  }
173 
174  /// Update the problem specs after adaptation:
176  {
177  // The functions used to update the no slip boundary conditions
178  // must be set on any new nodes that have been created during the
179  // mesh adaptation process.
180  // There is no mechanism by which auxiliary update functions
181  // are copied to newly created nodes.
182  // (because, unlike boundary conditions, they don't occur exclusively
183  // at boundaries)
184 
185  // The no-slip boundary is boundary 1 of the mesh
186  // Loop over the nodes on this boundary and reset the auxilliary
187  // node update function
188  unsigned n_node = Fluid_mesh_pt->nboundary_node(1);
189  for (unsigned n=0;n<n_node;n++)
190  {
191  Fluid_mesh_pt->boundary_node_pt(1,n)->set_auxiliary_node_update_fct_pt(
192  FSI_functions::apply_no_slip_on_moving_wall);
193  }
194 
195  // (Re-)setup fsi: Work out which fluid dofs affect wall elements
196  // the correspondance between wall dofs and fluid elements is handled
197  // during the remeshing, but the "reverse" association must be done
198  // separately.
199  // We need to set up the interaction every time because the fluid element
200  // adjacent to a given solid element's integration point may have changed
201  // We pass the boundary between the fluid and solid meshes and pointers
202  // to the meshes. The interaction boundary is boundary 1 of the 2D
203  // fluid mesh.
204  FSI_functions::setup_fluid_load_info_for_solid_elements<FLUID_ELEMENT,2>
205  (this,1,Fluid_mesh_pt,Wall_mesh_pt);
206  }
207 
208  /// Doc solution: Pass number of timestep, i (we append to tracefile
209  /// after every timestep but do a full doc only at certain intervals),
210  /// DocInfo object and tracefile
211  void doc_solution(const unsigned& i, DocInfo& doc_info, ofstream& trace_file,
212  const unsigned& i_case);
213 
214  /// Do dynamic run
215  void dynamic_run(const unsigned& i_case);
216 
217 private:
218 
219  /// Setup initial condition for both domains
221 
222  /// Setup initial condition for wall
224 
225  /// Setup initial condition for fluid
227 
228  /// Element used for documenting displacement
229  SOLID_ELEMENT* Doc_displacement_elem_pt;
230 
231  /// Pointer to wall mesh
232  OneDLagrangianMesh<SOLID_ELEMENT> *Wall_mesh_pt;
233 
234  /// Pointer to fluid mesh
235  AlgebraicRefineableQuarterCircleSectorMesh<FLUID_ELEMENT> *Fluid_mesh_pt;
236 
237  /// Pointer to geometric object that represents the undeformed wall shape
238  GeomObject* Undef_geom_pt;
239 
240  /// Pointer to wall timestepper
241  Newmark<2>* Wall_time_stepper_pt;
242 
243  /// Pointer to fluid timestepper
244  BDF<2>* Fluid_time_stepper_pt;
245 
246  /// Pointer to node on coarsest mesh on which velocity is traced
247  Node* Veloc_trace_node_pt;
248 
249  /// Amplitude of initial deformation
250  double Eps_ampl;
251 
252  /// Initial pcos
253  double Pcos_initial;
254 
255  /// Duration of initial pcos
256  double Pcos_duration;
257 
258 };
259 
260 
261 //===============================================================
262 /// Setup initial condition: When we're done here, all variables
263 /// represent the state at the initial time.
264 //===============================================================
266 {
267 
268  cout << "Setting wall ic" << std::endl;
269  set_wall_initial_condition();
270 
271  cout << "Setting fluid ic" << std::endl;
272  set_fluid_initial_condition();
273 
274 }
275 
276 
277 //===============================================================
278 /// Setup initial condition for fluid: Impulsive start
279 //===============================================================
281 {
282 
283  // Update fluid domain: Careful!!! This also applies the no slip conditions
284  // on all nodes on the wall! Since the wall might have moved since
285  // we created the mesh; we're therefore imposing a nonzero
286  // velocity on these nodes. Must wipe this afterwards (done
287  // by setting *all* velocities to zero) otherwise we get
288  // an impulsive start from a very bizarre initial velocity
289  // field! [Yes, it took me a while to figure this out...]
290  Fluid_mesh_pt->node_update();
291 
292  // Assign initial values for the velocities;
293  // pressures don't carry a time history and can be left alone.
294 
295  //Find number of nodes in fluid mesh
296  unsigned n_node = Fluid_mesh_pt->nnode();
297 
298  // Loop over the nodes to set initial guess everywhere
299  for(unsigned n=0;n<n_node;n++)
300  {
301  // Loop over velocity directions: Impulsive initial start from
302  // zero velocity!
303  for(unsigned i=0;i<2;i++)
304  {
305  Fluid_mesh_pt->node_pt(n)->set_value(i,0.0);
306  }
307  }
308 
309  // Do an impulsive start with the assigned velocity field
310  Fluid_mesh_pt->assign_initial_values_impulsive();
311 
312 }
313 
314 
315 //===============================================================
316 /// Setup initial condition: Impulsive start either from
317 /// deformed or undeformed wall shape.
318 //===============================================================
320 {
321 
322  // Geometric object that specifies the initial conditions:
323  // A ring that is bucked in a 2-lobed mode
324  GeomObject* ic_geom_object_pt=
325  new PseudoBucklingRing(Eps_ampl,Global_Physical_Variables::H,2,2,
326  Wall_time_stepper_pt);
327 
328  // Assign period of oscillation of the geometric object
329  static_cast<PseudoBucklingRing*>(ic_geom_object_pt)->set_T(1.0);
330 
331  //Set initial time (to deform wall into max. amplitude)
332  double time=0.25;
333 
334  // Assign initial radius of the object
335  static_cast<PseudoBucklingRing*>(ic_geom_object_pt)->set_R_0(1.00);
336 
337  // Setup object that specifies the initial conditions:
338  SolidInitialCondition* IC_pt = new SolidInitialCondition(ic_geom_object_pt);
339 
340  // Assign values of positional data of all elements on wall mesh
341  // so that the wall deforms into the shape specified by IC object.
342  SolidMesh::Solid_IC_problem.set_static_initial_condition(
343  this,Wall_mesh_pt,IC_pt,time);
344 
345 }
346 
347 
348 //========================================================================
349 /// Document solution: Pass number of timestep, i; we append to trace file
350 /// at every timestep and do a full doc only after a certain number
351 /// of steps.
352 //========================================================================
353 void FSIRingProblem::doc_solution(const unsigned& i,
354  DocInfo& doc_info, ofstream& trace_file, const unsigned& i_case)
355 {
356 
357  // Full doc every nskip steps
358  unsigned nskip=1; // ADJUST
359 
360  // If we at an integer multiple of nskip, full documentation.
361  if (i%nskip==0)
362  {
363  doc_info.enable_doc();
364  cout << "Full doc step " << doc_info.number()
365  << " for time " << time_stepper_pt()->time_pt()->time() << std::endl;
366  }
367  //Otherwise, just output the trace file
368  else
369  {
370  doc_info.disable_doc();
371  cout << "Only trace for time "
372  << time_stepper_pt()->time_pt()->time() << std::endl;
373  }
374 
375 
376  // If we are at a full documentation step, output the fluid solution
377  if (doc_info.is_doc_enabled())
378  {
379  //Variables used in the output file.
380  ofstream some_file; char filename[100];
381  //Construct the output filename from the doc_info number and the
382  //output directory
383  sprintf(filename,"%s/soln%i_%i.dat",doc_info.directory().c_str(),
384  i_case,doc_info.number());
385  //Open the output file
386  some_file.open(filename);
387  /// Output the solution using 5x5 plot points
388  Fluid_mesh_pt->output(some_file,5);
389  //Close the output file
390  some_file.close();
391  }
392 
393  //Temporary vector to give the local coordinate at which to document
394  //the wall displacment
395  Vector<double> s(1,1.0);
396  // Write to the trace file:
397  trace_file << time_pt()->time()
398  //Document the displacement at the end of the the chosen element
399  << " " << Doc_displacement_elem_pt->interpolated_x(s,1)
400  << " " << Veloc_trace_node_pt->x(0)
401  << " " << Veloc_trace_node_pt->x(1)
402  << " " << Veloc_trace_node_pt->value(0)
403  << " " << Veloc_trace_node_pt->value(1)
404  << " " << Fluid_mesh_pt->nelement()
405  << " " << ndof()
406  << " " << Fluid_mesh_pt->nrefinement_overruled()
407  << " " << Fluid_mesh_pt->max_error()
408  << " " << Fluid_mesh_pt->min_error()
409  << " " << Fluid_mesh_pt->max_permitted_error()
410  << " " << Fluid_mesh_pt->min_permitted_error()
411  << " " << Fluid_mesh_pt->max_keep_unrefined();
412 
413  // Output the number of the corresponding full documentation
414  // file number (or -1 if no full doc was made)
415  if (doc_info.is_doc_enabled())
416  {trace_file << " " <<doc_info.number() << " ";}
417  else {trace_file << " " <<-1 << " ";}
418 
419  //End the trace file
420  trace_file << std::endl;
421 
422  // Increment counter for full doc
423  if (doc_info.is_doc_enabled()) {doc_info.number()++;}
424 }
425 
426 //======================================================================
427 /// Constructor for FSI ring problem. Pass number of wall elements
428 /// and length of wall (in Lagrangian coordinates) amplitude of
429 /// initial deformation, pcos perturbation and duration.
430 //======================================================================
432  const double& eps_ampl,
433  const double& pcos_initial,
434  const double& pcos_duration,
435  const unsigned& i_case) :
436  Eps_ampl(eps_ampl), Pcos_initial(pcos_initial),
437  Pcos_duration(pcos_duration)
438 {
439  //-----------------------------------------------------------
440  // Create timesteppers
441  //-----------------------------------------------------------
442 
443  // Allocate the wall timestepper and add it to the problem's vector
444  // of timesteppers
445  Wall_time_stepper_pt = new Newmark<2>;
446  add_time_stepper_pt(Wall_time_stepper_pt);
447 
448  // Allocate the fluid timestepper and add it to the problem's Vector
449  // of timesteppers
450  Fluid_time_stepper_pt = new BDF<2>;
451  add_time_stepper_pt(Fluid_time_stepper_pt);
452 
453  //----------------------------------------------------------
454  // Create the wall mesh
455  //----------------------------------------------------------
456 
457  // Undeformed wall is an elliptical ring
458  Undef_geom_pt = new Ellipse(1.0,1.0);
459 
460  //Length of wall in Lagrangian coordinates
461  double L = 2.0*atan(1.0);
462 
463  //Now create the (Lagrangian!) mesh
464  Wall_mesh_pt = new
465  OneDLagrangianMesh<SOLID_ELEMENT>(N,L,Undef_geom_pt,Wall_time_stepper_pt);
466 
467  //----------------------------------------------------------
468  // Set the boundary conditions for wall mesh (problem)
469  //----------------------------------------------------------
470 
471  // Bottom boundary: (Boundary 0)
472  // No vertical displacement
473  Wall_mesh_pt->boundary_node_pt(0,0)->pin_position(1);
474  // Zero slope: Pin type 1 dof for displacement direction 0
475  Wall_mesh_pt->boundary_node_pt(0,0)->pin_position(1,0);
476 
477  // Top boundary: (Boundary 1)
478  // No horizontal displacement
479  Wall_mesh_pt->boundary_node_pt(1,0)->pin_position(0);
480  // Zero slope: Pin type 1 dof for displacement direction 1
481  Wall_mesh_pt->boundary_node_pt(1,0)->pin_position(1,1);
482 
483 
484  //-----------------------------------------------------------
485  // Create the fluid mesh:
486  //-----------------------------------------------------------
487 
488  // Fluid mesh is suspended from wall between the following Lagrangian
489  // coordinates:
490  double xi_lo=0.0;
491  double xi_hi=L;
492 
493  // Fractional position of dividing line for two outer blocks in mesh
494  double fract_mid=0.5;
495 
496  //Create a geometric object that represents the wall geometry from the
497  //wall mesh (one Lagrangian, two Eulerian coordinates).
498  MeshAsGeomObject *wall_mesh_as_geometric_object_pt
499  = new MeshAsGeomObject(Wall_mesh_pt);
500 
501  // Build fluid mesh using the wall mesh as a geometric object
502  Fluid_mesh_pt = new AlgebraicRefineableQuarterCircleSectorMesh<FLUID_ELEMENT >
503  (wall_mesh_as_geometric_object_pt,
504  xi_lo,fract_mid,xi_hi,Fluid_time_stepper_pt);
505 
506  // Set the error estimator
507  Z2ErrorEstimator* error_estimator_pt=new Z2ErrorEstimator;
508  Fluid_mesh_pt->spatial_error_estimator_pt()=error_estimator_pt;
509 
510  // Extract pointer to node at center of mesh
511  unsigned nnode=Fluid_mesh_pt->finite_element_pt(0)->nnode();
512  Veloc_trace_node_pt=Fluid_mesh_pt->finite_element_pt(0)->node_pt(nnode-1);
513 
514  // Do some random refinement
515  Vector<unsigned> elements_to_be_refined;
516  elements_to_be_refined.push_back(2);
517  Fluid_mesh_pt->refine_selected_elements(elements_to_be_refined);
518  Fluid_mesh_pt->refine_selected_elements(elements_to_be_refined);
519 
520  //-------------------------------------------------------
521  // Set the fluid boundary conditions
522  //-------------------------------------------------------
523 
524  // Bottom boundary (boundary 0):
525  {
526  unsigned n_node = Fluid_mesh_pt->nboundary_node(0);
527  for (unsigned n=0;n<n_node;n++)
528  {
529  // Pin vertical velocity
530  Fluid_mesh_pt->boundary_node_pt(0,n)->pin(1);
531  }
532  }
533 
534  // Ring boundary (boundary 1):
535  // No slip; this also implies that the velocity needs
536  // to be updated in response to wall motion
537  {
538  unsigned n_node = Fluid_mesh_pt->nboundary_node(1);
539  for (unsigned n=0;n<n_node;n++)
540  {
541  // Which node are we dealing with?
542  Node* node_pt=Fluid_mesh_pt->boundary_node_pt(1,n);
543 
544  // Set auxiliary update function pointer
545  node_pt->set_auxiliary_node_update_fct_pt(
546  FSI_functions::apply_no_slip_on_moving_wall);
547 
548  // Pin both velocities
549  for(unsigned i=0;i<2;i++) {node_pt->pin(i);}
550  }
551  }
552 
553  // Left boundary (boundary 2):
554  {
555  unsigned n_node = Fluid_mesh_pt->nboundary_node(2);
556  for (unsigned n=0;n<n_node;n++)
557  {
558  // Pin horizontal velocity
559  Fluid_mesh_pt->boundary_node_pt(2,n)->pin(0);
560  }
561  }
562 
563 
564  //--------------------------------------------------------
565  // Add the submeshes and build global mesh
566  // -------------------------------------------------------
567 
568  // Wall mesh
569  add_sub_mesh(Wall_mesh_pt);
570 
571  //Fluid mesh
572  add_sub_mesh(Fluid_mesh_pt);
573 
574  // Combine all submeshes into a single Mesh
575  build_global_mesh();
576 
577 
578  //----------------------------------------------------------
579  // Finish problem setup
580  // ---------------------------------------------------------
581 
582  //Find number of elements in fluid mesh
583  unsigned n_element = Fluid_mesh_pt->nelement();
584 
585  bool done=false;
586 
587  // Loop over the fluid elements to set up element-specific
588  // things that cannot be handled by constructor
589  for(unsigned e=0;e<n_element;e++)
590  {
591  // Upcast from FiniteElement to the present element
592  FLUID_ELEMENT *el_pt
593  = dynamic_cast<FLUID_ELEMENT*>(Fluid_mesh_pt->element_pt(e));
594 
595  //Set the Reynolds number, etc
596  el_pt->re_pt() = &Global_Physical_Variables::Re;
597  el_pt->re_st_pt() = &Global_Physical_Variables::ReSt;
598 
599  // Choose evaluation of shape derivatives
600 
601  // Direct FD
602  if (i_case==0)
603  {
604  el_pt->evaluate_shape_derivs_by_direct_fd();
605  if (!done) std::cout << "\n\n [CR residuals] Direct FD" << std::endl;
606  }
607  // Chain rule with/without FD
608  else if ( (i_case==1) || (i_case==2) )
609  {
610  // It's broken but let's call it anyway to keep self-test alive
611  bool i_know_what_i_am_doing=true;
612  el_pt->evaluate_shape_derivs_by_chain_rule(i_know_what_i_am_doing);
613  if (i_case==1)
614  {
615  el_pt->enable_always_evaluate_dresidual_dnodal_coordinates_by_fd();
616  if (!done) std::cout << "\n\n [CR residuals] Chain rule and FD"
617  << std::endl;
618  }
619  else
620  {
621  el_pt->disable_always_evaluate_dresidual_dnodal_coordinates_by_fd();
622  if (!done) std::cout << "\n\n [CR residuals] Chain rule and analytic"
623  << std::endl;
624  }
625  }
626  // Fastest with/without FD
627  else if ( (i_case==3) || (i_case==4) )
628  {
629 
630  // It's broken but let's call it anyway to keep self-test alive
631  bool i_know_what_i_am_doing=true;
632  el_pt->evaluate_shape_derivs_by_fastest_method(i_know_what_i_am_doing);
633  if (i_case==3)
634  {
635  el_pt->enable_always_evaluate_dresidual_dnodal_coordinates_by_fd();
636  if (!done) std::cout << "\n\n [CR residuals] Fastest and FD"
637  << std::endl;
638  }
639  else
640  {
641  el_pt->disable_always_evaluate_dresidual_dnodal_coordinates_by_fd();
642  if (!done) std::cout << "\n\n [CR residuals] Fastest and analytic"
643  << std::endl;
644  }
645  }
646  done=true;
647 
648  }
649 
650 
651  //Loop over the solid elements to set physical parameters etc.
652  unsigned n_wall_element = Wall_mesh_pt->nelement();
653  for(unsigned e=0;e<n_wall_element;e++)
654  {
655  //Cast to proper element type
656  SOLID_ELEMENT *el_pt = dynamic_cast<SOLID_ELEMENT*>(
657  Wall_mesh_pt->element_pt(e));
658 
659  //Set physical parameters for each element:
660  el_pt->h_pt() = &Global_Physical_Variables::H;
661  el_pt->lambda_sq_pt() = &Global_Physical_Variables::Lambda_sq;
662 
663  //Function that specifies the external load Vector
664  el_pt->load_vector_fct_pt() = &Global_Physical_Variables::pcos_load;
665 
666  // Function that specifies the load ratios
667  el_pt->q_pt() = &Global_Physical_Variables::Q;
668 
669  //Assign the undeformed beam shape
670  el_pt->undeformed_beam_pt() = Undef_geom_pt;
671  }
672 
673  // Establish control displacment: (even if no displacement control is applied
674  // we still want to doc the displacement at the same point)
675 
676  // Choose element: (This is the last one)
677  Doc_displacement_elem_pt = dynamic_cast<SOLID_ELEMENT*>(
678  Wall_mesh_pt->element_pt(n_wall_element-1));
679 
680  // Setup fsi: Work out which fluid dofs affect the wall elements
681  // the correspondance between wall dofs and fluid elements is handled
682  // during the remeshing, but the "reverse" association must be done
683  // separately.
684  // We pass the boundary between the fluid and solid meshes and pointers
685  // to the meshes. The interaction boundary is boundary 1 of the
686  // 2D fluid mesh.
687  FSI_functions::setup_fluid_load_info_for_solid_elements<FLUID_ELEMENT,2>
688  (this,1,Fluid_mesh_pt,Wall_mesh_pt);
689 
690  // Do equation numbering
691  cout << "# of dofs " << assign_eqn_numbers() << std::endl;
692 
693 }
694 
695 
696 //=========================================================================
697 /// Solver loop to perform unsteady run
698 //=========================================================================
699 void FSIRingProblem::dynamic_run(const unsigned& i_case)
700 {
701  // Setup documentation
702  //---------------------------------------------------------------
703 
704  /// Label for output
705  DocInfo doc_info;
706 
707  // Output directory
708  doc_info.set_directory("RESLT");
709 
710  // Step number
711  doc_info.number()=0;
712 
713  //Open a trace file
714  ofstream trace_file("RESLT/trace_ring.dat");
715 
716  // Write header for trace file
717  trace_file << "VARIABLES=\"time\",\"V_c_t_r_l\"";
718  trace_file << ",\"x<sub>1</sub><sup>(track)</sup>\"";
719  trace_file << ",\"x<sub>2</sub><sup>(track)</sup>\"";
720  trace_file << ",\"u<sub>1</sub><sup>(track)</sup>\"";
721  trace_file << ",\"u<sub>2</sub><sup>(track)</sup>\"";
722  trace_file << ",\"N<sub>element</sub>\"";
723  trace_file << ",\"N<sub>dof</sub>\"";
724  trace_file << ",\"# of under-refined elements\"";
725  trace_file << ",\"max. error\"";
726  trace_file << ",\"min. error\"";
727  trace_file << ",\"max. permitted error\"";
728  trace_file << ",\"min. permitted error\"";
729  trace_file << ",\"max. permitted # of unrefined elements\"";
730  trace_file << ",\"doc number\"";
731  trace_file << std::endl;
732 
733 
734  // Initialise timestepping
735  // -------------------------------------------------------------
736 
737  // Number of steps
738  unsigned nstep=300;
739 
740  // Nontrivial command line input: validation: only do three steps
741  if (CommandLineArgs::Argc>1)
742  {
743  nstep=1;
744  cout << "Only doing nstep steps for validation: " << nstep << std::endl;
745  }
746 
747  // Set initial timestep
748  double dt=0.004;
749 
750  // Set initial value for dt -- also assigns weights to the timesteppers
751  initialise_dt(dt);
752 
753  // Set physical parameters
754  //---------------------------------------------------------
755  using namespace Global_Physical_Variables;
756 
757  // Set Womersley number
758  Alpha_sq=100.0; // 50.0; // ADJUST
759 
760  // Set density ratio
761  Density_ratio=10.0; // 0.0; ADJUST
762 
763  // Wall thickness
764  H=1.0/20.0;
765 
766  // Set external pressure
767  Pext=0.0;
768 
769  /// Perturbation pressure
771 
772  // Assign/doc corresponding computational parameters
773  set_params();
774 
775 
776  // Refine uniformly and assign initial conditions
777  //--------------------------------------------------------------
778 
779  // Refine the problem uniformly
780  //refine_uniformly();
781  //refine_uniformly();
782 
783  // This sets up the solution at the initial time
785 
786  // Set targets for spatial adptivity
787  //---------------------------------------------------------------
788 
789  // Max. and min. error for adaptive refinement/unrefinement
790  Fluid_mesh_pt->max_permitted_error()=1.0e-2;
791  Fluid_mesh_pt->min_permitted_error()=1.0e-3;
792 
793  // Don't allow refinement to drop under given level
794  Fluid_mesh_pt->min_refinement_level()=2;
795 
796  // Don't allow refinement beyond given level
797  Fluid_mesh_pt->max_refinement_level()=6;
798 
799  // Don't bother adapting the mesh if no refinement is required
800  // and if less than ... elements are to be merged.
801  Fluid_mesh_pt->max_keep_unrefined()=20;
802 
803  // Doc refinement targets
804  Fluid_mesh_pt->doc_adaptivity_targets(cout);
805 
806 
807  // Do the timestepping
808  //----------------------------------------------------------------
809 
810  // Reset initial conditions after refinement for first step only
811  bool first=true;
812 
813  //Output initial data
814  doc_solution(0,doc_info,trace_file,i_case);
815 
816 
817 // {
818 // unsigned nel=Fluid_mesh_pt->nelement();
819 // for (unsigned e=0;e<nel;e++)
820 // {
821 // std::cout << "\n\nEl: " << e << std::endl << std::endl;
822 // FiniteElement* el_pt=Fluid_mesh_pt->finite_element_pt(e);
823 // unsigned n_dof=el_pt->ndof();
824 // Vector<double> residuals(n_dof);
825 // DenseDoubleMatrix jacobian(n_dof,n_dof);
826 // el_pt->get_jacobian(residuals,jacobian);
827 // }
828 // exit(0);
829 // }
830 
831  // Time integration loop
832  for(unsigned i=1;i<=nstep;i++)
833  {
834  // Switch doc off during solve
835  doc_info.disable_doc();
836 
837  // Solve
838  unsigned max_adapt=1;
839  unsteady_newton_solve(dt,max_adapt,first);
840 
841  // Now we've done the first step
842  first=false;
843 
844  // Doc solution
845  doc_solution(i,doc_info,trace_file,i_case);
846 
847  /// Switch off perturbation pressure
848  if (time_pt()->time()>Pcos_duration) {Pcos=0.0;}
849 
850  }
851 
852 }
853 
854 
855 //=====================================================================
856 /// Driver for fsi ring test problem
857 //=====================================================================
858 int main(int argc, char* argv[])
859 {
860 
861  // Store command line arguments
862  CommandLineArgs::setup(argc,argv);
863 
864  // Number of elements
865  unsigned nelem = 13;
866 
867  /// Perturbation pressure
868  double pcos_initial=1.0e-6; // ADJUST
869 
870  /// Duration of initial pcos perturbation
871  double pcos_duration=0.3; // ADJUST
872 
873  /// Amplitude of initial deformation
874  double eps_ampl=0.0; // ADJUST
875 
876 
877 
878  // Loop over all cases
879  for (unsigned i_case=0;i_case<5;i_case++)
880  {
881  std::cout << "[CR residuals] " << std::endl;
882  std::cout << "[CR residuals]=================================================="
883  << std::endl;
884  std::cout << "[CR residuals] " << std::endl;
885 
886  //Set up the problem
887  FSIRingProblem problem(nelem,eps_ampl,pcos_initial,pcos_duration,i_case);
888 
889  // Do parameter study
890  problem.dynamic_run(i_case);
891  }
892 
893 }
894 
895 
896 
897 
898 
899 
FSI Ring problem: a fluid-structure interaction problem in which a viscous fluid bounded by an initia...
double Pcos_initial
Initial pcos.
OneDLagrangianMesh< SOLID_ELEMENT > * Wall_mesh_pt
Pointer to wall mesh.
GeomObject * Undef_geom_pt
Pointer to geometric object that represents the undeformed wall shape.
AlgebraicElement< RefineableQCrouzeixRaviartElement< 2 > > FLUID_ELEMENT
There are very few element types that will work for this problem. Rather than passing the element typ...
Node * Veloc_trace_node_pt
Pointer to node on coarsest mesh on which velocity is traced.
void set_initial_condition()
Setup initial condition for both domains.
AlgebraicRefineableQuarterCircleSectorMesh< FLUID_ELEMENT > * Fluid_mesh_pt
Pointer to fluid mesh.
void set_wall_initial_condition()
Setup initial condition for wall.
void actions_after_newton_solve()
Update after solve (empty)
Newmark< 2 > * Wall_time_stepper_pt
Pointer to wall timestepper.
void doc_solution(const unsigned &i, DocInfo &doc_info, ofstream &trace_file)
Doc solution: Pass number of timestep, i (we append to tracefile after every timestep but do a full d...
BDF< 2 > * Fluid_time_stepper_pt
Pointer to fluid timestepper.
SOLID_ELEMENT * Doc_displacement_elem_pt
Element used for documenting displacement.
FSIRingProblem(const unsigned &nelement_wall, const double &eps_ampl, const double &pcos_initial, const double &pcos_duration)
Constructor: Number of elements in wall mesh, amplitude of the initial wall deformation,...
double Pcos_duration
Duration of initial pcos.
FSIHermiteBeamElement SOLID_ELEMENT
Typedef to specify the solid element used.
void actions_before_newton_solve()
Update before solve (empty)
void dynamic_run()
Do dynamic run.
void set_fluid_initial_condition()
Setup initial condition for fluid.
void actions_after_adapt()
Update the problem specs after adaptation:
void actions_before_newton_convergence_check()
Update the problem specs before checking Newton convergence.
int main(int argc, char *argv[])
Driver for fsi ring test problem.
Namespace for physical parameters.
Definition: fsi_osc_ring.cc:46
double Pext
External Pressure.
Definition: fsi_osc_ring.cc:58
double Alpha_sq
Square of Womersly number (a frequency parameter)
Definition: fsi_osc_ring.cc:52
double ReSt
Reynolds x Strouhal number.
Definition: fsi_osc_ring.cc:77
double Nu
Poisson ratio.
Definition: fsi_osc_ring.cc:61
void set_params()
Set the parameters that are used in the code as a function of the Womersley number,...
Definition: fsi_osc_ring.cc:87
double Q
Stress ratio.
Definition: fsi_osc_ring.cc:83
double Lambda_sq
Timescale ratio (non-dimensation density)
Definition: fsi_osc_ring.cc:80
void pcos_load(const Vector< double > &xi, const Vector< double > &x, const Vector< double > &N, Vector< double > &load)
Non-FSI load function, a constant external pressure plus a (small) sinusoidal perturbation of wavenum...
double Pcos
Perturbation pressure.
Definition: fsi_osc_ring.cc:67
double Re
Reynolds number.
Definition: fsi_osc_ring.cc:74
double Density_ratio
Density ratio of the solid and the fluid.
Definition: fsi_osc_ring.cc:55
double H
Nondimensional thickness of the beam.
Definition: fsi_osc_ring.cc:64