1438 *
const double g = 9.81;
1439 *
const double rho = 7700;
1447 *
template <
int dim>
1452 *
const unsigned int n_points = points.
size();
1456 *
for (
unsigned int p = 0; p < n_points; ++p)
1465 * <a name=
"ThecodeIncrementalBoundaryValuecodeclass"></a>
1503 *
template <
int dim>
1510 *
virtual void vector_value(
const Point<dim> &p,
1514 *
vector_value_list(
const std::vector<
Point<dim>> &points,
1524 *
template <
int dim>
1535 *
template <
int dim>
1548 *
template <
int dim>
1553 *
const unsigned int n_points = points.
size();
1557 *
for (
unsigned int p = 0; p < n_points; ++p)
1566 * <a name=
"ImplementationofthecodeTopLevelcodeclass"></a>
1576 *
template <
int dim>
1586 * <a name=
"Thepublicinterface"></a>
1598 *
template <
int dim>
1611 *
,
pcout(std::cout, this_mpi_process == 0)
1616 *
template <
int dim>
1619 *
dof_handler.clear();
1634 *
template <
int dim>
1647 * <a name=
"TopLevelcreate_coarse_grid"></a>
1669 *
template <
int dim>
1674 *
for (
const auto &cell :
triangulation.active_cell_iterators())
1676 *
if (face->at_boundary())
1681 *
face->set_boundary_id(0);
1683 *
face->set_boundary_id(1);
1687 *
face->set_boundary_id(2);
1689 *
face->set_boundary_id(3);
1714 * <a name=
"TopLevelsetup_system"></a>
1732 * template <int dim>
1733 * void TopLevel<dim>::setup_system()
1735 * dof_handler.distribute_dofs(fe);
1736 * locally_owned_dofs = dof_handler.locally_owned_dofs();
1737 * locally_relevant_dofs =
1738 * DoFTools::extract_locally_relevant_dofs(dof_handler);
1742 * The next step is to set up constraints due to hanging nodes. This has
1743 * been handled many times before:
1746 * hanging_node_constraints.clear();
1747 * DoFTools::make_hanging_node_constraints(dof_handler,
1748 * hanging_node_constraints);
1749 * hanging_node_constraints.close();
1753 * And then we have to set up the matrix. Here we deviate from @ref step_17 "step-17", in
1757 * at all
efficient:
if we don't give PETSc a clue as to which elements
1758 * are written to, it is (at least at the time of this writing) unbearably
1759 * slow when we set the elements in the matrix for the first time (i.e. in
1760 * the first timestep). Later on, when the elements have been allocated,
1761 * everything is much faster. In experiments we made, the first timestep
1762 * can be accelerated by almost two orders of magnitude if we instruct
1763 * PETSc which elements will be used and which are not.
1767 * To do so, we first generate the sparsity pattern of the matrix we are
1768 * going to work with, and make sure that the condensation of hanging node
1769 * constraints add the necessary additional entries in the sparsity
1773 * DynamicSparsityPattern sparsity_pattern(locally_relevant_dofs);
1774 * DoFTools::make_sparsity_pattern(dof_handler,
1776 * hanging_node_constraints,
1777 * /*keep constrained dofs*/ false);
1778 * SparsityTools::distribute_sparsity_pattern(sparsity_pattern,
1779 * locally_owned_dofs,
1781 * locally_relevant_dofs);
1784 * Note that we have used the <code>DynamicSparsityPattern</code> class
1785 * here that was already introduced in @ref step_11 "step-11", rather than the
1786 * <code>SparsityPattern</code> class that we have used in all other
1787 * cases. The reason for this is that for the latter class to work we have
1788 * to give an initial upper bound for the number of entries in each row, a
1789 * task that is traditionally done by
1790 * <code>DoFHandler::max_couplings_between_dofs()</code>. However, this
1791 * function suffers from a serious problem: it has to compute an upper
1792 * bound to the number of nonzero entries in each row, and this is a
1793 * rather complicated task, in particular in 3d. In effect, while it is
1794 * quite accurate in 2d, it often comes up with much too large a number in
1795 * 3d, and in that case the <code>SparsityPattern</code> allocates much
1796 * too much memory at first, often several 100 MBs. This is later
1797 * corrected when <code>DoFTools::make_sparsity_pattern</code> is called
1834 *
system_matrix.
reinit(locally_owned_dofs,
1835 *
locally_owned_dofs,
1837 *
mpi_communicator);
1862 * <a name=
"TopLevelassemble_system"></a>
1882 *
template <
int dim>
1886 *
system_matrix = 0;
1893 *
const unsigned int dofs_per_cell = fe.n_dofs_per_cell();
1899 *
std::vector<types::global_dof_index> local_dof_indices(dofs_per_cell);
1911 *
for (
const auto &cell : dof_handler.active_cell_iterators())
1912 *
if (cell->is_locally_owned())
1917 *
fe_values.
reinit(cell);
1932 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
1933 *
for (
unsigned int j = 0;
j < dofs_per_cell; ++
j)
1964 *
the quadrature points
on this cell:
1967 *
body_force.vector_value_list(fe_values.get_quadrature_points(),
1975 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
1978 *
fe.system_to_component_index(i).first;
1987 *
fe_values.shape_value(i,
q_point) -
1997 *
as in @
ref step_17
"step-17":
2000 *
cell->get_dof_indices(local_dof_indices);
2004 *
local_dof_indices,
2040 * want no movement at all, corresponding to the cylinder being clamped or
2041 * cemented in at this part of the boundary. At the top, however, we want
2042 * a prescribed vertical downward motion compressing the cylinder; in
2043 * addition, we only want to restrict the vertical movement, but not the
2044 * horizontal ones -- one can think of this situation as a well-greased
2045 * plate sitting on top of the cylinder pushing it downwards: the atoms of
2046 * the cylinder are forced to move downward, but they are free to slide
2047 * horizontally along the plate.
2051 * The way to describe this is as follows: for boundary indicator zero
2052 * (bottom face) we use a dim-dimensional zero function representing no
2053 * motion in any coordinate direction. For the boundary with indicator 1
2054 * (top surface), we use the <code>IncrementalBoundaryValues</code> class,
2055 * but we specify an additional argument to the
2056 * <code>VectorTools::interpolate_boundary_values</code> function denoting
2057 * which vector components it should apply to; this is a vector of bools
2058 * for each vector component and because we only want to restrict vertical
2059 * motion, it has only its last component set:
2062 * const FEValuesExtractors::Scalar z_component(dim - 1);
2063 * std::map<types::global_dof_index, double> boundary_values;
2064 * VectorTools::interpolate_boundary_values(dof_handler,
2066 * Functions::ZeroFunction<dim>(dim),
2068 * VectorTools::interpolate_boundary_values(
2071 * IncrementalBoundaryValues<dim>(present_time, present_timestep),
2073 * fe.component_mask(z_component));
2075 * PETScWrappers::MPI::Vector tmp(locally_owned_dofs, mpi_communicator);
2076 * MatrixTools::apply_boundary_values(
2077 * boundary_values, system_matrix, tmp, system_rhs, false);
2078 * incremental_displacement = tmp;
2086 * <a name="TopLevelsolve_timestep"></a>
2087 * <h4>TopLevel::solve_timestep</h4>
2091 * The next function is the one that controls what all has to happen within
2092 * a timestep. The order of things should be relatively self-explanatory
2093 * from the function names:
2096 * template <int dim>
2097 * void TopLevel<dim>::solve_timestep()
2099 * pcout << " Assembling system..." << std::flush;
2100 * assemble_system();
2101 * pcout << " norm of rhs is " << system_rhs.l2_norm() << std::endl;
2103 * const unsigned int n_iterations = solve_linear_problem();
2105 * pcout << " Solver converged in " << n_iterations << " iterations."
2108 * pcout << " Updating quadrature point data..." << std::flush;
2109 * update_quadrature_point_history();
2110 * pcout << std::endl;
2118 * <a name="TopLevelsolve_linear_problem"></a>
2119 * <h4>TopLevel::solve_linear_problem</h4>
2123 * Solving the linear system again works mostly as before. The only
2124 * difference is that we want to only keep a complete local copy of the
2125 * solution vector instead of the distributed one that we get as output from
2137 *
template <
int dim>
2141 *
locally_owned_dofs, mpi_communicator);
2151 *
cg.solve(system_matrix,
2160 *
return solver_control.last_step();
2168 * <a name=
"TopLeveloutput_results"></a>
2187 *
template <
int dim>
2191 *
data_out.attach_dof_handler(dof_handler);
2199 *
haven't implemented this case yet (another case of defensive
2203 * std::vector<std::string> solution_names;
2207 * solution_names.emplace_back("delta_x");
2210 * solution_names.emplace_back("delta_x");
2211 * solution_names.emplace_back("delta_y");
2214 * solution_names.emplace_back("delta_x");
2215 * solution_names.emplace_back("delta_y");
2216 * solution_names.emplace_back("delta_z");
2219 * Assert(false, ExcNotImplemented());
2222 * data_out.add_data_vector(incremental_displacement, solution_names);
2227 * The next thing is that we wanted to output something like the average
2228 * norm of the stresses that we have stored in each cell. This may seem
2229 * complicated, since on the present processor we only store the stresses
2230 * in quadrature points on those cells that actually belong to the present
2231 * process. In other words, it seems as if we can't compute
the average
2235 * for all the other cells as this information would not be touched. The
2236 * following little loop does this. We enclose the entire block into a
2237 * pair of braces to make sure that the iterator variables do not remain
2238 * accidentally visible beyond the end of the block in which they are
2242 * Vector<double> norm_of_stress(triangulation.n_active_cells());
2246 * Loop over all the cells...
2249 * for (auto &cell : triangulation.active_cell_iterators())
2250 * if (cell->is_locally_owned())
2254 * On these cells, add up the stresses over all quadrature
2258 * SymmetricTensor<2, dim> accumulated_stress;
2259 * for (unsigned int q = 0; q < quadrature_formula.size(); ++q)
2260 * accumulated_stress +=
2261 * reinterpret_cast<PointHistory<dim> *>(cell->user_pointer())[q]
2266 * ...then write the norm of the average to their destination:
2269 * norm_of_stress(cell->active_cell_index()) =
2270 * (accumulated_stress / quadrature_formula.size()).norm();
2274 * And on the cells that we are not interested in, set the respective
2275 * value in the vector to a bogus value (norms must be positive, and a
2276 * large negative value should catch your eye) in order to make sure
2277 * that if we were somehow wrong about our assumption that these
2278 * elements would not appear in the output file, that we would find out
2279 * by looking at the graphical output:
2283 * norm_of_stress(cell->active_cell_index()) = -1e+20;
2287 * Finally attach this vector as well to be treated for output:
2290 * data_out.add_data_vector(norm_of_stress, "norm_of_stress");
2294 * As a last piece of data, let us also add the partitioning of the domain
2295 * into subdomains associated with the processors if this is a parallel
2296 * job. This works in the exact same way as in the @ref step_17 "step-17" program:
2299 * std::vector<types::subdomain_id> partition_int(
2300 * triangulation.n_active_cells());
2301 * GridTools::get_subdomain_association(triangulation, partition_int);
2302 * const Vector<double> partitioning(partition_int.begin(),
2303 * partition_int.end());
2304 * data_out.add_data_vector(partitioning, "partitioning");
2308 * Finally, with all this data, we can instruct deal.II to munge the
2309 * information and produce some intermediate data structures that contain
2310 * all these solution and other data vectors:
2313 * data_out.build_patches();
2317 * Let us call a function that opens the necessary output files and writes
2318 * the data we have generated into them. The function automatically
2319 * constructs the file names from the given directory name (the first
2320 * argument) and file name base (second argument). It augments the resulting
2321 * string by pieces that result from the time step number and a "piece
2322 * number" that corresponds to a part of the overall domain that can consist
2323 * of one or more subdomains.
2327 * The function also writes a record files (with suffix `.pvd`) for Paraview
2328 * that describes how all of these output files combine into the data for
2329 * this single time step:
2332 * const std::string pvtu_filename = data_out.write_vtu_with_pvtu_record(
2333 * "./", "solution", timestep_no, mpi_communicator, 4);
2337 * The record files must be written only once and not by each processor,
2338 * so we do this on processor 0:
2341 * if (this_mpi_process == 0)
2345 * Finally, we write the paraview record, that references all .pvtu
2346 * files and their respective time. Note that the variable
2347 * times_and_names is declared static, so it will retain the entries
2348 * from the previous timesteps.
2351 * static std::vector<std::pair<double, std::string>> times_and_names;
2352 * times_and_names.emplace_back(present_time, pvtu_filename);
2353 * std::ofstream pvd_output("solution.pvd");
2354 * DataOutBase::write_pvd_record(pvd_output, times_and_names);
2363 * <a name="TopLeveldo_initial_timestep"></a>
2364 * <h4>TopLevel::do_initial_timestep</h4>
2368 * This and the next function handle the overall structure of the first and
2369 * following timesteps, respectively. The first timestep is slightly more
2370 * involved because we want to compute it multiple times on successively
2371 * refined meshes, each time starting from a clean state. At the end of
2372 * these computations, in which we compute the incremental displacements
2373 * each time, we use the last results obtained for the incremental
2374 * displacements to compute the resulting stress updates and move the mesh
2375 * accordingly. On this new mesh, we then output the solution and any
2376 * additional data we consider important.
2380 * All this is interspersed by generating output to the console to update
2381 * the person watching the screen on what is going on. As in @ref step_17 "step-17", the
2382 * use of <code>pcout</code> instead of <code>std::cout</code> makes sure
2383 * that only one of the parallel processes is actually writing to the
2384 * console, without having to explicitly code an if-statement in each place
2385 * where we generate output:
2388 * template <int dim>
2389 * void TopLevel<dim>::do_initial_timestep()
2391 * present_time += present_timestep;
2393 * pcout << "Timestep " << timestep_no << " at time " << present_time
2396 * for (unsigned int cycle = 0; cycle < 2; ++cycle)
2398 * pcout << " Cycle " << cycle << ':
' << std::endl;
2401 * create_coarse_grid();
2403 * refine_initial_grid();
2405 * pcout << " Number of active cells: "
2406 * << triangulation.n_active_cells() << " (by partition:";
2407 * for (unsigned int p = 0; p < n_mpi_processes; ++p)
2408 * pcout << (p == 0 ? ' ' : '+
')
2409 * << (GridTools::count_cells_with_subdomain_association(
2410 * triangulation, p));
2411 * pcout << ')
' << std::endl;
2415 * pcout << " Number of degrees of freedom: " << dof_handler.n_dofs()
2416 * << " (by partition:";
2417 * for (unsigned int p = 0; p < n_mpi_processes; ++p)
2418 * pcout << (p == 0 ? ' ' : '+
')
2419 * << (DoFTools::count_dofs_with_subdomain_association(dof_handler,
2421 * pcout << ')
' << std::endl;
2429 * pcout << std::endl;
2437 * <a name="TopLeveldo_timestep"></a>
2438 * <h4>TopLevel::do_timestep</h4>
2442 * Subsequent timesteps are simpler, and probably do not require any more
2443 * documentation given the explanations for the previous function above:
2446 * template <int dim>
2447 * void TopLevel<dim>::do_timestep()
2449 * present_time += present_timestep;
2451 * pcout << "Timestep " << timestep_no << " at time " << present_time
2453 * if (present_time > end_time)
2455 * present_timestep -= (present_time - end_time);
2456 * present_time = end_time;
2465 * pcout << std::endl;
2472 * <a name="TopLevelrefine_initial_grid"></a>
2473 * <h4>TopLevel::refine_initial_grid</h4>
2477 * The following function is called when solving the first time step on
2478 * successively refined meshes. After each iteration, it computes a
2479 * refinement criterion, refines the mesh, and sets up the history variables
2480 * in each quadrature point again to a clean state.
2483 * template <int dim>
2484 * void TopLevel<dim>::refine_initial_grid()
2488 * First, let each process compute error indicators for the cells it owns:
2491 * Vector<float> error_per_cell(triangulation.n_active_cells());
2492 * KellyErrorEstimator<dim>::estimate(
2494 * QGauss<dim - 1>(fe.degree + 1),
2495 * std::map<types::boundary_id, const Function<dim> *>(),
2496 * incremental_displacement,
2500 * MultithreadInfo::n_threads(),
2501 * this_mpi_process);
2505 * Then set up a global vector into which we merge the local indicators
2506 * from each of the %parallel processes:
2509 * const unsigned int n_local_cells =
2510 * triangulation.n_locally_owned_active_cells();
2512 * PETScWrappers::MPI::Vector distributed_error_per_cell(
2513 * mpi_communicator, triangulation.n_active_cells(), n_local_cells);
2515 * for (unsigned int i = 0; i < error_per_cell.size(); ++i)
2516 * if (error_per_cell(i) != 0)
2517 * distributed_error_per_cell(i) = error_per_cell(i);
2518 * distributed_error_per_cell.compress(VectorOperation::insert);
2522 * Once we have that, copy it back into local copies on all processors and
2523 * refine the mesh accordingly:
2526 * error_per_cell = distributed_error_per_cell;
2527 * GridRefinement::refine_and_coarsen_fixed_number(triangulation,
2531 * triangulation.execute_coarsening_and_refinement();
2535 * Finally, set up quadrature point data again on the new mesh, and only
2536 * on those cells that we have determined to be ours:
2539 * setup_quadrature_point_history();
2547 * <a name="TopLevelmove_mesh"></a>
2548 * <h4>TopLevel::move_mesh</h4>
2552 * At the end of each time step, we move the nodes of the mesh according to
2553 * the incremental displacements computed in this time step. To do this, we
2554 * keep a vector of flags that indicate for each vertex whether we have
2555 * already moved it around, and then loop over all cells and move those
2556 * vertices of the cell that have not been moved yet. It is worth noting
2557 * that it does not matter from which of the cells adjacent to a vertex we
2558 * move this vertex: since we compute the displacement using a continuous
2559 * finite element, the displacement field is continuous as well and we can
2560 * compute the displacement of a given vertex from each of the adjacent
2561 * cells. We only have to make sure that we move each node exactly once,
2562 * which is why we keep the vector of flags.
2566 * There are two noteworthy things in this function. First, how we get the
2567 * displacement field at a given vertex using the
2568 * <code>cell-@>vertex_dof_index(v,d)</code> function that returns the index
2569 * of the <code>d</code>th degree of freedom at vertex <code>v</code> of the
2570 * given cell. In the present case, displacement in the k-th coordinate
2571 * direction corresponds to the k-th component of the finite element. Using a
2572 * function like this bears a certain risk, because it uses knowledge of the
2573 * order of elements that we have taken together for this program in the
2574 * <code>FESystem</code> element. If we decided to add an additional
2575 * variable, for example a pressure variable for stabilization, and happened
2576 * to insert it as the first variable of the element, then the computation
2577 * below will start to produce nonsensical results. In addition, this
2578 * computation rests on other assumptions: first, that the element we use
2579 * has, indeed, degrees of freedom that are associated with vertices. This
2580 * is indeed the case for the present Q1 element, as would be for all Qp
2581 * elements of polynomial order <code>p</code>. However, it would not hold
2582 * for discontinuous elements, or elements for mixed formulations. Secondly,
2583 * it also rests on the assumption that the displacement at a vertex is
2584 * determined solely by the value of the degree of freedom associated with
2585 * this vertex; in other words, all shape functions corresponding to other
2586 * degrees of freedom are zero at this particular vertex. Again, this is the
2587 * case for the present element, but is not so for all elements that are
2588 * presently available in deal.II. Despite its risks, we choose to use this
2589 * way in order to present a way to query individual degrees of freedom
2590 * associated with vertices.
2594 * In this context, it is instructive to point out what a more general way
2595 * would be. For general finite elements, the way to go would be to take a
2596 * quadrature formula with the quadrature points in the vertices of a
2597 * cell. The <code>QTrapezoid</code> formula for the trapezoidal rule does
2598 * exactly this. With this quadrature formula, we would then initialize an
2599 * <code>FEValues</code> object in each cell, and use the
2600 * <code>FEValues::get_function_values</code> function to obtain the values
2601 * of the solution function in the quadrature points, i.e. the vertices of
2602 * the cell. These are the only values that we really need, i.e. we are not
2603 * at all interested in the weights (or the <code>JxW</code> values)
2604 * associated with this particular quadrature formula, and this can be
2605 * specified as the last argument in the constructor to
2606 * <code>FEValues</code>. The only point of minor inconvenience in this
2607 * scheme is that we have to figure out which quadrature point corresponds
2608 * to the vertex we consider at present, as they may or may not be ordered
2609 * in the same order.
2613 * This inconvenience could be avoided if finite elements have support
2614 * points on vertices (which the one here has; for the concept of support
2615 * points, see @ref GlossSupport "support points"). For such a case, one
2616 * could construct a custom quadrature rule using
2617 * FiniteElement::get_unit_support_points(). The first
2618 * <code>cell->n_vertices()*fe.dofs_per_vertex</code>
2619 * quadrature points will then correspond to the vertices of the cell and
2620 * are ordered consistent with <code>cell-@>vertex(i)</code>, taking into
2621 * account that support points for vector elements will be duplicated
2622 * <code>fe.dofs_per_vertex</code> times.
2626 * Another point worth explaining about this short function is the way in
2627 * which the triangulation class exports information about its vertices:
2628 * through the <code>Triangulation::n_vertices</code> function, it
2629 * advertises how many vertices there are in the triangulation. Not all of
2630 * them are actually in use all the time -- some are left-overs from cells
2631 * that have been coarsened previously and remain in existence since deal.II
2632 * never changes the number of a vertex once it has come into existence,
2633 * even if vertices with lower number go away. Secondly, the location
2634 * returned by <code>cell-@>vertex(v)</code> is not only a read-only object
2635 * of type <code>Point@<dim@></code>, but in fact a reference that can be
2636 * written to. This allows to move around the nodes of a mesh with relative
2637 * ease, but it is worth pointing out that it is the responsibility of an
2638 * application program using this feature to make sure that the resulting
2639 * cells are still useful, i.e. are not distorted so much that the cell is
2640 * degenerated (indicated, for example, by negative Jacobians). Note that we
2641 * do not have any provisions in this function to actually ensure this, we
2646 * After this lengthy introduction, here are the full 20 or so lines of
2650 * template <int dim>
2651 * void TopLevel<dim>::move_mesh()
2653 * pcout << " Moving mesh..." << std::endl;
2655 * std::vector<bool> vertex_touched(triangulation.n_vertices(), false);
2656 * for (auto &cell : dof_handler.active_cell_iterators())
2657 * for (const auto v : cell->vertex_indices())
2658 * if (vertex_touched[cell->vertex_index(v)] == false)
2660 * vertex_touched[cell->vertex_index(v)] = true;
2662 * Point<dim> vertex_displacement;
2663 * for (unsigned int d = 0; d < dim; ++d)
2664 * vertex_displacement[d] =
2665 * incremental_displacement(cell->vertex_dof_index(v, d));
2667 * cell->vertex(v) += vertex_displacement;
2675 * <a name="TopLevelsetup_quadrature_point_history"></a>
2676 * <h4>TopLevel::setup_quadrature_point_history</h4>
2680 * At the beginning of our computations, we needed to set up initial values
2681 * of the history variables, such as the existing stresses in the material,
2682 * that we store in each quadrature point. As mentioned above, we use the
2683 * <code>user_pointer</code> for this that is available in each cell.
2687 * To put this into larger perspective, we note that if we had previously
2688 * available stresses in our model (which we assume do not exist for the
2689 * purpose of this program), then we would need to interpolate the field of
2690 * preexisting stresses to the quadrature points. Likewise, if we were to
2691 * simulate elasto-plastic materials with hardening/softening, then we would
2692 * have to store additional history variables like the present yield stress
2693 * of the accumulated plastic strains in each quadrature
2694 * points. Pre-existing hardening or weakening would then be implemented by
2695 * interpolating these variables in the present function as well.
2698 * template <int dim>
2699 * void TopLevel<dim>::setup_quadrature_point_history()
2703 * For good measure, we set all user pointers of all cells, whether
2704 * ours of not, to the null pointer. This way, if we ever access the user
2705 * pointer of a cell which we should not have accessed, a segmentation
2706 * fault will let us know that this should not have happened:
2712 * triangulation.clear_user_data();
2716 * Next, allocate the quadrature objects that are within the responsibility
2717 * of this processor. This, of course, equals the number of cells that
2718 * belong to this processor times the number of quadrature points our
2719 * quadrature formula has on each cell. Since the `resize()` function does
2720 * not actually shrink the amount of allocated memory if the requested new
2721 * size is smaller than the old size, we resort to a trick to first free all
2722 * memory, and then reallocate it: we declare an empty vector as a temporary
2723 * variable and then swap the contents of the old vector and this temporary
2724 * variable. This makes sure that the `quadrature_point_history` is now
2725 * really empty, and we can let the temporary variable that now holds the
2726 * previous contents of the vector go out of scope and be destroyed. In the
2727 * next step we can then re-allocate as many elements as we need, with the
2728 * vector default-initializing the `PointHistory` objects, which includes
2729 * setting the stress variables to zero.
2733 * std::vector<PointHistory<dim>> tmp;
2734 * quadrature_point_history.swap(tmp);
2736 * quadrature_point_history.resize(
2737 * triangulation.n_locally_owned_active_cells() * quadrature_formula.size());
2741 * Finally loop over all cells again and set the user pointers from the
2742 * cells that belong to the present processor to point to the first
2743 * quadrature point objects corresponding to this cell in the vector of
2747 * unsigned int history_index = 0;
2748 * for (auto &cell : triangulation.active_cell_iterators())
2749 * if (cell->is_locally_owned())
2751 * cell->set_user_pointer(&quadrature_point_history[history_index]);
2752 * history_index += quadrature_formula.size();
2757 * At the end, for good measure make sure that our count of elements was
2758 * correct and that we have both used up all objects we allocated
2759 * previously, and not point to any objects beyond the end of the
2760 * vector. Such defensive programming strategies are always good checks to
2761 * avoid accidental errors and to guard against future changes to this
2762 * function that forget to update all uses of a variable at the same
2763 * time. Recall that constructs using the <code>Assert</code> macro are
2764 * optimized away in optimized mode, so do not affect the run time of
2768 * Assert(history_index == quadrature_point_history.size(),
2769 * ExcInternalError());
2777 * <a name="TopLevelupdate_quadrature_point_history"></a>
2778 * <h4>TopLevel::update_quadrature_point_history</h4>
2782 * At the end of each time step, we should have computed an incremental
2783 * displacement update so that the material in its new configuration
2784 * accommodates for the difference between the external body and boundary
2785 * forces applied during this time step minus the forces exerted through
2786 * preexisting internal stresses. In order to have the preexisting
2787 * stresses available at the next time step, we therefore have to update the
2788 * preexisting stresses with the stresses due to the incremental
2789 * displacement computed during the present time step. Ideally, the
2790 * resulting sum of internal stresses would exactly counter all external
2791 * forces. Indeed, a simple experiment can make sure that this is so: if we
2792 * choose boundary conditions and body forces to be time independent, then
2793 * the forcing terms (the sum of external forces and internal stresses)
2794 * should be exactly zero. If you make this experiment, you will realize
2795 * from the output of the norm of the right hand side in each time step that
2796 * this is almost the case: it is not exactly zero, since in the first time
2797 * step the incremental displacement and stress updates were computed
2798 * relative to the undeformed mesh, which was then deformed. In the second
2799 * time step, we again compute displacement and stress updates, but this
2800 * time in the deformed mesh -- there, the resulting updates are very small
2801 * but not quite zero. This can be iterated, and in each such iteration the
2802 * residual, i.e. the norm of the right hand side vector, is reduced; if one
2803 * makes this little experiment, one realizes that the norm of this residual
2804 * decays exponentially with the number of iterations, and after an initial
2805 * very rapid decline is reduced by roughly a factor of about 3.5 in each
2806 * iteration (for one testcase I looked at, other testcases, and other
2807 * numbers of unknowns change the factor, but not the exponential decay).
2811 * In a sense, this can then be considered as a quasi-timestepping scheme to
2812 * resolve the nonlinear problem of solving large-deformation elasticity on
2813 * a mesh that is moved along in a Lagrangian manner.
2817 * Another complication is that the existing (old) stresses are defined on
2818 * the old mesh, which we will move around after updating the stresses. If
2819 * this mesh update involves rotations of the cell, then we need to also
2820 * rotate the updated stress, since it was computed relative to the
2821 * coordinate system of the old cell.
2825 * Thus, what we need is the following: on each cell which the present
2826 * processor owns, we need to extract the old stress from the data stored
2827 * with each quadrature point, compute the stress update, add the two
2828 * together, and then rotate the result together with the incremental
2829 * rotation computed from the incremental displacement at the present
2830 * quadrature point. We will detail these steps below:
2833 * template <int dim>
2834 * void TopLevel<dim>::update_quadrature_point_history()
2838 * First, set up an <code>FEValues</code> object by which we will evaluate
2839 * the incremental displacements and the gradients thereof at the
2840 * quadrature points, together with a vector that will hold this
2844 * FEValues<dim> fe_values(fe,
2845 * quadrature_formula,
2846 * update_values | update_gradients);
2848 * std::vector<std::vector<Tensor<1, dim>>> displacement_increment_grads(
2849 * quadrature_formula.size(), std::vector<Tensor<1, dim>>(dim));
2853 * Then loop over all cells and do the job in the cells that belong to our
2857 * for (auto &cell : dof_handler.active_cell_iterators())
2858 * if (cell->is_locally_owned())
2862 * Next, get a pointer to the quadrature point history data local to
2863 * the present cell, and, as a defensive measure, make sure that
2864 * this pointer is within the bounds of the global array:
2867 * PointHistory<dim> *local_quadrature_points_history =
2868 * reinterpret_cast<PointHistory<dim> *>(cell->user_pointer());
2869 * Assert(local_quadrature_points_history >=
2870 * &quadrature_point_history.front(),
2871 * ExcInternalError());
2872 * Assert(local_quadrature_points_history <=
2873 * &quadrature_point_history.back(),
2874 * ExcInternalError());
2878 * Then initialize the <code>FEValues</code> object on the present
2879 * cell, and extract the gradients of the displacement at the
2880 * quadrature points for later computation of the strains
2883 * fe_values.reinit(cell);
2884 * fe_values.get_function_gradients(incremental_displacement,
2885 * displacement_increment_grads);
2889 * Then loop over the quadrature points of this cell:
2892 * for (unsigned int q = 0; q < quadrature_formula.size(); ++q)
2896 * On each quadrature point, compute the strain increment from
2897 * the gradients, and multiply it by the stress-strain tensor to
2898 * get the stress update. Then add this update to the already
2899 * existing strain at this point:
2902 * const SymmetricTensor<2, dim> new_stress =
2903 * (local_quadrature_points_history[q].old_stress +
2904 * (stress_strain_tensor *
2905 * get_strain(displacement_increment_grads[q])));
2909 * Finally, we have to rotate the result. For this, we first
2910 * have to compute a rotation matrix at the present quadrature
2911 * point from the incremental displacements. In fact, it can be
2912 * computed from the gradients, and we already have a function
2916 * const Tensor<2, dim> rotation =
2917 * get_rotation_matrix(displacement_increment_grads[q]);
2920 * Note that the result, a rotation matrix, is in general an
2921 * antisymmetric tensor of rank 2, so we must store it as a full
2926 * With this rotation matrix, we can compute the rotated tensor
2927 * by contraction from the left and right, after we expand the
2928 * symmetric tensor <code>new_stress</code> into a full tensor:
2931 * const SymmetricTensor<2, dim> rotated_new_stress =
2932 * symmetrize(transpose(rotation) *
2933 * static_cast<Tensor<2, dim>>(new_stress) * rotation);
2936 * Note that while the result of the multiplication of these
2937 * three matrices should be symmetric, it is not due to floating
2938 * point round off: we get an asymmetry on the order of 1e-16 of
2939 * the off-diagonal elements of the result. When assigning the
2940 * result to a <code>SymmetricTensor</code>, the constructor of
2941 * that class checks the symmetry and realizes that it isn't
2974 *
using namespace dealii;
2975 *
using namespace Step18;
2982 *
catch (std::exception &exc)
2984 *
std::cerr << std::endl
2986 *
<<
"----------------------------------------------------"
2988 *
std::cerr <<
"Exception on processing: " << std::endl
2989 *
<< exc.what() << std::endl
2990 *
<<
"Aborting!" << std::endl
2991 *
<<
"----------------------------------------------------"
2998 *
std::cerr << std::endl
3000 *
<<
"----------------------------------------------------"
3002 *
std::cerr <<
"Unknown exception!" << std::endl
3003 *
<<
"Aborting!" << std::endl
3004 *
<<
"----------------------------------------------------"
3148 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0002.0000.png"
3158 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0005.0000.png"
3168 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0007.0000.png"
3182 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0008.0000.png"
3192 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0009.0000.png"
3202 <
img src=
"https://www.dealii.org/images/steps/developer/step-18.sequential-0010.0000.png"
3228 Number
of active cells: 29696 (
by partition: 1808+1802+1894+1881+1870+1840+1884+1810+1876+1818+1870+1884+1854+1903+1816+1886)
3229 Number
of degrees
of freedom: 113100 (
by partition: 6936+6930+7305+7116+7326+6869+7331+6786+7193+6829+7093+7162+6920+7280+6843+7181)
3234 Number
of active cells: 102034 (
by partition: 6387+6202+6421+6341+6408+6201+6428+6428+6385+6294+6506+6244+6417+6527+6299+6546)
3235 Number
of degrees
of freedom: 359337 (
by partition: 23255+21308+24774+24019+22304+21415+22430+22184+22298+21796+22396+21592+22325+22553+21977+22711)
3267That's quite a good number of unknowns, given that we are in 3d. The output of
3268this program are 16 files for each time step:
3270\$ ls -l solution-0001*
3271-rw-r--r-- 1 wellsd2 user 761065 Feb 13 21:09 solution-0001.000.vtu
3272-rw-r--r-- 1 wellsd2 user 759277 Feb 13 21:09 solution-0001.001.vtu
3273-rw-r--r-- 1 wellsd2 user 761217 Feb 13 21:09 solution-0001.002.vtu
3274-rw-r--r-- 1 wellsd2 user 761605 Feb 13 21:09 solution-0001.003.vtu
3275-rw-r--r-- 1 wellsd2 user 756917 Feb 13 21:09 solution-0001.004.vtu
3276-rw-r--r-- 1 wellsd2 user 752669 Feb 13 21:09 solution-0001.005.vtu
3277-rw-r--r-- 1 wellsd2 user 735217 Feb 13 21:09 solution-0001.006.vtu
3278-rw-r--r-- 1 wellsd2 user 750065 Feb 13 21:09 solution-0001.007.vtu
3279-rw-r--r-- 1 wellsd2 user 760273 Feb 13 21:09 solution-0001.008.vtu
3280-rw-r--r-- 1 wellsd2 user 777265 Feb 13 21:09 solution-0001.009.vtu
3281-rw-r--r-- 1 wellsd2 user 772469 Feb 13 21:09 solution-0001.010.vtu
3282-rw-r--r-- 1 wellsd2 user 760833 Feb 13 21:09 solution-0001.011.vtu
3283-rw-r--r-- 1 wellsd2 user 782241 Feb 13 21:09 solution-0001.012.vtu
3284-rw-r--r-- 1 wellsd2 user 748905 Feb 13 21:09 solution-0001.013.vtu
3285-rw-r--r-- 1 wellsd2 user 738413 Feb 13 21:09 solution-0001.014.vtu
3286-rw-r--r-- 1 wellsd2 user 762133 Feb 13 21:09 solution-0001.015.vtu
3287-rw-r--r-- 1 wellsd2 user 1421 Feb 13 21:09 solution-0001.pvtu
3288-rw-r--r-- 1 wellsd2 user 364 Feb 13 21:09 solution-0001.visit
3292Here are first the mesh on which we compute as well as the partitioning
3293for the 16 processors:
3296<div class="twocolumn" style="width: 80%">
3297 <div class="parent">
3298 <div class="img" align="center">
3299 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-000mesh.png"
3300 alt="Discretization"
3303 <div class="text" align="center">
3307 <div class="parent">
3308 <div class="img" align="center">
3309 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0002.p.png"
3310 alt="Parallel partitioning"
3313 <div class="text" align="center">
3314 Parallel partitioning
3320Finally, here is the same output as we have shown before for the much smaller
3323<div class="threecolumn" style="width: 80%">
3324 <div class="parent">
3325 <div class="img" align="center">
3326 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0002.s.png"
3330 <div class="text" align="center">
3334 <div class="parent">
3335 <div class="img" align="center">
3336 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0005.s.png"
3340 <div class="text" align="center">
3344 <div class="parent">
3345 <div class="img" align="center">
3346 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0007.s.png"
3350 <div class="text" align="center">
3357<div class="threecolumn" style="width: 80%">
3358 <div class="parent">
3359 <div class="img" align="center">
3360 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0008.s.png"
3364 <div class="text" align="center">
3368 <div class="parent">
3369 <div class="img" align="center">
3370 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0009.s.png"
3374 <div class="text" align="center">
3378 <div class="parent">
3379 <div class="img" align="center">
3380 <img src="https://www.dealii.org/images/steps/developer/step-18.parallel-0010.s.png"
3384 <div class="text" align="center">
3391As before, we observe that at high axial compression the cylinder begins
3392to buckle, but this time ultimately collapses on itself. In contrast to our
3393first run, towards the end of the simulation the deflection pattern becomes
3394nonsymmetric (the central bulge deflects laterally). The model clearly does not
3395provide for this (all our forces and boundary deflections are symmetric) but the
3396effect is probably physically correct anyway: in reality, small inhomogeneities
3409the present one. One would need an even finer computation to find out. However,
3410the point may be moot: looking at the last picture in detail, it is pretty
3411obvious that not only is the linear small deformation model we chose completely
3412inadequate, but for a realistic simulation we would also need to make sure that
3413the body does not intersect itself during deformation (if we continued
3414compressing the cylinder we would observe some self-intersection).
3415Without such a formulation we cannot expect anything to make physical sense,
3416even if it produces nice pictures!
3419<a name="Possibilitiesforextensions"></a><h3>Possibilities for extensions</h3>
3422The program as is does not really solve an equation that has many applications
3423in practice: quasi-static material deformation based on a purely elastic law
3424is almost boring. However, the program may serve as the starting point for
3425more interesting experiments, and that indeed was the initial motivation for
3426writing it. Here are some suggestions of what the program is missing and in
3427what direction it may be extended:
3429<a name="Plasticitymodels"></a><h5>Plasticity models</h5>
3432 The most obvious extension is to use a more
3433realistic material model for large-scale quasistatic deformation. The natural
3434choice for this would be plasticity, in which a nonlinear relationship between
3435stress and strain replaces equation <a href="#step_18.stress-strain">[stress-strain]</a>. Plasticity
3436models are usually rather complicated to program since the stress-strain
3437dependence is generally non-smooth. The material can be thought of being able
3438to withstand only a maximal stress (the yield stress) after which it starts to
3439“flow”. A mathematical description to this can be given in the form of a
3440variational inequality, which alternatively can be treated as minimizing the
3444 (\varepsilon(\mathbf{u}), C\varepsilon(\mathbf{u}))_{\Omega}
3445 - (\mathbf{f}, \mathbf{u})_{\Omega} - (\mathbf{b}, \mathbf{u})_{\Gamma_N},
3447subject to the constraint
3449 f(\sigma(\mathbf{u})) \le 0
3451on the stress. This extension makes the problem to be solved in each time step
3452nonlinear, so we need another loop within each time step.
3454Without going into further details of this model, we refer to the excellent
3455book by Simo and Hughes on “Computational Inelasticity” for a
3456comprehensive overview of computational strategies for solving plastic
3457models. Alternatively, a brief but concise description of an algorithm for
3458plasticity is given in an article by S. Commend, A. Truty, and Th. Zimmermann;
3462<a name="Stabilizationissues"></a><h5>Stabilization issues</h5>
3465The formulation we have chosen, i.e. using
3466piecewise (bi-, tri-)linear elements for all components of the displacement
3467vector, and treating the stress as a variable dependent on the displacement is
3468appropriate for most materials. However, this so-called displacement-based
3469formulation becomes unstable and exhibits spurious modes for incompressible or
3470nearly-incompressible materials. While fluids are usually not elastic (in most
3471cases, the stress depends on velocity gradients, not displacement gradients,
3472although there are exceptions such as electro-rheologic fluids), there are a
3473few solids that are nearly incompressible, for example rubber. Another case is
3474that many plasticity models ultimately let the material become incompressible,
3475although this is outside the scope of the present program.
3477Incompressibility is characterized by Poisson's
ratio
3493<a name=
"Refinementduringtimesteps"></a><
h5>Refinement
during timesteps</
h5>
3522 per cell in 2d
and 8 in 3d.
3535 in
the quadrature points.)
3546 for (
unsigned int i=0; i<dim; ++i)
3547 for (
unsigned int j=0;
j<dim; ++
j)
3558 quadrature, quadrature,
3576 for (
unsigned int i=0; i<dim; ++i)
3577 for (
unsigned int j=0;
j<dim; ++
j)
3579 for (
unsigned int q=0;
q<quadrature.
size(); ++
q)
3621 for (
unsigned int i=0; i<dim; ++i)
3622 for (
unsigned int j=0;
j<dim; ++
j)
3630 for (
unsigned int q=0;
q<quadrature.
size(); ++
q)
3659<a name="PlainProg"></a>
3660<h1> The plain program</h1>
3661@include "step-18.cc"
value_type * data() const noexcept
void reinit(value_type *starting_element, const std::size_t n_elements)
active_cell_iterator begin_active(const unsigned int level=0) const
__global__ void reduction(Number *result, const Number *v, const size_type N)
__global__ void set(Number *val, const Number s, const size_type N)
#define Assert(cond, exc)
#define AssertDimension(dim1, dim2)
static ::ExceptionBase & ExcInternalError()
typename ActiveSelector::active_cell_iterator active_cell_iterator
void loop(ITERATOR begin, std_cxx20::type_identity_t< ITERATOR > end, DOFINFO &dinfo, INFOBOX &info, const std::function< void(DOFINFO &, typename INFOBOX::CellInfo &)> &cell_worker, const std::function< void(DOFINFO &, typename INFOBOX::CellInfo &)> &boundary_worker, const std::function< void(DOFINFO &, DOFINFO &, typename INFOBOX::CellInfo &, typename INFOBOX::CellInfo &)> &face_worker, ASSEMBLER &assembler, const LoopControl &lctrl=LoopControl())
@ update_values
Shape function values.
@ update_JxW_values
Transformed quadrature weights.
@ update_gradients
Shape function gradients.
@ update_quadrature_points
Transformed quadrature points.
void approximate(SynchronousIterators< std::tuple< typename DoFHandler< dim, spacedim >::active_cell_iterator, Vector< float >::iterator > > const &cell, const Mapping< dim, spacedim > &mapping, const DoFHandler< dim, spacedim > &dof_handler, const InputVector &solution, const unsigned int component)
void cylinder(Triangulation< dim > &tria, const double radius=1., const double half_length=1.)
void cylinder_shell(Triangulation< dim > &tria, const double length, const double inner_radius, const double outer_radius, const unsigned int n_radial_cells=0, const unsigned int n_axial_cells=0)
void refine(Triangulation< dim, spacedim > &tria, const Vector< Number > &criteria, const double threshold, const unsigned int max_to_mark=numbers::invalid_unsigned_int)
@ valid
Iterator points to a valid object.
@ matrix
Contents is actually a matrix.
@ symmetric
Matrix is symmetric.
@ general
No special properties.
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)
Point< spacedim > point(const gp_Pnt &p, const double tolerance=1e-10)
SymmetricTensor< 2, dim, Number > C(const Tensor< 2, dim, Number > &F)
Tensor< 2, dim, Number > l(const Tensor< 2, dim, Number > &F, const Tensor< 2, dim, Number > &dF_dt)
SymmetricTensor< 2, dim, Number > d(const Tensor< 2, dim, Number > &F, const Tensor< 2, dim, Number > &dF_dt)
constexpr ReturnType< rank, T >::value_type & extract(T &t, const ArrayType &indices)
void call(const std::function< RT()> &function, internal::return_value< RT > &ret_val)
unsigned int n_mpi_processes(const MPI_Comm mpi_communicator)
unsigned int this_mpi_process(const MPI_Comm mpi_communicator)
std::string compress(const std::string &input)
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 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 > sqrt(const ::VectorizedArray< Number, width > &)
const ::parallel::distributed::Triangulation< dim, spacedim > * triangulation
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)