conspire/constitutive/hybrid/elastic/multiplicative/
mod.rs#[cfg(test)]
mod test;
use crate::{
constitutive::{
hybrid::{Hybrid, Multiplicative, MultiplicativeTrait},
solid::{elastic::Elastic, Solid},
Constitutive, ConstitutiveError, Parameters,
},
math::{Rank2, Tensor, TensorRank2, IDENTITY_10, ZERO_10},
mechanics::{
CauchyStress, CauchyTangentStiffness, DeformationGradient, FirstPiolaKirchhoffStress,
FirstPiolaKirchhoffTangentStiffness, Scalar, SecondPiolaKirchhoffStress,
SecondPiolaKirchhoffTangentStiffness,
},
ABS_TOL,
};
impl<'a, C1: Elastic<'a>, C2: Elastic<'a>> Constitutive<'a> for Multiplicative<C1, C2> {
fn new(_parameters: Parameters<'a>) -> Self {
panic!()
}
}
impl<'a, C1: Elastic<'a>, C2: Elastic<'a>> Solid<'a> for Multiplicative<C1, C2> {
fn bulk_modulus(&self) -> &Scalar {
panic!()
}
fn shear_modulus(&self) -> &Scalar {
panic!()
}
}
impl<'a, C1: Elastic<'a>, C2: Elastic<'a>> Elastic<'a> for Multiplicative<C1, C2> {
fn cauchy_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<CauchyStress, ConstitutiveError> {
let (deformation_gradient_1, deformation_gradient_2) =
self.deformation_gradients(deformation_gradient)?;
Ok(self
.constitutive_model_1()
.cauchy_stress(&deformation_gradient_1)?
/ deformation_gradient_2.determinant())
}
fn cauchy_tangent_stiffness(
&self,
_: &DeformationGradient,
) -> Result<CauchyTangentStiffness, ConstitutiveError> {
panic!()
}
fn first_piola_kirchhoff_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<FirstPiolaKirchhoffStress, ConstitutiveError> {
let (deformation_gradient_1, deformation_gradient_2) =
self.deformation_gradients(deformation_gradient)?;
let deformation_gradient_2_inverse_transpose: TensorRank2<3, 0, 0> =
deformation_gradient_2.inverse_transpose().into();
Ok(self
.constitutive_model_1()
.first_piola_kirchhoff_stress(&deformation_gradient_1)?
* deformation_gradient_2_inverse_transpose)
}
fn first_piola_kirchhoff_tangent_stiffness(
&self,
_: &DeformationGradient,
) -> Result<FirstPiolaKirchhoffTangentStiffness, ConstitutiveError> {
panic!()
}
fn second_piola_kirchhoff_stress(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<SecondPiolaKirchhoffStress, ConstitutiveError> {
let (deformation_gradient_1, deformation_gradient_2) =
self.deformation_gradients(deformation_gradient)?;
let deformation_gradient_2_inverse: TensorRank2<3, 0, 0> =
deformation_gradient_2.inverse().into();
Ok(&deformation_gradient_2_inverse
* self
.constitutive_model_1()
.second_piola_kirchhoff_stress(&deformation_gradient_1)?
* deformation_gradient_2_inverse.transpose())
}
fn second_piola_kirchhoff_tangent_stiffness(
&self,
_: &DeformationGradient,
) -> Result<SecondPiolaKirchhoffTangentStiffness, ConstitutiveError> {
panic!()
}
}
impl<'a, C1: Elastic<'a>, C2: Elastic<'a>> MultiplicativeTrait for Multiplicative<C1, C2> {
fn deformation_gradients(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<(DeformationGradient, DeformationGradient), ConstitutiveError> {
if deformation_gradient.is_identity() {
Ok((IDENTITY_10, IDENTITY_10))
} else {
let mut deformation_gradient_1 = IDENTITY_10;
let mut deformation_gradient_2 = deformation_gradient.clone();
let mut deformation_gradient_2_old = IDENTITY_10;
let mut deformation_gradient_2_inverse_transpose: TensorRank2<3, 0, 0>;
let mut residual: FirstPiolaKirchhoffStress;
let mut residual_increment: FirstPiolaKirchhoffStress;
let mut residual_norm = 1.0;
let mut residual_old = ZERO_10;
let mut right_hand_side: FirstPiolaKirchhoffStress;
let mut steps: u8 = 0;
let steps_maximum: u8 = 50;
let mut step_size: Scalar;
while steps < steps_maximum {
deformation_gradient_1 =
(deformation_gradient * deformation_gradient_2.inverse()).into();
deformation_gradient_2_inverse_transpose =
deformation_gradient_2.inverse_transpose().into();
right_hand_side = (deformation_gradient_1.transpose()
* self
.constitutive_model_1()
.first_piola_kirchhoff_stress(&deformation_gradient_1)?
* deformation_gradient_2_inverse_transpose)
.into();
residual = self
.constitutive_model_2()
.first_piola_kirchhoff_stress(&deformation_gradient_2)?
- right_hand_side;
residual_norm = residual.norm();
if residual_norm >= ABS_TOL {
residual_increment = residual_old - &residual;
step_size = (deformation_gradient_2_old - &deformation_gradient_2)
.full_contraction(&residual_increment)
.abs()
/ residual_increment.norm_squared();
deformation_gradient_2_old = deformation_gradient_2.clone();
residual_old = residual.clone();
deformation_gradient_2 -= residual * step_size;
} else {
break;
}
steps += 1;
}
if residual_norm >= ABS_TOL && steps == steps_maximum {
panic!("MAX STEPS REACHED")
} else {
Ok((deformation_gradient_1, deformation_gradient_2))
}
}
}
}