Paul Eggert wrote in <https://gcc.gnu.org/bugzilla/show_bug.cgi?id=109914#c6>:
> So my idea that "[[reproducible]] and __attribute__((pure)) are supposed to be
> the same thing" is incorrect. Similarly, [[unsequenced]] and
> __attribute__((const) are not the same thing. Oh well. We may need to change
> Gnulib because of these discrepancies.

Here's a patch that adds macros REPRODUCIBLE and UNSEQUENCED to the
'attribute' module, as well as _GL_ prefixed macros for Gnulib-internal
use.

I consulted ISO C 23 and the standardization document
<https://www.open-std.org/jtc1/sc22/wg14/www/docs/n2956.htm>.


2024-05-27  Bruno Haible  <br...@clisp.org>

        attribute: Support ISO C 23 [[reproducible]] and [[unsequenced]].
        * m4/gnulib-common.m4 (gl_COMMON_BODY): Define _GL_ATTR_reproducible,
        _GL_ATTR_unsequenced, _GL_ATTRIBUTE_REPRODUCIBLE,
        _GL_ATTRIBUTE_UNSEQUENCED.
        * lib/attribute.h (UNSEQUENCED, REPRODUCIBLE): New macros.

diff --git a/lib/attribute.h b/lib/attribute.h
index 710341ba41..a7076c8b7e 100644
--- a/lib/attribute.h
+++ b/lib/attribute.h
@@ -195,22 +195,50 @@
 /* Applies to: function.  */
 #define ATTRIBUTE_ALWAYS_INLINE _GL_ATTRIBUTE_ALWAYS_INLINE
 
-/* It is OK for a compiler to omit duplicate calls with the same arguments.
+/* It is OK for a compiler to omit duplicate calls to the function with the
+   same arguments.
    This attribute is safe for a function that neither depends on
    nor affects observable state, and always returns exactly once -
    e.g., does not loop forever, and does not call longjmp.
-   (This attribute is stricter than ATTRIBUTE_PURE.)  */
+   (This attribute is stricter than ATTRIBUTE_PURE.  It is equivalent to
+   UNSEQUENCED for a function that has no pointer or array parameters.)  */
 /* Applies to: functions.  */
 #define ATTRIBUTE_CONST _GL_ATTRIBUTE_CONST
 
-/* It is OK for a compiler to omit duplicate calls with the same
+/* It is OK for a compiler to move calls to the function and to omit duplicate
+   calls to the function with the same arguments.
+   This attribute is safe for a function that is effectless, idempotent,
+   stateless, and independent; see ISO C 23 § 6.7.12.7 for a definition of
+   these terms.
+   (This attribute is stricter than REPRODUCIBLE.  It is equivalent to
+   ATTRIBUTE_CONST for a function that has no pointer or array parameters.)
+   See also <https://www.open-std.org/jtc1/sc22/wg14/www/docs/n2956.htm> and
+   <https://stackoverflow.com/questions/76847905/>.  */
+/* Applies to: functions.  */
+#define UNSEQUENCED _GL_ATTRIBUTE_UNSEQUENCED
+
+/* It is OK for a compiler to omit duplicate calls to the function with the 
same
    arguments if observable state is not changed between calls.
-   This attribute is safe for a function that does not affect
-   observable state, and always returns exactly once.
-   (This attribute is looser than ATTRIBUTE_CONST.)  */
+   This attribute is safe for a function that does not affect observable state
+   and always returns exactly once.
+   (This attribute is looser than ATTRIBUTE_CONST.  It is equivalent to
+   REPRODUCIBLE for a function for which all pointer or array parameters have
+   'const'-qualified target types.)  */
 /* Applies to: functions.  */
 #define ATTRIBUTE_PURE _GL_ATTRIBUTE_PURE
 
+/* It is OK for a compiler to omit duplicate calls to the function with the 
same
+   arguments if observable state is not changed between calls.
+   This attribute is safe for a function that is effectless and idempotent; see
+   ISO C 23 § 6.7.12.7 for a definition of these terms.
+   (This attribute is looser than UNSEQUENCED.  It is equivalent to
+   ATTRIBUTE_PURE for a function for which all pointer or array parameters have
+   'const'-qualified target types.)
+   See also <https://www.open-std.org/jtc1/sc22/wg14/www/docs/n2956.htm> and
+   <https://stackoverflow.com/questions/76847905/>.  */
+/* Applies to: functions.  */
+#define REPRODUCIBLE _GL_ATTRIBUTE_REPRODUCIBLE
+
 /* The function is rarely executed.  */
 /* Applies to: functions.  */
 #define ATTRIBUTE_COLD _GL_ATTRIBUTE_COLD
diff --git a/m4/gnulib-common.m4 b/m4/gnulib-common.m4
index cb73044950..044ecc93f1 100644
--- a/m4/gnulib-common.m4
+++ b/m4/gnulib-common.m4
@@ -1,5 +1,5 @@
 # gnulib-common.m4
-# serial 93
+# serial 94
 dnl Copyright (C) 2007-2024 Free Software Foundation, Inc.
 dnl This file is free software; the Free Software Foundation
 dnl gives unlimited permission to copy and/or distribute it,
@@ -114,8 +114,10 @@ AC_DEFUN([gl_COMMON_BODY]
 #  define _GL_ATTR_nothrow _GL_GNUC_PREREQ (3, 3)
 #  define _GL_ATTR_packed _GL_GNUC_PREREQ (2, 7)
 #  define _GL_ATTR_pure _GL_GNUC_PREREQ (2, 96)
+#  define _GL_ATTR_reproducible 0 /* not yet supported, as of GCC 14 */
 #  define _GL_ATTR_returns_nonnull _GL_GNUC_PREREQ (4, 9)
 #  define _GL_ATTR_sentinel _GL_GNUC_PREREQ (4, 0)
+#  define _GL_ATTR_unsequenced 0 /* not yet supported, as of GCC 14 */
 #  define _GL_ATTR_unused _GL_GNUC_PREREQ (2, 7)
 #  define _GL_ATTR_warn_unused_result _GL_GNUC_PREREQ (3, 4)
 # endif
@@ -206,7 +208,9 @@ AC_DEFUN([gl_COMMON_BODY]
    This attribute is safe for a function that neither depends on nor affects
    observable state, and always returns exactly once - e.g., does not loop
    forever, and does not call longjmp.
-   (This attribute is stricter than _GL_ATTRIBUTE_PURE.)  */
+   (This attribute is stricter than _GL_ATTRIBUTE_PURE.  It is equivalent to
+   _GL_ATTRIBUTE_UNSEQUENCED for a function that has no pointer or array
+   parameters.)  */
 /* Applies to: functions.  */
 #ifndef _GL_ATTRIBUTE_CONST
 # if _GL_HAS_ATTRIBUTE (const)
@@ -505,9 +509,9 @@ AC_DEFUN([gl_COMMON_BODY]
    minimizing the memory required.  */
 /* Applies to: struct members, struct, union,
    in C++ also: class.  */
+#ifndef _GL_ATTRIBUTE_PACKED
 /* Oracle Studio 12.6 miscompiles code with __attribute__ ((__packed__)) 
despite
    __has_attribute OK.  */
-#ifndef _GL_ATTRIBUTE_PACKED
 # if _GL_HAS_ATTRIBUTE (packed) && !defined __SUNPRO_C
 #  define _GL_ATTRIBUTE_PACKED __attribute__ ((__packed__))
 # else
@@ -515,12 +519,14 @@ AC_DEFUN([gl_COMMON_BODY]
 # endif
 #endif
 
-/* _GL_ATTRIBUTE_PURE declares that It is OK for a compiler to omit duplicate
+/* _GL_ATTRIBUTE_PURE declares that it is OK for a compiler to omit duplicate
    calls to the function with the same arguments if observable state is not
    changed between calls.
-   This attribute is safe for a function that does not affect
-   observable state, and always returns exactly once.
-   (This attribute is looser than _GL_ATTRIBUTE_CONST.)  */
+   This attribute is safe for a function that does not affect observable state
+   and always returns exactly once.
+   (This attribute is looser than _GL_ATTRIBUTE_CONST.  It is equivalent to
+   _GL_ATTRIBUTE_REPRODUCIBLE for a function for which all pointer or array
+   parameters have 'const'-qualified target types.)  */
 /* Applies to: functions.  */
 #ifndef _GL_ATTRIBUTE_PURE
 # if _GL_HAS_ATTRIBUTE (pure)
@@ -530,6 +536,23 @@ AC_DEFUN([gl_COMMON_BODY]
 # endif
 #endif
 
+/* _GL_ATTRIBUTE_REPRODUCIBLE declares that it is OK for a compiler to omit
+   duplicate calls to the function with the same arguments if observable state
+   is not changed between calls.
+   This attribute is safe for a function that is effectless and idempotent; see
+   ISO C 23 § 6.7.12.7 for a definition of these terms.
+   (This attribute is looser than _GL_ATTRIBUTE_UNSEQUENCED.  It is equivalent
+   to _GL_ATTRIBUTE_PURE for a function for which all pointer or array
+   parameters have 'const'-qualified target types.)  */
+/* Applies to: functions.  */
+#ifndef _GL_ATTRIBUTE_REPRODUCIBLE
+# if _GL_HAS_ATTRIBUTE (reproducible)
+#  define _GL_ATTRIBUTE_REPRODUCIBLE [[reproducible]]
+# else
+#  define _GL_ATTRIBUTE_REPRODUCIBLE
+# endif
+#endif
+
 /* _GL_ATTRIBUTE_RETURNS_NONNULL declares that the function's return value is
    a non-NULL pointer.  */
 /* Applies to: functions.  */
@@ -554,6 +577,24 @@ AC_DEFUN([gl_COMMON_BODY]
 # endif
 #endif
 
+/* _GL_ATTRIBUTE_UNSEQUENCED declares that it is OK for a compiler to move
+   calls to the function and to omit duplicate calls to the function with the
+   same arguments.
+   This attribute is safe for a function that is effectless, idempotent,
+   stateless, and independent; see ISO C 23 § 6.7.12.7 for a definition of
+   these terms.
+   (This attribute is stricter than _GL_ATTRIBUTE_REPRODUCIBLE.  It is
+   equivalent to _GL_ATTRIBUTE_CONST for a function that has no pointer or
+   array parameters.)  */
+/* Applies to: functions.  */
+#ifndef _GL_ATTRIBUTE_UNSEQUENCED
+# if _GL_HAS_ATTRIBUTE (unsequenced)
+#  define _GL_ATTRIBUTE_UNSEQUENCED [[unsequenced]]
+# else
+#  define _GL_ATTRIBUTE_UNSEQUENCED
+# endif
+#endif
+
 /* A helper macro.  Don't use it directly.  */
 #ifndef _GL_ATTRIBUTE_UNUSED
 # if _GL_HAS_ATTRIBUTE (unused)




Reply via email to