}
/* See pr #108522. */
+
+#define DEFSTRUCT(_s, _n) \
+ struct DS \
+ { \
+ char a[_n]; \
+ unsigned long long b; \
+ int c; \
+ char d[2 * _n]; \
+ } _s
+
size_t
__attribute__ ((noinline))
-test_dynarray_struct_member (size_t sz)
+test_dynarray_struct_member_b (size_t sz)
{
- struct
- {
- char a[sz];
- char b;
- } s;
+ DEFSTRUCT (s, sz);
return __builtin_dynamic_object_size (&s.b, 0);
}
size_t
__attribute__ ((noinline))
+test_dynarray_struct_member_c (size_t sz)
+{
+ DEFSTRUCT (s, sz);
+
+ return __builtin_dynamic_object_size (&s.c, 0);
+}
+
+size_t
+__attribute__ ((noinline))
+test_dynarray_struct_member_d (size_t sz, size_t offset)
+{
+ DEFSTRUCT (s, sz);
+
+ return __builtin_dynamic_object_size (&s.d[offset], 0);
+}
+
+size_t
+__attribute__ ((noinline))
+test_dynarray_struct_member_subobj_b (size_t sz)
+{
+ DEFSTRUCT (s, sz);
+
+ return __builtin_dynamic_object_size (&s.b, 1);
+}
+
+size_t
+__attribute__ ((noinline))
+test_dynarray_struct_member_subobj_c (size_t sz)
+{
+ DEFSTRUCT (s, sz);
+
+ return __builtin_dynamic_object_size (&s.c, 1);
+}
+
+size_t
+__attribute__ ((noinline))
+test_dynarray_struct_member_subobj_d (size_t sz, size_t offset)
+{
+ DEFSTRUCT (s, sz);
+
+ return __builtin_dynamic_object_size (&s.d[offset], 1);
+}
+
+size_t
+__attribute__ ((noinline))
test_substring (size_t sz, size_t off)
{
char str[sz];
if (test_dynarray_struct_subobj2 (42, 4, &objsz)
!= objsz - 4 - sizeof (long) - sizeof (int))
FAIL ();
- if (test_dynarray_struct_member (42) != sizeof (char))
+ DEFSTRUCT(ds, 64);
+ const size_t n = sizeof (ds.a);
+ if (test_dynarray_struct_member_b (n)
+ != sizeof (ds) - __builtin_offsetof (struct DS, b))
+ FAIL ();
+ if (test_dynarray_struct_member_c (n)
+ != sizeof (ds) - __builtin_offsetof (struct DS, c))
+ FAIL ();
+ if (test_dynarray_struct_member_d (n, 0)
+ != sizeof (ds) - __builtin_offsetof (struct DS, d))
+ FAIL ();
+ if (test_dynarray_struct_member_subobj_b (n) != sizeof (ds.b))
+ FAIL ();
+ if (test_dynarray_struct_member_subobj_c (n) != sizeof (ds.c))
+ FAIL ();
+ if (test_dynarray_struct_member_subobj_d (n, n - 2)
+ != sizeof (ds) - __builtin_offsetof (struct DS, d) - n + 2)
FAIL ();
if (test_substring_ptrplus (128, 4) != (128 - 4) * sizeof (int))
FAIL ();
tree wholesize;
};
-static tree compute_object_offset (const_tree, const_tree);
+static tree compute_object_offset (tree, const_tree);
static bool addr_object_size (struct object_size_info *,
const_tree, int, tree *, tree *t = NULL);
static tree alloc_object_size (const gcall *, int);
if unknown. */
static tree
-compute_object_offset (const_tree expr, const_tree var)
+compute_object_offset (tree expr, const_tree var)
{
enum tree_code code = PLUS_EXPR;
tree base, off, t;
t = TREE_OPERAND (expr, 1);
off = size_binop (PLUS_EXPR,
- (TREE_OPERAND (expr, 2) ? TREE_OPERAND (expr, 2)
- : DECL_FIELD_OFFSET (t)),
+ component_ref_field_offset (expr),
size_int (tree_to_uhwi (DECL_FIELD_BIT_OFFSET (t))
/ BITS_PER_UNIT));
break;