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/builtin-has-attribute-3.c | |
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/builtin-has-attribute-3.c')
-rw-r--r-- | gcc/testsuite/c-c++-common/builtin-has-attribute-3.c | 314 |
1 files changed, 314 insertions, 0 deletions
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" } */ |