aarch64: Fix parameter passing for [[no_unique_address]]

This patch makes the ABI code ignore zero-sized [[no_unique_address]]
fields when deciding whether something is a HFA or HVA.

As things stood, we'd get two sets of -Wpsabi warnings, one when
trying to decide whether something was an SVE function, and another
when actually processing the function definition or function call.
The patch therefore makes aapcs_vfp_sub_candidate honour the
CUMULATIVE_ARGS "silent_p" flag where applicable.

This doesn't stop all duplicate warnings for parameters, and I suspect
we'll get duplicate warnings for return values too, but it should be
better than nothing.

2020-04-29  Richard Sandiford  <richard.sandiford@arm.com>

gcc/
	* config/aarch64/aarch64.c (aarch64_function_arg_alignment): Add a
	comment explaining why we consider even zero-sized fields.
	(WARN_PSABI_EMPTY_CXX17_BASE): New constant.
	(WARN_PSABI_NO_UNIQUE_ADDRESS): Likewise.
	(aapcs_vfp_sub_candidate): Replace the boolean pointer parameter
	avoid_cxx17_empty_base with a pointer to a bitmask.  Ignore fields
	whose DECL_FIELD_ABI_IGNORED bit is set when determining whether
	something actually is a HFA or HVA.  Record whether we see a
	[[no_unique_address]] field that previous GCCs would not have
	ignored in this way.
	(aarch64_vfp_is_call_or_return_candidate): Add a parameter to say
	whether diagnostics should be suppressed.  Update the calls to
	aapcs_vfp_sub_candidate and report a -Wpsabi warning for the
	[[no_unique_address]] case.
	(aarch64_return_in_msb): Update call accordingly, never silencing
	diagnostics.
	(aarch64_function_value): Likewise.
	(aarch64_return_in_memory_1): Likewise.
	(aarch64_init_cumulative_args): Likewise.
	(aarch64_gimplify_va_arg_expr): Likewise.
	(aarch64_pass_by_reference_1): Take a CUMULATIVE_ARGS pointer and
	use it to decide whether arch64_vfp_is_call_or_return_candidate
	should be silent.
	(aarch64_pass_by_reference): Update calls accordingly.
	(aarch64_vfp_is_call_candidate): Use the CUMULATIVE_ARGS argument
	to decide whether arch64_vfp_is_call_or_return_candidate should be
	silent.

gcc/testsuite/
	* g++.target/aarch64/no_unique_address_1.C: New test.
	* g++.target/aarch64/no_unique_address_2.C: Likewise.
This commit is contained in:
Richard Sandiford 2020-04-29 10:56:49 +01:00
parent b5620fadc5
commit 56fe3ca30e
5 changed files with 553 additions and 53 deletions

View File

@ -1,3 +1,33 @@
2020-04-29 Richard Sandiford <richard.sandiford@arm.com>
* config/aarch64/aarch64.c (aarch64_function_arg_alignment): Add a
comment explaining why we consider even zero-sized fields.
(WARN_PSABI_EMPTY_CXX17_BASE): New constant.
(WARN_PSABI_NO_UNIQUE_ADDRESS): Likewise.
(aapcs_vfp_sub_candidate): Replace the boolean pointer parameter
avoid_cxx17_empty_base with a pointer to a bitmask. Ignore fields
whose DECL_FIELD_ABI_IGNORED bit is set when determining whether
something actually is a HFA or HVA. Record whether we see a
[[no_unique_address]] field that previous GCCs would not have
ignored in this way.
(aarch64_vfp_is_call_or_return_candidate): Add a parameter to say
whether diagnostics should be suppressed. Update the calls to
aapcs_vfp_sub_candidate and report a -Wpsabi warning for the
[[no_unique_address]] case.
(aarch64_return_in_msb): Update call accordingly, never silencing
diagnostics.
(aarch64_function_value): Likewise.
(aarch64_return_in_memory_1): Likewise.
(aarch64_init_cumulative_args): Likewise.
(aarch64_gimplify_va_arg_expr): Likewise.
(aarch64_pass_by_reference_1): Take a CUMULATIVE_ARGS pointer and
use it to decide whether arch64_vfp_is_call_or_return_candidate
should be silent.
(aarch64_pass_by_reference): Update calls accordingly.
(aarch64_vfp_is_call_candidate): Use the CUMULATIVE_ARGS argument
to decide whether arch64_vfp_is_call_or_return_candidate should be
silent.
2020-04-29 Haijian Zhang <z.zhanghaijian@huawei.com>
PR target/94820

View File

@ -286,7 +286,7 @@ static bool aarch64_return_in_memory_1 (const_tree);
static bool aarch64_vfp_is_call_or_return_candidate (machine_mode,
const_tree,
machine_mode *, int *,
bool *);
bool *, bool);
static void aarch64_elf_asm_constructor (rtx, int) ATTRIBUTE_UNUSED;
static void aarch64_elf_asm_destructor (rtx, int) ATTRIBUTE_UNUSED;
static void aarch64_override_options_after_change (void);
@ -5369,7 +5369,8 @@ aarch64_function_ok_for_sibcall (tree, tree exp)
passed in SVE registers. */
static bool
aarch64_pass_by_reference_1 (const function_arg_info &arg)
aarch64_pass_by_reference_1 (CUMULATIVE_ARGS *pcum,
const function_arg_info &arg)
{
HOST_WIDE_INT size;
machine_mode dummymode;
@ -5393,8 +5394,8 @@ aarch64_pass_by_reference_1 (const function_arg_info &arg)
/* Can this be a candidate to be passed in fp/simd register(s)? */
if (aarch64_vfp_is_call_or_return_candidate (arg.mode, arg.type,
&dummymode, &nregs,
NULL))
&dummymode, &nregs, NULL,
!pcum || pcum->silent_p))
return false;
/* Arguments which are variable sized or larger than 2 registers are
@ -5412,7 +5413,7 @@ aarch64_pass_by_reference (cumulative_args_t pcum_v,
CUMULATIVE_ARGS *pcum = get_cumulative_args (pcum_v);
if (!arg.type)
return aarch64_pass_by_reference_1 (arg);
return aarch64_pass_by_reference_1 (pcum, arg);
pure_scalable_type_info pst_info;
switch (pst_info.analyze (arg.type))
@ -5431,12 +5432,12 @@ aarch64_pass_by_reference (cumulative_args_t pcum_v,
|| pcum->aapcs_nprn + pst_info.num_pr () > NUM_PR_ARG_REGS);
case pure_scalable_type_info::DOESNT_MATTER:
gcc_assert (aarch64_pass_by_reference_1 (arg));
gcc_assert (aarch64_pass_by_reference_1 (pcum, arg));
return true;
case pure_scalable_type_info::NO_ABI_IDENTITY:
case pure_scalable_type_info::ISNT_PST:
return aarch64_pass_by_reference_1 (arg);
return aarch64_pass_by_reference_1 (pcum, arg);
}
gcc_unreachable ();
}
@ -5464,7 +5465,8 @@ aarch64_return_in_msb (const_tree valtype)
is always passed/returned in the least significant bits of fp/simd
register(s). */
if (aarch64_vfp_is_call_or_return_candidate (TYPE_MODE (valtype), valtype,
&dummy_mode, &dummy_int, NULL))
&dummy_mode, &dummy_int, NULL,
false))
return false;
/* Likewise pure scalable types for SVE vector and predicate registers. */
@ -5511,8 +5513,8 @@ aarch64_function_value (const_tree type, const_tree func,
int count;
machine_mode ag_mode;
if (aarch64_vfp_is_call_or_return_candidate (mode, type,
&ag_mode, &count, NULL))
if (aarch64_vfp_is_call_or_return_candidate (mode, type, &ag_mode, &count,
NULL, false))
{
gcc_assert (!sve_p);
if (!aarch64_composite_type_p (type, mode))
@ -5599,11 +5601,8 @@ aarch64_return_in_memory_1 (const_tree type)
/* Simple scalar types always returned in registers. */
return false;
if (aarch64_vfp_is_call_or_return_candidate (TYPE_MODE (type),
type,
&ag_mode,
&count,
NULL))
if (aarch64_vfp_is_call_or_return_candidate (TYPE_MODE (type), type,
&ag_mode, &count, NULL, false))
return false;
/* Types larger than 2 registers returned in memory. */
@ -5646,11 +5645,9 @@ aarch64_vfp_is_call_candidate (cumulative_args_t pcum_v, machine_mode mode,
const_tree type, int *nregs)
{
CUMULATIVE_ARGS *pcum = get_cumulative_args (pcum_v);
return aarch64_vfp_is_call_or_return_candidate (mode,
type,
return aarch64_vfp_is_call_or_return_candidate (mode, type,
&pcum->aapcs_vfp_rmode,
nregs,
NULL);
nregs, NULL, pcum->silent_p);
}
/* Given MODE and TYPE of a function argument, return the alignment in
@ -5684,6 +5681,19 @@ aarch64_function_arg_alignment (machine_mode mode, const_tree type,
for (tree field = TYPE_FIELDS (type); field; field = DECL_CHAIN (field))
if (TREE_CODE (field) == FIELD_DECL)
{
/* Note that we explicitly consider zero-sized fields here,
even though they don't map to AAPCS64 machine types.
For example, in:
struct __attribute__((aligned(8))) empty {};
struct s {
[[no_unique_address]] empty e;
int x;
};
"s" contains only one Fundamental Data Type (the int field)
but gains 8-byte alignment and size thanks to "e". */
alignment = std::max (alignment, DECL_ALIGN (field));
if (DECL_BIT_FIELD_TYPE (field))
bitfield_alignment
@ -5976,7 +5986,7 @@ aarch64_init_cumulative_args (CUMULATIVE_ARGS *pcum,
machine_mode mode ATTRIBUTE_UNUSED; /* To pass pointer as argument. */
int nregs ATTRIBUTE_UNUSED; /* Likewise. */
if (aarch64_vfp_is_call_or_return_candidate (TYPE_MODE (type), type,
&mode, &nregs, NULL))
&mode, &nregs, NULL, false))
aarch64_err_no_fpadvsimd (TYPE_MODE (type));
}
@ -16152,11 +16162,8 @@ aarch64_gimplify_va_arg_expr (tree valist, tree type, gimple_seq *pre_p,
dw_align = false;
adjust = 0;
if (aarch64_vfp_is_call_or_return_candidate (mode,
type,
&ag_mode,
&nregs,
&is_ha))
if (aarch64_vfp_is_call_or_return_candidate (mode, type, &ag_mode, &nregs,
&is_ha, false))
{
/* No frontends can create types with variable-sized modes, so we
shouldn't be asked to pass or return them. */
@ -16521,23 +16528,42 @@ aarch64_member_type_forces_blk (const_tree field_or_array, machine_mode mode)
return default_member_type_forces_blk (field_or_array, mode);
}
/* Bitmasks that indicate whether earlier versions of GCC would have
taken a different path through the ABI logic. This should result in
a -Wpsabi warning if the earlier path led to a different ABI decision.
WARN_PSABI_EMPTY_CXX17_BASE
Indicates that the type includes an artificial empty C++17 base field
that, prior to GCC 10.1, would prevent the type from being treated as
a HFA or HVA. See PR94383 for details.
WARN_PSABI_NO_UNIQUE_ADDRESS
Indicates that the type includes an empty [[no_unique_address]] field
that, prior to GCC 10.1, would prevent the type from being treated as
a HFA or HVA. */
const unsigned int WARN_PSABI_EMPTY_CXX17_BASE = 1U << 0;
const unsigned int WARN_PSABI_NO_UNIQUE_ADDRESS = 1U << 1;
/* Walk down the type tree of TYPE counting consecutive base elements.
If *MODEP is VOIDmode, then set it to the first valid floating point
type. If a non-floating point type is found, or if a floating point
type that doesn't match a non-VOIDmode *MODEP is found, then return -1,
otherwise return the count in the sub-tree.
The AVOID_CXX17_EMPTY_BASE argument is to allow the caller to check whether
this function has changed its behavior after the fix for PR94384 -- this fix
is to avoid artificial fields in empty base classes.
When called with this argument as a NULL pointer this function does not
avoid the artificial fields -- this is useful to check whether the function
returns something different after the fix.
When called pointing at a value, this function avoids such artificial fields
and sets the value to TRUE when one of these fields has been set. */
The WARN_PSABI_FLAGS argument allows the caller to check whether this
function has changed its behavior relative to earlier versions of GCC.
Normally the argument should be nonnull and point to a zero-initialized
variable. The function then records whether the ABI decision might
be affected by a known fix to the ABI logic, setting the associated
WARN_PSABI_* bits if so.
When the argument is instead a null pointer, the function tries to
simulate the behavior of GCC before all such ABI fixes were made.
This is useful to check whether the function returns something
different after the ABI fixes. */
static int
aapcs_vfp_sub_candidate (const_tree type, machine_mode *modep,
bool *avoid_cxx17_empty_base)
unsigned int *warn_psabi_flags)
{
machine_mode mode;
HOST_WIDE_INT size;
@ -16614,7 +16640,7 @@ aapcs_vfp_sub_candidate (const_tree type, machine_mode *modep,
return -1;
count = aapcs_vfp_sub_candidate (TREE_TYPE (type), modep,
avoid_cxx17_empty_base);
warn_psabi_flags);
if (count == -1
|| !index
|| !TYPE_MAX_VALUE (index)
@ -16652,18 +16678,30 @@ aapcs_vfp_sub_candidate (const_tree type, machine_mode *modep,
if (TREE_CODE (field) != FIELD_DECL)
continue;
/* Ignore C++17 empty base fields, while their type indicates
they do contain padding, they have zero size and thus don't
contain any padding. */
if (cxx17_empty_base_field_p (field)
&& avoid_cxx17_empty_base)
if (DECL_FIELD_ABI_IGNORED (field))
{
*avoid_cxx17_empty_base = true;
continue;
/* See whether this is something that earlier versions of
GCC failed to ignore. */
unsigned int flag;
if (lookup_attribute ("no_unique_address",
DECL_ATTRIBUTES (field)))
flag = WARN_PSABI_NO_UNIQUE_ADDRESS;
else if (cxx17_empty_base_field_p (field))
flag = WARN_PSABI_EMPTY_CXX17_BASE;
else
/* No compatibility problem. */
continue;
/* Simulate the old behavior when WARN_PSABI_FLAGS is null. */
if (warn_psabi_flags)
{
*warn_psabi_flags |= flag;
continue;
}
}
sub_count = aapcs_vfp_sub_candidate (TREE_TYPE (field), modep,
avoid_cxx17_empty_base);
warn_psabi_flags);
if (sub_count < 0)
return -1;
count += sub_count;
@ -16697,7 +16735,7 @@ aapcs_vfp_sub_candidate (const_tree type, machine_mode *modep,
continue;
sub_count = aapcs_vfp_sub_candidate (TREE_TYPE (field), modep,
avoid_cxx17_empty_base);
warn_psabi_flags);
if (sub_count < 0)
return -1;
count = count > sub_count ? count : sub_count;
@ -16796,14 +16834,20 @@ aarch64_composite_type_p (const_tree type,
Upon successful return, *COUNT returns the number of needed registers,
*BASE_MODE returns the mode of the individual register and when IS_HAF
is not NULL, *IS_HA indicates whether or not the argument is a homogeneous
floating-point aggregate or a homogeneous short-vector aggregate. */
floating-point aggregate or a homogeneous short-vector aggregate.
SILENT_P is true if the function should refrain from reporting any
diagnostics. This should only be used if the caller is certain that
any ABI decisions would eventually come through this function with
SILENT_P set to false. */
static bool
aarch64_vfp_is_call_or_return_candidate (machine_mode mode,
const_tree type,
machine_mode *base_mode,
int *count,
bool *is_ha)
bool *is_ha,
bool silent_p)
{
if (is_ha != NULL) *is_ha = false;
@ -16824,24 +16868,33 @@ aarch64_vfp_is_call_or_return_candidate (machine_mode mode,
}
else if (type && composite_p)
{
bool avoided = false;
int ag_count = aapcs_vfp_sub_candidate (type, &new_mode, &avoided);
unsigned int warn_psabi_flags = 0;
int ag_count = aapcs_vfp_sub_candidate (type, &new_mode,
&warn_psabi_flags);
if (ag_count > 0 && ag_count <= HA_MAX_NUM_FLDS)
{
static unsigned last_reported_type_uid;
unsigned uid = TYPE_UID (TYPE_MAIN_VARIANT (type));
int alt;
if (warn_psabi
&& avoided
if (!silent_p
&& warn_psabi
&& warn_psabi_flags
&& uid != last_reported_type_uid
&& ((alt = aapcs_vfp_sub_candidate (type, &new_mode, NULL))
!= ag_count))
{
gcc_assert (alt == -1);
last_reported_type_uid = uid;
inform (input_location, "parameter passing for argument of type "
"%qT when C++17 is enabled changed to match C++14 "
"in GCC 10.1", type);
/* Use TYPE_MAIN_VARIANT to strip any redundant const
qualification. */
if (warn_psabi_flags & WARN_PSABI_NO_UNIQUE_ADDRESS)
inform (input_location, "parameter passing for argument of "
"type %qT with %<[[no_unique_address]]%> members "
"changed in GCC 10.1", TYPE_MAIN_VARIANT (type));
else if (warn_psabi_flags & WARN_PSABI_EMPTY_CXX17_BASE)
inform (input_location, "parameter passing for argument of "
"type %qT when C++17 is enabled changed to match "
"C++14 in GCC 10.1", TYPE_MAIN_VARIANT (type));
}
if (is_ha != NULL) *is_ha = true;

View File

@ -1,3 +1,8 @@
2020-04-29 Richard Sandiford <richard.sandiford@arm.com>
* g++.target/aarch64/no_unique_address_1.C: New test.
* g++.target/aarch64/no_unique_address_2.C: Likewise.
2020-04-29 Richard Sandiford <richard.sandiford@arm.com>
* g++.target/arm/mve.exp: Restore the original dg-do-what-default

View File

@ -0,0 +1,206 @@
/* { dg-options "-std=c++11 -O -foptimize-sibling-calls -fpeephole2" } */
/* { dg-final { check-function-bodies "**" "" "" { target lp64 } } } */
struct X { };
struct Y { int : 0; };
struct Z { int : 0; Y y; };
struct W : public X { X q; };
struct A { float a; };
struct B : public X { float a; };
struct C : public Y { float a; };
struct D : public Z { float a; };
struct E : public W { float a; };
struct F { [[no_unique_address]] X x; float a; };
struct G { [[no_unique_address]] Y y; float a; };
struct H { [[no_unique_address]] Z z; float a; };
struct I { [[no_unique_address]] W w; float a; };
struct J { float a; [[no_unique_address]] X x; float b; };
struct K { float a; [[no_unique_address]] Y y; float b; };
struct L { float a; [[no_unique_address]] Z z; float b; };
struct M { float a; [[no_unique_address]] W w; float b; };
struct N : public A { float b; };
struct O { [[no_unique_address]] A a; float b; };
struct P : public Y { int : 0; float a, b, c, d; };
union Q { X x; float a; };
union R { [[no_unique_address]] X x; float a; };
union S { A a; float b; };
union T { F f; float b; };
union U { N n; O o; };
typedef S Salias;
typedef T Talias;
typedef U Ualias;
#define T(S, s) extern int callee_##s (S)
/*
** _Z8caller_aR1A:
** ldr s0, \[x0\]
** b .*
*/
T (A, a); int caller_a (A &a) { return callee_a (a); } /* { dg-bogus {argument of type 'A'} } */
/*
** _Z8caller_bR1B:
** ldr s0, \[x0\]
** b .*
*/
T (B, b); int caller_b (B &b) { return callee_b (b); } /* { dg-bogus {argument of type 'B'} } */
/*
** _Z8caller_cR1C:
** ldr s0, \[x0\]
** b .*
*/
T (C, c); int caller_c (C &c) { return callee_c (c); } /* { dg-bogus {argument of type 'C'} } */
/*
** _Z8caller_dR1D:
** ldr x0, \[x0\]
** b .*
*/
T (D, d); int caller_d (D &d) { return callee_d (d); } /* { dg-bogus {argument of type 'D'} } */
/*
** _Z8caller_eR1E:
** ldr x0, \[x0\]
** b .*
*/
T (E, e); int caller_e (E &e) { return callee_e (e); } /* { dg-bogus {argument of type 'E'} } */
/*
** _Z8caller_fR1F:
** ldr s0, \[x0\]
** b .*
*/
T (F, f); int caller_f (F &f) { return callee_f (f); } /* { dg-message {parameter passing for argument of type 'F' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_gR1G:
** ldr s0, \[x0\]
** b .*
*/
T (G, g); int caller_g (G &g) { return callee_g (g); } /* { dg-message {parameter passing for argument of type 'G' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_hR1H:
** ldr x0, \[x0\]
** b .*
*/
T (H, h); int caller_h (H &h) { return callee_h (h); } /* { dg-bogus {argument of type 'H'} } */
/*
** _Z8caller_iR1I:
** ldr x0, \[x0\]
** b .*
*/
T (I, i); int caller_i (I &i) { return callee_i (i); } /* { dg-bogus {argument of type 'I'} } */
/*
** _Z8caller_jR1J:
** ldp s0, s1, \[x0\]
** b .*
*/
T (J, j); int caller_j (J &j) { return callee_j (j); } /* { dg-message {parameter passing for argument of type 'J' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_kR1K:
** ldp s0, s1, \[x0\]
** b .*
*/
T (K, k); int caller_k (K &k) { return callee_k (k); } /* { dg-message {parameter passing for argument of type 'K' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_lR1L: { target aarch64_little_endian }
** (
** ldr w1, \[x0, 8\]
** ldr x0, \[x0\]
** |
** mov (x[0-9]+), x0
** ldr x0, \[x0\]
** ldr w1, \[\1, 8\]
** )
** b .*
*/
T (L, l); int caller_l (L &l) { return callee_l (l); } /* { dg-bogus {argument of type 'L'} } */
/*
** _Z8caller_mR1M: { target aarch64_little_endian }
** (
** ldr w1, \[x0, 8\]
** ldr x0, \[x0\]
** |
** mov (x[0-9]+), x0
** ldr x0, \[x0\]
** ldr w1, \[\1, 8\]
** )
** b .*
*/
T (M, m); int caller_m (M &m) { return callee_m (m); } /* { dg-bogus {argument of type 'M'} } */
/*
** _Z8caller_nR1N:
** ldp s0, s1, \[x0\]
** b .*
*/
T (N, n); int caller_n (N &n) { return callee_n (n); } /* { dg-bogus {argument of type 'N'} } */
/*
** _Z8caller_oR1O:
** ldp s0, s1, \[x0\]
** b .*
*/
T (O, o); int caller_o (O &o) { return callee_o (o); } /* { dg-bogus {argument of type 'O'} } */
/*
** _Z8caller_pR1P:
** ldp s0, s1, \[x0\]
** ldp s2, s3, \[x0, 8\]
** b .*
*/
T (P, p); int caller_p (P &p) { return callee_p (p); } /* { dg-bogus {argument of type 'P'} } */
/*
** _Z8caller_qR1Q: { target aarch64_little_endian }
** ldr w0, \[x0\]
** b .*
*/
T (Q, q); int caller_q (Q &q) { return callee_q (q); } /* { dg-bogus {argument of type 'Q'} } */
/*
** _Z8caller_rR1R: { target aarch64_little_endian }
** ldr w0, \[x0\]
** b .*
*/
T (R, r); int caller_r (R &r) { return callee_r (r); } /* { dg-bogus {argument of type 'R'} } */
/*
** _Z8caller_sR1S:
** ldr s0, \[x0\]
** b .*
*/
T (Salias, s); int caller_s (Salias &s) { return callee_s (s); } /* { dg-bogus {argument of type 'S'} } */
/*
** _Z8caller_tR1T:
** ldr s0, \[x0\]
** b .*
*/
T (Talias, t); int caller_t (Talias &t) { return callee_t (t); } /* { dg-message {parameter passing for argument of type 'T' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_uR1U:
** ldp s0, s1, \[x0\]
** b .*
*/
T (Ualias, u); int caller_u (Ualias &u) { return callee_u (u); } /* { dg-bogus {argument of type 'U'} } */
/* { dg-bogus {argument of type 'const} "should not be printed as const" { target *-*-*} 0 } */

View File

@ -0,0 +1,206 @@
/* { dg-options "-std=c++17 -O -foptimize-sibling-calls -fpeephole2" } */
/* { dg-final { check-function-bodies "**" "" "" { target lp64 } } } */
struct X { };
struct Y { int : 0; };
struct Z { int : 0; Y y; };
struct W : public X { X q; };
struct A { float a; };
struct B : public X { float a; };
struct C : public Y { float a; };
struct D : public Z { float a; };
struct E : public W { float a; };
struct F { [[no_unique_address]] X x; float a; };
struct G { [[no_unique_address]] Y y; float a; };
struct H { [[no_unique_address]] Z z; float a; };
struct I { [[no_unique_address]] W w; float a; };
struct J { float a; [[no_unique_address]] X x; float b; };
struct K { float a; [[no_unique_address]] Y y; float b; };
struct L { float a; [[no_unique_address]] Z z; float b; };
struct M { float a; [[no_unique_address]] W w; float b; };
struct N : public A { float b; };
struct O { [[no_unique_address]] A a; float b; };
struct P : public Y { int : 0; float a, b, c, d; };
union Q { X x; float a; };
union R { [[no_unique_address]] X x; float a; };
union S { A a; float b; };
union T { F f; float b; };
union U { N n; O o; };
typedef S Salias;
typedef T Talias;
typedef U Ualias;
#define T(S, s) extern int callee_##s (S)
/*
** _Z8caller_aR1A:
** ldr s0, \[x0\]
** b .*
*/
T (A, a); int caller_a (A &a) { return callee_a (a); } /* { dg-bogus {argument of type 'A'} } */
/*
** _Z8caller_bR1B:
** ldr s0, \[x0\]
** b .*
*/
T (B, b); int caller_b (B &b) { return callee_b (b); } /* { dg-message {parameter passing for argument of type 'B' when C\+\+17 is enabled changed to match C\+\+14 in GCC 10.1} } */
/*
** _Z8caller_cR1C:
** ldr s0, \[x0\]
** b .*
*/
T (C, c); int caller_c (C &c) { return callee_c (c); } /* { dg-message {parameter passing for argument of type 'C' when C\+\+17 is enabled changed to match C\+\+14 in GCC 10.1} } */
/*
** _Z8caller_dR1D:
** ldr x0, \[x0\]
** b .*
*/
T (D, d); int caller_d (D &d) { return callee_d (d); } /* { dg-bogus {argument of type 'D'} } */
/*
** _Z8caller_eR1E:
** ldr x0, \[x0\]
** b .*
*/
T (E, e); int caller_e (E &e) { return callee_e (e); } /* { dg-bogus {argument of type 'E'} } */
/*
** _Z8caller_fR1F:
** ldr s0, \[x0\]
** b .*
*/
T (F, f); int caller_f (F &f) { return callee_f (f); } /* { dg-message {parameter passing for argument of type 'F' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_gR1G:
** ldr s0, \[x0\]
** b .*
*/
T (G, g); int caller_g (G &g) { return callee_g (g); } /* { dg-message {parameter passing for argument of type 'G' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_hR1H:
** ldr x0, \[x0\]
** b .*
*/
T (H, h); int caller_h (H &h) { return callee_h (h); } /* { dg-bogus {argument of type 'H'} } */
/*
** _Z8caller_iR1I:
** ldr x0, \[x0\]
** b .*
*/
T (I, i); int caller_i (I &i) { return callee_i (i); } /* { dg-bogus {argument of type 'I'} } */
/*
** _Z8caller_jR1J:
** ldp s0, s1, \[x0\]
** b .*
*/
T (J, j); int caller_j (J &j) { return callee_j (j); } /* { dg-message {parameter passing for argument of type 'J' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_kR1K:
** ldp s0, s1, \[x0\]
** b .*
*/
T (K, k); int caller_k (K &k) { return callee_k (k); } /* { dg-message {parameter passing for argument of type 'K' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_lR1L: { target aarch64_little_endian }
** (
** ldr w1, \[x0, 8\]
** ldr x0, \[x0\]
** |
** mov (x[0-9]+), x0
** ldr x0, \[x0\]
** ldr w1, \[\1, 8\]
** )
** b .*
*/
T (L, l); int caller_l (L &l) { return callee_l (l); } /* { dg-bogus {argument of type 'L'} } */
/*
** _Z8caller_mR1M: { target aarch64_little_endian }
** (
** ldr w1, \[x0, 8\]
** ldr x0, \[x0\]
** |
** mov (x[0-9]+), x0
** ldr x0, \[x0\]
** ldr w1, \[\1, 8\]
** )
** b .*
*/
T (M, m); int caller_m (M &m) { return callee_m (m); } /* { dg-bogus {argument of type 'M'} } */
/*
** _Z8caller_nR1N:
** ldp s0, s1, \[x0\]
** b .*
*/
T (N, n); int caller_n (N &n) { return callee_n (n); } /* { dg-bogus {argument of type 'N'} } */
/*
** _Z8caller_oR1O:
** ldp s0, s1, \[x0\]
** b .*
*/
T (O, o); int caller_o (O &o) { return callee_o (o); } /* { dg-bogus {argument of type 'O'} } */
/*
** _Z8caller_pR1P:
** ldp s0, s1, \[x0\]
** ldp s2, s3, \[x0, 8\]
** b .*
*/
T (P, p); int caller_p (P &p) { return callee_p (p); } /* { dg-message {parameter passing for argument of type 'P' when C\+\+17 is enabled changed to match C\+\+14 in GCC 10.1} } */
/*
** _Z8caller_qR1Q: { target aarch64_little_endian }
** ldr w0, \[x0\]
** b .*
*/
T (Q, q); int caller_q (Q &q) { return callee_q (q); } /* { dg-bogus {argument of type 'Q'} } */
/*
** _Z8caller_rR1R: { target aarch64_little_endian }
** ldr w0, \[x0\]
** b .*
*/
T (R, r); int caller_r (R &r) { return callee_r (r); } /* { dg-bogus {argument of type 'R'} } */
/*
** _Z8caller_sR1S:
** ldr s0, \[x0\]
** b .*
*/
T (Salias, s); int caller_s (Salias &s) { return callee_s (s); } /* { dg-bogus {argument of type 'S'} } */
/*
** _Z8caller_tR1T:
** ldr s0, \[x0\]
** b .*
*/
T (Talias, t); int caller_t (Talias &t) { return callee_t (t); } /* { dg-message {parameter passing for argument of type 'T' with '\[\[no_unique_address\]\]' members changed in GCC 10.1} } */
/*
** _Z8caller_uR1U:
** ldp s0, s1, \[x0\]
** b .*
*/
T (Ualias, u); int caller_u (Ualias &u) { return callee_u (u); } /* { dg-bogus {argument of type 'U'} } */
/* { dg-bogus {argument of type 'const} "should not be printed as const" { target *-*-*} 0 } */