diff options
author | Martin Sebor <msebor@redhat.com> | 2018-11-21 02:50:02 +0000 |
---|---|---|
committer | Martin Sebor <msebor@gcc.gnu.org> | 2018-11-20 19:50:02 -0700 |
commit | 98f08eb8939735c1e9cbc1ec5cadebe79e935c90 (patch) | |
tree | 9fb2084c7e170349abee1d8f950047936556b98a /gcc/testsuite/c-c++-common | |
parent | 48d1f31d1b7131e8f809ede8256e4f1eb6c5c3ae (diff) | |
download | gcc-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.c | 206 | ||||
-rw-r--r-- | gcc/testsuite/c-c++-common/builtin-has-attribute-3.c | 314 | ||||
-rw-r--r-- | gcc/testsuite/c-c++-common/builtin-has-attribute-4.c | 285 | ||||
-rw-r--r-- | gcc/testsuite/c-c++-common/builtin-has-attribute.c | 60 |
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" } */ +} |