use crate::{
constitutive::solid::elastic_viscoplastic::ElasticViscoplastic,
fem::{
NodalCoordinates, NodalCoordinatesHistory,
block::{
Block, FiniteElementBlockError,
element::{
FiniteElementError, solid::elastic_viscoplastic::ElasticViscoplasticFiniteElement,
},
solid::{NodalForcesSolid, NodalStiffnessesSolid, SolidFiniteElementBlock},
},
},
math::{
Scalar, Tensor, TensorTupleListVec, TensorTupleListVec2D,
integrate::{ExplicitDaeFirstOrderRoot, IntegrationError},
optimize::{EqualityConstraint, FirstOrderRootFinding},
},
mechanics::{DeformationGradientPlastic, Times},
};
use std::array::from_fn;
pub type ViscoplasticStateVariables<const G: usize, Y> =
TensorTupleListVec<DeformationGradientPlastic, Y, G>;
pub type ViscoplasticStateVariablesHistory<const G: usize, Y> =
TensorTupleListVec2D<DeformationGradientPlastic, Y, G>;
pub type ElasticViscoplasticBCs = fn(Scalar) -> EqualityConstraint;
pub trait ElasticViscoplasticFiniteElementBlock<
C,
F,
const G: usize,
const M: usize,
const N: usize,
const P: usize,
Y,
> where
C: ElasticViscoplastic<Y>,
F: ElasticViscoplasticFiniteElement<C, G, M, N, P, Y>,
Self: SolidFiniteElementBlock<C, F, G, M, N, P>,
Y: Tensor,
{
fn nodal_forces(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<NodalForcesSolid, FiniteElementBlockError>;
fn nodal_stiffnesses(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<NodalStiffnessesSolid, FiniteElementBlockError>;
fn state_variables_evolution(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<ViscoplasticStateVariables<G, Y>, FiniteElementBlockError>;
fn root(
&self,
integrator: impl ExplicitDaeFirstOrderRoot<
NodalForcesSolid,
NodalStiffnessesSolid,
ViscoplasticStateVariables<G, Y>,
NodalCoordinates,
ViscoplasticStateVariablesHistory<G, Y>,
NodalCoordinatesHistory,
>,
solver: impl FirstOrderRootFinding<NodalForcesSolid, NodalStiffnessesSolid, NodalCoordinates>,
time: &[Scalar],
bcs: ElasticViscoplasticBCs,
) -> Result<
(
Times,
NodalCoordinatesHistory,
ViscoplasticStateVariablesHistory<G, Y>,
),
IntegrationError,
>;
}
impl<C, F, const G: usize, const M: usize, const N: usize, const P: usize, Y>
ElasticViscoplasticFiniteElementBlock<C, F, G, M, N, P, Y> for Block<C, F, G, M, N, P>
where
C: ElasticViscoplastic<Y>,
F: ElasticViscoplasticFiniteElement<C, G, M, N, P, Y>,
Self: SolidFiniteElementBlock<C, F, G, M, N, P>,
Y: Tensor,
{
fn nodal_forces(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<NodalForcesSolid, FiniteElementBlockError> {
let mut nodal_forces = NodalForcesSolid::zero(nodal_coordinates.len());
match self
.elements()
.iter()
.zip(self.connectivity())
.zip(state_variables)
.try_for_each(|((element, nodes), state_variables_element)| {
element
.nodal_forces(
self.constitutive_model(),
&Self::element_coordinates(nodal_coordinates, nodes),
state_variables_element,
)?
.into_iter()
.zip(nodes)
.for_each(|(nodal_force, &node)| nodal_forces[node] += nodal_force);
Ok::<(), FiniteElementError>(())
}) {
Ok(()) => Ok(nodal_forces),
Err(error) => Err(FiniteElementBlockError::Upstream(
format!("{error}"),
format!("{self:?}"),
)),
}
}
fn nodal_stiffnesses(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<NodalStiffnessesSolid, FiniteElementBlockError> {
let mut nodal_stiffnesses = NodalStiffnessesSolid::zero(nodal_coordinates.len());
match self
.elements()
.iter()
.zip(self.connectivity())
.zip(state_variables)
.try_for_each(|((element, nodes), state_variables_element)| {
element
.nodal_stiffnesses(
self.constitutive_model(),
&Self::element_coordinates(nodal_coordinates, nodes),
state_variables_element,
)?
.into_iter()
.zip(nodes)
.for_each(|(object, &node_a)| {
object
.into_iter()
.zip(nodes)
.for_each(|(nodal_stiffness, &node_b)| {
nodal_stiffnesses[node_a][node_b] += nodal_stiffness
})
});
Ok::<(), FiniteElementError>(())
}) {
Ok(()) => Ok(nodal_stiffnesses),
Err(error) => Err(FiniteElementBlockError::Upstream(
format!("{error}"),
format!("{self:?}"),
)),
}
}
fn state_variables_evolution(
&self,
nodal_coordinates: &NodalCoordinates,
state_variables: &ViscoplasticStateVariables<G, Y>,
) -> Result<ViscoplasticStateVariables<G, Y>, FiniteElementBlockError> {
match self
.elements()
.iter()
.zip(self.connectivity())
.zip(state_variables)
.map(|((element, nodes), element_state_variables)| {
element.state_variables_evolution(
self.constitutive_model(),
&Self::element_coordinates(nodal_coordinates, nodes),
element_state_variables,
)
})
.collect()
{
Ok(state_variables_evolution) => Ok(state_variables_evolution),
Err(error) => Err(FiniteElementBlockError::Upstream(
format!("{error}"),
format!("{self:?}"),
)),
}
}
fn root(
&self,
integrator: impl ExplicitDaeFirstOrderRoot<
NodalForcesSolid,
NodalStiffnessesSolid,
ViscoplasticStateVariables<G, Y>,
NodalCoordinates,
ViscoplasticStateVariablesHistory<G, Y>,
NodalCoordinatesHistory,
>,
solver: impl FirstOrderRootFinding<NodalForcesSolid, NodalStiffnessesSolid, NodalCoordinates>,
time: &[Scalar],
bcs: ElasticViscoplasticBCs,
) -> Result<
(
Times,
NodalCoordinatesHistory,
ViscoplasticStateVariablesHistory<G, Y>,
),
IntegrationError,
> {
let (time_history, state_variables_history, _, nodal_coordinates_history) = integrator
.integrate(
|_: Scalar,
state_variables: &ViscoplasticStateVariables<G, Y>,
nodal_coordinates: &NodalCoordinates| {
Ok(self.state_variables_evolution(nodal_coordinates, state_variables)?)
},
|_t: Scalar,
state_variables: &ViscoplasticStateVariables<G, Y>,
nodal_coordinates: &NodalCoordinates| {
Ok(self.nodal_forces(nodal_coordinates, state_variables)?)
},
|_t: Scalar,
state_variables: &ViscoplasticStateVariables<G, Y>,
nodal_coordinates: &NodalCoordinates| {
Ok(self.nodal_stiffnesses(nodal_coordinates, state_variables)?)
},
solver,
time,
(
self.elements()
.iter()
.map(|_| from_fn(|_| self.constitutive_model().initial_state()).into())
.collect(),
self.coordinates().clone().into(),
),
bcs,
)?;
Ok((
time_history,
nodal_coordinates_history,
state_variables_history,
))
}
}