cxx0x-attributes.cpp revision 176edba5311f6eff0cad2631449885ddf4fbc9ea
1// RUN: %clang_cc1 -fcxx-exceptions -fexceptions -fsyntax-only -verify -std=c++11 -Wc++14-compat %s 2 3// Need std::initializer_list 4namespace std { 5 typedef decltype(sizeof(int)) size_t; 6 7 // libc++'s implementation 8 template <class _E> 9 class initializer_list 10 { 11 const _E* __begin_; 12 size_t __size_; 13 14 initializer_list(const _E* __b, size_t __s) 15 : __begin_(__b), 16 __size_(__s) 17 {} 18 19 public: 20 typedef _E value_type; 21 typedef const _E& reference; 22 typedef const _E& const_reference; 23 typedef size_t size_type; 24 25 typedef const _E* iterator; 26 typedef const _E* const_iterator; 27 28 initializer_list() : __begin_(nullptr), __size_(0) {} 29 30 size_t size() const {return __size_;} 31 const _E* begin() const {return __begin_;} 32 const _E* end() const {return __begin_ + __size_;} 33 }; 34} 35 36 37// Declaration syntax checks 38[[]] int before_attr; 39int [[]] between_attr; 40const [[]] int between_attr_2 = 0; // expected-error {{an attribute list cannot appear here}} 41int after_attr [[]]; 42int * [[]] ptr_attr; 43int & [[]] ref_attr = after_attr; 44int & [[unknown]] ref_attr_2 = after_attr; // expected-warning {{unknown attribute 'unknown' ignored}} 45int & [[noreturn]] ref_attr_3 = after_attr; // expected-error {{'noreturn' attribute cannot be applied to types}} 46int && [[]] rref_attr = 0; 47int array_attr [1] [[]]; 48alignas(8) int aligned_attr; 49[[test::valid(for 42 [very] **** '+' symbols went on a trip and had a "good"_time; the end.)]] int garbage_attr; // expected-warning {{unknown attribute 'valid' ignored}} 50[[,,,static, class, namespace,, inline, constexpr, mutable,, bitand, bitor::compl(!.*_ Cx.!U^*R),,,]] int more_garbage_attr; // expected-warning {{unknown attribute 'static' ignored}} \ 51 // expected-warning {{unknown attribute 'class' ignored}} \ 52 // expected-warning {{unknown attribute 'namespace' ignored}} \ 53 // expected-warning {{unknown attribute 'inline' ignored}} \ 54 // expected-warning {{unknown attribute 'constexpr' ignored}} \ 55 // expected-warning {{unknown attribute 'mutable' ignored}} \ 56 // expected-warning {{unknown attribute 'bitand' ignored}} \ 57 // expected-warning {{unknown attribute 'compl' ignored}} 58[[u8"invalid!"]] int invalid_string_attr; // expected-error {{expected ']'}} 59void fn_attr () [[]]; 60void noexcept_fn_attr () noexcept [[]]; 61struct MemberFnOrder { 62 virtual void f() const volatile && noexcept [[]] final = 0; 63}; 64struct [[]] struct_attr; 65class [[]] class_attr {}; 66union [[]] union_attr; 67 68// Checks attributes placed at wrong syntactic locations of class specifiers. 69class [[]] [[]] 70 attr_after_class_name_decl [[]] [[]]; // expected-error {{an attribute list cannot appear here}} 71 72class [[]] [[]] 73 attr_after_class_name_definition [[]] [[]] [[]]{}; // expected-error {{an attribute list cannot appear here}} 74 75class [[]] c {}; 76class c [[]] [[]] x; 77class c [[]] [[]] y [[]] [[]]; 78class c final [(int){0}]; 79 80class base {}; 81class [[]] [[]] final_class 82 alignas(float) [[]] final // expected-error {{an attribute list cannot appear here}} 83 alignas(float) [[]] [[]] alignas(float): base{}; // expected-error {{an attribute list cannot appear here}} 84 85class [[]] [[]] final_class_another 86 [[]] [[]] alignas(16) final // expected-error {{an attribute list cannot appear here}} 87 [[]] [[]] alignas(16) [[]]{}; // expected-error {{an attribute list cannot appear here}} 88 89[[]] struct with_init_declarators {} init_declarator; 90[[]] struct no_init_declarators; // expected-error {{an attribute list cannot appear here}} 91template<typename> [[]] struct no_init_declarators_template; // expected-error {{an attribute list cannot appear here}} 92void fn_with_structs() { 93 [[]] struct with_init_declarators {} init_declarator; 94 [[]] struct no_init_declarators; // expected-error {{an attribute list cannot appear here}} 95} 96[[]]; 97struct ctordtor { 98 [[]] ctordtor(); 99 [[]] ~ctordtor(); 100}; 101[[]] ctordtor::ctordtor() {} 102[[]] ctordtor::~ctordtor() {} 103extern "C++" [[]] int extern_attr; 104template <typename T> [[]] void template_attr (); 105[[]] [[]] int [[]] [[]] multi_attr [[]] [[]]; 106 107int comma_attr [[,]]; 108int scope_attr [[foo::]]; // expected-error {{expected identifier}} 109int (paren_attr) [[]]; // expected-error {{an attribute list cannot appear here}} 110unsigned [[]] int attr_in_decl_spec; // expected-error {{an attribute list cannot appear here}} 111unsigned [[]] int [[]] const double_decl_spec = 0; // expected-error 2{{an attribute list cannot appear here}} 112class foo { 113 void const_after_attr () [[]] const; // expected-error {{expected ';'}} 114}; 115extern "C++" [[]] { } // expected-error {{an attribute list cannot appear here}} 116[[]] template <typename T> void before_template_attr (); // expected-error {{an attribute list cannot appear here}} 117[[]] namespace ns { int i; } // expected-error {{an attribute list cannot appear here}} expected-note {{declared here}} 118[[]] static_assert(true, ""); //expected-error {{an attribute list cannot appear here}} 119[[]] asm(""); // expected-error {{an attribute list cannot appear here}} 120 121[[]] using ns::i; // expected-error {{an attribute list cannot appear here}} 122[[unknown]] using namespace ns; // expected-warning {{unknown attribute 'unknown' ignored}} 123[[noreturn]] using namespace ns; // expected-error {{'noreturn' attribute only applies to functions}} 124namespace [[]] ns2 {} // expected-warning {{attributes on a namespace declaration are incompatible with C++ standards before C++1z}} 125 126using [[]] alignas(4) [[]] ns::i; // expected-error {{an attribute list cannot appear here}} 127using [[]] alignas(4) [[]] foobar = int; // expected-error {{an attribute list cannot appear here}} expected-error {{'alignas' attribute only applies to}} 128 129void bad_attributes_in_do_while() { 130 do {} while ( 131 [[ns::i); // expected-error {{expected ']'}} \ 132 // expected-note {{to match this '['}} \ 133 // expected-error {{expected expression}} 134 do {} while ( 135 [[a]b ns::i); // expected-error {{expected ']'}} \ 136 // expected-note {{to match this '['}} \ 137 // expected-error {{expected expression}} 138 do {} while ( 139 [[ab]ab] ns::i); // expected-error {{an attribute list cannot appear here}} 140 do {} while ( // expected-note {{to match this '('}} 141 alignas(4 ns::i; // expected-note {{to match this '('}} 142} // expected-error 2{{expected ')'}} expected-error {{expected expression}} 143 144[[]] using T = int; // expected-error {{an attribute list cannot appear here}} 145using T [[]] = int; // ok 146template<typename T> using U [[]] = T; 147using ns::i [[]]; // expected-error {{an attribute list cannot appear here}} 148using [[]] ns::i; // expected-error {{an attribute list cannot appear here}} 149using T [[unknown]] = int; // expected-warning {{unknown attribute 'unknown' ignored}} 150using T [[noreturn]] = int; // expected-error {{'noreturn' attribute only applies to functions}} 151using V = int; // expected-note {{previous}} 152using V [[gnu::vector_size(16)]] = int; // expected-error {{redefinition with different types}} 153 154auto trailing() -> [[]] const int; // expected-error {{an attribute list cannot appear here}} 155auto trailing() -> const [[]] int; // expected-error {{an attribute list cannot appear here}} 156auto trailing() -> const int [[]]; 157auto trailing_2() -> struct struct_attr [[]]; 158 159namespace N { 160 struct S {}; 161}; 162template<typename> struct Template {}; 163 164// FIXME: Improve this diagnostic 165struct [[]] N::S s; // expected-error {{an attribute list cannot appear here}} 166struct [[]] Template<int> t; // expected-error {{an attribute list cannot appear here}} 167struct [[]] ::template Template<int> u; // expected-error {{an attribute list cannot appear here}} 168template struct [[]] Template<char>; // expected-error {{an attribute list cannot appear here}} 169template <> struct [[]] Template<void>; 170 171enum [[]] E1 {}; 172enum [[]] E2; // expected-error {{forbids forward references}} 173enum [[]] E1; 174enum [[]] E3 : int; 175enum [[]] { 176 k_123 [[]] = 123 // expected-warning {{attributes on an enumerator declaration are incompatible with C++ standards before C++1z}} 177}; 178enum [[]] E1 e; // expected-error {{an attribute list cannot appear here}} 179enum [[]] class E4 { }; // expected-error {{an attribute list cannot appear here}} 180enum struct [[]] E5; 181 182struct S { 183 friend int f [[]] (); // expected-FIXME{{an attribute list cannot appear here}} 184 friend int f1 [[noreturn]] (); //expected-error{{an attribute list cannot appear here}} 185 friend int f2 [[]] [[noreturn]] () {} 186 [[]] friend int g(); // expected-error{{an attribute list cannot appear here}} 187 [[]] friend int h() { 188 } 189 [[]] friend int f3(), f4(), f5(); // expected-error{{an attribute list cannot appear here}} 190 friend int f6 [[noreturn]] (), f7 [[noreturn]] (), f8 [[noreturn]] (); // expected-error3 {{an attribute list cannot appear here}} 191 friend class [[]] C; // expected-error{{an attribute list cannot appear here}} 192 [[]] friend class D; // expected-error{{an attribute list cannot appear here}} 193 [[]] friend int; // expected-error{{an attribute list cannot appear here}} 194}; 195template<typename T> void tmpl(T) {} 196template void tmpl [[]] (int); // expected-FIXME {{an attribute list cannot appear here}} 197template [[]] void tmpl(char); // expected-error {{an attribute list cannot appear here}} 198template void [[]] tmpl(short); 199 200// Argument tests 201alignas int aligned_no_params; // expected-error {{expected '('}} 202alignas(i) int aligned_nonconst; // expected-error {{'aligned' attribute requires integer constant}} expected-note {{read of non-const variable 'i'}} 203 204// Statement tests 205void foo () { 206 [[]] ; 207 [[]] { } 208 [[]] if (0) { } 209 [[]] for (;;); 210 [[]] do { 211 [[]] continue; 212 } while (0); 213 [[]] while (0); 214 215 [[]] switch (i) { 216 [[]] case 0: 217 [[]] default: 218 [[]] break; 219 } 220 221 [[]] goto there; 222 [[]] there: 223 224 [[]] try { 225 } [[]] catch (...) { // expected-error {{an attribute list cannot appear here}} 226 } 227 struct S { int arr[2]; } s; 228 (void)s.arr[ [] { return 0; }() ]; // expected-error {{C++11 only allows consecutive left square brackets when introducing an attribute}} 229 int n = __builtin_offsetof(S, arr[ [] { return 0; }() ]); // expected-error {{C++11 only allows consecutive left square brackets when introducing an attribute}} 230 231 void bar [[noreturn]] ([[]] int i, [[]] int j); 232 using FuncType = void ([[]] int); 233 void baz([[]]...); // expected-error {{expected parameter declarator}} 234 235 [[]] return; 236} 237 238template<typename...Ts> void variadic() { 239 void bar [[noreturn...]] (); // expected-error {{attribute 'noreturn' cannot be used as an attribute pack}} 240} 241 242// Expression tests 243void bar () { 244 // FIXME: GCC accepts [[gnu::noreturn]] on a lambda, even though it appertains 245 // to the operator()'s type, and GCC does not otherwise accept attributes 246 // applied to types. Use that to test this. 247 [] () [[gnu::noreturn]] { return; } (); // expected-warning {{attribute 'noreturn' ignored}} FIXME-error {{should not return}} 248 [] () [[gnu::noreturn]] { throw; } (); // expected-warning {{attribute 'noreturn' ignored}} 249 new int[42][[]][5][[]]{}; 250} 251 252// Condition tests 253void baz () { 254 if ([[unknown]] bool b = true) { // expected-warning {{unknown attribute 'unknown' ignored}} 255 switch ([[unknown]] int n { 42 }) { // expected-warning {{unknown attribute 'unknown' ignored}} 256 default: 257 for ([[unknown]] int n = 0; [[unknown]] char b = n < 5; ++b) { // expected-warning 2{{unknown attribute 'unknown' ignored}} 258 } 259 } 260 } 261 int x; 262 // An attribute can be applied to an expression-statement, such as the first 263 // statement in a for. But it can't be applied to a condition which is an 264 // expression. 265 for ([[]] x = 0; ; ) {} // expected-error {{an attribute list cannot appear here}} 266 for (; [[]] x < 5; ) {} // expected-error {{an attribute list cannot appear here}} 267 while ([[]] bool k { false }) { 268 } 269 while ([[]] true) { // expected-error {{an attribute list cannot appear here}} 270 } 271 do { 272 } while ([[]] false); // expected-error {{an attribute list cannot appear here}} 273 274 for ([[unknown]] int n : { 1, 2, 3 }) { // expected-warning {{unknown attribute 'unknown' ignored}} 275 } 276} 277 278enum class __attribute__((visibility("hidden"))) SecretKeepers { 279 one, /* rest are deprecated */ two, three 280}; 281enum class [[]] EvenMoreSecrets {}; 282 283namespace arguments { 284 void f[[gnu::format(printf, 1, 2)]](const char*, ...); 285 void g() [[unknown::foo(ignore arguments for unknown attributes, even with symbols!)]]; // expected-warning {{unknown attribute 'foo' ignored}} 286 [[deprecated("with argument")]] int i; 287} 288 289// Forbid attributes on decl specifiers. 290unsigned [[gnu::used]] static int [[gnu::unused]] v1; // expected-error {{'unused' attribute cannot be applied to types}} \ 291 expected-error {{an attribute list cannot appear here}} 292typedef [[gnu::used]] unsigned long [[gnu::unused]] v2; // expected-error {{'unused' attribute cannot be applied to types}} \ 293 expected-error {{an attribute list cannot appear here}} 294int [[carries_dependency]] foo(int [[carries_dependency]] x); // expected-error 2{{'carries_dependency' attribute cannot be applied to types}} 295 296// Forbid [[gnu::...]] attributes on declarator chunks. 297int *[[gnu::unused]] v3; // expected-warning {{attribute 'unused' ignored}} 298int v4[2][[gnu::unused]]; // expected-warning {{attribute 'unused' ignored}} 299int v5()[[gnu::unused]]; // expected-warning {{attribute 'unused' ignored}} 300 301[[attribute_declaration]]; // expected-warning {{unknown attribute 'attribute_declaration' ignored}} 302[[noreturn]]; // expected-error {{'noreturn' attribute only applies to functions}} 303[[carries_dependency]]; // expected-error {{'carries_dependency' attribute only applies to functions, methods, and parameters}} 304 305class A { 306 A([[gnu::unused]] int a); 307}; 308A::A([[gnu::unused]] int a) {} 309 310namespace GccConst { 311 // GCC's tokenizer treats const and __const as the same token. 312 [[gnu::const]] int *f1(); 313 [[gnu::__const]] int *f2(); 314 [[gnu::__const__]] int *f3(); 315 void f(const int *); 316 void g() { f(f1()); f(f2()); } 317 void h() { f(f3()); } 318} 319 320namespace GccASan { 321 __attribute__((no_address_safety_analysis)) void f1(); 322 __attribute__((no_sanitize_address)) void f2(); 323 [[gnu::no_address_safety_analysis]] void f3(); 324 [[gnu::no_sanitize_address]] void f4(); 325} 326 327namespace { 328 [[deprecated]] void bar(); 329 [[deprecated("hello")]] void baz(); 330 [[deprecated()]] void foo(); // expected-error {{parentheses must be omitted if 'deprecated' attribute's argument list is empty}} 331 [[gnu::deprecated()]] void quux(); 332} 333