aboutsummaryrefslogtreecommitdiff
path: root/gcc/testsuite/c-c++-common
diff options
context:
space:
mode:
authorMartin Sebor <msebor@redhat.com>2018-11-21 02:50:02 +0000
committerMartin Sebor <msebor@gcc.gnu.org>2018-11-20 19:50:02 -0700
commit98f08eb8939735c1e9cbc1ec5cadebe79e935c90 (patch)
tree9fb2084c7e170349abee1d8f950047936556b98a /gcc/testsuite/c-c++-common
parent48d1f31d1b7131e8f809ede8256e4f1eb6c5c3ae (diff)
downloadgcc-98f08eb8939735c1e9cbc1ec5cadebe79e935c90.zip
gcc-98f08eb8939735c1e9cbc1ec5cadebe79e935c90.tar.gz
gcc-98f08eb8939735c1e9cbc1ec5cadebe79e935c90.tar.bz2
c-parser.c (c_parser_has_attribute_expression): New function.
gcc/c/ChangeLog: * c-parser.c (c_parser_has_attribute_expression): New function. (c_parser_attribute): New function. (c_parser_attributes): Move code into c_parser_attribute. (c_parser_unary_expression): Handle RID_HAS_ATTRIBUTE_EXPRESSION. gcc/c-family/ChangeLog: * c-attribs.c (type_for_vector_size): New function. (type_valid_for_vector_size): Same. (handle_vector_size_attribute): Move code to the functions above and call them. (validate_attribute, has_attribute): New functions. * c-common.h (has_attribute): Declare. (rid): Add RID_HAS_ATTRIBUTE_EXPRESSION. * c-common.c (c_common_resword): Same. gcc/cp/ChangeLog: * cp-tree.h (cp_check_const_attributes): Declare. * decl2.c (cp_check_const_attributes): Declare extern. * parser.c (cp_parser_has_attribute_expression): New function. (cp_parser_unary_expression): Handle RID_HAS_ATTRIBUTE_EXPRESSION. (cp_parser_gnu_attribute_list): Add argument. gcc/ChangeLog: * doc/extend.texi (Other Builtins): Add __builtin_has_attribute. gcc/testsuite/ChangeLog: * c-c++-common/builtin-has-attribute-2.c: New test. * c-c++-common/builtin-has-attribute-3.c: New test. * c-c++-common/builtin-has-attribute-4.c: New test. * c-c++-common/builtin-has-attribute.c: New test. * gcc.dg/builtin-has-attribute.c: New test. * gcc/testsuite/gcc.target/i386/builtin-has-attribute.c: New test. From-SVN: r266335
Diffstat (limited to 'gcc/testsuite/c-c++-common')
-rw-r--r--gcc/testsuite/c-c++-common/builtin-has-attribute-2.c206
-rw-r--r--gcc/testsuite/c-c++-common/builtin-has-attribute-3.c314
-rw-r--r--gcc/testsuite/c-c++-common/builtin-has-attribute-4.c285
-rw-r--r--gcc/testsuite/c-c++-common/builtin-has-attribute.c60
4 files changed, 865 insertions, 0 deletions
diff --git a/gcc/testsuite/c-c++-common/builtin-has-attribute-2.c b/gcc/testsuite/c-c++-common/builtin-has-attribute-2.c
new file mode 100644
index 0000000..0f692ff
--- /dev/null
+++ b/gcc/testsuite/c-c++-common/builtin-has-attribute-2.c
@@ -0,0 +1,206 @@
+/* Verify __builtin_has_attribute return value for types.
+ { dg-do compile }
+ { dg-options "-Wall -ftrack-macro-expansion=0" }
+ { dg-options "-Wall -Wno-narrowing -Wno-unused-local-typedefs -ftrack-macro-expansion=0" { target c++ } } */
+
+#define ATTR(...) __attribute__ ((__VA_ARGS__))
+
+#define A(expect, sym, attr) \
+ typedef int Assert [1 - 2 * !(__builtin_has_attribute (sym, attr) == expect)]
+
+struct ATTR (packed) Packed { char c; int i; };
+
+void fvoid (void);
+struct Packed fpacked (void);
+
+union OrdinaryUnion { void *p; int i; };
+union ATTR (transparent_union) TransparentUnion { void *p; int i; };
+
+/* Exercise __builtin_has_attribute with the first argument that
+ is a type. */
+
+void test_type (int n)
+{
+ /* Verify both forms of the attribute spelling. Unlike the attribute
+ keyword that can be spelled three ways (with either leading or
+ trailing underscores, or with both), attribute names can only be
+ spelled two ways. */
+ A (0, int, aligned);
+ A (0, int, __aligned__);
+
+ A (0, int, aligned (1));
+ A (0, int, aligned (2));
+ A (0, int[1], aligned);
+ A (0, int[1], aligned (2));
+ A (0, int[n], aligned);
+ A (0, int[n], aligned (4));
+
+ /* Again, verify both forms of the attribute spelling. */
+ A (1, ATTR (aligned) char, aligned);
+ A (1, ATTR (aligned (2)) short, aligned);
+ A (1, ATTR (aligned (4)) int, __aligned__);
+
+ A (0, int ATTR (aligned (4)), aligned (2));
+ A (0, int ATTR (aligned (2)), aligned (4));
+ /* GCC retains both attributes in the */
+ A (0, int ATTR (aligned (2), aligned (4)), aligned (2));
+ A (1, int ATTR (aligned (2), aligned (4)), aligned (4));
+ /* The following fails due to bug 87524.
+ A (1, int ATTR (aligned (4), aligned (2))), aligned (4)); */
+ A (0, int ATTR (aligned (4), aligned (2)), aligned (8));
+
+ A (1, int ATTR (aligned (8)), aligned (1 + 7));
+
+ enum { eight = 8 };
+ A (1, int ATTR (aligned (8)), aligned (eight));
+ A (1, int ATTR (aligned (eight)), aligned (1 + 7));
+
+ struct NotPacked { char c; int i; };
+ A (0, struct NotPacked, packed);
+ A (1, struct Packed, packed);
+
+ /* Exercise types returned from a function. */
+ A (0, fvoid (), packed);
+ A (1, fpacked (), packed);
+
+ struct ATTR (aligned (2), packed) Aligned2Packed { char c; int i; };
+ A (1, struct Aligned2Packed, aligned);
+ A (1, struct Aligned2Packed, aligned (2));
+ A (0, struct Aligned2Packed, aligned (4));
+ A (1, struct Aligned2Packed, packed);
+
+ A (0, int, may_alias);
+ A (1, ATTR (may_alias) int, may_alias);
+
+ A (0, char, warn_if_not_aligned (1));
+ A (0, char, warn_if_not_aligned (2));
+
+ A (1, ATTR (warn_if_not_aligned (2)) char, warn_if_not_aligned);
+ A (0, ATTR (warn_if_not_aligned (2)) char, warn_if_not_aligned (1));
+ A (1, ATTR (warn_if_not_aligned (2)) char, warn_if_not_aligned (2));
+ A (0, ATTR (warn_if_not_aligned (2)) char, warn_if_not_aligned (4));
+
+ A (0, union OrdinaryUnion, transparent_union);
+
+ A (1, union TransparentUnion, transparent_union);
+ A (1, const union TransparentUnion, transparent_union);
+}
+
+/* Exercise __builtin_has_attribute with the first argument that
+ is a typedef. */
+
+void test_typedef (int n)
+{
+ typedef char A1[1];
+ A (0, A1, aligned);
+ A (0, A1, aligned (1));
+ A (0, A1, aligned (2));
+
+ typedef char An[n];
+ A (0, An, aligned);
+ A (0, An, aligned (1));
+ A (0, An, aligned (2));
+
+ typedef ATTR (aligned (8)) short AI8;
+ A (1, AI8, aligned);
+ A (0, AI8, aligned (4));
+ A (1, AI8, aligned (8));
+ A (0, AI8, aligned (16));
+
+ A (1, const AI8, aligned);
+ A (1, const volatile AI8, aligned);
+
+ typedef ATTR (aligned (2), aligned (8), aligned (16)) int AI16;
+ A (1, AI16, aligned);
+ A (0, AI16, aligned (1));
+ A (0, AI16, aligned (2));
+ A (0, AI16, aligned (4));
+ A (0, AI16, aligned (8));
+ A (1, AI16, aligned (16));
+ A (0, AI16, aligned (32));
+
+ typedef const AI16 CAI16;
+ A (1, CAI16, aligned);
+ A (0, CAI16, aligned (1));
+ A (1, CAI16, aligned (16));
+
+ typedef int I;
+ A (0, I, may_alias);
+ A (0, AI8, may_alias);
+
+ typedef ATTR (may_alias) int MAI;
+ A (1, MAI, may_alias);
+
+ typedef ATTR (aligned (4), may_alias) char A4MAC;
+ A (0, A4MAC, aligned (0));
+ A (0, A4MAC, aligned (1));
+ A (0, A4MAC, aligned (2));
+ A (1, A4MAC, aligned (4));
+ A (0, A4MAC, aligned (8));
+ A (1, A4MAC, may_alias);
+
+ typedef ATTR (may_alias, aligned (8)) char A8MAC;
+ A (1, A8MAC, aligned);
+ A (0, A8MAC, aligned (0));
+ A (0, A8MAC, aligned (1));
+ A (0, A8MAC, aligned (2));
+ A (0, A8MAC, aligned (4));
+ A (1, A8MAC, aligned (8));
+ A (0, A8MAC, aligned (16));
+ A (1, A8MAC, may_alias);
+
+ typedef ATTR (may_alias) const AI8 CMAI8;
+ A (1, CMAI8, aligned);
+ A (1, CMAI8, may_alias);
+ A (0, CMAI8, aligned (4));
+ A (1, CMAI8, aligned (8));
+
+ typedef void Fnull (void*, void*, void*);
+ A (0, Fnull, nonnull);
+ A (0, Fnull, nonnull (1));
+ A (0, Fnull, nonnull (2));
+ A (0, Fnull, nonnull (3));
+
+ typedef ATTR (nonnull) Fnull Fnonnull;
+ A (1, Fnonnull, nonnull);
+ A (1, Fnonnull, nonnull (1));
+ A (1, Fnonnull, nonnull (2));
+ A (1, Fnonnull, nonnull (3));
+
+ typedef ATTR (nonnull (2)) void Fnonnull_2 (void*, void*, void*);
+ A (0, Fnonnull_2, nonnull);
+ A (0, Fnonnull_2, nonnull (1));
+ A (1, Fnonnull_2, nonnull (2));
+ A (0, Fnonnull_2, nonnull (3));
+
+ typedef ATTR (nonnull (1), nonnull (2), nonnull (3))
+ void Fnonnull_1_2_3 (void*, void*, void*);
+
+ /* The following fails because the built-in doesn't recognize that
+ a single nonnull with no arguments is the same as one nonnull for
+ each function parameter. Disable the testing for now.
+ A (1, Fnonnull_1_2_3, nonnull);
+ */
+ A (1, Fnonnull_1_2_3, nonnull (1));
+ A (1, Fnonnull_1_2_3, nonnull (2));
+ A (1, Fnonnull_1_2_3, nonnull (3));
+
+ typedef void Freturns (void);
+ A (0, Fnull, noreturn);
+ A (0, Freturns, noreturn);
+
+ typedef ATTR (warn_if_not_aligned (8)) char CWA8;
+ A (0, CWA8, warn_if_not_aligned (2));
+ A (0, CWA8, warn_if_not_aligned (4));
+ A (1, CWA8, warn_if_not_aligned (8));
+ A (0, CWA8, warn_if_not_aligned (16));
+
+ typedef union OrdinaryUnion OrdUnion;
+ A (0, OrdUnion, transparent_union);
+
+ /* The attribute is ignored on typedefs but GCC fails to diagnose
+ it (see bug ). */
+ typedef union ATTR (transparent_union)
+ OrdinaryUnion TransUnion; /* { dg-warning "\\\[-Wattributes" "pr87578" { xfail { ! { c++ } } } } */
+ A (0, TransUnion, transparent_union);
+}
diff --git a/gcc/testsuite/c-c++-common/builtin-has-attribute-3.c b/gcc/testsuite/c-c++-common/builtin-has-attribute-3.c
new file mode 100644
index 0000000..237dc72
--- /dev/null
+++ b/gcc/testsuite/c-c++-common/builtin-has-attribute-3.c
@@ -0,0 +1,314 @@
+/* Verify __builtin_has_attribute return value for functions.
+ { dg-do compile }
+ { dg-options "-Wall -ftrack-macro-expansion=0" }
+ { dg-options "-Wall -Wno-narrowing -Wno-unused-local-typedefs -ftrack-macro-expansion=0" { target c++ } } */
+
+#define ATTR(...) __attribute__ ((__VA_ARGS__))
+
+void fnone (void);
+
+ATTR (aligned) void faligned (void);
+ATTR (aligned (1)) void faligned_1 (void);
+ATTR (aligned (2)) void faligned_2 (void);
+ATTR (aligned (4)) void faligned_4 (void);
+ATTR (aligned (8)) void faligned_8 (void);
+
+ATTR (alloc_size (1)) void* falloc_size_1 (int, int);
+ATTR (alloc_size (2)) void* falloc_size_2 (int, int);
+ATTR (alloc_size (2, 4)) void* falloc_size_2_4 (int, int, int, int);
+
+ATTR (alloc_align (1)) void* falloc_align_1 (int, int);
+ATTR (alloc_align (2)) void* falloc_align_2 (int, int);
+ATTR (alloc_align (1), alloc_size (2)) void* falloc_align_1_size_2 (int, int);
+ATTR (alloc_align (2), alloc_size (1)) void* falloc_align_2_size_1 (int, int);
+
+#if __cplusplus
+extern "C"
+#endif
+ATTR (noreturn) void fnoreturn (void) { __builtin_abort (); }
+
+ATTR (alias ("fnoreturn")) void falias (void);
+
+#define A(expect, sym, attr) \
+ typedef int Assert [1 - 2 * !(__builtin_has_attribute (sym, attr) == expect)]
+
+void test_aligned (void)
+{
+ A (0, fnone, aligned);
+ A (0, fnone, aligned (0));
+ A (0, fnone, aligned (1));
+ A (0, fnone, aligned (2));
+ A (0, fnone, aligned (4));
+ A (0, fnone, aligned (8));
+ A (0, fnone, aligned (16));
+
+ A (1, faligned, aligned);
+ A (0, faligned, aligned (0));
+ A (0, faligned, aligned (1));
+ A (0, faligned, aligned (2));
+
+ A (1, faligned_1, aligned);
+ A (0, faligned_1, aligned (0));
+ A (1, faligned_1, aligned (1));
+ A (0, faligned_1, aligned (2));
+ A (0, faligned_1, aligned (4));
+
+ A (1, faligned_2, aligned);
+ A (0, faligned_2, aligned (0));
+ A (0, faligned_2, aligned (1));
+ A (1, faligned_2, aligned (2));
+ A (0, faligned_2, aligned (4));
+}
+
+
+void test_alloc_align (void)
+{
+ A (0, fnone, alloc_align);
+ A (0, falloc_size_1, alloc_align);
+ A (1, falloc_align_1, alloc_align);
+ A (1, falloc_align_2, alloc_align);
+
+ A (0, fnone, alloc_align (1)); /* { dg-warning "\\\[-Wattributes" } */
+ A (0, falloc_size_1, alloc_align (1));
+ A (1, falloc_align_1, alloc_align (1));
+ A (0, falloc_align_2, alloc_align (1));
+ A (1, falloc_align_2, alloc_align (2));
+}
+
+
+void test_alloc_size_malloc (void)
+{
+ A (0, fnone, alloc_size);
+ A (0, fnone, alloc_size (1)); /* { dg-warning "\\\[-Wattributes" } */
+ A (0, fnone, alloc_size (2)); /* { dg-warning "\\\[-Wattributes" } */
+ A (0, falloc_align_1, alloc_size (1));
+ A (0, falloc_align_2, alloc_size (1));
+ A (1, falloc_size_1, alloc_size (1));
+ A (0, falloc_size_1, alloc_size (2));
+ A (0, falloc_size_2, alloc_size (1));
+ A (1, falloc_size_2, alloc_size (2));
+
+ A (1, falloc_size_2_4, alloc_size);
+ /* It would probably make more sense to have the built-in return
+ true only when both alloc_size arguments match, not just one
+ or the other. */
+ A (0, falloc_size_2_4, alloc_size (1));
+ A (1, falloc_size_2_4, alloc_size (2));
+ A (0, falloc_size_2_4, alloc_size (3));
+ A (1, falloc_size_2_4, alloc_size (4));
+ A (1, falloc_size_2_4, alloc_size (2, 4));
+
+ extern ATTR (alloc_size (3))
+ void* fmalloc_size_3 (int, int, int);
+
+ A (1, fmalloc_size_3, alloc_size);
+ A (0, fmalloc_size_3, alloc_size (1));
+ A (0, fmalloc_size_3, alloc_size (2));
+ A (1, fmalloc_size_3, alloc_size (3));
+ A (0, fmalloc_size_3, malloc);
+
+ extern ATTR (malloc)
+ void* fmalloc_size_3 (int, int, int);
+
+ A (1, fmalloc_size_3, alloc_size (3));
+ A (1, fmalloc_size_3, malloc);
+}
+
+
+void test_alias (void)
+{
+ A (0, fnoreturn, alias);
+ A (1, falias, alias);
+ A (1, falias, alias ("fnoreturn"));
+ A (0, falias, alias ("falias"));
+ A (0, falias, alias ("fnone"));
+}
+
+
+void test_cold_hot (void)
+{
+ extern ATTR (cold) void fcold (void);
+ extern ATTR (hot) void fhot (void);
+
+ A (0, fnone, cold);
+ A (0, fnone, hot);
+
+ A (1, fcold, cold);
+ A (0, fcold, hot);
+
+ A (0, fhot, cold);
+ A (1, fhot, hot);
+}
+
+
+void test_const_leaf_pure (void)
+{
+ extern ATTR (const) int fconst (void);
+ extern ATTR (leaf) int fleaf (void);
+ extern ATTR (pure) int fpure (void);
+
+ A (0, fnone, const);
+ A (0, fnone, leaf);
+ A (0, fnone, pure);
+
+ A (1, fconst, const);
+ A (0, fconst, leaf);
+ A (0, fconst, pure);
+
+ A (0, fleaf, const);
+ A (1, fleaf, leaf);
+ A (0, fleaf, pure);
+
+ A (0, fpure, const);
+ A (0, fpure, leaf);
+ A (1, fpure, pure);
+
+ extern ATTR (const, leaf) int fconst_leaf (void);
+
+ A (1, fconst_leaf, const);
+ A (1, fconst_leaf, leaf);
+
+ extern ATTR (leaf, const) int fleaf_const (void);
+
+ A (1, fleaf_const, const);
+ A (1, fleaf_const, leaf);
+}
+
+
+void test_ctor_dtor (void)
+{
+ extern ATTR (constructor) void fctor (void);
+ extern ATTR (destructor) void fdtor (void);
+ extern ATTR (constructor, destructor) void fctor_dtor (void);
+
+ A (0, fnone, constructor);
+ A (0, fnone, destructor);
+
+ A (1, fctor, constructor);
+ A (1, fdtor, destructor);
+
+ extern ATTR (constructor) void fctor_dtor (void);
+ extern ATTR (destructor) void fctor_dtor (void);
+ extern ATTR (constructor, destructor) void fctor_dtor (void);
+
+ A (1, fctor_dtor, constructor);
+ A (1, fctor_dtor, destructor);
+
+ extern ATTR (constructor (123)) void fctor_123 (void);
+ A (1, fctor_123, constructor);
+ A (0, fctor_123, destructor);
+ A (1, fctor_123, constructor (123));
+ A (0, fctor_123, constructor (124));
+
+ extern ATTR (destructor (234)) void fctor_123 (void);
+ A (1, fctor_123, constructor (123));
+ A (1, fctor_123, destructor);
+ A (1, fctor_123, destructor (234));
+ A (0, fctor_123, destructor (235));
+}
+
+
+void test_externally_visible (void)
+{
+ extern void fexternally_visible (void);
+
+ A (0, fexternally_visible, externally_visible);
+
+ extern ATTR (externally_visible) void fexternally_visible (void);
+
+ A (1, fexternally_visible, externally_visible);
+}
+
+
+void test_flatten (void)
+{
+ extern void fflatten (void);
+
+ A (0, fflatten, flatten);
+
+ extern ATTR (flatten) void fflatten (void);
+
+ A (1, fflatten, flatten);
+
+ extern void fflatten (void);
+
+ A (1, fflatten, flatten);
+}
+
+
+ATTR (format (printf, 2, 4)) void
+fformat_printf_2_3 (int, const char*, int, ...);
+
+void test_format (void)
+{
+ A (0, fnone, format);
+ A (0, fnone, format (printf));
+ A (0, fnone, format (printf, 2));
+}
+
+
+inline void finline (void) { }
+inline ATTR (always_inline) void falways_inline (void) { }
+inline ATTR (always_inline, gnu_inline) void falways_gnu_inline (void) { }
+ATTR (noinline) void fnoinline () { }
+
+void test_inline (void)
+{
+ A (0, fnone, always_inline);
+ A (0, fnone, gnu_inline);
+ A (0, fnone, noinline);
+
+ A (0, finline, always_inline);
+ A (0, finline, gnu_inline);
+ A (0, finline, noinline);
+
+ A (1, falways_inline, always_inline);
+ A (0, falways_inline, gnu_inline);
+ A (0, falways_inline, noinline);
+
+ A (1, falways_gnu_inline, always_inline);
+ A (1, falways_gnu_inline, gnu_inline);
+ A (0, falways_gnu_inline, noinline);
+
+ A (0, fnoinline, always_inline);
+ A (0, fnoinline, gnu_inline);
+ A (1, fnoinline, noinline);
+}
+
+
+ATTR (no_instrument_function) void fno_instrument (void);
+
+ATTR (visibility ("default")) void fdefault (void);
+ATTR (visibility ("hidden")) void fhidden (void);
+ATTR (visibility ("internal")) void finternal (void);
+ATTR (visibility ("protected")) void fprotected (void);
+
+void test_visibility (void)
+{
+ A (0, fnone, visibility ("default"));
+ A (0, fnone, visibility ("hidden"));
+ A (0, fnone, visibility ("internal"));
+ A (0, fnone, visibility ("protected"));
+
+ A (1, fdefault, visibility ("default"));
+ A (0, fdefault, visibility ("hidden"));
+ A (0, fdefault, visibility ("internal"));
+ A (0, fdefault, visibility ("protected"));
+
+ A (0, fhidden, visibility ("default"));
+ A (1, fhidden, visibility ("hidden"));
+ A (0, fhidden, visibility ("internal"));
+ A (0, fhidden, visibility ("protected"));
+
+ A (0, finternal, visibility ("default"));
+ A (0, finternal, visibility ("hidden"));
+ A (1, finternal, visibility ("internal"));
+ A (0, finternal, visibility ("protected"));
+
+ A (0, fprotected, visibility ("default"));
+ A (0, fprotected, visibility ("hidden"));
+ A (0, fprotected, visibility ("internal"));
+ A (1, fprotected, visibility ("protected"));
+}
+
+/* { dg-prune-output "specifies less restrictive attribute" } */
diff --git a/gcc/testsuite/c-c++-common/builtin-has-attribute-4.c b/gcc/testsuite/c-c++-common/builtin-has-attribute-4.c
new file mode 100644
index 0000000..14bdd3f
--- /dev/null
+++ b/gcc/testsuite/c-c++-common/builtin-has-attribute-4.c
@@ -0,0 +1,285 @@
+/* Verify __builtin_has_attribute return value for variables.
+ { dg-do compile }
+ { dg-options "-Wall -ftrack-macro-expansion=0" }
+ { dg-options "-Wall -Wno-narrowing -Wno-unused -ftrack-macro-expansion=0" { target c++ } } */
+
+#define ATTR(...) __attribute__ ((__VA_ARGS__))
+
+#define A(expect, sym, attr) \
+ typedef int Assert [1 - 2 * !(__builtin_has_attribute (sym, attr) == expect)]
+
+int vnone;
+
+ATTR (aligned) char valigned;
+ATTR (aligned (1)) char valigned_1;
+ATTR (aligned (2)) char valigned_2;
+ATTR (aligned (4)) char valigned_4;
+ATTR (aligned (8)) char valigned_8;
+
+void test_aligned (void)
+{
+ A (0, vnone, aligned);
+ A (0, vnone, aligned (0));
+ A (0, vnone, aligned (1));
+ A (0, vnone, aligned (2));
+ A (0, vnone, aligned (4));
+ A (0, vnone, aligned (8));
+ A (0, vnone, aligned (16));
+
+ A (1, valigned, aligned);
+ A (0, valigned, aligned (0));
+ A (0, valigned, aligned (1));
+ A (0, valigned, aligned (2));
+
+ A (1, valigned_1, aligned);
+ A (0, valigned_1, aligned (0));
+ A (1, valigned_1, aligned (1));
+ A (0, valigned_1, aligned (2));
+ A (0, valigned_1, aligned (4));
+
+ A (1, valigned_2, aligned);
+ A (0, valigned_2, aligned (0));
+ A (0, valigned_2, aligned (1));
+ A (1, valigned_2, aligned (2));
+ A (0, valigned_2, aligned (4));
+}
+
+
+int vtarget;
+extern ATTR (alias ("vtarget")) int valias;
+
+void test_alias (void)
+{
+ A (0, vnone, alias);
+ A (1, valias, alias);
+ A (1, valias, alias ("vtarget"));
+ A (0, valias, alias ("vnone"));
+}
+
+
+void test_cleanup (void)
+{
+ extern void fpv (void*);
+ extern void fcleanup (void*);
+
+ int var;
+ ATTR (cleanup (fcleanup)) int var_cleanup;
+ A (0, var, cleanup);
+ A (1, var_cleanup, cleanup);
+ A (1, var_cleanup, cleanup (fcleanup));
+ A (0, var_cleanup, cleanup (fpv));
+}
+
+
+ATTR (common) int vcommon;
+ATTR (nocommon) int vnocommon;
+
+void test_common (void)
+{
+ A (0, vnone, common);
+ A (0, vnone, nocommon);
+
+ A (1, vcommon, common);
+ A (0, vcommon, nocommon);
+
+ A (0, vnocommon, common);
+ A (1, vnocommon, nocommon);
+}
+
+
+void test_externally_visible (void)
+{
+ extern int vexternally_visible;
+
+ A (0, vexternally_visible, externally_visible);
+
+ extern ATTR (externally_visible) int vexternally_visible;
+
+ A (1, vexternally_visible, externally_visible);
+}
+
+
+int test_mode (void)
+{
+ ATTR (mode (byte)) int i8;
+ return __builtin_has_attribute (i8, mode); /* { dg-warning ".mode. attribute not supported in .__builtin_has_attribute." } */
+}
+
+
+void test_nonstring (void)
+{
+ char arr[1];
+ char* ptr = arr;
+
+ ATTR (nonstring) char arr_nonstring[1];
+ ATTR (nonstring) char *ptr_nonstring = arr_nonstring;
+
+ A (0, arr, nonstring);
+ A (0, ptr, nonstring);
+
+ A (1, arr_nonstring, nonstring);
+ A (1, ptr_nonstring, nonstring);
+}
+
+struct PackedMember
+{
+ char c;
+ short s;
+ int i;
+ ATTR (packed) int a[2];
+} gpak[2];
+
+void test_packed (struct PackedMember *p)
+{
+ int vunpacked;
+ ATTR (packed) int vpacked; /* { dg-warning ".packed. attribute ignored" } */
+
+ A (0, vunpacked, packed);
+ A (0, vpacked, packed);
+
+ int arr_unpacked[2];
+ ATTR (packed) int arr_packed[2]; /* { dg-warning ".packed. attribute ignored" } */
+
+ A (0, arr_unpacked, packed);
+ A (0, arr_packed, packed);
+ A (0, arr_unpacked[0], packed);
+ A (0, arr_packed[0], packed);
+
+ A (0, gpak, packed);
+ A (0, gpak[0], packed);
+ A (0, *gpak, packed);
+ A (0, gpak[0].c, packed);
+ A (0, gpak[1].s, packed);
+ A (1, gpak->a, packed);
+ A (1, (*gpak).a[0], packed);
+
+ /* The following fails because in C it's represented as
+ INDIRECT_REF (POINTER_PLUS (NOP_EXPR (ADDR_EXPR (gpak)), ...))
+ with no reference to the member. Avoid testing it.
+ A (1, *gpak[9].a, packed); */
+
+ A (0, p->c, packed);
+ A (0, p->s, packed);
+ A (1, p->a, packed);
+ A (1, p->a[0], packed);
+ /* Similar to the comment above.
+ A (1, *p->a, packed); */
+}
+
+
+ATTR (section ("sectA")) int var_sectA;
+ATTR (section ("sectB")) int var_sectB;
+
+void test_section (void)
+{
+ int var = 0;
+ A (0, var, section);
+ A (0, var, section ("sectA"));
+
+ A (1, var_sectA, section);
+ A (1, var_sectA, section ("sectA"));
+ A (0, var_sectA, section ("sectB"));
+
+ A (1, var_sectB, section);
+ A (0, var_sectB, section ("sectA"));
+ A (1, var_sectB, section ("sectB"));
+}
+
+
+void test_vector_size (void)
+{
+ char c;
+ extern int arrx[];
+ extern int arr1[1];
+
+ A (0, c, vector_size);
+ A (0, c, vector_size (1));
+ A (0, arrx, vector_size);
+ A (0, arrx, vector_size (4));
+ A (0, arr1, vector_size);
+ A (0, arr1, vector_size (8));
+
+ ATTR (vector_size (4)) char cv4;
+ ATTR (vector_size (16)) int iv16;
+
+ A (1, cv4, vector_size);
+ A (0, cv4, vector_size (1));
+ A (0, cv4, vector_size (2));
+ A (1, cv4, vector_size (4));
+ A (0, cv4, vector_size (8));
+
+ A (1, iv16, vector_size);
+ A (0, iv16, vector_size (1));
+ A (0, iv16, vector_size (8));
+ A (1, iv16, vector_size (16));
+ A (0, iv16, vector_size (32));
+
+ ATTR (vector_size (8)) float afv8[4];
+ A (1, afv8, vector_size);
+ A (0, afv8, vector_size (1));
+ A (0, afv8, vector_size (2));
+ A (0, afv8, vector_size (4));
+ A (1, afv8, vector_size (8));
+ A (0, afv8, vector_size (16));
+}
+
+
+ATTR (visibility ("default")) int vdefault;
+ATTR (visibility ("hidden")) int vhidden;
+ATTR (visibility ("internal")) int vinternal;
+ATTR (visibility ("protected")) int vprotected;
+
+void test_visibility (void)
+{
+ A (0, vnone, visibility ("default"));
+ A (0, vnone, visibility ("hidden"));
+ A (0, vnone, visibility ("internal"));
+ A (0, vnone, visibility ("protected"));
+
+ A (1, vdefault, visibility ("default"));
+ A (0, vdefault, visibility ("hidden"));
+ A (0, vdefault, visibility ("internal"));
+ A (0, vdefault, visibility ("protected"));
+
+ A (0, vhidden, visibility ("default"));
+ A (1, vhidden, visibility ("hidden"));
+ A (0, vhidden, visibility ("internal"));
+ A (0, vhidden, visibility ("protected"));
+
+ A (0, vinternal, visibility ("default"));
+ A (0, vinternal, visibility ("hidden"));
+ A (1, vinternal, visibility ("internal"));
+ A (0, vinternal, visibility ("protected"));
+
+ A (0, vprotected, visibility ("default"));
+ A (0, vprotected, visibility ("hidden"));
+ A (0, vprotected, visibility ("internal"));
+ A (1, vprotected, visibility ("protected"));
+}
+
+
+int var_init_strong = 123;
+int var_uninit_strong;
+static int var_extern_strong;
+static int var_static_strong;
+
+ATTR (weak) int var_init_weak = 234;
+ATTR (weak) int var_uninit_weak;
+
+void test_weak (void)
+{
+ int var_local = 0;
+ static int var_static_local = 0;
+
+ A (0, var_init_strong, weak);
+ A (0, var_uninit_strong, weak);
+ A (0, var_extern_strong, weak);
+ A (0, var_static_strong, weak);
+ A (0, var_local, weak);
+ A (0, var_static_local, weak);
+
+ A (1, var_init_weak, weak);
+ A (1, var_uninit_weak, weak);
+}
+
+/* { dg-prune-output "specifies less restrictive attribute" } */
diff --git a/gcc/testsuite/c-c++-common/builtin-has-attribute.c b/gcc/testsuite/c-c++-common/builtin-has-attribute.c
new file mode 100644
index 0000000..67c792f
--- /dev/null
+++ b/gcc/testsuite/c-c++-common/builtin-has-attribute.c
@@ -0,0 +1,60 @@
+/* Verify __builtin_has_attribute error handling.
+ { dg-do compile }
+ { dg-options "-Wall -ftrack-macro-expansion=0" } */
+
+#define ATTR(list) __attribute__ (list)
+
+void fnone (void);
+
+ATTR ((aligned)) void faligned (void);
+ATTR ((aligned (8))) void faligned_8 (void);
+
+#define has_attr(x, attr) __builtin_has_attribute (x, attr)
+
+#define A(expect, sym, attr) \
+ typedef int Assert [1 - 2 * !(has_attr (sym, attr) == expect)]
+
+
+int b;
+
+/* Exercise syntactically invalid arguments. */
+
+void test_bad_arguments (void)
+{
+ b = __builtin_has_attribute (); /* { dg-error "expected \(primary-\)?expression|expected .,." } */
+ b = __builtin_has_attribute (1); /* { dg-error "expected .,." } */
+ b = __builtin_has_attribute (void); /* { dg-error "expected .,." } */
+ b = __builtin_has_attribute (foo); /* { dg-error ".foo. \(undeclared|was not declared\)" } */
+ /* { dg-error "expected .,." "missing comma" { target *-*-* } .-1 } */
+
+ /* Verify the implementationm doesn't ICE. */
+ b = __builtin_has_attribute (foobar, aligned); /* { dg-error ".foobar. \(undeclared|was not declared\)" } */
+
+ b = __builtin_has_attribute (1, 2, 3); /* { dg-error "expected identifier" } */
+ b = __builtin_has_attribute (int, 1 + 2); /* { dg-error "expected identifier" } */
+ b = __builtin_has_attribute (2, "aligned"); /* { dg-error "expected identifier" } */
+}
+
+/* Exercise syntactically valid arguments applied in invalid ways. */
+
+void test_invalid_arguments (void)
+{
+ b = has_attr (fnone, align); /* { dg-error "unknown attribute .align." } */
+ b = has_attr (b, aligned__); /* { dg-error "unknown attribute .aligned__." } */
+ b = has_attr (fnone, aligned (3)); /* { dg-error "alignment .3. is not a positive power of 2" } */
+
+ /* Verify the out-of-bounds arguments are diagnosed and the result
+ of the built-in is false. */
+ A (0, fnone, alloc_size (1)); /* { dg-warning "\\\[-Wattributes]" } */
+ A (0, fnone, alloc_size (2)); /* { dg-warning "\\\[-Wattributes]" } */
+
+ A (0, int, alloc_size (1)); /* { dg-warning ".alloc_size. attribute only applies to function types" } */
+
+ int i = 1;
+ A (0, i, alloc_size (1)); /* { dg-warning ".alloc_size. attribute only applies to function types" } */
+
+ A (0, faligned_8, aligned (i)); /* { dg-error "alignment is not an integer constant" } */
+
+ typedef ATTR ((aligned (2))) char CA2;
+ b = has_attr (CA2[2], aligned); /* { dg-error "alignment of array elements is greater than element size" } */
+}