10 real(kind=kreal),
parameter,
private :: pi=3.14159265358979d0
16 integer :: var,dimn,syze,bite
18 var = var + dimn*syze*bite
24 integer,
intent(in) :: n
25 integer,
intent(in),
optional :: n1
26 character(len=*),
intent(inout) :: fname
28 character(len=128) :: tmpname, tmp
30 npos = scan( fname,
'.')
31 nlen = len_trim( fname )
32 if( nlen>128 ) stop
"String too long(>128) in append_int2name"
33 if( n>100000 ) stop
"Integer too big>100000 in append_int2name"
36 write( fname,
'(a,i6)') fname(1:nlen),n
38 write( tmp,
'(i6,a)') n,tmpname(npos:nlen)
39 fname = tmpname(1:npos-1) // adjustl(tmp)
41 if(
present(n1).and.n1/=0)
then
43 fname = fname(1:len_trim(fname))//
'.'//adjustl(tmp)
49 integer,
intent(in) :: iin
50 integer,
pointer :: carray(:)
53 integer,
pointer :: dumarray(:) => null()
54 if( .not.
associated(carray) )
then
58 oldsize =
size( carray )
59 allocate( dumarray(oldsize) )
61 dumarray(i) = carray(i)
64 allocate( carray(oldsize+1) )
66 carray(i) = dumarray(i)
68 carray(oldsize+1) = iin
70 if(
associated(dumarray) )
deallocate( dumarray )
75 real(kind=kreal),
intent(in) :: tensor(6)
76 real(kind=kreal),
intent(out) :: eigval(3)
78 real(kind=kreal) :: i1,i2,i3,r,theta,q, x(3,3), xx(3,3)
79 real(kind=kreal),
parameter :: small = 1.0d-10
81 x(1,1)=tensor(1); x(2,2)=tensor(2); x(3,3)=tensor(3)
82 x(1,2)=tensor(4); x(2,1)=x(1,2)
83 x(2,3)=tensor(5); x(3,2)=x(2,3)
84 x(3,1)=tensor(6); x(1,3)=x(3,1)
87 i1= x(1,1)+x(2,2)+x(3,3)
88 i2= 0.5d0*( i1*i1 - (xx(1,1)+xx(2,2)+xx(3,3)) )
89 i3= x(1,1)*x(2,2)*x(3,3)+x(2,1)*x(3,2)*x(1,3)+x(3,1)*x(1,2)*x(2,3) &
90 -x(3,1)*x(2,2)*x(1,3)-x(2,1)*x(1,2)*x(3,3)-x(1,1)*x(3,2)*x(2,3)
92 r=(-2.d0*i1*i1*i1+9.d0*i1*i2-27.d0*i3)/54.d0
93 q=(i1*i1-3.d0*i2)/9.d0
97 if(r*r - q*q*q > small)
then
98 write(*,*)
'ERROR in tensor_eigen3: Discriminant R^2 - Q^3 > 0'
99 write(*,*)
'This indicates the algorithm has numerical issues.'
100 write(*,*)
'Discriminant =', r*r - q*q*q
101 stop
'Eigenvalue calculation failed - possible numerical error'
105 if(abs(q) < small)
then
112 if(abs(q*q*q) < small)
then
114 if(abs(r) < small)
then
130 theta = acos(max(-1.0d0, min(1.0d0, r/dsqrt(q*q*q))))
133 eigval(1) = -2.d0*dsqrt(q)*cos(theta/3.d0)+i1/3.d0
134 eigval(2) = -2.d0*dsqrt(q)*cos((theta+2.d0*pi)/3.d0)+i1/3.d0
135 eigval(3) = -2.d0*dsqrt(q)*cos((theta-2.d0*pi)/3.d0)+i1/3.d0
142 subroutine eigen3 (tensor, eigval, princ)
143 real(kind=kreal),
intent(in) :: tensor(6)
144 real(kind=kreal),
intent(out) :: eigval(3)
145 real(kind=kreal),
intent(out) :: princ(3, 3)
147 integer,
parameter :: msweep = 50
148 integer :: i,j, is, ip, iq, ir
149 real(kind=kreal) :: fsum, od, theta, t, c, s, tau, g, h, hd, btens(3,3), factor
151 btens(1,1)=tensor(1); btens(2,2)=tensor(2); btens(3,3)=tensor(3)
152 btens(1,2)=tensor(4); btens(2,1)=btens(1,2)
153 btens(2,3)=tensor(5); btens(3,2)=btens(2,3)
154 btens(3,1)=tensor(6); btens(1,3)=btens(3,1)
164 eigval(i) = btens(i, i)
165 factor = factor + dabs(btens(i,i))
168 if( factor == 0.d0 .or. factor /= factor )
then
171 eigval(1:3) = eigval(1:3)/factor
172 btens(1:3,1:3) = btens(1:3,1:3)/factor
182 fsum = fsum + abs( btens(ip, iq) )
188 if ( fsum < 1.d-10 )
then
189 eigval(1:3) = eigval(1:3)*factor
197 od = 100.d0 * abs(btens(ip, iq) )
198 if ( (od+abs(eigval(ip) ) /= abs(eigval(ip) )) &
199 .and. (od+abs(eigval(iq) ) /= abs(eigval(iq) )))
then
200 hd = eigval(iq) - eigval(ip)
204 if ( abs(hd) + od == abs(hd) )
then
205 t = btens(ip, iq) / hd
207 theta = 0.5d0 * hd / btens(ip, iq)
208 t = 1.d0 / (abs(theta) + sqrt(1.d0 + theta**2) )
209 if ( theta < 0.d0 ) t = - t
214 c = 1.d0 / sqrt(1.d0 + t**2)
217 h = t * btens(ip, iq)
218 eigval(ip) = eigval(ip) - h
219 eigval(iq) = eigval(iq) + h
224 g = btens(min(ir, ip), max(ir, ip) )
225 h = btens(min(ir, iq), max(ir, iq) )
226 btens(min(ir, ip), max(ir, ip) ) = g &
228 btens(min(ir, iq), max(ir, iq) ) = h &
236 princ(ir, ip) = g - s * (h + g * tau)
237 princ(ir, iq) = h + s * (g - h * tau)
247 stop
' Jacobi iteration unable to converge'
252 real(kind=kreal) :: mat(6)
253 real(kind=kreal) :: xj(3,3)
255 xj(1,1)=mat(1); xj(2,2)=mat(2); xj(3,3)=mat(3)
256 xj(1,2)=mat(4); xj(2,1)=xj(1,2)
257 xj(2,3)=mat(5); xj(3,2)=xj(2,3)
258 xj(3,1)=mat(6); xj(1,3)=xj(3,1)
261 +xj(2,1)*xj(3,2)*xj(1,3) &
262 +xj(3,1)*xj(1,2)*xj(2,3) &
263 -xj(3,1)*xj(2,2)*xj(1,3) &
264 -xj(2,1)*xj(1,2)*xj(3,3) &
265 -xj(1,1)*xj(3,2)*xj(2,3)
270 real(kind=kreal) :: xj(3,3)
273 +xj(2,1)*xj(3,2)*xj(1,3) &
274 +xj(3,1)*xj(1,2)*xj(2,3) &
275 -xj(3,1)*xj(2,2)*xj(1,3) &
276 -xj(2,1)*xj(1,2)*xj(3,3) &
277 -xj(1,1)*xj(3,2)*xj(2,3)
282 character(*) :: sub_name
283 integer(kind=kint) :: imsg
284 integer(kind=kint) :: ierr
287 write(imsg,*)
'Memory overflow at ', sub_name
288 write(*,*)
'Memory overflow at ', sub_name
289 call hecmw_abort( hecmw_comm_get_comm( ) )
295 integer,
intent(in) :: NN
296 real(kind=kreal),
intent(inout) :: a(nn,nn)
298 integer :: I, J,K,IW,LR,IP(NN)
299 real(kind=kreal) :: w,wmax,pivot,api,eps,det
318 write(*,
'(''PIVOT ERROR AT'',I5)') k
341 if (j.NE.k) a(i,j)=a(i,j)-w*a(k,j)
368 integer,
intent(in) :: NN
369 real(kind=kreal),
intent(inout) :: a(nn,nn)
370 real(kind=kreal),
intent(inout) :: b(nn)
372 integer :: i, j, k, imax
373 real(kind=kreal) :: w, piv
379 if( dabs(a(i,k)) > dabs(a(imax,k)) ) imax = i
383 w = a(k,j); a(k,j) = a(imax,j); a(imax,j) = w
385 w = b(k); b(k) = b(imax); b(imax) = w
391 a(i,j) = a(i,j) - w * a(k,j)
393 b(i) = b(i) - w * b(k)
399 b(i) = b(i) - a(i,j) * b(j)
406 real(kind=kreal),
intent(in) :: v1(3),v2(3)
407 real(kind=kreal),
intent(out) :: vn(3)
409 vn(1) = v1(2)*v2(3) - v1(3)*v2(2)
410 vn(2) = v1(3)*v2(1) - v1(1)*v2(3)
411 vn(3) = v1(1)*v2(2) - v1(2)*v2(1)
415 real(kind=kreal),
intent(in) :: jacob(3,3)
416 real(kind=kreal),
intent(out) :: tm(6,6)
422 tm(i,j)= jacob(i,j)*jacob(i,j)
424 tm(i,4) = jacob(i,1)*jacob(i,2)
425 tm(i,5) = jacob(i,2)*jacob(i,3)
426 tm(i,6) = jacob(i,3)*jacob(i,1)
428 tm(4,1) = 2.d0*jacob(1,1)*jacob(2,1)
429 tm(5,1) = 2.d0*jacob(2,1)*jacob(3,1)
430 tm(6,1) = 2.d0*jacob(3,1)*jacob(1,1)
431 tm(4,2) = 2.d0*jacob(1,2)*jacob(2,2)
432 tm(5,2) = 2.d0*jacob(2,2)*jacob(3,2)
433 tm(6,2) = 2.d0*jacob(3,2)*jacob(1,2)
434 tm(4,3) = 2.d0*jacob(1,3)*jacob(2,3)
435 tm(5,3) = 2.d0*jacob(2,3)*jacob(3,3)
436 tm(6,3) = 2.d0*jacob(3,3)*jacob(1,3)
437 tm(4,4) = jacob(1,1)*jacob(2,2) + jacob(1,2)*jacob(2,1)
438 tm(5,4) = jacob(2,1)*jacob(3,2) + jacob(2,2)*jacob(3,1)
439 tm(6,4) = jacob(3,1)*jacob(1,2) + jacob(3,2)*jacob(1,1)
440 tm(4,5) = jacob(1,2)*jacob(2,3) + jacob(1,3)*jacob(2,2)
441 tm(5,5) = jacob(2,2)*jacob(3,3) + jacob(2,3)*jacob(3,2)
442 tm(6,5) = jacob(3,2)*jacob(1,3) + jacob(3,3)*jacob(1,2)
443 tm(4,6) = jacob(1,3)*jacob(2,1) + jacob(1,1)*jacob(2,3)
444 tm(5,6) = jacob(2,3)*jacob(3,1) + jacob(2,1)*jacob(3,3)
445 tm(6,6) = jacob(3,3)*jacob(1,1) + jacob(3,1)*jacob(1,3)
453 real(kind=kreal) :: tensor(1:6)
454 real(kind=kreal) :: eigval(3)
455 real(kind=kreal) :: princmatrix(3,3)
456 real(kind=kreal) :: princnormal(3,3)
457 real(kind=kreal) :: tempv(3)
458 real(kind=kreal) :: temps
460 call eigen3(tensor,eigval,princnormal)
462 if (eigval(1)<eigval(2))
then
466 tempv(:)=princnormal(:,1)
467 princnormal(:,1)=princnormal(:,2)
468 princnormal(:,2)=tempv(:)
470 if (eigval(1)<eigval(3))
then
474 tempv(:)=princnormal(:,1)
475 princnormal(:,1)=princnormal(:,3)
476 princnormal(:,3)=tempv(:)
478 if (eigval(2)<eigval(3))
then
482 tempv(:)=princnormal(:,2)
483 princnormal(:,2)=princnormal(:,3)
484 princnormal(:,3)=tempv(:)
489 princmatrix(i,j) = princnormal(i,j) * eigval(j)
498 real(kind=kreal) :: tensor(6)
499 real(kind=kreal) :: eigval(3)
500 real(kind=kreal) :: princ(3,3)
502 real(kind=kreal) :: s11, s22, s33, s12, s23, s13, j1, j2, j3
503 real(kind=kreal) :: ml,nl
504 complex(kind=kreal):: x1,x2,x3
505 real(kind=kreal):: rtemp
516 j2 = -s11*s22 - s22*s33 - s33*s11 + s12**2 + s23**2 + s13**2
517 j3 = s11*s22*s33 + 2*s12*s23*s13 - s11*s23**2 - s22*s13**2 - s33*s12**2
521 call cardano(-j1, -j2, -j3, x1, x2, x3)
525 if (eigval(1)<eigval(2))
then
530 if (eigval(1)<eigval(3))
then
535 if (eigval(2)<eigval(3))
then
542 if (eigval(i)/(eigval(1)+eigval(2)+eigval(3)) < 1.0d-10 )
then
547 ml = ( s23*s13 - s12*(s33-eigval(i)) ) / ( -s23**2 + (s22-eigval(i))*(s33-eigval(i)) )
548 nl = ( s12**2 - (s22-eigval(i))*(s11-eigval(i)) ) / ( s12*s23 - s13*(s22-eigval(i)) )
549 if (abs(ml) >= huge(ml))
then
552 if (abs(nl) >= huge(nl))
then
555 princ(i,1) = eigval(i)/sqrt( 1 + ml**2 + nl**2)
556 princ(i,2) = ml * princ(i,1)
557 princ(i,3) = nl * princ(i,1)
564 real(kind=kreal):: a,b,c
565 real(kind=kreal):: p,q,d
566 complex(kind=kreal):: w
567 complex(kind=kreal):: u,v,y
568 complex(kind=kreal):: x1,x2,x3
569 w = (-1.0d0 + sqrt(dcmplx(-3.0d0)))/2.0d0
570 p = -a**2/9.0d0 + b/3.0d0
571 q = 2.0d0/2.7d1*a**3 - a*b/3.0d0 + c
572 d = q**2 + 4.0d0*p**3
574 u = ((-dcmplx(q) + sqrt(dcmplx(d)))/2.0d0)**(1.0d0/3.0d0)
578 x1 = u + v -dcmplx(a)/3.0d0
579 x2 = u*w + v*w**2 -dcmplx(a)/3.0d0
580 x3 = u*w**2 + v*w -dcmplx(a)/3.0d0
582 y = (-dcmplx(q))**(1.0d0/3.0d0)
583 x1 = y -dcmplx(a)/3.0d0
584 x2 = y*w -dcmplx(a)/3.0d0
585 x3 = y*w**2 -dcmplx(a)/3.0d0
591 real(kind=kreal),
intent(in) :: r(3)
592 real(kind=kreal),
intent(inout) :: v(3)
594 real(kind=kreal) :: rotv(3), rv
595 real(kind=kreal) :: cosx, sinc(2)
596 real(kind=kreal) :: x, x2, x4, x6
597 real(kind=kreal),
parameter :: c0 = 0.5d0
598 real(kind=kreal),
parameter :: c2 = -4.166666666666666d-002
599 real(kind=kreal),
parameter :: c4 = 1.388888888888889d-003
600 real(kind=kreal),
parameter :: c6 = -2.480158730158730d-005
602 x2 = dot_product(r,r)
608 sinc(1) = 1.d0-x2/6.d0+x4/120.d0
609 sinc(2) = c0+c2*x2+c4*x4+c6*x6
612 sinc(2) = (1.d0-cosx)/x2
616 rv = dot_product(r,v)
617 rotv(1:3) = cosx*v(1:3)
618 rotv(1:3) = rotv(1:3)+rv*sinc(2)*r(1:3)
619 rotv(1) = rotv(1) + (-v(2)*r(3)+v(3)*r(2))*sinc(1)
620 rotv(2) = rotv(2) + (-v(3)*r(1)+v(1)*r(3))*sinc(1)
621 rotv(3) = rotv(3) + (-v(1)*r(2)+v(2)*r(1))*sinc(1)
628 real(kind=kreal),
intent(in) :: dpydpx(3,3)
629 real(kind=kreal),
intent(out) :: dydx(6,6)
630 real(kind=kreal),
intent(in) :: eigv(3,3)
631 real(kind=kreal),
intent(in) :: px(3), py(3)
633 real(kind=kreal) :: x(6)
634 real(kind=kreal) :: ep(6,3), dx2dx(6,6)
635 integer(kind=kint) :: i, j, m1, m2, m3, ia, ib, ic, ip, jp, k
636 real(kind=kreal) :: xmax, dif12, dif23, c1, c2, c3, c4, d1, d2, d3
637 real(kind=kreal) :: xa, xc, ya, yc, daa, dac, dca, dcb, dcc
638 real(kind=kreal) :: xa_xc, xa_xc2, xa_xc3, ya_yc, xc2
639 real(kind=kreal) :: s1, s2, s3, s4, s5, s6
640 real(kind=kreal),
parameter :: i2(6) = [1.d0, 1.d0, 1.d0, 0.d0, 0.d0, 0.d0]
641 real(kind=kreal),
parameter :: is(6,6) = reshape([&
642 1.d0, 0.d0, 0.d0, 0.d0, 0.d0, 0.d0,&
643 0.d0, 1.d0, 0.d0, 0.d0, 0.d0, 0.d0,&
644 0.d0, 0.d0, 1.d0, 0.d0, 0.d0, 0.d0,&
645 0.d0, 0.d0, 0.d0, 0.5d0, 0.d0, 0.d0,&
646 0.d0, 0.d0, 0.d0, 0.d0, 0.5d0, 0.d0,&
647 0.d0, 0.d0, 0.d0, 0.d0, 0.d0, 0.5d0],&
649 real(kind=kreal),
parameter :: eps=1.d-6
652 ep(1,i) = eigv(1,i)*eigv(1,i)
653 ep(2,i) = eigv(2,i)*eigv(2,i)
654 ep(3,i) = eigv(3,i)*eigv(3,i)
655 ep(4,i) = eigv(1,i)*eigv(2,i)
656 ep(5,i) = eigv(2,i)*eigv(3,i)
657 ep(6,i) = eigv(3,i)*eigv(1,i)
662 x(j) = x(j)+px(i)*ep(j,i)
665 dx2dx = reshape([2.d0*x(1), 0.d0, 0.d0, x(4), 0.d0, x(6),&
666 & 0.d0, 2.d0*x(2), 0.d0, x(4), x(5), 0.d0,&
667 & 0.d0, 0.d0, 2.d0*x(3), 0.d0, x(5), x(6),&
668 & x(4), x(4), 0.d0, (x(1)+x(2))/2.d0, x(6)/2.d0, x(5)/2.d0,&
669 & 0.d0, x(5), x(6), x(6)/2.d0, (x(2)+x(3))/2.d0, x(4)/2.d0,&
670 & x(6), 0.d0, x(6), x(5)/2.d0, x(4)/2.d0, (x(1)+x(3))/2.d0],&
675 m1 = 1; m2 = 2; m3 = 3
679 xmax = maxval(abs(px), 1)
680 if( xmax < eps ) xmax = 1.d0
681 dif12 = abs(px(m1) - px(m2)) / xmax
682 dif23 = abs(px(m2) - px(m3)) / xmax
683 if( dif12 < eps .and. dif23 < eps )
then
684 c1 = dpydpx(1,1)-dpydpx(1,2)
688 dydx(i,j) = c1*is(i,j)+c2*i2(i)*i2(j)
691 else if( dif12 < eps .or. dif23 < eps )
then
692 if( dif12 < eps )
then
693 ia = m3; ib = m1; ic = m2
695 ia = m1; ib = m2; ic = m3
697 ya = py(ia); yc = py(ic)
698 xa = px(ia); xc = px(ic)
706 xa_xc3 = xa_xc2*xa_xc
716 s1 = c1+(dcb-dcc)/xa_xc
717 s2 = 2.d0*xc*c1+(dcb-dcc)*c4
718 s3 = 2.d0*c2+d3/xa_xc2
719 s4 = 2.d0*xc*c2+(dac-dcb)/xa_xc+d3*c3
720 s5 = 2.d0*xc*c2+(dca-dcb)/xa_xc+d3*c3
721 s6 = 2.d0*xc2*c2+(d1*xa-d2*xc)*c3-dcb*c4
724 dydx(i,j) = s1*dx2dx(i,j)-s2*is(i,j)-s3*x(i)*x(j)+s4*x(i)*i2(j)+s5*i2(i)*x(j)-s6*i2(i)*i2(j)
731 ia = 1; ib = 2; ic = 3
733 ia = 2; ib = 3; ic = 1
735 ia = 3; ib = 1; ic = 2
737 c1 = py(ia)/((px(ia)-px(ib))*(px(ia)-px(ic)))
739 c3 = (px(ia)-px(ib))+(px(ia)-px(ic))
743 dydx(i,j) = dydx(i,j)+c1*(dx2dx(i,j)-c2*is(i,j)-c3*ep(i,ia)*ep(j,ia)-c4*(ep(i,ib)*ep(j,ib)-ep(i,ic)*ep(j,ic)))
752 dydx(i,j) = dydx(i,j)+c1*ep(i,ip)*ep(j,jp)
This module provides aux functions.
subroutine eigen3(tensor, eigval, princ)
Compute eigenvalue and eigenvetor for symmetric 3*3 tensor using Jacobi iteration adapted from numeri...
subroutine cross_product(v1, v2, vn)
subroutine eigen3d(tensor, eigval, princ)
real(kind=kreal) function determinant33(XJ)
Compute determinant for 3*3 matrix.
subroutine insert_int2array(iin, carray)
Insert an integer into a integer array.
real(kind=kreal) function determinant(mat)
Compute determinant for symmetric 3*3 matrix.
subroutine append_int2name(n, fname, n1)
Insert an integer at end of a file name.
subroutine calsolve(NN, A, b)
Solve A x = b for dense NN x NN system (Gaussian elimination with partial pivoting)
subroutine tensor_eigen3(tensor, eigval)
Given symmetric 3x3 matrix M, compute the eigenvalues.
subroutine get_principal(tensor, eigval, princmatrix)
subroutine transformation(jacob, tm)
subroutine fstr_chk_alloc(imsg, sub_name, ierr)
subroutine deriv_general_iso_tensor_func_3d(dpydpx, dydx, eigv, px, py)
Compute derivative of a general isotropic tensor function of one tensor.
subroutine cardano(a, b, c, x1, x2, x3)
subroutine memget(var, dimn, syze)
Record used memory.
subroutine rotate_3dvector_by_rodrigues_formula(r, v)
subroutine calinverse(NN, A)
calculate inverse of matrix a