diff --git a/doc/content/bib/blackbear.bib b/doc/content/bib/blackbear.bib index 29c911b66..f5fa6ac07 100644 --- a/doc/content/bib/blackbear.bib +++ b/doc/content/bib/blackbear.bib @@ -244,3 +244,50 @@ @article{poyet2009temperature year={2009}, publisher={Elsevier} } + + @article{lee1998plastic, + title={Plastic-damage model for cyclic loading of concrete structures}, + author={Lee, Jeeho and Fenves, Gregory L}, + journal={Journal of engineering mechanics}, + volume={124}, + number={8}, + pages={892--900}, + year={1998}, + publisher={American Society of Civil Engineers} +} + +@book{lee1996theory, + title={Theory and implementation of plastic-damage model for concrete structures under cyclic and dynamic loading}, + author={Lee, Jeeho}, + year={1996}, + publisher={University of California, Berkeley} +} + +@article{wilkins2020method, + title = {A method for smoothing multiple yield functions}, + author = {Andy Wilkins and Benjamin W. Spencer and Amit Jain and Bora Gencturk}, + year = {2020}, +journal = {International Journal for Numerical Methods in Engineering}, + volume = {121}, + number = {3}, + pages = {434--449}, + doi = {10.1002/nme.6215} +} + +@article{lubliner1989plastic, + title={A plastic-damage model for concrete}, + author={Lubliner, Jacob and Oliver, Javier and Oller, Sand and Onate, EJIJos}, + journal={International Journal of solids and structures}, + volume={25}, + number={3}, + pages={299--326}, + year={1989}, + publisher={Elsevier} +} + +@article{krabbenhoft2002basic, + title={Basic computational plasticity}, + author={Krabbenh{\o}ft, KRISTIAN}, + journal={University of Denmark}, + year={2002} +} diff --git a/doc/content/media/Return_mapping_flow_chart.png b/doc/content/media/Return_mapping_flow_chart.png new file mode 100644 index 000000000..aebf37389 Binary files /dev/null and b/doc/content/media/Return_mapping_flow_chart.png differ diff --git a/doc/content/source/materials/ComputeMultipleInelasticDamageStress.md b/doc/content/source/materials/ComputeMultipleInelasticDamageStress.md new file mode 100644 index 000000000..eeaceb644 --- /dev/null +++ b/doc/content/source/materials/ComputeMultipleInelasticDamageStress.md @@ -0,0 +1,21 @@ +# Compute Multiple Inelastic Damage Stress + +!syntax description /Materials/ComputeMultipleInelasticDamageStress + +## Description + +`ComputeMultipleInelasticDamageStress` computes the damage stress. + +## Example Input Files + +The input settings for the inelastic material model is as follows + +!listing test/tests/damage_plasticity_model/uniaxial_test.i block=Materials/stress + +!syntax parameters /Materials/ComputeMultipleInelasticDamageStress + +!syntax inputs /Materials/ComputeMultipleInelasticDamageStress + +!syntax children /Materials/ComputeMultipleInelasticDamageStress + +!bibtex bibliography diff --git a/doc/content/source/materials/DamagePlasticityStressUpdate.md b/doc/content/source/materials/DamagePlasticityStressUpdate.md new file mode 100644 index 000000000..689cd4926 --- /dev/null +++ b/doc/content/source/materials/DamagePlasticityStressUpdate.md @@ -0,0 +1,346 @@ +# Damage Plasticity Model + +The [!cite](lee1996theory) model accounts for the independent damage in tension and compression. It also accounts for degradation of the elastic modulus of the concrete as the loading goes beyond yielding in either tension or compression. The model uses the incremental theory of plasticity and decomposes the total strain, $\boldsymbol{\varepsilon}$, into elastic strain, $\boldsymbol{\varepsilon}^{e}$, and plastic strain, $\boldsymbol{\varepsilon}^{p}$, as follows +\begin{equation} + \boldsymbol{\varepsilon} = \boldsymbol{\varepsilon}^{e} + \boldsymbol{\varepsilon}^{p} \label{eps_def} +\end{equation} +where bold symbol represents a vectoral or tensorial quantity. The relation between elastic strain and the stress, $\boldsymbol{\sigma}$, is given by +\begin{equation} + \boldsymbol{\varepsilon}^{e} = \boldsymbol{\mathfrak{E}}^{-1}:\boldsymbol{\sigma} \label{eps_e_def} +\end{equation} +where $\boldsymbol{\mathfrak{E}}$ is the elasticity tensor. Using Eqs. \eqref{eps_def}-\eqref{eps_e_def}, the relation between $\boldsymbol{\sigma}$ and $\boldsymbol{\varepsilon}^{p}$ is expressed as +\begin{equation} + \boldsymbol{\sigma} = \boldsymbol{\mathfrak{E}}:\left(\boldsymbol{\varepsilon} - \boldsymbol{\varepsilon}^{e}\right) +\end{equation} +Since the model considers the effect of damage in elastic stiffness, an effective stress, +$\boldsymbol{\sigma}^{e}$, is defined, where the stress for a given strain always corresponds to the +undamaged elastic stiffness of the material, $\boldsymbol{\mathfrak{E}}_{0}$ The relation between +$\boldsymbol{\sigma}^{e}$, $\boldsymbol{\varepsilon}$, and $\boldsymbol{\varepsilon}^{p}$ is given by +\begin{equation} + \boldsymbol{\sigma}^e = \boldsymbol{\mathfrak{E}}_0:\left(\boldsymbol{\varepsilon} - \boldsymbol{\varepsilon}^{e}\right) +\end{equation} +To consider the degradation of reinforced-concrete structures, an isotropic damage was +considered in concrete material. Hence, the relation between $\boldsymbol{\sigma}^e$ and $\boldsymbol{\sigma}$ can be established by +the isotropic scalar degradation damage variable, D, as follows +\begin{equation} + \boldsymbol{\sigma} = \left(1-D\right)\boldsymbol{\sigma}^e \label{sigma_def} +\end{equation} +\begin{equation} + \boldsymbol{\sigma} = \left(1-D\right)\boldsymbol{\mathfrak{E}}_0:\left(\boldsymbol{\varepsilon} - \boldsymbol{\varepsilon}^{e}\right)\label{sigma_def2} +\end{equation} +The Damage Plasticity Model has various attributes to define the mechanical behavior of concrete +in tension and compression such as the yield function, plastic potential, strength of material +in tension and compression, and hardening and softening of the yield surface. These attributes +are discussed in detail in the following sections. A method for the implementation of the +Damage Plasticity Model and for the estimation of crack width are also presented in the upcoming +sections. + +## Yield Function + +The yield function, $\mathfrak{F}$ is a function of $\boldsymbol{\sigma}$, the strength of the material in uniaxial tension, $f_t$, and the strength of the material in uniaxial compression, $f_c$. It was used to describe the admissible stress space. For this implementation, the yield function in stress space is defined as follows +\begin{equation} \label{yf} +\begin{gathered} + \mathfrak{F}\left(\boldsymbol{\sigma},f_t,f_c\right) = \frac{1}{1-\alpha} \\ + \left(\alpha I_1 + \sqrt{3J_2} + \beta\left(\boldsymbol{\kappa}\right)<{\hat{\boldsymbol{\sigma}}_{max}}>\right) - f_c\left(\boldsymbol{\kappa}\right) +\end{gathered} +\end{equation} +where $I_1$ and $J_2$ is first invariant of stress and second invariant of the deviatoric +component of the stress, respectively, $ =\frac{x+|x|}{2}$ is the Macaulay bracket function, ${\hat{\boldsymbol{\sigma}}_{max}}$ is algebraically maximum principal +stress, $\alpha = \frac{f_{b0}-f_{c0}}{2 f_{b0}-f_{c0}}$ is a parameter that relates +uniaxial, $f_{c0}$, and biaxial, $f_{b0}$, yield strength of concrete in compression, +$\beta\left(\boldsymbol{\kappa}\right)=\frac{f_c\left(\boldsymbol{\kappa}\right)}{f_t\left(\boldsymbol{\kappa}\right)}\left(\alpha-1\right)-\left(1+\alpha\right)$ is a parameter that +relates tensile, $f_t\left(\boldsymbol{\kappa}\right)$, and compressive, $f_c\left(\boldsymbol{\kappa}\right)$, yield strength which are +function of a vector of damage variable, $\boldsymbol{\kappa} = \{\kappa_t, \kappa_c\}$ and $\kappa_t$ +and $\kappa_c$ are the damage variables in tension and compression, respectively. + +The implementation first solves the given problem in the effective stress space and then transform the effective stress to stress space using Eq. \eqref{sigma_def2}. Thus, the yield strength of the concrete under uniaxial loading is expressed as effective yield strength as follows +\begin{equation} + f_t\left(\boldsymbol{\kappa}\right) = \left(1-D_t \left(\kappa_t\right)\right)f_{t}^{e}\left(\kappa_t\right) \label{ft} +\end{equation} +\begin{equation} + f_c\left(\boldsymbol{\kappa}\right) = \left(1-D_c \left(\kappa_c\right)\right)f_{c}^{e}\left(\kappa_c\right) \label{fc} +\end{equation} +where $f_{t}^{e}$ and $f_{c}^{e}$ are the yield strength of the concrete in tension and +compression, respectively and $D_t$ and $D_c$ are the degradation damage variables in +tension and compression, respectively such that $0\leq D_t$\textless 1 and $0\leq D_c$\textless 1. +The scalar degradation damage variable is expressed in terms of $D_t$ and $D_c$ as follows +\begin{equation} + D\left(\boldsymbol{\kappa}\right) = 1-\left(1-D_t\left(\kappa_t\right)\right)\left(1-D_c\left(\kappa_c\right)\right) \label{D} +\end{equation} +Hence, for uniaxial tension, $D=D_t$, while for uniaxial compression, $D=D_c$.The yield strength for multi-axial loading, i.e., Eqs. \eqref{ft}-\eqref{fc}, can be rewritten as +\begin{equation} + f_t\left(\boldsymbol{\kappa}\right) = \left(1-D\left(\boldsymbol{\kappa}\right)\right)f_{t}^{e}\left(\kappa_t\right) \label{ft_new} +\end{equation} +\begin{equation} + f_c\left(\boldsymbol{\kappa}\right) = \left(1-D\left(\boldsymbol{\kappa}\right)\right)f_{c}^{e}\left(\kappa_c\right) \label{fc_new} +\end{equation} +Similarly, the first invariant of $\boldsymbol{\sigma}^e$, $I_1^e$, and second invariant of the deviatoric component of $\boldsymbol{\sigma}^e$, $J_2^e$, can be rewritten in terms of $I_1$ and $J_2$ as follows +\begin{equation} + I_1^e = \left(1-D\left(\boldsymbol{\kappa}\right)\right)I_1 \label{I1e} +\end{equation} +\begin{equation} + J_2^e = \left(1-D\left(\boldsymbol{\kappa}\right)\right)^2J_2 \label{J2e} +\end{equation} +Since $D$ \textless 1, the maximum principal effective stress ${\hat{\boldsymbol{\sigma}}_{max}}^e$ is expressed in the terms of ${\hat{\boldsymbol{\sigma}}_{max}}$ as follows +\begin{equation} + {\hat{\boldsymbol{\sigma}}_{max}}^e = \left(1-D\left(\boldsymbol{\kappa}\right)\right){\hat{\boldsymbol{\sigma}}_{max}} \label{sig_max_e} +\end{equation} +Consequently, yield function $\left(\mathfrak{F}\left(\boldsymbol{\sigma},f_t,f_c\right)\right)$ is a homogenous +function, i.e., $x \mathfrak{F}\left(\boldsymbol{\sigma},f_t,f_c\right) = \mathfrak{F}\left(x \boldsymbol{\sigma},x f_t,x f_c\right)$ Hence, using Eqs. \eqref{ft_new}-\eqref{sig_max_e}, the yield function in the effective stress space was obtained by multiplying by a factor $\left(1-D\right)$ of both sides of Eq. \eqref{yf}, as follows +\begin{equation}\label{yf_e} +\begin{gathered} + \mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right) = \frac{1}{1-\alpha} \\ + \left(\alpha I_1^e + \sqrt{3J_2^e} + \beta\left(\boldsymbol{\kappa}\right)<{\hat{\boldsymbol{\sigma}}_{max}}^e>\right) - f_c^e\left(\boldsymbol{\kappa}\right) +\end{gathered} +\end{equation} + +## Plastic Potential + +It was found that for concrete, the Drucker-Prager flow rule describes the experimentally observed volumetric expansion of the material as opposed to the Von-Mises flow rule, which results in no volumetric expansion ([!cite](krabbenhoft2002basic)). Since all the equations are solved in the effective stress space, the plastic flow potential is also defined in the effective stress space ([!cite](lee1996theory)) as follows +\begin{equation} + \dot{\boldsymbol{\varepsilon}^p} = \dot{\gamma} \nabla_{\boldsymbol{\sigma}^e}\Phi\left(\boldsymbol{\sigma}^e\right) +\end{equation} +\begin{equation} \label{flowRule} + \Phi\left(\boldsymbol{\sigma}^e\right)=\alpha_p I_1^e+\|{s^e}\| +\end{equation} +where $\Phi$ is the plastic potential function, $s^e$ is the deviatoric component of the +$\boldsymbol{\sigma}^e$, and $\|\cdot\|$ is $L_2$ norm of $\alpha_p$ is a parameter that controls the +dilatancy of concrete, and $\dot{\gamma}$ is the plastic consistency parameter. + +## Strength Function + +Since the concrete shows strain-softening in tension and strain hardening and softening in compression, the concrete strength is expressed as a combination of two exponential functions as follows +\begin{equation} + f_N = f_{N0} \left(\left(1+a_N\right) e^{-b_N \varepsilon^p}- a_N e^{-2b_N \varepsilon^p}\right) \label{fN} +\end{equation} +where $f_{N0}$ is the initial yield stress of the material, $N = t$, for the uniaxial +tension, $N = c$, for uniaxial compression, $a_N$ and $b_N$, are the material constants +that describe the softening and hardening behavior of the concrete. Similarly, the +degradation of the elastic modulus is also expressed as another exponential function as +follows +\begin{equation} + D_N = 1 - e^{-d_N \varepsilon^p} \label{DN} +\end{equation} +where $d_N$ is a constant that determine the rate of degradation of $\boldsymbol{\mathfrak{E}}$ with the +increase in plastic strain. The strength of the material in the effective stress space was +obtained using Eqs. \eqref{ft_new}-\eqref{fc_new}, and \eqref{fN}-\eqref{DN}, as follows +\begin{equation}\label{fNe} + f_N^e = f_{N0} \left(\left(1+a_N\right) \left(e^{-b_N \varepsilon^p}\right)^{1-\frac{d_N}{b_N}}- + a_N \left(e^{-b_N \varepsilon^p}\right)^{2-\frac{d_N}{b_N}}\right) +\end{equation} +The damage variable, $\kappa_N$ is defined as +\begin{equation} + \kappa_N = \frac{1}{g_N}\int_0^{\varepsilon^p} {f_N\left(\varepsilon^p\right)d\varepsilon^p} \label{kN_def} +\end{equation} +where $g_N$ $\left(=\int_0^\infty {f_N\left(\varepsilon^p\right)d\varepsilon^p}=\frac {f_{N0}}{b_N}\left(1+\frac{a_N}{2}\right)\right)$ +is the fracture energy density during the process of cracking, which is derived from the +fracture energy, $G_N$, which is a material property. The relation between $G_N$ and $g_N$ +is expressed as follows +\begin{equation} + g_N = \frac{G_N}{l_N} \label{GN_def} +\end{equation} +where $l_N$ is characteristic length or the size of the deformation localization zone. +Thus, the plastic strain can be presented in terms of damage variable as follows +\begin{equation} + \varepsilon^p = \frac{1}{b_N} \log{\frac{\sqrt{\Phi_N}}{a_N}} \label{eps_p} +\end{equation} +where $\Phi_N = 1 + a_N \left(2+a_N \right)\kappa_N$. Using Eqs. \eqref{fN} and \eqref{eps_p}, the +strength of the concrete can be expressed in terms of the damage variable as follows +\begin{equation} + f_N = f_{N0} \frac{1+a_N-\sqrt{\Phi_N\left(\kappa_N\right)}}{a_N}\sqrt{\Phi_N\left(\kappa_N\right)} \label{fN_new} +\end{equation} +Thus, the strength of the material and degradation damage variable in the effective stress space can be written as + +\begin{equation} + f_N^e = f_{N0} \left(\frac{1+a_N-\sqrt{\Phi_N\left(\kappa_N\right)}}{a_N}\right)^{1-\frac{d_N}{b_N}} \sqrt{\Phi_N\left(\kappa_N\right)} \label{fNe_new} +\end{equation} +\begin{equation} + D_N = 1- \left(\frac{1+a_N-\sqrt{\Phi_N\left(\kappa_N\right)}}{a_N}\right)^{\frac{d_N}{b_N}} \label{DN_new} +\end{equation} +where $a_N$, $b_N$,and $d_N$ are the modeling parameters, which are evaluated from the +material properties. Since the maximum compressive strength of concrete, $f_{cm}$, was used +as a material property, $f_{cm}$ was obtained in terms of $a_c$ by finding maximum value of +compressive strength in Eq. \eqref{fNe} as follows +\begin{equation} + f_{cm} = \frac{f_{c0}\left(1+a_c\right)^2}{4a_c} \label{fcm} +\end{equation} +Thus, $a_c$ can be expressed as follows +\begin{equation} + a_c = 2\frac{f_{cm}}{f_{c0}}-1+2\sqrt{\left(\frac{f_{cm}}{f_{c0}}\right)^2-\frac{f_{cm}}{f_{c0}}} \label{ac} +\end{equation} +Similarly, if $G_c$ and $l_c$ are known then $b_c$ can be expressed in term of known quantities as follows +\begin{equation} + b_c = \frac{f_{c0}}{\frac{G_c}{l_c}}\left(1+\frac{a_c}{2}\right) \label{bc} +\end{equation} +A relationship between $a_t$ and $b_t$ is written as follows +\begin{equation} + b_t = \frac{f_{t0}}{\frac{G_t}{l_t}}\left(1+\frac{a_t}{2}\right) \label{bt} +\end{equation} +[!cite](lubliner1989plastic) suggested that if the slope of $\sigma$ versus $\varepsilon^p$ curve is +known at $\varepsilon^p=0$, then another relationship between $a_t$ and $b_t$ will be obtained +as follows +\begin{equation} + \left(\frac{d\sigma}{d\varepsilon^p}\right)_{\varepsilon^p=0} = f_{t0}b_t\left(a_t-1\right) \label{slope} +\end{equation} +Thus, $a_t$ was obtained using Eqs. \eqref{bt}-\eqref{slope} as follows +\begin{equation} + a_t = \sqrt{\frac{9}{4}+\frac{2\frac{G_t}{l_t} \left(\frac{d\sigma}{d\varepsilon^p}\right)_{\varepsilon^p=0}}{f_{t0}^2}}\label{at} +\end{equation} +The minimum slope of the $\sigma$ versus $\varepsilon^p$ curve is +$\left(\left(\frac{d\sigma}{d\varepsilon^p}\right)_{\varepsilon^p=0}\right)_{min}= +-\frac{9}{8}\frac{f_{t0}^2}{\frac{G_t}{l_t}}$, which is a function of the characteristic length in tension. +Therefore, a mesh independent slope parameter $\omega\in\left(0,1\right)$, is defined such that +\begin{equation} + \left(\frac{d\boldsymbol{\sigma}}{d\varepsilon^p}\right)_{\varepsilon^p=0} = \omega \left(\left(\frac{d\sigma}{d\varepsilon^p}\right)_{\varepsilon^p=0}\right)_{min} \label{slope_new} +\end{equation} +Using Eqs. \eqref{at}-\eqref{slope_new}, $a_t$ is rewritten as follows +\begin{equation} + a_t = \frac{3}{2}\sqrt{1-\omega}-\frac{1}{2}\label{at_new} +\end{equation} +The ratio of $\frac{d_c}{b_c}$ was obtained by specifying degradation values for uniaxial +compression case from experiments. If the degradation in the elastic modulus is known, +denoted as $\widetilde{D}_c$, when the concrete is unloaded from $\sigma =f_{cm}$, then $\frac{d_c}{b_c}$ will be obtained using the following relation +\begin{equation} + \widetilde{D}_c = 1 - \left(\frac{1+a_c}{2a_c}\right)^{\frac{d_c}{b_c}} \label{Dc_fcm} +\end{equation} +\begin{equation} + \frac{d_c}{b_c} = \frac{\log\left(1-\widetilde{D}_c\right)}{\log\left(\frac{1+a_c}{2a_c}\right)} \label{dcbc_fcm} +\end{equation} +Similarly, if degradation in the elastic modulus is known, denoted as $\widetilde{D}_t$, when the material is unloaded from $\sigma=\frac {f_{t0}}{2}$, on softening branch, then $\frac{d_t}{b_t}$ will be obtained using the following relation +\begin{equation} + \widetilde{D}_t = 1 - \left(\frac{1+a_t-\sqrt{1+a_t^2}}{2a_t}\right)^{\frac{d_t}{b_t}} \label{Dt_ft0_2} +\end{equation} +\begin{equation} + \frac{d_t}{b_t} = \frac{\log\left(1-\widetilde{D}_t\right)}{\log\left(\frac{1+a_t-\sqrt{1+a_t^2}}{2a_t}\right)} \label{Dt_ft0} +\end{equation} +Thus, material modeling parameters $a_N$,$b_N$, and $d_N$ were obtained, which were used in +defining the strength of concrete in both tension and compression as given in Eq. +\eqref{fNe_new}. These parameters are also used to define the degradation damage variable in +both tension and compression as indicated in Eq. \eqref{DN_new}. + +## Hardening Potential + +The vector of two damage variables, $\boldsymbol{\kappa}=\{\kappa_t, \kappa_c\}$, was used in the implementation as the state variable to store the state of damage in tension and compression, separately. The evolution of these damage variables is defined in terms of the hardening potential, $H$, as +\begin{equation} + \dot{\boldsymbol{\kappa}} = \dot{\gamma} H\left(\boldsymbol{\sigma}^e, \boldsymbol{\kappa}\right)\label{kappa} +\end{equation} +The evolution of the damage variable is expressed in terms of the evolution of $\boldsymbol{\varepsilon}^p$ as follows +\begin{equation} + \dot{\boldsymbol{\kappa}} = \frac{1}{g_N}f_N^e\left(\kappa_N\right)\dot{\boldsymbol{\varepsilon}^p} \label{kappa_ep} +\end{equation} +where $g_N$ is dissipated energy density during the process of cracking. The scalar $\dot{\boldsymbol{\varepsilon}^p}$, is extended to multi-dimensional case as follows +\begin{equation} + \dot{\boldsymbol{\varepsilon}^p} = \delta_{tN} r\left(\hat{\boldsymbol{\sigma}^e}\right)\dot{\varepsilon}^{p}_{max} + \delta_{cN} \left(1-r\left(\hat{\boldsymbol{\sigma}^e}\right)\right)\dot{\varepsilon}^{p}_{min} \label{ep_dot} +\end{equation} +where $\delta_{ij}$ is the Dirac delta function and $\hat{\boldsymbol{\sigma}^e}$ are eigenvalues of the $\boldsymbol{\sigma}^e$, +\begin{equation}\label{r_sige} + r\left(\hat{\boldsymbol{\sigma}^e}\right) = + \begin{cases} + 0,& \text{if } \boldsymbol{\sigma}^e = \boldsymbol{0}\\ + \frac{\sum_{i=1}^3<\sigma^e_i>}{\sum_{i=1}^3|\sigma^e_i|}, & \text{otherwise} + \end{cases} +\end{equation} +$\dot{\varepsilon}^{p}_{max}$ and $\dot{\varepsilon}^{p}_{min}$ +are the maximum and minimum principal plastic strain, respectively. From Eqs. \eqref{kappa_ep} - \eqref{r_sige}, the evolution of $\boldsymbol{\kappa}$ was obtained as +\begin{equation} + \dot{\boldsymbol{\kappa}} = \boldsymbol{h}\left(\hat{\boldsymbol{\sigma}^e}\right):\dot{\boldsymbol{\varepsilon}}^{\hat{p}} \label{kappa_h_ep} +\end{equation} +where +\begin{equation}\label{h} + \boldsymbol{h}\left(\hat{\boldsymbol{\sigma}^e}\right)= + \begin{bmatrix} + \frac{r\left(\hat{\boldsymbol{\sigma}^e}\right)}{g_t}f_t^e\left(\kappa_t\right)&0&0\\ + 0&1&0\\ + 0&0&\frac{1-r\left(\hat{\boldsymbol{\sigma}^e}\right)}{g_c}f_c^e\left(\kappa_c\right)\\ + \end{bmatrix} +\end{equation} +and ‘:’ represents products of two matrices. Hence, $H\left(\boldsymbol{\sigma}^e,\boldsymbol{\kappa}\right)$ in Eq. \eqref{kappa} was obtained as follows +\begin{equation} + H\left(\boldsymbol{\sigma}^e, \boldsymbol{\kappa}\right) = \boldsymbol{h}\cdot \nabla_{\hat{\boldsymbol{\sigma}^e}}\Phi\left(\hat{\boldsymbol{\sigma}^e}\right) \label{H_def} +\end{equation} +where ‘$\cdot$’ represents the dot product of a matrix and a vector, and $\nabla_{\hat{\boldsymbol{\sigma}^e}}$ is the gradient with respect to principal effective stress components, $\hat{\boldsymbol{\sigma}^e}$. Thus, the hardening potential that governs the evolution of damage variables is expressed in terms of effective stress space. + +## Return Mapping Algorithm + +Return mapping algorithm is summarized in a flowchart [flowChart]. For given strain increment and previous state of stress, trial effective stress is obtained by elastic increment in the elastic predictor step according to +\begin{equation} + \boldsymbol{\sigma}_{n+1}^{e^{tr}} = \boldsymbol{\mathfrak{E}}_0:\left(\varepsilon_{n+1}-\varepsilon_n^p\right) \label{predictor_step} +\end{equation} +If the state of trial effective stress lies inside the admissible domain, i.e, $\mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right) < 0$, the trial effective stress is considered as an admissible effective stress and old damage variables are supplied for the next step. +If the state of trial effective stress is outside admissible domain, i.e, $\mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right) > 0$, the effective stress is obtained using the plastic corrector step + +!media media/Return_mapping_flow_chart.png + id=flowChart + style=width:30%; + caption=Flow chart for return mapping algorithm. + +\begin{equation} +\boldsymbol{\sigma}_{n+1}^e = \boldsymbol{\sigma}_{n+1}^{e^{tr}}-\boldsymbol{\mathfrak{E}}_0:\varepsilon_n^p \label{plasticCorrector} +\end{equation} +Once admissible effective stress is obtained, degradation corrector step is utilized to account for the stiffness degradation on the state of stress according to +\begin{equation} +\boldsymbol{\sigma}_{n+1} = \left(1-D_{n}\right)\boldsymbol{\sigma}_{n+1}^{e} \label{degradation_corrector} +\end{equation} + + + +During the plastic corrector step, the returned effective stress should satisfy the Kuhn-Tucker conditions on the $\mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right)$ and $\dot{\gamma}$, according to +\begin{equation} + \begin{split} \label{khunTuckerConditions} + \dot{\gamma} > 0 \\ + \dot{\gamma}\mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right) = 0 \\ + \mathfrak{F}\left(\boldsymbol{\sigma}^e,f_t^e,f_c^e\right) = 0 + \end{split} +\end{equation} +As per flow rule in Eq. \eqref{flowRule}, the plastic corrector step, i.e., Eq. \eqref{plasticCorrector} can be rewritten as +\begin{equation} +\boldsymbol{\sigma^e}_{n+1} = \boldsymbol{\sigma}_{n+1}^{e^{tr}}-\dot{\gamma}\left(2G\frac{\boldsymbol{s}_{n+1}^e}{\|\boldsymbol{s}_{n+1}^e\|} + 3K\alpha_p\boldsymbol{I}\right) \label{returnMap1} +\end{equation} +where $G$ is shear modulus and $K$ is bulk modulus. After separating the volumetric and deviatoric components from Eq. \eqref{returnMap1} following relations can be obtained +\begin{equation} + I_{1|n+1} = I_{1|n+1}^{e^{tr}} - 9K\alpha \alpha_p \dot{\gamma} \label{stressRelation1} +\end{equation} +\begin{equation}\label{stressRelation2} + \begin{gathered} + \frac{\boldsymbol{s}_{n+1}^e}{\|\boldsymbol{s}_{n+1}^e\|} = \frac{\boldsymbol{s}_{{n+1}}^{e^{tr}}}{\|\boldsymbol{s}_{n+1}^{e^{tr}}\|} \\ + {\|\boldsymbol{s}^{e}_{n+1}\|} = {\|\boldsymbol{s}_{n+1}^{e^{tr}}\|} - 2G\dot{\gamma} + \end{gathered} +\end{equation} +Using Eqs. \eqref{stressRelation1} and \eqref{stressRelation2}, Eq. \eqref{returnMap1} can be written as +\begin{equation} + \boldsymbol{\sigma}_{n+1}^e = \boldsymbol{\sigma}_{n+1}^{e^{tr}}-\dot{\gamma}\left(2G\frac{\boldsymbol{s}^{e^{tr}}_{n+1}}{\|\boldsymbol{s}_{{n+1}}^{e^{tr}} \|}+ 3K\alpha_p\boldsymbol{I}\right) \label{returnMap2} +\end{equation} +In case of plastic deformation, the returned state of stress should lie on the yield surface as per Kuhn-Tucker conditions (Eq. \eqref{khunTuckerConditions}, therefore $\mathfrak{F}\left(\boldsymbol{\sigma}_{n+1}^e,f_t^e,f_c^e\right) = 0$, i.e., +\begin{equation} \label{yfnext} + \begin{gathered} + \alpha I_{1|n+1}^e + \sqrt{3J_{2|n+1}^e} + \beta\left(\boldsymbol{\kappa}\right)<\hat{\boldsymbol{\sigma}}^e_{n+1|max}> \\ - \left(1-\alpha\right)f_c^e\left(\boldsymbol{\kappa}\right) = 0 + \end{gathered} +\end{equation} +Using Eq. \eqref{stressRelation1}, \eqref{stressRelation2}, and \eqref{returnMap2}, Eq. \eqref{yfnext} can be written as +\begin{equation} \label{yfzero} + \begin{gathered} + \alpha\left(I_{1|n+1}^{e^{tr}} - 9K\alpha \alpha_p \dot{\gamma}\right) + + \left(\sqrt{\frac{3}{2}}\|\boldsymbol{s}_{{n+1}}^{e^{tr}}\| - \sqrt{6}G\dot{\gamma}\right)\\+ + \beta\left(\boldsymbol{\kappa}\right)<\hat{\boldsymbol{\sigma}}^e_{n+1|max}> - \left(1-\alpha\right)f_c^e\left(\boldsymbol{\kappa}\right) = 0 + \end{gathered} +\end{equation} +Thus, the plastic multiplier can be by solving Eq. \eqref{yfzero} as +\begin{equation}\label{gammaDef} + \dot{\gamma} = + \begin{cases} + \frac{\alpha I_{1|n+1}^{e^{tr}}+\sqrt{\frac{3}{2}}\|\boldsymbol{s}_{{n+1}}^{e^{tr}}\|-\left(1-\alpha\right)f_c^e\left(\boldsymbol{\kappa}\right)} + {9K \alpha_p + \sqrt{6}G}, & \text{if $\sigma_{m|n+1}^e < 0$}\\ + \frac{\alpha I_{1|n+1}^{e^{tr}}+\sqrt{\frac{3}{2}}\|\boldsymbol{s}_{{n+1}}^{e^{tr}}\|+\beta\left(\boldsymbol{\kappa}\right) \sigma_{m|n+1}^{e^{tr}}-\left(1-\alpha\right)f_c^e\left(\boldsymbol{\kappa}\right)} + {9K \alpha_p + \sqrt{6}G + \beta\left(\boldsymbol{\kappa}\right)\left(2G\frac{s^{e^{tr}}_{m|n+1}}{\|\boldsymbol{s}_{{n+1}}^{e^{tr}} \|}+ 3K\alpha_p\right)}, & \text{otherwise}. + \end{cases} +\end{equation} +where $\sigma_{m|n+1}^e$, $\sigma_{m|n+1}^{e^{tr}}$, and $s^{e^{tr}}_{m|n+1}$ are the $m^{th}$ component of the $\hat{\boldsymbol{\sigma}}_{n+1}^e$, $\boldsymbol{\sigma}_{n+1}^{e^{tr}}$, and $\boldsymbol{s}^{e^{tr}}_{n+1}$, respectively, which corresponds to maximum principal effective stress in $\left(n+1\right)^{th}$ step. Eq. \eqref{gammaDef} is solved iteratively. + + +!syntax parameters /Materials/DamagePlasticityStressUpdate + +!syntax inputs /Materials/DamagePlasticityStressUpdate + +!syntax children /Materials/DamagePlasticityStressUpdate + +!bibtex bibliography diff --git a/include/materials/ComputeMultipleInelasticDamageStress.h b/include/materials/ComputeMultipleInelasticDamageStress.h new file mode 100644 index 000000000..a569e8b7d --- /dev/null +++ b/include/materials/ComputeMultipleInelasticDamageStress.h @@ -0,0 +1,42 @@ +/****************************************************************/ +/* DO NOT MODIFY THIS HEADER */ +/* BlackBear */ +/* */ +/* (c) 2017 Battelle Energy Alliance, LLC */ +/* ALL RIGHTS RESERVED */ +/* */ +/* Prepared by Battelle Energy Alliance, LLC */ +/* Under Contract No. DE-AC07-05ID14517 */ +/* With the U. S. Department of Energy */ +/* */ +/* See COPYRIGHT for full restrictions */ +/****************************************************************/ + +#pragma once + +#include "ComputeMultipleInelasticStress.h" + +class ComputeMultipleInelasticDamageStress : public ComputeMultipleInelasticStress +{ +public: + static InputParameters validParams(); + ComputeMultipleInelasticDamageStress(const InputParameters & parameters); + +protected: + /// damage parameter for DamagePlasticityStressUpdate model + const MaterialProperty & _D; + const MaterialProperty & _D_old; + const MaterialProperty & _D_older; + + virtual void computeQpJacobianMult() override; + + virtual void computeAdmissibleState(unsigned model_number, + RankTwoTensor & elastic_strain_increment, + RankTwoTensor & inelastic_strain_increment, + RankFourTensor & consistent_tangent_operator) override; + + virtual void + updateQpStateSingleModel(unsigned model_number, + RankTwoTensor & elastic_strain_increment, + RankTwoTensor & combined_inelastic_strain_increment) override; +}; diff --git a/include/materials/DamagePlasticityStressUpdate.h b/include/materials/DamagePlasticityStressUpdate.h new file mode 100644 index 000000000..51b958a92 --- /dev/null +++ b/include/materials/DamagePlasticityStressUpdate.h @@ -0,0 +1,184 @@ +/****************************************************************/ +/* DO NOT MODIFY THIS HEADER */ +/* BlackBear */ +/* */ +/* (c) 2017 Battelle Energy Alliance, LLC */ +/* ALL RIGHTS RESERVED */ +/* */ +/* Prepared by Battelle Energy Alliance, LLC */ +/* Under Contract No. DE-AC07-05ID14517 */ +/* With the U. S. Department of Energy */ +/* */ +/* See COPYRIGHT for full restrictions */ +/****************************************************************/ + +#pragma once + +#include "MultiParameterPlasticityStressUpdate.h" + +class DamagePlasticityStressUpdate : public MultiParameterPlasticityStressUpdate +{ +public: + static InputParameters validParams(); + DamagePlasticityStressUpdate(const InputParameters & parameters); + + /** + * Does the model require the elasticity tensor to be isotropic? + */ + bool requiresIsotropicTensor() override { return true; } + +protected: + virtual void initQpStatefulProperties() override; + virtual void finalizeReturnProcess(const RankTwoTensor & rotation_increment) override; + +private: + const Real _f_tol; + const Real _alfa; + const Real _alfa_p; + const Real _s0; + + const Real _Chi; + const Real _Dt; + const Real _ft; + const Real _FEt; + + const Real _fyc; + const Real _Dc; + const Real _fc; + const Real _FEc; + + const Real _at; + const Real _ac; + const Real _zt; + const Real _zc; + const Real _dPhit; + const Real _dPhic; + const Real _sqrtPhit_max; + const Real _sqrtPhic_max; + const Real _dt_bt; + const Real _dc_bc; + const Real _ft0; + const Real _fc0; + const Real _small_smoother2; + + const Real _sqrt3; + + /// Whether to provide an estimate of the returned stress, based on perfect plasticity + const bool _perfect_guess; + + /// Eigenvectors of the trial stress as a RankTwoTensor, in order to rotate the returned stress back to stress space + RankTwoTensor _eigvecs; + + MaterialProperty & _max_principal; + MaterialProperty & _min_principal; + MaterialProperty & _intnl0; + MaterialProperty & _intnl1; + MaterialProperty & _ele_len; + MaterialProperty & _gt; + MaterialProperty & _gc; + + MaterialProperty & _tD; + MaterialProperty & _cD; + MaterialProperty & _D; + MaterialProperty & _min_ep; + MaterialProperty & _mid_ep; + MaterialProperty & _max_ep; + MaterialProperty & _sigma0; + MaterialProperty & _sigma1; + MaterialProperty & _sigma2; + + Real ft(const std::vector & intnl) const; /// tensile strength + Real dft(const std::vector & intnl) const; /// d(tensile strength)/d(intnl) + Real fc(const std::vector & intnl) const; /// compressive strength + Real dfc(const std::vector & intnl) const; /// d(compressive strength)/d(intnl) + Real beta(const std::vector & intnl) const; + Real dbeta0(const std::vector & intnl) const; + Real dbeta1(const std::vector & intnl) const; + void weighfac(const std::vector & stress_params, Real & wf) const; /// weight factor + void dweighfac(const std::vector & stress_params, + Real & wf, + std::vector & dwf) const; /// d(weight factor)/d(stress) + Real damageVar(const std::vector & stress_params, const std::vector & intnl) const; + + void computeStressParams(const RankTwoTensor & stress, + std::vector & stress_params) const override; + + std::vector dstress_param_dstress(const RankTwoTensor & stress) const override; + + std::vector d2stress_param_dstress(const RankTwoTensor & stress) const override; + + void setEffectiveElasticity(const RankFourTensor & Eijkl) override; + + virtual void preReturnMapV(const std::vector & trial_stress_params, + const RankTwoTensor & stress_trial, + const std::vector & intnl_old, + const std::vector & yf, + const RankFourTensor & Eijkl) override; + + virtual void setStressAfterReturnV(const RankTwoTensor & stress_trial, + const std::vector & stress_params, + Real gaE, + const std::vector & intnl, + const yieldAndFlow & smoothed_q, + const RankFourTensor & Eijkl, + RankTwoTensor & stress) const override; + + void yieldFunctionValuesV(const std::vector & stress_params, + const std::vector & intnl, + std::vector & yf) const override; + + void computeAllQV(const std::vector & stress_params, + const std::vector & intnl, + std::vector & all_q) const override; + + virtual void flowPotential(const std::vector & stress_params, + const std::vector & intnl, + std::vector & r) const; + + virtual void dflowPotential_dstress(const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dr_dstress) const; + + virtual void dflowPotential_dintnl(const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dr_dintnl) const; + + virtual void hardPotential(const std::vector & stress_params, + const std::vector & intnl, + std::vector & h) const; + + virtual void dhardPotential_dstress(const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dh_dsig) const; + + virtual void dhardPotential_dintnl(const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dh_dintnl) const; + + void initialiseVarsV(const std::vector & trial_stress_params, + const std::vector & intnl_old, + std::vector & stress_params, + Real & gaE, + std::vector & intnl) const; + + void setIntnlValuesV(const std::vector & trial_stress_params, + const std::vector & current_stress_params, + const std::vector & intnl_old, + std::vector & intnl) const override; + + void setIntnlDerivativesV(const std::vector & trial_stress_params, + const std::vector & current_stress_params, + const std::vector & intnl, + std::vector> & dintnl) const override; + + virtual void consistentTangentOperatorV(const RankTwoTensor & stress_trial, + const std::vector & trial_stress_params, + const RankTwoTensor & stress, + const std::vector & stress_params, + Real gaE, + const yieldAndFlow & smoothed_q, + const RankFourTensor & Eijkl, + bool compute_full_tangent_operator, + const std::vector> & dvar_dtrial, + RankFourTensor & cto) override; +}; diff --git a/src/materials/ComputeMultipleInelasticDamageStress.C b/src/materials/ComputeMultipleInelasticDamageStress.C new file mode 100644 index 000000000..4f92615e1 --- /dev/null +++ b/src/materials/ComputeMultipleInelasticDamageStress.C @@ -0,0 +1,73 @@ +/****************************************************************/ +/* DO NOT MODIFY THIS HEADER */ +/* BlackBear */ +/* */ +/* (c) 2017 Battelle Energy Alliance, LLC */ +/* ALL RIGHTS RESERVED */ +/* */ +/* Prepared by Battelle Energy Alliance, LLC */ +/* Under Contract No. DE-AC07-05ID14517 */ +/* With the U. S. Department of Energy */ +/* */ +/* See COPYRIGHT for full restrictions */ +/****************************************************************/ + +#include "ComputeMultipleInelasticDamageStress.h" +#include "StressUpdateBase.h" + +registerMooseObject("BlackBearApp", ComputeMultipleInelasticDamageStress); + +InputParameters +ComputeMultipleInelasticDamageStress::validParams() +{ + InputParameters params = ComputeMultipleInelasticStress::validParams(); + return params; +} + +ComputeMultipleInelasticDamageStress::ComputeMultipleInelasticDamageStress( + const InputParameters & parameters) + : ComputeMultipleInelasticStress(parameters), + _D(getMaterialProperty("elemental_damage_variable")), + _D_old(getMaterialPropertyOld("elemental_damage_variable")), + _D_older(getMaterialPropertyOlder("elemental_damage_variable")) +{ +} + +void +ComputeMultipleInelasticDamageStress::computeQpJacobianMult() +{ + ComputeMultipleInelasticStress::computeQpJacobianMult(); + _Jacobian_mult[_qp] = (1.0 - _D_older[_qp]) * _Jacobian_mult[_qp]; + // _Jacobian_mult[_qp] = (1.0 - _D[_qp]) * _Jacobian_mult[_qp]; +} + +void +ComputeMultipleInelasticDamageStress::updateQpStateSingleModel( + unsigned model_number, + RankTwoTensor & elastic_strain_increment, + RankTwoTensor & combined_inelastic_strain_increment) +{ + ComputeMultipleInelasticStress::updateQpStateSingleModel( + model_number, elastic_strain_increment, combined_inelastic_strain_increment); + _Jacobian_mult[_qp] = (1.0 - _D_older[_qp]) * _Jacobian_mult[_qp]; +} + +void +ComputeMultipleInelasticDamageStress::computeAdmissibleState( + unsigned model_number, + RankTwoTensor & elastic_strain_increment, + RankTwoTensor & inelastic_strain_increment, + RankFourTensor & consistent_tangent_operator) +{ + _models[model_number]->updateState(elastic_strain_increment, + inelastic_strain_increment, + _rotation_increment[_qp], + _stress[_qp], + _stress_old[_qp] / (1.0 - _D_older[_qp]), + // _stress_old[_qp] / (1.0 - _D[_qp]), + _elasticity_tensor[_qp], + _elastic_strain_old[_qp], + _tangent_operator_type == TangentOperatorEnum::nonlinear, + consistent_tangent_operator); + _stress[_qp] *= (1.0 - _D_older[_qp]); +} diff --git a/src/materials/DamagePlasticityStressUpdate.C b/src/materials/DamagePlasticityStressUpdate.C new file mode 100644 index 000000000..8f38c945b --- /dev/null +++ b/src/materials/DamagePlasticityStressUpdate.C @@ -0,0 +1,682 @@ +/****************************************************************/ +/* DO NOT MODIFY THIS HEADER */ +/* BlackBear */ +/* */ +/* (c) 2017 Battelle Energy Alliance, LLC */ +/* ALL RIGHTS RESERVED */ +/* */ +/* Prepared by Battelle Energy Alliance, LLC */ +/* Under Contract No. DE-AC07-05ID14517 */ +/* With the U. S. Department of Energy */ +/* */ +/* See COPYRIGHT for full restrictions */ +/****************************************************************/ +#include "DamagePlasticityStressUpdate.h" +#include "libmesh/utility.h" + +registerMooseObject("BlackBearApp", DamagePlasticityStressUpdate); + +InputParameters +DamagePlasticityStressUpdate::validParams() +{ + InputParameters params = MultiParameterPlasticityStressUpdate::validParams(); + params.addParam( + "yield_function_tolerance", + "If the yield function is less than this amount, the (stress, internal parameters) are " + "deemed admissible. A std::vector of tolerances must be entered for the multi-surface case"); + params.addRangeCheckedParam("factor_relating_biaxial_unixial_cmp_str", + 0.1, + "factor_relating_biaxial_unixial_cmp_str < 0.5 & " + "factor_relating_biaxial_unixial_cmp_str >= 0", + "Material parameter that relate biaxial and uniaxial " + "compressive strength, i.e., \alfa = (fb0-fc0)/(2*fb0-fc0)"); + params.addRequiredParam("factor_controlling_dilatancy", + "controls the dilation of concrete"); + params.addRangeCheckedParam("stiff_recovery_factor", + 0., + "stiff_recovery_factor <= 1. & stiff_recovery_factor >= 0", + "stiffness recovery factor"); + params.addRangeCheckedParam( + "ft_ep_slope_factor_at_zero_ep", + "ft_ep_slope_factor_at_zero_ep <= 1 & ft_ep_slope_factor_at_zero_ep >= 0", + "slope of ft vs plastic strain curve at zero plastic strain"); + params.addRequiredParam( + "tensile_damage_at_half_tensile_strength", + "fraction of the elastic recovery slope in tension at 0.5*ft0 after yielding"); + params.addRangeCheckedParam("yield_strength_in_tension", + "yield_strength_in_tension >= 0", + "Tensile yield strength of concrete"); + params.addRangeCheckedParam("fracture_energy_in_tension", + "fracture_energy_in_tension >= 0", + "Fracture energy of concrete in uniaxial tension"); + + params.addRangeCheckedParam("yield_strength_in_compression", + "yield_strength_in_compression >= 0", + "Absolute yield compressice strength"); + params.addRequiredParam("compressive_damage_at_max_compressive_strength", + "damage at maximum compressive strength"); + params.addRequiredParam("maximum_strength_in_compression", + "Absolute maximum compressive strength"); + params.addRangeCheckedParam("fracture_energy_in_compression", + "fracture_energy_in_compression >= 0", + "Fracture energy of concrete in uniaxial compression"); + + params.addRequiredRangeCheckedParam( + "tip_smoother", + "tip_smoother>=0", + "Smoothing parameter: the cone vertex at mean = cohesion*cot(friction_angle), will be " + "smoothed by the given amount. Typical value is 0.1*cohesion"); + params.addParam("perfect_guess", + true, + "Provide a guess to the Newton-Raphson proceedure " + "that is the result from perfect plasticity. With " + "severe hardening/softening this may be " + "suboptimal."); + params.addClassDescription("Damage Plasticity Model for concrete"); + return params; +} + +DamagePlasticityStressUpdate::DamagePlasticityStressUpdate(const InputParameters & parameters) + : MultiParameterPlasticityStressUpdate(parameters, 3, 1, 2), + _f_tol(getParam("yield_function_tol")), + + _alfa(getParam("factor_relating_biaxial_unixial_cmp_str")), + _alfa_p(getParam("factor_controlling_dilatancy")), + _s0(getParam("stiff_recovery_factor")), + + _Chi(getParam("ft_ep_slope_factor_at_zero_ep")), + _Dt(getParam("tensile_damage_at_half_tensile_strength")), + _ft(getParam("yield_strength_in_tension")), + _FEt(getParam("fracture_energy_in_tension")), + + _fyc(getParam("yield_strength_in_compression")), + _Dc(getParam("compressive_damage_at_max_compressive_strength")), + _fc(getParam("maximum_strength_in_compression")), + _FEc(getParam("fracture_energy_in_compression")), + + _at(1.5 * std::sqrt(1 - _Chi) - 0.5), + _ac((2. * (_fc / _fyc) - 1. + 2. * std::sqrt(std::pow((_fc / _fyc), 2.) - _fc / _fyc))), + + _zt((1. + _at) / _at), + _zc((1. + _ac) / _ac), + _dPhit(_at * (2. + _at)), + _dPhic(_ac * (2. + _ac)), + _sqrtPhit_max((1. + _at + sqrt(1. + _at * _at)) / 2.), + _sqrtPhic_max((1. + _ac) / 2.), + _dt_bt(log(1. - _Dt) / log((1. + _at - sqrt(1. + _at * _at)) / (2. * _at))), + _dc_bc(log(1. - _Dc) / log((1. + _ac) / (2. * _ac))), + _ft0(0.5 * _ft / + ((1. - _Dt) * pow((_zt - _sqrtPhit_max / _at), (1. - _dt_bt)) * _sqrtPhit_max)), + _fc0(_fc / ((1. - _Dc) * pow((_zc - _sqrtPhic_max / _ac), (1. - _dc_bc)) * _sqrtPhic_max)), + _small_smoother2(std::pow(getParam("tip_smoother"), 2)), + + _sqrt3(sqrt(3.)), + _perfect_guess(getParam("perfect_guess")), + _eigvecs(RankTwoTensor()), + _max_principal(declareProperty("max_principal_stress")), + _min_principal(declareProperty("min_principal_stress")), + _intnl0(declareProperty("damage_state_in_tension")), + _intnl1(declareProperty("damage_state_in_compression")), + _ele_len(declareProperty("element_length")), + _gt(declareProperty("elemental_fracture_energy_in_tension")), + _gc(declareProperty("elemental_fracture_energy_in_compression")), + _tD(declareProperty("elemental_tensile_damage")), + _cD(declareProperty("elemental_compression_damage")), + _D(declareProperty("elemental_damage_variable")), + _min_ep(declareProperty("min_ep")), + _mid_ep(declareProperty("mid_ep")), + _max_ep(declareProperty("max_ep")), + _sigma0(declareProperty("damaged_min_principal_stress")), + _sigma1(declareProperty("damaged_mid_principal_stress")), + _sigma2(declareProperty("damaged_max_principal_stress")) +{ +} + +void +DamagePlasticityStressUpdate::initQpStatefulProperties() +{ + // if (_current_elem->n_vertices() < 3) + // _ele_len[_qp] = _current_elem->length(0, 1); + // else if (_current_elem->n_vertices() < 5) + // _ele_len[_qp] = (_current_elem->length(0, 1) + _current_elem->length(1, 2)) / 2.; + // else + // _ele_len[_qp] = + // (_current_elem->length(0, 1) + _current_elem->length(1, 2) + _current_elem->length(0, 4)) + // / 3.; + _ele_len[_qp] = std::cbrt(_current_elem->volume()); + + _gt[_qp] = _FEt / _ele_len[_qp]; + _gc[_qp] = _FEc / _ele_len[_qp]; + + _min_ep[_qp] = 0.; + _mid_ep[_qp] = 0.; + _max_ep[_qp] = 0.; + _sigma0[_qp] = 0.; + _sigma1[_qp] = 0.; + _sigma2[_qp] = 0.; + _intnl0[_qp] = 0.; + _intnl1[_qp] = 0.; + _tD[_qp] = 0.; + _cD[_qp] = 0.; + _D[_qp] = 0.; + MultiParameterPlasticityStressUpdate::initQpStatefulProperties(); +} + +void +DamagePlasticityStressUpdate::finalizeReturnProcess(const RankTwoTensor & /*rotation_increment*/) +{ + std::vector eigstrain; + _plastic_strain[_qp].symmetricEigenvalues(eigstrain); + _min_ep[_qp] = eigstrain[0]; + _mid_ep[_qp] = eigstrain[1]; + _max_ep[_qp] = eigstrain[2]; +} + +void +DamagePlasticityStressUpdate::computeStressParams(const RankTwoTensor & stress, + std::vector & stress_params) const +{ + stress.symmetricEigenvalues(stress_params); +} + +std::vector +DamagePlasticityStressUpdate::dstress_param_dstress(const RankTwoTensor & stress) const +{ + std::vector sp; + std::vector dsp; + stress.dsymmetricEigenvalues(sp, dsp); + return dsp; +} + +std::vector +DamagePlasticityStressUpdate::d2stress_param_dstress(const RankTwoTensor & stress) const +{ + std::vector d2; + stress.d2symmetricEigenvalues(d2); + return d2; +} + +void +DamagePlasticityStressUpdate::setEffectiveElasticity(const RankFourTensor & Eijkl) +{ + // Eijkl is required to be isotropic, so we can use the + // frame where stress is diagonal + for (unsigned a = 0; a < _num_sp; ++a) + for (unsigned b = 0; b < _num_sp; ++b) + _Eij[a][b] = Eijkl(a, a, b, b); + _En = _Eij[2][2]; + const Real denom = _Eij[0][0] * (_Eij[0][0] + _Eij[0][1]) - 2 * Utility::pow<2>(_Eij[0][1]); + for (unsigned a = 0; a < _num_sp; ++a) + { + _Cij[a][a] = (_Eij[0][0] + _Eij[0][1]) / denom; + for (unsigned b = 0; b < a; ++b) + _Cij[a][b] = _Cij[b][a] = -_Eij[0][1] / denom; + } +} + +void +DamagePlasticityStressUpdate::preReturnMapV(const std::vector & /*trial_stress_params*/, + const RankTwoTensor & stress_trial, + const std::vector & /*intnl_old*/, + const std::vector & /*yf*/, + const RankFourTensor & /*Eijkl*/) +{ + std::vector eigvals; + stress_trial.symmetricEigenvaluesEigenvectors(eigvals, _eigvecs); +} + +void +DamagePlasticityStressUpdate::setStressAfterReturnV(const RankTwoTensor & /*stress_trial*/, + const std::vector & stress_params, + Real /*gaE*/, + const std::vector & intnl, + const yieldAndFlow & /*smoothed_q*/, + const RankFourTensor & /*Eijkl*/, + RankTwoTensor & stress) const +{ + // form the diagonal stress + stress = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0.0, 0.0, 0.0); + // rotate to the original frame + stress = _eigvecs * stress * (_eigvecs.transpose()); + // _dir[_qp] = _eigvecs; + Real D = damageVar(stress_params, intnl); + _sigma0[_qp] = (1. - D) * stress_params[0]; + _sigma1[_qp] = (1. - D) * stress_params[1]; + _sigma2[_qp] = (1. - D) * stress_params[2]; + _intnl0[_qp] = intnl[0]; + _intnl1[_qp] = intnl[1]; + _D[_qp] = D; +} + +void +DamagePlasticityStressUpdate::yieldFunctionValuesV(const std::vector & stress_params, + const std::vector & intnl, + std::vector & yf) const +{ + Real I1 = stress_params[0] + stress_params[1] + stress_params[2]; + Real J2 = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .secondInvariant(); + yf[0] = 1. / (1. - _alfa) * + (_alfa * I1 + _sqrt3 * sqrt(J2) + + beta(intnl) * (stress_params[2] < 0. ? 0. : stress_params[2])) - + fc(intnl); +} + +void +DamagePlasticityStressUpdate::computeAllQV(const std::vector & stress_params, + const std::vector & intnl, + std::vector & all_q) const +{ + Real I1 = stress_params[0] + stress_params[1] + stress_params[2]; + Real J2 = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .secondInvariant(); + RankTwoTensor d_sqrt_3J2; + if (J2 == 0) + d_sqrt_3J2 = RankTwoTensor(); + else + d_sqrt_3J2 = 0.5 * std::sqrt(3.0 / J2) * + RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .dsecondInvariant(); + + all_q[0].f = 1. / (1. - _alfa) * + (_alfa * I1 + _sqrt3 * sqrt(J2) + + beta(intnl) * (stress_params[2] < 0. ? 0. : stress_params[2])) - + fc(intnl); + + for (unsigned i = 0; i < _num_sp; ++i) + if (J2 == 0) + all_q[0].df[i] = + 1. / (1. - _alfa) * (_alfa + beta(intnl) * (stress_params[2] < 0. ? 0. : (i == 2))); + else + all_q[0].df[i] = + 1. / (1. - _alfa) * + (_alfa + d_sqrt_3J2(i, i) + beta(intnl) * (stress_params[2] < 0. ? 0. : (i == 2))); + all_q[0].df_di[0] = + 1. / (1. - _alfa) * (dbeta0(intnl) * (stress_params[2] < 0. ? 0. : stress_params[2])); + all_q[0].df_di[1] = + 1. / (1. - _alfa) * (dbeta1(intnl) * (stress_params[2] < 0. ? 0. : stress_params[2])) - + dfc(intnl); + + flowPotential(stress_params, intnl, all_q[0].dg); + dflowPotential_dstress(stress_params, intnl, all_q[0].d2g); + dflowPotential_dintnl(stress_params, intnl, all_q[0].d2g_di); +} + +void +DamagePlasticityStressUpdate::flowPotential(const std::vector & stress_params, + const std::vector & intnl, + std::vector & r) const +{ + Real J2 = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .secondInvariant(); + RankTwoTensor d_sqrt_2J2; + if (J2 == 0) + d_sqrt_2J2 = RankTwoTensor(); + else + d_sqrt_2J2 = 0.5 * std::sqrt(2.0 / J2) * + RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .dsecondInvariant(); + + Real D = damageVar(stress_params, intnl); + + for (unsigned int i = 0; i < _num_sp; ++i) + r[i] = (_alfa_p + d_sqrt_2J2(i, i)) * pow((1. - D), 1); +} + +void +DamagePlasticityStressUpdate::dflowPotential_dstress( + const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dr_dstress) const +{ + Real J2 = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .secondInvariant(); + RankTwoTensor dII = RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .dsecondInvariant(); + RankTwoTensor d_sqrt_2J2; + RankFourTensor dfp; + Real pre; + if (J2 == 0) + { + d_sqrt_2J2 = RankTwoTensor(); + dfp = RankFourTensor(); + pre = 0; + } + else + { + d_sqrt_2J2 = 0.5 * std::sqrt(2.0 / J2) * dII; + dfp = 0.5 * std::sqrt(2.0 / J2) * + RankTwoTensor(stress_params[0], stress_params[1], stress_params[2], 0, 0, 0) + .d2secondInvariant(); + pre = -0.25 * std::sqrt(2.0) * std::pow(J2, -1.5); + } + + for (unsigned i = 0; i < 3; ++i) + for (unsigned j = 0; j < 3; ++j) + for (unsigned k = 0; k < 3; ++k) + for (unsigned l = 0; l < 3; ++l) + dfp(i, j, k, l) += pre * dII(i, j) * dII(k, l); + + Real D = damageVar(stress_params, intnl); + + for (unsigned i = 0; i < _num_sp; ++i) + for (unsigned j = 0; j < (i + 1); ++j) + { + dr_dstress[i][i] = J2 < _f_tol ? 0. : dfp(i, i, j, j) * pow((1. - D), 2); + if (i != j) + dr_dstress[j][i] = dr_dstress[i][j]; + } +} + +void +DamagePlasticityStressUpdate::dflowPotential_dintnl( + const std::vector & /* stress_params */, + const std::vector & /* intnl */, + std::vector> & dr_dintnl) const +{ + for (unsigned i = 0; i < _num_sp; ++i) + for (unsigned j = 0; j < _num_intnl; ++j) + dr_dintnl[i][j] = 0.; +} + +void +DamagePlasticityStressUpdate::hardPotential(const std::vector & stress_params, + const std::vector & intnl, + std::vector & h) const +{ + Real wf; + weighfac(stress_params, wf); + std::vector r(3); + flowPotential(stress_params, intnl, r); + h[0] = wf * ft(intnl) / _gt[_qp] * r[2]; + h[1] = -(1. - wf) * fc(intnl) / _gc[_qp] * r[0]; +} + +void +DamagePlasticityStressUpdate::dhardPotential_dstress(const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dh_dsig) const +{ + Real wf; + std::vector dwf(3); + dweighfac(stress_params, wf, dwf); + + std::vector r(3); + flowPotential(stress_params, intnl, r); + std::vector> dr_dsig(3, std::vector(3)); + dflowPotential_dstress(stress_params, intnl, dr_dsig); + + for (unsigned i = 0; i < _num_sp; ++i) + { + dh_dsig[0][i] = (wf * dr_dsig[2][i] + dwf[i] * r[2]) * ft(intnl) / _gt[_qp]; + dh_dsig[1][i] = -((1. - wf) * dr_dsig[0][i] - dwf[i] * r[0]) * fc(intnl) / _gc[_qp]; + } +} + +void +DamagePlasticityStressUpdate::dhardPotential_dintnl( + const std::vector & stress_params, + const std::vector & intnl, + std::vector> & dh_dintnl) const +{ + Real wf; + weighfac(stress_params, wf); + std::vector r(3); + flowPotential(stress_params, intnl, r); + + dh_dintnl[0][0] = wf * dft(intnl) / _gt[_qp] * r[2]; + dh_dintnl[0][1] = 0.; + dh_dintnl[1][0] = 0.; + dh_dintnl[1][1] = -(1 - wf) * dfc(intnl) / _gc[_qp] * r[0]; +} + +void +DamagePlasticityStressUpdate::initialiseVarsV(const std::vector & trial_stress_params, + const std::vector & intnl_old, + std::vector & stress_params, + Real & /* gaE */, + std::vector & intnl) const +{ + setIntnlValuesV(trial_stress_params, stress_params, intnl_old, intnl); +} + +void +DamagePlasticityStressUpdate::setIntnlValuesV(const std::vector & trial_stress_params, + const std::vector & current_stress_params, + const std::vector & intnl_old, + std::vector & intnl) const +{ + Real I1_trial = trial_stress_params[0] + trial_stress_params[1] + trial_stress_params[2]; + Real J2_trial = + RankTwoTensor(trial_stress_params[0], trial_stress_params[1], trial_stress_params[2], 0, 0, 0) + .secondInvariant(); + Real invsqrt2J2_trial = 1. / sqrt(2. * J2_trial); + Real G = 0.5 * (_Eij[0][0] - _Eij[0][1]); // Lame's mu + Real K = _Eij[0][1] + 2. * G / 3.; // Bulk modulus + Real C1 = (2. * G * invsqrt2J2_trial); + Real C2 = -(I1_trial / 3. * G * invsqrt2J2_trial - 3. * K * _alfa_p); + Real C3 = 3. * K * _alfa_p; + + RankTwoTensor dsig = RankTwoTensor(trial_stress_params[0] - current_stress_params[0], + trial_stress_params[1] - current_stress_params[1], + trial_stress_params[2] - current_stress_params[2], + 0., + 0., + 0.); + RankTwoTensor fac = J2_trial < _f_tol ? C3 * RankTwoTensor(1., 1., 1., 0., 0., 0.) + : RankTwoTensor(C1 * trial_stress_params[0] - C2, + C1 * trial_stress_params[1] - C2, + C1 * trial_stress_params[2] - C2, + 0., + 0., + 0.); + + Real lam = dsig.L2norm() / fac.L2norm(); + std::vector h(2); + hardPotential(current_stress_params, intnl_old, h); + + intnl[0] = intnl_old[0] + lam * h[0]; + intnl[1] = intnl_old[1] + lam * h[1]; +} + +void +DamagePlasticityStressUpdate::setIntnlDerivativesV(const std::vector & trial_stress_params, + const std::vector & current_stress_params, + const std::vector & intnl, + std::vector> & dintnl) const +{ + Real I1_trial = trial_stress_params[0] + trial_stress_params[1] + trial_stress_params[2]; + Real J2_trial = + RankTwoTensor(trial_stress_params[0], trial_stress_params[1], trial_stress_params[2], 0, 0, 0) + .secondInvariant(); + Real invsqrt2J2_trial = 1. / sqrt(2. * J2_trial); + Real G = 0.5 * (_Eij[0][0] - _Eij[0][1]); // Lame's mu + Real K = _Eij[0][1] + 2. * G / 3.; // Bulk modulus + Real C1 = (2. * G * invsqrt2J2_trial); + Real C2 = -(I1_trial / 3. * G * invsqrt2J2_trial - 3. * K * _alfa_p); + Real C3 = 3. * K * _alfa_p; + + RankTwoTensor dsig = RankTwoTensor(trial_stress_params[0] - current_stress_params[0], + trial_stress_params[1] - current_stress_params[1], + trial_stress_params[2] - current_stress_params[2], + 0., + 0., + 0.); + RankTwoTensor fac = J2_trial < _f_tol ? C3 * RankTwoTensor(1., 1., 1., 0., 0., 0.) + : RankTwoTensor(C1 * trial_stress_params[0] - C2, + C1 * trial_stress_params[1] - C2, + C1 * trial_stress_params[2] - C2, + 0., + 0., + 0.); + + Real lam = dsig.L2norm() / fac.L2norm(); + + std::vector dlam_dsig(3); + for (unsigned i = 0; i < _num_sp; ++i) + dlam_dsig[i] = dsig.L2norm() == 0. ? 0. + : -(trial_stress_params[i] - current_stress_params[i]) / + (dsig.L2norm() * fac.L2norm()); + + std::vector h(2); + hardPotential(current_stress_params, intnl, h); + std::vector> dh_dsig(2, std::vector(3)); + dhardPotential_dstress(current_stress_params, intnl, dh_dsig); + std::vector> dh_dintnl(2, std::vector(2)); + dhardPotential_dintnl(current_stress_params, intnl, dh_dintnl); + + for (unsigned i = 0; i < _num_intnl; ++i) + for (unsigned j = 0; j < _num_sp; ++j) + dintnl[i][j] = dlam_dsig[j] * h[i] + lam * dh_dsig[i][j]; +} + +Real +DamagePlasticityStressUpdate::ft(const std::vector & intnl) const +{ + Real sqrtPhi_t = sqrt(1. + _at * (2. + _at) * intnl[0]); + if (_zt > sqrtPhi_t / _at) + return _ft0 * pow(_zt - sqrtPhi_t / _at, (1. - _dt_bt)) * sqrtPhi_t; + else + return _ft0 * 1.E-6; +} + +Real +DamagePlasticityStressUpdate::dft(const std::vector & intnl) const +{ + Real sqrtPhi_t = sqrt(1. + _at * (2. + _at) * intnl[0]); + if (_zt > sqrtPhi_t / _at) + return _ft0 * _dPhit / (2 * sqrtPhi_t) * pow(_zt - sqrtPhi_t / _at, -_dt_bt) * + (_zt - (2. - _dt_bt) * sqrtPhi_t / _at); + else + return 0.; +} + +Real +DamagePlasticityStressUpdate::fc(const std::vector & intnl) const +{ + Real sqrtPhi_c; + if (intnl[1] < 1.0) + sqrtPhi_c = sqrt(1. + _ac * (2. + _ac) * intnl[1]); + else + sqrtPhi_c = sqrt(1. + _ac * (2. + _ac) * 0.99); + return _fc0 * pow((_zc - sqrtPhi_c / _ac), (1. - _dc_bc)) * sqrtPhi_c; +} + +Real +DamagePlasticityStressUpdate::dfc(const std::vector & intnl) const +{ + if (intnl[1] < 1.0) + { + Real sqrtPhi_c = sqrt(1. + _ac * (2. + _ac) * intnl[1]); + return _fc0 * _dPhic / (2. * sqrtPhi_c) * pow(_zc - sqrtPhi_c / _ac, -_dc_bc) * + (_zc - (2. - _dc_bc) * sqrtPhi_c / _ac); + } + else + return 0.; +} + +Real +DamagePlasticityStressUpdate::beta(const std::vector & intnl) const +{ + return (1. - _alfa) * fc(intnl) / ft(intnl) - (1. + _alfa); +} + +Real +DamagePlasticityStressUpdate::dbeta0(const std::vector & intnl) const +{ + return -(1. - _alfa) * fc(intnl) * dft(intnl) / pow(ft(intnl), 2.); +} + +Real +DamagePlasticityStressUpdate::dbeta1(const std::vector & intnl) const +{ + return dfc(intnl) / ft(intnl) * (1. - _alfa); +} + +void +DamagePlasticityStressUpdate::weighfac(const std::vector & stress_params, Real & wf) const +{ + Real Dr = 0.; + Real Nr = 0.; + for (unsigned i = 0; i < _num_sp; ++i) + { + if (stress_params[i] > 0.) + { + Nr += stress_params[i]; + Dr += stress_params[i]; + } + else + Dr += -stress_params[i]; + } + wf = Nr / Dr; +} + +void +DamagePlasticityStressUpdate::dweighfac(const std::vector & stress_params, + Real & wf, + std::vector & dwf) const +{ + std::vector dNr(3, 0.), dDr(3, 0.); + Real Dr = 0.; + Real Nr = 0.; + for (unsigned i = 0; i < _num_sp; ++i) + { + if (stress_params[i] > 0.) + { + Nr += stress_params[i]; + dNr[i] = 1.; + Dr += stress_params[i]; + dDr[i] = 1.; + } + else + { + Dr += -stress_params[i]; + dDr[i] = -1.; + } + } + wf = Nr / Dr; + + for (unsigned i = 0; i < _num_sp; ++i) + dwf[i] = (dNr[i] - wf * dDr[i]) / Dr; +} + +Real +DamagePlasticityStressUpdate::damageVar(const std::vector & stress_params, + const std::vector & intnl) const +{ + Real sqrtPhi_t = sqrt(1. + _at * (2. + _at) * intnl[0]); + if (_zt > sqrtPhi_t / _at) + _tD[_qp] = 1. - pow(_zt - sqrtPhi_t / _at, _dt_bt); + else + _tD[_qp] = 1. - 1.E-6; + + Real wf; + weighfac(stress_params, wf); + Real s = _s0 + (1. - _s0) * wf; + + Real sqrtPhi_c; + if (intnl[1] < 1.0) + sqrtPhi_c = sqrt(1. + _ac * (2. + _ac) * intnl[1]); + else + sqrtPhi_c = sqrt(1. + _ac * (2. + _ac) * 0.99); + + _cD[_qp] = 1. - pow((_zc - sqrtPhi_c / _ac), _dc_bc); + return 1. - (1. - s * _tD[_qp]) * (1. - _cD[_qp]); +} + +void +DamagePlasticityStressUpdate::consistentTangentOperatorV( + const RankTwoTensor & /* stress_trial */, + const std::vector & /* trial_stress_params */, + const RankTwoTensor & /*stress*/, + const std::vector & /* stress_params */, + Real /*gaE*/, + const yieldAndFlow & /*smoothed_q*/, + const RankFourTensor & elasticity_tensor, + bool /* compute_full_tangent_operator */, + const std::vector> & /* dvar_dtrial */, + RankFourTensor & cto) +{ + cto = elasticity_tensor; + return; +} diff --git a/test/tests/damage_plasticity_model/gold/shear_test_out.csv b/test/tests/damage_plasticity_model/gold/shear_test_out.csv new file mode 100644 index 000000000..27901560b --- /dev/null +++ b/test/tests/damage_plasticity_model/gold/shear_test_out.csv @@ -0,0 +1,32 @@ +time,e_xy,ep_xy,s_xy +0,0,0,0 +5,7.8182642932574e-06,0,0.66498115646358 +10,1.5636528586514e-05,0,1.3299623129272 +15,2.345207495453e-05,4.8983821458532e-07,1.8597699614386 +20,3.11707777874e-05,2.3387363549052e-06,1.999713831302 +25,3.8959171524551e-05,4.3097281241562e-06,2.1346824057288 +30,4.6801703888548e-05,6.3885974716588e-06,2.2574870564089 +35,5.5637752385082e-05,1.207217846236e-05,2.3033174652061 +40,6.4839823260016e-05,1.844537790924e-05,2.3398394923518 +45,7.4332133712806e-05,2.5158614352263e-05,2.3733504627113 +50,8.4092539131669e-05,3.2187207599246e-05,2.4049417650395 +55,9.3888463826214e-05,3.9908256976132e-05,2.4266012372914 +60,0.00010323683612959,4.8992974581954e-05,2.4214548809772 +65,0.00011266255963277,5.8229398297501e-05,2.4142310514246 +70,0.00012214407352407,6.7602846421036e-05,2.4038645327461 +75,0.00013168927245092,7.7076797993163e-05,2.3923720472085 +80,0.00014129356869822,8.6636512382646e-05,2.3798545017804 +85,0.00015095301808515,9.6270716039463e-05,2.3664924645965 +90,0.00016066467929913,0.00010597105426996,2.3523986408369 +95,0.00017040819689737,0.00011571915090458,2.3387981408876 +100,0.00018018480850651,0.00012550953189172,2.3252270791176 +105,0.00018999579215526,0.00013533938039156,2.3114253216103 +110,0.00019983912189051,0.0001452051244901,2.2974371251615 +115,0.00020971300364264,0.00015510388179827,2.2833019593255 +120,0.00021959888110224,0.00016502273243373,2.270282271349 +125,0.00022950520377271,0.00017496197512972,2.2574608359554 +130,0.00023943148653538,0.00018492075750379,2.2447204907186 +135,0.00024937629740186,0.00019489784829704,2.2320557231984 +140,0.00025933856736405,0.0002048919743404,2.2194726111622 +145,0.00026931733603067,0.00021490203530275,2.2069758602706 +150,0.00027931174337632,0.00022492707416744,2.1945687759467 diff --git a/test/tests/damage_plasticity_model/gold/uniaxial_compression_out.csv b/test/tests/damage_plasticity_model/gold/uniaxial_compression_out.csv new file mode 100644 index 000000000..60820880c --- /dev/null +++ b/test/tests/damage_plasticity_model/gold/uniaxial_compression_out.csv @@ -0,0 +1,52 @@ +time,displacement_x,react_x,s_xx +0,0,0,0 +10,-0.0001,3.17,-3.17 +20,-0.0002,6.34,-6.34 +30,-0.0003,9.51,-9.51 +40,-0.0004,12.68,-12.68 +50,-0.0005,15.85,-15.85 +60,-0.0006,18.64578867,-18.64578867 +70,-0.0007,20.19367563,-20.19367563 +80,-0.0008,21.62512169,-21.62512169 +90,-0.0009,22.71489264,-22.71489264 +100,-0.001,23.71370544,-23.71370544 +110,-0.0011,24.62195323,-24.62195323 +120,-0.0012,25.44011227,-25.44011227 +130,-0.0013,26.16875604,-26.16875604 +140,-0.0014,26.80857009,-26.80857009 +150,-0.0015,27.3603668,-27.3603668 +160,-0.0016,27.82510011,-27.82510011 +170,-0.0017,28.20387957,-28.20387957 +180,-0.0018,28.49798676,-28.49798676 +190,-0.0019,28.70888792,-28.70888792 +200,-0.002,28.83824946,-28.83824946 +210,-0.0021,28.88794706,-28.88794706 +220,-0.0022,28.86008204,-28.86008204 +230,-0.0023,28.75699116,-28.75699116 +240,-0.0024,28.58125514,-28.58125514 +250,-0.0025,28.33570579,-28.33570579 +260,-0.0026,28.02342987,-28.02342987 +270,-0.0027,27.64776983,-27.64776983 +280,-0.0028,27.21232051,-27.21232051 +290,-0.0029,26.72092154,-26.72092154 +300,-0.003,26.17764477,-26.17764477 +310,-0.0031,25.58677634,-25.58677634 +320,-0.0032,24.95279339,-24.95279339 +330,-0.0033,24.28033472,-24.28033472 +340,-0.0034,23.57416623,-23.57416623 +350,-0.0035,22.83914111,-22.83914111 +360,-0.0036,22.08015565,-22.08015565 +370,-0.0037,21.30210089,-21.30210089 +380,-0.0038,20.50981323,-20.50981323 +390,-0.0039,19.70802252,-19.70802252 +400,-0.004,18.90130194,-18.90130194 +410,-0.0041,18.09401946,-18.09401946 +420,-0.0042,17.29029338,-17.29029338 +430,-0.0043,16.49395305,-16.49395305 +440,-0.0044,15.70850686,-15.70850686 +450,-0.0045,14.93711621,-14.93711621 +460,-0.0046,14.18257945,-14.18257945 +470,-0.0047,13.44732218,-13.44732218 +480,-0.0048,12.73339745,-12.73339745 +490,-0.0049,12.04249192,-12.04249192 +500,-0.005,11.37593971,-11.37593971 \ No newline at end of file diff --git a/test/tests/damage_plasticity_model/gold/uniaxial_tension_out.csv b/test/tests/damage_plasticity_model/gold/uniaxial_tension_out.csv new file mode 100644 index 000000000..d66dae2af --- /dev/null +++ b/test/tests/damage_plasticity_model/gold/uniaxial_tension_out.csv @@ -0,0 +1,32 @@ +time,displacement_x,react_x,s_xx +0,0,0,0 +5,0.00025,-3.5059949907068,3.5059949907068 +10,0.0005,-3.5433099131495,3.5433099131495 +15,0.00075,-3.4118850343689,3.4118850343689 +20,0.001,-3.1582664717401,3.1582664717401 +25,0.00125,-2.9155170891073,2.9155170891073 +30,0.0015,-2.6849823789444,2.6849823789444 +35,0.00175,-2.4674713932184,2.4674713932184 +40,0.002,-2.2633734276651,2.2633734276651 +45,0.00225,-2.0727543309067,2.0727543309067 +50,0.0025,-1.8954350159471,1.8954350159471 +55,0.00275,-1.7310555130409,1.7310555130409 +60,0.003,-1.5791262741523,1.5791262741523 +65,0.00325,-1.4390690718447,1.4390690718447 +70,0.0035,-1.3102492850223,1.3102492850223 +75,0.00375,-1.1920010941285,1.1920010941285 +80,0.004,-1.0836468859318,1.0836468859318 +85,0.00425,-0.98451197378197,0.98451197378197 +90,0.0045,-0.89393551236588,0.89393551236588 +95,0.00475,-0.81127845393839,0.81127845393839 +100,0.005,-0.73592901705959,0.73592901705959 +105,0.00525,-0.66730632692045,0.66730632692045 +110,0.0055,-0.60486257240887,0.60486257240887 +115,0.00575,-0.54808402869796,0.54808402869796 +120,0.006,-0.49649121765577,0.49649121765577 +125,0.00625,-0.44963843163719,0.44963843163719 +130,0.0065,-0.40711277281401,0.40711277281401 +135,0.00675,-0.36853289823744,0.36853289823744 +140,0.007,-0.3335474924857,0.3335474924857 +145,0.00725,-0.3018336337162,0.3018336337162 +150,0.0075,-0.27309508032506,0.27309508032506 diff --git a/test/tests/damage_plasticity_model/shear_test.i b/test/tests/damage_plasticity_model/shear_test.i new file mode 100644 index 000000000..cb51e1f17 --- /dev/null +++ b/test/tests/damage_plasticity_model/shear_test.i @@ -0,0 +1,214 @@ +[Mesh] + type = GeneratedMesh + dim = 3 + nx = 1 + ny = 1 + nz = 1 + xmin = -0.5 + xmax = 0.5 + ymin = -0.5 + ymax = 0.5 + zmin = -0.5 + zmax = 0.5 +[] + +[GlobalParams] + displacements = 'disp_x disp_y disp_z' +[] + +[Modules/TensorMechanics/Master] + [all] + add_variables = true + incremental = true + generate_output = 'stress_xx stress_xy stress_xz stress_yy stress_yz stress_zz plastic_strain_xx plastic_strain_xy plastic_strain_xz plastic_strain_yy plastic_strain_yz plastic_strain_zz strain_xx strain_xy strain_xz strain_yy strain_yz strain_zz' + save_in = 'resid_x resid_y resid_z' + [] +[] + +[AuxVariables] + [resid_x] + [] + [resid_y] + [] + [resid_z] + [] + [min_ep] + order = CONSTANT + family = MONOMIAL + [] + [mid_ep] + order = CONSTANT + family = MONOMIAL + [] + [max_ep] + order = CONSTANT + family = MONOMIAL + [] + [sigma0] + order = CONSTANT + family = MONOMIAL + [] + [sigma1] + order = CONSTANT + family = MONOMIAL + [] + [sigma2] + order = CONSTANT + family = MONOMIAL + [] + [f0] + order = CONSTANT + family = MONOMIAL + [] + [D] + order = CONSTANT + family = MONOMIAL + [] + [intnl0] + order = CONSTANT + family = MONOMIAL + [] + [intnl1] + order = CONSTANT + family = MONOMIAL + [] +[] + +[AuxKernels] + [f0_auxk] + type = MaterialStdVectorAux + property = plastic_yield_function + index = 0 + variable = f0 + [] + [D_auxk] + type = MaterialRealAux + property = elemental_damage_variable + variable = D + [] + [min_ep] + type = MaterialRealAux + property = min_ep + variable = min_ep + [] + [mid_ep] + type = MaterialRealAux + property = mid_ep + variable = mid_ep + [] + [max_ep] + type = MaterialRealAux + property = max_ep + variable = max_ep + [] + [sigma0] + type = MaterialRealAux + property = damaged_min_principal_stress + variable = sigma0 + [] + [sigma1] + type = MaterialRealAux + property = damaged_mid_principal_stress + variable = sigma1 + [] + [sigma2] + type = MaterialRealAux + property = damaged_max_principal_stress + variable = sigma2 + [] + [intnl0_auxk] + type = MaterialRealAux + property = damage_state_in_tension + variable = intnl0 + [] + [intnl1_auxk] + type = MaterialRealAux + property = damage_state_in_compression + variable = intnl1 + [] +[] + +[Postprocessors] + [s_xy] + type = ElementAverageValue + variable = stress_xx + [] + [e_xy] + type = ElementAverageValue + variable = strain_xy + [] + [ep_xy] + type = ElementAverageValue + variable = plastic_strain_xy + [] +[] + +[BCs] + [x_r] + type = FunctionDirichletBC + variable = disp_x + boundary = 'top' + function = '1E-5*t' + [] + [x_l] + type = FunctionDirichletBC + variable = disp_x + boundary = 'left' + function = '0' + [] + [y_l] + type = FunctionDirichletBC + variable = disp_y + boundary = 'bottom' + function = '0' + [] + [z_l] + type = FunctionDirichletBC + variable = disp_z + boundary = 'back' + function = '0' + [] +[] + +[Materials] + [elasticity_tensor] + type = ComputeIsotropicElasticityTensor + youngs_modulus = 3.17E4 + poissons_ratio = 0.18 + [] + [pdm] + type = DamagePlasticityStressUpdate + factor_relating_biaxial_unixial_cmp_str = 0.109 + factor_controlling_dilatancy = 0.23 + stiff_recovery_factor = 0.001 + + yield_strength_in_tension = 3.48 + ft_ep_slope_factor_at_zero_ep = 0.70 + tensile_damage_at_half_tensile_strength = 0.51 + fracture_energy_in_tension = 12.3E-3 + + yield_strength_in_compression = 18.30 + maximum_strength_in_compression = 27.60 + compressive_damage_at_max_compressive_strength = 0.25 + fracture_energy_in_compression = 6.889E-2 + + yield_function_tol = 1.E-5 + tip_smoother = 1.E-6 + smoothing_tol = 1.E-3 + [] + [stress] + type = ComputeMultipleInelasticDamageStress + inelastic_models = pdm + perform_finite_strain_rotations = false + [] +[] + +[Executioner] + end_time = 150 + dt = 5 + type = Transient +[] + +[Outputs] + csv = true +[] diff --git a/test/tests/damage_plasticity_model/tests b/test/tests/damage_plasticity_model/tests new file mode 100644 index 000000000..cc9b36f50 --- /dev/null +++ b/test/tests/damage_plasticity_model/tests @@ -0,0 +1,34 @@ +[Tests] + issues = '#184' + design = 'DamagePlasticityStressUpdate.md ComputeMultipleInelasticDamageStress.md' + [Damage_Plasticity_Model] + requirement = 'Blackbear shall demonstrate stress-strain behavior of concrete ' + [uniaxial_tension] + type = 'CSVDiff' + input = 'uniaxial_test.i' + csvdiff = 'uniaxial_tension_out.csv' + allow_test_objects = True + cli_args = 'Outputs/file_base=uniaxial_tension_out' + detail = ' in uniaxial tensile loading.' + [] + [uniaxial_compression] + type = 'CSVDiff' + input = 'uniaxial_test.i' + csvdiff = 'uniaxial_compression_out.csv' + allow_test_objects = True + cli_args = 'BCs/x_r/function=-2E-5*x*t + Executioner/end_time=500 + Executioner/dt=10 + Outputs/file_base=uniaxial_compression_out' + detail = ' in uniaxial compressive loading.' + [] + [shear] + type = 'CSVDiff' + input = 'shear_test.i' + csvdiff = 'shear_test_out.csv' + allow_test_objects = True + cli_args = 'Outputs/file_base=shear_test_out' + detail = ' in shear loading.' + [] + [] +[] diff --git a/test/tests/damage_plasticity_model/uniaxial_test.i b/test/tests/damage_plasticity_model/uniaxial_test.i new file mode 100644 index 000000000..3fc23789c --- /dev/null +++ b/test/tests/damage_plasticity_model/uniaxial_test.i @@ -0,0 +1,217 @@ +[Mesh] + type = GeneratedMesh + dim = 3 + nx = 1 + ny = 1 + nz = 1 + xmin = -0.5 + xmax = 0.5 + ymin = -0.5 + ymax = 0.5 + zmin = -0.5 + zmax = 0.5 +[] + +[GlobalParams] + displacements = 'disp_x disp_y disp_z' +[] + +[Modules/TensorMechanics/Master] + [all] + add_variables = true + incremental = true + generate_output = 'stress_xx stress_xy stress_xz stress_yy stress_yz stress_zz plastic_strain_xx plastic_strain_xy plastic_strain_xz plastic_strain_yy plastic_strain_yz plastic_strain_zz strain_xx strain_xy strain_xz strain_yy strain_yz strain_zz' + save_in = 'resid_x resid_y resid_z' + [] +[] + +[AuxVariables] + [resid_x] + [] + [resid_y] + [] + [resid_z] + [] + [min_ep] + order = CONSTANT + family = MONOMIAL + [] + [mid_ep] + order = CONSTANT + family = MONOMIAL + [] + [max_ep] + order = CONSTANT + family = MONOMIAL + [] + [sigma0] + order = CONSTANT + family = MONOMIAL + [] + [sigma1] + order = CONSTANT + family = MONOMIAL + [] + [sigma2] + order = CONSTANT + family = MONOMIAL + [] + [f0] + order = CONSTANT + family = MONOMIAL + [] + [D] + order = CONSTANT + family = MONOMIAL + [] + [intnl0] + order = CONSTANT + family = MONOMIAL + [] + [intnl1] + order = CONSTANT + family = MONOMIAL + [] +[] + +[AuxKernels] + [f0_auxk] + type = MaterialStdVectorAux + property = plastic_yield_function + index = 0 + variable = f0 + [] + [D_auxk] + type = MaterialRealAux + property = elemental_damage_variable + variable = D + [] + [min_ep] + type = MaterialRealAux + property = min_ep + variable = min_ep + [] + [mid_ep] + type = MaterialRealAux + property = mid_ep + variable = mid_ep + [] + [max_ep] + type = MaterialRealAux + property = max_ep + variable = max_ep + [] + [sigma0] + type = MaterialRealAux + property = damaged_min_principal_stress + variable = sigma0 + [] + [sigma1] + type = MaterialRealAux + property = damaged_mid_principal_stress + variable = sigma1 + [] + [sigma2] + type = MaterialRealAux + property = damaged_max_principal_stress + variable = sigma2 + [] + [intnl0_auxk] + type = MaterialRealAux + property = damage_state_in_tension + variable = intnl0 + [] + [intnl1_auxk] + type = MaterialRealAux + property = damage_state_in_compression + variable = intnl1 + [] +[] + +[Postprocessors] + [react_x] + type = NodalSum + variable = resid_x + boundary = 'left' + [] + [displacement_x] + type = AverageNodalVariableValue + variable = disp_x + boundary = 'right' + [] + [s_xx] + type = PointValue + point = '0 0 0' + variable = stress_xx + [] +[] + +[BCs] + [x_r] + type = FunctionDirichletBC + variable = disp_x + boundary = 'right' + function = '1E-4*x*t' + [] + [x_l] + type = FunctionDirichletBC + variable = disp_x + boundary = 'left' + function = '0' + [] + [y_l] + type = FunctionDirichletBC + variable = disp_y + boundary = 'bottom' + function = '0' + [] + [z_l] + type = FunctionDirichletBC + variable = disp_z + boundary = 'back' + function = '0' + [] +[] + +[Materials] + [elasticity_tensor] + type = ComputeIsotropicElasticityTensor + youngs_modulus = 3.17E4 + poissons_ratio = 0.18 + [] + [pdm] + type = DamagePlasticityStressUpdate + factor_relating_biaxial_unixial_cmp_str = 0.109 + factor_controlling_dilatancy = 0.23 + stiff_recovery_factor = 0.001 + + yield_strength_in_tension = 3.48 + ft_ep_slope_factor_at_zero_ep = 0.70 + tensile_damage_at_half_tensile_strength = 0.51 + fracture_energy_in_tension = 12.3E-3 + + yield_strength_in_compression = 18.30 + maximum_strength_in_compression = 27.60 + compressive_damage_at_max_compressive_strength = 0.25 + fracture_energy_in_compression = 6.889E-2 + + yield_function_tol = 1.E-5 + tip_smoother = 1.E-6 + smoothing_tol = 1.E-3 + [] + [stress] + type = ComputeMultipleInelasticDamageStress + inelastic_models = pdm + perform_finite_strain_rotations = false + [] +[] + +[Executioner] + type = Transient + end_time = 150 + dt = 5 +[] + +[Outputs] + csv = true +[]