40 face_midpoint(
const unsigned int face_no)
43 const auto face_reference_cell =
47 for (
const auto face_vertex_no : face_reference_cell.vertex_indices())
55 midpoint /= reference_cell.face_reference_cell(0).n_vertices();
63 std::vector<unsigned int>
64 get_dpo_vector_fe_p(
const unsigned int dim,
const unsigned int degree)
117 std::vector<Point<dim>>
118 unit_support_points_fe_p(
const unsigned int degree)
122 std::vector<Point<dim>> unit_points;
129 unit_points.emplace_back(
reference_cell.template barycenter<dim>());
134 const auto dpo = get_dpo_vector_fe_p(dim,
degree);
147 for (
unsigned int p = 0; p < dpo[1]; ++p)
148 unit_points.push_back((
double(dpo[1] - p) / (dpo[1] + 1)) * p0 +
149 (
double(p + 1) / (dpo[1] + 1)) * p1);
152 if (dim > 1 && dpo[2] > 0)
159 unit_points.push_back(face_midpoint<dim>(f));
166 std::vector<Point<0>>
167 unit_support_points_fe_p(
const unsigned int )
177 std::vector<std::vector<
Point<dim - 1>>>
178 unit_face_support_points_fe_p(
179 const unsigned int degree,
190 std::vector<std::vector<
Point<dim - 1>>> unit_face_points;
196 unit_face_points.emplace_back(
197 unit_support_points_fe_p<dim - 1>(degree));
200 return unit_face_points;
210 constraints_fe_p(
const unsigned int )
219 constraints_fe_p<2>(
const unsigned int degree)
221 constexpr int dim = 2;
229 std::vector<
Point<dim - 1>> constraint_points;
231 constraint_points.emplace_back(0.5);
233 for (
unsigned int i = 1; i < degree; ++i)
234 constraint_points.push_back(
238 for (
unsigned int i = 1; i < degree; ++i)
239 constraint_points.push_back(
246 const unsigned int n_dofs_constrained = constraint_points.size();
247 unsigned int n_dofs_per_face = degree + 1;
253 for (
unsigned int i = 0; i < n_dofs_constrained; ++i)
254 for (
unsigned int j = 0; j < n_dofs_per_face; ++j)
256 interface_constraints(i, j) =
257 poly.compute_value(j, constraint_points[i]);
263 if (std::fabs(interface_constraints(i, j)) < 1e-13)
264 interface_constraints(i, j) = 0;
266 return interface_constraints;
275 std::vector<unsigned int>
276 get_dpo_vector_fe_dgp(
const unsigned int dim,
const unsigned int degree)
281 std::vector<unsigned int> dpo(dim + 1, 0U);
292 const auto continuous_dpo = get_dpo_vector_fe_p(dim, degree);
299 continuous_dpo[dim]};
307 continuous_dpo[dim]};
317 continuous_dpo[dim]};
328template <
int dim,
int spacedim>
332 const bool prolongation_is_additive,
350template <
int dim,
int spacedim>
351std::pair<Table<2, bool>, std::vector<unsigned int>>
355 constant_modes.fill(
true);
356 return std::pair<Table<2, bool>, std::vector<unsigned int>>(
357 constant_modes, std::vector<unsigned int>(1, 0));
362template <
int dim,
int spacedim>
365 const unsigned int child,
386 if (this->
prolongation[refinement_case - 1][child].n() == 0)
391 if (this->
prolongation[refinement_case - 1][child].n() ==
401 std::vector<std::vector<FullMatrix<double>>> isotropic_matrices(
403 isotropic_matrices.back().resize(
411 this_nonconst.prolongation[refinement_case - 1] =
412 std::move(isotropic_matrices.back());
416 std::vector<std::vector<FullMatrix<double>>> matrices(
424 for (
unsigned int refinement_direction =
static_cast<unsigned int>(
426 refinement_direction <=
428 refinement_direction++)
429 this_nonconst.prolongation[refinement_direction - 1] =
430 std::move(matrices[refinement_direction - 1]);
442template <
int dim,
int spacedim>
445 const unsigned int face_dof_index,
446 const unsigned int face,
463 const unsigned int face_vertex =
465 const unsigned int dof_index_on_vertex =
471 face, face_vertex, combined_orientation) *
480 const unsigned int index =
484 const unsigned int dof_index_on_line = index % this->
n_dofs_per_line();
488 unsigned int adjusted_dof_index_on_line = 0;
497 adjusted_dof_index_on_line = dof_index_on_line;
499 adjusted_dof_index_on_line =
513 const auto cell_line_index =
516 combined_orientation);
517 const auto [canonical_face, canonical_line] =
523 const auto face_line_orientation =
527 combined_orientation,
531 if (face_line_orientation ==
533 adjusted_dof_index_on_line = dof_index_on_line;
536 Assert(face_line_orientation ==
539 adjusted_dof_index_on_line =
552 combined_orientation) *
554 adjusted_dof_index_on_line);
562 const unsigned int index =
583template <
int dim,
int spacedim>
586 const unsigned int child,
606 if (this->
restriction[refinement_case - 1][child].n() == 0)
611 if (this->
restriction[refinement_case - 1][child].n() ==
613 return this->
restriction[refinement_case - 1][child];
623 const double eps = 1e-12;
637 tria.
begin_active()->set_refine_choice(refinement_case);
640 const auto &child_cell = tria.
begin(0)->
child(child);
646 std::vector<Point<dim>> transformed_point(1);
661 if (this->
reference_cell().contains_point(transformed_point[0], eps))
663 restriction_mat[i][j] =
672 for (
unsigned int j = 0; j < this->n_dofs_per_cell(); j++)
673 sum += restriction_mat[i][j];
675 Assert(std::fabs(sum - 1) < eps || std::fabs(sum) < eps,
677 "The entries in a row of the local "
678 "restriction matrix do not add to zero or one. "
679 "This typically indicates that the "
680 "polynomial interpolation is "
681 "ill-conditioned such that round-off "
682 "prevents the sum to be one."));
687 for (
unsigned int i = 0; i < restriction_mat.
m(); ++i)
688 for (
unsigned int j = 0; j < restriction_mat.
n(); ++j)
690 if (std::fabs(restriction_mat(i, j)) < eps)
691 restriction_mat(i, j) = 0.;
692 if (std::fabs(restriction_mat(i, j) - 1) < eps)
693 restriction_mat(i, j) = 1.;
698 std::move(restriction_mat);
702 return this->
restriction[refinement_case - 1][child];
707template <
int dim,
int spacedim>
712 const unsigned int face_no)
const
726 const double eps = 2e-13 * this->degree * (dim - 1);
728 const std::vector<Point<dim>> face_quadrature_points =
736 for (
unsigned int j = 0; j < this->n_dofs_per_face(face_no); ++j)
738 double matrix_entry =
740 face_quadrature_points[i]);
745 if (std::fabs(matrix_entry - 1.0) < eps)
747 if (std::fabs(matrix_entry) < eps)
750 interpolation_matrix(i, j) = matrix_entry;
759 for (
unsigned int i = 0; i < this->n_dofs_per_face(face_no); ++i)
760 sum += interpolation_matrix(j, i);
779template <
int dim,
int spacedim>
783 const unsigned int subface,
785 const unsigned int face_no)
const
799 const double eps = 2e-13 * this->degree * (dim - 1);
811 for (
unsigned int j = 0; j < this->n_dofs_per_face(face_no); ++j)
813 double matrix_entry =
815 subface_quadrature.point(i));
820 if (std::fabs(matrix_entry - 1.0) < eps)
822 if (std::fabs(matrix_entry) < eps)
825 interpolation_matrix(i, j) = matrix_entry;
834 for (
unsigned int i = 0; i < this->n_dofs_per_face(face_no); ++i)
835 sum += interpolation_matrix(j, i);
854template <
int dim,
int spacedim>
863template <
int dim,
int spacedim>
868 std::vector<double> &nodal_values)
const
871 this->get_unit_support_points().size());
879 nodal_values[i] = support_point_values[i](0);
885template <
int dim,
int spacedim>
895 unit_support_points_fe_p<dim>(
degree),
897 constraints_fe_p<dim>(
degree))
909template <
int dim,
int spacedim>
910std::unique_ptr<FiniteElement<dim, spacedim>>
913 return std::make_unique<FE_SimplexP<dim, spacedim>>(*this);
918template <
int dim,
int spacedim>
922 std::ostringstream namebuf;
924 << this->degree <<
")";
926 return namebuf.str();
931template <
int dim,
int spacedim>
935 const unsigned int codim)
const
956 if (this->degree < fe_p_other->
degree)
958 else if (this->degree == fe_p_other->degree)
966 if (this->degree < fe_q_other->
degree)
968 else if (this->degree == fe_q_other->degree)
976 if (fe_nothing->is_dominating())
991template <
int dim,
int spacedim>
992std::vector<std::pair<unsigned int, unsigned int>>
1036template <
int dim,
int spacedim>
1037std::vector<std::pair<unsigned int, unsigned int>>
1053 std::vector<std::pair<unsigned int, unsigned int>> identities;
1055 for (
unsigned int i = 0; i < this->degree - 1; ++i)
1056 for (
unsigned int j = 0; j < fe_p_other->degree - 1; ++j)
1058 fe_p_other->unit_support_points[i + 3][0]) < 1e-14)
1059 identities.emplace_back(i, j);
1087 const std::vector<unsigned int> &index_map_inverse_q_other =
1088 fe_q_other->get_poly_space_numbering_inverse();
1090 std::vector<std::pair<unsigned int, unsigned int>> identities;
1092 for (
unsigned int i = 0; i < this->degree - 1; ++i)
1093 for (
unsigned int j = 0; j < fe_q_other->degree - 1; ++j)
1095 fe_q_other->get_unit_support_points()
1096 [index_map_inverse_q_other[j + 1]][0]) < 1e-14)
1097 identities.emplace_back(i, j);
1140template <
int dim,
int spacedim>
1150 unit_support_points_fe_p<dim>(
degree),
1152 constraints_fe_p<dim>(
degree))
1157template <
int dim,
int spacedim>
1158std::unique_ptr<FiniteElement<dim, spacedim>>
1161 return std::make_unique<FE_SimplexDGP<dim, spacedim>>(*this);
1166template <
int dim,
int spacedim>
1170 std::ostringstream namebuf;
1172 << this->degree <<
")";
1174 return namebuf.str();
1178template <
int dim,
int spacedim>
1182 const unsigned int codim)
const
1199 if (this->degree < fe_dgp_other->
degree)
1201 else if (this->degree == fe_dgp_other->degree)
1209 if (this->degree < fe_dgq_other->
degree)
1211 else if (this->degree == fe_dgq_other->degree)
1219 if (fe_nothing->is_dominating())
1234template <
int dim,
int spacedim>
1235std::vector<std::pair<unsigned int, unsigned int>>
1246template <
int dim,
int spacedim>
1247std::vector<std::pair<unsigned int, unsigned int>>
1258template <
int dim,
int spacedim>
1261 const unsigned int child,
1281 if (this->
restriction[refinement_case - 1][child].n() == 0)
1286 if (this->
restriction[refinement_case - 1][child].n() ==
1288 return this->
restriction[refinement_case - 1][child];
1296 std::vector<std::vector<FullMatrix<double>>> isotropic_matrices(
1298 isotropic_matrices.back().resize(
1306 this_nonconst.restriction[refinement_case - 1] =
1307 std::move(isotropic_matrices.back());
1311 std::vector<std::vector<FullMatrix<double>>> matrices(
1319 for (
unsigned int refinement_direction =
static_cast<unsigned int>(
1321 refinement_direction <=
1323 refinement_direction++)
1324 this_nonconst.restriction[refinement_direction - 1] =
1325 std::move(matrices[refinement_direction - 1]);
1332 return this->
restriction[refinement_case - 1][child];
1336#include "fe/fe_simplex_p.inst"
ArrayView< std::remove_reference_t< typename std::iterator_traits< Iterator >::reference >, MemorySpaceType > make_array_view(const Iterator begin, const Iterator end)
static BarycentricPolynomials< dim > get_fe_p_basis(const unsigned int degree)
TriaIterator< CellAccessor< dim, spacedim > > child(const unsigned int i) const
FE_Poly(const ScalarPolynomialsBase< dim > &poly_space, const FiniteElementData< dim > &fe_data, const std::vector< bool > &restriction_is_additive_flags, const std::vector< ComponentMask > &nonzero_components)
virtual double shape_value(const unsigned int i, const Point< dim > &p) const override
FiniteElementDomination::Domination compare_for_domination(const FiniteElement< dim, spacedim > &fe_other, const unsigned int codim) const override
std::string get_name() const override
std::vector< std::pair< unsigned int, unsigned int > > hp_line_dof_identities(const FiniteElement< dim, spacedim > &fe_other) const override
std::vector< std::pair< unsigned int, unsigned int > > hp_vertex_dof_identities(const FiniteElement< dim, spacedim > &fe_other) const override
FE_SimplexDGP(const unsigned int degree)
virtual const FullMatrix< double > & get_restriction_matrix(const unsigned int child, const RefinementCase< dim > &refinement_case=RefinementCase< dim >::isotropic_refinement) const override
std::unique_ptr< FiniteElement< dim, spacedim > > clone() const override
FE_SimplexP(const unsigned int degree)
std::vector< std::pair< unsigned int, unsigned int > > hp_line_dof_identities(const FiniteElement< dim, spacedim > &fe_other) const override
FiniteElementDomination::Domination compare_for_domination(const FiniteElement< dim, spacedim > &fe_other, const unsigned int codim) const override
std::vector< std::pair< unsigned int, unsigned int > > hp_vertex_dof_identities(const FiniteElement< dim, spacedim > &fe_other) const override
std::string get_name() const override
std::unique_ptr< FiniteElement< dim, spacedim > > clone() const override
FE_SimplexPoly(const BarycentricPolynomials< dim > polynomials, const FiniteElementData< dim > &fe_data, const bool prolongation_is_additive, const std::vector< Point< dim > > &unit_support_points, const std::vector< std::vector< Point< dim - 1 > > > unit_face_support_points, const FullMatrix< double > &interface_constraints)
virtual std::pair< Table< 2, bool >, std::vector< unsigned int > > get_constant_modes() const override
void get_subface_interpolation_matrix(const FiniteElement< dim, spacedim > &x_source_fe, const unsigned int subface, FullMatrix< double > &interpolation_matrix, const unsigned int face_no) const override
virtual const FullMatrix< double > & get_restriction_matrix(const unsigned int child, const RefinementCase< dim > &refinement_case=RefinementCase< dim >::isotropic_refinement) const override
virtual void convert_generalized_support_point_values_to_dof_values(const std::vector< Vector< double > > &support_point_values, std::vector< double > &nodal_values) const override
bool hp_constraints_are_implemented() const override
Threads::Mutex restriction_matrix_mutex
virtual const FullMatrix< double > & get_prolongation_matrix(const unsigned int child, const RefinementCase< dim > &refinement_case=RefinementCase< dim >::isotropic_refinement) const override
virtual unsigned int face_to_cell_index(const unsigned int face_dof_index, const unsigned int face, const types::geometric_orientation combined_orientation=numbers::default_geometric_orientation) const override
Threads::Mutex prolongation_matrix_mutex
void get_face_interpolation_matrix(const FiniteElement< dim, spacedim > &source_fe, FullMatrix< double > &interpolation_matrix, const unsigned int face_no) const override
unsigned int get_first_line_index() const
unsigned int n_dofs_per_vertex() const
const unsigned int degree
unsigned int n_dofs_per_cell() const
unsigned int n_dofs_per_line() const
unsigned int get_first_quad_index(const unsigned int quad_no=0) const
unsigned int n_dofs_per_face(unsigned int face_no=0, unsigned int child=0) const
ReferenceCell reference_cell() const
unsigned int get_first_face_line_index(const unsigned int face_no=0) const
unsigned int get_first_face_quad_index(const unsigned int face_no=0) const
unsigned int n_unique_faces() const
unsigned int n_dofs_per_quad(unsigned int face_no=0) const
const unsigned int dofs_per_cell
FiniteElementData(const std::vector< unsigned int > &dofs_per_object, const unsigned int n_components, const unsigned int degree, const Conformity conformity=unknown, const BlockIndices &block_indices=BlockIndices())
virtual Point< dim > unit_support_point(const unsigned int index) const
std::vector< std::vector< Point< dim - 1 > > > unit_face_support_points
std::vector< std::vector< FullMatrix< double > > > restriction
const std::vector< Point< dim > > & get_unit_support_points() const
const std::vector< Point< dim - 1 > > & get_unit_face_support_points(const unsigned int face_no=0) const
std::vector< int > adjust_line_dof_index_for_line_orientation_table
FiniteElement(const FiniteElementData< dim > &fe_data, const std::vector< bool > &restriction_is_additive_flags, const std::vector< ComponentMask > &nonzero_components)
std::vector< Point< dim > > unit_support_points
FullMatrix< double > interface_constraints
std::vector< std::vector< FullMatrix< double > > > prolongation
virtual void transform_points_real_to_unit_cell(const typename Triangulation< dim, spacedim >::cell_iterator &cell, const ArrayView< const Point< spacedim > > &real_points, const ArrayView< Point< dim > > &unit_points) const
static void project_to_subface(const ReferenceCell &reference_cell, const SubQuadrature &quadrature, const unsigned int face_no, const unsigned int subface_no, std::vector< Point< dim > > &q_points, const RefinementCase< dim - 1 > &ref_case=RefinementCase< dim - 1 >::isotropic_refinement)
static void project_to_face(const ReferenceCell &reference_cell, const SubQuadrature &quadrature, const unsigned int face_no, std::vector< Point< dim > > &q_points)
cell_iterator begin(const unsigned int level=0) const
virtual void execute_coarsening_and_refinement()
active_cell_iterator begin_active(const unsigned int level=0) const
#define DEAL_II_NAMESPACE_OPEN
constexpr bool running_in_debug_mode()
#define DEAL_II_NAMESPACE_CLOSE
#define DEAL_II_ASSERT_UNREACHABLE()
#define DEAL_II_NOT_IMPLEMENTED()
static ::ExceptionBase & ExcNotImplemented()
#define Assert(cond, exc)
static ::ExceptionBase & ExcImpossibleInDim(int arg1)
#define AssertDimension(dim1, dim2)
#define AssertIndexRange(index, range)
static ::ExceptionBase & ExcInternalError()
static ::ExceptionBase & ExcDimensionMismatch(std::size_t arg1, std::size_t arg2)
static ::ExceptionBase & ExcInterpolationNotImplemented()
#define AssertThrow(cond, exc)
const Mapping< dim, spacedim > & get_default_linear_mapping(const Triangulation< dim, spacedim > &triangulation)
@ either_element_can_dominate
@ other_element_dominates
@ neither_element_dominates
void reference_cell(Triangulation< dim, spacedim > &tria, const ReferenceCell &reference_cell)
constexpr ReferenceCell Triangle
constexpr ReferenceCell Tetrahedron
constexpr const ReferenceCell & get_simplex()
constexpr ReferenceCell Line
std::string dim_string(const int dim, const int spacedim)
constexpr types::geometric_orientation reverse_line_orientation
constexpr types::geometric_orientation default_geometric_orientation
unsigned char geometric_orientation
static Point< dim > child_to_cell_coordinates(const Point< dim > &p, const unsigned int child_index, const RefinementCase< dim > refine_case=RefinementCase< dim >::isotropic_refinement)