Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Linear elasticity example #799

Merged
merged 39 commits into from
Aug 19, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
3666310
start linear elasticity tutorial
kimauth Sep 14, 2023
5c855f1
start linear elasticity tutorial
kimauth Sep 23, 2023
7773404
runnable linear elasticity example
kimauth Sep 27, 2023
8cee8b7
suppress output after code blocks
kimauth Sep 27, 2023
9e148c9
load mesh from assets
kimauth Sep 27, 2023
de13a38
revert unindented changes on heat equation tutorial
kimauth Sep 27, 2023
0244357
an actual introduction for linear elasticity
kimauth Sep 27, 2023
ce6fb00
forgotten semicolon
kimauth Sep 27, 2023
d1a7aad
Update docs/src/literate-tutorials/linear_elasticity.jl
kimauth Sep 28, 2023
82ad164
remove internal force vector computation
kimauth Nov 1, 2023
5bde8aa
suppress gmsh output
kimauth Nov 1, 2023
97e3d6d
add comment on addfaceset
kimauth Nov 2, 2023
ea3cffe
add neumann bc
kimauth Nov 2, 2023
38ca056
changes I don't remember
kimauth Apr 9, 2024
9dfb691
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 4, 2024
e1bde02
Elasticity tutorial update and fixups
KnutAM Aug 4, 2024
3b8c26e
Further improvements, update figure
KnutAM Aug 4, 2024
e56d33d
Fix test on linux
KnutAM Aug 4, 2024
a906ec3
Minor formatting
KnutAM Aug 4, 2024
8b4ffd1
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 4, 2024
c6783e4
Disable test that fails seemingly unrelated
KnutAM Aug 4, 2024
26ceb4a
Fix some traction-related formatting
KnutAM Aug 4, 2024
bc41b54
Apply solution by @termi-official
KnutAM Aug 5, 2024
263ac8d
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 5, 2024
1038b51
Use VTKGridFile
KnutAM Aug 5, 2024
9cd5d64
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 8, 2024
d123875
Hopefully complete example
KnutAM Aug 8, 2024
95517a7
Fix links and test hiding
KnutAM Aug 8, 2024
30fbaf8
Improve figure and fix spelling/grammar/formatting
KnutAM Aug 10, 2024
6dd8418
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 10, 2024
c5820ab
Merge branch 'master' into ka/elasticity_example
KnutAM Aug 10, 2024
6103a81
Apply suggestions from reviews
KnutAM Aug 18, 2024
c8936ae
Download stress results to elasticity tutorial
KnutAM Aug 18, 2024
e9913bd
Supress outputs
KnutAM Aug 18, 2024
fd58706
Remove performance tips completely
KnutAM Aug 19, 2024
dc3067b
Apply suggestions from code review
KnutAM Aug 19, 2024
d223659
Adress review comments
KnutAM Aug 19, 2024
551dca3
4th order tensor to mathsf
KnutAM Aug 19, 2024
f5796ef
Remove unused figure
KnutAM Aug 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 1 addition & 3 deletions docs/src/literate-tutorials/heat_equation.jl
Original file line number Diff line number Diff line change
Expand Up @@ -173,9 +173,7 @@ end
#
# !!! note "Notation"
# Comparing again with [Introduction to FEM](@ref), `f` and `u` correspond to
# $\underline{\hat{f}}$ and $\underline{\hat{u}}$, since they represent the discretized
# versions. However, through the code we use `f` and `u` instead to reflect the strong
# connection between the weak form and the Ferrite implementation.
# $\underline{\hat{f}}$ and $\underline{\hat{u}}$, since they represent the discretized versions. However, through the code we use `f` and `u` instead to reflect the strong connection between the weak form and the Ferrite implementation.
kimauth marked this conversation as resolved.
Show resolved Hide resolved

function assemble_global(cellvalues::CellValues, K::SparseMatrixCSC, dh::DofHandler)
## Allocate the element stiffness matrix and element force vector
Expand Down
244 changes: 242 additions & 2 deletions docs/src/literate-tutorials/linear_elasticity.jl
Original file line number Diff line number Diff line change
@@ -1,3 +1,243 @@
# # Linear elasticity
# # [Linear elasticity](@id tutorial-linear-elasticity)
#
# TBW
# ![](linear_elasticity.png)
#
# *Figure 1*: Linear elastically deformed Ferrite logo.
#
#md # !!! tip
#md # This example is also available as a Jupyter notebook:
#md # [`heat_equation.ipynb`](@__NBVIEWER_ROOT_URL__/examples/heat_equation.ipynb).
kimauth marked this conversation as resolved.
Show resolved Hide resolved
#-
#
# ## Introduction
#
# The heat equation is the "Hello, world!" equation of finite elements.
# Here we solve the equation on a unit square, with a uniform internal source.
# The strong form of the (linear) heat equation is given by
#
# ```math
# -\boldsymbol{\sigma} \cdot \boldsymbol{\nabla} = \boldsymbol{b} \quad \textbf{x} \in \Omega,
termi-official marked this conversation as resolved.
Show resolved Hide resolved
# ```
#
# where $\boldsymbol{\sigma}$ is the stress tensor, $\boldsymbol{b}$ is the body force and
# $\Omega$ the domain.
#
# In this example, we use linear elasticity, such that
# ```math
# \boldsymbol{\sigma} = \boldsymbol{E} : \boldsymbol \varepsilon
# ```
# where $\boldsymbol{E}$ is the elastic stiffness tensor and $\boldsymbol{\varepsilon}$ is
# the small strain tensor that is computed from the displacement field $\boldsymbol{u}$ as
# ```math
# \boldsymbol{\varepsilon} = \frac{1}{2} \left(
# \boldsymbol{\nabla} \otimes \boldsymbol{u}
# +
# \boldsymbol{u} \otimes \boldsymbol{\nabla}
# \right)
# ```
#
# The resulting weak form is given given as follows: Find ``\boldsymbol{u} \in \mathbb{U}`` such that
# ```math
# \int_\Omega
# \boldsymbol{\sigma} : \left(\delta \boldsymbol{u} \otimes \boldsymbol{\nabla} \right)
KnutAM marked this conversation as resolved.
Show resolved Hide resolved
# \, \mathrm{d}V
# =
# \int_{\partial\Omega}
# \boldsymbol{t}^\ast \cdot \delta \boldsymbol{u}
# \, \mathrm{d}A
# +
# \int_\Omega
# \boldsymbol{b} \cdot \delta \boldsymbol{u}
# \, \mathrm{d}V
# \quad \forall \, \delta \boldsymbol{u} \in \mathbb{T},
# ```
# where $\delta \boldsymbol{u}$ is a vector valued test function, and where $\mathbb{U}$ and
# $\mathbb{T}$ are suitable trial and test function sets, respectively. The boundary traction
# is denoted $\boldsymbol{t}^\ast$ and body forces are denoted $\boldsymbol{b}$.
#
# However, for this example we will neglect all external loads and thus the weak form reads:
# ```math
# \int_\Omega
# \boldsymbol{\sigma} : \left(\delta \boldsymbol{u} \otimes \boldsymbol{\nabla} \right)
# \, \mathrm{d}V
# =
# 0 \,.
# ```

# First we load Ferrite, and some other packages we need.
using Ferrite, FerriteGmsh, SparseArrays
# Like for the Heat Equation example, we will use a unit square - but here we'll load the grid of the Ferrite logo! This is done by `FerriteGmsh` here.
grid = togrid("src/literate-tutorials/logo.geo") # TODO: where to store correctly and how to refer to this?
# By default the grid lacks the facesets for the boundaries, so we add them by Ferrite here.
# Note that approximate comparison to 0.0 doesn't work well, so we use a tolerance instead.
addfaceset!(grid, "top", x->x[2] ≈ 1.0)
addfaceset!(grid, "left", x->x[1] < 1e-6)
addfaceset!(grid, "bottom", x->x[2] < 1e-6);

# ### Trial and test functions
# We use linear Lagrange functions as test and trial functions. The grid is composed of triangular elements, thus we need the Lagrange functions defined on `RefTriangle`. All currently available interpolations can be found under [`Interpolation`](@ref).
#
# Since the displacement field $\boldsymbol{u}$ is vector valued, we use vector valued shape functions $\boldsymbol{N}_i$ to approximate the test and trial functions:
# ```math
# \boldsymbol{u} \approx \sum_{i=1}^N \boldsymbol{N}_i \left(\boldsymbol{x}\right) \, \hat{u}_i
# \qquad
# \delta \boldsymbol{u} \approx \sum_{i=1}^N \boldsymbol{N}_i \left(\boldsymbol{x}\right) \, \delta \hat{u}_i
# ```
# Here $N$ is the number of nodal variables and $\hat{u}_i$ / $\delta\hat{u}_i$ represent the i-th nodal value.
# Using the Einstein summation convention, we can write this in short form as
# $\boldsymbol{u} \approx \boldsymbol{N}_i \, \hat{u}_i$ and $\delta\boldsymbol{u} \approx \boldsymbol{N}_i \, \delta\hat{u}_i$.
#
# Here we use linear triangular elements (also called constant strain triangles) with a single
# quadrature point.
# The vector valued shape functions are constructed by raising the interpolation
# to the power `dim` (the dimension) since the displacement field has one component in each
# spatial dimension.
dim = 2
order = 1 # linear interpolation
ip = Lagrange{RefTriangle, order}()^dim # vector valued interpolation
qr = QuadratureRule{RefTriangle}(1) # 1 quadrature point
cellvalues = CellValues(qr, ip);

# ### Degrees of freedom
# For distributing degrees of freedom, we define a `DofHandler`. The `DofHandler` knows that `u` has two degrees of freedom per node because we vectorized the interpolation above.
dh = DofHandler(grid)
add!(dh, :u, ip)
close!(dh);

# ### Boundary conditions
# Now, we add boundary conditions. We simply support the bottom and the left side and prescribe a displacement upwards on the top edge.
# The last argument to `Dirichlet` determines which components of the field should be constrained.
ch = ConstraintHandler(dh)
add!(ch, Dirichlet(:u, getfaceset(grid, "bottom"), (x, t) -> 0.0, 2))
add!(ch, Dirichlet(:u, getfaceset(grid, "left"), (x, t) -> 0.0, 1))
add!(ch, Dirichlet(:u, getfaceset(grid, "top"), (x, t) -> 0.1, 2))
close!(ch);

# ### Material routine
# For the sake of structuring the program, we introduce a material routine here.
# We define a data structure that holds the material constants, here it stores the
# shear modulus $G$ and the bulk modulus $K$. The material routine then computes the stress
# $\boldsymbol{\sigma}$ and its tangent $\frac{\partial\boldsymbol{\sigma}}{\partial\boldsymbol{\varepsilon}}$
# based on the input strain $\boldsymbol{\varepsilon}$. Here, we use automatic differentiation
# for computing the stress tangent. You can read more about automatic differentiation on tensor
# operations in the [`Tensors.jl` docs](https://ferrite-fem.github.io/Tensors.jl/stable/man/automatic_differentiation/).
struct Elasticity
G::Float64
K::Float64
end

function material_routine(material::Elasticity, ε::SymmetricTensor{2})
(; G, K) = material
stress(ε) = 2G * dev(ε) + K * tr(ε) * one(ε)
∂σ∂ε, σ = gradient(stress, ε, :all)
return σ, ∂σ∂ε
end

E = 200e3 # Young's modulus [MPa]
ν = 0.3 # Poisson's ratio [-]
material = Elasticity(E/2(1+ν), E/3(1-2ν));

# ### Element routine
# The residual vector and stiffness matrix follow from the weak form such that
# ```math
# \left(\underline{R}\right)_i
# =
# \int_\Omega
# \boldsymbol{\sigma} : \boldsymbol{\nabla} \boldsymbol{N}_i
# \, \mathrm{d}V
# \,, \qquad
# \left(\underline{\underline{K}}\right)_{ij}
# =
# \int_\Omega
# \left(
# \frac{\partial \boldsymbol{\sigma}}{\partial \boldsymbol{\varepsilon}}
# :
# \boldsymbol{\nabla}^\mathrm{sym} \boldsymbol{N}_j
# \right)
# :
# \boldsymbol{\nabla} \boldsymbol{N}_i
# \, \mathrm{d}V
# ```
# The element routine computes the local residual vector `re` and stiffness matrix `ke`
# for a single element. `ke` and `re` are pre-allocated and reused for all elements.
function assemble_cell!(ke, fe, cellvalues, material, ue)
fill!(ke, 0.0)
fill!(fe, 0.0)

n_basefuncs = getnbasefunctions(cellvalues)
for q_point in 1:getnquadpoints(cellvalues)
## For each integration point, compute strain, stress and material stiffness
ε = function_symmetric_gradient(cellvalues, q_point, ue)
σ, ∂σ∂ε = material_routine(material, ε)

dΩ = getdetJdV(cellvalues, q_point)
for i in 1:n_basefuncs
∇Nᵢ = shape_gradient(cellvalues, q_point, i)# shape_symmetric_gradient(cellvalues, q_point, i)
KnutAM marked this conversation as resolved.
Show resolved Hide resolved
fe[i] += σ ⊡ ∇Nᵢ * dΩ # add internal force to residual
for j in 1:n_basefuncs
∇ˢʸᵐNⱼ = shape_symmetric_gradient(cellvalues, q_point, j)
ke[i, j] += (∂σ∂ε ⊡ ∇ˢʸᵐNⱼ) ⊡ ∇Nᵢ * dΩ
end
end
end
end
#md nothing # hide
kimauth marked this conversation as resolved.
Show resolved Hide resolved

# #### Global assembly
# We define the function `assemble_global` to loop over the elements and do the global
# assembly. The function takes the sparse matrix `K`, a guess of the displacement vector
# (this would be more relevant for non-linear problems), our DofHandler, our `cellvalues` and
# the material definition that we want to use for the stress computation
# as input arguments and fills the assembled global stiffness matrix, and the assembled
# global force vector.
function assemble_global!(K, f, a, dh, cellvalues, material)
## Allocate the element stiffness matrix and element force vector
n_basefuncs = getnbasefunctions(cellvalues)
ke = zeros(n_basefuncs, n_basefuncs)
fe = zeros(n_basefuncs)
## Create an assembler
assembler = start_assemble(K, f)
## Loop over all cells
for cell in CellIterator(dh)
reinit!(cellvalues, cell) # update spatial derivatives based on element coordinates
@views ue = a[celldofs(cell)]
## Compute element contribution
assemble_cell!(ke, fe, cellvalues, material, ue)
## Assemble ke and fe into K and f
assemble!(assembler, celldofs(cell), ke, fe)
end
return K, f
end
#md nothing # hide

# ### Solution of the system
# The last step is to solve the system. First we call `assemble_global`
# to obtain the global stiffness matrix `K` and force vector `f`.
K = create_sparsity_pattern(dh)
f = zeros(ndofs(dh))
a = zeros(ndofs(dh))
assemble_global!(K, f, a, dh, cellvalues, material);

# To account for the Dirichlet boundary conditions we use the `apply!` function.
# This modifies elements in `K` and `f` respectively, such that
# we can get the correct solution vector `u` by using `\`.
apply!(K, f, ch)
u = K \ f;

# ### Exporting to VTK
# To visualize the result we export the grid and our field `u`
# to a VTK-file, which can be viewed in e.g. [ParaView](https://www.paraview.org/).
vtk_grid("linear_elasticity", dh) do vtk
vtk_point_data(vtk, dh, u)
vtk_cellset(vtk, grid)
end
#md nothing # hide

#md # ## [Plain program](@id heat_equation-plain-program)
#md #
#md # Here follows a version of the program without any comments.
#md # The file is also available here: [`heat_equation.jl`](heat_equation.jl).
#md #
#md # ```julia
#md # @__CODE__
#md # ```
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
45 changes: 45 additions & 0 deletions docs/src/literate-tutorials/logo.geo
termi-official marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
Point (1) = {1.000000000000,1.000000000000,0.000000000000};
Point (2) = {1.000000000000,0.379757979263,-0.000000000000};
Point (3) = {0.801534880751,0.454963545532,-0.000000000000};
Point (4) = {0.656107331955,1.000000000000,-0.000000000000};
Point (5) = {0.600672553037,0.775245709538,-0.000000000000};
Point (6) = {0.000000000000,1.000000000000,0.000000000000};
Point (7) = {0.392825178821,0.672136259831,-0.000000000000};
Point (8) = {1.000000000000,0.000000000000,0.000000000000};
Point (9) = {0.547800422194,-0.000000000000,-0.000000000000};
Point (10) = {0.488710023938,0.224380304618,-0.000000000000};
Point (11) = {0.000000000000,0.000000000000,0.000000000000};
Point (12) = {-0.000000000000,0.324566579562,-0.000000000000};
Point (13) = {0.172066723668,0.367888021869,-0.000000000000};
Line (1) = {2,1};
Line (2) = {1,4};
Line (3) = {2,3};
Line (4) = {3,5};
Line (5) = {5,4};
Line (6) = {4,6};
Line (7) = {7,6};
Line (8) = {5,7};
Line (9) = {8,2};
Line (10) = {9,8};
Line (11) = {9,10};
Line (12) = {10,3};
Line (13) = {12,11};
Line (14) = {11,9};
Line (15) = {12,13};
Line (16) = {13,10};
Line (17) = {6,12};
Line (18) = {7,13};
Line Loop (1) = {-1,3,4,5,-2};
Plane Surface (1) = {1}; Physical Surface (1) = {1};
Line Loop (2) = {7,-6,-5,8};
Plane Surface (2) = {2}; Physical Surface (2) = {2};
Line Loop (3) = {-10,11,12,-3,-9};
Plane Surface (3) = {3}; Physical Surface (3) = {3};
Line Loop (4) = {-11,-14,-13,15,16};
Plane Surface (4) = {4}; Physical Surface (4) = {4};
Line Loop (5) = {-7,18,-15,-17};
Plane Surface (5) = {5}; Physical Surface (5) = {5};
Line Loop (6) = {-16,-18,-8,-4,-12};
Plane Surface (6) = {6}; Physical Surface (6) = {6};

ReverseMesh Surface{1,2,3,4,5,6};