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 #if defined(GFC_REAL_16_IS_FLOAT128)
32627f7eb2Smrg #define MATHFUNC(funcname) funcname ## q
33627f7eb2Smrg #else
34627f7eb2Smrg #define MATHFUNC(funcname) funcname ## l
35627f7eb2Smrg #endif
36627f7eb2Smrg
37627f7eb2Smrg #if defined (HAVE_GFC_REAL_16)
38627f7eb2Smrg
39627f7eb2Smrg
40627f7eb2Smrg
41627f7eb2Smrg #if (defined(GFC_REAL_16_IS_FLOAT128) || defined(HAVE_JNL))
42627f7eb2Smrg extern void bessel_jn_r16 (gfc_array_r16 * const restrict ret, int n1,
43627f7eb2Smrg int n2, GFC_REAL_16 x);
44627f7eb2Smrg export_proto(bessel_jn_r16);
45627f7eb2Smrg
46627f7eb2Smrg void
bessel_jn_r16(gfc_array_r16 * const restrict ret,int n1,int n2,GFC_REAL_16 x)47627f7eb2Smrg bessel_jn_r16 (gfc_array_r16 * const restrict ret, int n1, int n2, GFC_REAL_16 x)
48627f7eb2Smrg {
49627f7eb2Smrg int i;
50627f7eb2Smrg index_type stride;
51627f7eb2Smrg
52627f7eb2Smrg GFC_REAL_16 last1, last2, x2rev;
53627f7eb2Smrg
54627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
55627f7eb2Smrg
56627f7eb2Smrg if (ret->base_addr == NULL)
57627f7eb2Smrg {
58627f7eb2Smrg size_t size = n2 < n1 ? 0 : n2-n1+1;
59627f7eb2Smrg GFC_DIMENSION_SET(ret->dim[0], 0, size-1, 1);
60627f7eb2Smrg ret->base_addr = xmallocarray (size, sizeof (GFC_REAL_16));
61627f7eb2Smrg ret->offset = 0;
62627f7eb2Smrg }
63627f7eb2Smrg
64627f7eb2Smrg if (unlikely (n2 < n1))
65627f7eb2Smrg return;
66627f7eb2Smrg
67627f7eb2Smrg if (unlikely (compile_options.bounds_check)
68627f7eb2Smrg && GFC_DESCRIPTOR_EXTENT(ret,0) != (n2-n1+1))
69627f7eb2Smrg runtime_error("Incorrect extent in return value of BESSEL_JN "
70627f7eb2Smrg "(%ld vs. %ld)", (long int) n2-n1,
71627f7eb2Smrg (long int) GFC_DESCRIPTOR_EXTENT(ret,0));
72627f7eb2Smrg
73627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
74627f7eb2Smrg
75627f7eb2Smrg if (unlikely (x == 0))
76627f7eb2Smrg {
77627f7eb2Smrg ret->base_addr[0] = 1;
78627f7eb2Smrg for (i = 1; i <= n2-n1; i++)
79627f7eb2Smrg ret->base_addr[i*stride] = 0;
80627f7eb2Smrg return;
81627f7eb2Smrg }
82627f7eb2Smrg
83627f7eb2Smrg last1 = MATHFUNC(jn) (n2, x);
84627f7eb2Smrg ret->base_addr[(n2-n1)*stride] = last1;
85627f7eb2Smrg
86627f7eb2Smrg if (n1 == n2)
87627f7eb2Smrg return;
88627f7eb2Smrg
89627f7eb2Smrg last2 = MATHFUNC(jn) (n2 - 1, x);
90627f7eb2Smrg ret->base_addr[(n2-n1-1)*stride] = last2;
91627f7eb2Smrg
92627f7eb2Smrg if (n1 + 1 == n2)
93627f7eb2Smrg return;
94627f7eb2Smrg
95627f7eb2Smrg x2rev = GFC_REAL_16_LITERAL(2.)/x;
96627f7eb2Smrg
97627f7eb2Smrg for (i = n2-n1-2; i >= 0; i--)
98627f7eb2Smrg {
99627f7eb2Smrg ret->base_addr[i*stride] = x2rev * (i+1+n1) * last2 - last1;
100627f7eb2Smrg last1 = last2;
101627f7eb2Smrg last2 = ret->base_addr[i*stride];
102627f7eb2Smrg }
103627f7eb2Smrg }
104627f7eb2Smrg
105627f7eb2Smrg #endif
106627f7eb2Smrg
107627f7eb2Smrg #if (defined(GFC_REAL_16_IS_FLOAT128) || defined(HAVE_YNL))
108627f7eb2Smrg extern void bessel_yn_r16 (gfc_array_r16 * const restrict ret,
109627f7eb2Smrg int n1, int n2, GFC_REAL_16 x);
110627f7eb2Smrg export_proto(bessel_yn_r16);
111627f7eb2Smrg
112627f7eb2Smrg void
bessel_yn_r16(gfc_array_r16 * const restrict ret,int n1,int n2,GFC_REAL_16 x)113627f7eb2Smrg bessel_yn_r16 (gfc_array_r16 * const restrict ret, int n1, int n2,
114627f7eb2Smrg GFC_REAL_16 x)
115627f7eb2Smrg {
116627f7eb2Smrg int i;
117627f7eb2Smrg index_type stride;
118627f7eb2Smrg
119627f7eb2Smrg GFC_REAL_16 last1, last2, x2rev;
120627f7eb2Smrg
121627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
122627f7eb2Smrg
123627f7eb2Smrg if (ret->base_addr == NULL)
124627f7eb2Smrg {
125627f7eb2Smrg size_t size = n2 < n1 ? 0 : n2-n1+1;
126627f7eb2Smrg GFC_DIMENSION_SET(ret->dim[0], 0, size-1, 1);
127627f7eb2Smrg ret->base_addr = xmallocarray (size, sizeof (GFC_REAL_16));
128627f7eb2Smrg ret->offset = 0;
129627f7eb2Smrg }
130627f7eb2Smrg
131627f7eb2Smrg if (unlikely (n2 < n1))
132627f7eb2Smrg return;
133627f7eb2Smrg
134627f7eb2Smrg if (unlikely (compile_options.bounds_check)
135627f7eb2Smrg && GFC_DESCRIPTOR_EXTENT(ret,0) != (n2-n1+1))
136627f7eb2Smrg runtime_error("Incorrect extent in return value of BESSEL_JN "
137627f7eb2Smrg "(%ld vs. %ld)", (long int) n2-n1,
138627f7eb2Smrg (long int) GFC_DESCRIPTOR_EXTENT(ret,0));
139627f7eb2Smrg
140627f7eb2Smrg stride = GFC_DESCRIPTOR_STRIDE(ret,0);
141627f7eb2Smrg
142627f7eb2Smrg if (unlikely (x == 0))
143627f7eb2Smrg {
144627f7eb2Smrg for (i = 0; i <= n2-n1; i++)
145627f7eb2Smrg #if defined(GFC_REAL_16_INFINITY)
146627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_16_INFINITY;
147627f7eb2Smrg #else
148627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_16_HUGE;
149627f7eb2Smrg #endif
150627f7eb2Smrg return;
151627f7eb2Smrg }
152627f7eb2Smrg
153627f7eb2Smrg last1 = MATHFUNC(yn) (n1, x);
154627f7eb2Smrg ret->base_addr[0] = last1;
155627f7eb2Smrg
156627f7eb2Smrg if (n1 == n2)
157627f7eb2Smrg return;
158627f7eb2Smrg
159627f7eb2Smrg last2 = MATHFUNC(yn) (n1 + 1, x);
160627f7eb2Smrg ret->base_addr[1*stride] = last2;
161627f7eb2Smrg
162627f7eb2Smrg if (n1 + 1 == n2)
163627f7eb2Smrg return;
164627f7eb2Smrg
165627f7eb2Smrg x2rev = GFC_REAL_16_LITERAL(2.)/x;
166627f7eb2Smrg
167627f7eb2Smrg for (i = 2; i <= n2 - n1; i++)
168627f7eb2Smrg {
169627f7eb2Smrg #if defined(GFC_REAL_16_INFINITY)
170627f7eb2Smrg if (unlikely (last2 == -GFC_REAL_16_INFINITY))
171627f7eb2Smrg {
172627f7eb2Smrg ret->base_addr[i*stride] = -GFC_REAL_16_INFINITY;
173627f7eb2Smrg }
174627f7eb2Smrg else
175627f7eb2Smrg #endif
176627f7eb2Smrg {
177627f7eb2Smrg ret->base_addr[i*stride] = x2rev * (i-1+n1) * last2 - last1;
178627f7eb2Smrg last1 = last2;
179627f7eb2Smrg last2 = ret->base_addr[i*stride];
180627f7eb2Smrg }
181627f7eb2Smrg }
182627f7eb2Smrg }
183627f7eb2Smrg #endif
184627f7eb2Smrg
185627f7eb2Smrg #endif
186627f7eb2Smrg
187