#[cfg(test)]
mod test;
use super::*;
#[derive(Debug)]
pub struct SaintVenantKirchoff<'a> {
parameters: Parameters<'a>,
}
impl<'a> Constitutive<'a> for SaintVenantKirchoff<'a> {
fn new(parameters: Parameters<'a>) -> Self {
Self { parameters }
}
}
impl<'a> Solid<'a> for SaintVenantKirchoff<'a> {
fn get_bulk_modulus(&self) -> &Scalar {
&self.parameters[0]
}
fn get_shear_modulus(&self) -> &Scalar {
&self.parameters[1]
}
}
impl<'a> Viscous<'a> for SaintVenantKirchoff<'a> {
fn get_bulk_viscosity(&self) -> &Scalar {
&self.parameters[2]
}
fn get_shear_viscosity(&self) -> &Scalar {
&self.parameters[3]
}
}
impl<'a> Viscoelastic<'a> for SaintVenantKirchoff<'a> {
fn calculate_second_piola_kirchoff_stress(
&self,
deformation_gradient: &DeformationGradient,
deformation_gradient_rate: &DeformationGradientRate,
) -> Result<SecondPiolaKirchoffStress, ConstitutiveError> {
let jacobian = deformation_gradient.determinant();
if jacobian > 0.0 {
let (deviatoric_strain, strain_trace) = ((self
.calculate_right_cauchy_green_deformation(deformation_gradient)
- IDENTITY_00)
* 0.5)
.deviatoric_and_trace();
let first_term = deformation_gradient_rate.transpose() * deformation_gradient;
let (deviatoric_strain_rate, strain_rate_trace) =
((&first_term + first_term.transpose()) * 0.5).deviatoric_and_trace();
Ok(deviatoric_strain * (2.0 * self.get_shear_modulus())
+ deviatoric_strain_rate * (2.0 * self.get_shear_viscosity())
+ IDENTITY_00
* (self.get_bulk_modulus() * strain_trace
+ self.get_bulk_viscosity() * strain_rate_trace))
} else {
Err(ConstitutiveError::InvalidJacobian(
jacobian,
deformation_gradient.copy(),
format!("{:?}", &self),
))
}
}
fn calculate_second_piola_kirchoff_rate_tangent_stiffness(
&self,
deformation_gradient: &DeformationGradient,
_: &DeformationGradientRate,
) -> Result<SecondPiolaKirchoffRateTangentStiffness, ConstitutiveError> {
let jacobian = deformation_gradient.determinant();
if jacobian > 0.0 {
let scaled_deformation_gradient_transpose =
deformation_gradient.transpose() * self.get_shear_viscosity();
Ok(SecondPiolaKirchoffRateTangentStiffness::dyad_ik_jl(
&scaled_deformation_gradient_transpose,
&IDENTITY_00,
) + SecondPiolaKirchoffRateTangentStiffness::dyad_il_jk(
&IDENTITY_00,
&scaled_deformation_gradient_transpose,
) + SecondPiolaKirchoffRateTangentStiffness::dyad_ij_kl(
&(IDENTITY_00
* (self.get_bulk_viscosity() - TWO_THIRDS * self.get_shear_viscosity())),
deformation_gradient,
))
} else {
Err(ConstitutiveError::InvalidJacobian(
jacobian,
deformation_gradient.copy(),
format!("{:?}", &self),
))
}
}
}
impl<'a> ElasticHyperviscous<'a> for SaintVenantKirchoff<'a> {
fn calculate_viscous_dissipation(
&self,
deformation_gradient: &DeformationGradient,
deformation_gradient_rate: &DeformationGradientRate,
) -> Result<Scalar, ConstitutiveError> {
let jacobian = deformation_gradient.determinant();
if jacobian > 0.0 {
let first_term = deformation_gradient_rate.transpose() * deformation_gradient;
let strain_rate = (&first_term + first_term.transpose()) * 0.5;
Ok(self.get_shear_viscosity() * strain_rate.squared_trace()
+ 0.5
* (self.get_bulk_viscosity() - TWO_THIRDS * self.get_shear_viscosity())
* strain_rate.trace().powi(2))
} else {
Err(ConstitutiveError::InvalidJacobian(
jacobian,
deformation_gradient.copy(),
format!("{:?}", &self),
))
}
}
}
impl<'a> Hyperviscoelastic<'a> for SaintVenantKirchoff<'a> {
fn calculate_helmholtz_free_energy_density(
&self,
deformation_gradient: &DeformationGradient,
) -> Result<Scalar, ConstitutiveError> {
let jacobian = deformation_gradient.determinant();
if jacobian > 0.0 {
let strain = (self.calculate_right_cauchy_green_deformation(deformation_gradient)
- IDENTITY_00)
* 0.5;
Ok(self.get_shear_modulus() * strain.squared_trace()
+ 0.5
* (self.get_bulk_modulus() - TWO_THIRDS * self.get_shear_modulus())
* strain.trace().powi(2))
} else {
Err(ConstitutiveError::InvalidJacobian(
jacobian,
deformation_gradient.copy(),
format!("{:?}", &self),
))
}
}
}