Skip to content

Commit

Permalink
added refactored dqtcrt from NSWC library
Browse files Browse the repository at this point in the history
  • Loading branch information
jacobwilliams committed Jan 21, 2024
1 parent 094d55e commit 006c433
Show file tree
Hide file tree
Showing 3 changed files with 312 additions and 5 deletions.
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ Method name | Polynomial type | Coefficients | Roots | Reference
[`cpzero`](https://jacobwilliams.github.io/polyroots-fortran/proc/cpzero.html) | General | complex | complex | [SLATEC](https://netlib.org/slatec/src/cpzero.f)
[`rpqr79`](https://jacobwilliams.github.io/polyroots-fortran/proc/rpqr79.html) | General | real | complex | [SLATEC](https://netlib.org/slatec/src/rpqr79.f)
[`cpqr79`](https://jacobwilliams.github.io/polyroots-fortran/proc/cpqr79.html) | General | complex | complex | [SLATEC](https://netlib.org/slatec/src/cpqr79.f)
[`dqtcrt`](https://jacobwilliams.github.io/polyroots-fortran/proc/dqtcrt.html) | Quartic | real | complex | [NSWC Library](/~https://github.com/jacobwilliams/nswc)
[`dcbcrt`](https://jacobwilliams.github.io/polyroots-fortran/proc/dcbcrt.html) | Cubic | real | complex | [NSWC Library](/~https://github.com/jacobwilliams/nswc)
[`dqdcrt`](https://jacobwilliams.github.io/polyroots-fortran/proc/dqdcrt.html) | Quadratic | real | complex | [NSWC Library](/~https://github.com/jacobwilliams/nswc)
[`dpolz`](https://jacobwilliams.github.io/polyroots-fortran/proc/dpolz.html) | General | real | complex | [MATH77 Library](https://netlib.org/math/)
Expand Down
297 changes: 297 additions & 0 deletions src/polyroots_module.F90
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ module polyroots_module
! special polynomial routines:
public :: dcbcrt
public :: dqdcrt
public :: dqtcrt
public :: rroots_chebyshev_cubic

! utility routines:
Expand Down Expand Up @@ -1254,6 +1255,302 @@ pure subroutine dqdcrt(a, zr, zi)
end subroutine dqdcrt
!*****************************************************************************************

!*****************************************************************************************
!>
! dqtcrt computes the roots of the real polynomial
!```
! a(1) + a(2)*z + ... + a(5)*z**4
!```
! and stores the results in `zr` and `zi`. it is assumed
! that `a(5)` is nonzero.
!
!### History
! * Original version written by alfred h. morris,
! naval surface weapons center, dahlgren, virginia
!
!### See also
! * A. H. Morris, "NSWC Library of Mathematical Subroutines",
! Naval Surface Warfare Center, NSWCDD/TR-92/425, January 1993

subroutine dqtcrt(a,zr,zi)

real(wp) :: a(5) , zr(4) , zi(4)
real(wp) :: b , b2 , c , d , e , h , p , q , r , t , temp(4) , &
u , v , v1 , v2 , w(2) , x , x1 , x2 , x3

if ( a(1)==0.0_wp ) then
zr(1) = 0.0_wp
zi(1) = 0.0_wp
call dcbcrt(a(2),zr(2),zi(2))

Check warning on line 1284 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1282-L1284

Added lines #L1282 - L1284 were not covered by tests
else
b = a(4)/(4.0_wp*a(5))
c = a(3)/a(5)
d = a(2)/a(5)
e = a(1)/a(5)
b2 = b*b

p = 0.5_wp*(c-6.0_wp*b2)
q = d - 2.0_wp*b*(c-4.0_wp*b2)
r = b2*(c-3.0_wp*b2) - b*d + e

! solve the resolvent cubic equation. the cubic has
! at least one nonnegative real root. if w1, w2, w3
! are the roots of the cubic then the roots of the
! originial equation are
!
! z = -b + csqrt(w1) + csqrt(w2) + csqrt(w3)
!
! where the signs of the square roots are chosen so
! that csqrt(w1) * csqrt(w2) * csqrt(w3) = -q/8.

temp(1) = -q*q/64.0_wp
temp(2) = 0.25_wp*(p*p-r)
temp(3) = p
temp(4) = 1.0_wp
call dcbcrt(temp,zr,zi)
if ( zi(2)/=0.0_wp ) then

! the resolvent cubic has complex roots

t = zr(1)
x = 0.0_wp
if ( t<0 ) then
h = abs(zr(2)) + abs(zi(2))
if ( abs(t)>h ) then
v = sqrt(abs(t))
zr(1) = -b
zr(2) = -b
zr(3) = -b
zr(4) = -b
zi(1) = v
zi(2) = -v
zi(3) = v
zi(4) = -v
return

Check warning on line 1329 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1315-L1329

Added lines #L1315 - L1329 were not covered by tests
endif
elseif ( t/=0 ) then
x = sqrt(t)
if ( q>0.0_wp ) x = -x

Check warning on line 1333 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1331-L1333

Added lines #L1331 - L1333 were not covered by tests
endif
w(1) = zr(2)
w(2) = zi(2)
call dcsqrt(w,w)
u = 2.0_wp*w(1)
v = 2.0_wp*abs(w(2))
t = x - b
x1 = t + u
x2 = t - u
if ( abs(x1)>abs(x2) ) then
t = x1
x1 = x2
x2 = t

Check warning on line 1346 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1335-L1346

Added lines #L1335 - L1346 were not covered by tests
endif
u = -x - b
h = u*u + v*v
if ( x1*x1<1.0e-2_wp*min(x2*x2,h) ) x1 = e/(x2*h)
zr(1) = x1
zr(2) = x2
zi(1) = 0.0_wp
zi(2) = 0.0_wp
zr(3) = u
zr(4) = u
zi(3) = v
zi(4) = -v

Check warning on line 1358 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1348-L1358

Added lines #L1348 - L1358 were not covered by tests

else

! the resolvent cubic has only real roots
! reorder the roots in increasing order
x1 = zr(1)
x2 = zr(2)
x3 = zr(3)
if ( x1>x2 ) then
t = x1
x1 = x2
x2 = t

Check warning on line 1370 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1368-L1370

Added lines #L1368 - L1370 were not covered by tests
endif
if ( x2>x3 ) then
t = x2
x2 = x3
x3 = t
if ( x1>x2 ) then
t = x1
x1 = x2
x2 = t
endif
endif

u = 0.0_wp
if ( x3>0.0_wp ) u = sqrt(x3)
tmp : block
if ( x2<=0.0_wp ) then
v1 = sqrt(abs(x1))
v2 = sqrt(abs(x2))
if ( q<0.0_wp ) u = -u
else
if ( x1<0.0_wp ) then
if ( abs(x1)>x2 ) then
v1 = sqrt(abs(x1))
v2 = 0.0_wp
exit tmp

Check warning on line 1395 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1391-L1395

Added lines #L1391 - L1395 were not covered by tests
else
x1 = 0.0_wp

Check warning on line 1397 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1397

Added line #L1397 was not covered by tests
endif
endif
x1 = sqrt(x1)
x2 = sqrt(x2)
if ( q>0.0_wp ) x1 = -x1
zr(1) = ((x1+x2)+u) - b
zr(2) = ((-x1-x2)+u) - b
zr(3) = ((x1-x2)-u) - b
zr(4) = ((-x1+x2)-u) - b
call daord(zr,4)
if ( abs(zr(1))<0.1_wp*abs(zr(4)) ) then
t = zr(2)*zr(3)*zr(4)
if ( t/=0.0_wp ) zr(1) = e/t

Check warning on line 1410 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1400-L1410

Added lines #L1400 - L1410 were not covered by tests
endif
zi(1) = 0.0_wp
zi(2) = 0.0_wp
zi(3) = 0.0_wp
zi(4) = 0.0_wp
return

Check warning on line 1416 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1412-L1416

Added lines #L1412 - L1416 were not covered by tests
endif
end block tmp
zr(1) = -u - b
zi(1) = v1 - v2
zr(2) = zr(1)
zi(2) = -zi(1)
zr(3) = u - b
zi(3) = v1 + v2
zr(4) = zr(3)
zi(4) = -zi(3)
endif

endif

end subroutine dqtcrt
!*****************************************************************************************

!*****************************************************************************************
!>
! Used to reorder the elements of the double precision
! array a so that abs(a(i)) <= abs(a(i+1)) for i = 1,...,n-1.
! it is assumed that n >= 1.

subroutine daord(a,n)

Check warning on line 1440 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1440

Added line #L1440 was not covered by tests

integer,intent(in) :: n
real(wp),intent(inout) :: a(n)

integer :: i , ii , imax , j , jmax , ki , l , ll
real(wp) :: s

integer,dimension(10),parameter :: k = [1, 4, 13, 40, 121, 364, &
1093, 3280, 9841, 29524]

! selection of the increments k(i) = (3**i-1)/2
if ( n<2 ) return
imax = 1
do i = 3 , 10
if ( n<=k(i) ) exit
imax = imax + 1

Check warning on line 1456 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1452-L1456

Added lines #L1452 - L1456 were not covered by tests
enddo

! stepping through the increments k(imax),...,k(1)
i = imax
do ii = 1 , imax
ki = k(i)

Check warning on line 1462 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1460-L1462

Added lines #L1460 - L1462 were not covered by tests
! sorting elements that are ki positions apart
! so that abs(a(j)) <= abs(a(j+ki))
jmax = n - ki
do j = 1 , jmax
l = j
ll = j + ki
s = a(ll)
do while ( abs(s)<abs(a(l)) )
a(ll) = a(l)
ll = l
l = l - ki
if ( l<=0 ) exit

Check warning on line 1474 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1465-L1474

Added lines #L1465 - L1474 were not covered by tests
enddo
a(ll) = s

Check warning on line 1476 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1476

Added line #L1476 was not covered by tests
enddo
i = i - 1

Check warning on line 1478 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1478

Added line #L1478 was not covered by tests
enddo

end subroutine daord
!*****************************************************************************************

!*****************************************************************************************
!>
! `w = sqrt(z)` for the double precision complex number `z`
!
! z and w are interpreted as double precision complex numbers.
! it is assumed that z(1) and z(2) are the real and imaginary
! parts of the complex number z, and that w(1) and w(2) are
! the real and imaginary parts of w.

subroutine dcsqrt(z,w)

Check warning on line 1493 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1493

Added line #L1493 was not covered by tests

real(wp),intent(in) :: z(2)
real(wp),intent(out) :: w(2)

real(wp) :: x , y , r

x = z(1)
y = z(2)
if ( x<0 ) then
if ( y/=0.0_wp ) then
r = dcpabs(x,y)
w(2) = sqrt(0.5_wp*(r-x))
w(2) = sign(w(2),y)
w(1) = 0.5_wp*y/w(2)

Check warning on line 1507 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1500-L1507

Added lines #L1500 - L1507 were not covered by tests
else
w(1) = 0.0_wp
w(2) = sqrt(abs(x))

Check warning on line 1510 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1509-L1510

Added lines #L1509 - L1510 were not covered by tests
endif
elseif ( x==0.0_wp ) then
if ( y/=0.0_wp ) then
w(1) = sqrt(0.5_wp*abs(y))
w(2) = sign(w(1),y)

Check warning on line 1515 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1512-L1515

Added lines #L1512 - L1515 were not covered by tests
else
w(1) = 0.0_wp
w(2) = 0.0_wp

Check warning on line 1518 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1517-L1518

Added lines #L1517 - L1518 were not covered by tests
endif
elseif ( y/=0.0_wp ) then
r = dcpabs(x,y)
w(1) = sqrt(0.5_wp*(r+x))
w(2) = 0.5_wp*y/w(1)

Check warning on line 1523 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1520-L1523

Added lines #L1520 - L1523 were not covered by tests
else
w(1) = sqrt(x)
w(2) = 0.0_wp

Check warning on line 1526 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1525-L1526

Added lines #L1525 - L1526 were not covered by tests
endif

end subroutine dcsqrt

Check warning on line 1529 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1529

Added line #L1529 was not covered by tests
!*****************************************************************************************

!*****************************************************************************************
!>
! evaluation of `sqrt(x*x + y*y)`

real(wp) function dcpabs(x,y)

Check warning on line 1536 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1536

Added line #L1536 was not covered by tests

real(wp),intent(in) :: x , y
real(wp) :: a

if ( abs(x)>abs(y) ) then
a = y/x
dcpabs = abs(x)*sqrt(1.0_wp+a*a)
elseif ( y==0.0_wp ) then
dcpabs = 0.0_wp

Check warning on line 1545 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1541-L1545

Added lines #L1541 - L1545 were not covered by tests
else
a = x/y
dcpabs = abs(y)*sqrt(1.0_wp+a*a)

Check warning on line 1548 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1547-L1548

Added lines #L1547 - L1548 were not covered by tests
end if

end function dcpabs

Check warning on line 1551 in src/polyroots_module.F90

View check run for this annotation

Codecov / codecov/patch

src/polyroots_module.F90#L1551

Added line #L1551 was not covered by tests
!*****************************************************************************************

!*****************************************************************************************
!>
! Solve the real coefficient algebraic equation by the qr-method.
Expand Down
19 changes: 14 additions & 5 deletions test/dcbcrt_test.f90
Original file line number Diff line number Diff line change
Expand Up @@ -9,17 +9,26 @@ program dcbcrt_test

implicit none

real(wp), dimension(4) :: a !! coefficients
real(wp), dimension(3) :: zr !! real components of roots
real(wp), dimension(3) :: zi !! imaginary components of roots
real(wp), dimension(5) :: a !! coefficients
real(wp), dimension(4) :: zr !! real components of roots
real(wp), dimension(4) :: zi !! imaginary components of roots

integer :: i !! counter
complex(wp) :: z, root

a = [1.0_wp, 2.0_wp, 3.0_wp, 4.0_wp]
a = [1.0_wp, 2.0_wp, 3.0_wp, 4.0_wp, 5.0_wp]

write(*,'(/A)') 'dqtcrt test:'
call dqtcrt(a, zr(1:4), zi(1:4))
do i = 1, 4
z = cmplx(zr(i), zi(i), wp)
root = a(1) + a(2)*z + a(3)*z**2 + a(4)*z**3 + a(5)*z**4
write(*,'(A,1x,*(e22.15,1x))') 'root is: ', zr(i), zi(i), abs(root)
if (abs(root) > 100*epsilon(1.0_wp)) error stop 'Error: insufficient accuracy'
end do

write(*,'(/A)') 'dcbcrt test:'
call dcbcrt(a, zr, zi)
call dcbcrt(a, zr(1:3), zi(1:3))
do i = 1, 3
z = cmplx(zr(i), zi(i), wp)
root = a(1) + a(2)*z + a(3)*z**2 + a(4)*z**3
Expand Down

0 comments on commit 006c433

Please sign in to comment.