8#include <ipc/barrier/adaptive_stiffness.hpp>
9#include <ipc/utils/world_bbox_diagonal_length.hpp>
19 const double avg_mass,
20 const bool use_area_weighting,
21 const bool use_improved_max_operator,
22 const bool use_physical_barrier,
23 const bool use_adaptive_barrier_stiffness,
24 const bool is_time_dependent,
25 const bool enable_shape_derivatives,
26 const ipc::BroadPhaseMethod broad_phase_method,
27 const double ccd_tolerance,
28 const int ccd_max_iterations) :
ContactForm(collision_mesh, dhat, avg_mass, use_adaptive_barrier_stiffness, is_time_dependent, enable_shape_derivatives, broad_phase_method, ccd_tolerance, ccd_max_iterations), barrier_potential_(dhat, 1.0, use_physical_barrier)
46 ipc::NormalCollisions nonconvergent_constraints;
48 nonconvergent_constraints.set_use_area_weighting(
false);
49 nonconvergent_constraints.set_use_improved_max_approximator(
false);
50 nonconvergent_constraints.build(
62 double scaling_factor = 0;
63 if (!nonconvergent_constraints.empty())
72 scaling_factor = nonconvergent_potential / convergent_potential;
89 "Setting adaptive barrier stiffness to {} (max barrier stiffness: {})",
96 static Eigen::MatrixXd cached_displaced_surface;
97 if (cached_displaced_surface.size() == displaced_surface.size() && cached_displaced_surface == displaced_surface)
106 cached_displaced_surface = displaced_surface;
129 auto storage = utils::create_thread_storage<Eigen::VectorXd>(Eigen::VectorXd::Zero(num_vertices));
132 Eigen::VectorXd &local_storage = utils::get_local_thread_storage(storage, thread_id);
134 for (size_t i = start; i < end; i++)
137 const double potential = barrier_potential_(collision_set_[i], collision_set_[i].dof(V, E, F));
139 const int n_v = collision_set_[i].num_vertices();
140 const auto vis = collision_set_[i].vertex_ids(E, F);
141 for (int j = 0; j < n_v; j++)
143 assert(0 <= vis[j] && vis[j] < num_vertices);
144 local_storage[vis[j]] += potential / n_v;
149 Eigen::VectorXd out = Eigen::VectorXd::Zero(num_vertices);
150 for (
const auto &local_potential : storage)
152 out += local_potential;
155 Eigen::VectorXd out_full = Eigen::VectorXd::Zero(collision_mesh_.full_num_vertices());
156 for (
int i = 0; i < out.size(); i++)
157 out_full[collision_mesh_.to_full_vertex_id(i)] = out[i];
159 assert(std::abs(value_unweighted(
x) - out_full.sum()) < std::max(1e-10 * out_full.sum(), 1e-10));
164 void BarrierContactForm::first_derivative_unweighted(
const Eigen::VectorXd &
x, Eigen::VectorXd &gradv)
const
166 gradv = barrier_potential_.gradient(collision_set_, collision_mesh_, compute_displaced_surface(
x));
167 gradv = collision_mesh_.to_full_dof(gradv);
170 void BarrierContactForm::second_derivative_unweighted(
const Eigen::VectorXd &
x,
StiffnessMatrix &hessian)
const
174 ipc::PSDProjectionMethod psd_projection_method;
178 psd_projection_method = ipc::PSDProjectionMethod::CLAMP;
182 psd_projection_method = ipc::PSDProjectionMethod::NONE;
185 hessian = barrier_potential_.hessian(collision_set_, collision_mesh_, compute_displaced_surface(
x), psd_projection_method);
186 hessian = collision_mesh_.to_full_dof(hessian);
189 void BarrierContactForm::post_step(
const polysolve::nonlinear::PostStepData &data)
191 const Eigen::MatrixXd displaced_surface = compute_displaced_surface(data.x);
193 const double curr_distance = collision_set_.compute_minimum_distance(collision_mesh_, displaced_surface);
194 if (!std::isinf(curr_distance))
196 const double ratio = sqrt(curr_distance) / dhat();
197 const auto log_level = (ratio < 1e-6) ? spdlog::level::err : ((ratio < 1e-4) ? spdlog::level::warn : spdlog::level::debug);
198 polyfem::logger().log(log_level,
"Minimum distance during solve: {}, dhat: {}", sqrt(curr_distance), dhat());
201 if (data.iter_num == 0)
204 if (use_adaptive_barrier_stiffness_)
206 if (is_time_dependent_)
208 const double prev_barrier_stiffness = barrier_stiffness();
210 barrier_stiffness_ = ipc::update_barrier_stiffness(
211 prev_distance_, curr_distance, max_barrier_stiffness_,
212 barrier_stiffness(), ipc::world_bbox_diagonal_length(displaced_surface));
214 if (barrier_stiffness() != prev_barrier_stiffness)
217 "updated barrier stiffness from {:g} to {:g} (max barrier stiffness: )",
218 prev_barrier_stiffness, barrier_stiffness(), max_barrier_stiffness_);
228 prev_distance_ = curr_distance;
#define POLYFEM_SCOPED_TIMER(...)
void maybe_parallel_for(int size, const std::function< void(int, int, int)> &partial_for)
spdlog::logger & logger()
Retrieves the current logger.
Eigen::SparseMatrix< double, Eigen::ColMajor > StiffnessMatrix