Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * float.c
4 : : * Functions for the built-in floating-point types.
5 : : *
6 : : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : : * Portions Copyright (c) 1994, Regents of the University of California
8 : : *
9 : : *
10 : : * IDENTIFICATION
11 : : * src/backend/utils/adt/float.c
12 : : *
13 : : *-------------------------------------------------------------------------
14 : : */
15 : : #include "postgres.h"
16 : :
17 : : #include <ctype.h>
18 : : #include <float.h>
19 : : #include <math.h>
20 : : #include <limits.h>
21 : :
22 : : #include "catalog/pg_type.h"
23 : : #include "common/int.h"
24 : : #include "common/shortest_dec.h"
25 : : #include "libpq/pqformat.h"
26 : : #include "utils/array.h"
27 : : #include "utils/float.h"
28 : : #include "utils/fmgrprotos.h"
29 : : #include "utils/sortsupport.h"
30 : :
31 : :
32 : : /*
33 : : * Configurable GUC parameter
34 : : *
35 : : * If >0, use shortest-decimal format for output; this is both the default and
36 : : * allows for compatibility with clients that explicitly set a value here to
37 : : * get round-trip-accurate results. If 0 or less, then use the old, slow,
38 : : * decimal rounding method.
39 : : */
40 : : int extra_float_digits = 1;
41 : :
42 : : /* Cached constants for degree-based trig functions */
43 : : static bool degree_consts_set = false;
44 : : static float8 sin_30 = 0;
45 : : static float8 one_minus_cos_60 = 0;
46 : : static float8 asin_0_5 = 0;
47 : : static float8 acos_0_5 = 0;
48 : : static float8 atan_1_0 = 0;
49 : : static float8 tan_45 = 0;
50 : : static float8 cot_45 = 0;
51 : :
52 : : /*
53 : : * These are intentionally not static; don't "fix" them. They will never
54 : : * be referenced by other files, much less changed; but we don't want the
55 : : * compiler to know that, else it might try to precompute expressions
56 : : * involving them. See comments for init_degree_constants().
57 : : *
58 : : * The additional extern declarations are to silence
59 : : * -Wmissing-variable-declarations.
60 : : */
61 : : extern float8 degree_c_thirty;
62 : : extern float8 degree_c_forty_five;
63 : : extern float8 degree_c_sixty;
64 : : extern float8 degree_c_one_half;
65 : : extern float8 degree_c_one;
66 : : float8 degree_c_thirty = 30.0;
67 : : float8 degree_c_forty_five = 45.0;
68 : : float8 degree_c_sixty = 60.0;
69 : : float8 degree_c_one_half = 0.5;
70 : : float8 degree_c_one = 1.0;
71 : :
72 : : /* Local function prototypes */
73 : : static double sind_q1(double x);
74 : : static double cosd_q1(double x);
75 : : static void init_degree_constants(void);
76 : :
77 : :
78 : : /*
79 : : * We use these out-of-line ereport() calls to report float overflow,
80 : : * underflow, and zero-divide, because following our usual practice of
81 : : * repeating them at each call site would lead to a lot of code bloat.
82 : : *
83 : : * This does mean that you don't get a useful error location indicator.
84 : : */
85 : : pg_noinline void
2032 tgl@sss.pgh.pa.us 86 :CBC 42 : float_overflow_error(void)
87 : : {
88 [ + - ]: 42 : ereport(ERROR,
89 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
90 : : errmsg("value out of range: overflow")));
91 : : }
92 : :
93 : : pg_noinline void
94 : 15 : float_underflow_error(void)
95 : : {
96 [ + - ]: 15 : ereport(ERROR,
97 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
98 : : errmsg("value out of range: underflow")));
99 : : }
100 : :
101 : : pg_noinline void
102 : 36 : float_zero_divide_error(void)
103 : : {
104 [ + - ]: 36 : ereport(ERROR,
105 : : (errcode(ERRCODE_DIVISION_BY_ZERO),
106 : : errmsg("division by zero")));
107 : : }
108 : :
109 : :
110 : : /*
111 : : * Returns -1 if 'val' represents negative infinity, 1 if 'val'
112 : : * represents (positive) infinity, and 0 otherwise. On some platforms,
113 : : * this is equivalent to the isinf() macro, but not everywhere: C99
114 : : * does not specify that isinf() needs to distinguish between positive
115 : : * and negative infinity.
116 : : */
117 : : int
7846 neilc@samurai.com 118 :UBC 0 : is_infinite(double val)
119 : : {
7678 bruce@momjian.us 120 [ # # # # ]: 0 : int inf = isinf(val);
121 : :
7846 neilc@samurai.com 122 [ # # ]: 0 : if (inf == 0)
123 : 0 : return 0;
6822 bruce@momjian.us 124 [ # # ]: 0 : else if (val > 0)
7846 neilc@samurai.com 125 : 0 : return 1;
126 : : else
6822 bruce@momjian.us 127 : 0 : return -1;
128 : : }
129 : :
130 : :
131 : : /* ========== USER I/O ROUTINES ========== */
132 : :
133 : :
134 : : /*
135 : : * float4in - converts "num" to float4
136 : : *
137 : : * Note that this code now uses strtof(), where it used to use strtod().
138 : : *
139 : : * The motivation for using strtof() is to avoid a double-rounding problem:
140 : : * for certain decimal inputs, if you round the input correctly to a double,
141 : : * and then round the double to a float, the result is incorrect in that it
142 : : * does not match the result of rounding the decimal value to float directly.
143 : : *
144 : : * One of the best examples is 7.038531e-26:
145 : : *
146 : : * 0xAE43FDp-107 = 7.03853069185120912085...e-26
147 : : * midpoint 7.03853100000000022281...e-26
148 : : * 0xAE43FEp-107 = 7.03853130814879132477...e-26
149 : : *
150 : : * making 0xAE43FDp-107 the correct float result, but if you do the conversion
151 : : * via a double, you get
152 : : *
153 : : * 0xAE43FD.7FFFFFF8p-107 = 7.03853099999999907487...e-26
154 : : * midpoint 7.03853099999999964884...e-26
155 : : * 0xAE43FD.80000000p-107 = 7.03853100000000022281...e-26
156 : : * 0xAE43FD.80000008p-107 = 7.03853100000000137076...e-26
157 : : *
158 : : * so the value rounds to the double exactly on the midpoint between the two
159 : : * nearest floats, and then rounding again to a float gives the incorrect
160 : : * result of 0xAE43FEp-107.
161 : : *
162 : : */
163 : : Datum
9167 tgl@sss.pgh.pa.us 164 :CBC 359275 : float4in(PG_FUNCTION_ARGS)
165 : : {
166 : 359275 : char *num = PG_GETARG_CSTRING(0);
167 : :
990 andrew@dunslane.net 168 : 359275 : PG_RETURN_FLOAT4(float4in_internal(num, NULL, "real", num,
169 : : fcinfo->context));
170 : : }
171 : :
172 : : /*
173 : : * float4in_internal - guts of float4in()
174 : : *
175 : : * This is exposed for use by functions that want a reasonably
176 : : * platform-independent way of inputting floats. The behavior is
177 : : * essentially like strtof + ereturn on error.
178 : : *
179 : : * Uses the same API as float8in_internal below, so most of its
180 : : * comments also apply here, except regarding use in geometric types.
181 : : */
182 : : float4
183 : 364453 : float4in_internal(char *num, char **endptr_p,
184 : : const char *type_name, const char *orig_string,
185 : : struct Node *escontext)
186 : : {
187 : : float val;
188 : : char *endptr;
189 : :
190 : : /*
191 : : * endptr points to the first character _after_ the sequence we recognized
192 : : * as a valid floating point number. orig_string points to the original
193 : : * input string.
194 : : */
195 : :
196 : : /* skip leading whitespace */
4417 tgl@sss.pgh.pa.us 197 [ + + + + ]: 364558 : while (*num != '\0' && isspace((unsigned char) *num))
198 : 105 : num++;
199 : :
200 : : /*
201 : : * Check for an empty-string input to begin with, to avoid the vagaries of
202 : : * strtod() on different platforms.
203 : : */
7846 neilc@samurai.com 204 [ + + ]: 364453 : if (*num == '\0')
990 andrew@dunslane.net 205 [ + - ]: 6 : ereturn(escontext, 0,
206 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
207 : : errmsg("invalid input syntax for type %s: \"%s\"",
208 : : type_name, orig_string)));
209 : :
7218 bruce@momjian.us 210 : 364447 : errno = 0;
2397 rhodiumtoad@postgres 211 : 364447 : val = strtof(num, &endptr);
212 : :
213 : : /* did we not see anything that looks like a double? */
7828 tgl@sss.pgh.pa.us 214 [ + + + + ]: 364447 : if (endptr == num || errno != 0)
215 : : {
4836 bruce@momjian.us 216 : 68 : int save_errno = errno;
217 : :
218 : : /*
219 : : * C99 requires that strtof() accept NaN, [+-]Infinity, and [+-]Inf,
220 : : * but not all platforms support all of these (and some accept them
221 : : * but set ERANGE anyway...) Therefore, we check for these inputs
222 : : * ourselves if strtof() fails.
223 : : *
224 : : * Note: C99 also requires hexadecimal input as well as some extended
225 : : * forms of NaN, but we consider these forms unportable and don't try
226 : : * to support them. You can use 'em if your strtof() takes 'em.
227 : : */
7792 tgl@sss.pgh.pa.us 228 [ - + ]: 68 : if (pg_strncasecmp(num, "NaN", 3) == 0)
229 : : {
7845 tgl@sss.pgh.pa.us 230 :UBC 0 : val = get_float4_nan();
7846 neilc@samurai.com 231 : 0 : endptr = num + 3;
232 : : }
7792 tgl@sss.pgh.pa.us 233 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "Infinity", 8) == 0)
234 : : {
7845 tgl@sss.pgh.pa.us 235 :UBC 0 : val = get_float4_infinity();
7846 neilc@samurai.com 236 : 0 : endptr = num + 8;
237 : : }
4417 tgl@sss.pgh.pa.us 238 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "+Infinity", 9) == 0)
239 : : {
4417 tgl@sss.pgh.pa.us 240 :UBC 0 : val = get_float4_infinity();
241 : 0 : endptr = num + 9;
242 : : }
7792 tgl@sss.pgh.pa.us 243 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "-Infinity", 9) == 0)
244 : : {
7678 bruce@momjian.us 245 :UBC 0 : val = -get_float4_infinity();
7846 neilc@samurai.com 246 : 0 : endptr = num + 9;
247 : : }
4417 tgl@sss.pgh.pa.us 248 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "inf", 3) == 0)
249 : : {
4417 tgl@sss.pgh.pa.us 250 :UBC 0 : val = get_float4_infinity();
251 : 0 : endptr = num + 3;
252 : : }
4417 tgl@sss.pgh.pa.us 253 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "+inf", 4) == 0)
254 : : {
4417 tgl@sss.pgh.pa.us 255 :UBC 0 : val = get_float4_infinity();
256 : 0 : endptr = num + 4;
257 : : }
4417 tgl@sss.pgh.pa.us 258 [ - + ]:CBC 68 : else if (pg_strncasecmp(num, "-inf", 4) == 0)
259 : : {
4417 tgl@sss.pgh.pa.us 260 :UBC 0 : val = -get_float4_infinity();
261 : 0 : endptr = num + 4;
262 : : }
4966 tgl@sss.pgh.pa.us 263 [ + + ]:CBC 68 : else if (save_errno == ERANGE)
264 : : {
265 : : /*
266 : : * Some platforms return ERANGE for denormalized numbers (those
267 : : * that are not zero, but are too close to zero to have full
268 : : * precision). We'd prefer not to throw error for that, so try to
269 : : * detect whether it's a "real" out-of-range condition by checking
270 : : * to see if the result is zero or huge.
271 : : */
2397 rhodiumtoad@postgres 272 [ + + + + ]: 36 : if (val == 0.0 ||
273 : : #if !defined(HUGE_VALF)
274 : : isinf(val)
275 : : #else
276 [ + + ]: 9 : (val >= HUGE_VALF || val <= -HUGE_VALF)
277 : : #endif
278 : : )
279 : : {
280 : : /* see comments in float8in_internal for rationale */
990 andrew@dunslane.net 281 : 33 : char *errnumber = pstrdup(num);
282 : :
283 : 33 : errnumber[endptr - num] = '\0';
284 : :
285 [ + + ]: 33 : ereturn(escontext, 0,
286 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
287 : : errmsg("\"%s\" is out of range for type real",
288 : : errnumber)));
289 : : }
290 : : }
291 : : else
292 [ + + ]: 32 : ereturn(escontext, 0,
293 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
294 : : errmsg("invalid input syntax for type %s: \"%s\"",
295 : : type_name, orig_string)));
296 : : }
297 : :
298 : : /* skip trailing whitespace */
7828 tgl@sss.pgh.pa.us 299 [ + + + + ]: 364481 : while (*endptr != '\0' && isspace((unsigned char) *endptr))
7849 neilc@samurai.com 300 : 99 : endptr++;
301 : :
302 : : /* report stopping point if wanted, else complain if not end of string */
990 andrew@dunslane.net 303 [ - + ]: 364382 : if (endptr_p)
990 andrew@dunslane.net 304 :UBC 0 : *endptr_p = endptr;
990 andrew@dunslane.net 305 [ + + ]:CBC 364382 : else if (*endptr != '\0')
306 [ + - ]: 18 : ereturn(escontext, 0,
307 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
308 : : errmsg("invalid input syntax for type %s: \"%s\"",
309 : : type_name, orig_string)));
310 : :
311 : 364364 : return val;
312 : : }
313 : :
314 : : /*
315 : : * float4out - converts a float4 number to a string
316 : : * using a standard output format
317 : : */
318 : : Datum
9167 tgl@sss.pgh.pa.us 319 : 163057 : float4out(PG_FUNCTION_ARGS)
320 : : {
321 : 163057 : float4 num = PG_GETARG_FLOAT4(0);
2525 322 : 163057 : char *ascii = (char *) palloc(32);
323 : 163057 : int ndig = FLT_DIG + extra_float_digits;
324 : :
2397 rhodiumtoad@postgres 325 [ + + ]: 163057 : if (extra_float_digits > 0)
326 : : {
327 : 156651 : float_to_shortest_decimal_buf(num, ascii);
328 : 156651 : PG_RETURN_CSTRING(ascii);
329 : : }
330 : :
2525 tgl@sss.pgh.pa.us 331 : 6406 : (void) pg_strfromd(ascii, 32, ndig, num);
9167 332 : 6406 : PG_RETURN_CSTRING(ascii);
333 : : }
334 : :
335 : : /*
336 : : * float4recv - converts external binary format to float4
337 : : */
338 : : Datum
8156 tgl@sss.pgh.pa.us 339 :UBC 0 : float4recv(PG_FUNCTION_ARGS)
340 : : {
341 : 0 : StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
342 : :
343 : 0 : PG_RETURN_FLOAT4(pq_getmsgfloat4(buf));
344 : : }
345 : :
346 : : /*
347 : : * float4send - converts float4 to binary format
348 : : */
349 : : Datum
8156 tgl@sss.pgh.pa.us 350 :CBC 3246 : float4send(PG_FUNCTION_ARGS)
351 : : {
352 : 3246 : float4 num = PG_GETARG_FLOAT4(0);
353 : : StringInfoData buf;
354 : :
355 : 3246 : pq_begintypsend(&buf);
356 : 3246 : pq_sendfloat4(&buf, num);
357 : 3246 : PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
358 : : }
359 : :
360 : : /*
361 : : * float8in - converts "num" to float8
362 : : */
363 : : Datum
9167 364 : 339403 : float8in(PG_FUNCTION_ARGS)
365 : : {
366 : 339403 : char *num = PG_GETARG_CSTRING(0);
367 : :
1002 368 : 339403 : PG_RETURN_FLOAT8(float8in_internal(num, NULL, "double precision", num,
369 : : fcinfo->context));
370 : : }
371 : :
372 : : /*
373 : : * float8in_internal - guts of float8in()
374 : : *
375 : : * This is exposed for use by functions that want a reasonably
376 : : * platform-independent way of inputting doubles. The behavior is
377 : : * essentially like strtod + ereturn on error, but note the following
378 : : * differences:
379 : : * 1. Both leading and trailing whitespace are skipped.
380 : : * 2. If endptr_p is NULL, we report error if there's trailing junk.
381 : : * Otherwise, it's up to the caller to complain about trailing junk.
382 : : * 3. In event of a syntax error, the report mentions the given type_name
383 : : * and prints orig_string as the input; this is meant to support use of
384 : : * this function with types such as "box" and "point", where what we are
385 : : * parsing here is just a substring of orig_string.
386 : : *
387 : : * If escontext points to an ErrorSaveContext node, that is filled instead
388 : : * of throwing an error; the caller must check SOFT_ERROR_OCCURRED()
389 : : * to detect errors.
390 : : *
391 : : * "num" could validly be declared "const char *", but that results in an
392 : : * unreasonable amount of extra casting both here and in callers, so we don't.
393 : : */
394 : : float8
395 : 497669 : float8in_internal(char *num, char **endptr_p,
396 : : const char *type_name, const char *orig_string,
397 : : struct Node *escontext)
398 : : {
399 : : double val;
400 : : char *endptr;
401 : :
402 : : /* skip leading whitespace */
4417 403 [ + + + + ]: 498331 : while (*num != '\0' && isspace((unsigned char) *num))
404 : 662 : num++;
405 : :
406 : : /*
407 : : * Check for an empty-string input to begin with, to avoid the vagaries of
408 : : * strtod() on different platforms.
409 : : */
7846 neilc@samurai.com 410 [ + + ]: 497669 : if (*num == '\0')
1002 tgl@sss.pgh.pa.us 411 [ + - ]: 9 : ereturn(escontext, 0,
412 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
413 : : errmsg("invalid input syntax for type %s: \"%s\"",
414 : : type_name, orig_string)));
415 : :
7218 bruce@momjian.us 416 : 497660 : errno = 0;
10226 417 : 497660 : val = strtod(num, &endptr);
418 : :
419 : : /* did we not see anything that looks like a double? */
7828 tgl@sss.pgh.pa.us 420 [ + + + + ]: 497660 : if (endptr == num || errno != 0)
421 : : {
4836 bruce@momjian.us 422 : 135 : int save_errno = errno;
423 : :
424 : : /*
425 : : * C99 requires that strtod() accept NaN, [+-]Infinity, and [+-]Inf,
426 : : * but not all platforms support all of these (and some accept them
427 : : * but set ERANGE anyway...) Therefore, we check for these inputs
428 : : * ourselves if strtod() fails.
429 : : *
430 : : * Note: C99 also requires hexadecimal input as well as some extended
431 : : * forms of NaN, but we consider these forms unportable and don't try
432 : : * to support them. You can use 'em if your strtod() takes 'em.
433 : : */
7792 tgl@sss.pgh.pa.us 434 [ - + ]: 135 : if (pg_strncasecmp(num, "NaN", 3) == 0)
435 : : {
7845 tgl@sss.pgh.pa.us 436 :UBC 0 : val = get_float8_nan();
7846 neilc@samurai.com 437 : 0 : endptr = num + 3;
438 : : }
7792 tgl@sss.pgh.pa.us 439 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "Infinity", 8) == 0)
440 : : {
7845 tgl@sss.pgh.pa.us 441 :UBC 0 : val = get_float8_infinity();
7846 neilc@samurai.com 442 : 0 : endptr = num + 8;
443 : : }
4417 tgl@sss.pgh.pa.us 444 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "+Infinity", 9) == 0)
445 : : {
4417 tgl@sss.pgh.pa.us 446 :UBC 0 : val = get_float8_infinity();
447 : 0 : endptr = num + 9;
448 : : }
7792 tgl@sss.pgh.pa.us 449 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "-Infinity", 9) == 0)
450 : : {
7678 bruce@momjian.us 451 :UBC 0 : val = -get_float8_infinity();
7846 neilc@samurai.com 452 : 0 : endptr = num + 9;
453 : : }
4417 tgl@sss.pgh.pa.us 454 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "inf", 3) == 0)
455 : : {
4417 tgl@sss.pgh.pa.us 456 :UBC 0 : val = get_float8_infinity();
457 : 0 : endptr = num + 3;
458 : : }
4417 tgl@sss.pgh.pa.us 459 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "+inf", 4) == 0)
460 : : {
4417 tgl@sss.pgh.pa.us 461 :UBC 0 : val = get_float8_infinity();
462 : 0 : endptr = num + 4;
463 : : }
4417 tgl@sss.pgh.pa.us 464 [ - + ]:CBC 135 : else if (pg_strncasecmp(num, "-inf", 4) == 0)
465 : : {
4417 tgl@sss.pgh.pa.us 466 :UBC 0 : val = -get_float8_infinity();
467 : 0 : endptr = num + 4;
468 : : }
4966 tgl@sss.pgh.pa.us 469 [ + + ]:CBC 135 : else if (save_errno == ERANGE)
470 : : {
471 : : /*
472 : : * Some platforms return ERANGE for denormalized numbers (those
473 : : * that are not zero, but are too close to zero to have full
474 : : * precision). We'd prefer not to throw error for that, so try to
475 : : * detect whether it's a "real" out-of-range condition by checking
476 : : * to see if the result is zero or huge.
477 : : *
478 : : * On error, we intentionally complain about double precision not
479 : : * the given type name, and we print only the part of the string
480 : : * that is the current number.
481 : : */
482 [ + + + + : 63 : if (val == 0.0 || val >= HUGE_VAL || val <= -HUGE_VAL)
+ + ]
483 : : {
3447 484 : 54 : char *errnumber = pstrdup(num);
485 : :
486 : 54 : errnumber[endptr - num] = '\0';
1002 487 [ + + ]: 54 : ereturn(escontext, 0,
488 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
489 : : errmsg("\"%s\" is out of range for type double precision",
490 : : errnumber)));
491 : : }
492 : : }
493 : : else
494 [ + + ]: 72 : ereturn(escontext, 0,
495 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
496 : : errmsg("invalid input syntax for type %s: \"%s\"",
497 : : type_name, orig_string)));
498 : : }
499 : :
500 : : /* skip trailing whitespace */
7828 501 [ + + + + ]: 497736 : while (*endptr != '\0' && isspace((unsigned char) *endptr))
7849 neilc@samurai.com 502 : 202 : endptr++;
503 : :
504 : : /* report stopping point if wanted, else complain if not end of string */
3447 tgl@sss.pgh.pa.us 505 [ + + ]: 497534 : if (endptr_p)
506 : 158167 : *endptr_p = endptr;
507 [ + + ]: 339367 : else if (*endptr != '\0')
1002 508 [ + + ]: 21 : ereturn(escontext, 0,
509 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
510 : : errmsg("invalid input syntax for type %s: \"%s\"",
511 : : type_name, orig_string)));
512 : :
3447 513 : 497513 : return val;
514 : : }
515 : :
516 : :
517 : : /*
518 : : * float8out - converts float8 number to a string
519 : : * using a standard output format
520 : : */
521 : : Datum
9167 522 : 493280 : float8out(PG_FUNCTION_ARGS)
523 : : {
524 : 493280 : float8 num = PG_GETARG_FLOAT8(0);
525 : :
3447 526 : 493280 : PG_RETURN_CSTRING(float8out_internal(num));
527 : : }
528 : :
529 : : /*
530 : : * float8out_internal - guts of float8out()
531 : : *
532 : : * This is exposed for use by functions that want a reasonably
533 : : * platform-independent way of outputting doubles.
534 : : * The result is always palloc'd.
535 : : */
536 : : char *
537 : 2294604 : float8out_internal(double num)
538 : : {
2525 539 : 2294604 : char *ascii = (char *) palloc(32);
540 : 2294604 : int ndig = DBL_DIG + extra_float_digits;
541 : :
2397 rhodiumtoad@postgres 542 [ + + ]: 2294604 : if (extra_float_digits > 0)
543 : : {
544 : 2185902 : double_to_shortest_decimal_buf(num, ascii);
545 : 2185902 : return ascii;
546 : : }
547 : :
2525 tgl@sss.pgh.pa.us 548 : 108702 : (void) pg_strfromd(ascii, 32, ndig, num);
3447 549 : 108702 : return ascii;
550 : : }
551 : :
552 : : /*
553 : : * float8recv - converts external binary format to float8
554 : : */
555 : : Datum
8156 556 : 13 : float8recv(PG_FUNCTION_ARGS)
557 : : {
558 : 13 : StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
559 : :
560 : 13 : PG_RETURN_FLOAT8(pq_getmsgfloat8(buf));
561 : : }
562 : :
563 : : /*
564 : : * float8send - converts float8 to binary format
565 : : */
566 : : Datum
567 : 2578 : float8send(PG_FUNCTION_ARGS)
568 : : {
569 : 2578 : float8 num = PG_GETARG_FLOAT8(0);
570 : : StringInfoData buf;
571 : :
572 : 2578 : pq_begintypsend(&buf);
573 : 2578 : pq_sendfloat8(&buf, num);
574 : 2578 : PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
575 : : }
576 : :
577 : :
578 : : /* ========== PUBLIC ROUTINES ========== */
579 : :
580 : :
581 : : /*
582 : : * ======================
583 : : * FLOAT4 BASE OPERATIONS
584 : : * ======================
585 : : */
586 : :
587 : : /*
588 : : * float4abs - returns |arg1| (absolute value)
589 : : */
590 : : Datum
9167 591 : 15 : float4abs(PG_FUNCTION_ARGS)
592 : : {
593 : 15 : float4 arg1 = PG_GETARG_FLOAT4(0);
594 : :
1064 peter@eisentraut.org 595 : 15 : PG_RETURN_FLOAT4(fabsf(arg1));
596 : : }
597 : :
598 : : /*
599 : : * float4um - returns -arg1 (unary minus)
600 : : */
601 : : Datum
9167 tgl@sss.pgh.pa.us 602 : 8 : float4um(PG_FUNCTION_ARGS)
603 : : {
604 : 8 : float4 arg1 = PG_GETARG_FLOAT4(0);
605 : : float4 result;
606 : :
6044 607 : 8 : result = -arg1;
6822 bruce@momjian.us 608 : 8 : PG_RETURN_FLOAT4(result);
609 : : }
610 : :
611 : : Datum
8857 bruce@momjian.us 612 :UBC 0 : float4up(PG_FUNCTION_ARGS)
613 : : {
614 : 0 : float4 arg = PG_GETARG_FLOAT4(0);
615 : :
616 : 0 : PG_RETURN_FLOAT4(arg);
617 : : }
618 : :
619 : : Datum
9167 tgl@sss.pgh.pa.us 620 :CBC 9 : float4larger(PG_FUNCTION_ARGS)
621 : : {
622 : 9 : float4 arg1 = PG_GETARG_FLOAT4(0);
623 : 9 : float4 arg2 = PG_GETARG_FLOAT4(1);
624 : : float4 result;
625 : :
2596 tomas.vondra@postgre 626 [ + + ]: 9 : if (float4_gt(arg1, arg2))
8074 tgl@sss.pgh.pa.us 627 : 3 : result = arg1;
628 : : else
629 : 6 : result = arg2;
9167 630 : 9 : PG_RETURN_FLOAT4(result);
631 : : }
632 : :
633 : : Datum
9167 tgl@sss.pgh.pa.us 634 :UBC 0 : float4smaller(PG_FUNCTION_ARGS)
635 : : {
636 : 0 : float4 arg1 = PG_GETARG_FLOAT4(0);
637 : 0 : float4 arg2 = PG_GETARG_FLOAT4(1);
638 : : float4 result;
639 : :
2596 tomas.vondra@postgre 640 [ # # ]: 0 : if (float4_lt(arg1, arg2))
8074 tgl@sss.pgh.pa.us 641 : 0 : result = arg1;
642 : : else
643 : 0 : result = arg2;
9167 644 : 0 : PG_RETURN_FLOAT4(result);
645 : : }
646 : :
647 : : /*
648 : : * ======================
649 : : * FLOAT8 BASE OPERATIONS
650 : : * ======================
651 : : */
652 : :
653 : : /*
654 : : * float8abs - returns |arg1| (absolute value)
655 : : */
656 : : Datum
9167 tgl@sss.pgh.pa.us 657 :CBC 36234 : float8abs(PG_FUNCTION_ARGS)
658 : : {
659 : 36234 : float8 arg1 = PG_GETARG_FLOAT8(0);
660 : :
6822 bruce@momjian.us 661 : 36234 : PG_RETURN_FLOAT8(fabs(arg1));
662 : : }
663 : :
664 : :
665 : : /*
666 : : * float8um - returns -arg1 (unary minus)
667 : : */
668 : : Datum
9167 tgl@sss.pgh.pa.us 669 : 146 : float8um(PG_FUNCTION_ARGS)
670 : : {
671 : 146 : float8 arg1 = PG_GETARG_FLOAT8(0);
672 : : float8 result;
673 : :
6044 674 : 146 : result = -arg1;
9167 675 : 146 : PG_RETURN_FLOAT8(result);
676 : : }
677 : :
678 : : Datum
8857 bruce@momjian.us 679 :UBC 0 : float8up(PG_FUNCTION_ARGS)
680 : : {
681 : 0 : float8 arg = PG_GETARG_FLOAT8(0);
682 : :
683 : 0 : PG_RETURN_FLOAT8(arg);
684 : : }
685 : :
686 : : Datum
9167 tgl@sss.pgh.pa.us 687 :CBC 6438 : float8larger(PG_FUNCTION_ARGS)
688 : : {
689 : 6438 : float8 arg1 = PG_GETARG_FLOAT8(0);
690 : 6438 : float8 arg2 = PG_GETARG_FLOAT8(1);
691 : : float8 result;
692 : :
2596 tomas.vondra@postgre 693 [ + + ]: 6438 : if (float8_gt(arg1, arg2))
8074 tgl@sss.pgh.pa.us 694 : 6090 : result = arg1;
695 : : else
696 : 348 : result = arg2;
9167 697 : 6438 : PG_RETURN_FLOAT8(result);
698 : : }
699 : :
700 : : Datum
701 : 576 : float8smaller(PG_FUNCTION_ARGS)
702 : : {
703 : 576 : float8 arg1 = PG_GETARG_FLOAT8(0);
704 : 576 : float8 arg2 = PG_GETARG_FLOAT8(1);
705 : : float8 result;
706 : :
2596 tomas.vondra@postgre 707 [ + + ]: 576 : if (float8_lt(arg1, arg2))
8074 tgl@sss.pgh.pa.us 708 : 444 : result = arg1;
709 : : else
710 : 132 : result = arg2;
9167 711 : 576 : PG_RETURN_FLOAT8(result);
712 : : }
713 : :
714 : :
715 : : /*
716 : : * ====================
717 : : * ARITHMETIC OPERATORS
718 : : * ====================
719 : : */
720 : :
721 : : /*
722 : : * float4pl - returns arg1 + arg2
723 : : * float4mi - returns arg1 - arg2
724 : : * float4mul - returns arg1 * arg2
725 : : * float4div - returns arg1 / arg2
726 : : */
727 : : Datum
728 : 27 : float4pl(PG_FUNCTION_ARGS)
729 : : {
6044 730 : 27 : float4 arg1 = PG_GETARG_FLOAT4(0);
731 : 27 : float4 arg2 = PG_GETARG_FLOAT4(1);
732 : :
2596 tomas.vondra@postgre 733 : 27 : PG_RETURN_FLOAT4(float4_pl(arg1, arg2));
734 : : }
735 : :
736 : : Datum
9167 tgl@sss.pgh.pa.us 737 : 9 : float4mi(PG_FUNCTION_ARGS)
738 : : {
739 : 9 : float4 arg1 = PG_GETARG_FLOAT4(0);
740 : 9 : float4 arg2 = PG_GETARG_FLOAT4(1);
741 : :
2596 tomas.vondra@postgre 742 : 9 : PG_RETURN_FLOAT4(float4_mi(arg1, arg2));
743 : : }
744 : :
745 : : Datum
9167 tgl@sss.pgh.pa.us 746 : 18 : float4mul(PG_FUNCTION_ARGS)
747 : : {
748 : 18 : float4 arg1 = PG_GETARG_FLOAT4(0);
749 : 18 : float4 arg2 = PG_GETARG_FLOAT4(1);
750 : :
2596 tomas.vondra@postgre 751 : 18 : PG_RETURN_FLOAT4(float4_mul(arg1, arg2));
752 : : }
753 : :
754 : : Datum
9167 tgl@sss.pgh.pa.us 755 : 24 : float4div(PG_FUNCTION_ARGS)
756 : : {
757 : 24 : float4 arg1 = PG_GETARG_FLOAT4(0);
758 : 24 : float4 arg2 = PG_GETARG_FLOAT4(1);
759 : :
2596 tomas.vondra@postgre 760 : 24 : PG_RETURN_FLOAT4(float4_div(arg1, arg2));
761 : : }
762 : :
763 : : /*
764 : : * float8pl - returns arg1 + arg2
765 : : * float8mi - returns arg1 - arg2
766 : : * float8mul - returns arg1 * arg2
767 : : * float8div - returns arg1 / arg2
768 : : */
769 : : Datum
9167 tgl@sss.pgh.pa.us 770 : 53689 : float8pl(PG_FUNCTION_ARGS)
771 : : {
772 : 53689 : float8 arg1 = PG_GETARG_FLOAT8(0);
773 : 53689 : float8 arg2 = PG_GETARG_FLOAT8(1);
774 : :
2596 tomas.vondra@postgre 775 : 53689 : PG_RETURN_FLOAT8(float8_pl(arg1, arg2));
776 : : }
777 : :
778 : : Datum
9167 tgl@sss.pgh.pa.us 779 : 6276 : float8mi(PG_FUNCTION_ARGS)
780 : : {
781 : 6276 : float8 arg1 = PG_GETARG_FLOAT8(0);
782 : 6276 : float8 arg2 = PG_GETARG_FLOAT8(1);
783 : :
2596 tomas.vondra@postgre 784 : 6276 : PG_RETURN_FLOAT8(float8_mi(arg1, arg2));
785 : : }
786 : :
787 : : Datum
9167 tgl@sss.pgh.pa.us 788 : 933059 : float8mul(PG_FUNCTION_ARGS)
789 : : {
790 : 933059 : float8 arg1 = PG_GETARG_FLOAT8(0);
791 : 933059 : float8 arg2 = PG_GETARG_FLOAT8(1);
792 : :
2596 tomas.vondra@postgre 793 : 933059 : PG_RETURN_FLOAT8(float8_mul(arg1, arg2));
794 : : }
795 : :
796 : : Datum
9167 tgl@sss.pgh.pa.us 797 : 7634 : float8div(PG_FUNCTION_ARGS)
798 : : {
799 : 7634 : float8 arg1 = PG_GETARG_FLOAT8(0);
800 : 7634 : float8 arg2 = PG_GETARG_FLOAT8(1);
801 : :
2596 tomas.vondra@postgre 802 : 7634 : PG_RETURN_FLOAT8(float8_div(arg1, arg2));
803 : : }
804 : :
805 : :
806 : : /*
807 : : * ====================
808 : : * COMPARISON OPERATORS
809 : : * ====================
810 : : */
811 : :
812 : : /*
813 : : * float4{eq,ne,lt,le,gt,ge} - float4/float4 comparison operations
814 : : */
815 : : int
8892 tgl@sss.pgh.pa.us 816 : 7787137 : float4_cmp_internal(float4 a, float4 b)
817 : : {
2596 tomas.vondra@postgre 818 [ + + ]: 7787137 : if (float4_gt(a, b))
819 : 174395 : return 1;
820 [ + + ]: 7612742 : if (float4_lt(a, b))
821 : 1025719 : return -1;
822 : 6587023 : return 0;
823 : : }
824 : :
825 : : Datum
9167 tgl@sss.pgh.pa.us 826 : 31190 : float4eq(PG_FUNCTION_ARGS)
827 : : {
828 : 31190 : float4 arg1 = PG_GETARG_FLOAT4(0);
829 : 31190 : float4 arg2 = PG_GETARG_FLOAT4(1);
830 : :
2596 tomas.vondra@postgre 831 : 31190 : PG_RETURN_BOOL(float4_eq(arg1, arg2));
832 : : }
833 : :
834 : : Datum
9167 tgl@sss.pgh.pa.us 835 : 15 : float4ne(PG_FUNCTION_ARGS)
836 : : {
837 : 15 : float4 arg1 = PG_GETARG_FLOAT4(0);
838 : 15 : float4 arg2 = PG_GETARG_FLOAT4(1);
839 : :
2596 tomas.vondra@postgre 840 : 15 : PG_RETURN_BOOL(float4_ne(arg1, arg2));
841 : : }
842 : :
843 : : Datum
9167 tgl@sss.pgh.pa.us 844 : 37044 : float4lt(PG_FUNCTION_ARGS)
845 : : {
846 : 37044 : float4 arg1 = PG_GETARG_FLOAT4(0);
847 : 37044 : float4 arg2 = PG_GETARG_FLOAT4(1);
848 : :
2596 tomas.vondra@postgre 849 : 37044 : PG_RETURN_BOOL(float4_lt(arg1, arg2));
850 : : }
851 : :
852 : : Datum
9167 tgl@sss.pgh.pa.us 853 : 1914 : float4le(PG_FUNCTION_ARGS)
854 : : {
855 : 1914 : float4 arg1 = PG_GETARG_FLOAT4(0);
856 : 1914 : float4 arg2 = PG_GETARG_FLOAT4(1);
857 : :
2596 tomas.vondra@postgre 858 : 1914 : PG_RETURN_BOOL(float4_le(arg1, arg2));
859 : : }
860 : :
861 : : Datum
9167 tgl@sss.pgh.pa.us 862 : 2319 : float4gt(PG_FUNCTION_ARGS)
863 : : {
864 : 2319 : float4 arg1 = PG_GETARG_FLOAT4(0);
865 : 2319 : float4 arg2 = PG_GETARG_FLOAT4(1);
866 : :
2596 tomas.vondra@postgre 867 : 2319 : PG_RETURN_BOOL(float4_gt(arg1, arg2));
868 : : }
869 : :
870 : : Datum
9167 tgl@sss.pgh.pa.us 871 : 1914 : float4ge(PG_FUNCTION_ARGS)
872 : : {
873 : 1914 : float4 arg1 = PG_GETARG_FLOAT4(0);
874 : 1914 : float4 arg2 = PG_GETARG_FLOAT4(1);
875 : :
2596 tomas.vondra@postgre 876 : 1914 : PG_RETURN_BOOL(float4_ge(arg1, arg2));
877 : : }
878 : :
879 : : Datum
8892 tgl@sss.pgh.pa.us 880 : 942106 : btfloat4cmp(PG_FUNCTION_ARGS)
881 : : {
882 : 942106 : float4 arg1 = PG_GETARG_FLOAT4(0);
883 : 942106 : float4 arg2 = PG_GETARG_FLOAT4(1);
884 : :
885 : 942106 : PG_RETURN_INT32(float4_cmp_internal(arg1, arg2));
886 : : }
887 : :
888 : : static int
5022 889 : 6840000 : btfloat4fastcmp(Datum x, Datum y, SortSupport ssup)
890 : : {
891 : 6840000 : float4 arg1 = DatumGetFloat4(x);
892 : 6840000 : float4 arg2 = DatumGetFloat4(y);
893 : :
894 : 6840000 : return float4_cmp_internal(arg1, arg2);
895 : : }
896 : :
897 : : Datum
898 : 711 : btfloat4sortsupport(PG_FUNCTION_ARGS)
899 : : {
4836 bruce@momjian.us 900 : 711 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0);
901 : :
5022 tgl@sss.pgh.pa.us 902 : 711 : ssup->comparator = btfloat4fastcmp;
903 : 711 : PG_RETURN_VOID();
904 : : }
905 : :
906 : : /*
907 : : * float8{eq,ne,lt,le,gt,ge} - float8/float8 comparison operations
908 : : */
909 : : int
8892 910 : 11754220 : float8_cmp_internal(float8 a, float8 b)
911 : : {
2596 tomas.vondra@postgre 912 [ + + ]: 11754220 : if (float8_gt(a, b))
913 : 4230928 : return 1;
914 [ + + ]: 7523292 : if (float8_lt(a, b))
915 : 7387833 : return -1;
916 : 135459 : return 0;
917 : : }
918 : :
919 : : Datum
9167 tgl@sss.pgh.pa.us 920 : 304448 : float8eq(PG_FUNCTION_ARGS)
921 : : {
922 : 304448 : float8 arg1 = PG_GETARG_FLOAT8(0);
923 : 304448 : float8 arg2 = PG_GETARG_FLOAT8(1);
924 : :
2596 tomas.vondra@postgre 925 : 304448 : PG_RETURN_BOOL(float8_eq(arg1, arg2));
926 : : }
927 : :
928 : : Datum
9167 tgl@sss.pgh.pa.us 929 : 8184 : float8ne(PG_FUNCTION_ARGS)
930 : : {
931 : 8184 : float8 arg1 = PG_GETARG_FLOAT8(0);
932 : 8184 : float8 arg2 = PG_GETARG_FLOAT8(1);
933 : :
2596 tomas.vondra@postgre 934 : 8184 : PG_RETURN_BOOL(float8_ne(arg1, arg2));
935 : : }
936 : :
937 : : Datum
9167 tgl@sss.pgh.pa.us 938 : 84489 : float8lt(PG_FUNCTION_ARGS)
939 : : {
940 : 84489 : float8 arg1 = PG_GETARG_FLOAT8(0);
941 : 84489 : float8 arg2 = PG_GETARG_FLOAT8(1);
942 : :
2596 tomas.vondra@postgre 943 : 84489 : PG_RETURN_BOOL(float8_lt(arg1, arg2));
944 : : }
945 : :
946 : : Datum
9167 tgl@sss.pgh.pa.us 947 : 3010 : float8le(PG_FUNCTION_ARGS)
948 : : {
949 : 3010 : float8 arg1 = PG_GETARG_FLOAT8(0);
950 : 3010 : float8 arg2 = PG_GETARG_FLOAT8(1);
951 : :
2596 tomas.vondra@postgre 952 : 3010 : PG_RETURN_BOOL(float8_le(arg1, arg2));
953 : : }
954 : :
955 : : Datum
9167 tgl@sss.pgh.pa.us 956 : 15427 : float8gt(PG_FUNCTION_ARGS)
957 : : {
958 : 15427 : float8 arg1 = PG_GETARG_FLOAT8(0);
959 : 15427 : float8 arg2 = PG_GETARG_FLOAT8(1);
960 : :
2596 tomas.vondra@postgre 961 : 15427 : PG_RETURN_BOOL(float8_gt(arg1, arg2));
962 : : }
963 : :
964 : : Datum
9167 tgl@sss.pgh.pa.us 965 : 10652 : float8ge(PG_FUNCTION_ARGS)
966 : : {
967 : 10652 : float8 arg1 = PG_GETARG_FLOAT8(0);
968 : 10652 : float8 arg2 = PG_GETARG_FLOAT8(1);
969 : :
2596 tomas.vondra@postgre 970 : 10652 : PG_RETURN_BOOL(float8_ge(arg1, arg2));
971 : : }
972 : :
973 : : Datum
8892 tgl@sss.pgh.pa.us 974 : 1655 : btfloat8cmp(PG_FUNCTION_ARGS)
975 : : {
976 : 1655 : float8 arg1 = PG_GETARG_FLOAT8(0);
977 : 1655 : float8 arg2 = PG_GETARG_FLOAT8(1);
978 : :
979 : 1655 : PG_RETURN_INT32(float8_cmp_internal(arg1, arg2));
980 : : }
981 : :
982 : : static int
5022 983 : 3018182 : btfloat8fastcmp(Datum x, Datum y, SortSupport ssup)
984 : : {
985 : 3018182 : float8 arg1 = DatumGetFloat8(x);
986 : 3018182 : float8 arg2 = DatumGetFloat8(y);
987 : :
988 : 3018182 : return float8_cmp_internal(arg1, arg2);
989 : : }
990 : :
991 : : Datum
992 : 529 : btfloat8sortsupport(PG_FUNCTION_ARGS)
993 : : {
4836 bruce@momjian.us 994 : 529 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0);
995 : :
5022 tgl@sss.pgh.pa.us 996 : 529 : ssup->comparator = btfloat8fastcmp;
997 : 529 : PG_RETURN_VOID();
998 : : }
999 : :
1000 : : Datum
7969 tgl@sss.pgh.pa.us 1001 :GBC 17 : btfloat48cmp(PG_FUNCTION_ARGS)
1002 : : {
1003 : 17 : float4 arg1 = PG_GETARG_FLOAT4(0);
1004 : 17 : float8 arg2 = PG_GETARG_FLOAT8(1);
1005 : :
1006 : : /* widen float4 to float8 and then compare */
1007 : 17 : PG_RETURN_INT32(float8_cmp_internal(arg1, arg2));
1008 : : }
1009 : :
1010 : : Datum
1011 : 149 : btfloat84cmp(PG_FUNCTION_ARGS)
1012 : : {
1013 : 149 : float8 arg1 = PG_GETARG_FLOAT8(0);
1014 : 149 : float4 arg2 = PG_GETARG_FLOAT4(1);
1015 : :
1016 : : /* widen float4 to float8 and then compare */
1017 : 149 : PG_RETURN_INT32(float8_cmp_internal(arg1, arg2));
1018 : : }
1019 : :
1020 : : /*
1021 : : * in_range support function for float8.
1022 : : *
1023 : : * Note: we needn't supply a float8_float4 variant, as implicit coercion
1024 : : * of the offset value takes care of that scenario just as well.
1025 : : */
1026 : : Datum
2751 tgl@sss.pgh.pa.us 1027 :CBC 576 : in_range_float8_float8(PG_FUNCTION_ARGS)
1028 : : {
1029 : 576 : float8 val = PG_GETARG_FLOAT8(0);
1030 : 576 : float8 base = PG_GETARG_FLOAT8(1);
1031 : 576 : float8 offset = PG_GETARG_FLOAT8(2);
1032 : 576 : bool sub = PG_GETARG_BOOL(3);
1033 : 576 : bool less = PG_GETARG_BOOL(4);
1034 : : float8 sum;
1035 : :
1036 : : /*
1037 : : * Reject negative or NaN offset. Negative is per spec, and NaN is
1038 : : * because appropriate semantics for that seem non-obvious.
1039 : : */
1040 [ + + - + ]: 576 : if (isnan(offset) || offset < 0)
1041 [ + - ]: 3 : ereport(ERROR,
1042 : : (errcode(ERRCODE_INVALID_PRECEDING_OR_FOLLOWING_SIZE),
1043 : : errmsg("invalid preceding or following size in window function")));
1044 : :
1045 : : /*
1046 : : * Deal with cases where val and/or base is NaN, following the rule that
1047 : : * NaN sorts after non-NaN (cf float8_cmp_internal). The offset cannot
1048 : : * affect the conclusion.
1049 : : */
1050 [ + + ]: 573 : if (isnan(val))
1051 : : {
1052 [ + + ]: 93 : if (isnan(base))
1053 : 30 : PG_RETURN_BOOL(true); /* NAN = NAN */
1054 : : else
1055 : 63 : PG_RETURN_BOOL(!less); /* NAN > non-NAN */
1056 : : }
1057 [ + + ]: 480 : else if (isnan(base))
1058 : : {
1059 : 63 : PG_RETURN_BOOL(less); /* non-NAN < NAN */
1060 : : }
1061 : :
1062 : : /*
1063 : : * Deal with cases where both base and offset are infinite, and computing
1064 : : * base +/- offset would produce NaN. This corresponds to a window frame
1065 : : * whose boundary infinitely precedes +inf or infinitely follows -inf,
1066 : : * which is not well-defined. For consistency with other cases involving
1067 : : * infinities, such as the fact that +inf infinitely follows +inf, we
1068 : : * choose to assume that +inf infinitely precedes +inf and -inf infinitely
1069 : : * follows -inf, and therefore that all finite and infinite values are in
1070 : : * such a window frame.
1071 : : *
1072 : : * offset is known positive, so we need only check the sign of base in
1073 : : * this test.
1074 : : */
1874 1075 [ + + + + : 417 : if (isinf(offset) && isinf(base) &&
+ + + + ]
1076 : : (sub ? base > 0 : base < 0))
1077 : 87 : PG_RETURN_BOOL(true);
1078 : :
1079 : : /*
1080 : : * Otherwise it should be safe to compute base +/- offset. We trust the
1081 : : * FPU to cope if an input is +/-inf or the true sum would overflow, and
1082 : : * produce a suitably signed infinity, which will compare properly against
1083 : : * val whether or not that's infinity.
1084 : : */
2751 1085 [ + + ]: 330 : if (sub)
1086 : 180 : sum = base - offset;
1087 : : else
1088 : 150 : sum = base + offset;
1089 : :
1090 [ + + ]: 330 : if (less)
1091 : 129 : PG_RETURN_BOOL(val <= sum);
1092 : : else
1093 : 201 : PG_RETURN_BOOL(val >= sum);
1094 : : }
1095 : :
1096 : : /*
1097 : : * in_range support function for float4.
1098 : : *
1099 : : * We would need a float4_float8 variant in any case, so we supply that and
1100 : : * let implicit coercion take care of the float4_float4 case.
1101 : : */
1102 : : Datum
1103 : 576 : in_range_float4_float8(PG_FUNCTION_ARGS)
1104 : : {
2681 1105 : 576 : float4 val = PG_GETARG_FLOAT4(0);
1106 : 576 : float4 base = PG_GETARG_FLOAT4(1);
1107 : 576 : float8 offset = PG_GETARG_FLOAT8(2);
1108 : 576 : bool sub = PG_GETARG_BOOL(3);
1109 : 576 : bool less = PG_GETARG_BOOL(4);
1110 : : float8 sum;
1111 : :
1112 : : /*
1113 : : * Reject negative or NaN offset. Negative is per spec, and NaN is
1114 : : * because appropriate semantics for that seem non-obvious.
1115 : : */
1116 [ + + - + ]: 576 : if (isnan(offset) || offset < 0)
1117 [ + - ]: 3 : ereport(ERROR,
1118 : : (errcode(ERRCODE_INVALID_PRECEDING_OR_FOLLOWING_SIZE),
1119 : : errmsg("invalid preceding or following size in window function")));
1120 : :
1121 : : /*
1122 : : * Deal with cases where val and/or base is NaN, following the rule that
1123 : : * NaN sorts after non-NaN (cf float8_cmp_internal). The offset cannot
1124 : : * affect the conclusion.
1125 : : */
1126 [ + + ]: 573 : if (isnan(val))
1127 : : {
1128 [ + + ]: 93 : if (isnan(base))
1129 : 30 : PG_RETURN_BOOL(true); /* NAN = NAN */
1130 : : else
1131 : 63 : PG_RETURN_BOOL(!less); /* NAN > non-NAN */
1132 : : }
1133 [ + + ]: 480 : else if (isnan(base))
1134 : : {
1135 : 63 : PG_RETURN_BOOL(less); /* non-NAN < NAN */
1136 : : }
1137 : :
1138 : : /*
1139 : : * Deal with cases where both base and offset are infinite, and computing
1140 : : * base +/- offset would produce NaN. This corresponds to a window frame
1141 : : * whose boundary infinitely precedes +inf or infinitely follows -inf,
1142 : : * which is not well-defined. For consistency with other cases involving
1143 : : * infinities, such as the fact that +inf infinitely follows +inf, we
1144 : : * choose to assume that +inf infinitely precedes +inf and -inf infinitely
1145 : : * follows -inf, and therefore that all finite and infinite values are in
1146 : : * such a window frame.
1147 : : *
1148 : : * offset is known positive, so we need only check the sign of base in
1149 : : * this test.
1150 : : */
1874 1151 [ + + + + : 417 : if (isinf(offset) && isinf(base) &&
+ + + + ]
1152 : : (sub ? base > 0 : base < 0))
1153 : 87 : PG_RETURN_BOOL(true);
1154 : :
1155 : : /*
1156 : : * Otherwise it should be safe to compute base +/- offset. We trust the
1157 : : * FPU to cope if an input is +/-inf or the true sum would overflow, and
1158 : : * produce a suitably signed infinity, which will compare properly against
1159 : : * val whether or not that's infinity.
1160 : : */
2681 1161 [ + + ]: 330 : if (sub)
1162 : 180 : sum = base - offset;
1163 : : else
1164 : 150 : sum = base + offset;
1165 : :
1166 [ + + ]: 330 : if (less)
1167 : 129 : PG_RETURN_BOOL(val <= sum);
1168 : : else
1169 : 201 : PG_RETURN_BOOL(val >= sum);
1170 : : }
1171 : :
1172 : :
1173 : : /*
1174 : : * ===================
1175 : : * CONVERSION ROUTINES
1176 : : * ===================
1177 : : */
1178 : :
1179 : : /*
1180 : : * ftod - converts a float4 number to a float8 number
1181 : : */
1182 : : Datum
9167 1183 : 147 : ftod(PG_FUNCTION_ARGS)
1184 : : {
1185 : 147 : float4 num = PG_GETARG_FLOAT4(0);
1186 : :
1187 : 147 : PG_RETURN_FLOAT8((float8) num);
1188 : : }
1189 : :
1190 : :
1191 : : /*
1192 : : * dtof - converts a float8 number to a float4 number
1193 : : */
1194 : : Datum
1195 : 28 : dtof(PG_FUNCTION_ARGS)
1196 : : {
1197 : 28 : float8 num = PG_GETARG_FLOAT8(0);
1198 : : float4 result;
1199 : :
2032 1200 : 28 : result = (float4) num;
1201 [ + + + - ]: 28 : if (unlikely(isinf(result)) && !isinf(num))
1202 : 6 : float_overflow_error();
1203 [ + + + - ]: 22 : if (unlikely(result == 0.0f) && num != 0.0)
1204 : 6 : float_underflow_error();
1205 : :
1206 : 16 : PG_RETURN_FLOAT4(result);
1207 : : }
1208 : :
1209 : :
1210 : : /*
1211 : : * dtoi4 - converts a float8 number to an int4 number
1212 : : */
1213 : : Datum
9167 1214 : 545962 : dtoi4(PG_FUNCTION_ARGS)
1215 : : {
1216 : 545962 : float8 num = PG_GETARG_FLOAT8(0);
1217 : :
1218 : : /*
1219 : : * Get rid of any fractional part in the input. This is so we don't fail
1220 : : * on just-out-of-range values that would round into range. Note
1221 : : * assumption that rint() will pass through a NaN or Inf unchanged.
1222 : : */
2479 1223 : 545962 : num = rint(num);
1224 : :
1225 : : /* Range check */
2130 1226 [ + + + + : 545962 : if (unlikely(isnan(num) || !FLOAT8_FITS_IN_INT32(num)))
+ + + + ]
8077 1227 [ + - ]: 12 : ereport(ERROR,
1228 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1229 : : errmsg("integer out of range")));
1230 : :
2479 1231 : 545950 : PG_RETURN_INT32((int32) num);
1232 : : }
1233 : :
1234 : :
1235 : : /*
1236 : : * dtoi2 - converts a float8 number to an int2 number
1237 : : */
1238 : : Datum
9224 1239 : 45 : dtoi2(PG_FUNCTION_ARGS)
1240 : : {
1241 : 45 : float8 num = PG_GETARG_FLOAT8(0);
1242 : :
1243 : : /*
1244 : : * Get rid of any fractional part in the input. This is so we don't fail
1245 : : * on just-out-of-range values that would round into range. Note
1246 : : * assumption that rint() will pass through a NaN or Inf unchanged.
1247 : : */
2479 1248 : 45 : num = rint(num);
1249 : :
1250 : : /* Range check */
2130 1251 [ + - + + : 45 : if (unlikely(isnan(num) || !FLOAT8_FITS_IN_INT16(num)))
+ + + + ]
8077 1252 [ + - ]: 6 : ereport(ERROR,
1253 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1254 : : errmsg("smallint out of range")));
1255 : :
2479 1256 : 39 : PG_RETURN_INT16((int16) num);
1257 : : }
1258 : :
1259 : :
1260 : : /*
1261 : : * i4tod - converts an int4 number to a float8 number
1262 : : */
1263 : : Datum
9216 1264 : 1162353 : i4tod(PG_FUNCTION_ARGS)
1265 : : {
1266 : 1162353 : int32 num = PG_GETARG_INT32(0);
1267 : :
6822 bruce@momjian.us 1268 : 1162353 : PG_RETURN_FLOAT8((float8) num);
1269 : : }
1270 : :
1271 : :
1272 : : /*
1273 : : * i2tod - converts an int2 number to a float8 number
1274 : : */
1275 : : Datum
9224 tgl@sss.pgh.pa.us 1276 : 123 : i2tod(PG_FUNCTION_ARGS)
1277 : : {
1278 : 123 : int16 num = PG_GETARG_INT16(0);
1279 : :
6822 bruce@momjian.us 1280 : 123 : PG_RETURN_FLOAT8((float8) num);
1281 : : }
1282 : :
1283 : :
1284 : : /*
1285 : : * ftoi4 - converts a float4 number to an int4 number
1286 : : */
1287 : : Datum
9167 tgl@sss.pgh.pa.us 1288 : 12 : ftoi4(PG_FUNCTION_ARGS)
1289 : : {
1290 : 12 : float4 num = PG_GETARG_FLOAT4(0);
1291 : :
1292 : : /*
1293 : : * Get rid of any fractional part in the input. This is so we don't fail
1294 : : * on just-out-of-range values that would round into range. Note
1295 : : * assumption that rint() will pass through a NaN or Inf unchanged.
1296 : : */
2479 1297 : 12 : num = rint(num);
1298 : :
1299 : : /* Range check */
2130 1300 [ + - + + : 12 : if (unlikely(isnan(num) || !FLOAT4_FITS_IN_INT32(num)))
+ + + + ]
8077 1301 [ + - ]: 6 : ereport(ERROR,
1302 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1303 : : errmsg("integer out of range")));
1304 : :
2479 1305 : 6 : PG_RETURN_INT32((int32) num);
1306 : : }
1307 : :
1308 : :
1309 : : /*
1310 : : * ftoi2 - converts a float4 number to an int2 number
1311 : : */
1312 : : Datum
9224 1313 : 12 : ftoi2(PG_FUNCTION_ARGS)
1314 : : {
1315 : 12 : float4 num = PG_GETARG_FLOAT4(0);
1316 : :
1317 : : /*
1318 : : * Get rid of any fractional part in the input. This is so we don't fail
1319 : : * on just-out-of-range values that would round into range. Note
1320 : : * assumption that rint() will pass through a NaN or Inf unchanged.
1321 : : */
2479 1322 : 12 : num = rint(num);
1323 : :
1324 : : /* Range check */
2130 1325 [ + - + + : 12 : if (unlikely(isnan(num) || !FLOAT4_FITS_IN_INT16(num)))
+ + + + ]
8077 1326 [ + - ]: 6 : ereport(ERROR,
1327 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1328 : : errmsg("smallint out of range")));
1329 : :
2479 1330 : 6 : PG_RETURN_INT16((int16) num);
1331 : : }
1332 : :
1333 : :
1334 : : /*
1335 : : * i4tof - converts an int4 number to a float4 number
1336 : : */
1337 : : Datum
9216 1338 : 249 : i4tof(PG_FUNCTION_ARGS)
1339 : : {
1340 : 249 : int32 num = PG_GETARG_INT32(0);
1341 : :
6822 bruce@momjian.us 1342 : 249 : PG_RETURN_FLOAT4((float4) num);
1343 : : }
1344 : :
1345 : :
1346 : : /*
1347 : : * i2tof - converts an int2 number to a float4 number
1348 : : */
1349 : : Datum
9224 tgl@sss.pgh.pa.us 1350 :UBC 0 : i2tof(PG_FUNCTION_ARGS)
1351 : : {
1352 : 0 : int16 num = PG_GETARG_INT16(0);
1353 : :
6822 bruce@momjian.us 1354 : 0 : PG_RETURN_FLOAT4((float4) num);
1355 : : }
1356 : :
1357 : :
1358 : : /*
1359 : : * =======================
1360 : : * RANDOM FLOAT8 OPERATORS
1361 : : * =======================
1362 : : */
1363 : :
1364 : : /*
1365 : : * dround - returns ROUND(arg1)
1366 : : */
1367 : : Datum
9167 tgl@sss.pgh.pa.us 1368 :CBC 589488 : dround(PG_FUNCTION_ARGS)
1369 : : {
1370 : 589488 : float8 arg1 = PG_GETARG_FLOAT8(0);
1371 : :
6822 bruce@momjian.us 1372 : 589488 : PG_RETURN_FLOAT8(rint(arg1));
1373 : : }
1374 : :
1375 : : /*
1376 : : * dceil - returns the smallest integer greater than or
1377 : : * equal to the specified float
1378 : : */
1379 : : Datum
8358 1380 : 34230 : dceil(PG_FUNCTION_ARGS)
1381 : : {
1382 : 34230 : float8 arg1 = PG_GETARG_FLOAT8(0);
1383 : :
1384 : 34230 : PG_RETURN_FLOAT8(ceil(arg1));
1385 : : }
1386 : :
1387 : : /*
1388 : : * dfloor - returns the largest integer lesser than or
1389 : : * equal to the specified float
1390 : : */
1391 : : Datum
1392 : 30 : dfloor(PG_FUNCTION_ARGS)
1393 : : {
1394 : 30 : float8 arg1 = PG_GETARG_FLOAT8(0);
1395 : :
1396 : 30 : PG_RETURN_FLOAT8(floor(arg1));
1397 : : }
1398 : :
1399 : : /*
1400 : : * dsign - returns -1 if the argument is less than 0, 0
1401 : : * if the argument is equal to 0, and 1 if the
1402 : : * argument is greater than zero.
1403 : : */
1404 : : Datum
1405 : 15 : dsign(PG_FUNCTION_ARGS)
1406 : : {
1407 : 15 : float8 arg1 = PG_GETARG_FLOAT8(0);
1408 : : float8 result;
1409 : :
1410 [ + + ]: 15 : if (arg1 > 0)
1411 : 9 : result = 1.0;
1412 [ + + ]: 6 : else if (arg1 < 0)
1413 : 3 : result = -1.0;
1414 : : else
1415 : 3 : result = 0.0;
1416 : :
1417 : 15 : PG_RETURN_FLOAT8(result);
1418 : : }
1419 : :
1420 : : /*
1421 : : * dtrunc - returns truncation-towards-zero of arg1,
1422 : : * arg1 >= 0 ... the greatest integer less
1423 : : * than or equal to arg1
1424 : : * arg1 < 0 ... the least integer greater
1425 : : * than or equal to arg1
1426 : : */
1427 : : Datum
9167 tgl@sss.pgh.pa.us 1428 : 18 : dtrunc(PG_FUNCTION_ARGS)
1429 : : {
1430 : 18 : float8 arg1 = PG_GETARG_FLOAT8(0);
1431 : : float8 result;
1432 : :
1433 [ + + ]: 18 : if (arg1 >= 0)
1434 : 15 : result = floor(arg1);
1435 : : else
1436 : 3 : result = -floor(-arg1);
1437 : :
1438 : 18 : PG_RETURN_FLOAT8(result);
1439 : : }
1440 : :
1441 : :
1442 : : /*
1443 : : * dsqrt - returns square root of arg1
1444 : : */
1445 : : Datum
1446 : 2033 : dsqrt(PG_FUNCTION_ARGS)
1447 : : {
1448 : 2033 : float8 arg1 = PG_GETARG_FLOAT8(0);
1449 : : float8 result;
1450 : :
1451 [ - + ]: 2033 : if (arg1 < 0)
8077 tgl@sss.pgh.pa.us 1452 [ # # ]:UBC 0 : ereport(ERROR,
1453 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_POWER_FUNCTION),
1454 : : errmsg("cannot take square root of a negative number")));
1455 : :
9167 tgl@sss.pgh.pa.us 1456 :CBC 2033 : result = sqrt(arg1);
2032 1457 [ - + - - ]: 2033 : if (unlikely(isinf(result)) && !isinf(arg1))
2032 tgl@sss.pgh.pa.us 1458 :UBC 0 : float_overflow_error();
2032 tgl@sss.pgh.pa.us 1459 [ + + - + ]:CBC 2033 : if (unlikely(result == 0.0) && arg1 != 0.0)
2032 tgl@sss.pgh.pa.us 1460 :UBC 0 : float_underflow_error();
1461 : :
9167 tgl@sss.pgh.pa.us 1462 :CBC 2033 : PG_RETURN_FLOAT8(result);
1463 : : }
1464 : :
1465 : :
1466 : : /*
1467 : : * dcbrt - returns cube root of arg1
1468 : : */
1469 : : Datum
1470 : 18 : dcbrt(PG_FUNCTION_ARGS)
1471 : : {
1472 : 18 : float8 arg1 = PG_GETARG_FLOAT8(0);
1473 : : float8 result;
1474 : :
1475 : 18 : result = cbrt(arg1);
2032 1476 [ - + - - ]: 18 : if (unlikely(isinf(result)) && !isinf(arg1))
2032 tgl@sss.pgh.pa.us 1477 :UBC 0 : float_overflow_error();
2032 tgl@sss.pgh.pa.us 1478 [ + + - + ]:CBC 18 : if (unlikely(result == 0.0) && arg1 != 0.0)
2032 tgl@sss.pgh.pa.us 1479 :UBC 0 : float_underflow_error();
1480 : :
9167 tgl@sss.pgh.pa.us 1481 :CBC 18 : PG_RETURN_FLOAT8(result);
1482 : : }
1483 : :
1484 : :
1485 : : /*
1486 : : * dpow - returns pow(arg1,arg2)
1487 : : */
1488 : : Datum
1489 : 334 : dpow(PG_FUNCTION_ARGS)
1490 : : {
1491 : 334 : float8 arg1 = PG_GETARG_FLOAT8(0);
1492 : 334 : float8 arg2 = PG_GETARG_FLOAT8(1);
1493 : : float8 result;
1494 : :
1495 : : /*
1496 : : * The POSIX spec says that NaN ^ 0 = 1, and 1 ^ NaN = 1, while all other
1497 : : * cases with NaN inputs yield NaN (with no error). Many older platforms
1498 : : * get one or more of these cases wrong, so deal with them via explicit
1499 : : * logic rather than trusting pow(3).
1500 : : */
2687 1501 [ + + ]: 334 : if (isnan(arg1))
1502 : : {
1503 [ + + + + ]: 9 : if (isnan(arg2) || arg2 != 0.0)
1504 : 6 : PG_RETURN_FLOAT8(get_float8_nan());
1505 : 3 : PG_RETURN_FLOAT8(1.0);
1506 : : }
1507 [ + + ]: 325 : if (isnan(arg2))
1508 : : {
1509 [ + + ]: 9 : if (arg1 != 1.0)
1510 : 6 : PG_RETURN_FLOAT8(get_float8_nan());
1511 : 3 : PG_RETURN_FLOAT8(1.0);
1512 : : }
1513 : :
1514 : : /*
1515 : : * The SQL spec requires that we emit a particular SQLSTATE error code for
1516 : : * certain error conditions. Specifically, we don't return a
1517 : : * divide-by-zero error code for 0 ^ -1.
1518 : : */
6329 bruce@momjian.us 1519 [ + + + + ]: 316 : if (arg1 == 0 && arg2 < 0)
7783 neilc@samurai.com 1520 [ + - ]: 3 : ereport(ERROR,
1521 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_POWER_FUNCTION),
1522 : : errmsg("zero raised to a negative power is undefined")));
6329 bruce@momjian.us 1523 [ + + + + ]: 313 : if (arg1 < 0 && floor(arg2) != arg2)
1524 [ + - ]: 3 : ereport(ERROR,
1525 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_POWER_FUNCTION),
1526 : : errmsg("a negative number raised to a non-integer power yields a complex result")));
1527 : :
1528 : : /*
1529 : : * We don't trust the platform's pow() to handle infinity cases per POSIX
1530 : : * spec either, so deal with those explicitly too. It's easier to handle
1531 : : * infinite y first, so that it doesn't matter if x is also infinite.
1532 : : */
1909 tgl@sss.pgh.pa.us 1533 [ + + ]: 310 : if (isinf(arg2))
1534 : : {
1908 1535 : 51 : float8 absx = fabs(arg1);
1536 : :
1909 1537 [ + + ]: 51 : if (absx == 1.0)
1538 : 12 : result = 1.0;
1539 [ + + ]: 39 : else if (arg2 > 0.0) /* y = +Inf */
1540 : : {
1541 [ + + ]: 21 : if (absx > 1.0)
1542 : 12 : result = arg2;
1543 : : else
1544 : 9 : result = 0.0;
1545 : : }
1546 : : else /* y = -Inf */
1547 : : {
1548 [ + + ]: 18 : if (absx > 1.0)
1549 : 12 : result = 0.0;
1550 : : else
1551 : 6 : result = -arg2;
1552 : : }
1553 : : }
1554 [ + + ]: 259 : else if (isinf(arg1))
1555 : : {
1556 [ + + ]: 24 : if (arg2 == 0.0)
1557 : 6 : result = 1.0;
1558 [ + + ]: 18 : else if (arg1 > 0.0) /* x = +Inf */
1559 : : {
1560 [ + + ]: 6 : if (arg2 > 0.0)
1561 : 3 : result = arg1;
1562 : : else
1563 : 3 : result = 0.0;
1564 : : }
1565 : : else /* x = -Inf */
1566 : : {
1567 : : /*
1568 : : * Per POSIX, the sign of the result depends on whether y is an
1569 : : * odd integer. Since x < 0, we already know from the previous
1570 : : * domain check that y is an integer. It is odd if y/2 is not
1571 : : * also an integer.
1572 : : */
1908 1573 : 12 : float8 halfy = arg2 / 2; /* should be computed exactly */
1574 : 12 : bool yisoddinteger = (floor(halfy) != halfy);
1575 : :
1909 1576 [ + + ]: 12 : if (arg2 > 0.0)
1577 [ + + ]: 6 : result = yisoddinteger ? arg1 : -arg1;
1578 : : else
1579 [ + + ]: 6 : result = yisoddinteger ? -0.0 : 0.0;
1580 : : }
1581 : : }
1582 : : else
1583 : : {
1584 : : /*
1585 : : * pow() sets errno on only some platforms, depending on whether it
1586 : : * follows _IEEE_, _POSIX_, _XOPEN_, or _SVID_, so we must check both
1587 : : * errno and invalid output values. (We can't rely on just the
1588 : : * latter, either; some old platforms return a large-but-finite
1589 : : * HUGE_VAL when reporting overflow.)
1590 : : */
1591 : 235 : errno = 0;
1592 : 235 : result = pow(arg1, arg2);
1593 [ + - - + ]: 235 : if (errno == EDOM || isnan(result))
1594 : : {
1595 : : /*
1596 : : * We handled all possible domain errors above, so this should be
1597 : : * impossible. However, old glibc versions on x86 have a bug that
1598 : : * causes them to fail this way for abs(y) greater than 2^63:
1599 : : *
1600 : : * https://sourceware.org/bugzilla/show_bug.cgi?id=3866
1601 : : *
1602 : : * Hence, if we get here, assume y is finite but large (large
1603 : : * enough to be certainly even). The result should be 0 if x == 0,
1604 : : * 1.0 if abs(x) == 1.0, otherwise an overflow or underflow error.
1605 : : */
1909 tgl@sss.pgh.pa.us 1606 [ # # ]:UBC 0 : if (arg1 == 0.0)
1607 : 0 : result = 0.0; /* we already verified y is positive */
1608 : : else
1609 : : {
1908 1610 : 0 : float8 absx = fabs(arg1);
1611 : :
1909 1612 [ # # ]: 0 : if (absx == 1.0)
1613 : 0 : result = 1.0;
1614 [ # # # # ]: 0 : else if (arg2 >= 0.0 ? (absx > 1.0) : (absx < 1.0))
1615 : 0 : float_overflow_error();
1616 : : else
1617 : 0 : float_underflow_error();
1618 : : }
1619 : : }
1909 tgl@sss.pgh.pa.us 1620 [ + + ]:CBC 235 : else if (errno == ERANGE)
1621 : : {
1622 [ + - ]: 3 : if (result != 0.0)
1623 : 3 : float_overflow_error();
1624 : : else
1909 tgl@sss.pgh.pa.us 1625 :UBC 0 : float_underflow_error();
1626 : : }
1627 : : else
1628 : : {
1909 tgl@sss.pgh.pa.us 1629 [ - + ]:CBC 232 : if (unlikely(isinf(result)))
1909 tgl@sss.pgh.pa.us 1630 :UBC 0 : float_overflow_error();
1909 tgl@sss.pgh.pa.us 1631 [ + + - + ]:CBC 232 : if (unlikely(result == 0.0) && arg1 != 0.0)
1909 tgl@sss.pgh.pa.us 1632 :UBC 0 : float_underflow_error();
1633 : : }
1634 : : }
1635 : :
9167 tgl@sss.pgh.pa.us 1636 :CBC 307 : PG_RETURN_FLOAT8(result);
1637 : : }
1638 : :
1639 : :
1640 : : /*
1641 : : * dexp - returns the exponential function of arg1
1642 : : */
1643 : : Datum
1644 : 27 : dexp(PG_FUNCTION_ARGS)
1645 : : {
1646 : 27 : float8 arg1 = PG_GETARG_FLOAT8(0);
1647 : : float8 result;
1648 : :
1649 : : /*
1650 : : * Handle NaN and Inf cases explicitly. This avoids needing to assume
1651 : : * that the platform's exp() conforms to POSIX for these cases, and it
1652 : : * removes some edge cases for the overflow checks below.
1653 : : */
1910 1654 [ + + ]: 27 : if (isnan(arg1))
1655 : 3 : result = arg1;
1656 [ + + ]: 24 : else if (isinf(arg1))
1657 : : {
1658 : : /* Per POSIX, exp(-Inf) is 0 */
1659 [ + + ]: 6 : result = (arg1 > 0.0) ? arg1 : 0;
1660 : : }
1661 : : else
1662 : : {
1663 : : /*
1664 : : * On some platforms, exp() will not set errno but just return Inf or
1665 : : * zero to report overflow/underflow; therefore, test both cases.
1666 : : */
1667 : 18 : errno = 0;
1668 : 18 : result = exp(arg1);
1669 [ + + ]: 18 : if (unlikely(errno == ERANGE))
1670 : : {
1671 [ - + ]: 3 : if (result != 0.0)
1910 tgl@sss.pgh.pa.us 1672 :UBC 0 : float_overflow_error();
1673 : : else
1910 tgl@sss.pgh.pa.us 1674 :CBC 3 : float_underflow_error();
1675 : : }
1676 [ - + ]: 15 : else if (unlikely(isinf(result)))
1910 tgl@sss.pgh.pa.us 1677 :UBC 0 : float_overflow_error();
1910 tgl@sss.pgh.pa.us 1678 [ - + ]:CBC 15 : else if (unlikely(result == 0.0))
1910 tgl@sss.pgh.pa.us 1679 :UBC 0 : float_underflow_error();
1680 : : }
1681 : :
9167 tgl@sss.pgh.pa.us 1682 :CBC 24 : PG_RETURN_FLOAT8(result);
1683 : : }
1684 : :
1685 : :
1686 : : /*
1687 : : * dlog1 - returns the natural logarithm of arg1
1688 : : */
1689 : : Datum
1690 : 15 : dlog1(PG_FUNCTION_ARGS)
1691 : : {
1692 : 15 : float8 arg1 = PG_GETARG_FLOAT8(0);
1693 : : float8 result;
1694 : :
1695 : : /*
1696 : : * Emit particular SQLSTATE error codes for ln(). This is required by the
1697 : : * SQL standard.
1698 : : */
1699 [ + + ]: 15 : if (arg1 == 0.0)
8077 1700 [ + - ]: 3 : ereport(ERROR,
1701 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_LOG),
1702 : : errmsg("cannot take logarithm of zero")));
9167 1703 [ + + ]: 12 : if (arg1 < 0)
8077 1704 [ + - ]: 3 : ereport(ERROR,
1705 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_LOG),
1706 : : errmsg("cannot take logarithm of a negative number")));
1707 : :
9167 1708 : 9 : result = log(arg1);
2032 1709 [ - + - - ]: 9 : if (unlikely(isinf(result)) && !isinf(arg1))
2032 tgl@sss.pgh.pa.us 1710 :UBC 0 : float_overflow_error();
2032 tgl@sss.pgh.pa.us 1711 [ - + - - ]:CBC 9 : if (unlikely(result == 0.0) && arg1 != 1.0)
2032 tgl@sss.pgh.pa.us 1712 :UBC 0 : float_underflow_error();
1713 : :
9167 tgl@sss.pgh.pa.us 1714 :CBC 9 : PG_RETURN_FLOAT8(result);
1715 : : }
1716 : :
1717 : :
1718 : : /*
1719 : : * dlog10 - returns the base 10 logarithm of arg1
1720 : : */
1721 : : Datum
9167 tgl@sss.pgh.pa.us 1722 :UBC 0 : dlog10(PG_FUNCTION_ARGS)
1723 : : {
1724 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
1725 : : float8 result;
1726 : :
1727 : : /*
1728 : : * Emit particular SQLSTATE error codes for log(). The SQL spec doesn't
1729 : : * define log(), but it does define ln(), so it makes sense to emit the
1730 : : * same error code for an analogous error condition.
1731 : : */
1732 [ # # ]: 0 : if (arg1 == 0.0)
8077 1733 [ # # ]: 0 : ereport(ERROR,
1734 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_LOG),
1735 : : errmsg("cannot take logarithm of zero")));
9167 1736 [ # # ]: 0 : if (arg1 < 0)
8077 1737 [ # # ]: 0 : ereport(ERROR,
1738 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_LOG),
1739 : : errmsg("cannot take logarithm of a negative number")));
1740 : :
9167 1741 : 0 : result = log10(arg1);
2032 1742 [ # # # # ]: 0 : if (unlikely(isinf(result)) && !isinf(arg1))
1743 : 0 : float_overflow_error();
1744 [ # # # # ]: 0 : if (unlikely(result == 0.0) && arg1 != 1.0)
1745 : 0 : float_underflow_error();
1746 : :
9167 1747 : 0 : PG_RETURN_FLOAT8(result);
1748 : : }
1749 : :
1750 : :
1751 : : /*
1752 : : * dacos - returns the arccos of arg1 (radians)
1753 : : */
1754 : : Datum
1755 : 0 : dacos(PG_FUNCTION_ARGS)
1756 : : {
1757 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
1758 : : float8 result;
1759 : :
1760 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1761 [ # # ]: 0 : if (isnan(arg1))
1762 : 0 : PG_RETURN_FLOAT8(get_float8_nan());
1763 : :
1764 : : /*
1765 : : * The principal branch of the inverse cosine function maps values in the
1766 : : * range [-1, 1] to values in the range [0, Pi], so we should reject any
1767 : : * inputs outside that range and the result will always be finite.
1768 : : */
1769 [ # # # # ]: 0 : if (arg1 < -1.0 || arg1 > 1.0)
8077 1770 [ # # ]: 0 : ereport(ERROR,
1771 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1772 : : errmsg("input is out of range")));
1773 : :
3515 1774 : 0 : result = acos(arg1);
2032 1775 [ # # ]: 0 : if (unlikely(isinf(result)))
1776 : 0 : float_overflow_error();
1777 : :
9167 1778 : 0 : PG_RETURN_FLOAT8(result);
1779 : : }
1780 : :
1781 : :
1782 : : /*
1783 : : * dasin - returns the arcsin of arg1 (radians)
1784 : : */
1785 : : Datum
9167 tgl@sss.pgh.pa.us 1786 :CBC 55 : dasin(PG_FUNCTION_ARGS)
1787 : : {
1788 : 55 : float8 arg1 = PG_GETARG_FLOAT8(0);
1789 : : float8 result;
1790 : :
1791 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1792 [ - + ]: 55 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 1793 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
1794 : :
1795 : : /*
1796 : : * The principal branch of the inverse sine function maps values in the
1797 : : * range [-1, 1] to values in the range [-Pi/2, Pi/2], so we should reject
1798 : : * any inputs outside that range and the result will always be finite.
1799 : : */
3515 tgl@sss.pgh.pa.us 1800 [ + - - + ]:CBC 55 : if (arg1 < -1.0 || arg1 > 1.0)
8077 tgl@sss.pgh.pa.us 1801 [ # # ]:UBC 0 : ereport(ERROR,
1802 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1803 : : errmsg("input is out of range")));
1804 : :
3515 tgl@sss.pgh.pa.us 1805 :CBC 55 : result = asin(arg1);
2032 1806 [ - + ]: 55 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 1807 :UBC 0 : float_overflow_error();
1808 : :
9167 tgl@sss.pgh.pa.us 1809 :CBC 55 : PG_RETURN_FLOAT8(result);
1810 : : }
1811 : :
1812 : :
1813 : : /*
1814 : : * datan - returns the arctan of arg1 (radians)
1815 : : */
1816 : : Datum
9167 tgl@sss.pgh.pa.us 1817 :UBC 0 : datan(PG_FUNCTION_ARGS)
1818 : : {
1819 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
1820 : : float8 result;
1821 : :
1822 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1823 [ # # ]: 0 : if (isnan(arg1))
1824 : 0 : PG_RETURN_FLOAT8(get_float8_nan());
1825 : :
1826 : : /*
1827 : : * The principal branch of the inverse tangent function maps all inputs to
1828 : : * values in the range [-Pi/2, Pi/2], so the result should always be
1829 : : * finite, even if the input is infinite.
1830 : : */
9167 1831 : 0 : result = atan(arg1);
2032 1832 [ # # ]: 0 : if (unlikely(isinf(result)))
1833 : 0 : float_overflow_error();
1834 : :
9167 1835 : 0 : PG_RETURN_FLOAT8(result);
1836 : : }
1837 : :
1838 : :
1839 : : /*
1840 : : * atan2 - returns the arctan of arg1/arg2 (radians)
1841 : : */
1842 : : Datum
9167 tgl@sss.pgh.pa.us 1843 :CBC 20 : datan2(PG_FUNCTION_ARGS)
1844 : : {
1845 : 20 : float8 arg1 = PG_GETARG_FLOAT8(0);
1846 : 20 : float8 arg2 = PG_GETARG_FLOAT8(1);
1847 : : float8 result;
1848 : :
1849 : : /* Per the POSIX spec, return NaN if either input is NaN */
3515 1850 [ + - - + ]: 20 : if (isnan(arg1) || isnan(arg2))
3515 tgl@sss.pgh.pa.us 1851 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
1852 : :
1853 : : /*
1854 : : * atan2 maps all inputs to values in the range [-Pi, Pi], so the result
1855 : : * should always be finite, even if the inputs are infinite.
1856 : : */
9167 tgl@sss.pgh.pa.us 1857 :CBC 20 : result = atan2(arg1, arg2);
2032 1858 [ - + ]: 20 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 1859 :UBC 0 : float_overflow_error();
1860 : :
9167 tgl@sss.pgh.pa.us 1861 :CBC 20 : PG_RETURN_FLOAT8(result);
1862 : : }
1863 : :
1864 : :
1865 : : /*
1866 : : * dcos - returns the cosine of arg1 (radians)
1867 : : */
1868 : : Datum
1869 : 585 : dcos(PG_FUNCTION_ARGS)
1870 : : {
1871 : 585 : float8 arg1 = PG_GETARG_FLOAT8(0);
1872 : : float8 result;
1873 : :
1874 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1875 [ - + ]: 585 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 1876 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
1877 : :
1878 : : /*
1879 : : * cos() is periodic and so theoretically can work for all finite inputs,
1880 : : * but some implementations may choose to throw error if the input is so
1881 : : * large that there are no significant digits in the result. So we should
1882 : : * check for errors. POSIX allows an error to be reported either via
1883 : : * errno or via fetestexcept(), but currently we only support checking
1884 : : * errno. (fetestexcept() is rumored to report underflow unreasonably
1885 : : * early on some platforms, so it's not clear that believing it would be a
1886 : : * net improvement anyway.)
1887 : : *
1888 : : * For infinite inputs, POSIX specifies that the trigonometric functions
1889 : : * should return a domain error; but we won't notice that unless the
1890 : : * platform reports via errno, so also explicitly test for infinite
1891 : : * inputs.
1892 : : */
7218 bruce@momjian.us 1893 :CBC 585 : errno = 0;
9167 tgl@sss.pgh.pa.us 1894 : 585 : result = cos(arg1);
3515 1895 [ + - - + ]: 585 : if (errno != 0 || isinf(arg1))
8077 tgl@sss.pgh.pa.us 1896 [ # # ]:UBC 0 : ereport(ERROR,
1897 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1898 : : errmsg("input is out of range")));
2032 tgl@sss.pgh.pa.us 1899 [ - + ]:CBC 585 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 1900 :UBC 0 : float_overflow_error();
1901 : :
9167 tgl@sss.pgh.pa.us 1902 :CBC 585 : PG_RETURN_FLOAT8(result);
1903 : : }
1904 : :
1905 : :
1906 : : /*
1907 : : * dcot - returns the cotangent of arg1 (radians)
1908 : : */
1909 : : Datum
9167 tgl@sss.pgh.pa.us 1910 :UBC 0 : dcot(PG_FUNCTION_ARGS)
1911 : : {
1912 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
1913 : : float8 result;
1914 : :
1915 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1916 [ # # ]: 0 : if (isnan(arg1))
1917 : 0 : PG_RETURN_FLOAT8(get_float8_nan());
1918 : :
1919 : : /* Be sure to throw an error if the input is infinite --- see dcos() */
7218 bruce@momjian.us 1920 : 0 : errno = 0;
9167 tgl@sss.pgh.pa.us 1921 : 0 : result = tan(arg1);
3515 1922 [ # # # # ]: 0 : if (errno != 0 || isinf(arg1))
8077 1923 [ # # ]: 0 : ereport(ERROR,
1924 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1925 : : errmsg("input is out of range")));
1926 : :
9167 1927 : 0 : result = 1.0 / result;
1928 : : /* Not checking for overflow because cot(0) == Inf */
1929 : :
1930 : 0 : PG_RETURN_FLOAT8(result);
1931 : : }
1932 : :
1933 : :
1934 : : /*
1935 : : * dsin - returns the sine of arg1 (radians)
1936 : : */
1937 : : Datum
9167 tgl@sss.pgh.pa.us 1938 :CBC 487 : dsin(PG_FUNCTION_ARGS)
1939 : : {
1940 : 487 : float8 arg1 = PG_GETARG_FLOAT8(0);
1941 : : float8 result;
1942 : :
1943 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1944 [ - + ]: 487 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 1945 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
1946 : :
1947 : : /* Be sure to throw an error if the input is infinite --- see dcos() */
7218 bruce@momjian.us 1948 :CBC 487 : errno = 0;
9167 tgl@sss.pgh.pa.us 1949 : 487 : result = sin(arg1);
3515 1950 [ + - - + ]: 487 : if (errno != 0 || isinf(arg1))
8077 tgl@sss.pgh.pa.us 1951 [ # # ]:UBC 0 : ereport(ERROR,
1952 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1953 : : errmsg("input is out of range")));
2032 tgl@sss.pgh.pa.us 1954 [ - + ]:CBC 487 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 1955 :UBC 0 : float_overflow_error();
1956 : :
9167 tgl@sss.pgh.pa.us 1957 :CBC 487 : PG_RETURN_FLOAT8(result);
1958 : : }
1959 : :
1960 : :
1961 : : /*
1962 : : * dtan - returns the tangent of arg1 (radians)
1963 : : */
1964 : : Datum
9167 tgl@sss.pgh.pa.us 1965 :UBC 0 : dtan(PG_FUNCTION_ARGS)
1966 : : {
1967 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
1968 : : float8 result;
1969 : :
1970 : : /* Per the POSIX spec, return NaN if the input is NaN */
3515 1971 [ # # ]: 0 : if (isnan(arg1))
1972 : 0 : PG_RETURN_FLOAT8(get_float8_nan());
1973 : :
1974 : : /* Be sure to throw an error if the input is infinite --- see dcos() */
7218 bruce@momjian.us 1975 : 0 : errno = 0;
9167 tgl@sss.pgh.pa.us 1976 : 0 : result = tan(arg1);
3515 1977 [ # # # # ]: 0 : if (errno != 0 || isinf(arg1))
8077 1978 [ # # ]: 0 : ereport(ERROR,
1979 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1980 : : errmsg("input is out of range")));
1981 : : /* Not checking for overflow because tan(pi/2) == Inf */
1982 : :
9167 1983 : 0 : PG_RETURN_FLOAT8(result);
1984 : : }
1985 : :
1986 : :
1987 : : /* ========== DEGREE-BASED TRIGONOMETRIC FUNCTIONS ========== */
1988 : :
1989 : :
1990 : : /*
1991 : : * Initialize the cached constants declared at the head of this file
1992 : : * (sin_30 etc). The fact that we need those at all, let alone need this
1993 : : * Rube-Goldberg-worthy method of initializing them, is because there are
1994 : : * compilers out there that will precompute expressions such as sin(constant)
1995 : : * using a sin() function different from what will be used at runtime. If we
1996 : : * want exact results, we must ensure that none of the scaling constants used
1997 : : * in the degree-based trig functions are computed that way. To do so, we
1998 : : * compute them from the variables degree_c_thirty etc, which are also really
1999 : : * constants, but the compiler cannot assume that.
2000 : : *
2001 : : * Other hazards we are trying to forestall with this kluge include the
2002 : : * possibility that compilers will rearrange the expressions, or compute
2003 : : * some intermediate results in registers wider than a standard double.
2004 : : *
2005 : : * In the places where we use these constants, the typical pattern is like
2006 : : * volatile float8 sin_x = sin(x * RADIANS_PER_DEGREE);
2007 : : * return (sin_x / sin_30);
2008 : : * where we hope to get a value of exactly 1.0 from the division when x = 30.
2009 : : * The volatile temporary variable is needed on machines with wide float
2010 : : * registers, to ensure that the result of sin(x) is rounded to double width
2011 : : * the same as the value of sin_30 has been. Experimentation with gcc shows
2012 : : * that marking the temp variable volatile is necessary to make the store and
2013 : : * reload actually happen; hopefully the same trick works for other compilers.
2014 : : * (gcc's documentation suggests using the -ffloat-store compiler switch to
2015 : : * ensure this, but that is compiler-specific and it also pessimizes code in
2016 : : * many places where we don't care about this.)
2017 : : */
2018 : : static void
3421 tgl@sss.pgh.pa.us 2019 :CBC 3 : init_degree_constants(void)
2020 : : {
2021 : 3 : sin_30 = sin(degree_c_thirty * RADIANS_PER_DEGREE);
2022 : 3 : one_minus_cos_60 = 1.0 - cos(degree_c_sixty * RADIANS_PER_DEGREE);
2023 : 3 : asin_0_5 = asin(degree_c_one_half);
2024 : 3 : acos_0_5 = acos(degree_c_one_half);
2025 : 3 : atan_1_0 = atan(degree_c_one);
2026 : 3 : tan_45 = sind_q1(degree_c_forty_five) / cosd_q1(degree_c_forty_five);
2027 : 3 : cot_45 = cosd_q1(degree_c_forty_five) / sind_q1(degree_c_forty_five);
3514 2028 : 3 : degree_consts_set = true;
2029 : 3 : }
2030 : :
2031 : : #define INIT_DEGREE_CONSTANTS() \
2032 : : do { \
2033 : : if (!degree_consts_set) \
2034 : : init_degree_constants(); \
2035 : : } while(0)
2036 : :
2037 : :
2038 : : /*
2039 : : * asind_q1 - returns the inverse sine of x in degrees, for x in
2040 : : * the range [0, 1]. The result is an angle in the
2041 : : * first quadrant --- [0, 90] degrees.
2042 : : *
2043 : : * For the 3 special case inputs (0, 0.5 and 1), this
2044 : : * function will return exact values (0, 30 and 90
2045 : : * degrees respectively).
2046 : : */
2047 : : static double
3515 2048 : 42 : asind_q1(double x)
2049 : : {
2050 : : /*
2051 : : * Stitch together inverse sine and cosine functions for the ranges [0,
2052 : : * 0.5] and (0.5, 1]. Each expression below is guaranteed to return
2053 : : * exactly 30 for x=0.5, so the result is a continuous monotonic function
2054 : : * over the full range.
2055 : : */
2056 [ + + ]: 42 : if (x <= 0.5)
2057 : : {
3420 2058 : 24 : volatile float8 asin_x = asin(x);
2059 : :
2060 : 24 : return (asin_x / asin_0_5) * 30.0;
2061 : : }
2062 : : else
2063 : : {
2064 : 18 : volatile float8 acos_x = acos(x);
2065 : :
2066 : 18 : return 90.0 - (acos_x / acos_0_5) * 60.0;
2067 : : }
2068 : : }
2069 : :
2070 : :
2071 : : /*
2072 : : * acosd_q1 - returns the inverse cosine of x in degrees, for x in
2073 : : * the range [0, 1]. The result is an angle in the
2074 : : * first quadrant --- [0, 90] degrees.
2075 : : *
2076 : : * For the 3 special case inputs (0, 0.5 and 1), this
2077 : : * function will return exact values (0, 60 and 90
2078 : : * degrees respectively).
2079 : : */
2080 : : static double
3515 2081 : 18 : acosd_q1(double x)
2082 : : {
2083 : : /*
2084 : : * Stitch together inverse sine and cosine functions for the ranges [0,
2085 : : * 0.5] and (0.5, 1]. Each expression below is guaranteed to return
2086 : : * exactly 60 for x=0.5, so the result is a continuous monotonic function
2087 : : * over the full range.
2088 : : */
2089 [ + + ]: 18 : if (x <= 0.5)
2090 : : {
3420 2091 : 12 : volatile float8 asin_x = asin(x);
2092 : :
2093 : 12 : return 90.0 - (asin_x / asin_0_5) * 30.0;
2094 : : }
2095 : : else
2096 : : {
2097 : 6 : volatile float8 acos_x = acos(x);
2098 : :
2099 : 6 : return (acos_x / acos_0_5) * 60.0;
2100 : : }
2101 : : }
2102 : :
2103 : :
2104 : : /*
2105 : : * dacosd - returns the arccos of arg1 (degrees)
2106 : : */
2107 : : Datum
3515 2108 : 30 : dacosd(PG_FUNCTION_ARGS)
2109 : : {
2110 : 30 : float8 arg1 = PG_GETARG_FLOAT8(0);
2111 : : float8 result;
2112 : :
2113 : : /* Per the POSIX spec, return NaN if the input is NaN */
2114 [ - + ]: 30 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2115 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2116 : :
3514 tgl@sss.pgh.pa.us 2117 [ - + ]:CBC 30 : INIT_DEGREE_CONSTANTS();
2118 : :
2119 : : /*
2120 : : * The principal branch of the inverse cosine function maps values in the
2121 : : * range [-1, 1] to values in the range [0, 180], so we should reject any
2122 : : * inputs outside that range and the result will always be finite.
2123 : : */
3515 2124 [ + - - + ]: 30 : if (arg1 < -1.0 || arg1 > 1.0)
3515 tgl@sss.pgh.pa.us 2125 [ # # ]:UBC 0 : ereport(ERROR,
2126 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2127 : : errmsg("input is out of range")));
2128 : :
3515 tgl@sss.pgh.pa.us 2129 [ + + ]:CBC 30 : if (arg1 >= 0.0)
2130 : 18 : result = acosd_q1(arg1);
2131 : : else
2132 : 12 : result = 90.0 + asind_q1(-arg1);
2133 : :
2032 2134 [ - + ]: 30 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2135 :UBC 0 : float_overflow_error();
2136 : :
3515 tgl@sss.pgh.pa.us 2137 :CBC 30 : PG_RETURN_FLOAT8(result);
2138 : : }
2139 : :
2140 : :
2141 : : /*
2142 : : * dasind - returns the arcsin of arg1 (degrees)
2143 : : */
2144 : : Datum
2145 : 30 : dasind(PG_FUNCTION_ARGS)
2146 : : {
2147 : 30 : float8 arg1 = PG_GETARG_FLOAT8(0);
2148 : : float8 result;
2149 : :
2150 : : /* Per the POSIX spec, return NaN if the input is NaN */
2151 [ - + ]: 30 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2152 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2153 : :
3514 tgl@sss.pgh.pa.us 2154 [ - + ]:CBC 30 : INIT_DEGREE_CONSTANTS();
2155 : :
2156 : : /*
2157 : : * The principal branch of the inverse sine function maps values in the
2158 : : * range [-1, 1] to values in the range [-90, 90], so we should reject any
2159 : : * inputs outside that range and the result will always be finite.
2160 : : */
3515 2161 [ + - - + ]: 30 : if (arg1 < -1.0 || arg1 > 1.0)
3515 tgl@sss.pgh.pa.us 2162 [ # # ]:UBC 0 : ereport(ERROR,
2163 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2164 : : errmsg("input is out of range")));
2165 : :
3515 tgl@sss.pgh.pa.us 2166 [ + + ]:CBC 30 : if (arg1 >= 0.0)
2167 : 18 : result = asind_q1(arg1);
2168 : : else
2169 : 12 : result = -asind_q1(-arg1);
2170 : :
2032 2171 [ - + ]: 30 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2172 :UBC 0 : float_overflow_error();
2173 : :
3515 tgl@sss.pgh.pa.us 2174 :CBC 30 : PG_RETURN_FLOAT8(result);
2175 : : }
2176 : :
2177 : :
2178 : : /*
2179 : : * datand - returns the arctan of arg1 (degrees)
2180 : : */
2181 : : Datum
2182 : 30 : datand(PG_FUNCTION_ARGS)
2183 : : {
2184 : 30 : float8 arg1 = PG_GETARG_FLOAT8(0);
2185 : : float8 result;
2186 : : volatile float8 atan_arg1;
2187 : :
2188 : : /* Per the POSIX spec, return NaN if the input is NaN */
2189 [ - + ]: 30 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2190 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2191 : :
3514 tgl@sss.pgh.pa.us 2192 [ - + ]:CBC 30 : INIT_DEGREE_CONSTANTS();
2193 : :
2194 : : /*
2195 : : * The principal branch of the inverse tangent function maps all inputs to
2196 : : * values in the range [-90, 90], so the result should always be finite,
2197 : : * even if the input is infinite. Additionally, we take care to ensure
2198 : : * than when arg1 is 1, the result is exactly 45.
2199 : : */
3420 2200 : 30 : atan_arg1 = atan(arg1);
2201 : 30 : result = (atan_arg1 / atan_1_0) * 45.0;
2202 : :
2032 2203 [ - + ]: 30 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2204 :UBC 0 : float_overflow_error();
2205 : :
3515 tgl@sss.pgh.pa.us 2206 :CBC 30 : PG_RETURN_FLOAT8(result);
2207 : : }
2208 : :
2209 : :
2210 : : /*
2211 : : * atan2d - returns the arctan of arg1/arg2 (degrees)
2212 : : */
2213 : : Datum
2214 : 30 : datan2d(PG_FUNCTION_ARGS)
2215 : : {
2216 : 30 : float8 arg1 = PG_GETARG_FLOAT8(0);
2217 : 30 : float8 arg2 = PG_GETARG_FLOAT8(1);
2218 : : float8 result;
2219 : : volatile float8 atan2_arg1_arg2;
2220 : :
2221 : : /* Per the POSIX spec, return NaN if either input is NaN */
2222 [ + - - + ]: 30 : if (isnan(arg1) || isnan(arg2))
3515 tgl@sss.pgh.pa.us 2223 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2224 : :
3514 tgl@sss.pgh.pa.us 2225 [ - + ]:CBC 30 : INIT_DEGREE_CONSTANTS();
2226 : :
2227 : : /*
2228 : : * atan2d maps all inputs to values in the range [-180, 180], so the
2229 : : * result should always be finite, even if the inputs are infinite.
2230 : : *
2231 : : * Note: this coding assumes that atan(1.0) is a suitable scaling constant
2232 : : * to get an exact result from atan2(). This might well fail on us at
2233 : : * some point, requiring us to decide exactly what inputs we think we're
2234 : : * going to guarantee an exact result for.
2235 : : */
3420 2236 : 30 : atan2_arg1_arg2 = atan2(arg1, arg2);
2237 : 30 : result = (atan2_arg1_arg2 / atan_1_0) * 45.0;
2238 : :
2032 2239 [ - + ]: 30 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2240 :UBC 0 : float_overflow_error();
2241 : :
3515 tgl@sss.pgh.pa.us 2242 :CBC 30 : PG_RETURN_FLOAT8(result);
2243 : : }
2244 : :
2245 : :
2246 : : /*
2247 : : * sind_0_to_30 - returns the sine of an angle that lies between 0 and
2248 : : * 30 degrees. This will return exactly 0 when x is 0,
2249 : : * and exactly 0.5 when x is 30 degrees.
2250 : : */
2251 : : static double
2252 : 159 : sind_0_to_30(double x)
2253 : : {
3420 2254 : 159 : volatile float8 sin_x = sin(x * RADIANS_PER_DEGREE);
2255 : :
2256 : 159 : return (sin_x / sin_30) / 2.0;
2257 : : }
2258 : :
2259 : :
2260 : : /*
2261 : : * cosd_0_to_60 - returns the cosine of an angle that lies between 0
2262 : : * and 60 degrees. This will return exactly 1 when x
2263 : : * is 0, and exactly 0.5 when x is 60 degrees.
2264 : : */
2265 : : static double
3515 2266 : 267 : cosd_0_to_60(double x)
2267 : : {
3420 2268 : 267 : volatile float8 one_minus_cos_x = 1.0 - cos(x * RADIANS_PER_DEGREE);
2269 : :
3513 2270 : 267 : return 1.0 - (one_minus_cos_x / one_minus_cos_60) / 2.0;
2271 : : }
2272 : :
2273 : :
2274 : : /*
2275 : : * sind_q1 - returns the sine of an angle in the first quadrant
2276 : : * (0 to 90 degrees).
2277 : : */
2278 : : static double
3515 2279 : 213 : sind_q1(double x)
2280 : : {
2281 : : /*
2282 : : * Stitch together the sine and cosine functions for the ranges [0, 30]
2283 : : * and (30, 90]. These guarantee to return exact answers at their
2284 : : * endpoints, so the overall result is a continuous monotonic function
2285 : : * that gives exact results when x = 0, 30 and 90 degrees.
2286 : : */
2287 [ + + ]: 213 : if (x <= 30.0)
2288 : 105 : return sind_0_to_30(x);
2289 : : else
2290 : 108 : return cosd_0_to_60(90.0 - x);
2291 : : }
2292 : :
2293 : :
2294 : : /*
2295 : : * cosd_q1 - returns the cosine of an angle in the first quadrant
2296 : : * (0 to 90 degrees).
2297 : : */
2298 : : static double
2299 : 213 : cosd_q1(double x)
2300 : : {
2301 : : /*
2302 : : * Stitch together the sine and cosine functions for the ranges [0, 60]
2303 : : * and (60, 90]. These guarantee to return exact answers at their
2304 : : * endpoints, so the overall result is a continuous monotonic function
2305 : : * that gives exact results when x = 0, 60 and 90 degrees.
2306 : : */
2307 [ + + ]: 213 : if (x <= 60.0)
2308 : 159 : return cosd_0_to_60(x);
2309 : : else
2310 : 54 : return sind_0_to_30(90.0 - x);
2311 : : }
2312 : :
2313 : :
2314 : : /*
2315 : : * dcosd - returns the cosine of arg1 (degrees)
2316 : : */
2317 : : Datum
2318 : 99 : dcosd(PG_FUNCTION_ARGS)
2319 : : {
2320 : 99 : float8 arg1 = PG_GETARG_FLOAT8(0);
2321 : : float8 result;
3514 2322 : 99 : int sign = 1;
2323 : :
2324 : : /*
2325 : : * Per the POSIX spec, return NaN if the input is NaN and throw an error
2326 : : * if the input is infinite.
2327 : : */
3515 2328 [ - + ]: 99 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2329 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2330 : :
3515 tgl@sss.pgh.pa.us 2331 [ - + ]:CBC 99 : if (isinf(arg1))
3515 tgl@sss.pgh.pa.us 2332 [ # # ]:UBC 0 : ereport(ERROR,
2333 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2334 : : errmsg("input is out of range")));
2335 : :
3514 tgl@sss.pgh.pa.us 2336 [ - + ]:CBC 99 : INIT_DEGREE_CONSTANTS();
2337 : :
2338 : : /* Reduce the range of the input to [0,90] degrees */
3515 2339 : 99 : arg1 = fmod(arg1, 360.0);
2340 : :
2341 [ - + ]: 99 : if (arg1 < 0.0)
2342 : : {
2343 : : /* cosd(-x) = cosd(x) */
3515 tgl@sss.pgh.pa.us 2344 :UBC 0 : arg1 = -arg1;
2345 : : }
2346 : :
3515 tgl@sss.pgh.pa.us 2347 [ + + ]:CBC 99 : if (arg1 > 180.0)
2348 : : {
2349 : : /* cosd(360-x) = cosd(x) */
2350 : 27 : arg1 = 360.0 - arg1;
2351 : : }
2352 : :
2353 [ + + ]: 99 : if (arg1 > 90.0)
2354 : : {
2355 : : /* cosd(180-x) = -cosd(x) */
2356 : 27 : arg1 = 180.0 - arg1;
2357 : 27 : sign = -sign;
2358 : : }
2359 : :
2360 : 99 : result = sign * cosd_q1(arg1);
2361 : :
2032 2362 [ - + ]: 99 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2363 :UBC 0 : float_overflow_error();
2364 : :
3515 tgl@sss.pgh.pa.us 2365 :CBC 99 : PG_RETURN_FLOAT8(result);
2366 : : }
2367 : :
2368 : :
2369 : : /*
2370 : : * dcotd - returns the cotangent of arg1 (degrees)
2371 : : */
2372 : : Datum
2373 : 54 : dcotd(PG_FUNCTION_ARGS)
2374 : : {
2375 : 54 : float8 arg1 = PG_GETARG_FLOAT8(0);
2376 : : float8 result;
2377 : : volatile float8 cot_arg1;
3514 2378 : 54 : int sign = 1;
2379 : :
2380 : : /*
2381 : : * Per the POSIX spec, return NaN if the input is NaN and throw an error
2382 : : * if the input is infinite.
2383 : : */
3515 2384 [ - + ]: 54 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2385 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2386 : :
3515 tgl@sss.pgh.pa.us 2387 [ - + ]:CBC 54 : if (isinf(arg1))
3515 tgl@sss.pgh.pa.us 2388 [ # # ]:UBC 0 : ereport(ERROR,
2389 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2390 : : errmsg("input is out of range")));
2391 : :
3514 tgl@sss.pgh.pa.us 2392 [ - + ]:CBC 54 : INIT_DEGREE_CONSTANTS();
2393 : :
2394 : : /* Reduce the range of the input to [0,90] degrees */
3515 2395 : 54 : arg1 = fmod(arg1, 360.0);
2396 : :
2397 [ - + ]: 54 : if (arg1 < 0.0)
2398 : : {
2399 : : /* cotd(-x) = -cotd(x) */
3515 tgl@sss.pgh.pa.us 2400 :UBC 0 : arg1 = -arg1;
2401 : 0 : sign = -sign;
2402 : : }
2403 : :
3515 tgl@sss.pgh.pa.us 2404 [ + + ]:CBC 54 : if (arg1 > 180.0)
2405 : : {
2406 : : /* cotd(360-x) = -cotd(x) */
2407 : 18 : arg1 = 360.0 - arg1;
2408 : 18 : sign = -sign;
2409 : : }
2410 : :
2411 [ + + ]: 54 : if (arg1 > 90.0)
2412 : : {
2413 : : /* cotd(180-x) = -cotd(x) */
2414 : 18 : arg1 = 180.0 - arg1;
2415 : 18 : sign = -sign;
2416 : : }
2417 : :
3420 2418 : 54 : cot_arg1 = cosd_q1(arg1) / sind_q1(arg1);
2419 : 54 : result = sign * (cot_arg1 / cot_45);
2420 : :
2421 : : /*
2422 : : * On some machines we get cotd(270) = minus zero, but this isn't always
2423 : : * true. For portability, and because the user constituency for this
2424 : : * function probably doesn't want minus zero, force it to plain zero.
2425 : : */
3514 2426 [ + + ]: 54 : if (result == 0.0)
2427 : 12 : result = 0.0;
2428 : :
2429 : : /* Not checking for overflow because cotd(0) == Inf */
2430 : :
3515 2431 : 54 : PG_RETURN_FLOAT8(result);
2432 : : }
2433 : :
2434 : :
2435 : : /*
2436 : : * dsind - returns the sine of arg1 (degrees)
2437 : : */
2438 : : Datum
2439 : 99 : dsind(PG_FUNCTION_ARGS)
2440 : : {
2441 : 99 : float8 arg1 = PG_GETARG_FLOAT8(0);
2442 : : float8 result;
3514 2443 : 99 : int sign = 1;
2444 : :
2445 : : /*
2446 : : * Per the POSIX spec, return NaN if the input is NaN and throw an error
2447 : : * if the input is infinite.
2448 : : */
3515 2449 [ - + ]: 99 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2450 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2451 : :
3515 tgl@sss.pgh.pa.us 2452 [ - + ]:CBC 99 : if (isinf(arg1))
3515 tgl@sss.pgh.pa.us 2453 [ # # ]:UBC 0 : ereport(ERROR,
2454 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2455 : : errmsg("input is out of range")));
2456 : :
3514 tgl@sss.pgh.pa.us 2457 [ + + ]:CBC 99 : INIT_DEGREE_CONSTANTS();
2458 : :
2459 : : /* Reduce the range of the input to [0,90] degrees */
3515 2460 : 99 : arg1 = fmod(arg1, 360.0);
2461 : :
2462 [ - + ]: 99 : if (arg1 < 0.0)
2463 : : {
2464 : : /* sind(-x) = -sind(x) */
3515 tgl@sss.pgh.pa.us 2465 :UBC 0 : arg1 = -arg1;
2466 : 0 : sign = -sign;
2467 : : }
2468 : :
3515 tgl@sss.pgh.pa.us 2469 [ + + ]:CBC 99 : if (arg1 > 180.0)
2470 : : {
2471 : : /* sind(360-x) = -sind(x) */
2472 : 27 : arg1 = 360.0 - arg1;
2473 : 27 : sign = -sign;
2474 : : }
2475 : :
2476 [ + + ]: 99 : if (arg1 > 90.0)
2477 : : {
2478 : : /* sind(180-x) = sind(x) */
2479 : 27 : arg1 = 180.0 - arg1;
2480 : : }
2481 : :
2482 : 99 : result = sign * sind_q1(arg1);
2483 : :
2032 2484 [ - + ]: 99 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2485 :UBC 0 : float_overflow_error();
2486 : :
3515 tgl@sss.pgh.pa.us 2487 :CBC 99 : PG_RETURN_FLOAT8(result);
2488 : : }
2489 : :
2490 : :
2491 : : /*
2492 : : * dtand - returns the tangent of arg1 (degrees)
2493 : : */
2494 : : Datum
2495 : 54 : dtand(PG_FUNCTION_ARGS)
2496 : : {
2497 : 54 : float8 arg1 = PG_GETARG_FLOAT8(0);
2498 : : float8 result;
2499 : : volatile float8 tan_arg1;
3514 2500 : 54 : int sign = 1;
2501 : :
2502 : : /*
2503 : : * Per the POSIX spec, return NaN if the input is NaN and throw an error
2504 : : * if the input is infinite.
2505 : : */
3515 2506 [ - + ]: 54 : if (isnan(arg1))
3515 tgl@sss.pgh.pa.us 2507 :UBC 0 : PG_RETURN_FLOAT8(get_float8_nan());
2508 : :
3515 tgl@sss.pgh.pa.us 2509 [ - + ]:CBC 54 : if (isinf(arg1))
3515 tgl@sss.pgh.pa.us 2510 [ # # ]:UBC 0 : ereport(ERROR,
2511 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2512 : : errmsg("input is out of range")));
2513 : :
3514 tgl@sss.pgh.pa.us 2514 [ - + ]:CBC 54 : INIT_DEGREE_CONSTANTS();
2515 : :
2516 : : /* Reduce the range of the input to [0,90] degrees */
3515 2517 : 54 : arg1 = fmod(arg1, 360.0);
2518 : :
2519 [ - + ]: 54 : if (arg1 < 0.0)
2520 : : {
2521 : : /* tand(-x) = -tand(x) */
3515 tgl@sss.pgh.pa.us 2522 :UBC 0 : arg1 = -arg1;
2523 : 0 : sign = -sign;
2524 : : }
2525 : :
3515 tgl@sss.pgh.pa.us 2526 [ + + ]:CBC 54 : if (arg1 > 180.0)
2527 : : {
2528 : : /* tand(360-x) = -tand(x) */
2529 : 18 : arg1 = 360.0 - arg1;
2530 : 18 : sign = -sign;
2531 : : }
2532 : :
2533 [ + + ]: 54 : if (arg1 > 90.0)
2534 : : {
2535 : : /* tand(180-x) = -tand(x) */
2536 : 18 : arg1 = 180.0 - arg1;
2537 : 18 : sign = -sign;
2538 : : }
2539 : :
3420 2540 : 54 : tan_arg1 = sind_q1(arg1) / cosd_q1(arg1);
2541 : 54 : result = sign * (tan_arg1 / tan_45);
2542 : :
2543 : : /*
2544 : : * On some machines we get tand(180) = minus zero, but this isn't always
2545 : : * true. For portability, and because the user constituency for this
2546 : : * function probably doesn't want minus zero, force it to plain zero.
2547 : : */
3514 2548 [ + + ]: 54 : if (result == 0.0)
2549 : 18 : result = 0.0;
2550 : :
2551 : : /* Not checking for overflow because tand(90) == Inf */
2552 : :
3515 2553 : 54 : PG_RETURN_FLOAT8(result);
2554 : : }
2555 : :
2556 : :
2557 : : /*
2558 : : * degrees - returns degrees converted from radians
2559 : : */
2560 : : Datum
9167 2561 : 40 : degrees(PG_FUNCTION_ARGS)
2562 : : {
2563 : 40 : float8 arg1 = PG_GETARG_FLOAT8(0);
2564 : :
2596 tomas.vondra@postgre 2565 : 40 : PG_RETURN_FLOAT8(float8_div(arg1, RADIANS_PER_DEGREE));
2566 : : }
2567 : :
2568 : :
2569 : : /*
2570 : : * dpi - returns the constant PI
2571 : : */
2572 : : Datum
9167 tgl@sss.pgh.pa.us 2573 : 28 : dpi(PG_FUNCTION_ARGS)
2574 : : {
2575 : 28 : PG_RETURN_FLOAT8(M_PI);
2576 : : }
2577 : :
2578 : :
2579 : : /*
2580 : : * radians - returns radians converted from degrees
2581 : : */
2582 : : Datum
2583 : 955 : radians(PG_FUNCTION_ARGS)
2584 : : {
2585 : 955 : float8 arg1 = PG_GETARG_FLOAT8(0);
2586 : :
2596 tomas.vondra@postgre 2587 : 955 : PG_RETURN_FLOAT8(float8_mul(arg1, RADIANS_PER_DEGREE));
2588 : : }
2589 : :
2590 : :
2591 : : /* ========== HYPERBOLIC FUNCTIONS ========== */
2592 : :
2593 : :
2594 : : /*
2595 : : * dsinh - returns the hyperbolic sine of arg1
2596 : : */
2597 : : Datum
2370 tgl@sss.pgh.pa.us 2598 : 12 : dsinh(PG_FUNCTION_ARGS)
2599 : : {
2600 : 12 : float8 arg1 = PG_GETARG_FLOAT8(0);
2601 : : float8 result;
2602 : :
2603 : 12 : errno = 0;
2604 : 12 : result = sinh(arg1);
2605 : :
2606 : : /*
2607 : : * if an ERANGE error occurs, it means there is an overflow. For sinh,
2608 : : * the result should be either -infinity or infinity, depending on the
2609 : : * sign of arg1.
2610 : : */
2611 [ - + ]: 12 : if (errno == ERANGE)
2612 : : {
2370 tgl@sss.pgh.pa.us 2613 [ # # ]:UBC 0 : if (arg1 < 0)
2614 : 0 : result = -get_float8_infinity();
2615 : : else
2616 : 0 : result = get_float8_infinity();
2617 : : }
2618 : :
2370 tgl@sss.pgh.pa.us 2619 :CBC 12 : PG_RETURN_FLOAT8(result);
2620 : : }
2621 : :
2622 : :
2623 : : /*
2624 : : * dcosh - returns the hyperbolic cosine of arg1
2625 : : */
2626 : : Datum
2627 : 12 : dcosh(PG_FUNCTION_ARGS)
2628 : : {
2629 : 12 : float8 arg1 = PG_GETARG_FLOAT8(0);
2630 : : float8 result;
2631 : :
2632 : 12 : errno = 0;
2633 : 12 : result = cosh(arg1);
2634 : :
2635 : : /*
2636 : : * if an ERANGE error occurs, it means there is an overflow. As cosh is
2637 : : * always positive, it always means the result is positive infinity.
2638 : : */
2639 [ - + ]: 12 : if (errno == ERANGE)
2370 tgl@sss.pgh.pa.us 2640 :UBC 0 : result = get_float8_infinity();
2641 : :
2032 tgl@sss.pgh.pa.us 2642 [ - + ]:CBC 12 : if (unlikely(result == 0.0))
2032 tgl@sss.pgh.pa.us 2643 :UBC 0 : float_underflow_error();
2644 : :
2370 tgl@sss.pgh.pa.us 2645 :CBC 12 : PG_RETURN_FLOAT8(result);
2646 : : }
2647 : :
2648 : : /*
2649 : : * dtanh - returns the hyperbolic tangent of arg1
2650 : : */
2651 : : Datum
2652 : 12 : dtanh(PG_FUNCTION_ARGS)
2653 : : {
2654 : 12 : float8 arg1 = PG_GETARG_FLOAT8(0);
2655 : : float8 result;
2656 : :
2657 : : /*
2658 : : * For tanh, we don't need an errno check because it never overflows.
2659 : : */
2660 : 12 : result = tanh(arg1);
2661 : :
2032 2662 [ - + ]: 12 : if (unlikely(isinf(result)))
2032 tgl@sss.pgh.pa.us 2663 :UBC 0 : float_overflow_error();
2664 : :
2370 tgl@sss.pgh.pa.us 2665 :CBC 12 : PG_RETURN_FLOAT8(result);
2666 : : }
2667 : :
2668 : : /*
2669 : : * dasinh - returns the inverse hyperbolic sine of arg1
2670 : : */
2671 : : Datum
2672 : 12 : dasinh(PG_FUNCTION_ARGS)
2673 : : {
2674 : 12 : float8 arg1 = PG_GETARG_FLOAT8(0);
2675 : : float8 result;
2676 : :
2677 : : /*
2678 : : * For asinh, we don't need an errno check because it never overflows.
2679 : : */
2680 : 12 : result = asinh(arg1);
2681 : :
2682 : 12 : PG_RETURN_FLOAT8(result);
2683 : : }
2684 : :
2685 : : /*
2686 : : * dacosh - returns the inverse hyperbolic cosine of arg1
2687 : : */
2688 : : Datum
2689 : 9 : dacosh(PG_FUNCTION_ARGS)
2690 : : {
2691 : 9 : float8 arg1 = PG_GETARG_FLOAT8(0);
2692 : : float8 result;
2693 : :
2694 : : /*
2695 : : * acosh is only defined for inputs >= 1.0. By checking this ourselves,
2696 : : * we need not worry about checking for an EDOM error, which is a good
2697 : : * thing because some implementations will report that for NaN. Otherwise,
2698 : : * no error is possible.
2699 : : */
2700 [ + + ]: 9 : if (arg1 < 1.0)
2701 [ + - ]: 3 : ereport(ERROR,
2702 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2703 : : errmsg("input is out of range")));
2704 : :
2705 : 6 : result = acosh(arg1);
2706 : :
2707 : 6 : PG_RETURN_FLOAT8(result);
2708 : : }
2709 : :
2710 : : /*
2711 : : * datanh - returns the inverse hyperbolic tangent of arg1
2712 : : */
2713 : : Datum
2714 : 12 : datanh(PG_FUNCTION_ARGS)
2715 : : {
2716 : 12 : float8 arg1 = PG_GETARG_FLOAT8(0);
2717 : : float8 result;
2718 : :
2719 : : /*
2720 : : * atanh is only defined for inputs between -1 and 1. By checking this
2721 : : * ourselves, we need not worry about checking for an EDOM error, which is
2722 : : * a good thing because some implementations will report that for NaN.
2723 : : */
2724 [ + + + + ]: 12 : if (arg1 < -1.0 || arg1 > 1.0)
2725 [ + - ]: 6 : ereport(ERROR,
2726 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
2727 : : errmsg("input is out of range")));
2728 : :
2729 : : /*
2730 : : * Also handle the infinity cases ourselves; this is helpful because old
2731 : : * glibc versions may produce the wrong errno for this. All other inputs
2732 : : * cannot produce an error.
2733 : : */
2734 [ - + ]: 6 : if (arg1 == -1.0)
2370 tgl@sss.pgh.pa.us 2735 :UBC 0 : result = -get_float8_infinity();
2370 tgl@sss.pgh.pa.us 2736 [ - + ]:CBC 6 : else if (arg1 == 1.0)
2370 tgl@sss.pgh.pa.us 2737 :UBC 0 : result = get_float8_infinity();
2738 : : else
2370 tgl@sss.pgh.pa.us 2739 :CBC 6 : result = atanh(arg1);
2740 : :
2741 : 6 : PG_RETURN_FLOAT8(result);
2742 : : }
2743 : :
2744 : :
2745 : : /* ========== ERROR FUNCTIONS ========== */
2746 : :
2747 : :
2748 : : /*
2749 : : * derf - returns the error function: erf(arg1)
2750 : : */
2751 : : Datum
907 dean.a.rasheed@gmail 2752 : 3066 : derf(PG_FUNCTION_ARGS)
2753 : : {
2754 : 3066 : float8 arg1 = PG_GETARG_FLOAT8(0);
2755 : : float8 result;
2756 : :
2757 : : /*
2758 : : * For erf, we don't need an errno check because it never overflows.
2759 : : */
2760 : 3066 : result = erf(arg1);
2761 : :
2762 [ - + ]: 3066 : if (unlikely(isinf(result)))
907 dean.a.rasheed@gmail 2763 :UBC 0 : float_overflow_error();
2764 : :
907 dean.a.rasheed@gmail 2765 :CBC 3066 : PG_RETURN_FLOAT8(result);
2766 : : }
2767 : :
2768 : : /*
2769 : : * derfc - returns the complementary error function: 1 - erf(arg1)
2770 : : */
2771 : : Datum
2772 : 66 : derfc(PG_FUNCTION_ARGS)
2773 : : {
2774 : 66 : float8 arg1 = PG_GETARG_FLOAT8(0);
2775 : : float8 result;
2776 : :
2777 : : /*
2778 : : * For erfc, we don't need an errno check because it never overflows.
2779 : : */
2780 : 66 : result = erfc(arg1);
2781 : :
2782 [ - + ]: 66 : if (unlikely(isinf(result)))
907 dean.a.rasheed@gmail 2783 :UBC 0 : float_overflow_error();
2784 : :
907 dean.a.rasheed@gmail 2785 :CBC 66 : PG_RETURN_FLOAT8(result);
2786 : : }
2787 : :
2788 : :
2789 : : /* ========== GAMMA FUNCTIONS ========== */
2790 : :
2791 : :
2792 : : /*
2793 : : * dgamma - returns the gamma function of arg1
2794 : : */
2795 : : Datum
164 2796 : 39 : dgamma(PG_FUNCTION_ARGS)
2797 : : {
2798 : 39 : float8 arg1 = PG_GETARG_FLOAT8(0);
2799 : : float8 result;
2800 : :
2801 : : /*
2802 : : * Handle NaN and Inf cases explicitly. This simplifies the overflow
2803 : : * checks on platforms that do not set errno.
2804 : : */
2805 [ + + ]: 39 : if (isnan(arg1))
2806 : 3 : result = arg1;
2807 [ + + ]: 36 : else if (isinf(arg1))
2808 : : {
2809 : : /* Per POSIX, an input of -Inf causes a domain error */
2810 [ + + ]: 6 : if (arg1 < 0)
2811 : : {
2812 : 3 : float_overflow_error();
2813 : : result = get_float8_nan(); /* keep compiler quiet */
2814 : : }
2815 : : else
2816 : 3 : result = arg1;
2817 : : }
2818 : : else
2819 : : {
2820 : : /*
2821 : : * Note: the POSIX/C99 gamma function is called "tgamma", not "gamma".
2822 : : *
2823 : : * On some platforms, tgamma() will not set errno but just return Inf,
2824 : : * NaN, or zero to report overflow/underflow; therefore, test those
2825 : : * cases explicitly (note that, like the exponential function, the
2826 : : * gamma function has no zeros).
2827 : : */
2828 : 30 : errno = 0;
2829 : 30 : result = tgamma(arg1);
2830 : :
2831 [ + + + - : 30 : if (errno != 0 || isinf(result) || isnan(result))
- + ]
2832 : : {
2833 [ + + ]: 12 : if (result != 0.0)
2834 : 9 : float_overflow_error();
2835 : : else
2836 : 3 : float_underflow_error();
2837 : : }
2838 [ - + ]: 18 : else if (result == 0.0)
164 dean.a.rasheed@gmail 2839 :UBC 0 : float_underflow_error();
2840 : : }
2841 : :
164 dean.a.rasheed@gmail 2842 :CBC 24 : PG_RETURN_FLOAT8(result);
2843 : : }
2844 : :
2845 : :
2846 : : /*
2847 : : * dlgamma - natural logarithm of absolute value of gamma of arg1
2848 : : */
2849 : : Datum
2850 : 42 : dlgamma(PG_FUNCTION_ARGS)
2851 : : {
2852 : 42 : float8 arg1 = PG_GETARG_FLOAT8(0);
2853 : : float8 result;
2854 : :
2855 : : /*
2856 : : * Note: lgamma may not be thread-safe because it may write to a global
2857 : : * variable signgam, which may not be thread-local. However, this doesn't
2858 : : * matter to us, since we don't use signgam.
2859 : : */
2860 : 42 : errno = 0;
2861 : 42 : result = lgamma(arg1);
2862 : :
2863 : : /*
2864 : : * If an ERANGE error occurs, it means there was an overflow or a pole
2865 : : * error (which happens for zero and negative integer inputs).
2866 : : *
2867 : : * On some platforms, lgamma() will not set errno but just return infinity
2868 : : * to report overflow, but it should never underflow.
2869 : : */
2870 [ + + + + : 42 : if (errno == ERANGE || (isinf(result) && !isinf(arg1)))
- + ]
2871 : 9 : float_overflow_error();
2872 : :
2873 : 33 : PG_RETURN_FLOAT8(result);
2874 : : }
2875 : :
2876 : :
2877 : :
2878 : : /*
2879 : : * =========================
2880 : : * FLOAT AGGREGATE OPERATORS
2881 : : * =========================
2882 : : *
2883 : : * float8_accum - accumulate for AVG(), variance aggregates, etc.
2884 : : * float4_accum - same, but input data is float4
2885 : : * float8_avg - produce final result for float AVG()
2886 : : * float8_var_samp - produce final result for float VAR_SAMP()
2887 : : * float8_var_pop - produce final result for float VAR_POP()
2888 : : * float8_stddev_samp - produce final result for float STDDEV_SAMP()
2889 : : * float8_stddev_pop - produce final result for float STDDEV_POP()
2890 : : *
2891 : : * The naive schoolbook implementation of these aggregates works by
2892 : : * accumulating sum(X) and sum(X^2). However, this approach suffers from
2893 : : * large rounding errors in the final computation of quantities like the
2894 : : * population variance (N*sum(X^2) - sum(X)^2) / N^2, since each of the
2895 : : * intermediate terms is potentially very large, while the difference is often
2896 : : * quite small.
2897 : : *
2898 : : * Instead we use the Youngs-Cramer algorithm [1] which works by accumulating
2899 : : * Sx=sum(X) and Sxx=sum((X-Sx/N)^2), using a numerically stable algorithm to
2900 : : * incrementally update those quantities. The final computations of each of
2901 : : * the aggregate values is then trivial and gives more accurate results (for
2902 : : * example, the population variance is just Sxx/N). This algorithm is also
2903 : : * fairly easy to generalize to allow parallel execution without loss of
2904 : : * precision (see, for example, [2]). For more details, and a comparison of
2905 : : * this with other algorithms, see [3].
2906 : : *
2907 : : * The transition datatype for all these aggregates is a 3-element array
2908 : : * of float8, holding the values N, Sx, Sxx in that order.
2909 : : *
2910 : : * Note that we represent N as a float to avoid having to build a special
2911 : : * datatype. Given a reasonable floating-point implementation, there should
2912 : : * be no accuracy loss unless N exceeds 2 ^ 52 or so (by which time the
2913 : : * user will have doubtless lost interest anyway...)
2914 : : *
2915 : : * [1] Some Results Relevant to Choice of Sum and Sum-of-Product Algorithms,
2916 : : * E. A. Youngs and E. M. Cramer, Technometrics Vol 13, No 3, August 1971.
2917 : : *
2918 : : * [2] Updating Formulae and a Pairwise Algorithm for Computing Sample
2919 : : * Variances, T. F. Chan, G. H. Golub & R. J. LeVeque, COMPSTAT 1982.
2920 : : *
2921 : : * [3] Numerically Stable Parallel Computation of (Co-)Variance, Erich
2922 : : * Schubert and Michael Gertz, Proceedings of the 30th International
2923 : : * Conference on Scientific and Statistical Database Management, 2018.
2924 : : */
2925 : :
2926 : : static float8 *
6980 tgl@sss.pgh.pa.us 2927 : 746 : check_float8_array(ArrayType *transarray, const char *caller, int n)
2928 : : {
2929 : : /*
2930 : : * We expect the input to be an N-element float array; verify that. We
2931 : : * don't need to use deconstruct_array() since the array data is just
2932 : : * going to look like a C array of N float8 values.
2933 : : */
8412 2934 [ + - ]: 746 : if (ARR_NDIM(transarray) != 1 ||
6980 2935 [ + - ]: 746 : ARR_DIMS(transarray)[0] != n ||
7233 2936 [ + - ]: 746 : ARR_HASNULL(transarray) ||
8412 2937 [ - + ]: 746 : ARR_ELEMTYPE(transarray) != FLOAT8OID)
6980 tgl@sss.pgh.pa.us 2938 [ # # ]:UBC 0 : elog(ERROR, "%s: expected %d-element float8 array", caller, n);
9182 tgl@sss.pgh.pa.us 2939 [ - + ]:CBC 746 : return (float8 *) ARR_DATA_PTR(transarray);
2940 : : }
2941 : :
2942 : : /*
2943 : : * float8_combine
2944 : : *
2945 : : * An aggregate combine function used to combine two 3 fields
2946 : : * aggregate transition data into a single transition data.
2947 : : * This function is used only in two stage aggregation and
2948 : : * shouldn't be called outside aggregate context.
2949 : : */
2950 : : Datum
3438 rhaas@postgresql.org 2951 : 9 : float8_combine(PG_FUNCTION_ARGS)
2952 : : {
2953 : 9 : ArrayType *transarray1 = PG_GETARG_ARRAYTYPE_P(0);
2954 : 9 : ArrayType *transarray2 = PG_GETARG_ARRAYTYPE_P(1);
2955 : : float8 *transvalues1;
2956 : : float8 *transvalues2;
2957 : : float8 N1,
2958 : : Sx1,
2959 : : Sxx1,
2960 : : N2,
2961 : : Sx2,
2962 : : Sxx2,
2963 : : tmp,
2964 : : N,
2965 : : Sx,
2966 : : Sxx;
2967 : :
2968 : 9 : transvalues1 = check_float8_array(transarray1, "float8_combine", 3);
2969 : 9 : transvalues2 = check_float8_array(transarray2, "float8_combine", 3);
2970 : :
2527 dean.a.rasheed@gmail 2971 : 9 : N1 = transvalues1[0];
2972 : 9 : Sx1 = transvalues1[1];
2973 : 9 : Sxx1 = transvalues1[2];
2974 : :
2975 : 9 : N2 = transvalues2[0];
2976 : 9 : Sx2 = transvalues2[1];
2977 : 9 : Sxx2 = transvalues2[2];
2978 : :
2979 : : /*--------------------
2980 : : * The transition values combine using a generalization of the
2981 : : * Youngs-Cramer algorithm as follows:
2982 : : *
2983 : : * N = N1 + N2
2984 : : * Sx = Sx1 + Sx2
2985 : : * Sxx = Sxx1 + Sxx2 + N1 * N2 * (Sx1/N1 - Sx2/N2)^2 / N;
2986 : : *
2987 : : * It's worth handling the special cases N1 = 0 and N2 = 0 separately
2988 : : * since those cases are trivial, and we then don't need to worry about
2989 : : * division-by-zero errors in the general case.
2990 : : *--------------------
2991 : : */
2992 [ + + ]: 9 : if (N1 == 0.0)
2993 : : {
2994 : 3 : N = N2;
2995 : 3 : Sx = Sx2;
2996 : 3 : Sxx = Sxx2;
2997 : : }
2998 [ + + ]: 6 : else if (N2 == 0.0)
2999 : : {
3000 : 3 : N = N1;
3001 : 3 : Sx = Sx1;
3002 : 3 : Sxx = Sxx1;
3003 : : }
3004 : : else
3005 : : {
3006 : 3 : N = N1 + N2;
3007 : 3 : Sx = float8_pl(Sx1, Sx2);
3008 : 3 : tmp = Sx1 / N1 - Sx2 / N2;
3009 : 3 : Sxx = Sxx1 + Sxx2 + N1 * N2 * tmp * tmp / N;
2032 tgl@sss.pgh.pa.us 3010 [ - + - - : 3 : if (unlikely(isinf(Sxx)) && !isinf(Sxx1) && !isinf(Sxx2))
- - ]
2032 tgl@sss.pgh.pa.us 3011 :UBC 0 : float_overflow_error();
3012 : : }
3013 : :
3014 : : /*
3015 : : * If we're invoked as an aggregate, we can cheat and modify our first
3016 : : * parameter in-place to reduce palloc overhead. Otherwise we construct a
3017 : : * new array with the updated transition data and return it.
3018 : : */
2527 dean.a.rasheed@gmail 3019 [ - + ]:CBC 9 : if (AggCheckCallContext(fcinfo, NULL))
3020 : : {
2527 dean.a.rasheed@gmail 3021 :UBC 0 : transvalues1[0] = N;
3022 : 0 : transvalues1[1] = Sx;
3023 : 0 : transvalues1[2] = Sxx;
3024 : :
3025 : 0 : PG_RETURN_ARRAYTYPE_P(transarray1);
3026 : : }
3027 : : else
3028 : : {
3029 : : Datum transdatums[3];
3030 : : ArrayType *result;
3031 : :
2527 dean.a.rasheed@gmail 3032 :CBC 9 : transdatums[0] = Float8GetDatumFast(N);
3033 : 9 : transdatums[1] = Float8GetDatumFast(Sx);
3034 : 9 : transdatums[2] = Float8GetDatumFast(Sxx);
3035 : :
327 msawada@postgresql.o 3036 : 9 : result = construct_array_builtin(transdatums, 3, FLOAT8OID);
3037 : :
2527 dean.a.rasheed@gmail 3038 : 9 : PG_RETURN_ARRAYTYPE_P(result);
3039 : : }
3040 : : }
3041 : :
3042 : : Datum
9182 tgl@sss.pgh.pa.us 3043 : 181 : float8_accum(PG_FUNCTION_ARGS)
3044 : : {
3045 : 181 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3046 : 181 : float8 newval = PG_GETARG_FLOAT8(1);
3047 : : float8 *transvalues;
3048 : : float8 N,
3049 : : Sx,
3050 : : Sxx,
3051 : : tmp;
3052 : :
6980 3053 : 181 : transvalues = check_float8_array(transarray, "float8_accum", 3);
2527 dean.a.rasheed@gmail 3054 : 181 : N = transvalues[0];
3055 : 181 : Sx = transvalues[1];
3056 : 181 : Sxx = transvalues[2];
3057 : :
3058 : : /*
3059 : : * Use the Youngs-Cramer algorithm to incorporate the new value into the
3060 : : * transition values.
3061 : : */
3062 : 181 : N += 1.0;
3063 : 181 : Sx += newval;
3064 [ + + ]: 181 : if (transvalues[0] > 0.0)
3065 : : {
3066 : 132 : tmp = newval * N - Sx;
3067 : 132 : Sxx += tmp * tmp / (N * transvalues[0]);
3068 : :
3069 : : /*
3070 : : * Overflow check. We only report an overflow error when finite
3071 : : * inputs lead to infinite results. Note also that Sxx should be NaN
3072 : : * if any of the inputs are infinite, so we intentionally prevent Sxx
3073 : : * from becoming infinite.
3074 : : */
3075 [ + + - + ]: 132 : if (isinf(Sx) || isinf(Sxx))
3076 : : {
3077 [ + + + + : 12 : if (!isinf(transvalues[1]) && !isinf(newval))
- + ]
2032 tgl@sss.pgh.pa.us 3078 :UBC 0 : float_overflow_error();
3079 : :
2527 dean.a.rasheed@gmail 3080 :CBC 12 : Sxx = get_float8_nan();
3081 : : }
3082 : : }
3083 : : else
3084 : : {
3085 : : /*
3086 : : * At the first input, we normally can leave Sxx as 0. However, if
3087 : : * the first input is Inf or NaN, we'd better force Sxx to NaN;
3088 : : * otherwise we will falsely report variance zero when there are no
3089 : : * more inputs.
3090 : : */
1911 tgl@sss.pgh.pa.us 3091 [ + + + + ]: 49 : if (isnan(newval) || isinf(newval))
3092 : 24 : Sxx = get_float8_nan();
3093 : : }
3094 : :
3095 : : /*
3096 : : * If we're invoked as an aggregate, we can cheat and modify our first
3097 : : * parameter in-place to reduce palloc overhead. Otherwise we construct a
3098 : : * new array with the updated transition data and return it.
3099 : : */
5689 3100 [ + + ]: 181 : if (AggCheckCallContext(fcinfo, NULL))
3101 : : {
2527 dean.a.rasheed@gmail 3102 : 178 : transvalues[0] = N;
3103 : 178 : transvalues[1] = Sx;
3104 : 178 : transvalues[2] = Sxx;
3105 : :
7458 neilc@samurai.com 3106 : 178 : PG_RETURN_ARRAYTYPE_P(transarray);
3107 : : }
3108 : : else
3109 : : {
3110 : : Datum transdatums[3];
3111 : : ArrayType *result;
3112 : :
2527 dean.a.rasheed@gmail 3113 : 3 : transdatums[0] = Float8GetDatumFast(N);
3114 : 3 : transdatums[1] = Float8GetDatumFast(Sx);
3115 : 3 : transdatums[2] = Float8GetDatumFast(Sxx);
3116 : :
327 msawada@postgresql.o 3117 : 3 : result = construct_array_builtin(transdatums, 3, FLOAT8OID);
3118 : :
7458 neilc@samurai.com 3119 : 3 : PG_RETURN_ARRAYTYPE_P(result);
3120 : : }
3121 : : }
3122 : :
3123 : : Datum
9182 tgl@sss.pgh.pa.us 3124 : 144 : float4_accum(PG_FUNCTION_ARGS)
3125 : : {
3126 : 144 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3127 : :
3128 : : /* do computations as float8 */
6822 bruce@momjian.us 3129 : 144 : float8 newval = PG_GETARG_FLOAT4(1);
3130 : : float8 *transvalues;
3131 : : float8 N,
3132 : : Sx,
3133 : : Sxx,
3134 : : tmp;
3135 : :
6980 tgl@sss.pgh.pa.us 3136 : 144 : transvalues = check_float8_array(transarray, "float4_accum", 3);
2527 dean.a.rasheed@gmail 3137 : 144 : N = transvalues[0];
3138 : 144 : Sx = transvalues[1];
3139 : 144 : Sxx = transvalues[2];
3140 : :
3141 : : /*
3142 : : * Use the Youngs-Cramer algorithm to incorporate the new value into the
3143 : : * transition values.
3144 : : */
3145 : 144 : N += 1.0;
3146 : 144 : Sx += newval;
3147 [ + + ]: 144 : if (transvalues[0] > 0.0)
3148 : : {
3149 : 102 : tmp = newval * N - Sx;
3150 : 102 : Sxx += tmp * tmp / (N * transvalues[0]);
3151 : :
3152 : : /*
3153 : : * Overflow check. We only report an overflow error when finite
3154 : : * inputs lead to infinite results. Note also that Sxx should be NaN
3155 : : * if any of the inputs are infinite, so we intentionally prevent Sxx
3156 : : * from becoming infinite.
3157 : : */
3158 [ + - - + ]: 102 : if (isinf(Sx) || isinf(Sxx))
3159 : : {
2527 dean.a.rasheed@gmail 3160 [ # # # # :UBC 0 : if (!isinf(transvalues[1]) && !isinf(newval))
# # ]
2032 tgl@sss.pgh.pa.us 3161 : 0 : float_overflow_error();
3162 : :
2527 dean.a.rasheed@gmail 3163 : 0 : Sxx = get_float8_nan();
3164 : : }
3165 : : }
3166 : : else
3167 : : {
3168 : : /*
3169 : : * At the first input, we normally can leave Sxx as 0. However, if
3170 : : * the first input is Inf or NaN, we'd better force Sxx to NaN;
3171 : : * otherwise we will falsely report variance zero when there are no
3172 : : * more inputs.
3173 : : */
1911 tgl@sss.pgh.pa.us 3174 [ + + + + ]:CBC 42 : if (isnan(newval) || isinf(newval))
3175 : 12 : Sxx = get_float8_nan();
3176 : : }
3177 : :
3178 : : /*
3179 : : * If we're invoked as an aggregate, we can cheat and modify our first
3180 : : * parameter in-place to reduce palloc overhead. Otherwise we construct a
3181 : : * new array with the updated transition data and return it.
3182 : : */
5689 3183 [ + - ]: 144 : if (AggCheckCallContext(fcinfo, NULL))
3184 : : {
2527 dean.a.rasheed@gmail 3185 : 144 : transvalues[0] = N;
3186 : 144 : transvalues[1] = Sx;
3187 : 144 : transvalues[2] = Sxx;
3188 : :
7458 neilc@samurai.com 3189 : 144 : PG_RETURN_ARRAYTYPE_P(transarray);
3190 : : }
3191 : : else
3192 : : {
3193 : : Datum transdatums[3];
3194 : : ArrayType *result;
3195 : :
2527 dean.a.rasheed@gmail 3196 :UBC 0 : transdatums[0] = Float8GetDatumFast(N);
3197 : 0 : transdatums[1] = Float8GetDatumFast(Sx);
3198 : 0 : transdatums[2] = Float8GetDatumFast(Sxx);
3199 : :
327 msawada@postgresql.o 3200 : 0 : result = construct_array_builtin(transdatums, 3, FLOAT8OID);
3201 : :
7458 neilc@samurai.com 3202 : 0 : PG_RETURN_ARRAYTYPE_P(result);
3203 : : }
3204 : : }
3205 : :
3206 : : Datum
9182 tgl@sss.pgh.pa.us 3207 :CBC 31 : float8_avg(PG_FUNCTION_ARGS)
3208 : : {
3209 : 31 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3210 : : float8 *transvalues;
3211 : : float8 N,
3212 : : Sx;
3213 : :
6980 3214 : 31 : transvalues = check_float8_array(transarray, "float8_avg", 3);
9182 3215 : 31 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3216 : 31 : Sx = transvalues[1];
3217 : : /* ignore Sxx */
3218 : :
3219 : : /* SQL defines AVG of no values to be NULL */
9182 tgl@sss.pgh.pa.us 3220 [ + + ]: 31 : if (N == 0.0)
3221 : 3 : PG_RETURN_NULL();
3222 : :
2527 dean.a.rasheed@gmail 3223 : 28 : PG_RETURN_FLOAT8(Sx / N);
3224 : : }
3225 : :
3226 : : Datum
7120 neilc@samurai.com 3227 : 42 : float8_var_pop(PG_FUNCTION_ARGS)
3228 : : {
9182 tgl@sss.pgh.pa.us 3229 : 42 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3230 : : float8 *transvalues;
3231 : : float8 N,
3232 : : Sxx;
3233 : :
6980 3234 : 42 : transvalues = check_float8_array(transarray, "float8_var_pop", 3);
7120 neilc@samurai.com 3235 : 42 : N = transvalues[0];
3236 : : /* ignore Sx */
2527 dean.a.rasheed@gmail 3237 : 42 : Sxx = transvalues[2];
3238 : :
3239 : : /* Population variance is undefined when N is 0, so return NULL */
7120 neilc@samurai.com 3240 [ - + ]: 42 : if (N == 0.0)
7120 neilc@samurai.com 3241 :UBC 0 : PG_RETURN_NULL();
3242 : :
3243 : : /* Note that Sxx is guaranteed to be non-negative */
3244 : :
2527 dean.a.rasheed@gmail 3245 :CBC 42 : PG_RETURN_FLOAT8(Sxx / N);
3246 : : }
3247 : :
3248 : : Datum
7120 neilc@samurai.com 3249 : 21 : float8_var_samp(PG_FUNCTION_ARGS)
3250 : : {
3251 : 21 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3252 : : float8 *transvalues;
3253 : : float8 N,
3254 : : Sxx;
3255 : :
6980 tgl@sss.pgh.pa.us 3256 : 21 : transvalues = check_float8_array(transarray, "float8_var_samp", 3);
9182 3257 : 21 : N = transvalues[0];
3258 : : /* ignore Sx */
2527 dean.a.rasheed@gmail 3259 : 21 : Sxx = transvalues[2];
3260 : :
3261 : : /* Sample variance is undefined when N is 0 or 1, so return NULL */
9182 tgl@sss.pgh.pa.us 3262 [ + + ]: 21 : if (N <= 1.0)
8174 3263 : 18 : PG_RETURN_NULL();
3264 : :
3265 : : /* Note that Sxx is guaranteed to be non-negative */
3266 : :
2527 dean.a.rasheed@gmail 3267 : 3 : PG_RETURN_FLOAT8(Sxx / (N - 1.0));
3268 : : }
3269 : :
3270 : : Datum
7120 neilc@samurai.com 3271 : 21 : float8_stddev_pop(PG_FUNCTION_ARGS)
3272 : : {
3273 : 21 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3274 : : float8 *transvalues;
3275 : : float8 N,
3276 : : Sxx;
3277 : :
6980 tgl@sss.pgh.pa.us 3278 : 21 : transvalues = check_float8_array(transarray, "float8_stddev_pop", 3);
7120 neilc@samurai.com 3279 : 21 : N = transvalues[0];
3280 : : /* ignore Sx */
2527 dean.a.rasheed@gmail 3281 : 21 : Sxx = transvalues[2];
3282 : :
3283 : : /* Population stddev is undefined when N is 0, so return NULL */
7120 neilc@samurai.com 3284 [ - + ]: 21 : if (N == 0.0)
7120 neilc@samurai.com 3285 :UBC 0 : PG_RETURN_NULL();
3286 : :
3287 : : /* Note that Sxx is guaranteed to be non-negative */
3288 : :
2527 dean.a.rasheed@gmail 3289 :CBC 21 : PG_RETURN_FLOAT8(sqrt(Sxx / N));
3290 : : }
3291 : :
3292 : : Datum
7120 neilc@samurai.com 3293 : 24 : float8_stddev_samp(PG_FUNCTION_ARGS)
3294 : : {
9182 tgl@sss.pgh.pa.us 3295 : 24 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3296 : : float8 *transvalues;
3297 : : float8 N,
3298 : : Sxx;
3299 : :
6980 3300 : 24 : transvalues = check_float8_array(transarray, "float8_stddev_samp", 3);
9182 3301 : 24 : N = transvalues[0];
3302 : : /* ignore Sx */
2527 dean.a.rasheed@gmail 3303 : 24 : Sxx = transvalues[2];
3304 : :
3305 : : /* Sample stddev is undefined when N is 0 or 1, so return NULL */
9182 tgl@sss.pgh.pa.us 3306 [ + + ]: 24 : if (N <= 1.0)
8174 3307 : 18 : PG_RETURN_NULL();
3308 : :
3309 : : /* Note that Sxx is guaranteed to be non-negative */
3310 : :
2527 dean.a.rasheed@gmail 3311 : 6 : PG_RETURN_FLOAT8(sqrt(Sxx / (N - 1.0)));
3312 : : }
3313 : :
3314 : : /*
3315 : : * =========================
3316 : : * SQL2003 BINARY AGGREGATES
3317 : : * =========================
3318 : : *
3319 : : * As with the preceding aggregates, we use the Youngs-Cramer algorithm to
3320 : : * reduce rounding errors in the aggregate final functions.
3321 : : *
3322 : : * The transition datatype for all these aggregates is a 6-element array of
3323 : : * float8, holding the values N, Sx=sum(X), Sxx=sum((X-Sx/N)^2), Sy=sum(Y),
3324 : : * Syy=sum((Y-Sy/N)^2), Sxy=sum((X-Sx/N)*(Y-Sy/N)) in that order.
3325 : : *
3326 : : * Note that Y is the first argument to all these aggregates!
3327 : : *
3328 : : * It might seem attractive to optimize this by having multiple accumulator
3329 : : * functions that only calculate the sums actually needed. But on most
3330 : : * modern machines, a couple of extra floating-point multiplies will be
3331 : : * insignificant compared to the other per-tuple overhead, so I've chosen
3332 : : * to minimize code space instead.
3333 : : */
3334 : :
3335 : : Datum
6980 tgl@sss.pgh.pa.us 3336 : 159 : float8_regr_accum(PG_FUNCTION_ARGS)
3337 : : {
3338 : 159 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3339 : 159 : float8 newvalY = PG_GETARG_FLOAT8(1);
3340 : 159 : float8 newvalX = PG_GETARG_FLOAT8(2);
3341 : : float8 *transvalues;
3342 : : float8 N,
3343 : : Sx,
3344 : : Sxx,
3345 : : Sy,
3346 : : Syy,
3347 : : Sxy,
3348 : : tmpX,
3349 : : tmpY,
3350 : : scale;
3351 : :
3352 : 159 : transvalues = check_float8_array(transarray, "float8_regr_accum", 6);
3353 : 159 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3354 : 159 : Sx = transvalues[1];
3355 : 159 : Sxx = transvalues[2];
3356 : 159 : Sy = transvalues[3];
3357 : 159 : Syy = transvalues[4];
3358 : 159 : Sxy = transvalues[5];
3359 : :
3360 : : /*
3361 : : * Use the Youngs-Cramer algorithm to incorporate the new values into the
3362 : : * transition values.
3363 : : */
6980 tgl@sss.pgh.pa.us 3364 : 159 : N += 1.0;
2527 dean.a.rasheed@gmail 3365 : 159 : Sx += newvalX;
3366 : 159 : Sy += newvalY;
3367 [ + + ]: 159 : if (transvalues[0] > 0.0)
3368 : : {
3369 : 105 : tmpX = newvalX * N - Sx;
3370 : 105 : tmpY = newvalY * N - Sy;
3371 : 105 : scale = 1.0 / (N * transvalues[0]);
3372 : 105 : Sxx += tmpX * tmpX * scale;
3373 : 105 : Syy += tmpY * tmpY * scale;
3374 : 105 : Sxy += tmpX * tmpY * scale;
3375 : :
3376 : : /*
3377 : : * Overflow check. We only report an overflow error when finite
3378 : : * inputs lead to infinite results. Note also that Sxx, Syy and Sxy
3379 : : * should be NaN if any of the relevant inputs are infinite, so we
3380 : : * intentionally prevent them from becoming infinite.
3381 : : */
3382 [ + - + - : 105 : if (isinf(Sx) || isinf(Sxx) || isinf(Sy) || isinf(Syy) || isinf(Sxy))
+ - + - -
+ ]
3383 : : {
2527 dean.a.rasheed@gmail 3384 [ # # # # ]:UBC 0 : if (((isinf(Sx) || isinf(Sxx)) &&
3385 [ # # # # ]: 0 : !isinf(transvalues[1]) && !isinf(newvalX)) ||
3386 [ # # # # ]: 0 : ((isinf(Sy) || isinf(Syy)) &&
3387 [ # # # # ]: 0 : !isinf(transvalues[3]) && !isinf(newvalY)) ||
3388 [ # # ]: 0 : (isinf(Sxy) &&
3389 [ # # # # ]: 0 : !isinf(transvalues[1]) && !isinf(newvalX) &&
3390 [ # # # # ]: 0 : !isinf(transvalues[3]) && !isinf(newvalY)))
2032 tgl@sss.pgh.pa.us 3391 : 0 : float_overflow_error();
3392 : :
2527 dean.a.rasheed@gmail 3393 [ # # ]: 0 : if (isinf(Sxx))
3394 : 0 : Sxx = get_float8_nan();
3395 [ # # ]: 0 : if (isinf(Syy))
3396 : 0 : Syy = get_float8_nan();
3397 [ # # ]: 0 : if (isinf(Sxy))
3398 : 0 : Sxy = get_float8_nan();
3399 : : }
3400 : : }
3401 : : else
3402 : : {
3403 : : /*
3404 : : * At the first input, we normally can leave Sxx et al as 0. However,
3405 : : * if the first input is Inf or NaN, we'd better force the dependent
3406 : : * sums to NaN; otherwise we will falsely report variance zero when
3407 : : * there are no more inputs.
3408 : : */
1911 tgl@sss.pgh.pa.us 3409 [ + + + + ]:CBC 54 : if (isnan(newvalX) || isinf(newvalX))
3410 : 12 : Sxx = Sxy = get_float8_nan();
3411 [ + - - + ]: 54 : if (isnan(newvalY) || isinf(newvalY))
1911 tgl@sss.pgh.pa.us 3412 :UBC 0 : Syy = Sxy = get_float8_nan();
3413 : : }
3414 : :
3415 : : /*
3416 : : * If we're invoked as an aggregate, we can cheat and modify our first
3417 : : * parameter in-place to reduce palloc overhead. Otherwise we construct a
3418 : : * new array with the updated transition data and return it.
3419 : : */
5689 tgl@sss.pgh.pa.us 3420 [ + + ]:CBC 159 : if (AggCheckCallContext(fcinfo, NULL))
3421 : : {
6980 3422 : 156 : transvalues[0] = N;
2527 dean.a.rasheed@gmail 3423 : 156 : transvalues[1] = Sx;
3424 : 156 : transvalues[2] = Sxx;
3425 : 156 : transvalues[3] = Sy;
3426 : 156 : transvalues[4] = Syy;
3427 : 156 : transvalues[5] = Sxy;
3428 : :
6980 tgl@sss.pgh.pa.us 3429 : 156 : PG_RETURN_ARRAYTYPE_P(transarray);
3430 : : }
3431 : : else
3432 : : {
3433 : : Datum transdatums[6];
3434 : : ArrayType *result;
3435 : :
3436 : 3 : transdatums[0] = Float8GetDatumFast(N);
2527 dean.a.rasheed@gmail 3437 : 3 : transdatums[1] = Float8GetDatumFast(Sx);
3438 : 3 : transdatums[2] = Float8GetDatumFast(Sxx);
3439 : 3 : transdatums[3] = Float8GetDatumFast(Sy);
3440 : 3 : transdatums[4] = Float8GetDatumFast(Syy);
3441 : 3 : transdatums[5] = Float8GetDatumFast(Sxy);
3442 : :
327 msawada@postgresql.o 3443 : 3 : result = construct_array_builtin(transdatums, 6, FLOAT8OID);
3444 : :
6980 tgl@sss.pgh.pa.us 3445 : 3 : PG_RETURN_ARRAYTYPE_P(result);
3446 : : }
3447 : : }
3448 : :
3449 : : /*
3450 : : * float8_regr_combine
3451 : : *
3452 : : * An aggregate combine function used to combine two 6 fields
3453 : : * aggregate transition data into a single transition data.
3454 : : * This function is used only in two stage aggregation and
3455 : : * shouldn't be called outside aggregate context.
3456 : : */
3457 : : Datum
3438 rhaas@postgresql.org 3458 : 9 : float8_regr_combine(PG_FUNCTION_ARGS)
3459 : : {
3460 : 9 : ArrayType *transarray1 = PG_GETARG_ARRAYTYPE_P(0);
3461 : 9 : ArrayType *transarray2 = PG_GETARG_ARRAYTYPE_P(1);
3462 : : float8 *transvalues1;
3463 : : float8 *transvalues2;
3464 : : float8 N1,
3465 : : Sx1,
3466 : : Sxx1,
3467 : : Sy1,
3468 : : Syy1,
3469 : : Sxy1,
3470 : : N2,
3471 : : Sx2,
3472 : : Sxx2,
3473 : : Sy2,
3474 : : Syy2,
3475 : : Sxy2,
3476 : : tmp1,
3477 : : tmp2,
3478 : : N,
3479 : : Sx,
3480 : : Sxx,
3481 : : Sy,
3482 : : Syy,
3483 : : Sxy;
3484 : :
3485 : 9 : transvalues1 = check_float8_array(transarray1, "float8_regr_combine", 6);
3486 : 9 : transvalues2 = check_float8_array(transarray2, "float8_regr_combine", 6);
3487 : :
2527 dean.a.rasheed@gmail 3488 : 9 : N1 = transvalues1[0];
3489 : 9 : Sx1 = transvalues1[1];
3490 : 9 : Sxx1 = transvalues1[2];
3491 : 9 : Sy1 = transvalues1[3];
3492 : 9 : Syy1 = transvalues1[4];
3493 : 9 : Sxy1 = transvalues1[5];
3494 : :
3495 : 9 : N2 = transvalues2[0];
3496 : 9 : Sx2 = transvalues2[1];
3497 : 9 : Sxx2 = transvalues2[2];
3498 : 9 : Sy2 = transvalues2[3];
3499 : 9 : Syy2 = transvalues2[4];
3500 : 9 : Sxy2 = transvalues2[5];
3501 : :
3502 : : /*--------------------
3503 : : * The transition values combine using a generalization of the
3504 : : * Youngs-Cramer algorithm as follows:
3505 : : *
3506 : : * N = N1 + N2
3507 : : * Sx = Sx1 + Sx2
3508 : : * Sxx = Sxx1 + Sxx2 + N1 * N2 * (Sx1/N1 - Sx2/N2)^2 / N
3509 : : * Sy = Sy1 + Sy2
3510 : : * Syy = Syy1 + Syy2 + N1 * N2 * (Sy1/N1 - Sy2/N2)^2 / N
3511 : : * Sxy = Sxy1 + Sxy2 + N1 * N2 * (Sx1/N1 - Sx2/N2) * (Sy1/N1 - Sy2/N2) / N
3512 : : *
3513 : : * It's worth handling the special cases N1 = 0 and N2 = 0 separately
3514 : : * since those cases are trivial, and we then don't need to worry about
3515 : : * division-by-zero errors in the general case.
3516 : : *--------------------
3517 : : */
3518 [ + + ]: 9 : if (N1 == 0.0)
3519 : : {
3520 : 3 : N = N2;
3521 : 3 : Sx = Sx2;
3522 : 3 : Sxx = Sxx2;
3523 : 3 : Sy = Sy2;
3524 : 3 : Syy = Syy2;
3525 : 3 : Sxy = Sxy2;
3526 : : }
3527 [ + + ]: 6 : else if (N2 == 0.0)
3528 : : {
3529 : 3 : N = N1;
3530 : 3 : Sx = Sx1;
3531 : 3 : Sxx = Sxx1;
3532 : 3 : Sy = Sy1;
3533 : 3 : Syy = Syy1;
3534 : 3 : Sxy = Sxy1;
3535 : : }
3536 : : else
3537 : : {
3538 : 3 : N = N1 + N2;
3539 : 3 : Sx = float8_pl(Sx1, Sx2);
3540 : 3 : tmp1 = Sx1 / N1 - Sx2 / N2;
3541 : 3 : Sxx = Sxx1 + Sxx2 + N1 * N2 * tmp1 * tmp1 / N;
2032 tgl@sss.pgh.pa.us 3542 [ - + - - : 3 : if (unlikely(isinf(Sxx)) && !isinf(Sxx1) && !isinf(Sxx2))
- - ]
2032 tgl@sss.pgh.pa.us 3543 :UBC 0 : float_overflow_error();
2527 dean.a.rasheed@gmail 3544 :CBC 3 : Sy = float8_pl(Sy1, Sy2);
3545 : 3 : tmp2 = Sy1 / N1 - Sy2 / N2;
3546 : 3 : Syy = Syy1 + Syy2 + N1 * N2 * tmp2 * tmp2 / N;
2032 tgl@sss.pgh.pa.us 3547 [ - + - - : 3 : if (unlikely(isinf(Syy)) && !isinf(Syy1) && !isinf(Syy2))
- - ]
2032 tgl@sss.pgh.pa.us 3548 :UBC 0 : float_overflow_error();
2527 dean.a.rasheed@gmail 3549 :CBC 3 : Sxy = Sxy1 + Sxy2 + N1 * N2 * tmp1 * tmp2 / N;
2032 tgl@sss.pgh.pa.us 3550 [ - + - - : 3 : if (unlikely(isinf(Sxy)) && !isinf(Sxy1) && !isinf(Sxy2))
- - ]
2032 tgl@sss.pgh.pa.us 3551 :UBC 0 : float_overflow_error();
3552 : : }
3553 : :
3554 : : /*
3555 : : * If we're invoked as an aggregate, we can cheat and modify our first
3556 : : * parameter in-place to reduce palloc overhead. Otherwise we construct a
3557 : : * new array with the updated transition data and return it.
3558 : : */
2527 dean.a.rasheed@gmail 3559 [ - + ]:CBC 9 : if (AggCheckCallContext(fcinfo, NULL))
3560 : : {
2527 dean.a.rasheed@gmail 3561 :UBC 0 : transvalues1[0] = N;
3562 : 0 : transvalues1[1] = Sx;
3563 : 0 : transvalues1[2] = Sxx;
3564 : 0 : transvalues1[3] = Sy;
3565 : 0 : transvalues1[4] = Syy;
3566 : 0 : transvalues1[5] = Sxy;
3567 : :
3568 : 0 : PG_RETURN_ARRAYTYPE_P(transarray1);
3569 : : }
3570 : : else
3571 : : {
3572 : : Datum transdatums[6];
3573 : : ArrayType *result;
3574 : :
2527 dean.a.rasheed@gmail 3575 :CBC 9 : transdatums[0] = Float8GetDatumFast(N);
3576 : 9 : transdatums[1] = Float8GetDatumFast(Sx);
3577 : 9 : transdatums[2] = Float8GetDatumFast(Sxx);
3578 : 9 : transdatums[3] = Float8GetDatumFast(Sy);
3579 : 9 : transdatums[4] = Float8GetDatumFast(Syy);
3580 : 9 : transdatums[5] = Float8GetDatumFast(Sxy);
3581 : :
327 msawada@postgresql.o 3582 : 9 : result = construct_array_builtin(transdatums, 6, FLOAT8OID);
3583 : :
2527 dean.a.rasheed@gmail 3584 : 9 : PG_RETURN_ARRAYTYPE_P(result);
3585 : : }
3586 : : }
3587 : :
3588 : :
3589 : : Datum
6980 tgl@sss.pgh.pa.us 3590 : 15 : float8_regr_sxx(PG_FUNCTION_ARGS)
3591 : : {
3592 : 15 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3593 : : float8 *transvalues;
3594 : : float8 N,
3595 : : Sxx;
3596 : :
3597 : 15 : transvalues = check_float8_array(transarray, "float8_regr_sxx", 6);
3598 : 15 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3599 : 15 : Sxx = transvalues[2];
3600 : :
3601 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3602 [ - + ]: 15 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3603 :UBC 0 : PG_RETURN_NULL();
3604 : :
3605 : : /* Note that Sxx is guaranteed to be non-negative */
3606 : :
2527 dean.a.rasheed@gmail 3607 :CBC 15 : PG_RETURN_FLOAT8(Sxx);
3608 : : }
3609 : :
3610 : : Datum
6980 tgl@sss.pgh.pa.us 3611 : 15 : float8_regr_syy(PG_FUNCTION_ARGS)
3612 : : {
3613 : 15 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3614 : : float8 *transvalues;
3615 : : float8 N,
3616 : : Syy;
3617 : :
3618 : 15 : transvalues = check_float8_array(transarray, "float8_regr_syy", 6);
3619 : 15 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3620 : 15 : Syy = transvalues[4];
3621 : :
3622 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3623 [ - + ]: 15 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3624 :UBC 0 : PG_RETURN_NULL();
3625 : :
3626 : : /* Note that Syy is guaranteed to be non-negative */
3627 : :
2527 dean.a.rasheed@gmail 3628 :CBC 15 : PG_RETURN_FLOAT8(Syy);
3629 : : }
3630 : :
3631 : : Datum
6980 tgl@sss.pgh.pa.us 3632 : 15 : float8_regr_sxy(PG_FUNCTION_ARGS)
3633 : : {
3634 : 15 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3635 : : float8 *transvalues;
3636 : : float8 N,
3637 : : Sxy;
3638 : :
3639 : 15 : transvalues = check_float8_array(transarray, "float8_regr_sxy", 6);
3640 : 15 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3641 : 15 : Sxy = transvalues[5];
3642 : :
3643 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3644 [ - + ]: 15 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3645 :UBC 0 : PG_RETURN_NULL();
3646 : :
3647 : : /* A negative result is valid here */
3648 : :
2527 dean.a.rasheed@gmail 3649 :CBC 15 : PG_RETURN_FLOAT8(Sxy);
3650 : : }
3651 : :
3652 : : Datum
6980 tgl@sss.pgh.pa.us 3653 : 3 : float8_regr_avgx(PG_FUNCTION_ARGS)
3654 : : {
3655 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3656 : : float8 *transvalues;
3657 : : float8 N,
3658 : : Sx;
3659 : :
3660 : 3 : transvalues = check_float8_array(transarray, "float8_regr_avgx", 6);
3661 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3662 : 3 : Sx = transvalues[1];
3663 : :
3664 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3665 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3666 :UBC 0 : PG_RETURN_NULL();
3667 : :
2527 dean.a.rasheed@gmail 3668 :CBC 3 : PG_RETURN_FLOAT8(Sx / N);
3669 : : }
3670 : :
3671 : : Datum
6980 tgl@sss.pgh.pa.us 3672 : 3 : float8_regr_avgy(PG_FUNCTION_ARGS)
3673 : : {
3674 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3675 : : float8 *transvalues;
3676 : : float8 N,
3677 : : Sy;
3678 : :
3679 : 3 : transvalues = check_float8_array(transarray, "float8_regr_avgy", 6);
3680 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3681 : 3 : Sy = transvalues[3];
3682 : :
3683 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3684 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3685 :UBC 0 : PG_RETURN_NULL();
3686 : :
2527 dean.a.rasheed@gmail 3687 :CBC 3 : PG_RETURN_FLOAT8(Sy / N);
3688 : : }
3689 : :
3690 : : Datum
6980 tgl@sss.pgh.pa.us 3691 : 12 : float8_covar_pop(PG_FUNCTION_ARGS)
3692 : : {
3693 : 12 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3694 : : float8 *transvalues;
3695 : : float8 N,
3696 : : Sxy;
3697 : :
3698 : 12 : transvalues = check_float8_array(transarray, "float8_covar_pop", 6);
3699 : 12 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3700 : 12 : Sxy = transvalues[5];
3701 : :
3702 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3703 [ - + ]: 12 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3704 :UBC 0 : PG_RETURN_NULL();
3705 : :
2527 dean.a.rasheed@gmail 3706 :CBC 12 : PG_RETURN_FLOAT8(Sxy / N);
3707 : : }
3708 : :
3709 : : Datum
6980 tgl@sss.pgh.pa.us 3710 : 12 : float8_covar_samp(PG_FUNCTION_ARGS)
3711 : : {
3712 : 12 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3713 : : float8 *transvalues;
3714 : : float8 N,
3715 : : Sxy;
3716 : :
3717 : 12 : transvalues = check_float8_array(transarray, "float8_covar_samp", 6);
3718 : 12 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3719 : 12 : Sxy = transvalues[5];
3720 : :
3721 : : /* if N is <= 1 we should return NULL */
6980 tgl@sss.pgh.pa.us 3722 [ + + ]: 12 : if (N < 2.0)
3723 : 9 : PG_RETURN_NULL();
3724 : :
2527 dean.a.rasheed@gmail 3725 : 3 : PG_RETURN_FLOAT8(Sxy / (N - 1.0));
3726 : : }
3727 : :
3728 : : Datum
6980 tgl@sss.pgh.pa.us 3729 : 3 : float8_corr(PG_FUNCTION_ARGS)
3730 : : {
3731 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3732 : : float8 *transvalues;
3733 : : float8 N,
3734 : : Sxx,
3735 : : Syy,
3736 : : Sxy;
3737 : :
3738 : 3 : transvalues = check_float8_array(transarray, "float8_corr", 6);
3739 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3740 : 3 : Sxx = transvalues[2];
3741 : 3 : Syy = transvalues[4];
3742 : 3 : Sxy = transvalues[5];
3743 : :
3744 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3745 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3746 :UBC 0 : PG_RETURN_NULL();
3747 : :
3748 : : /* Note that Sxx and Syy are guaranteed to be non-negative */
3749 : :
3750 : : /* per spec, return NULL for horizontal and vertical lines */
2527 dean.a.rasheed@gmail 3751 [ + - - + ]:CBC 3 : if (Sxx == 0 || Syy == 0)
6980 tgl@sss.pgh.pa.us 3752 :UBC 0 : PG_RETURN_NULL();
3753 : :
2527 dean.a.rasheed@gmail 3754 :CBC 3 : PG_RETURN_FLOAT8(Sxy / sqrt(Sxx * Syy));
3755 : : }
3756 : :
3757 : : Datum
6980 tgl@sss.pgh.pa.us 3758 : 3 : float8_regr_r2(PG_FUNCTION_ARGS)
3759 : : {
3760 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3761 : : float8 *transvalues;
3762 : : float8 N,
3763 : : Sxx,
3764 : : Syy,
3765 : : Sxy;
3766 : :
3767 : 3 : transvalues = check_float8_array(transarray, "float8_regr_r2", 6);
3768 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3769 : 3 : Sxx = transvalues[2];
3770 : 3 : Syy = transvalues[4];
3771 : 3 : Sxy = transvalues[5];
3772 : :
3773 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3774 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3775 :UBC 0 : PG_RETURN_NULL();
3776 : :
3777 : : /* Note that Sxx and Syy are guaranteed to be non-negative */
3778 : :
3779 : : /* per spec, return NULL for a vertical line */
2527 dean.a.rasheed@gmail 3780 [ - + ]:CBC 3 : if (Sxx == 0)
6980 tgl@sss.pgh.pa.us 3781 :UBC 0 : PG_RETURN_NULL();
3782 : :
3783 : : /* per spec, return 1.0 for a horizontal line */
2527 dean.a.rasheed@gmail 3784 [ - + ]:CBC 3 : if (Syy == 0)
6980 tgl@sss.pgh.pa.us 3785 :UBC 0 : PG_RETURN_FLOAT8(1.0);
3786 : :
2527 dean.a.rasheed@gmail 3787 :CBC 3 : PG_RETURN_FLOAT8((Sxy * Sxy) / (Sxx * Syy));
3788 : : }
3789 : :
3790 : : Datum
6980 tgl@sss.pgh.pa.us 3791 : 3 : float8_regr_slope(PG_FUNCTION_ARGS)
3792 : : {
3793 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3794 : : float8 *transvalues;
3795 : : float8 N,
3796 : : Sxx,
3797 : : Sxy;
3798 : :
3799 : 3 : transvalues = check_float8_array(transarray, "float8_regr_slope", 6);
3800 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3801 : 3 : Sxx = transvalues[2];
3802 : 3 : Sxy = transvalues[5];
3803 : :
3804 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3805 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3806 :UBC 0 : PG_RETURN_NULL();
3807 : :
3808 : : /* Note that Sxx is guaranteed to be non-negative */
3809 : :
3810 : : /* per spec, return NULL for a vertical line */
2527 dean.a.rasheed@gmail 3811 [ - + ]:CBC 3 : if (Sxx == 0)
6980 tgl@sss.pgh.pa.us 3812 :UBC 0 : PG_RETURN_NULL();
3813 : :
2527 dean.a.rasheed@gmail 3814 :CBC 3 : PG_RETURN_FLOAT8(Sxy / Sxx);
3815 : : }
3816 : :
3817 : : Datum
6980 tgl@sss.pgh.pa.us 3818 : 3 : float8_regr_intercept(PG_FUNCTION_ARGS)
3819 : : {
3820 : 3 : ArrayType *transarray = PG_GETARG_ARRAYTYPE_P(0);
3821 : : float8 *transvalues;
3822 : : float8 N,
3823 : : Sx,
3824 : : Sxx,
3825 : : Sy,
3826 : : Sxy;
3827 : :
3828 : 3 : transvalues = check_float8_array(transarray, "float8_regr_intercept", 6);
3829 : 3 : N = transvalues[0];
2527 dean.a.rasheed@gmail 3830 : 3 : Sx = transvalues[1];
3831 : 3 : Sxx = transvalues[2];
3832 : 3 : Sy = transvalues[3];
3833 : 3 : Sxy = transvalues[5];
3834 : :
3835 : : /* if N is 0 we should return NULL */
6980 tgl@sss.pgh.pa.us 3836 [ - + ]: 3 : if (N < 1.0)
6980 tgl@sss.pgh.pa.us 3837 :UBC 0 : PG_RETURN_NULL();
3838 : :
3839 : : /* Note that Sxx is guaranteed to be non-negative */
3840 : :
3841 : : /* per spec, return NULL for a vertical line */
2527 dean.a.rasheed@gmail 3842 [ - + ]:CBC 3 : if (Sxx == 0)
6980 tgl@sss.pgh.pa.us 3843 :UBC 0 : PG_RETURN_NULL();
3844 : :
2527 dean.a.rasheed@gmail 3845 :CBC 3 : PG_RETURN_FLOAT8((Sy - Sx * Sxy / Sxx) / N);
3846 : : }
3847 : :
3848 : :
3849 : : /*
3850 : : * ====================================
3851 : : * MIXED-PRECISION ARITHMETIC OPERATORS
3852 : : * ====================================
3853 : : */
3854 : :
3855 : : /*
3856 : : * float48pl - returns arg1 + arg2
3857 : : * float48mi - returns arg1 - arg2
3858 : : * float48mul - returns arg1 * arg2
3859 : : * float48div - returns arg1 / arg2
3860 : : */
3861 : : Datum
9167 tgl@sss.pgh.pa.us 3862 : 12 : float48pl(PG_FUNCTION_ARGS)
3863 : : {
3864 : 12 : float4 arg1 = PG_GETARG_FLOAT4(0);
3865 : 12 : float8 arg2 = PG_GETARG_FLOAT8(1);
3866 : :
2596 tomas.vondra@postgre 3867 : 12 : PG_RETURN_FLOAT8(float8_pl((float8) arg1, arg2));
3868 : : }
3869 : :
3870 : : Datum
9167 tgl@sss.pgh.pa.us 3871 : 3 : float48mi(PG_FUNCTION_ARGS)
3872 : : {
3873 : 3 : float4 arg1 = PG_GETARG_FLOAT4(0);
3874 : 3 : float8 arg2 = PG_GETARG_FLOAT8(1);
3875 : :
2596 tomas.vondra@postgre 3876 : 3 : PG_RETURN_FLOAT8(float8_mi((float8) arg1, arg2));
3877 : : }
3878 : :
3879 : : Datum
9167 tgl@sss.pgh.pa.us 3880 : 4 : float48mul(PG_FUNCTION_ARGS)
3881 : : {
3882 : 4 : float4 arg1 = PG_GETARG_FLOAT4(0);
3883 : 4 : float8 arg2 = PG_GETARG_FLOAT8(1);
3884 : :
2596 tomas.vondra@postgre 3885 : 4 : PG_RETURN_FLOAT8(float8_mul((float8) arg1, arg2));
3886 : : }
3887 : :
3888 : : Datum
9167 tgl@sss.pgh.pa.us 3889 : 3 : float48div(PG_FUNCTION_ARGS)
3890 : : {
3891 : 3 : float4 arg1 = PG_GETARG_FLOAT4(0);
3892 : 3 : float8 arg2 = PG_GETARG_FLOAT8(1);
3893 : :
2596 tomas.vondra@postgre 3894 : 3 : PG_RETURN_FLOAT8(float8_div((float8) arg1, arg2));
3895 : : }
3896 : :
3897 : : /*
3898 : : * float84pl - returns arg1 + arg2
3899 : : * float84mi - returns arg1 - arg2
3900 : : * float84mul - returns arg1 * arg2
3901 : : * float84div - returns arg1 / arg2
3902 : : */
3903 : : Datum
9167 tgl@sss.pgh.pa.us 3904 : 6 : float84pl(PG_FUNCTION_ARGS)
3905 : : {
3906 : 6 : float8 arg1 = PG_GETARG_FLOAT8(0);
3907 : 6 : float4 arg2 = PG_GETARG_FLOAT4(1);
3908 : :
2596 tomas.vondra@postgre 3909 : 6 : PG_RETURN_FLOAT8(float8_pl(arg1, (float8) arg2));
3910 : : }
3911 : :
3912 : : Datum
9167 tgl@sss.pgh.pa.us 3913 :UBC 0 : float84mi(PG_FUNCTION_ARGS)
3914 : : {
3915 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
3916 : 0 : float4 arg2 = PG_GETARG_FLOAT4(1);
3917 : :
2596 tomas.vondra@postgre 3918 : 0 : PG_RETURN_FLOAT8(float8_mi(arg1, (float8) arg2));
3919 : : }
3920 : :
3921 : : Datum
9167 tgl@sss.pgh.pa.us 3922 : 0 : float84mul(PG_FUNCTION_ARGS)
3923 : : {
3924 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
3925 : 0 : float4 arg2 = PG_GETARG_FLOAT4(1);
3926 : :
2596 tomas.vondra@postgre 3927 : 0 : PG_RETURN_FLOAT8(float8_mul(arg1, (float8) arg2));
3928 : : }
3929 : :
3930 : : Datum
9167 tgl@sss.pgh.pa.us 3931 :CBC 3 : float84div(PG_FUNCTION_ARGS)
3932 : : {
3933 : 3 : float8 arg1 = PG_GETARG_FLOAT8(0);
3934 : 3 : float4 arg2 = PG_GETARG_FLOAT4(1);
3935 : :
2596 tomas.vondra@postgre 3936 : 3 : PG_RETURN_FLOAT8(float8_div(arg1, (float8) arg2));
3937 : : }
3938 : :
3939 : : /*
3940 : : * ====================
3941 : : * COMPARISON OPERATORS
3942 : : * ====================
3943 : : */
3944 : :
3945 : : /*
3946 : : * float48{eq,ne,lt,le,gt,ge} - float4/float8 comparison operations
3947 : : */
3948 : : Datum
9167 tgl@sss.pgh.pa.us 3949 : 1491 : float48eq(PG_FUNCTION_ARGS)
3950 : : {
3951 : 1491 : float4 arg1 = PG_GETARG_FLOAT4(0);
3952 : 1491 : float8 arg2 = PG_GETARG_FLOAT8(1);
3953 : :
2596 tomas.vondra@postgre 3954 : 1491 : PG_RETURN_BOOL(float8_eq((float8) arg1, arg2));
3955 : : }
3956 : :
3957 : : Datum
9167 tgl@sss.pgh.pa.us 3958 : 10047 : float48ne(PG_FUNCTION_ARGS)
3959 : : {
3960 : 10047 : float4 arg1 = PG_GETARG_FLOAT4(0);
3961 : 10047 : float8 arg2 = PG_GETARG_FLOAT8(1);
3962 : :
2596 tomas.vondra@postgre 3963 : 10047 : PG_RETURN_BOOL(float8_ne((float8) arg1, arg2));
3964 : : }
3965 : :
3966 : : Datum
9167 tgl@sss.pgh.pa.us 3967 : 2134 : float48lt(PG_FUNCTION_ARGS)
3968 : : {
3969 : 2134 : float4 arg1 = PG_GETARG_FLOAT4(0);
3970 : 2134 : float8 arg2 = PG_GETARG_FLOAT8(1);
3971 : :
2596 tomas.vondra@postgre 3972 : 2134 : PG_RETURN_BOOL(float8_lt((float8) arg1, arg2));
3973 : : }
3974 : :
3975 : : Datum
9167 tgl@sss.pgh.pa.us 3976 : 13183 : float48le(PG_FUNCTION_ARGS)
3977 : : {
3978 : 13183 : float4 arg1 = PG_GETARG_FLOAT4(0);
3979 : 13183 : float8 arg2 = PG_GETARG_FLOAT8(1);
3980 : :
2596 tomas.vondra@postgre 3981 : 13183 : PG_RETURN_BOOL(float8_le((float8) arg1, arg2));
3982 : : }
3983 : :
3984 : : Datum
9167 tgl@sss.pgh.pa.us 3985 : 2246 : float48gt(PG_FUNCTION_ARGS)
3986 : : {
3987 : 2246 : float4 arg1 = PG_GETARG_FLOAT4(0);
3988 : 2246 : float8 arg2 = PG_GETARG_FLOAT8(1);
3989 : :
2596 tomas.vondra@postgre 3990 : 2246 : PG_RETURN_BOOL(float8_gt((float8) arg1, arg2));
3991 : : }
3992 : :
3993 : : Datum
9167 tgl@sss.pgh.pa.us 3994 : 2446 : float48ge(PG_FUNCTION_ARGS)
3995 : : {
3996 : 2446 : float4 arg1 = PG_GETARG_FLOAT4(0);
3997 : 2446 : float8 arg2 = PG_GETARG_FLOAT8(1);
3998 : :
2596 tomas.vondra@postgre 3999 : 2446 : PG_RETURN_BOOL(float8_ge((float8) arg1, arg2));
4000 : : }
4001 : :
4002 : : /*
4003 : : * float84{eq,ne,lt,le,gt,ge} - float8/float4 comparison operations
4004 : : */
4005 : : Datum
9167 tgl@sss.pgh.pa.us 4006 : 906 : float84eq(PG_FUNCTION_ARGS)
4007 : : {
4008 : 906 : float8 arg1 = PG_GETARG_FLOAT8(0);
4009 : 906 : float4 arg2 = PG_GETARG_FLOAT4(1);
4010 : :
2596 tomas.vondra@postgre 4011 : 906 : PG_RETURN_BOOL(float8_eq(arg1, (float8) arg2));
4012 : : }
4013 : :
4014 : : Datum
9167 tgl@sss.pgh.pa.us 4015 :UBC 0 : float84ne(PG_FUNCTION_ARGS)
4016 : : {
4017 : 0 : float8 arg1 = PG_GETARG_FLOAT8(0);
4018 : 0 : float4 arg2 = PG_GETARG_FLOAT4(1);
4019 : :
2596 tomas.vondra@postgre 4020 : 0 : PG_RETURN_BOOL(float8_ne(arg1, (float8) arg2));
4021 : : }
4022 : :
4023 : : Datum
9167 tgl@sss.pgh.pa.us 4024 :CBC 1599 : float84lt(PG_FUNCTION_ARGS)
4025 : : {
4026 : 1599 : float8 arg1 = PG_GETARG_FLOAT8(0);
4027 : 1599 : float4 arg2 = PG_GETARG_FLOAT4(1);
4028 : :
2596 tomas.vondra@postgre 4029 : 1599 : PG_RETURN_BOOL(float8_lt(arg1, (float8) arg2));
4030 : : }
4031 : :
4032 : : Datum
9167 tgl@sss.pgh.pa.us 4033 : 1899 : float84le(PG_FUNCTION_ARGS)
4034 : : {
4035 : 1899 : float8 arg1 = PG_GETARG_FLOAT8(0);
4036 : 1899 : float4 arg2 = PG_GETARG_FLOAT4(1);
4037 : :
2596 tomas.vondra@postgre 4038 : 1899 : PG_RETURN_BOOL(float8_le(arg1, (float8) arg2));
4039 : : }
4040 : :
4041 : : Datum
9167 tgl@sss.pgh.pa.us 4042 : 1599 : float84gt(PG_FUNCTION_ARGS)
4043 : : {
4044 : 1599 : float8 arg1 = PG_GETARG_FLOAT8(0);
4045 : 1599 : float4 arg2 = PG_GETARG_FLOAT4(1);
4046 : :
2596 tomas.vondra@postgre 4047 : 1599 : PG_RETURN_BOOL(float8_gt(arg1, (float8) arg2));
4048 : : }
4049 : :
4050 : : Datum
9167 tgl@sss.pgh.pa.us 4051 : 1602 : float84ge(PG_FUNCTION_ARGS)
4052 : : {
4053 : 1602 : float8 arg1 = PG_GETARG_FLOAT8(0);
4054 : 1602 : float4 arg2 = PG_GETARG_FLOAT4(1);
4055 : :
2596 tomas.vondra@postgre 4056 : 1602 : PG_RETURN_BOOL(float8_ge(arg1, (float8) arg2));
4057 : : }
4058 : :
4059 : : /*
4060 : : * Implements the float8 version of the width_bucket() function
4061 : : * defined by SQL2003. See also width_bucket_numeric().
4062 : : *
4063 : : * 'bound1' and 'bound2' are the lower and upper bounds of the
4064 : : * histogram's range, respectively. 'count' is the number of buckets
4065 : : * in the histogram. width_bucket() returns an integer indicating the
4066 : : * bucket number that 'operand' belongs to in an equiwidth histogram
4067 : : * with the specified characteristics. An operand smaller than the
4068 : : * lower bound is assigned to bucket 0. An operand greater than or equal
4069 : : * to the upper bound is assigned to an additional bucket (with number
4070 : : * count+1). We don't allow the histogram bounds to be NaN or +/- infinity,
4071 : : * but we do allow those values for the operand (taking NaN to be larger
4072 : : * than any other value, as we do in comparisons).
4073 : : */
4074 : : Datum
6808 neilc@samurai.com 4075 : 435 : width_bucket_float8(PG_FUNCTION_ARGS)
4076 : : {
6505 bruce@momjian.us 4077 : 435 : float8 operand = PG_GETARG_FLOAT8(0);
4078 : 435 : float8 bound1 = PG_GETARG_FLOAT8(1);
4079 : 435 : float8 bound2 = PG_GETARG_FLOAT8(2);
4080 : 435 : int32 count = PG_GETARG_INT32(3);
4081 : : int32 result;
4082 : :
891 tgl@sss.pgh.pa.us 4083 [ + + ]: 435 : if (count <= 0)
6808 neilc@samurai.com 4084 [ + - ]: 6 : ereport(ERROR,
4085 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION),
4086 : : errmsg("count must be greater than zero")));
4087 : :
66 tgl@sss.pgh.pa.us 4088 [ + + - + ]:GNC 429 : if (isnan(bound1) || isnan(bound2))
6808 neilc@samurai.com 4089 [ + - ]:CBC 3 : ereport(ERROR,
4090 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION),
4091 : : errmsg("lower and upper bounds cannot be NaN")));
4092 : :
6044 tgl@sss.pgh.pa.us 4093 [ + + + + ]: 426 : if (isinf(bound1) || isinf(bound2))
6808 neilc@samurai.com 4094 [ + - ]: 9 : ereport(ERROR,
4095 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION),
4096 : : errmsg("lower and upper bounds must be finite")));
4097 : :
4098 [ + + ]: 417 : if (bound1 < bound2)
4099 : : {
66 tgl@sss.pgh.pa.us 4100 [ + + + + ]:GNC 300 : if (isnan(operand) || operand >= bound2)
4101 : : {
890 tgl@sss.pgh.pa.us 4102 [ + + ]:CBC 63 : if (pg_add_s32_overflow(count, 1, &result))
4103 [ + - ]: 3 : ereport(ERROR,
4104 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
4105 : : errmsg("integer out of range")));
4106 : : }
66 tgl@sss.pgh.pa.us 4107 [ + + ]:GNC 237 : else if (operand < bound1)
4108 : 57 : result = 0;
4109 : : else
4110 : : {
890 tgl@sss.pgh.pa.us 4111 [ + + + + ]:CBC 180 : if (!isinf(bound2 - bound1))
4112 : : {
4113 : : /* The quotient is surely in [0,1], so this can't overflow */
4114 : 171 : result = count * ((operand - bound1) / (bound2 - bound1));
4115 : : }
4116 : : else
4117 : : {
4118 : : /*
4119 : : * We get here if bound2 - bound1 overflows DBL_MAX. Since
4120 : : * both bounds are finite, their difference can't exceed twice
4121 : : * DBL_MAX; so we can perform the computation without overflow
4122 : : * by dividing all the inputs by 2. That should be exact too,
4123 : : * except in the case where a very small operand underflows to
4124 : : * zero, which would have negligible impact on the result
4125 : : * given such large bounds.
4126 : : */
4127 : 9 : result = count * ((operand / 2 - bound1 / 2) / (bound2 / 2 - bound1 / 2));
4128 : : }
4129 : : /* The quotient could round to 1.0, which would be a lie */
4130 [ + + ]: 180 : if (result >= count)
4131 : 3 : result = count - 1;
4132 : : /* Having done that, we can add 1 without fear of overflow */
4133 : 180 : result++;
4134 : : }
4135 : : }
6808 neilc@samurai.com 4136 [ + + ]: 117 : else if (bound1 > bound2)
4137 : : {
66 tgl@sss.pgh.pa.us 4138 [ + - + + ]:GNC 114 : if (isnan(operand) || operand > bound1)
890 tgl@sss.pgh.pa.us 4139 :CBC 6 : result = 0;
6808 neilc@samurai.com 4140 [ + + ]: 108 : else if (operand <= bound2)
4141 : : {
890 tgl@sss.pgh.pa.us 4142 [ + + ]: 15 : if (pg_add_s32_overflow(count, 1, &result))
4143 [ + - ]: 3 : ereport(ERROR,
4144 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
4145 : : errmsg("integer out of range")));
4146 : : }
4147 : : else
4148 : : {
4149 [ + + + + ]: 93 : if (!isinf(bound1 - bound2))
4150 : 84 : result = count * ((bound1 - operand) / (bound1 - bound2));
4151 : : else
4152 : 9 : result = count * ((bound1 / 2 - operand / 2) / (bound1 / 2 - bound2 / 2));
4153 [ + + ]: 93 : if (result >= count)
4154 : 3 : result = count - 1;
4155 : 93 : result++;
4156 : : }
4157 : : }
4158 : : else
4159 : : {
6808 neilc@samurai.com 4160 [ + - ]: 3 : ereport(ERROR,
4161 : : (errcode(ERRCODE_INVALID_ARGUMENT_FOR_WIDTH_BUCKET_FUNCTION),
4162 : : errmsg("lower bound cannot equal upper bound")));
4163 : : result = 0; /* keep the compiler quiet */
4164 : : }
4165 : :
4166 : 408 : PG_RETURN_INT32(result);
4167 : : }
|