conspire/fem/block/element/linear/tetrahedron/
mod.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#[cfg(test)]
mod test;

use super::*;
use crate::{
    constitutive::{Constitutive, Parameters},
    math::{tensor_rank_0_list, tensor_rank_1, tensor_rank_1_list, tensor_rank_1_list_2d},
    mechanics::Scalar,
};
use std::array::from_fn;

const G: usize = 1;
const M: usize = 3;
const N: usize = 4;
const P: usize = 1;

#[cfg(test)]
const Q: usize = 4;

pub type Tetrahedron<C> = Element<C, G, N>;

impl<'a, C> FiniteElement<'a, C, G, N> for Tetrahedron<C>
where
    C: Constitutive<'a>,
{
    fn new(
        constitutive_model_parameters: Parameters<'a>,
        reference_nodal_coordinates: ReferenceNodalCoordinates<N>,
    ) -> Self {
        let standard_gradient_operator = &Self::standard_gradient_operators()[0];
        let (operator, jacobian) = (reference_nodal_coordinates * standard_gradient_operator)
            .inverse_transpose_and_determinant();
        Self {
            constitutive_models: from_fn(|_| <C>::new(constitutive_model_parameters)),
            gradient_vectors: tensor_rank_1_list_2d([operator * standard_gradient_operator]),
            integration_weights: tensor_rank_0_list([jacobian * Self::integration_weight()]),
        }
    }
}

impl<'a, C> Tetrahedron<C>
where
    C: Constitutive<'a>,
{
    const fn integration_weight() -> Scalar {
        1.0 / 6.0
    }
    #[cfg(test)]
    const fn shape_functions_at_integration_points() -> ShapeFunctionsAtIntegrationPoints<G, Q> {
        tensor_rank_1_list([tensor_rank_1([0.25; Q])])
    }
    const fn standard_gradient_operators() -> StandardGradientOperators<M, N, P> {
        tensor_rank_1_list_2d([tensor_rank_1_list([
            tensor_rank_1([-1.0, -1.0, -1.0]),
            tensor_rank_1([1.0, 0.0, 0.0]),
            tensor_rank_1([0.0, 1.0, 0.0]),
            tensor_rank_1([0.0, 0.0, 1.0]),
        ])])
    }
}