2013-03-22 23:05:05 +05:30
! Copyright 2011-13 Max-Planck-Institut für Eisenforschung GmbH
!
! This file is part of DAMASK,
! the Düsseldorf Advanced MAterial Simulation Kit.
!
! DAMASK is free software: you can redistribute it and/or modify
! it under the terms of the GNU General Public License as published by
! the Free Software Foundation, either version 3 of the License, or
! (at your option) any later version.
!
! DAMASK is distributed in the hope that it will be useful,
! but WITHOUT ANY WARRANTY; without even the implied warranty of
! MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! GNU General Public License for more details.
!
! You should have received a copy of the GNU General Public License
! along with DAMASK. If not, see <http://www.gnu.org/licenses/>.
!
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
! $Id$
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
!> @author Pratheek Shanthraj, Max-Planck-Institut für Eisenforschung GmbH
!> @author Martin Diehl, Max-Planck-Institut für Eisenforschung GmbH
!> @author Philip Eisenlohr, Max-Planck-Institut für Eisenforschung GmbH
!> @brief Utilities used by the different spectral solver variants
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
module DAMASK_spectral_utilities
use , intrinsic :: iso_c_binding
2012-07-24 22:37:10 +05:30
use prec , only : &
pReal , &
pInt
2012-10-24 17:01:40 +05:30
2012-07-19 22:54:56 +05:30
implicit none
2013-03-28 16:07:00 +05:30
private
2012-12-15 23:37:49 +05:30
#ifdef PETSc
#include <finclude/petscsys.h>
#endif
2012-10-24 17:01:40 +05:30
logical , public :: cutBack = . false . !< cut back of BVP solver in case convergence is not achieved or a material point is terminally ill
2012-07-23 15:42:31 +05:30
!--------------------------------------------------------------------------------------------------
! variables storing information for spectral method and FFTW
2012-11-12 19:44:39 +05:30
real ( pReal ) , public , dimension ( : , : , : , : , : ) , pointer :: field_real !< real representation (some stress or deformation) of field_fourier
complex ( pReal ) , private , dimension ( : , : , : , : , : ) , pointer :: field_fourier !< field on which the Fourier transform operates
2012-10-24 17:01:40 +05:30
real ( pReal ) , private , dimension ( : , : , : , : , : , : , : ) , allocatable :: gamma_hat !< gamma operator (field) for spectral method
real ( pReal ) , private , dimension ( : , : , : , : ) , allocatable :: xi !< wave vector field for divergence and for gamma operator
real ( pReal ) , private , dimension ( 3 , 3 , 3 , 3 ) :: C_ref !< reference stiffness
2012-07-23 15:42:31 +05:30
!--------------------------------------------------------------------------------------------------
! debug fftw
2012-11-12 19:44:39 +05:30
complex ( pReal ) , private , dimension ( : , : , : ) , pointer :: scalarField_real , & !< scalar field real representation for debug of FFTW
scalarField_fourier !< scalar field complex representation for debug of FFTW
2012-07-20 21:03:13 +05:30
2012-07-23 15:42:31 +05:30
!--------------------------------------------------------------------------------------------------
! debug divergence
2013-04-10 15:49:16 +05:30
real ( pReal ) , private , dimension ( : , : , : , : ) , pointer :: divReal !< scalar field real representation for debugging divergence calculation
complex ( pReal ) , private , dimension ( : , : , : , : ) , pointer :: divFourier !< scalar field real representation for debugging divergence calculation
2012-07-23 15:42:31 +05:30
2012-11-12 19:44:39 +05:30
!--------------------------------------------------------------------------------------------------
! plans for FFTW
2013-04-10 15:49:16 +05:30
type ( C_PTR ) , private :: &
planForth , & !< FFTW plan P(x) to P(k)
planBack , & !< FFTW plan F(k) to F(x)
planDebugForth , & !< FFTW plan for scalar field (proof that order of usual transform is correct)
planDebugBack , & !< FFTW plan for scalar field inverse (proof that order of usual transform is correct)
planDiv !< plan for FFTW in case of debugging divergence calculation
2012-11-12 19:44:39 +05:30
2012-07-23 15:42:31 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
! variables controlling debugging
logical , public :: &
debugGeneral , & !< general debugging of spectral solver
debugDivergence , & !< debugging of divergence calculation (comparison to function used for post processing)
debugRestart , & !< debbuging of restart features
2012-11-28 20:34:05 +05:30
debugFFTW , & !< doing additional FFT on scalar field and compare to results of strided 3D FFT
2012-12-15 23:37:49 +05:30
debugRotation , & !< also printing out results in lab frame
debugPETSc !< use some in debug defined options for more verbose PETSc solution
2012-07-23 15:42:31 +05:30
2012-07-25 19:31:39 +05:30
!--------------------------------------------------------------------------------------------------
2012-08-03 14:55:48 +05:30
! derived types
2013-04-10 15:49:16 +05:30
type , public :: tSolutionState !< return type of solution from spectral solver variants
2012-10-24 17:01:40 +05:30
logical :: converged = . true .
logical :: regrid = . false .
logical :: termIll = . false .
integer ( pInt ) :: iterationsNeeded = 0_pInt
2012-10-02 20:56:56 +05:30
end type tSolutionState
2012-08-03 14:55:48 +05:30
2013-04-10 15:49:16 +05:30
type , public :: tBoundaryCondition !< set of parameters defining a boundary condition
2012-10-24 17:01:40 +05:30
real ( pReal ) , dimension ( 3 , 3 ) :: values = 0.0_pReal
real ( pReal ) , dimension ( 3 , 3 ) :: maskFloat = 0.0_pReal
2012-08-03 14:55:48 +05:30
logical , dimension ( 3 , 3 ) :: maskLogical = . false .
2012-10-24 17:01:40 +05:30
character ( len = 64 ) :: myType = 'None'
2012-10-02 20:56:56 +05:30
end type tBoundaryCondition
2012-10-24 17:01:40 +05:30
public :: &
utilities_init , &
utilities_updateGamma , &
utilities_FFTforward , &
utilities_FFTbackward , &
utilities_fourierConvolution , &
utilities_divergenceRMS , &
2013-03-07 01:04:30 +05:30
utilities_curlRMS , &
2012-10-24 17:01:40 +05:30
utilities_maskedCompliance , &
utilities_constitutiveResponse , &
utilities_calculateRate , &
utilities_forwardField , &
utilities_destroy
private :: &
utilities_getFilter
2012-08-03 14:55:48 +05:30
2012-07-23 15:42:31 +05:30
contains
2012-07-19 22:54:56 +05:30
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
2012-11-12 19:44:39 +05:30
!> @brief allocates all neccessary fields, sets debug flags, create plans for FFTW
!> @details Sets the debug levels for general, divergence, restart and FFTW from the biwise coding
2012-08-09 18:34:56 +05:30
!> provided by the debug module to logicals.
!> Allocates all fields used by FFTW and create the corresponding plans depending on the debug
!> level chosen.
!> Initializes FFTW.
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
subroutine utilities_init ( )
2012-08-27 13:34:47 +05:30
use , intrinsic :: iso_fortran_env ! to get compiler_version and compiler_options (at least for gfortran >4.6 at the moment)
2012-10-24 17:01:40 +05:30
use IO , only : &
2013-01-10 19:03:43 +05:30
IO_error , &
2013-02-25 22:04:59 +05:30
IO_warning , &
IO_timeStamp
2012-08-03 14:55:48 +05:30
use numerics , only : &
2012-07-20 21:03:13 +05:30
DAMASK_NumThreadsInt , &
fftw_planner_flag , &
2012-07-30 19:36:22 +05:30
fftw_timelimit , &
2012-12-15 23:37:49 +05:30
memory_efficient , &
petsc_options
2012-07-20 21:03:13 +05:30
use debug , only : &
debug_level , &
debug_spectral , &
debug_levelBasic , &
debug_spectralDivergence , &
debug_spectralRestart , &
2012-11-28 20:34:05 +05:30
debug_spectralFFTW , &
2013-01-11 16:10:16 +05:30
debug_spectralPETSc , &
2013-01-02 22:32:12 +05:30
debug_spectralRotation
2012-12-15 23:37:49 +05:30
#ifdef PETSc
2013-01-02 22:32:12 +05:30
use debug , only : &
2012-12-15 23:37:49 +05:30
debug_spectralPETSc , &
2013-01-02 22:32:12 +05:30
PETScDebug
2012-12-15 23:37:49 +05:30
#endif
2012-10-24 17:01:40 +05:30
use mesh , only : &
res , &
res1_red , &
2013-01-16 16:10:53 +05:30
scaledDim
2012-10-24 17:01:40 +05:30
use math ! must use the whole module for use of FFTW
2012-12-15 23:37:49 +05:30
2012-07-24 22:37:10 +05:30
implicit none
2012-12-15 23:37:49 +05:30
#ifdef PETSc
2013-02-13 23:24:56 +05:30
external :: &
PETScOptionsClear , &
PETScOptionsInsertString , &
MPI_Abort
2012-12-15 23:37:49 +05:30
PetscErrorCode :: ierr
#endif
2012-10-24 17:01:40 +05:30
integer ( pInt ) :: i , j , k
2012-07-24 22:37:10 +05:30
integer ( pInt ) , dimension ( 3 ) :: k_s
2012-10-24 17:01:40 +05:30
type ( C_PTR ) :: &
tensorField , & !< field cotaining data for FFTW in real and fourier space (in place)
scalarField_realC , & !< field cotaining data for FFTW in real space when debugging FFTW (no in place)
scalarField_fourierC , & !< field cotaining data for FFTW in fourier space when debugging FFTW (no in place)
2013-04-10 15:49:16 +05:30
div !< field cotaining data for FFTW in real and fourier space when debugging divergence (in place)
2013-03-27 17:58:55 +05:30
write ( 6 , '(/,a)' ) ' <<<+- DAMASK_spectral_utilities init -+>>>'
write ( 6 , '(a)' ) ' $Id$'
write ( 6 , '(a15,a)' ) ' Current time: ' , IO_timeStamp ( )
2012-07-19 22:54:56 +05:30
#include "compilation_info.f90"
!--------------------------------------------------------------------------------------------------
2012-07-23 15:42:31 +05:30
! set debugging parameters
2012-07-19 22:54:56 +05:30
debugGeneral = iand ( debug_level ( debug_spectral ) , debug_levelBasic ) / = 0
debugDivergence = iand ( debug_level ( debug_spectral ) , debug_spectralDivergence ) / = 0
debugRestart = iand ( debug_level ( debug_spectral ) , debug_spectralRestart ) / = 0
debugFFTW = iand ( debug_level ( debug_spectral ) , debug_spectralFFTW ) / = 0
2012-11-28 20:34:05 +05:30
debugRotation = iand ( debug_level ( debug_spectral ) , debug_spectralRotation ) / = 0
2012-12-15 23:37:49 +05:30
debugPETSc = iand ( debug_level ( debug_spectral ) , debug_spectralPETSc ) / = 0
2013-01-10 19:03:43 +05:30
#ifdef PETSc
2013-01-10 03:49:32 +05:30
if ( debugPETSc ) write ( 6 , '(/,a)' ) ' Initializing PETSc with debug options: ' , trim ( PETScDebug ) , &
' add more using the PETSc_Options keyword in numerics.config '
flush ( 6 )
call PetscOptionsClear ( ierr ) ; CHKERRQ ( ierr )
if ( debugPETSc ) call PetscOptionsInsertString ( trim ( PETScDebug ) , ierr ) ; CHKERRQ ( ierr )
call PetscOptionsInsertString ( trim ( petsc_options ) , ierr ) ; CHKERRQ ( ierr )
2013-01-10 19:03:43 +05:30
#else
call IO_warning ( 41_pInt , ext_msg = 'debug PETSc' )
2012-12-15 23:37:49 +05:30
#endif
2013-03-07 01:04:30 +05:30
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
! allocation
2012-11-12 19:44:39 +05:30
allocate ( xi ( 3 , res1_red , res ( 2 ) , res ( 3 ) ) , source = 0.0_pReal ) ! frequencies, only half the size for first dimension
tensorField = fftw_alloc_complex ( int ( res1_red * res ( 2 ) * res ( 3 ) * 9_pInt , C_SIZE_T ) ) ! allocate aligned data using a C function, C_SIZE_T is of type integer(8)
2013-04-10 15:49:16 +05:30
call c_f_pointer ( tensorField , field_real , [ res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) , res ( 2 ) , res ( 3 ) , 3 , 3 ] ) ! place a pointer for a real representation on tensorField
call c_f_pointer ( tensorField , field_fourier , [ res1_red , res ( 2 ) , res ( 3 ) , 3 , 3 ] ) ! place a pointer for a complex representation on tensorField
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
! general initialization of FFTW (see manual on fftw.org for more details)
2012-12-14 23:00:22 +05:30
if ( pReal / = C_DOUBLE . or . pInt / = C_INT ) call IO_error ( 0_pInt , ext_msg = 'Fortran to C' ) ! check for correct precision in C
2012-07-19 22:54:56 +05:30
!$ if(DAMASK_NumThreadsInt > 0_pInt) then
2012-10-24 17:01:40 +05:30
!$ i = fftw_init_threads() ! returns 0 in case of problem
!$ if (i == 0_pInt) call IO_error(error_ID = 809_pInt)
2012-07-19 22:54:56 +05:30
!$ call fftw_plan_with_nthreads(DAMASK_NumThreadsInt)
!$ endif
call fftw_set_timelimit ( fftw_timelimit ) ! set timelimit for plan creation
!--------------------------------------------------------------------------------------------------
2012-11-12 19:44:39 +05:30
! creating plans for the convolution
2013-04-10 15:49:16 +05:30
planForth = fftw_plan_many_dft_r2c ( 3 , [ res ( 3 ) , res ( 2 ) , res ( 1 ) ] , 9 , & ! dimensions, logical length in each dimension in reversed order, no. of transforms
field_real , [ res ( 3 ) , res ( 2 ) , res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ] , & ! input data, physical length in each dimension in reversed order
1 , res ( 3 ) * res ( 2 ) * ( res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ) , & ! striding, product of physical length in the 3 dimensions
field_fourier , [ res ( 3 ) , res ( 2 ) , res1_red ] , & ! output data, physical length in each dimension in reversed order
1 , res ( 3 ) * res ( 2 ) * res1_red , fftw_planner_flag ) ! striding, product of physical length in the 3 dimensions, planner precision
2012-07-19 22:54:56 +05:30
2013-04-10 15:49:16 +05:30
planBack = fftw_plan_many_dft_c2r ( 3 , [ res ( 3 ) , res ( 2 ) , res ( 1 ) ] , 9 , & ! dimensions, logical length in each dimension in reversed order, no. of transforms
field_fourier , [ res ( 3 ) , res ( 2 ) , res1_red ] , & ! input data, physical length in each dimension in reversed order
1 , res ( 3 ) * res ( 2 ) * res1_red , & ! striding, product of physical length in the 3 dimensions
field_real , [ res ( 3 ) , res ( 2 ) , res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ] , & ! output data, physical length in each dimension in reversed order
1 , res ( 3 ) * res ( 2 ) * ( res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ) , & ! striding, product of physical length in the 3 dimensions
fftw_planner_flag ) ! planner precision
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-11-12 19:44:39 +05:30
! depending on debug options, allocate more memory and create additional plans
2012-07-19 22:54:56 +05:30
if ( debugDivergence ) then
2013-04-10 15:49:16 +05:30
div = fftw_alloc_complex ( int ( res1_red * res ( 2 ) * res ( 3 ) * 3_pInt , C_SIZE_T ) )
call c_f_pointer ( div , divReal , [ res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) , res ( 2 ) , res ( 3 ) , 3 ] )
call c_f_pointer ( div , divFourier , [ res1_red , res ( 2 ) , res ( 3 ) , 3 ] )
planDiv = fftw_plan_many_dft_c2r ( 3 , [ res ( 3 ) , res ( 2 ) , res ( 1 ) ] , 3 , &
divFourier , [ res ( 3 ) , res ( 2 ) , res1_red ] , &
1 , res ( 3 ) * res ( 2 ) * res1_red , &
divReal , [ res ( 3 ) , res ( 2 ) , res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ] , &
1 , res ( 3 ) * res ( 2 ) * ( res ( 1 ) + 2_pInt - mod ( res ( 1 ) , 2_pInt ) ) , &
fftw_planner_flag )
2012-07-19 22:54:56 +05:30
endif
if ( debugFFTW ) then
2012-10-24 17:01:40 +05:30
scalarField_realC = fftw_alloc_complex ( int ( res ( 1 ) * res ( 2 ) * res ( 3 ) , C_SIZE_T ) ) ! allocate data for real representation (no in place transform)
scalarField_fourierC = fftw_alloc_complex ( int ( res ( 1 ) * res ( 2 ) * res ( 3 ) , C_SIZE_T ) ) ! allocate data for fourier representation (no in place transform)
call c_f_pointer ( scalarField_realC , scalarField_real , [ res ( 1 ) , res ( 2 ) , res ( 3 ) ] ) ! place a pointer for a real representation
call c_f_pointer ( scalarField_fourierC , scalarField_fourier , [ res ( 1 ) , res ( 2 ) , res ( 3 ) ] ) ! place a pointer for a fourier representation
2013-04-10 15:49:16 +05:30
planDebugForth = fftw_plan_dft_3d ( res ( 3 ) , res ( 2 ) , res ( 1 ) , & ! reversed order (C style)
2012-10-24 17:01:40 +05:30
scalarField_real , scalarField_fourier , - 1 , fftw_planner_flag ) ! input, output, forward FFT(-1), planner precision
2013-04-10 15:49:16 +05:30
planDebugBack = fftw_plan_dft_3d ( res ( 3 ) , res ( 2 ) , res ( 1 ) , & ! reversed order (C style)
2012-10-24 17:01:40 +05:30
scalarField_fourier , scalarField_real , + 1 , fftw_planner_flag ) ! input, output, backward (1), planner precision
2012-07-19 22:54:56 +05:30
endif
2013-01-10 03:49:32 +05:30
if ( debugGeneral ) write ( 6 , '(/,a)' ) ' FFTW initialized'
flush ( 6 )
2012-07-24 22:37:10 +05:30
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-07-23 15:42:31 +05:30
! calculation of discrete angular frequencies, ordered as in FFTW (wrap around)
do k = 1_pInt , res ( 3 )
k_s ( 3 ) = k - 1_pInt
2012-10-24 17:01:40 +05:30
if ( k > res ( 3 ) / 2_pInt + 1_pInt ) k_s ( 3 ) = k_s ( 3 ) - res ( 3 ) ! running from 0,1,...,N/2,N/2+1,-N/2,-N/2+1,...,-1
2012-07-23 15:42:31 +05:30
do j = 1_pInt , res ( 2 )
k_s ( 2 ) = j - 1_pInt
2012-10-24 17:01:40 +05:30
if ( j > res ( 2 ) / 2_pInt + 1_pInt ) k_s ( 2 ) = k_s ( 2 ) - res ( 2 ) ! running from 0,1,...,N/2,N/2+1,-N/2,-N/2+1,...,-1
2012-07-23 15:42:31 +05:30
do i = 1_pInt , res1_red
2012-10-24 17:01:40 +05:30
k_s ( 1 ) = i - 1_pInt ! symmetry, junst running from 0,1,...,N/2,N/2+1
2013-01-28 21:00:51 +05:30
xi ( 1 : 3 , i , j , k ) = real ( k_s , pReal ) / scaledDim ! if divergence_correction is set, frequencies are calculated on unit length
2012-07-23 15:42:31 +05:30
enddo ; enddo ; enddo
2012-07-24 22:37:10 +05:30
2012-07-19 22:54:56 +05:30
if ( memory_efficient ) then ! allocate just single fourth order tensor
2012-08-09 18:34:56 +05:30
allocate ( gamma_hat ( 3 , 3 , 3 , 3 , 1 , 1 , 1 ) , source = 0.0_pReal )
2012-07-19 22:54:56 +05:30
else ! precalculation of gamma_hat field
2012-10-24 17:01:40 +05:30
allocate ( gamma_hat ( 3 , 3 , 3 , 3 , res1_red , res ( 2 ) , res ( 3 ) ) , source = 0.0_pReal )
2012-07-19 22:54:56 +05:30
endif
2012-08-03 14:55:48 +05:30
2012-10-24 17:01:40 +05:30
end subroutine utilities_init
2012-07-26 19:28:47 +05:30
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief updates references stiffness and potentially precalculated gamma operator
2012-08-09 18:34:56 +05:30
!> @details Sets the current reference stiffness to the stiffness given as an argument.
!> If the gamma operator is precalculated, it is calculated with this stiffness.
!> In case of a on-the-fly calculation, only the reference stiffness is updated.
2012-10-24 17:01:40 +05:30
!> The gamma operator is filtered depening on the filter selected in numerics.
!> Also writes out the current reference stiffness for restart.
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
subroutine utilities_updateGamma ( C , saveReference )
use IO , only : &
IO_write_jobBinaryFile
2012-08-03 14:55:48 +05:30
use numerics , only : &
memory_efficient
2012-10-24 17:01:40 +05:30
use math , only : &
math_inv33
use mesh , only : &
res , &
res1_red
2012-08-03 14:55:48 +05:30
implicit none
2012-10-24 17:01:40 +05:30
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 , 3 , 3 ) :: C !< input stiffness to store as reference stiffness
logical , intent ( in ) :: saveReference !< save reference stiffness to file for restart
real ( pReal ) , dimension ( 3 , 3 ) :: temp33_Real , xiDyad
real ( pReal ) :: filter !< weighting of current component
integer ( pInt ) :: &
i , j , k , &
l , m , n , o
2012-07-26 19:28:47 +05:30
2012-08-03 14:55:48 +05:30
C_ref = C
2012-10-24 17:01:40 +05:30
if ( saveReference ) then
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a)' ) ' writing reference stiffness to file'
flush ( 6 )
2012-10-24 17:01:40 +05:30
call IO_write_jobBinaryFile ( 777 , 'C_ref' , size ( C_ref ) )
write ( 777 , rec = 1 ) C_ref
close ( 777 )
endif
2012-08-03 14:55:48 +05:30
if ( . not . memory_efficient ) then
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 ) ; do i = 1_pInt , res1_red
2012-10-24 17:01:40 +05:30
if ( any ( [ i , j , k ] / = 1_pInt ) ) then ! singular point at xi=(0.0,0.0,0.0) i.e. i=j=k=1
2012-08-03 14:55:48 +05:30
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
xiDyad ( l , m ) = xi ( l , i , j , k ) * xi ( m , i , j , k )
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
2012-11-29 00:16:07 +05:30
temp33_Real ( l , m ) = sum ( C_ref ( l , 1 : 3 , m , 1 : 3 ) * xiDyad )
2012-08-03 14:55:48 +05:30
temp33_Real = math_inv33 ( temp33_Real )
2012-10-24 17:01:40 +05:30
filter = utilities_getFilter ( xi ( 1 : 3 , i , j , k ) ) ! weighting factor computed by getFilter function
2012-08-03 14:55:48 +05:30
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt , n = 1_pInt : 3_pInt , o = 1_pInt : 3_pInt ) &
2012-08-09 18:34:56 +05:30
gamma_hat ( l , m , n , o , i , j , k ) = filter * temp33_Real ( l , n ) * xiDyad ( m , o )
2012-08-03 14:55:48 +05:30
endif
enddo ; enddo ; enddo
2012-10-24 17:01:40 +05:30
gamma_hat ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , 1 , 1 ) = 0.0_pReal ! singular point at xi=(0.0,0.0,0.0) i.e. i=j=k=1
2012-08-03 14:55:48 +05:30
endif
2012-10-24 17:01:40 +05:30
end subroutine utilities_updateGamma
2012-07-26 19:28:47 +05:30
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief forward FFT of data in field_real to field_fourier with highest freqs. removed
2012-11-12 19:44:39 +05:30
!> @detailed Does an unweighted FFT transform from real to complex.
2012-08-09 18:34:56 +05:30
!> In case of debugging the FFT, also one component of the tensor (specified by row and column)
!> is independetly transformed complex to complex and compared to the whole tensor transform
2012-08-03 14:55:48 +05:30
!--------------------------------------------------------------------------------------------------
2013-04-10 15:49:16 +05:30
subroutine utilities_FFTforward ( ) !< @ToDo make row and column between randomly between 1 and 3
2012-10-24 17:01:40 +05:30
use math
2012-08-03 14:55:48 +05:30
use mesh , only : &
2013-01-16 16:10:53 +05:30
scaledDim , &
2012-10-24 17:01:40 +05:30
res , &
res1_red
implicit none
2013-04-10 15:49:16 +05:30
integer ( pInt ) :: row , column ! if debug FFTW, compare 3D array field of row and column
integer ( pInt ) , dimension ( 2 : 3 , 2 ) :: Nyquist ! highest frequencies to be removed (1 if even, 2 if odd)
real ( pReal ) , dimension ( 2 ) :: myRand ! random numbers
2012-07-25 19:31:39 +05:30
!--------------------------------------------------------------------------------------------------
2012-07-19 22:54:56 +05:30
! copy one component of the stress field to to a single FT and check for mismatch
2013-04-10 15:49:16 +05:30
if ( debugFFTW ) then
call random_number ( myRand ) ! two numbers: 0 <= x < 1
row = nint ( myRand ( 1 ) * 2_pReal + 1_pReal , pInt )
column = nint ( myRand ( 2 ) * 2_pReal + 1_pReal , pInt )
scalarField_real = cmplx ( field_real ( 1 : res ( 1 ) , 1 : res ( 2 ) , 1 : res ( 3 ) , row , column ) , 0.0_pReal , pReal ) ! store the selected component
endif
2013-03-07 01:04:30 +05:30
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
! doing the FFT
2013-04-10 15:49:16 +05:30
call fftw_execute_dft_r2c ( planForth , field_real , field_fourier )
2012-07-26 19:28:47 +05:30
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
! comparing 1 and 3x3 FT results
2013-04-10 15:49:16 +05:30
if ( debugFFTW ) then
call fftw_execute_dft ( planDebugForth , scalarField_real , scalarField_fourier )
where ( abs ( scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) ) > tiny ( 1.0_pReal ) ) ! avoid division by zero
scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) = &
( scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) - &
field_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) , row , column ) ) / &
scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) )
else where
scalarField_real = cmplx ( 0.0 , 0.0 , pReal )
end where
write ( 6 , '(/,a,i1,1x,i1,a)' ) ' .. checking FT results of compontent ' , row , column , ' ..'
write ( 6 , '(/,a,2(es11.4,1x))' ) ' max FT relative error = ' , & ! print real and imaginary part seperately
maxval ( real ( scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) ) ) , &
maxval ( aimag ( scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) ) )
flush ( 6 )
endif
2012-10-24 17:01:40 +05:30
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
! removing highest frequencies
2013-04-10 15:49:16 +05:30
Nyquist ( 2 , 1 : 2 ) = [ res ( 2 ) / 2_pInt + 1_pInt , res ( 2 ) / 2_pInt + 1_pInt + mod ( res ( 2 ) , 2_pInt ) ]
Nyquist ( 3 , 1 : 2 ) = [ res ( 3 ) / 2_pInt + 1_pInt , res ( 3 ) / 2_pInt + 1_pInt + mod ( res ( 3 ) , 2_pInt ) ]
if ( res ( 1 ) / = 1_pInt ) & ! do not delete the whole slice in case of 2D calculation
field_fourier ( res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) , 1 : 3 , 1 : 3 ) &
2012-07-30 19:36:22 +05:30
= cmplx ( 0.0_pReal , 0.0_pReal , pReal )
2013-04-10 15:49:16 +05:30
if ( res ( 2 ) / = 1_pInt ) & ! do not delete the whole slice in case of 2D calculation
field_fourier ( 1 : res1_red , Nyquist ( 2 , 1 ) : Nyquist ( 2 , 2 ) , 1 : res ( 3 ) , 1 : 3 , 1 : 3 ) &
2012-07-30 19:36:22 +05:30
= cmplx ( 0.0_pReal , 0.0_pReal , pReal )
2013-04-10 15:49:16 +05:30
if ( res ( 3 ) / = 1_pInt ) & ! do not delete the whole slice in case of 2D calculation
field_fourier ( 1 : res1_red , 1 : res ( 2 ) , Nyquist ( 3 , 1 ) : Nyquist ( 3 , 2 ) , 1 : 3 , 1 : 3 ) &
2012-07-30 19:36:22 +05:30
= cmplx ( 0.0_pReal , 0.0_pReal , pReal )
2012-10-24 17:01:40 +05:30
end subroutine utilities_FFTforward
2012-07-26 19:28:47 +05:30
2012-08-09 18:34:56 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief backward FFT of data in field_fourier to field_real
2012-11-12 19:44:39 +05:30
!> @detailed Does an inverse FFT transform from complex to real
2012-08-09 18:34:56 +05:30
!> In case of debugging the FFT, also one component of the tensor (specified by row and column)
!> is independetly transformed complex to complex and compared to the whole tensor transform
!> results is weighted by number of points stored in wgt
!--------------------------------------------------------------------------------------------------
2013-04-10 15:49:16 +05:30
subroutine utilities_FFTbackward ( )
2012-10-24 17:01:40 +05:30
use math !< must use the whole module for use of FFTW
use mesh , only : &
wgt , &
res , &
res1_red
2012-07-26 19:28:47 +05:30
2012-10-24 17:01:40 +05:30
implicit none
2013-04-10 15:49:16 +05:30
integer ( pInt ) :: row , column !< if debug FFTW, compare 3D array field of row and column
2012-10-24 17:01:40 +05:30
integer ( pInt ) :: i , j , k , m , n
2013-04-10 15:49:16 +05:30
real ( pReal ) , dimension ( 2 ) :: myRand
2012-07-26 19:28:47 +05:30
!--------------------------------------------------------------------------------------------------
2012-11-12 19:44:39 +05:30
! unpack FFT data for conj complex symmetric part. This data is not transformed when using c2r
2013-04-10 15:49:16 +05:30
if ( debugFFTW ) then
call random_number ( myRand ) ! two numbers: 0 <= x < 1
row = nint ( myRand ( 1 ) * 2_pReal + 1_pReal , pInt )
column = nint ( myRand ( 2 ) * 2_pReal + 1_pReal , pInt )
scalarField_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) ) &
= field_fourier ( 1 : res1_red , 1 : res ( 2 ) , 1 : res ( 3 ) , row , column )
do i = 0_pInt , res ( 1 ) / 2_pInt - 2_pInt + mod ( res ( 1 ) , 2_pInt )
2012-10-24 17:01:40 +05:30
m = 1_pInt
do k = 1_pInt , res ( 3 )
n = 1_pInt
do j = 1_pInt , res ( 2 )
scalarField_fourier ( res ( 1 ) - i , j , k ) = conjg ( scalarField_fourier ( 2 + i , n , m ) )
if ( n == 1_pInt ) n = res ( 2 ) + 1_pInt
n = n - 1_pInt
enddo
if ( m == 1_pInt ) m = res ( 3 ) + 1_pInt
m = m - 1_pInt
enddo ; enddo
endif
!--------------------------------------------------------------------------------------------------
! doing the iFFT
2013-04-10 15:49:16 +05:30
call fftw_execute_dft_c2r ( planBack , field_fourier , field_real ) ! back transform of fluct deformation gradient
2012-08-03 14:55:48 +05:30
2012-07-26 19:28:47 +05:30
!--------------------------------------------------------------------------------------------------
! comparing 1 and 3x3 inverse FT results
2013-04-10 15:49:16 +05:30
if ( debugFFTW ) then
call fftw_execute_dft ( planDebugBack , scalarField_fourier , scalarField_real )
where ( abs ( real ( scalarField_real , pReal ) ) > tiny ( 1.0_pReal ) ) ! avoid division by zero
scalarField_real = ( scalarField_real &
- cmplx ( field_real ( 1 : res ( 1 ) , 1 : res ( 2 ) , 1 : res ( 3 ) , row , column ) , 0.0 , pReal ) ) / &
scalarField_real
else where
scalarField_real = cmplx ( 0.0 , 0.0 , pReal )
end where
write ( 6 , '(/,a,i1,1x,i1,a)' ) ' ... checking iFT results of compontent ' , row , column , ' ..'
write ( 6 , '(/,a,es11.4)' ) ' max iFT relative error = ' , maxval ( real ( scalarField_real , pReal ) )
flush ( 6 )
endif
field_real = field_real * wgt ! normalize the result by number of elements
2012-08-03 14:55:48 +05:30
2012-10-24 17:01:40 +05:30
end subroutine utilities_FFTbackward
2012-07-26 19:28:47 +05:30
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
!> @brief doing convolution gamma_hat * field_real, ensuring that average value = fieldAim
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
subroutine utilities_fourierConvolution ( fieldAim )
2012-07-30 19:36:22 +05:30
use numerics , only : &
memory_efficient
2012-10-24 17:01:40 +05:30
use math , only : &
math_inv33
use mesh , only : &
mesh_NcpElems , &
res , &
res1_red
2012-07-26 19:28:47 +05:30
2012-10-24 17:01:40 +05:30
implicit none
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 ) :: fieldAim !< desired average value of the field after convolution
real ( pReal ) , dimension ( 3 , 3 ) :: xiDyad , temp33_Real
real ( pReal ) :: filter !< weighting of current component
2012-11-12 19:44:39 +05:30
complex ( pReal ) , dimension ( 3 , 3 ) :: temp33_complex
2012-10-24 17:01:40 +05:30
integer ( pInt ) :: &
i , j , k , &
l , m , n , o
2012-10-02 20:56:56 +05:30
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a)' ) ' ... doing convolution .....................................................'
flush ( 6 )
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2013-04-18 22:10:49 +05:30
! do the actual spectral method calculation (mechanical equilibrium)
2012-10-24 17:01:40 +05:30
if ( memory_efficient ) then ! memory saving version, on-the-fly calculation of gamma_hat
2012-07-30 19:36:22 +05:30
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 ) ; do i = 1_pInt , res1_red
2012-11-29 00:16:07 +05:30
if ( any ( [ i , j , k ] / = 1_pInt ) ) then ! singular point at xi=(0.0,0.0,0.0) i.e. i=j=k=1
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
xiDyad ( l , m ) = xi ( l , i , j , k ) * xi ( m , i , j , k )
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
temp33_Real ( l , m ) = sum ( C_ref ( l , 1 : 3 , m , 1 : 3 ) * xiDyad )
temp33_Real = math_inv33 ( temp33_Real )
filter = utilities_getFilter ( xi ( 1 : 3 , i , j , k ) ) ! weighting factor computed by getFilter function
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt , n = 1_pInt : 3_pInt , o = 1_pInt : 3_pInt ) &
gamma_hat ( l , m , n , o , 1 , 1 , 1 ) = filter * temp33_Real ( l , n ) * xiDyad ( m , o )
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
temp33_Complex ( l , m ) = sum ( gamma_hat ( l , m , 1 : 3 , 1 : 3 , 1 , 1 , 1 ) * field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) )
field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) = temp33_Complex
2012-07-30 19:36:22 +05:30
endif
enddo ; enddo ; enddo
2012-10-24 17:01:40 +05:30
else ! use precalculated gamma-operator
2012-07-30 19:36:22 +05:30
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 ) ; do i = 1_pInt , res1_red
2013-03-01 15:06:45 +05:30
forall ( l = 1_pInt : 3_pInt , m = 1_pInt : 3_pInt ) &
temp33_Complex ( l , m ) = sum ( gamma_hat ( l , m , 1 : 3 , 1 : 3 , i , j , k ) * field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) )
2012-07-30 19:36:22 +05:30
field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) = temp33_Complex
enddo ; enddo ; enddo
endif
2012-10-24 17:01:40 +05:30
field_fourier ( 1 , 1 , 1 , 1 : 3 , 1 : 3 ) = cmplx ( fieldAim * real ( mesh_NcpElems , pReal ) , 0.0_pReal , pReal ) ! singular point at xi=(0.0,0.0,0.0) i.e. i=j=k=1
2012-07-30 19:36:22 +05:30
2012-10-24 17:01:40 +05:30
end subroutine utilities_fourierConvolution
2012-07-26 19:28:47 +05:30
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief calculate root mean square of divergence of field_fourier
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
real ( pReal ) function utilities_divergenceRMS ( )
use math !< must use the whole module for use of FFTW
use mesh , only : &
wgt , &
res , &
res1_red
implicit none
integer ( pInt ) :: i , j , k
real ( pReal ) :: &
err_real_div_RMS , & !< RMS of divergence in real space
err_div_max , & !< maximum value of divergence in Fourier space
err_real_div_max !< maximum value of divergence in real space
complex ( pReal ) , dimension ( 3 ) :: temp3_complex
2012-07-30 19:36:22 +05:30
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a)' ) ' ... calculating divergence ................................................'
flush ( 6 )
2013-04-10 15:49:16 +05:30
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-07-26 19:28:47 +05:30
! calculating RMS divergence criterion in Fourier space
2012-10-24 17:01:40 +05:30
utilities_divergenceRMS = 0.0_pReal
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 )
2013-04-10 15:49:16 +05:30
do i = 2_pInt , res1_red - 1_pInt ! Has somewhere a conj. complex counterpart. Therefore count it twice.
2012-10-24 17:01:40 +05:30
utilities_divergenceRMS = utilities_divergenceRMS &
2013-04-10 15:49:16 +05:30
+ 2.0_pReal * ( sum ( real ( math_mul33x3_complex ( field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) , & ! (sqrt(real(a)**2 + aimag(a)**2))**2 = real(a)**2 + aimag(a)**2. do not take square root and square again
xi ( 1 : 3 , i , j , k ) ) * TWOPIIMG ) ** 2.0_pReal ) & ! --> sum squared L_2 norm of vector
2012-10-24 17:01:40 +05:30
+ sum ( aimag ( math_mul33x3_complex ( field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) , &
xi ( 1 : 3 , i , j , k ) ) * TWOPIIMG ) ** 2.0_pReal ) )
enddo
2013-04-10 15:49:16 +05:30
utilities_divergenceRMS = utilities_divergenceRMS & ! these two layers (DC and Nyquist) do not have a conjugate complex counterpart (if res(1) /= 1)
+ sum ( real ( math_mul33x3_complex ( field_fourier ( 1 , j , k , 1 : 3 , 1 : 3 ) , &
xi ( 1 : 3 , 1 , j , k ) ) * TWOPIIMG ) ** 2.0_pReal ) &
+ sum ( aimag ( math_mul33x3_complex ( field_fourier ( 1 , j , k , 1 : 3 , 1 : 3 ) , &
xi ( 1 : 3 , 1 , j , k ) ) * TWOPIIMG ) ** 2.0_pReal ) &
+ sum ( real ( math_mul33x3_complex ( field_fourier ( res1_red , j , k , 1 : 3 , 1 : 3 ) , &
xi ( 1 : 3 , res1_red , j , k ) ) * TWOPIIMG ) ** 2.0_pReal ) &
+ sum ( aimag ( math_mul33x3_complex ( field_fourier ( res1_red , j , k , 1 : 3 , 1 : 3 ) , &
xi ( 1 : 3 , res1_red , j , k ) ) * TWOPIIMG ) ** 2.0_pReal )
2012-10-24 17:01:40 +05:30
enddo ; enddo
2013-04-10 15:49:16 +05:30
if ( res ( 1 ) == 1_pInt ) utilities_divergenceRMS = utilities_divergenceRMS * 0.5_pReal ! counted twice in case of res(1) == 1
utilities_divergenceRMS = sqrt ( utilities_divergenceRMS ) * wgt ! RMS in real space calculated with Parsevals theorem from Fourier space
2012-10-24 17:01:40 +05:30
2012-07-19 22:54:56 +05:30
!--------------------------------------------------------------------------------------------------
2012-07-26 19:28:47 +05:30
! calculate additional divergence criteria and report
2013-04-10 15:49:16 +05:30
if ( debugDivergence ) then ! calculate divergence again
2012-10-24 17:01:40 +05:30
err_div_max = 0.0_pReal
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 ) ; do i = 1_pInt , res1_red
2013-04-10 15:49:16 +05:30
temp3_Complex = math_mul33x3_complex ( field_fourier ( i , j , k , 1 : 3 , 1 : 3 ) * wgt , & ! weighting P_fourier
2012-10-24 17:01:40 +05:30
xi ( 1 : 3 , i , j , k ) ) * TWOPIIMG
err_div_max = max ( err_div_max , sum ( abs ( temp3_Complex ) ** 2.0_pReal ) )
2013-04-10 15:49:16 +05:30
divFourier ( i , j , k , 1 : 3 ) = temp3_Complex ! need divergence NOT squared
2012-10-24 17:01:40 +05:30
enddo ; enddo ; enddo
2013-04-10 15:49:16 +05:30
call fftw_execute_dft_c2r ( planDiv , divFourier , divReal ) ! already weighted
2012-07-25 19:31:39 +05:30
2013-04-10 15:49:16 +05:30
err_real_div_RMS = sqrt ( wgt * sum ( divReal ** 2.0_pReal ) ) ! RMS in real space
err_real_div_max = sqrt ( maxval ( sum ( divReal ** 2.0_pReal , dim = 4 ) ) ) ! max in real space
err_div_max = sqrt ( err_div_max ) ! max in Fourier space
2012-10-24 17:01:40 +05:30
2013-03-07 01:04:30 +05:30
write ( 6 , '(/,1x,a,es11.4)' ) 'error divergence FT RMS = ' , utilities_divergenceRMS
2012-10-24 17:01:40 +05:30
write ( 6 , '(1x,a,es11.4)' ) 'error divergence Real RMS = ' , err_real_div_RMS
write ( 6 , '(1x,a,es11.4)' ) 'error divergence FT max = ' , err_div_max
write ( 6 , '(1x,a,es11.4)' ) 'error divergence Real max = ' , err_real_div_max
2013-01-10 03:49:32 +05:30
flush ( 6 )
2012-10-24 17:01:40 +05:30
endif
end function utilities_divergenceRMS
2013-03-06 20:01:13 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief calculate max of curl of field_fourier
!--------------------------------------------------------------------------------------------------
real ( pReal ) function utilities_curlRMS ( )
use math !< must use the whole module for use of FFTW
use mesh , only : &
res , &
res1_red , &
wgt
implicit none
integer ( pInt ) :: i , j , k , l
complex ( pReal ) , dimension ( 3 , 3 ) :: curl_fourier
write ( 6 , '(/,a)' ) ' ... calculating curl ................................................'
flush ( 6 )
!--------------------------------------------------------------------------------------------------
! calculating max curl criterion in Fourier space
utilities_curlRMS = 0.0_pReal
do k = 1_pInt , res ( 3 ) ; do j = 1_pInt , res ( 2 ) ;
do i = 2_pInt , res1_red - 1_pInt
do l = 1_pInt , 3_pInt
curl_fourier ( l , 1 ) = ( field_fourier ( i , j , k , l , 3 ) * xi ( 2 , i , j , k ) &
- field_fourier ( i , j , k , l , 2 ) * xi ( 3 , i , j , k ) ) * TWOPIIMG
curl_fourier ( l , 2 ) = ( - field_fourier ( i , j , k , l , 3 ) * xi ( 1 , i , j , k ) &
+ field_fourier ( i , j , k , l , 1 ) * xi ( 3 , i , j , k ) ) * TWOPIIMG
curl_fourier ( l , 3 ) = ( field_fourier ( i , j , k , l , 2 ) * xi ( 1 , i , j , k ) &
- field_fourier ( i , j , k , l , 1 ) * xi ( 2 , i , j , k ) ) * TWOPIIMG
enddo
utilities_curlRMS = utilities_curlRMS + &
2.0_pReal * sum ( real ( curl_fourier ) ** 2.0_pReal + aimag ( curl_fourier ) ** 2.0_pReal )
enddo
do l = 1_pInt , 3_pInt
curl_fourier = ( field_fourier ( 1 , j , k , l , 3 ) * xi ( 2 , 1 , j , k ) &
- field_fourier ( 1 , j , k , l , 2 ) * xi ( 3 , 1 , j , k ) ) * TWOPIIMG
curl_fourier = ( - field_fourier ( 1 , j , k , l , 3 ) * xi ( 1 , 1 , j , k ) &
+ field_fourier ( 1 , j , k , l , 1 ) * xi ( 3 , 1 , j , k ) ) * TWOPIIMG
curl_fourier = ( field_fourier ( 1 , j , k , l , 2 ) * xi ( 1 , 1 , j , k ) &
- field_fourier ( 1 , j , k , l , 1 ) * xi ( 2 , 1 , j , k ) ) * TWOPIIMG
enddo
utilities_curlRMS = utilities_curlRMS + &
2.0_pReal * sum ( real ( curl_fourier ) ** 2.0_pReal + aimag ( curl_fourier ) ** 2.0_pReal )
do l = 1_pInt , 3_pInt
curl_fourier = ( field_fourier ( res1_red , j , k , l , 3 ) * xi ( 2 , res1_red , j , k ) &
- field_fourier ( res1_red , j , k , l , 2 ) * xi ( 3 , res1_red , j , k ) ) * TWOPIIMG
curl_fourier = ( - field_fourier ( res1_red , j , k , l , 3 ) * xi ( 1 , res1_red , j , k ) &
+ field_fourier ( res1_red , j , k , l , 1 ) * xi ( 3 , res1_red , j , k ) ) * TWOPIIMG
curl_fourier = ( field_fourier ( res1_red , j , k , l , 2 ) * xi ( 1 , res1_red , j , k ) &
- field_fourier ( res1_red , j , k , l , 1 ) * xi ( 2 , res1_red , j , k ) ) * TWOPIIMG
enddo
utilities_curlRMS = utilities_curlRMS + &
2.0_pReal * sum ( real ( curl_fourier ) ** 2.0_pReal + aimag ( curl_fourier ) ** 2.0_pReal )
enddo ; enddo
utilities_curlRMS = sqrt ( utilities_curlRMS ) * wgt
end function utilities_curlRMS
2012-07-24 22:37:10 +05:30
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
2013-04-18 22:10:49 +05:30
!> @brief calculates mask compliance tensor used to adjust F to fullfill stress BC
2012-07-30 19:36:22 +05:30
!--------------------------------------------------------------------------------------------------
2012-10-24 17:01:40 +05:30
function utilities_maskedCompliance ( rot_BC , mask_stress , C )
use IO , only : &
IO_error
use math , only : &
math_Plain3333to99 , &
math_plain99to3333 , &
math_rotate_forward3333 , &
math_rotate_forward33 , &
math_invert
2012-08-28 22:29:45 +05:30
implicit none
2012-10-24 17:01:40 +05:30
real ( pReal ) , dimension ( 3 , 3 , 3 , 3 ) :: utilities_maskedCompliance !< masked compliance
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 , 3 , 3 ) :: C !< current average stiffness
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 ) :: rot_BC !< rotation of load frame
logical , intent ( in ) , dimension ( 3 , 3 ) :: mask_stress !< mask of stress BC
2012-08-28 22:29:45 +05:30
integer ( pInt ) :: j , k , m , n
2012-10-19 14:14:21 +05:30
logical , dimension ( 9 ) :: mask_stressVector
2012-07-30 19:36:22 +05:30
real ( pReal ) , dimension ( 9 , 9 ) :: temp99_Real
integer ( pInt ) :: size_reduced = 0_pInt
2012-10-24 17:01:40 +05:30
real ( pReal ) , dimension ( : , : ) , allocatable :: &
s_reduced , & !< reduced compliance matrix (depending on number of stress BC)
c_reduced , & !< reduced stiffness (depending on number of stress BC)
sTimesC !< temp variable to check inversion
2012-07-30 19:36:22 +05:30
logical :: errmatinv
2012-08-28 22:29:45 +05:30
character ( len = 1024 ) :: formatString
2012-07-25 19:31:39 +05:30
2012-10-19 14:14:21 +05:30
mask_stressVector = reshape ( transpose ( mask_stress ) , [ 9 ] )
size_reduced = int ( count ( mask_stressVector ) , pInt )
2012-07-30 19:36:22 +05:30
if ( size_reduced > 0_pInt ) then
allocate ( c_reduced ( size_reduced , size_reduced ) , source = 0.0_pReal )
allocate ( s_reduced ( size_reduced , size_reduced ) , source = 0.0_pReal )
2012-08-28 22:29:45 +05:30
allocate ( sTimesC ( size_reduced , size_reduced ) , source = 0.0_pReal )
2012-07-30 19:36:22 +05:30
2012-10-24 17:01:40 +05:30
temp99_Real = math_Plain3333to99 ( math_rotate_forward3333 ( C , rot_BC ) )
2013-01-10 03:49:32 +05:30
if ( debugGeneral ) then
write ( 6 , '(/,a)' ) ' ... updating masked compliance ............................................'
2013-04-18 22:10:49 +05:30
write ( 6 , '(/,a,/,9(9(2x,f12.7,1x)/))' , advance = 'no' ) ' Stiffness C (load) / GPa =' , &
2012-10-24 17:01:40 +05:30
transpose ( temp99_Real ) / 1.e9_pReal
2013-01-10 03:49:32 +05:30
flush ( 6 )
endif
2012-10-24 17:01:40 +05:30
k = 0_pInt ! calculate reduced stiffness
2012-07-26 19:28:47 +05:30
do n = 1_pInt , 9_pInt
if ( mask_stressVector ( n ) ) then
k = k + 1_pInt
j = 0_pInt
do m = 1_pInt , 9_pInt
if ( mask_stressVector ( m ) ) then
j = j + 1_pInt
2012-07-30 19:36:22 +05:30
c_reduced ( k , j ) = temp99_Real ( n , m )
endif ; enddo ; endif ; enddo
2012-08-28 22:29:45 +05:30
call math_invert ( size_reduced , c_reduced , s_reduced , errmatinv ) ! invert reduced stiffness
2012-12-14 23:00:22 +05:30
if ( errmatinv ) call IO_error ( error_ID = 400_pInt , ext_msg = 'utilities_maskedCompliance' )
2012-10-24 17:01:40 +05:30
temp99_Real = 0.0_pReal ! fill up compliance with zeros
2012-07-30 19:36:22 +05:30
k = 0_pInt
do n = 1_pInt , 9_pInt
if ( mask_stressVector ( n ) ) then
k = k + 1_pInt
j = 0_pInt
do m = 1_pInt , 9_pInt
if ( mask_stressVector ( m ) ) then
j = j + 1_pInt
temp99_Real ( n , m ) = s_reduced ( k , j )
endif ; enddo ; endif ; enddo
2013-03-07 01:04:30 +05:30
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
2013-03-07 01:04:30 +05:30
! check if inversion was successful
2012-08-28 22:29:45 +05:30
sTimesC = matmul ( c_reduced , s_reduced )
do m = 1_pInt , size_reduced
do n = 1_pInt , size_reduced
2012-10-24 17:01:40 +05:30
if ( m == n . and . abs ( sTimesC ( m , n ) ) > ( 1.0_pReal + 1 0.0e-12_pReal ) ) errmatinv = . true . ! diagonal elements of S*C should be 1
if ( m / = n . and . abs ( sTimesC ( m , n ) ) > ( 0.0_pReal + 1 0.0e-12_pReal ) ) errmatinv = . true . ! off diagonal elements of S*C should be 0
2012-08-28 22:29:45 +05:30
enddo
enddo
2012-10-24 17:01:40 +05:30
if ( debugGeneral . or . errmatinv ) then ! report
2012-08-28 22:29:45 +05:30
write ( formatString , '(I16.16)' ) size_reduced
2013-01-10 03:49:32 +05:30
formatString = '(/,a,/,' / / trim ( formatString ) / / '(' / / trim ( formatString ) / / '(2x,es9.2,1x)/))'
2013-04-18 22:10:49 +05:30
write ( 6 , trim ( formatString ) , advance = 'no' ) ' C * S (load) ' , &
transpose ( matmul ( c_reduced , s_reduced ) )
write ( 6 , trim ( formatString ) , advance = 'no' ) ' S (load) ' , transpose ( s_reduced )
2012-08-28 22:29:45 +05:30
endif
2012-12-14 23:00:22 +05:30
if ( errmatinv ) call IO_error ( error_ID = 400_pInt , ext_msg = 'utilities_maskedCompliance' )
2012-07-30 19:36:22 +05:30
deallocate ( c_reduced )
deallocate ( s_reduced )
2012-08-28 22:29:45 +05:30
deallocate ( sTimesC )
2012-07-30 19:36:22 +05:30
else
temp99_real = 0.0_pReal
endif
2012-10-24 17:01:40 +05:30
if ( debugGeneral ) & ! report
2013-04-18 22:10:49 +05:30
write ( 6 , '(/,a,/,9(9(2x,f12.7,1x)/),/)' , advance = 'no' ) ' Masked Compliance (load) * GPa =' , &
2013-01-10 03:49:32 +05:30
transpose ( temp99_Real * 1.e9_pReal )
flush ( 6 )
2012-10-24 17:01:40 +05:30
utilities_maskedCompliance = math_Plain99to3333 ( temp99_Real )
end function utilities_maskedCompliance
!--------------------------------------------------------------------------------------------------
!> @brief calculates constitutive response
!--------------------------------------------------------------------------------------------------
2012-11-14 20:08:10 +05:30
subroutine utilities_constitutiveResponse ( F_lastInc , F , temperature , timeinc , &
2013-03-06 20:01:13 +05:30
P , C_volAvg , C_minmaxAvg , P_av , forwardData , rotation_BC )
2012-10-24 17:01:40 +05:30
use debug , only : &
debug_reset , &
debug_info
use math , only : &
math_transpose33 , &
2013-03-07 01:04:30 +05:30
math_rotate_forward33 , &
math_det33
2012-10-24 17:01:40 +05:30
use FEsolving , only : &
restartWrite
use mesh , only : &
res , &
2013-01-24 01:26:45 +05:30
wgt , &
mesh_NcpElems
2012-10-24 17:01:40 +05:30
use CPFEM , only : &
2013-03-01 17:18:29 +05:30
CPFEM_general , &
CPFEM_COLLECT , &
CPFEM_CALCRESULTS , &
CPFEM_AGERESULTS , &
CPFEM_BACKUPJACOBIAN , &
CPFEM_RESTOREJACOBIAN
2013-01-24 01:26:45 +05:30
use homogenization , only : &
materialpoint_F0 , &
materialpoint_F , &
materialpoint_Temperature , &
materialpoint_P , &
materialpoint_dPdF
2012-10-24 17:01:40 +05:30
implicit none
2012-11-12 19:44:39 +05:30
real ( pReal ) , intent ( inout ) :: temperature !< temperature (no field)
2012-10-24 17:01:40 +05:30
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: &
F_lastInc , & !< target deformation gradient
F !< previous deformation gradient
real ( pReal ) , intent ( in ) :: timeinc !< loading time
logical , intent ( in ) :: forwardData !< age results
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 ) :: rotation_BC !< rotation of load frame
2013-03-06 20:01:13 +05:30
real ( pReal ) , intent ( out ) , dimension ( 3 , 3 , 3 , 3 ) :: C_volAvg , C_minmaxAvg !< average stiffness
2012-10-24 17:01:40 +05:30
real ( pReal ) , intent ( out ) , dimension ( 3 , 3 ) :: P_av !< average PK stress
real ( pReal ) , intent ( out ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: P !< PK stress
integer ( pInt ) :: &
calcMode , & !< CPFEM mode for calculation
2013-04-10 15:49:16 +05:30
collectMode , & !< CPFEM mode for collection
j , k
2013-03-06 20:01:13 +05:30
real ( pReal ) , dimension ( 3 , 3 , 3 , 3 ) :: max_dPdF , min_dPdF
2013-04-10 15:49:16 +05:30
real ( pReal ) :: max_dPdF_norm , min_dPdF_norm , defgradDetMin , defgradDetMax , defgradDet
2012-10-24 17:01:40 +05:30
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a)' ) ' ... evaluating constitutive response ......................................'
2013-03-01 17:18:29 +05:30
calcMode = CPFEM_CALCRESULTS
collectMode = CPFEM_COLLECT
2012-10-24 17:01:40 +05:30
if ( forwardData ) then ! aging results
2013-03-01 17:18:29 +05:30
calcMode = ior ( calcMode , CPFEM_AGERESULTS )
collectMode = ior ( collectMode , CPFEM_BACKUPJACOBIAN )
2012-10-24 17:01:40 +05:30
endif
if ( cutBack ) then ! restore saved variables
2013-03-01 17:18:29 +05:30
collectMode = ior ( collectMode , CPFEM_RESTOREJACOBIAN )
collectMode = iand ( collectMode , not ( CPFEM_BACKUPJACOBIAN ) )
calcMode = iand ( calcMode , not ( CPFEM_AGERESULTS ) )
2012-10-24 17:01:40 +05:30
endif
2013-03-07 01:04:30 +05:30
2013-04-09 15:38:00 +05:30
call CPFEM_general ( collectMode , F_lastInc ( 1 : 3 , 1 : 3 , 1 , 1 , 1 ) , F ( 1 : 3 , 1 : 3 , 1 , 1 , 1 ) , & ! collect mode handles Jacobian backup / restoration
temperature , timeinc , 1_pInt , 1_pInt )
materialpoint_F0 = reshape ( F_lastInc , [ 3 , 3 , 1 , mesh_NcpElems ] )
materialpoint_F = reshape ( F , [ 3 , 3 , 1 , mesh_NcpElems ] )
materialpoint_Temperature = temperature
call debug_reset ( )
2012-11-12 19:44:39 +05:30
!--------------------------------------------------------------------------------------------------
! calculate bounds of det(F) and report
2013-03-07 01:04:30 +05:30
if ( debugGeneral ) then
defgradDetMax = - huge ( 1.0_pReal )
defgradDetMin = + huge ( 1.0_pReal )
2013-04-09 15:38:00 +05:30
do j = 1_pInt , mesh_NcpElems
defgradDet = math_det33 ( materialpoint_F ( 1 : 3 , 1 : 3 , 1 , j ) )
2013-03-07 01:04:30 +05:30
defgradDetMax = max ( defgradDetMax , defgradDet )
defgradDetMin = min ( defgradDetMin , defgradDet )
2013-04-09 15:38:00 +05:30
end do
2013-03-07 01:04:30 +05:30
write ( 6 , '(a,1x,es11.4)' ) ' max determinant of deformation =' , defgradDetMax
write ( 6 , '(a,1x,es11.4)' ) ' min determinant of deformation =' , defgradDetMin
flush ( 6 )
endif
2013-04-09 15:38:00 +05:30
2013-01-24 01:26:45 +05:30
call CPFEM_general ( calcMode , F_lastInc ( 1 : 3 , 1 : 3 , 1 , 1 , 1 ) , F ( 1 : 3 , 1 : 3 , 1 , 1 , 1 ) , & ! first call calculates everything
2013-04-09 15:38:00 +05:30
temperature , timeinc , 1_pInt , 1_pInt )
2013-03-06 20:01:13 +05:30
max_dPdF = 0.0_pReal
max_dPdF_norm = 0.0_pReal
min_dPdF = huge ( 1.0_pReal )
min_dPdF_norm = huge ( 1.0_pReal )
2013-03-07 01:04:30 +05:30
do k = 1_pInt , mesh_NcpElems
2013-03-06 20:01:13 +05:30
if ( max_dPdF_norm < sum ( materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k ) ** 2.0_pReal ) ) then
max_dPdF = materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k )
max_dPdF_norm = sum ( materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k ) ** 2.0_pReal )
endif
if ( min_dPdF_norm > sum ( materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k ) ** 2.0_pReal ) ) then
min_dPdF = materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k )
min_dPdF_norm = sum ( materialpoint_dPdF ( 1 : 3 , 1 : 3 , 1 : 3 , 1 : 3 , 1 , k ) ** 2.0_pReal )
endif
2013-04-09 15:38:00 +05:30
end do
2013-01-24 01:26:45 +05:30
P = reshape ( materialpoint_P , [ 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ] )
2013-03-06 20:01:13 +05:30
C_volAvg = sum ( sum ( materialpoint_dPdF , dim = 6 ) , dim = 5 ) * wgt
C_minmaxAvg = 0.5_pReal * ( max_dPdF + min_dPdF )
2012-10-24 17:01:40 +05:30
call debug_info ( )
restartWrite = . false . ! reset restartWrite status
cutBack = . false . ! reset cutBack status
2012-11-28 20:34:05 +05:30
P_av = sum ( sum ( sum ( P , dim = 5 ) , dim = 4 ) , dim = 3 ) * wgt ! average of P
if ( debugRotation ) &
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a,/,3(3(2x,f12.7,1x)/))' , advance = 'no' ) ' Piola--Kirchhoff stress (lab) / MPa =' , &
2012-11-28 20:34:05 +05:30
math_transpose33 ( P_av ) / 1.e6_pReal
P_av = math_rotate_forward33 ( P_av , rotation_BC )
2013-01-10 03:49:32 +05:30
write ( 6 , '(/,a,/,3(3(2x,f12.7,1x)/))' , advance = 'no' ) ' Piola--Kirchhoff stress / MPa =' , &
2012-10-24 17:01:40 +05:30
math_transpose33 ( P_av ) / 1.e6_pReal
2013-01-10 03:49:32 +05:30
2012-10-24 17:01:40 +05:30
end subroutine utilities_constitutiveResponse
2012-07-30 19:36:22 +05:30
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief calculates forward rate, either guessing or just add delta/timeinc
!--------------------------------------------------------------------------------------------------
2012-11-28 20:34:05 +05:30
pure function utilities_calculateRate ( avRate , timeinc_old , guess , field_lastInc , field )
2012-10-24 17:01:40 +05:30
use mesh , only : &
res
2012-08-28 22:29:45 +05:30
implicit none
2012-11-28 20:34:05 +05:30
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 ) :: avRate !< homogeneous addon
2012-10-24 17:01:40 +05:30
real ( pReal ) , intent ( in ) :: &
2012-11-09 01:02:00 +05:30
timeinc_old !< timeinc of last step
logical , intent ( in ) :: &
guess !< guess along former trajectory
2012-10-24 17:01:40 +05:30
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: &
field_lastInc , & !< data of previous step
field !< data of current step
real ( pReal ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: utilities_calculateRate
2012-08-03 14:55:48 +05:30
2012-11-12 19:44:39 +05:30
if ( guess ) then
2012-10-24 17:01:40 +05:30
utilities_calculateRate = ( field - field_lastInc ) / timeinc_old
2012-08-09 18:34:56 +05:30
else
2012-11-28 20:34:05 +05:30
utilities_calculateRate = spread ( spread ( spread ( avRate , 3 , res ( 1 ) ) , 4 , res ( 2 ) ) , 5 , res ( 3 ) )
2012-08-09 18:34:56 +05:30
endif
2012-10-24 17:01:40 +05:30
end function utilities_calculateRate
2012-10-02 20:56:56 +05:30
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
2013-01-08 15:42:03 +05:30
!> @brief forwards a field with a pointwise given rate, if aim is given,
!> ensures that the average matches the aim
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
2013-01-08 15:42:03 +05:30
pure function utilities_forwardField ( timeinc , field_lastInc , rate , aim )
2012-10-24 17:01:40 +05:30
use mesh , only : &
res , &
wgt
2012-10-02 20:56:56 +05:30
implicit none
2013-01-08 15:42:03 +05:30
real ( pReal ) , intent ( in ) :: &
timeinc !< timeinc of current step
real ( pReal ) , intent ( in ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: &
field_lastInc , & !< initial field
2012-10-24 17:01:40 +05:30
rate !< rate by which to forward
2013-01-08 15:42:03 +05:30
real ( pReal ) , intent ( in ) , optional , dimension ( 3 , 3 ) :: &
aim !< average field value aim
real ( pReal ) , dimension ( 3 , 3 , res ( 1 ) , res ( 2 ) , res ( 3 ) ) :: utilities_forwardField
real ( pReal ) , dimension ( 3 , 3 ) :: fieldDiff !< <a + adot*t> - aim
2012-08-07 22:53:13 +05:30
2012-10-24 17:01:40 +05:30
utilities_forwardField = field_lastInc + rate * timeinc
2013-01-08 15:42:03 +05:30
if ( present ( aim ) ) then !< correct to match average
fieldDiff = sum ( sum ( sum ( utilities_forwardField , dim = 5 ) , dim = 4 ) , dim = 3 ) * wgt - aim
utilities_forwardField = utilities_forwardField - &
2012-10-02 20:56:56 +05:30
spread ( spread ( spread ( fieldDiff , 3 , res ( 1 ) ) , 4 , res ( 2 ) ) , 5 , res ( 3 ) )
2013-01-08 15:42:03 +05:30
endif
2012-10-02 20:56:56 +05:30
2012-10-24 17:01:40 +05:30
end function utilities_forwardField
2012-10-02 20:56:56 +05:30
2012-08-07 22:53:13 +05:30
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief calculates filter for fourier convolution depending on type given in numerics.config
!--------------------------------------------------------------------------------------------------
real ( pReal ) function utilities_getFilter ( k )
use IO , only : &
IO_error
use numerics , only : &
2012-08-07 22:53:13 +05:30
myfilter
2012-10-24 17:01:40 +05:30
use mesh , only : &
res
use math , only : &
PI
2012-08-07 22:53:13 +05:30
2012-10-24 17:01:40 +05:30
implicit none
real ( pReal ) , intent ( in ) , dimension ( 3 ) :: k !< indices of frequency
2012-08-07 22:53:13 +05:30
2013-03-27 17:58:55 +05:30
utilities_getFilter = 1.0_pReal
2012-10-24 17:01:40 +05:30
select case ( myfilter )
2013-03-28 16:07:00 +05:30
case ( 'none' ) !< default is already nothing (1.0_pReal)
2012-10-24 17:01:40 +05:30
case ( 'cosine' ) !< cosine curve with 1 for avg and zero for highest freq
utilities_getFilter = ( 1.0_pReal + cos ( PI * k ( 3 ) / res ( 3 ) ) ) &
* ( 1.0_pReal + cos ( PI * k ( 2 ) / res ( 2 ) ) ) &
* ( 1.0_pReal + cos ( PI * k ( 1 ) / res ( 1 ) ) ) / 8.0_pReal
2012-08-29 00:49:47 +05:30
case default
call IO_error ( error_ID = 892_pInt , ext_msg = trim ( myfilter ) )
2012-08-07 22:53:13 +05:30
end select
2012-10-24 17:01:40 +05:30
end function utilities_getFilter
2012-08-03 14:55:48 +05:30
2012-10-02 20:56:56 +05:30
2012-10-24 17:01:40 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief cleans up
!--------------------------------------------------------------------------------------------------
subroutine utilities_destroy ( )
use math
2013-03-28 16:07:00 +05:30
2012-10-24 17:01:40 +05:30
implicit none
2012-07-25 19:31:39 +05:30
2013-04-10 15:49:16 +05:30
if ( debugDivergence ) call fftw_destroy_plan ( planDiv )
if ( debugFFTW ) call fftw_destroy_plan ( planDebugForth )
if ( debugFFTW ) call fftw_destroy_plan ( planDebugBack )
call fftw_destroy_plan ( planForth )
call fftw_destroy_plan ( planBack )
2012-07-25 19:31:39 +05:30
2012-10-24 17:01:40 +05:30
end subroutine utilities_destroy
2012-10-02 20:56:56 +05:30
2012-10-24 17:01:40 +05:30
end module DAMASK_spectral_utilities