conspire/constitutive/solid/elastic/
mod.rs#![doc = include_str!("doc.md")]
#[cfg(test)]
pub mod test;
mod almansi_hamel;
pub use almansi_hamel::AlmansiHamel;
use super::*;
use crate::math::optimize::{NewtonRaphson, SecondOrder};
pub trait Elastic<'a>
where
Self: Solid<'a>,
{
fn cauchy_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<CauchyStress, ConstitutiveError> {
Ok(deformation_gradient
* self.second_piola_kirchhoff_stress(deformation_gradient)?
* deformation_gradient.transpose()
/ deformation_gradient.determinant())
}
fn cauchy_tangent_stiffness(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<CauchyTangentStiffness, ConstitutiveError> {
let deformation_gradient_inverse_transpose = deformation_gradient.inverse_transpose();
let cauchy_stress = self.cauchy_stress(deformation_gradient)?;
let some_stress = &cauchy_stress * &deformation_gradient_inverse_transpose;
Ok(self
.second_piola_kirchhoff_tangent_stiffness(deformation_gradient)?
.contract_first_second_indices_with_second_indices_of(
deformation_gradient,
deformation_gradient,
)
/ deformation_gradient.determinant()
- CauchyTangentStiffness::dyad_ij_kl(
&cauchy_stress,
&deformation_gradient_inverse_transpose,
)
+ CauchyTangentStiffness::dyad_il_kj(&some_stress, &IDENTITY)
+ CauchyTangentStiffness::dyad_ik_jl(&IDENTITY, &some_stress))
}
fn first_piola_kirchhoff_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<FirstPiolaKirchhoffStress, ConstitutiveError> {
Ok(self.cauchy_stress(deformation_gradient)?
* deformation_gradient.inverse_transpose()
* deformation_gradient.determinant())
}
fn first_piola_kirchhoff_tangent_stiffness(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<FirstPiolaKirchhoffTangentStiffness, ConstitutiveError> {
let deformation_gradient_inverse_transpose = deformation_gradient.inverse_transpose();
let first_piola_kirchhoff_stress =
self.first_piola_kirchhoff_stress(deformation_gradient)?;
Ok(self
.cauchy_tangent_stiffness(deformation_gradient)?
.contract_second_index_with_first_index_of(&deformation_gradient_inverse_transpose)
* deformation_gradient.determinant()
+ FirstPiolaKirchhoffTangentStiffness::dyad_ij_kl(
&first_piola_kirchhoff_stress,
&deformation_gradient_inverse_transpose,
)
- FirstPiolaKirchhoffTangentStiffness::dyad_il_kj(
&first_piola_kirchhoff_stress,
&deformation_gradient_inverse_transpose,
))
}
fn second_piola_kirchhoff_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<SecondPiolaKirchhoffStress, ConstitutiveError> {
Ok(deformation_gradient.inverse()
* self.cauchy_stress(deformation_gradient)?
* deformation_gradient.inverse_transpose()
* deformation_gradient.determinant())
}
fn second_piola_kirchhoff_tangent_stiffness(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<SecondPiolaKirchhoffTangentStiffness, ConstitutiveError> {
let deformation_gradient_inverse_transpose = deformation_gradient.inverse_transpose();
let deformation_gradient_inverse = deformation_gradient_inverse_transpose.transpose();
let second_piola_kirchhoff_stress =
self.second_piola_kirchhoff_stress(deformation_gradient)?;
Ok(self
.cauchy_tangent_stiffness(deformation_gradient)?
.contract_first_second_indices_with_second_indices_of(
&deformation_gradient_inverse,
&deformation_gradient_inverse,
)
* deformation_gradient.determinant()
+ SecondPiolaKirchhoffTangentStiffness::dyad_ij_kl(
&second_piola_kirchhoff_stress,
&deformation_gradient_inverse_transpose,
)
- SecondPiolaKirchhoffTangentStiffness::dyad_il_kj(
&second_piola_kirchhoff_stress,
&deformation_gradient_inverse_transpose,
)
- SecondPiolaKirchhoffTangentStiffness::dyad_ik_jl(
&deformation_gradient_inverse,
&second_piola_kirchhoff_stress,
))
}
fn solve(
&self,
applied_load: AppliedLoad,
) -> Result<(DeformationGradient, CauchyStress), ConstitutiveError> {
let optimization = NewtonRaphson {
..Default::default()
};
let deformation_gradient = match applied_load {
AppliedLoad::UniaxialStress(deformation_gradient_11) => {
let deformation_gradient_33 = optimization.minimize(
|deformation_gradient_33: &Scalar| {
Ok(self.cauchy_stress(&DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, *deformation_gradient_33, 0.0],
[0.0, 0.0, *deformation_gradient_33],
]))?[2][2])
},
|deformation_gradient_33: &Scalar| {
Ok(self.cauchy_tangent_stiffness(&DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, *deformation_gradient_33, 0.0],
[0.0, 0.0, *deformation_gradient_33],
]))?[2][2][2][2])
},
1.0 / deformation_gradient_11.sqrt(),
None,
None,
)?;
DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, deformation_gradient_33, 0.0],
[0.0, 0.0, deformation_gradient_33],
])
}
AppliedLoad::BiaxialStress(deformation_gradient_11, deformation_gradient_22) => {
let deformation_gradient_33 = optimization.minimize(
|deformation_gradient_33: &Scalar| {
Ok(self.cauchy_stress(&DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, deformation_gradient_22, 0.0],
[0.0, 0.0, *deformation_gradient_33],
]))?[2][2])
},
|deformation_gradient_33: &Scalar| {
Ok(self.cauchy_tangent_stiffness(&DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, deformation_gradient_22, 0.0],
[0.0, 0.0, *deformation_gradient_33],
]))?[2][2][2][2])
},
1.0 / deformation_gradient_11 / deformation_gradient_22,
None,
None,
)?;
DeformationGradient::new([
[deformation_gradient_11, 0.0, 0.0],
[0.0, deformation_gradient_22, 0.0],
[0.0, 0.0, deformation_gradient_33],
])
}
};
let cauchy_stress = self.cauchy_stress(&deformation_gradient)?;
Ok((deformation_gradient, cauchy_stress))
}
}