56 if (formulation ==
"Helmholtz")
57 return std::make_shared<Helmholtz>();
58 else if (formulation ==
"Laplacian")
59 return std::make_shared<Laplacian>();
60 else if (formulation ==
"Electrostatics")
61 return std::make_shared<Electrostatics>();
63 else if (formulation ==
"Bilaplacian")
64 return std::make_shared<BilaplacianMain>();
65 else if (formulation ==
"BilaplacianAux")
66 return std::make_shared<BilaplacianAux>();
68 else if (formulation ==
"LinearElasticity")
69 return std::make_shared<LinearElasticity>();
70 else if (formulation ==
"HookeLinearElasticity")
71 return std::make_shared<HookeLinearElasticity>();
72 else if (formulation ==
"IncompressibleLinearElasticity")
73 return std::make_shared<IncompressibleLinearElasticityDispacement>();
74 else if (formulation ==
"IncompressibleLinearElasticityPressure")
75 return std::make_shared<IncompressibleLinearElasticityPressure>();
77 else if (formulation ==
"SaintVenant")
78 return std::make_shared<SaintVenantElasticity>();
79 else if (formulation ==
"NeoHookean")
80 return std::make_shared<NeoHookeanElasticity>();
81 else if (formulation ==
"IsochoricNeoHookean")
82 return std::make_shared<IsochoricNeoHookean>();
83 else if (formulation ==
"MooneyRivlin")
84 return std::make_shared<MooneyRivlinElasticity>();
85 else if (formulation ==
"MooneyRivlin3Param")
86 return std::make_shared<MooneyRivlin3ParamElasticity>();
87 else if (formulation ==
"MooneyRivlin3ParamSymbolic")
88 return std::make_shared<MooneyRivlin3ParamSymbolic>();
89 else if (formulation ==
"MultiModels")
90 return std::make_shared<MultiModel>();
91 else if (formulation ==
"MaterialSum")
92 return std::make_shared<SumModel>();
93 else if (formulation ==
"UnconstrainedOgden")
94 return std::make_shared<UnconstrainedOgdenElasticity>();
95 else if (formulation ==
"IncompressibleOgden")
96 return std::make_shared<IncompressibleOgdenElasticity>();
97 else if (formulation ==
"VolumePenalty")
98 return std::make_shared<VolumePenalty>();
100 else if (formulation ==
"HGOFiber")
101 return std::make_shared<HGOFiber>();
103 else if (formulation ==
"Stokes")
104 return std::make_shared<StokesVelocity>();
105 else if (formulation ==
"StokesPressure")
106 return std::make_shared<StokesPressure>();
107 else if (formulation ==
"NavierStokes")
108 return std::make_shared<NavierStokesVelocity>();
109 else if (formulation ==
"OperatorSplitting")
110 return std::make_shared<OperatorSplitting>();
112 else if (formulation ==
"AMIPS")
113 return std::make_shared<AMIPSEnergy>();
114 else if (formulation ==
"AMIPSAutodiff")
115 return std::make_shared<AMIPSEnergyAutodiff>();
116 else if (formulation ==
"FixedCorotational")
117 return std::make_shared<FixedCorotational>();
135 const int n_bases,
const int n_pressure_bases,
const int problem_dim,
const bool add_average,
139 assert(velocity_stiffness.rows() == velocity_stiffness.cols());
140 assert(velocity_stiffness.rows() == n_bases * problem_dim);
142 assert(mixed_stiffness.size() == 0 || mixed_stiffness.rows() == n_bases * problem_dim);
143 assert(mixed_stiffness.size() == 0 || mixed_stiffness.cols() == n_pressure_bases);
145 assert(pressure_stiffness.size() == 0 || pressure_stiffness.rows() == n_pressure_bases);
146 assert(pressure_stiffness.size() == 0 || pressure_stiffness.cols() == n_pressure_bases);
148 const int avg_offset = add_average ? 1 : 0;
150 std::vector<Eigen::Triplet<double>> blocks;
151 blocks.reserve(velocity_stiffness.nonZeros() + 2 * mixed_stiffness.nonZeros() + pressure_stiffness.nonZeros() + 2 * avg_offset * velocity_stiffness.rows());
153 for (
int k = 0; k < velocity_stiffness.outerSize(); ++k)
155 for (StiffnessMatrix::InnerIterator it(velocity_stiffness, k); it; ++it)
157 blocks.emplace_back(it.row(), it.col(), it.value());
161 for (
int k = 0; k < mixed_stiffness.outerSize(); ++k)
163 for (StiffnessMatrix::InnerIterator it(mixed_stiffness, k); it; ++it)
165 blocks.emplace_back(it.row(), n_bases * problem_dim + it.col(), it.value());
166 blocks.emplace_back(it.col() + n_bases * problem_dim, it.row(), it.value());
170 for (
int k = 0; k < pressure_stiffness.outerSize(); ++k)
172 for (StiffnessMatrix::InnerIterator it(pressure_stiffness, k); it; ++it)
174 blocks.emplace_back(n_bases * problem_dim + it.row(), n_bases * problem_dim + it.col(), it.value());
180 const double val = 1.0 / n_pressure_bases;
181 for (
int i = 0; i < n_pressure_bases; ++i)
183 blocks.emplace_back(n_bases * problem_dim + i, n_bases * problem_dim + n_pressure_bases,
val);
184 blocks.emplace_back(n_bases * problem_dim + n_pressure_bases, n_bases * problem_dim + i,
val);
188 stiffness.resize(n_bases * problem_dim + n_pressure_bases + avg_offset, n_bases * problem_dim + n_pressure_bases + avg_offset);
189 stiffness.setFromTriplets(blocks.begin(), blocks.end());
190 stiffness.makeCompressed();
202 if (assembler ==
"Mass")
206 return std::max(basis_degree * 2, 1);
208 return basis_degree * 2 + 1;
210 else if (assembler ==
"NavierStokes")
213 return std::max((basis_degree - 1) + basis_degree, 1);
215 return std::max(basis_degree * 2, 1);
217 return basis_degree * 2 + 1;
225 return std::max((basis_degree - 1) * 2, 1);
236 return std::max(basis_degree * 2, 1);
240 return (basis_degree - 1) * 2 + 1;
static int quadrature_order(const std::string &assembler, const int basis_degree, const BasisType &b_type, const int dim)
utility for retrieving the needed quadrature order to precisely integrate the given form on the given...
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,...