417 *
vector_value_list(
const std::vector<
Point<dim>> &points,
422 *
for (
unsigned int p = 0; p < points.size(); ++p)
430 * <a name=
"ThecodeParameterReadercodeclass"></a>
449 *
void declare_parameters();
466 * <a name=
"codeParameterReaderdeclare_parameterscode"></a>
467 * <
h4><
code>ParameterReader::declare_parameters</
code></
h4>
479 *
void ParameterReader::declare_parameters()
492 *
prm.enter_subsection(
"Mesh & geometry parameters");
494 *
prm.declare_entry(
"Number of refinements",
497 *
"Number of global mesh refinement steps "
498 *
"applied to initial coarse grid");
500 *
prm.declare_entry(
"Focal distance",
503 *
"Distance of the focal point of the lens "
506 *
prm.leave_subsection();
517 *
prm.enter_subsection(
"Physical constants");
523 *
prm.leave_subsection();
533 *
prm.enter_subsection(
"Output parameters");
535 *
prm.declare_entry(
"Output filename",
538 *
"Name of the output file (without extension)");
557 *
template-parameter-
dependent class.)
To find out what parameters
568 *
prm.leave_subsection();
574 * <a name=
"codeParameterReaderread_parameterscode"></a>
588 *
declare_parameters();
598 * <a name=
"ThecodeComputeIntensitycodeclass"></a>
669 * compute @
f$|
u|@
f$,
so we're good with the update_values flag.
673 * ComputeIntensity<dim>::ComputeIntensity()
674 * : DataPostprocessorScalar<dim>("Intensity", update_values)
680 * The actual postprocessing happens in the following function. Its input is
681 * an object that stores values of the function (which is here vector-valued)
682 * representing the data vector given to DataOut::add_data_vector, evaluated
683 * at all evaluation points where we generate output, and some tensor objects
719 *
const std::complex<double>
u(
inputs.solution_values[p](0),
720 *
inputs.solution_values[p](1));
730 * <a name=
"ThecodeUltrasoundProblemcodeclass"></a>
789 * <a name=
"codeUltrasoundProblemmake_gridcode"></a>
808 *
std::cout <<
"Generating grid... ";
818 *
prm.enter_subsection(
"Mesh & geometry parameters");
821 *
const unsigned int n_refinements = prm.get_integer(
"Number of refinements");
823 *
prm.leave_subsection();
861 *
if (face->at_boundary() &&
864 *
face->set_boundary_id(1);
865 *
face->set_manifold_id(1);
893 *
std::cout <<
"done (" << timer.cpu_time() <<
"s)" << std::endl;
895 *
std::cout <<
" Number of active cells: " <<
triangulation.n_active_cells()
903 * <a name=
"codeUltrasoundProblemsetup_systemcode"></a>
917 *
std::cout <<
"Setting up system... ";
920 *
dof_handler.distribute_dofs(fe);
924 *
sparsity_pattern.copy_from(
dsp);
926 *
system_matrix.
reinit(sparsity_pattern);
928 *
solution.
reinit(dof_handler.n_dofs());
931 *
std::cout <<
"done (" << timer.cpu_time() <<
"s)" << std::endl;
933 *
std::cout <<
" Number of degrees of freedom: " << dof_handler.n_dofs()
941 * <a name=
"codeUltrasoundProblemassemble_systemcode"></a>
947 * right
hand side vector:
953 *
std::cout <<
"Assembling system matrix... ";
966 *
prm.enter_subsection(
"Physical constants");
968 *
const double omega = prm.get_double(
"omega"), c = prm.get_double(
"c");
970 *
prm.leave_subsection();
985 *
dofs_per_cell = fe.n_dofs_per_cell();
991 * need the values and gradients of the shape functions, and of course the
992 * quadrature weights. For the terms involving the boundary integrals,
993 * only shape function values and the quadrature weights are necessary.
996 * FEValues<dim> fe_values(fe,
997 * quadrature_formula,
998 * update_values | update_gradients |
999 * update_JxW_values);
1001 * FEFaceValues<dim> fe_face_values(fe,
1002 * face_quadrature_formula,
1003 * update_values | update_JxW_values);
1007 * As usual, the system matrix is assembled cell by cell, and we need a
1008 * matrix for storing the local cell contributions as well as an index
1009 * vector to transfer the cell contributions to the appropriate location
1010 * in the global system matrix after.
1013 * FullMatrix<double> cell_matrix(dofs_per_cell, dofs_per_cell);
1014 * std::vector<types::global_dof_index> local_dof_indices(dofs_per_cell);
1016 * for (const auto &cell : dof_handler.active_cell_iterators())
1020 * On each cell, we first need to reset the local contribution matrix
1021 * and request the FEValues object to compute the shape functions for
1026 * fe_values.reinit(cell);
1028 * for (unsigned int i = 0; i < dofs_per_cell; ++i)
1030 * for (unsigned int j = 0; j < dofs_per_cell; ++j)
1034 * At this point, it is important to keep in mind that we are
1035 * dealing with a finite element system with two
1036 * components. Due to the way we constructed this FESystem,
1037 * namely as the Cartesian product of two scalar finite
1038 * element fields, each shape function has only a single
1039 * nonzero component (they are, in deal.II lingo, @ref
1040 * GlossPrimitive "primitive"). Hence, each shape function
1041 * can be viewed as one of the @f$\phi@f$'s
or @
f$\psi@
f$'s from the
1042 * introduction, and similarly the corresponding degrees of
1043 * freedom can be attributed to either @f$\alpha@f$ or @f$\beta@f$.
1044 * As we iterate through all the degrees of freedom on the
1045 * current cell however, they do not come in any particular
1046 * order, and so we cannot decide right away whether the DoFs
1047 * with index @f$i@f$ and @f$j@f$ belong to the real or imaginary part
1048 * of our solution. On the other hand, if you look at the
1049 * form of the system matrix in the introduction, this
1050 * distinction is crucial since it will determine to which
1051 * block in the system matrix the contribution of the current
1052 * pair of DoFs will go and hence which quantity we need to
1053 * compute from the given two shape functions. Fortunately,
1054 * the FESystem object can provide us with this information,
1055 * namely it has a function
1056 * FESystem::system_to_component_index, that for each local
1057 * DoF index returns a pair of integers of which the first
1058 * indicates to which component of the system the DoF
1059 * belongs. The second integer of the pair indicates which
1060 * index the DoF has in the scalar base finite element field,
1061 * but this information is not relevant here. If you want to
1062 * know more about this function and the underlying scheme
1063 * behind primitive vector valued elements, take a look at
1064 * @ref step_8 "step-8" or the @ref vector_valued module, where these topics
1065 * are explained in depth.
1068 * if (fe.system_to_component_index(i).first ==
1069 * fe.system_to_component_index(j).first)
1073 * If both DoFs @f$i@f$ and @f$j@f$ belong to same component,
1074 * i.e. their shape functions are both @f$\phi@f$'s
or both
1075 * @
f$\psi@
f$'s, the contribution will end up in one of the
1076 * diagonal blocks in our system matrix, and since the
1077 * corresponding entries are computed by the same formula,
1078 * we do not bother if they actually are @f$\phi@f$ or @f$\psi@f$
1079 * shape functions. We can simply compute the entry by
1080 * iterating over all quadrature points and adding up
1081 * their contributions, where values and gradients of the
1082 * shape functions are supplied by our FEValues object.
1088 * for (unsigned int q_point = 0; q_point < n_q_points;
1090 * cell_matrix(i, j) +=
1091 * (((fe_values.shape_value(i, q_point) *
1092 * fe_values.shape_value(j, q_point)) *
1093 * (-omega * omega) +
1094 * (fe_values.shape_grad(i, q_point) *
1095 * fe_values.shape_grad(j, q_point)) *
1097 * fe_values.JxW(q_point));
1101 * You might think that we would have to specify which
1102 * component of the shape function we'd like to evaluate
1125 *
for (
const auto face_no : cell->face_indices())
1126 *
if (cell->face(
face_no)->at_boundary() &&
1147 *
for (
unsigned int i = 0; i < dofs_per_cell; ++i)
1148 *
for (
unsigned int j = 0;
j < dofs_per_cell; ++
j)
1149 *
if ((fe.system_to_component_index(i).first !=
1150 *
fe.system_to_component_index(
j).first) &&
1151 *
fe.has_support_on_face(i,
face_no) &&
1152 *
fe.has_support_on_face(
j,
face_no))
1157 * it we would simply add up terms to the local cell
1158 * matrix that happen to be zero because at least one of
1159 * the shape functions happens to be zero. However, we can
1160 * save that work by adding the checks above.
1164 * In either case, these DoFs will contribute to the
1165 * boundary integrals in the off-diagonal blocks of the
1166 * system matrix. To compute the integral, we loop over
1167 * all the quadrature points on the face and sum up the
1168 * contribution weighted with the quadrature weights that
1169 * the face quadrature rule provides. In contrast to the
1170 * entries on the diagonal blocks, here it does matter
1171 * which one of the shape functions is a @f$\psi@f$ and which
1172 * one is a @f$\phi@f$, since that will determine the sign of
1173 * the entry. We account for this by a simple conditional
1174 * statement that determines the correct sign. Since we
1175 * already checked that DoF @f$i@f$ and @f$j@f$ belong to
1176 * different components, it suffices here to test for one
1177 * of them to which component it belongs.
1180 * for (unsigned int q_point = 0; q_point < n_face_q_points;
1182 * cell_matrix(i, j) +=
1183 * ((fe.system_to_component_index(i).first == 0) ? -1 :
1185 * fe_face_values.shape_value(i, q_point) *
1186 * fe_face_values.shape_value(j, q_point) * c * omega *
1187 * fe_face_values.JxW(q_point);
1192 * Now we are done with this cell and have to transfer its
1193 * contributions from the local to the global system matrix. To this
1194 * end, we first get a list of the global indices of the this cells
1198 * cell->get_dof_indices(local_dof_indices);
1203 * ...and then add the entries to the system matrix one by one:
1206 * for (unsigned int i = 0; i < dofs_per_cell; ++i)
1207 * for (unsigned int j = 0; j < dofs_per_cell; ++j)
1208 * system_matrix.add(local_dof_indices[i],
1209 * local_dof_indices[j],
1210 * cell_matrix(i, j));
1216 * The only thing left are the Dirichlet boundary values on @f$\Gamma_1@f$,
1217 * which is characterized by the boundary indicator 1. The Dirichlet
1218 * values are provided by the <code>DirichletBoundaryValues</code> class
1222 * std::map<types::global_dof_index, double> boundary_values;
1223 * VectorTools::interpolate_boundary_values(dof_handler,
1225 * DirichletBoundaryValues<dim>(),
1228 * MatrixTools::apply_boundary_values(boundary_values,
1234 * std::cout << "done (" << timer.cpu_time() << "s)" << std::endl;
1242 * <a name="codeUltrasoundProblemsolvecode"></a>
1243 * <h4><code>UltrasoundProblem::solve</code></h4>
1247 * As already mentioned in the introduction, the system matrix is neither
1248 * symmetric nor definite, and so it is not quite obvious how to come up
1249 * with an iterative solver and a preconditioner that do a good job on this
1250 * matrix. (For more on this topic, see also the
1251 * <a href="#extensions">Possibilities for extensions</a> section below.)
1252 * We chose instead to go a different way and solve the linear
1253 * system with the sparse LU decomposition provided by UMFPACK. This is
1254 * often a good first choice for 2d problems and works reasonably well even
1255 * for moderately large numbers of DoFs. The deal.II interface to UMFPACK
1256 * is implemented in the SparseDirectUMFPACK class, which is very easy to
1257 * use and allows us to solve our linear system with just 3 lines of code.
1261 * Note again that for compiling this example program, you need to have the
1262 * deal.II library built with UMFPACK support.
1265 * template <int dim>
1266 * void UltrasoundProblem<dim>::solve()
1268 * std::cout << "Solving linear system... ";
1273 * The code to solve the linear system is short: First, we allocate an
1274 * object of the right type. The following call to
1275 * SparseDirectUMFPACK::solve() takes as argument the matrix to decompose,
1276 * and a vector that upon input equals the right hand side of the linear
1277 * system to be solved, and upon output contains the solution of the linear
1278 * system. To satisfy this input/output requirement, we first assign the
1279 * right hand side vector to the `solution` variable.
1282 * SparseDirectUMFPACK A_direct;
1284 * solution = system_rhs;
1285 * A_direct.solve(system_matrix, solution);
1288 * std::cout << "done (" << timer.cpu_time() << "s)" << std::endl;
1296 * <a name="codeUltrasoundProblemoutput_resultscode"></a>
1297 * <h4><code>UltrasoundProblem::output_results</code></h4>
1301 * Here we output our solution @f$v@f$ and @f$w@f$ as well as the derived quantity
1302 * @f$|u|@f$ in the format specified in the parameter file. Most of the work for
1303 * deriving @f$|u|@f$ from @f$v@f$ and @f$w@f$ was already done in the implementation of
1304 * the <code>ComputeIntensity</code> class, so that the output routine is
1305 * rather straightforward and very similar to what is done in the previous
1309 * template <int dim>
1310 * void UltrasoundProblem<dim>::output_results() const
1312 * std::cout << "Generating output... ";
1317 * Define objects of our <code>ComputeIntensity</code> class and a DataOut
1321 * ComputeIntensity<dim> intensities;
1322 * DataOut<dim> data_out;
1324 * data_out.attach_dof_handler(dof_handler);
1328 * Next we query the output-related parameters from the ParameterHandler.
1329 * The DataOut::parse_parameters call acts as a counterpart to the
1330 * DataOutInterface<1>::declare_parameters call in
1331 * <code>ParameterReader::declare_parameters</code>. It collects all the
1332 * output format related parameters from the ParameterHandler and sets the
1333 * corresponding properties of the DataOut object accordingly.
1336 * prm.enter_subsection("Output parameters");
1338 * const std::string output_filename = prm.get("Output filename");
1339 * data_out.parse_parameters(prm);
1341 * prm.leave_subsection();
1345 * Now we put together the filename from the base name provided by the
1346 * ParameterHandler and the suffix which is provided by the DataOut class
1347 * (the default suffix is set to the right type that matches the one set
1348 * in the .prm file through parse_parameters()):
1351 * const std::string filename = output_filename + data_out.default_suffix();
1353 * std::ofstream output(filename);
1357 * The solution vectors @f$v@f$ and @f$w@f$ are added to the DataOut object in the
1361 * std::vector<std::string> solution_names;
1362 * solution_names.emplace_back("Re_u");
1363 * solution_names.emplace_back("Im_u");
1365 * data_out.add_data_vector(solution, solution_names);
1369 * For the intensity, we just call <code>add_data_vector</code> again, but
1370 * this with our <code>ComputeIntensity</code> object as the second
1371 * argument, which effectively adds @f$|u|@f$ to the output data:
1374 * data_out.add_data_vector(solution, intensities);
1378 * The last steps are as before. Note that the actual output format is now
1379 * determined by what is stated in the input file, i.e. one can change the
1380 * output format without having to re-compile this program:
1383 * data_out.build_patches();
1384 * data_out.write(output);
1387 * std::cout << "done (" << timer.cpu_time() << "s)" << std::endl;
1395 * <a name="codeUltrasoundProblemruncode"></a>
1396 * <h4><code>UltrasoundProblem::run</code></h4>
1400 * Here we simply execute our functions one after the other:
1403 * template <int dim>
1404 * void UltrasoundProblem<dim>::run()
1408 * assemble_system();
1412 * } // namespace Step29
1418 * <a name="Thecodemaincodefunction"></a>
1419 * <h4>The <code>main</code> function</h4>
1423 * Finally the <code>main</code> function of the program. It has the same
1424 * structure as in almost all of the other tutorial programs. The only
1425 * exception is that we define ParameterHandler and
1426 * <code>ParameterReader</code> objects, and let the latter read in the
1427 * parameter values from a textfile called <code>@ref step_29 "step-29".prm</code>. The
1428 * values so read are then handed over to an instance of the UltrasoundProblem
1436 * using namespace dealii;
1437 * using namespace Step29;
1439 * ParameterHandler prm;
1440 * ParameterReader param(prm);
1441 * param.read_parameters("step-29.prm");
1443 * UltrasoundProblem<2> ultrasound_problem(prm);
1444 * ultrasound_problem.run();
1446 * catch (std::exception &exc)
1448 * std::cerr << std::endl
1450 * << "----------------------------------------------------"
1452 * std::cerr << "Exception on processing: " << std::endl
1453 * << exc.what() << std::endl
1454 * << "Aborting!" << std::endl
1455 * << "----------------------------------------------------"
1461 * std::cerr << std::endl
1463 * << "----------------------------------------------------"
1465 * std::cerr << "Unknown exception!" << std::endl
1466 * << "Aborting!" << std::endl
1467 * << "----------------------------------------------------"
1474<a name="Results"></a>
1475<a name="Results"></a><h1>Results</h1>
1478The current program reads its run-time parameters from an input file
1479called <code>step-29.prm</code> that looks like this:
1481subsection Mesh & geometry parameters
1482 # Distance of the focal point of the lens to the x-axis
1483 set Focal distance = 0.3
1485 # Number of global mesh refinement steps applied to initial coarse grid
1486 set Number of refinements = 5
1490subsection Physical constants
1499subsection Output parameters
1500 # Name of the output file (without extension)
1501 set Output file = solution
1503 # A name for the output format to be used
1504 set Output format = vtu
1508As can be seen, we set
1509@f$d=0.3@f$, which amounts to a focus of the transducer lens
1510at @f$x=0.5@f$, @f$y=0.3@f$. The coarse mesh is refined 5 times,
1511resulting in 160x160 cells, and the output is written in vtu
1512format. The parameter reader understands many more parameters
1513pertaining in particular to the generation of output, but we
1514need none of these parameters here and therefore stick with
1515their default values.
1558<table
align=
"center" class=
"doxtable">
1561 <
img src=
"https://www.dealii.org/images/steps/developer/step-29.v.png" alt=
"v = Re(u)">
1564 <
img src=
"https://www.dealii.org/images/steps/developer/step-29.w.png" alt=
"w = Im(u)">
1569 <
img src=
"https://www.dealii.org/images/steps/developer/step-29.intensity.png" alt=
"|u|">
1586<table
align=
"center" class=
"doxtable">
1589 <
img src=
"https://www.dealii.org/images/steps/developer/step-29.surface.png" alt=
"|u|">
1592 <
img src=
"https://www.dealii.org/images/steps/developer/step-29.contours.png" alt=
"|u|">
1605[ 66%] Built target @ref step_29 "step-29"
1606[100%] Run @ref step_29 "step-29" with Release configuration
1607DEAL::Generating grid... done (0.0135260s)
1608DEAL:: Number of active cells: 25600
1609DEAL::Setting up system... done (0.0213910s)
1610DEAL:: Number of degrees of freedom: 51842
1611DEAL::Assembling system matrix... done (0.0414300s)
1612DEAL::Solving linear system... done (1.56621s)
1613DEAL::Generating output... done (0.729605s)
1614[100%] Built target run
1617[ 66%] Built target @ref step_29 "step-29"
1618[100%] Run @ref step_29 "step-29" with Release configuration
1619DEAL::Generating grid... done (0.0668490s)
1620DEAL:: Number of active cells: 102400
1621DEAL::Setting up system... done (0.109694s)
1622DEAL:: Number of degrees of freedom: 206082
1623DEAL::Assembling system matrix... done (0.160784s)
1624DEAL::Solving linear system... done (7.86577s)
1625DEAL::Generating output... done (2.89320s)
1626[100%] Built target run
1629[ 66%] Built target @ref step_29 "step-29"
1630[100%] Run @ref step_29 "step-29" with Release configuration
1631DEAL::Generating grid... done (0.293154s)
1632DEAL:: Number of active cells: 409600
1633DEAL::Setting up system... done (0.491301s)
1634DEAL:: Number of degrees of freedom: 821762
1635DEAL::Assembling system matrix... done (0.605386s)
1636DEAL::Solving linear system... done (45.1989s)
1637DEAL::Generating output... done (11.2292s)
1638[100%] Built target run
1641Each time we refine the mesh once, so the number of cells and degrees
1642of freedom roughly quadruples from each step to the next. As can be seen,
1643generating the grid, setting up degrees of freedom, assembling the
1644linear system, and generating output scale pretty closely to linear,
1645whereas solving the linear system is an operation that requires 8
1646times more time each time the number of degrees of freedom is
1647increased by a factor of 4, i.e. it is @f${\cal O}(N^{3/2})@f$. This can
1648be explained by the fact that (using optimal ordering) the
1649bandwidth of a finite element matrix is @f$B={\cal O}(N^{(dim-1)/dim})@f$,
1650and the effort to solve a banded linear system using LU decomposition
1651is @f${\cal O}(BN)@f$. This also explains why the program does run in 3d
1652as well (after changing the dimension on the
1653<code>UltrasoundProblem</code> object), but scales very badly and
1654takes extraordinary patience before it finishes solving the linear
1655system on a mesh with appreciable resolution, even though all the
1656other parts of the program scale very nicely.
1660<a name="extensions"></a>
1661<a name="Possibilitiesforextensions"></a><h3>Possibilities for extensions</h3>
1664An obvious possible extension for this program is to run it in 3d
1665— after all, the world around us is three-dimensional, and
1666ultrasound beams propagate in three-dimensional media. You can try
1667this by simply changing the template parameter of the principal class
1668in <code>main()</code> and running it. This won't get
you very far,
1670the parameter file.
You'll simply run out of memory as both the mesh
1671(with its @f$(2^5)^3 \cdot 5^3=2^{15}\cdot 125 \approx 4\cdot 10^6@f$ cells)
1672and in particular the sparse direct solver take too much memory. You
1673can solve with 3 global refinement steps, however, if you have a bit
1674of time: in early 2011, the direct solve takes about half an
1677solution
's waves accurately, and you can see this in plots of the
1678solution. Consequently, this is one of the cases where adaptivity is
1683<a name=
"PlainProg"></a>
value_type * data() const noexcept
void reinit(value_type *starting_element, const std::size_t n_elements)
static void declare_parameters(ParameterHandler &prm)
void add_data_vector(const VectorType &data, const std::vector< std::string > &names, const DataVectorType type=type_automatic, const std::vector< DataComponentInterpretation::DataComponentInterpretation > &data_component_interpretation={})
virtual void evaluate_vector_field(const DataPostprocessorInputs::Vector< dim > &input_data, std::vector< Vector< double > > &computed_quantities) const
virtual void evaluate_scalar_field(const DataPostprocessorInputs::Scalar< dim > &input_data, std::vector< Vector< double > > &computed_quantities) const
__global__ void set(Number *val, const Number s, const size_type N)
#define AssertDimension(dim1, dim2)
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())
void make_sparsity_pattern(const DoFHandler< dim, spacedim > &dof_handler, SparsityPatternBase &sparsity_pattern, const AffineConstraints< number > &constraints=AffineConstraints< number >(), const bool keep_constrained_dofs=true, const types::subdomain_id subdomain_id=numbers::invalid_subdomain_id)
@ update_hessians
Second derivatives of shape functions.
@ update_values
Shape function values.
@ update_normal_vectors
Normal vectors.
@ update_gradients
Shape function gradients.
void subdivided_hyper_cube(Triangulation< dim, spacedim > &tria, const unsigned int repetitions, const double left=0., const double right=1., const bool colorize=false)
@ matrix
Contents is actually a matrix.
@ diagonal
Matrix is diagonal.
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 > w(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)
void call(const std::function< RT()> &function, internal::return_value< RT > &ret_val)
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)
bool check(const ConstraintKinds kind_in, const unsigned int dim)
int(&) functions(const void *v1, const void *v2)
::VectorizedArray< Number, width > sqrt(const ::VectorizedArray< Number, width > &)
::VectorizedArray< Number, width > abs(const ::VectorizedArray< Number, width > &)
const ::parallel::distributed::Triangulation< dim, spacedim > * triangulation