1627f7eb2Smrg /* Implementation of the BESSEL_JN and BESSEL_YN transformational
2627f7eb2Smrg function using a recurrence algorithm.
3*4c3eb207Smrg Copyright (C) 2010-2020 Free Software Foundation, Inc.
4627f7eb2Smrg Contributed by Tobias Burnus <burnus@net-b.de>
5627f7eb2Smrg
6627f7eb2Smrg This file is part of the GNU Fortran runtime library (libgfortran).
7627f7eb2Smrg
8627f7eb2Smrg Libgfortran is free software; you can redistribute it and/or
9627f7eb2Smrg modify it under the terms of the GNU General Public
10627f7eb2Smrg License as published by the Free Software Foundation; either
11627f7eb2Smrg version 3 of the License, or (at your option) any later version.
12627f7eb2Smrg
13627f7eb2Smrg Libgfortran is distributed in the hope that it will be useful,
14627f7eb2Smrg but WITHOUT ANY WARRANTY; without even the implied warranty of
15627f7eb2Smrg MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16627f7eb2Smrg GNU General Public License for more details.
17627f7eb2Smrg
18627f7eb2Smrg Under Section 7 of GPL version 3, you are granted additional
19627f7eb2Smrg permissions described in the GCC Runtime Library Exception, version
20627f7eb2Smrg 3.1, as published by the Free Software Foundation.
21627f7eb2Smrg
22627f7eb2Smrg You should have received a copy of the GNU General Public License and
23627f7eb2Smrg a copy of the GCC Runtime Library Exception along with this program;
24627f7eb2Smrg see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
25627f7eb2Smrg <http://www.gnu.org/licenses/>. */
26627f7eb2Smrg
27627f7eb2Smrg #include "libgfortran.h"
28627f7eb2Smrg
29627f7eb2Smrg
30627f7eb2Smrg
31627f7eb2Smrg #define MATHFUNC(funcname) funcname ## f
32627f7eb2Smrg
33627f7eb2Smrg #if defined (HAVE_GFC_REAL_4)
34627f7eb2Smrg
35627f7eb2Smrg
36627f7eb2Smrg
37627f7eb2Smrg #if defined (HAVE_JNF)
38627f7eb2Smrg extern void bessel_jn_r4 (gfc_array_r4 * const restrict ret, int n1,
39627f7eb2Smrg int n2, GFC_REAL_4 x);
40627f7eb2Smrg export_proto(bessel_jn_r4);
41627f7eb2Smrg
42627f7eb2Smrg void
bessel_jn_r4(gfc_array_r4 * const restrict ret,int n1,int n2,GFC_REAL_4 x)43627f7eb2Smrg bessel_jn_r4 (gfc_array_r4 * const restrict ret, int n1, int n2, GFC_REAL_4 x)
44627f7eb2Smrg {
45627f7eb2Smrg int i;
46627f7eb2Smrg index_type stride;
47627f7eb2Smrg
48627f7eb2Smrg GFC_REAL_4 last1, last2, x2rev;
49627f7eb2Smrg
50627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
51627f7eb2Smrg
52627f7eb2Smrg if (ret->base_addr == NULL)
53627f7eb2Smrg {
54627f7eb2Smrg size_t size = n2 < n1 ? 0 : n2-n1+1;
55627f7eb2Smrg GFC_DIMENSION_SET(ret->dim[0], 0, size-1, 1);
56627f7eb2Smrg ret->base_addr = xmallocarray (size, sizeof (GFC_REAL_4));
57627f7eb2Smrg ret->offset = 0;
58627f7eb2Smrg }
59627f7eb2Smrg
60627f7eb2Smrg if (unlikely (n2 < n1))
61627f7eb2Smrg return;
62627f7eb2Smrg
63627f7eb2Smrg if (unlikely (compile_options.bounds_check)
64627f7eb2Smrg && GFC_DESCRIPTOR_EXTENT(ret,0) != (n2-n1+1))
65627f7eb2Smrg runtime_error("Incorrect extent in return value of BESSEL_JN "
66627f7eb2Smrg "(%ld vs. %ld)", (long int) n2-n1,
67627f7eb2Smrg (long int) GFC_DESCRIPTOR_EXTENT(ret,0));
68627f7eb2Smrg
69627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
70627f7eb2Smrg
71627f7eb2Smrg if (unlikely (x == 0))
72627f7eb2Smrg {
73627f7eb2Smrg ret->base_addr[0] = 1;
74627f7eb2Smrg for (i = 1; i <= n2-n1; i++)
75627f7eb2Smrg ret->base_addr[i*stride] = 0;
76627f7eb2Smrg return;
77627f7eb2Smrg }
78627f7eb2Smrg
79627f7eb2Smrg last1 = MATHFUNC(jn) (n2, x);
80627f7eb2Smrg ret->base_addr[(n2-n1)*stride] = last1;
81627f7eb2Smrg
82627f7eb2Smrg if (n1 == n2)
83627f7eb2Smrg return;
84627f7eb2Smrg
85627f7eb2Smrg last2 = MATHFUNC(jn) (n2 - 1, x);
86627f7eb2Smrg ret->base_addr[(n2-n1-1)*stride] = last2;
87627f7eb2Smrg
88627f7eb2Smrg if (n1 + 1 == n2)
89627f7eb2Smrg return;
90627f7eb2Smrg
91627f7eb2Smrg x2rev = GFC_REAL_4_LITERAL(2.)/x;
92627f7eb2Smrg
93627f7eb2Smrg for (i = n2-n1-2; i >= 0; i--)
94627f7eb2Smrg {
95627f7eb2Smrg ret->base_addr[i*stride] = x2rev * (i+1+n1) * last2 - last1;
96627f7eb2Smrg last1 = last2;
97627f7eb2Smrg last2 = ret->base_addr[i*stride];
98627f7eb2Smrg }
99627f7eb2Smrg }
100627f7eb2Smrg
101627f7eb2Smrg #endif
102627f7eb2Smrg
103627f7eb2Smrg #if defined (HAVE_YNF)
104627f7eb2Smrg extern void bessel_yn_r4 (gfc_array_r4 * const restrict ret,
105627f7eb2Smrg int n1, int n2, GFC_REAL_4 x);
106627f7eb2Smrg export_proto(bessel_yn_r4);
107627f7eb2Smrg
108627f7eb2Smrg void
bessel_yn_r4(gfc_array_r4 * const restrict ret,int n1,int n2,GFC_REAL_4 x)109627f7eb2Smrg bessel_yn_r4 (gfc_array_r4 * const restrict ret, int n1, int n2,
110627f7eb2Smrg GFC_REAL_4 x)
111627f7eb2Smrg {
112627f7eb2Smrg int i;
113627f7eb2Smrg index_type stride;
114627f7eb2Smrg
115627f7eb2Smrg GFC_REAL_4 last1, last2, x2rev;
116627f7eb2Smrg
117627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
118627f7eb2Smrg
119627f7eb2Smrg if (ret->base_addr == NULL)
120627f7eb2Smrg {
121627f7eb2Smrg size_t size = n2 < n1 ? 0 : n2-n1+1;
122627f7eb2Smrg GFC_DIMENSION_SET(ret->dim[0], 0, size-1, 1);
123627f7eb2Smrg ret->base_addr = xmallocarray (size, sizeof (GFC_REAL_4));
124627f7eb2Smrg ret->offset = 0;
125627f7eb2Smrg }
126627f7eb2Smrg
127627f7eb2Smrg if (unlikely (n2 < n1))
128627f7eb2Smrg return;
129627f7eb2Smrg
130627f7eb2Smrg if (unlikely (compile_options.bounds_check)
131627f7eb2Smrg && GFC_DESCRIPTOR_EXTENT(ret,0) != (n2-n1+1))
132627f7eb2Smrg runtime_error("Incorrect extent in return value of BESSEL_JN "
133627f7eb2Smrg "(%ld vs. %ld)", (long int) n2-n1,
134627f7eb2Smrg (long int) GFC_DESCRIPTOR_EXTENT(ret,0));
135627f7eb2Smrg
136627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
137627f7eb2Smrg
138627f7eb2Smrg if (unlikely (x == 0))
139627f7eb2Smrg {
140627f7eb2Smrg for (i = 0; i <= n2-n1; i++)
141627f7eb2Smrg #if defined(GFC_REAL_4_INFINITY)
142627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_4_INFINITY;
143627f7eb2Smrg #else
144627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_4_HUGE;
145627f7eb2Smrg #endif
146627f7eb2Smrg return;
147627f7eb2Smrg }
148627f7eb2Smrg
149627f7eb2Smrg last1 = MATHFUNC(yn) (n1, x);
150627f7eb2Smrg ret->base_addr[0] = last1;
151627f7eb2Smrg
152627f7eb2Smrg if (n1 == n2)
153627f7eb2Smrg return;
154627f7eb2Smrg
155627f7eb2Smrg last2 = MATHFUNC(yn) (n1 + 1, x);
156627f7eb2Smrg ret->base_addr[1*stride] = last2;
157627f7eb2Smrg
158627f7eb2Smrg if (n1 + 1 == n2)
159627f7eb2Smrg return;
160627f7eb2Smrg
161627f7eb2Smrg x2rev = GFC_REAL_4_LITERAL(2.)/x;
162627f7eb2Smrg
163627f7eb2Smrg for (i = 2; i <= n2 - n1; i++)
164627f7eb2Smrg {
165627f7eb2Smrg #if defined(GFC_REAL_4_INFINITY)
166627f7eb2Smrg if (unlikely (last2 == -GFC_REAL_4_INFINITY))
167627f7eb2Smrg {
168627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_4_INFINITY;
169627f7eb2Smrg }
170627f7eb2Smrg else
171627f7eb2Smrg #endif
172627f7eb2Smrg {
173627f7eb2Smrg ret->base_addr[i*stride] = x2rev * (i-1+n1) * last2 - last1;
174627f7eb2Smrg last1 = last2;
175627f7eb2Smrg last2 = ret->base_addr[i*stride];
176627f7eb2Smrg }
177627f7eb2Smrg }
178627f7eb2Smrg }
179627f7eb2Smrg #endif
180627f7eb2Smrg
181627f7eb2Smrg #endif
182627f7eb2Smrg
183