8#ifndef __CONTACTOPS_HPP__
9#define __CONTACTOPS_HPP__
14struct CommonData :
public boost::enable_shared_from_this<CommonData> {
28 static SmartPetscObj<Vec>
32 constexpr int ghosts[] = {0, 1, 2, 3, 4};
34 createGhostVector(m_field.
get_comm(),
58 return boost::shared_ptr<MatrixDouble>(shared_from_this(),
63 return boost::shared_ptr<MatrixDouble>(shared_from_this(), &
contactDisp);
67 return boost::shared_ptr<MatrixDouble>(shared_from_this(),
72 return boost::shared_ptr<VectorDouble>(shared_from_this(), &
sdfVals);
76 return boost::shared_ptr<MatrixDouble>(shared_from_this(), &
gradsSdf);
80 return boost::shared_ptr<MatrixDouble>(shared_from_this(), &
hessSdf);
84 return boost::shared_ptr<VectorDouble>(shared_from_this(), &
constraintVals);
93#ifdef ENABLE_PYTHON_BINDING
95 SDFPython() =
default;
96 virtual ~SDFPython() =
default;
98 MoFEMErrorCode sdfInit(
const std::string py_file) {
103 auto main_module = bp::import(
"__main__");
104 mainNamespace = main_module.attr(
"__dict__");
105 bp::exec_file(py_file.c_str(), mainNamespace, mainNamespace);
107 sdfFun = mainNamespace[
"sdf"];
108 sdfGradFun = mainNamespace[
"grad_sdf"];
109 sdfHessFun = mainNamespace[
"hess_sdf"];
111 }
catch (bp::error_already_set
const &) {
119 template <
typename T>
120 inline std::vector<T>
121 py_list_to_std_vector(
const boost::python::object &iterable) {
122 return std::vector<T>(boost::python::stl_input_iterator<T>(iterable),
123 boost::python::stl_input_iterator<T>());
128 double delta_t,
double t, np::ndarray x, np::ndarray y, np::ndarray z,
129 np::ndarray tx, np::ndarray ty, np::ndarray tz,
int block_id,
137 sdf = bp::extract<np::ndarray>(
138 sdfFun(delta_t,
t, x, y, z, tx, ty, tz, block_id));
140 }
catch (bp::error_already_set
const &) {
150 double delta_t,
double t, np::ndarray x, np::ndarray y, np::ndarray z,
151 np::ndarray tx, np::ndarray ty, np::ndarray tz,
int block_id,
152 np::ndarray &grad_sdf
159 grad_sdf = bp::extract<np::ndarray>(
160 sdfGradFun(delta_t,
t, x, y, z, tx, ty, tz, block_id));
162 }
catch (bp::error_already_set
const &) {
172 double delta_t,
double t, np::ndarray x, np::ndarray y, np::ndarray z,
173 np::ndarray tx, np::ndarray ty, np::ndarray tz,
int block_id,
174 np::ndarray &hess_sdf
181 hess_sdf = bp::extract<np::ndarray>(
182 sdfHessFun(delta_t,
t, x, y, z, tx, ty, tz, block_id));
184 }
catch (bp::error_already_set
const &) {
193 bp::object mainNamespace;
195 bp::object sdfGradFun;
196 bp::object sdfHessFun;
199static boost::weak_ptr<SDFPython> sdfPythonWeakPtr;
201inline np::ndarray convert_to_numpy(VectorDouble &data,
int nb_gauss_pts,
203 auto dtype = np::dtype::get_builtin<double>();
204 auto size = bp::make_tuple(nb_gauss_pts);
205 auto stride = bp::make_tuple(3 *
sizeof(
double));
206 return (np::from_data(&data[
id], dtype, size, stride, bp::object()));
212 double delta_t,
double t,
int nb_gauss_pts, MatrixDouble &spatial_coords,
213 MatrixDouble &normals_at_pts,
int block_id)>;
216 double delta_t,
double t,
int nb_gauss_pts, MatrixDouble &spatial_coords,
217 MatrixDouble &normals_at_pts,
int block_id)>;
220 double delta_t,
double t,
int nb_gauss_pts, MatrixDouble &spatial_coords,
221 MatrixDouble &normals_at_pts,
int block_id)>;
225 MatrixDouble &m_spatial_coords,
226 MatrixDouble &m_normals_at_pts,
229#ifdef ENABLE_PYTHON_BINDING
230 if (
auto sdf_ptr = sdfPythonWeakPtr.lock()) {
232 VectorDouble v_spatial_coords = m_spatial_coords.data();
233 VectorDouble v_normal_at_pts = m_normals_at_pts.data();
235 bp::list python_coords;
236 bp::list python_normals;
238 for (
int idx = 0; idx < 3; ++idx) {
239 python_coords.append(
240 convert_to_numpy(v_spatial_coords, nb_gauss_pts, idx));
241 python_normals.append(
242 convert_to_numpy(v_normal_at_pts, nb_gauss_pts, idx));
245 np::ndarray np_sdf = np::empty(bp::make_tuple(nb_gauss_pts),
246 np::dtype::get_builtin<double>());
248 bp::extract<np::ndarray>(python_coords[0]),
249 bp::extract<np::ndarray>(python_coords[1]),
250 bp::extract<np::ndarray>(python_coords[2]),
251 bp::extract<np::ndarray>(python_normals[0]),
252 bp::extract<np::ndarray>(python_normals[1]),
253 bp::extract<np::ndarray>(python_normals[2]),
255 "Failed python call");
258 if (np_sdf.get_nd() != 1 || np_sdf.get_shape()[0] != nb_gauss_pts) {
261 "Wrong number of dimensions or size of SDF returned from "
262 "python, expected: dim 1, got: " +
263 std::to_string(np_sdf.get_nd()) +
", expected size: (" +
264 std::to_string(nb_gauss_pts) +
")");
267 double *sdf_val_ptr =
reinterpret_cast<double *
>(np_sdf.get_data());
270 v_sdf.resize(nb_gauss_pts,
false);
272 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg)
273 v_sdf[gg] = *(sdf_val_ptr + gg);
279 v_sdf.resize(nb_gauss_pts,
false);
280 auto t_coords = getFTensor1FromPtr<3>(&m_spatial_coords(0, 0));
282 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg) {
283 v_sdf[gg] = -t_coords(2) - 0.1;
292 MatrixDouble &m_spatial_coords,
293 MatrixDouble &m_normals_at_pts,
int block_id) {
294#ifdef ENABLE_PYTHON_BINDING
295 if (
auto sdf_ptr = sdfPythonWeakPtr.lock()) {
297 VectorDouble v_spatial_coords = m_spatial_coords.data();
298 VectorDouble v_normal_at_pts = m_normals_at_pts.data();
300 bp::list python_coords;
301 bp::list python_normals;
303 for (
int idx = 0; idx < 3; ++idx) {
304 python_coords.append(
305 convert_to_numpy(v_spatial_coords, nb_gauss_pts, idx));
306 python_normals.append(
307 convert_to_numpy(v_normal_at_pts, nb_gauss_pts, idx));
310 np::ndarray np_grad_sdf = np::empty(bp::make_tuple(nb_gauss_pts, 3),
311 np::dtype::get_builtin<double>());
313 delta_t,
t, bp::extract<np::ndarray>(python_coords[0]),
314 bp::extract<np::ndarray>(python_coords[1]),
315 bp::extract<np::ndarray>(python_coords[2]),
316 bp::extract<np::ndarray>(python_normals[0]),
317 bp::extract<np::ndarray>(python_normals[1]),
318 bp::extract<np::ndarray>(python_normals[2]), block_id,
320 "Failed python call");
323 if (np_grad_sdf.get_shape()[0] != nb_gauss_pts ||
324 np_grad_sdf.get_shape()[1] != 3) {
326 "Wrong shape of gradient of SDF returned from "
327 "python, expected: (" +
328 std::to_string(nb_gauss_pts) +
", 3), got: (" +
329 std::to_string(np_grad_sdf.get_shape()[0]) +
", " +
330 std::to_string(np_grad_sdf.get_shape()[1]) +
")");
333 double *grad_ptr =
reinterpret_cast<double *
>(np_grad_sdf.get_data());
335 MatrixDouble m_grad_sdf;
336 m_grad_sdf.resize(nb_gauss_pts, 3,
false);
337 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg) {
338 for (
int idx = 0; idx < 3; ++idx)
339 m_grad_sdf(gg, idx) = *(grad_ptr + (3 * gg + idx));
344 MatrixDouble m_grad_sdf;
345 m_grad_sdf.resize(nb_gauss_pts, 3,
false);
348 auto t_grad_sdf = getFTensor1FromMat<3>(m_grad_sdf);
350 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg) {
351 t_grad_sdf(
i) = t_grad_sdf_set(
i);
360 MatrixDouble &m_spatial_coords,
361 MatrixDouble &m_normals_at_pts,
int block_id) {
362#ifdef ENABLE_PYTHON_BINDING
363 if (
auto sdf_ptr = sdfPythonWeakPtr.lock()) {
365 VectorDouble v_spatial_coords = m_spatial_coords.data();
366 VectorDouble v_normal_at_pts = m_normals_at_pts.data();
368 bp::list python_coords;
369 bp::list python_normals;
371 for (
int idx = 0; idx < 3; ++idx) {
372 python_coords.append(
373 convert_to_numpy(v_spatial_coords, nb_gauss_pts, idx));
374 python_normals.append(
375 convert_to_numpy(v_normal_at_pts, nb_gauss_pts, idx));
378 np::ndarray np_hess_sdf = np::empty(bp::make_tuple(nb_gauss_pts, 6),
379 np::dtype::get_builtin<double>());
381 delta_t,
t, bp::extract<np::ndarray>(python_coords[0]),
382 bp::extract<np::ndarray>(python_coords[1]),
383 bp::extract<np::ndarray>(python_coords[2]),
384 bp::extract<np::ndarray>(python_normals[0]),
385 bp::extract<np::ndarray>(python_normals[1]),
386 bp::extract<np::ndarray>(python_normals[2]), block_id,
388 "Failed python call");
391 if (np_hess_sdf.get_shape()[0] != nb_gauss_pts ||
392 np_hess_sdf.get_shape()[1] != 6) {
394 "Wrong shape of Hessian of SDF returned from "
395 "python, expected: (" +
396 std::to_string(nb_gauss_pts) +
", 6), got: (" +
397 std::to_string(np_hess_sdf.get_shape()[0]) +
", " +
398 std::to_string(np_hess_sdf.get_shape()[1]) +
")");
401 double *hess_ptr =
reinterpret_cast<double *
>(np_hess_sdf.get_data());
403 MatrixDouble m_hess_sdf;
404 m_hess_sdf.resize(nb_gauss_pts, 6,
false);
405 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg) {
406 for (
int idx = 0; idx < 6; ++idx)
407 m_hess_sdf(gg, idx) = *(hess_ptr + (6 * gg + idx));
412 MatrixDouble m_hess_sdf;
413 m_hess_sdf.resize(nb_gauss_pts, 6,
false);
417 auto t_hess_sdf = getFTensor2SymmetricFromMat<3>(m_hess_sdf);
419 for (
size_t gg = 0; gg < nb_gauss_pts; ++gg) {
420 t_hess_sdf(
i,
j) = t_hess_sdf_set(
i,
j);
426template <
int DIM, IntegrationType I,
typename BoundaryEleOp>
429template <
int DIM, IntegrationType I,
typename BoundaryEleOp>
432template <
int DIM, IntegrationType I,
typename BoundaryEleOp>
435template <
int DIM, IntegrationType I,
typename AssemblyBoundaryEleOp>
438template <
int DIM, IntegrationType I,
typename AssemblyBoundaryEleOp>
441template <
int DIM, IntegrationType I,
typename AssemblyBoundaryEleOp>
444template <
typename T1,
typename T2,
int DIM1,
int DIM2>
447 size_t nb_gauss_pts) {
448 MatrixDouble m_spatial_coords(nb_gauss_pts, 3);
449 m_spatial_coords.clear();
450 auto t_spatial_coords = getFTensor1FromPtr<3>(&m_spatial_coords(0, 0));
452 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
453 t_spatial_coords(
i) = t_coords(
i) + t_disp(
i);
458 return m_spatial_coords;
461template <
typename T1,
int DIM1>
463 size_t nb_gauss_pts) {
464 MatrixDouble m_normals_at_pts(nb_gauss_pts, 3);
465 m_normals_at_pts.clear();
467 auto t_set_normal = getFTensor1FromMat<3>(m_normals_at_pts);
468 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
469 t_set_normal(
i) = t_normal_at_pts(
i) / t_normal_at_pts.l2();
473 return m_normals_at_pts;
476template <
int DIM,
typename BoundaryEleOp>
480 boost::shared_ptr<CommonData> common_data_ptr,
double scale = 1,
482 MoFEMErrorCode doWork(
int side, EntityType
type,
EntData &data);
490template <
int DIM,
typename BoundaryEleOp>
494 boost::shared_ptr<CommonData> common_data_ptr,
496 boost::shared_ptr<Range> contact_range_ptr =
nullptr);
497 MoFEMErrorCode doWork(
int side, EntityType
type,
EntData &data);
509template <
int DIM,
typename BoundaryEleOp>
512 MoFEMErrorCode doWork(
int side, EntityType
type,
EntData &data);
524template <
int DIM,
typename AssemblyBoundaryEleOp>
528 boost::shared_ptr<CommonData> common_data_ptr,
530 MoFEMErrorCode iNtegrate(EntitiesFieldData::EntData &data);
541template <
int DIM,
typename AssemblyBoundaryEleOp>
545 const std::string col_field_name,
546 boost::shared_ptr<CommonData> common_data_ptr,
548 MoFEMErrorCode iNtegrate(EntitiesFieldData::EntData &row_data,
549 EntitiesFieldData::EntData &col_data);
561template <
int DIM,
typename AssemblyBoundaryEleOp>
565 const std::string row_field_name,
const std::string col_field_name,
566 boost::shared_ptr<CommonData> common_data_ptr,
568 MoFEMErrorCode iNtegrate(EntitiesFieldData::EntData &row_data,
569 EntitiesFieldData::EntData &col_data);
581 template <
int DIM, IntegrationType I>
585 template <
int DIM, IntegrationType I>
589 template <
int DIM, IntegrationType I>
597 template <
int DIM, IntegrationType I>
601 template <
int DIM, IntegrationType I>
605 template <
int DIM, IntegrationType I>
612 constexpr auto eps = std::numeric_limits<float>::epsilon();
613 if (std::abs(x) <
eps)
621inline double w(
const double sdf,
const double tn) {
639template <
int DIM,
typename BoundaryEleOp>
642 boost::shared_ptr<CommonData> common_data_ptr,
double scale,
645 commonDataPtr(common_data_ptr), scaleTraction(
scale),
648template <
int DIM,
typename BoundaryEleOp>
657 auto t_w = BoundaryEleOp::getFTensor0IntegrationWeight();
658 auto t_traction = getFTensor1FromMat<DIM>(commonDataPtr->contactTraction);
659 auto t_coords = BoundaryEleOp::getFTensor1CoordsAtGaussPts();
661 const auto nb_gauss_pts = BoundaryEleOp::getGaussPts().size2();
662 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
663 double jacobian = 1.;
664 if (isAxisymmetric) {
665 jacobian = 2. * M_PI * t_coords(0);
667 const double alpha = t_w * jacobian * BoundaryEleOp::getMeasure();
668 t_sum_t(
i) += alpha * t_traction(
i);
674 t_sum_t(
i) *= scaleTraction;
676 constexpr int ind[] = {0, 1, 2};
677 CHKERR VecSetValues(commonDataPtr->totalTraction, 3, ind, &t_sum_t(0),
682template <
int DIM,
typename BoundaryEleOp>
686 boost::shared_ptr<Range> contact_range_ptr)
689 contactRange(contact_range_ptr) {}
691template <
int DIM,
typename BoundaryEleOp>
697 auto fe_type = BoundaryEleOp::getFEType();
699 const auto fe_ent = BoundaryEleOp::getFEEntityHandle();
701 if (contactRange->find(fe_ent) != contactRange->end()) {
706 auto t_w = BoundaryEleOp::getFTensor0IntegrationWeight();
707 auto t_traction = getFTensor1FromMat<DIM>(commonDataPtr->contactTraction);
708 auto t_coords = BoundaryEleOp::getFTensor1CoordsAtGaussPts();
710 auto t_grad = getFTensor2FromMat<DIM, DIM>(commonDataPtr->contactDispGrad);
711 auto t_normal_at_pts = BoundaryEleOp::getFTensor1NormalsAtGaussPts();
713 const auto nb_gauss_pts = BoundaryEleOp::getGaussPts().size2();
715 BoundaryEleOp::getFTensor1CoordsAtGaussPts(),
716 getFTensor1FromMat<DIM>(commonDataPtr->contactDisp), nb_gauss_pts);
718 BoundaryEleOp::getFTensor1NormalsAtGaussPts(), nb_gauss_pts);
720 auto ts_time = BoundaryEleOp::getTStime();
721 auto ts_time_step = BoundaryEleOp::getTStimeStep();
724 surfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
725 m_spatial_coords, m_normals_at_pts, block_id);
726 auto m_grad_sdf = gradSurfaceDistanceFunction(
727 ts_time_step, ts_time, nb_gauss_pts, m_spatial_coords, m_normals_at_pts,
729 auto t_sdf = getFTensor0FromVec(v_sdf);
730 auto t_grad_sdf = getFTensor1FromMat<3>(m_grad_sdf);
731 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
732 double jacobian = 1.;
733 if (isAxisymmetric) {
734 jacobian = 2. * M_PI * t_coords(0);
736 auto tn = -t_traction(
i) * t_grad_sdf(
i);
738 double alpha = t_w * jacobian;
744 F(
i,
j) = t_grad(
i,
j) + kronecker_delta(
i,
j);
745 auto det = determinantTensor(
F);
746 CHKERR invertTensor(
F, det, invF);
747 t_normal_current(
i) = det * (invF(
j,
i) * t_normal_at_pts(
j));
749 alpha *= sqrt(t_normal_current(
i) * t_normal_current(
i));
751 if (fe_type == MBTRI) {
767 constexpr int ind[] = {3, 4};
768 CHKERR VecSetValues(commonDataPtr->totalTraction, 2, ind, &t_sum_a(0),
774template <
int DIM,
typename BoundaryEleOp>
776 boost::shared_ptr<CommonData> common_data_ptr)
778 commonDataPtr(common_data_ptr) {}
780template <
int DIM,
typename BoundaryEleOp>
786 const auto nb_gauss_pts = BoundaryEleOp::getGaussPts().size2();
787 auto &sdf_vec = commonDataPtr->sdfVals;
788 auto &grad_mat = commonDataPtr->gradsSdf;
789 auto &hess_mat = commonDataPtr->hessSdf;
790 auto &constraint_vec = commonDataPtr->constraintVals;
791 auto &contactTraction_mat = commonDataPtr->contactTraction;
793 sdf_vec.resize(nb_gauss_pts,
false);
794 grad_mat.resize(nb_gauss_pts, DIM,
false);
795 hess_mat.resize(nb_gauss_pts, (DIM * (DIM + 1)) / 2,
false);
796 constraint_vec.resize(nb_gauss_pts,
false);
798 auto t_traction = getFTensor1FromMat<DIM>(contactTraction_mat);
800 auto t_sdf = getFTensor0FromVec(sdf_vec);
801 auto t_grad_sdf = getFTensor1FromMat<DIM>(grad_mat);
802 auto t_hess_sdf = getFTensor2SymmetricFromMat<DIM>(hess_mat);
803 auto t_constraint = getFTensor0FromVec(constraint_vec);
805 auto t_disp = getFTensor1FromMat<DIM>(commonDataPtr->contactDisp);
809 auto ts_time = BoundaryEleOp::getTStime();
810 auto ts_time_step = BoundaryEleOp::getTStimeStep();
813 BoundaryEleOp::getFTensor1CoordsAtGaussPts(),
814 getFTensor1FromMat<DIM>(commonDataPtr->contactDisp), nb_gauss_pts);
816 BoundaryEleOp::getFTensor1NormalsAtGaussPts(), nb_gauss_pts);
822 surfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
823 m_spatial_coords, m_normals_at_pts, block_id);
826 gradSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
827 m_spatial_coords, m_normals_at_pts, block_id);
830 hessSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
831 m_spatial_coords, m_normals_at_pts, block_id);
833 auto t_sdf_v = getFTensor0FromVec(v_sdf);
834 auto t_grad_sdf_v = getFTensor1FromMat<3>(m_grad_sdf);
835 auto t_hess_sdf_v = getFTensor2SymmetricFromMat<3>(m_hess_sdf);
849 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
851 auto tn = -t_traction(
i) * t_grad_sdf_v(
i);
855 t_grad_sdf(
i) = t_grad_sdf_v(
i);
856 t_hess_sdf(
i,
j) = t_hess_sdf_v(
i,
j);
865template <
int DIM,
typename AssemblyBoundaryEleOp>
868 boost::shared_ptr<CommonData> common_data_ptr,
874template <
int DIM,
typename AssemblyBoundaryEleOp>
877 EntitiesFieldData::EntData &data) {
885 const size_t nb_gauss_pts = AssemblyBoundaryEleOp::getGaussPts().size2();
887 auto &nf = AssemblyBoundaryEleOp::locF;
889 auto t_w = AssemblyBoundaryEleOp::getFTensor0IntegrationWeight();
890 auto t_disp = getFTensor1FromMat<DIM>(commonDataPtr->contactDisp);
891 auto t_traction = getFTensor1FromMat<DIM>(commonDataPtr->contactTraction);
892 auto t_coords = AssemblyBoundaryEleOp::getFTensor1CoordsAtGaussPts();
894 size_t nb_base_functions = data.getN().size2() / 3;
895 auto t_base = data.getFTensor1N<3>();
898 BoundaryEleOp::getFTensor1CoordsAtGaussPts(),
899 getFTensor1FromMat<DIM>(commonDataPtr->contactDisp), nb_gauss_pts);
901 BoundaryEleOp::getFTensor1NormalsAtGaussPts(), nb_gauss_pts);
903 auto t_normal = getFTensor1FromMat<3>(m_normals_at_pts);
905 auto ts_time = AssemblyBoundaryEleOp::getTStime();
906 auto ts_time_step = AssemblyBoundaryEleOp::getTStimeStep();
912 surfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
913 m_spatial_coords, m_normals_at_pts, block_id);
916 gradSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
917 m_spatial_coords, m_normals_at_pts, block_id);
919 auto t_sdf = getFTensor0FromVec(v_sdf);
920 auto t_grad_sdf = getFTensor1FromMat<3>(m_grad_sdf);
922 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
924 auto t_nf = getFTensor1FromPtr<DIM>(&nf[0]);
926 double jacobian = 1.;
927 if (isAxisymmetric) {
928 jacobian = 2. * M_PI * t_coords(0);
930 const double alpha = t_w * jacobian * AssemblyBoundaryEleOp::getMeasure();
932 auto tn = -t_traction(
i) * t_grad_sdf(
i);
936 t_cP(
i,
j) = (
c * t_grad_sdf(
i)) * t_grad_sdf(
j);
938 t_cQ(
i,
j) = kronecker_delta(
i,
j) - t_cP(
i,
j);
947 t_cP(
i,
j) * t_disp(
j) +
948 c * (t_sdf * t_grad_sdf(
i));
951 for (; bb != AssemblyBoundaryEleOp::nbRows / DIM; ++bb) {
952 const double beta = alpha * (t_base(
i) * t_normal(
i));
953 t_nf(
i) -= beta * t_rhs(
i);
958 for (; bb < nb_base_functions; ++bb)
973template <
int DIM,
typename AssemblyBoundaryEleOp>
976 const std::string col_field_name,
977 boost::shared_ptr<CommonData> common_data_ptr,
982 AssemblyBoundaryEleOp::sYmm =
false;
985template <
int DIM,
typename AssemblyBoundaryEleOp>
988 EntitiesFieldData::EntData &row_data,
989 EntitiesFieldData::EntData &col_data) {
996 const size_t nb_gauss_pts = AssemblyBoundaryEleOp::getGaussPts().size2();
997 auto &locMat = AssemblyBoundaryEleOp::locMat;
999 auto t_traction = getFTensor1FromMat<DIM>(commonDataPtr->contactTraction);
1000 auto t_coords = AssemblyBoundaryEleOp::getFTensor1CoordsAtGaussPts();
1002 auto t_w = AssemblyBoundaryEleOp::getFTensor0IntegrationWeight();
1003 auto t_row_base = row_data.getFTensor1N<3>();
1004 size_t nb_face_functions = row_data.getN().size2() / 3;
1007 BoundaryEleOp::getFTensor1CoordsAtGaussPts(),
1008 getFTensor1FromMat<DIM>(commonDataPtr->contactDisp), nb_gauss_pts);
1010 BoundaryEleOp::getFTensor1NormalsAtGaussPts(), nb_gauss_pts);
1012 auto t_normal = getFTensor1FromMat<3>(m_normals_at_pts);
1014 auto ts_time = AssemblyBoundaryEleOp::getTStime();
1015 auto ts_time_step = AssemblyBoundaryEleOp::getTStimeStep();
1021 surfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
1022 m_spatial_coords, m_normals_at_pts, block_id);
1025 gradSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
1026 m_spatial_coords, m_normals_at_pts, block_id);
1029 hessSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
1030 m_spatial_coords, m_normals_at_pts, block_id);
1032 auto t_sdf = getFTensor0FromVec(v_sdf);
1033 auto t_grad_sdf = getFTensor1FromMat<3>(m_grad_sdf);
1034 auto t_hess_sdf = getFTensor2SymmetricFromMat<3>(m_hess_sdf);
1036 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1038 double jacobian = 1.;
1039 if (isAxisymmetric) {
1040 jacobian = 2. * M_PI * t_coords(0);
1042 const double alpha = t_w * jacobian * AssemblyBoundaryEleOp::getMeasure();
1044 auto tn = -t_traction(
i) * t_grad_sdf(
i);
1048 t_cP(
i,
j) = (
c * t_grad_sdf(
i)) * t_grad_sdf(
j);
1050 t_cQ(
i,
j) = kronecker_delta(
i,
j) - t_cP(
i,
j);
1053 t_res_dU(
i,
j) = kronecker_delta(
i,
j) + t_cP(
i,
j);
1058 (t_hess_sdf(
i,
j) * (t_grad_sdf(
k) * t_traction(
k)) +
1059 t_grad_sdf(
i) * t_hess_sdf(
k,
j) * t_traction(
k)) +
1060 c * t_sdf * t_hess_sdf(
i,
j);
1064 for (; rr != AssemblyBoundaryEleOp::nbRows / DIM; ++rr) {
1066 auto t_mat = getFTensor2FromArray<DIM, DIM, DIM>(locMat, DIM * rr);
1068 const double row_base = t_row_base(
i) * t_normal(
i);
1070 auto t_col_base = col_data.getFTensor0N(gg, 0);
1071 for (
size_t cc = 0; cc != AssemblyBoundaryEleOp::nbCols / DIM; ++cc) {
1072 const double beta = alpha * row_base * t_col_base;
1074 t_mat(
i,
j) -= beta * t_res_dU(
i,
j);
1082 for (; rr < nb_face_functions; ++rr)
1097template <
int DIM,
typename AssemblyBoundaryEleOp>
1100 const std::string row_field_name,
const std::string col_field_name,
1105 AssemblyBoundaryEleOp::sYmm =
false;
1108template <
int DIM,
typename AssemblyBoundaryEleOp>
1111 iNtegrate(EntitiesFieldData::EntData &row_data,
1112 EntitiesFieldData::EntData &col_data) {
1119 const size_t nb_gauss_pts = AssemblyBoundaryEleOp::getGaussPts().size2();
1120 auto &locMat = AssemblyBoundaryEleOp::locMat;
1122 auto t_traction = getFTensor1FromMat<DIM>(commonDataPtr->contactTraction);
1123 auto t_coords = AssemblyBoundaryEleOp::getFTensor1CoordsAtGaussPts();
1125 auto t_w = AssemblyBoundaryEleOp::getFTensor0IntegrationWeight();
1126 auto t_row_base = row_data.getFTensor1N<3>();
1127 size_t nb_face_functions = row_data.getN().size2() / 3;
1130 BoundaryEleOp::getFTensor1CoordsAtGaussPts(),
1131 getFTensor1FromMat<DIM>(commonDataPtr->contactDisp), nb_gauss_pts);
1133 BoundaryEleOp::getFTensor1NormalsAtGaussPts(), nb_gauss_pts);
1135 auto t_normal = getFTensor1FromMat<3>(m_normals_at_pts);
1137 auto ts_time = AssemblyBoundaryEleOp::getTStime();
1138 auto ts_time_step = AssemblyBoundaryEleOp::getTStimeStep();
1144 surfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
1145 m_spatial_coords, m_normals_at_pts, block_id);
1148 gradSurfaceDistanceFunction(ts_time_step, ts_time, nb_gauss_pts,
1149 m_spatial_coords, m_normals_at_pts, block_id);
1151 auto t_sdf = getFTensor0FromVec(v_sdf);
1152 auto t_grad_sdf = getFTensor1FromMat<3>(m_grad_sdf);
1154 for (
size_t gg = 0; gg != nb_gauss_pts; ++gg) {
1156 double jacobian = 1.;
1157 if (isAxisymmetric) {
1158 jacobian = 2. * M_PI * t_coords(0);
1160 const double alpha = t_w * jacobian * AssemblyBoundaryEleOp::getMeasure();
1162 auto tn = -t_traction(
i) * t_grad_sdf(
i);
1166 t_cP(
i,
j) = (
c * t_grad_sdf(
i)) * t_grad_sdf(
j);
1168 t_cQ(
i,
j) = kronecker_delta(
i,
j) - t_cP(
i,
j);
1174 for (; rr != AssemblyBoundaryEleOp::nbRows / DIM; ++rr) {
1176 auto t_mat = getFTensor2FromArray<DIM, DIM, DIM>(locMat, DIM * rr);
1177 const double row_base = t_row_base(
i) * t_normal(
i);
1179 auto t_col_base = col_data.getFTensor1N<3>(gg, 0);
1180 for (
size_t cc = 0; cc != AssemblyBoundaryEleOp::nbCols / DIM; ++cc) {
1181 const double col_base = t_col_base(
i) * t_normal(
i);
1182 const double beta = alpha * row_base * col_base;
1184 t_mat(
i,
j) -= beta * t_res_dt(
i,
j);
1192 for (; rr < nb_face_functions; ++rr)
1206template <
int DIM, AssemblyType A, IntegrationType I,
typename DomainEleOp>
1208 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1212 using B =
typename FormsIntegrators<DomainEleOp>::template Assembly<
1214 using OpMixDivURhs =
typename B::template OpMixDivTimesU<3, DIM, DIM>;
1215 using OpMixDivUCylRhs =
1216 typename B::template OpMixDivTimesU<3, DIM, DIM, CYLINDRICAL>;
1218 using OpMixLambdaGradURhs =
typename B::template OpMixTensorTimesGradU<DIM>;
1219 using OpMixUTimesDivLambdaRhs =
1220 typename B::template OpMixVecTimesDivLambda<SPACE_DIM>;
1221 using OpMixUTimesLambdaRhs =
1224 auto common_data_ptr = boost::make_shared<ContactOps::CommonData>();
1225 auto mat_grad_ptr = boost::make_shared<MatrixDouble>();
1226 auto div_stress_ptr = boost::make_shared<MatrixDouble>();
1227 auto contact_stress_ptr = boost::make_shared<MatrixDouble>();
1232 return 2. * M_PI * r;
1237 pip.push_back(
new OpCalculateVectorFieldValues<DIM>(
1238 u, common_data_ptr->contactDispPtr()));
1240 new OpCalculateHVecTensorField<DIM, DIM>(sigma, contact_stress_ptr));
1244 new OpCalculateHVecTensorDivergence<DIM, DIM>(sigma, div_stress_ptr));
1246 pip.push_back(
new OpCalculateHVecTensorDivergence<DIM, DIM, CYLINDRICAL>(
1247 sigma, div_stress_ptr));
1250 pip.push_back(
new OpCalculateVectorFieldGradient<DIM, DIM>(u, mat_grad_ptr));
1254 new OpMixDivURhs(sigma, common_data_ptr->contactDispPtr(), jacobian));
1256 pip.push_back(
new OpMixDivUCylRhs(sigma, common_data_ptr->contactDispPtr(),
1260 pip.push_back(
new OpMixLambdaGradURhs(sigma, mat_grad_ptr, jacobian));
1261 pip.push_back(
new OpMixUTimesDivLambdaRhs(u, div_stress_ptr, jacobian));
1262 pip.push_back(
new OpMixUTimesLambdaRhs(u, contact_stress_ptr, jacobian));
1268 using OpMixLhs::OpMixLhs;
1269 MoFEMErrorCode
doWork(
int row_side,
int col_side, EntityType row_type,
1270 EntityType col_type,
1271 EntitiesFieldData::EntData &row_data,
1272 EntitiesFieldData::EntData &col_data) {
1274 auto side_fe_entity = OpMixLhs::getSidePtrFE()->getFEEntityHandle();
1275 auto side_fe_data = OpMixLhs::getSideEntity(row_side, row_type);
1277 if (side_fe_entity == side_fe_data) {
1278 CHKERR OpMixLhs::doWork(row_side, col_side, row_type, col_type, row_data,
1285template <
int DIM, AssemblyType A, IntegrationType I,
typename DomainEle>
1288 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1289 std::string fe_domain_name, std::string sigma, std::string u,
1290 std::string geom, ForcesAndSourcesCore::RuleHookFun rule,
1294 using DomainEleOp =
typename DomainEle::UserDataOperator;
1296 auto op_loop_side =
new OpLoopSide<DomainEle>(
1297 m_field, fe_domain_name, DIM, Sev::noisy,
1298 boost::make_shared<ForcesAndSourcesCore::UserDataOperator::AdjCache>());
1299 pip.push_back(op_loop_side);
1304 using B =
typename FormsIntegrators<DomainEleOp>::template Assembly<
1307 using OpMixDivULhs =
typename B::template OpMixDivTimesVec<DIM>;
1308 using OpMixDivUCylLhs =
1309 typename B::template OpMixDivTimesVec<DIM, CYLINDRICAL>;
1310 using OpLambdaGraULhs =
typename B::template OpMixTensorTimesGrad<DIM>;
1316 auto unity = []() {
return 1; };
1320 return 2. * M_PI * r;
1326 op_loop_side->getOpPtrVector().push_back(
1327 new OpMixDivULhsSide(sigma, u, unity, jacobian,
true));
1329 op_loop_side->getOpPtrVector().push_back(
1330 new OpMixDivUCylLhsSide(sigma, u, unity, jacobian,
true));
1332 op_loop_side->getOpPtrVector().push_back(
1333 new OpLambdaGraULhsSide(sigma, u, unity, jacobian,
true));
1335 op_loop_side->getSideFEPtr()->getRuleHook = rule;
1339template <
int DIM, AssemblyType A, IntegrationType I,
typename BoundaryEleOp>
1341 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1347 auto common_data_ptr = boost::make_shared<ContactOps::CommonData>();
1349 pip.push_back(
new OpCalculateVectorFieldValues<DIM>(
1350 u, common_data_ptr->contactDispPtr()));
1351 pip.push_back(
new OpCalculateHVecTensorTrace<DIM, BoundaryEleOp>(
1352 sigma, common_data_ptr->contactTractionPtr()));
1354 new typename C::template Assembly<A>::template OpConstrainBoundaryLhs_dU<
1356 pip.push_back(
new typename C::template Assembly<A>::
1357 template OpConstrainBoundaryLhs_dTraction<DIM, GAUSS>(
1363template <
int DIM, AssemblyType A, IntegrationType I,
typename BoundaryEleOp>
1365 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1371 auto common_data_ptr = boost::make_shared<ContactOps::CommonData>();
1373 pip.push_back(
new OpCalculateVectorFieldValues<DIM>(
1374 u, common_data_ptr->contactDispPtr()));
1375 pip.push_back(
new OpCalculateHVecTensorTrace<DIM, BoundaryEleOp>(
1376 sigma, common_data_ptr->contactTractionPtr()));
1378 new typename C::template Assembly<A>::template OpConstrainBoundaryRhs<
1384template <
int DIM, IntegrationType I,
typename BoundaryEleOp>
1386 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1392 auto common_data_ptr = boost::make_shared<ContactOps::CommonData>();
1393 pip.push_back(
new OpCalculateHVecTensorTrace<DIM, BoundaryEleOp>(
1394 sigma, common_data_ptr->contactTractionPtr()));
1395 pip.push_back(
new typename C::template OpAssembleTotalContactTraction<DIM, I>(
1401template <
int DIM, IntegrationType I,
typename BoundaryEleOp>
1403 boost::ptr_deque<ForcesAndSourcesCore::UserDataOperator> &pip,
1404 OpLoopSide<SideEle> *op_loop_side, std::string sigma, std::string u,
1406 boost::shared_ptr<Range> contact_range_ptr =
nullptr) {
1410 auto common_data_ptr = boost::make_shared<ContactOps::CommonData>();
1412 op_loop_side->getOpPtrVector().push_back(
1413 new OpCalculateVectorFieldGradient<SPACE_DIM, SPACE_DIM>(
1414 "U", common_data_ptr->contactDispGradPtr()));
1416 if (contact_range_ptr) {
1417 pip.push_back(
new OpCalculateVectorFieldValues<DIM>(
1418 u, common_data_ptr->contactDispPtr()));
1419 pip.push_back(
new OpCalculateHVecTensorTrace<DIM, BoundaryEleOp>(
1420 sigma, common_data_ptr->contactTractionPtr()));
1421 pip.push_back(op_loop_side);
1422 pip.push_back(
new typename C::template OpAssembleTotalContactArea<DIM, I>(
#define CHK_THROW_MESSAGE(err, msg)
Check and throw MoFEM exception.
#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_OPERATION_UNSUCCESSFUL
@ MOFEM_DATA_INCONSISTENCY
#define MoFEMFunctionReturn(a)
Last executable line of each PETSc function used for error handling. Replaces return()
#define CHKERR
Inline error check.
FTensor::Index< 'i', SPACE_DIM > i
const double c
speed of light (cm/ns)
FTensor::Index< 'l', 3 > l
FTensor::Index< 'j', 3 > j
FTensor::Index< 'k', 3 > k
PetscErrorCode MoFEMErrorCode
MoFEM/PETSc error code.
hess_sdf(delta_t, t, x, y, z, tx, ty, tz, block_id)
grad_sdf(delta_t, t, x, y, z, tx, ty, tz, block_id)
FormsIntegrators< DomainEleOp >::Assembly< A >::LinearForm< I >::OpGradTimesTensor< 1, FIELD_DIM, SPACE_DIM > OpGradTimesTensor
constexpr double t
plate stiffness
constexpr auto field_name
virtual MPI_Comm & get_comm() const =0
virtual int get_comm_rank() const =0
Deprecated interface functions.