This set of patches adds -posix variants for the *z*printf-posix modules
that we have so far, along with small unit tests.


2024-06-24  Bruno Haible  <br...@clisp.org>

        obstack-zprintf-posix: Add tests.
        * tests/test-obstack-zprintf-posix.c: New file, based on
        tests/test-vazsprintf-posix.c.
        * modules/obstack-zprintf-posix-tests: New file.

        obstack-zprintf-posix: New module.
        * modules/obstack-zprintf-posix: New file.

2024-06-24  Bruno Haible  <br...@clisp.org>

        vazsprintf-posix: Add tests.
        * tests/test-vazsprintf-posix.c: New file, based on
        tests/test-zsnprintf-posix.h.
        * modules/vazsprintf-posix-tests: New file.

        vazsprintf-posix: New module.
        * modules/vazsprintf-posix: New file.

2024-06-24  Bruno Haible  <br...@clisp.org>

        zsprintf-posix: Add tests.
        * tests/test-zsprintf-posix.c: New file.
        * modules/zsprintf-posix-tests: New file.

        zsprintf-posix: New module.
        * modules/zsprintf-posix: New file.

2024-06-24  Bruno Haible  <br...@clisp.org>

        vzsprintf-posix: Add tests.
        * tests/test-vzsprintf-posix.c: New file.
        * tests/test-zsprintf-posix.h: New file, based on
        tests/test-zsnprintf-posix.h.
        * modules/vzsprintf-posix-tests: New file.

        vzsprintf-posix: New module.
        * modules/vzsprintf-posix: New file.

2024-06-24  Bruno Haible  <br...@clisp.org>

        zsnprintf-posix: Add tests.
        * tests/test-zsnprintf-posix.c: New file.
        * modules/zsnprintf-posix-tests: New file.

        zsnprintf-posix: New module.
        * modules/zsnprintf-posix: New file.

2024-06-24  Bruno Haible  <br...@clisp.org>

        vzsnprintf-posix: Add tests.
        * tests/test-vzsnprintf-posix.c: New file.
        * tests/test-zsnprintf-posix.h: New file, based on
        tests/test-snprintf-posix.h.
        * modules/vzsnprintf-posix-tests: New file.

        vzsnprintf-posix: New module.
        * modules/vzsnprintf-posix: New file.

>From 3b5067e98d78af7c971b4b6fd178092d76aaa611 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:21 +0200
Subject: [PATCH 01/12] vzsnprintf-posix: New module.

* modules/vzsnprintf-posix: New file.
---
 ChangeLog                |  5 +++++
 modules/vzsnprintf-posix | 22 ++++++++++++++++++++++
 2 files changed, 27 insertions(+)
 create mode 100644 modules/vzsnprintf-posix

diff --git a/ChangeLog b/ChangeLog
index 3162246429..6b7006d2da 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	vzsnprintf-posix: New module.
+	* modules/vzsnprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	Ease maintainability of *printf-gnu modules.
diff --git a/modules/vzsnprintf-posix b/modules/vzsnprintf-posix
new file mode 100644
index 0000000000..e1d30cf54f
--- /dev/null
+++ b/modules/vzsnprintf-posix
@@ -0,0 +1,22 @@
+Description:
+vzsnprintf() function with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+vzsnprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+LGPLv2+
+
+Maintainer:
+all
-- 
2.34.1

>From ccc053ead71ef63dbcef5d13d603ab0bc762fbc6 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:28 +0200
Subject: [PATCH 02/12] vzsnprintf-posix: Add tests.

* tests/test-vzsnprintf-posix.c: New file.
* tests/test-zsnprintf-posix.h: New file, based on
tests/test-snprintf-posix.h.
* modules/vzsnprintf-posix-tests: New file.
---
 ChangeLog                      |   6 ++
 modules/vzsnprintf-posix-tests |  14 +++
 tests/test-vzsnprintf-posix.c  |  48 +++++++++
 tests/test-zsnprintf-posix.h   | 183 +++++++++++++++++++++++++++++++++
 4 files changed, 251 insertions(+)
 create mode 100644 modules/vzsnprintf-posix-tests
 create mode 100644 tests/test-vzsnprintf-posix.c
 create mode 100644 tests/test-zsnprintf-posix.h

diff --git a/ChangeLog b/ChangeLog
index 6b7006d2da..84abf933b8 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,11 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	vzsnprintf-posix: Add tests.
+	* tests/test-vzsnprintf-posix.c: New file.
+	* tests/test-zsnprintf-posix.h: New file, based on
+	tests/test-snprintf-posix.h.
+	* modules/vzsnprintf-posix-tests: New file.
+
 	vzsnprintf-posix: New module.
 	* modules/vzsnprintf-posix: New file.
 
diff --git a/modules/vzsnprintf-posix-tests b/modules/vzsnprintf-posix-tests
new file mode 100644
index 0000000000..58df0a2436
--- /dev/null
+++ b/modules/vzsnprintf-posix-tests
@@ -0,0 +1,14 @@
+Files:
+tests/test-vzsnprintf-posix.c
+tests/test-zsnprintf-posix.h
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-vzsnprintf-posix
+check_PROGRAMS += test-vzsnprintf-posix
diff --git a/tests/test-vzsnprintf-posix.c b/tests/test-vzsnprintf-posix.c
new file mode 100644
index 0000000000..e44b94cafc
--- /dev/null
+++ b/tests/test-vzsnprintf-posix.c
@@ -0,0 +1,48 @@
+/* Test of POSIX compatible vzsnprintf() function.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2007.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stdarg.h>
+#include <stddef.h>
+#include <string.h>
+
+#include "macros.h"
+
+#include "test-zsnprintf-posix.h"
+
+static ptrdiff_t
+my_zsnprintf (char *str, size_t size, const char *format, ...)
+{
+  va_list args;
+  ptrdiff_t ret;
+
+  va_start (args, format);
+  ret = vzsnprintf (str, size, format, args);
+  va_end (args);
+  return ret;
+}
+
+int
+main (int argc, char *argv[])
+{
+  test_function (my_zsnprintf);
+  return test_exit_status;
+}
diff --git a/tests/test-zsnprintf-posix.h b/tests/test-zsnprintf-posix.h
new file mode 100644
index 0000000000..64639d4f17
--- /dev/null
+++ b/tests/test-zsnprintf-posix.h
@@ -0,0 +1,183 @@
+/* Test of POSIX compatible vzsnprintf() and zsnprintf() functions.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2024.  */
+
+/* This test exercises only a few POSIX compliance problems that are still
+   visible on platforms relevant in 2024.  For a much more complete test suite,
+   see test-snprintf-posix.h.  */
+
+#include "infinity.h"
+#include "nan.h"
+
+/* Test whether string[start_index..end_index-1] is a valid textual
+   representation of NaN.  */
+static int
+strisnan (const char *string, size_t start_index, size_t end_index, int uppercase)
+{
+  if (start_index < end_index)
+    {
+      if (string[start_index] == '-')
+        start_index++;
+      if (start_index + 3 <= end_index
+          && memcmp (string + start_index, uppercase ? "NAN" : "nan", 3) == 0)
+        {
+          start_index += 3;
+          if (start_index == end_index
+              || (string[start_index] == '(' && string[end_index - 1] == ')'))
+            return 1;
+        }
+    }
+  return 0;
+}
+
+static void
+test_function (ptrdiff_t (*my_zsnprintf) (char *, size_t, const char *, ...))
+{
+  char result[5000];
+
+  /* Test the support of the 'a' and 'A' conversion specifier for hexadecimal
+     output of floating-point numbers.  */
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%a %d", 3.1416015625, 33, 44, 55);
+    ASSERT (strcmp (result, "0x1.922p+1 33") == 0
+            || strcmp (result, "0x3.244p+0 33") == 0
+            || strcmp (result, "0x6.488p-1 33") == 0
+            || strcmp (result, "0xc.91p-2 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%A %d", -3.1416015625, 33, 44, 55);
+    ASSERT (strcmp (result, "-0X1.922P+1 33") == 0
+            || strcmp (result, "-0X3.244P+0 33") == 0
+            || strcmp (result, "-0X6.488P-1 33") == 0
+            || strcmp (result, "-0XC.91P-2 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on FreeBSD 6.1, NetBSD 10.0. */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%.2a %d", 1.51, 33, 44, 55);
+    ASSERT (strcmp (result, "0x1.83p+0 33") == 0
+            || strcmp (result, "0x3.05p-1 33") == 0
+            || strcmp (result, "0x6.0ap-2 33") == 0
+            || strcmp (result, "0xc.14p-3 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on macOS 14, FreeBSD 14.0, OpenBSD 7.5, AIX 7.3,
+       Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%.0a %d", 1.51, 33, 44, 55);
+    ASSERT (strcmp (result, "0x2p+0 33") == 0
+            || strcmp (result, "0x3p-1 33") == 0
+            || strcmp (result, "0x6p-2 33") == 0
+            || strcmp (result, "0xcp-3 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %f format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%f %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%f %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%010f %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "       inf 33") == 0
+            || strcmp (result, "  infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %e format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%e %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%e %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %g format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%g %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%g %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of large precision.  */
+
+  { /* This test would fail on AIX 7.1.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%.4000d %d", 1234567, 99);
+    size_t i;
+    for (i = 0; i < 4000 - 7; i++)
+      ASSERT (result[i] == '0');
+    ASSERT (strcmp (result + 4000 - 7, "1234567 99") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the 'b' conversion specifier for binary output of
+     integers.  */
+
+  { /* This test would fail on glibc 2.34, musl libc, macOS 14,
+       FreeBSD 13.2, NetBSD 10.0, OpenBSD 7.5, AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsnprintf (result, sizeof (result), "%b %d", 12345, 33, 44, 55);
+    ASSERT (strcmp (result, "11000000111001 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+}
-- 
2.34.1

>From 892d32e59c3603ab75116a83144dddab5d98cfb5 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:34 +0200
Subject: [PATCH 03/12] zsnprintf-posix: New module.

* modules/zsnprintf-posix: New file.
---
 ChangeLog               |  5 +++++
 modules/zsnprintf-posix | 22 ++++++++++++++++++++++
 2 files changed, 27 insertions(+)
 create mode 100644 modules/zsnprintf-posix

diff --git a/ChangeLog b/ChangeLog
index 84abf933b8..933190ff3b 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	zsnprintf-posix: New module.
+	* modules/zsnprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	vzsnprintf-posix: Add tests.
diff --git a/modules/zsnprintf-posix b/modules/zsnprintf-posix
new file mode 100644
index 0000000000..6e2f2c6589
--- /dev/null
+++ b/modules/zsnprintf-posix
@@ -0,0 +1,22 @@
+Description:
+zsnprintf() function with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+zsnprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+LGPLv2+
+
+Maintainer:
+all
-- 
2.34.1

>From c40e39a7f5f89ed376a711dfdf507a56220dcfd2 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:40 +0200
Subject: [PATCH 04/12] zsnprintf-posix: Add tests.

* tests/test-zsnprintf-posix.c: New file.
* modules/zsnprintf-posix-tests: New file.
---
 ChangeLog                     |  4 ++++
 modules/zsnprintf-posix-tests | 14 ++++++++++++++
 tests/test-zsnprintf-posix.c  | 35 +++++++++++++++++++++++++++++++++++
 3 files changed, 53 insertions(+)
 create mode 100644 modules/zsnprintf-posix-tests
 create mode 100644 tests/test-zsnprintf-posix.c

diff --git a/ChangeLog b/ChangeLog
index 933190ff3b..ffb2b7fecf 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,9 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	zsnprintf-posix: Add tests.
+	* tests/test-zsnprintf-posix.c: New file.
+	* modules/zsnprintf-posix-tests: New file.
+
 	zsnprintf-posix: New module.
 	* modules/zsnprintf-posix: New file.
 
diff --git a/modules/zsnprintf-posix-tests b/modules/zsnprintf-posix-tests
new file mode 100644
index 0000000000..75f7594293
--- /dev/null
+++ b/modules/zsnprintf-posix-tests
@@ -0,0 +1,14 @@
+Files:
+tests/test-zsnprintf-posix.c
+tests/test-zsnprintf-posix.h
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-zsnprintf-posix
+check_PROGRAMS += test-zsnprintf-posix
diff --git a/tests/test-zsnprintf-posix.c b/tests/test-zsnprintf-posix.c
new file mode 100644
index 0000000000..0acefb59dc
--- /dev/null
+++ b/tests/test-zsnprintf-posix.c
@@ -0,0 +1,35 @@
+/* Test of POSIX compatible zsnprintf() function.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2007.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stddef.h>
+#include <string.h>
+
+#include "macros.h"
+
+#include "test-zsnprintf-posix.h"
+
+int
+main (int argc, char *argv[])
+{
+  test_function (zsnprintf);
+  return test_exit_status;
+}
-- 
2.34.1

>From eae7f66a061c7e4e0e78e457d25caff296396cc4 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:45 +0200
Subject: [PATCH 05/12] vzsprintf-posix: New module.

* modules/vzsprintf-posix: New file.
---
 ChangeLog               |  5 +++++
 modules/vzsprintf-posix | 22 ++++++++++++++++++++++
 2 files changed, 27 insertions(+)
 create mode 100644 modules/vzsprintf-posix

diff --git a/ChangeLog b/ChangeLog
index ffb2b7fecf..aba08dc3e2 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	vzsprintf-posix: New module.
+	* modules/vzsprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	zsnprintf-posix: Add tests.
diff --git a/modules/vzsprintf-posix b/modules/vzsprintf-posix
new file mode 100644
index 0000000000..fb3428a8e1
--- /dev/null
+++ b/modules/vzsprintf-posix
@@ -0,0 +1,22 @@
+Description:
+vzsprintf() function with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+vzsprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+LGPL
+
+Maintainer:
+all
-- 
2.34.1

>From 20ac07997afaf79bcd26379c067f82c4c958988c Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:49 +0200
Subject: [PATCH 06/12] vzsprintf-posix: Add tests.

* tests/test-vzsprintf-posix.c: New file.
* tests/test-zsprintf-posix.h: New file, based on
tests/test-zsnprintf-posix.h.
* modules/vzsprintf-posix-tests: New file.
---
 ChangeLog                     |   6 ++
 modules/vzsprintf-posix-tests |  14 +++
 tests/test-vzsprintf-posix.c  |  48 +++++++++
 tests/test-zsprintf-posix.h   | 183 ++++++++++++++++++++++++++++++++++
 4 files changed, 251 insertions(+)
 create mode 100644 modules/vzsprintf-posix-tests
 create mode 100644 tests/test-vzsprintf-posix.c
 create mode 100644 tests/test-zsprintf-posix.h

diff --git a/ChangeLog b/ChangeLog
index aba08dc3e2..45a312df53 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,11 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	vzsprintf-posix: Add tests.
+	* tests/test-vzsprintf-posix.c: New file.
+	* tests/test-zsprintf-posix.h: New file, based on
+	tests/test-zsnprintf-posix.h.
+	* modules/vzsprintf-posix-tests: New file.
+
 	vzsprintf-posix: New module.
 	* modules/vzsprintf-posix: New file.
 
diff --git a/modules/vzsprintf-posix-tests b/modules/vzsprintf-posix-tests
new file mode 100644
index 0000000000..ea4682d504
--- /dev/null
+++ b/modules/vzsprintf-posix-tests
@@ -0,0 +1,14 @@
+Files:
+tests/test-vzsprintf-posix.c
+tests/test-zsprintf-posix.h
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-vzsprintf-posix
+check_PROGRAMS += test-vzsprintf-posix
diff --git a/tests/test-vzsprintf-posix.c b/tests/test-vzsprintf-posix.c
new file mode 100644
index 0000000000..a3dd260731
--- /dev/null
+++ b/tests/test-vzsprintf-posix.c
@@ -0,0 +1,48 @@
+/* Test of POSIX compatible vzsprintf() function.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2007.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stdarg.h>
+#include <stddef.h>
+#include <string.h>
+
+#include "macros.h"
+
+#include "test-zsprintf-posix.h"
+
+static ptrdiff_t
+my_zsprintf (char *str, const char *format, ...)
+{
+  va_list args;
+  ptrdiff_t ret;
+
+  va_start (args, format);
+  ret = vzsprintf (str, format, args);
+  va_end (args);
+  return ret;
+}
+
+int
+main (int argc, char *argv[])
+{
+  test_function (my_zsprintf);
+  return test_exit_status;
+}
diff --git a/tests/test-zsprintf-posix.h b/tests/test-zsprintf-posix.h
new file mode 100644
index 0000000000..c12754970d
--- /dev/null
+++ b/tests/test-zsprintf-posix.h
@@ -0,0 +1,183 @@
+/* Test of POSIX compatible vzsprintf() and zsprintf() functions.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2024.  */
+
+/* This test exercises only a few POSIX compliance problems that are still
+   visible on platforms relevant in 2024.  For a much more complete test suite,
+   see test-sprintf-posix.h.  */
+
+#include "infinity.h"
+#include "nan.h"
+
+/* Test whether string[start_index..end_index-1] is a valid textual
+   representation of NaN.  */
+static int
+strisnan (const char *string, size_t start_index, size_t end_index, int uppercase)
+{
+  if (start_index < end_index)
+    {
+      if (string[start_index] == '-')
+        start_index++;
+      if (start_index + 3 <= end_index
+          && memcmp (string + start_index, uppercase ? "NAN" : "nan", 3) == 0)
+        {
+          start_index += 3;
+          if (start_index == end_index
+              || (string[start_index] == '(' && string[end_index - 1] == ')'))
+            return 1;
+        }
+    }
+  return 0;
+}
+
+static void
+test_function (ptrdiff_t (*my_zsprintf) (char *, const char *, ...))
+{
+  char result[5000];
+
+  /* Test the support of the 'a' and 'A' conversion specifier for hexadecimal
+     output of floating-point numbers.  */
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%a %d", 3.1416015625, 33, 44, 55);
+    ASSERT (strcmp (result, "0x1.922p+1 33") == 0
+            || strcmp (result, "0x3.244p+0 33") == 0
+            || strcmp (result, "0x6.488p-1 33") == 0
+            || strcmp (result, "0xc.91p-2 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%A %d", -3.1416015625, 33, 44, 55);
+    ASSERT (strcmp (result, "-0X1.922P+1 33") == 0
+            || strcmp (result, "-0X3.244P+0 33") == 0
+            || strcmp (result, "-0X6.488P-1 33") == 0
+            || strcmp (result, "-0XC.91P-2 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on FreeBSD 6.1, NetBSD 10.0. */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%.2a %d", 1.51, 33, 44, 55);
+    ASSERT (strcmp (result, "0x1.83p+0 33") == 0
+            || strcmp (result, "0x3.05p-1 33") == 0
+            || strcmp (result, "0x6.0ap-2 33") == 0
+            || strcmp (result, "0xc.14p-3 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on macOS 14, FreeBSD 14.0, OpenBSD 7.5, AIX 7.3,
+       Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%.0a %d", 1.51, 33, 44, 55);
+    ASSERT (strcmp (result, "0x2p+0 33") == 0
+            || strcmp (result, "0x3p-1 33") == 0
+            || strcmp (result, "0x6p-2 33") == 0
+            || strcmp (result, "0xcp-3 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %f format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%f %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%f %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%010f %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "       inf 33") == 0
+            || strcmp (result, "  infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %e format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%e %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%e %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the %g format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%g %d", Infinityd (), 33, 44, 55);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%g %d", NaNd (), 33, 44, 55);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of large precision.  */
+
+  { /* This test would fail on AIX 7.1.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%.4000d %d", 1234567, 99);
+    size_t i;
+    for (i = 0; i < 4000 - 7; i++)
+      ASSERT (result[i] == '0');
+    ASSERT (strcmp (result + 4000 - 7, "1234567 99") == 0);
+    ASSERT (retval == strlen (result));
+  }
+
+  /* Test the support of the 'b' conversion specifier for binary output of
+     integers.  */
+
+  { /* This test would fail on glibc 2.34, musl libc, macOS 14,
+       FreeBSD 13.2, NetBSD 10.0, OpenBSD 7.5, AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t retval =
+      my_zsprintf (result, "%b %d", 12345, 33, 44, 55);
+    ASSERT (strcmp (result, "11000000111001 33") == 0);
+    ASSERT (retval == strlen (result));
+  }
+}
-- 
2.34.1

>From 60b27976e67e80a837033eb4546eef7c775b71a8 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:55 +0200
Subject: [PATCH 07/12] zsprintf-posix: New module.

* modules/zsprintf-posix: New file.
---
 ChangeLog              |  5 +++++
 modules/zsprintf-posix | 22 ++++++++++++++++++++++
 2 files changed, 27 insertions(+)
 create mode 100644 modules/zsprintf-posix

diff --git a/ChangeLog b/ChangeLog
index 45a312df53..820bcd3d22 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	zsprintf-posix: New module.
+	* modules/zsprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	vzsprintf-posix: Add tests.
diff --git a/modules/zsprintf-posix b/modules/zsprintf-posix
new file mode 100644
index 0000000000..c3ab891f88
--- /dev/null
+++ b/modules/zsprintf-posix
@@ -0,0 +1,22 @@
+Description:
+zsprintf() function with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+zsprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+LGPL
+
+Maintainer:
+all
-- 
2.34.1

>From 4ae432545a1606cf174d590737653d219c3d3dce Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:09:59 +0200
Subject: [PATCH 08/12] zsprintf-posix: Add tests.

* tests/test-zsprintf-posix.c: New file.
* modules/zsprintf-posix-tests: New file.
---
 ChangeLog                    |  4 ++++
 modules/zsprintf-posix-tests | 14 ++++++++++++++
 tests/test-zsprintf-posix.c  | 36 ++++++++++++++++++++++++++++++++++++
 3 files changed, 54 insertions(+)
 create mode 100644 modules/zsprintf-posix-tests
 create mode 100644 tests/test-zsprintf-posix.c

diff --git a/ChangeLog b/ChangeLog
index 820bcd3d22..3690156667 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,9 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	zsprintf-posix: Add tests.
+	* tests/test-zsprintf-posix.c: New file.
+	* modules/zsprintf-posix-tests: New file.
+
 	zsprintf-posix: New module.
 	* modules/zsprintf-posix: New file.
 
diff --git a/modules/zsprintf-posix-tests b/modules/zsprintf-posix-tests
new file mode 100644
index 0000000000..8ea6819c06
--- /dev/null
+++ b/modules/zsprintf-posix-tests
@@ -0,0 +1,14 @@
+Files:
+tests/test-zsprintf-posix.c
+tests/test-zsprintf-posix.h
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-zsprintf-posix
+check_PROGRAMS += test-zsprintf-posix
diff --git a/tests/test-zsprintf-posix.c b/tests/test-zsprintf-posix.c
new file mode 100644
index 0000000000..b5af845b15
--- /dev/null
+++ b/tests/test-zsprintf-posix.c
@@ -0,0 +1,36 @@
+/* Test of POSIX compatible zsprintf() function.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2007.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stdarg.h>
+#include <stddef.h>
+#include <string.h>
+
+#include "macros.h"
+
+#include "test-zsprintf-posix.h"
+
+int
+main (int argc, char *argv[])
+{
+  test_function (zsprintf);
+  return test_exit_status;
+}
-- 
2.34.1

>From 970f2cf6387bface37081bdd917d837eae8450da Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:10:03 +0200
Subject: [PATCH 09/12] vazsprintf-posix: New module.

* modules/vazsprintf-posix: New file.
---
 ChangeLog                |  5 +++++
 modules/vazsprintf-posix | 23 +++++++++++++++++++++++
 2 files changed, 28 insertions(+)
 create mode 100644 modules/vazsprintf-posix

diff --git a/ChangeLog b/ChangeLog
index 3690156667..6b4f07b427 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	vazsprintf-posix: New module.
+	* modules/vazsprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	zsprintf-posix: Add tests.
diff --git a/modules/vazsprintf-posix b/modules/vazsprintf-posix
new file mode 100644
index 0000000000..39a1da3800
--- /dev/null
+++ b/modules/vazsprintf-posix
@@ -0,0 +1,23 @@
+Description:
+azsprintf() and vazsprintf() functions
+with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+vazsprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+LGPLv2+
+
+Maintainer:
+all
-- 
2.34.1

>From 154a3d42c54b7e86e2d6daaa2877ae6ec5f2ee63 Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:10:08 +0200
Subject: [PATCH 10/12] vazsprintf-posix: Add tests.

* tests/test-vazsprintf-posix.c: New file, based on
tests/test-zsnprintf-posix.h.
* modules/vazsprintf-posix-tests: New file.
---
 ChangeLog                      |   5 +
 modules/vazsprintf-posix-tests |  13 ++
 tests/test-vazsprintf-posix.c  | 264 +++++++++++++++++++++++++++++++++
 3 files changed, 282 insertions(+)
 create mode 100644 modules/vazsprintf-posix-tests
 create mode 100644 tests/test-vazsprintf-posix.c

diff --git a/ChangeLog b/ChangeLog
index 6b4f07b427..b383122fed 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,10 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	vazsprintf-posix: Add tests.
+	* tests/test-vazsprintf-posix.c: New file, based on
+	tests/test-zsnprintf-posix.h.
+	* modules/vazsprintf-posix-tests: New file.
+
 	vazsprintf-posix: New module.
 	* modules/vazsprintf-posix: New file.
 
diff --git a/modules/vazsprintf-posix-tests b/modules/vazsprintf-posix-tests
new file mode 100644
index 0000000000..04c84e7861
--- /dev/null
+++ b/modules/vazsprintf-posix-tests
@@ -0,0 +1,13 @@
+Files:
+tests/test-vazsprintf-posix.c
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-vazsprintf-posix
+check_PROGRAMS += test-vazsprintf-posix
diff --git a/tests/test-vazsprintf-posix.c b/tests/test-vazsprintf-posix.c
new file mode 100644
index 0000000000..550596652c
--- /dev/null
+++ b/tests/test-vazsprintf-posix.c
@@ -0,0 +1,264 @@
+/* Test of POSIX compatible vazsprintf() and azsprintf() functions.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2024.  */
+
+/* This test exercises only a few POSIX compliance problems that are still
+   visible on platforms relevant in 2024.  For a much more complete test suite,
+   see test-vasprintf-posix.c.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stdarg.h>
+#include <stddef.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "macros.h"
+#include "infinity.h"
+#include "nan.h"
+
+/* Test whether string[start_index..end_index-1] is a valid textual
+   representation of NaN.  */
+static int
+strisnan (const char *string, size_t start_index, size_t end_index, int uppercase)
+{
+  if (start_index < end_index)
+    {
+      if (string[start_index] == '-')
+        start_index++;
+      if (start_index + 3 <= end_index
+          && memcmp (string + start_index, uppercase ? "NAN" : "nan", 3) == 0)
+        {
+          start_index += 3;
+          if (start_index == end_index
+              || (string[start_index] == '(' && string[end_index - 1] == ')'))
+            return 1;
+        }
+    }
+  return 0;
+}
+
+static void
+test_function (ptrdiff_t (*my_azsprintf) (char **, const char *, ...))
+{
+  char result[5000];
+
+  /* Test the support of the 'a' and 'A' conversion specifier for hexadecimal
+     output of floating-point numbers.  */
+
+  { /* This test would fail on Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%a %d", 3.1416015625, 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "0x1.922p+1 33") == 0
+            || strcmp (result, "0x3.244p+0 33") == 0
+            || strcmp (result, "0x6.488p-1 33") == 0
+            || strcmp (result, "0xc.91p-2 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%A %d", -3.1416015625, 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "-0X1.922P+1 33") == 0
+            || strcmp (result, "-0X3.244P+0 33") == 0
+            || strcmp (result, "-0X6.488P-1 33") == 0
+            || strcmp (result, "-0XC.91P-2 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on FreeBSD 6.1, NetBSD 10.0. */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%.2a %d", 1.51, 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "0x1.83p+0 33") == 0
+            || strcmp (result, "0x3.05p-1 33") == 0
+            || strcmp (result, "0x6.0ap-2 33") == 0
+            || strcmp (result, "0xc.14p-3 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on macOS 14, FreeBSD 14.0, OpenBSD 7.5, AIX 7.3,
+       Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%.0a %d", 1.51, 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "0x2p+0 33") == 0
+            || strcmp (result, "0x3p-1 33") == 0
+            || strcmp (result, "0x6p-2 33") == 0
+            || strcmp (result, "0xcp-3 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  /* Test the support of the %f format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%f %d", Infinityd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%f %d", NaNd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%010f %d", Infinityd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "       inf 33") == 0
+            || strcmp (result, "  infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  /* Test the support of the %e format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%e %d", Infinityd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%e %d", NaNd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  /* Test the support of the %g format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%g %d", Infinityd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "inf 33") == 0
+            || strcmp (result, "infinity 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%g %d", NaNd (), 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strlen (result) >= 3 + 3
+            && strisnan (result, 0, strlen (result) - 3, 0)
+            && strcmp (result + strlen (result) - 3, " 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  /* Test the support of large precision.  */
+
+  { /* This test would fail on AIX 7.1.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%.4000d %d", 1234567, 99);
+    ASSERT (result != NULL);
+    size_t i;
+    for (i = 0; i < 4000 - 7; i++)
+      ASSERT (result[i] == '0');
+    ASSERT (strcmp (result + 4000 - 7, "1234567 99") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+
+  /* Test the support of the 'b' conversion specifier for binary output of
+     integers.  */
+
+  { /* This test would fail on glibc 2.34, musl libc, macOS 14,
+       FreeBSD 13.2, NetBSD 10.0, OpenBSD 7.5, AIX 7.3, Solaris 11.4.  */
+    char *result;
+    ptrdiff_t retval =
+      my_azsprintf (&result, "%b %d", 12345, 33, 44, 55);
+    ASSERT (result != NULL);
+    ASSERT (strcmp (result, "11000000111001 33") == 0);
+    ASSERT (retval == strlen (result));
+    free (result);
+  }
+}
+
+static ptrdiff_t
+my_azsprintf (char **result, const char *format, ...)
+{
+  va_list args;
+  ptrdiff_t ret;
+
+  va_start (args, format);
+  ret = vazsprintf (result, format, args);
+  va_end (args);
+  return ret;
+}
+
+static void
+test_vazsprintf ()
+{
+  test_function (my_azsprintf);
+}
+
+static void
+test_azsprintf ()
+{
+  test_function (azsprintf);
+}
+
+int
+main (int argc, char *argv[])
+{
+  test_vazsprintf ();
+  test_azsprintf ();
+  return test_exit_status;
+}
-- 
2.34.1

>From 3a1e774a9931611444a23ab4267468f7f95186ee Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:10:30 +0200
Subject: [PATCH 11/12] obstack-zprintf-posix: New module.

* modules/obstack-zprintf-posix: New file.
---
 ChangeLog                     |  5 +++++
 modules/obstack-zprintf-posix | 23 +++++++++++++++++++++++
 2 files changed, 28 insertions(+)
 create mode 100644 modules/obstack-zprintf-posix

diff --git a/ChangeLog b/ChangeLog
index b383122fed..c95a133559 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,8 @@
+2024-06-24  Bruno Haible  <br...@clisp.org>
+
+	obstack-zprintf-posix: New module.
+	* modules/obstack-zprintf-posix: New file.
+
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
 	vazsprintf-posix: Add tests.
diff --git a/modules/obstack-zprintf-posix b/modules/obstack-zprintf-posix
new file mode 100644
index 0000000000..192b9b60fd
--- /dev/null
+++ b/modules/obstack-zprintf-posix
@@ -0,0 +1,23 @@
+Description:
+Formatted printing into an obstack (without INT_MAX limitation)
+with POSIX compliant format string interpretation
+
+Files:
+
+Depends-on:
+obstack-zprintf
+mixin/printf-posix
+
+configure.ac:
+gl_PREREQ_VASNPRINTF_WITH_POSIX_EXTRAS
+
+Makefile.am:
+
+Include:
+<stdio.h>
+
+License:
+GPL
+
+Maintainer:
+all
-- 
2.34.1

>From e967beab8b72086f4b9cea982e430b8f5cc8775b Mon Sep 17 00:00:00 2001
From: Bruno Haible <br...@clisp.org>
Date: Tue, 25 Jun 2024 01:10:34 +0200
Subject: [PATCH 12/12] obstack-zprintf-posix: Add tests.

* tests/test-obstack-zprintf-posix.c: New file, based on
tests/test-vazsprintf-posix.c.
* modules/obstack-zprintf-posix-tests: New file.
---
 ChangeLog                           |   5 +
 modules/obstack-zprintf-posix-tests |  14 ++
 tests/test-obstack-zprintf-posix.c  | 262 ++++++++++++++++++++++++++++
 3 files changed, 281 insertions(+)
 create mode 100644 modules/obstack-zprintf-posix-tests
 create mode 100644 tests/test-obstack-zprintf-posix.c

diff --git a/ChangeLog b/ChangeLog
index c95a133559..02fc3f35fa 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,10 @@
 2024-06-24  Bruno Haible  <br...@clisp.org>
 
+	obstack-zprintf-posix: Add tests.
+	* tests/test-obstack-zprintf-posix.c: New file, based on
+	tests/test-vazsprintf-posix.c.
+	* modules/obstack-zprintf-posix-tests: New file.
+
 	obstack-zprintf-posix: New module.
 	* modules/obstack-zprintf-posix: New file.
 
diff --git a/modules/obstack-zprintf-posix-tests b/modules/obstack-zprintf-posix-tests
new file mode 100644
index 0000000000..eeaf027340
--- /dev/null
+++ b/modules/obstack-zprintf-posix-tests
@@ -0,0 +1,14 @@
+Files:
+tests/test-obstack-zprintf-posix.c
+tests/infinity.h
+tests/macros.h
+
+Depends-on:
+nan
+
+configure.ac:
+
+Makefile.am:
+TESTS += test-obstack-zprintf-posix
+check_PROGRAMS += test-obstack-zprintf-posix
+test_obstack_zprintf_posix_LDADD = $(LDADD) @LIBINTL@
diff --git a/tests/test-obstack-zprintf-posix.c b/tests/test-obstack-zprintf-posix.c
new file mode 100644
index 0000000000..0002f79174
--- /dev/null
+++ b/tests/test-obstack-zprintf-posix.c
@@ -0,0 +1,262 @@
+/* Test of POSIX compatible obstack_[v]zprintf() functions.
+   Copyright (C) 2007-2024 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <br...@clisp.org>, 2024.  */
+
+/* This test exercises only a few POSIX compliance problems that are still
+   visible on platforms relevant in 2024.  */
+
+#include <config.h>
+
+#include <stdio.h>
+
+#include <stdarg.h>
+#include <stddef.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "obstack.h"
+
+#include "macros.h"
+#include "infinity.h"
+#include "nan.h"
+
+#define obstack_chunk_alloc malloc
+#define obstack_chunk_free free
+
+/* Test whether string[start_index..end_index-1] is a valid textual
+   representation of NaN.  */
+static int
+strisnan (const char *string, size_t start_index, size_t end_index, int uppercase)
+{
+  if (start_index < end_index)
+    {
+      if (string[start_index] == '-')
+        start_index++;
+      if (start_index + 3 <= end_index
+          && memcmp (string + start_index, uppercase ? "NAN" : "nan", 3) == 0)
+        {
+          start_index += 3;
+          if (start_index == end_index
+              || (string[start_index] == '(' && string[end_index - 1] == ')'))
+            return 1;
+        }
+    }
+  return 0;
+}
+
+static void
+test_function (ptrdiff_t (*my_obstack_zprintf) (struct obstack *, const char *, ...))
+{
+  struct obstack obs;
+  obstack_init (&obs);
+
+  #define RESULT_EQ(expected) \
+    (len == strlen (expected) && memcmp (result, expected, len) == 0)
+
+  /* Test the support of the 'a' and 'A' conversion specifier for hexadecimal
+     output of floating-point numbers.  */
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%a %d", 3.1416015625, 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("0x1.922p+1 33")
+            || RESULT_EQ ("0x3.244p+0 33")
+            || RESULT_EQ ("0x6.488p-1 33")
+            || RESULT_EQ ("0xc.91p-2 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%A %d", -3.1416015625, 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("-0X1.922P+1 33")
+            || RESULT_EQ ("-0X3.244P+0 33")
+            || RESULT_EQ ("-0X6.488P-1 33")
+            || RESULT_EQ ("-0XC.91P-2 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on FreeBSD 6.1, NetBSD 10.0. */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%.2a %d", 1.51, 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("0x1.83p+0 33")
+            || RESULT_EQ ("0x3.05p-1 33")
+            || RESULT_EQ ("0x6.0ap-2 33")
+            || RESULT_EQ ("0xc.14p-3 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on macOS 14, FreeBSD 14.0, OpenBSD 7.5, AIX 7.3,
+       Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%.0a %d", 1.51, 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("0x2p+0 33")
+            || RESULT_EQ ("0x3p-1 33")
+            || RESULT_EQ ("0x6p-2 33")
+            || RESULT_EQ ("0xcp-3 33"));
+    obstack_free (&obs, result);
+  }
+
+  /* Test the support of the %f format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%f %d", Infinityd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("inf 33")
+            || RESULT_EQ ("infinity 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%f %d", NaNd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (len >= 3 + 3
+            && strisnan (result, 0, len - 3, 0)
+            && memcmp (result + len - 3, " 33", 3) == 0);
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%010f %d", Infinityd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("       inf 33")
+            || RESULT_EQ ("  infinity 33"));
+    obstack_free (&obs, result);
+  }
+
+  /* Test the support of the %e format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%e %d", Infinityd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("inf 33")
+            || RESULT_EQ ("infinity 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%e %d", NaNd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (len >= 3 + 3
+            && strisnan (result, 0, len - 3, 0)
+            && memcmp (result + len - 3, " 33", 3) == 0);
+    obstack_free (&obs, result);
+  }
+
+  /* Test the support of the %g format directive.  */
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%g %d", Infinityd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("inf 33")
+            || RESULT_EQ ("infinity 33"));
+    obstack_free (&obs, result);
+  }
+
+  { /* This test would fail on AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%g %d", NaNd (), 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (len >= 3 + 3
+            && strisnan (result, 0, len - 3, 0)
+            && memcmp (result + len - 3, " 33", 3) == 0);
+    obstack_free (&obs, result);
+  }
+
+  /* Test the support of large precision.  */
+
+  { /* This test would fail on AIX 7.1.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%.4000d %d", 1234567, 99);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (len == 4000 + 3);
+    size_t i;
+    for (i = 0; i < 4000 - 7; i++)
+      ASSERT (result[i] == '0');
+    ASSERT (memcmp (result + 4000 - 7, "1234567 99", 7 + 3) == 0);
+    obstack_free (&obs, result);
+  }
+
+  /* Test the support of the 'b' conversion specifier for binary output of
+     integers.  */
+
+  { /* This test would fail on glibc 2.34, musl libc, macOS 14,
+       FreeBSD 13.2, NetBSD 10.0, OpenBSD 7.5, AIX 7.3, Solaris 11.4.  */
+    ptrdiff_t len =
+      my_obstack_zprintf (&obs, "%b %d", 12345, 33, 44, 55);
+    ASSERT (len >= 0);
+    char *result = obstack_finish (&obs);
+    ASSERT (RESULT_EQ ("11000000111001 33"));
+    obstack_free (&obs, result);
+  }
+
+  obstack_free (&obs, NULL);
+}
+
+static ptrdiff_t
+my_obstack_zprintf (struct obstack *obs, const char *format, ...)
+{
+  va_list args;
+  ptrdiff_t ret;
+
+  va_start (args, format);
+  ret = obstack_vzprintf (obs, format, args);
+  va_end (args);
+  return ret;
+}
+
+static void
+test_obstack_vzprintf ()
+{
+  test_function (my_obstack_zprintf);
+}
+
+static void
+test_obstack_zprintf ()
+{
+  test_function (obstack_zprintf);
+}
+
+int
+main (int argc, char *argv[])
+{
+  test_obstack_vzprintf ();
+  test_obstack_zprintf ();
+  return test_exit_status;
+}
-- 
2.34.1

Reply via email to