35#include <polysolve/linear/Solver.hpp>
36#include <polysolve/nonlinear/Solver.hpp>
44 using namespace solver;
45 using namespace time_integrator;
50 const bool contact_dhat_was_explicit = clean_args[
"contact"].value(
"_dhat_was_explicit",
false);
51 clean_args[
"contact"].erase(
"_dhat_was_explicit");
73 logger().info(
"Loading obstacles...");
92 const Eigen::MatrixXd &solution,
103 const int actual_dim =
problem->is_scalar() ? 1 :
mesh_->dimension();
104 const auto ¶view_options =
args[
"output"][
"paraview"][
"options"];
105 const bool explicit_fields = !options.
fields.empty();
107 const auto has_field = [&](
const std::string &
name) {
108 return std::any_of(fields.begin(), fields.end(), [&](
const io::OutputField &field) {
109 return field.association == io::OutputField::Association::Point && field.name == name;
113 const auto append_collision_dof_field = [&](
const std::string &
name,
const Eigen::MatrixXd &dof_values) {
114 if (has_field(
name) || dof_values.size() <= 0)
118 if (values.rows() == sample.
points.rows())
122 const auto append_collision_form_force = [&](
const std::string &name,
const std::shared_ptr<solver::Form> &form) {
123 if (!form || !form->enabled() || sample.points.rows() != collision_mesh.rest_positions().rows())
126 Eigen::VectorXd force;
127 form->first_derivative(solution.col(0), force);
128 const double acceleration_scaling =
129 solve_data.time_integrator ? solve_data.time_integrator->acceleration_scaling() : 1;
130 force *= -1.0 / acceleration_scaling;
131 append_collision_dof_field(name, force);
134 if (paraview_options[
"forces"] && !problem->is_scalar())
136 const double s = solve_data.time_integrator ? solve_data.time_integrator->acceleration_scaling() : 1;
137 for (
const auto &[name, form] : solve_data.named_forms())
139 const std::string field_name = name +
"_forces";
140 if (!options.export_field(field_name))
143 Eigen::VectorXd force;
144 if (form && form->enabled())
146 form->first_derivative(solution, force);
151 force.setZero(solution.size());
153 append_collision_dof_field(field_name, force);
157 if (options.export_field(
"gradient_of_elastic_potential") && solve_data.elastic_form)
159 Eigen::VectorXd potential_grad;
160 solve_data.elastic_form->first_derivative(solution, potential_grad);
161 append_collision_dof_field(
"gradient_of_elastic_potential", potential_grad);
164 if (options.export_field(
"gradient_of_contact_potential") && solve_data.contact_form && solve_data.contact_form->weight() > 0)
166 Eigen::VectorXd potential_grad;
167 solve_data.contact_form->first_derivative(solution, potential_grad);
168 potential_grad *= -solve_data.contact_form->barrier_stiffness() / solve_data.contact_form->weight();
169 append_collision_dof_field(
"gradient_of_contact_potential", potential_grad);
172 if (options.export_field(
"displacement"))
173 append_collision_dof_field(
"displacement", solution);
174 if (options.export_field(
"solution"))
175 append_collision_dof_field(
"solution", solution);
177 if ((paraview_options[
"contact_forces"] || explicit_fields) && options.export_field(
"contact_forces"))
178 append_collision_form_force(
"contact_forces", solve_data.contact_form);
179 if ((paraview_options[
"friction_forces"] || explicit_fields) && options.export_field(
"friction_forces"))
180 append_collision_form_force(
"friction_forces", solve_data.friction_form);
181 if ((paraview_options[
"normal_adhesion_forces"] || explicit_fields) && options.export_field(
"normal_adhesion_forces"))
182 append_collision_form_force(
"normal_adhesion_forces", solve_data.normal_adhesion_form);
183 if ((paraview_options[
"tangential_adhesion_forces"] || explicit_fields) && options.export_field(
"tangential_adhesion_forces"))
184 append_collision_form_force(
"tangential_adhesion_forces", solve_data.tangential_adhesion_form);
187 && options.export_field(
"adaptive_dhat")
188 && args[
"contact"][
"use_gcp_formulation"]
189 && args[
"contact"][
"use_adaptive_dhat"])
191 const auto smooth_contact = std::dynamic_pointer_cast<solver::SmoothContactForm>(solve_data.contact_form);
194 const auto &set = smooth_contact->collision_set();
197 Eigen::VectorXd dhats(collision_mesh.num_edges());
198 for (
int e = 0;
e < dhats.size(); ++
e)
199 dhats(e) = set.get_edge_dhat(e);
204 Eigen::VectorXd dhats(collision_mesh.num_faces());
205 for (
int f = 0;
f < dhats.size(); ++
f)
206 dhats(f) = set.get_face_dhat(f);
209 Eigen::VectorXd vertex_dhats(collision_mesh.num_vertices());
210 for (
int v = 0; v < vertex_dhats.size(); ++v)
211 vertex_dhats(v) = set.get_vert_dhat(v);
220 void NonlinearElasticVarForm::build_basis(
mesh::Mesh &mesh,
const bool iso_parametric,
const json &args)
222 ElasticVarForm::build_basis(mesh, iso_parametric, args);
227 const int n_fe_bases =
space_.n_bases;
228 space_.n_bases += obstacle.n_vertices();
230 logger().info(
"Building collision mesh...");
231 build_collision_mesh(mesh, args);
232 preprocess_contact_parameters();
238 for (
int i = n_fe_bases; i <
space_.n_bases; ++i)
240 for (
int d = 0; d < mesh.
dimension(); ++d)
241 boundary_.boundary_nodes.push_back(i * mesh.
dimension() + d);
244 boundary_.normalize_boundary_nodes();
247 void NonlinearElasticVarForm::preprocess_contact_parameters()
249 if (!is_contact_enabled())
252 double min_boundary_edge_length = std::numeric_limits<double>::max();
253 for (
const auto &edge : collision_mesh.edges().rowwise())
255 const VectorNd v0 = collision_mesh.rest_positions().row(edge(0));
256 const VectorNd v1 = collision_mesh.rest_positions().row(edge(1));
257 min_boundary_edge_length = std::min(min_boundary_edge_length, (v1 - v0).norm());
260 double dhat =
Units::convert(args[
"contact"][
"dhat"], units.length());
261 args[
"contact"][
"epsv"] =
Units::convert(args[
"contact"][
"epsv"], units.velocity());
263 if (!contact_dhat_was_explicit_
264 && std::isfinite(min_boundary_edge_length)
265 && dhat > min_boundary_edge_length)
267 dhat = args[
"contact"][
"dhat_percentage"].get<
double>() * min_boundary_edge_length;
268 logger().info(
"dhat set to {}", dhat);
270 else if (std::isfinite(min_boundary_edge_length) && dhat > min_boundary_edge_length)
272 logger().warn(
"dhat larger than min boundary edge, {} > {}", dhat, min_boundary_edge_length);
275 args[
"contact"][
"dhat"] = dhat;
278 void NonlinearElasticVarForm::build_rhs_assembler()
280 json rhs_solver_params = args[
"solver"][
"linear"];
281 if (!rhs_solver_params.contains(
"Pardiso"))
282 rhs_solver_params[
"Pardiso"] = {};
283 rhs_solver_params[
"Pardiso"][
"mtype"] = -2;
285 const int size = problem->is_scalar() ? 1 : mesh_->dimension();
287 solve_data.rhs_assembler = std::make_shared<assembler::RhsAssembler>(
288 *primary_assembler_, *mesh_, &obstacle,
289 boundary_.dirichlet_nodes, boundary_.neumann_nodes,
290 boundary_.dirichlet_nodes_position, boundary_.neumann_nodes_position,
291 space_.n_bases, size,
space_.basis_list(),
space_.geometry_basis_list(), mass_ass_vals_cache_, *problem,
292 args[
"space"][
"advanced"][
"bc_method"],
294 rhs_assembler_ = solve_data.rhs_assembler;
297 void NonlinearElasticVarForm::build_collision_mesh(
301 build_collision_mesh(
302 mesh,
space_.n_bases,
space_.basis_list(),
space_.geometry_basis_list(), boundary_.total_local_boundary, obstacle,
303 args, [
this](
const std::string &p) { return utils::resolve_path(p, root_path, false); },
304 space_.space_in_node_to_node, collision_mesh);
307 void NonlinearElasticVarForm::build_collision_mesh(
310 const std::vector<basis::ElementBases> &bases,
311 const std::vector<basis::ElementBases> &geom_bases,
312 const std::vector<mesh::LocalBoundary> &total_local_boundary,
315 const std::function<std::string(
const std::string &)> &resolve_input_path,
316 const Eigen::VectorXi &in_node_to_node,
317 ipc::CollisionMesh &collision_mesh)
319 Eigen::MatrixXd collision_vertices;
320 Eigen::VectorXi collision_codim_vids;
321 Eigen::MatrixXi collision_edges, collision_triangles;
322 std::vector<Eigen::Triplet<double>> displacement_map_entries;
324 if (args.contains(
"/contact/collision_mesh"_json_pointer)
325 && args.at(
"/contact/collision_mesh/enabled"_json_pointer).get<
bool>())
327 const json collision_mesh_args = args.at(
"/contact/collision_mesh"_json_pointer);
328 if (collision_mesh_args.contains(
"linear_map"))
330 assert(displacement_map_entries.empty());
331 assert(collision_mesh_args.contains(
"mesh"));
332 const std::string root_path = utils::json_value<std::string>(args,
"root_path",
"");
338 in_node_to_node, transformation, collision_vertices, collision_codim_vids,
339 collision_edges, collision_triangles, displacement_map_entries);
341 else if (collision_mesh_args.contains(
"max_edge_length"))
344 "Building collision proxy with max edge length={} ...",
345 collision_mesh_args[
"max_edge_length"].get<double>());
349 bases, geom_bases, total_local_boundary, n_bases, mesh.
dimension(),
350 collision_mesh_args[
"max_edge_length"], collision_vertices,
351 collision_triangles, displacement_map_entries,
352 collision_mesh_args[
"tessellation_type"]);
353 if (collision_triangles.size())
354 igl::edges(collision_triangles, collision_edges);
356 logger().debug(fmt::format(
357 std::locale(
"en_US.UTF-8"),
358 "Done (took {:g}s, {:L} vertices, {:L} triangles)",
359 timer.getElapsedTime(),
360 collision_vertices.rows(), collision_triangles.rows()));
365 mesh, n_bases - obstacle.
n_vertices(), bases, total_local_boundary,
366 collision_vertices, collision_edges, collision_triangles, displacement_map_entries);
372 mesh, n_bases - obstacle.
n_vertices(), bases, total_local_boundary,
373 collision_vertices, collision_edges, collision_triangles, displacement_map_entries);
376 std::vector<bool> is_orientable_vertex(collision_vertices.rows(),
true);
379 const int num_fe_nodes = n_bases - obstacle.
n_vertices();
380 const int num_fe_collision_vertices = collision_vertices.rows();
381 assert(collision_edges.size() == 0 || collision_edges.maxCoeff() < num_fe_collision_vertices);
382 assert(collision_triangles.size() == 0 || collision_triangles.maxCoeff() < num_fe_collision_vertices);
392 for (
int i = 0; i < obstacle.
n_vertices(); i++)
394 is_orientable_vertex.push_back(
false);
397 if (!displacement_map_entries.empty())
399 displacement_map_entries.reserve(displacement_map_entries.size() + obstacle.
n_vertices());
400 for (
int i = 0; i < obstacle.
n_vertices(); i++)
402 displacement_map_entries.emplace_back(num_fe_collision_vertices + i, num_fe_nodes + i, 1.0);
407 std::vector<bool> is_on_surface = ipc::CollisionMesh::construct_is_on_surface(
408 collision_vertices.rows(), collision_edges);
409 for (
const int vid : collision_codim_vids)
411 is_on_surface[vid] =
true;
414 Eigen::SparseMatrix<double> displacement_map;
415 if (!displacement_map_entries.empty())
417 displacement_map.resize(collision_vertices.rows(), n_bases);
418 displacement_map.setFromTriplets(displacement_map_entries.begin(), displacement_map_entries.end());
421 collision_mesh = ipc::CollisionMesh(
422 is_on_surface, is_orientable_vertex, collision_vertices, collision_edges, collision_triangles,
425 collision_mesh.can_collide = [&collision_mesh, num_fe_collision_vertices](
size_t vi,
size_t vj) {
427 return collision_mesh.to_full_vertex_id(vi) < num_fe_collision_vertices
428 || collision_mesh.to_full_vertex_id(vj) < num_fe_collision_vertices;
431 collision_mesh.init_area_jacobians();
434 std::shared_ptr<assembler::PressureAssembler> NonlinearElasticVarForm::build_pressure_assembler()
const
436 const int size = problem->is_scalar() ? 1 : mesh_->dimension();
438 return std::make_shared<assembler::PressureAssembler>(
439 *primary_assembler_, *mesh_, obstacle,
440 boundary_.local_pressure_boundary,
441 boundary_.local_pressure_cavity,
442 boundary_.boundary_nodes,
443 elastic_primitive_to_node(), elastic_node_to_primitive(),
447 void NonlinearElasticStaticVarForm::solve_problem(Eigen::MatrixXd &sol)
449 stats.spectrum.setZero();
453 logger().info(
"Solving {}", primary_assembler_->name());
465 initial_elastic_solution(sol);
468 sol.conservativeResize(Eigen::NoChange, 1);
470 init_solve(sol, 1.0);
472 solve_tensor_nonlinear(0, sol,
true);
474 const std::string state_path = resolve_output_path(args[
"output"][
"data"][
"state"]);
475 if (!state_path.empty())
479 timings.solving_time = timer.getElapsedTime();
480 logger().info(
" took {}s", timings.solving_time);
483 void NonlinearElasticTransientVarForm::solve_problem(Eigen::MatrixXd &sol)
485 const bool save_stats = args[
"output"][
"stats"];
486 stats.spectrum.setZero();
490 logger().info(
"Solving {}", primary_assembler_->name());
502 initial_elastic_solution(sol);
505 sol.conservativeResize(Eigen::NoChange, 1);
507 init_solve(sol, t0 + dt);
512 std::unique_ptr<io::EnergyCSVWriter> energy_csv =
nullptr;
513 std::unique_ptr<io::RuntimeStatsCSVWriter> stats_csv =
nullptr;
517 logger().debug(
"Saving nl stats to {} and {}", resolve_output_path(
"energy.csv"), resolve_output_path(
"stats.csv"));
518 energy_csv = std::make_unique<io::EnergyCSVWriter>(resolve_output_path(
"energy.csv"), solve_data);
520 stats_csv = std::make_unique<io::RuntimeStatsCSVWriter>(
521 resolve_output_path(
"stats.csv"),
529 energy_csv->write(save_i, sol);
530 save_timestep(t0, 0, t0, dt, sol);
534 for (
int t = 1; t <= time_steps; ++t)
536 double forward_solve_time = 0, remeshing_time = 0, global_relaxation_time = 0;
540 solve_tensor_nonlinear(t, sol,
true);
545 energy_csv->write(save_i, sol);
546 save_timestep(t0 + dt * t, t, t0, dt, sol);
552 solve_data.time_integrator->update_quantities(sol);
554 solve_data.nl_problem->update_quantities(t0 + (t + 1) * dt, sol);
556 solve_data.update_dt();
557 solve_data.update_barrier_stiffness(sol);
560 logger().info(
"{}/{} t={}", t, time_steps, t0 + dt * t);
561 notify_time_step(t, time_steps, t0, dt);
563 save_elastic_step_state(t0, dt, t, solve_data.time_integrator.get());
565 stats_csv->write(t, forward_solve_time, remeshing_time, global_relaxation_time);
569 timings.solving_time = timer.getElapsedTime();
570 logger().info(
" took {}s", timings.solving_time);
573 void NonlinearElasticVarForm::init_forms(
const json &args,
const int dim, Eigen::MatrixXd &sol,
const double t)
575 damping_assembler = std::make_shared<assembler::ViscousDamping>();
576 set_materials(*damping_assembler, mesh_->dimension());
578 elasticity_pressure_assembler = build_pressure_assembler();
581 damping_prev_assembler = std::make_shared<assembler::ViscousDampingPrev>();
582 set_materials(*damping_prev_assembler, mesh_->dimension());
587 forms = solve_data.init_forms(
590 dim, t,
space_.space_in_node_to_node,
592 space_.n_bases, *
space_.bases,
space_.geometry_basis_list(), *primary_assembler_, ass_vals_cache_, mass_ass_vals_cache_, args[
"solver"][
"advanced"][
"jacobian_threshold"], check_inversion,
594 0, boundary_.boundary_nodes, boundary_.local_boundary,
595 boundary_.local_neumann_boundary,
596 elastic_boundary_samples(), rhs_, sol, mass_assembler_->density(),
598 boundary_.local_pressure_boundary, boundary_.local_pressure_cavity, elasticity_pressure_assembler,
600 args.value(
"/time/quasistatic"_json_pointer,
true), mass_,
601 damping_assembler->is_valid() ? damping_assembler :
nullptr,
603 args[
"solver"][
"advanced"][
"lagged_regularization_weight"],
604 args[
"solver"][
"advanced"][
"lagged_regularization_iterations"],
606 obstacle.
ndof(), args[
"constraints"][
"hard"], args[
"constraints"][
"soft"],
608 args[
"contact"][
"enabled"], collision_mesh, args[
"contact"][
"dhat"],
609 avg_mass_, args[
"contact"][
"use_convergent_formulation"] ? bool(args[
"contact"][
"use_area_weighting"]) :
false,
610 args[
"contact"][
"use_convergent_formulation"] ? bool(args[
"contact"][
"use_improved_max_operator"]) :
false,
611 args[
"contact"][
"use_convergent_formulation"] ? bool(args[
"contact"][
"use_physical_barrier"]) :
false,
612 args[
"solver"][
"contact"][
"barrier_stiffness"],
613 args[
"solver"][
"contact"][
"initial_barrier_stiffness"],
614 args[
"solver"][
"contact"][
"CCD"][
"broad_phase"],
615 args[
"solver"][
"contact"][
"CCD"][
"tolerance"],
616 args[
"solver"][
"contact"][
"CCD"][
"max_iterations"],
619 args[
"contact"][
"use_gcp_formulation"],
620 args[
"contact"][
"alpha_t"],
621 args[
"contact"][
"alpha_n"],
622 args[
"contact"][
"use_adaptive_dhat"],
623 args[
"contact"][
"min_distance_ratio"],
625 args[
"contact"][
"adhesion"][
"adhesion_enabled"],
626 args[
"contact"][
"adhesion"][
"dhat_p"],
627 args[
"contact"][
"adhesion"][
"dhat_a"],
628 args[
"contact"][
"adhesion"][
"adhesion_strength"],
630 args[
"contact"][
"adhesion"][
"tangential_adhesion_coefficient"],
631 args[
"contact"][
"adhesion"][
"epsa"],
632 args[
"solver"][
"contact"][
"tangential_adhesion_iterations"],
636 false, Eigen::VectorXi(),
nullptr,
638 args[
"contact"][
"friction_coefficient"],
639 args[
"contact"][
"epsv"],
640 args[
"solver"][
"contact"][
"friction_iterations"],
642 args[
"solver"][
"rayleigh_damping"]);
644 for (
const auto &form : forms)
645 form->set_output_dir(output_path);
647 if (solve_data.contact_form !=
nullptr)
648 solve_data.contact_form->save_ccd_debug_meshes = args[
"output"][
"advanced"][
"save_ccd_debug_meshes"];
651 void NonlinearElasticVarForm::init_solve(Eigen::MatrixXd &sol,
const double t)
653 assert(sol.cols() == 1);
654 assert(!problem->is_scalar());
667 if (args[
"contact"][
"enabled"])
671 const Eigen::MatrixXd displaced = collision_mesh.displace_vertices(
674 if (ipc::has_intersections(collision_mesh, displaced, ipc::create_broad_phase(args[
"solver"][
"contact"][
"CCD"][
"broad_phase"]).get()))
677 resolve_output_path(
"intersection.obj"), displaced,
678 collision_mesh.edges(), collision_mesh.faces());
685 if (problem->is_time_dependent())
688 solve_data.time_integrator = ImplicitTimeIntegrator::construct_time_integrator(args[
"time"][
"integrator"]);
690 Eigen::MatrixXd solution, velocity, acceleration;
691 initial_elastic_solution(solution);
692 solution.col(0) = sol;
693 assert(solution.rows() == sol.size());
694 initial_velocity(velocity);
695 assert(velocity.rows() == sol.size());
696 initial_acceleration(acceleration);
697 assert(acceleration.rows() == sol.size());
699 solve_data.time_integrator->init(solution, velocity, acceleration, dt);
700 assert(solve_data.time_integrator !=
nullptr);
704 solve_data.time_integrator =
nullptr;
713 init_forms(args, mesh_->dimension(), sol, t);
715 double characteristic_length = 0;
716 if (args[
"solver"][
"advanced"][
"characteristic_length"] > 0)
718 characteristic_length = args[
"solver"][
"advanced"][
"characteristic_length"];
723 mesh_->bounding_box(min, max);
724 characteristic_length = (max - min).norm();
727 double characteristic_force_density = 0;
728 if (args[
"solver"][
"advanced"][
"characteristic_force_density"] <= 0)
730 logger().warn(
"No user-specified force density was provided, defaulting to 10000.");
731 characteristic_force_density = 10000;
735 characteristic_force_density = args[
"solver"][
"advanced"][
"characteristic_force_density"];
738 if (pure_mass_.size() == 0)
739 pure_mass_assembler_->assemble(mesh_->is_volume(),
space_.n_bases,
space_.basis_list(),
space_.geometry_basis_list(), pure_mass_ass_vals_cache_, 0, pure_mass_,
true);
741 const int ndof =
space_.n_bases * mesh_->dimension();
742 solve_data.nl_problem = std::make_shared<solver::NLProblem>(
743 ndof,
nullptr, t, forms, solve_data.al_form,
744 polysolve::linear::Solver::create(args[
"solver"][
"linear"],
logger()),
745 characteristic_length, characteristic_force_density, pure_mass_, mesh_->dimension());
746 solve_data.nl_problem->init(sol);
747 solve_data.nl_problem->update_quantities(t, sol);
750 stats.solver_info = json::array();
753 void NonlinearElasticVarForm::solve_tensor_nonlinear(
int step, Eigen::MatrixXd &sol,
const bool init_lagging)
755 assert(solve_data.nl_problem !=
nullptr);
758 assert(sol.size() == rhs_.size());
767 logger().info(
"Lagging iteration 1:");
770 save_subsolve(0, step, sol);
772 std::shared_ptr<polysolve::nonlinear::Solver> nl_solver =
773 polysolve::nonlinear::Solver::create(args[
"solver"][
"augmented_lagrangian"][
"nonlinear"], args[
"solver"][
"linear"], units.characteristic_length(),
logger());
777 args[
"solver"][
"augmented_lagrangian"][
"initial_weight"],
778 args[
"solver"][
"augmented_lagrangian"][
"scaling"],
779 args[
"solver"][
"augmented_lagrangian"][
"max_weight"],
780 args[
"solver"][
"augmented_lagrangian"][
"eta"],
781 [&](
const Eigen::VectorXd &
x) {
782 this->solve_data.update_barrier_stiffness(sol);
786 stats.solver_info.push_back(
787 {{
"type", al_weight > 0 ?
"al" :
"rc"},
789 {
"info", nl_solver->info()}});
791 stats.solver_info.back()[
"weight"] = al_weight;
792 save_subsolve(stats.solver_info.size(), step, sol);
795 Eigen::MatrixXd prev_sol = sol;
797 args[
"solver"][
"augmented_lagrangian"][
"nonlinear"], args[
"solver"][
"linear"], units.characteristic_length());
800 args[
"solver"][
"nonlinear"], args[
"solver"][
"linear"], units.characteristic_length());
802 if (args[
"space"][
"advanced"][
"count_flipped_els_continuous"])
805 logger().debug(
"Flipped elements (cnt {}) : {}", invalidList.size(), invalidList);
808 const double lagging_tol = args[
"solver"][
"contact"].value(
"friction_convergence_tol", 1e-2) * units.characteristic_length();
811 for (
int lag_i = 1; !lagging_converged; lag_i++)
817 Eigen::VectorXd grad;
819 const double delta_x_norm = (prev_sol - sol).lpNorm<Eigen::Infinity>();
820 logger().debug(
"Lagging convergence grad_norm={:g} tol={:g} (||Δx||={:g})", grad.norm(), lagging_tol, delta_x_norm);
821 if (grad.norm() <= lagging_tol)
824 "Lagging converged in {:d} iteration(s) (grad_norm={:g} tol={:g})",
825 lag_i, grad.norm(), lagging_tol);
826 lagging_converged =
true;
830 if (delta_x_norm <= 1e-12)
833 "Lagging produced tiny update between iterations {:d} and {:d} (grad_norm={:g} grad_tol={:g} ||Δx||={:g} Δx_tol={:g}); stopping early",
834 lag_i - 1, lag_i, grad.norm(), lagging_tol, delta_x_norm, 1e-6);
835 lagging_converged =
false;
842 "Lagging failed to converge with {:d} iteration(s) (grad_norm={:g} tol={:g})",
843 lag_i, grad.norm(), lagging_tol);
844 lagging_converged =
false;
848 logger().info(
"Lagging iteration {:d}:", lag_i + 1);
849 nl_problem.
init(sol);
850 solve_data.update_barrier_stiffness(sol);
852 nl_solver->minimize(nl_problem, tmp_sol);
857 stats.solver_info.push_back(
861 {
"info", nl_solver->info()}});
862 save_subsolve(stats.solver_info.size(), step, sol);
std::array< Matrix< int, 3, 3 >, 3 > space_
#define POLYFEM_SCOPED_TIMER(...)
static double convert(const json &val, const std::string &unit_type)
static bool write(const std::string &path, const Eigen::MatrixXd &v, const Eigen::MatrixXi &e, const Eigen::MatrixXi &f)
static void extract_boundary_mesh(const mesh::Mesh &mesh, const int n_bases, const std::vector< basis::ElementBases > &bases, const std::vector< mesh::LocalBoundary > &total_local_boundary, Eigen::MatrixXd &node_positions, Eigen::MatrixXi &boundary_edges, Eigen::MatrixXi &boundary_triangles, std::vector< Eigen::Triplet< double > > &displacement_map_entries)
extracts the boundary mesh
Abstract mesh class to capture 2d/3d conforming and non-conforming meshes.
int n_elements() const
utitlity to return the number of elements, cells or faces in 3d and 2d
int dimension() const
utily for dimension
const Eigen::MatrixXi & e() const
const Eigen::MatrixXi & f() const
const Eigen::MatrixXd & v() const
const Eigen::VectorXi & codim_v() const
void solve_reduced(NLProblem &nl_problem, Eigen::MatrixXd &sol, std::shared_ptr< polysolve::nonlinear::Solver > nl_solver)
std::function< void(const double)> post_subsolve
void solve_al(NLProblem &nl_problem, Eigen::MatrixXd &sol, std::shared_ptr< polysolve::nonlinear::Solver > nl_solver)
bool uses_lagging() const
int max_lagging_iterations() const
virtual void init(const TVector &x0) override
double normalize_forms() override
TVector full_to_reduced(const TVector &full) const
virtual void gradient(const TVector &x, TVector &gradv) override
void init_lagging(const TVector &x) override
TVector reduced_to_full(const TVector &reduced) const
void update_lagging(const TVector &x, const int iter_num) override
class to store time stepping data
std::shared_ptr< solver::ContactForm > contact_form
std::vector< std::pair< std::string, std::shared_ptr< solver::Form > > > named_forms() const
std::shared_ptr< solver::ElasticForm > elastic_form
std::shared_ptr< time_integrator::ImplicitTimeIntegrator > time_integrator
bool write_matrix(const std::string &path, const Mat &mat)
Writes a matrix to a file. Determines the file format based on the path's extension.
void load_collision_proxy(const std::string &mesh_filename, const std::string &weights_filename, const Eigen::VectorXi &in_node_to_node, const json &transformation, Eigen::MatrixXd &vertices, Eigen::VectorXi &codim_vertices, Eigen::MatrixXi &edges, Eigen::MatrixXi &faces, std::vector< Eigen::Triplet< double > > &displacement_map_entries)
Load a collision proxy mesh and displacement map from files.
void build_collision_proxy(const std::vector< basis::ElementBases > &bases, const std::vector< basis::ElementBases > &geom_bases, const std::vector< LocalBoundary > &total_local_boundary, const int n_bases, const int dim, const double max_edge_length, Eigen::MatrixXd &proxy_vertices, Eigen::MatrixXi &proxy_faces, std::vector< Eigen::Triplet< double > > &displacement_map_entries, const CollisionProxyTessellation tessellation)
Obstacle read_obstacle_geometry(const Units &units, const json &geometry, const std::vector< json > &displacements, const std::vector< json > &dirichlets, const std::string &root_path, const int dim, const std::vector< std::string > &_names, const std::vector< Eigen::MatrixXd > &_vertices, const std::vector< Eigen::MatrixXi > &_cells, const bool non_conforming)
read a FEM mesh from a geometry JSON
std::string resolve_path(const std::string &path, const std::string &input_file_path, const bool only_if_exists=false)
std::vector< T > json_as_array(const json &j)
Return the value of a json object as an array.
Eigen::MatrixXd unflatten(const Eigen::VectorXd &x, int dim)
Unflatten rowwises, so every dim elements in x become a row.
std::vector< int > count_invalid(const int dim, const std::vector< basis::ElementBases > &bases, const std::vector< basis::ElementBases > &gbases, const Eigen::VectorXd &u)
void append_rows(DstMat &dst, const SrcMat &src)
spdlog::logger & logger()
Retrieves the current logger.
Eigen::Matrix< double, Eigen::Dynamic, 1, 0, 3, 1 > VectorNd
Eigen::Matrix< double, 1, Eigen::Dynamic, Eigen::RowMajor, 1, 3 > RowVectorNd
void log_and_throw_error(const std::string &msg)
std::vector< std::string > fields