/* Testing the correct usage of the new __builtin_counted_by_ref. */ /* { dg-do compile } */ /* { dg-options "-O" } */ #include struct annotated { size_t b; int other; int c[] __attribute ((counted_by (b))); } *array_annotated; struct flex { size_t b; int other; int c[]; } *array_flex; #define MAX(A, B) (A > B) ? (A) : (B) #define MY_ALLOC(P, FAM, COUNT) ({ \ __auto_type __p = &(P); \ __auto_type __c = (COUNT); \ size_t __size = MAX (sizeof (*(*__p)),\ __builtin_offsetof (__typeof(*(*__p)),FAM) \ + sizeof (*((*__p)->FAM)) * __c); \ if ((*__p = __builtin_malloc(__size))) { \ __builtin_memset(*__p, 0, __size); \ __auto_type ret = __builtin_counted_by_ref((*__p)->FAM); \ *_Generic(ret, void *: &(size_t){0}, default: ret) = __c; \ if (sizeof (__builtin_counted_by_ref ((*__p)->FAM)) != sizeof (char *)) \ __builtin_abort (); \ } \ }) extern char c_count; extern short s_count; extern int i_count; extern long l_count; extern float f_count; extern int * foo (); int main(int argc, char *argv[]) { /* The good usages. */ MY_ALLOC(array_annotated, c, 10); MY_ALLOC(array_flex, c, 20); MY_ALLOC(array_annotated, c, c_count); MY_ALLOC(array_flex, c, i_count); MY_ALLOC(array_annotated, c, l_count); MY_ALLOC(array_flex, c, c_count * 3); MY_ALLOC(array_annotated, c, l_count * i_count); /* The bad usages, issue errors. */ __builtin_counted_by_ref (); /* { dg-error "wrong number of arguments to" } */ __builtin_counted_by_ref (array_annotated->c, 10); /* { dg-error "wrong number of arguments to" } */ __builtin_counted_by_ref (array_annotated->other); /* { dg-error "must be an array" } */ __builtin_counted_by_ref (foo()); /* { dg-error "must be an array" } */ return 0; }