version 0.4.1
truss.hh
Go to the documentation of this file.
1// SPDX-FileCopyrightText: 2021-2025 The Ikarus Developers mueller@ibb.uni-stuttgart.de
2// SPDX-License-Identifier: LGPL-3.0-or-later
3
10#pragma once
11
12#include <optional>
13
14#include <dune/localfefunctions/eigenDuneTransformations.hh>
15
18
19namespace Ikarus {
20
21template <typename PreFE, typename FE>
22class Truss;
23
28{
29 double E; // Young's modulus
30 double A; // Cross-section area
31
32 template <typename PreFE, typename FE>
34};
35
44template <typename PreFE, typename FE>
45class Truss
46 : public ResultTypeBase<ResultTypes::cauchyAxialForce, ResultTypes::PK2AxialForce, ResultTypes::linearAxialForce>
47{
48public:
51 using FlatBasis = typename Traits::FlatBasis;
53 using LocalView = typename Traits::LocalView;
54 using Geometry = typename Traits::Geometry;
55 using GridView = typename Traits::GridView;
56 using Element = typename Traits::Element;
57 using Pre = TrussPre;
58
59 static constexpr int myDim = Traits::mydim;
60 static constexpr int worldDim = Traits::worlddim;
61 static_assert(myDim == 1, "Truss elements should have myDim == 1");
62
71 template <typename ST>
73 {
74 double L;
75 ST l;
76 ST Elin;
77 ST Egl;
78 Eigen::VectorX<ST> dEdu;
79 Eigen::MatrixX<ST> ddEddu;
80 };
81
86 explicit Truss(const Pre& pre)
87 : E{pre.E},
88 A{pre.A} {}
89
98 template <typename ST = double>
99 auto displacement(const Requirement& par,
100 const std::optional<std::reference_wrapper<const Eigen::VectorX<ST>>>& dx = std::nullopt) const {
101 const auto& d = par.globalSolution();
102 auto disp = Ikarus::FEHelper::localSolutionBlockVector<Traits>(d, underlying().localView(), dx);
103 return disp;
104 }
105
115 template <class ST = double>
117 const Requirement& par,
118 const std::optional<std::reference_wrapper<const Eigen::VectorX<ST>>>& dx = std::nullopt) const {
120 const auto& localView = underlying().localView();
121 const auto& tree = localView.tree();
122 const auto numberOfNodes = tree.child(0).finiteElement().size();
123 auto& ele = localView.element();
124 const auto X1 = Dune::toEigen(ele.geometry().corner(0));
125 const auto X2 = Dune::toEigen(ele.geometry().corner(1));
126 const auto u = Dune::viewAsEigenMatrixAsDynFixed(displacement(par, dx)).transpose().eval();
127 const auto A1 = X2 - X1;
128 const auto ud1 = u.col(1) - u.col(0);
129
130 const Eigen::Vector<ST, worldDim> x1 = X1 + u.col(0);
131 const Eigen::Vector<ST, worldDim> x2 = X2 + u.col(1);
132
133 const double Lsquared = A1.squaredNorm();
134 const ST lsquared = (x2 - x1).squaredNorm();
135
136 kin.L = sqrt(Lsquared);
137 kin.l = sqrt(lsquared);
138
139 // Linear strain
140 kin.Elin = (kin.l - kin.L) / kin.L;
141
142 // Green-Lagrange strains
143 kin.Egl = 0.5 * (lsquared - Lsquared) / Lsquared;
144
145 // First derivative of Egl w.r.t displacements
146 kin.dEdu.setZero(worldDim * numberOfNodes);
147 kin.dEdu << -A1 - ud1, A1 + ud1;
148 kin.dEdu /= Lsquared;
149
150 // Second derivative of Egl w.r.t displacements
151 kin.ddEddu.setIdentity(worldDim * numberOfNodes, worldDim * numberOfNodes);
152 kin.ddEddu.template topRightCorner<worldDim, worldDim>() = -Eigen::Matrix<ST, worldDim, worldDim>::Identity();
153 kin.ddEddu.template bottomLeftCorner<worldDim, worldDim>() = -Eigen::Matrix<ST, worldDim, worldDim>::Identity();
154 kin.ddEddu /= Lsquared;
155
156 return kin;
157 }
158
159public:
170 template <template <typename, int, int> class RT>
171 requires(supportsResultType<RT>())
172 auto calculateAtImpl(const Requirement& req, [[maybe_unused]] const Dune::FieldVector<double, Traits::mydim>& local,
173 Dune::PriorityTag<0>) const {
175 const auto [L, l, Elin, Egl, dEdu, ddEddu] = computeStrain(req);
176 if constexpr (isSameResultType<RT, ResultTypes::cauchyAxialForce>) {
177 auto N = Eigen::Vector<double, 1>{E * A * Egl * l / L}; // Axial force in deformed configuration
178 return RTWrapper{N};
179 }
180 if constexpr (isSameResultType<RT, ResultTypes::PK2AxialForce>) {
181 auto N = Eigen::Vector<double, 1>{E * A * Egl}; // Axial force in undeformed configuration
182 return RTWrapper{N};
183 }
184 if constexpr (isSameResultType<RT, ResultTypes::linearAxialForce>) {
185 auto N = Eigen::Vector<double, 1>{E * A * Elin};
186 return RTWrapper{N};
187 }
188 }
189
190private:
191 //> CRTP
192 const auto& underlying() const { return static_cast<const FE&>(*this); }
193 auto& underlying() { return static_cast<FE&>(*this); }
194
195 double E;
196 double A;
197
198protected:
199 template <typename ScalarType>
201 const Requirement& par, const MatrixAffordance& affordance, typename Traits::template MatrixType<> K,
202 const std::optional<std::reference_wrapper<const Eigen::VectorX<ScalarType>>>& dx = std::nullopt) const {
203 const auto [L, l, Elin, Egl, dEdu, ddEddu] = computeStrain(par, dx);
204 K += E * A * L * (dEdu * dEdu.transpose() + ddEddu * Egl);
205 }
206
207 template <typename ScalarType>
209 const std::optional<std::reference_wrapper<const Eigen::VectorX<ScalarType>>>& dx =
210 std::nullopt) const -> ScalarType {
211 const auto [L, l, Elin, Egl, dEdu, ddEddu] = computeStrain(par, dx);
212 return 0.5 * E * A * L * Egl * Egl;
213 }
214
215 template <typename ScalarType>
217 const Requirement& par, VectorAffordance affordance, typename Traits::template VectorType<ScalarType> force,
218 const std::optional<std::reference_wrapper<const Eigen::VectorX<ScalarType>>>& dx = std::nullopt) const {
219 const auto [L, l, Elin, Egl, dEdu, ddEddu] = computeStrain(par, dx);
220 force += E * A * Egl * L * dEdu;
221 }
222};
223
230inline auto truss(const double E, const double A) {
231 TrussPre pre(E, A);
232
233 return pre;
234}
235} // namespace Ikarus
Definition of the LinearElastic class for finite element mechanics computations.
auto viewAsEigenMatrixAsDynFixed(Dune::BlockVector< ValueType > &blockedVector)
View Dune::BlockVector as an Eigen::Matrix with dynamic rows and fixed columns depending on the size ...
Definition: linearalgebrahelper.hh:88
Definition: assemblermanipulatorbuildingblocks.hh:22
auto truss(const double E, const double A)
A helper function to create a truss pre finite element.
Definition: truss.hh:230
MatrixAffordance
A strongly typed enum class representing the matrix affordance.
Definition: ferequirements.hh:64
VectorAffordance
A strongly typed enum class representing the vector affordance.
Definition: ferequirements.hh:49
ScalarAffordance
A strongly typed enum class representing the scalar affordance.
Definition: ferequirements.hh:38
FE class is a base class for all finite elements.
Definition: febase.hh:79
FETraits< BH, useEigenRef, useFlat > Traits
Definition: febase.hh:38
Class representing the requirements for finite element calculations.
Definition: ferequirements.hh:223
const SolutionVectorType & globalSolution() const
Get the global solution vector.
Definition: ferequirements.hh:313
Container that is used for FE Results. It gives access to the stored value, but can also be used to a...
Definition: feresulttypes.hh:164
Base class for element definitions that provides common functionality for ResultTypes.
Definition: feresulttypes.hh:277
Traits for handling finite elements.
Definition: fetraits.hh:25
typename Basis::LocalView LocalView
Type of the local view.
Definition: fetraits.hh:42
typename Element::Geometry Geometry
Type of the element geometry.
Definition: fetraits.hh:51
BH BasisHandler
Type of the basis of the finite element.
Definition: fetraits.hh:27
typename Basis::GridView GridView
Type of the grid view.
Definition: fetraits.hh:45
static constexpr int worlddim
Dimension of the world space.
Definition: fetraits.hh:60
typename BasisHandler::FlatBasis FlatBasis
Type of the flat basis.
Definition: fetraits.hh:33
typename LocalView::Element Element
Type of the grid element.
Definition: fetraits.hh:48
static constexpr int mydim
Dimension of the geometry.
Definition: fetraits.hh:63
Truss class represents a truss finite element.
Definition: truss.hh:47
void calculateVectorImpl(const Requirement &par, VectorAffordance affordance, typename Traits::template VectorType< ScalarType > force, const std::optional< std::reference_wrapper< const Eigen::VectorX< ScalarType > > > &dx=std::nullopt) const
Definition: truss.hh:216
auto calculateAtImpl(const Requirement &req, const Dune::FieldVector< double, Traits::mydim > &local, Dune::PriorityTag< 0 >) const
Calculates a requested result at a specific local position.
Definition: truss.hh:172
typename Traits::LocalView LocalView
Definition: truss.hh:53
Truss(const Pre &pre)
Constructor for the Truss class.
Definition: truss.hh:86
static constexpr int worldDim
Definition: truss.hh:60
void calculateMatrixImpl(const Requirement &par, const MatrixAffordance &affordance, typename Traits::template MatrixType<> K, const std::optional< std::reference_wrapper< const Eigen::VectorX< ScalarType > > > &dx=std::nullopt) const
Definition: truss.hh:200
typename Traits::Geometry Geometry
Definition: truss.hh:54
typename Traits::FlatBasis FlatBasis
Definition: truss.hh:51
typename Traits::GridView GridView
Definition: truss.hh:55
typename Traits::BasisHandler BasisHandler
Definition: truss.hh:50
static constexpr int myDim
Definition: truss.hh:59
auto calculateScalarImpl(const Requirement &par, ScalarAffordance affordance, const std::optional< std::reference_wrapper< const Eigen::VectorX< ScalarType > > > &dx=std::nullopt) const -> ScalarType
Definition: truss.hh:208
typename Traits::Element Element
Definition: truss.hh:56
KinematicVariables< ST > computeStrain(const Requirement &par, const std::optional< std::reference_wrapper< const Eigen::VectorX< ST > > > &dx=std::nullopt) const
Gets the strain for the given Requirement and optional displacement vector.
Definition: truss.hh:116
auto displacement(const Requirement &par, const std::optional< std::reference_wrapper< const Eigen::VectorX< ST > > > &dx=std::nullopt) const
Gets the displacement for the given Requirement and optional displacement vector.
Definition: truss.hh:99
A PreFE struct for truss elements.
Definition: truss.hh:28
double E
Definition: truss.hh:29
double A
Definition: truss.hh:30
A structure representing kinematic variables.
Definition: truss.hh:73
double L
Length of the reference geometry.
Definition: truss.hh:74
Eigen::MatrixX< ST > ddEddu
second derivative of Egl w.r.t displacements
Definition: truss.hh:79
ST l
Length of the deformed geometry.
Definition: truss.hh:75
ST Egl
Green-Lagrange strain.
Definition: truss.hh:77
ST Elin
Linear strain.
Definition: truss.hh:76
Eigen::VectorX< ST > dEdu
first derivative of Egl w.r.t displacements
Definition: truss.hh:78
Definition: utils/dirichletvalues.hh:32