| // RUN: %clang_cc1 %s -ast-dump | FileCheck %s | 
 | // RUN: %clang_cc1 -fexperimental-late-parse-attributes %s -ast-dump | FileCheck %s | 
 |  | 
 | #define __counted_by(f)  __attribute__((counted_by(f))) | 
 |  | 
 | struct size_unknown; | 
 | struct size_known { | 
 |   int field; | 
 | }; | 
 |  | 
 | //============================================================================== | 
 | // __counted_by on struct member pointer in decl attribute position | 
 | //============================================================================== | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_member_pointer_complete_ty definition | 
 | // CHECK-NEXT: |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT: `-FieldDecl {{.+}} buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | struct on_member_pointer_complete_ty { | 
 |   int count; | 
 |   struct size_known * buf __counted_by(count); | 
 | }; | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_pointer_anon_buf definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  |-RecordDecl {{.+}} struct definition | 
 | // CHECK-NEXT:  | `-FieldDecl {{.+}} buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} implicit 'struct on_pointer_anon_buf::(anonymous at [[ANON_STRUCT_PATH:.+]])' | 
 | // CHECK-NEXT:  `-IndirectFieldDecl {{.+}} implicit buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | // CHECK-NEXT:    |-Field {{.+}} field_index 1 'struct on_pointer_anon_buf::(anonymous at [[ANON_STRUCT_PATH]])' | 
 | // CHECK-NEXT:    `-Field {{.+}} 'buf' 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | struct on_pointer_anon_buf { | 
 |   int count; | 
 |   struct { | 
 |     struct size_known *buf __counted_by(count); | 
 |   }; | 
 | }; | 
 |  | 
 | struct on_pointer_anon_count { | 
 |   struct { | 
 |     int count; | 
 |   }; | 
 |   struct size_known *buf __counted_by(count); | 
 | }; | 
 |  | 
 | //============================================================================== | 
 | // __counted_by on struct member pointer in type attribute position | 
 | //============================================================================== | 
 | // TODO: Correctly parse counted_by as a type attribute. Currently it is parsed | 
 | // as a declaration attribute | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_member_pointer_complete_ty_ty_pos definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  `-FieldDecl {{.+}} buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | struct on_member_pointer_complete_ty_ty_pos { | 
 |   int count; | 
 |   struct size_known *__counted_by(count) buf; | 
 | }; | 
 |  | 
 | // TODO: This should be forbidden but isn't due to counted_by being treated as a | 
 | // declaration attribute. The attribute ends up on the outer most pointer | 
 | // (allowed by sema) even though syntactically its supposed to be on the inner | 
 | // pointer (would not allowed by sema due to pointee being a function type). | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_member_pointer_fn_ptr_ty_ty_pos_inner definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  `-FieldDecl {{.+}} fn_ptr 'void (** __counted_by(count))(void)':'void (**)(void)' | 
 | struct on_member_pointer_fn_ptr_ty_ty_pos_inner { | 
 |   int count; | 
 |   void (* __counted_by(count) * fn_ptr)(void); | 
 | }; | 
 |  | 
 | // FIXME: The generated AST here is wrong. The attribute should be on the inner | 
 | // pointer. | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_nested_pointer_inner definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  `-FieldDecl {{.+}} buf 'struct size_known ** __counted_by(count)':'struct size_known **' | 
 | struct on_nested_pointer_inner { | 
 |   int count; | 
 |   // TODO: This should be disallowed because in the `-fbounds-safety` model | 
 |   // `__counted_by` can only be nested when used in function parameters. | 
 |   struct size_known *__counted_by(count) *buf; | 
 | }; | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_nested_pointer_outer definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  `-FieldDecl {{.+}} buf 'struct size_known ** __counted_by(count)':'struct size_known **' | 
 | struct on_nested_pointer_outer { | 
 |   int count; | 
 |   struct size_known **__counted_by(count) buf; | 
 | }; | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_pointer_anon_buf_ty_pos definition | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} referenced count 'int' | 
 | // CHECK-NEXT:  |-RecordDecl {{.+}} struct definition | 
 | // CHECK-NEXT:  | `-FieldDecl {{.+}} buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} implicit 'struct on_pointer_anon_buf_ty_pos::(anonymous at [[ANON_STRUCT_PATH2:.+]])' | 
 | // CHECK-NEXT:  `-IndirectFieldDecl {{.+}} implicit buf 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | // CHECK-NEXT:    |-Field {{.+}} field_index 1 'struct on_pointer_anon_buf_ty_pos::(anonymous at [[ANON_STRUCT_PATH2]])' | 
 | // CHECK-NEXT:    `-Field {{.+}} 'buf' 'struct size_known * __counted_by(count)':'struct size_known *' | 
 | struct on_pointer_anon_buf_ty_pos { | 
 |   int count; | 
 |   struct { | 
 |     struct size_known * __counted_by(count) buf; | 
 |   }; | 
 | }; | 
 |  | 
 | // CHECK-LABEL: RecordDecl {{.+}} struct on_pointer_anon_count_ty_pos definition | 
 | // CHECK-NEXT:  |-RecordDecl {{.+}} struct definition | 
 | // CHECK-NEXT:  | `-FieldDecl {{.+}} count 'int' | 
 | // CHECK-NEXT:  |-FieldDecl {{.+}} implicit 'struct on_pointer_anon_count_ty_pos::(anonymous at [[ANON_STRUCT_PATH3:.+]])' | 
 | // CHECK-NEXT:  |-IndirectFieldDecl {{.+}} implicit referenced count 'int' | 
 | // CHECK-NEXT:  | |-Field {{.+}} field_index 0 'struct on_pointer_anon_count_ty_pos::(anonymous at [[ANON_STRUCT_PATH3]])' | 
 | // CHECK-NEXT:  | `-Field {{.+}} 'count' 'int' | 
 | struct on_pointer_anon_count_ty_pos { | 
 |   struct { | 
 |     int count; | 
 |   }; | 
 |   struct size_known *__counted_by(count) buf; | 
 | }; |