PolyFEM
Loading...
Searching...
No Matches
ScalarVarForm.cpp
Go to the documentation of this file.
1#include "ScalarVarForm.hpp"
2
5
8
11
17
18#include <unsupported/Eigen/SparseExtra>
19
20#include <polysolve/linear/FEMSolver.hpp>
21
22#include <algorithm>
23
24namespace polyfem::varform
25{
26 namespace
27 {
28 void write_matrix_market(const json &args, const StiffnessMatrix &stiffness)
29 {
30 const std::string full_mat_path = args["output"]["data"]["full_mat"];
31 if (!full_mat_path.empty())
32 Eigen::saveMarket(stiffness, full_mat_path);
33 }
34 } // namespace
35
37 {
39 space_.reset();
44 rhs_assembler_ = nullptr;
45 mass_.resize(0, 0);
46 pure_mass_.resize(0, 0);
47 avg_mass_ = 0;
48 rhs_.resize(0, 0);
49 primary_assembler_ = nullptr;
50 mass_assembler_ = nullptr;
51 pure_mass_assembler_ = nullptr;
52 t0 = 0;
53 time_steps = 0;
54 dt = 0;
55 time_integrator = nullptr;
56 }
57
58 void ScalarVarForm::init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path)
59 {
60 VarForm::init(formulation, units, args, out_path);
61 const bool is_time_dependent = args.contains("time") && !args["time"].is_null();
62
64 assert(primary_assembler_->name() == formulation);
65 assert(primary_assembler_->is_linear());
66 assert(!primary_assembler_->is_tensor());
67 mass_assembler_ = std::make_shared<assembler::Mass>();
68 pure_mass_assembler_ = std::make_shared<assembler::HRZMass>();
69
70 if (!args.contains("preset_problem"))
71 {
72 problem = std::make_shared<assembler::GenericScalarProblem>("GenericScalar");
73 problem->clear();
74
75 json tmp;
76 tmp["is_time_dependent"] = is_time_dependent;
77 problem->set_parameters(tmp, root_path);
78
79 auto bc = args["boundary_conditions"];
80 bc["root_path"] = root_path;
81 problem->set_parameters(bc, root_path);
82 problem->set_parameters(args["initial_conditions"], root_path);
83 problem->set_parameters(args["output"], root_path);
84 }
85 else
86 {
87 problem = problem::ProblemFactory::factory().get_problem(args["preset_problem"]["type"]);
88 problem->clear();
89 problem->set_parameters(args["preset_problem"], root_path);
90 }
91
92 problem->set_units(*primary_assembler_, units);
93
94 t0 = is_time_dependent ? args["time"]["t0"].get<double>() : 0.0;
95 time_steps = is_time_dependent ? args["time"]["time_steps"].get<int>() : 0;
96 dt = is_time_dependent ? args["time"]["dt"].get<double>() : 0.0;
97 }
98
99 void ScalarVarForm::load_mesh(const mesh::Mesh &mesh, const json &args)
100 {
103 pure_mass_assembler_->set_size(mass_assembler_->size());
104
105 problem->init(mesh);
106 }
107
108 void ScalarVarForm::build_basis(mesh::Mesh &mesh, const bool iso_parametric, const json &args)
109 {
110 assert(problem);
111 assert(primary_assembler_);
112 assert(mass_assembler_);
113 assert(pure_mass_assembler_);
114
115 Eigen::VectorXi space_disc_orders;
116 assign_discr_orders(args["space"]["discr_order"], mesh, space_disc_orders);
117
118 if (args["space"]["use_p_ref"])
119 {
121 mesh,
122 args["space"]["advanced"]["B"],
123 args["space"]["advanced"]["h1_formula"],
124 args["space"]["discr_order"],
125 args["space"]["advanced"]["discr_order_max"],
126 stats,
127 space_disc_orders);
128
129 logger().info("min p: {} max p: {}", space_disc_orders.minCoeff(), space_disc_orders.maxCoeff());
130 }
131
133 mesh,
134 iso_parametric,
135 space_disc_orders,
136 args["space"]["basis_type"],
137 args["space"]["poly_basis_type"],
139 /*value_dim=*/1,
140 args["space"]["advanced"]["quadrature_order"],
141 args["space"]["advanced"]["mass_quadrature_order"],
142 args["space"]["advanced"]["use_corner_quadrature"],
143 args["space"]["advanced"]["n_harmonic_samples"],
144 args["space"]["advanced"]["integral_constraints"],
145 space_,
146 boundary_);
147
149
150 problem->update_nodes(space_.space_in_node_to_node);
152
153 problem->setup_bc(
154 mesh,
156 /*fe_space_id=*/-1,
161 std::vector<int> unused_neumann_boundary_nodes;
162 problem->setup_bc(
163 mesh,
165 /*fe_space_id=*/-1,
169 unused_neumann_boundary_nodes);
170
171 problem->setup_nodal_bc(
172 mesh,
174 /*fe_space_id=*/-1,
177 problem->setup_nodal_bc(
178 mesh,
180 /*fe_space_id=*/-1,
183
184 for (const int n_id : boundary_.dirichlet_nodes)
185 {
186 const int tag = mesh.get_node_id(n_id);
187 if (problem->is_nodal_dimension_dirichlet(n_id, tag, 0))
188 boundary_.boundary_nodes.push_back(n_id);
189 }
190
192
195
196 const auto &current_bases = space_.geometry_basis_list();
197 if (args["space"]["advanced"]["count_flipped_els"])
198 stats.count_flipped_elements(mesh, current_bases);
199
200 const int n_samples = 10;
201 stats.compute_mesh_size(mesh, current_bases, n_samples, args["output"]["advanced"]["curved_mesh_size"]);
202
203 logger().info("flipped elements {}", stats.n_flipped);
204 logger().info("h: {}", stats.mesh_size);
205
206 if (space_.n_bases <= args["solver"]["advanced"]["cache_size"])
207 {
208 igl::Timer timer;
209 timer.start();
210 logger().info("Building cache...");
211 ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases);
212 mass_ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases, true);
213 pure_mass_ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases, true);
214 logger().info(" took {}s", timer.getElapsedTime());
215 }
216 else
217 {
221 }
222 }
223
225 {
226 json rhs_solver_params = args["solver"]["linear"];
227 if (!rhs_solver_params.contains("Pardiso"))
228 rhs_solver_params["Pardiso"] = {};
229 rhs_solver_params["Pardiso"]["mtype"] = -2;
230
231 rhs_assembler_ = std::make_shared<assembler::RhsAssembler>(
232 *primary_assembler_, *mesh_, nullptr,
236 args["space"]["advanced"]["bc_method"],
237 rhs_solver_params);
238 }
239
241 {
242 igl::Timer timer;
243 json p_params = {};
244 p_params["formulation"] = primary_assembler_->name();
245 p_params["root_path"] = root_path;
246 {
247 RowVectorNd min, max, delta;
248 mesh.bounding_box(min, max);
249 delta = (max - min) / 2. + min;
250 if (mesh.is_volume())
251 p_params["bbox_center"] = {delta(0), delta(1), delta(2)};
252 else
253 p_params["bbox_center"] = {delta(0), delta(1)};
254 }
255 problem->set_parameters(p_params, root_path);
256
257 rhs_.resize(0, 0);
258
259 timer.start();
260 logger().info("Assigning rhs...");
261
263 assert(rhs_assembler_ != nullptr);
264 rhs_assembler_->assemble(mass_assembler_->density(), rhs_);
265 rhs_ *= -1;
266
267 timings.assigning_rhs_time = timer.getElapsedTime();
268 logger().info(" took {}s", timings.assigning_rhs_time);
269 }
270
271 void ScalarVarForm::assemble_mass_mat(const mesh::Mesh &mesh, const json &args)
272 {
273 if (!problem->is_time_dependent())
274 {
275 avg_mass_ = 1;
277 return;
278 }
279
280 mass_.resize(0, 0);
281
282 igl::Timer timer;
283 timer.start();
284 logger().info("Assembling mass mat...");
285
287
288 assert(mass_.size() > 0);
289
290 avg_mass_ = 0;
291 for (int k = 0; k < mass_.outerSize(); ++k)
292 {
293 for (StiffnessMatrix::InnerIterator it(mass_, k); it; ++it)
294 {
295 assert(it.col() == k);
296 avg_mass_ += it.value();
297 }
298 }
299
300 avg_mass_ /= mass_.rows();
301 logger().info("average mass {}", avg_mass_);
302
303 if (args["solver"]["advanced"]["lump_mass_matrix"])
305
306 timer.stop();
307 timings.assembling_mass_mat_time = timer.getElapsedTime();
308 logger().info(" took {}s", timings.assembling_mass_mat_time);
309
310 stats.nn_zero = mass_.nonZeros();
311 stats.num_dofs = mass_.rows();
312 stats.mat_size = (long long)mass_.rows() * (long long)mass_.cols();
313 logger().info("sparsity: {}/{}", stats.nn_zero, stats.mat_size);
314 }
315
316 void ScalarVarForm::prepare_initial_solution(Eigen::MatrixXd &solution) const
317 {
318 assert(rhs_assembler_ != nullptr);
319
320 const bool was_solution_loaded = read_initial_x_from_file(
321 resolve_input_path(args["input"]["data"]["state"]), "u",
322 args["input"]["data"]["reorder"], space_.space_in_node_to_node,
323 /*dim=*/1, solution);
324
325 if (!was_solution_loaded)
326 {
327 if (problem->is_time_dependent())
328 rhs_assembler_->initial_solution(solution);
329 else
330 {
331 solution.resize(rhs_.size(), 1);
332 solution.setZero();
333 }
334 }
335 }
336
337 void ScalarVarForm::save_json(const Eigen::MatrixXd &solution, std::ostream &out) const
338 {
339 if (!mesh_)
340 {
341 logger().error("Load the mesh first!");
342 return;
343 }
344 if (solution.size() <= 0)
345 {
346 logger().error("Solve the problem first!");
347 return;
348 }
349
350 logger().info("Saving json...");
351 const int primary_size = space_.n_bases;
352 const Eigen::MatrixXd stats_solution =
353 solution.rows() >= primary_size
354 ? solution.topRows(primary_size).eval()
355 : solution;
356
357 nlohmann::json j;
359 args, space_.n_bases, /*n_auxiliary_bases=*/0,
360 stats_solution, *mesh_, space_.disc_orders, space_.disc_ordersq, *problem,
362 args["output"]["advanced"]["sol_at_node"], j);
363 out << j.dump(4) << std::endl;
364 }
365
367 {
368 Eigen::VectorXi output_orders = space_.disc_orders;
369 if (mesh_ && space_.disc_ordersq.size() == space_.disc_orders.size())
370 {
371 for (int e = 0; e < output_orders.size(); ++e)
372 {
373 if (mesh_->is_prism(e))
374 output_orders(e) = std::max(space_.disc_orders(e), space_.disc_ordersq(e));
375 }
376 }
377
378 return {
379 mesh_.get(),
381 output_orders,
382 &space_.polys,
385 nullptr,
386 nullptr,
389 }
390
391 io::OutStatsData ScalarVarForm::compute_errors(const Eigen::MatrixXd &solution)
392 {
393 if (!args["output"]["advanced"]["compute_error"])
394 return stats;
395
396 double tend = 0;
397 if (!args["time"].is_null())
398 tend = args["time"]["tend"];
399
401 return stats;
402 }
403
404 void ScalarVarForm::export_data(const Eigen::MatrixXd &solution) const
405 {
406 const io::OutputSpace space = output_space();
407 if (!space.mesh)
408 {
409 logger().error("Load the mesh first!");
410 return;
411 }
412 if (solution.size() <= 0)
413 {
414 logger().error("Solve the problem first!");
415 return;
416 }
417
419
420 const std::string vis_mesh_path = resolve_output_path(args["output"]["paraview"]["file_name"]);
421 const bool has_time = args.contains("time") && !args["time"].is_null();
422 double tend = has_time ? args["time"]["tend"].get<double>() : 1.0;
423 double dt = 1;
424 if (has_time)
425 dt = args["time"]["dt"];
426
427 const auto opts = export_options(space);
429 space,
430 output_field_function(solution, opts),
431 has_time,
432 tend, dt,
433 opts,
434 vis_mesh_path);
435
436 const std::string solution_path = resolve_output_path(args["output"]["data"]["solution"]);
437 if (!solution_path.empty())
438 {
439 const int primary_ndof = std::min<int>(solution.rows(), space_.n_bases);
440 const Eigen::MatrixXd primary_solution = solution.topRows(primary_ndof);
441 if (opts.reorder_output && space_.space_in_node_to_node.size() > 0)
442 {
443 const Eigen::MatrixXd nodal_solution = utils::unflatten(primary_solution, 1);
444 Eigen::MatrixXd reordered = Eigen::MatrixXd::Zero(nodal_solution.rows(), nodal_solution.cols());
445 for (int input_node = 0; input_node < space_.space_in_node_to_node.size(); ++input_node)
446 {
447 const int node = space_.space_in_node_to_node(input_node);
448 if (node >= 0 && node < nodal_solution.rows() && input_node < reordered.rows())
449 reordered.row(input_node) = nodal_solution.row(node);
450 }
451 io::write_matrix(solution_path, reordered);
452 }
453 else
454 {
455 io::write_matrix(solution_path, primary_solution);
456 }
457 }
458
459 const std::string nodes_path = resolve_output_path(args["output"]["data"]["nodes"]);
460 if (!nodes_path.empty())
461 {
462 Eigen::MatrixXd nodes = Eigen::MatrixXd::Zero(space_.n_bases, mesh_->dimension());
463 for (const basis::ElementBases &element_bases : space_.basis_list())
464 for (const basis::Basis &basis : element_bases.bases)
465 for (const auto &global : basis.global())
466 nodes.row(global.index) = global.node;
467 io::write_matrix(nodes_path, nodes);
468 }
469
470 const std::string stress_path = resolve_output_path(args["output"]["data"]["stress_mat"]);
471 const std::string mises_path = resolve_output_path(args["output"]["data"]["mises"]);
472 if ((!stress_path.empty() || !mises_path.empty()) && primary_assembler_)
473 {
474 Eigen::MatrixXd stress;
475 Eigen::VectorXd mises;
479 stress, mises);
480 if (!stress_path.empty())
481 io::write_matrix(stress_path, stress);
482 if (!mises_path.empty())
483 io::write_matrix(mises_path, mises);
484 }
485 }
486
487 std::vector<io::OutputField> ScalarVarForm::output_fields(
488 const io::OutputSample &sample,
489 const Eigen::MatrixXd &solution,
490 const io::OutputFieldOptions &options) const
491 {
492 std::vector<io::OutputField> fields;
493 if (!mesh_ || !problem || solution.size() <= 0)
494 return fields;
495
496 assert(problem->is_scalar());
497 const bool has_element_samples = sample.local_points.rows() > 0 && sample.local_points.rows() == sample.element_ids.size();
498 const int output_rows = sample.points.rows() > 0 ? sample.points.rows() : std::max<int>(sample.local_points.rows(), sample.node_ids.size());
499 const int primary_ndof = std::min<int>(solution.rows(), space_.n_bases);
500 const Eigen::MatrixXd primary_solution = solution.topRows(primary_ndof);
501
502 const auto sample_dof_field = [&](const Eigen::MatrixXd &dof_values, Eigen::MatrixXd &values, Eigen::MatrixXd *gradients = nullptr) -> bool {
503 if (dof_values.size() <= 0)
504 return false;
505
506 if (has_element_samples)
507 {
508 values.resize(sample.local_points.rows(), 1);
509 if (gradients)
510 gradients->resize(sample.local_points.rows(), mesh_->dimension());
511 for (int i = 0; i < sample.local_points.rows(); ++i)
512 {
513 const int element_id = sample.element_ids(i);
514 if (element_id < 0)
515 {
516 values(i) = 0;
517 if (gradients)
518 gradients->row(i).setZero();
519 continue;
520 }
521
522 Eigen::MatrixXd local_sol, local_grad;
525 element_id, sample.local_points.row(i), dof_values, local_sol, local_grad);
526 values(i) = local_sol(0);
527 if (gradients)
528 gradients->row(i) = local_grad;
529 }
530
531 if (output_rows > values.rows())
532 {
533 const int previous_rows = values.rows();
534 values.conservativeResize(output_rows, Eigen::NoChange);
535 values.bottomRows(output_rows - previous_rows).setZero();
536 if (gradients)
537 {
538 gradients->conservativeResize(output_rows, Eigen::NoChange);
539 gradients->bottomRows(output_rows - previous_rows).setZero();
540 }
541 }
542 return true;
543 }
544
545 if (sample.node_ids.size() > 0)
546 {
547 values.resize(sample.node_ids.size(), 1);
548 for (int i = 0; i < sample.node_ids.size(); ++i)
549 {
550 const int node_id = sample.node_ids(i);
551 if (node_id < 0 || node_id >= dof_values.rows())
552 return false;
553 values(i) = dof_values(node_id);
554 }
555 return sample.points.rows() == 0 || sample.points.rows() == values.rows();
556 }
557
558 return false;
559 };
560
561 const auto &paraview_options = args["output"]["paraview"]["options"];
562 if (has_element_samples && problem->has_exact_sol() && sample.points.rows() == output_rows)
563 {
564 Eigen::MatrixXd exact;
565 problem->exact(sample.points, sample.time, exact);
566 if (exact.rows() == output_rows)
567 {
568 if (options.export_field("exact"))
569 fields.push_back({"exact", exact, io::OutputField::Association::Point});
570 if (options.export_field("error"))
571 {
572 Eigen::MatrixXd values;
573 if (sample_dof_field(primary_solution, values))
574 fields.push_back({"error", (values - exact).rowwise().norm(), io::OutputField::Association::Point});
575 }
576 }
577 }
578
579 if ((paraview_options["nodes"] || (!options.fields.empty() && options.export_field("nodes")))
580 && has_element_samples
581 && sample.primitive_ids.size() == 0)
582 {
583 Eigen::MatrixXd dof_ids(primary_ndof, 1);
584 dof_ids.col(0).setLinSpaced(primary_ndof, 0, primary_ndof - 1);
585 Eigen::MatrixXd values;
586 if (sample_dof_field(dof_ids, values))
587 fields.push_back({"nodes", values, io::OutputField::Association::Point});
588 }
589
590 if ((paraview_options["jacobian_validity"] || (!options.fields.empty() && options.export_field("validity")))
591 && has_element_samples
592 && mesh_->dimension() == 1
593 && sample.primitive_ids.size() == 0)
594 {
595 const auto invalid_elements = utils::count_invalid(mesh_->dimension(), space_.basis_list(), space_.geometry_basis_list(), primary_solution);
596 Eigen::MatrixXd validity = Eigen::MatrixXd::Zero(output_rows, 1);
597 for (int i = 0; i < sample.element_ids.size(); ++i)
598 validity(i) = std::find(invalid_elements.begin(), invalid_elements.end(), sample.element_ids(i)) != invalid_elements.end();
599 fields.push_back({"validity", validity, io::OutputField::Association::Point});
600 }
601
602 const bool export_solution_gradient =
603 !options.fields.empty() && options.export_field("solution_gradient");
604 if (options.export_field("solution") || export_solution_gradient)
605 {
606 Eigen::MatrixXd values, gradients;
607 if (sample_dof_field(
608 solution, values,
609 export_solution_gradient ? &gradients : nullptr))
610 {
611 if (options.export_field("solution"))
612 fields.push_back({"solution", values, io::OutputField::Association::Point});
613 if (export_solution_gradient)
614 fields.push_back({"solution_gradient", gradients, io::OutputField::Association::Point});
615 }
616 }
617
618 if (paraview_options["material"] && has_element_samples)
619 {
620 const auto &params = primary_assembler_->parameters();
621 std::map<std::string, Eigen::MatrixXd> param_values;
622 for (const auto &[p, _] : params)
623 param_values[p].setZero(output_rows, 1);
624
625 Eigen::MatrixXd rhos = Eigen::MatrixXd::Zero(output_rows, 1);
626 const auto &density = mass_assembler_->density();
627 for (int i = 0; i < sample.local_points.rows(); ++i)
628 {
629 const int element_id = sample.element_ids(i);
630 if (element_id < 0)
631 continue;
632
633 for (const auto &[p, func] : params)
634 param_values.at(p)(i) = func(sample.local_points.row(i), sample.points.row(i), sample.time, element_id);
635 rhos(i) = density(sample.local_points.row(i), sample.points.row(i), sample.time, element_id);
636 }
637
638 for (const auto &[name, values] : param_values)
639 if (options.export_field(name))
640 fields.push_back({name, values, io::OutputField::Association::Point});
641 if (options.export_field("rho"))
642 fields.push_back({"rho", rhos, io::OutputField::Association::Point});
643 }
644
645 if (paraview_options["body_ids"] && options.export_field("body_ids") && has_element_samples)
646 {
647 Eigen::MatrixXd ids = Eigen::MatrixXd::Zero(output_rows, 1);
648 for (int i = 0; i < sample.element_ids.size(); ++i)
649 {
650 const int element_id = sample.element_ids(i);
651 if (element_id >= 0)
652 ids(i) = mesh_->get_body_id(element_id);
653 }
654 fields.push_back({"body_ids", ids, io::OutputField::Association::Point});
655 }
656
657 return fields;
658 }
659
660 void ScalarVarForm::build_stiffness_mat(StiffnessMatrix &stiffness)
661 {
662 igl::Timer timer;
663 timer.start();
664 logger().info("Assembling stiffness mat...");
665 assert(primary_assembler_->is_linear());
666 assert(problem->is_scalar());
667
668 primary_assembler_->assemble(mesh_->is_volume(), space_.n_bases, space_.basis_list(), space_.geometry_basis_list(), ass_vals_cache_, 0, stiffness);
669
670 timer.stop();
671 timings.assembling_stiffness_mat_time = timer.getElapsedTime();
672 logger().info(" took {}s", timings.assembling_stiffness_mat_time);
673
674 stats.nn_zero = stiffness.nonZeros();
675 stats.num_dofs = stiffness.rows();
676 stats.mat_size = (long long)stiffness.rows() * (long long)stiffness.cols();
677 logger().info("sparsity: {}/{}", stats.nn_zero, stats.mat_size);
678
679 write_matrix_market(args, stiffness);
680 }
681
682 void ScalarVarForm::solve_linear_system(
683 const std::unique_ptr<polysolve::linear::Solver> &solver,
685 Eigen::VectorXd &b,
686 const bool compute_spectrum,
687 Eigen::MatrixXd &sol)
688 {
689 assert(primary_assembler_->is_linear());
690 assert(problem->is_scalar());
691 assert(rhs_assembler_ != nullptr);
692
693 Eigen::VectorXd x;
694 stats.spectrum = dirichlet_solve(
695 *solver,
696 A,
697 b,
698 boundary_.boundary_nodes,
699 x,
700 space_.n_bases,
701 args["output"]["data"]["stiffness_mat"],
702 compute_spectrum,
703 /*is_problem_mixed=*/false,
704 /*use_avg_pressure=*/false);
705
706 sol = x;
707 solver->get_info(stats.solver_info);
708
709 const auto error = (A * x - b).norm();
710 if (error > 1e-4)
711 logger().error("Solver error: {}", error);
712 else
713 logger().debug("Solver error: {}", error);
714 }
715
716 void ScalarVarForm::solve_static(Eigen::MatrixXd &sol)
717 {
718 auto solver = polysolve::linear::Solver::create(args["solver"]["linear"], logger());
719 logger().info("{}...", solver->name());
720
721 const int gdiscr_order = mesh_->orders().size() <= 0 ? 1 : mesh_->orders().maxCoeff();
722 const QuadratureOrders boundary_samples = n_boundary_samples(space_.disc_orders.maxCoeff(), gdiscr_order);
723
724 rhs_assembler_->set_bc(
725 boundary_.local_boundary, boundary_.boundary_nodes, boundary_samples,
726 (primary_assembler_->name() != "Bilaplacian") ? boundary_.local_neumann_boundary : std::vector<mesh::LocalBoundary>(), rhs_);
727
729 build_stiffness_mat(A);
730
731 Eigen::VectorXd b = rhs_;
732 solve_linear_system(solver, A, b, args["output"]["advanced"]["spectrum"], sol);
733 }
734
735 void ScalarVarForm::solve_transient(Eigen::MatrixXd &sol)
736 {
737 assert(problem->is_time_dependent());
738 assert(rhs_assembler_ != nullptr);
739
740 auto solver = polysolve::linear::Solver::create(args["solver"]["linear"], logger());
741 logger().info("{}...", solver->name());
742
743 auto bdf = make_bdf_time_integrator();
744 bdf->init(sol, Eigen::VectorXd::Zero(sol.size()), Eigen::VectorXd::Zero(sol.size()), dt);
745 time_integrator = bdf;
746
747 save_timestep(t0, 0, t0, dt, sol);
748
749 Eigen::MatrixXd current_rhs = rhs_;
750
751 StiffnessMatrix stiffness;
752 build_stiffness_mat(stiffness);
753
754 const int gdiscr_order = mesh_->orders().size() <= 0 ? 1 : mesh_->orders().maxCoeff();
755 const QuadratureOrders n_b_samples = n_boundary_samples(space_.disc_orders.maxCoeff(), gdiscr_order);
756 for (int t = 1; t <= time_steps; ++t)
757 {
758 const double time = t0 + t * dt;
759
760 rhs_assembler_->compute_energy_grad(
761 boundary_.local_boundary, boundary_.boundary_nodes, mass_assembler_->density(), n_b_samples,
762 boundary_.local_neumann_boundary, rhs_, time, current_rhs);
763
764 rhs_assembler_->set_bc(
765 boundary_.local_boundary, boundary_.boundary_nodes, n_b_samples, boundary_.local_neumann_boundary, current_rhs, sol, time);
766
767 StiffnessMatrix A = mass_ / bdf->beta_dt() + stiffness;
768 Eigen::VectorXd b = (mass_ * bdf->weighted_sum_x_prevs()) / bdf->beta_dt();
769 for (int i : boundary_.boundary_nodes)
770 b[i] = 0;
771 b += current_rhs;
772
773 solve_linear_system(solver, A, b, args["output"]["advanced"]["spectrum"].get<bool>() && t == time_steps, sol);
774
775 bdf->update_quantities(sol);
776 save_timestep(time, t, t0, dt, sol);
777 save_step_state(t0, dt, t, time_integrator.get());
778
779 logger().info("{}/{} t={}", t, time_steps, time);
780 notify_time_step(t, time_steps, t0, dt);
781 }
782 }
783
784 void ScalarVarForm::solve_problem(Eigen::MatrixXd &sol)
785 {
786 stats.spectrum.setZero();
787
788 igl::Timer timer;
789 timer.start();
790 logger().info("Solving {}", primary_assembler_->name());
791
792 {
793 POLYFEM_SCOPED_TIMER("Setup RHS");
794
795 if (sol.size() <= 0)
796 prepare_initial_solution(sol);
797
798 if (sol.cols() > 1)
799 sol.conservativeResize(Eigen::NoChange, 1);
800 }
801
802 time_integrator = nullptr;
803 if (problem->is_time_dependent())
804 solve_transient(sol);
805 else
806 solve_static(sol);
807
808 timer.stop();
809 timings.solving_time = timer.getElapsedTime();
810 logger().info(" took {}s", timings.solving_time);
811 }
812} // namespace polyfem::varform
int x
std::array< Matrix< int, 3, 3 >, 3 > space_
#define POLYFEM_SCOPED_TIMER(...)
Definition Timer.hpp:10
static std::shared_ptr< Assembler > make_assembler(const std::string &formulation)
void init(const bool is_volume, const std::vector< basis::ElementBases > &bases, const std::vector< basis::ElementBases > &gbases, const bool is_mass=false)
computes the basis evaluation and geometric mapping for each of the given ElementBases in bases initi...
void init_empty(const bool is_mass=false)
initialize an empty cache.
Represents one basis function and its gradient.
Definition Basis.hpp:44
Stores the basis functions for a given element in a mesh (facet in 2d, cell in 3d).
static void interpolate_at_local_vals(const mesh::Mesh &mesh, const bool is_problem_scalar, const std::vector< basis::ElementBases > &bases, const std::vector< basis::ElementBases > &gbases, const int el_index, const Eigen::MatrixXd &local_pts, const Eigen::MatrixXd &fun, Eigen::MatrixXd &result, Eigen::MatrixXd &result_grad)
interpolate solution and gradient at element (calls interpolate_at_local_vals with sol)
static void compute_stress_at_quadrature_points(const mesh::Mesh &mesh, const bool is_problem_scalar, const std::vector< basis::ElementBases > &bases, const std::vector< basis::ElementBases > &gbases, const Eigen::VectorXi &disc_orders, const Eigen::VectorXi &disc_ordersq, const assembler::Assembler &assembler, const Eigen::MatrixXd &fun, const double t, Eigen::MatrixXd &result, Eigen::VectorXd &von_mises)
compute von mises stress at quadrature points for the function fun, also compute the interpolated fun...
void export_data(const OutputSpace &space, const OutputFieldFunction &output_fields, const bool is_time_dependent, const double tend_in, const double dt, const ExportOptions &opts, const std::string &vis_mesh_path) const
exports everytihng, txt, vtu, etc
Definition OutData.cpp:1124
double assigning_rhs_time
time to computing the rhs
double assembling_mass_mat_time
time to assembly mass
all stats from polyfem
int n_flipped
number of flipped elements, compute only when using count_flipped_els (false by default)
void count_flipped_elements(const polyfem::mesh::Mesh &mesh, const std::vector< polyfem::basis::ElementBases > &gbases)
counts the number of flipped elements
Definition OutData.cpp:1811
void compute_errors(const int n_bases, const std::vector< polyfem::basis::ElementBases > &bases, const std::vector< polyfem::basis::ElementBases > &gbases, const polyfem::mesh::Mesh &mesh, const assembler::Problem &problem, const double tend, const Eigen::MatrixXd &sol)
compute errors
Definition OutData.cpp:1856
void compute_mesh_size(const polyfem::mesh::Mesh &mesh_in, const std::vector< polyfem::basis::ElementBases > &bases_in, const int n_samples, const bool use_curved_mesh_size)
computes the mesh size, it samples every edges n_samples times uses curved_mesh_size (false by defaul...
Definition OutData.cpp:1728
long long nn_zero
non zeros and sytem matrix size num dof is the total dof in the system
double mesh_size
max edge lenght
void save_json(const nlohmann::json &args, const int n_bases, const int n_pressure_bases, const Eigen::MatrixXd &sol, const mesh::Mesh &mesh, const Eigen::VectorXi &disc_orders, const Eigen::VectorXi &disc_ordersq, const assembler::Problem &problem, const OutRuntimeData &runtime, const std::string &formulation, const bool isoparametric, const int sol_at_node_id, nlohmann::json &j) const
saves the output statistic to a json object
Definition OutData.cpp:2099
Abstract mesh class to capture 2d/3d conforming and non-conforming meshes.
Definition Mesh.hpp:41
virtual void bounding_box(RowVectorNd &min, RowVectorNd &max) const =0
computes the bbox of the mesh
virtual bool is_volume() const =0
checks if mesh is volume
void update_nodes(const Eigen::VectorXi &in_node_to_node)
Update the node ids to reorder them.
Definition Mesh.cpp:371
virtual int get_node_id(const int node_id) const
Get the boundary selection of a node.
Definition Mesh.hpp:497
static const ProblemFactory & factory()
std::shared_ptr< assembler::Problem > get_problem(const std::string &problem) const
static void p_refine(const mesh::Mesh &mesh, const double B, const bool h1_formula, const int base_p, const int discr_order_max, io::OutStatsData &stats, Eigen::VectorXi &disc_orders)
compute a priori prefinement
Definition APriori.cpp:242
const std::vector< basis::ElementBases > & geometry_basis_list() const
Definition FESpace.hpp:115
Eigen::VectorXi disc_orders
Primary polynomial degree for each mesh element.
Definition FESpace.hpp:71
Eigen::VectorXi disc_ordersq
Secondary polynomial degree for anisotropic bases, e.g. prisms.
Definition FESpace.hpp:74
int n_bases
Number of globally indexed scalar basis functions in the space.
Definition FESpace.hpp:65
Eigen::VectorXi space_in_node_to_node
Definition FESpace.hpp:91
std::map< int, std::pair< Eigen::MatrixXd, Eigen::MatrixXi > > polys_3d
Physical vertices and face connectivity for 3D polyhedral elements.
Definition FESpace.hpp:83
std::map< int, Eigen::MatrixXd > polys
Physical boundary samples for 2D polygonal elements.
Definition FESpace.hpp:80
const std::vector< basis::ElementBases > & basis_list() const
Definition FESpace.hpp:109
bool is_iso_parametric() const
Definition FESpace.hpp:104
void save_json(const Eigen::MatrixXd &solution, std::ostream &out) const override
Save the solution to a JSON file, for output purposes.
std::shared_ptr< assembler::HRZMass > pure_mass_assembler_
std::shared_ptr< time_integrator::ImplicitTimeIntegrator > time_integrator
std::shared_ptr< assembler::Mass > mass_assembler_
io::OutStatsData compute_errors(const Eigen::MatrixXd &solution) override
Get the error statistics of the variational formulation, for output purposes.
std::string name() const override
Get the name of the variational formulation.
void build_basis(mesh::Mesh &mesh, const bool iso_parametric, const json &args) override
assembler::AssemblyValsCache pure_mass_ass_vals_cache_
assembler::AssemblyValsCache mass_ass_vals_cache_
assembler::AssemblyValsCache ass_vals_cache_
void export_data(const Eigen::MatrixXd &solution) const override
void assemble_rhs(const mesh::Mesh &mesh) override
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.
void assemble_mass_mat(const mesh::Mesh &mesh, const json &args) override
void prepare_initial_solution(Eigen::MatrixXd &solution) const
std::shared_ptr< assembler::Assembler > primary_assembler_
std::shared_ptr< assembler::RhsAssembler > rhs_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.
void load_mesh(const mesh::Mesh &mesh, const json &args) override
io::OutputSpace output_space() const override
Get the output space of the variational formulation, for output purposes.
std::string resolve_input_path(const std::string &path, const bool only_if_exists=false) const
Definition VarForm.cpp:1057
static void rebuild_node_positions(const std::vector< basis::ElementBases > &bases, const std::vector< int > &node_ids, std::vector< RowVectorNd > &positions)
Definition VarForm.cpp:1071
std::shared_ptr< assembler::Problem > problem
current problem, it contains rhs and bc
Definition VarForm.hpp:191
std::unique_ptr< mesh::Mesh > mesh_
Definition VarForm.hpp:203
io::OutStatsData stats
Definition VarForm.hpp:195
static bool read_initial_x_from_file(const std::string &state_path, const std::string &x_name, const bool reorder, const Eigen::VectorXi &in_node_to_node, const int dim, Eigen::MatrixXd &x)
Definition VarForm.cpp:39
io::OutGeometryData::ExportOptions export_options(const io::OutputSpace &space) const
Definition VarForm.cpp:846
io::OutGeometryData output_geometry_
Definition VarForm.hpp:207
io::OutputFieldFunction output_field_function(const Eigen::MatrixXd &solution, const io::OutGeometryData::ExportOptions &opts) const
Definition VarForm.cpp:855
std::string resolve_output_path(const std::string &path) const
Definition VarForm.cpp:1062
void ensure_output_sampler() const
Definition VarForm.cpp:832
virtual void init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path)
Initialize the variational formulation with the given parameters.
Definition VarForm.cpp:280
void build_fe_space(mesh::Mesh &mesh, const bool iso_parametric, const Eigen::VectorXi &disc_orders, const std::string &basis_type, const std::string &poly_basis_type, const assembler::Assembler &space_assembler, const int value_dim, const int quadrature_order, const int mass_quadrature_order, const bool use_corner_quadrature, const int n_harmonic_samples, const int integral_constraints, FESpace &space, VarFormBoundaryState &boundary, std::shared_ptr< GeometryMapping > geometry=nullptr)
Definition VarForm.cpp:323
io::OutRuntimeData timings
runtime statistics
Definition VarForm.hpp:198
void set_materials(assembler::Assembler &assembler, const int size) const
Definition VarForm.cpp:817
virtual void reset()=0
Definition VarForm.cpp:268
void assign_discr_orders(const json &discr_order, const mesh::Mesh &mesh, Eigen::VectorXi &disc_orders)
Definition VarForm.cpp:745
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
Eigen::SparseMatrix< double > lump_matrix(const Eigen::SparseMatrix< double > &M)
Lump each row of a matrix into the diagonal.
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
bool write_matrix_market(const json &args, const StiffnessMatrix &stiffness)
spdlog::logger & logger()
Retrieves the current logger.
Definition Logger.cpp:44
std::array< int, 2 > QuadratureOrders
Definition Types.hpp:19
nlohmann::json json
Definition Common.hpp:9
Eigen::Matrix< double, 1, Eigen::Dynamic, Eigen::RowMajor, 1, 3 > RowVectorNd
Definition Types.hpp:13
Eigen::SparseMatrix< double, Eigen::ColMajor > StiffnessMatrix
Definition Types.hpp:24
bool export_field(const std::string &field) const
Definition OutData.cpp:51
std::vector< std::string > fields
Eigen::VectorXi node_ids
Eigen::VectorXi primitive_ids
Eigen::VectorXi element_ids
Eigen::MatrixXd local_points
const mesh::Mesh * mesh
std::vector< RowVectorNd > neumann_nodes_position
Definition FESpace.hpp:163
std::vector< mesh::LocalBoundary > local_boundary
Definition FESpace.hpp:155
std::vector< mesh::LocalBoundary > local_neumann_boundary
Definition FESpace.hpp:156
std::vector< mesh::LocalBoundary > total_local_boundary
Definition FESpace.hpp:154
std::vector< RowVectorNd > dirichlet_nodes_position
Definition FESpace.hpp:161