changeset 16680:fcdfd8bdad38

New module 'log2l'. * lib/math.in.h (log2l): New declaration. * lib/log2l.c: New file. * m4/log2l.m4: New file. * m4/math_h.m4 (gl_MATH_H): Test whether log2l is declared. (gl_MATH_H_DEFAULTS): Initialize GNULIB_LOG2L, HAVE_DECL_LOG2L, REPLACE_LOG2L. * modules/math (Makefile.am): Substitute GNULIB_LOG2L, HAVE_DECL_LOG2L, REPLACE_LOG2L. * modules/log2l: New file. * tests/test-math-c++.cc: Check the declaration of log2l. * doc/posix-functions/log2l.texi: Mention the new module and the IRIX and OSF/1 problems.
author Bruno Haible <bruno@clisp.org>
date Sun, 11 Mar 2012 20:50:08 +0100
parents bb4c399d2025
children 14592d8d106b
files ChangeLog doc/posix-functions/log2l.texi lib/log2l.c lib/math.in.h m4/log2l.m4 m4/math_h.m4 modules/log2l modules/math tests/test-math-c++.cc
diffstat 9 files changed, 288 insertions(+), 6 deletions(-) [+]
line wrap: on
line diff
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,19 @@
+2012-03-11  Bruno Haible  <bruno@clisp.org>
+
+	New module 'log2l'.
+	* lib/math.in.h (log2l): New declaration.
+	* lib/log2l.c: New file.
+	* m4/log2l.m4: New file.
+	* m4/math_h.m4 (gl_MATH_H): Test whether log2l is declared.
+	(gl_MATH_H_DEFAULTS): Initialize GNULIB_LOG2L, HAVE_DECL_LOG2L,
+	REPLACE_LOG2L.
+	* modules/math (Makefile.am): Substitute GNULIB_LOG2L, HAVE_DECL_LOG2L,
+	REPLACE_LOG2L.
+	* modules/log2l: New file.
+	* tests/test-math-c++.cc: Check the declaration of log2l.
+	* doc/posix-functions/log2l.texi: Mention the new module and the IRIX
+	and OSF/1 problems.
+
 2012-03-11  Bruno Haible  <bruno@clisp.org>
 
 	Tests for module 'log2f'.
--- a/doc/posix-functions/log2l.texi
+++ b/doc/posix-functions/log2l.texi
@@ -4,15 +4,21 @@
 
 POSIX specification:@* @url{http://www.opengroup.org/onlinepubs/9699919799/functions/log2l.html}
 
-Gnulib module: ---
+Gnulib module: log2l
 
 Portability problems fixed by Gnulib:
 @itemize
+@item
+This function is missing on some platforms:
+FreeBSD 6.0, NetBSD 5.0, OpenBSD 3.8, Minix 3.1.8, AIX 5.1, HP-UX 11, older IRIX 6.5, Solaris 9, Cygwin, MSVC 9, Interix 3.5, BeOS.
+@item
+This function is not declared on some platforms:
+IRIX 6.5.
+@item
+This function returns a wrong value for a minus zero argument on some platforms:
+OSF/1 5.1.
 @end itemize
 
 Portability problems not fixed by Gnulib:
 @itemize
-@item
-This function is missing on some platforms:
-FreeBSD 6.0, NetBSD 5.0, OpenBSD 3.8, Minix 3.1.8, AIX 5.1, HP-UX 11, IRIX 6.5, Solaris 9, Cygwin, MSVC 9, Interix 3.5, BeOS.
 @end itemize
new file mode 100644
--- /dev/null
+++ b/lib/log2l.c
@@ -0,0 +1,85 @@
+/* Base 2 logarithm.
+   Copyright (C) 2011-2012 Free Software Foundation, Inc.
+
+   This program is free software: you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 3 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
+
+#include <config.h>
+
+/* Specification.  */
+#include <math.h>
+
+#if HAVE_SAME_LONG_DOUBLE_AS_DOUBLE
+
+long double
+log2l (long double x)
+{
+  return log2 (x);
+}
+
+#else
+
+/* Best possible approximation of log(2) as a 'long double'.  */
+#define LOG2 0.693147180559945309417232121458176568075L
+
+/* Best possible approximation of 1/log(2) as a 'long double'.  */
+#define LOG2_INVERSE 1.44269504088896340735992468100189213743L
+
+/* sqrt(0.5).  */
+#define SQRT_HALF 0.707106781186547524400844362104849039284L
+
+long double
+log2l (long double x)
+{
+  if (isnanl (x))
+    return x;
+
+  if (x <= 0.0L)
+    {
+      if (x == 0.0L)
+        /* Return -Infinity.  */
+        return - HUGE_VALL;
+      else
+        {
+          /* Return NaN.  */
+#if defined _MSC_VER || (defined __sgi && !defined __GNUC__)
+          static long double zero;
+          return zero / zero;
+#else
+          return 0.0L / 0.0L;
+#endif
+        }
+    }
+
+  /* Decompose x into
+       x = 2^e * y
+     where
+       e is an integer,
+       1/2 < y < 2.
+     Then log2(x) = e + log2(y) = e + log(y)/log(2).  */
+  {
+    int e;
+    long double y;
+
+    y = frexpl (x, &e);
+    if (y < SQRT_HALF)
+      {
+        y = 2.0L * y;
+        e = e - 1;
+      }
+
+    return (long double) e + logl (y) * LOG2_INVERSE;
+  }
+}
+
+#endif
--- a/lib/math.in.h
+++ b/lib/math.in.h
@@ -1339,6 +1339,29 @@
 # endif
 #endif
 
+#if @GNULIB_LOG2L@
+# if @REPLACE_LOG2L@
+#  if !(defined __cplusplus && defined GNULIB_NAMESPACE)
+#   undef log2l
+#   define log2l rpl_log2l
+#  endif
+_GL_FUNCDECL_RPL (log2l, long double, (long double x));
+_GL_CXXALIAS_RPL (log2l, long double, (long double x));
+# else
+#  if !@HAVE_DECL_LOG2L@
+_GL_FUNCDECL_SYS (log2l, long double, (long double x));
+#  endif
+_GL_CXXALIAS_SYS (log2l, long double, (long double x));
+# endif
+_GL_CXXALIASWARN (log2l);
+#elif defined GNULIB_POSIXCHECK
+# undef log2l
+# if HAVE_RAW_DECL_LOG2L
+_GL_WARN_ON_USE (log2l, "log2l is unportable - "
+                 "use gnulib module log2l for portability");
+# endif
+#endif
+
 
 #if @GNULIB_MODFF@
 # if @REPLACE_MODFF@
new file mode 100644
--- /dev/null
+++ b/m4/log2l.m4
@@ -0,0 +1,108 @@
+# log2l.m4 serial 1
+dnl Copyright (C) 2010-2012 Free Software Foundation, Inc.
+dnl This file is free software; the Free Software Foundation
+dnl gives unlimited permission to copy and/or distribute it,
+dnl with or without modifications, as long as this notice is preserved.
+
+AC_DEFUN([gl_FUNC_LOG2L],
+[
+  AC_REQUIRE([gl_MATH_H_DEFAULTS])
+  AC_REQUIRE([gl_LONG_DOUBLE_VS_DOUBLE])
+  AC_REQUIRE([gl_FUNC_LOG2])
+
+  dnl Persuade glibc <math.h> to declare log2l().
+  AC_REQUIRE([gl_USE_SYSTEM_EXTENSIONS])
+
+  dnl Test whether log2l() exists. Assume that log2l(), if it exists, is
+  dnl defined in the same library as log2().
+  save_LIBS="$LIBS"
+  LIBS="$LIBS $LOG2_LIBM"
+  AC_CHECK_FUNCS([log2l])
+  LIBS="$save_LIBS"
+  if test $ac_cv_func_log2l = yes; then
+    LOG2L_LIBM="$LOG2_LIBM"
+    HAVE_LOG2L=1
+    dnl Also check whether it's declared.
+    dnl IRIX 6.5 has log2l() in libm but doesn't declare it in <math.h>.
+    AC_CHECK_DECL([log2l], , [HAVE_DECL_LOG2L=0], [[#include <math.h>]])
+
+    save_LIBS="$LIBS"
+    LIBS="$LIBS $LOG2L_LIBM"
+    gl_FUNC_LOG2L_WORKS
+    LIBS="$save_LIBS"
+    case "$gl_cv_func_log2l_works" in
+      *yes) ;;
+      *) REPLACE_LOG2L=1 ;;
+    esac
+  else
+    HAVE_LOG2L=0
+    HAVE_DECL_LOG2L=0
+  fi
+  if test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; then
+    dnl Find libraries needed to link lib/log2l.c.
+    if test $HAVE_SAME_LONG_DOUBLE_AS_DOUBLE = 1; then
+      LOG2L_LIBM="$LOG2_LIBM"
+    else
+      AC_REQUIRE([gl_FUNC_ISNANL])
+      AC_REQUIRE([gl_FUNC_FREXPL])
+      AC_REQUIRE([gl_FUNC_LOGL])
+      LOG2L_LIBM=
+      dnl Append $ISNANL_LIBM to LOG2L_LIBM, avoiding gratuitous duplicates.
+      case " $LOG2L_LIBM " in
+        *" $ISNANL_LIBM "*) ;;
+        *) LOG2L_LIBM="$LOG2L_LIBM $ISNANL_LIBM" ;;
+      esac
+      dnl Append $FREXPL_LIBM to LOG2L_LIBM, avoiding gratuitous duplicates.
+      case " $LOG2L_LIBM " in
+        *" $FREXPL_LIBM "*) ;;
+        *) LOG2L_LIBM="$LOG2L_LIBM $FREXPL_LIBM" ;;
+      esac
+      dnl Append $LOGL_LIBM to LOG2L_LIBM, avoiding gratuitous duplicates.
+      case " $LOG2L_LIBM " in
+        *" $LOGL_LIBM "*) ;;
+        *) LOG2L_LIBM="$LOG2L_LIBM $LOGL_LIBM" ;;
+      esac
+    fi
+  fi
+  AC_SUBST([LOG2L_LIBM])
+])
+
+dnl Test whether log2l() works.
+dnl On OSF/1 5.1, log2l(-0.0) is NaN.
+AC_DEFUN([gl_FUNC_LOG2L_WORKS],
+[
+  AC_REQUIRE([AC_PROG_CC])
+  AC_REQUIRE([AC_CANONICAL_HOST]) dnl for cross-compiles
+  AC_CACHE_CHECK([whether log2l works], [gl_cv_func_log2l_works],
+    [
+      AC_RUN_IFELSE(
+        [AC_LANG_SOURCE([[
+#include <math.h>
+#ifndef log2l /* for AIX */
+extern
+#ifdef __cplusplus
+"C"
+#endif
+long double log2l (long double);
+#endif
+volatile long double x;
+volatile long double y;
+int main ()
+{
+  /* This test fails on OSF/1 5.1.  */
+  x = -0.0L;
+  y = log2l (x);
+  if (!(y + y == y))
+    return 1;
+  return 0;
+}
+]])],
+        [gl_cv_func_log2l_works=yes],
+        [gl_cv_func_log2l_works=no],
+        [case "$host_os" in
+           osf*) gl_cv_func_log2l_works="guessing no";;
+           *)    gl_cv_func_log2l_works="guessing yes";;
+         esac
+        ])
+    ])
+])
--- a/m4/math_h.m4
+++ b/m4/math_h.m4
@@ -1,4 +1,4 @@
-# math_h.m4 serial 102
+# math_h.m4 serial 103
 dnl Copyright (C) 2007-2012 Free Software Foundation, Inc.
 dnl This file is free software; the Free Software Foundation
 dnl gives unlimited permission to copy and/or distribute it,
@@ -45,7 +45,7 @@
      fabsf fabsl floorf floorl fma fmaf fmal
      fmod fmodf fmodl frexpf frexpl hypotf hypotl
      ldexpf ldexpl
-     logb log logf logl log10f log10l log1p log1pf log1pl log2 log2f
+     logb log logf logl log10f log10l log1p log1pf log1pl log2 log2f log2l
      modf modff modfl powf
      remainder remainderf remainderl
      rint rintf rintl round roundf roundl sinf sinl sinhf sqrtf sqrtl
@@ -126,6 +126,7 @@
   GNULIB_LOG1PL=0;     AC_SUBST([GNULIB_LOG1PL])
   GNULIB_LOG2=0;       AC_SUBST([GNULIB_LOG2])
   GNULIB_LOG2F=0;      AC_SUBST([GNULIB_LOG2F])
+  GNULIB_LOG2L=0;      AC_SUBST([GNULIB_LOG2L])
   GNULIB_MODF=0;       AC_SUBST([GNULIB_MODF])
   GNULIB_MODFF=0;      AC_SUBST([GNULIB_MODFF])
   GNULIB_MODFL=0;      AC_SUBST([GNULIB_MODFL])
@@ -230,6 +231,7 @@
   HAVE_DECL_LOG10L=1;          AC_SUBST([HAVE_DECL_LOG10L])
   HAVE_DECL_LOG2=1;            AC_SUBST([HAVE_DECL_LOG2])
   HAVE_DECL_LOG2F=1;           AC_SUBST([HAVE_DECL_LOG2F])
+  HAVE_DECL_LOG2L=1;           AC_SUBST([HAVE_DECL_LOG2L])
   HAVE_DECL_REMAINDER=1;       AC_SUBST([HAVE_DECL_REMAINDER])
   HAVE_DECL_REMAINDERL=1;      AC_SUBST([HAVE_DECL_REMAINDERL])
   HAVE_DECL_RINTF=1;           AC_SUBST([HAVE_DECL_RINTF])
@@ -280,6 +282,7 @@
   REPLACE_LOG1PL=0;            AC_SUBST([REPLACE_LOG1PL])
   REPLACE_LOG2=0;              AC_SUBST([REPLACE_LOG2])
   REPLACE_LOG2F=0;             AC_SUBST([REPLACE_LOG2F])
+  REPLACE_LOG2L=0;             AC_SUBST([REPLACE_LOG2L])
   REPLACE_MODF=0;              AC_SUBST([REPLACE_MODF])
   REPLACE_MODFF=0;             AC_SUBST([REPLACE_MODFF])
   REPLACE_MODFL=0;             AC_SUBST([REPLACE_MODFL])
new file mode 100644
--- /dev/null
+++ b/modules/log2l
@@ -0,0 +1,35 @@
+Description:
+log2() function: base 2 logarithm.
+
+Files:
+lib/log2l.c
+m4/log2l.m4
+
+Depends-on:
+math
+extensions
+log2            [{ test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; } && test $HAVE_SAME_LONG_DOUBLE_AS_DOUBLE = 1]
+isnanl          [{ test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; } && test $HAVE_SAME_LONG_DOUBLE_AS_DOUBLE = 0]
+frexpl          [{ test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; } && test $HAVE_SAME_LONG_DOUBLE_AS_DOUBLE = 0]
+logl            [{ test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; } && test $HAVE_SAME_LONG_DOUBLE_AS_DOUBLE = 0]
+
+configure.ac:
+gl_FUNC_LOG2L
+if test $HAVE_LOG2L = 0 || test $REPLACE_LOG2L = 1; then
+  AC_LIBOBJ([log2l])
+fi
+gl_MATH_MODULE_INDICATOR([log2l])
+
+Makefile.am:
+
+Include:
+<math.h>
+
+Link:
+$(LOG2L_LIBM)
+
+License:
+LGPL
+
+Maintainer:
+Bruno Haible
--- a/modules/math
+++ b/modules/math
@@ -91,6 +91,7 @@
 	      -e 's/@''GNULIB_LOG1PL''@/$(GNULIB_LOG1PL)/g' \
 	      -e 's/@''GNULIB_LOG2''@/$(GNULIB_LOG2)/g' \
 	      -e 's/@''GNULIB_LOG2F''@/$(GNULIB_LOG2F)/g' \
+	      -e 's/@''GNULIB_LOG2L''@/$(GNULIB_LOG2L)/g' \
 	      -e 's/@''GNULIB_MODF''@/$(GNULIB_MODF)/g' \
 	      -e 's/@''GNULIB_MODFF''@/$(GNULIB_MODFF)/g' \
 	      -e 's/@''GNULIB_MODFL''@/$(GNULIB_MODFL)/g' \
@@ -195,6 +196,7 @@
 	      -e 's|@''HAVE_DECL_LOG10L''@|$(HAVE_DECL_LOG10L)|g' \
 	      -e 's|@''HAVE_DECL_LOG2''@|$(HAVE_DECL_LOG2)|g' \
 	      -e 's|@''HAVE_DECL_LOG2F''@|$(HAVE_DECL_LOG2F)|g' \
+	      -e 's|@''HAVE_DECL_LOG2L''@|$(HAVE_DECL_LOG2L)|g' \
 	      -e 's|@''HAVE_DECL_REMAINDER''@|$(HAVE_DECL_REMAINDER)|g' \
 	      -e 's|@''HAVE_DECL_REMAINDERL''@|$(HAVE_DECL_REMAINDERL)|g' \
 	      -e 's|@''HAVE_DECL_RINTF''@|$(HAVE_DECL_RINTF)|g' \
@@ -247,6 +249,7 @@
 	      -e 's|@''REPLACE_LOG1PL''@|$(REPLACE_LOG1PL)|g' \
 	      -e 's|@''REPLACE_LOG2''@|$(REPLACE_LOG2)|g' \
 	      -e 's|@''REPLACE_LOG2F''@|$(REPLACE_LOG2F)|g' \
+	      -e 's|@''REPLACE_LOG2L''@|$(REPLACE_LOG2L)|g' \
 	      -e 's|@''REPLACE_MODF''@|$(REPLACE_MODF)|g' \
 	      -e 's|@''REPLACE_MODFF''@|$(REPLACE_MODFF)|g' \
 	      -e 's|@''REPLACE_MODFL''@|$(REPLACE_MODFL)|g' \
--- a/tests/test-math-c++.cc
+++ b/tests/test-math-c++.cc
@@ -247,6 +247,9 @@
 #if GNULIB_TEST_LOG2
 SIGNATURE_CHECK (GNULIB_NAMESPACE::log2, double, (double));
 #endif
+#if GNULIB_TEST_LOG2L
+SIGNATURE_CHECK (GNULIB_NAMESPACE::log2l, long double, (long double));
+#endif
 
 #if GNULIB_TEST_MODFF
 SIGNATURE_CHECK (GNULIB_NAMESPACE::modff, float, (float, float *));