2021-04-23 22:50:07 +05:30
|
|
|
|
"""
|
|
|
|
|
Finite-strain continuum mechanics.
|
|
|
|
|
|
2021-04-24 10:43:36 +05:30
|
|
|
|
All routines operate on numpy.ndarrays of shape (...,3,3).
|
2021-04-23 22:50:07 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2020-11-16 03:44:46 +05:30
|
|
|
|
|
2022-01-12 18:48:38 +05:30
|
|
|
|
from typing import Sequence as _Sequence
|
2020-11-16 03:44:46 +05:30
|
|
|
|
|
2020-04-10 16:00:39 +05:30
|
|
|
|
import numpy as _np
|
2019-10-19 00:20:03 +05:30
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
from . import tensor as _tensor
|
|
|
|
|
from . import _rotation
|
|
|
|
|
|
2020-11-16 03:44:46 +05:30
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def deformation_Cauchy_Green_left(F: _np.ndarray) -> _np.ndarray:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Calculate left Cauchy-Green deformation tensor (Finger deformation tensor).
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
F : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Deformation gradient.
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
B : numpy.ndarray, shape (...,3,3)
|
2020-11-19 19:08:54 +05:30
|
|
|
|
Left Cauchy-Green deformation tensor.
|
2019-10-30 22:35:44 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2020-11-19 18:35:59 +05:30
|
|
|
|
return _np.matmul(F,_tensor.transpose(F))
|
2019-11-27 16:49:37 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def deformation_Cauchy_Green_right(F: _np.ndarray) -> _np.ndarray:
|
2019-11-27 16:49:37 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Calculate right Cauchy-Green deformation tensor.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-11-27 16:49:37 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
F : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Deformation gradient.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
C : numpy.ndarray, shape (...,3,3)
|
2020-11-19 19:08:54 +05:30
|
|
|
|
Right Cauchy-Green deformation tensor.
|
2019-11-27 16:49:37 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2020-11-19 18:35:59 +05:30
|
|
|
|
return _np.matmul(_tensor.transpose(F),F)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def equivalent_strain_Mises(epsilon: _np.ndarray) -> _np.ndarray:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Calculate the Mises equivalent of a strain tensor.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
epsilon : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Symmetric strain tensor of which the von Mises equivalent is computed.
|
2019-10-19 16:40:46 +05:30
|
|
|
|
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
epsilon_vM : numpy.ndarray, shape (...)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Von Mises equivalent strain of epsilon.
|
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-18 03:26:22 +05:30
|
|
|
|
return _equivalent_Mises(epsilon,2.0/3.0)
|
2019-10-19 16:24:16 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def equivalent_stress_Mises(sigma: _np.ndarray) -> _np.ndarray:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Calculate the Mises equivalent of a stress tensor.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
sigma : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Symmetric stress tensor of which the von Mises equivalent is computed.
|
2019-10-19 16:40:46 +05:30
|
|
|
|
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
sigma_vM : numpy.ndarray, shape (...)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Von Mises equivalent stress of sigma.
|
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-18 03:26:22 +05:30
|
|
|
|
return _equivalent_Mises(sigma,3.0/2.0)
|
2019-10-19 16:24:16 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def maximum_shear(T_sym: _np.ndarray) -> _np.ndarray:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Calculate the maximum shear component of a symmetric tensor.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T_sym : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Symmetric tensor of which the maximum shear is computed.
|
2019-10-19 16:40:46 +05:30
|
|
|
|
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
gamma_max : numpy.ndarray, shape (...)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Maximum shear of T_sym.
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-19 18:35:59 +05:30
|
|
|
|
w = _tensor.eigenvalues(T_sym)
|
|
|
|
|
return (w[...,0] - w[...,2])*0.5
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2020-03-03 03:41:05 +05:30
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def rotation(T: _np.ndarray) -> _rotation.Rotation:
|
2020-02-15 18:26:15 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Calculate the rotational part of a tensor.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Tensor of which the rotational part is computed.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
R : damask.Rotation, shape (...)
|
2020-11-20 03:06:19 +05:30
|
|
|
|
Rotational part of the vector.
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
2020-02-15 18:26:15 +05:30
|
|
|
|
"""
|
2020-11-20 03:06:19 +05:30
|
|
|
|
return _rotation.Rotation.from_matrix(_polar_decomposition(T,'R')[0])
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def strain(F: _np.ndarray, t: str, m: float) -> _np.ndarray:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-16 05:42:23 +05:30
|
|
|
|
Calculate strain tensor (Seth–Hill family).
|
2020-02-15 18:40:16 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
F : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Deformation gradient.
|
2020-02-15 18:40:16 +05:30
|
|
|
|
t : {‘V’, ‘U’}
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Type of the polar decomposition, ‘V’ for left stretch tensor
|
2020-11-19 19:08:54 +05:30
|
|
|
|
and ‘U’ for right stretch tensor.
|
2020-02-15 18:40:16 +05:30
|
|
|
|
m : float
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Order of the strain.
|
2019-10-25 17:00:20 +05:30
|
|
|
|
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
epsilon : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Strain of F.
|
|
|
|
|
|
2021-05-07 23:12:23 +05:30
|
|
|
|
References
|
|
|
|
|
----------
|
|
|
|
|
https://en.wikipedia.org/wiki/Finite_strain_theory
|
|
|
|
|
https://de.wikipedia.org/wiki/Verzerrungstensor
|
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-02-15 18:40:16 +05:30
|
|
|
|
if t == 'V':
|
2020-11-18 03:26:22 +05:30
|
|
|
|
w,n = _np.linalg.eigh(deformation_Cauchy_Green_left(F))
|
2020-02-15 18:40:16 +05:30
|
|
|
|
elif t == 'U':
|
2020-11-18 03:26:22 +05:30
|
|
|
|
w,n = _np.linalg.eigh(deformation_Cauchy_Green_right(F))
|
2019-10-25 17:00:20 +05:30
|
|
|
|
|
2020-02-15 18:40:16 +05:30
|
|
|
|
if m > 0.0:
|
2020-11-16 11:42:37 +05:30
|
|
|
|
eps = 1.0/(2.0*abs(m)) * (+ _np.einsum('...j,...kj,...lj',w**m,n,n) - _np.eye(3))
|
2020-02-15 18:40:16 +05:30
|
|
|
|
elif m < 0.0:
|
2020-11-16 11:42:37 +05:30
|
|
|
|
eps = 1.0/(2.0*abs(m)) * (- _np.einsum('...j,...kj,...lj',w**m,n,n) + _np.eye(3))
|
2020-02-15 18:40:16 +05:30
|
|
|
|
else:
|
2020-11-16 11:42:37 +05:30
|
|
|
|
eps = _np.einsum('...j,...kj,...lj',0.5*_np.log(w),n,n)
|
2019-10-25 17:00:20 +05:30
|
|
|
|
|
2020-06-10 00:57:08 +05:30
|
|
|
|
return eps
|
2020-02-15 18:40:16 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def stress_Cauchy(P: _np.ndarray, F: _np.ndarray) -> _np.ndarray:
|
2020-11-19 18:35:59 +05:30
|
|
|
|
"""
|
|
|
|
|
Calculate the Cauchy stress (true stress).
|
|
|
|
|
|
|
|
|
|
Resulting tensor is symmetrized as the Cauchy stress needs to be symmetric.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
P : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
First Piola-Kirchhoff stress.
|
2021-12-06 12:08:40 +05:30
|
|
|
|
F : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Deformation gradient.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
sigma : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Cauchy stress.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return _tensor.symmetric(_np.einsum('...,...ij,...kj',1.0/_np.linalg.det(F),P,F))
|
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def stress_second_Piola_Kirchhoff(P: _np.ndarray, F: _np.ndarray) -> _np.ndarray:
|
2020-11-19 18:35:59 +05:30
|
|
|
|
"""
|
|
|
|
|
Calculate the second Piola-Kirchhoff stress.
|
|
|
|
|
|
|
|
|
|
Resulting tensor is symmetrized as the second Piola-Kirchhoff stress
|
|
|
|
|
needs to be symmetric.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
P : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
First Piola-Kirchhoff stress.
|
2021-12-06 12:08:40 +05:30
|
|
|
|
F : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Deformation gradient.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
S : numpy.ndarray, shape (...,3,3)
|
2020-11-19 18:35:59 +05:30
|
|
|
|
Second Piola-Kirchhoff stress.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return _tensor.symmetric(_np.einsum('...ij,...jk',_np.linalg.inv(F),P))
|
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def stretch_left(T: _np.ndarray) -> _np.ndarray:
|
2020-11-16 05:31:32 +05:30
|
|
|
|
"""
|
2020-11-19 19:08:54 +05:30
|
|
|
|
Calculate left stretch of a tensor.
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Tensor of which the left stretch is computed.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
V : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Left stretch tensor from Polar decomposition of T.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return _polar_decomposition(T,'V')[0]
|
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def stretch_right(T: _np.ndarray) -> _np.ndarray:
|
2020-11-16 05:31:32 +05:30
|
|
|
|
"""
|
2020-11-19 19:08:54 +05:30
|
|
|
|
Calculate right stretch of a tensor.
|
2020-11-16 05:31:32 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Tensor of which the right stretch is computed.
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
-------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
U : numpy.ndarray, shape (...,3,3)
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Left stretch tensor from Polar decomposition of T.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
return _polar_decomposition(T,'U')[0]
|
|
|
|
|
|
|
|
|
|
|
2022-01-12 18:48:38 +05:30
|
|
|
|
def _polar_decomposition(T: _np.ndarray, requested: _Sequence[str]) -> tuple:
|
2019-10-30 22:35:44 +05:30
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
Perform singular value decomposition.
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2019-10-30 22:35:44 +05:30
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Tensor of which the singular values are computed.
|
2019-11-22 00:16:05 +05:30
|
|
|
|
requested : iterable of str
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Requested outputs: ‘R’ for the rotation tensor,
|
|
|
|
|
‘V’ for left stretch tensor and ‘U’ for right stretch tensor.
|
2019-10-30 22:35:44 +05:30
|
|
|
|
|
|
|
|
|
"""
|
2020-11-16 05:31:32 +05:30
|
|
|
|
u, _, vh = _np.linalg.svd(T)
|
2020-11-16 11:42:37 +05:30
|
|
|
|
R = _np.einsum('...ij,...jk',u,vh)
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
output = []
|
2019-10-30 22:35:44 +05:30
|
|
|
|
if 'R' in requested:
|
2021-11-01 03:20:41 +05:30
|
|
|
|
output+=[R]
|
2019-10-30 22:35:44 +05:30
|
|
|
|
if 'V' in requested:
|
2021-11-01 03:20:41 +05:30
|
|
|
|
output+=[_np.einsum('...ij,...kj',T,R)]
|
2019-10-30 22:35:44 +05:30
|
|
|
|
if 'U' in requested:
|
2021-11-01 03:20:41 +05:30
|
|
|
|
output+=[_np.einsum('...ji,...jk',R,T)]
|
2020-02-15 18:26:15 +05:30
|
|
|
|
|
2020-11-19 15:09:41 +05:30
|
|
|
|
if len(output) == 0:
|
2020-11-20 03:06:19 +05:30
|
|
|
|
raise ValueError('output needs to be out of V, R, U')
|
2020-11-19 15:09:41 +05:30
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
return tuple(output)
|
2020-02-15 18:40:16 +05:30
|
|
|
|
|
|
|
|
|
|
2021-11-01 03:20:41 +05:30
|
|
|
|
def _equivalent_Mises(T_sym: _np.ndarray, s: float) -> _np.ndarray:
|
2020-02-15 18:40:16 +05:30
|
|
|
|
"""
|
2020-11-16 11:42:37 +05:30
|
|
|
|
Base equation for Mises equivalent of a stress or strain tensor.
|
2020-02-15 18:40:16 +05:30
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
2021-12-06 12:08:40 +05:30
|
|
|
|
T_sym : numpy.ndarray, shape (...,3,3)
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Symmetric tensor of which the von Mises equivalent is computed.
|
2020-02-15 18:40:16 +05:30
|
|
|
|
s : float
|
2020-03-15 02:23:48 +05:30
|
|
|
|
Scaling factor (2/3 for strain, 3/2 for stress).
|
2020-03-03 03:41:05 +05:30
|
|
|
|
|
2020-02-15 18:40:16 +05:30
|
|
|
|
"""
|
2020-11-19 19:08:54 +05:30
|
|
|
|
d = _tensor.deviatoric(T_sym)
|
2020-11-16 11:42:37 +05:30
|
|
|
|
return _np.sqrt(s*_np.sum(d**2.0,axis=(-1,-2)))
|