ETS  \$Id: Doxyfile 2162 2020-02-26 14:16:09Z g2dpc $
 All Classes Files Functions Variables Pages
zero.f90
Go to the documentation of this file.
1 subroutine zero(x1, y1, x2, y2, func, err, x, y, izero, ll)
2 
3 ! ******************************************************************
4 ! * the zero y=func(x)=0 on the interval (x1,x2) is found. *
5 ! * the function func(x) should be provided by an external. *
6 ! * upon return izero is the number of iterations which were *
7 ! * required to obtain abs(y).le.err. *
8 ! * diagnostic information is printed if l.ne.0. *
9 ! * the argument ll has the double function of communicating the *
10 ! * output unit iout=ll/10 and the print switch l=ll-iout*10. *
11 ! * iout=0 (l=ll): file is "output". *
12 ! * modified by jan rem to improve convergence 25/08/84. *
13 ! ******************************************************************
14 
15  use itm_types
16 
17  implicit none
18 
19  real(r8), intent(inout) :: x1, y1, x2, y2
20  real(r8), external :: func
21  real(r8), intent(in) :: err
22  real(r8), intent(out) :: x, y
23  integer(itm_i4), intent(out) :: izero
24  integer(itm_i4), intent(in) :: ll
25 
26  real(r8) :: x1s, y1s, x2s, y2s, sig, x0, y0, a, b, c, xn, yn
27  integer(itm_i4) :: l, iout
28  integer(itm_i4) , parameter :: nizero = 50
29 
30  iout = ll / 10
31  l = ll - iout * 10
32 
33  if (iout == 0) iout = 6
34 
35  if ((y1 < 0._r8 .and. y2 < 0._r8) .or. (y1 > 0._r8 &
36  .and. y2 > 0._r8)) then
37  write(iout, 5)
38  return
39  end if
40  if (l /= 0) write(iout, 6)
41  x1s = x1
42  y1s = y1
43  x2s = x2
44  y2s = y2
45  if (x1 > x2) then
46  x1 = x2s
47  y1 = y2s
48  x2 = x1s
49  y2 = y1s
50  end if
51  sig = 1._r8
52  if (y1 >= 0._r8) then
53  sig = -1._r8
54  y1 = -y1
55  y2 = -y2
56  end if
57 
58 !-- begin loop on izero
59  izero = 0
60 
61  loop_zero : do
62  x0 = x1 - (x2 - x1) * y1 / (y2 - y1)
63  y0 = sig * func(x0)
64  izero = izero + 1
65  if (l /= 0) write(iout, 11) izero, x1, y1, x2, y2, x0, y0
66 
67  if (abs(y0) <= err) then
68  x = x0
69  y = y0
70  exit
71  end if
72 
73  if (abs(y0) >= 0.2_r8 * min(-y1, y2)) then
74  a = ((y2 - y0) / (x2 - x0) - (y0 - y1) / (x0 - x1)) / (x2 - x1)
75  b = (y2 - y1) / (x2 - x1) - a * (x2 + x1)
76  c = y0 - a * x0 * x0 - b * x0
77  xn = (-b + sqrt(b * b - 4._r8 * a * c)) / (2._r8 * a)
78  yn = sig * func(xn)
79  izero = izero + 1
80  if (l /= 0) write(iout, 12) izero, x1, y1, x2, y2, xn, yn
81 
82  if (abs(yn) <= err) then
83  x = xn
84  y = yn
85  exit
86  end if
87 
88  if (yn < 0._r8) then
89  x1 = xn
90  y1 = yn
91  if (y0 > 0._r8) then
92  x2 = x0
93  y2 = y0
94  end if
95  else
96  x2 = xn
97  y2 = yn
98  if (y0 < 0._r8) then
99  x1 = x0
100  y1 = y0
101  end if
102  end if
103  else if (y0 > err) then
104  x2 = x0
105  y2 = y0
106  else
107  x1 = x0
108  y1 = y0
109  end if
110  if (izero >= nizero) then
111  write(iout, 13) nizero
112  x = x0
113  y = y0
114  exit
115  end if
116  end do loop_zero
117 !-- end loop on izero
118 
119  x1 = x1s
120  y1 = y1s
121  x2 = x2s
122  y2 = y2s
123 
124  return
125 
126  5 format(/1x, '***subroutine zero: y1 and y2 violate requirements')
127  6 format(/30x, '==== subroutine zero ====')
128 11 format(1x, 'izero=', i3, 3x, 'x1=', 1pe12.4, 3x, 'y1=', 1pe12.4, &
129  3x, 'x2=', 1pe12.4, 3x, 'y2=', 1pe12.4, 3x, 'x0=', 1pe12.4, &
130  3x, 'y0=', 1pe12.4)
131 12 format(1x, 'izero=', i3, 3x, 'x1=', 1pe12.4, 3x, 'y1=', 1pe12.4, &
132  3x, 'x2=', 1pe12.4, 3x, 'y2=', 1pe12.4, 3x, 'xn=', 1pe12.4, &
133  3x, 'yn=', 1pe12.4)
134 13 format(/1x, '***subroutine zero: no convergence for x in ', i3, &
135  ' steps')
136 
137 end subroutine zero
subroutine zero(x1, y1, x2, y2, func, err, x, y, izero, ll)
Definition: zero.f90:1