avoiding numerical errors (if quaternion norm is > 1.)

use consistently "self" instead of "this"
function for misorientation
This commit is contained in:
Martin Diehl 2019-02-01 20:17:12 +01:00
parent 878331e5e9
commit 08009079ff
2 changed files with 53 additions and 45 deletions

View File

@ -65,17 +65,9 @@ module rotations
!------------------------------------------ !------------------------------------------
procedure, public :: rotVector procedure, public :: rotVector
procedure, public :: rotTensor procedure, public :: rotTensor
procedure, public :: misorientation
end type rotation end type rotation
public :: &
asQuaternion, &
asEulerAngles, &
asAxisAnglePair, &
asRotationMatrix, &
asRodriguesFrankVector, &
asHomochoric, &
fromRotationMatrix, &
rotVector, &
rotTensor
contains contains
@ -83,76 +75,76 @@ contains
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
! Return rotation in different represenations ! Return rotation in different represenations
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asQuaternion(this) function asQuaternion(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(4) :: asQuaternion real(pReal), dimension(4) :: asQuaternion
asQuaternion = [this%q%w, this%q%x, this%q%y, this%q%z] asQuaternion = [self%q%w, self%q%x, self%q%y, self%q%z]
end function asQuaternion end function asQuaternion
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asEulerAngles(this) function asEulerAngles(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(3) :: asEulerAngles real(pReal), dimension(3) :: asEulerAngles
asEulerAngles = qu2eu(this%q) asEulerAngles = qu2eu(self%q)
end function asEulerAngles end function asEulerAngles
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asAxisAnglePair(this) function asAxisAnglePair(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(4) :: asAxisAnglePair real(pReal), dimension(4) :: asAxisAnglePair
asAxisAnglePair = qu2ax(this%q) asAxisAnglePair = qu2ax(self%q)
end function asAxisAnglePair end function asAxisAnglePair
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asRotationMatrix(this) function asRotationMatrix(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(3,3) :: asRotationMatrix real(pReal), dimension(3,3) :: asRotationMatrix
asRotationMatrix = qu2om(this%q) asRotationMatrix = qu2om(self%q)
end function asRotationMatrix end function asRotationMatrix
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asRodriguesFrankVector(this) function asRodriguesFrankVector(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(4) :: asRodriguesFrankVector real(pReal), dimension(4) :: asRodriguesFrankVector
asRodriguesFrankVector = qu2ro(this%q) asRodriguesFrankVector = qu2ro(self%q)
end function asRodriguesFrankVector end function asRodriguesFrankVector
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function asHomochoric(this) function asHomochoric(self)
implicit none implicit none
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), dimension(3) :: asHomochoric real(pReal), dimension(3) :: asHomochoric
asHomochoric = qu2ho(this%q) asHomochoric = qu2ho(self%q)
end function asHomochoric end function asHomochoric
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
! Initialize rotation from different representations ! Initialize rotation from different representations
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
subroutine fromRotationMatrix(this,om) subroutine fromRotationMatrix(self,om)
implicit none implicit none
class(rotation), intent(out) :: this class(rotation), intent(out) :: self
real(pReal), dimension(3,3), intent(in) :: om real(pReal), dimension(3,3), intent(in) :: om
this%q = om2qu(om) self%q = om2qu(om)
end subroutine end subroutine
@ -162,13 +154,13 @@ end subroutine
!> @brief rotate a vector passively (default) or actively !> @brief rotate a vector passively (default) or actively
!> @details: rotation is based on unit quaternion or rotation matrix (fallback) !> @details: rotation is based on unit quaternion or rotation matrix (fallback)
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function rotVector(this,v,active) function rotVector(self,v,active)
use prec, only: & use prec, only: &
dEq dEq
implicit none implicit none
real(pReal), dimension(3) :: rotVector real(pReal), dimension(3) :: rotVector
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), intent(in), dimension(3) :: v real(pReal), intent(in), dimension(3) :: v
logical, intent(in), optional :: active logical, intent(in), optional :: active
@ -176,16 +168,16 @@ function rotVector(this,v,active)
if (dEq(norm2(v),1.0_pReal,tol=1.0e-15_pReal)) then if (dEq(norm2(v),1.0_pReal,tol=1.0e-15_pReal)) then
passive: if (merge(.not. active, .true., present(active))) then ! ToDo: not save (PGI) passive: if (merge(.not. active, .true., present(active))) then ! ToDo: not save (PGI)
q = this%q * (quaternion([0.0_pReal, v(1), v(2), v(3)]) * conjg(this%q) ) q = self%q * (quaternion([0.0_pReal, v(1), v(2), v(3)]) * conjg(self%q) )
else passive else passive
q = conjg(this%q) * (quaternion([0.0_pReal, v(1), v(2), v(3)]) * this%q ) q = conjg(self%q) * (quaternion([0.0_pReal, v(1), v(2), v(3)]) * self%q )
endif passive endif passive
rotVector = [q%x,q%y,q%z] rotVector = [q%x,q%y,q%z]
else else
passive2: if (merge(.not. active, .true., present(active))) then ! ToDo: not save (PGI) passive2: if (merge(.not. active, .true., present(active))) then ! ToDo: not save (PGI)
rotVector = matmul(this%asRotationMatrix(),v) rotVector = matmul(self%asRotationMatrix(),v)
else passive2 else passive2
rotVector = matmul(transpose(this%asRotationMatrix()),v) rotVector = matmul(transpose(self%asRotationMatrix()),v)
endif passive2 endif passive2
endif endif
@ -197,24 +189,37 @@ end function rotVector
!> @brief rotate a second rank tensor passively (default) or actively !> @brief rotate a second rank tensor passively (default) or actively
!> @details: rotation is based on rotation matrix !> @details: rotation is based on rotation matrix
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
function rotTensor(this,m,active) function rotTensor(self,m,active)
implicit none implicit none
real(pReal), dimension(3,3) :: rotTensor real(pReal), dimension(3,3) :: rotTensor
class(rotation), intent(in) :: this class(rotation), intent(in) :: self
real(pReal), intent(in), dimension(3,3) :: m real(pReal), intent(in), dimension(3,3) :: m
logical, intent(in), optional :: active logical, intent(in), optional :: active
passive: if (merge(.not. active, .true., present(active))) then passive: if (merge(.not. active, .true., present(active))) then
rotTensor = matmul(matmul(this%asRotationMatrix(),m),transpose(this%asRotationMatrix())) rotTensor = matmul(matmul(self%asRotationMatrix(),m),transpose(self%asRotationMatrix()))
else passive else passive
rotTensor = matmul(matmul(transpose(this%asRotationMatrix()),m),this%asRotationMatrix()) rotTensor = matmul(matmul(transpose(self%asRotationMatrix()),m),self%asRotationMatrix())
endif passive endif passive
end function rotTensor end function rotTensor
!---------------------------------------------------------------------------------------------------
!> @brief misorientation
!---------------------------------------------------------------------------------------------------
function misorientation(self,other)
implicit none
type(rotation) :: misorientation
class(rotation), intent(in) :: self, other
misorientation%q = conjg(self%q) * other%q !ToDo: this is the convention used in math
end function misorientation
!--------------------------------------------------------------------------------------------------- !---------------------------------------------------------------------------------------------------
! The following routines convert between different representations ! The following routines convert between different representations
@ -760,7 +765,8 @@ pure function qu2ax(qu) result(ax)
dEq0, & dEq0, &
dNeq0 dNeq0
use math, only: & use math, only: &
PI PI, &
math_clip
implicit none implicit none
type(quaternion), intent(in) :: qu type(quaternion), intent(in) :: qu
@ -768,7 +774,7 @@ pure function qu2ax(qu) result(ax)
real(pReal) :: omega, s real(pReal) :: omega, s
omega = 2.0 * acos(qu%w) omega = 2.0 * acos(math_clip(qu%w,0.0_pReal,1.0_pReal))
! if the angle equals zero, then we return the rotation axis as [001] ! if the angle equals zero, then we return the rotation axis as [001]
if (dEq0(omega)) then if (dEq0(omega)) then
ax = [ 0.0, 0.0, 1.0, 0.0 ] ax = [ 0.0, 0.0, 1.0, 0.0 ]
@ -818,6 +824,8 @@ end function qu2ro
pure function qu2ho(qu) result(ho) pure function qu2ho(qu) result(ho)
use prec, only: & use prec, only: &
dEq0 dEq0
use math, only: &
math_clip
implicit none implicit none
type(quaternion), intent(in) :: qu type(quaternion), intent(in) :: qu
@ -825,7 +833,7 @@ pure function qu2ho(qu) result(ho)
real(pReal) :: omega, f real(pReal) :: omega, f
omega = 2.0 * acos(qu%w) omega = 2.0 * acos(math_clip(qu%w,0.0_pReal,1.0_pReal))
if (dEq0(omega)) then if (dEq0(omega)) then
ho = [ 0.0, 0.0, 0.0 ] ho = [ 0.0, 0.0, 0.0 ]