19 template <
typename Derived>
23 Eigen::Matrix<double, dim * dim, dim> B(size() * size(), size());
26 for (
int i = 0; i < dim; ++i)
27 for (
int j = 0; j < dim; ++j)
28 B(i * dim + j, i) = g(j);
33 template <
typename Derived>
38 template <
typename Derived>
44 const std::function<Eigen::MatrixXd(
const Eigen::MatrixXd &)> &fun)
const
46 Eigen::MatrixXd deformation_grad(size(), size());
47 Eigen::MatrixXd stress_tensor(size(), size());
51 const auto &displacement = data.
fun;
53 const auto &bs = data.
bs;
54 const auto &gbs = data.
gbs;
55 const auto el_id = data.
el_id;
57 assert(displacement.cols() == 1);
59 all.resize(local_pts.rows(), all_size);
62 Eigen::Matrix<Diff, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> def_grad(size(), size());
65 vals.
compute(el_id, size() == 3, local_pts, bs, gbs);
67 for (
long p = 0; p < local_pts.rows(); ++p)
72 for (
int d = 0; d < size(); ++d)
73 deformation_grad(d, d) += 1;
77 all.row(p) = fun(deformation_grad);
81 for (
int d1 = 0; d1 < size(); ++d1)
83 for (
int d2 = 0; d2 < size(); ++d2)
84 def_grad(d1, d2) =
Diff(d1 * size() + d2, deformation_grad(d1, d2));
87 const auto val = derived().elastic_energy(local_pts.row(p), data.
t,
vals.
element_id, def_grad);
89 for (
int d1 = 0; d1 < size(); ++d1)
91 for (
int d2 = 0; d2 < size(); ++d2)
92 stress_tensor(d1, d2) =
val.getGradient()(d1 * size() + d2);
95 stress_tensor = 1.0 / deformation_grad.determinant() * stress_tensor * deformation_grad.transpose();
102 all.row(p) = fun(stress_tensor);
106 template <
typename Derived>
109 return compute_energy_aux<double>(data);
112 template <
typename Derived>
116 return assemble_gradient_full_ad(data);
120 auto grad = assemble_gradient_stress_ad(data);
121 auto grad_full = assemble_gradient_full_ad(data);
122 assert((std::isnan(grad.norm()) && std::isnan(grad_full.norm())) || (grad - grad_full).norm() < 1e-6);
124 return assemble_gradient_stress_ad(data);
129 auto grad = assemble_gradient_stress_noad(data);
130 auto grad_full = assemble_gradient_stress_ad(data);
131 assert((std::isnan(grad.norm()) && std::isnan(grad_full.norm())) || (grad - grad_full).norm() < 1e-6);
133 return assemble_gradient_stress_noad(data);
137 template <
typename Derived>
142 size(), n_bases, data,
143 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 6, 1>>>(data); },
144 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 8, 1>>>(data); },
145 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 12, 1>>>(data); },
146 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 18, 1>>>(data); },
147 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 24, 1>>>(data); },
148 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 30, 1>>>(data); },
149 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 60, 1>>>(data); },
150 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, 81, 1>>>(data); },
151 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, SMALL_N, 1>>>(data); },
152 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, BIG_N, 1>>>(data); },
153 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar1<double, Eigen::VectorXd>>(data); });
156 template <
typename Derived>
159 Eigen::Matrix<double, Eigen::Dynamic, 1> gradient;
168 compute_gradient_from_stress<3, 2>(data, gradient);
174 compute_gradient_from_stress<6, 2>(data, gradient);
180 compute_gradient_from_stress<10, 2>(data, gradient);
186 compute_gradient_from_stress<Eigen::Dynamic, 2>(data, gradient);
199 compute_gradient_from_stress<4, 3>(data, gradient);
205 compute_gradient_from_stress<10, 3>(data, gradient);
211 compute_gradient_from_stress<20, 3>(data, gradient);
217 compute_gradient_from_stress<Eigen::Dynamic, 3>(data, gradient);
226 template <
typename Derived>
229 Eigen::Matrix<double, Eigen::Dynamic, 1> gradient;
238 compute_gradient_from_stress_noad<3, 2>(data, gradient);
244 compute_gradient_from_stress_noad<6, 2>(data, gradient);
250 compute_gradient_from_stress_noad<10, 2>(data, gradient);
256 compute_gradient_from_stress_noad<Eigen::Dynamic, 2>(data, gradient);
269 compute_gradient_from_stress_noad<4, 3>(data, gradient);
275 compute_gradient_from_stress_noad<10, 3>(data, gradient);
281 compute_gradient_from_stress_noad<20, 3>(data, gradient);
287 compute_gradient_from_stress_noad<Eigen::Dynamic, 3>(data, gradient);
296 template <
typename Derived>
300 return assemble_hessian_full_ad(data);
304 auto hessian = assemble_hessian_stress_ad(data);
305 auto hessian_full = assemble_hessian_full_ad(data);
306 assert((std::isnan(hessian.norm()) && std::isnan(hessian_full.norm())) || (hessian - hessian_full).norm() < 1e-5);
308 return assemble_hessian_stress_ad(data);
313 auto hessian = assemble_hessian_stress_noad(data);
314 auto hessian_full = assemble_hessian_stress_ad(data);
315 assert((std::isnan(hessian.norm()) && std::isnan(hessian_full.norm())) || (hessian - hessian_full).norm() < 1e-5);
317 return assemble_hessian_stress_noad(data);
321 template <
typename Derived>
326 size(), n_bases, data,
327 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 6, 1>, Eigen::Matrix<double, 6, 6>>>(data); },
328 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 8, 1>, Eigen::Matrix<double, 8, 8>>>(data); },
329 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 12, 1>, Eigen::Matrix<double, 12, 12>>>(data); },
330 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 18, 1>, Eigen::Matrix<double, 18, 18>>>(data); },
331 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 24, 1>, Eigen::Matrix<double, 24, 24>>>(data); },
332 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 30, 1>, Eigen::Matrix<double, 30, 30>>>(data); },
333 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 60, 1>, Eigen::Matrix<double, 60, 60>>>(data); },
334 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, 81, 1>, Eigen::Matrix<double, 81, 81>>>(data); },
335 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, SMALL_N, 1>, Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic, 0, SMALL_N, SMALL_N>>>(data); },
336 [&](
const NonLinearAssemblerData &data) {
return compute_energy_aux<DScalar2<double, Eigen::VectorXd, Eigen::MatrixXd>>(data); });
339 template <
typename Derived>
342 Eigen::MatrixXd hessian;
350 hessian.resize(6, 6);
352 compute_hessian_from_stress<3, 2>(data, hessian);
357 hessian.resize(12, 12);
359 compute_hessian_from_stress<6, 2>(data, hessian);
364 hessian.resize(20, 20);
366 compute_hessian_from_stress<10, 2>(data, hessian);
373 compute_hessian_from_stress<Eigen::Dynamic, 2>(data, hessian);
385 hessian.resize(12, 12);
387 compute_hessian_from_stress<4, 3>(data, hessian);
392 hessian.resize(30, 30);
394 compute_hessian_from_stress<10, 3>(data, hessian);
399 hessian.resize(60, 60);
401 compute_hessian_from_stress<20, 3>(data, hessian);
408 compute_hessian_from_stress<Eigen::Dynamic, 3>(data, hessian);
417 template <
typename Derived>
420 Eigen::MatrixXd hessian;
428 hessian.resize(6, 6);
430 compute_hessian_from_stress_noad<3, 2>(data, hessian);
435 hessian.resize(12, 12);
437 compute_hessian_from_stress_noad<6, 2>(data, hessian);
442 hessian.resize(20, 20);
444 compute_hessian_from_stress_noad<10, 2>(data, hessian);
451 compute_hessian_from_stress_noad<Eigen::Dynamic, 2>(data, hessian);
463 hessian.resize(12, 12);
465 compute_hessian_from_stress_noad<4, 3>(data, hessian);
470 hessian.resize(30, 30);
472 compute_hessian_from_stress_noad<10, 3>(data, hessian);
477 hessian.resize(60, 60);
479 compute_hessian_from_stress_noad<20, 3>(data, hessian);
486 compute_hessian_from_stress_noad<Eigen::Dynamic, 3>(data, hessian);
495 template <
typename Derived>
498 const Eigen::MatrixXd &mat,
499 Eigen::MatrixXd &stress,
500 Eigen::MatrixXd &result)
const
502 typedef DScalar2<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, 9, 1>, Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic, 0, 9, 9>>
Diff;
504 const double t = data.
t;
505 const int el_id = data.
el_id;
506 const Eigen::MatrixXd &local_pts = data.
local_pts;
507 const Eigen::MatrixXd &global_pts = data.
global_pts;
508 const Eigen::MatrixXd &grad_u_i = data.
grad_u_i;
511 Eigen::Matrix<Diff, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> def_grad(size(), size());
513 Eigen::MatrixXd
F = grad_u_i;
514 for (
int d = 0; d < size(); ++d)
517 assert(local_pts.rows() == 1);
518 for (
int i = 0; i < size(); ++i)
519 for (
int j = 0; j < size(); ++j)
520 def_grad(i, j) =
Diff(i + j * size(),
F(i, j));
522 auto energy = derived().elastic_energy(global_pts, t, el_id, def_grad);
525 Eigen::MatrixXd grad = energy.getGradient().reshaped(size(), size());
527 Eigen::MatrixXd hess = energy.getHessian();
532 result = (hess * mat.reshaped(size() * size(), 1)).reshaped(size(), size());
535 template <
typename Derived>
538 Eigen::MatrixXd &stress,
539 Eigen::MatrixXd &result)
const
541 typedef DScalar2<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, 9, 1>, Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic, 0, 9, 9>>
Diff;
543 const double t = data.
t;
544 const int el_id = data.
el_id;
545 const Eigen::MatrixXd &local_pts = data.
local_pts;
546 const Eigen::MatrixXd &global_pts = data.
global_pts;
547 const Eigen::MatrixXd &grad_u_i = data.
grad_u_i;
550 Eigen::Matrix<Diff, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> def_grad(size(), size());
552 Eigen::MatrixXd
F = grad_u_i;
553 for (
int d = 0; d < size(); ++d)
556 assert(local_pts.rows() == 1);
557 for (
int i = 0; i < size(); ++i)
558 for (
int j = 0; j < size(); ++j)
559 def_grad(i, j) =
Diff(i + j * size(),
F(i, j));
561 auto energy = derived().elastic_energy(global_pts, t, el_id, def_grad);
564 Eigen::MatrixXd grad = energy.getGradient().reshaped(size(), size());
566 Eigen::MatrixXd hess = energy.getHessian();
571 result = (hess * stress.reshaped(size() * size(), 1)).reshaped(size(), size());
574 template <
typename Derived>
577 const Eigen::MatrixXd &vect,
578 Eigen::MatrixXd &stress,
579 Eigen::MatrixXd &result)
const
581 typedef DScalar2<double, Eigen::Matrix<double, Eigen::Dynamic, 1, 0, 9, 1>, Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic, 0, 9, 9>>
Diff;
583 const double t = data.
t;
584 const int el_id = data.
el_id;
585 const Eigen::MatrixXd &local_pts = data.
local_pts;
586 const Eigen::MatrixXd &global_pts = data.
global_pts;
587 const Eigen::MatrixXd &grad_u_i = data.
grad_u_i;
590 Eigen::Matrix<Diff, Eigen::Dynamic, Eigen::Dynamic, 0, 3, 3> def_grad(size(), size());
592 Eigen::MatrixXd
F = grad_u_i;
593 for (
int d = 0; d < size(); ++d)
596 assert(local_pts.rows() == 1);
597 for (
int i = 0; i < size(); ++i)
598 for (
int j = 0; j < size(); ++j)
599 def_grad(i, j) =
Diff(i + j * size(),
F(i, j));
601 auto energy = derived().elastic_energy(global_pts, t, el_id, def_grad);
604 Eigen::MatrixXd grad = energy.getGradient().reshaped(size(), size());
606 Eigen::MatrixXd hess = energy.getHessian();
610 result.resize(hess.rows(), vect.size());
611 for (
int i = 0; i < hess.rows(); ++i)
612 if (vect.rows() == 1)
614 result.row(i) = vect * hess.row(i).reshaped(size(), size());
617 result.row(i) = hess.row(i).reshaped(size(), size()) * vect;
ElementAssemblyValues vals
stores per element basis values at given quadrature points and geometric mapping
std::vector< AssemblyValues > basis_values
void compute(const int el_index, const bool is_volume, const Eigen::MatrixXd &pts, const basis::ElementBases &basis, const basis::ElementBases &gbasis)
computes the per element values at the local (ref el) points (pts) sets basis_values,...
void compute_stress_grad_multiply_stress(const OptAssemblerData &data, Eigen::MatrixXd &stress, Eigen::MatrixXd &result) const override
void assign_stress_tensor(const OutputData &data, const int all_size, const ElasticityTensorType &type, Eigen::MatrixXd &all, const std::function< Eigen::MatrixXd(const Eigen::MatrixXd &)> &fun) const override
Eigen::Matrix< double, dim *dim, dim > compute_B_block(const Eigen::Matrix< double, 1, dim > &g) const
Eigen::VectorXd assemble_gradient_full_ad(const NonLinearAssemblerData &data) const
Eigen::MatrixXd assemble_hessian_full_ad(const NonLinearAssemblerData &data) const
Eigen::MatrixXd assemble_hessian_stress_noad(const NonLinearAssemblerData &data) const
Eigen::VectorXd assemble_gradient_stress_noad(const NonLinearAssemblerData &data) const
Eigen::MatrixXd assemble_hessian_stress_ad(const NonLinearAssemblerData &data) const
Eigen::MatrixXd assemble_hessian(const NonLinearAssemblerData &data) const override
void compute_stress_grad_multiply_vect(const OptAssemblerData &data, const Eigen::MatrixXd &vect, Eigen::MatrixXd &stress, Eigen::MatrixXd &result) const override
Eigen::VectorXd assemble_gradient(const NonLinearAssemblerData &data) const override
Eigen::VectorXd assemble_gradient_stress_ad(const NonLinearAssemblerData &data) const
double compute_energy(const NonLinearAssemblerData &data) const override
void compute_stress_grad_multiply_mat(const OptAssemblerData &data, const Eigen::MatrixXd &mat, Eigen::MatrixXd &stress, Eigen::MatrixXd &result) const override
const ElementAssemblyValues & vals
const Eigen::MatrixXd & local_pts
const Eigen::MatrixXd & global_pts
const Eigen::MatrixXd & grad_u_i
const basis::ElementBases & bs
const Eigen::MatrixXd & fun
const Eigen::MatrixXd & local_pts
const basis::ElementBases & gbs
DScalar1< double, Eigen::Matrix< double, Eigen::Dynamic, 1 > > Diff
Eigen::MatrixXd pk2_from_cauchy(const Eigen::MatrixXd &stress, const Eigen::MatrixXd &F)
Eigen::MatrixXd hessian_from_energy(const int size, const int n_bases, const assembler::NonLinearAssemblerData &data, const std::function< DScalar2< double, Eigen::Matrix< double, 6, 1 >, Eigen::Matrix< double, 6, 6 > >(const assembler::NonLinearAssemblerData &)> &fun6, const std::function< DScalar2< double, Eigen::Matrix< double, 8, 1 >, Eigen::Matrix< double, 8, 8 > >(const assembler::NonLinearAssemblerData &)> &fun8, const std::function< DScalar2< double, Eigen::Matrix< double, 12, 1 >, Eigen::Matrix< double, 12, 12 > >(const assembler::NonLinearAssemblerData &)> &fun12, const std::function< DScalar2< double, Eigen::Matrix< double, 18, 1 >, Eigen::Matrix< double, 18, 18 > >(const assembler::NonLinearAssemblerData &)> &fun18, const std::function< DScalar2< double, Eigen::Matrix< double, 24, 1 >, Eigen::Matrix< double, 24, 24 > >(const assembler::NonLinearAssemblerData &)> &fun24, const std::function< DScalar2< double, Eigen::Matrix< double, 30, 1 >, Eigen::Matrix< double, 30, 30 > >(const assembler::NonLinearAssemblerData &)> &fun30, const std::function< DScalar2< double, Eigen::Matrix< double, 60, 1 >, Eigen::Matrix< double, 60, 60 > >(const assembler::NonLinearAssemblerData &)> &fun60, const std::function< DScalar2< double, Eigen::Matrix< double, 81, 1 >, Eigen::Matrix< double, 81, 81 > >(const assembler::NonLinearAssemblerData &)> &fun81, const std::function< DScalar2< double, Eigen::Matrix< double, Eigen::Dynamic, 1, 0, SMALL_N, 1 >, Eigen::Matrix< double, Eigen::Dynamic, Eigen::Dynamic, 0, SMALL_N, SMALL_N > >(const assembler::NonLinearAssemblerData &)> &funN, const std::function< DScalar2< double, Eigen::VectorXd, Eigen::MatrixXd >(const assembler::NonLinearAssemblerData &)> &funn)
void compute_diplacement_grad(const int size, const ElementAssemblyValues &vals, const Eigen::MatrixXd &local_pts, const int p, const Eigen::MatrixXd &displacement, Eigen::MatrixXd &displacement_grad)
Eigen::MatrixXd pk1_from_cauchy(const Eigen::MatrixXd &stress, const Eigen::MatrixXd &F)
Eigen::VectorXd gradient_from_energy(const int size, const int n_bases, const assembler::NonLinearAssemblerData &data, const std::function< DScalar1< double, Eigen::Matrix< double, 6, 1 > >(const assembler::NonLinearAssemblerData &)> &fun6, const std::function< DScalar1< double, Eigen::Matrix< double, 8, 1 > >(const assembler::NonLinearAssemblerData &)> &fun8, const std::function< DScalar1< double, Eigen::Matrix< double, 12, 1 > >(const assembler::NonLinearAssemblerData &)> &fun12, const std::function< DScalar1< double, Eigen::Matrix< double, 18, 1 > >(const assembler::NonLinearAssemblerData &)> &fun18, const std::function< DScalar1< double, Eigen::Matrix< double, 24, 1 > >(const assembler::NonLinearAssemblerData &)> &fun24, const std::function< DScalar1< double, Eigen::Matrix< double, 30, 1 > >(const assembler::NonLinearAssemblerData &)> &fun30, const std::function< DScalar1< double, Eigen::Matrix< double, 60, 1 > >(const assembler::NonLinearAssemblerData &)> &fun60, const std::function< DScalar1< double, Eigen::Matrix< double, 81, 1 > >(const assembler::NonLinearAssemblerData &)> &fun81, const std::function< DScalar1< double, Eigen::Matrix< double, Eigen::Dynamic, 1, 0, SMALL_N, 1 > >(const assembler::NonLinearAssemblerData &)> &funN, const std::function< DScalar1< double, Eigen::Matrix< double, Eigen::Dynamic, 1, 0, BIG_N, 1 > >(const assembler::NonLinearAssemblerData &)> &funBigN, const std::function< DScalar1< double, Eigen::VectorXd >(const assembler::NonLinearAssemblerData &)> &funn)
Automatic differentiation scalar with first-order derivatives.
Automatic differentiation scalar with first- and second-order derivatives.
static void setVariableCount(size_t value)
Set the independent variable count used by the automatic differentiation layer.