Biaxial stress test#
Attempt to reproduce Figure 8 in [1].
[1] Holzapfel, Gerhard A., and Ray W. Ogden. “Constitutive modelling of passive myocardium: a structurally based framework for material characterization. “Philosophical Transactions of the Royal Society of London A: Mathematical, Physical and Engineering Sciences 367.1902 (2009): 3445-3475.
import dolfin
import matplotlib.pyplot as plt
import numpy as np
try:
from dolfin_adjoint import (
Constant,
DirichletBC,
Expression,
UnitCubeMesh,
interpolate,
)
except ImportError:
from dolfin import (
Constant,
DirichletBC,
interpolate,
Expression,
UnitCubeMesh,
)
# -
import pulse
# Create mesh
N = 4
mesh = UnitCubeMesh(N, N, N)
# Create a facet fuction in order to mark the subdomains
ffun = dolfin.MeshFunction("size_t", mesh, 2)
ffun.set_all(0)
# Mark subdomains
xlow = dolfin.CompiledSubDomain("near(x[0], 0) && on_boundary")
xlow_marker = 1
xlow.mark(ffun, xlow_marker)
xhigh = dolfin.CompiledSubDomain("near(x[0], 1.0) && on_boundary")
xhigh_marker = 2
xhigh.mark(ffun, xhigh_marker)
ylow = dolfin.CompiledSubDomain("near(x[1], 0) && on_boundary")
ylow_marker = 3
ylow.mark(ffun, ylow_marker)
yhigh = dolfin.CompiledSubDomain("near(x[1], 1) && on_boundary")
yhigh_marker = 4
yhigh.mark(ffun, yhigh_marker)
center = dolfin.CompiledSubDomain(
"near(x[0], 0.5) && near(x[1], 0.5) && near(x[2], 0.5)",
)
center_marker = 5
center.mark(ffun, center_marker)
dolfin.File("ffun.pvd") << ffun
# Collect the functions containing the markers
marker_functions = pulse.MarkerFunctions(ffun=ffun)
# Create mictrotructure
V_f = dolfin.VectorFunctionSpace(mesh, "CG", 1)
# Fibers
f0 = interpolate(Expression(("1.0", "0.0", "0.0"), degree=1), V_f)
# Sheets
s0 = interpolate(Expression(("0.0", "1.0", "0.0"), degree=1), V_f)
# Fiber-sheet normal
n0 = interpolate(Expression(("0.0", "0.0", "1.0"), degree=1), V_f)
# Collect the mictrotructure
microstructure = pulse.Microstructure(f0=f0, s0=s0, n0=n0)
# Create the geometry
geometry = pulse.Geometry(
mesh=mesh,
marker_functions=marker_functions,
microstructure=microstructure,
)
# Use the default material parameters
material_parameters = {
"a": 2.28,
"b": 9.726,
"a_f": 1.685,
"b_f": 15.779,
"a_s": 0,
"a_fs": 0,
}
# Create material
material = pulse.HolzapfelOgden(parameters=material_parameters)
# Eff / Ess strain ratio
strain_ratio = Constant(1.0)
# Create costants defined for the dirichlet BC
u0 = Constant(0.0)
x_strain = u0 * strain_ratio / 2
y_strain = u0 * (1 / strain_ratio) / 2
# Make Dirichlet boundary conditions
def dirichlet_bc(W):
V = W if W.sub(0).num_sub_spaces() == 0 else W.sub(0)
return [
DirichletBC(V.sub(0), Constant(-x_strain), xlow),
DirichletBC(V.sub(0), Constant(x_strain), xhigh),
DirichletBC(V.sub(1), Constant(-y_strain), ylow),
DirichletBC(V.sub(1), Constant(y_strain), yhigh),
DirichletBC(V, np.zeros(3), center, method="pointwise"),
]
# Collect Boundary Conditions
bcs = pulse.BoundaryConditions(dirichlet=(dirichlet_bc,))
# Create problem
problem = pulse.MechanicsProblem(geometry, material, bcs)
# Solve problem
max_xi = [0.1, 0.15, 0.1]
data = {}
for i, sr in enumerate([2.05, 1.02, 0.48]):
strain_ratio.assign(sr)
Effs = []
Sffs = []
Esss = []
Ssss = []
for xi in np.linspace(0, max_xi[i], 10):
try:
pulse.iterate.iterate(problem, u0, Constant(xi), reinit_each_step=True)
except pulse.mechanicsproblem.SolverDidNotConverge:
continue
S = problem.SecondPiolaStress()
E = problem.GreenLagrangeStrain()
Sff = dolfin.assemble(dolfin.inner(f0, S * f0) * dolfin.dx)
Eff = dolfin.assemble(dolfin.inner(f0, E * f0) * dolfin.dx)
Effs.append(Eff)
Sffs.append(Sff)
Sss = dolfin.assemble(dolfin.inner(s0, S * s0) * dolfin.dx)
Ess = dolfin.assemble(dolfin.inner(s0, E * s0) * dolfin.dx)
Esss.append(Ess)
Ssss.append(Sss)
u, p = problem.state.split(deepcopy=True)
data[sr] = {"Eff": Effs, "Sff": Sffs, "Ess": Esss, "Sss": Ssss}
2024-09-05 11:44:36,065 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:44:36,066 [248] INFO pulse.iterate: Current control: f_31 = 0.000
2024-09-05 11:44:36,066 [248] INFO pulse.iterate: Target: 0.000
2024-09-05 11:44:36,195 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,195 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,196 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,197 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,197 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,198 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:36,375 [248] DEBUG UFL_LEGACY: Blocks of each mode:
1 full
2024-09-05 11:44:47,110 [248] DEBUG UFL_LEGACY: Blocks of each mode:
1 full
2024-09-05 11:44:47,583 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,584 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,584 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,584 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,585 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,585 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:47,805 [248] DEBUG UFL_LEGACY: Blocks of each mode:
1 full
2024-09-05 11:44:48,427 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:44:48,428 [248] INFO pulse.iterate: Current control: f_31 = 0.000
2024-09-05 11:44:48,428 [248] INFO pulse.iterate: Target: 0.011
2024-09-05 11:44:48,483 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,483 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,484 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,484 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,485 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,485 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:44:48,955 [248] DEBUG UFL_LEGACY: Blocks of each mode:
99 full
2024-09-05 11:45:39,845 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,846 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,847 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,847 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,848 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,848 [248] INFO UFL_LEGACY: Adjusting missing element cell to tetrahedron.
2024-09-05 11:45:39,986 [248] DEBUG UFL_LEGACY: Blocks of each mode:
10 full
2024-09-05 11:45:41,295 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:41,295 [248] INFO pulse.iterate: Current control: f_31 = 0.011
2024-09-05 11:45:41,296 [248] INFO pulse.iterate: Target: 0.022
2024-09-05 11:45:41,919 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:41,920 [248] INFO pulse.iterate: Current control: f_31 = 0.022
2024-09-05 11:45:41,920 [248] INFO pulse.iterate: Target: 0.033
2024-09-05 11:45:42,540 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:42,541 [248] INFO pulse.iterate: Current control: f_31 = 0.033
2024-09-05 11:45:42,541 [248] INFO pulse.iterate: Target: 0.044
2024-09-05 11:45:43,167 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:43,168 [248] INFO pulse.iterate: Current control: f_31 = 0.044
2024-09-05 11:45:43,168 [248] INFO pulse.iterate: Target: 0.056
2024-09-05 11:45:43,795 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:43,795 [248] INFO pulse.iterate: Current control: f_31 = 0.056
2024-09-05 11:45:43,796 [248] INFO pulse.iterate: Target: 0.067
2024-09-05 11:45:44,424 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:44,425 [248] INFO pulse.iterate: Current control: f_31 = 0.067
2024-09-05 11:45:44,425 [248] INFO pulse.iterate: Target: 0.078
2024-09-05 11:45:45,046 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:45,046 [248] INFO pulse.iterate: Current control: f_31 = 0.078
2024-09-05 11:45:45,047 [248] INFO pulse.iterate: Target: 0.089
2024-09-05 11:45:45,665 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:45,666 [248] INFO pulse.iterate: Current control: f_31 = 0.089
2024-09-05 11:45:45,666 [248] INFO pulse.iterate: Target: 0.100
2024-09-05 11:45:46,353 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:46,353 [248] INFO pulse.iterate: Current control: f_31 = 0.100
2024-09-05 11:45:46,354 [248] INFO pulse.iterate: Target: 0.000
2024-09-05 11:45:47,257 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:47,258 [248] INFO pulse.iterate: Current control: f_31 = 0.000
2024-09-05 11:45:47,258 [248] INFO pulse.iterate: Target: 0.017
2024-09-05 11:45:47,875 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:47,875 [248] INFO pulse.iterate: Current control: f_31 = 0.017
2024-09-05 11:45:47,876 [248] INFO pulse.iterate: Target: 0.033
2024-09-05 11:45:48,497 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:48,497 [248] INFO pulse.iterate: Current control: f_31 = 0.033
2024-09-05 11:45:48,498 [248] INFO pulse.iterate: Target: 0.050
2024-09-05 11:45:49,115 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:49,116 [248] INFO pulse.iterate: Current control: f_31 = 0.050
2024-09-05 11:45:49,116 [248] INFO pulse.iterate: Target: 0.067
2024-09-05 11:45:49,737 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:49,738 [248] INFO pulse.iterate: Current control: f_31 = 0.067
2024-09-05 11:45:49,739 [248] INFO pulse.iterate: Target: 0.083
2024-09-05 11:45:50,357 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:50,358 [248] INFO pulse.iterate: Current control: f_31 = 0.083
2024-09-05 11:45:50,358 [248] INFO pulse.iterate: Target: 0.100
2024-09-05 11:45:50,977 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:50,977 [248] INFO pulse.iterate: Current control: f_31 = 0.100
2024-09-05 11:45:50,978 [248] INFO pulse.iterate: Target: 0.117
2024-09-05 11:45:51,596 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:51,597 [248] INFO pulse.iterate: Current control: f_31 = 0.117
2024-09-05 11:45:51,597 [248] INFO pulse.iterate: Target: 0.133
2024-09-05 11:45:52,216 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:52,217 [248] INFO pulse.iterate: Current control: f_31 = 0.133
2024-09-05 11:45:52,217 [248] INFO pulse.iterate: Target: 0.150
2024-09-05 11:45:52,907 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:52,908 [248] INFO pulse.iterate: Current control: f_31 = 0.150
2024-09-05 11:45:52,909 [248] INFO pulse.iterate: Target: 0.000
2024-09-05 11:45:53,793 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:53,793 [248] INFO pulse.iterate: Current control: f_31 = 0.000
2024-09-05 11:45:53,794 [248] INFO pulse.iterate: Target: 0.011
2024-09-05 11:45:54,411 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:54,412 [248] INFO pulse.iterate: Current control: f_31 = 0.011
2024-09-05 11:45:54,413 [248] INFO pulse.iterate: Target: 0.022
2024-09-05 11:45:55,034 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:55,035 [248] INFO pulse.iterate: Current control: f_31 = 0.022
2024-09-05 11:45:55,036 [248] INFO pulse.iterate: Target: 0.033
2024-09-05 11:45:55,653 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:55,654 [248] INFO pulse.iterate: Current control: f_31 = 0.033
2024-09-05 11:45:55,655 [248] INFO pulse.iterate: Target: 0.044
2024-09-05 11:45:56,271 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:56,272 [248] INFO pulse.iterate: Current control: f_31 = 0.044
2024-09-05 11:45:56,272 [248] INFO pulse.iterate: Target: 0.056
2024-09-05 11:45:56,893 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:56,893 [248] INFO pulse.iterate: Current control: f_31 = 0.056
2024-09-05 11:45:56,894 [248] INFO pulse.iterate: Target: 0.067
2024-09-05 11:45:57,512 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:57,513 [248] INFO pulse.iterate: Current control: f_31 = 0.067
2024-09-05 11:45:57,513 [248] INFO pulse.iterate: Target: 0.078
2024-09-05 11:45:58,143 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:58,143 [248] INFO pulse.iterate: Current control: f_31 = 0.078
2024-09-05 11:45:58,144 [248] INFO pulse.iterate: Target: 0.089
2024-09-05 11:45:58,765 [248] INFO pulse.iterate: Iterating to target control (f_31)...
2024-09-05 11:45:58,766 [248] INFO pulse.iterate: Current control: f_31 = 0.089
2024-09-05 11:45:58,767 [248] INFO pulse.iterate: Target: 0.100
fig, ax = plt.subplots(1, 2, figsize=(10, 6))
markers = ["^", "s", "o"]
for i, (sr, v) in enumerate(data.items()):
ax[0].plot(v["Eff"], v["Sff"], marker=markers[i], label=f"SR: {sr:.2f}")
ax[1].plot(v["Ess"], v["Sss"], marker=markers[i], label=f"SR: {sr:.2f}")
ax[0].set_ylabel("$S_{ff}$ (kPa)")
ax[1].set_ylabel("$S_{ss}$ (kPa)")
ax[0].set_xlabel("$E_{ff}$")
ax[1].set_xlabel("$E_{ss}$")
for axi in ax:
axi.legend()
axi.set_ylim(0, 12)
axi.grid()
plt.show()
