597 * We start by including all the necessary deal.II header files and some
C++
598 * related ones. They have been discussed in detail in previous tutorial
599 * programs, so you need only refer to past tutorials
for details.
605 * #include <deal.II/base/function.h>
606 * #include <deal.II/base/parameter_handler.h>
607 * #include <deal.II/base/
point.h>
608 * #include <deal.II/base/quadrature_lib.h>
609 * #include <deal.II/base/symmetric_tensor.h>
610 * #include <deal.II/base/tensor.h>
611 * #include <deal.II/base/timer.h>
612 * #include <deal.II/base/work_stream.h>
613 * #include <deal.II/base/mpi.h>
614 * #include <deal.II/base/quadrature_point_data.h>
616 * #include <deal.II/differentiation/ad.h>
618 * #include <deal.II/distributed/shared_tria.h>
620 * #include <deal.II/dofs/dof_renumbering.h>
621 * #include <deal.II/dofs/dof_tools.h>
622 * #include <deal.II/dofs/dof_accessor.h>
624 * #include <deal.II/grid/filtered_iterator.h>
625 * #include <deal.II/grid/grid_generator.h>
626 * #include <deal.II/grid/grid_tools.h>
627 * #include <deal.II/grid/grid_in.h>
628 * #include <deal.II/grid/grid_out.h>
629 * #include <deal.II/grid/manifold_lib.h>
630 * #include <deal.II/grid/tria_accessor.h>
631 * #include <deal.II/grid/tria_iterator.h>
633 * #include <deal.II/fe/fe_dgp_monomial.h>
634 * #include <deal.II/fe/fe_q.h>
635 * #include <deal.II/fe/fe_system.h>
636 * #include <deal.II/fe/fe_tools.h>
637 * #include <deal.II/fe/fe_values.h>
639 * #include <deal.II/lac/block_sparsity_pattern.h>
640 * #include <deal.II/lac/affine_constraints.h>
641 * #include <deal.II/lac/dynamic_sparsity_pattern.h>
642 * #include <deal.II/lac/full_matrix.h>
644 * #include <deal.II/lac/packaged_operation.h>
646 * #include <deal.II/lac/trilinos_block_sparse_matrix.h>
647 * #include <deal.II/lac/trilinos_linear_operator.h>
648 * #include <deal.II/lac/trilinos_parallel_block_vector.h>
649 * #include <deal.II/lac/trilinos_precondition.h>
650 * #include <deal.II/lac/trilinos_sparse_matrix.h>
651 * #include <deal.II/lac/trilinos_sparsity_pattern.h>
652 * #include <deal.II/lac/trilinos_solver.h>
653 * #include <deal.II/lac/trilinos_vector.h>
655 * #include <deal.II/lac/block_vector.h>
656 * #include <deal.II/lac/vector.h>
658 * #include <deal.II/numerics/data_postprocessor.h>
659 * #include <deal.II/numerics/data_out.h>
660 * #include <deal.II/numerics/data_out_faces.h>
661 * #include <deal.II/numerics/fe_field_function.h>
662 * #include <deal.II/numerics/vector_tools.h>
664 * #include <deal.II/physics/transformations.h>
665 * #include <deal.II/physics/elasticity/kinematics.h>
666 * #include <deal.II/physics/elasticity/standard_tensors.h>
668 * #include <iostream>
676 * We create a
namespace for everything that relates to
677 * the nonlinear poro-viscoelastic formulation,
678 * and
import all the deal.II function and class names into it:
681 * namespace NonLinearPoroViscoElasticity
683 * using namespace dealii;
688 * <a name=
"nonlinear-poro-viscoelasticity.cc-Runtimeparameters"></a>
689 * <h3>Run-time parameters</h3>
694 * introduced by the user through the file
"parameters.prm"
697 *
namespace Parameters
702 * <a name=
"nonlinear-poro-viscoelasticity.cc-FiniteElementsystem"></a>
703 * <h4>Finite Element system</h4>
704 * Here we specify the polynomial order used to
approximate the solution,
705 * both
for the displacements and pressure unknowns.
706 * The quadrature order should be adjusted accordingly.
711 *
unsigned int poly_degree_displ;
712 *
unsigned int poly_degree_pore;
713 *
unsigned int quad_order;
724 * prm.enter_subsection(
"Finite element system");
726 * prm.declare_entry(
"Polynomial degree displ",
"2",
728 *
"Displacement system polynomial order");
730 * prm.declare_entry(
"Polynomial degree pore",
"1",
732 *
"Pore pressure system polynomial order");
734 * prm.declare_entry(
"Quadrature order",
"3",
736 *
"Gauss quadrature order");
738 * prm.leave_subsection();
743 * prm.enter_subsection(
"Finite element system");
745 * poly_degree_displ = prm.get_integer(
"Polynomial degree displ");
746 * poly_degree_pore = prm.get_integer(
"Polynomial degree pore");
747 * quad_order = prm.get_integer(
"Quadrature order");
749 * prm.leave_subsection();
755 * <a name=
"nonlinear-poro-viscoelasticity.cc-Geometry"></a>
757 * These parameters are related to the geometry definition and mesh generation.
758 * We select the type of problem to solve and introduce the desired load values.
763 * std::string geom_type;
764 *
unsigned int global_refinement;
766 * std::string load_type;
768 *
unsigned int num_cycle_sets;
770 *
double drained_pressure;
781 * prm.enter_subsection(
"Geometry");
783 * prm.declare_entry(
"Geometry type",
"Ehlers_tube_step_load",
785 *
"|Ehlers_tube_increase_load"
786 *
"|Ehlers_cube_consolidation"
787 *
"|Franceschini_consolidation"
788 *
"|Budday_cube_tension_compression"
789 *
"|Budday_cube_tension_compression_fully_fixed"
790 *
"|Budday_cube_shear_fully_fixed"),
791 *
"Type of geometry used. "
792 *
"For Ehlers verification examples see Ehlers and Eipper (1999). "
793 *
"For Franceschini brain consolidation see Franceschini et al. (2006)"
794 *
"For Budday brain examples see Budday et al. (2017)");
796 * prm.declare_entry(
"Global refinement",
"1",
798 *
"Global refinement level");
800 * prm.declare_entry(
"Grid scale",
"1.0",
802 *
"Global grid scaling factor");
804 * prm.declare_entry(
"Load type",
"pressure",
806 *
"Type of loading");
808 * prm.declare_entry(
"Load value",
"-7.5e+6",
812 * prm.declare_entry(
"Number of cycle sets",
"1",
814 *
"Number of times each set of 3 cycles is repeated, only for "
815 *
"Budday_cube_tension_compression and Budday_cube_tension_compression_fully_fixed. "
816 *
"Load value is doubled in second set, load rate is kept constant."
817 *
"Final time indicates end of second cycle set.");
819 * prm.declare_entry(
"Fluid flow value",
"0.0",
821 *
"Prescribed fluid flow. Not implemented in any example yet.");
823 * prm.declare_entry(
"Drained pressure",
"0.0",
825 *
"Increase of pressure value at drained boundary w.r.t the atmospheric pressure.");
827 * prm.leave_subsection();
832 * prm.enter_subsection(
"Geometry");
834 * geom_type = prm.get(
"Geometry type");
835 * global_refinement = prm.get_integer(
"Global refinement");
836 *
scale = prm.get_double(
"Grid scale");
837 * load_type = prm.get(
"Load type");
838 * load = prm.get_double(
"Load value");
839 * num_cycle_sets = prm.get_integer(
"Number of cycle sets");
840 * fluid_flow = prm.get_double(
"Fluid flow value");
841 * drained_pressure = prm.get_double(
"Drained pressure");
843 * prm.leave_subsection();
849 * <a name=
"nonlinear-poro-viscoelasticity.cc-Materials"></a>
854 * Here we select the type of material
for the solid component
855 * and define the corresponding material parameters.
856 * Then we define he fluid data, including the type of
857 * seepage velocity definition to use.
862 * std::string mat_type;
868 *
double alpha1_infty;
869 *
double alpha2_infty;
870 *
double alpha3_infty;
874 *
double alpha1_mode_1;
875 *
double alpha2_mode_1;
876 *
double alpha3_mode_1;
877 *
double viscosity_mode_1;
878 * std::string fluid_type;
879 *
double solid_vol_frac;
880 *
double kappa_darcy;
881 *
double init_intrinsic_perm;
882 *
double viscosity_FR;
883 *
double init_darcy_coef;
886 *
int gravity_direction;
887 *
double gravity_value;
901 * prm.enter_subsection(
"Material properties");
903 * prm.declare_entry(
"material",
"Neo-Hooke",
905 *
"Type of material used in the problem");
907 * prm.declare_entry(
"lambda",
"8.375e6",
909 *
"First Lamé parameter for extension function related to compactation point in solid material [Pa].");
911 * prm.declare_entry(
"shear modulus",
"5.583e6",
913 *
"shear modulus for Neo-Hooke materials [Pa].");
915 * prm.declare_entry(
"eigen solver",
"QL Implicit Shifts",
917 *
"The type of eigen solver to be used for Ogden and visco-Ogden models.");
919 * prm.declare_entry(
"mu1",
"0.0",
921 *
"Shear material parameter 'mu1' for Ogden material [Pa].");
923 * prm.declare_entry(
"mu2",
"0.0",
925 *
"Shear material parameter 'mu2' for Ogden material [Pa].");
927 * prm.declare_entry(
"mu3",
"0.0",
929 *
"Shear material parameter 'mu1' for Ogden material [Pa].");
931 * prm.declare_entry(
"alpha1",
"1.0",
933 *
"Stiffness material parameter 'alpha1' for Ogden material [-].");
935 * prm.declare_entry(
"alpha2",
"1.0",
937 *
"Stiffness material parameter 'alpha2' for Ogden material [-].");
939 * prm.declare_entry(
"alpha3",
"1.0",
941 *
"Stiffness material parameter 'alpha3' for Ogden material [-].");
943 * prm.declare_entry(
"mu1_1",
"0.0",
945 *
"Shear material parameter 'mu1' for first viscous mode in Ogden material [Pa].");
947 * prm.declare_entry(
"mu2_1",
"0.0",
949 *
"Shear material parameter 'mu2' for first viscous mode in Ogden material [Pa].");
951 * prm.declare_entry(
"mu3_1",
"0.0",
953 *
"Shear material parameter 'mu1' for first viscous mode in Ogden material [Pa].");
955 * prm.declare_entry(
"alpha1_1",
"1.0",
957 *
"Stiffness material parameter 'alpha1' for first viscous mode in Ogden material [-].");
959 * prm.declare_entry(
"alpha2_1",
"1.0",
961 *
"Stiffness material parameter 'alpha2' for first viscous mode in Ogden material [-].");
963 * prm.declare_entry(
"alpha3_1",
"1.0",
965 *
"Stiffness material parameter 'alpha3' for first viscous mode in Ogden material [-].");
967 * prm.declare_entry(
"viscosity_1",
"1e-10",
969 *
"Deformation-independent viscosity parameter 'eta_1' for first viscous mode in Ogden material [-].");
971 * prm.declare_entry(
"seepage definition",
"Ehlers",
973 *
"Type of formulation used to define the seepage velocity in the problem. "
974 *
"Choose between Markert formulation of deformation-dependent intrinsic permeability "
975 *
"and Ehlers formulation of deformation-dependent Darcy flow coefficient.");
977 * prm.declare_entry(
"initial solid volume fraction",
"0.67",
979 *
"Initial porosity (solid volume fraction, 0 < n_0s < 1)");
981 * prm.declare_entry(
"kappa",
"0.0",
983 *
"Deformation-dependency control parameter for specific permeability (kappa >= 0)");
985 * prm.declare_entry(
"initial intrinsic permeability",
"0.0",
987 *
"Initial intrinsic permeability parameter [m^2] (isotropic permeability). To be used with Markert formulation.");
989 * prm.declare_entry(
"fluid viscosity",
"0.0",
991 *
"Effective shear viscosity parameter of the fluid [Pa·s, (N·s)/m^2]. To be used with Markert formulation.");
993 * prm.declare_entry(
"initial Darcy coefficient",
"1.0e-4",
995 *
"Initial Darcy flow coefficient [m/s] (isotropic permeability). To be used with Ehlers formulation.");
997 * prm.declare_entry(
"fluid weight",
"1.0e4",
999 *
"Effective weight of the fluid [N/m^3]. To be used with Ehlers formulation.");
1001 * prm.declare_entry(
"gravity term",
"false",
1003 *
"Gravity term considered (true) or neglected (false)");
1005 * prm.declare_entry(
"fluid density",
"1.0",
1007 *
"Real (or effective) density of the fluid");
1009 * prm.declare_entry(
"solid density",
"1.0",
1011 *
"Real (or effective) density of the solid");
1013 * prm.declare_entry(
"gravity direction",
"2",
1015 *
"Direction of gravity (unit vector 0 for x, 1 for y, 2 for z)");
1017 * prm.declare_entry(
"gravity value",
"-9.81",
1019 *
"Value of gravity (be careful to have consistent units!)");
1021 * prm.leave_subsection();
1026 * prm.enter_subsection(
"Material properties");
1033 * mat_type = prm.get(
"material");
1034 *
lambda = prm.get_double(
"lambda");
1035 * mu = prm.get_double(
"shear modulus");
1036 * mu1_infty = prm.get_double(
"mu1");
1037 * mu2_infty = prm.get_double(
"mu2");
1038 * mu3_infty = prm.get_double(
"mu3");
1039 * alpha1_infty = prm.get_double(
"alpha1");
1040 * alpha2_infty = prm.get_double(
"alpha2");
1041 * alpha3_infty = prm.get_double(
"alpha3");
1042 * mu1_mode_1 = prm.get_double(
"mu1_1");
1043 * mu2_mode_1 = prm.get_double(
"mu2_1");
1044 * mu3_mode_1 = prm.get_double(
"mu3_1");
1045 * alpha1_mode_1 = prm.get_double(
"alpha1_1");
1046 * alpha2_mode_1 = prm.get_double(
"alpha2_1");
1047 * alpha3_mode_1 = prm.get_double(
"alpha3_1");
1048 * viscosity_mode_1 = prm.get_double(
"viscosity_1");
1054 * fluid_type = prm.get(
"seepage definition");
1055 * solid_vol_frac = prm.get_double(
"initial solid volume fraction");
1056 * kappa_darcy = prm.get_double(
"kappa");
1057 * init_intrinsic_perm = prm.get_double(
"initial intrinsic permeability");
1058 * viscosity_FR = prm.get_double(
"fluid viscosity");
1059 * init_darcy_coef = prm.get_double(
"initial Darcy coefficient");
1060 * weight_FR = prm.get_double(
"fluid weight");
1066 * gravity_term = prm.get_bool(
"gravity term");
1067 * density_FR = prm.get_double(
"fluid density");
1068 * density_SR = prm.get_double(
"solid density");
1069 * gravity_direction = prm.get_integer(
"gravity direction");
1070 * gravity_value = prm.get_double(
"gravity value");
1072 *
if ( (fluid_type ==
"Markert") && ((init_intrinsic_perm == 0.0) || (viscosity_FR == 0.0)) )
1073 *
AssertThrow(
false, ExcMessage(
"Markert seepage velocity formulation requires the definition of "
1074 *
"'initial intrinsic permeability' and 'fluid viscosity' greater than 0.0."));
1076 *
if ( (fluid_type ==
"Ehlers") && ((init_darcy_coef == 0.0) || (weight_FR == 0.0)) )
1077 *
AssertThrow(
false, ExcMessage(
"Ehler seepage velocity formulation requires the definition of "
1078 *
"'initial Darcy coefficient' and 'fluid weight' greater than 0.0."));
1080 *
const std::string eigen_solver_type = prm.get(
"eigen solver");
1081 *
if (eigen_solver_type ==
"QL Implicit Shifts")
1083 *
else if (eigen_solver_type ==
"Jacobi")
1087 *
AssertThrow(
false, ExcMessage(
"Unknown eigen solver selected."));
1090 * prm.leave_subsection();
1096 * <a name=
"nonlinear-poro-viscoelasticity.cc-Nonlinearsolver"></a>
1097 * <h4>Nonlinear solver</h4>
1101 * We now define the tolerances and the maximum number of iterations
for the
1102 * Newton-Raphson scheme used to solve the nonlinear system of governing equations.
1105 *
struct NonlinearSolver
1107 *
unsigned int max_iterations_NR;
1110 *
double tol_p_fluid;
1121 * prm.enter_subsection(
"Nonlinear solver");
1123 * prm.declare_entry(
"Max iterations Newton-Raphson",
"15",
1125 *
"Number of Newton-Raphson iterations allowed");
1127 * prm.declare_entry(
"Tolerance force",
"1.0e-8",
1129 *
"Force residual tolerance");
1131 * prm.declare_entry(
"Tolerance displacement",
"1.0e-6",
1133 *
"Displacement error tolerance");
1135 * prm.declare_entry(
"Tolerance pore pressure",
"1.0e-6",
1137 *
"Pore pressure error tolerance");
1139 * prm.leave_subsection();
1144 * prm.enter_subsection(
"Nonlinear solver");
1146 * max_iterations_NR = prm.get_integer(
"Max iterations Newton-Raphson");
1147 * tol_f = prm.get_double(
"Tolerance force");
1148 * tol_u = prm.get_double(
"Tolerance displacement");
1149 * tol_p_fluid = prm.get_double(
"Tolerance pore pressure");
1151 * prm.leave_subsection();
1157 * <a name=
"nonlinear-poro-viscoelasticity.cc-Time"></a>
1159 * Here we set the timestep size @f$ \varDelta t @f$ and the simulation
end-time.
1175 * prm.enter_subsection(
"Time");
1177 * prm.declare_entry(
"End time",
"10.0",
1181 * prm.declare_entry(
"Time step size",
"0.002",
1183 *
"Time step size. The value must be larger than the displacement error tolerance defined.");
1185 * prm.leave_subsection();
1190 * prm.enter_subsection(
"Time");
1192 * end_time = prm.get_double(
"End time");
1193 * delta_t = prm.get_double(
"Time step size");
1195 * prm.leave_subsection();
1202 * <a name=
"nonlinear-poro-viscoelasticity.cc-Output"></a>
1204 * We can choose the frequency of the data
for the output files.
1207 *
struct OutputParam
1210 * std::string outfiles_requested;
1211 *
unsigned int timestep_output;
1212 * std::string outtype;
1223 * prm.enter_subsection(
"Output parameters");
1225 * prm.declare_entry(
"Output files",
"true",
1227 *
"Paraview output files to generate.");
1228 * prm.declare_entry(
"Time step number output",
"1",
1230 *
"Output data for time steps multiple of the given "
1231 *
"integer value.");
1232 * prm.declare_entry(
"Averaged results",
"nodes",
1234 *
"Output data associated with integration point values"
1235 *
" averaged on elements or on nodes.");
1237 * prm.leave_subsection();
1242 * prm.enter_subsection(
"Output parameters");
1244 * outfiles_requested = prm.get(
"Output files");
1245 * timestep_output = prm.get_integer(
"Time step number output");
1246 * outtype = prm.get(
"Averaged results");
1248 * prm.leave_subsection();
1254 * <a name=
"nonlinear-poro-viscoelasticity.cc-Allparameters"></a>
1255 * <h4>All parameters</h4>
1256 * We
finally consolidate all of the above structures into a single container that holds all the
run-time selections.
1259 *
struct AllParameters :
public FESystem,
1262 *
public NonlinearSolver,
1264 *
public OutputParam
1266 * AllParameters(
const std::string &input_file);
1269 * declare_parameters(ParameterHandler &prm);
1272 * parse_parameters(ParameterHandler &prm);
1275 * AllParameters::AllParameters(
const std::string &input_file)
1278 * declare_parameters(prm);
1279 * prm.parse_input(input_file);
1280 * parse_parameters(prm);
1285 * FESystem::declare_parameters(prm);
1286 * Geometry::declare_parameters(prm);
1287 * Materials::declare_parameters(prm);
1288 * NonlinearSolver::declare_parameters(prm);
1289 * Time::declare_parameters(prm);
1290 * OutputParam::declare_parameters(prm);
1295 * FESystem::parse_parameters(prm);
1296 * Geometry::parse_parameters(prm);
1297 * Materials::parse_parameters(prm);
1298 * NonlinearSolver::parse_parameters(prm);
1299 * Time::parse_parameters(prm);
1300 * OutputParam::parse_parameters(prm);
1307 * <a name=
"nonlinear-poro-viscoelasticity.cc-Timeclass"></a>
1308 * <h3>Time
class</h3>
1309 *
A simple
class to store time data.
1310 * For simplicity we assume a
constant time step size.
1316 * Time (
const double time_end,
1317 *
const double delta_t)
1320 * time_current(0.0),
1321 * time_end(time_end),
1328 *
double get_current() const
1330 *
return time_current;
1332 *
double get_end() const
1336 *
double get_delta_t() const
1340 *
unsigned int get_timestep() const
1344 *
void increment_time ()
1346 * time_current += delta_t;
1351 *
unsigned int timestep;
1352 *
double time_current;
1354 *
const double delta_t;
1360 * <a name=
"nonlinear-poro-viscoelasticity.cc-Constitutiveequationforthesolidcomponentofthebiphasicmaterial"></a>
1361 * <h3>Constitutive equation
for the solid component of the biphasic material</h3>
1366 * <a name=
"nonlinear-poro-viscoelasticity.cc-Baseclassgenerichyperelasticmaterial"></a>
1367 * <h4>Base
class:
generic hyperelastic material</h4>
1368 * The
"extra" Kirchhoff stress in the solid component is the
sum of isochoric
1369 * and a volumetric part:
1370 * @f$\mathbf{\tau} = \mathbf{\tau}_E^{(\bullet)} + \mathbf{\tau}^{\textrm{vol}}@f$.
1371 * The deviatoric part changes depending on the type of material model selected:
1372 * Neo-Hooken hyperelasticity, Ogden hyperelasticiy,
1373 * or a single-mode finite viscoelasticity based on the Ogden hyperelastic model.
1374 * In
this base
class we declare it as a virtual function,
1375 * and it will be defined
for each model type in the corresponding derived
class.
1376 * We define here the volumetric component, which depends on the
1377 * extension function @f$U(J_S)@f$ selected, and in
this case is the same
for all models.
1378 * We use the function proposed by
1379 * Ehlers & Eipper 1999 doi:10.1023/
A:1006565509095.
1380 * We also define some
public functions to access and update the
internal variables.
1383 *
template <
int dim,
typename NumberType = Sacado::Fad::DFad<
double> >
1384 *
class Material_Hyperelastic
1387 * Material_Hyperelastic(
const Parameters::AllParameters ¶meters,
1390 * n_OS (parameters.solid_vol_frac),
1391 *
lambda (parameters.lambda),
1394 * det_F_converged (1.0),
1395 * eigen_solver (parameters.eigen_solver)
1397 * ~Material_Hyperelastic()
1403 *
return ( get_tau_E_base(F) + get_tau_E_ext_func(F) );
1410 *
Assert(det_F > 0, ExcInternalError());
1411 *
return get_tau_E(F)*NumberType(1/det_F);
1415 * get_converged_det_F() const
1417 *
return det_F_converged;
1421 * update_end_timestep()
1423 * det_F_converged = det_F;
1433 * get_viscous_dissipation( )
const = 0;
1435 *
const double n_OS;
1439 *
double det_F_converged;
1447 *
Assert(det_F > 0, ExcInternalError());
1451 *
return ( NumberType(lambda * (1.0-n_OS)*(1.0-n_OS)
1452 * * (det_F/(1.0-n_OS) - det_F/(det_F-n_OS))) * I );
1462 * <a name=
"nonlinear-poro-viscoelasticity.cc-DerivedclassNeoHookeanhyperelasticmaterial"></a>
1463 * <h4>Derived
class: Neo-Hookean hyperelastic material</h4>
1466 *
template <
int dim,
typename NumberType = Sacado::Fad::DFad<
double> >
1467 *
class NeoHooke :
public Material_Hyperelastic < dim, NumberType >
1470 * NeoHooke(
const Parameters::AllParameters ¶meters,
1473 * Material_Hyperelastic< dim, NumberType > (parameters,time),
1476 *
virtual ~NeoHooke()
1480 * get_viscous_dissipation() const override
1488 * SymmetricTensor<2, dim, NumberType>
1489 * get_tau_E_base(
const Tensor<2,dim, NumberType> &F)
const override
1491 *
static const SymmetricTensor< 2, dim, double>
1494 *
const bool use_standard_model =
true;
1496 *
if (use_standard_model)
1500 * Standard Neo-Hooke
1509 * Neo-Hooke in terms of principal stretches
1512 *
const SymmetricTensor<2, dim, NumberType>
1514 *
const std::array< std::pair< NumberType, Tensor< 1, dim, NumberType > >, dim >
1517 * SymmetricTensor<2, dim, NumberType> B_ev;
1518 *
for (
unsigned int d=0;
d<dim; ++
d)
1521 *
return ( mu*(B_ev-I) );
1529 * <a name=
"nonlinear-poro-viscoelasticity.cc-DerivedclassOgdenhyperelasticmaterial"></a>
1530 * <h4>Derived
class: Ogden hyperelastic material</h4>
1533 *
template <
int dim,
typename NumberType = Sacado::Fad::DFad<
double> >
1534 *
class Ogden :
public Material_Hyperelastic < dim, NumberType >
1537 * Ogden(
const Parameters::AllParameters ¶meters,
1540 * Material_Hyperelastic< dim, NumberType > (parameters,time),
1541 * mu({parameters.mu1_infty,
1542 * parameters.mu2_infty,
1543 * parameters.mu3_infty}),
1544 * alpha({parameters.alpha1_infty,
1545 * parameters.alpha2_infty,
1546 * parameters.alpha3_infty})
1552 * get_viscous_dissipation() const override
1558 * std::vector<double> mu;
1559 * std::vector<double> alpha;
1561 * SymmetricTensor<2, dim, NumberType>
1562 * get_tau_E_base(
const Tensor<2,dim, NumberType> &F)
const override
1564 *
const SymmetricTensor<2, dim, NumberType>
1567 *
const std::array< std::pair< NumberType, Tensor< 1, dim, NumberType > >, dim >
1570 * SymmetricTensor<2, dim, NumberType> tau;
1571 *
static const SymmetricTensor< 2, dim, double>
1574 *
for (
unsigned int i = 0; i < 3; ++i)
1576 *
for (
unsigned int A = 0;
A < dim; ++
A)
1578 * SymmetricTensor<2, dim, NumberType> tau_aux1 =
symmetrize(
1580 * tau_aux1 *= mu[i]*
std::pow(eigen_B[A].first, (alpha[i]/2.) );
1583 * SymmetricTensor<2, dim, NumberType> tau_aux2 (I);
1584 * tau_aux2 *= mu[i];
1594 * <a name=
"nonlinear-poro-viscoelasticity.cc-DerivedclassSinglemodeOgdenviscoelasticmaterial"></a>
1595 * <h4>Derived
class: Single-mode Ogden viscoelastic material</h4>
1596 * We use the finite viscoelastic model described in
1597 * Reese & Govindjee (1998) doi:10.1016/S0020-7683(97)00217-5
1598 * The algorithm for the implicit exponential time integration is given in
1599 * Budday et al. (2017) doi: 10.1016/j.actbio.2017.06.024
1602 * template <
int dim, typename NumberType = Sacado::Fad::DFad<
double> >
1603 * class visco_Ogden : public Material_Hyperelastic < dim, NumberType >
1606 * visco_Ogden(
const Parameters::AllParameters ¶meters,
1609 * Material_Hyperelastic< dim, NumberType > (parameters,time),
1610 * mu_infty({parameters.mu1_infty,
1611 * parameters.mu2_infty,
1612 * parameters.mu3_infty}),
1613 * alpha_infty({parameters.alpha1_infty,
1614 * parameters.alpha2_infty,
1615 * parameters.alpha3_infty}),
1616 * mu_mode_1({parameters.mu1_mode_1,
1617 * parameters.mu2_mode_1,
1618 * parameters.mu3_mode_1}),
1619 * alpha_mode_1({parameters.alpha1_mode_1,
1620 * parameters.alpha2_mode_1,
1621 * parameters.alpha3_mode_1}),
1622 * viscosity_mode_1(parameters.viscosity_mode_1),
1626 *
virtual ~visco_Ogden()
1632 * Material_Hyperelastic < dim, NumberType >::update_internal_equilibrium(
F);
1634 * this->Cinv_v_1 = this->Cinv_v_1_converged;
1637 *
const std::array< std::pair< NumberType, Tensor< 1, dim, NumberType > >, dim >
1638 * eigen_B_e_1_tr =
eigenvectors(B_e_1_tr, this->eigen_solver);
1642 *
for (
int a = 0; a < dim; ++a)
1644 * lambdas_e_1_tr[a] =
std::sqrt(eigen_B_e_1_tr[a].first);
1645 * epsilon_e_1_tr[a] =
std::log(lambdas_e_1_tr[a]);
1648 *
const double tolerance = 1
e-8;
1649 *
double residual_check = tolerance*10.0;
1655 * std::vector<NumberType> lambdas_e_1_iso(dim);
1657 *
int iteration = 0;
1661 * epsilon_e_1 = epsilon_e_1_tr;
1663 *
while(residual_check > tolerance)
1665 * NumberType aux_J_e_1 = 1.0;
1666 *
for (
unsigned int a = 0; a < dim; ++a)
1668 * lambdas_e_1[a] =
std::exp(epsilon_e_1[a]);
1669 * aux_J_e_1 *= lambdas_e_1[a];
1672 * J_e_1 = aux_J_e_1;
1674 *
for (
unsigned int a = 0; a < dim; ++a)
1675 * lambdas_e_1_iso[a] = lambdas_e_1[a]*
std::pow(J_e_1,-1.0/dim);
1677 *
for (
unsigned int a = 0; a < dim; ++a)
1679 * residual[a] = get_beta_mode_1(lambdas_e_1_iso, a);
1680 * residual[a] *= this->time.get_delta_t()/(2.0*viscosity_mode_1);
1681 * residual[a] += epsilon_e_1[a];
1682 * residual[a] -= epsilon_e_1_tr[a];
1684 *
for (
unsigned int b = 0;
b < dim; ++
b)
1686 * tangent[a][
b] = get_gamma_mode_1(lambdas_e_1_iso, a, b);
1687 * tangent[a][
b] *= this->time.get_delta_t()/(2.0*viscosity_mode_1);
1688 * tangent[a][
b] += I[a][
b];
1692 * epsilon_e_1 -=
invert(tangent)*residual;
1694 * residual_check = 0.0;
1695 *
for (
unsigned int a = 0; a < dim; ++a)
1697 *
if (
std::abs(residual[a]) > residual_check)
1701 *
if (iteration > 15 )
1702 *
AssertThrow(
false, ExcMessage(
"No convergence in local Newton iteration for the "
1703 *
"viscoelastic exponential time integration algorithm."));
1706 * NumberType aux_J_e_1 = 1.0;
1707 *
for (
unsigned int a = 0; a < dim; ++a)
1709 * lambdas_e_1[a] =
std::exp(epsilon_e_1[a]);
1710 * aux_J_e_1 *= lambdas_e_1[a];
1712 * J_e_1 = aux_J_e_1;
1714 *
for (
unsigned int a = 0; a < dim; ++a)
1715 * lambdas_e_1_iso[a] = lambdas_e_1[a]*
std::pow(J_e_1,-1.0/dim);
1717 *
for (
unsigned int a = 0; a < dim; ++a)
1721 * B_e_1_aux *= lambdas_e_1[a] * lambdas_e_1[a];
1722 * B_e_1 += B_e_1_aux;
1727 * this->tau_neq_1 = 0;
1728 *
for (
unsigned int a = 0; a < dim; ++a)
1732 * tau_neq_1_aux *= get_beta_mode_1(lambdas_e_1_iso, a);
1733 * this->tau_neq_1 += tau_neq_1_aux;
1741 *
for (
unsigned int a = 0; a < dim; ++a)
1742 *
for (
unsigned int b = 0;
b < dim; ++
b)
1746 *
void update_end_timestep() override
1748 * Material_Hyperelastic < dim, NumberType >::update_end_timestep();
1749 * this->Cinv_v_1_converged = this->Cinv_v_1;
1752 *
double get_viscous_dissipation() const override
1754 * NumberType dissipation_term = get_tau_E_neq() * get_tau_E_neq();
1755 * dissipation_term /= (2*viscosity_mode_1);
1757 *
return dissipation_term.val();
1761 * std::vector<double> mu_infty;
1762 * std::vector<double> alpha_infty;
1763 * std::vector<double> mu_mode_1;
1764 * std::vector<double> alpha_mode_1;
1765 *
double viscosity_mode_1;
1773 *
return ( get_tau_E_neq() + get_tau_E_eq(F) );
1781 * std::array< std::pair< NumberType, Tensor< 1, dim, NumberType > >, dim > eigen_B;
1788 *
for (
unsigned int i = 0; i < 3; ++i)
1790 *
for (
unsigned int A = 0;
A < dim; ++
A)
1794 * tau_aux1 *= mu_infty[i]*
std::pow(eigen_B[A].first, (alpha_infty[i]/2.) );
1798 * tau_aux2 *= mu_infty[i];
1805 * get_tau_E_neq() const
1811 * get_beta_mode_1(std::vector< NumberType > &lambda,
const int &A)
const
1813 * NumberType beta = 0.0;
1815 *
for (
unsigned int i = 0; i < 3; ++i)
1818 * NumberType aux = 0.0;
1819 *
for (
int p = 0; p < dim; ++p)
1820 * aux +=
std::pow(lambda[p],alpha_mode_1[i]);
1823 * aux +=
std::pow(lambda[A], alpha_mode_1[i]);
1824 * aux *= mu_mode_1[i];
1832 * get_gamma_mode_1(std::vector< NumberType > &lambda,
1834 *
const int &B )
const
1836 * NumberType
gamma = 0.0;
1840 *
for (
unsigned int i = 0; i < 3; ++i)
1842 * NumberType aux = 0.0;
1843 *
for (
int p = 0; p < dim; ++p)
1844 * aux +=
std::pow(lambda[p],alpha_mode_1[i]);
1846 * aux *= 1.0/(dim*dim);
1847 * aux += 1.0/dim *
std::pow(lambda[A], alpha_mode_1[i]);
1848 * aux *= mu_mode_1[i]*alpha_mode_1[i];
1855 *
for (
unsigned int i = 0; i < 3; ++i)
1857 * NumberType aux = 0.0;
1858 *
for (
int p = 0; p < dim; ++p)
1859 * aux +=
std::pow(lambda[p],alpha_mode_1[i]);
1861 * aux *= 1.0/(dim*dim);
1862 * aux -= 1.0/dim *
std::pow(lambda[A], alpha_mode_1[i]);
1863 * aux -= 1.0/dim *
std::pow(lambda[B], alpha_mode_1[i]);
1864 * aux *= mu_mode_1[i]*alpha_mode_1[i];
1878 * <a name=
"nonlinear-poro-viscoelasticity.cc-Constitutiveequationforthefluidcomponentofthebiphasicmaterial"></a>
1879 * <h3>Constitutive equation
for the fluid component of the biphasic material</h3>
1880 * We consider two slightly different definitions to define the seepage velocity with a Darcy-like law.
1881 * Ehlers & Eipper 1999, doi:10.1023/
A:1006565509095
1882 * Markert 2007, doi:10.1007/s11242-007-9107-6
1883 * The selection of
one or another is made by the user via the parameters file.
1886 *
template <
int dim,
typename NumberType = Sacado::Fad::DFad<
double> >
1887 *
class Material_Darcy_Fluid
1890 * Material_Darcy_Fluid(
const Parameters::AllParameters ¶meters)
1892 * fluid_type(parameters.fluid_type),
1893 * n_OS(parameters.solid_vol_frac),
1894 * initial_intrinsic_permeability(parameters.init_intrinsic_perm),
1895 * viscosity_FR(parameters.viscosity_FR),
1896 * initial_darcy_coefficient(parameters.init_darcy_coef),
1897 * weight_FR(parameters.weight_FR),
1898 * kappa_darcy(parameters.kappa_darcy),
1899 * gravity_term(parameters.gravity_term),
1900 * density_FR(parameters.density_FR),
1901 * gravity_direction(parameters.gravity_direction),
1902 * gravity_value(parameters.gravity_value)
1904 *
Assert(kappa_darcy >= 0, ExcInternalError());
1906 * ~Material_Darcy_Fluid()
1914 *
Assert(det_F > 0.0, ExcInternalError());
1918 *
if (fluid_type ==
"Markert")
1919 * permeability_term = get_instrinsic_permeability_current(F) / viscosity_FR;
1921 *
else if (fluid_type ==
"Ehlers")
1922 * permeability_term = get_darcy_flow_current(F) / weight_FR;
1926 *
"Material_Darcy_Fluid --> Only Markert "
1927 *
"and Ehlers formulations have been implemented."));
1929 *
return ( -1.0 * permeability_term * det_F
1930 * * (grad_p_fluid - get_body_force_FR_current()) );
1936 * NumberType dissipation_term;
1941 *
Assert(det_F > 0.0, ExcInternalError());
1943 *
if (fluid_type ==
"Markert")
1945 * permeability_term = get_instrinsic_permeability_current(F) / viscosity_FR;
1946 * seepage_velocity = get_seepage_velocity_current(F,grad_p_fluid);
1948 *
else if (fluid_type ==
"Ehlers")
1950 * permeability_term = get_darcy_flow_current(F) / weight_FR;
1951 * seepage_velocity = get_seepage_velocity_current(F,grad_p_fluid);
1955 *
"Material_Darcy_Fluid --> Only Markert and Ehlers "
1956 *
"formulations have been implemented."));
1958 * dissipation_term = (
invert(permeability_term) * seepage_velocity ) * seepage_velocity;
1959 * dissipation_term *= 1.0/(det_F*det_F);
1964 *
const std::string fluid_type;
1965 *
const double n_OS;
1966 *
const double initial_intrinsic_permeability;
1967 *
const double viscosity_FR;
1968 *
const double initial_darcy_coefficient;
1969 *
const double weight_FR;
1970 *
const double kappa_darcy;
1971 *
const bool gravity_term;
1972 *
const double density_FR;
1973 *
const int gravity_direction;
1974 *
const double gravity_value;
1985 *
Assert(det_F > 0.0, ExcInternalError());
1987 *
const NumberType fraction = (det_F - n_OS)/(1 - n_OS);
1988 *
return ( NumberType (
std::pow(fraction, kappa_darcy))
1989 * * initial_instrinsic_permeability_tensor );
2001 *
Assert(det_F > 0.0, ExcInternalError());
2003 *
const NumberType fraction = (1.0 - (n_OS / det_F) )/(1.0 - n_OS);
2004 *
return ( NumberType (
std::pow(fraction, kappa_darcy))
2005 * * initial_darcy_flow_tensor);
2009 * get_body_force_FR_current() const
2013 *
if (gravity_term ==
true)
2016 * gravity_vector[gravity_direction] = gravity_value;
2017 * body_force_FR_current = density_FR * gravity_vector;
2019 *
return body_force_FR_current;
2026 * <a name=
"nonlinear-poro-viscoelasticity.cc-Quadraturepointhistory"></a>
2028 * As seen in @ref step_18
"step-18", the <code> PointHistory </code>
class offers a method
2029 *
for storing data at the quadrature points. Here each quadrature
point
2030 * holds a pointer to a material description. Thus, different material models
2031 * can be used in different regions of the domain. Among other data, we
2032 * choose to store the
"extra" Kirchhoff stress @f$\boldsymbol{\tau}_E@f$ and
2033 * the dissipation values @f$\mathcal{D}_p@f$ and @f$\mathcal{D}_v@f$.
2036 *
template <
int dim,
typename NumberType = Sacado::Fad::DFad<
double> >
2037 *
class PointHistory
2043 *
virtual ~PointHistory()
2046 *
void setup_lqp (
const Parameters::AllParameters ¶meters,
2049 *
if (parameters.mat_type ==
"Neo-Hooke")
2050 * solid_material.reset(
new NeoHooke<dim,NumberType>(parameters,time));
2051 *
else if (parameters.mat_type ==
"Ogden")
2052 * solid_material.reset(
new Ogden<dim,NumberType>(parameters,time));
2053 *
else if (parameters.mat_type ==
"visco-Ogden")
2054 * solid_material.reset(
new visco_Ogden<dim,NumberType>(parameters,time));
2056 *
Assert (
false, ExcMessage(
"Material type not implemented"));
2058 * fluid_material.reset(
new Material_Darcy_Fluid<dim,NumberType>(parameters));
2064 *
return solid_material->get_tau_E(F);
2070 *
return solid_material->get_Cauchy_E(F);
2074 * get_converged_det_F() const
2076 *
return solid_material->get_converged_det_F();
2080 * update_end_timestep()
2082 * solid_material->update_end_timestep();
2088 * solid_material->update_internal_equilibrium(F);
2092 * get_viscous_dissipation() const
2094 *
return solid_material->get_viscous_dissipation();
2101 *
return fluid_material->get_seepage_velocity_current(F, grad_p_fluid);
2108 *
return fluid_material->get_porous_dissipation(F, grad_p_fluid);
2113 *
const Parameters::AllParameters ¶meters)
const
2117 *
if (parameters.gravity_term ==
true)
2120 *
Assert(det_F_AD > 0.0, ExcInternalError());
2122 *
const NumberType overall_density_ref
2123 * = parameters.density_SR * parameters.solid_vol_frac
2124 * + parameters.density_FR
2125 * * (det_F_AD - parameters.solid_vol_frac);
2128 * gravity_vector[parameters.gravity_direction] = parameters.gravity_value;
2129 * body_force = overall_density_ref * gravity_vector;
2132 *
return body_force;
2135 * std::shared_ptr< Material_Hyperelastic<dim, NumberType> > solid_material;
2136 * std::shared_ptr< Material_Darcy_Fluid<dim, NumberType> > fluid_material;
2142 * <a name=
"nonlinear-poro-viscoelasticity.cc-Nonlinearporoviscoelasticsolid"></a>
2143 * <h3>Nonlinear poro-viscoelastic solid</h3>
2144 * The Solid
class is the central class as it represents the problem at hand:
2145 * the nonlinear poro-viscoelastic solid
2148 * template <int dim>
2152 * Solid(
const Parameters::AllParameters ¶meters);
2157 *
using ADNumberType = Sacado::Fad::DFad<double>;
2159 * std::ofstream outfile;
2160 * std::ofstream pointfile;
2162 *
struct PerTaskData_ASM;
2163 *
template<
typename NumberType =
double>
struct ScratchData_ASM;
2170 *
virtual void make_grid() = 0;
2174 * Define points
for post-processing
2177 *
virtual void define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) = 0;
2181 * Set up the finite element system to be solved:
2184 *
void system_setup(TrilinosWrappers::MPI::BlockVector &solution_delta_OUT);
2188 * Extract sub-blocks from the global
matrix
2191 *
void determine_component_extractors();
2195 * Several
functions to
assemble the system and right hand side matrices
using multithreading.
2198 *
void assemble_system
2199 * (
const TrilinosWrappers::MPI::BlockVector &solution_delta_OUT );
2200 *
void assemble_system_one_cell
2202 * ScratchData_ASM<ADNumberType> &scratch,
2203 * PerTaskData_ASM &data)
const;
2204 *
void copy_local_to_global_system(
const PerTaskData_ASM &data);
2208 * Define boundary conditions
2211 *
virtual void make_constraints(
const int &it_nr);
2212 *
virtual void make_dirichlet_constraints(AffineConstraints<double> &constraints) = 0;
2213 *
virtual Tensor<1,dim> get_neumann_traction
2215 *
const Point<dim> &pt,
2216 *
const Tensor<1,dim> &N)
const = 0;
2217 *
virtual double get_prescribed_fluid_flow
2219 *
const Point<dim> &pt)
const = 0;
2221 * get_reaction_boundary_id_for_output ()
const = 0;
2222 *
virtual std::pair<types::boundary_id,types::boundary_id>
2223 * get_drained_boundary_id_for_output ()
const = 0;
2224 *
virtual std::vector<double> get_dirichlet_load
2226 *
const int &direction)
const = 0;
2230 * Create and update the quadrature points.
2237 * Solve non-linear system
using a Newton-Raphson scheme
2240 *
void solve_nonlinear_timestep(TrilinosWrappers::MPI::BlockVector &solution_delta_OUT);
2244 * Solve the linearized equations
using a direct solver
2247 *
void solve_linear_system ( TrilinosWrappers::MPI::BlockVector &newton_update_OUT);
2251 * Retrieve the solution
2254 * TrilinosWrappers::MPI::BlockVector
2255 * get_total_solution(
const TrilinosWrappers::MPI::BlockVector &solution_delta_IN)
const;
2259 * Store the converged
values of the internal variables at the
end of each timestep
2262 *
void update_end_timestep();
2266 * Post-processing and writing data to files
2269 *
void output_results_to_vtu(
const unsigned int timestep,
2270 *
const double current_time,
2271 * TrilinosWrappers::MPI::BlockVector solution)
const;
2272 *
void output_results_to_plot(
const unsigned int timestep,
2273 *
const double current_time,
2274 * TrilinosWrappers::MPI::BlockVector solution,
2275 * std::vector<Point<dim> > &tracked_vertices,
2276 * std::ofstream &pointfile)
const;
2280 * Headers and footer
for the output files
2283 *
void print_console_file_header( std::ofstream &outfile)
const;
2284 *
void print_plot_file_header(std::vector<Point<dim> > &tracked_vertices,
2285 * std::ofstream &pointfile)
const;
2286 *
void print_console_file_footer(std::ofstream &outfile)
const;
2287 *
void print_plot_file_footer( std::ofstream &pointfile)
const;
2291 * For parallel communication
2294 * MPI_Comm mpi_communicator;
2297 *
mutable ConditionalOStream pcout;
2301 *
A collection of the parameters used to describe the problem setup
2304 *
const Parameters::AllParameters ¶meters;
2308 * Declare an instance of dealii Triangulation
class (mesh)
2311 * parallel::shared::Triangulation<dim> triangulation;
2315 * Keep track of the current time and the time spent evaluating certain
functions
2319 * TimerOutput timerconsole;
2320 * TimerOutput timerfile;
2324 *
A storage
object for quadrature
point information.
2327 * CellDataStorage<typename Triangulation<dim>::cell_iterator, PointHistory<dim,ADNumberType> > quadrature_point_history;
2331 * Integers to store polynomial degree (needed
for output)
2334 *
const unsigned int degree_displ;
2335 *
const unsigned int degree_pore;
2339 * Declare an instance of dealii FESystem
class (finite element definition)
2342 *
const FESystem<dim> fe;
2346 * Declare an instance of dealii DoFHandler
class (assign DoFs to mesh)
2349 * DoFHandler<dim> dof_handler_ref;
2353 * Integer to store DoFs per element (
this value will be used often)
2356 *
const unsigned int dofs_per_cell;
2360 * Declare an instance of dealii Extractor objects used to retrieve information from the solution vectors
2361 * We will use
"u_fe" and
"p_fluid_fe"as subscript in
operator [] expressions on FEValues and FEFaceValues
2362 * objects to
extract the components of the displacement vector and fluid pressure, respectively.
2365 *
const FEValuesExtractors::Vector u_fe;
2366 *
const FEValuesExtractors::Scalar p_fluid_fe;
2370 * Description of how the block-system is arranged. There are 3 blocks:
2371 * 0 - vector DOF displacements u
2372 * 1 -
scalar DOF fluid pressure p_fluid
2375 *
static const unsigned int n_blocks = 2;
2376 *
static const unsigned int n_components = dim+1;
2377 *
static const unsigned int first_u_component = 0;
2378 *
static const unsigned int p_fluid_component = dim;
2391 *
const FEValuesExtractors::Scalar x_displacement;
2392 *
const FEValuesExtractors::Scalar y_displacement;
2393 *
const FEValuesExtractors::Scalar z_displacement;
2394 *
const FEValuesExtractors::Scalar pressure;
2401 * std::vector<unsigned int> block_component;
2408 * std::vector<IndexSet> all_locally_owned_dofs;
2409 * IndexSet locally_owned_dofs;
2410 * IndexSet locally_relevant_dofs;
2411 * std::vector<IndexSet> locally_owned_partitioning;
2412 * std::vector<IndexSet> locally_relevant_partitioning;
2414 * std::vector<types::global_dof_index> dofs_per_block;
2415 * std::vector<types::global_dof_index> element_indices_u;
2416 * std::vector<types::global_dof_index> element_indices_p_fluid;
2420 * Declare an instance of dealii QGauss
class (The Gauss-Legendre family of quadrature rules
for numerical integration)
2421 * Gauss Points in element, with n quadrature points (in each of the dim space directions)
2424 *
const QGauss<dim> qf_cell;
2427 * Gauss Points on element faces (used
for definition of BCs)
2430 *
const QGauss<dim - 1> qf_face;
2433 * Integer to store num GPs per element (
this value will be used often)
2436 *
const unsigned int n_q_points;
2439 * Integer to store num GPs per face (
this value will be used often)
2442 *
const unsigned int n_q_points_f;
2446 * Declare an instance of dealii AffineConstraints
class (linear constraints on DoFs due to hanging nodes or BCs)
2449 * AffineConstraints<double> constraints;
2453 * Declare an instance of dealii classes necessary
for FE system set-up and assembly
2454 * Store elements of tangent
matrix (indicated by SparsityPattern
class) as sparse
matrix (more efficient)
2457 * TrilinosWrappers::BlockSparseMatrix tangent_matrix;
2458 * TrilinosWrappers::BlockSparseMatrix tangent_matrix_preconditioner;
2461 * Right hand side vector of forces
2464 * TrilinosWrappers::MPI::BlockVector system_rhs;
2467 * Total displacement
values + pressure (accumulated solution to FE system)
2470 * TrilinosWrappers::MPI::BlockVector solution_n;
2474 * Non-block system
for the direct solver. We will
copy the block system into these to solve the linearized system of equations.
2477 * TrilinosWrappers::SparseMatrix tangent_matrix_nb;
2478 * TrilinosWrappers::MPI::Vector system_rhs_nb;
2482 * We define variables to store norms and update norms and normalisation factors.
2489 *
norm(1.0), u(1.0), p_fluid(1.0)
2498 *
void normalise(
const Errors &rhs)
2500 *
if (rhs.norm != 0.0)
2504 *
if (rhs.p_fluid != 0.0)
2505 * p_fluid /= rhs.p_fluid;
2508 *
double norm, u, p_fluid;
2513 * Declare several instances of the
"Error" structure
2516 * Errors error_residual, error_residual_0, error_residual_norm, error_update,
2517 * error_update_0, error_update_norm;
2521 * Methods to calculate error measures
2524 *
void get_error_residual(Errors &error_residual_OUT);
2525 *
void get_error_update
2526 * (
const TrilinosWrappers::MPI::BlockVector &newton_update_IN,
2527 * Errors &error_update_OUT);
2531 * Print information to screen
2534 *
void print_conv_header();
2535 *
void print_conv_footer();
2539 * NOTE: In all
functions, we pass by reference (&), so these
functions work on the original
copy (not a clone copy),
2548 * <a name=
"nonlinear-poro-viscoelasticity.cc-ImplementationofthecodeSolidcodeclass"></a>
2549 * <h3>Implementation of the <code>Solid</code>
class</h3>
2551 * <a name=
"nonlinear-poro-viscoelasticity.cc-Publicinterface"></a>
2552 * <h4>Public interface</h4>
2553 * We initialise the Solid
class using data extracted from the parameter file.
2556 *
template <
int dim>
2557 * Solid<dim>::Solid(
const Parameters::AllParameters ¶meters)
2559 * mpi_communicator(MPI_COMM_WORLD),
2562 * pcout(std::cout, this_mpi_process == 0),
2563 * parameters(parameters),
2565 * time(parameters.end_time, parameters.delta_t),
2566 * timerconsole( mpi_communicator,
2570 * timerfile( mpi_communicator,
2574 * degree_displ(parameters.poly_degree_displ),
2575 * degree_pore(parameters.poly_degree_pore),
2576 * fe(
FE_Q<dim>(parameters.poly_degree_displ), dim,
2577 *
FE_Q<dim>(parameters.poly_degree_pore), 1 ),
2578 * dof_handler_ref(triangulation),
2579 * dofs_per_cell (fe.dofs_per_cell),
2580 * u_fe(first_u_component),
2581 * p_fluid_fe(p_fluid_component),
2582 * x_displacement(first_u_component),
2583 * y_displacement(first_u_component+1),
2584 * z_displacement(first_u_component+2),
2585 * pressure(p_fluid_component),
2586 * dofs_per_block(n_blocks),
2587 * qf_cell(parameters.quad_order),
2588 * qf_face(parameters.quad_order),
2589 * n_q_points (qf_cell.size()),
2590 * n_q_points_f (qf_face.size())
2592 *
Assert(dim==3, ExcMessage(
"This problem only works in 3 space dimensions."));
2593 * determine_component_extractors();
2598 * The
class destructor simply clears the data held by the DOFHandler
2601 *
template <
int dim>
2602 * Solid<dim>::~Solid()
2604 * dof_handler_ref.clear();
2609 * Runs the 3D solid problem
2612 *
template <
int dim>
2613 *
void Solid<dim>::run()
2617 * The current solution increment is defined as a block vector to reflect the structure
2618 * of the PDE system, with multiple solution components
2628 *
if (this_mpi_process == 0)
2630 * outfile.open(
"console-output.sol");
2631 * print_console_file_header(outfile);
2643 * Assign DOFs and create the stiffness and right-hand-side force vector
2646 * system_setup(solution_delta);
2650 * Define points
for post-processing
2653 * std::vector<Point<dim> > tracked_vertices (2);
2654 * define_tracked_vertices(tracked_vertices);
2655 * std::vector<Point<dim>> reaction_force;
2657 *
if (this_mpi_process == 0)
2659 * pointfile.open(
"data-for-gnuplot.sol");
2660 * print_plot_file_header(tracked_vertices, pointfile);
2665 * Print results to output file
2668 *
if (parameters.outfiles_requested ==
"true")
2670 * output_results_to_vtu(time.get_timestep(),
2671 * time.get_current(),
2675 * output_results_to_plot(time.get_timestep(),
2676 * time.get_current(),
2683 * Increment time step (=load step)
2684 * NOTE: In solving the quasi-
static problem, the time becomes a loading parameter,
2685 * i.e. we increase the loading linearly with time, making the two
concepts interchangeable.
2688 * time.increment_time();
2692 * Print information on screen
2695 * pcout <<
"\nSolver:";
2696 * pcout <<
"\n CST = make constraints";
2697 * pcout <<
"\n ASM_SYS = assemble system";
2698 * pcout <<
"\n SLV = linear solver \n";
2702 * Print information on file
2705 * outfile <<
"\nSolver:";
2706 * outfile <<
"\n CST = make constraints";
2707 * outfile <<
"\n ASM_SYS = assemble system";
2708 * outfile <<
"\n SLV = linear solver \n";
2710 *
while ( (time.get_end() - time.get_current()) > -1.0*parameters.tol_u )
2714 * Initialize the current solution increment to
zero
2717 * solution_delta = 0.0;
2721 * Solve the non-linear system
using a Newton-Rapshon scheme
2724 * solve_nonlinear_timestep(solution_delta);
2728 * Add the computed solution increment to total solution
2731 * solution_n += solution_delta;
2735 * Store the converged values of the
internal variables
2738 * update_end_timestep();
2745 *
if (( (time.get_timestep()%parameters.timestep_output) == 0 )
2746 * && (parameters.outfiles_requested ==
"true") )
2748 * output_results_to_vtu(time.get_timestep(),
2749 * time.get_current(),
2753 * output_results_to_plot(time.get_timestep(),
2754 * time.get_current(),
2761 * Increment the time step (=load step)
2764 * time.increment_time();
2769 * Print the footers and close files
2772 *
if (this_mpi_process == 0)
2774 * print_plot_file_footer(pointfile);
2775 * pointfile.close ();
2776 * print_console_file_footer(outfile);
2780 * NOTE: ideally, we should close the outfile here [ >> outfile.close (); ]
2781 * But
if we
do, then the timer output will not be printed. That is why we leave it open.
2790 * <a name=
"nonlinear-poro-viscoelasticity.cc-Privateinterface"></a>
2791 * <h4>Private interface</h4>
2792 * We define the structures needed
for parallelization with Threading Building Blocks (TBB)
2793 * Tangent
matrix and right-hand side force vector assembly structures.
2794 * PerTaskData_ASM stores local contributions
2797 *
template <
int dim>
2798 *
struct Solid<dim>::PerTaskData_ASM
2801 * Vector<double> cell_rhs;
2802 * std::vector<types::global_dof_index> local_dof_indices;
2804 * PerTaskData_ASM(
const unsigned int dofs_per_cell)
2807 * cell_rhs(dofs_per_cell),
2808 * local_dof_indices(dofs_per_cell)
2820 * ScratchData_ASM stores larger objects used during the assembly
2823 *
template <
int dim>
2824 *
template <
typename NumberType>
2825 *
struct Solid<dim>::ScratchData_ASM
2827 *
const TrilinosWrappers::MPI::BlockVector &solution_total;
2831 * Integration helper
2834 * FEValues<dim> fe_values_ref;
2835 * FEFaceValues<dim> fe_face_values_ref;
2839 * Quadrature
point solution
2842 * std::vector<NumberType> local_dof_values;
2843 * std::vector<Tensor<2, dim, NumberType> > solution_grads_u_total;
2844 * std::vector<NumberType> solution_values_p_fluid_total;
2845 * std::vector<Tensor<1, dim, NumberType> > solution_grads_p_fluid_total;
2846 * std::vector<Tensor<1, dim, NumberType> > solution_grads_face_p_fluid_total;
2853 * std::vector<std::vector<Tensor<1,dim>>> Nx;
2854 * std::vector<std::vector<double>> Nx_p_fluid;
2860 * std::vector<std::vector<Tensor<2,dim, NumberType>>> grad_Nx;
2861 * std::vector<std::vector<SymmetricTensor<2,dim, NumberType>>> symm_grad_Nx;
2862 * std::vector<std::vector<Tensor<1,dim, NumberType>>> grad_Nx_p_fluid;
2864 * ScratchData_ASM(
const FiniteElement<dim> &fe_cell,
2865 *
const QGauss<dim> &qf_cell,
const UpdateFlags uf_cell,
2866 *
const QGauss<dim - 1> & qf_face,
const UpdateFlags uf_face,
2867 *
const TrilinosWrappers::MPI::BlockVector &solution_total )
2869 * solution_total (solution_total),
2870 * fe_values_ref(fe_cell, qf_cell, uf_cell),
2871 * fe_face_values_ref(fe_cell, qf_face, uf_face),
2872 * local_dof_values(fe_cell.dofs_per_cell),
2873 * solution_grads_u_total(qf_cell.size()),
2874 * solution_values_p_fluid_total(qf_cell.size()),
2875 * solution_grads_p_fluid_total(qf_cell.size()),
2876 * solution_grads_face_p_fluid_total(qf_face.size()),
2877 * Nx(qf_cell.size(), std::vector<Tensor<1,dim>>(fe_cell.dofs_per_cell)),
2878 * Nx_p_fluid(qf_cell.size(), std::vector<double>(fe_cell.dofs_per_cell)),
2879 * grad_Nx(qf_cell.size(), std::vector<Tensor<2, dim, NumberType>>(fe_cell.dofs_per_cell)),
2880 * symm_grad_Nx(qf_cell.size(), std::vector<SymmetricTensor<2, dim, NumberType>> (fe_cell.dofs_per_cell)),
2881 * grad_Nx_p_fluid(qf_cell.size(), std::vector<Tensor<1, dim, NumberType>>(fe_cell.dofs_per_cell))
2884 * ScratchData_ASM(
const ScratchData_ASM &rhs)
2886 * solution_total (rhs.solution_total),
2887 * fe_values_ref(rhs.fe_values_ref.get_fe(),
2888 * rhs.fe_values_ref.get_quadrature(),
2889 * rhs.fe_values_ref.get_update_flags()),
2890 * fe_face_values_ref(rhs.fe_face_values_ref.get_fe(),
2891 * rhs.fe_face_values_ref.get_quadrature(),
2892 * rhs.fe_face_values_ref.get_update_flags()),
2893 * local_dof_values(rhs.local_dof_values),
2894 * solution_grads_u_total(rhs.solution_grads_u_total),
2895 * solution_values_p_fluid_total(rhs.solution_values_p_fluid_total),
2896 * solution_grads_p_fluid_total(rhs.solution_grads_p_fluid_total),
2897 * solution_grads_face_p_fluid_total(rhs.solution_grads_face_p_fluid_total),
2899 * Nx_p_fluid(rhs.Nx_p_fluid),
2900 * grad_Nx(rhs.grad_Nx),
2901 * symm_grad_Nx(rhs.symm_grad_Nx),
2902 * grad_Nx_p_fluid(rhs.grad_Nx_p_fluid)
2907 *
const unsigned int n_q_points = Nx_p_fluid.size();
2908 *
const unsigned int n_dofs_per_cell = Nx_p_fluid[0].size();
2910 *
Assert(local_dof_values.size() == n_dofs_per_cell, ExcInternalError());
2912 *
for (
unsigned int k = 0; k < n_dofs_per_cell; ++k)
2914 * local_dof_values[k] = 0.0;
2917 *
Assert(solution_grads_u_total.size() == n_q_points, ExcInternalError());
2918 *
Assert(solution_values_p_fluid_total.size() == n_q_points, ExcInternalError());
2919 *
Assert(solution_grads_p_fluid_total.size() == n_q_points, ExcInternalError());
2921 *
Assert(Nx.size() == n_q_points, ExcInternalError());
2922 *
Assert(grad_Nx.size() == n_q_points, ExcInternalError());
2923 *
Assert(symm_grad_Nx.size() == n_q_points, ExcInternalError());
2925 *
for (
unsigned int q_point = 0; q_point < n_q_points; ++q_point)
2927 *
Assert( Nx[q_point].size() == n_dofs_per_cell, ExcInternalError());
2928 *
Assert( grad_Nx[q_point].size() == n_dofs_per_cell, ExcInternalError());
2929 *
Assert( symm_grad_Nx[q_point].size() == n_dofs_per_cell, ExcInternalError());
2931 * solution_grads_u_total[q_point] = 0.0;
2932 * solution_values_p_fluid_total[q_point] = 0.0;
2933 * solution_grads_p_fluid_total[q_point] = 0.0;
2935 *
for (
unsigned int k = 0; k < n_dofs_per_cell; ++k)
2937 * Nx[q_point][k] = 0.0;
2938 * Nx_p_fluid[q_point][k] = 0.0;
2939 * grad_Nx[q_point][k] = 0.0;
2940 * symm_grad_Nx[q_point][k] = 0.0;
2941 * grad_Nx_p_fluid[q_point][k] = 0.0;
2945 *
const unsigned int n_f_q_points = solution_grads_face_p_fluid_total.size();
2946 *
Assert(solution_grads_face_p_fluid_total.size() == n_f_q_points, ExcInternalError());
2948 *
for (
unsigned int f_q_point = 0; f_q_point < n_f_q_points; ++f_q_point)
2949 * solution_grads_face_p_fluid_total[f_q_point] = 0.0;
2955 * Define the boundary conditions on the mesh
2958 *
template <
int dim>
2959 *
void Solid<dim>::make_constraints(
const int &it_nr_IN)
2961 * pcout <<
" CST " << std::flush;
2962 * outfile <<
" CST " << std::flush;
2964 *
if (it_nr_IN > 1)
return;
2966 *
const bool apply_dirichlet_bc = (it_nr_IN == 0);
2968 *
if (apply_dirichlet_bc)
2970 * constraints.clear();
2972 * constraints.reinit(locally_owned_dofs, locally_relevant_dofs);
2974 * constraints.reinit(locally_relevant_dofs);
2976 * make_dirichlet_constraints(constraints);
2980 *
for (
const auto i : locally_relevant_dofs)
2981 *
if (constraints.is_inhomogeneously_constrained(i) ==
true)
2982 * constraints.set_inhomogeneity(i,0.0);
2984 * constraints.close();
2989 * Set-up the FE system
2992 *
template <
int dim>
2995 * timerconsole.enter_subsection(
"Setup system");
2996 * timerfile.enter_subsection(
"Setup system");
3000 * Determine number of components per block
3003 * std::vector<unsigned int> block_component(n_components, u_block);
3004 * block_component[p_fluid_component] = p_fluid_block;
3008 * The DOF handler is initialised and we renumber the grid in an efficient manner.
3011 * dof_handler_ref.distribute_dofs(fe);
3017 * Count the number of DoFs in each block
3024 * Setup the sparsity pattern and tangent
matrix
3028 * std::vector<IndexSet> all_locally_relevant_dofs
3031 * locally_owned_dofs.clear();
3032 * locally_owned_partitioning.clear();
3033 *
Assert(all_locally_owned_dofs.size() > this_mpi_process, ExcInternalError());
3036 * locally_relevant_dofs.clear();
3037 * locally_relevant_partitioning.clear();
3038 *
Assert(all_locally_relevant_dofs.size() > this_mpi_process, ExcInternalError());
3041 * locally_owned_partitioning.reserve(n_blocks);
3042 * locally_relevant_partitioning.reserve(n_blocks);
3047 * = std::accumulate(dofs_per_block.begin(),
3048 * std::next(dofs_per_block.begin(),b), 0);
3050 * = std::accumulate(dofs_per_block.begin(),
3051 * std::next(dofs_per_block.begin(),b+1), 0);
3052 * locally_owned_partitioning.push_back(locally_owned_dofs.get_view(idx_begin, idx_end));
3053 * locally_relevant_partitioning.push_back(locally_relevant_dofs.get_view(idx_begin, idx_end));
3058 * Print information on screen
3061 * pcout <<
"\nTriangulation:\n"
3062 * <<
" Number of active cells: "
3063 * << triangulation.n_active_cells()
3064 * <<
" (by partition:";
3066 * pcout << (p==0 ?
' ' :
'+')
3070 * pcout <<
" Number of degrees of freedom: "
3071 * << dof_handler_ref.n_dofs()
3072 * <<
" (by partition:";
3074 * pcout << (p==0 ?
' ' :
'+')
3078 * pcout <<
" Number of degrees of freedom per block: "
3079 * <<
"[n_u, n_p_fluid] = ["
3080 * << dofs_per_block[u_block]
3082 * << dofs_per_block[p_fluid_block]
3088 * Print information to file
3091 * outfile <<
"\nTriangulation:\n"
3092 * <<
" Number of active cells: "
3093 * << triangulation.n_active_cells()
3094 * <<
" (by partition:";
3096 * outfile << (p==0 ?
' ' :
'+')
3100 * outfile <<
" Number of degrees of freedom: "
3101 * << dof_handler_ref.n_dofs()
3102 * <<
" (by partition:";
3104 * outfile << (p==0 ?
' ' :
'+')
3108 * outfile <<
" Number of degrees of freedom per block: "
3109 * <<
"[n_u, n_p_fluid] = ["
3110 * << dofs_per_block[u_block]
3112 * << dofs_per_block[p_fluid_block]
3118 * We optimise the sparsity pattern to reflect
this structure and prevent
3119 * unnecessary data creation
for the right-
diagonal block components.
3123 *
for (
unsigned int ii = 0; ii < n_components; ++ii)
3124 *
for (
unsigned int jj = 0; jj < n_components; ++jj)
3128 * Identify
"zero" matrix components of FE-system (The two components
do not couple)
3131 *
if (((ii == p_fluid_component) && (jj < p_fluid_component))
3132 * || ((ii < p_fluid_component) && (jj == p_fluid_component)) )
3137 * The rest of components
always couple
3144 * mpi_communicator);
3147 *
false, this_mpi_process);
3152 * Reinitialize the (sparse) tangent
matrix with the given sparsity pattern.
3155 * tangent_matrix.reinit (bsp);
3159 * Initialize the right hand side and solution vectors with number of DoFs
3162 * system_rhs.reinit(locally_owned_partitioning, mpi_communicator);
3163 * solution_n.reinit(locally_owned_partitioning, mpi_communicator);
3164 * solution_delta_OUT.reinit(locally_owned_partitioning, mpi_communicator);
3172 * mpi_communicator);
3174 *
false, this_mpi_process);
3176 * tangent_matrix_nb.reinit (sp);
3177 * system_rhs_nb.reinit(locally_owned_dofs, mpi_communicator);
3181 * Set up the quadrature
point history
3186 * timerconsole.leave_subsection();
3187 * timerfile.leave_subsection();
3192 * Component extractors: used to
extract sub-blocks from the global
matrix
3193 * Description of which local element DOFs are attached to which block component
3196 *
template <
int dim>
3197 *
void Solid<dim>::determine_component_extractors()
3199 * element_indices_u.clear();
3200 * element_indices_p_fluid.clear();
3202 *
for (
unsigned int k = 0; k < fe.dofs_per_cell; ++k)
3204 *
const unsigned int k_group = fe.system_to_base_index(k).first.first;
3205 *
if (k_group == u_block)
3206 * element_indices_u.push_back(k);
3207 *
else if (k_group == p_fluid_block)
3208 * element_indices_p_fluid.push_back(k);
3211 *
Assert(k_group <= p_fluid_block, ExcInternalError());
3218 * Set-up quadrature
point history (QPH) data objects
3221 *
template <
int dim>
3222 *
void Solid<dim>::setup_qph()
3224 * pcout <<
"\nSetting up quadrature point data..." << std::endl;
3225 * outfile <<
"\nSetting up quadrature point data..." << std::endl;
3229 * Create QPH data objects.
3232 * quadrature_point_history.initialize(triangulation.begin_active(),
3233 * triangulation.end(), n_q_points);
3237 * Setup the
initial quadrature
point data
using the info stored in parameters
3242 * dof_handler_ref.begin_active()),
3244 * dof_handler_ref.end());
3245 *
for (; cell!=endc; ++cell)
3247 *
Assert(cell->is_locally_owned(), ExcInternalError());
3248 *
Assert(cell->subdomain_id() == this_mpi_process, ExcInternalError());
3250 *
const std::vector<std::shared_ptr<PointHistory<dim, ADNumberType> > >
3251 * lqph = quadrature_point_history.get_data(cell);
3252 *
Assert(lqph.size() == n_q_points, ExcInternalError());
3254 *
for (
unsigned int q_point = 0; q_point < n_q_points; ++q_point)
3255 * lqph[q_point]->setup_lqp(parameters, time);
3261 * Solve the non-linear system
using a Newton-Raphson scheme
3264 *
template <
int dim>
3269 * Print the load step
3272 * pcout << std::endl
3274 * << time.get_timestep()
3276 * << time.get_current()
3279 * outfile << std::endl
3281 * << time.get_timestep()
3283 * << time.get_current()
3289 * Declare newton_update vector (solution of a Newton iteration),
3290 * which must have as many positions as global DoFs.
3294 * (locally_owned_partitioning, mpi_communicator);
3298 * Reset the error storage objects
3301 * error_residual.reset();
3302 * error_residual_0.reset();
3303 * error_residual_norm.reset();
3304 * error_update.reset();
3305 * error_update_0.reset();
3306 * error_update_norm.reset();
3308 * print_conv_header();
3312 * Declare and initialize iterator
for the Newton-Raphson algorithm steps
3315 *
unsigned int newton_iteration = 0;
3319 * Iterate until error is below tolerance or
max number iterations are reached
3322 *
while(newton_iteration < parameters.max_iterations_NR)
3324 * pcout <<
" " << std::setw(2) << newton_iteration <<
" " << std::flush;
3325 * outfile <<
" " << std::setw(2) << newton_iteration <<
" " << std::flush;
3329 * Initialize global stiffness
matrix and global force vector to
zero
3332 * tangent_matrix = 0.0;
3335 * tangent_matrix_nb = 0.0;
3336 * system_rhs_nb = 0.0;
3340 * Apply boundary conditions
3343 * make_constraints(newton_iteration);
3344 * assemble_system(solution_delta_OUT);
3348 * Compute the rhs residual (error between external and
internal forces in FE system)
3351 * get_error_residual(error_residual);
3355 * error_residual in first iteration is stored to normalize posterior error measures
3358 *
if (newton_iteration == 0)
3359 * error_residual_0 = error_residual;
3363 * Determine the normalised residual error
3366 * error_residual_norm = error_residual;
3367 * error_residual_norm.normalise(error_residual_0);
3371 * If both errors are below the tolerances, exit the
loop.
3372 * We need to
check the residual vector directly
for convergence
3373 * in the load steps where no external forces or displacements are imposed.
3376 *
if ( ((newton_iteration > 0)
3377 * && (error_update_norm.u <= parameters.tol_u)
3378 * && (error_update_norm.p_fluid <= parameters.tol_p_fluid)
3379 * && (error_residual_norm.u <= parameters.tol_f)
3380 * && (error_residual_norm.p_fluid <= parameters.tol_f))
3381 * || ( (newton_iteration > 0)
3382 * && system_rhs.l2_norm() <= parameters.tol_f) )
3384 * pcout <<
"\n ***** CONVERGED! ***** "
3385 * << system_rhs.l2_norm() <<
" "
3386 * <<
" " << error_residual_norm.norm
3387 * <<
" " << error_residual_norm.u
3388 * <<
" " << error_residual_norm.p_fluid
3389 * <<
" " << error_update_norm.norm
3390 * <<
" " << error_update_norm.u
3391 * <<
" " << error_update_norm.p_fluid
3392 * <<
" " << std::endl;
3393 * outfile <<
"\n ***** CONVERGED! ***** "
3394 * << system_rhs.l2_norm() <<
" "
3395 * <<
" " << error_residual_norm.norm
3396 * <<
" " << error_residual_norm.u
3397 * <<
" " << error_residual_norm.p_fluid
3398 * <<
" " << error_update_norm.norm
3399 * <<
" " << error_update_norm.u
3400 * <<
" " << error_update_norm.p_fluid
3401 * <<
" " << std::endl;
3402 * print_conv_footer();
3409 * Solve the linearized system
3412 * solve_linear_system(newton_update);
3413 * constraints.distribute(newton_update);
3417 * Compute the displacement error
3420 * get_error_update(newton_update, error_update);
3424 * error_update in first iteration is stored to normalize posterior error measures
3427 *
if (newton_iteration == 0)
3428 * error_update_0 = error_update;
3432 * Determine the normalised Newton update error
3435 * error_update_norm = error_update;
3436 * error_update_norm.normalise(error_update_0);
3440 * Determine the normalised residual error
3443 * error_residual_norm = error_residual;
3444 * error_residual_norm.normalise(error_residual_0);
3448 * Print error values
3451 * pcout <<
" | " << std::fixed << std::setprecision(3)
3452 * << std::setw(7) << std::scientific
3453 * << system_rhs.l2_norm()
3454 * <<
" " << error_residual_norm.norm
3455 * <<
" " << error_residual_norm.u
3456 * <<
" " << error_residual_norm.p_fluid
3457 * <<
" " << error_update_norm.norm
3458 * <<
" " << error_update_norm.u
3459 * <<
" " << error_update_norm.p_fluid
3460 * <<
" " << std::endl;
3462 * outfile <<
" | " << std::fixed << std::setprecision(3)
3463 * << std::setw(7) << std::scientific
3464 * << system_rhs.l2_norm()
3465 * <<
" " << error_residual_norm.norm
3466 * <<
" " << error_residual_norm.u
3467 * <<
" " << error_residual_norm.p_fluid
3468 * <<
" " << error_update_norm.norm
3469 * <<
" " << error_update_norm.u
3470 * <<
" " << error_update_norm.p_fluid
3471 * <<
" " << std::endl;
3478 * solution_delta_OUT += newton_update;
3479 * newton_update = 0.0;
3480 * newton_iteration++;
3485 * If maximum allowed number of iterations
for Newton algorithm are reached, print non-convergence message and
abort program
3488 *
AssertThrow (newton_iteration < parameters.max_iterations_NR, ExcMessage(
"No convergence in nonlinear solver!"));
3493 * Prints the header
for convergence info on console
3496 *
template <
int dim>
3497 *
void Solid<dim>::print_conv_header()
3499 *
static const unsigned int l_width = 120;
3501 *
for (
unsigned int i = 0; i < l_width; ++i)
3507 * pcout << std::endl;
3508 * outfile << std::endl;
3510 * pcout <<
"\n SOLVER STEP | SYS_RES "
3511 * <<
"RES_NORM RES_U RES_P "
3512 * <<
"NU_NORM NU_U NU_P " << std::endl;
3513 * outfile <<
"\n SOLVER STEP | SYS_RES "
3514 * <<
"RES_NORM RES_U RES_P "
3515 * <<
"NU_NORM NU_U NU_P " << std::endl;
3517 *
for (
unsigned int i = 0; i < l_width; ++i)
3522 * pcout << std::endl << std::endl;
3523 * outfile << std::endl << std::endl;
3528 * Prints the footer
for convergence info on console
3531 *
template <
int dim>
3532 *
void Solid<dim>::print_conv_footer()
3534 *
static const unsigned int l_width = 120;
3536 *
for (
unsigned int i = 0; i < l_width; ++i)
3541 * pcout << std::endl << std::endl;
3542 * outfile << std::endl << std::endl;
3544 * pcout <<
"Relative errors:" << std::endl
3545 * <<
"Displacement: "
3546 * << error_update.u / error_update_0.u << std::endl
3547 * <<
"Force (displ): "
3548 * << error_residual.u / error_residual_0.u << std::endl
3549 * <<
"Pore pressure: "
3550 * << error_update.p_fluid / error_update_0.p_fluid << std::endl
3551 * <<
"Force (pore): "
3552 * << error_residual.p_fluid / error_residual_0.p_fluid << std::endl;
3553 * outfile <<
"Relative errors:" << std::endl
3554 * <<
"Displacement: "
3555 * << error_update.u / error_update_0.u << std::endl
3556 * <<
"Force (displ): "
3557 * << error_residual.u / error_residual_0.u << std::endl
3558 * <<
"Pore pressure: "
3559 * << error_update.p_fluid / error_update_0.p_fluid << std::endl
3560 * <<
"Force (pore): "
3561 * << error_residual.p_fluid / error_residual_0.p_fluid << std::endl;
3566 * Determine the
true residual error
for the problem
3569 *
template <
int dim>
3570 *
void Solid<dim>::get_error_residual(Errors &error_residual_OUT)
3573 * constraints.set_zero(error_res);
3575 * error_residual_OUT.norm = error_res.l2_norm();
3576 * error_residual_OUT.u = error_res.block(u_block).l2_norm();
3577 * error_residual_OUT.p_fluid = error_res.block(p_fluid_block).l2_norm();
3582 * Determine the
true Newton update error
for the problem
3585 *
template <
int dim>
3586 *
void Solid<dim>::get_error_update
3588 * Errors &error_update_OUT)
3591 * constraints.set_zero(error_ud);
3593 * error_update_OUT.norm = error_ud.l2_norm();
3594 * error_update_OUT.u = error_ud.block(u_block).l2_norm();
3595 * error_update_OUT.p_fluid = error_ud.block(p_fluid_block).l2_norm();
3600 * Compute the total solution, which is
valid at any Newton step. This is required as, to
reduce
3601 * computational error, the total solution is only updated at the
end of the timestep.
3604 *
template <
int dim>
3610 * Cell interpolation -> Ghosted vector
3614 * solution_total (locally_owned_partitioning,
3615 * locally_relevant_partitioning,
3619 * solution_total = solution_n;
3620 * tmp = solution_delta_IN;
3621 * solution_total += tmp;
3622 *
return solution_total;
3627 * Compute elemental stiffness tensor and right-hand side force vector, and
assemble into global ones
3630 *
template <
int dim>
3633 * timerconsole.enter_subsection(
"Assemble system");
3634 * timerfile.enter_subsection(
"Assemble system");
3635 * pcout <<
" ASM_SYS " << std::flush;
3636 * outfile <<
" ASM_SYS " << std::flush;
3642 * Info given to
FEValues and
FEFaceValues constructors, to indicate which data will be needed at each element.
3656 * Setup a
copy of the data structures required
for the process and pass them, along with the
3660 * PerTaskData_ASM per_task_data(dofs_per_cell);
3661 * ScratchData_ASM<ADNumberType> scratch_data(fe, qf_cell, uf_cell,
3667 * dof_handler_ref.begin_active()),
3669 * dof_handler_ref.end());
3670 *
for (; cell != endc; ++cell)
3672 *
Assert(cell->is_locally_owned(), ExcInternalError());
3673 *
Assert(cell->subdomain_id() == this_mpi_process, ExcInternalError());
3675 * assemble_system_one_cell(cell, scratch_data, per_task_data);
3676 * copy_local_to_global_system(per_task_data);
3684 * timerconsole.leave_subsection();
3685 * timerfile.leave_subsection();
3690 * Add the local elemental contribution to the global stiffness tensor
3691 * We
do it twice,
for the block and the non-block systems
3694 *
template <
int dim>
3695 *
void Solid<dim>::copy_local_to_global_system (
const PerTaskData_ASM &data)
3697 * constraints.distribute_local_to_global(data.cell_matrix,
3699 * data.local_dof_indices,
3703 * constraints.distribute_local_to_global(data.cell_matrix,
3705 * data.local_dof_indices,
3706 * tangent_matrix_nb,
3712 * Compute stiffness
matrix and corresponding rhs
for one element
3715 *
template <
int dim>
3716 *
void Solid<dim>::assemble_system_one_cell
3718 * ScratchData_ASM<ADNumberType> &scratch,
3719 * PerTaskData_ASM &data)
const
3721 *
Assert(cell->is_locally_owned(), ExcInternalError());
3725 * scratch.fe_values_ref.reinit(cell);
3726 * cell->get_dof_indices(data.local_dof_indices);
3730 * Setup automatic differentiation
3733 *
for (
unsigned int k = 0; k < dofs_per_cell; ++k)
3737 * Initialise the dofs
for the cell
using the current solution.
3740 * scratch.local_dof_values[k] = scratch.solution_total[data.local_dof_indices[k]];
3746 * scratch.local_dof_values[k].diff(k, dofs_per_cell);
3751 * Update the quadrature
point solution
3752 * Compute the values and
gradients of the solution in terms of the AD variables
3755 *
for (
unsigned int q = 0; q < n_q_points; ++q)
3757 *
for (
unsigned int k = 0; k < dofs_per_cell; ++k)
3759 *
const unsigned int k_group = fe.system_to_base_index(k).first.first;
3760 *
if (k_group == u_block)
3763 * scratch.fe_values_ref[u_fe].gradient(k, q);
3764 *
for (
unsigned int dd = 0; dd < dim; ++dd)
3766 *
for (
unsigned int ee = 0; ee < dim; ++ee)
3768 * scratch.solution_grads_u_total[q][dd][ee]
3769 * += scratch.local_dof_values[k] * Grad_Nx_u[dd][ee];
3773 *
else if (k_group == p_fluid_block)
3775 *
const double Nx_p = scratch.fe_values_ref[p_fluid_fe].value(k, q);
3777 * scratch.fe_values_ref[p_fluid_fe].gradient(k, q);
3779 * scratch.solution_values_p_fluid_total[q]
3780 * += scratch.local_dof_values[k] * Nx_p;
3781 *
for (
unsigned int dd = 0; dd < dim; ++dd)
3783 * scratch.solution_grads_p_fluid_total[q][dd]
3784 * += scratch.local_dof_values[k] * Grad_Nx_p[dd];
3788 *
Assert(k_group <= p_fluid_block, ExcInternalError());
3794 * Set up pointer
"lgph" to the PointHistory
object of
this element
3797 *
const std::vector<std::shared_ptr<const PointHistory<dim, ADNumberType> > >
3798 * lqph = quadrature_point_history.get_data(cell);
3799 *
Assert(lqph.size() == n_q_points, ExcInternalError());
3804 * Precalculate the element shape function values and
gradients
3807 *
for (
unsigned int q_point = 0; q_point < n_q_points; ++q_point)
3814 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
3816 *
const unsigned int i_group = fe.system_to_base_index(i).first.first;
3818 *
if (i_group == u_block)
3820 * scratch.Nx[q_point][i] =
3821 * scratch.fe_values_ref[u_fe].value(i, q_point);
3822 * scratch.grad_Nx[q_point][i] =
3823 * scratch.fe_values_ref[u_fe].gradient(i, q_point)*F_inv_AD;
3824 * scratch.symm_grad_Nx[q_point][i] =
3827 *
else if (i_group == p_fluid_block)
3829 * scratch.Nx_p_fluid[q_point][i] =
3830 * scratch.fe_values_ref[p_fluid_fe].value(i, q_point);
3831 * scratch.grad_Nx_p_fluid[q_point][i] =
3832 * scratch.fe_values_ref[p_fluid_fe].gradient(i, q_point)*F_inv_AD;
3835 *
Assert(i_group <= p_fluid_block, ExcInternalError());
3841 * Assemble the stiffness
matrix and rhs vector
3844 * std::vector<ADNumberType> residual_ad (dofs_per_cell, ADNumberType(0.0));
3845 *
for (
unsigned int q_point = 0; q_point < n_q_points; ++q_point)
3849 *
const ADNumberType det_F_AD =
determinant(F_AD);
3851 *
Assert(det_F_AD > 0, ExcInternalError());
3854 *
const ADNumberType p_fluid = scratch.solution_values_p_fluid_total[q_point];
3857 * PointHistory<dim, ADNumberType> *lqph_q_point_nc =
3858 *
const_cast<PointHistory<dim, ADNumberType>*
>(lqph[q_point].get());
3859 * lqph_q_point_nc->update_internal_equilibrium(F_AD);
3864 * Get some info from constitutive model of solid
3870 * tau_E = lqph[q_point]->get_tau_E(F_AD);
3872 * tau_fluid_vol *= -1.0 * p_fluid * det_F_AD;
3876 * Get some info from constitutive model of fluid
3879 *
const ADNumberType det_F_aux = lqph[q_point]->get_converged_det_F();
3882 * = lqph[q_point]->get_overall_body_force(F_AD, parameters);
3886 * Define some aliases to make the assembly process easier to follow
3889 *
const std::vector<Tensor<1,dim>> &Nu = scratch.Nx[q_point];
3890 *
const std::vector<SymmetricTensor<2, dim, ADNumberType>>
3891 * &symm_grad_Nu = scratch.symm_grad_Nx[q_point];
3892 *
const std::vector<double> &Np = scratch.Nx_p_fluid[q_point];
3893 *
const std::vector<Tensor<1, dim, ADNumberType> > &grad_Np
3894 * = scratch.grad_Nx_p_fluid[q_point];
3896 * = scratch.solution_grads_p_fluid_total[q_point]*F_inv_AD;
3897 *
const double JxW = scratch.fe_values_ref.JxW(q_point);
3899 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
3901 *
const unsigned int i_group = fe.system_to_base_index(i).first.first;
3903 *
if (i_group == u_block)
3905 * residual_ad[i] += symm_grad_Nu[i] * ( tau_E + tau_fluid_vol ) * JxW;
3906 * residual_ad[i] -= Nu[i] * overall_body_force * JxW;
3908 *
else if (i_group == p_fluid_block)
3911 * = lqph[q_point]->get_seepage_velocity_current(F_AD, grad_p);
3912 * residual_ad[i] += Np[i] * (det_F_AD - det_F_converged) * JxW;
3913 * residual_ad[i] -= time.get_delta_t() * grad_Np[i]
3914 * * seepage_vel_current * JxW;
3917 *
Assert(i_group <= p_fluid_block, ExcInternalError());
3923 * Assemble the Neumann contribution (external force contribution).
3926 *
for (
unsigned int face = 0; face < GeometryInfo<dim>::faces_per_cell; ++face)
3928 *
if (cell->face(face)->at_boundary() ==
true)
3930 * scratch.fe_face_values_ref.reinit(cell, face);
3932 *
for (
unsigned int f_q_point = 0; f_q_point < n_q_points_f; ++f_q_point)
3935 * = scratch.fe_face_values_ref.normal_vector(f_q_point);
3937 * = scratch.fe_face_values_ref.quadrature_point(f_q_point);
3939 * = get_neumann_traction(cell->face(face)->boundary_id(), pt, N);
3941 * = get_prescribed_fluid_flow(cell->face(face)->boundary_id(), pt);
3943 *
if ( (traction.norm() < 1e-12) && (
std::abs(flow) < 1e-12) )
continue;
3945 *
const double JxW_f = scratch.fe_face_values_ref.JxW(f_q_point);
3947 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
3949 *
const unsigned int i_group = fe.system_to_base_index(i).first.first;
3951 *
if ((i_group == u_block) && (traction.norm() > 1e-12))
3953 *
const unsigned int component_i
3954 * = fe.system_to_component_index(i).first;
3956 * = scratch.fe_face_values_ref.shape_value(i, f_q_point);
3957 * residual_ad[i] -= (Nu_f * traction[component_i]) * JxW_f;
3959 *
if ((i_group == p_fluid_block) && (
std::abs(flow) > 1e-12))
3962 * = scratch.fe_face_values_ref.shape_value(i, f_q_point);
3963 * residual_ad[i] -= (Nu_p * flow) * JxW_f;
3972 * Linearise the residual
3975 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
3977 *
const ADNumberType &R_i = residual_ad[i];
3979 * data.cell_rhs(i) -= R_i.val();
3980 *
for (
unsigned int j=0; j<dofs_per_cell; ++j)
3981 * data.cell_matrix(i,j) += R_i.fastAccessDx(j);
3987 * Store the converged values of the
internal variables
3990 *
template <
int dim>
3991 *
void Solid<dim>::update_end_timestep()
3995 * dof_handler_ref.begin_active()),
3997 * dof_handler_ref.end());
3998 *
for (; cell!=endc; ++cell)
4000 *
Assert(cell->is_locally_owned(), ExcInternalError());
4001 *
Assert(cell->subdomain_id() == this_mpi_process, ExcInternalError());
4003 *
const std::vector<std::shared_ptr<PointHistory<dim, ADNumberType> > >
4004 * lqph = quadrature_point_history.get_data(cell);
4005 *
Assert(lqph.size() == n_q_points, ExcInternalError());
4006 *
for (
unsigned int q_point = 0; q_point < n_q_points; ++q_point)
4007 * lqph[q_point]->update_end_timestep();
4014 * Solve the linearized equations
4017 *
template <
int dim>
4021 * timerconsole.enter_subsection(
"Linear solver");
4022 * timerfile.enter_subsection(
"Linear solver");
4023 * pcout <<
" SLV " << std::flush;
4024 * outfile <<
" SLV " << std::flush;
4027 * newton_update_nb.
reinit(locally_owned_dofs, mpi_communicator);
4030 * 1.0e-6 * system_rhs_nb.l2_norm());
4032 * solver.solve(tangent_matrix_nb, newton_update_nb, system_rhs_nb);
4036 * Copy the non-block solution back to block system
4039 *
for (
unsigned int i=0; i<locally_owned_dofs.n_elements(); ++i)
4042 * = locally_owned_dofs.nth_index_in_set(i);
4043 * newton_update_OUT(idx_i) = newton_update_nb(idx_i);
4047 * timerconsole.leave_subsection();
4048 * timerfile.leave_subsection();
4053 * Class to compute
gradient of the pressure
4056 *
template <
int dim>
4060 * GradientPostprocessor (
const unsigned int p_fluid_component)
4064 * p_fluid_component (p_fluid_component)
4067 *
virtual ~GradientPostprocessor(){}
4070 * evaluate_vector_field
4071 * (
const DataPostprocessorInputs::Vector<dim> &input_data,
4072 * std::vector<Vector<double> > &computed_quantities)
const override
4075 * computed_quantities.size());
4076 *
for (
unsigned int p=0; p<input_data.solution_gradients.size(); ++p)
4079 *
for (
unsigned int d=0;
d<dim; ++
d)
4080 * computed_quantities[p][d]
4081 * = input_data.solution_gradients[p][p_fluid_component][d];
4086 *
const unsigned int p_fluid_component;
4092 * Print results to
vtu file
4095 *
template <
int dim>
void Solid<dim>::output_results_to_vtu
4096 * (
const unsigned int timestep,
4097 *
const double current_time,
4101 * locally_relevant_partitioning,
4104 * solution_total = solution_IN;
4106 *
material_id.reinit(triangulation.n_active_cells());
4107 * std::vector<types::subdomain_id> partition_int(triangulation.n_active_cells());
4108 * GradientPostprocessor<dim> gradient_postprocessor(p_fluid_component);
4112 * Declare local variables with number of stress components
4116 *
unsigned int num_comp_symm_tensor = 6;
4120 * Declare local vectors to store values
4121 * OUTPUT AVERAGED ON ELEMENTS -------------------------------------------
4124 * std::vector<Vector<double>>cauchy_stresses_total_elements
4125 * (num_comp_symm_tensor,
4127 * std::vector<Vector<double>>cauchy_stresses_E_elements
4128 * (num_comp_symm_tensor,
4130 * std::vector<Vector<double>>stretches_elements
4133 * std::vector<Vector<double>>seepage_velocity_elements
4137 * (triangulation.n_active_cells());
4139 * (triangulation.n_active_cells());
4141 * (triangulation.n_active_cells());
4145 * OUTPUT AVERAGED ON NODES ----------------------------------------------
4146 * We need to create a
new FE space with a single dof per node to avoid
4147 * duplication of the output on nodes
for our problem with dim+1 dofs.
4152 * vertex_handler_ref.distribute_dofs(fe_vertex);
4153 *
AssertThrow(vertex_handler_ref.n_dofs() == triangulation.n_vertices(),
4154 * ExcDimensionMismatch(vertex_handler_ref.n_dofs(),
4155 * triangulation.n_vertices()));
4158 * (vertex_handler_ref.n_dofs());
4160 * (vertex_handler_ref.n_dofs());
4162 * std::vector<Vector<double>>cauchy_stresses_total_vertex_mpi
4163 * (num_comp_symm_tensor,
4165 * std::vector<Vector<double>>sum_cauchy_stresses_total_vertex
4166 * (num_comp_symm_tensor,
4168 * std::vector<Vector<double>>cauchy_stresses_E_vertex_mpi
4169 * (num_comp_symm_tensor,
4171 * std::vector<Vector<double>>sum_cauchy_stresses_E_vertex
4172 * (num_comp_symm_tensor,
4174 * std::vector<Vector<double>>stretches_vertex_mpi
4177 * std::vector<Vector<double>>sum_stretches_vertex
4180 *
Vector<double> porous_dissipation_vertex_mpi(vertex_handler_ref.n_dofs());
4181 *
Vector<double> sum_porous_dissipation_vertex(vertex_handler_ref.n_dofs());
4182 *
Vector<double> viscous_dissipation_vertex_mpi(vertex_handler_ref.n_dofs());
4183 *
Vector<double> sum_viscous_dissipation_vertex(vertex_handler_ref.n_dofs());
4184 *
Vector<double> solid_vol_fraction_vertex_mpi(vertex_handler_ref.n_dofs());
4185 *
Vector<double> sum_solid_vol_fraction_vertex(vertex_handler_ref.n_dofs());
4189 * We need to create a
new FE space with a dim dof per node to
4190 * be able to output data on nodes in vector form
4195 * vertex_vec_handler_ref.distribute_dofs(fe_vertex_vec);
4196 *
AssertThrow(vertex_vec_handler_ref.n_dofs() == (dim*triangulation.n_vertices()),
4197 * ExcDimensionMismatch(vertex_vec_handler_ref.n_dofs(),
4198 * (dim*triangulation.n_vertices())));
4200 *
Vector<double> seepage_velocity_vertex_vec_mpi(vertex_vec_handler_ref.n_dofs());
4201 *
Vector<double> sum_seepage_velocity_vertex_vec(vertex_vec_handler_ref.n_dofs());
4202 *
Vector<double> counter_on_vertices_vec_mpi(vertex_vec_handler_ref.n_dofs());
4203 *
Vector<double> sum_counter_on_vertices_vec(vertex_vec_handler_ref.n_dofs());
4206 * -----------------------------------------------------------------------
4210 * Declare and initialize local unit vectors (to construct tensor basis)
4213 * std::vector<Tensor<1,dim>> basis_vectors (dim,
Tensor<1,dim>() );
4214 *
for (
unsigned int i=0; i<dim; ++i)
4215 * basis_vectors[i][i] = 1;
4219 * Declare an instance of the material
class object
4222 *
if (parameters.mat_type ==
"Neo-Hooke")
4223 * NeoHooke<dim,ADNumberType> material(parameters,time);
4224 *
else if (parameters.mat_type ==
"Ogden")
4225 * Ogden<dim,ADNumberType> material(parameters,time);
4226 *
else if (parameters.mat_type ==
"visco-Ogden")
4227 * visco_Ogden <dim,ADNumberType>material(parameters,time);
4229 *
Assert (
false, ExcMessage(
"Material type not implemented"));
4233 * Define a local instance of
FEValues to compute updated values required
4234 * to calculate stresses
4243 * Iterate through elements (cells) and Gauss Points
4248 * dof_handler_ref.begin_active()),
4250 * dof_handler_ref.end()),
4252 * vertex_handler_ref.begin_active()),
4254 * vertex_vec_handler_ref.begin_active());
4260 *
for (; cell!=endc; ++cell, ++cell_v, ++cell_v_vec)
4262 *
Assert(cell->is_locally_owned(), ExcInternalError());
4263 *
Assert(cell->subdomain_id() == this_mpi_process, ExcInternalError());
4266 *
static_cast<int>(cell->material_id());
4268 * fe_values_ref.reinit(cell);
4270 * std::vector<Tensor<2,dim>> solution_grads_u(n_q_points);
4271 * fe_values_ref[u_fe].get_function_gradients(solution_total,
4272 * solution_grads_u);
4274 * std::vector<double> solution_values_p_fluid_total(n_q_points);
4275 * fe_values_ref[p_fluid_fe].get_function_values(solution_total,
4276 * solution_values_p_fluid_total);
4278 * std::vector<Tensor<1,dim>> solution_grads_p_fluid_AD (n_q_points);
4279 * fe_values_ref[p_fluid_fe].get_function_gradients(solution_total,
4280 * solution_grads_p_fluid_AD);
4287 *
for (
unsigned int q_point=0; q_point<n_q_points; ++q_point)
4294 *
const std::vector<std::shared_ptr<const PointHistory<dim,ADNumberType>>>
4295 * lqph = quadrature_point_history.get_data(cell);
4296 *
Assert(lqph.size() == n_q_points, ExcInternalError());
4298 *
const double p_fluid = solution_values_p_fluid_total[q_point];
4309 * lqph[q_point]->get_Cauchy_E(F_AD);
4311 *
for (
unsigned int i=0; i<dim; ++i)
4312 *
for (
unsigned int j=0; j<dim; ++j)
4316 * sigma_fluid_vol *= -p_fluid;
4324 *
const double solid_vol_fraction = (parameters.solid_vol_frac)/det_F;
4328 * Green-Lagrange strain
4340 * solution_grads_p_fluid_AD[q_point]*F_inv;
4342 * lqph[q_point]->get_seepage_velocity_current(F_AD, grad_p_fluid_AD);
4349 *
const double porous_dissipation =
4350 * lqph[q_point]->get_porous_dissipation(F_AD, grad_p_fluid_AD);
4351 *
const double viscous_dissipation =
4352 * lqph[q_point]->get_viscous_dissipation();
4356 * OUTPUT AVERAGED ON ELEMENTS -------------------------------------------
4357 * Both average on elements and on nodes is NOT weighted with the
4359 * integration
point to the average. Ideally, it should be weighted,
4360 * but I haven
't invested time in getting it to work properly.
4363 * if (parameters.outtype == "elements")
4365 * for (unsigned int j=0; j<dim; ++j)
4367 * cauchy_stresses_total_elements[j](cell->active_cell_index())
4368 * += ((sigma*basis_vectors[j])*basis_vectors[j])/n_q_points;
4369 * cauchy_stresses_E_elements[j](cell->active_cell_index())
4370 * += ((sigma_E*basis_vectors[j])*basis_vectors[j])/n_q_points;
4371 * stretches_elements[j](cell->active_cell_index())
4372 * += std::sqrt(1.0+2.0*Tensor<0,dim,double>(E_strain[j][j]))
4374 * seepage_velocity_elements[j](cell->active_cell_index())
4375 * += Tensor<0,dim,double>(seepage_vel_AD[j])/n_q_points;
4378 * porous_dissipation_elements(cell->active_cell_index())
4379 * += porous_dissipation/n_q_points;
4380 * viscous_dissipation_elements(cell->active_cell_index())
4381 * += viscous_dissipation/n_q_points;
4382 * solid_vol_fraction_elements(cell->active_cell_index())
4383 * += solid_vol_fraction/n_q_points;
4385 * cauchy_stresses_total_elements[3](cell->active_cell_index())
4386 * += ((sigma*basis_vectors[0])*basis_vectors[1])/n_q_points; //sig_xy
4387 * cauchy_stresses_total_elements[4](cell->active_cell_index())
4388 * += ((sigma*basis_vectors[0])*basis_vectors[2])/n_q_points;//sig_xz
4389 * cauchy_stresses_total_elements[5](cell->active_cell_index())
4390 * += ((sigma*basis_vectors[1])*basis_vectors[2])/n_q_points;//sig_yz
4392 * cauchy_stresses_E_elements[3](cell->active_cell_index())
4393 * += ((sigma_E*basis_vectors[0])* basis_vectors[1])/n_q_points; //sig_xy
4394 * cauchy_stresses_E_elements[4](cell->active_cell_index())
4395 * += ((sigma_E*basis_vectors[0])* basis_vectors[2])/n_q_points;//sig_xz
4396 * cauchy_stresses_E_elements[5](cell->active_cell_index())
4397 * += ((sigma_E*basis_vectors[1])* basis_vectors[2])/n_q_points;//sig_yz
4402 * OUTPUT AVERAGED ON NODES -------------------------------------------
4405 * else if (parameters.outtype == "nodes")
4407 * for (unsigned int v=0; v<(GeometryInfo<dim>::vertices_per_cell); ++v)
4409 * types::global_dof_index local_vertex_indices =
4410 * cell_v->vertex_dof_index(v, 0);
4411 * counter_on_vertices_mpi(local_vertex_indices) += 1;
4412 * for (unsigned int k=0; k<dim; ++k)
4414 * cauchy_stresses_total_vertex_mpi[k](local_vertex_indices)
4415 * += (sigma*basis_vectors[k])*basis_vectors[k];
4416 * cauchy_stresses_E_vertex_mpi[k](local_vertex_indices)
4417 * += (sigma_E*basis_vectors[k])*basis_vectors[k];
4418 * stretches_vertex_mpi[k](local_vertex_indices)
4419 * += std::sqrt(1.0+2.0*Tensor<0,dim,double>(E_strain[k][k]));
4421 * types::global_dof_index local_vertex_vec_indices =
4422 * cell_v_vec->vertex_dof_index(v, k);
4423 * counter_on_vertices_vec_mpi(local_vertex_vec_indices) += 1;
4424 * seepage_velocity_vertex_vec_mpi(local_vertex_vec_indices)
4425 * += Tensor<0,dim,double>(seepage_vel_AD[k]);
4428 * porous_dissipation_vertex_mpi(local_vertex_indices)
4429 * += porous_dissipation;
4430 * viscous_dissipation_vertex_mpi(local_vertex_indices)
4431 * += viscous_dissipation;
4432 * solid_vol_fraction_vertex_mpi(local_vertex_indices)
4433 * += solid_vol_fraction;
4435 * cauchy_stresses_total_vertex_mpi[3](local_vertex_indices)
4436 * += (sigma*basis_vectors[0])*basis_vectors[1]; //sig_xy
4437 * cauchy_stresses_total_vertex_mpi[4](local_vertex_indices)
4438 * += (sigma*basis_vectors[0])*basis_vectors[2];//sig_xz
4439 * cauchy_stresses_total_vertex_mpi[5](local_vertex_indices)
4440 * += (sigma*basis_vectors[1])*basis_vectors[2]; //sig_yz
4442 * cauchy_stresses_E_vertex_mpi[3](local_vertex_indices)
4443 * += (sigma_E*basis_vectors[0])*basis_vectors[1]; //sig_xy
4444 * cauchy_stresses_E_vertex_mpi[4](local_vertex_indices)
4445 * += (sigma_E*basis_vectors[0])*basis_vectors[2];//sig_xz
4446 * cauchy_stresses_E_vertex_mpi[5](local_vertex_indices)
4447 * += (sigma_E*basis_vectors[1])*basis_vectors[2]; //sig_yz
4452 * ---------------------------------------------------------------
4455 * } //end gauss point loop
4460 * Different nodes might have different amount of contributions, e.g.,
4461 * corner nodes have less integration points contributing to the averaged.
4462 * This is why we need a counter and divide at the end, outside the cell loop.
4465 * if (parameters.outtype == "nodes")
4467 * for (unsigned int d=0; d<(vertex_handler_ref.n_dofs()); ++d)
4469 * sum_counter_on_vertices[d] =
4470 * Utilities::MPI::sum(counter_on_vertices_mpi[d],
4471 * mpi_communicator);
4472 * sum_porous_dissipation_vertex[d] =
4473 * Utilities::MPI::sum(porous_dissipation_vertex_mpi[d],
4474 * mpi_communicator);
4475 * sum_viscous_dissipation_vertex[d] =
4476 * Utilities::MPI::sum(viscous_dissipation_vertex_mpi[d],
4477 * mpi_communicator);
4478 * sum_solid_vol_fraction_vertex[d] =
4479 * Utilities::MPI::sum(solid_vol_fraction_vertex_mpi[d],
4480 * mpi_communicator);
4482 * for (unsigned int k=0; k<num_comp_symm_tensor; ++k)
4484 * sum_cauchy_stresses_total_vertex[k][d] =
4485 * Utilities::MPI::sum(cauchy_stresses_total_vertex_mpi[k][d],
4486 * mpi_communicator);
4487 * sum_cauchy_stresses_E_vertex[k][d] =
4488 * Utilities::MPI::sum(cauchy_stresses_E_vertex_mpi[k][d],
4489 * mpi_communicator);
4491 * for (unsigned int k=0; k<dim; ++k)
4493 * sum_stretches_vertex[k][d] =
4494 * Utilities::MPI::sum(stretches_vertex_mpi[k][d],
4495 * mpi_communicator);
4499 * for (unsigned int d=0; d<(vertex_vec_handler_ref.n_dofs()); ++d)
4501 * sum_counter_on_vertices_vec[d] =
4502 * Utilities::MPI::sum(counter_on_vertices_vec_mpi[d],
4503 * mpi_communicator);
4504 * sum_seepage_velocity_vertex_vec[d] =
4505 * Utilities::MPI::sum(seepage_velocity_vertex_vec_mpi[d],
4506 * mpi_communicator);
4509 * for (unsigned int d=0; d<(vertex_handler_ref.n_dofs()); ++d)
4511 * if (sum_counter_on_vertices[d]>0)
4513 * for (unsigned int i=0; i<num_comp_symm_tensor; ++i)
4515 * sum_cauchy_stresses_total_vertex[i][d] /= sum_counter_on_vertices[d];
4516 * sum_cauchy_stresses_E_vertex[i][d] /= sum_counter_on_vertices[d];
4518 * for (unsigned int i=0; i<dim; ++i)
4520 * sum_stretches_vertex[i][d] /= sum_counter_on_vertices[d];
4522 * sum_porous_dissipation_vertex[d] /= sum_counter_on_vertices[d];
4523 * sum_viscous_dissipation_vertex[d] /= sum_counter_on_vertices[d];
4524 * sum_solid_vol_fraction_vertex[d] /= sum_counter_on_vertices[d];
4528 * for (unsigned int d=0; d<(vertex_vec_handler_ref.n_dofs()); ++d)
4530 * if (sum_counter_on_vertices_vec[d]>0)
4532 * sum_seepage_velocity_vertex_vec[d] /= sum_counter_on_vertices_vec[d];
4540 * Add the results to the solution to create the output file for Paraview
4543 * DataOut<dim> data_out;
4544 * std::vector<DataComponentInterpretation::DataComponentInterpretation>
4546 * DataComponentInterpretation::component_is_part_of_vector);
4547 * comp_type.push_back(DataComponentInterpretation::component_is_scalar);
4549 * GridTools::get_subdomain_association(triangulation, partition_int);
4551 * std::vector<std::string> solution_name(dim, "displacement");
4552 * solution_name.push_back("pore_pressure");
4554 * data_out.attach_dof_handler(dof_handler_ref);
4555 * data_out.add_data_vector(solution_total,
4557 * DataOut<dim>::type_dof_data,
4560 * data_out.add_data_vector(solution_total,
4561 * gradient_postprocessor);
4563 * const Vector<double> partitioning(partition_int.begin(),
4564 * partition_int.end());
4566 * data_out.add_data_vector(partitioning, "partitioning");
4567 * data_out.add_data_vector(material_id, "material_id");
4571 * Integration point results -----------------------------------------------------------
4574 * if (parameters.outtype == "elements")
4576 * data_out.add_data_vector(cauchy_stresses_total_elements[0], "cauchy_xx");
4577 * data_out.add_data_vector(cauchy_stresses_total_elements[1], "cauchy_yy");
4578 * data_out.add_data_vector(cauchy_stresses_total_elements[2], "cauchy_zz");
4579 * data_out.add_data_vector(cauchy_stresses_total_elements[3], "cauchy_xy");
4580 * data_out.add_data_vector(cauchy_stresses_total_elements[4], "cauchy_xz");
4581 * data_out.add_data_vector(cauchy_stresses_total_elements[5], "cauchy_yz");
4583 * data_out.add_data_vector(cauchy_stresses_E_elements[0], "cauchy_E_xx");
4584 * data_out.add_data_vector(cauchy_stresses_E_elements[1], "cauchy_E_yy");
4585 * data_out.add_data_vector(cauchy_stresses_E_elements[2], "cauchy_E_zz");
4586 * data_out.add_data_vector(cauchy_stresses_E_elements[3], "cauchy_E_xy");
4587 * data_out.add_data_vector(cauchy_stresses_E_elements[4], "cauchy_E_xz");
4588 * data_out.add_data_vector(cauchy_stresses_E_elements[5], "cauchy_E_yz");
4590 * data_out.add_data_vector(stretches_elements[0], "stretch_xx");
4591 * data_out.add_data_vector(stretches_elements[1], "stretch_yy");
4592 * data_out.add_data_vector(stretches_elements[2], "stretch_zz");
4594 * data_out.add_data_vector(seepage_velocity_elements[0], "seepage_vel_x");
4595 * data_out.add_data_vector(seepage_velocity_elements[1], "seepage_vel_y");
4596 * data_out.add_data_vector(seepage_velocity_elements[2], "seepage_vel_z");
4598 * data_out.add_data_vector(porous_dissipation_elements, "dissipation_porous");
4599 * data_out.add_data_vector(viscous_dissipation_elements, "dissipation_viscous");
4600 * data_out.add_data_vector(solid_vol_fraction_elements, "solid_vol_fraction");
4602 * else if (parameters.outtype == "nodes")
4604 * data_out.add_data_vector(vertex_handler_ref,
4605 * sum_cauchy_stresses_total_vertex[0],
4607 * data_out.add_data_vector(vertex_handler_ref,
4608 * sum_cauchy_stresses_total_vertex[1],
4610 * data_out.add_data_vector(vertex_handler_ref,
4611 * sum_cauchy_stresses_total_vertex[2],
4613 * data_out.add_data_vector(vertex_handler_ref,
4614 * sum_cauchy_stresses_total_vertex[3],
4616 * data_out.add_data_vector(vertex_handler_ref,
4617 * sum_cauchy_stresses_total_vertex[4],
4619 * data_out.add_data_vector(vertex_handler_ref,
4620 * sum_cauchy_stresses_total_vertex[5],
4623 * data_out.add_data_vector(vertex_handler_ref,
4624 * sum_cauchy_stresses_E_vertex[0],
4626 * data_out.add_data_vector(vertex_handler_ref,
4627 * sum_cauchy_stresses_E_vertex[1],
4629 * data_out.add_data_vector(vertex_handler_ref,
4630 * sum_cauchy_stresses_E_vertex[2],
4632 * data_out.add_data_vector(vertex_handler_ref,
4633 * sum_cauchy_stresses_E_vertex[3],
4635 * data_out.add_data_vector(vertex_handler_ref,
4636 * sum_cauchy_stresses_E_vertex[4],
4638 * data_out.add_data_vector(vertex_handler_ref,
4639 * sum_cauchy_stresses_E_vertex[5],
4642 * data_out.add_data_vector(vertex_handler_ref,
4643 * sum_stretches_vertex[0],
4645 * data_out.add_data_vector(vertex_handler_ref,
4646 * sum_stretches_vertex[1],
4648 * data_out.add_data_vector(vertex_handler_ref,
4649 * sum_stretches_vertex[2],
4652 * std::vector<DataComponentInterpretation::DataComponentInterpretation>
4653 * comp_type_vec(dim,
4654 * DataComponentInterpretation::component_is_part_of_vector);
4655 * std::vector<std::string> solution_name_vec(dim,"seepage_velocity");
4657 * data_out.add_data_vector(vertex_vec_handler_ref,
4658 * sum_seepage_velocity_vertex_vec,
4659 * solution_name_vec,
4662 * data_out.add_data_vector(vertex_handler_ref,
4663 * sum_porous_dissipation_vertex,
4664 * "dissipation_porous");
4665 * data_out.add_data_vector(vertex_handler_ref,
4666 * sum_viscous_dissipation_vertex,
4667 * "dissipation_viscous");
4668 * data_out.add_data_vector(vertex_handler_ref,
4669 * sum_solid_vol_fraction_vertex,
4670 * "solid_vol_fraction");
4674 * ---------------------------------------------------------------------
4680 * data_out.build_patches(degree_displ);
4684 * static std::string get_filename_vtu(unsigned int process,
4685 * unsigned int timestep,
4686 * const unsigned int n_digits = 5)
4688 * std::ostringstream filename_vtu;
4691 * << Utilities::int_to_string(process, n_digits)
4693 * << Utilities::int_to_string(timestep, n_digits)
4695 * return filename_vtu.str();
4698 * static std::string get_filename_pvtu(unsigned int timestep,
4699 * const unsigned int n_digits = 5)
4701 * std::ostringstream filename_vtu;
4704 * << Utilities::int_to_string(timestep, n_digits)
4706 * return filename_vtu.str();
4709 * static std::string get_filename_pvd (void)
4711 * std::ostringstream filename_vtu;
4713 * << "solution.pvd";
4714 * return filename_vtu.str();
4718 * const std::string filename_vtu = Filename::get_filename_vtu(this_mpi_process,
4720 * std::ofstream output(filename_vtu.c_str());
4721 * data_out.write_vtu(output);
4725 * We have a collection of files written in parallel
4726 * This next set of steps should only be performed by master process
4729 * if (this_mpi_process == 0)
4733 * List of all files written out at this timestep by all processors
4736 * std::vector<std::string> parallel_filenames_vtu;
4737 * for (unsigned int p=0; p<n_mpi_processes; ++p)
4739 * parallel_filenames_vtu.push_back(Filename::get_filename_vtu(p, timestep));
4742 * const std::string filename_pvtu(Filename::get_filename_pvtu(timestep));
4743 * std::ofstream pvtu_master(filename_pvtu.c_str());
4744 * data_out.write_pvtu_record(pvtu_master,
4745 * parallel_filenames_vtu);
4749 * Time dependent data master file
4752 * static std::vector<std::pair<double,std::string>> time_and_name_history;
4753 * time_and_name_history.push_back(std::make_pair(current_time,
4755 * const std::string filename_pvd(Filename::get_filename_pvd());
4756 * std::ofstream pvd_output(filename_pvd.c_str());
4757 * DataOutBase::write_pvd_record(pvd_output, time_and_name_history);
4764 * Print results to plotting file
4767 * template <int dim>
4768 * void Solid<dim>::output_results_to_plot(
4769 * const unsigned int timestep,
4770 * const double current_time,
4771 * TrilinosWrappers::MPI::BlockVector solution_IN,
4772 * std::vector<Point<dim> > &tracked_vertices_IN,
4773 * std::ofstream &plotpointfile) const
4775 * TrilinosWrappers::MPI::BlockVector solution_total(locally_owned_partitioning,
4776 * locally_relevant_partitioning,
4781 * solution_total = solution_IN;
4785 * Variables needed to print the solution file for plotting
4788 * Point<dim> reaction_force;
4789 * Point<dim> reaction_force_pressure;
4790 * Point<dim> reaction_force_extra;
4791 * double total_fluid_flow = 0.0;
4792 * double total_porous_dissipation = 0.0;
4793 * double total_viscous_dissipation = 0.0;
4794 * double total_solid_vol = 0.0;
4795 * double total_vol_current = 0.0;
4796 * double total_vol_reference = 0.0;
4797 * std::vector<Point<dim+1>> solution_vertices(tracked_vertices_IN.size());
4801 * Auxiliary variables needed for mpi processing
4804 * Tensor<1,dim> sum_reaction_mpi;
4805 * Tensor<1,dim> sum_reaction_pressure_mpi;
4806 * Tensor<1,dim> sum_reaction_extra_mpi;
4807 * sum_reaction_mpi = 0.0;
4808 * sum_reaction_pressure_mpi = 0.0;
4809 * sum_reaction_extra_mpi = 0.0;
4810 * double sum_total_flow_mpi = 0.0;
4811 * double sum_porous_dissipation_mpi = 0.0;
4812 * double sum_viscous_dissipation_mpi = 0.0;
4813 * double sum_solid_vol_mpi = 0.0;
4814 * double sum_vol_current_mpi = 0.0;
4815 * double sum_vol_reference_mpi = 0.0;
4819 * Declare an instance of the material class object
4822 * if (parameters.mat_type == "Neo-Hooke")
4823 * NeoHooke<dim,ADNumberType> material(parameters,time);
4824 * else if (parameters.mat_type == "Ogden")
4825 * Ogden<dim,ADNumberType> material(parameters, time);
4826 * else if (parameters.mat_type == "visco-Ogden")
4827 * visco_Ogden <dim,ADNumberType>material(parameters,time);
4829 * Assert (false, ExcMessage("Material type not implemented"));
4833 * Define a local instance of FEValues to compute updated values required
4834 * to calculate stresses
4837 * const UpdateFlags uf_cell(update_values | update_gradients |
4838 * update_JxW_values);
4839 * FEValues<dim> fe_values_ref (fe, qf_cell, uf_cell);
4843 * Iterate through elements (cells) and Gauss Points
4846 * FilteredIterator<typename DoFHandler<dim>::active_cell_iterator>
4847 * cell(IteratorFilters::LocallyOwnedCell(),
4848 * dof_handler_ref.begin_active()),
4849 * endc(IteratorFilters::LocallyOwnedCell(),
4850 * dof_handler_ref.end());
4856 * for (; cell!=endc; ++cell)
4858 * Assert(cell->is_locally_owned(), ExcInternalError());
4859 * Assert(cell->subdomain_id() == this_mpi_process, ExcInternalError());
4861 * fe_values_ref.reinit(cell);
4863 * std::vector<Tensor<2,dim>> solution_grads_u(n_q_points);
4864 * fe_values_ref[u_fe].get_function_gradients(solution_total,
4865 * solution_grads_u);
4867 * std::vector<double> solution_values_p_fluid_total(n_q_points);
4868 * fe_values_ref[p_fluid_fe].get_function_values(solution_total,
4869 * solution_values_p_fluid_total);
4871 * std::vector<Tensor<1,dim >> solution_grads_p_fluid_AD(n_q_points);
4872 * fe_values_ref[p_fluid_fe].get_function_gradients(solution_total,
4873 * solution_grads_p_fluid_AD);
4877 * start gauss point loop
4880 * for (unsigned int q_point=0; q_point<n_q_points; ++q_point)
4882 * const Tensor<2,dim,ADNumberType>
4883 * F_AD = Physics::Elasticity::Kinematics::F(solution_grads_u[q_point]);
4884 * ADNumberType det_F_AD = determinant(F_AD);
4885 * const double det_F = Tensor<0,dim,double>(det_F_AD);
4887 * const std::vector<std::shared_ptr<const PointHistory<dim,ADNumberType>>>
4888 * lqph = quadrature_point_history.get_data(cell);
4889 * Assert(lqph.size() == n_q_points, ExcInternalError());
4891 * double JxW = fe_values_ref.JxW(q_point);
4898 * sum_vol_current_mpi += det_F * JxW;
4899 * sum_vol_reference_mpi += JxW;
4900 * sum_solid_vol_mpi += parameters.solid_vol_frac * JxW * det_F;
4907 * const Tensor<2,dim,ADNumberType> F_inv = invert(F_AD);
4908 * const Tensor<1,dim,ADNumberType>
4909 * grad_p_fluid_AD = solution_grads_p_fluid_AD[q_point]*F_inv;
4910 * const Tensor<1,dim,ADNumberType> seepage_vel_AD
4911 * = lqph[q_point]->get_seepage_velocity_current(F_AD, grad_p_fluid_AD);
4918 * const double porous_dissipation =
4919 * lqph[q_point]->get_porous_dissipation(F_AD, grad_p_fluid_AD);
4920 * sum_porous_dissipation_mpi += porous_dissipation * det_F * JxW;
4922 * const double viscous_dissipation = lqph[q_point]->get_viscous_dissipation();
4923 * sum_viscous_dissipation_mpi += viscous_dissipation * det_F * JxW;
4927 * ---------------------------------------------------------------
4930 * } //end gauss point loop
4934 * Compute reaction force on load boundary & total fluid flow across
4936 * Define a local instance of FEFaceValues to compute values required
4937 * to calculate reaction force
4940 * const UpdateFlags uf_face( update_values | update_gradients |
4941 * update_normal_vectors | update_JxW_values );
4942 * FEFaceValues<dim> fe_face_values_ref(fe, qf_face, uf_face);
4949 * for (unsigned int face=0; face<GeometryInfo<dim>::faces_per_cell; ++face)
4956 * if (cell->face(face)->at_boundary() == true &&
4957 * cell->face(face)->boundary_id() == get_reaction_boundary_id_for_output() )
4959 * fe_face_values_ref.reinit(cell, face);
4963 * Get displacement gradients for current face
4966 * std::vector<Tensor<2,dim> > solution_grads_u_f(n_q_points_f);
4967 * fe_face_values_ref[u_fe].get_function_gradients
4969 * solution_grads_u_f);
4973 * Get pressure for current element
4976 * std::vector< double > solution_values_p_fluid_total_f(n_q_points_f);
4977 * fe_face_values_ref[p_fluid_fe].get_function_values
4979 * solution_values_p_fluid_total_f);
4983 * start gauss points on faces loop
4986 * for (unsigned int f_q_point=0; f_q_point<n_q_points_f; ++f_q_point)
4988 * const Tensor<1,dim> &N = fe_face_values_ref.normal_vector(f_q_point);
4989 * const double JxW_f = fe_face_values_ref.JxW(f_q_point);
4993 * Compute deformation gradient from displacements gradient
4994 * (present configuration)
4997 * const Tensor<2,dim,ADNumberType> F_AD =
4998 * Physics::Elasticity::Kinematics::F(solution_grads_u_f[f_q_point]);
5000 * const std::vector<std::shared_ptr<const PointHistory<dim,ADNumberType>>>
5001 * lqph = quadrature_point_history.get_data(cell);
5002 * Assert(lqph.size() == n_q_points, ExcInternalError());
5004 * const double p_fluid = solution_values_p_fluid_total[f_q_point];
5011 * static const SymmetricTensor<2,dim,double>
5012 * I (Physics::Elasticity::StandardTensors<dim>::I);
5013 * SymmetricTensor<2,dim> sigma_E;
5014 * const SymmetricTensor<2,dim,ADNumberType> sigma_E_AD =
5015 * lqph[f_q_point]->get_Cauchy_E(F_AD);
5017 * for (unsigned int i=0; i<dim; ++i)
5018 * for (unsigned int j=0; j<dim; ++j)
5019 * sigma_E[i][j] = Tensor<0,dim,double>(sigma_E_AD[i][j]);
5021 * SymmetricTensor<2,dim> sigma_fluid_vol(I);
5022 * sigma_fluid_vol *= -1.0*p_fluid;
5023 * const SymmetricTensor<2,dim> sigma = sigma_E+sigma_fluid_vol;
5024 * sum_reaction_mpi += sigma * N * JxW_f;
5025 * sum_reaction_pressure_mpi += sigma_fluid_vol * N * JxW_f;
5026 * sum_reaction_extra_mpi += sigma_E * N * JxW_f;
5027 * }//end gauss points on faces loop
5035 * if (cell->face(face)->at_boundary() == true &&
5036 * (cell->face(face)->boundary_id() ==
5037 * get_drained_boundary_id_for_output().first ||
5038 * cell->face(face)->boundary_id() ==
5039 * get_drained_boundary_id_for_output().second ) )
5041 * fe_face_values_ref.reinit(cell, face);
5045 * Get displacement gradients for current face
5048 * std::vector<Tensor<2,dim>> solution_grads_u_f(n_q_points_f);
5049 * fe_face_values_ref[u_fe].get_function_gradients
5051 * solution_grads_u_f);
5055 * Get pressure gradients for current face
5058 * std::vector<Tensor<1,dim>> solution_grads_p_f(n_q_points_f);
5059 * fe_face_values_ref[p_fluid_fe].get_function_gradients
5061 * solution_grads_p_f);
5065 * start gauss points on faces loop
5068 * for (unsigned int f_q_point=0; f_q_point<n_q_points_f; ++f_q_point)
5070 * const Tensor<1,dim> &N =
5071 * fe_face_values_ref.normal_vector(f_q_point);
5072 * const double JxW_f = fe_face_values_ref.JxW(f_q_point);
5076 * Deformation gradient and inverse from displacements gradient
5077 * (present configuration)
5080 * const Tensor<2,dim,ADNumberType> F_AD
5081 * = Physics::Elasticity::Kinematics::F(solution_grads_u_f[f_q_point]);
5083 * const Tensor<2,dim,ADNumberType> F_inv_AD = invert(F_AD);
5084 * ADNumberType det_F_AD = determinant(F_AD);
5086 * const std::vector<std::shared_ptr<const PointHistory<dim,ADNumberType>>>
5087 * lqph = quadrature_point_history.get_data(cell);
5088 * Assert(lqph.size() == n_q_points, ExcInternalError());
5095 * Tensor<1,dim> seepage;
5096 * double det_F = Tensor<0,dim,double>(det_F_AD);
5097 * const Tensor<1,dim,ADNumberType> grad_p
5098 * = solution_grads_p_f[f_q_point]*F_inv_AD;
5099 * const Tensor<1,dim,ADNumberType> seepage_AD
5100 * = lqph[f_q_point]->get_seepage_velocity_current(F_AD, grad_p);
5102 * for (unsigned int i=0; i<dim; ++i)
5103 * seepage[i] = Tensor<0,dim,double>(seepage_AD[i]);
5105 * sum_total_flow_mpi += (seepage/det_F) * N * JxW_f;
5106 * }//end gauss points on faces loop
5113 * Sum the results from different MPI process and then add to the reaction_force vector
5114 * In theory, the solution on each surface (each cell) only exists in one MPI process
5115 * so, we add all MPI process, one will have the solution and the others will be zero
5118 * for (unsigned int d=0; d<dim; ++d)
5120 * reaction_force[d] = Utilities::MPI::sum(sum_reaction_mpi[d],
5121 * mpi_communicator);
5122 * reaction_force_pressure[d] = Utilities::MPI::sum(sum_reaction_pressure_mpi[d],
5123 * mpi_communicator);
5124 * reaction_force_extra[d] = Utilities::MPI::sum(sum_reaction_extra_mpi[d],
5125 * mpi_communicator);
5130 * Same for total fluid flow, and for porous and viscous dissipations
5133 * total_fluid_flow = Utilities::MPI::sum(sum_total_flow_mpi,
5134 * mpi_communicator);
5135 * total_porous_dissipation = Utilities::MPI::sum(sum_porous_dissipation_mpi,
5136 * mpi_communicator);
5137 * total_viscous_dissipation = Utilities::MPI::sum(sum_viscous_dissipation_mpi,
5138 * mpi_communicator);
5139 * total_solid_vol = Utilities::MPI::sum(sum_solid_vol_mpi,
5140 * mpi_communicator);
5141 * total_vol_current = Utilities::MPI::sum(sum_vol_current_mpi,
5142 * mpi_communicator);
5143 * total_vol_reference = Utilities::MPI::sum(sum_vol_reference_mpi,
5144 * mpi_communicator);
5148 * Extract solution for tracked vectors
5149 * Copying an MPI::BlockVector into MPI::Vector is not possible,
5150 * so we copy each block of MPI::BlockVector into an MPI::Vector
5151 * And then we copy the MPI::Vector into "normal" Vectors
5154 * TrilinosWrappers::MPI::Vector solution_vector_u_MPI(solution_total.block(u_block));
5155 * TrilinosWrappers::MPI::Vector solution_vector_p_MPI(solution_total.block(p_fluid_block));
5156 * Vector<double> solution_u_vector(solution_vector_u_MPI);
5157 * Vector<double> solution_p_vector(solution_vector_p_MPI);
5159 * if (this_mpi_process == 0)
5163 * Append the pressure solution vector to the displacement solution vector,
5164 * creating a single solution vector equivalent to the original BlockVector
5165 * so FEFieldFunction will work with the dof_handler_ref.
5168 * Vector<double> solution_vector(solution_p_vector.size()
5169 * +solution_u_vector.size());
5171 * for (unsigned int d=0; d<(solution_u_vector.size()); ++d)
5172 * solution_vector[d] = solution_u_vector[d];
5174 * for (unsigned int d=0; d<(solution_p_vector.size()); ++d)
5175 * solution_vector[solution_u_vector.size()+d] = solution_p_vector[d];
5177 * Functions::FEFieldFunction<dim,Vector<double>>
5178 * find_solution(dof_handler_ref, solution_vector);
5180 * for (unsigned int p=0; p<tracked_vertices_IN.size(); ++p)
5182 * Vector<double> update(dim+1);
5183 * Point<dim> pt_ref;
5185 * pt_ref[0]= tracked_vertices_IN[p][0];
5186 * pt_ref[1]= tracked_vertices_IN[p][1];
5187 * pt_ref[2]= tracked_vertices_IN[p][2];
5189 * find_solution.vector_value(pt_ref, update);
5191 * for (unsigned int d=0; d<(dim+1); ++d)
5195 * For values close to zero, set to 0.0
5198 * if (abs(update[d])<1.5*parameters.tol_u)
5200 * solution_vertices[p][d] = update[d];
5205 * Write the results to the plotting file.
5206 * Add two blank lines between cycles in the cyclic loading examples so GNUPLOT can detect each cycle as a different block
5209 * if (( (parameters.geom_type == "Budday_cube_tension_compression_fully_fixed")||
5210 * (parameters.geom_type == "Budday_cube_tension_compression")||
5211 * (parameters.geom_type == "Budday_cube_shear_fully_fixed") ) &&
5212 * ( (abs(current_time - parameters.end_time/3.) <0.9*parameters.delta_t)||
5213 * (abs(current_time - 2.*parameters.end_time/3.)<0.9*parameters.delta_t) ) &&
5214 * parameters.num_cycle_sets == 1 )
5216 * plotpointfile << std::endl<< std::endl;
5218 * if (( (parameters.geom_type == "Budday_cube_tension_compression_fully_fixed")||
5219 * (parameters.geom_type == "Budday_cube_tension_compression")||
5220 * (parameters.geom_type == "Budday_cube_shear_fully_fixed") ) &&
5221 * ( (abs(current_time - parameters.end_time/9.) <0.9*parameters.delta_t)||
5222 * (abs(current_time - 2.*parameters.end_time/9.)<0.9*parameters.delta_t)||
5223 * (abs(current_time - 3.*parameters.end_time/9.)<0.9*parameters.delta_t)||
5224 * (abs(current_time - 5.*parameters.end_time/9.)<0.9*parameters.delta_t)||
5225 * (abs(current_time - 7.*parameters.end_time/9.)<0.9*parameters.delta_t) ) &&
5226 * parameters.num_cycle_sets == 2 )
5228 * plotpointfile << std::endl<< std::endl;
5231 * plotpointfile << std::setprecision(6) << std::scientific;
5232 * plotpointfile << std::setw(16) << current_time << ","
5233 * << std::setw(15) << total_vol_reference << ","
5234 * << std::setw(15) << total_vol_current << ","
5235 * << std::setw(15) << total_solid_vol << ",";
5237 * if (current_time == 0.0)
5239 * for (unsigned int p=0; p<tracked_vertices_IN.size(); ++p)
5241 * for (unsigned int d=0; d<dim; ++d)
5242 * plotpointfile << std::setw(15) << 0.0 << ",";
5244 * plotpointfile << std::setw(15) << parameters.drained_pressure << ",";
5246 * for (unsigned int d=0; d<(3*dim+2); ++d)
5247 * plotpointfile << std::setw(15) << 0.0 << ",";
5249 * plotpointfile << std::setw(15) << 0.0;
5253 * for (unsigned int p=0; p<tracked_vertices_IN.size(); ++p)
5254 * for (unsigned int d=0; d<(dim+1); ++d)
5255 * plotpointfile << std::setw(15) << solution_vertices[p][d]<< ",";
5257 * for (unsigned int d=0; d<dim; ++d)
5258 * plotpointfile << std::setw(15) << reaction_force[d] << ",";
5260 * for (unsigned int d=0; d<dim; ++d)
5261 * plotpointfile << std::setw(15) << reaction_force_pressure[d] << ",";
5263 * for (unsigned int d=0; d<dim; ++d)
5264 * plotpointfile << std::setw(15) << reaction_force_extra[d] << ",";
5266 * plotpointfile << std::setw(15) << total_fluid_flow << ","
5267 * << std::setw(15) << total_porous_dissipation<< ","
5268 * << std::setw(15) << total_viscous_dissipation;
5270 * plotpointfile << std::endl;
5276 * Header for console output file
5279 * template <int dim>
5280 * void Solid<dim>::print_console_file_header(std::ofstream &outputfile) const
5282 * outputfile << "/*-----------------------------------------------------------------------------------------";
5283 * outputfile << "\n\n Poro-viscoelastic formulation to solve nonlinear solid mechanics problems using deal.ii";
5284 * outputfile << "\n\n Problem setup by E Comellas and J-P Pelteret, University of Erlangen-Nuremberg, 2018";
5285 * outputfile << "\n\n/*-----------------------------------------------------------------------------------------";
5286 * outputfile << "\n\nCONSOLE OUTPUT: \n\n";
5291 * Header for plotting output file
5294 * template <int dim>
5295 * void Solid<dim>::print_plot_file_header(std::vector<Point<dim> > &tracked_vertices,
5296 * std::ofstream &plotpointfile) const
5298 * plotpointfile << "#\n# *** Solution history for tracked vertices -- DOF: 0 = Ux, 1 = Uy, 2 = Uz, 3 = P ***"
5301 * for (unsigned int p=0; p<tracked_vertices.size(); ++p)
5303 * plotpointfile << "# Point " << p << " coordinates: ";
5304 * for (unsigned int d=0; d<dim; ++d)
5306 * plotpointfile << tracked_vertices[p][d];
5307 * if (!( (p == tracked_vertices.size()-1) && (d == dim-1) ))
5308 * plotpointfile << ", ";
5310 * plotpointfile << std::endl;
5312 * plotpointfile << "# The reaction force is the integral over the loaded surfaces in the "
5313 * << "undeformed configuration of the Cauchy stress times the normal surface unit vector.\n"
5314 * << "# reac(p) corresponds to the volumetric part of the Cauchy stress due to the pore fluid pressure"
5315 * << " and reac(E) corresponds to the extra part of the Cauchy stress due to the solid contribution."
5317 * << "# The fluid flow is the integral over the drained surfaces in the "
5318 * << "undeformed configuration of the seepage velocity times the normal surface unit vector."
5320 * << "# Column number:"
5324 * unsigned int columns = 24;
5325 * for (unsigned int d=1; d<columns; ++d)
5326 * plotpointfile << std::setw(15)<< d <<",";
5328 * plotpointfile << std::setw(15)<< columns
5331 * << std::right << std::setw(16) << "Time,"
5332 * << std::right << std::setw(16) << "ref vol,"
5333 * << std::right << std::setw(16) << "def vol,"
5334 * << std::right << std::setw(16) << "solid vol,";
5335 * for (unsigned int p=0; p<tracked_vertices.size(); ++p)
5336 * for (unsigned int d=0; d<(dim+1); ++d)
5337 * plotpointfile << std::right<< std::setw(11)
5338 * <<"P" << p << "[" << d << "],";
5340 * for (unsigned int d=0; d<dim; ++d)
5341 * plotpointfile << std::right<< std::setw(13)
5342 * << "reaction [" << d << "],";
5344 * for (unsigned int d=0; d<dim; ++d)
5345 * plotpointfile << std::right<< std::setw(13)
5346 * << "reac(p) [" << d << "],";
5348 * for (unsigned int d=0; d<dim; ++d)
5349 * plotpointfile << std::right<< std::setw(13)
5350 * << "reac(E) [" << d << "],";
5352 * plotpointfile << std::right<< std::setw(16)<< "fluid flow,"
5353 * << std::right<< std::setw(16)<< "porous dissip,"
5354 * << std::right<< std::setw(15)<< "viscous dissip"
5360 * Footer for console output file
5363 * template <int dim>
5364 * void Solid<dim>::print_console_file_footer(std::ofstream &outputfile) const
5368 * Copy "parameters" file at end of output file.
5371 * std::ifstream infile("parameters.prm");
5372 * std::string content = "";
5375 * for(i=0 ; infile.eof()!=true ; i++)
5377 * char aux = infile.get();
5379 * if(aux=='\n
') content += '#
';
5383 * content.erase(content.end()-1);
5386 * outputfile << "\n\n\n\n PARAMETERS FILE USED IN THIS COMPUTATION: \n#"
5393 * Footer for plotting output file
5396 * template <int dim>
5397 * void Solid<dim>::print_plot_file_footer(std::ofstream &plotpointfile) const
5401 * Copy "parameters" file at end of output file.
5404 * std::ifstream infile("parameters.prm");
5405 * std::string content = "";
5408 * for(i=0 ; infile.eof()!=true ; i++)
5410 * char aux = infile.get();
5412 * if(aux=='\n
') content += '#
';
5416 * content.erase(content.end()-1);
5419 * plotpointfile << "#"<< std::endl
5420 * << "#"<< std::endl
5421 * << "# PARAMETERS FILE USED IN THIS COMPUTATION:" << std::endl
5422 * << "#"<< std::endl
5430 * <a name="nonlinear-poro-viscoelasticity.cc-VerificationexamplesfromEhlersandEipper1999"></a>
5431 * <h3>Verification examples from Ehlers and Eipper 1999</h3>
5432 * We group the definition of the geometry, boundary and loading conditions specific to
5433 * the verification examples from Ehlers and Eipper 1999 into specific classes.
5438 * <a name="nonlinear-poro-viscoelasticity.cc-BaseclassTubegeometryandboundaryconditions"></a>
5439 * <h4>Base class: Tube geometry and boundary conditions</h4>
5442 * template <int dim>
5443 * class VerificationEhlers1999TubeBase
5444 * : public Solid<dim>
5447 * VerificationEhlers1999TubeBase (const Parameters::AllParameters ¶meters)
5448 * : Solid<dim> (parameters)
5451 * virtual ~VerificationEhlers1999TubeBase () {}
5454 * virtual void make_grid() override
5456 * GridGenerator::cylinder( this->triangulation,
5460 * const double rot_angle = 3.0*numbers::PI/2.0;
5461 * GridTools::rotate( Point<3>::unit_vector(1), rot_angle, this->triangulation);
5463 * this->triangulation.reset_manifold(0);
5464 * static const CylindricalManifold<dim> manifold_description_3d(2);
5465 * this->triangulation.set_manifold (0, manifold_description_3d);
5466 * GridTools::scale(this->parameters.scale, this->triangulation);
5467 * this->triangulation.refine_global(std::max (1U, this->parameters.global_refinement));
5468 * this->triangulation.reset_manifold(0);
5471 * virtual void define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
5473 * tracked_vertices[0][0] = 0.0*this->parameters.scale;
5474 * tracked_vertices[0][1] = 0.0*this->parameters.scale;
5475 * tracked_vertices[0][2] = 0.5*this->parameters.scale;
5477 * tracked_vertices[1][0] = 0.0*this->parameters.scale;
5478 * tracked_vertices[1][1] = 0.0*this->parameters.scale;
5479 * tracked_vertices[1][2] = -0.5*this->parameters.scale;
5482 * virtual void make_dirichlet_constraints(AffineConstraints<double> &constraints) override
5484 * if (this->time.get_timestep() < 2)
5486 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5488 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
5490 * (this->fe.component_mask(this->pressure)));
5494 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5496 * Functions::ZeroFunction<dim>(this->n_components),
5498 * (this->fe.component_mask(this->pressure)));
5501 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5503 * Functions::ZeroFunction<dim>(this->n_components),
5505 * (this->fe.component_mask(this->x_displacement)|
5506 * this->fe.component_mask(this->y_displacement) ) );
5508 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5510 * Functions::ZeroFunction<dim>(this->n_components),
5512 * (this->fe.component_mask(this->x_displacement) |
5513 * this->fe.component_mask(this->y_displacement) |
5514 * this->fe.component_mask(this->z_displacement) ));
5518 * get_prescribed_fluid_flow (const types::boundary_id &boundary_id,
5519 * const Point<dim> &pt) const override
5522 * (void)boundary_id;
5526 * virtual types::boundary_id
5527 * get_reaction_boundary_id_for_output() const override
5532 * virtual std::pair<types::boundary_id,types::boundary_id>
5533 * get_drained_boundary_id_for_output() const override
5535 * return std::make_pair(2,2);
5538 * virtual std::vector<double>
5539 * get_dirichlet_load(const types::boundary_id &boundary_id,
5540 * const int &direction) const override
5542 * std::vector<double> displ_incr(dim, 0.0);
5543 * (void)boundary_id;
5545 * AssertThrow(false, ExcMessage("Displacement loading not implemented for Ehlers verification examples."));
5547 * return displ_incr;
5554 * <a name="nonlinear-poro-viscoelasticity.cc-DerivedclassSteploadexample"></a>
5555 * <h4>Derived class: Step load example</h4>
5558 * template <int dim>
5559 * class VerificationEhlers1999StepLoad
5560 * : public VerificationEhlers1999TubeBase<dim>
5563 * VerificationEhlers1999StepLoad (const Parameters::AllParameters ¶meters)
5564 * : VerificationEhlers1999TubeBase<dim> (parameters)
5567 * virtual ~VerificationEhlers1999StepLoad () {}
5570 * virtual Tensor<1,dim>
5571 * get_neumann_traction (const types::boundary_id &boundary_id,
5572 * const Point<dim> &pt,
5573 * const Tensor<1,dim> &N) const override
5575 * if (this->parameters.load_type == "pressure")
5577 * if (boundary_id == 2)
5579 * return this->parameters.load * N;
5585 * return Tensor<1,dim>();
5592 * <a name="nonlinear-poro-viscoelasticity.cc-DerivedclassLoadincreasingexample"></a>
5593 * <h4>Derived class: Load increasing example</h4>
5596 * template <int dim>
5597 * class VerificationEhlers1999IncreaseLoad
5598 * : public VerificationEhlers1999TubeBase<dim>
5601 * VerificationEhlers1999IncreaseLoad (const Parameters::AllParameters ¶meters)
5602 * : VerificationEhlers1999TubeBase<dim> (parameters)
5605 * virtual ~VerificationEhlers1999IncreaseLoad () {}
5608 * virtual Tensor<1,dim>
5609 * get_neumann_traction (const types::boundary_id &boundary_id,
5610 * const Point<dim> &pt,
5611 * const Tensor<1,dim> &N) const override
5613 * if (this->parameters.load_type == "pressure")
5615 * if (boundary_id == 2)
5617 * const double initial_load = this->parameters.load;
5618 * const double final_load = 20.0*initial_load;
5619 * const double initial_time = this->time.get_delta_t();
5620 * const double final_time = this->time.get_end();
5621 * const double current_time = this->time.get_current();
5622 * const double load = initial_load + (final_load-initial_load)*(current_time-initial_time)/(final_time-initial_time);
5629 * return Tensor<1,dim>();
5636 * <a name="nonlinear-poro-viscoelasticity.cc-ClassConsolidationcube"></a>
5637 * <h4>Class: Consolidation cube</h4>
5640 * template <int dim>
5641 * class VerificationEhlers1999CubeConsolidation
5642 * : public Solid<dim>
5645 * VerificationEhlers1999CubeConsolidation (const Parameters::AllParameters ¶meters)
5646 * : Solid<dim> (parameters)
5649 * virtual ~VerificationEhlers1999CubeConsolidation () {}
5653 * make_grid() override
5655 * GridGenerator::hyper_rectangle(this->triangulation,
5656 * Point<dim>(0.0, 0.0, 0.0),
5657 * Point<dim>(1.0, 1.0, 1.0),
5660 * GridTools::scale(this->parameters.scale, this->triangulation);
5661 * this->triangulation.refine_global(std::max (1U, this->parameters.global_refinement));
5663 * typename Triangulation<dim>::active_cell_iterator cell =
5664 * this->triangulation.begin_active(), endc = this->triangulation.end();
5665 * for (; cell != endc; ++cell)
5667 * for (unsigned int face = 0; face < GeometryInfo<dim>::faces_per_cell; ++face)
5668 * if (cell->face(face)->at_boundary() == true &&
5669 * cell->face(face)->center()[2] == 1.0 * this->parameters.scale)
5671 * if (cell->face(face)->center()[0] < 0.5 * this->parameters.scale &&
5672 * cell->face(face)->center()[1] < 0.5 * this->parameters.scale)
5673 * cell->face(face)->set_boundary_id(100);
5675 * cell->face(face)->set_boundary_id(101);
5681 * define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
5683 * tracked_vertices[0][0] = 0.0*this->parameters.scale;
5684 * tracked_vertices[0][1] = 0.0*this->parameters.scale;
5685 * tracked_vertices[0][2] = 1.0*this->parameters.scale;
5687 * tracked_vertices[1][0] = 0.0*this->parameters.scale;
5688 * tracked_vertices[1][1] = 0.0*this->parameters.scale;
5689 * tracked_vertices[1][2] = 0.0*this->parameters.scale;
5693 * make_dirichlet_constraints(AffineConstraints<double> &constraints) override
5695 * if (this->time.get_timestep() < 2)
5697 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5699 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
5701 * (this->fe.component_mask(this->pressure)));
5705 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5707 * Functions::ZeroFunction<dim>(this->n_components),
5709 * (this->fe.component_mask(this->pressure)));
5712 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5714 * Functions::ZeroFunction<dim>(this->n_components),
5716 * this->fe.component_mask(this->x_displacement));
5718 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5720 * Functions::ZeroFunction<dim>(this->n_components),
5722 * this->fe.component_mask(this->x_displacement));
5724 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5726 * Functions::ZeroFunction<dim>(this->n_components),
5728 * this->fe.component_mask(this->y_displacement));
5730 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5732 * Functions::ZeroFunction<dim>(this->n_components),
5734 * this->fe.component_mask(this->y_displacement));
5736 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5738 * Functions::ZeroFunction<dim>(this->n_components),
5740 * ( this->fe.component_mask(this->x_displacement) |
5741 * this->fe.component_mask(this->y_displacement) |
5742 * this->fe.component_mask(this->z_displacement) ));
5745 * virtual Tensor<1,dim>
5746 * get_neumann_traction (const types::boundary_id &boundary_id,
5747 * const Point<dim> &pt,
5748 * const Tensor<1,dim> &N) const override
5750 * if (this->parameters.load_type == "pressure")
5752 * if (boundary_id == 100)
5754 * return this->parameters.load * N;
5760 * return Tensor<1,dim>();
5764 * get_prescribed_fluid_flow (const types::boundary_id &boundary_id,
5765 * const Point<dim> &pt) const override
5768 * (void)boundary_id;
5772 * virtual types::boundary_id
5773 * get_reaction_boundary_id_for_output() const override
5778 * virtual std::pair<types::boundary_id,types::boundary_id>
5779 * get_drained_boundary_id_for_output() const override
5781 * return std::make_pair(101,101);
5784 * virtual std::vector<double>
5785 * get_dirichlet_load(const types::boundary_id &boundary_id,
5786 * const int &direction) const override
5788 * std::vector<double> displ_incr(dim, 0.0);
5789 * (void)boundary_id;
5791 * AssertThrow(false, ExcMessage("Displacement loading not implemented for Ehlers verification examples."));
5793 * return displ_incr;
5800 * <a name="nonlinear-poro-viscoelasticity.cc-Franceschiniexperiments"></a>
5801 * <h4>Franceschini experiments</h4>
5804 * template <int dim>
5805 * class Franceschini2006Consolidation
5806 * : public Solid<dim>
5809 * Franceschini2006Consolidation (const Parameters::AllParameters ¶meters)
5810 * : Solid<dim> (parameters)
5813 * virtual ~Franceschini2006Consolidation () {}
5816 * virtual void make_grid() override
5818 * const Point<dim-1> mesh_center(0.0, 0.0);
5819 * const double radius = 0.5;
5822 * const double height = 0.27; //8.1 mm for 30 mm radius
5825 * const double height = 0.23; //6.9 mm for 30 mm radius
5826 * Triangulation<dim-1> triangulation_in;
5827 * GridGenerator::hyper_ball( triangulation_in,
5831 * GridGenerator::extrude_triangulation(triangulation_in,
5834 * this->triangulation);
5836 * const CylindricalManifold<dim> cylinder_3d(2);
5837 * const types::manifold_id cylinder_id = 0;
5840 * this->triangulation.set_manifold(cylinder_id, cylinder_3d);
5842 * for (auto cell : this->triangulation.active_cell_iterators())
5844 * for (unsigned int face = 0; face < GeometryInfo<dim>::faces_per_cell; ++face)
5846 * if (cell->face(face)->at_boundary() == true)
5848 * if (cell->face(face)->center()[2] == 0.0)
5849 * cell->face(face)->set_boundary_id(1);
5851 * else if (cell->face(face)->center()[2] == height)
5852 * cell->face(face)->set_boundary_id(2);
5856 * cell->face(face)->set_boundary_id(0);
5857 * cell->face(face)->set_all_manifold_ids(cylinder_id);
5863 * GridTools::scale(this->parameters.scale, this->triangulation);
5864 * this->triangulation.refine_global(std::max (1U, this->parameters.global_refinement));
5867 * virtual void define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
5869 * tracked_vertices[0][0] = 0.0*this->parameters.scale;
5870 * tracked_vertices[0][1] = 0.0*this->parameters.scale;
5873 * tracked_vertices[0][2] = 0.27*this->parameters.scale;
5876 * tracked_vertices[0][2] = 0.23*this->parameters.scale;
5878 * tracked_vertices[1][0] = 0.0*this->parameters.scale;
5879 * tracked_vertices[1][1] = 0.0*this->parameters.scale;
5880 * tracked_vertices[1][2] = 0.0*this->parameters.scale;
5883 * virtual void make_dirichlet_constraints(AffineConstraints<double> &constraints) override
5885 * if (this->time.get_timestep() < 2)
5887 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5889 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
5891 * (this->fe.component_mask(this->pressure)));
5893 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5895 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
5897 * (this->fe.component_mask(this->pressure)));
5901 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5903 * Functions::ZeroFunction<dim>(this->n_components),
5905 * (this->fe.component_mask(this->pressure)));
5907 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
5909 * Functions::ZeroFunction<dim>(this->n_components),
5911 * (this->fe.component_mask(this->pressure)));
5914 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5916 * Functions::ZeroFunction<dim>(this->n_components),
5918 * (this->fe.component_mask(this->x_displacement)|
5919 * this->fe.component_mask(this->y_displacement) ) );
5921 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5923 * Functions::ZeroFunction<dim>(this->n_components),
5925 * (this->fe.component_mask(this->x_displacement) |
5926 * this->fe.component_mask(this->y_displacement) |
5927 * this->fe.component_mask(this->z_displacement) ));
5929 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
5931 * Functions::ZeroFunction<dim>(this->n_components),
5933 * (this->fe.component_mask(this->x_displacement) |
5934 * this->fe.component_mask(this->y_displacement) ));
5938 * get_prescribed_fluid_flow (const types::boundary_id &boundary_id,
5939 * const Point<dim> &pt) const override
5942 * (void)boundary_id;
5946 * virtual types::boundary_id
5947 * get_reaction_boundary_id_for_output() const override
5952 * virtual std::pair<types::boundary_id,types::boundary_id>
5953 * get_drained_boundary_id_for_output() const override
5955 * return std::make_pair(1,2);
5958 * virtual std::vector<double>
5959 * get_dirichlet_load(const types::boundary_id &boundary_id,
5960 * const int &direction) const override
5962 * std::vector<double> displ_incr(dim, 0.0);
5963 * (void)boundary_id;
5965 * AssertThrow(false, ExcMessage("Displacement loading not implemented for Franceschini examples."));
5967 * return displ_incr;
5970 * virtual Tensor<1,dim>
5971 * get_neumann_traction (const types::boundary_id &boundary_id,
5972 * const Point<dim> &pt,
5973 * const Tensor<1,dim> &N) const override
5975 * if (this->parameters.load_type == "pressure")
5977 * if (boundary_id == 2)
5979 * return (this->parameters.load * N);
5981 * const double final_load = this->parameters.load;
5982 * const double final_load_time = 10 * this->time.get_delta_t();
5983 * const double current_time = this->time.get_current();
5986 * const double c = final_load_time / 2.0;
5987 * const double r = 200.0 * 0.03 / c;
5989 * const double load = final_load * std::exp(r * current_time)
5990 * / ( std::exp(c * current_time) + std::exp(r * current_time));
5998 * return Tensor<1,dim>();
6005 * <a name="nonlinear-poro-viscoelasticity.cc-ExamplestoreproduceexperimentsbyBuddayetal2017"></a>
6006 * <h3>Examples to reproduce experiments by Budday et al. 2017</h3>
6007 * We group the definition of the geometry, boundary and loading conditions specific to
6008 * the examples to reproduce experiments by Budday et al. 2017 into specific classes.
6013 * <a name="nonlinear-poro-viscoelasticity.cc-BaseclassCubegeometryandloadingpattern"></a>
6014 * <h4>Base class: Cube geometry and loading pattern</h4>
6017 * template <int dim>
6018 * class BrainBudday2017BaseCube
6019 * : public Solid<dim>
6022 * BrainBudday2017BaseCube (const Parameters::AllParameters ¶meters)
6023 * : Solid<dim> (parameters)
6026 * virtual ~BrainBudday2017BaseCube () {}
6030 * make_grid() override
6032 * GridGenerator::hyper_cube(this->triangulation,
6037 * typename Triangulation<dim>::active_cell_iterator cell =
6038 * this->triangulation.begin_active(), endc = this->triangulation.end();
6039 * for (; cell != endc; ++cell)
6041 * for (unsigned int face = 0; face < GeometryInfo<dim>::faces_per_cell; ++face)
6042 * if (cell->face(face)->at_boundary() == true &&
6043 * ( cell->face(face)->boundary_id() == 0 ||
6044 * cell->face(face)->boundary_id() == 1 ||
6045 * cell->face(face)->boundary_id() == 2 ||
6046 * cell->face(face)->boundary_id() == 3 ) )
6048 * cell->face(face)->set_boundary_id(100);
6052 * GridTools::scale(this->parameters.scale, this->triangulation);
6053 * this->triangulation.refine_global(std::max (1U, this->parameters.global_refinement));
6057 * get_prescribed_fluid_flow (const types::boundary_id &boundary_id,
6058 * const Point<dim> &pt) const override
6061 * (void)boundary_id;
6065 * virtual std::pair<types::boundary_id,types::boundary_id>
6066 * get_drained_boundary_id_for_output() const override
6068 * return std::make_pair(100,100);
6075 * <a name="nonlinear-poro-viscoelasticity.cc-DerivedclassUniaxialboundaryconditions"></a>
6076 * <h4>Derived class: Uniaxial boundary conditions</h4>
6079 * template <int dim>
6080 * class BrainBudday2017CubeTensionCompression
6081 * : public BrainBudday2017BaseCube<dim>
6084 * BrainBudday2017CubeTensionCompression (const Parameters::AllParameters ¶meters)
6085 * : BrainBudday2017BaseCube<dim> (parameters)
6088 * virtual ~BrainBudday2017CubeTensionCompression () {}
6092 * define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
6094 * tracked_vertices[0][0] = 0.5*this->parameters.scale;
6095 * tracked_vertices[0][1] = 0.5*this->parameters.scale;
6096 * tracked_vertices[0][2] = 1.0*this->parameters.scale;
6098 * tracked_vertices[1][0] = 0.5*this->parameters.scale;
6099 * tracked_vertices[1][1] = 0.5*this->parameters.scale;
6100 * tracked_vertices[1][2] = 0.5*this->parameters.scale;
6104 * make_dirichlet_constraints(AffineConstraints<double> &constraints) override
6106 * if (this->time.get_timestep() < 2)
6108 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
6110 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
6112 * (this->fe.component_mask(this->pressure)));
6116 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6118 * Functions::ZeroFunction<dim>(this->n_components),
6120 * (this->fe.component_mask(this->pressure)));
6122 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6124 * Functions::ZeroFunction<dim>(this->n_components),
6126 * this->fe.component_mask(this->z_displacement) );
6128 * Point<dim> fix_node(0.5*this->parameters.scale, 0.5*this->parameters.scale, 0.0);
6129 * typename DoFHandler<dim>::active_cell_iterator
6130 * cell = this->dof_handler_ref.begin_active(), endc = this->dof_handler_ref.end();
6131 * for (; cell != endc; ++cell)
6132 * for (unsigned int node = 0; node < GeometryInfo<dim>::vertices_per_cell; ++node)
6134 * if ( (abs(cell->vertex(node)[2]-fix_node[2]) < (1e-6 * this->parameters.scale))
6135 * && (abs(cell->vertex(node)[0]-fix_node[0]) < (1e-6 * this->parameters.scale)))
6136 * constraints.add_line(cell->vertex_dof_index(node, 0));
6138 * if ( (abs(cell->vertex(node)[2]-fix_node[2]) < (1e-6 * this->parameters.scale))
6139 * && (abs(cell->vertex(node)[1]-fix_node[1]) < (1e-6 * this->parameters.scale)))
6140 * constraints.add_line(cell->vertex_dof_index(node, 1));
6143 * if (this->parameters.load_type == "displacement")
6145 * const std::vector<double> value = get_dirichlet_load(5,2);
6146 * const FEValuesExtractors::Scalar direction(this->z_displacement);
6148 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6150 * Functions::ConstantFunction<dim>(value[2],this->n_components),
6152 * this->fe.component_mask(direction));
6156 * virtual Tensor<1,dim>
6157 * get_neumann_traction (const types::boundary_id &boundary_id,
6158 * const Point<dim> &pt,
6159 * const Tensor<1,dim> &N) const override
6161 * if (this->parameters.load_type == "pressure")
6163 * if (boundary_id == 5)
6165 * const double final_load = this->parameters.load;
6166 * const double current_time = this->time.get_current();
6167 * const double final_time = this->time.get_end();
6168 * const double num_cycles = 3.0;
6170 * return final_load/2.0 * (1.0 - std::sin(numbers::PI * (2.0*num_cycles*current_time/final_time + 0.5))) * N;
6176 * return Tensor<1,dim>();
6179 * virtual types::boundary_id
6180 * get_reaction_boundary_id_for_output() const override
6185 * virtual std::vector<double>
6186 * get_dirichlet_load(const types::boundary_id &boundary_id,
6187 * const int &direction) const override
6189 * std::vector<double> displ_incr(dim,0.0);
6191 * if ( (boundary_id == 5) && (direction == 2) )
6193 * const double final_displ = this->parameters.load;
6194 * const double current_time = this->time.get_current();
6195 * const double final_time = this->time.get_end();
6196 * const double delta_time = this->time.get_delta_t();
6197 * const double num_cycles = 3.0;
6198 * double current_displ = 0.0;
6199 * double previous_displ = 0.0;
6201 * if (this->parameters.num_cycle_sets == 1)
6203 * current_displ = final_displ/2.0 * (1.0
6204 * - std::sin(numbers::PI * (2.0*num_cycles*current_time/final_time + 0.5)));
6205 * previous_displ = final_displ/2.0 * (1.0
6206 * - std::sin(numbers::PI * (2.0*num_cycles*(current_time-delta_time)/final_time + 0.5)));
6210 * if ( current_time <= (final_time*1.0/3.0) )
6212 * current_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI *
6213 * (2.0*num_cycles*current_time/(final_time*1.0/3.0) + 0.5)));
6214 * previous_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI *
6215 * (2.0*num_cycles*(current_time-delta_time)/(final_time*1.0/3.0) + 0.5)));
6219 * current_displ = final_displ * (1.0 - std::sin(numbers::PI *
6220 * (2.0*num_cycles*current_time / (final_time*2.0/3.0)
6221 * - (num_cycles - 0.5) )));
6222 * previous_displ = final_displ * (1.0 - std::sin(numbers::PI *
6223 * (2.0*num_cycles*(current_time-delta_time) / (final_time*2.0/3.0)
6224 * - (num_cycles - 0.5))));
6227 * displ_incr[2] = current_displ - previous_displ;
6229 * return displ_incr;
6236 * <a name="nonlinear-poro-viscoelasticity.cc-DerivedclassNolateraldisplacementinloadingsurfaces"></a>
6237 * <h4>Derived class: No lateral displacement in loading surfaces</h4>
6240 * template <int dim>
6241 * class BrainBudday2017CubeTensionCompressionFullyFixed
6242 * : public BrainBudday2017BaseCube<dim>
6245 * BrainBudday2017CubeTensionCompressionFullyFixed (const Parameters::AllParameters ¶meters)
6246 * : BrainBudday2017BaseCube<dim> (parameters)
6249 * virtual ~BrainBudday2017CubeTensionCompressionFullyFixed () {}
6253 * define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
6255 * tracked_vertices[0][0] = 0.5*this->parameters.scale;
6256 * tracked_vertices[0][1] = 0.5*this->parameters.scale;
6257 * tracked_vertices[0][2] = 1.0*this->parameters.scale;
6259 * tracked_vertices[1][0] = 0.5*this->parameters.scale;
6260 * tracked_vertices[1][1] = 0.5*this->parameters.scale;
6261 * tracked_vertices[1][2] = 0.5*this->parameters.scale;
6265 * make_dirichlet_constraints(AffineConstraints<double> &constraints) override
6267 * if (this->time.get_timestep() < 2)
6269 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
6271 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
6273 * (this->fe.component_mask(this->pressure)));
6277 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6279 * Functions::ZeroFunction<dim>(this->n_components),
6281 * (this->fe.component_mask(this->pressure)));
6284 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6286 * Functions::ZeroFunction<dim>(this->n_components),
6288 * (this->fe.component_mask(this->x_displacement) |
6289 * this->fe.component_mask(this->y_displacement) |
6290 * this->fe.component_mask(this->z_displacement) ));
6293 * if (this->parameters.load_type == "displacement")
6295 * const std::vector<double> value = get_dirichlet_load(5,2);
6296 * const FEValuesExtractors::Scalar direction(this->z_displacement);
6298 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6300 * Functions::ConstantFunction<dim>(value[2],this->n_components),
6302 * this->fe.component_mask(direction) );
6304 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6306 * Functions::ZeroFunction<dim>(this->n_components),
6308 * (this->fe.component_mask(this->x_displacement) |
6309 * this->fe.component_mask(this->y_displacement) ));
6313 * virtual Tensor<1,dim>
6314 * get_neumann_traction (const types::boundary_id &boundary_id,
6315 * const Point<dim> &pt,
6316 * const Tensor<1,dim> &N) const override
6318 * if (this->parameters.load_type == "pressure")
6320 * if (boundary_id == 5)
6322 * const double final_load = this->parameters.load;
6323 * const double current_time = this->time.get_current();
6324 * const double final_time = this->time.get_end();
6325 * const double num_cycles = 3.0;
6327 * return final_load/2.0 * (1.0 - std::sin(numbers::PI * (2.0*num_cycles*current_time/final_time + 0.5))) * N;
6333 * return Tensor<1,dim>();
6336 * virtual types::boundary_id
6337 * get_reaction_boundary_id_for_output() const override
6342 * virtual std::vector<double>
6343 * get_dirichlet_load(const types::boundary_id &boundary_id,
6344 * const int &direction) const override
6346 * std::vector<double> displ_incr(dim,0.0);
6348 * if ( (boundary_id == 5) && (direction == 2) )
6350 * const double final_displ = this->parameters.load;
6351 * const double current_time = this->time.get_current();
6352 * const double final_time = this->time.get_end();
6353 * const double delta_time = this->time.get_delta_t();
6354 * const double num_cycles = 3.0;
6355 * double current_displ = 0.0;
6356 * double previous_displ = 0.0;
6358 * if (this->parameters.num_cycle_sets == 1)
6360 * current_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI * (2.0*num_cycles*current_time/final_time + 0.5)));
6361 * previous_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI * (2.0*num_cycles*(current_time-delta_time)/final_time + 0.5)));
6365 * if ( current_time <= (final_time*1.0/3.0) )
6367 * current_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI *
6368 * (2.0*num_cycles*current_time/(final_time*1.0/3.0) + 0.5)));
6369 * previous_displ = final_displ/2.0 * (1.0 - std::sin(numbers::PI *
6370 * (2.0*num_cycles*(current_time-delta_time)/(final_time*1.0/3.0) + 0.5)));
6374 * current_displ = final_displ * (1.0 - std::sin(numbers::PI *
6375 * (2.0*num_cycles*current_time / (final_time*2.0/3.0)
6376 * - (num_cycles - 0.5) )));
6377 * previous_displ = final_displ * (1.0 - std::sin(numbers::PI *
6378 * (2.0*num_cycles*(current_time-delta_time) / (final_time*2.0/3.0)
6379 * - (num_cycles - 0.5))));
6382 * displ_incr[2] = current_displ - previous_displ;
6384 * return displ_incr;
6391 * <a name="nonlinear-poro-viscoelasticity.cc-DerivedclassNolateralorverticaldisplacementinloadingsurface"></a>
6392 * <h4>Derived class: No lateral or vertical displacement in loading surface</h4>
6395 * template <int dim>
6396 * class BrainBudday2017CubeShearFullyFixed
6397 * : public BrainBudday2017BaseCube<dim>
6400 * BrainBudday2017CubeShearFullyFixed (const Parameters::AllParameters ¶meters)
6401 * : BrainBudday2017BaseCube<dim> (parameters)
6404 * virtual ~BrainBudday2017CubeShearFullyFixed () {}
6408 * define_tracked_vertices(std::vector<Point<dim> > &tracked_vertices) override
6410 * tracked_vertices[0][0] = 0.75*this->parameters.scale;
6411 * tracked_vertices[0][1] = 0.5*this->parameters.scale;
6412 * tracked_vertices[0][2] = 0.0*this->parameters.scale;
6414 * tracked_vertices[1][0] = 0.25*this->parameters.scale;
6415 * tracked_vertices[1][1] = 0.5*this->parameters.scale;
6416 * tracked_vertices[1][2] = 0.0*this->parameters.scale;
6420 * make_dirichlet_constraints(AffineConstraints<double> &constraints) override
6422 * if (this->time.get_timestep() < 2)
6424 * VectorTools::interpolate_boundary_values(this->dof_handler_ref,
6426 * Functions::ConstantFunction<dim>(this->parameters.drained_pressure,this->n_components),
6428 * (this->fe.component_mask(this->pressure)));
6432 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6434 * Functions::ZeroFunction<dim>(this->n_components),
6436 * (this->fe.component_mask(this->pressure)));
6439 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6441 * Functions::ZeroFunction<dim>(this->n_components),
6443 * (this->fe.component_mask(this->x_displacement) |
6444 * this->fe.component_mask(this->y_displacement) |
6445 * this->fe.component_mask(this->z_displacement) ));
6448 * if (this->parameters.load_type == "displacement")
6450 * const std::vector<double> value = get_dirichlet_load(4,0);
6451 * const FEValuesExtractors::Scalar direction(this->x_displacement);
6453 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6455 * Functions::ConstantFunction<dim>(value[0],this->n_components),
6457 * this->fe.component_mask(direction));
6459 * VectorTools::interpolate_boundary_values( this->dof_handler_ref,
6461 * Functions::ZeroFunction<dim>(this->n_components),
6463 * (this->fe.component_mask(this->y_displacement) |
6464 * this->fe.component_mask(this->z_displacement) ));
6468 * virtual Tensor<1,dim>
6469 * get_neumann_traction (const types::boundary_id &boundary_id,
6470 * const Point<dim> &pt,
6471 * const Tensor<1,dim> &N) const override
6473 * if (this->parameters.load_type == "pressure")
6475 * if (boundary_id == 4)
6477 * const double final_load = this->parameters.load;
6478 * const double current_time = this->time.get_current();
6479 * const double final_time = this->time.get_end();
6480 * const double num_cycles = 3.0;
6481 * const Tensor<1,3> axis ({0.0,1.0,0.0});
6482 * const double angle = numbers::PI;
6483 * static const Tensor< 2, dim, double> R(Physics::Transformations::Rotations::rotation_matrix_3d(axis,angle));
6485 * return (final_load * (std::sin(2.0*(numbers::PI)*num_cycles*current_time/final_time)) * (R * N));
6491 * return Tensor<1,dim>();
6494 * virtual types::boundary_id
6495 * get_reaction_boundary_id_for_output() const override
6500 * virtual std::vector<double>
6501 * get_dirichlet_load(const types::boundary_id &boundary_id,
6502 * const int &direction) const override
6504 * std::vector<double> displ_incr (dim, 0.0);
6506 * if ( (boundary_id == 4) && (direction == 0) )
6508 * const double final_displ = this->parameters.load;
6509 * const double current_time = this->time.get_current();
6510 * const double final_time = this->time.get_end();
6511 * const double delta_time = this->time.get_delta_t();
6512 * const double num_cycles = 3.0;
6513 * double current_displ = 0.0;
6514 * double previous_displ = 0.0;
6516 * if (this->parameters.num_cycle_sets == 1)
6518 * current_displ = final_displ * (std::sin(2.0*(numbers::PI)*num_cycles*current_time/final_time));
6519 * previous_displ = final_displ * (std::sin(2.0*(numbers::PI)*num_cycles*(current_time-delta_time)/final_time));
6523 * AssertThrow(false, ExcMessage("Problem type not defined. Budday shear experiments implemented only for one set of cycles."));
6525 * displ_incr[0] = current_displ - previous_displ;
6527 * return displ_incr;
6536 * <a name="nonlinear-poro-viscoelasticity.cc-Mainfunction"></a>
6537 * <h3>Main function</h3>
6538 * Lastly we provide the main driver function which is similar to the other tutorials.
6541 * int main (int argc, char *argv[])
6543 * using namespace dealii;
6544 * using namespace NonLinearPoroViscoElasticity;
6546 * const unsigned int n_tbb_processes = 1;
6547 * Utilities::MPI::MPI_InitFinalize mpi_initialization(argc, argv, n_tbb_processes);
6551 * Parameters::AllParameters parameters ("parameters.prm");
6552 * if (parameters.geom_type == "Ehlers_tube_step_load")
6554 * VerificationEhlers1999StepLoad<3> solid_3d(parameters);
6557 * else if (parameters.geom_type == "Ehlers_tube_increase_load")
6559 * VerificationEhlers1999IncreaseLoad<3> solid_3d(parameters);
6562 * else if (parameters.geom_type == "Ehlers_cube_consolidation")
6564 * VerificationEhlers1999CubeConsolidation<3> solid_3d(parameters);
6567 * else if (parameters.geom_type == "Franceschini_consolidation")
6569 * Franceschini2006Consolidation<3> solid_3d(parameters);
6572 * else if (parameters.geom_type == "Budday_cube_tension_compression")
6574 * BrainBudday2017CubeTensionCompression<3> solid_3d(parameters);
6577 * else if (parameters.geom_type == "Budday_cube_tension_compression_fully_fixed")
6579 * BrainBudday2017CubeTensionCompressionFullyFixed<3> solid_3d(parameters);
6582 * else if (parameters.geom_type == "Budday_cube_shear_fully_fixed")
6584 * BrainBudday2017CubeShearFullyFixed<3> solid_3d(parameters);
6589 * AssertThrow(false, ExcMessage("Problem type not defined. Current setting: " + parameters.geom_type));
6593 * catch (std::exception &exc)
6595 * if (Utilities::MPI::this_mpi_process(MPI_COMM_WORLD) == 0)
6597 * std::cerr << std::endl << std::endl
6598 * << "----------------------------------------------------"
6600 * std::cerr << "Exception on processing: " << std::endl << exc.what()
6601 * << std::endl << "Aborting!" << std::endl
6602 * << "----------------------------------------------------"
6610 * if (Utilities::MPI::this_mpi_process(MPI_COMM_WORLD) == 0)
6612 * std::cerr << std::endl << std::endl
6613 * << "----------------------------------------------------"
6615 * std::cerr << "Unknown exception!" << std::endl << "Aborting!"
6617 * << "----------------------------------------------------"
DataPostprocessorVector(const std::string &name, const UpdateFlags update_flags)
static constexpr const SymmetricTensor< 2, dim > I
void reinit(const Vector &v, const bool omit_zeroing_entries=false, const bool allow_different_maps=false)
#define DEAL_II_VERSION_GTE(major, minor, subminor)
#define Assert(cond, exc)
#define AssertDimension(dim1, dim2)
#define AssertThrow(cond, exc)
typename ActiveSelector::active_cell_iterator active_cell_iterator
LinearOperator< Range, Domain, Payload > linear_operator(const OperatorExemplar &, const Matrix &)
void loop(IteratorType begin, std_cxx20::type_identity_t< IteratorType > end, DOFINFO &dinfo, INFOBOX &info, const std::function< void(std_cxx20::type_identity_t< DOFINFO > &, typename INFOBOX::CellInfo &)> &cell_worker, const std::function< void(std_cxx20::type_identity_t< DOFINFO > &, typename INFOBOX::CellInfo &)> &boundary_worker, const std::function< void(std_cxx20::type_identity_t< DOFINFO > &, std_cxx20::type_identity_t< DOFINFO > &, typename INFOBOX::CellInfo &, typename INFOBOX::CellInfo &)> &face_worker, AssemblerType &assembler, const LoopControl &lctrl=LoopControl())
const bool IsBlockVector< VectorType >::value
void make_sparsity_pattern(const DoFHandler< dim, spacedim > &dof_handler, SparsityPatternBase &sparsity_pattern, const AffineConstraints< number > &constraints={}, const bool keep_constrained_dofs=true, const types::subdomain_id subdomain_id=numbers::invalid_subdomain_id)
@ update_values
Shape function values.
@ update_normal_vectors
Normal vectors.
@ update_JxW_values
Transformed quadrature weights.
@ update_gradients
Shape function gradients.
@ update_quadrature_points
Transformed quadrature points.
void approximate(const SynchronousIterators< std::tuple< typename DoFHandler< dim, spacedim >::active_cell_iterator, Vector< float >::iterator > > &cell, const Mapping< dim, spacedim > &mapping, const DoFHandler< dim, spacedim > &dof_handler, const InputVector &solution, const unsigned int component)
void component_wise(DoFHandler< dim, spacedim > &dof_handler, const std::vector< unsigned int > &target_component=std::vector< unsigned int >())
void Cuthill_McKee(DoFHandler< dim, spacedim > &dof_handler, const bool reversed_numbering=false, const bool use_constraints=false, const std::vector< types::global_dof_index > &starting_indices=std::vector< types::global_dof_index >())
@ valid
Iterator points to a valid object.
@ matrix
Contents is actually a matrix.
@ diagonal
Matrix is diagonal.
constexpr types::blas_int zero
constexpr types::blas_int one
void cell_matrix(FullMatrix< double > &M, const FEValuesBase< dim > &fe, const FEValuesBase< dim > &fetest, const ArrayView< const std::vector< double > > &velocity, const double factor=1.)
double norm(const FEValuesBase< dim > &fe, const ArrayView< const std::vector< Tensor< 1, dim > > > &Du)
std::enable_if_t< IsBlockVector< VectorType >::value, unsigned int > n_blocks(const VectorType &vector)
Point< spacedim > point(const gp_Pnt &p, const double tolerance=1e-10)
SymmetricTensor< 2, dim, Number > C(const Tensor< 2, dim, Number > &F)
SymmetricTensor< 2, dim, Number > e(const Tensor< 2, dim, Number > &F)
SymmetricTensor< 2, dim, Number > b(const Tensor< 2, dim, Number > &F)
SymmetricTensor< 2, dim, Number > d(const Tensor< 2, dim, Number > &F, const Tensor< 2, dim, Number > &dF_dt)
Tensor< 2, dim, Number > F(const Tensor< 2, dim, Number > &Grad_u)
constexpr ReturnType< rank, T >::value_type & extract(T &t, const ArrayType &indices)
VectorType::value_type * end(VectorType &V)
T sum(const T &t, const MPI_Comm mpi_communicator)
unsigned int n_mpi_processes(const MPI_Comm mpi_communicator)
unsigned int this_mpi_process(const MPI_Comm mpi_communicator)
T reduce(const T &local_value, const MPI_Comm comm, const std::function< T(const T &, const T &)> &combiner, const unsigned int root_process=0)
void run(const Iterator &begin, const std_cxx20::type_identity_t< Iterator > &end, Worker worker, Copier copier, const ScratchData &sample_scratch_data, const CopyData &sample_copy_data, const unsigned int queue_length, const unsigned int chunk_size)
void abort(const ExceptionBase &exc) noexcept
bool check(const ConstraintKinds kind_in, const unsigned int dim)
long double gamma(const unsigned int n)
void copy(const T *begin, const T *end, U *dest)
int(& functions)(const void *v1, const void *v2)
void assemble(const MeshWorker::DoFInfoBox< dim, DOFINFO > &dinfo, A *assembler)
::VectorizedArray< Number, width > log(const ::VectorizedArray< Number, width > &)
::VectorizedArray< Number, width > exp(const ::VectorizedArray< Number, width > &)
::VectorizedArray< Number, width > sqrt(const ::VectorizedArray< Number, width > &)
::VectorizedArray< Number, width > pow(const ::VectorizedArray< Number, width > &, const Number p)
::VectorizedArray< Number, width > abs(const ::VectorizedArray< Number, width > &)
unsigned int global_dof_index
DEAL_II_HOST constexpr Number determinant(const SymmetricTensor< 2, dim, Number > &)
DEAL_II_HOST constexpr SymmetricTensor< 2, dim, Number > symmetrize(const Tensor< 2, dim, Number > &t)
DEAL_II_HOST constexpr SymmetricTensor< 2, dim, Number > invert(const SymmetricTensor< 2, dim, Number > &)
std::array< std::pair< Number, Tensor< 1, dim, Number > >, dim > eigenvectors(const SymmetricTensor< 2, dim, Number > &T, const SymmetricTensorEigenvectorMethod method=SymmetricTensorEigenvectorMethod::ql_implicit_shifts)
DEAL_II_HOST constexpr SymmetricTensor< 4, dim, Number > outer_product(const SymmetricTensor< 2, dim, Number > &t1, const SymmetricTensor< 2, dim, Number > &t2)
SymmetricTensorEigenvectorMethod