On Sat, Oct 22, 2022 at 1:03 AM Joseph Myers <jos...@codesourcery.com> wrote:
>
> C2x allows function prototypes to be given as (...), a prototype
> meaning a variable-argument function with no named arguments.  To
> allow such functions to access their arguments, requirements for
> va_start calls are relaxed so it ignores all but its first argument
> (i.e. subsequent arguments, if any, can be arbitrary pp-token
> sequences).
>
> Implement this feature accordingly.  The va_start relaxation in
> <stdarg.h> is itself easy: __builtin_va_start already supports a
> second argument of 0 instead of a parameter name, and calls get
> converted internally to the form using 0 for that argument, so
> <stdarg.h> just needs changing to use a variadic macro that passes 0
> as the second argument of __builtin_va_start.  (This is done only in
> C2x mode, on the expectation that users of older standard would expect
> unsupported uses of va_start to be diagnosed.)
>
> For the (...) functions, it's necessary to distinguish these from
> unprototyped functions, whereas previously C++ (...) functions and
> unprototyped functions both used NULL TYPE_ARG_TYPES.  A flag is added
> to tree_type_common to mark the (...) functions; as discussed on gcc@,
> doing things this way is likely to be safer for unchanged code in GCC
> than adding a different form of representation in TYPE_ARG_TYPES, or
> adding a flag that instead signals that the function is unprototyped.
>
> There was previously an option
> -fallow-parameterless-variadic-functions to enable support for (...)
> prototypes.  The support was incomplete - it treated the functions as
> unprototyped, and only parsed some declarations, not e.g.
> "int g (int (...));".  This option is changed into a no-op ignored
> option; (...) is always accepted syntactically, with a pedwarn_c11
> call to given required diagnostics when appropriate.  The peculiarity
> of a parameter list with __attribute__ followed by '...' being
> accepted with that option is removed.
>
> Interfaces in tree.cc that create function types are adjusted to set
> this flag as appropriate.  It is of course possible that some existing
> users of the functions to create variable-argument functions actually
> wanted unprototyped functions in the no-named-argument case, rather
> than functions with a (...) prototype; some such cases in c-common.cc
> (for built-in functions and implicit function declarations) turn out
> to need updating for that reason.
>
> I didn't do anything to change how the C++ front end creates (...)
> function types.  It's very likely there are unchanged places in the
> compiler that in fact turn out to need changes to work properly with
> (...) function prototypes.
>
> Target setup_incoming_varargs hooks, where they used the information
> passed about the last named argument, needed updating to avoid using
> that information in the (...) case.  Note that apart from the x86
> changes, I haven't done any testing of those target changes beyond
> building cc1 to check for syntax errors.  It's possible further
> target-specific fixes will be needed; target maintainers should watch
> out for failures of c2x-stdarg-4.c, the execution test, which would
> indicate that this feature is not working correctly.
>
> Bootstrapped with no regressions for x86_64-pc-linux-gnu.  OK to commit?

You are missing to stream the new type flag in tree-streamer-{in,out}.cc
and checking for tree merging in lto-common.cc:compare_tree_sccs_1

Otherwise looks reasonable.  Can you add a (multi TU) runtime testcase to the
torture exercising the feature so we can see any ABI issues?

Thanks,
Richard.

> gcc/
>         * config/aarch64/aarch64.cc (aarch64_setup_incoming_varargs):
>         Check TYPE_NO_NAMED_ARGS_STDARG_P.
>         * config/alpha/alpha.cc (alpha_setup_incoming_varargs): Likewise.
>         * config/arc/arc.cc (arc_setup_incoming_varargs): Likewise.
>         * config/arm/arm.cc (arm_setup_incoming_varargs): Likewise.
>         * config/csky/csky.cc (csky_setup_incoming_varargs): Likewise.
>         * config/epiphany/epiphany.cc (epiphany_setup_incoming_varargs):
>         Likewise.
>         * config/fr30/fr30.cc (fr30_setup_incoming_varargs): Likewise.
>         * config/frv/frv.cc (frv_setup_incoming_varargs): Likewise.
>         * config/ft32/ft32.cc (ft32_setup_incoming_varargs): Likewise.
>         * config/i386/i386.cc (ix86_setup_incoming_varargs): Likewise.
>         * config/ia64/ia64.cc (ia64_setup_incoming_varargs): Likewise.
>         * config/loongarch/loongarch.cc
>         (loongarch_setup_incoming_varargs): Likewise.
>         * config/m32r/m32r.cc (m32r_setup_incoming_varargs): Likewise.
>         * config/mcore/mcore.cc (mcore_setup_incoming_varargs): Likewise.
>         * config/mips/mips.cc (mips_setup_incoming_varargs): Likewise.
>         * config/mmix/mmix.cc (mmix_setup_incoming_varargs): Likewise.
>         * config/nds32/nds32.cc (nds32_setup_incoming_varargs): Likewise.
>         * config/nios2/nios2.cc (nios2_setup_incoming_varargs): Likewise.
>         * config/riscv/riscv.cc (riscv_setup_incoming_varargs): Likewise.
>         * config/rs6000/rs6000-call.cc (setup_incoming_varargs): Likewise.
>         * config/sh/sh.cc (sh_setup_incoming_varargs): Likewise.
>         * config/visium/visium.cc (visium_setup_incoming_varargs):
>         Likewise.
>         * config/vms/vms-c.cc (vms_c_common_override_options): Do not set
>         flag_allow_parameterless_variadic_functions.
>         * doc/invoke.texi (-fallow-parameterless-variadic-functions): Do
>         not document option.
>         * function.cc (assign_parms): Call assign_parms_setup_varargs for
>         TYPE_NO_NAMED_ARGS_STDARG_P case.
>         * ginclude/stdarg.h [__STDC_VERSION__ > 201710L] (va_start): Make
>         variadic macro.  Pass second argument of 0 to __builtin_va_start.
>         * target.def (setup_incoming_varargs): Update documentation.
>         * doc/tm.texi: Regenerate.
>         * tree-core.h (struct tree_type_common): Add
>         no_named_args_stdarg_p.
>         * tree.cc (type_cache_hasher::equal): Compare
>         TYPE_NO_NAMED_ARGS_STDARG_P.
>         (build_function_type): Add argument no_named_args_stdarg_p.
>         (build_function_type_list_1, build_function_type_array_1)
>         (reconstruct_complex_type): Update calls to build_function_type.
>         (stdarg_p, prototype_p): Return true for (...) functions.
>         (gimple_canonical_types_compatible_p): Compare
>         TYPE_NO_NAMED_ARGS_STDARG_P.
>         * tree.h (TYPE_NO_NAMED_ARGS_STDARG_P): New.
>         (build_function_type): Update prototype.
>
> gcc/c-family/
>         * c-common.cc (def_fn_type): Call build_function_type for
>         zero-argument variable-argument function.
>         (c_common_nodes_and_builtins): Build default_function_type with
>         build_function_type.
>         * c.opt (fallow-parameterless-variadic-functions): Mark as ignored
>         option.
>
> gcc/c/
>         * c-decl.cc (grokdeclarator): Pass
>         arg_info->no_named_args_stdarg_p to build_function_type.
>         (grokparms): Check arg_info->no_named_args_stdarg_p before
>         converting () to (void).
>         (build_arg_info): Initialize no_named_args_stdarg_p.
>         (get_parm_info): Set no_named_args_stdarg_p.
>         (start_function): Pass TYPE_NO_NAMED_ARGS_STDARG_P to
>         build_function_type.
>         (store_parm_decls): Count (...) functions as prototyped.
>         * c-parser.cc (c_parser_direct_declarator): Allow '...' after open
>         parenthesis to start parameter list.
>         (c_parser_parms_list_declarator): Always allow '...' with no
>         arguments, call pedwarn_c11 and set no_named_args_stdarg_p.
>         * c-tree.h (struct c_arg_info): Add field no_named_args_stdarg_p.
>         * c-typeck.cc (composite_type): Handle
>         TYPE_NO_NAMED_ARGS_STDARG_P.
>         (function_types_compatible_p): Compare
>         TYPE_NO_NAMED_ARGS_STDARG_P.
>
>         gcc/objc/
>         * objc-next-runtime-abi-01.cc (build_next_objc_exception_stuff):
>         Use build_function_type to build type of objc_setjmp_decl.
>
> gcc/testsuite/
>         * gcc.dg/c11-stdarg-1.c, gcc.dg/c11-stdarg-2.c,
>         gcc.dg/c11-stdarg-3.c, gcc.dg/c2x-stdarg-1.c,
>         gcc.dg/c2x-stdarg-2.c, gcc.dg/c2x-stdarg-3.c,
>         gcc.dg/c2x-stdarg-4.c, gcc.dg/gnu2x-stdarg-1.c: New tests.
>         * gcc.dg/Wold-style-definition-2.c, gcc.dg/format/sentinel-1.c:
>         Update expected diagnostics.
>         * gcc.dg/c2x-nullptr-1.c (test5): Cast unused parameter to (void).
>         * gcc.dg/diagnostic-token-ranges.c: Use -pedantic.  Expect warning
>         in place of error.
>
> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
> index cd54c517b18..5890c18bdc3 100644
> --- a/gcc/c-family/c-common.cc
> +++ b/gcc/c-family/c-common.cc
> @@ -4064,7 +4064,8 @@ static tree builtin_types[(int) BT_LAST + 1];
>
>  /* A helper function for c_common_nodes_and_builtins.  Build function type
>     for DEF with return type RET and N arguments.  If VAR is true, then the
> -   function should be variadic after those N arguments.
> +   function should be variadic after those N arguments, or, if N is zero,
> +   unprototyped.
>
>     Takes special care not to ICE if any of the types involved are
>     error_mark_node, which indicates that said type is not in fact available
> @@ -4093,7 +4094,10 @@ def_fn_type (builtin_type def, builtin_type ret, bool 
> var, int n, ...)
>    if (t == error_mark_node)
>      goto egress;
>    if (var)
> -    t = build_varargs_function_type_array (t, n, args);
> +    if (n == 0)
> +      t = build_function_type (t, NULL_TREE);
> +    else
> +      t = build_varargs_function_type_array (t, n, args);
>    else
>      t = build_function_type_array (t, n, args);
>
> @@ -4661,8 +4665,7 @@ c_common_nodes_and_builtins (void)
>      uintptr_type_node =
>        TREE_TYPE (identifier_global_value (c_get_ident (UINTPTR_TYPE)));
>
> -  default_function_type
> -    = build_varargs_function_type_list (integer_type_node, NULL_TREE);
> +  default_function_type = build_function_type (integer_type_node, NULL_TREE);
>    unsigned_ptrdiff_type_node = c_common_unsigned_type (ptrdiff_type_node);
>
>    lang_hooks.decls.pushdecl
> diff --git a/gcc/c-family/c.opt b/gcc/c-family/c.opt
> index 01d480759ae..5f9f65f9d5b 100644
> --- a/gcc/c-family/c.opt
> +++ b/gcc/c-family/c.opt
> @@ -1508,8 +1508,8 @@ fall-virtual
>  C++ ObjC++ WarnRemoved
>
>  fallow-parameterless-variadic-functions
> -C ObjC Var(flag_allow_parameterless_variadic_functions)
> -Allow variadic functions without named parameter.
> +C ObjC Ignore
> +Does nothing. Preserved for backward compatibility.
>
>  falt-external-templates
>  C++ ObjC++ WarnRemoved
> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
> index 80f6e912187..1c29a29b50f 100644
> --- a/gcc/c/c-decl.cc
> +++ b/gcc/c/c-decl.cc
> @@ -7252,7 +7252,8 @@ grokdeclarator (const struct c_declarator *declarator,
>               }
>             type_quals = TYPE_UNQUALIFIED;
>
> -           type = build_function_type (type, arg_types);
> +           type = build_function_type (type, arg_types,
> +                                       arg_info->no_named_args_stdarg_p);
>             declarator = declarator->declarator;
>
>             /* Set the TYPE_CONTEXTs for each tagged type which is local to
> @@ -8017,7 +8018,8 @@ grokparms (struct c_arg_info *arg_info, bool 
> funcdef_flag)
>        /* In C2X, convert () to (void).  */
>        if (flag_isoc2x
>           && !arg_types
> -         && !arg_info->parms)
> +         && !arg_info->parms
> +         && !arg_info->no_named_args_stdarg_p)
>         arg_types = arg_info->types = void_list_node;
>
>        /* If there is a parameter of incomplete type in a definition,
> @@ -8087,6 +8089,7 @@ build_arg_info (void)
>    ret->others = NULL_TREE;
>    ret->pending_sizes = NULL;
>    ret->had_vla_unspec = 0;
> +  ret->no_named_args_stdarg_p = 0;
>    return ret;
>  }
>
> @@ -8278,6 +8281,7 @@ get_parm_info (bool ellipsis, tree expr)
>    arg_info->types = types;
>    arg_info->others = others;
>    arg_info->pending_sizes = expr;
> +  arg_info->no_named_args_stdarg_p = ellipsis && !types;
>    return arg_info;
>  }
>
> @@ -9815,7 +9819,8 @@ start_function (struct c_declspecs *declspecs, struct 
> c_declarator *declarator,
>        /* Make it return void instead.  */
>        TREE_TYPE (decl1)
>         = build_function_type (void_type_node,
> -                              TYPE_ARG_TYPES (TREE_TYPE (decl1)));
> +                              TYPE_ARG_TYPES (TREE_TYPE (decl1)),
> +                              TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE 
> (decl1)));
>      }
>
>    if (warn_about_return_type)
> @@ -10414,7 +10419,7 @@ store_parm_decls (void)
>       empty argument list was converted to (void) in grokparms; in
>       older C standard versions, it does not give the function a type
>       with a prototype for future calls.  */
> -  proto = arg_info->types != 0;
> +  proto = arg_info->types != 0 || arg_info->no_named_args_stdarg_p;
>
>    if (proto)
>      store_parm_decls_newstyle (fndecl, arg_info);
> diff --git a/gcc/c/c-parser.cc b/gcc/c/c-parser.cc
> index 602e0235f2d..31438464e4e 100644
> --- a/gcc/c/c-parser.cc
> +++ b/gcc/c/c-parser.cc
> @@ -4119,7 +4119,8 @@ c_parser_direct_declarator (c_parser *parser, bool 
> type_seen_p, c_dtr_syn kind,
>        if (kind != C_DTR_NORMAL
>           && (c_parser_next_token_starts_declspecs (parser)
>               || (!have_gnu_attrs
> -                 && c_parser_nth_token_starts_std_attributes (parser, 1))
> +                 && (c_parser_nth_token_starts_std_attributes (parser, 1)
> +                     || c_parser_next_token_is (parser, CPP_ELLIPSIS)))
>               || c_parser_next_token_is (parser, CPP_CLOSE_PAREN)))
>         {
>           struct c_arg_info *args
> @@ -4395,25 +4396,18 @@ c_parser_parms_list_declarator (c_parser *parser, 
> tree attrs, tree expr,
>        c_parser_consume_token (parser);
>        return ret;
>      }
> -  if (c_parser_next_token_is (parser, CPP_ELLIPSIS))
> +  if (c_parser_next_token_is (parser, CPP_ELLIPSIS) && !have_gnu_attrs)
>      {
>        struct c_arg_info *ret = build_arg_info ();
>
> -      if (flag_allow_parameterless_variadic_functions)
> -        {
> -          /* F (...) is allowed.  */
> -          ret->types = NULL_TREE;
> -        }
> -      else
> -        {
> -          /* Suppress -Wold-style-definition for this case.  */
> -          ret->types = error_mark_node;
> -          error_at (c_parser_peek_token (parser)->location,
> -                    "ISO C requires a named argument before %<...%>");
> -        }
> +      ret->types = NULL_TREE;
> +      pedwarn_c11 (c_parser_peek_token (parser)->location, OPT_Wpedantic,
> +                  "ISO C requires a named argument before %<...%> "
> +                  "before C2X");
>        c_parser_consume_token (parser);
>        if (c_parser_next_token_is (parser, CPP_CLOSE_PAREN))
>         {
> +         ret->no_named_args_stdarg_p = true;
>           c_parser_consume_token (parser);
>           return ret;
>         }
> diff --git a/gcc/c/c-tree.h b/gcc/c/c-tree.h
> index e7cdd2f11dc..1bdadfffa52 100644
> --- a/gcc/c/c-tree.h
> +++ b/gcc/c/c-tree.h
> @@ -457,6 +457,8 @@ struct c_arg_info {
>    tree pending_sizes;
>    /* True when these arguments had [*].  */
>    BOOL_BITFIELD had_vla_unspec : 1;
> +  /* True when the arguments are a (...) prototype.  */
> +  BOOL_BITFIELD no_named_args_stdarg_p : 1;
>  };
>
>  /* A declarator.  */
> diff --git a/gcc/c/c-typeck.cc b/gcc/c/c-typeck.cc
> index fdb96c28c51..868fb781c47 100644
> --- a/gcc/c/c-typeck.cc
> +++ b/gcc/c/c-typeck.cc
> @@ -542,17 +542,19 @@ composite_type (tree t1, tree t2)
>
>         /* Simple way if one arg fails to specify argument types.  */
>         if (TYPE_ARG_TYPES (t1) == NULL_TREE)
> -        {
> -           t1 = build_function_type (valtype, TYPE_ARG_TYPES (t2));
> +         {
> +           t1 = build_function_type (valtype, TYPE_ARG_TYPES (t2),
> +                                     TYPE_NO_NAMED_ARGS_STDARG_P (t2));
>             t1 = build_type_attribute_variant (t1, attributes);
>             return qualify_type (t1, t2);
>          }
>         if (TYPE_ARG_TYPES (t2) == NULL_TREE)
> -        {
> -          t1 = build_function_type (valtype, TYPE_ARG_TYPES (t1));
> -          t1 = build_type_attribute_variant (t1, attributes);
> -          return qualify_type (t1, t2);
> -        }
> +         {
> +           t1 = build_function_type (valtype, TYPE_ARG_TYPES (t1),
> +                                     TYPE_NO_NAMED_ARGS_STDARG_P (t1));
> +           t1 = build_type_attribute_variant (t1, attributes);
> +           return qualify_type (t1, t2);
> +         }
>
>         /* If both args specify argument types, we must merge the two
>            lists, argument by argument.  */
> @@ -1700,6 +1702,8 @@ function_types_compatible_p (const_tree f1, const_tree 
> f2,
>
>    if (args1 == NULL_TREE)
>      {
> +      if (TYPE_NO_NAMED_ARGS_STDARG_P (f1) != TYPE_NO_NAMED_ARGS_STDARG_P 
> (f2))
> +       return 0;
>        if (!self_promoting_args_p (args2))
>         return 0;
>        /* If one of these types comes from a non-prototype fn definition,
> @@ -1713,6 +1717,8 @@ function_types_compatible_p (const_tree f1, const_tree 
> f2,
>      }
>    if (args2 == NULL_TREE)
>      {
> +      if (TYPE_NO_NAMED_ARGS_STDARG_P (f1) != TYPE_NO_NAMED_ARGS_STDARG_P 
> (f2))
> +       return 0;
>        if (!self_promoting_args_p (args1))
>         return 0;
>        if (TYPE_ACTUAL_ARG_TYPES (f2)
> diff --git a/gcc/config/aarch64/aarch64.cc b/gcc/config/aarch64/aarch64.cc
> index 1d0f994f281..3e32d3f2b23 100644
> --- a/gcc/config/aarch64/aarch64.cc
> +++ b/gcc/config/aarch64/aarch64.cc
> @@ -19856,7 +19856,8 @@ aarch64_setup_incoming_varargs (cumulative_args_t 
> cum_v,
>       argument.  Advance a local copy of CUM past the last "real" named
>       argument, to find out how many registers are left over.  */
>    local_cum = *cum;
> -  aarch64_function_arg_advance (pack_cumulative_args(&local_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    aarch64_function_arg_advance (pack_cumulative_args(&local_cum), arg);
>
>    /* Found out how many registers we need to save.
>       Honor tree-stdvar analysis results.  */
> diff --git a/gcc/config/alpha/alpha.cc b/gcc/config/alpha/alpha.cc
> index 66c17149d4d..333f2c602c4 100644
> --- a/gcc/config/alpha/alpha.cc
> +++ b/gcc/config/alpha/alpha.cc
> @@ -6084,8 +6084,9 @@ alpha_setup_incoming_varargs (cumulative_args_t pcum,
>  {
>    CUMULATIVE_ARGS cum = *get_cumulative_args (pcum);
>
> -  /* Skip the current argument.  */
> -  targetm.calls.function_arg_advance (pack_cumulative_args (&cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    /* Skip the current argument.  */
> +    targetm.calls.function_arg_advance (pack_cumulative_args (&cum), arg);
>
>  #if TARGET_ABI_OPEN_VMS
>    /* For VMS, we allocate space for all 6 arg registers plus a count.
> diff --git a/gcc/config/arc/arc.cc b/gcc/config/arc/arc.cc
> index e6f52d87714..604a116e966 100644
> --- a/gcc/config/arc/arc.cc
> +++ b/gcc/config/arc/arc.cc
> @@ -2450,7 +2450,8 @@ arc_setup_incoming_varargs (cumulative_args_t 
> args_so_far,
>    /* We must treat `__builtin_va_alist' as an anonymous arg.  */
>
>    next_cum = *get_cumulative_args (args_so_far);
> -  arc_function_arg_advance (pack_cumulative_args (&next_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    arc_function_arg_advance (pack_cumulative_args (&next_cum), arg);
>    first_anon_arg = next_cum;
>
>    if (FUNCTION_ARG_REGNO_P (first_anon_arg))
> diff --git a/gcc/config/arm/arm.cc b/gcc/config/arm/arm.cc
> index ee8f1babf8a..2eb4d51e4a3 100644
> --- a/gcc/config/arm/arm.cc
> +++ b/gcc/config/arm/arm.cc
> @@ -29143,7 +29143,8 @@ arm_setup_incoming_varargs (cumulative_args_t pcum_v,
>    if (pcum->pcs_variant <= ARM_PCS_AAPCS_LOCAL)
>      {
>        nregs = pcum->aapcs_ncrn;
> -      if (nregs & 1)
> +      if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl))
> +         && (nregs & 1))
>         {
>           int res = arm_needs_doubleword_align (arg.mode, arg.type);
>           if (res < 0 && warn_psabi)
> diff --git a/gcc/config/csky/csky.cc b/gcc/config/csky/csky.cc
> index f7b2bf8e7c1..537eee6ab88 100644
> --- a/gcc/config/csky/csky.cc
> +++ b/gcc/config/csky/csky.cc
> @@ -2086,7 +2086,8 @@ csky_setup_incoming_varargs (cumulative_args_t pcum_v,
>
>    cfun->machine->uses_anonymous_args = 1;
>    local_cum = *pcum;
> -  csky_function_arg_advance (local_cum_v, arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    csky_function_arg_advance (local_cum_v, arg);
>    regs_to_push = CSKY_NPARM_REGS - local_cum.reg;
>    if (regs_to_push)
>      *pretend_size  = regs_to_push * UNITS_PER_WORD;
> diff --git a/gcc/config/epiphany/epiphany.cc b/gcc/config/epiphany/epiphany.cc
> index f8c04934085..c4e3ceaeb2a 100644
> --- a/gcc/config/epiphany/epiphany.cc
> +++ b/gcc/config/epiphany/epiphany.cc
> @@ -727,11 +727,13 @@ epiphany_setup_incoming_varargs (cumulative_args_t cum,
>    machine_function_t *mf = MACHINE_FUNCTION (cfun);
>
>    /* All BLKmode values are passed by reference.  */
> -  gcc_assert (arg.mode != BLKmode);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    gcc_assert (arg.mode != BLKmode);
>
>    next_cum = *get_cumulative_args (cum);
> -  next_cum = (ROUND_ADVANCE_CUM (next_cum, arg.mode, arg.type)
> -             + ROUND_ADVANCE_ARG (arg.mode, arg.type));
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    next_cum = (ROUND_ADVANCE_CUM (next_cum, arg.mode, arg.type)
> +               + ROUND_ADVANCE_ARG (arg.mode, arg.type));
>    first_anon_arg = next_cum;
>
>    if (first_anon_arg < MAX_EPIPHANY_PARM_REGS && !no_rtl)
> diff --git a/gcc/config/fr30/fr30.cc b/gcc/config/fr30/fr30.cc
> index c9b061d218b..334bb44e37f 100644
> --- a/gcc/config/fr30/fr30.cc
> +++ b/gcc/config/fr30/fr30.cc
> @@ -471,16 +471,19 @@ fr30_setup_incoming_varargs (cumulative_args_t 
> arg_regs_used_so_far_v,
>      = get_cumulative_args (arg_regs_used_so_far_v);
>    int size;
>
> -  /* All BLKmode values are passed by reference.  */
> -  gcc_assert (arg.mode != BLKmode);
> -
> -  /* ??? This run-time test as well as the code inside the if
> -     statement is probably unnecessary.  */
> -  if (targetm.calls.strict_argument_naming (arg_regs_used_so_far_v))
> -    /* If TARGET_STRICT_ARGUMENT_NAMING returns true, then the last named
> -       arg must not be treated as an anonymous arg.  */
> -    /* ??? This is a pointer increment, which makes no sense.  */
> -    arg_regs_used_so_far += fr30_num_arg_regs (arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    {
> +      /* All BLKmode values are passed by reference.  */
> +      gcc_assert (arg.mode != BLKmode);
> +
> +      /* ??? This run-time test as well as the code inside the if
> +        statement is probably unnecessary.  */
> +      if (targetm.calls.strict_argument_naming (arg_regs_used_so_far_v))
> +       /* If TARGET_STRICT_ARGUMENT_NAMING returns true, then the last named
> +          arg must not be treated as an anonymous arg.  */
> +       /* ??? This is a pointer increment, which makes no sense.  */
> +       arg_regs_used_so_far += fr30_num_arg_regs (arg);
> +    }
>
>    size = FR30_NUM_ARG_REGS - (* arg_regs_used_so_far);
>
> diff --git a/gcc/config/frv/frv.cc b/gcc/config/frv/frv.cc
> index 6f1904b358c..5cdb0bfe6e9 100644
> --- a/gcc/config/frv/frv.cc
> +++ b/gcc/config/frv/frv.cc
> @@ -2104,7 +2104,8 @@ frv_setup_incoming_varargs (cumulative_args_t cum_v,
>  {
>    CUMULATIVE_ARGS *cum = get_cumulative_args (cum_v);
>
> -  if (TARGET_DEBUG_ARG)
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl))
> +      && TARGET_DEBUG_ARG)
>      fprintf (stderr,
>              "setup_vararg: words = %2d, mode = %4s, pretend_size = %d, 
> second_time = %d\n",
>              *cum, GET_MODE_NAME (arg.mode), *pretend_size, second_time);
> diff --git a/gcc/config/ft32/ft32.cc b/gcc/config/ft32/ft32.cc
> index ed2d1229d61..d6b73d48686 100644
> --- a/gcc/config/ft32/ft32.cc
> +++ b/gcc/config/ft32/ft32.cc
> @@ -634,8 +634,10 @@ ft32_setup_incoming_varargs (cumulative_args_t cum_v,
>                              int *pretend_size, int no_rtl ATTRIBUTE_UNUSED)
>  {
>    CUMULATIVE_ARGS *cum = get_cumulative_args (cum_v);
> -  int named_size =
> -    GET_MODE_SIZE (SImode) * (*cum - FT32_R0) + GET_MODE_SIZE (arg.mode);
> +  int named_size = 0;
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    named_size =
> +      GET_MODE_SIZE (SImode) * (*cum - FT32_R0) + GET_MODE_SIZE (arg.mode);
>
>    if (named_size < 24)
>      *pretend_size = 24 - named_size;
> diff --git a/gcc/config/i386/i386.cc b/gcc/config/i386/i386.cc
> index 480db35f6cd..e92b12129b8 100644
> --- a/gcc/config/i386/i386.cc
> +++ b/gcc/config/i386/i386.cc
> @@ -4559,7 +4559,8 @@ ix86_setup_incoming_varargs (cumulative_args_t cum_v,
>    /* For varargs, we do not want to skip the dummy va_dcl argument.
>       For stdargs, we do want to skip the last named argument.  */
>    next_cum = *cum;
> -  if (stdarg_p (fntype))
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl))
> +      && stdarg_p (fntype))
>      ix86_function_arg_advance (pack_cumulative_args (&next_cum), arg);
>
>    if (cum->call_abi == MS_ABI)
> diff --git a/gcc/config/ia64/ia64.cc b/gcc/config/ia64/ia64.cc
> index 995ff906940..6df1ce736bc 100644
> --- a/gcc/config/ia64/ia64.cc
> +++ b/gcc/config/ia64/ia64.cc
> @@ -4596,8 +4596,9 @@ ia64_setup_incoming_varargs (cumulative_args_t cum,
>  {
>    CUMULATIVE_ARGS next_cum = *get_cumulative_args (cum);
>
> -  /* Skip the current argument.  */
> -  ia64_function_arg_advance (pack_cumulative_args (&next_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    /* Skip the current argument.  */
> +    ia64_function_arg_advance (pack_cumulative_args (&next_cum), arg);
>
>    if (next_cum.words < MAX_ARGUMENT_SLOTS)
>      {
> diff --git a/gcc/config/loongarch/loongarch.cc 
> b/gcc/config/loongarch/loongarch.cc
> index e9ba3374e35..f54c233f90c 100644
> --- a/gcc/config/loongarch/loongarch.cc
> +++ b/gcc/config/loongarch/loongarch.cc
> @@ -756,7 +756,8 @@ loongarch_setup_incoming_varargs (cumulative_args_t cum,
>       argument.  Advance a local copy of CUM past the last "real" named
>       argument, to find out how many registers are left over.  */
>    local_cum = *get_cumulative_args (cum);
> -  loongarch_function_arg_advance (pack_cumulative_args (&local_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    loongarch_function_arg_advance (pack_cumulative_args (&local_cum), arg);
>
>    /* Found out how many registers we need to save.  */
>    gp_saved = MAX_ARGS_IN_REGISTERS - local_cum.num_gprs;
> diff --git a/gcc/config/m32r/m32r.cc b/gcc/config/m32r/m32r.cc
> index bca768172b7..e3489fb4dc0 100644
> --- a/gcc/config/m32r/m32r.cc
> +++ b/gcc/config/m32r/m32r.cc
> @@ -1287,11 +1287,15 @@ m32r_setup_incoming_varargs (cumulative_args_t cum,
>      return;
>
>    /* All BLKmode values are passed by reference.  */
> -  gcc_assert (arg.mode != BLKmode);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    gcc_assert (arg.mode != BLKmode);
>
> -  first_anon_arg = (ROUND_ADVANCE_CUM (*get_cumulative_args (cum),
> -                                      arg.mode, arg.type)
> -                   + ROUND_ADVANCE_ARG (arg.mode, arg.type));
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    first_anon_arg = (ROUND_ADVANCE_CUM (*get_cumulative_args (cum),
> +                                        arg.mode, arg.type)
> +                     + ROUND_ADVANCE_ARG (arg.mode, arg.type));
> +  else
> +    first_anon_arg = *get_cumulative_args (cum);
>
>    if (first_anon_arg < M32R_MAX_PARM_REGS)
>      {
> diff --git a/gcc/config/mcore/mcore.cc b/gcc/config/mcore/mcore.cc
> index 28e707496d1..605d63b6a70 100644
> --- a/gcc/config/mcore/mcore.cc
> +++ b/gcc/config/mcore/mcore.cc
> @@ -1953,8 +1953,9 @@ mcore_setup_incoming_varargs (cumulative_args_t 
> args_so_far_v,
>    /* We need to know how many argument registers are used before
>       the varargs start, so that we can push the remaining argument
>       registers during the prologue.  */
> -  number_of_regs_before_varargs
> -    = *args_so_far + mcore_num_arg_regs (arg.mode, arg.type);
> +  number_of_regs_before_varargs = *args_so_far;
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    number_of_regs_before_varargs += mcore_num_arg_regs (arg.mode, arg.type);
>
>    /* There is a bug somewhere in the arg handling code.
>       Until I can find it this workaround always pushes the
> diff --git a/gcc/config/mips/mips.cc b/gcc/config/mips/mips.cc
> index 387376b3df8..53478e9227b 100644
> --- a/gcc/config/mips/mips.cc
> +++ b/gcc/config/mips/mips.cc
> @@ -6683,7 +6683,8 @@ mips_setup_incoming_varargs (cumulative_args_t cum,
>       argument.  Advance a local copy of CUM past the last "real" named
>       argument, to find out how many registers are left over.  */
>    local_cum = *get_cumulative_args (cum);
> -  mips_function_arg_advance (pack_cumulative_args (&local_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    mips_function_arg_advance (pack_cumulative_args (&local_cum), arg);
>
>    /* Found out how many registers we need to save.  */
>    gp_saved = MAX_ARGS_IN_REGISTERS - local_cum.num_gprs;
> diff --git a/gcc/config/mmix/mmix.cc b/gcc/config/mmix/mmix.cc
> index ffdd8c71cc1..1ac7b883ac5 100644
> --- a/gcc/config/mmix/mmix.cc
> +++ b/gcc/config/mmix/mmix.cc
> @@ -999,7 +999,8 @@ mmix_setup_incoming_varargs (cumulative_args_t 
> args_so_farp_v,
>
>    /* We assume that one argument takes up one register here.  That should
>       be true until we start messing with multi-reg parameters.  */
> -  if ((7 + (MMIX_FUNCTION_ARG_SIZE (arg.mode, arg.type))) / 8 != 1)
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl))
> +      && (7 + (MMIX_FUNCTION_ARG_SIZE (arg.mode, arg.type))) / 8 != 1)
>      internal_error ("MMIX Internal: Last named vararg would not fit in a 
> register");
>  }
>
> diff --git a/gcc/config/nds32/nds32.cc b/gcc/config/nds32/nds32.cc
> index 67a612130fe..639baef6c17 100644
> --- a/gcc/config/nds32/nds32.cc
> +++ b/gcc/config/nds32/nds32.cc
> @@ -2377,9 +2377,12 @@ nds32_setup_incoming_varargs (cumulative_args_t ca,
>       for varargs.  */
>    total_args_regs
>      = NDS32_MAX_GPR_REGS_FOR_ARGS + NDS32_GPR_ARG_FIRST_REGNUM;
> -  num_of_used_regs
> -    = NDS32_AVAILABLE_REGNUM_FOR_GPR_ARG (cum->gpr_offset, arg.mode, 
> arg.type)
> -      + NDS32_NEED_N_REGS_FOR_ARG (arg.mode, arg.type);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    num_of_used_regs
> +      = NDS32_AVAILABLE_REGNUM_FOR_GPR_ARG (cum->gpr_offset, arg.mode, 
> arg.type)
> +        + NDS32_NEED_N_REGS_FOR_ARG (arg.mode, arg.type);
> +  else
> +    num_of_used_regs = cum->gpr_offset + NDS32_GPR_ARG_FIRST_REGNUM;
>
>    remaining_reg_count = total_args_regs - num_of_used_regs;
>    *pretend_args_size = remaining_reg_count * UNITS_PER_WORD;
> diff --git a/gcc/config/nios2/nios2.cc b/gcc/config/nios2/nios2.cc
> index 1a33c88f19f..6a894ec345e 100644
> --- a/gcc/config/nios2/nios2.cc
> +++ b/gcc/config/nios2/nios2.cc
> @@ -3524,7 +3524,8 @@ nios2_setup_incoming_varargs (cumulative_args_t cum_v,
>
>    cfun->machine->uses_anonymous_args = 1;
>    local_cum = *cum;
> -  nios2_function_arg_advance (local_cum_v, arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    nios2_function_arg_advance (local_cum_v, arg);
>
>    regs_to_push = NUM_ARG_REGS - local_cum.regs_used;
>
> diff --git a/gcc/config/riscv/riscv.cc b/gcc/config/riscv/riscv.cc
> index ad57b995e7b..04d21ec8f3a 100644
> --- a/gcc/config/riscv/riscv.cc
> +++ b/gcc/config/riscv/riscv.cc
> @@ -3728,7 +3728,8 @@ riscv_setup_incoming_varargs (cumulative_args_t cum,
>       argument.  Advance a local copy of CUM past the last "real" named
>       argument, to find out how many registers are left over.  */
>    local_cum = *get_cumulative_args (cum);
> -  riscv_function_arg_advance (pack_cumulative_args (&local_cum), arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    riscv_function_arg_advance (pack_cumulative_args (&local_cum), arg);
>
>    /* Found out how many registers we need to save.  */
>    gp_saved = MAX_ARGS_IN_REGISTERS - local_cum.num_gprs;
> diff --git a/gcc/config/rs6000/rs6000-call.cc 
> b/gcc/config/rs6000/rs6000-call.cc
> index ac3cb7e3d36..6da4de67137 100644
> --- a/gcc/config/rs6000/rs6000-call.cc
> +++ b/gcc/config/rs6000/rs6000-call.cc
> @@ -2253,7 +2253,9 @@ setup_incoming_varargs (cumulative_args_t cum,
>
>    /* Skip the last named argument.  */
>    next_cum = *get_cumulative_args (cum);
> -  rs6000_function_arg_advance_1 (&next_cum, arg.mode, arg.type, arg.named, 
> 0);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    rs6000_function_arg_advance_1 (&next_cum, arg.mode, arg.type, arg.named,
> +                                  0);
>
>    if (DEFAULT_ABI == ABI_V4)
>      {
> @@ -2327,7 +2329,8 @@ setup_incoming_varargs (cumulative_args_t cum,
>        first_reg_offset = next_cum.words;
>        save_area = crtl->args.internal_arg_pointer;
>
> -      if (targetm.calls.must_pass_in_stack (arg))
> +      if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl))
> +         && targetm.calls.must_pass_in_stack (arg))
>         first_reg_offset += rs6000_arg_size (TYPE_MODE (arg.type), arg.type);
>      }
>
> diff --git a/gcc/config/sh/sh.cc b/gcc/config/sh/sh.cc
> index 9bee618b639..1aec70a23d8 100644
> --- a/gcc/config/sh/sh.cc
> +++ b/gcc/config/sh/sh.cc
> @@ -8183,11 +8183,12 @@ sh_setup_incoming_varargs (cumulative_args_t ca,
>    gcc_assert (cfun->stdarg);
>    if (TARGET_VARARGS_PRETEND_ARGS (current_function_decl))
>      {
> -      int named_parm_regs, anon_parm_regs;
> +      int named_parm_regs = 0, anon_parm_regs;
>
> -      named_parm_regs = (sh_round_reg (*get_cumulative_args (ca), arg.mode)
> -                        + CEIL (arg.promoted_size_in_bytes (),
> -                                UNITS_PER_WORD));
> +      if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +       named_parm_regs = (sh_round_reg (*get_cumulative_args (ca), arg.mode)
> +                          + CEIL (arg.promoted_size_in_bytes (),
> +                                  UNITS_PER_WORD));
>        anon_parm_regs = NPARM_REGS (SImode) - named_parm_regs;
>        if (anon_parm_regs > 0)
>         *pretend_arg_size = anon_parm_regs * 4;
> diff --git a/gcc/config/visium/visium.cc b/gcc/config/visium/visium.cc
> index 03c1a33e1b9..e7d15960fc7 100644
> --- a/gcc/config/visium/visium.cc
> +++ b/gcc/config/visium/visium.cc
> @@ -1481,7 +1481,8 @@ visium_setup_incoming_varargs (cumulative_args_t pcum_v,
>    /* The caller has advanced ARGS_SO_FAR up to, but not beyond, the last 
> named
>       argument.  Advance a local copy of ARGS_SO_FAR past the last "real" 
> named
>       argument, to find out how many registers are left over.  */
> -  TARGET_FUNCTION_ARG_ADVANCE (local_args_so_far, arg);
> +  if (!TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (current_function_decl)))
> +    TARGET_FUNCTION_ARG_ADVANCE (local_args_so_far, arg);
>
>    /* Find how many registers we need to save.  */
>    locargs = get_cumulative_args (local_args_so_far);
> diff --git a/gcc/config/vms/vms-c.cc b/gcc/config/vms/vms-c.cc
> index 2f74fb574cd..ccf6d5fe3b6 100644
> --- a/gcc/config/vms/vms-c.cc
> +++ b/gcc/config/vms/vms-c.cc
> @@ -455,9 +455,6 @@ vms_c_register_includes (const char *sysroot,
>  void
>  vms_c_common_override_options (void)
>  {
> -  /* Allow variadic functions without parameters (as declared in starlet).  
> */
> -  flag_allow_parameterless_variadic_functions = TRUE;
> -
>    /* Initialize c_default_pointer_mode.  */
>    switch (flag_vms_pointer_size)
>      {
> diff --git a/gcc/doc/invoke.texi b/gcc/doc/invoke.texi
> index cd4d3c1d72c..9f1481bbeb1 100644
> --- a/gcc/doc/invoke.texi
> +++ b/gcc/doc/invoke.texi
> @@ -198,7 +198,7 @@ in the following sections.
>  @item C Language Options
>  @xref{C Dialect Options,,Options Controlling C Dialect}.
>  @gccoptlist{-ansi  -std=@var{standard}  -aux-info @var{filename} @gol
> --fallow-parameterless-variadic-functions  -fno-asm  @gol
> +-fno-asm  @gol
>  -fno-builtin  -fno-builtin-@var{function}  -fcond-mismatch @gol
>  -ffreestanding  -fgimple  -fgnu-tm  -fgnu89-inline  -fhosted @gol
>  -flax-vector-conversions  -fms-extensions @gol
> @@ -2514,14 +2514,6 @@ character).  In the case of function definitions, a 
> K&R-style list of
>  arguments followed by their declarations is also provided, inside
>  comments, after the declaration.
>
> -@item -fallow-parameterless-variadic-functions
> -@opindex fallow-parameterless-variadic-functions
> -Accept variadic functions without named parameters.
> -
> -Although it is possible to define such a function, this is not very
> -useful as it is not possible to read the arguments.  This is only
> -supported for C as this construct is allowed by C++.
> -
>  @item -fno-asm
>  @opindex fno-asm
>  @opindex fasm
> diff --git a/gcc/doc/tm.texi b/gcc/doc/tm.texi
> index 110f8dfa0a9..63c8a3177ee 100644
> --- a/gcc/doc/tm.texi
> +++ b/gcc/doc/tm.texi
> @@ -5378,7 +5378,9 @@ pass all their arguments on the stack.
>  The argument @var{args_so_far} points to the @code{CUMULATIVE_ARGS} data
>  structure, containing the values that are obtained after processing the
>  named arguments.  The argument @var{arg} describes the last of these named
> -arguments.
> +arguments.  The argument @var{arg} should not be used if the function type
> +satisfies @code{TYPE_NO_NAMED_ARGS_STDARG_P}, since in that case there are
> +no named arguments and all arguments are accessed with @code{va_arg}.
>
>  The target hook should do two things: first, push onto the stack all the
>  argument registers @emph{not} used for the named arguments, and second,
> diff --git a/gcc/function.cc b/gcc/function.cc
> index 6474a663b30..d3da20ede7f 100644
> --- a/gcc/function.cc
> +++ b/gcc/function.cc
> @@ -3647,6 +3647,12 @@ assign_parms (tree fndecl)
>    assign_parms_initialize_all (&all);
>    fnargs = assign_parms_augmented_arg_list (&all);
>
> +  if (TYPE_NO_NAMED_ARGS_STDARG_P (TREE_TYPE (fndecl)))
> +    {
> +      struct assign_parm_data_one data = {};
> +      assign_parms_setup_varargs (&all, &data, false);
> +    }
> +
>    FOR_EACH_VEC_ELT (fnargs, i, parm)
>      {
>        struct assign_parm_data_one data;
> diff --git a/gcc/ginclude/stdarg.h b/gcc/ginclude/stdarg.h
> index 7545ed30424..c704c9ffcf2 100644
> --- a/gcc/ginclude/stdarg.h
> +++ b/gcc/ginclude/stdarg.h
> @@ -44,7 +44,11 @@ typedef __builtin_va_list __gnuc_va_list;
>     if this invocation was from the user program.  */
>  #ifdef _STDARG_H
>
> +#if defined __STDC_VERSION__ && __STDC_VERSION__ > 201710L
> +#define va_start(v, ...)       __builtin_va_start(v, 0)
> +#else
>  #define va_start(v,l)  __builtin_va_start(v,l)
> +#endif
>  #define va_end(v)      __builtin_va_end(v)
>  #define va_arg(v,l)    __builtin_va_arg(v,l)
>  #if !defined(__STRICT_ANSI__) || __STDC_VERSION__ + 0 >= 199900L \
> diff --git a/gcc/objc/objc-next-runtime-abi-01.cc 
> b/gcc/objc/objc-next-runtime-abi-01.cc
> index 409b777b9e5..8d41886902b 100644
> --- a/gcc/objc/objc-next-runtime-abi-01.cc
> +++ b/gcc/objc/objc-next-runtime-abi-01.cc
> @@ -2443,7 +2443,7 @@ build_next_objc_exception_stuff (void)
>    /* int _setjmp(...); */
>    /* If the user includes <setjmp.h>, this shall be superseded by
>       'int _setjmp(jmp_buf);' */
> -  temp_type = build_varargs_function_type_list (integer_type_node, 
> NULL_TREE);
> +  temp_type = build_function_type (integer_type_node, NULL_TREE);
>    objc_setjmp_decl
>      = add_builtin_function (TAG_SETJMP, temp_type, 0, NOT_BUILT_IN, NULL, 
> NULL_TREE);
>
> diff --git a/gcc/target.def b/gcc/target.def
> index a3d3b04a165..25f94c19fa7 100644
> --- a/gcc/target.def
> +++ b/gcc/target.def
> @@ -4680,7 +4680,9 @@ pass all their arguments on the stack.\n\
>  The argument @var{args_so_far} points to the @code{CUMULATIVE_ARGS} data\n\
>  structure, containing the values that are obtained after processing the\n\
>  named arguments.  The argument @var{arg} describes the last of these named\n\
> -arguments.\n\
> +arguments.  The argument @var{arg} should not be used if the function type\n\
> +satisfies @code{TYPE_NO_NAMED_ARGS_STDARG_P}, since in that case there are\n\
> +no named arguments and all arguments are accessed with @code{va_arg}.\n\
>  \n\
>  The target hook should do two things: first, push onto the stack all the\n\
>  argument registers @emph{not} used for the named arguments, and second,\n\
> diff --git a/gcc/testsuite/gcc.dg/Wold-style-definition-2.c 
> b/gcc/testsuite/gcc.dg/Wold-style-definition-2.c
> index a69aae6fd27..8e297c96411 100644
> --- a/gcc/testsuite/gcc.dg/Wold-style-definition-2.c
> +++ b/gcc/testsuite/gcc.dg/Wold-style-definition-2.c
> @@ -5,6 +5,6 @@
>  /* { dg-do compile } */
>  /* { dg-options "-Wold-style-definition" } */
>
> -void bar1 ( ... ) {} /* { dg-error "ISO C requires a named argument" } */
> +void bar1 ( ... ) {}
>
>  void bar2 (int a, ... ) {}
> diff --git a/gcc/testsuite/gcc.dg/c11-stdarg-1.c 
> b/gcc/testsuite/gcc.dg/c11-stdarg-1.c
> new file mode 100644
> index 00000000000..984577fe656
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c11-stdarg-1.c
> @@ -0,0 +1,7 @@
> +/* Test variadic functions with no named parameters not supported in C11.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c11 -pedantic-errors" } */
> +
> +int f (...); /* { dg-error "ISO C requires a named argument before" } */
> +int g (int (...)); /* { dg-error "ISO C requires a named argument before" } 
> */
> +int h (...) { return 0; } /* { dg-error "ISO C requires a named argument 
> before" } */
> diff --git a/gcc/testsuite/gcc.dg/c11-stdarg-2.c 
> b/gcc/testsuite/gcc.dg/c11-stdarg-2.c
> new file mode 100644
> index 00000000000..bd115e8850c
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c11-stdarg-2.c
> @@ -0,0 +1,7 @@
> +/* Test variadic functions with no named parameters not supported in C11.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c11 -pedantic" } */
> +
> +int f (...); /* { dg-warning "ISO C requires a named argument before" } */
> +int g (int (...)); /* { dg-warning "ISO C requires a named argument before" 
> } */
> +int h (...) { return 0; } /* { dg-warning "ISO C requires a named argument 
> before" } */
> diff --git a/gcc/testsuite/gcc.dg/c11-stdarg-3.c 
> b/gcc/testsuite/gcc.dg/c11-stdarg-3.c
> new file mode 100644
> index 00000000000..009292461bd
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c11-stdarg-3.c
> @@ -0,0 +1,8 @@
> +/* Test variadic functions with no named parameters not supported in C11, but
> +   diagnostic disabled with -Wno-c11-c2x-compat.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c11 -pedantic-errors -Wno-c11-c2x-compat" } */
> +
> +int f (...);
> +int g (int (...));
> +int h (...) { return 0; }
> diff --git a/gcc/testsuite/gcc.dg/c2x-nullptr-1.c 
> b/gcc/testsuite/gcc.dg/c2x-nullptr-1.c
> index 9501b514f1c..9f2cb6c8256 100644
> --- a/gcc/testsuite/gcc.dg/c2x-nullptr-1.c
> +++ b/gcc/testsuite/gcc.dg/c2x-nullptr-1.c
> @@ -226,6 +226,7 @@ test4 (void)
>  static void
>  test5 (int i, ...)
>  {
> +  (void) i;
>    va_list ap;
>    va_start (ap, i);
>    if (va_arg (ap, void *))
> diff --git a/gcc/testsuite/gcc.dg/c2x-stdarg-1.c 
> b/gcc/testsuite/gcc.dg/c2x-stdarg-1.c
> new file mode 100644
> index 00000000000..7def49d3ce2
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c2x-stdarg-1.c
> @@ -0,0 +1,22 @@
> +/* Test C2x variadic functions with no named parameters.  Compilation tests,
> +   valid code.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c2x -pedantic-errors" } */
> +
> +int f (...);
> +int g (int (...));
> +int h (...) { return 0; }
> +
> +typedef int A[];
> +typedef int A2[2];
> +
> +A *f1 (...);
> +A2 *f1 (...);
> +A *f1 (...) { return 0; }
> +
> +A2 *f2 (...);
> +A *f2 (...);
> +A2 *f2 (...) { return 0; }
> +typeof (f1) f2;
> +
> +int t () { return f () + f (1) + f (1, 2) + h () + h (1.5, 2, f1) + g (f); }
> diff --git a/gcc/testsuite/gcc.dg/c2x-stdarg-2.c 
> b/gcc/testsuite/gcc.dg/c2x-stdarg-2.c
> new file mode 100644
> index 00000000000..27782401c93
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c2x-stdarg-2.c
> @@ -0,0 +1,22 @@
> +/* Test C2x variadic functions with no named parameters.  Compilation tests,
> +   valid code, verify not considered unprototyped functions.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c2x -pedantic-errors -Wstrict-prototypes 
> -Wold-style-definition" } */
> +
> +int f (...);
> +int g (int (...));
> +int h (...) { return 0; }
> +
> +typedef int A[];
> +typedef int A2[2];
> +
> +A *f1 (...);
> +A2 *f1 (...);
> +A *f1 (...) { return 0; }
> +
> +A2 *f2 (...);
> +A *f2 (...);
> +A2 *f2 (...) { return 0; }
> +typeof (f1) f2;
> +
> +int t () { return f () + f (1) + f (1, 2) + h () + h (1.5, 2, f1) + g (f); }
> diff --git a/gcc/testsuite/gcc.dg/c2x-stdarg-3.c 
> b/gcc/testsuite/gcc.dg/c2x-stdarg-3.c
> new file mode 100644
> index 00000000000..e2e14063e91
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c2x-stdarg-3.c
> @@ -0,0 +1,16 @@
> +/* Test C2x variadic functions with no named parameters.  Compilation tests,
> +   invalid code.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=c2x -pedantic-errors" } */
> +
> +int f (...); /* { dg-message "previous declaration" } */
> +int f (); /* { dg-error "conflicting types" } */
> +
> +int f2 (...); /* { dg-message "previous declaration" } */
> +int f2 (int); /* { dg-error "conflicting types" } */
> +
> +int g (); /* { dg-message "previous declaration" } */
> +int g (...); /* { dg-error "conflicting types" } */
> +
> +int g2 (int); /* { dg-message "previous declaration" } */
> +int g2 (...); /* { dg-error "conflicting types" } */
> diff --git a/gcc/testsuite/gcc.dg/c2x-stdarg-4.c 
> b/gcc/testsuite/gcc.dg/c2x-stdarg-4.c
> new file mode 100644
> index 00000000000..cb7a5cb9317
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/c2x-stdarg-4.c
> @@ -0,0 +1,42 @@
> +/* Test C2x variadic functions with no named parameters.  Execution tests.  
> */
> +/* { dg-do run } */
> +/* { dg-options "-std=c2x -pedantic-errors" } */
> +
> +#include <stdarg.h>
> +
> +extern void abort (void);
> +extern void exit (int);
> +
> +double
> +f (...)
> +{
> +  va_list ap;
> +  va_start (ap);
> +  double ret = va_arg (ap, int);
> +  ret += va_arg (ap, double);
> +  ret += va_arg (ap, int);
> +  ret += va_arg (ap, double);
> +  va_end (ap);
> +  return ret;
> +}
> +
> +void
> +g (...)
> +{
> +  va_list ap;
> +  va_start (ap, random ! ignored, ignored ** text);
> +  for (int i = 0; i < 10; i++)
> +    if (va_arg (ap, double) != i)
> +      abort ();
> +  va_end (ap);
> +}
> +
> +int
> +main ()
> +{
> +  if (f (1, 2.0, 3, 4.0) != 10.0)
> +    abort ();
> +  g (0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0);
> +  g (0.0f, 1.0f, 2.0f, 3.0f, 4.0f, 5.0f, 6.0f, 7.0f, 8.0f, 9.0f);
> +  exit (0);
> +}
> diff --git a/gcc/testsuite/gcc.dg/diagnostic-token-ranges.c 
> b/gcc/testsuite/gcc.dg/diagnostic-token-ranges.c
> index 7d7ec0a9e0b..31085be1421 100644
> --- a/gcc/testsuite/gcc.dg/diagnostic-token-ranges.c
> +++ b/gcc/testsuite/gcc.dg/diagnostic-token-ranges.c
> @@ -1,4 +1,4 @@
> -/* { dg-options "-fdiagnostics-show-caret -Wc++-compat -std=c11" } */
> +/* { dg-options "-fdiagnostics-show-caret -Wc++-compat -std=c11 -pedantic" } 
> */
>
>  /* Verify that various diagnostics show source code ranges.  */
>
> @@ -48,7 +48,7 @@ void test_identifier_conflicts_with_cplusplus (void)
>  }
>
>  extern void
> -bogus_varargs (...); /* { dg-error "ISO C requires a named argument before 
> '...'" } */
> +bogus_varargs (...); /* { dg-warning "ISO C requires a named argument before 
> '...'" } */
>  /*
>  { dg-begin-multiline-output "" }
>   bogus_varargs (...);
> diff --git a/gcc/testsuite/gcc.dg/format/sentinel-1.c 
> b/gcc/testsuite/gcc.dg/format/sentinel-1.c
> index 0c8a2ac7737..16c75a8a961 100644
> --- a/gcc/testsuite/gcc.dg/format/sentinel-1.c
> +++ b/gcc/testsuite/gcc.dg/format/sentinel-1.c
> @@ -15,7 +15,7 @@ extern char *envp[];
>  extern int a ATTR; /* { dg-warning "applies to function types" "sentinel" } 
> */
>
>  extern void foo1 (const char *, ...) ATTR; /* { dg-message "note: declared 
> here" } */
> -extern void foo2 (...) ATTR; /* { dg-error "ISO C requires|named arguments" 
> "sentinel" } */
> +extern void foo2 (...) ATTR;
>  extern void foo3 () ATTR; /* { dg-warning "named arguments" "sentinel" } */
>  extern void foo4 (const char *, int) ATTR; /* { dg-warning "variadic 
> functions" "sentinel" } */
>  extern void foo5 (const char *, ...) __attribute__ ((__sentinel__(1)));
> diff --git a/gcc/testsuite/gcc.dg/gnu2x-stdarg-1.c 
> b/gcc/testsuite/gcc.dg/gnu2x-stdarg-1.c
> new file mode 100644
> index 00000000000..bb64cdeb48a
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/gnu2x-stdarg-1.c
> @@ -0,0 +1,8 @@
> +/* Test variadic functions with no named parameters do not accept GNU
> +   attributes before '...'.  */
> +/* { dg-do compile } */
> +/* { dg-options "-std=gnu2x" } */
> +
> +int f (__attribute__(()) ...); /* { dg-error "expected" } */
> +int g (int (__attribute__(()) ...)); /* { dg-error "expected" } */
> +int h (__attribute__(()) ...) { return 0; } /* { dg-error "expected" } */
> diff --git a/gcc/tree-core.h b/gcc/tree-core.h
> index 80b886cc3e4..af75522504f 100644
> --- a/gcc/tree-core.h
> +++ b/gcc/tree-core.h
> @@ -1717,7 +1717,8 @@ struct GTY(()) tree_type_common {
>    unsigned typeless_storage : 1;
>    unsigned empty_flag : 1;
>    unsigned indivisible_p : 1;
> -  unsigned spare : 16;
> +  unsigned no_named_args_stdarg_p : 1;
> +  unsigned spare : 15;
>
>    alias_set_type alias_set;
>    tree pointer_to;
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index 81a6ceaf181..48ab60d78fe 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -6112,7 +6112,9 @@ type_cache_hasher::equal (type_hash *a, type_hash *b)
>                                       TYPE_FIELDS (b->type))));
>
>      case FUNCTION_TYPE:
> -      if (TYPE_ARG_TYPES (a->type) == TYPE_ARG_TYPES (b->type)
> +      if ((TYPE_ARG_TYPES (a->type) == TYPE_ARG_TYPES (b->type)
> +          && (TYPE_NO_NAMED_ARGS_STDARG_P (a->type)
> +              == TYPE_NO_NAMED_ARGS_STDARG_P (b->type)))
>           || (TYPE_ARG_TYPES (a->type)
>               && TREE_CODE (TYPE_ARG_TYPES (a->type)) == TREE_LIST
>               && TYPE_ARG_TYPES (b->type)
> @@ -7364,10 +7366,13 @@ maybe_canonicalize_argtypes (tree argtypes,
>     given arguments of types ARG_TYPES.
>     ARG_TYPES is a chain of TREE_LIST nodes whose TREE_VALUEs
>     are data type nodes for the arguments of the function.
> +   NO_NAMED_ARGS_STDARG_P is true if this is a prototyped
> +   variable-arguments function with (...) prototype (no named arguments).
>     If such a type has already been constructed, reuse it.  */
>
>  tree
> -build_function_type (tree value_type, tree arg_types)
> +build_function_type (tree value_type, tree arg_types,
> +                    bool no_named_args_stdarg_p)
>  {
>    tree t;
>    inchash::hash hstate;
> @@ -7386,6 +7391,11 @@ build_function_type (tree value_type, tree arg_types)
>    t = make_node (FUNCTION_TYPE);
>    TREE_TYPE (t) = value_type;
>    TYPE_ARG_TYPES (t) = arg_types;
> +  if (no_named_args_stdarg_p)
> +    {
> +      gcc_assert (arg_types == NULL_TREE);
> +      TYPE_NO_NAMED_ARGS_STDARG_P (t) = 1;
> +    }
>
>    /* If we already have such a type, use the old one.  */
>    hashval_t hash = type_hash_canon_hash (t);
> @@ -7436,7 +7446,7 @@ build_function_type_list_1 (bool vaargs, tree 
> return_type, va_list argp)
>        args = nreverse (args);
>        TREE_CHAIN (last) = void_list_node;
>      }
> -  args = build_function_type (return_type, args);
> +  args = build_function_type (return_type, args, vaargs && args == 
> NULL_TREE);
>
>    return args;
>  }
> @@ -7491,7 +7501,7 @@ build_function_type_array_1 (bool vaargs, tree 
> return_type, int n,
>    for (i = n - 1; i >= 0; i--)
>      t = tree_cons (NULL_TREE, arg_types[i], t);
>
> -  return build_function_type (return_type, t);
> +  return build_function_type (return_type, t, vaargs && n == 0);
>  }
>
>  /* Build a function type.  RETURN_TYPE is the type returned by the
> @@ -9994,7 +10004,8 @@ reconstruct_complex_type (tree type, tree bottom)
>    else if (TREE_CODE (type) == FUNCTION_TYPE)
>      {
>        inner = reconstruct_complex_type (TREE_TYPE (type), bottom);
> -      outer = build_function_type (inner, TYPE_ARG_TYPES (type));
> +      outer = build_function_type (inner, TYPE_ARG_TYPES (type),
> +                                  TYPE_NO_NAMED_ARGS_STDARG_P (type));
>      }
>    else if (TREE_CODE (type) == METHOD_TYPE)
>      {
> @@ -11612,6 +11623,9 @@ stdarg_p (const_tree fntype)
>    if (!fntype)
>      return false;
>
> +  if (TYPE_NO_NAMED_ARGS_STDARG_P (fntype))
> +    return true;
> +
>    FOREACH_FUNCTION_ARGS (fntype, t, args_iter)
>      {
>        n = t;
> @@ -11629,6 +11643,9 @@ prototype_p (const_tree fntype)
>
>    gcc_assert (fntype != NULL_TREE);
>
> +  if (TYPE_NO_NAMED_ARGS_STDARG_P (fntype))
> +    return true;
> +
>    t = TYPE_ARG_TYPES (fntype);
>    return (t != NULL_TREE);
>  }
> @@ -13647,7 +13664,9 @@ gimple_canonical_types_compatible_p (const_tree t1, 
> const_tree t2,
>                                                 trust_type_canonical))
>         return false;
>
> -      if (TYPE_ARG_TYPES (t1) == TYPE_ARG_TYPES (t2))
> +      if (TYPE_ARG_TYPES (t1) == TYPE_ARG_TYPES (t2)
> +         && (TYPE_NO_NAMED_ARGS_STDARG_P (t1)
> +             == TYPE_NO_NAMED_ARGS_STDARG_P (t2)))
>         return true;
>        else
>         {
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 9af971cf401..885df4d0d66 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -772,6 +772,12 @@ extern void omp_clause_range_check_failed (const_tree, 
> const char *, int,
>     normal GNU extensions for target-specific vector types.  */
>  #define TYPE_INDIVISIBLE_P(NODE) (TYPE_CHECK 
> (NODE)->type_common.indivisible_p)
>
> +/* True if this is a stdarg function with no named arguments (C2x
> +   (...) prototype, where arguments can be accessed with va_start and
> +   va_arg), as opposed to an unprototyped function.  */
> +#define TYPE_NO_NAMED_ARGS_STDARG_P(NODE) \
> +  (TYPE_CHECK (NODE)->type_common.no_named_args_stdarg_p)
> +
>  /* In an IDENTIFIER_NODE, this means that assemble_name was called with
>     this string as an argument.  */
>  #define TREE_SYMBOL_REFERENCED(NODE) \
> @@ -4727,7 +4733,7 @@ extern tree build_array_type_1 (tree, tree, bool, bool, 
> bool);
>  extern tree build_array_type (tree, tree, bool = false);
>  extern tree build_nonshared_array_type (tree, tree);
>  extern tree build_array_type_nelts (tree, poly_uint64);
> -extern tree build_function_type (tree, tree);
> +extern tree build_function_type (tree, tree, bool = false);
>  extern tree build_function_type_list (tree, ...);
>  extern tree build_varargs_function_type_list (tree, ...);
>  extern tree build_function_type_array (tree, int, tree *);
>
> --
> Joseph S. Myers
> jos...@codesourcery.com

Reply via email to