12#ifndef __POSTPROCBROKENMESHINMOABBASE_HPP
13#define __POSTPROCBROKENMESHINMOABBASE_HPP
25 std::vector<MatrixDouble>
28 std::vector<MatrixDouble>
54 boost::shared_ptr<PostProcGenerateRefMeshBase>;
97 std::string opts_prefix =
"");
99 boost::shared_ptr<moab::Core> core_mesh_ptr,
100 std::string opts_prefix =
"");
165 boost::shared_ptr<moab::Core>
coreMeshPtr = boost::make_shared<moab::Core>();
170 std::map<EntityType, PostProcGenerateRefMeshPtr>
187 std::vector<Tag> tags_to_transfer) {
189 tagsToTransfer.swap(tags_to_transfer);
194 auto get_element_max_dofs_order = [&]() {
196 auto dofs_vec = E::getDataVectorDofsPtr();
197 for (
auto &dof : *dofs_vec) {
198 const int dof_order = dof->getDofOrder();
199 max_order = (max_order < dof_order) ? dof_order : max_order;
203 const auto dof_max_order = get_element_max_dofs_order();
204 return (dof_max_order > 0) ? (dof_max_order - 1) / 2 : 0;
211 auto &calc_mesh = this->mField.get_moab();
213 auto name = [&](
auto tag) {
219 auto data_type = [&](
auto tag) {
220 moab::DataType data_type;
226 auto type = [&](
auto tag) {
228 CHK_MOAB_THROW(calc_mesh.tag_get_type(tag, type),
"get tag type");
232 auto length = [&](
auto tag) {
234 CHK_MOAB_THROW(calc_mesh.tag_get_length(tag, length),
"get length ");
238 auto default_value = [&](
auto tag) {
241 CHK_MOAB_THROW(calc_mesh.tag_get_default_value(tag, def_val, size),
242 "get default tag value");
246 std::vector<double> tag_data_vec;
247 auto data_ptr = [&](
auto tag) {
248 const void *tag_data;
249 auto core_ent = this->getFEEntityHandle();
250 CHK_MOAB_THROW(calc_mesh.tag_get_by_ptr(tag, &core_ent, 1, &tag_data),
252 if (data_type(tag) == MB_TYPE_DOUBLE) {
253 tag_data_vec.resize(length(tag));
254 std::copy(
static_cast<const double *
>(tag_data),
255 static_cast<const double *
>(tag_data) + length(tag),
256 tag_data_vec.begin());
257 for (
auto &
v : tag_data_vec)
258 v = std::abs(
v) < std::numeric_limits<float>::min() ? 0. :
v;
259 for (
auto &
v : tag_data_vec)
260 v =
v > std::numeric_limits<float>::max()
261 ? std::numeric_limits<float>::max()
263 for (
auto &
v : tag_data_vec)
264 v =
v < std::numeric_limits<float>::lowest()
265 ? std::numeric_limits<float>::lowest()
267 return static_cast<const void *
>(tag_data_vec.data());
272 for (
auto tag : tagsToTransfer) {
274 CHKERR getPostProcMesh().tag_get_handle(
275 name(tag).c_str(), length(tag), data_type(tag), tag_postproc,
276 type(tag) | MB_TAG_CREAT, default_value(tag));
277 CHKERR getPostProcMesh().tag_clear_data(tag_postproc, postProcElements,
287 :
E(m_field), optionsPrefix(opts_prefix) {}
292 std::string opts_prefix)
299 ParallelComm *pcomm_post_proc_mesh =
301 if (pcomm_post_proc_mesh != NULL)
302 delete pcomm_post_proc_mesh;
318 ref_ele = refElementsMap.at(type);
319 }
catch (
const out_of_range &e) {
322 "Generation of reference elements for type <%s> is not implemented",
323 moab::CN::EntityTypeName(type));
326 auto set_gauss_pts = [&](
auto &level_gauss_pts_on_ref_mesh,
auto &level_ref,
327 auto &level_shape_functions,
329 auto start_vert_handle,
auto start_ele_handle,
330 auto &verts_array,
auto &conn,
auto &ver_count,
336 size_t level = getMaxLevel();
337 level = std::min(level, level_gauss_pts_on_ref_mesh.size() - 1);
339 auto &level_ref_gauss_pts = level_gauss_pts_on_ref_mesh[level];
340 auto &level_ref_ele = level_ref[level];
341 auto &shape_functions = level_shape_functions[level];
342 E::gaussPts.resize(level_ref_gauss_pts.size1(), level_ref_gauss_pts.size2(),
344 noalias(E::gaussPts) = level_ref_gauss_pts;
346 const auto fe_ent = E::numeredEntFiniteElementPtr->getEnt();
347 auto get_fe_coords = [&]() {
351 E::mField.get_moab().get_connectivity(fe_ent, conn, num_nodes,
true),
352 "error get connectivity");
355 E::mField.get_moab().get_coords(conn, num_nodes, &*coords.begin()),
356 "error get coordinates");
360 auto coords = get_fe_coords();
362 const int num_nodes = level_ref_gauss_pts.size2();
363 mapGaussPts.resize(level_ref_gauss_pts.size2());
367 &*shape_functions.data().begin());
369 &verts_array[0][ver_count], &verts_array[1][ver_count],
370 &verts_array[2][ver_count]);
371 for (
int gg = 0; gg != num_nodes; ++gg, ++ver_count) {
373 mapGaussPts[gg] = start_vert_handle + ver_count;
375 auto set_float_precision = [](
const double x) {
376 if (std::abs(x) < std::numeric_limits<float>::epsilon())
383 auto t_ele_coords = getFTensor1FromArray<3, 3>(coords);
384 for (
int nn = 0; nn != CN::VerticesPerEntity(type); ++nn) {
385 t_coords(
i) += t_n * t_ele_coords(
i);
390 for (
auto ii : {0, 1, 2})
391 t_coords(ii) = set_float_precision(t_coords(ii));
397 int def_in_the_loop = -1;
398 CHKERR getPostProcMesh().tag_get_handle(
399 "NB_IN_THE_LOOP", 1, MB_TYPE_INTEGER,
th, MB_TAG_CREAT | MB_TAG_SPARSE,
402 postProcElements.clear();
403 const int num_el = level_ref_ele.size1();
404 const int num_nodes_on_ele = level_ref_ele.size2();
405 auto start_e = start_ele_handle + ele_count;
406 postProcElements =
Range(start_e, start_e + num_el - 1);
407 for (
auto tt = 0; tt != level_ref_ele.size1(); ++tt, ++ele_count) {
408 for (
int nn = 0; nn != num_nodes_on_ele; ++nn) {
409 conn[num_nodes_on_ele * ele_count + nn] =
410 mapGaussPts[level_ref_ele(tt, nn)];
414 const int n_in_the_loop = E::nInTheLoop;
415 CHKERR getPostProcMesh().tag_clear_data(
th, postProcElements,
424 ref_ele->levelGaussPtsOnRefMesh, ref_ele->levelRef,
425 ref_ele->levelShapeFunctions,
427 ref_ele->startingVertEleHandle, ref_ele->startingEleHandle,
428 ref_ele->verticesOnEleArrays, ref_ele->eleConn, ref_ele->countVertEle,
440auto get_ref_ele = [&](
const EntityType type) {
443 auto it = refElementsMap.find(type);
444 if (it != refElementsMap.end()) {
445 ref_ele_ptr = it->second;
449 ref_ele_ptr = boost::make_shared<PostProcGenerateRefMesh<MBTET>>();
452 ref_ele_ptr = boost::make_shared<PostProcGenerateRefMesh<MBHEX>>();
455 ref_ele_ptr = boost::make_shared<PostProcGenerateRefMesh<MBTRI>>();
458 ref_ele_ptr = boost::make_shared<PostProcGenerateRefMesh<MBQUAD>>();
461 ref_ele_ptr = boost::make_shared<PostProcGenerateRefMesh<MBEDGE>>();
465 <<
"Generation of reference elements for type < "
466 << moab::CN::EntityTypeName(type) <<
" > is not implemented";
472 "Error when generating reference element");
474 refElementsMap[type] = ref_ele_ptr;
480 auto fe_ptr = this->problemPtr->numeredFiniteElementsPtr;
483 fe_ptr->template get<Composite_Name_And_Part_mi_tag>().lower_bound(
484 boost::make_tuple(this->getFEName(), this->getLoFERank()));
486 fe_ptr->template get<Composite_Name_And_Part_mi_tag>().upper_bound(
487 boost::make_tuple(this->getFEName(), this->getHiFERank()));
489 const int number_of_ents_in_the_loop = this->getLoopSize();
490 if (std::distance(miit, hi_miit) != number_of_ents_in_the_loop) {
492 "Wrong size of indicies. Inconsistent size number of iterated "
493 "elements iterated by problem and from range.");
496 for (
auto &
m : refElementsMap) {
497 m.second->nbVertices = 0;
498 m.second->nbEles = 0;
499 m.second->countEle = 0;
500 m.second->countVertEle = 0;
503 for (; miit != hi_miit; ++miit) {
504 auto type = (*miit)->getEntType();
505 auto ref_ele = get_ref_ele(type);
509 E::numeredEntFiniteElementPtr = *miit;
511 if (E::exeTestHook) {
512 add = E::exeTestHook(
this);
516 size_t level = getMaxLevel();
517 level = std::min(level, ref_ele->levelGaussPtsOnRefMesh.size() - 1);
518 ref_ele->nbVertices += ref_ele->levelGaussPtsOnRefMesh[level].size2();
519 ref_ele->nbEles += ref_ele->levelRef[level].size1();
523 auto alloc_vertices_and_elements_on_post_proc_mesh = [&]() {
526 ReadUtilIface *iface;
527 CHKERR getPostProcMesh().query_interface(iface);
529 for (
auto &
m : refElementsMap) {
530 if (
m.second->nbEles) {
531 CHKERR iface->get_node_coords(3,
m.second->nbVertices, 0,
532 m.second->startingVertEleHandle,
533 m.second->verticesOnEleArrays);
534 CHKERR iface->get_element_connect(
535 m.second->nbEles,
m.second->levelRef[0].size2(),
m.first, 0,
536 m.second->startingEleHandle,
m.second->eleConn);
538 m.second->countEle = 0;
539 m.second->countVertEle = 0;
546 CHKERR alloc_vertices_and_elements_on_post_proc_mesh();
555 auto update_elements = [&]() {
556 ReadUtilIface *iface;
557 CHKERR getPostProcMesh().query_interface(iface);
561 for (
auto &
m : refElementsMap) {
562 if (
m.second->nbEles) {
564 <<
"Update < " << moab::CN::EntityTypeName(
m.first)
565 <<
" number of processed " <<
m.second->countEle;
566 CHKERR iface->update_adjacencies(
567 m.second->startingEleHandle,
m.second->countEle,
568 m.second->levelRef[0].size2(),
m.second->eleConn);
569 ents.merge(
Range(
m.second->startingEleHandle,
570 m.second->startingEleHandle +
m.second->countEle - 1));
577 auto remove_obsolete_entities = [&]() {
580 for (
auto &
m : refElementsMap) {
581 if (
m.second->nbEles) {
582 ents.merge(
Range(
m.second->startingEleHandle,
583 m.second->startingEleHandle +
m.second->countEle - 1));
584 const int dim = moab::CN::Dimension(
m.first);
585 for (
auto d = 1; d != dim; ++d) {
586 CHKERR getPostProcMesh().get_adjacencies(ents, d,
false, adj,
587 moab::Interface::UNION);
591 CHKERR getPostProcMesh().delete_entities(adj);
595 auto set_proc_tags = [&]() {
597 ParallelComm *pcomm_post_proc_mesh =
598 ParallelComm::get_pcomm(&(getPostProcMesh()),
MYPCOMM_INDEX);
599 if (pcomm_post_proc_mesh) {
601 for (
auto &
m : refElementsMap) {
602 if (
m.second->nbEles) {
604 Range(
m.second->startingEleHandle,
605 m.second->startingEleHandle +
m.second->countEle - 1));
609 int rank = E::mField.get_comm_rank();
610 CHKERR getPostProcMesh().tag_clear_data(pcomm_post_proc_mesh->part_tag(),
617 CHKERR remove_obsolete_entities();
627 ParallelComm *pcomm_post_proc_mesh =
629 if (pcomm_post_proc_mesh != NULL)
630 delete pcomm_post_proc_mesh;
631 CHKERR getPostProcMesh().delete_mesh();
632 pcomm_post_proc_mesh =
633 new ParallelComm(&(getPostProcMesh()), PETSC_COMM_WORLD);
644 ParallelComm *pcomm_post_proc_mesh =
646 if(pcomm_post_proc_mesh ==
nullptr)
649 CHKERR postProcPostProc();
650 CHKERR pcomm_post_proc_mesh->resolve_shared_ents(0);
662 moab::Interface &post_proc_mesh_interface = *coreMeshPtr;
663 return post_proc_mesh_interface;
668 return postProcElements;
675 ParallelComm *pcomm_post_proc_mesh =
677 if (pcomm_post_proc_mesh == NULL)
679 "ParallelComm not allocated");
680 CHKERR getPostProcMesh().write_file(file_name.c_str(),
"MOAB",
681 "PARALLEL=WRITE_PART");
719 using DataMapVec = std::map<std::string, boost::shared_ptr<VectorDouble>>;
720 using DataMapMat = std::map<std::string, boost::shared_ptr<MatrixDouble>>;
734 std::vector<EntityHandle> &map_gauss_pts,
742 std::fill(&O::doEntities[MBEDGE], &O::doEntities[MBMAXTYPE],
false);
756template <
int DIM1,
int DIM2,
typename O>
762 std::array<double, 9> def;
763 std::fill(def.begin(), def.end(), 0);
765 auto get_tag = [&](
const std::string name,
size_t size) {
767 CHKERR postProcMesh.tag_get_handle(name.c_str(), size, MB_TYPE_DOUBLE,
th,
768 MB_TAG_CREAT | MB_TAG_SPARSE,
777 for (
size_t r = 0; r !=
DIM1; ++r)
784 for (
size_t r = 0; r !=
DIM1; ++r)
785 for (
size_t c = 0;
c !=
DIM2; ++
c)
792 for (
size_t r = 0; r !=
DIM1; ++r)
793 for (
size_t c = 0;
c !=
DIM1; ++
c)
804 auto set_float_precision = [](
const double x) {
805 if (std::abs(x) < std::numeric_limits<float>::min())
806 return static_cast<float>(0.);
808 return static_cast<float>(x);
811 auto set_float_max = [](
const double x) {
812 if (x > std::numeric_limits<float>::max())
813 return std::numeric_limits<float>::max();
815 return static_cast<float>(x);
818 auto set_float_lowest = [](
const double x) {
819 if (x < std::numeric_limits<float>::lowest())
820 return std::numeric_limits<float>::lowest();
822 return static_cast<float>(x);
826 for (
auto &
v : mat.data())
827 v = set_float_max(set_float_lowest(set_float_precision(
v)));
828 return postProcMesh.tag_set_data(
th, &mapGaussPts[gg], 1,
829 &*mat.data().begin());
832 for (
auto &
m : dataMapScalar) {
834 auto th = get_tag(
m.first, 1);
836 auto nb_integration_pts = O::getGaussPts().size2();
837 for (
int gg = 0; gg != nb_integration_pts; ++gg) {
838 CHKERR set_tag(
th, gg, set_scalar(t_scl));
844 for (
auto &
m : dataMapVec) {
846 auto th = get_tag(
m.first, 3);
847 auto t_vec = getFTensor1FromMat<DIM1>(*
m.second);
848 auto nb_integration_pts = O::getGaussPts().size2();
849 for (
int gg = 0; gg != nb_integration_pts; ++gg) {
850 CHKERR set_tag(
th, gg, set_vector_3d(t_vec));
856 for (
auto &
m : dataMapMat) {
858 auto th = get_tag(
m.first, 9);
859 auto t_mat = getFTensor2FromMat<DIM1, DIM2>(*
m.second);
860 auto nb_integration_pts = O::getGaussPts().size2();
861 for (
int gg = 0; gg != nb_integration_pts; ++gg) {
862 CHKERR set_tag(
th, gg, set_matrix_3d(t_mat));
868 for (
auto &
m : dataMapSymmMat) {
870 auto th = get_tag(
m.first, 9);
871 auto t_mat = getFTensor2SymmetricFromMat<DIM1>(*
m.second);
872 auto nb_integration_pts = O::getGaussPts().size2();
873 for (
int gg = 0; gg != nb_integration_pts; ++gg) {
874 CHKERR set_tag(
th, gg, set_matrix_symm_3d(t_mat));
891 std::string opts_prefix =
"")
896 ParallelComm *pcomm_post_proc_mesh =
898 if (pcomm_post_proc_mesh != NULL)
899 delete pcomm_post_proc_mesh;
901 pcomm_post_proc_mesh =
912template <
typename PostProcEle>
struct PostProcBrokenMeshInMoabBaseContImpl;
920 std::string opts_prefix =
"")
930template <
typename PostProcEle>
struct PostProcBrokenMeshInMoabBaseEndImpl;
938 std::string opts_prefix =
"")
945 ParallelComm *pcomm_post_proc_mesh =
947 if (pcomm_post_proc_mesh ==
nullptr)
949 "PComm not allocated");
950 CHKERR pcomm_post_proc_mesh->resolve_shared_ents(0);
#define MOFEM_TAG_AND_LOG(channel, severity, tag)
Tag and log in channel.
#define CHK_THROW_MESSAGE(err, msg)
Check and throw MoFEM exception.
#define MoFEMFunctionReturnHot(a)
Last executable line of each PETSc function used for error handling. Replaces return()
#define MYPCOMM_INDEX
default communicator number PCOMM
#define MoFEMFunctionBegin
First executable line of each MoFEM function, used for error handling. Final line of MoFEM functions ...
#define CHK_MOAB_THROW(err, msg)
Check error code of MoAB function and throw MoFEM exception.
@ MOFEM_DATA_INCONSISTENCY
#define MoFEMFunctionReturn(a)
Last executable line of each PETSc function used for error handling. Replaces return()
#define CHKERR
Inline error check.
#define MoFEMFunctionBeginHot
First executable line of each MoFEM function, used for error handling. Final line of MoFEM functions ...
constexpr int order
Order displacement.
MoFEMErrorCode writeFile(const std::string file_name)
wrote results in (MOAB) format, use "file_name.h5m"
#define MOFEM_LOG(channel, severity)
Log.
FTensor::Index< 'i', SPACE_DIM > i
const double c
speed of light (cm/ns)
const double v
phase velocity of light in medium (cm/ns)
PetscErrorCode MoFEMErrorCode
MoFEM/PETSc error code.
implementation of Data Operators for Forces and Sources
auto type_from_handle(const EntityHandle h)
get type from entity handle
boost::shared_ptr< PostProcGenerateRefMeshBase > PostProcGenerateRefMeshPtr
static auto getFTensor0FromVec(ublas::vector< T, A > &data)
Get tensor rank 0 (scalar) form data vector.
constexpr double t
plate stiffness
FTensor::Index< 'm', 3 > m
Deprecated interface functions.
Data on single entity (This is passed as argument to DataOperator::doWork)
Structure for user loop methods on finite elements.
@ OPSPACE
operator do Work is execute on space data
Post post-proc data at points from hash maps.
MoFEMErrorCode doWork(int side, EntityType type, EntitiesFieldData::EntData &data)
Operator for linear form, usually to calculate values on right hand side.
DataMapMat dataMapSymmMat
std::vector< EntityHandle > & mapGaussPts
std::map< std::string, boost::shared_ptr< MatrixDouble > > DataMapMat
std::map< std::string, boost::shared_ptr< VectorDouble > > DataMapVec
OpPostProcMapInMoab(moab::Interface &post_proc_mesh, std::vector< EntityHandle > &map_gauss_pts, DataMapVec data_map_scalar, DataMapMat data_map_vec, DataMapMat data_map_mat, DataMapMat data_symm_map_mat)
Construct a new OpPostProcMapInMoab object.
moab::Interface & postProcMesh
MoFEMErrorCode operator()()
MoFEMErrorCode preProcess()
MoFEMErrorCode postProcess()
PostProcBrokenMeshInMoabBaseBeginImpl(MoFEM::Interface &m_field, boost::shared_ptr< moab::Core > core_mesh_ptr, std::string opts_prefix="")
PostProcBrokenMeshInMoabBaseContImpl(MoFEM::Interface &m_field, boost::shared_ptr< moab::Core > core_mesh_ptr, std::string opts_prefix="")
MoFEMErrorCode postProcess()
MoFEMErrorCode preProcess()
MoFEMErrorCode operator()()
MoFEMErrorCode preProcess()
MoFEMErrorCode postProcess()
PostProcBrokenMeshInMoabBaseEndImpl(MoFEM::Interface &m_field, boost::shared_ptr< moab::Core > core_mesh_ptr, std::string opts_prefix="")
auto & getPostProcElements()
Get postprocessing elements.
MoFEMErrorCode preProcess()
Generate vertices and elements.
virtual MoFEMErrorCode postProcPostProc()
MoFEMErrorCode setGaussPts(int order)
std::vector< Tag > tagsToTransfer
std::string optionsPrefix
Prefix for options.
PostProcBrokenMeshInMoabBase(MoFEM::Interface &m_field, boost::shared_ptr< moab::Core > core_mesh_ptr, std::string opts_prefix="")
PostProcBrokenMeshInMoabBase(MoFEM::Interface &m_field, std::string opts_prefix="")
virtual MoFEMErrorCode transferTags()
auto & getMapGaussPts()
Get vector of vectors associated to integration points.
auto & getPostProcMesh()
Get postprocessing mesh.
MoFEMErrorCode postProcess()
boost::shared_ptr< moab::Core > coreMeshPtr
std::vector< EntityHandle > mapGaussPts
virtual int getMaxLevel() const
Determine refinement level based on fields approx ordre.
virtual ~PostProcBrokenMeshInMoabBase()
MoFEMErrorCode setTagsToTransfer(std::vector< Tag > tags_to_transfer)
Set tags to be transferred to post-processing mesh.
std::map< EntityType, PostProcGenerateRefMeshPtr > refElementsMap
virtual MoFEMErrorCode preProcPostProc()
EntityHandle startingEleHandle
virtual MoFEMErrorCode generateReferenceElementMesh()=0
virtual ~PostProcGenerateRefMeshBase()=default
std::vector< ublas::matrix< int > > levelRef
std::vector< double * > verticesOnEleArrays
virtual MoFEMErrorCode getOptions(std::string prefix)
std::vector< MatrixDouble > levelShapeFunctions
PostProcGenerateRefMeshBase()
std::vector< MatrixDouble > levelGaussPtsOnRefMesh
EntityHandle startingVertEleHandle
Element for postprocessing. Uses MoAB to generate post-processing mesh.
Volume finite element base.