PolyFEM
Loading...
Searching...
No Matches
NonlinearElasticVarForm.cpp
Go to the documentation of this file.
2
5
10
17
21
29
30#include <igl/Timer.h>
31#include <igl/edges.h>
32
33#include <ipc/ipc.hpp>
34
35#include <polysolve/linear/Solver.hpp>
36#include <polysolve/nonlinear/Solver.hpp>
37
38#include <algorithm>
39#include <cmath>
40#include <limits>
41
42namespace polyfem::varform
43{
44 using namespace solver;
45 using namespace time_integrator;
46
47 void NonlinearElasticVarForm::init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path)
48 {
49 json clean_args = args;
50 const bool contact_dhat_was_explicit = clean_args["contact"].value("_dhat_was_explicit", false);
51 clean_args["contact"].erase("_dhat_was_explicit");
52 ElasticVarForm::init(formulation, units, clean_args, out_path);
53 contact_dhat_was_explicit_ = contact_dhat_was_explicit;
54 }
55
57 {
59 collision_mesh = ipc::CollisionMesh();
62 forms.clear();
64 damping_assembler = nullptr;
65 damping_prev_assembler = nullptr;
67 }
68
69 void NonlinearElasticVarForm::load_mesh(const mesh::Mesh &mesh, const json &args)
70 {
72
73 logger().info("Loading obstacles...");
75 units,
76 args["geometry"],
77 utils::json_as_array(args["boundary_conditions"]["obstacle_displacements"]),
78 utils::json_as_array(args["boundary_conditions"]["dirichlet_boundary"]),
79 root_path, mesh.dimension());
80 }
81
83 {
84 auto space = ElasticVarForm::output_space();
85 space.collision_mesh = is_contact_enabled() ? &collision_mesh : nullptr;
86 space.obstacle = &obstacle;
87 return space;
88 }
89
90 std::vector<io::OutputField> NonlinearElasticVarForm::output_fields(
91 const io::OutputSample &sample,
92 const Eigen::MatrixXd &solution,
93 const io::OutputFieldOptions &options) const
94 {
95 std::vector<io::OutputField> fields = elastic_output_fields(
96 sample, solution, options, &obstacle, solve_data.time_integrator.get(),
98 if (!mesh_ || !problem || solution.size() <= 0)
99 return fields;
101 return fields;
102
103 const int actual_dim = problem->is_scalar() ? 1 : mesh_->dimension();
104 const auto &paraview_options = args["output"]["paraview"]["options"];
105 const bool explicit_fields = !options.fields.empty();
106
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;
110 });
111 };
112
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)
115 return;
116
117 Eigen::MatrixXd values = collision_mesh.map_displacements(utils::unflatten(dof_values, actual_dim));
118 if (values.rows() == sample.points.rows())
119 fields.push_back({name, values, io::OutputField::Association::Point});
120 };
121
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())
124 return;
125
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);
132 };
133
134 if (paraview_options["forces"] && !problem->is_scalar())
135 {
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())
138 {
139 const std::string field_name = name + "_forces";
140 if (!options.export_field(field_name))
141 continue;
142
143 Eigen::VectorXd force;
144 if (form && form->enabled())
145 {
146 form->first_derivative(solution, force);
147 force *= -1.0 / s;
148 }
149 else
150 {
151 force.setZero(solution.size());
152 }
153 append_collision_dof_field(field_name, force);
154 }
155 }
156
157 if (options.export_field("gradient_of_elastic_potential") && solve_data.elastic_form)
158 {
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);
162 }
163
164 if (options.export_field("gradient_of_contact_potential") && solve_data.contact_form && solve_data.contact_form->weight() > 0)
165 {
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);
170 }
171
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);
176
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);
185
186 if (explicit_fields
187 && options.export_field("adaptive_dhat")
188 && args["contact"]["use_gcp_formulation"]
189 && args["contact"]["use_adaptive_dhat"])
190 {
191 const auto smooth_contact = std::dynamic_pointer_cast<solver::SmoothContactForm>(solve_data.contact_form);
192 if (smooth_contact)
193 {
194 const auto &set = smooth_contact->collision_set();
195 if (actual_dim == 2)
196 {
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);
200 fields.push_back({"dhat", dhats, io::OutputField::Association::Cell});
201 }
202 else
203 {
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);
207 fields.push_back({"dhat_face", dhats, io::OutputField::Association::Cell});
208
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);
212 fields.push_back({"dhat_vert", vertex_dhats, io::OutputField::Association::Point});
213 }
214 }
215 }
216
217 return fields;
218 }
219
220 void NonlinearElasticVarForm::build_basis(mesh::Mesh &mesh, const bool iso_parametric, const json &args)
221 {
222 ElasticVarForm::build_basis(mesh, iso_parametric, args);
223
224 // Legacy nonlinear/contact code assumes the displacement space includes obstacle vertices.
225 // The shared build path only counts FE bases, so extend it here
226 // before constructing collision/contact state.
227 const int n_fe_bases = space_.n_bases;
228 space_.n_bases += obstacle.n_vertices();
229
230 logger().info("Building collision mesh...");
231 build_collision_mesh(mesh, args);
232 preprocess_contact_parameters();
233 // FIXME!! handle periodic collision mesh
234 // if (periodic_bc && args["contact"]["periodic"])
235 // build_periodic_collision_mesh();
236 logger().info("Done!");
237
238 for (int i = n_fe_bases; i < space_.n_bases; ++i)
239 {
240 for (int d = 0; d < mesh.dimension(); ++d)
241 boundary_.boundary_nodes.push_back(i * mesh.dimension() + d);
242 }
243
244 boundary_.normalize_boundary_nodes();
245 }
246
247 void NonlinearElasticVarForm::preprocess_contact_parameters()
248 {
249 if (!is_contact_enabled())
250 return;
251
252 double min_boundary_edge_length = std::numeric_limits<double>::max();
253 for (const auto &edge : collision_mesh.edges().rowwise())
254 {
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());
258 }
259
260 double dhat = Units::convert(args["contact"]["dhat"], units.length());
261 args["contact"]["epsv"] = Units::convert(args["contact"]["epsv"], units.velocity());
262
263 if (!contact_dhat_was_explicit_
264 && std::isfinite(min_boundary_edge_length)
265 && dhat > min_boundary_edge_length)
266 {
267 dhat = args["contact"]["dhat_percentage"].get<double>() * min_boundary_edge_length;
268 logger().info("dhat set to {}", dhat);
269 }
270 else if (std::isfinite(min_boundary_edge_length) && dhat > min_boundary_edge_length)
271 {
272 logger().warn("dhat larger than min boundary edge, {} > {}", dhat, min_boundary_edge_length);
273 }
274
275 args["contact"]["dhat"] = dhat;
276 }
277
278 void NonlinearElasticVarForm::build_rhs_assembler()
279 {
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;
284
285 const int size = problem->is_scalar() ? 1 : mesh_->dimension();
286
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"],
293 rhs_solver_params);
294 rhs_assembler_ = solve_data.rhs_assembler;
295 }
296
297 void NonlinearElasticVarForm::build_collision_mesh(
298 const mesh::Mesh &mesh,
299 const json &args)
300 {
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);
305 }
306
307 void NonlinearElasticVarForm::build_collision_mesh(
308 const mesh::Mesh &mesh,
309 const int n_bases,
310 const std::vector<basis::ElementBases> &bases,
311 const std::vector<basis::ElementBases> &geom_bases,
312 const std::vector<mesh::LocalBoundary> &total_local_boundary,
313 const mesh::Obstacle &obstacle,
314 const json &args,
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)
318 {
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;
323
324 if (args.contains("/contact/collision_mesh"_json_pointer)
325 && args.at("/contact/collision_mesh/enabled"_json_pointer).get<bool>())
326 {
327 const json collision_mesh_args = args.at("/contact/collision_mesh"_json_pointer);
328 if (collision_mesh_args.contains("linear_map"))
329 {
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", "");
333 // TODO: handle transformation per geometry
334 const json transformation = utils::json_as_array(args["geometry"])[0]["transformation"];
336 utils::resolve_path(collision_mesh_args["mesh"], root_path),
337 utils::resolve_path(collision_mesh_args["linear_map"], root_path),
338 in_node_to_node, transformation, collision_vertices, collision_codim_vids,
339 collision_edges, collision_triangles, displacement_map_entries);
340 }
341 else if (collision_mesh_args.contains("max_edge_length"))
342 {
343 logger().debug(
344 "Building collision proxy with max edge length={} ...",
345 collision_mesh_args["max_edge_length"].get<double>());
346 igl::Timer timer;
347 timer.start();
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);
355 timer.stop();
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()));
361 }
362 else
363 {
365 mesh, n_bases - obstacle.n_vertices(), bases, total_local_boundary,
366 collision_vertices, collision_edges, collision_triangles, displacement_map_entries);
367 }
368 }
369 else
370 {
372 mesh, n_bases - obstacle.n_vertices(), bases, total_local_boundary,
373 collision_vertices, collision_edges, collision_triangles, displacement_map_entries);
374 }
375
376 std::vector<bool> is_orientable_vertex(collision_vertices.rows(), true);
377
378 // n_bases already contains the obstacle vertices
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);
383
384 // Append the obstacles to the collision mesh
385 if (obstacle.n_vertices() > 0)
386 {
387 utils::append_rows(collision_vertices, obstacle.v());
388 utils::append_rows(collision_codim_vids, obstacle.codim_v().array() + num_fe_collision_vertices);
389 utils::append_rows(collision_edges, obstacle.e().array() + num_fe_collision_vertices);
390 utils::append_rows(collision_triangles, obstacle.f().array() + num_fe_collision_vertices);
391
392 for (int i = 0; i < obstacle.n_vertices(); i++)
393 {
394 is_orientable_vertex.push_back(false);
395 }
396
397 if (!displacement_map_entries.empty())
398 {
399 displacement_map_entries.reserve(displacement_map_entries.size() + obstacle.n_vertices());
400 for (int i = 0; i < obstacle.n_vertices(); i++)
401 {
402 displacement_map_entries.emplace_back(num_fe_collision_vertices + i, num_fe_nodes + i, 1.0);
403 }
404 }
405 }
406
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)
410 {
411 is_on_surface[vid] = true;
412 }
413
414 Eigen::SparseMatrix<double> displacement_map;
415 if (!displacement_map_entries.empty())
416 {
417 displacement_map.resize(collision_vertices.rows(), n_bases);
418 displacement_map.setFromTriplets(displacement_map_entries.begin(), displacement_map_entries.end());
419 }
420
421 collision_mesh = ipc::CollisionMesh(
422 is_on_surface, is_orientable_vertex, collision_vertices, collision_edges, collision_triangles,
423 displacement_map);
424
425 collision_mesh.can_collide = [&collision_mesh, num_fe_collision_vertices](size_t vi, size_t vj) {
426 // obstacles do not collide with other obstacles
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;
429 };
430
431 collision_mesh.init_area_jacobians();
432 }
433
434 std::shared_ptr<assembler::PressureAssembler> NonlinearElasticVarForm::build_pressure_assembler() const
435 {
436 const int size = problem->is_scalar() ? 1 : mesh_->dimension();
437
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(),
444 space_.n_bases, size, space_.basis_list(), space_.geometry_basis_list(), *problem);
445 }
446
447 void NonlinearElasticStaticVarForm::solve_problem(Eigen::MatrixXd &sol)
448 {
449 stats.spectrum.setZero();
450
451 igl::Timer timer;
452 timer.start();
453 logger().info("Solving {}", primary_assembler_->name());
454
455 {
456 POLYFEM_SCOPED_TIMER("Setup RHS");
457
458 // FIXME
459 // read_initial_x_from_file(
460 // resolve_input_path(args["input"]["data"]["state"]), "u",
461 // args["input"]["data"]["reorder"], in_node_to_node,
462 // mesh->dimension(), solution);
463
464 if (sol.size() <= 0)
465 initial_elastic_solution(sol);
466
467 if (sol.cols() > 1) // ignore previous solutions
468 sol.conservativeResize(Eigen::NoChange, 1);
469 }
470 init_solve(sol, 1.0);
471
472 solve_tensor_nonlinear(0, sol, true);
473
474 const std::string state_path = resolve_output_path(args["output"]["data"]["state"]);
475 if (!state_path.empty())
476 io::write_matrix(state_path, "u", sol);
477
478 timer.stop();
479 timings.solving_time = timer.getElapsedTime();
480 logger().info(" took {}s", timings.solving_time);
481 }
482
483 void NonlinearElasticTransientVarForm::solve_problem(Eigen::MatrixXd &sol)
484 {
485 const bool save_stats = args["output"]["stats"];
486 stats.spectrum.setZero();
487
488 igl::Timer timer;
489 timer.start();
490 logger().info("Solving {}", primary_assembler_->name());
491
492 {
493 POLYFEM_SCOPED_TIMER("Setup RHS");
494
495 // FIXME
496 // read_initial_x_from_file(
497 // resolve_input_path(args["input"]["data"]["state"]), "u",
498 // args["input"]["data"]["reorder"], in_node_to_node,
499 // mesh->dimension(), solution);
500
501 if (sol.size() <= 0)
502 initial_elastic_solution(sol);
503
504 if (sol.cols() > 1) // ignore previous solutions
505 sol.conservativeResize(Eigen::NoChange, 1);
506 }
507 init_solve(sol, t0 + dt);
508
509 // Write the total energy to a CSV file
510 int save_i = 0;
511
512 std::unique_ptr<io::EnergyCSVWriter> energy_csv = nullptr;
513 std::unique_ptr<io::RuntimeStatsCSVWriter> stats_csv = nullptr;
514
515 if (save_stats)
516 {
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);
519 const io::OutputSpace space = output_space();
520 stats_csv = std::make_unique<io::RuntimeStatsCSVWriter>(
521 resolve_output_path("stats.csv"),
522 space_.n_bases,
523 space.mesh ? space.mesh->n_elements() : 0,
524 t0, dt);
525 }
526
527 // Save the initial solution
528 if (energy_csv)
529 energy_csv->write(save_i, sol);
530 save_timestep(t0, 0, t0, dt, sol);
531
532 save_i++;
533
534 for (int t = 1; t <= time_steps; ++t)
535 {
536 double forward_solve_time = 0, remeshing_time = 0, global_relaxation_time = 0;
537
538 {
539 POLYFEM_SCOPED_TIMER(forward_solve_time);
540 solve_tensor_nonlinear(t, sol, true);
541 }
542
543 // Always save the solution for consistency
544 if (energy_csv)
545 energy_csv->write(save_i, sol);
546 save_timestep(t0 + dt * t, t, t0, dt, sol);
547 save_i++;
548
549 {
550 POLYFEM_SCOPED_TIMER("Update quantities");
551
552 solve_data.time_integrator->update_quantities(sol);
553
554 solve_data.nl_problem->update_quantities(t0 + (t + 1) * dt, sol);
555
556 solve_data.update_dt();
557 solve_data.update_barrier_stiffness(sol);
558 }
559
560 logger().info("{}/{} t={}", t, time_steps, t0 + dt * t);
561 notify_time_step(t, time_steps, t0, dt);
562
563 save_elastic_step_state(t0, dt, t, solve_data.time_integrator.get());
564 if (stats_csv)
565 stats_csv->write(t, forward_solve_time, remeshing_time, global_relaxation_time);
566 }
567
568 timer.stop();
569 timings.solving_time = timer.getElapsedTime();
570 logger().info(" took {}s", timings.solving_time);
571 }
572
573 void NonlinearElasticVarForm::init_forms(const json &args, const int dim, Eigen::MatrixXd &sol, const double t)
574 {
575 damping_assembler = std::make_shared<assembler::ViscousDamping>();
576 set_materials(*damping_assembler, mesh_->dimension());
577
578 elasticity_pressure_assembler = build_pressure_assembler();
579
580 // for backward solve
581 damping_prev_assembler = std::make_shared<assembler::ViscousDampingPrev>();
582 set_materials(*damping_prev_assembler, mesh_->dimension());
583
584 const ElementInversionCheck check_inversion = args["solver"]["advanced"]["check_inversion"];
585
586 // NOTE: some stuff are legacy and hardcoded to be off
587 forms = solve_data.init_forms(
588 // General
589 units,
590 dim, t, space_.space_in_node_to_node,
591 // Elastic form
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,
593 // Body form
594 0, boundary_.boundary_nodes, boundary_.local_boundary,
595 boundary_.local_neumann_boundary,
596 elastic_boundary_samples(), rhs_, sol, mass_assembler_->density(),
597 // Pressure form
598 boundary_.local_pressure_boundary, boundary_.local_pressure_cavity, elasticity_pressure_assembler,
599 // Inertia form
600 args.value("/time/quasistatic"_json_pointer, true), mass_,
601 damping_assembler->is_valid() ? damping_assembler : nullptr,
602 // Lagged regularization form
603 args["solver"]["advanced"]["lagged_regularization_weight"],
604 args["solver"]["advanced"]["lagged_regularization_iterations"],
605 // Augmented lagrangian form
606 obstacle.ndof(), args["constraints"]["hard"], args["constraints"]["soft"],
607 // Contact form
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"],
617 false,
618 // Smooth Contact Form
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"],
624 // Normal Adhesion Form
625 args["contact"]["adhesion"]["adhesion_enabled"],
626 args["contact"]["adhesion"]["dhat_p"],
627 args["contact"]["adhesion"]["dhat_a"],
628 args["contact"]["adhesion"]["adhesion_strength"],
629 // Tangential Adhesion Form
630 args["contact"]["adhesion"]["tangential_adhesion_coefficient"],
631 args["contact"]["adhesion"]["epsa"],
632 args["solver"]["contact"]["tangential_adhesion_iterations"],
633 // Homogenization
635 // Periodic contact
636 false, Eigen::VectorXi(), nullptr,
637 // Friction form
638 args["contact"]["friction_coefficient"],
639 args["contact"]["epsv"],
640 args["solver"]["contact"]["friction_iterations"],
641 // Rayleigh damping form
642 args["solver"]["rayleigh_damping"]);
643
644 for (const auto &form : forms)
645 form->set_output_dir(output_path);
646
647 if (solve_data.contact_form != nullptr)
648 solve_data.contact_form->save_ccd_debug_meshes = args["output"]["advanced"]["save_ccd_debug_meshes"];
649 }
650
651 void NonlinearElasticVarForm::init_solve(Eigen::MatrixXd &sol, const double t)
652 {
653 assert(sol.cols() == 1);
654 assert(!problem->is_scalar()); // tensor
655
656 // FIXME
657 // if (optimization_enabled != solver::CacheLevel::None)
658 // {
659 // if (initial_sol_update.size() == ndof())
660 // sol = initial_sol_update;
661 // else
662 // initial_sol_update = sol;
663 // }
664
665 // --------------------------------------------------------------------
666 // Check for initial intersections
667 if (args["contact"]["enabled"])
668 {
669 POLYFEM_SCOPED_TIMER("Check for initial intersections");
670
671 const Eigen::MatrixXd displaced = collision_mesh.displace_vertices(
672 utils::unflatten(sol, mesh_->dimension()));
673
674 if (ipc::has_intersections(collision_mesh, displaced, ipc::create_broad_phase(args["solver"]["contact"]["CCD"]["broad_phase"]).get()))
675 {
677 resolve_output_path("intersection.obj"), displaced,
678 collision_mesh.edges(), collision_mesh.faces());
679 log_and_throw_error("Unable to solve, initial solution has intersections!");
680 }
681 }
682
683 // --------------------------------------------------------------------
684
685 if (problem->is_time_dependent())
686 {
687 POLYFEM_SCOPED_TIMER("Initialize time integrator");
688 solve_data.time_integrator = ImplicitTimeIntegrator::construct_time_integrator(args["time"]["integrator"]);
689
690 Eigen::MatrixXd solution, velocity, acceleration;
691 initial_elastic_solution(solution); // Reload this because we need all previous solutions
692 solution.col(0) = sol; // Make sure the current solution is the same as `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());
698
699 solve_data.time_integrator->init(solution, velocity, acceleration, dt);
700 assert(solve_data.time_integrator != nullptr);
701 }
702 else
703 {
704 solve_data.time_integrator = nullptr;
705 }
706
707 // --------------------------------------------------------------------
708 // Initialize forms
709
710 // --------------------------------------------------------------------
711 // Initialize nonlinear problems
712
713 init_forms(args, mesh_->dimension(), sol, t);
714
715 double characteristic_length = 0;
716 if (args["solver"]["advanced"]["characteristic_length"] > 0)
717 {
718 characteristic_length = args["solver"]["advanced"]["characteristic_length"];
719 }
720 else
721 {
722 RowVectorNd min, max;
723 mesh_->bounding_box(min, max);
724 characteristic_length = (max - min).norm();
725 }
726
727 double characteristic_force_density = 0;
728 if (args["solver"]["advanced"]["characteristic_force_density"] <= 0)
729 {
730 logger().warn("No user-specified force density was provided, defaulting to 10000.");
731 characteristic_force_density = 10000;
732 }
733 else
734 {
735 characteristic_force_density = args["solver"]["advanced"]["characteristic_force_density"];
736 }
737
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);
740
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);
748 // --------------------------------------------------------------------
749
750 stats.solver_info = json::array();
751 }
752
753 void NonlinearElasticVarForm::solve_tensor_nonlinear(int step, Eigen::MatrixXd &sol, const bool init_lagging)
754 {
755 assert(solve_data.nl_problem != nullptr);
756 solver::NLProblem &nl_problem = *(solve_data.nl_problem);
757
758 assert(sol.size() == rhs_.size());
759
760 if (nl_problem.uses_lagging())
761 {
762 if (init_lagging)
763 {
764 POLYFEM_SCOPED_TIMER("Initializing lagging");
765 nl_problem.init_lagging(sol);
766 }
767 logger().info("Lagging iteration 1:");
768 }
769
770 save_subsolve(0, step, sol);
771
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());
774
775 ALSolver al_solver(
776 solve_data.al_form,
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);
783 });
784
785 al_solver.post_subsolve = [&](const double al_weight) {
786 stats.solver_info.push_back(
787 {{"type", al_weight > 0 ? "al" : "rc"},
788 {"t", step},
789 {"info", nl_solver->info()}});
790 if (al_weight > 0)
791 stats.solver_info.back()["weight"] = al_weight;
792 save_subsolve(stats.solver_info.size(), step, sol);
793 };
794
795 Eigen::MatrixXd prev_sol = sol;
796 al_solver.solve_al(nl_problem, sol,
797 args["solver"]["augmented_lagrangian"]["nonlinear"], args["solver"]["linear"], units.characteristic_length());
798
799 al_solver.solve_reduced(nl_problem, sol,
800 args["solver"]["nonlinear"], args["solver"]["linear"], units.characteristic_length());
801
802 if (args["space"]["advanced"]["count_flipped_els_continuous"])
803 {
804 const auto invalidList = utils::count_invalid(mesh_->dimension(), space_.basis_list(), space_.geometry_basis_list(), sol);
805 logger().debug("Flipped elements (cnt {}) : {}", invalidList.size(), invalidList);
806 }
807
808 const double lagging_tol = args["solver"]["contact"].value("friction_convergence_tol", 1e-2) * units.characteristic_length();
809
810 bool lagging_converged = !nl_problem.uses_lagging();
811 for (int lag_i = 1; !lagging_converged; lag_i++)
812 {
813 Eigen::VectorXd tmp_sol = nl_problem.full_to_reduced(sol);
814
815 nl_problem.update_lagging(tmp_sol, lag_i);
816
817 Eigen::VectorXd grad;
818 nl_problem.gradient(tmp_sol, 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)
822 {
823 logger().info(
824 "Lagging converged in {:d} iteration(s) (grad_norm={:g} tol={:g})",
825 lag_i, grad.norm(), lagging_tol);
826 lagging_converged = true;
827 break;
828 }
829
830 if (delta_x_norm <= 1e-12)
831 {
832 logger().warn(
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;
836 break;
837 }
838
839 if (lag_i >= nl_problem.max_lagging_iterations())
840 {
841 logger().warn(
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;
845 break;
846 }
847
848 logger().info("Lagging iteration {:d}:", lag_i + 1);
849 nl_problem.init(sol);
850 solve_data.update_barrier_stiffness(sol);
851 nl_problem.normalize_forms();
852 nl_solver->minimize(nl_problem, tmp_sol);
853 nl_problem.finish();
854 prev_sol = sol;
855 sol = nl_problem.reduced_to_full(tmp_sol);
856
857 stats.solver_info.push_back(
858 {{"type", "rc"},
859 {"t", step},
860 {"lag_i", lag_i},
861 {"info", nl_solver->info()}});
862 save_subsolve(stats.solver_info.size(), step, sol);
863 }
864 }
865
866} // namespace polyfem::varform
int x
std::array< Matrix< int, 3, 3 >, 3 > space_
#define POLYFEM_SCOPED_TIMER(...)
Definition Timer.hpp:10
static double convert(const json &val, const std::string &unit_type)
Definition Units.cpp:31
static bool write(const std::string &path, const Eigen::MatrixXd &v, const Eigen::MatrixXi &e, const Eigen::MatrixXi &f)
Definition OBJWriter.cpp:18
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
Definition OutData.cpp:95
Abstract mesh class to capture 2d/3d conforming and non-conforming meshes.
Definition Mesh.hpp:41
int n_elements() const
utitlity to return the number of elements, cells or faces in 3d and 2d
Definition Mesh.hpp:163
int dimension() const
utily for dimension
Definition Mesh.hpp:153
const Eigen::MatrixXi & e() const
Definition Obstacle.hpp:45
const Eigen::MatrixXi & f() const
Definition Obstacle.hpp:44
const Eigen::MatrixXd & v() const
Definition Obstacle.hpp:42
const Eigen::VectorXi & codim_v() const
Definition Obstacle.hpp:43
void solve_reduced(NLProblem &nl_problem, Eigen::MatrixXd &sol, std::shared_ptr< polysolve::nonlinear::Solver > nl_solver)
Definition ALSolver.hpp:41
std::function< void(const double)> post_subsolve
Definition ALSolver.hpp:53
void solve_al(NLProblem &nl_problem, Eigen::MatrixXd &sol, std::shared_ptr< polysolve::nonlinear::Solver > nl_solver)
Definition ALSolver.hpp:29
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
Definition SolveData.hpp:59
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
io::OutputSpace output_space() const override
Get the output space of the variational formulation, for output purposes.
void init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path) override
Initialize the variational formulation with the given parameters.
std::vector< io::OutputField > elastic_output_fields(const io::OutputSample &sample, const Eigen::MatrixXd &solution, const io::OutputFieldOptions &options, const mesh::Obstacle *obstacle, const time_integrator::ImplicitTimeIntegrator *time_integrator, const std::vector< std::pair< std::string, std::shared_ptr< solver::Form > > > &named_forms, const solver::Form *elastic_form, const solver::ContactForm *contact_form=nullptr) const
void load_mesh(const mesh::Mesh &mesh, const json &args) override
std::shared_ptr< assembler::PressureAssembler > elasticity_pressure_assembler
std::vector< io::OutputField > output_fields(const io::OutputSample &sample, const Eigen::MatrixXd &solution, const io::OutputFieldOptions &options) const override
Get the output fields of the variational formulation, for output purposes.
std::shared_ptr< assembler::ViscousDamping > damping_assembler
void init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path) override
Initialize the variational formulation with the given parameters.
io::OutputSpace output_space() const override
Get the output space of the variational formulation, for output purposes.
std::shared_ptr< assembler::ViscousDampingPrev > damping_prev_assembler
void load_mesh(const mesh::Mesh &mesh, const json &args) override
bool is_contact_enabled() const override
Check if contact is enabled for the variational formulation, for output purposes.
std::vector< std::shared_ptr< solver::Form > > forms
std::shared_ptr< assembler::Problem > problem
current problem, it contains rhs and bc
Definition VarForm.hpp:191
virtual std::string name() const =0
Get the name of the variational formulation.
std::unique_ptr< mesh::Mesh > mesh_
Definition VarForm.hpp:203
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.
Definition MatrixIO.cpp:42
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.
Definition JSONUtils.hpp:38
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)
Definition Jacobian.cpp:212
void append_rows(DstMat &dst, const SrcMat &src)
spdlog::logger & logger()
Retrieves the current logger.
Definition Logger.cpp:44
Eigen::Matrix< double, Eigen::Dynamic, 1, 0, 3, 1 > VectorNd
Definition Types.hpp:11
nlohmann::json json
Definition Common.hpp:9
Eigen::Matrix< double, 1, Eigen::Dynamic, Eigen::RowMajor, 1, 3 > RowVectorNd
Definition Types.hpp:13
void log_and_throw_error(const std::string &msg)
Definition Logger.cpp:73
std::vector< std::string > fields
const mesh::Mesh * mesh