1160 auto &m_field = getPtrFE()->mField;
1161 auto fe_ent = getNumeredEntFiniteElementPtr()->getEnt();
1170 const auto nb_gauss_pts = getGaussPts().size2();
1172 auto t_disp_h1 = getFTensor1FromMat<3>(*
uH1Ptr);
1173 auto t_coords = getFTensor1CoordsAtGaussPts();
1174 auto t_traction = getFTensor1FromMat<3>(
commonDataPtr->contactTraction);
1182 auto get_ele_centre = [
i](
auto t_ele_coords) {
1184 t_ele_center(
i) = 0;
1185 for (
int nn = 0; nn != 3; nn++) {
1186 t_ele_center(
i) += t_ele_coords(
i);
1189 t_ele_center(
i) /= 3;
1190 return t_ele_center;
1193 auto get_ele_radius = [
i](
auto t_ele_center,
auto t_ele_coords) {
1195 t_n0(
i) = t_ele_center(
i) - t_ele_coords(
i);
1199 auto get_face_conn = [
this](
auto face) {
1203 face, conn, num_nodes,
true),
1205 if (num_nodes != 3) {
1211 auto get_face_coords = [
this](
auto conn) {
1212 std::array<double, 9> coords;
1217 auto get_closet_face = [
this](
auto *point_ptr,
auto r) {
1219 std::vector<EntityHandle> faces_out;
1223 "get closest faces");
1227 auto get_faces_out = [
this](
auto *point_ptr,
auto *unit_ray_ptr,
auto radius,
1229 std::vector<double> distances_out;
1230 std::vector<EntityHandle> faces_out;
1235 point_ptr, unit_ray_ptr, &radius),
1237 "get closest faces");
1238 return std::make_pair(faces_out, distances_out);
1241 auto get_normal = [](
auto &ele_coords) {
1243 Tools::getTriNormal(ele_coords.data(), &t_normal(0));
1247 auto make_map = [&](
auto &face_out,
auto &face_dist,
auto &t_ray_point,
1248 auto &t_unit_ray,
auto &t_master_coord) {
1251 std::map<double, EntityHandle>
m;
1252 for (
auto ii = 0; ii != face_out.size(); ++ii) {
1253 auto face_conn = get_face_conn(face_out[ii]);
1256 t_face_normal.normalize();
1258 t_x(
i) = t_ray_point(
i) + t_unit_ray(
i) * face_dist[ii];
1261 t_x(
i) * t_face_normal(
j) - t_master_coord(
i) * t_unit_ray(
j);
1262 if (t_unit_ray(
i) * t_face_normal(
i) > std::cos(M_PI / 3)) {
1263 auto dot = std::sqrt(t_chi(
i,
j) * t_chi(
i,
j));
1264 m[dot] = face_out[ii];
1270 auto get_tag_data = [
this](
auto tag,
auto face,
auto &vec) {
1274 vec.resize(tag_length);
1280 auto create_tag = [
this](
const std::string tag_name,
const int size) {
1281 double def_VAL[] = {0, 0, 0, 0, 0, 0, 0, 0, 0};
1285 tag_name.c_str(), size, MB_TYPE_DOUBLE,
th,
1286 MB_TAG_CREAT | MB_TAG_SPARSE, def_VAL);
1293 auto set_float_precision = [](
const double x) {
1294 if (std::abs(x) < std::numeric_limits<float>::epsilon())
1301 auto save_scal_tag = [&](
auto &
th,
auto v,
const int gg) {
1304 v = set_float_precision(
v);
1311 auto get_fe_adjacencies = [
this](
auto fe_ent) {
1314 &fe_ent, 1, 2,
false, adj_faces, moab::Interface::UNION),
1316 std::set<int> adj_ids;
1317 for (
auto f : adj_faces) {
1323 auto get_face_id = [
this](
auto face) {
1332 auto get_body_id = [
this](
auto face) {
1341 auto get_face_part = [
this](
auto face) {
1342 ParallelComm *pcomm_post_proc_mesh = ParallelComm::get_pcomm(
1346 pcomm_post_proc_mesh->part_tag(), &face, 1, &part) == MB_SUCCESS) {
1352 auto check_face = [&](
auto face,
auto fe_id,
auto part) {
1353 auto face_id = get_face_id(face);
1354 auto face_part = get_face_part(face);
1355 if (face_id == fe_id && face_part == part)
1363 auto save_vec_tag = [&](
auto &
th,
auto &t_d,
const int gg) {
1367 for (
auto &
a :
v.data())
1368 a = set_float_precision(
a);
1370 &*
v.data().begin());
1375 Tag th_mark = create_tag(
"contact_mark", 1);
1376 Tag th_mark_slave = create_tag(
"contact_mark_slave", 1);
1377 Tag th_body_id = create_tag(
"contact_body_id", 1);
1378 Tag th_gap = create_tag(
"contact_gap", 1);
1379 Tag th_tn_master = create_tag(
"contact_tn_master", 1);
1380 Tag th_tn_slave = create_tag(
"contact_tn_slave", 1);
1381 Tag th_contact_traction = create_tag(
"contact_traction", 3);
1382 Tag th_contact_traction_master = create_tag(
"contact_traction_master", 3);
1383 Tag th_contact_traction_slave = create_tag(
"contact_traction_slave", 3);
1384 Tag th_c = create_tag(
"contact_c", 1);
1385 Tag th_normal = create_tag(
"contact_normal", 3);
1386 Tag th_dist = create_tag(
"contact_dip", 3);
1388 auto t_ele_centre = get_ele_centre(getFTensor1Coords());
1389 auto ele_radius = get_ele_radius(t_ele_centre, getFTensor1Coords());
1395 auto adj_fe_ids = get_fe_adjacencies(fe_ent);
1397 for (
auto gg = 0; gg != nb_gauss_pts; ++gg) {
1400 t_spatial_coords(
i) = t_coords(
i) + t_disp_h1(
i);
1403 CHKERR save_vec_tag(th_contact_traction, t_traction, gg);
1406 auto faces_close = get_closet_face(&t_spatial_coords(0), ele_radius);
1407 for (
auto face_close : faces_close) {
1408 if (check_face(face_close, fe_id, m_field.get_comm_rank())) {
1410 auto body_id = get_body_id(face_close);
1412 auto master_face_conn = get_face_conn(face_close);
1413 std::array<double, 9> master_coords;
1416 master_coords.data());
1417 std::array<double, 9> master_traction;
1420 master_traction.data());
1421 auto t_normal_face_close = get_normal(master_coords);
1422 t_normal_face_close.normalize();
1426 CHKERR save_scal_tag(th_mark,
m, gg);
1427 CHKERR save_scal_tag(th_body_id,
static_cast<double>(body_id), gg);
1428 CHKERR save_vec_tag(th_normal, t_normal_face_close, gg);
1429 CHKERR save_vec_tag(th_contact_traction, t_traction, gg);
1433 t_unit_ray(
i) = -t_normal_face_close(
i);
1436 t_spatial_coords(
i) -
1439 constexpr
double eps = 1e-3;
1440 auto [faces_out, faces_dist] =
1441 get_faces_out(&t_ray_point(0), &t_unit_ray(0),
1445 auto m = make_map(faces_out, faces_dist, t_ray_point, t_unit_ray,
1447 for (
auto m_it =
m.begin(); m_it !=
m.end(); ++m_it) {
1448 auto face = m_it->second;
1449 if (face != face_close) {
1453 body_id != get_body_id(face) &&
1454 (adj_fe_ids.find(get_face_id(face)) == adj_fe_ids.end() ||
1455 get_face_part(face) != m_field.get_comm_rank())
1459 shadow_vec.push_back(ContactTree::FaceData());
1460 shadow_vec.back().gaussPtNb = gg;
1462 auto slave_face_conn = get_face_conn(face);
1463 std::array<double, 9> slave_coords;
1466 slave_coords.data());
1467 auto t_normal_face = get_normal(slave_coords);
1468 std::array<double, 9> slave_tractions;
1471 slave_tractions.data());
1473 auto t_master_point =
1474 getFTensor1FromPtr<3>(shadow_vec.back().masterPoint.data());
1475 auto t_slave_point =
1476 getFTensor1FromPtr<3>(shadow_vec.back().slavePoint.data());
1477 auto t_ray_point_data =
1478 getFTensor1FromPtr<3>(shadow_vec.back().rayPoint.data());
1479 auto t_unit_ray_data =
1480 getFTensor1FromPtr<3>(shadow_vec.back().unitRay.data());
1482 t_slave_point(
i) = t_ray_point(
i) + m_it->first * t_unit_ray(
i);
1484 auto eval_position = [&](
auto &&t_elem_coords,
auto &&t_point) {
1485 std::array<double, 2> loc_coords;
1487 Tools::getLocalCoordinatesOnReferenceThreeNodeTri(
1488 &t_elem_coords(0, 0), &t_point(0), 1,
1490 "get local coords");
1499 t_point_out(
i) = t_shape_fun(
j) * t_elem_coords(
j,
i);
1503 auto t_master_point_updated = eval_position(
1504 getFTensor2FromPtr<3, 3>(master_coords.data()),
1505 getFTensor1FromPtr<3>(shadow_vec.back().masterPoint.data()));
1506 t_master_point(
i) = t_master_point_updated(
i);
1508 auto t_slave_point_updated = eval_position(
1509 getFTensor2FromPtr<3, 3>(slave_coords.data()),
1510 getFTensor1FromPtr<3>(shadow_vec.back().slavePoint.data()));
1511 t_slave_point(
i) = t_slave_point_updated(
i);
1513 t_ray_point_data(
i) = t_ray_point(
i);
1514 t_unit_ray_data(
i) = t_unit_ray(
i);
1516 std::copy(master_coords.begin(), master_coords.end(),
1517 shadow_vec.back().masterPointNodes.begin());
1518 std::copy(master_traction.begin(), master_traction.end(),
1519 shadow_vec.back().masterTractionNodes.begin());
1520 std::copy(slave_coords.begin(), slave_coords.end(),
1521 shadow_vec.back().slavePointNodes.begin());
1522 std::copy(slave_tractions.begin(), slave_tractions.end(),
1523 shadow_vec.back().slaveTractionNodes.begin());
1525 shadow_vec.back().eleRadius = ele_radius;
1535 auto [gap, tn_master, tn_slave,
c, t_master_traction,
1537 multiGetGap(&(shadow_vec.back()), t_spatial_coords);
1539 t_gap_vec(
i) = t_slave_point(
i) - t_spatial_coords(
i);
1540 CHKERR save_scal_tag(th_gap, gap, gg);
1541 CHKERR save_scal_tag(th_tn_master, tn_master, gg);
1542 CHKERR save_scal_tag(th_tn_slave, tn_slave, gg);
1543 CHKERR save_scal_tag(th_c,
c, gg);
1545 CHKERR save_scal_tag(th_mark_slave,
m, gg);
1546 CHKERR save_vec_tag(th_dist, t_gap_vec, gg);
1547 CHKERR save_vec_tag(th_contact_traction_master,
1548 t_master_traction, gg);
1549 CHKERR save_vec_tag(th_contact_traction_slave, t_slave_traction,