PolyFEM
Loading...
Searching...
No Matches
BilaplacianVarForm.cpp
Go to the documentation of this file.
2
19
20#include <polysolve/linear/FEMSolver.hpp>
21
22namespace polyfem::varform
23{
24 using namespace varform::internal;
25
27 {
29 space_.reset();
34 rhs_assembler_ = nullptr;
35 mass_.resize(0, 0);
36 pure_mass_.resize(0, 0);
37 avg_mass_ = 0;
38 rhs_.resize(0, 0);
42 primary_assembler_ = nullptr;
43 mass_assembler_ = nullptr;
44 pure_mass_assembler_ = nullptr;
45 mixed_assembler_ = nullptr;
46 pressure_assembler_ = nullptr;
47 t0 = 0;
48 time_steps = 0;
49 dt = 0;
50 time_integrator = nullptr;
51 }
52
53 void BilaplacianVarForm::init(const std::string &formulation, const Units &units, const json &args, const std::string &out_path)
54 {
55 VarForm::init(formulation, units, args, out_path);
56 const bool is_time_dependent = args.contains("time") && !args["time"].is_null();
57
59 mass_assembler_ = std::make_shared<assembler::Mass>();
60 pure_mass_assembler_ = std::make_shared<assembler::HRZMass>();
63
64 if (!args.contains("preset_problem"))
65 {
66 problem = std::make_shared<assembler::GenericScalarProblem>("GenericScalar");
67 problem->clear();
68 json tmp;
69 tmp["is_time_dependent"] = is_time_dependent;
70 problem->set_parameters(tmp, root_path);
71
72 auto bc = args["boundary_conditions"];
73 bc["root_path"] = root_path;
74 problem->set_parameters(bc, root_path);
75 problem->set_parameters(args["initial_conditions"], root_path);
76 problem->set_parameters(args["output"], root_path);
77 }
78 else
79 {
80 problem = problem::ProblemFactory::factory().get_problem(args["preset_problem"]["type"]);
81 problem->clear();
82 problem->set_parameters(args["preset_problem"], root_path);
83 }
84
85 problem->set_units(*primary_assembler_, units);
86 t0 = is_time_dependent ? args["time"]["t0"].get<double>() : 0.0;
87 time_steps = is_time_dependent ? args["time"]["time_steps"].get<int>() : 0;
88 dt = is_time_dependent ? args["time"]["dt"].get<double>() : 0.0;
89 }
90
91 void BilaplacianVarForm::save_json(const Eigen::MatrixXd &solution, std::ostream &out) const
92 {
93 if (!mesh_)
94 {
95 logger().error("Load the mesh first!");
96 return;
97 }
98 if (solution.size() <= 0)
99 {
100 logger().error("Solve the problem first!");
101 return;
102 }
103
104 logger().info("Saving json...");
105 const Eigen::MatrixXd stats_solution =
106 solution.rows() >= space_.n_bases
107 ? solution.topRows(space_.n_bases).eval()
108 : solution;
109
110 nlohmann::json j;
113 stats_solution, *mesh_, space_.disc_orders, space_.disc_ordersq, *problem,
115 args["output"]["advanced"]["sol_at_node"], j);
116 out << j.dump(4) << std::endl;
117 }
118
120 {
121 Eigen::VectorXi output_orders = space_.disc_orders;
122 if (mesh_ && space_.disc_ordersq.size() == space_.disc_orders.size())
123 {
124 for (int e = 0; e < output_orders.size(); ++e)
125 {
126 if (mesh_->is_prism(e))
127 output_orders(e) = std::max(space_.disc_orders(e), space_.disc_ordersq(e));
128 }
129 }
130
131 return {
132 mesh_.get(),
134 output_orders,
135 &space_.polys,
138 nullptr,
139 nullptr,
142 }
143
145 {
146 if (!args["output"]["advanced"]["compute_error"])
147 return stats;
148
149 double tend = 0;
150 if (!args["time"].is_null())
151 tend = args["time"]["tend"];
152
153 Eigen::MatrixXd value, pressure;
154 split_solution(solution, value, pressure);
156 return stats;
157 }
158
159 void BilaplacianVarForm::export_data(const Eigen::MatrixXd &solution) const
160 {
161 const io::OutputSpace space = output_space();
162 if (!space.mesh)
163 {
164 logger().error("Load the mesh first!");
165 return;
166 }
167 if (solution.size() <= 0)
168 {
169 logger().error("Solve the problem first!");
170 return;
171 }
172
174
175 const std::string vis_mesh_path = resolve_output_path(args["output"]["paraview"]["file_name"]);
176 const bool has_time = args.contains("time") && !args["time"].is_null();
177 double tend = has_time ? args["time"]["tend"].get<double>() : 1.0;
178 double dt = 1;
179 if (has_time)
180 dt = args["time"]["dt"];
181
182 const auto opts = export_options(space);
184 space,
185 output_field_function(solution, opts),
186 has_time,
187 tend, dt,
188 opts,
189 vis_mesh_path);
190
191 Eigen::MatrixXd value, pressure;
192 split_solution(solution, value, pressure);
193
194 const std::string solution_path = resolve_output_path(args["output"]["data"]["solution"]);
195 if (!solution_path.empty())
196 {
197 const int primary_ndof = std::min<int>(value.rows(), space_.n_bases);
198 const Eigen::MatrixXd primary_solution = value.topRows(primary_ndof);
199 if (opts.reorder_output && space_.space_in_node_to_node.size() > 0)
200 {
201 const Eigen::MatrixXd nodal_solution = utils::unflatten(primary_solution, 1);
202 Eigen::MatrixXd reordered = Eigen::MatrixXd::Zero(nodal_solution.rows(), nodal_solution.cols());
203 for (int input_node = 0; input_node < space_.space_in_node_to_node.size(); ++input_node)
204 {
205 const int node = space_.space_in_node_to_node(input_node);
206 if (node >= 0 && node < nodal_solution.rows() && input_node < reordered.rows())
207 reordered.row(input_node) = nodal_solution.row(node);
208 }
209 io::write_matrix(solution_path, reordered);
210 }
211 else
212 {
213 io::write_matrix(solution_path, primary_solution);
214 }
215 }
216
217 const std::string nodes_path = resolve_output_path(args["output"]["data"]["nodes"]);
218 if (!nodes_path.empty())
219 {
220 Eigen::MatrixXd nodes = Eigen::MatrixXd::Zero(space_.n_bases, mesh_->dimension());
221 for (const basis::ElementBases &element_bases : space_.basis_list())
222 for (const basis::Basis &basis : element_bases.bases)
223 for (const auto &global : basis.global())
224 nodes.row(global.index) = global.node;
225 io::write_matrix(nodes_path, nodes);
226 }
227
228 const std::string stress_path = resolve_output_path(args["output"]["data"]["stress_mat"]);
229 const std::string mises_path = resolve_output_path(args["output"]["data"]["mises"]);
230 if ((!stress_path.empty() || !mises_path.empty()) && primary_assembler_)
231 {
232 Eigen::MatrixXd stress;
233 Eigen::VectorXd mises;
237 stress, mises);
238 if (!stress_path.empty())
239 io::write_matrix(stress_path, stress);
240 if (!mises_path.empty())
241 io::write_matrix(mises_path, mises);
242 }
243 }
244
245 void BilaplacianVarForm::load_mesh(const mesh::Mesh &mesh, const json &args)
246 {
249 pure_mass_assembler_->set_size(mass_assembler_->size());
250 problem->init(mesh);
251
253 mixed_assembler_->set_size(1);
256 }
257
258 std::shared_ptr<assembler::RhsAssembler> BilaplacianVarForm::build_rhs_assembler(
259 const int n_bases,
260 const std::vector<basis::ElementBases> &bases,
261 const assembler::AssemblyValsCache &ass_vals_cache)
262 {
263 json rhs_solver_params = args["solver"]["linear"];
264 if (!rhs_solver_params.contains("Pardiso"))
265 rhs_solver_params["Pardiso"] = {};
266 rhs_solver_params["Pardiso"]["mtype"] = -2;
267
268 return std::make_shared<assembler::RhsAssembler>(
269 *primary_assembler_, *mesh_, nullptr,
272 n_bases, 1, bases, space_.geometry_basis_list(), ass_vals_cache, *problem,
273 args["space"]["advanced"]["bc_method"],
274 rhs_solver_params);
275 }
276
277 void BilaplacianVarForm::build_basis(mesh::Mesh &mesh, const bool iso_parametric, const json &args)
278 {
279 assert(problem);
280 assert(primary_assembler_);
281 assert(mass_assembler_);
282 assert(pure_mass_assembler_);
283
284 Eigen::VectorXi space_disc_orders;
285 assign_discr_orders(args["space"]["discr_order"], mesh, space_disc_orders);
286
287 if (args["space"]["use_p_ref"])
288 {
290 mesh,
291 args["space"]["advanced"]["B"],
292 args["space"]["advanced"]["h1_formula"],
293 args["space"]["discr_order"],
294 args["space"]["advanced"]["discr_order_max"],
295 stats,
296 space_disc_orders);
297
298 logger().info("min p: {} max p: {}", space_disc_orders.minCoeff(), space_disc_orders.maxCoeff());
299 }
300
302 mesh,
303 iso_parametric,
304 space_disc_orders,
305 args["space"]["basis_type"],
306 args["space"]["poly_basis_type"],
308 /*value_dim=*/1,
309 args["space"]["advanced"]["quadrature_order"],
310 args["space"]["advanced"]["mass_quadrature_order"],
311 args["space"]["advanced"]["use_corner_quadrature"],
312 args["space"]["advanced"]["n_harmonic_samples"],
313 args["space"]["advanced"]["integral_constraints"],
314 space_,
315 boundary_);
316
317 problem->update_nodes(space_.space_in_node_to_node);
319
320 const auto &current_bases = space_.geometry_basis_list();
321 if (args["space"]["advanced"]["count_flipped_els"])
322 stats.count_flipped_elements(mesh, current_bases);
323
324 const int n_samples = 10;
325 stats.compute_mesh_size(mesh, current_bases, n_samples, args["output"]["advanced"]["curved_mesh_size"]);
326
327 logger().info("flipped elements {}", stats.n_flipped);
328 logger().info("h: {}", stats.mesh_size);
329
330 if (space_.n_bases <= args["solver"]["advanced"]["cache_size"])
331 {
332 igl::Timer timer;
333 timer.start();
334 logger().info("Building cache...");
335 ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases);
336 mass_ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases, true);
337 pure_mass_ass_vals_cache_.init(mesh.is_volume(), space_.basis_list(), current_bases, true);
338 logger().info(" took {}s", timer.getElapsedTime());
339 }
340 else
341 {
345 }
346
347 if (space_.disc_orders.maxCoeff() != space_.disc_orders.minCoeff())
348 log_and_throw_error("p refinement not supported in mixed formulation!");
349 if (!space_.poly_edge_to_data.empty())
350 log_and_throw_error("Polygonal bases are not supported in mixed formulations!");
351
352 const int prev_bases = space_.n_bases;
353 const auto &all_boundary = boundary_.total_local_boundary;
354 const bool use_corner_quadrature = args["space"]["advanced"]["use_corner_quadrature"];
355 const int quadrature_order = args["space"]["advanced"]["quadrature_order"].get<int>();
356 const int mass_quadrature_order = args["space"]["advanced"]["mass_quadrature_order"].get<int>();
357 const int order = args["space"]["pressure_discr_order"];
358 Eigen::VectorXi pressure_disc_orders(mesh.n_elements());
359 pressure_disc_orders.setConstant(order);
360 // to avoid serendipity
361 const std::string pressure_basis_type = args["space"]["basis_type"].get<std::string>() == "Bernstein" ? "Bernstein" : "Lagrange";
363 mesh,
364 /*iso_parametric=*/true,
365 pressure_disc_orders,
366 pressure_basis_type,
367 args["space"]["poly_basis_type"],
369 /*value_dim=*/1,
370 quadrature_order,
371 mass_quadrature_order,
372 use_corner_quadrature,
373 args["space"]["advanced"]["n_harmonic_samples"],
374 args["space"]["advanced"]["integral_constraints"],
378
379 assert(space_.basis_list().size() == pressure_space_.basis_list().size());
380 for (int i = 0; i < int(pressure_space_.basis_list().size()); ++i)
381 {
383 space_.basis_list()[i].compute_quadrature(b_quad);
384 (*pressure_space_.bases)[i].set_quadrature([b_quad](quadrature::Quadrature &quad) { quad = b_quad; });
385 }
386
388 for (const auto &lb : all_boundary)
389 boundary_.local_boundary.emplace_back(lb);
391
392 problem->setup_bc(
393 mesh, space_.n_bases,
403
406
407 for (int i = prev_bases; i < space_.n_bases; ++i)
408 boundary_.boundary_nodes.push_back(i);
409
411
412 if (space_.n_bases <= args["solver"]["advanced"]["cache_size"])
414 else
416
418
419 logger().info("n pressure bases: {}", pressure_space_.n_bases);
420 }
421
423 {
424 igl::Timer timer;
425 json p_params = {};
426 p_params["formulation"] = primary_assembler_->name();
427 p_params["root_path"] = root_path;
428 {
429 RowVectorNd min, max, delta;
430 mesh.bounding_box(min, max);
431 delta = (max - min) / 2. + min;
432 if (mesh.is_volume())
433 p_params["bbox_center"] = {delta(0), delta(1), delta(2)};
434 else
435 p_params["bbox_center"] = {delta(0), delta(1)};
436 }
437 problem->set_parameters(p_params, root_path);
438
439 rhs_.resize(0, 0);
440
441 timer.start();
442 logger().info("Assigning rhs...");
443
445 assert(rhs_assembler_ != nullptr);
446 rhs_assembler_->assemble(mass_assembler_->density(), rhs_);
447 rhs_ *= -1;
448
449 timings.assigning_rhs_time = timer.getElapsedTime();
450 logger().info(" took {}s", timings.assigning_rhs_time);
451
452 const int prev_size = rhs_.rows();
453 rhs_.conservativeResize(prev_size + pressure_space_.n_bases, rhs_.cols());
455 {
456 rhs_.bottomRows(pressure_space_.n_bases).setZero();
457 }
458 else
459 {
460 Eigen::MatrixXd tmp = Eigen::MatrixXd::Zero(pressure_space_.n_bases, 1);
462 const int gdiscr_order = mesh_->orders().size() <= 0 ? 1 : mesh_->orders().maxCoeff();
463 const QuadratureOrders boundary_samples = n_boundary_samples(space_.disc_orders.maxCoeff(), gdiscr_order);
464 tmp_rhs_assembler->set_bc(
465 std::vector<mesh::LocalBoundary>(), std::vector<int>(), boundary_samples, boundary_.local_neumann_boundary, tmp);
466 rhs_.bottomRows(pressure_space_.n_bases) = tmp;
467 }
468 }
469
471 {
472 if (!problem->is_time_dependent())
473 {
474 avg_mass_ = 1;
476 return;
477 }
478
479 mass_.resize(0, 0);
480 igl::Timer timer;
481 timer.start();
482 logger().info("Assembling mass mat...");
484 avg_mass_ = 0;
485 for (int k = 0; k < mass_.outerSize(); ++k)
486 for (StiffnessMatrix::InnerIterator it(mass_, k); it; ++it)
487 {
488 assert(it.col() == k);
489 avg_mass_ += it.value();
490 }
491 avg_mass_ /= std::max(1, int(mass_.rows()));
492 if (args["solver"]["advanced"]["lump_mass_matrix"])
494 timer.stop();
495 timings.assembling_mass_mat_time = timer.getElapsedTime();
496 logger().info(" took {}s", timings.assembling_mass_mat_time);
497 stats.nn_zero = mass_.nonZeros();
498 stats.num_dofs = mass_.rows();
499 stats.mat_size = (long long)mass_.rows() * (long long)mass_.cols();
500 }
501
506
507 void BilaplacianVarForm::prepare_initial_solution(Eigen::MatrixXd &sol) const
508 {
509 if (sol.size() <= 0)
510 {
511 assert(rhs_assembler_ != nullptr);
512 const bool was_solution_loaded = read_initial_x_from_file(
513 resolve_input_path(args["input"]["data"]["state"]), "u",
514 args["input"]["data"]["reorder"], space_.space_in_node_to_node,
515 /*dim=*/1, sol);
516
517 if (!was_solution_loaded)
518 {
519 if (problem->is_time_dependent())
520 rhs_assembler_->initial_solution(sol);
521 else
522 {
523 sol.resize(rhs_.size(), 1);
524 sol.setZero();
525 }
526 }
527 }
528 if (sol.cols() > 1)
529 sol.conservativeResize(Eigen::NoChange, 1);
530 sol.conservativeResize(stacked_ndof(), sol.cols());
531 sol.bottomRows(pressure_space_.n_bases).setZero();
532 }
533
534 void BilaplacianVarForm::split_solution(const Eigen::MatrixXd &stacked, Eigen::MatrixXd &primary, Eigen::MatrixXd &pressure) const
535 {
536 const int cols = std::max(1, int(stacked.cols()));
537 primary.setZero(space_.n_bases, cols);
538 pressure.setZero(pressure_space_.n_bases, cols);
539 const int primary_rows = std::min(space_.n_bases, int(stacked.rows()));
540 if (primary_rows > 0)
541 primary.topRows(primary_rows) = stacked.topRows(primary_rows);
542 if (stacked.rows() > space_.n_bases)
543 {
544 const int pressure_rows = std::min(pressure_space_.n_bases, int(stacked.rows()) - space_.n_bases);
545 if (pressure_rows > 0)
546 pressure.topRows(pressure_rows) = stacked.middleRows(space_.n_bases, pressure_rows);
547 }
548 }
549
551 {
552 igl::Timer timer;
553 timer.start();
554 logger().info("Assembling stiffness mat...");
555
556 StiffnessMatrix main_stiffness, mixed_stiffness, aux_stiffness;
557 primary_assembler_->assemble(mesh_->is_volume(), space_.n_bases, space_.basis_list(), space_.geometry_basis_list(), ass_vals_cache_, 0, main_stiffness);
560
562 space_.n_bases, pressure_space_.n_bases, 1, /*add_average=*/false,
563 main_stiffness, mixed_stiffness, aux_stiffness, stiffness);
564
565 timer.stop();
566 timings.assembling_stiffness_mat_time = timer.getElapsedTime();
567 logger().info(" took {}s", timings.assembling_stiffness_mat_time);
568 stats.nn_zero = stiffness.nonZeros();
569 stats.num_dofs = stiffness.rows();
570 stats.mat_size = (long long)stiffness.rows() * (long long)stiffness.cols();
571 write_matrix_market(args, stiffness);
572 }
573
575 const std::unique_ptr<polysolve::linear::Solver> &solver,
577 Eigen::VectorXd &b,
578 const bool compute_spectrum,
579 Eigen::MatrixXd &sol)
580 {
581 Eigen::VectorXd x;
582 stats.spectrum = dirichlet_solve(
583 *solver,
584 A,
585 b,
587 x,
589 args["output"]["data"]["stiffness_mat"],
590 compute_spectrum,
591 /*is_fluid=*/false,
592 /*use_avg_pressure=*/false);
593 sol = x;
594 solver->get_info(stats.solver_info);
595 }
596
598 {
599 auto solver = polysolve::linear::Solver::create(args["solver"]["linear"], logger());
600 logger().info("{}...", solver->name());
601 const int gdiscr_order = mesh_->orders().size() <= 0 ? 1 : mesh_->orders().maxCoeff();
602 const QuadratureOrders boundary_samples = n_boundary_samples(space_.disc_orders.maxCoeff(), gdiscr_order);
603 rhs_assembler_->set_bc(
605 (primary_assembler_->name() != "Bilaplacian") ? boundary_.local_neumann_boundary : std::vector<mesh::LocalBoundary>(), rhs_);
608 Eigen::VectorXd b = rhs_;
609 solve_linear_system(solver, A, b, args["output"]["advanced"]["spectrum"], sol);
610 }
611
613 {
614 auto solver = polysolve::linear::Solver::create(args["solver"]["linear"], logger());
615 logger().info("{}...", solver->name());
616
617 Eigen::MatrixXd value, pressure;
618 split_solution(sol, value, pressure);
619 auto bdf = make_bdf_time_integrator();
620 bdf->init(
621 value,
622 Eigen::MatrixXd::Zero(value.rows(), value.cols()),
623 Eigen::MatrixXd::Zero(value.rows(), value.cols()),
624 dt);
625 time_integrator = bdf;
626
627 save_timestep(t0, 0, t0, dt, sol);
628
629 Eigen::MatrixXd current_rhs = rhs_;
630 StiffnessMatrix stiffness, expanded_mass;
631 build_stiffness_mat(stiffness);
632 expand_primary_matrix(stacked_ndof(), mass_, expanded_mass);
633 const int gdiscr_order = mesh_->orders().size() <= 0 ? 1 : mesh_->orders().maxCoeff();
634 const QuadratureOrders boundary_samples = n_boundary_samples(space_.disc_orders.maxCoeff(), gdiscr_order);
635
636 for (int t = 1; t <= time_steps; ++t)
637 {
638 const double time = t0 + t * dt;
639 rhs_assembler_->compute_energy_grad(
641 current_rhs);
642 rhs_assembler_->set_bc(
643 boundary_.local_boundary, boundary_.boundary_nodes, boundary_samples, boundary_.local_neumann_boundary, current_rhs, value, time);
644
645 if (current_rhs.rows() != stacked_ndof())
646 {
647 const int old_rows = current_rhs.rows();
648 current_rhs.conservativeResize(stacked_ndof(), current_rhs.cols());
649 if (stacked_ndof() > old_rows)
650 current_rhs.bottomRows(stacked_ndof() - old_rows).setZero();
651 }
652 current_rhs.bottomRows(pressure_space_.n_bases).setZero();
653
654 StiffnessMatrix A = expanded_mass / bdf->beta_dt() + stiffness;
655 Eigen::VectorXd b = Eigen::VectorXd::Zero(stacked_ndof());
656 b.head(space_.n_bases) = (mass_ * bdf->weighted_sum_x_prevs()) / bdf->beta_dt();
657 for (int i : boundary_.boundary_nodes)
658 b[i] = 0;
659 b += current_rhs;
660
661 solve_linear_system(solver, A, b, args["output"]["advanced"]["spectrum"].get<bool>() && t == time_steps, sol);
662 split_solution(sol, value, pressure);
663 bdf->update_quantities(value.col(0));
664
665 save_timestep(time, t, t0, dt, sol);
667 logger().info("{}/{} t={}", t, time_steps, time);
669 }
670 }
671
672 void BilaplacianVarForm::solve_problem(Eigen::MatrixXd &sol)
673 {
674 stats.spectrum.setZero();
675 igl::Timer timer;
676 timer.start();
677 logger().info("Solving {}", primary_assembler_->name());
679 if (problem->is_time_dependent())
681 else
682 {
683 time_integrator = nullptr;
685 }
686 timer.stop();
687 timings.solving_time = timer.getElapsedTime();
688 logger().info(" took {}s", timings.solving_time);
689 }
690
691 std::vector<io::OutputField> BilaplacianVarForm::output_fields(
692 const io::OutputSample &sample,
693 const Eigen::MatrixXd &solution,
694 const io::OutputFieldOptions &options) const
695 {
696 std::vector<io::OutputField> fields;
697 if (!mesh_ || !problem || solution.size() <= 0)
698 return fields;
699
700 Eigen::MatrixXd value, pressure;
701 split_solution(solution, value, pressure);
702
703 const bool has_element_samples = sample.local_points.rows() > 0 && sample.local_points.rows() == sample.element_ids.size();
704 const int output_rows = sample.points.rows() > 0 ? sample.points.rows() : std::max<int>(sample.local_points.rows(), sample.node_ids.size());
705 const int primary_ndof = std::min<int>(value.rows(), space_.n_bases);
706 const Eigen::MatrixXd primary_solution = value.topRows(primary_ndof);
707
708 const auto sample_dof_field = [&](const Eigen::MatrixXd &dof_values, Eigen::MatrixXd &values, Eigen::MatrixXd *gradients = nullptr) -> bool {
709 if (dof_values.size() <= 0)
710 return false;
711
712 if (has_element_samples)
713 {
714 values.resize(sample.local_points.rows(), 1);
715 if (gradients)
716 gradients->resize(sample.local_points.rows(), mesh_->dimension());
717 for (int i = 0; i < sample.local_points.rows(); ++i)
718 {
719 const int element_id = sample.element_ids(i);
720 if (element_id < 0)
721 {
722 values(i) = 0;
723 if (gradients)
724 gradients->row(i).setZero();
725 continue;
726 }
727
728 Eigen::MatrixXd local_sol, local_grad;
731 element_id, sample.local_points.row(i), dof_values, local_sol, local_grad);
732 values(i) = local_sol(0);
733 if (gradients)
734 gradients->row(i) = local_grad;
735 }
736
737 if (output_rows > values.rows())
738 {
739 const int previous_rows = values.rows();
740 values.conservativeResize(output_rows, Eigen::NoChange);
741 values.bottomRows(output_rows - previous_rows).setZero();
742 if (gradients)
743 {
744 gradients->conservativeResize(output_rows, Eigen::NoChange);
745 gradients->bottomRows(output_rows - previous_rows).setZero();
746 }
747 }
748 return true;
749 }
750
751 if (sample.node_ids.size() > 0)
752 {
753 values.resize(sample.node_ids.size(), 1);
754 for (int i = 0; i < sample.node_ids.size(); ++i)
755 {
756 const int node_id = sample.node_ids(i);
757 if (node_id < 0 || node_id >= dof_values.rows())
758 return false;
759 values(i) = dof_values(node_id);
760 }
761 return sample.points.rows() == 0 || sample.points.rows() == values.rows();
762 }
763
764 return false;
765 };
766
767 const auto &paraview_options = args["output"]["paraview"]["options"];
768 if (has_element_samples && problem->has_exact_sol() && sample.points.rows() == output_rows)
769 {
770 Eigen::MatrixXd exact;
771 problem->exact(sample.points, sample.time, exact);
772 if (exact.rows() == output_rows)
773 {
774 if (options.export_field("exact"))
775 fields.push_back({"exact", exact, io::OutputField::Association::Point});
776 if (options.export_field("error"))
777 {
778 Eigen::MatrixXd values;
779 if (sample_dof_field(primary_solution, values))
780 fields.push_back({"error", (values - exact).rowwise().norm(), io::OutputField::Association::Point});
781 }
782 }
783 }
784
785 if ((paraview_options["nodes"] || (!options.fields.empty() && options.export_field("nodes")))
786 && has_element_samples
787 && sample.primitive_ids.size() == 0)
788 {
789 Eigen::MatrixXd dof_ids(primary_ndof, 1);
790 dof_ids.col(0).setLinSpaced(primary_ndof, 0, primary_ndof - 1);
791 Eigen::MatrixXd values;
792 if (sample_dof_field(dof_ids, values))
793 fields.push_back({"nodes", values, io::OutputField::Association::Point});
794 }
795
796 if ((paraview_options["jacobian_validity"] || (!options.fields.empty() && options.export_field("validity")))
797 && has_element_samples
798 && mesh_->dimension() == 1
799 && sample.primitive_ids.size() == 0)
800 {
801 const auto invalid_elements = utils::count_invalid(mesh_->dimension(), space_.basis_list(), space_.geometry_basis_list(), primary_solution);
802 Eigen::MatrixXd validity = Eigen::MatrixXd::Zero(output_rows, 1);
803 for (int i = 0; i < sample.element_ids.size(); ++i)
804 validity(i) = std::find(invalid_elements.begin(), invalid_elements.end(), sample.element_ids(i)) != invalid_elements.end();
805 fields.push_back({"validity", validity, io::OutputField::Association::Point});
806 }
807
808 const bool export_solution_gradient =
809 !options.fields.empty() && options.export_field("solution_gradient");
810 if (options.export_field("solution") || export_solution_gradient)
811 {
812 Eigen::MatrixXd values, gradients;
813 if (sample_dof_field(
814 value, values,
815 export_solution_gradient ? &gradients : nullptr))
816 {
817 if (options.export_field("solution"))
818 fields.push_back({"solution", values, io::OutputField::Association::Point});
819 if (export_solution_gradient)
820 fields.push_back({"solution_gradient", gradients, io::OutputField::Association::Point});
821 }
822 }
823
824 if (paraview_options["material"] && has_element_samples)
825 {
826 const auto &params = primary_assembler_->parameters();
827 std::map<std::string, Eigen::MatrixXd> param_values;
828 for (const auto &[p, _] : params)
829 param_values[p].setZero(output_rows, 1);
830
831 Eigen::MatrixXd rhos = Eigen::MatrixXd::Zero(output_rows, 1);
832 const auto &density = mass_assembler_->density();
833 for (int i = 0; i < sample.local_points.rows(); ++i)
834 {
835 const int element_id = sample.element_ids(i);
836 if (element_id < 0)
837 continue;
838
839 for (const auto &[p, func] : params)
840 param_values.at(p)(i) = func(sample.local_points.row(i), sample.points.row(i), sample.time, element_id);
841 rhos(i) = density(sample.local_points.row(i), sample.points.row(i), sample.time, element_id);
842 }
843
844 for (const auto &[name, values] : param_values)
845 if (options.export_field(name))
846 fields.push_back({name, values, io::OutputField::Association::Point});
847 if (options.export_field("rho"))
848 fields.push_back({"rho", rhos, io::OutputField::Association::Point});
849 }
850
851 if (paraview_options["body_ids"] && options.export_field("body_ids") && has_element_samples)
852 {
853 Eigen::MatrixXd ids = Eigen::MatrixXd::Zero(output_rows, 1);
854 for (int i = 0; i < sample.element_ids.size(); ++i)
855 {
856 const int element_id = sample.element_ids(i);
857 if (element_id >= 0)
858 ids(i) = mesh_->get_body_id(element_id);
859 }
860 fields.push_back({"body_ids", ids, io::OutputField::Association::Point});
861 }
862
863 const bool export_pressure_gradient =
864 !options.fields.empty() && options.export_field("pressure_gradient");
865 if (mesh_ && (options.export_field("pressure") || export_pressure_gradient || (!options.fields.empty() && options.export_field("auxiliary"))))
866 {
867 Eigen::MatrixXd values, gradients;
869 *mesh_, pressure_space_.basis_list(), space_.geometry_basis_list(), sample, pressure, values,
870 export_pressure_gradient ? &gradients : nullptr))
871 {
872 if (options.export_field("pressure"))
873 fields.push_back({"pressure", values, io::OutputField::Association::Point});
874 if (export_pressure_gradient)
875 fields.push_back({"pressure_gradient", gradients, io::OutputField::Association::Point});
876 if (!options.fields.empty() && options.export_field("auxiliary"))
877 fields.push_back({"auxiliary", values, io::OutputField::Association::Point});
878 }
879 }
880 return fields;
881 }
882} // namespace polyfem::varform
int x
std::array< Matrix< int, 3, 3 >, 3 > space_
static std::shared_ptr< MixedAssembler > make_mixed_assembler(const std::string &formulation)
static std::string other_assembler_name(const std::string &formulation)
static void merge_mixed_matrices(const int n_bases, const int n_pressure_bases, const int problem_dim, const bool add_average, const StiffnessMatrix &velocity_stiffness, const StiffnessMatrix &mixed_stiffness, const StiffnessMatrix &pressure_stiffness, StiffnessMatrix &stiffness)
utility to merge 3 blocks of mixed matrices, A=velocity_stiffness, B=mixed_stiffness,...
static std::shared_ptr< Assembler > make_assembler(const std::string &formulation)
Caches basis evaluation and geometric mapping at every element.
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 assembling_stiffness_mat_time
time to assembly
double assigning_rhs_time
time to computing the rhs
double assembling_mass_mat_time
time to assembly mass
double solving_time
time to solve
all stats from polyfem
int n_flipped
number of flipped elements, compute only when using count_flipped_els (false by default)
json solver_info
information of the solver, eg num iteration, time, errors, etc the informations varies depending on t...
Eigen::Vector4d spectrum
spectrum of the stiffness matrix, enable only if POLYSOLVE_WITH_SPECTRA is ON (off 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
int n_elements() const
utitlity to return the number of elements, cells or faces in 3d and 2d
Definition Mesh.hpp:163
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
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
std::string name() const override
Get the name of the variational formulation.
void save_json(const Eigen::MatrixXd &solution, std::ostream &out) const override
Save the solution to a JSON file, for output purposes.
io::OutStatsData compute_errors(const Eigen::MatrixXd &solution) override
Get the error statistics of the variational formulation, for output purposes.
void build_basis(mesh::Mesh &mesh, const bool iso_parametric, const json &args) override
std::shared_ptr< time_integrator::ImplicitTimeIntegrator > time_integrator
assembler::AssemblyValsCache mass_ass_vals_cache_
std::shared_ptr< assembler::MixedAssembler > mixed_assembler_
void solve_static_linear(Eigen::MatrixXd &sol)
void solve_problem(Eigen::MatrixXd &sol) override
std::shared_ptr< assembler::Mass > mass_assembler_
std::shared_ptr< assembler::HRZMass > pure_mass_assembler_
assembler::AssemblyValsCache pure_mass_ass_vals_cache_
void assemble_mass_mat(const mesh::Mesh &mesh, const json &args) override
void assemble_rhs(const mesh::Mesh &mesh) override
void solve_transient_linear(Eigen::MatrixXd &sol)
assembler::AssemblyValsCache ass_vals_cache_
void load_mesh(const mesh::Mesh &mesh, const json &args) 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 build_stiffness_mat(StiffnessMatrix &stiffness)
void prepare_initial_solution(Eigen::MatrixXd &sol) const
void export_data(const Eigen::MatrixXd &solution) const override
std::shared_ptr< assembler::Assembler > primary_assembler_
io::OutputSpace output_space() const override
Get the output space of the variational formulation, for output purposes.
assembler::AssemblyValsCache pressure_ass_vals_cache_
void split_solution(const Eigen::MatrixXd &stacked, Eigen::MatrixXd &primary, Eigen::MatrixXd &pressure) const
void solve_linear_system(const std::unique_ptr< polysolve::linear::Solver > &solver, StiffnessMatrix &A, Eigen::VectorXd &b, const bool compute_spectrum, Eigen::MatrixXd &sol)
std::shared_ptr< assembler::Assembler > pressure_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.
const std::vector< basis::ElementBases > & geometry_basis_list() const
Definition FESpace.hpp:115
std::shared_ptr< std::vector< basis::ElementBases > > bases
Per-element basis data.
Definition FESpace.hpp:68
std::shared_ptr< GeometryMapping > geometry
Geometric mapping used to integrate this FE space.
Definition FESpace.hpp:89
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
std::map< int, basis::InterfaceData > poly_edge_to_data
Polygonal-basis construction data, indexed by element ID.
Definition FESpace.hpp:77
const std::vector< basis::ElementBases > & basis_list() const
Definition FESpace.hpp:109
bool is_iso_parametric() const
Definition FESpace.hpp:104
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
void notify_time_step(const int t, const int time_steps, const double t0, const double dt) const
Definition VarForm.cpp:951
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
std::shared_ptr< time_integrator::BDF > make_bdf_time_integrator() const
Definition VarForm.cpp:874
void save_timestep(const double time, const int t, const double t0, const double dt, const Eigen::MatrixXd &solution) const
Definition VarForm.cpp:907
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 save_step_state(const double t0, const double dt, const int t, const time_integrator::ImplicitTimeIntegrator *time_integrator, const bool rest_mesh_written=false) const
Definition VarForm.cpp:892
QuadratureOrders n_boundary_samples(const int discr_order, const int gdiscr_order) const
Definition VarForm.cpp:259
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)
bool sample_scalar_field(const mesh::Mesh &mesh, const std::vector< basis::ElementBases > &field_bases, const std::vector< basis::ElementBases > &gbases, const io::OutputSample &sample, const Eigen::MatrixXd &dof_values, Eigen::MatrixXd &values, Eigen::MatrixXd *gradients=nullptr)
void expand_primary_matrix(const int full_size, const StiffnessMatrix &primary, StiffnessMatrix &expanded)
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
void log_and_throw_error(const std::string &msg)
Definition Logger.cpp:73
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
std::vector< mesh::LocalBoundary > local_pressure_boundary
Definition FESpace.hpp:157
std::unordered_map< int, std::vector< mesh::LocalBoundary > > local_pressure_cavity
Definition FESpace.hpp:158