From 990fea847f2765822be74e30d502132aed364eca Mon Sep 17 00:00:00 2001 From: Bruce Momjian Date: Wed, 3 Jan 2007 14:35:24 +0000 Subject: [PATCH] Attempt to return proper overflow/underflow messages for platforms that only return Nan and set errno for pow/exp overflow/underflow. --- src/backend/utils/adt/float.c | 29 +++++++++++++++++++++++------ 1 file changed, 23 insertions(+), 6 deletions(-) diff --git a/src/backend/utils/adt/float.c b/src/backend/utils/adt/float.c index 2660bd3e31..d4e551b5ed 100644 --- a/src/backend/utils/adt/float.c +++ b/src/backend/utils/adt/float.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $PostgreSQL: pgsql/src/backend/utils/adt/float.c,v 1.136 2007/01/03 04:21:47 momjian Exp $ + * $PostgreSQL: pgsql/src/backend/utils/adt/float.c,v 1.137 2007/01/03 14:35:24 momjian Exp $ * *------------------------------------------------------------------------- */ @@ -1440,11 +1440,19 @@ dpow(PG_FUNCTION_ARGS) /* * pow() sets errno only on some platforms, depending on whether it - * follows _IEEE_, _POSIX_, _XOPEN_, or _SVID_, so, for consistency, - * we don't consult it and just do our check below. + * follows _IEEE_, _POSIX_, _XOPEN_, or _SVID_, and some return Nan, + * so we check and set result properly. */ + errno = 0; result = pow(arg1, arg2); - + if (errno == ERANGE && isnan(result)) + { + if ((fabs(arg1) > 1 && arg2 >= 0) || (fabs(arg1) < 1 && arg2 < 0)) + result = (arg1 >= 0) ? get_float8_infinity() : -get_float8_infinity(); + else + result = 0; + } + CHECKFLOATVAL(result, isinf(arg1) || isinf(arg2), arg1 == 0); PG_RETURN_FLOAT8(result); } @@ -1461,10 +1469,19 @@ dexp(PG_FUNCTION_ARGS) /* * exp() sets errno only on some platforms, depending on whether it - * follows _IEEE_, _POSIX_, _XOPEN_, or _SVID_, so, for consistency, - * we don't consult it and just do our check below. + * follows _IEEE_, _POSIX_, _XOPEN_, or _SVID_, and some return Nan, + * so we check and set result properly. */ + errno = 0; result = exp(arg1); + if (errno == ERANGE && isnan(result)) + { + if (arg1 >= 0) + result = get_float8_infinity(); + else + result = 0; + } + CHECKFLOATVAL(result, isinf(arg1), false); PG_RETURN_FLOAT8(result);